1302 lines
43 KiB
HTML
1302 lines
43 KiB
HTML
<html>
|
||
<head>
|
||
<meta charset="UTF-8">
|
||
<meta name="viewport" content="width=device-width, initial-scale=1, maximum-scale=1" />
|
||
<meta name="color-scheme" content="light dark">
|
||
<title>llama.cpp - chat</title>
|
||
|
||
<style>
|
||
body {
|
||
font-family: system-ui;
|
||
font-size: 90%;
|
||
}
|
||
|
||
.grid-container {
|
||
display: grid;
|
||
grid-template-columns: auto auto auto;
|
||
padding: 10px;
|
||
}
|
||
|
||
.grid-item {
|
||
padding: 5px;
|
||
/* font-size: 30px; */
|
||
text-align: center;
|
||
}
|
||
|
||
#container {
|
||
margin: 0em auto;
|
||
display: flex;
|
||
flex-direction: column;
|
||
justify-content: space-between;
|
||
height: 100%;
|
||
}
|
||
|
||
main {
|
||
margin: 3px;
|
||
display: flex;
|
||
flex-direction: column;
|
||
justify-content: space-between;
|
||
gap: 1em;
|
||
|
||
flex-grow: 1;
|
||
overflow-y: auto;
|
||
|
||
border: 1px solid #ccc;
|
||
border-radius: 5px;
|
||
padding: 0.5em;
|
||
}
|
||
|
||
h1 {
|
||
text-align: center;
|
||
}
|
||
|
||
.customlink:link {
|
||
color: white;
|
||
background-color: #007aff;
|
||
font-weight: 600;
|
||
text-decoration: none;
|
||
float: right;
|
||
margin-top: 30px;
|
||
display: flex;
|
||
flex-direction: row;
|
||
gap: 0.5em;
|
||
justify-content: flex-end;
|
||
border-radius: 4px;
|
||
padding: 8px;
|
||
}
|
||
|
||
.customlink:visited {
|
||
color: white;
|
||
background-color: #007aff;
|
||
font-weight: 600;
|
||
text-decoration: none;
|
||
float: right;
|
||
margin-top: 30px;
|
||
display: flex;
|
||
flex-direction: row;
|
||
gap: 0.5em;
|
||
justify-content: flex-end;
|
||
padding: 8px;
|
||
}
|
||
|
||
.customlink:hover {
|
||
color: white;
|
||
background-color: #0070ee;
|
||
font-weight: 600;
|
||
text-decoration: none;
|
||
float: right;
|
||
margin-top: 30px;
|
||
display: flex;
|
||
flex-direction: row;
|
||
gap: 0.5em;
|
||
justify-content: flex-end;
|
||
padding: 8px;
|
||
}
|
||
|
||
.customlink:active {
|
||
color: #0070ee;
|
||
background-color: #80b3ef;
|
||
font-weight: 600;
|
||
text-decoration: none;
|
||
float: right;
|
||
margin-top: 30px;
|
||
display: flex;
|
||
flex-direction: row;
|
||
gap: 0.5em;
|
||
justify-content: flex-end;
|
||
padding: 8px;
|
||
}
|
||
|
||
body {
|
||
max-width: 600px;
|
||
min-width: 300px;
|
||
line-height: 1.2;
|
||
margin: 0 auto;
|
||
padding: 0 0.5em;
|
||
}
|
||
|
||
p {
|
||
overflow-wrap: break-word;
|
||
word-wrap: break-word;
|
||
hyphens: auto;
|
||
margin-top: 0.5em;
|
||
margin-bottom: 0.5em;
|
||
}
|
||
|
||
#write form {
|
||
margin: 1em 0 0 0;
|
||
display: flex;
|
||
flex-direction: column;
|
||
gap: 0.5em;
|
||
align-items: stretch;
|
||
}
|
||
|
||
.message-controls {
|
||
display: flex;
|
||
justify-content: flex-end;
|
||
}
|
||
.message-controls > div:nth-child(2) {
|
||
display: flex;
|
||
flex-direction: column;
|
||
gap: 0.5em;
|
||
}
|
||
.message-controls > div:nth-child(2) > div {
|
||
display: flex;
|
||
margin-left: auto;
|
||
gap: 0.5em;
|
||
}
|
||
|
||
fieldset {
|
||
border: none;
|
||
padding: 0;
|
||
margin: 0;
|
||
}
|
||
|
||
fieldset.two {
|
||
display: grid;
|
||
grid-template: "a a";
|
||
gap: 1em;
|
||
}
|
||
|
||
fieldset.three {
|
||
display: grid;
|
||
grid-template: "a a a";
|
||
gap: 1em;
|
||
}
|
||
|
||
details {
|
||
border: 1px solid #aaa;
|
||
border-radius: 4px;
|
||
padding: 0.5em 0.5em 0;
|
||
margin-top: 0.5em;
|
||
}
|
||
|
||
summary {
|
||
font-weight: bold;
|
||
margin: -0.5em -0.5em 0;
|
||
padding: 0.5em;
|
||
cursor: pointer;
|
||
}
|
||
|
||
details[open] {
|
||
padding: 0.5em;
|
||
}
|
||
|
||
.prob-set {
|
||
padding: 0.3em;
|
||
border-bottom: 1px solid #ccc;
|
||
}
|
||
|
||
.popover-content {
|
||
position: absolute;
|
||
background-color: white;
|
||
padding: 0.2em;
|
||
box-shadow: 0 0 10px rgba(0, 0, 0, 0.1);
|
||
}
|
||
|
||
textarea {
|
||
padding: 5px;
|
||
flex-grow: 1;
|
||
width: 100%;
|
||
}
|
||
|
||
pre code {
|
||
display: block;
|
||
background-color: #222;
|
||
color: #ddd;
|
||
}
|
||
|
||
code {
|
||
font-family: monospace;
|
||
padding: 0.1em 0.3em;
|
||
border-radius: 3px;
|
||
}
|
||
|
||
fieldset label {
|
||
margin: 0.5em 0;
|
||
display: block;
|
||
}
|
||
|
||
fieldset label.slim {
|
||
margin: 0 0.5em;
|
||
display: inline;
|
||
}
|
||
|
||
header,
|
||
footer {
|
||
text-align: center;
|
||
}
|
||
|
||
footer {
|
||
font-size: 80%;
|
||
color: #888;
|
||
}
|
||
|
||
.mode-chat textarea[name=prompt] {
|
||
height: 4.5em;
|
||
}
|
||
|
||
.mode-completion textarea[name=prompt] {
|
||
height: 10em;
|
||
}
|
||
|
||
[contenteditable] {
|
||
display: inline-block;
|
||
white-space: pre-wrap;
|
||
outline: 0px solid transparent;
|
||
}
|
||
|
||
@keyframes loading-bg-wipe {
|
||
0% {
|
||
background-position: 0%;
|
||
}
|
||
|
||
100% {
|
||
background-position: 100%;
|
||
}
|
||
}
|
||
|
||
.loading {
|
||
--loading-color-1: #eeeeee00;
|
||
--loading-color-2: #eeeeeeff;
|
||
background-size: 50% 100%;
|
||
background-image: linear-gradient(90deg, var(--loading-color-1), var(--loading-color-2), var(--loading-color-1));
|
||
animation: loading-bg-wipe 2s linear infinite;
|
||
}
|
||
|
||
@media (prefers-color-scheme: dark) {
|
||
.loading {
|
||
--loading-color-1: #22222200;
|
||
--loading-color-2: #222222ff;
|
||
}
|
||
|
||
.popover-content {
|
||
background-color: black;
|
||
}
|
||
}
|
||
</style>
|
||
|
||
<script type="module">
|
||
import {
|
||
html, h, signal, effect, computed, render, useSignal, useEffect, useRef, Component
|
||
} from './index.js';
|
||
|
||
import { llama } from './completion.js';
|
||
import { SchemaConverter } from './json-schema-to-grammar.mjs';
|
||
|
||
let selected_image = false;
|
||
var slot_id = -1;
|
||
|
||
const session = signal({
|
||
prompt: "This is a conversation between User and Llama, a friendly chatbot. Llama is helpful, kind, honest, good at writing, and never fails to answer any requests immediately and with precision.",
|
||
template: "{{prompt}}\n\n{{history}}\n{{char}}:",
|
||
historyTemplate: "{{name}}: {{message}}",
|
||
transcript: [],
|
||
type: "chat", // "chat" | "completion"
|
||
char: "Llama",
|
||
user: "User",
|
||
image_selected: ''
|
||
})
|
||
|
||
const params = signal({
|
||
n_predict: 400,
|
||
temperature: 0.7,
|
||
repeat_last_n: 256, // 0 = disable penalty, -1 = context size
|
||
repeat_penalty: 1.18, // 1.0 = disabled
|
||
dry_multiplier: 0.0, // 0.0 = disabled, 0.8 works well
|
||
dry_base: 1.75, // 0.0 = disabled
|
||
dry_allowed_length: 2, // tokens extending repetitions beyond this receive penalty, 2 works well
|
||
dry_penalty_last_n: -1, // how many tokens to scan for repetitions (0 = disable penalty, -1 = context size)
|
||
top_k: 40, // <= 0 to use vocab size
|
||
top_p: 0.95, // 1.0 = disabled
|
||
min_p: 0.05, // 0 = disabled
|
||
xtc_probability: 0.0, // 0 = disabled;
|
||
xtc_threshold: 0.1, // > 0.5 disables XTC;
|
||
typical_p: 1.0, // 1.0 = disabled
|
||
presence_penalty: 0.0, // 0.0 = disabled
|
||
frequency_penalty: 0.0, // 0.0 = disabled
|
||
mirostat: 0, // 0/1/2
|
||
mirostat_tau: 5, // target entropy
|
||
mirostat_eta: 0.1, // learning rate
|
||
grammar: '',
|
||
n_probs: 0, // no completion_probabilities,
|
||
min_keep: 0, // min probs from each sampler,
|
||
image_data: [],
|
||
cache_prompt: true,
|
||
api_key: ''
|
||
})
|
||
|
||
/* START: Support for storing prompt templates and parameters in browsers LocalStorage */
|
||
|
||
const local_storage_storageKey = "llamacpp_server_local_storage";
|
||
|
||
function local_storage_setDataFromObject(tag, content) {
|
||
localStorage.setItem(local_storage_storageKey + '/' + tag, JSON.stringify(content));
|
||
}
|
||
|
||
function local_storage_setDataFromRawText(tag, content) {
|
||
localStorage.setItem(local_storage_storageKey + '/' + tag, content);
|
||
}
|
||
|
||
function local_storage_getDataAsObject(tag) {
|
||
const item = localStorage.getItem(local_storage_storageKey + '/' + tag);
|
||
if (!item) {
|
||
return null;
|
||
} else {
|
||
return JSON.parse(item);
|
||
}
|
||
}
|
||
|
||
function local_storage_getDataAsRawText(tag) {
|
||
const item = localStorage.getItem(local_storage_storageKey + '/' + tag);
|
||
if (!item) {
|
||
return null;
|
||
} else {
|
||
return item;
|
||
}
|
||
}
|
||
|
||
// create a container for user templates and settings
|
||
|
||
const savedUserTemplates = signal({})
|
||
const selectedUserTemplate = signal({ name: '', template: { session: {}, params: {} } })
|
||
|
||
// let's import locally saved templates and settings if there are any
|
||
// user templates and settings are stored in one object
|
||
// in form of { "templatename": "templatedata" } and { "settingstemplatename":"settingsdata" }
|
||
|
||
console.log('Importing saved templates')
|
||
|
||
let importedTemplates = local_storage_getDataAsObject('user_templates')
|
||
|
||
if (importedTemplates) {
|
||
// saved templates were successfully imported.
|
||
|
||
console.log('Processing saved templates and updating default template')
|
||
params.value = { ...params.value, image_data: [] };
|
||
|
||
//console.log(importedTemplates);
|
||
savedUserTemplates.value = importedTemplates;
|
||
|
||
//override default template
|
||
savedUserTemplates.value.default = { session: session.value, params: params.value }
|
||
local_storage_setDataFromObject('user_templates', savedUserTemplates.value)
|
||
} else {
|
||
// no saved templates detected.
|
||
|
||
console.log('Initializing LocalStorage and saving default template')
|
||
|
||
savedUserTemplates.value = { "default": { session: session.value, params: params.value } }
|
||
local_storage_setDataFromObject('user_templates', savedUserTemplates.value)
|
||
}
|
||
|
||
function userTemplateResetToDefault() {
|
||
console.log('Resetting template to default')
|
||
selectedUserTemplate.value.name = 'default';
|
||
selectedUserTemplate.value.data = savedUserTemplates.value['default'];
|
||
}
|
||
|
||
function userTemplateApply(t) {
|
||
session.value = t.data.session;
|
||
session.value = { ...session.value, image_selected: '' };
|
||
params.value = t.data.params;
|
||
params.value = { ...params.value, image_data: [] };
|
||
}
|
||
|
||
function userTemplateResetToDefaultAndApply() {
|
||
userTemplateResetToDefault()
|
||
userTemplateApply(selectedUserTemplate.value)
|
||
}
|
||
|
||
function userTemplateLoadAndApplyAutosaved() {
|
||
// get autosaved last used template
|
||
let lastUsedTemplate = local_storage_getDataAsObject('user_templates_last')
|
||
|
||
if (lastUsedTemplate) {
|
||
|
||
console.log('Autosaved template found, restoring')
|
||
|
||
selectedUserTemplate.value = lastUsedTemplate
|
||
}
|
||
else {
|
||
|
||
console.log('No autosaved template found, using default template')
|
||
// no autosaved last used template was found, so load from default.
|
||
|
||
userTemplateResetToDefault()
|
||
}
|
||
|
||
console.log('Applying template')
|
||
// and update internal data from templates
|
||
|
||
userTemplateApply(selectedUserTemplate.value)
|
||
}
|
||
|
||
//console.log(savedUserTemplates.value)
|
||
//console.log(selectedUserTemplate.value)
|
||
|
||
function userTemplateAutosave() {
|
||
console.log('Template Autosave...')
|
||
if (selectedUserTemplate.value.name == 'default') {
|
||
// we don't want to save over default template, so let's create a new one
|
||
let newTemplateName = 'UserTemplate-' + Date.now().toString()
|
||
let newTemplate = { 'name': newTemplateName, 'data': { 'session': session.value, 'params': params.value } }
|
||
|
||
console.log('Saving as ' + newTemplateName)
|
||
|
||
// save in the autosave slot
|
||
local_storage_setDataFromObject('user_templates_last', newTemplate)
|
||
|
||
// and load it back and apply
|
||
userTemplateLoadAndApplyAutosaved()
|
||
} else {
|
||
local_storage_setDataFromObject('user_templates_last', { 'name': selectedUserTemplate.value.name, 'data': { 'session': session.value, 'params': params.value } })
|
||
}
|
||
}
|
||
|
||
console.log('Checking for autosaved last used template')
|
||
userTemplateLoadAndApplyAutosaved()
|
||
|
||
/* END: Support for storing prompt templates and parameters in browsers LocalStorage */
|
||
|
||
const tts = window.speechSynthesis;
|
||
const ttsVoice = signal(null)
|
||
|
||
const llamaStats = signal(null)
|
||
const controller = signal(null)
|
||
|
||
// currently generating a completion?
|
||
const generating = computed(() => controller.value != null)
|
||
|
||
// has the user started a chat?
|
||
const chatStarted = computed(() => session.value.transcript.length > 0)
|
||
|
||
const transcriptUpdate = (transcript) => {
|
||
session.value = {
|
||
...session.value,
|
||
transcript
|
||
}
|
||
}
|
||
|
||
// simple template replace
|
||
const template = (str, extraSettings) => {
|
||
let settings = session.value;
|
||
if (extraSettings) {
|
||
settings = { ...settings, ...extraSettings };
|
||
}
|
||
return String(str).replaceAll(/\{\{(.*?)\}\}/g, (_, key) => template(settings[key]));
|
||
}
|
||
|
||
async function runLlama(prompt, llamaParams, char) {
|
||
const currentMessages = [];
|
||
const history = session.value.transcript;
|
||
if (controller.value) {
|
||
throw new Error("already running");
|
||
}
|
||
controller.value = new AbortController();
|
||
for await (const chunk of llama(prompt, llamaParams, { controller: controller.value, api_url: new URL('.', document.baseURI).href })) {
|
||
const data = chunk.data;
|
||
|
||
if (data.stop) {
|
||
while (
|
||
currentMessages.length > 0 &&
|
||
currentMessages[currentMessages.length - 1].content.match(/\n$/) != null
|
||
) {
|
||
currentMessages.pop();
|
||
}
|
||
transcriptUpdate([...history, [char, currentMessages]])
|
||
console.log("Completion finished: '", currentMessages.map(msg => msg.content).join(''), "', summary: ", data);
|
||
} else {
|
||
currentMessages.push(data);
|
||
slot_id = data.slot_id;
|
||
if (selected_image && !data.multimodal) {
|
||
alert("The server was not compiled for multimodal or the model projector can't be loaded.");
|
||
return;
|
||
}
|
||
transcriptUpdate([...history, [char, currentMessages]])
|
||
}
|
||
|
||
if (data.timings) {
|
||
llamaStats.value = data;
|
||
}
|
||
}
|
||
|
||
controller.value = null;
|
||
}
|
||
|
||
// send message to server
|
||
const chat = async (msg) => {
|
||
if (controller.value) {
|
||
console.log('already running...');
|
||
return;
|
||
}
|
||
|
||
transcriptUpdate([...session.value.transcript, ["{{user}}", msg]])
|
||
|
||
let prompt = template(session.value.template, {
|
||
message: msg,
|
||
history: session.value.transcript.flatMap(
|
||
([name, data]) =>
|
||
template(
|
||
session.value.historyTemplate,
|
||
{
|
||
name,
|
||
message: Array.isArray(data) ?
|
||
data.map(msg => msg.content).join('').replace(/^\s/, '') :
|
||
data,
|
||
}
|
||
)
|
||
).join("\n"),
|
||
});
|
||
if (selected_image) {
|
||
prompt = `A chat between a curious human and an artificial intelligence assistant. The assistant gives helpful, detailed, and polite answers to the human's questions.\nUSER:[img-10]${msg}\nASSISTANT:`;
|
||
}
|
||
await runLlama(prompt, {
|
||
...params.value,
|
||
slot_id: slot_id,
|
||
stop: ["</s>", template("{{char}}:"), template("{{user}}:")],
|
||
}, "{{char}}");
|
||
}
|
||
|
||
const runCompletion = () => {
|
||
if (controller.value) {
|
||
console.log('already running...');
|
||
return;
|
||
}
|
||
const { prompt } = session.value;
|
||
transcriptUpdate([...session.value.transcript, ["", prompt]]);
|
||
runLlama(prompt, {
|
||
...params.value,
|
||
slot_id: slot_id,
|
||
stop: [],
|
||
}, "").finally(() => {
|
||
session.value.prompt = session.value.transcript.map(([_, data]) =>
|
||
Array.isArray(data) ? data.map(msg => msg.content).join('') : data
|
||
).join('');
|
||
session.value.transcript = [];
|
||
})
|
||
}
|
||
|
||
const stop = (e) => {
|
||
e.preventDefault();
|
||
if (controller.value) {
|
||
controller.value.abort();
|
||
controller.value = null;
|
||
}
|
||
}
|
||
|
||
const reset = (e) => {
|
||
stop(e);
|
||
transcriptUpdate([]);
|
||
}
|
||
|
||
const uploadImage = (e) => {
|
||
e.preventDefault();
|
||
document.getElementById("fileInput").click();
|
||
document.getElementById("fileInput").addEventListener("change", function (event) {
|
||
const selectedFile = event.target.files[0];
|
||
if (selectedFile) {
|
||
const reader = new FileReader();
|
||
reader.onload = function () {
|
||
const image_data = reader.result;
|
||
session.value = { ...session.value, image_selected: image_data };
|
||
params.value = {
|
||
...params.value, image_data: [
|
||
{ data: image_data.replace(/data:image\/[^;]+;base64,/, ''), id: 10 }]
|
||
}
|
||
};
|
||
selected_image = true;
|
||
reader.readAsDataURL(selectedFile);
|
||
}
|
||
});
|
||
}
|
||
|
||
const SpeechRecognition = window.SpeechRecognition || window.webkitSpeechRecognition;
|
||
const talkRecognition = SpeechRecognition ? new SpeechRecognition() : null;
|
||
function MessageInput() {
|
||
const message = useSignal("");
|
||
|
||
const talkActive = useSignal(false);
|
||
const sendOnTalk = useSignal(false);
|
||
const talkStop = (e) => {
|
||
if (e) e.preventDefault();
|
||
|
||
talkActive.value = false;
|
||
talkRecognition?.stop();
|
||
}
|
||
const talk = (e) => {
|
||
e.preventDefault();
|
||
|
||
if (talkRecognition)
|
||
talkRecognition.start();
|
||
else
|
||
alert("Speech recognition is not supported by this browser.");
|
||
}
|
||
if(talkRecognition) {
|
||
talkRecognition.onstart = () => {
|
||
talkActive.value = true;
|
||
}
|
||
talkRecognition.onresult = (e) => {
|
||
if (event.results.length > 0) {
|
||
message.value = event.results[0][0].transcript;
|
||
if (sendOnTalk.value) {
|
||
submit(e);
|
||
}
|
||
}
|
||
}
|
||
talkRecognition.onspeechend = () => {
|
||
talkStop();
|
||
}
|
||
}
|
||
|
||
const ttsVoices = useSignal(tts?.getVoices() || []);
|
||
const ttsVoiceDefault = computed(() => ttsVoices.value.find(v => v.default));
|
||
if (tts) {
|
||
tts.onvoiceschanged = () => {
|
||
ttsVoices.value = tts.getVoices();
|
||
}
|
||
}
|
||
|
||
const submit = (e) => {
|
||
stop(e);
|
||
chat(message.value);
|
||
message.value = "";
|
||
}
|
||
|
||
const enterSubmits = (event) => {
|
||
if (event.which === 13 && !event.shiftKey) {
|
||
submit(event);
|
||
}
|
||
}
|
||
|
||
return html`
|
||
<form onsubmit=${submit}>
|
||
<div>
|
||
<textarea
|
||
className=${generating.value ? "loading" : null}
|
||
oninput=${(e) => message.value = e.target.value}
|
||
onkeypress=${enterSubmits}
|
||
placeholder="Say something..."
|
||
rows=2
|
||
type="text"
|
||
value="${message}"
|
||
/>
|
||
</div>
|
||
<div class="message-controls">
|
||
<div> </div>
|
||
<div>
|
||
<div>
|
||
<button type="submit" disabled=${generating.value || talkActive.value}>Send</button>
|
||
<button disabled=${generating.value || talkActive.value} onclick=${uploadImage}>Upload Image</button>
|
||
<button onclick=${stop} disabled=${!generating.value}>Stop</button>
|
||
<button onclick=${reset}>Reset</button>
|
||
</div>
|
||
<div>
|
||
<a href="#" style="cursor: help;" title="Help" onclick=${e => {
|
||
e.preventDefault();
|
||
alert(`STT supported by your browser: ${SpeechRecognition ? 'Yes' : 'No'}\n` +
|
||
`(TTS and speech recognition are not provided by llama.cpp)\n` +
|
||
`Note: STT requires HTTPS to work.`);
|
||
}}>[?]</a>
|
||
<button disabled=${generating.value} onclick=${talkActive.value ? talkStop : talk}>${talkActive.value ? "Stop Talking" : "Talk"}</button>
|
||
<div>
|
||
<input type="checkbox" id="send-on-talk" name="send-on-talk" checked="${sendOnTalk}" onchange=${(e) => sendOnTalk.value = e.target.checked} />
|
||
<label for="send-on-talk" style="line-height: initial;">Send after talking</label>
|
||
</div>
|
||
</div>
|
||
<div>
|
||
<a href="#" style="cursor: help;" title="Help" onclick=${e => {
|
||
e.preventDefault();
|
||
alert(`TTS supported by your browser: ${tts ? 'Yes' : 'No'}\n(TTS and speech recognition are not provided by llama.cpp)`);
|
||
}}>[?]</a>
|
||
<label for="tts-voices" style="line-height: initial;">Bot Voice:</label>
|
||
<select id="tts-voices" name="tts-voices" onchange=${(e) => ttsVoice.value = e.target.value} style="max-width: 100px;">
|
||
<option value="" selected="${!ttsVoice.value}">None</option>
|
||
${[
|
||
...(ttsVoiceDefault.value ? [ttsVoiceDefault.value] : []),
|
||
...ttsVoices.value.filter(v => !v.default),
|
||
].map(
|
||
v => html`<option value="${v.name}" selected="${ttsVoice.value === v.name}">${v.name} (${v.lang}) ${v.default ? '(default)' : ''}</option>`
|
||
)}
|
||
</select>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</form>
|
||
`
|
||
}
|
||
|
||
function CompletionControls() {
|
||
const submit = (e) => {
|
||
stop(e);
|
||
runCompletion();
|
||
}
|
||
return html`
|
||
<div>
|
||
<button onclick=${submit} type="button" disabled=${generating.value}>Start</button>
|
||
<button onclick=${stop} disabled=${!generating.value}>Stop</button>
|
||
<button onclick=${reset}>Reset</button>
|
||
</div>`;
|
||
}
|
||
|
||
const ChatLog = (props) => {
|
||
const messages = session.value.transcript;
|
||
const container = useRef(null)
|
||
|
||
useEffect(() => {
|
||
// scroll to bottom (if needed)
|
||
const parent = container.current.parentElement;
|
||
if (parent && parent.scrollHeight <= parent.scrollTop + parent.offsetHeight + 300) {
|
||
parent.scrollTo(0, parent.scrollHeight)
|
||
}
|
||
}, [messages])
|
||
|
||
const ttsChatLineActiveIx = useSignal(undefined);
|
||
const ttsChatLine = (e, ix, msg) => {
|
||
if (e) e.preventDefault();
|
||
|
||
if (!tts || !ttsVoice.value || !('SpeechSynthesisUtterance' in window)) return;
|
||
|
||
const ttsVoices = tts.getVoices();
|
||
const voice = ttsVoices.find(v => v.name === ttsVoice.value);
|
||
if (!voice) return;
|
||
|
||
if (ttsChatLineActiveIx.value !== undefined) {
|
||
tts.cancel();
|
||
if (ttsChatLineActiveIx.value === ix) {
|
||
ttsChatLineActiveIx.value = undefined;
|
||
return;
|
||
}
|
||
}
|
||
|
||
ttsChatLineActiveIx.value = ix;
|
||
let ttsUtter = new SpeechSynthesisUtterance(msg);
|
||
ttsUtter.voice = voice;
|
||
ttsUtter.onend = e => {
|
||
ttsChatLineActiveIx.value = undefined;
|
||
};
|
||
tts.speak(ttsUtter);
|
||
}
|
||
|
||
const isCompletionMode = session.value.type === 'completion'
|
||
|
||
// Try play the last bot message
|
||
const lastCharChatLinesIxs = useSignal([]);
|
||
const lastCharChatLinesIxsOld = useSignal([]);
|
||
useEffect(() => {
|
||
if (
|
||
!isCompletionMode
|
||
&& lastCharChatLinesIxs.value.length !== lastCharChatLinesIxsOld.value.length
|
||
&& !generating.value
|
||
) {
|
||
const ix = lastCharChatLinesIxs.value[lastCharChatLinesIxs.value.length - 1];
|
||
if (ix !== undefined) {
|
||
const msg = messages[ix];
|
||
ttsChatLine(null, ix, Array.isArray(msg) ? msg[1].map(m => m.content).join('') : msg);
|
||
}
|
||
|
||
lastCharChatLinesIxsOld.value = structuredClone(lastCharChatLinesIxs.value);
|
||
}
|
||
}, [generating.value]);
|
||
|
||
const chatLine = ([user, data], index) => {
|
||
let message
|
||
const isArrayMessage = Array.isArray(data);
|
||
const text = isArrayMessage ?
|
||
data.map(msg => msg.content).join('') :
|
||
data;
|
||
if (params.value.n_probs > 0 && isArrayMessage) {
|
||
message = html`<${Probabilities} data=${data} />`
|
||
} else {
|
||
message = isCompletionMode ?
|
||
text :
|
||
html`<${Markdownish} text=${template(text)} />`
|
||
}
|
||
|
||
const fromBot = user && user === '{{char}}';
|
||
if (fromBot && !lastCharChatLinesIxs.value.includes(index))
|
||
lastCharChatLinesIxs.value.push(index);
|
||
|
||
if (user) {
|
||
return html`
|
||
<div>
|
||
<p key=${index}><strong>${template(user)}:</strong> ${message}</p>
|
||
${
|
||
fromBot && ttsVoice.value
|
||
&& html`<button disabled=${generating.value} onclick=${e => ttsChatLine(e, index, text)} aria-label=${ttsChatLineActiveIx.value === index ? 'Pause' : 'Play'}>${ ttsChatLineActiveIx.value === index ? '⏸️' : '▶️' }</div>`
|
||
}
|
||
</div>
|
||
`;
|
||
} else {
|
||
return isCompletionMode ?
|
||
html`<span key=${index}>${message}</span>` :
|
||
html`<div><p key=${index}>${message}</p></div>`
|
||
}
|
||
};
|
||
|
||
const handleCompletionEdit = (e) => {
|
||
session.value.prompt = e.target.innerText;
|
||
session.value.transcript = [];
|
||
}
|
||
|
||
return html`
|
||
<div id="chat" ref=${container} key=${messages.length}>
|
||
<img style="width: 60%;${!session.value.image_selected ? `display: none;` : ``}" src="${session.value.image_selected}"/>
|
||
<span contenteditable=${isCompletionMode} ref=${container} oninput=${handleCompletionEdit}>
|
||
${messages.flatMap(chatLine)}
|
||
</span>
|
||
</div>`;
|
||
};
|
||
|
||
const ConfigForm = (props) => {
|
||
const updateSession = (el) => session.value = { ...session.value, [el.target.name]: el.target.value }
|
||
const updateParams = (el) => params.value = { ...params.value, [el.target.name]: el.target.value }
|
||
const updateParamsFloat = (el) => params.value = { ...params.value, [el.target.name]: parseFloat(el.target.value) }
|
||
const updateParamsInt = (el) => params.value = { ...params.value, [el.target.name]: Math.floor(parseFloat(el.target.value)) }
|
||
const updateParamsBool = (el) => params.value = { ...params.value, [el.target.name]: el.target.checked }
|
||
|
||
const grammarJsonSchemaPropOrder = signal('')
|
||
const updateGrammarJsonSchemaPropOrder = (el) => grammarJsonSchemaPropOrder.value = el.target.value
|
||
const convertJSONSchemaGrammar = async () => {
|
||
try {
|
||
let schema = JSON.parse(params.value.grammar)
|
||
const converter = new SchemaConverter({
|
||
prop_order: grammarJsonSchemaPropOrder.value
|
||
.split(',')
|
||
.reduce((acc, cur, i) => ({ ...acc, [cur.trim()]: i }), {}),
|
||
allow_fetch: true,
|
||
})
|
||
schema = await converter.resolveRefs(schema, 'input')
|
||
converter.visit(schema, '')
|
||
params.value = {
|
||
...params.value,
|
||
grammar: converter.formatGrammar(),
|
||
}
|
||
} catch (e) {
|
||
alert(`Convert failed: ${e.message}`)
|
||
}
|
||
}
|
||
|
||
const FloatField = ({ label, max, min, name, step, value }) => {
|
||
return html`
|
||
<div>
|
||
<label for="${name}">${label}</label>
|
||
<input type="range" id="${name}" min="${min}" max="${max}" step="${step}" name="${name}" value="${value}" oninput=${updateParamsFloat} />
|
||
<span>${value}</span>
|
||
</div>
|
||
`
|
||
};
|
||
|
||
const IntField = ({ label, max, min, name, value }) => {
|
||
return html`
|
||
<div>
|
||
<label for="${name}">${label}</label>
|
||
<input type="range" id="${name}" min="${min}" max="${max}" name="${name}" value="${value}" oninput=${updateParamsInt} />
|
||
<span>${value}</span>
|
||
</div>
|
||
`
|
||
};
|
||
|
||
const BoolField = ({ label, name, value }) => {
|
||
return html`
|
||
<div>
|
||
<label for="${name}">${label}</label>
|
||
<input type="checkbox" id="${name}" name="${name}" checked="${value}" onclick=${updateParamsBool} />
|
||
</div>
|
||
`
|
||
};
|
||
|
||
const userTemplateReset = (e) => {
|
||
e.preventDefault();
|
||
userTemplateResetToDefaultAndApply()
|
||
}
|
||
|
||
const UserTemplateResetButton = () => {
|
||
if (selectedUserTemplate.value.name == 'default') {
|
||
return html`
|
||
<button disabled>Using default template</button>
|
||
`
|
||
}
|
||
|
||
return html`
|
||
<button onclick=${userTemplateReset}>Reset all to default</button>
|
||
`
|
||
};
|
||
|
||
useEffect(() => {
|
||
// autosave template on every change
|
||
userTemplateAutosave()
|
||
}, [session.value, params.value])
|
||
|
||
const GrammarControl = () => (
|
||
html`
|
||
<div>
|
||
<label for="template">Grammar</label>
|
||
<textarea id="grammar" name="grammar" placeholder="Use gbnf or JSON Schema+convert" value="${params.value.grammar}" rows=4 oninput=${updateParams}/>
|
||
<input type="text" name="prop-order" placeholder="order: prop1,prop2,prop3" oninput=${updateGrammarJsonSchemaPropOrder} />
|
||
<button type="button" onclick=${convertJSONSchemaGrammar}>Convert JSON Schema</button>
|
||
</div>
|
||
`
|
||
);
|
||
|
||
const PromptControlFieldSet = () => (
|
||
html`
|
||
<fieldset>
|
||
<div>
|
||
<label htmlFor="prompt">Prompt</label>
|
||
<textarea type="text" name="prompt" value="${session.value.prompt}" oninput=${updateSession}/>
|
||
</div>
|
||
</fieldset>
|
||
`
|
||
);
|
||
|
||
const ChatConfigForm = () => (
|
||
html`
|
||
${PromptControlFieldSet()}
|
||
|
||
<fieldset class="two">
|
||
<div>
|
||
<label for="user">User name</label>
|
||
<input type="text" name="user" value="${session.value.user}" oninput=${updateSession} />
|
||
</div>
|
||
|
||
<div>
|
||
<label for="bot">Bot name</label>
|
||
<input type="text" name="char" value="${session.value.char}" oninput=${updateSession} />
|
||
</div>
|
||
</fieldset>
|
||
|
||
<fieldset>
|
||
<div>
|
||
<label for="template">Prompt template</label>
|
||
<textarea id="template" name="template" value="${session.value.template}" rows=4 oninput=${updateSession}/>
|
||
</div>
|
||
|
||
<div>
|
||
<label for="template">Chat history template</label>
|
||
<textarea id="template" name="historyTemplate" value="${session.value.historyTemplate}" rows=1 oninput=${updateSession}/>
|
||
</div>
|
||
${GrammarControl()}
|
||
</fieldset>
|
||
`
|
||
);
|
||
|
||
const CompletionConfigForm = () => (
|
||
html`
|
||
${PromptControlFieldSet()}
|
||
<fieldset>${GrammarControl()}</fieldset>
|
||
`
|
||
);
|
||
|
||
return html`
|
||
<form>
|
||
<fieldset class="two">
|
||
<${UserTemplateResetButton}/>
|
||
<div>
|
||
<label class="slim"><input type="radio" name="type" value="chat" checked=${session.value.type === "chat"} oninput=${updateSession} /> Chat</label>
|
||
<label class="slim"><input type="radio" name="type" value="completion" checked=${session.value.type === "completion"} oninput=${updateSession} /> Completion</label>
|
||
</div>
|
||
</fieldset>
|
||
|
||
${session.value.type === 'chat' ? ChatConfigForm() : CompletionConfigForm()}
|
||
|
||
<fieldset class="two">
|
||
${IntField({ label: "Predictions", max: 2048, min: -1, name: "n_predict", value: params.value.n_predict })}
|
||
${FloatField({ label: "Temperature", max: 2.0, min: 0.0, name: "temperature", step: 0.01, value: params.value.temperature })}
|
||
${FloatField({ label: "Penalize repeat sequence", max: 2.0, min: 0.0, name: "repeat_penalty", step: 0.01, value: params.value.repeat_penalty })}
|
||
${IntField({ label: "Consider N tokens for penalize", max: 2048, min: 0, name: "repeat_last_n", value: params.value.repeat_last_n })}
|
||
${IntField({ label: "Top-K sampling", max: 100, min: -1, name: "top_k", value: params.value.top_k })}
|
||
${FloatField({ label: "Top-P sampling", max: 1.0, min: 0.0, name: "top_p", step: 0.01, value: params.value.top_p })}
|
||
${FloatField({ label: "Min-P sampling", max: 1.0, min: 0.0, name: "min_p", step: 0.01, value: params.value.min_p })}
|
||
</fieldset>
|
||
<details>
|
||
<summary>More options</summary>
|
||
<fieldset class="two">
|
||
${FloatField({ label: "Typical P", max: 1.0, min: 0.0, name: "typical_p", step: 0.01, value: params.value.typical_p })}
|
||
${FloatField({ label: "Presence penalty", max: 1.0, min: 0.0, name: "presence_penalty", step: 0.01, value: params.value.presence_penalty })}
|
||
${FloatField({ label: "Frequency penalty", max: 1.0, min: 0.0, name: "frequency_penalty", step: 0.01, value: params.value.frequency_penalty })}
|
||
${FloatField({ label: "DRY Penalty Multiplier", max: 5.0, min: 0.0, name: "dry_multiplier", step: 0.01, value: params.value.dry_multiplier })}
|
||
${FloatField({ label: "DRY Base", max: 3.0, min: 1.0, name: "dry_base", step: 0.01, value: params.value.dry_base })}
|
||
${IntField({ label: "DRY Allowed Length", max: 10, min: 2, step: 1, name: "dry_allowed_length", value: params.value.dry_allowed_length })}
|
||
${IntField({ label: "DRY Penalty Last N", max: 2048, min: -1, step: 16, name: "dry_penalty_last_n", value: params.value.dry_penalty_last_n })}
|
||
${FloatField({ label: "XTC probability", max: 1.0, min: 0.0, name: "xtc_probability", step: 0.01, value: params.value.xtc_probability })}
|
||
${FloatField({ label: "XTC threshold", max: 0.5, min: 0.0, name: "xtc_threshold", step: 0.01, value: params.value.xtc_threshold })}
|
||
</fieldset>
|
||
<hr />
|
||
<fieldset class="three">
|
||
<div>
|
||
<label><input type="radio" name="mirostat" value="0" checked=${params.value.mirostat == 0} oninput=${updateParamsInt} /> no Mirostat</label>
|
||
<label><input type="radio" name="mirostat" value="1" checked=${params.value.mirostat == 1} oninput=${updateParamsInt} /> Mirostat v1</label>
|
||
<label><input type="radio" name="mirostat" value="2" checked=${params.value.mirostat == 2} oninput=${updateParamsInt} /> Mirostat v2</label>
|
||
</div>
|
||
${FloatField({ label: "Mirostat tau", max: 10.0, min: 0.0, name: "mirostat_tau", step: 0.01, value: params.value.mirostat_tau })}
|
||
${FloatField({ label: "Mirostat eta", max: 1.0, min: 0.0, name: "mirostat_eta", step: 0.01, value: params.value.mirostat_eta })}
|
||
</fieldset>
|
||
<fieldset>
|
||
${IntField({ label: "Show Probabilities", max: 10, min: 0, name: "n_probs", value: params.value.n_probs })}
|
||
</fieldset>
|
||
<fieldset>
|
||
${IntField({ label: "Min Probabilities from each Sampler", max: 10, min: 0, name: "min_keep", value: params.value.min_keep })}
|
||
</fieldset>
|
||
<fieldset>
|
||
<label for="api_key">API Key</label>
|
||
<input type="text" name="api_key" value="${params.value.api_key}" placeholder="Enter API key" oninput=${updateParams} />
|
||
</fieldset>
|
||
</details>
|
||
</form>
|
||
`
|
||
}
|
||
|
||
const probColor = (p) => {
|
||
const r = Math.floor(192 * (1 - p));
|
||
const g = Math.floor(192 * p);
|
||
return `rgba(${r},${g},0,0.3)`;
|
||
}
|
||
|
||
const Probabilities = (params) => {
|
||
return params.data.map(msg => {
|
||
const { completion_probabilities } = msg;
|
||
if (
|
||
!completion_probabilities ||
|
||
completion_probabilities.length === 0
|
||
) return msg.content
|
||
|
||
if (completion_probabilities.length > 1) {
|
||
// Not for byte pair
|
||
if (completion_probabilities[0].content.startsWith('byte: \\')) return msg.content
|
||
|
||
const splitData = completion_probabilities.map(prob => ({
|
||
content: prob.content,
|
||
completion_probabilities: [prob]
|
||
}))
|
||
return html`<${Probabilities} data=${splitData} />`
|
||
}
|
||
|
||
const { probs, content } = completion_probabilities[0]
|
||
const found = probs.find(p => p.tok_str === msg.content)
|
||
const pColor = found ? probColor(found.prob) : 'transparent'
|
||
|
||
const popoverChildren = html`
|
||
<div class="prob-set">
|
||
${probs.map((p, index) => {
|
||
return html`
|
||
<div
|
||
key=${index}
|
||
title=${`prob: ${p.prob}`}
|
||
style=${{
|
||
padding: '0.3em',
|
||
backgroundColor: p.tok_str === content ? probColor(p.prob) : 'transparent'
|
||
}}
|
||
>
|
||
<span>${p.tok_str}: </span>
|
||
<span>${Math.floor(p.prob * 100)}%</span>
|
||
</div>
|
||
`
|
||
})}
|
||
</div>
|
||
`
|
||
|
||
return html`
|
||
<${Popover} style=${{ backgroundColor: pColor }} popoverChildren=${popoverChildren}>
|
||
${msg.content.match(/\n/gim) ? html`<br />` : msg.content}
|
||
</>
|
||
`
|
||
});
|
||
}
|
||
|
||
// poor mans markdown replacement
|
||
const Markdownish = (params) => {
|
||
const chunks = params.text.split('```');
|
||
|
||
for (let i = 0; i < chunks.length; i++) {
|
||
if (i % 2 === 0) { // outside code block
|
||
chunks[i] = chunks[i]
|
||
.replace(/&/g, '&')
|
||
.replace(/</g, '<')
|
||
.replace(/>/g, '>')
|
||
.replace(/(^|\n)#{1,6} ([^\n]*)(?=([^`]*`[^`]*`)*[^`]*$)/g, '$1<h3>$2</h3>')
|
||
.replace(/\*\*(.*?)\*\*(?=([^`]*`[^`]*`)*[^`]*$)/g, '<strong>$1</strong>')
|
||
.replace(/__(.*?)__(?=([^`]*`[^`]*`)*[^`]*$)/g, '<strong>$1</strong>')
|
||
.replace(/\*(.*?)\*(?=([^`]*`[^`]*`)*[^`]*$)/g, '<em>$1</em>')
|
||
.replace(/_(.*?)_(?=([^`]*`[^`]*`)*[^`]*$)/g, '<em>$1</em>')
|
||
.replace(/```.*?\n([\s\S]*?)```/g, '<pre><code>$1</code></pre>')
|
||
.replace(/`(.*?)`/g, '<code>$1</code>')
|
||
.replace(/\n/gim, '<br />');
|
||
} else { // inside code block
|
||
chunks[i] = `<pre><code>${chunks[i]}</code></pre>`;
|
||
}
|
||
}
|
||
|
||
const restoredText = chunks.join('');
|
||
|
||
return html`<span dangerouslySetInnerHTML=${{ __html: restoredText }} />`;
|
||
};
|
||
|
||
const ModelGenerationInfo = (params) => {
|
||
if (!llamaStats.value) {
|
||
return html`<span/>`
|
||
}
|
||
return html`
|
||
<span>
|
||
${llamaStats.value.tokens_predicted} predicted, ${llamaStats.value.tokens_cached} cached, ${llamaStats.value.timings.predicted_per_token_ms.toFixed()}ms per token, ${llamaStats.value.timings.predicted_per_second.toFixed(2)} tokens per second
|
||
</span>
|
||
`
|
||
}
|
||
|
||
|
||
// simple popover impl
|
||
const Popover = (props) => {
|
||
const isOpen = useSignal(false);
|
||
const position = useSignal({ top: '0px', left: '0px' });
|
||
const buttonRef = useRef(null);
|
||
const popoverRef = useRef(null);
|
||
|
||
const togglePopover = () => {
|
||
if (buttonRef.current) {
|
||
const rect = buttonRef.current.getBoundingClientRect();
|
||
position.value = {
|
||
top: `${rect.bottom + window.scrollY}px`,
|
||
left: `${rect.left + window.scrollX}px`,
|
||
};
|
||
}
|
||
isOpen.value = !isOpen.value;
|
||
};
|
||
|
||
const handleClickOutside = (event) => {
|
||
if (popoverRef.current && !popoverRef.current.contains(event.target) && !buttonRef.current.contains(event.target)) {
|
||
isOpen.value = false;
|
||
}
|
||
};
|
||
|
||
useEffect(() => {
|
||
document.addEventListener('mousedown', handleClickOutside);
|
||
return () => {
|
||
document.removeEventListener('mousedown', handleClickOutside);
|
||
};
|
||
}, []);
|
||
|
||
return html`
|
||
<span style=${props.style} ref=${buttonRef} onClick=${togglePopover}>${props.children}</span>
|
||
${isOpen.value && html`
|
||
<${Portal} into="#portal">
|
||
<div
|
||
ref=${popoverRef}
|
||
class="popover-content"
|
||
style=${{
|
||
top: position.value.top,
|
||
left: position.value.left,
|
||
}}
|
||
>
|
||
${props.popoverChildren}
|
||
</div>
|
||
</${Portal}>
|
||
`}
|
||
`;
|
||
};
|
||
|
||
// Source: preact-portal (https://github.com/developit/preact-portal/blob/master/src/preact-portal.js)
|
||
/** Redirect rendering of descendants into the given CSS selector */
|
||
class Portal extends Component {
|
||
componentDidUpdate(props) {
|
||
for (let i in props) {
|
||
if (props[i] !== this.props[i]) {
|
||
return setTimeout(this.renderLayer);
|
||
}
|
||
}
|
||
}
|
||
|
||
componentDidMount() {
|
||
this.isMounted = true;
|
||
this.renderLayer = this.renderLayer.bind(this);
|
||
this.renderLayer();
|
||
}
|
||
|
||
componentWillUnmount() {
|
||
this.renderLayer(false);
|
||
this.isMounted = false;
|
||
if (this.remote && this.remote.parentNode) this.remote.parentNode.removeChild(this.remote);
|
||
}
|
||
|
||
findNode(node) {
|
||
return typeof node === 'string' ? document.querySelector(node) : node;
|
||
}
|
||
|
||
renderLayer(show = true) {
|
||
if (!this.isMounted) return;
|
||
|
||
// clean up old node if moving bases:
|
||
if (this.props.into !== this.intoPointer) {
|
||
this.intoPointer = this.props.into;
|
||
if (this.into && this.remote) {
|
||
this.remote = render(html`<${PortalProxy} />`, this.into, this.remote);
|
||
}
|
||
this.into = this.findNode(this.props.into);
|
||
}
|
||
|
||
this.remote = render(html`
|
||
<${PortalProxy} context=${this.context}>
|
||
${show && this.props.children || null}
|
||
</${PortalProxy}>
|
||
`, this.into, this.remote);
|
||
}
|
||
|
||
render() {
|
||
return null;
|
||
}
|
||
}
|
||
// high-order component that renders its first child if it exists.
|
||
// used as a conditional rendering proxy.
|
||
class PortalProxy extends Component {
|
||
getChildContext() {
|
||
return this.props.context;
|
||
}
|
||
render({ children }) {
|
||
return children || null;
|
||
}
|
||
}
|
||
|
||
function App(props) {
|
||
useEffect(() => {
|
||
const query = new URLSearchParams(location.search).get("q");
|
||
if (query) chat(query);
|
||
}, []);
|
||
|
||
return html`
|
||
<div class="mode-${session.value.type}">
|
||
<header>
|
||
<div class="grid-container">
|
||
<div class="grid-item"></div>
|
||
<div class="grid-item"><h1>llama.cpp</h1></div>
|
||
<div class="grid-item"><a class="customlink" href="index-new.html">New UI</a></div>
|
||
</div>
|
||
</header>
|
||
|
||
<main id="content">
|
||
<${chatStarted.value ? ChatLog : ConfigForm} />
|
||
</main>
|
||
|
||
<section id="write">
|
||
<${session.value.type === 'chat' ? MessageInput : CompletionControls} />
|
||
</section>
|
||
|
||
<footer>
|
||
<p><${ModelGenerationInfo} /></p>
|
||
<p>Powered by <a href="https://github.com/ggml-org/llama.cpp">llama.cpp</a> and <a href="https://ggml.ai">ggml.ai</a>.</p>
|
||
</footer>
|
||
</div>
|
||
`;
|
||
}
|
||
|
||
render(h(App), document.querySelector('#container'));
|
||
</script>
|
||
</head>
|
||
|
||
<body>
|
||
<div id="container">
|
||
<input type="file" id="fileInput" accept="image/*" style="display: none;">
|
||
</div>
|
||
<div id="portal"></div>
|
||
</body>
|
||
|
||
</html>
|