2023-07-20 20:32:15 +03:00
import {
saveSettingsDebounced ,
systemUserName ,
hideSwipeButtons ,
showSwipeButtons ,
callPopup ,
getRequestHeaders ,
event _types ,
eventSource ,
2023-07-22 21:12:23 +03:00
appendImageToMessage ,
2023-07-22 23:57:48 +03:00
generateQuietPrompt ,
this _chid ,
2023-09-03 00:41:26 +03:00
getCurrentChatId ,
2023-07-20 20:32:15 +03:00
} from "../../../script.js" ;
2023-09-03 00:41:26 +03:00
import { getApiUrl , getContext , extension _settings , doExtrasFetch , modules , renderExtensionTemplate } from "../../extensions.js" ;
2023-07-22 23:57:48 +03:00
import { selected _group } from "../../group-chats.js" ;
2023-09-04 18:00:15 +03:00
import { stringFormat , initScrollHeight , resetScrollHeight , getCharaFilename , saveBase64AsFile } from "../../utils.js" ;
2023-08-22 18:32:18 +03:00
import { getMessageTimeStamp , humanizedDateTime } from "../../RossAscends-mods.js" ;
2023-09-03 00:41:26 +03:00
import { SECRET _KEYS , secret _state } from "../../secrets.js" ;
2023-09-04 18:00:15 +03:00
import { getNovelUnlimitedImageGeneration , getNovelAnlas , loadNovelSubscriptionData } from "../../nai-settings.js" ;
2023-07-20 20:32:15 +03:00
export { MODULE _NAME } ;
// Wraps a string into monospace font-face span
const m = x => ` <span class="monospace"> ${ x } </span> ` ;
// Joins an array of strings with ' / '
const j = a => a . join ( ' / ' ) ;
// Wraps a string into paragraph block
const p = a => ` <p> ${ a } </p> `
const MODULE _NAME = 'sd' ;
const UPDATE _INTERVAL = 1000 ;
2023-09-03 00:41:26 +03:00
const sources = {
extras : 'extras' ,
horde : 'horde' ,
auto : 'auto' ,
novel : 'novel' ,
2023-10-07 18:30:06 +03:00
vlad : 'vlad' ,
2023-11-06 21:47:00 +02:00
openai : 'openai' ,
2023-09-03 00:41:26 +03:00
}
2023-07-20 20:32:15 +03:00
const generationMode = {
CHARACTER : 0 ,
USER : 1 ,
SCENARIO : 2 ,
RAW _LAST : 3 ,
NOW : 4 ,
FACE : 5 ,
FREE : 6 ,
2023-08-18 00:41:21 +03:00
BACKGROUND : 7 ,
2023-07-20 20:32:15 +03:00
}
2023-07-22 21:12:23 +03:00
const modeLabels = {
[ generationMode . CHARACTER ] : 'Character ("Yourself")' ,
[ generationMode . FACE ] : 'Portrait ("Your Face")' ,
[ generationMode . USER ] : 'User ("Me")' ,
[ generationMode . SCENARIO ] : 'Scenario ("The Whole Story")' ,
[ generationMode . NOW ] : 'Last Message' ,
[ generationMode . RAW _LAST ] : 'Raw Last Message' ,
2023-08-18 00:41:21 +03:00
[ generationMode . BACKGROUND ] : 'Background' ,
2023-07-22 21:12:23 +03:00
}
2023-07-20 20:32:15 +03:00
const triggerWords = {
[ generationMode . CHARACTER ] : [ 'you' ] ,
[ generationMode . USER ] : [ 'me' ] ,
[ generationMode . SCENARIO ] : [ 'scene' ] ,
[ generationMode . RAW _LAST ] : [ 'raw_last' ] ,
[ generationMode . NOW ] : [ 'last' ] ,
[ generationMode . FACE ] : [ 'face' ] ,
2023-08-18 00:41:21 +03:00
[ generationMode . BACKGROUND ] : [ 'background' ] ,
2023-07-20 20:32:15 +03:00
}
2023-11-01 22:58:59 +02:00
const messageTrigger = {
activationRegex : /\b(send|mail|imagine|generate|make|create|draw|paint|render)\b.*\b(pic|picture|image|drawing|painting|photo|photograph)\b(?:\s+of)?(?:\s+(?:a|an|the)?)?(.+)/i ,
specialCases : {
[ generationMode . CHARACTER ] : [ 'you' , 'yourself' ] ,
[ generationMode . USER ] : [ 'me' , 'myself' ] ,
[ generationMode . SCENARIO ] : [ 'story' , 'scenario' , 'whole story' ] ,
[ generationMode . NOW ] : [ 'last message' ] ,
[ generationMode . FACE ] : [ 'your face' , 'your portrait' , 'your selfie' ] ,
[ generationMode . BACKGROUND ] : [ 'background' , 'scene background' , 'scene' , 'scenery' , 'surroundings' , 'environment' ] ,
} ,
}
2023-07-22 21:12:23 +03:00
const promptTemplates = {
2023-07-20 20:32:15 +03:00
/*OLD: [generationMode.CHARACTER]: "Pause your roleplay and provide comma-delimited list of phrases and keywords which describe {{char}}'s physical appearance and clothing. Ignore {{char}}'s personality traits, and chat history when crafting this description. End your response once the comma-delimited list is complete. Do not roleplay when writing this description, and do not attempt to continue the story.", */
[ generationMode . CHARACTER ] : "[In the next response I want you to provide only a detailed comma-delimited list of keywords and phrases which describe {{char}}. The list must include all of the following items in this order: name, species and race, gender, age, clothing, occupation, physical features and appearances. Do not include descriptions of non-visual qualities such as personality, movements, scents, mental traits, or anything which could not be seen in a still photograph. Do not write in full sentences. Prefix your description with the phrase 'full body portrait,']" ,
//face-specific prompt
[ generationMode . FACE ] : "[In the next response I want you to provide only a detailed comma-delimited list of keywords and phrases which describe {{char}}. The list must include all of the following items in this order: name, species and race, gender, age, facial features and expressions, occupation, hair and hair accessories (if any), what they are wearing on their upper body (if anything). Do not describe anything below their neck. Do not include descriptions of non-visual qualities such as personality, movements, scents, mental traits, or anything which could not be seen in a still photograph. Do not write in full sentences. Prefix your description with the phrase 'close up facial portrait,']" ,
//prompt for only the last message
[ generationMode . USER ] : "[Pause your roleplay and provide a detailed description of {{user}}'s physical appearance from the perspective of {{char}} in the form of a comma-delimited list of keywords and phrases. The list must include all of the following items in this order: name, species and race, gender, age, clothing, occupation, physical features and appearances. Do not include descriptions of non-visual qualities such as personality, movements, scents, mental traits, or anything which could not be seen in a still photograph. Do not write in full sentences. Prefix your description with the phrase 'full body portrait,'. Ignore the rest of the story when crafting this description. Do not roleplay as {{char}} when writing this description, and do not attempt to continue the story.]" ,
[ generationMode . SCENARIO ] : "[Pause your roleplay and provide a detailed description for all of the following: a brief recap of recent events in the story, {{char}}'s appearance, and {{char}}'s surroundings. Do not roleplay while writing this description.]" ,
[ generationMode . NOW ] : ` [Pause your roleplay. Your next response must be formatted as a single comma-delimited list of concise keywords. The list will describe of the visual details included in the last chat message.
Only mention characters by using pronouns ( 'he' , 'his' , 'she' , 'her' , 'it' , 'its' ) or neutral nouns ( 'male' , 'the man' , 'female' , 'the woman' ) .
Ignore non - visible things such as feelings , personality traits , thoughts , and spoken dialog .
Add keywords in this precise order :
a keyword to describe the location of the scene ,
a keyword to mention how many characters of each gender or type are present in the scene ( minimum of two characters :
{ { user } } and { { char } } , example : '2 men ' or '1 man 1 woman ' , '1 man 3 robots' ) ,
keywords to describe the relative physical positioning of the characters to each other ( if a commonly known term for the positioning is known use it instead of describing the positioning in detail ) + 'POV' ,
a single keyword or phrase to describe the primary act taking place in the last chat message ,
keywords to describe { { char } } ' s physical appearance and facial expression ,
keywords to describe { { char } } ' s actions ,
keywords to describe { { user } } ' s physical appearance and actions .
If character actions involve direct physical interaction with another character , mention specifically which body parts interacting and how .
A correctly formatted example response would be :
'(location),(character list by gender),(primary action), (relative character position) POV, (character 1' s description and actions ) , ( character 2 's description and actions)' ] ` ,
[ generationMode . RAW _LAST ] : "[Pause your roleplay and provide ONLY the last chat message string back to me verbatim. Do not write anything after the string. Do not roleplay at all in your response. Do not continue the roleplay story.]" ,
2023-08-18 00:41:21 +03:00
[ generationMode . BACKGROUND ] : "[Pause your roleplay and provide a detailed description of {{char}}'s surroundings in the form of a comma-delimited list of keywords and phrases. The list must include all of the following items in this order: location, time of day, weather, lighting, and any other relevant details. Do not include descriptions of characters and non-visual qualities such as names, personality, movements, scents, mental traits, or anything which could not be seen in a still photograph. Do not write in full sentences. Prefix your description with the phrase 'background,'. Ignore the rest of the story when crafting this description. Do not roleplay as {{user}} when writing this description, and do not attempt to continue the story.]" ,
2023-07-20 20:32:15 +03:00
}
const helpString = [
2023-11-04 01:21:20 +02:00
` ${ m ( '(argument)' ) } – requests SD to make an image. Supported arguments: ${ m ( j ( Object . values ( triggerWords ) . flat ( ) ) ) } . ` ,
` Anything else would trigger a "free mode" to make SD generate whatever you prompted. Example: '/sd apple tree' would generate a picture of an apple tree. ` ,
] . join ( ' ' ) ;
2023-07-20 20:32:15 +03:00
2023-10-23 10:41:39 +03:00
const defaultPrefix = 'best quality, absurdres, aesthetic,' ;
2023-10-22 00:10:48 +03:00
const defaultNegative = 'lowres, bad anatomy, bad hands, text, error, cropped, worst quality, low quality, normal quality, jpeg artifacts, signature, watermark, username, blurry' ;
const defaultStyles = [
{
name : 'Default' ,
negative : defaultNegative ,
prefix : defaultPrefix ,
} ,
] ;
2023-07-20 20:32:15 +03:00
const defaultSettings = {
2023-09-03 00:41:26 +03:00
source : sources . extras ,
2023-07-20 20:32:15 +03:00
// CFG Scale
scale _min : 1 ,
scale _max : 30 ,
scale _step : 0.5 ,
scale : 7 ,
// Sampler steps
steps _min : 1 ,
steps _max : 150 ,
steps _step : 1 ,
steps : 20 ,
// Image dimensions (Width & Height)
dimension _min : 64 ,
dimension _max : 2048 ,
dimension _step : 64 ,
width : 512 ,
height : 512 ,
2023-10-22 00:10:48 +03:00
prompt _prefix : defaultPrefix ,
negative _prompt : defaultNegative ,
2023-07-20 20:32:15 +03:00
sampler : 'DDIM' ,
model : '' ,
// Automatic1111/Horde exclusives
restore _faces : false ,
enable _hr : false ,
// Horde settings
horde : false ,
horde _nsfw : false ,
horde _karras : true ,
2023-10-08 22:29:33 +03:00
horde _sanitize : true ,
2023-07-20 20:32:15 +03:00
// Refine mode
refine _mode : false ,
2023-10-20 15:03:26 +03:00
expand : false ,
2023-11-01 22:58:59 +02:00
interactive _mode : false ,
2023-07-22 21:12:23 +03:00
prompts : promptTemplates ,
2023-09-03 00:41:26 +03:00
// AUTOMATIC1111 settings
auto _url : 'http://localhost:7860' ,
2023-09-03 01:19:31 +03:00
auto _auth : '' ,
2023-09-03 14:56:02 +03:00
2023-10-07 18:30:06 +03:00
vlad _url : 'http://localhost:7860' ,
vlad _auth : '' ,
2023-09-03 14:56:02 +03:00
hr _upscaler : 'Latent' ,
hr _scale : 2.0 ,
hr _scale _min : 1.0 ,
hr _scale _max : 4.0 ,
hr _scale _step : 0.1 ,
denoising _strength : 0.7 ,
denoising _strength _min : 0.0 ,
denoising _strength _max : 1.0 ,
denoising _strength _step : 0.01 ,
hr _second _pass _steps : 0 ,
hr _second _pass _steps _min : 0 ,
hr _second _pass _steps _max : 150 ,
hr _second _pass _steps _step : 1 ,
2023-09-04 18:00:15 +03:00
// NovelAI settings
novel _upscale _ratio _min : 1.0 ,
novel _upscale _ratio _max : 4.0 ,
novel _upscale _ratio _step : 0.1 ,
novel _upscale _ratio : 1.0 ,
novel _anlas _guard : false ,
2023-10-22 00:10:48 +03:00
2023-11-06 21:47:00 +02:00
// OpenAI settings
openai _style : 'vivid' ,
openai _quality : 'standard' ,
2023-10-22 00:10:48 +03:00
style : 'Default' ,
styles : defaultStyles ,
2023-09-03 00:41:26 +03:00
}
2023-11-01 22:58:59 +02:00
function processTriggers ( chat , _ , abort ) {
if ( ! extension _settings . sd . interactive _mode ) {
return ;
}
const lastMessage = chat [ chat . length - 1 ] ;
if ( ! lastMessage ) {
return ;
}
const message = lastMessage . mes ;
const isUser = lastMessage . is _user ;
if ( ! message || ! isUser ) {
return ;
}
const messageLower = message . toLowerCase ( ) ;
try {
const activationRegex = new RegExp ( messageTrigger . activationRegex , 'i' ) ;
const activationMatch = messageLower . match ( activationRegex ) ;
if ( ! activationMatch ) {
return ;
}
let subject = activationMatch [ 3 ] . trim ( ) ;
if ( ! subject ) {
return ;
}
console . log ( ` SD: Triggered by " ${ message } ", detected subject: ${ subject } " ` ) ;
for ( const [ specialMode , triggers ] of Object . entries ( messageTrigger . specialCases ) ) {
for ( const trigger of triggers ) {
if ( subject === trigger ) {
subject = triggerWords [ specialMode ] [ 0 ] ;
console . log ( ` SD: Detected special case " ${ trigger } ", switching to mode ${ specialMode } ` ) ;
break ;
}
}
}
abort ( true ) ;
setTimeout ( ( ) => generatePicture ( 'sd' , subject , message ) , 1 ) ;
} catch {
console . log ( 'SD: Failed to process triggers.' ) ;
return ;
}
}
window [ 'SD_ProcessTriggers' ] = processTriggers ;
2023-10-07 18:30:06 +03:00
function getSdRequestBody ( ) {
switch ( extension _settings . sd . source ) {
case sources . vlad :
return { url : extension _settings . sd . vlad _url , auth : extension _settings . sd . vlad _auth } ;
case sources . auto :
return { url : extension _settings . sd . auto _url , auth : extension _settings . sd . auto _auth } ;
default :
throw new Error ( 'Invalid SD source.' ) ;
}
}
2023-09-03 01:19:31 +03:00
2023-09-03 00:41:26 +03:00
function toggleSourceControls ( ) {
$ ( '.sd_settings [data-sd-source]' ) . each ( function ( ) {
2023-10-07 18:30:06 +03:00
const source = $ ( this ) . data ( 'sd-source' ) . split ( ',' ) ;
$ ( this ) . toggle ( source . includes ( extension _settings . sd . source ) ) ;
2023-09-03 00:41:26 +03:00
} ) ;
2023-07-20 20:32:15 +03:00
}
async function loadSettings ( ) {
2023-09-03 00:41:26 +03:00
// Initialize settings
2023-07-20 20:32:15 +03:00
if ( Object . keys ( extension _settings . sd ) . length === 0 ) {
Object . assign ( extension _settings . sd , defaultSettings ) ;
}
2023-09-03 00:41:26 +03:00
// Insert missing settings
for ( const [ key , value ] of Object . entries ( defaultSettings ) ) {
if ( extension _settings . sd [ key ] === undefined ) {
extension _settings . sd [ key ] = value ;
}
}
2023-07-22 21:12:23 +03:00
if ( extension _settings . sd . prompts === undefined ) {
extension _settings . sd . prompts = promptTemplates ;
}
2023-08-18 00:41:21 +03:00
// Insert missing templates
for ( const [ key , value ] of Object . entries ( promptTemplates ) ) {
if ( extension _settings . sd . prompts [ key ] === undefined ) {
extension _settings . sd . prompts [ key ] = value ;
}
}
2023-07-22 23:57:48 +03:00
if ( extension _settings . sd . character _prompts === undefined ) {
extension _settings . sd . character _prompts = { } ;
}
2023-10-22 00:10:48 +03:00
if ( ! Array . isArray ( extension _settings . sd . styles ) ) {
extension _settings . sd . styles = defaultStyles ;
}
2023-09-03 00:41:26 +03:00
$ ( '#sd_source' ) . val ( extension _settings . sd . source ) ;
2023-07-20 20:32:15 +03:00
$ ( '#sd_scale' ) . val ( extension _settings . sd . scale ) . trigger ( 'input' ) ;
$ ( '#sd_steps' ) . val ( extension _settings . sd . steps ) . trigger ( 'input' ) ;
$ ( '#sd_prompt_prefix' ) . val ( extension _settings . sd . prompt _prefix ) . trigger ( 'input' ) ;
$ ( '#sd_negative_prompt' ) . val ( extension _settings . sd . negative _prompt ) . trigger ( 'input' ) ;
$ ( '#sd_width' ) . val ( extension _settings . sd . width ) . trigger ( 'input' ) ;
$ ( '#sd_height' ) . val ( extension _settings . sd . height ) . trigger ( 'input' ) ;
2023-09-03 14:56:02 +03:00
$ ( '#sd_hr_scale' ) . val ( extension _settings . sd . hr _scale ) . trigger ( 'input' ) ;
$ ( '#sd_denoising_strength' ) . val ( extension _settings . sd . denoising _strength ) . trigger ( 'input' ) ;
$ ( '#sd_hr_second_pass_steps' ) . val ( extension _settings . sd . hr _second _pass _steps ) . trigger ( 'input' ) ;
2023-09-04 18:00:15 +03:00
$ ( '#sd_novel_upscale_ratio' ) . val ( extension _settings . sd . novel _upscale _ratio ) . trigger ( 'input' ) ;
$ ( '#sd_novel_anlas_guard' ) . prop ( 'checked' , extension _settings . sd . novel _anlas _guard ) ;
2023-07-20 20:32:15 +03:00
$ ( '#sd_horde' ) . prop ( 'checked' , extension _settings . sd . horde ) ;
$ ( '#sd_horde_nsfw' ) . prop ( 'checked' , extension _settings . sd . horde _nsfw ) ;
$ ( '#sd_horde_karras' ) . prop ( 'checked' , extension _settings . sd . horde _karras ) ;
2023-10-08 22:29:33 +03:00
$ ( '#sd_horde_sanitize' ) . prop ( 'checked' , extension _settings . sd . horde _sanitize ) ;
2023-07-20 20:32:15 +03:00
$ ( '#sd_restore_faces' ) . prop ( 'checked' , extension _settings . sd . restore _faces ) ;
$ ( '#sd_enable_hr' ) . prop ( 'checked' , extension _settings . sd . enable _hr ) ;
$ ( '#sd_refine_mode' ) . prop ( 'checked' , extension _settings . sd . refine _mode ) ;
2023-10-20 15:03:26 +03:00
$ ( '#sd_expand' ) . prop ( 'checked' , extension _settings . sd . expand ) ;
2023-09-03 00:41:26 +03:00
$ ( '#sd_auto_url' ) . val ( extension _settings . sd . auto _url ) ;
2023-09-03 01:19:31 +03:00
$ ( '#sd_auto_auth' ) . val ( extension _settings . sd . auto _auth ) ;
2023-10-07 18:30:06 +03:00
$ ( '#sd_vlad_url' ) . val ( extension _settings . sd . vlad _url ) ;
$ ( '#sd_vlad_auth' ) . val ( extension _settings . sd . vlad _auth ) ;
2023-11-01 22:58:59 +02:00
$ ( '#sd_interactive_mode' ) . prop ( 'checked' , extension _settings . sd . interactive _mode ) ;
2023-11-06 21:47:00 +02:00
$ ( '#sd_openai_style' ) . val ( extension _settings . sd . openai _style ) ;
$ ( '#sd_openai_quality' ) . val ( extension _settings . sd . openai _quality ) ;
2023-07-20 20:32:15 +03:00
2023-10-22 00:10:48 +03:00
for ( const style of extension _settings . sd . styles ) {
const option = document . createElement ( 'option' ) ;
option . value = style . name ;
option . text = style . name ;
option . selected = style . name === extension _settings . sd . style ;
$ ( '#sd_style' ) . append ( option ) ;
}
2023-09-03 00:41:26 +03:00
toggleSourceControls ( ) ;
2023-07-22 21:12:23 +03:00
addPromptTemplates ( ) ;
2023-07-20 20:32:15 +03:00
await Promise . all ( [ loadSamplers ( ) , loadModels ( ) ] ) ;
}
2023-07-22 21:12:23 +03:00
function addPromptTemplates ( ) {
$ ( '#sd_prompt_templates' ) . empty ( ) ;
for ( const [ name , prompt ] of Object . entries ( extension _settings . sd . prompts ) ) {
const label = $ ( '<label></label>' )
. text ( modeLabels [ name ] )
. attr ( 'for' , ` sd_prompt_ ${ name } ` ) ;
const textarea = $ ( '<textarea></textarea>' )
2023-07-22 23:57:48 +03:00
. addClass ( 'textarea_compact text_pole' )
2023-07-22 21:12:23 +03:00
. attr ( 'id' , ` sd_prompt_ ${ name } ` )
2023-10-31 22:16:33 +02:00
. attr ( 'rows' , 3 )
2023-07-22 21:12:23 +03:00
. val ( prompt ) . on ( 'input' , ( ) => {
extension _settings . sd . prompts [ name ] = textarea . val ( ) ;
saveSettingsDebounced ( ) ;
} ) ;
const button = $ ( '<button></button>' )
. addClass ( 'menu_button fa-solid fa-undo' )
. attr ( 'title' , 'Restore default' )
. on ( 'click' , ( ) => {
textarea . val ( promptTemplates [ name ] ) ;
extension _settings . sd . prompts [ name ] = promptTemplates [ name ] ;
saveSettingsDebounced ( ) ;
} ) ;
const container = $ ( '<div></div>' )
. addClass ( 'title_restorable' )
. append ( label )
. append ( button )
$ ( '#sd_prompt_templates' ) . append ( container ) ;
$ ( '#sd_prompt_templates' ) . append ( textarea ) ;
}
}
2023-11-01 22:58:59 +02:00
function onInteractiveModeInput ( ) {
extension _settings . sd . interactive _mode = ! ! $ ( this ) . prop ( 'checked' ) ;
saveSettingsDebounced ( ) ;
}
2023-10-22 00:10:48 +03:00
function onStyleSelect ( ) {
const selectedStyle = String ( $ ( '#sd_style' ) . find ( ':selected' ) . val ( ) ) ;
const styleObject = extension _settings . sd . styles . find ( x => x . name === selectedStyle ) ;
if ( ! styleObject ) {
console . warn ( ` Could not find style object for ${ selectedStyle } ` ) ;
return ;
}
$ ( '#sd_prompt_prefix' ) . val ( styleObject . prefix ) . trigger ( 'input' ) ;
$ ( '#sd_negative_prompt' ) . val ( styleObject . negative ) . trigger ( 'input' ) ;
extension _settings . sd . style = selectedStyle ;
saveSettingsDebounced ( ) ;
}
async function onSaveStyleClick ( ) {
const userInput = await callPopup ( 'Enter style name:' , 'input' , '' , { okButton : 'Save' } ) ;
if ( ! userInput ) {
return ;
}
const name = String ( userInput ) . trim ( ) ;
const prefix = String ( $ ( '#sd_prompt_prefix' ) . val ( ) ) ;
const negative = String ( $ ( '#sd_negative_prompt' ) . val ( ) ) ;
const existingStyle = extension _settings . sd . styles . find ( x => x . name === name ) ;
if ( existingStyle ) {
existingStyle . prefix = prefix ;
existingStyle . negative = negative ;
$ ( '#sd_style' ) . val ( name ) ;
saveSettingsDebounced ( ) ;
return ;
}
const styleObject = {
name : name ,
prefix : prefix ,
negative : negative ,
} ;
extension _settings . sd . styles . push ( styleObject ) ;
const option = document . createElement ( 'option' ) ;
option . value = styleObject . name ;
option . text = styleObject . name ;
option . selected = true ;
$ ( '#sd_style' ) . append ( option ) ;
$ ( '#sd_style' ) . val ( styleObject . name ) ;
saveSettingsDebounced ( ) ;
}
2023-10-20 15:03:26 +03:00
async function expandPrompt ( prompt ) {
try {
const response = await fetch ( '/api/sd/expand' , {
method : 'POST' ,
headers : getRequestHeaders ( ) ,
body : JSON . stringify ( { prompt : prompt } ) ,
} ) ;
if ( ! response . ok ) {
throw new Error ( 'API returned an error.' ) ;
}
const data = await response . json ( ) ;
return data . prompt ;
} catch {
return prompt ;
}
}
2023-10-20 15:43:55 +03:00
/ * *
* Modifies prompt based on auto - expansion and user inputs .
* @ param { string } prompt Prompt to refine
* @ param { boolean } allowExpand Whether to allow auto - expansion
* @ returns { Promise < string > } Refined prompt
* /
async function refinePrompt ( prompt , allowExpand ) {
if ( allowExpand && extension _settings . sd . expand ) {
2023-10-20 15:03:26 +03:00
prompt = await expandPrompt ( prompt ) ;
}
2023-07-22 20:32:58 +03:00
if ( extension _settings . sd . refine _mode ) {
2023-10-07 18:30:06 +03:00
const refinedPrompt = await callPopup ( '<h3>Review and edit the prompt:</h3>Press "Cancel" to abort the image generation.' , 'input' , prompt . trim ( ) , { rows : 5 , okButton : 'Generate' } ) ;
2023-07-22 20:32:58 +03:00
if ( refinedPrompt ) {
return refinedPrompt ;
} else {
throw new Error ( 'Generation aborted by user.' ) ;
}
}
return prompt ;
}
2023-07-22 23:57:48 +03:00
function onChatChanged ( ) {
if ( this _chid === undefined || selected _group ) {
$ ( '#sd_character_prompt_block' ) . hide ( ) ;
return ;
}
$ ( '#sd_character_prompt_block' ) . show ( ) ;
const key = getCharaFilename ( this _chid ) ;
$ ( '#sd_character_prompt' ) . val ( key ? ( extension _settings . sd . character _prompts [ key ] || '' ) : '' ) ;
}
function onCharacterPromptInput ( ) {
const key = getCharaFilename ( this _chid ) ;
extension _settings . sd . character _prompts [ key ] = $ ( '#sd_character_prompt' ) . val ( ) ;
2023-07-23 00:28:23 +03:00
resetScrollHeight ( $ ( this ) ) ;
2023-07-22 23:57:48 +03:00
saveSettingsDebounced ( ) ;
}
function getCharacterPrefix ( ) {
2023-10-07 18:30:06 +03:00
if ( ! this _chid || selected _group ) {
2023-07-22 23:57:48 +03:00
return '' ;
}
const key = getCharaFilename ( this _chid ) ;
if ( key ) {
return extension _settings . sd . character _prompts [ key ] || '' ;
}
return '' ;
}
2023-10-22 00:10:48 +03:00
/ * *
* Combines two prompt prefixes into one .
* @ param { string } str1 Base string
* @ param { string } str2 Secondary string
* @ param { string } macro Macro to replace with the secondary string
* @ returns { string } Combined string with a comma between them
* /
function combinePrefixes ( str1 , str2 , macro = '' ) {
2023-07-22 23:57:48 +03:00
if ( ! str2 ) {
return str1 ;
}
// Remove leading/trailing white spaces and commas from the strings
str1 = str1 . trim ( ) . replace ( /^,|,$/g , '' ) ;
str2 = str2 . trim ( ) . replace ( /^,|,$/g , '' ) ;
2023-10-22 00:10:48 +03:00
// Combine the strings with a comma between them)
const result = macro && str1 . includes ( macro ) ? str1 . replace ( macro , str2 ) : ` ${ str1 } , ${ str2 } , ` ;
2023-07-22 23:57:48 +03:00
return result ;
}
2023-10-20 15:03:26 +03:00
function onExpandInput ( ) {
extension _settings . sd . expand = ! ! $ ( this ) . prop ( 'checked' ) ;
saveSettingsDebounced ( ) ;
}
2023-07-20 20:32:15 +03:00
function onRefineModeInput ( ) {
extension _settings . sd . refine _mode = ! ! $ ( '#sd_refine_mode' ) . prop ( 'checked' ) ;
saveSettingsDebounced ( ) ;
}
function onScaleInput ( ) {
extension _settings . sd . scale = Number ( $ ( '#sd_scale' ) . val ( ) ) ;
$ ( '#sd_scale_value' ) . text ( extension _settings . sd . scale . toFixed ( 1 ) ) ;
saveSettingsDebounced ( ) ;
}
function onStepsInput ( ) {
extension _settings . sd . steps = Number ( $ ( '#sd_steps' ) . val ( ) ) ;
$ ( '#sd_steps_value' ) . text ( extension _settings . sd . steps ) ;
saveSettingsDebounced ( ) ;
}
function onPromptPrefixInput ( ) {
extension _settings . sd . prompt _prefix = $ ( '#sd_prompt_prefix' ) . val ( ) ;
resetScrollHeight ( $ ( this ) ) ;
saveSettingsDebounced ( ) ;
}
function onNegativePromptInput ( ) {
extension _settings . sd . negative _prompt = $ ( '#sd_negative_prompt' ) . val ( ) ;
resetScrollHeight ( $ ( this ) ) ;
saveSettingsDebounced ( ) ;
}
function onSamplerChange ( ) {
extension _settings . sd . sampler = $ ( '#sd_sampler' ) . find ( ':selected' ) . val ( ) ;
saveSettingsDebounced ( ) ;
}
function onWidthInput ( ) {
extension _settings . sd . width = Number ( $ ( '#sd_width' ) . val ( ) ) ;
$ ( '#sd_width_value' ) . text ( extension _settings . sd . width ) ;
saveSettingsDebounced ( ) ;
}
function onHeightInput ( ) {
extension _settings . sd . height = Number ( $ ( '#sd_height' ) . val ( ) ) ;
$ ( '#sd_height_value' ) . text ( extension _settings . sd . height ) ;
saveSettingsDebounced ( ) ;
}
2023-09-03 00:41:26 +03:00
async function onSourceChange ( ) {
extension _settings . sd . source = $ ( '#sd_source' ) . find ( ':selected' ) . val ( ) ;
2023-07-20 20:32:15 +03:00
extension _settings . sd . model = null ;
extension _settings . sd . sampler = null ;
2023-09-03 00:41:26 +03:00
toggleSourceControls ( ) ;
2023-07-20 20:32:15 +03:00
saveSettingsDebounced ( ) ;
await Promise . all ( [ loadModels ( ) , loadSamplers ( ) ] ) ;
}
2023-11-06 21:47:00 +02:00
async function onOpenAiStyleSelect ( ) {
extension _settings . sd . openai _style = String ( $ ( '#sd_openai_style' ) . find ( ':selected' ) . val ( ) ) ;
saveSettingsDebounced ( ) ;
}
async function onOpenAiQualitySelect ( ) {
extension _settings . sd . openai _quality = String ( $ ( '#sd_openai_quality' ) . find ( ':selected' ) . val ( ) ) ;
saveSettingsDebounced ( ) ;
}
2023-09-04 18:00:15 +03:00
async function onViewAnlasClick ( ) {
const result = await loadNovelSubscriptionData ( ) ;
if ( ! result ) {
toastr . warning ( 'Are you subscribed?' , 'Could not load NovelAI subscription data' ) ;
return ;
}
const anlas = getNovelAnlas ( ) ;
const unlimitedGeneration = getNovelUnlimitedImageGeneration ( ) ;
toastr . info ( ` Free image generation: ${ unlimitedGeneration ? 'Yes' : 'No' } ` , ` Anlas: ${ anlas } ` ) ;
}
function onNovelUpscaleRatioInput ( ) {
extension _settings . sd . novel _upscale _ratio = Number ( $ ( '#sd_novel_upscale_ratio' ) . val ( ) ) ;
$ ( '#sd_novel_upscale_ratio_value' ) . text ( extension _settings . sd . novel _upscale _ratio . toFixed ( 1 ) ) ;
saveSettingsDebounced ( ) ;
}
function onNovelAnlasGuardInput ( ) {
extension _settings . sd . novel _anlas _guard = ! ! $ ( '#sd_novel_anlas_guard' ) . prop ( 'checked' ) ;
saveSettingsDebounced ( ) ;
}
2023-10-08 22:29:33 +03:00
function onHordeNsfwInput ( ) {
2023-07-20 20:32:15 +03:00
extension _settings . sd . horde _nsfw = ! ! $ ( this ) . prop ( 'checked' ) ;
saveSettingsDebounced ( ) ;
}
2023-10-08 22:29:33 +03:00
function onHordeKarrasInput ( ) {
2023-07-20 20:32:15 +03:00
extension _settings . sd . horde _karras = ! ! $ ( this ) . prop ( 'checked' ) ;
saveSettingsDebounced ( ) ;
}
2023-10-08 22:29:33 +03:00
function onHordeSanitizeInput ( ) {
extension _settings . sd . horde _sanitize = ! ! $ ( this ) . prop ( 'checked' ) ;
saveSettingsDebounced ( ) ;
}
2023-07-20 20:32:15 +03:00
function onRestoreFacesInput ( ) {
extension _settings . sd . restore _faces = ! ! $ ( this ) . prop ( 'checked' ) ;
saveSettingsDebounced ( ) ;
}
function onHighResFixInput ( ) {
extension _settings . sd . enable _hr = ! ! $ ( this ) . prop ( 'checked' ) ;
saveSettingsDebounced ( ) ;
}
2023-09-03 00:41:26 +03:00
function onAutoUrlInput ( ) {
extension _settings . sd . auto _url = $ ( '#sd_auto_url' ) . val ( ) ;
saveSettingsDebounced ( ) ;
}
2023-09-03 01:19:31 +03:00
function onAutoAuthInput ( ) {
extension _settings . sd . auto _auth = $ ( '#sd_auto_auth' ) . val ( ) ;
saveSettingsDebounced ( ) ;
}
2023-10-07 18:30:06 +03:00
function onVladUrlInput ( ) {
extension _settings . sd . vlad _url = $ ( '#sd_vlad_url' ) . val ( ) ;
saveSettingsDebounced ( ) ;
}
function onVladAuthInput ( ) {
extension _settings . sd . vlad _auth = $ ( '#sd_vlad_auth' ) . val ( ) ;
saveSettingsDebounced ( ) ;
}
2023-09-03 14:56:02 +03:00
function onHrUpscalerChange ( ) {
extension _settings . sd . hr _upscaler = $ ( '#sd_hr_upscaler' ) . find ( ':selected' ) . val ( ) ;
saveSettingsDebounced ( ) ;
}
function onHrScaleInput ( ) {
extension _settings . sd . hr _scale = Number ( $ ( '#sd_hr_scale' ) . val ( ) ) ;
$ ( '#sd_hr_scale_value' ) . text ( extension _settings . sd . hr _scale . toFixed ( 1 ) ) ;
saveSettingsDebounced ( ) ;
}
function onDenoisingStrengthInput ( ) {
extension _settings . sd . denoising _strength = Number ( $ ( '#sd_denoising_strength' ) . val ( ) ) ;
$ ( '#sd_denoising_strength_value' ) . text ( extension _settings . sd . denoising _strength . toFixed ( 2 ) ) ;
saveSettingsDebounced ( ) ;
}
function onHrSecondPassStepsInput ( ) {
extension _settings . sd . hr _second _pass _steps = Number ( $ ( '#sd_hr_second_pass_steps' ) . val ( ) ) ;
$ ( '#sd_hr_second_pass_steps_value' ) . text ( extension _settings . sd . hr _second _pass _steps ) ;
saveSettingsDebounced ( ) ;
}
2023-09-03 00:41:26 +03:00
async function validateAutoUrl ( ) {
try {
if ( ! extension _settings . sd . auto _url ) {
throw new Error ( 'URL is not set.' ) ;
}
const result = await fetch ( '/api/sd/ping' , {
method : 'POST' ,
headers : getRequestHeaders ( ) ,
2023-10-07 18:30:06 +03:00
body : JSON . stringify ( getSdRequestBody ( ) ) ,
2023-09-03 00:41:26 +03:00
} ) ;
if ( ! result . ok ) {
throw new Error ( 'SD WebUI returned an error.' ) ;
}
await loadSamplers ( ) ;
await loadModels ( ) ;
toastr . success ( 'SD WebUI API connected.' ) ;
} catch ( error ) {
toastr . error ( ` Could not validate SD WebUI API: ${ error . message } ` ) ;
}
}
2023-10-07 18:30:06 +03:00
async function validateVladUrl ( ) {
try {
if ( ! extension _settings . sd . vlad _url ) {
throw new Error ( 'URL is not set.' ) ;
}
const result = await fetch ( '/api/sd/ping' , {
method : 'POST' ,
headers : getRequestHeaders ( ) ,
body : JSON . stringify ( getSdRequestBody ( ) ) ,
} ) ;
if ( ! result . ok ) {
throw new Error ( 'SD.Next returned an error.' ) ;
}
await loadSamplers ( ) ;
await loadModels ( ) ;
toastr . success ( 'SD.Next API connected.' ) ;
} catch ( error ) {
toastr . error ( ` Could not validate SD.Next API: ${ error . message } ` ) ;
}
}
2023-07-20 20:32:15 +03:00
async function onModelChange ( ) {
extension _settings . sd . model = $ ( '#sd_model' ) . find ( ':selected' ) . val ( ) ;
saveSettingsDebounced ( ) ;
2023-11-06 21:47:00 +02:00
const cloudSources = [ sources . horde , sources . novel , sources . openai ] ;
2023-09-03 00:41:26 +03:00
if ( cloudSources . includes ( extension _settings . sd . source ) ) {
return ;
}
toastr . info ( 'Updating remote model...' , 'Please wait' ) ;
if ( extension _settings . sd . source === sources . extras ) {
2023-07-20 20:32:15 +03:00
await updateExtrasRemoteModel ( ) ;
}
2023-10-07 18:30:06 +03:00
if ( extension _settings . sd . source === sources . auto || extension _settings . sd . source === sources . vlad ) {
2023-09-03 00:41:26 +03:00
await updateAutoRemoteModel ( ) ;
}
toastr . success ( 'Model successfully loaded!' , 'Stable Diffusion' ) ;
}
async function getAutoRemoteModel ( ) {
try {
const result = await fetch ( '/api/sd/get-model' , {
method : 'POST' ,
headers : getRequestHeaders ( ) ,
2023-10-07 18:30:06 +03:00
body : JSON . stringify ( getSdRequestBody ( ) ) ,
2023-09-03 00:41:26 +03:00
} ) ;
if ( ! result . ok ) {
throw new Error ( 'SD WebUI returned an error.' ) ;
}
const data = await result . text ( ) ;
return data ;
} catch ( error ) {
console . error ( error ) ;
return null ;
}
}
2023-09-03 14:56:02 +03:00
async function getAutoRemoteUpscalers ( ) {
try {
const result = await fetch ( '/api/sd/upscalers' , {
method : 'POST' ,
headers : getRequestHeaders ( ) ,
2023-10-07 18:30:06 +03:00
body : JSON . stringify ( getSdRequestBody ( ) ) ,
2023-09-03 14:56:02 +03:00
} ) ;
if ( ! result . ok ) {
throw new Error ( 'SD WebUI returned an error.' ) ;
}
const data = await result . json ( ) ;
return data ;
} catch ( error ) {
console . error ( error ) ;
return [ extension _settings . sd . hr _upscaler ] ;
}
}
2023-10-07 18:30:06 +03:00
async function getVladRemoteUpscalers ( ) {
try {
const result = await fetch ( '/api/sd-next/upscalers' , {
method : 'POST' ,
headers : getRequestHeaders ( ) ,
body : JSON . stringify ( getSdRequestBody ( ) ) ,
} ) ;
if ( ! result . ok ) {
throw new Error ( 'SD.Next returned an error.' ) ;
}
const data = await result . json ( ) ;
return data ;
} catch ( error ) {
console . error ( error ) ;
return [ extension _settings . sd . hr _upscaler ] ;
}
}
2023-09-03 00:41:26 +03:00
async function updateAutoRemoteModel ( ) {
try {
const result = await fetch ( '/api/sd/set-model' , {
method : 'POST' ,
headers : getRequestHeaders ( ) ,
2023-10-07 18:30:06 +03:00
body : JSON . stringify ( { ... getSdRequestBody ( ) , model : extension _settings . sd . model } ) ,
2023-09-03 00:41:26 +03:00
} ) ;
if ( ! result . ok ) {
throw new Error ( 'SD WebUI returned an error.' ) ;
}
console . log ( 'Model successfully updated on SD WebUI remote.' ) ;
} catch ( error ) {
console . error ( error ) ;
toastr . error ( ` Could not update SD WebUI model: ${ error . message } ` ) ;
}
2023-07-20 20:32:15 +03:00
}
async function updateExtrasRemoteModel ( ) {
const url = new URL ( getApiUrl ( ) ) ;
url . pathname = '/api/image/model' ;
const getCurrentModelResult = await doExtrasFetch ( url , {
method : 'POST' ,
body : JSON . stringify ( { model : extension _settings . sd . model } ) ,
} ) ;
if ( getCurrentModelResult . ok ) {
console . log ( 'Model successfully updated on SD remote.' ) ;
}
}
async function loadSamplers ( ) {
$ ( '#sd_sampler' ) . empty ( ) ;
let samplers = [ ] ;
2023-09-03 00:41:26 +03:00
switch ( extension _settings . sd . source ) {
case sources . extras :
samplers = await loadExtrasSamplers ( ) ;
break ;
case sources . horde :
samplers = await loadHordeSamplers ( ) ;
break ;
case sources . auto :
samplers = await loadAutoSamplers ( ) ;
break ;
case sources . novel :
samplers = await loadNovelSamplers ( ) ;
break ;
2023-10-07 18:30:06 +03:00
case sources . vlad :
samplers = await loadVladSamplers ( ) ;
break ;
2023-11-06 21:47:00 +02:00
case sources . openai :
samplers = await loadOpenAiSamplers ( ) ;
break ;
2023-07-20 20:32:15 +03:00
}
for ( const sampler of samplers ) {
const option = document . createElement ( 'option' ) ;
option . innerText = sampler ;
option . value = sampler ;
option . selected = sampler === extension _settings . sd . sampler ;
$ ( '#sd_sampler' ) . append ( option ) ;
}
}
async function loadHordeSamplers ( ) {
2023-09-12 20:45:36 +03:00
const result = await fetch ( '/api/horde/sd-samplers' , {
2023-07-20 20:32:15 +03:00
method : 'POST' ,
headers : getRequestHeaders ( ) ,
} ) ;
if ( result . ok ) {
const data = await result . json ( ) ;
return data ;
}
return [ ] ;
}
async function loadExtrasSamplers ( ) {
if ( ! modules . includes ( 'sd' ) ) {
return [ ] ;
}
const url = new URL ( getApiUrl ( ) ) ;
url . pathname = '/api/image/samplers' ;
const result = await doExtrasFetch ( url ) ;
if ( result . ok ) {
const data = await result . json ( ) ;
return data . samplers ;
}
return [ ] ;
}
2023-09-03 00:41:26 +03:00
async function loadAutoSamplers ( ) {
if ( ! extension _settings . sd . auto _url ) {
return [ ] ;
}
try {
const result = await fetch ( '/api/sd/samplers' , {
method : 'POST' ,
headers : getRequestHeaders ( ) ,
2023-10-07 18:30:06 +03:00
body : JSON . stringify ( getSdRequestBody ( ) ) ,
2023-09-03 00:41:26 +03:00
} ) ;
if ( ! result . ok ) {
throw new Error ( 'SD WebUI returned an error.' ) ;
}
const data = await result . json ( ) ;
return data ;
} catch ( error ) {
return [ ] ;
}
}
2023-11-06 21:47:00 +02:00
async function loadOpenAiSamplers ( ) {
return [ 'N/A' ] ;
}
2023-10-07 18:30:06 +03:00
async function loadVladSamplers ( ) {
if ( ! extension _settings . sd . vlad _url ) {
return [ ] ;
}
try {
const result = await fetch ( '/api/sd/samplers' , {
method : 'POST' ,
headers : getRequestHeaders ( ) ,
body : JSON . stringify ( getSdRequestBody ( ) ) ,
} ) ;
if ( ! result . ok ) {
throw new Error ( 'SD.Next returned an error.' ) ;
}
const data = await result . json ( ) ;
return data ;
} catch ( error ) {
return [ ] ;
}
}
2023-09-03 00:41:26 +03:00
async function loadNovelSamplers ( ) {
if ( ! secret _state [ SECRET _KEYS . NOVEL ] ) {
2023-09-04 18:00:15 +03:00
console . debug ( 'NovelAI API key is not set.' ) ;
2023-09-03 00:41:26 +03:00
return [ ] ;
}
return [
'k_dpmpp_2m' ,
'k_dpmpp_sde' ,
'k_dpmpp_2s_ancestral' ,
'k_euler' ,
'k_euler_ancestral' ,
'k_dpm_fast' ,
'ddim' ,
] ;
}
2023-07-20 20:32:15 +03:00
async function loadModels ( ) {
$ ( '#sd_model' ) . empty ( ) ;
let models = [ ] ;
2023-09-03 00:41:26 +03:00
switch ( extension _settings . sd . source ) {
case sources . extras :
models = await loadExtrasModels ( ) ;
break ;
case sources . horde :
models = await loadHordeModels ( ) ;
break ;
case sources . auto :
models = await loadAutoModels ( ) ;
break ;
case sources . novel :
models = await loadNovelModels ( ) ;
break ;
2023-10-07 18:30:06 +03:00
case sources . vlad :
models = await loadVladModels ( ) ;
break ;
2023-11-06 21:47:00 +02:00
case sources . openai :
models = await loadOpenAiModels ( ) ;
break ;
2023-07-20 20:32:15 +03:00
}
for ( const model of models ) {
const option = document . createElement ( 'option' ) ;
option . innerText = model . text ;
option . value = model . value ;
option . selected = model . value === extension _settings . sd . model ;
$ ( '#sd_model' ) . append ( option ) ;
}
}
async function loadHordeModels ( ) {
2023-09-12 20:45:36 +03:00
const result = await fetch ( '/api/horde/sd-models' , {
2023-07-20 20:32:15 +03:00
method : 'POST' ,
headers : getRequestHeaders ( ) ,
} ) ;
if ( result . ok ) {
const data = await result . json ( ) ;
data . sort ( ( a , b ) => b . count - a . count ) ;
const models = data . map ( x => ( { value : x . name , text : ` ${ x . name } (ETA: ${ x . eta } s, Queue: ${ x . queued } , Workers: ${ x . count } ) ` } ) ) ;
return models ;
}
return [ ] ;
}
async function loadExtrasModels ( ) {
if ( ! modules . includes ( 'sd' ) ) {
return [ ] ;
}
const url = new URL ( getApiUrl ( ) ) ;
url . pathname = '/api/image/model' ;
const getCurrentModelResult = await doExtrasFetch ( url ) ;
if ( getCurrentModelResult . ok ) {
const data = await getCurrentModelResult . json ( ) ;
extension _settings . sd . model = data . model ;
}
url . pathname = '/api/image/models' ;
const getModelsResult = await doExtrasFetch ( url ) ;
if ( getModelsResult . ok ) {
const data = await getModelsResult . json ( ) ;
const view _models = data . models . map ( x => ( { value : x , text : x } ) ) ;
return view _models ;
}
return [ ] ;
}
2023-09-03 00:41:26 +03:00
async function loadAutoModels ( ) {
if ( ! extension _settings . sd . auto _url ) {
return [ ] ;
}
try {
const currentModel = await getAutoRemoteModel ( ) ;
if ( currentModel ) {
extension _settings . sd . model = currentModel ;
}
const result = await fetch ( '/api/sd/models' , {
method : 'POST' ,
headers : getRequestHeaders ( ) ,
2023-10-07 18:30:06 +03:00
body : JSON . stringify ( getSdRequestBody ( ) ) ,
2023-09-03 00:41:26 +03:00
} ) ;
if ( ! result . ok ) {
throw new Error ( 'SD WebUI returned an error.' ) ;
}
2023-09-03 14:56:02 +03:00
const upscalers = await getAutoRemoteUpscalers ( ) ;
if ( Array . isArray ( upscalers ) && upscalers . length > 0 ) {
$ ( '#sd_hr_upscaler' ) . empty ( ) ;
for ( const upscaler of upscalers ) {
const option = document . createElement ( 'option' ) ;
option . innerText = upscaler ;
option . value = upscaler ;
option . selected = upscaler === extension _settings . sd . hr _upscaler ;
$ ( '#sd_hr_upscaler' ) . append ( option ) ;
}
}
2023-09-03 00:41:26 +03:00
const data = await result . json ( ) ;
return data ;
} catch ( error ) {
return [ ] ;
}
}
2023-11-06 21:47:00 +02:00
async function loadOpenAiModels ( ) {
return [
{ value : 'dall-e-2' , text : 'DALL-E 2' } ,
{ value : 'dall-e-3' , text : 'DALL-E 3' } ,
] ;
}
2023-10-07 18:30:06 +03:00
async function loadVladModels ( ) {
if ( ! extension _settings . sd . vlad _url ) {
return [ ] ;
}
try {
const currentModel = await getAutoRemoteModel ( ) ;
if ( currentModel ) {
extension _settings . sd . model = currentModel ;
}
const result = await fetch ( '/api/sd/models' , {
method : 'POST' ,
headers : getRequestHeaders ( ) ,
body : JSON . stringify ( getSdRequestBody ( ) ) ,
} ) ;
if ( ! result . ok ) {
throw new Error ( 'SD WebUI returned an error.' ) ;
}
const upscalers = await getVladRemoteUpscalers ( ) ;
if ( Array . isArray ( upscalers ) && upscalers . length > 0 ) {
$ ( '#sd_hr_upscaler' ) . empty ( ) ;
for ( const upscaler of upscalers ) {
const option = document . createElement ( 'option' ) ;
option . innerText = upscaler ;
option . value = upscaler ;
option . selected = upscaler === extension _settings . sd . hr _upscaler ;
$ ( '#sd_hr_upscaler' ) . append ( option ) ;
}
}
const data = await result . json ( ) ;
return data ;
} catch ( error ) {
return [ ] ;
}
}
2023-09-03 00:41:26 +03:00
async function loadNovelModels ( ) {
if ( ! secret _state [ SECRET _KEYS . NOVEL ] ) {
2023-09-04 18:00:15 +03:00
console . debug ( 'NovelAI API key is not set.' ) ;
2023-09-03 00:41:26 +03:00
return [ ] ;
}
return [
2023-10-23 01:06:49 +03:00
{
value : 'nai-diffusion-2' ,
text : 'NAI Diffusion Anime V2' ,
} ,
2023-09-03 00:41:26 +03:00
{
value : 'nai-diffusion' ,
2023-10-23 01:06:49 +03:00
text : 'NAI Diffusion Anime V1 (Full)' ,
2023-09-03 00:41:26 +03:00
} ,
{
value : 'safe-diffusion' ,
2023-10-23 01:06:49 +03:00
text : 'NAI Diffusion Anime V1 (Curated)' ,
2023-09-03 00:41:26 +03:00
} ,
{
value : 'nai-diffusion-furry' ,
2023-10-23 01:06:49 +03:00
text : 'NAI Diffusion Furry' ,
2023-09-03 00:41:26 +03:00
} ,
] ;
}
2023-07-20 20:32:15 +03:00
function getGenerationType ( prompt ) {
for ( const [ key , values ] of Object . entries ( triggerWords ) ) {
for ( const value of values ) {
if ( value . toLowerCase ( ) === prompt . toLowerCase ( ) . trim ( ) ) {
return Number ( key ) ;
}
}
}
return generationMode . FREE ;
}
function getQuietPrompt ( mode , trigger ) {
if ( mode === generationMode . FREE ) {
return trigger ;
}
2023-08-20 12:29:20 +03:00
return stringFormat ( extension _settings . sd . prompts [ mode ] , trigger ) ;
2023-07-20 20:32:15 +03:00
}
function processReply ( str ) {
if ( ! str ) {
return '' ;
}
str = str . replaceAll ( '"' , '' )
str = str . replaceAll ( '“' , '' )
str = str . replaceAll ( '.' , ',' )
str = str . replaceAll ( '\n' , ', ' )
str = str . replace ( /[^a-zA-Z0-9,:()]+/g , ' ' ) // Replace everything except alphanumeric characters and commas with spaces
str = str . replace ( /\s+/g , ' ' ) ; // Collapse multiple whitespaces into one
str = str . trim ( ) ;
str = str
. split ( ',' ) // list split by commas
. map ( x => x . trim ( ) ) // trim each entry
. filter ( x => x ) // remove empty entries
. join ( ', ' ) ; // join it back with proper spacing
return str ;
}
function getRawLastMessage ( ) {
2023-08-29 18:33:30 +03:00
const getLastUsableMessage = ( ) => {
for ( const message of context . chat . slice ( ) . reverse ( ) ) {
if ( message . is _system ) {
continue ;
}
return message . mes ;
}
toastr . warning ( 'No usable messages found.' , 'Stable Diffusion' ) ;
throw new Error ( 'No usable messages found.' ) ;
}
2023-07-20 20:32:15 +03:00
const context = getContext ( ) ;
2023-08-29 18:33:30 +03:00
const lastMessage = getLastUsableMessage ( ) ,
2023-07-20 20:32:15 +03:00
characterDescription = context . characters [ context . characterId ] . description ,
situation = context . characters [ context . characterId ] . scenario ;
return ` (( ${ processReply ( lastMessage ) } )), ( ${ processReply ( situation ) } :0.7), ( ${ processReply ( characterDescription ) } :0.5) `
}
async function generatePicture ( _ , trigger , message , callback ) {
if ( ! trigger || trigger . trim ( ) . length === 0 ) {
console . log ( 'Trigger word empty, aborting' ) ;
return ;
}
2023-09-03 00:41:26 +03:00
if ( ! isValidState ( ) ) {
2023-07-20 20:32:15 +03:00
toastr . warning ( "Extensions API is not connected or doesn't provide SD module. Enable Stable Horde to generate images." ) ;
return ;
}
extension _settings . sd . sampler = $ ( '#sd_sampler' ) . find ( ':selected' ) . val ( ) ;
extension _settings . sd . model = $ ( '#sd_model' ) . find ( ':selected' ) . val ( ) ;
trigger = trigger . trim ( ) ;
const generationType = getGenerationType ( trigger ) ;
console . log ( 'Generation mode' , generationType , 'triggered with' , trigger ) ;
const quiet _prompt = getQuietPrompt ( generationType , trigger ) ;
const context = getContext ( ) ;
2023-08-19 23:01:09 -04:00
// if context.characterId is not null, then we get context.characters[context.characterId].avatar, else we get groupId and context.groups[groupId].id
// sadly, groups is not an array, but is a dict with keys being index numbers, so we have to filter it
2023-10-07 18:30:06 +03:00
const characterName = context . characterId ? context . characters [ context . characterId ] . name : context . groups [ Object . keys ( context . groups ) . filter ( x => context . groups [ x ] . id === context . groupId ) [ 0 ] ] ? . id ? . toString ( ) ;
2023-08-19 23:01:09 -04:00
2023-09-04 12:18:37 +03:00
if ( generationType == generationMode . BACKGROUND ) {
2023-08-18 00:41:21 +03:00
const callbackOriginal = callback ;
2023-11-01 22:58:59 +02:00
callback = async function ( prompt , imagePath , generationType ) {
2023-10-21 17:43:25 +03:00
const imgUrl = ` url(" ${ encodeURI ( imagePath ) } ") ` ;
eventSource . emit ( event _types . FORCE _SET _BACKGROUND , { url : imgUrl , path : imagePath } ) ;
2023-08-18 00:41:21 +03:00
if ( typeof callbackOriginal === 'function' ) {
2023-11-01 22:58:59 +02:00
callbackOriginal ( prompt , imagePath , generationType ) ;
2023-08-18 00:41:21 +03:00
} else {
2023-11-01 22:58:59 +02:00
sendMessage ( prompt , imagePath , generationType ) ;
2023-08-18 00:41:21 +03:00
}
}
2023-07-20 20:32:15 +03:00
}
2023-11-01 22:58:59 +02:00
const dimensions = setTypeSpecificDimensions ( generationType ) ;
2023-07-20 20:32:15 +03:00
try {
const prompt = await getPrompt ( generationType , message , trigger , quiet _prompt ) ;
console . log ( 'Processed Stable Diffusion prompt:' , prompt ) ;
context . deactivateSendButtons ( ) ;
hideSwipeButtons ( ) ;
2023-08-19 23:01:09 -04:00
await sendGenerationRequest ( generationType , prompt , characterName , callback ) ;
2023-07-20 20:32:15 +03:00
} catch ( err ) {
console . trace ( err ) ;
throw new Error ( 'SD prompt text generation failed.' )
}
finally {
2023-11-01 22:58:59 +02:00
restoreOriginalDimensions ( dimensions ) ;
2023-07-20 20:32:15 +03:00
context . activateSendButtons ( ) ;
showSwipeButtons ( ) ;
}
}
2023-11-01 22:58:59 +02:00
function setTypeSpecificDimensions ( generationType ) {
const prevSDHeight = extension _settings . sd . height ;
const prevSDWidth = extension _settings . sd . width ;
const aspectRatio = extension _settings . sd . width / extension _settings . sd . height ;
// Face images are always portrait (pun intended)
if ( generationType == generationMode . FACE && aspectRatio >= 1 ) {
// Round to nearest multiple of 64
extension _settings . sd . height = Math . round ( extension _settings . sd . width * 1.5 / 64 ) * 64 ;
}
if ( generationType == generationMode . BACKGROUND ) {
// Background images are always landscape
if ( aspectRatio <= 1 ) {
// Round to nearest multiple of 64
extension _settings . sd . width = Math . round ( extension _settings . sd . height * 1.8 / 64 ) * 64 ;
}
}
return { height : prevSDHeight , width : prevSDWidth } ;
}
function restoreOriginalDimensions ( savedParams ) {
extension _settings . sd . height = savedParams . height ;
extension _settings . sd . width = savedParams . width ;
}
2023-07-20 20:32:15 +03:00
async function getPrompt ( generationType , message , trigger , quiet _prompt ) {
let prompt ;
switch ( generationType ) {
case generationMode . RAW _LAST :
prompt = message || getRawLastMessage ( ) ;
break ;
case generationMode . FREE :
prompt = trigger . trim ( ) ;
break ;
default :
prompt = await generatePrompt ( quiet _prompt ) ;
break ;
}
2023-07-22 20:32:58 +03:00
if ( generationType !== generationMode . FREE ) {
2023-10-20 15:43:55 +03:00
prompt = await refinePrompt ( prompt , true ) ;
2023-07-22 20:32:58 +03:00
}
2023-07-20 20:32:15 +03:00
return prompt ;
}
async function generatePrompt ( quiet _prompt ) {
2023-10-22 00:10:48 +03:00
const reply = await generateQuietPrompt ( quiet _prompt , false , false ) ;
2023-07-22 21:12:23 +03:00
return processReply ( reply ) ;
2023-07-20 20:32:15 +03:00
}
2023-08-20 12:37:38 +03:00
async function sendGenerationRequest ( generationType , prompt , characterName = null , callback ) {
2023-11-01 22:58:59 +02:00
const prefix = ( generationType !== generationMode . BACKGROUND && generationType !== generationMode . FREE )
2023-08-18 00:41:21 +03:00
? combinePrefixes ( extension _settings . sd . prompt _prefix , getCharacterPrefix ( ) )
: extension _settings . sd . prompt _prefix ;
2023-10-22 00:10:48 +03:00
const prefixedPrompt = combinePrefixes ( prefix , prompt , '{prompt}' ) ;
2023-09-03 00:41:26 +03:00
let result = { format : '' , data : '' } ;
const currentChatId = getCurrentChatId ( ) ;
try {
switch ( extension _settings . sd . source ) {
case sources . extras :
result = await generateExtrasImage ( prefixedPrompt ) ;
break ;
case sources . horde :
result = await generateHordeImage ( prefixedPrompt ) ;
break ;
2023-10-07 18:30:06 +03:00
case sources . vlad :
result = await generateAutoImage ( prefixedPrompt ) ;
break ;
2023-09-03 00:41:26 +03:00
case sources . auto :
result = await generateAutoImage ( prefixedPrompt ) ;
break ;
case sources . novel :
result = await generateNovelImage ( prefixedPrompt ) ;
break ;
2023-11-06 21:47:00 +02:00
case sources . openai :
result = await generateOpenAiImage ( prefixedPrompt ) ;
break ;
2023-09-03 00:41:26 +03:00
}
if ( ! result . data ) {
2023-11-06 21:47:00 +02:00
throw new Error ( 'Endpoint did not return image data.' ) ;
2023-09-03 00:41:26 +03:00
}
} catch ( err ) {
2023-11-06 21:47:00 +02:00
console . error ( err ) ;
toastr . error ( 'Image generation failed. Please try again.' + '\n\n' + String ( err ) , 'Stable Diffusion' ) ;
2023-09-03 00:41:26 +03:00
return ;
}
if ( currentChatId !== getCurrentChatId ( ) ) {
console . warn ( 'Chat changed, aborting SD result saving' ) ;
toastr . warning ( 'Chat changed, generated image discarded.' , 'Stable Diffusion' ) ;
return ;
2023-07-20 20:32:15 +03:00
}
2023-09-03 00:41:26 +03:00
const filename = ` ${ characterName } _ ${ humanizedDateTime ( ) } ` ;
const base64Image = await saveBase64AsFile ( result . data , characterName , filename , result . format ) ;
2023-11-01 22:58:59 +02:00
callback ? callback ( prompt , base64Image , generationType ) : sendMessage ( prompt , base64Image , generationType ) ;
2023-07-20 20:32:15 +03:00
}
2023-08-20 00:15:57 -04:00
/ * *
2023-09-03 00:41:26 +03:00
* Generates an "extras" image using a provided prompt and other settings .
2023-08-20 12:16:29 +03:00
*
2023-08-20 00:33:37 -04:00
* @ param { string } prompt - The main instruction used to guide the image generation .
2023-09-03 00:41:26 +03:00
* @ returns { Promise < { format : string , data : string } > } - A promise that resolves when the image generation and processing are complete .
2023-08-20 00:15:57 -04:00
* /
2023-09-03 00:41:26 +03:00
async function generateExtrasImage ( prompt ) {
2023-07-20 20:32:15 +03:00
const url = new URL ( getApiUrl ( ) ) ;
url . pathname = '/api/image' ;
const result = await doExtrasFetch ( url , {
method : 'POST' ,
2023-09-03 14:04:53 +03:00
headers : {
'Content-Type' : 'application/json' ,
} ,
2023-07-20 20:32:15 +03:00
body : JSON . stringify ( {
prompt : prompt ,
sampler : extension _settings . sd . sampler ,
steps : extension _settings . sd . steps ,
scale : extension _settings . sd . scale ,
width : extension _settings . sd . width ,
height : extension _settings . sd . height ,
negative _prompt : extension _settings . sd . negative _prompt ,
restore _faces : ! ! extension _settings . sd . restore _faces ,
enable _hr : ! ! extension _settings . sd . enable _hr ,
karras : ! ! extension _settings . sd . horde _karras ,
2023-09-03 14:56:02 +03:00
hr _upscaler : extension _settings . sd . hr _upscaler ,
hr _scale : extension _settings . sd . hr _scale ,
denoising _strength : extension _settings . sd . denoising _strength ,
hr _second _pass _steps : extension _settings . sd . hr _second _pass _steps ,
2023-07-20 20:32:15 +03:00
} ) ,
} ) ;
if ( result . ok ) {
const data = await result . json ( ) ;
2023-09-03 00:41:26 +03:00
return { format : 'jpg' , data : data . image } ;
2023-07-20 20:32:15 +03:00
} else {
2023-11-06 21:47:00 +02:00
const text = await result . text ( ) ;
throw new Error ( text ) ;
2023-07-20 20:32:15 +03:00
}
}
2023-08-20 00:15:57 -04:00
/ * *
2023-09-03 00:41:26 +03:00
* Generates a "horde" image using the provided prompt and configuration settings .
2023-08-20 12:16:29 +03:00
*
2023-08-20 00:33:37 -04:00
* @ param { string } prompt - The main instruction used to guide the image generation .
2023-09-03 00:41:26 +03:00
* @ returns { Promise < { format : string , data : string } > } - A promise that resolves when the image generation and processing are complete .
2023-08-20 00:15:57 -04:00
* /
2023-09-03 00:41:26 +03:00
async function generateHordeImage ( prompt ) {
2023-09-12 20:45:36 +03:00
const result = await fetch ( '/api/horde/generate-image' , {
2023-07-20 20:32:15 +03:00
method : 'POST' ,
headers : getRequestHeaders ( ) ,
body : JSON . stringify ( {
prompt : prompt ,
sampler : extension _settings . sd . sampler ,
steps : extension _settings . sd . steps ,
scale : extension _settings . sd . scale ,
width : extension _settings . sd . width ,
height : extension _settings . sd . height ,
negative _prompt : extension _settings . sd . negative _prompt ,
model : extension _settings . sd . model ,
nsfw : extension _settings . sd . horde _nsfw ,
restore _faces : ! ! extension _settings . sd . restore _faces ,
enable _hr : ! ! extension _settings . sd . enable _hr ,
2023-10-08 22:29:33 +03:00
sanitize : ! ! extension _settings . sd . horde _sanitize ,
2023-07-20 20:32:15 +03:00
} ) ,
} ) ;
if ( result . ok ) {
const data = await result . text ( ) ;
2023-09-03 00:41:26 +03:00
return { format : 'webp' , data : data } ;
2023-07-20 20:32:15 +03:00
} else {
2023-11-06 21:47:00 +02:00
const text = await result . text ( ) ;
throw new Error ( text ) ;
2023-09-03 00:41:26 +03:00
}
}
/ * *
* Generates an image in SD WebUI API using the provided prompt and configuration settings .
*
* @ param { string } prompt - The main instruction used to guide the image generation .
* @ returns { Promise < { format : string , data : string } > } - A promise that resolves when the image generation and processing are complete .
* /
async function generateAutoImage ( prompt ) {
const result = await fetch ( '/api/sd/generate' , {
method : 'POST' ,
headers : getRequestHeaders ( ) ,
body : JSON . stringify ( {
2023-10-07 18:30:06 +03:00
... getSdRequestBody ( ) ,
2023-09-03 00:41:26 +03:00
prompt : prompt ,
negative _prompt : extension _settings . sd . negative _prompt ,
sampler _name : extension _settings . sd . sampler ,
steps : extension _settings . sd . steps ,
cfg _scale : extension _settings . sd . scale ,
width : extension _settings . sd . width ,
height : extension _settings . sd . height ,
restore _faces : ! ! extension _settings . sd . restore _faces ,
2023-09-03 14:56:02 +03:00
enable _hr : ! ! extension _settings . sd . enable _hr ,
hr _upscaler : extension _settings . sd . hr _upscaler ,
hr _scale : extension _settings . sd . hr _scale ,
denoising _strength : extension _settings . sd . denoising _strength ,
hr _second _pass _steps : extension _settings . sd . hr _second _pass _steps ,
2023-09-03 00:41:26 +03:00
// Ensure generated img is saved to disk
save _images : true ,
send _images : true ,
do _not _save _grid : false ,
do _not _save _samples : false ,
} ) ,
} ) ;
if ( result . ok ) {
const data = await result . json ( ) ;
return { format : 'png' , data : data . images [ 0 ] } ;
} else {
2023-11-06 21:47:00 +02:00
const text = await result . text ( ) ;
throw new Error ( text ) ;
2023-09-03 00:41:26 +03:00
}
}
/ * *
* Generates an image in NovelAI API using the provided prompt and configuration settings .
*
* @ param { string } prompt - The main instruction used to guide the image generation .
* @ returns { Promise < { format : string , data : string } > } - A promise that resolves when the image generation and processing are complete .
* /
async function generateNovelImage ( prompt ) {
2023-09-04 18:00:15 +03:00
const { steps , width , height } = getNovelParams ( ) ;
2023-09-03 00:41:26 +03:00
const result = await fetch ( '/api/novelai/generate-image' , {
method : 'POST' ,
headers : getRequestHeaders ( ) ,
body : JSON . stringify ( {
prompt : prompt ,
model : extension _settings . sd . model ,
sampler : extension _settings . sd . sampler ,
2023-09-04 18:00:15 +03:00
steps : steps ,
2023-09-03 00:41:26 +03:00
scale : extension _settings . sd . scale ,
2023-09-04 18:00:15 +03:00
width : width ,
height : height ,
2023-09-03 00:41:26 +03:00
negative _prompt : extension _settings . sd . negative _prompt ,
2023-09-04 18:00:15 +03:00
upscale _ratio : extension _settings . sd . novel _upscale _ratio ,
2023-09-03 00:41:26 +03:00
} ) ,
} ) ;
if ( result . ok ) {
const data = await result . text ( ) ;
return { format : 'png' , data : data } ;
} else {
2023-11-06 21:47:00 +02:00
const text = await result . text ( ) ;
throw new Error ( text ) ;
2023-07-20 20:32:15 +03:00
}
}
2023-09-04 18:00:15 +03:00
/ * *
* Adjusts extension parameters for NovelAI . Applies Anlas guard if needed .
* @ returns { { steps : number , width : number , height : number } } - A tuple of parameters for NovelAI API .
* /
function getNovelParams ( ) {
let steps = extension _settings . sd . steps ;
let width = extension _settings . sd . width ;
let height = extension _settings . sd . height ;
2023-10-23 10:41:39 +03:00
// Don't apply Anlas guard if it's disabled.
2023-09-04 18:00:15 +03:00
if ( ! extension _settings . sd . novel _anlas _guard ) {
return { steps , width , height } ;
}
const MAX _STEPS = 28 ;
2023-10-23 10:41:39 +03:00
const MAX _PIXELS = 1024 * 1024 ;
2023-09-04 18:00:15 +03:00
if ( width * height > MAX _PIXELS ) {
const ratio = Math . sqrt ( MAX _PIXELS / ( width * height ) ) ;
// Calculate new width and height while maintaining aspect ratio.
var newWidth = Math . round ( width * ratio ) ;
var newHeight = Math . round ( height * ratio ) ;
// Ensure new dimensions are multiples of 64. If not, reduce accordingly.
if ( newWidth % 64 !== 0 ) {
newWidth = newWidth - newWidth % 64 ;
}
if ( newHeight % 64 !== 0 ) {
newHeight = newHeight - newHeight % 64 ;
}
// If total pixel count after rounding still exceeds MAX_PIXELS, decrease dimension size by 64 accordingly.
while ( newWidth * newHeight > MAX _PIXELS ) {
if ( newWidth > newHeight ) {
newWidth -= 64 ;
} else {
newHeight -= 64 ;
}
}
console . log ( ` Anlas Guard: Image size ( ${ width } x ${ height } ) > ${ MAX _PIXELS } , reducing size to ${ newWidth } x ${ newHeight } ` ) ;
width = newWidth ;
height = newHeight ;
}
if ( steps > MAX _STEPS ) {
console . log ( ` Anlas Guard: Steps ( ${ steps } ) > ${ MAX _STEPS } , reducing steps to ${ MAX _STEPS } ` ) ;
steps = MAX _STEPS ;
}
return { steps , width , height } ;
}
2023-11-06 21:47:00 +02:00
async function generateOpenAiImage ( prompt ) {
const dalle2PromptLimit = 1000 ;
const dalle3PromptLimit = 4000 ;
const isDalle2 = extension _settings . sd . model === 'dall-e-2' ;
const isDalle3 = extension _settings . sd . model === 'dall-e-3' ;
if ( isDalle2 && prompt . length > dalle2PromptLimit ) {
prompt = prompt . substring ( 0 , dalle2PromptLimit ) ;
}
if ( isDalle3 && prompt . length > dalle3PromptLimit ) {
prompt = prompt . substring ( 0 , dalle3PromptLimit ) ;
}
let width = 1024 ;
let height = 1024 ;
let aspectRatio = extension _settings . sd . width / extension _settings . sd . height ;
if ( isDalle3 && aspectRatio < 1 ) {
height = 1792 ;
}
if ( isDalle3 && aspectRatio > 1 ) {
width = 1792 ;
}
if ( isDalle2 && ( extension _settings . sd . width <= 512 && extension _settings . sd . height <= 512 ) ) {
width = 512 ;
height = 512 ;
}
const result = await fetch ( '/api/openai/generate-image' , {
method : 'POST' ,
headers : getRequestHeaders ( ) ,
body : JSON . stringify ( {
prompt : prompt ,
model : extension _settings . sd . model ,
size : ` ${ width } x ${ height } ` ,
n : 1 ,
quality : isDalle3 ? extension _settings . sd . openai _quality : undefined ,
style : isDalle3 ? extension _settings . sd . openai _style : undefined ,
response _format : 'b64_json' ,
} ) ,
} ) ;
if ( result . ok ) {
const data = await result . json ( ) ;
return { format : 'png' , data : data ? . data [ 0 ] ? . b64 _json } ;
} else {
const text = await result . text ( ) ;
throw new Error ( text ) ;
}
}
2023-11-01 22:58:59 +02:00
async function sendMessage ( prompt , image , generationType ) {
2023-07-20 20:32:15 +03:00
const context = getContext ( ) ;
const messageText = ` [ ${ context . name2 } sends a picture that contains: ${ prompt } ] ` ;
const message = {
name : context . groupId ? systemUserName : context . name2 ,
is _user : false ,
2023-08-18 00:41:21 +03:00
is _system : true ,
2023-08-22 18:32:18 +03:00
send _date : getMessageTimeStamp ( ) ,
2023-07-20 20:32:15 +03:00
mes : context . groupId ? p ( messageText ) : messageText ,
extra : {
image : image ,
title : prompt ,
2023-11-01 22:58:59 +02:00
generationType : generationType ,
2023-07-20 20:32:15 +03:00
} ,
} ;
context . chat . push ( message ) ;
context . addOneMessage ( message ) ;
context . saveChat ( ) ;
}
function addSDGenButtons ( ) {
const buttonHtml = `
< div id = "sd_gen" class = "list-group-item flex-container flexGap5" >
< div class = "fa-solid fa-paintbrush extensionsMenuExtensionButton" title = "Trigger Stable Diffusion" / > < / d i v >
Stable Diffusion
< / d i v >
` ;
const waitButtonHtml = `
< div id = "sd_gen_wait" class = "fa-solid fa-hourglass-half" / > < / d i v >
`
const dropdownHtml = `
< div id = "sd_dropdown" >
< ul class = "list-group" >
< span > Send me a picture of : < / s p a n >
< li class = "list-group-item" id = "sd_you" data - value = "you" > Yourself < / l i >
< li class = "list-group-item" id = "sd_face" data - value = "face" > Your Face < / l i >
< li class = "list-group-item" id = "sd_me" data - value = "me" > Me < / l i >
< li class = "list-group-item" id = "sd_world" data - value = "world" > The Whole Story < / l i >
< li class = "list-group-item" id = "sd_last" data - value = "last" > The Last Message < / l i >
< li class = "list-group-item" id = "sd_raw_last" data - value = "raw_last" > Raw Last Message < / l i >
2023-08-18 00:41:21 +03:00
< li class = "list-group-item" id = "sd_background" data - value = "background" > Background < / l i >
2023-07-20 20:32:15 +03:00
< / u l >
< / d i v > ` ;
$ ( '#extensionsMenu' ) . prepend ( buttonHtml ) ;
$ ( '#extensionsMenu' ) . prepend ( waitButtonHtml ) ;
$ ( document . body ) . append ( dropdownHtml ) ;
const messageButton = $ ( '.sd_message_gen' ) ;
const button = $ ( '#sd_gen' ) ;
const waitButton = $ ( "#sd_gen_wait" ) ;
const dropdown = $ ( '#sd_dropdown' ) ;
waitButton . hide ( ) ;
dropdown . hide ( ) ;
button . hide ( ) ;
messageButton . hide ( ) ;
let popper = Popper . createPopper ( button . get ( 0 ) , dropdown . get ( 0 ) , {
placement : 'top' ,
} ) ;
$ ( document ) . on ( 'click' , '.sd_message_gen' , sdMessageButton ) ;
$ ( document ) . on ( 'click touchend' , function ( e ) {
const target = $ ( e . target ) ;
if ( target . is ( dropdown ) ) return ;
2023-08-24 01:37:44 +03:00
if ( target . is ( button ) && ! dropdown . is ( ":visible" ) && $ ( "#send_but" ) . is ( ":visible" ) ) {
2023-07-20 20:32:15 +03:00
e . preventDefault ( ) ;
dropdown . fadeIn ( 250 ) ;
popper . update ( ) ;
} else {
dropdown . fadeOut ( 250 ) ;
}
} ) ;
}
2023-09-03 00:41:26 +03:00
function isValidState ( ) {
switch ( extension _settings . sd . source ) {
case sources . extras :
return modules . includes ( 'sd' ) ;
case sources . horde :
return true ;
case sources . auto :
return ! ! extension _settings . sd . auto _url ;
2023-10-07 18:30:06 +03:00
case sources . vlad :
return ! ! extension _settings . sd . vlad _url ;
2023-09-03 00:41:26 +03:00
case sources . novel :
return secret _state [ SECRET _KEYS . NOVEL ] ;
2023-11-06 21:47:00 +02:00
case sources . openai :
return secret _state [ SECRET _KEYS . OPENAI ] ;
2023-09-03 00:41:26 +03:00
}
2023-07-20 20:32:15 +03:00
}
async function moduleWorker ( ) {
2023-09-03 00:41:26 +03:00
if ( isValidState ( ) ) {
2023-07-20 20:32:15 +03:00
$ ( '#sd_gen' ) . show ( ) ;
$ ( '.sd_message_gen' ) . show ( ) ;
}
else {
$ ( '#sd_gen' ) . hide ( ) ;
$ ( '.sd_message_gen' ) . hide ( ) ;
}
}
addSDGenButtons ( ) ;
setInterval ( moduleWorker , UPDATE _INTERVAL ) ;
async function sdMessageButton ( e ) {
function setBusyIcon ( isBusy ) {
$icon . toggleClass ( 'fa-paintbrush' , ! isBusy ) ;
$icon . toggleClass ( busyClass , isBusy ) ;
}
const busyClass = 'fa-hourglass' ;
const context = getContext ( ) ;
const $icon = $ ( e . currentTarget ) ;
const $mes = $icon . closest ( '.mes' ) ;
const message _id = $mes . attr ( 'mesid' ) ;
const message = context . chat [ message _id ] ;
const characterName = message ? . name || context . name2 ;
2023-10-07 18:30:06 +03:00
const characterFileName = context . characterId ? context . characters [ context . characterId ] . name : context . groups [ Object . keys ( context . groups ) . filter ( x => context . groups [ x ] . id === context . groupId ) [ 0 ] ] ? . id ? . toString ( ) ;
2023-08-20 12:29:20 +03:00
const messageText = message ? . mes ;
2023-07-20 20:32:15 +03:00
const hasSavedImage = message ? . extra ? . image && message ? . extra ? . title ;
if ( $icon . hasClass ( busyClass ) ) {
console . log ( 'Previous image is still being generated...' ) ;
return ;
}
2023-11-01 22:58:59 +02:00
let dimensions = null ;
2023-07-20 20:32:15 +03:00
try {
setBusyIcon ( true ) ;
if ( hasSavedImage ) {
2023-10-20 15:43:55 +03:00
const prompt = await refinePrompt ( message . extra . title , false ) ;
2023-07-22 20:32:58 +03:00
message . extra . title = prompt ;
2023-11-01 22:58:59 +02:00
const generationType = message ? . extra ? . generationType ? ? generationMode . FREE ;
2023-07-20 20:32:15 +03:00
console . log ( 'Regenerating an image, using existing prompt:' , prompt ) ;
2023-11-01 22:58:59 +02:00
dimensions = setTypeSpecificDimensions ( generationType ) ;
await sendGenerationRequest ( generationType , prompt , characterFileName , saveGeneratedImage ) ;
2023-07-20 20:32:15 +03:00
}
else {
console . log ( "doing /sd raw last" ) ;
await generatePicture ( 'sd' , 'raw_last' , ` ${ characterName } said: ${ messageText } ` , saveGeneratedImage ) ;
}
}
catch ( error ) {
console . error ( 'Could not generate inline image: ' , error ) ;
}
finally {
setBusyIcon ( false ) ;
2023-11-01 22:58:59 +02:00
if ( dimensions ) {
restoreOriginalDimensions ( dimensions ) ;
}
2023-07-20 20:32:15 +03:00
}
2023-11-01 22:58:59 +02:00
function saveGeneratedImage ( prompt , image , generationType ) {
2023-07-20 20:32:15 +03:00
// Some message sources may not create the extra object
if ( typeof message . extra !== 'object' ) {
message . extra = { } ;
}
// If already contains an image and it's not inline - leave it as is
message . extra . inline _image = message . extra . image && ! message . extra . inline _image ? false : true ;
message . extra . image = image ;
message . extra . title = prompt ;
2023-11-01 22:58:59 +02:00
message . extra . generationType = generationType ;
2023-07-20 20:32:15 +03:00
appendImageToMessage ( message , $mes ) ;
context . saveChat ( ) ;
}
} ;
$ ( "#sd_dropdown [id]" ) . on ( "click" , function ( ) {
2023-08-18 00:41:21 +03:00
const id = $ ( this ) . attr ( "id" ) ;
const idParamMap = {
"sd_you" : "you" ,
"sd_face" : "face" ,
"sd_me" : "me" ,
"sd_world" : "scene" ,
"sd_last" : "last" ,
"sd_raw_last" : "raw_last" ,
"sd_background" : "background"
} ;
2023-07-20 20:32:15 +03:00
2023-08-18 00:41:21 +03:00
const param = idParamMap [ id ] ;
2023-07-20 20:32:15 +03:00
2023-08-18 00:41:21 +03:00
if ( param ) {
console . log ( "doing /sd " + param )
generatePicture ( 'sd' , param ) ;
2023-07-20 20:32:15 +03:00
}
} ) ;
jQuery ( async ( ) => {
getContext ( ) . registerSlashCommand ( 'sd' , generatePicture , [ ] , helpString , true , true ) ;
2023-09-03 00:41:26 +03:00
$ ( '#extensions_settings' ) . append ( renderExtensionTemplate ( 'stable-diffusion' , 'settings' , defaultSettings ) ) ;
$ ( '#sd_source' ) . on ( 'change' , onSourceChange ) ;
2023-07-20 20:32:15 +03:00
$ ( '#sd_scale' ) . on ( 'input' , onScaleInput ) ;
$ ( '#sd_steps' ) . on ( 'input' , onStepsInput ) ;
$ ( '#sd_model' ) . on ( 'change' , onModelChange ) ;
$ ( '#sd_sampler' ) . on ( 'change' , onSamplerChange ) ;
$ ( '#sd_prompt_prefix' ) . on ( 'input' , onPromptPrefixInput ) ;
$ ( '#sd_negative_prompt' ) . on ( 'input' , onNegativePromptInput ) ;
$ ( '#sd_width' ) . on ( 'input' , onWidthInput ) ;
$ ( '#sd_height' ) . on ( 'input' , onHeightInput ) ;
$ ( '#sd_horde_nsfw' ) . on ( 'input' , onHordeNsfwInput ) ;
$ ( '#sd_horde_karras' ) . on ( 'input' , onHordeKarrasInput ) ;
2023-10-08 22:29:33 +03:00
$ ( '#sd_horde_sanitize' ) . on ( 'input' , onHordeSanitizeInput ) ;
2023-07-20 20:32:15 +03:00
$ ( '#sd_restore_faces' ) . on ( 'input' , onRestoreFacesInput ) ;
$ ( '#sd_enable_hr' ) . on ( 'input' , onHighResFixInput ) ;
$ ( '#sd_refine_mode' ) . on ( 'input' , onRefineModeInput ) ;
2023-07-22 23:57:48 +03:00
$ ( '#sd_character_prompt' ) . on ( 'input' , onCharacterPromptInput ) ;
2023-09-03 00:41:26 +03:00
$ ( '#sd_auto_validate' ) . on ( 'click' , validateAutoUrl ) ;
$ ( '#sd_auto_url' ) . on ( 'input' , onAutoUrlInput ) ;
2023-09-03 01:19:31 +03:00
$ ( '#sd_auto_auth' ) . on ( 'input' , onAutoAuthInput ) ;
2023-10-07 18:30:06 +03:00
$ ( '#sd_vlad_validate' ) . on ( 'click' , validateVladUrl ) ;
$ ( '#sd_vlad_url' ) . on ( 'input' , onVladUrlInput ) ;
$ ( '#sd_vlad_auth' ) . on ( 'input' , onVladAuthInput ) ;
2023-09-03 14:56:02 +03:00
$ ( '#sd_hr_upscaler' ) . on ( 'change' , onHrUpscalerChange ) ;
$ ( '#sd_hr_scale' ) . on ( 'input' , onHrScaleInput ) ;
$ ( '#sd_denoising_strength' ) . on ( 'input' , onDenoisingStrengthInput ) ;
$ ( '#sd_hr_second_pass_steps' ) . on ( 'input' , onHrSecondPassStepsInput ) ;
2023-09-04 18:00:15 +03:00
$ ( '#sd_novel_upscale_ratio' ) . on ( 'input' , onNovelUpscaleRatioInput ) ;
$ ( '#sd_novel_anlas_guard' ) . on ( 'input' , onNovelAnlasGuardInput ) ;
$ ( '#sd_novel_view_anlas' ) . on ( 'click' , onViewAnlasClick ) ;
2023-10-20 15:03:26 +03:00
$ ( '#sd_expand' ) . on ( 'input' , onExpandInput ) ;
2023-10-22 00:10:48 +03:00
$ ( '#sd_style' ) . on ( 'change' , onStyleSelect ) ;
$ ( '#sd_save_style' ) . on ( 'click' , onSaveStyleClick ) ;
2023-07-22 23:57:48 +03:00
$ ( '#sd_character_prompt_block' ) . hide ( ) ;
2023-11-01 22:58:59 +02:00
$ ( '#sd_interactive_mode' ) . on ( 'input' , onInteractiveModeInput ) ;
2023-11-06 21:47:00 +02:00
$ ( '#sd_openai_style' ) . on ( 'change' , onOpenAiStyleSelect ) ;
$ ( '#sd_openai_quality' ) . on ( 'change' , onOpenAiQualitySelect ) ;
2023-07-20 20:32:15 +03:00
$ ( '.sd_settings .inline-drawer-toggle' ) . on ( 'click' , function ( ) {
initScrollHeight ( $ ( "#sd_prompt_prefix" ) ) ;
initScrollHeight ( $ ( "#sd_negative_prompt" ) ) ;
2023-07-23 00:28:23 +03:00
initScrollHeight ( $ ( "#sd_character_prompt" ) ) ;
2023-07-20 20:32:15 +03:00
} )
eventSource . on ( event _types . EXTRAS _CONNECTED , async ( ) => {
await Promise . all ( [ loadSamplers ( ) , loadModels ( ) ] ) ;
} ) ;
2023-07-22 23:57:48 +03:00
eventSource . on ( event _types . CHAT _CHANGED , onChatChanged ) ;
2023-07-20 20:32:15 +03:00
await loadSettings ( ) ;
$ ( 'body' ) . addClass ( 'sd' ) ;
} ) ;