diff --git a/bbcode/parser.ts b/bbcode/parser.ts index 7983ec3..c5e026b 100644 --- a/bbcode/parser.ts +++ b/bbcode/parser.ts @@ -127,7 +127,10 @@ export class BBCodeParser { isAllowed = (name) => self.isAllowed(name) && parentAllowed(name); currentTag = this._currentTag = {tag: self.tag, line: this._line, column: this._column}; } - let tagStart = -1, paramStart = -1, mark = start; + let tagStart = -1, + paramStart = -1, + mark = start, + isInCollapseParam = false; for(let i = start; i < input.length; ++i) { const c = input[i]; ++this._column; @@ -135,12 +138,22 @@ export class BBCodeParser { ++this._line; this._column = 1; } - if(c === '[') { + if(c === '[' && !isInCollapseParam) { tagStart = i; paramStart = -1; - } else if(c === '=' && paramStart === -1) - paramStart = i; + } else if(c === '=' && paramStart === -1) { + paramStart = i; + + const paramIndex = paramStart === -1 ? i : paramStart; + let tagKey = input + .substring(tagStart + 1, paramIndex) + .trim() + .toLowerCase(); + + if (tagKey == "collapse") isInCollapseParam = true; + } else if(c === ']') { + const paramIndex = paramStart === -1 ? i : paramStart; let tagKey = input.substring(tagStart + 1, paramIndex).trim().toLowerCase(); if(tagKey.length === 0) { @@ -154,6 +167,17 @@ export class BBCodeParser { tagStart = -1; continue; } + if (isInCollapseParam) { + let fullTagKey = input + .substring(tagStart + 1, i + 1) + .trim() + .toLowerCase(); + if (fullTagKey.endsWith("[hr]")) { + continue; + } + } + + isInCollapseParam = false; if(!close) { const tag = this._tags[tagKey]!; const allowed = isAllowed(tagKey); diff --git a/bbcode/standard.ts b/bbcode/standard.ts index 80f851b..1d677d9 100644 --- a/bbcode/standard.ts +++ b/bbcode/standard.ts @@ -73,7 +73,11 @@ export class StandardBBCodeParser extends CoreBBCodeParser { icon.className = 'fas fa-chevron-down'; icon.style.marginRight = '10px'; headerText.appendChild(icon); - headerText.appendChild(document.createTextNode(param)); + // HACK: to allow [hr] in header part + param = Utils.replaceAll(param, "[hr]", "
"); + const headerText2 = parser.createElement("div"); + headerText2.innerHTML = param; + headerText.appendChild(headerText2); outer.appendChild(headerText); const body = parser.createElement('div'); body.className = 'bbcode-collapse-body'; diff --git a/site/utils.ts b/site/utils.ts index bf4d313..11975ab 100644 --- a/site/utils.ts +++ b/site/utils.ts @@ -82,3 +82,13 @@ export function init(s: Settings, c: SimpleCharacter[]): void { settings = s; characters = c; } + +function escapeRegExp(string: string): +string { + return string.replace(/[.*+?^${}()|[\]\\]/g, "\\$&"); // $& means the whole matched string +} + +export function replaceAll(str: string, find: string, replace: string): +string { + return str.replace(new RegExp(escapeRegExp(find), "g"), replace); +} \ No newline at end of file