ibc-ai-web-sdk 2.0.5 → 2.1.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/index.cjs.js +277 -618
- package/dist/index.esm.js +277 -618
- package/dist/index.umd.js +5268 -5609
- package/dist/index.umd.min.js +1 -1
- package/dist-ts/AIChatDialog.d.ts +184 -0
- package/dist-ts/config/defaultConfig.d.ts +43 -0
- package/dist-ts/config/themes.d.ts +1568 -0
- package/dist-ts/core/AIChatClient.d.ts +56 -0
- package/dist-ts/index.d.ts +45 -0
- package/dist-ts/types.d.ts +456 -0
- package/dist-ts/utils/errorCodes.d.ts +15 -0
- package/dist-ts/utils/logger.d.ts +16 -0
- package/dist-ts/utils/messageIdGenerator.d.ts +12 -0
- package/package.json +17 -9
- package/dist/index.cjs.js.map +0 -1
- package/dist/index.esm.js.map +0 -1
- package/dist/index.umd.js.map +0 -1
- package/index.d.ts +0 -130
- package/src/types.ts +0 -599
package/dist/index.cjs.js
CHANGED
|
@@ -6,7 +6,6 @@ Object.defineProperty(exports, '__esModule', { value: true });
|
|
|
6
6
|
* 错误码常量
|
|
7
7
|
* 统一定义所有错误码,方便维护
|
|
8
8
|
*/
|
|
9
|
-
|
|
10
9
|
const ERROR_CODES = {
|
|
11
10
|
// 客户端错误 (40000-49999)
|
|
12
11
|
AGENT_NOT_FOUND: 40001,
|
|
@@ -15,7 +14,6 @@ const ERROR_CODES = {
|
|
|
15
14
|
// 提示词不能为空
|
|
16
15
|
AUTH_FAILED: 40301,
|
|
17
16
|
// 鉴权失败
|
|
18
|
-
|
|
19
17
|
// 服务器错误 (50000-59999)
|
|
20
18
|
NETWORK_ERROR: 500,
|
|
21
19
|
// 网络请求失败
|
|
@@ -23,7 +21,6 @@ const ERROR_CODES = {
|
|
|
23
21
|
// 服务器内部错误
|
|
24
22
|
TIMEOUT_ERROR: 50002,
|
|
25
23
|
// 请求超时
|
|
26
|
-
|
|
27
24
|
// 取消
|
|
28
25
|
REQUEST_CANCELLED: 0 // 请求已取消
|
|
29
26
|
};
|
|
@@ -624,14 +621,14 @@ class AIChatClient {
|
|
|
624
621
|
const message = parsed && (parsed.message || parsed.errorMessage) || data || '流式请求失败';
|
|
625
622
|
const code = parsed && (parsed.code || parsed.errCode || parsed.status);
|
|
626
623
|
if (code && this.isAuthErrorCode(code)) {
|
|
627
|
-
const
|
|
628
|
-
|
|
629
|
-
|
|
630
|
-
throw
|
|
624
|
+
const authErr = new Error(message);
|
|
625
|
+
authErr.authError = true;
|
|
626
|
+
authErr.code = code;
|
|
627
|
+
throw authErr;
|
|
631
628
|
}
|
|
632
|
-
const
|
|
633
|
-
|
|
634
|
-
throw
|
|
629
|
+
const sseErr = new Error(message);
|
|
630
|
+
sseErr.sseData = parsed || data;
|
|
631
|
+
throw sseErr;
|
|
635
632
|
}
|
|
636
633
|
if ((META_EVENTS.has(event) || MESSAGE_EVENTS.has(event)) && data.startsWith('[META]')) {
|
|
637
634
|
const meta = tryParseJson(data.slice('[META]'.length));
|
|
@@ -876,7 +873,7 @@ class AIChatClient {
|
|
|
876
873
|
}
|
|
877
874
|
|
|
878
875
|
/**
|
|
879
|
-
* marked v18.0.
|
|
876
|
+
* marked v18.0.2 - a markdown parser
|
|
880
877
|
* Copyright (c) 2018-2026, MarkedJS. (MIT License)
|
|
881
878
|
* Copyright (c) 2011-2018, Christopher Jeffrey. (MIT License)
|
|
882
879
|
* https://github.com/markedjs/marked
|
|
@@ -887,12 +884,12 @@ class AIChatClient {
|
|
|
887
884
|
* The code in this file is generated from files in ./src/
|
|
888
885
|
*/
|
|
889
886
|
|
|
890
|
-
function
|
|
891
|
-
]`).replace("lheading",
|
|
887
|
+
function z(){return {async:false,breaks:false,extensions:null,gfm:true,hooks:null,pedantic:false,renderer:null,silent:false,tokenizer:null,walkTokens:null}}var T=z();function G(l){T=l;}var _={exec:()=>null};function k(l,e=""){let t=typeof l=="string"?l:l.source,n={replace:(s,r)=>{let i=typeof r=="string"?r:r.source;return i=i.replace(m.caret,"$1"),t=t.replace(s,i),n},getRegex:()=>new RegExp(t,e)};return n}var Re=((l="")=>{try{return !!new RegExp("(?<=1)(?<!1)"+l)}catch{return false}})(),m={codeRemoveIndent:/^(?: {1,4}| {0,3}\t)/gm,outputLinkReplace:/\\([\[\]])/g,indentCodeCompensation:/^(\s+)(?:```)/,beginningSpace:/^\s+/,endingHash:/#$/,startingSpaceChar:/^ /,endingSpaceChar:/ $/,nonSpaceChar:/[^ ]/,newLineCharGlobal:/\n/g,tabCharGlobal:/\t/g,multipleSpaceGlobal:/\s+/g,blankLine:/^[ \t]*$/,doubleBlankLine:/\n[ \t]*\n[ \t]*$/,blockquoteStart:/^ {0,3}>/,blockquoteSetextReplace:/\n {0,3}((?:=+|-+) *)(?=\n|$)/g,blockquoteSetextReplace2:/^ {0,3}>[ \t]?/gm,listReplaceNesting:/^ {1,4}(?=( {4})*[^ ])/g,listIsTask:/^\[[ xX]\] +\S/,listReplaceTask:/^\[[ xX]\] +/,listTaskCheckbox:/\[[ xX]\]/,anyLine:/\n.*\n/,hrefBrackets:/^<(.*)>$/,tableDelimiter:/[:|]/,tableAlignChars:/^\||\| *$/g,tableRowBlankLine:/\n[ \t]*$/,tableAlignRight:/^ *-+: *$/,tableAlignCenter:/^ *:-+: *$/,tableAlignLeft:/^ *:-+ *$/,startATag:/^<a /i,endATag:/^<\/a>/i,startPreScriptTag:/^<(pre|code|kbd|script)(\s|>)/i,endPreScriptTag:/^<\/(pre|code|kbd|script)(\s|>)/i,startAngleBracket:/^</,endAngleBracket:/>$/,pedanticHrefTitle:/^([^'"]*[^\s])\s+(['"])(.*)\2/,unicodeAlphaNumeric:/[\p{L}\p{N}]/u,escapeTest:/[&<>"']/,escapeReplace:/[&<>"']/g,escapeTestNoEncode:/[<>"']|&(?!(#\d{1,7}|#[Xx][a-fA-F0-9]{1,6}|\w+);)/,escapeReplaceNoEncode:/[<>"']|&(?!(#\d{1,7}|#[Xx][a-fA-F0-9]{1,6}|\w+);)/g,caret:/(^|[^\[])\^/g,percentDecode:/%25/g,findPipe:/\|/g,splitPipe:/ \|/,slashPipe:/\\\|/g,carriageReturn:/\r\n|\r/g,spaceLine:/^ +$/gm,notSpaceStart:/^\S*/,endingNewline:/\n$/,listItemRegex:l=>new RegExp(`^( {0,3}${l})((?:[ ][^\\n]*)?(?:\\n|$))`),nextBulletRegex:l=>new RegExp(`^ {0,${Math.min(3,l-1)}}(?:[*+-]|\\d{1,9}[.)])((?:[ ][^\\n]*)?(?:\\n|$))`),hrRegex:l=>new RegExp(`^ {0,${Math.min(3,l-1)}}((?:- *){3,}|(?:_ *){3,}|(?:\\* *){3,})(?:\\n+|$)`),fencesBeginRegex:l=>new RegExp(`^ {0,${Math.min(3,l-1)}}(?:\`\`\`|~~~)`),headingBeginRegex:l=>new RegExp(`^ {0,${Math.min(3,l-1)}}#`),htmlBeginRegex:l=>new RegExp(`^ {0,${Math.min(3,l-1)}}<(?:[a-z].*>|!--)`,"i"),blockquoteBeginRegex:l=>new RegExp(`^ {0,${Math.min(3,l-1)}}>`)},Te=/^(?:[ \t]*(?:\n|$))+/,Oe=/^((?: {4}| {0,3}\t)[^\n]+(?:\n(?:[ \t]*(?:\n|$))*)?)+/,we=/^ {0,3}(`{3,}(?=[^`\n]*(?:\n|$))|~{3,})([^\n]*)(?:\n|$)(?:|([\s\S]*?)(?:\n|$))(?: {0,3}\1[~`]* *(?=\n|$)|$)/,I=/^ {0,3}((?:-[\t ]*){3,}|(?:_[ \t]*){3,}|(?:\*[ \t]*){3,})(?:\n+|$)/,ye=/^ {0,3}(#{1,6})(?=\s|$)(.*)(?:\n+|$)/,Q=/ {0,3}(?:[*+-]|\d{1,9}[.)])/,ie=/^(?!bull |blockCode|fences|blockquote|heading|html|table)((?:.|\n(?!\s*?\n|bull |blockCode|fences|blockquote|heading|html|table))+?)\n {0,3}(=+|-+) *(?:\n+|$)/,oe=k(ie).replace(/bull/g,Q).replace(/blockCode/g,/(?: {4}| {0,3}\t)/).replace(/fences/g,/ {0,3}(?:`{3,}|~{3,})/).replace(/blockquote/g,/ {0,3}>/).replace(/heading/g,/ {0,3}#{1,6}/).replace(/html/g,/ {0,3}<[^\n>]+>\n/).replace(/\|table/g,"").getRegex(),Pe=k(ie).replace(/bull/g,Q).replace(/blockCode/g,/(?: {4}| {0,3}\t)/).replace(/fences/g,/ {0,3}(?:`{3,}|~{3,})/).replace(/blockquote/g,/ {0,3}>/).replace(/heading/g,/ {0,3}#{1,6}/).replace(/html/g,/ {0,3}<[^\n>]+>\n/).replace(/table/g,/ {0,3}\|?(?:[:\- ]*\|)+[\:\- ]*\n/).getRegex(),j=/^([^\n]+(?:\n(?!hr|heading|lheading|blockquote|fences|list|html|table| +\n)[^\n]+)*)/,Se=/^[^\n]+/,F=/(?!\s*\])(?:\\[\s\S]|[^\[\]\\])+/,$e=k(/^ {0,3}\[(label)\]: *(?:\n[ \t]*)?([^<\s][^\s]*|<.*?>)(?:(?: +(?:\n[ \t]*)?| *\n[ \t]*)(title))? *(?:\n+|$)/).replace("label",F).replace("title",/(?:"(?:\\"?|[^"\\])*"|'[^'\n]*(?:\n[^'\n]+)*\n?'|\([^()]*\))/).getRegex(),Le=k(/^(bull)([ \t][^\n]+?)?(?:\n|$)/).replace(/bull/g,Q).getRegex(),v="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|search|section|summary|table|tbody|td|tfoot|th|thead|title|tr|track|ul",U=/<!--(?:-?>|[\s\S]*?(?:-->|$))/,_e=k("^ {0,3}(?:<(script|pre|style|textarea)[\\s>][\\s\\S]*?(?:</\\1>[^\\n]*\\n+|$)|comment[^\\n]*(\\n+|$)|<\\?[\\s\\S]*?(?:\\?>\\n*|$)|<![A-Z][\\s\\S]*?(?:>\\n*|$)|<!\\[CDATA\\[[\\s\\S]*?(?:\\]\\]>\\n*|$)|</?(tag)(?: +|\\n|/?>)[\\s\\S]*?(?:(?:\\n[ ]*)+\\n|$)|<(?!script|pre|style|textarea)([a-z][\\w-]*)(?:attribute)*? */?>(?=[ \\t]*(?:\\n|$))[\\s\\S]*?(?:(?:\\n[ ]*)+\\n|$)|</(?!script|pre|style|textarea)[a-z][\\w-]*\\s*>(?=[ \\t]*(?:\\n|$))[\\s\\S]*?(?:(?:\\n[ ]*)+\\n|$))","i").replace("comment",U).replace("tag",v).replace("attribute",/ +[a-zA-Z:_][\w.:-]*(?: *= *"[^"\n]*"| *= *'[^'\n]*'| *= *[^\s"'=<>`]+)?/).getRegex(),ae=k(j).replace("hr",I).replace("heading"," {0,3}#{1,6}(?:\\s|$)").replace("|lheading","").replace("|table","").replace("blockquote"," {0,3}>").replace("fences"," {0,3}(?:`{3,}(?=[^`\\n]*\\n)|~{3,})[^\\n]*\\n").replace("list"," {0,3}(?:[*+-]|1[.)])[ \\t]").replace("html","</?(?:tag)(?: +|\\n|/?>)|<(?:script|pre|style|textarea|!--)").replace("tag",v).getRegex(),Me=k(/^( {0,3}> ?(paragraph|[^\n]*)(?:\n|$))+/).replace("paragraph",ae).getRegex(),K={blockquote:Me,code:Oe,def:$e,fences:we,heading:ye,hr:I,html:_e,lheading:oe,list:Le,newline:Te,paragraph:ae,table:_,text:Se},re=k("^ *([^\\n ].*)\\n {0,3}((?:\\| *)?:?-+:? *(?:\\| *:?-+:? *)*(?:\\| *)?)(?:\\n((?:(?! *\\n|hr|heading|blockquote|code|fences|list|html).*(?:\\n|$))*)\\n*|$)").replace("hr",I).replace("heading"," {0,3}#{1,6}(?:\\s|$)").replace("blockquote"," {0,3}>").replace("code","(?: {4}| {0,3} )[^\\n]").replace("fences"," {0,3}(?:`{3,}(?=[^`\\n]*\\n)|~{3,})[^\\n]*\\n").replace("list"," {0,3}(?:[*+-]|1[.)])[ \\t]").replace("html","</?(?:tag)(?: +|\\n|/?>)|<(?:script|pre|style|textarea|!--)").replace("tag",v).getRegex(),ze={...K,lheading:Pe,table:re,paragraph:k(j).replace("hr",I).replace("heading"," {0,3}#{1,6}(?:\\s|$)").replace("|lheading","").replace("table",re).replace("blockquote"," {0,3}>").replace("fences"," {0,3}(?:`{3,}(?=[^`\\n]*\\n)|~{3,})[^\\n]*\\n").replace("list"," {0,3}(?:[*+-]|1[.)])[ \\t]").replace("html","</?(?:tag)(?: +|\\n|/?>)|<(?:script|pre|style|textarea|!--)").replace("tag",v).getRegex()},Ee={...K,html:k(`^ *(?:comment *(?:\\n|\\s*$)|<(tag)[\\s\\S]+?</\\1> *(?:\\n{2,}|\\s*$)|<tag(?:"[^"]*"|'[^']*'|\\s[^'"/>\\s]*)*?/?> *(?:\\n{2,}|\\s*$))`).replace("comment",U).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:/^ *\[([^\]]+)\]: *<?([^\s>]+)>?(?: +(["(][^\n]+[")]))? *(?:\n+|$)/,heading:/^(#{1,6})(.*)(?:\n+|$)/,fences:_,lheading:/^(.+?)\n {0,3}(=+|-+) *(?:\n+|$)/,paragraph:k(j).replace("hr",I).replace("heading",` *#{1,6} *[^
|
|
888
|
+
]`).replace("lheading",oe).replace("|table","").replace("blockquote"," {0,3}>").replace("|fences","").replace("|list","").replace("|html","").replace("|tag","").getRegex()},Ae=/^\\([!"#$%&'()*+,\-./:;<=>?@\[\]\\^_`{|}~])/,Ce=/^(`+)([^`]|[^`][\s\S]*?[^`])\1(?!`)/,le=/^( {2,}|\\)\n(?!\s*$)/,Ie=/^(`+|[^`])(?:(?= {2,}\n)|[\s\S]*?(?:(?=[\\<!\[`*_]|\b_|$)|[^ ](?= {2,}\n)))/,E=/[\p{P}\p{S}]/u,H=/[\s\p{P}\p{S}]/u,W=/[^\s\p{P}\p{S}]/u,Be=k(/^((?![*_])punctSpace)/,"u").replace(/punctSpace/g,H).getRegex(),ue=/(?!~)[\p{P}\p{S}]/u,De=/(?!~)[\s\p{P}\p{S}]/u,qe=/(?:[^\s\p{P}\p{S}]|~)/u,ve=k(/link|precode-code|html/,"g").replace("link",/\[(?:[^\[\]`]|(?<a>`+)[^`]+\k<a>(?!`))*?\]\((?:\\[\s\S]|[^\\\(\)]|\((?:\\[\s\S]|[^\\\(\)])*\))*\)/).replace("precode-",Re?"(?<!`)()":"(^^|[^`])").replace("code",/(?<b>`+)[^`]+\k<b>(?!`)/).replace("html",/<(?! )[^<>]*?>/).getRegex(),pe=/^(?:\*+(?:((?!\*)punct)|([^\s*]))?)|^_+(?:((?!_)punct)|([^\s_]))?/,He=k(pe,"u").replace(/punct/g,E).getRegex(),Ze=k(pe,"u").replace(/punct/g,ue).getRegex(),ce="^[^_*]*?__[^_*]*?\\*[^_*]*?(?=__)|[^*]+(?=[^*])|(?!\\*)punct(\\*+)(?=[\\s]|$)|notPunctSpace(\\*+)(?!\\*)(?=punctSpace|$)|(?!\\*)punctSpace(\\*+)(?=notPunctSpace)|[\\s](\\*+)(?!\\*)(?=punct)|(?!\\*)punct(\\*+)(?!\\*)(?=punct)|notPunctSpace(\\*+)(?=notPunctSpace)",Ge=k(ce,"gu").replace(/notPunctSpace/g,W).replace(/punctSpace/g,H).replace(/punct/g,E).getRegex(),Ne=k(ce,"gu").replace(/notPunctSpace/g,qe).replace(/punctSpace/g,De).replace(/punct/g,ue).getRegex(),Qe=k("^[^_*]*?\\*\\*[^_*]*?_[^_*]*?(?=\\*\\*)|[^_]+(?=[^_])|(?!_)punct(_+)(?=[\\s]|$)|notPunctSpace(_+)(?!_)(?=punctSpace|$)|(?!_)punctSpace(_+)(?=notPunctSpace)|[\\s](_+)(?!_)(?=punct)|(?!_)punct(_+)(?!_)(?=punct)","gu").replace(/notPunctSpace/g,W).replace(/punctSpace/g,H).replace(/punct/g,E).getRegex(),je=k(/^~~?(?:((?!~)punct)|[^\s~])/,"u").replace(/punct/g,E).getRegex(),Fe="^[^~]+(?=[^~])|(?!~)punct(~~?)(?=[\\s]|$)|notPunctSpace(~~?)(?!~)(?=punctSpace|$)|(?!~)punctSpace(~~?)(?=notPunctSpace)|[\\s](~~?)(?!~)(?=punct)|(?!~)punct(~~?)(?!~)(?=punct)|notPunctSpace(~~?)(?=notPunctSpace)",Ue=k(Fe,"gu").replace(/notPunctSpace/g,W).replace(/punctSpace/g,H).replace(/punct/g,E).getRegex(),Ke=k(/\\(punct)/,"gu").replace(/punct/g,E).getRegex(),We=k(/^<(scheme:[^\s\x00-\x1f<>]*|email)>/).replace("scheme",/[a-zA-Z][a-zA-Z0-9+.-]{1,31}/).replace("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])?)+(?![-_])/).getRegex(),Xe=k(U).replace("(?:-->|$)","-->").getRegex(),Je=k("^comment|^</[a-zA-Z][\\w:-]*\\s*>|^<[a-zA-Z][\\w-]*(?:attribute)*?\\s*/?>|^<\\?[\\s\\S]*?\\?>|^<![a-zA-Z]+\\s[\\s\\S]*?>|^<!\\[CDATA\\[[\\s\\S]*?\\]\\]>").replace("comment",Xe).replace("attribute",/\s+[a-zA-Z:_][\w.:-]*(?:\s*=\s*"[^"]*"|\s*=\s*'[^']*'|\s*=\s*[^\s"'=<>`]+)?/).getRegex(),q=/(?:\[(?:\\[\s\S]|[^\[\]\\])*\]|\\[\s\S]|`+(?!`)[^`]*?`+(?!`)|``+(?=\])|[^\[\]\\`])*?/,Ve=k(/^!?\[(label)\]\(\s*(href)(?:(?:[ \t]+(?:\n[ \t]*)?|\n[ \t]*)(title))?\s*\)/).replace("label",q).replace("href",/<(?:\\.|[^\n<>\\])+>|[^ \t\n\x00-\x1f]*/).replace("title",/"(?:\\"?|[^"\\])*"|'(?:\\'?|[^'\\])*'|\((?:\\\)?|[^)\\])*\)/).getRegex(),he=k(/^!?\[(label)\]\[(ref)\]/).replace("label",q).replace("ref",F).getRegex(),ke=k(/^!?\[(ref)\](?:\[\])?/).replace("ref",F).getRegex(),Ye=k("reflink|nolink(?!\\()","g").replace("reflink",he).replace("nolink",ke).getRegex(),se=/[hH][tT][tT][pP][sS]?|[fF][tT][pP]/,X={_backpedal:_,anyPunctuation:Ke,autolink:We,blockSkip:ve,br:le,code:Ce,del:_,delLDelim:_,delRDelim:_,emStrongLDelim:He,emStrongRDelimAst:Ge,emStrongRDelimUnd:Qe,escape:Ae,link:Ve,nolink:ke,punctuation:Be,reflink:he,reflinkSearch:Ye,tag:Je,text:Ie,url:_},et={...X,link:k(/^!?\[(label)\]\((.*?)\)/).replace("label",q).getRegex(),reflink:k(/^!?\[(label)\]\s*\[([^\]]*)\]/).replace("label",q).getRegex()},N={...X,emStrongRDelimAst:Ne,emStrongLDelim:Ze,delLDelim:je,delRDelim:Ue,url:k(/^((?:protocol):\/\/|www\.)(?:[a-zA-Z0-9\-]+\.?)+[^\s<]*|^email/).replace("protocol",se).replace("email",/[A-Za-z0-9._+-]+(@)[a-zA-Z0-9-_]+(?:\.[a-zA-Z0-9-_]*[a-zA-Z0-9])+(?![-_])/).getRegex(),_backpedal:/(?:[^?!.,:;*_'"~()&]+|\([^)]*\)|&(?![a-zA-Z0-9]+;$)|[?!.,:;*_'"~)]+(?!$))+/,del:/^(~~?)(?=[^\s~])((?:\\[\s\S]|[^\\])*?(?:\\[\s\S]|[^\s~\\]))\1(?=[^~]|$)/,text:k(/^([`~]+|[^`~])(?:(?= {2,}\n)|(?=[a-zA-Z0-9.!#$%&'*+\/=?_`{\|}~-]+@)|[\s\S]*?(?:(?=[\\<!\[`*~_]|\b_|protocol:\/\/|www\.|$)|[^ ](?= {2,}\n)|[^a-zA-Z0-9.!#$%&'*+\/=?_`{\|}~-](?=[a-zA-Z0-9.!#$%&'*+\/=?_`{\|}~-]+@)))/).replace("protocol",se).getRegex()},tt={...N,br:k(le).replace("{2,}","*").getRegex(),text:k(N.text).replace("\\b_","\\b_| {2,}\\n").replace(/\{2,\}/g,"*").getRegex()},B={normal:K,gfm:ze,pedantic:Ee},A={normal:X,gfm:N,breaks:tt,pedantic:et};var nt={"&":"&","<":"<",">":">",'"':""","'":"'"},de=l=>nt[l];function O(l,e){if(e){if(m.escapeTest.test(l))return l.replace(m.escapeReplace,de)}else if(m.escapeTestNoEncode.test(l))return l.replace(m.escapeReplaceNoEncode,de);return l}function J(l){try{l=encodeURI(l).replace(m.percentDecode,"%");}catch{return null}return l}function V(l,e){let t=l.replace(m.findPipe,(r,i,o)=>{let u=false,a=i;for(;--a>=0&&o[a]==="\\";)u=!u;return u?"|":" |"}),n=t.split(m.splitPipe),s=0;if(n[0].trim()||n.shift(),n.length>0&&!n.at(-1)?.trim()&&n.pop(),e)if(n.length>e)n.splice(e);else for(;n.length<e;)n.push("");for(;s<n.length;s++)n[s]=n[s].trim().replace(m.slashPipe,"|");return n}function $(l,e,t){let n=l.length;if(n===0)return "";let s=0;for(;s<n;){let r=l.charAt(n-s-1);if(r===e&&true)s++;else break}return l.slice(0,n-s)}function Y(l){let e=l.split(`
|
|
892
889
|
`),t=e.length-1;for(;t>=0&&m.blankLine.test(e[t]);)t--;return e.length-t<=2?l:e.slice(0,t+1).join(`
|
|
893
|
-
`)}function
|
|
890
|
+
`)}function ge(l,e){if(l.indexOf(e[1])===-1)return -1;let t=0;for(let n=0;n<l.length;n++)if(l[n]==="\\")n++;else if(l[n]===e[0])t++;else if(l[n]===e[1]&&(t--,t<0))return n;return t>0?-2:-1}function fe(l,e=0){let t=e,n="";for(let s of l)if(s===" "){let r=4-t%4;n+=" ".repeat(r),t+=r;}else n+=s,t++;return n}function me(l,e,t,n,s){let r=e.href,i=e.title||null,o=l[1].replace(s.other.outputLinkReplace,"$1");n.state.inLink=true;let u={type:l[0].charAt(0)==="!"?"image":"link",raw:t,href:r,title:i,text:o,tokens:n.inlineTokens(o)};return n.state.inLink=false,u}function rt(l,e,t){let n=l.match(t.other.indentCodeCompensation);if(n===null)return e;let s=n[1];return e.split(`
|
|
894
891
|
`).map(r=>{let i=r.match(t.other.beginningSpace);if(i===null)return r;let[o]=i;return o.length>=s.length?r.slice(s.length):r}).join(`
|
|
895
|
-
`)}var w=class{options;rules;lexer;constructor(e){this.options=e||T;}space(e){let t=this.rules.block.newline.exec(e);if(t&&t[0].length>0)return {type:"space",raw:t[0]}}code(e){let t=this.rules.block.code.exec(e);if(t){let n=this.options.pedantic?t[0]:
|
|
892
|
+
`)}var w=class{options;rules;lexer;constructor(e){this.options=e||T;}space(e){let t=this.rules.block.newline.exec(e);if(t&&t[0].length>0)return {type:"space",raw:t[0]}}code(e){let t=this.rules.block.code.exec(e);if(t){let n=this.options.pedantic?t[0]:Y(t[0]),s=n.replace(this.rules.other.codeRemoveIndent,"");return {type:"code",raw:n,codeBlockStyle:"indented",text:s}}}fences(e){let t=this.rules.block.fences.exec(e);if(t){let n=t[0],s=rt(n,t[3]||"",this.rules);return {type:"code",raw:n,lang:t[2]?t[2].trim().replace(this.rules.inline.anyPunctuation,"$1"):t[2],text:s}}}heading(e){let t=this.rules.block.heading.exec(e);if(t){let n=t[2].trim();if(this.rules.other.endingHash.test(n)){let s=$(n,"#");(this.options.pedantic||!s||this.rules.other.endingSpaceChar.test(s))&&(n=s.trim());}return {type:"heading",raw:$(t[0],`
|
|
896
893
|
`),depth:t[1].length,text:n,tokens:this.lexer.inline(n)}}}hr(e){let t=this.rules.block.hr.exec(e);if(t)return {type:"hr",raw:$(t[0],`
|
|
897
894
|
`)}}blockquote(e){let t=this.rules.block.blockquote.exec(e);if(t){let n=$(t[0],`
|
|
898
895
|
`).split(`
|
|
@@ -900,24 +897,24 @@ function M(){return {async:false,breaks:false,extensions:null,gfm:true,hooks:nul
|
|
|
900
897
|
`),p=c.replace(this.rules.other.blockquoteSetextReplace,`
|
|
901
898
|
$1`).replace(this.rules.other.blockquoteSetextReplace2,"");s=s?`${s}
|
|
902
899
|
${c}`:c,r=r?`${r}
|
|
903
|
-
${p}`:p;let
|
|
900
|
+
${p}`:p;let d=this.lexer.state.top;if(this.lexer.state.top=true,this.lexer.blockTokens(p,i,true),this.lexer.state.top=d,n.length===0)break;let h=i.at(-1);if(h?.type==="code")break;if(h?.type==="blockquote"){let R=h,f=R.raw+`
|
|
904
901
|
`+n.join(`
|
|
905
902
|
`),S=this.blockquote(f);i[i.length-1]=S,s=s.substring(0,s.length-R.raw.length)+S.raw,r=r.substring(0,r.length-R.text.length)+S.text;break}else if(h?.type==="list"){let R=h,f=R.raw+`
|
|
906
903
|
`+n.join(`
|
|
907
904
|
`),S=this.list(f);i[i.length-1]=S,s=s.substring(0,s.length-h.raw.length)+S.raw,r=r.substring(0,r.length-R.raw.length)+S.raw,n=f.substring(i.at(-1).raw.length).split(`
|
|
908
|
-
`);continue}}return {type:"blockquote",raw:s,tokens:i,text:r}}}list(e){let t=this.rules.block.list.exec(e);if(t){let n=t[1].trim(),s=n.length>1,r={type:"list",raw:"",ordered:s,start:s?+n.slice(0,-1):"",loose:false,items:[]};n=s?`\\d{1,9}\\${n.slice(-1)}`:`\\${n}`,this.options.pedantic&&(n=s?n:"[*+-]");let i=this.rules.other.listItemRegex(n),o=false;for(;e;){let a=false,c="",p="";if(!(t=i.exec(e))||this.rules.block.hr.test(e))break;c=t[0],e=e.substring(c.length);let
|
|
905
|
+
`);continue}}return {type:"blockquote",raw:s,tokens:i,text:r}}}list(e){let t=this.rules.block.list.exec(e);if(t){let n=t[1].trim(),s=n.length>1,r={type:"list",raw:"",ordered:s,start:s?+n.slice(0,-1):"",loose:false,items:[]};n=s?`\\d{1,9}\\${n.slice(-1)}`:`\\${n}`,this.options.pedantic&&(n=s?n:"[*+-]");let i=this.rules.other.listItemRegex(n),o=false;for(;e;){let a=false,c="",p="";if(!(t=i.exec(e))||this.rules.block.hr.test(e))break;c=t[0],e=e.substring(c.length);let d=fe(t[2].split(`
|
|
909
906
|
`,1)[0],t[1].length),h=e.split(`
|
|
910
|
-
`,1)[0],R=!
|
|
911
|
-
`,e=e.substring(h.length+1),a=true),!a){let S=this.rules.other.nextBulletRegex(f),
|
|
912
|
-
`,1)[0],C;if(h=
|
|
913
|
-
`+C.slice(f);else {if(R||
|
|
914
|
-
`+h;}R=!h.trim(),c+=
|
|
915
|
-
`,e=e.substring(
|
|
916
|
-
`),href:s,title:r}}}table(e){let t=this.rules.block.table.exec(e);if(!t||!this.rules.other.tableDelimiter.test(t[2]))return;let n=
|
|
907
|
+
`,1)[0],R=!d.trim(),f=0;if(this.options.pedantic?(f=2,p=d.trimStart()):R?f=t[1].length+1:(f=d.search(this.rules.other.nonSpaceChar),f=f>4?1:f,p=d.slice(f),f+=t[1].length),R&&this.rules.other.blankLine.test(h)&&(c+=h+`
|
|
908
|
+
`,e=e.substring(h.length+1),a=true),!a){let S=this.rules.other.nextBulletRegex(f),ee=this.rules.other.hrRegex(f),te=this.rules.other.fencesBeginRegex(f),ne=this.rules.other.headingBeginRegex(f),xe=this.rules.other.htmlBeginRegex(f),be=this.rules.other.blockquoteBeginRegex(f);for(;e;){let Z=e.split(`
|
|
909
|
+
`,1)[0],C;if(h=Z,this.options.pedantic?(h=h.replace(this.rules.other.listReplaceNesting," "),C=h):C=h.replace(this.rules.other.tabCharGlobal," "),te.test(h)||ne.test(h)||xe.test(h)||be.test(h)||S.test(h)||ee.test(h))break;if(C.search(this.rules.other.nonSpaceChar)>=f||!h.trim())p+=`
|
|
910
|
+
`+C.slice(f);else {if(R||d.replace(this.rules.other.tabCharGlobal," ").search(this.rules.other.nonSpaceChar)>=4||te.test(d)||ne.test(d)||ee.test(d))break;p+=`
|
|
911
|
+
`+h;}R=!h.trim(),c+=Z+`
|
|
912
|
+
`,e=e.substring(Z.length+1),d=C.slice(f);}}r.loose||(o?r.loose=true:this.rules.other.doubleBlankLine.test(c)&&(o=true)),r.items.push({type:"list_item",raw:c,task:!!this.options.gfm&&this.rules.other.listIsTask.test(p),loose:false,text:p,tokens:[]}),r.raw+=c;}let u=r.items.at(-1);if(u)u.raw=u.raw.trimEnd(),u.text=u.text.trimEnd();else return;r.raw=r.raw.trimEnd();for(let a of r.items){if(this.lexer.state.top=false,a.tokens=this.lexer.blockTokens(a.text,[]),a.task){if(a.text=a.text.replace(this.rules.other.listReplaceTask,""),a.tokens[0]?.type==="text"||a.tokens[0]?.type==="paragraph"){a.tokens[0].raw=a.tokens[0].raw.replace(this.rules.other.listReplaceTask,""),a.tokens[0].text=a.tokens[0].text.replace(this.rules.other.listReplaceTask,"");for(let p=this.lexer.inlineQueue.length-1;p>=0;p--)if(this.rules.other.listIsTask.test(this.lexer.inlineQueue[p].src)){this.lexer.inlineQueue[p].src=this.lexer.inlineQueue[p].src.replace(this.rules.other.listReplaceTask,"");break}}let c=this.rules.other.listTaskCheckbox.exec(a.raw);if(c){let p={type:"checkbox",raw:c[0]+" ",checked:c[0]!=="[ ]"};a.checked=p.checked,r.loose?a.tokens[0]&&["paragraph","text"].includes(a.tokens[0].type)&&"tokens"in a.tokens[0]&&a.tokens[0].tokens?(a.tokens[0].raw=p.raw+a.tokens[0].raw,a.tokens[0].text=p.raw+a.tokens[0].text,a.tokens[0].tokens.unshift(p)):a.tokens.unshift({type:"paragraph",raw:p.raw,text:p.raw,tokens:[p]}):a.tokens.unshift(p);}}if(!r.loose){let c=a.tokens.filter(d=>d.type==="space"),p=c.length>0&&c.some(d=>this.rules.other.anyLine.test(d.raw));r.loose=p;}}if(r.loose)for(let a of r.items){a.loose=true;for(let c of a.tokens)c.type==="text"&&(c.type="paragraph");}return r}}html(e){let t=this.rules.block.html.exec(e);if(t){let n=Y(t[0]);return {type:"html",block:true,raw:n,pre:t[1]==="pre"||t[1]==="script"||t[1]==="style",text:n}}}def(e){let t=this.rules.block.def.exec(e);if(t){let n=t[1].toLowerCase().replace(this.rules.other.multipleSpaceGlobal," "),s=t[2]?t[2].replace(this.rules.other.hrefBrackets,"$1").replace(this.rules.inline.anyPunctuation,"$1"):"",r=t[3]?t[3].substring(1,t[3].length-1).replace(this.rules.inline.anyPunctuation,"$1"):t[3];return {type:"def",tag:n,raw:$(t[0],`
|
|
913
|
+
`),href:s,title:r}}}table(e){let t=this.rules.block.table.exec(e);if(!t||!this.rules.other.tableDelimiter.test(t[2]))return;let n=V(t[1]),s=t[2].replace(this.rules.other.tableAlignChars,"").split("|"),r=t[3]?.trim()?t[3].replace(this.rules.other.tableRowBlankLine,"").split(`
|
|
917
914
|
`):[],i={type:"table",raw:$(t[0],`
|
|
918
|
-
`),header:[],align:[],rows:[]};if(n.length===s.length){for(let o of s)this.rules.other.tableAlignRight.test(o)?i.align.push("right"):this.rules.other.tableAlignCenter.test(o)?i.align.push("center"):this.rules.other.tableAlignLeft.test(o)?i.align.push("left"):i.align.push(null);for(let o=0;o<n.length;o++)i.header.push({text:n[o],tokens:this.lexer.inline(n[o]),header:true,align:i.align[o]});for(let o of r)i.rows.push(
|
|
915
|
+
`),header:[],align:[],rows:[]};if(n.length===s.length){for(let o of s)this.rules.other.tableAlignRight.test(o)?i.align.push("right"):this.rules.other.tableAlignCenter.test(o)?i.align.push("center"):this.rules.other.tableAlignLeft.test(o)?i.align.push("left"):i.align.push(null);for(let o=0;o<n.length;o++)i.header.push({text:n[o],tokens:this.lexer.inline(n[o]),header:true,align:i.align[o]});for(let o of r)i.rows.push(V(o,i.header.length).map((u,a)=>({text:u,tokens:this.lexer.inline(u),header:false,align:i.align[a]})));return i}}lheading(e){let t=this.rules.block.lheading.exec(e);if(t){let n=t[1].trim();return {type:"heading",raw:$(t[0],`
|
|
919
916
|
`),depth:t[2].charAt(0)==="="?1:2,text:n,tokens:this.lexer.inline(n)}}}paragraph(e){let t=this.rules.block.paragraph.exec(e);if(t){let n=t[1].charAt(t[1].length-1)===`
|
|
920
|
-
`?t[1].slice(0,-1):t[1];return {type:"paragraph",raw:t[0],text:n,tokens:this.lexer.inline(n)}}}text(e){let t=this.rules.block.text.exec(e);if(t)return {type:"text",raw:t[0],text:t[0],tokens:this.lexer.inline(t[0])}}escape(e){let t=this.rules.inline.escape.exec(e);if(t)return {type:"escape",raw:t[0],text:t[1]}}tag(e){let t=this.rules.inline.tag.exec(e);if(t)return !this.lexer.state.inLink&&this.rules.other.startATag.test(t[0])?this.lexer.state.inLink=true:this.lexer.state.inLink&&this.rules.other.endATag.test(t[0])&&(this.lexer.state.inLink=false),!this.lexer.state.inRawBlock&&this.rules.other.startPreScriptTag.test(t[0])?this.lexer.state.inRawBlock=true:this.lexer.state.inRawBlock&&this.rules.other.endPreScriptTag.test(t[0])&&(this.lexer.state.inRawBlock=false),{type:"html",raw:t[0],inLink:this.lexer.state.inLink,inRawBlock:this.lexer.state.inRawBlock,block:false,text:t[0]}}link(e){let t=this.rules.inline.link.exec(e);if(t){let n=t[2].trim();if(!this.options.pedantic&&this.rules.other.startAngleBracket.test(n)){if(!this.rules.other.endAngleBracket.test(n))return;let i=$(n.slice(0,-1),"\\");if((n.length-i.length)%2===0)return}else {let i=
|
|
917
|
+
`?t[1].slice(0,-1):t[1];return {type:"paragraph",raw:t[0],text:n,tokens:this.lexer.inline(n)}}}text(e){let t=this.rules.block.text.exec(e);if(t)return {type:"text",raw:t[0],text:t[0],tokens:this.lexer.inline(t[0])}}escape(e){let t=this.rules.inline.escape.exec(e);if(t)return {type:"escape",raw:t[0],text:t[1]}}tag(e){let t=this.rules.inline.tag.exec(e);if(t)return !this.lexer.state.inLink&&this.rules.other.startATag.test(t[0])?this.lexer.state.inLink=true:this.lexer.state.inLink&&this.rules.other.endATag.test(t[0])&&(this.lexer.state.inLink=false),!this.lexer.state.inRawBlock&&this.rules.other.startPreScriptTag.test(t[0])?this.lexer.state.inRawBlock=true:this.lexer.state.inRawBlock&&this.rules.other.endPreScriptTag.test(t[0])&&(this.lexer.state.inRawBlock=false),{type:"html",raw:t[0],inLink:this.lexer.state.inLink,inRawBlock:this.lexer.state.inRawBlock,block:false,text:t[0]}}link(e){let t=this.rules.inline.link.exec(e);if(t){let n=t[2].trim();if(!this.options.pedantic&&this.rules.other.startAngleBracket.test(n)){if(!this.rules.other.endAngleBracket.test(n))return;let i=$(n.slice(0,-1),"\\");if((n.length-i.length)%2===0)return}else {let i=ge(t[2],"()");if(i===-2)return;if(i>-1){let u=(t[0].indexOf("!")===0?5:4)+t[1].length+i;t[2]=t[2].substring(0,i),t[0]=t[0].substring(0,u).trim(),t[3]="";}}let s=t[2],r="";if(this.options.pedantic){let i=this.rules.other.pedanticHrefTitle.exec(s);i&&(s=i[1],r=i[3]);}else r=t[3]?t[3].slice(1,-1):"";return s=s.trim(),this.rules.other.startAngleBracket.test(s)&&(this.options.pedantic&&!this.rules.other.endAngleBracket.test(n)?s=s.slice(1):s=s.slice(1,-1)),me(t,{href:s&&s.replace(this.rules.inline.anyPunctuation,"$1"),title:r&&r.replace(this.rules.inline.anyPunctuation,"$1")},t[0],this.lexer,this.rules)}}reflink(e,t){let n;if((n=this.rules.inline.reflink.exec(e))||(n=this.rules.inline.nolink.exec(e))){let s=(n[2]||n[1]).replace(this.rules.other.multipleSpaceGlobal," "),r=t[s.toLowerCase()];if(!r){let i=n[0].charAt(0);return {type:"text",raw:i,text:i}}return me(n,r,n[0],this.lexer,this.rules)}}emStrong(e,t,n=""){let s=this.rules.inline.emStrongLDelim.exec(e);if(!s||!s[1]&&!s[2]&&!s[3]&&!s[4]||s[4]&&n.match(this.rules.other.unicodeAlphaNumeric))return;if(!(s[1]||s[3]||"")||!n||this.rules.inline.punctuation.exec(n)){let i=[...s[0]].length-1,o,u,a=i,c=0,p=s[0][0]==="*"?this.rules.inline.emStrongRDelimAst:this.rules.inline.emStrongRDelimUnd;for(p.lastIndex=0,t=t.slice(-1*e.length+i);(s=p.exec(t))!==null;){if(o=s[1]||s[2]||s[3]||s[4]||s[5]||s[6],!o)continue;if(u=[...o].length,s[3]||s[4]){a+=u;continue}else if((s[5]||s[6])&&i%3&&!((i+u)%3)){c+=u;continue}if(a-=u,a>0)continue;u=Math.min(u,u+a+c);let d=[...s[0]][0].length,h=e.slice(0,i+s.index+d+u);if(Math.min(i,u)%2){let f=h.slice(1,-1);return {type:"em",raw:h,text:f,tokens:this.lexer.inlineTokens(f)}}let R=h.slice(2,-2);return {type:"strong",raw:h,text:R,tokens:this.lexer.inlineTokens(R)}}}}codespan(e){let t=this.rules.inline.code.exec(e);if(t){let n=t[2].replace(this.rules.other.newLineCharGlobal," "),s=this.rules.other.nonSpaceChar.test(n),r=this.rules.other.startingSpaceChar.test(n)&&this.rules.other.endingSpaceChar.test(n);return s&&r&&(n=n.substring(1,n.length-1)),{type:"codespan",raw:t[0],text:n}}}br(e){let t=this.rules.inline.br.exec(e);if(t)return {type:"br",raw:t[0]}}del(e,t,n=""){let s=this.rules.inline.delLDelim.exec(e);if(!s)return;if(!(s[1]||"")||!n||this.rules.inline.punctuation.exec(n)){let i=[...s[0]].length-1,o,u,a=i,c=this.rules.inline.delRDelim;for(c.lastIndex=0,t=t.slice(-1*e.length+i);(s=c.exec(t))!==null;){if(o=s[1]||s[2]||s[3]||s[4]||s[5]||s[6],!o||(u=[...o].length,u!==i))continue;if(s[3]||s[4]){a+=u;continue}if(a-=u,a>0)continue;u=Math.min(u,u+a);let p=[...s[0]][0].length,d=e.slice(0,i+s.index+p+u),h=d.slice(i,-i);return {type:"del",raw:d,text:h,tokens:this.lexer.inlineTokens(h)}}}}autolink(e){let t=this.rules.inline.autolink.exec(e);if(t){let n,s;return t[2]==="@"?(n=t[1],s="mailto:"+n):(n=t[1],s=n),{type:"link",raw:t[0],text:n,href:s,tokens:[{type:"text",raw:n,text:n}]}}}url(e){let t;if(t=this.rules.inline.url.exec(e)){let n,s;if(t[2]==="@")n=t[0],s="mailto:"+n;else {let r;do r=t[0],t[0]=this.rules.inline._backpedal.exec(t[0])?.[0]??"";while(r!==t[0]);n=t[0],t[1]==="www."?s="http://"+t[0]:s=t[0];}return {type:"link",raw:t[0],text:n,href:s,tokens:[{type:"text",raw:n,text:n}]}}}inlineText(e){let t=this.rules.inline.text.exec(e);if(t){let n=this.lexer.state.inRawBlock;return {type:"text",raw:t[0],text:t[0],escaped:n}}}};var x=class l{tokens;options;state;inlineQueue;tokenizer;constructor(e){this.tokens=[],this.tokens.links=Object.create(null),this.options=e||T,this.options.tokenizer=this.options.tokenizer||new w,this.tokenizer=this.options.tokenizer,this.tokenizer.options=this.options,this.tokenizer.lexer=this,this.inlineQueue=[],this.state={inLink:false,inRawBlock:false,top:true};let t={other:m,block:B.normal,inline:A.normal};this.options.pedantic?(t.block=B.pedantic,t.inline=A.pedantic):this.options.gfm&&(t.block=B.gfm,this.options.breaks?t.inline=A.breaks:t.inline=A.gfm),this.tokenizer.rules=t;}static get rules(){return {block:B,inline:A}}static lex(e,t){return new l(t).lex(e)}static lexInline(e,t){return new l(t).inlineTokens(e)}lex(e){e=e.replace(m.carriageReturn,`
|
|
921
918
|
`),this.blockTokens(e,this.tokens);for(let t=0;t<this.inlineQueue.length;t++){let n=this.inlineQueue[t];this.inlineTokens(n.src,n.tokens);}return this.inlineQueue=[],this.tokens}blockTokens(e,t=[],n=false){this.tokenizer.lexer=this,this.options.pedantic&&(e=e.replace(m.tabCharGlobal," ").replace(m.spaceLine,""));let s=1/0;for(;e;){if(e.length<s)s=e.length;else {this.infiniteLoopError(e.charCodeAt(0));break}let r;if(this.options.extensions?.block?.some(o=>(r=o.call({lexer:this},e,t))?(e=e.substring(r.raw.length),t.push(r),true):false))continue;if(r=this.tokenizer.space(e)){e=e.substring(r.raw.length);let o=t.at(-1);r.raw.length===1&&o!==void 0?o.raw+=`
|
|
922
919
|
`:t.push(r);continue}if(r=this.tokenizer.code(e)){e=e.substring(r.raw.length);let o=t.at(-1);o?.type==="paragraph"||o?.type==="text"?(o.raw+=(o.raw.endsWith(`
|
|
923
920
|
`)?"":`
|
|
@@ -931,7 +928,7 @@ ${p}`:p;let k=this.lexer.state.top;if(this.lexer.state.top=true,this.lexer.block
|
|
|
931
928
|
`+r.text,this.inlineQueue.pop(),this.inlineQueue.at(-1).src=o.text):t.push(r),n=i.length!==e.length,e=e.substring(r.raw.length);continue}if(r=this.tokenizer.text(e)){e=e.substring(r.raw.length);let o=t.at(-1);o?.type==="text"?(o.raw+=(o.raw.endsWith(`
|
|
932
929
|
`)?"":`
|
|
933
930
|
`)+r.raw,o.text+=`
|
|
934
|
-
`+r.text,this.inlineQueue.pop(),this.inlineQueue.at(-1).src=o.text):t.push(r);continue}if(e){this.infiniteLoopError(e.charCodeAt(0));break}}return this.state.top=true,t}inline(e,t=[]){return this.inlineQueue.push({src:e,tokens:t}),t}inlineTokens(e,t=[]){this.tokenizer.lexer=this;let n=e,s=null;if(this.tokens.links){let a=Object.keys(this.tokens.links);if(a.length>0)for(;(s=this.tokenizer.rules.inline.reflinkSearch.exec(n))!==null;)a.includes(s[0].slice(s[0].lastIndexOf("[")+1,-1))&&(n=n.slice(0,s.index)+"["+"a".repeat(s[0].length-2)+"]"+n.slice(this.tokenizer.rules.inline.reflinkSearch.lastIndex));}for(;(s=this.tokenizer.rules.inline.anyPunctuation.exec(n))!==null;)n=n.slice(0,s.index)+"++"+n.slice(this.tokenizer.rules.inline.anyPunctuation.lastIndex);let r;for(;(s=this.tokenizer.rules.inline.blockSkip.exec(n))!==null;)r=s[2]?s[2].length:0,n=n.slice(0,s.index+r)+"["+"a".repeat(s[0].length-r-2)+"]"+n.slice(this.tokenizer.rules.inline.blockSkip.lastIndex);n=this.options.hooks?.emStrongMask?.call({lexer:this},n)??n;let i=false,o="",u=1/0;for(;e;){if(e.length<u)u=e.length;else {this.infiniteLoopError(e.charCodeAt(0));break}i||(o=""),i=false;let a;if(this.options.extensions?.inline?.some(p=>(a=p.call({lexer:this},e,t))?(e=e.substring(a.raw.length),t.push(a),true):false))continue;if(a=this.tokenizer.escape(e)){e=e.substring(a.raw.length),t.push(a);continue}if(a=this.tokenizer.tag(e)){e=e.substring(a.raw.length),t.push(a);continue}if(a=this.tokenizer.link(e)){e=e.substring(a.raw.length),t.push(a);continue}if(a=this.tokenizer.reflink(e,this.tokens.links)){e=e.substring(a.raw.length);let p=t.at(-1);a.type==="text"&&p?.type==="text"?(p.raw+=a.raw,p.text+=a.text):t.push(a);continue}if(a=this.tokenizer.emStrong(e,n,o)){e=e.substring(a.raw.length),t.push(a);continue}if(a=this.tokenizer.codespan(e)){e=e.substring(a.raw.length),t.push(a);continue}if(a=this.tokenizer.br(e)){e=e.substring(a.raw.length),t.push(a);continue}if(a=this.tokenizer.del(e,n,o)){e=e.substring(a.raw.length),t.push(a);continue}if(a=this.tokenizer.autolink(e)){e=e.substring(a.raw.length),t.push(a);continue}if(!this.state.inLink&&(a=this.tokenizer.url(e))){e=e.substring(a.raw.length),t.push(a);continue}let c=e;if(this.options.extensions?.startInline){let p=1/0,
|
|
931
|
+
`+r.text,this.inlineQueue.pop(),this.inlineQueue.at(-1).src=o.text):t.push(r);continue}if(e){this.infiniteLoopError(e.charCodeAt(0));break}}return this.state.top=true,t}inline(e,t=[]){return this.inlineQueue.push({src:e,tokens:t}),t}inlineTokens(e,t=[]){this.tokenizer.lexer=this;let n=e,s=null;if(this.tokens.links){let a=Object.keys(this.tokens.links);if(a.length>0)for(;(s=this.tokenizer.rules.inline.reflinkSearch.exec(n))!==null;)a.includes(s[0].slice(s[0].lastIndexOf("[")+1,-1))&&(n=n.slice(0,s.index)+"["+"a".repeat(s[0].length-2)+"]"+n.slice(this.tokenizer.rules.inline.reflinkSearch.lastIndex));}for(;(s=this.tokenizer.rules.inline.anyPunctuation.exec(n))!==null;)n=n.slice(0,s.index)+"++"+n.slice(this.tokenizer.rules.inline.anyPunctuation.lastIndex);let r;for(;(s=this.tokenizer.rules.inline.blockSkip.exec(n))!==null;)r=s[2]?s[2].length:0,n=n.slice(0,s.index+r)+"["+"a".repeat(s[0].length-r-2)+"]"+n.slice(this.tokenizer.rules.inline.blockSkip.lastIndex);n=this.options.hooks?.emStrongMask?.call({lexer:this},n)??n;let i=false,o="",u=1/0;for(;e;){if(e.length<u)u=e.length;else {this.infiniteLoopError(e.charCodeAt(0));break}i||(o=""),i=false;let a;if(this.options.extensions?.inline?.some(p=>(a=p.call({lexer:this},e,t))?(e=e.substring(a.raw.length),t.push(a),true):false))continue;if(a=this.tokenizer.escape(e)){e=e.substring(a.raw.length),t.push(a);continue}if(a=this.tokenizer.tag(e)){e=e.substring(a.raw.length),t.push(a);continue}if(a=this.tokenizer.link(e)){e=e.substring(a.raw.length),t.push(a);continue}if(a=this.tokenizer.reflink(e,this.tokens.links)){e=e.substring(a.raw.length);let p=t.at(-1);a.type==="text"&&p?.type==="text"?(p.raw+=a.raw,p.text+=a.text):t.push(a);continue}if(a=this.tokenizer.emStrong(e,n,o)){e=e.substring(a.raw.length),t.push(a);continue}if(a=this.tokenizer.codespan(e)){e=e.substring(a.raw.length),t.push(a);continue}if(a=this.tokenizer.br(e)){e=e.substring(a.raw.length),t.push(a);continue}if(a=this.tokenizer.del(e,n,o)){e=e.substring(a.raw.length),t.push(a);continue}if(a=this.tokenizer.autolink(e)){e=e.substring(a.raw.length),t.push(a);continue}if(!this.state.inLink&&(a=this.tokenizer.url(e))){e=e.substring(a.raw.length),t.push(a);continue}let c=e;if(this.options.extensions?.startInline){let p=1/0,d=e.slice(1),h;this.options.extensions.startInline.forEach(R=>{h=R.call({lexer:this},d),typeof h=="number"&&h>=0&&(p=Math.min(p,h));}),p<1/0&&p>=0&&(c=e.substring(0,p+1));}if(a=this.tokenizer.inlineText(c)){e=e.substring(a.raw.length),a.raw.slice(-1)!=="_"&&(o=a.raw.slice(-1)),i=true;let p=t.at(-1);p?.type==="text"?(p.raw+=a.raw,p.text+=a.text):t.push(a);continue}if(e){this.infiniteLoopError(e.charCodeAt(0));break}}return t}infiniteLoopError(e){let t="Infinite loop on byte: "+e;if(this.options.silent)console.error(t);else throw new Error(t)}};var y=class{options;parser;constructor(e){this.options=e||T;}space(e){return ""}code({text:e,lang:t,escaped:n}){let s=(t||"").match(m.notSpaceStart)?.[0],r=e.replace(m.endingNewline,"")+`
|
|
935
932
|
`;return s?'<pre><code class="language-'+O(s)+'">'+(n?r:O(r,true))+`</code></pre>
|
|
936
933
|
`:"<pre><code>"+(n?r:O(r,true))+`</code></pre>
|
|
937
934
|
`}blockquote({tokens:e}){return `<blockquote>
|
|
@@ -949,68 +946,27 @@ ${this.parser.parse(e)}</blockquote>
|
|
|
949
946
|
`}tablerow({text:e}){return `<tr>
|
|
950
947
|
${e}</tr>
|
|
951
948
|
`}tablecell(e){let t=this.parser.parseInline(e.tokens),n=e.header?"th":"td";return (e.align?`<${n} align="${e.align}">`:`<${n}>`)+t+`</${n}>
|
|
952
|
-
`}strong({tokens:e}){return `<strong>${this.parser.parseInline(e)}</strong>`}em({tokens:e}){return `<em>${this.parser.parseInline(e)}</em>`}codespan({text:e}){return `<code>${O(e,true)}</code>`}br(e){return "<br>"}del({tokens:e}){return `<del>${this.parser.parseInline(e)}</del>`}link({href:e,title:t,tokens:n}){let s=this.parser.parseInline(n),r=
|
|
953
|
-
Please report this to https://github.com/markedjs/marked.`,e){let s="<p>An error occurred:</p><pre>"+O(n.message+"",true)+"</pre>";return t?Promise.resolve(s):s}if(t)return Promise.reject(n);throw n}}};var
|
|
954
|
-
|
|
955
|
-
/*! @license DOMPurify 3.4.
|
|
956
|
-
|
|
957
|
-
|
|
958
|
-
|
|
959
|
-
|
|
960
|
-
|
|
961
|
-
|
|
962
|
-
|
|
963
|
-
|
|
964
|
-
|
|
965
|
-
|
|
966
|
-
|
|
967
|
-
|
|
968
|
-
|
|
969
|
-
|
|
970
|
-
|
|
971
|
-
|
|
972
|
-
|
|
973
|
-
f = true,
|
|
974
|
-
o = false;
|
|
975
|
-
try {
|
|
976
|
-
if (i = (t = t.call(r)).next, 0 === l) ; else for (; !(f = (e = i.call(t)).done) && (a.push(e.value), a.length !== l); f = !0);
|
|
977
|
-
} catch (r) {
|
|
978
|
-
o = true, n = r;
|
|
979
|
-
} finally {
|
|
980
|
-
try {
|
|
981
|
-
if (!f && null != t.return && (u = t.return(), Object(u) !== u)) return;
|
|
982
|
-
} finally {
|
|
983
|
-
if (o) throw n;
|
|
984
|
-
}
|
|
985
|
-
}
|
|
986
|
-
return a;
|
|
987
|
-
}
|
|
988
|
-
}
|
|
989
|
-
function _nonIterableRest() {
|
|
990
|
-
throw new TypeError("Invalid attempt to destructure non-iterable instance.\nIn order to be iterable, non-array objects must have a [Symbol.iterator]() method.");
|
|
991
|
-
}
|
|
992
|
-
function _slicedToArray(r, e) {
|
|
993
|
-
return _arrayWithHoles(r) || _iterableToArrayLimit(r, e) || _unsupportedIterableToArray(r, e) || _nonIterableRest();
|
|
994
|
-
}
|
|
995
|
-
function _unsupportedIterableToArray(r, a) {
|
|
996
|
-
if (r) {
|
|
997
|
-
if ("string" == typeof r) return _arrayLikeToArray(r, a);
|
|
998
|
-
var t = {}.toString.call(r).slice(8, -1);
|
|
999
|
-
return "Object" === t && r.constructor && (t = r.constructor.name), "Map" === t || "Set" === t ? Array.from(r) : "Arguments" === t || /^(?:Ui|I)nt(?:8|16|32)(?:Clamped)?Array$/.test(t) ? _arrayLikeToArray(r, a) : void 0;
|
|
1000
|
-
}
|
|
1001
|
-
}
|
|
1002
|
-
|
|
1003
|
-
const entries = Object.entries,
|
|
1004
|
-
setPrototypeOf = Object.setPrototypeOf,
|
|
1005
|
-
isFrozen = Object.isFrozen,
|
|
1006
|
-
getPrototypeOf = Object.getPrototypeOf,
|
|
1007
|
-
getOwnPropertyDescriptor = Object.getOwnPropertyDescriptor;
|
|
1008
|
-
let freeze = Object.freeze,
|
|
1009
|
-
seal = Object.seal,
|
|
1010
|
-
create = Object.create; // eslint-disable-line import/no-mutable-exports
|
|
1011
|
-
let _ref = typeof Reflect !== 'undefined' && Reflect,
|
|
1012
|
-
apply = _ref.apply,
|
|
1013
|
-
construct = _ref.construct;
|
|
949
|
+
`}strong({tokens:e}){return `<strong>${this.parser.parseInline(e)}</strong>`}em({tokens:e}){return `<em>${this.parser.parseInline(e)}</em>`}codespan({text:e}){return `<code>${O(e,true)}</code>`}br(e){return "<br>"}del({tokens:e}){return `<del>${this.parser.parseInline(e)}</del>`}link({href:e,title:t,tokens:n}){let s=this.parser.parseInline(n),r=J(e);if(r===null)return s;e=r;let i='<a href="'+e+'"';return t&&(i+=' title="'+O(t)+'"'),i+=">"+s+"</a>",i}image({href:e,title:t,text:n,tokens:s}){s&&(n=this.parser.parseInline(s,this.parser.textRenderer));let r=J(e);if(r===null)return O(n);e=r;let i=`<img src="${e}" alt="${O(n)}"`;return t&&(i+=` title="${O(t)}"`),i+=">",i}text(e){return "tokens"in e&&e.tokens?this.parser.parseInline(e.tokens):"escaped"in e&&e.escaped?e.text:O(e.text)}};var L=class{strong({text:e}){return e}em({text:e}){return e}codespan({text:e}){return e}del({text:e}){return e}html({text:e}){return e}text({text:e}){return e}link({text:e}){return ""+e}image({text:e}){return ""+e}br(){return ""}checkbox({raw:e}){return e}};var b=class l{options;renderer;textRenderer;constructor(e){this.options=e||T,this.options.renderer=this.options.renderer||new y,this.renderer=this.options.renderer,this.renderer.options=this.options,this.renderer.parser=this,this.textRenderer=new L;}static parse(e,t){return new l(t).parse(e)}static parseInline(e,t){return new l(t).parseInline(e)}parse(e){this.renderer.parser=this;let t="";for(let n=0;n<e.length;n++){let s=e[n];if(this.options.extensions?.renderers?.[s.type]){let i=s,o=this.options.extensions.renderers[i.type].call({parser:this},i);if(o!==false||!["space","hr","heading","code","table","blockquote","list","html","def","paragraph","text"].includes(i.type)){t+=o||"";continue}}let r=s;switch(r.type){case "space":{t+=this.renderer.space(r);break}case "hr":{t+=this.renderer.hr(r);break}case "heading":{t+=this.renderer.heading(r);break}case "code":{t+=this.renderer.code(r);break}case "table":{t+=this.renderer.table(r);break}case "blockquote":{t+=this.renderer.blockquote(r);break}case "list":{t+=this.renderer.list(r);break}case "checkbox":{t+=this.renderer.checkbox(r);break}case "html":{t+=this.renderer.html(r);break}case "def":{t+=this.renderer.def(r);break}case "paragraph":{t+=this.renderer.paragraph(r);break}case "text":{t+=this.renderer.text(r);break}default:{let i='Token with "'+r.type+'" type was not found.';if(this.options.silent)return console.error(i),"";throw new Error(i)}}}return t}parseInline(e,t=this.renderer){this.renderer.parser=this;let n="";for(let s=0;s<e.length;s++){let r=e[s];if(this.options.extensions?.renderers?.[r.type]){let o=this.options.extensions.renderers[r.type].call({parser:this},r);if(o!==false||!["escape","html","link","image","strong","em","codespan","br","del","text"].includes(r.type)){n+=o||"";continue}}let i=r;switch(i.type){case "escape":{n+=t.text(i);break}case "html":{n+=t.html(i);break}case "link":{n+=t.link(i);break}case "image":{n+=t.image(i);break}case "checkbox":{n+=t.checkbox(i);break}case "strong":{n+=t.strong(i);break}case "em":{n+=t.em(i);break}case "codespan":{n+=t.codespan(i);break}case "br":{n+=t.br(i);break}case "del":{n+=t.del(i);break}case "text":{n+=t.text(i);break}default:{let o='Token with "'+i.type+'" type was not found.';if(this.options.silent)return console.error(o),"";throw new Error(o)}}}return n}};var P=class{options;block;constructor(e){this.options=e||T;}static passThroughHooks=new Set(["preprocess","postprocess","processAllTokens","emStrongMask"]);static passThroughHooksRespectAsync=new Set(["preprocess","postprocess","processAllTokens"]);preprocess(e){return e}postprocess(e){return e}processAllTokens(e){return e}emStrongMask(e){return e}provideLexer(e=this.block){return e?x.lex:x.lexInline}provideParser(e=this.block){return e?b.parse:b.parseInline}};var D=class{defaults=z();options=this.setOptions;parse=this.parseMarkdown(true);parseInline=this.parseMarkdown(false);Parser=b;Renderer=y;TextRenderer=L;Lexer=x;Tokenizer=w;Hooks=P;constructor(...e){this.use(...e);}walkTokens(e,t){let n=[];for(let s of e)switch(n=n.concat(t.call(this,s)),s.type){case "table":{let r=s;for(let i of r.header)n=n.concat(this.walkTokens(i.tokens,t));for(let i of r.rows)for(let o of i)n=n.concat(this.walkTokens(o.tokens,t));break}case "list":{let r=s;n=n.concat(this.walkTokens(r.items,t));break}default:{let r=s;this.defaults.extensions?.childTokens?.[r.type]?this.defaults.extensions.childTokens[r.type].forEach(i=>{let o=r[i].flat(1/0);n=n.concat(this.walkTokens(o,t));}):r.tokens&&(n=n.concat(this.walkTokens(r.tokens,t)));}}return n}use(...e){let t=this.defaults.extensions||{renderers:{},childTokens:{}};return e.forEach(n=>{let s={...n};if(s.async=this.defaults.async||s.async||false,n.extensions&&(n.extensions.forEach(r=>{if(!r.name)throw new Error("extension name required");if("renderer"in r){let i=t.renderers[r.name];i?t.renderers[r.name]=function(...o){let u=r.renderer.apply(this,o);return u===false&&(u=i.apply(this,o)),u}:t.renderers[r.name]=r.renderer;}if("tokenizer"in r){if(!r.level||r.level!=="block"&&r.level!=="inline")throw new Error("extension level must be 'block' or 'inline'");let i=t[r.level];i?i.unshift(r.tokenizer):t[r.level]=[r.tokenizer],r.start&&(r.level==="block"?t.startBlock?t.startBlock.push(r.start):t.startBlock=[r.start]:r.level==="inline"&&(t.startInline?t.startInline.push(r.start):t.startInline=[r.start]));}"childTokens"in r&&r.childTokens&&(t.childTokens[r.name]=r.childTokens);}),s.extensions=t),n.renderer){let r=this.defaults.renderer||new y(this.defaults);for(let i in n.renderer){if(!(i in r))throw new Error(`renderer '${i}' does not exist`);if(["options","parser"].includes(i))continue;let o=i,u=n.renderer[o],a=r[o];r[o]=(...c)=>{let p=u.apply(r,c);return p===false&&(p=a.apply(r,c)),p||""};}s.renderer=r;}if(n.tokenizer){let r=this.defaults.tokenizer||new w(this.defaults);for(let i in n.tokenizer){if(!(i in r))throw new Error(`tokenizer '${i}' does not exist`);if(["options","rules","lexer"].includes(i))continue;let o=i,u=n.tokenizer[o],a=r[o];r[o]=(...c)=>{let p=u.apply(r,c);return p===false&&(p=a.apply(r,c)),p};}s.tokenizer=r;}if(n.hooks){let r=this.defaults.hooks||new P;for(let i in n.hooks){if(!(i in r))throw new Error(`hook '${i}' does not exist`);if(["options","block"].includes(i))continue;let o=i,u=n.hooks[o],a=r[o];P.passThroughHooks.has(i)?r[o]=c=>{if(this.defaults.async&&P.passThroughHooksRespectAsync.has(i))return (async()=>{let d=await u.call(r,c);return a.call(r,d)})();let p=u.call(r,c);return a.call(r,p)}:r[o]=(...c)=>{if(this.defaults.async)return (async()=>{let d=await u.apply(r,c);return d===false&&(d=await a.apply(r,c)),d})();let p=u.apply(r,c);return p===false&&(p=a.apply(r,c)),p};}s.hooks=r;}if(n.walkTokens){let r=this.defaults.walkTokens,i=n.walkTokens;s.walkTokens=function(o){let u=[];return u.push(i.call(this,o)),r&&(u=u.concat(r.call(this,o))),u};}this.defaults={...this.defaults,...s};}),this}setOptions(e){return this.defaults={...this.defaults,...e},this}lexer(e,t){return x.lex(e,t??this.defaults)}parser(e,t){return b.parse(e,t??this.defaults)}parseMarkdown(e){return (n,s)=>{let r={...s},i={...this.defaults,...r},o=this.onError(!!i.silent,!!i.async);if(this.defaults.async===true&&r.async===false)return o(new Error("marked(): The async option was set to true by an extension. Remove async: false from the parse options object to return a Promise."));if(typeof n>"u"||n===null)return o(new Error("marked(): input parameter is undefined or null"));if(typeof n!="string")return o(new Error("marked(): input parameter is of type "+Object.prototype.toString.call(n)+", string expected"));if(i.hooks&&(i.hooks.options=i,i.hooks.block=e),i.async)return (async()=>{let u=i.hooks?await i.hooks.preprocess(n):n,c=await(i.hooks?await i.hooks.provideLexer(e):e?x.lex:x.lexInline)(u,i),p=i.hooks?await i.hooks.processAllTokens(c):c;i.walkTokens&&await Promise.all(this.walkTokens(p,i.walkTokens));let h=await(i.hooks?await i.hooks.provideParser(e):e?b.parse:b.parseInline)(p,i);return i.hooks?await i.hooks.postprocess(h):h})().catch(o);try{i.hooks&&(n=i.hooks.preprocess(n));let a=(i.hooks?i.hooks.provideLexer(e):e?x.lex:x.lexInline)(n,i);i.hooks&&(a=i.hooks.processAllTokens(a)),i.walkTokens&&this.walkTokens(a,i.walkTokens);let p=(i.hooks?i.hooks.provideParser(e):e?b.parse:b.parseInline)(a,i);return i.hooks&&(p=i.hooks.postprocess(p)),p}catch(u){return o(u)}}}onError(e,t){return n=>{if(n.message+=`
|
|
950
|
+
Please report this to https://github.com/markedjs/marked.`,e){let s="<p>An error occurred:</p><pre>"+O(n.message+"",true)+"</pre>";return t?Promise.resolve(s):s}if(t)return Promise.reject(n);throw n}}};var M=new D;function g(l,e){return M.parse(l,e)}g.options=g.setOptions=function(l){return M.setOptions(l),g.defaults=M.defaults,G(g.defaults),g};g.getDefaults=z;g.defaults=T;g.use=function(...l){return M.use(...l),g.defaults=M.defaults,G(g.defaults),g};g.walkTokens=function(l,e){return M.walkTokens(l,e)};g.parseInline=M.parseInline;g.Parser=b;g.parser=b.parse;g.Renderer=y;g.TextRenderer=L;g.Lexer=x;g.lexer=x.lex;g.Tokenizer=w;g.Hooks=P;g.parse=g;g.options;g.setOptions;g.use;g.walkTokens;g.parseInline;b.parse;x.lex;
|
|
951
|
+
|
|
952
|
+
/*! @license DOMPurify 3.4.1 | (c) Cure53 and other contributors | Released under the Apache license 2.0 and Mozilla Public License 2.0 | github.com/cure53/DOMPurify/blob/3.4.1/LICENSE */
|
|
953
|
+
|
|
954
|
+
const {
|
|
955
|
+
entries,
|
|
956
|
+
setPrototypeOf,
|
|
957
|
+
isFrozen,
|
|
958
|
+
getPrototypeOf,
|
|
959
|
+
getOwnPropertyDescriptor
|
|
960
|
+
} = Object;
|
|
961
|
+
let {
|
|
962
|
+
freeze,
|
|
963
|
+
seal,
|
|
964
|
+
create
|
|
965
|
+
} = Object; // eslint-disable-line import/no-mutable-exports
|
|
966
|
+
let {
|
|
967
|
+
apply,
|
|
968
|
+
construct
|
|
969
|
+
} = typeof Reflect !== 'undefined' && Reflect;
|
|
1014
970
|
if (!freeze) {
|
|
1015
971
|
freeze = function freeze(x) {
|
|
1016
972
|
return x;
|
|
@@ -1147,10 +1103,7 @@ function cleanArray(array) {
|
|
|
1147
1103
|
*/
|
|
1148
1104
|
function clone(object) {
|
|
1149
1105
|
const newObject = create(null);
|
|
1150
|
-
for (const
|
|
1151
|
-
var _ref3 = _slicedToArray(_ref2, 2);
|
|
1152
|
-
const property = _ref3[0];
|
|
1153
|
-
const value = _ref3[1];
|
|
1106
|
+
for (const [property, value] of entries(object)) {
|
|
1154
1107
|
const isPropertyExist = objectHasOwnProperty(object, property);
|
|
1155
1108
|
if (isPropertyExist) {
|
|
1156
1109
|
if (arrayIsArray(value)) {
|
|
@@ -1264,14 +1217,15 @@ const mathMl$1 = freeze(['math', 'menclose', 'merror', 'mfenced', 'mfrac', 'mgly
|
|
|
1264
1217
|
const mathMlDisallowed = freeze(['maction', 'maligngroup', 'malignmark', 'mlongdiv', 'mscarries', 'mscarry', 'msgroup', 'mstack', 'msline', 'msrow', 'semantics', 'annotation', 'annotation-xml', 'mprescripts', 'none']);
|
|
1265
1218
|
const text = freeze(['#text']);
|
|
1266
1219
|
|
|
1267
|
-
const html = freeze(['accept', 'action', 'align', 'alt', 'autocapitalize', 'autocomplete', 'autopictureinpicture', 'autoplay', 'background', 'bgcolor', 'border', 'capture', 'cellpadding', 'cellspacing', 'checked', 'cite', 'class', 'clear', 'color', 'cols', 'colspan', '
|
|
1220
|
+
const html = freeze(['accept', 'action', 'align', 'alt', 'autocapitalize', 'autocomplete', 'autopictureinpicture', 'autoplay', 'background', 'bgcolor', 'border', 'capture', 'cellpadding', 'cellspacing', 'checked', 'cite', 'class', 'clear', 'color', 'cols', 'colspan', 'controls', 'controlslist', 'coords', 'crossorigin', 'datetime', 'decoding', 'default', 'dir', 'disabled', 'disablepictureinpicture', 'disableremoteplayback', 'download', 'draggable', 'enctype', 'enterkeyhint', 'exportparts', 'face', 'for', 'headers', 'height', 'hidden', 'high', 'href', 'hreflang', 'id', 'inert', 'inputmode', 'integrity', 'ismap', 'kind', 'label', 'lang', 'list', 'loading', 'loop', 'low', 'max', 'maxlength', 'media', 'method', 'min', 'minlength', 'multiple', 'muted', 'name', 'nonce', 'noshade', 'novalidate', 'nowrap', 'open', 'optimum', 'part', 'pattern', 'placeholder', 'playsinline', 'popover', 'popovertarget', 'popovertargetaction', 'poster', 'preload', 'pubdate', 'radiogroup', 'readonly', 'rel', 'required', 'rev', 'reversed', 'role', 'rows', 'rowspan', 'spellcheck', 'scope', 'selected', 'shape', 'size', 'sizes', 'slot', 'span', 'srclang', 'start', 'src', 'srcset', 'step', 'style', 'summary', 'tabindex', 'title', 'translate', 'type', 'usemap', 'valign', 'value', 'width', 'wrap', 'xmlns']);
|
|
1268
1221
|
const svg = freeze(['accent-height', 'accumulate', 'additive', 'alignment-baseline', 'amplitude', 'ascent', 'attributename', 'attributetype', 'azimuth', 'basefrequency', 'baseline-shift', 'begin', 'bias', 'by', 'class', 'clip', 'clippathunits', 'clip-path', 'clip-rule', 'color', 'color-interpolation', 'color-interpolation-filters', 'color-profile', 'color-rendering', 'cx', 'cy', 'd', 'dx', 'dy', 'diffuseconstant', 'direction', 'display', 'divisor', 'dur', 'edgemode', 'elevation', 'end', 'exponent', 'fill', 'fill-opacity', 'fill-rule', 'filter', 'filterunits', 'flood-color', 'flood-opacity', 'font-family', 'font-size', 'font-size-adjust', 'font-stretch', 'font-style', 'font-variant', 'font-weight', 'fx', 'fy', 'g1', 'g2', 'glyph-name', 'glyphref', 'gradientunits', 'gradienttransform', 'height', 'href', 'id', 'image-rendering', 'in', 'in2', 'intercept', 'k', 'k1', 'k2', 'k3', 'k4', 'kerning', 'keypoints', 'keysplines', 'keytimes', 'lang', 'lengthadjust', 'letter-spacing', 'kernelmatrix', 'kernelunitlength', 'lighting-color', 'local', 'marker-end', 'marker-mid', 'marker-start', 'markerheight', 'markerunits', 'markerwidth', 'maskcontentunits', 'maskunits', 'max', 'mask', 'mask-type', 'media', 'method', 'mode', 'min', 'name', 'numoctaves', 'offset', 'operator', 'opacity', 'order', 'orient', 'orientation', 'origin', 'overflow', 'paint-order', 'path', 'pathlength', 'patterncontentunits', 'patterntransform', 'patternunits', 'points', 'preservealpha', 'preserveaspectratio', 'primitiveunits', 'r', 'rx', 'ry', 'radius', 'refx', 'refy', 'repeatcount', 'repeatdur', 'restart', 'result', 'rotate', 'scale', 'seed', 'shape-rendering', 'slope', 'specularconstant', 'specularexponent', 'spreadmethod', 'startoffset', 'stddeviation', 'stitchtiles', 'stop-color', 'stop-opacity', 'stroke-dasharray', 'stroke-dashoffset', 'stroke-linecap', 'stroke-linejoin', 'stroke-miterlimit', 'stroke-opacity', 'stroke', 'stroke-width', 'style', 'surfacescale', 'systemlanguage', 'tabindex', 'tablevalues', 'targetx', 'targety', 'transform', 'transform-origin', 'text-anchor', 'text-decoration', 'text-rendering', 'textlength', 'type', 'u1', 'u2', 'unicode', 'values', 'viewbox', 'visibility', 'version', 'vert-adv-y', 'vert-origin-x', 'vert-origin-y', 'width', 'word-spacing', 'wrap', 'writing-mode', 'xchannelselector', 'ychannelselector', 'x', 'x1', 'x2', 'xmlns', 'y', 'y1', 'y2', 'z', 'zoomandpan']);
|
|
1269
1222
|
const mathMl = freeze(['accent', 'accentunder', 'align', 'bevelled', 'close', 'columnalign', 'columnlines', 'columnspacing', 'columnspan', 'denomalign', 'depth', 'dir', 'display', 'displaystyle', 'encoding', 'fence', 'frame', 'height', 'href', 'id', 'largeop', 'length', 'linethickness', 'lquote', 'lspace', 'mathbackground', 'mathcolor', 'mathsize', 'mathvariant', 'maxsize', 'minsize', 'movablelimits', 'notation', 'numalign', 'open', 'rowalign', 'rowlines', 'rowspacing', 'rowspan', 'rspace', 'rquote', 'scriptlevel', 'scriptminsize', 'scriptsizemultiplier', 'selection', 'separator', 'separators', 'stretchy', 'subscriptshift', 'supscriptshift', 'symmetric', 'voffset', 'width', 'xmlns']);
|
|
1270
1223
|
const xml = freeze(['xlink:href', 'xml:id', 'xlink:title', 'xml:space', 'xmlns:xlink']);
|
|
1271
1224
|
|
|
1272
|
-
|
|
1273
|
-
const
|
|
1274
|
-
const
|
|
1225
|
+
// eslint-disable-next-line unicorn/better-regex
|
|
1226
|
+
const MUSTACHE_EXPR = seal(/\{\{[\w\W]*|[\w\W]*\}\}/gm); // Specify template detection regex for SAFE_FOR_TEMPLATES mode
|
|
1227
|
+
const ERB_EXPR = seal(/<%[\w\W]*|[\w\W]*%>/gm);
|
|
1228
|
+
const TMPLIT_EXPR = seal(/\$\{[\w\W]*/gm); // eslint-disable-line unicorn/better-regex
|
|
1275
1229
|
const DATA_ATTR = seal(/^data-[\-\w.\u00B7-\uFFFF]+$/); // eslint-disable-line no-useless-escape
|
|
1276
1230
|
const ARIA_ATTR = seal(/^aria-[\-\w]+$/); // eslint-disable-line no-useless-escape
|
|
1277
1231
|
const IS_ALLOWED_URI = seal(/^(?:(?:(?:f|ht)tps?|mailto|tel|callto|sms|cid|xmpp|matrix):|[^a-z]|[a-z+.\-]+(?:[^a-z+.\-:]|$))/i // eslint-disable-line no-useless-escape
|
|
@@ -1282,24 +1236,29 @@ const ATTR_WHITESPACE = seal(/[\u0000-\u0020\u00A0\u1680\u180E\u2000-\u2029\u205
|
|
|
1282
1236
|
const DOCTYPE_NAME = seal(/^html$/i);
|
|
1283
1237
|
const CUSTOM_ELEMENT = seal(/^[a-z][.\w]*(-[.\w]+)+$/i);
|
|
1284
1238
|
|
|
1239
|
+
var EXPRESSIONS = /*#__PURE__*/Object.freeze({
|
|
1240
|
+
__proto__: null,
|
|
1241
|
+
ARIA_ATTR: ARIA_ATTR,
|
|
1242
|
+
ATTR_WHITESPACE: ATTR_WHITESPACE,
|
|
1243
|
+
CUSTOM_ELEMENT: CUSTOM_ELEMENT,
|
|
1244
|
+
DATA_ATTR: DATA_ATTR,
|
|
1245
|
+
DOCTYPE_NAME: DOCTYPE_NAME,
|
|
1246
|
+
ERB_EXPR: ERB_EXPR,
|
|
1247
|
+
IS_ALLOWED_URI: IS_ALLOWED_URI,
|
|
1248
|
+
IS_SCRIPT_OR_DATA: IS_SCRIPT_OR_DATA,
|
|
1249
|
+
MUSTACHE_EXPR: MUSTACHE_EXPR,
|
|
1250
|
+
TMPLIT_EXPR: TMPLIT_EXPR
|
|
1251
|
+
});
|
|
1252
|
+
|
|
1285
1253
|
/* eslint-disable @typescript-eslint/indent */
|
|
1286
1254
|
// https://developer.mozilla.org/en-US/docs/Web/API/Node/nodeType
|
|
1287
1255
|
const NODE_TYPE = {
|
|
1288
1256
|
element: 1,
|
|
1289
|
-
attribute: 2,
|
|
1290
1257
|
text: 3,
|
|
1291
|
-
cdataSection: 4,
|
|
1292
|
-
entityReference: 5,
|
|
1293
|
-
// Deprecated
|
|
1294
|
-
entityNode: 6,
|
|
1295
1258
|
// Deprecated
|
|
1296
1259
|
progressingInstruction: 7,
|
|
1297
1260
|
comment: 8,
|
|
1298
|
-
document: 9
|
|
1299
|
-
documentType: 10,
|
|
1300
|
-
documentFragment: 11,
|
|
1301
|
-
notation: 12 // Deprecated
|
|
1302
|
-
};
|
|
1261
|
+
document: 9};
|
|
1303
1262
|
const getGlobal = function getGlobal() {
|
|
1304
1263
|
return typeof window === 'undefined' ? null : window;
|
|
1305
1264
|
};
|
|
@@ -1357,7 +1316,7 @@ const _createHooksMap = function _createHooksMap() {
|
|
|
1357
1316
|
function createDOMPurify() {
|
|
1358
1317
|
let window = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : getGlobal();
|
|
1359
1318
|
const DOMPurify = root => createDOMPurify(root);
|
|
1360
|
-
DOMPurify.version = '3.4.
|
|
1319
|
+
DOMPurify.version = '3.4.1';
|
|
1361
1320
|
DOMPurify.removed = [];
|
|
1362
1321
|
if (!window || !window.document || window.document.nodeType !== NODE_TYPE.document || !window.Element) {
|
|
1363
1322
|
// Not running in a browser, provide a factory function
|
|
@@ -1365,29 +1324,28 @@ function createDOMPurify() {
|
|
|
1365
1324
|
DOMPurify.isSupported = false;
|
|
1366
1325
|
return DOMPurify;
|
|
1367
1326
|
}
|
|
1368
|
-
let
|
|
1327
|
+
let {
|
|
1328
|
+
document
|
|
1329
|
+
} = window;
|
|
1369
1330
|
const originalDocument = document;
|
|
1370
1331
|
const currentScript = originalDocument.currentScript;
|
|
1371
|
-
|
|
1372
|
-
|
|
1373
|
-
|
|
1374
|
-
|
|
1375
|
-
|
|
1376
|
-
|
|
1377
|
-
|
|
1378
|
-
|
|
1379
|
-
|
|
1380
|
-
trustedTypes
|
|
1332
|
+
const {
|
|
1333
|
+
DocumentFragment,
|
|
1334
|
+
HTMLTemplateElement,
|
|
1335
|
+
Node,
|
|
1336
|
+
Element,
|
|
1337
|
+
NodeFilter,
|
|
1338
|
+
NamedNodeMap = window.NamedNodeMap || window.MozNamedAttrMap,
|
|
1339
|
+
HTMLFormElement,
|
|
1340
|
+
DOMParser,
|
|
1341
|
+
trustedTypes
|
|
1342
|
+
} = window;
|
|
1381
1343
|
const ElementPrototype = Element.prototype;
|
|
1382
1344
|
const cloneNode = lookupGetter(ElementPrototype, 'cloneNode');
|
|
1383
1345
|
const remove = lookupGetter(ElementPrototype, 'remove');
|
|
1384
1346
|
const getNextSibling = lookupGetter(ElementPrototype, 'nextSibling');
|
|
1385
1347
|
const getChildNodes = lookupGetter(ElementPrototype, 'childNodes');
|
|
1386
1348
|
const getParentNode = lookupGetter(ElementPrototype, 'parentNode');
|
|
1387
|
-
const getShadowRoot = lookupGetter(ElementPrototype, 'shadowRoot');
|
|
1388
|
-
const getAttributes = lookupGetter(ElementPrototype, 'attributes');
|
|
1389
|
-
const getNodeType = Node && Node.prototype ? lookupGetter(Node.prototype, 'nodeType') : null;
|
|
1390
|
-
const getNodeName = Node && Node.prototype ? lookupGetter(Node.prototype, 'nodeName') : null;
|
|
1391
1349
|
// As per issue #47, the web-components registry is inherited by a
|
|
1392
1350
|
// new document created via createHTMLDocument. As per the spec
|
|
1393
1351
|
// (http://w3c.github.io/webcomponents/spec/custom/#creating-and-passing-registries)
|
|
@@ -1402,43 +1360,33 @@ function createDOMPurify() {
|
|
|
1402
1360
|
}
|
|
1403
1361
|
let trustedTypesPolicy;
|
|
1404
1362
|
let emptyHTML = '';
|
|
1405
|
-
|
|
1406
|
-
|
|
1407
|
-
|
|
1408
|
-
|
|
1409
|
-
|
|
1410
|
-
|
|
1411
|
-
const
|
|
1412
|
-
|
|
1413
|
-
|
|
1414
|
-
}
|
|
1415
|
-
IN_POLICY_CREATE_HTML++;
|
|
1416
|
-
try {
|
|
1417
|
-
return trustedTypesPolicy.createHTML(html);
|
|
1418
|
-
} finally {
|
|
1419
|
-
IN_POLICY_CREATE_HTML--;
|
|
1420
|
-
}
|
|
1421
|
-
};
|
|
1422
|
-
const _document = document,
|
|
1423
|
-
implementation = _document.implementation,
|
|
1424
|
-
createNodeIterator = _document.createNodeIterator,
|
|
1425
|
-
createDocumentFragment = _document.createDocumentFragment,
|
|
1426
|
-
getElementsByTagName = _document.getElementsByTagName;
|
|
1427
|
-
const importNode = originalDocument.importNode;
|
|
1363
|
+
const {
|
|
1364
|
+
implementation,
|
|
1365
|
+
createNodeIterator,
|
|
1366
|
+
createDocumentFragment,
|
|
1367
|
+
getElementsByTagName
|
|
1368
|
+
} = document;
|
|
1369
|
+
const {
|
|
1370
|
+
importNode
|
|
1371
|
+
} = originalDocument;
|
|
1428
1372
|
let hooks = _createHooksMap();
|
|
1429
1373
|
/**
|
|
1430
1374
|
* Expose whether this browser supports running the full DOMPurify.
|
|
1431
1375
|
*/
|
|
1432
1376
|
DOMPurify.isSupported = typeof entries === 'function' && typeof getParentNode === 'function' && implementation && implementation.createHTMLDocument !== undefined;
|
|
1433
|
-
const
|
|
1434
|
-
|
|
1435
|
-
|
|
1436
|
-
|
|
1437
|
-
|
|
1438
|
-
|
|
1439
|
-
|
|
1440
|
-
|
|
1441
|
-
|
|
1377
|
+
const {
|
|
1378
|
+
MUSTACHE_EXPR,
|
|
1379
|
+
ERB_EXPR,
|
|
1380
|
+
TMPLIT_EXPR,
|
|
1381
|
+
DATA_ATTR,
|
|
1382
|
+
ARIA_ATTR,
|
|
1383
|
+
IS_SCRIPT_OR_DATA,
|
|
1384
|
+
ATTR_WHITESPACE,
|
|
1385
|
+
CUSTOM_ELEMENT
|
|
1386
|
+
} = EXPRESSIONS;
|
|
1387
|
+
let {
|
|
1388
|
+
IS_ALLOWED_URI: IS_ALLOWED_URI$1
|
|
1389
|
+
} = EXPRESSIONS;
|
|
1442
1390
|
/**
|
|
1443
1391
|
* We consider the elements and attributes below to be safe. Ideally
|
|
1444
1392
|
* don't add any new ones but feel free to remove unwanted ones.
|
|
@@ -1746,47 +1694,19 @@ function createDOMPurify() {
|
|
|
1746
1694
|
throw typeErrorCreate('TRUSTED_TYPES_POLICY configuration option must provide a "createScriptURL" hook.');
|
|
1747
1695
|
}
|
|
1748
1696
|
// Overwrite existing TrustedTypes policy.
|
|
1749
|
-
const previousTrustedTypesPolicy = trustedTypesPolicy;
|
|
1750
1697
|
trustedTypesPolicy = cfg.TRUSTED_TYPES_POLICY;
|
|
1751
|
-
// Sign local variables required by `sanitize`.
|
|
1752
|
-
|
|
1753
|
-
// throws via the re-entrancy guard. Restore the previous policy first so
|
|
1754
|
-
// the instance is not left in a poisoned state. See #1422.
|
|
1755
|
-
try {
|
|
1756
|
-
emptyHTML = _createTrustedHTML('');
|
|
1757
|
-
} catch (error) {
|
|
1758
|
-
trustedTypesPolicy = previousTrustedTypesPolicy;
|
|
1759
|
-
throw error;
|
|
1760
|
-
}
|
|
1698
|
+
// Sign local variables required by `sanitize`.
|
|
1699
|
+
emptyHTML = trustedTypesPolicy.createHTML('');
|
|
1761
1700
|
} else {
|
|
1762
1701
|
// Uninitialized policy, attempt to initialize the internal dompurify policy.
|
|
1763
|
-
if (trustedTypesPolicy === undefined
|
|
1702
|
+
if (trustedTypesPolicy === undefined) {
|
|
1764
1703
|
trustedTypesPolicy = _createTrustedTypesPolicy(trustedTypes, currentScript);
|
|
1765
1704
|
}
|
|
1766
1705
|
// If creating the internal policy succeeded sign internal variables.
|
|
1767
|
-
|
|
1768
|
-
|
|
1769
|
-
// policy has been initialized yet) must be excluded here, otherwise we
|
|
1770
|
-
// would call `.createHTML` on a non-policy and throw. See #1422.
|
|
1771
|
-
if (trustedTypesPolicy && typeof emptyHTML === 'string') {
|
|
1772
|
-
emptyHTML = _createTrustedHTML('');
|
|
1706
|
+
if (trustedTypesPolicy !== null && typeof emptyHTML === 'string') {
|
|
1707
|
+
emptyHTML = trustedTypesPolicy.createHTML('');
|
|
1773
1708
|
}
|
|
1774
1709
|
}
|
|
1775
|
-
/*
|
|
1776
|
-
* Mirror the clone-before-mutate pattern already applied above for
|
|
1777
|
-
* cfg.ADD_TAGS / cfg.ADD_ATTR: if any uponSanitize* hook is
|
|
1778
|
-
* registered AND the set still points at the default constant,
|
|
1779
|
-
* clone it. The hook then mutates the clone (in-call widening
|
|
1780
|
-
* still works exactly as documented) and the next default-cfg
|
|
1781
|
-
* call rebinds to the untouched original via the reassignment at
|
|
1782
|
-
* the top of this function.
|
|
1783
|
-
*/
|
|
1784
|
-
if ((hooks.uponSanitizeElement.length > 0 || hooks.uponSanitizeAttribute.length > 0) && ALLOWED_TAGS === DEFAULT_ALLOWED_TAGS) {
|
|
1785
|
-
ALLOWED_TAGS = clone(ALLOWED_TAGS);
|
|
1786
|
-
}
|
|
1787
|
-
if (hooks.uponSanitizeAttribute.length > 0 && ALLOWED_ATTR === DEFAULT_ALLOWED_ATTR) {
|
|
1788
|
-
ALLOWED_ATTR = clone(ALLOWED_ATTR);
|
|
1789
|
-
}
|
|
1790
1710
|
// Prevent further manipulation of configuration.
|
|
1791
1711
|
// Not available in IE8, Safari 5, etc.
|
|
1792
1712
|
if (freeze) {
|
|
@@ -1946,7 +1866,7 @@ function createDOMPurify() {
|
|
|
1946
1866
|
// Root of XHTML doc must contain xmlns declaration (see https://www.w3.org/TR/xhtml1/normative.html#strict)
|
|
1947
1867
|
dirty = '<html xmlns="http://www.w3.org/1999/xhtml"><head></head><body>' + dirty + '</body></html>';
|
|
1948
1868
|
}
|
|
1949
|
-
const dirtyPayload = trustedTypesPolicy ?
|
|
1869
|
+
const dirtyPayload = trustedTypesPolicy ? trustedTypesPolicy.createHTML(dirty) : dirty;
|
|
1950
1870
|
/*
|
|
1951
1871
|
* Use the DOMParser API by default, fallback later if needs be
|
|
1952
1872
|
* DOMParser not work for svg when has multiple root element.
|
|
@@ -1986,142 +1906,23 @@ function createDOMPurify() {
|
|
|
1986
1906
|
// eslint-disable-next-line no-bitwise
|
|
1987
1907
|
NodeFilter.SHOW_ELEMENT | NodeFilter.SHOW_COMMENT | NodeFilter.SHOW_TEXT | NodeFilter.SHOW_PROCESSING_INSTRUCTION | NodeFilter.SHOW_CDATA_SECTION, null);
|
|
1988
1908
|
};
|
|
1989
|
-
/**
|
|
1990
|
-
* Strip template-engine expressions ({{...}}, ${...}, <%...%>) from the
|
|
1991
|
-
* character data of an element subtree. Used as the final safety net for
|
|
1992
|
-
* SAFE_FOR_TEMPLATES on every DOM-returning code path so that expressions
|
|
1993
|
-
* which only form after text-node normalization (e.g. fragments split across
|
|
1994
|
-
* stripped elements) cannot survive into a template-evaluating framework.
|
|
1995
|
-
*
|
|
1996
|
-
* Walks text/comment/CDATA/processing-instruction nodes and mutates `.data`
|
|
1997
|
-
* in place rather than round-tripping through innerHTML. This preserves
|
|
1998
|
-
* descendant node references (important for IN_PLACE callers), avoids a
|
|
1999
|
-
* serialize/reparse cycle, and reads literal character data — which means
|
|
2000
|
-
* `<%...%>` in text content matches the ERB regex against its real bytes
|
|
2001
|
-
* instead of the HTML-entity-escaped form innerHTML would produce.
|
|
2002
|
-
*
|
|
2003
|
-
* Attribute values are not visited here; SAFE_FOR_TEMPLATES handling for
|
|
2004
|
-
* attributes is performed during the per-node `_sanitizeAttributes` pass.
|
|
2005
|
-
*
|
|
2006
|
-
* @param node The root element whose character data should be scrubbed.
|
|
2007
|
-
*/
|
|
2008
|
-
const _scrubTemplateExpressions2 = function _scrubTemplateExpressions(node) {
|
|
2009
|
-
var _node$querySelectorAl, _node$querySelectorAl2;
|
|
2010
|
-
node.normalize();
|
|
2011
|
-
const walker = createNodeIterator.call(node.ownerDocument || node, node,
|
|
2012
|
-
// eslint-disable-next-line no-bitwise
|
|
2013
|
-
NodeFilter.SHOW_TEXT | NodeFilter.SHOW_COMMENT | NodeFilter.SHOW_CDATA_SECTION | NodeFilter.SHOW_PROCESSING_INSTRUCTION, null);
|
|
2014
|
-
let currentNode = walker.nextNode();
|
|
2015
|
-
while (currentNode) {
|
|
2016
|
-
let data = currentNode.data;
|
|
2017
|
-
arrayForEach([MUSTACHE_EXPR$1, ERB_EXPR$1, TMPLIT_EXPR$1], expr => {
|
|
2018
|
-
data = stringReplace(data, expr, ' ');
|
|
2019
|
-
});
|
|
2020
|
-
currentNode.data = data;
|
|
2021
|
-
currentNode = walker.nextNode();
|
|
2022
|
-
}
|
|
2023
|
-
// NodeIterator does not descend into <template>.content per the DOM spec,
|
|
2024
|
-
// so we must explicitly recurse into each template's content fragment,
|
|
2025
|
-
// mirroring the approach used by _sanitizeShadowDOM.
|
|
2026
|
-
const templates = (_node$querySelectorAl = (_node$querySelectorAl2 = node.querySelectorAll) === null || _node$querySelectorAl2 === void 0 ? void 0 : _node$querySelectorAl2.call(node, 'template')) !== null && _node$querySelectorAl !== void 0 ? _node$querySelectorAl : [];
|
|
2027
|
-
arrayForEach(Array.from(templates), tmpl => {
|
|
2028
|
-
if (_isDocumentFragment(tmpl.content)) {
|
|
2029
|
-
_scrubTemplateExpressions2(tmpl.content);
|
|
2030
|
-
}
|
|
2031
|
-
});
|
|
2032
|
-
};
|
|
2033
1909
|
/**
|
|
2034
1910
|
* _isClobbered
|
|
2035
1911
|
*
|
|
2036
|
-
* Detect DOM-clobbering on HTMLFormElement nodes. Form is the only HTML
|
|
2037
|
-
* interface with [LegacyOverrideBuiltIns]; a descendant element with a
|
|
2038
|
-
* `name` attribute matching a prototype property shadows that property
|
|
2039
|
-
* on direct reads. We use this check at the IN_PLACE entry-point and
|
|
2040
|
-
* during attribute sanitization to refuse clobbered forms.
|
|
2041
|
-
*
|
|
2042
1912
|
* @param element element to check for clobbering attacks
|
|
2043
1913
|
* @return true if clobbered, false if safe
|
|
2044
1914
|
*/
|
|
2045
1915
|
const _isClobbered = function _isClobbered(element) {
|
|
2046
|
-
|
|
2047
|
-
// name at all, we can't reason about clobbering — return false
|
|
2048
|
-
// (the caller's other defences still apply).
|
|
2049
|
-
const realTagName = getNodeName ? getNodeName(element) : null;
|
|
2050
|
-
if (typeof realTagName !== 'string') {
|
|
2051
|
-
return false;
|
|
2052
|
-
}
|
|
2053
|
-
if (transformCaseFunc(realTagName) !== 'form') {
|
|
2054
|
-
return false;
|
|
2055
|
-
}
|
|
2056
|
-
return typeof element.nodeName !== 'string' || typeof element.textContent !== 'string' || typeof element.removeChild !== 'function' ||
|
|
2057
|
-
// Realm-safe NamedNodeMap detection: equality against the cached
|
|
2058
|
-
// prototype getter. Clobbered .attributes (e.g. <input name="attributes">)
|
|
2059
|
-
// makes the direct read diverge from the cached read; a clean form
|
|
2060
|
-
// (same-realm OR foreign-realm) has both reads pointing at the same
|
|
2061
|
-
// canonical NamedNodeMap.
|
|
2062
|
-
element.attributes !== getAttributes(element) || typeof element.removeAttribute !== 'function' || typeof element.setAttribute !== 'function' || typeof element.namespaceURI !== 'string' || typeof element.insertBefore !== 'function' || typeof element.hasChildNodes !== 'function' ||
|
|
2063
|
-
// NodeType clobbering probe. Cached Node.prototype.nodeType getter
|
|
2064
|
-
// returns the integer 1 for any Element regardless of realm; direct
|
|
2065
|
-
// read on a clobbered form (e.g. <input name="nodeType">) returns
|
|
2066
|
-
// the named child element. Cheap addition — nodeType is read from
|
|
2067
|
-
// an internal slot, no serialization cost — and removes a residual
|
|
2068
|
-
// clobbering surface used by several mXSS / PI / comment branches
|
|
2069
|
-
// in _sanitizeElements that compare currentNode.nodeType directly.
|
|
2070
|
-
element.nodeType !== getNodeType(element) ||
|
|
2071
|
-
// HTMLFormElement has [LegacyOverrideBuiltIns]: a descendant named
|
|
2072
|
-
// "childNodes" shadows the prototype getter. Direct reads of
|
|
2073
|
-
// form.childNodes from a clobbered form return the named child
|
|
2074
|
-
// instead of the real NodeList, so any walk that reads it directly
|
|
2075
|
-
// skips the form's real children. Compare the direct read to the
|
|
2076
|
-
// cached Node.prototype getter — when the form's named-property
|
|
2077
|
-
// getter intercepts the read, the two values differ and we flag
|
|
2078
|
-
// the form. This catches every clobbering child type (input,
|
|
2079
|
-
// select, etc.) regardless of whether the named child happens to
|
|
2080
|
-
// carry a numeric .length, which a typeof-based probe would miss
|
|
2081
|
-
// (e.g. HTMLSelectElement.length is a defined unsigned-long).
|
|
2082
|
-
element.childNodes !== getChildNodes(element);
|
|
2083
|
-
};
|
|
2084
|
-
/**
|
|
2085
|
-
* Checks whether the given value is a DocumentFragment from any realm.
|
|
2086
|
-
*
|
|
2087
|
-
* The realm-independent replacement reads `nodeType` through the cached
|
|
2088
|
-
* Node.prototype getter and compares to the DOCUMENT_FRAGMENT_NODE
|
|
2089
|
-
* constant (11). nodeType is a numeric value resolved from the node's
|
|
2090
|
-
* internal slot, identical across realms for the same kind of node.
|
|
2091
|
-
*
|
|
2092
|
-
* @param value object to check
|
|
2093
|
-
* @return true if value is a DocumentFragment-shaped node from any realm
|
|
2094
|
-
*/
|
|
2095
|
-
const _isDocumentFragment = function _isDocumentFragment(value) {
|
|
2096
|
-
if (!getNodeType || typeof value !== 'object' || value === null) {
|
|
2097
|
-
return false;
|
|
2098
|
-
}
|
|
2099
|
-
try {
|
|
2100
|
-
return getNodeType(value) === NODE_TYPE.documentFragment;
|
|
2101
|
-
} catch (_) {
|
|
2102
|
-
return false;
|
|
2103
|
-
}
|
|
1916
|
+
return element instanceof HTMLFormElement && (typeof element.nodeName !== 'string' || typeof element.textContent !== 'string' || typeof element.removeChild !== 'function' || !(element.attributes instanceof NamedNodeMap) || typeof element.removeAttribute !== 'function' || typeof element.setAttribute !== 'function' || typeof element.namespaceURI !== 'string' || typeof element.insertBefore !== 'function' || typeof element.hasChildNodes !== 'function');
|
|
2104
1917
|
};
|
|
2105
1918
|
/**
|
|
2106
|
-
* Checks whether the given object is a DOM node
|
|
2107
|
-
* originate from a different window/realm (e.g. an iframe's
|
|
2108
|
-
* contentDocument). The previous `value instanceof Node` check was
|
|
2109
|
-
* realm-bound: nodes from a different window failed it, causing
|
|
2110
|
-
* sanitize() to silently stringify them and reset IN_PLACE to false,
|
|
2111
|
-
* returning the original node unsanitized. See GHSA-4w3q-35jp-p934.
|
|
1919
|
+
* Checks whether the given object is a DOM node.
|
|
2112
1920
|
*
|
|
2113
1921
|
* @param value object to check whether it's a DOM node
|
|
2114
|
-
* @return true
|
|
1922
|
+
* @return true is object is a DOM node
|
|
2115
1923
|
*/
|
|
2116
1924
|
const _isNode = function _isNode(value) {
|
|
2117
|
-
|
|
2118
|
-
return false;
|
|
2119
|
-
}
|
|
2120
|
-
try {
|
|
2121
|
-
return typeof getNodeType(value) === 'number';
|
|
2122
|
-
} catch (_) {
|
|
2123
|
-
return false;
|
|
2124
|
-
}
|
|
1925
|
+
return typeof Node === 'function' && value instanceof Node;
|
|
2125
1926
|
};
|
|
2126
1927
|
function _executeHooks(hooks, currentNode, data) {
|
|
2127
1928
|
arrayForEach(hooks, hook => {
|
|
@@ -2147,7 +1948,7 @@ function createDOMPurify() {
|
|
|
2147
1948
|
return true;
|
|
2148
1949
|
}
|
|
2149
1950
|
/* Now let's check the element's type and name */
|
|
2150
|
-
const tagName = transformCaseFunc(
|
|
1951
|
+
const tagName = transformCaseFunc(currentNode.nodeName);
|
|
2151
1952
|
/* Execute a hook if present */
|
|
2152
1953
|
_executeHooks(hooks.uponSanitizeElement, currentNode, {
|
|
2153
1954
|
tagName,
|
|
@@ -2184,17 +1985,10 @@ function createDOMPurify() {
|
|
|
2184
1985
|
return false;
|
|
2185
1986
|
}
|
|
2186
1987
|
}
|
|
2187
|
-
/* Keep content except for bad-listed elements
|
|
2188
|
-
Use the cached prototype getters exclusively — the previous code
|
|
2189
|
-
had `|| currentNode.parentNode` / `|| currentNode.childNodes`
|
|
2190
|
-
fallbacks, but the cached getters always return the canonical
|
|
2191
|
-
value (or null for a real parent-less node), so the fallback
|
|
2192
|
-
path was dead in safe cases and a clobbering surface in unsafe
|
|
2193
|
-
ones. Falsy cached results stay falsy; the `if (childNodes &&
|
|
2194
|
-
parentNode)` check already gates correctly. */
|
|
1988
|
+
/* Keep content except for bad-listed elements */
|
|
2195
1989
|
if (KEEP_CONTENT && !FORBID_CONTENTS[tagName]) {
|
|
2196
|
-
const parentNode = getParentNode(currentNode);
|
|
2197
|
-
const childNodes = getChildNodes(currentNode);
|
|
1990
|
+
const parentNode = getParentNode(currentNode) || currentNode.parentNode;
|
|
1991
|
+
const childNodes = getChildNodes(currentNode) || currentNode.childNodes;
|
|
2198
1992
|
if (childNodes && parentNode) {
|
|
2199
1993
|
const childCount = childNodes.length;
|
|
2200
1994
|
for (let i = childCount - 1; i >= 0; --i) {
|
|
@@ -2206,14 +2000,8 @@ function createDOMPurify() {
|
|
|
2206
2000
|
_forceRemove(currentNode);
|
|
2207
2001
|
return true;
|
|
2208
2002
|
}
|
|
2209
|
-
/* Check whether element has a valid namespace
|
|
2210
|
-
|
|
2211
|
-
nodeType getter rather than `instanceof Element`, which is realm-
|
|
2212
|
-
bound and short-circuits to false for any node minted in a different
|
|
2213
|
-
realm — letting a foreign-realm element with a forbidden namespace
|
|
2214
|
-
slip past the namespace check entirely. */
|
|
2215
|
-
const nt = getNodeType ? getNodeType(currentNode) : currentNode.nodeType;
|
|
2216
|
-
if (nt === NODE_TYPE.element && !_checkValidNamespace(currentNode)) {
|
|
2003
|
+
/* Check whether element has a valid namespace */
|
|
2004
|
+
if (currentNode instanceof Element && !_checkValidNamespace(currentNode)) {
|
|
2217
2005
|
_forceRemove(currentNode);
|
|
2218
2006
|
return true;
|
|
2219
2007
|
}
|
|
@@ -2226,7 +2014,7 @@ function createDOMPurify() {
|
|
|
2226
2014
|
if (SAFE_FOR_TEMPLATES && currentNode.nodeType === NODE_TYPE.text) {
|
|
2227
2015
|
/* Get the element's text content */
|
|
2228
2016
|
content = currentNode.textContent;
|
|
2229
|
-
arrayForEach([MUSTACHE_EXPR
|
|
2017
|
+
arrayForEach([MUSTACHE_EXPR, ERB_EXPR, TMPLIT_EXPR], expr => {
|
|
2230
2018
|
content = stringReplace(content, expr, ' ');
|
|
2231
2019
|
});
|
|
2232
2020
|
if (currentNode.textContent !== content) {
|
|
@@ -2258,12 +2046,11 @@ function createDOMPurify() {
|
|
|
2258
2046
|
if (SANITIZE_DOM && (lcName === 'id' || lcName === 'name') && (value in document || value in formElement)) {
|
|
2259
2047
|
return false;
|
|
2260
2048
|
}
|
|
2261
|
-
const nameIsPermitted = ALLOWED_ATTR[lcName] || EXTRA_ELEMENT_HANDLING.attributeCheck instanceof Function && EXTRA_ELEMENT_HANDLING.attributeCheck(lcName, lcTag);
|
|
2262
2049
|
/* Allow valid data-* attributes: At least one character after "-"
|
|
2263
2050
|
(https://html.spec.whatwg.org/multipage/dom.html#embedding-custom-non-visible-data-with-the-data-*-attributes)
|
|
2264
2051
|
XML-compatible (https://html.spec.whatwg.org/multipage/infrastructure.html#xml-compatible and http://www.w3.org/TR/xml/#d0e804)
|
|
2265
2052
|
We don't need to check the value; it's always URI safe. */
|
|
2266
|
-
if (ALLOW_DATA_ATTR && !FORBID_ATTR[lcName] && regExpTest(DATA_ATTR
|
|
2053
|
+
if (ALLOW_DATA_ATTR && !FORBID_ATTR[lcName] && regExpTest(DATA_ATTR, lcName)) ; else if (ALLOW_ARIA_ATTR && regExpTest(ARIA_ATTR, lcName)) ; else if (EXTRA_ELEMENT_HANDLING.attributeCheck instanceof Function && EXTRA_ELEMENT_HANDLING.attributeCheck(lcName, lcTag)) ; else if (!ALLOWED_ATTR[lcName] || FORBID_ATTR[lcName]) {
|
|
2267
2054
|
if (
|
|
2268
2055
|
// First condition does a very basic check if a) it's basically a valid custom element tagname AND
|
|
2269
2056
|
// b) if the tagName passes whatever the user has configured for CUSTOM_ELEMENT_HANDLING.tagNameCheck
|
|
@@ -2275,7 +2062,7 @@ function createDOMPurify() {
|
|
|
2275
2062
|
return false;
|
|
2276
2063
|
}
|
|
2277
2064
|
/* Check value is safe. First, is attr inert? If so, is safe */
|
|
2278
|
-
} else if (URI_SAFE_ATTRIBUTES[lcName]) ; else if (regExpTest(IS_ALLOWED_URI$1, stringReplace(value, ATTR_WHITESPACE
|
|
2065
|
+
} else if (URI_SAFE_ATTRIBUTES[lcName]) ; else if (regExpTest(IS_ALLOWED_URI$1, stringReplace(value, ATTR_WHITESPACE, ''))) ; else if ((lcName === 'src' || lcName === 'xlink:href' || lcName === 'href') && lcTag !== 'script' && stringIndexOf(value, 'data:') === 0 && DATA_URI_TAGS[lcTag]) ; else if (ALLOW_UNKNOWN_PROTOCOLS && !regExpTest(IS_SCRIPT_OR_DATA, stringReplace(value, ATTR_WHITESPACE, ''))) ; else if (value) {
|
|
2279
2066
|
return false;
|
|
2280
2067
|
} else ;
|
|
2281
2068
|
return true;
|
|
@@ -2293,7 +2080,7 @@ function createDOMPurify() {
|
|
|
2293
2080
|
* @returns Returns true if the tag name meets the basic criteria for a custom element, otherwise false.
|
|
2294
2081
|
*/
|
|
2295
2082
|
const _isBasicCustomElement = function _isBasicCustomElement(tagName) {
|
|
2296
|
-
return !RESERVED_CUSTOM_ELEMENT_NAMES[stringToLowerCase(tagName)] && regExpTest(CUSTOM_ELEMENT
|
|
2083
|
+
return !RESERVED_CUSTOM_ELEMENT_NAMES[stringToLowerCase(tagName)] && regExpTest(CUSTOM_ELEMENT, tagName);
|
|
2297
2084
|
};
|
|
2298
2085
|
/**
|
|
2299
2086
|
* _sanitizeAttributes
|
|
@@ -2308,7 +2095,9 @@ function createDOMPurify() {
|
|
|
2308
2095
|
const _sanitizeAttributes = function _sanitizeAttributes(currentNode) {
|
|
2309
2096
|
/* Execute a hook if present */
|
|
2310
2097
|
_executeHooks(hooks.beforeSanitizeAttributes, currentNode, null);
|
|
2311
|
-
const
|
|
2098
|
+
const {
|
|
2099
|
+
attributes
|
|
2100
|
+
} = currentNode;
|
|
2312
2101
|
/* Check if we have attributes; if not we might have a text node */
|
|
2313
2102
|
if (!attributes || _isClobbered(currentNode)) {
|
|
2314
2103
|
return;
|
|
@@ -2324,9 +2113,11 @@ function createDOMPurify() {
|
|
|
2324
2113
|
/* Go backwards over all attributes; safely remove bad ones */
|
|
2325
2114
|
while (l--) {
|
|
2326
2115
|
const attr = attributes[l];
|
|
2327
|
-
const
|
|
2328
|
-
|
|
2329
|
-
|
|
2116
|
+
const {
|
|
2117
|
+
name,
|
|
2118
|
+
namespaceURI,
|
|
2119
|
+
value: attrValue
|
|
2120
|
+
} = attr;
|
|
2330
2121
|
const lcName = transformCaseFunc(name);
|
|
2331
2122
|
const initValue = attrValue;
|
|
2332
2123
|
let value = name === 'value' ? initValue : stringTrim(initValue);
|
|
@@ -2374,7 +2165,7 @@ function createDOMPurify() {
|
|
|
2374
2165
|
}
|
|
2375
2166
|
/* Sanitize attribute content to be template-safe */
|
|
2376
2167
|
if (SAFE_FOR_TEMPLATES) {
|
|
2377
|
-
arrayForEach([MUSTACHE_EXPR
|
|
2168
|
+
arrayForEach([MUSTACHE_EXPR, ERB_EXPR, TMPLIT_EXPR], expr => {
|
|
2378
2169
|
value = stringReplace(value, expr, ' ');
|
|
2379
2170
|
});
|
|
2380
2171
|
}
|
|
@@ -2390,7 +2181,7 @@ function createDOMPurify() {
|
|
|
2390
2181
|
switch (trustedTypes.getAttributeType(lcTag, lcName)) {
|
|
2391
2182
|
case 'TrustedHTML':
|
|
2392
2183
|
{
|
|
2393
|
-
value =
|
|
2184
|
+
value = trustedTypesPolicy.createHTML(value);
|
|
2394
2185
|
break;
|
|
2395
2186
|
}
|
|
2396
2187
|
case 'TrustedScriptURL':
|
|
@@ -2440,98 +2231,14 @@ function createDOMPurify() {
|
|
|
2440
2231
|
_sanitizeElements(shadowNode);
|
|
2441
2232
|
/* Check attributes next */
|
|
2442
2233
|
_sanitizeAttributes(shadowNode);
|
|
2443
|
-
/* Deep shadow DOM detected
|
|
2444
|
-
|
|
2445
|
-
DOCUMENT_FRAGMENT_NODE constant rather than instanceof, so we
|
|
2446
|
-
recurse into <template>.content from foreign realms too. */
|
|
2447
|
-
if (_isDocumentFragment(shadowNode.content)) {
|
|
2234
|
+
/* Deep shadow DOM detected */
|
|
2235
|
+
if (shadowNode.content instanceof DocumentFragment) {
|
|
2448
2236
|
_sanitizeShadowDOM2(shadowNode.content);
|
|
2449
2237
|
}
|
|
2450
|
-
/* An element iterated here may itself host an attached
|
|
2451
|
-
shadow root. The default NodeIterator does not enter shadow
|
|
2452
|
-
trees, so a shadow root nested inside template.content was
|
|
2453
|
-
previously reached by no walk at all (the pre-pass at
|
|
2454
|
-
_sanitizeAttachedShadowRoots descends via childNodes, which
|
|
2455
|
-
doesn't enter template.content; the template-content recursion
|
|
2456
|
-
above iterates the content but never inspected shadowRoot).
|
|
2457
|
-
Walk it explicitly. The nodeType guard avoids reading
|
|
2458
|
-
shadowRoot off text / comment / CDATA / PI nodes that the
|
|
2459
|
-
iterator also surfaces. */
|
|
2460
|
-
const shadowNodeType = getNodeType ? getNodeType(shadowNode) : shadowNode.nodeType;
|
|
2461
|
-
if (shadowNodeType === NODE_TYPE.element) {
|
|
2462
|
-
const innerSr = getShadowRoot ? getShadowRoot(shadowNode) : shadowNode.shadowRoot;
|
|
2463
|
-
if (_isDocumentFragment(innerSr)) {
|
|
2464
|
-
_sanitizeAttachedShadowRoots2(innerSr);
|
|
2465
|
-
_sanitizeShadowDOM2(innerSr);
|
|
2466
|
-
}
|
|
2467
|
-
}
|
|
2468
2238
|
}
|
|
2469
2239
|
/* Execute a hook if present */
|
|
2470
2240
|
_executeHooks(hooks.afterSanitizeShadowDOM, fragment, null);
|
|
2471
2241
|
};
|
|
2472
|
-
/**
|
|
2473
|
-
* _sanitizeAttachedShadowRoots
|
|
2474
|
-
*
|
|
2475
|
-
* Walks `root` and feeds every attached shadow root we encounter into
|
|
2476
|
-
* the existing _sanitizeShadowDOM pipeline. The default node iterator
|
|
2477
|
-
* does not descend into shadow trees, so nodes inside an attached
|
|
2478
|
-
* shadow root would otherwise be skipped entirely.
|
|
2479
|
-
*
|
|
2480
|
-
* Two real input paths put attached shadow roots in front of us:
|
|
2481
|
-
* 1. IN_PLACE on a DOM node that already has shadow roots attached.
|
|
2482
|
-
* 2. DOM-node input where importNode(dirty, true) deep-clones the
|
|
2483
|
-
* shadow root because it was created with `clonable: true`.
|
|
2484
|
-
*
|
|
2485
|
-
* This pass runs once, up front, so the main iteration loop (and the
|
|
2486
|
-
* existing _sanitizeShadowDOM template-content recursion) stay
|
|
2487
|
-
* untouched — string-input paths are not affected.
|
|
2488
|
-
*
|
|
2489
|
-
* @param root the subtree root to walk for attached shadow roots
|
|
2490
|
-
*/
|
|
2491
|
-
const _sanitizeAttachedShadowRoots2 = function _sanitizeAttachedShadowRoots(root) {
|
|
2492
|
-
const nodeType = getNodeType ? getNodeType(root) : root.nodeType;
|
|
2493
|
-
if (nodeType === NODE_TYPE.element) {
|
|
2494
|
-
const sr = getShadowRoot ? getShadowRoot(root) : root.shadowRoot;
|
|
2495
|
-
// Realm-safe check (GHSA-hpcv-96wg-7vj8): use nodeType-based
|
|
2496
|
-
// detection rather than `instanceof DocumentFragment`, which is
|
|
2497
|
-
// realm-bound and silently skipped shadow roots whose host element
|
|
2498
|
-
// belonged to a foreign realm (e.g. iframe.contentDocument
|
|
2499
|
-
// attachShadow). A foreign-realm ShadowRoot extends the foreign
|
|
2500
|
-
// realm's DocumentFragment, not ours, so the old instanceof check
|
|
2501
|
-
// returned false and the shadow subtree was never walked.
|
|
2502
|
-
if (_isDocumentFragment(sr)) {
|
|
2503
|
-
// Recurse first so that nested shadow roots are reached even if
|
|
2504
|
-
// _sanitizeShadowDOM removes hosts at this level.
|
|
2505
|
-
_sanitizeAttachedShadowRoots2(sr);
|
|
2506
|
-
_sanitizeShadowDOM2(sr);
|
|
2507
|
-
}
|
|
2508
|
-
}
|
|
2509
|
-
// Snapshot children before recursing. Sanitization of one subtree
|
|
2510
|
-
// (e.g. via an uponSanitizeShadowNode hook) may detach siblings,
|
|
2511
|
-
// and naive nextSibling traversal would silently skip the rest of
|
|
2512
|
-
// the list once a node is detached.
|
|
2513
|
-
const childNodes = getChildNodes ? getChildNodes(root) : root.childNodes;
|
|
2514
|
-
if (!childNodes) {
|
|
2515
|
-
return;
|
|
2516
|
-
}
|
|
2517
|
-
const snapshot = [];
|
|
2518
|
-
arrayForEach(childNodes, child => {
|
|
2519
|
-
arrayPush(snapshot, child);
|
|
2520
|
-
});
|
|
2521
|
-
for (const child of snapshot) {
|
|
2522
|
-
_sanitizeAttachedShadowRoots2(child);
|
|
2523
|
-
}
|
|
2524
|
-
/* When the root is a <template>, also descend into root.content */
|
|
2525
|
-
if (nodeType === NODE_TYPE.element) {
|
|
2526
|
-
const rootName = getNodeName ? getNodeName(root) : null;
|
|
2527
|
-
if (typeof rootName === 'string' && transformCaseFunc(rootName) === 'template') {
|
|
2528
|
-
const content = root.content;
|
|
2529
|
-
if (_isDocumentFragment(content)) {
|
|
2530
|
-
_sanitizeAttachedShadowRoots2(content);
|
|
2531
|
-
}
|
|
2532
|
-
}
|
|
2533
|
-
}
|
|
2534
|
-
};
|
|
2535
2242
|
// eslint-disable-next-line complexity
|
|
2536
2243
|
DOMPurify.sanitize = function (dirty) {
|
|
2537
2244
|
let cfg = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : {};
|
|
@@ -2568,35 +2275,15 @@ function createDOMPurify() {
|
|
|
2568
2275
|
IN_PLACE = false;
|
|
2569
2276
|
}
|
|
2570
2277
|
if (IN_PLACE) {
|
|
2571
|
-
/* Do some early pre-sanitization to avoid unsafe root nodes
|
|
2572
|
-
|
|
2573
|
-
child named "nodeName" on the form root would otherwise shadow
|
|
2574
|
-
the property and let this check skip the root-allowlist
|
|
2575
|
-
validation entirely. */
|
|
2576
|
-
const nn = getNodeName ? getNodeName(dirty) : dirty.nodeName;
|
|
2278
|
+
/* Do some early pre-sanitization to avoid unsafe root nodes */
|
|
2279
|
+
const nn = dirty.nodeName;
|
|
2577
2280
|
if (typeof nn === 'string') {
|
|
2578
2281
|
const tagName = transformCaseFunc(nn);
|
|
2579
2282
|
if (!ALLOWED_TAGS[tagName] || FORBID_TAGS[tagName]) {
|
|
2580
2283
|
throw typeErrorCreate('root node is forbidden and cannot be sanitized in-place');
|
|
2581
2284
|
}
|
|
2582
2285
|
}
|
|
2583
|
-
|
|
2584
|
-
removal path can not detach a parent-less root: _forceRemove
|
|
2585
|
-
falls through to Element.prototype.remove(), which per spec
|
|
2586
|
-
is a no-op on a node with no parent. A clobbered root would
|
|
2587
|
-
then survive the main loop with its attributes uninspected,
|
|
2588
|
-
because _sanitizeAttributes early-returns on _isClobbered. The
|
|
2589
|
-
result would be an attacker-controlled form, complete with any
|
|
2590
|
-
event-handler attributes the caller passed in, handed back to
|
|
2591
|
-
the application unsanitized. Refuse to sanitize such a root
|
|
2592
|
-
the same way we refuse a forbidden tag. GHSA-r47g-fvhr-h676. */
|
|
2593
|
-
if (_isClobbered(dirty)) {
|
|
2594
|
-
throw typeErrorCreate('root node is clobbered and cannot be sanitized in-place');
|
|
2595
|
-
}
|
|
2596
|
-
/* Sanitize attached shadow roots before the main iterator runs.
|
|
2597
|
-
The iterator does not descend into shadow trees. */
|
|
2598
|
-
_sanitizeAttachedShadowRoots2(dirty);
|
|
2599
|
-
} else if (_isNode(dirty)) {
|
|
2286
|
+
} else if (dirty instanceof Node) {
|
|
2600
2287
|
/* If dirty is a DOM element, append to an empty document to avoid
|
|
2601
2288
|
elements being stripped by the parser */
|
|
2602
2289
|
body = _initDocument('<!---->');
|
|
@@ -2610,18 +2297,12 @@ function createDOMPurify() {
|
|
|
2610
2297
|
// eslint-disable-next-line unicorn/prefer-dom-node-append
|
|
2611
2298
|
body.appendChild(importedNode);
|
|
2612
2299
|
}
|
|
2613
|
-
/* Clonable shadow roots are deep-cloned by importNode(); sanitize
|
|
2614
|
-
them before the main iterator runs, since the iterator does not
|
|
2615
|
-
descend into shadow trees. The walk routes every read through a
|
|
2616
|
-
cached prototype getter so clobbering descendants on a form root
|
|
2617
|
-
cannot hide a shadow host from this pass. */
|
|
2618
|
-
_sanitizeAttachedShadowRoots2(importedNode);
|
|
2619
2300
|
} else {
|
|
2620
2301
|
/* Exit directly if we have nothing to do */
|
|
2621
2302
|
if (!RETURN_DOM && !SAFE_FOR_TEMPLATES && !WHOLE_DOCUMENT &&
|
|
2622
2303
|
// eslint-disable-next-line unicorn/prefer-includes
|
|
2623
2304
|
dirty.indexOf('<') === -1) {
|
|
2624
|
-
return trustedTypesPolicy && RETURN_TRUSTED_TYPE ?
|
|
2305
|
+
return trustedTypesPolicy && RETURN_TRUSTED_TYPE ? trustedTypesPolicy.createHTML(dirty) : dirty;
|
|
2625
2306
|
}
|
|
2626
2307
|
/* Initialize the document to work on */
|
|
2627
2308
|
body = _initDocument(dirty);
|
|
@@ -2642,25 +2323,24 @@ function createDOMPurify() {
|
|
|
2642
2323
|
_sanitizeElements(currentNode);
|
|
2643
2324
|
/* Check attributes next */
|
|
2644
2325
|
_sanitizeAttributes(currentNode);
|
|
2645
|
-
/* Shadow DOM detected, sanitize it
|
|
2646
|
-
|
|
2647
|
-
instead of instanceof, so foreign-realm <template>.content is
|
|
2648
|
-
walked correctly. */
|
|
2649
|
-
if (_isDocumentFragment(currentNode.content)) {
|
|
2326
|
+
/* Shadow DOM detected, sanitize it */
|
|
2327
|
+
if (currentNode.content instanceof DocumentFragment) {
|
|
2650
2328
|
_sanitizeShadowDOM2(currentNode.content);
|
|
2651
2329
|
}
|
|
2652
2330
|
}
|
|
2653
2331
|
/* If we sanitized `dirty` in-place, return it. */
|
|
2654
2332
|
if (IN_PLACE) {
|
|
2655
|
-
if (SAFE_FOR_TEMPLATES) {
|
|
2656
|
-
_scrubTemplateExpressions2(dirty);
|
|
2657
|
-
}
|
|
2658
2333
|
return dirty;
|
|
2659
2334
|
}
|
|
2660
2335
|
/* Return sanitized string or DOM */
|
|
2661
2336
|
if (RETURN_DOM) {
|
|
2662
2337
|
if (SAFE_FOR_TEMPLATES) {
|
|
2663
|
-
|
|
2338
|
+
body.normalize();
|
|
2339
|
+
let html = body.innerHTML;
|
|
2340
|
+
arrayForEach([MUSTACHE_EXPR, ERB_EXPR, TMPLIT_EXPR], expr => {
|
|
2341
|
+
html = stringReplace(html, expr, ' ');
|
|
2342
|
+
});
|
|
2343
|
+
body.innerHTML = html;
|
|
2664
2344
|
}
|
|
2665
2345
|
if (RETURN_DOM_FRAGMENT) {
|
|
2666
2346
|
returnNode = createDocumentFragment.call(body.ownerDocument);
|
|
@@ -2690,11 +2370,11 @@ function createDOMPurify() {
|
|
|
2690
2370
|
}
|
|
2691
2371
|
/* Sanitize final string template-safe */
|
|
2692
2372
|
if (SAFE_FOR_TEMPLATES) {
|
|
2693
|
-
arrayForEach([MUSTACHE_EXPR
|
|
2373
|
+
arrayForEach([MUSTACHE_EXPR, ERB_EXPR, TMPLIT_EXPR], expr => {
|
|
2694
2374
|
serializedHTML = stringReplace(serializedHTML, expr, ' ');
|
|
2695
2375
|
});
|
|
2696
2376
|
}
|
|
2697
|
-
return trustedTypesPolicy && RETURN_TRUSTED_TYPE ?
|
|
2377
|
+
return trustedTypesPolicy && RETURN_TRUSTED_TYPE ? trustedTypesPolicy.createHTML(serializedHTML) : serializedHTML;
|
|
2698
2378
|
};
|
|
2699
2379
|
DOMPurify.setConfig = function () {
|
|
2700
2380
|
let cfg = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : {};
|
|
@@ -2741,7 +2421,6 @@ var purify = createDOMPurify();
|
|
|
2741
2421
|
* 预设主题配置
|
|
2742
2422
|
* 提供多种常用主题,用户可以直接使用或自定义
|
|
2743
2423
|
*/
|
|
2744
|
-
|
|
2745
2424
|
/**
|
|
2746
2425
|
* 默认主题(蓝色渐变)
|
|
2747
2426
|
*/
|
|
@@ -2792,7 +2471,6 @@ const DEFAULT_THEME = {
|
|
|
2792
2471
|
button: '0 4px 12px rgba(24, 144, 255, 0.35)'
|
|
2793
2472
|
}
|
|
2794
2473
|
};
|
|
2795
|
-
|
|
2796
2474
|
/**
|
|
2797
2475
|
* 深色主题
|
|
2798
2476
|
*/
|
|
@@ -2843,7 +2521,6 @@ const DARK_THEME = {
|
|
|
2843
2521
|
button: '0 4px 12px rgba(24, 144, 255, 0.4)'
|
|
2844
2522
|
}
|
|
2845
2523
|
};
|
|
2846
|
-
|
|
2847
2524
|
/**
|
|
2848
2525
|
* 清新主题(绿色系)
|
|
2849
2526
|
*/
|
|
@@ -2894,7 +2571,6 @@ const FRESH_THEME = {
|
|
|
2894
2571
|
button: '0 4px 12px rgba(82, 196, 26, 0.35)'
|
|
2895
2572
|
}
|
|
2896
2573
|
};
|
|
2897
|
-
|
|
2898
2574
|
/**
|
|
2899
2575
|
* 活力主题(橙色系)
|
|
2900
2576
|
*/
|
|
@@ -2945,7 +2621,6 @@ const VIBRANT_THEME = {
|
|
|
2945
2621
|
button: '0 4px 12px rgba(250, 140, 22, 0.35)'
|
|
2946
2622
|
}
|
|
2947
2623
|
};
|
|
2948
|
-
|
|
2949
2624
|
/**
|
|
2950
2625
|
* 浪漫主题(粉色系)
|
|
2951
2626
|
*/
|
|
@@ -2996,7 +2671,6 @@ const ROMANTIC_THEME = {
|
|
|
2996
2671
|
button: '0 4px 12px rgba(235, 47, 150, 0.35)'
|
|
2997
2672
|
}
|
|
2998
2673
|
};
|
|
2999
|
-
|
|
3000
2674
|
/**
|
|
3001
2675
|
* 紫色主题(优雅神秘)
|
|
3002
2676
|
*/
|
|
@@ -3047,7 +2721,6 @@ const PURPLE_THEME = {
|
|
|
3047
2721
|
button: '0 4px 12px rgba(114, 46, 209, 0.35)'
|
|
3048
2722
|
}
|
|
3049
2723
|
};
|
|
3050
|
-
|
|
3051
2724
|
/**
|
|
3052
2725
|
* 海洋主题(深蓝深海)
|
|
3053
2726
|
*/
|
|
@@ -3098,7 +2771,6 @@ const OCEAN_THEME = {
|
|
|
3098
2771
|
button: '0 4px 12px rgba(0, 119, 182, 0.35)'
|
|
3099
2772
|
}
|
|
3100
2773
|
};
|
|
3101
|
-
|
|
3102
2774
|
/**
|
|
3103
2775
|
* 暮光主题(紫罗兰渐变)
|
|
3104
2776
|
*/
|
|
@@ -3149,7 +2821,6 @@ const TWILIGHT_THEME = {
|
|
|
3149
2821
|
button: '0 4px 12px rgba(147, 51, 234, 0.35)'
|
|
3150
2822
|
}
|
|
3151
2823
|
};
|
|
3152
|
-
|
|
3153
2824
|
/**
|
|
3154
2825
|
* 薄荷主题(青绿色)
|
|
3155
2826
|
*/
|
|
@@ -3200,7 +2871,6 @@ const MINT_THEME = {
|
|
|
3200
2871
|
button: '0 4px 12px rgba(20, 184, 166, 0.35)'
|
|
3201
2872
|
}
|
|
3202
2873
|
};
|
|
3203
|
-
|
|
3204
2874
|
/**
|
|
3205
2875
|
* 玫瑰主题(深红色)
|
|
3206
2876
|
*/
|
|
@@ -3251,7 +2921,6 @@ const ROSE_THEME = {
|
|
|
3251
2921
|
button: '0 4px 12px rgba(225, 29, 72, 0.35)'
|
|
3252
2922
|
}
|
|
3253
2923
|
};
|
|
3254
|
-
|
|
3255
2924
|
/**
|
|
3256
2925
|
* 极光主题(蓝绿渐变)
|
|
3257
2926
|
*/
|
|
@@ -3302,7 +2971,6 @@ const AURORA_THEME = {
|
|
|
3302
2971
|
button: '0 4px 12px rgba(14, 165, 233, 0.35)'
|
|
3303
2972
|
}
|
|
3304
2973
|
};
|
|
3305
|
-
|
|
3306
2974
|
/**
|
|
3307
2975
|
* 薰衣草主题(淡紫色)
|
|
3308
2976
|
*/
|
|
@@ -3353,7 +3021,6 @@ const LAVENDER_THEME = {
|
|
|
3353
3021
|
button: '0 4px 12px rgba(139, 92, 246, 0.35)'
|
|
3354
3022
|
}
|
|
3355
3023
|
};
|
|
3356
|
-
|
|
3357
3024
|
/**
|
|
3358
3025
|
* 珊瑚主题(粉橙色)
|
|
3359
3026
|
*/
|
|
@@ -3404,7 +3071,6 @@ const CORAL_THEME = {
|
|
|
3404
3071
|
button: '0 4px 12px rgba(249, 115, 22, 0.35)'
|
|
3405
3072
|
}
|
|
3406
3073
|
};
|
|
3407
|
-
|
|
3408
3074
|
/**
|
|
3409
3075
|
* 翡翠主题(深绿色)
|
|
3410
3076
|
*/
|
|
@@ -3455,7 +3121,6 @@ const JADE_THEME = {
|
|
|
3455
3121
|
button: '0 4px 12px rgba(5, 150, 105, 0.35)'
|
|
3456
3122
|
}
|
|
3457
3123
|
};
|
|
3458
|
-
|
|
3459
3124
|
/**
|
|
3460
3125
|
* 星空主题(深蓝紫色)
|
|
3461
3126
|
*/
|
|
@@ -3506,7 +3171,6 @@ const STARSKY_THEME = {
|
|
|
3506
3171
|
button: '0 4px 12px rgba(99, 102, 241, 0.4)'
|
|
3507
3172
|
}
|
|
3508
3173
|
};
|
|
3509
|
-
|
|
3510
3174
|
/**
|
|
3511
3175
|
* 日落主题(暖色调)
|
|
3512
3176
|
*/
|
|
@@ -3557,7 +3221,6 @@ const SUNSET_THEME = {
|
|
|
3557
3221
|
button: '0 4px 12px rgba(234, 88, 12, 0.35)'
|
|
3558
3222
|
}
|
|
3559
3223
|
};
|
|
3560
|
-
|
|
3561
3224
|
/**
|
|
3562
3225
|
* 所有预设主题列表
|
|
3563
3226
|
*/
|
|
@@ -3583,27 +3246,21 @@ const PRESET_THEMES = {
|
|
|
3583
3246
|
/**
|
|
3584
3247
|
* AIChatDialog - 纯JavaScript实现的AI对话组件 (Web Components)
|
|
3585
3248
|
* 样式与功能完全对齐 Vue 版本(1:1 复刻)
|
|
3586
|
-
*
|
|
3249
|
+
*
|
|
3587
3250
|
* @author IBC AI Team
|
|
3588
|
-
* @version 2.0
|
|
3251
|
+
* @version 2.1.0
|
|
3589
3252
|
*/
|
|
3590
|
-
|
|
3591
|
-
|
|
3253
|
+
// 导入 marked Markdown 解析库 + DOMPurify XSS防护 + 主题配置
|
|
3592
3254
|
// 运行时占位符(对齐 portal)
|
|
3593
3255
|
const RUNTIME_PLACEHOLDERS = ['请求已受理', '任务已受理', '正在思考', '正在执行', '结果处理中', '已进入助手运行时', '任务运行时', 'SSE 连接已建立'];
|
|
3594
3256
|
const DEFAULT_DIALOG_TOP = '96px';
|
|
3595
|
-
|
|
3596
3257
|
// ========== marked 配置(安全且功能完整) ==========
|
|
3258
|
+
// marked v18 移除了 headerIds/mangle,仅保留 breaks 和 gfm
|
|
3597
3259
|
g.setOptions({
|
|
3598
3260
|
breaks: true,
|
|
3599
3261
|
// 支持GFM换行(单换行变<br>)
|
|
3600
|
-
gfm: true
|
|
3601
|
-
|
|
3602
|
-
headerIds: false,
|
|
3603
|
-
// 不生成标题id
|
|
3604
|
-
mangle: false // 不转义邮箱
|
|
3605
|
-
});
|
|
3606
|
-
|
|
3262
|
+
gfm: true // GitHub Flavored Markdown
|
|
3263
|
+
} /* marked v18 types are strict */);
|
|
3607
3264
|
// Vue themes.js 的 camelCase key → JS _themeVars CSS变量名 映射表
|
|
3608
3265
|
const THEME_COLOR_KEY_MAP = {
|
|
3609
3266
|
primary: '--ai-primary',
|
|
@@ -3632,14 +3289,82 @@ const THEME_COLOR_KEY_MAP = {
|
|
|
3632
3289
|
error: '--ai-error',
|
|
3633
3290
|
warning: '--ai-warning'
|
|
3634
3291
|
};
|
|
3635
|
-
|
|
3636
3292
|
// ========== 16种预设主题(与Vue版themes.js完全一致) ==========
|
|
3637
3293
|
class AIChatDialog extends HTMLElement {
|
|
3294
|
+
// ====== 类型声明(HTMLElement 子类必需) ======
|
|
3638
3295
|
static get observedAttributes() {
|
|
3639
3296
|
return ['visible'];
|
|
3640
3297
|
}
|
|
3641
3298
|
constructor() {
|
|
3642
3299
|
super();
|
|
3300
|
+
// 核心状态
|
|
3301
|
+
this._messages = [];
|
|
3302
|
+
this._isLoading = false;
|
|
3303
|
+
this._config = null;
|
|
3304
|
+
this._mockMode = false;
|
|
3305
|
+
this._isDragging = false;
|
|
3306
|
+
this._isExpanded = false;
|
|
3307
|
+
this._chatClient = null;
|
|
3308
|
+
this._attachments = [];
|
|
3309
|
+
this._dragStartX = 0;
|
|
3310
|
+
this._dragStartY = 0;
|
|
3311
|
+
this._dialogX = 0;
|
|
3312
|
+
this._dialogY = 0;
|
|
3313
|
+
this._appDetail = null;
|
|
3314
|
+
this._historyVisible = false;
|
|
3315
|
+
this._historyRecords = [];
|
|
3316
|
+
this._historyLoading = false;
|
|
3317
|
+
this._historyLoaded = false;
|
|
3318
|
+
this._historyPageIndex = 1;
|
|
3319
|
+
this._historyHasMore = false;
|
|
3320
|
+
this._isMobile = false;
|
|
3321
|
+
// 流式渲染
|
|
3322
|
+
this._streamTextEl = null;
|
|
3323
|
+
this._streamInitDone = false;
|
|
3324
|
+
this._streamFullText = '';
|
|
3325
|
+
this._streamVisibleLength = 0;
|
|
3326
|
+
this._streamTypeRaf = null;
|
|
3327
|
+
this._streaming = false;
|
|
3328
|
+
// 运行时面板
|
|
3329
|
+
this._runtimePanelEl = null;
|
|
3330
|
+
this._runtimeEventsContainerEl = null;
|
|
3331
|
+
this._runtimeStatusEl = null;
|
|
3332
|
+
this._runtimeCountEl = null;
|
|
3333
|
+
// 页面上下文
|
|
3334
|
+
this._context = {};
|
|
3335
|
+
// 主题
|
|
3336
|
+
this._themeVars = {};
|
|
3337
|
+
// 事件处理器引用(用于清理)
|
|
3338
|
+
this._handleOnline = null;
|
|
3339
|
+
this._handleOffline = null;
|
|
3340
|
+
this._handleResize = null;
|
|
3341
|
+
// DOM 缓存引用
|
|
3342
|
+
this._dialog = null;
|
|
3343
|
+
this._body = null;
|
|
3344
|
+
this._input = null;
|
|
3345
|
+
this._sendBtn = null;
|
|
3346
|
+
this._attachBtn = null;
|
|
3347
|
+
this._attachmentInput = null;
|
|
3348
|
+
this._closeBtn = null;
|
|
3349
|
+
this._expandBtn = null;
|
|
3350
|
+
this._newChatBtn = null;
|
|
3351
|
+
this._historyBtn = null;
|
|
3352
|
+
this._floatBtn = null;
|
|
3353
|
+
this._inputContainer = null;
|
|
3354
|
+
// 会话
|
|
3355
|
+
this._conversationId = null;
|
|
3356
|
+
// 防抖
|
|
3357
|
+
this._lastSendTime = 0;
|
|
3358
|
+
// 拖拽处理器引用
|
|
3359
|
+
this._dragMouseDownHandler = null;
|
|
3360
|
+
this._dragMouseMoveHandler = null;
|
|
3361
|
+
this._dragMouseUpHandler = null;
|
|
3362
|
+
this._dragTouchStartHandler = null;
|
|
3363
|
+
this._dragTouchMoveHandler = null;
|
|
3364
|
+
this._dragTouchEndHandler = null;
|
|
3365
|
+
this._dragInitialized = false;
|
|
3366
|
+
// 错误标记
|
|
3367
|
+
this._ERROR_CONTENT_PATTERNS = ['(请求超时', '(等待AI处理', '(无回复)', '(已停止)', '(AI 处理超时', '错误:', '网络错误:', '抱歉,'];
|
|
3643
3368
|
this.attachShadow({
|
|
3644
3369
|
mode: 'open'
|
|
3645
3370
|
});
|
|
@@ -3678,7 +3403,6 @@ class AIChatDialog extends HTMLElement {
|
|
|
3678
3403
|
this._runtimeCountEl = null;
|
|
3679
3404
|
// P3: 页面上下文数据(宿主页面选中的数据)
|
|
3680
3405
|
this._context = {};
|
|
3681
|
-
|
|
3682
3406
|
// ====== 默认主题配置(与 themes.js DEFAULT_THEME 完全一致) ======
|
|
3683
3407
|
this._themeVars = {
|
|
3684
3408
|
'--ai-primary': '#1890ff',
|
|
@@ -3721,9 +3445,7 @@ class AIChatDialog extends HTMLElement {
|
|
|
3721
3445
|
this.render();
|
|
3722
3446
|
this.bindEvents();
|
|
3723
3447
|
}
|
|
3724
|
-
|
|
3725
3448
|
// ==================== 生命周期 ====================
|
|
3726
|
-
|
|
3727
3449
|
connectedCallback() {
|
|
3728
3450
|
// P1-8: 网络状态监控(与Vue版一致)
|
|
3729
3451
|
this._handleOnline = () => {
|
|
@@ -3736,12 +3458,11 @@ class AIChatDialog extends HTMLElement {
|
|
|
3736
3458
|
tip.className = 'network-tip';
|
|
3737
3459
|
tip.textContent = '网络连接已断开,请检查网络后重试';
|
|
3738
3460
|
this._dialog?.prepend(tip);
|
|
3739
|
-
setTimeout(() => tip.remove(), 5000);
|
|
3461
|
+
window.setTimeout(() => tip.remove(), 5000);
|
|
3740
3462
|
}
|
|
3741
3463
|
};
|
|
3742
3464
|
window.addEventListener('online', this._handleOnline);
|
|
3743
3465
|
window.addEventListener('offline', this._handleOffline);
|
|
3744
|
-
|
|
3745
3466
|
// P2-17: 移动端环境检测(与Vue版 detectEnvironment 一致)
|
|
3746
3467
|
this._detectMobile();
|
|
3747
3468
|
// P2-17: 窗口resize时重新检测移动端
|
|
@@ -3754,7 +3475,6 @@ class AIChatDialog extends HTMLElement {
|
|
|
3754
3475
|
}
|
|
3755
3476
|
};
|
|
3756
3477
|
window.addEventListener('resize', this._handleResize);
|
|
3757
|
-
|
|
3758
3478
|
// 恢复后台Token刷新定时器(SPA路由切换后 re-attach 场景)
|
|
3759
3479
|
if (this._chatClient) this._chatClient.startBackgroundRefresh();
|
|
3760
3480
|
}
|
|
@@ -3772,7 +3492,6 @@ class AIChatDialog extends HTMLElement {
|
|
|
3772
3492
|
this._chatClient.cancelCurrentRequest();
|
|
3773
3493
|
}
|
|
3774
3494
|
}
|
|
3775
|
-
|
|
3776
3495
|
// P2-17: 移动端4维检测(与Vue版 isMobileDevice 计算属性完全一致)
|
|
3777
3496
|
_detectMobile() {
|
|
3778
3497
|
// 1. 优先使用显式配置
|
|
@@ -3780,31 +3499,25 @@ class AIChatDialog extends HTMLElement {
|
|
|
3780
3499
|
this._isMobile = !!this._config.mobileMode;
|
|
3781
3500
|
return;
|
|
3782
3501
|
}
|
|
3783
|
-
|
|
3784
3502
|
// 2. 微信小程序环境检测
|
|
3785
3503
|
if (typeof wx !== 'undefined' && wx.getSystemInfoSync) {
|
|
3786
3504
|
this._isMobile = true;
|
|
3787
3505
|
return;
|
|
3788
3506
|
}
|
|
3789
|
-
|
|
3790
3507
|
// 3. 浏览器环境
|
|
3791
3508
|
if (typeof window === 'undefined') {
|
|
3792
3509
|
this._isMobile = false;
|
|
3793
3510
|
return;
|
|
3794
3511
|
}
|
|
3795
|
-
|
|
3796
3512
|
// 4. User Agent 正则匹配
|
|
3797
3513
|
const ua = navigator.userAgent || '';
|
|
3798
3514
|
const isMobileUA = /Android|webOS|iPhone|iPad|iPod|BlackBerry|IEMobile|Opera Mini/i.test(ua);
|
|
3799
|
-
|
|
3800
3515
|
// 5. 屏幕宽度 < 768px + 触摸支持
|
|
3801
3516
|
const isSmallScreen = window.innerWidth < 768;
|
|
3802
3517
|
const hasTouch = 'ontouchstart' in window || navigator.maxTouchPoints > 0;
|
|
3803
|
-
|
|
3804
3518
|
// 综合判断:UA匹配 或 (小屏幕+触摸)
|
|
3805
3519
|
this._isMobile = isMobileUA || isSmallScreen && hasTouch;
|
|
3806
3520
|
}
|
|
3807
|
-
|
|
3808
3521
|
// P2-17: 应用移动端/桌面端布局(自动全屏或恢复浮动气泡)
|
|
3809
3522
|
_applyMobileLayout() {
|
|
3810
3523
|
if (!this._dialog) return;
|
|
@@ -3849,9 +3562,7 @@ class AIChatDialog extends HTMLElement {
|
|
|
3849
3562
|
if (visible) this.showDialog();else this.hideDialog();
|
|
3850
3563
|
}
|
|
3851
3564
|
}
|
|
3852
|
-
|
|
3853
3565
|
// ==================== 属性访问器 ====================
|
|
3854
|
-
|
|
3855
3566
|
get visible() {
|
|
3856
3567
|
return this.hasAttribute('visible');
|
|
3857
3568
|
}
|
|
@@ -3867,9 +3578,7 @@ class AIChatDialog extends HTMLElement {
|
|
|
3867
3578
|
get config() {
|
|
3868
3579
|
return this._config;
|
|
3869
3580
|
}
|
|
3870
|
-
|
|
3871
3581
|
// ==================== 公共API ====================
|
|
3872
|
-
|
|
3873
3582
|
init(config = {}) {
|
|
3874
3583
|
this._config = {
|
|
3875
3584
|
apiBaseUrl: '',
|
|
@@ -4028,7 +3737,6 @@ class AIChatDialog extends HTMLElement {
|
|
|
4028
3737
|
this._mockMode = false;
|
|
4029
3738
|
return this;
|
|
4030
3739
|
}
|
|
4031
|
-
|
|
4032
3740
|
/**
|
|
4033
3741
|
* 设置页面上下文数据(宿主页面选中的数据,随每次请求发送)
|
|
4034
3742
|
* @param {Object} context - 上下文数据对象,传 null 清空
|
|
@@ -4040,7 +3748,6 @@ class AIChatDialog extends HTMLElement {
|
|
|
4040
3748
|
} : {};
|
|
4041
3749
|
return this;
|
|
4042
3750
|
}
|
|
4043
|
-
|
|
4044
3751
|
/**
|
|
4045
3752
|
* 获取当前上下文数据
|
|
4046
3753
|
* @returns {Object}
|
|
@@ -4050,7 +3757,6 @@ class AIChatDialog extends HTMLElement {
|
|
|
4050
3757
|
...this._context
|
|
4051
3758
|
};
|
|
4052
3759
|
}
|
|
4053
|
-
|
|
4054
3760
|
/**
|
|
4055
3761
|
* 外部触发表态发送(业务层主动调用,如点击外部按钮触发 AI 请求)
|
|
4056
3762
|
* @param {string} text - 要发送的消息内容
|
|
@@ -4066,9 +3772,7 @@ class AIChatDialog extends HTMLElement {
|
|
|
4066
3772
|
this.handleSend();
|
|
4067
3773
|
return this;
|
|
4068
3774
|
}
|
|
4069
|
-
|
|
4070
3775
|
// ==================== 显示/隐藏 ====================
|
|
4071
|
-
|
|
4072
3776
|
showDialog() {
|
|
4073
3777
|
const dialog = this.shadowRoot.querySelector('.ai-chat-dialog');
|
|
4074
3778
|
const btn = this.shadowRoot.querySelector('.float-button');
|
|
@@ -4083,9 +3787,7 @@ class AIChatDialog extends HTMLElement {
|
|
|
4083
3787
|
if (dialog) dialog.classList.remove('dialog-visible');
|
|
4084
3788
|
if (btn) btn.style.display = 'flex';
|
|
4085
3789
|
}
|
|
4086
|
-
|
|
4087
3790
|
// ==================== 渲染(完整CSS对齐theme.css) ====================
|
|
4088
|
-
|
|
4089
3791
|
render() {
|
|
4090
3792
|
const cssVarsStr = Object.entries(this._themeVars).map(([k, v]) => `${k}:${v};`).join('\n ');
|
|
4091
3793
|
this.shadowRoot.innerHTML = `
|
|
@@ -5176,16 +4878,43 @@ class AIChatDialog extends HTMLElement {
|
|
|
5176
4878
|
:host(.dark-theme) blockquote { background: rgba(110, 168, 254, 0.08); }
|
|
5177
4879
|
:host(.dark-theme) th { background: rgba(110, 168, 254, 0.12); }
|
|
5178
4880
|
:host(.dark-theme) hr { border-top-color: var(--ai-border); }
|
|
4881
|
+
|
|
4882
|
+
/* ========== 插槽样式(自定义内容区域) ========== */
|
|
4883
|
+
/* header-title 插槽 */
|
|
4884
|
+
::slotted([slot="header-title"]) {
|
|
4885
|
+
font-size: var(--ai-font-title, 15px);
|
|
4886
|
+
font-weight: 650;
|
|
4887
|
+
color: var(--ai-text);
|
|
4888
|
+
overflow: hidden;
|
|
4889
|
+
text-overflow: ellipsis;
|
|
4890
|
+
white-space: nowrap;
|
|
4891
|
+
}
|
|
4892
|
+
/* input-prepend 插槽 — 输入区上方附加内容 */
|
|
4893
|
+
::slotted([slot="input-prepend"]) {
|
|
4894
|
+
display: block;
|
|
4895
|
+
padding: 0 4px 8px;
|
|
4896
|
+
}
|
|
4897
|
+
/* footer 插槽 */
|
|
4898
|
+
::slotted([slot="footer"]) {
|
|
4899
|
+
display: flex;
|
|
4900
|
+
justify-content: space-between;
|
|
4901
|
+
gap: 12px;
|
|
4902
|
+
color: #b5bbc5;
|
|
4903
|
+
font-size: 12px;
|
|
4904
|
+
line-height: 1;
|
|
4905
|
+
}
|
|
5179
4906
|
</style>
|
|
5180
4907
|
|
|
5181
4908
|
<div class="ai-float-container">
|
|
5182
4909
|
<!-- 对话框 -->
|
|
5183
4910
|
<div class="ai-chat-dialog">
|
|
5184
|
-
|
|
5185
|
-
|
|
5186
|
-
|
|
5187
|
-
|
|
5188
|
-
|
|
4911
|
+
<!-- 头部(含展开/关闭按钮) -->
|
|
4912
|
+
<div class="dialog-header">
|
|
4913
|
+
<div class="header-left">
|
|
4914
|
+
<slot name="header-title">
|
|
4915
|
+
<span class="header-title">${this.escapeHtml(this._config?.title || 'AI 助手')}</span>
|
|
4916
|
+
</slot>
|
|
4917
|
+
</div>
|
|
5189
4918
|
<div class="header-right">
|
|
5190
4919
|
<button class="new-chat-btn history-btn" type="button" title="历史记录" aria-label="历史记录">
|
|
5191
4920
|
<svg viewBox="0 0 24 24" fill="none" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><path d="M3 12a9 9 0 1 0 3-6.7"/><path d="M3 4v5h5"/><path d="M12 7v5l3 2"/></svg>
|
|
@@ -5209,9 +4938,10 @@ class AIChatDialog extends HTMLElement {
|
|
|
5209
4938
|
<!-- 消息区域 -->
|
|
5210
4939
|
<div class="dialog-body" id="messagesContainer"></div>
|
|
5211
4940
|
|
|
5212
|
-
|
|
5213
|
-
|
|
5214
|
-
|
|
4941
|
+
<!-- 输入区域 -->
|
|
4942
|
+
<div class="dialog-footer">
|
|
4943
|
+
<slot name="input-prepend"></slot>
|
|
4944
|
+
<div class="input-container" id="inputContainer">
|
|
5215
4945
|
<textarea class="message-input" id="msgInput"
|
|
5216
4946
|
placeholder="请输入您的问题..." autocomplete="off" rows="2"></textarea>
|
|
5217
4947
|
<div class="input-tools">
|
|
@@ -5224,10 +4954,12 @@ class AIChatDialog extends HTMLElement {
|
|
|
5224
4954
|
</button>
|
|
5225
4955
|
</div>
|
|
5226
4956
|
</div>
|
|
5227
|
-
|
|
5228
|
-
|
|
5229
|
-
|
|
5230
|
-
|
|
4957
|
+
<div class="footer-note">
|
|
4958
|
+
<slot name="footer">
|
|
4959
|
+
<span class="footer-note-left">${this.escapeHtml(this._config?.footerDisclaimer || '')}</span>
|
|
4960
|
+
<span class="footer-note-right">${this.escapeHtml(this._config?.footerIdentity || '')}</span>
|
|
4961
|
+
</slot>
|
|
4962
|
+
</div>
|
|
5231
4963
|
</div>
|
|
5232
4964
|
</div>
|
|
5233
4965
|
|
|
@@ -5235,7 +4967,6 @@ class AIChatDialog extends HTMLElement {
|
|
|
5235
4967
|
<div class="float-button" id="floatBtn" title="打开AI助手"></div>
|
|
5236
4968
|
</div>
|
|
5237
4969
|
`;
|
|
5238
|
-
|
|
5239
4970
|
// 缓存DOM引用
|
|
5240
4971
|
this._dialog = this.shadowRoot.querySelector('.ai-chat-dialog');
|
|
5241
4972
|
this._body = this.shadowRoot.querySelector('#messagesContainer');
|
|
@@ -5257,28 +4988,21 @@ class AIChatDialog extends HTMLElement {
|
|
|
5257
4988
|
requestAnimationFrame(() => {
|
|
5258
4989
|
// 关闭按钮
|
|
5259
4990
|
if (this._closeBtn) this._closeBtn.addEventListener('click', () => this.close());
|
|
5260
|
-
|
|
5261
4991
|
// 展开/收起按钮
|
|
5262
4992
|
if (this._expandBtn) this._expandBtn.addEventListener('click', () => this.toggleExpand());
|
|
5263
|
-
|
|
5264
4993
|
// 新会话按钮
|
|
5265
4994
|
if (this._newChatBtn) this._newChatBtn.addEventListener('click', () => this.clearChat());
|
|
5266
|
-
|
|
5267
4995
|
// 历史记录按钮
|
|
5268
4996
|
if (this._historyBtn) this._historyBtn.addEventListener('click', () => this.toggleHistory());
|
|
5269
|
-
|
|
5270
4997
|
// 悬浮按钮打开
|
|
5271
4998
|
if (this._floatBtn) this._floatBtn.addEventListener('click', () => this.open());
|
|
5272
|
-
|
|
5273
4999
|
// 发送按钮
|
|
5274
5000
|
if (this._sendBtn) this._sendBtn.addEventListener('click', () => this.handleSend());
|
|
5275
|
-
|
|
5276
5001
|
// 附件选择
|
|
5277
5002
|
if (this._attachBtn && this._attachmentInput) {
|
|
5278
5003
|
this._attachBtn.addEventListener('click', () => this._attachmentInput.click());
|
|
5279
5004
|
this._attachmentInput.addEventListener('change', () => this._handleAttachmentSelect());
|
|
5280
5005
|
}
|
|
5281
|
-
|
|
5282
5006
|
// 输入回车发送
|
|
5283
5007
|
if (this._input) {
|
|
5284
5008
|
this._autoResizeInput();
|
|
@@ -5294,19 +5018,15 @@ class AIChatDialog extends HTMLElement {
|
|
|
5294
5018
|
}
|
|
5295
5019
|
});
|
|
5296
5020
|
}
|
|
5297
|
-
|
|
5298
5021
|
// 拖拽(内嵌模式禁用)— 由 init() 在 _config 就绪后触发
|
|
5299
5022
|
// initDraggable 延迟到 init() 中执行以免 _config 为空
|
|
5300
5023
|
});
|
|
5301
5024
|
}
|
|
5302
|
-
|
|
5303
5025
|
// ==================== 展开/收起 ====================
|
|
5304
|
-
|
|
5305
5026
|
toggleExpand() {
|
|
5306
5027
|
this._isExpanded = !this._isExpanded;
|
|
5307
5028
|
const dlg = this._dialog;
|
|
5308
5029
|
if (!dlg) return;
|
|
5309
|
-
|
|
5310
5030
|
// P2-17: 移动端始终全屏,展开/收起无意义(与Vue版 dialogStyle 一致)
|
|
5311
5031
|
if (this._isMobile) {
|
|
5312
5032
|
this._isExpanded = false; // 移动端不记录展开状态
|
|
@@ -5343,13 +5063,10 @@ class AIChatDialog extends HTMLElement {
|
|
|
5343
5063
|
detail: this._isExpanded
|
|
5344
5064
|
}));
|
|
5345
5065
|
}
|
|
5346
|
-
|
|
5347
5066
|
// ==================== 拖拽(与Vue startDrag/onDrag/stopDrag 一致) ====================
|
|
5348
|
-
|
|
5349
5067
|
initDraggable() {
|
|
5350
5068
|
const header = this.shadowRoot?.querySelector('.dialog-header');
|
|
5351
5069
|
if (!header) return;
|
|
5352
|
-
|
|
5353
5070
|
// 清理上一次遗留的 document 级监听(SPA 路由切换场景)
|
|
5354
5071
|
this._cleanupDragListeners();
|
|
5355
5072
|
const self = this;
|
|
@@ -5377,7 +5094,6 @@ class AIChatDialog extends HTMLElement {
|
|
|
5377
5094
|
self._dialog.classList.remove('dragging');
|
|
5378
5095
|
}
|
|
5379
5096
|
};
|
|
5380
|
-
|
|
5381
5097
|
// 鼠标事件(桌面端)— 保存引用以便清理
|
|
5382
5098
|
this._dragMouseDownHandler = e => {
|
|
5383
5099
|
if (e.target.closest('button, .header-icon')) return;
|
|
@@ -5391,7 +5107,6 @@ class AIChatDialog extends HTMLElement {
|
|
|
5391
5107
|
header.addEventListener('mousedown', this._dragMouseDownHandler);
|
|
5392
5108
|
document.addEventListener('mousemove', this._dragMouseMoveHandler);
|
|
5393
5109
|
document.addEventListener('mouseup', this._dragMouseUpHandler);
|
|
5394
|
-
|
|
5395
5110
|
// P1-7: 触摸事件(移动端,与Vue版一致)
|
|
5396
5111
|
this._dragTouchStartHandler = e => {
|
|
5397
5112
|
if (self._isExpanded) return;
|
|
@@ -5414,7 +5129,6 @@ class AIChatDialog extends HTMLElement {
|
|
|
5414
5129
|
document.addEventListener('touchend', this._dragTouchEndHandler);
|
|
5415
5130
|
this._dragInitialized = true;
|
|
5416
5131
|
}
|
|
5417
|
-
|
|
5418
5132
|
// P0: 清理拖拽事件监听(SPA disconnectedCallback / re-init 调用)
|
|
5419
5133
|
_cleanupDragListeners() {
|
|
5420
5134
|
if (this._dragMouseDownHandler) {
|
|
@@ -5446,12 +5160,10 @@ class AIChatDialog extends HTMLElement {
|
|
|
5446
5160
|
// 更新标题
|
|
5447
5161
|
const titleEl = this.shadowRoot.querySelector('.header-title');
|
|
5448
5162
|
if (titleEl && this._config.title) titleEl.textContent = this._config.title;
|
|
5449
|
-
|
|
5450
5163
|
// 更新placeholder
|
|
5451
5164
|
if (this._input && this._config.placeholder) this._input.placeholder = this._config.placeholder;
|
|
5452
5165
|
// P1-11: 动态设置输入框最大长度
|
|
5453
5166
|
if (this._input && this._config.maxLength) this._input.maxLength = this._config.maxLength;
|
|
5454
|
-
|
|
5455
5167
|
// ====== 主题处理:16种字符串预设 + 对象自定义 ======
|
|
5456
5168
|
const t = this._config.theme;
|
|
5457
5169
|
const host = this.shadowRoot.host;
|
|
@@ -5480,7 +5192,6 @@ class AIChatDialog extends HTMLElement {
|
|
|
5480
5192
|
}
|
|
5481
5193
|
host.classList.toggle('dark-theme', ['dark', 'starsky'].includes(t.toLowerCase()));
|
|
5482
5194
|
}
|
|
5483
|
-
|
|
5484
5195
|
// 对象形式自定义主题 - 也先重置再合并
|
|
5485
5196
|
if (t && typeof t === 'object') {
|
|
5486
5197
|
this._resetThemeVars();
|
|
@@ -5495,7 +5206,6 @@ class AIChatDialog extends HTMLElement {
|
|
|
5495
5206
|
});
|
|
5496
5207
|
host.classList.remove('dark-theme');
|
|
5497
5208
|
}
|
|
5498
|
-
|
|
5499
5209
|
// 应用所有CSS变量到host
|
|
5500
5210
|
Object.entries(this._themeVars).forEach(([k, v]) => host.style.setProperty(k, v));
|
|
5501
5211
|
}
|
|
@@ -5550,9 +5260,7 @@ class AIChatDialog extends HTMLElement {
|
|
|
5550
5260
|
questions
|
|
5551
5261
|
};
|
|
5552
5262
|
}
|
|
5553
|
-
|
|
5554
5263
|
// ==================== 渲染消息(与Vue模板结构一致) ====================
|
|
5555
|
-
|
|
5556
5264
|
renderMessages(options = {}) {
|
|
5557
5265
|
if (!this._body) return;
|
|
5558
5266
|
if (this._historyVisible) {
|
|
@@ -5614,14 +5322,12 @@ class AIChatDialog extends HTMLElement {
|
|
|
5614
5322
|
</div>
|
|
5615
5323
|
</div>`;
|
|
5616
5324
|
});
|
|
5617
|
-
|
|
5618
5325
|
// 加载状态 — 有执行过程时不展示思考中
|
|
5619
5326
|
if (this._isLoading && !runtimeExists) {
|
|
5620
5327
|
html += '<div class="message-item ai">' + '<div class="message-content">' + '<div class="message-bubble loading"><span class="thinking-text">思考中</span><span class="typing-indicator"><span class="typing-dot"></span><span class="typing-dot"></span><span class="typing-dot"></span></span></div>' + '</div>' + '</div>';
|
|
5621
5328
|
}
|
|
5622
5329
|
html += '</div>';
|
|
5623
5330
|
this._body.innerHTML = html;
|
|
5624
|
-
|
|
5625
5331
|
// 绑定复制事件
|
|
5626
5332
|
this._body.querySelectorAll('.copy-btn').forEach(btn => {
|
|
5627
5333
|
btn.addEventListener('click', () => {
|
|
@@ -5630,7 +5336,6 @@ class AIChatDialog extends HTMLElement {
|
|
|
5630
5336
|
if (msg) this.copyMessage(msg.content, idx);
|
|
5631
5337
|
});
|
|
5632
5338
|
});
|
|
5633
|
-
|
|
5634
5339
|
// 绑定重新生成事件
|
|
5635
5340
|
this._body.querySelectorAll('.regenerate-btn').forEach(btn => {
|
|
5636
5341
|
btn.addEventListener('click', () => {
|
|
@@ -5638,7 +5343,6 @@ class AIChatDialog extends HTMLElement {
|
|
|
5638
5343
|
this.regenerate(idx);
|
|
5639
5344
|
});
|
|
5640
5345
|
});
|
|
5641
|
-
|
|
5642
5346
|
// P4: 绑定自定义操作按钮
|
|
5643
5347
|
this._body.querySelectorAll('.custom-action-btn').forEach(btn => {
|
|
5644
5348
|
btn.addEventListener('click', () => {
|
|
@@ -5656,7 +5360,6 @@ class AIChatDialog extends HTMLElement {
|
|
|
5656
5360
|
}
|
|
5657
5361
|
});
|
|
5658
5362
|
});
|
|
5659
|
-
|
|
5660
5363
|
// 绑定执行过程面板折叠/展开(Shadow DOM 下 inline onclick 不可靠)
|
|
5661
5364
|
this._body.querySelectorAll('.runtime-header').forEach(header => {
|
|
5662
5365
|
header.addEventListener('click', () => {
|
|
@@ -5899,9 +5602,7 @@ class AIChatDialog extends HTMLElement {
|
|
|
5899
5602
|
}
|
|
5900
5603
|
this._updateSendButtonState();
|
|
5901
5604
|
}
|
|
5902
|
-
|
|
5903
5605
|
// ==================== 发送消息(与Vue sendMessage 一致) ====================
|
|
5904
|
-
|
|
5905
5606
|
async handleSend() {
|
|
5906
5607
|
const content = this._input ? this._input.value.trim() : '';
|
|
5907
5608
|
const attachments = [...(this._attachments || [])];
|
|
@@ -5910,14 +5611,12 @@ class AIChatDialog extends HTMLElement {
|
|
|
5910
5611
|
return;
|
|
5911
5612
|
}
|
|
5912
5613
|
if (this._isLoading || this._streaming) return;
|
|
5913
|
-
|
|
5914
5614
|
// P1-11: 字数限制检查
|
|
5915
5615
|
const maxLen = this._config?.maxLength ?? 500;
|
|
5916
5616
|
if (content.length > maxLen) {
|
|
5917
5617
|
this.showToast(`输入内容不能超过${maxLen}个字符(当前${content.length}字)`);
|
|
5918
5618
|
return;
|
|
5919
5619
|
}
|
|
5920
|
-
|
|
5921
5620
|
// 防抖:500ms内不允许重复发送
|
|
5922
5621
|
const now = Date.now();
|
|
5923
5622
|
if (this._lastSendTime && now - this._lastSendTime < 500) return;
|
|
@@ -5965,7 +5664,6 @@ class AIChatDialog extends HTMLElement {
|
|
|
5965
5664
|
attachmentIds.push(res.data.attachmentId);
|
|
5966
5665
|
}
|
|
5967
5666
|
}
|
|
5968
|
-
|
|
5969
5667
|
// P3: 收集页面上下文(setContext + onBeforeSend)
|
|
5970
5668
|
let context = {
|
|
5971
5669
|
...(this._context || {})
|
|
@@ -6044,10 +5742,9 @@ class AIChatDialog extends HTMLElement {
|
|
|
6044
5742
|
}
|
|
6045
5743
|
}
|
|
6046
5744
|
async mockResponse(userContent) {
|
|
6047
|
-
await new Promise(r => setTimeout(r, 600 + Math.random() * 800));
|
|
5745
|
+
await new Promise(r => window.setTimeout(r, 600 + Math.random() * 800));
|
|
6048
5746
|
const responses = [`关于"${(userContent || '').substring(0, 20)}"这个问题,我来为您详细解答...`, `这是一个很好的问题!根据我的理解:\n\n1. 首先,我们需要分析需求\n2. 其次,设计实现方案\n3. 最后,进行测试验证`, `收到您的问题。让我来帮您分析一下...\n\n根据您的描述,我建议您可以按照以下步骤操作:`, `感谢您的提问!以下是我的看法:\n\n• 第一点很关键,需要注意细节\n• 第二点值得关注\n• 第三点是核心所在`];
|
|
6049
5747
|
const mockContent = responses[Math.floor(Math.random() * responses.length)];
|
|
6050
|
-
|
|
6051
5748
|
// 流式效果
|
|
6052
5749
|
const tempId = this.generateId();
|
|
6053
5750
|
this._messages.push({
|
|
@@ -6058,7 +5755,7 @@ class AIChatDialog extends HTMLElement {
|
|
|
6058
5755
|
});
|
|
6059
5756
|
this.renderMessages();
|
|
6060
5757
|
for (let i = 0; i < mockContent.length; i++) {
|
|
6061
|
-
await new Promise(r => setTimeout(r, 12 + Math.random() * 20));
|
|
5758
|
+
await new Promise(r => window.setTimeout(r, 12 + Math.random() * 20));
|
|
6062
5759
|
const msgIdx = this._messages.findIndex(m => m.id === tempId);
|
|
6063
5760
|
if (msgIdx >= 0) {
|
|
6064
5761
|
this._messages[msgIdx].content = mockContent.substring(0, i + 1);
|
|
@@ -6090,9 +5787,7 @@ class AIChatDialog extends HTMLElement {
|
|
|
6090
5787
|
if (idx >= 0) delete this._messages[idx]._isStreaming;
|
|
6091
5788
|
this.renderMessages();
|
|
6092
5789
|
}
|
|
6093
|
-
|
|
6094
5790
|
// P0: 会话ID管理
|
|
6095
|
-
_conversationId = null;
|
|
6096
5791
|
async _directApiCallStream(msgId, content, requestOptions = {}) {
|
|
6097
5792
|
if (!this._chatClient) {
|
|
6098
5793
|
this._chatClient = new AIChatClient(this._config || {});
|
|
@@ -6283,10 +5978,8 @@ class AIChatDialog extends HTMLElement {
|
|
|
6283
5978
|
const idx = this._messages.findIndex(m => m.id === msgId);
|
|
6284
5979
|
if (idx < 0) return;
|
|
6285
5980
|
this._messages[idx].content = content;
|
|
6286
|
-
|
|
6287
5981
|
// Detect if there are runtime events (for typewriter mode check)
|
|
6288
5982
|
const hasRuntime = !!(this._messages[idx]._runtimeEvents && this._messages[idx]._runtimeEvents.length > 0);
|
|
6289
|
-
|
|
6290
5983
|
// Accumulate full content
|
|
6291
5984
|
if (content.startsWith(this._streamFullText)) {
|
|
6292
5985
|
this._streamFullText = content;
|
|
@@ -6347,7 +6040,6 @@ class AIChatDialog extends HTMLElement {
|
|
|
6347
6040
|
this.renderMessages();
|
|
6348
6041
|
}
|
|
6349
6042
|
}
|
|
6350
|
-
|
|
6351
6043
|
// 运行时占位符过滤(对齐 portal)
|
|
6352
6044
|
_isPlaceholder(text) {
|
|
6353
6045
|
if (!text || typeof text !== 'string') return false;
|
|
@@ -6366,14 +6058,11 @@ class AIChatDialog extends HTMLElement {
|
|
|
6366
6058
|
} catch (e) {}
|
|
6367
6059
|
return false;
|
|
6368
6060
|
}
|
|
6369
|
-
|
|
6370
6061
|
// 判断错误标记内容(替换脆弱的前缀检查 startsWith('('))
|
|
6371
|
-
_ERROR_CONTENT_PATTERNS = ['(请求超时', '(等待AI处理', '(无回复)', '(已停止)', '(AI 处理超时', '错误:', '网络错误:', '抱歉,'];
|
|
6372
6062
|
_isErrorContent(content) {
|
|
6373
6063
|
if (!content) return true;
|
|
6374
6064
|
return this._ERROR_CONTENT_PATTERNS.some(p => content.startsWith(p));
|
|
6375
6065
|
}
|
|
6376
|
-
|
|
6377
6066
|
// P3: 将上下文对象格式化为可读文本(拼入 prompt 前缀)
|
|
6378
6067
|
_formatContextText(context) {
|
|
6379
6068
|
if (!context || typeof context !== 'object') return '';
|
|
@@ -6394,7 +6083,6 @@ class AIChatDialog extends HTMLElement {
|
|
|
6394
6083
|
}
|
|
6395
6084
|
return lines.join('\n');
|
|
6396
6085
|
}
|
|
6397
|
-
|
|
6398
6086
|
// 运行时事件管理 – incremental DOM, no renderMessages
|
|
6399
6087
|
_addRuntimeEvent(msgId, event) {
|
|
6400
6088
|
const idx = this._messages.findIndex(m => m.id === msgId);
|
|
@@ -6422,7 +6110,6 @@ class AIChatDialog extends HTMLElement {
|
|
|
6422
6110
|
if (this._shouldAutoScroll()) this.scrollToBottom();
|
|
6423
6111
|
return;
|
|
6424
6112
|
}
|
|
6425
|
-
|
|
6426
6113
|
// Path 2: first event – inject runtime panel DOM next to the streaming bubble
|
|
6427
6114
|
const msgContent = this._findStreamingMessageContent();
|
|
6428
6115
|
if (msgContent) {
|
|
@@ -6489,7 +6176,6 @@ class AIChatDialog extends HTMLElement {
|
|
|
6489
6176
|
panel.appendChild(eventsContainer);
|
|
6490
6177
|
return panel;
|
|
6491
6178
|
}
|
|
6492
|
-
|
|
6493
6179
|
// P4: 获取自定义操作按钮列表(缓存结果避免重复调用 onMessageActions)
|
|
6494
6180
|
_getCustomActions(msg, idx) {
|
|
6495
6181
|
const key = '_customActions_' + idx;
|
|
@@ -6503,7 +6189,6 @@ class AIChatDialog extends HTMLElement {
|
|
|
6503
6189
|
}
|
|
6504
6190
|
return msg[key] || [];
|
|
6505
6191
|
}
|
|
6506
|
-
|
|
6507
6192
|
// P4: 生成自定义操作按钮 HTML
|
|
6508
6193
|
_buildCustomActions(msg, idx) {
|
|
6509
6194
|
const actions = this._getCustomActions(msg, idx);
|
|
@@ -6529,7 +6214,7 @@ class AIChatDialog extends HTMLElement {
|
|
|
6529
6214
|
}
|
|
6530
6215
|
async _fetchConversationResult(conversationId) {
|
|
6531
6216
|
for (let i = 0; i < 8; i++) {
|
|
6532
|
-
await new Promise(r => setTimeout(r, 1000));
|
|
6217
|
+
await new Promise(r => window.setTimeout(r, 1000));
|
|
6533
6218
|
try {
|
|
6534
6219
|
const response = await this._chatClient.queryConversation({
|
|
6535
6220
|
conversationId,
|
|
@@ -6545,9 +6230,7 @@ class AIChatDialog extends HTMLElement {
|
|
|
6545
6230
|
}
|
|
6546
6231
|
return '(AI 处理超时,请稍后查看对话记录)';
|
|
6547
6232
|
}
|
|
6548
|
-
|
|
6549
6233
|
// ==================== 重新生成(与Vue regenerate 一致) ====================
|
|
6550
|
-
|
|
6551
6234
|
async regenerate(index) {
|
|
6552
6235
|
if (this._isLoading || this._streaming) return;
|
|
6553
6236
|
const aiMessage = this._messages[index];
|
|
@@ -6592,14 +6275,12 @@ class AIChatDialog extends HTMLElement {
|
|
|
6592
6275
|
toast.className = 'error-toast';
|
|
6593
6276
|
toast.textContent = msg;
|
|
6594
6277
|
this._dialog?.appendChild(toast);
|
|
6595
|
-
setTimeout(() => toast.remove(), 3000);
|
|
6278
|
+
window.setTimeout(() => toast.remove(), 3000);
|
|
6596
6279
|
}
|
|
6597
|
-
|
|
6598
6280
|
// P1-6: 精细化错误处理(与Vue版 handleErrorMsg 一致,按HTTP状态码+业务错误码分类)
|
|
6599
6281
|
handleError(err) {
|
|
6600
6282
|
let displayMsg = '抱歉,网络请求失败,请稍后重试';
|
|
6601
6283
|
const errMsg = err?.message || '';
|
|
6602
|
-
|
|
6603
6284
|
// 按错误类型分类
|
|
6604
6285
|
if (errMsg.includes('Failed to fetch') || errMsg.includes('NetworkError') || errMsg.includes('网络')) {
|
|
6605
6286
|
displayMsg = '网络连接失败,请检查网络后重试';
|
|
@@ -6633,9 +6314,7 @@ class AIChatDialog extends HTMLElement {
|
|
|
6633
6314
|
detail: err
|
|
6634
6315
|
}));
|
|
6635
6316
|
}
|
|
6636
|
-
|
|
6637
6317
|
// ==================== 复制(与Vue copyMessage 一致) ====================
|
|
6638
|
-
|
|
6639
6318
|
async copyContent(text) {
|
|
6640
6319
|
// P1-10: 微信环境优先使用 wx.setClipboardData(与Vue版一致)
|
|
6641
6320
|
if (typeof wx !== 'undefined' && wx.setClipboardData) {
|
|
@@ -6678,7 +6357,7 @@ class AIChatDialog extends HTMLElement {
|
|
|
6678
6357
|
await this.copyContent(content);
|
|
6679
6358
|
if (this._messages[idx]) this._messages[idx].copied = true;
|
|
6680
6359
|
this.renderMessages();
|
|
6681
|
-
setTimeout(() => {
|
|
6360
|
+
window.setTimeout(() => {
|
|
6682
6361
|
if (this._messages[idx]) {
|
|
6683
6362
|
this._messages[idx].copied = false;
|
|
6684
6363
|
this.renderMessages();
|
|
@@ -6727,9 +6406,7 @@ class AIChatDialog extends HTMLElement {
|
|
|
6727
6406
|
}
|
|
6728
6407
|
return response;
|
|
6729
6408
|
}
|
|
6730
|
-
|
|
6731
6409
|
// ==================== 工具函数 ====================
|
|
6732
|
-
|
|
6733
6410
|
generateId() {
|
|
6734
6411
|
return `msg_${Date.now()}_${Math.random().toString(36).substr(2, 9)}`;
|
|
6735
6412
|
}
|
|
@@ -6737,7 +6414,6 @@ class AIChatDialog extends HTMLElement {
|
|
|
6737
6414
|
const n = new Date();
|
|
6738
6415
|
return `${String(n.getHours()).padStart(2, '0')}:${String(n.getMinutes()).padStart(2, '0')}`;
|
|
6739
6416
|
}
|
|
6740
|
-
|
|
6741
6417
|
/**
|
|
6742
6418
|
* 调试日志(仅在 config.debug === true 时输出)
|
|
6743
6419
|
*/
|
|
@@ -6761,7 +6437,6 @@ class AIChatDialog extends HTMLElement {
|
|
|
6761
6437
|
if (!s) return '';
|
|
6762
6438
|
return s.replace(/&/g, '&').replace(/"/g, '"').replace(/'/g, ''').replace(/</g, '<').replace(/>/g, '>');
|
|
6763
6439
|
}
|
|
6764
|
-
|
|
6765
6440
|
/**
|
|
6766
6441
|
* 重置 _themeVars 为默认值(主题切换时调用,防止残留)
|
|
6767
6442
|
*/
|
|
@@ -6803,7 +6478,6 @@ class AIChatDialog extends HTMLElement {
|
|
|
6803
6478
|
'--ai-font-small': '11px'
|
|
6804
6479
|
});
|
|
6805
6480
|
}
|
|
6806
|
-
|
|
6807
6481
|
/**
|
|
6808
6482
|
* Markdown → HTML 解析(仅用于AI回复,用户消息保持纯文本)
|
|
6809
6483
|
* 使用 marked + DOMPurify 双重保障安全
|
|
@@ -6827,7 +6501,6 @@ class AIChatDialog extends HTMLElement {
|
|
|
6827
6501
|
}
|
|
6828
6502
|
}
|
|
6829
6503
|
}
|
|
6830
|
-
|
|
6831
6504
|
// 注册自定义元素
|
|
6832
6505
|
if (!customElements.get('ai-chat-dialog')) {
|
|
6833
6506
|
customElements.define('ai-chat-dialog', AIChatDialog);
|
|
@@ -6839,17 +6512,15 @@ if (typeof window !== 'undefined') {
|
|
|
6839
6512
|
/**
|
|
6840
6513
|
* IBC AI Web SDK - 纯JavaScript版本
|
|
6841
6514
|
* 支持任意前端框架:Vue, React, Angular, 原生HTML等
|
|
6842
|
-
*
|
|
6515
|
+
*
|
|
6843
6516
|
* @author IBC AI Team
|
|
6844
|
-
* @version 2.0
|
|
6517
|
+
* @version 2.1.0
|
|
6845
6518
|
*/
|
|
6846
|
-
|
|
6847
|
-
|
|
6519
|
+
// 导出组件
|
|
6848
6520
|
// 快速初始化函数
|
|
6849
6521
|
function createAIChatDialog(options = {}) {
|
|
6850
6522
|
// 创建自定义元素实例
|
|
6851
6523
|
const dialog = document.createElement('ai-chat-dialog');
|
|
6852
|
-
|
|
6853
6524
|
// 设置属性
|
|
6854
6525
|
if (options.title) dialog.setAttribute('title', options.title);
|
|
6855
6526
|
if (options.placeholder) dialog.setAttribute('placeholder', options.placeholder);
|
|
@@ -6858,7 +6529,6 @@ function createAIChatDialog(options = {}) {
|
|
|
6858
6529
|
if (options.height) dialog.style.setProperty('--ai-dialog-height', typeof options.height === 'number' ? `${options.height}px` : options.height);
|
|
6859
6530
|
if (options.top != null) dialog.style.setProperty('--ai-dialog-top', typeof options.top === 'number' ? `${options.top}px` : options.top);
|
|
6860
6531
|
if (options.right != null) dialog.style.setProperty('--ai-dialog-right', typeof options.right === 'number' ? `${options.right}px` : options.right);
|
|
6861
|
-
|
|
6862
6532
|
// 挂载到指定容器或 body
|
|
6863
6533
|
if (options.target) {
|
|
6864
6534
|
const container = typeof options.target === 'string' ? document.querySelector(options.target) : options.target;
|
|
@@ -6872,12 +6542,10 @@ function createAIChatDialog(options = {}) {
|
|
|
6872
6542
|
} else {
|
|
6873
6543
|
document.body.appendChild(dialog);
|
|
6874
6544
|
}
|
|
6875
|
-
|
|
6876
6545
|
// 初始化配置
|
|
6877
6546
|
if (Object.keys(options).length > 0) {
|
|
6878
6547
|
dialog.init(options);
|
|
6879
6548
|
}
|
|
6880
|
-
|
|
6881
6549
|
// 行内模式:挂载到 DOM 后直接展开
|
|
6882
6550
|
if (options.target) {
|
|
6883
6551
|
requestAnimationFrame(() => {
|
|
@@ -6902,14 +6570,12 @@ function createAIChatDialog(options = {}) {
|
|
|
6902
6570
|
}
|
|
6903
6571
|
return dialog;
|
|
6904
6572
|
}
|
|
6905
|
-
|
|
6906
6573
|
// Vue插件安装方式(向后兼容)
|
|
6907
6574
|
function installVuePlugin(Vue) {
|
|
6908
6575
|
if (!Vue) {
|
|
6909
6576
|
console.warn('[AI Web SDK] Vue instance not provided');
|
|
6910
6577
|
return;
|
|
6911
6578
|
}
|
|
6912
|
-
|
|
6913
6579
|
// 注册全局组件包装器
|
|
6914
6580
|
Vue.component('AiChatWrapper', {
|
|
6915
6581
|
props: {
|
|
@@ -6965,7 +6631,6 @@ function installVuePlugin(Vue) {
|
|
|
6965
6631
|
methods: {
|
|
6966
6632
|
initDialog() {
|
|
6967
6633
|
this.dialogInstance = createAIChatDialog(this.config || {});
|
|
6968
|
-
|
|
6969
6634
|
// 监听事件并转发
|
|
6970
6635
|
const events = ['open', 'close', 'message-send', 'message-received', 'error'];
|
|
6971
6636
|
events.forEach(event => {
|
|
@@ -6993,9 +6658,6 @@ function installVuePlugin(Vue) {
|
|
|
6993
6658
|
}
|
|
6994
6659
|
});
|
|
6995
6660
|
}
|
|
6996
|
-
|
|
6997
|
-
// React Hook(可选) — 通过 window.React 获取 hooks,兼容 CDN 全局引入
|
|
6998
|
-
/* global React */
|
|
6999
6661
|
function useAIChat(options = {}) {
|
|
7000
6662
|
if (typeof React === 'undefined') {
|
|
7001
6663
|
console.warn('[AI Web SDK] React is not loaded, useAIChat skipped');
|
|
@@ -7019,7 +6681,6 @@ function useAIChat(options = {}) {
|
|
|
7019
6681
|
useEffect(() => {
|
|
7020
6682
|
if (!dialogRef.current) {
|
|
7021
6683
|
dialogRef.current = createAIChatDialog(options);
|
|
7022
|
-
|
|
7023
6684
|
// 监听事件
|
|
7024
6685
|
dialogRef.current.addEventListener('message-send', e => {
|
|
7025
6686
|
setMessages(prev => [...prev, e.detail]);
|
|
@@ -7074,7 +6735,6 @@ function useAIChat(options = {}) {
|
|
|
7074
6735
|
ref: dialogRef
|
|
7075
6736
|
};
|
|
7076
6737
|
}
|
|
7077
|
-
|
|
7078
6738
|
// 默认导出
|
|
7079
6739
|
var index = {
|
|
7080
6740
|
AIChatDialog,
|
|
@@ -7082,14 +6742,13 @@ var index = {
|
|
|
7082
6742
|
createAIChatDialog,
|
|
7083
6743
|
installVuePlugin,
|
|
7084
6744
|
useAIChat,
|
|
7085
|
-
version: '2.0
|
|
6745
|
+
version: '2.1.0'
|
|
7086
6746
|
};
|
|
7087
|
-
|
|
7088
6747
|
// 全局暴露(UMD/浏览器环境)
|
|
7089
6748
|
if (typeof window !== 'undefined') {
|
|
7090
6749
|
window.AICreateChatDialog = createAIChatDialog;
|
|
7091
6750
|
window.AIWebSDK = {
|
|
7092
|
-
version: '2.0
|
|
6751
|
+
version: '2.1.0',
|
|
7093
6752
|
create: createAIChatDialog,
|
|
7094
6753
|
AIChatDialog,
|
|
7095
6754
|
AIChatClient,
|