| | import { |
| | MAX_INJECTION_DEPTH, |
| | animation_duration, |
| | chat_metadata, |
| | eventSource, |
| | event_types, |
| | extension_prompt_roles, |
| | extension_prompt_types, |
| | saveSettingsDebounced, |
| | this_chid, |
| | } from '../script.js'; |
| | import { selected_group } from './group-chats.js'; |
| | import { extension_settings, getContext, saveMetadataDebounced } from './extensions.js'; |
| | import { getCharaFilename, debounce, delay } from './utils.js'; |
| | import { getTokenCountAsync } from './tokenizers.js'; |
| | import { debounce_timeout } from './constants.js'; |
| | import { SlashCommandParser } from './slash-commands/SlashCommandParser.js'; |
| | import { SlashCommand } from './slash-commands/SlashCommand.js'; |
| | import { ARGUMENT_TYPE, SlashCommandArgument } from './slash-commands/SlashCommandArgument.js'; |
| | export { MODULE_NAME as NOTE_MODULE_NAME }; |
| | import { t } from './i18n.js'; |
| | import { MacrosParser } from './macros.js'; |
| |
|
| | const MODULE_NAME = '2_floating_prompt'; |
| |
|
| | export var shouldWIAddPrompt = false; |
| |
|
| | export const metadata_keys = { |
| | prompt: 'note_prompt', |
| | interval: 'note_interval', |
| | depth: 'note_depth', |
| | position: 'note_position', |
| | role: 'note_role', |
| | }; |
| |
|
| | const chara_note_position = { |
| | replace: 0, |
| | before: 1, |
| | after: 2, |
| | }; |
| |
|
| | function setNoteTextCommand(_, text) { |
| | if (text) { |
| | $('#extension_floating_prompt').val(text).trigger('input'); |
| | toastr.success(t`Author's Note text updated`); |
| | } |
| | return chat_metadata[metadata_keys.prompt]; |
| | } |
| |
|
| | function setNoteDepthCommand(_, text) { |
| | if (text) { |
| | const value = Number(text); |
| |
|
| | if (Number.isNaN(value)) { |
| | toastr.error(t`Not a valid number`); |
| | return; |
| | } |
| |
|
| | $('#extension_floating_depth').val(Math.abs(value)).trigger('input'); |
| | toastr.success(t`Author's Note depth updated`); |
| | } |
| | return chat_metadata[metadata_keys.depth]; |
| | } |
| |
|
| | function setNoteIntervalCommand(_, text) { |
| | if (text) { |
| | const value = Number(text); |
| |
|
| | if (Number.isNaN(value)) { |
| | toastr.error(t`Not a valid number`); |
| | return; |
| | } |
| |
|
| | $('#extension_floating_interval').val(Math.abs(value)).trigger('input'); |
| | toastr.success(t`Author's Note frequency updated`); |
| | } |
| | return chat_metadata[metadata_keys.interval]; |
| | } |
| |
|
| | function setNotePositionCommand(_, text) { |
| | const validPositions = { |
| | 'after': 0, |
| | 'scenario': 0, |
| | 'chat': 1, |
| | 'before_scenario': 2, |
| | 'before': 2, |
| | }; |
| |
|
| | if (text) { |
| | const position = validPositions[text?.trim()?.toLowerCase()]; |
| |
|
| | if (typeof position === 'undefined') { |
| | toastr.error(t`Not a valid position`); |
| | return; |
| | } |
| |
|
| | $(`input[name="extension_floating_position"][value="${position}"]`).prop('checked', true).trigger('input'); |
| | toastr.info(t`Author's Note position updated`); |
| | } |
| | return Object.keys(validPositions).find(key => validPositions[key] == chat_metadata[metadata_keys.position]); |
| | } |
| |
|
| | function setNoteRoleCommand(_, text) { |
| | const validRoles = { |
| | 'system': 0, |
| | 'user': 1, |
| | 'assistant': 2, |
| | }; |
| |
|
| | if (text) { |
| | const role = validRoles[text?.trim()?.toLowerCase()]; |
| |
|
| | if (typeof role === 'undefined') { |
| | toastr.error(t`Not a valid role`); |
| | return; |
| | } |
| |
|
| | $('#extension_floating_role').val(Math.abs(role)).trigger('input'); |
| | toastr.info(t`Author's Note role updated`); |
| | } |
| | return Object.keys(validRoles).find(key => validRoles[key] == chat_metadata[metadata_keys.role]); |
| | } |
| |
|
| | function updateSettings() { |
| | saveSettingsDebounced(); |
| | loadSettings(); |
| | setFloatingPrompt(); |
| | } |
| |
|
| | const setMainPromptTokenCounterDebounced = debounce(async (value) => $('#extension_floating_prompt_token_counter').text(await getTokenCountAsync(value)), debounce_timeout.relaxed); |
| | const setCharaPromptTokenCounterDebounced = debounce(async (value) => $('#extension_floating_chara_token_counter').text(await getTokenCountAsync(value)), debounce_timeout.relaxed); |
| | const setDefaultPromptTokenCounterDebounced = debounce(async (value) => $('#extension_floating_default_token_counter').text(await getTokenCountAsync(value)), debounce_timeout.relaxed); |
| |
|
| | async function onExtensionFloatingPromptInput() { |
| | chat_metadata[metadata_keys.prompt] = $(this).val(); |
| | setMainPromptTokenCounterDebounced(chat_metadata[metadata_keys.prompt]); |
| | updateSettings(); |
| | saveMetadataDebounced(); |
| | } |
| |
|
| | async function onExtensionFloatingIntervalInput() { |
| | chat_metadata[metadata_keys.interval] = Number($(this).val()); |
| | updateSettings(); |
| | saveMetadataDebounced(); |
| | } |
| |
|
| | async function onExtensionFloatingDepthInput() { |
| | let value = Number($(this).val()); |
| |
|
| | if (value < 0) { |
| | value = Math.abs(value); |
| | $(this).val(value); |
| | } |
| |
|
| | chat_metadata[metadata_keys.depth] = value; |
| | updateSettings(); |
| | saveMetadataDebounced(); |
| | } |
| |
|
| | async function onExtensionFloatingPositionInput(e) { |
| | chat_metadata[metadata_keys.position] = Number(e.target.value); |
| | updateSettings(); |
| | saveMetadataDebounced(); |
| | } |
| |
|
| | async function onDefaultPositionInput(e) { |
| | extension_settings.note.defaultPosition = Number(e.target.value); |
| | saveSettingsDebounced(); |
| | } |
| |
|
| | async function onDefaultDepthInput() { |
| | let value = Number($(this).val()); |
| |
|
| | if (value < 0) { |
| | value = Math.abs(value); |
| | $(this).val(value); |
| | } |
| |
|
| | extension_settings.note.defaultDepth = value; |
| | saveSettingsDebounced(); |
| | } |
| |
|
| | async function onDefaultIntervalInput() { |
| | extension_settings.note.defaultInterval = Number($(this).val()); |
| | saveSettingsDebounced(); |
| | } |
| |
|
| | function onExtensionFloatingRoleInput(e) { |
| | chat_metadata[metadata_keys.role] = Number(e.target.value); |
| | updateSettings(); |
| | } |
| |
|
| | function onExtensionDefaultRoleInput(e) { |
| | extension_settings.note.defaultRole = Number(e.target.value); |
| | saveSettingsDebounced(); |
| | } |
| |
|
| | async function onExtensionFloatingCharPositionInput(e) { |
| | const value = e.target.value; |
| | const charaNote = extension_settings.note.chara.find((e) => e.name === getCharaFilename()); |
| |
|
| | if (charaNote) { |
| | charaNote.position = Number(value); |
| | updateSettings(); |
| | } |
| | } |
| |
|
| | function onExtensionFloatingCharaPromptInput() { |
| | const tempPrompt = $(this).val(); |
| | const avatarName = getCharaFilename(); |
| | let tempCharaNote = { |
| | name: avatarName, |
| | prompt: tempPrompt, |
| | }; |
| |
|
| | setCharaPromptTokenCounterDebounced(tempPrompt); |
| |
|
| | let existingCharaNoteIndex; |
| | let existingCharaNote; |
| |
|
| | if (extension_settings.note.chara) { |
| | existingCharaNoteIndex = extension_settings.note.chara.findIndex((e) => e.name === avatarName); |
| | existingCharaNote = extension_settings.note.chara[existingCharaNoteIndex]; |
| | } |
| |
|
| | if (tempPrompt.length === 0 && |
| | extension_settings.note.chara && |
| | existingCharaNote && |
| | !existingCharaNote.useChara |
| | ) { |
| | extension_settings.note.chara.splice(existingCharaNoteIndex, 1); |
| | } |
| | else if (extension_settings.note.chara && existingCharaNote) { |
| | Object.assign(existingCharaNote, tempCharaNote); |
| | } |
| | else if (avatarName && tempPrompt.length > 0) { |
| | if (!extension_settings.note.chara) { |
| | extension_settings.note.chara = []; |
| | } |
| | Object.assign(tempCharaNote, { useChara: false, position: chara_note_position.replace }); |
| |
|
| | extension_settings.note.chara.push(tempCharaNote); |
| | } else { |
| | console.log('Character author\'s note error: No avatar name key could be found.'); |
| | toastr.error(t`Something went wrong. Could not save character's author's note.`); |
| |
|
| | |
| | return; |
| | } |
| |
|
| | updateSettings(); |
| | } |
| |
|
| | function onExtensionFloatingCharaCheckboxChanged() { |
| | const value = !!$(this).prop('checked'); |
| | const charaNote = extension_settings.note.chara.find((e) => e.name === getCharaFilename()); |
| |
|
| | if (charaNote) { |
| | charaNote.useChara = value; |
| |
|
| | updateSettings(); |
| | } |
| | } |
| |
|
| | function onExtensionFloatingDefaultInput() { |
| | extension_settings.note.default = $(this).val(); |
| | setDefaultPromptTokenCounterDebounced(extension_settings.note.default); |
| | updateSettings(); |
| | } |
| |
|
| | function loadSettings() { |
| | const DEFAULT_DEPTH = 4; |
| | const DEFAULT_POSITION = 1; |
| | const DEFAULT_INTERVAL = 1; |
| | const DEFAULT_ROLE = extension_prompt_roles.SYSTEM; |
| |
|
| | if (extension_settings.note.defaultPosition === undefined) { |
| | extension_settings.note.defaultPosition = DEFAULT_POSITION; |
| | } |
| |
|
| | if (extension_settings.note.defaultDepth === undefined) { |
| | extension_settings.note.defaultDepth = DEFAULT_DEPTH; |
| | } |
| |
|
| | if (extension_settings.note.defaultInterval === undefined) { |
| | extension_settings.note.defaultInterval = DEFAULT_INTERVAL; |
| | } |
| |
|
| | if (extension_settings.note.defaultRole === undefined) { |
| | extension_settings.note.defaultRole = DEFAULT_ROLE; |
| | } |
| |
|
| | chat_metadata[metadata_keys.prompt] = chat_metadata[metadata_keys.prompt] ?? extension_settings.note.default ?? ''; |
| | chat_metadata[metadata_keys.interval] = chat_metadata[metadata_keys.interval] ?? extension_settings.note.defaultInterval ?? DEFAULT_INTERVAL; |
| | chat_metadata[metadata_keys.position] = chat_metadata[metadata_keys.position] ?? extension_settings.note.defaultPosition ?? DEFAULT_POSITION; |
| | chat_metadata[metadata_keys.depth] = chat_metadata[metadata_keys.depth] ?? extension_settings.note.defaultDepth ?? DEFAULT_DEPTH; |
| | chat_metadata[metadata_keys.role] = chat_metadata[metadata_keys.role] ?? extension_settings.note.defaultRole ?? DEFAULT_ROLE; |
| | $('#extension_floating_prompt').val(chat_metadata[metadata_keys.prompt]); |
| | $('#extension_floating_interval').val(chat_metadata[metadata_keys.interval]); |
| | $('#extension_floating_allow_wi_scan').prop('checked', extension_settings.note.allowWIScan ?? false); |
| | $('#extension_floating_depth').val(chat_metadata[metadata_keys.depth]); |
| | $('#extension_floating_role').val(chat_metadata[metadata_keys.role]); |
| | $(`input[name="extension_floating_position"][value="${chat_metadata[metadata_keys.position]}"]`).prop('checked', true); |
| |
|
| | if (extension_settings.note.chara && getContext().characterId !== undefined) { |
| | const charaNote = extension_settings.note.chara.find((e) => e.name === getCharaFilename()); |
| |
|
| | $('#extension_floating_chara').val(charaNote ? charaNote.prompt : ''); |
| | $('#extension_use_floating_chara').prop('checked', charaNote ? charaNote.useChara : false); |
| | $(`input[name="extension_floating_char_position"][value="${charaNote?.position ?? chara_note_position.replace}"]`).prop('checked', true); |
| | } else { |
| | $('#extension_floating_chara').val(''); |
| | $('#extension_use_floating_chara').prop('checked', false); |
| | $(`input[name="extension_floating_char_position"][value="${chara_note_position.replace}"]`).prop('checked', true); |
| | } |
| |
|
| | $('#extension_floating_default').val(extension_settings.note.default); |
| | $('#extension_default_depth').val(extension_settings.note.defaultDepth); |
| | $('#extension_default_interval').val(extension_settings.note.defaultInterval); |
| | $('#extension_default_role').val(extension_settings.note.defaultRole); |
| | $(`input[name="extension_default_position"][value="${extension_settings.note.defaultPosition}"]`).prop('checked', true); |
| | } |
| |
|
| | export function setFloatingPrompt() { |
| | const context = getContext(); |
| | if (!context.groupId && context.characterId === undefined) { |
| | console.debug('setFloatingPrompt: Not in a chat. Skipping.'); |
| | shouldWIAddPrompt = false; |
| | return; |
| | } |
| |
|
| | |
| | let lastMessageNumber = Array.isArray(context.chat) && context.chat.length ? context.chat.filter(m => m.is_user).length : 0; |
| |
|
| | console.debug(` |
| | setFloatingPrompt entered |
| | ------ |
| | lastMessageNumber = ${lastMessageNumber} |
| | metadata_keys.interval = ${chat_metadata[metadata_keys.interval]} |
| | metadata_keys.position = ${chat_metadata[metadata_keys.position]} |
| | metadata_keys.depth = ${chat_metadata[metadata_keys.depth]} |
| | metadata_keys.role = ${chat_metadata[metadata_keys.role]} |
| | ------ |
| | `); |
| |
|
| | |
| | if (chat_metadata[metadata_keys.interval] === 1) { |
| | lastMessageNumber = 1; |
| | } |
| |
|
| | if (lastMessageNumber <= 0 || chat_metadata[metadata_keys.interval] <= 0) { |
| | context.setExtensionPrompt(MODULE_NAME, '', extension_prompt_types.NONE, MAX_INJECTION_DEPTH); |
| | $('#extension_floating_counter').text('(disabled)'); |
| | shouldWIAddPrompt = false; |
| | return; |
| | } |
| |
|
| | const messagesTillInsertion = lastMessageNumber >= chat_metadata[metadata_keys.interval] |
| | ? (lastMessageNumber % chat_metadata[metadata_keys.interval]) |
| | : (chat_metadata[metadata_keys.interval] - lastMessageNumber); |
| | const shouldAddPrompt = messagesTillInsertion == 0; |
| | shouldWIAddPrompt = shouldAddPrompt; |
| |
|
| | let prompt = shouldAddPrompt ? $('#extension_floating_prompt').val() : ''; |
| | if (shouldAddPrompt && extension_settings.note.chara && getContext().characterId !== undefined) { |
| | const charaNote = extension_settings.note.chara.find((e) => e.name === getCharaFilename()); |
| |
|
| | |
| | if (charaNote && charaNote.useChara) { |
| | switch (charaNote.position) { |
| | case chara_note_position.before: |
| | prompt = charaNote.prompt + '\n' + prompt; |
| | break; |
| | case chara_note_position.after: |
| | prompt = prompt + '\n' + charaNote.prompt; |
| | break; |
| | default: |
| | prompt = charaNote.prompt; |
| | break; |
| | } |
| | } |
| | } |
| | context.setExtensionPrompt( |
| | MODULE_NAME, |
| | String(prompt), |
| | chat_metadata[metadata_keys.position], |
| | chat_metadata[metadata_keys.depth], |
| | extension_settings.note.allowWIScan, |
| | chat_metadata[metadata_keys.role], |
| | ); |
| | $('#extension_floating_counter').text(shouldAddPrompt ? '0' : messagesTillInsertion); |
| | } |
| |
|
| | function onANMenuItemClick() { |
| | if (!selected_group && this_chid === undefined) { |
| | toastr.warning(t`Select a character before trying to use Author's Note`, '', { timeOut: 2000 }); |
| | return; |
| | } |
| |
|
| | |
| | const $ANcontainer = $('#floatingPrompt'); |
| | if ($ANcontainer.css('display') !== 'flex') { |
| | $ANcontainer.addClass('resizing'); |
| | $ANcontainer.css('display', 'flex'); |
| | $ANcontainer.css('opacity', 0.0); |
| | $ANcontainer.transition({ |
| | opacity: 1.0, |
| | duration: animation_duration, |
| | }, async function () { |
| | await delay(50); |
| | $ANcontainer.removeClass('resizing'); |
| | }); |
| |
|
| | |
| | if ($('#ANBlockToggle') |
| | .siblings('.inline-drawer-content') |
| | .css('display') !== 'block') { |
| | $ANcontainer.addClass('resizing'); |
| | $('#ANBlockToggle').trigger('click'); |
| | } |
| | } else { |
| | |
| | $ANcontainer.addClass('resizing'); |
| | $ANcontainer.transition({ |
| | opacity: 0.0, |
| | duration: animation_duration, |
| | }, async function () { |
| | await delay(50); |
| | $ANcontainer.removeClass('resizing'); |
| | }); |
| | setTimeout(function () { |
| | $ANcontainer.hide(); |
| | }, animation_duration); |
| | } |
| |
|
| | |
| | |
| | $('#options').stop().fadeOut(animation_duration); |
| | } |
| |
|
| | async function onChatChanged() { |
| | loadSettings(); |
| | setFloatingPrompt(); |
| | const context = getContext(); |
| |
|
| | |
| | $('#extension_floating_chara').prop('disabled', !!context.groupId); |
| |
|
| | const tokenCounter1 = chat_metadata[metadata_keys.prompt] ? await getTokenCountAsync(chat_metadata[metadata_keys.prompt]) : 0; |
| | $('#extension_floating_prompt_token_counter').text(tokenCounter1); |
| |
|
| | let tokenCounter2; |
| | if (extension_settings.note.chara && context.characterId !== undefined) { |
| | const charaNote = extension_settings.note.chara.find((e) => e.name === getCharaFilename()); |
| |
|
| | if (charaNote) { |
| | tokenCounter2 = await getTokenCountAsync(charaNote.prompt); |
| | } |
| | } |
| |
|
| | $('#extension_floating_chara_token_counter').text(tokenCounter2 || 0); |
| |
|
| | const tokenCounter3 = extension_settings.note.default ? await getTokenCountAsync(extension_settings.note.default) : 0; |
| | $('#extension_floating_default_token_counter').text(tokenCounter3); |
| | } |
| |
|
| | function onAllowWIScanCheckboxChanged() { |
| | extension_settings.note.allowWIScan = !!$(this).prop('checked'); |
| | updateSettings(); |
| | } |
| |
|
| | |
| | |
| | |
| | |
| | export function initAuthorsNote() { |
| | $('#extension_floating_prompt').on('input', onExtensionFloatingPromptInput); |
| | $('#extension_floating_interval').on('input', onExtensionFloatingIntervalInput); |
| | $('#extension_floating_depth').on('input', onExtensionFloatingDepthInput); |
| | $('#extension_floating_chara').on('input', onExtensionFloatingCharaPromptInput); |
| | $('#extension_use_floating_chara').on('input', onExtensionFloatingCharaCheckboxChanged); |
| | $('#extension_floating_default').on('input', onExtensionFloatingDefaultInput); |
| | $('#extension_default_depth').on('input', onDefaultDepthInput); |
| | $('#extension_default_interval').on('input', onDefaultIntervalInput); |
| | $('#extension_floating_allow_wi_scan').on('input', onAllowWIScanCheckboxChanged); |
| | $('#extension_floating_role').on('input', onExtensionFloatingRoleInput); |
| | $('#extension_default_role').on('input', onExtensionDefaultRoleInput); |
| | $('input[name="extension_floating_position"]').on('change', onExtensionFloatingPositionInput); |
| | $('input[name="extension_default_position"]').on('change', onDefaultPositionInput); |
| | $('input[name="extension_floating_char_position"]').on('change', onExtensionFloatingCharPositionInput); |
| | $('#ANClose').on('click', function () { |
| | $('#floatingPrompt').transition({ |
| | opacity: 0, |
| | duration: animation_duration, |
| | easing: 'ease-in-out', |
| | }); |
| | setTimeout(function () { $('#floatingPrompt').hide(); }, animation_duration); |
| | }); |
| | $('#option_toggle_AN').on('click', onANMenuItemClick); |
| |
|
| | SlashCommandParser.addCommandObject(SlashCommand.fromProps({ |
| | name: 'note', |
| | callback: setNoteTextCommand, |
| | returns: 'current author\'s note', |
| | unnamedArgumentList: [ |
| | new SlashCommandArgument( |
| | 'text', [ARGUMENT_TYPE.STRING], false, |
| | ), |
| | ], |
| | helpString: ` |
| | <div> |
| | Sets an author's note for the currently selected chat if specified and returns the current note. |
| | </div> |
| | `, |
| | })); |
| | SlashCommandParser.addCommandObject(SlashCommand.fromProps({ |
| | name: 'note-depth', |
| | aliases: ['depth'], |
| | callback: setNoteDepthCommand, |
| | returns: 'current author\'s note depth', |
| | unnamedArgumentList: [ |
| | new SlashCommandArgument( |
| | 'number', [ARGUMENT_TYPE.NUMBER], false, |
| | ), |
| | ], |
| | helpString: ` |
| | <div> |
| | Sets an author's note depth for in-chat positioning if specified and returns the current depth. |
| | </div> |
| | `, |
| | })); |
| | SlashCommandParser.addCommandObject(SlashCommand.fromProps({ |
| | name: 'note-frequency', |
| | aliases: ['freq', 'note-freq'], |
| | callback: setNoteIntervalCommand, |
| | returns: 'current author\'s note insertion frequency', |
| | namedArgumentList: [], |
| | unnamedArgumentList: [ |
| | new SlashCommandArgument( |
| | 'number', [ARGUMENT_TYPE.NUMBER], false, |
| | ), |
| | ], |
| | helpString: ` |
| | <div> |
| | Sets an author's note insertion frequency if specified and returns the current frequency. |
| | </div> |
| | `, |
| | })); |
| | SlashCommandParser.addCommandObject(SlashCommand.fromProps({ |
| | name: 'note-position', |
| | callback: setNotePositionCommand, |
| | aliases: ['pos', 'note-pos'], |
| | returns: 'current author\'s note insertion position', |
| | namedArgumentList: [], |
| | unnamedArgumentList: [ |
| | new SlashCommandArgument( |
| | 'position', [ARGUMENT_TYPE.STRING], false, false, null, ['before', 'after', 'chat'], |
| | ), |
| | ], |
| | helpString: ` |
| | <div> |
| | Sets an author's note position if specified and returns the current position. |
| | </div> |
| | `, |
| | })); |
| | SlashCommandParser.addCommandObject(SlashCommand.fromProps({ |
| | name: 'note-role', |
| | callback: setNoteRoleCommand, |
| | returns: 'current author\'s note chat insertion role', |
| | namedArgumentList: [], |
| | unnamedArgumentList: [ |
| | new SlashCommandArgument( |
| | 'role', [ARGUMENT_TYPE.STRING], false, false, null, ['system', 'user', 'assistant'], |
| | ), |
| | ], |
| | helpString: ` |
| | <div> |
| | Sets an author's note chat insertion role if specified and returns the current role. |
| | </div> |
| | `, |
| | })); |
| | eventSource.on(event_types.CHAT_CHANGED, onChatChanged); |
| |
|
| | MacrosParser.registerMacro('authorsNote', () => chat_metadata[metadata_keys.prompt] ?? '', t`The contents of the Author's Note`); |
| | MacrosParser.registerMacro('charAuthorsNote', () => this_chid !== undefined ? (extension_settings.note.chara.find((e) => e.name === getCharaFilename())?.prompt ?? '') : '', t`The contents of the Character Author's Note`); |
| | MacrosParser.registerMacro('defaultAuthorsNote', () => extension_settings.note.default ?? '', t`The contents of the Default Author's Note`); |
| | } |
| |
|