|
import { |
|
animation_duration, |
|
chat_metadata, |
|
eventSource, |
|
event_types, |
|
extension_prompt_roles, |
|
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_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, |
|
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; |
|
} |
|
|
|
|
|
if ($('#floatingPrompt').css('display') !== 'flex') { |
|
$('#floatingPrompt').addClass('resizing'); |
|
$('#floatingPrompt').css('display', 'flex'); |
|
$('#floatingPrompt').css('opacity', 0.0); |
|
$('#floatingPrompt').transition({ |
|
opacity: 1.0, |
|
duration: animation_duration, |
|
}, async function () { |
|
await delay(50); |
|
$('#floatingPrompt').removeClass('resizing'); |
|
}); |
|
|
|
|
|
if ($('#ANBlockToggle') |
|
.siblings('.inline-drawer-content') |
|
.css('display') !== 'block') { |
|
$('#floatingPrompt').addClass('resizing'); |
|
$('#ANBlockToggle').click(); |
|
} |
|
} else { |
|
|
|
$('#floatingPrompt').addClass('resizing'); |
|
$('#floatingPrompt').transition({ |
|
opacity: 0.0, |
|
duration: animation_duration, |
|
}, async function () { |
|
await delay(50); |
|
$('#floatingPrompt').removeClass('resizing'); |
|
}); |
|
setTimeout(function () { |
|
$('#floatingPrompt').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`); |
|
} |
|
|