mirror of
				https://github.com/SillyTavern/SillyTavern.git
				synced 2025-06-05 21:59:27 +02:00 
			
		
		
		
	Add ability to omit settings from Connection Profiles
This commit is contained in:
		
							
								
								
									
										16
									
								
								public/scripts/extensions/connection-manager/edit.html
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										16
									
								
								public/scripts/extensions/connection-manager/edit.html
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,16 @@ | |||||||
|  | <div> | ||||||
|  |     <h3>Included settings:</h3> | ||||||
|  |     <div class="justifyLeft flex-container flexFlowColumn flexNoGap"> | ||||||
|  |         {{#each settings}} | ||||||
|  |         <label class="checkbox_label"> | ||||||
|  |             {{#if this}} | ||||||
|  |                 <input type="checkbox" value="{{@key}}" name="exclude" checked> | ||||||
|  |             {{else}} | ||||||
|  |                 <input type="checkbox" value="{{@key}}" name="exclude"> | ||||||
|  |             {{/if}} | ||||||
|  |             <span>{{@key}}</span> | ||||||
|  |         </label> | ||||||
|  |         {{/each}} | ||||||
|  |     </div> | ||||||
|  |     <h3>Profile name:</h3> | ||||||
|  | </div> | ||||||
| @@ -135,6 +135,7 @@ const profilesProvider = () => [ | |||||||
|  * @property {string} [context] Context Template |  * @property {string} [context] Context Template | ||||||
|  * @property {string} [instruct-state] Instruct Mode |  * @property {string} [instruct-state] Instruct Mode | ||||||
|  * @property {string} [tokenizer] Tokenizer |  * @property {string} [tokenizer] Tokenizer | ||||||
|  |  * @property {string[]} [exclude] Commands to exclude | ||||||
|  */ |  */ | ||||||
|  |  | ||||||
| /** | /** | ||||||
| @@ -171,8 +172,13 @@ function findProfileByName(value) { | |||||||
| async function readProfileFromCommands(mode, profile, cleanUp = false) { | async function readProfileFromCommands(mode, profile, cleanUp = false) { | ||||||
|     const commands = mode === 'cc' ? CC_COMMANDS : TC_COMMANDS; |     const commands = mode === 'cc' ? CC_COMMANDS : TC_COMMANDS; | ||||||
|     const opposingCommands = mode === 'cc' ? TC_COMMANDS : CC_COMMANDS; |     const opposingCommands = mode === 'cc' ? TC_COMMANDS : CC_COMMANDS; | ||||||
|  |     const excludeList = Array.isArray(profile.exclude) ? profile.exclude : []; | ||||||
|     for (const command of commands) { |     for (const command of commands) { | ||||||
|         try { |         try { | ||||||
|  |             if (excludeList.includes(command)) { | ||||||
|  |                 continue; | ||||||
|  |             } | ||||||
|  |  | ||||||
|             const args = getNamedArguments(); |             const args = getNamedArguments(); | ||||||
|             const result = await SlashCommandParser.commands[command].callback(args, ''); |             const result = await SlashCommandParser.commands[command].callback(args, ''); | ||||||
|             if (result) { |             if (result) { | ||||||
| @@ -208,15 +214,36 @@ async function readProfileFromCommands(mode, profile, cleanUp = false) { | |||||||
| async function createConnectionProfile(forceName = null) { | async function createConnectionProfile(forceName = null) { | ||||||
|     const mode = main_api === 'openai' ? 'cc' : 'tc'; |     const mode = main_api === 'openai' ? 'cc' : 'tc'; | ||||||
|     const id = uuidv4(); |     const id = uuidv4(); | ||||||
|  |     /** @type {ConnectionProfile} */ | ||||||
|     const profile = { |     const profile = { | ||||||
|         id, |         id, | ||||||
|         mode, |         mode, | ||||||
|  |         exclude: [], | ||||||
|     }; |     }; | ||||||
|  |  | ||||||
|     await readProfileFromCommands(mode, profile); |     await readProfileFromCommands(mode, profile); | ||||||
|  |  | ||||||
|     const profileForDisplay = makeFancyProfile(profile); |     const profileForDisplay = makeFancyProfile(profile); | ||||||
|     const template = await renderExtensionTemplateAsync(MODULE_NAME, 'profile', { profile: profileForDisplay }); |     const template = $(await renderExtensionTemplateAsync(MODULE_NAME, 'profile', { profile: profileForDisplay })); | ||||||
|  |     template.find('input[name="exclude"]').on('input', function () { | ||||||
|  |         const fancyName = String($(this).val()); | ||||||
|  |         const keyName = Object.entries(FANCY_NAMES).find(x => x[1] === fancyName)?.[0]; | ||||||
|  |         if (!keyName) { | ||||||
|  |             console.warn('Key not found for fancy name:', fancyName); | ||||||
|  |             return; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         if (!Array.isArray(profile.exclude)) { | ||||||
|  |             profile.exclude = []; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         const excludeState = !$(this).prop('checked'); | ||||||
|  |         if (excludeState) { | ||||||
|  |             profile.exclude.push(keyName); | ||||||
|  |         } else { | ||||||
|  |             profile.exclude.splice(profile.exclude.indexOf(keyName), 1); | ||||||
|  |         } | ||||||
|  |     }); | ||||||
|     const isNameTaken = (n) => extension_settings.connectionManager.profiles.some(p => p.name === n); |     const isNameTaken = (n) => extension_settings.connectionManager.profiles.some(p => p.name === n); | ||||||
|     const suggestedName = getUniqueName(collapseSpaces(`${profile.api ?? ''} ${profile.model ?? ''} - ${profile.preset ?? ''}`), isNameTaken); |     const suggestedName = getUniqueName(collapseSpaces(`${profile.api ?? ''} ${profile.model ?? ''} - ${profile.preset ?? ''}`), isNameTaken); | ||||||
|     const name = forceName ?? await callGenericPopup(template, POPUP_TYPE.INPUT, suggestedName, { rows: 2 }); |     const name = forceName ?? await callGenericPopup(template, POPUP_TYPE.INPUT, suggestedName, { rows: 2 }); | ||||||
| @@ -230,7 +257,13 @@ async function createConnectionProfile(forceName = null) { | |||||||
|         return null; |         return null; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     profile.name = name; |     if (Array.isArray(profile.exclude)) { | ||||||
|  |         for (const command of profile.exclude) { | ||||||
|  |             delete profile[command]; | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     profile.name = String(name); | ||||||
|     return profile; |     return profile; | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -357,6 +390,9 @@ async function renderDetailsContent(detailsContent) { | |||||||
|     const profile = extension_settings.connectionManager.profiles.find(p => p.id === selectedProfile); |     const profile = extension_settings.connectionManager.profiles.find(p => p.id === selectedProfile); | ||||||
|     if (profile) { |     if (profile) { | ||||||
|         const profileForDisplay = makeFancyProfile(profile); |         const profileForDisplay = makeFancyProfile(profile); | ||||||
|  |         if (Array.isArray(profile.exclude) && profile.exclude.length > 0) { | ||||||
|  |             profileForDisplay['Omitted Settings'] = profile.exclude.map(e => FANCY_NAMES[e]).join(', '); | ||||||
|  |         } | ||||||
|         const template = await renderExtensionTemplateAsync(MODULE_NAME, 'view', { profile: profileForDisplay }); |         const template = await renderExtensionTemplateAsync(MODULE_NAME, 'view', { profile: profileForDisplay }); | ||||||
|         detailsContent.innerHTML = template; |         detailsContent.innerHTML = template; | ||||||
|     } else { |     } else { | ||||||
| @@ -472,8 +508,8 @@ async function renderDetailsContent(detailsContent) { | |||||||
|         await eventSource.emit(event_types.CONNECTION_PROFILE_LOADED, NONE); |         await eventSource.emit(event_types.CONNECTION_PROFILE_LOADED, NONE); | ||||||
|     }); |     }); | ||||||
|  |  | ||||||
|     const renameButton = document.getElementById('rename_connection_profile'); |     const editButton = document.getElementById('edit_connection_profile'); | ||||||
|     renameButton.addEventListener('click', async () => { |     editButton.addEventListener('click', async () => { | ||||||
|         const selectedProfile = extension_settings.connectionManager.selectedProfile; |         const selectedProfile = extension_settings.connectionManager.selectedProfile; | ||||||
|         const profile = extension_settings.connectionManager.profiles.find(p => p.id === selectedProfile); |         const profile = extension_settings.connectionManager.profiles.find(p => p.id === selectedProfile); | ||||||
|         if (!profile) { |         if (!profile) { | ||||||
| @@ -481,20 +517,48 @@ async function renderDetailsContent(detailsContent) { | |||||||
|             return; |             return; | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         const newName = await Popup.show.input('Enter a new name', null, profile.name, { rows: 2 }); |         if (!Array.isArray(profile.exclude)) { | ||||||
|  |             profile.exclude = []; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         const commands = profile.mode === 'cc' ? CC_COMMANDS : TC_COMMANDS; | ||||||
|  |         const settings = commands.reduce((acc, command) => { | ||||||
|  |             const fancyName = FANCY_NAMES[command]; | ||||||
|  |             acc[fancyName] = !profile.exclude.includes(command); | ||||||
|  |             return acc; | ||||||
|  |         }, {}); | ||||||
|  |         const template = $(await renderExtensionTemplateAsync(MODULE_NAME, 'edit', { name: profile.name, settings })); | ||||||
|  |         const newName = await callGenericPopup(template, POPUP_TYPE.INPUT, profile.name); | ||||||
|  |  | ||||||
|         if (!newName) { |         if (!newName) { | ||||||
|             return; |             return; | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         if (extension_settings.connectionManager.profiles.some(p => p.name === newName)) { |         if (profile.name !== newName && extension_settings.connectionManager.profiles.some(p => p.name === newName)) { | ||||||
|             toastr.error('A profile with the same name already exists.'); |             toastr.error('A profile with the same name already exists.'); | ||||||
|             return; |             return; | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         profile.name = newName; |         const newExcludeList = template.find('input[name="exclude"]:not(:checked)').map(function () { | ||||||
|  |             return Object.entries(FANCY_NAMES).find(x => x[1] === String($(this).val()))?.[0]; | ||||||
|  |         }).get(); | ||||||
|  |  | ||||||
|  |         if (newExcludeList.length !== profile.exclude.length || !newExcludeList.every(e => profile.exclude.includes(e))) { | ||||||
|  |             profile.exclude = newExcludeList; | ||||||
|  |             for (const command of newExcludeList) { | ||||||
|  |                 delete profile[command]; | ||||||
|  |             } | ||||||
|  |             toastr.info('Press "Update" to record them into the profile.', 'Included settings list updated'); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         if (profile.name !== newName) { | ||||||
|  |             toastr.success('Connection profile renamed.'); | ||||||
|  |             profile.name = String(newName); | ||||||
|  |         } | ||||||
|  |  | ||||||
|         saveSettingsDebounced(); |         saveSettingsDebounced(); | ||||||
|         renderConnectionProfiles(profiles); |         renderConnectionProfiles(profiles); | ||||||
|         toastr.success('Connection profile renamed', '', { timeOut: 1500 }); |         await renderDetailsContent(detailsContent); | ||||||
|     }); |     }); | ||||||
|  |  | ||||||
|     /** @type {HTMLElement} */ |     /** @type {HTMLElement} */ | ||||||
|   | |||||||
| @@ -2,11 +2,18 @@ | |||||||
|     <h2 data-i18n="Creating a Connection Profile"> |     <h2 data-i18n="Creating a Connection Profile"> | ||||||
|         Creating a Connection Profile |         Creating a Connection Profile | ||||||
|     </h2> |     </h2> | ||||||
|     <ul class="justifyLeft"> |     <div class="justifyLeft flex-container flexFlowColumn flexNoGap"> | ||||||
|         {{#each profile}} |         {{#each profile}} | ||||||
|         <li><strong data-i18n="{{@key}}">{{@key}}:</strong> {{this}}</li> |         <label class="checkbox_label"> | ||||||
|  |             <input type="checkbox" value="{{@key}}" name="exclude" checked> | ||||||
|  |             <span><strong data-i18n="{{@key}}">{{@key}}:</strong> {{this}}</span> | ||||||
|  |         </label> | ||||||
|         {{/each}} |         {{/each}} | ||||||
|     </ul> |     </div> | ||||||
|  |     <small> | ||||||
|  |         <b>Hint:</b> | ||||||
|  |         <i>Click on the setting name to omit it from the profile.</i> | ||||||
|  |     </small> | ||||||
|     <h3 data-i18n="Enter a name:"> |     <h3 data-i18n="Enter a name:"> | ||||||
|         Enter a name: |         Enter a name: | ||||||
|     </h3> |     </h3> | ||||||
|   | |||||||
| @@ -13,7 +13,7 @@ | |||||||
|         <i id="view_connection_profile" class="menu_button fa-solid fa-info-circle" title="View connection profile details" data-i18n="[title]View connection profile details"></i> |         <i id="view_connection_profile" class="menu_button fa-solid fa-info-circle" title="View connection profile details" data-i18n="[title]View connection profile details"></i> | ||||||
|         <i id="create_connection_profile" class="menu_button fa-solid fa-file-circle-plus" title="Create a new connection profile" data-i18n="[title]Create a new connection profile"></i> |         <i id="create_connection_profile" class="menu_button fa-solid fa-file-circle-plus" title="Create a new connection profile" data-i18n="[title]Create a new connection profile"></i> | ||||||
|         <i id="update_connection_profile" class="menu_button fa-solid fa-save" title="Update a connection profile" data-i18n="[title]Update a connection profile"></i> |         <i id="update_connection_profile" class="menu_button fa-solid fa-save" title="Update a connection profile" data-i18n="[title]Update a connection profile"></i> | ||||||
|         <i id="rename_connection_profile" class="menu_button fa-solid fa-pencil" title="Rename a connection profile" data-i18n="[title]Rename a connection profile"></i> |         <i id="edit_connection_profile" class="menu_button fa-solid fa-pencil" title="Edit a connection profile" data-i18n="[title]Edit a connection profile"></i> | ||||||
|         <i id="reload_connection_profile" class="menu_button fa-solid fa-recycle" title="Reload a connection profile" data-i18n="[title]Reload a connection profile"></i> |         <i id="reload_connection_profile" class="menu_button fa-solid fa-recycle" title="Reload a connection profile" data-i18n="[title]Reload a connection profile"></i> | ||||||
|         <i id="delete_connection_profile" class="menu_button fa-solid fa-trash-can" title="Delete a connection profile" data-i18n="[title]Delete a connection profile"></i> |         <i id="delete_connection_profile" class="menu_button fa-solid fa-trash-can" title="Delete a connection profile" data-i18n="[title]Delete a connection profile"></i> | ||||||
|     </div> |     </div> | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user