mirror of
				https://github.com/SillyTavern/SillyTavern.git
				synced 2025-06-05 21:59:27 +02:00 
			
		
		
		
	
		
			
				
	
	
		
			354 lines
		
	
	
		
			10 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			354 lines
		
	
	
		
			10 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
| import {
 | |
|     characters,
 | |
|     saveChat,
 | |
|     system_messages,
 | |
|     system_message_types,
 | |
|     this_chid,
 | |
|     openCharacterChat,
 | |
|     chat_metadata,
 | |
|     callPopup,
 | |
|     getRequestHeaders,
 | |
|     getThumbnailUrl,
 | |
|     getCharacters,
 | |
|     chat,
 | |
|     saveChatConditional,
 | |
|     saveItemizedPrompts,
 | |
| } from '../script.js';
 | |
| import { humanizedDateTime, getMessageTimeStamp } from './RossAscends-mods.js';
 | |
| import {
 | |
|     getGroupPastChats,
 | |
|     group_activation_strategy,
 | |
|     groups,
 | |
|     openGroupById,
 | |
|     openGroupChat,
 | |
|     saveGroupBookmarkChat,
 | |
|     selected_group,
 | |
| } from './group-chats.js';
 | |
| import { Popup } from './popup.js';
 | |
| import { createTagMapFromList } from './tags.js';
 | |
| 
 | |
| import {
 | |
|     delay,
 | |
|     getUniqueName,
 | |
| } from './utils.js';
 | |
| 
 | |
| export {
 | |
|     createNewBookmark,
 | |
|     showBookmarksButtons,
 | |
| };
 | |
| 
 | |
| const bookmarkNameToken = 'Checkpoint #';
 | |
| 
 | |
| async function getExistingChatNames() {
 | |
|     if (selected_group) {
 | |
|         const data = await getGroupPastChats(selected_group);
 | |
|         return data.map(x => x.file_name);
 | |
|     } else {
 | |
|         const response = await fetch('/api/characters/chats', {
 | |
|             method: 'POST',
 | |
|             headers: getRequestHeaders(),
 | |
|             body: JSON.stringify({ avatar_url: characters[this_chid].avatar }),
 | |
|         });
 | |
| 
 | |
|         if (response.ok) {
 | |
|             const data = await response.json();
 | |
|             return Object.values(data).map(x => x.file_name.replace('.jsonl', ''));
 | |
|         }
 | |
|     }
 | |
| }
 | |
| 
 | |
| async function getBookmarkName() {
 | |
|     const chatNames = await getExistingChatNames();
 | |
|     const popupText = `<h3>Enter the checkpoint name:<h3>
 | |
|     <small>Leave empty to auto-generate.</small>`;
 | |
|     let name = await callPopup(popupText, 'input');
 | |
| 
 | |
|     if (name === false) {
 | |
|         return null;
 | |
|     }
 | |
|     else if (name === '') {
 | |
|         for (let i = chatNames.length; i < 1000; i++) {
 | |
|             name = bookmarkNameToken + i;
 | |
|             if (!chatNames.includes(name)) {
 | |
|                 break;
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     return `${name} - ${humanizedDateTime()}`;
 | |
| }
 | |
| 
 | |
| function getMainChatName() {
 | |
|     if (chat_metadata) {
 | |
|         if (chat_metadata['main_chat']) {
 | |
|             return chat_metadata['main_chat'];
 | |
|         }
 | |
|         // groups didn't support bookmarks before chat metadata was introduced
 | |
|         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;
 | |
| }
 | |
| 
 | |
| function showBookmarksButtons() {
 | |
|     try {
 | |
|         if (selected_group) {
 | |
|             $('#option_convert_to_group').hide();
 | |
|         } else {
 | |
|             $('#option_convert_to_group').show();
 | |
|         }
 | |
| 
 | |
|         if (chat_metadata['main_chat']) {
 | |
|             // In bookmark chat
 | |
|             $('#option_back_to_main').show();
 | |
|             $('#option_new_bookmark').show();
 | |
|         } else if (!selected_group && !characters[this_chid].chat) {
 | |
|             // No chat recorded on character
 | |
|             $('#option_back_to_main').hide();
 | |
|             $('#option_new_bookmark').hide();
 | |
|         } else {
 | |
|             // In main chat
 | |
|             $('#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 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(name, newMetadata, mesId);
 | |
|     }
 | |
|     // append to branches list if it exists
 | |
|     // otherwise create it
 | |
|     if (typeof lastMes.extra !== 'object') {
 | |
|         lastMes.extra = {};
 | |
|     }
 | |
|     if (typeof lastMes.extra['branches'] !== 'object') {
 | |
|         lastMes.extra['branches'] = [];
 | |
|     }
 | |
|     lastMes.extra['branches'].push(name);
 | |
|     return name;
 | |
| }
 | |
| 
 | |
| async function createNewBookmark(mesId) {
 | |
|     if (!chat.length) {
 | |
|         toastr.warning('The chat is empty.', 'Checkpoint creation failed');
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     if (mesId < 0 || mesId >= chat.length) {
 | |
|         toastr.warning('Invalid message ID.', 'Checkpoint creation failed');
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     const lastMes = chat[mesId];
 | |
| 
 | |
|     if (typeof lastMes.extra !== 'object') {
 | |
|         lastMes.extra = {};
 | |
|     }
 | |
| 
 | |
|     if (lastMes.extra.bookmark_link) {
 | |
|         const confirm = await callPopup('Checkpoint for the last message already exists. Would you like to replace it?', 'confirm');
 | |
| 
 | |
|         if (!confirm) {
 | |
|             return;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     await delay(250);
 | |
|     let name = await getBookmarkName();
 | |
| 
 | |
|     if (!name) {
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     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(name, newMetadata, mesId);
 | |
|     }
 | |
| 
 | |
|     lastMes.extra['bookmark_link'] = name;
 | |
|     $(`.mes[mesid="${mesId}"]`).attr('bookmark_link', name);
 | |
| 
 | |
|     await saveChatConditional();
 | |
|     toastr.success('Click the flag icon in the last message to open the checkpoint chat.', 'Checkpoint created', { timeOut: 10000 });
 | |
| }
 | |
| 
 | |
| 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);
 | |
|         }
 | |
|     }
 | |
| }
 | |
| 
 | |
| 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('Convert to group chat', 'Are you sure you want to convert this chat to a group chat?<br />This cannot be reverted.');
 | |
|     if (!confirm) {
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     const character = characters[this_chid];
 | |
| 
 | |
|     // Populate group required fields
 | |
|     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 activationStrategy = group_activation_strategy.NATURAL;
 | |
|     const allowSelfResponses = false;
 | |
|     const favChecked = character.fav || character.fav == 'true';
 | |
|     const metadata = Object.assign({}, chat_metadata);
 | |
|     delete metadata.main_chat;
 | |
| 
 | |
|     const createGroupResponse = await fetch('/api/groups/create', {
 | |
|         method: 'POST',
 | |
|         headers: getRequestHeaders(),
 | |
|         body: JSON.stringify({
 | |
|             name: name,
 | |
|             members: members,
 | |
|             avatar_url: avatar,
 | |
|             allow_self_responses: activationStrategy,
 | |
|             activation_strategy: allowSelfResponses,
 | |
|             disabled_members: [],
 | |
|             chat_metadata: metadata,
 | |
|             fav: favChecked,
 | |
|             chat_id: chatName,
 | |
|             chats: chats,
 | |
|         }),
 | |
|     });
 | |
| 
 | |
|     if (!createGroupResponse.ok) {
 | |
|         console.error('Group creation unsuccessful');
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     const group = await createGroupResponse.json();
 | |
| 
 | |
|     // Convert tags list and assign to group
 | |
|     createTagMapFromList('#tagList', group.id);
 | |
| 
 | |
|     // Update chars list
 | |
|     await getCharacters();
 | |
| 
 | |
|     // Convert chat to group format
 | |
|     const groupChat = chat.slice();
 | |
|     const genIdFirst = Date.now();
 | |
| 
 | |
|     // Add something if the chat is empty
 | |
|     if (groupChat.length === 0) {
 | |
|         const newMessage = {
 | |
|             ...system_messages[system_message_types.GROUP],
 | |
|             send_date: getMessageTimeStamp(),
 | |
|             extra: { type: system_message_types.GROUP },
 | |
|         };
 | |
|         groupChat.push(newMessage);
 | |
|     }
 | |
| 
 | |
|     for (let index = 0; index < groupChat.length; index++) {
 | |
|         const message = groupChat[index];
 | |
| 
 | |
|         // Save group-chat marker
 | |
|         if (index == 0) {
 | |
|             message.is_group = true;
 | |
|         }
 | |
| 
 | |
|         // Skip messages we don't care about
 | |
|         if (message.is_user || message.is_system || message.extra?.type === system_message_types.NARRATOR || message.force_avatar !== undefined) {
 | |
|             continue;
 | |
|         }
 | |
| 
 | |
|         // Set force fields for solo character
 | |
|         message.name = character.name;
 | |
|         message.original_avatar = character.avatar;
 | |
|         message.force_avatar = getThumbnailUrl('avatar', character.avatar);
 | |
| 
 | |
|         // Allow regens of a single message in group
 | |
|         if (typeof message.extra !== 'object') {
 | |
|             message.extra = { gen_id: genIdFirst + index };
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     // Save group chat
 | |
|     const createChatResponse = await fetch('/api/chats/group/save', {
 | |
|         method: 'POST',
 | |
|         headers: getRequestHeaders(),
 | |
|         body: JSON.stringify({ id: chatName, chat: groupChat }),
 | |
|     });
 | |
| 
 | |
|     if (!createChatResponse.ok) {
 | |
|         console.error('Group chat creation unsuccessful');
 | |
|         toastr.error('Group chat creation unsuccessful');
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     // Click on the freshly selected group to open it
 | |
|     await openGroupById(group.id);
 | |
| 
 | |
|     toastr.success('The chat has been successfully converted!');
 | |
| }
 | |
| 
 | |
| jQuery(function () {
 | |
|     $('#option_new_bookmark').on('click', saveBookmarkMenu);
 | |
|     $('#option_back_to_main').on('click', backToMainChat);
 | |
|     $('#option_convert_to_group').on('click', convertSoloToGroupChat);
 | |
| });
 |