updated lite, added autorope config based on trained ctxlen, hotfix for falcon gpu broken

This commit is contained in:
Concedo 2023-08-30 16:50:55 +08:00
parent 89495c0716
commit d4c22a8b02
4 changed files with 654 additions and 74 deletions

View file

@ -328,7 +328,7 @@ static std::string FileFormatTokenizeID(int id, FileFormat file_format)
{ {
return std::string(llama_v3_token_to_str(llama_ctx_v3, id)); return std::string(llama_v3_token_to_str(llama_ctx_v3, id));
} }
else if( file_format == FileFormat::GGUF_LLAMA) else if(file_format == FileFormat::GGUF_LLAMA || file_format==FileFormat::GGUF_FALCON)
{ {
return std::string(llama_token_to_str(llama_ctx_v4, id)); return std::string(llama_token_to_str(llama_ctx_v4, id));
} }
@ -340,7 +340,7 @@ static std::string FileFormatTokenizeID(int id, FileFormat file_format)
static void TokenizeString(const std::string & str_to_tokenize, std::vector<int> & output_tokens, FileFormat file_format) static void TokenizeString(const std::string & str_to_tokenize, std::vector<int> & output_tokens, FileFormat file_format)
{ {
if (file_format == FileFormat::GGML || file_format == FileFormat::GGHF || file_format == FileFormat::GGJT || file_format == FileFormat::GGJT_2 || file_format == FileFormat::GGJT_3 || file_format == FileFormat::GGUF_LLAMA) if (file_format == FileFormat::GGML || file_format == FileFormat::GGHF || file_format == FileFormat::GGJT || file_format == FileFormat::GGJT_2 || file_format == FileFormat::GGJT_3 || file_format == FileFormat::GGUF_LLAMA || file_format==FileFormat::GGUF_FALCON)
{ {
if(file_format == FileFormat::GGHF || file_format == FileFormat::GGJT || file_format == FileFormat::GGJT_2 ) if(file_format == FileFormat::GGHF || file_format == FileFormat::GGJT || file_format == FileFormat::GGJT_2 )
{ {
@ -432,7 +432,13 @@ ModelLoadResult gpttype_load_model(const load_model_inputs inputs, FileFormat in
else else
{ {
//approximate NTK aware ctx //approximate NTK aware ctx
rope_freq_base = (params.n_ctx <= 3072 ? 26000.0f : (params.n_ctx <= 4096 ? 32000.0f : (params.n_ctx <= 6144 ? 54000.0f : (params.n_ctx <= 8192 ? 82684.0f : (params.n_ctx <= 12288 ? 140000.0f : 200000.0f))))); auto effectivenctx = params.n_ctx;
if((file_format == FileFormat::GGUF_LLAMA || file_format==FileFormat::GGUF_FALCON) && llama_ctx_v4->model.hparams.n_ctx_train>2048)
{
float factor = llama_ctx_v4->model.hparams.n_ctx_train/2048;
effectivenctx = effectivenctx/factor;
}
rope_freq_base = (effectivenctx <= 3072 ? 26000.0f : (effectivenctx <= 4096 ? 32000.0f : (effectivenctx <= 6144 ? 54000.0f : (effectivenctx <= 8192 ? 82684.0f : (effectivenctx <= 12288 ? 140000.0f : 200000.0f)))));
} }
@ -585,7 +591,7 @@ ModelLoadResult gpttype_load_model(const load_model_inputs inputs, FileFormat in
} }
return ModelLoadResult::SUCCESS; return ModelLoadResult::SUCCESS;
} }
else if(file_format==FileFormat::GGUF_LLAMA) else if(file_format==FileFormat::GGUF_LLAMA || file_format==FileFormat::GGUF_FALCON)
{ {
llama_context_params llama_ctx_params = llama_context_default_params(); llama_context_params llama_ctx_params = llama_context_default_params();
llama_ctx_params.n_ctx = inputs.max_context_length; llama_ctx_params.n_ctx = inputs.max_context_length;
@ -598,6 +604,11 @@ ModelLoadResult gpttype_load_model(const load_model_inputs inputs, FileFormat in
llama_ctx_params.use_mmap = inputs.use_mmap; llama_ctx_params.use_mmap = inputs.use_mmap;
llama_ctx_params.use_mlock = inputs.use_mlock; llama_ctx_params.use_mlock = inputs.use_mlock;
llama_ctx_params.n_gpu_layers = inputs.gpulayers; llama_ctx_params.n_gpu_layers = inputs.gpulayers;
if(file_format==FileFormat::GGUF_FALCON && llama_ctx_params.n_gpu_layers>0)
{
printf("\nGPU layer offload for GGUF FALCON is known to have issues, it has been set to 0.\n");
llama_ctx_params.n_gpu_layers = 0;
}
llama_ctx_params.main_gpu = cu_parseinfo_maindevice; llama_ctx_params.main_gpu = cu_parseinfo_maindevice;
llama_ctx_params.rope_freq_base = rope_freq_base; llama_ctx_params.rope_freq_base = rope_freq_base;
llama_ctx_params.rope_freq_scale = rope_freq_scale; llama_ctx_params.rope_freq_scale = rope_freq_scale;
@ -1120,7 +1131,7 @@ generation_outputs gpttype_generate(const generation_inputs inputs, generation_o
{ {
//for non llama, limit to 256 //for non llama, limit to 256
int bbs = blasbatchsize; int bbs = blasbatchsize;
if (file_format != FileFormat::GGML && file_format != FileFormat::GGHF && file_format != FileFormat::GGJT && file_format != FileFormat::GGJT_2 && file_format != FileFormat::GGJT_3 && file_format != FileFormat::GGUF_LLAMA) if (file_format != FileFormat::GGML && file_format != FileFormat::GGHF && file_format != FileFormat::GGJT && file_format != FileFormat::GGJT_2 && file_format != FileFormat::GGJT_3 && file_format != FileFormat::GGUF_LLAMA && file_format!=FileFormat::GGUF_FALCON)
{ {
bbs = (blasbatchsize > 256 ? 256 : blasbatchsize); bbs = (blasbatchsize > 256 ? 256 : blasbatchsize);
} }
@ -1180,7 +1191,7 @@ generation_outputs gpttype_generate(const generation_inputs inputs, generation_o
{ {
n_vocab = llama_v3_n_vocab(llama_ctx_v3); n_vocab = llama_v3_n_vocab(llama_ctx_v3);
} }
else if(file_format == FileFormat::GGUF_LLAMA) else if(file_format == FileFormat::GGUF_LLAMA || file_format==FileFormat::GGUF_FALCON)
{ {
n_vocab = llama_n_vocab(llama_ctx_v4); n_vocab = llama_n_vocab(llama_ctx_v4);
} }
@ -1331,7 +1342,7 @@ generation_outputs gpttype_generate(const generation_inputs inputs, generation_o
{ {
evalres = (llama_v3_eval(llama_ctx_v3, embd.data(), embdsize, n_past, params.n_threads)==0); evalres = (llama_v3_eval(llama_ctx_v3, embd.data(), embdsize, n_past, params.n_threads)==0);
} }
else if(file_format == FileFormat::GGUF_LLAMA) else if(file_format == FileFormat::GGUF_LLAMA || file_format==FileFormat::GGUF_FALCON)
{ {
evalres = (llama_eval(llama_ctx_v4, embd.data(), embdsize, n_past, params.n_threads)==0); evalres = (llama_eval(llama_ctx_v4, embd.data(), embdsize, n_past, params.n_threads)==0);
} }
@ -1439,9 +1450,9 @@ generation_outputs gpttype_generate(const generation_inputs inputs, generation_o
unsigned int eosID = 0; unsigned int eosID = 0;
float * logitsPtr; float * logitsPtr;
int btsize = banned_token_ids.size(); int btsize = banned_token_ids.size();
if(file_format == FileFormat::GGML || file_format == FileFormat::GGHF || file_format == FileFormat::GGJT || file_format == FileFormat::GGJT_2 || file_format == FileFormat::GGJT_3 || file_format == FileFormat::GGUF_LLAMA) if(file_format == FileFormat::GGML || file_format == FileFormat::GGHF || file_format == FileFormat::GGJT || file_format == FileFormat::GGJT_2 || file_format == FileFormat::GGJT_3 || file_format == FileFormat::GGUF_LLAMA || file_format==FileFormat::GGUF_FALCON)
{ {
if(file_format == FileFormat::GGUF_LLAMA) if(file_format == FileFormat::GGUF_LLAMA || file_format==FileFormat::GGUF_FALCON)
{ {
logitsPtr = llama_get_logits(llama_ctx_v4); logitsPtr = llama_get_logits(llama_ctx_v4);
eosID = llama_token_eos(llama_ctx_v4); eosID = llama_token_eos(llama_ctx_v4);

View file

@ -1,6 +1,6 @@
<!-- <!--
An embedded version of Kobold Lite for use in koboldcpp An embedded version of Kobold Lite for use in koboldcpp
Current version: 57 Current version: 58
Please go to https://github.com/LostRuins/lite.koboldai.net for updates on Kobold Lite. Please go to https://github.com/LostRuins/lite.koboldai.net for updates on Kobold Lite.
Kobold Lite is under the AGPL v3.0 License for the purposes of koboldcpp. Please do not remove this line. Kobold Lite is under the AGPL v3.0 License for the purposes of koboldcpp. Please do not remove this line.
- Concedo - Concedo
@ -1704,8 +1704,22 @@ Kobold Lite is under the AGPL v3.0 License for the purposes of koboldcpp. Please
/*--------- end chat window---------------*/ /*--------- end chat window---------------*/
</style> </style>
<style>
/*--------- begin instruct-UI ------------*/
.ui-settings-inline { font-size: 12px; display:flex; flex-direction: row; }
.instruct-settings-input { margin: 0px 2px; font-size:10px; }
.instruct-settings-input input { width:40px; height:20px; }
#code-block-background-colorselector, #code-block-foreground-colorselector { text-align: center; margin: 0px 5px; }
#you-text-colorselector, #you-speech-colorselector, #you-action-colorselector, #AI-text-colorselector, #AI-speech-colorselector, #AI-action-colorselector, #sys-text-colorselector, #sys-speech-colorselector, #sys-action-colorselector { text-align: center; margin: 0px 5px; }
#you-bubble-colorselector, #AI-bubble-colorselector, #sys-bubble-colorselector, #you-portrait, #AI-portrait { text-align: center; margin: 0px 10px; border-radius: 1rem; padding: 1px 6px; }
@media screen and (max-width: 880px) {
#aesthetic_text_preview_panel { display: none; }
}
/*--------- end instruct-UI -----------*/
</style>
<script> <script>
const niko_square = ""; const niko_square = "";
const human_square = ""; const human_square = "";
const favicon_busy = ""; const favicon_busy = "";
const favivon_normal = ""; const favivon_normal = "";
@ -2825,6 +2839,7 @@ Kobold Lite is under the AGPL v3.0 License for the purposes of koboldcpp. Please
autoscroll: true, //automatically scroll to bottom on render autoscroll: true, //automatically scroll to bottom on render
trimsentences: true, //trim to last punctuation trimsentences: true, //trim to last punctuation
trimwhitespace: false, //trim trailing whitespace trimwhitespace: false, //trim trailing whitespace
unban_tokens: false, //allow the EOS token when using locally
opmode: 1, //what mode are we in? 1=story, 2=adventure, 3=chat, 4=instruct opmode: 1, //what mode are we in? 1=story, 2=adventure, 3=chat, 4=instruct
adventure_is_action: false, //in adventure mode, determine story or action adventure_is_action: false, //in adventure mode, determine story or action
adventure_context_mod: true, //extra injection for adventure mode adventure_context_mod: true, //extra injection for adventure mode
@ -2908,6 +2923,26 @@ Kobold Lite is under the AGPL v3.0 License for the purposes of koboldcpp. Please
//attempt to load settings //attempt to load settings
function init() { function init() {
//polyfill for forEach
if (window.NodeList && !NodeList.prototype.forEach) {
NodeList.prototype.forEach = function (callback, thisArg) {
thisArg = thisArg || window;
for (var i = 0; i < this.length; i++) {
callback.call(thisArg, this[i], i, this);
}
};
}
//polyfill for object.entries
if (!Object.entries)
{
Object.entries = function( obj ){
var ownProps = Object.keys( obj ),i = ownProps.length,resArray = new Array(i); // preallocate the Array
while (i--){resArray[i] = [ownProps[i], obj[ownProps[i]]];}
return resArray;
};
}
//uncompress compacted scenarios //uncompress compacted scenarios
for(let i=0;i<compressed_scenario_db.length;++i) for(let i=0;i<compressed_scenario_db.length;++i)
{ {
@ -2916,7 +2951,7 @@ Kobold Lite is under the AGPL v3.0 License for the purposes of koboldcpp. Please
} }
//disable debug log if not local //disable debug log if not local
const dbgmode = urlParams.get('dbg'); let dbgmode = urlParams.get('dbg');
if (localflag) if (localflag)
{ {
@ -3023,6 +3058,9 @@ Kobold Lite is under the AGPL v3.0 License for the purposes of koboldcpp. Please
//setup drag and drop zone for files //setup drag and drop zone for files
setupDragDrop(); setupDragDrop();
//setup customization UI functionality
initializeInstructUIFunctionality();
//fix for iphone zooming //fix for iphone zooming
if(navigator.userAgent.indexOf('iPhone') > -1 ) if(navigator.userAgent.indexOf('iPhone') > -1 )
{ {
@ -3868,8 +3906,6 @@ Kobold Lite is under the AGPL v3.0 License for the purposes of koboldcpp. Please
"selective": itm.selective, "selective": itm.selective,
"constant": itm.constant "constant": itm.constant
}; };
nwi.content = replaceAll(nwi.content,"{{char}}",chatopponent,true);
nwi.content = replaceAll(nwi.content,"{{user}}",myname,true);
current_wi.push(nwi); current_wi.push(nwi);
} }
} }
@ -3889,14 +3925,6 @@ Kobold Lite is under the AGPL v3.0 License for the purposes of koboldcpp. Please
let greeting = obj.first_mes?obj.first_mes:""; let greeting = obj.first_mes?obj.first_mes:"";
//post process //post process
memory = replaceAll(memory,"{{char}}",chatopponent,true);
scenario = replaceAll(scenario,"{{char}}",chatopponent,true);
greeting = replaceAll(greeting,"{{char}}",chatopponent,true);
examplemsg = replaceAll(examplemsg,"{{char}}",chatopponent,true);
memory = replaceAll(memory,"{{user}}",myname,true);
scenario = replaceAll(scenario,"{{user}}",myname,true);
greeting = replaceAll(greeting,"{{user}}",myname,true);
examplemsg = replaceAll(examplemsg,"{{user}}",myname,true);
if(scenario!="") if(scenario!="")
{ {
scenario = "\n[Scenario: "+scenario+"]"; scenario = "\n[Scenario: "+scenario+"]";
@ -3936,14 +3964,6 @@ Kobold Lite is under the AGPL v3.0 License for the purposes of koboldcpp. Please
let examplemsg = obj.example_dialogue?obj.example_dialogue:""; let examplemsg = obj.example_dialogue?obj.example_dialogue:"";
let greeting = obj.char_greeting?obj.char_greeting:""; let greeting = obj.char_greeting?obj.char_greeting:"";
//post process //post process
memory = replaceAll(memory,"{{char}}",chatopponent);
scenario = replaceAll(scenario,"{{char}}",chatopponent);
greeting = replaceAll(greeting,"{{char}}",chatopponent);
examplemsg = replaceAll(examplemsg,"{{char}}",chatopponent);
memory = replaceAll(memory,"{{user}}",myname);
scenario = replaceAll(scenario,"{{user}}",myname);
greeting = replaceAll(greeting,"{{user}}",myname);
examplemsg = replaceAll(examplemsg,"{{user}}",myname);
if(scenario!="") if(scenario!="")
{ {
scenario = "\n[Scenario: "+scenario+"]"; scenario = "\n[Scenario: "+scenario+"]";
@ -4104,7 +4124,7 @@ Kobold Lite is under the AGPL v3.0 License for the purposes of koboldcpp. Please
} }
if(temp_scenario.opmode == 4) if(temp_scenario.opmode == 4)
{ {
if (temp_scenario.enhanced_instruct_ui) { localsettings.enhanced_instruct_ui = temp_scenario.enhanced_chat_ui; } if (temp_scenario.enhanced_instruct_ui) { localsettings.enhanced_instruct_ui = temp_scenario.enhanced_instruct_ui; }
if (temp_scenario.instruct_starttag) { localsettings.instruct_starttag = temp_scenario.instruct_starttag; } if (temp_scenario.instruct_starttag) { localsettings.instruct_starttag = temp_scenario.instruct_starttag; }
if (temp_scenario.instruct_endtag) { localsettings.instruct_endtag = temp_scenario.instruct_endtag; } if (temp_scenario.instruct_endtag) { localsettings.instruct_endtag = temp_scenario.instruct_endtag; }
} }
@ -5273,6 +5293,7 @@ Kobold Lite is under the AGPL v3.0 License for the purposes of koboldcpp. Please
document.getElementById("invert_colors").checked = localsettings.invert_colors; document.getElementById("invert_colors").checked = localsettings.invert_colors;
document.getElementById("trimsentences").checked = localsettings.trimsentences; document.getElementById("trimsentences").checked = localsettings.trimsentences;
document.getElementById("trimwhitespace").checked = localsettings.trimwhitespace; document.getElementById("trimwhitespace").checked = localsettings.trimwhitespace;
document.getElementById("unban_tokens").checked = localsettings.unban_tokens;
document.getElementById("persist_session").checked = localsettings.persist_session; document.getElementById("persist_session").checked = localsettings.persist_session;
document.getElementById("opmode").value = localsettings.opmode; document.getElementById("opmode").value = localsettings.opmode;
document.getElementById("chatname").value = localsettings.chatname; document.getElementById("chatname").value = localsettings.chatname;
@ -5442,6 +5463,7 @@ Kobold Lite is under the AGPL v3.0 License for the purposes of koboldcpp. Please
localsettings.invert_colors = (document.getElementById("invert_colors").checked ? true : false); localsettings.invert_colors = (document.getElementById("invert_colors").checked ? true : false);
localsettings.trimsentences = (document.getElementById("trimsentences").checked ? true : false); localsettings.trimsentences = (document.getElementById("trimsentences").checked ? true : false);
localsettings.trimwhitespace = (document.getElementById("trimwhitespace").checked ? true : false); localsettings.trimwhitespace = (document.getElementById("trimwhitespace").checked ? true : false);
localsettings.unban_tokens = (document.getElementById("unban_tokens").checked ? true : false);
localsettings.persist_session = (document.getElementById("persist_session").checked ? true : false); localsettings.persist_session = (document.getElementById("persist_session").checked ? true : false);
localsettings.enhanced_chat_ui = (document.getElementById("enhanced_chat_ui").checked ? true : false); localsettings.enhanced_chat_ui = (document.getElementById("enhanced_chat_ui").checked ? true : false);
localsettings.enhanced_instruct_ui = (document.getElementById("enhanced_instruct_ui").checked ? true : false); localsettings.enhanced_instruct_ui = (document.getElementById("enhanced_instruct_ui").checked ? true : false);
@ -5497,6 +5519,10 @@ Kobold Lite is under the AGPL v3.0 License for the purposes of koboldcpp. Please
} }
} }
//show or hide the 'Customize UI' button based on whether the Aesthetic Instruct UI Mode is active or not.
if (localsettings.opmode == 4 && localsettings.enhanced_instruct_ui) { document.getElementById('btn_aesthetics').classList.remove('hidden'); }
else { document.getElementById('btn_aesthetics').classList.add('hidden'); }
//validate samplers, if fail, reset to default //validate samplers, if fail, reset to default
validate_samplers(true); validate_samplers(true);
localsettings.last_selected_preset = document.getElementById("presets").value; localsettings.last_selected_preset = document.getElementById("presets").value;
@ -5594,7 +5620,7 @@ Kobold Lite is under the AGPL v3.0 License for the purposes of koboldcpp. Please
let long_story = (truncated_context.length>1800?true:false); let long_story = (truncated_context.length>1800?true:false);
truncated_context += "\n### Instruction:Summarize the above text in a single paragraph of up to "+(long_story?"ten":"five")+" detailed sentences.\n### Response:"; truncated_context += "\n### Instruction:Summarize the above text in a single paragraph of up to "+(long_story?"ten":"five")+" detailed sentences.\n### Response:";
truncated_context = replace_placeholders(truncated_context);
let submit_payload = { let submit_payload = {
"prompt": truncated_context, "prompt": truncated_context,
"params": { "params": {
@ -5722,10 +5748,8 @@ Kobold Lite is under the AGPL v3.0 License for the purposes of koboldcpp. Please
function btn_editmode() function btn_editmode()
{ {
if (gametext_arr.length > 0) { document.getElementById("allowediting").checked = true;
document.getElementById("allowediting").checked = true; toggle_editable();
toggle_editable();
}
} }
function toggle_editable() { function toggle_editable() {
@ -5750,6 +5774,19 @@ Kobold Lite is under the AGPL v3.0 License for the purposes of koboldcpp. Please
render_gametext(); render_gametext();
} }
function replace_placeholders(inputtxt)
{
//only do this for chat and instruct modes
if(localsettings.opmode==3||localsettings.opmode==4)
{
inputtxt = replaceAll(inputtxt,"{{user}}",localsettings.chatname?localsettings.chatname:"You");
inputtxt = replaceAll(inputtxt,"{{char}}",localsettings.chatopponent?localsettings.chatopponent:defaultchatopponent);
inputtxt = replaceAll(inputtxt,"{{[INPUT]}}",get_instruct_starttag(false));
inputtxt = replaceAll(inputtxt,"{{[OUTPUT]}}",get_instruct_endtag(false));
}
return inputtxt;
}
function end_trim_to_sentence(input,include_newline=false) { function end_trim_to_sentence(input,include_newline=false) {
let last = -1; let last = -1;
let enders = ['.', '!', '?', '`', '*', '"', ')', '}', '`', ']', ';']; let enders = ['.', '!', '?', '`', '*', '"', ')', '}', '`', ']', ';'];
@ -5793,14 +5830,17 @@ Kobold Lite is under the AGPL v3.0 License for the purposes of koboldcpp. Please
var event = event || window.event; var event = event || window.event;
var charCode = event.keyCode || event.which; var charCode = event.keyCode || event.which;
if (!event.shiftKey && charCode == 13) { if (!event.shiftKey && (charCode == 13||(charCode == 10 && event.ctrlKey))) {
let willsubmit = (document.getElementById("entersubmit").checked ? true : false); let willsubmit = (document.getElementById("entersubmit").checked ? true : false);
let newgennotempty = (document.getElementById("input_text").value != ""); let newgennotempty = (document.getElementById("input_text").value != "");
if (willsubmit) { if (willsubmit) {
event.preventDefault(); event.preventDefault();
//enter pressed, trigger auto submit //enter pressed, trigger auto submit
if (newgennotempty && !document.getElementById("btnsend").disabled) { if (!document.getElementById("btnsend").disabled) {
submit_generation(); if(newgennotempty || event.ctrlKey)
{
submit_generation();
}
} }
} }
} }
@ -5870,6 +5910,7 @@ Kobold Lite is under the AGPL v3.0 License for the purposes of koboldcpp. Please
function manual_gen_image() { function manual_gen_image() {
let truncated_context = concat_gametext(true, ""); let truncated_context = concat_gametext(true, "");
truncated_context = replace_placeholders(truncated_context);
var tclen = truncated_context.length; var tclen = truncated_context.length;
if (tclen > 0) { if (tclen > 0) {
var sentence = truncated_context.substring(tclen - 300, tclen); var sentence = truncated_context.substring(tclen - 300, tclen);
@ -6228,6 +6269,7 @@ Kobold Lite is under the AGPL v3.0 License for the purposes of koboldcpp. Please
truncated_context = truncated_context.slice(0, anote_insert_idx) + truncated_anote + truncated_context.slice(anote_insert_idx); truncated_context = truncated_context.slice(0, anote_insert_idx) + truncated_anote + truncated_context.slice(anote_insert_idx);
truncated_context = truncated_memory + truncated_context; truncated_context = truncated_memory + truncated_context;
} }
truncated_context = replace_placeholders(truncated_context);
last_token_budget = truncated_context.length + "/" + max_allowed_characters; last_token_budget = truncated_context.length + "/" + max_allowed_characters;
@ -6340,6 +6382,12 @@ Kobold Lite is under the AGPL v3.0 License for the purposes of koboldcpp. Please
} }
} }
//version 1.2.4 and later supports unban tokens
if (kobold_endpoint_version && kobold_endpoint_version != "" && compare_version_str(kobold_endpoint_version, "1.2.3") > 0)
{
submit_payload.use_default_badwordids = (localsettings.unban_tokens?true:false);
}
let pseudostreaming = should_use_pseudostreaming(); let pseudostreaming = should_use_pseudostreaming();
let pstreamamount = urlParams.get('streamamount'); let pstreamamount = urlParams.get('streamamount');
let streamchunk = 8; //8 tokens per stream tick by default let streamchunk = 8; //8 tokens per stream tick by default
@ -6940,9 +6988,7 @@ Kobold Lite is under the AGPL v3.0 License for the purposes of koboldcpp. Please
img.queue = 0; img.queue = 0;
let origImg = "data:image/jpeg;base64," + finalimg.generations[0].img; let origImg = "data:image/jpeg;base64," + finalimg.generations[0].img;
//console.log("Original image: " + origImg); //console.log("Original image: " + origImg);
compressImage(origImg, (newDataUri) => { compressImage(origImg, (newDataUri) => { img.result = newDataUri; }, true);
img.result = newDataUri;
});
} }
}) })
.catch((error) => { .catch((error) => {
@ -6991,7 +7037,7 @@ Kobold Lite is under the AGPL v3.0 License for the purposes of koboldcpp. Please
} }
} }
function compressImage(inputDataUri, onDone) { function compressImage(inputDataUri, onDone, isJpeg=true) {
let img = document.createElement('img'); let img = document.createElement('img');
let wantedWidth = 256; let wantedWidth = 256;
let wantedHeight = 256; let wantedHeight = 256;
@ -7009,7 +7055,16 @@ Kobold Lite is under the AGPL v3.0 License for the purposes of koboldcpp. Please
// We resize the image with the canvas method // We resize the image with the canvas method
ctx.drawImage(this, 0, 0, wantedWidth, wantedHeight); ctx.drawImage(this, 0, 0, wantedWidth, wantedHeight);
var dataURI = canvas.toDataURL('image/jpeg', 0.33); var dataURI = "";
if(isJpeg)
{
dataURI = canvas.toDataURL(`image/jpeg`, 0.33);
}
else
{
//png does not support compression by default, not recommended!
dataURI = canvas.toDataURL(`image/png`);
}
onDone(dataURI); onDone(dataURI);
}; };
@ -7493,6 +7548,7 @@ Kobold Lite is under the AGPL v3.0 License for the purposes of koboldcpp. Please
fulltxt = concat_gametext(false, "", "%SpnStg%", "%SpnEtg%",true); fulltxt = concat_gametext(false, "", "%SpnStg%", "%SpnEtg%",true);
} else { } else {
fulltxt = concat_gametext(false, "", "", "",true); fulltxt = concat_gametext(false, "", "", "",true);
fulltxt = replace_placeholders(fulltxt);
} }
@ -7520,8 +7576,8 @@ Kobold Lite is under the AGPL v3.0 License for the purposes of koboldcpp. Please
fulltxt = simpleMarkdown(fulltxt); fulltxt = simpleMarkdown(fulltxt);
} }
fulltxt = replaceAll(fulltxt, `%SpcStg%`, `<hr class="hr_instruct"><span class="color_cyan"><img src="`+human_square+`" style="padding:3px 6px 3px 3px;border-radius: 8%;"/>`); fulltxt = replaceAll(fulltxt, `%SpcStg%`, `<hr class="hr_instruct"><span class="color_cyan"><img src="`+human_square+`" style="height:38px;width:auto;padding:3px 6px 3px 3px;border-radius: 8%;"/>`);
fulltxt = replaceAll(fulltxt, `%SpcEtg%`, `</span><hr class="hr_instruct"><img src="`+niko_square+`" style="padding:3px 6px 3px 3px;border-radius: 8%;"/>`); fulltxt = replaceAll(fulltxt, `%SpcEtg%`, `</span><hr class="hr_instruct"><img src="`+niko_square+`" style="height:38px;width:auto;padding:3px 6px 3px 3px;border-radius: 8%;"/>`);
}else{ }else{
fulltxt = replaceAll(fulltxt, get_instruct_starttag(true), `%SclStg%`+escapeHtml(get_instruct_starttag(true))+`%SpnEtg%`); fulltxt = replaceAll(fulltxt, get_instruct_starttag(true), `%SclStg%`+escapeHtml(get_instruct_starttag(true))+`%SpnEtg%`);
@ -7706,11 +7762,13 @@ Kobold Lite is under the AGPL v3.0 License for the purposes of koboldcpp. Please
// Render onto enhanced chat interface if selected. Currently only applicable to Chat & Instruct modes. // Render onto enhanced chat interface if selected. Currently only applicable to Chat & Instruct modes.
let isStyleApplicable = (localsettings.opmode == 3 && localsettings.enhanced_chat_ui) || (localsettings.opmode == 4 && localsettings.enhanced_instruct_ui); let isStyleApplicable = (localsettings.opmode == 3 && localsettings.enhanced_chat_ui) || (localsettings.opmode == 4 && localsettings.enhanced_instruct_ui);
let inEditMode = (document.getElementById("allowediting").checked ? true : false); let inEditMode = (document.getElementById("allowediting").checked ? true : false);
if (localsettings.enhanced_chat_ui && !inEditMode && isStyleApplicable) if (!inEditMode && isStyleApplicable)
{ {
let textToRender = (gametext_arr.length == 0 ? document.getElementById("gametext").innerHTML : concat_gametext(false, "", "", "", true)); let textToRender = (gametext_arr.length == 0 ? document.getElementById("gametext").innerHTML : concat_gametext(false, "", "", "", true));
textToRender = replace_placeholders(textToRender);
if (localsettings.opmode == 3) { render_enhanced_chat(textToRender); } if (localsettings.opmode == 3) { render_enhanced_chat(textToRender); }
else if (localsettings.opmode == 4) { render_enhanced_chat_instruct(textToRender); } else if (localsettings.opmode == 4) { document.getElementById("chat_msg_body").innerHTML = render_enhanced_chat_instruct(textToRender); }
// Show the 'AI is typing' message if an answer is pending, and prevent the 'send button' from being clicked again. // Show the 'AI is typing' message if an answer is pending, and prevent the 'send button' from being clicked again.
if (pending_response_id=="") { document.getElementById("chatistyping").classList.add("hidden"); } if (pending_response_id=="") { document.getElementById("chatistyping").classList.add("hidden"); }
@ -7869,28 +7927,6 @@ Kobold Lite is under the AGPL v3.0 License for the purposes of koboldcpp. Please
chatbody.innerHTML = newbodystr; chatbody.innerHTML = newbodystr;
} }
function render_enhanced_chat_instruct(input)
{
var chatbody = document.getElementById("chat_msg_body");
if (!chatbody) { return; }
let you = get_instruct_starttag(); let bot = get_instruct_endtag(); // Instruct tags will be used to wrap text in styled bubbles.
let sysStyle = `</p></div></div></div><div class="incoming_msg"><div class="chat_received_msg" style='width:100%'><div class="chat_received_withd_msg" style='width:100%;background-color:rgba(20, 40, 40, 0.8)'><p style='opacity:.8'>`;
let youStyle = `</p></div></div></div><div class="incoming_msg"><div class="chat_received_msg" style='width:100%'><div class="chat_received_withd_msg" style='width:100%;background-color:#29343a'><p>`;
let botStyle = `</p></div></div></div><div class="incoming_msg"><div class="chat_received_msg" style='width:100%'><div class="chat_received_withd_msg" style='width:100%;background-color:rgba(20, 20, 40, 1)'><p>`;
// We'll transform the input to a well-formatted HTML string. First, trim any newline characters after seq_start and seq_end.
let newbodystr = sysStyle + input.replaceAll(you + '\n', you).replaceAll(bot + '\n', bot).replaceAll(you + ' ', you).replaceAll(bot + ' ', bot);
if (newbodystr.endsWith(bot)) { newbodystr = newbodystr.slice(0, -bot.length); } // Remove the last endtag if it exists.
if (localsettings.instruct_has_markdown) { // Style *Expressions*, "Speech", and code blocks if markdown is toggled.
newbodystr = newbodystr.replaceAll(/\*(\S[^*]+\S)\*/g,`<em style='opacity:0.7'>$1</em>`);
newbodystr = newbodystr.replaceAll(/&quot;(.*?)&quot;/g, `<em style='color:rgba(150, 150, 200, 1)'>"$1"</em>`);
let blocks = newbodystr.split(/(```[\s\S]*?\n[\s\S]*?```)/g); for (var i = 0; i < blocks.length; i++) { if (blocks[i].startsWith("```")) { blocks[i] = blocks[i].replace(/```[\s\S]*?\n([\s\S]*?)```/g, `</p><pre style='margin:0px 100px 0px 20px;background-color:black;color:rgba(180, 35, 40, 1)'>$1</pre><p>`); } else { blocks[i] = blocks[i].replaceAll("```","`").replaceAll("``","`").replaceAll(/`(.*?)`/g, `<code style='background-color:black'>$1</code>`);} } newbodystr = blocks.join('');
}
newbodystr = newbodystr.replaceAll(you, youStyle).replaceAll(bot, botStyle); // Style incoming messages and outcoming messages appropriately.
chatbody.innerHTML = newbodystr.replaceAll('\r\n','<br>').replaceAll('\n','<br>'); // Replace the HTML and handle newlines as needed.
}
function chat_handle_typing(event) function chat_handle_typing(event)
{ {
var event = event || window.event; var event = event || window.event;
@ -8175,9 +8211,349 @@ Kobold Lite is under the AGPL v3.0 License for the purposes of koboldcpp. Please
let v = isNaN(val) ? 0 : val; let v = isNaN(val) ? 0 : val;
return clamp(v, min, max); return clamp(v, min, max);
}; };
</script> </script>
<!-- Aesthetic UI scripts -->
<script>
const aestheticTextStyleTypes = ['text', 'speech', 'action']; // One style per speech type. Could add more later I guess.
const aestheticTextStyleRoles = ['uniform', 'you', 'AI', 'sys']; // Uniform for when you want all roles use the same styles.
class AestheticTextStyle {
constructor(options) {
options = options || {};
this.color = options.color || 'rgba(255,255,255, 1)';
this.fontWeight = options.bold ? 'bold' : 'normal';
this.fontStyle = options.italic ? 'italic' : 'normal';
this.opacity = options.opacity || 1;
}
}
class AestheticInstructUISettings {
constructor() {
this.bubbleColor_sys = 'rgba(20, 40, 40, 0.8)';
this.bubbleColor_you = '#29343a';
this.bubbleColor_AI = 'rgba(20, 20, 40, 1)';
// this.background_anchor_style_you = 'left';
// this.background_anchor_style_AI = 'left';
this.background_margin = [10, 10, 5, 0];
this.background_padding = [40, 40, 10, 0];
this.background_minHeight = 100;
this.show_portraits = true; // Shows/hides the rest of the fields below on the UI, and is also used on the display part of the code.
this.border_style = 'Rect';
this.portrait_width = 100;
this.portrait_height = 100;
this.you_portrait = null;
this.AI_portrait = niko_square;
this.font_size = 12;
this.use_markdown = true;
this.use_uniform_colors = true; // Hides 'you, AI, sys' if set to true via settings UI.
for (let role of aestheticTextStyleRoles) {
this[`text_color_${role}`] = new AestheticTextStyle({ color: 'rgba(255,255,255,1)'} );
this[`speech_color_${role}`] = new AestheticTextStyle({ color: 'rgba(150, 150, 200, 1)', italic: true });
this[`action_color_${role}`] = new AestheticTextStyle({ color: 'rgba(255,255,255, 0.7)', italic: true });
}
this.code_block_background = 'black';
this.code_block_foreground = 'rgba(180, 35, 40, 1)';
}
padding() { return `${this.background_padding[2]}px ${this.background_padding[1]}px ${this.background_padding[3]}px ${this.background_padding[0]}px`; }
margin() { return `${this.background_margin[2]}px ${this.background_margin[1]}px ${this.background_margin[3]}px ${this.background_margin[0]}px`; }
portraitSize() { return { width: this.portrait_width, height: this.border_style == 'Circle' ? this.portrait_width : this.portrait_height }; }
portraitRadius() { return this.border_style == 'Circle' ? '1000rem' : (this.border_style == 'Rounded' ? '2rem' : '0.1rem'); }
}
const sideMapping = { 'left': 0, 'right': 1, 'top': 2, 'bottom': 3 };
let aestheticInstructUISettings = new AestheticInstructUISettings();
let tempAestheticInstructUISettings = null; // These exist to act as backup when customizing, to revert when pressing the 'Cancel' button.
function initializeInstructUIFunctionality() {
// Load the default settings (by default), or the latest chosen ones.
if (localsettings.persist_session === true) {
const jsonString = localStorage.getItem((localmode?"e_":"")+'koboldLiteUICustomizationOptions');
if (jsonString) {
var obj = JSON.parse(jsonString);
for (let key in obj) { if (aestheticInstructUISettings.hasOwnProperty(key)) { aestheticInstructUISettings[key] = obj[key]; } }
}
}else{
//reset to defaults
localStorage.setItem((localmode?"e_":"")+'koboldLiteUICustomizationOptions', JSON.stringify(aestheticInstructUISettings, null, 2));
}
// Initialize text colorPicker and background color pickers. Text colorPicker should change the foreground, and background colorPicker should change the background.
document.querySelectorAll('.enhancedTextColorPicker, .enhancedStandardColorPicker').forEach(element => {
element.addEventListener('click', (e) => {
let clrPicker = e.target.classList.contains('enhancedTextColorPicker') ? colorPicker : colorPicker_background;
clrPicker.clickedElement = e.target;
clrPicker.click();
});
element.addEventListener('mouseover', () => element.style.cursor = "pointer");
});
document.getElementById('colorPicker').addEventListener('change', function() { this.clickedElement.style.color = this.value; this.clickedElement = null; refreshPreview(); });
document.getElementById('colorPicker_background').addEventListener('change', function() { this.clickedElement.style.backgroundColor = this.value; this.clickedElement = null; refreshPreview(); });
// Initialize functionality for the margin & padding input fields.
document.querySelectorAll('.instruct-settings-input').forEach(element => {
const input = element.querySelector('input');
const type = element.getAttribute('data-type');
const side = element.getAttribute('data-side');
input.addEventListener('input', function() {
let clippedvalue = parseInt(this.value, 10);
clippedvalue = cleannum(clippedvalue, 0, 300);
if (type === 'margin') { aestheticInstructUISettings.background_margin[sideMapping[side]] = parseInt(clippedvalue, 10); }
else if (type === 'padding') { aestheticInstructUISettings.background_padding[sideMapping[side]] = parseInt(clippedvalue, 10); }
});
});
// Initialize functionality for the portrait pickers.
document.querySelectorAll('#you-portrait, #AI-portrait').forEach(element => {
element.addEventListener('click', (e) => {
let finput = document.getElementById('portraitFileInput');
finput.click();
finput.onchange = (event) => {
if (event.target.files.length > 0 && event.target.files[0]) {
const file = event.target.files[0];
const reader = new FileReader();
reader.onload = function(img) {
compressImage(img.target.result, loadCompressedImage, true);
function loadCompressedImage(compressedImageURI) {
let isSelfPortrait = (element.id=="you-portrait");
if(isSelfPortrait)
{
aestheticInstructUISettings.you_portrait = compressedImageURI;
}
else
{
aestheticInstructUISettings.AI_portrait = compressedImageURI;
}
updateDataFromUI(); updateTextPreview();
}
}
reader.readAsDataURL(file);
}
finput.value = "";
};
});
element.addEventListener('mouseover', () => element.style.cursor = "pointer");
});
document.getElementById("reset-portrait").addEventListener('click', (e) => {
aestheticInstructUISettings.you_portrait = null;
aestheticInstructUISettings.AI_portrait = niko_square;
updateDataFromUI(); updateTextPreview();
});
document.getElementById("reset-all-aesthetic-instruct").addEventListener('click', (e) => {
let ns = new AestheticInstructUISettings();
aestheticInstructUISettings = deepCopyAestheticSettings(ns);
updateUIFromData();
updateTextPreview();
});
refreshPreview(false);
}
function openAestheticUISettingsMenu() {
tempAestheticInstructUISettings = deepCopyAestheticSettings(aestheticInstructUISettings);
document.getElementById("aestheticsettingscontainer").classList.remove("hidden");
updateTextPreview();
}
function hideAestheticUISettingsMenu(confirm) {
if (!confirm) { aestheticInstructUISettings = deepCopyAestheticSettings(tempAestheticInstructUISettings); updateUIFromData(); }
tempAestheticInstructUISettings = null;
document.getElementById("aestheticsettingscontainer").classList.add("hidden");
render_gametext();
}
function deepCopyAestheticSettings(original) {
let copy = new AestheticInstructUISettings();
for (let [key, value] of Object.entries(original)) {
if (value instanceof AestheticTextStyle) { copy[key] = new AestheticTextStyle({ color: value.color, bold: value.fontWeight === 'bold', italic: value.fontStyle === 'italic', opacity: value.opacity }); }
else { copy[key] = value; }
}
return copy;
}
function refreshPreview(updateFromUI = true) {
if (updateFromUI) { updateDataFromUI(); }
updateUIFromData();
updateTextPreview();
}
function updateDataFromUI() {
for (let role of aestheticTextStyleRoles) {
for (let type of aestheticTextStyleTypes) {
aestheticInstructUISettings[`${type}_color_${role}`] = getTextStyleFromElement(`${role}-${type}-colorselector`);
}
if (role != 'uniform') { aestheticInstructUISettings[`bubbleColor_${role}`] = document.getElementById(`${role}-bubble-colorselector`).style.backgroundColor; }
}
aestheticInstructUISettings.use_markdown = document.getElementById('instructModeMarkdown').checked;
aestheticInstructUISettings.use_uniform_colors = !document.getElementById('instructModeCustomized').checked;
aestheticInstructUISettings.code_block_background = document.getElementById('code-block-background-colorselector').style.color;
aestheticInstructUISettings.code_block_foreground = document.getElementById('code-block-foreground-colorselector').style.color;
aestheticInstructUISettings.font_size = document.getElementById('instruct-font-size').value;
aestheticInstructUISettings.border_style = document.getElementById('instructBorderStyle').value;
aestheticInstructUISettings.portrait_width = document.getElementById('portrait_width').value;
aestheticInstructUISettings.portrait_height = document.getElementById('portrait_height').value;
aestheticInstructUISettings.background_minHeight = document.getElementById('instruct-min-backgroundHeight').value;
// aestheticInstructUISettings.background_anchor_style_you = document.getElementById('background-anchor-style-you').value;
// aestheticInstructUISettings.background_anchor_style_AI = document.getElementById('background-anchor-style-AI').value;
//basic sanitization
aestheticInstructUISettings.font_size = cleannum(aestheticInstructUISettings.font_size, 5, 50);
aestheticInstructUISettings.portrait_width = cleannum(aestheticInstructUISettings.portrait_width, 10, 250);
aestheticInstructUISettings.portrait_height = cleannum(aestheticInstructUISettings.portrait_height, 10, 250);
aestheticInstructUISettings.background_minHeight = cleannum(aestheticInstructUISettings.background_minHeight, 0, 300);
// NOTE: Portraits are loaded automatically from the json, and are stored to aestheticInstructUISettings directly.
localStorage.setItem((localmode?"e_":"")+'koboldLiteUICustomizationOptions', JSON.stringify(aestheticInstructUISettings, null, 2));
function getTextStyleFromElement(id) {
let element = document.getElementById(id);
let computedStyle = window.getComputedStyle(element);
return new AestheticTextStyle({color: computedStyle.color, bold: computedStyle.fontWeight > 400, italic: computedStyle.fontStyle == 'italic', opacity: computedStyle.opacity});
}
}
function updateUIFromData() {
// Parse color settings and apply to the related parts in the UI.
for (let role of aestheticTextStyleRoles) {
for (let type of aestheticTextStyleTypes) {
setElementColor(`${role}-${type}-colorselector`, aestheticInstructUISettings[`${type}_color_${role}`]);
}
if (role != 'uniform') { document.getElementById(`${role}-bubble-colorselector`).style.backgroundColor = aestheticInstructUISettings[`bubbleColor_${role}`]; }
}
// Apply the settings from the json file to the UI.
document.getElementById('instructModeMarkdown').checked = aestheticInstructUISettings.use_markdown;
document.getElementById('instructModeCustomized').checked = !aestheticInstructUISettings.use_uniform_colors;
document.getElementById('code-block-background-colorselector').style.color = aestheticInstructUISettings.code_block_background;
document.getElementById('code-block-foreground-colorselector').style.color = aestheticInstructUISettings.code_block_foreground;
document.getElementById('instruct-font-size').value = aestheticInstructUISettings.font_size;
document.getElementById('instructBorderStyle').value = aestheticInstructUISettings.border_style;
document.getElementById('portrait_width').value = aestheticInstructUISettings.portrait_width;
document.getElementById('portrait_height').value = aestheticInstructUISettings.portrait_height;
document.getElementById('instruct-min-backgroundHeight').value = aestheticInstructUISettings.background_minHeight;
// document.getElementById('background-anchor-style-AI').value = aestheticInstructUISettings.background_anchor_style_AI;
// document.getElementById('background-anchor-style-you').value = aestheticInstructUISettings.background_anchor_style_you;
// Show or hide custumization UI elements based on whether they should be visible in the UI or not.
showOrHide('.uniform-mode-font', document.getElementById('instructModeCustomized').checked == false);
showOrHide('.custom-mode-font', document.getElementById('instructModeCustomized').checked == true);
showOrHide('.instruct-markdown-user', document.getElementById('instructModeMarkdown').checked == true);
showOrHide('.rectPortraitMode', document.getElementById('instructBorderStyle').value != 'Circle');
document.querySelectorAll('.instruct-settings-input').forEach(element => {
const input = element.querySelector('input');
const type = element.getAttribute('data-type');
const side = element.getAttribute('data-side');
if (type === 'margin') { input.value = aestheticInstructUISettings.background_margin[sideMapping[side]]; }
else if (type === 'padding') { input.value = aestheticInstructUISettings.background_padding[sideMapping[side]]; }
});
function setElementColor(id, textStyle) {
let element = document.getElementById(id);
if (!element) { console.warn(`Element with ID: ${id} not found.`); return; }
element.style.color = textStyle.color;
element.style.opacity = textStyle.opacity;
element.style.fontWeight = textStyle.fontWeight;
element.style.fontStyle = textStyle.fontStyle;
}
function showOrHide(classID, value) {
if (value) { document.querySelectorAll(classID).forEach((x) => x.classList.remove('hidden')); }
else { document.querySelectorAll(classID).forEach((x) => x.classList.add('hidden')); }
}
}
function render_enhanced_chat_instruct(input)
{
const contextDict = { sysOpen: '<sys_context_koboldlite_internal>', youOpen: '<user_context_koboldlite_internal>', AIOpen: '<AI_context_koboldlite_internal>', closeTag: '<end_of_context_koboldlite_internal>' }
let you = get_instruct_starttag(); let bot = get_instruct_endtag(); // Instruct tags will be used to wrap text in styled bubbles.
let as = aestheticInstructUISettings; // ..and use this as shortcut to avoid typing it each time.
//update portraits globally as css classes, that way multiple lines can reuse them
let portraitsStyling = `
<style>
.you-portrait-image {margin: 10px 6px; background:url(`+ as.you_portrait +`); background-clip: content-box; background-position: 50% 50%; background-size: 100% 100%; background-origin: content-box; background-repeat: no-repeat; border:none;}
.AI-portrait-image {margin: 10px 6px; background:url(`+ as.AI_portrait +`); background-clip: content-box; background-position: 50% 50%; background-size: 100% 100%; background-origin: content-box; background-repeat: no-repeat; border:none;}
</style>
`;
// We'll transform the input to a well-formatted HTML string.
let newbodystr = (input.startsWith(you) || input.startsWith(bot)) ? input : style('sys') + input; // First, create the string we'll transform. Style system bubble if we should.
if (newbodystr.endsWith(bot)) { newbodystr = newbodystr.slice(0, -bot.length); } // Reduce any unnecessary spaces or newlines. Trim empty replies if they exist.
newbodystr = replaceAll(newbodystr,you + '\n', you);
newbodystr = replaceAll(newbodystr, bot + '\n', bot);
newbodystr = replaceAll(newbodystr, you + ' ', you);
newbodystr = replaceAll(newbodystr,bot + ' ', bot);
newbodystr = replaceAll(newbodystr,'"', '&quot;');
newbodystr = replaceAll(newbodystr,you + '\n', you);
newbodystr = replaceAll(newbodystr,you, style('you'));
newbodystr = replaceAll(newbodystr,bot, style('AI'));
newbodystr += contextDict.closeTag; // Style background of incoming and outgoing messages appropriately.
if (aestheticInstructUISettings.use_markdown) { // If markdown is enabled, style the content of each bubble as well.
let internalHTMLparts = []; // We'll cache the embedded HTML parts here to keep them intact.
for (let role of aestheticTextStyleRoles) { // ..starting by the "speech" and *actions* for each role.
let styleRole = aestheticInstructUISettings.use_uniform_colors ? 'uniform' : role; // Uniform role is preferred if it's active on the settings.
newbodystr = newbodystr.replace(new RegExp(`${contextDict[`${role}Open`]}([^]*?)${contextDict.closeTag}`, 'g'), (match, p) => {
let replacedText = match.replace(/<[^>]*>/g, (htmlPart) => { internalHTMLparts.push(htmlPart); return `<internal_html_${internalHTMLparts.length - 1}>`; });
replacedText = replacedText.replace(/\*(\S[^*]+\S)\*/g, wrapperSpan(styleRole, 'action')); // Apply the actions style to *actions*.
replacedText = replacedText.replace(/&quot;(.*?)&quot;/g, wrapperSpan(styleRole, 'speech')); // Apply the speech style to "speech".
return replacedText;
});
}
internalHTMLparts.forEach((part, index) => { newbodystr = newbodystr.replace(`<internal_html_${index}>`, part); }); // Bring back the embedded HTML parts.
newbodystr = applyStylizedCodeBlocks(); // Apply the code-block styling, if markdown is used.
}
let ret = newbodystr;
ret = replaceAll(ret,'\r\n','<br>');
ret = replaceAll(ret,'\n','<br>');
return portraitsStyling + ret + '</p></div></div><br>'; // Finally, convert newlines to HTML format and return the stylized string.
// Helper functions to allow styling the chat log properly. These affect both the background of the chat bubbles and its content.
function style(role) { return `${contextDict.closeTag}</p></div></div><div style='display:flex; align-items:stretch; flex-direction: row;'>${image(role)}<div style='flex: 1; color: ${as[`text_color_${as.use_uniform_colors ? 'uniform' : role}`].color}; background-color:${as[`bubbleColor_${role}`]}; padding: ${as.padding()}; margin: ${as.margin()}; min-height:${as.background_minHeight}px; font-size: ${as.font_size}px; display: flex; flex-direction:column; align-items: flex-start; justify-content: center; border-radius: 15px'><p>${contextDict[`${role}Open`]}`; }
function wrapperSpan(role, type) { let textStyle = as[`${type}_color_${role}`]; return `<span style='color: ${textStyle.color}; font-style: ${textStyle.fontStyle}; font-weight: ${textStyle.fontWeight}'>$1</span>`; }
function image(role) {
const portraitSrc = as[`${role}_portrait`];
if (!portraitSrc || as.border_style == 'None' || role == 'sys') { return ''; }
return `<div class='${role}-portrait-image' style='width:${as.portraitSize().width}px; height:${as.portraitSize().height}px; border-radius: ${as.portraitRadius()}'></div>`;
}
function applyStylizedCodeBlocks() {
let blocks = newbodystr.split(/(```[\s\S]*?\n[\s\S]*?```)/g);
for (var i = 0; i < blocks.length; i++) {
if (blocks[i].startsWith('```')) { blocks[i] = blocks[i].replace(/```[\s\S]*?\n([\s\S]*?)```/g, `</p><pre style='min-width:80%;margin:0px 40px 0px 20px;background-color:${as.code_block_background};color:${as.code_block_foreground}'>$1</pre><p>`); }
else {
let bi = blocks[i];
bi = replaceAll(bi,'```','`');
bi = replaceAll(bi,'``','`');
bi = bi.replace(/`(.*?)`/g, `<code style='background-color:black'>$1</code>`);
blocks[i] = bi;
}
}
return blocks.join('');
}
}
function updateTextPreview() {
let preview = `You are Mikago, a prestigious bot that's a supervillain.\n\nRoleplay in first person, be prestigious, don't be a bot. This is a fantasy world.\n\nCode blocks should be wrapped in triple backticks, like so:\nqqq\n<Some_\n-- multiline\n--- code here$\nqqq\n[AI_REPLY]\n*takes my hat off to greet the squad* "Greetings, I am Mikago, the prestigious!" *bows to the crew*\n*clears my throat* "Now, I'm sure there are many questions, but all will be answered in due time." *deep breath*\n[USER_REPLY]\n*draws my sword* "Yes. You should know the code to calculate the factorial of a number."\nThe crew also draws their weapons and point them at you, not giving you any space.\n[AI_REPLY]\n*backs off* "Woah, easy there.." *makes some steps backwards, but then stops*\n"I would normally take this as an insult to my prestige, but I understand your caution.." *takes a deep breath*\n"Well, if it's to prove myself, here goes the python code to calculate the factorial of a number.."\n\nMikago opens a live-code-portal with his magic and writes the code that was requested.\nqqq\ndef factorial(n):\n if n == 0:\n return 1\n else:\n return n * factorial(n-1)\nqqq\n*looks at you, getting impatient* "Are we ok now.. or do you want me to write the code of a game next?"\n[USER_REPLY]\n*sheathes my sword and approaches for a hug* "Oh, Mikago, my old friend, it is really you!"`;
preview = replaceAll(preview,'qqq', '```');
preview = replaceAll(preview,'\n[USER_REPLY]\n', get_instruct_starttag());
preview = replaceAll(preview,'\n[AI_REPLY]\n', get_instruct_endtag());
document.getElementById('aesthetic_text_preview').innerHTML = render_enhanced_chat_instruct(preview);
}
</script>
</head> </head>
@ -8240,8 +8616,6 @@ Kobold Lite is under the AGPL v3.0 License for the purposes of koboldcpp. Please
<a class="nav-link" href="#" id="btn_share" <a class="nav-link" href="#" id="btn_share"
onclick="export_share_story()">Share</a> onclick="export_share_story()">Share</a>
</li> </li>
</ul> </ul>
</div> </div>
</nav> </nav>
@ -8320,6 +8694,7 @@ Kobold Lite is under the AGPL v3.0 License for the purposes of koboldcpp. Please
<button type="button" class="btn btn-primary" id="btn_actredo2" onpointerdown="btn_redo_longpress_start()" onpointerleave="btn_redo_longpress_end()" onpointerup="btn_redo_longpress_end()" onclick="btn_redo()">Redo</button> <button type="button" class="btn btn-primary" id="btn_actredo2" onpointerdown="btn_redo_longpress_start()" onpointerleave="btn_redo_longpress_end()" onpointerup="btn_redo_longpress_end()" onclick="btn_redo()">Redo</button>
<button type="button" class="btn btn-primary" id="btn_actretry2" onclick="btn_retry()">Retry</button> <button type="button" class="btn btn-primary" id="btn_actretry2" onclick="btn_retry()">Retry</button>
<button type="button" class="btn btn-primary bg_green" id="btn_genimg2" onclick="manual_gen_image()">Add Img</button> <button type="button" class="btn btn-primary bg_green" id="btn_genimg2" onclick="manual_gen_image()">Add Img</button>
<button type="button" class="btn btn-primary" id="btn_aesthetics" onclick="openAestheticUISettingsMenu()">Customize UI</button>
<button type="button" class="btn btn-primary" id="btn_editmode" onclick="btn_editmode()">Edit</button> <button type="button" class="btn btn-primary" id="btn_editmode" onclick="btn_editmode()">Edit</button>
</div> </div>
@ -8855,6 +9230,10 @@ Kobold Lite is under the AGPL v3.0 License for the purposes of koboldcpp. Please
<div class="justifyleft settingsmall" title="Trim trailing whitespace at the end of context">Trim Whitespace </div> <div class="justifyleft settingsmall" title="Trim trailing whitespace at the end of context">Trim Whitespace </div>
<input type="checkbox" id="trimwhitespace" style="margin:0px 0 0;"> <input type="checkbox" id="trimwhitespace" style="margin:0px 0 0;">
</div> </div>
<div class="settinglabel">
<div class="justifyleft settingsmall" title="Allow the EOS token and others to be used">Unban Tokens (KAI) </div>
<input type="checkbox" id="unban_tokens" style="margin:0px 0 0;">
</div>
<div class="settinglabel"> <div class="settinglabel">
<div class="justifyleft settingsmall" title="Autosaves your current story and settings on exit, reloads when you return">Persist Session </div> <div class="justifyleft settingsmall" title="Autosaves your current story and settings on exit, reloads when you return">Persist Session </div>
<input type="checkbox" id="persist_session" style="margin:0px 0 0;"> <input type="checkbox" id="persist_session" style="margin:0px 0 0;">
@ -9087,8 +9466,193 @@ Kobold Lite is under the AGPL v3.0 License for the purposes of koboldcpp. Please
</div> </div>
</div> </div>
<div class="popupcontainer flex hidden" id="aestheticsettingscontainer">
<div class="popupbg flex"></div>
<div class="nspopup highest" style="margin-left: 20px; margin-right: 20px;">
<div class="popuptitlebar" id="aesthetic_customization_panel">
<div class="popuptitletext">Aesthetic Instruct UI customization panel</div>
</div>
<div class="aidgpopuplistheader" style="display: flex; flex-direction: row; height:70vh;">
<!-- Settings panel -->
<div style="background-color: #122b40;" onchange="refreshPreview()">
<div style="padding: 10px; width:350px; height:100%">
<!-- PRESET LOADING //TODO: Make exporting/importing work -->
<!-- <div class="hidden">
<div class="settinglabel" style="display: flex;flex-direction: column; margin-top:5px; border-top: solid 1px rgba(180, 180, 255, 0.2);">
<div class="justifyleft settingsmall" style="font-size: 14px; margin-bottom: 2px;">RAWRRR Presets</div>
<div class="ui-settings-inline">
<select class="form-control" id="presets" style="width:300px;height:24px;padding:0;" onchange="toggle_UI_preset()">
<option value="1" title="Known Working Settings">[Default]</option>
<option value="1000" title="Known Working Settings">[Custom]</option>
</select>
<button class="btn btn-primary" style="height:25px; font-size: 12px;text-align: center; margin-left: 10px;">Save</button>
</div>
</div>
</div> -->
<!-- BACKGROUND STYLE SETTINGS -->
<div>
<div class="settinglabel" style="display: flex;flex-direction: column; margin-top:5px; border-top: solid 1px rgba(180, 180, 255, 0.2);">
<!-- Background style header -->
<div class="justifyleft settingsmall" style="font-size: 14px; margin-bottom: 2px;">Background Style</div>
<!-- Background style main settings -->
<div style="margin-left: 12px;">
<div class="ui-settings-inline">
<div style="margin-right: 5px">Bubble Color: </div>
<div class="enhancedStandardColorPicker" id="sys-bubble-colorselector">System 🖌️</div>
<div class="enhancedStandardColorPicker" id="you-bubble-colorselector">You 🖌️</div>
<div class="enhancedStandardColorPicker" id="AI-bubble-colorselector">AI 🖌️</div>
</div>
<!-- ANCHOR NOT IMPLEMENTED -->
<!-- <div class="ui-settings-inline hidden">
<div style="margin-right: 38px">Anchor: </div>
<div style="text-align: center; margin: 0px 10px">You:</div>
<select class="form-control" id="background-anchor-style-you" style="width:50px;height:16px;padding:0; font-size: 10px;">
<option value="0">Left</option>
<option value="1">Right</option>
</select>
<div style="text-align: center; margin: 0px 10px">AI:</div>
<select class="form-control" id="background-anchor-style-AI" style="width:50px;height:16px;padding:0; font-size: 10px;">
<option value="0">Left</option>
<option value="1">Right</option>
</select>
</div> -->
<!-- ^^ TODO: IMPLEMENT ANCHOR ABOVE ^^-->
<div class="ui-settings-inline">
<div style="margin-right:20px;">Min Height: </div>
<div class="instruct-settings-input"><input id ="instruct-min-backgroundHeight" type="number"/> px</div>
</div>
<div class="ui-settings-inline">
<div style="margin-right:20px;">Margin (px): </div>
<div class="instruct-settings-input" data-type="margin" data-side="left" >L: <input type="number"/></div>
<div class="instruct-settings-input" data-type="margin" data-side="right" >R: <input type="number"/></div>
<div class="instruct-settings-input" data-type="margin" data-side="top" >T: <input type="number"/></div>
<div class="instruct-settings-input" data-type="margin" data-side="bottom">B: <input type="number"/></div>
</div>
<div class="ui-settings-inline">
<div style="margin-right:13px">Padding (px): </div>
<div class="instruct-settings-input" data-type="padding" data-side="left" >L: <input type="number"/></div>
<div class="instruct-settings-input" data-type="padding" data-side="right" >R: <input type="number"/></div>
<div class="instruct-settings-input" data-type="padding" data-side="top" >T: <input type="number"/></div>
<div class="instruct-settings-input" data-type="padding" data-side="bottom">B: <input type="number"/></div>
</div>
</div>
</div>
</div>
<!-- PORTRAIT STYLE SETTINGS -->
<div>
<div class="settinglabel" style="display: flex;flex-direction: column; margin-top:5px; border-top: solid 1px rgba(180, 180, 255, 0.2);">
<!-- Portrait style header -->
<div class="justifyleft settingsmall" style="font-size: 15px; margin-bottom: 5px;">Portrait Style</div>
<!-- Portrait style main settings -->
<div style="margin-left: 12px;">
<div class="ui-settings-inline">
<div style="margin-right: 27px">Portraits: </div>
<div id="you-portrait">🖼️ Your Portrait</div>
<div id="AI-portrait">🖼️ AI's Portrait</div>
</div>
</div>
<div style="margin-left: 12px;">
<div class="ui-settings-inline">
<div style="margin-right:17px;">Portrait Style: </div>
<select class="form-control" id="instructBorderStyle" style="width:60px;height:16px;padding:0; font-size: 10px;">
<option value="None">None</option>
<option value="Circle">Circle</option>
<option value="Rounded">Rounded</option>
<option value="Rect">Rect</option>
</select>
<div id="reset-portrait" style="margin-left: 10px;"><a href="#" class="color_blueurl">(Reset Image)</a></div>
</div>
<div class="ui-settings-inline">
<div style="margin-right:18px;">Portrait Size: </div>
<div> <span class="rectPortraitMode">W: </span><input id="portrait_width" type="number" placeholder="100" value="100" style='width:40px;height:20px;font-size:10px;'/></div>
<div style="margin-left:5px"><span class="rectPortraitMode">H: </span><input id="portrait_height" type="number" placeholder="100" value="100" style='width:40px;height:20px;font-size:10px;' class="rectPortraitMode"/></div>
<div style="align-self: center;">px</div>
</div>
</div>
</div>
</div>
<!-- FONT STYLE SETTINGS -->
<div>
<div class="settinglabel" style="display: flex;flex-direction: column; margin-top:5px; border-top: solid 1px rgba(180, 180, 255, 0.2);">
<!-- Font style header -->
<div class="justifyleft settingsmall" style="font-size: 15px; margin-bottom:5px;">Font Style</div>
<!-- Font style main settings -->
<div style="margin-left: 12px;">
<div class="ui-settings-inline">
<div style="margin-right:20px;text-align: center;">Font Size: </div>
<div style="margin: 0px 10px"><input id="instruct-font-size" type="number" min="8" max="40" style='width:40px;height:20px;font-size:10px;'/> px</div>
</div>
<div class="ui-settings-inline">
<div style="font-size: 12px; margin-right:27px; text-align: center;">Customize: </div>
<div class="ui-settings-inline" style="font-size: 10px">
<div style="padding-top: 2px;">Per-entity: </div>
<input id="instructModeCustomized" type="checkbox" style="height: 10px;">
</div>
<div class="ui-settings-inline" style="font-size: 10px; margin-left: 10px">
<div style="padding-top: 2px;">Markdown: </div>
<input id="instructModeMarkdown" type="checkbox" style="height: 10px">
</div>
</div>
<div class="ui-settings-inline uniform-mode-font">
<div style="margin-right:48px; text-align: center;">Colors: </div>
<div class="enhancedTextColorPicker" id="uniform-text-colorselector">text🖌</div>
<div class="enhancedTextColorPicker instruct-markdown-user" id="uniform-speech-colorselector">"speech"🖌️</div>
<div class="enhancedTextColorPicker instruct-markdown-user" id="uniform-action-colorselector">*action*🖌️</div>
</div>
<div class="ui-settings-inline custom-mode-font">
<div style="margin-right:58px; text-align: center;">You: </div>
<div class="enhancedTextColorPicker" id="you-text-colorselector">text🖌</div>
<div class="enhancedTextColorPicker instruct-markdown-user" id="you-speech-colorselector">"speech"🖌️</div>
<div class="enhancedTextColorPicker instruct-markdown-user" id="you-action-colorselector">*action*🖌️</div>
</div>
<div class="ui-settings-inline custom-mode-font">
<div style="margin-right:67px; text-align: center;">AI: </div>
<div class="enhancedTextColorPicker" id="AI-text-colorselector">text🖌</div>
<div class="enhancedTextColorPicker instruct-markdown-user" id="AI-speech-colorselector">"speech"🖌️</div>
<div class="enhancedTextColorPicker instruct-markdown-user" id="AI-action-colorselector">*action*🖌️</div>
</div>
<div class="ui-settings-inline custom-mode-font">
<div style="margin-right:38px; text-align: center;">System: </div>
<div class="enhancedTextColorPicker" id="sys-text-colorselector">text🖌</div>
<div class="enhancedTextColorPicker instruct-markdown-user" id="sys-speech-colorselector">"speech"🖌️</div>
<div class="enhancedTextColorPicker instruct-markdown-user" id="sys-action-colorselector">*action*🖌️</div>
</div>
<div class="ui-settings-inline instruct-markdown-user">
<div style="margin-right:11px; text-align: center;">Code blocks: </div>
<div class="enhancedTextColorPicker" id="code-block-background-colorselector">background🖌</div>
<div class="enhancedTextColorPicker" id="code-block-foreground-colorselector">foreground🖌</div>
</div>
</div>
<br>
<div id="reset-all-aesthetic-instruct" style="margin-left: 10px;"><a href="#" class="color_blueurl">(Reset All Styles)</a></div>
</div>
</div>
</div>
<div class="popupfooter" id="aesthetic_instruct_footer" style="margin-top: -55px;height:55px;">
<button type="button" class="btn btn-primary" id="btn_settingsaccept" onclick="hideAestheticUISettingsMenu(true)">OK</button>
<button type="button" class="btn btn-primary" id="btn_settingsclose" onclick="hideAestheticUISettingsMenu(false)">Cancel</button>
</div>
</div>
<div id="aesthetic_text_preview_panel" style="background-color: black; padding: 10px; height:100%; overflow-y: auto; ">
<p>Style Preview</p>
<div id="aesthetic_text_preview" style="background-color: black; margin: 2px; text-align: left; word-wrap: break-word;"></div>
</div>
<input type="color" id="colorPicker" style="display: none;">
<input type="color" id="colorPicker_background" style="display: none;">
<input type="file" id="portraitFileInput" style="display:none" accept="image/*">
</div>
</div>
</div>
</body> </body>
<script> <script>
init(); init();

View file

@ -271,6 +271,11 @@ void print_tok_vec(std::vector<float> &embd)
{ {
fileformat = FileFormat::GGUF_LLAMA; fileformat = FileFormat::GGUF_LLAMA;
} }
else if(modelarch=="falcon")
{
fileformat = FileFormat::GGUF_FALCON; //uses the same loader
printf("\nDetected GGUF FALCON format.\n");
}
else else
{ {
printf("\nERROR: Detected unimplemented GGUF Arch: %s\n",modelarch.c_str()); printf("\nERROR: Detected unimplemented GGUF Arch: %s\n",modelarch.c_str());

View file

@ -47,7 +47,7 @@ enum FileFormat
MPT_1=500, //first supported mpt version MPT_1=500, //first supported mpt version
GGUF_FALCON=600, //GGUF (falcon)
}; };