mirror of
https://github.com/SillyTavern/SillyTavern.git
synced 2025-01-05 13:36:47 +01:00
509 lines
19 KiB
JavaScript
509 lines
19 KiB
JavaScript
// eslint-disable-next-line no-unused-vars
|
|
import { QuickReply } from '../src/QuickReply.js';
|
|
import { QuickReplyContextLink } from '../src/QuickReplyContextLink.js';
|
|
import { QuickReplySet } from '../src/QuickReplySet.js';
|
|
// eslint-disable-next-line no-unused-vars
|
|
import { QuickReplySettings } from '../src/QuickReplySettings.js';
|
|
// eslint-disable-next-line no-unused-vars
|
|
import { SettingsUi } from '../src/ui/SettingsUi.js';
|
|
import { onlyUnique } from '../../../utils.js';
|
|
|
|
export class QuickReplyApi {
|
|
/**@type {QuickReplySettings}*/ settings;
|
|
/**@type {SettingsUi}*/ settingsUi;
|
|
|
|
|
|
|
|
|
|
constructor(/**@type {QuickReplySettings}*/settings, /**@type {SettingsUi}*/settingsUi) {
|
|
this.settings = settings;
|
|
this.settingsUi = settingsUi;
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
* @param {QuickReply} qr
|
|
* @returns {QuickReplySet}
|
|
*/
|
|
getSetByQr(qr) {
|
|
return QuickReplySet.list.find(it=>it.qrList.includes(qr));
|
|
}
|
|
|
|
/**
|
|
* Finds and returns an existing Quick Reply Set by its name.
|
|
*
|
|
* @param {string} name name of the quick reply set
|
|
* @returns the quick reply set, or undefined if not found
|
|
*/
|
|
getSetByName(name) {
|
|
return QuickReplySet.get(name);
|
|
}
|
|
|
|
/**
|
|
* Finds and returns an existing Quick Reply by its set's name and its label.
|
|
*
|
|
* @param {string} setName name of the quick reply set
|
|
* @param {string|number} label label or numeric ID of the quick reply
|
|
* @returns the quick reply, or undefined if not found
|
|
*/
|
|
getQrByLabel(setName, label) {
|
|
const set = this.getSetByName(setName);
|
|
if (!set) return;
|
|
if (Number.isInteger(label)) return set.qrList.find(it=>it.id == label);
|
|
return set.qrList.find(it=>it.label == label);
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
* Executes a quick reply by its index and returns the result.
|
|
*
|
|
* @param {Number} idx the index (zero-based) of the quick reply to execute
|
|
* @returns the return value of the quick reply, or undefined if not found
|
|
*/
|
|
async executeQuickReplyByIndex(idx) {
|
|
const qr = [...this.settings.config.setList, ...(this.settings.chatConfig?.setList ?? [])]
|
|
.map(it=>it.set.qrList)
|
|
.flat()[idx]
|
|
;
|
|
if (qr) {
|
|
return await qr.onExecute();
|
|
} else {
|
|
throw new Error(`No quick reply at index "${idx}"`);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Executes an existing quick reply.
|
|
*
|
|
* @param {string} setName name of the existing quick reply set
|
|
* @param {string|number} label label of the existing quick reply (text on the button) or its numeric ID
|
|
* @param {object} [args] optional arguments
|
|
* @param {import('../../../slash-commands.js').ExecuteSlashCommandsOptions} [options] optional execution options
|
|
*/
|
|
async executeQuickReply(setName, label, args = {}, options = {}) {
|
|
const qr = this.getQrByLabel(setName, label);
|
|
if (!qr) {
|
|
throw new Error(`No quick reply with label "${label}" in set "${setName}" found.`);
|
|
}
|
|
return await qr.execute(args, false, false, options);
|
|
}
|
|
|
|
|
|
/**
|
|
* Adds or removes a quick reply set to the list of globally active quick reply sets.
|
|
*
|
|
* @param {string} name the name of the set
|
|
* @param {boolean} isVisible whether to show the set's buttons or not
|
|
*/
|
|
toggleGlobalSet(name, isVisible = true) {
|
|
const set = this.getSetByName(name);
|
|
if (!set) {
|
|
throw new Error(`No quick reply set with name "${name}" found.`);
|
|
}
|
|
if (this.settings.config.hasSet(set)) {
|
|
this.settings.config.removeSet(set);
|
|
} else {
|
|
this.settings.config.addSet(set, isVisible);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Adds a quick reply set to the list of globally active quick reply sets.
|
|
*
|
|
* @param {string} name the name of the set
|
|
* @param {boolean} isVisible whether to show the set's buttons or not
|
|
*/
|
|
addGlobalSet(name, isVisible = true) {
|
|
const set = this.getSetByName(name);
|
|
if (!set) {
|
|
throw new Error(`No quick reply set with name "${name}" found.`);
|
|
}
|
|
this.settings.config.addSet(set, isVisible);
|
|
}
|
|
|
|
/**
|
|
* Removes a quick reply set from the list of globally active quick reply sets.
|
|
*
|
|
* @param {string} name the name of the set
|
|
*/
|
|
removeGlobalSet(name) {
|
|
const set = this.getSetByName(name);
|
|
if (!set) {
|
|
throw new Error(`No quick reply set with name "${name}" found.`);
|
|
}
|
|
this.settings.config.removeSet(set);
|
|
}
|
|
|
|
|
|
/**
|
|
* Adds or removes a quick reply set to the list of the current chat's active quick reply sets.
|
|
*
|
|
* @param {string} name the name of the set
|
|
* @param {boolean} isVisible whether to show the set's buttons or not
|
|
*/
|
|
toggleChatSet(name, isVisible = true) {
|
|
if (!this.settings.chatConfig) return;
|
|
const set = this.getSetByName(name);
|
|
if (!set) {
|
|
throw new Error(`No quick reply set with name "${name}" found.`);
|
|
}
|
|
if (this.settings.chatConfig.hasSet(set)) {
|
|
this.settings.chatConfig.removeSet(set);
|
|
} else {
|
|
this.settings.chatConfig.addSet(set, isVisible);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Adds a quick reply set to the list of the current chat's active quick reply sets.
|
|
*
|
|
* @param {string} name the name of the set
|
|
* @param {boolean} isVisible whether to show the set's buttons or not
|
|
*/
|
|
addChatSet(name, isVisible = true) {
|
|
if (!this.settings.chatConfig) return;
|
|
const set = this.getSetByName(name);
|
|
if (!set) {
|
|
throw new Error(`No quick reply set with name "${name}" found.`);
|
|
}
|
|
this.settings.chatConfig.addSet(set, isVisible);
|
|
}
|
|
|
|
/**
|
|
* Removes a quick reply set from the list of the current chat's active quick reply sets.
|
|
*
|
|
* @param {string} name the name of the set
|
|
*/
|
|
removeChatSet(name) {
|
|
if (!this.settings.chatConfig) return;
|
|
const set = this.getSetByName(name);
|
|
if (!set) {
|
|
throw new Error(`No quick reply set with name "${name}" found.`);
|
|
}
|
|
this.settings.chatConfig.removeSet(set);
|
|
}
|
|
|
|
|
|
/**
|
|
* Creates a new quick reply in an existing quick reply set.
|
|
*
|
|
* @param {string} setName name of the quick reply set to insert the new quick reply into
|
|
* @param {string} label label for the new quick reply (text on the button)
|
|
* @param {object} [props]
|
|
* @param {string} [props.icon] the icon to show on the QR button
|
|
* @param {boolean} [props.showLabel] whether to show the label even when an icon is assigned
|
|
* @param {string} [props.message] the message to be sent or slash command to be executed by the new quick reply
|
|
* @param {string} [props.title] the title / tooltip to be shown on the quick reply button
|
|
* @param {boolean} [props.isHidden] whether to hide or show the button
|
|
* @param {boolean} [props.executeOnStartup] whether to execute the quick reply when SillyTavern starts
|
|
* @param {boolean} [props.executeOnUser] whether to execute the quick reply after a user has sent a message
|
|
* @param {boolean} [props.executeOnAi] whether to execute the quick reply after the AI has sent a message
|
|
* @param {boolean} [props.executeOnChatChange] whether to execute the quick reply when a new chat is loaded
|
|
* @param {boolean} [props.executeOnGroupMemberDraft] whether to execute the quick reply when a group member is selected
|
|
* @param {boolean} [props.executeOnNewChat] whether to execute the quick reply when a new chat is created
|
|
* @param {string} [props.automationId] when not empty, the quick reply will be executed when the WI with the given automation ID is activated
|
|
* @returns {QuickReply} the new quick reply
|
|
*/
|
|
createQuickReply(setName, label, {
|
|
icon,
|
|
showLabel,
|
|
message,
|
|
title,
|
|
isHidden,
|
|
executeOnStartup,
|
|
executeOnUser,
|
|
executeOnAi,
|
|
executeOnChatChange,
|
|
executeOnGroupMemberDraft,
|
|
executeOnNewChat,
|
|
automationId,
|
|
} = {}) {
|
|
const set = this.getSetByName(setName);
|
|
if (!set) {
|
|
throw new Error(`No quick reply set with named "${setName}" found.`);
|
|
}
|
|
const qr = set.addQuickReply();
|
|
qr.label = label ?? '';
|
|
qr.icon = icon ?? '';
|
|
qr.showLabel = showLabel ?? false;
|
|
qr.message = message ?? '';
|
|
qr.title = title ?? '';
|
|
qr.isHidden = isHidden ?? false;
|
|
qr.executeOnStartup = executeOnStartup ?? false;
|
|
qr.executeOnUser = executeOnUser ?? false;
|
|
qr.executeOnAi = executeOnAi ?? false;
|
|
qr.executeOnChatChange = executeOnChatChange ?? false;
|
|
qr.executeOnGroupMemberDraft = executeOnGroupMemberDraft ?? false;
|
|
qr.executeOnNewChat = executeOnNewChat ?? false;
|
|
qr.automationId = automationId ?? '';
|
|
qr.onUpdate();
|
|
return qr;
|
|
}
|
|
|
|
/**
|
|
* Updates an existing quick reply.
|
|
*
|
|
* @param {string} setName name of the existing quick reply set
|
|
* @param {string|number} label label of the existing quick reply (text on the button) or its numeric ID
|
|
* @param {object} [props]
|
|
* @param {string} [props.icon] the icon to show on the QR button
|
|
* @param {boolean} [props.showLabel] whether to show the label even when an icon is assigned
|
|
* @param {string} [props.newLabel] new label for quick reply (text on the button)
|
|
* @param {string} [props.message] the message to be sent or slash command to be executed by the quick reply
|
|
* @param {string} [props.title] the title / tooltip to be shown on the quick reply button
|
|
* @param {boolean} [props.isHidden] whether to hide or show the button
|
|
* @param {boolean} [props.executeOnStartup] whether to execute the quick reply when SillyTavern starts
|
|
* @param {boolean} [props.executeOnUser] whether to execute the quick reply after a user has sent a message
|
|
* @param {boolean} [props.executeOnAi] whether to execute the quick reply after the AI has sent a message
|
|
* @param {boolean} [props.executeOnChatChange] whether to execute the quick reply when a new chat is loaded
|
|
* @param {boolean} [props.executeOnGroupMemberDraft] whether to execute the quick reply when a group member is selected
|
|
* @param {boolean} [props.executeOnNewChat] whether to execute the quick reply when a new chat is created
|
|
* @param {string} [props.automationId] when not empty, the quick reply will be executed when the WI with the given automation ID is activated
|
|
* @returns {QuickReply} the altered quick reply
|
|
*/
|
|
updateQuickReply(setName, label, {
|
|
icon,
|
|
showLabel,
|
|
newLabel,
|
|
message,
|
|
title,
|
|
isHidden,
|
|
executeOnStartup,
|
|
executeOnUser,
|
|
executeOnAi,
|
|
executeOnChatChange,
|
|
executeOnGroupMemberDraft,
|
|
executeOnNewChat,
|
|
automationId,
|
|
} = {}) {
|
|
const qr = this.getQrByLabel(setName, label);
|
|
if (!qr) {
|
|
throw new Error(`No quick reply with label "${label}" in set "${setName}" found.`);
|
|
}
|
|
qr.updateIcon(icon ?? qr.icon);
|
|
qr.updateShowLabel(showLabel ?? qr.showLabel);
|
|
qr.updateLabel(newLabel ?? qr.label);
|
|
qr.updateMessage(message ?? qr.message);
|
|
qr.updateTitle(title ?? qr.title);
|
|
qr.isHidden = isHidden ?? qr.isHidden;
|
|
qr.executeOnStartup = executeOnStartup ?? qr.executeOnStartup;
|
|
qr.executeOnUser = executeOnUser ?? qr.executeOnUser;
|
|
qr.executeOnAi = executeOnAi ?? qr.executeOnAi;
|
|
qr.executeOnChatChange = executeOnChatChange ?? qr.executeOnChatChange;
|
|
qr.executeOnGroupMemberDraft = executeOnGroupMemberDraft ?? qr.executeOnGroupMemberDraft;
|
|
qr.executeOnNewChat = executeOnNewChat ?? qr.executeOnNewChat;
|
|
qr.automationId = automationId ?? qr.automationId;
|
|
qr.onUpdate();
|
|
return qr;
|
|
}
|
|
|
|
/**
|
|
* Deletes an existing quick reply.
|
|
*
|
|
* @param {string} setName name of the existing quick reply set
|
|
* @param {string|number} label label of the existing quick reply (text on the button) or its numeric ID
|
|
*/
|
|
deleteQuickReply(setName, label) {
|
|
const qr = this.getQrByLabel(setName, label);
|
|
if (!qr) {
|
|
throw new Error(`No quick reply with label "${label}" in set "${setName}" found.`);
|
|
}
|
|
qr.delete();
|
|
}
|
|
|
|
|
|
/**
|
|
* Adds an existing quick reply set as a context menu to an existing quick reply.
|
|
*
|
|
* @param {string} setName name of the existing quick reply set containing the quick reply
|
|
* @param {string|number} label label of the existing quick reply or its numeric ID
|
|
* @param {string} contextSetName name of the existing quick reply set to be used as a context menu
|
|
* @param {boolean} isChained whether or not to chain the context menu quick replies
|
|
*/
|
|
createContextItem(setName, label, contextSetName, isChained = false) {
|
|
const qr = this.getQrByLabel(setName, label);
|
|
const set = this.getSetByName(contextSetName);
|
|
if (!qr) {
|
|
throw new Error(`No quick reply with label "${label}" in set "${setName}" found.`);
|
|
}
|
|
if (!set) {
|
|
throw new Error(`No quick reply set with name "${contextSetName}" found.`);
|
|
}
|
|
const cl = new QuickReplyContextLink();
|
|
cl.set = set;
|
|
cl.isChained = isChained;
|
|
qr.addContextLink(cl);
|
|
}
|
|
|
|
/**
|
|
* Removes a quick reply set from a quick reply's context menu.
|
|
*
|
|
* @param {string} setName name of the existing quick reply set containing the quick reply
|
|
* @param {string|number} label label of the existing quick reply or its numeric ID
|
|
* @param {string} contextSetName name of the existing quick reply set to be used as a context menu
|
|
*/
|
|
deleteContextItem(setName, label, contextSetName) {
|
|
const qr = this.getQrByLabel(setName, label);
|
|
const set = this.getSetByName(contextSetName);
|
|
if (!qr) {
|
|
throw new Error(`No quick reply with label "${label}" in set "${setName}" found.`);
|
|
}
|
|
if (!set) {
|
|
throw new Error(`No quick reply set with name "${contextSetName}" found.`);
|
|
}
|
|
qr.removeContextLink(set.name);
|
|
}
|
|
|
|
/**
|
|
* Removes all entries from a quick reply's context menu.
|
|
*
|
|
* @param {string} setName name of the existing quick reply set containing the quick reply
|
|
* @param {string|number} label label of the existing quick reply or its numeric ID
|
|
*/
|
|
clearContextMenu(setName, label) {
|
|
const qr = this.getQrByLabel(setName, label);
|
|
if (!qr) {
|
|
throw new Error(`No quick reply with label "${label}" in set "${setName}" found.`);
|
|
}
|
|
qr.clearContextLinks();
|
|
}
|
|
|
|
|
|
/**
|
|
* Create a new quick reply set.
|
|
*
|
|
* @param {string} name name of the new quick reply set
|
|
* @param {object} [props]
|
|
* @param {boolean} [props.disableSend] whether or not to send the quick replies or put the message or slash command into the char input box
|
|
* @param {boolean} [props.placeBeforeInput] whether or not to place the quick reply contents before the existing user input
|
|
* @param {boolean} [props.injectInput] whether or not to automatically inject the user input at the end of the quick reply
|
|
* @returns {Promise<QuickReplySet>} the new quick reply set
|
|
*/
|
|
async createSet(name, {
|
|
disableSend,
|
|
placeBeforeInput,
|
|
injectInput,
|
|
} = {}) {
|
|
const set = new QuickReplySet();
|
|
set.name = name;
|
|
set.disableSend = disableSend ?? false;
|
|
set.placeBeforeInput = placeBeforeInput ?? false;
|
|
set.injectInput = injectInput ?? false;
|
|
const oldSet = this.getSetByName(name);
|
|
if (oldSet) {
|
|
QuickReplySet.list.splice(QuickReplySet.list.indexOf(oldSet), 1, set);
|
|
} else {
|
|
const idx = QuickReplySet.list.findIndex(it=>it.name.localeCompare(name) == 1);
|
|
if (idx > -1) {
|
|
QuickReplySet.list.splice(idx, 0, set);
|
|
} else {
|
|
QuickReplySet.list.push(set);
|
|
}
|
|
}
|
|
await set.save();
|
|
this.settingsUi.rerender();
|
|
return set;
|
|
}
|
|
|
|
/**
|
|
* Update an existing quick reply set.
|
|
*
|
|
* @param {string} name name of the existing quick reply set
|
|
* @param {object} [props]
|
|
* @param {boolean} [props.disableSend] whether or not to send the quick replies or put the message or slash command into the char input box
|
|
* @param {boolean} [props.placeBeforeInput] whether or not to place the quick reply contents before the existing user input
|
|
* @param {boolean} [props.injectInput] whether or not to automatically inject the user input at the end of the quick reply
|
|
* @returns {Promise<QuickReplySet>} the altered quick reply set
|
|
*/
|
|
async updateSet(name, {
|
|
disableSend,
|
|
placeBeforeInput,
|
|
injectInput,
|
|
} = {}) {
|
|
const set = this.getSetByName(name);
|
|
if (!set) {
|
|
throw new Error(`No quick reply set with name "${name}" found.`);
|
|
}
|
|
set.disableSend = disableSend ?? false;
|
|
set.placeBeforeInput = placeBeforeInput ?? false;
|
|
set.injectInput = injectInput ?? false;
|
|
await set.save();
|
|
this.settingsUi.rerender();
|
|
return set;
|
|
}
|
|
|
|
/**
|
|
* Delete an existing quick reply set.
|
|
*
|
|
* @param {string} name name of the existing quick reply set
|
|
*/
|
|
async deleteSet(name) {
|
|
const set = this.getSetByName(name);
|
|
if (!set) {
|
|
throw new Error(`No quick reply set with name "${name}" found.`);
|
|
}
|
|
await set.delete();
|
|
this.settingsUi.rerender();
|
|
}
|
|
|
|
|
|
/**
|
|
* Gets a list of all quick reply sets.
|
|
*
|
|
* @returns array with the names of all quick reply sets
|
|
*/
|
|
listSets() {
|
|
return QuickReplySet.list.map(it=>it.name);
|
|
}
|
|
/**
|
|
* Gets a list of all globally active quick reply sets.
|
|
*
|
|
* @returns array with the names of all quick reply sets
|
|
*/
|
|
listGlobalSets() {
|
|
return this.settings.config.setList.map(it=>it.set.name);
|
|
}
|
|
/**
|
|
* Gets a list of all quick reply sets activated by the current chat.
|
|
*
|
|
* @returns array with the names of all quick reply sets
|
|
*/
|
|
listChatSets() {
|
|
return this.settings.chatConfig?.setList?.flatMap(it=>it.set.name) ?? [];
|
|
}
|
|
|
|
/**
|
|
* Gets a list of all quick replies in the quick reply set.
|
|
*
|
|
* @param {string} setName name of the existing quick reply set
|
|
* @returns array with the labels of this set's quick replies
|
|
*/
|
|
listQuickReplies(setName) {
|
|
const set = this.getSetByName(setName);
|
|
if (!set) {
|
|
throw new Error(`No quick reply set with name "${name}" found.`);
|
|
}
|
|
return set.qrList.map(it=>it.label);
|
|
}
|
|
|
|
/**
|
|
* Gets a list of all Automation IDs used by quick replies.
|
|
*
|
|
* @returns {String[]} array with all automation IDs used by quick replies
|
|
*/
|
|
listAutomationIds() {
|
|
return this
|
|
.listSets()
|
|
.flatMap(it => ({ set: it, qrs: this.listQuickReplies(it) }))
|
|
.map(it => it.qrs?.map(qr => this.getQrByLabel(it.set, qr)?.automationId))
|
|
.flat()
|
|
.filter(Boolean)
|
|
.filter(onlyUnique)
|
|
.map(String);
|
|
}
|
|
}
|