From 68e9faa5af88245f92f62eacac52fac7885d491a Mon Sep 17 00:00:00 2001 From: Tyler <68524461+TySP-Dev@users.noreply.github.com> Date: Mon, 25 May 2026 01:50:39 -0400 Subject: [PATCH] Testing --- background.js | 171 +++++ content.js | 15 + generate_icons.html | 203 ++++++ highlight.min.js | 1213 ++++++++++++++++++++++++++++++++++ icons/icon128.png | Bin 0 -> 1353 bytes icons/icon16.png | Bin 0 -> 261 bytes icons/icon48.png | Bin 0 -> 547 bytes manifest.json | 45 ++ marked.min.css | 7 + marked.min.js | 6 + sidepanel.css | 579 ++++++++++++++++ sidepanel.html | 167 +++++ sidepanel.js | 1535 +++++++++++++++++++++++++++++++++++++++++++ 13 files changed, 3941 insertions(+) create mode 100644 background.js create mode 100644 content.js create mode 100644 generate_icons.html create mode 100644 highlight.min.js create mode 100644 icons/icon128.png create mode 100644 icons/icon16.png create mode 100644 icons/icon48.png create mode 100644 manifest.json create mode 100644 marked.min.css create mode 100644 marked.min.js create mode 100644 sidepanel.css create mode 100644 sidepanel.html create mode 100644 sidepanel.js diff --git a/background.js b/background.js new file mode 100644 index 0000000..f4d37a8 --- /dev/null +++ b/background.js @@ -0,0 +1,171 @@ +// ── Side panel ─────────────────────────────────────────────────────────────── + +if (chrome.sidePanel) { + chrome.sidePanel.setPanelBehavior({ openPanelOnActionClick: true }).catch(() => {}); +} +chrome.action.onClicked.addListener((tab) => { + if (chrome.sidePanel && tab.id) chrome.sidePanel.open({ tabId: tab.id }).catch(() => {}); +}); + +// ── Active stream registry (for cancellation) ──────────────────────────────── + +const activeStreams = new Map(); // requestId → AbortController + +// ── Message router ──────────────────────────────────────────────────────────── + +chrome.runtime.onMessage.addListener((message, sender, sendResponse) => { + if (message.action === 'OLLAMA_GET') { + handleGet(message, sendResponse); + return true; + } + if (message.action === 'OLLAMA_FETCH') { + handleStream(message); + sendResponse({ started: true }); + return false; + } + if (message.action === 'CANCEL_STREAM') { + const ctrl = activeStreams.get(message.requestId); + if (ctrl) { ctrl.abort(); activeStreams.delete(message.requestId); } + sendResponse({ ok: true }); + return false; + } + if (message.action === 'GET_PAGE_CONTENT') { + handleGetPageContent(message.tabId, sendResponse); + return true; + } +}); + +// ── Helpers ─────────────────────────────────────────────────────────────────── + +function broadcast(msg) { + chrome.runtime.sendMessage(msg).catch(() => {}); +} + +// ── Page content relay ──────────────────────────────────────────────────────── + +function handleGetPageContent(tabId, sendResponse) { + chrome.tabs.sendMessage(tabId, { type: 'GET_PAGE_CONTENT' }, (response) => { + if (chrome.runtime.lastError) { + sendResponse({ error: chrome.runtime.lastError.message }); + return; + } + sendResponse(response || { error: 'Content script did not respond' }); + }); +} + +// ── Non-streaming GET ───────────────────────────────────────────────────────── + +async function handleGet(message, sendResponse) { + try { + const res = await fetch(message.url, { method: 'GET', headers: message.headers || {} }); + if (!res.ok) { sendResponse({ ok: false, status: res.status, error: 'HTTP ' + res.status }); return; } + const data = await res.json(); + sendResponse({ ok: true, data }); + } catch (e) { + sendResponse({ ok: false, error: e.message }); + } +} + +// ── Streaming POST ──────────────────────────────────────────────────────────── +// Requests go through the background service worker so they carry no page +// Origin header — avoiding 403s from reverse proxies that block extension origins. + +async function handleStream(message) { + const { url, headers, body, requestId } = message; + const controller = new AbortController(); + activeStreams.set(requestId, controller); + + try { + const res = await fetch(url, { + method: 'POST', + headers: headers || {}, + body, + signal: controller.signal + }); + + if (!res.ok) { + let errBody = ''; + try { errBody = await res.text(); } catch (_) {} + broadcast({ action: 'STREAM_ERROR', requestId, + error: 'HTTP ' + res.status + (errBody ? ': ' + errBody.slice(0, 200) : '') }); + return; + } + + const reader = res.body.getReader(); + const decoder = new TextDecoder(); + let buf = ''; + + while (true) { + const { done, value } = await reader.read(); + if (done) break; + + buf += decoder.decode(value, { stream: true }); + const lines = buf.split('\n'); + buf = lines.pop() || ''; + + for (const line of lines) { + const trimmed = line.trim(); + if (!trimmed) continue; + let chunk; + try { chunk = JSON.parse(trimmed); } catch (_) { continue; } + + if (chunk.error) { + broadcast({ action: 'STREAM_ERROR', requestId, error: chunk.error }); + return; + } + + if (chunk.message) { + if (chunk.message.thinking) { + broadcast({ action: 'STREAM_THINKING', requestId, text: chunk.message.thinking }); + } + if (chunk.message.content) { + broadcast({ action: 'STREAM_CHUNK', requestId, text: chunk.message.content }); + } + } + + if (chunk.done) { + if (chunk.eval_count || chunk.total_duration) { + broadcast({ + action: 'STREAM_METRICS', requestId, + metrics: { + eval_count: chunk.eval_count || 0, + eval_duration: chunk.eval_duration || 0, + prompt_eval_count: chunk.prompt_eval_count || 0, + total_duration: chunk.total_duration || 0 + } + }); + } + broadcast({ action: 'STREAM_DONE', requestId }); + return; + } + } + } + + // Drain remaining buffer + if (buf.trim()) { + try { + const last = JSON.parse(buf.trim()); + if (last.message && last.message.content) { + broadcast({ action: 'STREAM_CHUNK', requestId, text: last.message.content }); + } + if (last.error) { broadcast({ action: 'STREAM_ERROR', requestId, error: last.error }); return; } + if (last.done && (last.eval_count || last.total_duration)) { + broadcast({ action: 'STREAM_METRICS', requestId, + metrics: { eval_count: last.eval_count || 0, eval_duration: last.eval_duration || 0, + prompt_eval_count: last.prompt_eval_count || 0, total_duration: last.total_duration || 0 } }); + } + } catch (_) {} + } + + broadcast({ action: 'STREAM_DONE', requestId }); + + } catch (e) { + if (e.name === 'AbortError') { + broadcast({ action: 'STREAM_DONE', requestId, aborted: true }); + } else { + broadcast({ action: 'STREAM_ERROR', requestId, error: e.message }); + } + } finally { + activeStreams.delete(requestId); + } +} diff --git a/content.js b/content.js new file mode 100644 index 0000000..484501d --- /dev/null +++ b/content.js @@ -0,0 +1,15 @@ +console.log('[Ollama Sidebar] content script loaded on', location.href); + +chrome.runtime.onMessage.addListener((message, sender, sendResponse) => { + if (message.type === 'GET_PAGE_CONTENT') { + try { + const clone = document.body.cloneNode(true); + clone.querySelectorAll('script, style, noscript, iframe, svg, canvas').forEach((el) => el.remove()); + const text = (clone.textContent || '').replace(/\s+/g, ' ').trim().slice(0, 8000); + sendResponse({ title: document.title, url: location.href, content: text }); + } catch (e) { + sendResponse({ title: document.title, url: location.href, content: '' }); + } + return true; + } +}); diff --git a/generate_icons.html b/generate_icons.html new file mode 100644 index 0000000..4b41ff7 --- /dev/null +++ b/generate_icons.html @@ -0,0 +1,203 @@ + + + + + Ollama Sidebar — Icon Generator + + + + +

Ollama Sidebar — Icon Generator

+

+ Click Save All Icons (or save each individually), then place the downloaded + files inside the icons/ folder before loading the extension. +

+ +
+ + + +
+ Steps to use: +
    +
  1. Click Save All Icons — your browser will download three PNG files.
  2. +
  3. Move icon16.png, icon48.png, and icon128.png into the icons/ folder.
  4. +
  5. Load the extension: Extensions → Manage Extensions → Load unpacked (Chrome) or Tools → Extensions → Install from Disk (Orion).
  6. +
+
+ + + + diff --git a/highlight.min.js b/highlight.min.js new file mode 100644 index 0000000..5d699ae --- /dev/null +++ b/highlight.min.js @@ -0,0 +1,1213 @@ +/*! + Highlight.js v11.9.0 (git: f47103d4f1) + (c) 2006-2023 undefined and other contributors + License: BSD-3-Clause + */ +var hljs=function(){"use strict";function e(n){ +return n instanceof Map?n.clear=n.delete=n.set=()=>{ +throw Error("map is read-only")}:n instanceof Set&&(n.add=n.clear=n.delete=()=>{ +throw Error("set is read-only") +}),Object.freeze(n),Object.getOwnPropertyNames(n).forEach((t=>{ +const a=n[t],i=typeof a;"object"!==i&&"function"!==i||Object.isFrozen(a)||e(a) +})),n}class n{constructor(e){ +void 0===e.data&&(e.data={}),this.data=e.data,this.isMatchIgnored=!1} +ignoreMatch(){this.isMatchIgnored=!0}}function t(e){ +return e.replace(/&/g,"&").replace(//g,">").replace(/"/g,""").replace(/'/g,"'") +}function a(e,...n){const t=Object.create(null);for(const n in e)t[n]=e[n] +;return n.forEach((e=>{for(const n in e)t[n]=e[n]})),t}const i=e=>!!e.scope +;class r{constructor(e,n){ +this.buffer="",this.classPrefix=n.classPrefix,e.walk(this)}addText(e){ +this.buffer+=t(e)}openNode(e){if(!i(e))return;const n=((e,{prefix:n})=>{ +if(e.startsWith("language:"))return e.replace("language:","language-") +;if(e.includes(".")){const t=e.split(".") +;return[`${n}${t.shift()}`,...t.map(((e,n)=>`${e}${"_".repeat(n+1)}`))].join(" ") +}return`${n}${e}`})(e.scope,{prefix:this.classPrefix});this.span(n)} +closeNode(e){i(e)&&(this.buffer+="")}value(){return this.buffer}span(e){ +this.buffer+=``}}const s=(e={})=>{const n={children:[]} +;return Object.assign(n,e),n};class o{constructor(){ +this.rootNode=s(),this.stack=[this.rootNode]}get top(){ +return this.stack[this.stack.length-1]}get root(){return this.rootNode}add(e){ +this.top.children.push(e)}openNode(e){const n=s({scope:e}) +;this.add(n),this.stack.push(n)}closeNode(){ +if(this.stack.length>1)return this.stack.pop()}closeAllNodes(){ +for(;this.closeNode(););}toJSON(){return JSON.stringify(this.rootNode,null,4)} +walk(e){return this.constructor._walk(e,this.rootNode)}static _walk(e,n){ +return"string"==typeof n?e.addText(n):n.children&&(e.openNode(n), +n.children.forEach((n=>this._walk(e,n))),e.closeNode(n)),e}static _collapse(e){ +"string"!=typeof e&&e.children&&(e.children.every((e=>"string"==typeof e))?e.children=[e.children.join("")]:e.children.forEach((e=>{ +o._collapse(e)})))}}class l extends o{constructor(e){super(),this.options=e} +addText(e){""!==e&&this.add(e)}startScope(e){this.openNode(e)}endScope(){ +this.closeNode()}__addSublanguage(e,n){const t=e.root +;n&&(t.scope="language:"+n),this.add(t)}toHTML(){ +return new r(this,this.options).value()}finalize(){ +return this.closeAllNodes(),!0}}function c(e){ +return e?"string"==typeof e?e:e.source:null}function d(e){return b("(?=",e,")")} +function g(e){return b("(?:",e,")*")}function u(e){return b("(?:",e,")?")} +function b(...e){return e.map((e=>c(e))).join("")}function m(...e){const n=(e=>{ +const n=e[e.length-1] +;return"object"==typeof n&&n.constructor===Object?(e.splice(e.length-1,1),n):{} +})(e);return"("+(n.capture?"":"?:")+e.map((e=>c(e))).join("|")+")"} +function p(e){return RegExp(e.toString()+"|").exec("").length-1} +const _=/\[(?:[^\\\]]|\\.)*\]|\(\??|\\([1-9][0-9]*)|\\./ +;function h(e,{joinWith:n}){let t=0;return e.map((e=>{t+=1;const n=t +;let a=c(e),i="";for(;a.length>0;){const e=_.exec(a);if(!e){i+=a;break} +i+=a.substring(0,e.index), +a=a.substring(e.index+e[0].length),"\\"===e[0][0]&&e[1]?i+="\\"+(Number(e[1])+n):(i+=e[0], +"("===e[0]&&t++)}return i})).map((e=>`(${e})`)).join(n)} +const f="[a-zA-Z]\\w*",E="[a-zA-Z_]\\w*",y="\\b\\d+(\\.\\d+)?",N="(-?)(\\b0[xX][a-fA-F0-9]+|(\\b\\d+(\\.\\d*)?|\\.\\d+)([eE][-+]?\\d+)?)",w="\\b(0b[01]+)",v={ +begin:"\\\\[\\s\\S]",relevance:0},O={scope:"string",begin:"'",end:"'", +illegal:"\\n",contains:[v]},k={scope:"string",begin:'"',end:'"',illegal:"\\n", +contains:[v]},x=(e,n,t={})=>{const i=a({scope:"comment",begin:e,end:n, +contains:[]},t);i.contains.push({scope:"doctag", +begin:"[ ]*(?=(TODO|FIXME|NOTE|BUG|OPTIMIZE|HACK|XXX):)", +end:/(TODO|FIXME|NOTE|BUG|OPTIMIZE|HACK|XXX):/,excludeBegin:!0,relevance:0}) +;const r=m("I","a","is","so","us","to","at","if","in","it","on",/[A-Za-z]+['](d|ve|re|ll|t|s|n)/,/[A-Za-z]+[-][a-z]+/,/[A-Za-z][a-z]{2,}/) +;return i.contains.push({begin:b(/[ ]+/,"(",r,/[.]?[:]?([.][ ]|[ ])/,"){3}")}),i +},M=x("//","$"),S=x("/\\*","\\*/"),A=x("#","$");var C=Object.freeze({ +__proto__:null,APOS_STRING_MODE:O,BACKSLASH_ESCAPE:v,BINARY_NUMBER_MODE:{ +scope:"number",begin:w,relevance:0},BINARY_NUMBER_RE:w,COMMENT:x, +C_BLOCK_COMMENT_MODE:S,C_LINE_COMMENT_MODE:M,C_NUMBER_MODE:{scope:"number", +begin:N,relevance:0},C_NUMBER_RE:N,END_SAME_AS_BEGIN:e=>Object.assign(e,{ +"on:begin":(e,n)=>{n.data._beginMatch=e[1]},"on:end":(e,n)=>{ +n.data._beginMatch!==e[1]&&n.ignoreMatch()}}),HASH_COMMENT_MODE:A,IDENT_RE:f, +MATCH_NOTHING_RE:/\b\B/,METHOD_GUARD:{begin:"\\.\\s*"+E,relevance:0}, +NUMBER_MODE:{scope:"number",begin:y,relevance:0},NUMBER_RE:y, +PHRASAL_WORDS_MODE:{ +begin:/\b(a|an|the|are|I'm|isn't|don't|doesn't|won't|but|just|should|pretty|simply|enough|gonna|going|wtf|so|such|will|you|your|they|like|more)\b/ +},QUOTE_STRING_MODE:k,REGEXP_MODE:{scope:"regexp",begin:/\/(?=[^/\n]*\/)/, +end:/\/[gimuy]*/,contains:[v,{begin:/\[/,end:/\]/,relevance:0,contains:[v]}]}, +RE_STARTERS_RE:"!|!=|!==|%|%=|&|&&|&=|\\*|\\*=|\\+|\\+=|,|-|-=|/=|/|:|;|<<|<<=|<=|<|===|==|=|>>>=|>>=|>=|>>>|>>|>|\\?|\\[|\\{|\\(|\\^|\\^=|\\||\\|=|\\|\\||~", +SHEBANG:(e={})=>{const n=/^#![ ]*\// +;return e.binary&&(e.begin=b(n,/.*\b/,e.binary,/\b.*/)),a({scope:"meta",begin:n, +end:/$/,relevance:0,"on:begin":(e,n)=>{0!==e.index&&n.ignoreMatch()}},e)}, +TITLE_MODE:{scope:"title",begin:f,relevance:0},UNDERSCORE_IDENT_RE:E, +UNDERSCORE_TITLE_MODE:{scope:"title",begin:E,relevance:0}});function T(e,n){ +"."===e.input[e.index-1]&&n.ignoreMatch()}function R(e,n){ +void 0!==e.className&&(e.scope=e.className,delete e.className)}function D(e,n){ +n&&e.beginKeywords&&(e.begin="\\b("+e.beginKeywords.split(" ").join("|")+")(?!\\.)(?=\\b|\\s)", +e.__beforeBegin=T,e.keywords=e.keywords||e.beginKeywords,delete e.beginKeywords, +void 0===e.relevance&&(e.relevance=0))}function I(e,n){ +Array.isArray(e.illegal)&&(e.illegal=m(...e.illegal))}function L(e,n){ +if(e.match){ +if(e.begin||e.end)throw Error("begin & end are not supported with match") +;e.begin=e.match,delete e.match}}function B(e,n){ +void 0===e.relevance&&(e.relevance=1)}const $=(e,n)=>{if(!e.beforeMatch)return +;if(e.starts)throw Error("beforeMatch cannot be used with starts") +;const t=Object.assign({},e);Object.keys(e).forEach((n=>{delete e[n] +})),e.keywords=t.keywords,e.begin=b(t.beforeMatch,d(t.begin)),e.starts={ +relevance:0,contains:[Object.assign(t,{endsParent:!0})] +},e.relevance=0,delete t.beforeMatch +},z=["of","and","for","in","not","or","if","then","parent","list","value"],F="keyword" +;function U(e,n,t=F){const a=Object.create(null) +;return"string"==typeof e?i(t,e.split(" ")):Array.isArray(e)?i(t,e):Object.keys(e).forEach((t=>{ +Object.assign(a,U(e[t],n,t))})),a;function i(e,t){ +n&&(t=t.map((e=>e.toLowerCase()))),t.forEach((n=>{const t=n.split("|") +;a[t[0]]=[e,j(t[0],t[1])]}))}}function j(e,n){ +return n?Number(n):(e=>z.includes(e.toLowerCase()))(e)?0:1}const P={},K=e=>{ +console.error(e)},H=(e,...n)=>{console.log("WARN: "+e,...n)},q=(e,n)=>{ +P[`${e}/${n}`]||(console.log(`Deprecated as of ${e}. ${n}`),P[`${e}/${n}`]=!0) +},G=Error();function Z(e,n,{key:t}){let a=0;const i=e[t],r={},s={} +;for(let e=1;e<=n.length;e++)s[e+a]=i[e],r[e+a]=!0,a+=p(n[e-1]) +;e[t]=s,e[t]._emit=r,e[t]._multi=!0}function W(e){(e=>{ +e.scope&&"object"==typeof e.scope&&null!==e.scope&&(e.beginScope=e.scope, +delete e.scope)})(e),"string"==typeof e.beginScope&&(e.beginScope={ +_wrap:e.beginScope}),"string"==typeof e.endScope&&(e.endScope={_wrap:e.endScope +}),(e=>{if(Array.isArray(e.begin)){ +if(e.skip||e.excludeBegin||e.returnBegin)throw K("skip, excludeBegin, returnBegin not compatible with beginScope: {}"), +G +;if("object"!=typeof e.beginScope||null===e.beginScope)throw K("beginScope must be object"), +G;Z(e,e.begin,{key:"beginScope"}),e.begin=h(e.begin,{joinWith:""})}})(e),(e=>{ +if(Array.isArray(e.end)){ +if(e.skip||e.excludeEnd||e.returnEnd)throw K("skip, excludeEnd, returnEnd not compatible with endScope: {}"), +G +;if("object"!=typeof e.endScope||null===e.endScope)throw K("endScope must be object"), +G;Z(e,e.end,{key:"endScope"}),e.end=h(e.end,{joinWith:""})}})(e)}function Q(e){ +function n(n,t){ +return RegExp(c(n),"m"+(e.case_insensitive?"i":"")+(e.unicodeRegex?"u":"")+(t?"g":"")) +}class t{constructor(){ +this.matchIndexes={},this.regexes=[],this.matchAt=1,this.position=0} +addRule(e,n){ +n.position=this.position++,this.matchIndexes[this.matchAt]=n,this.regexes.push([n,e]), +this.matchAt+=p(e)+1}compile(){0===this.regexes.length&&(this.exec=()=>null) +;const e=this.regexes.map((e=>e[1]));this.matcherRe=n(h(e,{joinWith:"|" +}),!0),this.lastIndex=0}exec(e){this.matcherRe.lastIndex=this.lastIndex +;const n=this.matcherRe.exec(e);if(!n)return null +;const t=n.findIndex(((e,n)=>n>0&&void 0!==e)),a=this.matchIndexes[t] +;return n.splice(0,t),Object.assign(n,a)}}class i{constructor(){ +this.rules=[],this.multiRegexes=[], +this.count=0,this.lastIndex=0,this.regexIndex=0}getMatcher(e){ +if(this.multiRegexes[e])return this.multiRegexes[e];const n=new t +;return this.rules.slice(e).forEach((([e,t])=>n.addRule(e,t))), +n.compile(),this.multiRegexes[e]=n,n}resumingScanAtSamePosition(){ +return 0!==this.regexIndex}considerAll(){this.regexIndex=0}addRule(e,n){ +this.rules.push([e,n]),"begin"===n.type&&this.count++}exec(e){ +const n=this.getMatcher(this.regexIndex);n.lastIndex=this.lastIndex +;let t=n.exec(e) +;if(this.resumingScanAtSamePosition())if(t&&t.index===this.lastIndex);else{ +const n=this.getMatcher(0);n.lastIndex=this.lastIndex+1,t=n.exec(e)} +return t&&(this.regexIndex+=t.position+1, +this.regexIndex===this.count&&this.considerAll()),t}} +if(e.compilerExtensions||(e.compilerExtensions=[]), +e.contains&&e.contains.includes("self"))throw Error("ERR: contains `self` is not supported at the top-level of a language. See documentation.") +;return e.classNameAliases=a(e.classNameAliases||{}),function t(r,s){const o=r +;if(r.isCompiled)return o +;[R,L,W,$].forEach((e=>e(r,s))),e.compilerExtensions.forEach((e=>e(r,s))), +r.__beforeBegin=null,[D,I,B].forEach((e=>e(r,s))),r.isCompiled=!0;let l=null +;return"object"==typeof r.keywords&&r.keywords.$pattern&&(r.keywords=Object.assign({},r.keywords), +l=r.keywords.$pattern, +delete r.keywords.$pattern),l=l||/\w+/,r.keywords&&(r.keywords=U(r.keywords,e.case_insensitive)), +o.keywordPatternRe=n(l,!0), +s&&(r.begin||(r.begin=/\B|\b/),o.beginRe=n(o.begin),r.end||r.endsWithParent||(r.end=/\B|\b/), +r.end&&(o.endRe=n(o.end)), +o.terminatorEnd=c(o.end)||"",r.endsWithParent&&s.terminatorEnd&&(o.terminatorEnd+=(r.end?"|":"")+s.terminatorEnd)), +r.illegal&&(o.illegalRe=n(r.illegal)), +r.contains||(r.contains=[]),r.contains=[].concat(...r.contains.map((e=>(e=>(e.variants&&!e.cachedVariants&&(e.cachedVariants=e.variants.map((n=>a(e,{ +variants:null},n)))),e.cachedVariants?e.cachedVariants:X(e)?a(e,{ +starts:e.starts?a(e.starts):null +}):Object.isFrozen(e)?a(e):e))("self"===e?r:e)))),r.contains.forEach((e=>{t(e,o) +})),r.starts&&t(r.starts,s),o.matcher=(e=>{const n=new i +;return e.contains.forEach((e=>n.addRule(e.begin,{rule:e,type:"begin" +}))),e.terminatorEnd&&n.addRule(e.terminatorEnd,{type:"end" +}),e.illegal&&n.addRule(e.illegal,{type:"illegal"}),n})(o),o}(e)}function X(e){ +return!!e&&(e.endsWithParent||X(e.starts))}class V extends Error{ +constructor(e,n){super(e),this.name="HTMLInjectionError",this.html=n}} +const J=t,Y=a,ee=Symbol("nomatch"),ne=t=>{ +const a=Object.create(null),i=Object.create(null),r=[];let s=!0 +;const o="Could not find the language '{}', did you forget to load/include a language module?",c={ +disableAutodetect:!0,name:"Plain text",contains:[]};let p={ +ignoreUnescapedHTML:!1,throwUnescapedHTML:!1,noHighlightRe:/^(no-?highlight)$/i, +languageDetectRe:/\blang(?:uage)?-([\w-]+)\b/i,classPrefix:"hljs-", +cssSelector:"pre code",languages:null,__emitter:l};function _(e){ +return p.noHighlightRe.test(e)}function h(e,n,t){let a="",i="" +;"object"==typeof n?(a=e, +t=n.ignoreIllegals,i=n.language):(q("10.7.0","highlight(lang, code, ...args) has been deprecated."), +q("10.7.0","Please use highlight(code, options) instead.\nhttps://github.com/highlightjs/highlight.js/issues/2277"), +i=e,a=n),void 0===t&&(t=!0);const r={code:a,language:i};x("before:highlight",r) +;const s=r.result?r.result:f(r.language,r.code,t) +;return s.code=r.code,x("after:highlight",s),s}function f(e,t,i,r){ +const l=Object.create(null);function c(){if(!x.keywords)return void S.addText(A) +;let e=0;x.keywordPatternRe.lastIndex=0;let n=x.keywordPatternRe.exec(A),t="" +;for(;n;){t+=A.substring(e,n.index) +;const i=w.case_insensitive?n[0].toLowerCase():n[0],r=(a=i,x.keywords[a]);if(r){ +const[e,a]=r +;if(S.addText(t),t="",l[i]=(l[i]||0)+1,l[i]<=7&&(C+=a),e.startsWith("_"))t+=n[0];else{ +const t=w.classNameAliases[e]||e;g(n[0],t)}}else t+=n[0] +;e=x.keywordPatternRe.lastIndex,n=x.keywordPatternRe.exec(A)}var a +;t+=A.substring(e),S.addText(t)}function d(){null!=x.subLanguage?(()=>{ +if(""===A)return;let e=null;if("string"==typeof x.subLanguage){ +if(!a[x.subLanguage])return void S.addText(A) +;e=f(x.subLanguage,A,!0,M[x.subLanguage]),M[x.subLanguage]=e._top +}else e=E(A,x.subLanguage.length?x.subLanguage:null) +;x.relevance>0&&(C+=e.relevance),S.__addSublanguage(e._emitter,e.language) +})():c(),A=""}function g(e,n){ +""!==e&&(S.startScope(n),S.addText(e),S.endScope())}function u(e,n){let t=1 +;const a=n.length-1;for(;t<=a;){if(!e._emit[t]){t++;continue} +const a=w.classNameAliases[e[t]]||e[t],i=n[t];a?g(i,a):(A=i,c(),A=""),t++}} +function b(e,n){ +return e.scope&&"string"==typeof e.scope&&S.openNode(w.classNameAliases[e.scope]||e.scope), +e.beginScope&&(e.beginScope._wrap?(g(A,w.classNameAliases[e.beginScope._wrap]||e.beginScope._wrap), +A=""):e.beginScope._multi&&(u(e.beginScope,n),A="")),x=Object.create(e,{parent:{ +value:x}}),x}function m(e,t,a){let i=((e,n)=>{const t=e&&e.exec(n) +;return t&&0===t.index})(e.endRe,a);if(i){if(e["on:end"]){const a=new n(e) +;e["on:end"](t,a),a.isMatchIgnored&&(i=!1)}if(i){ +for(;e.endsParent&&e.parent;)e=e.parent;return e}} +if(e.endsWithParent)return m(e.parent,t,a)}function _(e){ +return 0===x.matcher.regexIndex?(A+=e[0],1):(D=!0,0)}function h(e){ +const n=e[0],a=t.substring(e.index),i=m(x,e,a);if(!i)return ee;const r=x +;x.endScope&&x.endScope._wrap?(d(), +g(n,x.endScope._wrap)):x.endScope&&x.endScope._multi?(d(), +u(x.endScope,e)):r.skip?A+=n:(r.returnEnd||r.excludeEnd||(A+=n), +d(),r.excludeEnd&&(A=n));do{ +x.scope&&S.closeNode(),x.skip||x.subLanguage||(C+=x.relevance),x=x.parent +}while(x!==i.parent);return i.starts&&b(i.starts,e),r.returnEnd?0:n.length} +let y={};function N(a,r){const o=r&&r[0];if(A+=a,null==o)return d(),0 +;if("begin"===y.type&&"end"===r.type&&y.index===r.index&&""===o){ +if(A+=t.slice(r.index,r.index+1),!s){const n=Error(`0 width match regex (${e})`) +;throw n.languageName=e,n.badRule=y.rule,n}return 1} +if(y=r,"begin"===r.type)return(e=>{ +const t=e[0],a=e.rule,i=new n(a),r=[a.__beforeBegin,a["on:begin"]] +;for(const n of r)if(n&&(n(e,i),i.isMatchIgnored))return _(t) +;return a.skip?A+=t:(a.excludeBegin&&(A+=t), +d(),a.returnBegin||a.excludeBegin||(A=t)),b(a,e),a.returnBegin?0:t.length})(r) +;if("illegal"===r.type&&!i){ +const e=Error('Illegal lexeme "'+o+'" for mode "'+(x.scope||"")+'"') +;throw e.mode=x,e}if("end"===r.type){const e=h(r);if(e!==ee)return e} +if("illegal"===r.type&&""===o)return 1 +;if(R>1e5&&R>3*r.index)throw Error("potential infinite loop, way more iterations than matches") +;return A+=o,o.length}const w=v(e) +;if(!w)throw K(o.replace("{}",e)),Error('Unknown language: "'+e+'"') +;const O=Q(w);let k="",x=r||O;const M={},S=new p.__emitter(p);(()=>{const e=[] +;for(let n=x;n!==w;n=n.parent)n.scope&&e.unshift(n.scope) +;e.forEach((e=>S.openNode(e)))})();let A="",C=0,T=0,R=0,D=!1;try{ +if(w.__emitTokens)w.__emitTokens(t,S);else{for(x.matcher.considerAll();;){ +R++,D?D=!1:x.matcher.considerAll(),x.matcher.lastIndex=T +;const e=x.matcher.exec(t);if(!e)break;const n=N(t.substring(T,e.index),e) +;T=e.index+n}N(t.substring(T))}return S.finalize(),k=S.toHTML(),{language:e, +value:k,relevance:C,illegal:!1,_emitter:S,_top:x}}catch(n){ +if(n.message&&n.message.includes("Illegal"))return{language:e,value:J(t), +illegal:!0,relevance:0,_illegalBy:{message:n.message,index:T, +context:t.slice(T-100,T+100),mode:n.mode,resultSoFar:k},_emitter:S};if(s)return{ +language:e,value:J(t),illegal:!1,relevance:0,errorRaised:n,_emitter:S,_top:x} +;throw n}}function E(e,n){n=n||p.languages||Object.keys(a);const t=(e=>{ +const n={value:J(e),illegal:!1,relevance:0,_top:c,_emitter:new p.__emitter(p)} +;return n._emitter.addText(e),n})(e),i=n.filter(v).filter(k).map((n=>f(n,e,!1))) +;i.unshift(t);const r=i.sort(((e,n)=>{ +if(e.relevance!==n.relevance)return n.relevance-e.relevance +;if(e.language&&n.language){if(v(e.language).supersetOf===n.language)return 1 +;if(v(n.language).supersetOf===e.language)return-1}return 0})),[s,o]=r,l=s +;return l.secondBest=o,l}function y(e){let n=null;const t=(e=>{ +let n=e.className+" ";n+=e.parentNode?e.parentNode.className:"" +;const t=p.languageDetectRe.exec(n);if(t){const n=v(t[1]) +;return n||(H(o.replace("{}",t[1])), +H("Falling back to no-highlight mode for this block.",e)),n?t[1]:"no-highlight"} +return n.split(/\s+/).find((e=>_(e)||v(e)))})(e);if(_(t))return +;if(x("before:highlightElement",{el:e,language:t +}),e.dataset.highlighted)return void console.log("Element previously highlighted. To highlight again, first unset `dataset.highlighted`.",e) +;if(e.children.length>0&&(p.ignoreUnescapedHTML||(console.warn("One of your code blocks includes unescaped HTML. This is a potentially serious security risk."), +console.warn("https://github.com/highlightjs/highlight.js/wiki/security"), +console.warn("The element with unescaped HTML:"), +console.warn(e)),p.throwUnescapedHTML))throw new V("One of your code blocks includes unescaped HTML.",e.innerHTML) +;n=e;const a=n.textContent,r=t?h(a,{language:t,ignoreIllegals:!0}):E(a) +;e.innerHTML=r.value,e.dataset.highlighted="yes",((e,n,t)=>{const a=n&&i[n]||t +;e.classList.add("hljs"),e.classList.add("language-"+a) +})(e,t,r.language),e.result={language:r.language,re:r.relevance, +relevance:r.relevance},r.secondBest&&(e.secondBest={ +language:r.secondBest.language,relevance:r.secondBest.relevance +}),x("after:highlightElement",{el:e,result:r,text:a})}let N=!1;function w(){ +"loading"!==document.readyState?document.querySelectorAll(p.cssSelector).forEach(y):N=!0 +}function v(e){return e=(e||"").toLowerCase(),a[e]||a[i[e]]} +function O(e,{languageName:n}){"string"==typeof e&&(e=[e]),e.forEach((e=>{ +i[e.toLowerCase()]=n}))}function k(e){const n=v(e) +;return n&&!n.disableAutodetect}function x(e,n){const t=e;r.forEach((e=>{ +e[t]&&e[t](n)}))} +"undefined"!=typeof window&&window.addEventListener&&window.addEventListener("DOMContentLoaded",(()=>{ +N&&w()}),!1),Object.assign(t,{highlight:h,highlightAuto:E,highlightAll:w, +highlightElement:y, +highlightBlock:e=>(q("10.7.0","highlightBlock will be removed entirely in v12.0"), +q("10.7.0","Please use highlightElement now."),y(e)),configure:e=>{p=Y(p,e)}, +initHighlighting:()=>{ +w(),q("10.6.0","initHighlighting() deprecated. Use highlightAll() now.")}, +initHighlightingOnLoad:()=>{ +w(),q("10.6.0","initHighlightingOnLoad() deprecated. Use highlightAll() now.") +},registerLanguage:(e,n)=>{let i=null;try{i=n(t)}catch(n){ +if(K("Language definition for '{}' could not be registered.".replace("{}",e)), +!s)throw n;K(n),i=c} +i.name||(i.name=e),a[e]=i,i.rawDefinition=n.bind(null,t),i.aliases&&O(i.aliases,{ +languageName:e})},unregisterLanguage:e=>{delete a[e] +;for(const n of Object.keys(i))i[n]===e&&delete i[n]}, +listLanguages:()=>Object.keys(a),getLanguage:v,registerAliases:O, +autoDetection:k,inherit:Y,addPlugin:e=>{(e=>{ +e["before:highlightBlock"]&&!e["before:highlightElement"]&&(e["before:highlightElement"]=n=>{ +e["before:highlightBlock"](Object.assign({block:n.el},n)) +}),e["after:highlightBlock"]&&!e["after:highlightElement"]&&(e["after:highlightElement"]=n=>{ +e["after:highlightBlock"](Object.assign({block:n.el},n))})})(e),r.push(e)}, +removePlugin:e=>{const n=r.indexOf(e);-1!==n&&r.splice(n,1)}}),t.debugMode=()=>{ +s=!1},t.safeMode=()=>{s=!0},t.versionString="11.9.0",t.regex={concat:b, +lookahead:d,either:m,optional:u,anyNumberOfTimes:g} +;for(const n in C)"object"==typeof C[n]&&e(C[n]);return Object.assign(t,C),t +},te=ne({});te.newInstance=()=>ne({});var ae=te;const ie=e=>({IMPORTANT:{ +scope:"meta",begin:"!important"},BLOCK_COMMENT:e.C_BLOCK_COMMENT_MODE,HEXCOLOR:{ +scope:"number",begin:/#(([0-9a-fA-F]{3,4})|(([0-9a-fA-F]{2}){3,4}))\b/}, +FUNCTION_DISPATCH:{className:"built_in",begin:/[\w-]+(?=\()/}, +ATTRIBUTE_SELECTOR_MODE:{scope:"selector-attr",begin:/\[/,end:/\]/,illegal:"$", +contains:[e.APOS_STRING_MODE,e.QUOTE_STRING_MODE]},CSS_NUMBER_MODE:{ +scope:"number", +begin:e.NUMBER_RE+"(%|em|ex|ch|rem|vw|vh|vmin|vmax|cm|mm|in|pt|pc|px|deg|grad|rad|turn|s|ms|Hz|kHz|dpi|dpcm|dppx)?", +relevance:0},CSS_VARIABLE:{className:"attr",begin:/--[A-Za-z_][A-Za-z0-9_-]*/} +}),re=["a","abbr","address","article","aside","audio","b","blockquote","body","button","canvas","caption","cite","code","dd","del","details","dfn","div","dl","dt","em","fieldset","figcaption","figure","footer","form","h1","h2","h3","h4","h5","h6","header","hgroup","html","i","iframe","img","input","ins","kbd","label","legend","li","main","mark","menu","nav","object","ol","p","q","quote","samp","section","span","strong","summary","sup","table","tbody","td","textarea","tfoot","th","thead","time","tr","ul","var","video"],se=["any-hover","any-pointer","aspect-ratio","color","color-gamut","color-index","device-aspect-ratio","device-height","device-width","display-mode","forced-colors","grid","height","hover","inverted-colors","monochrome","orientation","overflow-block","overflow-inline","pointer","prefers-color-scheme","prefers-contrast","prefers-reduced-motion","prefers-reduced-transparency","resolution","scan","scripting","update","width","min-width","max-width","min-height","max-height"],oe=["active","any-link","blank","checked","current","default","defined","dir","disabled","drop","empty","enabled","first","first-child","first-of-type","fullscreen","future","focus","focus-visible","focus-within","has","host","host-context","hover","indeterminate","in-range","invalid","is","lang","last-child","last-of-type","left","link","local-link","not","nth-child","nth-col","nth-last-child","nth-last-col","nth-last-of-type","nth-of-type","only-child","only-of-type","optional","out-of-range","past","placeholder-shown","read-only","read-write","required","right","root","scope","target","target-within","user-invalid","valid","visited","where"],le=["after","backdrop","before","cue","cue-region","first-letter","first-line","grammar-error","marker","part","placeholder","selection","slotted","spelling-error"],ce=["align-content","align-items","align-self","all","animation","animation-delay","animation-direction","animation-duration","animation-fill-mode","animation-iteration-count","animation-name","animation-play-state","animation-timing-function","backface-visibility","background","background-attachment","background-blend-mode","background-clip","background-color","background-image","background-origin","background-position","background-repeat","background-size","block-size","border","border-block","border-block-color","border-block-end","border-block-end-color","border-block-end-style","border-block-end-width","border-block-start","border-block-start-color","border-block-start-style","border-block-start-width","border-block-style","border-block-width","border-bottom","border-bottom-color","border-bottom-left-radius","border-bottom-right-radius","border-bottom-style","border-bottom-width","border-collapse","border-color","border-image","border-image-outset","border-image-repeat","border-image-slice","border-image-source","border-image-width","border-inline","border-inline-color","border-inline-end","border-inline-end-color","border-inline-end-style","border-inline-end-width","border-inline-start","border-inline-start-color","border-inline-start-style","border-inline-start-width","border-inline-style","border-inline-width","border-left","border-left-color","border-left-style","border-left-width","border-radius","border-right","border-right-color","border-right-style","border-right-width","border-spacing","border-style","border-top","border-top-color","border-top-left-radius","border-top-right-radius","border-top-style","border-top-width","border-width","bottom","box-decoration-break","box-shadow","box-sizing","break-after","break-before","break-inside","caption-side","caret-color","clear","clip","clip-path","clip-rule","color","column-count","column-fill","column-gap","column-rule","column-rule-color","column-rule-style","column-rule-width","column-span","column-width","columns","contain","content","content-visibility","counter-increment","counter-reset","cue","cue-after","cue-before","cursor","direction","display","empty-cells","filter","flex","flex-basis","flex-direction","flex-flow","flex-grow","flex-shrink","flex-wrap","float","flow","font","font-display","font-family","font-feature-settings","font-kerning","font-language-override","font-size","font-size-adjust","font-smoothing","font-stretch","font-style","font-synthesis","font-variant","font-variant-caps","font-variant-east-asian","font-variant-ligatures","font-variant-numeric","font-variant-position","font-variation-settings","font-weight","gap","glyph-orientation-vertical","grid","grid-area","grid-auto-columns","grid-auto-flow","grid-auto-rows","grid-column","grid-column-end","grid-column-start","grid-gap","grid-row","grid-row-end","grid-row-start","grid-template","grid-template-areas","grid-template-columns","grid-template-rows","hanging-punctuation","height","hyphens","icon","image-orientation","image-rendering","image-resolution","ime-mode","inline-size","isolation","justify-content","left","letter-spacing","line-break","line-height","list-style","list-style-image","list-style-position","list-style-type","margin","margin-block","margin-block-end","margin-block-start","margin-bottom","margin-inline","margin-inline-end","margin-inline-start","margin-left","margin-right","margin-top","marks","mask","mask-border","mask-border-mode","mask-border-outset","mask-border-repeat","mask-border-slice","mask-border-source","mask-border-width","mask-clip","mask-composite","mask-image","mask-mode","mask-origin","mask-position","mask-repeat","mask-size","mask-type","max-block-size","max-height","max-inline-size","max-width","min-block-size","min-height","min-inline-size","min-width","mix-blend-mode","nav-down","nav-index","nav-left","nav-right","nav-up","none","normal","object-fit","object-position","opacity","order","orphans","outline","outline-color","outline-offset","outline-style","outline-width","overflow","overflow-wrap","overflow-x","overflow-y","padding","padding-block","padding-block-end","padding-block-start","padding-bottom","padding-inline","padding-inline-end","padding-inline-start","padding-left","padding-right","padding-top","page-break-after","page-break-before","page-break-inside","pause","pause-after","pause-before","perspective","perspective-origin","pointer-events","position","quotes","resize","rest","rest-after","rest-before","right","row-gap","scroll-margin","scroll-margin-block","scroll-margin-block-end","scroll-margin-block-start","scroll-margin-bottom","scroll-margin-inline","scroll-margin-inline-end","scroll-margin-inline-start","scroll-margin-left","scroll-margin-right","scroll-margin-top","scroll-padding","scroll-padding-block","scroll-padding-block-end","scroll-padding-block-start","scroll-padding-bottom","scroll-padding-inline","scroll-padding-inline-end","scroll-padding-inline-start","scroll-padding-left","scroll-padding-right","scroll-padding-top","scroll-snap-align","scroll-snap-stop","scroll-snap-type","scrollbar-color","scrollbar-gutter","scrollbar-width","shape-image-threshold","shape-margin","shape-outside","speak","speak-as","src","tab-size","table-layout","text-align","text-align-all","text-align-last","text-combine-upright","text-decoration","text-decoration-color","text-decoration-line","text-decoration-style","text-emphasis","text-emphasis-color","text-emphasis-position","text-emphasis-style","text-indent","text-justify","text-orientation","text-overflow","text-rendering","text-shadow","text-transform","text-underline-position","top","transform","transform-box","transform-origin","transform-style","transition","transition-delay","transition-duration","transition-property","transition-timing-function","unicode-bidi","vertical-align","visibility","voice-balance","voice-duration","voice-family","voice-pitch","voice-range","voice-rate","voice-stress","voice-volume","white-space","widows","width","will-change","word-break","word-spacing","word-wrap","writing-mode","z-index"].reverse(),de=oe.concat(le) +;var ge="[0-9](_*[0-9])*",ue=`\\.(${ge})`,be="[0-9a-fA-F](_*[0-9a-fA-F])*",me={ +className:"number",variants:[{ +begin:`(\\b(${ge})((${ue})|\\.)?|(${ue}))[eE][+-]?(${ge})[fFdD]?\\b`},{ +begin:`\\b(${ge})((${ue})[fFdD]?\\b|\\.([fFdD]\\b)?)`},{ +begin:`(${ue})[fFdD]?\\b`},{begin:`\\b(${ge})[fFdD]\\b`},{ +begin:`\\b0[xX]((${be})\\.?|(${be})?\\.(${be}))[pP][+-]?(${ge})[fFdD]?\\b`},{ +begin:"\\b(0|[1-9](_*[0-9])*)[lL]?\\b"},{begin:`\\b0[xX](${be})[lL]?\\b`},{ +begin:"\\b0(_*[0-7])*[lL]?\\b"},{begin:"\\b0[bB][01](_*[01])*[lL]?\\b"}], +relevance:0};function pe(e,n,t){return-1===t?"":e.replace(n,(a=>pe(e,n,t-1)))} +const _e="[A-Za-z$_][0-9A-Za-z$_]*",he=["as","in","of","if","for","while","finally","var","new","function","do","return","void","else","break","catch","instanceof","with","throw","case","default","try","switch","continue","typeof","delete","let","yield","const","class","debugger","async","await","static","import","from","export","extends"],fe=["true","false","null","undefined","NaN","Infinity"],Ee=["Object","Function","Boolean","Symbol","Math","Date","Number","BigInt","String","RegExp","Array","Float32Array","Float64Array","Int8Array","Uint8Array","Uint8ClampedArray","Int16Array","Int32Array","Uint16Array","Uint32Array","BigInt64Array","BigUint64Array","Set","Map","WeakSet","WeakMap","ArrayBuffer","SharedArrayBuffer","Atomics","DataView","JSON","Promise","Generator","GeneratorFunction","AsyncFunction","Reflect","Proxy","Intl","WebAssembly"],ye=["Error","EvalError","InternalError","RangeError","ReferenceError","SyntaxError","TypeError","URIError"],Ne=["setInterval","setTimeout","clearInterval","clearTimeout","require","exports","eval","isFinite","isNaN","parseFloat","parseInt","decodeURI","decodeURIComponent","encodeURI","encodeURIComponent","escape","unescape"],we=["arguments","this","super","console","window","document","localStorage","sessionStorage","module","global"],ve=[].concat(Ne,Ee,ye) +;function Oe(e){const n=e.regex,t=_e,a={begin:/<[A-Za-z0-9\\._:-]+/, +end:/\/[A-Za-z0-9\\._:-]+>|\/>/,isTrulyOpeningTag:(e,n)=>{ +const t=e[0].length+e.index,a=e.input[t] +;if("<"===a||","===a)return void n.ignoreMatch();let i +;">"===a&&(((e,{after:n})=>{const t="",M={ +match:[/const|var|let/,/\s+/,t,/\s*/,/=\s*/,/(async\s*)?/,n.lookahead(x)], +keywords:"async",className:{1:"keyword",3:"title.function"},contains:[f]} +;return{name:"JavaScript",aliases:["js","jsx","mjs","cjs"],keywords:i,exports:{ +PARAMS_CONTAINS:h,CLASS_REFERENCE:y},illegal:/#(?![$_A-z])/, +contains:[e.SHEBANG({label:"shebang",binary:"node",relevance:5}),{ +label:"use_strict",className:"meta",relevance:10, +begin:/^\s*['"]use (strict|asm)['"]/ +},e.APOS_STRING_MODE,e.QUOTE_STRING_MODE,d,g,u,b,m,{match:/\$\d+/},l,y,{ +className:"attr",begin:t+n.lookahead(":"),relevance:0},M,{ +begin:"("+e.RE_STARTERS_RE+"|\\b(case|return|throw)\\b)\\s*", +keywords:"return throw case",relevance:0,contains:[m,e.REGEXP_MODE,{ +className:"function",begin:x,returnBegin:!0,end:"\\s*=>",contains:[{ +className:"params",variants:[{begin:e.UNDERSCORE_IDENT_RE,relevance:0},{ +className:null,begin:/\(\s*\)/,skip:!0},{begin:/\(/,end:/\)/,excludeBegin:!0, +excludeEnd:!0,keywords:i,contains:h}]}]},{begin:/,/,relevance:0},{match:/\s+/, +relevance:0},{variants:[{begin:"<>",end:""},{ +match:/<[A-Za-z0-9\\._:-]+\s*\/>/},{begin:a.begin, +"on:begin":a.isTrulyOpeningTag,end:a.end}],subLanguage:"xml",contains:[{ +begin:a.begin,end:a.end,skip:!0,contains:["self"]}]}]},N,{ +beginKeywords:"while if switch catch for"},{ +begin:"\\b(?!function)"+e.UNDERSCORE_IDENT_RE+"\\([^()]*(\\([^()]*(\\([^()]*\\)[^()]*)*\\)[^()]*)*\\)\\s*\\{", +returnBegin:!0,label:"func.def",contains:[f,e.inherit(e.TITLE_MODE,{begin:t, +className:"title.function"})]},{match:/\.\.\./,relevance:0},O,{match:"\\$"+t, +relevance:0},{match:[/\bconstructor(?=\s*\()/],className:{1:"title.function"}, +contains:[f]},w,{relevance:0,match:/\b[A-Z][A-Z_0-9]+\b/, +className:"variable.constant"},E,k,{match:/\$[(.]/}]}} +const ke=e=>b(/\b/,e,/\w$/.test(e)?/\b/:/\B/),xe=["Protocol","Type"].map(ke),Me=["init","self"].map(ke),Se=["Any","Self"],Ae=["actor","any","associatedtype","async","await",/as\?/,/as!/,"as","borrowing","break","case","catch","class","consume","consuming","continue","convenience","copy","default","defer","deinit","didSet","distributed","do","dynamic","each","else","enum","extension","fallthrough",/fileprivate\(set\)/,"fileprivate","final","for","func","get","guard","if","import","indirect","infix",/init\?/,/init!/,"inout",/internal\(set\)/,"internal","in","is","isolated","nonisolated","lazy","let","macro","mutating","nonmutating",/open\(set\)/,"open","operator","optional","override","postfix","precedencegroup","prefix",/private\(set\)/,"private","protocol",/public\(set\)/,"public","repeat","required","rethrows","return","set","some","static","struct","subscript","super","switch","throws","throw",/try\?/,/try!/,"try","typealias",/unowned\(safe\)/,/unowned\(unsafe\)/,"unowned","var","weak","where","while","willSet"],Ce=["false","nil","true"],Te=["assignment","associativity","higherThan","left","lowerThan","none","right"],Re=["#colorLiteral","#column","#dsohandle","#else","#elseif","#endif","#error","#file","#fileID","#fileLiteral","#filePath","#function","#if","#imageLiteral","#keyPath","#line","#selector","#sourceLocation","#warning"],De=["abs","all","any","assert","assertionFailure","debugPrint","dump","fatalError","getVaList","isKnownUniquelyReferenced","max","min","numericCast","pointwiseMax","pointwiseMin","precondition","preconditionFailure","print","readLine","repeatElement","sequence","stride","swap","swift_unboxFromSwiftValueWithType","transcode","type","unsafeBitCast","unsafeDowncast","withExtendedLifetime","withUnsafeMutablePointer","withUnsafePointer","withVaList","withoutActuallyEscaping","zip"],Ie=m(/[/=\-+!*%<>&|^~?]/,/[\u00A1-\u00A7]/,/[\u00A9\u00AB]/,/[\u00AC\u00AE]/,/[\u00B0\u00B1]/,/[\u00B6\u00BB\u00BF\u00D7\u00F7]/,/[\u2016-\u2017]/,/[\u2020-\u2027]/,/[\u2030-\u203E]/,/[\u2041-\u2053]/,/[\u2055-\u205E]/,/[\u2190-\u23FF]/,/[\u2500-\u2775]/,/[\u2794-\u2BFF]/,/[\u2E00-\u2E7F]/,/[\u3001-\u3003]/,/[\u3008-\u3020]/,/[\u3030]/),Le=m(Ie,/[\u0300-\u036F]/,/[\u1DC0-\u1DFF]/,/[\u20D0-\u20FF]/,/[\uFE00-\uFE0F]/,/[\uFE20-\uFE2F]/),Be=b(Ie,Le,"*"),$e=m(/[a-zA-Z_]/,/[\u00A8\u00AA\u00AD\u00AF\u00B2-\u00B5\u00B7-\u00BA]/,/[\u00BC-\u00BE\u00C0-\u00D6\u00D8-\u00F6\u00F8-\u00FF]/,/[\u0100-\u02FF\u0370-\u167F\u1681-\u180D\u180F-\u1DBF]/,/[\u1E00-\u1FFF]/,/[\u200B-\u200D\u202A-\u202E\u203F-\u2040\u2054\u2060-\u206F]/,/[\u2070-\u20CF\u2100-\u218F\u2460-\u24FF\u2776-\u2793]/,/[\u2C00-\u2DFF\u2E80-\u2FFF]/,/[\u3004-\u3007\u3021-\u302F\u3031-\u303F\u3040-\uD7FF]/,/[\uF900-\uFD3D\uFD40-\uFDCF\uFDF0-\uFE1F\uFE30-\uFE44]/,/[\uFE47-\uFEFE\uFF00-\uFFFD]/),ze=m($e,/\d/,/[\u0300-\u036F\u1DC0-\u1DFF\u20D0-\u20FF\uFE20-\uFE2F]/),Fe=b($e,ze,"*"),Ue=b(/[A-Z]/,ze,"*"),je=["attached","autoclosure",b(/convention\(/,m("swift","block","c"),/\)/),"discardableResult","dynamicCallable","dynamicMemberLookup","escaping","freestanding","frozen","GKInspectable","IBAction","IBDesignable","IBInspectable","IBOutlet","IBSegueAction","inlinable","main","nonobjc","NSApplicationMain","NSCopying","NSManaged",b(/objc\(/,Fe,/\)/),"objc","objcMembers","propertyWrapper","requires_stored_property_inits","resultBuilder","Sendable","testable","UIApplicationMain","unchecked","unknown","usableFromInline","warn_unqualified_access"],Pe=["iOS","iOSApplicationExtension","macOS","macOSApplicationExtension","macCatalyst","macCatalystApplicationExtension","watchOS","watchOSApplicationExtension","tvOS","tvOSApplicationExtension","swift"] +;var Ke=Object.freeze({__proto__:null,grmr_bash:e=>{const n=e.regex,t={},a={ +begin:/\$\{/,end:/\}/,contains:["self",{begin:/:-/,contains:[t]}]} +;Object.assign(t,{className:"variable",variants:[{ +begin:n.concat(/\$[\w\d#@][\w\d_]*/,"(?![\\w\\d])(?![$])")},a]});const i={ +className:"subst",begin:/\$\(/,end:/\)/,contains:[e.BACKSLASH_ESCAPE]},r={ +begin:/<<-?\s*(?=\w+)/,starts:{contains:[e.END_SAME_AS_BEGIN({begin:/(\w+)/, +end:/(\w+)/,className:"string"})]}},s={className:"string",begin:/"/,end:/"/, +contains:[e.BACKSLASH_ESCAPE,t,i]};i.contains.push(s);const o={begin:/\$?\(\(/, +end:/\)\)/,contains:[{begin:/\d+#[0-9a-f]+/,className:"number"},e.NUMBER_MODE,t] +},l=e.SHEBANG({binary:"(fish|bash|zsh|sh|csh|ksh|tcsh|dash|scsh)",relevance:10 +}),c={className:"function",begin:/\w[\w\d_]*\s*\(\s*\)\s*\{/,returnBegin:!0, +contains:[e.inherit(e.TITLE_MODE,{begin:/\w[\w\d_]*/})],relevance:0};return{ +name:"Bash",aliases:["sh"],keywords:{$pattern:/\b[a-z][a-z0-9._-]+\b/, +keyword:["if","then","else","elif","fi","for","while","until","in","do","done","case","esac","function","select"], +literal:["true","false"], +built_in:["break","cd","continue","eval","exec","exit","export","getopts","hash","pwd","readonly","return","shift","test","times","trap","umask","unset","alias","bind","builtin","caller","command","declare","echo","enable","help","let","local","logout","mapfile","printf","read","readarray","source","type","typeset","ulimit","unalias","set","shopt","autoload","bg","bindkey","bye","cap","chdir","clone","comparguments","compcall","compctl","compdescribe","compfiles","compgroups","compquote","comptags","comptry","compvalues","dirs","disable","disown","echotc","echoti","emulate","fc","fg","float","functions","getcap","getln","history","integer","jobs","kill","limit","log","noglob","popd","print","pushd","pushln","rehash","sched","setcap","setopt","stat","suspend","ttyctl","unfunction","unhash","unlimit","unsetopt","vared","wait","whence","where","which","zcompile","zformat","zftp","zle","zmodload","zparseopts","zprof","zpty","zregexparse","zsocket","zstyle","ztcp","chcon","chgrp","chown","chmod","cp","dd","df","dir","dircolors","ln","ls","mkdir","mkfifo","mknod","mktemp","mv","realpath","rm","rmdir","shred","sync","touch","truncate","vdir","b2sum","base32","base64","cat","cksum","comm","csplit","cut","expand","fmt","fold","head","join","md5sum","nl","numfmt","od","paste","ptx","pr","sha1sum","sha224sum","sha256sum","sha384sum","sha512sum","shuf","sort","split","sum","tac","tail","tr","tsort","unexpand","uniq","wc","arch","basename","chroot","date","dirname","du","echo","env","expr","factor","groups","hostid","id","link","logname","nice","nohup","nproc","pathchk","pinky","printenv","printf","pwd","readlink","runcon","seq","sleep","stat","stdbuf","stty","tee","test","timeout","tty","uname","unlink","uptime","users","who","whoami","yes"] +},contains:[l,e.SHEBANG(),c,o,e.HASH_COMMENT_MODE,r,{match:/(\/[a-z._-]+)+/},s,{ +match:/\\"/},{className:"string",begin:/'/,end:/'/},{match:/\\'/},t]}}, +grmr_c:e=>{const n=e.regex,t=e.COMMENT("//","$",{contains:[{begin:/\\\n/}] +}),a="decltype\\(auto\\)",i="[a-zA-Z_]\\w*::",r="("+a+"|"+n.optional(i)+"[a-zA-Z_]\\w*"+n.optional("<[^<>]+>")+")",s={ +className:"type",variants:[{begin:"\\b[a-z\\d_]*_t\\b"},{ +match:/\batomic_[a-z]{3,6}\b/}]},o={className:"string",variants:[{ +begin:'(u8?|U|L)?"',end:'"',illegal:"\\n",contains:[e.BACKSLASH_ESCAPE]},{ +begin:"(u8?|U|L)?'(\\\\(x[0-9A-Fa-f]{2}|u[0-9A-Fa-f]{4,8}|[0-7]{3}|\\S)|.)", +end:"'",illegal:"."},e.END_SAME_AS_BEGIN({ +begin:/(?:u8?|U|L)?R"([^()\\ ]{0,16})\(/,end:/\)([^()\\ ]{0,16})"/})]},l={ +className:"number",variants:[{begin:"\\b(0b[01']+)"},{ +begin:"(-?)\\b([\\d']+(\\.[\\d']*)?|\\.[\\d']+)((ll|LL|l|L)(u|U)?|(u|U)(ll|LL|l|L)?|f|F|b|B)" +},{ +begin:"(-?)(\\b0[xX][a-fA-F0-9']+|(\\b[\\d']+(\\.[\\d']*)?|\\.[\\d']+)([eE][-+]?[\\d']+)?)" +}],relevance:0},c={className:"meta",begin:/#\s*[a-z]+\b/,end:/$/,keywords:{ +keyword:"if else elif endif define undef warning error line pragma _Pragma ifdef ifndef include" +},contains:[{begin:/\\\n/,relevance:0},e.inherit(o,{className:"string"}),{ +className:"string",begin:/<.*?>/},t,e.C_BLOCK_COMMENT_MODE]},d={ +className:"title",begin:n.optional(i)+e.IDENT_RE,relevance:0 +},g=n.optional(i)+e.IDENT_RE+"\\s*\\(",u={ +keyword:["asm","auto","break","case","continue","default","do","else","enum","extern","for","fortran","goto","if","inline","register","restrict","return","sizeof","struct","switch","typedef","union","volatile","while","_Alignas","_Alignof","_Atomic","_Generic","_Noreturn","_Static_assert","_Thread_local","alignas","alignof","noreturn","static_assert","thread_local","_Pragma"], +type:["float","double","signed","unsigned","int","short","long","char","void","_Bool","_Complex","_Imaginary","_Decimal32","_Decimal64","_Decimal128","const","static","complex","bool","imaginary"], +literal:"true false NULL", +built_in:"std string wstring cin cout cerr clog stdin stdout stderr stringstream istringstream ostringstream auto_ptr deque list queue stack vector map set pair bitset multiset multimap unordered_set unordered_map unordered_multiset unordered_multimap priority_queue make_pair array shared_ptr abort terminate abs acos asin atan2 atan calloc ceil cosh cos exit exp fabs floor fmod fprintf fputs free frexp fscanf future isalnum isalpha iscntrl isdigit isgraph islower isprint ispunct isspace isupper isxdigit tolower toupper labs ldexp log10 log malloc realloc memchr memcmp memcpy memset modf pow printf putchar puts scanf sinh sin snprintf sprintf sqrt sscanf strcat strchr strcmp strcpy strcspn strlen strncat strncmp strncpy strpbrk strrchr strspn strstr tanh tan vfprintf vprintf vsprintf endl initializer_list unique_ptr" +},b=[c,s,t,e.C_BLOCK_COMMENT_MODE,l,o],m={variants:[{begin:/=/,end:/;/},{ +begin:/\(/,end:/\)/},{beginKeywords:"new throw return else",end:/;/}], +keywords:u,contains:b.concat([{begin:/\(/,end:/\)/,keywords:u, +contains:b.concat(["self"]),relevance:0}]),relevance:0},p={ +begin:"("+r+"[\\*&\\s]+)+"+g,returnBegin:!0,end:/[{;=]/,excludeEnd:!0, +keywords:u,illegal:/[^\w\s\*&:<>.]/,contains:[{begin:a,keywords:u,relevance:0},{ +begin:g,returnBegin:!0,contains:[e.inherit(d,{className:"title.function"})], +relevance:0},{relevance:0,match:/,/},{className:"params",begin:/\(/,end:/\)/, +keywords:u,relevance:0,contains:[t,e.C_BLOCK_COMMENT_MODE,o,l,s,{begin:/\(/, +end:/\)/,keywords:u,relevance:0,contains:["self",t,e.C_BLOCK_COMMENT_MODE,o,l,s] +}]},s,t,e.C_BLOCK_COMMENT_MODE,c]};return{name:"C",aliases:["h"],keywords:u, +disableAutodetect:!0,illegal:"=]/,contains:[{ +beginKeywords:"final class struct"},e.TITLE_MODE]}]),exports:{preprocessor:c, +strings:o,keywords:u}}},grmr_cpp:e=>{const n=e.regex,t=e.COMMENT("//","$",{ +contains:[{begin:/\\\n/}] +}),a="decltype\\(auto\\)",i="[a-zA-Z_]\\w*::",r="(?!struct)("+a+"|"+n.optional(i)+"[a-zA-Z_]\\w*"+n.optional("<[^<>]+>")+")",s={ +className:"type",begin:"\\b[a-z\\d_]*_t\\b"},o={className:"string",variants:[{ +begin:'(u8?|U|L)?"',end:'"',illegal:"\\n",contains:[e.BACKSLASH_ESCAPE]},{ +begin:"(u8?|U|L)?'(\\\\(x[0-9A-Fa-f]{2}|u[0-9A-Fa-f]{4,8}|[0-7]{3}|\\S)|.)", +end:"'",illegal:"."},e.END_SAME_AS_BEGIN({ +begin:/(?:u8?|U|L)?R"([^()\\ ]{0,16})\(/,end:/\)([^()\\ ]{0,16})"/})]},l={ +className:"number",variants:[{begin:"\\b(0b[01']+)"},{ +begin:"(-?)\\b([\\d']+(\\.[\\d']*)?|\\.[\\d']+)((ll|LL|l|L)(u|U)?|(u|U)(ll|LL|l|L)?|f|F|b|B)" +},{ +begin:"(-?)(\\b0[xX][a-fA-F0-9']+|(\\b[\\d']+(\\.[\\d']*)?|\\.[\\d']+)([eE][-+]?[\\d']+)?)" +}],relevance:0},c={className:"meta",begin:/#\s*[a-z]+\b/,end:/$/,keywords:{ +keyword:"if else elif endif define undef warning error line pragma _Pragma ifdef ifndef include" +},contains:[{begin:/\\\n/,relevance:0},e.inherit(o,{className:"string"}),{ +className:"string",begin:/<.*?>/},t,e.C_BLOCK_COMMENT_MODE]},d={ +className:"title",begin:n.optional(i)+e.IDENT_RE,relevance:0 +},g=n.optional(i)+e.IDENT_RE+"\\s*\\(",u={ +type:["bool","char","char16_t","char32_t","char8_t","double","float","int","long","short","void","wchar_t","unsigned","signed","const","static"], +keyword:["alignas","alignof","and","and_eq","asm","atomic_cancel","atomic_commit","atomic_noexcept","auto","bitand","bitor","break","case","catch","class","co_await","co_return","co_yield","compl","concept","const_cast|10","consteval","constexpr","constinit","continue","decltype","default","delete","do","dynamic_cast|10","else","enum","explicit","export","extern","false","final","for","friend","goto","if","import","inline","module","mutable","namespace","new","noexcept","not","not_eq","nullptr","operator","or","or_eq","override","private","protected","public","reflexpr","register","reinterpret_cast|10","requires","return","sizeof","static_assert","static_cast|10","struct","switch","synchronized","template","this","thread_local","throw","transaction_safe","transaction_safe_dynamic","true","try","typedef","typeid","typename","union","using","virtual","volatile","while","xor","xor_eq"], +literal:["NULL","false","nullopt","nullptr","true"],built_in:["_Pragma"], +_type_hints:["any","auto_ptr","barrier","binary_semaphore","bitset","complex","condition_variable","condition_variable_any","counting_semaphore","deque","false_type","future","imaginary","initializer_list","istringstream","jthread","latch","lock_guard","multimap","multiset","mutex","optional","ostringstream","packaged_task","pair","promise","priority_queue","queue","recursive_mutex","recursive_timed_mutex","scoped_lock","set","shared_future","shared_lock","shared_mutex","shared_timed_mutex","shared_ptr","stack","string_view","stringstream","timed_mutex","thread","true_type","tuple","unique_lock","unique_ptr","unordered_map","unordered_multimap","unordered_multiset","unordered_set","variant","vector","weak_ptr","wstring","wstring_view"] +},b={className:"function.dispatch",relevance:0,keywords:{ +_hint:["abort","abs","acos","apply","as_const","asin","atan","atan2","calloc","ceil","cerr","cin","clog","cos","cosh","cout","declval","endl","exchange","exit","exp","fabs","floor","fmod","forward","fprintf","fputs","free","frexp","fscanf","future","invoke","isalnum","isalpha","iscntrl","isdigit","isgraph","islower","isprint","ispunct","isspace","isupper","isxdigit","labs","launder","ldexp","log","log10","make_pair","make_shared","make_shared_for_overwrite","make_tuple","make_unique","malloc","memchr","memcmp","memcpy","memset","modf","move","pow","printf","putchar","puts","realloc","scanf","sin","sinh","snprintf","sprintf","sqrt","sscanf","std","stderr","stdin","stdout","strcat","strchr","strcmp","strcpy","strcspn","strlen","strncat","strncmp","strncpy","strpbrk","strrchr","strspn","strstr","swap","tan","tanh","terminate","to_underlying","tolower","toupper","vfprintf","visit","vprintf","vsprintf"] +}, +begin:n.concat(/\b/,/(?!decltype)/,/(?!if)/,/(?!for)/,/(?!switch)/,/(?!while)/,e.IDENT_RE,n.lookahead(/(<[^<>]+>|)\s*\(/)) +},m=[b,c,s,t,e.C_BLOCK_COMMENT_MODE,l,o],p={variants:[{begin:/=/,end:/;/},{ +begin:/\(/,end:/\)/},{beginKeywords:"new throw return else",end:/;/}], +keywords:u,contains:m.concat([{begin:/\(/,end:/\)/,keywords:u, +contains:m.concat(["self"]),relevance:0}]),relevance:0},_={className:"function", +begin:"("+r+"[\\*&\\s]+)+"+g,returnBegin:!0,end:/[{;=]/,excludeEnd:!0, +keywords:u,illegal:/[^\w\s\*&:<>.]/,contains:[{begin:a,keywords:u,relevance:0},{ +begin:g,returnBegin:!0,contains:[d],relevance:0},{begin:/::/,relevance:0},{ +begin:/:/,endsWithParent:!0,contains:[o,l]},{relevance:0,match:/,/},{ +className:"params",begin:/\(/,end:/\)/,keywords:u,relevance:0, +contains:[t,e.C_BLOCK_COMMENT_MODE,o,l,s,{begin:/\(/,end:/\)/,keywords:u, +relevance:0,contains:["self",t,e.C_BLOCK_COMMENT_MODE,o,l,s]}] +},s,t,e.C_BLOCK_COMMENT_MODE,c]};return{name:"C++", +aliases:["cc","c++","h++","hpp","hh","hxx","cxx"],keywords:u,illegal:"",keywords:u,contains:["self",s]},{begin:e.IDENT_RE+"::",keywords:u},{ +match:[/\b(?:enum(?:\s+(?:class|struct))?|class|struct|union)/,/\s+/,/\w+/], +className:{1:"keyword",3:"title.class"}}])}},grmr_csharp:e=>{const n={ +keyword:["abstract","as","base","break","case","catch","class","const","continue","do","else","event","explicit","extern","finally","fixed","for","foreach","goto","if","implicit","in","interface","internal","is","lock","namespace","new","operator","out","override","params","private","protected","public","readonly","record","ref","return","scoped","sealed","sizeof","stackalloc","static","struct","switch","this","throw","try","typeof","unchecked","unsafe","using","virtual","void","volatile","while"].concat(["add","alias","and","ascending","async","await","by","descending","equals","from","get","global","group","init","into","join","let","nameof","not","notnull","on","or","orderby","partial","remove","select","set","unmanaged","value|0","var","when","where","with","yield"]), +built_in:["bool","byte","char","decimal","delegate","double","dynamic","enum","float","int","long","nint","nuint","object","sbyte","short","string","ulong","uint","ushort"], +literal:["default","false","null","true"]},t=e.inherit(e.TITLE_MODE,{ +begin:"[a-zA-Z](\\.?\\w)*"}),a={className:"number",variants:[{ +begin:"\\b(0b[01']+)"},{ +begin:"(-?)\\b([\\d']+(\\.[\\d']*)?|\\.[\\d']+)(u|U|l|L|ul|UL|f|F|b|B)"},{ +begin:"(-?)(\\b0[xX][a-fA-F0-9']+|(\\b[\\d']+(\\.[\\d']*)?|\\.[\\d']+)([eE][-+]?[\\d']+)?)" +}],relevance:0},i={className:"string",begin:'@"',end:'"',contains:[{begin:'""'}] +},r=e.inherit(i,{illegal:/\n/}),s={className:"subst",begin:/\{/,end:/\}/, +keywords:n},o=e.inherit(s,{illegal:/\n/}),l={className:"string",begin:/\$"/, +end:'"',illegal:/\n/,contains:[{begin:/\{\{/},{begin:/\}\}/ +},e.BACKSLASH_ESCAPE,o]},c={className:"string",begin:/\$@"/,end:'"',contains:[{ +begin:/\{\{/},{begin:/\}\}/},{begin:'""'},s]},d=e.inherit(c,{illegal:/\n/, +contains:[{begin:/\{\{/},{begin:/\}\}/},{begin:'""'},o]}) +;s.contains=[c,l,i,e.APOS_STRING_MODE,e.QUOTE_STRING_MODE,a,e.C_BLOCK_COMMENT_MODE], +o.contains=[d,l,r,e.APOS_STRING_MODE,e.QUOTE_STRING_MODE,a,e.inherit(e.C_BLOCK_COMMENT_MODE,{ +illegal:/\n/})];const g={variants:[c,l,i,e.APOS_STRING_MODE,e.QUOTE_STRING_MODE] +},u={begin:"<",end:">",contains:[{beginKeywords:"in out"},t] +},b=e.IDENT_RE+"(<"+e.IDENT_RE+"(\\s*,\\s*"+e.IDENT_RE+")*>)?(\\[\\])?",m={ +begin:"@"+e.IDENT_RE,relevance:0};return{name:"C#",aliases:["cs","c#"], +keywords:n,illegal:/::/,contains:[e.COMMENT("///","$",{returnBegin:!0, +contains:[{className:"doctag",variants:[{begin:"///",relevance:0},{ +begin:"\x3c!--|--\x3e"},{begin:""}]}] +}),e.C_LINE_COMMENT_MODE,e.C_BLOCK_COMMENT_MODE,{className:"meta",begin:"#", +end:"$",keywords:{ +keyword:"if else elif endif define undef warning error line region endregion pragma checksum" +}},g,a,{beginKeywords:"class interface",relevance:0,end:/[{;=]/, +illegal:/[^\s:,]/,contains:[{beginKeywords:"where class" +},t,u,e.C_LINE_COMMENT_MODE,e.C_BLOCK_COMMENT_MODE]},{beginKeywords:"namespace", +relevance:0,end:/[{;=]/,illegal:/[^\s:]/, +contains:[t,e.C_LINE_COMMENT_MODE,e.C_BLOCK_COMMENT_MODE]},{ +beginKeywords:"record",relevance:0,end:/[{;=]/,illegal:/[^\s:]/, +contains:[t,u,e.C_LINE_COMMENT_MODE,e.C_BLOCK_COMMENT_MODE]},{className:"meta", +begin:"^\\s*\\[(?=[\\w])",excludeBegin:!0,end:"\\]",excludeEnd:!0,contains:[{ +className:"string",begin:/"/,end:/"/}]},{ +beginKeywords:"new return throw await else",relevance:0},{className:"function", +begin:"("+b+"\\s+)+"+e.IDENT_RE+"\\s*(<[^=]+>\\s*)?\\(",returnBegin:!0, +end:/\s*[{;=]/,excludeEnd:!0,keywords:n,contains:[{ +beginKeywords:"public private protected static internal protected abstract async extern override unsafe virtual new sealed partial", +relevance:0},{begin:e.IDENT_RE+"\\s*(<[^=]+>\\s*)?\\(",returnBegin:!0, +contains:[e.TITLE_MODE,u],relevance:0},{match:/\(\)/},{className:"params", +begin:/\(/,end:/\)/,excludeBegin:!0,excludeEnd:!0,keywords:n,relevance:0, +contains:[g,a,e.C_BLOCK_COMMENT_MODE] +},e.C_LINE_COMMENT_MODE,e.C_BLOCK_COMMENT_MODE]},m]}},grmr_css:e=>{ +const n=e.regex,t=ie(e),a=[e.APOS_STRING_MODE,e.QUOTE_STRING_MODE];return{ +name:"CSS",case_insensitive:!0,illegal:/[=|'\$]/,keywords:{ +keyframePosition:"from to"},classNameAliases:{keyframePosition:"selector-tag"}, +contains:[t.BLOCK_COMMENT,{begin:/-(webkit|moz|ms|o)-(?=[a-z])/ +},t.CSS_NUMBER_MODE,{className:"selector-id",begin:/#[A-Za-z0-9_-]+/,relevance:0 +},{className:"selector-class",begin:"\\.[a-zA-Z-][a-zA-Z0-9_-]*",relevance:0 +},t.ATTRIBUTE_SELECTOR_MODE,{className:"selector-pseudo",variants:[{ +begin:":("+oe.join("|")+")"},{begin:":(:)?("+le.join("|")+")"}] +},t.CSS_VARIABLE,{className:"attribute",begin:"\\b("+ce.join("|")+")\\b"},{ +begin:/:/,end:/[;}{]/, +contains:[t.BLOCK_COMMENT,t.HEXCOLOR,t.IMPORTANT,t.CSS_NUMBER_MODE,...a,{ +begin:/(url|data-uri)\(/,end:/\)/,relevance:0,keywords:{built_in:"url data-uri" +},contains:[...a,{className:"string",begin:/[^)]/,endsWithParent:!0, +excludeEnd:!0}]},t.FUNCTION_DISPATCH]},{begin:n.lookahead(/@/),end:"[{;]", +relevance:0,illegal:/:/,contains:[{className:"keyword",begin:/@-?\w[\w]*(-\w+)*/ +},{begin:/\s/,endsWithParent:!0,excludeEnd:!0,relevance:0,keywords:{ +$pattern:/[a-z-]+/,keyword:"and or not only",attribute:se.join(" ")},contains:[{ +begin:/[a-z-]+(?=:)/,className:"attribute"},...a,t.CSS_NUMBER_MODE]}]},{ +className:"selector-tag",begin:"\\b("+re.join("|")+")\\b"}]}},grmr_diff:e=>{ +const n=e.regex;return{name:"Diff",aliases:["patch"],contains:[{ +className:"meta",relevance:10, +match:n.either(/^@@ +-\d+,\d+ +\+\d+,\d+ +@@/,/^\*\*\* +\d+,\d+ +\*\*\*\*$/,/^--- +\d+,\d+ +----$/) +},{className:"comment",variants:[{ +begin:n.either(/Index: /,/^index/,/={3,}/,/^-{3}/,/^\*{3} /,/^\+{3}/,/^diff --git/), +end:/$/},{match:/^\*{15}$/}]},{className:"addition",begin:/^\+/,end:/$/},{ +className:"deletion",begin:/^-/,end:/$/},{className:"addition",begin:/^!/, +end:/$/}]}},grmr_go:e=>{const n={ +keyword:["break","case","chan","const","continue","default","defer","else","fallthrough","for","func","go","goto","if","import","interface","map","package","range","return","select","struct","switch","type","var"], +type:["bool","byte","complex64","complex128","error","float32","float64","int8","int16","int32","int64","string","uint8","uint16","uint32","uint64","int","uint","uintptr","rune"], +literal:["true","false","iota","nil"], +built_in:["append","cap","close","complex","copy","imag","len","make","new","panic","print","println","real","recover","delete"] +};return{name:"Go",aliases:["golang"],keywords:n,illegal:"{const n=e.regex;return{name:"GraphQL",aliases:["gql"], +case_insensitive:!0,disableAutodetect:!1,keywords:{ +keyword:["query","mutation","subscription","type","input","schema","directive","interface","union","scalar","fragment","enum","on"], +literal:["true","false","null"]}, +contains:[e.HASH_COMMENT_MODE,e.QUOTE_STRING_MODE,e.NUMBER_MODE,{ +scope:"punctuation",match:/[.]{3}/,relevance:0},{scope:"punctuation", +begin:/[\!\(\)\:\=\[\]\{\|\}]{1}/,relevance:0},{scope:"variable",begin:/\$/, +end:/\W/,excludeEnd:!0,relevance:0},{scope:"meta",match:/@\w+/,excludeEnd:!0},{ +scope:"symbol",begin:n.concat(/[_A-Za-z][_0-9A-Za-z]*/,n.lookahead(/\s*:/)), +relevance:0}],illegal:[/[;<']/,/BEGIN/]}},grmr_ini:e=>{const n=e.regex,t={ +className:"number",relevance:0,variants:[{begin:/([+-]+)?[\d]+_[\d_]+/},{ +begin:e.NUMBER_RE}]},a=e.COMMENT();a.variants=[{begin:/;/,end:/$/},{begin:/#/, +end:/$/}];const i={className:"variable",variants:[{begin:/\$[\w\d"][\w\d_]*/},{ +begin:/\$\{(.*?)\}/}]},r={className:"literal", +begin:/\bon|off|true|false|yes|no\b/},s={className:"string", +contains:[e.BACKSLASH_ESCAPE],variants:[{begin:"'''",end:"'''",relevance:10},{ +begin:'"""',end:'"""',relevance:10},{begin:'"',end:'"'},{begin:"'",end:"'"}] +},o={begin:/\[/,end:/\]/,contains:[a,r,i,s,t,"self"],relevance:0 +},l=n.either(/[A-Za-z0-9_-]+/,/"(\\"|[^"])*"/,/'[^']*'/);return{ +name:"TOML, also INI",aliases:["toml"],case_insensitive:!0,illegal:/\S/, +contains:[a,{className:"section",begin:/\[+/,end:/\]+/},{ +begin:n.concat(l,"(\\s*\\.\\s*",l,")*",n.lookahead(/\s*=\s*[^#\s]/)), +className:"attr",starts:{end:/$/,contains:[a,o,r,i,s,t]}}]}},grmr_java:e=>{ +const n=e.regex,t="[\xc0-\u02b8a-zA-Z_$][\xc0-\u02b8a-zA-Z_$0-9]*",a=t+pe("(?:<"+t+"~~~(?:\\s*,\\s*"+t+"~~~)*>)?",/~~~/g,2),i={ +keyword:["synchronized","abstract","private","var","static","if","const ","for","while","strictfp","finally","protected","import","native","final","void","enum","else","break","transient","catch","instanceof","volatile","case","assert","package","default","public","try","switch","continue","throws","protected","public","private","module","requires","exports","do","sealed","yield","permits"], +literal:["false","true","null"], +type:["char","boolean","long","float","int","byte","short","double"], +built_in:["super","this"]},r={className:"meta",begin:"@"+t,contains:[{ +begin:/\(/,end:/\)/,contains:["self"]}]},s={className:"params",begin:/\(/, +end:/\)/,keywords:i,relevance:0,contains:[e.C_BLOCK_COMMENT_MODE],endsParent:!0} +;return{name:"Java",aliases:["jsp"],keywords:i,illegal:/<\/|#/, +contains:[e.COMMENT("/\\*\\*","\\*/",{relevance:0,contains:[{begin:/\w+@/, +relevance:0},{className:"doctag",begin:"@[A-Za-z]+"}]}),{ +begin:/import java\.[a-z]+\./,keywords:"import",relevance:2 +},e.C_LINE_COMMENT_MODE,e.C_BLOCK_COMMENT_MODE,{begin:/"""/,end:/"""/, +className:"string",contains:[e.BACKSLASH_ESCAPE] +},e.APOS_STRING_MODE,e.QUOTE_STRING_MODE,{ +match:[/\b(?:class|interface|enum|extends|implements|new)/,/\s+/,t],className:{ +1:"keyword",3:"title.class"}},{match:/non-sealed/,scope:"keyword"},{ +begin:[n.concat(/(?!else)/,t),/\s+/,t,/\s+/,/=(?!=)/],className:{1:"type", +3:"variable",5:"operator"}},{begin:[/record/,/\s+/,t],className:{1:"keyword", +3:"title.class"},contains:[s,e.C_LINE_COMMENT_MODE,e.C_BLOCK_COMMENT_MODE]},{ +beginKeywords:"new throw return else",relevance:0},{ +begin:["(?:"+a+"\\s+)",e.UNDERSCORE_IDENT_RE,/\s*(?=\()/],className:{ +2:"title.function"},keywords:i,contains:[{className:"params",begin:/\(/, +end:/\)/,keywords:i,relevance:0, +contains:[r,e.APOS_STRING_MODE,e.QUOTE_STRING_MODE,me,e.C_BLOCK_COMMENT_MODE] +},e.C_LINE_COMMENT_MODE,e.C_BLOCK_COMMENT_MODE]},me,r]}},grmr_javascript:Oe, +grmr_json:e=>{const n=["true","false","null"],t={scope:"literal", +beginKeywords:n.join(" ")};return{name:"JSON",keywords:{literal:n},contains:[{ +className:"attr",begin:/"(\\.|[^\\"\r\n])*"(?=\s*:)/,relevance:1.01},{ +match:/[{}[\],:]/,className:"punctuation",relevance:0 +},e.QUOTE_STRING_MODE,t,e.C_NUMBER_MODE,e.C_LINE_COMMENT_MODE,e.C_BLOCK_COMMENT_MODE], +illegal:"\\S"}},grmr_kotlin:e=>{const n={ +keyword:"abstract as val var vararg get set class object open private protected public noinline crossinline dynamic final enum if else do while for when throw try catch finally import package is in fun override companion reified inline lateinit init interface annotation data sealed internal infix operator out by constructor super tailrec where const inner suspend typealias external expect actual", +built_in:"Byte Short Char Int Long Boolean Float Double Void Unit Nothing", +literal:"true false null"},t={className:"symbol",begin:e.UNDERSCORE_IDENT_RE+"@" +},a={className:"subst",begin:/\$\{/,end:/\}/,contains:[e.C_NUMBER_MODE]},i={ +className:"variable",begin:"\\$"+e.UNDERSCORE_IDENT_RE},r={className:"string", +variants:[{begin:'"""',end:'"""(?=[^"])',contains:[i,a]},{begin:"'",end:"'", +illegal:/\n/,contains:[e.BACKSLASH_ESCAPE]},{begin:'"',end:'"',illegal:/\n/, +contains:[e.BACKSLASH_ESCAPE,i,a]}]};a.contains.push(r);const s={ +className:"meta", +begin:"@(?:file|property|field|get|set|receiver|param|setparam|delegate)\\s*:(?:\\s*"+e.UNDERSCORE_IDENT_RE+")?" +},o={className:"meta",begin:"@"+e.UNDERSCORE_IDENT_RE,contains:[{begin:/\(/, +end:/\)/,contains:[e.inherit(r,{className:"string"}),"self"]}] +},l=me,c=e.COMMENT("/\\*","\\*/",{contains:[e.C_BLOCK_COMMENT_MODE]}),d={ +variants:[{className:"type",begin:e.UNDERSCORE_IDENT_RE},{begin:/\(/,end:/\)/, +contains:[]}]},g=d;return g.variants[1].contains=[d],d.variants[1].contains=[g], +{name:"Kotlin",aliases:["kt","kts"],keywords:n, +contains:[e.COMMENT("/\\*\\*","\\*/",{relevance:0,contains:[{className:"doctag", +begin:"@[A-Za-z]+"}]}),e.C_LINE_COMMENT_MODE,c,{className:"keyword", +begin:/\b(break|continue|return|this)\b/,starts:{contains:[{className:"symbol", +begin:/@\w+/}]}},t,s,o,{className:"function",beginKeywords:"fun",end:"[(]|$", +returnBegin:!0,excludeEnd:!0,keywords:n,relevance:5,contains:[{ +begin:e.UNDERSCORE_IDENT_RE+"\\s*\\(",returnBegin:!0,relevance:0, +contains:[e.UNDERSCORE_TITLE_MODE]},{className:"type",begin://, +keywords:"reified",relevance:0},{className:"params",begin:/\(/,end:/\)/, +endsParent:!0,keywords:n,relevance:0,contains:[{begin:/:/,end:/[=,\/]/, +endsWithParent:!0,contains:[d,e.C_LINE_COMMENT_MODE,c],relevance:0 +},e.C_LINE_COMMENT_MODE,c,s,o,r,e.C_NUMBER_MODE]},c]},{ +begin:[/class|interface|trait/,/\s+/,e.UNDERSCORE_IDENT_RE],beginScope:{ +3:"title.class"},keywords:"class interface trait",end:/[:\{(]|$/,excludeEnd:!0, +illegal:"extends implements",contains:[{ +beginKeywords:"public protected internal private constructor" +},e.UNDERSCORE_TITLE_MODE,{className:"type",begin://,excludeBegin:!0, +excludeEnd:!0,relevance:0},{className:"type",begin:/[,:]\s*/,end:/[<\(,){\s]|$/, +excludeBegin:!0,returnEnd:!0},s,o]},r,{className:"meta",begin:"^#!/usr/bin/env", +end:"$",illegal:"\n"},l]}},grmr_less:e=>{ +const n=ie(e),t=de,a="[\\w-]+",i="("+a+"|@\\{"+a+"\\})",r=[],s=[],o=e=>({ +className:"string",begin:"~?"+e+".*?"+e}),l=(e,n,t)=>({className:e,begin:n, +relevance:t}),c={$pattern:/[a-z-]+/,keyword:"and or not only", +attribute:se.join(" ")},d={begin:"\\(",end:"\\)",contains:s,keywords:c, +relevance:0} +;s.push(e.C_LINE_COMMENT_MODE,e.C_BLOCK_COMMENT_MODE,o("'"),o('"'),n.CSS_NUMBER_MODE,{ +begin:"(url|data-uri)\\(",starts:{className:"string",end:"[\\)\\n]", +excludeEnd:!0} +},n.HEXCOLOR,d,l("variable","@@?"+a,10),l("variable","@\\{"+a+"\\}"),l("built_in","~?`[^`]*?`"),{ +className:"attribute",begin:a+"\\s*:",end:":",returnBegin:!0,excludeEnd:!0 +},n.IMPORTANT,{beginKeywords:"and not"},n.FUNCTION_DISPATCH);const g=s.concat({ +begin:/\{/,end:/\}/,contains:r}),u={beginKeywords:"when",endsWithParent:!0, +contains:[{beginKeywords:"and not"}].concat(s)},b={begin:i+"\\s*:", +returnBegin:!0,end:/[;}]/,relevance:0,contains:[{begin:/-(webkit|moz|ms|o)-/ +},n.CSS_VARIABLE,{className:"attribute",begin:"\\b("+ce.join("|")+")\\b", +end:/(?=:)/,starts:{endsWithParent:!0,illegal:"[<=$]",relevance:0,contains:s}}] +},m={className:"keyword", +begin:"@(import|media|charset|font-face|(-[a-z]+-)?keyframes|supports|document|namespace|page|viewport|host)\\b", +starts:{end:"[;{}]",keywords:c,returnEnd:!0,contains:s,relevance:0}},p={ +className:"variable",variants:[{begin:"@"+a+"\\s*:",relevance:15},{begin:"@"+a +}],starts:{end:"[;}]",returnEnd:!0,contains:g}},_={variants:[{ +begin:"[\\.#:&\\[>]",end:"[;{}]"},{begin:i,end:/\{/}],returnBegin:!0, +returnEnd:!0,illegal:"[<='$\"]",relevance:0, +contains:[e.C_LINE_COMMENT_MODE,e.C_BLOCK_COMMENT_MODE,u,l("keyword","all\\b"),l("variable","@\\{"+a+"\\}"),{ +begin:"\\b("+re.join("|")+")\\b",className:"selector-tag" +},n.CSS_NUMBER_MODE,l("selector-tag",i,0),l("selector-id","#"+i),l("selector-class","\\."+i,0),l("selector-tag","&",0),n.ATTRIBUTE_SELECTOR_MODE,{ +className:"selector-pseudo",begin:":("+oe.join("|")+")"},{ +className:"selector-pseudo",begin:":(:)?("+le.join("|")+")"},{begin:/\(/, +end:/\)/,relevance:0,contains:g},{begin:"!important"},n.FUNCTION_DISPATCH]},h={ +begin:a+":(:)?"+`(${t.join("|")})`,returnBegin:!0,contains:[_]} +;return r.push(e.C_LINE_COMMENT_MODE,e.C_BLOCK_COMMENT_MODE,m,p,h,b,_,u,n.FUNCTION_DISPATCH), +{name:"Less",case_insensitive:!0,illegal:"[=>'/<($\"]",contains:r}}, +grmr_lua:e=>{const n="\\[=*\\[",t="\\]=*\\]",a={begin:n,end:t,contains:["self"] +},i=[e.COMMENT("--(?!"+n+")","$"),e.COMMENT("--"+n,t,{contains:[a],relevance:10 +})];return{name:"Lua",keywords:{$pattern:e.UNDERSCORE_IDENT_RE, +literal:"true false nil", +keyword:"and break do else elseif end for goto if in local not or repeat return then until while", +built_in:"_G _ENV _VERSION __index __newindex __mode __call __metatable __tostring __len __gc __add __sub __mul __div __mod __pow __concat __unm __eq __lt __le assert collectgarbage dofile error getfenv getmetatable ipairs load loadfile loadstring module next pairs pcall print rawequal rawget rawset require select setfenv setmetatable tonumber tostring type unpack xpcall arg self coroutine resume yield status wrap create running debug getupvalue debug sethook getmetatable gethook setmetatable setlocal traceback setfenv getinfo setupvalue getlocal getregistry getfenv io lines write close flush open output type read stderr stdin input stdout popen tmpfile math log max acos huge ldexp pi cos tanh pow deg tan cosh sinh random randomseed frexp ceil floor rad abs sqrt modf asin min mod fmod log10 atan2 exp sin atan os exit setlocale date getenv difftime remove time clock tmpname rename execute package preload loadlib loaded loaders cpath config path seeall string sub upper len gfind rep find match char dump gmatch reverse byte format gsub lower table setn insert getn foreachi maxn foreach concat sort remove" +},contains:i.concat([{className:"function",beginKeywords:"function",end:"\\)", +contains:[e.inherit(e.TITLE_MODE,{ +begin:"([_a-zA-Z]\\w*\\.)*([_a-zA-Z]\\w*:)?[_a-zA-Z]\\w*"}),{className:"params", +begin:"\\(",endsWithParent:!0,contains:i}].concat(i) +},e.C_NUMBER_MODE,e.APOS_STRING_MODE,e.QUOTE_STRING_MODE,{className:"string", +begin:n,end:t,contains:[a],relevance:5}])}},grmr_makefile:e=>{const n={ +className:"variable",variants:[{begin:"\\$\\("+e.UNDERSCORE_IDENT_RE+"\\)", +contains:[e.BACKSLASH_ESCAPE]},{begin:/\$[@%{ +const n={begin:/<\/?[A-Za-z_]/,end:">",subLanguage:"xml",relevance:0},t={ +variants:[{begin:/\[.+?\]\[.*?\]/,relevance:0},{ +begin:/\[.+?\]\(((data|javascript|mailto):|(?:http|ftp)s?:\/\/).*?\)/, +relevance:2},{ +begin:e.regex.concat(/\[.+?\]\(/,/[A-Za-z][A-Za-z0-9+.-]*/,/:\/\/.*?\)/), +relevance:2},{begin:/\[.+?\]\([./?&#].*?\)/,relevance:1},{ +begin:/\[.*?\]\(.*?\)/,relevance:0}],returnBegin:!0,contains:[{match:/\[(?=\])/ +},{className:"string",relevance:0,begin:"\\[",end:"\\]",excludeBegin:!0, +returnEnd:!0},{className:"link",relevance:0,begin:"\\]\\(",end:"\\)", +excludeBegin:!0,excludeEnd:!0},{className:"symbol",relevance:0,begin:"\\]\\[", +end:"\\]",excludeBegin:!0,excludeEnd:!0}]},a={className:"strong",contains:[], +variants:[{begin:/_{2}(?!\s)/,end:/_{2}/},{begin:/\*{2}(?!\s)/,end:/\*{2}/}] +},i={className:"emphasis",contains:[],variants:[{begin:/\*(?![*\s])/,end:/\*/},{ +begin:/_(?![_\s])/,end:/_/,relevance:0}]},r=e.inherit(a,{contains:[] +}),s=e.inherit(i,{contains:[]});a.contains.push(s),i.contains.push(r) +;let o=[n,t];return[a,i,r,s].forEach((e=>{e.contains=e.contains.concat(o) +})),o=o.concat(a,i),{name:"Markdown",aliases:["md","mkdown","mkd"],contains:[{ +className:"section",variants:[{begin:"^#{1,6}",end:"$",contains:o},{ +begin:"(?=^.+?\\n[=-]{2,}$)",contains:[{begin:"^[=-]*$"},{begin:"^",end:"\\n", +contains:o}]}]},n,{className:"bullet",begin:"^[ \t]*([*+-]|(\\d+\\.))(?=\\s+)", +end:"\\s+",excludeEnd:!0},a,i,{className:"quote",begin:"^>\\s+",contains:o, +end:"$"},{className:"code",variants:[{begin:"(`{3,})[^`](.|\\n)*?\\1`*[ ]*"},{ +begin:"(~{3,})[^~](.|\\n)*?\\1~*[ ]*"},{begin:"```",end:"```+[ ]*$"},{ +begin:"~~~",end:"~~~+[ ]*$"},{begin:"`.+?`"},{begin:"(?=^( {4}|\\t))", +contains:[{begin:"^( {4}|\\t)",end:"(\\n)$"}],relevance:0}]},{ +begin:"^[-\\*]{3,}",end:"$"},t,{begin:/^\[[^\n]+\]:/,returnBegin:!0,contains:[{ +className:"symbol",begin:/\[/,end:/\]/,excludeBegin:!0,excludeEnd:!0},{ +className:"link",begin:/:\s*/,end:/$/,excludeBegin:!0}]}]}},grmr_objectivec:e=>{ +const n=/[a-zA-Z@][a-zA-Z0-9_]*/,t={$pattern:n, +keyword:["@interface","@class","@protocol","@implementation"]};return{ +name:"Objective-C",aliases:["mm","objc","obj-c","obj-c++","objective-c++"], +keywords:{"variable.language":["this","super"],$pattern:n, +keyword:["while","export","sizeof","typedef","const","struct","for","union","volatile","static","mutable","if","do","return","goto","enum","else","break","extern","asm","case","default","register","explicit","typename","switch","continue","inline","readonly","assign","readwrite","self","@synchronized","id","typeof","nonatomic","IBOutlet","IBAction","strong","weak","copy","in","out","inout","bycopy","byref","oneway","__strong","__weak","__block","__autoreleasing","@private","@protected","@public","@try","@property","@end","@throw","@catch","@finally","@autoreleasepool","@synthesize","@dynamic","@selector","@optional","@required","@encode","@package","@import","@defs","@compatibility_alias","__bridge","__bridge_transfer","__bridge_retained","__bridge_retain","__covariant","__contravariant","__kindof","_Nonnull","_Nullable","_Null_unspecified","__FUNCTION__","__PRETTY_FUNCTION__","__attribute__","getter","setter","retain","unsafe_unretained","nonnull","nullable","null_unspecified","null_resettable","class","instancetype","NS_DESIGNATED_INITIALIZER","NS_UNAVAILABLE","NS_REQUIRES_SUPER","NS_RETURNS_INNER_POINTER","NS_INLINE","NS_AVAILABLE","NS_DEPRECATED","NS_ENUM","NS_OPTIONS","NS_SWIFT_UNAVAILABLE","NS_ASSUME_NONNULL_BEGIN","NS_ASSUME_NONNULL_END","NS_REFINED_FOR_SWIFT","NS_SWIFT_NAME","NS_SWIFT_NOTHROW","NS_DURING","NS_HANDLER","NS_ENDHANDLER","NS_VALUERETURN","NS_VOIDRETURN"], +literal:["false","true","FALSE","TRUE","nil","YES","NO","NULL"], +built_in:["dispatch_once_t","dispatch_queue_t","dispatch_sync","dispatch_async","dispatch_once"], +type:["int","float","char","unsigned","signed","short","long","double","wchar_t","unichar","void","bool","BOOL","id|0","_Bool"] +},illegal:"/,end:/$/,illegal:"\\n" +},e.C_LINE_COMMENT_MODE,e.C_BLOCK_COMMENT_MODE]},{className:"class", +begin:"("+t.keyword.join("|")+")\\b",end:/(\{|$)/,excludeEnd:!0,keywords:t, +contains:[e.UNDERSCORE_TITLE_MODE]},{begin:"\\."+e.UNDERSCORE_IDENT_RE, +relevance:0}]}},grmr_perl:e=>{const n=e.regex,t=/[dualxmsipngr]{0,12}/,a={ +$pattern:/[\w.]+/, +keyword:"abs accept alarm and atan2 bind binmode bless break caller chdir chmod chomp chop chown chr chroot close closedir connect continue cos crypt dbmclose dbmopen defined delete die do dump each else elsif endgrent endhostent endnetent endprotoent endpwent endservent eof eval exec exists exit exp fcntl fileno flock for foreach fork format formline getc getgrent getgrgid getgrnam gethostbyaddr gethostbyname gethostent getlogin getnetbyaddr getnetbyname getnetent getpeername getpgrp getpriority getprotobyname getprotobynumber getprotoent getpwent getpwnam getpwuid getservbyname getservbyport getservent getsockname getsockopt given glob gmtime goto grep gt hex if index int ioctl join keys kill last lc lcfirst length link listen local localtime log lstat lt ma map mkdir msgctl msgget msgrcv msgsnd my ne next no not oct open opendir or ord our pack package pipe pop pos print printf prototype push q|0 qq quotemeta qw qx rand read readdir readline readlink readpipe recv redo ref rename require reset return reverse rewinddir rindex rmdir say scalar seek seekdir select semctl semget semop send setgrent sethostent setnetent setpgrp setpriority setprotoent setpwent setservent setsockopt shift shmctl shmget shmread shmwrite shutdown sin sleep socket socketpair sort splice split sprintf sqrt srand stat state study sub substr symlink syscall sysopen sysread sysseek system syswrite tell telldir tie tied time times tr truncate uc ucfirst umask undef unless unlink unpack unshift untie until use utime values vec wait waitpid wantarray warn when while write x|0 xor y|0" +},i={className:"subst",begin:"[$@]\\{",end:"\\}",keywords:a},r={begin:/->\{/, +end:/\}/},s={variants:[{begin:/\$\d/},{ +begin:n.concat(/[$%@](\^\w\b|#\w+(::\w+)*|\{\w+\}|\w+(::\w*)*)/,"(?![A-Za-z])(?![@$%])") +},{begin:/[$%@][^\s\w{]/,relevance:0}] +},o=[e.BACKSLASH_ESCAPE,i,s],l=[/!/,/\//,/\|/,/\?/,/'/,/"/,/#/],c=(e,a,i="\\1")=>{ +const r="\\1"===i?i:n.concat(i,a) +;return n.concat(n.concat("(?:",e,")"),a,/(?:\\.|[^\\\/])*?/,r,/(?:\\.|[^\\\/])*?/,i,t) +},d=(e,a,i)=>n.concat(n.concat("(?:",e,")"),a,/(?:\\.|[^\\\/])*?/,i,t),g=[s,e.HASH_COMMENT_MODE,e.COMMENT(/^=\w/,/=cut/,{ +endsWithParent:!0}),r,{className:"string",contains:o,variants:[{ +begin:"q[qwxr]?\\s*\\(",end:"\\)",relevance:5},{begin:"q[qwxr]?\\s*\\[", +end:"\\]",relevance:5},{begin:"q[qwxr]?\\s*\\{",end:"\\}",relevance:5},{ +begin:"q[qwxr]?\\s*\\|",end:"\\|",relevance:5},{begin:"q[qwxr]?\\s*<",end:">", +relevance:5},{begin:"qw\\s+q",end:"q",relevance:5},{begin:"'",end:"'", +contains:[e.BACKSLASH_ESCAPE]},{begin:'"',end:'"'},{begin:"`",end:"`", +contains:[e.BACKSLASH_ESCAPE]},{begin:/\{\w+\}/,relevance:0},{ +begin:"-?\\w+\\s*=>",relevance:0}]},{className:"number", +begin:"(\\b0[0-7_]+)|(\\b0x[0-9a-fA-F_]+)|(\\b[1-9][0-9_]*(\\.[0-9_]+)?)|[0_]\\b", +relevance:0},{ +begin:"(\\/\\/|"+e.RE_STARTERS_RE+"|\\b(split|return|print|reverse|grep)\\b)\\s*", +keywords:"split return print reverse grep",relevance:0, +contains:[e.HASH_COMMENT_MODE,{className:"regexp",variants:[{ +begin:c("s|tr|y",n.either(...l,{capture:!0}))},{begin:c("s|tr|y","\\(","\\)")},{ +begin:c("s|tr|y","\\[","\\]")},{begin:c("s|tr|y","\\{","\\}")}],relevance:2},{ +className:"regexp",variants:[{begin:/(m|qr)\/\//,relevance:0},{ +begin:d("(?:m|qr)?",/\//,/\//)},{begin:d("m|qr",n.either(...l,{capture:!0 +}),/\1/)},{begin:d("m|qr",/\(/,/\)/)},{begin:d("m|qr",/\[/,/\]/)},{ +begin:d("m|qr",/\{/,/\}/)}]}]},{className:"function",beginKeywords:"sub", +end:"(\\s*\\(.*?\\))?[;{]",excludeEnd:!0,relevance:5,contains:[e.TITLE_MODE]},{ +begin:"-\\w\\b",relevance:0},{begin:"^__DATA__$",end:"^__END__$", +subLanguage:"mojolicious",contains:[{begin:"^@@.*",end:"$",className:"comment"}] +}];return i.contains=g,r.contains=g,{name:"Perl",aliases:["pl","pm"],keywords:a, +contains:g}},grmr_php:e=>{ +const n=e.regex,t=/(?![A-Za-z0-9])(?![$])/,a=n.concat(/[a-zA-Z_\x7f-\xff][a-zA-Z0-9_\x7f-\xff]*/,t),i=n.concat(/(\\?[A-Z][a-z0-9_\x7f-\xff]+|\\?[A-Z]+(?=[A-Z][a-z0-9_\x7f-\xff])){1,}/,t),r={ +scope:"variable",match:"\\$+"+a},s={scope:"subst",variants:[{begin:/\$\w+/},{ +begin:/\{\$/,end:/\}/}]},o=e.inherit(e.APOS_STRING_MODE,{illegal:null +}),l="[ \t\n]",c={scope:"string",variants:[e.inherit(e.QUOTE_STRING_MODE,{ +illegal:null,contains:e.QUOTE_STRING_MODE.contains.concat(s)}),o,{ +begin:/<<<[ \t]*(?:(\w+)|"(\w+)")\n/,end:/[ \t]*(\w+)\b/, +contains:e.QUOTE_STRING_MODE.contains.concat(s),"on:begin":(e,n)=>{ +n.data._beginMatch=e[1]||e[2]},"on:end":(e,n)=>{ +n.data._beginMatch!==e[1]&&n.ignoreMatch()}},e.END_SAME_AS_BEGIN({ +begin:/<<<[ \t]*'(\w+)'\n/,end:/[ \t]*(\w+)\b/})]},d={scope:"number",variants:[{ +begin:"\\b0[bB][01]+(?:_[01]+)*\\b"},{begin:"\\b0[oO][0-7]+(?:_[0-7]+)*\\b"},{ +begin:"\\b0[xX][\\da-fA-F]+(?:_[\\da-fA-F]+)*\\b"},{ +begin:"(?:\\b\\d+(?:_\\d+)*(\\.(?:\\d+(?:_\\d+)*))?|\\B\\.\\d+)(?:[eE][+-]?\\d+)?" +}],relevance:0 +},g=["false","null","true"],u=["__CLASS__","__DIR__","__FILE__","__FUNCTION__","__COMPILER_HALT_OFFSET__","__LINE__","__METHOD__","__NAMESPACE__","__TRAIT__","die","echo","exit","include","include_once","print","require","require_once","array","abstract","and","as","binary","bool","boolean","break","callable","case","catch","class","clone","const","continue","declare","default","do","double","else","elseif","empty","enddeclare","endfor","endforeach","endif","endswitch","endwhile","enum","eval","extends","final","finally","float","for","foreach","from","global","goto","if","implements","instanceof","insteadof","int","integer","interface","isset","iterable","list","match|0","mixed","new","never","object","or","private","protected","public","readonly","real","return","string","switch","throw","trait","try","unset","use","var","void","while","xor","yield"],b=["Error|0","AppendIterator","ArgumentCountError","ArithmeticError","ArrayIterator","ArrayObject","AssertionError","BadFunctionCallException","BadMethodCallException","CachingIterator","CallbackFilterIterator","CompileError","Countable","DirectoryIterator","DivisionByZeroError","DomainException","EmptyIterator","ErrorException","Exception","FilesystemIterator","FilterIterator","GlobIterator","InfiniteIterator","InvalidArgumentException","IteratorIterator","LengthException","LimitIterator","LogicException","MultipleIterator","NoRewindIterator","OutOfBoundsException","OutOfRangeException","OuterIterator","OverflowException","ParentIterator","ParseError","RangeException","RecursiveArrayIterator","RecursiveCachingIterator","RecursiveCallbackFilterIterator","RecursiveDirectoryIterator","RecursiveFilterIterator","RecursiveIterator","RecursiveIteratorIterator","RecursiveRegexIterator","RecursiveTreeIterator","RegexIterator","RuntimeException","SeekableIterator","SplDoublyLinkedList","SplFileInfo","SplFileObject","SplFixedArray","SplHeap","SplMaxHeap","SplMinHeap","SplObjectStorage","SplObserver","SplPriorityQueue","SplQueue","SplStack","SplSubject","SplTempFileObject","TypeError","UnderflowException","UnexpectedValueException","UnhandledMatchError","ArrayAccess","BackedEnum","Closure","Fiber","Generator","Iterator","IteratorAggregate","Serializable","Stringable","Throwable","Traversable","UnitEnum","WeakReference","WeakMap","Directory","__PHP_Incomplete_Class","parent","php_user_filter","self","static","stdClass"],m={ +keyword:u,literal:(e=>{const n=[];return e.forEach((e=>{ +n.push(e),e.toLowerCase()===e?n.push(e.toUpperCase()):n.push(e.toLowerCase()) +})),n})(g),built_in:b},p=e=>e.map((e=>e.replace(/\|\d+$/,""))),_={variants:[{ +match:[/new/,n.concat(l,"+"),n.concat("(?!",p(b).join("\\b|"),"\\b)"),i],scope:{ +1:"keyword",4:"title.class"}}]},h=n.concat(a,"\\b(?!\\()"),f={variants:[{ +match:[n.concat(/::/,n.lookahead(/(?!class\b)/)),h],scope:{2:"variable.constant" +}},{match:[/::/,/class/],scope:{2:"variable.language"}},{ +match:[i,n.concat(/::/,n.lookahead(/(?!class\b)/)),h],scope:{1:"title.class", +3:"variable.constant"}},{match:[i,n.concat("::",n.lookahead(/(?!class\b)/))], +scope:{1:"title.class"}},{match:[i,/::/,/class/],scope:{1:"title.class", +3:"variable.language"}}]},E={scope:"attr", +match:n.concat(a,n.lookahead(":"),n.lookahead(/(?!::)/))},y={relevance:0, +begin:/\(/,end:/\)/,keywords:m,contains:[E,r,f,e.C_BLOCK_COMMENT_MODE,c,d,_] +},N={relevance:0, +match:[/\b/,n.concat("(?!fn\\b|function\\b|",p(u).join("\\b|"),"|",p(b).join("\\b|"),"\\b)"),a,n.concat(l,"*"),n.lookahead(/(?=\()/)], +scope:{3:"title.function.invoke"},contains:[y]};y.contains.push(N) +;const w=[E,f,e.C_BLOCK_COMMENT_MODE,c,d,_];return{case_insensitive:!1, +keywords:m,contains:[{begin:n.concat(/#\[\s*/,i),beginScope:"meta",end:/]/, +endScope:"meta",keywords:{literal:g,keyword:["new","array"]},contains:[{ +begin:/\[/,end:/]/,keywords:{literal:g,keyword:["new","array"]}, +contains:["self",...w]},...w,{scope:"meta",match:i}] +},e.HASH_COMMENT_MODE,e.COMMENT("//","$"),e.COMMENT("/\\*","\\*/",{contains:[{ +scope:"doctag",match:"@[A-Za-z]+"}]}),{match:/__halt_compiler\(\);/, +keywords:"__halt_compiler",starts:{scope:"comment",end:e.MATCH_NOTHING_RE, +contains:[{match:/\?>/,scope:"meta",endsParent:!0}]}},{scope:"meta",variants:[{ +begin:/<\?php/,relevance:10},{begin:/<\?=/},{begin:/<\?/,relevance:.1},{ +begin:/\?>/}]},{scope:"variable.language",match:/\$this\b/},r,N,f,{ +match:[/const/,/\s/,a],scope:{1:"keyword",3:"variable.constant"}},_,{ +scope:"function",relevance:0,beginKeywords:"fn function",end:/[;{]/, +excludeEnd:!0,illegal:"[$%\\[]",contains:[{beginKeywords:"use" +},e.UNDERSCORE_TITLE_MODE,{begin:"=>",endsParent:!0},{scope:"params", +begin:"\\(",end:"\\)",excludeBegin:!0,excludeEnd:!0,keywords:m, +contains:["self",r,f,e.C_BLOCK_COMMENT_MODE,c,d]}]},{scope:"class",variants:[{ +beginKeywords:"enum",illegal:/[($"]/},{beginKeywords:"class interface trait", +illegal:/[:($"]/}],relevance:0,end:/\{/,excludeEnd:!0,contains:[{ +beginKeywords:"extends implements"},e.UNDERSCORE_TITLE_MODE]},{ +beginKeywords:"namespace",relevance:0,end:";",illegal:/[.']/, +contains:[e.inherit(e.UNDERSCORE_TITLE_MODE,{scope:"title.class"})]},{ +beginKeywords:"use",relevance:0,end:";",contains:[{ +match:/\b(as|const|function)\b/,scope:"keyword"},e.UNDERSCORE_TITLE_MODE]},c,d]} +},grmr_php_template:e=>({name:"PHP template",subLanguage:"xml",contains:[{ +begin:/<\?(php|=)?/,end:/\?>/,subLanguage:"php",contains:[{begin:"/\\*", +end:"\\*/",skip:!0},{begin:'b"',end:'"',skip:!0},{begin:"b'",end:"'",skip:!0 +},e.inherit(e.APOS_STRING_MODE,{illegal:null,className:null,contains:null, +skip:!0}),e.inherit(e.QUOTE_STRING_MODE,{illegal:null,className:null, +contains:null,skip:!0})]}]}),grmr_plaintext:e=>({name:"Plain text", +aliases:["text","txt"],disableAutodetect:!0}),grmr_python:e=>{ +const n=e.regex,t=/[\p{XID_Start}_]\p{XID_Continue}*/u,a=["and","as","assert","async","await","break","case","class","continue","def","del","elif","else","except","finally","for","from","global","if","import","in","is","lambda","match","nonlocal|10","not","or","pass","raise","return","try","while","with","yield"],i={ +$pattern:/[A-Za-z]\w+|__\w+__/,keyword:a, +built_in:["__import__","abs","all","any","ascii","bin","bool","breakpoint","bytearray","bytes","callable","chr","classmethod","compile","complex","delattr","dict","dir","divmod","enumerate","eval","exec","filter","float","format","frozenset","getattr","globals","hasattr","hash","help","hex","id","input","int","isinstance","issubclass","iter","len","list","locals","map","max","memoryview","min","next","object","oct","open","ord","pow","print","property","range","repr","reversed","round","set","setattr","slice","sorted","staticmethod","str","sum","super","tuple","type","vars","zip"], +literal:["__debug__","Ellipsis","False","None","NotImplemented","True"], +type:["Any","Callable","Coroutine","Dict","List","Literal","Generic","Optional","Sequence","Set","Tuple","Type","Union"] +},r={className:"meta",begin:/^(>>>|\.\.\.) /},s={className:"subst",begin:/\{/, +end:/\}/,keywords:i,illegal:/#/},o={begin:/\{\{/,relevance:0},l={ +className:"string",contains:[e.BACKSLASH_ESCAPE],variants:[{ +begin:/([uU]|[bB]|[rR]|[bB][rR]|[rR][bB])?'''/,end:/'''/, +contains:[e.BACKSLASH_ESCAPE,r],relevance:10},{ +begin:/([uU]|[bB]|[rR]|[bB][rR]|[rR][bB])?"""/,end:/"""/, +contains:[e.BACKSLASH_ESCAPE,r],relevance:10},{ +begin:/([fF][rR]|[rR][fF]|[fF])'''/,end:/'''/, +contains:[e.BACKSLASH_ESCAPE,r,o,s]},{begin:/([fF][rR]|[rR][fF]|[fF])"""/, +end:/"""/,contains:[e.BACKSLASH_ESCAPE,r,o,s]},{begin:/([uU]|[rR])'/,end:/'/, +relevance:10},{begin:/([uU]|[rR])"/,end:/"/,relevance:10},{ +begin:/([bB]|[bB][rR]|[rR][bB])'/,end:/'/},{begin:/([bB]|[bB][rR]|[rR][bB])"/, +end:/"/},{begin:/([fF][rR]|[rR][fF]|[fF])'/,end:/'/, +contains:[e.BACKSLASH_ESCAPE,o,s]},{begin:/([fF][rR]|[rR][fF]|[fF])"/,end:/"/, +contains:[e.BACKSLASH_ESCAPE,o,s]},e.APOS_STRING_MODE,e.QUOTE_STRING_MODE] +},c="[0-9](_?[0-9])*",d=`(\\b(${c}))?\\.(${c})|\\b(${c})\\.`,g="\\b|"+a.join("|"),u={ +className:"number",relevance:0,variants:[{ +begin:`(\\b(${c})|(${d}))[eE][+-]?(${c})[jJ]?(?=${g})`},{begin:`(${d})[jJ]?`},{ +begin:`\\b([1-9](_?[0-9])*|0+(_?0)*)[lLjJ]?(?=${g})`},{ +begin:`\\b0[bB](_?[01])+[lL]?(?=${g})`},{begin:`\\b0[oO](_?[0-7])+[lL]?(?=${g})` +},{begin:`\\b0[xX](_?[0-9a-fA-F])+[lL]?(?=${g})`},{begin:`\\b(${c})[jJ](?=${g})` +}]},b={className:"comment",begin:n.lookahead(/# type:/),end:/$/,keywords:i, +contains:[{begin:/# type:/},{begin:/#/,end:/\b\B/,endsWithParent:!0}]},m={ +className:"params",variants:[{className:"",begin:/\(\s*\)/,skip:!0},{begin:/\(/, +end:/\)/,excludeBegin:!0,excludeEnd:!0,keywords:i, +contains:["self",r,u,l,e.HASH_COMMENT_MODE]}]};return s.contains=[l,u,r],{ +name:"Python",aliases:["py","gyp","ipython"],unicodeRegex:!0,keywords:i, +illegal:/(<\/|\?)|=>/,contains:[r,u,{begin:/\bself\b/},{beginKeywords:"if", +relevance:0},l,b,e.HASH_COMMENT_MODE,{match:[/\bdef/,/\s+/,t],scope:{ +1:"keyword",3:"title.function"},contains:[m]},{variants:[{ +match:[/\bclass/,/\s+/,t,/\s*/,/\(\s*/,t,/\s*\)/]},{match:[/\bclass/,/\s+/,t]}], +scope:{1:"keyword",3:"title.class",6:"title.class.inherited"}},{ +className:"meta",begin:/^[\t ]*@/,end:/(?=#)|$/,contains:[u,m,l]}]}}, +grmr_python_repl:e=>({aliases:["pycon"],contains:[{className:"meta.prompt", +starts:{end:/ |$/,starts:{end:"$",subLanguage:"python"}},variants:[{ +begin:/^>>>(?=[ ]|$)/},{begin:/^\.\.\.(?=[ ]|$)/}]}]}),grmr_r:e=>{ +const n=e.regex,t=/(?:(?:[a-zA-Z]|\.[._a-zA-Z])[._a-zA-Z0-9]*)|\.(?!\d)/,a=n.either(/0[xX][0-9a-fA-F]+\.[0-9a-fA-F]*[pP][+-]?\d+i?/,/0[xX][0-9a-fA-F]+(?:[pP][+-]?\d+)?[Li]?/,/(?:\d+(?:\.\d*)?|\.\d+)(?:[eE][+-]?\d+)?[Li]?/),i=/[=!<>:]=|\|\||&&|:::?|<-|<<-|->>|->|\|>|[-+*\/?!$&|:<=>@^~]|\*\*/,r=n.either(/[()]/,/[{}]/,/\[\[/,/[[\]]/,/\\/,/,/) +;return{name:"R",keywords:{$pattern:t, +keyword:"function if in break next repeat else for while", +literal:"NULL NA TRUE FALSE Inf NaN NA_integer_|10 NA_real_|10 NA_character_|10 NA_complex_|10", +built_in:"LETTERS letters month.abb month.name pi T F abs acos acosh all any anyNA Arg as.call as.character as.complex as.double as.environment as.integer as.logical as.null.default as.numeric as.raw asin asinh atan atanh attr attributes baseenv browser c call ceiling class Conj cos cosh cospi cummax cummin cumprod cumsum digamma dim dimnames emptyenv exp expression floor forceAndCall gamma gc.time globalenv Im interactive invisible is.array is.atomic is.call is.character is.complex is.double is.environment is.expression is.finite is.function is.infinite is.integer is.language is.list is.logical is.matrix is.na is.name is.nan is.null is.numeric is.object is.pairlist is.raw is.recursive is.single is.symbol lazyLoadDBfetch length lgamma list log max min missing Mod names nargs nzchar oldClass on.exit pos.to.env proc.time prod quote range Re rep retracemem return round seq_along seq_len seq.int sign signif sin sinh sinpi sqrt standardGeneric substitute sum switch tan tanh tanpi tracemem trigamma trunc unclass untracemem UseMethod xtfrm" +},contains:[e.COMMENT(/#'/,/$/,{contains:[{scope:"doctag",match:/@examples/, +starts:{end:n.lookahead(n.either(/\n^#'\s*(?=@[a-zA-Z]+)/,/\n^(?!#')/)), +endsParent:!0}},{scope:"doctag",begin:"@param",end:/$/,contains:[{ +scope:"variable",variants:[{match:t},{match:/`(?:\\.|[^`\\])+`/}],endsParent:!0 +}]},{scope:"doctag",match:/@[a-zA-Z]+/},{scope:"keyword",match:/\\[a-zA-Z]+/}] +}),e.HASH_COMMENT_MODE,{scope:"string",contains:[e.BACKSLASH_ESCAPE], +variants:[e.END_SAME_AS_BEGIN({begin:/[rR]"(-*)\(/,end:/\)(-*)"/ +}),e.END_SAME_AS_BEGIN({begin:/[rR]"(-*)\{/,end:/\}(-*)"/ +}),e.END_SAME_AS_BEGIN({begin:/[rR]"(-*)\[/,end:/\](-*)"/ +}),e.END_SAME_AS_BEGIN({begin:/[rR]'(-*)\(/,end:/\)(-*)'/ +}),e.END_SAME_AS_BEGIN({begin:/[rR]'(-*)\{/,end:/\}(-*)'/ +}),e.END_SAME_AS_BEGIN({begin:/[rR]'(-*)\[/,end:/\](-*)'/}),{begin:'"',end:'"', +relevance:0},{begin:"'",end:"'",relevance:0}]},{relevance:0,variants:[{scope:{ +1:"operator",2:"number"},match:[i,a]},{scope:{1:"operator",2:"number"}, +match:[/%[^%]*%/,a]},{scope:{1:"punctuation",2:"number"},match:[r,a]},{scope:{ +2:"number"},match:[/[^a-zA-Z0-9._]|^/,a]}]},{scope:{3:"operator"}, +match:[t,/\s+/,/<-/,/\s+/]},{scope:"operator",relevance:0,variants:[{match:i},{ +match:/%[^%]*%/}]},{scope:"punctuation",relevance:0,match:r},{begin:"`",end:"`", +contains:[{begin:/\\./}]}]}},grmr_ruby:e=>{ +const n=e.regex,t="([a-zA-Z_]\\w*[!?=]?|[-+~]@|<<|>>|=~|===?|<=>|[<>]=?|\\*\\*|[-/+%^&*~`|]|\\[\\]=?)",a=n.either(/\b([A-Z]+[a-z0-9]+)+/,/\b([A-Z]+[a-z0-9]+)+[A-Z]+/),i=n.concat(a,/(::\w+)*/),r={ +"variable.constant":["__FILE__","__LINE__","__ENCODING__"], +"variable.language":["self","super"], +keyword:["alias","and","begin","BEGIN","break","case","class","defined","do","else","elsif","end","END","ensure","for","if","in","module","next","not","or","redo","require","rescue","retry","return","then","undef","unless","until","when","while","yield","include","extend","prepend","public","private","protected","raise","throw"], +built_in:["proc","lambda","attr_accessor","attr_reader","attr_writer","define_method","private_constant","module_function"], +literal:["true","false","nil"]},s={className:"doctag",begin:"@[A-Za-z]+"},o={ +begin:"#<",end:">"},l=[e.COMMENT("#","$",{contains:[s] +}),e.COMMENT("^=begin","^=end",{contains:[s],relevance:10 +}),e.COMMENT("^__END__",e.MATCH_NOTHING_RE)],c={className:"subst",begin:/#\{/, +end:/\}/,keywords:r},d={className:"string",contains:[e.BACKSLASH_ESCAPE,c], +variants:[{begin:/'/,end:/'/},{begin:/"/,end:/"/},{begin:/`/,end:/`/},{ +begin:/%[qQwWx]?\(/,end:/\)/},{begin:/%[qQwWx]?\[/,end:/\]/},{ +begin:/%[qQwWx]?\{/,end:/\}/},{begin:/%[qQwWx]?/},{begin:/%[qQwWx]?\//, +end:/\//},{begin:/%[qQwWx]?%/,end:/%/},{begin:/%[qQwWx]?-/,end:/-/},{ +begin:/%[qQwWx]?\|/,end:/\|/},{begin:/\B\?(\\\d{1,3})/},{ +begin:/\B\?(\\x[A-Fa-f0-9]{1,2})/},{begin:/\B\?(\\u\{?[A-Fa-f0-9]{1,6}\}?)/},{ +begin:/\B\?(\\M-\\C-|\\M-\\c|\\c\\M-|\\M-|\\C-\\M-)[\x20-\x7e]/},{ +begin:/\B\?\\(c|C-)[\x20-\x7e]/},{begin:/\B\?\\?\S/},{ +begin:n.concat(/<<[-~]?'?/,n.lookahead(/(\w+)(?=\W)[^\n]*\n(?:[^\n]*\n)*?\s*\1\b/)), +contains:[e.END_SAME_AS_BEGIN({begin:/(\w+)/,end:/(\w+)/, +contains:[e.BACKSLASH_ESCAPE,c]})]}]},g="[0-9](_?[0-9])*",u={className:"number", +relevance:0,variants:[{ +begin:`\\b([1-9](_?[0-9])*|0)(\\.(${g}))?([eE][+-]?(${g})|r)?i?\\b`},{ +begin:"\\b0[dD][0-9](_?[0-9])*r?i?\\b"},{begin:"\\b0[bB][0-1](_?[0-1])*r?i?\\b" +},{begin:"\\b0[oO][0-7](_?[0-7])*r?i?\\b"},{ +begin:"\\b0[xX][0-9a-fA-F](_?[0-9a-fA-F])*r?i?\\b"},{ +begin:"\\b0(_?[0-7])+r?i?\\b"}]},b={variants:[{match:/\(\)/},{ +className:"params",begin:/\(/,end:/(?=\))/,excludeBegin:!0,endsParent:!0, +keywords:r}]},m=[d,{variants:[{match:[/class\s+/,i,/\s+<\s+/,i]},{ +match:[/\b(class|module)\s+/,i]}],scope:{2:"title.class", +4:"title.class.inherited"},keywords:r},{match:[/(include|extend)\s+/,i],scope:{ +2:"title.class"},keywords:r},{relevance:0,match:[i,/\.new[. (]/],scope:{ +1:"title.class"}},{relevance:0,match:/\b[A-Z][A-Z_0-9]+\b/, +className:"variable.constant"},{relevance:0,match:a,scope:"title.class"},{ +match:[/def/,/\s+/,t],scope:{1:"keyword",3:"title.function"},contains:[b]},{ +begin:e.IDENT_RE+"::"},{className:"symbol", +begin:e.UNDERSCORE_IDENT_RE+"(!|\\?)?:",relevance:0},{className:"symbol", +begin:":(?!\\s)",contains:[d,{begin:t}],relevance:0},u,{className:"variable", +begin:"(\\$\\W)|((\\$|@@?)(\\w+))(?=[^@$?])(?![A-Za-z])(?![@$?'])"},{ +className:"params",begin:/\|/,end:/\|/,excludeBegin:!0,excludeEnd:!0, +relevance:0,keywords:r},{begin:"("+e.RE_STARTERS_RE+"|unless)\\s*", +keywords:"unless",contains:[{className:"regexp",contains:[e.BACKSLASH_ESCAPE,c], +illegal:/\n/,variants:[{begin:"/",end:"/[a-z]*"},{begin:/%r\{/,end:/\}[a-z]*/},{ +begin:"%r\\(",end:"\\)[a-z]*"},{begin:"%r!",end:"![a-z]*"},{begin:"%r\\[", +end:"\\][a-z]*"}]}].concat(o,l),relevance:0}].concat(o,l) +;c.contains=m,b.contains=m;const p=[{begin:/^\s*=>/,starts:{end:"$",contains:m} +},{className:"meta.prompt", +begin:"^([>?]>|[\\w#]+\\(\\w+\\):\\d+:\\d+[>*]|(\\w+-)?\\d+\\.\\d+\\.\\d+(p\\d+)?[^\\d][^>]+>)(?=[ ])", +starts:{end:"$",keywords:r,contains:m}}];return l.unshift(o),{name:"Ruby", +aliases:["rb","gemspec","podspec","thor","irb"],keywords:r,illegal:/\/\*/, +contains:[e.SHEBANG({binary:"ruby"})].concat(p).concat(l).concat(m)}}, +grmr_rust:e=>{const n=e.regex,t={className:"title.function.invoke",relevance:0, +begin:n.concat(/\b/,/(?!let|for|while|if|else|match\b)/,e.IDENT_RE,n.lookahead(/\s*\(/)) +},a="([ui](8|16|32|64|128|size)|f(32|64))?",i=["drop ","Copy","Send","Sized","Sync","Drop","Fn","FnMut","FnOnce","ToOwned","Clone","Debug","PartialEq","PartialOrd","Eq","Ord","AsRef","AsMut","Into","From","Default","Iterator","Extend","IntoIterator","DoubleEndedIterator","ExactSizeIterator","SliceConcatExt","ToString","assert!","assert_eq!","bitflags!","bytes!","cfg!","col!","concat!","concat_idents!","debug_assert!","debug_assert_eq!","env!","eprintln!","panic!","file!","format!","format_args!","include_bytes!","include_str!","line!","local_data_key!","module_path!","option_env!","print!","println!","select!","stringify!","try!","unimplemented!","unreachable!","vec!","write!","writeln!","macro_rules!","assert_ne!","debug_assert_ne!"],r=["i8","i16","i32","i64","i128","isize","u8","u16","u32","u64","u128","usize","f32","f64","str","char","bool","Box","Option","Result","String","Vec"] +;return{name:"Rust",aliases:["rs"],keywords:{$pattern:e.IDENT_RE+"!?",type:r, +keyword:["abstract","as","async","await","become","box","break","const","continue","crate","do","dyn","else","enum","extern","false","final","fn","for","if","impl","in","let","loop","macro","match","mod","move","mut","override","priv","pub","ref","return","self","Self","static","struct","super","trait","true","try","type","typeof","unsafe","unsized","use","virtual","where","while","yield"], +literal:["true","false","Some","None","Ok","Err"],built_in:i},illegal:""},t]}}, +grmr_scss:e=>{const n=ie(e),t=le,a=oe,i="@[a-z-]+",r={className:"variable", +begin:"(\\$[a-zA-Z-][a-zA-Z0-9_-]*)\\b",relevance:0};return{name:"SCSS", +case_insensitive:!0,illegal:"[=/|']", +contains:[e.C_LINE_COMMENT_MODE,e.C_BLOCK_COMMENT_MODE,n.CSS_NUMBER_MODE,{ +className:"selector-id",begin:"#[A-Za-z0-9_-]+",relevance:0},{ +className:"selector-class",begin:"\\.[A-Za-z0-9_-]+",relevance:0 +},n.ATTRIBUTE_SELECTOR_MODE,{className:"selector-tag", +begin:"\\b("+re.join("|")+")\\b",relevance:0},{className:"selector-pseudo", +begin:":("+a.join("|")+")"},{className:"selector-pseudo", +begin:":(:)?("+t.join("|")+")"},r,{begin:/\(/,end:/\)/, +contains:[n.CSS_NUMBER_MODE]},n.CSS_VARIABLE,{className:"attribute", +begin:"\\b("+ce.join("|")+")\\b"},{ +begin:"\\b(whitespace|wait|w-resize|visible|vertical-text|vertical-ideographic|uppercase|upper-roman|upper-alpha|underline|transparent|top|thin|thick|text|text-top|text-bottom|tb-rl|table-header-group|table-footer-group|sw-resize|super|strict|static|square|solid|small-caps|separate|se-resize|scroll|s-resize|rtl|row-resize|ridge|right|repeat|repeat-y|repeat-x|relative|progress|pointer|overline|outside|outset|oblique|nowrap|not-allowed|normal|none|nw-resize|no-repeat|no-drop|newspaper|ne-resize|n-resize|move|middle|medium|ltr|lr-tb|lowercase|lower-roman|lower-alpha|loose|list-item|line|line-through|line-edge|lighter|left|keep-all|justify|italic|inter-word|inter-ideograph|inside|inset|inline|inline-block|inherit|inactive|ideograph-space|ideograph-parenthesis|ideograph-numeric|ideograph-alpha|horizontal|hidden|help|hand|groove|fixed|ellipsis|e-resize|double|dotted|distribute|distribute-space|distribute-letter|distribute-all-lines|disc|disabled|default|decimal|dashed|crosshair|collapse|col-resize|circle|char|center|capitalize|break-word|break-all|bottom|both|bolder|bold|block|bidi-override|below|baseline|auto|always|all-scroll|absolute|table|table-cell)\\b" +},{begin:/:/,end:/[;}{]/,relevance:0, +contains:[n.BLOCK_COMMENT,r,n.HEXCOLOR,n.CSS_NUMBER_MODE,e.QUOTE_STRING_MODE,e.APOS_STRING_MODE,n.IMPORTANT,n.FUNCTION_DISPATCH] +},{begin:"@(page|font-face)",keywords:{$pattern:i,keyword:"@page @font-face"}},{ +begin:"@",end:"[{;]",returnBegin:!0,keywords:{$pattern:/[a-z-]+/, +keyword:"and or not only",attribute:se.join(" ")},contains:[{begin:i, +className:"keyword"},{begin:/[a-z-]+(?=:)/,className:"attribute" +},r,e.QUOTE_STRING_MODE,e.APOS_STRING_MODE,n.HEXCOLOR,n.CSS_NUMBER_MODE] +},n.FUNCTION_DISPATCH]}},grmr_shell:e=>({name:"Shell Session", +aliases:["console","shellsession"],contains:[{className:"meta.prompt", +begin:/^\s{0,3}[/~\w\d[\]()@-]*[>%$#][ ]?/,starts:{end:/[^\\](?=\s*$)/, +subLanguage:"bash"}}]}),grmr_sql:e=>{ +const n=e.regex,t=e.COMMENT("--","$"),a=["true","false","unknown"],i=["bigint","binary","blob","boolean","char","character","clob","date","dec","decfloat","decimal","float","int","integer","interval","nchar","nclob","national","numeric","real","row","smallint","time","timestamp","varchar","varying","varbinary"],r=["abs","acos","array_agg","asin","atan","avg","cast","ceil","ceiling","coalesce","corr","cos","cosh","count","covar_pop","covar_samp","cume_dist","dense_rank","deref","element","exp","extract","first_value","floor","json_array","json_arrayagg","json_exists","json_object","json_objectagg","json_query","json_table","json_table_primitive","json_value","lag","last_value","lead","listagg","ln","log","log10","lower","max","min","mod","nth_value","ntile","nullif","percent_rank","percentile_cont","percentile_disc","position","position_regex","power","rank","regr_avgx","regr_avgy","regr_count","regr_intercept","regr_r2","regr_slope","regr_sxx","regr_sxy","regr_syy","row_number","sin","sinh","sqrt","stddev_pop","stddev_samp","substring","substring_regex","sum","tan","tanh","translate","translate_regex","treat","trim","trim_array","unnest","upper","value_of","var_pop","var_samp","width_bucket"],s=["create table","insert into","primary key","foreign key","not null","alter table","add constraint","grouping sets","on overflow","character set","respect nulls","ignore nulls","nulls first","nulls last","depth first","breadth first"],o=r,l=["abs","acos","all","allocate","alter","and","any","are","array","array_agg","array_max_cardinality","as","asensitive","asin","asymmetric","at","atan","atomic","authorization","avg","begin","begin_frame","begin_partition","between","bigint","binary","blob","boolean","both","by","call","called","cardinality","cascaded","case","cast","ceil","ceiling","char","char_length","character","character_length","check","classifier","clob","close","coalesce","collate","collect","column","commit","condition","connect","constraint","contains","convert","copy","corr","corresponding","cos","cosh","count","covar_pop","covar_samp","create","cross","cube","cume_dist","current","current_catalog","current_date","current_default_transform_group","current_path","current_role","current_row","current_schema","current_time","current_timestamp","current_path","current_role","current_transform_group_for_type","current_user","cursor","cycle","date","day","deallocate","dec","decimal","decfloat","declare","default","define","delete","dense_rank","deref","describe","deterministic","disconnect","distinct","double","drop","dynamic","each","element","else","empty","end","end_frame","end_partition","end-exec","equals","escape","every","except","exec","execute","exists","exp","external","extract","false","fetch","filter","first_value","float","floor","for","foreign","frame_row","free","from","full","function","fusion","get","global","grant","group","grouping","groups","having","hold","hour","identity","in","indicator","initial","inner","inout","insensitive","insert","int","integer","intersect","intersection","interval","into","is","join","json_array","json_arrayagg","json_exists","json_object","json_objectagg","json_query","json_table","json_table_primitive","json_value","lag","language","large","last_value","lateral","lead","leading","left","like","like_regex","listagg","ln","local","localtime","localtimestamp","log","log10","lower","match","match_number","match_recognize","matches","max","member","merge","method","min","minute","mod","modifies","module","month","multiset","national","natural","nchar","nclob","new","no","none","normalize","not","nth_value","ntile","null","nullif","numeric","octet_length","occurrences_regex","of","offset","old","omit","on","one","only","open","or","order","out","outer","over","overlaps","overlay","parameter","partition","pattern","per","percent","percent_rank","percentile_cont","percentile_disc","period","portion","position","position_regex","power","precedes","precision","prepare","primary","procedure","ptf","range","rank","reads","real","recursive","ref","references","referencing","regr_avgx","regr_avgy","regr_count","regr_intercept","regr_r2","regr_slope","regr_sxx","regr_sxy","regr_syy","release","result","return","returns","revoke","right","rollback","rollup","row","row_number","rows","running","savepoint","scope","scroll","search","second","seek","select","sensitive","session_user","set","show","similar","sin","sinh","skip","smallint","some","specific","specifictype","sql","sqlexception","sqlstate","sqlwarning","sqrt","start","static","stddev_pop","stddev_samp","submultiset","subset","substring","substring_regex","succeeds","sum","symmetric","system","system_time","system_user","table","tablesample","tan","tanh","then","time","timestamp","timezone_hour","timezone_minute","to","trailing","translate","translate_regex","translation","treat","trigger","trim","trim_array","true","truncate","uescape","union","unique","unknown","unnest","update","upper","user","using","value","values","value_of","var_pop","var_samp","varbinary","varchar","varying","versioning","when","whenever","where","width_bucket","window","with","within","without","year","add","asc","collation","desc","final","first","last","view"].filter((e=>!r.includes(e))),c={ +begin:n.concat(/\b/,n.either(...o),/\s*\(/),relevance:0,keywords:{built_in:o}} +;return{name:"SQL",case_insensitive:!0,illegal:/[{}]|<\//,keywords:{ +$pattern:/\b[\w\.]+/,keyword:((e,{exceptions:n,when:t}={})=>{const a=t +;return n=n||[],e.map((e=>e.match(/\|\d+$/)||n.includes(e)?e:a(e)?e+"|0":e)) +})(l,{when:e=>e.length<3}),literal:a,type:i, +built_in:["current_catalog","current_date","current_default_transform_group","current_path","current_role","current_schema","current_transform_group_for_type","current_user","session_user","system_time","system_user","current_time","localtime","current_timestamp","localtimestamp"] +},contains:[{begin:n.either(...s),relevance:0,keywords:{$pattern:/[\w\.]+/, +keyword:l.concat(s),literal:a,type:i}},{className:"type", +begin:n.either("double precision","large object","with timezone","without timezone") +},c,{className:"variable",begin:/@[a-z0-9][a-z0-9_]*/},{className:"string", +variants:[{begin:/'/,end:/'/,contains:[{begin:/''/}]}]},{begin:/"/,end:/"/, +contains:[{begin:/""/}]},e.C_NUMBER_MODE,e.C_BLOCK_COMMENT_MODE,t,{ +className:"operator",begin:/[-+*/=%^~]|&&?|\|\|?|!=?|<(?:=>?|<|>)?|>[>=]?/, +relevance:0}]}},grmr_swift:e=>{const n={match:/\s+/,relevance:0 +},t=e.COMMENT("/\\*","\\*/",{contains:["self"]}),a=[e.C_LINE_COMMENT_MODE,t],i={ +match:[/\./,m(...xe,...Me)],className:{2:"keyword"}},r={match:b(/\./,m(...Ae)), +relevance:0},s=Ae.filter((e=>"string"==typeof e)).concat(["_|0"]),o={variants:[{ +className:"keyword", +match:m(...Ae.filter((e=>"string"!=typeof e)).concat(Se).map(ke),...Me)}]},l={ +$pattern:m(/\b\w+/,/#\w+/),keyword:s.concat(Re),literal:Ce},c=[i,r,o],g=[{ +match:b(/\./,m(...De)),relevance:0},{className:"built_in", +match:b(/\b/,m(...De),/(?=\()/)}],u={match:/->/,relevance:0},p=[u,{ +className:"operator",relevance:0,variants:[{match:Be},{match:`\\.(\\.|${Le})+`}] +}],_="([0-9]_*)+",h="([0-9a-fA-F]_*)+",f={className:"number",relevance:0, +variants:[{match:`\\b(${_})(\\.(${_}))?([eE][+-]?(${_}))?\\b`},{ +match:`\\b0x(${h})(\\.(${h}))?([pP][+-]?(${_}))?\\b`},{match:/\b0o([0-7]_*)+\b/ +},{match:/\b0b([01]_*)+\b/}]},E=(e="")=>({className:"subst",variants:[{ +match:b(/\\/,e,/[0\\tnr"']/)},{match:b(/\\/,e,/u\{[0-9a-fA-F]{1,8}\}/)}] +}),y=(e="")=>({className:"subst",match:b(/\\/,e,/[\t ]*(?:[\r\n]|\r\n)/) +}),N=(e="")=>({className:"subst",label:"interpol",begin:b(/\\/,e,/\(/),end:/\)/ +}),w=(e="")=>({begin:b(e,/"""/),end:b(/"""/,e),contains:[E(e),y(e),N(e)] +}),v=(e="")=>({begin:b(e,/"/),end:b(/"/,e),contains:[E(e),N(e)]}),O={ +className:"string", +variants:[w(),w("#"),w("##"),w("###"),v(),v("#"),v("##"),v("###")] +},k=[e.BACKSLASH_ESCAPE,{begin:/\[/,end:/\]/,relevance:0, +contains:[e.BACKSLASH_ESCAPE]}],x={begin:/\/[^\s](?=[^/\n]*\/)/,end:/\//, +contains:k},M=e=>{const n=b(e,/\//),t=b(/\//,e);return{begin:n,end:t, +contains:[...k,{scope:"comment",begin:`#(?!.*${t})`,end:/$/}]}},S={ +scope:"regexp",variants:[M("###"),M("##"),M("#"),x]},A={match:b(/`/,Fe,/`/) +},C=[A,{className:"variable",match:/\$\d+/},{className:"variable", +match:`\\$${ze}+`}],T=[{match:/(@|#(un)?)available/,scope:"keyword",starts:{ +contains:[{begin:/\(/,end:/\)/,keywords:Pe,contains:[...p,f,O]}]}},{ +scope:"keyword",match:b(/@/,m(...je))},{scope:"meta",match:b(/@/,Fe)}],R={ +match:d(/\b[A-Z]/),relevance:0,contains:[{className:"type", +match:b(/(AV|CA|CF|CG|CI|CL|CM|CN|CT|MK|MP|MTK|MTL|NS|SCN|SK|UI|WK|XC)/,ze,"+") +},{className:"type",match:Ue,relevance:0},{match:/[?!]+/,relevance:0},{ +match:/\.\.\./,relevance:0},{match:b(/\s+&\s+/,d(Ue)),relevance:0}]},D={ +begin://,keywords:l,contains:[...a,...c,...T,u,R]};R.contains.push(D) +;const I={begin:/\(/,end:/\)/,relevance:0,keywords:l,contains:["self",{ +match:b(Fe,/\s*:/),keywords:"_|0",relevance:0 +},...a,S,...c,...g,...p,f,O,...C,...T,R]},L={begin://, +keywords:"repeat each",contains:[...a,R]},B={begin:/\(/,end:/\)/,keywords:l, +contains:[{begin:m(d(b(Fe,/\s*:/)),d(b(Fe,/\s+/,Fe,/\s*:/))),end:/:/, +relevance:0,contains:[{className:"keyword",match:/\b_\b/},{className:"params", +match:Fe}]},...a,...c,...p,f,O,...T,R,I],endsParent:!0,illegal:/["']/},$={ +match:[/(func|macro)/,/\s+/,m(A.match,Fe,Be)],className:{1:"keyword", +3:"title.function"},contains:[L,B,n],illegal:[/\[/,/%/]},z={ +match:[/\b(?:subscript|init[?!]?)/,/\s*(?=[<(])/],className:{1:"keyword"}, +contains:[L,B,n],illegal:/\[|%/},F={match:[/operator/,/\s+/,Be],className:{ +1:"keyword",3:"title"}},U={begin:[/precedencegroup/,/\s+/,Ue],className:{ +1:"keyword",3:"title"},contains:[R],keywords:[...Te,...Ce],end:/}/} +;for(const e of O.variants){const n=e.contains.find((e=>"interpol"===e.label)) +;n.keywords=l;const t=[...c,...g,...p,f,O,...C];n.contains=[...t,{begin:/\(/, +end:/\)/,contains:["self",...t]}]}return{name:"Swift",keywords:l, +contains:[...a,$,z,{beginKeywords:"struct protocol class extension enum actor", +end:"\\{",excludeEnd:!0,keywords:l,contains:[e.inherit(e.TITLE_MODE,{ +className:"title.class",begin:/[A-Za-z$_][\u00C0-\u02B80-9A-Za-z$_]*/}),...c] +},F,U,{beginKeywords:"import",end:/$/,contains:[...a],relevance:0 +},S,...c,...g,...p,f,O,...C,...T,R,I]}},grmr_typescript:e=>{ +const n=Oe(e),t=_e,a=["any","void","number","boolean","string","object","never","symbol","bigint","unknown"],i={ +beginKeywords:"namespace",end:/\{/,excludeEnd:!0, +contains:[n.exports.CLASS_REFERENCE]},r={beginKeywords:"interface",end:/\{/, +excludeEnd:!0,keywords:{keyword:"interface extends",built_in:a}, +contains:[n.exports.CLASS_REFERENCE]},s={$pattern:_e, +keyword:he.concat(["type","namespace","interface","public","private","protected","implements","declare","abstract","readonly","enum","override"]), +literal:fe,built_in:ve.concat(a),"variable.language":we},o={className:"meta", +begin:"@"+t},l=(e,n,t)=>{const a=e.contains.findIndex((e=>e.label===n)) +;if(-1===a)throw Error("can not find mode to replace");e.contains.splice(a,1,t)} +;return Object.assign(n.keywords,s), +n.exports.PARAMS_CONTAINS.push(o),n.contains=n.contains.concat([o,i,r]), +l(n,"shebang",e.SHEBANG()),l(n,"use_strict",{className:"meta",relevance:10, +begin:/^\s*['"]use strict['"]/ +}),n.contains.find((e=>"func.def"===e.label)).relevance=0,Object.assign(n,{ +name:"TypeScript",aliases:["ts","tsx","mts","cts"]}),n},grmr_vbnet:e=>{ +const n=e.regex,t=/\d{1,2}\/\d{1,2}\/\d{4}/,a=/\d{4}-\d{1,2}-\d{1,2}/,i=/(\d|1[012])(:\d+){0,2} *(AM|PM)/,r=/\d{1,2}(:\d{1,2}){1,2}/,s={ +className:"literal",variants:[{begin:n.concat(/# */,n.either(a,t),/ *#/)},{ +begin:n.concat(/# */,r,/ *#/)},{begin:n.concat(/# */,i,/ *#/)},{ +begin:n.concat(/# */,n.either(a,t),/ +/,n.either(i,r),/ *#/)}] +},o=e.COMMENT(/'''/,/$/,{contains:[{className:"doctag",begin:/<\/?/,end:/>/}] +}),l=e.COMMENT(null,/$/,{variants:[{begin:/'/},{begin:/([\t ]|^)REM(?=\s)/}]}) +;return{name:"Visual Basic .NET",aliases:["vb"],case_insensitive:!0, +classNameAliases:{label:"symbol"},keywords:{ +keyword:"addhandler alias aggregate ansi as async assembly auto binary by byref byval call case catch class compare const continue custom declare default delegate dim distinct do each equals else elseif end enum erase error event exit explicit finally for friend from function get global goto group handles if implements imports in inherits interface into iterator join key let lib loop me mid module mustinherit mustoverride mybase myclass namespace narrowing new next notinheritable notoverridable of off on operator option optional order overloads overridable overrides paramarray partial preserve private property protected public raiseevent readonly redim removehandler resume return select set shadows shared skip static step stop structure strict sub synclock take text then throw to try unicode until using when where while widening with withevents writeonly yield", +built_in:"addressof and andalso await directcast gettype getxmlnamespace is isfalse isnot istrue like mod nameof new not or orelse trycast typeof xor cbool cbyte cchar cdate cdbl cdec cint clng cobj csbyte cshort csng cstr cuint culng cushort", +type:"boolean byte char date decimal double integer long object sbyte short single string uinteger ulong ushort", +literal:"true false nothing"}, +illegal:"//|\\{|\\}|endif|gosub|variant|wend|^\\$ ",contains:[{ +className:"string",begin:/"(""|[^/n])"C\b/},{className:"string",begin:/"/, +end:/"/,illegal:/\n/,contains:[{begin:/""/}]},s,{className:"number",relevance:0, +variants:[{begin:/\b\d[\d_]*((\.[\d_]+(E[+-]?[\d_]+)?)|(E[+-]?[\d_]+))[RFD@!#]?/ +},{begin:/\b\d[\d_]*((U?[SIL])|[%&])?/},{begin:/&H[\dA-F_]+((U?[SIL])|[%&])?/},{ +begin:/&O[0-7_]+((U?[SIL])|[%&])?/},{begin:/&B[01_]+((U?[SIL])|[%&])?/}]},{ +className:"label",begin:/^\w+:/},o,l,{className:"meta", +begin:/[\t ]*#(const|disable|else|elseif|enable|end|externalsource|if|region)\b/, +end:/$/,keywords:{ +keyword:"const disable else elseif enable end externalsource if region then"}, +contains:[l]}]}},grmr_wasm:e=>{e.regex;const n=e.COMMENT(/\(;/,/;\)/) +;return n.contains.push("self"),{name:"WebAssembly",keywords:{$pattern:/[\w.]+/, +keyword:["anyfunc","block","br","br_if","br_table","call","call_indirect","data","drop","elem","else","end","export","func","global.get","global.set","local.get","local.set","local.tee","get_global","get_local","global","if","import","local","loop","memory","memory.grow","memory.size","module","mut","nop","offset","param","result","return","select","set_global","set_local","start","table","tee_local","then","type","unreachable"] +},contains:[e.COMMENT(/;;/,/$/),n,{match:[/(?:offset|align)/,/\s*/,/=/], +className:{1:"keyword",3:"operator"}},{className:"variable",begin:/\$[\w_]+/},{ +match:/(\((?!;)|\))+/,className:"punctuation",relevance:0},{ +begin:[/(?:func|call|call_indirect)/,/\s+/,/\$[^\s)]+/],className:{1:"keyword", +3:"title.function"}},e.QUOTE_STRING_MODE,{match:/(i32|i64|f32|f64)(?!\.)/, +className:"type"},{className:"keyword", +match:/\b(f32|f64|i32|i64)(?:\.(?:abs|add|and|ceil|clz|const|convert_[su]\/i(?:32|64)|copysign|ctz|demote\/f64|div(?:_[su])?|eqz?|extend_[su]\/i32|floor|ge(?:_[su])?|gt(?:_[su])?|le(?:_[su])?|load(?:(?:8|16|32)_[su])?|lt(?:_[su])?|max|min|mul|nearest|neg?|or|popcnt|promote\/f32|reinterpret\/[fi](?:32|64)|rem_[su]|rot[lr]|shl|shr_[su]|store(?:8|16|32)?|sqrt|sub|trunc(?:_[su]\/f(?:32|64))?|wrap\/i64|xor))\b/ +},{className:"number",relevance:0, +match:/[+-]?\b(?:\d(?:_?\d)*(?:\.\d(?:_?\d)*)?(?:[eE][+-]?\d(?:_?\d)*)?|0x[\da-fA-F](?:_?[\da-fA-F])*(?:\.[\da-fA-F](?:_?[\da-fA-D])*)?(?:[pP][+-]?\d(?:_?\d)*)?)\b|\binf\b|\bnan(?::0x[\da-fA-F](?:_?[\da-fA-D])*)?\b/ +}]}},grmr_xml:e=>{ +const n=e.regex,t=n.concat(/[\p{L}_]/u,n.optional(/[\p{L}0-9_.-]*:/u),/[\p{L}0-9_.-]*/u),a={ +className:"symbol",begin:/&[a-z]+;|&#[0-9]+;|&#x[a-f0-9]+;/},i={begin:/\s/, +contains:[{className:"keyword",begin:/#?[a-z_][a-z1-9_-]+/,illegal:/\n/}] +},r=e.inherit(i,{begin:/\(/,end:/\)/}),s=e.inherit(e.APOS_STRING_MODE,{ +className:"string"}),o=e.inherit(e.QUOTE_STRING_MODE,{className:"string"}),l={ +endsWithParent:!0,illegal:/`]+/}]}]}]};return{ +name:"HTML, XML", +aliases:["html","xhtml","rss","atom","xjb","xsd","xsl","plist","wsf","svg"], +case_insensitive:!0,unicodeRegex:!0,contains:[{className:"meta",begin://,relevance:10,contains:[i,o,s,r,{begin:/\[/,end:/\]/,contains:[{ +className:"meta",begin://,contains:[i,r,o,s]}]}] +},e.COMMENT(//,{relevance:10}),{begin://, +relevance:10},a,{className:"meta",end:/\?>/,variants:[{begin:/<\?xml/, +relevance:10,contains:[o]},{begin:/<\?[a-z][a-z0-9]+/}]},{className:"tag", +begin:/)/,end:/>/,keywords:{name:"style"},contains:[l],starts:{ +end:/<\/style>/,returnEnd:!0,subLanguage:["css","xml"]}},{className:"tag", +begin:/)/,end:/>/,keywords:{name:"script"},contains:[l],starts:{ +end:/<\/script>/,returnEnd:!0,subLanguage:["javascript","handlebars","xml"]}},{ +className:"tag",begin:/<>|<\/>/},{className:"tag", +begin:n.concat(//,/>/,/\s/)))), +end:/\/?>/,contains:[{className:"name",begin:t,relevance:0,starts:l}]},{ +className:"tag",begin:n.concat(/<\//,n.lookahead(n.concat(t,/>/))),contains:[{ +className:"name",begin:t,relevance:0},{begin:/>/,relevance:0,endsParent:!0}]}]} +},grmr_yaml:e=>{ +const n="true false yes no null",t="[\\w#;/?:@&=+$,.~*'()[\\]]+",a={ +className:"string",relevance:0,variants:[{begin:/'/,end:/'/},{begin:/"/,end:/"/ +},{begin:/\S+/}],contains:[e.BACKSLASH_ESCAPE,{className:"template-variable", +variants:[{begin:/\{\{/,end:/\}\}/},{begin:/%\{/,end:/\}/}]}]},i=e.inherit(a,{ +variants:[{begin:/'/,end:/'/},{begin:/"/,end:/"/},{begin:/[^\s,{}[\]]+/}]}),r={ +end:",",endsWithParent:!0,excludeEnd:!0,keywords:n,relevance:0},s={begin:/\{/, +end:/\}/,contains:[r],illegal:"\\n",relevance:0},o={begin:"\\[",end:"\\]", +contains:[r],illegal:"\\n",relevance:0},l=[{className:"attr",variants:[{ +begin:"\\w[\\w :\\/.-]*:(?=[ \t]|$)"},{begin:'"\\w[\\w :\\/.-]*":(?=[ \t]|$)'},{ +begin:"'\\w[\\w :\\/.-]*':(?=[ \t]|$)"}]},{className:"meta",begin:"^---\\s*$", +relevance:10},{className:"string", +begin:"[\\|>]([1-9]?[+-])?[ ]*\\n( +)[^ ][^\\n]*\\n(\\2[^\\n]+\\n?)*"},{ +begin:"<%[%=-]?",end:"[%-]?%>",subLanguage:"ruby",excludeBegin:!0,excludeEnd:!0, +relevance:0},{className:"type",begin:"!\\w+!"+t},{className:"type", +begin:"!<"+t+">"},{className:"type",begin:"!"+t},{className:"type",begin:"!!"+t +},{className:"meta",begin:"&"+e.UNDERSCORE_IDENT_RE+"$"},{className:"meta", +begin:"\\*"+e.UNDERSCORE_IDENT_RE+"$"},{className:"bullet",begin:"-(?=[ ]|$)", +relevance:0},e.HASH_COMMENT_MODE,{beginKeywords:n,keywords:{literal:n}},{ +className:"number", +begin:"\\b[0-9]{4}(-[0-9][0-9]){0,2}([Tt \\t][0-9][0-9]?(:[0-9][0-9]){2})?(\\.[0-9]*)?([ \\t])*(Z|[-+][0-9][0-9]?(:[0-9][0-9])?)?\\b" +},{className:"number",begin:e.C_NUMBER_RE+"\\b",relevance:0},s,o,a],c=[...l] +;return c.pop(),c.push(i),r.contains=c,{name:"YAML",case_insensitive:!0, +aliases:["yml"],contains:l}}});const He=ae;for(const e of Object.keys(Ke)){ +const n=e.replace("grmr_","").replace("_","-");He.registerLanguage(n,Ke[e])} +return He}() +;"object"==typeof exports&&"undefined"!=typeof module&&(module.exports=hljs); \ No newline at end of file diff --git a/icons/icon128.png b/icons/icon128.png new file mode 100644 index 0000000000000000000000000000000000000000..9790306e2f12c1157454eb1e2ee087cab716bbe0 GIT binary patch literal 1353 zcmYLJeN>WH9KIkbNDi7JC8waWJkBL^nUoaZ(zH#@?1NJ~#Im!>rNh3c6<=v3YJPOq znWsoy9aCEqvQ?y0%&%EVW4V>7h@xgWC3!RDJxH)i?QDPC_de&nAHVzD=Xq{!a#EbH zPoNI~z;{D@ObYB3?uPe*-&W49P5?Z~8)Blj9P738kEQsB`=2q_eat7vhO|-UQuzKx zn^dTNLW%8U&l}U98VkxkeEe zYgi(24ST9r#WywR4X>u1L{S-g>bgs>suIu48O>!CHGVhps5O1YX}siZGTKJVSVx16 zb;FM=xhwa6n~^{=+t05HmYqOVE9#y$t?r4QGk*;dZk_rwa1DZ3MI)6Mlq((Tyk{7- zXDT)Gnts45nKb#4Rc-vUP4!F_T{;x{n_&~a+coG4c1YwT+ffr9wiY@~~Kx1p$9|^c5qBJc3aS(3Z!&s?5S0S_++FXe0ZXTZ~LsiMg{_7Y|kT;2ANWEE?|1SrcwkDn(eHptXBsU0Hp=q4U1LbAYHIuIg;)D)0paE>)4>kN`oAsi2M@ z;b@Nr$Hr%Nd%Qe2Chh@b$XKb0{Hfu|{u5*33n1A_t!f0x;)(cX?M5^e%WFA`<@Ks5 z3wRx>$UGA>(3wU9y*t;YvBeK@3I(idtlqu~6kB%GVtyPf!h*i5p=rNv(>Go*ld!;e zMX*evzO%pG7#irz@&y@kSvnBBdJ;+lSZB+dZW(Dh(<_9wa_O))hEb~Dv_~8Ek&FX^ zps2TYZx5PqpWoL72_Xp!-w5h!Z8>hHoZ@QupSm#7}W6yX0|^^4uw?#j|UDFER1(LW&QML?PqN++7~U z$4wmh%ynpq3VHfT^f{6ZDr9Nd=yPbl6jcAY${I3A*J3JW3Q8-@o10mf3dhN-%LX^M zkg)M@$9Hs5GoKbr;_hv{ZqDf47wCF+nd#1p-vBgxquYSprzatlqH!4z=Zc`=cRCo* z@bC}}9mlaG65=)%%negUhW|VRzedY|K z9{1yTlGCVr(|dFpPdrx`4%HZ{%6AQboB3!S@)qViz&W9RVez;zmn;m?HNqD{z5)xw zsHGy5;aT`|k<7VRw%GOe)NU$L` LDMtO-S6Tl6p^%8A literal 0 HcmV?d00001 diff --git a/icons/icon16.png b/icons/icon16.png new file mode 100644 index 0000000000000000000000000000000000000000..7abca9a4a95c87d28bf261ed90975d895669d9c5 GIT binary patch literal 261 zcmeAS@N?(olHy`uVBq!ia0vp^0wB!61|;P_|4#%`XFOdTLn>}9y!+lw=u8Dkw3kO-{>Is@#A+Y8+|^UdCDYgQ#xNk*k)ycqu8^Le(vqZ zgFbMpYpb8vk_(krxV!DLMP%f&5SuAXcV|9L{d&e|`6I_(?h^57IhMVuTJNsETd?MK zn0{b;b*bKu=fSBD*NMz{;CGmL=7ViOM%d-w|0b&aV15`nw|{n>y%o@344$rjF6*2U FngG?6bEyCT literal 0 HcmV?d00001 diff --git a/icons/icon48.png b/icons/icon48.png new file mode 100644 index 0000000000000000000000000000000000000000..c4601fd41b31816e9e347ca50043f277a70b6136 GIT binary patch literal 547 zcmV+;0^I$HP)p0+Y$&Bb*?aH3;lHv+yAfrFIK zKCqW6OLhS->bxrPR+8UI`)sHG32AltEd^`&%}iNrWWEGzd1Splf}{MJ4&W+}tY#sC zv-}DMaF&+=oaL9ZpMC&W`6UeCEH48%%P;n*0Iu>27|;lLn4chA%gX?l@?qj8uH@%+ zfUSH~aT8nl5(YF&USlI)>irO2BOaTuKWxx%ZRF=uS)YA zRbB=pEw7(>^E^d<#t)!fQb7T6-t*>pihL0Rs5h+bC1swc$QN@EKwS?^fw(tpP(WPy zX&FG=ZGr;g+_2_(ihKbBsEe8HdSITX$QN?&!n4^bO^KxCr*r^yB@SB+;>ha&ihLde zZ1?#%>w#;hZevH^Ib2jpmGQh!?RI}tKFu>(!U{rD2$41M`fJS`7{w3~Geq0A6 z^#Sf$eysmZc%0>B0B8A89l%w71OqtB%K*;u!#aSg{LladaF&+=IIQJyILZ&!2jH-l l*TY&~4{Q1QY1u!lF~4>rPLjLrM+g7_002ovPDHLkV1gzQ`4a#D literal 0 HcmV?d00001 diff --git a/manifest.json b/manifest.json new file mode 100644 index 0000000..1c53ed0 --- /dev/null +++ b/manifest.json @@ -0,0 +1,45 @@ +{ + "manifest_version": 3, + "name": "Ollama Sidebar", + "version": "1.0.0", + "description": "Chat with local Ollama models from a browser sidebar", + "permissions": ["storage", "sidePanel", "tabs", "activeTab", "scripting"], + "host_permissions": [""], + "background": { + "service_worker": "background.js" + }, + "action": { + "default_title": "Ollama Sidebar", + "default_icon": { + "16": "icons/icon16.png", + "48": "icons/icon48.png", + "128": "icons/icon128.png" + } + }, + "side_panel": { + "default_path": "sidepanel.html" + }, + "sidebar_action": { + "default_panel": "sidepanel.html", + "default_title": "Ollama Sidebar", + "default_icon": "icons/icon48.png" + }, + "content_scripts": [ + { + "matches": [""], + "js": ["content.js"], + "run_at": "document_idle" + } + ], + "web_accessible_resources": [ + { + "resources": ["marked.min.js", "highlight.min.js", "marked.min.css"], + "matches": [""] + } + ], + "icons": { + "16": "icons/icon16.png", + "48": "icons/icon48.png", + "128": "icons/icon128.png" + } +} diff --git a/marked.min.css b/marked.min.css new file mode 100644 index 0000000..53ada59 --- /dev/null +++ b/marked.min.css @@ -0,0 +1,7 @@ + +404 Not Found + +

404 Not Found

+
nginx
+ + diff --git a/marked.min.js b/marked.min.js new file mode 100644 index 0000000..64f81db --- /dev/null +++ b/marked.min.js @@ -0,0 +1,6 @@ +/** + * marked v9.1.2 - a markdown parser + * Copyright (c) 2011-2023, Christopher Jeffrey. (MIT Licensed) + * https://github.com/markedjs/marked + */ +!function(e,t){"object"==typeof exports&&"undefined"!=typeof module?t(exports):"function"==typeof define&&define.amd?define(["exports"],t):t((e="undefined"!=typeof globalThis?globalThis:e||self).marked={})}(this,(function(e){"use strict";function t(){return{async:!1,breaks:!1,extensions:null,gfm:!0,hooks:null,pedantic:!1,renderer:null,silent:!1,tokenizer:null,walkTokens:null}}function n(t){e.defaults=t}e.defaults={async:!1,breaks:!1,extensions:null,gfm:!0,hooks:null,pedantic:!1,renderer:null,silent:!1,tokenizer:null,walkTokens:null};const s=/[&<>"']/,r=new RegExp(s.source,"g"),i=/[<>"']|&(?!(#\d{1,7}|#[Xx][a-fA-F0-9]{1,6}|\w+);)/,l=new RegExp(i.source,"g"),o={"&":"&","<":"<",">":">",'"':""","'":"'"},a=e=>o[e];function c(e,t){if(t){if(s.test(e))return e.replace(r,a)}else if(i.test(e))return e.replace(l,a);return e}const h=/&(#(?:\d+)|(?:#x[0-9A-Fa-f]+)|(?:\w+));?/gi;const p=/(^|[^\[])\^/g;function u(e,t){e="string"==typeof e?e:e.source,t=t||"";const n={replace:(t,s)=>(s=(s="object"==typeof s&&"source"in s?s.source:s).replace(p,"$1"),e=e.replace(t,s),n),getRegex:()=>new RegExp(e,t)};return n}function g(e){try{e=encodeURI(e).replace(/%25/g,"%")}catch(e){return null}return e}const k={exec:()=>null};function f(e,t){const n=e.replace(/\|/g,((e,t,n)=>{let s=!1,r=t;for(;--r>=0&&"\\"===n[r];)s=!s;return s?"|":" |"})).split(/ \|/);let s=0;if(n[0].trim()||n.shift(),n.length>0&&!n[n.length-1].trim()&&n.pop(),t)if(n.length>t)n.splice(t);else for(;n.length0)return{type:"space",raw:t[0]}}code(e){const t=this.rules.block.code.exec(e);if(t){const e=t[0].replace(/^ {1,4}/gm,"");return{type:"code",raw:t[0],codeBlockStyle:"indented",text:this.options.pedantic?e:d(e,"\n")}}}fences(e){const t=this.rules.block.fences.exec(e);if(t){const e=t[0],n=function(e,t){const n=e.match(/^(\s+)(?:```)/);if(null===n)return t;const s=n[1];return t.split("\n").map((e=>{const t=e.match(/^\s+/);if(null===t)return e;const[n]=t;return n.length>=s.length?e.slice(s.length):e})).join("\n")}(e,t[3]||"");return{type:"code",raw:e,lang:t[2]?t[2].trim().replace(this.rules.inline._escapes,"$1"):t[2],text:n}}}heading(e){const t=this.rules.block.heading.exec(e);if(t){let e=t[2].trim();if(/#$/.test(e)){const t=d(e,"#");this.options.pedantic?e=t.trim():t&&!/ $/.test(t)||(e=t.trim())}return{type:"heading",raw:t[0],depth:t[1].length,text:e,tokens:this.lexer.inline(e)}}}hr(e){const t=this.rules.block.hr.exec(e);if(t)return{type:"hr",raw:t[0]}}blockquote(e){const t=this.rules.block.blockquote.exec(e);if(t){const e=d(t[0].replace(/^ *>[ \t]?/gm,""),"\n"),n=this.lexer.state.top;this.lexer.state.top=!0;const s=this.lexer.blockTokens(e);return this.lexer.state.top=n,{type:"blockquote",raw:t[0],tokens:s,text:e}}}list(e){let t=this.rules.block.list.exec(e);if(t){let n=t[1].trim();const s=n.length>1,r={type:"list",raw:"",ordered:s,start:s?+n.slice(0,-1):"",loose:!1,items:[]};n=s?`\\d{1,9}\\${n.slice(-1)}`:`\\${n}`,this.options.pedantic&&(n=s?n:"[*+-]");const i=new RegExp(`^( {0,3}${n})((?:[\t ][^\\n]*)?(?:\\n|$))`);let l="",o="",a=!1;for(;e;){let n=!1;if(!(t=i.exec(e)))break;if(this.rules.block.hr.test(e))break;l=t[0],e=e.substring(l.length);let s=t[2].split("\n",1)[0].replace(/^\t+/,(e=>" ".repeat(3*e.length))),c=e.split("\n",1)[0],h=0;this.options.pedantic?(h=2,o=s.trimStart()):(h=t[2].search(/[^ ]/),h=h>4?1:h,o=s.slice(h),h+=t[1].length);let p=!1;if(!s&&/^ *$/.test(c)&&(l+=c+"\n",e=e.substring(c.length+1),n=!0),!n){const t=new RegExp(`^ {0,${Math.min(3,h-1)}}(?:[*+-]|\\d{1,9}[.)])((?:[ \t][^\\n]*)?(?:\\n|$))`),n=new RegExp(`^ {0,${Math.min(3,h-1)}}((?:- *){3,}|(?:_ *){3,}|(?:\\* *){3,})(?:\\n+|$)`),r=new RegExp(`^ {0,${Math.min(3,h-1)}}(?:\`\`\`|~~~)`),i=new RegExp(`^ {0,${Math.min(3,h-1)}}#`);for(;e;){const a=e.split("\n",1)[0];if(c=a,this.options.pedantic&&(c=c.replace(/^ {1,4}(?=( {4})*[^ ])/g," ")),r.test(c))break;if(i.test(c))break;if(t.test(c))break;if(n.test(e))break;if(c.search(/[^ ]/)>=h||!c.trim())o+="\n"+c.slice(h);else{if(p)break;if(s.search(/[^ ]/)>=4)break;if(r.test(s))break;if(i.test(s))break;if(n.test(s))break;o+="\n"+c}p||c.trim()||(p=!0),l+=a+"\n",e=e.substring(a.length+1),s=c.slice(h)}}r.loose||(a?r.loose=!0:/\n *\n *$/.test(l)&&(a=!0));let u,g=null;this.options.gfm&&(g=/^\[[ xX]\] /.exec(o),g&&(u="[ ] "!==g[0],o=o.replace(/^\[[ xX]\] +/,""))),r.items.push({type:"list_item",raw:l,task:!!g,checked:u,loose:!1,text:o,tokens:[]}),r.raw+=l}r.items[r.items.length-1].raw=l.trimEnd(),r.items[r.items.length-1].text=o.trimEnd(),r.raw=r.raw.trimEnd();for(let e=0;e"space"===e.type)),n=t.length>0&&t.some((e=>/\n.*\n/.test(e.raw)));r.loose=n}if(r.loose)for(let e=0;e$/,"$1").replace(this.rules.inline._escapes,"$1"):"",s=t[3]?t[3].substring(1,t[3].length-1).replace(this.rules.inline._escapes,"$1"):t[3];return{type:"def",tag:e,raw:t[0],href:n,title:s}}}table(e){const t=this.rules.block.table.exec(e);if(t){if(!/[:|]/.test(t[2]))return;const e={type:"table",raw:t[0],header:f(t[1]).map((e=>({text:e,tokens:[]}))),align:t[2].replace(/^\||\| *$/g,"").split("|"),rows:t[3]&&t[3].trim()?t[3].replace(/\n[ \t]*$/,"").split("\n"):[]};if(e.header.length===e.align.length){let t,n,s,r,i=e.align.length;for(t=0;t({text:e,tokens:[]})));for(i=e.header.length,n=0;n/i.test(t[0])&&(this.lexer.state.inLink=!1),!this.lexer.state.inRawBlock&&/^<(pre|code|kbd|script)(\s|>)/i.test(t[0])?this.lexer.state.inRawBlock=!0:this.lexer.state.inRawBlock&&/^<\/(pre|code|kbd|script)(\s|>)/i.test(t[0])&&(this.lexer.state.inRawBlock=!1),{type:"html",raw:t[0],inLink:this.lexer.state.inLink,inRawBlock:this.lexer.state.inRawBlock,block:!1,text:t[0]}}link(e){const t=this.rules.inline.link.exec(e);if(t){const e=t[2].trim();if(!this.options.pedantic&&/^$/.test(e))return;const t=d(e.slice(0,-1),"\\");if((e.length-t.length)%2==0)return}else{const e=function(e,t){if(-1===e.indexOf(t[1]))return-1;let n=0;for(let s=0;s-1){const n=(0===t[0].indexOf("!")?5:4)+t[1].length+e;t[2]=t[2].substring(0,e),t[0]=t[0].substring(0,n).trim(),t[3]=""}}let n=t[2],s="";if(this.options.pedantic){const e=/^([^'"]*[^\s])\s+(['"])(.*)\2/.exec(n);e&&(n=e[1],s=e[3])}else s=t[3]?t[3].slice(1,-1):"";return n=n.trim(),/^$/.test(e)?n.slice(1):n.slice(1,-1)),x(t,{href:n?n.replace(this.rules.inline._escapes,"$1"):n,title:s?s.replace(this.rules.inline._escapes,"$1"):s},t[0],this.lexer)}}reflink(e,t){let n;if((n=this.rules.inline.reflink.exec(e))||(n=this.rules.inline.nolink.exec(e))){let e=(n[2]||n[1]).replace(/\s+/g," ");if(e=t[e.toLowerCase()],!e){const e=n[0].charAt(0);return{type:"text",raw:e,text:e}}return x(n,e,n[0],this.lexer)}}emStrong(e,t,n=""){let s=this.rules.inline.emStrong.lDelim.exec(e);if(!s)return;if(s[3]&&n.match(/[\p{L}\p{N}]/u))return;if(!(s[1]||s[2]||"")||!n||this.rules.inline.punctuation.exec(n)){const n=[...s[0]].length-1;let r,i,l=n,o=0;const a="*"===s[0][0]?this.rules.inline.emStrong.rDelimAst:this.rules.inline.emStrong.rDelimUnd;for(a.lastIndex=0,t=t.slice(-1*e.length+s[0].length-1);null!=(s=a.exec(t));){if(r=s[1]||s[2]||s[3]||s[4]||s[5]||s[6],!r)continue;if(i=[...r].length,s[3]||s[4]){l+=i;continue}if((s[5]||s[6])&&n%3&&!((n+i)%3)){o+=i;continue}if(l-=i,l>0)continue;i=Math.min(i,i+l+o);const t=[...e].slice(0,n+s.index+i+1).join("");if(Math.min(n,i)%2){const e=t.slice(1,-1);return{type:"em",raw:t,text:e,tokens:this.lexer.inlineTokens(e)}}const a=t.slice(2,-2);return{type:"strong",raw:t,text:a,tokens:this.lexer.inlineTokens(a)}}}}codespan(e){const t=this.rules.inline.code.exec(e);if(t){let e=t[2].replace(/\n/g," ");const n=/[^ ]/.test(e),s=/^ /.test(e)&&/ $/.test(e);return n&&s&&(e=e.substring(1,e.length-1)),e=c(e,!0),{type:"codespan",raw:t[0],text:e}}}br(e){const t=this.rules.inline.br.exec(e);if(t)return{type:"br",raw:t[0]}}del(e){const t=this.rules.inline.del.exec(e);if(t)return{type:"del",raw:t[0],text:t[2],tokens:this.lexer.inlineTokens(t[2])}}autolink(e){const t=this.rules.inline.autolink.exec(e);if(t){let e,n;return"@"===t[2]?(e=c(t[1]),n="mailto:"+e):(e=c(t[1]),n=e),{type:"link",raw:t[0],text:e,href:n,tokens:[{type:"text",raw:e,text:e}]}}}url(e){let t;if(t=this.rules.inline.url.exec(e)){let e,n;if("@"===t[2])e=c(t[0]),n="mailto:"+e;else{let s;do{s=t[0],t[0]=this.rules.inline._backpedal.exec(t[0])[0]}while(s!==t[0]);e=c(t[0]),n="www."===t[1]?"http://"+t[0]:t[0]}return{type:"link",raw:t[0],text:e,href:n,tokens:[{type:"text",raw:e,text:e}]}}}inlineText(e){const t=this.rules.inline.text.exec(e);if(t){let e;return e=this.lexer.state.inRawBlock?t[0]:c(t[0]),{type:"text",raw:t[0],text:e}}}}const m={newline:/^(?: *(?:\n|$))+/,code:/^( {4}[^\n]+(?:\n(?: *(?:\n|$))*)?)+/,fences:/^ {0,3}(`{3,}(?=[^`\n]*(?:\n|$))|~{3,})([^\n]*)(?:\n|$)(?:|([\s\S]*?)(?:\n|$))(?: {0,3}\1[~`]* *(?=\n|$)|$)/,hr:/^ {0,3}((?:-[\t ]*){3,}|(?:_[ \t]*){3,}|(?:\*[ \t]*){3,})(?:\n+|$)/,heading:/^ {0,3}(#{1,6})(?=\s|$)(.*)(?:\n+|$)/,blockquote:/^( {0,3}> ?(paragraph|[^\n]*)(?:\n|$))+/,list:/^( {0,3}bull)([ \t][^\n]+?)?(?:\n|$)/,html:"^ {0,3}(?:<(script|pre|style|textarea)[\\s>][\\s\\S]*?(?:[^\\n]*\\n+|$)|comment[^\\n]*(\\n+|$)|<\\?[\\s\\S]*?(?:\\?>\\n*|$)|\\n*|$)|\\n*|$)|)[\\s\\S]*?(?:(?:\\n *)+\\n|$)|<(?!script|pre|style|textarea)([a-z][\\w-]*)(?:attribute)*? */?>(?=[ \\t]*(?:\\n|$))[\\s\\S]*?(?:(?:\\n *)+\\n|$)|(?=[ \\t]*(?:\\n|$))[\\s\\S]*?(?:(?:\\n *)+\\n|$))",def:/^ {0,3}\[(label)\]: *(?:\n *)?([^<\s][^\s]*|<.*?>)(?:(?: +(?:\n *)?| *\n *)(title))? *(?:\n+|$)/,table:k,lheading:/^(?!bull )((?:.|\n(?!\s*?\n|bull ))+?)\n {0,3}(=+|-+) *(?:\n+|$)/,_paragraph:/^([^\n]+(?:\n(?!hr|heading|lheading|blockquote|fences|list|html|table| +\n)[^\n]+)*)/,text:/^[^\n]+/,_label:/(?!\s*\])(?:\\.|[^\[\]\\])+/,_title:/(?:"(?:\\"?|[^"\\])*"|'[^'\n]*(?:\n[^'\n]+)*\n?'|\([^()]*\))/};m.def=u(m.def).replace("label",m._label).replace("title",m._title).getRegex(),m.bullet=/(?:[*+-]|\d{1,9}[.)])/,m.listItemStart=u(/^( *)(bull) */).replace("bull",m.bullet).getRegex(),m.list=u(m.list).replace(/bull/g,m.bullet).replace("hr","\\n+(?=\\1?(?:(?:- *){3,}|(?:_ *){3,}|(?:\\* *){3,})(?:\\n+|$))").replace("def","\\n+(?="+m.def.source+")").getRegex(),m._tag="address|article|aside|base|basefont|blockquote|body|caption|center|col|colgroup|dd|details|dialog|dir|div|dl|dt|fieldset|figcaption|figure|footer|form|frame|frameset|h[1-6]|head|header|hr|html|iframe|legend|li|link|main|menu|menuitem|meta|nav|noframes|ol|optgroup|option|p|param|section|source|summary|table|tbody|td|tfoot|th|thead|title|tr|track|ul",m._comment=/|$)/,m.html=u(m.html,"i").replace("comment",m._comment).replace("tag",m._tag).replace("attribute",/ +[a-zA-Z:_][\w.:-]*(?: *= *"[^"\n]*"| *= *'[^'\n]*'| *= *[^\s"'=<>`]+)?/).getRegex(),m.lheading=u(m.lheading).replace(/bull/g,m.bullet).getRegex(),m.paragraph=u(m._paragraph).replace("hr",m.hr).replace("heading"," {0,3}#{1,6} ").replace("|lheading","").replace("|table","").replace("blockquote"," {0,3}>").replace("fences"," {0,3}(?:`{3,}(?=[^`\\n]*\\n)|~{3,})[^\\n]*\\n").replace("list"," {0,3}(?:[*+-]|1[.)]) ").replace("html",")|<(?:script|pre|style|textarea|!--)").replace("tag",m._tag).getRegex(),m.blockquote=u(m.blockquote).replace("paragraph",m.paragraph).getRegex(),m.normal={...m},m.gfm={...m.normal,table:"^ *([^\\n ].*)\\n {0,3}((?:\\| *)?:?-+:? *(?:\\| *:?-+:? *)*(?:\\| *)?)(?:\\n((?:(?! *\\n|hr|heading|blockquote|code|fences|list|html).*(?:\\n|$))*)\\n*|$)"},m.gfm.table=u(m.gfm.table).replace("hr",m.hr).replace("heading"," {0,3}#{1,6} ").replace("blockquote"," {0,3}>").replace("code"," {4}[^\\n]").replace("fences"," {0,3}(?:`{3,}(?=[^`\\n]*\\n)|~{3,})[^\\n]*\\n").replace("list"," {0,3}(?:[*+-]|1[.)]) ").replace("html",")|<(?:script|pre|style|textarea|!--)").replace("tag",m._tag).getRegex(),m.gfm.paragraph=u(m._paragraph).replace("hr",m.hr).replace("heading"," {0,3}#{1,6} ").replace("|lheading","").replace("table",m.gfm.table).replace("blockquote"," {0,3}>").replace("fences"," {0,3}(?:`{3,}(?=[^`\\n]*\\n)|~{3,})[^\\n]*\\n").replace("list"," {0,3}(?:[*+-]|1[.)]) ").replace("html",")|<(?:script|pre|style|textarea|!--)").replace("tag",m._tag).getRegex(),m.pedantic={...m.normal,html:u("^ *(?:comment *(?:\\n|\\s*$)|<(tag)[\\s\\S]+? *(?:\\n{2,}|\\s*$)|\\s]*)*?/?> *(?:\\n{2,}|\\s*$))").replace("comment",m._comment).replace(/tag/g,"(?!(?:a|em|strong|small|s|cite|q|dfn|abbr|data|time|code|var|samp|kbd|sub|sup|i|b|u|mark|ruby|rt|rp|bdi|bdo|span|br|wbr|ins|del|img)\\b)\\w+(?!:|[^\\w\\s@]*@)\\b").getRegex(),def:/^ *\[([^\]]+)\]: *]+)>?(?: +(["(][^\n]+[")]))? *(?:\n+|$)/,heading:/^(#{1,6})(.*)(?:\n+|$)/,fences:k,lheading:/^(.+?)\n {0,3}(=+|-+) *(?:\n+|$)/,paragraph:u(m.normal._paragraph).replace("hr",m.hr).replace("heading"," *#{1,6} *[^\n]").replace("lheading",m.lheading).replace("blockquote"," {0,3}>").replace("|fences","").replace("|list","").replace("|html","").getRegex()};const w={escape:/^\\([!"#$%&'()*+,\-./:;<=>?@\[\]\\^_`{|}~])/,autolink:/^<(scheme:[^\s\x00-\x1f<>]*|email)>/,url:k,tag:"^comment|^|^<[a-zA-Z][\\w-]*(?:attribute)*?\\s*/?>|^<\\?[\\s\\S]*?\\?>|^|^",link:/^!?\[(label)\]\(\s*(href)(?:\s+(title))?\s*\)/,reflink:/^!?\[(label)\]\[(ref)\]/,nolink:/^!?\[(ref)\](?:\[\])?/,reflinkSearch:"reflink|nolink(?!\\()",emStrong:{lDelim:/^(?:\*+(?:((?!\*)[punct])|[^\s*]))|^_+(?:((?!_)[punct])|([^\s_]))/,rDelimAst:/^[^_*]*?__[^_*]*?\*[^_*]*?(?=__)|[^*]+(?=[^*])|(?!\*)[punct](\*+)(?=[\s]|$)|[^punct\s](\*+)(?!\*)(?=[punct\s]|$)|(?!\*)[punct\s](\*+)(?=[^punct\s])|[\s](\*+)(?!\*)(?=[punct])|(?!\*)[punct](\*+)(?!\*)(?=[punct])|[^punct\s](\*+)(?=[^punct\s])/,rDelimUnd:/^[^_*]*?\*\*[^_*]*?_[^_*]*?(?=\*\*)|[^_]+(?=[^_])|(?!_)[punct](_+)(?=[\s]|$)|[^punct\s](_+)(?!_)(?=[punct\s]|$)|(?!_)[punct\s](_+)(?=[^punct\s])|[\s](_+)(?!_)(?=[punct])|(?!_)[punct](_+)(?!_)(?=[punct])/},code:/^(`+)([^`]|[^`][\s\S]*?[^`])\1(?!`)/,br:/^( {2,}|\\)\n(?!\s*$)/,del:k,text:/^(`+|[^`])(?:(?= {2,}\n)|[\s\S]*?(?:(?=[\\`^|~"};w.punctuation=u(w.punctuation,"u").replace(/punctuation/g,w._punctuation).getRegex(),w.blockSkip=/\[[^[\]]*?\]\([^\(\)]*?\)|`[^`]*?`|<[^<>]*?>/g,w.anyPunctuation=/\\[punct]/g,w._escapes=/\\([punct])/g,w._comment=u(m._comment).replace("(?:--\x3e|$)","--\x3e").getRegex(),w.emStrong.lDelim=u(w.emStrong.lDelim,"u").replace(/punct/g,w._punctuation).getRegex(),w.emStrong.rDelimAst=u(w.emStrong.rDelimAst,"gu").replace(/punct/g,w._punctuation).getRegex(),w.emStrong.rDelimUnd=u(w.emStrong.rDelimUnd,"gu").replace(/punct/g,w._punctuation).getRegex(),w.anyPunctuation=u(w.anyPunctuation,"gu").replace(/punct/g,w._punctuation).getRegex(),w._escapes=u(w._escapes,"gu").replace(/punct/g,w._punctuation).getRegex(),w._scheme=/[a-zA-Z][a-zA-Z0-9+.-]{1,31}/,w._email=/[a-zA-Z0-9.!#$%&'*+/=?^_`{|}~-]+(@)[a-zA-Z0-9](?:[a-zA-Z0-9-]{0,61}[a-zA-Z0-9])?(?:\.[a-zA-Z0-9](?:[a-zA-Z0-9-]{0,61}[a-zA-Z0-9])?)+(?![-_])/,w.autolink=u(w.autolink).replace("scheme",w._scheme).replace("email",w._email).getRegex(),w._attribute=/\s+[a-zA-Z:_][\w.:-]*(?:\s*=\s*"[^"]*"|\s*=\s*'[^']*'|\s*=\s*[^\s"'=<>`]+)?/,w.tag=u(w.tag).replace("comment",w._comment).replace("attribute",w._attribute).getRegex(),w._label=/(?:\[(?:\\.|[^\[\]\\])*\]|\\.|`[^`]*`|[^\[\]\\`])*?/,w._href=/<(?:\\.|[^\n<>\\])+>|[^\s\x00-\x1f]*/,w._title=/"(?:\\"?|[^"\\])*"|'(?:\\'?|[^'\\])*'|\((?:\\\)?|[^)\\])*\)/,w.link=u(w.link).replace("label",w._label).replace("href",w._href).replace("title",w._title).getRegex(),w.reflink=u(w.reflink).replace("label",w._label).replace("ref",m._label).getRegex(),w.nolink=u(w.nolink).replace("ref",m._label).getRegex(),w.reflinkSearch=u(w.reflinkSearch,"g").replace("reflink",w.reflink).replace("nolink",w.nolink).getRegex(),w.normal={...w},w.pedantic={...w.normal,strong:{start:/^__|\*\*/,middle:/^__(?=\S)([\s\S]*?\S)__(?!_)|^\*\*(?=\S)([\s\S]*?\S)\*\*(?!\*)/,endAst:/\*\*(?!\*)/g,endUnd:/__(?!_)/g},em:{start:/^_|\*/,middle:/^()\*(?=\S)([\s\S]*?\S)\*(?!\*)|^_(?=\S)([\s\S]*?\S)_(?!_)/,endAst:/\*(?!\*)/g,endUnd:/_(?!_)/g},link:u(/^!?\[(label)\]\((.*?)\)/).replace("label",w._label).getRegex(),reflink:u(/^!?\[(label)\]\s*\[([^\]]*)\]/).replace("label",w._label).getRegex()},w.gfm={...w.normal,escape:u(w.escape).replace("])","~|])").getRegex(),_extended_email:/[A-Za-z0-9._+-]+(@)[a-zA-Z0-9-_]+(?:\.[a-zA-Z0-9-_]*[a-zA-Z0-9])+(?![-_])/,url:/^((?:ftp|https?):\/\/|www\.)(?:[a-zA-Z0-9\-]+\.?)+[^\s<]*|^email/,_backpedal:/(?:[^?!.,:;*_'"~()&]+|\([^)]*\)|&(?![a-zA-Z0-9]+;$)|[?!.,:;*_'"~)]+(?!$))+/,del:/^(~~?)(?=[^\s~])([\s\S]*?[^\s~])\1(?=[^~]|$)/,text:/^([`~]+|[^`~])(?:(?= {2,}\n)|(?=[a-zA-Z0-9.!#$%&'*+\/=?_`{\|}~-]+@)|[\s\S]*?(?:(?=[\\t+" ".repeat(n.length)));e;)if(!(this.options.extensions&&this.options.extensions.block&&this.options.extensions.block.some((s=>!!(n=s.call({lexer:this},e,t))&&(e=e.substring(n.raw.length),t.push(n),!0)))))if(n=this.tokenizer.space(e))e=e.substring(n.raw.length),1===n.raw.length&&t.length>0?t[t.length-1].raw+="\n":t.push(n);else if(n=this.tokenizer.code(e))e=e.substring(n.raw.length),s=t[t.length-1],!s||"paragraph"!==s.type&&"text"!==s.type?t.push(n):(s.raw+="\n"+n.raw,s.text+="\n"+n.text,this.inlineQueue[this.inlineQueue.length-1].src=s.text);else if(n=this.tokenizer.fences(e))e=e.substring(n.raw.length),t.push(n);else if(n=this.tokenizer.heading(e))e=e.substring(n.raw.length),t.push(n);else if(n=this.tokenizer.hr(e))e=e.substring(n.raw.length),t.push(n);else if(n=this.tokenizer.blockquote(e))e=e.substring(n.raw.length),t.push(n);else if(n=this.tokenizer.list(e))e=e.substring(n.raw.length),t.push(n);else if(n=this.tokenizer.html(e))e=e.substring(n.raw.length),t.push(n);else if(n=this.tokenizer.def(e))e=e.substring(n.raw.length),s=t[t.length-1],!s||"paragraph"!==s.type&&"text"!==s.type?this.tokens.links[n.tag]||(this.tokens.links[n.tag]={href:n.href,title:n.title}):(s.raw+="\n"+n.raw,s.text+="\n"+n.raw,this.inlineQueue[this.inlineQueue.length-1].src=s.text);else if(n=this.tokenizer.table(e))e=e.substring(n.raw.length),t.push(n);else if(n=this.tokenizer.lheading(e))e=e.substring(n.raw.length),t.push(n);else{if(r=e,this.options.extensions&&this.options.extensions.startBlock){let t=1/0;const n=e.slice(1);let s;this.options.extensions.startBlock.forEach((e=>{s=e.call({lexer:this},n),"number"==typeof s&&s>=0&&(t=Math.min(t,s))})),t<1/0&&t>=0&&(r=e.substring(0,t+1))}if(this.state.top&&(n=this.tokenizer.paragraph(r)))s=t[t.length-1],i&&"paragraph"===s.type?(s.raw+="\n"+n.raw,s.text+="\n"+n.text,this.inlineQueue.pop(),this.inlineQueue[this.inlineQueue.length-1].src=s.text):t.push(n),i=r.length!==e.length,e=e.substring(n.raw.length);else if(n=this.tokenizer.text(e))e=e.substring(n.raw.length),s=t[t.length-1],s&&"text"===s.type?(s.raw+="\n"+n.raw,s.text+="\n"+n.text,this.inlineQueue.pop(),this.inlineQueue[this.inlineQueue.length-1].src=s.text):t.push(n);else if(e){const t="Infinite loop on byte: "+e.charCodeAt(0);if(this.options.silent){console.error(t);break}throw new Error(t)}}return this.state.top=!0,t}inline(e,t=[]){return this.inlineQueue.push({src:e,tokens:t}),t}inlineTokens(e,t=[]){let n,s,r,i,l,o,a=e;if(this.tokens.links){const e=Object.keys(this.tokens.links);if(e.length>0)for(;null!=(i=this.tokenizer.rules.inline.reflinkSearch.exec(a));)e.includes(i[0].slice(i[0].lastIndexOf("[")+1,-1))&&(a=a.slice(0,i.index)+"["+"a".repeat(i[0].length-2)+"]"+a.slice(this.tokenizer.rules.inline.reflinkSearch.lastIndex))}for(;null!=(i=this.tokenizer.rules.inline.blockSkip.exec(a));)a=a.slice(0,i.index)+"["+"a".repeat(i[0].length-2)+"]"+a.slice(this.tokenizer.rules.inline.blockSkip.lastIndex);for(;null!=(i=this.tokenizer.rules.inline.anyPunctuation.exec(a));)a=a.slice(0,i.index)+"++"+a.slice(this.tokenizer.rules.inline.anyPunctuation.lastIndex);for(;e;)if(l||(o=""),l=!1,!(this.options.extensions&&this.options.extensions.inline&&this.options.extensions.inline.some((s=>!!(n=s.call({lexer:this},e,t))&&(e=e.substring(n.raw.length),t.push(n),!0)))))if(n=this.tokenizer.escape(e))e=e.substring(n.raw.length),t.push(n);else if(n=this.tokenizer.tag(e))e=e.substring(n.raw.length),s=t[t.length-1],s&&"text"===n.type&&"text"===s.type?(s.raw+=n.raw,s.text+=n.text):t.push(n);else if(n=this.tokenizer.link(e))e=e.substring(n.raw.length),t.push(n);else if(n=this.tokenizer.reflink(e,this.tokens.links))e=e.substring(n.raw.length),s=t[t.length-1],s&&"text"===n.type&&"text"===s.type?(s.raw+=n.raw,s.text+=n.text):t.push(n);else if(n=this.tokenizer.emStrong(e,a,o))e=e.substring(n.raw.length),t.push(n);else if(n=this.tokenizer.codespan(e))e=e.substring(n.raw.length),t.push(n);else if(n=this.tokenizer.br(e))e=e.substring(n.raw.length),t.push(n);else if(n=this.tokenizer.del(e))e=e.substring(n.raw.length),t.push(n);else if(n=this.tokenizer.autolink(e))e=e.substring(n.raw.length),t.push(n);else if(this.state.inLink||!(n=this.tokenizer.url(e))){if(r=e,this.options.extensions&&this.options.extensions.startInline){let t=1/0;const n=e.slice(1);let s;this.options.extensions.startInline.forEach((e=>{s=e.call({lexer:this},n),"number"==typeof s&&s>=0&&(t=Math.min(t,s))})),t<1/0&&t>=0&&(r=e.substring(0,t+1))}if(n=this.tokenizer.inlineText(r))e=e.substring(n.raw.length),"_"!==n.raw.slice(-1)&&(o=n.raw.slice(-1)),l=!0,s=t[t.length-1],s&&"text"===s.type?(s.raw+=n.raw,s.text+=n.text):t.push(n);else if(e){const t="Infinite loop on byte: "+e.charCodeAt(0);if(this.options.silent){console.error(t);break}throw new Error(t)}}else e=e.substring(n.raw.length),t.push(n);return t}}class y{options;constructor(t){this.options=t||e.defaults}code(e,t,n){const s=(t||"").match(/^\S*/)?.[0];return e=e.replace(/\n$/,"")+"\n",s?'
'+(n?e:c(e,!0))+"
\n":"
"+(n?e:c(e,!0))+"
\n"}blockquote(e){return`
\n${e}
\n`}html(e,t){return e}heading(e,t,n){return`${e}\n`}hr(){return"
\n"}list(e,t,n){const s=t?"ol":"ul";return"<"+s+(t&&1!==n?' start="'+n+'"':"")+">\n"+e+"\n"}listitem(e,t,n){return`
  • ${e}
  • \n`}checkbox(e){return"'}paragraph(e){return`

    ${e}

    \n`}table(e,t){return t&&(t=`${t}`),"\n\n"+e+"\n"+t+"
    \n"}tablerow(e){return`\n${e}\n`}tablecell(e,t){const n=t.header?"th":"td";return(t.align?`<${n} align="${t.align}">`:`<${n}>`)+e+`\n`}strong(e){return`${e}`}em(e){return`${e}`}codespan(e){return`${e}`}br(){return"
    "}del(e){return`${e}`}link(e,t,n){const s=g(e);if(null===s)return n;let r='",r}image(e,t,n){const s=g(e);if(null===s)return n;let r=`${n}"colon"===(t=t.toLowerCase())?":":"#"===t.charAt(0)?"x"===t.charAt(1)?String.fromCharCode(parseInt(t.substring(2),16)):String.fromCharCode(+t.substring(1)):"")));continue}case"code":{const e=r;n+=this.renderer.code(e.text,e.lang,!!e.escaped);continue}case"table":{const e=r;let t="",s="";for(let t=0;t0&&"paragraph"===n.tokens[0].type?(n.tokens[0].text=e+" "+n.tokens[0].text,n.tokens[0].tokens&&n.tokens[0].tokens.length>0&&"text"===n.tokens[0].tokens[0].type&&(n.tokens[0].tokens[0].text=e+" "+n.tokens[0].tokens[0].text)):n.tokens.unshift({type:"text",text:e+" "}):o+=e+" "}o+=this.parse(n.tokens,i),l+=this.renderer.listitem(o,r,!!s)}n+=this.renderer.list(l,t,s);continue}case"html":{const e=r;n+=this.renderer.html(e.text,e.block);continue}case"paragraph":{const e=r;n+=this.renderer.paragraph(this.parseInline(e.tokens));continue}case"text":{let i=r,l=i.tokens?this.parseInline(i.tokens):i.text;for(;s+1{n=n.concat(this.walkTokens(e[s],t))})):e.tokens&&(n=n.concat(this.walkTokens(e.tokens,t)))}}return n}use(...e){const t=this.defaults.extensions||{renderers:{},childTokens:{}};return e.forEach((e=>{const n={...e};if(n.async=this.defaults.async||n.async||!1,e.extensions&&(e.extensions.forEach((e=>{if(!e.name)throw new Error("extension name required");if("renderer"in e){const n=t.renderers[e.name];t.renderers[e.name]=n?function(...t){let s=e.renderer.apply(this,t);return!1===s&&(s=n.apply(this,t)),s}:e.renderer}if("tokenizer"in e){if(!e.level||"block"!==e.level&&"inline"!==e.level)throw new Error("extension level must be 'block' or 'inline'");const n=t[e.level];n?n.unshift(e.tokenizer):t[e.level]=[e.tokenizer],e.start&&("block"===e.level?t.startBlock?t.startBlock.push(e.start):t.startBlock=[e.start]:"inline"===e.level&&(t.startInline?t.startInline.push(e.start):t.startInline=[e.start]))}"childTokens"in e&&e.childTokens&&(t.childTokens[e.name]=e.childTokens)})),n.extensions=t),e.renderer){const t=this.defaults.renderer||new y(this.defaults);for(const n in e.renderer){const s=e.renderer[n],r=n,i=t[r];t[r]=(...e)=>{let n=s.apply(t,e);return!1===n&&(n=i.apply(t,e)),n||""}}n.renderer=t}if(e.tokenizer){const t=this.defaults.tokenizer||new b(this.defaults);for(const n in e.tokenizer){const s=e.tokenizer[n],r=n,i=t[r];t[r]=(...e)=>{let n=s.apply(t,e);return!1===n&&(n=i.apply(t,e)),n}}n.tokenizer=t}if(e.hooks){const t=this.defaults.hooks||new T;for(const n in e.hooks){const s=e.hooks[n],r=n,i=t[r];T.passThroughHooks.has(n)?t[r]=e=>{if(this.defaults.async)return Promise.resolve(s.call(t,e)).then((e=>i.call(t,e)));const n=s.call(t,e);return i.call(t,n)}:t[r]=(...e)=>{let n=s.apply(t,e);return!1===n&&(n=i.apply(t,e)),n}}n.hooks=t}if(e.walkTokens){const t=this.defaults.walkTokens,s=e.walkTokens;n.walkTokens=function(e){let n=[];return n.push(s.call(this,e)),t&&(n=n.concat(t.call(this,e))),n}}this.defaults={...this.defaults,...n}})),this}setOptions(e){return this.defaults={...this.defaults,...e},this}#e(e,t){return(n,s)=>{const r={...s},i={...this.defaults,...r};!0===this.defaults.async&&!1===r.async&&(i.silent||console.warn("marked(): The async option was set to true by an extension. The async: false option sent to parse will be ignored."),i.async=!0);const l=this.#t(!!i.silent,!!i.async);if(null==n)return l(new Error("marked(): input parameter is undefined or null"));if("string"!=typeof n)return l(new Error("marked(): input parameter is of type "+Object.prototype.toString.call(n)+", string expected"));if(i.hooks&&(i.hooks.options=i),i.async)return Promise.resolve(i.hooks?i.hooks.preprocess(n):n).then((t=>e(t,i))).then((e=>i.walkTokens?Promise.all(this.walkTokens(e,i.walkTokens)).then((()=>e)):e)).then((e=>t(e,i))).then((e=>i.hooks?i.hooks.postprocess(e):e)).catch(l);try{i.hooks&&(n=i.hooks.preprocess(n));const s=e(n,i);i.walkTokens&&this.walkTokens(s,i.walkTokens);let r=t(s,i);return i.hooks&&(r=i.hooks.postprocess(r)),r}catch(e){return l(e)}}}#t(e,t){return n=>{if(n.message+="\nPlease report this to https://github.com/markedjs/marked.",e){const e="

    An error occurred:

    "+c(n.message+"",!0)+"
    ";return t?Promise.resolve(e):e}if(t)return Promise.reject(n);throw n}}}const S=new R;function A(e,t){return S.parse(e,t)}A.options=A.setOptions=function(e){return S.setOptions(e),A.defaults=S.defaults,n(A.defaults),A},A.getDefaults=t,A.defaults=e.defaults,A.use=function(...e){return S.use(...e),A.defaults=S.defaults,n(A.defaults),A},A.walkTokens=function(e,t){return S.walkTokens(e,t)},A.parseInline=S.parseInline,A.Parser=z,A.parser=z.parse,A.Renderer=y,A.TextRenderer=$,A.Lexer=_,A.lexer=_.lex,A.Tokenizer=b,A.Hooks=T,A.parse=A;const I=A.options,E=A.setOptions,Z=A.use,q=A.walkTokens,L=A.parseInline,D=A,P=z.parse,v=_.lex;e.Hooks=T,e.Lexer=_,e.Marked=R,e.Parser=z,e.Renderer=y,e.TextRenderer=$,e.Tokenizer=b,e.getDefaults=t,e.lexer=v,e.marked=A,e.options=I,e.parse=D,e.parseInline=L,e.parser=P,e.setOptions=E,e.use=Z,e.walkTokens=q})); diff --git a/sidepanel.css b/sidepanel.css new file mode 100644 index 0000000..32e7824 --- /dev/null +++ b/sidepanel.css @@ -0,0 +1,579 @@ +/* ── Reset ─────────────────────────────────────────────────── */ +*, *::before, *::after { box-sizing: border-box; margin: 0; padding: 0; } + +/* ── Dark theme (default) ──────────────────────────────────── */ +:root { + --bg0: #111114; + --bg1: #1a1a1f; + --bg2: #222228; + --bg3: #2a2a32; + --bg4: #32323c; + --fg1: #e2e2e8; + --fg2: #8888a0; + --fg3: #55556a; + --accent: #7c3aed; + --acc-h: #6d28d9; + --acc-d: rgba(124,58,237,.15); + --acc-dd: rgba(124,58,237,.08); + --border: #2e2e38; + --focus: #7c3aed; + --ok: #22c55e; + --err: #ef4444; + --warn: #f59e0b; + --scroll: #3a3a48; + --code-bg: #0d0d12; + --ububble: #1d4ed8; + --ububble-text: #fff; + --abubble: #1e1e26; + --abubble-border: #2e2e38; + --think-bg: rgba(245,158,11,.07); + --think-border: rgba(245,158,11,.25); + --think-text: #c8a840; + --meta-text: #66667a; + --chip-bg: #2a2a38; + --chip-text: #a0a0c0; + --danger: #ef4444; +} + +/* ── Light theme ───────────────────────────────────────────── */ +[data-theme="light"] { + --bg0: #f0f2f5; + --bg1: #ffffff; + --bg2: #f8f8fb; + --bg3: #eeeff4; + --bg4: #e4e5ec; + --fg1: #1a1a2e; + --fg2: #55556e; + --fg3: #9999b0; + --border: #dddde8; + --focus: #7c3aed; + --ok: #16a34a; + --err: #dc2626; + --warn: #d97706; + --scroll: #ccccdc; + --code-bg: #f4f4f8; + --ububble: #2563eb; + --ububble-text: #fff; + --abubble: #ffffff; + --abubble-border: #dddde8; + --think-bg: rgba(217,119,6,.07); + --think-border: rgba(217,119,6,.3); + --think-text: #92660a; + --meta-text: #9999b0; + --chip-bg: #e8e8f2; + --chip-text: #5555a0; +} + +/* ── Base ──────────────────────────────────────────────────── */ +html, body { + width: 100%; height: 100%; overflow: hidden; + font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', system-ui, sans-serif; + font-size: 13px; line-height: 1.5; + background: var(--bg0); color: var(--fg1); + -webkit-font-smoothing: antialiased; +} + +#app { + display: flex; flex-direction: column; + height: 100vh; width: 100%; overflow: hidden; + background: var(--bg0); color: var(--fg1); + transition: background .2s, color .2s; +} + +/* ── Scrollbar ─────────────────────────────────────────────── */ +::-webkit-scrollbar { width: 4px; height: 4px; } +::-webkit-scrollbar-track { background: transparent; } +::-webkit-scrollbar-thumb { background: var(--scroll); border-radius: 2px; } + +/* ── Header ────────────────────────────────────────────────── */ +header { + display: flex; align-items: center; justify-content: space-between; + padding: 8px 11px; + background: var(--bg1); border-bottom: 1px solid var(--border); + flex-shrink: 0; min-height: 40px; z-index: 2; +} +.h-left, .h-right { display: flex; align-items: center; gap: 5px; } +.app-title { font-weight: 700; font-size: 13px; letter-spacing: .02em; } + +.status-dot { + width: 7px; height: 7px; border-radius: 50%; + background: var(--warn); flex-shrink: 0; + transition: background .3s; +} +.status-dot.connected { background: var(--ok); } +.status-dot.error { background: var(--err); } + +/* ── Buttons ───────────────────────────────────────────────── */ +.icon-btn { + background: none; border: none; cursor: pointer; + color: var(--fg2); font-size: 16px; + padding: 4px 5px; border-radius: 5px; line-height: 1; + transition: background .12s, color .12s; + display: inline-flex; align-items: center; justify-content: center; +} +.icon-btn:hover { background: var(--bg3); color: var(--fg1); } +.icon-btn.active { color: var(--accent); } +.icon-btn.sm { font-size: 13px; padding: 3px 4px; } +.icon-btn.danger:hover { color: var(--err); } + +.primary-btn { + width: 100%; padding: 8px 12px; + background: var(--accent); color: #fff; + border: none; border-radius: 6px; cursor: pointer; + font-size: 13px; font-weight: 500; font-family: inherit; + transition: background .12s; +} +.primary-btn:hover { background: var(--acc-h); } +.primary-btn:active { transform: scale(.98); } +.primary-btn.sm { padding: 5px 10px; font-size: 12px; width: auto; } + +.pill-btn { + background: var(--bg3); border: 1px solid var(--border); + color: var(--fg2); padding: 3px 10px; + border-radius: 20px; cursor: pointer; font-size: 11.5px; font-family: inherit; + transition: background .12s, border-color .12s; + white-space: nowrap; +} +.pill-btn:hover { background: var(--bg4); border-color: var(--accent); color: var(--fg1); } + +/* ── Drop panels (settings + sessions) ────────────────────── */ +.drop-panel { + background: var(--bg1); border-bottom: 1px solid var(--border); + padding: 11px; flex-shrink: 0; + display: flex; flex-direction: column; gap: 9px; + overflow-y: auto; max-height: 52vh; +} +.drop-panel.hidden { display: none; } + +.field-group { display: flex; flex-direction: column; gap: 4px; } +.field-lbl { + font-size: 10px; font-weight: 600; text-transform: uppercase; + letter-spacing: .06em; color: var(--fg2); + display: flex; align-items: center; gap: 4px; +} +.hint { text-transform: none; font-weight: 400; letter-spacing: 0; color: var(--fg3); } + +.field-group input { + width: 100%; background: var(--bg0); + border: 1px solid var(--border); color: var(--fg1); + padding: 6px 9px; border-radius: 6px; + font-size: 13px; font-family: inherit; outline: none; + transition: border-color .12s; +} +.field-group input:focus { border-color: var(--focus); } + +.input-row-inline { display: flex; gap: 4px; align-items: center; } +.input-row-inline input { flex: 1; } + +.status-line { + font-size: 11.5px; text-align: center; min-height: 14px; + color: var(--fg3); transition: color .2s; +} +.status-line.ok { color: var(--ok); } +.status-line.err { color: var(--err); } + +/* ── Sessions panel ────────────────────────────────────────── */ +.panel-hdr { + display: flex; align-items: center; justify-content: space-between; + margin-bottom: 2px; +} +.panel-title { font-size: 11px; font-weight: 600; text-transform: uppercase; + letter-spacing: .06em; color: var(--fg2); } + +.sessions-list { display: flex; flex-direction: column; gap: 2px; } + +.session-item { + display: flex; align-items: center; gap: 6px; + padding: 7px 9px; border-radius: 7px; cursor: pointer; + border: 1px solid transparent; + transition: background .12s, border-color .12s; + min-height: 36px; +} +.session-item:hover { background: var(--bg3); } +.session-item.active { background: var(--acc-d); border-color: var(--accent); } + +.session-item-name { + flex: 1; font-size: 12.5px; white-space: nowrap; + overflow: hidden; text-overflow: ellipsis; min-width: 0; +} +.session-item-meta { font-size: 10.5px; color: var(--fg3); white-space: nowrap; } +.session-item-actions { display: flex; gap: 2px; opacity: 0; transition: opacity .15s; } +.session-item:hover .session-item-actions { opacity: 1; } + +/* ── Session bar ───────────────────────────────────────────── */ +.session-bar { + display: flex; align-items: center; gap: 6px; + padding: 5px 10px 5px 12px; + background: var(--bg1); border-bottom: 1px solid var(--border); + flex-shrink: 0; min-height: 34px; +} +.session-name { + flex: 1; font-size: 12.5px; font-weight: 500; + white-space: nowrap; overflow: hidden; text-overflow: ellipsis; + cursor: text; color: var(--fg1); +} +.session-bar-actions { display: flex; gap: 2px; flex-shrink: 0; } + +/* ── Chat area ─────────────────────────────────────────────── */ +.chat-area { + flex: 1; overflow-y: auto; padding: 12px; + display: flex; flex-direction: column; scroll-behavior: smooth; +} + +.messages { display: flex; flex-direction: column; gap: 14px; flex: 1; } + +/* ── Empty state ───────────────────────────────────────────── */ +.empty-state { + display: flex; flex-direction: column; align-items: center; + justify-content: center; flex: 1; gap: 10px; + text-align: center; padding: 32px 16px; user-select: none; +} +.empty-state .ei { font-size: 34px; line-height: 1; } +.empty-state p { font-size: 13px; line-height: 1.7; max-width: 230px; color: var(--fg2); } + +/* ── Messages ──────────────────────────────────────────────── */ +.message { + display: flex; flex-direction: column; max-width: 100%; + animation: msgIn .15s ease-out; +} +@keyframes msgIn { from { opacity:0; transform:translateY(4px); } to { opacity:1; transform:translateY(0); } } + +.message.user { align-items: flex-end; } +.message.assistant { align-items: flex-start; } + +.msg-role { + font-size: 9.5px; color: var(--fg3); margin-bottom: 3px; padding: 0 4px; + font-weight: 600; letter-spacing: .05em; text-transform: uppercase; +} + +.msg-body { display: flex; flex-direction: column; max-width: 92%; gap: 3px; } + +.msg-bubble { + padding: 8px 13px; border-radius: 14px; + word-break: break-word; overflow-wrap: break-word; + font-size: 13px; line-height: 1.65; white-space: pre-wrap; +} + +.message.user .msg-bubble { + background: var(--ububble); color: var(--ububble-text); + border-bottom-right-radius: 4px; +} +.message.assistant .msg-bubble { + background: var(--abubble); color: var(--fg1); + border: 1px solid var(--abubble-border); + border-bottom-left-radius: 4px; white-space: normal; +} + +/* Streaming cursor */ +.msg-bubble.streaming::after { + content: '\25AE'; display: inline-block; margin-left: 2px; + color: var(--accent); animation: cblink .7s step-end infinite; +} +@keyframes cblink { 50% { opacity: 0; } } + +/* Error bubble */ +.message.error .msg-bubble { + background: rgba(239,68,68,.1); + border: 1px solid rgba(239,68,68,.3); color: #fca5a5; +} + +/* Inline stream error note */ +.stream-err-note { + margin-top: 6px; padding: 5px 9px; + background: rgba(239,68,68,.08); border: 1px solid rgba(239,68,68,.25); + border-radius: 5px; color: #fca5a5; font-size: 11.5px; +} + +/* ── Thinking block ────────────────────────────────────────── */ +.think-block { + border: 1px solid var(--think-border); + background: var(--think-bg); border-radius: 8px; + overflow: hidden; font-size: 12px; max-width: 100%; +} +.think-block summary { + padding: 5px 10px; cursor: pointer; user-select: none; + color: var(--think-text); font-weight: 500; font-size: 11.5px; + list-style: none; display: flex; align-items: center; gap: 5px; +} +.think-block summary::-webkit-details-marker { display: none; } +.think-block summary::before { content: '▶'; font-size: 9px; transition: transform .2s; } +.think-block[open] summary::before { transform: rotate(90deg); } +.think-content { + padding: 8px 10px 9px; color: var(--think-text); + border-top: 1px solid var(--think-border); + max-height: 200px; overflow-y: auto; + white-space: pre-wrap; word-break: break-word; + font-size: 12px; line-height: 1.55; +} + +/* ── Message metadata ──────────────────────────────────────── */ +.msg-meta { display: flex; flex-direction: column; gap: 3px; } + +.source-chip { + display: inline-flex; align-items: center; gap: 4px; + padding: 2px 8px; border-radius: 4px; + background: var(--acc-dd); color: var(--accent); + font-size: 11px; border: 1px solid rgba(124,58,237,.2); + max-width: 100%; overflow: hidden; text-overflow: ellipsis; white-space: nowrap; +} + +.metrics-line { + font-size: 10.5px; color: var(--meta-text); + padding-left: 2px; +} + +/* ── Bubble action buttons ─────────────────────────────────── */ +.msg-actions { + display: flex; gap: 3px; padding: 1px 2px; + opacity: 0; transition: opacity .15s; +} +.message:hover .msg-actions { opacity: 1; } +.act-btn { + background: var(--bg3); border: 1px solid var(--border); + color: var(--fg2); padding: 2px 7px; + border-radius: 4px; cursor: pointer; font-size: 11.5px; + font-family: inherit; transition: background .12s, color .12s, border-color .12s; + display: inline-flex; align-items: center; gap: 3px; +} +.act-btn:hover { background: var(--bg4); border-color: var(--accent); color: var(--fg1); } + +/* Edit mode */ +.edit-textarea { + width: 100%; background: var(--bg0); border: 1px solid var(--focus); + color: var(--fg1); padding: 8px 10px; border-radius: 8px; + font-size: 13px; font-family: inherit; resize: vertical; + outline: none; min-height: 60px; max-height: 180px; +} +.edit-actions { display: flex; gap: 6px; margin-top: 5px; } +.secondary-btn { + background: var(--bg3); border: 1px solid var(--border); + color: var(--fg2); padding: 5px 12px; + border-radius: 6px; cursor: pointer; font-size: 12px; font-family: inherit; + transition: background .12s; +} +.secondary-btn:hover { background: var(--bg4); color: var(--fg1); } + +/* Code inside bubbles */ +.msg-bubble pre { + background: var(--code-bg); border: 1px solid var(--bg3); + border-radius: 6px; padding: 9px 11px; + overflow-x: auto; margin: 6px 0; white-space: pre; +} +.msg-bubble code { + font-family: 'Fira Code','Cascadia Code','Menlo','Consolas', monospace; font-size: 12px; +} +.msg-bubble pre code { background: none; padding: 0; font-size: 11.5px; line-height: 1.5; } +.msg-bubble :not(pre) > code { + background: rgba(0,0,0,.25); padding: 1px 5px; border-radius: 3px; font-size: 12px; +} +[data-theme="light"] .msg-bubble :not(pre) > code { background: rgba(0,0,0,.07); } +.msg-bubble strong { font-weight: 600; } +.msg-bubble em { font-style: italic; } + +/* ── Input area ────────────────────────────────────────────── */ +.input-area { + flex-shrink: 0; background: var(--bg1); + border-top: 1px solid var(--border); + padding: 8px 11px 11px; + display: flex; flex-direction: column; gap: 6px; +} + +/* Collapsibles (system prompt, params) */ +.collapsible { + background: var(--bg2); border: 1px solid var(--border); + border-radius: 8px; overflow: hidden; +} +.collapsible.hidden { display: none; } +.coll-hdr { + display: flex; align-items: center; justify-content: space-between; + padding: 5px 10px; background: var(--bg3); + font-size: 11px; font-weight: 600; text-transform: uppercase; + letter-spacing: .05em; color: var(--fg2); +} + +.collapsible textarea, #input-system-prompt { + width: 100%; background: var(--bg2); border: none; + color: var(--fg1); padding: 8px 10px; + font-size: 13px; font-family: inherit; resize: none; + outline: none; max-height: 100px; overflow-y: auto; + border-top: 1px solid var(--border); +} + +/* Params grid */ +.params-grid { + padding: 8px 10px; display: grid; + grid-template-columns: auto 1fr auto; + gap: 5px 8px; align-items: center; +} +.param-lbl { font-size: 11.5px; color: var(--fg2); white-space: nowrap; } +.param-input { + width: 100%; background: var(--bg0); border: 1px solid var(--border); + color: var(--fg1); padding: 3px 6px; border-radius: 5px; + font-size: 12px; font-family: inherit; outline: none; +} +.param-input:focus { border-color: var(--focus); } +.param-val { font-size: 11px; color: var(--fg3); text-align: right; min-width: 28px; } + +/* File chips */ +.file-chips-area { display: flex; flex-wrap: wrap; gap: 5px; } +.file-chips-area.hidden { display: none; } +.file-chip { + display: inline-flex; align-items: center; gap: 5px; + background: var(--chip-bg); color: var(--chip-text); + border: 1px solid var(--border); border-radius: 5px; + padding: 3px 8px; font-size: 11.5px; max-width: 160px; +} +.file-chip-name { overflow: hidden; text-overflow: ellipsis; white-space: nowrap; min-width: 0; } +.file-chip-del { + background: none; border: none; cursor: pointer; + color: var(--fg3); font-size: 11px; padding: 0 1px; + flex-shrink: 0; +} +.file-chip-del:hover { color: var(--err); } + +/* Toolbar */ +.toolbar { display: flex; align-items: center; gap: 5px; } +.tb-spacer { flex: 1; } +.toolbar-btn { + display: inline-flex; align-items: center; gap: 4px; + background: var(--bg3); border: 1px solid var(--border); + color: var(--fg2); padding: 4px 8px; border-radius: 5px; + cursor: pointer; font-size: 11.5px; font-family: inherit; + transition: border-color .12s, color .12s, background .12s; + white-space: nowrap; max-width: 130px; +} +.toolbar-btn:hover { border-color: var(--accent); color: var(--fg1); } +.toolbar-btn.active { background: var(--acc-d); border-color: var(--accent); color: var(--accent); } +.tbi { font-size: 12px; flex-shrink: 0; line-height: 1; } + +/* Page context label truncation */ +#page-ctx-lbl { overflow: hidden; text-overflow: ellipsis; white-space: nowrap; max-width: 90px; } + +/* Textarea */ +#input-message { + width: 100%; background: var(--bg0); border: 1px solid var(--border); + color: var(--fg1); padding: 8px 10px; border-radius: 8px; + font-size: 13px; font-family: inherit; line-height: 1.5; + resize: none; outline: none; + min-height: 54px; max-height: 160px; + transition: border-color .12s; overflow-y: auto; display: block; +} +#input-message:focus { border-color: var(--focus); } +#input-message::placeholder { color: var(--fg3); opacity: .7; } +#input-message:disabled { opacity: .5; cursor: not-allowed; } + +/* Template dropdown */ +.tmpl-dropdown { + background: var(--bg2); border: 1px solid var(--border); + border-radius: 8px; overflow: hidden; z-index: 10; +} +.tmpl-dropdown.hidden { display: none; } +.tmpl-dd-header { + display: flex; align-items: center; justify-content: space-between; + padding: 6px 10px; background: var(--bg3); + font-size: 11px; font-weight: 600; text-transform: uppercase; + letter-spacing: .05em; color: var(--fg2); +} +.tmpl-dd-list { max-height: 160px; overflow-y: auto; } +.tmpl-dd-item { + padding: 7px 11px; cursor: pointer; + border-bottom: 1px solid var(--border); + display: flex; flex-direction: column; gap: 2px; + transition: background .1s; +} +.tmpl-dd-item:last-child { border-bottom: none; } +.tmpl-dd-item:hover { background: var(--bg3); } +.tmpl-dd-name { font-size: 12.5px; color: var(--fg1); font-weight: 500; } +.tmpl-dd-preview { font-size: 11px; color: var(--fg3); white-space: nowrap; overflow: hidden; text-overflow: ellipsis; } + +/* Send row */ +.send-row { display: flex; gap: 6px; align-items: center; } + +#select-model { + flex: 1; min-width: 0; + background: var(--bg0); border: 1px solid var(--border); + color: var(--fg1); padding: 6px 24px 6px 8px; + border-radius: 7px; font-size: 12px; font-family: inherit; + outline: none; cursor: pointer; + -webkit-appearance: none; appearance: none; + background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='10' height='6'%3E%3Cpath d='M0 0l5 6 5-6z' fill='%23666'/%3E%3C/svg%3E"); + background-repeat: no-repeat; background-position: right 7px center; + transition: border-color .12s; +} +#select-model:focus { border-color: var(--focus); } +#select-model option { background: var(--bg2); color: var(--fg1); } + +.send-btn { + width: 34px; height: 34px; flex-shrink: 0; + background: var(--accent); color: #fff; border: none; + border-radius: 7px; cursor: pointer; font-size: 13px; + display: inline-flex; align-items: center; justify-content: center; + transition: background .12s, transform .1s; +} +.send-btn:hover { background: var(--acc-h); } +.send-btn:active { transform: scale(.93); } +.send-btn:disabled { background: var(--bg4); color: var(--fg3); cursor: not-allowed; transform: none; } + +.stop-btn { + width: 34px; height: 34px; flex-shrink: 0; + background: var(--err); color: #fff; border: none; + border-radius: 7px; cursor: pointer; font-size: 12px; + display: inline-flex; align-items: center; justify-content: center; + transition: background .12s; +} +.stop-btn:hover { background: #dc2626; } +.stop-btn.hidden { display: none; } +.send-btn.hidden { display: none; } + +/* ── Template manager overlay ──────────────────────────────── */ +.overlay { + position: absolute; inset: 0; z-index: 20; + background: rgba(0,0,0,.55); + display: flex; align-items: flex-start; justify-content: center; + padding-top: 40px; +} +.overlay.hidden { display: none; } + +.overlay-panel { + background: var(--bg1); border: 1px solid var(--border); + border-radius: 10px; width: calc(100% - 24px); max-width: 380px; + display: flex; flex-direction: column; max-height: 80vh; overflow: hidden; +} +.overlay-hdr { + display: flex; align-items: center; justify-content: space-between; + padding: 10px 14px; border-bottom: 1px solid var(--border); + font-weight: 600; font-size: 14px; flex-shrink: 0; +} + +.tmpl-list { + overflow-y: auto; flex: 1; display: flex; flex-direction: column; gap: 0; +} +.tmpl-item { + display: flex; align-items: flex-start; gap: 8px; + padding: 9px 13px; border-bottom: 1px solid var(--border); + transition: background .1s; +} +.tmpl-item:hover { background: var(--bg2); } +.tmpl-item-body { flex: 1; min-width: 0; } +.tmpl-item-name { font-size: 13px; font-weight: 500; color: var(--fg1); } +.tmpl-item-text { font-size: 11.5px; color: var(--fg3); white-space: nowrap; overflow: hidden; text-overflow: ellipsis; margin-top: 2px; } +.tmpl-item-del { + background: none; border: none; cursor: pointer; + color: var(--fg3); font-size: 13px; padding: 2px 4px; flex-shrink: 0; +} +.tmpl-item-del:hover { color: var(--err); } + +.tmpl-add-form { + padding: 11px 13px; border-top: 1px solid var(--border); + display: flex; flex-direction: column; gap: 7px; flex-shrink: 0; +} +.tmpl-add-form input, .tmpl-add-form textarea { + background: var(--bg0); border: 1px solid var(--border); + color: var(--fg1); padding: 7px 9px; border-radius: 6px; + font-size: 13px; font-family: inherit; outline: none; +} +.tmpl-add-form input:focus, .tmpl-add-form textarea:focus { border-color: var(--focus); } +.tmpl-add-form textarea { resize: vertical; min-height: 70px; } + +/* ── Relative positioning for overlay ─────────────────────── */ +#app { position: relative; } diff --git a/sidepanel.html b/sidepanel.html new file mode 100644 index 0000000..2f55738 --- /dev/null +++ b/sidepanel.html @@ -0,0 +1,167 @@ + + + + + + Ollama Sidebar + + + +
    + + +
    +
    + + Ollama + +
    +
    + + +
    +
    + + + + + + + + +
    + +
    + + + +
    +
    + + +
    +
    +
    + + +
    + + + + + +
    + + + +
    + +
    + + + + + + + + +
    + + + +
    + +
    + + + + + + + +
    + + + + + + diff --git a/sidepanel.js b/sidepanel.js new file mode 100644 index 0000000..fb86caf --- /dev/null +++ b/sidepanel.js @@ -0,0 +1,1535 @@ +(function () { +'use strict'; + +// ── Constants ───────────────────────────────────────────────────────────────── + +var DEF_PARAMS = { + temperature: 0.7, + top_p: 0.9, + top_k: 40, + repeat_penalty: 1.1, + seed: -1, + num_ctx: 4096, + num_predict: -1 +}; + +var DEFAULT_SYSTEM_PROMPT = 'You are a helpful AI assistant.'; + +var PARAM_META = [ + { key: 'temperature', label: 'Temperature', min: 0, max: 2, step: 0.05 }, + { key: 'top_p', label: 'Top-P', min: 0, max: 1, step: 0.05 }, + { key: 'top_k', label: 'Top-K', min: 1, max: 200, step: 1 }, + { key: 'repeat_penalty', label: 'Repeat penalty',min: 0.5, max: 2, step: 0.05 }, + { key: 'seed', label: 'Seed (-1=rand)',min: -1, max: 999999,step: 1 }, + { key: 'num_ctx', label: 'Context (tokens)',min:512,max:131072, step: 512 }, + { key: 'num_predict', label: 'Max tokens (-1=∞)',min:-1,max:8192, step: 1 } +]; + +// ── State ───────────────────────────────────────────────────────────────────── + +var S = { + settings: { url: 'http://localhost:11434', token: '' }, + theme: 'dark', + models: [], // [{ name, paramSize, sizeLabel }] + modelParams: {}, // { [modelName]: { ...params } } + templates: [], // [{ id, name, content }] + sessions: [], // [{ id, name, model, updatedAt }] index only + activeSession: null, // full session object + pageContext: null, + usePageContext: false, + attachments: [], // [{ name, content }] + isStreaming: false, + currentStream: null // { requestId, bubble, thinkBlock, thinkContent, + // mainContent, metrics, sources, msgIndex } +}; + +// ── DOM refs ────────────────────────────────────────────────────────────────── + +var app = document.getElementById('app'); +var statusDot = document.getElementById('status-dot'); +var btnSessions = document.getElementById('btn-sessions'); +var sessionsPanel = document.getElementById('sessions-panel'); +var sessionsList = document.getElementById('sessions-list'); +var btnNewSession = document.getElementById('btn-new-session'); +var btnSettings = document.getElementById('btn-settings'); +var settingsPanel = document.getElementById('settings-panel'); +var inputUrl = document.getElementById('input-url'); +var inputToken = document.getElementById('input-token'); +var btnShowToken = document.getElementById('btn-show-token'); +var btnSave = document.getElementById('btn-save'); +var settingsStatus = document.getElementById('settings-status'); +var btnTheme = document.getElementById('btn-theme'); +var sessionNameEl = document.getElementById('session-name-display'); +var btnRenameSession = document.getElementById('btn-rename-session'); +var btnExportSession = document.getElementById('btn-export-session'); +var btnDeleteSession = document.getElementById('btn-delete-session'); +var chatArea = document.getElementById('chat-area'); +var messagesEl = document.getElementById('messages'); +var systemPanel = document.getElementById('system-prompt-panel'); +var inputSysPrompt = document.getElementById('input-system-prompt'); +var btnToggleSystem = document.getElementById('btn-toggle-system'); +var btnCloseSystem = document.getElementById('btn-close-system'); +var paramsPanel = document.getElementById('params-panel'); +var paramsGrid = document.getElementById('params-grid'); +var btnToggleParams = document.getElementById('btn-toggle-params'); +var btnCloseParams = document.getElementById('btn-close-params'); +var fileChipsArea = document.getElementById('file-chips-area'); +var fileChipsEl = document.getElementById('file-chips'); +var btnPageContext = document.getElementById('btn-page-context'); +var pageCtxLbl = document.getElementById('page-ctx-lbl'); +var btnAttachFile = document.getElementById('btn-attach-file'); +var btnTemplatesOpen = document.getElementById('btn-templates-open'); +var btnClear = document.getElementById('btn-clear'); +var inputMessage = document.getElementById('input-message'); +var templateDropdown = document.getElementById('template-dropdown'); +var tmplDdList = document.getElementById('tmpl-dd-list'); +var btnTmplManageInline = document.getElementById('btn-tmpl-manage-inline'); +var selectModel = document.getElementById('select-model'); +var btnStop = document.getElementById('btn-stop'); +var btnSend = document.getElementById('btn-send'); +var tmplOverlay = document.getElementById('tmpl-overlay'); +var tmplList = document.getElementById('tmpl-list'); +var tmplNewName = document.getElementById('tmpl-new-name'); +var tmplNewContent = document.getElementById('tmpl-new-content'); +var btnTmplAdd = document.getElementById('btn-tmpl-add'); +var btnTmplOverlayClose = document.getElementById('btn-tmpl-overlay-close'); +var fileInput = document.getElementById('file-input'); + +// ── Utilities ───────────────────────────────────────────────────────────────── + +function genId() { + return Date.now().toString(36) + Math.random().toString(36).slice(2, 7); +} + +function escapeHtml(s) { + return String(s) + .replace(/&/g,'&').replace(//g,'>').replace(/"/g,'"'); +} + +function formatBytes(b) { + if (b >= 1e9) return (b / 1e9).toFixed(1) + ' GB'; + if (b >= 1e6) return (b / 1e6).toFixed(0) + ' MB'; + return b + ' B'; +} + +function formatMetrics(m) { + if (!m || !m.eval_count) return ''; + var tps = m.eval_duration ? (m.eval_count / (m.eval_duration / 1e9)).toFixed(1) : '?'; + var secs = m.total_duration ? (m.total_duration / 1e9).toFixed(2) : '?'; + return m.eval_count + ' tokens · ' + tps + ' tok/s · ' + secs + 's'; +} + +function humanizeError(raw) { + if (typeof raw !== 'string') return 'Unknown error'; + if (raw.indexOf('HTTP 401') !== -1) return 'Unauthorized — bearer token required or incorrect'; + if (raw.indexOf('HTTP 403') !== -1) return 'Access denied — check your bearer token'; + if (raw.indexOf('HTTP 405') !== -1) return 'Method not allowed — check server CORS configuration'; + if (raw.indexOf('HTTP 5') !== -1) return 'Server error — ' + raw; + if (/Failed to fetch|NetworkError|Load failed|ECONNREFUSED|network/i.test(raw)) + return 'Cannot reach Ollama — check the URL and that Ollama is running'; + return raw; +} + +function getHeaders() { + var h = { 'Content-Type': 'application/json' }; + var tok = S.settings.token && S.settings.token.trim(); + if (tok) h['Authorization'] = 'Bearer ' + tok; + return h; +} + +function setStatus(state, title) { + statusDot.className = 'status-dot' + (state ? ' ' + state : ''); + statusDot.title = title || state || 'Unknown'; +} + +function setSettingsStatus(msg, type) { + settingsStatus.textContent = msg; + settingsStatus.className = 'status-line' + (type ? ' ' + type : ''); +} + +function setSendEnabled(on) { + btnSend.disabled = !on; + inputMessage.disabled = !on; + if (on) { + btnStop.classList.add('hidden'); + btnSend.classList.remove('hidden'); + } else { + btnStop.classList.remove('hidden'); + btnSend.classList.add('hidden'); + } +} + +// ── Markdown renderer ───────────────────────────────────────────────────────── + +// Import marked.js for proper markdown rendering +var marked; +function loadMarked() { + if (!marked) { + marked = new (window.marked || window.require('marked'))(); + // Use GitHub-style markdown preset + marked.setOptions({ + gfm: true, + breaks: true, + sanitize: false + }); + } +} + +function renderInline(text) { + var h = escapeHtml(text); + h = h.replace(/`([^`\n]+)`/g, '$1'); + h = h.replace(/\*\*([^*\n]+?)\*\*/g, '$1'); + h = h.replace(/\*([^*\n]+?)\*/g, '$1'); + h = h.replace(/\n/g, '
    '); + return h; +} + +function renderMarkdown(raw) { + if (!raw) return ''; + loadMarked(); + try { + var html = marked.parse(raw, { breaks: true }); + // Add syntax highlighting + html = html.replace(/
    ]*)>([\s\S]*?)<\/code><\/pre>/g, function(match, attr, code) {
    +      var lang = attr.match(/[^\s]*=/)?.replace('=', '').trim() || '';
    +      var hl = '';
    +      if (lang && hl) hl = ' data-lang="' + escapeHtml(lang) + '"';
    +      return '' + escapeHtml(code) + '
    '; + }); + return html; + } catch (e) { + // Fallback to simple rendering if marked fails + return renderInline(raw); + } +} + +function renderMarkdown(raw) { + if (!raw) return ''; + var parts = raw.split(/(```[^\n]*\n[\s\S]*?```)/g); + return parts.map(function (part, i) { + if (i % 2 === 1) { + var m = part.match(/```([^\n]*)\n([\s\S]*?)```/); + if (m) { + var la = m[1].trim() ? ' data-lang="' + escapeHtml(m[1].trim()) + '"' : ''; + return '' + escapeHtml(m[2]) + ''; + } + return escapeHtml(part); + } + return renderInline(part); + }).join(''); +} + +// ── RAG helpers ─────────────────────────────────────────────────────────────── + +var STOP_WORDS = new Set( + 'the a an is are was were be been have has had do does did will would could should may might can this that these those i you he she it we they and or but in on at to for of with by from as'.split(' ') +); + +function chunkText(text, size) { + size = size || 250; + var words = text.split(/\s+/); + var chunks = []; + for (var i = 0; i < words.length; i += size) { + chunks.push(words.slice(i, i + size).join(' ')); + } + return chunks; +} + +function scoreChunk(chunk, queryTerms) { + if (!queryTerms.length) return 1; // no terms = include everything (short text) + var lower = chunk.toLowerCase(); + var score = 0; + queryTerms.forEach(function (t) { + var re = new RegExp('\\b' + t.replace(/[.*+?^${}()|[\]\\]/g,'\\$&') + '\\b', 'gi'); + var m = lower.match(re); + if (m) score += m.length; + }); + return score; +} + +function getTopChunks(text, query, maxChunks, maxChars) { + maxChunks = maxChunks || 4; + maxChars = maxChars || 3000; + if (!text || !text.trim()) return []; + var terms = (query || '').toLowerCase().split(/\W+/) + .filter(function (w) { return w.length > 2 && !STOP_WORDS.has(w); }); + var chunks = chunkText(text); + var scored = chunks.map(function (c, idx) { + return { c: c, s: scoreChunk(c, terms), idx: idx }; + }); + // If no query terms, include first N chunks + if (!terms.length) { + scored.sort(function (a, b) { return a.idx - b.idx; }); + } else { + scored = scored.filter(function (x) { return x.s > 0; }); + scored.sort(function (a, b) { return b.s - a.s; }); + } + var out = []; var chars = 0; + for (var i = 0; i < scored.length && out.length < maxChunks; i++) { + if (chars + scored[i].c.length > maxChars) break; + out.push(scored[i].c); + chars += scored[i].c.length; + } + return out; +} + +// Build context block from active page context + attachments, using RAG scoring +function buildContextBlock(userQuery) { + var sources = []; var names = []; + + if (S.usePageContext && S.pageContext && S.pageContext.content) { + var chunks = getTopChunks(S.pageContext.content, userQuery); + if (chunks.length) { + sources.push('=== ' + (S.pageContext.title || S.pageContext.url || 'Page') + ' ===\n' + chunks.join('\n\n')); + names.push(S.pageContext.title || 'Page'); + } + } + + S.attachments.forEach(function (att) { + var chunks = getTopChunks(att.content, userQuery); + if (chunks.length) { + sources.push('=== ' + att.name + ' ===\n' + chunks.join('\n\n')); + names.push(att.name); + } + }); + + if (!sources.length) return null; + return { text: '[Context]\n' + sources.join('\n\n'), names: names }; +} + +// ── Storage ─────────────────────────────────────────────────────────────────── + +function loadAllData(cb) { + chrome.storage.local.get( + ['ollama_s','ollama_t','ollama_si','ollama_sa','ollama_mp','ollama_tp'], + function (r) { + // settings + var saved = r.ollama_s || {}; + S.settings.url = saved.url || 'http://localhost:11434'; + S.settings.token = saved.token || ''; + inputUrl.value = S.settings.url; + inputToken.value = S.settings.token; + + // theme + S.theme = r.ollama_t || 'dark'; + applyTheme(S.theme); + + // default system prompt + loadDefaultSystemPrompt(); + + // sessions index + S.sessions = r.ollama_si || []; + + // model params + S.modelParams = r.ollama_mp || {}; + + // templates + S.templates = r.ollama_tp || []; + + // load active session + var activeId = r.ollama_sa; + if (!activeId && S.sessions.length) activeId = S.sessions[0].id; + + if (activeId) { + chrome.storage.local.get('ollama_ss_' + activeId, function (r2) { + var sess = r2['ollama_ss_' + activeId]; + if (sess) { + S.activeSession = sess; + } else { + S.activeSession = createSessionObj('New Chat'); + saveSession(S.activeSession); + } + if (cb) cb(); + }); + } else { + S.activeSession = createSessionObj('New Chat'); + saveSession(S.activeSession); + if (cb) cb(); + } + } + ); +} + +function saveSettings() { + S.settings.url = inputUrl.value.trim().replace(/\/+$/, '') || 'http://localhost:11434'; + S.settings.token = inputToken.value; + inputUrl.value = S.settings.url; + chrome.storage.local.set({ ollama_s: { url: S.settings.url, token: S.settings.token } }); + // Save default system prompt if changed + if (inputSysPrompt.value !== DEFAULT_SYSTEM_PROMPT) { + chrome.storage.local.set({ ollama_s_default_sys: inputSysPrompt.value }); + } +} + +function loadDefaultSystemPrompt() { + chrome.storage.local.get('ollama_s_default_sys', function(r) { + var saved = r['ollama_s_default_sys']; + if (saved && typeof saved === 'string') { + inputSysPrompt.value = saved; + if (S.activeSession) S.activeSession.systemPrompt = saved; + } + }); +} + +function saveSession(sess) { + sess.updatedAt = Date.now(); + // Upsert in index + var idx = -1; + for (var i = 0; i < S.sessions.length; i++) { + if (S.sessions[i].id === sess.id) { idx = i; break; } + } + var meta = { id: sess.id, name: sess.name, model: sess.model, updatedAt: sess.updatedAt }; + if (idx === -1) S.sessions.unshift(meta); + else S.sessions[idx] = meta; + var store = {}; + store['ollama_ss_' + sess.id] = sess; + store['ollama_si'] = S.sessions; + store['ollama_sa'] = sess.id; + chrome.storage.local.set(store); +} + +var _saveTimer = null; +function debouncedSaveSession() { + clearTimeout(_saveTimer); + _saveTimer = setTimeout(function () { + if (S.activeSession) saveSession(S.activeSession); + }, 600); +} + +function deleteSessionFromStorage(id) { + S.sessions = S.sessions.filter(function (s) { return s.id !== id; }); + chrome.storage.local.remove('ollama_ss_' + id); + chrome.storage.local.set({ ollama_si: S.sessions }); +} + +function saveModelParams() { + chrome.storage.local.set({ ollama_mp: S.modelParams }); +} + +function saveTemplates() { + chrome.storage.local.set({ ollama_tp: S.templates }); +} + +// ── Session management ──────────────────────────────────────────────────────── + +function createSessionObj(name) { + var model = (selectModel && selectModel.value) || ''; + // Auto-generate title from first message if no name provided + var generatedName = ''; + if (!name && S.attachments.length === 0 && S.pageContext) { + var ctx = S.pageContext.title || S.pageContext.url || 'page'; + generatedName = '[From ' + ctx + ']'; + } + return { + id: genId(), + name: name || generatedName || 'New Chat', + model: model, + systemPrompt: '', + messages: [], + createdAt: Date.now(), + updatedAt: Date.now() + }; +} + +function newSession() { + if (S.activeSession) saveSession(S.activeSession); + var sess = createSessionObj('New Chat'); + S.activeSession = sess; + saveSession(sess); + renderSession(); + updateSessionsPanel(); + closePanels(); +} + +function switchSession(id) { + if (S.activeSession && S.activeSession.id === id) { closePanels(); return; } + if (S.activeSession) saveSession(S.activeSession); + chrome.storage.local.get('ollama_ss_' + id, function (r) { + var sess = r['ollama_ss_' + id]; + if (!sess) return; + S.activeSession = sess; + chrome.storage.local.set({ ollama_sa: id }); + renderSession(); + updateSessionsPanel(); + closePanels(); + }); +} + +function deleteSession(id) { + if (!confirm('Delete this conversation?')) return; + deleteSessionFromStorage(id); + if (S.activeSession && S.activeSession.id === id) { + if (S.sessions.length) { + switchSession(S.sessions[0].id); + } else { + S.activeSession = createSessionObj('New Chat'); + saveSession(S.activeSession); + renderSession(); + } + } + updateSessionsPanel(); +} + +function renameSession(id) { + var meta = S.sessions.find(function (s) { return s.id === id; }); + if (!meta) return; + var name = prompt('Rename conversation:', meta.name); + if (!name || !name.trim()) return; + name = name.trim().slice(0, 80); + meta.name = name; + if (S.activeSession && S.activeSession.id === id) { + S.activeSession.name = name; + saveSession(S.activeSession); + sessionNameEl.textContent = name; + } else { + chrome.storage.local.get('ollama_ss_' + id, function (r) { + var sess = r['ollama_ss_' + id]; + if (sess) { sess.name = name; saveSession(sess); } + else { chrome.storage.local.set({ ollama_si: S.sessions }); } + }); + } + updateSessionsPanel(); +} + +function exportSession(sess) { + if (!sess) return; + var lines = ['# ' + sess.name, '', + '> **Model:** ' + (sess.model || 'unknown') + ' ', + '> **Created:** ' + new Date(sess.createdAt).toLocaleString(), + '']; + if (sess.systemPrompt) { + lines.push('## System Prompt', '', sess.systemPrompt, ''); + } + lines.push('---', ''); + (sess.messages || []).forEach(function (m) { + lines.push('**' + (m.role === 'user' ? 'You' : 'Assistant') + ':**', '', m.content, '', '---', ''); + }); + var blob = new Blob([lines.join('\n')], { type: 'text/markdown' }); + var url = URL.createObjectURL(blob); + var a = document.createElement('a'); + a.href = url; + a.download = (sess.name || 'chat').replace(/[^a-z0-9\-_ ]/gi, '-').slice(0,60) + '.md'; + document.body.appendChild(a); + a.click(); + document.body.removeChild(a); + URL.revokeObjectURL(url); +} + +// ── Model management ────────────────────────────────────────────────────────── + +function fetchModels() { + setStatus('', 'Connecting…'); + return new Promise(function (resolve) { + chrome.runtime.sendMessage( + { action: 'OLLAMA_GET', url: S.settings.url + '/api/tags', headers: getHeaders() }, + function (response) { + if (chrome.runtime.lastError || !response) { + var e = chrome.runtime.lastError ? chrome.runtime.lastError.message : 'No response'; + setStatus('error', humanizeError(e)); + setSettingsStatus(humanizeError(e), 'err'); + resolve(); return; + } + if (!response.ok) { + var msg = humanizeError(response.error || ('HTTP ' + response.status)); + setStatus('error', msg); + setSettingsStatus(msg, 'err'); + resolve(); return; + } + var raw = response.data.models || []; + S.models = raw.map(function (m) { + return { + name: m.name, + paramSize: (m.details && m.details.parameter_size) || '', + sizeLabel: m.size ? formatBytes(m.size) : '' + }; + }); + populateModels(); + setStatus('connected', 'Connected'); + setSettingsStatus('Connected — ' + S.models.length + ' model(s)', 'ok'); + resolve(); + } + ); + }); +} + +function populateModels() { + var current = (S.activeSession && S.activeSession.model) || ''; + selectModel.innerHTML = ''; + S.models.forEach(function (m) { + var opt = document.createElement('option'); + opt.value = m.name; + var lbl = m.name; + if (m.paramSize) lbl += ' (' + m.paramSize + ')'; + if (m.sizeLabel) lbl += ' · ' + m.sizeLabel; + opt.textContent = lbl; + if (m.name === current) opt.selected = true; + selectModel.appendChild(opt); + }); + if (selectModel.value && S.activeSession) S.activeSession.model = selectModel.value; +} + +function getModelOptions() { + var model = (S.activeSession && S.activeSession.model) || selectModel.value || ''; + return Object.assign({}, DEF_PARAMS, S.modelParams[model] || {}); +} + +// ── Page context ────────────────────────────────────────────────────────────── + +function getPageContent() { + return new Promise(function (resolve, reject) { + chrome.tabs.query({ active: true, currentWindow: true }, function (tabs) { + if (!tabs || !tabs[0]) { reject(new Error('No active tab')); return; } + chrome.runtime.sendMessage( + { action: 'GET_PAGE_CONTENT', tabId: tabs[0].id }, + function (response) { + if (chrome.runtime.lastError) { reject(new Error(chrome.runtime.lastError.message)); return; } + if (!response) { reject(new Error('No response from background')); return; } + if (response.error) { reject(new Error(response.error)); return; } + resolve(response); + } + ); + }); + }); +} + +async function togglePageContext() { + if (S.usePageContext) { + S.usePageContext = false; S.pageContext = null; + btnPageContext.classList.remove('active'); + pageCtxLbl.textContent = 'Page'; + return; + } + btnPageContext.disabled = true; + pageCtxLbl.textContent = 'Loading…'; + try { + var ctx = await getPageContent(); + S.pageContext = ctx; S.usePageContext = true; + btnPageContext.classList.add('active'); + var title = (ctx.title || ctx.url || 'page').trim(); + pageCtxLbl.textContent = title.length > 20 ? title.slice(0, 18) + '…' : title; + } catch (e) { + pageCtxLbl.textContent = 'Page'; + appendSystemMsg('Could not read page: ' + e.message); + } finally { + btnPageContext.disabled = false; + } +} + +// ── File attachments ────────────────────────────────────────────────────────── + +function extractPdfText(buffer) { + var bytes = new Uint8Array(buffer); + var runs = []; var cur = ''; + for (var i = 0; i < bytes.length; i++) { + var b = bytes[i]; + if (b >= 32 && b <= 126) { cur += String.fromCharCode(b); } + else if (b === 10 || b === 13) { if (cur.length > 3) runs.push(cur); cur = ''; } + else { if (cur.length > 3) runs.push(cur); cur = ''; } + } + if (cur.length > 3) runs.push(cur); + return runs.filter(function (s) { return !/^[\/\-\(\)\.]{2,}$/.test(s.trim()); }) + .join(' ').replace(/\s+/g, ' ').trim().slice(0, 12000); +} + +function handleFiles(files) { + Array.prototype.forEach.call(files, function (file) { + var name = file.name; + var ext = name.split('.').pop().toLowerCase(); + if (ext === 'pdf') { + var reader = new FileReader(); + reader.onload = function (e) { + var text = extractPdfText(e.target.result); + S.attachments.push({ name: name, content: text }); + renderFileChips(); + }; + reader.readAsArrayBuffer(file); + } else { + var reader2 = new FileReader(); + reader2.onload = function (e) { + S.attachments.push({ name: name, content: (e.target.result || '').slice(0, 12000) }); + renderFileChips(); + }; + reader2.readAsText(file); + } + }); + fileInput.value = ''; +} + +function removeAttachment(idx) { + S.attachments.splice(idx, 1); + renderFileChips(); +} + +function renderFileChips() { + if (!S.attachments.length) { + fileChipsArea.classList.add('hidden'); return; + } + fileChipsArea.classList.remove('hidden'); + fileChipsEl.innerHTML = ''; + S.attachments.forEach(function (att, i) { + var chip = document.createElement('div'); + chip.className = 'file-chip'; + chip.innerHTML = + '' + escapeHtml(att.name) + '' + + ''; + fileChipsEl.appendChild(chip); + }); +} + +// ── Template management ─────────────────────────────────────────────────────── + +function renderTmplOverlay() { + tmplList.innerHTML = ''; + if (!S.templates.length) { + tmplList.innerHTML = '
    No templates yet.
    '; + return; + } + S.templates.forEach(function (t) { + var el = document.createElement('div'); + el.className = 'tmpl-item'; + el.innerHTML = + '
    ' + + '
    ' + escapeHtml(t.name) + '
    ' + + '
    ' + escapeHtml((t.content || '').slice(0, 80)) + '
    ' + + '
    ' + + ''; + tmplList.appendChild(el); + }); +} + +function renderTmplDropdown(filter) { + var filtered = S.templates.filter(function (t) { + if (!filter) return true; + return t.name.toLowerCase().includes(filter) || t.content.toLowerCase().includes(filter); + }); + if (!filtered.length) { templateDropdown.classList.add('hidden'); return; } + tmplDdList.innerHTML = ''; + filtered.slice(0, 8).forEach(function (t) { + var el = document.createElement('div'); + el.className = 'tmpl-dd-item'; + el.innerHTML = + '
    ' + escapeHtml(t.name) + '
    ' + + '
    ' + escapeHtml((t.content || '').slice(0, 60)) + '
    '; + el.addEventListener('click', function () { + inputMessage.value = t.content; + inputMessage.style.height = 'auto'; + inputMessage.style.height = Math.min(inputMessage.scrollHeight, 160) + 'px'; + templateDropdown.classList.add('hidden'); + inputMessage.focus(); + }); + tmplDdList.appendChild(el); + }); + templateDropdown.classList.remove('hidden'); +} + +// ── UI helpers ──────────────────────────────────────────────────────────────── + +function applyTheme(t) { + S.theme = t; + app.setAttribute('data-theme', t); + chrome.storage.local.set({ ollama_t: t }); +} + +function closePanels() { + settingsPanel.classList.add('hidden'); + btnSettings.classList.remove('active'); + sessionsPanel.classList.add('hidden'); + btnSessions.classList.remove('active'); +} + +function updateSessionsPanel() { + if (!S.sessions.length) { + sessionsList.innerHTML = '
    No saved sessions.
    '; + return; + } + sessionsList.innerHTML = ''; + S.sessions.forEach(function (s) { + var el = document.createElement('div'); + el.className = 'session-item' + (S.activeSession && S.activeSession.id === s.id ? ' active' : ''); + var ts = s.updatedAt ? new Date(s.updatedAt).toLocaleDateString() : ''; + el.innerHTML = + '
    ' + escapeHtml(s.name) + '
    ' + + '
    ' + escapeHtml(ts) + '
    ' + + '
    ' + + '' + + '' + + '
    '; + el.addEventListener('click', function (e) { + if (e.target.closest('[data-action]')) return; + switchSession(s.id); + }); + sessionsList.appendChild(el); + }); +} + +function updateSessionBar() { + if (!S.activeSession) return; + sessionNameEl.textContent = S.activeSession.name; +} + +function updateModelDisplay() { + var model = S.activeSession && S.activeSession.model || selectModel.value || ''; + var display = model || '(no model)'; + document.getElementById('model-display').textContent = display; +} + +function renderParamsPanel() { + var opts = getModelOptions(); + paramsGrid.innerHTML = ''; + PARAM_META.forEach(function (pm) { + var val = opts[pm.key] !== undefined ? opts[pm.key] : DEF_PARAMS[pm.key]; + var lbl = document.createElement('label'); + lbl.className = 'param-lbl'; + lbl.htmlFor = 'param_' + pm.key; + lbl.textContent = pm.label; + + var inp = document.createElement('input'); + inp.id = 'param_' + pm.key; + inp.type = 'number'; + inp.className = 'param-input'; + inp.value = val; + inp.min = pm.min; + inp.max = pm.max; + inp.step = pm.step; + inp.addEventListener('change', function () { + var model = (S.activeSession && S.activeSession.model) || selectModel.value || ''; + if (!S.modelParams[model]) S.modelParams[model] = {}; + S.modelParams[model][pm.key] = parseFloat(this.value); + saveModelParams(); + }); + + var valEl = document.createElement('span'); + valEl.className = 'param-val'; + valEl.textContent = val; + inp.addEventListener('input', function () { valEl.textContent = this.value; }); + + paramsGrid.appendChild(lbl); + paramsGrid.appendChild(inp); + paramsGrid.appendChild(valEl); + }); +} + +function renderSession() { + if (!S.activeSession) return; + updateSessionBar(); + // Sync system prompt field + inputSysPrompt.value = S.activeSession.systemPrompt || ''; + // Sync model selector + if (S.activeSession.model && selectModel) { + selectModel.value = S.activeSession.model; + if (!selectModel.value && S.activeSession.model) { + // Model not in list yet, will be set after fetchModels + } + } + renderAllMessages(); +} + +// ── Message rendering ───────────────────────────────────────────────────────── + +function scrollToBottom() { chatArea.scrollTop = chatArea.scrollHeight; } + +function showEmptyState() { + if (messagesEl.children.length) return; + var el = document.createElement('div'); + el.id = 'empty-state'; el.className = 'empty-state'; + el.innerHTML = + '
    🦙
    ' + + '

    Ask anything.
    Use Page or Attach to include context.

    '; + messagesEl.appendChild(el); +} + +function removeEmptyState() { + var el = document.getElementById('empty-state'); + if (el) el.remove(); +} + +function appendSystemMsg(text) { + var w = document.createElement('div'); + w.className = 'message error'; + var b = document.createElement('div'); + b.className = 'msg-bubble'; b.textContent = humanizeError(text); + var body = document.createElement('div'); + body.className = 'msg-body'; body.appendChild(b); + w.appendChild(body); messagesEl.appendChild(w); scrollToBottom(); +} + +function buildMessageEl(msg, index) { + var w = document.createElement('div'); + w.className = 'message ' + msg.role; + w.dataset.index = index; + + var roleEl = document.createElement('div'); + roleEl.className = 'msg-role'; + roleEl.textContent = msg.role === 'user' ? 'You' : 'Assistant'; + + var body = document.createElement('div'); + body.className = 'msg-body'; + + if (msg.role === 'assistant') { + // Thinking block + if (msg.thinking) { + var det = document.createElement('details'); + det.className = 'think-block'; + var sum = document.createElement('summary'); + sum.textContent = 'Thinking'; + var tc = document.createElement('div'); + tc.className = 'think-content'; + tc.innerHTML = renderMarkdown(msg.thinking); + det.appendChild(sum); det.appendChild(tc); + body.appendChild(det); + } + // Bubble + var bubble = document.createElement('div'); + bubble.className = 'msg-bubble'; + bubble.innerHTML = renderMarkdown(msg.content || ''); + body.appendChild(bubble); + // Meta (source + metrics) + var meta = document.createElement('div'); + meta.className = 'msg-meta'; + if (msg.sources && msg.sources.length) { + var sc = document.createElement('div'); + sc.className = 'source-chip'; + sc.textContent = '📎 ' + msg.sources.join(', '); + meta.appendChild(sc); + } + if (msg.metrics) { + var ml = document.createElement('div'); + ml.className = 'metrics-line'; + ml.textContent = formatMetrics(msg.metrics); + if (ml.textContent) meta.appendChild(ml); + } + if (meta.children.length) body.appendChild(meta); + // Actions + var acts = document.createElement('div'); + acts.className = 'msg-actions'; + acts.innerHTML = + ''; + body.appendChild(acts); + } else { + // User bubble + var ub = document.createElement('div'); + ub.className = 'msg-bubble'; + ub.textContent = msg.content; + body.appendChild(ub); + var ua = document.createElement('div'); + ua.className = 'msg-actions'; + ua.innerHTML = ''; + body.appendChild(ua); + } + + w.appendChild(roleEl); w.appendChild(body); + return w; +} + +function renderAllMessages() { + messagesEl.innerHTML = ''; + if (!S.activeSession || !S.activeSession.messages.length) { showEmptyState(); return; } + S.activeSession.messages.forEach(function (msg, i) { + messagesEl.appendChild(buildMessageEl(msg, i)); + }); + // Add regen button to last assistant message + updateRegenButton(); + scrollToBottom(); +} + +function updateRegenButton() { + // Remove any existing regen buttons + messagesEl.querySelectorAll('[data-action="regen"]').forEach(function (b) { b.remove(); }); + // Find last assistant message + var msgs = S.activeSession && S.activeSession.messages; + if (!msgs || !msgs.length) return; + for (var i = msgs.length - 1; i >= 0; i--) { + if (msgs[i].role === 'assistant') { + var el = messagesEl.querySelector('.message[data-index="' + i + '"] .msg-actions'); + if (el) { + var rb = document.createElement('button'); + rb.className = 'act-btn'; rb.dataset.action = 'regen'; rb.title = 'Regenerate'; + rb.innerHTML = '↻ Regen'; + el.appendChild(rb); + } + break; + } + } +} + +// ── Chat ────────────────────────────────────────────────────────────────────── + +function buildApiMessages() { + var out = []; + if (S.activeSession && S.activeSession.systemPrompt) { + out.push({ role: 'system', content: S.activeSession.systemPrompt }); + } + (S.activeSession && S.activeSession.messages || []).forEach(function (m) { + out.push({ role: m.role, content: m.content }); + }); + return out; +} + +function doSendChat(text) { + if (S.isStreaming) return; + if (!text || !text.trim()) return; + if (!S.activeSession) return; + + var model = selectModel.value; + if (!model) { appendSystemMsg('Please select a model first.'); return; } + S.activeSession.model = model; + + // Collect context + var ctx = buildContextBlock(text); + + // Build actual user content (with context prefix) + var apiContent = text; + if (ctx) apiContent = ctx.text + '\n\n' + text; + + removeEmptyState(); + inputMessage.value = ''; + inputMessage.style.height = ''; + setSendEnabled(false); + S.isStreaming = true; + + // User message (display only the typed text) + var userMsg = { id: genId(), role: 'user', content: text, ts: Date.now() }; + S.activeSession.messages.push(userMsg); + var userIdx = S.activeSession.messages.length - 1; + messagesEl.appendChild(buildMessageEl(userMsg, userIdx)); + scrollToBottom(); + + // Generate title from first message if this is the first message + if (S.activeSession.messages.length === 2 && !S.activeSession.name) { + // First message just sent - generate a short title + var shortTitle = text.slice(0, 40).trim().replace(/[^a-zA-Z0-9 ]/g, ''); + S.activeSession.name = shortTitle || 'New Chat'; + saveSession(S.activeSession); + } + + // Build API history with context-injected version of the last message + var apiMsgs = buildApiMessages(); + apiMsgs[apiMsgs.length - 1] = { role: 'user', content: apiContent }; + + // Assistant bubble + var asstIdx = S.activeSession.messages.length; // will be pushed on DONE + var asstMsgEl = document.createElement('div'); + asstMsgEl.className = 'message assistant'; + asstMsgEl.dataset.index = asstIdx; + + var roleEl = document.createElement('div'); + roleEl.className = 'msg-role'; roleEl.textContent = 'Assistant'; + + var body = document.createElement('div'); + body.className = 'msg-body'; + + // Thinking block (hidden until content arrives) + var thinkBlock = document.createElement('details'); + thinkBlock.className = 'think-block hidden'; + var thinkSum = document.createElement('summary'); + thinkSum.textContent = 'Thinking'; + var thinkContent = document.createElement('div'); + thinkContent.className = 'think-content'; + thinkBlock.appendChild(thinkSum); thinkBlock.appendChild(thinkContent); + body.appendChild(thinkBlock); + + var bubble = document.createElement('div'); + bubble.className = 'msg-bubble streaming'; + body.appendChild(bubble); + + asstMsgEl.appendChild(roleEl); asstMsgEl.appendChild(body); + messagesEl.appendChild(asstMsgEl); + scrollToBottom(); + + var requestId = genId(); + S.currentStream = { + requestId: requestId, + bubble: bubble, + thinkBlock: thinkBlock, + thinkContent: thinkContent, + thinkText: '', + mainText: '', + inThinkTag: false, + metrics: null, + sources: ctx ? ctx.names : null, + msgIndex: asstIdx, + msgEl: asstMsgEl, + bodyEl: body + }; + + chrome.runtime.sendMessage({ + action: 'OLLAMA_FETCH', + url: S.settings.url + '/api/chat', + headers: getHeaders(), + body: JSON.stringify({ + model: model, + messages: apiMsgs, + stream: true, + options: getModelOptions() + }), + requestId: requestId + }); +} + +function sendChat() { + var text = inputMessage.value.trim(); + if (text) doSendChat(text); +} + +function stopStream() { + if (!S.currentStream) return; + chrome.runtime.sendMessage({ action: 'CANCEL_STREAM', requestId: S.currentStream.requestId }); +} + +function regenerateLastResponse() { + if (S.isStreaming || !S.activeSession) return; + var msgs = S.activeSession.messages; + // Remove trailing assistant messages + while (msgs.length && msgs[msgs.length - 1].role === 'assistant') msgs.pop(); + if (!msgs.length || msgs[msgs.length - 1].role !== 'user') return; + var lastUserText = msgs[msgs.length - 1].content; + msgs.pop(); // will be re-added by doSendChat + saveSession(S.activeSession); + renderAllMessages(); + doSendChat(lastUserText); +} + +function startEditMessage(msgEl, index) { + if (S.isStreaming) return; + var msgs = S.activeSession && S.activeSession.messages; + if (!msgs || index >= msgs.length) return; + var original = msgs[index].content; + + var bubble = msgEl.querySelector('.msg-bubble'); + bubble.innerHTML = ''; + var ta = document.createElement('textarea'); + ta.className = 'edit-textarea'; ta.value = original; + bubble.appendChild(ta); + + var acts = msgEl.querySelector('.msg-actions'); + acts.innerHTML = ''; + var saveBtn = document.createElement('button'); + saveBtn.className = 'primary-btn sm'; saveBtn.textContent = 'Save & Send'; + var cancelBtn = document.createElement('button'); + cancelBtn.className = 'secondary-btn'; cancelBtn.textContent = 'Cancel'; + + saveBtn.addEventListener('click', function () { + var newText = ta.value.trim(); + if (!newText) return; + // Truncate history to before this message + S.activeSession.messages = msgs.slice(0, index); + saveSession(S.activeSession); + renderAllMessages(); + doSendChat(newText); + }); + cancelBtn.addEventListener('click', function () { + bubble.innerHTML = escapeHtml(original).replace(/\n/g,'
    '); + acts.innerHTML = ''; + }); + + var editActs = document.createElement('div'); + editActs.className = 'edit-actions'; + editActs.appendChild(saveBtn); editActs.appendChild(cancelBtn); + msgEl.querySelector('.msg-body').appendChild(editActs); + ta.focus(); ta.select(); +} + +// ── Stream message listener ─────────────────────────────────────────────────── + +chrome.runtime.onMessage.addListener(function (msg) { + if (!S.currentStream || msg.requestId !== S.currentStream.requestId) return; + var cs = S.currentStream; + + if (msg.action === 'STREAM_THINKING') { + cs.thinkText += msg.text; + cs.thinkContent.innerHTML = renderMarkdown(cs.thinkText); + cs.thinkBlock.classList.remove('hidden'); + cs.thinkBlock.open = true; + scrollToBottom(); + return; + } + + if (msg.action === 'STREAM_CHUNK') { + var delta = msg.text; + + // Parse inline ... tags from content + if (!cs.inThinkTag && !cs.mainText && delta.trimStart().startsWith('')) { + cs.inThinkTag = true; + delta = delta.replace(/^[\s\S]*?/, ''); + } + if (cs.inThinkTag) { + var closeIdx = delta.indexOf(''); + if (closeIdx !== -1) { + cs.thinkText += delta.slice(0, closeIdx); + cs.inThinkTag = false; + delta = delta.slice(closeIdx + 8).replace(/^\s+/, ''); + } else { + cs.thinkText += delta; + delta = ''; + } + } + if (cs.thinkText) { + cs.thinkContent.innerHTML = renderMarkdown(cs.thinkText); + cs.thinkBlock.classList.remove('hidden'); + if (cs.inThinkTag) cs.thinkBlock.open = true; + } + if (delta) { + cs.mainText += delta; + cs.bubble.innerHTML = renderMarkdown(cs.mainText); + } + scrollToBottom(); + return; + } + + if (msg.action === 'STREAM_METRICS') { + cs.metrics = msg.metrics; + return; + } + + if (msg.action === 'STREAM_DONE' || msg.action === 'STREAM_ERROR') { + var isErr = msg.action === 'STREAM_ERROR'; + var content = cs.mainText; + + cs.bubble.classList.remove('streaming'); + if (cs.thinkBlock && cs.thinkText) cs.thinkBlock.open = false; // collapse when done + + if (isErr && !content) { + // No partial content — show error in bubble + if (S.activeSession && S.activeSession.messages.length) { + S.activeSession.messages.pop(); // roll back user message + } + cs.bubble.parentElement.parentElement.className = 'message error'; + cs.bubble.textContent = humanizeError(msg.error || 'Unknown error'); + } else { + if (isErr) { + // Partial content + error note + var note = document.createElement('div'); + note.className = 'stream-err-note'; + note.textContent = humanizeError(msg.error || 'Stream interrupted'); + cs.bubble.parentElement.appendChild(note); + } + + // Add source chip if context was used + if (cs.sources && cs.sources.length) { + var sc = document.createElement('div'); + sc.className = 'source-chip'; + sc.textContent = '📎 ' + cs.sources.join(', '); + var metaEl = document.createElement('div'); + metaEl.className = 'msg-meta'; + metaEl.appendChild(sc); + if (cs.metrics) { + var ml = document.createElement('div'); + ml.className = 'metrics-line'; + ml.textContent = formatMetrics(cs.metrics); + if (ml.textContent) metaEl.appendChild(ml); + } + cs.bodyEl.insertBefore(metaEl, cs.bubble.nextSibling); + } else if (cs.metrics) { + var mEl = document.createElement('div'); + mEl.className = 'msg-meta'; + var ml2 = document.createElement('div'); + ml2.className = 'metrics-line'; + ml2.textContent = formatMetrics(cs.metrics); + if (ml2.textContent) { mEl.appendChild(ml2); cs.bodyEl.insertBefore(mEl, cs.bubble.nextSibling); } + } + + // Add actions row + var actsEl = document.createElement('div'); + actsEl.className = 'msg-actions'; + actsEl.innerHTML = ''; + cs.bodyEl.appendChild(actsEl); + + // Push to session + if (S.activeSession) { + var asstMsg = { + id: genId(), + role: 'assistant', + content: content || '', + thinking: cs.thinkText || null, + metrics: cs.metrics || null, + sources: cs.sources || null, + ts: Date.now() + }; + S.activeSession.messages.push(asstMsg); + cs.msgEl.dataset.index = S.activeSession.messages.length - 1; + saveSession(S.activeSession); + updateRegenButton(); + } + } + + S.currentStream = null; + S.isStreaming = false; + setSendEnabled(true); + scrollToBottom(); + inputMessage.focus(); + } +}); + +// ── Event handlers ──────────────────────────────────────────────────────────── + +function initEventHandlers() { + + // Header + btnSettings.addEventListener('click', function () { + var wasOpen = !settingsPanel.classList.contains('hidden'); + closePanels(); + if (!wasOpen) { settingsPanel.classList.remove('hidden'); btnSettings.classList.add('active'); } + }); + + btnSessions.addEventListener('click', function () { + var wasOpen = !sessionsPanel.classList.contains('hidden'); + closePanels(); + if (!wasOpen) { + updateSessionsPanel(); + sessionsPanel.classList.remove('hidden'); + btnSessions.classList.add('active'); + } + }); + + btnTheme.addEventListener('click', function () { + applyTheme(S.theme === 'dark' ? 'light' : 'dark'); + }); + + // Settings + btnShowToken.addEventListener('click', function () { + inputToken.type = inputToken.type === 'password' ? 'text' : 'password'; + }); + + var urlTimer = null; + inputUrl.addEventListener('input', function () { + clearTimeout(urlTimer); + var val = this.value.trim().replace(/\/+$/, ''); + if (!val || !/^https?:\/\//.test(val)) return; + urlTimer = setTimeout(function () { + setStatus('', 'Checking…'); + var lh = { 'Content-Type': 'application/json' }; + var tok = inputToken.value.trim(); + if (tok) lh['Authorization'] = 'Bearer ' + tok; + chrome.runtime.sendMessage( + { action: 'OLLAMA_GET', url: val + '/api/tags', headers: lh }, + function (r) { + if (!r || !r.ok) { setStatus('error', 'Cannot connect'); return; } + setStatus('connected', 'Connected'); + var models = (r.data.models || []).map(function (m) { + return { name: m.name, paramSize: (m.details && m.details.parameter_size) || '', sizeLabel: m.size ? formatBytes(m.size) : '' }; + }); + S.models = models; + populateModels(); + } + ); + }, 800); + }); + + btnSave.addEventListener('click', async function () { + saveSettings(); + setSettingsStatus('Connecting…', ''); + await fetchModels(); + }); + + // Sessions panel + btnNewSession.addEventListener('click', newSession); + + sessionsList.addEventListener('click', function (e) { + var btn = e.target.closest('[data-action]'); + if (!btn) return; + var id = btn.dataset.id; + if (btn.dataset.action === 'rename') renameSession(id); + if (btn.dataset.action === 'delete') deleteSession(id); + }); + + // Session bar + btnRenameSession.addEventListener('click', function () { + if (S.activeSession) renameSession(S.activeSession.id); + }); + btnExportSession.addEventListener('click', function () { exportSession(S.activeSession); }); + btnDeleteSession.addEventListener('click', function () { + if (S.activeSession) deleteSession(S.activeSession.id); + }); + sessionNameEl.addEventListener('dblclick', function () { + if (S.activeSession) renameSession(S.activeSession.id); + }); + + // Model selector (in settings panel) + selectModel.addEventListener('change', function () { + if (!S.activeSession) return; + S.activeSession.model = this.value; + updateModelDisplay(); + debouncedSaveSession(); + renderParamsPanel(); + }); + + // System prompt (now in settings panel, but also accessible from input area as reminder) + btnToggleSystem.addEventListener('click', function () { + // Toggle visibility in input area (for quick editing) + var open = systemPanel.classList.toggle('hidden') === false; + btnToggleSystem.classList.toggle('active', !systemPanel.classList.contains('hidden')); + if (open) inputSysPrompt.focus(); + }); + btnCloseSystem.addEventListener('click', function () { + systemPanel.classList.add('hidden'); + btnToggleSystem.classList.remove('active'); + }); + inputSysPrompt.addEventListener('input', function () { + if (!S.activeSession) return; + S.activeSession.systemPrompt = this.value; + debouncedSaveSession(); + }); + + // Settings panel system prompt + var settingsSysPanel = document.getElementById('system-prompt-panel'); + if (settingsSysPanel) { + var settingsBtnCloseSystem = settingsSysPanel.querySelector('#btn-close-system'); + if (settingsBtnCloseSystem) { + settingsBtnCloseSystem.addEventListener('click', function () { + settingsSysPanel.classList.add('collapsed'); + }); + } + var settingsInputSysPrompt = settingsSysPanel.querySelector('#input-system-prompt'); + if (settingsInputSysPrompt) { + settingsInputSysPrompt.addEventListener('input', function () { + if (!S.activeSession) return; + S.activeSession.systemPrompt = this.value; + debouncedSaveSession(); + }); + } + } + + // Params panel + btnToggleParams.addEventListener('click', function () { + var open = !paramsPanel.classList.contains('hidden'); + paramsPanel.classList.toggle('hidden', open); + btnToggleParams.classList.toggle('active', !open); + if (!open) renderParamsPanel(); + }); + btnCloseParams.addEventListener('click', function () { + paramsPanel.classList.add('hidden'); + btnToggleParams.classList.remove('active'); + }); + + // File chips (delegation) + fileChipsEl.addEventListener('click', function (e) { + var btn = e.target.closest('.file-chip-del'); + if (!btn) return; + var idx = parseInt(btn.dataset.idx); + removeAttachment(idx); + }); + + // File attach + btnAttachFile.addEventListener('click', function () { fileInput.click(); }); + fileInput.addEventListener('change', function () { if (this.files.length) handleFiles(this.files); }); + + // Page context + btnPageContext.addEventListener('click', togglePageContext); + + // Clear + btnClear.addEventListener('click', function () { + if (S.isStreaming) return; + if (!confirm('Clear this conversation?')) return; + if (S.activeSession) { S.activeSession.messages = []; saveSession(S.activeSession); } + renderAllMessages(); + }); + + // Textarea input & template dropdown + inputMessage.addEventListener('input', function () { + this.style.height = 'auto'; + this.style.height = Math.min(this.scrollHeight, 160) + 'px'; + var val = this.value; + if (val.startsWith('/')) { + renderTmplDropdown(val.slice(1).toLowerCase()); + } else { + templateDropdown.classList.add('hidden'); + } + }); + + inputMessage.addEventListener('keydown', function (e) { + if (e.key === 'Enter' && !e.shiftKey) { + e.preventDefault(); + sendChat(); + } + if (e.key === 'Escape') templateDropdown.classList.add('hidden'); + }); + + // Templates button (open manager) + btnTemplatesOpen.addEventListener('click', function () { + renderTmplOverlay(); + tmplOverlay.classList.remove('hidden'); + }); + btnTmplManageInline.addEventListener('click', function () { + templateDropdown.classList.add('hidden'); + renderTmplOverlay(); + tmplOverlay.classList.remove('hidden'); + }); + + // Template overlay + btnTmplOverlayClose.addEventListener('click', function () { + tmplOverlay.classList.add('hidden'); + }); + tmplOverlay.addEventListener('click', function (e) { + if (e.target === tmplOverlay) tmplOverlay.classList.add('hidden'); + }); + btnTmplAdd.addEventListener('click', function () { + var name = tmplNewName.value.trim(); + var content = tmplNewContent.value.trim(); + if (!name || !content) return; + S.templates.push({ id: genId(), name: name, content: content }); + saveTemplates(); + tmplNewName.value = ''; tmplNewContent.value = ''; + renderTmplOverlay(); + }); + tmplList.addEventListener('click', function (e) { + var btn = e.target.closest('.tmpl-item-del'); + if (!btn) return; + var id = btn.dataset.id; + S.templates = S.templates.filter(function (t) { return t.id !== id; }); + saveTemplates(); + renderTmplOverlay(); + }); + + // Send / Stop + btnSend.addEventListener('click', sendChat); + btnStop.addEventListener('click', stopStream); + + // Message actions (delegation) + messagesEl.addEventListener('click', function (e) { + var btn = e.target.closest('.act-btn[data-action]'); + if (!btn) return; + var action = btn.dataset.action; + var msgEl = btn.closest('.message'); + var index = msgEl ? parseInt(msgEl.dataset.index) : -1; + + if (action === 'copy') { + var bubble = msgEl && msgEl.querySelector('.msg-bubble'); + var text = bubble ? (bubble.innerText || bubble.textContent || '') : ''; + navigator.clipboard.writeText(text).catch(function () { + // fallback: select text + var r = document.createRange(); r.selectNode(bubble); + window.getSelection().removeAllRanges(); + window.getSelection().addRange(r); + document.execCommand('copy'); + window.getSelection().removeAllRanges(); + }); + btn.textContent = '✓ Copied'; + setTimeout(function () { btn.innerHTML = '⧉ Copy'; }, 1500); + } + if (action === 'edit') { startEditMessage(msgEl, index); } + if (action === 'regen') { regenerateLastResponse(); } + }); +} + +// ── Init ────────────────────────────────────────────────────────────────────── + +loadAllData(async function () { + renderSession(); + initEventHandlers(); + renderParamsPanel(); + await fetchModels(); + // After models load, sync session model to selector + if (S.activeSession && S.activeSession.model) { + selectModel.value = S.activeSession.model; + if (!selectModel.value) { + // model stored in session not in list; pick first + if (selectModel.options.length > 1) { + selectModel.selectedIndex = 1; + S.activeSession.model = selectModel.value; + } + } + } +}); + +})();