mirror of
				https://github.com/SillyTavern/SillyTavern.git
				synced 2025-06-05 21:59:27 +02:00 
			
		
		
		
	
		
			
				
	
	
		
			580 lines
		
	
	
		
			21 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			580 lines
		
	
	
		
			21 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
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';
 | 
						|
 | 
						|
const MODULE_NAME = '2_floating_prompt'; // <= Deliberate, for sorting lower than memory
 | 
						|
 | 
						|
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.`);
 | 
						|
 | 
						|
        // Don't save settings if something went wrong
 | 
						|
        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;
 | 
						|
    }
 | 
						|
 | 
						|
    // take the count of messages
 | 
						|
    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]}
 | 
						|
    ------
 | 
						|
    `);
 | 
						|
 | 
						|
    // interval 1 should be inserted no matter what
 | 
						|
    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());
 | 
						|
 | 
						|
        // Only replace with the chara note if the user checked the box
 | 
						|
        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;
 | 
						|
    }
 | 
						|
 | 
						|
    //show AN if it's hidden
 | 
						|
    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');
 | 
						|
        });
 | 
						|
 | 
						|
        //auto-open the main AN inline drawer
 | 
						|
        if ($('#ANBlockToggle')
 | 
						|
            .siblings('.inline-drawer-content')
 | 
						|
            .css('display') !== 'block') {
 | 
						|
            $('#floatingPrompt').addClass('resizing');
 | 
						|
            $('#ANBlockToggle').click();
 | 
						|
        }
 | 
						|
    } else {
 | 
						|
        //hide AN if it's already displayed
 | 
						|
        $('#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);
 | 
						|
    }
 | 
						|
 | 
						|
    //duplicate options menu close handler from script.js
 | 
						|
    //because this listener takes priority
 | 
						|
    $('#options').stop().fadeOut(animation_duration);
 | 
						|
}
 | 
						|
 | 
						|
async function onChatChanged() {
 | 
						|
    loadSettings();
 | 
						|
    setFloatingPrompt();
 | 
						|
    const context = getContext();
 | 
						|
 | 
						|
    // Disable the chara note if in a group
 | 
						|
    $('#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();
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 * Inject author's note options and setup event listeners.
 | 
						|
 */
 | 
						|
// Inserts the extension first since it's statically imported
 | 
						|
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);
 | 
						|
}
 |