mirror of
				https://github.com/SillyTavern/SillyTavern.git
				synced 2025-06-05 21:59:27 +02:00 
			
		
		
		
	* set pipe to empty string on empty closure * fix missing parser flags and scope * add closure serializing * add enum provider function to slash command arguments * add enum providers for /bg, /ask, and /go * fix index out of bounds returning undefined * keep whitespace as is in mixed unnamed args (string+closure) * add _hasUnnamedArgument to named arguments dictionary * allow /var key=x retrieval * add enum provider to /tag-add * fix typo (case) * add option to make enum matching optional * add executor to enum provider * change /tag-add enum provider to only show tags not already assigned * add enum provider to /tag-remove * fix name enum provider excluding groups * remove void from slash command callback return types * Lint undefined and null pipes * enable pointer events in chat autocomplete * fix type hint --------- Co-authored-by: LenAnderson <Anderson.Len@outlook.com> Co-authored-by: Cohee <18619528+Cohee1207@users.noreply.github.com>
		
			
				
	
	
		
			185 lines
		
	
	
		
			8.3 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			185 lines
		
	
	
		
			8.3 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
| import { AutoCompleteNameResult } from '../autocomplete/AutoCompleteNameResult.js';
 | |
| import { AutoCompleteOption } from '../autocomplete/AutoCompleteOption.js';
 | |
| import { AutoCompleteSecondaryNameResult } from '../autocomplete/AutoCompleteSecondaryNameResult.js';
 | |
| import { SlashCommand } from './SlashCommand.js';
 | |
| import { SlashCommandNamedArgument } from './SlashCommandArgument.js';
 | |
| import { SlashCommandClosure } from './SlashCommandClosure.js';
 | |
| import { SlashCommandCommandAutoCompleteOption } from './SlashCommandCommandAutoCompleteOption.js';
 | |
| import { SlashCommandEnumAutoCompleteOption } from './SlashCommandEnumAutoCompleteOption.js';
 | |
| import { SlashCommandExecutor } from './SlashCommandExecutor.js';
 | |
| import { SlashCommandNamedArgumentAutoCompleteOption } from './SlashCommandNamedArgumentAutoCompleteOption.js';
 | |
| 
 | |
| export class SlashCommandAutoCompleteNameResult extends AutoCompleteNameResult {
 | |
|     /**@type {SlashCommandExecutor}*/ executor;
 | |
| 
 | |
|     /**
 | |
|      * @param {SlashCommandExecutor} executor
 | |
|      * @param {Object.<string,SlashCommand>} commands
 | |
|      */
 | |
|     constructor(executor, commands) {
 | |
|         super(
 | |
|             executor.name,
 | |
|             executor.start,
 | |
|             Object
 | |
|                 .keys(commands)
 | |
|                 .map(key=>new SlashCommandCommandAutoCompleteOption(commands[key], key))
 | |
|             ,
 | |
|             false,
 | |
|             ()=>`No matching slash commands for "/${this.name}"`,
 | |
|             ()=>'No slash commands found!',
 | |
|         );
 | |
|         this.executor = executor;
 | |
|     }
 | |
| 
 | |
|     getSecondaryNameAt(text, index, isSelect) {
 | |
|         const namedResult = this.getNamedArgumentAt(text, index, isSelect);
 | |
|         if (!namedResult || namedResult.optionList.length == 0 || !namedResult.isRequired) {
 | |
|             const unnamedResult = this.getUnnamedArgumentAt(text, index, isSelect);
 | |
|             if (!namedResult) return unnamedResult;
 | |
|             if (namedResult && unnamedResult) {
 | |
|                 const combinedResult = new AutoCompleteSecondaryNameResult(
 | |
|                     namedResult.name,
 | |
|                     namedResult.start,
 | |
|                     [...namedResult.optionList, ...unnamedResult.optionList],
 | |
|                 );
 | |
|                 combinedResult.isRequired = namedResult.isRequired || unnamedResult.isRequired;
 | |
|                 combinedResult.forceMatch = namedResult.forceMatch && unnamedResult.forceMatch;
 | |
|                 return combinedResult;
 | |
|             }
 | |
|         }
 | |
|         return namedResult;
 | |
|     }
 | |
| 
 | |
|     getNamedArgumentAt(text, index, isSelect) {
 | |
|         function getSplitRegex() {
 | |
|             try {
 | |
|                 return new RegExp('(?<==)');
 | |
|             } catch {
 | |
|                 // For browsers that don't support lookbehind
 | |
|                 return new RegExp('=(.*)');
 | |
|             }
 | |
|         }
 | |
|         if (!Array.isArray(this.executor.command?.namedArgumentList)) {
 | |
|             return null;
 | |
|         }
 | |
|         const notProvidedNamedArguments = this.executor.command.namedArgumentList.filter(arg=>!this.executor.namedArgumentList.find(it=>it.name == arg.name));
 | |
|         let name;
 | |
|         let value;
 | |
|         let start;
 | |
|         let cmdArg;
 | |
|         let argAssign;
 | |
|         const unamedArgLength = this.executor.endUnnamedArgs - this.executor.startUnnamedArgs;
 | |
|         const namedArgsFollowedBySpace = text[this.executor.endNamedArgs] == ' ';
 | |
|         if (this.executor.startNamedArgs <= index && this.executor.endNamedArgs + (namedArgsFollowedBySpace ? 1 : 0) >= index) {
 | |
|             // cursor is somewhere within the named arguments (including final space)
 | |
|             argAssign = this.executor.namedArgumentList.find(it=>it.start <= index && it.end >= index);
 | |
|             if (argAssign) {
 | |
|                 const [argName, ...v] = text.slice(argAssign.start, index).split(getSplitRegex());
 | |
|                 name = argName;
 | |
|                 value = v.join('');
 | |
|                 start = argAssign.start;
 | |
|                 cmdArg = this.executor.command.namedArgumentList.find(it=>[it.name, `${it.name}=`].includes(argAssign.name));
 | |
|                 if (cmdArg) notProvidedNamedArguments.push(cmdArg);
 | |
|             } else {
 | |
|                 name = '';
 | |
|                 start = index;
 | |
|             }
 | |
|         } else if (unamedArgLength > 0 && index >= this.executor.startUnnamedArgs && index <= this.executor.endUnnamedArgs) {
 | |
|             // cursor is somewhere within the unnamed arguments
 | |
|             //TODO if index is in first array item and that is a string, treat it as an unfinished named arg
 | |
|             if (typeof this.executor.unnamedArgumentList[0].value == 'string') {
 | |
|                 if (index <= this.executor.startUnnamedArgs + this.executor.unnamedArgumentList[0].value.length) {
 | |
|                     name = this.executor.unnamedArgumentList[0].value.slice(0, index - this.executor.startUnnamedArgs);
 | |
|                     start = this.executor.startUnnamedArgs;
 | |
|                 } else {
 | |
|                     return null;
 | |
|                 }
 | |
|             } else {
 | |
|                 return null;
 | |
|             }
 | |
|         } else {
 | |
|             return null;
 | |
|         }
 | |
| 
 | |
|         if (name.includes('=') && cmdArg) {
 | |
|             // if cursor is already behind "=" check for enums
 | |
|             const enumList = cmdArg?.enumProvider?.(this.executor) ?? cmdArg?.enumList;
 | |
|             if (cmdArg && enumList?.length) {
 | |
|                 if (isSelect && enumList.find(it=>it.value == value) && argAssign && argAssign.end == index) {
 | |
|                     return null;
 | |
|                 }
 | |
|                 const result = new AutoCompleteSecondaryNameResult(
 | |
|                     value,
 | |
|                     start + name.length,
 | |
|                     enumList.map(it=>new SlashCommandEnumAutoCompleteOption(this.executor.command, it)),
 | |
|                     true,
 | |
|                 );
 | |
|                 result.isRequired = true;
 | |
|                 result.forceMatch = cmdArg.forceEnum;
 | |
|                 return result;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         if (notProvidedNamedArguments.length > 0) {
 | |
|             const result = new AutoCompleteSecondaryNameResult(
 | |
|                 name,
 | |
|                 start,
 | |
|                 notProvidedNamedArguments.map(it=>new SlashCommandNamedArgumentAutoCompleteOption(it, this.executor.command)),
 | |
|                 false,
 | |
|             );
 | |
|             result.isRequired = notProvidedNamedArguments.find(it=>it.isRequired) != null;
 | |
|             return result;
 | |
|         }
 | |
| 
 | |
|         return null;
 | |
|     }
 | |
| 
 | |
|     getUnnamedArgumentAt(text, index, isSelect) {
 | |
|         if (!Array.isArray(this.executor.command?.unnamedArgumentList)) {
 | |
|             return null;
 | |
|         }
 | |
|         const lastArgIsBlank = this.executor.unnamedArgumentList.slice(-1)[0]?.value == '';
 | |
|         const notProvidedArguments = this.executor.command.unnamedArgumentList.slice(this.executor.unnamedArgumentList.length - (lastArgIsBlank ? 1 : 0));
 | |
|         let value;
 | |
|         let start;
 | |
|         let cmdArg;
 | |
|         let argAssign;
 | |
|         if (this.executor.startUnnamedArgs <= index && this.executor.endUnnamedArgs + 1 >= index) {
 | |
|             // cursor is somwehere in the unnamed args
 | |
|             const idx = this.executor.unnamedArgumentList.findIndex(it=>it.start <= index && it.end >= index);
 | |
|             if (idx > -1) {
 | |
|                 argAssign = this.executor.unnamedArgumentList[idx];
 | |
|                 cmdArg = this.executor.command.unnamedArgumentList[idx];
 | |
|                 const enumList = cmdArg?.enumProvider?.(this.executor) ?? cmdArg?.enumList;
 | |
|                 if (cmdArg && enumList.length > 0) {
 | |
|                     value = argAssign.value.toString().slice(0, index - argAssign.start);
 | |
|                     start = argAssign.start;
 | |
|                 } else {
 | |
|                     return null;
 | |
|                 }
 | |
|             } else {
 | |
|                 value = '';
 | |
|                 start = index;
 | |
|                 cmdArg = notProvidedArguments[0];
 | |
|             }
 | |
|         } else {
 | |
|             return null;
 | |
|         }
 | |
| 
 | |
|         const enumList = cmdArg?.enumProvider?.(this.executor) ?? cmdArg?.enumList;
 | |
|         if (cmdArg == null || enumList.length == 0) return null;
 | |
| 
 | |
|         const result = new AutoCompleteSecondaryNameResult(
 | |
|             value,
 | |
|             start,
 | |
|             enumList.map(it=>new SlashCommandEnumAutoCompleteOption(this.executor.command, it)),
 | |
|             false,
 | |
|         );
 | |
|         const isCompleteValue = enumList.find(it=>it.value == value);
 | |
|         const isSelectedValue = isSelect && isCompleteValue;
 | |
|         result.isRequired = cmdArg.isRequired && !isSelectedValue && !isCompleteValue;
 | |
|         result.forceMatch = cmdArg.forceEnum;
 | |
|         return result;
 | |
|     }
 | |
| }
 |