/** * SCEditor BBCode Plugin * http://www.sceditor.com/ * * Copyright (C) 2011-2017, Sam Clarke (samclarke.com) * * SCEditor is licensed under the MIT license: * http://www.opensource.org/licenses/mit-license.php * * @fileoverview SCEditor BBCode Format * @author Sam Clarke */ (function (sceditor) { /*eslint max-depth: off*/ 'use strict'; var escapeEntities = sceditor.escapeEntities; var escapeUriScheme = sceditor.escapeUriScheme; var dom = sceditor.dom; var utils = sceditor.utils; var css = dom.css; var attr = dom.attr; var is = dom.is; var extend = utils.extend; var each = utils.each; var IE_VER = sceditor.ie; // In IE < 11 a BR at the end of a block level element // causes a double line break. var IE_BR_FIX = IE_VER && IE_VER < 11; var EMOTICON_DATA_ATTR = 'data-sceditor-emoticon'; var getEditorCommand = sceditor.command.get; var QuoteType = { /** @lends BBCodeParser.QuoteType */ /** * Always quote the attribute value * @type {Number} */ always: 1, /** * Never quote the attributes value * @type {Number} */ never: 2, /** * Only quote the attributes value when it contains spaces to equals * @type {Number} */ auto: 3 }; var defaultCommandsOverrides = { bold: { txtExec: ['[b]', '[/b]'] }, italic: { txtExec: ['[i]', '[/i]'] }, underline: { txtExec: ['[u]', '[/u]'] }, strike: { txtExec: ['[s]', '[/s]'] }, subscript: { txtExec: ['[sub]', '[/sub]'] }, superscript: { txtExec: ['[sup]', '[/sup]'] }, left: { txtExec: ['[left]', '[/left]'] }, center: { txtExec: ['[center]', '[/center]'] }, right: { txtExec: ['[right]', '[/right]'] }, justify: { txtExec: ['[justify]', '[/justify]'] }, font: { txtExec: function (caller) { var editor = this; getEditorCommand('font')._dropDown( editor, caller, function (fontName) { editor.insertText( '[font=' + fontName + ']', '[/font]' ); } ); } }, size: { txtExec: function (caller) { var editor = this; getEditorCommand('size')._dropDown( editor, caller, function (fontSize) { editor.insertText( '[size=' + fontSize + ']', '[/size]' ); } ); } }, color: { txtExec: function (caller) { var editor = this; getEditorCommand('color')._dropDown( editor, caller, function (color) { editor.insertText( '[color=' + color + ']', '[/color]' ); } ); } }, bulletlist: { txtExec: function (caller, selected) { var content = ''; each(selected.split(/\r?\n/), function () { content += (content ? '\n' : '') + '[li]' + this + '[/li]'; }); this.insertText('[ul]\n' + content + '\n[/ul]'); } }, orderedlist: { txtExec: function (caller, selected) { var content = ''; each(selected.split(/\r?\n/), function () { content += (content ? '\n' : '') + '[li]' + this + '[/li]'; }); this.insertText('[ol]\n' + content + '\n[/ol]'); } }, table: { txtExec: ['[table][tr][td]', '[/td][/tr][/table]'] }, horizontalrule: { txtExec: ['[hr]'] }, code: { txtExec: ['[code]', '[/code]'] }, image: { txtExec: function (caller, selected) { var editor = this; getEditorCommand('image')._dropDown( editor, caller, selected, function (url, width, height) { var attrs = ''; if (width) { attrs += ' width=' + width; } if (height) { attrs += ' height=' + height; } editor.insertText( '[img' + attrs + ']' + url + '[/img]' ); } ); } }, email: { txtExec: function (caller, selected) { var editor = this; getEditorCommand('email')._dropDown( editor, caller, function (url, text) { editor.insertText( '[email=' + url + ']' + (text || selected || url) + '[/email]' ); } ); } }, link: { txtExec: function (caller, selected) { var editor = this; getEditorCommand('link')._dropDown( editor, caller, function (url, text) { editor.insertText( '[url=' + url + ']' + (text || selected || url) + '[/url]' ); } ); } }, quote: { txtExec: ['[quote]', '[/quote]'] }, youtube: { txtExec: function (caller) { var editor = this; getEditorCommand('youtube')._dropDown( editor, caller, function (id) { editor.insertText('[youtube]' + id + '[/youtube]'); } ); } }, rtl: { txtExec: ['[rtl]', '[/rtl]'] }, ltr: { txtExec: ['[ltr]', '[/ltr]'] } }; var bbcodeHandlers = { // START_COMMAND: Bold b: { tags: { b: null, strong: null }, styles: { // 401 is for FF 3.5 'font-weight': ['bold', 'bolder', '401', '700', '800', '900'] }, format: '[b]{0}[/b]', html: '{0}' }, // END_COMMAND // START_COMMAND: Italic i: { tags: { i: null, em: null }, styles: { 'font-style': ['italic', 'oblique'] }, format: '[i]{0}[/i]', html: '{0}' }, // END_COMMAND // START_COMMAND: Underline u: { tags: { u: null }, styles: { 'text-decoration': ['underline'] }, format: '[u]{0}[/u]', html: '{0}' }, // END_COMMAND // START_COMMAND: Strikethrough s: { tags: { s: null, strike: null }, styles: { 'text-decoration': ['line-through'] }, format: '[s]{0}[/s]', html: '{0}' }, // END_COMMAND // START_COMMAND: Subscript sub: { tags: { sub: null }, format: '[sub]{0}[/sub]', html: '{0}' }, // END_COMMAND // START_COMMAND: Superscript sup: { tags: { sup: null }, format: '[sup]{0}[/sup]', html: '{0}' }, // END_COMMAND // START_COMMAND: Font font: { tags: { font: { face: null } }, styles: { 'font-family': null }, quoteType: QuoteType.never, format: function (element, content) { var font; if (!is(element, 'font') || !(font = attr(element, 'face'))) { font = css(element, 'font-family'); } return '[font=' + _stripQuotes(font) + ']' + content + '[/font]'; }, html: '{0}' }, // END_COMMAND // START_COMMAND: Size size: { tags: { font: { size: null } }, styles: { 'font-size': null }, format: function (element, content) { var fontSize = attr(element, 'size'), size = 2; if (!fontSize) { fontSize = css(element, 'fontSize'); } // Most browsers return px value but IE returns 1-7 if (fontSize.indexOf('px') > -1) { // convert size to an int fontSize = fontSize.replace('px', '') - 0; if (fontSize < 12) { size = 1; } if (fontSize > 15) { size = 3; } if (fontSize > 17) { size = 4; } if (fontSize > 23) { size = 5; } if (fontSize > 31) { size = 6; } if (fontSize > 47) { size = 7; } } else { size = fontSize; } return '[size=' + size + ']' + content + '[/size]'; }, html: '{!0}' }, // END_COMMAND // START_COMMAND: Color color: { tags: { font: { color: null } }, styles: { color: null }, quoteType: QuoteType.never, format: function (elm, content) { var color; if (!is(elm, 'font') || !(color = attr(elm, 'color'))) { color = elm.style.color || css(elm, 'color'); } return '[color=' + _normaliseColour(color) + ']' + content + '[/color]'; }, html: function (token, attrs, content) { return '' + content + ''; } }, // END_COMMAND // START_COMMAND: Lists ul: { tags: { ul: null }, breakStart: true, isInline: false, skipLastLineBreak: true, format: '[ul]{0}[/ul]', html: '' }, list: { breakStart: true, isInline: false, skipLastLineBreak: true, html: '' }, ol: { tags: { ol: null }, breakStart: true, isInline: false, skipLastLineBreak: true, format: '[ol]{0}[/ol]', html: '
    {0}
' }, li: { tags: { li: null }, isInline: false, closedBy: ['/ul', '/ol', '/list', '*', 'li'], format: '[li]{0}[/li]', html: '
  • {0}
  • ' }, '*': { isInline: false, closedBy: ['/ul', '/ol', '/list', '*', 'li'], html: '
  • {0}
  • ' }, // END_COMMAND // START_COMMAND: Table table: { tags: { table: null }, isInline: false, isHtmlInline: true, skipLastLineBreak: true, format: '[table]{0}[/table]', html: '{0}
    ' }, tr: { tags: { tr: null }, isInline: false, skipLastLineBreak: true, format: '[tr]{0}[/tr]', html: '{0}' }, th: { tags: { th: null }, allowsEmpty: true, isInline: false, format: '[th]{0}[/th]', html: '{0}' }, td: { tags: { td: null }, allowsEmpty: true, isInline: false, format: '[td]{0}[/td]', html: '{0}' }, // END_COMMAND // START_COMMAND: Emoticons emoticon: { allowsEmpty: true, tags: { img: { src: null, 'data-sceditor-emoticon': null } }, format: function (element, content) { return attr(element, EMOTICON_DATA_ATTR) + content; }, html: '{0}' }, // END_COMMAND // START_COMMAND: Horizontal Rule hr: { tags: { hr: null }, allowsEmpty: true, isSelfClosing: true, isInline: false, format: '[hr]{0}', html: '
    ' }, // END_COMMAND // START_COMMAND: Image img: { allowsEmpty: true, tags: { img: { src: null } }, allowedChildren: ['#'], quoteType: QuoteType.never, format: function (element, content) { var width, height, attribs = '', style = function (name) { return element.style ? element.style[name] : null; }; // check if this is an emoticon image if (attr(element, EMOTICON_DATA_ATTR)) { return content; } width = attr(element, 'width') || style('width'); height = attr(element, 'height') || style('height'); // only add width and height if one is specified if ((element.complete && (width || height)) || (width && height)) { attribs = '=' + dom.width(element) + 'x' + dom.height(element); } return '[img' + attribs + ']' + attr(element, 'src') + '[/img]'; }, html: function (token, attrs, content) { var undef, width, height, match, attribs = ''; // handle [img width=340 height=240]url[/img] width = attrs.width; height = attrs.height; // handle [img=340x240]url[/img] if (attrs.defaultattr) { match = attrs.defaultattr.split(/x/i); width = match[0]; height = (match.length === 2 ? match[1] : match[0]); } if (width !== undef) { attribs += ' width="' + escapeEntities(width, true) + '"'; } if (height !== undef) { attribs += ' height="' + escapeEntities(height, true) + '"'; } return ''; } }, // END_COMMAND // START_COMMAND: URL url: { allowsEmpty: true, tags: { a: { href: null } }, quoteType: QuoteType.never, format: function (element, content) { var url = attr(element, 'href'); // make sure this link is not an e-mail, // if it is return e-mail BBCode if (url.substr(0, 7) === 'mailto:') { return '[email="' + url.substr(7) + '"]' + content + '[/email]'; } return '[url=' + url + ']' + content + '[/url]'; }, html: function (token, attrs, content) { attrs.defaultattr = escapeEntities(attrs.defaultattr, true) || content; return '' + content + ''; } }, // END_COMMAND // START_COMMAND: E-mail email: { quoteType: QuoteType.never, html: function (token, attrs, content) { return '' + content + ''; } }, // END_COMMAND // START_COMMAND: Quote quote: { tags: { blockquote: null }, isInline: false, quoteType: QuoteType.never, format: function (element, content) { var authorAttr = 'data-author'; var author = ''; var cite; var children = element.children; for (var i = 0; !cite && i < children.length; i++) { if (is(children[i], 'cite')) { cite = children[i]; } } if (cite || attr(element, authorAttr)) { author = cite && cite.textContent || attr(element, authorAttr); attr(element, authorAttr, author); if (cite) { element.removeChild(cite); } content = this.elementToBbcode(element); author = '=' + author.replace(/(^\s+|\s+$)/g, ''); if (cite) { element.insertBefore(cite, element.firstChild); } } return '[quote' + author + ']' + content + '[/quote]'; }, html: function (token, attrs, content) { if (attrs.defaultattr) { content = '' + escapeEntities(attrs.defaultattr) + '' + content; } return '
    ' + content + '
    '; } }, // END_COMMAND // START_COMMAND: Code code: { tags: { code: null }, isInline: false, allowedChildren: ['#', '#newline'], format: '[code]{0}[/code]', html: '{0}' }, // END_COMMAND // START_COMMAND: Left left: { styles: { 'text-align': [ 'left', '-webkit-left', '-moz-left', '-khtml-left' ] }, isInline: false, format: '[left]{0}[/left]', html: '
    {0}
    ' }, // END_COMMAND // START_COMMAND: Centre center: { styles: { 'text-align': [ 'center', '-webkit-center', '-moz-center', '-khtml-center' ] }, isInline: false, format: '[center]{0}[/center]', html: '
    {0}
    ' }, // END_COMMAND // START_COMMAND: Right right: { styles: { 'text-align': [ 'right', '-webkit-right', '-moz-right', '-khtml-right' ] }, isInline: false, format: '[right]{0}[/right]', html: '
    {0}
    ' }, // END_COMMAND // START_COMMAND: Justify justify: { styles: { 'text-align': [ 'justify', '-webkit-justify', '-moz-justify', '-khtml-justify' ] }, isInline: false, format: '[justify]{0}[/justify]', html: '
    {0}
    ' }, // END_COMMAND // START_COMMAND: YouTube youtube: { allowsEmpty: true, tags: { iframe: { 'data-youtube-id': null } }, format: function (element, content) { element = attr(element, 'data-youtube-id'); return element ? '[youtube]' + element + '[/youtube]' : content; }, html: '' }, // END_COMMAND // START_COMMAND: Rtl rtl: { styles: { direction: ['rtl'] }, isInline: false, format: '[rtl]{0}[/rtl]', html: '
    {0}
    ' }, // END_COMMAND // START_COMMAND: Ltr ltr: { styles: { direction: ['ltr'] }, isInline: false, format: '[ltr]{0}[/ltr]', html: '
    {0}
    ' }, // END_COMMAND // this is here so that commands above can be removed // without having to remove the , after the last one. // Needed for IE. ignore: {} }; /** * Formats a string replacing {name} with the values of * obj.name properties. * * If there is no property for the specified {name} then * it will be left intact. * * @param {string} str * @param {Object} obj * @return {string} * @since 2.0.0 */ function formatBBCodeString(str, obj) { return str.replace(/\{([^}]+)\}/g, function (match, group) { var undef, escape = true; if (group.charAt(0) === '!') { escape = false; group = group.substring(1); } if (group === '0') { escape = false; } if (obj[group] === undef) { return match; } return escape ? escapeEntities(obj[group], true) : obj[group]; }); } /** * Removes the first and last divs from the HTML. * * This is needed for pasting * @param {string} html * @return {string} * @private */ function removeFirstLastDiv(html) { var node, next, removeDiv, output = document.createElement('div'); removeDiv = function (node, isFirst) { // Don't remove divs that have styling if (dom.hasStyling(node)) { return; } if (IE_BR_FIX || (node.childNodes.length !== 1 || !is(node.firstChild, 'br'))) { while ((next = node.firstChild)) { output.insertBefore(next, node); } } if (isFirst) { var lastChild = output.lastChild; if (node !== lastChild && is(lastChild, 'div') && node.nextSibling === lastChild) { output.insertBefore(document.createElement('br'), node); } } output.removeChild(node); }; css(output, 'display', 'none'); output.innerHTML = html.replace(/<\/div>\n/g, ''); if ((node = output.firstChild) && is(node, 'div')) { removeDiv(node, true); } if ((node = output.lastChild) && is(node, 'div')) { removeDiv(node); } return output.innerHTML; } function isFunction(fn) { return typeof fn === 'function'; } /** * Removes any leading or trailing quotes ('") * * @return string * @since v1.4.0 */ function _stripQuotes(str) { return str ? str.replace(/\\(.)/g, '$1').replace(/^(["'])(.*?)\1$/, '$2') : str; } /** * Formats a string replacing {0}, {1}, {2}, ect. with * the params provided * * @param {string} str The string to format * @param {...string} arg The strings to replace * @return {string} * @since v1.4.0 */ function _formatString(str) { var undef; var args = arguments; return str.replace(/\{(\d+)\}/g, function (_, matchNum) { return args[matchNum - 0 + 1] !== undef ? args[matchNum - 0 + 1] : '{' + matchNum + '}'; }); } var TOKEN_OPEN = 'open'; var TOKEN_CONTENT = 'content'; var TOKEN_NEWLINE = 'newline'; var TOKEN_CLOSE = 'close'; /* * @typedef {Object} TokenizeToken * @property {string} type * @property {string} name * @property {string} val * @property {Object.} attrs * @property {array} children * @property {TokenizeToken} closing */ /** * Tokenize token object * * @param {string} type The type of token this is, * should be one of tokenType * @param {string} name The name of this token * @param {string} val The originally matched string * @param {array} attrs Any attributes. Only set on * TOKEN_TYPE_OPEN tokens * @param {array} children Any children of this token * @param {TokenizeToken} closing This tokens closing tag. * Only set on TOKEN_TYPE_OPEN tokens * @class {TokenizeToken} * @name {TokenizeToken} * @memberOf BBCodeParser.prototype */ // eslint-disable-next-line max-params function TokenizeToken(type, name, val, attrs, children, closing) { var base = this; base.type = type; base.name = name; base.val = val; base.attrs = attrs || {}; base.children = children || []; base.closing = closing || null; }; TokenizeToken.prototype = { /** @lends BBCodeParser.prototype.TokenizeToken */ /** * Clones this token * * @return {TokenizeToken} */ clone: function () { var base = this; return new TokenizeToken( base.type, base.name, base.val, extend({}, base.attrs), [], base.closing ? base.closing.clone() : null ); }, /** * Splits this token at the specified child * * @param {TokenizeToken} splitAt The child to split at * @return {TokenizeToken} The right half of the split token or * empty clone if invalid splitAt lcoation */ splitAt: function (splitAt) { var offsetLength; var base = this; var clone = base.clone(); var offset = base.children.indexOf(splitAt); if (offset > -1) { // Work out how many items are on the right side of the split // to pass to splice() offsetLength = base.children.length - offset; clone.children = base.children.splice(offset, offsetLength); } return clone; } }; /** * SCEditor BBCode parser class * * @param {Object} options * @class BBCodeParser * @name BBCodeParser * @since v1.4.0 */ function BBCodeParser(options) { var base = this; base.opts = extend({}, BBCodeParser.defaults, options); /** * Takes a BBCode string and splits it into open, * content and close tags. * * It does no checking to verify a tag has a matching open * or closing tag or if the tag is valid child of any tag * before it. For that the tokens should be passed to the * parse function. * * @param {string} str * @return {array} * @memberOf BBCodeParser.prototype */ base.tokenize = function (str) { var matches, type, i; var tokens = []; // The token types in reverse order of precedence // (they're looped in reverse) var tokenTypes = [ { type: TOKEN_CONTENT, regex: /^([^\[\r\n]+|\[)/ }, { type: TOKEN_NEWLINE, regex: /^(\r\n|\r|\n)/ }, { type: TOKEN_OPEN, regex: /^\[[^\[\]]+\]/ }, // Close must come before open as they are // the same except close has a / at the start. { type: TOKEN_CLOSE, regex: /^\[\/[^\[\]]+\]/ } ]; strloop: while (str.length) { i = tokenTypes.length; while (i--) { type = tokenTypes[i].type; // Check if the string matches any of the tokens if (!(matches = str.match(tokenTypes[i].regex)) || !matches[0]) { continue; } // Add the match to the tokens list tokens.push(tokenizeTag(type, matches[0])); // Remove the match from the string str = str.substr(matches[0].length); // The token has been added so start again continue strloop; } // If there is anything left in the string which doesn't match // any of the tokens then just assume it's content and add it. if (str.length) { tokens.push(tokenizeTag(TOKEN_CONTENT, str)); } str = ''; } return tokens; }; /** * Extracts the name an params from a tag * * @param {string} type * @param {string} val * @return {Object} * @private */ function tokenizeTag(type, val) { var matches, attrs, name, openRegex = /\[([^\]\s=]+)(?:([^\]]+))?\]/, closeRegex = /\[\/([^\[\]]+)\]/; // Extract the name and attributes from opening tags and // just the name from closing tags. if (type === TOKEN_OPEN && (matches = val.match(openRegex))) { name = lower(matches[1]); if (matches[2] && (matches[2] = matches[2].trim())) { attrs = tokenizeAttrs(matches[2]); } } if (type === TOKEN_CLOSE && (matches = val.match(closeRegex))) { name = lower(matches[1]); } if (type === TOKEN_NEWLINE) { name = '#newline'; } // Treat all tokens without a name and // all unknown BBCodes as content if (!name || ((type === TOKEN_OPEN || type === TOKEN_CLOSE) && !bbcodeHandlers[name])) { type = TOKEN_CONTENT; name = '#'; } return new TokenizeToken(type, name, val, attrs); } /** * Extracts the individual attributes from a string containing * all the attributes. * * @param {string} attrs * @return {Object} Assoc array of attributes * @private */ function tokenizeAttrs(attrs) { var matches, /* ([^\s=]+) Anything that's not a space or equals = Equals sign = (?: (?: (["']) The opening quote ( (?:\\\2|[^\2])*? Anything that isn't the unescaped opening quote ) \2 The opening quote again which will close the string ) | If not a quoted string then match ( (?:.(?!\s\S+=))*.? Anything that isn't part of [space][non-space][=] which would be a new attribute ) ) */ attrRegex = /([^\s=]+)=(?:(?:(["'])((?:\\\2|[^\2])*?)\2)|((?:.(?!\s\S+=))*.))/g, ret = {}; // if only one attribute then remove the = from the start and // strip any quotes if (attrs.charAt(0) === '=' && attrs.indexOf('=', 1) < 0) { ret.defaultattr = _stripQuotes(attrs.substr(1)); } else { if (attrs.charAt(0) === '=') { attrs = 'defaultattr' + attrs; } // No need to strip quotes here, the regex will do that. while ((matches = attrRegex.exec(attrs))) { ret[lower(matches[1])] = _stripQuotes(matches[3]) || matches[4]; } } return ret; } /** * Parses a string into an array of BBCodes * * @param {string} str * @param {boolean} preserveNewLines If to preserve all new lines, not * strip any based on the passed * formatting options * @return {array} Array of BBCode objects * @memberOf BBCodeParser.prototype */ base.parse = function (str, preserveNewLines) { var ret = parseTokens(base.tokenize(str)); var opts = base.opts; if (opts.fixInvalidNesting) { fixNesting(ret); } normaliseNewLines(ret, null, preserveNewLines); if (opts.removeEmptyTags) { removeEmpty(ret); } return ret; }; /** * Checks if an array of TokenizeToken's contains the * specified token. * * Checks the tokens name and type match another tokens * name and type in the array. * * @param {string} name * @param {string} type * @param {array} arr * @return {Boolean} * @private */ function hasTag(name, type, arr) { var i = arr.length; while (i--) { if (arr[i].type === type && arr[i].name === name) { return true; } } return false; } /** * Checks if the child tag is allowed as one * of the parent tags children. * * @param {TokenizeToken} parent * @param {TokenizeToken} child * @return {Boolean} * @private */ function isChildAllowed(parent, child) { var parentBBCode = parent ? bbcodeHandlers[parent.name] : {}, allowedChildren = parentBBCode.allowedChildren; if (base.opts.fixInvalidChildren && allowedChildren) { return allowedChildren.indexOf(child.name || '#') > -1; } return true; } // TODO: Tidy this parseTokens() function up a bit. /** * Parses an array of tokens created by tokenize() * * @param {array} toks * @return {array} Parsed tokens * @see tokenize() * @private */ function parseTokens(toks) { var token, bbcode, curTok, clone, i, next, cloned = [], output = [], openTags = [], /** * Returns the currently open tag or undefined * @return {TokenizeToken} */ currentTag = function () { return last(openTags); }, /** * Adds a tag to either the current tags children * or to the output array. * @param {TokenizeToken} token * @private */ addTag = function (token) { if (currentTag()) { currentTag().children.push(token); } else { output.push(token); } }, /** * Checks if this tag closes the current tag * @param {string} name * @return {Void} */ closesCurrentTag = function (name) { return currentTag() && (bbcode = bbcodeHandlers[currentTag().name]) && bbcode.closedBy && bbcode.closedBy.indexOf(name) > -1; }; while ((token = toks.shift())) { next = toks[0]; /* * Fixes any invalid children. * * If it is an element which isn't allowed as a child of it's * parent then it will be converted to content of the parent * element. i.e. * [code]Code [b]only[/b] allows text.[/code] * Will become: * Code [b]only[/b] allows text. * Instead of: * Code only allows text. */ // Ignore tags that can't be children if (!isChildAllowed(currentTag(), token)) { // exclude closing tags of current tag if (token.type !== TOKEN_CLOSE || !currentTag() || token.name !== currentTag().name) { token.name = '#'; token.type = TOKEN_CONTENT; } } switch (token.type) { case TOKEN_OPEN: // Check it this closes a parent, // e.g. for lists [*]one [*]two if (closesCurrentTag(token.name)) { openTags.pop(); } addTag(token); bbcode = bbcodeHandlers[token.name]; // If this tag is not self closing and it has a closing // tag then it is open and has children so add it to the // list of open tags. If has the closedBy property then // it is closed by other tags so include everything as // it's children until one of those tags is reached. if (bbcode && !bbcode.isSelfClosing && (bbcode.closedBy || hasTag(token.name, TOKEN_CLOSE, toks))) { openTags.push(token); } else if (!bbcode || !bbcode.isSelfClosing) { token.type = TOKEN_CONTENT; } break; case TOKEN_CLOSE: // check if this closes the current tag, // e.g. [/list] would close an open [*] if (currentTag() && token.name !== currentTag().name && closesCurrentTag('/' + token.name)) { openTags.pop(); } // If this is closing the currently open tag just pop // the close tag off the open tags array if (currentTag() && token.name === currentTag().name) { currentTag().closing = token; openTags.pop(); // If this is closing an open tag that is the parent of // the current tag then clone all the tags including the // current one until reaching the parent that is being // closed. Close the parent and then add the clones back // in. } else if (hasTag(token.name, TOKEN_OPEN, openTags)) { // Remove the tag from the open tags while ((curTok = openTags.pop())) { // If it's the tag that is being closed then // discard it and break the loop. if (curTok.name === token.name) { curTok.closing = token; break; } // Otherwise clone this tag and then add any // previously cloned tags as it's children clone = curTok.clone(); if (cloned.length) { clone.children.push(last(cloned)); } cloned.push(clone); } // Place block linebreak before cloned tags if (next && next.type === TOKEN_NEWLINE) { bbcode = bbcodeHandlers[token.name]; if (bbcode && bbcode.isInline === false) { addTag(next); toks.shift(); } } // Add the last cloned child to the now current tag // (the parent of the tag which was being closed) addTag(last(cloned)); // Add all the cloned tags to the open tags list i = cloned.length; while (i--) { openTags.push(cloned[i]); } cloned.length = 0; // This tag is closing nothing so treat it as content } else { token.type = TOKEN_CONTENT; addTag(token); } break; case TOKEN_NEWLINE: // handle things like // [*]list\nitem\n[*]list1 // where it should come out as // [*]list\nitem[/*]\n[*]list1[/*] // instead of // [*]list\nitem\n[/*][*]list1[/*] if (currentTag() && next && closesCurrentTag( (next.type === TOKEN_CLOSE ? '/' : '') + next.name )) { // skip if the next tag is the closing tag for // the option tag, i.e. [/*] if (!(next.type === TOKEN_CLOSE && next.name === currentTag().name)) { bbcode = bbcodeHandlers[currentTag().name]; if (bbcode && bbcode.breakAfter) { openTags.pop(); } else if (bbcode && bbcode.isInline === false && base.opts.breakAfterBlock && bbcode.breakAfter !== false) { openTags.pop(); } } } addTag(token); break; default: // content addTag(token); break; } } return output; } /** * Normalise all new lines * * Removes any formatting new lines from the BBCode * leaving only content ones. I.e. for a list: * * [list] * [*] list item one * with a line break * [*] list item two * [/list] * * would become * * [list] [*] list item one * with a line break [*] list item two [/list] * * Which makes it easier to convert to HTML or add * the formatting new lines back in when converting * back to BBCode * * @param {array} children * @param {TokenizeToken} parent * @param {boolean} onlyRemoveBreakAfter * @return {void} */ function normaliseNewLines(children, parent, onlyRemoveBreakAfter) { var token, left, right, parentBBCode, bbcode, removedBreakEnd, removedBreakBefore, remove; var childrenLength = children.length; // TODO: this function really needs tidying up if (parent) { parentBBCode = bbcodeHandlers[parent.name]; } var i = childrenLength; while (i--) { if (!(token = children[i])) { continue; } if (token.type === TOKEN_NEWLINE) { left = i > 0 ? children[i - 1] : null; right = i < childrenLength - 1 ? children[i + 1] : null; remove = false; // Handle the start and end new lines // e.g. [tag]\n and \n[/tag] if (!onlyRemoveBreakAfter && parentBBCode && parentBBCode.isSelfClosing !== true) { // First child of parent so must be opening line break // (breakStartBlock, breakStart) e.g. [tag]\n if (!left) { if (parentBBCode.isInline === false && base.opts.breakStartBlock && parentBBCode.breakStart !== false) { remove = true; } if (parentBBCode.breakStart) { remove = true; } // Last child of parent so must be end line break // (breakEndBlock, breakEnd) // e.g. \n[/tag] // remove last line break (breakEndBlock, breakEnd) } else if (!removedBreakEnd && !right) { if (parentBBCode.isInline === false && base.opts.breakEndBlock && parentBBCode.breakEnd !== false) { remove = true; } if (parentBBCode.breakEnd) { remove = true; } removedBreakEnd = remove; } } if (left && left.type === TOKEN_OPEN) { if ((bbcode = bbcodeHandlers[left.name])) { if (!onlyRemoveBreakAfter) { if (bbcode.isInline === false && base.opts.breakAfterBlock && bbcode.breakAfter !== false) { remove = true; } if (bbcode.breakAfter) { remove = true; } } else if (bbcode.isInline === false) { remove = true; } } } if (!onlyRemoveBreakAfter && !removedBreakBefore && right && right.type === TOKEN_OPEN) { if ((bbcode = bbcodeHandlers[right.name])) { if (bbcode.isInline === false && base.opts.breakBeforeBlock && bbcode.breakBefore !== false) { remove = true; } if (bbcode.breakBefore) { remove = true; } removedBreakBefore = remove; if (remove) { children.splice(i, 1); continue; } } } if (remove) { children.splice(i, 1); } // reset double removedBreakBefore removal protection. // This is needed for cases like \n\n[\tag] where // only 1 \n should be removed but without this they both // would be. removedBreakBefore = false; } else if (token.type === TOKEN_OPEN) { normaliseNewLines(token.children, token, onlyRemoveBreakAfter); } } } /** * Fixes any invalid nesting. * * If it is a block level element inside 1 or more inline elements * then those inline elements will be split at the point where the * block level is and the block level element placed between the split * parts. i.e. * [inline]A[blocklevel]B[/blocklevel]C[/inline] * Will become: * [inline]A[/inline][blocklevel]B[/blocklevel][inline]C[/inline] * * @param {array} children * @param {array} [parents] Null if there is no parents * @param {boolea} [insideInline] If inside an inline element * @param {array} [rootArr] Root array if there is one * @return {array} * @private */ function fixNesting(children, parents, insideInline, rootArr) { var token, i, parent, parentIndex, parentParentChildren, right; var isInline = function (token) { var bbcode = bbcodeHandlers[token.name]; return !bbcode || bbcode.isInline !== false; }; parents = parents || []; rootArr = rootArr || children; // This must check the length each time as it can change when // tokens are moved to fix the nesting. for (i = 0; i < children.length; i++) { if (!(token = children[i]) || token.type !== TOKEN_OPEN) { continue; } if (insideInline && !isInline(token)) { // if this is a blocklevel element inside an inline one then // split the parent at the block level element parent = last(parents); right = parent.splitAt(token); parentParentChildren = parents.length > 1 ? parents[parents.length - 2].children : rootArr; // If parent inline is allowed inside this tag, clone it and // wrap this tags children in it. if (isChildAllowed(token, parent)) { var clone = parent.clone(); clone.children = token.children; token.children = [clone]; } parentIndex = parentParentChildren.indexOf(parent); if (parentIndex > -1) { // remove the block level token from the right side of // the split inline element right.children.splice(0, 1); // insert the block level token and the right side after // the left side of the inline token parentParentChildren.splice( parentIndex + 1, 0, token, right ); // If token is a block and is followed by a newline, // then move the newline along with it to the new parent var next = right.children[0]; if (next && next.type === TOKEN_NEWLINE) { if (!isInline(token)) { right.children.splice(0, 1); parentParentChildren.splice( parentIndex + 2, 0, next ); } } // return to parents loop as the // children have now increased return; } } parents.push(token); fixNesting( token.children, parents, insideInline || isInline(token), rootArr ); parents.pop(); } } /** * Removes any empty BBCodes which are not allowed to be empty. * * @param {array} tokens * @private */ function removeEmpty(tokens) { var token, bbcode; /** * Checks if all children are whitespace or not * @private */ var isTokenWhiteSpace = function (children) { var j = children.length; while (j--) { var type = children[j].type; if (type === TOKEN_OPEN || type === TOKEN_CLOSE) { return false; } if (type === TOKEN_CONTENT && /\S|\u00A0/.test(children[j].val)) { return false; } } return true; }; var i = tokens.length; while (i--) { // So skip anything that isn't a tag since only tags can be // empty, content can't if (!(token = tokens[i]) || token.type !== TOKEN_OPEN) { continue; } bbcode = bbcodeHandlers[token.name]; // Remove any empty children of this tag first so that if they // are all removed this one doesn't think it's not empty. removeEmpty(token.children); if (isTokenWhiteSpace(token.children) && bbcode && !bbcode.isSelfClosing && !bbcode.allowsEmpty) { tokens.splice.apply(tokens, [i, 1].concat(token.children)); } } } /** * Converts a BBCode string to HTML * * @param {string} str * @param {boolean} preserveNewLines If to preserve all new lines, not * strip any based on the passed * formatting options * @return {string} * @memberOf BBCodeParser.prototype */ base.toHTML = function (str, preserveNewLines) { return convertToHTML(base.parse(str, preserveNewLines), true); }; /** * @private */ function convertToHTML(tokens, isRoot) { var undef, token, bbcode, content, html, needsBlockWrap, blockWrapOpen, isInline, lastChild, ret = []; isInline = function (bbcode) { return (!bbcode || (bbcode.isHtmlInline !== undef ? bbcode.isHtmlInline : bbcode.isInline)) !== false; }; while (tokens.length > 0) { if (!(token = tokens.shift())) { continue; } if (token.type === TOKEN_OPEN) { lastChild = token.children[token.children.length - 1] || {}; bbcode = bbcodeHandlers[token.name]; needsBlockWrap = isRoot && isInline(bbcode); content = convertToHTML(token.children, false); if (bbcode && bbcode.html) { // Only add a line break to the end if this is // blocklevel and the last child wasn't block-level if (!isInline(bbcode) && isInline(bbcodeHandlers[lastChild.name]) && !bbcode.isPreFormatted && !bbcode.skipLastLineBreak) { // Add placeholder br to end of block level elements // in all browsers apart from IE < 9 which handle // new lines differently and doesn't need one. if (!IE_BR_FIX) { content += '
    '; } } if (!isFunction(bbcode.html)) { token.attrs['0'] = content; html = formatBBCodeString( bbcode.html, token.attrs ); } else { html = bbcode.html.call( base, token, token.attrs, content ); } } else { html = token.val + content + (token.closing ? token.closing.val : ''); } } else if (token.type === TOKEN_NEWLINE) { if (!isRoot) { ret.push('
    '); continue; } // If not already in a block wrap then start a new block if (!blockWrapOpen) { ret.push('
    '); } // Putting BR in a div in IE causes it // to do a double line break. if (!IE_BR_FIX) { ret.push('
    '); } // Normally the div acts as a line-break with by moving // whatever comes after onto a new line. // If this is the last token, add an extra line-break so it // shows as there will be nothing after it. if (!tokens.length) { ret.push('
    '); } ret.push('
    \n'); blockWrapOpen = false; continue; // content } else { needsBlockWrap = isRoot; html = escapeEntities(token.val, true); } if (needsBlockWrap && !blockWrapOpen) { ret.push('
    '); blockWrapOpen = true; } else if (!needsBlockWrap && blockWrapOpen) { ret.push('
    \n'); blockWrapOpen = false; } ret.push(html); } if (blockWrapOpen) { ret.push('\n'); } return ret.join(''); } /** * Takes a BBCode string, parses it then converts it back to BBCode. * * This will auto fix the BBCode and format it with the specified * options. * * @param {string} str * @param {boolean} preserveNewLines If to preserve all new lines, not * strip any based on the passed * formatting options * @return {string} * @memberOf BBCodeParser.prototype */ base.toBBCode = function (str, preserveNewLines) { return convertToBBCode(base.parse(str, preserveNewLines)); }; /** * Converts parsed tokens back into BBCode with the * formatting specified in the options and with any * fixes specified. * * @param {array} toks Array of parsed tokens from base.parse() * @return {string} * @private */ function convertToBBCode(toks) { var token, attr, bbcode, isBlock, isSelfClosing, quoteType, breakBefore, breakStart, breakEnd, breakAfter, // Create an array of strings which are joined together // before being returned as this is faster in slow browsers. // (Old versions of IE). ret = []; while (toks.length > 0) { if (!(token = toks.shift())) { continue; } // TODO: tidy this bbcode = bbcodeHandlers[token.name]; isBlock = !(!bbcode || bbcode.isInline !== false); isSelfClosing = bbcode && bbcode.isSelfClosing; breakBefore = (isBlock && base.opts.breakBeforeBlock && bbcode.breakBefore !== false) || (bbcode && bbcode.breakBefore); breakStart = (isBlock && !isSelfClosing && base.opts.breakStartBlock && bbcode.breakStart !== false) || (bbcode && bbcode.breakStart); breakEnd = (isBlock && base.opts.breakEndBlock && bbcode.breakEnd !== false) || (bbcode && bbcode.breakEnd); breakAfter = (isBlock && base.opts.breakAfterBlock && bbcode.breakAfter !== false) || (bbcode && bbcode.breakAfter); quoteType = (bbcode ? bbcode.quoteType : null) || base.opts.quoteType || QuoteType.auto; if (!bbcode && token.type === TOKEN_OPEN) { ret.push(token.val); if (token.children) { ret.push(convertToBBCode(token.children)); } if (token.closing) { ret.push(token.closing.val); } } else if (token.type === TOKEN_OPEN) { if (breakBefore) { ret.push('\n'); } // Convert the tag and it's attributes to BBCode ret.push('[' + token.name); if (token.attrs) { if (token.attrs.defaultattr) { ret.push('=', quote( token.attrs.defaultattr, quoteType, 'defaultattr' )); delete token.attrs.defaultattr; } for (attr in token.attrs) { if (token.attrs.hasOwnProperty(attr)) { ret.push(' ', attr, '=', quote(token.attrs[attr], quoteType, attr)); } } } ret.push(']'); if (breakStart) { ret.push('\n'); } // Convert the tags children to BBCode if (token.children) { ret.push(convertToBBCode(token.children)); } // add closing tag if not self closing if (!isSelfClosing && !bbcode.excludeClosing) { if (breakEnd) { ret.push('\n'); } ret.push('[/' + token.name + ']'); } if (breakAfter) { ret.push('\n'); } // preserve whatever was recognized as the // closing tag if it is a self closing tag if (token.closing && isSelfClosing) { ret.push(token.closing.val); } } else { ret.push(token.val); } } return ret.join(''); } /** * Quotes an attribute * * @param {string} str * @param {BBCodeParser.QuoteType} quoteType * @param {string} name * @return {string} * @private */ function quote(str, quoteType, name) { var needsQuotes = /\s|=/.test(str); if (isFunction(quoteType)) { return quoteType(str, name); } if (quoteType === QuoteType.never || (quoteType === QuoteType.auto && !needsQuotes)) { return str; } return '"' + str.replace('\\', '\\\\').replace('"', '\\"') + '"'; } /** * Returns the last element of an array or null * * @param {array} arr * @return {Object} Last element * @private */ function last(arr) { if (arr.length) { return arr[arr.length - 1]; } return null; } /** * Converts a string to lowercase. * * @param {string} str * @return {string} Lowercase version of str * @private */ function lower(str) { return str.toLowerCase(); } }; /** * Quote type * @type {Object} * @class QuoteType * @name BBCodeParser.QuoteType * @since v1.4.0 */ BBCodeParser.QuoteType = QuoteType; /** * Default BBCode parser options * @type {Object} */ BBCodeParser.defaults = { /** * If to add a new line before block level elements * * @type {Boolean} */ breakBeforeBlock: false, /** * If to add a new line after the start of block level elements * * @type {Boolean} */ breakStartBlock: false, /** * If to add a new line before the end of block level elements * * @type {Boolean} */ breakEndBlock: false, /** * If to add a new line after block level elements * * @type {Boolean} */ breakAfterBlock: true, /** * If to remove empty tags * * @type {Boolean} */ removeEmptyTags: true, /** * If to fix invalid nesting, * i.e. block level elements inside inline elements. * * @type {Boolean} */ fixInvalidNesting: true, /** * If to fix invalid children. * i.e. A tag which is inside a parent that doesn't * allow that type of tag. * * @type {Boolean} */ fixInvalidChildren: true, /** * Attribute quote type * * @type {BBCodeParser.QuoteType} * @since 1.4.1 */ quoteType: QuoteType.auto }; /** * Converts a number 0-255 to hex. * * Will return 00 if number is not a valid number. * * @param {any} number * @return {string} * @private */ function toHex(number) { number = parseInt(number, 10); if (isNaN(number)) { return '00'; } number = Math.max(0, Math.min(number, 255)).toString(16); return number.length < 2 ? '0' + number : number; } /** * Normalises a CSS colour to hex #xxxxxx format * * @param {string} colorStr * @return {string} * @private */ function _normaliseColour(colorStr) { var match; colorStr = colorStr || '#000'; // rgb(n,n,n); if ((match = colorStr.match(/rgb\((\d{1,3}),\s*?(\d{1,3}),\s*?(\d{1,3})\)/i))) { return '#' + toHex(match[1]) + toHex(match[2]) + toHex(match[3]); } // expand shorthand if ((match = colorStr.match(/#([0-f])([0-f])([0-f])\s*?$/i))) { return '#' + match[1] + match[1] + match[2] + match[2] + match[3] + match[3]; } return colorStr; } /** * SCEditor BBCode format * @since 2.0.0 */ function bbcodeFormat() { var base = this; base.stripQuotes = _stripQuotes; /** * cache of all the tags pointing to their bbcodes to enable * faster lookup of which bbcode a tag should have * @private */ var tagsToBBCodes = {}; /** * Same as tagsToBBCodes but instead of HTML tags it's styles * @private */ var stylesToBBCodes = {}; /** * Allowed children of specific HTML tags. Empty array if no * children other than text nodes are allowed * @private */ var validChildren = { ul: ['li', 'ol', 'ul'], ol: ['li', 'ol', 'ul'], table: ['tr'], tr: ['td', 'th'], code: ['br', 'p', 'div'] }; /** * Populates tagsToBBCodes and stylesToBBCodes to enable faster lookups * * @private */ function buildBbcodeCache() { each(bbcodeHandlers, function (bbcode) { var isBlock, tags = bbcodeHandlers[bbcode].tags, styles = bbcodeHandlers[bbcode].styles; if (tags) { each(tags, function (tag, values) { isBlock = bbcodeHandlers[bbcode].isInline === false; tagsToBBCodes[tag] = tagsToBBCodes[tag] || {}; tagsToBBCodes[tag][isBlock] = tagsToBBCodes[tag][isBlock] || {}; tagsToBBCodes[tag][isBlock][bbcode] = values; }); } if (styles) { each(styles, function (style, values) { isBlock = bbcodeHandlers[bbcode].isInline === false; stylesToBBCodes[isBlock] = stylesToBBCodes[isBlock] || {}; stylesToBBCodes[isBlock][style] = stylesToBBCodes[isBlock][style] || {}; stylesToBBCodes[isBlock][style][bbcode] = values; }); } }); }; /** * Checks if any bbcode styles match the elements styles * * @param {!HTMLElement} element * @param {string} content * @param {boolean} [blockLevel=false] * @return {string} Content with any matching * bbcode tags wrapped around it. * @private */ function handleStyles(element, content, blockLevel) { var styleValue, format, getStyle = dom.getStyle; // convert blockLevel to boolean blockLevel = !!blockLevel; if (!stylesToBBCodes[blockLevel]) { return content; } each(stylesToBBCodes[blockLevel], function (property, bbcodes) { styleValue = getStyle(element, property); // if the parent has the same style use that instead of this one // so you don't end up with [i]parent[i]child[/i][/i] if (!styleValue || getStyle(element.parentNode, property) === styleValue) { return; } each(bbcodes, function (bbcode, values) { if (!values || values.indexOf(styleValue.toString()) > -1) { format = bbcodeHandlers[bbcode].format; if (isFunction(format)) { content = format.call(base, element, content); } else { content = _formatString(format, content); } } }); }); return content; } /** * Handles adding newlines after block level elements * * @param {HTMLElement} element The element to convert * @param {string} content The tags text content * @return {string} * @private */ function handleBlockNewlines(element, content) { var tag = element.nodeName.toLowerCase(); var isInline = dom.isInline; if (!isInline(element, true) || tag === 'br') { var isLastBlockChild, parent, parentLastChild, previousSibling = element.previousSibling; // Skips selection makers and ignored elements // Skip empty inline elements while (previousSibling && previousSibling.nodeType === 1 && !is(previousSibling, 'br') && isInline(previousSibling, true) && !previousSibling.firstChild) { previousSibling = previousSibling.previousSibling; } // If it's the last block of an inline that is the last // child of a block then it shouldn't cause a line break // except in IE < 11 //
    do { parent = element.parentNode; parentLastChild = parent && parent.lastChild; isLastBlockChild = parentLastChild === element; element = parent; } while (parent && isLastBlockChild && isInline(parent, true)); // If this block is: // * Not the last child of a block level element // * Is a
  • tag (lists are blocks) // * Is IE < 11 and the tag is BR. IE < 11 never collapses BR // tags. if (!isLastBlockChild || tag === 'li' || (tag === 'br' && IE_BR_FIX)) { content += '\n'; } // Check for: // texttext // // The second opening opening tag should cause a // line break because the previous sibing is inline. if (tag !== 'br' && previousSibling && !is(previousSibling, 'br') && isInline(previousSibling, true)) { content = '\n' + content; } } return content; } /** * Handles a HTML tag and finds any matching bbcodes * * @param {HTMLElement} element The element to convert * @param {string} content The Tags text content * @param {boolean} [blockLevel=false] If to convert block level tags * @return {string} Content with any matching bbcode tags * wrapped around it. * @private */ function handleTags(element, content, blockLevel) { var convertBBCode, format, tag = element.nodeName.toLowerCase(); // convert blockLevel to boolean blockLevel = !!blockLevel; if (tagsToBBCodes[tag] && tagsToBBCodes[tag][blockLevel]) { // loop all bbcodes for this tag each(tagsToBBCodes[tag][blockLevel], function ( bbcode, bbcodeAttribs) { // if the bbcode requires any attributes then check this has // all needed if (bbcodeAttribs) { convertBBCode = false; // loop all the bbcode attribs each(bbcodeAttribs, function (attrib, values) { // Skip if the element doesn't have the attibue or // the attribute doesn't match one of the require // values if (!attr(element, attrib) || (values && values.indexOf(attr(element, attrib)) < 0)) { return; } // break this loop as we have matched this bbcode convertBBCode = true; return false; }); if (!convertBBCode) { return; } } format = bbcodeHandlers[bbcode].format; if (isFunction(format)) { content = format.call(base, element, content); } else { content = _formatString(format, content); } }); } return content; } /** * Converts a HTML dom element to BBCode starting from * the innermost element and working backwards * * @private * @param {HTMLElement} element * @return {string} BBCode * @memberOf SCEditor.plugins.bbcode.prototype */ function elementToBbcode(element) { var toBBCode = function (node, vChildren) { var ret = ''; dom.traverse(node, function (node) { var curTag = '', nodeType = node.nodeType, tag = node.nodeName.toLowerCase(), vChild = validChildren[tag], firstChild = node.firstChild, isValidChild = true; if (typeof vChildren === 'object') { isValidChild = vChildren.indexOf(tag) > -1; // Emoticons should always be converted if (is(node, 'img') && attr(node, EMOTICON_DATA_ATTR)) { isValidChild = true; } // if this tag is one of the parents allowed children // then set this tags allowed children to whatever it // allows, otherwise set to what the parent allows if (!isValidChild) { vChild = vChildren; } } // 3 = text and 1 = element if (nodeType !== 3 && nodeType !== 1) { return; } if (nodeType === 1) { // skip empty nlf elements (new lines automatically // added after block level elements like quotes) if (is(node, '.sceditor-nlf')) { if (!firstChild || (!IE_BR_FIX && node.childNodes.length === 1 && /br/i.test(firstChild.nodeName))) { return; } } // don't convert iframe contents if (tag !== 'iframe') { curTag = toBBCode(node, vChild); } // TODO: isValidChild is no longer needed. Should use // valid children bbcodes instead by creating BBCode // tokens like the parser. if (isValidChild) { // code tags should skip most styles if (tag !== 'code') { // handle inline bbcodes curTag = handleStyles(node, curTag); curTag = handleTags(node, curTag); // handle blocklevel bbcodes curTag = handleStyles(node, curTag, true); } curTag = handleTags(node, curTag, true); ret += handleBlockNewlines(node, curTag); } else { ret += curTag; } } else { ret += node.nodeValue; } }, false, true); return ret; }; return toBBCode(element); }; /** * Initializer * @private */ base.init = function () { base.opts = this.opts; base.elementToBbcode = elementToBbcode; // build the BBCode cache buildBbcodeCache(); this.commands = extend( true, {}, defaultCommandsOverrides, this.commands ); // Add BBCode helper methods this.toBBCode = base.toSource; this.fromBBCode = base.toHtml; }; /** * Converts BBCode into HTML * * @param {boolean} asFragment * @param {string} source * @param {boolean} [legacyAsFragment] Used by fromBBCode() method */ function toHtml(asFragment, source, legacyAsFragment) { var parser = new BBCodeParser(base.opts.parserOptions); var html = parser.toHTML( base.opts.bbcodeTrim ? source.trim() : source ); return (asFragment || legacyAsFragment) ? removeFirstLastDiv(html) : html; } /** * Converts HTML into BBCode * * @param {boolean} asFragment * @param {string} html * @param {!Document} [context] * @param {!HTMLElement} [parent] * @return {string} * @private */ function toSource(asFragment, html, context, parent) { context = context || document; var bbcode, elements; var containerParent = context.createElement('div'); var container = context.createElement('div'); var parser = new BBCodeParser(base.opts.parserOptions); container.innerHTML = html; css(containerParent, 'visibility', 'hidden'); containerParent.appendChild(container); context.body.appendChild(containerParent); if (asFragment) { // Add text before and after so removeWhiteSpace doesn't remove // leading and trailing whitespace containerParent.insertBefore( context.createTextNode('#'), containerParent.firstChild ); containerParent.appendChild(context.createTextNode('#')); } // Match parents white-space handling if (parent) { css(container, 'whiteSpace', css(parent, 'whiteSpace')); } // Remove all nodes with sceditor-ignore class elements = container.getElementsByClassName('sceditor-ignore'); while (elements.length) { elements[0].parentNode.removeChild(elements[0]); } dom.removeWhiteSpace(containerParent); bbcode = elementToBbcode(container); context.body.removeChild(containerParent); bbcode = parser.toBBCode(bbcode, true); if (base.opts.bbcodeTrim) { bbcode = bbcode.trim(); } return bbcode; }; base.toHtml = toHtml.bind(null, false); base.fragmentToHtml = toHtml.bind(null, true); base.toSource = toSource.bind(null, false); base.fragmentToSource = toSource.bind(null, true); }; /** * Gets a BBCode * * @param {string} name * @return {Object|null} * @since 2.0.0 */ bbcodeFormat.get = function (name) { return bbcodeHandlers[name] || null; }; /** * Adds a BBCode to the parser or updates an existing * BBCode if a BBCode with the specified name already exists. * * @param {string} name * @param {Object} bbcode * @return {this} * @since 2.0.0 */ bbcodeFormat.set = function (name, bbcode) { if (name && bbcode) { // merge any existing command properties bbcode = extend(bbcodeHandlers[name] || {}, bbcode); bbcode.remove = function () { delete bbcodeHandlers[name]; }; bbcodeHandlers[name] = bbcode; } return this; }; /** * Renames a BBCode * * This does not change the format or HTML handling, those must be * changed manually. * * @param {string} name [description] * @param {string} newName [description] * @return {this|false} * @since 2.0.0 */ bbcodeFormat.rename = function (name, newName) { if (name in bbcodeHandlers) { bbcodeHandlers[newName] = bbcodeHandlers[name]; delete bbcodeHandlers[name]; } return this; }; /** * Removes a BBCode * * @param {string} name * @return {this} * @since 2.0.0 */ bbcodeFormat.remove = function (name) { if (name in bbcodeHandlers) { delete bbcodeHandlers[name]; } return this; }; bbcodeFormat.formatBBCodeString = formatBBCodeString; sceditor.formats.bbcode = bbcodeFormat; sceditor.BBCodeParser = BBCodeParser; }(sceditor));