| import { |
| characters, |
| saveChat, |
| system_message_types, |
| this_chid, |
| openCharacterChat, |
| chat_metadata, |
| getRequestHeaders, |
| getThumbnailUrl, |
| getCharacters, |
| chat, |
| saveChatConditional, |
| saveItemizedPrompts, |
| setActiveGroup, |
| } from '../script.js'; |
| import { humanizedDateTime } from './RossAscends-mods.js'; |
| import { |
| DEFAULT_AUTO_MODE_DELAY, |
| group_activation_strategy, |
| group_generation_mode, |
| groups, |
| openGroupById, |
| openGroupChat, |
| saveGroupBookmarkChat, |
| selected_group, |
| } from './group-chats.js'; |
| import { hideLoader, showLoader } from './loader.js'; |
| import { getLastMessageId } from './macros.js'; |
| import { Popup } from './popup.js'; |
| import { SlashCommand } from './slash-commands/SlashCommand.js'; |
| import { ARGUMENT_TYPE, SlashCommandArgument, SlashCommandNamedArgument } from './slash-commands/SlashCommandArgument.js'; |
| import { commonEnumProviders } from './slash-commands/SlashCommandCommonEnumsProvider.js'; |
| import { SlashCommandParser } from './slash-commands/SlashCommandParser.js'; |
| import { createTagMapFromList } from './tags.js'; |
| import { renderTemplateAsync } from './templates.js'; |
| import { t } from './i18n.js'; |
|
|
| import { |
| getUniqueName, |
| isTrueBoolean, |
| } from './utils.js'; |
|
|
| const bookmarkNameToken = 'Checkpoint #'; |
|
|
| |
| |
| |
| |
| async function getExistingChatNames() { |
| if (selected_group) { |
| const group = groups.find(x => x.id == selected_group); |
| if (group && Array.isArray(group.chats)) { |
| return [...group.chats]; |
| } |
|
|
| return []; |
| } |
|
|
| if (this_chid === undefined) { |
| return []; |
| } |
|
|
| const character = characters[this_chid]; |
| if (!character) { |
| return []; |
| } |
|
|
| const response = await fetch('/api/characters/chats', { |
| method: 'POST', |
| headers: getRequestHeaders(), |
| body: JSON.stringify({ avatar_url: character.avatar, simple: true }), |
| }); |
|
|
| if (response.ok) { |
| const data = await response.json(); |
| const chats = Object.values(data).map(x => x.file_name.replace('.jsonl', '')); |
| return [...chats]; |
| } |
|
|
| return []; |
| } |
|
|
| async function getBookmarkName({ isReplace = false, forceName = null } = {}) { |
| const chatNames = await getExistingChatNames(); |
|
|
| const body = await renderTemplateAsync('createCheckpoint', { isReplace: isReplace }); |
| let name = forceName ?? await Popup.show.input('Create Checkpoint', body); |
| |
| if (name === '') { |
| for (let i = chatNames.length; i < 1000; i++) { |
| name = bookmarkNameToken + i; |
| if (!chatNames.includes(name)) { |
| break; |
| } |
| } |
| } |
| if (!name) { |
| return null; |
| } |
|
|
| return `${name} - ${humanizedDateTime()}`; |
| } |
|
|
| function getMainChatName() { |
| if (chat_metadata) { |
| if (chat_metadata['main_chat']) { |
| return chat_metadata['main_chat']; |
| } |
| |
| else if (selected_group) { |
| return null; |
| } |
| else if (characters[this_chid].chat && characters[this_chid].chat.includes(bookmarkNameToken)) { |
| const tokenIndex = characters[this_chid].chat.lastIndexOf(bookmarkNameToken); |
| chat_metadata['main_chat'] = characters[this_chid].chat.substring(0, tokenIndex).trim(); |
| return chat_metadata['main_chat']; |
| } |
| } |
| return null; |
| } |
|
|
| export function showBookmarksButtons() { |
| try { |
| if (selected_group) { |
| $('#option_convert_to_group').hide(); |
| } else { |
| $('#option_convert_to_group').show(); |
| } |
|
|
| if (chat_metadata['main_chat']) { |
| |
| $('#option_back_to_main').show(); |
| $('#option_new_bookmark').show(); |
| } else if (!selected_group && !characters[this_chid].chat) { |
| |
| $('#option_back_to_main').hide(); |
| $('#option_new_bookmark').hide(); |
| } else { |
| |
| $('#option_back_to_main').hide(); |
| $('#option_new_bookmark').show(); |
| } |
| } |
| catch { |
| $('#option_back_to_main').hide(); |
| $('#option_new_bookmark').hide(); |
| $('#option_convert_to_group').hide(); |
| } |
| } |
|
|
| async function saveBookmarkMenu() { |
| if (!chat.length) { |
| toastr.warning('The chat is empty.', 'Checkpoint creation failed'); |
| return; |
| } |
|
|
| return await createNewBookmark(chat.length - 1); |
| } |
|
|
| |
| export async function createBranch(mesId) { |
| if (!chat.length) { |
| toastr.warning('The chat is empty.', 'Branch creation failed'); |
| return; |
| } |
|
|
| if (mesId < 0 || mesId >= chat.length) { |
| toastr.warning('Invalid message ID.', 'Branch creation failed'); |
| return; |
| } |
|
|
| const lastMes = chat[mesId]; |
| const mainChat = selected_group ? groups?.find(x => x.id == selected_group)?.chat_id : characters[this_chid].chat; |
| const newMetadata = { main_chat: mainChat }; |
| let name = `Branch #${mesId} - ${humanizedDateTime()}`; |
|
|
| if (selected_group) { |
| await saveGroupBookmarkChat(selected_group, name, newMetadata, mesId); |
| } else { |
| await saveChat({ chatName: name, withMetadata: newMetadata, mesId }); |
| } |
| |
| |
| if (typeof lastMes.extra !== 'object') { |
| lastMes.extra = {}; |
| } |
| if (typeof lastMes.extra['branches'] !== 'object') { |
| lastMes.extra['branches'] = []; |
| } |
| lastMes.extra['branches'].push(name); |
| return name; |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| export async function createNewBookmark(mesId, { forceName = null } = {}) { |
| if (this_chid === undefined && !selected_group) { |
| toastr.info('No character selected.', 'Create Checkpoint'); |
| return null; |
| } |
| if (!chat.length) { |
| toastr.warning('The chat is empty.', 'Create Checkpoint'); |
| return null; |
| } |
| if (!chat[mesId]) { |
| toastr.warning('Invalid message ID.', 'Create Checkpoint'); |
| return null; |
| } |
|
|
| const lastMes = chat[mesId]; |
|
|
| if (typeof lastMes.extra !== 'object') { |
| lastMes.extra = {}; |
| } |
|
|
| const isReplace = lastMes.extra.bookmark_link; |
|
|
| let name = await getBookmarkName({ isReplace: isReplace, forceName: forceName }); |
| if (!name) { |
| return null; |
| } |
|
|
| const mainChat = selected_group ? groups?.find(x => x.id == selected_group)?.chat_id : characters[this_chid].chat; |
| const newMetadata = { main_chat: mainChat }; |
| await saveItemizedPrompts(name); |
|
|
| if (selected_group) { |
| await saveGroupBookmarkChat(selected_group, name, newMetadata, mesId); |
| } else { |
| await saveChat({ chatName: name, withMetadata: newMetadata, mesId }); |
| } |
|
|
| lastMes.extra['bookmark_link'] = name; |
|
|
| const mes = $(`.mes[mesid="${mesId}"]`); |
| updateBookmarkDisplay(mes, name); |
|
|
| await saveChatConditional(); |
| toastr.success('Click the flag icon next to the message to open the checkpoint chat.', 'Create Checkpoint', { timeOut: 10000 }); |
| return name; |
| } |
|
|
|
|
| |
| |
| |
| |
| |
| export function updateBookmarkDisplay(mes, newBookmarkLink = null) { |
| newBookmarkLink && mes.attr('bookmark_link', newBookmarkLink); |
| const bookmarkFlag = mes.find('.mes_bookmark'); |
| bookmarkFlag.attr('title', `Checkpoint\n${mes.attr('bookmark_link')}\n\n${bookmarkFlag.data('tooltip')}`); |
| } |
|
|
| async function backToMainChat() { |
| const mainChatName = getMainChatName(); |
| const allChats = await getExistingChatNames(); |
|
|
| if (allChats.includes(mainChatName)) { |
| if (selected_group) { |
| await openGroupChat(selected_group, mainChatName); |
| } else { |
| await openCharacterChat(mainChatName); |
| } |
| return mainChatName; |
| } |
|
|
| return null; |
| } |
|
|
| export async function convertSoloToGroupChat() { |
| if (selected_group) { |
| console.log('Already in group. No need for conversion'); |
| return; |
| } |
|
|
| if (this_chid === undefined) { |
| console.log('Need to have a character selected'); |
| return; |
| } |
|
|
| const confirm = await Popup.show.confirm(t`Convert to group chat`, t`Are you sure you want to convert this chat to a group chat?` + '<br />' + t`This cannot be reverted.`); |
| if (!confirm) { |
| return; |
| } |
|
|
| const character = characters[this_chid]; |
|
|
| |
| const name = getUniqueName(`Group: ${character.name}`, y => groups.findIndex(x => x.name === y) !== -1); |
| const avatar = getThumbnailUrl('avatar', character.avatar); |
| const chatName = humanizedDateTime(); |
| const chats = [chatName]; |
| const members = [character.avatar]; |
| const favChecked = character.fav || character.fav == 'true'; |
| |
| const metadata = Object.assign({}, chat_metadata); |
| delete metadata.main_chat; |
| |
| const chatHeader = { |
| chat_metadata: metadata, |
| user_name: 'unused', |
| character_name: 'unused', |
| }; |
| |
| const groupCreateModel = { |
| name: name, |
| members: members, |
| avatar_url: avatar, |
| allow_self_responses: false, |
| activation_strategy: group_activation_strategy.NATURAL, |
| disabled_members: [], |
| fav: favChecked, |
| chat_id: chatName, |
| chats: chats, |
| hideMutedSprites: false, |
| generation_mode: group_generation_mode.SWAP, |
| auto_mode_delay: DEFAULT_AUTO_MODE_DELAY, |
| }; |
|
|
| const createGroupResponse = await fetch('/api/groups/create', { |
| method: 'POST', |
| headers: getRequestHeaders(), |
| body: JSON.stringify(groupCreateModel), |
| }); |
|
|
| if (!createGroupResponse.ok) { |
| console.error('Group creation unsuccessful'); |
| return; |
| } |
|
|
| |
| const group = await createGroupResponse.json(); |
|
|
| |
| createTagMapFromList('#tagList', group.id); |
|
|
| |
| await getCharacters(); |
|
|
| |
| const groupChat = [...chat].map(m => structuredClone(m)); |
| const genIdFirst = Date.now(); |
|
|
| for (let index = 0; index < groupChat.length; index++) { |
| const message = groupChat[index]; |
|
|
| |
| if (message.is_user || message.is_system || message.extra?.type === system_message_types.NARRATOR || message.force_avatar !== undefined) { |
| continue; |
| } |
|
|
| if (!message.extra || typeof message.extra !== 'object') { |
| message.extra = {}; |
| } |
|
|
| |
| message.name = character.name; |
| message.original_avatar = character.avatar; |
| message.force_avatar = getThumbnailUrl('avatar', character.avatar); |
| |
| message.extra.gen_id = genIdFirst + index; |
| } |
|
|
| |
| const createChatResponse = await fetch('/api/chats/group/save', { |
| method: 'POST', |
| headers: getRequestHeaders(), |
| body: JSON.stringify({ id: chatName, chat: [chatHeader, ...groupChat] }), |
| }); |
|
|
| if (!createChatResponse.ok) { |
| console.error('Group chat creation unsuccessful'); |
| toastr.error('Group chat creation unsuccessful'); |
| return; |
| } |
|
|
| |
| setActiveGroup(group.id); |
| await openGroupById(group.id); |
|
|
| toastr.success(t`The chat has been successfully converted!`); |
| } |
|
|
| |
| |
| |
| |
| |
| export async function branchChat(mesId) { |
| if (this_chid === undefined && !selected_group) { |
| toastr.info('No character selected.', 'Create Branch'); |
| return null; |
| } |
|
|
| const fileName = await createBranch(mesId); |
| await saveItemizedPrompts(fileName); |
|
|
| if (selected_group) { |
| await openGroupChat(selected_group, fileName); |
| } else { |
| await openCharacterChat(fileName); |
| } |
|
|
| return fileName; |
| } |
|
|
| function registerBookmarksSlashCommands() { |
| |
| |
| |
| |
| |
| |
| |
| function validateMessageId(mesId, context) { |
| if (isNaN(mesId)) { |
| toastr.warning('Invalid message ID was provided', context); |
| return false; |
| } |
| if (!chat[mesId]) { |
| toastr.warning(`Message for id ${mesId} not found`, context); |
| return false; |
| } |
| return true; |
| } |
|
|
| SlashCommandParser.addCommandObject(SlashCommand.fromProps({ |
| name: 'branch-create', |
| returns: 'Name of the new branch', |
| callback: async (args, text) => { |
| const mesId = Number(args.mesId ?? text ?? getLastMessageId()); |
| if (!validateMessageId(mesId, 'Create Branch')) return ''; |
|
|
| const branchName = await branchChat(mesId); |
| return branchName ?? ''; |
| }, |
| unnamedArgumentList: [ |
| SlashCommandArgument.fromProps({ |
| description: 'Message ID', |
| typeList: [ARGUMENT_TYPE.NUMBER], |
| enumProvider: commonEnumProviders.messages(), |
| }), |
| ], |
| helpString: ` |
| <div> |
| Create a new branch from the selected message. If no message id is provided, will use the last message. |
| </div> |
| <div> |
| Creating a branch will automatically choose a name for the branch.<br /> |
| After creating the branch, the branch chat will be automatically opened. |
| </div> |
| <div> |
| Use Checkpoints and <code>/checkpoint-create</code> instead if you do not want to jump to the new chat. |
| </div>`, |
| })); |
| SlashCommandParser.addCommandObject(SlashCommand.fromProps({ |
| name: 'checkpoint-create', |
| returns: 'Name of the new checkpoint', |
| callback: async (args, text) => { |
| const mesId = Number(args.mesId ?? getLastMessageId()); |
| if (!validateMessageId(mesId, 'Create Checkpoint')) return ''; |
|
|
| if (typeof text !== 'string') { |
| toastr.warning('Checkpoint name must be a string or empty', 'Create Checkpoint'); |
| return ''; |
| } |
|
|
| const checkPointName = await createNewBookmark(mesId, { forceName: text }); |
| return checkPointName ?? ''; |
| }, |
| namedArgumentList: [ |
| SlashCommandNamedArgument.fromProps({ |
| name: 'mesId', |
| description: 'Message ID', |
| typeList: [ARGUMENT_TYPE.NUMBER], |
| enumProvider: commonEnumProviders.messages(), |
| }), |
| ], |
| unnamedArgumentList: [ |
| SlashCommandArgument.fromProps({ |
| description: 'Checkpoint name', |
| typeList: [ARGUMENT_TYPE.STRING], |
| }), |
| ], |
| helpString: ` |
| <div> |
| Create a new checkpoint for the selected message with the provided name. If no message id is provided, will use the last message.<br /> |
| Leave the checkpoint name empty to auto-generate one. |
| </div> |
| <div> |
| A created checkpoint will be permanently linked with the message.<br /> |
| If a checkpoint already exists, the link to it will be overwritten.<br /> |
| After creating the checkpoint, the checkpoint chat can be opened with the checkpoint flag, |
| using the <code>/go</code> command with the checkpoint name or the <code>/checkpoint-go</code> command on the message. |
| </div> |
| <div> |
| Use Branches and <code>/branch-create</code> instead if you do want to jump to the new chat. |
| </div> |
| <div> |
| <strong>Example:</strong> |
| <ul> |
| <li> |
| <pre><code>/checkpoint-create mes={{lastCharMessage}} Checkpoint for char reply | /setvar key=rememberCheckpoint {{pipe}}</code></pre> |
| Will create a new checkpoint to the latest message of the current character, and save it as a local variable for future use. |
| </li> |
| </ul> |
| </div>`, |
| })); |
| SlashCommandParser.addCommandObject(SlashCommand.fromProps({ |
| name: 'checkpoint-go', |
| returns: 'Name of the checkpoint', |
| callback: async (args, text) => { |
| const mesId = Number(args.mesId ?? text ?? getLastMessageId()); |
| if (!validateMessageId(mesId, 'Open Checkpoint')) return ''; |
|
|
| const checkPointName = chat[mesId].extra?.bookmark_link; |
| if (!checkPointName) { |
| toastr.warning('No checkpoint is linked to the selected message', 'Open Checkpoint'); |
| return ''; |
| } |
|
|
| if (selected_group) { |
| await openGroupChat(selected_group, checkPointName); |
| } else { |
| await openCharacterChat(checkPointName); |
| } |
|
|
| return checkPointName; |
| }, |
| unnamedArgumentList: [ |
| SlashCommandArgument.fromProps({ |
| description: 'Message ID', |
| typeList: [ARGUMENT_TYPE.NUMBER], |
| enumProvider: commonEnumProviders.messages(), |
| }), |
| ], |
| helpString: ` |
| <div> |
| Open the checkpoint linked to the selected message. If no message id is provided, will use the last message. |
| </div> |
| <div> |
| Use <code>/checkpoint-get</code> if you want to make sure that the selected message has a checkpoint. |
| </div>`, |
| })); |
| SlashCommandParser.addCommandObject(SlashCommand.fromProps({ |
| name: 'checkpoint-exit', |
| returns: 'The name of the chat exited to. Returns an empty string if not in a checkpoint chat.', |
| callback: async () => { |
| const mainChat = await backToMainChat(); |
| return mainChat ?? ''; |
| }, |
| helpString: 'Exit the checkpoint chat.<br />If not in a checkpoint chat, returns empty string.', |
| })); |
| SlashCommandParser.addCommandObject(SlashCommand.fromProps({ |
| name: 'checkpoint-parent', |
| returns: 'Name of the parent chat for this checkpoint', |
| callback: async () => { |
| const mainChatName = getMainChatName(); |
| return mainChatName ?? ''; |
| }, |
| helpString: 'Get the name of the parent chat for this checkpoint.<br />If not in a checkpoint chat, returns empty string.', |
| })); |
| SlashCommandParser.addCommandObject(SlashCommand.fromProps({ |
| name: 'checkpoint-get', |
| returns: 'Name of the chat', |
| callback: async (args, text) => { |
| const mesId = Number(args.mesId ?? text ?? getLastMessageId()); |
| if (!validateMessageId(mesId, 'Get Checkpoint')) return ''; |
|
|
| const checkPointName = chat[mesId].extra?.bookmark_link; |
| return checkPointName ?? ''; |
| }, |
| unnamedArgumentList: [ |
| SlashCommandArgument.fromProps({ |
| description: 'Message ID', |
| typeList: [ARGUMENT_TYPE.NUMBER], |
| enumProvider: commonEnumProviders.messages(), |
| }), |
| ], |
| helpString: ` |
| <div> |
| Get the name of the checkpoint linked to the selected message. If no message id is provided, will use the last message.<br /> |
| If no checkpoint is linked, the result will be empty. |
| </div>`, |
| })); |
| SlashCommandParser.addCommandObject(SlashCommand.fromProps({ |
| name: 'checkpoint-list', |
| returns: 'JSON array of all existing checkpoints in this chat, as an array', |
| |
| callback: async (args, _) => { |
| const result = Object.entries(chat) |
| .filter(([_, message]) => message.extra?.bookmark_link) |
| .map(([mesId, message]) => isTrueBoolean(args.links) ? message.extra.bookmark_link : Number(mesId)); |
| return JSON.stringify(result); |
| }, |
| namedArgumentList: [ |
| SlashCommandNamedArgument.fromProps({ |
| name: 'links', |
| description: 'Get a list of all links / chat names of the checkpoints, instead of the message ids', |
| typeList: [ARGUMENT_TYPE.BOOLEAN], |
| enumList: commonEnumProviders.boolean('trueFalse')(), |
| defaultValue: 'false', |
| }), |
| ], |
| helpString: ` |
| <div> |
| List all existing checkpoints in this chat. |
| </div> |
| <div> |
| Returns a list of all message ids that have a checkpoint, or all checkpoint links if <code>links</code> is set to <code>true</code>.<br /> |
| The value will be a JSON array. |
| </div>`, |
| })); |
| } |
|
|
| export function initBookmarks() { |
| $('#option_new_bookmark').on('click', saveBookmarkMenu); |
| $('#option_back_to_main').on('click', backToMainChat); |
| $('#option_convert_to_group').on('click', convertSoloToGroupChat); |
|
|
| $(document).on('click', '.select_chat_block, .mes_bookmark', async function (e) { |
| |
| const mes = $(this).closest('.mes'); |
| if (e.shiftKey && mes.length) { |
| const selectedMesId = mes.attr('mesid'); |
| await createNewBookmark(Number(selectedMesId)); |
| return; |
| } |
|
|
| const fileName = $(this).hasClass('mes_bookmark') |
| ? $(this).closest('.mes').attr('bookmark_link') |
| : $(this).attr('file_name').replace('.jsonl', ''); |
|
|
| if (!fileName) { |
| return; |
| } |
|
|
| try { |
| showLoader(); |
| if (selected_group) { |
| await openGroupChat(selected_group, fileName); |
| } else { |
| await openCharacterChat(fileName); |
| } |
| } finally { |
| await hideLoader(); |
| } |
|
|
| $('#shadow_select_chat_popup').css('display', 'none'); |
| }); |
|
|
| $(document).on('click', '.mes_create_bookmark', async function () { |
| const mesId = $(this).closest('.mes').attr('mesid'); |
| if (mesId !== undefined) { |
| await createNewBookmark(Number(mesId)); |
| } |
| }); |
|
|
| $(document).on('click', '.mes_create_branch', async function () { |
| const mesId = $(this).closest('.mes').attr('mesid'); |
| if (mesId !== undefined) { |
| await branchChat(Number(mesId)); |
| } |
| }); |
|
|
| registerBookmarksSlashCommands(); |
| } |
|
|