mirror of
https://github.com/SillyTavern/SillyTavern.git
synced 2025-06-05 21:59:27 +02:00
Unleash the variables / STscript alpha
This commit is contained in:
@ -194,6 +194,7 @@ import { hideLoader, showLoader } from "./scripts/loader.js";
|
||||
import { CharacterContextMenu, BulkEditOverlay } from "./scripts/BulkEditOverlay.js";
|
||||
import { loadMancerModels } from "./scripts/mancer-settings.js";
|
||||
import { hasPendingFileAttachment, populateFileAttachment } from "./scripts/chats.js";
|
||||
import { replaceVariableMacros } from "./scripts/variables.js";
|
||||
|
||||
//exporting functions and vars for mods
|
||||
export {
|
||||
@ -2008,6 +2009,7 @@ function substituteParams(content, _name1, _name2, _original, _group, _replaceCh
|
||||
content = content.replace(/{{original}}/i, _original);
|
||||
}
|
||||
|
||||
content = replaceVariableMacros(content);
|
||||
content = content.replace(/{{input}}/gi, String($('#send_textarea').val()));
|
||||
|
||||
if (_replaceCharacterCard) {
|
||||
|
@ -33,6 +33,7 @@ import { autoSelectPersona } from "./personas.js";
|
||||
import { getContext } from "./extensions.js";
|
||||
import { hideChatMessage, unhideChatMessage } from "./chats.js";
|
||||
import { stringToRange } from "./utils.js";
|
||||
import { registerVariableCommands } from "./variables.js";
|
||||
export {
|
||||
executeSlashCommands,
|
||||
registerSlashCommand,
|
||||
@ -160,6 +161,7 @@ parser.addCommand('delswipe', deleteSwipeCallback, ['swipedel'], '<span class="m
|
||||
parser.addCommand('echo', echoCallback, [], '<span class="monospace">(text)</span> – echoes the text to toast message. Useful for pipes debugging.', true, true);
|
||||
parser.addCommand('gen', generateCallback, [], '<span class="monospace">(prompt)</span> – generates text using the provided prompt and passes it to the next command through the pipe.', true, true);
|
||||
parser.addCommand('addswipe', addSwipeCallback, ['swipeadd'], '<span class="monospace">(text)</span> – adds a swipe to the last chat message.', true, true);
|
||||
registerVariableCommands();
|
||||
|
||||
const NARRATOR_NAME_KEY = 'narrator_name';
|
||||
const NARRATOR_NAME_DEFAULT = 'System';
|
||||
@ -179,12 +181,12 @@ async function generateCallback(_, arg) {
|
||||
}
|
||||
|
||||
async function echoCallback(_, arg) {
|
||||
if (!arg) {
|
||||
if (!String(arg)) {
|
||||
console.warn('WARN: No argument provided for /echo command');
|
||||
return;
|
||||
}
|
||||
|
||||
toastr.info(arg);
|
||||
toastr.info(String(arg));
|
||||
return arg;
|
||||
}
|
||||
|
||||
@ -961,6 +963,11 @@ function setBackgroundCallback(_, bg) {
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Executes slash commands in the provided text
|
||||
* @param {string} text Slash command text
|
||||
* @returns {Promise<{interrupt: boolean, newText: string, pipe: string} | boolean>}
|
||||
*/
|
||||
async function executeSlashCommands(text) {
|
||||
if (!text) {
|
||||
return false;
|
||||
@ -1006,7 +1013,7 @@ async function executeSlashCommands(text) {
|
||||
|
||||
const newText = lines.filter(x => linesToRemove.indexOf(x) === -1).join('\n');
|
||||
|
||||
return { interrupt, newText };
|
||||
return { interrupt, newText, pipe: pipeResult };
|
||||
}
|
||||
|
||||
function setSlashCommandAutocomplete(textarea) {
|
||||
|
@ -1,4 +1,6 @@
|
||||
System-wide Replacement Macros (in order of evaluation):
|
||||
<div>
|
||||
System-wide Replacement Macros (in order of evaluation):
|
||||
</div>
|
||||
<ul>
|
||||
<li><tt>{{original}}</tt> – global prompts defined in API settings. Only valid in Advanced Definitions prompt overrides.</li>
|
||||
<li><tt>{{input}}</tt> – the user input</li>
|
||||
@ -27,3 +29,16 @@ System-wide Replacement Macros (in order of evaluation):
|
||||
<li><tt>{{roll:(formula)}}</tt> – rolls a dice. (ex: {{roll:1d6}} will roll a 6- sided dice and return a number between 1 and 6)</li>
|
||||
<li><tt>{{banned "text here"}}</tt> – dynamically add text in the quotes to banned words sequences, if Text Generation WebUI backend used. Do nothing for others backends. Can be used anywhere (Character description, WI, AN, etc.) Quotes around the text are important.</li>
|
||||
</ul>
|
||||
<div>
|
||||
Chat variables Macros:
|
||||
</div>
|
||||
<div><small>Local variables = unique to the current chat</small></div>
|
||||
<div><small>Global variables = works in any chat for any character</small></div>
|
||||
<ul>
|
||||
<li><tt>{{getvar::name}}</tt> – replaced with the value of the local variable "name"</li>
|
||||
<li><tt>{{setvar::name::value}}</tt> – replaced with empty string, sets the local variable "name" to "value"</li>
|
||||
<li><tt>{{addvar::name::increment}}</tt> – replaced with the result of addition numeric value of "increment" to the local variable "name"</li>
|
||||
<li><tt>{{getglobalvar::name}}</tt> – replaced with the value of the global variable "name"</li>
|
||||
<li><tt>{{setglobalvar::name::value}}</tt> – replaced with empty string, sets the global variable "name" to "value"</li>
|
||||
<li><tt>{{addglobalvar::name::value}}</tt> – replaced with the result of addition numeric value of "increment" to the global variable "name"</li>
|
||||
</ul>
|
||||
|
@ -1,6 +1,6 @@
|
||||
import { chat_metadata, getCurrentChatId, sendSystemMessage, system_message_types } from "../script.js";
|
||||
import { extension_settings } from "./extensions.js";
|
||||
import { registerSlashCommand } from "./slash-commands.js";
|
||||
import { executeSlashCommands, registerSlashCommand } from "./slash-commands.js";
|
||||
|
||||
function getLocalVariable(name) {
|
||||
const localVariable = chat_metadata?.variables[name];
|
||||
@ -14,6 +14,7 @@ function setLocalVariable(name, value) {
|
||||
}
|
||||
|
||||
chat_metadata.variables[name] = value;
|
||||
return value;
|
||||
}
|
||||
|
||||
function getGlobalVariable(name) {
|
||||
@ -26,6 +27,42 @@ function setGlobalVariable(name, value) {
|
||||
extension_settings.variables.global[name] = value;
|
||||
}
|
||||
|
||||
function addLocalVariable(name, value) {
|
||||
const currentValue = getLocalVariable(name) || 0;
|
||||
const increment = Number(value);
|
||||
|
||||
if (isNaN(increment)) {
|
||||
return '';
|
||||
}
|
||||
|
||||
const newValue = Number(currentValue) + increment;
|
||||
|
||||
if (isNaN(newValue)) {
|
||||
return '';
|
||||
}
|
||||
|
||||
setLocalVariable(name, newValue);
|
||||
return newValue;
|
||||
}
|
||||
|
||||
function addGlobalVariable(name, value) {
|
||||
const currentValue = getGlobalVariable(name) || 0;
|
||||
const increment = Number(value);
|
||||
|
||||
if (isNaN(increment)) {
|
||||
return '';
|
||||
}
|
||||
|
||||
const newValue = Number(currentValue) + increment;
|
||||
|
||||
if (isNaN(newValue)) {
|
||||
return '';
|
||||
}
|
||||
|
||||
setGlobalVariable(name, newValue);
|
||||
return newValue;
|
||||
}
|
||||
|
||||
export function replaceVariableMacros(str) {
|
||||
// Replace {{getvar::name}} with the value of the variable name
|
||||
str = str.replace(/{{getvar::([^}]+)}}/gi, (_, name) => {
|
||||
@ -43,21 +80,7 @@ export function replaceVariableMacros(str) {
|
||||
// Replace {{addvar::name::value}} with empty string and add value to the variable value
|
||||
str = str.replace(/{{addvar::([^:]+)::([^}]+)}}/gi, (_, name, value) => {
|
||||
name = name.toLowerCase().trim();
|
||||
const currentValue = getLocalVariable(name) || 0;
|
||||
const increment = Number(value);
|
||||
|
||||
if (isNaN(increment)) {
|
||||
return '';
|
||||
}
|
||||
|
||||
const newValue = Number(currentValue) + increment;
|
||||
|
||||
if (isNaN(newValue)) {
|
||||
return '';
|
||||
}
|
||||
|
||||
setLocalVariable(name, newValue);
|
||||
return '';
|
||||
return addLocalVariable(name, value);;
|
||||
});
|
||||
|
||||
// Replace {{getglobalvar::name}} with the value of the global variable name
|
||||
@ -76,21 +99,7 @@ export function replaceVariableMacros(str) {
|
||||
// Replace {{addglobalvar::name::value}} with empty string and add value to the global variable value
|
||||
str = str.replace(/{{addglobalvar::([^:]+)::([^}]+)}}/gi, (_, name, value) => {
|
||||
name = name.toLowerCase().trim();
|
||||
const currentValue = getGlobalVariable(name) || 0;
|
||||
const increment = Number(value);
|
||||
|
||||
if (isNaN(increment)) {
|
||||
return '';
|
||||
}
|
||||
|
||||
const newValue = Number(currentValue) + increment;
|
||||
|
||||
if (isNaN(newValue)) {
|
||||
return '';
|
||||
}
|
||||
|
||||
setGlobalVariable(name, newValue);
|
||||
return '';
|
||||
return addGlobalVariable(name, value);
|
||||
});
|
||||
|
||||
return str;
|
||||
@ -110,11 +119,80 @@ function listVariablesCallback() {
|
||||
|
||||
const converter = new showdown.Converter();
|
||||
const message = `### Local variables (${chatName}):\n${localVariablesString}\n\n### Global variables:\n${globalVariablesString}`;
|
||||
const htmlMessage = converter.makeHtml(message);
|
||||
const htmlMessage = DOMPurify.sanitize(converter.makeHtml(message));
|
||||
|
||||
sendSystemMessage(system_message_types.GENERIC, htmlMessage);
|
||||
}
|
||||
|
||||
async function ifCallback(args, command) {
|
||||
const a = getLocalVariable(args.a) || getGlobalVariable(args.a) || Number(args.a);
|
||||
const b = getLocalVariable(args.b) || getGlobalVariable(args.b) || Number(args.b);
|
||||
const rule = args.rule;
|
||||
|
||||
if (!a || !b || !rule) {
|
||||
toastr.warning('Both operands and the rule must be specified for the /if command.', 'Invalid /if command');
|
||||
return '';
|
||||
}
|
||||
|
||||
const aNumber = Number(a);
|
||||
const bNumber = Number(b);
|
||||
|
||||
if (isNaN(aNumber) || isNaN(bNumber)) {
|
||||
toastr.warning('Both operands must be numbers for the /if command.', 'Invalid /if command');
|
||||
return '';
|
||||
}
|
||||
|
||||
let result = false;
|
||||
|
||||
switch (rule) {
|
||||
case 'gt':
|
||||
result = aNumber > bNumber;
|
||||
break;
|
||||
case 'gte':
|
||||
result = aNumber >= bNumber;
|
||||
break;
|
||||
case 'lt':
|
||||
result = aNumber < bNumber;
|
||||
break;
|
||||
case 'lte':
|
||||
result = aNumber <= bNumber;
|
||||
break;
|
||||
case 'eq':
|
||||
result = aNumber === bNumber;
|
||||
break;
|
||||
default:
|
||||
toastr.warning('Unknown rule for the /if command.', 'Invalid /if command');
|
||||
return '';
|
||||
}
|
||||
|
||||
if (result && command) {
|
||||
if (command.startsWith('"')) {
|
||||
command = command.slice(1);
|
||||
}
|
||||
|
||||
if (command.endsWith('"')) {
|
||||
command = command.slice(0, -1);
|
||||
}
|
||||
|
||||
const result = await executeSlashCommands(command);
|
||||
|
||||
if (!result || typeof result !== 'object') {
|
||||
return '';
|
||||
}
|
||||
|
||||
return result?.pipe || '';
|
||||
}
|
||||
|
||||
return '';
|
||||
}
|
||||
|
||||
export function registerVariableCommands() {
|
||||
registerSlashCommand('listvar', listVariablesCallback, [''], ' – list registered chat variables', true, true);
|
||||
registerSlashCommand('setvar', (args, value) => setLocalVariable(args.key || args.name, value), [], '<span class="monospace">key=varname (value)</span> – set a local variable value and pass it down the pipe, e.g. <tt>/setvar key=color green</tt>', true, true);
|
||||
registerSlashCommand('getvar', (_, value) => getLocalVariable(value), [], '<span class="monospace">(key)</span> – get a local variable value and pass it down the pipe, e.g. <tt>/getvar height</tt>', true, true);
|
||||
registerSlashCommand('addvar', (args, value) => addLocalVariable(args.key || args.name, value), [], '<span class="monospace">key=varname (increment)</span> – add a value to a local variable and pass the result down the pipe, e.g. <tt>/addvar score 10</tt>', true, true);
|
||||
registerSlashCommand('setglobalvar', (args, value) => setGlobalVariable(args.key || args.name, value), [], '<span class="monospace">key=varname (value)</span> – set a global variable value and pass it down the pipe, e.g. <tt>/setglobalvar key=color green</tt>', true, true);
|
||||
registerSlashCommand('getglobalvar', (_, value) => getGlobalVariable(value), [], '<span class="monospace">(key)</span> – get a global variable value and pass it down the pipe, e.g. <tt>/getglobalvar height</tt>', true, true);
|
||||
registerSlashCommand('addglobalvar', (args, value) => addGlobalVariable(args.key || args.name, value), [], '<span class="monospace">key=varname (increment)</span> – add a value to a global variable and pass the result down the pipe, e.g. <tt>/addglobalvar score 10</tt>', true, true);
|
||||
registerSlashCommand('if', ifCallback, [], '<span class="monospace">a=varname1 b=varname2 rule=comparison "(command)"</span> – compare the value of variable "a" with the value of variable "b", and if the condition yields true, then execute any valid slash command enclosed in quotes and pass the result of the command execution down the pipe. Numeric values for "a" and "b" supported. Available rules: gt => a > b, gte => a >= b, lt => a < b, lte => a <= b, eq => a == b e.g. <tt>/if a=score a=10 rule=gte "/speak You win"</tt> triggers a /speak command if the value of "score" is greater or equals 10.', true, true);
|
||||
}
|
||||
|
Reference in New Issue
Block a user