mirror of
https://github.com/SillyTavern/SillyTavern.git
synced 2024-12-13 18:07:20 +01:00
464 lines
20 KiB
JavaScript
464 lines
20 KiB
JavaScript
/*
|
|
TODO:
|
|
*/
|
|
//const DEBUG_TONY_SAMA_FORK_MODE = true
|
|
|
|
import { getRequestHeaders, processDroppedFiles, eventSource, event_types } from '../../../script.js';
|
|
import { deleteExtension, extensionNames, getContext, installExtension, renderExtensionTemplateAsync } from '../../extensions.js';
|
|
import { POPUP_TYPE, Popup, callGenericPopup } from '../../popup.js';
|
|
import { executeSlashCommands } from '../../slash-commands.js';
|
|
import { flashHighlight, getStringHash, isValidUrl } from '../../utils.js';
|
|
export { MODULE_NAME };
|
|
|
|
const MODULE_NAME = 'assets';
|
|
const DEBUG_PREFIX = '<Assets module> ';
|
|
let previewAudio = null;
|
|
let ASSETS_JSON_URL = 'https://raw.githubusercontent.com/SillyTavern/SillyTavern-Content/main/index.json';
|
|
|
|
|
|
// DBG
|
|
//if (DEBUG_TONY_SAMA_FORK_MODE)
|
|
// ASSETS_JSON_URL = "https://raw.githubusercontent.com/Tony-sama/SillyTavern-Content/main/index.json"
|
|
let availableAssets = {};
|
|
let currentAssets = {};
|
|
|
|
//#############################//
|
|
// Extension UI and Settings //
|
|
//#############################//
|
|
|
|
function filterAssets() {
|
|
const searchValue = String($('#assets_search').val()).toLowerCase().trim();
|
|
const typeValue = String($('#assets_type_select').val());
|
|
|
|
if (typeValue === '') {
|
|
$('#assets_menu .assets-list-div').show();
|
|
$('#assets_menu .assets-list-div h3').show();
|
|
} else {
|
|
$('#assets_menu .assets-list-div h3').hide();
|
|
$('#assets_menu .assets-list-div').hide();
|
|
$(`#assets_menu .assets-list-div[data-type="${typeValue}"]`).show();
|
|
}
|
|
|
|
if (searchValue === '') {
|
|
$('#assets_menu .asset-block').show();
|
|
} else {
|
|
$('#assets_menu .asset-block').hide();
|
|
$('#assets_menu .asset-block').filter(function () {
|
|
return $(this).text().toLowerCase().includes(searchValue);
|
|
}).show();
|
|
}
|
|
}
|
|
|
|
const KNOWN_TYPES = {
|
|
'extension': 'Extensions',
|
|
'character': 'Characters',
|
|
'ambient': 'Ambient sounds',
|
|
'bgm': 'Background music',
|
|
'blip': 'Blip sounds',
|
|
};
|
|
|
|
function downloadAssetsList(url) {
|
|
updateCurrentAssets().then(function () {
|
|
fetch(url, { cache: 'no-cache' })
|
|
.then(response => response.json())
|
|
.then(json => {
|
|
|
|
availableAssets = {};
|
|
$('#assets_menu').empty();
|
|
|
|
console.debug(DEBUG_PREFIX, 'Received assets dictionary', json);
|
|
|
|
for (const i of json) {
|
|
//console.log(DEBUG_PREFIX,i)
|
|
if (availableAssets[i['type']] === undefined)
|
|
availableAssets[i['type']] = [];
|
|
|
|
availableAssets[i['type']].push(i);
|
|
}
|
|
|
|
console.debug(DEBUG_PREFIX, 'Updated available assets to', availableAssets);
|
|
// First extensions, then everything else
|
|
const assetTypes = Object.keys(availableAssets).sort((a, b) => (a === 'extension') ? -1 : (b === 'extension') ? 1 : 0);
|
|
|
|
$('#assets_type_select').empty();
|
|
$('#assets_search').val('');
|
|
$('#assets_type_select').append($('<option />', { value: '', text: 'All' }));
|
|
|
|
for (const type of assetTypes) {
|
|
const option = $('<option />', { value: type, text: KNOWN_TYPES[type] || type });
|
|
$('#assets_type_select').append(option);
|
|
}
|
|
|
|
if (assetTypes.includes('extension')) {
|
|
$('#assets_type_select').val('extension');
|
|
}
|
|
|
|
$('#assets_type_select').off('change').on('change', filterAssets);
|
|
$('#assets_search').off('input').on('input', filterAssets);
|
|
|
|
for (const assetType of assetTypes) {
|
|
let assetTypeMenu = $('<div />', { id: 'assets_audio_ambient_div', class: 'assets-list-div' });
|
|
assetTypeMenu.attr('data-type', assetType);
|
|
assetTypeMenu.append(`<h3>${KNOWN_TYPES[assetType] || assetType}</h3>`).hide();
|
|
|
|
if (assetType == 'extension') {
|
|
assetTypeMenu.append(`
|
|
<div class="assets-list-git">
|
|
To download extensions from this page, you need to have <a href="https://git-scm.com/downloads" target="_blank">Git</a> installed.<br>
|
|
Click the <i class="fa-solid fa-sm fa-arrow-up-right-from-square"></i> icon to visit the Extension's repo for tips on how to use it.
|
|
</div>`);
|
|
}
|
|
|
|
for (const i in availableAssets[assetType].sort((a, b) => a?.name && b?.name && a['name'].localeCompare(b['name']))) {
|
|
const asset = availableAssets[assetType][i];
|
|
const elemId = `assets_install_${assetType}_${i}`;
|
|
let element = $('<div />', { id: elemId, class: 'asset-download-button right_menu_button' });
|
|
const label = $('<i class="fa-fw fa-solid fa-download fa-lg"></i>');
|
|
element.append(label);
|
|
|
|
//if (DEBUG_TONY_SAMA_FORK_MODE)
|
|
// asset["url"] = asset["url"].replace("https://github.com/SillyTavern/","https://github.com/Tony-sama/"); // DBG
|
|
|
|
console.debug(DEBUG_PREFIX, 'Checking asset', asset['id'], asset['url']);
|
|
|
|
const assetInstall = async function () {
|
|
element.off('click');
|
|
label.removeClass('fa-download');
|
|
this.classList.add('asset-download-button-loading');
|
|
await installAsset(asset['url'], assetType, asset['id']);
|
|
label.addClass('fa-check');
|
|
this.classList.remove('asset-download-button-loading');
|
|
element.on('click', assetDelete);
|
|
element.on('mouseenter', function () {
|
|
label.removeClass('fa-check');
|
|
label.addClass('fa-trash');
|
|
label.addClass('redOverlayGlow');
|
|
}).on('mouseleave', function () {
|
|
label.addClass('fa-check');
|
|
label.removeClass('fa-trash');
|
|
label.removeClass('redOverlayGlow');
|
|
});
|
|
};
|
|
|
|
const assetDelete = async function () {
|
|
if (assetType === 'character') {
|
|
toastr.error('Go to the characters menu to delete a character.', 'Character deletion not supported');
|
|
await executeSlashCommands(`/go ${asset['id']}`);
|
|
return;
|
|
}
|
|
element.off('click');
|
|
await deleteAsset(assetType, asset['id']);
|
|
label.removeClass('fa-check');
|
|
label.removeClass('redOverlayGlow');
|
|
label.removeClass('fa-trash');
|
|
label.addClass('fa-download');
|
|
element.off('mouseenter').off('mouseleave');
|
|
element.on('click', assetInstall);
|
|
};
|
|
|
|
if (isAssetInstalled(assetType, asset['id'])) {
|
|
console.debug(DEBUG_PREFIX, 'installed, checked');
|
|
label.toggleClass('fa-download');
|
|
label.toggleClass('fa-check');
|
|
element.on('click', assetDelete);
|
|
element.on('mouseenter', function () {
|
|
label.removeClass('fa-check');
|
|
label.addClass('fa-trash');
|
|
label.addClass('redOverlayGlow');
|
|
}).on('mouseleave', function () {
|
|
label.addClass('fa-check');
|
|
label.removeClass('fa-trash');
|
|
label.removeClass('redOverlayGlow');
|
|
});
|
|
}
|
|
else {
|
|
console.debug(DEBUG_PREFIX, 'not installed, unchecked');
|
|
element.prop('checked', false);
|
|
element.on('click', assetInstall);
|
|
}
|
|
|
|
console.debug(DEBUG_PREFIX, 'Created element for ', asset['id']);
|
|
|
|
const displayName = DOMPurify.sanitize(asset['name'] || asset['id']);
|
|
const description = DOMPurify.sanitize(asset['description'] || '');
|
|
const url = isValidUrl(asset['url']) ? asset['url'] : '';
|
|
const title = assetType === 'extension' ? `Extension repo/guide: ${url}` : 'Preview in browser';
|
|
const previewIcon = (assetType === 'extension' || assetType === 'character') ? 'fa-arrow-up-right-from-square' : 'fa-headphones-simple';
|
|
|
|
const assetBlock = $('<i></i>')
|
|
.append(element)
|
|
.append(`<div class="flex-container flexFlowColumn flexNoGap">
|
|
<span class="asset-name flex-container alignitemscenter">
|
|
<b>${displayName}</b>
|
|
<a class="asset_preview" href="${url}" target="_blank" title="${title}">
|
|
<i class="fa-solid fa-sm ${previewIcon}"></i>
|
|
</a>
|
|
</span>
|
|
<small class="asset-description">
|
|
${description}
|
|
</small>
|
|
</div>`);
|
|
|
|
if (assetType === 'character') {
|
|
if (asset.highlight) {
|
|
assetBlock.find('.asset-name').append('<i class="fa-solid fa-sm fa-trophy"></i>');
|
|
}
|
|
assetBlock.find('.asset-name').prepend(`<div class="avatar"><img src="${asset['url']}" alt="${displayName}"></div>`);
|
|
}
|
|
|
|
assetBlock.addClass('asset-block');
|
|
|
|
assetTypeMenu.append(assetBlock);
|
|
}
|
|
assetTypeMenu.appendTo('#assets_menu');
|
|
assetTypeMenu.on('click', 'a.asset_preview', previewAsset);
|
|
}
|
|
|
|
filterAssets();
|
|
$('#assets_filters').show();
|
|
$('#assets_menu').show();
|
|
})
|
|
.catch((error) => {
|
|
// Info hint if the user maybe... likely accidently was trying to install an extension and we wanna help guide them? uwu :3
|
|
const installButton = $('#third_party_extension_button');
|
|
flashHighlight(installButton, 10_000);
|
|
toastr.info('Click the flashing button at the top right corner of the menu.', 'Trying to install a custom extension?', { timeOut: 10_000 });
|
|
|
|
// Error logged after, to appear on top
|
|
console.error(error);
|
|
toastr.error('Problem with assets URL', DEBUG_PREFIX + 'Cannot get assets list');
|
|
$('#assets-connect-button').addClass('fa-plug-circle-exclamation');
|
|
$('#assets-connect-button').addClass('redOverlayGlow');
|
|
});
|
|
});
|
|
}
|
|
|
|
function previewAsset(e) {
|
|
const href = $(this).attr('href');
|
|
const audioExtensions = ['.mp3', '.ogg', '.wav'];
|
|
|
|
if (audioExtensions.some(ext => href.endsWith(ext))) {
|
|
e.preventDefault();
|
|
|
|
if (previewAudio) {
|
|
previewAudio.pause();
|
|
|
|
if (previewAudio.src === href) {
|
|
previewAudio = null;
|
|
return;
|
|
}
|
|
}
|
|
|
|
previewAudio = new Audio(href);
|
|
previewAudio.play();
|
|
return;
|
|
}
|
|
}
|
|
|
|
function isAssetInstalled(assetType, filename) {
|
|
let assetList = currentAssets[assetType];
|
|
|
|
if (assetType == 'extension') {
|
|
const thirdPartyMarker = 'third-party/';
|
|
assetList = extensionNames.filter(x => x.startsWith(thirdPartyMarker)).map(x => x.replace(thirdPartyMarker, ''));
|
|
}
|
|
|
|
if (assetType == 'character') {
|
|
assetList = getContext().characters.map(x => x.avatar);
|
|
}
|
|
|
|
for (const i of assetList) {
|
|
//console.debug(DEBUG_PREFIX,i,filename)
|
|
if (i.includes(filename))
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
async function installAsset(url, assetType, filename) {
|
|
console.debug(DEBUG_PREFIX, 'Downloading ', url);
|
|
const category = assetType;
|
|
try {
|
|
if (category === 'extension') {
|
|
console.debug(DEBUG_PREFIX, 'Installing extension ', url);
|
|
await installExtension(url);
|
|
console.debug(DEBUG_PREFIX, 'Extension installed.');
|
|
return;
|
|
}
|
|
|
|
const body = { url, category, filename };
|
|
const result = await fetch('/api/assets/download', {
|
|
method: 'POST',
|
|
headers: getRequestHeaders(),
|
|
body: JSON.stringify(body),
|
|
cache: 'no-cache',
|
|
});
|
|
if (result.ok) {
|
|
console.debug(DEBUG_PREFIX, 'Download success.');
|
|
if (category === 'character') {
|
|
console.debug(DEBUG_PREFIX, 'Importing character ', filename);
|
|
const blob = await result.blob();
|
|
const file = new File([blob], filename, { type: blob.type });
|
|
await processDroppedFiles([file], true);
|
|
console.debug(DEBUG_PREFIX, 'Character downloaded.');
|
|
}
|
|
}
|
|
}
|
|
catch (err) {
|
|
console.log(err);
|
|
return [];
|
|
}
|
|
}
|
|
|
|
async function deleteAsset(assetType, filename) {
|
|
console.debug(DEBUG_PREFIX, 'Deleting ', assetType, filename);
|
|
const category = assetType;
|
|
try {
|
|
if (category === 'extension') {
|
|
console.debug(DEBUG_PREFIX, 'Deleting extension ', filename);
|
|
await deleteExtension(filename);
|
|
console.debug(DEBUG_PREFIX, 'Extension deleted.');
|
|
}
|
|
|
|
const body = { category, filename };
|
|
const result = await fetch('/api/assets/delete', {
|
|
method: 'POST',
|
|
headers: getRequestHeaders(),
|
|
body: JSON.stringify(body),
|
|
cache: 'no-cache',
|
|
});
|
|
if (result.ok) {
|
|
console.debug(DEBUG_PREFIX, 'Deletion success.');
|
|
}
|
|
}
|
|
catch (err) {
|
|
console.log(err);
|
|
return [];
|
|
}
|
|
}
|
|
|
|
async function openCharacterBrowser(forceDefault) {
|
|
const url = forceDefault ? ASSETS_JSON_URL : String($('#assets-json-url-field').val());
|
|
const fetchResult = await fetch(url, { cache: 'no-cache' });
|
|
const json = await fetchResult.json();
|
|
const characters = json.filter(x => x.type === 'character');
|
|
|
|
if (!characters.length) {
|
|
toastr.error('No characters found in the assets list', 'Character browser');
|
|
return;
|
|
}
|
|
|
|
const template = $(await renderExtensionTemplateAsync(MODULE_NAME, 'market', {}));
|
|
|
|
for (const character of characters.sort((a, b) => a.name.localeCompare(b.name))) {
|
|
const listElement = template.find(character.highlight ? '.contestWinnersList' : '.featuredCharactersList');
|
|
const characterElement = $(await renderExtensionTemplateAsync(MODULE_NAME, 'character', character));
|
|
const downloadButton = characterElement.find('.characterAssetDownloadButton');
|
|
const checkMark = characterElement.find('.characterAssetCheckMark');
|
|
const isInstalled = isAssetInstalled('character', character.id);
|
|
|
|
downloadButton.toggle(!isInstalled).on('click', async () => {
|
|
downloadButton.toggleClass('fa-download fa-spinner fa-spin');
|
|
await installAsset(character.url, 'character', character.id);
|
|
downloadButton.hide();
|
|
checkMark.show();
|
|
});
|
|
|
|
checkMark.toggle(isInstalled);
|
|
|
|
listElement.append(characterElement);
|
|
}
|
|
|
|
callGenericPopup(template, POPUP_TYPE.TEXT, '', { okButton: 'Close', wide: true, large: true, allowVerticalScrolling: true, allowHorizontalScrolling: false });
|
|
}
|
|
|
|
//#############################//
|
|
// API Calls //
|
|
//#############################//
|
|
|
|
async function updateCurrentAssets() {
|
|
console.debug(DEBUG_PREFIX, 'Checking installed assets...');
|
|
try {
|
|
const result = await fetch('/api/assets/get', {
|
|
method: 'POST',
|
|
headers: getRequestHeaders(),
|
|
});
|
|
currentAssets = result.ok ? (await result.json()) : {};
|
|
}
|
|
catch (err) {
|
|
console.log(err);
|
|
}
|
|
console.debug(DEBUG_PREFIX, 'Current assets found:', currentAssets);
|
|
}
|
|
|
|
|
|
//#############################//
|
|
// Extension load //
|
|
//#############################//
|
|
|
|
// This function is called when the extension is loaded
|
|
jQuery(async () => {
|
|
// This is an example of loading HTML from a file
|
|
const windowTemplate = await renderExtensionTemplateAsync(MODULE_NAME, 'window', {});
|
|
const windowHtml = $(windowTemplate);
|
|
|
|
const assetsJsonUrl = windowHtml.find('#assets-json-url-field');
|
|
assetsJsonUrl.val(ASSETS_JSON_URL);
|
|
|
|
const charactersButton = windowHtml.find('#assets-characters-button');
|
|
charactersButton.on('click', async function () {
|
|
openCharacterBrowser(false);
|
|
});
|
|
|
|
const installHintButton = windowHtml.find('.assets-install-hint-link');
|
|
installHintButton.on('click', async function () {
|
|
const installButton = $('#third_party_extension_button');
|
|
flashHighlight(installButton, 5000);
|
|
toastr.info('Click the flashing button to install extensions.', 'How to install extensions?');
|
|
});
|
|
|
|
const connectButton = windowHtml.find('#assets-connect-button');
|
|
connectButton.on('click', async function () {
|
|
const url = DOMPurify.sanitize(String(assetsJsonUrl.val()));
|
|
const rememberKey = `Assets_SkipConfirm_${getStringHash(url)}`;
|
|
const skipConfirm = localStorage.getItem(rememberKey) === 'true';
|
|
|
|
const confirmation = skipConfirm || await Popup.show.confirm('Loading Asset List', `<span>Are you sure you want to connect to the following url?</span><var>${url}</var>`, {
|
|
customInputs: [{ id: 'assets-remember', label: 'Don\'t ask again for this URL' }],
|
|
onClose: popup => {
|
|
if (popup.result) {
|
|
const rememberValue = popup.inputResults.get('assets-remember');
|
|
localStorage.setItem(rememberKey, String(rememberValue));
|
|
}
|
|
},
|
|
});
|
|
|
|
if (confirmation) {
|
|
try {
|
|
console.debug(DEBUG_PREFIX, 'Confimation, loading assets...');
|
|
downloadAssetsList(url);
|
|
connectButton.removeClass('fa-plug-circle-exclamation');
|
|
connectButton.removeClass('redOverlayGlow');
|
|
connectButton.addClass('fa-plug-circle-check');
|
|
} catch (error) {
|
|
console.error('Error:', error);
|
|
toastr.error(`Cannot get assets list from ${url}`);
|
|
connectButton.removeClass('fa-plug-circle-check');
|
|
connectButton.addClass('fa-plug-circle-exclamation');
|
|
connectButton.removeClass('redOverlayGlow');
|
|
}
|
|
}
|
|
else {
|
|
console.debug(DEBUG_PREFIX, 'Connection refused by user');
|
|
}
|
|
});
|
|
|
|
windowHtml.find('#assets_filters').hide();
|
|
$('#assets_container').append(windowHtml);
|
|
|
|
eventSource.on(event_types.OPEN_CHARACTER_LIBRARY, async (forceDefault) => {
|
|
openCharacterBrowser(forceDefault);
|
|
});
|
|
});
|