From cc8449b67e249c2186ffc7c44fa9ad5c89993594 Mon Sep 17 00:00:00 2001 From: joyqi Date: Wed, 30 Nov 2016 10:50:51 +0800 Subject: [PATCH] * add Hyperdown.js * add Mysqli --- admin/editor-js.php | 24 +- admin/js/hyperdown.js | 969 ++++++++++++++++++++++++++++++++++++++++++ install.php | 9 +- var/HyperDown.php | 425 ++++++++++++------ 4 files changed, 1277 insertions(+), 150 deletions(-) create mode 100644 admin/js/hyperdown.js diff --git a/admin/editor-js.php b/admin/editor-js.php index 1dd25cf2..7e0b62b2 100644 --- a/admin/editor-js.php +++ b/admin/editor-js.php @@ -1,5 +1,6 @@ markdown): ?> + @@ -57,31 +58,18 @@ $(document).ready(function () { help: '' }; - var converter = new Markdown.Converter(), + var converter = new HyperDown(), editor = new Markdown.Editor(converter, '', options), diffMatch = new diff_match_patch(), last = '', preview = $('#wmd-preview'), mark = '@mark' + Math.ceil(Math.random() * 100000000) + '@', span = '', cache = {}; - Markdown.Extra.init(converter, { - 'extensions' : ["tables", "fenced_code_gfm", "def_list", "attr_list", "footnotes", "strikethrough", "newlines"] - }); - // 自动跟随 - converter.hooks.chain('postConversion', function (html) { - // clear special html tags - html = html.replace(/<\/?(\!doctype|html|head|body|link|title|input|select|button|textarea|style|noscript)[^>]*>/ig, function (all) { - return all.replace(/&/g, '&') - .replace(//g, '>') - .replace(/'/g, ''') - .replace(/"/g, '"'); - }); - - // clear hard breaks - html = html.replace(/\s*((?:
\n)+)\s*(<\/?(?:p|div|h[1-6]|blockquote|pre|table|dl|ol|ul|address|form|fieldset|iframe|hr|legend|article|section|nav|aside|hgroup|header|footer|figcaption|li|dd|dt)[^\w])/gm, '$2'); - + converter.hook('makeHtml', function (html) { + // convert all comment + html = html.replace(/<!--(.+?)-->/g, ''); + if (html.indexOf('') > 0) { var parts = html.split(/\s*<\!\-\-more\-\->\s*/), summary = parts.shift(), diff --git a/admin/js/hyperdown.js b/admin/js/hyperdown.js new file mode 100644 index 00000000..acd11bbe --- /dev/null +++ b/admin/js/hyperdown.js @@ -0,0 +1,969 @@ +// Generated by CoffeeScript 1.11.1 +(function() { + var Parser, + slice = [].slice; + + Parser = (function() { + var array_keys, array_values, htmlspecialchars, preg_quote, str_replace, trim, ucfirst; + + ucfirst = function(str) { + return (str.charAt(0)).toUpperCase() + str.substring(1); + }; + + preg_quote = function(str) { + return str.replace(/[-\[\]\/\{\}\(\)\*\+\?\.\\\^\$\|]/g, "\\$&"); + }; + + str_replace = function(search, replace, str) { + var i, j, l, len, len1, val; + if (search instanceof Array) { + if (replace instanceof Array) { + for (i = j = 0, len = search.length; j < len; i = ++j) { + val = search[i]; + str = str_replace(val, replace[i], str); + } + } else { + for (l = 0, len1 = search.length; l < len1; l++) { + val = search[l]; + str = str_replace(val, replace, str); + } + } + } else { + search = preg_quote(search); + str = str.replace(new RegExp(search, 'g'), replace); + } + return str; + }; + + htmlspecialchars = function(str) { + return str.replace(/&/g, '&').replace(//g, '>').replace(/"/g, '"'); + }; + + trim = function(str, ch) { + var c, i, j, ref, search; + if (ch == null) { + ch = null; + } + if (ch != null) { + search = ''; + for (i = j = 0, ref = ch.length - 1; 0 <= ref ? j <= ref : j >= ref; i = 0 <= ref ? ++j : --j) { + c = ch[i]; + c = preg_quote(c); + search += c; + } + search = '[' + search + ']*'; + return str.replace(new RegExp('^' + search), '').replace(new RegExp(search + '$'), ''); + } else { + return str.replace(/^\s*/, '').replace(/\s*$/, ''); + } + }; + + array_keys = function(arr) { + var _, j, k, len, result; + result = []; + if (arr instanceof Array) { + for (k = j = 0, len = arr.length; j < len; k = ++j) { + _ = arr[k]; + result.push(k); + } + } else { + for (k in arr) { + result.push(k); + } + } + return result; + }; + + array_values = function(arr) { + var _, j, len, result, v; + result = []; + if (arr instanceof Array) { + for (j = 0, len = arr.length; j < len; j++) { + v = arr[j]; + result.push(v); + } + } else { + for (_ in arr) { + v = arr[_]; + result.push(v); + } + } + return result; + }; + + function Parser() { + this.commonWhiteList = 'kbd|b|i|strong|em|sup|sub|br|code|del|a|hr|small'; + this.specialWhiteList = { + table: 'table|tbody|thead|tfoot|tr|td|th' + }; + this.hooks = {}; + } + + Parser.prototype.makeHtml = function(text) { + var html; + this.footnotes = []; + this.definitions = {}; + this.holders = {}; + this.uniqid = (Math.ceil(Math.random() * 10000000)) + (Math.ceil(Math.random() * 10000000)); + this.id = 0; + text = this.initText(text); + html = this.parse(text); + html = this.makeFootnotes(html); + return this.call('makeHtml', html); + }; + + Parser.prototype.hook = function(type, cb) { + if (this.hooks[type] == null) { + this.hooks[type] = []; + } + return this.hooks[type].push(cb); + }; + + Parser.prototype.makeHolder = function(str) { + var key; + key = "|\r" + this.uniqid + this.id + "\r|"; + this.id += 1; + this.holders[key] = str; + return key; + }; + + Parser.prototype.initText = function(text) { + return text.replace(/\t/g, ' ').replace(/\r/g, ''); + }; + + Parser.prototype.makeFootnotes = function(html) { + var index, val; + if (this.footnotes.length > 0) { + html += '

    '; + index = 1; + while (val = this.footnotes.shift()) { + if (typeof val === 'string') { + val += " "; + } else { + val[val.length - 1] += " "; + val = val.length > 1 ? this.parse(val.join("\n")) : this.parseInline(val[0]); + } + html += "
  1. " + val + "
  2. "; + index += 1; + } + html += '
'; + } + return html; + }; + + Parser.prototype.parse = function(text) { + var block, blocks, end, extract, html, j, len, lines, method, result, start, type, value; + lines = []; + blocks = this.parseBlock(text, lines); + html = ''; + for (j = 0, len = blocks.length; j < len; j++) { + block = blocks[j]; + type = block[0], start = block[1], end = block[2], value = block[3]; + extract = lines.slice(start, end + 1); + method = 'parse' + ucfirst(type); + extract = this.call('before' + ucfirst(method), extract, value); + result = this[method](extract, value); + result = this.call('after' + ucfirst(method), result, value); + html += result; + } + return html; + }; + + Parser.prototype.call = function() { + var args, callback, j, len, ref, type, value; + type = arguments[0], args = 2 <= arguments.length ? slice.call(arguments, 1) : []; + value = args[0]; + if (this.hooks[type] == null) { + return value; + } + ref = this.hooks[type]; + for (j = 0, len = ref.length; j < len; j++) { + callback = ref[j]; + value = callback.apply(this, args); + args[0] = value; + } + return value; + }; + + Parser.prototype.releaseHolder = function(text, clearHolders) { + var deep; + if (clearHolders == null) { + clearHolders = true; + } + deep = 0; + while ((text.indexOf("\r")) >= 0 && deep < 10) { + text = str_replace(array_keys(this.holders), array_values(this.holders), text); + deep += 1; + } + if (clearHolders) { + this.holders = {}; + } + return text; + }; + + Parser.prototype.parseInline = function(text, whiteList, clearHolders, enableAutoLink) { + if (whiteList == null) { + whiteList = ''; + } + if (clearHolders == null) { + clearHolders = true; + } + if (enableAutoLink == null) { + enableAutoLink = true; + } + text = this.call('beforeParseInline', text); + text = text.replace(/\\(.)/g, (function(_this) { + return function() { + var escaped, matches; + matches = 1 <= arguments.length ? slice.call(arguments, 0) : []; + escaped = htmlspecialchars(matches[1]); + escaped = escaped.replace(/\$/g, '$'); + return _this.makeHolder(escaped); + }; + })(this)); + text = text.replace(/(^|[^\\])(`+)(.+?)\2/mg, (function(_this) { + return function() { + var matches; + matches = 1 <= arguments.length ? slice.call(arguments, 0) : []; + return matches[1] + _this.makeHolder('' + (htmlspecialchars(matches[3])) + ''); + }; + })(this)); + text = text.replace(/<(https?:\/\/.+)>/ig, (function(_this) { + return function() { + var link, matches, url; + matches = 1 <= arguments.length ? slice.call(arguments, 0) : []; + url = _this.cleanUrl(matches[1]); + link = _this.call('parseLink', matches[1]); + return _this.makeHolder("" + link + ""); + }; + })(this)); + text = text.replace(/<(\/?)([a-z0-9-]+)(\s+[^>]*)?>/ig, (function(_this) { + return function() { + var matches; + matches = 1 <= arguments.length ? slice.call(arguments, 0) : []; + if ((('|' + _this.commonWhiteList + '|' + whiteList + '|').indexOf('|' + matches[2].toLowerCase() + '|')) >= 0) { + return _this.makeHolder(matches[0]); + } else { + return htmlspecialchars(matches[0]); + } + }; + })(this)); + text = str_replace(['<', '>'], ['<', '>'], text); + text = text.replace(/\[\^((?:[^\]]|\\\]|\\\[)+?)\]/g, (function(_this) { + return function() { + var id, matches; + matches = 1 <= arguments.length ? slice.call(arguments, 0) : []; + id = _this.footnotes.indexOf(matches[1]); + if (id < 0) { + id = _this.footnotes.length + 1; + _this.footnotes.push(_this.parseInline(matches[1], '', false)); + } + return _this.makeHolder("" + id + ""); + }; + })(this)); + text = text.replace(/!\[((?:[^\]]|\\\]|\\\[)*?)\]\(((?:[^\)]|\\\)|\\\()+?)\)/g, (function(_this) { + return function() { + var escaped, matches, url; + matches = 1 <= arguments.length ? slice.call(arguments, 0) : []; + escaped = _this.escapeBracket(matches[1]); + url = _this.escapeBracket(matches[2]); + url = _this.cleanUrl(url); + return _this.makeHolder("\"""); + }; + })(this)); + text = text.replace(/!\[((?:[^\]]|\\\]|\\\[)*?)\]\[((?:[^\]]|\\\]|\\\[)+?)\]/g, (function(_this) { + return function() { + var escaped, matches, result; + matches = 1 <= arguments.length ? slice.call(arguments, 0) : []; + escaped = _this.escapeBracket(matches[1]); + result = _this.definitions[matches[2]] != null ? "\""" : escaped; + return _this.makeHolder(result); + }; + })(this)); + text = text.replace(/\[((?:[^\]]|\\\]|\\\[)+?)\]\(((?:[^\)]|\\\)|\\\()+?)\)/g, (function(_this) { + return function() { + var escaped, matches, url; + matches = 1 <= arguments.length ? slice.call(arguments, 0) : []; + escaped = _this.parseInline(_this.escapeBracket(matches[1]), '', false, false); + url = _this.escapeBracket(matches[2]); + url = _this.cleanUrl(url); + return _this.makeHolder("" + escaped + ""); + }; + })(this)); + text = text.replace(/\[((?:[^\]]|\\\]|\\\[)+?)\]\[((?:[^\]]|\\\]|\\\[)+?)\]/g, (function(_this) { + return function() { + var escaped, matches, result; + matches = 1 <= arguments.length ? slice.call(arguments, 0) : []; + escaped = _this.parseInline(_this.escapeBracket(matches[1]), '', false, false); + result = _this.definitions[matches[2]] != null ? "" + escaped + "" : escaped; + return _this.makeHolder(result); + }; + })(this)); + text = this.parseInlineCallback(text); + text = text.replace(/<([_a-z0-9-\.\+]+@[^@]+\.[a-z]{2,})>/ig, '$1'); + if (enableAutoLink) { + text = text.replace(/(^|[^"])((https?):[x80-xff_a-z0-9-\.\/%#!@\?\+=~\|\,&\(\)]+)($|[^"])/ig, (function(_this) { + return function() { + var link, matches; + matches = 1 <= arguments.length ? slice.call(arguments, 0) : []; + link = _this.call('parseLink', matches[2]); + return matches[1] + "" + link + "" + matches[4]; + }; + })(this)); + } + text = this.call('afterParseInlineBeforeRelease', text); + text = this.releaseHolder(text, clearHolders); + text = this.call('afterParseInline', text); + return text; + }; + + Parser.prototype.parseInlineCallback = function(text) { + text = text.replace(/(\*{3})((?:.|\r)+?)\1/mg, (function(_this) { + return function() { + var matches; + matches = 1 <= arguments.length ? slice.call(arguments, 0) : []; + return '' + (_this.parseInlineCallback(matches[2])) + ''; + }; + })(this)); + text = text.replace(/(\*{2})((?:.|\r)+?)\1/mg, (function(_this) { + return function() { + var matches; + matches = 1 <= arguments.length ? slice.call(arguments, 0) : []; + return '' + (_this.parseInlineCallback(matches[2])) + ''; + }; + })(this)); + text = text.replace(/(\*)((?:.|\r)+?)\1/mg, (function(_this) { + return function() { + var matches; + matches = 1 <= arguments.length ? slice.call(arguments, 0) : []; + return '' + (_this.parseInlineCallback(matches[2])) + ''; + }; + })(this)); + text = text.replace(/(\s+|^)(_{3})((?:.|\r)+?)\2(\s+|$)/mg, (function(_this) { + return function() { + var matches; + matches = 1 <= arguments.length ? slice.call(arguments, 0) : []; + return matches[1] + '' + (_this.parseInlineCallback(matches[3])) + '' + matches[4]; + }; + })(this)); + text = text.replace(/(\s+|^)(_{2})((?:.|\r)+?)\2(\s+|$)/mg, (function(_this) { + return function() { + var matches; + matches = 1 <= arguments.length ? slice.call(arguments, 0) : []; + return matches[1] + '' + (_this.parseInlineCallback(matches[3])) + '' + matches[4]; + }; + })(this)); + text = text.replace(/(\s+|^)(_)((?:.|\r)+?)\2(\s+|$)/mg, (function(_this) { + return function() { + var matches; + matches = 1 <= arguments.length ? slice.call(arguments, 0) : []; + return matches[1] + '' + (_this.parseInlineCallback(matches[3])) + '' + matches[4]; + }; + })(this)); + text = text.replace(/(~{2})((?:.|\r)+?)\1/mg, (function(_this) { + return function() { + var matches; + matches = 1 <= arguments.length ? slice.call(arguments, 0) : []; + return '' + (_this.parseInlineCallback(matches[2])) + ''; + }; + })(this)); + return text; + }; + + Parser.prototype.parseBlock = function(text, lines) { + var align, aligns, block, emptyCount, head, isAfterList, j, key, l, len, len1, len2, line, m, matches, num, ref, row, rows, space, special, tag; + ref = text.split("\n"); + for (j = 0, len = ref.length; j < len; j++) { + line = ref[j]; + lines.push(line); + } + this.blocks = []; + this.current = 'normal'; + this.pos = -1; + special = (array_keys(this.specialWhiteList)).join('|'); + emptyCount = 0; + for (key = l = 0, len1 = lines.length; l < len1; key = ++l) { + line = lines[key]; + block = this.getBlock(); + if (block != null) { + block = block.slice(0); + } + if (!!(matches = line.match(/^(\s*)(~|`){3,}([^`~]*)$/i))) { + if (this.isBlock('code')) { + isAfterList = block[3][2]; + if (isAfterList) { + this.combineBlock().setBlock(key); + } else { + (this.setBlock(key)).endBlock(); + } + } else { + isAfterList = false; + if (this.isBlock('list')) { + space = block[3]; + isAfterList = (space > 0 && matches[1].length >= space) || matches[1].length > space; + } + this.startBlock('code', key, [matches[1], matches[3], isAfterList]); + } + continue; + } else if (this.isBlock('code')) { + this.setBlock(key); + continue; + } + if (!!(matches = line.match(new RegExp("^\\s*<(" + special + ")(\\s+[^>]*)?>", 'i')))) { + tag = matches[1].toLowerCase(); + if (!(this.isBlock('html', tag)) && !(this.isBlock('pre'))) { + this.startBlock('html', key, tag); + } + continue; + } else if (!!(matches = line.match(new RegExp("\\s*$", 'i')))) { + tag = matches[1].toLowerCase(); + if (this.isBlock('html', tag)) { + this.setBlock(key).endBlock(); + } + continue; + } else if (this.isBlock('html')) { + this.setBlock(key); + continue; + } + switch (true) { + case !!(line.match(/^ {4}/)): + emptyCount = 0; + if ((this.isBlock('pre')) || this.isBlock('list')) { + this.setBlock(key); + } else { + this.startBlock('pre', key); + } + break; + case !!(matches = line.match(/^(\s*)((?:[0-9a-z]+\.)|\-|\+|\*)\s+/)): + space = matches[1].length; + emptyCount = 0; + if (this.isBlock('list')) { + this.setBlock(key, space); + } else { + this.startBlock('list', key, space); + } + break; + case !!(matches = line.match(/^\[\^((?:[^\]]|\]|\[)+?)\]:/)): + space = matches[0].length - 1; + this.startBlock('footnote', key, [space, matches[1]]); + break; + case !!(matches = line.match(/^\s*\[((?:[^\]]|\]|\[)+?)\]:\s*(.+)$/)): + this.definitions[matches[1]] = this.cleanUrl(matches[2]); + this.startBlock('definition', key).endBlock(); + break; + case !!(line.match(/^\s*>/)): + if (this.isBlock('quote')) { + this.setBlock(key); + } else { + this.startBlock('quote', key); + } + break; + case !!(matches = line.match(/^((?:(?:(?:[ :]*\-[ :]*)+(?:\||\+))|(?:(?:\||\+)(?:[ :]*\-[ :]*)+)|(?:(?:[ :]*\-[ :]*)+(?:\||\+)(?:[ :]*\-[ :]*)+))+)$/)): + if (this.isBlock('table')) { + block[3][0].push(block[3][2]); + block[3][2] += 1; + this.setBlock(key, block[3]); + } else { + head = 0; + if ((block == null) || block[0] !== 'normal' || lines[block[2]].match(/^\s*$/)) { + this.startBlock('table', key); + } else { + head = 1; + this.backBlock(1, 'table'); + } + if (matches[1][0] === '|') { + matches[1] = matches[1].substring(1); + if (matches[1][matches[1].length - 1] === '|') { + matches[1] = matches[1].substring(0, matches[1].length - 1); + } + } + rows = matches[1].split(/\+|\|/); + aligns = []; + for (m = 0, len2 = rows.length; m < len2; m++) { + row = rows[m]; + align = 'none'; + if (!!(matches = row.match(/^\s*(:?)\-+(:?)\s*$/))) { + if (!!matches[1] && !!matches[2]) { + align = 'center'; + } else if (!!matches[1]) { + align = 'left'; + } else if (!!matches[2]) { + align = 'right'; + } + } + aligns.push(align); + } + this.setBlock(key, [[head], aligns, head + 1]); + } + break; + case !!(matches = line.match(/^(#+)(.*)$/)): + num = Math.min(matches[1].length, 6); + this.startBlock('sh', key, num).endBlock(); + break; + case !!(matches = line.match(/^\s*((=|-){2,})\s*$/)) && ((block != null) && block[0] === 'normal' && !lines[block[2]].match(/^\s*$/)): + if (this.isBlock('normal')) { + this.backBlock(1, 'mh', matches[1][0] === '=' ? 1 : 2).setBlock(key).endBlock(); + } else { + this.startBlock('normal', key); + } + break; + case !!(line.match(/^[-\*]{3,}\s*$/)): + this.startBlock('hr', key).endBlock(); + break; + default: + if (this.isBlock('list')) { + if (line.match(/^(\s*)/)) { + if (emptyCount > 0) { + this.startBlock('normal', key); + } else { + this.setBlock(key); + } + emptyCount += 1; + } else if (emptyCount === 0) { + this.setBlock(key); + } else { + this.startBlock('normal', key); + } + } else if (this.isBlock('footnote')) { + matches = line.match(/^(\s*)/); + if (matches[1].length >= block[3][0]) { + this.setBlock(key); + } else { + this.startBlock('normal', key); + } + } else if (this.isBlock('table')) { + if (0 <= line.indexOf('|')) { + block[3][2] += 1; + this.setBlock(key, block[3]); + } else { + this.startBlock('normal', key); + } + } else if (this.isBlock('pre')) { + if (line.match(/^\s*$/)) { + if (emptyCount > 0) { + this.startBlock('normal', key); + } else { + this.setBlock(key); + } + emptyCount += 1; + } else { + this.startBlock('normal', key); + } + } else if (this.isBlock('quote')) { + if (line.match(/^(\s*)/)) { + if (emptyCount > 0) { + this.startBlock('normal', key); + } else { + this.setBlock(key); + } + emptyCount += 1; + } else if (emptyCount === 0) { + this.setBlock(key); + } else { + this.startBlock('normal', key); + } + } else { + if ((block == null) || block[0] !== 'normal') { + this.startBlock('normal', key); + } else { + this.setBlock(key); + } + } + } + } + return this.optimizeBlocks(this.blocks, lines); + }; + + Parser.prototype.optimizeBlocks = function(_blocks, _lines) { + var block, blocks, from, isEmpty, key, lines, moved, nextBlock, prevBlock, to, type, types; + blocks = _blocks.slice(0); + lines = _lines.slice(0); + blocks = this.call('beforeOptimizeBlocks', blocks, lines); + key = 0; + while (blocks[key] != null) { + moved = false; + block = blocks[key]; + prevBlock = blocks[key - 1] != null ? blocks[key - 1] : null; + nextBlock = blocks[key + 1] != null ? blocks[key + 1] : null; + type = block[0], from = block[1], to = block[2]; + if ('pre' === type) { + isEmpty = lines.reduce(function(result, line) { + return (line.match(/^\s*$/)) && result; + }, true); + if (isEmpty) { + block[0] = type = 'normal'; + } + } + if ('normal' === type) { + types = ['list', 'quote']; + if (from === to && (lines[from].match(/^\s*$/)) && (prevBlock != null) && (nextBlock != null)) { + if (prevBlock[0] === nextBlock[0] && (types.indexOf(prevBlock[0])) >= 0) { + blocks[key - 1] = [prevBlock[0], prevBlock[1], nextBlock[2], null]; + blocks.splice(key, 2); + moved = true; + } + } + } + if (!moved) { + key += 1; + } + } + return this.call('afterOptimizeBlocks', blocks, lines); + }; + + Parser.prototype.parseCode = function(lines, parts) { + var blank, count, lang, rel, str; + blank = parts[0], lang = parts[1]; + lang = trim(lang); + count = blank.length; + if (!lang.match(/^[_a-z0-9-\+\#\:\.]+$/i)) { + lang = null; + } else { + parts = lang.split(':'); + if (parts.length > 1) { + lang = parts[0], rel = parts[1]; + lang = trim(lang); + rel = trim(rel); + } + } + lines = lines.slice(1, -1).map(function(line) { + return line.replace(new RegExp("/^[ ]{" + count + "}/"), ''); + }); + str = lines.join("\n"); + if (str.match(/^\s*$/)) { + return ''; + } else { + return '
' + (htmlspecialchars(str)) + '
'; + } + }; + + Parser.prototype.parsePre = function(lines) { + var str; + lines = lines.map(function(line) { + return htmlspecialchars(line.substring(4)); + }); + str = lines.join("\n"); + if (str.match(/^\s*$/)) { + return ''; + } else { + return '
' + str + '
'; + } + }; + + Parser.prototype.parseSh = function(lines, num) { + var line; + line = this.parseInline(trim(lines[0], '# ')); + if (line.match(/^\s*$/)) { + return ''; + } else { + return "" + line + ""; + } + }; + + Parser.prototype.parseMh = function(lines, num) { + return this.parseSh(lines, num); + }; + + Parser.prototype.parseQuote = function(lines) { + var str; + lines = lines.map(function(line) { + return line.replace(/^\s*> ?/, ''); + }); + str = lines.join("\n"); + if (str.match(/^\s*$/)) { + return ''; + } else { + return '
' + (this.parse(str)) + '
'; + } + }; + + Parser.prototype.parseList = function(lines) { + var found, html, j, key, l, lastType, leftLines, len, len1, len2, line, m, matches, minSpace, row, rows, secondMinSpace, space, text, type; + html = ''; + minSpace = 99999; + rows = []; + for (key = j = 0, len = lines.length; j < len; key = ++j) { + line = lines[key]; + if (matches = line.match(/^(\s*)((?:[0-9a-z]+\.?)|\-|\+|\*)(\s+)(.*)$/)) { + space = matches[1].length; + type = 0 <= '+-*'.indexOf(matches[2]) ? 'ul' : 'ol'; + minSpace = Math.min(space, minSpace); + rows.push([space, type, line, matches[4]]); + } else { + rows.push(line); + } + } + found = false; + secondMinSpace = 99999; + for (l = 0, len1 = rows.length; l < len1; l++) { + row = rows[l]; + if (row instanceof Array && row[0] !== minSpace) { + secondMinSpace = Math.min(secondMinSpace, row[0]); + found = true; + } + } + secondMinSpace = found ? secondMinSpace : minSpace; + lastType = ''; + leftLines = []; + for (m = 0, len2 = rows.length; m < len2; m++) { + row = rows[m]; + if (row instanceof Array) { + space = row[0], type = row[1], line = row[2], text = row[3]; + if (space !== minSpace) { + leftLines.push(line.replace(new RegExp("^\\s{" + secondMinSpace + "}"), '')); + } else { + if (leftLines.length > 0) { + html += '
  • ' + (this.parse(leftLines.join("\n"))) + '
  • '; + } + if (lastType !== type) { + if (!!lastType) { + html += ""; + } + html += "<" + type + ">"; + } + leftLines = [text]; + lastType = type; + } + } else { + leftLines.push(row.replace(new RegExp("^\\s{" + secondMinSpace + "}"), '')); + } + } + if (leftLines.length > 0) { + html += '
  • ' + (this.parse(leftLines.join("\n"))) + ("
  • "); + } + return html; + }; + + Parser.prototype.parseTable = function(lines, value) { + var aligns, body, column, columns, head, html, ignores, j, key, l, last, len, len1, line, num, output, row, rows, tag, text; + ignores = value[0], aligns = value[1]; + head = ignores.length > 0 && (ignores.reduce(function(prev, curr) { + return curr + prev; + })) > 0; + html = ''; + body = head ? null : true; + output = false; + for (key = j = 0, len = lines.length; j < len; key = ++j) { + line = lines[key]; + if (0 <= ignores.indexOf(key)) { + if (head && output) { + head = false; + body = true; + } + continue; + } + line = trim(line); + output = true; + if (line[0] === '|') { + line = line.substring(1); + if (line[line.length - 1] === '|') { + line = line.substring(0, line.length - 1); + } + } + rows = line.split('|').map(function(row) { + if (row.match(/^\s+$/)) { + return ''; + } else { + return trim(row); + } + }); + columns = {}; + last = -1; + for (l = 0, len1 = rows.length; l < len1; l++) { + row = rows[l]; + if (row.length > 0) { + last += 1; + columns[last] = [(columns[last] != null ? columns[last][0] + 1 : 1), row]; + } else if (columns[last] != null) { + columns[last][0] += 1; + } else { + columns[0] = [1, row]; + } + } + if (head) { + html += ''; + } else if (body) { + html += ''; + } + html += ''; + for (key in columns) { + column = columns[key]; + num = column[0], text = column[1]; + tag = head ? 'th' : 'td'; + html += "<" + tag; + if (num > 1) { + html += " colspan=\"" + num + "\""; + } + if ((aligns[key] != null) && aligns[key] !== 'none') { + html += " align=\"" + aligns[key] + "\""; + } + html += '>' + (this.parseInline(text)) + (""); + } + html += ''; + if (head) { + html += ''; + } else if (body) { + body = false; + } + } + if (body !== null) { + html += ''; + } + return html += '
    '; + }; + + Parser.prototype.parseHr = function() { + return '
    '; + }; + + Parser.prototype.parseNormal = function(lines) { + var str; + lines = lines.map((function(_this) { + return function(line) { + return _this.parseInline(line); + }; + })(this)); + str = trim(lines.join("\n")); + str = str.replace(/(\n\s*){2,}/g, '

    '); + str = str.replace(/\n/g, '
    '); + if (str.match(/^\s*$/)) { + return ''; + } else { + return "

    " + str + "

    "; + } + }; + + Parser.prototype.parseFootnote = function(lines, value) { + var index, note, space; + space = value[0], note = value[1]; + index = this.footnotes.indexOf(note); + if (index >= 0) { + lines = lines.slice(0); + lines[0] = lines[0].replace(/^\[\^((?:[^\]]|\]|\[)+?)\]:/, ''); + this.footnotes[index] = lines; + } + return ''; + }; + + Parser.prototype.parseDefinition = function() { + return ''; + }; + + Parser.prototype.parseHtml = function(lines, type) { + lines = lines.map((function(_this) { + return function(line) { + return _this.parseInline(line, _this.specialWhiteList[type] != null ? _this.specialWhiteList[type] : ''); + }; + })(this)); + return lines.join("\n"); + }; + + Parser.prototype.cleanUrl = function(url) { + var matches; + if (!!(matches = url.match(/^\s*((http|https|ftp|mailto):[x80-xff_a-z0-9-\.\/%#@\?\+=~\|\,&\(\)]+)/i))) { + matches[1]; + } + if (!!(matches = url.match(/^\s*([x80-xff_a-z0-9-\.\/%#@\?\+=~\|\,&]+)/i))) { + return matches[1]; + } else { + return '#'; + } + }; + + Parser.prototype.escapeBracket = function(str) { + return str_replace(['\\[', '\\]', '\\(', '\\)'], ['[', ']', '(', ')'], str); + }; + + Parser.prototype.startBlock = function(type, start, value) { + if (value == null) { + value = null; + } + this.pos += 1; + this.current = type; + this.blocks.push([type, start, start, value]); + return this; + }; + + Parser.prototype.endBlock = function() { + this.current = 'normal'; + return this; + }; + + Parser.prototype.isBlock = function(type, value) { + if (value == null) { + value = null; + } + return this.current === type && (null === value ? true : this.blocks[this.pos][3] === value); + }; + + Parser.prototype.getBlock = function() { + if (this.blocks[this.pos] != null) { + return this.blocks[this.pos]; + } else { + return null; + } + }; + + Parser.prototype.setBlock = function(to, value) { + if (to == null) { + to = null; + } + if (value == null) { + value = null; + } + if (to !== null) { + this.blocks[this.pos][2] = to; + } + if (value !== null) { + this.blocks[this.pos][3] = value; + } + return this; + }; + + Parser.prototype.backBlock = function(step, type, value) { + var item, last; + if (value == null) { + value = null; + } + if (this.pos < 0) { + return this.startBlock(type, 0, value); + } + last = this.blocks[this.pos][2]; + this.blocks[this.pos][2] = last - step; + item = [type, last - step + 1, last, value]; + if (this.blocks[this.pos][1] <= this.blocks[this.pos][2]) { + this.pos += 1; + this.blocks.push(item); + } else { + this.blocks[this.pos] = item; + } + this.current = type; + return this; + }; + + Parser.prototype.combineBlock = function() { + var current, prev; + if (this.pos < 1) { + return this; + } + prev = this.blocks[this.pos - 1].slice(0); + current = this.blocks[this.pos].slice(0); + prev[2] = current[2]; + this.blocks[this.pos - 1] = prev; + this.current = prev[0]; + this.blocks = this.blocks.slice(0, -1); + this.pos -= 1; + return this; + }; + + return Parser; + + })(); + + if (typeof module !== "undefined" && module !== null) { + module.exports = Parser; + } else if (typeof window !== "undefined" && window !== null) { + window.HyperDown = Parser; + } + +}).call(this); diff --git a/install.php b/install.php index cba21c97..7638c7f5 100644 --- a/install.php +++ b/install.php @@ -126,6 +126,8 @@ function _p($adapter) { switch ($adapter) { case 'Mysql': return Typecho_Db_Adapter_Mysql::isAvailable(); + case 'Mysqli': + return Typecho_Db_Adapter_Mysqli::isAvailable(); case 'Pdo_Mysql': return Typecho_Db_Adapter_Pdo_Mysql::isAvailable(); case 'SQLite': @@ -444,15 +446,16 @@ Typecho_Cookie::set('__typecho_lang', $lang);

    diff --git a/var/HyperDown.php b/var/HyperDown.php index e9e1854d..8350e051 100644 --- a/var/HyperDown.php +++ b/var/HyperDown.php @@ -1,7 +1,7 @@ @@ -14,7 +14,7 @@ class HyperDown * * @var string */ - private $_commonWhiteList = 'kbd|b|i|strong|em|sup|sub|br|code|del|a|hr|small'; + public $_commonWhiteList = 'kbd|b|i|strong|em|sup|sub|br|code|del|a|hr|small'; /** * _specialWhiteList @@ -23,7 +23,7 @@ class HyperDown * @access private */ private $_specialWhiteList = array( - 'table' => 'table|tbody|thead|tfoot|tr|td|th' + 'table' => 'table|tbody|thead|tfoot|tr|td|th' ); /** @@ -31,7 +31,7 @@ class HyperDown * * @var array */ - private $_footnotes; + public $_footnotes; /** * _blocks @@ -59,7 +59,7 @@ class HyperDown * * @var array */ - private $_definitions; + public $_definitions; /** * @var array @@ -97,7 +97,9 @@ class HyperDown $text = $this->initText($text); $html = $this->parse($text); - return $this->makeFootnotes($html); + $html = $this->makeFootnotes($html); + + return $this->call('makeHtml', $html); } /** @@ -115,7 +117,7 @@ class HyperDown */ public function makeHolder($str) { - $key = "|\r" . $this->_uniqid . $this->_id . "\r|"; + $key = "\r" . $this->_uniqid . $this->_id . "\r"; $this->_id ++; $this->_holders[$key] = $str; @@ -128,7 +130,7 @@ class HyperDown */ private function initText($text) { - $text = str_replace(array("\t", "\r"), array(' ', ''), $text); + $text = str_replace(array("\t", "\r"), array(' ', ''), $text); return $text; } @@ -216,7 +218,7 @@ class HyperDown private function releaseHolder($text, $clearHolders = true) { $deep = 0; - while (strpos($text, "|\r") !== false && $deep < 10) { + while (strpos($text, "\r") !== false && $deep < 10) { $text = str_replace(array_keys($this->_holders), array_values($this->_holders), $text); $deep ++; } @@ -234,91 +236,161 @@ class HyperDown * @param string $text * @param string $whiteList * @param bool $clearHolders + * @param bool $enableAutoLink * @return string */ - private function parseInline($text, $whiteList = '', $clearHolders = true) + public function parseInline($text, $whiteList = '', $clearHolders = true, $enableAutoLink = true) { + $self = $this; $text = $this->call('beforeParseInline', $text); + // escape + $text = preg_replace_callback( + "/\\\(.)/u", + function ($matches) use ($self) { + $escaped = htmlspecialchars($matches[1]); + $escaped = str_replace('$', '$', $escaped); + return $self->makeHolder($escaped); + }, + $text + ); + // code - $text = preg_replace_callback("/(^|[^\\\])(`+)(.+?)\\2/", function ($matches) { - return $matches[1] . $this->makeHolder('' . htmlspecialchars($matches[3]) . ''); - }, $text); + $text = preg_replace_callback( + "/(^|[^\\\])(`+)(.+?)\\2/", + function ($matches) use ($self) { + return $matches[1] . $self->makeHolder( + '' . htmlspecialchars($matches[3]) . '' + ); + }, + $text + ); // link - $text = preg_replace_callback("/<(https?:\/\/.+)>/i", function ($matches) { - return $this->makeHolder("{$matches[1]}"); - }, $text); + $text = preg_replace_callback( + "/<(https?:\/\/.+)>/i", + function ($matches) use ($self) { + $url = $self->cleanUrl($matches[1]); + $link = $self->call('parseLink', $matches[1]); + + return $self->makeHolder( + "{$link}" + ); + }, + $text + ); // encode unsafe tags - $text = preg_replace_callback("/<(\/?)([a-z0-9-]+)(\s+[^>]*)?>/i", function ($matches) use ($whiteList) { - if (stripos($this->_commonWhiteList . '|' . $whiteList, $matches[2]) !== false) { - return $this->makeHolder($matches[0]); - } else { - return htmlspecialchars($matches[0]); - } - }, $text); + $text = preg_replace_callback( + "/<(\/?)([a-z0-9-]+)(\s+[^>]*)?>/i", + function ($matches) use ($self, $whiteList) { + if (false !== stripos( + '|' . $self->_commonWhiteList . '|' . $whiteList . '|', '|' . $matches[2] . '|' + )) { + return $self->makeHolder($matches[0]); + } else { + return htmlspecialchars($matches[0]); + } + }, + $text + ); - $text = str_replace(array('<', '>'), array('<', '>'), $text); + $text = str_replace(array('<', '>'), array('<', '>'), $text); // footnote - $text = preg_replace_callback("/\[\^((?:[^\]]|\\]|\\[)+?)\]/", function ($matches) { - $id = array_search($matches[1], $this->_footnotes); + $text = preg_replace_callback( + "/\[\^((?:[^\]]|\\\\\]|\\\\\[)+?)\]/", + function ($matches) use ($self) { + $id = array_search($matches[1], $self->_footnotes); - if (false === $id) { - $id = count($this->_footnotes) + 1; - $this->_footnotes[$id] = $this->parseInline($matches[1], '', false); - } + if (false === $id) { + $id = count($self->_footnotes) + 1; + $self->_footnotes[$id] = $self->parseInline($matches[1], '', false); + } - return $this->makeHolder("{$id}"); - }, $text); + return $self->makeHolder( + "{$id}" + ); + }, + $text + ); // image - $text = preg_replace_callback("/!\[((?:[^\]]|\\]|\\[)*?)\]\(((?:[^\)]|\\)|\\()+?)\)/", function ($matches) { - $escaped = $this->escapeBracket($matches[1]); - $url = $this->escapeBracket($matches[2]); - return $this->makeHolder("\"{$escaped}\""); - }, $text); + $text = preg_replace_callback( + "/!\[((?:[^\]]|\\\\\]|\\\\\[)*?)\]\(((?:[^\)]|\\\\\)|\\\\\()+?)\)/", + function ($matches) use ($self) { + $escaped = $self->escapeBracket($matches[1]); + $url = $self->escapeBracket($matches[2]); + $url = $self->cleanUrl($url); + return $self->makeHolder( + "\"{$escaped}\"" + ); + }, + $text + ); - $text = preg_replace_callback("/!\[((?:[^\]]|\\]|\\[)*?)\]\[((?:[^\]]|\\]|\\[)+?)\]/", function ($matches) { - $escaped = $this->escapeBracket($matches[1]); + $text = preg_replace_callback( + "/!\[((?:[^\]]|\\\\\]|\\\\\[)*?)\]\[((?:[^\]]|\\\\\]|\\\\\[)+?)\]/", + function ($matches) use ($self) { + $escaped = $self->escapeBracket($matches[1]); - $result = isset($this->_definitions[$matches[2]]) ? - "_definitions[$matches[2]]}\" alt=\"{$escaped}\" title=\"{$escaped}\">" - : $escaped; + $result = isset( $self->_definitions[$matches[2]] ) ? + "_definitions[$matches[2]]}\" alt=\"{$escaped}\" title=\"{$escaped}\">" + : $escaped; - return $this->makeHolder($result); - }, $text); + return $self->makeHolder($result); + }, + $text + ); // link - $text = preg_replace_callback("/\[((?:[^\]]|\\]|\\[)+?)\]\(((?:[^\)]|\\)|\\()+?)\)/", function ($matches) { - $escaped = $this->parseInline($this->escapeBracket($matches[1]), '', false); - $url = $this->escapeBracket($matches[2]); - return $this->makeHolder("{$escaped}"); - }, $text); + $text = preg_replace_callback( + "/\[((?:[^\]]|\\\\\]|\\\\\[)+?)\]\(((?:[^\)]|\\\\\)|\\\\\()+?)\)/", + function ($matches) use ($self) { + $escaped = $self->parseInline( + $self->escapeBracket($matches[1]), '', false, false + ); + $url = $self->escapeBracket($matches[2]); + $url = $self->cleanUrl($url); + return $self->makeHolder("{$escaped}"); + }, + $text + ); - $text = preg_replace_callback("/\[((?:[^\]]|\\]|\\[)+?)\]\[((?:[^\]]|\\]|\\[)+?)\]/", function ($matches) { - $escaped = $this->parseInline($this->escapeBracket($matches[1]), '', false); + $text = preg_replace_callback( + "/\[((?:[^\]]|\\\\\]|\\\\\[)+?)\]\[((?:[^\]]|\\\\\]|\\\\\[)+?)\]/", + function ($matches) use ($self) { + $escaped = $self->parseInline( + $self->escapeBracket($matches[1]), '', false + ); + $result = isset( $self->_definitions[$matches[2]] ) ? + "_definitions[$matches[2]]}\">{$escaped}" + : $escaped; - $result = isset($this->_definitions[$matches[2]]) ? - "_definitions[$matches[2]]}\">{$escaped}" - : $escaped; - - return $this->makeHolder($result); - }, $text); - - // escape - $text = preg_replace_callback("/\\\(`|\*|_|~)/", function ($matches) { - return $this->makeHolder(htmlspecialchars($matches[1])); - }, $text); + return $self->makeHolder($result); + }, + $text + ); // strong and em and some fuck $text = $this->parseInlineCallback($text); - $text = preg_replace("/<([_a-z0-9-\.\+]+@[^@]+\.[a-z]{2,})>/i", "\\1", $text); + $text = preg_replace( + "/<([_a-z0-9-\.\+]+@[^@]+\.[a-z]{2,})>/i", + "\\1", + $text + ); // autolink url - $text = preg_replace("/(^|[^\"])((http|https|ftp|mailto):[_a-z0-9-\.\/%#@\?\+=~\|\,&\(\)]+)($|[^\"])/i", - "\\1\\2\\4", $text); + if ($enableAutoLink) { + $text = preg_replace_callback( + "/(^|[^\"])((https?):[x80-xff_a-z0-9-\.\/%#@\?\+=~\|\,&\(\)]+)($|[^\"])/i", + function ($matches) use ($self) { + $link = $self->call('parseLink', $matches[2]); + return "{$matches[1]}{$link}{$matches[4]}"; + }, + $text + ); + } $text = $this->call('afterParseInlineBeforeRelease', $text); $text = $this->releaseHolder($text, $clearHolders); @@ -332,35 +404,79 @@ class HyperDown * @param $text * @return mixed */ - private function parseInlineCallback($text) + public function parseInlineCallback($text) { - $text = preg_replace_callback("/(\*{3})(.+?)\\1/", function ($matches) { - return '' . $this->parseInlineCallback($matches[2]) . ''; - }, $text); + $self = $this; - $text = preg_replace_callback("/(\*{2})(.+?)\\1/", function ($matches) { - return '' . $this->parseInlineCallback($matches[2]) . ''; - }, $text); + $text = preg_replace_callback( + "/(\*{3})(.+?)\\1/", + function ($matches) use ($self) { + return '' . + $self->parseInlineCallback($matches[2]) . + ''; + }, + $text + ); - $text = preg_replace_callback("/(\*)(.+?)\\1/", function ($matches) { - return '' . $this->parseInlineCallback($matches[2]) . ''; - }, $text); + $text = preg_replace_callback( + "/(\*{2})(.+?)\\1/", + function ($matches) use ($self) { + return '' . + $self->parseInlineCallback($matches[2]) . + ''; + }, + $text + ); - $text = preg_replace_callback("/(\s+|^)(_{3})(.+?)\\2(\s+|$)/", function ($matches) { - return $matches[1] . '' . $this->parseInlineCallback($matches[3]) . '' . $matches[4]; - }, $text); + $text = preg_replace_callback( + "/(\*)(.+?)\\1/", + function ($matches) use ($self) { + return '' . + $self->parseInlineCallback($matches[2]) . + ''; + }, + $text + ); - $text = preg_replace_callback("/(\s+|^)(_{2})(.+?)\\2(\s+|$)/", function ($matches) { - return $matches[1] . '' . $this->parseInlineCallback($matches[3]) . '' . $matches[4]; - }, $text); + $text = preg_replace_callback( + "/(\s+|^)(_{3})(.+?)\\2(\s+|$)/", + function ($matches) use ($self) { + return $matches[1] . '' . + $self->parseInlineCallback($matches[3]) . + '' . $matches[4]; + }, + $text + ); - $text = preg_replace_callback("/(\s+|^)(_)(.+?)\\2(\s+|$)/", function ($matches) { - return $matches[1] . '' . $this->parseInlineCallback($matches[3]) . '' . $matches[4]; - }, $text); + $text = preg_replace_callback( + "/(\s+|^)(_{2})(.+?)\\2(\s+|$)/", + function ($matches) use ($self) { + return $matches[1] . '' . + $self->parseInlineCallback($matches[3]) . + '' . $matches[4]; + }, + $text + ); - $text = preg_replace_callback("/(~{2})(.+?)\\1/", function ($matches) { - return '' . $this->parseInlineCallback($matches[2]) . ''; - }, $text); + $text = preg_replace_callback( + "/(\s+|^)(_)(.+?)\\2(\s+|$)/", + function ($matches) use ($self) { + return $matches[1] . '' . + $self->parseInlineCallback($matches[3]) . + '' . $matches[4]; + }, + $text + ); + + $text = preg_replace_callback( + "/(~{2})(.+?)\\1/", + function ($matches) use ($self) { + return '' . + $self->parseInlineCallback($matches[2]) . + ''; + }, + $text + ); return $text; } @@ -384,7 +500,7 @@ class HyperDown // analyze by line foreach ($lines as $key => $line) { $block = $this->getBlock(); - + // code block is special if (preg_match("/^(\s*)(~|`){3,}([^`~]*)$/i", $line, $matches)) { if ($this->isBlock('code')) { @@ -407,7 +523,9 @@ class HyperDown || strlen($matches[1]) > $space; } - $this->startBlock('code', $key, array($matches[1], $matches[3], $isAfterList)); + $this->startBlock('code', $key, array( + $matches[1], $matches[3], $isAfterList + )); } continue; @@ -416,19 +534,6 @@ class HyperDown continue; } - // pre block - if (preg_match("/^ {4}/", $line)) { - $emptyCount = 0; - - if ($this->isBlock('pre') || $this->isBlock('list')) { - $this->setBlock($key); - continue; - } else if ($this->isBlock('normal')) { - $this->startBlock('pre', $key); - continue; - } - } - // html block is special too if (preg_match("/^\s*<({$special})(\s+[^>]*)?>/i", $line, $matches)) { $tag = strtolower($matches[1]); @@ -452,6 +557,17 @@ class HyperDown } switch (true) { + // pre block + case preg_match("/^ {4}/", $line): + $emptyCount = 0; + + if ($this->isBlock('pre') || $this->isBlock('list')) { + $this->setBlock($key); + } else if ($this->isBlock('normal')) { + $this->startBlock('pre', $key); + } + break; + // list case preg_match("/^(\s*)((?:[0-9a-z]+\.)|\-|\+|\*)\s+/", $line, $matches): $space = strlen($matches[1]); @@ -463,17 +579,19 @@ class HyperDown } else { $this->startBlock('list', $key, $space); } - break; + break; // footnote case preg_match("/^\[\^((?:[^\]]|\\]|\\[)+?)\]:/", $line, $matches): $space = strlen($matches[0]) - 1; - $this->startBlock('footnote', $key, array($space, $matches[1])); + $this->startBlock('footnote', $key, array( + $space, $matches[1] + )); break; // definition case preg_match("/^\s*\[((?:[^\]]|\\]|\\[)+?)\]:\s*(.+)$/", $line, $matches): - $this->_definitions[$matches[1]] = $matches[2]; + $this->_definitions[$matches[1]] = $this->cleanUrl($matches[2]); $this->startBlock('definition', $key) ->endBlock(); break; @@ -489,15 +607,19 @@ class HyperDown // table case preg_match("/^((?:(?:(?:[ :]*\-[ :]*)+(?:\||\+))|(?:(?:\||\+)(?:[ :]*\-[ :]*)+)|(?:(?:[ :]*\-[ :]*)+(?:\||\+)(?:[ :]*\-[ :]*)+))+)$/", $line, $matches): - if ($this->isBlock('normal')) { - $head = false; + if ($this->isBlock('table')) { + $block[3][0][] = $block[3][2]; + $block[3][2] ++; + $this->setBlock($key, $block[3]); + } else { + $head = 0; if (empty($block) || $block[0] != 'normal' || preg_match("/^\s*$/", $lines[$block[2]])) { $this->startBlock('table', $key); } else { - $head = true; + $head = 1; $this->backBlock(1, 'table'); } @@ -527,7 +649,7 @@ class HyperDown $aligns[] = $align; } - $this->setBlock($key, array($head, $aligns)); + $this->setBlock($key, array(array($head), $aligns, $head + 1)); } break; @@ -581,7 +703,8 @@ class HyperDown } } else if ($this->isBlock('table')) { if (false !== strpos($line, '|')) { - $this->setBlock($key); + $block[3][2] ++; + $this->setBlock($key, $block[3]); } else { $this->startBlock('normal', $key); } @@ -634,7 +757,11 @@ class HyperDown { $blocks = $this->call('beforeOptimizeBlocks', $blocks, $lines); - foreach ($blocks as $key => &$block) { + $key = 0; + while (isset($blocks[$key])) { + $moved = false; + + $block = &$blocks[$key]; $prevBlock = isset($blocks[$key - 1]) ? $blocks[$key - 1] : NULL; $nextBlock = isset($blocks[$key + 1]) ? $blocks[$key + 1] : NULL; @@ -658,11 +785,20 @@ class HyperDown && !empty($prevBlock) && !empty($nextBlock)) { if ($prevBlock[0] == $nextBlock[0] && in_array($prevBlock[0], $types)) { // combine 3 blocks - $blocks[$key - 1] = array($prevBlock[0], $prevBlock[1], $nextBlock[2], NULL); + $blocks[$key - 1] = array( + $prevBlock[0], $prevBlock[1], $nextBlock[2], NULL + ); array_splice($blocks, $key, 2); + + // do not move + $moved = true; } } } + + if (!$moved) { + $key ++; + } } return $this->call('afterOptimizeBlocks', $blocks, $lines); @@ -681,8 +817,15 @@ class HyperDown $lang = trim($lang); $count = strlen($blank); - if (!preg_match("/^[_a-z0-9-\+\#]+$/i", $lang)) { + if (! preg_match("/^[_a-z0-9-\+\#\:\.]+$/i", $lang)) { $lang = NULL; + } else { + $parts = explode(':', $lang); + if (count($parts) > 1) { + list ($lang, $rel) = $parts; + $lang = trim($lang); + $rel = trim($rel); + } } $lines = array_map(function ($line) use ($count) { @@ -691,8 +834,9 @@ class HyperDown $str = implode("\n", $lines); return preg_match("/^\s*$/", $str) ? '' : - '
    '
    -            . htmlspecialchars(implode("\n", $lines)) . '
    '; + '
    '
    +            . htmlspecialchars($str) . '
    '; } /** @@ -733,8 +877,7 @@ class HyperDown */ private function parseMh(array $lines, $num) { - $line = $this->parseInline(trim($lines[0], '# ')); - return preg_match("/^\s*$/", $line) ? '' : "{$line}"; + return $this->parseSh($lines, $num); } /** @@ -786,7 +929,7 @@ class HyperDown $found = true; } } - $secondMinSpace = $found ?: $minSpace; + $secondMinSpace = $found ? $secondMinSpace : $minSpace; $lastType = ''; $leftLines = array(); @@ -832,21 +975,24 @@ class HyperDown */ private function parseTable(array $lines, array $value) { - list ($head, $aligns) = $value; - $ignore = $head ? 1 : 0; + list ($ignores, $aligns) = $value; + $head = count($ignores) > 0 && array_sum($ignores) > 0; $html = ''; - $body = NULL; + $body = $head ? NULL : true; + $output = false; foreach ($lines as $key => $line) { - if ($key == $ignore) { - $head = false; - $body = true; + if (in_array($key, $ignores)) { + if ($head && $output) { + $head = false; + $body = true; + } continue; } - $line = trim($line); + $output = true; if ($line[0] == '|') { $line = substr($line, 1); @@ -870,7 +1016,9 @@ class HyperDown foreach ($rows as $row) { if (strlen($row) > 0) { $last ++; - $columns[$last] = array(isset($columns[$last]) ? $columns[$last][0] + 1 : 1, $row); + $columns[$last] = array( + isset($columns[$last]) ? $columns[$last][0] + 1 : 1, $row + ); } else if (isset($columns[$last])) { $columns[$last][0] ++; } else { @@ -995,13 +1143,30 @@ class HyperDown return implode("\n", $lines); } + /** + * @param $url + * @return string + */ + public function cleanUrl($url) + { + if (preg_match("/^\s*((http|https|ftp|mailto):[x80-xff_a-z0-9-\.\/%#@\?\+=~\|\,&\(\)]+)/i", $url, $matches)) { + return $matches[1]; + } else if (preg_match("/^\s*([x80-xff_a-z0-9-\.\/%#@\?\+=~\|\,&]+)/i", $url, $matches)) { + return $matches[1]; + } else { + return '#'; + } + } + /** * @param $str * @return mixed */ - private function escapeBracket($str) + public function escapeBracket($str) { - return str_replace(array('\[', '\]', '\(', '\)'), array('[', ']', '(', ')'), $str); + return str_replace( + array('\[', '\]', '\(', '\)'), array('[', ']', '(', ')'), $str + ); } /** @@ -1098,7 +1263,9 @@ class HyperDown } $this->_current = $type; - $this->_blocks[$this->_pos] = array($type, $last - $step + 1, $last, $value); + $this->_blocks[$this->_pos] = array( + $type, $last - $step + 1, $last, $value + ); return $this; }