mirror of
				https://github.com/SillyTavern/SillyTavern.git
				synced 2025-06-05 21:59:27 +02:00 
			
		
		
		
	Merge pull request #3725 from SillyTavern/feat/expressions-filter-available
Adds filtering to expressions to ignore labels that do not have sprites available
This commit is contained in:
		| @@ -4,7 +4,7 @@ import { characters, eventSource, event_types, generateRaw, getRequestHeaders, m | |||||||
| import { dragElement, isMobile } from '../../RossAscends-mods.js'; | import { dragElement, isMobile } from '../../RossAscends-mods.js'; | ||||||
| import { getContext, getApiUrl, modules, extension_settings, ModuleWorkerWrapper, doExtrasFetch, renderExtensionTemplateAsync } from '../../extensions.js'; | import { getContext, getApiUrl, modules, extension_settings, ModuleWorkerWrapper, doExtrasFetch, renderExtensionTemplateAsync } from '../../extensions.js'; | ||||||
| import { loadMovingUIState, performFuzzySearch, power_user } from '../../power-user.js'; | import { loadMovingUIState, performFuzzySearch, power_user } from '../../power-user.js'; | ||||||
| import { onlyUnique, debounce, getCharaFilename, trimToEndSentence, trimToStartSentence, waitUntilCondition, findChar } from '../../utils.js'; | import { onlyUnique, debounce, getCharaFilename, trimToEndSentence, trimToStartSentence, waitUntilCondition, findChar, isFalseBoolean } from '../../utils.js'; | ||||||
| import { hideMutedSprites, selected_group } from '../../group-chats.js'; | import { hideMutedSprites, selected_group } from '../../group-chats.js'; | ||||||
| import { isJsonSchemaSupported } from '../../textgen-settings.js'; | import { isJsonSchemaSupported } from '../../textgen-settings.js'; | ||||||
| import { debounce_timeout } from '../../constants.js'; | import { debounce_timeout } from '../../constants.js'; | ||||||
| @@ -679,7 +679,7 @@ async function setSpriteFolderCommand(_, folder) { | |||||||
|     return ''; |     return ''; | ||||||
| } | } | ||||||
|  |  | ||||||
| async function classifyCallback(/** @type {{api: string?, prompt: string?}} */ { api = null, prompt = null }, text) { | async function classifyCallback(/** @type {{api: string?, filter: string?, prompt: string?}} */ { api = null, filter = null, prompt = null }, text) { | ||||||
|     if (!text) { |     if (!text) { | ||||||
|         toastr.error('No text provided'); |         toastr.error('No text provided'); | ||||||
|         return ''; |         return ''; | ||||||
| @@ -690,13 +690,14 @@ async function classifyCallback(/** @type {{api: string?, prompt: string?}} */ { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     const expressionApi = EXPRESSION_API[api] || extension_settings.expressions.api; |     const expressionApi = EXPRESSION_API[api] || extension_settings.expressions.api; | ||||||
|  |     const filterAvailable = !isFalseBoolean(filter); | ||||||
|  |  | ||||||
|     if (!modules.includes('classify') && expressionApi == EXPRESSION_API.extras) { |     if (!modules.includes('classify') && expressionApi == EXPRESSION_API.extras) { | ||||||
|         toastr.warning('Text classification is disabled or not available'); |         toastr.warning('Text classification is disabled or not available'); | ||||||
|         return ''; |         return ''; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     const label = await getExpressionLabel(text, expressionApi, { customPrompt: prompt }); |     const label = await getExpressionLabel(text, expressionApi, { filterAvailable: filterAvailable, customPrompt: prompt }); | ||||||
|     console.debug(`Classification result for "${text}": ${label}`); |     console.debug(`Classification result for "${text}": ${label}`); | ||||||
|     return label; |     return label; | ||||||
| } | } | ||||||
| @@ -992,10 +993,11 @@ function onTextGenSettingsReady(args) { | |||||||
|  * @param {string} text - The text to classify and retrieve the expression label for. |  * @param {string} text - The text to classify and retrieve the expression label for. | ||||||
|  * @param {EXPRESSION_API} [expressionsApi=extension_settings.expressions.api] - The expressions API to use for classification. |  * @param {EXPRESSION_API} [expressionsApi=extension_settings.expressions.api] - The expressions API to use for classification. | ||||||
|  * @param {object} [options={}] - Optional arguments. |  * @param {object} [options={}] - Optional arguments. | ||||||
|  |  * @param {boolean?} [options.filterAvailable=null] - Whether to filter available expressions. If not specified, uses the extension setting. | ||||||
|  * @param {string?} [options.customPrompt=null] - The custom prompt to use for classification. |  * @param {string?} [options.customPrompt=null] - The custom prompt to use for classification. | ||||||
|  * @returns {Promise<string?>} - The label of the expression. |  * @returns {Promise<string?>} - The label of the expression. | ||||||
|  */ |  */ | ||||||
| export async function getExpressionLabel(text, expressionsApi = extension_settings.expressions.api, { customPrompt = null } = {}) { | export async function getExpressionLabel(text, expressionsApi = extension_settings.expressions.api, { filterAvailable = null, customPrompt = null } = {}) { | ||||||
|     // Return if text is undefined, saving a costly fetch request |     // Return if text is undefined, saving a costly fetch request | ||||||
|     if ((!modules.includes('classify') && expressionsApi == EXPRESSION_API.extras) || !text) { |     if ((!modules.includes('classify') && expressionsApi == EXPRESSION_API.extras) || !text) { | ||||||
|         return extension_settings.expressions.fallback_expression; |         return extension_settings.expressions.fallback_expression; | ||||||
| @@ -1007,6 +1009,11 @@ export async function getExpressionLabel(text, expressionsApi = extension_settin | |||||||
|  |  | ||||||
|     text = sampleClassifyText(text); |     text = sampleClassifyText(text); | ||||||
|  |  | ||||||
|  |     filterAvailable ??= extension_settings.expressions.filterAvailable; | ||||||
|  |     if (filterAvailable && ![EXPRESSION_API.llm, EXPRESSION_API.webllm].includes(expressionsApi)) { | ||||||
|  |         console.debug('Filter available is only supported for LLM and WebLLM expressions'); | ||||||
|  |     } | ||||||
|  |  | ||||||
|     try { |     try { | ||||||
|         switch (expressionsApi) { |         switch (expressionsApi) { | ||||||
|             // Local BERT pipeline |             // Local BERT pipeline | ||||||
| @@ -1031,7 +1038,7 @@ export async function getExpressionLabel(text, expressionsApi = extension_settin | |||||||
|                     return extension_settings.expressions.fallback_expression; |                     return extension_settings.expressions.fallback_expression; | ||||||
|                 } |                 } | ||||||
|  |  | ||||||
|                 const expressionsList = await getExpressionsList(); |                 const expressionsList = await getExpressionsList({ filterAvailable: filterAvailable }); | ||||||
|                 const prompt = substituteParamsExtended(customPrompt, { labels: expressionsList }) || await getLlmPrompt(expressionsList); |                 const prompt = substituteParamsExtended(customPrompt, { labels: expressionsList }) || await getLlmPrompt(expressionsList); | ||||||
|                 eventSource.once(event_types.TEXT_COMPLETION_SETTINGS_READY, onTextGenSettingsReady); |                 eventSource.once(event_types.TEXT_COMPLETION_SETTINGS_READY, onTextGenSettingsReady); | ||||||
|                 const emotionResponse = await generateRaw(text, main_api, false, false, prompt); |                 const emotionResponse = await generateRaw(text, main_api, false, false, prompt); | ||||||
| @@ -1044,7 +1051,7 @@ export async function getExpressionLabel(text, expressionsApi = extension_settin | |||||||
|                     return extension_settings.expressions.fallback_expression; |                     return extension_settings.expressions.fallback_expression; | ||||||
|                 } |                 } | ||||||
|  |  | ||||||
|                 const expressionsList = await getExpressionsList(); |                 const expressionsList = await getExpressionsList({ filterAvailable: filterAvailable }); | ||||||
|                 const prompt = substituteParamsExtended(customPrompt, { labels: expressionsList }) || await getLlmPrompt(expressionsList); |                 const prompt = substituteParamsExtended(customPrompt, { labels: expressionsList }) || await getLlmPrompt(expressionsList); | ||||||
|                 const messages = [ |                 const messages = [ | ||||||
|                     { role: 'user', content: text + '\n\n' + prompt }, |                     { role: 'user', content: text + '\n\n' + prompt }, | ||||||
| @@ -1324,12 +1331,28 @@ function getCachedExpressions() { | |||||||
|     return [...expressionsList, ...extension_settings.expressions.custom].filter(onlyUnique); |     return [...expressionsList, ...extension_settings.expressions.custom].filter(onlyUnique); | ||||||
| } | } | ||||||
|  |  | ||||||
| export async function getExpressionsList() { | export async function getExpressionsList({ filterAvailable = false } = {}) { | ||||||
|     // Return cached list if available |     // If there is no cached list, load and cache it | ||||||
|     if (Array.isArray(expressionsList)) { |     if (!Array.isArray(expressionsList)) { | ||||||
|         return getCachedExpressions(); |         expressionsList = await resolveExpressionsList(); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     const expressions = getCachedExpressions(); | ||||||
|  |  | ||||||
|  |     // Filtering is only available for llm and webllm APIs | ||||||
|  |     if (!filterAvailable || ![EXPRESSION_API.llm, EXPRESSION_API.webllm].includes(extension_settings.expressions.api)) { | ||||||
|  |         return expressions; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     // Get expressions with available sprites | ||||||
|  |     const currentLastMessage = selected_group ? getLastCharacterMessage() : null; | ||||||
|  |     const spriteFolderName = getSpriteFolderName(currentLastMessage, currentLastMessage?.name); | ||||||
|  |  | ||||||
|  |     return expressions.filter(label => { | ||||||
|  |         const expression = spriteCache[spriteFolderName]?.find(x => x.label === label); | ||||||
|  |         return (expression?.files.length ?? 0) > 0; | ||||||
|  |     }); | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * Returns the list of expressions from the API or fallback in offline mode. |      * Returns the list of expressions from the API or fallback in offline mode. | ||||||
|      * @returns {Promise<string[]>} |      * @returns {Promise<string[]>} | ||||||
| @@ -1376,9 +1399,6 @@ export async function getExpressionsList() { | |||||||
|         expressionsList = DEFAULT_EXPRESSIONS.slice(); |         expressionsList = DEFAULT_EXPRESSIONS.slice(); | ||||||
|         return expressionsList; |         return expressionsList; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     const result = await resolveExpressionsList(); |  | ||||||
|     return [...result, ...extension_settings.expressions.custom].filter(onlyUnique); |  | ||||||
| } | } | ||||||
|  |  | ||||||
| /** | /** | ||||||
| @@ -2106,6 +2126,10 @@ function migrateSettings() { | |||||||
|             extension_settings.expressions.rerollIfSame = !!$(this).prop('checked'); |             extension_settings.expressions.rerollIfSame = !!$(this).prop('checked'); | ||||||
|             saveSettingsDebounced(); |             saveSettingsDebounced(); | ||||||
|         }); |         }); | ||||||
|  |         $('#expressions_filter_available').prop('checked', extension_settings.expressions.filterAvailable).on('input', function () { | ||||||
|  |             extension_settings.expressions.filterAvailable = !!$(this).prop('checked'); | ||||||
|  |             saveSettingsDebounced(); | ||||||
|  |         }); | ||||||
|         $('#expression_override_cleanup_button').on('click', onClickExpressionOverrideRemoveAllButton); |         $('#expression_override_cleanup_button').on('click', onClickExpressionOverrideRemoveAllButton); | ||||||
|         $(document).on('dragstart', '.expression', (e) => { |         $(document).on('dragstart', '.expression', (e) => { | ||||||
|             e.preventDefault(); |             e.preventDefault(); | ||||||
| @@ -2283,13 +2307,13 @@ function migrateSettings() { | |||||||
|     SlashCommandParser.addCommandObject(SlashCommand.fromProps({ |     SlashCommandParser.addCommandObject(SlashCommand.fromProps({ | ||||||
|         name: 'expression-list', |         name: 'expression-list', | ||||||
|         aliases: ['expressions'], |         aliases: ['expressions'], | ||||||
|         /** @type {(args: {return: string}) => Promise<string>} */ |         /** @type {(args: {return: string, filter: string}) => Promise<string>} */ | ||||||
|         callback: async (args) => { |         callback: async (args) => { | ||||||
|             let returnType = |             let returnType = | ||||||
|                 /** @type {import('../../slash-commands/SlashCommandReturnHelper.js').SlashCommandReturnType} */ |                 /** @type {import('../../slash-commands/SlashCommandReturnHelper.js').SlashCommandReturnType} */ | ||||||
|                 (args.return); |                 (args.return); | ||||||
|  |  | ||||||
|             const list = await getExpressionsList(); |             const list = await getExpressionsList({ filterAvailable: !isFalseBoolean(args.filter) }); | ||||||
|  |  | ||||||
|             return await slashCommandReturnHelper.doReturn(returnType ?? 'pipe', list, { objectToStringFunc: list => list.join(', ') }); |             return await slashCommandReturnHelper.doReturn(returnType ?? 'pipe', list, { objectToStringFunc: list => list.join(', ') }); | ||||||
|         }, |         }, | ||||||
| @@ -2302,6 +2326,13 @@ function migrateSettings() { | |||||||
|                 enumList: slashCommandReturnHelper.enumList({ allowObject: true }), |                 enumList: slashCommandReturnHelper.enumList({ allowObject: true }), | ||||||
|                 forceEnum: true, |                 forceEnum: true, | ||||||
|             }), |             }), | ||||||
|  |             SlashCommandNamedArgument.fromProps({ | ||||||
|  |                 name: 'filter', | ||||||
|  |                 description: 'Filter the list to only include expressions that have available sprites for the current character.', | ||||||
|  |                 typeList: [ARGUMENT_TYPE.BOOLEAN], | ||||||
|  |                 enumList: commonEnumProviders.boolean('trueFalse')(), | ||||||
|  |                 defaultValue: 'true', | ||||||
|  |             }), | ||||||
|         ], |         ], | ||||||
|         returns: 'The comma-separated list of available expressions, including custom expressions.', |         returns: 'The comma-separated list of available expressions, including custom expressions.', | ||||||
|         helpString: 'Returns a list of available expressions, including custom expressions.', |         helpString: 'Returns a list of available expressions, including custom expressions.', | ||||||
| @@ -2317,6 +2348,13 @@ function migrateSettings() { | |||||||
|                 typeList: [ARGUMENT_TYPE.STRING], |                 typeList: [ARGUMENT_TYPE.STRING], | ||||||
|                 enumList: Object.keys(EXPRESSION_API).map(api => new SlashCommandEnumValue(api, null, enumTypes.enum)), |                 enumList: Object.keys(EXPRESSION_API).map(api => new SlashCommandEnumValue(api, null, enumTypes.enum)), | ||||||
|             }), |             }), | ||||||
|  |             SlashCommandNamedArgument.fromProps({ | ||||||
|  |                 name: 'filter', | ||||||
|  |                 description: 'Filter the list to only include expressions that have available sprites for the current character.', | ||||||
|  |                 typeList: [ARGUMENT_TYPE.BOOLEAN], | ||||||
|  |                 enumList: commonEnumProviders.boolean('trueFalse')(), | ||||||
|  |                 defaultValue: 'true', | ||||||
|  |             }), | ||||||
|             SlashCommandNamedArgument.fromProps({ |             SlashCommandNamedArgument.fromProps({ | ||||||
|                 name: 'prompt', |                 name: 'prompt', | ||||||
|                 description: 'Custom prompt for classification. Only relevant if Classifier API is set to LLM.', |                 description: 'Custom prompt for classification. Only relevant if Classifier API is set to LLM.', | ||||||
|   | |||||||
| @@ -29,7 +29,11 @@ | |||||||
|                 </select> |                 </select> | ||||||
|             </div> |             </div> | ||||||
|             <div class="expression_llm_prompt_block m-b-1 m-t-1"> |             <div class="expression_llm_prompt_block m-b-1 m-t-1"> | ||||||
|                 <label for="expression_llm_prompt" class="title_restorable"> |                 <label class="checkbox_label" for="expressions_filter_available" title="When using LLM or WebLLM classifier, only show and use expressions that have sprites assigned to them." data-i18n="[title]When using LLM or WebLLM classifier, only show and use expressions that have sprites assigned to them."> | ||||||
|  |                     <input id="expressions_filter_available" type="checkbox"> | ||||||
|  |                     <span data-i18n="Filter expressions for available sprites">Filter expressions for available sprites</span> | ||||||
|  |                 </label> | ||||||
|  |                 <label for="expression_llm_prompt" class="title_restorable m-t-1"> | ||||||
|                     <span data-i18n="LLM Prompt">LLM Prompt</span> |                     <span data-i18n="LLM Prompt">LLM Prompt</span> | ||||||
|                     <div id="expression_llm_prompt_restore" title="Restore default value" class="right_menu_button"> |                     <div id="expression_llm_prompt_restore" title="Restore default value" class="right_menu_button"> | ||||||
|                         <i class="fa-solid fa-clock-rotate-left fa-sm"></i> |                         <i class="fa-solid fa-clock-rotate-left fa-sm"></i> | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user