Add solo chat to group conversion

This commit is contained in:
SillyLossy
2023-05-01 15:45:44 +03:00
parent 8853e636ae
commit 6f08519858
4 changed files with 180 additions and 65 deletions

View File

@ -944,7 +944,7 @@ function messageFormating(mes, ch_name, isSystem, forceAvatar) {
if (this_chid != undefined && !isSystem)
mes = mes.replaceAll("<", "&lt;").replaceAll(">", "&gt;"); //for welcome message
if ((this_chid === undefined || this_chid == "invalid-safety-id") && !selected_group) {
if ((this_chid === undefined || this_chid === "invalid-safety-id") && !selected_group) {
mes = mes
.replace(/\*\*(.+?)\*\*/g, "<b>$1</b>")
.replace(/\n/g, "<br/>");
@ -1033,7 +1033,7 @@ function addOneMessage(mes, { type = "normal", insertAfter = null, scroll = true
if (!mes["is_user"]) {
if (mes.force_avatar) {
avatarImg = mes.force_avatar;
} else if (this_chid == undefined || this_chid == "invalid-safety-id") {
} else if (this_chid === undefined || this_chid === "invalid-safety-id") {
avatarImg = system_avatar;
} else {
if (characters[this_chid].avatar != "none") {
@ -1172,11 +1172,11 @@ function sendSystemMessage(type, text) {
newMessage.mes += getSlashCommandsHelp();
}
if (!newMessage.extras) {
newMessage.extras = {};
if (!newMessage.extra) {
newMessage.extra = {};
}
newMessage.extras.type = type;
newMessage.extra.type = type;
chat.push(newMessage);
addOneMessage(newMessage);
@ -2156,8 +2156,7 @@ async function Generate(type, automatic_trigger, force_name2) {
} //rungenerate ends
} else { //generate's primary loop ends, after this is error handling for no-connection or safety-id
if (this_chid == undefined || this_chid == 'invalid-safety-id') {
if (this_chid === undefined || this_chid === 'invalid-safety-id') {
//send ch sel
popup_type = 'char_not_selected';
callPopup('<h3>Сharacter is not selected</h3>');
@ -2535,7 +2534,7 @@ async function renameCharacter() {
// Async delay to update UI
await delay(1);
if (this_chid == -1) {
if (this_chid === -1) {
throw new Error('New character not selected');
}

View File

@ -9,10 +9,17 @@ import {
chat_metadata,
callPopup,
getRequestHeaders,
getThumbnailUrl,
getCharacters,
chat,
} from "../script.js";
import { selected_group } from "./group-chats.js";
import { humanizedDateTime } from "./RossAscends-mods.js";
import { group_activation_strategy, groups, selected_group } from "./group-chats.js";
import { createTagMapFromList } from "./tags.js";
import {
delay,
getUniqueName,
stringFormat,
} from "./utils.js";
@ -100,8 +107,7 @@ function showBookmarksButtons() {
}
}
$(document).ready(function () {
$('#option_new_bookmark').on('click', async function () {
async function createNewBookmark() {
if (selected_group) {
alert('Chat bookmarks unsupported for groups');
throw new Error();
@ -118,14 +124,130 @@ $(document).ready(function () {
let mainMessage = stringFormat(system_messages[system_message_types.BOOKMARK_CREATED].mes, name, name);
sendSystemMessage(system_message_types.BOOKMARK_CREATED, mainMessage);
saveChat();
});
}
$('#option_back_to_main').on('click', async function () {
async function backToMainChat() {
const mainChatName = getMainChatName(characters[this_chid].chat);
const allChats = await getExistingChatNames();
if (allChats.includes(mainChatName)) {
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 character = characters[this_chid];
// Populate group required fields
const name = getUniqueName(`Chat with ${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 == 'true';
const metadata = Object.assign({}, chat_metadata);
const createGroupResponse = await fetch("/creategroup", {
method: "POST",
headers: getRequestHeaders(),
body: JSON.stringify({
name: name,
members: members,
avatar_url: avatar,
allow_self_responses: activationStrategy,
activation_strategy: allowSelfResponses,
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: humanizedDateTime(),
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) {
continue;
}
// Set force fields for solo character
message.name = character.name;
message.original_avatar = character.avatar;
message.force_avatar = getThumbnailUrl('avatar', character.avatar);
message.is_name = true;
// 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("/savegroupchat", {
method: "POST",
headers: getRequestHeaders(),
body: JSON.stringify({ id: chatName, chat: groupChat }),
});
if (!createChatResponse.ok) {
console.error('Group chat creation unsuccessful');
return;
}
// Click on the freshly selected group to open it
$(`.group_select[grid="${group.id}"]`).click();
await delay(1);
callPopup('The chat has been successfully converted!', 'text');
}
$(document).ready(function () {
$('#option_new_bookmark').on('click', createNewBookmark);
$('#option_back_to_main').on('click', backToMainChat);
$('#option_convert_to_group').on('click', convertSoloToGroupChat);
});

View File

@ -71,7 +71,7 @@ let group_generation_id = null;
let fav_grp_checked = false;
let group_rm_panel_mode;
const group_activation_strategy = {
export const group_activation_strategy = {
NATURAL: 0,
LIST: 1,
};

View File

@ -1,24 +1,8 @@
export {
onlyUnique,
shuffle,
download,
urlContentToDataUri,
getBase64Async,
getStringHash,
debounce,
delay,
isSubsetOf,
incrementString,
stringFormat,
parseJsonFile,
};
/// UTILS
function onlyUnique(value, index, array) {
export function onlyUnique(value, index, array) {
return array.indexOf(value) === index;
}
function shuffle(array) {
export function shuffle(array) {
let currentIndex = array.length,
randomIndex;
@ -33,7 +17,7 @@ function shuffle(array) {
return array;
}
function download(content, fileName, contentType) {
export function download(content, fileName, contentType) {
const a = document.createElement("a");
const file = new Blob([content], { type: contentType });
a.href = URL.createObjectURL(file);
@ -41,7 +25,7 @@ function download(content, fileName, contentType) {
a.click();
}
async function urlContentToDataUri(url, params) {
export async function urlContentToDataUri(url, params) {
const response = await fetch(url, params);
const blob = await response.blob();
return await new Promise(callback => {
@ -51,7 +35,7 @@ async function urlContentToDataUri(url, params) {
});
}
function getBase64Async(file) {
export function getBase64Async(file) {
return new Promise((resolve, reject) => {
const reader = new FileReader();
reader.readAsDataURL(file);
@ -64,7 +48,7 @@ function getBase64Async(file) {
});
}
async function parseJsonFile(file) {
export async function parseJsonFile(file) {
return new Promise((resolve, reject) => {
const fileReader = new FileReader();
fileReader.onload = event => resolve(JSON.parse(event.target.result));
@ -73,7 +57,7 @@ async function parseJsonFile(file) {
});
}
function getStringHash(str, seed = 0) {
export function getStringHash(str, seed = 0) {
let h1 = 0xdeadbeef ^ seed,
h2 = 0x41c6ce57 ^ seed;
for (let i = 0, ch; i < str.length; i++) {
@ -88,7 +72,7 @@ function getStringHash(str, seed = 0) {
return 4294967296 * (2097151 & h2) + (h1 >>> 0);
};
function debounce(func, timeout = 300) {
export function debounce(func, timeout = 300) {
let timer;
return (...args) => {
clearTimeout(timer);
@ -96,10 +80,20 @@ function debounce(func, timeout = 300) {
};
}
const delay = (ms) => new Promise((res) => setTimeout(res, ms));
const isSubsetOf = (a, b) => (Array.isArray(a) && Array.isArray(b)) ? b.every(val => a.includes(val)) : false;
export function getUniqueName(name, exists) {
let i = 1;
let baseName = name;
while (exists(name)) {
name = `${baseName} (${i})`;
i++;
}
return name;
}
function incrementString(str) {
export const delay = (ms) => new Promise((res) => setTimeout(res, ms));
export const isSubsetOf = (a, b) => (Array.isArray(a) && Array.isArray(b)) ? b.every(val => a.includes(val)) : false;
export function incrementString(str) {
// Find the trailing number or it will match the empty string
const count = str.match(/\d*$/);
@ -108,7 +102,7 @@ function incrementString(str) {
return str.substr(0, count.index) + (++count[0]);
};
function stringFormat(format) {
export function stringFormat(format) {
const args = Array.prototype.slice.call(arguments, 1);
return format.replace(/{(\d+)}/g, function (match, number) {
return typeof args[number] != 'undefined'