2023-07-20 19:32:15 +02:00
import {
shuffle ,
onlyUnique ,
debounce ,
delay ,
isDataURL ,
createThumbnail ,
2023-07-30 22:10:37 +02:00
extractAllWords ,
2023-07-20 19:32:15 +02:00
} from './utils.js' ;
import { RA _CountCharTokens , humanizedDateTime , dragElement } from "./RossAscends-mods.js" ;
import { sortCharactersList , sortGroupMembers , loadMovingUIState } from './power-user.js' ;
import {
chat ,
sendSystemMessage ,
printMessages ,
substituteParams ,
characters ,
default _avatar ,
addOneMessage ,
callPopup ,
clearChat ,
Generate ,
select _rm _info ,
setCharacterId ,
setCharacterName ,
setEditedMessageId ,
is _send _press ,
name1 ,
resetChatState ,
setSendButtonState ,
getCharacters ,
system _message _types ,
online _status ,
talkativeness _default ,
selectRightMenuWithAnimation ,
setRightTabSelectedClass ,
default _ch _mes ,
deleteLastMessage ,
showSwipeButtons ,
hideSwipeButtons ,
chat _metadata ,
updateChatMetadata ,
isStreamingEnabled ,
getThumbnailUrl ,
streamingProcessor ,
getRequestHeaders ,
setMenuType ,
menu _type ,
select _selected _character ,
cancelTtsPlay ,
isMultigenEnabled ,
displayPastChats ,
sendMessageAsUser ,
getBiasStrings ,
saveChatConditional ,
deactivateSendButtons ,
activateSendButtons ,
eventSource ,
event _types ,
getCurrentChatId ,
setScenarioOverride ,
getCropPopup ,
} from "../script.js" ;
import { appendTagToList , createTagMapFromList , getTagsList , applyTagsOnCharacterSelect , tag _map } from './tags.js' ;
export {
selected _group ,
is _group _automode _enabled ,
is _group _generating ,
group _generation _id ,
groups ,
saveGroupChat ,
generateGroupWrapper ,
deleteGroup ,
getGroupAvatar ,
getGroups ,
printGroups ,
regenerateGroup ,
resetSelectedGroup ,
select _group _chats ,
}
let is _group _generating = false ; // Group generation flag
let is _group _automode _enabled = false ;
let groups = [ ] ;
let selected _group = null ;
let group _generation _id = null ;
let fav _grp _checked = false ;
let fav _filter _on = false ;
export const group _activation _strategy = {
NATURAL : 0 ,
LIST : 1 ,
} ;
const groupAutoModeInterval = setInterval ( groupChatAutoModeWorker , 5000 ) ;
const saveGroupDebounced = debounce ( async ( group ) => await _save ( group ) , 500 ) ;
async function _save ( group , reload = true ) {
await fetch ( "/editgroup" , {
method : "POST" ,
headers : getRequestHeaders ( ) ,
body : JSON . stringify ( group ) ,
} ) ;
if ( reload ) {
await getCharacters ( ) ;
}
}
// Group chats
async function regenerateGroup ( ) {
let generationId = getLastMessageGenerationId ( ) ;
while ( chat . length > 0 ) {
const lastMes = chat [ chat . length - 1 ] ;
const this _generationId = lastMes . extra ? . gen _id ;
// for new generations after the update
if ( ( generationId && this _generationId ) && generationId !== this _generationId ) {
break ;
}
// legacy for generations before the update
else if ( lastMes . is _user || lastMes . is _system ) {
break ;
}
await deleteLastMessage ( ) ;
}
generateGroupWrapper ( ) ;
}
async function loadGroupChat ( chatId ) {
const response = await fetch ( "/getgroupchat" , {
method : "POST" ,
headers : getRequestHeaders ( ) ,
body : JSON . stringify ( { id : chatId } ) ,
} ) ;
if ( response . ok ) {
const data = await response . json ( ) ;
return data ;
}
return [ ] ;
}
export async function getGroupChat ( groupId ) {
const group = groups . find ( ( x ) => x . id === groupId ) ;
const chat _id = group . chat _id ;
const data = await loadGroupChat ( chat _id ) ;
if ( Array . isArray ( data ) && data . length ) {
data [ 0 ] . is _group = true ;
for ( let key of data ) {
chat . push ( key ) ;
}
printMessages ( ) ;
} else {
sendSystemMessage ( system _message _types . GROUP , '' , { isSmallSys : true } ) ;
if ( group && Array . isArray ( group . members ) ) {
for ( let member of group . members ) {
const character = characters . find ( x => x . avatar === member || x . name === member ) ;
if ( ! character ) {
continue ;
}
const mes = getFirstCharacterMessage ( character ) ;
chat . push ( mes ) ;
addOneMessage ( mes ) ;
}
}
}
if ( group ) {
let metadata = group . chat _metadata ? ? { } ;
updateChatMetadata ( metadata , true ) ;
}
await saveGroupChat ( groupId , true ) ;
eventSource . emit ( event _types . CHAT _CHANGED , getCurrentChatId ( ) ) ;
}
function getFirstCharacterMessage ( character ) {
let messageText = character . first _mes ;
// if there are alternate greetings, pick one at random
if ( Array . isArray ( character . data ? . alternate _greetings ) ) {
const messageTexts = [ character . first _mes , ... character . data . alternate _greetings ] . filter ( x => x ) ;
messageText = messageTexts [ Math . floor ( Math . random ( ) * messageTexts . length ) ] ;
}
const mes = { } ;
mes [ "is_user" ] = false ;
mes [ "is_system" ] = false ;
mes [ "name" ] = character . name ;
mes [ "is_name" ] = true ;
mes [ "send_date" ] = humanizedDateTime ( ) ;
mes [ "original_avatar" ] = character . avatar ;
mes [ "extra" ] = { "gen_id" : Date . now ( ) * Math . random ( ) * 1000000 } ;
mes [ "mes" ] = messageText
? substituteParams ( messageText . trim ( ) , name1 , character . name )
: default _ch _mes ;
mes [ "force_avatar" ] =
character . avatar != "none"
? getThumbnailUrl ( 'avatar' , character . avatar )
: default _avatar ;
return mes ;
}
function resetSelectedGroup ( ) {
selected _group = null ;
is _group _generating = false ;
}
async function saveGroupChat ( groupId , shouldSaveGroup ) {
const group = groups . find ( x => x . id == groupId ) ;
const chat _id = group . chat _id ;
group [ 'date_last_chat' ] = Date . now ( ) ;
const response = await fetch ( "/savegroupchat" , {
method : "POST" ,
headers : getRequestHeaders ( ) ,
body : JSON . stringify ( { id : chat _id , chat : [ ... chat ] } ) ,
} ) ;
if ( shouldSaveGroup && response . ok ) {
await editGroup ( groupId ) ;
}
sortCharactersList ( ) ;
}
export async function renameGroupMember ( oldAvatar , newAvatar , newName ) {
// Scan every group for our renamed character
for ( const group of groups ) {
try {
// Try finding the member by old avatar link
const memberIndex = group . members . findIndex ( x => x == oldAvatar ) ;
// Character was not present in the group...
if ( memberIndex == - 1 ) {
continue ;
}
// Replace group member avatar id and save the changes
group . members [ memberIndex ] = newAvatar ;
await editGroup ( group . id , true ) ;
console . log ( ` Renamed character ${ newName } in group: ${ group . name } ` )
// Load all chats from this group
for ( const chatId of group . chats ) {
const messages = await loadGroupChat ( chatId ) ;
// Only save the chat if there were any changes to the chat content
let hadChanges = false ;
// Chat shouldn't be empty
if ( Array . isArray ( messages ) && messages . length ) {
// Iterate over every chat message
for ( const message of messages ) {
// Only look at character messages
if ( message . is _user || message . is _system ) {
continue ;
}
// Message belonged to the old-named character:
// Update name, avatar thumbnail URL and original avatar link
if ( message . force _avatar && message . force _avatar . indexOf ( encodeURIComponent ( oldAvatar ) ) !== - 1 ) {
message . name = newName ;
message . force _avatar = message . force _avatar . replace ( encodeURIComponent ( oldAvatar ) , encodeURIComponent ( newAvatar ) ) ;
message . original _avatar = newAvatar ;
hadChanges = true ;
}
}
if ( hadChanges ) {
const saveChatResponse = await fetch ( "/savegroupchat" , {
method : "POST" ,
headers : getRequestHeaders ( ) ,
body : JSON . stringify ( { id : chatId , chat : [ ... messages ] } ) ,
} ) ;
if ( saveChatResponse . ok ) {
console . log ( ` Renamed character ${ newName } in group chat: ${ chatId } ` ) ;
}
}
}
}
}
catch ( error ) {
console . log ( ` An error during renaming the character ${ newName } in group: ${ group . name } ` ) ;
console . error ( error ) ;
}
}
}
async function getGroups ( ) {
const response = await fetch ( "/getgroups" , {
method : "POST" ,
headers : getRequestHeaders ( )
} ) ;
if ( response . ok ) {
const data = await response . json ( ) ;
groups = data . sort ( ( a , b ) => a . id - b . id ) ;
// Convert groups to new format
for ( const group of groups ) {
if ( group . disabled _members == undefined ) {
group . disabled _members = [ ] ;
}
if ( group . chat _id == undefined ) {
group . chat _id = group . id ;
group . chats = [ group . id ] ;
group . members = group . members
. map ( x => characters . find ( y => y . name == x ) ? . avatar )
. filter ( x => x )
. filter ( onlyUnique )
}
if ( group . past _metadata == undefined ) {
group . past _metadata = { } ;
}
if ( typeof group . chat _id === 'number' ) {
group . chat _id = String ( group . chat _id ) ;
}
if ( Array . isArray ( group . chats ) && group . chats . some ( x => typeof x === 'number' ) ) {
group . chats = group . chats . map ( x => String ( x ) ) ;
}
}
}
}
function printGroups ( ) {
for ( let group of groups ) {
const template = $ ( "#group_list_template .group_select" ) . clone ( ) ;
template . data ( "id" , group . id ) ;
template . attr ( "grid" , group . id ) ;
template . find ( ".ch_name" ) . html ( group . name ) ;
template . find ( '.group_fav_icon' ) . css ( "display" , 'none' ) ;
template . addClass ( group . fav ? 'is_fav' : '' ) ;
template . find ( ".ch_fav" ) . val ( group . fav ) ;
// Display inline tags
const tags = getTagsList ( group . id ) ;
const tagsElement = template . find ( '.tags' ) ;
tags . forEach ( tag => appendTagToList ( tagsElement , tag , { } ) ) ;
$ ( "#rm_print_characters_block" ) . prepend ( template ) ;
updateGroupAvatar ( group ) ;
}
}
function updateGroupAvatar ( group ) {
$ ( "#rm_print_characters_block .group_select" ) . each ( function ( ) {
if ( $ ( this ) . data ( "id" ) == group . id ) {
const avatar = getGroupAvatar ( group ) ;
if ( avatar ) {
$ ( this ) . find ( ".avatar" ) . replaceWith ( avatar ) ;
}
}
} ) ;
}
function getGroupAvatar ( group ) {
if ( ! group ) {
return $ ( ` <div class="avatar"><img src=" ${ default _avatar } "></div> ` ) ;
}
if ( isDataURL ( group . avatar _url ) ) {
return $ ( ` <div class="avatar"><img src=" ${ group . avatar _url } "></div> ` ) ;
}
const memberAvatars = [ ] ;
if ( group && Array . isArray ( group . members ) && group . members . length ) {
for ( const member of group . members ) {
const charIndex = characters . findIndex ( x => x . avatar === member ) ;
if ( charIndex !== - 1 && characters [ charIndex ] . avatar !== "none" ) {
const avatar = getThumbnailUrl ( 'avatar' , characters [ charIndex ] . avatar ) ;
memberAvatars . push ( avatar ) ;
}
if ( memberAvatars . length === 4 ) {
break ;
}
}
}
const avatarCount = memberAvatars . length ;
if ( avatarCount >= 1 && avatarCount <= 4 ) {
const groupAvatar = $ ( ` #group_avatars_template .collage_ ${ avatarCount } ` ) . clone ( ) ;
for ( let i = 0 ; i < avatarCount ; i ++ ) {
groupAvatar . find ( ` .img_ ${ i + 1 } ` ) . attr ( "src" , memberAvatars [ i ] ) ;
}
return groupAvatar ;
}
// default avatar
const groupAvatar = $ ( "#group_avatars_template .collage_1" ) . clone ( ) ;
groupAvatar . find ( ".img_1" ) . attr ( "src" , group . avatar _url ) ;
return groupAvatar ;
}
async function generateGroupWrapper ( by _auto _mode , type = null , params = { } ) {
if ( online _status === "no_connection" ) {
is _group _generating = false ;
setSendButtonState ( false ) ;
return ;
}
if ( is _group _generating ) {
return false ;
}
// Auto-navigate back to group menu
if ( menu _type !== "group_edit" ) {
select _group _chats ( selected _group ) ;
await delay ( 1 ) ;
}
const group = groups . find ( ( x ) => x . id === selected _group ) ;
let typingIndicator = $ ( "#chat .typing_indicator" ) ;
if ( ! group || ! Array . isArray ( group . members ) || ! group . members . length ) {
sendSystemMessage ( system _message _types . EMPTY , '' , { isSmallSys : true } ) ;
return ;
}
try {
hideSwipeButtons ( ) ;
is _group _generating = true ;
setCharacterName ( '' ) ;
setCharacterId ( undefined ) ;
const userInput = $ ( "#send_textarea" ) . val ( ) ;
if ( typingIndicator . length === 0 && ! isStreamingEnabled ( ) ) {
typingIndicator = $ (
"#typing_indicator_template .typing_indicator"
) . clone ( ) ;
typingIndicator . hide ( ) ;
$ ( "#chat" ) . append ( typingIndicator ) ;
}
// id of this specific batch for regeneration purposes
group _generation _id = Date . now ( ) ;
const lastMessage = chat [ chat . length - 1 ] ;
let messagesBefore = chat . length ;
let lastMessageText = lastMessage ? . mes || '' ;
let activationText = "" ;
let isUserInput = false ;
let isGenerationDone = false ;
let isGenerationAborted = false ;
if ( userInput ? . length && ! by _auto _mode ) {
isUserInput = true ;
activationText = userInput ;
messagesBefore ++ ;
} else {
if ( lastMessage && ! lastMessage . is _system ) {
activationText = lastMessage . mes ;
}
}
const resolveOriginal = params . resolve ;
const rejectOriginal = params . reject ;
if ( params . signal instanceof AbortSignal ) {
if ( params . signal . aborted ) {
throw new Error ( 'Already aborted signal passed. Group generation stopped' ) ;
}
params . signal . onabort = ( ) => {
isGenerationAborted = true ;
} ;
}
if ( typeof params . resolve === 'function' ) {
params . resolve = function ( ) {
isGenerationDone = true ;
resolveOriginal . apply ( this , arguments ) ;
} ;
}
if ( typeof params . reject === 'function' ) {
params . reject = function ( ) {
isGenerationDone = true ;
rejectOriginal . apply ( this , arguments ) ;
}
}
const activationStrategy = Number ( group . activation _strategy ? ? group _activation _strategy . NATURAL ) ;
const enabledMembers = group . members . filter ( x => ! group . disabled _members . includes ( x ) ) ;
let activatedMembers = [ ] ;
if ( params && typeof params . force _chid == 'number' ) {
activatedMembers = [ params . force _chid ] ;
} else if ( type === "quiet" ) {
activatedMembers = activateSwipe ( group . members ) ;
if ( activatedMembers . length === 0 ) {
activatedMembers = activateListOrder ( group . members . slice ( 0 , 1 ) ) ;
}
}
else if ( type === "swipe" || type === 'continue' ) {
activatedMembers = activateSwipe ( group . members ) ;
if ( activatedMembers . length === 0 ) {
toastr . warning ( 'Deleted group member swiped. To get a reply, add them back to the group.' ) ;
throw new Error ( 'Deleted group member swiped' ) ;
}
}
else if ( type === "impersonate" ) {
$ ( "#send_textarea" ) . attr ( "disabled" , true ) ;
activatedMembers = activateImpersonate ( group . members ) ;
}
else if ( activationStrategy === group _activation _strategy . NATURAL ) {
activatedMembers = activateNaturalOrder ( enabledMembers , activationText , lastMessage , group . allow _self _responses , isUserInput ) ;
}
else if ( activationStrategy === group _activation _strategy . LIST ) {
activatedMembers = activateListOrder ( enabledMembers ) ;
}
if ( activatedMembers . length === 0 ) {
toastr . warning ( 'All group members are disabled. Enable at least one to get a reply.' ) ;
// Send user message as is
const bias = getBiasStrings ( userInput , type ) ;
await sendMessageAsUser ( userInput , bias . messageBias ) ;
await saveChatConditional ( ) ;
$ ( '#send_textarea' ) . val ( '' ) . trigger ( 'input' ) ;
}
// now the real generation begins: cycle through every activated character
for ( const chId of activatedMembers ) {
deactivateSendButtons ( ) ;
isGenerationDone = false ;
const generateType = type == "swipe" || type == "impersonate" || type == "quiet" || type == 'continue' ? type : "group_chat" ;
setCharacterId ( chId ) ;
setCharacterName ( characters [ chId ] . name )
await Generate ( generateType , { automatic _trigger : by _auto _mode , ... ( params || { } ) } ) ;
if ( type !== "swipe" && type !== "impersonate" && ! isMultigenEnabled ( ) && ! isStreamingEnabled ( ) ) {
// update indicator and scroll down
typingIndicator
. find ( ".typing_indicator_name" )
. text ( characters [ chId ] . name ) ;
$ ( "#chat" ) . append ( typingIndicator ) ;
typingIndicator . show ( 200 , function ( ) {
typingIndicator . get ( 0 ) . scrollIntoView ( { behavior : "smooth" } ) ;
} ) ;
}
// TODO: This is awful. Refactor this
while ( true ) {
deactivateSendButtons ( ) ;
if ( isGenerationAborted ) {
throw new Error ( 'Group generation aborted' ) ;
}
// if not swipe - check if message generated already
if ( generateType === "group_chat" && ! isMultigenEnabled ( ) && chat . length == messagesBefore ) {
await delay ( 100 ) ;
}
// if swipe - see if message changed
else if ( type === "swipe" ) {
if ( isStreamingEnabled ( ) ) {
if ( streamingProcessor && ! streamingProcessor . isFinished ) {
await delay ( 100 ) ;
}
else {
break ;
}
}
else if ( isMultigenEnabled ( ) ) {
if ( isGenerationDone ) {
break ;
} else {
await delay ( 100 ) ;
}
}
else {
if ( lastMessageText === chat [ chat . length - 1 ] . mes ) {
await delay ( 100 ) ;
}
else {
break ;
}
}
}
else if ( type === "impersonate" ) {
if ( isStreamingEnabled ( ) ) {
if ( streamingProcessor && ! streamingProcessor . isFinished ) {
await delay ( 100 ) ;
}
else {
break ;
}
}
else if ( isMultigenEnabled ( ) ) {
if ( isGenerationDone ) {
break ;
} else {
await delay ( 100 ) ;
}
}
else {
if ( ! $ ( "#send_textarea" ) . val ( ) || $ ( "#send_textarea" ) . val ( ) == userInput ) {
await delay ( 100 ) ;
}
else {
break ;
}
}
}
else if ( type === 'quiet' ) {
if ( isGenerationDone ) {
break ;
} else {
await delay ( 100 ) ;
}
}
else if ( isMultigenEnabled ( ) ) {
if ( isGenerationDone ) {
messagesBefore ++ ;
break ;
} else {
await delay ( 100 ) ;
}
}
else if ( isStreamingEnabled ( ) ) {
if ( streamingProcessor && ! streamingProcessor . isFinished ) {
await delay ( 100 ) ;
} else {
messagesBefore ++ ;
break ;
}
}
else {
messagesBefore ++ ;
break ;
}
}
}
} finally {
// hide and reapply the indicator to the bottom of the list
typingIndicator . hide ( 200 ) ;
$ ( "#chat" ) . append ( typingIndicator ) ;
is _group _generating = false ;
$ ( "#send_textarea" ) . attr ( "disabled" , false ) ;
setSendButtonState ( false ) ;
setCharacterId ( undefined ) ;
setCharacterName ( '' ) ;
activateSendButtons ( ) ;
showSwipeButtons ( ) ;
}
}
function getLastMessageGenerationId ( ) {
let generationId = null ;
if ( chat . length > 0 ) {
const lastMes = chat [ chat . length - 1 ] ;
if ( ! lastMes . is _user && ! lastMes . is _system && lastMes . extra ) {
generationId = lastMes . extra . gen _id ;
}
}
return generationId ;
}
function activateImpersonate ( members ) {
const randomIndex = Math . floor ( Math . random ( ) * members . length ) ;
const activatedMembers = [ members [ randomIndex ] ] ;
const memberIds = activatedMembers
. map ( ( x ) => characters . findIndex ( ( y ) => y . avatar === x ) )
. filter ( ( x ) => x !== - 1 ) ;
return memberIds ;
}
function activateSwipe ( members ) {
let activatedNames = [ ] ;
// pre-update group chat swipe
if ( ! chat [ chat . length - 1 ] . original _avatar ) {
const matches = characters . filter ( x => x . name == chat [ chat . length - 1 ] . name ) ;
for ( const match of matches ) {
if ( members . includes ( match . avatar ) ) {
activatedNames . push ( match . avatar ) ;
break ;
}
}
}
else {
activatedNames . push ( chat [ chat . length - 1 ] . original _avatar ) ;
}
const memberIds = activatedNames
. map ( ( x ) => characters . findIndex ( ( y ) => y . avatar === x ) )
. filter ( ( x ) => x !== - 1 ) ;
return memberIds ;
}
function activateListOrder ( members ) {
let activatedMembers = members . filter ( onlyUnique ) ;
// map to character ids
const memberIds = activatedMembers
. map ( ( x ) => characters . findIndex ( ( y ) => y . avatar === x ) )
. filter ( ( x ) => x !== - 1 ) ;
return memberIds ;
}
function activateNaturalOrder ( members , input , lastMessage , allowSelfResponses , isUserInput ) {
let activatedMembers = [ ] ;
// prevents the same character from speaking twice
let bannedUser = ! isUserInput && lastMessage && ! lastMessage . is _user && lastMessage . name ;
// ...unless allowed to do so
if ( allowSelfResponses ) {
bannedUser = undefined ;
}
// find mentions (excluding self)
if ( input && input . length ) {
for ( let inputWord of extractAllWords ( input ) ) {
for ( let member of members ) {
const character = characters . find ( x => x . avatar === member )
if ( ! character || character . name === bannedUser ) {
continue ;
}
if ( extractAllWords ( character . name ) . includes ( inputWord ) ) {
activatedMembers . push ( member ) ;
break ;
}
}
}
}
// activation by talkativeness (in shuffled order, except banned)
const shuffledMembers = shuffle ( [ ... members ] ) ;
for ( let member of shuffledMembers ) {
const character = characters . find ( ( x ) => x . avatar === member ) ;
if ( ! character || character . name === bannedUser ) {
continue ;
}
const rollValue = Math . random ( ) ;
let talkativeness = Number ( character . talkativeness ) ;
talkativeness = Number . isNaN ( talkativeness )
? talkativeness _default
: talkativeness ;
if ( talkativeness >= rollValue ) {
activatedMembers . push ( member ) ;
}
}
// pick 1 at random if no one was activated
let retries = 0 ;
while ( activatedMembers . length === 0 && ++ retries <= members . length ) {
const randomIndex = Math . floor ( Math . random ( ) * members . length ) ;
const character = characters . find ( ( x ) => x . avatar === members [ randomIndex ] ) ;
if ( ! character ) {
continue ;
}
activatedMembers . push ( members [ randomIndex ] ) ;
}
// de-duplicate array of character avatars
activatedMembers = activatedMembers . filter ( onlyUnique ) ;
// map to character ids
const memberIds = activatedMembers
. map ( ( x ) => characters . findIndex ( ( y ) => y . avatar === x ) )
. filter ( ( x ) => x !== - 1 ) ;
return memberIds ;
}
async function deleteGroup ( id ) {
const response = await fetch ( "/deletegroup" , {
method : "POST" ,
headers : getRequestHeaders ( ) ,
body : JSON . stringify ( { id : id } ) ,
} ) ;
if ( response . ok ) {
selected _group = null ;
delete tag _map [ id ] ;
resetChatState ( ) ;
clearChat ( ) ;
printMessages ( ) ;
await getCharacters ( ) ;
$ ( "#rm_info_avatar" ) . html ( "" ) ;
$ ( "#rm_info_block" ) . transition ( { opacity : 0 , duration : 0 } ) ;
select _rm _info ( "group_delete" , id ) ;
$ ( "#rm_info_block" ) . transition ( { opacity : 1.0 , duration : 2000 } ) ;
$ ( "#rm_button_selected_ch" ) . children ( "h2" ) . text ( '' ) ;
setRightTabSelectedClass ( ) ;
}
}
export async function editGroup ( id , immediately , reload = true ) {
let group = groups . find ( ( x ) => x . id === id ) ;
if ( ! group ) {
return ;
}
group [ 'chat_metadata' ] = chat _metadata ;
if ( immediately ) {
return await _save ( group , reload ) ;
}
saveGroupDebounced ( group ) ;
}
let groupAutoModeAbortController = null ;
async function groupChatAutoModeWorker ( ) {
if ( ! is _group _automode _enabled || online _status === "no_connection" ) {
return ;
}
if ( ! selected _group || is _send _press || is _group _generating ) {
return ;
}
const group = groups . find ( ( x ) => x . id === selected _group ) ;
if ( ! group || ! Array . isArray ( group . members ) || ! group . members . length ) {
return ;
}
groupAutoModeAbortController = new AbortController ( ) ;
await generateGroupWrapper ( true , 'auto' , { signal : groupAutoModeAbortController . signal } ) ;
}
async function modifyGroupMember ( chat _id , groupMember , isDelete ) {
const id = groupMember . data ( "id" ) ;
const template = groupMember . clone ( ) ;
let _thisGroup = groups . find ( ( x ) => x . id == chat _id ) ;
template . data ( "id" , id ) ;
if ( isDelete ) {
$ ( "#rm_group_add_members" ) . prepend ( template ) ;
} else {
$ ( "#rm_group_members" ) . prepend ( template ) ;
}
if ( _thisGroup ) {
if ( isDelete ) {
const index = _thisGroup . members . findIndex ( ( x ) => x === id ) ;
if ( index !== - 1 ) {
_thisGroup . members . splice ( index , 1 ) ;
}
} else {
_thisGroup . members . push ( id ) ;
template . css ( { 'order' : _thisGroup . members . length } ) ;
}
await editGroup ( selected _group ) ;
updateGroupAvatar ( _thisGroup ) ;
}
else {
template . css ( { 'order' : 'unset' } ) ;
}
groupMember . remove ( ) ;
const groupHasMembers = ! ! $ ( "#rm_group_members" ) . children ( ) . length ;
$ ( "#rm_group_submit" ) . prop ( "disabled" , ! groupHasMembers ) ;
}
async function reorderGroupMember ( chat _id , groupMember , direction ) {
const id = groupMember . data ( "id" ) ;
const group = groups . find ( ( x ) => x . id == chat _id ) ;
// Existing groups need to modify members list
if ( group && group . members . length > 1 ) {
const indexOf = group . members . indexOf ( id ) ;
if ( direction == 'down' ) {
const next = group . members [ indexOf + 1 ] ;
if ( next ) {
group . members [ indexOf + 1 ] = group . members [ indexOf ] ;
group . members [ indexOf ] = next ;
}
}
if ( direction == 'up' ) {
const prev = group . members [ indexOf - 1 ] ;
if ( prev ) {
group . members [ indexOf - 1 ] = group . members [ indexOf ] ;
group . members [ indexOf ] = prev ;
}
}
await editGroup ( chat _id ) ;
updateGroupAvatar ( group ) ;
// stupid but lifts the manual reordering
select _group _chats ( chat _id , true ) ;
}
// New groups just can't be DOM-ordered
else {
if ( direction == 'down' ) {
groupMember . insertAfter ( groupMember . next ( ) ) ;
}
if ( direction == 'up' ) {
groupMember . insertBefore ( groupMember . prev ( ) ) ;
}
}
}
function select _group _chats ( groupId , skipAnimation ) {
const group = groupId && groups . find ( ( x ) => x . id == groupId ) ;
const groupName = group ? . name ? ? "" ;
setMenuType ( ! ! group ? 'group_edit' : 'group_create' ) ;
$ ( "#group_avatar_preview" ) . empty ( ) . append ( getGroupAvatar ( group ) ) ;
$ ( "#rm_group_restore_avatar" ) . toggle ( ! ! group && isDataURL ( group . avatar _url ) ) ;
$ ( "#rm_group_chat_name" ) . val ( groupName ) ;
$ ( "#rm_group_chat_name" ) . off ( ) ;
$ ( "#rm_group_chat_name" ) . on ( "input" , async function ( ) {
if ( groupId ) {
let _thisGroup = groups . find ( ( x ) => x . id == groupId ) ;
_thisGroup . name = $ ( this ) . val ( ) ;
$ ( "#rm_button_selected_ch" ) . children ( "h2" ) . text ( _thisGroup . name ) ;
await editGroup ( groupId ) ;
}
} ) ;
$ ( "#rm_group_filter" ) . val ( "" ) . trigger ( "input" ) ;
$ ( 'input[name="rm_group_activation_strategy"]' ) . off ( ) ;
$ ( 'input[name="rm_group_activation_strategy"]' ) . on ( "input" , async function ( e ) {
if ( groupId ) {
let _thisGroup = groups . find ( ( x ) => x . id == groupId ) ;
_thisGroup . activation _strategy = Number ( e . target . value ) ;
await editGroup ( groupId ) ;
}
} ) ;
const replyStrategy = Number ( group ? . activation _strategy ? ? group _activation _strategy . NATURAL ) ;
$ ( ` input[name="rm_group_activation_strategy"][value=" ${ replyStrategy } "] ` ) . prop ( 'checked' , true ) ;
if ( ! skipAnimation ) {
selectRightMenuWithAnimation ( 'rm_group_chats_block' ) ;
}
// render characters list
$ ( "#rm_group_add_members" ) . empty ( ) ;
$ ( "#rm_group_members" ) . empty ( ) ;
for ( let character of characters ) {
const avatar =
character . avatar != "none"
? getThumbnailUrl ( 'avatar' , character . avatar )
: default _avatar ;
const template = $ ( "#group_member_template .group_member" ) . clone ( ) ;
const isFav = character . fav || character . fav == 'true' ;
template . data ( "id" , character . avatar ) ;
template . find ( ".avatar img" ) . attr ( "src" , avatar ) ;
template . find ( ".avatar img" ) . attr ( "title" , character . avatar ) ;
template . find ( ".ch_name" ) . text ( character . name ) ;
template . attr ( "chid" , characters . indexOf ( character ) ) ;
template . find ( '.ch_fav' ) . val ( isFav ) ;
template . toggleClass ( 'is_fav' , isFav ) ;
// Display inline tags
const tags = getTagsList ( character . avatar ) ;
const tagsElement = template . find ( '.tags' ) ;
tags . forEach ( tag => appendTagToList ( tagsElement , tag , { } ) ) ;
if ( ! group ) {
template . find ( '[data-action="speak"]' ) . hide ( ) ;
}
if (
group &&
Array . isArray ( group . members ) &&
group . members . includes ( character . avatar )
) {
template . css ( { 'order' : group . members . indexOf ( character . avatar ) } ) ;
template . toggleClass ( 'disabled' , group . disabled _members . includes ( character . avatar ) ) ;
$ ( "#rm_group_members" ) . append ( template ) ;
} else {
$ ( "#rm_group_add_members" ) . append ( template ) ;
}
}
sortGroupMembers ( "#rm_group_add_members .group_member" ) ;
const groupHasMembers = ! ! $ ( "#rm_group_members" ) . children ( ) . length ;
$ ( "#rm_group_submit" ) . prop ( "disabled" , ! groupHasMembers ) ;
$ ( "#rm_group_allow_self_responses" ) . prop ( "checked" , group && group . allow _self _responses ) ;
// bottom buttons
if ( groupId ) {
$ ( "#rm_group_submit" ) . hide ( ) ;
$ ( "#rm_group_delete" ) . show ( ) ;
$ ( "#rm_group_scenario" ) . show ( ) ;
} else {
$ ( "#rm_group_submit" ) . show ( ) ;
if ( $ ( "#groupAddMemberListToggle .inline-drawer-content" ) . css ( 'display' ) !== 'block' ) {
$ ( "#groupAddMemberListToggle" ) . trigger ( 'click' ) ;
}
$ ( "#rm_group_delete" ) . hide ( ) ;
$ ( "#rm_group_scenario" ) . hide ( ) ;
}
$ ( "#rm_group_delete" ) . off ( ) ;
$ ( "#rm_group_delete" ) . on ( "click" , function ( ) {
if ( is _group _generating ) {
toastr . warning ( 'Not so fast! Wait for the characters to stop typing before deleting the group.' ) ;
return ;
}
$ ( "#dialogue_popup" ) . data ( "group_id" , groupId ) ;
callPopup ( '<h3>Delete the group?</h3><p>This will also delete all your chats with that group. If you want to delete a single conversation, select a "View past chats" option in the lower left menu.</p>' , "del_group" ) ;
} ) ;
updateFavButtonState ( group ? . fav ? ? false ) ;
$ ( "#group_favorite_button" ) . off ( 'click' ) ;
$ ( "#group_favorite_button" ) . on ( 'click' , async function ( ) {
updateFavButtonState ( ! fav _grp _checked ) ;
if ( group ) {
let _thisGroup = groups . find ( ( x ) => x . id == groupId ) ;
_thisGroup . fav = fav _grp _checked ;
await editGroup ( groupId ) ;
}
} ) ;
$ ( "#rm_group_allow_self_responses" ) . off ( ) ;
$ ( "#rm_group_allow_self_responses" ) . on ( "input" , async function ( ) {
if ( group ) {
let _thisGroup = groups . find ( ( x ) => x . id == groupId ) ;
const value = $ ( this ) . prop ( "checked" ) ;
_thisGroup . allow _self _responses = value ;
await editGroup ( groupId ) ;
}
} ) ;
// top bar
if ( group ) {
$ ( "#rm_group_automode_label" ) . show ( ) ;
$ ( "#rm_button_selected_ch" ) . children ( "h2" ) . text ( groupName ) ;
setRightTabSelectedClass ( 'rm_button_selected_ch' ) ;
}
else {
$ ( "#rm_group_automode_label" ) . hide ( ) ;
}
$ ( "#group_avatar_button" ) . off ( 'input' ) . on ( "input" , uploadGroupAvatar ) ;
$ ( "#rm_group_restore_avatar" ) . off ( 'click' ) . on ( "click" , restoreGroupAvatar ) ;
async function uploadGroupAvatar ( event ) {
const file = event . target . files [ 0 ] ;
if ( ! file ) {
return ;
}
const e = await new Promise ( ( resolve , reject ) => {
const reader = new FileReader ( ) ;
reader . onload = resolve ;
reader . onerror = reject ;
reader . readAsDataURL ( file ) ;
} ) ;
$ ( '#dialogue_popup' ) . addClass ( 'large_dialogue_popup wide_dialogue_popup' ) ;
const croppedImage = await callPopup ( getCropPopup ( e . target . result ) , 'avatarToCrop' ) ;
if ( ! croppedImage ) {
return ;
}
const thumbnail = await createThumbnail ( croppedImage , 96 , 144 ) ;
if ( ! groupId ) {
$ ( '#group_avatar_preview img' ) . attr ( 'src' , thumbnail ) ;
$ ( '#rm_group_restore_avatar' ) . show ( ) ;
return ;
}
let _thisGroup = groups . find ( ( x ) => x . id == groupId ) ;
_thisGroup . avatar _url = thumbnail ;
$ ( "#group_avatar_preview" ) . empty ( ) . append ( getGroupAvatar ( _thisGroup ) ) ;
$ ( "#rm_group_restore_avatar" ) . show ( ) ;
await editGroup ( groupId , true , true ) ;
}
async function restoreGroupAvatar ( ) {
const confirm = await callPopup ( '<h3>Are you sure you want to restore the group avatar?</h3> Your custom image will be deleted, and a collage will be used instead.' , 'confirm' ) ;
if ( ! confirm ) {
return ;
}
if ( ! groupId ) {
$ ( "#group_avatar_preview img" ) . attr ( "src" , default _avatar ) ;
$ ( "#rm_group_restore_avatar" ) . hide ( ) ;
return ;
}
let _thisGroup = groups . find ( ( x ) => x . id == groupId ) ;
_thisGroup . avatar _url = '' ;
$ ( "#group_avatar_preview" ) . empty ( ) . append ( getGroupAvatar ( _thisGroup ) ) ;
$ ( "#rm_group_restore_avatar" ) . hide ( ) ;
await editGroup ( groupId , true , true ) ;
}
$ ( document ) . off ( "click" , ".group_member .right_menu_button" ) ;
$ ( document ) . on ( "click" , ".group_member .right_menu_button" , async function ( event ) {
event . stopPropagation ( ) ;
const action = $ ( this ) . data ( 'action' ) ;
const member = $ ( this ) . closest ( '.group_member' ) ;
if ( action === 'remove' ) {
await modifyGroupMember ( groupId , member , true ) ;
}
if ( action === 'add' ) {
await modifyGroupMember ( groupId , member , false ) ;
}
if ( action === 'enable' ) {
member . removeClass ( 'disabled' ) ;
const _thisGroup = groups . find ( x => x . id === groupId ) ;
const index = _thisGroup . disabled _members . indexOf ( member . data ( 'id' ) ) ;
if ( index !== - 1 ) {
_thisGroup . disabled _members . splice ( index , 1 ) ;
}
await editGroup ( groupId ) ;
}
if ( action === 'disable' ) {
member . addClass ( 'disabled' ) ;
const _thisGroup = groups . find ( x => x . id === groupId ) ;
_thisGroup . disabled _members . push ( member . data ( 'id' ) ) ;
await editGroup ( groupId ) ;
}
if ( action === 'up' || action === 'down' ) {
await reorderGroupMember ( groupId , member , action ) ;
}
if ( action === 'view' ) {
openCharacterDefinition ( member ) ;
}
if ( action === 'speak' ) {
const chid = Number ( member . attr ( 'chid' ) ) ;
if ( Number . isInteger ( chid ) ) {
Generate ( 'normal' , { force _chid : chid } ) ;
}
}
sortGroupMembers ( "#rm_group_add_members .group_member" ) ;
await eventSource . emit ( event _types . GROUP _UPDATED ) ;
} ) ;
2023-06-04 17:45:03 +02:00
2023-06-05 13:21:07 +02:00
eventSource . emit ( 'groupSelected' , { detail : { id : groupId , group : group } } ) ;
2023-07-20 19:32:15 +02:00
}
function updateFavButtonState ( state ) {
fav _grp _checked = state ;
$ ( "#rm_group_fav" ) . val ( fav _grp _checked ) ;
$ ( "#group_favorite_button" ) . toggleClass ( 'fav_on' , fav _grp _checked ) ;
$ ( "#group_favorite_button" ) . toggleClass ( 'fav_off' , ! fav _grp _checked ) ;
}
async function selectGroup ( ) {
const groupId = $ ( this ) . data ( "id" ) ;
if ( ! is _send _press && ! is _group _generating ) {
if ( selected _group !== groupId ) {
cancelTtsPlay ( ) ;
selected _group = groupId ;
setCharacterId ( undefined ) ;
setCharacterName ( '' ) ;
setEditedMessageId ( undefined ) ;
clearChat ( ) ;
updateChatMetadata ( { } , true ) ;
chat . length = 0 ;
await getGroupChat ( groupId ) ;
}
select _group _chats ( groupId ) ;
}
}
function openCharacterDefinition ( characterSelect ) {
if ( is _group _generating ) {
console . warn ( "Can't peek a character def while group reply is being generated" ) ;
return ;
}
const chid = characterSelect . attr ( 'chid' ) ;
if ( chid === null || chid === undefined ) {
return ;
}
setCharacterId ( chid ) ;
select _selected _character ( chid ) ;
// Gentle nudge to recalculate tokens
RA _CountCharTokens ( ) ;
// Do a little tomfoolery to spoof the tag selector
applyTagsOnCharacterSelect . call ( characterSelect ) ;
}
function filterGroupMembers ( ) {
const searchValue = $ ( this ) . val ( ) . trim ( ) . toLowerCase ( ) ;
if ( ! searchValue ) {
$ ( "#rm_group_add_members .group_member" ) . removeClass ( 'hiddenBySearch' ) ;
} else {
$ ( "#rm_group_add_members .group_member" ) . each ( function ( ) {
const isValidSearch = $ ( this ) . find ( ".ch_name" ) . text ( ) . toLowerCase ( ) . includes ( searchValue ) ;
$ ( this ) . toggleClass ( 'hiddenBySearch' , ! isValidSearch ) ;
} ) ;
}
}
async function createGroup ( ) {
let name = $ ( "#rm_group_chat_name" ) . val ( ) ;
let allow _self _responses = ! ! $ ( "#rm_group_allow_self_responses" ) . prop ( "checked" ) ;
let activation _strategy = $ ( 'input[name="rm_group_activation_strategy"]:checked' ) . val ( ) ? ? group _activation _strategy . NATURAL ;
const members = $ ( "#rm_group_members .group_member" )
. map ( ( _ , x ) => $ ( x ) . data ( "id" ) )
. toArray ( ) ;
const memberNames = characters . filter ( x => members . includes ( x . avatar ) ) . map ( x => x . name ) . join ( ", " ) ;
if ( ! name ) {
name = ` Group: ${ memberNames } ` ;
}
const avatar _url = $ ( '#group_avatar_preview img' ) . attr ( 'src' ) ;
const chatName = humanizedDateTime ( ) ;
const chats = [ chatName ] ;
const createGroupResponse = await fetch ( "/creategroup" , {
method : "POST" ,
headers : getRequestHeaders ( ) ,
body : JSON . stringify ( {
name : name ,
members : members ,
avatar _url : isDataURL ( avatar _url ) ? avatar _url : default _avatar ,
allow _self _responses : allow _self _responses ,
activation _strategy : activation _strategy ,
disabled _members : [ ] ,
chat _metadata : { } ,
fav : fav _grp _checked ,
chat _id : chatName ,
chats : chats ,
} ) ,
} ) ;
if ( createGroupResponse . ok ) {
const data = await createGroupResponse . json ( ) ;
createTagMapFromList ( "#groupTagList" , data . id ) ;
await getCharacters ( ) ;
select _rm _info ( 'group_create' , data . id ) ;
}
}
export async function createNewGroupChat ( groupId ) {
const group = groups . find ( x => x . id === groupId ) ;
if ( ! group ) {
return ;
}
const oldChatName = group . chat _id ;
const newChatName = humanizedDateTime ( ) ;
if ( typeof group . past _metadata !== 'object' ) {
group . past _metadata = { } ;
}
clearChat ( ) ;
chat . length = 0 ;
if ( oldChatName ) {
group . past _metadata [ oldChatName ] = Object . assign ( { } , chat _metadata ) ;
}
group . chats . push ( newChatName ) ;
group . chat _id = newChatName ;
group . chat _metadata = { } ;
updateChatMetadata ( group . chat _metadata , true ) ;
await editGroup ( group . id , true ) ;
await getGroupChat ( group . id ) ;
}
export async function getGroupPastChats ( groupId ) {
const group = groups . find ( x => x . id === groupId ) ;
if ( ! group ) {
return [ ] ;
}
const chats = [ ] ;
try {
for ( const chatId of group . chats ) {
const messages = await loadGroupChat ( chatId ) ;
let this _chat _file _size = ( JSON . stringify ( messages ) . length / 1024 ) . toFixed ( 2 ) + "kb" ;
let chat _items = messages . length ;
const lastMessage = messages . length ? messages [ messages . length - 1 ] . mes : '[The chat is empty]' ;
const lastMessageDate = messages . length ? ( messages [ messages . length - 1 ] . send _date || Date . now ( ) ) : Date . now ( ) ;
chats . push ( {
'file_name' : chatId ,
'mes' : lastMessage ,
'last_mes' : lastMessageDate ,
'file_size' : this _chat _file _size ,
'chat_items' : chat _items ,
} ) ;
}
} catch ( err ) {
console . error ( err ) ;
}
finally {
return chats ;
}
}
export async function openGroupChat ( groupId , chatId ) {
const group = groups . find ( x => x . id === groupId ) ;
if ( ! group || ! group . chats . includes ( chatId ) ) {
return ;
}
clearChat ( ) ;
chat . length = 0 ;
const previousChat = group . chat _id ;
group . past _metadata [ previousChat ] = Object . assign ( { } , chat _metadata ) ;
group . chat _id = chatId ;
group . chat _metadata = group . past _metadata [ chatId ] || { } ;
group [ 'date_last_chat' ] = Date . now ( ) ;
updateChatMetadata ( group . chat _metadata , true ) ;
await editGroup ( groupId , true ) ;
await getGroupChat ( groupId ) ;
sortCharactersList ( ) ;
}
export async function renameGroupChat ( groupId , oldChatId , newChatId ) {
const group = groups . find ( x => x . id === groupId ) ;
if ( ! group || ! group . chats . includes ( oldChatId ) ) {
return ;
}
if ( group . chat _id === oldChatId ) {
group . chat _id = newChatId ;
}
group . chats . splice ( group . chats . indexOf ( oldChatId ) , 1 ) ;
group . chats . push ( newChatId ) ;
group . past _metadata [ newChatId ] = ( group . past _metadata [ oldChatId ] || { } ) ;
delete group . past _metadata [ oldChatId ] ;
await editGroup ( groupId , true , true ) ;
}
export async function deleteGroupChat ( groupId , chatId ) {
const group = groups . find ( x => x . id === groupId ) ;
if ( ! group || ! group . chats . includes ( chatId ) ) {
return ;
}
group . chats . splice ( group . chats . indexOf ( chatId ) , 1 ) ;
group . chat _metadata = { } ;
group . chat _id = '' ;
delete group . past _metadata [ chatId ] ;
updateChatMetadata ( group . chat _metadata , true ) ;
const response = await fetch ( '/deletegroupchat' , {
method : 'POST' ,
headers : getRequestHeaders ( ) ,
body : JSON . stringify ( { id : chatId } ) ,
} ) ;
if ( response . ok ) {
if ( group . chats . length ) {
await openGroupChat ( groupId , group . chats [ 0 ] ) ;
} else {
await createNewGroupChat ( groupId ) ;
}
}
}
export async function importGroupChat ( formData ) {
await jQuery . ajax ( {
type : "POST" ,
url : "/importgroupchat" ,
data : formData ,
beforeSend : function ( ) {
} ,
cache : false ,
contentType : false ,
processData : false ,
success : async function ( data ) {
if ( data . res ) {
const chatId = data . res ;
const group = groups . find ( x => x . id == selected _group ) ;
if ( group ) {
group . chats . push ( chatId ) ;
await editGroup ( selected _group , true , true ) ;
await displayPastChats ( ) ;
}
}
} ,
error : function ( ) {
$ ( "#create_button" ) . removeAttr ( "disabled" ) ;
} ,
} ) ;
}
export async function saveGroupBookmarkChat ( groupId , name , metadata , mesId ) {
const group = groups . find ( x => x . id === groupId ) ;
if ( ! group ) {
return ;
}
group . past _metadata [ name ] = { ... chat _metadata , ... ( metadata || { } ) } ;
group . chats . push ( name ) ;
const trimmed _chat = ( mesId !== undefined && mesId >= 0 && mesId < chat . length )
? chat . slice ( 0 , parseInt ( mesId ) + 1 )
: chat ;
await editGroup ( groupId , true ) ;
await fetch ( "/savegroupchat" , {
method : "POST" ,
headers : getRequestHeaders ( ) ,
body : JSON . stringify ( { id : name , chat : [ ... trimmed _chat ] } ) ,
} ) ;
}
function onSendTextareaInput ( ) {
if ( is _group _automode _enabled ) {
// Wait for current automode generation to finish
is _group _automode _enabled = false ;
$ ( "#rm_group_automode" ) . prop ( "checked" , false ) ;
}
}
function stopAutoModeGeneration ( ) {
if ( groupAutoModeAbortController ) {
groupAutoModeAbortController . abort ( ) ;
}
is _group _automode _enabled = false ;
$ ( "#rm_group_automode" ) . prop ( "checked" , false ) ;
}
function doCurMemberListPopout ( ) {
//repurposes the zoomed avatar template to server as a floating group member list
if ( $ ( "#groupMemberListPopout" ) . length === 0 ) {
console . debug ( 'did not see popout yet, creating' )
const memberListClone = $ ( this ) . parent ( ) . parent ( ) . find ( '.inline-drawer-content' ) . html ( )
const template = $ ( '#zoomed_avatar_template' ) . html ( ) ;
const controlBarHtml = ` <div class="panelControlBar flex-container">
< div id = "groupMemberListPopoutheader" class = "fa-solid fa-grip drag-grabber hoverglow" > < / d i v >
< div id = "groupMemberListPopoutClose" class = "fa-solid fa-circle-xmark hoverglow" > < / d i v >
< / d i v > `
const newElement = $ ( template ) ;
newElement . attr ( 'id' , 'groupMemberListPopout' ) ;
newElement . removeClass ( 'zoomed_avatar' )
newElement . empty ( )
newElement . append ( controlBarHtml ) . append ( memberListClone )
$ ( 'body' ) . append ( newElement ) ;
loadMovingUIState ( ) ;
$ ( "#groupMemberListPopout" ) . fadeIn ( 250 )
dragElement ( newElement )
$ ( '#groupMemberListPopoutClose' ) . off ( 'click' ) . on ( 'click' , function ( ) {
$ ( "#groupMemberListPopout" ) . fadeOut ( 250 , ( ) => { $ ( "#groupMemberListPopout" ) . remove ( ) } )
} )
} else {
console . debug ( 'saw existing popout, removing' )
$ ( "#groupMemberListPopout" ) . fadeOut ( 250 , ( ) => { $ ( "#groupMemberListPopout" ) . remove ( ) } ) ;
}
}
jQuery ( ( ) => {
$ ( document ) . on ( "click" , ".group_select" , selectGroup ) ;
$ ( "#rm_group_filter" ) . on ( "input" , filterGroupMembers ) ;
$ ( "#rm_group_submit" ) . on ( "click" , createGroup ) ;
$ ( "#rm_group_scenario" ) . on ( "click" , setScenarioOverride ) ;
$ ( "#rm_group_automode" ) . on ( "input" , function ( ) {
const value = $ ( this ) . prop ( "checked" ) ;
is _group _automode _enabled = value ;
eventSource . once ( event _types . GENERATION _STOPPED , stopAutoModeGeneration ) ;
} ) ;
$ ( "#send_textarea" ) . on ( "keyup" , onSendTextareaInput ) ;
$ ( "#groupCurrentMemberPopoutButton" ) . on ( 'click' , doCurMemberListPopout ) ;
} ) ;