diff --git a/.gitignore b/.gitignore
index 7d48fd879..7a88d5bec 100644
--- a/.gitignore
+++ b/.gitignore
@@ -45,6 +45,7 @@ access.log
/vectors/
/cache/
public/css/user.css
+public/error/
/plugins/
/data
/default/scaffold
@@ -52,3 +53,5 @@ public/scripts/extensions/third-party
/certs
.aider*
.env
+/StartDev.bat
+
diff --git a/default/config.yaml b/default/config.yaml
index a9d38a4d0..f3e7db625 100644
--- a/default/config.yaml
+++ b/default/config.yaml
@@ -70,7 +70,7 @@ perUserBasicAuth: false
## Set to a positive number to expire session after a certain time of inactivity
## Set to 0 to expire session when the browser is closed
## Set to a negative number to disable session expiration
-sessionTimeout: 86400
+sessionTimeout: -1
# Used to sign session cookies. Will be auto-generated if not set
cookieSecret: ''
# Disable CSRF protection - NOT RECOMMENDED
@@ -133,24 +133,26 @@ whitelistImportDomains:
## headers:
## User-Agent: "Googlebot/2.1 (+http://www.google.com/bot.html)"
requestOverrides: []
-# -- EXTENSIONS CONFIGURATION --
-# Enable UI extensions
-enableExtensions: true
-# Automatically update extensions when a release version changes
-enableExtensionsAutoUpdate: true
+
+# EXTENSIONS CONFIGURATION
+extensions:
+ # Enable UI extensions
+ enabled: true
+ # Automatically update extensions when a release version changes
+ autoUpdate: true
+ models:
+ # Enables automatic model download from HuggingFace
+ autoDownload: true
+ # Additional models for extensions. Expects model IDs from HuggingFace model hub in ONNX format
+ classification: Cohee/distilbert-base-uncased-go-emotions-onnx
+ captioning: Xenova/vit-gpt2-image-captioning
+ embedding: Cohee/jina-embeddings-v2-base-en
+ speechToText: Xenova/whisper-small
+ textToSpeech: Xenova/speecht5_tts
+
# Additional model tokenizers can be downloaded on demand.
# Disabling will fallback to another locally available tokenizer.
enableDownloadableTokenizers: true
-# Extension settings
-extras:
- # Disables automatic model download from HuggingFace
- disableAutoDownload: false
- # Extra models for plugins. Expects model IDs from HuggingFace model hub in ONNX format
- classificationModel: Cohee/distilbert-base-uncased-go-emotions-onnx
- captioningModel: Xenova/vit-gpt2-image-captioning
- embeddingModel: Cohee/jina-embeddings-v2-base-en
- speechToTextModel: Xenova/whisper-small
- textToSpeechModel: Xenova/speecht5_tts
# -- OPENAI CONFIGURATION --
# A placeholder message to use in strict prompt post-processing mode when the prompt doesn't start with a user message
promptPlaceholder: "[Start a new chat]"
diff --git a/default/content/backgrounds/_black.jpg b/default/content/backgrounds/_black.jpg
index a451bc161..30ced914c 100644
Binary files a/default/content/backgrounds/_black.jpg and b/default/content/backgrounds/_black.jpg differ
diff --git a/default/content/backgrounds/_white.jpg b/default/content/backgrounds/_white.jpg
index a7c12e675..f363fbde0 100644
Binary files a/default/content/backgrounds/_white.jpg and b/default/content/backgrounds/_white.jpg differ
diff --git a/default/content/index.json b/default/content/index.json
index 6df03b78d..82387d0a9 100644
--- a/default/content/index.json
+++ b/default/content/index.json
@@ -782,5 +782,13 @@
{
"filename": "presets/context/Mistral V7.json",
"type": "context"
+ },
+ {
+ "filename": "presets/instruct/DeepSeek-V2.5.json",
+ "type": "instruct"
+ },
+ {
+ "filename": "presets/context/DeepSeek-V2.5.json",
+ "type": "context"
}
]
diff --git a/default/content/presets/context/DeepSeek-V2.5.json b/default/content/presets/context/DeepSeek-V2.5.json
new file mode 100644
index 000000000..49efaba59
--- /dev/null
+++ b/default/content/presets/context/DeepSeek-V2.5.json
@@ -0,0 +1,11 @@
+{
+ "story_string": "{{#if system}}{{system}}\n{{/if}}{{#if wiBefore}}{{wiBefore}}\n{{/if}}{{#if description}}{{description}}\n{{/if}}{{#if personality}}{{char}}'s personality: {{personality}}\n{{/if}}{{#if scenario}}Scenario: {{scenario}}\n{{/if}}{{#if wiAfter}}{{wiAfter}}\n{{/if}}{{#if persona}}{{persona}}\n{{/if}}{{trim}}\n",
+ "example_separator": "",
+ "chat_start": "",
+ "use_stop_strings": false,
+ "allow_jailbreak": false,
+ "always_force_name2": true,
+ "trim_sentences": false,
+ "single_line": false,
+ "name": "DeepSeek-V2.5"
+}
diff --git a/default/content/presets/instruct/DeepSeek-V2.5.json b/default/content/presets/instruct/DeepSeek-V2.5.json
new file mode 100644
index 000000000..7990d13c0
--- /dev/null
+++ b/default/content/presets/instruct/DeepSeek-V2.5.json
@@ -0,0 +1,22 @@
+{
+ "input_sequence": "<|User|>",
+ "output_sequence": "<|Assistant|>",
+ "first_output_sequence": "",
+ "last_output_sequence": "",
+ "system_sequence_prefix": "",
+ "system_sequence_suffix": "",
+ "stop_sequence": "",
+ "wrap": false,
+ "macro": true,
+ "names_behavior": "force",
+ "activation_regex": "",
+ "skip_examples": false,
+ "output_suffix": "<|end▁of▁sentence|>",
+ "input_suffix": "",
+ "system_sequence": "",
+ "system_suffix": "",
+ "user_alignment_message": "",
+ "last_system_sequence": "",
+ "system_same_as_user": true,
+ "name": "DeepSeek-V2.5"
+}
diff --git a/default/user.css b/default/public/css/user.css
similarity index 100%
rename from default/user.css
rename to default/public/css/user.css
diff --git a/default/public/error/forbidden-by-whitelist.html b/default/public/error/forbidden-by-whitelist.html
new file mode 100644
index 000000000..70ff71852
--- /dev/null
+++ b/default/public/error/forbidden-by-whitelist.html
@@ -0,0 +1,22 @@
+
+
+
+
+ Forbidden
+
+
+
+
Forbidden
+
+ If you are the system administrator, add your IP address to the
+ whitelist or disable whitelist mode by editing
+ config.yaml in the root directory of your installation.
+
+
+
+ Connection from {{ipDetails}} has been blocked. This attempt
+ has been logged.
+
+ If you are the system administrator, you can configure the
+ basicAuthUser credentials by editing
+ config.yaml in the root directory of your installation.
+
+
+
+
diff --git a/default/public/error/url-not-found.html b/default/public/error/url-not-found.html
new file mode 100644
index 000000000..87974145f
--- /dev/null
+++ b/default/public/error/url-not-found.html
@@ -0,0 +1,15 @@
+
+
+
+
+ Not found
+
+
+
+
Not found
+
+ The requested URL was not found on this server.
+
+ Allows using function tools.
+ Can be utilized by various extensions to provide additional functionality.
+
+
+ Sends images in prompts if the model supports it (e.g. GPT-4V, Claude 3 or Llava 13B). Use the
+
+ action on any message or the
+
+ menu to attach an image file to the chat.
+
-
+
@@ -2278,7 +2280,7 @@
Model Providers
-
-
+
Auto-Continue
@@ -5106,7 +5111,8 @@
Creator's Notes
-
+ Character details are hidden.
+
@@ -5444,7 +5450,7 @@
- - Advanced
+ - Advanced
Definitions
@@ -5734,7 +5740,7 @@
☰
-
+
@@ -5967,7 +5973,7 @@
-
+
@@ -5980,7 +5986,7 @@
-
+
@@ -5993,7 +5999,7 @@
-
+
@@ -6322,7 +6328,10 @@
-
+
+
+
+
diff --git a/public/locales/ar-sa.json b/public/locales/ar-sa.json
index 4766366db..46f8a2805 100644
--- a/public/locales/ar-sa.json
+++ b/public/locales/ar-sa.json
@@ -879,7 +879,7 @@
"popup-button-no": "لا",
"popup-button-cancel": "يلغي",
"popup-button-import": "يستورد",
- "Advanced Defininitions": "تعريفات متقدمة",
+ "Advanced Definitions": "تعريفات متقدمة",
"Prompt Overrides": "التجاوزات السريعة",
"(For Chat Completion and Instruct Mode)": "(لاستكمال الدردشة ووضع التعليمات)",
"Insert {{original}} into either box to include the respective default prompt from system settings.": "أدخل {{original}} في أي مربع لتضمين التعليمات الافتراضية المعنية من إعدادات النظام.",
@@ -1376,7 +1376,7 @@
"char_import_2": "Chub Lorebook (رابط مباشر أو معرف)",
"char_import_3": "حرف JanitorAI (رابط مباشر أو UUID)",
"char_import_4": "حرف Pygmalion.chat (رابط مباشر أو UUID)",
- "char_import_5": "حرف AICharacterCard.com (رابط مباشر أو معرف)",
+ "char_import_5": "حرف AICharacterCards.com (رابط مباشر أو معرف)",
"char_import_6": "رابط PNG المباشر (راجع",
"char_import_7": "للمضيفين المسموح بهم)",
"char_import_8": "شخصية RisuRealm (رابط مباشر)",
diff --git a/public/locales/de-de.json b/public/locales/de-de.json
index 8453123a2..a10d80e0b 100644
--- a/public/locales/de-de.json
+++ b/public/locales/de-de.json
@@ -879,7 +879,7 @@
"popup-button-no": "NEIN",
"popup-button-cancel": "Stornieren",
"popup-button-import": "Importieren",
- "Advanced Defininitions": "Erweiterte Definitionen",
+ "Advanced Definitions": "Erweiterte Definitionen",
"Prompt Overrides": "Eingabeaufforderungsüberschreibungen",
"(For Chat Completion and Instruct Mode)": "(Für Chat-Abschluss und Anweisungsmodus)",
"Insert {{original}} into either box to include the respective default prompt from system settings.": "Füge {{original}} in eines der Felder ein, um den jeweiligen Standardprompt aus den Systemeinstellungen einzuschließen.",
@@ -1376,7 +1376,7 @@
"char_import_2": "Chub Lorebook (Direktlink oder ID)",
"char_import_3": "JanitorAI-Charakter (Direktlink oder UUID)",
"char_import_4": "Pygmalion.chat-Charakter (Direktlink oder UUID)",
- "char_import_5": "AICharacterCard.com-Charakter (Direktlink oder ID)",
+ "char_import_5": "AICharacterCards.com-Charakter (Direktlink oder ID)",
"char_import_6": "Direkter PNG-Link (siehe",
"char_import_7": "für erlaubte Hosts)",
"char_import_8": "RisuRealm-Charakter (Direktlink)",
diff --git a/public/locales/es-es.json b/public/locales/es-es.json
index bbe2830fd..334a283b3 100644
--- a/public/locales/es-es.json
+++ b/public/locales/es-es.json
@@ -879,7 +879,7 @@
"popup-button-no": "No",
"popup-button-cancel": "Cancelar",
"popup-button-import": "Importar",
- "Advanced Defininitions": "Definiciones avanzadas",
+ "Advanced Definitions": "Definiciones avanzadas",
"Prompt Overrides": "Anulaciones rápidas",
"(For Chat Completion and Instruct Mode)": "(Para completar el chat y el modo de instrucción)",
"Insert {{original}} into either box to include the respective default prompt from system settings.": "Inserte {{original}} en cualquiera de las casillas para incluir las indicaciones predeterminadas respectivas de la configuración del sistema.",
@@ -1376,7 +1376,7 @@
"char_import_2": "Chub Lorebook (enlace directo o ID)",
"char_import_3": "Carácter de JanitorAI (enlace directo o UUID)",
"char_import_4": "Carácter Pygmalion.chat (enlace directo o UUID)",
- "char_import_5": "Carácter AICharacterCard.com (enlace directo o ID)",
+ "char_import_5": "Carácter AICharacterCards.com (enlace directo o ID)",
"char_import_6": "Enlace PNG directo (consulte",
"char_import_7": "para hosts permitidos)",
"char_import_8": "Personaje RisuRealm (Enlace directo)",
@@ -1439,5 +1439,112 @@
"Still have questions?": "¿Todavía tienes preguntas?",
"Join the SillyTavern Discord": "Únete al Discord de SillyTavern",
"Post a GitHub issue": "Publicar un problema en GitHub",
- "Contact the developers": "Contactar a los desarrolladores"
+ "Contact the developers": "Contactar a los desarrolladores",
+ "Prome (Visual Novel Extension)": "Prome (Extensión de Modo Waifu/Novela Visual)",
+ "Brought to you by": "Presentado por",
+ "and Prometheus.": "y Prometheus.",
+ "Enable Prome": "Iniciar Prome",
+ "Toggles Prome, VN Mode and other Prome features.": "Activa Prome, el modo VN y otras funciones de Prome.",
+ "Features marked with a": "Las funciones marcadas con un",
+ "require Prome to be enabled.": "requieren que Prome esté iniciado.",
+ "Sheld Configuration": "Configuración de interfaz (Sheld)",
+ "Hide Sheld (Message Box)": "Ocultar Sheld (cuadro de mensajes)",
+ "Hide the message box (sheld) in the ST UI.": "Ocultar el cuadro de mensajes (sheld) en la interfaz de usuario de SillyTavern.",
+ "Enable Traditional VN Mode": "Modo VN tradicional*",
+ "Only Show Last Message in Chat (Requires Prome to be enabled).": "Mostrar solo el último mensaje en el chat (Requiere que Prome esté iniciado).",
+ "Letterbox Configuration*": "Configuración de la franja negra*",
+ "Letterbox Mode": "Modo de franja negra",
+ "Select the letterbox mode for the Prome VN UI.": "Seleccione el modo de franja negra para la interfaz de usuario VN de Prome.",
+ "Horizontal Letterbox": "Franja negra horizontal",
+ "Vertical Letterbox": "Franja negra vertical",
+ "Letterbox Color": "Color de la franja negra",
+ "Select the color of the letterbox.": "Seleccione el color de la franja negra.",
+ "Letterbox Size": "Tamaño de la franja negra",
+ "Set the size of the letterbox.": "Establezca el tamaño de la franja negra.",
+ "Sprite Configuration": "Configuración de sprites",
+ "Emulate Character Card as Sprite": "[BETA] Emular tarjeta de personaje como sprite",
+ "Emulates the character card of a character to be a sprite. (Requires Prome to be enabled).": "Emula la tarjeta de personaje de un personaje para que sea un sprite. (Requiere que Prome esté iniciado).",
+ "Enable Sprite Shake": "[BETA] Sprite Shake",
+ "Shakes the character sprite when the character is speaking (Only works if Streaming is enabled in Preset Settings).": "Agita el sprite del personaje cuando el personaje está hablando (solo funciona si la transmisión está habilitada en la configuración preestablecida).",
+ "Enable Focus Mode": "Modo de enfoque",
+ "Focuses the current speaking character in chat. (Requires Prome to be enabled).": "Enfoca al personaje que está hablando en el chat. (Requiere que Prome esté iniciado).",
+ "Darken Unfocused Character Sprites": "Oscurecer sprites de personajes",
+ "Darkens non-speaking (unfocused) characters. (Requires Prome to be enabled).": "Oscurece a los personajes que no hablan (no enfocados). (Requiere que Prome esté iniciado).",
+ "Auto-Hide Sprites": "Ocultar sprites automáticamente",
+ "Auto-hides characters from the screen that haven't been in the conversation for a while up to X characters. (Requires Prome to be enabled).": "Oculta automáticamente a los personajes de la pantalla que no han estado en la conversación durante un tiempo hasta X personajes. (Requiere que Prome esté iniciado).",
+ "Max Visible Sprites": "Máximo de personajes visibles",
+ "Set the maximum number of visible sprites that appears in the VN screen.": "Establezca el número máximo de sprites visibles que aparecen en la pantalla VN.",
+ "Sprite Shadow Configuration": "Configuración de sombra de sprites",
+ "Enable Sprite Shadow": "Sombra de sprite",
+ "Adds a shadow to the character sprite.": "Agrega una sombra al sprite del personaje.",
+ "Shadow X Offset": "Desplazamiento X de la sombra",
+ "Set the X offset of the character shadow.": "Establezca el desplazamiento X de la sombra del personaje.",
+ "Shadow Y Offset": "Desplazamiento Y de la sombra",
+ "Set the Y offset of the character shadow.": "Establezca el desplazamiento Y de la sombra del personaje.",
+ "Shadow Blur": "Desenfoque de sombra",
+ "Set the blur of the character shadow.": "Establezca el desenfoque de la sombra del personaje.",
+ "Focus Mode Settings": "Configuración del modo de enfoque",
+ "Focus Mode Animation": "Animación de modo de enfoque",
+ "Select the animation for focus mode.": "Seleccione la animación para el modo de enfoque.",
+ "Focus Mode Animation Speed": "Velocidad de animación del modo de enfoque",
+ "Set the speed of the focus animation.": "Establezca la velocidad de la animación de enfoque.",
+ "User Sprite Configuration": "[BETA] Configuración de sprites de usuario",
+ "Enable User Sprite": "Sprites de usuario",
+ "Enables the ability to use a user sprite for your persona.": "Habilita la capacidad de usar un sprite de usuario para tu persona.",
+ "Sprite set": "Conjunto de sprites",
+ "Type the name of the sprite set to use for your persona. (Place your sprites in the 'characters' folder in SillyTavern).": "Escriba el nombre del conjunto de sprites que desea utilizar para su persona. (Coloque sus sprites en la carpeta 'characters' en SillyTavern).",
+ "Note: Create a sprite folder in the ": "Nota: Cree una carpeta de sprites en el ",
+ " folder of your user directory (typically 'data/default-user'). Place your expressions there.": " carpeta de su directorio de usuario (normalmente 'data/default-user'). Coloque sus expresiones allí.",
+ "Tint Configuration": "Configuración de tono",
+ "Enable Chat Tint": "Tinte de chat",
+ "Tints the chat background and/or character sprites.": "Tinta el fondo del chat y/o los sprites de personajes.",
+ "Share World Tint With Characters": "Compartir tinte mundial con personajes*",
+ "Applies the world tint to character sprites (Requires Prome to be enabled. This will override your character tint settings).": "Aplica el tinte mundial a los sprites de personajes (Requiere que Prome esté iniciado. Esto anulará la configuración de tinte de su personaje).",
+ "Tint Presets": "Preajustes de tinte",
+ "Select the tint preset to use for the Prome VN UI.": "Seleccione el preajuste de tinte que desea utilizar para la interfaz de usuario VN de Prome.",
+ "World Tint Settings": "Configuración de tinte mundial",
+ "Tints the world background.": "Tinta el fondo mundial.",
+ "Enable World Tint": "Tinte mundial",
+ "Set the strength of the world blur.": "Establezca la fuerza del desenfoque mundial.",
+ "Brightness": "Brillo",
+ "Set the brightness of the world.": "Establezca el brillo del mundo.",
+ "Contrast": "Contraste",
+ "Set the contrast of the world.": "Establezca el contraste del mundial.",
+ "Grayscale": "Escala de grises",
+ "Makes the world black and white.": "Hace que el mundo sea en blanco y negro.",
+ "Hue": "Matiz",
+ "Set the hue of the world tint.": "Establezca el matiz del tinte mundial.",
+ "Invert": "Invertir",
+ "Inverts the world colors.": "Invierte los colores del mundo.",
+ "Saturate": "Saturar",
+ "Saturates the world colors.": "Satura los colores del mundo.",
+ "Makes the world warmer in color.": "Hace que el mundo sea más cálido en color.",
+ "Character Tint Settings": "Configuración de tinte de sprites*",
+ "Enable Character Tint (Requires Prome to be enabled)": "Tinte de sprites (Requiere que Prome esté iniciado)",
+ "Set the strength of the character blur.": "Establezca la fuerza del desenfoque de personajes.",
+ "Set the brightness of the character.": "Establezca el brillo de personaje.",
+ "Set the contrast of the character.": "Establezca el contraste de personajes.",
+ "Makes the character black and white.": "Hace que el personaje sea en blanco y negro.",
+ "Set the hue of the character.": "Establezca el matiz del tinte de personajes.",
+ "Inverts the character colors.": "Invierte los colores del personaje.",
+ "Saturates the character colors.": "Satura los colores del personaje.",
+ "Makes the character warmer in color.": "Hace que el personaje sea más cálido en color.",
+ "Keybinds": "Atajos de teclado",
+ "Commands": "Comandos",
+ "Prome Keybinds": "Combinaciones de teclas de Prome",
+ "Hide/Show SillyTavern's Sheld (Message Box)": "Ocultar/Mostrar el estante de SillyTavern (cuadro de mensaje)",
+ "Prome Commands": "Comandos de Prome",
+ "Show/Hide the letterbox (black bars) in the VN UI": "Mostrar/Ocultar la franja negra (barras negras) en la interfaz VN",
+ "Toggles focus mode on character sprites": "Alterna el modo de enfoque en los sprites de personajes",
+ "Sets the focus mode animation": "Establece la animación del modo de enfoque",
+ "Toggles the defocus tint on non-speaking character sprites": "Alterna el tinte de desenfoque en los sprites de personajes que no hablan",
+ "Toggles the shake animation when a character speaks on character sprites": "Alterna la animación de sacudida cuando un personaje habla en los sprites de personajes",
+ "Toggles sprite shadows on character sprites": "Alterna las sombras de los sprites de personajes",
+ "Toggles world/character tint on the VN UI": "Alterna el tinte mundial/de personajes en la interfaz VN",
+ "Toggles world tint on the VN UI": "Alterna el tinte mundial en la interfaz VN",
+ "Toggles character tint on the VN UI": "Alterna el tinte de personajes en la interfaz VN",
+ "Toggles sharing world tint with character sprites (This will override Character Tint)": "Alterna el tinte mundial compartido con los sprites de personajes (esto anulará el tinte de personajes)",
+ "Sets the expression of the user sprite": "Establece la expresión del sprite de usuario",
+ "Sets the user sprite set to use for the user sprite": "Establece el conjunto de sprites de usuario para usar en el sprite de usuario",
+ "Toggles the user sprite on the VN UI": "Alterna el sprite de usuario en la interfaz VN"
}
diff --git a/public/locales/fr-fr.json b/public/locales/fr-fr.json
index 664f0967c..b2fc0b361 100644
--- a/public/locales/fr-fr.json
+++ b/public/locales/fr-fr.json
@@ -1,31 +1,31 @@
{
- "Favorite": "Préféré",
- "Tag": "Étiqueter",
+ "Favorite": "Favoris",
+ "Tag": "Tag",
"Duplicate": "Dupliquer",
- "Persona": "Personnage",
+ "Persona": "Persona",
"Delete": "Supprimer",
"AI Response Configuration": "Configuration de la réponse de l'IA",
"AI Configuration panel will stay open": "Le panneau de configuration de l'IA restera ouvert",
"clickslidertips": "Cliquez sur le curseur pour saisir les valeurs manuellement.",
- "MAD LAB MODE ON": "MODE LABORATOIRE MAD ACTIVÉ",
+ "MAD LAB MODE ON": "MODE MAD LAB ACTIVÉ",
"Documentation on sampling parameters": "Documentation sur les paramètres d'échantillonnage",
- "kobldpresets": "Préréglages de Kobold",
+ "kobldpresets": "Presets de Kobold",
"guikoboldaisettings": "Paramètres de l'interface utilisateur de KoboldAI",
- "Update current preset": "Mettre à jour le préréglage actuel",
- "Save preset as": "Enregistrer le préréglage sous",
- "Import preset": "Importer le préréglage",
- "Export preset": "Exporter le préréglage",
- "Restore current preset": "Restaurer le préréglage actuel",
- "Delete the preset": "Supprimer le préréglage",
- "novelaipresets": "Préréglages de NovelAI",
+ "Update current preset": "Mettre à jour le preset actuel",
+ "Save preset as": "Enregistrer le preset sous",
+ "Import preset": "Importer le preset",
+ "Export preset": "Exporter le preset",
+ "Restore current preset": "Restaurer le preset actuel",
+ "Delete the preset": "Supprimer le preset",
+ "novelaipresets": "Presets de NovelAI",
"Default": "Par défaut",
- "openaipresets": "Préréglages d'OpenAI",
- "Text Completion presets": "Préréglages de complétion de texte",
+ "openaipresets": "Presets d'OpenAI",
+ "Text Completion presets": "Presets de complétion de texte",
"AI Module": "Module IA",
"Changes the style of the generated text.": "Modifie le style du texte généré.",
"No Module": "Aucun module",
"Instruct": "Instruire",
- "Prose Augmenter": "Augmenteur de prose",
+ "Prose Augmenter": "Amélioration de prose",
"Text Adventure": "Aventure textuelle",
"response legth(tokens)": "Longueur de la réponse (en tokens)",
"Streaming": "Diffusion en continu",
@@ -33,8 +33,8 @@
"context size(tokens)": "Taille du contexte (en tokens)",
"unlocked": "Déverrouillé",
"Only enable this if your model supports context sizes greater than 8192 tokens": "Activez cela uniquement si votre modèle prend en charge des tailles de contexte supérieures à 8192 tokens",
- "Max prompt cost:": "Coût rapide maximum :",
- "Display the response bit by bit as it is generated.": "Afficher la réponse morceau par morceau au fur et à mesure de sa génération.",
+ "Max prompt cost:": "Coût maximum du prompt:",
+ "Display the response bit by bit as it is generated.": "Afficher la réponse bit par bit au fur et à mesure de sa génération.",
"When this is off, responses will be displayed all at once when they are complete.": "Lorsque cette fonction est désactivée, les réponses s'affichent toutes en une fois lorsqu'elles sont complètes.",
"Temperature": "Température",
"rep.pen": "Pénalité de répétition",
@@ -58,48 +58,45 @@
"Enable OpenAI completion streaming": "Activer le streaming de complétion OpenAI",
"Frequency Penalty": "Pénalité de fréquence",
"Presence Penalty": "Pénalité de présence",
- "Count Penalty": "Pénalité de décompte",
"Top K": "Top K",
"Top P": "Top P",
"Repetition Penalty": "Pénalité de répétition",
"Min P": "Min P",
"Top A": "Top A",
- "Quick Prompts Edit": "Édition rapide des invitations",
+ "Quick Prompts Edit": "Édition rapide des prompts",
"Main": "Principal",
- "NSFW": "NSFW",
- "Jailbreak": "Jailbreak",
- "Utility Prompts": "Invitations utilitaires",
+ "Utility Prompts": "Prompts utilitaires",
"Impersonation prompt": "Prompt d'usurpation",
- "Restore default prompt": "Restaurer l'instruction par défaut",
+ "Restore default prompt": "Restaurer le prompt par défaut",
"Prompt that is used for Impersonation function": "Prompt utilisée pour la fonction d'usurpation",
- "World Info Format Template": "Modèle de format d'informations mondiales",
+ "World Info Format Template": "Modèle de format des World Info",
"Restore default format": "Restaurer le format par défaut",
- "Wraps activated World Info entries before inserting into the prompt.": "Encapsule les entrées World Info activées avant de les insérer dans l'invite.",
+ "Wraps activated World Info entries before inserting into the prompt.": "Encapsule les entrées World Info activées avant de les insérer dans le prompt.",
"scenario_format_template_part_1": "Utiliser",
"scenario_format_template_part_2": "pour marquer un endroit où le contenu est inséré.",
"Scenario Format Template": "Modèle de format de scénario",
"Personality Format Template": "Modèle de format de personnalité",
- "Group Nudge Prompt Template": "Modèle d'invite de groupe Nudge",
+ "Group Nudge Prompt Template": "Modèle de prompt de groupe Nudge",
"Sent at the end of the group chat history to force reply from a specific character.": "Envoyé à la fin de l'historique des discussions de groupe pour forcer la réponse d'un personnage spécifique.",
"New Chat": "Nouvelle discussion",
- "Restore new chat prompt": "Restaurer une nouvelle invite de discussion",
+ "Restore new chat prompt": "Restaurer le prompt de nouvelle discussion",
"Set at the beginning of the chat history to indicate that a new chat is about to start.": "Placez-le au début de l'historique de discussion pour indiquer qu'une nouvelle discussion est sur le point de démarrer.",
"New Group Chat": "Nouvelle discussion de groupe",
- "Restore new group chat prompt": "Restaurer l'invite par défaut",
+ "Restore new group chat prompt": "Restaurer le prompt pour les nouvelle discussion de groupe",
"Set at the beginning of the chat history to indicate that a new group chat is about to start.": "Défini au début de l'historique des discussions pour indiquer qu'une nouvelle discussion de groupe est sur le point de démarrer.",
"New Example Chat": "Nouvel exemple de discussion",
- "Set at the beginning of Dialogue examples to indicate that a new example chat is about to start.": "Défini au début des exemples de dialogue pour indiquer qu'un nouvel exemple de discussion est sur le point de démarrer.",
+ "Set at the beginning of Dialogue examples to indicate that a new example chat is about to start.": "Défini au début des exemples de dialogue pour indiquer qu'un nouvel exemple est sur le point de démarrer.",
"Continue nudge": "Continuer le coup de pouce",
"Set at the end of the chat history when the continue button is pressed.": "Défini à la fin de l’historique des discussions lorsque vous appuyez sur le bouton Continuer.",
"Replace empty message": "Remplacer le message vide",
"Send this text instead of nothing when the text box is empty.": "Envoyer ce texte à la place de rien lorsque la zone de texte est vide.",
"Seed": "Graine",
"Set to get deterministic results. Use -1 for random seed.": "Réglé pour obtenir des résultats déterministes. Utilisez -1 pour les graines aléatoires.",
- "Temperature controls the randomness in token selection": "La température contrôle l'aléatoire dans la sélection des tokens:\n- Une température basse (<1.0) entraîne un texte plus prévisible, en donnant la priorité aux tokens à forte probabilité.\n- Une température élevée (>1.0) favorise la créativité et la diversité de sortie, en donnant plus de chances aux tokens à faible probabilité.\nRéglez la valeur à 1.0 pour les probabilités d'origine.",
+ "Temperature controls the randomness in token selection": "La température contrôle l'aléatoire dans la sélection des tokens:\n- Une température basse (<1.0) entraîne un texte plus prévisible, en donnant la priorité aux tokens à forte probabilité.\n- Une température élevée (>1.0) favorise la créativité et la diversité, en donnant plus de chances aux tokens à faible probabilité.\nRéglez la valeur à 1.0 pour les probabilités d'origine.",
"Top_K_desc": "Top K définit une quantité maximale de tokens les plus fréquents qui peuvent être sélectionnés.",
"Top_P_desc": "Top P (alias échantillonnage du noyau) regroupe tous les tokens supérieurs nécessaires pour atteindre un pourcentage spécifique.\nAutrement dit, si les deux premiers tokens représentent 25 % et que Top-P est de 0,50, seuls ces deux tokens sont considérés.\nRéglez la valeur à 1.0 pour la désactiver.",
- "Typical P": "P typique",
- "Typical_P_desc": "L'échantillonnage P typique privilégie les tokens en fonction de leur écart par rapport à l'entropie moyenne de l'ensemble.\nLes tokens dont la probabilité cumulée est proche du seuil spécifié (par exemple, 0.5) sont conservés, ce qui distingue ceux contenant une information moyenne.\nRéglez la valeur à 1.0 pour la désactiver.",
+ "Typical P": "P Typique",
+ "Typical_P_desc": "L'échantillonnage P Typique privilégie les tokens en fonction de leur écart par rapport à l'entropie moyenne de l'ensemble.\nLes tokens dont la probabilité cumulée est proche du seuil spécifié (par exemple, 0.5) sont conservés, ce qui distingue ceux contenant une information moyenne.\nRéglez la valeur à 1.0 pour la désactiver.",
"Min_P_desc": "Min P définit une probabilité minimale de base. Elle est optimisée en fonction de la probabilité du token supérieur.\nSi la probabilité du token supérieur est de 80 % et que Min P est de 0.1, seuls les tokens avec une probabilité supérieure à 8 % sont considérés.\nRéglez la valeur à 0 pour la désactiver.",
"Top_A_desc": "Top A définit un seuil pour la sélection des tokens en fonction du carré de la probabilité du token le plus élevé.\nSi Top A est de 0.2 et que la probabilité du token le plus élevé est de 50 %, les tokens avec une probabilité inférieure à 5 % sont exclus (0.2 * 0.5^2).\nRéglez la valeur à 0 pour la désactiver.",
"Tail_Free_Sampling_desc": "Échantillonnage sans queue (TFS) recherche les tokens de queue ayant une faible probabilité dans la distribution,\n en analysant le taux de changement des probabilités des tokens à l'aide de dérivées. Les tokens sont conservés jusqu'au seuil (par exemple, 0.3), en fonction de la dérivée seconde uniforme.\nPlus la valeur se rapproche de 0, plus le nombre de tokens rejetés augmente. Réglez la valeur à 1.0 pour la désactiver.",
@@ -115,8 +112,8 @@
"Ban_EOS_Token_desc": "Interdisez le jeton de fin de séquence (EOS) avec KoboldCpp (et éventuellement aussi d'autres jetons avec KoboldAI).\rIdéal pour l'écriture d'histoires, mais ne doit pas être utilisé pour le mode chat et instruction.",
"GBNF Grammar": "Grammaire GBNF",
"Type in the desired custom grammar": "Saisissez la grammaire personnalisée souhaitée",
- "Samplers Order": "Ordre des échantillons",
- "Samplers will be applied in a top-down order. Use with caution.": "Les échantillons seront appliqués dans un ordre de haut en bas. Utilisez avec prudence.",
+ "Samplers Order": "Ordre des échantillonneurs",
+ "Samplers will be applied in a top-down order. Use with caution.": "Les échantillonneurs seront appliqués dans un ordre de haut en bas. Utilisez avec prudence.",
"Tail Free Sampling": "Échantillonnage sans queue",
"Load koboldcpp order": "Charger l'ordre koboldcpp",
"Preamble": "Préambule",
@@ -127,9 +124,9 @@
"Add": "Ajouter",
"Helps to ban or reenforce the usage of certain words": "Aide à interdire ou à renforcer l'utilisation de certains mots",
"CFG Scale": "Échelle CFG",
- "Negative Prompt": "Indication négative",
+ "Negative Prompt": "Prompt négatif",
"Add text here that would make the AI generate things you don't want in your outputs.": "Ajoutez ici du texte qui ferait générer à l'IA des choses que vous ne voulez pas dans vos sorties.",
- "Used if CFG Scale is unset globally, per chat or character": "Utilisé si l'échelle CFG n'est pas définie globalement, par chat ou par caractère.",
+ "Used if CFG Scale is unset globally, per chat or character": "Utilisé si l'échelle CFG n'est pas définie globalement, par chat ou par personnage.",
"Mirostat Tau": "Tau Mirostat",
"Mirostat LR": "Mirostat LR",
"Min Length": "Longueur minimale",
@@ -143,13 +140,13 @@
"Customize displayed samplers or add custom samplers.": "Personnalisez les échantillonneurs affichés ou ajoutez des échantillonneurs personnalisés.",
"Epsilon Cutoff": "Coupure epsilon",
"Epsilon cutoff sets a probability floor below which tokens are excluded from being sampled": "La coupure epsilon définit un seuil de probabilité en dessous duquel les tokens sont exclus de l'échantillonnage.\nEn unités 1e-4; la valeur appropriée est 3. Réglez-la à 0 pour la désactiver.",
- "Eta Cutoff": "Coupure eta",
+ "Eta Cutoff": "Coupure Eta",
"Eta_Cutoff_desc": "Le seuil Eta est le principal paramètre de la technique d'échantillonnage Eta spéciale.
En unités de 1e-4 ; une valeur raisonnable est 3.
Réglez sur 0 pour désactiver.
Voir l'article Truncation Sampling as Language Model Desmoothing par Hewitt et al. (2022) pour plus de détails.",
- "rep.pen decay": "Carie du stylo de représentation",
+ "rep.pen decay": "Déclin de la pénalité de répétition",
"Encoder Rep. Pen.": "Pénalité de répétition de l'encodeur",
"No Repeat Ngram Size": "Taille de n-gramme sans répétition",
"Skew": "Fausser",
- "Max Tokens Second": "Nombre maximum de jetons par seconde",
+ "Max Tokens Second": "Nombre maximum de token par seconde",
"Smooth Sampling": "Échantillonnage fluide",
"Smooth_Sampling_desc": "Vous permet d'utiliser des transformations quadratiques/cubiques pour ajuster la distribution. Les valeurs inférieures du facteur de lissage seront plus créatives, généralement entre 0,2 et 0,3 comme point idéal (en supposant que la courbe = 1). Des valeurs de courbe de lissage plus élevées rendront la courbe plus raide, ce qui punira plus agressivement les choix à faible probabilité. La courbe 1,0 équivaut à utiliser uniquement le facteur de lissage.",
"Smoothing Factor": "Facteur de lissage",
@@ -164,7 +161,7 @@
"Allowed Length": "Longueur autorisée",
"Penalty Range": "Fourchette de pénalité",
"DRY_Sequence_Breakers_desc": "Jetons sur lesquels la correspondance de séquence n'est pas poursuivie. Spécifié sous la forme d'une liste de chaînes entre guillemets séparées par des virgules.",
- "Sequence Breakers": "Disjoncteurs de séquence",
+ "Sequence Breakers": "Interrupteurs de séquence",
"JSON-serialized array of strings.": "Tableau de chaînes sérialisées JSON.",
"Dynamic Temperature": "Température dynamique",
"Scale Temperature dynamically per token, based on the variation of probabilities": "Échelonnez dynamiquement la température par token, en fonction de la variation des probabilités.",
@@ -178,46 +175,41 @@
"Mirostat Eta": "Eta Mirostat",
"Learning rate of Mirostat": "Taux d'apprentissage de Mirostat.",
"Beam search": "Recherche par faisceau",
- "Helpful tip coming soon.": "Conseil utile à venir.",
- "Number of Beams": "Nombre de faisceaux",
"Length Penalty": "Pénalité de longueur",
"Early Stopping": "Arrêt précoce",
"Contrastive search": "Recherche contrastive",
"Penalty Alpha": "Alpha de pénalité",
- "Strength of the Contrastive Search regularization term. Set to 0 to disable CS": "Force du terme de régularisation de la recherche contrastive. Réglez la valeur à 0 pour désactiver CS.",
+ "Strength of the Contrastive Search regularization term. Set to 0 to disable CS": "Force du terme de régularisation de la recherche contrastive. Réglez la valeur à 0 pour la désactiver.",
"Do Sample": "Faire une échantillon",
"Add BOS Token": "Ajouter le token BOS",
"Add the bos_token to the beginning of prompts. Disabling this can make the replies more creative": "Ajoutez le token BOS au début des prompts. Désactiver cela peut rendre les réponses plus créatives",
"Ban the eos_token. This forces the model to never end the generation prematurely": "Interdire le token EOS. Cela force le modèle à ne jamais terminer la génération prématurément",
- "Ignore EOS Token": "Ignorer le jeton EOS",
- "Ignore the EOS Token even if it generates.": "Ignorez le jeton EOS même s'il est généré.",
+ "Ignore EOS Token": "Ignorer le token EOS",
+ "Ignore the EOS Token even if it generates.": "Ignorez le token EOS même s'il est généré.",
"Skip Special Tokens": "Ignorer les tokens spéciaux",
"Temperature Last": "Température en dernier",
"Temperature_Last_desc": "Utilisez le réglage de température en dernier. Cela est généralement logique.\nLorsqu'il est activé : une sélection de tokens potentiels est d'abord effectuée, puis la température est appliquée pour corriger leurs probabilités relatives (techniquement, les log-likelihoods).\nLorsqu'il est désactivé : la température est d'abord appliquée pour corriger les probabilités relatives de tous les tokens, puis une sélection de tokens potentiels est effectuée parmi eux.\nDésactivez la température en dernier.",
"Speculative Ngram": "Ngramme spéculatif",
"Use a different speculative decoding method without a draft model": "Utilisez une méthode de décodage spéculative différente sans projet de modèle.\rIl est préférable d’utiliser une ébauche de modèle. Le ngram spéculatif n’est pas aussi efficace.",
- "Spaces Between Special Tokens": "Espaces entre les jetons spéciaux",
+ "Spaces Between Special Tokens": "Espaces entre les tokens spéciaux",
"LLaMA / Mistral / Yi models only": "Modèles LLaMA / Mistral / Yi uniquement. Assurez-vous de sélectionner d'abord l'analyste approprié.\nLes chaînes de caractères ne doivent pas apparaître dans les résultats.\nUne chaîne par ligne. Texte ou [identifiants de tokens].\nDe nombreux tokens commencent par un espace. Utilisez un compteur de tokens si vous n'êtes pas sûr.",
- "Example: some text [42, 69, 1337]": "Exemple:\nun certain texte\n[42, 69, 1337]",
+ "Example: some text [42, 69, 1337]": "Exemple:\nun texte\n[42, 69, 1337]",
"Classifier Free Guidance. More helpful tip coming soon": "Guidance gratuite du classificateur. Des conseils plus utiles arrivent bientôt.",
"Scale": "Échelle",
"JSON Schema": "Schéma JSON",
"Type in the desired JSON schema": "Tapez le schéma JSON souhaité",
"Grammar String": "Chaîne de grammaire",
"GBNF or EBNF, depends on the backend in use. If you're using this you should know which.": "GBNF ou EBNF dépend du backend utilisé. Si vous l'utilisez, vous devez savoir lequel.",
- "Top P & Min P": "P supérieur et P minimal",
+ "Top P & Min P": "Top P et Min P",
"Load default order": "Charger l'ordre par défaut",
- "llama.cpp only. Determines the order of samplers. If Mirostat mode is not 0, sampler order is ignored.": "lama.cpp uniquement. Détermine l’ordre des échantillonneurs. Si le mode Mirostat n'est pas 0, l'ordre de l'échantillonneur est ignoré.",
+ "llama.cpp only. Determines the order of samplers. If Mirostat mode is not 0, sampler order is ignored.": "lama.cpp uniquement. Détermine l’ordre des échantillonneurs. Si le mode Mirostat n'est pas à 0, l'ordre de l'échantillonneur est ignoré.",
"Sampler Priority": "Priorité de l'échantillonneur",
"Ooba only. Determines the order of samplers.": "Ooba uniquement. Détermine l'ordre des échantillonneurs.",
"Character Names Behavior": "Comportement des noms de personnages",
- "Helps the model to associate messages with characters.": "Aide le modèle à associer des messages à des personnages.",
+ "Helps the model to associate messages with characters.": "Aide le modèle à associer les messages avec les personnages.",
"None": "Aucun",
- "character_names_default": "Sauf pour les groupes et les personnages passés. Sinon, assurez-vous de fournir des noms dans l'invite.",
- "Don't add character names.": "N'ajoutez pas de noms de personnages.",
- "Completion": "Objet d'achèvement",
+ "character_names_default": "Sauf pour les groupes et les personnages passés. Sinon, assurez-vous de fournir des noms dans le prompt.",
"character_names_completion": "Des restrictions s'appliquent : uniquement les caractères alphanumériques latins et les traits de soulignement. Ne fonctionne pas pour toutes les sources, notamment : Claude, MistralAI, Google.",
- "Add character names to completion objects.": "Ajoutez des noms de personnages aux objets de complétion.",
"Message Content": "Contenu du message",
"Prepend character names to message contents.": "Ajoutez les noms de caractères au contenu du message.",
"Continue Postfix": "Continuer Postfix",
@@ -235,53 +227,45 @@
"Combines consecutive system messages into one (excluding example dialogues). May improve coherence for some models.": "Combine les messages système consécutifs en un seul (à l'exclusion des dialogues d'exemple). Peut améliorer la cohérence pour certains modèles.",
"Enable function calling": "Activer l'appel de fonction",
"Send inline images": "Envoyer des images en ligne",
- "image_inlining_hint_1": "Envoie des images dans les invites si le modèle le prend en charge (par exemple GPT-4V, Claude 3 ou Llava 13B).\nUtilisez le",
+ "image_inlining_hint_1": "Envoie des images dans les prompts si le modèle le prend en charge (par exemple GPT-4V, Claude 3 ou Llava 13B).\nUtilisez le",
"image_inlining_hint_2": "action sur n'importe quel message ou le",
"image_inlining_hint_3": "menu pour joindre un fichier image au chat.",
"Inline Image Quality": "Qualité d'image en ligne",
"openai_inline_image_quality_auto": "Auto",
"openai_inline_image_quality_low": "Faible",
"openai_inline_image_quality_high": "Haut",
- "Use AI21 Tokenizer": "Utiliser le tokeniseur AI21",
- "Use the appropriate tokenizer for Jurassic models, which is more efficient than GPT's.": "Utilisez le tokenizer approprié pour les modèles Jurassic, qui est plus efficace que celui de GPT.",
- "Use Google Tokenizer": "Utiliser le tokenizer Google",
- "Use the appropriate tokenizer for Google models via their API. Slower prompt processing, but offers much more accurate token counting.": "Utilisez le tokenizer approprié pour les modèles Google via leur API. Traitement des invitations plus lent, mais offre un décompte de jetons beaucoup plus précis.",
- "Use system prompt": "Utiliser l'invite système",
- "(Gemini 1.5 Pro/Flash only)": "(Gemini 1.5 Pro/Flash uniquement)",
+ "Use system prompt": "Utiliser le prompt système",
"Merges_all_system_messages_desc_1": "Fusionne tous les messages système jusqu'au premier message avec un rôle non-système et les envoie dans un",
"Merges_all_system_messages_desc_2": "champ.",
"Assistant Prefill": "Pré-remplissage de l'assistant",
"Start Claude's answer with...": "Commencer la réponse de Claude par...",
"Assistant Impersonation Prefill": "Pré-remplir l'usurpation d'identité de l'assistant",
- "Use system prompt (Claude 2.1+ only)": "Utiliser l'invite système (uniquement Claude 2.1+)",
- "Send the system prompt for supported models. If disabled, the user message is added to the beginning of the prompt.": "Envoyer l'invite système pour les modèles pris en charge. Si désactivé, le message de l'utilisateur est ajouté au début de l'invite.",
- "User first message": "Premier message de l'utilisateur",
- "Restore User first message": "Restaurer le premier message de l'utilisateur",
- "Human message": "Message humain, instruction, etc.\nN'ajoute rien lorsqu'il est vide, c'est-à-dire nécessite une nouvelle invite avec le rôle « utilisateur ».",
- "New preset": "Nouveau préréglage",
- "Delete preset": "Supprimer le préréglage",
- "View / Edit bias preset": "Afficher/Modifier le préréglage de biais",
+ "Use system prompt (Claude 2.1+ only)": "Utiliser le prompt système (uniquement Claude 2.1+)",
+ "Send the system prompt for supported models. If disabled, the user message is added to the beginning of the prompt.": "Envoyer le prompt système pour les modèles pris en charge. Si désactivé, le message de l'utilisateur est ajouté au début du prompt.",
+ "New preset": "Nouveau preset",
+ "Delete preset": "Supprimer le preset",
+ "View / Edit bias preset": "Afficher/Modifier le preset de biais",
"Add bias entry": "Ajouter une entrée de biais",
- "Most tokens have a leading space.": "La plupart des jetons ont un espace de début.",
+ "Most tokens have a leading space.": "La plupart des tokens sont précédés d'un espace.",
"API Connections": "Connexions API",
- "Text Completion": "Achèvement du texte",
- "Chat Completion": "Achèvement du chat",
+ "Text Completion": "Complétion de texte",
+ "Chat Completion": "Complétion de chat",
"NovelAI": "NovelAI",
"AI Horde": "AI Horde",
"KoboldAI": "KoboldAI",
"Avoid sending sensitive information to the Horde.": "Évitez d'envoyer des informations sensibles à la Horde.",
"Review the Privacy statement": "Examiner la déclaration de confidentialité",
- "Register a Horde account for faster queue times": "Inscrivez-vous à un compte Horde pour des temps d'attente plus courts",
- "Learn how to contribute your idle GPU cycles to the Horde": "Apprenez comment contribuer vos cycles GPU inactifs à la Horde",
+ "Register a Horde account for faster queue times": "Enregistrez un compte Horde pour accélérer les files d'attente",
+ "Learn how to contribute your idle GPU cycles to the Horde": "Découvrez comment contribuer à la Horde avec vos cycles GPU inactifs.",
"Adjust context size to worker capabilities": "Ajuster la taille du contexte aux capacités des travailleurs",
"Adjust response length to worker capabilities": "Ajuster la longueur de la réponse aux capacités des travailleurs",
"Can help with bad responses by queueing only the approved workers. May slowdown the response time.": "Peut aider avec les mauvaises réponses en mettant en file d'attente uniquement les travailleurs approuvés. Peut ralentir le temps de réponse.",
"Trusted workers only": "Travailleurs de confiance uniquement",
"API key": "Clé API",
- "Get it here:": "Obtenez-le ici :",
+ "Get it here:": "Obtenez-la ici :",
"Register": "S'inscrire",
"View my Kudos": "Voir mes Kudos",
- "Enter": "Entrer",
+ "Enter": "Entrez",
"to use anonymous mode.": "pour utiliser le mode anonyme.",
"Clear your API key": "Effacer votre clé API",
"For privacy reasons, your API key will be hidden after you reload the page.": "Pour des raisons de confidentialité, votre clé API sera masquée après le rechargement de la page.",
@@ -291,7 +275,7 @@
"Not connected...": "Non connecté...",
"API url": "URL de l'API",
"Example: http://127.0.0.1:5000/api ": "Exemple : http://127.0.0.1:5000/api",
- "Connect": "Se connecter",
+ "Connect": "Connection",
"Cancel": "Annuler",
"Novel API key": "Clé API de NovelAI",
"Get your NovelAI API Key": "Obtenez votre clé API NovelAI",
@@ -299,7 +283,6 @@
"Novel AI Model": "Modèle NovelAI",
"No connection...": "Pas de connexion...",
"API Type": "Type d'API",
- "Default (completions compatible)": "Par défaut [OpenAI /complétions compatibles : oobabooga, LM Studio, etc.]",
"TogetherAI API Key": "Clé API de TogetherAI",
"TogetherAI Model": "Modèle TogetherAI",
"-- Connect to the API --": "-- Se connecter à l'API --",
@@ -338,25 +321,23 @@
"Authorize": "Autoriser",
"Get your OpenRouter API token using OAuth flow. You will be redirected to openrouter.ai": "Obtenez votre jeton API OpenRouter en utilisant le flux OAuth. Vous serez redirigé vers openrouter.ai",
"Bypass status check": "Contourner la vérification de l'état",
- "Chat Completion Source": "Source de complétion de la conversation",
- "Reverse Proxy": "Proxy inverse",
- "Proxy Presets": "Préréglages proxy",
+ "Chat Completion Source": "Source de complétion de chat",
+ "Reverse Proxy": "Reverse Proxy",
+ "Proxy Presets": "Presets proxy",
"Saved addresses and passwords.": "Adresses et mots de passe enregistrés.",
"Save Proxy": "Enregistrer le proxy",
"Delete Proxy": "Supprimer le proxy",
"Proxy Name": "Nom du proxy",
- "This will show up as your saved preset.": "Cela apparaîtra comme votre préréglage enregistré.",
+ "This will show up as your saved preset.": "Cela apparaîtra comme votre preset enregistré.",
"Proxy Server URL": "URL du serveur proxy",
"Alternative server URL (leave empty to use the default value).": "URL du serveur alternatif (laissez vide pour utiliser la valeur par défaut).",
- "Remove your real OAI API Key from the API panel BEFORE typing anything into this box": "Supprimez votre véritable clé API OAI du panneau API AVANT de taper quoi que ce soit dans cette boîte",
- "We cannot provide support for problems encountered while using an unofficial OpenAI proxy": "Nous ne pouvons pas fournir de support pour les problèmes rencontrés lors de l'utilisation d'un proxy OpenAI non officiel",
"Doesn't work? Try adding": "Ça ne marche pas ? Essayez d'ajouter",
"at the end!": "à la fin!",
"Proxy Password": "Mot de passe proxy",
"Will be used as a password for the proxy instead of API key.": "Sera utilisé comme mot de passe pour le proxy au lieu de la clé API.",
"Peek a password": "Jetez un œil à un mot de passe",
"OpenAI API key": "Clé API OpenAI",
- "View API Usage Metrics": "Afficher les mesures d'utilisation de l'API",
+ "View API Usage Metrics": "Afficher les statistiques d'utilisation de l'API",
"Follow": "Suivre",
"these directions": "ces instructions",
"to get your OpenAI API key.": "pour obtenir votre clé API OpenAI.",
@@ -374,7 +355,6 @@
"Context Size": "Taille du contexte",
"Group by vendors": "Regrouper par fournisseurs",
"Group by vendors Description": "Placez les modèles OpenAI dans un groupe, les modèles Anthropic dans un autre groupe, etc. Peut être combiné avec le tri.",
- "Allow fallback routes": "Autoriser les itinéraires de secours",
"Allow fallback routes Description": "Le modèle alternatif est automatiquement sélectionné si le modèle choisi ne peut pas répondre à votre demande.",
"Scale API Key": "Clé API Scale",
"Clear your cookie": "Effacer vos cookies",
@@ -387,15 +367,15 @@
"MistralAI Model": "Modèle MistralAI",
"Groq API Key": "Clé API Groq",
"Groq Model": "Modèle Groq",
- "Perplexity API Key": "Clé API de perplexité",
- "Perplexity Model": "Modèle de perplexité",
- "Cohere API Key": "Cohérer la clé API",
- "Cohere Model": "Modèle de cohérence",
+ "Perplexity API Key": "Clé API Perplexity",
+ "Perplexity Model": "Modèle Perplexity",
+ "Cohere API Key": "Clé API Cohere",
+ "Cohere Model": "Modèle Cohere",
"Custom Endpoint (Base URL)": "Point de terminaison personnalisé (URL de base)",
"Custom API Key": "Clé API personnalisée",
"Available Models": "Modèles disponibles",
- "Prompt Post-Processing": "Post-traitement rapide",
- "Applies additional processing to the prompt before sending it to the API.": "Applique un traitement supplémentaire à l'invite avant de l'envoyer à l'API.",
+ "Prompt Post-Processing": "Post-traitement de prompt",
+ "Applies additional processing to the prompt before sending it to the API.": "Applique un traitement supplémentaire au prompt avant de l'envoyer à l'API.",
"Verifies your API connection by sending a short test message. Be aware that you'll be credited for it!": "Vérifie votre connexion API en envoyant un court message de test. Soyez conscient que vous serez crédité pour cela !",
"Test Message": "Message de test",
"Auto-connect to Last Server": "Connexion automatique au dernier serveur",
@@ -405,93 +385,65 @@
"AI Response Formatting": "Formatage de la réponse de l'IA",
"Advanced Formatting": "Formatage avancé",
"Context Template": "Modèle de contexte",
- "Auto-select this preset for Instruct Mode": "Sélectionner automatiquement ce préréglage pour le mode Instruct",
"Story String": "Chaîne d'histoire",
"Example Separator": "Séparateur d'exemple",
"Chat Start": "Début de la discussion",
"Add Chat Start and Example Separator to a list of stopping strings.": "Ajoutez Chat Start et Sample Separator à une liste de chaînes d’arrêt.",
- "Use as Stop Strings": "Utiliser comme chaînes d'arrêt",
- "context_allow_jailbreak": "Inclut Jailbreak à la fin de l'invite, s'il est défini dans la carte du personnage ET « Prefer Char. Le jailbreak'' est activé.\nCECI N'EST PAS RECOMMANDÉ POUR LES MODÈLES DE COMPLÉTION DE TEXTE, PEUT ENTRAÎNER UNE MAUVAIS SORTIE.",
- "Allow Jailbreak": "Autoriser le jailbreak",
- "Context Order": "Ordre contextuel",
- "Summary": "Résumé",
- "Author's Note": "Note de l'auteur",
- "Example Dialogues": "Exemples de dialogues",
- "Hint": "Indice:",
- "In-Chat Position not affected": "Les commandes de résumé et de note de l'auteur ne sont affectées que lorsqu'elles n'ont pas de position définie dans le chat.",
- "Instruct Mode": "Mode d'instruction",
+ "Author's Note": "Note d'auteur",
"Enabled": "Activé",
"instruct_bind_to_context": "Si cette option est activée, les modèles de contexte seront automatiquement sélectionnés en fonction du nom du modèle d'instruction sélectionné ou par préférence.",
- "Bind to Context": "Lier au contexte",
- "Presets": "Préréglages",
- "Auto-select this preset on API connection": "Sélectionner automatiquement ce préréglage lors de la connexion à l'API",
"Activation Regex": "Regex d'activation",
"Wrap Sequences with Newline": "Envelopper les séquences avec un saut de ligne",
- "Replace Macro in Sequences": "Remplacer le macro dans les séquences",
+ "Replace Macro in Sequences": "Remplacer la macro dans les séquences",
"Skip Example Dialogues Formatting": "Ignorer le formatage des dialogues d'exemple",
"Include Names": "Inclure les noms",
- "Force for Groups and Personas": "Forcer pour les groupes et les personnages",
- "System Prompt": "Promp système",
- "Instruct Mode Sequences": "Séquences du mode d'instruction",
- "System Prompt Wrapping": "Enveloppement d’invite système",
- "Inserted before a System prompt.": "Inséré avant une invite système.",
- "System Prompt Prefix": "Préfixe d'invite système",
- "Inserted after a System prompt.": "Inséré après une invite système.",
- "System Prompt Suffix": "Suffixe d'invite système",
- "Chat Messages Wrapping": "Enveloppement des messages de discussion",
- "Inserted before a User message and as a last prompt line when impersonating.": "Inséré avant un message utilisateur et comme dernière ligne d'invite lors de l'usurpation d'identité.",
- "User Message Prefix": "Préfixe du message utilisateur",
+ "System Prompt": "Prompt système",
+ "Inserted before a System prompt.": "Inséré avant un prompt système.",
+ "System Prompt Prefix": "Préfixe de prompt système",
+ "Inserted after a System prompt.": "Inséré après un prompt système.",
+ "System Prompt Suffix": "Suffixe de prompt système",
+ "Inserted before a User message and as a last prompt line when impersonating.": "Inséré avant un message utilisateur et comme dernière ligne du prompt lors de l'usurpation d'identité.",
"Inserted after a User message.": "Inséré après un message utilisateur.",
- "User Message Suffix": "Suffixe du message utilisateur",
- "Inserted before an Assistant message and as a last prompt line when generating an AI reply.": "Inséré avant un message de l'Assistant et comme dernière ligne d'invite lors de la génération d'une réponse AI.",
- "Assistant Message Prefix": "Préfixe du message de l'assistant",
+ "Inserted before an Assistant message and as a last prompt line when generating an AI reply.": "Inséré avant un message de l'Assistant et comme dernière ligne du prompt lors de la génération d'une réponse de l'IA.",
"Inserted after an Assistant message.": "Inséré après un message de l'assistant.",
- "Assistant Message Suffix": "Suffixe du message de l'assistant",
"Inserted before a System (added by slash commands or extensions) message.": "Inséré avant un message système (ajouté par des commandes slash ou des extensions).",
- "System Message Prefix": "Préfixe du message système",
"Inserted after a System message.": "Inséré après un message système.",
- "System Message Suffix": "Suffixe du message système",
"If enabled, System Sequences will be the same as User Sequences.": "Si cette option est activée, les séquences système seront les mêmes que les séquences utilisateur.",
"System same as User": "Système identique à l'utilisateur",
"Misc. Sequences": "Séquences diverses",
- "Inserted before the first Assistant's message.": "Inséré avant le message du premier assistant.",
+ "Inserted before the first Assistant's message.": "Inséré avant le premier message de l'assistant.",
"First Assistant Prefix": "Préfixe du premier assistant",
- "instruct_last_output_sequence": "Inséré avant le dernier message de l'assistant ou comme dernière ligne d'invite lors de la génération d'une réponse IA (sauf un rôle neutre/système).",
+ "instruct_last_output_sequence": "Inséré avant le dernier message de l'assistant ou comme dernière ligne du prompt lors de la génération d'une réponse de l'IA (sauf un rôle neutre/système).",
"Last Assistant Prefix": "Préfixe du dernier assistant",
- "Will be inserted as a last prompt line when using system/neutral generation.": "Sera inséré comme dernière ligne d’invite lors de l’utilisation de la génération système/neutre.",
+ "Will be inserted as a last prompt line when using system/neutral generation.": "Sera inséré comme dernière ligne du prompt lors de l’utilisation de la génération système/neutre.",
"System Instruction Prefix": "Préfixe d'instruction système",
"If a stop sequence is generated, everything past it will be removed from the output (inclusive).": "Si une séquence d'arrêt est générée, tout ce qui se trouve au-delà sera supprimé de la sortie (inclus).",
"Stop Sequence": "Séquence d'arrêt",
"Will be inserted at the start of the chat history if it doesn't start with a User message.": "Sera inséré au début de l'historique des discussions s'il ne commence pas par un message utilisateur.",
"User Filler Message": "Message de remplissage de l'utilisateur",
"Context Formatting": "Formatage du contexte",
- "(Saved to Context Template)": "(Enregistré dans le modèle de contexte)",
- "Always add character's name to prompt": "Ajouter toujours le nom du personnage au support",
- "Generate only one line per request": "Générer seulement une ligne par demande",
+ "Always add character's name to prompt": "Toujours ajouter le nom du personnage au prompt",
+ "Generate only one line per request": "Générer seulement une ligne par requête",
"Trim Incomplete Sentences": "Supprimer les phrases incomplètes",
- "Include Newline": "Inclure un retour à la ligne",
- "Misc. Settings": "Paramètres divers",
"Collapse Consecutive Newlines": "Réduire les sauts de ligne consécutifs",
"Trim spaces": "Supprimer les espaces",
"Tokenizer": "Tokeniseur",
- "Token Padding": "Remplissage de jeton",
+ "Token Padding": "Remplissage de token",
"Start Reply With": "Commencer la réponse avec",
- "AI reply prefix": "Préfixe de réponse de l'IA",
"Show reply prefix in chat": "Afficher le préfixe de réponse dans la conversation",
"Non-markdown strings": "Chaînes non Markdown",
- "separate with commas w/o space between": "séparez avec des virgules sans espace entre",
"Custom Stopping Strings": "Chaînes d'arrêt personnalisées",
"JSON serialized array of strings": "Tableau de chaînes sérialisé JSON",
"Replace Macro in Custom Stopping Strings": "Remplacer les macro dans les chaînes d'arrêt personnalisées",
- "Auto-Continue": "Auto-Continuer",
- "Allow for Chat Completion APIs": "Autoriser les APIs de complétion de la discussion",
- "Target length (tokens)": "Longueur cible (jetons)",
- "World Info": "Informations sur le monde",
- "Locked = World Editor will stay open": "Verrouillé = l'éditeur de monde restera ouvert",
- "Worlds/Lorebooks": "Mondes/Livres de lore",
+ "Auto-Continue": "Auto-Continue",
+ "Allow for Chat Completion APIs": "Autoriser les APIs de complétion de chat",
+ "Target length (tokens)": "Longueur cible (tokens)",
+ "World Info": "World Info",
+ "Locked = World Editor will stay open": "Verrouillé = l'éditeur des World Info restera ouvert",
+ "Worlds/Lorebooks": "Worlds/Lorebooks",
"Active World(s) for all chats": "Monde(s) actif(s) pour toutes les conversations",
- "-- World Info not found --": "-- Informations sur le monde non trouvées --",
- "Global World Info/Lorebook activation settings": "Paramètres d'activation de Global World Info/Lorebook",
+ "-- World Info not found --": "-- World Info non trouvées --",
+ "Global World Info/Lorebook activation settings": "Paramètres d'activation globales des World Info/Lorebook",
"Click to expand": "Cliquez pour agrandir",
"Scan Depth": "Profondeur de scan",
"Context %": "Pourcentage de contexte",
@@ -518,41 +470,39 @@
"New": "Nouveau",
"or": "ou",
"--- Pick to Edit ---": "--- Sélectionnez pour éditer ---",
- "Rename World Info": "Renommer les informations sur le monde",
+ "Rename World Info": "Renommer le World Info",
"Open all Entries": "Ouvrir toutes les entrées",
"Close all Entries": "Fermer toutes les entrées",
"New Entry": "Nouvelle entrée",
"Fill empty Memo/Titles with Keywords": "Remplir les mémos/titres vides avec des mots-clés",
- "Import World Info": "Importer les informations sur le monde",
- "Export World Info": "Exporter les informations sur le monde",
- "Duplicate World Info": "Dupliquer les informations sur le monde",
- "Delete World Info": "Supprimer les informations sur le monde",
+ "Import World Info": "Importer un Wolrd Info",
+ "Export World Info": "Exporter le Wolrd Info",
+ "Duplicate World Info": "Dupliquer le World Info",
+ "Delete World Info": "Supprimer le World Info",
"Search...": "Rechercher...",
"Search": "Recherche",
"Priority": "Priorité",
"Custom": "Personnalisé",
"Title A-Z": "Titre A-Z",
"Title Z-A": "Titre Z-A",
- "Tokens ↗": "Jetons ↗",
- "Tokens ↘": "Jetons ↘",
+ "Tokens ↗": "Tokens ↗",
+ "Tokens ↘": "Tokens ↘",
"Depth ↗": "Profondeur ↗",
"Depth ↘": "Profondeur ↘",
- "Order ↗": "Commande ↗",
- "Order ↘": "Commande ↘",
+ "Order ↗": "Ordre ↗",
+ "Order ↘": "Ordre ↘",
"UID ↗": "UID ↗",
"UID ↘": "UID ↘",
"Trigger% ↗": "Déclencheur% ↗",
"Trigger% ↘": "Déclencheur% ↘",
"Refresh": "Actualiser",
"User Settings": "Paramètres utilisateur",
- "Simple": "Simple",
- "Advanced": "Avancé",
"UI Language": "Langue",
"Account": "Compte",
- "Admin Panel": "panneau d'administration",
- "Logout": "Se déconnecter",
- "Search Settings": "Paramètres de recherche",
- "UI Theme": "Thème de l'interface utilisateur",
+ "Admin Panel": "Panneau d'administration",
+ "Logout": "Déconnection",
+ "Search Settings": "Recherche de paramètres",
+ "UI Theme": "Thème de l'UI",
"Import a theme file": "Importer un fichier de thème",
"Export a theme file": "Exporter un fichier de thème",
"Delete a theme": "Supprimer un thème",
@@ -582,8 +532,8 @@
"Width of the main chat window in % of screen width": "Largeur de la fenêtre de discussion principale en % de la largeur de l'écran",
"Font Scale": "Échelle de police",
"Font size": "Taille de police",
- "Blur Strength": "Force de flou",
- "Blur strength on UI panels.": "Force de flou sur les panneaux de l'interface utilisateur.",
+ "Blur Strength": "Force du flou",
+ "Blur strength on UI panels.": "Force du flou sur les panneaux de l'interface utilisateur.",
"Text Shadow Width": "Largeur de l'ombre du texte",
"Strength of the text shadows": "Force des ombres du texte",
"Disables animations and transitions": "Désactive les animations et transitions",
@@ -595,11 +545,11 @@
"Reduce chat height, and put a static sprite behind the chat window": "Réduire la hauteur de la discussion et placer un sprite statique derrière la fenêtre de discussion",
"Waifu Mode": "Mode Waifu",
"Always show the full list of the Message Actions context items for chat messages, instead of hiding them behind '...'": "Afficher toujours la liste complète des éléments de contexte Actions de message pour les messages de discussion, au lieu de les cacher derrière '...' ",
- "Auto-Expand Message Actions": "Extension automatique des actions de message",
+ "Auto-Expand Message Actions": "Extension automatique des actions du message",
"Alternative UI for numeric sampling parameters with fewer steps": "Interface utilisateur alternative pour les paramètres d'échantillonnage numérique avec moins d'étapes",
"Zen Sliders": "Curseurs Zen",
"Entirely unrestrict all numeric sampling parameters": "Déverrouiller entièrement tous les paramètres d'échantillonnage numérique",
- "Mad Lab Mode": "Mode laboratoire fou",
+ "Mad Lab Mode": "Mode Mad Lab",
"Time the AI's message generation, and show the duration in the chat log": "Chronométrer la génération de messages de l'IA et afficher la durée dans le journal de discussion",
"Message Timer": "Minuteur de message",
"Show a timestamp for each message in the chat log": "Afficher un horodatage pour chaque message dans le journal de discussion",
@@ -607,38 +557,35 @@
"Show an icon for the API that generated the message": "Afficher une icône pour l'API qui a généré le message",
"Model Icon": "Icône du modèle",
"Show sequential message numbers in the chat log": "Afficher les numéros de message séquentiels dans le journal de discussion",
- "Message IDs": "Identifiants de message",
+ "Message IDs": "Identifiants du message",
"Hide avatars in chat messages.": "Masquez les avatars dans les messages de discussion.",
"Hide Chat Avatars": "Masquer les avatars du chat",
- "Show the number of tokens in each message in the chat log": "Afficher le nombre de jetons dans chaque message dans le journal de discussion",
- "Show Message Token Count": "Afficher le nombre de jetons de message",
+ "Show the number of tokens in each message in the chat log": "Afficher le nombre de tokens dans chaque message dans le journal de discussion",
+ "Show Message Token Count": "Afficher le nombre de tokens du message",
"Single-row message input area. Mobile only, no effect on PC": "Zone de saisie de message sur une seule ligne. Mobile uniquement, aucun effet sur PC",
"Compact Input Area (Mobile)": "Zone de saisie compacte (Mobile)",
"In the Character Management panel, show quick selection buttons for favorited characters": "Dans le panneau de gestion des personnages, afficher des boutons de sélection rapide pour les personnages favoris",
"Characters Hotswap": "Échange rapide de personnages",
"Enable magnification for zoomed avatar display.": "Activer le grossissement pour l'affichage de l'avatar zoomé.",
- "Avatar Hover Magnification": "Agrandissement du survol de l'avatar",
+ "Avatar Hover Magnification": "Agrandissement au survol de l'avatar",
"Enables a magnification effect on hover when you display the zoomed avatar after clicking an avatar's image in chat.": "Active un effet de grossissement au survol lorsque vous affichez l'avatar zoomé après avoir cliqué sur l'image d'un avatar dans le chat.",
"Show tagged character folders in the character list": "Afficher les dossiers de personnages tagués dans la liste de personnages",
- "Tags as Folders": "Balises comme dossiers",
- "Tags_as_Folders_desc": "Modification récente : les balises doivent être marquées comme dossiers dans le menu Gestion des balises pour apparaître comme telles. Cliquez ici pour l'afficher.",
- "Character Handling": "Gestion des caractères",
+ "Tags as Folders": "Tags comme dossiers",
+ "Tags_as_Folders_desc": "Modification récente : les tags doivent être marqués comme dossiers dans le menu Gestion des tags pour apparaître comme telles. Cliquez ici pour l'afficher.",
+ "Character Handling": "Gestion des personnages",
"If set in the advanced character definitions, this field will be displayed in the characters list.": "Si défini dans les définitions de personnage avancées, ce champ sera affiché dans la liste des personnages.",
- "Char List Subheader": "Sous-en-tête de la liste de caractères",
+ "Char List Subheader": "Sous-en-tête de la liste des personnages",
"Character Version": "Version du personnage",
"Created by": "Créé par",
"Use fuzzy matching, and search characters in the list by all data fields, not just by a name substring": "Utiliser la correspondance floue et rechercher des personnages dans la liste par tous les champs de données, pas seulement par une sous-chaîne de nom",
"Advanced Character Search": "Recherche de personnage avancée",
- "If checked and the character card contains a prompt override (System Prompt), use that instead": "Si cochée et si la carte de personnage contient un remplacement d'invite (invite système), l'utiliser à la place",
- "Prefer Character Card Prompt": "Préférer l'invite de carte de personnage",
- "If checked and the character card contains a jailbreak override (Post History Instruction), use that instead": "Si cochée et si la carte de personnage contient un contournement de jailbreak (Instruction d'historique des messages), l'utiliser à la place",
- "Prefer Character Card Jailbreak": "Préférer le jailbreak de carte de personnage",
+ "If checked and the character card contains a prompt override (System Prompt), use that instead": "Si cochée et si la carte de personnage contient un prompt de remplacement (prompt système), l'utiliser à la place",
+ "Prefer Character Card Prompt": "Préférer le prompt du personnage",
"Avoid cropping and resizing imported character images. When off, crop/resize to 512x768": "Évitez de recadrer et de redimensionner les images de personnages importés. Lorsqu'il est désactivé, recadrez/redimensionnez à 512 x 768.",
"Never resize avatars": "Ne jamais redimensionner les avatars",
"Show actual file names on the disk, in the characters list display only": "Afficher les noms de fichier réels sur le disque, dans l'affichage de la liste de personnages uniquement",
"Show avatar filenames": "Afficher les noms de fichier des avatars",
- "Prompt to import embedded card tags on character import. Otherwise embedded tags are ignored": "Proposer d'importer les balises de carte intégrées lors de l'importation de personnages. Sinon, les balises intégrées sont ignorées",
- "Import Card Tags": "Importer les balises de carte",
+ "Import Card Tags": "Importer les tags de carte",
"Hide character definitions from the editor panel behind a spoiler button": "Masquer les définitions de personnages du panneau d'édition derrière un bouton spoiler",
"Spoiler Free Mode": "Mode sans spoiler",
"Miscellaneous": "Divers",
@@ -651,42 +598,42 @@
"Fast": "Rapide",
"Play a sound when a message generation finishes": "Jouer un son lorsque la génération de message est terminée",
"Message Sound": "Son de message",
- "Only play a sound when ST's browser tab is unfocused": "Jouer un son uniquement lorsque l'onglet du navigateur ST n'est pas centré",
+ "Only play a sound when ST's browser tab is unfocused": "Jouer un son uniquement lorsque l'onglet ST du navigateur n'est pas active",
"Background Sound Only": "Son de fond uniquement",
"Reduce the formatting requirements on API URLs": "Réduire les exigences de formatage sur les URL d'API",
"Relaxed API URLS": "URLs d'API détendues",
- "Ask to import the World Info/Lorebook for every new character with embedded lorebook. If unchecked, a brief message will be shown instead": "Demander d'importer les informations du monde/Lorebook pour chaque nouveau personnage avec Lorebook intégré. Si non cochée, un message bref sera affiché à la place",
+ "Ask to import the World Info/Lorebook for every new character with embedded lorebook. If unchecked, a brief message will be shown instead": "Demander d'importer le Word Info/Lorebook pour chaque nouveau personnage avec Lorebook intégré. Si non cochée, un message bref sera affiché à la place",
"Lorebook Import Dialog": "Boîte de dialogue d'importation de Lorebook",
"Restore unsaved user input on page refresh": "Restaurer les saisies utilisateur non enregistrées lors du rafraîchissement de la page",
"Restore User Input": "Restaurer l'entrée utilisateur",
"Allow repositioning certain UI elements by dragging them. PC only, no effect on mobile": "Permettre le repositionnement de certains éléments de l'interface utilisateur en les faisant glisser. PC uniquement, aucun effet sur mobile",
"Movable UI Panels": "Panels d'UI déplaçables",
- "MovingUI preset. Predefined/saved draggable positions": "Préréglage MovingUI. Positions prédéfinies/enregistrées pouvant être déplacées",
- "MUI Preset": "Préréglage MUI",
+ "MovingUI preset. Predefined/saved draggable positions": "Preset MovingUI. Positions prédéfinies/enregistrées pouvant être déplacées",
+ "MUI Preset": "Preset MUI",
"Save movingUI changes to a new file": "Enregistrer les modifications de MovingUI dans un nouveau fichier",
"Reset MovingUI panel sizes/locations.": "Réinitialisez les tailles/emplacements des panneaux MovingUI.",
"Apply a custom CSS style to all of the ST GUI": "Appliquer un style CSS personnalisé à toute l'interface utilisateur de ST",
"Custom CSS": "CSS personnalisé",
- "Expand the editor": "Développez l'éditeur",
+ "Expand the editor": "Agrandir l'éditeur",
"Chat/Message Handling": "Gestion de la discussion/des messages",
- "# Messages to Load": "# Message. charger",
+ "# Messages to Load": "# Messages à charger",
"The number of chat history messages to load before pagination.": "Le nombre de messages de l'historique des discussions à charger avant la pagination.",
"(0 = All)": "(0 = Tout)",
- "Streaming FPS": "Images par seconde en streaming",
+ "Streaming FPS": "FPS en streaming",
"Update speed of streamed text.": "Vitesse de mise à jour du texte diffusé.",
"Example Messages Behavior": "Comportement des messages d'exemple",
"Gradual push-out": "Poussée progressive",
- "Always include examples": "Inclure toujours des exemples",
- "Never include examples": "Ne jamais inclure d'exemples",
+ "Always include examples": "Toujours inclure les exemples",
+ "Never include examples": "Ne jamais inclure les exemples",
"Send on Enter": "Envoyer sur Entrée",
"Disabled": "Désactivé",
"Automatic (PC)": "Automatique (PC)",
"Press Send to continue": "Appuyez sur Envoyer pour continuer",
"Show a button in the input area to ask the AI to continue (extend) its last message": "Afficher un bouton dans la zone de saisie pour demander à l'IA de continuer (étendre) son dernier message",
- "Quick 'Continue' button": "Bouton 'Continuer' rapide",
+ "Quick 'Continue' button": "Raccourcis 'Continuer'",
"Show arrow buttons on the last in-chat message to generate alternative AI responses. Both PC and mobile": "Afficher des boutons fléchés sur le dernier message de discussion pour générer des réponses alternatives de l'IA. Sur PC et mobile",
"Swipes": "Balayages",
- "Allow using swiping gestures on the last in-chat message to trigger swipe generation. Mobile only, no effect on PC": "Autoriser l'utilisation de gestes de balayage sur le dernier message de discussion pour déclencher la génération de glissement. Mobile uniquement, aucun effet sur PC",
+ "Allow using swiping gestures on the last in-chat message to trigger swipe generation. Mobile only, no effect on PC": "Autoriser l'utilisation de gestes de balayage sur le dernier message de discussion pour déclencher la génération de messages alternatifs. Mobile uniquement, aucun effet sur PC",
"Gestures": "Gestes",
"Auto-load Last Chat": "Chargement automatique de la dernière discussion",
"Auto-scroll Chat": "Défilement automatique de la discussion",
@@ -700,20 +647,20 @@
"Allow {{user}}: in bot messages": "Autoriser {{user}} : dans les messages du bot",
"Skip encoding and characters in message text, allowing a subset of HTML markup as well as Markdown": "Ignorer l'encodage et les caractères < et > dans le texte du message, permettant un sous-ensemble du balisage HTML ainsi que Markdown",
"Show tags in responses": "Afficher les tags dans les réponses",
- "Allow AI messages in groups to contain lines spoken by other group members": "Autoriser les messages AI dans les groupes à contenir des lignes prononcées par d'autres membres du groupe",
+ "Allow AI messages in groups to contain lines spoken by other group members": "Autoriser les messages de l'IA dans les groupes à contenir des lignes prononcées par d'autres membres du groupe",
"Relax message trim in Groups": "Relaxer la taille des messages dans les groupes",
- "Log prompts to console": "Journaliser les invites dans la console",
- "Requests logprobs from the API for the Token Probabilities feature": "Demande des logprobs de l'API pour la fonctionnalité des probabilités de jetons",
- "Request token probabilities": "Probabilités de jeton de demande",
+ "Log prompts to console": "Journaliser les prompts dans la console",
+ "Requests logprobs from the API for the Token Probabilities feature": "Demande des logprobs de l'API pour la fonctionnalité des probabilités des tokens",
+ "Request token probabilities": "Demande des probabilités de tokens",
"Automatically reject and re-generate AI message based on configurable criteria": "Rejeter automatiquement et régénérer un message AI en fonction de critères configurables",
- "Auto-swipe": "Glissement automatique",
- "Enable the auto-swipe function. Settings in this section only have an effect when auto-swipe is enabled": "Activer la fonction de glissement automatique. Les paramètres de cette section n'ont d'effet que lorsque le glissement automatique est activé",
+ "Auto-swipe": "Balayage automatique",
+ "Enable the auto-swipe function. Settings in this section only have an effect when auto-swipe is enabled": "Activer la fonction de balayage automatique. Les paramètres de cette section n'ont d'effet que lorsque le balayage automatique est activé",
"Minimum generated message length": "Longueur minimale du message généré",
- "If the generated message is shorter than this, trigger an auto-swipe": "Si le message généré est plus court que cela, déclenchez un glissement automatique",
+ "If the generated message is shorter than this, trigger an auto-swipe": "Si le message généré est plus court que cela, déclenchez un balayage automatique",
"Blacklisted words": "Mots en liste noire",
"words you dont want generated separated by comma ','": "mots que vous ne voulez pas générer séparés par des virgules ','",
- "Blacklisted word count to swipe": "Nombre de mots en liste noire à glisser",
- "Minimum number of blacklisted words detected to trigger an auto-swipe": "Nombre minimum de mots en liste noire détectés pour déclencher un glissement automatique",
+ "Blacklisted word count to swipe": "Nombre de mots en liste noire pour balayer",
+ "Minimum number of blacklisted words detected to trigger an auto-swipe": "Nombre minimum de mots en liste noire détectés pour déclencher un balayage automatique",
"AutoComplete Settings": "Paramètres de saisie semi-automatique",
"Automatically hide details": "Masquer automatiquement les détails",
"Determines how entries are found for autocomplete.": "Détermine comment les entrées sont trouvées pour la saisie semi-automatique.",
@@ -736,115 +683,112 @@
"Parser Flags": "Indicateurs de l'analyseur",
"Switch to stricter escaping, allowing all delimiting characters to be escaped with a backslash, and backslashes to be escaped as well.": "Passez à un échappement plus strict, permettant à tous les caractères de délimitation d'être échappés avec une barre oblique inverse, ainsi qu'aux barres obliques inverses.",
"STRICT_ESCAPING": "STRICT_ESCAPING",
- "Replace all {{getvar::}} and {{getglobalvar::}} macros with scoped variables to avoid double macro substitution.": "Remplacez toutes les macros {{getvar::}} et {{getglobalvar::}} par des variables à portée limitée pour éviter la double substitution de macro.",
"REPLACE_GETVAR": "REPLACE_GETVAR",
- "Change Background Image": "Changer l'image de fond",
+ "Change Background Image": "Changer l'arrière-plans",
"Filter": "Filtre",
"Automatically select a background based on the chat context": "Sélectionner automatiquement un arrière-plan en fonction du contexte de la discussion",
"Auto-select": "Sélection automatique",
"System Backgrounds": "Arrière-plans du système",
"Chat Backgrounds": "Arrière-plans de la discussion",
- "bg_chat_hint_1": "Arrière-plans de discussion générés avec le",
- "bg_chat_hint_2": "l'extension apparaîtra ici.",
+ "bg_chat_hint_1": "Les arrière-plans générés avec l'extension ",
+ "bg_chat_hint_2": " apparaîtrons ici.",
"Extensions": "Extensions",
- "Notify on extension updates": "Notifier les mises à jour de l'extension",
+ "Notify on extension updates": "Notifier les mises à jour des extensions",
"Manage extensions": "Gérer les extensions",
"Import Extension From Git Repo": "Importer une extension depuis un dépôt Git",
- "Install extension": "Installer l'extension",
+ "Install extension": "Installer une extension",
"Extras API:": "API supplémentaires :",
"Auto-connect": "Connexion automatique",
"Extras API URL": "URL de l'API des extras",
"Extras API key (optional)": "Clé API supplémentaire (facultatif)",
- "Persona Management": "Gestion de la personnalité",
- "How do I use this?": "Comment puis-je utiliser ceci?",
+ "Persona Management": "Gestion des personas",
"Click for stats!": "Cliquez pour les statistiques!",
"Usage Stats": "Statistiques d'utilisation",
"Backup your personas to a file": "Sauvegardez vos personas dans un fichier",
"Backup": "Sauvegarde",
"Restore your personas from a file": "Restaurez vos personas à partir d'un fichier",
"Restore": "Restaurer",
- "Create a dummy persona": "Créer une personnalité factice",
+ "Create a dummy persona": "Créer une persona factice",
"Create": "Créer",
"Toggle grid view": "Basculer en mode grille",
"No persona description": "[Pas de description]",
"Name": "Nom",
"Enter your name": "Entrez votre nom",
"Click to set a new User Name": "Cliquez pour définir un nouveau nom d'utilisateur",
- "Click to lock your selected persona to the current chat. Click again to remove the lock.": "Cliquez pour verrouiller votre personnalité sélectionnée sur la discussion actuelle. Cliquez à nouveau pour supprimer le verrou.",
+ "Click to lock your selected persona to the current chat. Click again to remove the lock.": "Cliquez pour verrouiller votre persona sélectionnée sur la discussion actuelle. Cliquez à nouveau pour supprimer le verrou.",
"Click to set user name for all messages": "Cliquez pour définir le nom d'utilisateur pour tous les messages",
- "Persona Description": "Description de la personnalité",
+ "Persona Description": "Description de la persona",
"Example: [{{user}} is a 28-year-old Romanian cat girl.]": "Exemple : [{{user}} est une fille chat roumaine de 28 ans.]",
- "Tokens persona description": "Description des jetons",
+ "Tokens persona description": "Description des tokens",
"Position:": "Position :",
- "In Story String / Prompt Manager": "Dans la chaîne d'histoire / Gestionnaire d'instructions",
- "Top of Author's Note": "En haut de la note de l'auteur",
- "Bottom of Author's Note": "En bas de la note de l'auteur",
+ "In Story String / Prompt Manager": "Dans la chaîne d'histoire / Gestionnaire de prompt",
+ "Top of Author's Note": "En haut de la note d'auteur",
+ "Bottom of Author's Note": "En bas de la note d'auteur",
"In-chat @ Depth": "Dans le chat @ Profondeur",
"Depth:": "Profondeur :",
"Role:": "Rôle:",
"System": "Système",
"User": "Utilisateur",
"Assistant": "Assistant",
- "Show notifications on switching personas": "Afficher les notifications lors du changement de personnalités",
+ "Show notifications on switching personas": "Afficher les notifications lors du changement de persona",
"Character Management": "Gestion des personnages",
"Locked = Character Management panel will stay open": "Verrouillé = le panneau de gestion des personnages restera ouvert",
"Select/Create Characters": "Sélectionner/Créer des personnages",
- "Favorite characters to add them to HotSwaps": "Favoriser les personnages pour les ajouter aux HotSwaps",
- "Token counts may be inaccurate and provided just for reference.": "Les comptages de jetons peuvent être inexacts et fournis uniquement à titre de référence.",
- "Total tokens": "Total des jetons",
- "Calculating...": "Calculateur...",
- "Tokens": "Jetons",
- "Permanent tokens": "Jetons permanents",
+ "Favorite characters to add them to HotSwaps": "Mettre les personnages en favoris pour les ajouter aux HotSwaps",
+ "Token counts may be inaccurate and provided just for reference.": "Le comptage des tokens peut être inexacts et est fournis uniquement à titre de référence.",
+ "Total tokens": "Total des tokens",
+ "Calculating...": "Calcul en cours...",
+ "Tokens": "Tokens",
+ "Permanent tokens": "Tokens permanents",
"Permanent": "Permanent",
- "About Token 'Limits'": "À propos des « limites » des jetons",
+ "About Token 'Limits'": "À propos des « limites » des tokens",
"Toggle character info panel": "Basculer le panneau d'informations sur le personnage",
"Name this character": "Nommer ce personnage",
- "extension_token_counter": "Jetons :",
- "Click to select a new avatar for this character": "Cliquez pour sélectionner une nouvelle avatar pour ce personnage",
+ "extension_token_counter": "Tokens :",
+ "Click to select a new avatar for this character": "Cliquez pour sélectionner un nouvel avatar pour ce personnage",
"Add to Favorites": "Ajouter aux favoris",
"Advanced Definition": "Définition avancée",
- "Character Lore": "Lore du personnage",
- "Chat Lore": "Histoire du chat",
+ "Chat Lore": "Lore du chat",
"Export and Download": "Exporter et Télécharger",
"Duplicate Character": "Dupliquer le personnage",
"Create Character": "Créer un personnage",
"Delete Character": "Supprimer le personnage",
"More...": "Plus...",
- "Link to World Info": "Lien vers les informations sur le monde",
- "Import Card Lore": "Importer la lore de la carte",
+ "Link to World Info": "Lien vers le World Info",
+ "Import Card Lore": "Importer le lore de la carte",
"Scenario Override": "Remplacement du scénario",
"Convert to Persona": "Convertir en Persona",
"Rename": "Renommer",
"Link to Source": "Lien vers la source",
"Replace / Update": "Remplacer/Mettre à jour",
- "Import Tags": "Importer des balises",
- "Search / Create Tags": "Rechercher / Créer des balises",
- "View all tags": "Voir toutes les balises",
+ "Import Tags": "Importer les tags",
+ "Search / Create Tags": "Rechercher / Créer des tags",
+ "View all tags": "Voir tout les tags",
"Creator's Notes": "Notes du créateur",
"Show / Hide Description and First Message": "Afficher / Masquer la description et le premier message",
"Character Description": "Description du personnage",
"Click to allow/forbid the use of external media for this character.": "Cliquez pour autoriser/interdire l’utilisation de médias externes pour ce personnage.",
- "Ext. Media": "Poste. Médias",
+ "Ext. Media": "Médias Ext.",
"Describe your character's physical and mental traits here.": "Décrivez les traits physiques et mentaux de votre personnage ici.",
"First message": "Premier message",
"Click to set additional greeting messages": "Cliquez pour définir des messages de salutation supplémentaires",
- "Alt. Greetings": "Alt. Salutations",
+ "Alt. Greetings": "Salutations Alt.",
"This will be the first message from the character that starts every chat.": "Ce sera le premier message du personnage qui démarre chaque discussion.",
"Group Controls": "Contrôles de groupe",
"Chat Name (Optional)": "Nom de la discussion (Facultatif)",
- "Click to select a new avatar for this group": "Cliquez pour sélectionner une nouvelle avatar pour ce groupe",
+ "Click to select a new avatar for this group": "Cliquez pour sélectionner un nouvel avatar pour ce groupe",
"Group reply strategy": "Stratégie de réponse de groupe",
"Natural order": "Ordre naturel",
"List order": "Ordre de la liste",
"Group generation handling mode": "Mode de gestion de la génération de groupe",
"Swap character cards": "Échanger les cartes de personnages",
- "Join character cards (exclude muted)": "Rejoignez les cartes de personnage (excluez les cartes en sourdine)",
- "Join character cards (include muted)": "Rejoignez les cartes de personnage (y compris en sourdine)",
+ "Join character cards (exclude muted)": "Joignez les cartes de personnage (excluez les cartes en sourdine)",
+ "Join character cards (include muted)": "Joignez les cartes de personnage (y compris en sourdine)",
"Inserted before each part of the joined fields.": "Inséré avant chaque partie des champs joints.",
- "Join Prefix": "Rejoindre le préfixe",
- "When 'Join character cards' is selected, all respective fields of the characters are being joined together.This means that in the story string for example all character descriptions will be joined to one big text.If you want those fields to be separated, you can define a prefix or suffix here.This value supports normal macros and will also replace {{char}} with the relevant char's name and with the name of the part (e.g.: description, personality, scenario, etc.)": "Lorsque « Rejoindre les cartes de personnage » est sélectionné, tous les champs respectifs des personnages sont réunis.\rCela signifie que dans la chaîne d'histoire, par exemple, toutes les descriptions des personnages seront réunies en un seul grand texte.\rSi vous souhaitez que ces champs soient séparés, vous pouvez définir ici un préfixe ou un suffixe.\r\rCette valeur prend en charge les macros normales et remplacera également {{char}} par le nom du caractère concerné et par le nom de la pièce (par exemple : description, personnalité, scénario, etc.)",
+ "Join Prefix": "Joindre les préfixes",
+ "When 'Join character cards' is selected, all respective fields of the characters are being joined together.This means that in the story string for example all character descriptions will be joined to one big text.If you want those fields to be separated, you can define a prefix or suffix here.This value supports normal macros and will also replace {{char}} with the relevant char's name and with the name of the part (e.g.: description, personality, scenario, etc.)": "Lorsque « Joindre les cartes de personnage » est sélectionné, tous les champs respectifs des personnages sont réunis.\rCela signifie que dans la chaîne d'histoire, par exemple, toutes les descriptions des personnages seront réunies en un seul grand texte.\rSi vous souhaitez que ces champs soient séparés, vous pouvez définir ici un préfixe ou un suffixe.\r\rCette valeur prend en charge les macros normales et remplacera également {{char}} par le nom du caractère concerné et par le nom de la pièce (par exemple : description, personnalité, scénario, etc.)",
"Inserted after each part of the joined fields.": "Inséré après chaque partie des champs joints.",
- "Join Suffix": "Rejoindre le suffixe",
+ "Join Suffix": "Suffixe de jointure",
"Set a group chat scenario": "Définir un scénario de discussion de groupe",
"Click to allow/forbid the use of external media for this group.": "Cliquez pour autoriser/interdire l’utilisation de médias externes pour ce groupe.",
"Restore collage avatar": "Restaurer l'avatar du collage",
@@ -857,8 +801,8 @@
"Create New Character": "Créer un nouveau personnage",
"Import Character from File": "Importer un personnage à partir d'un fichier",
"Import content from external URL": "Importer du contenu depuis une URL externe",
- "Create New Chat Group": "Créer un nouveau groupe de discussion",
- "Characters sorting order": "Ordre de tri des personnages",
+ "Create New Chat Group": "Créer une nouvelle discussion de groupe",
+ "Characters sorting order": "Ordre des personnages",
"A-Z": "A-Z",
"Z-A": "Z-A",
"Newest": "Plus récent",
@@ -867,8 +811,8 @@
"Recent": "Récent",
"Most chats": "Plus de discussions",
"Least chats": "Moins de discussions",
- "Most tokens": "La plupart des jetons",
- "Least tokens": "Moins de jetons",
+ "Most tokens": "Tokens maximum",
+ "Least tokens": "Tokens minimum",
"Random": "Aléatoire",
"Toggle character grid view": "Basculer vers la vue en grille des personnages",
"Bulk_edit_characters": "Édition en masse des personnages",
@@ -879,7 +823,7 @@
"popup-button-no": "Non",
"popup-button-cancel": "Annuler",
"popup-button-import": "Importer",
- "Advanced Defininitions": "Définitions avancées",
+ "Advanced Definitions": "Définitions avancées",
"Prompt Overrides": "Remplacements d'invite",
"(For Chat Completion and Instruct Mode)": "(Pour l'achèvement du chat et le mode instruction)",
"Insert {{original}} into either box to include the respective default prompt from system settings.": "Insérez {{original}} dans l'un ou l'autre des champs pour inclure l'instruction par défaut respective des paramètres système.",
@@ -888,13 +832,13 @@
"Any contents here will replace the default Jailbreak Prompt used for this character. (v2 spec: post_history_instructions)": "Tout contenu ici remplacera l'instruction de jailbreak par défaut utilisée pour ce personnage. (spécification v2 : post_history_instructions)",
"Creator's Metadata (Not sent with the AI prompt)": "Métadonnées du créateur (Non envoyées avec l'instruction de l'IA)",
"Creator's Metadata": "Métadonnées du créateur",
- "(Not sent with the AI Prompt)": "(Non envoyé avec l'invite AI)",
+ "(Not sent with the AI Prompt)": "(Non envoyé avec le prompt IA)",
"Everything here is optional": "Tout ce qui se trouve ici est facultatif",
"(Botmaker's name / Contact Info)": "(Nom du créateur du bot / Informations de contact)",
"(If you want to track character versions)": "(Si vous voulez suivre les versions du personnage)",
"(Describe the bot, give use tips, or list the chat models it has been tested on. This will be displayed in the character list.)": "(Décrivez le bot, donnez des conseils d'utilisation ou répertoriez les modèles de discussion sur lesquels il a été testé. Cela s'affichera dans la liste des personnages.)",
- "Tags to Embed": "Balises à intégrer",
- "(Write a comma-separated list of tags)": "(Écrivez une liste de balises séparées par des virgules)",
+ "Tags to Embed": "Tags à intégrer",
+ "(Write a comma-separated list of tags)": "(Écrivez une liste de tags séparés par des virgules)",
"Personality summary": "Résumé de la personnalité",
"(A brief description of the personality)": "(Une brève description de la personnalité)",
"Scenario": "Scénario",
@@ -912,58 +856,56 @@
"Chatty": "Bavard",
"Examples of dialogue": "Exemples de dialogue",
"Important to set the character's writing style.": "Important de définir le style d'écriture du personnage.",
- "(Examples of chat dialog. Begin each example with START on a new line.)": "(Exemples de dialogue de chat. Commencez chaque exemple par DÉBUT sur une nouvelle ligne.)",
+ "(Examples of chat dialog. Begin each example with START on a new line.)": "(Exemples de dialogue. Commencez chaque exemple par START sur une nouvelle ligne.)",
"Save": "Enregistrer",
"Chat History": "Historique de la conversation",
"Import Chat": "Importer une discussion",
"Copy to system backgrounds": "Copier vers les arrière-plans du système",
"Rename background": "Renommer l'arrière-plan",
- "Lock": "Verrouillage",
- "Unlock": "Ouvrir",
+ "Lock": "Verrouiller",
+ "Unlock": "Déverrouiller",
"Delete background": "Supprimer l'arrière-plan",
"Chat Scenario Override": "Remplacement du scénario de discussion",
"Remove": "Supprimer",
"Type here...": "Tapez ici...",
- "Chat Lorebook": "Livre de contes pour",
+ "Chat Lorebook": "Lorebook pour",
"Chat Lorebook for": "Chat Lorebook pour",
- "chat_world_template_txt": "Un World Info sélectionné sera lié à ce chat. Lors de la génération d'une réponse AI,\n il sera combiné avec les entrées des livres de connaissances globaux et de personnages.",
- "Select a World Info file for": "Sélectionnez un fichier d'informations sur le monde pour",
- "Primary Lorebook": "Livre de lore principal",
- "A selected World Info will be bound to this character as its own Lorebook.": "Une information mondiale sélectionnée sera liée à ce personnage comme son propre livre de lore.",
- "When generating an AI reply, it will be combined with the entries from a global World Info selector.": "Lors de la génération d'une réponse de l'IA, elle sera combinée avec les entrées d'un sélecteur d'informations mondiales global.",
- "Exporting a character would also export the selected Lorebook file embedded in the JSON data.": "L'exportation d'un personnage exporterait également le fichier de livre de lore sélectionné incorporé dans les données JSON.",
- "Additional Lorebooks": "Livres de lore supplémentaires",
- "Associate one or more auxillary Lorebooks with this character.": "Associer un ou plusieurs livres de lore auxiliaires à ce personnage.",
+ "chat_world_template_txt": "Un World Info sélectionné sera lié à ce chat. Lors de la génération d'une réponse IA,\n il sera combiné avec les entrées des livres de connaissances globaux et de personnages.",
+ "Select a World Info file for": "Sélectionnez un fichier Word Info pour",
+ "Primary Lorebook": "Lorebook principal",
+ "A selected World Info will be bound to this character as its own Lorebook.": "Un World Info sélectionnée sera liée à ce personnage comme son propre lorebook.",
+ "When generating an AI reply, it will be combined with the entries from a global World Info selector.": "Lors de la génération d'une réponse de l'IA, elle sera combinée avec les entrées d'un sélecteur de World Info global.",
+ "Exporting a character would also export the selected Lorebook file embedded in the JSON data.": "L'exportation d'un personnage exporterait également le fichier de lorebook sélectionné incorporé dans les données JSON.",
+ "Additional Lorebooks": "Lorebooks supplémentaires",
+ "Associate one or more auxillary Lorebooks with this character.": "Associer un ou plusieurs lorebooks auxiliaires à ce personnage.",
"NOTE: These choices are optional and won't be preserved on character export!": "REMARQUE : Ces choix sont facultatifs et ne seront pas conservés lors de l'exportation du personnage !",
"Rename chat file": "Renommer le fichier de discussion",
"Export JSONL chat file": "Exporter le fichier de discussion au format JSONL",
"Download chat as plain text document": "Télécharger la discussion sous forme de document texte brut",
"Delete chat file": "Supprimer le fichier de discussion",
- "Use tag as folder": "Marquer comme dossier",
- "Delete tag": "Supprimer la balise",
+ "Use tag as folder": "Utiliser les tags comme dossier",
+ "Delete tag": "Supprimer le tag'",
"Entry Title/Memo": "Titre de l'entrée/Mémo",
- "WI Entry Status:🔵 Constant🟢 Normal🔗 Vectorized❌ Disabled": "Statut d'entrée WI :\r🔵 Constante\r🟢 Normal\r🔗 Vectorisé\r❌ Désactivé",
"WI_Entry_Status_Constant": "Constante",
"WI_Entry_Status_Normal": "Normale",
"WI_Entry_Status_Vectorized": "Vectorisé",
- "WI_Entry_Status_Disabled": "Désactivé",
- "T_Position": "↑Caractère : avant les définitions de caractères\n↓Caractère : après les définitions de caractères\n↑AN : avant les notes d'auteur\n↓AN : après les notes d'auteur\n@D : à la profondeur",
- "Before Char Defs": "Avant les définitions de caractères",
- "After Char Defs": "Après les définitions de caractères",
+ "T_Position": "↑Personnage : avant les définitions de personnage\n↓Personnage : après les définitions de personnage\n↑AN : avant les notes d'auteur\n↓AN : après les notes d'auteur\n@D : à la profondeur",
+ "Before Char Defs": "Avant les définitions de personnage",
+ "After Char Defs": "Après les définitions de personnage",
"Before EM": "↑EM",
"After EM": "↓EM",
- "Before AN": "Avant AN",
- "After AN": "Après AN",
+ "Before AN": "Avant NA",
+ "After AN": "Après NA",
"at Depth System": "@D ⚙️",
"at Depth User": "@D 👤",
"at Depth AI": "@D 🤖",
"Depth": "Profondeur",
- "Order:": "Commande :",
+ "Order:": "Ordre :",
"Order": "Ordre :",
- "Trigger %:": "Déclenchement %:",
+ "Trigger %:": "% Déclenchement :",
"Probability": "Probabilité",
- "Duplicate world info entry": "Entrée d'informations mondiales en double",
- "Delete world info entry": "Supprimer l'entrée d'informations sur le monde",
+ "Duplicate world info entry": "Dupliquer l'entrée du World Info",
+ "Delete world info entry": "Supprimer l'entrée du World Info",
"Comma separated (required)": "Séparé par des virgules (requis)",
"Primary Keywords": "Mots-clés principaux",
"Keywords or Regexes": "Mots-clés ou expressions régulières",
@@ -982,19 +924,15 @@
"Case-Sensitive": "Sensible à la casse",
"Yes": "Oui",
"No": "Non",
- "Can be used to automatically activate Quick Replies": "Peut être utilisé pour activer automatiquement les réponses rapides",
+ "Can be used to automatically activate Quick Replies": "Peut être utilisé pour activer automatiquement les Quick Replies",
"Automation ID": "ID d'automatisation",
"( None )": "( Aucun )",
"Content": "Contenu",
- "Exclude from recursion": "Exclure de la récursion",
"Prevent further recursion (this entry will not activate others)": "Empêcher toute récursion ultérieure (cette entrée n’en activera pas d’autres)",
- "Delay until recursion (this entry can only be activated on recursive checking)": "Délai jusqu'à récursion (cette entrée ne peut être activée que sur vérification récursive)",
"What this keyword should mean to the AI, sent verbatim": "Ce que ce mot-clé devrait signifier pour l'IA, envoyé textuellement",
- "Filter to Character(s)": "Filtrer par personnage(s)",
- "Character Exclusion": "Exclusion de personnage",
"-- Characters not found --": "-- Personnages non trouvés --",
"Inclusion Group": "Groupe d'inclusion",
- "Inclusion Groups ensure only one entry from a group is activated at a time, if multiple are triggered.Documentation: World Info - Inclusion Group": "Les groupes d'inclusion garantissent qu'une seule entrée d'un groupe est activée à la fois, si plusieurs sont déclenchées.\rPrend en charge plusieurs groupes séparés par des virgules.\r\rDocumentation : Info Monde - Groupe Inclusion",
+ "Inclusion Groups ensure only one entry from a group is activated at a time, if multiple are triggered.Documentation: World Info - Inclusion Group": "Les groupes d'inclusion garantissent qu'une seule entrée d'un groupe est activée à la fois, si plusieurs sont déclenchées.\rPrend en charge plusieurs groupes séparés par des virgules.\r\rDocumentation : World Info - Groupe Inclusion",
"Prioritize this entry: When checked, this entry is prioritized out of all selections.If multiple are prioritized, the one with the highest 'Order' is chosen.": "Prioriser cette entrée : Lorsque cette case est cochée, cette entrée est prioritaire parmi toutes les sélections.\rSi plusieurs sont prioritaires, celui avec l'ordre le plus élevé est choisi.",
"Only one entry with the same label will be activated": "Seule une entrée avec la même étiquette sera activée",
"A relative likelihood of entry activation within the group": "Une probabilité relative d’activation d’entrée au sein du groupe",
@@ -1002,21 +940,20 @@
"Selective": "Sélectif",
"Use Probability": "Utiliser la probabilité",
"Add Memo": "Ajouter un mémo",
- "Text or token ids": "Texte ou [identifiants de jeton]",
+ "Text or token ids": "Texte ou [token ids]",
"close": "fermer",
"prompt_manager_edit": "Modifier",
"prompt_manager_name": "Nom",
- "A name for this prompt.": "Un nom pour cette invite.",
+ "A name for this prompt.": "Un nom pour ce prompt.",
"To whom this message will be attributed.": "À qui ce message sera attribué.",
"AI Assistant": "Assistant IA",
"prompt_manager_position": "Position",
- "Injection position. Next to other prompts (relative) or in-chat (absolute).": "Position d'injection. À côté d’autres invites (relatives) ou dans le chat (absolues).",
"prompt_manager_relative": "Relatif",
"prompt_manager_depth": "Profondeur",
"Injection depth. 0 = after the last message, 1 = before the last message, etc.": "Profondeur d'injection. 0 = après le dernier message, 1 = avant le dernier message, etc.",
- "Prompt": "Inciter",
- "The prompt to be sent.": "L'invite à envoyer.",
- "This prompt cannot be overridden by character cards, even if overrides are preferred.": "Cette invite ne peut pas être annulée par les cartes de personnage, même si les remplacements sont préférés.",
+ "Prompt": "Prompt",
+ "The prompt to be sent.": "Le prompt à envoyer.",
+ "This prompt cannot be overridden by character cards, even if overrides are preferred.": "Ce prompt ne peut pas être remplacé par les cartes de personnage, même si les remplacements sont préférés.",
"prompt_manager_forbid_overrides": "Interdire les remplacements",
"reset": "réinitialiser",
"save": "sauvegarder",
@@ -1025,13 +962,12 @@
"Translate message": "Traduire le message",
"Generate Image": "Générer une image",
"Narrate": "Narrer",
- "Exclude message from prompts": "Exclure le message des invitations",
- "Include message in prompts": "Inclure le message dans les invitations",
+ "Exclude message from prompts": "Exclure le message des prompts",
+ "Include message in prompts": "Inclure le message dans les prompts",
"Embed file or image": "Intégrer un fichier ou une image",
"Create checkpoint": "Créer un point de contrôle",
"Create Branch": "Créer une branche",
"Copy": "Copier",
- "Open checkpoint chat": "Ouvrir le chat du point de contrôle",
"Edit": "Modifier",
"Confirm": "Confirmer",
"Copy this message": "Copier ce message",
@@ -1039,28 +975,25 @@
"Move message up": "Déplacer le message vers le haut",
"Move message down": "Déplacer le message vers le bas",
"Enlarge": "Agrandir",
- "Welcome to SillyTavern!": "Bienvenue à SillyTaverne !",
- "welcome_message_part_1": "Lis le",
+ "Welcome to SillyTavern!": "Bienvenue sur SillyTavern !",
+ "welcome_message_part_1": "Lisez la",
"welcome_message_part_2": "Documentation officielle",
"welcome_message_part_3": null,
- "welcome_message_part_4": "Taper",
+ "welcome_message_part_4": "Tapez",
"welcome_message_part_5": "dans le chat pour les commandes et les macros.",
- "welcome_message_part_6": "Rejoins",
- "Discord server": "Serveur Discorde",
- "welcome_message_part_7": "pour infos et annonces.",
+ "welcome_message_part_6": "Rejoignez le",
+ "Discord server": "Serveur Discord",
+ "welcome_message_part_7": "pour les infos et annonces.",
"SillyTavern is aimed at advanced users.": "SillyTavern s'adresse aux utilisateurs avancés.",
- "If you're new to this, enable the simplified UI mode below.": "Si vous êtes nouveau dans ce domaine, activez le mode d'interface utilisateur simplifié ci-dessous.",
- "Change it later in the 'User Settings' panel.": "Modifiez-le plus tard dans le panneau « Paramètres utilisateur ».",
- "Enable simple UI mode": "Activer le mode interface utilisateur simple",
- "Looking for AI characters?": "Vous recherchez des personnages IA ?",
+ "Looking for AI characters?": "Vous recherchez des personnages ?",
"onboarding_import": "Importer",
- "from supported sources or view": "à partir de sources prises en charge ou afficher",
+ "from supported sources or view": "à partir de sources prises en charge ou afficher des",
"Sample characters": "Exemples de personnages",
- "Your Persona": "Votre personnalité",
- "Before you get started, you must select a persona name.": "Avant de commencer, vous devez sélectionner un nom de personnage.",
- "welcome_message_part_8": "Ceci peut être modifié à tout moment via le",
- "welcome_message_part_9": "icône.",
- "Persona Name:": "Nom du personnage :",
+ "Your Persona": "Votre persona",
+ "Before you get started, you must select a persona name.": "Avant de commencer, vous devez sélectionner un nom pour votre persona.",
+ "welcome_message_part_8": "Ceci peut être modifié à tout moment via l'icône",
+ "welcome_message_part_9": ".",
+ "Persona Name:": "Nom de la persona :",
"Temporarily disable automatic replies from this character": "Désactiver temporairement les réponses automatiques de ce personnage",
"Enable automatic replies from this character": "Activer les réponses automatiques de ce personnage",
"Trigger a message from this character": "Déclencher un message de ce personnage",
@@ -1071,7 +1004,6 @@
"Add to group": "Ajouter au groupe",
"Alternate Greetings": "Salutations alternatives",
"Alternate_Greetings_desc": "Ceux-ci seront affichés sous forme de balayages sur le premier message lors du démarrage d’une nouvelle discussion.\n Les membres du groupe peuvent en sélectionner un pour lancer la conversation.",
- "Alternate Greetings Hint": "Cliquez sur le bouton pour commencer !",
"(This will be the first message from the character that starts every chat)": "(Ce sera le premier message du personnage qui démarre chaque discussion)",
"Forbid Media Override explanation": "Capacité du personnage/groupe actuel à utiliser des médias externes dans les discussions.",
"Forbid Media Override subtitle": "Médias : images, vidéos, audio. Externe : non hébergé sur le serveur local.",
@@ -1081,46 +1013,46 @@
"Remove the file": "Supprimer le fichier",
"Unique to this chat": "Unique à ce chat",
"Checkpoints inherit the Note from their parent, and can be changed individually after that.": "Les points de contrôle héritent de la note de leur parent et peuvent ensuite être modifiés individuellement.",
- "Include in World Info Scanning": "Inclure dans l'analyse des informations mondiales",
- "Before Main Prompt / Story String": "Avant l'invite principale/la chaîne d'histoire",
- "After Main Prompt / Story String": "Après l'invite principale/la chaîne d'histoire",
+ "Include in World Info Scanning": "Inclure dans l'analyse des Wold Info",
+ "Before Main Prompt / Story String": "Avant le prompt principal/la chaîne d'histoire",
+ "After Main Prompt / Story String": "Après le prompt principal/la chaîne d'histoire",
"as": "comme",
"Insertion Frequency": "Fréquence d'insertion",
"(0 = Disable, 1 = Always)": "(0 = Désactiver, 1 = Toujours)",
"User inputs until next insertion:": "Saisies de l'utilisateur jusqu'à la prochaine insertion :",
- "Character Author's Note (Private)": "Note de l'auteur du personnage (privé)",
+ "Character Author's Note (Private)": "Note d'auteur du personnage (privé)",
"Won't be shared with the character card on export.": "Ne sera pas partagé avec la carte de personnage lors de l'exportation.",
- "Will be automatically added as the author's note for this character. Will be used in groups, but can't be modified when a group chat is open.": "Sera automatiquement ajouté comme note de l'auteur pour ce personnage. Sera utilisé en groupe, mais\n ne peut pas être modifié lorsqu'une discussion de groupe est ouverte.",
- "Use character author's note": "Utiliser la note de l'auteur du personnage",
- "Replace Author's Note": "Remplacer la note de l'auteur",
- "Default Author's Note": "Note de l'auteur par défaut",
- "Will be automatically added as the Author's Note for all new chats.": "Sera automatiquement ajouté en tant que note de l'auteur pour toutes les nouvelles discussions.",
+ "Will be automatically added as the author's note for this character. Will be used in groups, but can't be modified when a group chat is open.": "Sera automatiquement ajouté comme note d'auteur pour ce personnage. Sera utilisé en groupe, mais\n ne peut pas être modifié lorsqu'une discussion de groupe est ouverte.",
+ "Use character author's note": "Utiliser la note d'auteur du personnage",
+ "Replace Author's Note": "Remplacer la note d'auteur",
+ "Default Author's Note": "Note d'auteur par défaut",
+ "Will be automatically added as the Author's Note for all new chats.": "Sera automatiquement ajouté en tant que note d'auteur pour toutes les nouvelles discussions.",
"Chat CFG": "Chat CFG",
"1 = disabled": "1 = désactivé",
"write short replies, write replies using past tense": "rédiger des réponses courtes, rédiger des réponses en utilisant le passé",
- "Positive Prompt": "Invite positive",
- "Use character CFG scales": "Utiliser les échelles CFG de caractères",
- "Character CFG": "Personnage CFG",
+ "Positive Prompt": "Prompt positif",
+ "Use character CFG scales": "Utiliser les échelles CFG du personnage",
+ "Character CFG": "CFG du personnage",
"Will be automatically added as the CFG for this character.": "Sera automatiquement ajouté en tant que CFG pour ce personnage.",
- "Global CFG": "CFG mondial",
+ "Global CFG": "CFG global",
"Will be used as the default CFG options for every chat unless overridden.": "Sera utilisé comme options CFG par défaut pour chaque discussion, sauf remplacement.",
- "CFG Prompt Cascading": "Invite CFG en cascade",
- "Combine positive/negative prompts from other boxes.": "Combinez les invites positives/négatives d’autres cases.",
- "For example, ticking the chat, global, and character boxes combine all negative prompts into a comma-separated string.": "Par exemple, en cochant les cases Chat, Global et Caractère, vous combinez toutes les invites négatives dans une chaîne séparée par des virgules.",
+ "CFG Prompt Cascading": "Prompt CFG en cascade",
+ "Combine positive/negative prompts from other boxes.": "Combinez les prompts positifs/négatifs d’autres cases.",
+ "For example, ticking the chat, global, and character boxes combine all negative prompts into a comma-separated string.": "Par exemple, en cochant les cases Chat, Global et Personnage, vous combinez tout les prompt négatifs dans une chaîne séparée par des virgules.",
"Always Include": "Toujours inclure",
"Chat Negatives": "Points négatifs du chat",
- "Character Negatives": "Points négatifs sur les personnages",
- "Global Negatives": "Points négatifs mondiaux",
+ "Character Negatives": "Points négatifs du personnage",
+ "Global Negatives": "Points négatifs globaux",
"Custom Separator:": "Séparateur personnalisé :",
"Insertion Depth:": "Profondeur d'insertion :",
- "Token Probabilities": "Probabilités de jetons",
- "Select a token to see alternatives considered by the AI.": "Sélectionnez un jeton pour voir les alternatives envisagées par l'IA.",
+ "Token Probabilities": "Probabilités des tokens",
+ "Select a token to see alternatives considered by the AI.": "Sélectionnez un token pour voir les alternatives envisagées par l'IA.",
"Not connected to API!": "Non connecté à l'API !",
- "Type a message, or /? for help": "Tapez un message, ou /? pour aider",
+ "Type a message, or /? for help": "Tapez un message, ou /? pour l'aide",
"Continue script execution": "Continuer l'exécution du script",
"Pause script execution": "Suspendre l'exécution du script",
"Abort script execution": "Abandonner l'exécution du script",
- "Abort request": "Annuler la demande",
+ "Abort request": "Annuler la requête",
"Continue the last message": "Continuer le dernier message",
"Send a message": "Envoyer un message",
"Close chat": "Fermer la discussion",
@@ -1135,7 +1067,7 @@
"Ask AI to write your message for you": "Demander à l'IA d'écrire votre message pour vous",
"Impersonate": "Usurper l'identité",
"Continue": "Continuer",
- "Bind user name to that avatar": "Lier le nom d'utilisateur à cet avatar",
+ "Bind user name to that avatar": "Lier le nom d'utilisateur à cette persona",
"Change persona image": "Changer l'image de la persona",
"Select this as default persona for the new chats.": "Sélectionner ceci comme persona par défaut pour les nouvelles discussions.",
"Delete persona": "Supprimer la persona",
@@ -1144,8 +1076,8 @@
"These characters are the finalists of character design contests and have remarkable quality.": "Ces personnages sont les finalistes des concours de conception de personnages et ont une qualité remarquable.",
"Featured Characters": "Personnages en vedette",
"Attach a File": "Joindre un fichier",
- "Open Data Bank": "Banque de données ouverte",
- "Enter a URL or the ID of a Fandom wiki page to scrape:": "Entrez une URL ou l'ID d'une page wiki Fandom à récupérer :",
+ "Open Data Bank": "Ouvrir la banque de données",
+ "Enter a URL or the ID of a Fandom wiki page to scrape:": "Entrez une URL ou l'ID d'une page wiki Fandom à :",
"Examples:": "Exemples:",
"Example:": "Exemple:",
"Single file": "Un seul fichier",
@@ -1168,44 +1100,40 @@
"Global Attachments": "Pièces jointes globales",
"These files are available for all characters in all chats.": "Ces fichiers sont disponibles pour tous les personnages de toutes les discussions.",
"Character Attachments": "Pièces jointes aux personnages",
- "These files are available the current character in all chats they are in.": "Ces fichiers sont disponibles pour le personnage actuel dans toutes les discussions dans lesquelles ils se trouvent.",
"Saved locally. Not exported.": "Enregistré localement. Non exporté.",
"Chat Attachments": "Pièces jointes au chat",
- "These files are available to all characters in the current chat.": "Ces fichiers sont disponibles pour tous les personnages du chat en cours.",
- "Enter a base URL of the MediaWiki to scrape.": "Entrez une URL de base du MediaWiki à gratter.",
+ "Enter a base URL of the MediaWiki to scrape.": "Entrez une URL de base du MediaWiki à récupérer.",
"Don't include the page name!": "N'incluez pas le nom de la page !",
"Enter web URLs to scrape (one per line):": "Entrez les URL Web à récupérer (une par ligne) :",
- "Enter a video URL to download its transcript.": "Saisissez l'URL ou l'identifiant de la vidéo pour télécharger sa transcription.",
- "Expression API": "Locale\nSuppléments\nLLM",
+ "Enter a video URL to download its transcript.": "Saisissez l'URL de la vidéo pour télécharger sa transcription.",
"ext_sum_with": "Résumez avec :",
"ext_sum_main_api": "API principale",
"ext_sum_current_summary": "Résumé actuel :",
- "ext_sum_restore_previous": "Rétablir précédente",
+ "ext_sum_restore_previous": "Rétablir le précédente",
"ext_sum_memory_placeholder": "Le résumé sera généré ici...",
- "Trigger a summary update right now.": "Résumez maintenant",
"ext_sum_force_text": "Résumez maintenant",
"Disable automatic summary updates. While paused, the summary remains as-is. You can still force an update by pressing the Summarize now button (which is only available with the Main API).": "Désactivez les mises à jour automatiques du résumé. Pendant la pause, le résumé reste tel quel. Vous pouvez toujours forcer une mise à jour en appuyant sur le bouton Résumer maintenant (qui n'est disponible qu'avec l'API principale).",
"ext_sum_pause": "Pause",
- "Omit World Info and Author's Note from text to be summarized. Only has an effect when using the Main API. The Extras API always omits WI/AN.": "Omettre World Info et la note de l'auteur du texte à résumer. N'a d'effet que lors de l'utilisation de l'API principale. L'API Extras omet toujours WI/AN.",
- "ext_sum_no_wi_an": "Pas de WI/AN",
- "ext_sum_settings_tip": "Modifier l'invite de résumé, la position d'insertion, etc.",
- "ext_sum_settings": "Paramètres récapitulatifs",
- "ext_sum_prompt_builder": "Générateur d'invites",
- "ext_sum_prompt_builder_1_desc": "L'extension créera sa propre invite en utilisant des messages qui n'ont pas encore été résumés. Bloque le chat jusqu'à ce que le résumé soit généré.",
+ "Omit World Info and Author's Note from text to be summarized. Only has an effect when using the Main API. The Extras API always omits WI/AN.": "Omettre World Info et la note d'auteur du texte à résumer. N'a d'effet que lors de l'utilisation de l'API principale. L'API Extras omet toujours WI/AN.",
+ "ext_sum_no_wi_an": "Pas de WI/NA",
+ "ext_sum_settings_tip": "Modifier le prompt de résumé, la position d'insertion, etc.",
+ "ext_sum_settings": "Paramètres du résumé",
+ "ext_sum_prompt_builder": "Générateur de prompt",
+ "ext_sum_prompt_builder_1_desc": "L'extension créera son propre prompt en utilisant des messages qui n'ont pas encore été résumés. Bloque le chat jusqu'à ce que le résumé soit généré.",
"ext_sum_prompt_builder_1": "Brut, bloquant",
- "ext_sum_prompt_builder_2_desc": "L'extension créera sa propre invite en utilisant des messages qui n'ont pas encore été résumés. Ne bloque pas le chat pendant la génération du résumé. Tous les backends ne prennent pas en charge ce mode.",
+ "ext_sum_prompt_builder_2_desc": "L'extension créera son propre prompt en utilisant des messages qui n'ont pas encore été résumés. Ne bloque pas le chat pendant la génération du résumé. Tous les backends ne prennent pas en charge ce mode.",
"ext_sum_prompt_builder_2": "Brut, non bloquant",
- "ext_sum_prompt_builder_3_desc": "L'extension utilisera le générateur d'invite principal standard et y ajoutera la demande de résumé comme dernier message système.",
+ "ext_sum_prompt_builder_3_desc": "L'extension utilisera le générateur de prompt principal standard et y ajoutera la demande de résumé comme dernier message système.",
"ext_sum_prompt_builder_3": "Classique, bloquant",
- "Summary Prompt": "Invite de résumé",
- "ext_sum_restore_default_prompt_tip": "Restaurer l'invite par défaut",
- "ext_sum_prompt_placeholder": "Cette invite sera envoyée à AI pour demander la génération du résumé. {{words}} sera résolu par le paramètre \"Nombre de mots\".",
+ "Summary Prompt": "Prompt de résumé",
+ "ext_sum_restore_default_prompt_tip": "Restaurer le prompt par défaut",
+ "ext_sum_prompt_placeholder": "Ce prompt sera envoyée à l'IA pour demander la génération du résumé. {{words}} sera résolu par le paramètre \"Nombre de mots\".",
"ext_sum_target_length_1": "Longueur cible du résumé",
"ext_sum_target_length_2": null,
"ext_sum_target_length_3": "mots)",
"ext_sum_api_response_length_1": "Longueur de réponse de l'API",
"ext_sum_api_response_length_2": null,
- "ext_sum_api_response_length_3": "jetons)",
+ "ext_sum_api_response_length_3": "tokens)",
"ext_sum_0_default": "0 = par défaut",
"ext_sum_raw_max_msg": "[Brut] Nombre maximum de messages par requête",
"ext_sum_0_unlimited": "0 = illimité",
@@ -1225,12 +1153,12 @@
"ext_regex_new_global_script": "+ Mondial",
"ext_regex_new_scoped_script": "+ Portée",
"ext_regex_import_script": "Importer",
- "ext_regex_global_scripts": "Scripts mondiaux",
+ "ext_regex_global_scripts": "Scripts globaux",
"ext_regex_global_scripts_desc": "Disponible pour tous les personnages. Enregistré dans les paramètres locaux.",
"ext_regex_scoped_scripts": "Scripts étendus",
"ext_regex_scoped_scripts_desc": "Uniquement disponible pour ce personnage. Enregistré dans les données de la carte.",
"Regex Editor": "Éditeur d'expressions régulières",
- "Test Mode": "Mode d'essai",
+ "Test Mode": "Mode de test",
"ext_regex_desc": "Regex est un outil pour rechercher/remplacer des chaînes à l'aide d'expressions régulières. Si vous souhaitez en savoir plus, cliquez sur ? à côté du titre.",
"Input": "Saisir",
"ext_regex_test_input_placeholder": "Écrivez ici...",
@@ -1246,17 +1174,16 @@
"ext_regex_user_input": "Entrée de l'utilisateur",
"ext_regex_ai_output": "Sortie IA",
"Slash Commands": "Commandes barre oblique",
- "ext_regex_min_depth_desc": "Lorsqu'il est appliqué aux invites ou à l'affichage, n'affecte que les messages d'au moins N niveaux de profondeur. 0 = dernier message, 1 = avant-dernier message, etc. Ne compte que les entrées WI @Depth et les messages utilisables, c'est-à-dire non cachés ou système.",
+ "ext_regex_min_depth_desc": "Lorsqu'il est appliqué aux prompts ou à l'affichage, n'affecte que les messages d'au moins N niveaux de profondeur. 0 = dernier message, 1 = avant-dernier message, etc. Ne compte que les entrées WI @Depth et les messages utilisables, c'est-à-dire non cachés ou système.",
"Min Depth": "Profondeur minimale",
"ext_regex_min_depth_placeholder": "Illimité",
- "ext_regex_max_depth_desc": "Lorsqu'il est appliqué aux invites ou à l'affichage, n'affecte que les messages en profondeur ne dépassant pas N niveaux. 0 = dernier message, 1 = avant-dernier message, etc. Ne compte que les entrées WI @Depth et les messages utilisables, c'est-à-dire non cachés ou système.",
+ "ext_regex_max_depth_desc": "Lorsqu'il est appliqué aux prompts ou à l'affichage, n'affecte que les messages en profondeur ne dépassant pas N niveaux. 0 = dernier message, 1 = avant-dernier message, etc. Ne compte que les entrées WI @Depth et les messages utilisables, c'est-à-dire non cachés ou système.",
"ext_regex_other_options": "Autres options",
"Only Format Display": "Format d'affichage uniquement",
"ext_regex_only_format_prompt_desc": "L'historique des discussions ne changera pas, seule l'invite lors de l'envoi de la demande (lors de la génération).",
"Only Format Prompt (?)": "Uniquement l'invite de formatage",
"Run On Edit": "Exécuter sur Modifier",
"ext_regex_substitute_regex_desc": "Remplacez {{macros}} dans Find Regex avant de l'exécuter",
- "Substitute Regex": "Remplacer l'expression régulière",
"ext_regex_import_target": "Importer vers :",
"ext_regex_disable_script": "Désactiver le script",
"ext_regex_enable_script": "Activer le script",
@@ -1265,23 +1192,21 @@
"ext_regex_move_to_scoped": "Passer aux scripts étendus",
"ext_regex_export_script": "Exporter le script",
"ext_regex_delete_script": "Supprimer le script",
- "Trigger Stable Diffusion": "Déclencher une diffusion stable",
- "sd_Yourself": "Toi-même",
- "sd_Your_Face": "Ton visage",
+ "Trigger Stable Diffusion": "Déclencher Stable Diffusion",
+ "sd_Yourself": "Vous-même",
+ "sd_Your_Face": "Votre visage",
"sd_Me": "Moi",
"sd_The_Whole_Story": "Toute l'histoire",
"sd_The_Last_Message": "Le dernier message",
"sd_Raw_Last_Message": "Dernier message brut",
"sd_Background": "Arrière-plan",
"Image Generation": "Génération d'images",
- "sd_refine_mode": "Autoriser la modification manuelle des invites avant de les envoyer à l'API de génération",
- "sd_refine_mode_txt": "Modifier les invites avant la génération",
+ "sd_refine_mode": "Autoriser la modification manuelle des prompts avant de les envoyer à l'API de génération",
+ "sd_refine_mode_txt": "Modifier les prompts avant la génération",
"sd_interactive_mode": "Générez automatiquement des images lors de l'envoi de messages tels que « envoyez-moi une photo de chat ».",
"sd_interactive_mode_txt": "Mode interactif",
- "sd_multimodal_captioning": "Utilisez le sous-titrage multimodal pour générer des invites pour les portraits des utilisateurs et des personnages en fonction de leurs avatars.",
+ "sd_multimodal_captioning": "Utilisez le sous-titrage multimodal pour générer des prompts pour les portraits des utilisateurs et des personnages en fonction de leurs avatars.",
"sd_multimodal_captioning_txt": "Utiliser le sous-titrage multimodal pour les portraits",
- "sd_expand": "Étendre automatiquement les invites à l'aide du modèle de génération de texte",
- "sd_expand_txt": "Invites d’amélioration automatique",
"sd_snap": "Requêtes de génération d'instantanés avec un rapport hauteur/largeur forcé (portraits, arrière-plans) à la résolution connue la plus proche, tout en essayant de préserver le nombre absolu de pixels (recommandé pour SDXL).",
"sd_snap_txt": "Résolutions ajustées automatiquement",
"Source": "Source",
@@ -1297,56 +1222,53 @@
"The server must be accessible from the SillyTavern host machine.": "Le serveur doit être accessible depuis la machine hôte de SillyTavern.",
"Hint: Save an API key in AI Horde API settings to use it here.": "Astuce : enregistrez une clé API dans les paramètres de l'API AI Horde pour l'utiliser ici.",
"Allow NSFW images from Horde": "Autoriser les images NSFW de la Horde",
- "Sanitize prompts (recommended)": "Désinfecter les invites (recommandé)",
+ "Sanitize prompts (recommended)": "Désinfecter les prompts (recommandé)",
"Automatically adjust generation parameters to ensure free image generations.": "Ajustez automatiquement les paramètres de génération pour garantir des générations d’images gratuites.",
- "Avoid spending Anlas": "Évitez de dépenser Anlas",
- "Opus tier": "(Niveau Opus)",
- "View my Anlas": "Voir mon Anlas",
+ "Avoid spending Anlas": "Évitez de dépenser des Anlas",
+ "Opus tier": "(Tier Opus)",
+ "View my Anlas": "Voir mes Anlas",
"These settings only apply to DALL-E 3": "Ces paramètres s'appliquent uniquement à DALL-E 3",
- "Image Style": "Style d'image",
- "Image Quality": "Qualité d'image",
+ "Image Style": "Style de l'image",
+ "Image Quality": "Qualité de l'image",
"Standard": "Standard",
"HD": "HD",
"sd_comfy_url": "Exemple : {{comfy_url}}",
"Open workflow editor": "Ouvrir l'éditeur de workflow",
- "Create new workflow": "Créer un nouveau flux de travail",
- "Delete workflow": "Supprimer le flux de travail",
+ "Create new workflow": "Créer un nouveau workflow",
+ "Delete workflow": "Supprimer le workflow",
"Enhance": "Améliorer",
- "Refine": "Affiner",
"Decrisper": "Décrypteur",
"Sampling steps": "Étapes d'échantillonnage ()",
- "Width": "Largeur ()",
- "Height": "Hauteur ()",
+ "Width": "Largeur",
+ "Height": "Hauteur",
"Resolution": "Résolution",
"Model": "Modèle",
"Sampling method": "Méthode d'échantillonnage",
- "Karras (not all samplers supported)": "Karras (tous les échantillonneurs ne sont pas pris en charge)",
"SMEA versions of samplers are modified to perform better at high resolution.": "Les versions SMEA des échantillonneurs sont modifiées pour mieux fonctionner à haute résolution.",
- "SMEA": "PMEA",
+ "SMEA": "SDMEA",
"DYN variants of SMEA samplers often lead to more varied output, but may fail at very high resolutions.": "Les variantes DYN des échantillonneurs SMEA conduisent souvent à des sorties plus variées, mais peuvent échouer à des résolutions très élevées.",
"DYN": "DYN",
"Scheduler": "Planificateur",
"Restore Faces": "Restaurer les visages",
- "Hires. Fix": "Embauches. Réparer",
+ "Hires. Fix": "Hires. Fix",
"Upscaler": "Upscaleur",
- "Upscale by": "Mise à niveau par",
+ "Upscale by": "Upscale par",
"Denoising strength": "Force de débruitage",
- "Hires steps (2nd pass)": "Embauches marches (2ème passage)",
- "Preset for prompt prefix and negative prompt": "Préréglage pour le préfixe d'invite et l'invite négative",
+ "Hires steps (2nd pass)": "Nombre d'étapes Hires (2ème passage)",
+ "Preset for prompt prefix and negative prompt": "Preset pour le préfixe de prompt et le prompt négatif",
"Style": "Style",
"Save style": "Enregistrer le style",
"Delete style": "Supprimer le style",
- "Common prompt prefix": "Préfixe d'invite commun",
- "sd_prompt_prefix_placeholder": "Utilisez {prompt} pour spécifier où l'invite générée sera insérée",
- "Negative common prompt prefix": "Préfixe d'invite commun négatif",
- "Character-specific prompt prefix": "Préfixe d'invite spécifique au caractère",
+ "Common prompt prefix": "Préfixe de prompt commun",
+ "sd_prompt_prefix_placeholder": "Utilisez {prompt} pour spécifier où le prompt générée sera insérée",
+ "Negative common prompt prefix": "Préfixe de prompt commun négatif",
+ "Character-specific prompt prefix": "Préfixe de prompt spécifique au personnage",
"Won't be used in groups.": "Ne sera pas utilisé en groupe.",
- "sd_character_prompt_placeholder": "Toutes les caractéristiques qui décrivent le personnage actuellement sélectionné. Sera ajouté après un préfixe d'invite commun.\nExemple : femme, yeux verts, cheveux bruns, chemise rose",
- "Character-specific negative prompt prefix": "Préfixe d'invite négatif spécifique au caractère",
- "sd_character_negative_prompt_placeholder": "Toutes les caractéristiques qui ne devraient pas apparaître pour le personnage sélectionné. Sera ajouté après un préfixe d’invite commun négatif.\nExemple : bijoux, chaussures, lunettes",
+ "sd_character_prompt_placeholder": "Toutes les caractéristiques qui décrivent le personnage actuellement sélectionné. Sera ajouté après un préfixe de prompt commun.\nExemple : femme, yeux verts, cheveux bruns, chemise rose",
+ "Character-specific negative prompt prefix": "Préfixe de prompt négatif spécifique au personnage",
+ "sd_character_negative_prompt_placeholder": "Toutes les caractéristiques qui ne devraient pas apparaître pour le personnage sélectionné. Sera ajouté après un préfixe dde prompt commun négatif.\nExemple : bijoux, chaussures, lunettes",
"Shareable": "Partageable",
- "Image Prompt Templates": "Modèles d'invite d'image",
- "Vectors Model Warning": "Il est recommandé de purger les vecteurs lors du changement de modèle en cours de discussion. Sinon, cela conduira à des résultats médiocres.",
+ "Image Prompt Templates": "Modèles de prompt d'image",
"Translate files into English before processing": "Traduire les fichiers en anglais avant le traitement",
"Manager Users": "gérer les utilisateurs",
"New User": "Nouvel utilisateur",
@@ -1360,7 +1282,6 @@
"Current Password:": "Mot de passe actuel:",
"New Password:": "Nouveau mot de passe:",
"Confirm New Password:": "Confirmer le nouveau mot de passe:",
- "Debug Warning": "Les fonctions de cette catégorie sont réservées aux utilisateurs avancés. Ne cliquez sur rien si vous n'êtes pas sûr des conséquences.",
"Execute": "Exécuter",
"Are you sure you want to delete this user?": "Êtes-vous sûr de vouloir supprimer cet utilisateur ?",
"Deleting:": "Suppression :",
@@ -1368,18 +1289,18 @@
"Warning:": "Avertissement:",
"This action is irreversible.": "Cette action est irréversible.",
"Type the user's handle below to confirm:": "Tapez le pseudo de l'utilisateur ci-dessous pour confirmer :",
- "Import Characters": "Importer des caractères",
+ "Import Characters": "Importer des personnages",
"Enter the URL of the content to import": "Saisissez l'URL du contenu à importer",
"Supported sources:": "Sources prises en charge :",
- "char_import_1": "Personnage Chub (lien direct ou identifiant)",
+ "char_import_1": "Personnage de Chub (lien direct ou identifiant)",
"char_import_example": "Exemple:",
- "char_import_2": "Livre de traditions de Chub (lien direct ou ID)",
- "char_import_3": "Personnage JanitorAI (lien direct ou UUID)",
- "char_import_4": "Caractère Pygmalion.chat (lien direct ou UUID)",
- "char_import_5": "Personnage AICharacterCard.com (lien direct ou identifiant)",
+ "char_import_2": "Lorebook de Chub (lien direct ou ID)",
+ "char_import_3": "Personnage de JanitorAI (lien direct ou UUID)",
+ "char_import_4": "Personnage de Pygmalion.chat (lien direct ou UUID)",
+ "char_import_5": "Personnage de AICharacterCards.com (lien direct ou identifiant)",
"char_import_6": "Lien PNG direct (voir",
"char_import_7": "pour les hôtes autorisés)",
- "char_import_8": "Personnage RisuRealm (lien direct)",
+ "char_import_8": "Personnage de RisuRealm (lien direct)",
"Supports importing multiple characters.": "Prend en charge l'importation de plusieurs caractères.",
"Write each URL or ID into a new line.": "Écrivez chaque URL ou identifiant dans une nouvelle ligne.",
"Export for character": "Exportation pour le personnage",
@@ -1394,10 +1315,10 @@
"New prompt": "Nouvelle invitation",
"Prompts": "Invitations",
"Total Tokens:": "Tokens totaux :",
- "prompt_manager_tokens": "Jetons",
+ "prompt_manager_tokens": "Tokens",
"Are you sure you want to reset your settings to factory defaults?": "Êtes-vous sûr de vouloir réinitialiser vos paramètres aux valeurs d'usine par défaut ?",
"Don't forget to save a snapshot of your settings before proceeding.": "N'oubliez pas d'enregistrer un instantané de vos paramètres avant de continuer.",
- "Settings Snapshots": "Paramètres instantanés",
+ "Settings Snapshots": "Paramètres des instantanés",
"Record a snapshot of your current settings.": "Enregistrez un instantané de vos paramètres actuels.",
"Make a Snapshot": "Faire un instantané",
"Restore this snapshot": "Restaurer cet instantané",
@@ -1425,19 +1346,709 @@
"Want to update?": "Envie de mettre à jour ?",
"How to start chatting?": "Comment commencer à discuter ?",
"Click _space": "Cliquez sur",
- "and select a": "et sélectionnez un",
- "Chat API": "API de chat",
"and pick a character.": "et choisissez un personnage.",
- "You can browse a list of bundled characters in the": "Vous pouvez parcourir une liste de caractères regroupés dans le",
"Download Extensions & Assets": "Télécharger des extensions et des ressources",
- "menu within": "menu à l'intérieur",
+ "menu within": "à l'intérieur de",
"Confused or lost?": "Confus ou perdu ?",
"click these icons!": "cliquez sur ces icônes !",
"in the chat bar": "dans la barre de chat",
- "SillyTavern Documentation Site": "Site de documentation de SillyTavern",
- "Extras Installation Guide": "Guide d'installation des extras",
+ "SillyTavern Documentation Site": "Documentation de SillyTavern",
"Still have questions?": "Vous avez encore des questions ?",
"Join the SillyTavern Discord": "Rejoignez le Discord de SillyTavern",
"Post a GitHub issue": "Poster un problème sur GitHub",
- "Contact the developers": "Contacter les développeurs"
+ "Contact the developers": "Contacter les développeurs",
+ "Rename current preset": "Renommer le preset actuel",
+ "Linear": "Linéaire",
+ "Quad": "Quad",
+ "Conf": "Conf",
+ "Middle-out Transform": "Transformation Middle-out",
+ "Auto": "Auto",
+ "Allow": "Autoriser",
+ "Forbid": "Interdire",
+ "Auxiliary": "Auxiliaire",
+ "Post-History Instructions": "Instructions post-histoire",
+ "Unified Sampling": "Échantillonnage unifié",
+ "Top nsigma": "Top nsigma",
+ "Exclude Top Choices (XTC)": "Exclure les Top Choices (XTC)",
+ "Threshold": "Seuil",
+ "A greedy, brute-force algorithm used in LLM sampling to find the most likely sequence of words or tokens. It expands multiple candidate sequences at once, maintaining a fixed number (beam width) of top sequences at each step.": "Un algorithme gourmand et brutal utilisé dans l'échantillonnage LLM pour trouver la séquence de mots ou de tokens la plus probable. Il développe plusieurs séquences candidates à la fois, en conservant un nombre fixe (largeur du faisceau) de séquences supérieures à chaque étape.",
+ "# of Beams": "# de faisceaux",
+ "The number of sequences generated at each step with Beam Search.": "Le nombre de séquences générées à chaque étape avec Beam Search.",
+ "Penalize sequences based on their length.": "Pénaliser les séquences en fonction de leur longueur.",
+ "Controls the stopping condition for beam search. If checked, the generation stops as soon as there are '# of Beams' sequences. If not checked, a heuristic is applied and the generation is stopped when it's very unlikely to find better candidates.": "Contrôle la condition d'arrêt de la recherche de faisceaux. Si cette case est cochée, la génération s'arrête dès qu'il y a « # of Beams » séquences. Si cette case n'est pas cochée, une heuristique est appliquée et la génération est arrêtée lorsqu'il est très peu probable de trouver de meilleurs candidats.",
+ "Seed_desc": "Une graine aléatoire à utiliser pour obtenir des résultats déterministes et reproductibles. Définir à -1 pour utiliser une graine aléatoire.",
+ "Sampler Order": "Ordre des échantillonneurs",
+ "Aphrodite only. Determines the order of samplers. Skew is always applied post-softmax, so it's not included here.": "Aphrodite seulement. Determines l'ordres des échantillonneurs. L'obliquité est toujours appliquée après le softmax, elle n'est donc pas incluse ici.",
+ "Aphrodite only. Determines the order of samplers.": "Aphrodite seulement. Determines l'ordres des échantillonneurs.",
+ "character_names_none": "Ne jamais ajouter de préfixes de noms de personnages. Peut mal se comporter en groupe, à choisir avec précaution.",
+ "Completion Object": "Completion Object",
+ "enable_functions_desc_1": "Autorise l'utilisation",
+ "enable_functions_desc_2": "outils de fonction",
+ "enable_functions_desc_3": "Peut être utilisé par diverses extensions pour fournir des fonctionnalités supplémentaires.",
+ "Show model thoughts": "Afficher les pensées du modèle",
+ "Display the model's internal thoughts in the response.": "Afficher les pensées internes du modèle dans la réponse.",
+ "Confirm token parsing with": "Confirmer l'analyse des tokens avec",
+ "openai_logit_bias_no_items": "Aucun élément",
+ "api_no_connection": "Pas de connection...",
+ "AI Horde Website": "Site Web de AI Horde",
+ "Generic (OpenAI-compatible) [LM Studio, LiteLLM, etc.]": "Générique (OpenAI-compatible) [LM Studio, LiteLLM, etc.]",
+ "Allow fallback providers": "Autoriser les fournisseurs de secours",
+ "Model ID (optional)": "Model ID (optionnel)",
+ "Featherless Model Selection": "Séléction de modèle Featherless",
+ "category": "categorie",
+ "Top": "Top",
+ "All": "Tous",
+ "class": "Toutes les classes",
+ "No model description": "[Pas de déscription]",
+ "HuggingFace Token": "Token HuggingFace",
+ "Endpoint URL": "URL Endpoint",
+ "Example: https://****.endpoints.huggingface.cloud": "Exemple: https://****.endpoints.huggingface.cloud",
+ "Tabby Model": "Modèle Tabby",
+ "must be set in Tabby's config.yml to switch models.": "doit être défini dans le fichier config.yml de Tabby pour changer de modèle.",
+ "Use an admin API key.": "Utiliser une clé API d'administrateur.",
+ "Derive context size from backend": "Déterminer la taille du contexte à partir du backend",
+ "Custom (OpenAI-compatible)": "Personnalisé (OpenAI-compatible)",
+ "Using a proxy that you're not running yourself is a risk to your data privacy.": "L'utilisation d'un proxy que vous ne gérez pas vous-même présente un risque pour la confidentialité de vos données.",
+ "ANY support requests will be REFUSED if you are using a proxy.": "TOUTE demande d'assistance sera REFUSÉE si vous utilisez un proxy.",
+ "Do not proceed if you do not agree to this!": "Ne continuez pas si vous n'êtes pas d'accord avec cela !",
+ "Claude API Key": "Clé API Claude",
+ "Allow fallback models": "Autoriser les modèles de secours",
+ "NanoGPT API Key": "Clé API NanoGPT",
+ "NanoGPT Model": "Modèle NanoGPT",
+ "DeepSeek API Key": "Clé API DeepSeek",
+ "DeepSeek Model": "Modèle DeepSeek",
+ "Block Entropy API Key": "Clé API Block Entropy",
+ "Select a Model": "Sélectionner un modèle",
+ "Example: http://localhost:1234/v1": "Exemple: http://localhost:1234/v1",
+ "at the end of the URL!": "à la fin de l'URL !",
+ "(Optional)": "(Optionnel)",
+ "Enter a Model ID": "Saisir un ID de modèle",
+ "Example: gpt-3.5-turbo": "Exemple: gpt-3.5-turbo",
+ "prompt_post_processing_none": "Aucun",
+ "prompt_post_processing_merge": "Fusionner les rôles consécutifs",
+ "prompt_post_processing_semi": "Semi-strict (rôles alternés)",
+ "prompt_post_processing_strict": "Strict (l'utilisateur d'abord, rôles alternés)",
+ "01.AI API Key": "01.Clé IA API",
+ "01.AI Model": "01.Modèle IA",
+ "Additional Parameters": "Paramètres supplémentaires",
+ "Import Advanced Formatting settings": "Importer les paramètres de formatage avancé\n\nVous pouvez également fournir des fichiers hérités pour les modèles Instruct et Context..",
+ "Master Import": "Master Import",
+ "Export Advanced Formatting settings": "Exporter les paramètres de formatage avancé",
+ "Master Export": "Master Export",
+ "context_derived": "Dériver des métadonnées du modèle, si possible.",
+ "Select your current Context Template": "Sélectionnez votre modèle de contexte actuel",
+ "Update current template": "Mise à jour du modèle actuel",
+ "Rename current template": "Renommer le modèle actuel",
+ "Save template as": "Enregistrer le modèle sous",
+ "Import template": "Importer un modèle",
+ "Export template": "Exporter un modèle",
+ "Restore current template": "Restaurer le modèle actuel",
+ "Delete the template": "Supprimer le modèle",
+ "Disabling is not recommended.": "La désactivation n'est pas recommandée.",
+ "Separators as Stop Strings": "Séparateurs comme chaînes d'arrêt",
+ "Add Character and User names to a list of stopping strings.": "Ajouter les noms de personnages et d'utilisateurs à une liste de chaînes d'arrêt.",
+ "Names as Stop Strings": "Noms comme chaînes d'arrêt",
+ "context_allow_post_history_instructions": "Inclut les instructions post-historiques à la fin du prompt, si elles sont définies dans la fiche de personnage ET si l'option 'Préférer les instructions de personnage' est activée.\nN'EST PAS RECOMMANDÉ POUR LES MODÈLES DE COMPLÉTION DE TEXTE, CAR IL PEUT ENTRAÎNER DE MAUVAIS RÉSULTATS.",
+ "Allow Post-History Instructions": "Autoriser les instructions post-histoire",
+ "Instruct Template": "Modèle d'instruction",
+ "instruct_derived": "Dériver des métadonnées du modèle, si possible.",
+ "instruct_enabled": "Activer le mode d'instruction",
+ "Select your current Instruct Template": "Sélectionnez votre modèle d'instruction actuel",
+ "Delete template": "Supprimer le modèle",
+ "instruct_template_activation_regex_desc": "Lors de la connexion à une API ou du choix d'un modèle, ce modèle d'instruction est automatiquement activé si le nom du modèle correspond à l'expression régulière fournie.",
+ "Never": "Jamais",
+ "Groups and Past Personas": "Groupes et personas antérieures",
+ "Always": "Toujours",
+ "Instruct Sequences": "Séquences d'instruction",
+ "User Message Sequences": "Séquences de messages d'utilisateurs",
+ "User Prefix": "Préfixe du message de l'utilisateur",
+ "User Suffix": "Suffixe du message de l'utilisateur",
+ "Assistant Message Sequences": "Séquences de messages de l'assistant",
+ "Assistant Prefix": "Préfixe du message de l'assistant",
+ "Assistant Suffix": "Suffixe du message de l'assistant",
+ "System Message Sequences": "Séquences de messages système",
+ "System Prefix": "Préfixe des messages système",
+ "System Suffix": "Suffixe des messages du système",
+ "System Prompt Sequences": "Séquences du prompt système",
+ "Inserted before the first User's message.": "Inséré avant le premier message de l'utilisateur.",
+ "First User Prefix": "Préfixe du premier utilisateur",
+ "instruct_last_input_sequence": "Inséré avant le dernier message de l'utilisateur.",
+ "Last User Prefix": "Préfixe du dernier utilisateur",
+ "sysprompt_enabled": "Activer le prompt système",
+ "Select your current System Prompt": "Sélectionnez votre prompt système actuel",
+ "Update current prompt": "Mise à jour du prompt actuel",
+ "Rename current prompt": "Renommer le prompt actuel",
+ "Save prompt as": "Enregistrer le prompt sous",
+ "Restore current prompt": "Restaurer le prompt actuel",
+ "Prompt Content": "Contenu du prompt",
+ "comma delimited,no spaces between": "délimité par des virgules, sans espace entre",
+ "(disabled when max recursion steps are used)": "(désactivé lorsque le nombre maximum de pas de récursivité est utilisé)",
+ "Cap the number of entry activation recursions": "Plafonner le nombre de récursions d'activation d'entrée",
+ "Max Recursion Steps": "Nombre maximal d'étapes de récursivité",
+ "0 = unlimited, 1 = scans once and doesn't recurse, 2 = scans once and recurses once, etc\\n(disabled when min activations are used)": "0 = illimité, 1 = scanne une fois et ne récure pas, 2 = scanne une fois et récure une fois, etc.\n(désactivé lorsque des activations minimales sont utilisées)",
+ "Include names with each message into the context for scanning": "Inclure les noms dans chaque message dans le contexte pour l'analyse.",
+ "Apply current sorting as Order": "Appliquer le tri actuel comme ordre",
+ "Display swipe numbers for all messages, not just the last.": "Afficher le nombre de balayage sur tous les messages, et pas seulement le dernier.",
+ "Swipe # for All Messages": "Balayage # pour tous les messages",
+ "Defines on importing cards which action should be chosen for importing its listed tags. 'Ask' will always display the dialog.": "Définit, lors de l'importation de cartes, l'action à choisir pour importer les tags répertoriés. L'action 'Demander' affichera toujours la boîte de dialogue.",
+ "Ask": "Demander",
+ "tag_import_none": "Aucun",
+ "tag_import_all": "Tous",
+ "Existing": "Existant",
+ "If checked and the character card contains a Post-History Instructions override, use that instead": "Si cette case est cochée et que la carte de personnage contient une instruction de remplacement pour le post-histoire, utilisez-la à la place.",
+ "Prefer Character Card Instructions": "Préférer les instructions du personnage",
+ "Enable auto-select of input text in some text fields when clicking/selecting them. Applies to popup input textboxes, and possible other custom input fields.": "Active la sélection automatique du texte saisi dans certains champs de texte lorsque l'on clique dessus ou qu'on les sélectionne. S'applique aux zones de texte de type popup et, éventuellement, à d'autres champs de saisie personnalisés.",
+ "Auto-select Input Text": "Sélection automatique du texte d'entrée",
+ "markdown_hotkeys_desc": "Activer les raccourcis clavier pour l'insertion de caractères au format markdown dans certaines zones de saisie de texte. Voir '/help hotkeys'.",
+ "Markdown Hotkeys": "Markdown Hotkeys",
+ "mui_reset": "Réinitialisation",
+ "Show a button in the input area to ask the AI to impersonate your character for a single message": "Affichez un bouton dans la zone de saisie pour demander à l'IA de se faire passer pour votre personnage le temps d'un message.",
+ "Quick 'Impersonate' button": "Bouton \"Usurpation\" rapide",
+ "In group chat, highlight the character(s) that are currently queued to generate responses and the order in which they will respond.": "Dans le chat de groupe, mettez en évidence le(s) personnage(s) qui sont actuellement en file d'attente pour générer des réponses et l'ordre dans lequel ils répondront.",
+ "Show group chat queue": "Afficher la file d'attente du chat de groupe",
+ "Keyboard": "Clavier:",
+ "Select with Tab or Enter": "Sélectionner avec Tab ou Entrée",
+ "Select with Tab": "Sélectionner avec Tab",
+ "Select with Enter": "Sélectionner avec Entrée",
+ "stscript_parser_flag_replace_getvar_label": "Empêche les macros {{getvar::}} {{getglobalvar::}} d'avoir des valeurs littérales de type macro auto-évaluées.\nPar exemple, \"{{newline}}\" reste la chaîne littérale \"{{newline}}\"\n\n((Cela se fait en remplaçant en interne les macros {{getvar::}} {{getglobalvar::}} par des variables scopées)",
+ "Background Image": "Arrière-plans",
+ "Background Fitting": "Ajustement de l'arrière-plan",
+ "Classic": "Classique",
+ "Cover": "Couvrir",
+ "Contain": "Contenir",
+ "Stretch": "Étirer",
+ "Center": "Centrer",
+ "Persona Lore Alt+Click to open the lorebook": "Lore de persona\nAlt+Click pour ouvrir le lorebook",
+ "None (disabled)": "Aucun (désactivé)",
+ "world_button_title": "Lore de personnage\n\nCliquer pour charger\nMajuscule-clic pour ouvrir la fenêtre contextuelle « Lien vers le Wold Info ».",
+ "Chat Lore Alt+Click to open the lorebook": "Lore du chat\nAlt+Click pour ouvrir le lorebook",
+ "Manual": "Manuel",
+ "Duplicate persona": "Dupliquer la persona",
+ "popup-button-crop": "Recadrer",
+ "Close popup": "Fermer le popup",
+ "Close": "Fermer",
+ "Any contents here will replace the default Post-History Instructions used for this character. (v2 spec: post_history_instructions)": "Tout contenu ici remplacera les instructions post-histore par défaut utilisées pour ce personnage.\n(v2 spec: post_history_instructions)",
+ "WI Entry Status:🔵 Constant🟢 Normal🔗 Vectorized": "WI Statut de l'entrée :\r🔵 Constant\r🟢 Normal\r🔗 Vectorisé",
+ "Use global": "Utiliser les données globales",
+ "Whole Words": "Mots entiers",
+ "Group Scoring": "Notation des groupes",
+ "delay_until_recursion_level": "Définit les niveaux de délai pour les balayages récursifs.\r\rDans un premier temps, seul le premier niveau (le plus petit nombre) sera pris en compte.\rSi aucune correspondance n'est trouvée, le niveau suivant devient éligible pour la correspondance.\rCette opération est répétée jusqu'à ce que tous les niveaux soient vérifiés.\r\rLié au réglage \"Délai avant récursion\".",
+ "Recursion Level": "Niveau de récursivité",
+ "Non-recursable (will not be activated by another)": "Non récursable (ne peut être activé par un autre)",
+ "Delay until recursion (can only be activated on recursive checking)": "Délai jusqu'à la récursion (ne peut être activé que lors d'un contrôle récursif)",
+ "Prioritize": "Priorités",
+ "Sticky entries will stay active for N messages after being triggered.": "Les entrées collantes restent actives pendant N messages après avoir été déclenchées.",
+ "Sticky": "Collant",
+ "Entries with a cooldown can't be activated N messages after being triggered.": "Les entrées avec un temps de recharge ne peuvent pas être activées N messages après avoir été déclenchées.",
+ "Cooldown": "Récupération",
+ "Entries with a delay can't be activated until there are N messages present in the chat.": "Les entrées avec un délai ne peuvent pas être activées tant qu'il n'y a pas N messages dans le chat.",
+ "Delay": "Délai",
+ "Filter to Characters or Tags": "Filtre sur les personnages ou les tags",
+ "Switch the Character/Tags filter around to exclude the listed characters and tags from matching for this entry": "Changez le filtre Personnages/Tags pour exclure les personnages et tags listés de la correspondance pour cette entrée.",
+ "Exclude": "Exclure",
+ "Injection position. Relative (to other prompts in prompt manager) or In-chat @ Depth.": "Position d'injection. Relative (par rapport à d'autres prompts dans le gestionnaire de prompts) ou In-chat @ Depth.",
+ "prompt_manager_in_chat": "In-chat",
+ "The content of this prompt is pulled from elsewhere and cannot be edited here.": "Le contenu de ce message est tiré d'autres sources et ne peut être modifié ici..",
+ "Open checkpoint chat\nShift+Click to replace the existing checkpoint with a new one": "Cliquer pour ouvrir le chat du point de contrôle\nShift+Click pour remplacer le point de contrôle existant par un nouveau.",
+ "Caption": "Légende",
+ "Swipe left": "Balayer à gauche",
+ "Swipe right": "Balayer à droite",
+ "alternate_greetings_hint_1": "Cliquez sur le",
+ "alternate_greetings_hint_2": "bouton pour commencer !",
+ "Alternate Greeting #": "Salutation alternative #",
+ "Reroll with the entire prefix": "Renouvellement avec l'ensemble du préfixe",
+ "Load a custom asset list or select": "Charger une liste de ressources personnalisée ou sélectionner",
+ "to install 3rd party extensions.": "pour installer des extensions tierces.",
+ "Assets URL": "URL des ressources",
+ "load_asset_list_desc": "Chargement d'une liste d'extensions et de ressources sur la base d'un fichier de liste de ressources.\n\nL'URL des ressources par défaut dans ce champ pointe vers la liste des extensions et des ressources officielles.\nSi vous disposez d'une liste de ressources personnalisée, vous pouvez l'insérer ici.\n\nPour installer une seule extension tierce, utilisez le bouton \"Installer des extensions\" en haut à droite.",
+ "Load an asset list": "Charger une liste de ressources",
+ "Load Asset List": "Charger la liste des ressources",
+ "Characters": "Personnages",
+ "Disable": "Désactiver",
+ "Enable": "Activer",
+ "These files are available for the current character in all chats they are in.": "Ces fichiers sont disponibles pour le personnage actuel dans tous les chats auxquels il participe.",
+ "These files are available for all characters in the current chat.": "Ces fichiers sont disponibles pour tous les personnages du chat actuel.",
+ "Image Captioning": "Légende des images",
+ "Local": "Local",
+ "Multimodal (OpenAI / Anthropic / llama / Google)": "Multimodal (OpenAI / Anthropic / llama / Google)",
+ "Extras": "Extras",
+ "Horde": "Horde",
+ "API": "API",
+ "Text Generation WebUI (oobabooga)": "Text Generation WebUI (oobabooga)",
+ "currently_selected": "[Actuellement sélectionné]",
+ "currently_loaded": "[Actuellement chargée]",
+ "Allow reverse proxy": "Autoriser le reverse proxy",
+ "Hint:": "Indice:",
+ "Set your API keys and endpoints in the 'API Connections' tab first.": "Définissez d'abord vos clés API et vos points de terminaison dans l'onglet « Connexions API ».",
+ "Caption Prompt": "Légende Prompt",
+ "Ask every time": "Demander à chaque fois",
+ "Message Template": "Modèle de message",
+ "(use _space": "(use",
+ "macro)": "macro)",
+ "Automatically caption images": "Légende automatique des images",
+ "Edit captions before saving": "Modifier les légendes avant l'enregistrement",
+ "Profile name:": "Nom du profil :",
+ "Creating a Connection Profile": "Création d'un profil de connexion",
+ "{{@key}}": "{{@key}}:",
+ "Enter a name:": "Saisir un nom :",
+ "Connection Profile": "Profil de connexion",
+ "View connection profile details": "Voir les détails du profil de connexion",
+ "Create a new connection profile": "Créer un nouveau profil de connexion",
+ "Update a connection profile": "Mise à jour d'un profil de connexion",
+ "Edit a connection profile": "Modifier un profil de connexion",
+ "Reload a connection profile": "Recharger un profil de connexion",
+ "Delete a connection profile": "Supprimer un profil de connexion",
+ "Omitted Settings:": "Réglages omis :",
+ "Character Expressions": "Expressions de personnages",
+ "Translate text to English before classification": "Traduire le texte en anglais avant de le classer",
+ "Show default images (emojis) if sprite missing": "Afficher les images par défaut (emojis) si le sprite est manquant",
+ "Image Type - talkinghead (extras)": "Type d'image - talkinghead (extras)",
+ "Classifier API": "API de classification",
+ "Select the API for classifying expressions.": "Sélectionnez l'API pour classer les expressions.",
+ "Main API": "API principale",
+ "WebLLM Extension": "Extension WebLLM",
+ "LLM Prompt": "Prompt LLM",
+ "Will be used if the API doesn't support JSON schemas or function calling.": "Sera utilisé si l'API ne prend pas en charge les schémas JSON ou les appels de fonction.",
+ "Default / Fallback Expression": "Expression par défaut / de secours",
+ "Set the default and fallback expression being used when no matching expression is found.": "Définir l'expression par défaut et l'expression de secours utilisées lorsqu'aucune expression correspondante n'est trouvée.",
+ "Custom Expressions": "Expressions personnalisées",
+ "Can be set manually or with an _space": "Peut être réglé manuellement ou à l'aide d'une",
+ "space_ slash command.": "commande slash.",
+ "Open a chat to see the character expressions.": "Ouvrez un chat pour voir les expressions du personnage.",
+ "You are in offline mode. Click on the image below to set the expression.": "Vous êtes en mode hors ligne. Cliquez sur l'image ci-dessous pour définir l'expression.",
+ "Sprite Folder Override": "Remplacement du dossier des sprites",
+ "Use a forward slash to specify a subfolder. Example: _space": "Utilisez une barre oblique pour spécifier un sous-dossier. Exemple :",
+ "Upload sprite pack (ZIP)": "Charger le pack de sprites (ZIP)",
+ "Remove all image overrides": "Suppression de toutes les surcharges d'image",
+ "Create new folder in the _space": "Créer un nouveau dossier dans le",
+ "folder of your user data directory and name it as the name of the character.": "de votre répertoire de données utilisateur et nommez-le comme le nom du personnage.",
+ "Put images with expressions there. File names should follow the pattern:": "Placez-y des images avec des expressions. Les noms de fichiers doivent suivre le modèle :",
+ "expression_label_pattern": "[expression_label].[image_format]",
+ "Sprite set:": "Ensemble de sprites :",
+ "Show Gallery": "Show Gallery",
+ "ext_sum_title": "Résumer",
+ "ext_sum_webllm": "Extension WebLLM",
+ "ext_sum_restore_tip": "Rétablir un résumé précédent ; à utiliser plusieurs fois pour effacer l'état du résumé pour ce chat.",
+ "ext_sum_force_tip": "Déclencher une mise à jour du résumé dès maintenant.",
+ "ext_sum_include_wi_scan_desc": "Inclure le dernier résumé dans l'analyse WI.",
+ "ext_sum_include_wi_scan": "Inclure dans l'analyse des World Info",
+ "None (not injected)": "Aucune (pas d'injection)",
+ "ext_sum_injection_position_none": "Le résumé ne sera pas injecté dans le prompt. Vous pouvez toujours y accéder via la macro {{summary}}.",
+ "Labels and Message": "Étiquettes et message",
+ "Label": "Étiquette",
+ "(label of the button, if no icon is chosen) ": "(étiquette du bouton, si aucune icône n'est choisie)",
+ "Title": "Titre",
+ "(tooltip, leave empty to show message or /command)": "(info-bulle, laisser vide pour afficher un message ou une commande)",
+ "Message / Command:": "Message / Commande :",
+ "Word wrap": "Enveloppe de mot",
+ "Tab size:": "Tab size:",
+ "Ctrl+Enter to execute": "Ctrl+Entrée pour exécuter",
+ "Context Menu": "Menu Contexte",
+ "Auto-Execute": "Auto-Exécution",
+ "Don't trigger auto-execute": "Ne pas déclencher l'exécution automatique",
+ "Invisible (auto-execute only)": "Invisible (exécution automatique uniquement)",
+ "Execute on startup": "Exécuter au démarrage",
+ "Execute on user message": "Exécuter sur message de l'utilisateur",
+ "Execute on AI message": "Exécuter sur message de l'IA",
+ "Execute on chat change": "Exécuter sur changement de chat",
+ "Execute on new chat": "Exécution sur un nouveau chat",
+ "Execute on group member draft": "Execute on group member draft",
+ "Automation ID:": "Automation ID",
+ "Testing": "Essai",
+ "Quick Reply": "Quick Reply",
+ "Enable Quick Replies": "Activer les Quick Replies",
+ "Combine Quick Replies": "Combiner les Quick Replies",
+ "Show Popout Button": "Afficher le bouton contextuel (sur le bureau)",
+ "Global Quick Reply Sets": "Global Quick Reply Sets",
+ "Chat Quick Reply Sets": "Chat Quick Reply Sets",
+ "Edit Quick Replies": "Editer les Quick Replies",
+ "Disable Send (Insert Into Input Field)": "Désactiver l'envoi (insérer dans le champ de saisie)",
+ "Place Quick Reply Before Input": "Placez le quick reply avant la saisie",
+ "Inject user input automatically": "Injecter automatiquement les données de l'utilisateur",
+ "(if disabled, use ": "(si désactivé, utiliser",
+ "macro for manual injection)": "macro pour injection manuelle)",
+ "Color": "Couleur",
+ "Only apply color as accent": "N'appliquez la couleur qu'en tant qu'accent",
+ "ext_regex_new_global_script_desc": "Nouveau script global de regex",
+ "ext_regex_new_scoped_script_desc": "Nouveau script délimitée de regex",
+ "ext_regex_disallow_scoped": "Interdire l'utilisation de regex délimitée",
+ "ext_regex_allow_scoped": "Autoriser l'utilisation de regex délimitées",
+ "ext_regex_user_input_desc": "Messages envoyés par l'utilisateur.",
+ "ext_regex_ai_input_desc": "Messages reçus de l'API Génération.",
+ "ext_regex_slash_desc": "Messages envoyés à l'aide de commandes STscript.",
+ "ext_regex_wi_desc": "Contenu de l'entrée Lorebook/World Info. Nécessite de cocher la case 'Only Format Prompt' !",
+ "ext_regex_run_on_edit_desc": "Exécute le script regex lorsque le message appartenant à un ou plusieurs rôles spécifiés est modifié.",
+ "Macro in Find Regex": "Macros dans Find Regex",
+ "Don't substitute": "Ne pas substituer",
+ "Substitute (raw)": "Substituer (raw)",
+ "Substitute (escaped)": "Substituer (escaped)",
+ "Ephemerality": "Éphémérité",
+ "ext_regex_only_format_visual_desc": "Le contenu du fichier d'historique du chat ne changera pas, mais les regex seront appliquées aux messages affichés dans l'UI du chat.",
+ "Abort current image generation task": "Abandonner la tâche de génération d'images en cours",
+ "Stop Image Generation": "Arrêt de la génération d'images",
+ "sd_function_tool": "Utilisez les outils pour détecter automatiquement les intentions de générer des images.",
+ "sd_function_tool_txt": "Utilisez les outils",
+ "sd_free_extend": "Étendre automatiquement les prompts de sujet en mode libre (pas les portraits ou les arrière-plans) à l'aide d'un LLM actuellement sélectionné.",
+ "sd_free_extend_txt": "Étendre les prompts de sujet en mode libre",
+ "sd_free_extend_small": "(interactif/commandes)",
+ "Model ID": "Model ID",
+ "e.g. black-forest-labs/FLUX.1-dev": "ex. black-forest-labs/FLUX.1-dev",
+ "API Key": "Clé API",
+ "Click to set": "Cliquez pour définir",
+ "You can find your API key in the Stability AI dashboard.": "Vous trouverez votre clé API dans le tableau de bord de Stability AI.",
+ "Style Preset": "Preset de style",
+ "Prompt Upsampling": "Prompt Upsampling",
+ "Swap width and height": "Intervertir la largeur et la hauteur",
+ "CLIP Skip": "CLIP Skip",
+ "Karras": "Karras",
+ "Not all samplers supported.": "Tous les échantillonneurs ne sont pas pris en charge.",
+ "sd_adetailer_face": "Utiliser ADetailer avec le modèle de visage pendant la génération. L'extension ADetailer doit être installée sur le backend.",
+ "Use ADetailer (Face)": "Utiliser ADetailer (Face)",
+ "(-1 for random)": "(-1 pour aléatoire)",
+ "Chat Message Visibility (by source)": "Visibilité des messages de chat (par source)",
+ "Uncheck to hide the extension's messages in chat prompts.": "Décochez cette case pour masquer les messages de l'extension dans les prompts de chat.",
+ "Extensions Menu": "Menu des extensions",
+ "Slash Command": "Commande Slash",
+ "Interactive Mode": "Mode interactif",
+ "Function Tool": "Outils",
+ "ext_translate_btn_chat": "Traduire le chat",
+ "ext_translate_btn_input": "Traduire la saisie",
+ "ext_translate_delete_confirm_1": "Êtes-vous sûr ?",
+ "ext_translate_delete_confirm_2": "Cette action supprime le texte traduit de tous les messages de la discussion en cours. Cette action ne peut pas être annulée.",
+ "ext_translate_title": "Traduction du chat",
+ "ext_translate_auto_mode": "Mode auto",
+ "ext_translate_mode_none": "Aucun",
+ "ext_translate_mode_responses": "Traduire les réponses",
+ "ext_translate_mode_inputs": "Traduire la saisie",
+ "ext_translate_mode_both": "Traduire les deux",
+ "ext_translate_mode_provider": "Fournisseur",
+ "ext_translate_target_lang": "Langue cible",
+ "ext_translate_clear": "Clear Translations",
+ "Select TTS Provider": "Sélectionner le fournisseur TTS",
+ "tts_enabled": "Activé",
+ "Narrate user messages": "Narrer les messages de l'utilisateur",
+ "Auto Generation": "Generation Auto",
+ "Requires auto generation to be enabled.": "La génération automatique doit être activée.",
+ "Narrate by paragraphs (when streaming)": "Narration par paragraphes (en cas de diffusion en continu)",
+ "Only narrate quotes": "Ne narrer que les \"citations\"",
+ "Ignore text, even quotes, inside asterisk": "Ignorer le *texte, même les \"citations\", à l'intérieur d'astérisques*",
+ "Narrate only the translated text": "Ne narrer que le texte traduit",
+ "Skip codeblocks": "Sauter les blocs de code",
+ "Skip tagged blocks": "Skip blocks",
+ "Pass Asterisks to TTS Engine": "Transmettre les astérisques au moteur TTS",
+ "Audio Playback Speed": "Vitesse de lecture audio",
+ "Vector Storage": "Vector Storage",
+ "Vectorization Source": "Source de vectorisation",
+ "Local (Transformers)": "Local (Transformers)",
+ "Vectorization Model": "Modèle de vectorisation",
+ "Keep model in memory": "Garder le modèle en mémoire",
+ "Hint: Set the URL in the API connection settings.": "Conseil : définissez l'URL dans les paramètres de connexion à l'API.",
+ "The server MUST be started with the --embedding flag to use this feature!": "Le serveur DOIT être démarré avec l'option --embedding pour utiliser cette fonctionnalité !",
+ "NomicAI API Key": "Clé API NomicAI",
+ "Query messages": "Messages de requête",
+ "Score threshold": "Seuil de score",
+ "Chunk boundary": "Limite des morceaux",
+ "World Info settings": "Réglages desWorld Info",
+ "Enable for World Info": "Activer pour les World Info",
+ "Enabled for all entries": "Activé pour toutes les entrées",
+ "Checked: all entries except ❌ status can be activated.": "Vérifié : toutes les entrées sauf l'état ❌ peuvent être activées..",
+ "Unchecked: only entries with ❌ status can be activated.": "Non coché : seules les entrées ayant un statut 🔗 peuvent être activées.",
+ "Max Entries": "Nombre maximal d'entrées",
+ "File vectorization settings": "Paramètres de vectorisation des fichiers",
+ "Enable for files": "Activer pour les fichiers",
+ "Message attachments": "Pièces jointes aux messages",
+ "Size threshold (KB)": "Seuil de taille (KB)",
+ "Chunk size (chars)": "Taille des morceaux (chars)",
+ "Chunk overlap (%)": "Chevauchement des morceaux (%)",
+ "Retrieve chunks": "Récupérer des morceaux",
+ "Data Bank files": "Fichiers de la banque de données",
+ "Injection Template": "Modèle d'injection",
+ "Injection Position": "Position d'injection",
+ "Vectorize All": "Vectoriser tout",
+ "Purge Vectors": "Vecteurs de purge",
+ "Chat vectorization settings": "Paramètres de vectorisation du chat",
+ "Enabled for chat messages": "Activé pour les messages de chat",
+ "Retain#": "Conserver#",
+ "Insert#": "Insérer#",
+ "Vector Summarization": "Résumés vectoriels",
+ "Summarize chat messages for vector generation": "Résumer les messages de chat pour la génération de vecteurs",
+ "Warning: This will slow down vector generation drastically, as all messages have to be summarized first.": "Attention : Cela ralentira considérablement la génération de vecteurs, car tous les messages doivent d'abord être résumés.",
+ "Summarize chat messages when sending": "Résumer les messages de chat lors de l'envoi",
+ "Warning: This might cause your sent messages to take a bit to process and slow down response time.": "Attention : Le traitement des messages envoyés risque d'être un peu plus long et de ralentir le temps de réponse.",
+ "Extras API": "Extras API",
+ "Only used when Main API or WebLLM Extension is selected.": "Utilisé uniquement lorsque l'API principale ou l'extension WebLLM est sélectionnée.",
+ "Old messages are vectorized gradually as you chat. To process all previous messages, click the button below.": "Les anciens messages sont vectorisés progressivement au fur et à mesure que vous discutez.\n To process all previous messages, click the button below.",
+ "View Stats": "Voir les statistiques",
+ "Note:": "Note:",
+ "this chat is temporary and will be deleted as soon as you leave it.": "ce chat est temporaire et sera supprimé dès que vous le quitterez.",
+ "Enter a new display name:": "Saisissez un nouveau nom d'affichage :",
+ "Import Tags For _begin": "Importer des tags pour ",
+ "Click remove on any tag to remove it from this import. Select one of the import options to finish importing the tags.": "Cliquez sur le bouton « Supprimer » d'un tag pour le retirer de l'importation..\n Sélectionnez l'une des options d'importation pour terminer l'importation des tags.",
+ "Existing Tags": "Tags existantes",
+ "New Tags": "Nouveaux tags",
+ "Folder Tags": "Tags de dossier",
+ "The following tags will be auto-imported based on the currently selected folders": "Les tags suivants seront importés automatiquement en fonction des dossiers sélectionnés.",
+ "Import None": "Aucun Import",
+ "Import All": "Importer Tout",
+ "Import Existing": "Importer l'existants",
+ "Import": "Importer",
+ "chat_rename_1": "Saisissez le nouveau nom du chat :",
+ "chat_rename_2": "!!L'utilisation d'un nom de fichier existant produira une erreur !!!",
+ "chat_rename_3": "Cela rompra le lien entre les points de contrôle des chats .",
+ "chat_rename_4": "Il n'est pas nécessaire d'ajouter '.jsonl' à la fin.",
+ "Enter Checkpoint Name:": "Saisir le nom du point de contrôle :",
+ "(Leave empty to auto-generate)": "(Laisser vide pour une génération automatique)",
+ "The currently existing checkpoint will be unlinked and replaced with the new checkpoint, but can still be found in the Chat Management.": "Le point de contrôle existant sera dissocié et remplacé par le nouveau point de contrôle, mais pourra toujours être trouvé dans la gestion du chat.",
+ "Include Body Parameters": "Inclure les paramètres du corps",
+ "custom_include_body_desc": "Paramètres à inclure dans le corps de la requête Chat Completion (objet YAML)\n\nExemple:\n- top_k: 20\n- repetition_penalty: 1.1",
+ "Exclude Body Parameters": "Exclure les paramètres du corps",
+ "custom_exclude_body_desc": "Paramètres à exclure du corps de la requête Chat Completion (tableau YAML)\n\nExemple:\n- frequency_penalty\n- presence_penalty",
+ "Include Request Headers": "Inclure les en-têtes des requêtes",
+ "custom_include_headers_desc": "En-têtes supplémentaires pour les demandes de Chat Completion (objet YAML)\n\nExemple:\n- CustomHeader: custom-value\n- AnotherHeader: custom-value",
+ "Functions in this category are for advanced users only. Don't click anything if you're not sure about the consequences.": "Les fonctions de cette catégorie sont réservées aux utilisateurs avancés. Ne cliquez sur rien si vous n'êtes pas sûr des conséquences.",
+ "THIS IS PERMANENT!": "CECI EST PERMANENT !",
+ "Also delete the chat files": "Supprimez également les fichiers de chat",
+ "Are you sure you want to duplicate this character?": "Êtes-vous sûr de vouloir dupliquer ce personnage ?",
+ "If you just want to start a new chat with the same character...": "Si vous voulez simplement commencer un nouveau chat avec le même personnage, utilisez l'option \"Commencer un nouveau chat\" dans le menu d'options en bas à gauche.",
+ "forbid_media_global_state_forbidden": "(interdit)",
+ "forbid_media_global_state_allowed": "(autorisé)",
+ "help_format_1": "Commandes de mise en forme du texte :",
+ "help_format_2": "*texte*",
+ "help_format_3": "s'affiche comme",
+ "help_format_4": "italique",
+ "help_format_5": "**texte**",
+ "help_format_6": "s'affiche comme",
+ "help_format_7": "gras",
+ "help_format_8": "***text***",
+ "help_format_9": "s'affiche comme",
+ "help_format_10": "gras italique",
+ "help_format_11": "__texte__",
+ "help_format_12": "s'affiche comme",
+ "help_format_13": "souligné",
+ "help_format_14": "~~texte~~",
+ "help_format_15": "displays as a",
+ "help_format_16": "strikethough",
+ "help_format_17": "[texte](url)",
+ "help_format_18": "s'affiche comme un",
+ "help_format_19": "hyperlink",
+ "help_format_20": "",
+ "help_format_21": "s'affiche sous forme d'image",
+ "help_format_22": "```texte```",
+ "help_format_23": "s'affiche comme un bloc de code (de nouvelles lignes sont autorisées entre les backticks)",
+ "help_format_like_this": "comme cela",
+ "help_format_24": "`texte`",
+ "help_format_25": "s'affiche comme du",
+ "help_format_26": "code inline",
+ "help_format_27": "> texte",
+ "help_format_28": "s'affiche comme une citation en bloc (notez l'espace après >)",
+ "help_format_29": "# texte",
+ "help_format_30": "s'affiche comme un grand en-tête (notez l'espace)",
+ "help_format_32": "## texte",
+ "help_format_33": "s'affiche comme un en-tête moyen (notez l'espace)",
+ "help_format_35": "### texte",
+ "help_format_36": "s'affiche comme un petit en-tête (notez l'espace)",
+ "help_1": "Bonjour à tous ! Veuillez sélectionner la rubrique d'aide sur laquelle vous souhaitez en savoir plus :",
+ "help_2": "Commands Slash",
+ "help_or": "ou",
+ "help_3": "Formatage",
+ "help_4": "Raccourcis",
+ "help_5": "{{Macros}}",
+ "help_6": "Vous avez encore des questions ? Le",
+ "help_7": "la documentation officielle de SillyTavern",
+ "help_8": "contient beaucoup plus d'informations !",
+ "help_hotkeys_0": "Raccourcis du chat",
+ "help_hotkeys_1": "Haut",
+ "help_hotkeys_2": "Modifier le dernier message dans le chat",
+ "help_hotkeys_3": "Ctrl+Haut",
+ "help_hotkeys_4": "Modifier le dernier message de l'utilisateur dans le chat",
+ "help_hotkeys_5": "Gauche",
+ "help_hotkeys_6": "balayage gauche",
+ "help_hotkeys_7": "Droite",
+ "help_hotkeys_8": "balayage droite (REMARQUE : les touches de raccourci sont désactivées lorsque quelque chose est tapé dans la barre de discussion.)",
+ "help_hotkeys_9": "Entrée",
+ "help_hotkeys_10": "(avec la barre de chat sélectionnée)",
+ "help_hotkeys_10_1": "envoyez votre message à l'IA",
+ "help_hotkeys_11": "Ctrl+Entrée",
+ "help_hotkeys_12": "Régénérer la dernière réponse de l'IA",
+ "help_hotkeys_13": "Alt+Entrée",
+ "help_hotkeys_14": "Poursuivre la dernière réponse de l'IA",
+ "help_hotkeys_15": "Echap",
+ "help_hotkeys_16": "arrêter la génération de réponses de l'IA, fermer les panneaux de l'interface utilisateur, annuler l'édition du message",
+ "help_hotkeys_17": "Ctrl+Maj+Haut",
+ "help_hotkeys_18": "Défiler jusqu'à la ligne de contexte",
+ "help_hotkeys_19": "Ctrl+Maj+Bas",
+ "help_hotkeys_20": "Raccourcis Markdown",
+ "help_hotkeys_21": "Fonctionne dans la barre de chat et les zones de texte marquées par cette icône :",
+ "help_hotkeys_22": "**gras**",
+ "help_hotkeys_23": "*italique*",
+ "help_hotkeys_24": "__underline__",
+ "help_hotkeys_25": "`code inline`",
+ "help_hotkeys_26": "~~strikethrough~~",
+ "Enter the Git URL of the extension to install": "Saisir l'URL Git de l'extension à installer",
+ "Disclaimer:": "Disclaimer:",
+ "Please be aware that using external extensions can have unintended side effects and may pose security risks. Always make sure you trust the source before importing an extension. We are not responsible for any damage caused by third-party extensions.": "Veuillez noter que l'utilisation d'extensions externes peut avoir des effets secondaires inattendus et présenter des risques pour la sécurité. Assurez-vous toujours de la fiabilité de la source avant d'importer une extension. Nous ne sommes pas responsables des dommages causés par des extensions tierces.",
+ "Prompt Itemization": "Prompt Itemization",
+ "Show Raw Prompt": "Afficher le prompt brut",
+ "Copy Prompt": "Copier le prompt",
+ "Show Prompt Differences": "Montrer les différences entre les prompts",
+ "API/Model:": "API/Modèle:",
+ "Preset:": "Preset:",
+ "Tokenizer:": "Tokenizer:",
+ "Only the white numbers really matter. All numbers are estimates. Grey color items may not have been included in the context due to certain prompt format settings.": "Seuls les chiffres blancs comptent vraiment. Tous les chiffres sont des estimations.\n Les éléments de couleur grise peuvent ne pas avoir été inclus dans le contexte en raison de certains paramètres du format du prompt.",
+ "System Info:": "Info système :",
+ "Prompt Tokens:": "Prompt Tokens:",
+ "World Info:": "World Info:",
+ "Chat History:": "Historique du chat :",
+ "Extensions:": "Extensions:",
+ "Bias:": "Biais:",
+ "Total Tokens in Prompt:": "Total des Tokens dans le Prompt:",
+ "Max Context": "Contexte Maximums",
+ "(Context Size - Response Length)": "(Taille du contexte - Longueur de la réponse)",
+ "System-wide Replacement Macros (in order of evaluation):": "Macros de remplacement globales (par ordre d'évaluation) :",
+ "help_macros_1": "uniquement pour la mise en lot des commandes slash. Remplacé par le résultat de la commande précédente.",
+ "help_macros_2": "insère simplement une nouvelle ligne.",
+ "help_macros_3": "supprime les nouvelles lignes entourant cette macro.",
+ "help_macros_4": "pas d'opération, juste une chaîne vide.",
+ "help_macros_5": "prompts globaux définis dans les paramètres de l'API. Valable uniquement dans les overrides de définitions avancées.",
+ "help_macros_6": "l'entrée de l'utilisateur",
+ "help_macros_7": "l'override du prompt principal du personnage",
+ "help_macros_8": "les overrides des instructions post-histoire du personnage",
+ "help_macros_9": "la description du personnage",
+ "help_macros_10": "la personnalité du personnage",
+ "help_macros_11": "le scénario du personnage",
+ "help_macros_12": "la description de votre persona actuelle",
+ "help_macros_13": "Exemples de dialogue du personnage",
+ "help_macros_14": "Exemples de dialogues non formatés",
+ "(only for Story String)": "(uniquement pour Story String)",
+ "help_macros_summary": "le dernier résumé du chat généré par l'extension \"Summarize\" (si disponible).",
+ "help_macros_15": "votre nom de persona actuel",
+ "help_macros_16": "le nom du personnage",
+ "help_macros_17": "le numéro de version du personnage",
+ "help_macros_18": "une liste de noms de membres du groupe séparés par des virgules (y compris les personnes en sourdine) ou le nom du personnage dans les discussions en solo. Alias: {{charIfNotGroup}}",
+ "help_groupNotMuted": "la même chose que {{group}}, mais exclut les membres en sourdine",
+ "help_macros_19": "un nom de modèle de génération de texte pour l'API actuellement sélectionnée.",
+ "Can be inaccurate!": "Peut être imprécis !",
+ "help_macros_20": "le texte du dernier message de chat.",
+ "help_macros_lastUser": "le texte du dernier message de chat de l'utilisateur.",
+ "help_macros_lastChar": "le texte du dernier message de chat du personnage.",
+ "help_macros_21": "index # du dernier message de chat. Utile pour la mise en lot des commandes slash.",
+ "help_macros_22": "l'ID du premier message inclus dans le contexte. La génération doit être exécutée au moins une fois dans la session en cours.",
+ "help_macros_23": "l'ID basé sur 1 du swipe actuel dans le dernier message de chat. Chaîne vide si le dernier message est caché par l'utilisateur ou le prompt.",
+ "help_macros_24": "le nombre de swipes dans le dernier message de chat. Chaîne vide si le dernier message est caché par l'utilisateur ou le prompt.",
+ "help_macros_reverse": "inverse le contenu de la macro.",
+ "help_macros_25": "vous pouvez laisser une note ici, et la macro sera remplacée par un contenu vide. Non visible pour l'IA.",
+ "help_macros_26": "l'heure actuelle",
+ "help_macros_27": "la date du jour",
+ "help_macros_28": "le jour de la semaine",
+ "help_macros_29": "l'heure ISO actuelle (24 heures)",
+ "help_macros_30": "la date ISO actuelle (AAAA-MM-JJ)",
+ "help_macros_31": "la date et l'heure actuelles dans le format spécifié, par exemple pour la date/heure allemandes :",
+ "help_macros_32": "l'heure actuelle dans le fuseau horaire UTC spécifié, ex. UTC-4 or UTC+2",
+ "help_macros_33": "la différence de temps entre time1 et time2. Accepte les macros d'heure et de date. (Ex: {{timeDiff::{{isodate}} {{time}}::2024/5/11 12:30:00}})",
+ "help_macros_34": "le temps écoulé depuis l'envoi du dernier message de l'utilisateur",
+ "help_macros_35": "définit un biais comportemental pour l'IA jusqu'à la prochaine entrée de l'utilisateur. Les guillemets autour du texte sont importants.",
+ "help_macros_36": "lance un dé. (ex :",
+ "space_ will roll a 6-sided dice and return a number between 1 and 6)": "lancera un dé à 6 faces et renverra un nombre entre 1 et 6)",
+ "help_macros_37": "renvoie un élément aléatoire de la liste. (ex :",
+ "space_ will return 1 of the 4 numbers at random. Works with text lists too.": "renverra 1 des 4 nombres au hasard. Fonctionne également avec les listes de texte.",
+ "help_macros_38": "syntaxe alternative pour random qui permet d'utiliser des virgules dans les éléments de la liste.",
+ "help_macros_39": "choisit un élément aléatoire dans la liste. Fonctionne de la même manière que {{random}}, avec les mêmes options syntaxiques possibles, mais le choix restera cohérent pour ce chat une fois choisi et ne sera pas relancé sur les messages consécutifs et le traitement des prompts.",
+ "help_macros_40": "ajoute dynamiquement du texte entre guillemets aux séquences de mots interdits, si le backend Text Generation WebUI est utilisé. Ne fait rien pour les autres backends. Peut être utilisé n'importe où (description de personnage, WI, AN, etc.) Les guillemets autour du texte sont importants.",
+ "help_macros_isMobile": "\"true\" si ST fonctionne actuellement dans un environnement mobile, \"false\" sinon",
+ "Instruct Mode and Context Template Macros:": "Mode d'instruction et macros de modèle de contexte :",
+ "(enabled in the Advanced Formatting settings)": "(activée dans les paramètres de formatage avancés)",
+ "help_macros_41": "longueur maximale autorisée du prompt en tokens = (taille du contexte - longueur de la réponse)",
+ "help_macros_42": "contexte modèle exemple dialogues séparateur",
+ "help_macros_43": "contexte modèle chat ligne de début",
+ "help_macros_44": "contenu du prompt système s'il est activé (soit override du prompt du personnage si c'est autorisée, soit defaultSystemPrompt)",
+ "help_macros_45": "contenu du prompt système",
+ "help_macros_46": "séquence de préfixes du prompt système d'instruction",
+ "help_macros_47": "séquence de suffixes du prompt système d'instruction",
+ "help_macros_48": "séquence de préfixes de l'utilisateur d'instruction",
+ "help_macros_49": "séquence de suffixes de l'utilisateur d'instruction",
+ "help_macros_50": "séquence de préfixes de l'assistant d'instruction",
+ "help_macros_51": "séquence de suffixes de l'assistant d'instruction",
+ "help_macros_52": "instruct assistant first output sequence",
+ "help_macros_53": "instruct assistant last output sequence",
+ "help_macros_54": "instruct system message prefix sequence",
+ "help_macros_55": "instruct system message suffix sequence",
+ "help_macros_56": "instruct system instruction prefix",
+ "help_macros_57": "instruct first user message filler",
+ "help_macros_58": "instruct stop sequence",
+ "help_macros_first_user": "instruct user first input sequence",
+ "help_macros_last_user": "instruct user last input sequence",
+ "Chat variables Macros:": "Macros de variables de chat :",
+ "Local variables = unique to the current chat": "Variables locales = propres au chat actuel",
+ "Global variables = works in any chat for any character": "Variables globales = fonctionnent dans n'importe quel chat pour n'importe quel personnage",
+ "Scoped variables = works in STscript": "Variables scopés = fonctionne dans STscript",
+ "help_macros_59": "remplacée par la valeur de la variable locale \"name\"",
+ "help_macros_60": "remplacée par une chaîne vide, définit la variable locale \"name\" à \"value\"",
+ "help_macros_61": "remplacés par des chaînes vides, ajoute une valeur numérique de \"increment\" à la variable locale \"name\"",
+ "help_macros_62": "remplacé par le résultat de l'incrémentation de la valeur de la variable \"name\" de 1",
+ "help_macros_63": "remplacé par le résultat de la décrémentation de la valeur de la variable \"name\" de 1",
+ "help_macros_64": "remplacée par la valeur de la variable globale \"name\"",
+ "help_macros_65": "remplacée par une chaîne vide, définit la variable globale \"name\" à \"value\"",
+ "help_macros_66": "remplacé par une chaîne vide, ajoute une valeur numérique de \"increment\" à la variable globale \"name\"",
+ "help_macros_67": "remplacé par le résultat de l'incrémentation de la valeur de la variable globale \"name\" de 1",
+ "help_macros_68": "remplacé par le résultat de la décrémentation de la valeur de la variable globale \"name\" de 1",
+ "help_macros_69": "remplacée par la valeur de la variable scopée \"name\"",
+ "help_macros_70": "remplacée par la valeur de l'élément à l'index (pour les tableaux / listes ou objets / dictionnaires) de la variable scopée \"name\"",
+ "Choose what to export": "Choisir ce qui doit être exporté",
+ "{{name}}": "{{name}}",
+ "Choose what to import": "Choisir ce qu'il faut importer",
+ "If necessary, you can later restore this chat file from the /backups folder": "Si nécessaire, vous pouvez restaurer ultérieurement ce fichier de chat à partir du dossier /backups",
+ "Also delete the current chat file": "Supprime également le fichier de chat actuel",
+ "Persona Lorebook for": "Persona Lorebook pour",
+ "persona_world_template_txt": "Une Wolrd Info sélectionnée sera liée à ce personnage. Lors de la création d'une réponse de l'IA,\n il sera combiné avec les entrées des lorebooks globaux, de personnages et de chat.",
+ "Are you sure you want to connect to the following proxy URL?": "Êtes-vous sûr de vouloir vous connecter à l'URL proxy suivante ?",
+ "Encountered an error while processing your request.": "Une erreur s'est produite lors du traitement de votre requête.",
+ "Check you have credits available on your": "Vérifiez que vous avez des crédits disponibles sur votre",
+ "OpenAI account quora_error": "compte OpenAI",
+ "dot quota_error": ".",
+ "If you have sufficient credits, please try again later.": "Si vous avez suffisamment de crédits, veuillez réessayer plus tard.",
+ "Enter your password below to confirm:": "Entrez votre mot de passe ci-dessous pour confirmer :",
+ "Unique to this chat.": "Unique à ce chat.",
+ "All group members will use the following scenario text instead of what is specified in their character cards.": "Tous les membres du groupe utiliseront le texte du scénario suivant au lieu de ce qui est spécifié dans leur fiche de personnage.",
+ "The following scenario text will be used instead of the value set in the character card.": "Le texte du scénario suivant sera utilisé à la place de la valeur définie dans la fiche de personnage.",
+ "Checkpoints inherit the scenario override from their parent, and can be changed individually after that.": "Les points de contrôle héritent du scénario de leur parent et peuvent ensuite être modifiés individuellement.",
+ "Download Model": "Télécharger le modèle",
+ "Downloader Options": "Options de téléchargement",
+ "Extra parameters for downloading/HuggingFace API": "Paramètres supplémentaires pour le téléchargement/HuggingFace API.\rSi vous n'êtes pas sûr, laissez ces paramètres vides.",
+ "Revision": "Révision",
+ "Folder Name": "Nom du dossier de sortie",
+ "HF Token": "Token HF",
+ "Include Patterns": "Inclure les patterns",
+ "Glob patterns of files to include in the download.": "Patterns globaux de fichiers à inclure dans le téléchargement.\rSéparer chaque pattern par une nouvelle ligne.",
+ "Exclude Patterns": "Exclure les patterns",
+ "Glob patterns of files to exclude in the download.": "Patterns globaux de fichiers à exclure du téléchargement.\rSéparer chaque pattern par une nouvelle ligne.",
+ "Tag Management": "Gestion des tags",
+ "Save your tags to a file": "Enregistrer vos tags dans un fichier",
+ "Restore tags from a file": "Enregistrer vos tags dans un fichier",
+ "Create a new tag": "Créer un nouveau tag",
+ "Drag handle to reorder. Click name to rename. Click color to change display.": "Faites glisser la poignée pour réorganiser. Cliquez sur le nom pour le renommer. Cliquez sur la couleur pour modifier l'affichage.",
+ "Click on the folder icon to use this tag as a folder.": "Cliquez sur l'icône de dossier pour utiliser ce tag comme dossier.",
+ "Use alphabetical sorting": "Utiliser le tri alphabétique",
+ "tags_sorting_desc": "Si cette option est activée, les tags seront automatiquement triés par ordre alphabétique lors de leur création ou de leur renommage.\nSi cette option est désactivée, les nouveaux tags seront ajoutés à la fin.\n\nSi un tag est réorganisé manuellement en le faisant glisser, le tri automatique sera désactivé.",
+ "Are you sure you want to delete the theme?": "Êtes-vous sûr de vouloir supprimer le thème ?",
+ "This will delete all your settings and data. There will be no undo button. Make sure you have a backup before proceeding.": "Cette opération effacera tous vos paramètres et toutes vos données. Il n'y aura pas de bouton d'annulation.\n Assurez-vous de disposer d'une sauvegarde avant de poursuivre.",
+ "Account reset code has been posted to the server console.": "Le code de réinitialisation du compte a été affiché sur la console du serveur.",
+ "and connect to an": "et se connectez à une",
+ "You can add more": "Vous pouvez ajouter d'autres",
+ "or_welcome": "ou",
+ "from other websites": "d'autres sites web.",
+ "Go to the": "Aller au menu ",
+ "to install additional features.": "pour installer des fonctionnalités supplémentaires.",
+ "If you're connected to an API, try asking me something!": "Si vous êtes connecté à une API, essayez de me demander quelque chose !",
+ "Title/Memo": "Titre/Memo",
+ "Strategy": "Stratégie",
+ "Position": "Position",
+ "Trigger %": "Déclencheur %",
+ "Only chunk on custom boundary": "Only chunk on custom boundary",
+ "Generate Caption": "Générer une légende",
+ "Use System Prompt": "Utiliser le prompt système:",
+ "Settings Preset": "Preset de réglages:",
+ "System Prompt Name": "Nom du prompt système:",
+ "Instruct Mode": "Mode Instruction:",
+ "Save and Update": "Sauvegarder et mettre à jour",
+ "Don't ask again for this URL": "Ne plus demander pour cette URL"
}
diff --git a/public/locales/is-is.json b/public/locales/is-is.json
index 5783ee8ee..2cacc1511 100644
--- a/public/locales/is-is.json
+++ b/public/locales/is-is.json
@@ -879,7 +879,7 @@
"popup-button-no": "Nei",
"popup-button-cancel": "Hætta við",
"popup-button-import": "Flytja inn",
- "Advanced Defininitions": "Ítarleg skilgreiningar",
+ "Advanced Definitions": "Ítarleg skilgreiningar",
"Prompt Overrides": "Hnekkja hvetjandi",
"(For Chat Completion and Instruct Mode)": "(Til að ljúka spjalli og leiðbeiningarham)",
"Insert {{original}} into either box to include the respective default prompt from system settings.": "Settu inn {{original}} í hvora kassa til að innifela viðkomandi sjálfgefna framkallan frá kerfisstillingum.",
@@ -1376,7 +1376,7 @@
"char_import_2": "Chub Lorebook (beinn hlekkur eða auðkenni)",
"char_import_3": "JanitorAI karakter (beinn hlekkur eða UUID)",
"char_import_4": "Pygmalion.chat karakter (beinn hlekkur eða UUID)",
- "char_import_5": "AICharacterCard.com Karakter (beinn hlekkur eða auðkenni)",
+ "char_import_5": "AICharacterCards.com Karakter (beinn hlekkur eða auðkenni)",
"char_import_6": "Beinn PNG hlekkur (sjá",
"char_import_7": "fyrir leyfilega gestgjafa)",
"char_import_8": "RisuRealm karakter (beinn hlekkur)",
diff --git a/public/locales/it-it.json b/public/locales/it-it.json
index 6ce249db9..eeea5c0fb 100644
--- a/public/locales/it-it.json
+++ b/public/locales/it-it.json
@@ -879,7 +879,7 @@
"popup-button-no": "NO",
"popup-button-cancel": "Annulla",
"popup-button-import": "Importare",
- "Advanced Defininitions": "Definizioni avanzate",
+ "Advanced Definitions": "Definizioni avanzate",
"Prompt Overrides": "Sostituzioni richieste",
"(For Chat Completion and Instruct Mode)": "(Per il completamento della chat e la modalità istruzione)",
"Insert {{original}} into either box to include the respective default prompt from system settings.": "Inserisci {{originale}} in uno dei due riquadri per includere il prompt predefinito corrispondente dalle impostazioni di sistema.",
@@ -1376,7 +1376,7 @@
"char_import_2": "Lorebook di Chub (collegamento diretto o ID)",
"char_import_3": "Carattere JanitorAI (collegamento diretto o UUID)",
"char_import_4": "Carattere Pygmalion.chat (collegamento diretto o UUID)",
- "char_import_5": "Carattere AICharacterCard.com (Link diretto o ID)",
+ "char_import_5": "Carattere AICharacterCards.com (Link diretto o ID)",
"char_import_6": "Collegamento PNG diretto (fare riferimento a",
"char_import_7": "per gli host consentiti)",
"char_import_8": "Personaggio RisuRealm (collegamento diretto)",
diff --git a/public/locales/ja-jp.json b/public/locales/ja-jp.json
index bb552d2fd..2b3119e6e 100644
--- a/public/locales/ja-jp.json
+++ b/public/locales/ja-jp.json
@@ -879,7 +879,7 @@
"popup-button-no": "いいえ",
"popup-button-cancel": "キャンセル",
"popup-button-import": "インポート",
- "Advanced Defininitions": "高度な定義",
+ "Advanced Definitions": "高度な定義",
"Prompt Overrides": "プロンプトのオーバーライド",
"(For Chat Completion and Instruct Mode)": "(チャット補完と指示モード用)",
"Insert {{original}} into either box to include the respective default prompt from system settings.": "システム設定からの対応するデフォルトのプロンプトを含めるには、どちらかのボックスに{{original}}を挿入します。",
@@ -1378,7 +1378,7 @@
"char_import_2": "Chub ロアブック (直接リンクまたは ID)",
"char_import_3": "JanitorAI キャラクター (直接リンクまたは UUID)",
"char_import_4": "Pygmalion.chat キャラクター (直接リンクまたは UUID)",
- "char_import_5": "AICharacterCard.com キャラクター (直接リンクまたは ID)",
+ "char_import_5": "AICharacterCards.com キャラクター (直接リンクまたは ID)",
"char_import_6": "直接PNGリンク(参照",
"char_import_7": "許可されたホストの場合)",
"char_import_8": "RisuRealm キャラクター (直接リンク)",
diff --git a/public/locales/ko-kr.json b/public/locales/ko-kr.json
index 636d8cd7f..2c2dfca86 100644
--- a/public/locales/ko-kr.json
+++ b/public/locales/ko-kr.json
@@ -895,7 +895,7 @@
"popup-button-no": "아니요",
"popup-button-cancel": "취소",
"popup-button-import": "불러오기",
- "Advanced Defininitions": "고급 정의",
+ "Advanced Definitions": "고급 정의",
"Prompt Overrides": "프롬프트 무시",
"(For Chat Completion and Instruct Mode)": "(채팅 완료 및 지시 모드의 경우)",
"Insert {{original}} into either box to include the respective default prompt from system settings.": "{{original}}를 해당 상자에 넣어 시스템 설정의 기본 프롬프트를 포함합니다.",
@@ -1395,7 +1395,7 @@
"char_import_2": "Chub Lorebook(직접 링크 또는 ID)",
"char_import_3": "JanitorAI 캐릭터(직접 링크 또는 UUID)",
"char_import_4": "Pygmalion.chat 문자(직접 링크 또는 UUID)",
- "char_import_5": "AICharacterCard.com 캐릭터(직접 링크 또는 ID)",
+ "char_import_5": "AICharacterCards.com 캐릭터(직접 링크 또는 ID)",
"char_import_6": "직접 PNG 링크(참조",
"char_import_7": "허용된 호스트의 경우)",
"char_import_8": "RisuRealm 캐릭터 (직접링크)",
diff --git a/public/locales/nl-nl.json b/public/locales/nl-nl.json
index 9938c3a7f..c4820f5b8 100644
--- a/public/locales/nl-nl.json
+++ b/public/locales/nl-nl.json
@@ -879,7 +879,7 @@
"popup-button-no": "Nee",
"popup-button-cancel": "Annuleren",
"popup-button-import": "Importeren",
- "Advanced Defininitions": "Geavanceerde definities",
+ "Advanced Definitions": "Geavanceerde definities",
"Prompt Overrides": "Prompt-overschrijvingen",
"(For Chat Completion and Instruct Mode)": "(Voor voltooiing van chat en instructiemodus)",
"Insert {{original}} into either box to include the respective default prompt from system settings.": "Voeg {{original}} in in elk vak in om de respectievelijke standaardprompt vanuit systeeminstellingen op te nemen.",
@@ -1376,7 +1376,7 @@
"char_import_2": "Chub Lorebook (directe link of ID)",
"char_import_3": "JanitorAI-personage (directe link of UUID)",
"char_import_4": "Pygmalion.chat-teken (directe link of UUID)",
- "char_import_5": "AICharacterCard.com-teken (directe link of ID)",
+ "char_import_5": "AICharacterCards.com-teken (directe link of ID)",
"char_import_6": "Directe PNG-link (zie",
"char_import_7": "voor toegestane hosts)",
"char_import_8": "RisuRealm-personage (directe link)",
diff --git a/public/locales/pt-pt.json b/public/locales/pt-pt.json
index 98260b8b2..5a2fba569 100644
--- a/public/locales/pt-pt.json
+++ b/public/locales/pt-pt.json
@@ -879,7 +879,7 @@
"popup-button-no": "Não",
"popup-button-cancel": "Cancelar",
"popup-button-import": "Importar",
- "Advanced Defininitions": "Definições Avançadas",
+ "Advanced Definitions": "Definições Avançadas",
"Prompt Overrides": "Substituições de prompt",
"(For Chat Completion and Instruct Mode)": "(Para conclusão de bate-papo e modo de instrução)",
"Insert {{original}} into either box to include the respective default prompt from system settings.": "Insira {{original}} em qualquer caixa para incluir o prompt padrão respectivo das configurações do sistema.",
@@ -1376,7 +1376,7 @@
"char_import_2": "Chub Lorebook (link direto ou ID)",
"char_import_3": "Personagem JanitorAI (Link Direto ou UUID)",
"char_import_4": "Caractere Pygmalion.chat (Link Direto ou UUID)",
- "char_import_5": "Personagem AICharacterCard.com (link direto ou ID)",
+ "char_import_5": "Personagem AICharacterCards.com (link direto ou ID)",
"char_import_6": "Link PNG direto (consulte",
"char_import_7": "para hosts permitidos)",
"char_import_8": "Personagem RisuRealm (link direto)",
diff --git a/public/locales/ru-ru.json b/public/locales/ru-ru.json
index 5c506a88b..4c67beaa9 100644
--- a/public/locales/ru-ru.json
+++ b/public/locales/ru-ru.json
@@ -265,7 +265,7 @@
"Current Members": "Текущие участники",
"Delete": "Удалить",
"Cancel": "Отменить",
- "Advanced Defininitions": "Расширенное описание",
+ "Advanced Definitions": "Расширенное описание",
"Personality summary": "Резюме по личности",
"Scenario": "Сценарий",
"Talkativeness": "Разговорчивость",
@@ -294,7 +294,7 @@
"Avoid sending sensitive information to the Horde.": "Избегайте отправки личной информации Horde",
"Review the Privacy statement": "Ознакомиться с заявлением о конфиденциальности",
"Trusted workers only": "Только доверенные рабочие машины",
- "For privacy reasons, your API key will be hidden after you reload the page.": "По причинам безопасности ваш API-ключ будет скрыт после перезагрузки страницы.",
+ "For privacy reasons, your API key will be hidden after you reload the page.": "Из соображений безопасности ваш API-ключ будет скрыт после перезагрузки страницы.",
"-- Horde models not loaded --": "--Модель Horde не загружена--",
"Example: http://127.0.0.1:5000/api ": "Пример: http://127.0.0.1:5000/api",
"No connection...": "Нет соединения...",
@@ -331,7 +331,6 @@
"UID ↘": "UID ↘",
"Trigger% ↗": "Триггер% ↗",
"Trigger% ↘": "Триггер% ↘",
- "Order:": "Порядок:",
"Depth:": "Глубина:",
"Character Lore First": "Сначала лор персонажа",
"Global Lore First": "Сначала глобальный лор",
@@ -347,7 +346,7 @@
"After Char Defs": "↓Перс.",
"Before AN": "↑АЗ",
"After AN": "↓АЗ",
- "Order": "Порядок:",
+ "Order": "Очерёдность:",
"Update a theme file": "Обновить файл темы",
"Save as a new theme": "Сохранить как новую тему",
"Minimum number of blacklisted words detected to trigger an auto-swipe": "Минимальное количество обнаруженных запрещённых слов, при котором срабатывает авто-свайп.",
@@ -497,7 +496,7 @@
"What this keyword should mean to the AI, sent verbatim": "Что это ключевое слово должно означать для ИИ, отправляется дословно",
"Filter to Character(s)": "Фильтр по персонажу(ам)",
"Character Exclusion": "Исключить персонажей",
- "Inclusion Group": "Включить персонажей",
+ "Inclusion Group": "Группа записей",
"Only one entry with the same label will be activated": "Будет активна только одна запись с одинаковой меткой",
"-- Characters not found --": "-- Персонажей не найдено --",
"(This will be the first message from the character that starts every chat)": "(Это будет первое сообщение от персонажа, когда вы начинаете новый чат)",
@@ -541,7 +540,7 @@
"NOT ANY": "НЕ ЛЮБОЙ",
"Optional Filter": "Дополнительный фильтр",
"New Entry": "Новая запись",
- "Fill empty Memo/Titles with Keywords": "Заполните пустые Заметки/Названия ключевыми словами",
+ "Fill empty Memo/Titles with Keywords": "Заполнить пустые названия ключевыми словами",
"AI Response Formatting": "Формат ответа ИИ",
"Change Background Image": "Изменить фон",
"Extensions": "Расширения",
@@ -628,7 +627,7 @@
"UI Theme": "Тема UI",
"This message is invisible for the AI": "Это сообщение невидимо для ИИ",
"Sampler Priority": "Приоритет сэмплеров",
- "Ooba only. Determines the order of samplers.": "Только Ooba. Определяет порядок сэмплеров.",
+ "Ooba only. Determines the order of samplers.": "Только oobabooga. Определяет порядок сэмплеров.",
"Load default order": "Загрузить стандартный порядок",
"Max Tokens Second": "Макс. кол-во токенов в секунду",
"CFG": "CFG",
@@ -967,7 +966,7 @@
"char_import_2": "Лорбук с Chub (прямая ссылка или ID)",
"char_import_3": "Персонаж с JanitorAI (прямая ссылка или UUID)",
"char_import_4": "Персонаж с Pygmalion.chat (прямая ссылка или UUID)",
- "char_import_5": "Персонаж с AICharacterCard.com (прямая ссылка или ID)",
+ "char_import_5": "Персонаж с AICharacterCards.com (прямая ссылка или ID)",
"char_import_6": "Прямая ссылка на PNG-файл (чтобы узнать список разрешённых хостов, загляните в",
"char_import_7": ")",
"Grammar String": "Грамматика",
@@ -1132,7 +1131,7 @@
"Custom API Key": "Ключ от кастомного API",
"(Optional)": "(необязательно)",
"Enter a Model ID": "Введите идентификатор модели",
- "Example: gpt-3.5-turbo": "Пример: gpt-3.5-turbo",
+ "Example: gpt-4o": "Пример: gpt-4o",
"Available Models": "Доступные модели",
"Prompt Post-Processing": "Постобработка промпта",
"Applies additional processing to the prompt before sending it to the API.": "Позволяет обработать промпт перед отправкой в API.",
@@ -1342,7 +1341,7 @@
"Chat Lorebook": "Chat Lorebook for",
"chat_world_template_txt": "A selected World Info will be bound to this chat. When generating an AI reply,\n it will be combined with the entries from global and character lorebooks.",
"Use tag as folder": "Tag as folder",
- "WI Entry Status:🔵 Constant🟢 Normal🔗 Vectorized❌ Disabled": "Статус записи:\n 🔵 Постоянная\n 🟢 Обычная\n 🔗 Векторизованная\n ❌ Отключена",
+ "WI Entry Status:🔵 Constant🟢 Normal🔗 Vectorized": "Статус записи:\n 🔵 Постоянная\n 🟢 Обычная\n 🔗 Векторизованная",
"WI_Entry_Status_Constant": "Постоянная",
"WI_Entry_Status_Normal": "Обычная",
"WI_Entry_Status_Vectorized": "Векторизованная",
@@ -1366,13 +1365,7 @@
"Can be used to automatically activate Quick Replies": "Can be used to automatically activate Quick Replies",
"Automation ID": "Automation ID",
"( None )": "( None )",
- "Delay until recursion (this entry can only be activated on recursive checking)": "Delay until recursion (this entry can only be activated on recursive checking)",
- "Inclusion Groups ensure only one entry from a group is activated at a time, if multiple are triggered.Documentation: World Info - Inclusion Group": "Inclusion Groups ensure only one entry from a group is activated at a time, if multiple are triggered.\rSupports multiple comma-separated groups.\r\rDocumentation: World Info - Inclusion Group",
- "Prioritize this entry: When checked, this entry is prioritized out of all selections.If multiple are prioritized, the one with the highest 'Order' is chosen.": "Prioritize this entry: When checked, this entry is prioritized out of all selections.\rIf multiple are prioritized, the one with the highest 'Order' is chosen.\r",
- "A relative likelihood of entry activation within the group": "A relative likelihood of entry activation within the group",
- "Group Weight": "Group Weight",
"Add Memo": "Add Memo",
- "close": "close",
"reset": "reset",
"save": "save",
"Open checkpoint chat": "Открыть чат из чекпоинта",
@@ -2035,5 +2028,121 @@
"tags_sorting_desc": "Автоматически сортировать теги по алфавиту после создания или переименования одного из них.\nПри выключении новые теги будут просто добавляться в конец.\n\nЕсли вручную перетащить один из тегов на другое место, автосортировка отключится.",
"Imported tags:": "Импортируемые теги:",
"Importing Tags": "Импорт тегов",
- "Couldn't import tags:": "Не удалось импортировать теги:"
+ "Couldn't import tags:": "Не удалось импортировать теги:",
+ "Allow fallback models": "Разрешить fallback-модели",
+ "Allow fallback providers": "Разрешить fallback-провайдеров",
+ "To use instruct formatting, switch to OpenRouter under Text Completion API.": "Переключитесь на OpenRouter в Text Completion API, чтобы использовать форматирование Instruct-режима.",
+ "Select providers. No selection = all providers.": "Выберите провайдера. Нет выбранного = выбраны все.",
+ "Model Providers": "Провайдеры моделей",
+ "Select a model": "Выберите модель",
+ "Search models...": "Искать по моделям...",
+ "[Currently loaded]": "[Загруженная сейчас]",
+ "Search providers...": "Искать по провайдерам...",
+ "Automatically chooses an alternative provider if chosen providers can't serve your request.": "Автоматически переключаться на другого провайдера, если текущий не может обслужить запрос.",
+ "Example: 127.0.0.1:8000": "Пример: 127.0.0.1:8000",
+ "Edit a connection profile": "Редактировать профиль соединения",
+ "System Prompt Name": "Название системного промпта",
+ "Use System Prompt": "Использовать системный промпт",
+ "Hint:": "Подсказка:",
+ "Click on the setting name to omit it from the profile.": "Нажмите на название настройки, чтобы исключить её из профиля",
+ "Included settings:": "Сохранённые параметры:",
+ "Server URL": "Адрес сервера",
+ "NanoGPT API Key": "Ключ от API NanoGPT",
+ "NanoGPT Model": "Модель NanoGPT",
+ "Use extension settings": "Использовать настройки из расширения",
+ "DeepSeek API Key": "Ключ от API DeepSeek",
+ "DeepSeek Model": "Модель DeepSeek",
+ "prompt_post_processing_merge": "Объединять идущие подряд сообщения с одной ролью",
+ "prompt_post_processing_semi": "Semi-strict (чередовать роли)",
+ "prompt_post_processing_strict": "Strict (чередовать роли, сначала пользователь)",
+ "Select Horde models": "Выбрать модель из Horde",
+ "Model ID (optional)": "Идентификатор модели (необязательно)",
+ "Derive context size from backend": "Использовать бэкенд для определения размера контекста",
+ "Rename current preset": "Переименовать пресет",
+ "No Worlds active. Click here to select.": "Нет активных миров. Нажмите, чтобы выбрать.",
+ "Title/Memo": "Название",
+ "Strategy": "Статус",
+ "Position": "Позиция",
+ "Trigger %": "% срабатывания",
+ "Use global": "Глоб. настройка",
+ "Whole Words": "Целые слова",
+ "Non-recursable (will not be activated by another)": "Не рекурсивная (не активируется другими)",
+ "Delay until recursion (can only be activated on recursive checking)": "Рекурсивная (активируется только рекурсией)",
+ "Toggle entry's active state.": "Вкл/выкл запись.",
+ "Prioritize": "Важная",
+ "Prioritize this entry: When checked, this entry is prioritized out of all selections.If multiple are prioritized, the one with the highest 'Order' is chosen.": "Важная запись получает приоритет среди всех выбранных. Если важных записей несколько, выбирается та, у которой выше \"Очерёдность\".",
+ "Group Weight": "Вес в группе",
+ "A relative likelihood of entry activation within the group": "Относительная вероятность активации записи в рамках группы",
+ "Sticky": "Липучка",
+ "Sticky entries will stay active for N messages after being triggered.": "Запись-\"липучка\" останется активной в течение N сообщений после срабатывания.",
+ "Cooldown": "Кулдаун",
+ "Entries with a cooldown can't be activated N messages after being triggered.": "Запись с заданным кулдауном не активируется следующие N сообщений после срабатывания.",
+ "Delay": "Задержка",
+ "Entries with a delay can't be activated until there are N messages present in the chat.": "Запись с заданной задержкой может активироваться только после того, как в чате наберётся N сообщений.",
+ "Non-sticky": "Нет",
+ "No cooldown": "Нет",
+ "No delay": "Нет",
+ "Filter to Characters or Tags": "Фильтровать по персонажам или тегам",
+ "Switch to plaintext mode": "Вкл/выкл режим чистого текста",
+ "Exclude": "Режим исключения",
+ "Switch the Character/Tags filter around to exclude the listed characters and tags from matching for this entry": "Инвертировать логику: для выбранных в фильтре персонажей/тегов данная запись активна НЕ БУДЕТ",
+ "Apply current sorting as Order": "Настроить Очерёдность в соответствии с текущей сортировкой",
+ "Create a new World Info": "Создать новый мир",
+ "Enter a name for the new file:": "Название нового файла:",
+ "Inclusion Groups ensure only one entry from a group is activated at a time, if multiple are triggered.Documentation: World Info - Inclusion Group": "Если сразу несколько записей из одной группы окажутся активированными, по факту сработает только одна. Одна запись может входить в несколько групп, отделяются запятыми. Раздел в документации: World Info - Inclusion Group",
+ "Valid World Info file name is required": "Требуется корректное имя для файла мира",
+ "World Info file has an invalid format": "Файл мира имеет неизвестный формат",
+ "World Info file has no entries": "В файле нет ни одной записи",
+ "Character not found.": "Персонаж не найден",
+ "Open a chat to get a name of the chat-bound lorebook": "Чтобы получить название привязанного к чату лорбука, требуется открыть чат.",
+ "File is not valid: ${0}": "Файл повреждён или имеет неизвестный формат: ${0}",
+ "The world with ${0} is invalid or corrupted.": "Мир ${0} повреждён или имеет неизвестный формат.",
+ "Deactivated all worlds": "Все миры отключены",
+ "No world found named: ${0}": "Не найдено мира с название ${0}",
+ "Activated world: ${0}": "Мир ${0} включён",
+ "Deactivated world: ${0}": "Мир ${0} отключён",
+ "World was not active: ${0}": "Мир ${0} не включён",
+ "The world '${0}' has been imported and linked to the character successfully.": "Мир ${0} успешно импортирован и привязан к персонажу.",
+ "World/Lorebook imported": "Мир/лорбук импортирован",
+ "Are you sure you want to import '${0}'?": "Вы точно хотите импортировать '${0}'?",
+ "It will overwrite the World/Lorebook with the same name.": "Существующий мир с таким же названием будет перезаписан.",
+ "Automatically 'continue' a response if the model stopped before reaching a certain amount of tokens.": "Автоматически \"продолжать\" ответ, если он оказался короче, чем целевая длина (в токенах).",
+ "None (not injected)": "Никуда",
+ "ext_sum_injection_position_none": "Пересказ не будет вставляться в промпт. Однако его по-прежнему можно будет вставить с помощью макроса {{summary}}",
+ "ext_sum_include_wi_scan": "Учитывать при сканировании лорбуком",
+ "ext_sum_include_wi_scan_desc": "Учитывать актуальный пересказ при сканировании промпта лорбуком.",
+ "ext_sum_force_tip": "Отправить запрос на создание пересказа прямо сейчас.",
+ "ext_sum_restore_tip": "Откатиться к предыдущему пересказу; используйте несколько раз, чтобы очистить историю пересказов для чата",
+ "Built-in Extensions:": "Комплектные расширения:",
+ "Installed Extensions:": "Установленные расширения:",
+ "Loading third-party extensions... Please wait...": "Загрузка сторонних расширений... Пожалуйста, подождите...",
+ "The page will be reloaded shortly...": "Страница будет перезагружена...",
+ "Extensions state changed": "Изменено состояние расширения",
+ "Error loading extensions. See browser console for details.": "Не удалось загрузить расширения. Подробности см. в консоли браузера.",
+ "You don't have permission to update global extensions.": "Отсутствуют права на обновление глобальных расширений.",
+ "Extension update failed": "Не удалось обновить расширение",
+ "Extension ${0} updated to ${1}": "Расширение ${0} обновлено до ${1}",
+ "Reload the page to apply updates": "Чтобы изменения вступили в силу, обновите страницу",
+ "You don't have permission to delete global extensions.": "Отсутствуют права на удаление глобальных расширений.",
+ "Are you sure you want to delete ${0}?": "Вы точно хотите удалить ${0}?",
+ "You don't have permission to move extensions.": "Отсутствуют права на перемещение расширений.",
+ "Are you sure you want to move ${0} to your local extensions? This will make it available only for you.": "Вы точно хотите сделать ${0} локальным расширением? После этого оно будет доступно только вам.",
+ "Are you sure you want to move ${0} to the global extensions? This will make it available for all users.": "Вы точно хотите сделать ${0} глобальным расширением? После этого оно будет доступно всем пользователям.",
+ "Extension ${0} moved.": "Расширение ${0} перемещено.",
+ "Extension ${0} deleted": "Расширение ${0} удалено.",
+ "Please wait...": "Пожалуйста, подождите...",
+ "Installing extension": "Идёт установка расширения",
+ "Extension installation failed": "Не удалось установить расширение",
+ "Extension '${0}' by ${1} (version ${2}) has been installed successfully!": "Расширение ${0} от автора ${1} (версия ${2}) успешно установлено!",
+ "Extension installation successful": "Расширение установлено",
+ "Extension updates available": "Доступных обновлений расширений",
+ "Auto-updating extensions. This may take several minutes.": "Запущено автоматическое обновление расширений. Это может занять несколько минут.",
+ "Install just for me": "Установить только мне",
+ "Install": "Установить",
+ "Install for all users": "Установить для всех пользователей",
+ "Modules provided by your Extras API:": "Модули из Extras API:",
+ "Not connected to the API!": "Нет соединения с API!",
+ "ext_type_system": "Это комплектное расширение. Его нельзя удалить, а обновляется оно вместе со всей системой.",
+ "Update all": "Обновить все",
+ "Close": "Закрыть"
}
diff --git a/public/locales/uk-ua.json b/public/locales/uk-ua.json
index ccfa17f45..216ed3928 100644
--- a/public/locales/uk-ua.json
+++ b/public/locales/uk-ua.json
@@ -879,7 +879,7 @@
"popup-button-no": "Немає",
"popup-button-cancel": "Скасувати",
"popup-button-import": "Імпорт",
- "Advanced Defininitions": "Розширені визначення",
+ "Advanced Definitions": "Розширені визначення",
"Prompt Overrides": "Перевизначення підказок",
"(For Chat Completion and Instruct Mode)": "(Для завершення чату та режиму інструктажу)",
"Insert {{original}} into either box to include the respective default prompt from system settings.": "Вставте {{original}} в будь-яке поле, щоб включити відповідний запит з налаштувань системи.",
@@ -1376,7 +1376,7 @@
"char_import_2": "Chub Lorebook (пряме посилання або ID)",
"char_import_3": "Символ JanitorAI (пряме посилання або UUID)",
"char_import_4": "Символ Pygmalion.chat (пряме посилання або UUID)",
- "char_import_5": "Символ AICharacterCard.com (пряме посилання або ідентифікатор)",
+ "char_import_5": "Символ AICharacterCards.com (пряме посилання або ідентифікатор)",
"char_import_6": "Пряме посилання на PNG (див",
"char_import_7": "для дозволених хостів)",
"char_import_8": "Персонаж RisuRealm (пряме посилання)",
diff --git a/public/locales/vi-vn.json b/public/locales/vi-vn.json
index 0b7175d01..36459e94f 100644
--- a/public/locales/vi-vn.json
+++ b/public/locales/vi-vn.json
@@ -879,7 +879,7 @@
"popup-button-no": "KHÔNG",
"popup-button-cancel": "Hủy",
"popup-button-import": "Nhập",
- "Advanced Defininitions": "Các Định nghĩa Nâng cao",
+ "Advanced Definitions": "Các Định nghĩa Nâng cao",
"Prompt Overrides": "Ghi đè Prompt",
"(For Chat Completion and Instruct Mode)": "(Đối với chế độ hoàn thành trò chuyện và hướng dẫn)",
"Insert {{original}} into either box to include the respective default prompt from system settings.": "Chèn {{original}} vào bất kỳ hộp nào để bao gồm Prompt mặc định tương ứng từ cài đặt hệ thống.",
@@ -1376,7 +1376,7 @@
"char_import_2": "Chub (Nhập URL trực tiếp hoặc ID)",
"char_import_3": "JanitorAI (Nhập URL trực tiếp hoặc UUID)",
"char_import_4": "Pygmalion.chat (Nhập URL trực tiếp hoặc UUID)",
- "char_import_5": "AICharacterCard.com (Nhập URL trực tiếp hoặc ID)",
+ "char_import_5": "AICharacterCards.com (Nhập URL trực tiếp hoặc ID)",
"char_import_6": "Nhập PNG trực tiếp (tham khảo",
"char_import_7": "đối với các máy chủ được phép)",
"char_import_8": "RisuRealm (URL trực tiếp)",
diff --git a/public/locales/zh-cn.json b/public/locales/zh-cn.json
index eb9b9b1a2..17a6ab6e8 100644
--- a/public/locales/zh-cn.json
+++ b/public/locales/zh-cn.json
@@ -78,7 +78,7 @@
"Allow": "Allow",
"Forbid": "Forbid",
"Enable OpenAI completion streaming": "启用OpenAI文本补全流式传输",
- "Display the response bit by bit as it is generated.": "随着回复的生成,逐位显示结果。",
+ "Display the response bit by bit as it is generated.": "随着回复的生成,逐词逐句地显示结果。",
"When this is off, responses will be displayed all at once when they are complete.": "当此选项关闭时,回复将在完成后一次性显示。",
"Frequency Penalty": "频率惩罚",
"Presence Penalty": "存在惩罚",
@@ -185,13 +185,13 @@
"Mirostat Eta": "Mirostat η",
"Learning rate of Mirostat": "Mirostat 的学习率。",
"Beam search": "束搜索",
- "A greedy, brute-force algorithm used in LLM sampling to find the most likely sequence of words or tokens. It expands multiple candidate sequences at once, maintaining a fixed number (beam width) of top sequences at each step.": "A greedy, brute-force algorithm used in LLM sampling to find the most likely sequence of words or tokens. It expands multiple candidate sequences at once, maintaining a fixed number (beam width) of top sequences at each step.",
- "# of Beams": "# of Beams",
+ "A greedy, brute-force algorithm used in LLM sampling to find the most likely sequence of words or tokens. It expands multiple candidate sequences at once, maintaining a fixed number (beam width) of top sequences at each step.": "一种在LLM采样中使用的贪婪暴力算法,用于找到最可能的单词或标记序列。它一次扩展多个候选序列,在每一步保留固定数量(光束宽度)的最佳序列。",
+ "# of Beams": "光束数量",
"The number of sequences generated at each step with Beam Search.": "The number of sequences generated at each step with Beam Search.",
"Length Penalty": "长度惩罚",
"Penalize sequences based on their length.": "Penalize sequences based on their length.",
"Early Stopping": "提前停止",
- "Controls the stopping condition for beam search. If checked, the generation stops as soon as there are '# of Beams' sequences. If not checked, a heuristic is applied and the generation is stopped when it's very unlikely to find better candidates.": "Controls the stopping condition for beam search. If checked, the generation stops as soon as there are '# of Beams' sequences. If not checked, a heuristic is applied and the generation is stopped when it's very unlikely to find better candidates.",
+ "Controls the stopping condition for beam search. If checked, the generation stops as soon as there are '# of Beams' sequences. If not checked, a heuristic is applied and the generation is stopped when it's very unlikely to find better candidates.": "控制光束搜索的停止条件。勾选时,当生成到达‘光束数量’的序列时停止。如果未勾选,则采用启发式方法,当几乎不可能找到更好的候选项时停止生成。",
"Contrastive search": "对比搜索",
"Contrastive_search_txt": "通过利用大多数 LLM 表示空间的各向同性以鼓励多样性同时保持一致性的采样器。详见 Su 等人在 2022 年发表的论文 《A Contrastive Framework for Neural Text Generation》。",
"Penalty Alpha": "惩罚系数 α",
@@ -208,7 +208,7 @@
"Speculative Ngram": "推测性 Ngram",
"Use a different speculative decoding method without a draft model": "使用不同的推测解码方法(不采用草稿模型)。最好使用草稿模型。推测性 Ngram 的效果不太好。",
"Spaces Between Special Tokens": "特殊词符之间的空格",
- "Seed_desc": "A random seed to use for deterministic and reproducable outputs. Set to -1 to use a random seed.",
+ "Seed_desc": "一个用于生成确定性和可复现的输出的随机种子。设置为 -1 时会使用随机种子。",
"LLaMA / Mistral / Yi models only": "LLaMA / Mistral / Yi模型专用。首先确保您选择了适当的词符化器。\n这项设置决定了你不想在结果中看到的字符串。\n每行一个字符串。可以是文本或者[词符id]。\n许多词符以空格开头。如果不确定,请使用词符计数器。",
"Example: some text [42, 69, 1337]": "例如:\n一些文本\n[42, 69, 1337]",
"CFG": "CFG",
@@ -227,8 +227,8 @@
"llama.cpp only. Determines the order of samplers. If Mirostat mode is not 0, sampler order is ignored.": "仅限 llama.cpp。确定采样器的顺序。如果 Mirostat 模式不为 0,则忽略采样器顺序。",
"Sampler Priority": "采样器优先级",
"Ooba only. Determines the order of samplers.": "确定采样器的顺序(仅适用于Ooba)",
- "Aphrodite only. Determines the order of samplers. Skew is always applied post-softmax, so it's not included here.": "Aphrodite only. Determines the order of samplers. Skew is always applied post-softmax, so it's not included here.",
- "Aphrodite only. Determines the order of samplers.": "Aphrodite only. Determines the order of samplers.",
+ "Aphrodite only. Determines the order of samplers. Skew is always applied post-softmax, so it's not included here.": "仅适用于 Aphrodite,用于确定采样器的顺序。偏移(Skew)始终在 softmax 后应用,因此不包含在这里。",
+ "Aphrodite only. Determines the order of samplers.": "仅适用于 Aphrodite。用于确定采样器的顺序。",
"Character Names Behavior": "角色名称行为",
"Helps the model to associate messages with characters.": "有助于模型将消息与角色关联起来。",
"None": "无",
@@ -266,13 +266,15 @@
"Use system prompt": "使用系统提示词",
"Merges_all_system_messages_desc_1": "合并所有系统消息,直到第一条具有非系统角色的消息,然后通过",
"Merges_all_system_messages_desc_2": "字段发送。",
+ "Show model thoughts": "展示思维链",
+ "Display the model's internal thoughts in the response.": "展示模型在回复时的内部思维链。",
"Assistant Prefill": "AI预填",
"Expand the editor": "展开编辑器",
"Start Claude's answer with...": "以如下内容开始Claude的回答...",
"Assistant Impersonation Prefill": "AI帮答预填",
"Use system prompt (Claude 2.1+ only)": "使用系统提示词(仅适用于Claude 2.1+)",
"Send the system prompt for supported models. If disabled, the user message is added to the beginning of the prompt.": "为支持的模型发送系统提示词。如果禁用,则用户消息将添加到提示词的开头。",
- "Confirm token parsing with": "Confirm token parsing with",
+ "Confirm token parsing with": "确认使用以下工具进行词符解析",
"Tokenizer": "词符化器",
"New preset": "新预设",
"Delete preset": "删除预设",
@@ -316,9 +318,9 @@
"Get your NovelAI API Key": "获取您的 NovelAI API 密钥",
"Enter it in the box below": "在下面的框中输入",
"Novel AI Model": "Novel AI 模型",
- "No connection...": "没有连接...",
+ "No connection...": "无连接...",
"API Type": "API 类型",
- "Generic (OpenAI-compatible) [LM Studio, LiteLLM, etc.]": "Generic (OpenAI-compatible) [LM Studio, LiteLLM, etc.]",
+ "Generic (OpenAI-compatible) [LM Studio, LiteLLM, etc.]": "通用(兼容 OpenAI)[LM Studio、LiteLLM 等]",
"TogetherAI API Key": "TogetherAI API 密钥",
"TogetherAI Model": "TogetherAI 模型",
"-- Connect to the API --": "-- 连接到API --",
@@ -350,7 +352,7 @@
"All": "All",
"class": "All Classes",
"Toggle grid view": "切换网格视图",
- "No model description": "[No description]",
+ "No model description": "[无描述]",
"vllm-project/vllm": "vllm-project/vllm(OpenAI API 包装器模式)",
"vLLM API key": "vLLM API 密钥",
"Example: 127.0.0.1:8000": "示例:http://127.0.0.1:8000",
@@ -399,12 +401,12 @@
"View API Usage Metrics": "查看API使用情况",
"Follow": "按照",
"these directions": "这些步骤",
- "to get your OpenAI API key.": "获取您的OpenAI API密钥。",
+ "to get your OpenAI API key.": "获取您的 OpenAI API 密钥。",
"Use Proxy password field instead. This input will be ignored.": "请使用“代理密码”字段。此输入将被忽略。",
"OpenAI Model": "OpenAI 模型",
"Bypass API status check": "绕过API状态检查",
"Show External models (provided by API)": "显示外部模型(由API提供)",
- "Claude API Key": "Claude API Key",
+ "Claude API Key": "Claude API 密钥",
"Get your key from": "从以下位置获取您的密钥",
"Anthropic's developer console": "Anthropic 开发者控制台",
"Claude Model": "Claude 模型",
@@ -430,23 +432,28 @@
"Groq Model": "Groq 模型",
"NanoGPT API Key": "NanoGPT API Key",
"NanoGPT Model": "NanoGPT Model",
+ "DeepSeek API Key": "DeepSeek API 密钥",
+ "DeepSeek Model": "DeepSeek 模型",
"Perplexity API Key": "Perplexity API 密钥",
"Perplexity Model": "Perplexity 模型",
"Cohere API Key": "Cohere API 密钥",
"Cohere Model": "Cohere 模型",
- "Block Entropy API Key": "Block Entropy API Key",
- "Select a Model": "Select a Model",
- "Custom Endpoint (Base URL)": "自定义端点(基本 URL)",
+ "Block Entropy API Key": "Block Entropy API 密钥",
+ "Select a Model": "选择一个模型",
+ "Custom Endpoint (Base URL)": "自定义端点(基础 URL)",
"Example: http://localhost:1234/v1": "例如:http://localhost:1234/v1",
"at the end of the URL!": "到 URL 的末尾!",
"Custom API Key": "自定义 API 密钥",
"(Optional)": "(可选)",
"Enter a Model ID": "输入模型名",
- "Example: gpt-3.5-turbo": "例如:gpt-3.5-turbo",
+ "Example: gpt-4o": "例如:gpt-4o",
"Available Models": "可用模型",
"Prompt Post-Processing": "提示词后处理",
"Applies additional processing to the prompt before sending it to the API.": "在将提示词发送到 API 之前对其进行额外处理。",
"prompt_post_processing_none": "未选择",
+ "prompt_post_processing_merge": "合并相同角色连续的发言",
+ "prompt_post_processing_semi": "半严格(强制对话角色交替)",
+ "prompt_post_processing_strict": "严格(强制对话角色交替、用户最先)",
"01.AI API Key": "01.AI API密钥",
"01.AI Model": "01.AI 模型",
"Additional Parameters": "附加参数",
@@ -503,25 +510,25 @@
"Never": "永不",
"Groups and Past Personas": "Groups and Past Personas",
"Always": "永远",
- "Instruct Sequences": "Instruct Sequences",
- "User Message Sequences": "User Message Sequences",
+ "Instruct Sequences": "指令序列",
+ "User Message Sequences": "用户消息序列",
"Inserted before a User message and as a last prompt line when impersonating.": "插入到用户消息之前并作为模拟时的最后一行提示词。",
"User Prefix": "用户消息前缀",
"Inserted after a User message.": "插入到用户消息之后。",
"User Suffix": "用户消息后缀",
- "Assistant Message Sequences": "Assistant Message Sequences",
+ "Assistant Message Sequences": "助手消息序列",
"Inserted before an Assistant message and as a last prompt line when generating an AI reply.": "插入到助手消息之前并作为生成 AI 回复时的最后一行提示词。",
"Assistant Prefix": "助手消息前缀",
"Inserted after an Assistant message.": "插入于助手消息之后。",
"Assistant Suffix": "助手消息后缀",
- "System Message Sequences": "System Message Sequences",
+ "System Message Sequences": "系统消息序列",
"Inserted before a System (added by slash commands or extensions) message.": "插入到系统(由快捷命令或扩展添加)消息之前。",
- "System Prefix": "System Message Prefix",
+ "System Prefix": "系统消息前缀",
"Inserted after a System message.": "插入到系统消息之后。",
- "System Suffix": "System Message Suffix",
+ "System Suffix": "系统消息后缀",
"If enabled, System Sequences will be the same as User Sequences.": "如果启用,系统序列将与用户序列相同。",
"System same as User": "系统与用户相同",
- "System Prompt Sequences": "System Prompt Sequences",
+ "System Prompt Sequences": "系统提示词序列",
"Inserted before a System prompt.": "插入到系统提示词之前。",
"System Prompt Prefix": "系统提示词前缀",
"Inserted after a System prompt.": "在系统提示词后插入。",
@@ -556,7 +563,7 @@
"Token Padding": "词符填充",
"Miscellaneous": "杂项",
"Non-markdown strings": "非 Markdown 字符串",
- "comma delimited,no spaces between": "comma delimited,no spaces between",
+ "comma delimited,no spaces between": "以逗号分隔,无需空格",
"Start Reply With": "以...开始回复",
"Show reply prefix in chat": "在聊天中显示回复前缀",
"World Info": "世界信息",
@@ -688,8 +695,8 @@
"Show Message Token Count": "显示消息词符数",
"Single-row message input area. Mobile only, no effect on PC": "将输入框限制为一行。仅适用于移动设备,对PC无影响",
"Compact Input Area (Mobile)": "紧凑输入区域(移动端)",
- "Display swipe numbers for all messages, not just the last.": "Display swipe numbers for all messages, not just the last.",
- "Swipe # for All Messages": "Swipe # for All Messages",
+ "Display swipe numbers for all messages, not just the last.": "显示所有信息的滑动编号,而非仅限最后一条。",
+ "Swipe # for All Messages": "给所有信息分配滑动编号 #",
"In the Character Management panel, show quick selection buttons for favorited characters": "在角色管理面板中,显示快速选择按钮以选择收藏的角色",
"Characters Hotswap": "角色卡热切换",
"Enable magnification for zoomed avatar display.": "启用放大功能以放大头像显示。",
@@ -894,7 +901,7 @@
"Favorite characters to add them to HotSwaps": "收藏角色以将它们添加到快速热切换区",
"Token counts may be inaccurate and provided just for reference.": "词符计数可能不准确,仅供参考。",
"Total tokens": "总词符数",
- "Calculating...": "正在计算…",
+ "Calculating...": "正在计算...",
"Tokens": "词符数",
"Permanent tokens": "永久词符",
"Permanent": "恒定的",
@@ -936,10 +943,10 @@
"Chat Name (Optional)": "聊天名称(可选)",
"Chat Lore": "聊天知识",
"Click to select a new avatar for this group": "单击选择该群聊的新头像",
- "Group reply strategy": "群聊回复策略",
+ "Group reply strategy": "群聊发言顺序",
"Manual": "手动",
"Natural order": "自然顺序",
- "List order": "上下顺序",
+ "List order": "从上到下",
"Group generation handling mode": "群组生成处理模式",
"Swap character cards": "交换角色卡",
"Join character cards (exclude muted)": "加入角色卡(不包括被禁言的)",
@@ -990,7 +997,7 @@
"popup-button-import": "导入",
"popup-button-crop": "裁剪",
"Close popup": "关闭弹出窗口",
- "Advanced Defininitions": "高级定义",
+ "Advanced Definitions": "高级定义",
"Prompt Overrides": "提示词覆盖",
"(For Chat Completion and Instruct Mode)": "(用于聊天补全和指导模式)",
"Insert {{original}} into either box to include the respective default prompt from system settings.": "将{{original}}插入到任一框中,以包含系统设置中的相应默认提示词。",
@@ -1023,7 +1030,7 @@
"Chatty": "高",
"Examples of dialogue": "对话示例",
"Important to set the character's writing style.": "设置角色的写作风格,很重要!",
- "(Examples of chat dialog. Begin each example with START on a new line.)": "(聊天对话的示例,每个示例都另起一行以开头。)",
+ "(Examples of chat dialog. Begin each example with START on a new line.)": "(聊天对话的示例,每个示例都另起一行以开头。)",
"Save": "保存",
"Chat History": "聊天记录",
"Import Chat": "导入聊天",
@@ -1107,12 +1114,15 @@
"A relative likelihood of entry activation within the group": "组内进入激活的相对可能性",
"Group Weight": "组权重",
"Sticky entries will stay active for N messages after being triggered.": "粘性条目在被触发后将保持活跃状态 N 条消息。",
- "Sticky": "粘性",
+ "Sticky": "黏性",
+ "Non-sticky": "无黏性",
"Entries with a cooldown can't be activated N messages after being triggered.": "具有冷却时间的条目在触发后 N 条消息内无法被激活。",
"Cooldown": "冷却",
+ "No cooldown": "无冷却",
"Entries with a delay can't be activated until there are N messages present in the chat.": "直到聊天中出现 N 条消息时,延迟的条目才能被激活。",
"Delay": "延迟",
- "Filter to Characters or Tags": "应用到角色或标签",
+ "No delay": "无延迟",
+ "Filter to Characters or Tags": "绑定到角色或标签",
"Switch the Character/Tags filter around to exclude the listed characters and tags from matching for this entry": "切换角色/标签筛选方式,将列出的角色和标签排除在匹配范围之外",
"Exclude": "排除",
"-- Characters not found --": "-- 未找到角色 --",
@@ -1132,8 +1142,8 @@
"prompt_manager_relative": "相对",
"prompt_manager_in_chat": "聊天中",
"prompt_manager_depth": "深度",
- "Injection depth. 0 = after the last message, 1 = before the last message, etc.": "注入深度。0 = 在最后一条消息之后,1 = 在最后一条消息之前,等等。",
- "The content of this prompt is pulled from elsewhere and cannot be edited here.": "The content of this prompt is pulled from elsewhere and cannot be edited here.",
+ "Injection depth. 0 = after the last message, 1 = before the last message, etc.": "注入深度。“0”为在最后一条消息之后,“1”为在最后一条消息之前,等等。",
+ "The content of this prompt is pulled from elsewhere and cannot be edited here.": "此提示词的内容是从其他地方提取的,无法在此处进行编辑。",
"Prompt": "提示词",
"The prompt to be sent.": "要发送的提示词。",
"This prompt cannot be overridden by character cards, even if overrides are preferred.": "即使选择覆盖,此提示词也不能被角色卡覆盖。",
@@ -1181,9 +1191,9 @@
"welcome_message_part_8": "您可随时通过",
"welcome_message_part_9": "图标来更改此设置。",
"Persona Name:": "用户角色名称:",
- "Temporarily disable automatic replies from this character": "暂时禁用此角色的自动回复",
- "Enable automatic replies from this character": "启用此角色的自动回复",
- "Trigger a message from this character": "从此角色触发消息",
+ "Temporarily disable automatic replies from this character": "临时禁言此角色",
+ "Enable automatic replies from this character": "解除禁言此角色",
+ "Trigger a message from this character": "强制触发该角色发言",
"Move up": "向上移动",
"Move down": "向下移动",
"View character card": "查看角色卡片",
@@ -1194,7 +1204,7 @@
"alternate_greetings_hint_1": "点击",
"alternate_greetings_hint_2": "按钮即可开始!",
"Alternate Greeting #": "额外问候语 #",
- "(This will be the first message from the character that starts every chat)": "(这是每次聊天开始时角色的第一条消息)",
+ "(This will be the first message from the character that starts every chat)": "(这是每次聊天开始时角色的第一条消息)",
"View contents": "查看内容",
"Remove the file": "删除文件",
"Author's Note": "作者注释",
@@ -1207,7 +1217,7 @@
"Insertion Frequency": "插入频率",
"(0 = Disable, 1 = Always)": "(“0”为禁用,“1”为始终)",
"User inputs until next insertion:": "用户输入直到下一次插入:",
- "Character Author's Note (Private)": "人物作者注(私有)",
+ "Character Author's Note (Private)": "人物作者注(私密)",
"Won't be shared with the character card on export.": "导出时不会与角色卡共享。",
"Will be automatically added as the author's note for this character. Will be used in groups, but can't be modified when a group chat is open.": "会自动添加为该角色的作者注解,在群聊中使用,但群聊开启时无法修改。",
"Use character author's note": "使用角色作者的注释",
@@ -1234,9 +1244,9 @@
"Insertion Depth:": "插入深度:",
"Token Probabilities": "词符概率",
"Select a token to see alternatives considered by the AI.": "选择一个词符来查看 AI 考虑的替代方案。",
- "Reroll with the entire prefix": "Reroll with the entire prefix",
+ "Reroll with the entire prefix": "使用完整前缀重新生成",
"Not connected to API!": "未连接到API!",
- "Type a message, or /? for help": "键入消息,或 /? 获取帮助",
+ "Type a message, or /? for help": "输入想发送的消息,或输入 /? 获取帮助",
"Continue script execution": "继续执行脚本",
"Pause script execution": "暂停执行脚本",
"Abort script execution": "中止执行脚本",
@@ -1355,7 +1365,7 @@
"You are in offline mode. Click on the image below to set the expression.": "您处于离线模式。点击下方图片即可设置表情。",
"Sprite Folder Override": "表情文件夹覆盖",
"Use a forward slash to specify a subfolder. Example: _space": "使用正斜杠指定子文件夹。例如:",
- "Upload sprite pack (ZIP)": "上传表情包 (ZIP)",
+ "Upload sprite pack (ZIP)": "上传表情包(ZIP)",
"Remove all image overrides": "删除所有图片覆盖",
"Create new folder in the _space": "创建新文件夹到",
"folder of your user data directory and name it as the name of the character.": "用户数据目录的文件夹并将其命名为角色的名称。",
@@ -1460,12 +1470,12 @@
"ext_regex_global_scripts": "全局正则脚本",
"ext_regex_global_scripts_desc": "影响所有角色,保存在本地设定中",
"ext_regex_scoped_scripts": "局部正则脚本",
- "ext_regex_disallow_scoped": "Disallow using scoped regex",
- "ext_regex_allow_scoped": "允许使用范围正则表达式",
+ "ext_regex_disallow_scoped": "不允许使用局部正则",
+ "ext_regex_allow_scoped": "允许使用局部正则",
"ext_regex_scoped_scripts_desc": "只影响当前角色,保存在角色卡片中",
"Regex Editor": "正则表达式编辑器",
"Test Mode": "测试模式",
- "ext_regex_desc": "正则是一款使用正则表达式查找/替换字符串的工具。如果您想了解更多信息,请点击标题旁边的 ?。",
+ "ext_regex_desc": "“正则”是一个使用“正则表达式”来查找/替换字符串的工具。如果您想了解更多信息,请点击标题旁边的“?”。",
"Input": "输入",
"ext_regex_test_input_placeholder": "在此输入...",
"Output": "输出",
@@ -1566,7 +1576,7 @@
"Click to set": "点击设置",
"You can find your API key in the Stability AI dashboard.": "您可以在 Stability AI 仪表板中找到您的 API 密钥。",
"Style Preset": "风格预设",
- "Prompt Upsampling": "Prompt Upsampling",
+ "Prompt Upsampling": "提示词增强(Upsampling)",
"Sampling method": "采样方法",
"Scheduler": "调度器",
"Resolution": "分辨率",
@@ -1584,7 +1594,7 @@
"Karras": "Karras",
"Not all samplers supported.": "并非所有采样器都受支持。",
"sd_adetailer_face": "Use ADetailer with face model during the generation. The ADetailer extension must be installed on the backend.",
- "Use ADetailer (Face)": "使用 ADetailer (Face)",
+ "Use ADetailer (Face)": "使用 ADetailer(脸部)",
"SMEA versions of samplers are modified to perform better at high resolution.": "SMEA 版本的采样器经过修改,在高分辨率下性能更佳。",
"SMEA": "中小企业协会",
"DYN variants of SMEA samplers often lead to more varied output, but may fail at very high resolutions.": "SMEA 采样器的 DYN 变体通常会产生更加多样化的输出,但在非常高的分辨率下可能会失败。",
@@ -1627,9 +1637,9 @@
"Select TTS Provider": "选择 文本转语音 的服务提供商",
"tts_enabled": "已启用",
"Narrate user messages": "朗读用户消息",
- "Auto Generation": "Auto Generation",
- "Requires auto generation to be enabled.": "Requires auto generation to be enabled.",
- "Narrate by paragraphs (when streaming)": "Narrate by paragraphs (when streaming)",
+ "Auto Generation": "自动生成",
+ "Requires auto generation to be enabled.": "需要启用自动生成功能。",
+ "Narrate by paragraphs (when streaming)": "按段朗读(流式播放时)",
"Only narrate quotes": "只朗读引号内文本",
"Ignore text, even quotes, inside asterisk": "不朗读所有*星号内文本*,即使其被引号包裹",
"Narrate only the translated text": "只朗读翻译后文本",
@@ -1658,9 +1668,9 @@
"Enable for files": "为文件启用",
"Translate files into English before processing": "处理之前将文件翻译成英文",
"Message attachments": "消息附件",
- "Size threshold (KB)": "大小阈值 (KB)",
+ "Size threshold (KB)": "大小阈值(KB)",
"Chunk size (chars)": "块大小(字符)",
- "Chunk overlap (%)": "块重叠 (%)",
+ "Chunk overlap (%)": "块重叠(%)",
"Retrieve chunks": "检索块",
"Data Bank files": "数据库文件",
"Injection Template": "注入模板",
@@ -1691,6 +1701,7 @@
"This will create a new subfolder...": "这将在 /data/ 目录中创建一个新的子文件夹,以用户的句柄作为文件夹名称。",
"Note:": "提示:",
"this chat is temporary and will be deleted as soon as you leave it.": "此聊天会话是临时的,会在你离开时被删除。",
+ "Enter a new display name:": "输入一个新的昵称:",
"Current Password:": "当前密码:",
"New Password:": "新密码:",
"Confirm New Password:": "确认新密码:",
@@ -1711,22 +1722,25 @@
"chat_rename_1": "输入聊天的新名称:",
"chat_rename_2": "注意!!使用已有文件名会导致错误!!",
"chat_rename_3": "此举会将次聊天与标记为“检查点”的聊天解绑。",
- "chat_rename_4": "不需要在结尾添加 '.jsonl'",
+ "chat_rename_4": "不需要在结尾添加 '.JSONL'",
+ "Enter Checkpoint Name:": "输入检查点名称:",
+ "(Leave empty to auto-generate)": "(留空以自动生成)",
+ "The currently existing checkpoint will be unlinked and replaced with the new checkpoint, but can still be found in the Chat Management.": "当前检查点将会被解绑并替换为新的检查点,但仍可在聊天管理中找到。",
"Include Body Parameters": "包括主体参数",
"custom_include_body_desc": "聊天完成请求主体中要包含的参数(YAML 对象)\n\n示例:\n- top_k:20\n- repetition_penalty:1.1",
"Exclude Body Parameters": "排除主体参数",
"custom_exclude_body_desc": "要从聊天完成请求主体中排除的参数(YAML 数组)\n\n示例:\n- frequency_penalty\n- presence_penalty",
"Include Request Headers": "包含请求标头",
"custom_include_headers_desc": "聊天完成请求的附加标头(YAML 对象)\n\n示例:\n- CustomHeader:自定义值\n- AnotherHeader:自定义值",
- "Debug Warning": "此类别中的功能仅供高级用户使用。如果您不确定后果,请不要点击任何内容。",
+ "Functions in this category are for advanced users only. Don't click anything if you're not sure about the consequences.": "此类别中的功能仅供高级用户使用。如果您不确定后果,请不要点击任何内容。",
"THIS IS PERMANENT!": "此操作不可逆!",
"Also delete the chat files": "同时删除聊天文件",
"Are you sure you want to delete this user?": "您确定要删除该用户吗?",
"Deleting:": "删除:",
- "Also wipe user data.": "也清空用户数据。",
+ "Also wipe user data.": "同时清空用户数据",
"Warning:": "警告:",
"This action is irreversible.": "此操作不可逆。",
- "Type the user's handle below to confirm:": "在下面输入用户的名称以确认:",
+ "Type the user's handle below to confirm:": "在下面输入此用户的用户句柄以确认删除操作:",
"Are you sure you want to duplicate this character?": "你确定要复制这个角色吗?",
"If you just want to start a new chat with the same character...": "如果你只是想要与此角色开启一个新的聊天,只需点击聊天左下方菜单中的“开始新聊天”按钮。",
"Forbid Media Override explanation": "当前角色/群聊成员使用外部媒体的能力。",
@@ -1764,11 +1778,11 @@
"help_format_26": "内联代码",
"help_format_27": "> 文本",
"help_format_28": "显示为块引用(请注意 > 后面的空格)",
- "help_format_29": "# 文本",
+ "help_format_29": "# 文本",
"help_format_30": "显示为大标题(注意空格)",
- "help_format_32": "## 文本",
+ "help_format_32": "## 文本",
"help_format_33": "显示为中等标题(注意空格)",
- "help_format_35": "### 文本",
+ "help_format_35": "### 文本",
"help_format_36": "显示为小标题(注意空格)",
"help_1": "您好!请选择您想要详细了解的帮助主题:",
"help_2": "斜线命令",
@@ -1815,15 +1829,33 @@
"char_import_2": "Chub 知识书(直链或ID)",
"char_import_3": "JanitorAI 角色(直链或UUID)",
"char_import_4": "Pygmalion.chat 角色(直链或UUID)",
- "char_import_5": "AICharacterCard.com 角色(直链或ID)",
+ "char_import_5": "AICharacterCards.com 角色(直链或ID)",
"char_import_6": "被允许的PNG直链(请参阅",
"char_import_7": ")",
"char_import_8": "RisuRealm 角色(直链)",
"Supports importing multiple characters.": "支持导入多个角色。",
"Write each URL or ID into a new line.": "将每个 URL 或 ID 写入新行。",
- "Show Raw Prompt": "显示原始提示",
- "Copy Prompt": "复制提示",
- "Show Prompt Differences": "显示提示差异",
+ "Enter the Git URL of the extension to install": "输入扩展程序的 Git URL 以安装",
+ "Disclaimer:": "免责声明:",
+ "Please be aware that using external extensions can have unintended side effects and may pose security risks. Always make sure you trust the source before importing an extension. We are not responsible for any damage caused by third-party extensions.": "使用外部的扩展程序可能存在意料外的副作用和安全隐患。在导入扩展程序前,请一定确认其来源可信。我们不为第三方扩展程序造成的任何损失负责。",
+ "Prompt Itemization": "提示词拆分",
+ "Show Raw Prompt": "显示原始提示词",
+ "Copy Prompt": "复制提示词",
+ "Show Prompt Differences": "显示提示词差异",
+ "API/Model:": "API/模型:",
+ "Preset:": "预设:",
+ "Tokenizer:": "分词器:",
+ "Only the white numbers really matter. All numbers are estimates. Grey color items may not have been included in the context due to certain prompt format settings.": "只有白色的数字作数。所有数字均为估算。\n 灰色的数字可能因特定的提示词处理规则而被排除在外。",
+ "System Info:": "系统信息:",
+ "Prompt Tokens:": "提示词Token:",
+ "World Info:": "世界书:",
+ "Chat History:": "聊天记录:",
+ "Extensions:": "扩展程序:",
+ "Bias:": "Bias:",
+ "Total Tokens in Prompt:": "提示词的总Token数量:",
+ "Max Context": "最大上下文:",
+ "(Context Size - Response Length)": "(上下文长度 - 回复长度)",
+ ":": ":",
"System-wide Replacement Macros (in order of evaluation):": "系统范围的替换宏(按评估顺序):",
"help_macros_1": "仅适用于斜线命令批处理。替换为上一个命令的返回结果。",
"help_macros_2": "仅插入一个换行符。",
@@ -1913,7 +1945,9 @@
"help_macros_68": "替换为全局变量“name”的值减 1 的结果",
"help_macros_69": "替换为范围变量“name”的值",
"help_macros_70": "用范围变量“name”的索引处的项目值(对于数组/列表或对象/字典)替换",
+ "Choose what to export": "选择您想要导出什么:",
"{{name}}": "{{name}}",
+ "Choose what to import": "选择您想要导入什么:",
"If necessary, you can later restore this chat file from the /backups folder": "若需要,您可稍后在 /backups 文件夹中恢复此聊天文件。",
"Also delete the current chat file": "同时删除当前聊天文件",
"Persona Lorebook for": "Persona Lorebook for",
@@ -1938,8 +1972,13 @@
"If you have sufficient credits, please try again later.": "若您有足够的余额,请稍后再试。",
"Are you sure you want to reset your settings to factory defaults?": "您确定要将您的设置重置为出厂默认设置吗?",
"Don't forget to save a snapshot of your settings before proceeding.": "在继续之前,不要忘记保存您的设置快照。",
+ "Enter your password below to confirm:": "输入您的密码以确认:",
"Chat Scenario Override": "聊天场景覆盖",
"Remove": "移除",
+ "Unique to this chat.": "Unique to this chat.",
+ "All group members will use the following scenario text instead of what is specified in their character cards.": "All group members will use the following scenario text instead of what is specified in their character cards.",
+ "The following scenario text will be used instead of the value set in the character card.": "The following scenario text will be used instead of the value set in the character card.",
+ "Checkpoints inherit the scenario override from their parent, and can be changed individually after that.": "Checkpoints inherit the scenario override from their parent, and can be changed individually after that.",
"Settings Snapshots": "设置快照",
"Record a snapshot of your current settings.": "记录当前设置的快照。",
"Make a Snapshot": "制作快照",
@@ -1962,6 +2001,7 @@
"Click on the folder icon to use this tag as a folder.": "点击文件夹图标来将此标签作为一个文件夹。",
"Use alphabetical sorting": "按字母顺序排列",
"tags_sorting_desc": "启用后,标签在创建或重命名时会自动按字母顺序排序。\n禁用后,新标签会追加到末尾。\n\n如果通过拖动手动重新排列标签,则自动排序将被禁用。",
+ "Are you sure you want to delete the theme?": "你确定要删除这个主题吗?",
"Hi,": "嗨,",
"To enable multi-account features, restart the SillyTavern server with": "要启用多帐户功能,请使用以下命令重新启动 SillyTavern 服务器",
"set to true in the config.yaml file.": "在 config.yaml 文件中设置为 true。",
@@ -1982,6 +2022,8 @@
"Reset Settings": "重置设置",
"Wipe all user data and reset your account to factory settings.": "删除所有用户数据并将您的账号重置为默认设置。",
"Reset Everything": "重置一切",
+ "This will delete all your settings and data. There will be no undo button. Make sure you have a backup before proceeding.": "This will delete all your settings and data. There will be no undo button.\n Make sure you have a backup before proceeding.",
+ "Account reset code has been posted to the server console.": "Account reset code has been posted to the server console.",
"Reset Code:": "重置代码:",
"Want to update?": "获取最新版本",
"How to start chatting?": "如何快速开始聊天?",
@@ -2003,8 +2045,8 @@
"Post a GitHub issue": "在 GitHub 发布问题",
"Contact the developers": "联系开发者",
"If you're connected to an API, try asking me something!": "若您已经配置好API,尝试发送些什么吧!",
- "Title/Memo": "标题/备忘录",
- "Strategy": "Strategy",
- "Position": "位置",
- "Trigger %": "触发率 %"
+ "Title/Memo": "标题(备忘)",
+ "Strategy": "触发策略",
+ "Position": "插入位置",
+ "Trigger %": "触发概率%"
}
diff --git a/public/locales/zh-tw.json b/public/locales/zh-tw.json
index e1a5eec72..062aa7942 100644
--- a/public/locales/zh-tw.json
+++ b/public/locales/zh-tw.json
@@ -28,14 +28,14 @@
"Prose Augmenter": "散文增強器",
"Text Adventure": "文字冒險",
"response legth(tokens)": "回應長度(符元數)",
- "Streaming": "串流",
- "Streaming_desc": "生成時逐位顯示回應。當此功能關閉時,回應將在完成後一次顯示。",
+ "Streaming": "即時串流",
+ "Streaming_desc": "逐字顯示生成中的回應內容。關閉時,回應將在生成完成後一次性顯示。",
"context size(tokens)": "上下文長度(符元數)",
"unlocked": "解鎖",
"Only enable this if your model supports context sizes greater than 8192 tokens": "僅在您的模型支援超過 8192 個符元的上下文長度時啟用此功能",
- "Max prompt cost:": "最大提示詞費用",
- "Display the response bit by bit as it is generated.": "生成時逐位顯示回應。",
- "When this is off, responses will be displayed all at once when they are complete.": "關閉時,回應將在完成後一次性顯示。",
+ "Max prompt cost:": "最大提示詞費用:",
+ "Display the response bit by bit as it is generated.": "逐字顯示生成中的回應內容。",
+ "When this is off, responses will be displayed all at once when they are complete.": "關閉時,回應將在生成完成後一次性顯示。",
"Temperature": "溫度",
"rep.pen": "重複懲罰",
"Rep. Pen. Range.": "重複懲罰範圍",
@@ -880,7 +880,7 @@
"popup-button-no": "取消",
"popup-button-cancel": "關閉",
"popup-button-import": "匯入",
- "Advanced Defininitions": "- 進階定義",
+ "Advanced Definitions": "進階定義",
"Prompt Overrides": "提示詞覆寫",
"(For Chat Completion and Instruct Mode)": "(用於聊天補全和指令模式)",
"Insert {{original}} into either box to include the respective default prompt from system settings.": "在任一框中插入 {{original}} 以包含系統設定中的預設提示詞。",
@@ -1381,7 +1381,7 @@
"char_import_2": "Chub Lorebook(直接連結或 ID)",
"char_import_3": "JanitorAI 角色(直接連結或 ID)",
"char_import_4": "Pygmalion.chat 角色(直接連結或 ID)",
- "char_import_5": "AICharacterCard.com 角色(直接連結或 ID)",
+ "char_import_5": "AICharacterCards.com 角色(直接連結或 ID)",
"char_import_6": "直接 PNG 連結(請參閱",
"char_import_7": "對於允許的主機)",
"char_import_8": "RisuRealm角色(直接連結)",
@@ -1454,7 +1454,7 @@
"Any contents here will replace the default Post-History Instructions used for this character. (v2 specpost_history_instructions)": "此處填入的內容將取代該角色的默認聊天歷史後指示(Post-History Instructions)。\n(v2 格式:specpost_history_instructions)",
"comma delimited,no spaces between": "逗號分割,無需空格",
"e.g. black-forest-labs/FLUX.1-dev": "例如:black-forest-labs/FLUX.1-dev",
- "Example: gpt-3.5-turbo": "例如:gpt-3.5-turbo",
+ "Example: gpt-4o": "例如:gpt-4o",
"Example: http://localhost:1234/v1": "例如:http://localhost:1234/v1",
"popup-button-crop": "裁剪",
"(disabled when max recursion steps are used)": "(當最大遞歸步驟數使用時將停用)",
@@ -1745,7 +1745,7 @@
"Sprite Folder Override": "表情立繪資料夾覆蓋",
"Sprite set:": "立繪組:",
"Show Gallery": "查看圖庫",
- "Sticky": "固定",
+ "Sticky": "黏性",
"Style Preset": "預設樣式",
"Summarize chat messages for vector generation": "摘要聊天訊息以進行向量化處理",
"Summarize chat messages when sending": "傳送時摘要聊天內容",
@@ -2124,7 +2124,7 @@
"from other websites": "以從其他網站新增。",
"Go to the": "前往",
"to install additional features.": "以安裝更多功能。",
- "If you're connected to an API, try asking me something!": "若您已連接 API,嘗試問我一些問題吧!",
+ "If you're connected to an API, try asking me something!": "若您已連線 API,嘗試問我一些問題吧!",
"Title/Memo": "標題/備註",
"Strategy": "插入策略",
"Position": "位置",
@@ -2386,5 +2386,71 @@
"Your preset contains proxy and/or custom endpoint settings.": "此預設包含代理和/或自訂端點設定。",
"Do you want to remove these fields before exporting?": "是否要在匯出前移除這些欄位?",
"Delete the preset? This action is irreversible and your current settings will be overwritten.": "確定刪除此預設?刪除後無法復原,且此設定將被覆蓋。",
- "Update all": "全部更新"
+ "Update all": "全部更新",
+ "Automatically chooses an alternative provider if chosen providers can't serve your request.": "當所選提供者無法滿足您的請求時,自動選擇替代提供者。",
+ "Use extension settings": "使用擴充功能設定",
+ "To use instruct formatting, switch to OpenRouter under Text Completion API.": "若要使用指令格式,請在文本補全 API 下切換至 OpenRouter。",
+ "Automatically 'continue' a response if the model stopped before reaching a certain amount of tokens.": "如果模型在達到一定數量的符元前停止,則自動繼續生成回應。",
+ "Toggle entry's active state.": "切換條目的啟用狀態。",
+ "Non-sticky": "無黏性",
+ "No cooldown": "無冷卻時間",
+ "No delay": "無延遲",
+ "Included settings:": "包含設定:",
+ "Click on the setting name to omit it from the profile.": "點擊設定名稱以從設定檔中省略。",
+ "Tints the chat background and/or character sprites.": "調整聊天背景或角色圖片的色調。",
+ "Only chunk on custom boundary": "僅在自定邊界進行分塊(chunk)",
+ "help_macros_firstDisplayedMessageId": "載入到可見聊天中的第一則訊息的 ID。",
+ "Couldn't import tags:": "無法匯入標籤:",
+ "Select providers. No selection = all providers.": "選擇供應商。未選擇=所有供應商。",
+ "Select a model": "選擇模型",
+ "Search models...": "搜尋模型⋯",
+ "[Currently loaded]": "[當前加載]",
+ "Search providers...": "搜尋供應商⋯",
+ "No-sticky": "無固定",
+ "Create a new World Info": "建立新世界資訊",
+ "Enter a name for the new file:": "輸入新檔案的名稱:",
+ "Valid World Info file name is required": "需要有效的世界資訊檔案名稱",
+ "World Info file has an invalid format": "世界資訊檔案格式無效",
+ "World Info file has no entries": "世界資訊檔案中無任何條目",
+ "Character not found.": "未找到該角色。",
+ "Open a chat to get a name of the chat-bound lorebook": "開啟聊天以取得綁定聊天的知識書名稱。",
+ "File is not valid: ${0}": "檔案無效或格式不正確:${0}",
+ "The world with ${0} is invalid or corrupted.": "世界 ${0} 無效或已損壞。",
+ "Deactivated all worlds": "已停用所有世界",
+ "No world found named: ${0}": "未找到名為 ${0} 的世界",
+ "Activated world: ${0}": "已啟用世界:${0}",
+ "Deactivated world: ${0}": "已停用世界:${0}",
+ "World was not active: ${0}": "世界 ${0} 未啟用",
+ "The world '${0}' has been imported and linked to the character successfully.": "世界「${0}」已成功匯入並綁定至角色。",
+ "World/Lorebook imported": "世界/知識書已匯入",
+ "Are you sure you want to import '${0}'?": "確定要匯入「${0}」嗎?",
+ "Built-in Extensions:": "內建擴充功能:",
+ "Installed Extensions:": "已安裝的擴充功能:",
+ "Loading third-party extensions... Please wait...": "正在載入第三方擴充功能,請稍候⋯",
+ "The page will be reloaded shortly...": "頁面即將重新載入⋯",
+ "Extensions state changed": "擴充功能狀態已更改",
+ "Error loading extensions. See browser console for details.": "載入擴充功能時出現錯誤。詳情請查看瀏覽器控制台。",
+ "You don't have permission to update global extensions.": "您無權更新全域擴充功能。",
+ "Extension update failed": "擴充功能更新失敗",
+ "Extension ${0} updated to ${1}": "擴充功能 ${0} 已更新至 ${1}",
+ "Reload the page to apply updates": "重新加載頁面以使用更新",
+ "You don't have permission to delete global extensions.": "您無權刪除全域擴充功能。",
+ "Are you sure you want to delete ${0}?": "確定要刪除 ${0} 嗎?",
+ "You don't have permission to move extensions.": "您無權移動擴充功能。",
+ "Are you sure you want to move ${0} to your local extensions? This will make it available only for you.": "確定要將 ${0} 移至本地擴充功能嗎?此後僅您可使用。",
+ "Are you sure you want to move ${0} to the global extensions? This will make it available for all users.": "確定要將 ${0} 移至全域擴充功能嗎?此後所有使用者皆可使用。",
+ "Extension ${0} moved.": "擴充功能 ${0} 已移動。",
+ "Extension ${0} deleted": "擴充功能 ${0} 已刪除。",
+ "Please wait...": "請稍候⋯",
+ "Installing extension": "正在安裝擴充功能",
+ "Extension installation failed": "擴充功能安裝失敗",
+ "Extension '${0}' by ${1} (version ${2}) has been installed successfully!": "擴充功能 '${0}' 由 ${1} 提供(版本 ${2})已成功安裝!",
+ "Extension installation successful": "擴充功能安裝成功",
+ "Extension updates available": "有可用的擴充功能更新",
+ "Auto-updating extensions. This may take several minutes.": "正在自動更新擴充功能。這可能需要幾分鐘時間。",
+ "Install": "安裝",
+ "Modules provided by your Extras API:": "由您的 Extras API 提供的模組:",
+ "Not connected to the API!": "未連線到 API!",
+ "ext_type_system": "這是內建的擴充功能,無法刪除,且會跟隨系統更新。",
+ "Valid": "已驗證"
}
diff --git a/public/script.js b/public/script.js
index d5ed97d4d..dc3d299da 100644
--- a/public/script.js
+++ b/public/script.js
@@ -167,6 +167,8 @@ import {
flashHighlight,
isTrueBoolean,
toggleDrawer,
+ isElementInViewport,
+ copyText,
} from './scripts/utils.js';
import { debounce_timeout } from './scripts/constants.js';
@@ -441,6 +443,7 @@ export const event_types = {
MESSAGE_DELETED: 'message_deleted',
MESSAGE_UPDATED: 'message_updated',
MESSAGE_FILE_EMBEDDED: 'message_file_embedded',
+ MORE_MESSAGES_LOADED: 'more_messages_loaded',
IMPERSONATE_READY: 'impersonate_ready',
CHAT_CHANGED: 'chat_id_changed',
GENERATION_AFTER_COMMANDS: 'GENERATION_AFTER_COMMANDS',
@@ -573,7 +576,7 @@ export const DEFAULT_SAVE_EDIT_TIMEOUT = debounce_timeout.relaxed;
/** @type {debounce_timeout} The debounce timeout used for printing. debounce_timeout.quick: 100 ms */
export const DEFAULT_PRINT_TIMEOUT = debounce_timeout.quick;
-export const saveSettingsDebounced = debounce(() => saveSettings(), DEFAULT_SAVE_EDIT_TIMEOUT);
+export const saveSettingsDebounced = debounce((loopCounter = 0) => saveSettings(loopCounter), DEFAULT_SAVE_EDIT_TIMEOUT);
export const saveCharacterDebounced = debounce(() => $('#create_button').trigger('click'), DEFAULT_SAVE_EDIT_TIMEOUT);
/**
@@ -721,6 +724,7 @@ async function getSystemMessages() {
is_user: false,
is_system: true,
mes: await renderTemplateAsync('assistantNote'),
+ uses_system_ui: true,
extra: {
isSmallSys: true,
},
@@ -1827,10 +1831,10 @@ export async function replaceCurrentChat() {
}
}
-export function showMoreMessages() {
+export async function showMoreMessages(messagesToLoad = null) {
const firstDisplayedMesId = $('#chat').children('.mes').first().attr('mesid');
let messageId = Number(firstDisplayedMesId);
- let count = power_user.chat_truncation || Number.MAX_SAFE_INTEGER;
+ let count = messagesToLoad || power_user.chat_truncation || Number.MAX_SAFE_INTEGER;
// If there are no messages displayed, or the message somehow has no mesid, we default to one higher than last message id,
// so the first "new" message being shown will be the last available message
@@ -1840,6 +1844,7 @@ export function showMoreMessages() {
console.debug('Inserting messages before', messageId, 'count', count, 'chat length', chat.length);
const prevHeight = $('#chat').prop('scrollHeight');
+ const isButtonInView = isElementInViewport($('#show_more_messages')[0]);
while (messageId > 0 && count > 0) {
let newMessageId = messageId - 1;
@@ -1852,8 +1857,12 @@ export function showMoreMessages() {
$('#show_more_messages').remove();
}
- const newHeight = $('#chat').prop('scrollHeight');
- $('#chat').scrollTop(newHeight - prevHeight);
+ if (isButtonInView) {
+ const newHeight = $('#chat').prop('scrollHeight');
+ $('#chat').scrollTop(newHeight - prevHeight);
+ }
+
+ await eventSource.emit(event_types.MORE_MESSAGES_LOADED);
}
export async function printMessages() {
@@ -2311,16 +2320,15 @@ export function addCopyToCodeBlocks(messageElement) {
const codeBlocks = $(messageElement).find('pre code');
for (let i = 0; i < codeBlocks.length; i++) {
hljs.highlightElement(codeBlocks.get(i));
- if (navigator.clipboard !== undefined) {
- const copyButton = document.createElement('i');
- copyButton.classList.add('fa-solid', 'fa-copy', 'code-copy', 'interactable');
- copyButton.title = 'Copy code';
- codeBlocks.get(i).appendChild(copyButton);
- copyButton.addEventListener('pointerup', function (event) {
- navigator.clipboard.writeText(codeBlocks.get(i).innerText);
- toastr.info(t`Copied!`, '', { timeOut: 2000 });
- });
- }
+ const copyButton = document.createElement('i');
+ copyButton.classList.add('fa-solid', 'fa-copy', 'code-copy', 'interactable');
+ copyButton.title = 'Copy code';
+ codeBlocks.get(i).appendChild(copyButton);
+ copyButton.addEventListener('pointerup', async function () {
+ const text = codeBlocks.get(i).innerText;
+ await copyText(text);
+ toastr.info(t`Copied!`, '', { timeOut: 2000 });
+ });
}
}
@@ -2714,12 +2722,13 @@ export function getStoppingStrings(isImpersonate, isContinue) {
* @param {string} quietImage Image to use for the quiet prompt
* @param {string} quietName Name to use for the quiet prompt (defaults to "System:")
* @param {number} [responseLength] Maximum response length. If unset, the global default value is used.
+ * @param {number} force_chid Character ID to use for this generation run. Works in groups only.
* @returns
*/
-export async function generateQuietPrompt(quiet_prompt, quietToLoud, skipWIAN, quietImage = null, quietName = null, responseLength = null) {
+export async function generateQuietPrompt(quiet_prompt, quietToLoud, skipWIAN, quietImage = null, quietName = null, responseLength = null, force_chid = null) {
console.log('got into genQuietPrompt');
const responseLengthCustomized = typeof responseLength === 'number' && responseLength > 0;
- let originalResponseLength = -1;
+ let eventHook = () => { };
try {
/** @type {GenerateOptions} */
const options = {
@@ -2729,12 +2738,17 @@ export async function generateQuietPrompt(quiet_prompt, quietToLoud, skipWIAN, q
force_name2: true,
quietImage: quietImage,
quietName: quietName,
+ force_chid: force_chid,
};
- originalResponseLength = responseLengthCustomized ? saveResponseLength(main_api, responseLength) : -1;
+ if (responseLengthCustomized) {
+ TempResponseLength.save(main_api, responseLength);
+ eventHook = TempResponseLength.setupEventHook(main_api);
+ }
return await Generate('quiet', options);
} finally {
- if (responseLengthCustomized) {
- restoreResponseLength(main_api, originalResponseLength);
+ if (responseLengthCustomized && TempResponseLength.isCustomized()) {
+ TempResponseLength.restore(main_api);
+ TempResponseLength.removeEventHook(main_api, eventHook);
}
}
}
@@ -3378,9 +3392,9 @@ export async function generateRaw(prompt, api, instructOverride, quietToLoud, sy
const abortController = new AbortController();
const responseLengthCustomized = typeof responseLength === 'number' && responseLength > 0;
- let originalResponseLength = -1;
const isInstruct = power_user.instruct.enabled && api !== 'openai' && api !== 'novel' && !instructOverride;
const isQuiet = true;
+ let eventHook = () => { };
if (systemPrompt) {
systemPrompt = substituteParams(systemPrompt);
@@ -3394,7 +3408,9 @@ export async function generateRaw(prompt, api, instructOverride, quietToLoud, sy
prompt = isInstruct ? (prompt + formatInstructModePrompt(name2, false, '', name1, name2, isQuiet, quietToLoud)) : (prompt + '\n');
try {
- originalResponseLength = responseLengthCustomized ? saveResponseLength(api, responseLength) : -1;
+ if (responseLengthCustomized) {
+ TempResponseLength.save(api, responseLength);
+ }
let generateData = {};
switch (api) {
@@ -3407,20 +3423,24 @@ export async function generateRaw(prompt, api, instructOverride, quietToLoud, sy
const koboldSettings = koboldai_settings[koboldai_setting_names[preset_settings]];
generateData = getKoboldGenerationData(prompt, koboldSettings, amount_gen, max_context, isHorde, 'quiet');
}
+ TempResponseLength.restore(api);
break;
case 'novel': {
const novelSettings = novelai_settings[novelai_setting_names[nai_settings.preset_settings_novel]];
generateData = getNovelGenerationData(prompt, novelSettings, amount_gen, false, false, null, 'quiet');
+ TempResponseLength.restore(api);
break;
}
case 'textgenerationwebui':
generateData = getTextGenGenerationData(prompt, amount_gen, false, false, null, 'quiet');
+ TempResponseLength.restore(api);
break;
case 'openai': {
generateData = [{ role: 'user', content: prompt.trim() }];
if (systemPrompt) {
generateData.unshift({ role: 'system', content: systemPrompt.trim() });
}
+ eventHook = TempResponseLength.setupEventHook(api);
} break;
}
@@ -3462,41 +3482,100 @@ export async function generateRaw(prompt, api, instructOverride, quietToLoud, sy
return message;
} finally {
- if (responseLengthCustomized) {
- restoreResponseLength(api, originalResponseLength);
+ if (responseLengthCustomized && TempResponseLength.isCustomized()) {
+ TempResponseLength.restore(api);
+ TempResponseLength.removeEventHook(api, eventHook);
}
}
}
-/**
- * Temporarily change the response length for the specified API.
- * @param {string} api API to use.
- * @param {number} responseLength Target response length.
- * @returns {number} The original response length.
- */
-function saveResponseLength(api, responseLength) {
- let oldValue = -1;
- if (api === 'openai') {
- oldValue = oai_settings.openai_max_tokens;
- oai_settings.openai_max_tokens = responseLength;
- } else {
- oldValue = amount_gen;
- amount_gen = responseLength;
- }
- return oldValue;
-}
+class TempResponseLength {
+ static #originalResponseLength = -1;
+ static #lastApi = null;
-/**
- * Restore the original response length for the specified API.
- * @param {string} api API to use.
- * @param {number} responseLength Target response length.
- * @returns {void}
- */
-function restoreResponseLength(api, responseLength) {
- if (api === 'openai') {
- oai_settings.openai_max_tokens = responseLength;
- } else {
- amount_gen = responseLength;
+ static isCustomized() {
+ return this.#originalResponseLength > -1;
+ }
+
+ /**
+ * Save the current response length for the specified API.
+ * @param {string} api API identifier
+ * @param {number} responseLength New response length
+ */
+ static save(api, responseLength) {
+ if (api === 'openai') {
+ this.#originalResponseLength = oai_settings.openai_max_tokens;
+ oai_settings.openai_max_tokens = responseLength;
+ } else {
+ this.#originalResponseLength = amount_gen;
+ amount_gen = responseLength;
+ }
+
+ this.#lastApi = api;
+ console.log('[TempResponseLength] Saved original response length:', TempResponseLength.#originalResponseLength);
+ }
+
+ /**
+ * Restore the original response length for the specified API.
+ * @param {string|null} api API identifier
+ * @returns {void}
+ */
+ static restore(api) {
+ if (this.#originalResponseLength === -1) {
+ return;
+ }
+ if (!api && this.#lastApi) {
+ api = this.#lastApi;
+ }
+ if (api === 'openai') {
+ oai_settings.openai_max_tokens = this.#originalResponseLength;
+ } else {
+ amount_gen = this.#originalResponseLength;
+ }
+
+ console.log('[TempResponseLength] Restored original response length:', this.#originalResponseLength);
+ this.#originalResponseLength = -1;
+ this.#lastApi = null;
+ }
+
+ /**
+ * Sets up an event hook to restore the original response length when the event is emitted.
+ * @param {string} api API identifier
+ * @returns {function(): void} Event hook function
+ */
+ static setupEventHook(api) {
+ const eventHook = () => {
+ if (this.isCustomized()) {
+ this.restore(api);
+ }
+ };
+
+ switch (api) {
+ case 'openai':
+ eventSource.once(event_types.CHAT_COMPLETION_SETTINGS_READY, eventHook);
+ break;
+ default:
+ eventSource.once(event_types.GENERATE_AFTER_DATA, eventHook);
+ break;
+ }
+
+ return eventHook;
+ }
+
+ /**
+ * Removes the event hook for the specified API.
+ * @param {string} api API identifier
+ * @param {function(): void} eventHook Previously set up event hook
+ */
+ static removeEventHook(api, eventHook) {
+ switch (api) {
+ case 'openai':
+ eventSource.removeListener(event_types.CHAT_COMPLETION_SETTINGS_READY, eventHook);
+ break;
+ default:
+ eventSource.removeListener(event_types.GENERATE_AFTER_DATA, eventHook);
+ break;
+ }
}
}
@@ -3827,7 +3906,7 @@ export async function Generate(type, { automatic_trigger, force_name2, quiet_pro
* @returns {string[]} Examples array with block heading
*/
function parseMesExamples(examplesStr) {
- if (examplesStr.length === 0 || examplesStr === '') {
+ if (!examplesStr || examplesStr.length === 0 || examplesStr === '') {
return [];
}
@@ -5394,7 +5473,7 @@ async function promptItemize(itemizedPrompts, requestedMesId) {
} else {
diffPrevPrompt.style.display = 'none';
}
- popup.dlg.querySelector('#copyPromptToClipboard').addEventListener('click', function () {
+ popup.dlg.querySelector('#copyPromptToClipboard').addEventListener('pointerup', async function () {
let rawPrompt = itemizedPrompts[PromptArrayItemForRawPromptDisplay].rawPrompt;
let rawPromptValues = rawPrompt;
@@ -5402,7 +5481,7 @@ async function promptItemize(itemizedPrompts, requestedMesId) {
rawPromptValues = rawPrompt.map(x => x.content).join('\n');
}
- navigator.clipboard.writeText(rawPromptValues);
+ await copyText(rawPromptValues);
toastr.info(t`Copied!`);
});
@@ -5423,20 +5502,24 @@ async function promptItemize(itemizedPrompts, requestedMesId) {
await popup.show();
}
-function setInContextMessages(lastmsg, type) {
+function setInContextMessages(msgInContextCount, type) {
$('#chat .mes').removeClass('lastInContext');
if (type === 'swipe' || type === 'regenerate' || type === 'continue') {
- lastmsg++;
+ msgInContextCount++;
}
- const lastMessageBlock = $('#chat .mes:not([is_system="true"])').eq(-lastmsg);
+ const lastMessageBlock = $('#chat .mes:not([is_system="true"])').eq(-msgInContextCount);
lastMessageBlock.addClass('lastInContext');
if (lastMessageBlock.length === 0) {
const firstMessageId = getFirstDisplayedMessageId();
$(`#chat .mes[mesid="${firstMessageId}"`).addClass('lastInContext');
}
+
+ // Update last id to chat. No metadata save on purpose, gets hopefully saved via another call
+ const lastMessageId = Math.max(0, chat.length - msgInContextCount);
+ chat_metadata['lastInContextMessageId'] = lastMessageId;
}
/**
@@ -5576,20 +5659,31 @@ function extractMessageFromData(data) {
return data;
}
- switch (main_api) {
- case 'kobold':
- return data.results[0].text;
- case 'koboldhorde':
- return data.text;
- case 'textgenerationwebui':
- return data.choices?.[0]?.text ?? data.content ?? data.response ?? '';
- case 'novel':
- return data.output;
- case 'openai':
- return data?.choices?.[0]?.message?.content ?? data?.choices?.[0]?.text ?? data?.text ?? data?.message?.content?.[0]?.text ?? data?.message?.tool_plan ?? '';
- default:
- return '';
+ function getTextContext() {
+ switch (main_api) {
+ case 'kobold':
+ return data.results[0].text;
+ case 'koboldhorde':
+ return data.text;
+ case 'textgenerationwebui':
+ return data.choices?.[0]?.text ?? data.content ?? data.response ?? '';
+ case 'novel':
+ return data.output;
+ case 'openai':
+ return data?.choices?.[0]?.message?.content ?? data?.choices?.[0]?.text ?? data?.text ?? data?.message?.content?.[0]?.text ?? data?.message?.tool_plan ?? '';
+ default:
+ return '';
+ }
}
+
+ const content = getTextContext();
+
+ if (main_api === 'openai' && oai_settings.chat_completion_source === chat_completion_sources.DEEPSEEK && oai_settings.show_thoughts) {
+ const thoughts = data?.choices?.[0]?.message?.reasoning_content ?? '';
+ return [thoughts, content].filter(x => x).join('\n\n');
+ }
+
+ return content;
}
/**
@@ -6855,12 +6949,23 @@ function selectKoboldGuiPreset() {
.trigger('change');
}
-export async function saveSettings(type) {
+export async function saveSettings(loopCounter = 0) {
if (!settingsReady) {
console.warn('Settings not ready, aborting save');
return;
}
+ const MAX_RETRIES = 3;
+ if (TempResponseLength.isCustomized()) {
+ if (loopCounter < MAX_RETRIES) {
+ console.warn('Response length is currently being overridden, scheduling another save');
+ saveSettingsDebounced(++loopCounter);
+ return;
+ }
+ console.error('Response length is currently being overridden, but the save loop has reached the maximum number of retries');
+ TempResponseLength.restore(null);
+ }
+
//console.log('Entering settings with name1 = '+name1);
return jQuery.ajax({
type: 'POST',
@@ -7299,7 +7404,7 @@ export function select_rm_info(type, charId, previousCharId = null) {
// Set a timeout so multiple flashes don't overlap
clearTimeout(importFlashTimeout);
importFlashTimeout = setTimeout(function () {
- if (type === 'char_import' || type === 'char_create') {
+ if (type === 'char_import' || type === 'char_create' || type === 'char_import_no_toast') {
// Find the page at which the character is located
const avatarFileName = charId;
const charData = getEntitiesList({ doFilter: true });
@@ -8851,24 +8956,61 @@ export async function processDroppedFiles(files, data = new Map()) {
'charx',
];
+ const avatarFileNames = [];
for (const file of files) {
const extension = file.name.split('.').pop().toLowerCase();
if (allowedMimeTypes.some(x => file.type.startsWith(x)) || allowedExtensions.includes(extension)) {
const preservedName = data instanceof Map && data.get(file);
- await importCharacter(file, preservedName);
+ const avatarFileName = await importCharacter(file, { preserveFileName: preservedName });
+ if (avatarFileName !== undefined) {
+ avatarFileNames.push(avatarFileName);
+ }
} else {
toastr.warning(t`Unsupported file type: ` + file.name);
}
}
+
+ if (avatarFileNames.length > 0) {
+ await importCharactersTags(avatarFileNames);
+ selectImportedChar(avatarFileNames[avatarFileNames.length - 1]);
+ }
+}
+
+/**
+ * Imports tags for the given characters
+ * @param {string[]} avatarFileNames character avatar filenames whose tags are to import
+ */
+async function importCharactersTags(avatarFileNames) {
+ await getCharacters();
+ for (let i = 0; i < avatarFileNames.length; i++) {
+ if (power_user.tag_import_setting !== tag_import_setting.NONE) {
+ const importedCharacter = characters.find(character => character.avatar === avatarFileNames[i]);
+ await importTags(importedCharacter);
+ }
+ }
+}
+
+/**
+ * Selects the given imported char
+ * @param {string} charId char to select
+ */
+function selectImportedChar(charId) {
+ let oldSelectedChar = null;
+ if (this_chid !== undefined) {
+ oldSelectedChar = characters[this_chid].avatar;
+ }
+ select_rm_info('char_import_no_toast', charId, oldSelectedChar);
}
/**
* Imports a character from a file.
* @param {File} file File to import
- * @param {string?} preserveFileName Whether to preserve original file name
- * @returns {Promise}
+ * @param {object} [options] - Options
+ * @param {string} [options.preserveFileName] Whether to preserve original file name
+ * @param {Boolean} [options.importTags=false] Whether to import tags
+ * @returns {Promise}
*/
-async function importCharacter(file, preserveFileName = '') {
+async function importCharacter(file, { preserveFileName = '', importTags = false } = {}) {
if (is_group_generating || is_send_press) {
toastr.error(t`Cannot import characters while generating. Stop the request and try again.`, t`Import aborted`);
throw new Error('Cannot import character while generating');
@@ -8904,19 +9046,14 @@ async function importCharacter(file, preserveFileName = '') {
if (data.file_name !== undefined) {
$('#character_search_bar').val('').trigger('input');
- let oldSelectedChar = null;
- if (this_chid !== undefined) {
- oldSelectedChar = characters[this_chid].avatar;
- }
+ toastr.success(t`Character Created: ${String(data.file_name).replace('.png', '')}`);
+ let avatarFileName = `${data.file_name}.png`;
+ if (importTags) {
+ await importCharactersTags([avatarFileName]);
- await getCharacters();
- select_rm_info('char_import', data.file_name, oldSelectedChar);
- if (power_user.tag_import_setting !== tag_import_setting.NONE) {
- let currentContext = getContext();
- let avatarFileName = `${data.file_name}.png`;
- let importedCharacter = currentContext.characters.find(character => character.avatar === avatarFileName);
- await importTags(importedCharacter);
+ selectImportedChar(data.file_name);
}
+ return avatarFileName;
}
}
@@ -9385,7 +9522,7 @@ API Settings: ${JSON.stringify(getSettingsContents[getSettingsContents.main_api
//console.log(logMessage);
try {
- await navigator.clipboard.writeText(logMessage);
+ await copyText(logMessage);
toastr.info('Your ST API setup data has been copied to the clipboard.');
} catch (error) {
toastr.error('Failed to copy ST Setup to clipboard:', error);
@@ -10450,24 +10587,18 @@ jQuery(async function () {
setTimeout(function () { $('#shadow_select_chat_popup').css('display', 'none'); }, animation_duration);
});
- if (navigator.clipboard === undefined) {
- // No clipboard support
- $('.mes_copy').remove();
- }
- else {
- $(document).on('pointerup', '.mes_copy', function () {
- if (this_chid !== undefined || selected_group || name2 === neutralCharacterName) {
- try {
- const messageId = $(this).closest('.mes').attr('mesid');
- const text = chat[messageId]['mes'];
- navigator.clipboard.writeText(text);
- toastr.info('Copied!', '', { timeOut: 2000 });
- } catch (err) {
- console.error('Failed to copy: ', err);
- }
+ $(document).on('pointerup', '.mes_copy', async function () {
+ if (this_chid !== undefined || selected_group || name2 === neutralCharacterName) {
+ try {
+ const messageId = $(this).closest('.mes').attr('mesid');
+ const text = chat[messageId]['mes'];
+ await copyText(text);
+ toastr.info('Copied!', '', { timeOut: 2000 });
+ } catch (err) {
+ console.error('Failed to copy: ', err);
}
- });
- }
+ }
+ });
$(document).on('pointerup', '.mes_prompt', async function () {
let mesIdForItemization = $(this).closest('.mes').attr('mesId');
@@ -10795,8 +10926,17 @@ jQuery(async function () {
return;
}
+ const avatarFileNames = [];
for (const file of e.target.files) {
- await importCharacter(file);
+ const avatarFileName = await importCharacter(file);
+ if (avatarFileName !== undefined) {
+ avatarFileNames.push(avatarFileName);
+ }
+ }
+
+ if (avatarFileNames.length > 0) {
+ await importCharactersTags(avatarFileNames);
+ selectImportedChar(avatarFileNames[avatarFileNames.length - 1]);
}
});
@@ -11329,8 +11469,8 @@ jQuery(async function () {
$('#avatar-and-name-block').slideToggle();
});
- $(document).on('mouseup touchend', '#show_more_messages', () => {
- showMoreMessages();
+ $(document).on('mouseup touchend', '#show_more_messages', async function () {
+ await showMoreMessages();
});
$(document).on('click', '.open_characters_library', async function () {
@@ -11352,4 +11492,3 @@ jQuery(async function () {
initCustomSelectedSamplers();
});
-
diff --git a/public/scripts/PromptManager.js b/public/scripts/PromptManager.js
index a49e4527b..783fb0c81 100644
--- a/public/scripts/PromptManager.js
+++ b/public/scripts/PromptManager.js
@@ -11,6 +11,7 @@ import { debounce_timeout } from './constants.js';
import { renderTemplateAsync } from './templates.js';
import { Popup } from './popup.js';
import { t } from './i18n.js';
+import { isMobile } from './RossAscends-mods.js';
function debouncePromise(func, delay) {
let timeoutId;
@@ -1562,6 +1563,7 @@ class PromptManager {
listItemHtml += `
- Hint:
- Click on the setting name to omit it from the profile.
+ Hint:
+ Click on the setting name to omit it from the profile.
diff --git a/public/scripts/extensions/gallery/index.js b/public/scripts/extensions/gallery/index.js
index a39634603..c34ea56e3 100644
--- a/public/scripts/extensions/gallery/index.js
+++ b/public/scripts/extensions/gallery/index.js
@@ -277,7 +277,7 @@ function makeMovable(id = 'gallery') {
const newElement = $(template);
newElement.css('background-color', 'var(--SmartThemeBlurTintColor)');
newElement.attr('forChar', id);
- newElement.attr('id', `${id}`);
+ newElement.attr('id', id);
newElement.find('.drag-grabber').attr('id', `${id}header`);
newElement.find('.dragTitle').text('Image Gallery');
//add a div for the gallery
diff --git a/public/scripts/extensions/quick-reply/html/settings.html b/public/scripts/extensions/quick-reply/html/settings.html
index a9a810fd8..210ad4f32 100644
--- a/public/scripts/extensions/quick-reply/html/settings.html
+++ b/public/scripts/extensions/quick-reply/html/settings.html
@@ -46,10 +46,12 @@
Edit Quick Replies
+
+
diff --git a/public/scripts/extensions/quick-reply/src/ui/SettingsUi.js b/public/scripts/extensions/quick-reply/src/ui/SettingsUi.js
index 920c2cd30..df6bb081f 100644
--- a/public/scripts/extensions/quick-reply/src/ui/SettingsUi.js
+++ b/public/scripts/extensions/quick-reply/src/ui/SettingsUi.js
@@ -1,4 +1,4 @@
-import { callPopup } from '../../../../../script.js';
+import { Popup } from '../../../../popup.js';
import { getSortableDelay } from '../../../../utils.js';
import { log, warn } from '../../index.js';
import { QuickReply } from '../QuickReply.js';
@@ -111,6 +111,7 @@ export class SettingsUi {
prepareQrEditor() {
// qr editor
+ this.dom.querySelector('#qr--set-rename').addEventListener('click', async () => this.renameQrSet());
this.dom.querySelector('#qr--set-new').addEventListener('click', async()=>this.addQrSet());
/**@type {HTMLInputElement}*/
const importFile = this.dom.querySelector('#qr--set-importFile');
@@ -119,7 +120,8 @@ export class SettingsUi {
importFile.value = null;
});
this.dom.querySelector('#qr--set-import').addEventListener('click', ()=>importFile.click());
- this.dom.querySelector('#qr--set-export').addEventListener('click', async()=>this.exportQrSet());
+ this.dom.querySelector('#qr--set-export').addEventListener('click', async () => this.exportQrSet());
+ this.dom.querySelector('#qr--set-duplicate').addEventListener('click', async () => this.duplicateQrSet());
this.dom.querySelector('#qr--set-delete').addEventListener('click', async()=>this.deleteQrSet());
this.dom.querySelector('#qr--set-add').addEventListener('click', async()=>{
this.currentQrSet.addQuickReply();
@@ -279,7 +281,7 @@ export class SettingsUi {
}
async deleteQrSet() {
- const confirmed = await callPopup(`Are you sure you want to delete the Quick Reply Set "${this.currentQrSet.name}"? This cannot be undone.`, 'confirm');
+ const confirmed = await Popup.show.confirm('Delete Quick Reply Set', `Are you sure you want to delete the Quick Reply Set "${this.currentQrSet.name}"? This cannot be undone.`);
if (confirmed) {
await this.doDeleteQrSet(this.currentQrSet);
this.rerender();
@@ -303,12 +305,52 @@ export class SettingsUi {
this.settings.save();
}
+ async renameQrSet() {
+ const newName = await Popup.show.input('Rename Quick Reply Set', 'Enter a new name:', this.currentQrSet.name);
+ if (newName && newName.length > 0) {
+ const existingSet = QuickReplySet.get(newName);
+ if (existingSet) {
+ toastr.error(`A Quick Reply Set named "${newName}" already exists.`);
+ return;
+ }
+ const oldName = this.currentQrSet.name;
+ this.currentQrSet.name = newName;
+ await this.currentQrSet.save();
+
+ // Update it in both set lists
+ this.settings.config.setList.forEach(set => {
+ if (set.set.name === oldName) {
+ set.set.name = newName;
+ }
+ });
+ this.settings.chatConfig?.setList.forEach(set => {
+ if (set.set.name === oldName) {
+ set.set.name = newName;
+ }
+ });
+ this.settings.save();
+
+ // Update the option in the current selected QR dropdown. All others will be refreshed via the prepare calls below.
+ /** @type {HTMLOptionElement} */
+ const option = this.currentSet.querySelector(`#qr--set option[value="${oldName}"]`);
+ option.value = newName;
+ option.textContent = newName;
+
+ this.currentSet.value = newName;
+ this.onQrSetChange();
+ this.prepareGlobalSetList();
+ this.prepareChatSetList();
+
+ console.info(`Quick Reply Set renamed from ""${oldName}" to "${newName}".`);
+ }
+ }
+
async addQrSet() {
- const name = await callPopup('Quick Reply Set Name:', 'input');
+ const name = await Popup.show.input('Create a new World Info', 'Enter a name for the new Quick Reply Set:');
if (name && name.length > 0) {
const oldQrs = QuickReplySet.get(name);
if (oldQrs) {
- const replace = await callPopup(`A Quick Reply Set named "${name}" already exists. Do you want to overwrite the existing Quick Reply Set? The existing set will be deleted. This cannot be undone.`, 'confirm');
+ const replace = Popup.show.confirm('Replace existing World Info', `A Quick Reply Set named "${name}" already exists. Do you want to overwrite the existing Quick Reply Set? The existing set will be deleted. This cannot be undone.`);
if (replace) {
const idx = QuickReplySet.list.indexOf(oldQrs);
await this.doDeleteQrSet(oldQrs);
@@ -369,7 +411,7 @@ export class SettingsUi {
qrs.init();
const oldQrs = QuickReplySet.get(props.name);
if (oldQrs) {
- const replace = await callPopup(`A Quick Reply Set named "${qrs.name}" already exists. Do you want to overwrite the existing Quick Reply Set? The existing set will be deleted. This cannot be undone.`, 'confirm');
+ const replace = Popup.show.confirm('Replace existing World Info', `A Quick Reply Set named "${name}" already exists. Do you want to overwrite the existing Quick Reply Set? The existing set will be deleted. This cannot be undone.`);
if (replace) {
const idx = QuickReplySet.list.indexOf(oldQrs);
await this.doDeleteQrSet(oldQrs);
@@ -421,6 +463,40 @@ export class SettingsUi {
URL.revokeObjectURL(url);
}
+ async duplicateQrSet() {
+ const newName = await Popup.show.input('Duplicate Quick Reply Set', 'Enter a name for the new Quick Reply Set:', `${this.currentQrSet.name} (Copy)`);
+ if (newName && newName.length > 0) {
+ const existingSet = QuickReplySet.get(newName);
+ if (existingSet) {
+ toastr.error(`A Quick Reply Set named "${newName}" already exists.`);
+ return;
+ }
+ const newQrSet = QuickReplySet.from(JSON.parse(JSON.stringify(this.currentQrSet)));
+ newQrSet.name = newName;
+ newQrSet.qrList = this.currentQrSet.qrList.map(qr => QuickReply.from(JSON.parse(JSON.stringify(qr))));
+ newQrSet.addQuickReply();
+ const idx = QuickReplySet.list.findIndex(it => it.name.toLowerCase().localeCompare(newName.toLowerCase()) == 1);
+ if (idx > -1) {
+ QuickReplySet.list.splice(idx, 0, newQrSet);
+ } else {
+ QuickReplySet.list.push(newQrSet);
+ }
+ const opt = document.createElement('option'); {
+ opt.value = newQrSet.name;
+ opt.textContent = newQrSet.name;
+ if (idx > -1) {
+ this.currentSet.children[idx].insertAdjacentElement('beforebegin', opt);
+ } else {
+ this.currentSet.append(opt);
+ }
+ }
+ this.currentSet.value = newName;
+ this.onQrSetChange();
+ this.prepareGlobalSetList();
+ this.prepareChatSetList();
+ }
+ }
+
selectQrSet(qrs) {
this.currentSet.value = qrs.name;
this.onQrSetChange();
diff --git a/public/scripts/extensions/tts/alltalk.js b/public/scripts/extensions/tts/alltalk.js
index 05e3db5d8..593ff3eac 100644
--- a/public/scripts/extensions/tts/alltalk.js
+++ b/public/scripts/extensions/tts/alltalk.js
@@ -388,7 +388,7 @@ class AllTalkTtsProvider {
}
async fetchRvcVoiceObjects() {
- if (this.settings.server_version !== 'v2') {
+ if (this.settings.server_version == 'v2') {
console.log('Skipping RVC voices fetch for V1 server');
return [];
}
diff --git a/public/scripts/extensions/vectors/index.js b/public/scripts/extensions/vectors/index.js
index 36b910f06..30278ddea 100644
--- a/public/scripts/extensions/vectors/index.js
+++ b/public/scripts/extensions/vectors/index.js
@@ -30,6 +30,8 @@ import { textgen_types, textgenerationwebui_settings } from '../../textgen-setti
import { SlashCommandParser } from '../../slash-commands/SlashCommandParser.js';
import { SlashCommand } from '../../slash-commands/SlashCommand.js';
import { ARGUMENT_TYPE, SlashCommandArgument, SlashCommandNamedArgument } from '../../slash-commands/SlashCommandArgument.js';
+import { SlashCommandEnumValue, enumTypes } from '../../slash-commands/SlashCommandEnumValue.js';
+import { slashCommandReturnHelper } from '../../slash-commands/SlashCommandReturnHelper.js';
import { callGenericPopup, POPUP_RESULT, POPUP_TYPE } from '../../popup.js';
import { generateWebLlmChatPrompt, isWebLlmSupported } from '../shared.js';
@@ -1613,25 +1615,55 @@ jQuery(async () => {
callback: async (args, query) => {
const clamp = (v) => Number.isNaN(v) ? null : Math.min(1, Math.max(0, v));
const threshold = clamp(Number(args?.threshold ?? settings.score_threshold));
+ const validateCount = (v) => Number.isNaN(v) || !Number.isInteger(v) || v < 1 ? null : v;
+ const count = validateCount(Number(args?.count)) ?? settings.chunk_count_db;
const source = String(args?.source ?? '');
const attachments = source ? getDataBankAttachmentsForSource(source, false) : getDataBankAttachments(false);
const collectionIds = await ingestDataBankAttachments(String(source));
- const queryResults = await queryMultipleCollections(collectionIds, String(query), settings.chunk_count_db, threshold);
-
- // Map collection IDs to file URLs
+ const queryResults = await queryMultipleCollections(collectionIds, String(query), count, threshold);
+
+ // Get URLs
const urls = Object
.keys(queryResults)
.map(x => attachments.find(y => getFileCollectionId(y.url) === x))
.filter(x => x)
.map(x => x.url);
+
+ // Gets the actual text content of chunks
+ const getChunksText = () => {
+ let textResult = '';
+ for (const collectionId in queryResults) {
+ const metadata = queryResults[collectionId].metadata?.filter(x => x.text)?.sort((a, b) => a.index - b.index)?.map(x => x.text)?.filter(onlyUnique) || [];
+ textResult += metadata.join('\n') + '\n\n';
+ }
+ return textResult;
+ };
+
+ if (args.return === 'chunks') {
+ return getChunksText();
+ }
- return JSON.stringify(urls);
+ // @ts-ignore
+ return slashCommandReturnHelper.doReturn(args.return ?? 'object', urls, { objectToStringFunc: list => list.join('\n') });
+
},
aliases: ['databank-search', 'data-bank-search'],
helpString: 'Search the Data Bank for a specific query using vector similarity. Returns a list of file URLs with the most relevant content.',
namedArgumentList: [
new SlashCommandNamedArgument('threshold', 'Threshold for the similarity score in the [0, 1] range. Uses the global config value if not set.', ARGUMENT_TYPE.NUMBER, false, false, ''),
+ new SlashCommandNamedArgument('count', 'Maximum number of query results to return.', ARGUMENT_TYPE.NUMBER, false, false, ''),
new SlashCommandNamedArgument('source', 'Optional filter for the attachments by source.', ARGUMENT_TYPE.STRING, false, false, '', ['global', 'character', 'chat']),
+ SlashCommandNamedArgument.fromProps({
+ name: 'return',
+ description: 'How you want the return value to be provided',
+ typeList: [ARGUMENT_TYPE.STRING],
+ defaultValue: 'object',
+ enumList: [
+ new SlashCommandEnumValue('chunks', 'Return the actual content chunks', enumTypes.enum, '{}'),
+ ...slashCommandReturnHelper.enumList({ allowObject: true })
+ ],
+ forceEnum: true,
+ })
],
unnamedArgumentList: [
new SlashCommandArgument('Query to search by.', ARGUMENT_TYPE.STRING, true, false),
diff --git a/public/scripts/group-chats.js b/public/scripts/group-chats.js
index 635343f33..c0d7f39e9 100644
--- a/public/scripts/group-chats.js
+++ b/public/scripts/group-chats.js
@@ -81,6 +81,7 @@ import { t } from './i18n.js';
export {
selected_group,
+ openGroupId,
is_group_automode_enabled,
hideMutedSprites,
is_group_generating,
@@ -1367,6 +1368,15 @@ function getGroupCharacterBlock(character) {
template.find('.ch_fav').val(isFav);
template.toggleClass('is_fav', isFav);
+ const auxFieldName = power_user.aux_field || 'character_version';
+ const auxFieldValue = (character.data && character.data[auxFieldName]) || '';
+ if (auxFieldValue) {
+ template.find('.character_version').text(auxFieldValue);
+ }
+ else {
+ template.find('.character_version').hide();
+ }
+
let queuePosition = groupChatQueueOrder.get(character.avatar);
if (queuePosition) {
template.find('.queue_position').text(queuePosition);
diff --git a/public/scripts/macros.js b/public/scripts/macros.js
index cbc38ded4..64f0b4647 100644
--- a/public/scripts/macros.js
+++ b/public/scripts/macros.js
@@ -26,6 +26,12 @@ Handlebars.registerHelper('helperMissing', function () {
* @typedef {(nonce: string) => string} MacroFunction
*/
+/**
+ * @typedef {Object} CustomMacro
+ * @property {string} key - Macro name (key)
+ * @property {string} description - Optional description of the macro
+ */
+
export class MacrosParser {
/**
* A map of registered macros.
@@ -33,12 +39,29 @@ export class MacrosParser {
*/
static #macros = new Map();
+ /**
+ * A map of macro descriptions.
+ * @type {Map}
+ */
+ static #descriptions = new Map();
+
+ /**
+ * Returns an iterator over all registered macros.
+ * @returns {IterableIterator}
+ */
+ static [Symbol.iterator] = function* () {
+ for (const macro of MacrosParser.#macros.keys()) {
+ yield { key: macro, description: MacrosParser.#descriptions.get(macro) };
+ }
+ };
+
/**
* Registers a global macro that can be used anywhere where substitution is allowed.
* @param {string} key Macro name (key)
* @param {string|MacroFunction} value A string or a function that returns a string
+ * @param {string} [description] Optional description of the macro
*/
- static registerMacro(key, value) {
+ static registerMacro(key, value, description = '') {
if (typeof key !== 'string') {
throw new Error('Macro key must be a string');
}
@@ -64,6 +87,10 @@ export class MacrosParser {
}
this.#macros.set(key, value);
+
+ if (typeof description === 'string' && description) {
+ this.#descriptions.set(key, description);
+ }
}
/**
@@ -88,6 +115,8 @@ export class MacrosParser {
if (!deleted) {
console.warn(`Macro ${key} was not registered`);
}
+
+ this.#descriptions.delete(key);
}
/**
@@ -202,10 +231,19 @@ export function getLastMessageId({ exclude_swipe_in_propress = true, filter = nu
* @returns {number|null} The ID of the first message in the context
*/
function getFirstIncludedMessageId() {
- const index = Number(document.querySelector('.lastInContext')?.getAttribute('mesid'));
+ return chat_metadata['lastInContextMessageId'];
+}
- if (!isNaN(index) && index >= 0) {
- return index;
+/**
+ * Returns the ID of the first displayed message in the chat.
+ *
+ * @returns {number|null} The ID of the first displayed message
+ */
+function getFirstDisplayedMessageId() {
+ const mesId = Number(document.querySelector('#chat .mes')?.getAttribute('mesid'));
+
+ if (!isNaN(mesId) && mesId >= 0) {
+ return mesId;
}
return null;
@@ -467,6 +505,7 @@ export function evaluateMacros(content, env, postProcessFn) {
{ regex: /{{lastUserMessage}}/gi, replace: () => getLastUserMessage() },
{ regex: /{{lastCharMessage}}/gi, replace: () => getLastCharMessage() },
{ regex: /{{firstIncludedMessageId}}/gi, replace: () => String(getFirstIncludedMessageId() ?? '') },
+ { regex: /{{firstDisplayedMessageId}}/gi, replace: () => String(getFirstDisplayedMessageId() ?? '') },
{ regex: /{{lastSwipeId}}/gi, replace: () => String(getLastSwipeId() ?? '') },
{ regex: /{{currentSwipeId}}/gi, replace: () => String(getCurrentSwipeId() ?? '') },
{ regex: /{{reverse:(.+?)}}/gi, replace: (_, str) => Array.from(str).reverse().join('') },
diff --git a/public/scripts/openai.js b/public/scripts/openai.js
index 54a53a181..d715377fb 100644
--- a/public/scripts/openai.js
+++ b/public/scripts/openai.js
@@ -1922,7 +1922,7 @@ async function sendOpenAIRequest(type, messages, signal) {
}
// Proxy is only supported for Claude, OpenAI, Mistral, and Google MakerSuite
- if (oai_settings.reverse_proxy && [chat_completion_sources.CLAUDE, chat_completion_sources.OPENAI, chat_completion_sources.MISTRALAI, chat_completion_sources.MAKERSUITE].includes(oai_settings.chat_completion_source)) {
+ if (oai_settings.reverse_proxy && [chat_completion_sources.CLAUDE, chat_completion_sources.OPENAI, chat_completion_sources.MISTRALAI, chat_completion_sources.MAKERSUITE, chat_completion_sources.DEEPSEEK].includes(oai_settings.chat_completion_source)) {
await validateReverseProxy();
generate_data['reverse_proxy'] = oai_settings.reverse_proxy;
generate_data['proxy_password'] = oai_settings.proxy_password;
@@ -2030,6 +2030,16 @@ async function sendOpenAIRequest(type, messages, signal) {
// https://api-docs.deepseek.com/api/create-chat-completion
if (isDeepSeek) {
generate_data.top_p = generate_data.top_p || Number.EPSILON;
+
+ if (generate_data.model.endsWith('-reasoner')) {
+ delete generate_data.top_p;
+ delete generate_data.temperature;
+ delete generate_data.frequency_penalty;
+ delete generate_data.presence_penalty;
+ delete generate_data.top_logprobs;
+ delete generate_data.logprobs;
+ delete generate_data.logit_bias;
+ }
}
if ((isOAI || isOpenRouter || isMistral || isCustom || isCohere || isNano) && oai_settings.seed >= 0) {
@@ -2085,6 +2095,7 @@ async function sendOpenAIRequest(type, messages, signal) {
let text = '';
const swipes = [];
const toolCalls = [];
+ const state = {};
while (true) {
const { done, value } = await reader.read();
if (done) return;
@@ -2095,9 +2106,9 @@ async function sendOpenAIRequest(type, messages, signal) {
if (Array.isArray(parsed?.choices) && parsed?.choices?.[0]?.index > 0) {
const swipeIndex = parsed.choices[0].index - 1;
- swipes[swipeIndex] = (swipes[swipeIndex] || '') + getStreamingReply(parsed);
+ swipes[swipeIndex] = (swipes[swipeIndex] || '') + getStreamingReply(parsed, state);
} else {
- text += getStreamingReply(parsed);
+ text += getStreamingReply(parsed, state);
}
ToolManager.parseToolCalls(toolCalls, parsed);
@@ -2129,14 +2140,27 @@ async function sendOpenAIRequest(type, messages, signal) {
}
}
-function getStreamingReply(data) {
+/**
+ * Extracts the reply from the response data from a chat completions-like source
+ * @param {object} data Response data from the chat completions-like source
+ * @param {object} state Additional state to keep track of
+ * @returns {string} The reply extracted from the response data
+ */
+function getStreamingReply(data, state) {
if (oai_settings.chat_completion_source === chat_completion_sources.CLAUDE) {
return data?.delta?.text || '';
} else if (oai_settings.chat_completion_source === chat_completion_sources.MAKERSUITE) {
return data?.candidates?.[0]?.content?.parts?.filter(x => oai_settings.show_thoughts || !x.thought)?.map(x => x.text)?.filter(x => x)?.join('\n\n') || '';
} else if (oai_settings.chat_completion_source === chat_completion_sources.COHERE) {
return data?.delta?.message?.content?.text || data?.delta?.message?.tool_plan || '';
- } else {
+ } else if (oai_settings.chat_completion_source === chat_completion_sources.DEEPSEEK) {
+ const hadThoughts = state.hadThoughts;
+ const thoughts = data.choices?.filter(x => oai_settings.show_thoughts || !x?.delta?.reasoning_content)?.[0]?.delta?.reasoning_content || '';
+ const content = data.choices?.[0]?.delta?.content || '';
+ state.hadThoughts = !!thoughts;
+ const separator = hadThoughts && !thoughts ? '\n\n' : '';
+ return [thoughts, separator, content].filter(x => x).join('\n\n');
+ } else {
return data.choices?.[0]?.delta?.content ?? data.choices?.[0]?.message?.content ?? data.choices?.[0]?.text ?? '';
}
}
@@ -3346,7 +3370,7 @@ async function getStatusOpen() {
chat_completion_source: oai_settings.chat_completion_source,
};
- if (oai_settings.reverse_proxy && [chat_completion_sources.CLAUDE, chat_completion_sources.OPENAI, chat_completion_sources.MISTRALAI, chat_completion_sources.MAKERSUITE].includes(oai_settings.chat_completion_source)) {
+ if (oai_settings.reverse_proxy && [chat_completion_sources.CLAUDE, chat_completion_sources.OPENAI, chat_completion_sources.MISTRALAI, chat_completion_sources.MAKERSUITE, chat_completion_sources.DEEPSEEK].includes(oai_settings.chat_completion_source)) {
await validateReverseProxy();
}
@@ -3971,6 +3995,8 @@ function onSettingsPresetChange() {
settings: oai_settings,
savePreset: saveOpenAIPreset,
}).finally(r => {
+ $('.model_custom_select').empty();
+
for (const [key, [selector, setting, isCheckbox]] of Object.entries(settingsToUpdate)) {
if (preset[key] !== undefined) {
if (isCheckbox) {
@@ -4202,7 +4228,7 @@ async function onModelChange() {
$('#openai_max_context').attr('max', max_32k);
} else if (value.includes('gemini-1.5-pro') || value.includes('gemini-exp-1206')) {
$('#openai_max_context').attr('max', max_2mil);
- } else if (value.includes('gemini-1.5-flash') || value.includes('gemini-2.0-flash-exp')) {
+ } else if (value.includes('gemini-1.5-flash') || value.includes('gemini-2.0-flash-exp') || value.includes('gemini-2.0-flash-thinking-exp')) {
$('#openai_max_context').attr('max', max_1mil);
} else if (value.includes('gemini-1.0-pro') || value === 'gemini-pro') {
$('#openai_max_context').attr('max', max_32k);
@@ -4486,7 +4512,7 @@ async function onModelChange() {
if (oai_settings.chat_completion_source === chat_completion_sources.DEEPSEEK) {
if (oai_settings.max_context_unlocked) {
$('#openai_max_context').attr('max', unlocked_max);
- } else if (oai_settings.deepseek_model == 'deepseek-chat') {
+ } else if (['deepseek-reasoner', 'deepseek-chat'].includes(oai_settings.deepseek_model)) {
$('#openai_max_context').attr('max', max_64k);
} else if (oai_settings.deepseek_model == 'deepseek-coder') {
$('#openai_max_context').attr('max', max_16k);
@@ -4723,7 +4749,7 @@ async function onConnectButtonClick(e) {
await writeSecret(SECRET_KEYS.DEEPSEEK, api_key_deepseek);
}
- if (!secret_state[SECRET_KEYS.DEEPSEEK]) {
+ if (!secret_state[SECRET_KEYS.DEEPSEEK] && !oai_settings.reverse_proxy) {
console.log('No secret key saved for DeepSeek');
return;
}
@@ -4899,6 +4925,8 @@ export function isImageInliningSupported() {
const visionSupportedModels = [
'gpt-4-vision',
'gemini-2.0-flash-thinking-exp-1219',
+ 'gemini-2.0-flash-thinking-exp-01-21',
+ 'gemini-2.0-flash-thinking-exp',
'gemini-2.0-flash-exp',
'gemini-1.5-flash',
'gemini-1.5-flash-latest',
@@ -5489,8 +5517,8 @@ export function initOpenAI() {
if (!isMobile()) {
$('#model_openrouter_select').select2({
- placeholder: 'Select a model',
- searchInputPlaceholder: 'Search models...',
+ placeholder: t`Select a model`,
+ searchInputPlaceholder: t`Search models...`,
searchInputCssClass: 'text_pole',
width: '100%',
templateResult: getOpenRouterModelTemplate,
diff --git a/public/scripts/power-user.js b/public/scripts/power-user.js
index 1e69d3608..bfe8eca2e 100644
--- a/public/scripts/power-user.js
+++ b/public/scripts/power-user.js
@@ -2534,7 +2534,7 @@ async function loadUntilMesId(mesId) {
let target;
while (getFirstDisplayedMessageId() > mesId && getFirstDisplayedMessageId() !== 0) {
- showMoreMessages();
+ await showMoreMessages();
await delay(1);
target = $('#chat').find(`.mes[mesid=${mesId}]`);
@@ -3962,6 +3962,95 @@ $(document).ready(() => {
`,
}));
+ SlashCommandParser.addCommandObject(SlashCommand.fromProps({
+ name: 'css-var',
+ /** @param {{to: string, varname: string }} args @param {string} value @returns {string} */
+ callback: (args, value) => {
+ // Map enum to target selector
+ const targetSelector = {
+ chat: '#chat',
+ background: '#bg1',
+ gallery: '#gallery',
+ zoomedAvatar: 'div.zoomed_avatar',
+ }[args.to || 'chat'];
+
+ if (!targetSelector) {
+ toastr.error(`Invalid target: ${args.to}`);
+ return;
+ }
+
+ if (!args.varname) {
+ toastr.error('CSS variable name is required');
+ return;
+ }
+ if (!args.varname.startsWith('--')) {
+ toastr.error('CSS variable names must start with "--"');
+ return;
+ }
+
+ const elements = document.querySelectorAll(targetSelector);
+ if (elements.length === 0) {
+ toastr.error(`No elements found for ${args.to ?? 'chat'} with selector "${targetSelector}"`);
+ return;
+ }
+
+ elements.forEach(element => {
+ element.style.setProperty(args.varname, value);
+ });
+
+ console.info(`Set CSS variable "${args.varname}" to "${value}" on "${targetSelector}"`);
+ },
+ namedArgumentList: [
+ SlashCommandNamedArgument.fromProps({
+ name: 'varname',
+ description: 'CSS variable name (starting with double dashes)',
+ typeList: [ARGUMENT_TYPE.STRING],
+ isRequired: true,
+ }),
+ SlashCommandNamedArgument.fromProps({
+ name: 'to',
+ description: 'The target element to which the CSS variable will be applied',
+ typeList: [ARGUMENT_TYPE.STRING],
+ enumList: [
+ new SlashCommandEnumValue('chat', null, enumTypes.enum, enumIcons.message),
+ new SlashCommandEnumValue('background', null, enumTypes.enum, enumIcons.image),
+ new SlashCommandEnumValue('zoomedAvatar', null, enumTypes.enum, enumIcons.character),
+ new SlashCommandEnumValue('gallery', null, enumTypes.enum, enumIcons.image),
+ ],
+ defaultValue: 'chat',
+ }),
+ ],
+ unnamedArgumentList: [
+ SlashCommandArgument.fromProps({
+ description: 'CSS variable value',
+ typeList: [ARGUMENT_TYPE.STRING],
+ isRequired: true,
+ }),
+ ],
+ helpString: `
+
+ Sets a CSS variable to a specified value on a target element.
+
+ Only setting of variable names is supported. They have to be prefixed with double dashes ("--exampleVar").
+ Setting actual CSS properties is not supported. Custom CSS in the theme settings can be used for that.
+
{{lastUserMessage}} – the text of the latest user chat message.
{{lastCharMessage}} – the text of the latest character chat message.
{{lastMessageId}} – index # of the latest chat message. Useful for slash command batching.
-
{{firstIncludedMessageId}} – the ID of the first message included in the context. Requires generation to be ran at least once in the current session.
+
{{firstIncludedMessageId}} – the ID of the first message included in the context. Requires generation to be run at least once in the current session. Will only be updated on generation.
+
{{firstDisplayedMessageId}} – the ID of the first message loaded into the visible chat.
{{currentSwipeId}} – the 1-based ID of the current swipe in the last chat message. Empty string if the last message is user or prompt-hidden.
{{lastSwipeId}} – the number of swipes in the last chat message. Empty string if the last message is user or prompt-hidden.
{{reverse:(content)}} – reverses the content of the macro.
diff --git a/public/scripts/textgen-models.js b/public/scripts/textgen-models.js
index 851c3940d..824d5eec4 100644
--- a/public/scripts/textgen-models.js
+++ b/public/scripts/textgen-models.js
@@ -5,6 +5,7 @@ import { textgenerationwebui_settings as textgen_settings, textgen_types } from
import { tokenizers } from './tokenizers.js';
import { renderTemplateAsync } from './templates.js';
import { POPUP_TYPE, callGenericPopup } from './popup.js';
+import { t } from './i18n.js';
let mancerModels = [];
let togetherModels = [];
@@ -318,8 +319,6 @@ export async function loadFeatherlessModels(data) {
return;
}
- // Sort the data by model id (default A-Z)
- data.sort((a, b) => a.id.localeCompare(b.id));
originalModels = data; // Store the original data for search
featherlessModels = data;
@@ -333,10 +332,8 @@ export async function loadFeatherlessModels(data) {
// Retrieve the stored number of items per page or default to 10
const perPage = Number(localStorage.getItem(storageKey)) || 10;
- // Initialize pagination with the full set of models
- const currentModelIndex = data.findIndex(x => x.id === textgen_settings.featherless_model);
- featherlessCurrentPage = currentModelIndex >= 0 ? (currentModelIndex / perPage) + 1 : 1;
- setupPagination(originalModels, perPage);
+ // Initialize pagination
+ applyFiltersAndSort();
// Function to set up pagination (also used for filtered results)
function setupPagination(models, perPage, pageNumber = featherlessCurrentPage) {
@@ -382,7 +379,7 @@ export async function loadFeatherlessModels(data) {
const dateAddedDiv = document.createElement('div');
dateAddedDiv.classList.add('model-date-added');
- dateAddedDiv.textContent = `Added On: ${new Date(model.updated_at).toLocaleDateString()}`;
+ dateAddedDiv.textContent = `Added On: ${new Date(model.created * 1000).toLocaleDateString()}`;
detailsContainer.appendChild(modelClassDiv);
detailsContainer.appendChild(contextLengthDiv);
@@ -471,6 +468,7 @@ export async function loadFeatherlessModels(data) {
featherlessTop = await fetchFeatherlessStats();
}
const featherlessIds = featherlessTop.map(stat => stat.id);
+
if (selectedCategory === 'New') {
featherlessNew = await fetchFeatherlessNew();
}
@@ -492,7 +490,7 @@ export async function loadFeatherlessModels(data) {
return matchesSearch && matchesClass && matchesNew;
}
else {
- return matchesSearch;
+ return matchesSearch && matchesClass;
}
});
@@ -501,11 +499,14 @@ export async function loadFeatherlessModels(data) {
} else if (selectedSortOrder === 'desc') {
filteredModels.sort((a, b) => b.id.localeCompare(a.id));
} else if (selectedSortOrder === 'date_asc') {
- filteredModels.sort((a, b) => a.updated_at.localeCompare(b.updated_at));
+ filteredModels.sort((a, b) => a.created - b.created);
} else if (selectedSortOrder === 'date_desc') {
- filteredModels.sort((a, b) => b.updated_at.localeCompare(a.updated_at));
+ filteredModels.sort((a, b) => b.created - a.created);
}
+ const currentModelIndex = filteredModels.findIndex(x => x.id === textgen_settings.featherless_model);
+ featherlessCurrentPage = currentModelIndex >= 0 ? (currentModelIndex / perPage) + 1 : 1;
+
setupPagination(filteredModels, Number(localStorage.getItem(storageKey)) || perPage, featherlessCurrentPage);
}
@@ -527,7 +528,7 @@ async function fetchFeatherlessStats() {
}
async function fetchFeatherlessNew() {
- const response = await fetch('https://api.featherless.ai/feather/models?sort=-created_at&perPage=10');
+ const response = await fetch('https://api.featherless.ai/feather/models?sort=-created_at&perPage=20');
const data = await response.json();
return data.items;
}
@@ -936,71 +937,71 @@ export function initTextGenModels() {
if (!isMobile()) {
$('#mancer_model').select2({
- placeholder: 'Select a model',
- searchInputPlaceholder: 'Search models...',
+ placeholder: t`Select a model`,
+ searchInputPlaceholder: t`Search models...`,
searchInputCssClass: 'text_pole',
width: '100%',
templateResult: getMancerModelTemplate,
});
$('#model_togetherai_select').select2({
- placeholder: 'Select a model',
- searchInputPlaceholder: 'Search models...',
+ placeholder: t`Select a model`,
+ searchInputPlaceholder: t`Search models...`,
searchInputCssClass: 'text_pole',
width: '100%',
templateResult: getTogetherModelTemplate,
});
$('#ollama_model').select2({
- placeholder: 'Select a model',
- searchInputPlaceholder: 'Search models...',
+ placeholder: t`Select a model`,
+ searchInputPlaceholder: t`Search models...`,
searchInputCssClass: 'text_pole',
width: '100%',
});
$('#tabby_model').select2({
- placeholder: '[Currently loaded]',
- searchInputPlaceholder: 'Search models...',
+ placeholder: t`[Currently loaded]`,
+ searchInputPlaceholder: t`Search models...`,
searchInputCssClass: 'text_pole',
width: '100%',
allowClear: true,
});
$('#model_infermaticai_select').select2({
- placeholder: 'Select a model',
- searchInputPlaceholder: 'Search models...',
+ placeholder: t`Select a model`,
+ searchInputPlaceholder: t`Search models...`,
searchInputCssClass: 'text_pole',
width: '100%',
templateResult: getInfermaticAIModelTemplate,
});
$('#model_dreamgen_select').select2({
- placeholder: 'Select a model',
- searchInputPlaceholder: 'Search models...',
+ placeholder: t`Select a model`,
+ searchInputPlaceholder: t`Search models...`,
searchInputCssClass: 'text_pole',
width: '100%',
templateResult: getDreamGenModelTemplate,
});
$('#openrouter_model').select2({
- placeholder: 'Select a model',
- searchInputPlaceholder: 'Search models...',
+ placeholder: t`Select a model`,
+ searchInputPlaceholder: t`Search models...`,
searchInputCssClass: 'text_pole',
width: '100%',
templateResult: getOpenRouterModelTemplate,
});
$('#vllm_model').select2({
- placeholder: 'Select a model',
- searchInputPlaceholder: 'Search models...',
+ placeholder: t`Select a model`,
+ searchInputPlaceholder: t`Search models...`,
searchInputCssClass: 'text_pole',
width: '100%',
templateResult: getVllmModelTemplate,
});
$('#aphrodite_model').select2({
- placeholder: 'Select a model',
- searchInputPlaceholder: 'Search models...',
+ placeholder: t`Select a model`,
+ searchInputPlaceholder: t`Search models...`,
searchInputCssClass: 'text_pole',
width: '100%',
templateResult: getAphroditeModelTemplate,
});
providersSelect.select2({
sorter: data => data.sort((a, b) => a.text.localeCompare(b.text)),
- placeholder: 'Select providers. No selection = all providers.',
- searchInputPlaceholder: 'Search providers...',
+ placeholder: t`Select providers. No selection = all providers.`,
+ searchInputPlaceholder: t`Search providers...`,
searchInputCssClass: 'text_pole',
width: '100%',
closeOnSelect: false,
diff --git a/public/scripts/textgen-settings.js b/public/scripts/textgen-settings.js
index e80c7e910..cd1004991 100644
--- a/public/scripts/textgen-settings.js
+++ b/public/scripts/textgen-settings.js
@@ -1231,7 +1231,7 @@ export function getTextGenGenerationData(finalPrompt, maxTokens, isImpersonate,
'top_p': settings.top_p,
'typical_p': settings.typical_p,
'typical': settings.typical_p,
- 'sampler_seed': settings.seed,
+ 'sampler_seed': settings.seed >= 0 ? settings.seed : undefined,
'min_p': settings.min_p,
'repetition_penalty': settings.rep_pen,
'frequency_penalty': settings.freq_pen,
@@ -1294,7 +1294,7 @@ export function getTextGenGenerationData(finalPrompt, maxTokens, isImpersonate,
'temperature_last': (settings.type === OOBA || settings.type === APHRODITE || settings.type == TABBY) ? settings.temperature_last : undefined,
'speculative_ngram': settings.type === TABBY ? settings.speculative_ngram : undefined,
'do_sample': settings.type === OOBA ? settings.do_sample : undefined,
- 'seed': settings.seed,
+ 'seed': settings.seed >= 0 ? settings.seed : undefined,
'guidance_scale': cfgValues?.guidanceScale?.value ?? settings.guidance_scale ?? 1,
'negative_prompt': cfgValues?.negativePrompt ?? substituteParams(settings.negative_prompt) ?? '',
'grammar_string': settings.grammar_string,
diff --git a/public/scripts/utils.js b/public/scripts/utils.js
index eb944b586..a479aee52 100644
--- a/public/scripts/utils.js
+++ b/public/scripts/utils.js
@@ -67,6 +67,16 @@ export function escapeHtml(str) {
return String(str).replace(/&/g, '&').replace(//g, '>').replace(/"/g, '"');
}
+/**
+ * Make string safe for use as a CSS selector.
+ * @param {string} str String to sanitize
+ * @param {string} replacement Replacement for invalid characters
+ * @returns {string} Sanitized string
+ */
+export function sanitizeSelector(str, replacement = '_') {
+ return String(str).replace(/[^a-z0-9_-]/ig, replacement);
+}
+
export function isValidUrl(value) {
try {
new URL(value);
@@ -381,6 +391,26 @@ export function getStringHash(str, seed = 0) {
return 4294967296 * (2097151 & h2) + (h1 >>> 0);
}
+/**
+ * Copy text to clipboard. Use navigator.clipboard.writeText if available, otherwise use document.execCommand.
+ * @param {string} text - The text to copy to the clipboard.
+ * @returns {Promise} A promise that resolves when the text has been copied to the clipboard.
+ */
+export function copyText(text) {
+ if (navigator.clipboard) {
+ return navigator.clipboard.writeText(text);
+ }
+
+ const parent = document.querySelector('dialog[open]:last-of-type') ?? document.body;
+ const textArea = document.createElement('textarea');
+ textArea.value = text;
+ parent.appendChild(textArea);
+ textArea.focus();
+ textArea.select();
+ document.execCommand('copy');
+ parent.removeChild(textArea);
+}
+
/**
* Map of debounced functions to their timers.
* Weak map is used to avoid memory leaks.
diff --git a/public/scripts/world-info.js b/public/scripts/world-info.js
index 9f18f46c5..ee4db166a 100644
--- a/public/scripts/world-info.js
+++ b/public/scripts/world-info.js
@@ -20,6 +20,7 @@ import { SlashCommandClosure } from './slash-commands/SlashCommandClosure.js';
import { callGenericPopup, Popup, POPUP_TYPE } from './popup.js';
import { StructuredCloneMap } from './util/StructuredCloneMap.js';
import { renderTemplateAsync } from './templates.js';
+import { t } from './i18n.js';
export const world_info_insertion_strategy = {
evenly: 0,
@@ -909,21 +910,21 @@ function registerWorldInfoSlashCommands() {
async function getEntriesFromFile(file) {
if (!file || !world_names.includes(file)) {
- toastr.warning('Valid World Info file name is required');
+ toastr.warning(t`Valid World Info file name is required`);
return '';
}
const data = await loadWorldInfo(file);
if (!data || !('entries' in data)) {
- toastr.warning('World Info file has an invalid format');
+ toastr.warning(t`World Info file has an invalid format`);
return '';
}
const entries = Object.values(data.entries);
if (!entries || entries.length === 0) {
- toastr.warning('World Info file has no entries');
+ toastr.warning(t`World Info file has no entries`);
return '';
}
@@ -951,7 +952,7 @@ function registerWorldInfoSlashCommands() {
name = String(name ?? '') || context.characters[context.characterId]?.avatar || null;
const character = findChar({ name });
if (!character) {
- toastr.error('Character not found.');
+ toastr.error(t`Character not found.`);
return '';
}
const books = [];
@@ -977,7 +978,7 @@ function registerWorldInfoSlashCommands() {
const chatId = getCurrentChatId();
if (!chatId) {
- toastr.warning('Open a chat to get a name of the chat-bound lorebook');
+ toastr.warning(t`Open a chat to get a name of the chat-bound lorebook`);
return '';
}
@@ -4773,7 +4774,7 @@ export async function importEmbeddedWorldInfo(skipPopup = false) {
const bookName = characters[chid]?.data?.character_book?.name || `${characters[chid]?.name}'s Lorebook`;
if (!skipPopup) {
- const confirmation = await Popup.show.confirm(`Are you sure you want to import "${bookName}"?`, world_names.includes(bookName) ? 'It will overwrite the World/Lorebook with the same name.' : '');
+ const confirmation = await Popup.show.confirm(t`Are you sure you want to import '${bookName}'?`, world_names.includes(bookName) ? t`It will overwrite the World/Lorebook with the same name.` : '');
if (!confirmation) {
return;
}
@@ -4785,7 +4786,7 @@ export async function importEmbeddedWorldInfo(skipPopup = false) {
await updateWorldInfoList();
$('#character_world').val(bookName).trigger('change');
- toastr.success(`The world "${bookName}" has been imported and linked to the character successfully.`, 'World/Lorebook imported');
+ toastr.success(t`The world '${bookName}' has been imported and linked to the character successfully.`, t`World/Lorebook imported`);
const newIndex = world_names.indexOf(bookName);
if (newIndex >= 0) {
@@ -4813,9 +4814,9 @@ export function onWorldInfoChange(args, text) {
if (selected_world_info.includes(name)) {
selected_world_info.splice(selected_world_info.indexOf(name), 1);
wiElement.prop('selected', false);
- if (!silent) toastr.success(`Deactivated world: ${name}`);
+ if (!silent) toastr.success(t`Deactivated world: ${name}`);
} else {
- if (!silent) toastr.error(`World was not active: ${name}`);
+ if (!silent) toastr.error(t`World was not active: ${name}`);
}
break;
}
@@ -4823,11 +4824,11 @@ export function onWorldInfoChange(args, text) {
if (selected_world_info.includes(name)) {
selected_world_info.splice(selected_world_info.indexOf(name), 1);
wiElement.prop('selected', false);
- if (!silent) toastr.success(`Deactivated world: ${name}`);
+ if (!silent) toastr.success(t`Deactivated world: ${name}`);
} else {
selected_world_info.push(name);
wiElement.prop('selected', true);
- if (!silent) toastr.success(`Activated world: ${name}`);
+ if (!silent) toastr.success(t`Activated world: ${name}`);
}
break;
}
@@ -4835,16 +4836,16 @@ export function onWorldInfoChange(args, text) {
default: {
selected_world_info.push(name);
wiElement.prop('selected', true);
- if (!silent) toastr.success(`Activated world: ${name}`);
+ if (!silent) toastr.success(t`Activated world: ${name}`);
}
}
} else {
- if (!silent) toastr.error(`No world found named: ${worldName}`);
+ if (!silent) toastr.error(t`No world found named: ${worldName}`);
}
});
$('#world_info').trigger('change');
} else { // if no args, unset all worlds
- if (!silent) toastr.success('Deactivated all worlds');
+ if (!silent) toastr.success(t`Deactivated all worlds`);
selected_world_info = [];
$('#world_info').val(null).trigger('change');
}
@@ -4860,7 +4861,7 @@ export function onWorldInfoChange(args, text) {
} else {
const wiElement = getWIElement(existingWorldName);
wiElement.prop('selected', false);
- toastr.error(`The world with ${existingWorldName} is invalid or corrupted.`);
+ toastr.error(t`The world with ${existingWorldName} is invalid or corrupted.`);
}
});
}
@@ -4892,7 +4893,7 @@ export async function importWorldInfo(file) {
}
if (jsonData === undefined || jsonData === null) {
- toastr.error(`File is not valid: ${file.name}`);
+ toastr.error(t`File is not valid: ${file.name}`);
return;
}
@@ -5038,7 +5039,7 @@ jQuery(() => {
$('#world_create_button').on('click', async () => {
const tempName = getFreeWorldName();
- const finalName = await Popup.show.input('Create a new World Info', 'Enter a name for the new file:', tempName);
+ const finalName = await Popup.show.input(t`Create a new World Info`, t`Enter a name for the new file:`, tempName);
if (finalName) {
await createNewWorldInfo(finalName, { interactive: true });
diff --git a/public/style.css b/public/style.css
index ce076f915..95dcd314c 100644
--- a/public/style.css
+++ b/public/style.css
@@ -1318,6 +1318,10 @@ button {
transition: clip-path 200ms;
}
+#send_textarea:placeholder-shown {
+ overflow: hidden;
+}
+
#send_textarea:focus-visible {
/* Disable outline for the chat bar itself, we add it to the outer div */
outline: none;
@@ -2917,7 +2921,7 @@ input[type=search]:focus::-webkit-search-cancel-button {
position: relative;
}
-#rm_print_characters_block .ch_name,
+.character_name_block .ch_name,
.avatar-container .ch_name {
flex: 1 1 auto;
white-space: nowrap;
@@ -2927,6 +2931,13 @@ input[type=search]:focus::-webkit-search-cancel-button {
display: block;
}
+.character_name_block .character_version {
+ text-overflow: ellipsis;
+ overflow: hidden;
+ text-wrap: nowrap;
+ max-width: 50%;
+}
+
#rm_print_characters_block .character_name_block> :last-child {
flex: 0 100000 auto;
/* Force shrinking first */
@@ -3259,6 +3270,18 @@ grammarly-extension {
align-items: center;
}
+#creators_notes_div {
+ align-items: baseline;
+}
+
+#creators_note_desc_hidden {
+ display: none;
+}
+
+#creators_notes_div:has(#spoiler_free_desc_button.fa-eye-slash) #creators_note_desc_hidden {
+ display: initial;
+}
+
.alternate_greeting details {
padding: 2px;
}
@@ -4799,7 +4822,7 @@ body:not(.sd) .mes_img_swipes {
.img_enlarged {
object-fit: contain;
- width: 100%;
+ max-width: 100%;
height: 100%;
cursor: zoom-in
}
@@ -5627,6 +5650,7 @@ body:not(.movingUI) .drawer-content.maximized {
.model-card .details-container {
text-align: right;
+ line-height: 0.9;
}
.model-card:hover {
@@ -5649,7 +5673,7 @@ body:not(.movingUI) .drawer-content.maximized {
}
.model-title {
- font-size: 13px;
+ font-size: calc(var(--mainFontSize) * 0.95);
font-weight: bold;
overflow: hidden;
}
@@ -5665,7 +5689,7 @@ body:not(.movingUI) .drawer-content.maximized {
.model-class,
.model-context-length,
.model-date-added {
- font-size: 10px;
+ font-size: calc(var(--mainFontSize) * 0.75);
}
.model-class,
@@ -5735,3 +5759,29 @@ body:not(.movingUI) .drawer-content.maximized {
}
}
+
+/* Constrict the scroll of alternate greetings to only the dynamic form section */
+.alternate_grettings {
+ display: grid;
+ grid-template-rows: auto auto auto 1fr;
+ max-height: 100%;
+ overflow: hidden;
+}
+
+.alternate_greetings_list {
+ overflow-y: scroll;
+}
+
+.mes_text div[data-type="assistant_note"]:has(.assistant_note_export) {
+ display: flex;
+ flex-direction: row;
+ flex-wrap: nowrap;
+ justify-content: space-between;
+ align-items: center;
+ gap: 10px;
+ padding: 0 2px;
+}
+
+.mes_text div[data-type="assistant_note"]:has(.assistant_note_export)>div:not(.assistant_note_export) {
+ flex: 1;
+}
diff --git a/server.js b/server.js
index db7ba9306..6cd78d753 100644
--- a/server.js
+++ b/server.js
@@ -18,10 +18,9 @@ import { hideBin } from 'yargs/helpers';
// express/server related library imports
import cors from 'cors';
-import { doubleCsrf } from 'csrf-csrf';
+import { csrfSync } from 'csrf-sync';
import express from 'express';
import compression from 'compression';
-import cookieParser from 'cookie-parser';
import cookieSession from 'cookie-session';
import multer from 'multer';
import responseTime from 'response-time';
@@ -40,7 +39,6 @@ util.inspect.defaultOptions.depth = 4;
import { loadPlugins } from './src/plugin-loader.js';
import {
initUserStorage,
- getCsrfSecret,
getCookieSecret,
getCookieSessionName,
getAllEnabledUsers,
@@ -67,6 +65,7 @@ import {
forwardFetchResponse,
removeColorFormatting,
getSeparator,
+ safeReadFileSync,
} from './src/util.js';
import { UPLOADS_DIRECTORY } from './src/constants.js';
import { ensureThumbnailCache } from './src/endpoints/thumbnails.js';
@@ -347,8 +346,8 @@ if (enableCorsProxy) {
}
function getSessionCookieAge() {
- // Defaults to 24 hours in seconds if not set
- const configValue = getConfigValue('sessionTimeout', 24 * 60 * 60);
+ // Defaults to "no expiration" if not set
+ const configValue = getConfigValue('sessionTimeout', -1);
// Convert to milliseconds
if (configValue > 0) {
@@ -377,27 +376,38 @@ app.use(setUserDataMiddleware);
// CSRF Protection //
if (!disableCsrf) {
- const COOKIES_SECRET = getCookieSecret();
-
- const { generateToken, doubleCsrfProtection } = doubleCsrf({
- getSecret: getCsrfSecret,
- cookieName: 'X-CSRF-Token',
- cookieOptions: {
- sameSite: 'strict',
- secure: false,
+ const csrfSyncProtection = csrfSync({
+ getTokenFromState: (req) => {
+ if (!req.session) {
+ console.error('(CSRF error) getTokenFromState: Session object not initialized');
+ return;
+ }
+ return req.session.csrfToken;
},
- size: 64,
- getTokenFromRequest: (req) => req.headers['x-csrf-token'],
+ getTokenFromRequest: (req) => {
+ return req.headers['x-csrf-token']?.toString();
+ },
+ storeTokenInState: (req, token) => {
+ if (!req.session) {
+ console.error('(CSRF error) storeTokenInState: Session object not initialized');
+ return;
+ }
+ req.session.csrfToken = token;
+ },
+ size: 32,
});
app.get('/csrf-token', (req, res) => {
res.json({
- 'token': generateToken(res, req),
+ 'token': csrfSyncProtection.generateToken(req),
});
});
- app.use(cookieParser(COOKIES_SECRET));
- app.use(doubleCsrfProtection);
+ // Customize the error message
+ csrfSyncProtection.invalidCsrfTokenError.message = color.red('Invalid CSRF token. Please refresh the page and try again.');
+ csrfSyncProtection.invalidCsrfTokenError.stack = undefined;
+
+ app.use(csrfSyncProtection.csrfSynchronisedProtection);
} else {
console.warn('\nCSRF protection is disabled. This will make your server vulnerable to CSRF attacks.\n');
app.get('/csrf-token', (req, res) => {
@@ -921,6 +931,16 @@ async function verifySecuritySettings() {
}
}
+/**
+ * Registers a not-found error response if a not-found error page exists. Should only be called after all other middlewares have been registered.
+ */
+function apply404Middleware() {
+ const notFoundWebpage = safeReadFileSync('./public/error/url-not-found.html') ?? '';
+ app.use((req, res) => {
+ res.status(404).send(notFoundWebpage);
+ });
+}
+
// User storage module needs to be initialized before starting the server
initUserStorage(dataRoot)
.then(ensurePublicDirectoriesExist)
@@ -928,4 +948,5 @@ initUserStorage(dataRoot)
.then(migrateSystemPrompts)
.then(verifySecuritySettings)
.then(preSetupTasks)
+ .then(apply404Middleware)
.finally(startServer);
diff --git a/src/endpoints/avatars.js b/src/endpoints/avatars.js
index 73b995ffb..f84527670 100644
--- a/src/endpoints/avatars.js
+++ b/src/endpoints/avatars.js
@@ -9,6 +9,7 @@ import { sync as writeFileAtomicSync } from 'write-file-atomic';
import { jsonParser, urlencodedParser } from '../express-common.js';
import { AVATAR_WIDTH, AVATAR_HEIGHT } from '../constants.js';
import { getImages, tryParse } from '../util.js';
+import { getFileNameValidationFunction } from '../middleware/validateFileName.js';
export const router = express.Router();
@@ -17,7 +18,7 @@ router.post('/get', jsonParser, function (request, response) {
response.send(JSON.stringify(images));
});
-router.post('/delete', jsonParser, function (request, response) {
+router.post('/delete', jsonParser, getFileNameValidationFunction('avatar'), function (request, response) {
if (!request.body) return response.sendStatus(400);
if (request.body.avatar !== sanitize(request.body.avatar)) {
diff --git a/src/endpoints/backends/chat-completions.js b/src/endpoints/backends/chat-completions.js
index 252e16d95..884a95c65 100644
--- a/src/endpoints/backends/chat-completions.js
+++ b/src/endpoints/backends/chat-completions.js
@@ -37,6 +37,8 @@ import {
getTiktokenTokenizer,
sentencepieceTokenizers,
TEXT_COMPLETION_MODELS,
+ webTokenizers,
+ getWebTokenizer,
} from '../tokenizers.js';
const API_OPENAI = 'https://api.openai.com/v1';
@@ -61,6 +63,7 @@ const API_DEEPSEEK = 'https://api.deepseek.com/beta';
* @returns
*/
function postProcessPrompt(messages, type, names) {
+ const addAssistantPrefix = x => x.length && (x[x.length - 1].role !== 'assistant' || (x[x.length - 1].prefix = true)) ? x : x;
switch (type) {
case 'merge':
case 'claude':
@@ -70,7 +73,9 @@ function postProcessPrompt(messages, type, names) {
case 'strict':
return mergeMessages(messages, names, true, true);
case 'deepseek':
- return (x => x.length && (x[x.length - 1].role !== 'assistant' || (x[x.length - 1].prefix = true)) ? x : x)(mergeMessages(messages, names, true, false));
+ return addAssistantPrefix(mergeMessages(messages, names, true, false));
+ case 'deepseek-reasoner':
+ return addAssistantPrefix(mergeMessages(messages, names, true, true));
default:
return messages;
}
@@ -636,6 +641,89 @@ async function sendCohereRequest(request, response) {
}
}
+/**
+ * Sends a request to DeepSeek API.
+ * @param {express.Request} request Express request
+ * @param {express.Response} response Express response
+ */
+async function sendDeepSeekRequest(request, response) {
+ const apiUrl = new URL(request.body.reverse_proxy || API_DEEPSEEK).toString();
+ const apiKey = request.body.reverse_proxy ? request.body.proxy_password : readSecret(request.user.directories, SECRET_KEYS.DEEPSEEK);
+
+ if (!apiKey && !request.body.reverse_proxy) {
+ console.log('DeepSeek API key is missing.');
+ return response.status(400).send({ error: true });
+ }
+
+ const controller = new AbortController();
+ request.socket.removeAllListeners('close');
+ request.socket.on('close', function () {
+ controller.abort();
+ });
+
+ try {
+ let bodyParams = {};
+
+ if (request.body.logprobs > 0) {
+ bodyParams['top_logprobs'] = request.body.logprobs;
+ bodyParams['logprobs'] = true;
+ }
+
+ const postProcessType = String(request.body.model).endsWith('-reasoner') ? 'deepseek-reasoner' : 'deepseek';
+ const processedMessages = postProcessPrompt(request.body.messages, postProcessType, getPromptNames(request));
+
+ const requestBody = {
+ 'messages': processedMessages,
+ 'model': request.body.model,
+ 'temperature': request.body.temperature,
+ 'max_tokens': request.body.max_tokens,
+ 'stream': request.body.stream,
+ 'presence_penalty': request.body.presence_penalty,
+ 'frequency_penalty': request.body.frequency_penalty,
+ 'top_p': request.body.top_p,
+ 'stop': request.body.stop,
+ 'seed': request.body.seed,
+ ...bodyParams,
+ };
+
+ const config = {
+ method: 'POST',
+ headers: {
+ 'Content-Type': 'application/json',
+ 'Authorization': 'Bearer ' + apiKey,
+ },
+ body: JSON.stringify(requestBody),
+ signal: controller.signal,
+ };
+
+ console.log('DeepSeek request:', requestBody);
+
+ const generateResponse = await fetch(apiUrl + '/chat/completions', config);
+
+ if (request.body.stream) {
+ forwardFetchResponse(generateResponse, response);
+ } else {
+ if (!generateResponse.ok) {
+ const errorText = await generateResponse.text();
+ console.log(`DeepSeek API returned error: ${generateResponse.status} ${generateResponse.statusText} ${errorText}`);
+ const errorJson = tryParse(errorText) ?? { error: true };
+ return response.status(500).send(errorJson);
+ }
+ const generateResponseJson = await generateResponse.json();
+ console.log('DeepSeek response:', generateResponseJson);
+ return response.send(generateResponseJson);
+ }
+ } catch (error) {
+ console.log('Error communicating with DeepSeek API: ', error);
+ if (!response.headersSent) {
+ response.send({ error: true });
+ } else {
+ response.end();
+ }
+ }
+}
+
+
export const router = express.Router();
router.post('/status', jsonParser, async function (request, response_getstatus_openai) {
@@ -680,8 +768,8 @@ router.post('/status', jsonParser, async function (request, response_getstatus_o
api_key_openai = readSecret(request.user.directories, SECRET_KEYS.NANOGPT);
headers = {};
} else if (request.body.chat_completion_source === CHAT_COMPLETION_SOURCES.DEEPSEEK) {
- api_url = API_DEEPSEEK.replace('/beta', '');
- api_key_openai = readSecret(request.user.directories, SECRET_KEYS.DEEPSEEK);
+ api_url = new URL(request.body.reverse_proxy || API_DEEPSEEK.replace('/beta', ''));
+ api_key_openai = request.body.reverse_proxy ? request.body.proxy_password : readSecret(request.user.directories, SECRET_KEYS.DEEPSEEK);
headers = {};
} else {
console.log('This chat completion source is not supported yet.');
@@ -777,6 +865,14 @@ router.post('/bias', jsonParser, async function (request, response) {
return response.send({});
}
encodeFunction = (text) => new Uint32Array(instance.encodeIds(text));
+ } else if (webTokenizers.includes(model)) {
+ const tokenizer = getWebTokenizer(model);
+ const instance = await tokenizer?.get();
+ if (!instance) {
+ console.warn('Tokenizer not initialized:', model);
+ return response.send({});
+ }
+ encodeFunction = (text) => new Uint32Array(instance.encode(text));
} else {
const tokenizer = getTiktokenTokenizer(model);
encodeFunction = (tokenizer.encode.bind(tokenizer));
@@ -841,6 +937,7 @@ router.post('/generate', jsonParser, function (request, response) {
case CHAT_COMPLETION_SOURCES.MAKERSUITE: return sendMakerSuiteRequest(request, response);
case CHAT_COMPLETION_SOURCES.MISTRALAI: return sendMistralAIRequest(request, response);
case CHAT_COMPLETION_SOURCES.COHERE: return sendCohereRequest(request, response);
+ case CHAT_COMPLETION_SOURCES.DEEPSEEK: return sendDeepSeekRequest(request, response);
}
let apiUrl;
@@ -954,18 +1051,6 @@ router.post('/generate', jsonParser, function (request, response) {
apiKey = readSecret(request.user.directories, SECRET_KEYS.BLOCKENTROPY);
headers = {};
bodyParams = {};
- } else if (request.body.chat_completion_source === CHAT_COMPLETION_SOURCES.DEEPSEEK) {
- apiUrl = API_DEEPSEEK;
- apiKey = readSecret(request.user.directories, SECRET_KEYS.DEEPSEEK);
- headers = {};
- bodyParams = {};
-
- if (request.body.logprobs > 0) {
- bodyParams['top_logprobs'] = request.body.logprobs;
- bodyParams['logprobs'] = true;
- }
-
- request.body.messages = postProcessPrompt(request.body.messages, 'deepseek', getPromptNames(request));
} else {
console.log('This chat completion source is not supported yet.');
return response.status(400).send({ error: true });
@@ -1103,4 +1188,3 @@ router.post('/generate', jsonParser, function (request, response) {
}
}
});
-
diff --git a/src/endpoints/backgrounds.js b/src/endpoints/backgrounds.js
index 0638415e6..13705fa7a 100644
--- a/src/endpoints/backgrounds.js
+++ b/src/endpoints/backgrounds.js
@@ -7,6 +7,7 @@ import sanitize from 'sanitize-filename';
import { jsonParser, urlencodedParser } from '../express-common.js';
import { invalidateThumbnail } from './thumbnails.js';
import { getImages } from '../util.js';
+import { getFileNameValidationFunction } from '../middleware/validateFileName.js';
export const router = express.Router();
@@ -15,7 +16,7 @@ router.post('/all', jsonParser, function (request, response) {
response.send(JSON.stringify(images));
});
-router.post('/delete', jsonParser, function (request, response) {
+router.post('/delete', jsonParser, getFileNameValidationFunction('bg'), function (request, response) {
if (!request.body) return response.sendStatus(400);
if (request.body.bg !== sanitize(request.body.bg)) {
diff --git a/src/endpoints/characters.js b/src/endpoints/characters.js
index 2aa2cca10..1e5df80bd 100644
--- a/src/endpoints/characters.js
+++ b/src/endpoints/characters.js
@@ -14,6 +14,7 @@ import jimp from 'jimp';
import { AVATAR_WIDTH, AVATAR_HEIGHT } from '../constants.js';
import { jsonParser, urlencodedParser } from '../express-common.js';
+import { default as validateAvatarUrlMiddleware, getFileNameValidationFunction } from '../middleware/validateFileName.js';
import { deepMerge, humanizedISO8601DateTime, tryParse, extractFileFromZipBuffer, MemoryLimitedMap, getConfigValue } from '../util.js';
import { TavernCardValidator } from '../validator/TavernCardValidator.js';
import { parse, write } from '../character-card-parser.js';
@@ -73,12 +74,18 @@ async function writeCharacterData(inputFile, data, outputFile, request, crop = u
* Read the image, resize, and save it as a PNG into the buffer.
* @returns {Promise} Image buffer
*/
- function getInputImage() {
- if (Buffer.isBuffer(inputFile)) {
- return parseImageBuffer(inputFile, crop);
- }
+ async function getInputImage() {
+ try {
+ if (Buffer.isBuffer(inputFile)) {
+ return await parseImageBuffer(inputFile, crop);
+ }
- return tryReadImage(inputFile, crop);
+ return await tryReadImage(inputFile, crop);
+ } catch (error) {
+ const message = Buffer.isBuffer(inputFile) ? 'Failed to read image buffer.' : `Failed to read image: ${inputFile}.`;
+ console.warn(message, 'Using a fallback image.', error);
+ return await fs.promises.readFile(defaultAvatarPath);
+ }
}
const inputImage = await getInputImage();
@@ -756,7 +763,7 @@ router.post('/create', urlencodedParser, async function (request, response) {
}
});
-router.post('/rename', jsonParser, async function (request, response) {
+router.post('/rename', jsonParser, validateAvatarUrlMiddleware, async function (request, response) {
if (!request.body.avatar_url || !request.body.new_name) {
return response.sendStatus(400);
}
@@ -803,7 +810,7 @@ router.post('/rename', jsonParser, async function (request, response) {
}
});
-router.post('/edit', urlencodedParser, async function (request, response) {
+router.post('/edit', urlencodedParser, validateAvatarUrlMiddleware, async function (request, response) {
if (!request.body) {
console.error('Error: no response body detected');
response.status(400).send('Error: no response body detected');
@@ -852,7 +859,7 @@ router.post('/edit', urlencodedParser, async function (request, response) {
* @param {Object} response - The HTTP response object.
* @returns {void}
*/
-router.post('/edit-attribute', jsonParser, async function (request, response) {
+router.post('/edit-attribute', jsonParser, validateAvatarUrlMiddleware, async function (request, response) {
console.log(request.body);
if (!request.body) {
console.error('Error: no response body detected');
@@ -898,7 +905,7 @@ router.post('/edit-attribute', jsonParser, async function (request, response) {
*
* @returns {void}
* */
-router.post('/merge-attributes', jsonParser, async function (request, response) {
+router.post('/merge-attributes', jsonParser, getFileNameValidationFunction('avatar'), async function (request, response) {
try {
const update = request.body;
const avatarPath = path.join(request.user.directories.characters, update.avatar);
@@ -929,7 +936,7 @@ router.post('/merge-attributes', jsonParser, async function (request, response)
}
});
-router.post('/delete', jsonParser, async function (request, response) {
+router.post('/delete', jsonParser, validateAvatarUrlMiddleware, async function (request, response) {
if (!request.body || !request.body.avatar_url) {
return response.sendStatus(400);
}
@@ -992,7 +999,7 @@ router.post('/all', jsonParser, async function (request, response) {
}
});
-router.post('/get', jsonParser, async function (request, response) {
+router.post('/get', jsonParser, validateAvatarUrlMiddleware, async function (request, response) {
try {
if (!request.body) return response.sendStatus(400);
const item = request.body.avatar_url;
@@ -1011,7 +1018,7 @@ router.post('/get', jsonParser, async function (request, response) {
}
});
-router.post('/chats', jsonParser, async function (request, response) {
+router.post('/chats', jsonParser, validateAvatarUrlMiddleware, async function (request, response) {
if (!request.body) return response.sendStatus(400);
const characterDirectory = (request.body.avatar_url).replace('.png', '');
@@ -1160,7 +1167,7 @@ router.post('/import', urlencodedParser, async function (request, response) {
}
});
-router.post('/duplicate', jsonParser, async function (request, response) {
+router.post('/duplicate', jsonParser, validateAvatarUrlMiddleware, async function (request, response) {
try {
if (!request.body.avatar_url) {
console.log('avatar URL not found in request body');
@@ -1207,7 +1214,7 @@ router.post('/duplicate', jsonParser, async function (request, response) {
}
});
-router.post('/export', jsonParser, async function (request, response) {
+router.post('/export', jsonParser, validateAvatarUrlMiddleware, async function (request, response) {
try {
if (!request.body.format || !request.body.avatar_url) {
return response.sendStatus(400);
diff --git a/src/endpoints/chats.js b/src/endpoints/chats.js
index 62c790400..c3bab87ac 100644
--- a/src/endpoints/chats.js
+++ b/src/endpoints/chats.js
@@ -9,6 +9,7 @@ import { sync as writeFileAtomicSync } from 'write-file-atomic';
import _ from 'lodash';
import { jsonParser, urlencodedParser } from '../express-common.js';
+import validateAvatarUrlMiddleware from '../middleware/validateFileName.js';
import {
getConfigValue,
humanizedISO8601DateTime,
@@ -264,9 +265,37 @@ function flattenChubChat(userName, characterName, lines) {
return (lines ?? []).map(convert).join('\n');
}
+/**
+ * Imports a chat from RisuAI format.
+ * @param {string} userName User name
+ * @param {string} characterName Character name
+ * @param {object} jsonData Imported chat data
+ * @returns {string} Chat data
+ */
+function importRisuChat(userName, characterName, jsonData) {
+ /** @type {object[]} */
+ const chat = [{
+ user_name: userName,
+ character_name: characterName,
+ create_date: humanizedISO8601DateTime(),
+ }];
+
+ for (const message of jsonData.data.message) {
+ const isUser = message.role === 'user';
+ chat.push({
+ name: message.name ?? (isUser ? userName : characterName),
+ is_user: isUser,
+ send_date: Number(message.time ?? Date.now()),
+ mes: message.data ?? '',
+ });
+ }
+
+ return chat.map(obj => JSON.stringify(obj)).join('\n');
+}
+
export const router = express.Router();
-router.post('/save', jsonParser, function (request, response) {
+router.post('/save', jsonParser, validateAvatarUrlMiddleware, function (request, response) {
try {
const directoryName = String(request.body.avatar_url).replace('.png', '');
const chatData = request.body.chat;
@@ -282,7 +311,7 @@ router.post('/save', jsonParser, function (request, response) {
}
});
-router.post('/get', jsonParser, function (request, response) {
+router.post('/get', jsonParser, validateAvatarUrlMiddleware, function (request, response) {
try {
const dirName = String(request.body.avatar_url).replace('.png', '');
const directoryPath = path.join(request.user.directories.chats, dirName);
@@ -319,7 +348,7 @@ router.post('/get', jsonParser, function (request, response) {
});
-router.post('/rename', jsonParser, async function (request, response) {
+router.post('/rename', jsonParser, validateAvatarUrlMiddleware, async function (request, response) {
if (!request.body || !request.body.original_file || !request.body.renamed_file) {
return response.sendStatus(400);
}
@@ -344,7 +373,7 @@ router.post('/rename', jsonParser, async function (request, response) {
return response.send({ ok: true, sanitizedFileName });
});
-router.post('/delete', jsonParser, function (request, response) {
+router.post('/delete', jsonParser, validateAvatarUrlMiddleware, function (request, response) {
const dirName = String(request.body.avatar_url).replace('.png', '');
const fileName = String(request.body.chatfile);
const filePath = path.join(request.user.directories.chats, dirName, sanitize(fileName));
@@ -360,7 +389,7 @@ router.post('/delete', jsonParser, function (request, response) {
return response.send('ok');
});
-router.post('/export', jsonParser, async function (request, response) {
+router.post('/export', jsonParser, validateAvatarUrlMiddleware, async function (request, response) {
if (!request.body.file || (!request.body.avatar_url && request.body.is_group === false)) {
return response.sendStatus(400);
}
@@ -450,7 +479,7 @@ router.post('/group/import', urlencodedParser, function (request, response) {
}
});
-router.post('/import', urlencodedParser, function (request, response) {
+router.post('/import', urlencodedParser, validateAvatarUrlMiddleware, function (request, response) {
if (!request.body) return response.sendStatus(400);
const format = request.body.file_type;
@@ -481,6 +510,8 @@ router.post('/import', urlencodedParser, function (request, response) {
importFunc = importOobaChat;
} else if (Array.isArray(jsonData.messages)) { // Agnai's format
importFunc = importAgnaiChat;
+ } else if (jsonData.type === 'risuChat') { // RisuAI format
+ importFunc = importRisuChat;
} else { // Unknown format
console.log('Incorrect chat format .json');
return response.send({ error: true });
@@ -596,7 +627,7 @@ router.post('/group/save', jsonParser, (request, response) => {
return response.send({ ok: true });
});
-router.post('/search', jsonParser, function (request, response) {
+router.post('/search', jsonParser, validateAvatarUrlMiddleware, function (request, response) {
try {
const { query, avatar_url, group_id } = request.body;
let chatFiles = [];
diff --git a/src/endpoints/search.js b/src/endpoints/search.js
index e29e53008..c999f8264 100644
--- a/src/endpoints/search.js
+++ b/src/endpoints/search.js
@@ -4,6 +4,8 @@ import express from 'express';
import { decode } from 'html-entities';
import { readSecret, SECRET_KEYS } from './secrets.js';
import { jsonParser } from '../express-common.js';
+import { trimV1 } from '../util.js';
+import { setAdditionalHeaders } from '../additional-headers.js';
export const router = express.Router();
@@ -257,6 +259,41 @@ router.post('/tavily', jsonParser, async (request, response) => {
}
});
+router.post('/koboldcpp', jsonParser, async (request, response) => {
+ try {
+ const { query, url } = request.body;
+
+ if (!url) {
+ console.error('No URL provided for KoboldCpp search');
+ return response.sendStatus(400);
+ }
+
+ console.debug('KoboldCpp search query', query);
+
+ const baseUrl = trimV1(url);
+ const args = {
+ method: 'POST',
+ headers: {},
+ body: JSON.stringify({ q: query }),
+ };
+
+ setAdditionalHeaders(request, args, baseUrl);
+ const result = await fetch(`${baseUrl}/api/extra/websearch`, args);
+
+ if (!result.ok) {
+ const text = await result.text();
+ console.error('KoboldCpp request failed', result.statusText, text);
+ return response.status(500).send(text);
+ }
+
+ const data = await result.json();
+ return response.json(data);
+ } catch (error) {
+ console.error(error);
+ return response.sendStatus(500);
+ }
+});
+
router.post('/visit', jsonParser, async (request, response) => {
try {
const url = request.body.url;
diff --git a/src/endpoints/settings.js b/src/endpoints/settings.js
index 0c4a7cf28..4df4978cc 100644
--- a/src/endpoints/settings.js
+++ b/src/endpoints/settings.js
@@ -9,9 +9,10 @@ import { SETTINGS_FILE } from '../constants.js';
import { getConfigValue, generateTimestamp, removeOldBackups } from '../util.js';
import { jsonParser } from '../express-common.js';
import { getAllUserHandles, getUserDirectories } from '../users.js';
+import { getFileNameValidationFunction } from '../middleware/validateFileName.js';
-const ENABLE_EXTENSIONS = getConfigValue('enableExtensions', true);
-const ENABLE_EXTENSIONS_AUTO_UPDATE = getConfigValue('enableExtensionsAutoUpdate', true);
+const ENABLE_EXTENSIONS = !!getConfigValue('extensions.enabled', true);
+const ENABLE_EXTENSIONS_AUTO_UPDATE = !!getConfigValue('extensions.autoUpdate', true);
const ENABLE_ACCOUNTS = getConfigValue('enableUserAccounts', false);
// 10 minutes
@@ -296,7 +297,7 @@ router.post('/get-snapshots', jsonParser, async (request, response) => {
}
});
-router.post('/load-snapshot', jsonParser, async (request, response) => {
+router.post('/load-snapshot', jsonParser, getFileNameValidationFunction('name'), async (request, response) => {
try {
const userFilesPattern = getFilePrefix(request.user.profile.handle);
@@ -330,7 +331,7 @@ router.post('/make-snapshot', jsonParser, async (request, response) => {
}
});
-router.post('/restore-snapshot', jsonParser, async (request, response) => {
+router.post('/restore-snapshot', jsonParser, getFileNameValidationFunction('name'), async (request, response) => {
try {
const userFilesPattern = getFilePrefix(request.user.profile.handle);
diff --git a/src/endpoints/tokenizers.js b/src/endpoints/tokenizers.js
index bd6fdeec0..5f693b751 100644
--- a/src/endpoints/tokenizers.js
+++ b/src/endpoints/tokenizers.js
@@ -238,6 +238,15 @@ export const sentencepieceTokenizers = [
'jamba',
];
+export const webTokenizers = [
+ 'claude',
+ 'llama3',
+ 'command-r',
+ 'qwen2',
+ 'nemo',
+ 'deepseek',
+];
+
/**
* Gets the Sentencepiece tokenizer by the model name.
* @param {string} model Sentencepiece model name
@@ -275,6 +284,39 @@ export function getSentencepiceTokenizer(model) {
return null;
}
+/**
+ * Gets the Web tokenizer by the model name.
+ * @param {string} model Web tokenizer model name
+ * @returns {WebTokenizer|null} Web tokenizer
+ */
+export function getWebTokenizer(model) {
+ if (model.includes('llama3')) {
+ return llama3_tokenizer;
+ }
+
+ if (model.includes('claude')) {
+ return claude_tokenizer;
+ }
+
+ if (model.includes('command-r')) {
+ return commandTokenizer;
+ }
+
+ if (model.includes('qwen2')) {
+ return qwen2Tokenizer;
+ }
+
+ if (model.includes('nemo')) {
+ return nemoTokenizer;
+ }
+
+ if (model.includes('deepseek')) {
+ return deepseekTokenizer;
+ }
+
+ return null;
+}
+
/**
* Counts the token ids for the given text using the Sentencepiece tokenizer.
* @param {SentencePieceTokenizer} tokenizer Sentencepiece tokenizer
diff --git a/src/endpoints/users-private.js b/src/endpoints/users-private.js
index 81b60acea..1fde87083 100644
--- a/src/endpoints/users-private.js
+++ b/src/endpoints/users-private.js
@@ -23,6 +23,7 @@ router.post('/logout', async (request, response) => {
}
request.session.handle = null;
+ request.session.csrfToken = null;
request.session = null;
return response.sendStatus(204);
} catch (error) {
diff --git a/src/endpoints/vectors.js b/src/endpoints/vectors.js
index 325684601..b0b9819ac 100644
--- a/src/endpoints/vectors.js
+++ b/src/endpoints/vectors.js
@@ -164,7 +164,7 @@ function getSourceSettings(source, request) {
};
case 'transformers':
return {
- model: getConfigValue('extras.embeddingModel', ''),
+ model: getConfigValue('extensions.models.embedding', ''),
};
case 'palm':
return {
diff --git a/src/middleware/basicAuth.js b/src/middleware/basicAuth.js
index 87b7fbcf8..b75856289 100644
--- a/src/middleware/basicAuth.js
+++ b/src/middleware/basicAuth.js
@@ -5,17 +5,18 @@
import { Buffer } from 'node:buffer';
import storage from 'node-persist';
import { getAllUserHandles, toKey, getPasswordHash } from '../users.js';
-import { getConfig, getConfigValue } from '../util.js';
+import { getConfig, getConfigValue, safeReadFileSync } from '../util.js';
const PER_USER_BASIC_AUTH = getConfigValue('perUserBasicAuth', false);
const ENABLE_ACCOUNTS = getConfigValue('enableUserAccounts', false);
-const unauthorizedResponse = (res) => {
- res.set('WWW-Authenticate', 'Basic realm="SillyTavern", charset="UTF-8"');
- return res.status(401).send('Authentication required');
-};
-
const basicAuthMiddleware = async function (request, response, callback) {
+ const unauthorizedWebpage = safeReadFileSync('./public/error/unauthorized.html') ?? '';
+ const unauthorizedResponse = (res) => {
+ res.set('WWW-Authenticate', 'Basic realm="SillyTavern", charset="UTF-8"');
+ return res.status(401).send(unauthorizedWebpage);
+ };
+
const config = getConfig();
const authHeader = request.headers.authorization;
diff --git a/src/middleware/validateFileName.js b/src/middleware/validateFileName.js
new file mode 100644
index 000000000..ccfb0e88d
--- /dev/null
+++ b/src/middleware/validateFileName.js
@@ -0,0 +1,34 @@
+import path from 'node:path';
+
+/**
+ * Gets a middleware function that validates the field in the request body.
+ * @param {string} fieldName Field name
+ * @returns {import('express').RequestHandler} Middleware function
+ */
+export function getFileNameValidationFunction(fieldName) {
+ /**
+ * Validates the field in the request body.
+ * @param {import('express').Request} req Request object
+ * @param {import('express').Response} res Response object
+ * @param {import('express').NextFunction} next Next middleware
+ */
+ return function validateAvatarUrlMiddleware(req, res, next) {
+ if (req.body && fieldName in req.body && typeof req.body[fieldName] === 'string') {
+ const forbiddenRegExp = path.sep === '/' ? /[/\x00]/ : /[/\x00\\]/;
+ if (forbiddenRegExp.test(req.body[fieldName])) {
+ console.error('An error occurred while validating the request body', {
+ handle: req.user.profile.handle,
+ path: req.originalUrl,
+ field: fieldName,
+ value: req.body[fieldName],
+ });
+ return res.sendStatus(400);
+ }
+ }
+
+ next();
+ };
+}
+
+const avatarUrlValidationFunction = getFileNameValidationFunction('avatar_url');
+export default avatarUrlValidationFunction;
diff --git a/src/middleware/webpack-serve.js b/src/middleware/webpack-serve.js
index 546eebc27..950e0a941 100644
--- a/src/middleware/webpack-serve.js
+++ b/src/middleware/webpack-serve.js
@@ -5,8 +5,6 @@ import { publicLibConfig } from '../../webpack.config.js';
export default function getWebpackServeMiddleware() {
const outputPath = publicLibConfig.output?.path;
const outputFile = publicLibConfig.output?.filename;
- /** @type {import('webpack').Compiler|null} */
- let compiler = webpack(publicLibConfig);
/**
* A very spartan recreation of webpack-dev-middleware.
@@ -28,12 +26,9 @@ export default function getWebpackServeMiddleware() {
* @returns {Promise}
*/
devMiddleware.runWebpackCompiler = () => {
- return new Promise((resolve) => {
- if (compiler === null) {
- console.warn('Webpack compiler is already closed.');
- return resolve();
- }
+ const compiler = webpack(publicLibConfig);
+ return new Promise((resolve) => {
console.log();
console.log('Compiling frontend libraries...');
compiler.run((_error, stats) => {
@@ -42,11 +37,7 @@ export default function getWebpackServeMiddleware() {
console.log(output);
console.log();
}
- if (compiler === null) {
- return resolve();
- }
compiler.close(() => {
- compiler = null;
resolve();
});
});
diff --git a/src/middleware/whitelist.js b/src/middleware/whitelist.js
index 9864e19ae..4df5a6798 100644
--- a/src/middleware/whitelist.js
+++ b/src/middleware/whitelist.js
@@ -1,10 +1,11 @@
import path from 'node:path';
import fs from 'node:fs';
import process from 'node:process';
+import Handlebars from 'handlebars';
import ipMatching from 'ip-matching';
import { getIpFromRequest } from '../express-common.js';
-import { color, getConfigValue } from '../util.js';
+import { color, getConfigValue, safeReadFileSync } from '../util.js';
const whitelistPath = path.join(process.cwd(), './whitelist.txt');
const enableForwardedWhitelist = getConfigValue('enableForwardedWhitelist', false);
@@ -52,12 +53,16 @@ function getForwardedIp(req) {
* @returns {import('express').RequestHandler} The middleware function
*/
export default function whitelistMiddleware(whitelistMode, listen) {
+ const forbiddenWebpage = Handlebars.compile(
+ safeReadFileSync('./public/error/forbidden-by-whitelist.html') ?? '',
+ );
+
return function (req, res, next) {
const clientIp = getIpFromRequest(req);
const forwardedIp = getForwardedIp(req);
+ const userAgent = req.headers['user-agent'];
if (listen && !knownIPs.has(clientIp)) {
- const userAgent = req.headers['user-agent'];
console.log(color.yellow(`New connection from ${clientIp}; User Agent: ${userAgent}\n`));
knownIPs.add(clientIp);
@@ -76,9 +81,15 @@ export default function whitelistMiddleware(whitelistMode, listen) {
|| forwardedIp && whitelistMode === true && !whitelist.some(x => ipMatching.matches(forwardedIp, ipMatching.getMatch(x)))
) {
// Log the connection attempt with real IP address
- const ipDetails = forwardedIp ? `${clientIp} (forwarded from ${forwardedIp})` : clientIp;
- console.log(color.red('Forbidden: Connection attempt from ' + ipDetails + '. If you are attempting to connect, please add your IP address in whitelist or disable whitelist mode in config.yaml in root of SillyTavern folder.\n'));
- return res.status(403).send('Forbidden: Connection attempt from ' + ipDetails + '. If you are attempting to connect, please add your IP address in whitelist or disable whitelist mode in config.yaml in root of SillyTavern folder.');
+ const ipDetails = forwardedIp
+ ? `${clientIp} (forwarded from ${forwardedIp})`
+ : clientIp;
+ console.log(
+ color.red(
+ `Blocked connection from ${clientIp}; User Agent: ${userAgent}\n\tTo allow this connection, add its IP address to the whitelist or disable whitelist mode by editing config.yaml in the root directory of your SillyTavern installation.\n`,
+ ),
+ );
+ return res.status(403).send(forbiddenWebpage({ ipDetails }));
}
next();
};
diff --git a/src/prompt-converters.js b/src/prompt-converters.js
index e38813c99..49afc1f2f 100644
--- a/src/prompt-converters.js
+++ b/src/prompt-converters.js
@@ -360,6 +360,8 @@ export function convertCohereMessages(messages, names) {
*/
export function convertGooglePrompt(messages, model, useSysPrompt, names) {
const visionSupportedModels = [
+ 'gemini-2.0-flash-thinking-exp',
+ 'gemini-2.0-flash-thinking-exp-01-21',
'gemini-2.0-flash-thinking-exp-1219',
'gemini-2.0-flash-exp',
'gemini-1.5-flash',
diff --git a/src/transformers.js b/src/transformers.js
index 3413101a6..c8f8da4f6 100644
--- a/src/transformers.js
+++ b/src/transformers.js
@@ -19,31 +19,31 @@ const tasks = {
'text-classification': {
defaultModel: 'Cohee/distilbert-base-uncased-go-emotions-onnx',
pipeline: null,
- configField: 'extras.classificationModel',
+ configField: 'extensions.models.classification',
quantized: true,
},
'image-to-text': {
defaultModel: 'Xenova/vit-gpt2-image-captioning',
pipeline: null,
- configField: 'extras.captioningModel',
+ configField: 'extensions.models.captioning',
quantized: true,
},
'feature-extraction': {
defaultModel: 'Xenova/all-mpnet-base-v2',
pipeline: null,
- configField: 'extras.embeddingModel',
+ configField: 'extensions.models.embedding',
quantized: true,
},
'automatic-speech-recognition': {
defaultModel: 'Xenova/whisper-small',
pipeline: null,
- configField: 'extras.speechToTextModel',
+ configField: 'extensions.models.speechToText',
quantized: true,
},
'text-to-speech': {
defaultModel: 'Xenova/speecht5_tts',
pipeline: null,
- configField: 'extras.textToSpeechModel',
+ configField: 'extensions.models.textToSpeech',
quantized: false,
},
};
@@ -132,7 +132,7 @@ export async function getPipeline(task, forceModel = '') {
const cacheDir = path.join(globalThis.DATA_ROOT, '_cache');
const model = forceModel || getModelForTask(task);
- const localOnly = getConfigValue('extras.disableAutoDownload', false);
+ const localOnly = !getConfigValue('extensions.models.autoDownload', true);
console.log('Initializing transformers.js pipeline for task', task, 'with model', model);
const instance = await pipeline(task, model, { cache_dir: cacheDir, quantized: tasks[task].quantized ?? true, local_files_only: localOnly });
tasks[task].pipeline = instance;
diff --git a/src/users.js b/src/users.js
index 36a5d62b3..c8df39779 100644
--- a/src/users.js
+++ b/src/users.js
@@ -458,7 +458,8 @@ export function getPasswordSalt() {
*/
export function getCookieSessionName() {
// Get server hostname and hash it to generate a session suffix
- const suffix = crypto.createHash('sha256').update(os.hostname()).digest('hex').slice(0, 8);
+ const hostname = os.hostname() || 'localhost';
+ const suffix = crypto.createHash('sha256').update(hostname).digest('hex').slice(0, 8);
return `session-${suffix}`;
}
diff --git a/src/util.js b/src/util.js
index 0fe03e76c..19bd3ccc7 100644
--- a/src/util.js
+++ b/src/util.js
@@ -871,3 +871,14 @@ export class MemoryLimitedMap {
return this.map[Symbol.iterator]();
}
}
+
+/**
+ * A 'safe' version of `fs.readFileSync()`. Returns the contents of a file if it exists, falling back to a default value if not.
+ * @param {string} filePath Path of the file to be read.
+ * @param {Parameters[1]} options Options object to pass through to `fs.readFileSync()` (default: `{ encoding: 'utf-8' }`).
+ * @returns The contents at `filePath` if it exists, or `null` if not.
+ */
+export function safeReadFileSync(filePath, options = { encoding: 'utf-8' }) {
+ if (fs.existsSync(filePath)) return fs.readFileSync(filePath, options);
+ return null;
+}