mirror of
https://github.com/SillyTavern/SillyTavern.git
synced 2025-06-05 21:59:27 +02:00
Move reasoning stuff to reasoning.js
This commit is contained in:
@@ -225,7 +225,7 @@ import {
|
|||||||
instruct_presets,
|
instruct_presets,
|
||||||
selectContextPreset,
|
selectContextPreset,
|
||||||
} from './scripts/instruct-mode.js';
|
} from './scripts/instruct-mode.js';
|
||||||
import { getCurrentLocale, initLocales, t } from './scripts/i18n.js';
|
import { initLocales, t } from './scripts/i18n.js';
|
||||||
import { getFriendlyTokenizerName, getTokenCount, getTokenCountAsync, initTokenizers, saveTokenCache, TOKENIZER_SUPPORTED_KEY } from './scripts/tokenizers.js';
|
import { getFriendlyTokenizerName, getTokenCount, getTokenCountAsync, initTokenizers, saveTokenCache, TOKENIZER_SUPPORTED_KEY } from './scripts/tokenizers.js';
|
||||||
import {
|
import {
|
||||||
user_avatar,
|
user_avatar,
|
||||||
@@ -269,7 +269,7 @@ import { initSettingsSearch } from './scripts/setting-search.js';
|
|||||||
import { initBulkEdit } from './scripts/bulk-edit.js';
|
import { initBulkEdit } from './scripts/bulk-edit.js';
|
||||||
import { deriveTemplatesFromChatTemplate } from './scripts/chat-templates.js';
|
import { deriveTemplatesFromChatTemplate } from './scripts/chat-templates.js';
|
||||||
import { getContext } from './scripts/st-context.js';
|
import { getContext } from './scripts/st-context.js';
|
||||||
import { initReasoning, PromptReasoning } from './scripts/reasoning.js';
|
import { extractReasoningFromData, initReasoning, PromptReasoning, updateReasoningTimeUI } from './scripts/reasoning.js';
|
||||||
|
|
||||||
// API OBJECT FOR EXTERNAL WIRING
|
// API OBJECT FOR EXTERNAL WIRING
|
||||||
globalThis.SillyTavern = {
|
globalThis.SillyTavern = {
|
||||||
@@ -5763,56 +5763,6 @@ function extractMessageFromData(data) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Extracts the reasoning from the response data.
|
|
||||||
* @param {object} data Response data
|
|
||||||
* @returns {string} Extracted reasoning
|
|
||||||
*/
|
|
||||||
function extractReasoningFromData(data) {
|
|
||||||
switch (main_api) {
|
|
||||||
case 'textgenerationwebui':
|
|
||||||
switch (textgen_settings.type) {
|
|
||||||
case textgen_types.OPENROUTER:
|
|
||||||
return data?.choices?.[0]?.reasoning ?? '';
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
|
|
||||||
case 'openai':
|
|
||||||
if (!oai_settings.show_thoughts) break;
|
|
||||||
|
|
||||||
switch (oai_settings.chat_completion_source) {
|
|
||||||
case chat_completion_sources.DEEPSEEK:
|
|
||||||
return data?.choices?.[0]?.message?.reasoning_content ?? '';
|
|
||||||
case chat_completion_sources.OPENROUTER:
|
|
||||||
return data?.choices?.[0]?.message?.reasoning ?? '';
|
|
||||||
case chat_completion_sources.MAKERSUITE:
|
|
||||||
return data?.responseContent?.parts?.filter(part => part.thought)?.map(part => part.text)?.join('\n\n') ?? '';
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
return '';
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Updates the Reasoning controls
|
|
||||||
* @param {HTMLElement} element The element to update
|
|
||||||
* @param {number?} duration The duration of the reasoning in milliseconds
|
|
||||||
* @param {object} [options={}] Options for the function
|
|
||||||
* @param {boolean} [options.forceEnd=false] If true, there will be no "Thinking..." when no duration exists
|
|
||||||
*/
|
|
||||||
function updateReasoningTimeUI(element, duration, { forceEnd = false } = {}) {
|
|
||||||
if (duration) {
|
|
||||||
const durationStr = moment.duration(duration).locale(getCurrentLocale()).humanize({ s: 50, ss: 9 });
|
|
||||||
element.textContent = t`Thought for ${durationStr}`;
|
|
||||||
} else if (forceEnd) {
|
|
||||||
element.textContent = t`Thought for some time`;
|
|
||||||
} else {
|
|
||||||
element.textContent = t`Thinking...`;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Extracts multiswipe swipes from the response data.
|
* Extracts multiswipe swipes from the response data.
|
||||||
* @param {Object} data Response data
|
* @param {Object} data Response data
|
||||||
@@ -10872,18 +10822,6 @@ jQuery(async function () {
|
|||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
$(document).on('click', '.mes_reasoning_header', function () {
|
|
||||||
// If we are in message edit mode and reasoning area is closed, a click opens and edits it
|
|
||||||
const mes = $(this).closest('.mes');
|
|
||||||
const mesEditArea = mes.find('#curEditTextarea');
|
|
||||||
if (mesEditArea.length) {
|
|
||||||
const summary = $(mes).find('.mes_reasoning_summary');
|
|
||||||
if (!summary.attr('open')) {
|
|
||||||
summary.find('.mes_reasoning_edit').trigger('click');
|
|
||||||
}
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
$(document).on('input', '#curEditTextarea', function () {
|
$(document).on('input', '#curEditTextarea', function () {
|
||||||
if (power_user.auto_save_msg_edits === true) {
|
if (power_user.auto_save_msg_edits === true) {
|
||||||
messageEditAuto($(this));
|
messageEditAuto($(this));
|
||||||
@@ -11438,17 +11376,6 @@ jQuery(async function () {
|
|||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
$(document).on('click', '.mes_reasoning_summary', function () {
|
|
||||||
// If you toggle summary header while editing reasoning, yup - we just cancel it
|
|
||||||
$(this).closest('.mes').find('.mes_reasoning_edit_cancel:visible').trigger('click');
|
|
||||||
});
|
|
||||||
|
|
||||||
$(document).on('click', '.mes_reasoning_details', function (e) {
|
|
||||||
if (!e.target.closest('.mes_reasoning_actions') && !e.target.closest('.mes_reasoning_header')) {
|
|
||||||
e.preventDefault();
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
$(document).keyup(function (e) {
|
$(document).keyup(function (e) {
|
||||||
if (e.key === 'Escape') {
|
if (e.key === 'Escape') {
|
||||||
const isEditVisible = $('#curEditTextarea').is(':visible') || $('.reasoning_edit_textarea').length > 0;
|
const isEditVisible = $('#curEditTextarea').is(':visible') || $('.reasoning_edit_textarea').length > 0;
|
||||||
|
@@ -1,13 +1,18 @@
|
|||||||
import { chat, closeMessageEditor, event_types, eventSource, saveChatConditional, saveSettingsDebounced, substituteParams, updateMessageBlock } from '../script.js';
|
import {
|
||||||
|
moment,
|
||||||
|
} from '../lib.js';
|
||||||
|
import { chat, closeMessageEditor, event_types, eventSource, main_api, saveChatConditional, saveSettingsDebounced, substituteParams, updateMessageBlock } from '../script.js';
|
||||||
import { getRegexedString, regex_placement } from './extensions/regex/engine.js';
|
import { getRegexedString, regex_placement } from './extensions/regex/engine.js';
|
||||||
import { t } from './i18n.js';
|
import { getCurrentLocale, t } from './i18n.js';
|
||||||
import { MacrosParser } from './macros.js';
|
import { MacrosParser } from './macros.js';
|
||||||
|
import { chat_completion_sources, oai_settings } from './openai.js';
|
||||||
import { Popup } from './popup.js';
|
import { Popup } from './popup.js';
|
||||||
import { power_user } from './power-user.js';
|
import { power_user } from './power-user.js';
|
||||||
import { SlashCommand } from './slash-commands/SlashCommand.js';
|
import { SlashCommand } from './slash-commands/SlashCommand.js';
|
||||||
import { ARGUMENT_TYPE, SlashCommandArgument, SlashCommandNamedArgument } from './slash-commands/SlashCommandArgument.js';
|
import { ARGUMENT_TYPE, SlashCommandArgument, SlashCommandNamedArgument } from './slash-commands/SlashCommandArgument.js';
|
||||||
import { commonEnumProviders } from './slash-commands/SlashCommandCommonEnumsProvider.js';
|
import { commonEnumProviders } from './slash-commands/SlashCommandCommonEnumsProvider.js';
|
||||||
import { SlashCommandParser } from './slash-commands/SlashCommandParser.js';
|
import { SlashCommandParser } from './slash-commands/SlashCommandParser.js';
|
||||||
|
import { textgen_types, textgenerationwebui_settings } from './textgen-settings.js';
|
||||||
import { copyText, escapeRegex, isFalseBoolean } from './utils.js';
|
import { copyText, escapeRegex, isFalseBoolean } from './utils.js';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -34,6 +39,55 @@ function toggleReasoningAutoExpand() {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Extracts the reasoning from the response data.
|
||||||
|
* @param {object} data Response data
|
||||||
|
* @returns {string} Extracted reasoning
|
||||||
|
*/
|
||||||
|
export function extractReasoningFromData(data) {
|
||||||
|
switch (main_api) {
|
||||||
|
case 'textgenerationwebui':
|
||||||
|
switch (textgenerationwebui_settings.type) {
|
||||||
|
case textgen_types.OPENROUTER:
|
||||||
|
return data?.choices?.[0]?.reasoning ?? '';
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
|
||||||
|
case 'openai':
|
||||||
|
if (!oai_settings.show_thoughts) break;
|
||||||
|
|
||||||
|
switch (oai_settings.chat_completion_source) {
|
||||||
|
case chat_completion_sources.DEEPSEEK:
|
||||||
|
return data?.choices?.[0]?.message?.reasoning_content ?? '';
|
||||||
|
case chat_completion_sources.OPENROUTER:
|
||||||
|
return data?.choices?.[0]?.message?.reasoning ?? '';
|
||||||
|
case chat_completion_sources.MAKERSUITE:
|
||||||
|
return data?.responseContent?.parts?.filter(part => part.thought)?.map(part => part.text)?.join('\n\n') ?? '';
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
return '';
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Updates the Reasoning controls
|
||||||
|
* @param {HTMLElement} element The element to update
|
||||||
|
* @param {number?} duration The duration of the reasoning in milliseconds
|
||||||
|
* @param {object} [options={}] Options for the function
|
||||||
|
* @param {boolean} [options.forceEnd=false] If true, there will be no "Thinking..." when no duration exists
|
||||||
|
*/
|
||||||
|
export function updateReasoningTimeUI(element, duration, { forceEnd = false } = {}) {
|
||||||
|
if (duration) {
|
||||||
|
const durationStr = moment.duration(duration).locale(getCurrentLocale()).humanize({ s: 50, ss: 9 });
|
||||||
|
element.textContent = t`Thought for ${durationStr}`;
|
||||||
|
} else if (forceEnd) {
|
||||||
|
element.textContent = t`Thought for some time`;
|
||||||
|
} else {
|
||||||
|
element.textContent = t`Thinking...`;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Helper class for adding reasoning to messages.
|
* Helper class for adding reasoning to messages.
|
||||||
* Keeps track of the number of reasoning additions.
|
* Keeps track of the number of reasoning additions.
|
||||||
@@ -247,6 +301,29 @@ function registerReasoningMacros() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
function setReasoningEventHandlers() {
|
function setReasoningEventHandlers() {
|
||||||
|
$(document).on('click', '.mes_reasoning_summary', function () {
|
||||||
|
// If you toggle summary header while editing reasoning, yup - we just cancel it
|
||||||
|
$(this).closest('.mes').find('.mes_reasoning_edit_cancel:visible').trigger('click');
|
||||||
|
});
|
||||||
|
|
||||||
|
$(document).on('click', '.mes_reasoning_details', function (e) {
|
||||||
|
if (!e.target.closest('.mes_reasoning_actions') && !e.target.closest('.mes_reasoning_header')) {
|
||||||
|
e.preventDefault();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
$(document).on('click', '.mes_reasoning_header', function () {
|
||||||
|
// If we are in message edit mode and reasoning area is closed, a click opens and edits it
|
||||||
|
const mes = $(this).closest('.mes');
|
||||||
|
const mesEditArea = mes.find('#curEditTextarea');
|
||||||
|
if (mesEditArea.length) {
|
||||||
|
const summary = $(mes).find('.mes_reasoning_summary');
|
||||||
|
if (!summary.attr('open')) {
|
||||||
|
summary.find('.mes_reasoning_edit').trigger('click');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
$(document).on('click', '.mes_reasoning_copy', (e) => {
|
$(document).on('click', '.mes_reasoning_copy', (e) => {
|
||||||
e.stopPropagation();
|
e.stopPropagation();
|
||||||
e.preventDefault();
|
e.preventDefault();
|
||||||
|
Reference in New Issue
Block a user