diff --git a/system/admin/editor.md/.editorconfig b/system/admin/editor.md/.editorconfig new file mode 100644 index 00000000..e291365a --- /dev/null +++ b/system/admin/editor.md/.editorconfig @@ -0,0 +1,9 @@ +root = true + +[*] +charset = utf-8 +indent_style = space +indent_size = 4 +end_of_line = lf +insert_final_newline = true +trim_trailing_whitespace = true diff --git a/system/admin/editor.md/.gitignore b/system/admin/editor.md/.gitignore new file mode 100644 index 00000000..9a85439c --- /dev/null +++ b/system/admin/editor.md/.gitignore @@ -0,0 +1,15 @@ +logs +*.log +*.pid +*.seed +node_modules/ +.sass-cache/ +research/ +test/ +backup/ +examples/uploads/**/* +*.bat +*.sh +.project +.url +css/*.map \ No newline at end of file diff --git a/system/admin/editor.md/.jshintrc b/system/admin/editor.md/.jshintrc new file mode 100644 index 00000000..b58a50e0 --- /dev/null +++ b/system/admin/editor.md/.jshintrc @@ -0,0 +1,22 @@ +{ + "asi": true, + "esnext": true, + "bitwise": true, + "camelcase": true, + "curly": true, + "eqeqeq": true, + "immed": true, + "indent": 2, + "latedef": true, + "newcap": true, + "noarg": true, + "quotmark": "singlequote", + "regexp": true, + "undef": false, + "unused": false, + "strict": true, + "trailing": true, + "smarttabs": true, + "white": true, + "laxbreak": true +} diff --git a/system/admin/editor.md/CHANGE.md b/system/admin/editor.md/CHANGE.md new file mode 100644 index 00000000..a450bd96 --- /dev/null +++ b/system/admin/editor.md/CHANGE.md @@ -0,0 +1,553 @@ +## 更新日志 + +### v1.0.x + +##### v1.0.0 beta + +预览版:基本功能完成; + +##### v1.0.0 releases + +发布 v1.0.0 正式版。 + +主要更新: + +- 新建分支 `mathjax-version`,但不打算继续对此分支进行开发; + +- 移除 MathJax,改用 KaTeX [#2](https://github.com/pandao/editor.md/issues/2),解析和预览响应速度大幅度提高 [#3](https://github.com/pandao/editor.md/issues/3); + - 移除 `mathjax` 配置项; + - 移除 `mathjaxURL` 属性; + - 移除 `setMathJaxConfig()` 方法; + - 移除 `loadMathJax()` 方法; + - 移除MathJax的所有示例; + - 新增 `tex` 配置项,表示是否开启支持科学公式 TeX ,基于 KaTeX; + - 新增 `katexURL` 属性; + - 新增 `loadKaTeX` 方法; + - 新增 KaTeX 的示例; + +- `setCodeEditor()` 方法更名为 `setCodeMirror()`; + +- 合并 CodeMirror 使用到的多个 JS 模块文件,大幅减少 HTTP 请求,加快下载速度; + - 新增合并后的两个模块文件:`./lib/codemirror/modes.min.js`、`./lib/codemirror/addons.min.js` ; + - `Gulpfile.js` 新增合并 CodeMirror 模块文件的任务方法 `codemirror-mode` 和 `codemirror-addon` ; + - 另外在使用 Require.js 时,因为 CodeMirror 的严格模块依赖的限制,不能使用上述合并的模块文件,仍然采用动态加载多个模块文件; + +- 更新 `README.md` 等相关文档和示例; + +- 解决 Sea.js 环境下 Raphael.js 无法运行导致不支持流程图和时序图的问题,即必须先加载 Raphael.js ,后加载 Sea.js ; + +### v1.1.x + +##### v1.1.0 + +主要更新: + +- 设计并更换了 Logo; +- 新增添加图片、链接、锚点链接、代码块、预格式文本等操作弹出对话框层及示例; +- 新增支持图片(跨域)上传; +- 改用 ` + +``` + +> Tip: Editor.md can auto append `") + markdownTextarea = this.markdownTextarea = editor.children("textarea") + } + markdownTextarea + .addClass(classNames.textarea.markdown) + .attr("placeholder", settings.placeholder) + if ( + typeof markdownTextarea.attr("name") === "undefined" || + markdownTextarea.attr("name") === "" + ) { + markdownTextarea.attr( + "name", + settings.name !== "" ? settings.name : id + "-markdown-doc" + ) + } + var appendElements = [ + !settings.readOnly + ? '' + : "", + settings.saveHTMLToTextarea + ? '' + : "", + '
', + '
', + '
' + ].join("\n") + editor.append(appendElements).addClass(classPrefix + "vertical") + if (settings.theme !== "") { + editor.addClass(classPrefix + "theme-" + settings.theme) + } + this.mask = editor.children("." + classPrefix + "mask") + this.containerMask = editor.children("." + classPrefix + "container-mask") + if (settings.markdown !== "") { + markdownTextarea.val(settings.markdown) + } + if (settings.appendMarkdown !== "") { + markdownTextarea.val(markdownTextarea.val() + settings.appendMarkdown) + } + this.htmlTextarea = editor.children("." + classNames.textarea.html) + this.preview = editor.children("." + classPrefix + "preview") + this.previewContainer = this.preview.children( + "." + classPrefix + "preview-container" + ) + if (settings.previewTheme !== "") { + this.preview.addClass( + classPrefix + "preview-theme-" + settings.previewTheme + ) + } + if (typeof define === "function" && define.amd) { + if (typeof katex !== "undefined") { + editormd.$katex = katex + } + if (settings.searchReplace && !settings.readOnly) { + editormd.loadCSS(settings.path + "codemirror/addon/dialog/dialog") + editormd.loadCSS( + settings.path + "codemirror/addon/search/matchesonscrollbar" + ) + } + } + if ( + (typeof define === "function" && define.amd) || + !settings.autoLoadModules + ) { + if (typeof CodeMirror !== "undefined") { + editormd.$CodeMirror = CodeMirror + } + + if (typeof marked !== "undefined") { + editormd.$marked = marked + } + this.setCodeMirror() + .setToolbar() + .loadedDisplay() + } else { + this.loadQueues() + } + return this + }, + + /** + * 所需组件加载队列 + * Required components loading queue + * + * @returns {editormd} 返回editormd的实例对象 + */ + + loadQueues: function() { + var _this = this + var settings = this.settings + var loadPath = settings.path + var loadFlowChartOrSequenceDiagram = function() { + if (editormd.isIE8) { + _this.loadedDisplay() + return + } + if (settings.flowChart || settings.sequenceDiagram) { + editormd.loadScript(loadPath + "raphael.min", function() { + editormd.loadScript(loadPath + "underscore.min", function() { + if (!settings.flowChart && settings.sequenceDiagram) { + editormd.loadScript( + loadPath + "sequence-diagram.min", + function() { + _this.loadedDisplay() + } + ) + } else if (settings.flowChart && !settings.sequenceDiagram) { + editormd.loadScript(loadPath + "flowchart.min", function() { + editormd.loadScript( + loadPath + "jquery.flowchart.min", + function() { + _this.loadedDisplay() + } + ) + }) + } else if (settings.flowChart && settings.sequenceDiagram) { + editormd.loadScript(loadPath + "flowchart.min", function() { + editormd.loadScript( + loadPath + "jquery.flowchart.min", + function() { + editormd.loadScript( + loadPath + "sequence-diagram.min", + function() { + _this.loadedDisplay() + } + ) + } + ) + }) + } + }) + }) + } else { + _this.loadedDisplay() + } + } + + editormd.loadCSS(loadPath + "codemirror/codemirror.min") + if (settings.searchReplace && !settings.readOnly) { + editormd.loadCSS(loadPath + "codemirror/addon/dialog/dialog") + editormd.loadCSS( + loadPath + "codemirror/addon/search/matchesonscrollbar" + ) + } + if (settings.codeFold) { + editormd.loadCSS(loadPath + "codemirror/addon/fold/foldgutter") + } + editormd.loadScript(loadPath + "codemirror/codemirror.min", function() { + editormd.$CodeMirror = CodeMirror + editormd.loadScript(loadPath + "codemirror/modes.min", function() { + editormd.loadScript(loadPath + "codemirror/addons.min", function() { + _this.setCodeMirror() + if (settings.mode !== "gfm" && settings.mode !== "markdown") { + _this.loadedDisplay() + return false + } + _this.setToolbar() + editormd.loadScript(loadPath + "marked.min", function() { + editormd.$marked = marked + + if (settings.previewCodeHighlight) { + editormd.loadScript(loadPath + "prettify.min", function() { + loadFlowChartOrSequenceDiagram() + }) + } else { + loadFlowChartOrSequenceDiagram() + } + }) + }) + }) + }) + return this + }, + + /** + * 设置 Editor.md 的整体主题,主要是工具栏 + * Setting Editor.md theme + * + * @returns {editormd} 返回editormd的实例对象 + */ + + setTheme: function(theme) { + var editor = this.editor + var oldTheme = this.settings.theme + var themePrefix = this.classPrefix + "theme-" + editor.removeClass(themePrefix + oldTheme).addClass(themePrefix + theme) + this.settings.theme = theme + return this + }, + + /** + * 设置 CodeMirror(编辑区)的主题 + * Setting CodeMirror (Editor area) theme + * + * @returns {editormd} 返回editormd的实例对象 + */ + + setEditorTheme: function(theme) { + var settings = this.settings + settings.editorTheme = theme + if (theme !== "default") { + editormd.loadCSS( + settings.path + "codemirror/theme/" + settings.editorTheme + ) + } + this.cm.setOption("theme", theme) + return this + }, + + /** + * setEditorTheme() 的别名 + * setEditorTheme() alias + * + * @returns {editormd} 返回editormd的实例对象 + */ + + setCodeMirrorTheme: function(theme) { + this.setEditorTheme(theme) + return this + }, + + /** + * 设置 Editor.md 的主题 + * Setting Editor.md theme + * + * @returns {editormd} 返回editormd的实例对象 + */ + + setPreviewTheme: function(theme) { + var preview = this.preview + var oldTheme = this.settings.previewTheme + var themePrefix = this.classPrefix + "preview-theme-" + preview.removeClass(themePrefix + oldTheme).addClass(themePrefix + theme) + this.settings.previewTheme = theme + return this + }, + + /** + * 配置和初始化CodeMirror组件 + * CodeMirror initialization + * + * @returns {editormd} 返回editormd的实例对象 + */ + + setCodeMirror: function() { + var settings = this.settings + var editor = this.editor + if (settings.editorTheme !== "default") { + editormd.loadCSS( + settings.path + "codemirror/theme/" + settings.editorTheme + ) + } + var codeMirrorConfig = { + mode: settings.mode, + theme: settings.editorTheme, + tabSize: settings.tabSize, + dragDrop: false, + autofocus: settings.autoFocus, + autoCloseTags: settings.autoCloseTags, + readOnly: settings.readOnly ? "nocursor" : false, + indentUnit: settings.indentUnit, + lineNumbers: settings.lineNumbers, + lineWrapping: settings.lineWrapping, + extraKeys: { + "Ctrl-Q": function(cm) { + cm.foldCode(cm.getCursor()) + } + }, + foldGutter: settings.codeFold, + gutters: ["CodeMirror-linenumbers", "CodeMirror-foldgutter"], + matchBrackets: settings.matchBrackets, + indentWithTabs: settings.indentWithTabs, + styleActiveLine: settings.styleActiveLine, + styleSelectedText: settings.styleSelectedText, + autoCloseBrackets: settings.autoCloseBrackets, + showTrailingSpace: settings.showTrailingSpace, + highlightSelectionMatches: !settings.matchWordHighlight + ? false + : { + showToken: + settings.matchWordHighlight === "onselected" ? false : /\w/ + } + } + + this.codeEditor = this.cm = editormd.$CodeMirror.fromTextArea( + this.markdownTextarea[0], + codeMirrorConfig + ) + this.codeMirror = this.cmElement = editor.children(".CodeMirror") + + if (settings.value !== "") { + this.cm.setValue(settings.value) + } + + this.codeMirror.css({ + fontSize: settings.fontSize, + width: !settings.watch ? "100%" : "50%" + }) + + if (settings.autoHeight) { + this.codeMirror.css({ + height: "auto", + minHeight: settings.minHeight, + maxHeight: settings.maxHeight + }) + this.codeMirror.find(".CodeMirror-gutters").css({ + minHeight: settings.minHeight + }) + this.codeMirror.find(".CodeMirror-scroll").css({ + minHeight: settings.minHeight, + maxHeight: settings.maxHeight + }) + this.cm.setOption("viewportMargin", Infinity) + } + + if (!settings.lineNumbers) { + this.codeMirror.find(".CodeMirror-gutters").css("border-right", "none") + } + + return this + }, + + /** + * 获取CodeMirror的配置选项 + * Get CodeMirror setting options + * + * @returns {Mixed} return CodeMirror setting option value + */ + + getCodeMirrorOption: function(key) { + return this.cm.getOption(key) + }, + + /** + * 配置和重配置CodeMirror的选项 + * CodeMirror setting options / resettings + * + * @returns {editormd} 返回editormd的实例对象 + */ + + setCodeMirrorOption: function(key, value) { + this.cm.setOption(key, value) + + return this + }, + + /** + * 添加 CodeMirror 键盘快捷键 + * Add CodeMirror keyboard shortcuts key map + * + * @returns {editormd} 返回editormd的实例对象 + */ + + addKeyMap: function(map, bottom) { + this.cm.addKeyMap(map, bottom) + + return this + }, + + /** + * 移除 CodeMirror 键盘快捷键 + * Remove CodeMirror keyboard shortcuts key map + * + * @returns {editormd} 返回editormd的实例对象 + */ + + removeKeyMap: function(map) { + this.cm.removeKeyMap(map) + + return this + }, + + /** + * 跳转到指定的行 + * Goto CodeMirror line + * + * @param {String|Intiger} line line number or "first"|"last" + * @returns {editormd} 返回editormd的实例对象 + */ + + gotoLine: function(line) { + var settings = this.settings + if (!settings.gotoLine) { + return this + } + var cm = this.cm + var editor = this.editor + var count = cm.lineCount() + var preview = this.preview + if (typeof line === "string") { + if (line === "last") { + line = count + } + if (line === "first") { + line = 1 + } + } + if (typeof line !== "number") { + alert("Error: The line number must be an integer.") + return this + } + line = parseInt(line) - 1 + if (line > count) { + alert("Error: The line number range 1-" + count) + return this + } + cm.setCursor({ line: line, ch: 0 }) + var scrollInfo = cm.getScrollInfo() + var clientHeight = scrollInfo.clientHeight + var coords = cm.charCoords({ line: line, ch: 0 }, "local") + cm.scrollTo(null, (coords.top + coords.bottom - clientHeight) / 2) + if (settings.watch) { + var cmScroll = this.codeMirror.find(".CodeMirror-scroll")[0] + var height = $(cmScroll).height() + var scrollTop = cmScroll.scrollTop + var percent = scrollTop / cmScroll.scrollHeight + if (scrollTop === 0) { + preview.scrollTop(0) + } else if (scrollTop + height >= cmScroll.scrollHeight - 16) { + preview.scrollTop(preview[0].scrollHeight) + } else { + preview.scrollTop(preview[0].scrollHeight * percent) + } + } + cm.focus() + return this + }, + + /** + * 扩展当前实例对象,可同时设置多个或者只设置一个 + * Extend editormd instance object, can mutil setting. + * + * @returns {editormd} this(editormd instance object.) + */ + + extend: function() { + if (typeof arguments[1] !== "undefined") { + if (typeof arguments[1] === "function") { + arguments[1] = $.proxy(arguments[1], this) + } + this[arguments[0]] = arguments[1] + } + if ( + typeof arguments[0] === "object" && + typeof arguments[0].length === "undefined" + ) { + $.extend(true, this, arguments[0]) + } + return this + }, + + /** + * 设置或扩展当前实例对象,单个设置 + * Extend editormd instance object, one by one + * + * @param {String|Object} key option key + * @param {String|Object} value option value + * @returns {editormd} this(editormd instance object.) + */ + + set: function(key, value) { + if (typeof value !== "undefined" && typeof value === "function") { + value = $.proxy(value, this) + } + this[key] = value + return this + }, + + /** + * 重新配置 + * Resetting editor options + * + * @param {String|Object} key option key + * @param {String|Object} value option value + * @returns {editormd} this(editormd instance object.) + */ + + config: function(key, value) { + var settings = this.settings + if (typeof key === "object") { + settings = $.extend(true, settings, key) + } + if (typeof key === "string") { + settings[key] = value + } + this.settings = settings + this.recreate() + return this + }, + + /** + * 注册事件处理方法 + * Bind editor event handle + * + * @param {String} eventType event type + * @param {Function} callback 回调函数 + * @returns {editormd} this(editormd instance object.) + */ + + on: function(eventType, callback) { + var settings = this.settings + if (typeof settings["on" + eventType] !== "undefined") { + settings["on" + eventType] = $.proxy(callback, this) + } + return this + }, + + /** + * 解除事件处理方法 + * Unbind editor event handle + * + * @param {String} eventType event type + * @returns {editormd} this(editormd instance object.) + */ + + off: function(eventType) { + var settings = this.settings + if (typeof settings["on" + eventType] !== "undefined") { + settings["on" + eventType] = function() {} + } + return this + }, + + /** + * 显示工具栏 + * Display toolbar + * + * @param {Function} [callback=function(){}] 回调函数 + * @returns {editormd} 返回editormd的实例对象 + */ + + showToolbar: function(callback) { + var settings = this.settings + if (settings.readOnly) { + return this + } + if ( + settings.toolbar && + (this.toolbar.length < 1 || + this.toolbar.find("." + this.classPrefix + "menu").html() === "") + ) { + this.setToolbar() + } + settings.toolbar = true + this.toolbar.show() + this.resize() + $.proxy(callback || function() {}, this)() + return this + }, + + /** + * 隐藏工具栏 + * Hide toolbar + * + * @param {Function} [callback=function(){}] 回调函数 + * @returns {editormd} this(editormd instance object.) + */ + + hideToolbar: function(callback) { + var settings = this.settings + settings.toolbar = false + this.toolbar.hide() + this.resize() + $.proxy(callback || function() {}, this)() + return this + }, + + /** + * 页面滚动时工具栏的固定定位 + * Set toolbar in window scroll auto fixed position + * + * @returns {editormd} 返回editormd的实例对象 + */ + + setToolbarAutoFixed: function(fixed) { + var state = this.state + var editor = this.editor + var toolbar = this.toolbar + var settings = this.settings + if (typeof fixed !== "undefined") { + settings.toolbarAutoFixed = fixed + } + var autoFixedHandle = function() { + var $window = $(window) + var top = $window.scrollTop() + if (!settings.toolbarAutoFixed) { + return false + } + if (top - editor.offset().top > 10 && top < editor.height()) { + toolbar.css({ + position: "fixed", + width: editor.width() + "px", + left: ($window.width() - editor.width()) / 2 + "px" + }) + } else { + toolbar.css({ + position: "absolute", + width: "100%", + left: 0 + }) + } + } + if ( + !state.fullscreen && + !state.preview && + settings.toolbar && + settings.toolbarAutoFixed + ) { + $(window).bind("scroll", autoFixedHandle) + } + return this + }, + + /** + * 配置和初始化工具栏 + * Set toolbar and Initialization + * + * @returns {editormd} 返回editormd的实例对象 + */ + + setToolbar: function() { + var settings = this.settings + if (settings.readOnly) { + return this + } + var editor = this.editor + var preview = this.preview + var classPrefix = this.classPrefix + var toolbar = (this.toolbar = editor.children( + "." + classPrefix + "toolbar" + )) + if (settings.toolbar && toolbar.length < 1) { + var toolbarHTML = + '
' + editor.append(toolbarHTML) + toolbar = this.toolbar = editor.children("." + classPrefix + "toolbar") + } + if (!settings.toolbar) { + toolbar.hide() + return this + } + toolbar.show() + var icons = + typeof settings.toolbarIcons === "function" + ? settings.toolbarIcons() + : typeof settings.toolbarIcons === "string" + ? editormd.toolbarModes[settings.toolbarIcons] + : settings.toolbarIcons + var toolbarMenu = toolbar.find("." + this.classPrefix + "menu"), + menu = "" + var pullRight = false + for (var i = 0, len = icons.length; i < len; i++) { + var name = icons[i] + if (name === "||") { + pullRight = true + } else if (name === "|") { + menu += '
  • |
  • ' + } else { + var isHeader = /h(\d)/.test(name) + var index = name + if (name === "watch" && !settings.watch) { + index = "unwatch" + } + var title = settings.lang.toolbar[index] + var iconTexts = settings.toolbarIconTexts[index] + var iconClass = settings.toolbarIconsClass[index] + title = typeof title === "undefined" ? "" : title + iconTexts = typeof iconTexts === "undefined" ? "" : iconTexts + iconClass = typeof iconClass === "undefined" ? "" : iconClass + var menuItem = pullRight ? '
  • ' : "
  • " + if ( + typeof settings.toolbarCustomIcons[name] !== "undefined" && + typeof settings.toolbarCustomIcons[name] !== "function" + ) { + menuItem += settings.toolbarCustomIcons[name] + } else { + menuItem += + '' + menuItem += + '' + + (isHeader + ? name.toUpperCase() + : iconClass === "" ? iconTexts : "") + + "" + menuItem += "" + } + menuItem += "
  • " + menu = pullRight ? menuItem + menu : menu + menuItem + } + } + toolbarMenu.html(menu) + toolbarMenu + .find('[title="Lowercase"]') + .attr("title", settings.lang.toolbar.lowercase) + toolbarMenu + .find('[title="ucwords"]') + .attr("title", settings.lang.toolbar.ucwords) + this.setToolbarHandler() + this.setToolbarAutoFixed() + return this + }, + + /** + * 工具栏图标事件处理对象序列 + * Get toolbar icons event handlers + * + * @param {Object} cm CodeMirror的实例对象 + * @param {String} name 要获取的事件处理器名称 + * @returns {Object} 返回处理对象序列 + */ + + dialogLockScreen: function() { + $.proxy(editormd.dialogLockScreen, this)() + return this + }, + dialogShowMask: function(dialog) { + $.proxy(editormd.dialogShowMask, this)(dialog) + return this + }, + getToolbarHandles: function(name) { + var toolbarHandlers = (this.toolbarHandlers = editormd.toolbarHandlers) + return name && typeof toolbarIconHandlers[name] !== "undefined" + ? toolbarHandlers[name] + : toolbarHandlers + }, + + /** + * 工具栏图标事件处理器 + * Bind toolbar icons event handle + * + * @returns {editormd} 返回editormd的实例对象 + */ + + setToolbarHandler: function() { + var _this = this + var settings = this.settings + if (!settings.toolbar || settings.readOnly) { + return this + } + var toolbar = this.toolbar + var cm = this.cm + var classPrefix = this.classPrefix + var toolbarIcons = (this.toolbarIcons = toolbar.find( + "." + classPrefix + "menu > li > a" + )) + var toolbarIconHandlers = this.getToolbarHandles() + toolbarIcons.bind(editormd.mouseOrTouch("click", "touchend"), function( + event + ) { + var icon = $(this).children(".fa") + var name = icon.attr("name") + var cursor = cm.getCursor() + var selection = cm.getSelection() + if (name === "") { + return + } + _this.activeIcon = icon + if (typeof toolbarIconHandlers[name] !== "undefined") { + $.proxy(toolbarIconHandlers[name], _this)(cm) + } else { + if (typeof settings.toolbarHandlers[name] !== "undefined") { + $.proxy(settings.toolbarHandlers[name], _this)( + cm, + icon, + cursor, + selection + ) + } + } + if ( + name !== "link" && + name !== "reference-link" && + name !== "image" && + name !== "code-block" && + name !== "preformatted-text" && + name !== "watch" && + name !== "preview" && + name !== "search" && + name !== "fullscreen" && + name !== "info" + ) { + cm.focus() + } + return false + }) + + return this + }, + + /** + * 动态创建对话框 + * Creating custom dialogs + * + * @param {Object} options 配置项键值对 Key/Value + * @returns {dialog} 返回创建的dialog的jQuery实例对象 + */ + + createDialog: function(options) { + return $.proxy(editormd.createDialog, this)(options) + }, + + /** + * 创建关于Editor.md的对话框 + * Create about Editor.md dialog + * + * @returns {editormd} 返回editormd的实例对象 + */ + + createInfoDialog: function() { + var _this = this + var editor = this.editor + var classPrefix = this.classPrefix + var infoDialogHTML = [ + '
    ', + '
    ', + '

    ' + + editormd.title + + "v" + + editormd.version + + "

    ", + "

    " + this.lang.description + "

    ", + '

    ' + + editormd.homePage + + '

    ', + '

    Copyright © 2015 Pandao, The MIT License.

    ', + "
    ", + '', + "
    " + ].join("\n") + editor.append(infoDialogHTML) + var infoDialog = (this.infoDialog = editor.children( + "." + classPrefix + "dialog-info" + )) + infoDialog + .find("." + classPrefix + "dialog-close") + .bind(editormd.mouseOrTouch("click", "touchend"), function() { + _this.hideInfoDialog() + }) + infoDialog + .css("border", editormd.isIE8 ? "1px solid #ddd" : "") + .css("z-index", editormd.dialogZindex) + .show() + this.infoDialogPosition() + return this + }, + + /** + * 关于Editor.md对话居中定位 + * Editor.md dialog position handle + * + * @returns {editormd} 返回editormd的实例对象 + */ + + infoDialogPosition: function() { + var infoDialog = this.infoDialog + var _infoDialogPosition = function() { + infoDialog.css({ + top: ($(window).height() - infoDialog.height()) / 2 + "px", + left: ($(window).width() - infoDialog.width()) / 2 + "px" + }) + } + _infoDialogPosition() + $(window).resize(_infoDialogPosition) + return this + }, + + /** + * 显示关于Editor.md + * Display about Editor.md dialog + * + * @returns {editormd} 返回editormd的实例对象 + */ + + showInfoDialog: function() { + $("html,body").css("overflow-x", "hidden") + var _this = this + var editor = this.editor + var settings = this.settings + var infoDialog = (this.infoDialog = editor.children( + "." + this.classPrefix + "dialog-info" + )) + if (infoDialog.length < 1) { + this.createInfoDialog() + } + this.lockScreen(true) + this.mask + .css({ + opacity: settings.dialogMaskOpacity, + backgroundColor: settings.dialogMaskBgColor + }) + .show() + infoDialog.css("z-index", editormd.dialogZindex).show() + this.infoDialogPosition() + return this + }, + + /** + * 隐藏关于Editor.md + * Hide about Editor.md dialog + * + * @returns {editormd} 返回editormd的实例对象 + */ + + hideInfoDialog: function() { + $("html,body").css("overflow-x", "") + this.infoDialog.hide() + this.mask.hide() + this.lockScreen(false) + return this + }, + + /** + * 锁屏 + * lock screen + * + * @param {Boolean} lock Boolean 布尔值,是否锁屏 + * @returns {editormd} 返回editormd的实例对象 + */ + + lockScreen: function(lock) { + editormd.lockScreen(lock) + this.resize() + return this + }, + + /** + * 编辑器界面重建,用于动态语言包或模块加载等 + * Recreate editor + * + * @returns {editormd} 返回editormd的实例对象 + */ + + recreate: function() { + var _this = this + var editor = this.editor + var settings = this.settings + this.codeMirror.remove() + this.setCodeMirror() + if (!settings.readOnly) { + if (editor.find(".editormd-dialog").length > 0) { + editor.find(".editormd-dialog").remove() + } + if (settings.toolbar) { + this.getToolbarHandles() + this.setToolbar() + } + } + this.loadedDisplay(true) + return this + }, + + /** + * 高亮预览HTML的pre代码部分 + * highlight of preview codes + * + * @returns {editormd} 返回editormd的实例对象 + */ + + previewCodeHighlight: function() { + var settings = this.settings + var previewContainer = this.previewContainer + if (settings.previewCodeHighlight) { + previewContainer.find("pre").addClass("prettyprint linenums") + if (typeof prettyPrint !== "undefined") { + prettyPrint() + } + } + + return this + }, + + /** + * 解析TeX(KaTeX)科学公式 + * TeX(KaTeX) Renderer + * + * @returns {editormd} 返回editormd的实例对象 + */ + + katexRender: function() { + if (timer === null) { + return this + } + this.previewContainer + .find("." + editormd.classNames.tex) + .each(function() { + var tex = $(this) + editormd.$katex.render(tex.text(), tex[0]) + tex.find(".katex").css("font-size", "1.6em") + }) + return this + }, + + /** + * 解析和渲染流程图及时序图 + * FlowChart and SequenceDiagram Renderer + * + * @returns {editormd} 返回editormd的实例对象 + */ + + flowChartAndSequenceDiagramRender: function() { + var $this = this + var settings = this.settings + var previewContainer = this.previewContainer + if (editormd.isIE8) { + return this + } + if (settings.flowChart) { + if (flowchartTimer === null) { + return this + } + previewContainer.find(".flowchart").flowChart() + } + + if (settings.sequenceDiagram) { + previewContainer + .find(".sequence-diagram") + .sequenceDiagram({ theme: "simple" }) + } + var preview = $this.preview + var codeMirror = $this.codeMirror + var codeView = codeMirror.find(".CodeMirror-scroll") + var height = codeView.height() + var scrollTop = codeView.scrollTop() + var percent = scrollTop / codeView[0].scrollHeight + var tocHeight = 0 + preview.find(".markdown-toc-list").each(function() { + tocHeight += $(this).height() + }) + var tocMenuHeight = preview.find(".editormd-toc-menu").height() + tocMenuHeight = !tocMenuHeight ? 0 : tocMenuHeight + if (settings.syncScrolling) { + if (scrollTop === 0) { + preview.scrollTop(0) + } else if (scrollTop + height >= codeView[0].scrollHeight - 16) { + preview.scrollTop(preview[0].scrollHeight) + } else { + preview.scrollTop( + (preview[0].scrollHeight + tocHeight + tocMenuHeight) * percent + ) + } + } + return this + }, + + /** + * 注册键盘快捷键处理 + * Register CodeMirror keyMaps (keyboard shortcuts). + * + * @param {Object} keyMap KeyMap key/value {"(Ctrl/Shift/Alt)-Key" : function(){}} + * @returns {editormd} return this + */ + + registerKeyMaps: function(keyMap) { + var _this = this + var cm = this.cm + var settings = this.settings + var toolbarHandlers = editormd.toolbarHandlers + var disabledKeyMaps = settings.disabledKeyMaps + keyMap = keyMap || null + if (keyMap) { + for (var i in keyMap) { + if ($.inArray(i, disabledKeyMaps) < 0) { + var map = {} + map[i] = keyMap[i] + cm.addKeyMap(keyMap) + } + } + } else { + for (var k in editormd.keyMaps) { + var _keyMap = editormd.keyMaps[k] + var handle = + typeof _keyMap === "string" + ? $.proxy(toolbarHandlers[_keyMap], _this) + : $.proxy(_keyMap, _this) + if ( + $.inArray(k, ["F9", "F10", "F11"]) < 0 && + $.inArray(k, disabledKeyMaps) < 0 + ) { + var _map = {} + _map[k] = handle + cm.addKeyMap(_map) + } + } + $(window).keydown(function(event) { + var keymaps = { + "120": "F9", + "121": "F10", + "122": "F11" + } + if ($.inArray(keymaps[event.keyCode], disabledKeyMaps) < 0) { + switch (event.keyCode) { + case 120: + $.proxy(toolbarHandlers.watch, _this)() + return false + case 121: + $.proxy(toolbarHandlers.preview, _this)() + return false + case 122: + $.proxy(toolbarHandlers.fullscreen, _this)() + return false + default: + break + } + } + }) + } + return this + }, + + /** + * 绑定同步滚动 + * @returns {editormd} return this + */ + + bindScrollEvent: function() { + var _this = this + var preview = this.preview + var settings = this.settings + var codeMirror = this.codeMirror + var mouseOrTouch = editormd.mouseOrTouch + if (!settings.syncScrolling) { + return this + } + var cmBindScroll = function() { + codeMirror + .find(".CodeMirror-scroll") + .bind(mouseOrTouch("scroll", "touchmove"), function(event) { + var height = $(this).height() + var scrollTop = $(this).scrollTop() + var percent = scrollTop / $(this)[0].scrollHeight + var tocHeight = 0 + preview.find(".markdown-toc-list").each(function() { + tocHeight += $(this).height() + }) + var tocMenuHeight = preview.find(".editormd-toc-menu").height() + tocMenuHeight = !tocMenuHeight ? 0 : tocMenuHeight + if (scrollTop === 0) { + preview.scrollTop(0) + } else if (scrollTop + height >= $(this)[0].scrollHeight - 16) { + preview.scrollTop(preview[0].scrollHeight) + } else { + preview.scrollTop( + (preview[0].scrollHeight + tocHeight + tocMenuHeight) * percent + ) + } + $.proxy(settings.onscroll, _this)(event) + }) + } + var cmUnbindScroll = function() { + codeMirror + .find(".CodeMirror-scroll") + .unbind(mouseOrTouch("scroll", "touchmove")) + } + var previewBindScroll = function() { + preview.bind(mouseOrTouch("scroll", "touchmove"), function(event) { + var height = $(this).height() + var scrollTop = $(this).scrollTop() + var percent = scrollTop / $(this)[0].scrollHeight + var codeView = codeMirror.find(".CodeMirror-scroll") + if (scrollTop === 0) { + codeView.scrollTop(0) + } else if (scrollTop + height >= $(this)[0].scrollHeight) { + codeView.scrollTop(codeView[0].scrollHeight) + } else { + codeView.scrollTop(codeView[0].scrollHeight * percent) + } + $.proxy(settings.onpreviewscroll, _this)(event) + }) + } + + var previewUnbindScroll = function() { + preview.unbind(mouseOrTouch("scroll", "touchmove")) + } + codeMirror.bind({ + mouseover: cmBindScroll, + mouseout: cmUnbindScroll, + touchstart: cmBindScroll, + touchend: cmUnbindScroll + }) + if (settings.syncScrolling === "single") { + return this + } + preview.bind({ + mouseover: previewBindScroll, + mouseout: previewUnbindScroll, + touchstart: previewBindScroll, + touchend: previewUnbindScroll + }) + return this + }, + + bindChangeEvent: function() { + var _this = this + var cm = this.cm + var settings = this.settings + + // if (!settings.syncScrolling) { + // return this; + // } + + cm.on("change", function(_cm, changeObj) { + if (settings.watch) { + _this.previewContainer.css( + "padding", + settings.autoHeight ? "20px 20px 50px 40px" : "20px" + ) + } + timer = setTimeout(function() { + clearTimeout(timer) + _this.save() + timer = null + }, settings.delay) + }) + return this + }, + + /** + * 加载队列完成之后的显示处理 + * Display handle of the module queues loaded after. + * + * @param {Boolean} recreate 是否为重建编辑器 + * @returns {editormd} 返回editormd的实例对象 + */ + + loadedDisplay: function(recreate) { + recreate = recreate || false + var _this = this + var editor = this.editor + var preview = this.preview + var settings = this.settings + this.containerMask.hide() + this.save() + if (settings.watch) { + preview.show() + } + editor.data("oldWidth", editor.width()).data("oldHeight", editor.height()) // 为了兼容Zepto + this.resize() + this.registerKeyMaps() + $(window).resize(function() { + _this.resize() + }) + this.bindScrollEvent().bindChangeEvent() + if (!recreate) { + $.proxy(settings.onload, this)() + } + this.state.loaded = true + return this + }, + + /** + * 设置编辑器的宽度 + * Set editor width + * + * @param {Number|String} width 编辑器宽度值 + * @returns {editormd} 返回editormd的实例对象 + */ + + width: function(width) { + this.editor.css("width", typeof width === "number" ? width + "px" : width) + this.resize() + return this + }, + + /** + * 设置编辑器的高度 + * Set editor height + * + * @param {Number|String} height 编辑器高度值 + * @returns {editormd} 返回editormd的实例对象 + */ + + height: function(height) { + this.editor.css( + "height", + typeof height === "number" ? height + "px" : height + ) + this.resize() + return this + }, + + /** + * 调整编辑器的尺寸和布局 + * Resize editor layout + * + * @param {Number|String} [width=null] 编辑器宽度值 + * @param {Number|String} [height=null] 编辑器高度值 + * @returns {editormd} 返回editormd的实例对象 + */ + + resize: function(width, height) { + width = width || null + height = height || null + var state = this.state + var editor = this.editor + var preview = this.preview + var toolbar = this.toolbar + var settings = this.settings + var codeMirror = this.codeMirror + if (width) { + editor.css("width", typeof width === "number" ? width + "px" : width) + } + if (settings.autoHeight && !state.fullscreen && !state.preview) { + editor.css({ + height: "auto", + minHeight: settings.minHeight + }) + codeMirror.css({ + height: "auto", + minHeight: settings.minHeight, + maxHeight: settings.maxHeight + }) + } else { + if (height) { + editor.css( + "height", + typeof height === "number" ? height + "px" : height + ) + } + if (state.fullscreen) { + editor.height($(window).height()) + } + if (settings.toolbar && !settings.readOnly) { + codeMirror + .css("margin-top", toolbar.height() + 1) + .height(editor.height() - toolbar.height()) + } else { + codeMirror.css("margin-top", 0).height(editor.height()) + } + } + if (settings.watch) { + codeMirror.width(editor.width() / 2) + preview.width(!state.preview ? editor.width() / 2 : editor.width()) + this.previewContainer.css( + "padding", + settings.autoHeight ? "20px 20px 50px 40px" : "20px" + ) + if (settings.toolbar && !settings.readOnly) { + preview.css("top", toolbar.height() + 1) + } else { + preview.css("top", 0) + } + if (settings.autoHeight && !state.fullscreen && !state.preview) { + preview.height("") + } else { + var previewHeight = + settings.toolbar && !settings.readOnly + ? editor.height() - toolbar.height() + : editor.height() + preview.height(previewHeight) + } + } else { + codeMirror.width(editor.width()) + preview.hide() + } + if (state.loaded) { + $.proxy(settings.onresize, this)() + } + return this + }, + + /** + * 解析和保存Markdown代码 + * Parse & Saving Markdown source code + * + * @returns {editormd} 返回editormd的实例对象 + */ + + save: function() { + if (timer === null) { + return this + } + var _this = this + var state = this.state + var settings = this.settings + var cm = this.cm + var cmValue = cm.getValue() + var previewContainer = this.previewContainer + if (settings.mode !== "gfm" && settings.mode !== "markdown") { + this.markdownTextarea.val(cmValue) + return this + } + var marked = editormd.$marked + var markdownToC = (this.markdownToC = []) + var rendererOptions = (this.markedRendererOptions = { + toc: settings.toc, + tocm: settings.tocm, + tocStartLevel: settings.tocStartLevel, + pageBreak: settings.pageBreak, + taskList: settings.taskList, + emoji: settings.emoji, + tex: settings.tex, + atLink: settings.atLink, // for @link + emailLink: settings.emailLink, // for mail address auto link + flowChart: settings.flowChart, + sequenceDiagram: settings.sequenceDiagram, + previewCodeHighlight: settings.previewCodeHighlight + }) + var markedOptions = (this.markedOptions = { + renderer: editormd.markedRenderer(markdownToC, rendererOptions), + gfm: true, + tables: true, + breaks: true, + pedantic: false, + sanitize: settings.htmlDecode ? false : true, // 关闭忽略HTML标签,即开启识别HTML标签,默认为false + smartLists: true, + smartypants: true + }) + marked.setOptions(markedOptions) + var newMarkdownDoc = editormd.$marked(cmValue, markedOptions) + newMarkdownDoc = editormd.filterHTMLTags( + newMarkdownDoc, + settings.htmlDecode + ) + this.markdownTextarea.text(cmValue) + cm.save() + if (settings.saveHTMLToTextarea) { + this.htmlTextarea.text(newMarkdownDoc) + } + if (settings.watch || (!settings.watch && state.preview)) { + previewContainer.html(newMarkdownDoc) + this.previewCodeHighlight() + if (settings.toc) { + var tocContainer = + settings.tocContainer === "" + ? previewContainer + : $(settings.tocContainer) + var tocMenu = tocContainer.find("." + this.classPrefix + "toc-menu") + tocContainer.attr( + "previewContainer", + settings.tocContainer === "" ? "true" : "false" + ) + if (settings.tocContainer !== "" && tocMenu.length > 0) { + tocMenu.remove() + } + editormd.markdownToCRenderer( + markdownToC, + tocContainer, + settings.tocDropdown, + settings.tocStartLevel + ) + if ( + settings.tocDropdown || + tocContainer.find("." + this.classPrefix + "toc-menu").length > 0 + ) { + editormd.tocDropdownMenu( + tocContainer, + settings.tocTitle !== "" ? settings.tocTitle : this.lang.tocTitle + ) + } + if (settings.tocContainer !== "") { + previewContainer.find(".markdown-toc").css("border", "none") + } + } + if (settings.tex) { + if (!editormd.kaTeXLoaded && settings.autoLoadModules) { + editormd.loadKaTeX(function() { + editormd.$katex = katex + editormd.kaTeXLoaded = true + _this.katexRender() + }) + } else { + editormd.$katex = katex + this.katexRender() + } + } + if (settings.flowChart || settings.sequenceDiagram) { + flowchartTimer = setTimeout(function() { + clearTimeout(flowchartTimer) + _this.flowChartAndSequenceDiagramRender() + flowchartTimer = null + }, 10) + } + if (state.loaded) { + $.proxy(settings.onchange, this)() + } + } + return this + }, + + /** + * 聚焦光标位置 + * Focusing the cursor position + * + * @returns {editormd} 返回editormd的实例对象 + */ + + focus: function() { + this.cm.focus() + return this + }, + + /** + * 设置光标的位置 + * Set cursor position + * + * @param {Object} cursor 要设置的光标位置键值对象,例:{line:1, ch:0} + * @returns {editormd} 返回editormd的实例对象 + */ + + setCursor: function(cursor) { + this.cm.setCursor(cursor) + return this + }, + + /** + * 获取当前光标的位置 + * Get the current position of the cursor + * + * @returns {Cursor} 返回一个光标Cursor对象 + */ + + getCursor: function() { + return this.cm.getCursor() + }, + + /** + * 设置光标选中的范围 + * Set cursor selected ranges + * + * @param {Object} from 开始位置的光标键值对象,例:{line:1, ch:0} + * @param {Object} to 结束位置的光标键值对象,例:{line:1, ch:0} + * @returns {editormd} 返回editormd的实例对象 + */ + + setSelection: function(from, to) { + this.cm.setSelection(from, to) + return this + }, + + /** + * 获取光标选中的文本 + * Get the texts from cursor selected + * + * @returns {String} 返回选中文本的字符串形式 + */ + + getSelection: function() { + return this.cm.getSelection() + }, + + /** + * 设置光标选中的文本范围 + * Set the cursor selection ranges + * + * @param {Array} ranges cursor selection ranges array + * @returns {Array} return this + */ + + setSelections: function(ranges) { + this.cm.setSelections(ranges) + return this + }, + + /** + * 获取光标选中的文本范围 + * Get the cursor selection ranges + * + * @returns {Array} return selection ranges array + */ + + getSelections: function() { + return this.cm.getSelections() + }, + + /** + * 替换当前光标选中的文本或在当前光标处插入新字符 + * Replace the text at the current cursor selected or insert a new character at the current cursor position + * + * @param {String} value 要插入的字符值 + * @returns {editormd} 返回editormd的实例对象 + */ + + replaceSelection: function(value) { + this.cm.replaceSelection(value) + return this + }, + + /** + * 在当前光标处插入新字符 + * Insert a new character at the current cursor position + * + * 同replaceSelection()方法 + * With the replaceSelection() method + * + * @param {String} value 要插入的字符值 + * @returns {editormd} 返回editormd的实例对象 + */ + + insertValue: function(value) { + this.replaceSelection(value) + return this + }, + + /** + * 追加markdown + * append Markdown to editor + * + * @param {String} md 要追加的markdown源文档 + * @returns {editormd} 返回editormd的实例对象 + */ + + appendMarkdown: function(md) { + var settings = this.settings + var cm = this.cm + cm.setValue(cm.getValue() + md) + return this + }, + + /** + * 设置和传入编辑器的markdown源文档 + * Set Markdown source document + * + * @param {String} md 要传入的markdown源文档 + * @returns {editormd} 返回editormd的实例对象 + */ + + setMarkdown: function(md) { + this.cm.setValue(md || this.settings.markdown) + return this + }, + + /** + * 获取编辑器的markdown源文档 + * Set Editor.md markdown/CodeMirror value + * + * @returns {editormd} 返回editormd的实例对象 + */ + + getMarkdown: function() { + return this.cm.getValue() + }, + + /** + * 获取编辑器的源文档 + * Get CodeMirror value + * + * @returns {editormd} 返回editormd的实例对象 + */ + + getValue: function() { + return this.cm.getValue() + }, + + /** + * 设置编辑器的源文档 + * Set CodeMirror value + * + * @param {String} value set code/value/string/text + * @returns {editormd} 返回editormd的实例对象 + */ + + setValue: function(value) { + this.cm.setValue(value) + return this + }, + + /** + * 清空编辑器 + * Empty CodeMirror editor container + * + * @returns {editormd} 返回editormd的实例对象 + */ + + clear: function() { + this.cm.setValue("") + return this + }, + + /** + * 获取解析后存放在Textarea的HTML源码 + * Get parsed html code from Textarea + * + * @returns {String} 返回HTML源码 + */ + + getHTML: function() { + if (!this.settings.saveHTMLToTextarea) { + alert("Error: settings.saveHTMLToTextarea == false") + return false + } + return this.htmlTextarea.val() + }, + + /** + * getHTML()的别名 + * getHTML (alias) + * + * @returns {String} Return html code 返回HTML源码 + */ + + getTextareaSavedHTML: function() { + return this.getHTML() + }, + + /** + * 获取预览窗口的HTML源码 + * Get html from preview container + * + * @returns {editormd} 返回editormd的实例对象 + */ + + getPreviewedHTML: function() { + if (!this.settings.watch) { + alert("Error: settings.watch == false") + return false + } + return this.previewContainer.html() + }, + + /** + * 开启实时预览 + * Enable real-time watching + * + * @returns {editormd} 返回editormd的实例对象 + */ + + watch: function(callback) { + var settings = this.settings + if ($.inArray(settings.mode, ["gfm", "markdown"]) < 0) { + return this + } + this.state.watching = settings.watch = true + this.preview.show() + if (this.toolbar) { + var watchIcon = settings.toolbarIconsClass.watch + var unWatchIcon = settings.toolbarIconsClass.unwatch + var icon = this.toolbar.find(".fa[name=watch]") + icon.parent().attr("title", settings.lang.toolbar.watch) + icon.removeClass(unWatchIcon).addClass(watchIcon) + } + this.codeMirror + .css("border-right", "1px solid #ddd") + .width(this.editor.width() / 2) + timer = 0 + this.save().resize() + if (!settings.onwatch) { + settings.onwatch = callback || function() {} + } + $.proxy(settings.onwatch, this)() + return this + }, + + /** + * 关闭实时预览 + * Disable real-time watching + * + * @returns {editormd} 返回editormd的实例对象 + */ + + unwatch: function(callback) { + var settings = this.settings + this.state.watching = settings.watch = false + this.preview.hide() + if (this.toolbar) { + var watchIcon = settings.toolbarIconsClass.watch + var unWatchIcon = settings.toolbarIconsClass.unwatch + var icon = this.toolbar.find(".fa[name=watch]") + icon.parent().attr("title", settings.lang.toolbar.unwatch) + icon.removeClass(watchIcon).addClass(unWatchIcon) + } + this.codeMirror.css("border-right", "none").width(this.editor.width()) + this.resize() + if (!settings.onunwatch) { + settings.onunwatch = callback || function() {} + } + $.proxy(settings.onunwatch, this)() + return this + }, + + /** + * 显示编辑器 + * Show editor + * + * @param {Function} [callback=function()] 回调函数 + * @returns {editormd} 返回editormd的实例对象 + */ + + show: function(callback) { + callback = callback || function() {} + var _this = this + this.editor.show(0, function() { + $.proxy(callback, _this)() + }) + return this + }, + + /** + * 隐藏编辑器 + * Hide editor + * + * @param {Function} [callback=function()] 回调函数 + * @returns {editormd} 返回editormd的实例对象 + */ + + hide: function(callback) { + callback = callback || function() {} + var _this = this + this.editor.hide(0, function() { + $.proxy(callback, _this)() + }) + return this + }, + + /** + * 隐藏编辑器部分,只预览HTML + * Enter preview html state + * + * @returns {editormd} 返回editormd的实例对象 + */ + + previewing: function() { + var _this = this + var editor = this.editor + var preview = this.preview + var toolbar = this.toolbar + var settings = this.settings + var codeMirror = this.codeMirror + var previewContainer = this.previewContainer + if ($.inArray(settings.mode, ["gfm", "markdown"]) < 0) { + return this + } + if (settings.toolbar && toolbar) { + toolbar.toggle() + toolbar.find(".fa[name=preview]").toggleClass("active") + } + codeMirror.toggle() + var escHandle = function(event) { + if (event.shiftKey && event.keyCode === 27) { + _this.previewed() + } + } + if (codeMirror.css("display") === "none") { + // 为了兼容Zepto,而不使用codeMirror.is(":hidden") + this.state.preview = true + if (this.state.fullscreen) { + preview.css("background", "#fff") + } + editor + .find("." + this.classPrefix + "preview-close-btn") + .show() + .bind(editormd.mouseOrTouch("click", "touchend"), function() { + _this.previewed() + }) + if (!settings.watch) { + this.save() + } else { + previewContainer.css("padding", "") + } + previewContainer.addClass(this.classPrefix + "preview-active") + preview.show().css({ + position: "", + top: 0, + width: editor.width(), + height: + settings.autoHeight && !this.state.fullscreen + ? "auto" + : editor.height() + }) + if (this.state.loaded) { + $.proxy(settings.onpreviewing, this)() + } + $(window).bind("keyup", escHandle) + } else { + $(window).unbind("keyup", escHandle) + this.previewed() + } + }, + + /** + * 显示编辑器部分,退出只预览HTML + * Exit preview html state + * + * @returns {editormd} 返回editormd的实例对象 + */ + + previewed: function() { + var editor = this.editor + var preview = this.preview + var toolbar = this.toolbar + var settings = this.settings + var previewContainer = this.previewContainer + var previewCloseBtn = editor.find( + "." + this.classPrefix + "preview-close-btn" + ) + this.state.preview = false + this.codeMirror.show() + if (settings.toolbar) { + toolbar.show() + } + preview[settings.watch ? "show" : "hide"]() + previewCloseBtn.hide().unbind(editormd.mouseOrTouch("click", "touchend")) + previewContainer.removeClass(this.classPrefix + "preview-active") + if (settings.watch) { + previewContainer.css("padding", "20px") + } + preview.css({ + background: null, + position: "absolute", + width: editor.width() / 2, + height: + settings.autoHeight && !this.state.fullscreen + ? "auto" + : editor.height() - toolbar.height(), + top: settings.toolbar ? toolbar.height() : 0 + }) + if (this.state.loaded) { + $.proxy(settings.onpreviewed, this)() + } + return this + }, + + /** + * 编辑器全屏显示 + * Fullscreen show + * + * @returns {editormd} 返回editormd的实例对象 + */ + + fullscreen: function() { + var _this = this + var state = this.state + var editor = this.editor + var preview = this.preview + var toolbar = this.toolbar + var settings = this.settings + var fullscreenClass = this.classPrefix + "fullscreen" + if (toolbar) { + toolbar + .find(".fa[name=fullscreen]") + .parent() + .toggleClass("active") + } + var escHandle = function(event) { + if (!event.shiftKey && event.keyCode === 27) { + if (state.fullscreen) { + _this.fullscreenExit() + } + } + } + if (!editor.hasClass(fullscreenClass)) { + state.fullscreen = true + $("html,body").css("overflow", "hidden") + editor + .css({ + width: $(window).width(), + height: $(window).height() + }) + .addClass(fullscreenClass) + this.resize() + $.proxy(settings.onfullscreen, this)() + $(window).bind("keyup", escHandle) + } else { + $(window).unbind("keyup", escHandle) + this.fullscreenExit() + } + return this + }, + + /** + * 编辑器退出全屏显示 + * Exit fullscreen state + * + * @returns {editormd} 返回editormd的实例对象 + */ + + fullscreenExit: function() { + var editor = this.editor + var settings = this.settings + var toolbar = this.toolbar + var fullscreenClass = this.classPrefix + "fullscreen" + this.state.fullscreen = false + if (toolbar) { + toolbar + .find(".fa[name=fullscreen]") + .parent() + .removeClass("active") + } + $("html,body").css("overflow", "") + editor + .css({ + width: editor.data("oldWidth"), + height: editor.data("oldHeight") + }) + .removeClass(fullscreenClass) + this.resize() + $.proxy(settings.onfullscreenExit, this)() + return this + }, + + /** + * 加载并执行插件 + * Load and execute the plugin + * + * @param {String} name plugin name / function name + * @param {String} path plugin load path + * @returns {editormd} 返回editormd的实例对象 + */ + + executePlugin: function(name, path) { + var _this = this + var cm = this.cm + var settings = this.settings + path = settings.pluginPath + path + if (typeof define === "function") { + if (typeof this[name] === "undefined") { + alert( + "Error: " + + name + + " plugin is not found, you are not load this plugin." + ) + return this + } + this[name](cm) + return this + } + if ($.inArray(path, editormd.loadFiles.plugin) < 0) { + editormd.loadPlugin(path, function() { + editormd.loadPlugins[name] = _this[name] + _this[name](cm) + }) + } else { + $.proxy(editormd.loadPlugins[name], this)(cm) + } + return this + }, + + /** + * 搜索替换 + * Search & replace + * + * @param {String} command CodeMirror serach commands, "find, fintNext, fintPrev, clearSearch, replace, replaceAll" + * @returns {editormd} return this + */ + + search: function(command) { + var settings = this.settings + if (!settings.searchReplace) { + alert("Error: settings.searchReplace == false") + return this + } + if (!settings.readOnly) { + this.cm.execCommand(command || "find") + } + return this + }, + searchReplace: function() { + this.search("replace") + return this + }, + searchReplaceAll: function() { + this.search("replaceAll") + return this + } + } + + editormd.fn.init.prototype = editormd.fn + + /** + * 锁屏 + * lock screen when dialog opening + * + * @returns {void} + */ + + editormd.dialogLockScreen = function() { + var settings = this.settings || { dialogLockScreen: true } + if (settings.dialogLockScreen) { + $("html,body").css("overflow", "hidden") + this.resize() + } + } + + /** + * 显示透明背景层 + * Display mask layer when dialog opening + * + * @param {Object} dialog dialog jQuery object + * @returns {void} + */ + + editormd.dialogShowMask = function(dialog) { + var editor = this.editor + var settings = this.settings || { dialogShowMask: true } + dialog.css({ + top: ($(window).height() - dialog.height()) / 2 + "px", + left: ($(window).width() - dialog.width()) / 2 + "px" + }) + if (settings.dialogShowMask) { + editor + .children("." + this.classPrefix + "mask") + .css("z-index", parseInt(dialog.css("z-index")) - 1) + .show() + } + } + + editormd.toolbarHandlers = { + undo: function() { + this.cm.undo() + }, + redo: function() { + this.cm.redo() + }, + bold: function() { + var cm = this.cm + var cursor = cm.getCursor() + var selection = cm.getSelection() + cm.replaceSelection("**" + selection + "**") + if (selection === "") { + cm.setCursor(cursor.line, cursor.ch + 2) + } + }, + del: function() { + var cm = this.cm + var cursor = cm.getCursor() + var selection = cm.getSelection() + cm.replaceSelection("~~" + selection + "~~") + if (selection === "") { + cm.setCursor(cursor.line, cursor.ch + 2) + } + }, + italic: function() { + var cm = this.cm + var cursor = cm.getCursor() + var selection = cm.getSelection() + cm.replaceSelection("*" + selection + "*") + if (selection === "") { + cm.setCursor(cursor.line, cursor.ch + 1) + } + }, + quote: function() { + var cm = this.cm + var cursor = cm.getCursor() + var selection = cm.getSelection() + if (cursor.ch !== 0) { + cm.setCursor(cursor.line, 0) + cm.replaceSelection("> " + selection) + cm.setCursor(cursor.line, cursor.ch + 2) + } else { + cm.replaceSelection("> " + selection) + } + //cm.replaceSelection("> " + selection); + //cm.setCursor(cursor.line, (selection === "") ? cursor.ch + 2 : cursor.ch + selection.length + 2); + }, + ucfirst: function() { + var cm = this.cm + var selection = cm.getSelection() + var selections = cm.listSelections() + cm.replaceSelection(editormd.firstUpperCase(selection)) + cm.setSelections(selections) + }, + ucwords: function() { + var cm = this.cm + var selection = cm.getSelection() + var selections = cm.listSelections() + cm.replaceSelection(editormd.wordsFirstUpperCase(selection)) + cm.setSelections(selections) + }, + uppercase: function() { + var cm = this.cm + var selection = cm.getSelection() + var selections = cm.listSelections() + cm.replaceSelection(selection.toUpperCase()) + cm.setSelections(selections) + }, + lowercase: function() { + var cm = this.cm + var cursor = cm.getCursor() + var selection = cm.getSelection() + var selections = cm.listSelections() + cm.replaceSelection(selection.toLowerCase()) + cm.setSelections(selections) + }, + h1: function() { + var cm = this.cm + var cursor = cm.getCursor() + var selection = cm.getSelection() + if (cursor.ch !== 0) { + cm.setCursor(cursor.line, 0) + cm.replaceSelection("# " + selection) + cm.setCursor(cursor.line, cursor.ch + 2) + } else { + cm.replaceSelection("# " + selection) + } + }, + h2: function() { + var cm = this.cm + var cursor = cm.getCursor() + var selection = cm.getSelection() + if (cursor.ch !== 0) { + cm.setCursor(cursor.line, 0) + cm.replaceSelection("## " + selection) + cm.setCursor(cursor.line, cursor.ch + 3) + } else { + cm.replaceSelection("## " + selection) + } + }, + h3: function() { + var cm = this.cm + var cursor = cm.getCursor() + var selection = cm.getSelection() + if (cursor.ch !== 0) { + cm.setCursor(cursor.line, 0) + cm.replaceSelection("### " + selection) + cm.setCursor(cursor.line, cursor.ch + 4) + } else { + cm.replaceSelection("### " + selection) + } + }, + h4: function() { + var cm = this.cm + var cursor = cm.getCursor() + var selection = cm.getSelection() + if (cursor.ch !== 0) { + cm.setCursor(cursor.line, 0) + cm.replaceSelection("#### " + selection) + cm.setCursor(cursor.line, cursor.ch + 5) + } else { + cm.replaceSelection("#### " + selection) + } + }, + h5: function() { + var cm = this.cm + var cursor = cm.getCursor() + var selection = cm.getSelection() + if (cursor.ch !== 0) { + cm.setCursor(cursor.line, 0) + cm.replaceSelection("##### " + selection) + cm.setCursor(cursor.line, cursor.ch + 6) + } else { + cm.replaceSelection("##### " + selection) + } + }, + h6: function() { + var cm = this.cm + var cursor = cm.getCursor() + var selection = cm.getSelection() + if (cursor.ch !== 0) { + cm.setCursor(cursor.line, 0) + cm.replaceSelection("###### " + selection) + cm.setCursor(cursor.line, cursor.ch + 7) + } else { + cm.replaceSelection("###### " + selection) + } + }, + "list-ul": function() { + var cm = this.cm + var cursor = cm.getCursor() + var selection = cm.getSelection() + if (selection === "") { + cm.replaceSelection("- " + selection) + } else { + var selectionText = selection.split("\n") + for (var i = 0, len = selectionText.length; i < len; i++) { + selectionText[i] = + selectionText[i] === "" ? "" : "- " + selectionText[i] + } + cm.replaceSelection(selectionText.join("\n")) + } + }, + "list-ol": function() { + var cm = this.cm + var cursor = cm.getCursor() + var selection = cm.getSelection() + if (selection === "") { + cm.replaceSelection("1. " + selection) + } else { + var selectionText = selection.split("\n") + for (var i = 0, len = selectionText.length; i < len; i++) { + selectionText[i] = + selectionText[i] === "" ? "" : i + 1 + ". " + selectionText[i] + } + cm.replaceSelection(selectionText.join("\n")) + } + }, + hr: function() { + var cm = this.cm + var cursor = cm.getCursor() + var selection = cm.getSelection() + cm.replaceSelection( + (cursor.ch !== 0 ? "\n\n" : "\n") + "------------\n\n" + ) + }, + tex: function() { + if (!this.settings.tex) { + alert("settings.tex === false") + return this + } + var cm = this.cm + var cursor = cm.getCursor() + var selection = cm.getSelection() + cm.replaceSelection("$$" + selection + "$$") + if (selection === "") { + cm.setCursor(cursor.line, cursor.ch + 2) + } + }, + link: function() { + this.executePlugin("linkDialog", "link-dialog/link-dialog") + }, + "reference-link": function() { + this.executePlugin( + "referenceLinkDialog", + "reference-link-dialog/reference-link-dialog" + ) + }, + pagebreak: function() { + if (!this.settings.pageBreak) { + alert("settings.pageBreak === false") + return this + } + var cm = this.cm + var selection = cm.getSelection() + cm.replaceSelection("\r\n[========]\r\n") + }, + image: function() { + this.executePlugin("imageDialog", "image-dialog/image-dialog") + }, + code: function() { + var cm = this.cm + var cursor = cm.getCursor() + var selection = cm.getSelection() + cm.replaceSelection("`" + selection + "`") + if (selection === "") { + cm.setCursor(cursor.line, cursor.ch + 1) + } + }, + "code-block": function() { + this.executePlugin( + "codeBlockDialog", + "code-block-dialog/code-block-dialog" + ) + }, + "preformatted-text": function() { + this.executePlugin( + "preformattedTextDialog", + "preformatted-text-dialog/preformatted-text-dialog" + ) + }, + table: function() { + this.executePlugin("tableDialog", "table-dialog/table-dialog") + }, + datetime: function() { + var cm = this.cm + var selection = cm.getSelection() + var date = new Date() + var langName = this.settings.lang.name + var datefmt = + editormd.dateFormat() + + " " + + editormd.dateFormat( + langName === "zh-cn" || langName === "zh-tw" + ? "cn-week-day" + : "week-day" + ) + cm.replaceSelection(datefmt) + }, + emoji: function() { + this.executePlugin("emojiDialog", "emoji-dialog/emoji-dialog") + }, + "html-entities": function() { + this.executePlugin( + "htmlEntitiesDialog", + "html-entities-dialog/html-entities-dialog" + ) + }, + "goto-line": function() { + this.executePlugin("gotoLineDialog", "goto-line-dialog/goto-line-dialog") + }, + watch: function() { + this[this.settings.watch ? "unwatch" : "watch"]() + }, + preview: function() { + this.previewing() + }, + fullscreen: function() { + this.fullscreen() + }, + clear: function() { + this.clear() + }, + search: function() { + this.search() + }, + help: function() { + this.executePlugin("helpDialog", "help-dialog/help-dialog") + }, + info: function() { + this.showInfoDialog() + } + } + + var isMac = navigator.platform.toUpperCase().indexOf("MAC") >= 0 + var key = isMac ? "Cmd" : "Ctrl" + editormd.keyMaps = { + [key + "-1"]: "h1", + [key + "-2"]: "h2", + [key + "-3"]: "h3", + [key + "-4"]: "h4", + [key + "-5"]: "h5", + [key + "-6"]: "h6", + [key + "-B"]: "bold", // if this is string == editormd.toolbarHandlers.xxxx + [key + "-D"]: "datetime", + [key + "Ctrl-E"]: function() { + // emoji + var cm = this.cm + var cursor = cm.getCursor() + var selection = cm.getSelection() + if (!this.settings.emoji) { + alert("Error: settings.emoji == false") + return + } + cm.replaceSelection(":" + selection + ":") + if (selection === "") { + cm.setCursor(cursor.line, cursor.ch + 1) + } + }, + [key + "-Alt-G"]: "goto-line", + [key + "-H"]: "hr", + [key + "-I"]: "italic", + [key + "-K"]: "code", + "Ctrl-L": function() { + var cm = this.cm + var cursor = cm.getCursor() + var selection = cm.getSelection() + var title = selection === "" ? "" : ' "' + selection + '"' + cm.replaceSelection("[" + selection + "](" + title + ")") + if (selection === "") { + cm.setCursor(cursor.line, cursor.ch + 1) + } + }, + [key + "-U"]: "list-ul", + "Shift-Ctrl-A": function() { + var cm = this.cm + var cursor = cm.getCursor() + var selection = cm.getSelection() + if (!this.settings.atLink) { + alert("Error: settings.atLink == false") + return + } + cm.replaceSelection("@" + selection) + if (selection === "") { + cm.setCursor(cursor.line, cursor.ch + 1) + } + }, + ["Shift" + key + "-C"]: "code", + ["Shift" + key + "Q"]: "quote", + ["Shift" + key + "S"]: "del", + ["Shift" + key + "K"]: "tex", // KaTeX + "Shift-Alt-C": function() { + var cm = this.cm + var cursor = cm.getCursor() + var selection = cm.getSelection() + cm.replaceSelection(["```", selection, "```"].join("\n")) + if (selection === "") { + cm.setCursor(cursor.line, cursor.ch + 3) + } + }, + ["Shift-" + key + "-Alt-C"]: "code-block", + ["Shift-" + key + "-H"]: "html-entities", + "Shift-Alt-H": "help", + ["Shift-" + key + "-E"]: "emoji", + ["Shift-" + key + "-U"]: "uppercase", + "Shift-Alt-U": "ucwords", + ["Shift-" + key + "-Alt-U"]: "ucfirst", + "Shift-Alt-L": "lowercase", + ["Shift-" + key + "-I"]: function() { + var cm = this.cm + var cursor = cm.getCursor() + var selection = cm.getSelection() + var title = selection === "" ? "" : ' "' + selection + '"' + cm.replaceSelection("![" + selection + "](" + title + ")") + if (selection === "") { + cm.setCursor(cursor.line, cursor.ch + 4) + } + }, + ["Shift-" + key + "-Alt-I"]: "image", + ["Shift-" + key + "-L"]: "link", + ["Shift-" + key + "-O"]: "list-ol", + ["Shift-" + key + "-P"]: "preformatted-text", + ["Shift-" + key + "-T"]: "table", + "Shift-Alt-P": "pagebreak", + F9: "watch", + F10: "preview", + F11: "fullscreen" + } + + /** + * 清除字符串两边的空格 + * Clear the space of strings both sides. + * + * @param {String} str string + * @returns {String} trimed string + */ + + var trim = function(str) { + return !String.prototype.trim + ? str.replace(/^[\s\uFEFF\xA0]+|[\s\uFEFF\xA0]+$/g, "") + : str.trim() + } + + editormd.trim = trim + + /** + * 所有单词首字母大写 + * Words first to uppercase + * + * @param {String} str string + * @returns {String} string + */ + + var ucwords = function(str) { + return str.toLowerCase().replace(/\b(\w)|\s(\w)/g, function($1) { + return $1.toUpperCase() + }) + } + + editormd.ucwords = editormd.wordsFirstUpperCase = ucwords + + /** + * 字符串首字母大写 + * Only string first char to uppercase + * + * @param {String} str string + * @returns {String} string + */ + + var firstUpperCase = function(str) { + return str.toLowerCase().replace(/\b(\w)/, function($1) { + return $1.toUpperCase() + }) + } + + var ucfirst = firstUpperCase + editormd.firstUpperCase = editormd.ucfirst = firstUpperCase + editormd.urls = { + atLinkBase: "https://github.com/" + } + editormd.regexs = { + atLink: /@(\w+)/g, + email: /(\w+)@(\w+)\.(\w+)\.?(\w+)?/g, + emailLink: /(mailto:)?([\w\.\_]+)@(\w+)\.(\w+)\.?(\w+)?/g, + emoji: /:([\w\+-]+):/g, + emojiDatetime: /(\d{1,2}:\d{1,2}:\d{1,2})/g, + twemoji: /:(tw-([\w]+)-?(\w+)?):/g, + fontAwesome: /:(fa-([\w]+)(-(\w+)){0,}):/g, + editormdLogo: /:(editormd-logo-?(\w+)?):/g, + pageBreak: /^\[[=]{8,}\]$/ + } + // Emoji graphics files url path + editormd.emoji = { + path: "http://www.emoji-cheat-sheet.com/graphics/emojis/", + ext: ".png" + } + // Twitter Emoji (Twemoji) graphics files url path + editormd.twemoji = { + path: "http://twemoji.maxcdn.com/36x36/", + ext: ".png" + } + + /** + * 自定义marked的解析器 + * Custom Marked renderer rules + * + * @param {Array} markdownToC 传入用于接收TOC的数组 + * @returns {Renderer} markedRenderer 返回marked的Renderer自定义对象 + */ + + editormd.markedRenderer = function(markdownToC, options) { + var defaults = { + toc: true, // Table of contents + tocm: false, + tocStartLevel: 1, // Said from H1 to create ToC + pageBreak: true, + atLink: true, // for @link + emailLink: true, // for mail address auto link + taskList: false, // Enable Github Flavored Markdown task lists + emoji: false, // :emoji: , Support Twemoji, fontAwesome, Editor.md logo emojis. + tex: false, // TeX(LaTeX), based on KaTeX + flowChart: false, // flowChart.js only support IE9+ + sequenceDiagram: false // sequenceDiagram.js only support IE9+ + } + var settings = $.extend(defaults, options || {}) + var marked = editormd.$marked + var markedRenderer = new marked.Renderer() + markdownToC = markdownToC || [] + + var regexs = editormd.regexs + var atLinkReg = regexs.atLink + var emojiReg = regexs.emoji + var emailReg = regexs.email + var emailLinkReg = regexs.emailLink + var twemojiReg = regexs.twemoji + var faIconReg = regexs.fontAwesome + var editormdLogoReg = regexs.editormdLogo + var pageBreakReg = regexs.pageBreak + + markedRenderer.emoji = function(text) { + text = text.replace(editormd.regexs.emojiDatetime, function($1) { + return $1.replace(/:/g, ":") + }) + + var matchs = text.match(emojiReg) + + if (!matchs || !settings.emoji) { + return text + } + + for (var i = 0, len = matchs.length; i < len; i++) { + if (matchs[i] === ":+1:") { + matchs[i] = ":\\+1:" + } + + text = text.replace(new RegExp(matchs[i]), function($1, $2) { + var faMatchs = $1.match(faIconReg) + var name = $1.replace(/:/g, "") + if (faMatchs) { + for (var fa = 0, len1 = faMatchs.length; fa < len1; fa++) { + var faName = faMatchs[fa].replace(/:/g, "") + return ( + '' + ) + } + } else { + var emdlogoMathcs = $1.match(editormdLogoReg) + var twemojiMatchs = $1.match(twemojiReg) + if (emdlogoMathcs) { + for (var x = 0, len2 = emdlogoMathcs.length; x < len2; x++) { + var logoName = emdlogoMathcs[x].replace(/:/g, "") + return ( + '' + ) + } + } else if (twemojiMatchs) { + for (var t = 0, len3 = twemojiMatchs.length; t < len3; t++) { + var twe = twemojiMatchs[t].replace(/:/g, "").replace("tw-", "") + return ( + 'twemoji-' +
+                  twe +
+                  '' + ) + } + } else { + var src = name === "+1" ? "plus1" : name + src = src === "black_large_square" ? "black_square" : src + src = src === "moon" ? "waxing_gibbous_moon" : src + return ( + ':' +
+                name +
+                ':' + ) + } + } + }) + } + return text + } + + markedRenderer.atLink = function(text) { + if (atLinkReg.test(text)) { + if (settings.atLink) { + text = text.replace(emailReg, function($1, $2, $3, $4) { + return $1.replace(/@/g, "_#_@_#_") + }) + text = text + .replace(atLinkReg, function($1, $2) { + return ( + '' + + $1 + + "" + ) + }) + .replace(/_#_@_#_/g, "@") + } + if (settings.emailLink) { + text = text.replace(emailLinkReg, function($1, $2, $3, $4, $5) { + return !$2 && + $.inArray($5, "jpg|jpeg|png|gif|webp|ico|icon|pdf".split("|")) < 0 + ? '' + $1 + "" + : $1 + }) + } + return text + } + return text + } + + markedRenderer.link = function(href, title, text) { + if (this.options.sanitize) { + try { + var prot = decodeURIComponent(unescape(href)) + .replace(/[^\w:]/g, "") + .toLowerCase() + if (prot.indexOf("javascript:") === 0) { + return "" + } + } catch (e) { + return "" + } + } + var out = '' + text.replace(/@/g, "@") + "" + } + if (title) { + out += ' title="' + title + '"' + } + out += ">" + text + "" + return out + } + + markedRenderer.heading = function(text, level, raw) { + var linkText = text + var hasLinkReg = /\s*]*)\>(.*)<\/a\>\s*/ + var getLinkTextReg = /\s*]+)\>([^\>]*)<\/a\>\s*/g + if (hasLinkReg.test(text)) { + var tempText = [] + text = text.split(/]+)\>([^\>]*)<\/a\>/) + for (var i = 0, len = text.length; i < len; i++) { + tempText.push(text[i].replace(/\s*href\=\"(.*)\"\s*/g, "")) + } + text = tempText.join(" ") + } + text = trim(text) + var escapedText = text.toLowerCase().replace(/[^\w]+/g, "-") + var toc = { + text: text, + level: level, + slug: escapedText + } + var isChinese = /^[\u4e00-\u9fa5]+$/.test(text) + var id = isChinese + ? escape(text).replace(/\%/g, "") + : text.toLowerCase().replace(/[^\w]+/g, "-") + markdownToC.push(toc) + var headingHTML = + "' + headingHTML += '' + headingHTML += '' + headingHTML += hasLinkReg + ? this.atLink(this.emoji(linkText)) + : this.atLink(this.emoji(text)) + headingHTML += "" + return headingHTML + } + markedRenderer.pageBreak = function(text) { + if (pageBreakReg.test(text) && settings.pageBreak) { + text = + '
    ' + } + return text + } + markedRenderer.paragraph = function(text) { + var isTeXInline = /\$\$(.*)\$\$/g.test(text) + var isTeXLine = /^\$\$(.*)\$\$$/.test(text) + var isTeXAddClass = isTeXLine + ? ' class="' + editormd.classNames.tex + '"' + : "" + var isToC = settings.tocm + ? /^(\[TOC\]|\[TOCM\])$/.test(text) + : /^\[TOC\]$/.test(text) + var isToCMenu = /^\[TOCM\]$/.test(text) + if (!isTeXLine && isTeXInline) { + text = text.replace(/(\$\$([^\$]*)\$\$)+/g, function($1, $2) { + return ( + '' + + $2.replace(/\$/g, "") + + "" + ) + }) + } else { + text = isTeXLine ? text.replace(/\$/g, "") : text + } + var tocHTML = + '
    ' + text + "
    " + return isToC + ? isToCMenu + ? '
    ' + tocHTML + "

    " + : tocHTML + : pageBreakReg.test(text) + ? this.pageBreak(text) + : "" + + this.atLink(this.emoji(text)) + + "

    \n" + } + markedRenderer.code = function(code, lang, escaped) { + if (lang === "seq" || lang === "sequence") { + return '
    ' + code + "
    " + } else if (lang === "flow") { + return '
    ' + code + "
    " + } else if (lang === "math" || lang === "latex" || lang === "katex") { + return '

    ' + code + "

    " + } else { + return marked.Renderer.prototype.code.apply(this, arguments) + } + } + markedRenderer.tablecell = function(content, flags) { + var type = flags.header ? "th" : "td" + var tag = flags.align + ? "<" + type + ' style="text-align:' + flags.align + '">' + : "<" + type + ">" + return tag + this.atLink(this.emoji(content)) + "\n" + } + markedRenderer.listitem = function(text) { + if (settings.taskList && /^\s*\[[x\s]\]\s*/.test(text)) { + text = text + .replace( + /^\s*\[\s\]\s*/, + ' ' + ) + .replace( + /^\s*\[x\]\s*/, + ' ' + ) + return ( + '
  • ' + + this.atLink(this.emoji(text)) + + "
  • " + ) + } else { + return "
  • " + this.atLink(this.emoji(text)) + "
  • " + } + } + return markedRenderer + } + + /** + * + * 生成TOC(Table of Contents) + * Creating ToC (Table of Contents) + * + * @param {Array} toc 从marked获取的TOC数组列表 + * @param {Element} container 插入TOC的容器元素 + * @param {Integer} startLevel Hx 起始层级 + * @returns {Object} tocContainer 返回ToC列表容器层的jQuery对象元素 + */ + + editormd.markdownToCRenderer = function( + toc, + container, + tocDropdown, + startLevel + ) { + var html = "" + var lastLevelArr = [0] + // var lastLevel = 0 + var classPrefix = this.classPrefix + startLevel = startLevel || 1 + for (var i = 0, len = toc.length; i < len; i++) { + var text = toc[i].text + var level = toc[i].level + var stageLen = lastLevelArr.length + var lastLevel = lastLevelArr[stageLen - 1] + var goBack = 0 + if (level < startLevel) { + continue + } + if (level > lastLevel) { + html += "" + lastLevelArr.push(level) + } else if (level < lastLevel) { + for (var j = stageLen - 1; j >= 0; j--) { + if (level <= lastLevelArr[j]) { + goBack ++ + lastLevelArr.pop() + } else { + lastLevelArr.push(level) + break + } + } + html += new Array(goBack + 1).join("") + } else { + html += "" + } + html += + '
  • ' + + text + + "
      " + } + + var tocContainer = container.find(".markdown-toc") + + if ( + tocContainer.length < 1 && + container.attr("previewContainer") === "false" + ) { + var tocHTML = + '
      ' + tocHTML = tocDropdown + ? '
      ' + tocHTML + "
      " + : tocHTML + container.html(tocHTML) + tocContainer = container.find(".markdown-toc") + } + if (tocDropdown) { + tocContainer.wrap('

      ') + } + tocContainer + .html('
        ') + .children(".markdown-toc-list") + .html(html.replace(/\r?\n?<\/ul\>/g, "")) + return tocContainer + } + + /** + * + * 生成TOC下拉菜单 + * Creating ToC dropdown menu + * + * @param {Object} container 插入TOC的容器jQuery对象元素 + * @param {String} tocTitle ToC title + * @returns {Object} return toc-menu object + */ + + editormd.tocDropdownMenu = function(container, tocTitle) { + tocTitle = tocTitle || "Table of Contents" + var zindex = 400 + var tocMenus = container.find("." + this.classPrefix + "toc-menu") + tocMenus.each(function() { + var $this = $(this) + var toc = $this.children(".markdown-toc") + var icon = '' + var btn = + '' + + icon + + tocTitle + + "" + var menu = toc.children("ul") + var list = menu.find("li") + toc.append(btn) + list.first().before("
      • " + tocTitle + " " + icon + "

      • ") + $this + .mouseover(function() { + menu.show() + list.each(function() { + var li = $(this) + var ul = li.children("ul") + if (ul.html() === "") { + ul.remove() + } + if (ul.length > 0 && ul.html() !== "") { + var firstA = li.children("a").first() + if (firstA.children(".fa").length < 1) { + firstA.append( + $(icon).css({ float: "right", paddingTop: "4px" }) + ) + } + } + li + .mouseover(function() { + ul.css("z-index", zindex).show() + zindex += 1 + }) + .mouseleave(function() { + ul.hide() + }) + }) + }) + .mouseleave(function() { + menu.hide() + }) + }) + + return tocMenus + } + + /** + * 简单地过滤指定的HTML标签 + * Filter custom html tags + * + * @param {String} html 要过滤HTML + * @param {String} filters 要过滤的标签 + * @returns {String} html 返回过滤的HTML + */ + + editormd.filterHTMLTags = function(html, filters) { + if (typeof html !== "string") { + html = new String(html) + } + if (typeof filters !== "string") { + return html + } + var expression = filters.split("|") + var filterTags = expression[0].split(",") + var attrs = expression[1] + for (var i = 0, len = filterTags.length; i < len; i++) { + var tag = filterTags[i] + html = html.replace( + new RegExp("]*)>([^>]*)", "igm"), + "" + ) + } + //return html; + if (typeof attrs !== "undefined") { + var htmlTagRegex = /<(\w+)\s*([^\>]*)\>([^\>]*)<\/(\w+)\>/gi + if (attrs === "*") { + html = html.replace(htmlTagRegex, function($1, $2, $3, $4, $5) { + return "<" + $2 + ">" + $4 + "" + }) + } else if (attrs === "on*") { + html = html.replace(htmlTagRegex, function($1, $2, $3, $4, $5) { + var el = $("<" + $2 + ">" + $4 + "") + var _attrs = $($1)[0].attributes + var $attrs = {} + $.each(_attrs, function(i, e) { + if (e.nodeName !== '"') { + $attrs[e.nodeName] = e.nodeValue + // fixed will cause xss problem + if (e.nodeName === "href" && e.nodeValue.toLowerCase().indexOf('javascript:') >= 0) { + $attrs[e.nodeName] = 'javascript:;'; + } + } + }) + $.each($attrs, function(i) { + if (i.indexOf("on") === 0) { + delete $attrs[i] + } + }) + el.attr($attrs) + var text = typeof el[1] !== "undefined" ? $(el[1]).text() : "" + // FIXED 使用 on* 过滤标签的属性,图片加链接的语法会出错的问题 + if ($2 && !isNaN($2) && $2 !== $5) { + text += "<" + $2 + ">" + $4 + "" + } else if ($2 && isNaN($2) && $2 !== $5) { + text += "" + return el[0].outerHTML + text + } else { + return el[0].outerHTML + text + } + }) + } else { + html = html.replace(htmlTagRegex, function($1, $2, $3, $4) { + var filterAttrs = attrs.split(",") + var el = $($1) + el.html($4) + $.each(filterAttrs, function(i) { + el.attr(filterAttrs[i], null) + }) + return el[0].outerHTML + }) + } + } + return html + } + + /** + * 将Markdown文档解析为HTML用于前台显示 + * Parse Markdown to HTML for Font-end preview. + * + * @param {String} id 用于显示HTML的对象ID + * @param {Object} [options={}] 配置选项,可选 + * @returns {Object} div 返回jQuery对象元素 + */ + + editormd.markdownToHTML = function(id, options) { + var defaults = { + gfm: true, + toc: true, + tocm: false, + tocStartLevel: 1, + tocTitle: "目录", + tocDropdown: false, + tocContainer: "", + markdown: "", + markdownSourceCode: false, + htmlDecode: false, + autoLoadKaTeX: true, + pageBreak: true, + atLink: true, // for @link + emailLink: true, // for mail address auto link + tex: false, + taskList: false, // Github Flavored Markdown task lists + emoji: false, + flowChart: false, + sequenceDiagram: false, + previewCodeHighlight: true + } + + editormd.$marked = marked + + var div = $("#" + id) + var settings = (div.settings = $.extend(true, defaults, options || {})) + var saveTo = div.find("textarea") + if (saveTo.length < 1) { + div.append("") + saveTo = div.find("textarea") + } + var markdownDoc = + settings.markdown === "" ? saveTo.val() : settings.markdown + var markdownToC = [] + var rendererOptions = { + toc: settings.toc, + tocm: settings.tocm, + tocStartLevel: settings.tocStartLevel, + taskList: settings.taskList, + emoji: settings.emoji, + tex: settings.tex, + pageBreak: settings.pageBreak, + atLink: settings.atLink, // for @link + emailLink: settings.emailLink, // for mail address auto link + flowChart: settings.flowChart, + sequenceDiagram: settings.sequenceDiagram, + previewCodeHighlight: settings.previewCodeHighlight + } + var markedOptions = { + renderer: editormd.markedRenderer(markdownToC, rendererOptions), + gfm: settings.gfm, + tables: true, + breaks: true, + pedantic: false, + sanitize: settings.htmlDecode ? false : true, // 是否忽略HTML标签,即是否开启HTML标签解析,为了安全性,默认不开启 + smartLists: true, + smartypants: true + } + markdownDoc = new String(markdownDoc) + var markdownParsed = marked(markdownDoc, markedOptions) + markdownParsed = editormd.filterHTMLTags( + markdownParsed, + settings.htmlDecode + ) + if (settings.markdownSourceCode) { + saveTo.text(markdownDoc) + } else { + saveTo.remove() + } + div + .addClass("markdown-body " + this.classPrefix + "html-preview") + .append(markdownParsed) + var tocContainer = + settings.tocContainer !== "" ? $(settings.tocContainer) : div + if (settings.tocContainer !== "") { + tocContainer.attr("previewContainer", false) + } + if (settings.toc) { + div.tocContainer = this.markdownToCRenderer( + markdownToC, + tocContainer, + settings.tocDropdown, + settings.tocStartLevel + ) + if ( + settings.tocDropdown || + div.find("." + this.classPrefix + "toc-menu").length > 0 + ) { + this.tocDropdownMenu(div, settings.tocTitle) + } + if (settings.tocContainer !== "") { + div.find(".editormd-toc-menu, .editormd-markdown-toc").remove() + } + } + if (settings.previewCodeHighlight) { + div.find("pre").addClass("prettyprint linenums") + prettyPrint() + } + if (!editormd.isIE8) { + if (settings.flowChart) { + div.find(".flowchart").flowChart() + } + if (settings.sequenceDiagram) { + div.find(".sequence-diagram").sequenceDiagram({ theme: "simple" }) + } + } + if (settings.tex) { + var katexHandle = function() { + div.find("." + editormd.classNames.tex).each(function() { + var tex = $(this) + katex.render( + tex + .html() + .replace(/</g, "<") + .replace(/>/g, ">"), + tex[0] + ) + tex.find(".katex").css("font-size", "1.6em") + }) + } + if (settings.autoLoadKaTeX && !editormd.$katex && !editormd.kaTeXLoaded) { + this.loadKaTeX(function() { + editormd.$katex = katex + editormd.kaTeXLoaded = true + katexHandle() + }) + } else { + katexHandle() + } + } + div.getMarkdown = function() { + return saveTo.val() + } + return div + } + + // Editor.md themes, change toolbar themes etc. + // added @1.5.0 + editormd.themes = ["default", "dark"] + // Preview area themes + // added @1.5.0 + editormd.previewThemes = ["default", "dark"] + // CodeMirror / editor area themes + // @1.5.0 rename -> editorThemes, old version -> themes + editormd.editorThemes = [ + "default", + "3024-day", + "3024-night", + "ambiance", + "ambiance-mobile", + "base16-dark", + "base16-light", + "blackboard", + "cobalt", + "eclipse", + "elegant", + "erlang-dark", + "lesser-dark", + "mbo", + "mdn-like", + "midnight", + "monokai", + "neat", + "neo", + "night", + "paraiso-dark", + "paraiso-light", + "pastel-on-dark", + "rubyblue", + "solarized", + "the-matrix", + "tomorrow-night-eighties", + "twilight", + "vibrant-ink", + "xq-dark", + "xq-light" + ] + editormd.loadPlugins = {} + editormd.loadFiles = { + js: [], + css: [], + plugin: [] + } + /** + * 动态加载Editor.md插件,但不立即执行 + * Load editor.md plugins + * + * @param {String} fileName 插件文件路径 + * @param {Function} [callback=function()] 加载成功后执行的回调函数 + * @param {String} [into="head"] 嵌入页面的位置 + */ + + editormd.loadPlugin = function(fileName, callback, into) { + callback = callback || function() {} + this.loadScript( + fileName, + function() { + editormd.loadFiles.plugin.push(fileName) + callback() + }, + into + ) + } + + /** + * 动态加载CSS文件的方法 + * Load css file method + * + * @param {String} fileName CSS文件名 + * @param {Function} [callback=function()] 加载成功后执行的回调函数 + * @param {String} [into="head"] 嵌入页面的位置 + */ + + editormd.loadCSS = function(fileName, callback, into) { + into = into || "head" + callback = callback || function() {} + + var css = document.createElement("link") + css.type = "text/css" + css.rel = "stylesheet" + css.onload = css.onreadystatechange = function() { + editormd.loadFiles.css.push(fileName) + callback() + } + css.href = fileName + ".css" + if (into === "head") { + document.getElementsByTagName("head")[0].appendChild(css) + } else { + document.body.appendChild(css) + } + } + + editormd.isIE = navigator.appName === "Microsoft Internet Explorer" + editormd.isIE8 = editormd.isIE && navigator.appVersion.match(/8./i) === "8." + + /** + * 动态加载JS文件的方法 + * Load javascript file method + * + * @param {String} fileName JS文件名 + * @param {Function} [callback=function()] 加载成功后执行的回调函数 + * @param {String} [into="head"] 嵌入页面的位置 + */ + + editormd.loadScript = function(fileName, callback, into) { + into = into || "head" + callback = callback || function() {} + var script = null + script = document.createElement("script") + script.id = fileName.replace(/[\./]+/g, "-") + script.type = "text/javascript" + script.src = fileName + ".js" + if (editormd.isIE8) { + script.onreadystatechange = function() { + if (script.readyState) { + if ( + script.readyState === "loaded" || + script.readyState === "complete" + ) { + script.onreadystatechange = null + editormd.loadFiles.js.push(fileName) + callback() + } + } + } + } else { + script.onload = function() { + editormd.loadFiles.js.push(fileName) + callback() + } + } + if (into === "head") { + document.getElementsByTagName("head")[0].appendChild(script) + } else { + document.body.appendChild(script) + } + } + + // 使用国外的CDN,加载速度有时会很慢,或者自定义URL + // You can custom KaTeX load url. + editormd.katexURL = { + css: "//cdnjs.cloudflare.com/ajax/libs/KaTeX/0.3.0/katex.min", + js: "//cdnjs.cloudflare.com/ajax/libs/KaTeX/0.3.0/katex.min" + } + editormd.kaTeXLoaded = false + /** + * 加载KaTeX文件 + * load KaTeX files + * + * @param {Function} [callback=function()] 加载成功后执行的回调函数 + */ + editormd.loadKaTeX = function(callback) { + editormd.loadCSS(editormd.katexURL.css, function() { + editormd.loadScript(editormd.katexURL.js, callback || function() {}) + }) + } + + /** + * 锁屏 + * lock screen + * + * @param {Boolean} lock Boolean 布尔值,是否锁屏 + * @returns {void} + */ + + editormd.lockScreen = function(lock) { + $("html,body").css("overflow", lock ? "hidden" : "") + } + + /** + * 动态创建对话框 + * Creating custom dialogs + * + * @param {Object} options 配置项键值对 Key/Value + * @returns {dialog} 返回创建的dialog的jQuery实例对象 + */ + + editormd.createDialog = function(options) { + var defaults = { + name: "", + width: 420, + height: 240, + title: "", + drag: true, + closed: true, + content: "", + mask: true, + maskStyle: { + backgroundColor: "#fff", + opacity: 0.1 + }, + lockScreen: true, + footer: true, + buttons: false + } + + options = $.extend(true, defaults, options) + + var $this = this + var editor = this.editor + var classPrefix = editormd.classPrefix + var guid = new Date().getTime() + var dialogName = + options.name === "" ? classPrefix + "dialog-" + guid : options.name + var mouseOrTouch = editormd.mouseOrTouch + var html = '
        ' + if (options.title !== "") { + html += + '
        " + html += + '' + + options.title + + "" + html += "
        " + } + if (options.closed) { + html += + '
        ' + } + html += + '
        ' + options.content + if (options.footer || typeof options.footer === "string") { + html += + '" + } + html += "
        " + html += + '
        ' + html += + '
        ' + html += "
        " + editor.append(html) + var dialog = editor.find("." + dialogName) + dialog.lockScreen = function(lock) { + if (options.lockScreen) { + $("html,body").css("overflow", lock ? "hidden" : "") + $this.resize() + } + return dialog + } + dialog.showMask = function() { + if (options.mask) { + editor + .find("." + classPrefix + "mask") + .css(options.maskStyle) + .css("z-index", editormd.dialogZindex - 1) + .show() + } + return dialog + } + dialog.hideMask = function() { + if (options.mask) { + editor.find("." + classPrefix + "mask").hide() + } + return dialog + } + dialog.loading = function(show) { + var loading = dialog.find("." + classPrefix + "dialog-mask") + loading[show ? "show" : "hide"]() + return dialog + } + dialog.lockScreen(true).showMask() + dialog.show().css({ + zIndex: editormd.dialogZindex, + border: editormd.isIE8 ? "1px solid #ddd" : "", + width: + typeof options.width === "number" + ? options.width + "px" + : options.width, + height: + typeof options.height === "number" + ? options.height + "px" + : options.height + }) + var dialogPosition = function() { + dialog.css({ + top: ($(window).height() - dialog.height()) / 2 + "px", + left: ($(window).width() - dialog.width()) / 2 + "px" + }) + } + dialogPosition() + $(window).resize(dialogPosition) + dialog + .children("." + classPrefix + "dialog-close") + .bind(mouseOrTouch("click", "touchend"), function() { + dialog + .hide() + .lockScreen(false) + .hideMask() + }) + if (typeof options.buttons === "object") { + var footer = (dialog.footer = dialog.find( + "." + classPrefix + "dialog-footer" + )) + for (var key in options.buttons) { + var btn = options.buttons[key] + var btnClassName = classPrefix + key + "-btn" + footer.append( + '" + ) + btn[1] = $.proxy(btn[1], dialog) + footer + .children("." + btnClassName) + .bind(mouseOrTouch("click", "touchend"), btn[1]) + } + } + if (options.title !== "" && options.drag) { + var posX, posY + var dialogHeader = dialog.children("." + classPrefix + "dialog-header") + if (!options.mask) { + dialogHeader.bind(mouseOrTouch("click", "touchend"), function() { + editormd.dialogZindex += 2 + dialog.css("z-index", editormd.dialogZindex) + }) + } + var userCanSelect = function(obj) { + obj.removeClass(classPrefix + "user-unselect").off("selectstart") + } + var userUnselect = function(obj) { + obj + .addClass(classPrefix + "user-unselect") + .on("selectstart", function(event) { + // selectstart for IE + return false + }) + } + var moveAction = function(e) { + e = e || window.event //IE + var left, + top, + nowLeft = parseInt(dialog[0].style.left), + nowTop = parseInt(dialog[0].style.top) + if (nowLeft >= 0) { + if (nowLeft + dialog.width() <= $(window).width()) { + left = e.clientX - posX + } else { + left = $(window).width() - dialog.width() + document.onmousemove = null + } + } else { + left = 0 + document.onmousemove = null + } + if (nowTop >= 0) { + top = e.clientY - posY + } else { + top = 0 + document.onmousemove = null + } + document.onselectstart = function() { + return false + } + userUnselect($("body")) + userUnselect(dialog) + dialog[0].style.left = left + "px" + dialog[0].style.top = top + "px" + } + dialogHeader.mousedown(function(e) { + e = e || window.event //IE + posX = e.clientX - parseInt(dialog[0].style.left) + posY = e.clientY - parseInt(dialog[0].style.top) + document.onmousemove = moveAction + }) + document.onmouseup = function() { + userCanSelect($("body")) + userCanSelect(dialog) + document.onselectstart = null + document.onmousemove = null + } + dialogHeader.touchDraggable = function() { + var offset = null + var start = function(e) { + var orig = e.originalEvent + var pos = $(this) + .parent() + .position() + offset = { + x: orig.changedTouches[0].pageX - pos.left, + y: orig.changedTouches[0].pageY - pos.top + } + } + var move = function(e) { + e.preventDefault() + var orig = e.originalEvent + $(this) + .parent() + .css({ + top: orig.changedTouches[0].pageY - offset.y, + left: orig.changedTouches[0].pageX - offset.x + }) + } + this.bind("touchstart", start).bind("touchmove", move) + } + dialogHeader.touchDraggable() + } + editormd.dialogZindex += 2 + return dialog + } + + /** + * 鼠标和触摸事件的判断/选择方法 + * MouseEvent or TouchEvent type switch + * + * @param {String} [mouseEventType="click"] 供选择的鼠标事件 + * @param {String} [touchEventType="touchend"] 供选择的触摸事件 + * @returns {String} EventType 返回事件类型名称 + */ + + editormd.mouseOrTouch = function(mouseEventType, touchEventType) { + mouseEventType = mouseEventType || "click" + touchEventType = touchEventType || "touchend" + var eventType = mouseEventType + try { + document.createEvent("TouchEvent") + eventType = touchEventType + } catch (e) {} + return eventType + } + + /** + * 日期时间的格式化方法 + * Datetime format method + * + * @param {String} [format=""] 日期时间的格式,类似PHP的格式 + * @returns {String} datefmt 返回格式化后的日期时间字符串 + */ + + editormd.dateFormat = function(format) { + format = format || "" + var addZero = function(d) { + return d < 10 ? "0" + d : d + } + var date = new Date() + var year = date.getFullYear() + var year2 = year.toString().slice(2, 4) + var month = addZero(date.getMonth() + 1) + var day = addZero(date.getDate()) + var weekDay = date.getDay() + var hour = addZero(date.getHours()) + var min = addZero(date.getMinutes()) + var second = addZero(date.getSeconds()) + var ms = addZero(date.getMilliseconds()) + var datefmt = "" + var ymd = year2 + "-" + month + "-" + day + var fymd = year + "-" + month + "-" + day + var hms = hour + ":" + min + ":" + second + switch (format) { + case "UNIX Time": + datefmt = date.getTime() + break + case "UTC": + datefmt = date.toUTCString() + break + case "yy": + datefmt = year2 + break + case "year": + case "yyyy": + datefmt = year + break + case "month": + case "mm": + datefmt = month + break + case "cn-week-day": + case "cn-wd": + var cnWeekDays = ["日", "一", "二", "三", "四", "五", "六"] + datefmt = "星期" + cnWeekDays[weekDay] + break + case "week-day": + case "wd": + var weekDays = [ + "Sunday", + "Monday", + "Tuesday", + "Wednesday", + "Thursday", + "Friday", + "Saturday" + ] + datefmt = weekDays[weekDay] + break + case "day": + case "dd": + datefmt = day + break + case "hour": + case "hh": + datefmt = hour + break + case "min": + case "ii": + datefmt = min + break + case "second": + case "ss": + datefmt = second + break + case "ms": + datefmt = ms + break + case "yy-mm-dd": + datefmt = ymd + break + case "yyyy-mm-dd": + datefmt = fymd + break + case "yyyy-mm-dd h:i:s ms": + case "full + ms": + datefmt = fymd + " " + hms + " " + ms + break + case "full": + case "yyyy-mm-dd h:i:s": + break + default: + datefmt = fymd + " " + hms + break + } + return datefmt + } + return editormd +}) diff --git a/system/admin/editor.md/editormd.amd.min.js b/system/admin/editor.md/editormd.amd.min.js new file mode 100644 index 00000000..6e7c32c6 --- /dev/null +++ b/system/admin/editor.md/editormd.amd.min.js @@ -0,0 +1,2 @@ +/*! Editor.md v2.0.1 | editormd.amd.min.js | Open source online markdown editor. | MIT License | By: Pandao | https://github.com/pandao/editor.md | 2018-03-29 */ +"use strict";function _defineProperty(e,t,i){return t in e?Object.defineProperty(e,t,{value:i,enumerable:!0,configurable:!0,writable:!0}):e[t]=i,e}function _typeof(e){return(_typeof="function"==typeof Symbol&&"symbol"==typeof Symbol.iterator?function(e){return typeof e}:function(e){return e&&"function"==typeof Symbol&&e.constructor===Symbol&&e!==Symbol.prototype?"symbol":typeof e})(e)}!function(e){if("function"==typeof require&&"object"===("undefined"==typeof exports?"undefined":_typeof(exports))&&"object"===("undefined"==typeof module?"undefined":_typeof(module)))module.exports=e;else if("function"==typeof define)if(define.amd){var t="codemirror/mode/",i="codemirror/addon/",o=["jquery","marked","prettify","katex","raphael","underscore","flowchart","jqueryflowchart","sequenceDiagram","codemirror/lib/codemirror",t+"css/css",t+"sass/sass",t+"shell/shell",t+"sql/sql",t+"clike/clike",t+"php/php",t+"xml/xml",t+"markdown/markdown",t+"javascript/javascript",t+"htmlmixed/htmlmixed",t+"gfm/gfm",t+"http/http",t+"go/go",t+"dart/dart",t+"coffeescript/coffeescript",t+"nginx/nginx",t+"python/python",t+"perl/perl",t+"lua/lua",t+"r/r",t+"ruby/ruby",t+"rst/rst",t+"smartymixed/smartymixed",t+"vb/vb",t+"vbscript/vbscript",t+"velocity/velocity",t+"xquery/xquery",t+"yaml/yaml",t+"erlang/erlang",t+"jade/jade",i+"edit/trailingspace",i+"dialog/dialog",i+"search/searchcursor",i+"search/search",i+"scroll/annotatescrollbar",i+"search/matchesonscrollbar",i+"display/placeholder",i+"edit/closetag",i+"fold/foldcode",i+"fold/foldgutter",i+"fold/indent-fold",i+"fold/brace-fold",i+"fold/xml-fold",i+"fold/markdown-fold",i+"fold/comment-fold",i+"mode/overlay",i+"selection/active-line",i+"edit/closebrackets",i+"display/fullscreen",i+"search/match-highlighter"];define(o,e)}else define(["jquery"],e);else window.editormd=e()}(function(){var e;"function"==typeof define&&define.amd&&(t=arguments[0],marked=arguments[1],prettify=arguments[2],katex=arguments[3],Raphael=arguments[4],_=arguments[5],flowchart=arguments[6],CodeMirror=arguments[9]);var t="undefined"!=typeof jQuery?jQuery:Zepto;if(void 0!==t){var i=function e(t,i){return new e.fn.init(t,i)};i.title=i.$name="Editor.md",i.version="1.5.0",i.homePage="https://pandao.github.io/editor.md/",i.classPrefix="editormd-",i.toolbarModes={full:["undo","redo","|","bold","del","italic","quote","ucwords","uppercase","lowercase","|","h1","h2","h3","h4","h5","h6","|","list-ul","list-ol","hr","|","link","reference-link","image","code","preformatted-text","code-block","table","datetime","emoji","html-entities","pagebreak","|","goto-line","watch","preview","fullscreen","clear","search","|","help","info"],simple:["undo","redo","|","bold","del","italic","quote","uppercase","lowercase","|","h1","h2","h3","h4","h5","h6","|","list-ul","list-ol","hr","|","watch","preview","fullscreen","|","help","info"],mini:["undo","redo","|","watch","preview","|","help","info"]},i.defaults={mode:"gfm",name:"",value:"",theme:"",editorTheme:"default",previewTheme:"",markdown:"",appendMarkdown:"",width:"100%",height:"100%",path:"./lib/",pluginPath:"",delay:300,autoLoadModules:!0,watch:!0,placeholder:"Enjoy Markdown! coding now...",gotoLine:!0,codeFold:!1,autoHeight:!1,minHeight:"150px",maxHeight:"none",autoFocus:!0,autoCloseTags:!0,searchReplace:!0,syncScrolling:!0,readOnly:!1,tabSize:4,indentUnit:4,lineNumbers:!0,lineWrapping:!0,autoCloseBrackets:!0,showTrailingSpace:!0,matchBrackets:!0,indentWithTabs:!0,styleSelectedText:!0,matchWordHighlight:!0,styleActiveLine:!0,dialogLockScreen:!0,dialogShowMask:!0,dialogDraggable:!0,dialogMaskBgColor:"#fff",dialogMaskOpacity:.1,fontSize:"13px",saveHTMLToTextarea:!1,disabledKeyMaps:[],onload:function(){},onresize:function(){},onchange:function(){},onwatch:null,onunwatch:null,onpreviewing:function(){},onpreviewed:function(){},onfullscreen:function(){},onfullscreenExit:function(){},onscroll:function(){},onpreviewscroll:function(){},imageUpload:!1,imageFormats:["jpg","jpeg","gif","png","bmp","webp"],imageUploadURL:"",crossDomainUpload:!1,uploadCallbackURL:"",toc:!0,tocm:!1,tocTitle:"",tocDropdown:!1,tocContainer:"",tocStartLevel:1,htmlDecode:!1,pageBreak:!0,atLink:!0,emailLink:!0,taskList:!1,emoji:!1,tex:!1,flowChart:!1,sequenceDiagram:!1,previewCodeHighlight:!0,toolbar:!0,toolbarAutoFixed:!0,toolbarIcons:"full",toolbarTitles:{},toolbarHandlers:{ucwords:function(){return i.toolbarHandlers.ucwords},lowercase:function(){return i.toolbarHandlers.lowercase}},toolbarCustomIcons:{lowercase:'a',ucwords:'Aa'},toolbarIconsClass:{undo:"fa-undo",redo:"fa-repeat",bold:"fa-bold",del:"fa-strikethrough",italic:"fa-italic",quote:"fa-quote-left",uppercase:"fa-font",h1:i.classPrefix+"bold",h2:i.classPrefix+"bold",h3:i.classPrefix+"bold",h4:i.classPrefix+"bold",h5:i.classPrefix+"bold",h6:i.classPrefix+"bold","list-ul":"fa-list-ul","list-ol":"fa-list-ol",hr:"fa-minus",link:"fa-link","reference-link":"fa-anchor",image:"fa-picture-o",code:"fa-code","preformatted-text":"fa-file-code-o","code-block":"fa-file-code-o",table:"fa-table",datetime:"fa-clock-o",emoji:"fa-smile-o","html-entities":"fa-copyright",pagebreak:"fa-newspaper-o","goto-line":"fa-terminal",watch:"fa-eye-slash",unwatch:"fa-eye",preview:"fa-desktop",search:"fa-search",fullscreen:"fa-arrows-alt",clear:"fa-eraser",help:"fa-question-circle",info:"fa-info-circle"},toolbarIconTexts:{},lang:{name:"zh-cn",description:"开源在线Markdown编辑器
        Open source online Markdown editor.",tocTitle:"目录",toolbar:{undo:"撤销(Ctrl+Z)",redo:"重做(Ctrl+Y)",bold:"粗体",del:"删除线",italic:"斜体",quote:"引用",ucwords:"将每个单词首字母转成大写",uppercase:"将所选转换成大写",lowercase:"将所选转换成小写",h1:"标题1",h2:"标题2",h3:"标题3",h4:"标题4",h5:"标题5",h6:"标题6","list-ul":"无序列表","list-ol":"有序列表",hr:"横线",link:"链接","reference-link":"引用链接",image:"添加图片",code:"行内代码","preformatted-text":"预格式文本 / 代码块(缩进风格)","code-block":"代码块(多语言风格)",table:"添加表格",datetime:"日期时间",emoji:"Emoji表情","html-entities":"HTML实体字符",pagebreak:"插入分页符","goto-line":"跳转到行",watch:"关闭实时预览",unwatch:"开启实时预览",preview:"全窗口预览HTML(按 Shift + ESC还原)",fullscreen:"全屏(按ESC还原)",clear:"清空",search:"搜索",help:"使用帮助",info:"关于"+i.title},buttons:{enter:"确定",cancel:"取消",close:"关闭"},dialog:{link:{title:"添加链接",url:"链接地址",urlTitle:"链接标题",urlEmpty:"错误:请填写链接地址。"},referenceLink:{title:"添加引用链接",name:"引用名称",url:"链接地址",urlId:"链接ID",urlTitle:"链接标题",nameEmpty:"错误:引用链接的名称不能为空。",idEmpty:"错误:请填写引用链接的ID。",urlEmpty:"错误:请填写引用链接的URL地址。"},image:{title:"添加图片",url:"图片地址",link:"图片链接",alt:"图片描述",uploadButton:"本地上传",imageURLEmpty:"错误:图片地址不能为空。",uploadFileEmpty:"错误:上传的图片不能为空。",formatNotAllowed:"错误:只允许上传图片文件,允许上传的图片文件格式有:"},preformattedText:{title:"添加预格式文本或代码块",emptyAlert:"错误:请填写预格式文本或代码的内容。"},codeBlock:{title:"添加代码块",selectLabel:"代码语言:",selectDefaultText:"请选择代码语言",otherLanguage:"其他语言",unselectedLanguageAlert:"错误:请选择代码所属的语言类型。",codeEmptyAlert:"错误:请填写代码内容。"},htmlEntities:{title:"HTML 实体字符"},help:{title:"使用帮助"}}}},i.classNames={tex:i.classPrefix+"tex"},i.dialogZindex=99999,i.$katex=null,i.$marked=null,i.$CodeMirror=null,i.$prettyPrint=null;var o,r;i.prototype=i.fn={state:{watching:!1,loaded:!1,preview:!1,fullscreen:!1},init:function(e,o){o=o||{},"object"===_typeof(e)&&(o=e);var r=this.classPrefix=i.classPrefix,n=this.settings=t.extend(!0,i.defaults,o);e="object"===_typeof(e)?n.id:e;var a=this.editor=t("#"+e);this.id=e,this.lang=n.lang;var s=this.classNames={textarea:{html:r+"html-textarea",markdown:r+"markdown-textarea"}};n.pluginPath=""===n.pluginPath?n.path+"../plugins/":n.pluginPath,this.state.watching=!!n.watch,a.hasClass("editormd")||a.addClass("editormd"),a.css({width:"number"==typeof n.width?n.width+"px":n.width,height:"number"==typeof n.height?n.height+"px":n.height}),n.autoHeight&&a.css({height:"auto",minHeight:n.minHeight});var l=this.markdownTextarea=a.children("textarea");l.length<1&&(a.append(""),l=this.markdownTextarea=a.children("textarea")),l.addClass(s.textarea.markdown).attr("placeholder",n.placeholder),void 0!==l.attr("name")&&""!==l.attr("name")||l.attr("name",""!==n.name?n.name:e+"-markdown-doc");var c=[n.readOnly?"":'',n.saveHTMLToTextarea?'':"",'
        ','
        ','
        '].join("\n");return a.append(c).addClass(r+"vertical"),""!==n.theme&&a.addClass(r+"theme-"+n.theme),this.mask=a.children("."+r+"mask"),this.containerMask=a.children("."+r+"container-mask"),""!==n.markdown&&l.val(n.markdown),""!==n.appendMarkdown&&l.val(l.val()+n.appendMarkdown),this.htmlTextarea=a.children("."+s.textarea.html),this.preview=a.children("."+r+"preview"),this.previewContainer=this.preview.children("."+r+"preview-container"),""!==n.previewTheme&&this.preview.addClass(r+"preview-theme-"+n.previewTheme),"function"==typeof define&&define.amd&&("undefined"!=typeof katex&&(i.$katex=katex),n.searchReplace&&!n.readOnly&&(i.loadCSS(n.path+"codemirror/addon/dialog/dialog"),i.loadCSS(n.path+"codemirror/addon/search/matchesonscrollbar"))),"function"==typeof define&&define.amd||!n.autoLoadModules?("undefined"!=typeof CodeMirror&&(i.$CodeMirror=CodeMirror),"undefined"!=typeof marked&&(i.$marked=marked),this.setCodeMirror().setToolbar().loadedDisplay()):this.loadQueues(),this},loadQueues:function(){var e=this,t=this.settings,o=t.path,r=function(){if(i.isIE8)return void e.loadedDisplay();t.flowChart||t.sequenceDiagram?i.loadScript(o+"raphael.min",function(){i.loadScript(o+"underscore.min",function(){!t.flowChart&&t.sequenceDiagram?i.loadScript(o+"sequence-diagram.min",function(){e.loadedDisplay()}):t.flowChart&&!t.sequenceDiagram?i.loadScript(o+"flowchart.min",function(){i.loadScript(o+"jquery.flowchart.min",function(){e.loadedDisplay()})}):t.flowChart&&t.sequenceDiagram&&i.loadScript(o+"flowchart.min",function(){i.loadScript(o+"jquery.flowchart.min",function(){i.loadScript(o+"sequence-diagram.min",function(){e.loadedDisplay()})})})})}):e.loadedDisplay()};return i.loadCSS(o+"codemirror/codemirror.min"),t.searchReplace&&!t.readOnly&&(i.loadCSS(o+"codemirror/addon/dialog/dialog"),i.loadCSS(o+"codemirror/addon/search/matchesonscrollbar")),t.codeFold&&i.loadCSS(o+"codemirror/addon/fold/foldgutter"),i.loadScript(o+"codemirror/codemirror.min",function(){i.$CodeMirror=CodeMirror,i.loadScript(o+"codemirror/modes.min",function(){i.loadScript(o+"codemirror/addons.min",function(){if(e.setCodeMirror(),"gfm"!==t.mode&&"markdown"!==t.mode)return e.loadedDisplay(),!1;e.setToolbar(),i.loadScript(o+"marked.min",function(){i.$marked=marked,t.previewCodeHighlight?i.loadScript(o+"prettify.min",function(){r()}):r()})})})}),this},setTheme:function(e){var t=this.editor,i=this.settings.theme,o=this.classPrefix+"theme-";return t.removeClass(o+i).addClass(o+e),this.settings.theme=e,this},setEditorTheme:function(e){var t=this.settings;return t.editorTheme=e,"default"!==e&&i.loadCSS(t.path+"codemirror/theme/"+t.editorTheme),this.cm.setOption("theme",e),this},setCodeMirrorTheme:function(e){return this.setEditorTheme(e),this},setPreviewTheme:function(e){var t=this.preview,i=this.settings.previewTheme,o=this.classPrefix+"preview-theme-";return t.removeClass(o+i).addClass(o+e),this.settings.previewTheme=e,this},setCodeMirror:function(){var e=this.settings,t=this.editor;"default"!==e.editorTheme&&i.loadCSS(e.path+"codemirror/theme/"+e.editorTheme);var o={mode:e.mode,theme:e.editorTheme,tabSize:e.tabSize,dragDrop:!1,autofocus:e.autoFocus,autoCloseTags:e.autoCloseTags,readOnly:!!e.readOnly&&"nocursor",indentUnit:e.indentUnit,lineNumbers:e.lineNumbers,lineWrapping:e.lineWrapping,extraKeys:{"Ctrl-Q":function(e){e.foldCode(e.getCursor())}},foldGutter:e.codeFold,gutters:["CodeMirror-linenumbers","CodeMirror-foldgutter"],matchBrackets:e.matchBrackets,indentWithTabs:e.indentWithTabs,styleActiveLine:e.styleActiveLine,styleSelectedText:e.styleSelectedText,autoCloseBrackets:e.autoCloseBrackets,showTrailingSpace:e.showTrailingSpace,highlightSelectionMatches:!!e.matchWordHighlight&&{showToken:"onselected"!==e.matchWordHighlight&&/\w/}};return this.codeEditor=this.cm=i.$CodeMirror.fromTextArea(this.markdownTextarea[0],o),this.codeMirror=this.cmElement=t.children(".CodeMirror"),""!==e.value&&this.cm.setValue(e.value),this.codeMirror.css({fontSize:e.fontSize,width:e.watch?"50%":"100%"}),e.autoHeight&&(this.codeMirror.css({height:"auto",minHeight:e.minHeight,maxHeight:e.maxHeight}),this.codeMirror.find(".CodeMirror-gutters").css({minHeight:e.minHeight}),this.codeMirror.find(".CodeMirror-scroll").css({minHeight:e.minHeight,maxHeight:e.maxHeight}),this.cm.setOption("viewportMargin",1/0)),e.lineNumbers||this.codeMirror.find(".CodeMirror-gutters").css("border-right","none"),this},getCodeMirrorOption:function(e){return this.cm.getOption(e)},setCodeMirrorOption:function(e,t){return this.cm.setOption(e,t),this},addKeyMap:function(e,t){return this.cm.addKeyMap(e,t),this},removeKeyMap:function(e){return this.cm.removeKeyMap(e),this},gotoLine:function(e){var i=this.settings;if(!i.gotoLine)return this;var o=this.cm,r=(this.editor,o.lineCount()),n=this.preview;if("string"==typeof e&&("last"===e&&(e=r),"first"===e&&(e=1)),"number"!=typeof e)return alert("Error: The line number must be an integer."),this;if((e=parseInt(e)-1)>r)return alert("Error: The line number range 1-"+r),this;o.setCursor({line:e,ch:0});var a=o.getScrollInfo(),s=a.clientHeight,l=o.charCoords({line:e,ch:0},"local");if(o.scrollTo(null,(l.top+l.bottom-s)/2),i.watch){var c=this.codeMirror.find(".CodeMirror-scroll")[0],h=t(c).height(),d=c.scrollTop,u=d/c.scrollHeight;0===d?n.scrollTop(0):d+h>=c.scrollHeight-16?n.scrollTop(n[0].scrollHeight):n.scrollTop(n[0].scrollHeight*u)}return o.focus(),this},extend:function(){return void 0!==arguments[1]&&("function"==typeof arguments[1]&&(arguments[1]=t.proxy(arguments[1],this)),this[arguments[0]]=arguments[1]),"object"===_typeof(arguments[0])&&void 0===arguments[0].length&&t.extend(!0,this,arguments[0]),this},set:function(e,i){return void 0!==i&&"function"==typeof i&&(i=t.proxy(i,this)),this[e]=i,this},config:function(e,i){var o=this.settings;return"object"===_typeof(e)&&(o=t.extend(!0,o,e)),"string"==typeof e&&(o[e]=i),this.settings=o,this.recreate(),this},on:function(e,i){var o=this.settings;return void 0!==o["on"+e]&&(o["on"+e]=t.proxy(i,this)),this},off:function(e){var t=this.settings;return void 0!==t["on"+e]&&(t["on"+e]=function(){}),this},showToolbar:function(e){var i=this.settings;return i.readOnly?this:(i.toolbar&&(this.toolbar.length<1||""===this.toolbar.find("."+this.classPrefix+"menu").html())&&this.setToolbar(),i.toolbar=!0,this.toolbar.show(),this.resize(),t.proxy(e||function(){},this)(),this)},hideToolbar:function(e){return this.settings.toolbar=!1,this.toolbar.hide(),this.resize(),t.proxy(e||function(){},this)(),this},setToolbarAutoFixed:function(e){var i=this.state,o=this.editor,r=this.toolbar,n=this.settings;void 0!==e&&(n.toolbarAutoFixed=e);var a=function(){var e=t(window),i=e.scrollTop();if(!n.toolbarAutoFixed)return!1;i-o.offset().top>10&&i
          ';t.append(n),r=this.toolbar=t.children("."+o+"toolbar")}if(!e.toolbar)return r.hide(),this;r.show();for(var a="function"==typeof e.toolbarIcons?e.toolbarIcons():"string"==typeof e.toolbarIcons?i.toolbarModes[e.toolbarIcons]:e.toolbarIcons,s=r.find("."+this.classPrefix+"menu"),l="",c=!1,h=0,d=a.length;h|';else{var f=/h(\d)/.test(u),p=u;"watch"!==u||e.watch||(p="unwatch");var g=e.lang.toolbar[p],m=e.toolbarIconTexts[p],w=e.toolbarIconsClass[p];g=void 0===g?"":g,m=void 0===m?"":m,w=void 0===w?"":w;var v=c?'
        • ':"
        • ";void 0!==e.toolbarCustomIcons[u]&&"function"!=typeof e.toolbarCustomIcons[u]?v+=e.toolbarCustomIcons[u]:(v+='',v+=''+(f?u.toUpperCase():""===w?m:"")+"",v+=""),v+="
        • ",l=c?v+l:l+v}}return s.html(l),s.find('[title="Lowercase"]').attr("title",e.lang.toolbar.lowercase),s.find('[title="ucwords"]').attr("title",e.lang.toolbar.ucwords),this.setToolbarHandler(),this.setToolbarAutoFixed(),this},dialogLockScreen:function(){return t.proxy(i.dialogLockScreen,this)(),this},dialogShowMask:function(e){return t.proxy(i.dialogShowMask,this)(e),this},getToolbarHandles:function(e){var t=this.toolbarHandlers=i.toolbarHandlers;return e&&void 0!==toolbarIconHandlers[e]?t[e]:t},setToolbarHandler:function(){var e=this,o=this.settings;if(!o.toolbar||o.readOnly)return this;var r=this.toolbar,n=this.cm,a=this.classPrefix,s=this.toolbarIcons=r.find("."+a+"menu > li > a"),l=this.getToolbarHandles();return s.bind(i.mouseOrTouch("click","touchend"),function(i){var r=t(this).children(".fa"),a=r.attr("name"),s=n.getCursor(),c=n.getSelection();if(""!==a)return e.activeIcon=r,void 0!==l[a]?t.proxy(l[a],e)(n):void 0!==o.toolbarHandlers[a]&&t.proxy(o.toolbarHandlers[a],e)(n,r,s,c),"link"!==a&&"reference-link"!==a&&"image"!==a&&"code-block"!==a&&"preformatted-text"!==a&&"watch"!==a&&"preview"!==a&&"search"!==a&&"fullscreen"!==a&&"info"!==a&&n.focus(),!1}),this},createDialog:function(e){return t.proxy(i.createDialog,this)(e)},createInfoDialog:function(){var e=this,t=this.editor,o=this.classPrefix,r=['
          ','
          ','

          '+i.title+"v"+i.version+"

          ","

          "+this.lang.description+"

          ",'

          '+i.homePage+'

          ','

          Copyright © 2015 Pandao, The MIT License.

          ',"
          ",'',"
          "].join("\n");t.append(r);var n=this.infoDialog=t.children("."+o+"dialog-info");return n.find("."+o+"dialog-close").bind(i.mouseOrTouch("click","touchend"),function(){e.hideInfoDialog()}),n.css("border",i.isIE8?"1px solid #ddd":"").css("z-index",i.dialogZindex).show(),this.infoDialogPosition(),this},infoDialogPosition:function(){var e=this.infoDialog,i=function(){e.css({top:(t(window).height()-e.height())/2+"px",left:(t(window).width()-e.width())/2+"px"})};return i(),t(window).resize(i),this},showInfoDialog:function(){t("html,body").css("overflow-x","hidden");var e=this.editor,o=this.settings,r=this.infoDialog=e.children("."+this.classPrefix+"dialog-info");return r.length<1&&this.createInfoDialog(),this.lockScreen(!0),this.mask.css({opacity:o.dialogMaskOpacity,backgroundColor:o.dialogMaskBgColor}).show(),r.css("z-index",i.dialogZindex).show(),this.infoDialogPosition(),this},hideInfoDialog:function(){return t("html,body").css("overflow-x",""),this.infoDialog.hide(),this.mask.hide(),this.lockScreen(!1),this},lockScreen:function(e){return i.lockScreen(e),this.resize(),this},recreate:function(){var e=this.editor,t=this.settings;return this.codeMirror.remove(),this.setCodeMirror(),t.readOnly||(e.find(".editormd-dialog").length>0&&e.find(".editormd-dialog").remove(),t.toolbar&&(this.getToolbarHandles(),this.setToolbar())),this.loadedDisplay(!0),this},previewCodeHighlight:function(){var e=this.settings,t=this.previewContainer;return e.previewCodeHighlight&&(t.find("pre").addClass("prettyprint linenums"),"undefined"!=typeof prettyPrint&&prettyPrint()),this},katexRender:function(){return null===o?this:(this.previewContainer.find("."+i.classNames.tex).each(function(){var e=t(this);i.$katex.render(e.text(),e[0]),e.find(".katex").css("font-size","1.6em")}),this)},flowChartAndSequenceDiagramRender:function(){var e=this,o=this.settings,n=this.previewContainer;if(i.isIE8)return this;if(o.flowChart){if(null===r)return this;n.find(".flowchart").flowChart()}o.sequenceDiagram&&n.find(".sequence-diagram").sequenceDiagram({theme:"simple"});var a=e.preview,s=e.codeMirror,l=s.find(".CodeMirror-scroll"),c=l.height(),h=l.scrollTop(),d=h/l[0].scrollHeight,u=0;a.find(".markdown-toc-list").each(function(){u+=t(this).height()});var f=a.find(".editormd-toc-menu").height();return f=f||0,o.syncScrolling&&(0===h?a.scrollTop(0):h+c>=l[0].scrollHeight-16?a.scrollTop(a[0].scrollHeight):a.scrollTop((a[0].scrollHeight+u+f)*d)),this},registerKeyMaps:function(e){var o=this,r=this.cm,n=this.settings,a=i.toolbarHandlers,s=n.disabledKeyMaps;if(e=e||null){for(var l in e)if(t.inArray(l,s)<0){var c={};c[l]=e[l],r.addKeyMap(e)}}else{for(var h in i.keyMaps){var d=i.keyMaps[h],u="string"==typeof d?t.proxy(a[d],o):t.proxy(d,o);if(t.inArray(h,["F9","F10","F11"])<0&&t.inArray(h,s)<0){var f={};f[h]=u,r.addKeyMap(f)}}t(window).keydown(function(e){var i={120:"F9",121:"F10",122:"F11"};if(t.inArray(i[e.keyCode],s)<0)switch(e.keyCode){case 120:return t.proxy(a.watch,o)(),!1;case 121:return t.proxy(a.preview,o)(),!1;case 122:return t.proxy(a.fullscreen,o)(),!1}})}return this},bindScrollEvent:function(){var e=this,o=this.preview,r=this.settings,n=this.codeMirror,a=i.mouseOrTouch;if(!r.syncScrolling)return this;var s=function(){n.find(".CodeMirror-scroll").bind(a("scroll","touchmove"),function(i){var n=t(this).height(),a=t(this).scrollTop(),s=a/t(this)[0].scrollHeight,l=0;o.find(".markdown-toc-list").each(function(){l+=t(this).height()});var c=o.find(".editormd-toc-menu").height();c=c||0,0===a?o.scrollTop(0):a+n>=t(this)[0].scrollHeight-16?o.scrollTop(o[0].scrollHeight):o.scrollTop((o[0].scrollHeight+l+c)*s),t.proxy(r.onscroll,e)(i)})},l=function(){n.find(".CodeMirror-scroll").unbind(a("scroll","touchmove"))},c=function(){o.bind(a("scroll","touchmove"),function(i){var o=t(this).height(),a=t(this).scrollTop(),s=a/t(this)[0].scrollHeight,l=n.find(".CodeMirror-scroll");0===a?l.scrollTop(0):a+o>=t(this)[0].scrollHeight?l.scrollTop(l[0].scrollHeight):l.scrollTop(l[0].scrollHeight*s),t.proxy(r.onpreviewscroll,e)(i)})},h=function(){o.unbind(a("scroll","touchmove"))};return n.bind({mouseover:s,mouseout:l,touchstart:s,touchend:l}),"single"===r.syncScrolling?this:(o.bind({mouseover:c,mouseout:h,touchstart:c,touchend:h}),this)},bindChangeEvent:function(){var e=this,t=this.cm,i=this.settings;return t.on("change",function(t,r){i.watch&&e.previewContainer.css("padding",i.autoHeight?"20px 20px 50px 40px":"20px"),o=setTimeout(function(){clearTimeout(o),e.save(),o=null},i.delay)}),this},loadedDisplay:function(e){e=e||!1;var i=this,o=this.editor,r=this.preview,n=this.settings;return this.containerMask.hide(),this.save(),n.watch&&r.show(),o.data("oldWidth",o.width()).data("oldHeight",o.height()),this.resize(),this.registerKeyMaps(),t(window).resize(function(){i.resize()}),this.bindScrollEvent().bindChangeEvent(),e||t.proxy(n.onload,this)(),this.state.loaded=!0,this},width:function(e){return this.editor.css("width","number"==typeof e?e+"px":e),this.resize(),this},height:function(e){return this.editor.css("height","number"==typeof e?e+"px":e),this.resize(),this},resize:function(e,i){e=e||null,i=i||null;var o=this.state,r=this.editor,n=this.preview,a=this.toolbar,s=this.settings,l=this.codeMirror;if(e&&r.css("width","number"==typeof e?e+"px":e),!s.autoHeight||o.fullscreen||o.preview?(i&&r.css("height","number"==typeof i?i+"px":i),o.fullscreen&&r.height(t(window).height()),s.toolbar&&!s.readOnly?l.css("margin-top",a.height()+1).height(r.height()-a.height()):l.css("margin-top",0).height(r.height())):(r.css({height:"auto",minHeight:s.minHeight}),l.css({height:"auto",minHeight:s.minHeight,maxHeight:s.maxHeight})),s.watch)if(l.width(r.width()/2),n.width(o.preview?r.width():r.width()/2),this.previewContainer.css("padding",s.autoHeight?"20px 20px 50px 40px":"20px"),s.toolbar&&!s.readOnly?n.css("top",a.height()+1):n.css("top",0),!s.autoHeight||o.fullscreen||o.preview){var c=s.toolbar&&!s.readOnly?r.height()-a.height():r.height();n.height(c)}else n.height("");else l.width(r.width()),n.hide();return o.loaded&&t.proxy(s.onresize,this)(),this},save:function(){if(null===o)return this;var e=this,n=this.state,a=this.settings,s=this.cm,l=s.getValue(),c=this.previewContainer;if("gfm"!==a.mode&&"markdown"!==a.mode)return this.markdownTextarea.val(l),this;var h=i.$marked,d=this.markdownToC=[],u=this.markedRendererOptions={toc:a.toc,tocm:a.tocm,tocStartLevel:a.tocStartLevel,pageBreak:a.pageBreak,taskList:a.taskList,emoji:a.emoji,tex:a.tex,atLink:a.atLink,emailLink:a.emailLink,flowChart:a.flowChart,sequenceDiagram:a.sequenceDiagram,previewCodeHighlight:a.previewCodeHighlight},f=this.markedOptions={renderer:i.markedRenderer(d,u),gfm:!0,tables:!0,breaks:!0,pedantic:!1,sanitize:!a.htmlDecode,smartLists:!0,smartypants:!0};h.setOptions(f);var p=i.$marked(l,f);if(p=i.filterHTMLTags(p,a.htmlDecode),this.markdownTextarea.text(l),s.save(),a.saveHTMLToTextarea&&this.htmlTextarea.text(p),a.watch||!a.watch&&n.preview){if(c.html(p),this.previewCodeHighlight(),a.toc){var g=""===a.tocContainer?c:t(a.tocContainer),m=g.find("."+this.classPrefix+"toc-menu");g.attr("previewContainer",""===a.tocContainer?"true":"false"),""!==a.tocContainer&&m.length>0&&m.remove(),i.markdownToCRenderer(d,g,a.tocDropdown,a.tocStartLevel),(a.tocDropdown||g.find("."+this.classPrefix+"toc-menu").length>0)&&i.tocDropdownMenu(g,""!==a.tocTitle?a.tocTitle:this.lang.tocTitle),""!==a.tocContainer&&c.find(".markdown-toc").css("border","none")}a.tex&&(!i.kaTeXLoaded&&a.autoLoadModules?i.loadKaTeX(function(){i.$katex=katex,i.kaTeXLoaded=!0,e.katexRender()}):(i.$katex=katex,this.katexRender())),(a.flowChart||a.sequenceDiagram)&&(r=setTimeout(function(){clearTimeout(r),e.flowChartAndSequenceDiagramRender(),r=null},10)),n.loaded&&t.proxy(a.onchange,this)()}return this},focus:function(){return this.cm.focus(),this},setCursor:function(e){return this.cm.setCursor(e),this},getCursor:function(){return this.cm.getCursor()},setSelection:function(e,t){return this.cm.setSelection(e,t),this},getSelection:function(){return this.cm.getSelection()},setSelections:function(e){return this.cm.setSelections(e),this},getSelections:function(){return this.cm.getSelections()},replaceSelection:function(e){return this.cm.replaceSelection(e),this},insertValue:function(e){return this.replaceSelection(e),this},appendMarkdown:function(e){var t=(this.settings,this.cm);return t.setValue(t.getValue()+e),this},setMarkdown:function(e){return this.cm.setValue(e||this.settings.markdown),this},getMarkdown:function(){return this.cm.getValue()},getValue:function(){return this.cm.getValue()},setValue:function(e){return this.cm.setValue(e),this},clear:function(){return this.cm.setValue(""),this},getHTML:function(){return this.settings.saveHTMLToTextarea?this.htmlTextarea.val():(alert("Error: settings.saveHTMLToTextarea == false"),!1)},getTextareaSavedHTML:function(){return this.getHTML()},getPreviewedHTML:function(){return this.settings.watch?this.previewContainer.html():(alert("Error: settings.watch == false"),!1)},watch:function(e){var i=this.settings;if(t.inArray(i.mode,["gfm","markdown"])<0)return this;if(this.state.watching=i.watch=!0,this.preview.show(),this.toolbar){var r=i.toolbarIconsClass.watch,n=i.toolbarIconsClass.unwatch,a=this.toolbar.find(".fa[name=watch]");a.parent().attr("title",i.lang.toolbar.watch),a.removeClass(n).addClass(r)}return this.codeMirror.css("border-right","1px solid #ddd").width(this.editor.width()/2),o=0,this.save().resize(),i.onwatch||(i.onwatch=e||function(){}),t.proxy(i.onwatch,this)(),this},unwatch:function(e){var i=this.settings;if(this.state.watching=i.watch=!1,this.preview.hide(),this.toolbar){var o=i.toolbarIconsClass.watch,r=i.toolbarIconsClass.unwatch,n=this.toolbar.find(".fa[name=watch]");n.parent().attr("title",i.lang.toolbar.unwatch),n.removeClass(o).addClass(r)}return this.codeMirror.css("border-right","none").width(this.editor.width()),this.resize(),i.onunwatch||(i.onunwatch=e||function(){}),t.proxy(i.onunwatch,this)(),this},show:function(e){e=e||function(){};var i=this;return this.editor.show(0,function(){t.proxy(e,i)()}),this},hide:function(e){e=e||function(){};var i=this;return this.editor.hide(0,function(){t.proxy(e,i)()}),this},previewing:function(){var e=this,o=this.editor,r=this.preview,n=this.toolbar,a=this.settings,s=this.codeMirror,l=this.previewContainer;if(t.inArray(a.mode,["gfm","markdown"])<0)return this;a.toolbar&&n&&(n.toggle(),n.find(".fa[name=preview]").toggleClass("active")),s.toggle();var c=function(t){t.shiftKey&&27===t.keyCode&&e.previewed()};"none"===s.css("display")?(this.state.preview=!0,this.state.fullscreen&&r.css("background","#fff"),o.find("."+this.classPrefix+"preview-close-btn").show().bind(i.mouseOrTouch("click","touchend"),function(){e.previewed()}),a.watch?l.css("padding",""):this.save(),l.addClass(this.classPrefix+"preview-active"),r.show().css({position:"",top:0,width:o.width(),height:a.autoHeight&&!this.state.fullscreen?"auto":o.height()}),this.state.loaded&&t.proxy(a.onpreviewing,this)(),t(window).bind("keyup",c)):(t(window).unbind("keyup",c),this.previewed())},previewed:function(){var e=this.editor,o=this.preview,r=this.toolbar,n=this.settings,a=this.previewContainer,s=e.find("."+this.classPrefix+"preview-close-btn");return this.state.preview=!1,this.codeMirror.show(),n.toolbar&&r.show(),o[n.watch?"show":"hide"](),s.hide().unbind(i.mouseOrTouch("click","touchend")),a.removeClass(this.classPrefix+"preview-active"),n.watch&&a.css("padding","20px"),o.css({background:null,position:"absolute",width:e.width()/2,height:n.autoHeight&&!this.state.fullscreen?"auto":e.height()-r.height(),top:n.toolbar?r.height():0}),this.state.loaded&&t.proxy(n.onpreviewed,this)(),this},fullscreen:function(){var e=this,i=this.state,o=this.editor,r=(this.preview,this.toolbar),n=this.settings,a=this.classPrefix+"fullscreen";r&&r.find(".fa[name=fullscreen]").parent().toggleClass("active");var s=function(t){t.shiftKey||27!==t.keyCode||i.fullscreen&&e.fullscreenExit()};return o.hasClass(a)?(t(window).unbind("keyup",s),this.fullscreenExit()):(i.fullscreen=!0,t("html,body").css("overflow","hidden"),o.css({width:t(window).width(),height:t(window).height()}).addClass(a),this.resize(),t.proxy(n.onfullscreen,this)(),t(window).bind("keyup",s)),this},fullscreenExit:function(){var e=this.editor,i=this.settings,o=this.toolbar,r=this.classPrefix+"fullscreen";return this.state.fullscreen=!1,o&&o.find(".fa[name=fullscreen]").parent().removeClass("active"),t("html,body").css("overflow",""),e.css({width:e.data("oldWidth"),height:e.data("oldHeight")}).removeClass(r),this.resize(),t.proxy(i.onfullscreenExit,this)(),this},executePlugin:function(e,o){var r=this,n=this.cm;return o=this.settings.pluginPath+o,"function"==typeof define?void 0===this[e]?(alert("Error: "+e+" plugin is not found, you are not load this plugin."),this):(this[e](n),this):(t.inArray(o,i.loadFiles.plugin)<0?i.loadPlugin(o,function(){i.loadPlugins[e]=r[e],r[e](n)}):t.proxy(i.loadPlugins[e],this)(n),this)},search:function(e){var t=this.settings;return t.searchReplace?(t.readOnly||this.cm.execCommand(e||"find"),this):(alert("Error: settings.searchReplace == false"),this)},searchReplace:function(){return this.search("replace"),this},searchReplaceAll:function(){return this.search("replaceAll"),this}},i.fn.init.prototype=i.fn,i.dialogLockScreen=function(){(this.settings||{dialogLockScreen:!0}).dialogLockScreen&&(t("html,body").css("overflow","hidden"),this.resize())},i.dialogShowMask=function(e){var i=this.editor,o=this.settings||{dialogShowMask:!0};e.css({top:(t(window).height()-e.height())/2+"px",left:(t(window).width()-e.width())/2+"px"}),o.dialogShowMask&&i.children("."+this.classPrefix+"mask").css("z-index",parseInt(e.css("z-index"))-1).show()},i.toolbarHandlers={undo:function(){this.cm.undo()},redo:function(){this.cm.redo()},bold:function(){var e=this.cm,t=e.getCursor(),i=e.getSelection();e.replaceSelection("**"+i+"**"),""===i&&e.setCursor(t.line,t.ch+2)},del:function(){var e=this.cm,t=e.getCursor(),i=e.getSelection();e.replaceSelection("~~"+i+"~~"),""===i&&e.setCursor(t.line,t.ch+2)},italic:function(){var e=this.cm,t=e.getCursor(),i=e.getSelection();e.replaceSelection("*"+i+"*"),""===i&&e.setCursor(t.line,t.ch+1)},quote:function(){var e=this.cm,t=e.getCursor(),i=e.getSelection();0!==t.ch?(e.setCursor(t.line,0),e.replaceSelection("> "+i),e.setCursor(t.line,t.ch+2)):e.replaceSelection("> "+i)},ucfirst:function(){var e=this.cm,t=e.getSelection(),o=e.listSelections();e.replaceSelection(i.firstUpperCase(t)),e.setSelections(o)},ucwords:function(){var e=this.cm,t=e.getSelection(),o=e.listSelections();e.replaceSelection(i.wordsFirstUpperCase(t)),e.setSelections(o)},uppercase:function(){var e=this.cm,t=e.getSelection(),i=e.listSelections();e.replaceSelection(t.toUpperCase()),e.setSelections(i)},lowercase:function(){var e=this.cm,t=(e.getCursor(),e.getSelection()),i=e.listSelections();e.replaceSelection(t.toLowerCase()),e.setSelections(i)},h1:function(){var e=this.cm,t=e.getCursor(),i=e.getSelection();0!==t.ch?(e.setCursor(t.line,0),e.replaceSelection("# "+i),e.setCursor(t.line,t.ch+2)):e.replaceSelection("# "+i)},h2:function(){var e=this.cm,t=e.getCursor(),i=e.getSelection();0!==t.ch?(e.setCursor(t.line,0),e.replaceSelection("## "+i),e.setCursor(t.line,t.ch+3)):e.replaceSelection("## "+i)},h3:function(){var e=this.cm,t=e.getCursor(),i=e.getSelection();0!==t.ch?(e.setCursor(t.line,0),e.replaceSelection("### "+i),e.setCursor(t.line,t.ch+4)):e.replaceSelection("### "+i)},h4:function(){var e=this.cm,t=e.getCursor(),i=e.getSelection();0!==t.ch?(e.setCursor(t.line,0),e.replaceSelection("#### "+i),e.setCursor(t.line,t.ch+5)):e.replaceSelection("#### "+i)},h5:function(){var e=this.cm,t=e.getCursor(),i=e.getSelection();0!==t.ch?(e.setCursor(t.line,0),e.replaceSelection("##### "+i),e.setCursor(t.line,t.ch+6)):e.replaceSelection("##### "+i)},h6:function(){var e=this.cm,t=e.getCursor(),i=e.getSelection();0!==t.ch?(e.setCursor(t.line,0),e.replaceSelection("###### "+i),e.setCursor(t.line,t.ch+7)):e.replaceSelection("###### "+i)},"list-ul":function(){var e=this.cm,t=(e.getCursor(),e.getSelection());if(""===t)e.replaceSelection("- "+t);else{for(var i=t.split("\n"),o=0,r=i.length;o=0,a=n?"Cmd":"Ctrl";i.keyMaps=(e={},_defineProperty(e,a+"-1","h1"),_defineProperty(e,a+"-2","h2"),_defineProperty(e,a+"-3","h3"),_defineProperty(e,a+"-4","h4"),_defineProperty(e,a+"-5","h5"),_defineProperty(e,a+"-6","h6"),_defineProperty(e,a+"-B","bold"),_defineProperty(e,a+"-D","datetime"),_defineProperty(e,a+"Ctrl-E",function(){var e=this.cm,t=e.getCursor(),i=e.getSelection();if(!this.settings.emoji)return void alert("Error: settings.emoji == false");e.replaceSelection(":"+i+":"),""===i&&e.setCursor(t.line,t.ch+1)}),_defineProperty(e,a+"-Alt-G","goto-line"),_defineProperty(e,a+"-H","hr"),_defineProperty(e,a+"-I","italic"),_defineProperty(e,a+"-K","code"),_defineProperty(e,"Ctrl-L",function(){var e=this.cm,t=e.getCursor(),i=e.getSelection(),o=""===i?"":' "'+i+'"';e.replaceSelection("["+i+"]("+o+")"),""===i&&e.setCursor(t.line,t.ch+1)}),_defineProperty(e,a+"-U","list-ul"),_defineProperty(e,"Shift-Ctrl-A",function(){var e=this.cm,t=e.getCursor(),i=e.getSelection();if(!this.settings.atLink)return void alert("Error: settings.atLink == false");e.replaceSelection("@"+i),""===i&&e.setCursor(t.line,t.ch+1)}),_defineProperty(e,"Shift"+a+"-C","code"),_defineProperty(e,"Shift"+a+"Q","quote"),_defineProperty(e,"Shift"+a+"S","del"),_defineProperty(e,"Shift"+a+"K","tex"),_defineProperty(e,"Shift-Alt-C",function(){var e=this.cm,t=e.getCursor(),i=e.getSelection();e.replaceSelection(["```",i,"```"].join("\n")),""===i&&e.setCursor(t.line,t.ch+3)}),_defineProperty(e,"Shift-"+a+"-Alt-C","code-block"),_defineProperty(e,"Shift-"+a+"-H","html-entities"),_defineProperty(e,"Shift-Alt-H","help"),_defineProperty(e,"Shift-"+a+"-E","emoji"),_defineProperty(e,"Shift-"+a+"-U","uppercase"),_defineProperty(e,"Shift-Alt-U","ucwords"),_defineProperty(e,"Shift-"+a+"-Alt-U","ucfirst"),_defineProperty(e,"Shift-Alt-L","lowercase"),_defineProperty(e,"Shift-"+a+"-I",function(){var e=this.cm,t=e.getCursor(),i=e.getSelection(),o=""===i?"":' "'+i+'"';e.replaceSelection("!["+i+"]("+o+")"),""===i&&e.setCursor(t.line,t.ch+4)}),_defineProperty(e,"Shift-"+a+"-Alt-I","image"),_defineProperty(e,"Shift-"+a+"-L","link"),_defineProperty(e,"Shift-"+a+"-O","list-ol"),_defineProperty(e,"Shift-"+a+"-P","preformatted-text"),_defineProperty(e,"Shift-"+a+"-T","table"),_defineProperty(e,"Shift-Alt-P","pagebreak"),_defineProperty(e,"F9","watch"),_defineProperty(e,"F10","preview"),_defineProperty(e,"F11","fullscreen"),e);var s=function(e){return String.prototype.trim?e.trim():e.replace(/^[\s\uFEFF\xA0]+|[\s\uFEFF\xA0]+$/g,"")};i.trim=s;var l=function(e){return e.toLowerCase().replace(/\b(\w)|\s(\w)/g,function(e){return e.toUpperCase()})};i.ucwords=i.wordsFirstUpperCase=l;var c=function(e){return e.toLowerCase().replace(/\b(\w)/,function(e){return e.toUpperCase()})};return i.firstUpperCase=i.ucfirst=c,i.urls={atLinkBase:"https://github.com/"},i.regexs={atLink:/@(\w+)/g,email:/(\w+)@(\w+)\.(\w+)\.?(\w+)?/g,emailLink:/(mailto:)?([\w\.\_]+)@(\w+)\.(\w+)\.?(\w+)?/g,emoji:/:([\w\+-]+):/g,emojiDatetime:/(\d{1,2}:\d{1,2}:\d{1,2})/g,twemoji:/:(tw-([\w]+)-?(\w+)?):/g,fontAwesome:/:(fa-([\w]+)(-(\w+)){0,}):/g,editormdLogo:/:(editormd-logo-?(\w+)?):/g,pageBreak:/^\[[=]{8,}\]$/},i.emoji={path:"http://www.emoji-cheat-sheet.com/graphics/emojis/",ext:".png"},i.twemoji={path:"http://twemoji.maxcdn.com/36x36/",ext:".png"},i.markedRenderer=function(e,o){var r={toc:!0,tocm:!1,tocStartLevel:1,pageBreak:!0,atLink:!0,emailLink:!0,taskList:!1,emoji:!1,tex:!1,flowChart:!1,sequenceDiagram:!1},n=t.extend(r,o||{}),a=i.$marked,l=new a.Renderer;e=e||[];var c=i.regexs,h=c.atLink,d=c.emoji,u=c.email,f=c.emailLink,p=c.twemoji,g=c.fontAwesome,m=c.editormdLogo,w=c.pageBreak;return l.emoji=function(e){e=e.replace(i.regexs.emojiDatetime,function(e){return e.replace(/:/g,":")});var t=e.match(d);if(!t||!n.emoji)return e;for(var o=0,r=t.length;o'}else{var l=e.match(m),c=e.match(p);if(l)for(var h=0,d=l.length;h'}else{if(!c){var f="+1"===r?"plus1":r;return f="black_large_square"===f?"black_square":f,f="moon"===f?"waxing_gibbous_moon":f,':'+r+':'}for(var w=0,v=c.length;w'}}}});return e},l.atLink=function(e){return h.test(e)?(n.atLink&&(e=e.replace(u,function(e,t,i,o){return e.replace(/@/g,"_#_@_#_")}),e=e.replace(h,function(e,t){return''+e+""}).replace(/_#_@_#_/g,"@")),n.emailLink&&(e=e.replace(f,function(e,i,o,r,n){return!i&&t.inArray(n,"jpg|jpeg|png|gif|webp|ico|icon|pdf".split("|"))<0?''+e+"":e})),e):e},l.link=function(e,t,i){if(this.options.sanitize)try{if(0===decodeURIComponent(unescape(e)).replace(/[^\w:]/g,"").toLowerCase().indexOf("javascript:"))return""}catch(e){return""}var o=''+i.replace(/@/g,"@")+""):(t&&(o+=' title="'+t+'"'),o+=">"+i+"")},l.heading=function(t,i,o){var r=t,n=/\s*]*)\>(.*)<\/a\>\s*/;if(n.test(t)){var a=[];t=t.split(/]+)\>([^\>]*)<\/a\>/);for(var l=0,c=t.length;l';return p+='',p+='',p+=n?this.atLink(this.emoji(r)):this.atLink(this.emoji(t)),p+=""},l.pageBreak=function(e){return w.test(e)&&n.pageBreak&&(e='
          '),e},l.paragraph=function(e){var t=/\$\$(.*)\$\$/g.test(e),o=/^\$\$(.*)\$\$$/.test(e),r=o?' class="'+i.classNames.tex+'"':"",a=n.tocm?/^(\[TOC\]|\[TOCM\])$/.test(e):/^\[TOC\]$/.test(e),s=/^\[TOCM\]$/.test(e);e=!o&&t?e.replace(/(\$\$([^\$]*)\$\$)+/g,function(e,t){return''+t.replace(/\$/g,"")+""}):o?e.replace(/\$/g,""):e;var l='
          '+e+"
          ";return a?s?'
          '+l+"

          ":l:w.test(e)?this.pageBreak(e):""+this.atLink(this.emoji(e))+"

          \n"},l.code=function(e,t,o){return"seq"===t||"sequence"===t?'
          '+e+"
          ":"flow"===t?'
          '+e+"
          ":"math"===t||"latex"===t||"katex"===t?'

          '+e+"

          ":a.Renderer.prototype.code.apply(this,arguments)},l.tablecell=function(e,t){var i=t.header?"th":"td";return(t.align?"<"+i+' style="text-align:'+t.align+'">':"<"+i+">")+this.atLink(this.emoji(e))+"\n"},l.listitem=function(e){return n.taskList&&/^\s*\[[x\s]\]\s*/.test(e)?(e=e.replace(/^\s*\[\s\]\s*/,' ').replace(/^\s*\[x\]\s*/,' '),'
        • '+this.atLink(this.emoji(e))+"
        • "):"
        • "+this.atLink(this.emoji(e))+"
        • "},l},i.markdownToCRenderer=function(e,t,i,o){var r="",n=[0],a=this.classPrefix;o=o||1;for(var s=0,l=e.length;su)r+="",n.push(h);else if(h=0;p--){if(!(h<=n[p])){n.push(h);break}f++,n.pop()}r+=new Array(f+1).join("
      • ")}else r+="";r+='
      • '+c+"
          "}}var g=t.find(".markdown-toc");if(g.length<1&&"false"===t.attr("previewContainer")){var m='
          ';m=i?'
          '+m+"
          ":m,t.html(m),g=t.find(".markdown-toc")}return i&&g.wrap('

          '),g.html('
            ').children(".markdown-toc-list").html(r.replace(/\r?\n?<\/ul\>/g,"")),g},i.tocDropdownMenu=function(e,i){i=i||"Table of Contents";var o=400,r=e.find("."+this.classPrefix+"toc-menu");return r.each(function(){var e=t(this),r=e.children(".markdown-toc"),n='',a=''+n+i+"",s=r.children("ul"),l=s.find("li");r.append(a),l.first().before("
          • "+i+" "+n+"

          • "),e.mouseover(function(){s.show(),l.each(function(){var e=t(this),i=e.children("ul");if(""===i.html()&&i.remove(),i.length>0&&""!==i.html()){var r=e.children("a").first();r.children(".fa").length<1&&r.append(t(n).css({float:"right",paddingTop:"4px"}))}e.mouseover(function(){i.css("z-index",o).show(),o+=1}).mouseleave(function(){i.hide()})})}).mouseleave(function(){s.hide()})}),r},i.filterHTMLTags=function(e,i){if("string"!=typeof e&&(e=new String(e)),"string"!=typeof i)return e;for(var o=i.split("|"),r=o[0].split(","),n=o[1],a=0,s=r.length;a]*)>([^>]*)","igm"),"")}if(void 0!==n){var c=/<(\w+)\s*([^\>]*)\>([^\>]*)<\/(\w+)\>/gi;e="*"===n?e.replace(c,function(e,t,i,o,r){return"<"+t+">"+o+""}):"on*"===n?e.replace(c,function(e,i,o,r,n){var a=t("<"+i+">"+r+""),s=t(e)[0].attributes,l={};t.each(s,function(e,t){'"'!==t.nodeName&&(l[t.nodeName]=t.nodeValue,"href"===t.nodeName&&t.nodeValue.toLowerCase().indexOf("javascript:")>=0&&(l[t.nodeName]="javascript:;"))}),t.each(l,function(e){0===e.indexOf("on")&&delete l[e]}),a.attr(l);var c=void 0!==a[1]?t(a[1]).text():"";if(!i||isNaN(i)||i===n)return i&&isNaN(i)&&i!==n?(c+="",a[0].outerHTML+c):a[0].outerHTML+c;c+="<"+i+">"+r+""}):e.replace(c,function(e,i,o,r){var a=n.split(","),s=t(e);return s.html(r),t.each(a,function(e){s.attr(a[e],null)}),s[0].outerHTML})}return e},i.markdownToHTML=function(e,o){var r={gfm:!0,toc:!0,tocm:!1,tocStartLevel:1,tocTitle:"目录",tocDropdown:!1,tocContainer:"",markdown:"",markdownSourceCode:!1,htmlDecode:!1,autoLoadKaTeX:!0,pageBreak:!0,atLink:!0,emailLink:!0,tex:!1,taskList:!1,emoji:!1,flowChart:!1,sequenceDiagram:!1,previewCodeHighlight:!0};i.$marked=marked;var n=t("#"+e),a=n.settings=t.extend(!0,r,o||{}),s=n.find("textarea");s.length<1&&(n.append(""),s=n.find("textarea"));var l=""===a.markdown?s.val():a.markdown,c=[],h={toc:a.toc,tocm:a.tocm,tocStartLevel:a.tocStartLevel,taskList:a.taskList,emoji:a.emoji,tex:a.tex,pageBreak:a.pageBreak,atLink:a.atLink,emailLink:a.emailLink,flowChart:a.flowChart,sequenceDiagram:a.sequenceDiagram,previewCodeHighlight:a.previewCodeHighlight},d={renderer:i.markedRenderer(c,h),gfm:a.gfm,tables:!0,breaks:!0,pedantic:!1,sanitize:!a.htmlDecode,smartLists:!0,smartypants:!0};l=new String(l);var u=marked(l,d);u=i.filterHTMLTags(u,a.htmlDecode),a.markdownSourceCode?s.text(l):s.remove(),n.addClass("markdown-body "+this.classPrefix+"html-preview").append(u);var f=""!==a.tocContainer?t(a.tocContainer):n;if(""!==a.tocContainer&&f.attr("previewContainer",!1),a.toc&&(n.tocContainer=this.markdownToCRenderer(c,f,a.tocDropdown,a.tocStartLevel),(a.tocDropdown||n.find("."+this.classPrefix+"toc-menu").length>0)&&this.tocDropdownMenu(n,a.tocTitle),""!==a.tocContainer&&n.find(".editormd-toc-menu, .editormd-markdown-toc").remove()),a.previewCodeHighlight&&(n.find("pre").addClass("prettyprint linenums"),prettyPrint()),i.isIE8||(a.flowChart&&n.find(".flowchart").flowChart(),a.sequenceDiagram&&n.find(".sequence-diagram").sequenceDiagram({theme:"simple"})),a.tex){var p=function(){n.find("."+i.classNames.tex).each(function(){var e=t(this);katex.render(e.html().replace(/</g,"<").replace(/>/g,">"),e[0]),e.find(".katex").css("font-size","1.6em")})};!a.autoLoadKaTeX||i.$katex||i.kaTeXLoaded?p():this.loadKaTeX(function(){i.$katex=katex,i.kaTeXLoaded=!0,p()})}return n.getMarkdown=function(){return s.val()},n},i.themes=["default","dark"],i.previewThemes=["default","dark"],i.editorThemes=["default","3024-day","3024-night","ambiance","ambiance-mobile","base16-dark","base16-light","blackboard","cobalt","eclipse","elegant","erlang-dark","lesser-dark","mbo","mdn-like","midnight","monokai","neat","neo","night","paraiso-dark","paraiso-light","pastel-on-dark","rubyblue","solarized","the-matrix","tomorrow-night-eighties","twilight","vibrant-ink","xq-dark","xq-light"],i.loadPlugins={},i.loadFiles={js:[],css:[],plugin:[]},i.loadPlugin=function(e,t,o){t=t||function(){},this.loadScript(e,function(){i.loadFiles.plugin.push(e),t()},o)},i.loadCSS=function(e,t,o){o=o||"head",t=t||function(){};var r=document.createElement("link");r.type="text/css",r.rel="stylesheet",r.onload=r.onreadystatechange=function(){i.loadFiles.css.push(e),t()},r.href=e+".css","head"===o?document.getElementsByTagName("head")[0].appendChild(r):document.body.appendChild(r)},i.isIE="Microsoft Internet Explorer"===navigator.appName,i.isIE8=i.isIE&&"8."===navigator.appVersion.match(/8./i),i.loadScript=function(e,t,o){o=o||"head",t=t||function(){};var r=null;r=document.createElement("script"),r.id=e.replace(/[\.\/]+/g,"-"),r.type="text/javascript",r.src=e+".js",i.isIE8?r.onreadystatechange=function(){r.readyState&&("loaded"!==r.readyState&&"complete"!==r.readyState||(r.onreadystatechange=null,i.loadFiles.js.push(e),t()))}:r.onload=function(){i.loadFiles.js.push(e),t()},"head"===o?document.getElementsByTagName("head")[0].appendChild(r):document.body.appendChild(r)},i.katexURL={css:"//cdnjs.cloudflare.com/ajax/libs/KaTeX/0.3.0/katex.min",js:"//cdnjs.cloudflare.com/ajax/libs/KaTeX/0.3.0/katex.min"},i.kaTeXLoaded=!1,i.loadKaTeX=function(e){i.loadCSS(i.katexURL.css,function(){i.loadScript(i.katexURL.js,e||function(){})})},i.lockScreen=function(e){t("html,body").css("overflow",e?"hidden":"")},i.createDialog=function(e){var o={name:"",width:420,height:240,title:"",drag:!0,closed:!0,content:"",mask:!0,maskStyle:{backgroundColor:"#fff",opacity:.1},lockScreen:!0,footer:!0,buttons:!1};e=t.extend(!0,o,e);var r=this,n=this.editor,a=i.classPrefix,s=(new Date).getTime(),l=""===e.name?a+"dialog-"+s:e.name,c=i.mouseOrTouch,h='
            ';""!==e.title&&(h+='
            ",h+=''+e.title+"",h+="
            "),e.closed&&(h+=''),h+='
            '+e.content,(e.footer||"string"==typeof e.footer)&&(h+='"),h+="
            ",h+='
            ',h+='
            ',h+="
            ",n.append(h);var d=n.find("."+l);d.lockScreen=function(i){return e.lockScreen&&(t("html,body").css("overflow",i?"hidden":""),r.resize()),d},d.showMask=function(){return e.mask&&n.find("."+a+"mask").css(e.maskStyle).css("z-index",i.dialogZindex-1).show(),d},d.hideMask=function(){return e.mask&&n.find("."+a+"mask").hide(),d},d.loading=function(e){return d.find("."+a+"dialog-mask")[e?"show":"hide"](),d},d.lockScreen(!0).showMask(),d.show().css({zIndex:i.dialogZindex,border:i.isIE8?"1px solid #ddd":"",width:"number"==typeof e.width?e.width+"px":e.width,height:"number"==typeof e.height?e.height+"px":e.height});var u=function(){d.css({top:(t(window).height()-d.height())/2+"px",left:(t(window).width()-d.width())/2+"px"})};if(u(),t(window).resize(u),d.children("."+a+"dialog-close").bind(c("click","touchend"),function(){d.hide().lockScreen(!1).hideMask()}),"object"===_typeof(e.buttons)){var f=d.footer=d.find("."+a+"dialog-footer");for(var p in e.buttons){var g=e.buttons[p],m=a+p+"-btn";f.append('"),g[1]=t.proxy(g[1],d),f.children("."+m).bind(c("click","touchend"),g[1])}}if(""!==e.title&&e.drag){var w,v,k=d.children("."+a+"dialog-header");e.mask||k.bind(c("click","touchend"),function(){i.dialogZindex+=2,d.css("z-index",i.dialogZindex)});var y=function(e){e.removeClass(a+"user-unselect").off("selectstart")},b=function(e){e.addClass(a+"user-unselect").on("selectstart",function(e){return!1})},x=function(e){e=e||window.event;var i,o,r=parseInt(d[0].style.left),n=parseInt(d[0].style.top);r>=0?r+d.width()<=t(window).width()?i=e.clientX-w:(i=t(window).width()-d.width(),document.onmousemove=null):(i=0,document.onmousemove=null),n>=0?o=e.clientY-v:(o=0,document.onmousemove=null),document.onselectstart=function(){return!1},b(t("body")),b(d),d[0].style.left=i+"px",d[0].style.top=o+"px"};k.mousedown(function(e){e=e||window.event,w=e.clientX-parseInt(d[0].style.left),v=e.clientY-parseInt(d[0].style.top),document.onmousemove=x}),document.onmouseup=function(){y(t("body")),y(d),document.onselectstart=null,document.onmousemove=null},k.touchDraggable=function(){var e=null,i=function(i){var o=i.originalEvent,r=t(this).parent().position();e={x:o.changedTouches[0].pageX-r.left,y:o.changedTouches[0].pageY-r.top}},o=function(i){i.preventDefault();var o=i.originalEvent;t(this).parent().css({top:o.changedTouches[0].pageY-e.y,left:o.changedTouches[0].pageX-e.x})};this.bind("touchstart",i).bind("touchmove",o)},k.touchDraggable()}return i.dialogZindex+=2,d},i.mouseOrTouch=function(e,t){e=e||"click",t=t||"touchend";var i=e;try{document.createEvent("TouchEvent"),i=t}catch(e){}return i},i.dateFormat=function(e){e=e||"";var t=function(e){return e<10?"0"+e:e},i=new Date,o=i.getFullYear(),r=o.toString().slice(2,4),n=t(i.getMonth()+1),a=t(i.getDate()),s=i.getDay(),l=t(i.getHours()),c=t(i.getMinutes()),h=t(i.getSeconds()),d=t(i.getMilliseconds()),u="",f=r+"-"+n+"-"+a,p=o+"-"+n+"-"+a,g=l+":"+c+":"+h;switch(e){case"UNIX Time":u=i.getTime();break;case"UTC":u=i.toUTCString();break;case"yy":u=r;break;case"year":case"yyyy":u=o;break;case"month":case"mm":u=n;break;case"cn-week-day":case"cn-wd":u="星期"+["日","一","二","三","四","五","六"][s];break;case"week-day":case"wd":u=["Sunday","Monday","Tuesday","Wednesday","Thursday","Friday","Saturday"][s];break;case"day":case"dd":u=a;break;case"hour":case"hh":u=l;break;case"min":case"ii":u=c;break;case"second":case"ss":u=h;break;case"ms":u=d;break;case"yy-mm-dd":u=f;break;case"yyyy-mm-dd":u=p;break;case"yyyy-mm-dd h:i:s ms":case"full + ms":u=p+" "+g+" "+d;break;case"full":case"yyyy-mm-dd h:i:s":break;default:u=p+" "+g}return u},i}}); \ No newline at end of file diff --git a/system/admin/editor.md/editormd.js b/system/admin/editor.md/editormd.js new file mode 100644 index 00000000..46372d13 --- /dev/null +++ b/system/admin/editor.md/editormd.js @@ -0,0 +1,4262 @@ +/* + * Editor.md + * + * @file editormd.js + * @version v2.0.1 + * @description Open source online markdown editor. + * @license MIT License + * @author Pandao + * {@link https://github.com/pandao/editor.md} + * @updateTime 2018-03-29 + */ + +;(function(factory) { + "use strict" + + // CommonJS/Node.js + if ( + typeof require === "function" && + typeof exports === "object" && + typeof module === "object" + ) { + module.exports = factory + } else if (typeof define === "function") { + // AMD/CMD/Sea.js + if (define.amd) { + // for Require.js + /* Require.js define replace */ + } else { + define(["jquery"], factory) // for Sea.js + } + } else { + window.editormd = factory() + } +})(function() { + /* Require.js assignment replace */ + + "use strict" + + var $ = typeof jQuery !== "undefined" ? jQuery : Zepto + + if (typeof $ === "undefined") { + return + } + + /** + * editormd + * + * @param {String} id 编辑器的ID + * @param {Object} options 配置选项 Key/Value + * @returns {Object} editormd 返回editormd对象 + */ + + var editormd = function(id, options) { + return new editormd.fn.init(id, options) + } + + editormd.title = editormd.$name = "Editor.md" + editormd.version = "1.5.0" + editormd.homePage = "https://pandao.github.io/editor.md/" + editormd.classPrefix = "editormd-" + + editormd.toolbarModes = { + full: [ + "undo", + "redo", + "|", + "bold", + "del", + "italic", + "quote", + "ucwords", + "uppercase", + "lowercase", + "|", + "h1", + "h2", + "h3", + "h4", + "h5", + "h6", + "|", + "list-ul", + "list-ol", + "hr", + "|", + "link", + "reference-link", + "image", + "code", + "preformatted-text", + "code-block", + "table", + "datetime", + "emoji", + "html-entities", + "pagebreak", + "|", + "goto-line", + "watch", + "preview", + "fullscreen", + "clear", + "search", + "|", + "help", + "info" + ], + simple: [ + "undo", + "redo", + "|", + "bold", + "del", + "italic", + "quote", + "uppercase", + "lowercase", + "|", + "h1", + "h2", + "h3", + "h4", + "h5", + "h6", + "|", + "list-ul", + "list-ol", + "hr", + "|", + "watch", + "preview", + "fullscreen", + "|", + "help", + "info" + ], + mini: ["undo", "redo", "|", "watch", "preview", "|", "help", "info"] + } + + editormd.defaults = { + mode: "gfm", //gfm or markdown + name: "", // Form element name + value: "", // value for CodeMirror, if mode not gfm/markdown + theme: "", // Editor.md self themes, before v1.5.0 is CodeMirror theme, default empty + editorTheme: "default", // Editor area, this is CodeMirror theme at v1.5.0 + previewTheme: "", // Preview area theme, default empty + markdown: "", // Markdown source code + appendMarkdown: "", // if in init textarea value not empty, append markdown to textarea + width: "100%", + height: "100%", + path: "./lib/", // Dependents module file directory + pluginPath: "", // If this empty, default use settings.path + "../plugins/" + delay: 300, // Delay parse markdown to html, Uint : ms + autoLoadModules: true, // Automatic load dependent module files + watch: true, + placeholder: "Enjoy Markdown! coding now...", + gotoLine: true, + codeFold: false, + autoHeight: false, + minHeight: "150px", + maxHeight: "none", // default "none" + autoFocus: true, + autoCloseTags: true, + searchReplace: true, + syncScrolling: true, // true | false | "single", default true + readOnly: false, + tabSize: 4, + indentUnit: 4, + lineNumbers: true, + lineWrapping: true, + autoCloseBrackets: true, + showTrailingSpace: true, + matchBrackets: true, + indentWithTabs: true, + styleSelectedText: true, + matchWordHighlight: true, // options: true, false, "onselected" + styleActiveLine: true, // Highlight the current line + dialogLockScreen: true, + dialogShowMask: true, + dialogDraggable: true, + dialogMaskBgColor: "#fff", + dialogMaskOpacity: 0.1, + fontSize: "13px", + saveHTMLToTextarea: false, + disabledKeyMaps: [], + onload: function() {}, + onresize: function() {}, + onchange: function() {}, + onwatch: null, + onunwatch: null, + onpreviewing: function() {}, + onpreviewed: function() {}, + onfullscreen: function() {}, + onfullscreenExit: function() {}, + onscroll: function() {}, + onpreviewscroll: function() {}, + imageUpload: false, + imageFormats: ["jpg", "jpeg", "gif", "png", "bmp", "webp"], + imageUploadURL: "", + crossDomainUpload: false, + uploadCallbackURL: "", + toc: true, // Table of contents + tocm: false, // Using [TOCM], auto create ToC dropdown menu + tocTitle: "", // for ToC dropdown menu btn + tocDropdown: false, + tocContainer: "", + tocStartLevel: 1, // Said from H1 to create ToC + htmlDecode: false, // Open the HTML tag identification + pageBreak: true, // Enable parse page break [========] + atLink: true, // for @link + emailLink: true, // for email address auto link + taskList: false, // Enable Github Flavored Markdown task lists + emoji: false, // :emoji: , Support Github emoji, Twitter Emoji (Twemoji); + // Support FontAwesome icon emoji :fa-xxx: > Using fontAwesome icon web fonts; + // Support Editor.md logo icon emoji :editormd-logo: :editormd-logo-1x: > 1~8x; + tex: false, // TeX(LaTeX), based on KaTeX + flowChart: false, // flowChart.js only support IE9+ + sequenceDiagram: false, // sequenceDiagram.js only support IE9+ + previewCodeHighlight: true, + toolbar: true, // show/hide toolbar + toolbarAutoFixed: true, // on window scroll auto fixed position + toolbarIcons: "full", + toolbarTitles: {}, + toolbarHandlers: { + ucwords: function() { + return editormd.toolbarHandlers.ucwords + }, + lowercase: function() { + return editormd.toolbarHandlers.lowercase + } + }, + toolbarCustomIcons: { + // using html tag create toolbar icon, unused default tag. + lowercase: + 'a', + ucwords: + 'Aa' + }, + toolbarIconsClass: { + undo: "fa-undo", + redo: "fa-repeat", + bold: "fa-bold", + del: "fa-strikethrough", + italic: "fa-italic", + quote: "fa-quote-left", + uppercase: "fa-font", + h1: editormd.classPrefix + "bold", + h2: editormd.classPrefix + "bold", + h3: editormd.classPrefix + "bold", + h4: editormd.classPrefix + "bold", + h5: editormd.classPrefix + "bold", + h6: editormd.classPrefix + "bold", + "list-ul": "fa-list-ul", + "list-ol": "fa-list-ol", + hr: "fa-minus", + link: "fa-link", + "reference-link": "fa-anchor", + image: "fa-picture-o", + code: "fa-code", + "preformatted-text": "fa-file-code-o", + "code-block": "fa-file-code-o", + table: "fa-table", + datetime: "fa-clock-o", + emoji: "fa-smile-o", + "html-entities": "fa-copyright", + pagebreak: "fa-newspaper-o", + "goto-line": "fa-terminal", // fa-crosshairs + watch: "fa-eye-slash", + unwatch: "fa-eye", + preview: "fa-desktop", + search: "fa-search", + fullscreen: "fa-arrows-alt", + clear: "fa-eraser", + help: "fa-question-circle", + info: "fa-info-circle" + }, + toolbarIconTexts: {}, + lang: { + name: "zh-cn", + description: + "开源在线Markdown编辑器
            Open source online Markdown editor.", + tocTitle: "目录", + toolbar: { + undo: "撤销(Ctrl+Z)", + redo: "重做(Ctrl+Y)", + bold: "粗体", + del: "删除线", + italic: "斜体", + quote: "引用", + ucwords: "将每个单词首字母转成大写", + uppercase: "将所选转换成大写", + lowercase: "将所选转换成小写", + h1: "标题1", + h2: "标题2", + h3: "标题3", + h4: "标题4", + h5: "标题5", + h6: "标题6", + "list-ul": "无序列表", + "list-ol": "有序列表", + hr: "横线", + link: "链接", + "reference-link": "引用链接", + image: "添加图片", + code: "行内代码", + "preformatted-text": "预格式文本 / 代码块(缩进风格)", + "code-block": "代码块(多语言风格)", + table: "添加表格", + datetime: "日期时间", + emoji: "Emoji表情", + "html-entities": "HTML实体字符", + pagebreak: "插入分页符", + "goto-line": "跳转到行", + watch: "关闭实时预览", + unwatch: "开启实时预览", + preview: "全窗口预览HTML(按 Shift + ESC还原)", + fullscreen: "全屏(按ESC还原)", + clear: "清空", + search: "搜索", + help: "使用帮助", + info: "关于" + editormd.title + }, + buttons: { + enter: "确定", + cancel: "取消", + close: "关闭" + }, + dialog: { + link: { + title: "添加链接", + url: "链接地址", + urlTitle: "链接标题", + urlEmpty: "错误:请填写链接地址。" + }, + referenceLink: { + title: "添加引用链接", + name: "引用名称", + url: "链接地址", + urlId: "链接ID", + urlTitle: "链接标题", + nameEmpty: "错误:引用链接的名称不能为空。", + idEmpty: "错误:请填写引用链接的ID。", + urlEmpty: "错误:请填写引用链接的URL地址。" + }, + image: { + title: "添加图片", + url: "图片地址", + link: "图片链接", + alt: "图片描述", + uploadButton: "本地上传", + imageURLEmpty: "错误:图片地址不能为空。", + uploadFileEmpty: "错误:上传的图片不能为空。", + formatNotAllowed: + "错误:只允许上传图片文件,允许上传的图片文件格式有:" + }, + preformattedText: { + title: "添加预格式文本或代码块", + emptyAlert: "错误:请填写预格式文本或代码的内容。" + }, + codeBlock: { + title: "添加代码块", + selectLabel: "代码语言:", + selectDefaultText: "请选择代码语言", + otherLanguage: "其他语言", + unselectedLanguageAlert: "错误:请选择代码所属的语言类型。", + codeEmptyAlert: "错误:请填写代码内容。" + }, + htmlEntities: { + title: "HTML 实体字符" + }, + help: { + title: "使用帮助" + } + } + } + } + editormd.classNames = { + tex: editormd.classPrefix + "tex" + } + editormd.dialogZindex = 99999 + editormd.$katex = null + editormd.$marked = null + editormd.$CodeMirror = null + editormd.$prettyPrint = null + + var timer, flowchartTimer + + editormd.prototype = editormd.fn = { + state: { + watching: false, + loaded: false, + preview: false, + fullscreen: false + }, + + /** + * 构造函数/实例初始化 + * Constructor / instance initialization + * + * @param {String} id 编辑器的ID + * @param {Object} [options={}] 配置选项 Key/Value + * @returns {editormd} 返回editormd的实例对象 + */ + + init: function(id, options) { + options = options || {} + if (typeof id === "object") { + options = id + } + var _this = this + var classPrefix = (this.classPrefix = editormd.classPrefix) + var settings = (this.settings = $.extend( + true, + editormd.defaults, + options + )) + id = typeof id === "object" ? settings.id : id + + var editor = (this.editor = $("#" + id)) + this.id = id + this.lang = settings.lang + var classNames = (this.classNames = { + textarea: { + html: classPrefix + "html-textarea", + markdown: classPrefix + "markdown-textarea" + } + }) + settings.pluginPath = + settings.pluginPath === "" + ? settings.path + "../plugins/" + : settings.pluginPath + + this.state.watching = settings.watch ? true : false + if (!editor.hasClass("editormd")) { + editor.addClass("editormd") + } + editor.css({ + width: + typeof settings.width === "number" + ? settings.width + "px" + : settings.width, + height: + typeof settings.height === "number" + ? settings.height + "px" + : settings.height + }) + if (settings.autoHeight) { + editor.css({ + height: "auto", + minHeight: settings.minHeight + }) + } + var markdownTextarea = (this.markdownTextarea = editor.children( + "textarea" + )) + if (markdownTextarea.length < 1) { + editor.append("") + markdownTextarea = this.markdownTextarea = editor.children("textarea") + } + markdownTextarea + .addClass(classNames.textarea.markdown) + .attr("placeholder", settings.placeholder) + if ( + typeof markdownTextarea.attr("name") === "undefined" || + markdownTextarea.attr("name") === "" + ) { + markdownTextarea.attr( + "name", + settings.name !== "" ? settings.name : id + "-markdown-doc" + ) + } + var appendElements = [ + !settings.readOnly + ? '' + : "", + settings.saveHTMLToTextarea + ? '' + : "", + '
            ', + '
            ', + '
            ' + ].join("\n") + editor.append(appendElements).addClass(classPrefix + "vertical") + if (settings.theme !== "") { + editor.addClass(classPrefix + "theme-" + settings.theme) + } + this.mask = editor.children("." + classPrefix + "mask") + this.containerMask = editor.children("." + classPrefix + "container-mask") + if (settings.markdown !== "") { + markdownTextarea.val(settings.markdown) + } + if (settings.appendMarkdown !== "") { + markdownTextarea.val(markdownTextarea.val() + settings.appendMarkdown) + } + this.htmlTextarea = editor.children("." + classNames.textarea.html) + this.preview = editor.children("." + classPrefix + "preview") + this.previewContainer = this.preview.children( + "." + classPrefix + "preview-container" + ) + if (settings.previewTheme !== "") { + this.preview.addClass( + classPrefix + "preview-theme-" + settings.previewTheme + ) + } + if (typeof define === "function" && define.amd) { + if (typeof katex !== "undefined") { + editormd.$katex = katex + } + if (settings.searchReplace && !settings.readOnly) { + editormd.loadCSS(settings.path + "codemirror/addon/dialog/dialog") + editormd.loadCSS( + settings.path + "codemirror/addon/search/matchesonscrollbar" + ) + } + } + if ( + (typeof define === "function" && define.amd) || + !settings.autoLoadModules + ) { + if (typeof CodeMirror !== "undefined") { + editormd.$CodeMirror = CodeMirror + } + + if (typeof marked !== "undefined") { + editormd.$marked = marked + } + this.setCodeMirror() + .setToolbar() + .loadedDisplay() + } else { + this.loadQueues() + } + return this + }, + + /** + * 所需组件加载队列 + * Required components loading queue + * + * @returns {editormd} 返回editormd的实例对象 + */ + + loadQueues: function() { + var _this = this + var settings = this.settings + var loadPath = settings.path + var loadFlowChartOrSequenceDiagram = function() { + if (editormd.isIE8) { + _this.loadedDisplay() + return + } + if (settings.flowChart || settings.sequenceDiagram) { + editormd.loadScript(loadPath + "raphael.min", function() { + editormd.loadScript(loadPath + "underscore.min", function() { + if (!settings.flowChart && settings.sequenceDiagram) { + editormd.loadScript( + loadPath + "sequence-diagram.min", + function() { + _this.loadedDisplay() + } + ) + } else if (settings.flowChart && !settings.sequenceDiagram) { + editormd.loadScript(loadPath + "flowchart.min", function() { + editormd.loadScript( + loadPath + "jquery.flowchart.min", + function() { + _this.loadedDisplay() + } + ) + }) + } else if (settings.flowChart && settings.sequenceDiagram) { + editormd.loadScript(loadPath + "flowchart.min", function() { + editormd.loadScript( + loadPath + "jquery.flowchart.min", + function() { + editormd.loadScript( + loadPath + "sequence-diagram.min", + function() { + _this.loadedDisplay() + } + ) + } + ) + }) + } + }) + }) + } else { + _this.loadedDisplay() + } + } + + editormd.loadCSS(loadPath + "codemirror/codemirror.min") + if (settings.searchReplace && !settings.readOnly) { + editormd.loadCSS(loadPath + "codemirror/addon/dialog/dialog") + editormd.loadCSS( + loadPath + "codemirror/addon/search/matchesonscrollbar" + ) + } + if (settings.codeFold) { + editormd.loadCSS(loadPath + "codemirror/addon/fold/foldgutter") + } + editormd.loadScript(loadPath + "codemirror/codemirror.min", function() { + editormd.$CodeMirror = CodeMirror + editormd.loadScript(loadPath + "codemirror/modes.min", function() { + editormd.loadScript(loadPath + "codemirror/addons.min", function() { + _this.setCodeMirror() + if (settings.mode !== "gfm" && settings.mode !== "markdown") { + _this.loadedDisplay() + return false + } + _this.setToolbar() + editormd.loadScript(loadPath + "marked.min", function() { + editormd.$marked = marked + + if (settings.previewCodeHighlight) { + editormd.loadScript(loadPath + "prettify.min", function() { + loadFlowChartOrSequenceDiagram() + }) + } else { + loadFlowChartOrSequenceDiagram() + } + }) + }) + }) + }) + return this + }, + + /** + * 设置 Editor.md 的整体主题,主要是工具栏 + * Setting Editor.md theme + * + * @returns {editormd} 返回editormd的实例对象 + */ + + setTheme: function(theme) { + var editor = this.editor + var oldTheme = this.settings.theme + var themePrefix = this.classPrefix + "theme-" + editor.removeClass(themePrefix + oldTheme).addClass(themePrefix + theme) + this.settings.theme = theme + return this + }, + + /** + * 设置 CodeMirror(编辑区)的主题 + * Setting CodeMirror (Editor area) theme + * + * @returns {editormd} 返回editormd的实例对象 + */ + + setEditorTheme: function(theme) { + var settings = this.settings + settings.editorTheme = theme + if (theme !== "default") { + editormd.loadCSS( + settings.path + "codemirror/theme/" + settings.editorTheme + ) + } + this.cm.setOption("theme", theme) + return this + }, + + /** + * setEditorTheme() 的别名 + * setEditorTheme() alias + * + * @returns {editormd} 返回editormd的实例对象 + */ + + setCodeMirrorTheme: function(theme) { + this.setEditorTheme(theme) + return this + }, + + /** + * 设置 Editor.md 的主题 + * Setting Editor.md theme + * + * @returns {editormd} 返回editormd的实例对象 + */ + + setPreviewTheme: function(theme) { + var preview = this.preview + var oldTheme = this.settings.previewTheme + var themePrefix = this.classPrefix + "preview-theme-" + preview.removeClass(themePrefix + oldTheme).addClass(themePrefix + theme) + this.settings.previewTheme = theme + return this + }, + + /** + * 配置和初始化CodeMirror组件 + * CodeMirror initialization + * + * @returns {editormd} 返回editormd的实例对象 + */ + + setCodeMirror: function() { + var settings = this.settings + var editor = this.editor + if (settings.editorTheme !== "default") { + editormd.loadCSS( + settings.path + "codemirror/theme/" + settings.editorTheme + ) + } + var codeMirrorConfig = { + mode: settings.mode, + theme: settings.editorTheme, + tabSize: settings.tabSize, + dragDrop: false, + autofocus: settings.autoFocus, + autoCloseTags: settings.autoCloseTags, + readOnly: settings.readOnly ? "nocursor" : false, + indentUnit: settings.indentUnit, + lineNumbers: settings.lineNumbers, + lineWrapping: settings.lineWrapping, + extraKeys: { + "Ctrl-Q": function(cm) { + cm.foldCode(cm.getCursor()) + } + }, + foldGutter: settings.codeFold, + gutters: ["CodeMirror-linenumbers", "CodeMirror-foldgutter"], + matchBrackets: settings.matchBrackets, + indentWithTabs: settings.indentWithTabs, + styleActiveLine: settings.styleActiveLine, + styleSelectedText: settings.styleSelectedText, + autoCloseBrackets: settings.autoCloseBrackets, + showTrailingSpace: settings.showTrailingSpace, + highlightSelectionMatches: !settings.matchWordHighlight + ? false + : { + showToken: + settings.matchWordHighlight === "onselected" ? false : /\w/ + } + } + + this.codeEditor = this.cm = editormd.$CodeMirror.fromTextArea( + this.markdownTextarea[0], + codeMirrorConfig + ) + this.codeMirror = this.cmElement = editor.children(".CodeMirror") + + if (settings.value !== "") { + this.cm.setValue(settings.value) + } + + this.codeMirror.css({ + fontSize: settings.fontSize, + width: !settings.watch ? "100%" : "50%" + }) + + if (settings.autoHeight) { + this.codeMirror.css({ + height: "auto", + minHeight: settings.minHeight, + maxHeight: settings.maxHeight + }) + this.codeMirror.find(".CodeMirror-gutters").css({ + minHeight: settings.minHeight + }) + this.codeMirror.find(".CodeMirror-scroll").css({ + minHeight: settings.minHeight, + maxHeight: settings.maxHeight + }) + this.cm.setOption("viewportMargin", Infinity) + } + + if (!settings.lineNumbers) { + this.codeMirror.find(".CodeMirror-gutters").css("border-right", "none") + } + + return this + }, + + /** + * 获取CodeMirror的配置选项 + * Get CodeMirror setting options + * + * @returns {Mixed} return CodeMirror setting option value + */ + + getCodeMirrorOption: function(key) { + return this.cm.getOption(key) + }, + + /** + * 配置和重配置CodeMirror的选项 + * CodeMirror setting options / resettings + * + * @returns {editormd} 返回editormd的实例对象 + */ + + setCodeMirrorOption: function(key, value) { + this.cm.setOption(key, value) + + return this + }, + + /** + * 添加 CodeMirror 键盘快捷键 + * Add CodeMirror keyboard shortcuts key map + * + * @returns {editormd} 返回editormd的实例对象 + */ + + addKeyMap: function(map, bottom) { + this.cm.addKeyMap(map, bottom) + + return this + }, + + /** + * 移除 CodeMirror 键盘快捷键 + * Remove CodeMirror keyboard shortcuts key map + * + * @returns {editormd} 返回editormd的实例对象 + */ + + removeKeyMap: function(map) { + this.cm.removeKeyMap(map) + + return this + }, + + /** + * 跳转到指定的行 + * Goto CodeMirror line + * + * @param {String|Intiger} line line number or "first"|"last" + * @returns {editormd} 返回editormd的实例对象 + */ + + gotoLine: function(line) { + var settings = this.settings + if (!settings.gotoLine) { + return this + } + var cm = this.cm + var editor = this.editor + var count = cm.lineCount() + var preview = this.preview + if (typeof line === "string") { + if (line === "last") { + line = count + } + if (line === "first") { + line = 1 + } + } + if (typeof line !== "number") { + alert("Error: The line number must be an integer.") + return this + } + line = parseInt(line) - 1 + if (line > count) { + alert("Error: The line number range 1-" + count) + return this + } + cm.setCursor({ line: line, ch: 0 }) + var scrollInfo = cm.getScrollInfo() + var clientHeight = scrollInfo.clientHeight + var coords = cm.charCoords({ line: line, ch: 0 }, "local") + cm.scrollTo(null, (coords.top + coords.bottom - clientHeight) / 2) + if (settings.watch) { + var cmScroll = this.codeMirror.find(".CodeMirror-scroll")[0] + var height = $(cmScroll).height() + var scrollTop = cmScroll.scrollTop + var percent = scrollTop / cmScroll.scrollHeight + if (scrollTop === 0) { + preview.scrollTop(0) + } else if (scrollTop + height >= cmScroll.scrollHeight - 16) { + preview.scrollTop(preview[0].scrollHeight) + } else { + preview.scrollTop(preview[0].scrollHeight * percent) + } + } + cm.focus() + return this + }, + + /** + * 扩展当前实例对象,可同时设置多个或者只设置一个 + * Extend editormd instance object, can mutil setting. + * + * @returns {editormd} this(editormd instance object.) + */ + + extend: function() { + if (typeof arguments[1] !== "undefined") { + if (typeof arguments[1] === "function") { + arguments[1] = $.proxy(arguments[1], this) + } + this[arguments[0]] = arguments[1] + } + if ( + typeof arguments[0] === "object" && + typeof arguments[0].length === "undefined" + ) { + $.extend(true, this, arguments[0]) + } + return this + }, + + /** + * 设置或扩展当前实例对象,单个设置 + * Extend editormd instance object, one by one + * + * @param {String|Object} key option key + * @param {String|Object} value option value + * @returns {editormd} this(editormd instance object.) + */ + + set: function(key, value) { + if (typeof value !== "undefined" && typeof value === "function") { + value = $.proxy(value, this) + } + this[key] = value + return this + }, + + /** + * 重新配置 + * Resetting editor options + * + * @param {String|Object} key option key + * @param {String|Object} value option value + * @returns {editormd} this(editormd instance object.) + */ + + config: function(key, value) { + var settings = this.settings + if (typeof key === "object") { + settings = $.extend(true, settings, key) + } + if (typeof key === "string") { + settings[key] = value + } + this.settings = settings + this.recreate() + return this + }, + + /** + * 注册事件处理方法 + * Bind editor event handle + * + * @param {String} eventType event type + * @param {Function} callback 回调函数 + * @returns {editormd} this(editormd instance object.) + */ + + on: function(eventType, callback) { + var settings = this.settings + if (typeof settings["on" + eventType] !== "undefined") { + settings["on" + eventType] = $.proxy(callback, this) + } + return this + }, + + /** + * 解除事件处理方法 + * Unbind editor event handle + * + * @param {String} eventType event type + * @returns {editormd} this(editormd instance object.) + */ + + off: function(eventType) { + var settings = this.settings + if (typeof settings["on" + eventType] !== "undefined") { + settings["on" + eventType] = function() {} + } + return this + }, + + /** + * 显示工具栏 + * Display toolbar + * + * @param {Function} [callback=function(){}] 回调函数 + * @returns {editormd} 返回editormd的实例对象 + */ + + showToolbar: function(callback) { + var settings = this.settings + if (settings.readOnly) { + return this + } + if ( + settings.toolbar && + (this.toolbar.length < 1 || + this.toolbar.find("." + this.classPrefix + "menu").html() === "") + ) { + this.setToolbar() + } + settings.toolbar = true + this.toolbar.show() + this.resize() + $.proxy(callback || function() {}, this)() + return this + }, + + /** + * 隐藏工具栏 + * Hide toolbar + * + * @param {Function} [callback=function(){}] 回调函数 + * @returns {editormd} this(editormd instance object.) + */ + + hideToolbar: function(callback) { + var settings = this.settings + settings.toolbar = false + this.toolbar.hide() + this.resize() + $.proxy(callback || function() {}, this)() + return this + }, + + /** + * 页面滚动时工具栏的固定定位 + * Set toolbar in window scroll auto fixed position + * + * @returns {editormd} 返回editormd的实例对象 + */ + + setToolbarAutoFixed: function(fixed) { + var state = this.state + var editor = this.editor + var toolbar = this.toolbar + var settings = this.settings + if (typeof fixed !== "undefined") { + settings.toolbarAutoFixed = fixed + } + var autoFixedHandle = function() { + var $window = $(window) + var top = $window.scrollTop() + if (!settings.toolbarAutoFixed) { + return false + } + if (top - editor.offset().top > 10 && top < editor.height()) { + toolbar.css({ + position: "fixed", + width: editor.width() + "px", + left: ($window.width() - editor.width()) / 2 + "px" + }) + } else { + toolbar.css({ + position: "absolute", + width: "100%", + left: 0 + }) + } + } + if ( + !state.fullscreen && + !state.preview && + settings.toolbar && + settings.toolbarAutoFixed + ) { + $(window).bind("scroll", autoFixedHandle) + } + return this + }, + + /** + * 配置和初始化工具栏 + * Set toolbar and Initialization + * + * @returns {editormd} 返回editormd的实例对象 + */ + + setToolbar: function() { + var settings = this.settings + if (settings.readOnly) { + return this + } + var editor = this.editor + var preview = this.preview + var classPrefix = this.classPrefix + var toolbar = (this.toolbar = editor.children( + "." + classPrefix + "toolbar" + )) + if (settings.toolbar && toolbar.length < 1) { + var toolbarHTML = + '
              ' + editor.append(toolbarHTML) + toolbar = this.toolbar = editor.children("." + classPrefix + "toolbar") + } + if (!settings.toolbar) { + toolbar.hide() + return this + } + toolbar.show() + var icons = + typeof settings.toolbarIcons === "function" + ? settings.toolbarIcons() + : typeof settings.toolbarIcons === "string" + ? editormd.toolbarModes[settings.toolbarIcons] + : settings.toolbarIcons + var toolbarMenu = toolbar.find("." + this.classPrefix + "menu"), + menu = "" + var pullRight = false + for (var i = 0, len = icons.length; i < len; i++) { + var name = icons[i] + if (name === "||") { + pullRight = true + } else if (name === "|") { + menu += '
            • |
            • ' + } else { + var isHeader = /h(\d)/.test(name) + var index = name + if (name === "watch" && !settings.watch) { + index = "unwatch" + } + var title = settings.lang.toolbar[index] + var iconTexts = settings.toolbarIconTexts[index] + var iconClass = settings.toolbarIconsClass[index] + title = typeof title === "undefined" ? "" : title + iconTexts = typeof iconTexts === "undefined" ? "" : iconTexts + iconClass = typeof iconClass === "undefined" ? "" : iconClass + var menuItem = pullRight ? '
            • ' : "
            • " + if ( + typeof settings.toolbarCustomIcons[name] !== "undefined" && + typeof settings.toolbarCustomIcons[name] !== "function" + ) { + menuItem += settings.toolbarCustomIcons[name] + } else { + menuItem += + '' + menuItem += + '' + + (isHeader + ? name.toUpperCase() + : iconClass === "" ? iconTexts : "") + + "" + menuItem += "" + } + menuItem += "
            • " + menu = pullRight ? menuItem + menu : menu + menuItem + } + } + toolbarMenu.html(menu) + toolbarMenu + .find('[title="Lowercase"]') + .attr("title", settings.lang.toolbar.lowercase) + toolbarMenu + .find('[title="ucwords"]') + .attr("title", settings.lang.toolbar.ucwords) + this.setToolbarHandler() + this.setToolbarAutoFixed() + return this + }, + + /** + * 工具栏图标事件处理对象序列 + * Get toolbar icons event handlers + * + * @param {Object} cm CodeMirror的实例对象 + * @param {String} name 要获取的事件处理器名称 + * @returns {Object} 返回处理对象序列 + */ + + dialogLockScreen: function() { + $.proxy(editormd.dialogLockScreen, this)() + return this + }, + dialogShowMask: function(dialog) { + $.proxy(editormd.dialogShowMask, this)(dialog) + return this + }, + getToolbarHandles: function(name) { + var toolbarHandlers = (this.toolbarHandlers = editormd.toolbarHandlers) + return name && typeof toolbarIconHandlers[name] !== "undefined" + ? toolbarHandlers[name] + : toolbarHandlers + }, + + /** + * 工具栏图标事件处理器 + * Bind toolbar icons event handle + * + * @returns {editormd} 返回editormd的实例对象 + */ + + setToolbarHandler: function() { + var _this = this + var settings = this.settings + if (!settings.toolbar || settings.readOnly) { + return this + } + var toolbar = this.toolbar + var cm = this.cm + var classPrefix = this.classPrefix + var toolbarIcons = (this.toolbarIcons = toolbar.find( + "." + classPrefix + "menu > li > a" + )) + var toolbarIconHandlers = this.getToolbarHandles() + toolbarIcons.bind(editormd.mouseOrTouch("click", "touchend"), function( + event + ) { + var icon = $(this).children(".fa") + var name = icon.attr("name") + var cursor = cm.getCursor() + var selection = cm.getSelection() + if (name === "") { + return + } + _this.activeIcon = icon + if (typeof toolbarIconHandlers[name] !== "undefined") { + $.proxy(toolbarIconHandlers[name], _this)(cm) + } else { + if (typeof settings.toolbarHandlers[name] !== "undefined") { + $.proxy(settings.toolbarHandlers[name], _this)( + cm, + icon, + cursor, + selection + ) + } + } + if ( + name !== "link" && + name !== "reference-link" && + name !== "image" && + name !== "code-block" && + name !== "preformatted-text" && + name !== "watch" && + name !== "preview" && + name !== "search" && + name !== "fullscreen" && + name !== "info" + ) { + cm.focus() + } + return false + }) + + return this + }, + + /** + * 动态创建对话框 + * Creating custom dialogs + * + * @param {Object} options 配置项键值对 Key/Value + * @returns {dialog} 返回创建的dialog的jQuery实例对象 + */ + + createDialog: function(options) { + return $.proxy(editormd.createDialog, this)(options) + }, + + /** + * 创建关于Editor.md的对话框 + * Create about Editor.md dialog + * + * @returns {editormd} 返回editormd的实例对象 + */ + + createInfoDialog: function() { + var _this = this + var editor = this.editor + var classPrefix = this.classPrefix + var infoDialogHTML = [ + '
              ', + '
              ', + '

              ' + + editormd.title + + "v" + + editormd.version + + "

              ", + "

              " + this.lang.description + "

              ", + '

              ' + + editormd.homePage + + '

              ', + '

              Copyright © 2015 Pandao, The MIT License.

              ', + "
              ", + '', + "
              " + ].join("\n") + editor.append(infoDialogHTML) + var infoDialog = (this.infoDialog = editor.children( + "." + classPrefix + "dialog-info" + )) + infoDialog + .find("." + classPrefix + "dialog-close") + .bind(editormd.mouseOrTouch("click", "touchend"), function() { + _this.hideInfoDialog() + }) + infoDialog + .css("border", editormd.isIE8 ? "1px solid #ddd" : "") + .css("z-index", editormd.dialogZindex) + .show() + this.infoDialogPosition() + return this + }, + + /** + * 关于Editor.md对话居中定位 + * Editor.md dialog position handle + * + * @returns {editormd} 返回editormd的实例对象 + */ + + infoDialogPosition: function() { + var infoDialog = this.infoDialog + var _infoDialogPosition = function() { + infoDialog.css({ + top: ($(window).height() - infoDialog.height()) / 2 + "px", + left: ($(window).width() - infoDialog.width()) / 2 + "px" + }) + } + _infoDialogPosition() + $(window).resize(_infoDialogPosition) + return this + }, + + /** + * 显示关于Editor.md + * Display about Editor.md dialog + * + * @returns {editormd} 返回editormd的实例对象 + */ + + showInfoDialog: function() { + $("html,body").css("overflow-x", "hidden") + var _this = this + var editor = this.editor + var settings = this.settings + var infoDialog = (this.infoDialog = editor.children( + "." + this.classPrefix + "dialog-info" + )) + if (infoDialog.length < 1) { + this.createInfoDialog() + } + this.lockScreen(true) + this.mask + .css({ + opacity: settings.dialogMaskOpacity, + backgroundColor: settings.dialogMaskBgColor + }) + .show() + infoDialog.css("z-index", editormd.dialogZindex).show() + this.infoDialogPosition() + return this + }, + + /** + * 隐藏关于Editor.md + * Hide about Editor.md dialog + * + * @returns {editormd} 返回editormd的实例对象 + */ + + hideInfoDialog: function() { + $("html,body").css("overflow-x", "") + this.infoDialog.hide() + this.mask.hide() + this.lockScreen(false) + return this + }, + + /** + * 锁屏 + * lock screen + * + * @param {Boolean} lock Boolean 布尔值,是否锁屏 + * @returns {editormd} 返回editormd的实例对象 + */ + + lockScreen: function(lock) { + editormd.lockScreen(lock) + this.resize() + return this + }, + + /** + * 编辑器界面重建,用于动态语言包或模块加载等 + * Recreate editor + * + * @returns {editormd} 返回editormd的实例对象 + */ + + recreate: function() { + var _this = this + var editor = this.editor + var settings = this.settings + this.codeMirror.remove() + this.setCodeMirror() + if (!settings.readOnly) { + if (editor.find(".editormd-dialog").length > 0) { + editor.find(".editormd-dialog").remove() + } + if (settings.toolbar) { + this.getToolbarHandles() + this.setToolbar() + } + } + this.loadedDisplay(true) + return this + }, + + /** + * 高亮预览HTML的pre代码部分 + * highlight of preview codes + * + * @returns {editormd} 返回editormd的实例对象 + */ + + previewCodeHighlight: function() { + var settings = this.settings + var previewContainer = this.previewContainer + if (settings.previewCodeHighlight) { + previewContainer.find("pre").addClass("prettyprint linenums") + if (typeof prettyPrint !== "undefined") { + prettyPrint() + } + } + + return this + }, + + /** + * 解析TeX(KaTeX)科学公式 + * TeX(KaTeX) Renderer + * + * @returns {editormd} 返回editormd的实例对象 + */ + + katexRender: function() { + if (timer === null) { + return this + } + this.previewContainer + .find("." + editormd.classNames.tex) + .each(function() { + var tex = $(this) + editormd.$katex.render(tex.text(), tex[0]) + tex.find(".katex").css("font-size", "1.6em") + }) + return this + }, + + /** + * 解析和渲染流程图及时序图 + * FlowChart and SequenceDiagram Renderer + * + * @returns {editormd} 返回editormd的实例对象 + */ + + flowChartAndSequenceDiagramRender: function() { + var $this = this + var settings = this.settings + var previewContainer = this.previewContainer + if (editormd.isIE8) { + return this + } + if (settings.flowChart) { + if (flowchartTimer === null) { + return this + } + previewContainer.find(".flowchart").flowChart() + } + + if (settings.sequenceDiagram) { + previewContainer + .find(".sequence-diagram") + .sequenceDiagram({ theme: "simple" }) + } + var preview = $this.preview + var codeMirror = $this.codeMirror + var codeView = codeMirror.find(".CodeMirror-scroll") + var height = codeView.height() + var scrollTop = codeView.scrollTop() + var percent = scrollTop / codeView[0].scrollHeight + var tocHeight = 0 + preview.find(".markdown-toc-list").each(function() { + tocHeight += $(this).height() + }) + var tocMenuHeight = preview.find(".editormd-toc-menu").height() + tocMenuHeight = !tocMenuHeight ? 0 : tocMenuHeight + if (settings.syncScrolling) { + if (scrollTop === 0) { + preview.scrollTop(0) + } else if (scrollTop + height >= codeView[0].scrollHeight - 16) { + preview.scrollTop(preview[0].scrollHeight) + } else { + preview.scrollTop( + (preview[0].scrollHeight + tocHeight + tocMenuHeight) * percent + ) + } + } + return this + }, + + /** + * 注册键盘快捷键处理 + * Register CodeMirror keyMaps (keyboard shortcuts). + * + * @param {Object} keyMap KeyMap key/value {"(Ctrl/Shift/Alt)-Key" : function(){}} + * @returns {editormd} return this + */ + + registerKeyMaps: function(keyMap) { + var _this = this + var cm = this.cm + var settings = this.settings + var toolbarHandlers = editormd.toolbarHandlers + var disabledKeyMaps = settings.disabledKeyMaps + keyMap = keyMap || null + if (keyMap) { + for (var i in keyMap) { + if ($.inArray(i, disabledKeyMaps) < 0) { + var map = {} + map[i] = keyMap[i] + cm.addKeyMap(keyMap) + } + } + } else { + for (var k in editormd.keyMaps) { + var _keyMap = editormd.keyMaps[k] + var handle = + typeof _keyMap === "string" + ? $.proxy(toolbarHandlers[_keyMap], _this) + : $.proxy(_keyMap, _this) + if ( + $.inArray(k, ["F9", "F10", "F11"]) < 0 && + $.inArray(k, disabledKeyMaps) < 0 + ) { + var _map = {} + _map[k] = handle + cm.addKeyMap(_map) + } + } + $(window).keydown(function(event) { + var keymaps = { + "120": "F9", + "121": "F10", + "122": "F11" + } + if ($.inArray(keymaps[event.keyCode], disabledKeyMaps) < 0) { + switch (event.keyCode) { + case 120: + $.proxy(toolbarHandlers.watch, _this)() + return false + case 121: + $.proxy(toolbarHandlers.preview, _this)() + return false + case 122: + $.proxy(toolbarHandlers.fullscreen, _this)() + return false + default: + break + } + } + }) + } + return this + }, + + /** + * 绑定同步滚动 + * @returns {editormd} return this + */ + + bindScrollEvent: function() { + var _this = this + var preview = this.preview + var settings = this.settings + var codeMirror = this.codeMirror + var mouseOrTouch = editormd.mouseOrTouch + if (!settings.syncScrolling) { + return this + } + var cmBindScroll = function() { + codeMirror + .find(".CodeMirror-scroll") + .bind(mouseOrTouch("scroll", "touchmove"), function(event) { + var height = $(this).height() + var scrollTop = $(this).scrollTop() + var percent = scrollTop / $(this)[0].scrollHeight + var tocHeight = 0 + preview.find(".markdown-toc-list").each(function() { + tocHeight += $(this).height() + }) + var tocMenuHeight = preview.find(".editormd-toc-menu").height() + tocMenuHeight = !tocMenuHeight ? 0 : tocMenuHeight + if (scrollTop === 0) { + preview.scrollTop(0) + } else if (scrollTop + height >= $(this)[0].scrollHeight - 16) { + preview.scrollTop(preview[0].scrollHeight) + } else { + preview.scrollTop( + (preview[0].scrollHeight + tocHeight + tocMenuHeight) * percent + ) + } + $.proxy(settings.onscroll, _this)(event) + }) + } + var cmUnbindScroll = function() { + codeMirror + .find(".CodeMirror-scroll") + .unbind(mouseOrTouch("scroll", "touchmove")) + } + var previewBindScroll = function() { + preview.bind(mouseOrTouch("scroll", "touchmove"), function(event) { + var height = $(this).height() + var scrollTop = $(this).scrollTop() + var percent = scrollTop / $(this)[0].scrollHeight + var codeView = codeMirror.find(".CodeMirror-scroll") + if (scrollTop === 0) { + codeView.scrollTop(0) + } else if (scrollTop + height >= $(this)[0].scrollHeight) { + codeView.scrollTop(codeView[0].scrollHeight) + } else { + codeView.scrollTop(codeView[0].scrollHeight * percent) + } + $.proxy(settings.onpreviewscroll, _this)(event) + }) + } + + var previewUnbindScroll = function() { + preview.unbind(mouseOrTouch("scroll", "touchmove")) + } + codeMirror.bind({ + mouseover: cmBindScroll, + mouseout: cmUnbindScroll, + touchstart: cmBindScroll, + touchend: cmUnbindScroll + }) + if (settings.syncScrolling === "single") { + return this + } + preview.bind({ + mouseover: previewBindScroll, + mouseout: previewUnbindScroll, + touchstart: previewBindScroll, + touchend: previewUnbindScroll + }) + return this + }, + + bindChangeEvent: function() { + var _this = this + var cm = this.cm + var settings = this.settings + + // if (!settings.syncScrolling) { + // return this; + // } + + cm.on("change", function(_cm, changeObj) { + if (settings.watch) { + _this.previewContainer.css( + "padding", + settings.autoHeight ? "20px 20px 50px 40px" : "20px" + ) + } + timer = setTimeout(function() { + clearTimeout(timer) + _this.save() + timer = null + }, settings.delay) + }) + return this + }, + + /** + * 加载队列完成之后的显示处理 + * Display handle of the module queues loaded after. + * + * @param {Boolean} recreate 是否为重建编辑器 + * @returns {editormd} 返回editormd的实例对象 + */ + + loadedDisplay: function(recreate) { + recreate = recreate || false + var _this = this + var editor = this.editor + var preview = this.preview + var settings = this.settings + this.containerMask.hide() + this.save() + if (settings.watch) { + preview.show() + } + editor.data("oldWidth", editor.width()).data("oldHeight", editor.height()) // 为了兼容Zepto + this.resize() + this.registerKeyMaps() + $(window).resize(function() { + _this.resize() + }) + this.bindScrollEvent().bindChangeEvent() + if (!recreate) { + $.proxy(settings.onload, this)() + } + this.state.loaded = true + return this + }, + + /** + * 设置编辑器的宽度 + * Set editor width + * + * @param {Number|String} width 编辑器宽度值 + * @returns {editormd} 返回editormd的实例对象 + */ + + width: function(width) { + this.editor.css("width", typeof width === "number" ? width + "px" : width) + this.resize() + return this + }, + + /** + * 设置编辑器的高度 + * Set editor height + * + * @param {Number|String} height 编辑器高度值 + * @returns {editormd} 返回editormd的实例对象 + */ + + height: function(height) { + this.editor.css( + "height", + typeof height === "number" ? height + "px" : height + ) + this.resize() + return this + }, + + /** + * 调整编辑器的尺寸和布局 + * Resize editor layout + * + * @param {Number|String} [width=null] 编辑器宽度值 + * @param {Number|String} [height=null] 编辑器高度值 + * @returns {editormd} 返回editormd的实例对象 + */ + + resize: function(width, height) { + width = width || null + height = height || null + var state = this.state + var editor = this.editor + var preview = this.preview + var toolbar = this.toolbar + var settings = this.settings + var codeMirror = this.codeMirror + if (width) { + editor.css("width", typeof width === "number" ? width + "px" : width) + } + if (settings.autoHeight && !state.fullscreen && !state.preview) { + editor.css({ + height: "auto", + minHeight: settings.minHeight + }) + codeMirror.css({ + height: "auto", + minHeight: settings.minHeight, + maxHeight: settings.maxHeight + }) + } else { + if (height) { + editor.css( + "height", + typeof height === "number" ? height + "px" : height + ) + } + if (state.fullscreen) { + editor.height($(window).height()) + } + if (settings.toolbar && !settings.readOnly) { + codeMirror + .css("margin-top", toolbar.height() + 1) + .height(editor.height() - toolbar.height()) + } else { + codeMirror.css("margin-top", 0).height(editor.height()) + } + } + if (settings.watch) { + codeMirror.width(editor.width() / 2) + preview.width(!state.preview ? editor.width() / 2 : editor.width()) + this.previewContainer.css( + "padding", + settings.autoHeight ? "20px 20px 50px 40px" : "20px" + ) + if (settings.toolbar && !settings.readOnly) { + preview.css("top", toolbar.height() + 1) + } else { + preview.css("top", 0) + } + if (settings.autoHeight && !state.fullscreen && !state.preview) { + preview.height("") + } else { + var previewHeight = + settings.toolbar && !settings.readOnly + ? editor.height() - toolbar.height() + : editor.height() + preview.height(previewHeight) + } + } else { + codeMirror.width(editor.width()) + preview.hide() + } + if (state.loaded) { + $.proxy(settings.onresize, this)() + } + return this + }, + + /** + * 解析和保存Markdown代码 + * Parse & Saving Markdown source code + * + * @returns {editormd} 返回editormd的实例对象 + */ + + save: function() { + if (timer === null) { + return this + } + var _this = this + var state = this.state + var settings = this.settings + var cm = this.cm + var cmValue = cm.getValue() + var previewContainer = this.previewContainer + if (settings.mode !== "gfm" && settings.mode !== "markdown") { + this.markdownTextarea.val(cmValue) + return this + } + var marked = editormd.$marked + var markdownToC = (this.markdownToC = []) + var rendererOptions = (this.markedRendererOptions = { + toc: settings.toc, + tocm: settings.tocm, + tocStartLevel: settings.tocStartLevel, + pageBreak: settings.pageBreak, + taskList: settings.taskList, + emoji: settings.emoji, + tex: settings.tex, + atLink: settings.atLink, // for @link + emailLink: settings.emailLink, // for mail address auto link + flowChart: settings.flowChart, + sequenceDiagram: settings.sequenceDiagram, + previewCodeHighlight: settings.previewCodeHighlight + }) + var markedOptions = (this.markedOptions = { + renderer: editormd.markedRenderer(markdownToC, rendererOptions), + gfm: true, + tables: true, + breaks: true, + pedantic: false, + sanitize: settings.htmlDecode ? false : true, // 关闭忽略HTML标签,即开启识别HTML标签,默认为false + smartLists: true, + smartypants: true + }) + marked.setOptions(markedOptions) + var newMarkdownDoc = editormd.$marked(cmValue, markedOptions) + newMarkdownDoc = editormd.filterHTMLTags( + newMarkdownDoc, + settings.htmlDecode + ) + this.markdownTextarea.text(cmValue) + cm.save() + if (settings.saveHTMLToTextarea) { + this.htmlTextarea.text(newMarkdownDoc) + } + if (settings.watch || (!settings.watch && state.preview)) { + previewContainer.html(newMarkdownDoc) + this.previewCodeHighlight() + if (settings.toc) { + var tocContainer = + settings.tocContainer === "" + ? previewContainer + : $(settings.tocContainer) + var tocMenu = tocContainer.find("." + this.classPrefix + "toc-menu") + tocContainer.attr( + "previewContainer", + settings.tocContainer === "" ? "true" : "false" + ) + if (settings.tocContainer !== "" && tocMenu.length > 0) { + tocMenu.remove() + } + editormd.markdownToCRenderer( + markdownToC, + tocContainer, + settings.tocDropdown, + settings.tocStartLevel + ) + if ( + settings.tocDropdown || + tocContainer.find("." + this.classPrefix + "toc-menu").length > 0 + ) { + editormd.tocDropdownMenu( + tocContainer, + settings.tocTitle !== "" ? settings.tocTitle : this.lang.tocTitle + ) + } + if (settings.tocContainer !== "") { + previewContainer.find(".markdown-toc").css("border", "none") + } + } + if (settings.tex) { + if (!editormd.kaTeXLoaded && settings.autoLoadModules) { + editormd.loadKaTeX(function() { + editormd.$katex = katex + editormd.kaTeXLoaded = true + _this.katexRender() + }) + } else { + editormd.$katex = katex + this.katexRender() + } + } + if (settings.flowChart || settings.sequenceDiagram) { + flowchartTimer = setTimeout(function() { + clearTimeout(flowchartTimer) + _this.flowChartAndSequenceDiagramRender() + flowchartTimer = null + }, 10) + } + if (state.loaded) { + $.proxy(settings.onchange, this)() + } + } + return this + }, + + /** + * 聚焦光标位置 + * Focusing the cursor position + * + * @returns {editormd} 返回editormd的实例对象 + */ + + focus: function() { + this.cm.focus() + return this + }, + + /** + * 设置光标的位置 + * Set cursor position + * + * @param {Object} cursor 要设置的光标位置键值对象,例:{line:1, ch:0} + * @returns {editormd} 返回editormd的实例对象 + */ + + setCursor: function(cursor) { + this.cm.setCursor(cursor) + return this + }, + + /** + * 获取当前光标的位置 + * Get the current position of the cursor + * + * @returns {Cursor} 返回一个光标Cursor对象 + */ + + getCursor: function() { + return this.cm.getCursor() + }, + + /** + * 设置光标选中的范围 + * Set cursor selected ranges + * + * @param {Object} from 开始位置的光标键值对象,例:{line:1, ch:0} + * @param {Object} to 结束位置的光标键值对象,例:{line:1, ch:0} + * @returns {editormd} 返回editormd的实例对象 + */ + + setSelection: function(from, to) { + this.cm.setSelection(from, to) + return this + }, + + /** + * 获取光标选中的文本 + * Get the texts from cursor selected + * + * @returns {String} 返回选中文本的字符串形式 + */ + + getSelection: function() { + return this.cm.getSelection() + }, + + /** + * 设置光标选中的文本范围 + * Set the cursor selection ranges + * + * @param {Array} ranges cursor selection ranges array + * @returns {Array} return this + */ + + setSelections: function(ranges) { + this.cm.setSelections(ranges) + return this + }, + + /** + * 获取光标选中的文本范围 + * Get the cursor selection ranges + * + * @returns {Array} return selection ranges array + */ + + getSelections: function() { + return this.cm.getSelections() + }, + + /** + * 替换当前光标选中的文本或在当前光标处插入新字符 + * Replace the text at the current cursor selected or insert a new character at the current cursor position + * + * @param {String} value 要插入的字符值 + * @returns {editormd} 返回editormd的实例对象 + */ + + replaceSelection: function(value) { + this.cm.replaceSelection(value) + return this + }, + + /** + * 在当前光标处插入新字符 + * Insert a new character at the current cursor position + * + * 同replaceSelection()方法 + * With the replaceSelection() method + * + * @param {String} value 要插入的字符值 + * @returns {editormd} 返回editormd的实例对象 + */ + + insertValue: function(value) { + this.replaceSelection(value) + return this + }, + + /** + * 追加markdown + * append Markdown to editor + * + * @param {String} md 要追加的markdown源文档 + * @returns {editormd} 返回editormd的实例对象 + */ + + appendMarkdown: function(md) { + var settings = this.settings + var cm = this.cm + cm.setValue(cm.getValue() + md) + return this + }, + + /** + * 设置和传入编辑器的markdown源文档 + * Set Markdown source document + * + * @param {String} md 要传入的markdown源文档 + * @returns {editormd} 返回editormd的实例对象 + */ + + setMarkdown: function(md) { + this.cm.setValue(md || this.settings.markdown) + return this + }, + + /** + * 获取编辑器的markdown源文档 + * Set Editor.md markdown/CodeMirror value + * + * @returns {editormd} 返回editormd的实例对象 + */ + + getMarkdown: function() { + return this.cm.getValue() + }, + + /** + * 获取编辑器的源文档 + * Get CodeMirror value + * + * @returns {editormd} 返回editormd的实例对象 + */ + + getValue: function() { + return this.cm.getValue() + }, + + /** + * 设置编辑器的源文档 + * Set CodeMirror value + * + * @param {String} value set code/value/string/text + * @returns {editormd} 返回editormd的实例对象 + */ + + setValue: function(value) { + this.cm.setValue(value) + return this + }, + + /** + * 清空编辑器 + * Empty CodeMirror editor container + * + * @returns {editormd} 返回editormd的实例对象 + */ + + clear: function() { + this.cm.setValue("") + return this + }, + + /** + * 获取解析后存放在Textarea的HTML源码 + * Get parsed html code from Textarea + * + * @returns {String} 返回HTML源码 + */ + + getHTML: function() { + if (!this.settings.saveHTMLToTextarea) { + alert("Error: settings.saveHTMLToTextarea == false") + return false + } + return this.htmlTextarea.val() + }, + + /** + * getHTML()的别名 + * getHTML (alias) + * + * @returns {String} Return html code 返回HTML源码 + */ + + getTextareaSavedHTML: function() { + return this.getHTML() + }, + + /** + * 获取预览窗口的HTML源码 + * Get html from preview container + * + * @returns {editormd} 返回editormd的实例对象 + */ + + getPreviewedHTML: function() { + if (!this.settings.watch) { + alert("Error: settings.watch == false") + return false + } + return this.previewContainer.html() + }, + + /** + * 开启实时预览 + * Enable real-time watching + * + * @returns {editormd} 返回editormd的实例对象 + */ + + watch: function(callback) { + var settings = this.settings + if ($.inArray(settings.mode, ["gfm", "markdown"]) < 0) { + return this + } + this.state.watching = settings.watch = true + this.preview.show() + if (this.toolbar) { + var watchIcon = settings.toolbarIconsClass.watch + var unWatchIcon = settings.toolbarIconsClass.unwatch + var icon = this.toolbar.find(".fa[name=watch]") + icon.parent().attr("title", settings.lang.toolbar.watch) + icon.removeClass(unWatchIcon).addClass(watchIcon) + } + this.codeMirror + .css("border-right", "1px solid #ddd") + .width(this.editor.width() / 2) + timer = 0 + this.save().resize() + if (!settings.onwatch) { + settings.onwatch = callback || function() {} + } + $.proxy(settings.onwatch, this)() + return this + }, + + /** + * 关闭实时预览 + * Disable real-time watching + * + * @returns {editormd} 返回editormd的实例对象 + */ + + unwatch: function(callback) { + var settings = this.settings + this.state.watching = settings.watch = false + this.preview.hide() + if (this.toolbar) { + var watchIcon = settings.toolbarIconsClass.watch + var unWatchIcon = settings.toolbarIconsClass.unwatch + var icon = this.toolbar.find(".fa[name=watch]") + icon.parent().attr("title", settings.lang.toolbar.unwatch) + icon.removeClass(watchIcon).addClass(unWatchIcon) + } + this.codeMirror.css("border-right", "none").width(this.editor.width()) + this.resize() + if (!settings.onunwatch) { + settings.onunwatch = callback || function() {} + } + $.proxy(settings.onunwatch, this)() + return this + }, + + /** + * 显示编辑器 + * Show editor + * + * @param {Function} [callback=function()] 回调函数 + * @returns {editormd} 返回editormd的实例对象 + */ + + show: function(callback) { + callback = callback || function() {} + var _this = this + this.editor.show(0, function() { + $.proxy(callback, _this)() + }) + return this + }, + + /** + * 隐藏编辑器 + * Hide editor + * + * @param {Function} [callback=function()] 回调函数 + * @returns {editormd} 返回editormd的实例对象 + */ + + hide: function(callback) { + callback = callback || function() {} + var _this = this + this.editor.hide(0, function() { + $.proxy(callback, _this)() + }) + return this + }, + + /** + * 隐藏编辑器部分,只预览HTML + * Enter preview html state + * + * @returns {editormd} 返回editormd的实例对象 + */ + + previewing: function() { + var _this = this + var editor = this.editor + var preview = this.preview + var toolbar = this.toolbar + var settings = this.settings + var codeMirror = this.codeMirror + var previewContainer = this.previewContainer + if ($.inArray(settings.mode, ["gfm", "markdown"]) < 0) { + return this + } + if (settings.toolbar && toolbar) { + toolbar.toggle() + toolbar.find(".fa[name=preview]").toggleClass("active") + } + codeMirror.toggle() + var escHandle = function(event) { + if (event.shiftKey && event.keyCode === 27) { + _this.previewed() + } + } + if (codeMirror.css("display") === "none") { + // 为了兼容Zepto,而不使用codeMirror.is(":hidden") + this.state.preview = true + if (this.state.fullscreen) { + preview.css("background", "#fff") + } + editor + .find("." + this.classPrefix + "preview-close-btn") + .show() + .bind(editormd.mouseOrTouch("click", "touchend"), function() { + _this.previewed() + }) + if (!settings.watch) { + this.save() + } else { + previewContainer.css("padding", "") + } + previewContainer.addClass(this.classPrefix + "preview-active") + preview.show().css({ + position: "", + top: 0, + width: editor.width(), + height: + settings.autoHeight && !this.state.fullscreen + ? "auto" + : editor.height() + }) + if (this.state.loaded) { + $.proxy(settings.onpreviewing, this)() + } + $(window).bind("keyup", escHandle) + } else { + $(window).unbind("keyup", escHandle) + this.previewed() + } + }, + + /** + * 显示编辑器部分,退出只预览HTML + * Exit preview html state + * + * @returns {editormd} 返回editormd的实例对象 + */ + + previewed: function() { + var editor = this.editor + var preview = this.preview + var toolbar = this.toolbar + var settings = this.settings + var previewContainer = this.previewContainer + var previewCloseBtn = editor.find( + "." + this.classPrefix + "preview-close-btn" + ) + this.state.preview = false + this.codeMirror.show() + if (settings.toolbar) { + toolbar.show() + } + preview[settings.watch ? "show" : "hide"]() + previewCloseBtn.hide().unbind(editormd.mouseOrTouch("click", "touchend")) + previewContainer.removeClass(this.classPrefix + "preview-active") + if (settings.watch) { + previewContainer.css("padding", "20px") + } + preview.css({ + background: null, + position: "absolute", + width: editor.width() / 2, + height: + settings.autoHeight && !this.state.fullscreen + ? "auto" + : editor.height() - toolbar.height(), + top: settings.toolbar ? toolbar.height() : 0 + }) + if (this.state.loaded) { + $.proxy(settings.onpreviewed, this)() + } + return this + }, + + /** + * 编辑器全屏显示 + * Fullscreen show + * + * @returns {editormd} 返回editormd的实例对象 + */ + + fullscreen: function() { + var _this = this + var state = this.state + var editor = this.editor + var preview = this.preview + var toolbar = this.toolbar + var settings = this.settings + var fullscreenClass = this.classPrefix + "fullscreen" + if (toolbar) { + toolbar + .find(".fa[name=fullscreen]") + .parent() + .toggleClass("active") + } + var escHandle = function(event) { + if (!event.shiftKey && event.keyCode === 27) { + if (state.fullscreen) { + _this.fullscreenExit() + } + } + } + if (!editor.hasClass(fullscreenClass)) { + state.fullscreen = true + $("html,body").css("overflow", "hidden") + editor + .css({ + width: $(window).width(), + height: $(window).height() + }) + .addClass(fullscreenClass) + this.resize() + $.proxy(settings.onfullscreen, this)() + $(window).bind("keyup", escHandle) + } else { + $(window).unbind("keyup", escHandle) + this.fullscreenExit() + } + return this + }, + + /** + * 编辑器退出全屏显示 + * Exit fullscreen state + * + * @returns {editormd} 返回editormd的实例对象 + */ + + fullscreenExit: function() { + var editor = this.editor + var settings = this.settings + var toolbar = this.toolbar + var fullscreenClass = this.classPrefix + "fullscreen" + this.state.fullscreen = false + if (toolbar) { + toolbar + .find(".fa[name=fullscreen]") + .parent() + .removeClass("active") + } + $("html,body").css("overflow", "") + editor + .css({ + width: editor.data("oldWidth"), + height: editor.data("oldHeight") + }) + .removeClass(fullscreenClass) + this.resize() + $.proxy(settings.onfullscreenExit, this)() + return this + }, + + /** + * 加载并执行插件 + * Load and execute the plugin + * + * @param {String} name plugin name / function name + * @param {String} path plugin load path + * @returns {editormd} 返回editormd的实例对象 + */ + + executePlugin: function(name, path) { + var _this = this + var cm = this.cm + var settings = this.settings + path = settings.pluginPath + path + if (typeof define === "function") { + if (typeof this[name] === "undefined") { + alert( + "Error: " + + name + + " plugin is not found, you are not load this plugin." + ) + return this + } + this[name](cm) + return this + } + if ($.inArray(path, editormd.loadFiles.plugin) < 0) { + editormd.loadPlugin(path, function() { + editormd.loadPlugins[name] = _this[name] + _this[name](cm) + }) + } else { + $.proxy(editormd.loadPlugins[name], this)(cm) + } + return this + }, + + /** + * 搜索替换 + * Search & replace + * + * @param {String} command CodeMirror serach commands, "find, fintNext, fintPrev, clearSearch, replace, replaceAll" + * @returns {editormd} return this + */ + + search: function(command) { + var settings = this.settings + if (!settings.searchReplace) { + alert("Error: settings.searchReplace == false") + return this + } + if (!settings.readOnly) { + this.cm.execCommand(command || "find") + } + return this + }, + searchReplace: function() { + this.search("replace") + return this + }, + searchReplaceAll: function() { + this.search("replaceAll") + return this + } + } + + editormd.fn.init.prototype = editormd.fn + + /** + * 锁屏 + * lock screen when dialog opening + * + * @returns {void} + */ + + editormd.dialogLockScreen = function() { + var settings = this.settings || { dialogLockScreen: true } + if (settings.dialogLockScreen) { + $("html,body").css("overflow", "hidden") + this.resize() + } + } + + /** + * 显示透明背景层 + * Display mask layer when dialog opening + * + * @param {Object} dialog dialog jQuery object + * @returns {void} + */ + + editormd.dialogShowMask = function(dialog) { + var editor = this.editor + var settings = this.settings || { dialogShowMask: true } + dialog.css({ + top: ($(window).height() - dialog.height()) / 2 + "px", + left: ($(window).width() - dialog.width()) / 2 + "px" + }) + if (settings.dialogShowMask) { + editor + .children("." + this.classPrefix + "mask") + .css("z-index", parseInt(dialog.css("z-index")) - 1) + .show() + } + } + + editormd.toolbarHandlers = { + undo: function() { + this.cm.undo() + }, + redo: function() { + this.cm.redo() + }, + bold: function() { + var cm = this.cm + var cursor = cm.getCursor() + var selection = cm.getSelection() + cm.replaceSelection("**" + selection + "**") + if (selection === "") { + cm.setCursor(cursor.line, cursor.ch + 2) + } + }, + del: function() { + var cm = this.cm + var cursor = cm.getCursor() + var selection = cm.getSelection() + cm.replaceSelection("~~" + selection + "~~") + if (selection === "") { + cm.setCursor(cursor.line, cursor.ch + 2) + } + }, + italic: function() { + var cm = this.cm + var cursor = cm.getCursor() + var selection = cm.getSelection() + cm.replaceSelection("*" + selection + "*") + if (selection === "") { + cm.setCursor(cursor.line, cursor.ch + 1) + } + }, + quote: function() { + var cm = this.cm + var cursor = cm.getCursor() + var selection = cm.getSelection() + if (cursor.ch !== 0) { + cm.setCursor(cursor.line, 0) + cm.replaceSelection("> " + selection) + cm.setCursor(cursor.line, cursor.ch + 2) + } else { + cm.replaceSelection("> " + selection) + } + //cm.replaceSelection("> " + selection); + //cm.setCursor(cursor.line, (selection === "") ? cursor.ch + 2 : cursor.ch + selection.length + 2); + }, + ucfirst: function() { + var cm = this.cm + var selection = cm.getSelection() + var selections = cm.listSelections() + cm.replaceSelection(editormd.firstUpperCase(selection)) + cm.setSelections(selections) + }, + ucwords: function() { + var cm = this.cm + var selection = cm.getSelection() + var selections = cm.listSelections() + cm.replaceSelection(editormd.wordsFirstUpperCase(selection)) + cm.setSelections(selections) + }, + uppercase: function() { + var cm = this.cm + var selection = cm.getSelection() + var selections = cm.listSelections() + cm.replaceSelection(selection.toUpperCase()) + cm.setSelections(selections) + }, + lowercase: function() { + var cm = this.cm + var cursor = cm.getCursor() + var selection = cm.getSelection() + var selections = cm.listSelections() + cm.replaceSelection(selection.toLowerCase()) + cm.setSelections(selections) + }, + h1: function() { + var cm = this.cm + var cursor = cm.getCursor() + var selection = cm.getSelection() + if (cursor.ch !== 0) { + cm.setCursor(cursor.line, 0) + cm.replaceSelection("# " + selection) + cm.setCursor(cursor.line, cursor.ch + 2) + } else { + cm.replaceSelection("# " + selection) + } + }, + h2: function() { + var cm = this.cm + var cursor = cm.getCursor() + var selection = cm.getSelection() + if (cursor.ch !== 0) { + cm.setCursor(cursor.line, 0) + cm.replaceSelection("## " + selection) + cm.setCursor(cursor.line, cursor.ch + 3) + } else { + cm.replaceSelection("## " + selection) + } + }, + h3: function() { + var cm = this.cm + var cursor = cm.getCursor() + var selection = cm.getSelection() + if (cursor.ch !== 0) { + cm.setCursor(cursor.line, 0) + cm.replaceSelection("### " + selection) + cm.setCursor(cursor.line, cursor.ch + 4) + } else { + cm.replaceSelection("### " + selection) + } + }, + h4: function() { + var cm = this.cm + var cursor = cm.getCursor() + var selection = cm.getSelection() + if (cursor.ch !== 0) { + cm.setCursor(cursor.line, 0) + cm.replaceSelection("#### " + selection) + cm.setCursor(cursor.line, cursor.ch + 5) + } else { + cm.replaceSelection("#### " + selection) + } + }, + h5: function() { + var cm = this.cm + var cursor = cm.getCursor() + var selection = cm.getSelection() + if (cursor.ch !== 0) { + cm.setCursor(cursor.line, 0) + cm.replaceSelection("##### " + selection) + cm.setCursor(cursor.line, cursor.ch + 6) + } else { + cm.replaceSelection("##### " + selection) + } + }, + h6: function() { + var cm = this.cm + var cursor = cm.getCursor() + var selection = cm.getSelection() + if (cursor.ch !== 0) { + cm.setCursor(cursor.line, 0) + cm.replaceSelection("###### " + selection) + cm.setCursor(cursor.line, cursor.ch + 7) + } else { + cm.replaceSelection("###### " + selection) + } + }, + "list-ul": function() { + var cm = this.cm + var cursor = cm.getCursor() + var selection = cm.getSelection() + if (selection === "") { + cm.replaceSelection("- " + selection) + } else { + var selectionText = selection.split("\n") + for (var i = 0, len = selectionText.length; i < len; i++) { + selectionText[i] = + selectionText[i] === "" ? "" : "- " + selectionText[i] + } + cm.replaceSelection(selectionText.join("\n")) + } + }, + "list-ol": function() { + var cm = this.cm + var cursor = cm.getCursor() + var selection = cm.getSelection() + if (selection === "") { + cm.replaceSelection("1. " + selection) + } else { + var selectionText = selection.split("\n") + for (var i = 0, len = selectionText.length; i < len; i++) { + selectionText[i] = + selectionText[i] === "" ? "" : i + 1 + ". " + selectionText[i] + } + cm.replaceSelection(selectionText.join("\n")) + } + }, + hr: function() { + var cm = this.cm + var cursor = cm.getCursor() + var selection = cm.getSelection() + cm.replaceSelection( + (cursor.ch !== 0 ? "\n\n" : "\n") + "------------\n\n" + ) + }, + tex: function() { + if (!this.settings.tex) { + alert("settings.tex === false") + return this + } + var cm = this.cm + var cursor = cm.getCursor() + var selection = cm.getSelection() + cm.replaceSelection("$$" + selection + "$$") + if (selection === "") { + cm.setCursor(cursor.line, cursor.ch + 2) + } + }, + link: function() { + this.executePlugin("linkDialog", "link-dialog/link-dialog") + }, + "reference-link": function() { + this.executePlugin( + "referenceLinkDialog", + "reference-link-dialog/reference-link-dialog" + ) + }, + pagebreak: function() { + if (!this.settings.pageBreak) { + alert("settings.pageBreak === false") + return this + } + var cm = this.cm + var selection = cm.getSelection() + cm.replaceSelection("\r\n[========]\r\n") + }, + image: function() { + this.executePlugin("imageDialog", "image-dialog/image-dialog") + }, + code: function() { + var cm = this.cm + var cursor = cm.getCursor() + var selection = cm.getSelection() + cm.replaceSelection("`" + selection + "`") + if (selection === "") { + cm.setCursor(cursor.line, cursor.ch + 1) + } + }, + "code-block": function() { + this.executePlugin( + "codeBlockDialog", + "code-block-dialog/code-block-dialog" + ) + }, + "preformatted-text": function() { + this.executePlugin( + "preformattedTextDialog", + "preformatted-text-dialog/preformatted-text-dialog" + ) + }, + table: function() { + this.executePlugin("tableDialog", "table-dialog/table-dialog") + }, + datetime: function() { + var cm = this.cm + var selection = cm.getSelection() + var date = new Date() + var langName = this.settings.lang.name + var datefmt = + editormd.dateFormat() + + " " + + editormd.dateFormat( + langName === "zh-cn" || langName === "zh-tw" + ? "cn-week-day" + : "week-day" + ) + cm.replaceSelection(datefmt) + }, + emoji: function() { + this.executePlugin("emojiDialog", "emoji-dialog/emoji-dialog") + }, + "html-entities": function() { + this.executePlugin( + "htmlEntitiesDialog", + "html-entities-dialog/html-entities-dialog" + ) + }, + "goto-line": function() { + this.executePlugin("gotoLineDialog", "goto-line-dialog/goto-line-dialog") + }, + watch: function() { + this[this.settings.watch ? "unwatch" : "watch"]() + }, + preview: function() { + this.previewing() + }, + fullscreen: function() { + this.fullscreen() + }, + clear: function() { + this.clear() + }, + search: function() { + this.search() + }, + help: function() { + this.executePlugin("helpDialog", "help-dialog/help-dialog") + }, + info: function() { + this.showInfoDialog() + } + } + + var isMac = navigator.platform.toUpperCase().indexOf("MAC") >= 0 + var key = isMac ? "Cmd" : "Ctrl" + editormd.keyMaps = { + [key + "-1"]: "h1", + [key + "-2"]: "h2", + [key + "-3"]: "h3", + [key + "-4"]: "h4", + [key + "-5"]: "h5", + [key + "-6"]: "h6", + [key + "-B"]: "bold", // if this is string == editormd.toolbarHandlers.xxxx + [key + "-D"]: "datetime", + [key + "Ctrl-E"]: function() { + // emoji + var cm = this.cm + var cursor = cm.getCursor() + var selection = cm.getSelection() + if (!this.settings.emoji) { + alert("Error: settings.emoji == false") + return + } + cm.replaceSelection(":" + selection + ":") + if (selection === "") { + cm.setCursor(cursor.line, cursor.ch + 1) + } + }, + [key + "-Alt-G"]: "goto-line", + [key + "-H"]: "hr", + [key + "-I"]: "italic", + [key + "-K"]: "code", + "Ctrl-L": function() { + var cm = this.cm + var cursor = cm.getCursor() + var selection = cm.getSelection() + var title = selection === "" ? "" : ' "' + selection + '"' + cm.replaceSelection("[" + selection + "](" + title + ")") + if (selection === "") { + cm.setCursor(cursor.line, cursor.ch + 1) + } + }, + [key + "-U"]: "list-ul", + "Shift-Ctrl-A": function() { + var cm = this.cm + var cursor = cm.getCursor() + var selection = cm.getSelection() + if (!this.settings.atLink) { + alert("Error: settings.atLink == false") + return + } + cm.replaceSelection("@" + selection) + if (selection === "") { + cm.setCursor(cursor.line, cursor.ch + 1) + } + }, + ["Shift" + key + "-C"]: "code", + ["Shift" + key + "Q"]: "quote", + ["Shift" + key + "S"]: "del", + ["Shift" + key + "K"]: "tex", // KaTeX + "Shift-Alt-C": function() { + var cm = this.cm + var cursor = cm.getCursor() + var selection = cm.getSelection() + cm.replaceSelection(["```", selection, "```"].join("\n")) + if (selection === "") { + cm.setCursor(cursor.line, cursor.ch + 3) + } + }, + ["Shift-" + key + "-Alt-C"]: "code-block", + ["Shift-" + key + "-H"]: "html-entities", + "Shift-Alt-H": "help", + ["Shift-" + key + "-E"]: "emoji", + ["Shift-" + key + "-U"]: "uppercase", + "Shift-Alt-U": "ucwords", + ["Shift-" + key + "-Alt-U"]: "ucfirst", + "Shift-Alt-L": "lowercase", + ["Shift-" + key + "-I"]: function() { + var cm = this.cm + var cursor = cm.getCursor() + var selection = cm.getSelection() + var title = selection === "" ? "" : ' "' + selection + '"' + cm.replaceSelection("![" + selection + "](" + title + ")") + if (selection === "") { + cm.setCursor(cursor.line, cursor.ch + 4) + } + }, + ["Shift-" + key + "-Alt-I"]: "image", + ["Shift-" + key + "-L"]: "link", + ["Shift-" + key + "-O"]: "list-ol", + ["Shift-" + key + "-P"]: "preformatted-text", + ["Shift-" + key + "-T"]: "table", + "Shift-Alt-P": "pagebreak", + F9: "watch", + F10: "preview", + F11: "fullscreen" + } + + /** + * 清除字符串两边的空格 + * Clear the space of strings both sides. + * + * @param {String} str string + * @returns {String} trimed string + */ + + var trim = function(str) { + return !String.prototype.trim + ? str.replace(/^[\s\uFEFF\xA0]+|[\s\uFEFF\xA0]+$/g, "") + : str.trim() + } + + editormd.trim = trim + + /** + * 所有单词首字母大写 + * Words first to uppercase + * + * @param {String} str string + * @returns {String} string + */ + + var ucwords = function(str) { + return str.toLowerCase().replace(/\b(\w)|\s(\w)/g, function($1) { + return $1.toUpperCase() + }) + } + + editormd.ucwords = editormd.wordsFirstUpperCase = ucwords + + /** + * 字符串首字母大写 + * Only string first char to uppercase + * + * @param {String} str string + * @returns {String} string + */ + + var firstUpperCase = function(str) { + return str.toLowerCase().replace(/\b(\w)/, function($1) { + return $1.toUpperCase() + }) + } + + var ucfirst = firstUpperCase + editormd.firstUpperCase = editormd.ucfirst = firstUpperCase + editormd.urls = { + atLinkBase: "https://github.com/" + } + editormd.regexs = { + atLink: /@(\w+)/g, + email: /(\w+)@(\w+)\.(\w+)\.?(\w+)?/g, + emailLink: /(mailto:)?([\w\.\_]+)@(\w+)\.(\w+)\.?(\w+)?/g, + emoji: /:([\w\+-]+):/g, + emojiDatetime: /(\d{1,2}:\d{1,2}:\d{1,2})/g, + twemoji: /:(tw-([\w]+)-?(\w+)?):/g, + fontAwesome: /:(fa-([\w]+)(-(\w+)){0,}):/g, + editormdLogo: /:(editormd-logo-?(\w+)?):/g, + pageBreak: /^\[[=]{8,}\]$/ + } + // Emoji graphics files url path + editormd.emoji = { + path: "http://www.emoji-cheat-sheet.com/graphics/emojis/", + ext: ".png" + } + // Twitter Emoji (Twemoji) graphics files url path + editormd.twemoji = { + path: "http://twemoji.maxcdn.com/36x36/", + ext: ".png" + } + + /** + * 自定义marked的解析器 + * Custom Marked renderer rules + * + * @param {Array} markdownToC 传入用于接收TOC的数组 + * @returns {Renderer} markedRenderer 返回marked的Renderer自定义对象 + */ + + editormd.markedRenderer = function(markdownToC, options) { + var defaults = { + toc: true, // Table of contents + tocm: false, + tocStartLevel: 1, // Said from H1 to create ToC + pageBreak: true, + atLink: true, // for @link + emailLink: true, // for mail address auto link + taskList: false, // Enable Github Flavored Markdown task lists + emoji: false, // :emoji: , Support Twemoji, fontAwesome, Editor.md logo emojis. + tex: false, // TeX(LaTeX), based on KaTeX + flowChart: false, // flowChart.js only support IE9+ + sequenceDiagram: false // sequenceDiagram.js only support IE9+ + } + var settings = $.extend(defaults, options || {}) + var marked = editormd.$marked + var markedRenderer = new marked.Renderer() + markdownToC = markdownToC || [] + + var regexs = editormd.regexs + var atLinkReg = regexs.atLink + var emojiReg = regexs.emoji + var emailReg = regexs.email + var emailLinkReg = regexs.emailLink + var twemojiReg = regexs.twemoji + var faIconReg = regexs.fontAwesome + var editormdLogoReg = regexs.editormdLogo + var pageBreakReg = regexs.pageBreak + + markedRenderer.emoji = function(text) { + text = text.replace(editormd.regexs.emojiDatetime, function($1) { + return $1.replace(/:/g, ":") + }) + + var matchs = text.match(emojiReg) + + if (!matchs || !settings.emoji) { + return text + } + + for (var i = 0, len = matchs.length; i < len; i++) { + if (matchs[i] === ":+1:") { + matchs[i] = ":\\+1:" + } + + text = text.replace(new RegExp(matchs[i]), function($1, $2) { + var faMatchs = $1.match(faIconReg) + var name = $1.replace(/:/g, "") + if (faMatchs) { + for (var fa = 0, len1 = faMatchs.length; fa < len1; fa++) { + var faName = faMatchs[fa].replace(/:/g, "") + return ( + '' + ) + } + } else { + var emdlogoMathcs = $1.match(editormdLogoReg) + var twemojiMatchs = $1.match(twemojiReg) + if (emdlogoMathcs) { + for (var x = 0, len2 = emdlogoMathcs.length; x < len2; x++) { + var logoName = emdlogoMathcs[x].replace(/:/g, "") + return ( + '' + ) + } + } else if (twemojiMatchs) { + for (var t = 0, len3 = twemojiMatchs.length; t < len3; t++) { + var twe = twemojiMatchs[t].replace(/:/g, "").replace("tw-", "") + return ( + 'twemoji-' +
+                  twe +
+                  '' + ) + } + } else { + var src = name === "+1" ? "plus1" : name + src = src === "black_large_square" ? "black_square" : src + src = src === "moon" ? "waxing_gibbous_moon" : src + return ( + ':' +
+                name +
+                ':' + ) + } + } + }) + } + return text + } + + markedRenderer.atLink = function(text) { + if (atLinkReg.test(text)) { + if (settings.atLink) { + text = text.replace(emailReg, function($1, $2, $3, $4) { + return $1.replace(/@/g, "_#_@_#_") + }) + text = text + .replace(atLinkReg, function($1, $2) { + return ( + '' + + $1 + + "" + ) + }) + .replace(/_#_@_#_/g, "@") + } + if (settings.emailLink) { + text = text.replace(emailLinkReg, function($1, $2, $3, $4, $5) { + return !$2 && + $.inArray($5, "jpg|jpeg|png|gif|webp|ico|icon|pdf".split("|")) < 0 + ? '' + $1 + "" + : $1 + }) + } + return text + } + return text + } + + markedRenderer.link = function(href, title, text) { + if (this.options.sanitize) { + try { + var prot = decodeURIComponent(unescape(href)) + .replace(/[^\w:]/g, "") + .toLowerCase() + if (prot.indexOf("javascript:") === 0) { + return "" + } + } catch (e) { + return "" + } + } + var out = '' + text.replace(/@/g, "@") + "" + } + if (title) { + out += ' title="' + title + '"' + } + out += ">" + text + "" + return out + } + + markedRenderer.heading = function(text, level, raw) { + var linkText = text + var hasLinkReg = /\s*]*)\>(.*)<\/a\>\s*/ + var getLinkTextReg = /\s*]+)\>([^\>]*)<\/a\>\s*/g + if (hasLinkReg.test(text)) { + var tempText = [] + text = text.split(/]+)\>([^\>]*)<\/a\>/) + for (var i = 0, len = text.length; i < len; i++) { + tempText.push(text[i].replace(/\s*href\=\"(.*)\"\s*/g, "")) + } + text = tempText.join(" ") + } + text = trim(text) + var escapedText = text.toLowerCase().replace(/[^\w]+/g, "-") + var toc = { + text: text, + level: level, + slug: escapedText + } + var isChinese = /^[\u4e00-\u9fa5]+$/.test(text) + var id = isChinese + ? escape(text).replace(/\%/g, "") + : text.toLowerCase().replace(/[^\w]+/g, "-") + markdownToC.push(toc) + var headingHTML = + "' + headingHTML += '' + headingHTML += '' + headingHTML += hasLinkReg + ? this.atLink(this.emoji(linkText)) + : this.atLink(this.emoji(text)) + headingHTML += "" + return headingHTML + } + markedRenderer.pageBreak = function(text) { + if (pageBreakReg.test(text) && settings.pageBreak) { + text = + '
              ' + } + return text + } + markedRenderer.paragraph = function(text) { + var isTeXLine = /^(?:\$\$|\$|\[|\()(.*)(?:\$\$|\$|\]|\))$/.test(text) + var isTeXAddClass = isTeXLine + ? ' class="' + editormd.classNames.tex + '"' + : "" + var isToC = settings.tocm + ? /^(\[TOC\]|\[TOCM\])$/.test(text) + : /^\[TOC\]$/.test(text) + var isToCMenu = /^\[TOCM\]$/.test(text) + var tocHTML = + '
              ' + text + "
              " + return isToC + ? isToCMenu + ? '
              ' + tocHTML + "

              " + : tocHTML + : pageBreakReg.test(text) + ? this.pageBreak(text) + : "" + + this.atLink(this.emoji(text)) + + "

              \n" + } + markedRenderer.code = function(code, lang, escaped) { + if (lang === "seq" || lang === "sequence") { + return '
              ' + code + "
              " + } else if (lang === "flow") { + return '
              ' + code + "
              " + } else if (lang === "math" || lang === "latex" || lang === "katex") { + return '

              ' + code + "

              " + } else { + return marked.Renderer.prototype.code.apply(this, arguments) + } + } + markedRenderer.tablecell = function(content, flags) { + var type = flags.header ? "th" : "td" + var tag = flags.align + ? "<" + type + ' style="text-align:' + flags.align + '">' + : "<" + type + ">" + return tag + this.atLink(this.emoji(content)) + "\n" + } + markedRenderer.listitem = function(text) { + if (settings.taskList && /^\s*\[[x\s]\]\s*/.test(text)) { + text = text + .replace( + /^\s*\[\s\]\s*/, + ' ' + ) + .replace( + /^\s*\[x\]\s*/, + ' ' + ) + return ( + '
            • ' + + this.atLink(this.emoji(text)) + + "
            • " + ) + } else { + return "
            • " + this.atLink(this.emoji(text)) + "
            • " + } + } + return markedRenderer + } + + /** + * + * 生成TOC(Table of Contents) + * Creating ToC (Table of Contents) + * + * @param {Array} toc 从marked获取的TOC数组列表 + * @param {Element} container 插入TOC的容器元素 + * @param {Integer} startLevel Hx 起始层级 + * @returns {Object} tocContainer 返回ToC列表容器层的jQuery对象元素 + */ + + editormd.markdownToCRenderer = function( + toc, + container, + tocDropdown, + startLevel + ) { + var html = "" + var lastLevelArr = [0] + // var lastLevel = 0 + var classPrefix = this.classPrefix + startLevel = startLevel || 1 + for (var i = 0, len = toc.length; i < len; i++) { + var text = toc[i].text + var level = toc[i].level + var stageLen = lastLevelArr.length + var lastLevel = lastLevelArr[stageLen - 1] + var goBack = 0 + if (level < startLevel) { + continue + } + if (level > lastLevel) { + html += "" + lastLevelArr.push(level) + } else if (level < lastLevel) { + for (var j = stageLen - 1; j >= 0; j--) { + if (level <= lastLevelArr[j]) { + goBack ++ + lastLevelArr.pop() + } else { + lastLevelArr.push(level) + break + } + } + html += new Array(goBack + 1).join("
          • ") + } else { + html += "" + } + html += + '
          • ' + + text + + "
              " + } + + var tocContainer = container.find(".markdown-toc") + + if ( + tocContainer.length < 1 && + container.attr("previewContainer") === "false" + ) { + var tocHTML = + '
              ' + tocHTML = tocDropdown + ? '
              ' + tocHTML + "
              " + : tocHTML + container.html(tocHTML) + tocContainer = container.find(".markdown-toc") + } + if (tocDropdown) { + tocContainer.wrap('

              ') + } + tocContainer + .html('
                ') + .children(".markdown-toc-list") + .html(html.replace(/\r?\n?<\/ul\>/g, "")) + return tocContainer + } + + /** + * + * 生成TOC下拉菜单 + * Creating ToC dropdown menu + * + * @param {Object} container 插入TOC的容器jQuery对象元素 + * @param {String} tocTitle ToC title + * @returns {Object} return toc-menu object + */ + + editormd.tocDropdownMenu = function(container, tocTitle) { + tocTitle = tocTitle || "Table of Contents" + var zindex = 400 + var tocMenus = container.find("." + this.classPrefix + "toc-menu") + tocMenus.each(function() { + var $this = $(this) + var toc = $this.children(".markdown-toc") + var icon = '' + var btn = + '' + + icon + + tocTitle + + "" + var menu = toc.children("ul") + var list = menu.find("li") + toc.append(btn) + list.first().before("
              • " + tocTitle + " " + icon + "

              • ") + $this + .mouseover(function() { + menu.show() + list.each(function() { + var li = $(this) + var ul = li.children("ul") + if (ul.html() === "") { + ul.remove() + } + if (ul.length > 0 && ul.html() !== "") { + var firstA = li.children("a").first() + if (firstA.children(".fa").length < 1) { + firstA.append( + $(icon).css({ float: "right", paddingTop: "4px" }) + ) + } + } + li + .mouseover(function() { + ul.css("z-index", zindex).show() + zindex += 1 + }) + .mouseleave(function() { + ul.hide() + }) + }) + }) + .mouseleave(function() { + menu.hide() + }) + }) + + return tocMenus + } + + /** + * 简单地过滤指定的HTML标签 + * Filter custom html tags + * + * @param {String} html 要过滤HTML + * @param {String} filters 要过滤的标签 + * @returns {String} html 返回过滤的HTML + */ + + editormd.filterHTMLTags = function(html, filters) { + if (typeof html !== "string") { + html = new String(html) + } + if (typeof filters !== "string") { + return html + } + var expression = filters.split("|") + var filterTags = expression[0].split(",") + var attrs = expression[1] + for (var i = 0, len = filterTags.length; i < len; i++) { + var tag = filterTags[i] + html = html.replace( + new RegExp("]*)>([^>]*)", "igm"), + "" + ) + } + //return html; + if (typeof attrs !== "undefined") { + var htmlTagRegex = /<(\w+)\s*([^\>]*)\>([^\>]*)<\/(\w+)\>/gi + if (attrs === "*") { + html = html.replace(htmlTagRegex, function($1, $2, $3, $4, $5) { + return "<" + $2 + ">" + $4 + "" + }) + } else if (attrs === "on*") { + html = html.replace(htmlTagRegex, function($1, $2, $3, $4, $5) { + var el = $("<" + $2 + ">" + $4 + "") + var _attrs = $($1)[0].attributes + var $attrs = {} + $.each(_attrs, function(i, e) { + if (e.nodeName !== '"') { + $attrs[e.nodeName] = e.nodeValue + // fixed will cause xss problem + if (e.nodeName === "href" && e.nodeValue.toLowerCase().indexOf('javascript:') >= 0) { + $attrs[e.nodeName] = 'javascript:;'; + } + } + }) + $.each($attrs, function(i) { + if (i.indexOf("on") === 0) { + delete $attrs[i] + } + }) + el.attr($attrs) + var text = typeof el[1] !== "undefined" ? $(el[1]).text() : "" + // FIXED 使用 on* 过滤标签的属性,图片加链接的语法会出错的问题 + if ($2 && !isNaN($2) && $2 !== $5) { + text += "<" + $2 + ">" + $4 + "" + } else if ($2 && isNaN($2) && $2 !== $5) { + text += "" + return el[0].outerHTML + text + } else { + return el[0].outerHTML + text + } + }) + } else { + html = html.replace(htmlTagRegex, function($1, $2, $3, $4) { + var filterAttrs = attrs.split(",") + var el = $($1) + el.html($4) + $.each(filterAttrs, function(i) { + el.attr(filterAttrs[i], null) + }) + return el[0].outerHTML + }) + } + } + return html + } + + /** + * 将Markdown文档解析为HTML用于前台显示 + * Parse Markdown to HTML for Font-end preview. + * + * @param {String} id 用于显示HTML的对象ID + * @param {Object} [options={}] 配置选项,可选 + * @returns {Object} div 返回jQuery对象元素 + */ + + editormd.markdownToHTML = function(id, options) { + var defaults = { + gfm: true, + toc: true, + tocm: false, + tocStartLevel: 1, + tocTitle: "目录", + tocDropdown: false, + tocContainer: "", + markdown: "", + markdownSourceCode: false, + htmlDecode: false, + autoLoadKaTeX: true, + pageBreak: true, + atLink: true, // for @link + emailLink: true, // for mail address auto link + tex: false, + taskList: false, // Github Flavored Markdown task lists + emoji: false, + flowChart: false, + sequenceDiagram: false, + previewCodeHighlight: true + } + + editormd.$marked = marked + + var div = $("#" + id) + var settings = (div.settings = $.extend(true, defaults, options || {})) + var saveTo = div.find("textarea") + if (saveTo.length < 1) { + div.append("") + saveTo = div.find("textarea") + } + var markdownDoc = + settings.markdown === "" ? saveTo.val() : settings.markdown + var markdownToC = [] + var rendererOptions = { + toc: settings.toc, + tocm: settings.tocm, + tocStartLevel: settings.tocStartLevel, + taskList: settings.taskList, + emoji: settings.emoji, + tex: settings.tex, + pageBreak: settings.pageBreak, + atLink: settings.atLink, // for @link + emailLink: settings.emailLink, // for mail address auto link + flowChart: settings.flowChart, + sequenceDiagram: settings.sequenceDiagram, + previewCodeHighlight: settings.previewCodeHighlight + } + var markedOptions = { + renderer: editormd.markedRenderer(markdownToC, rendererOptions), + gfm: settings.gfm, + tables: true, + breaks: true, + pedantic: false, + sanitize: settings.htmlDecode ? false : true, // 是否忽略HTML标签,即是否开启HTML标签解析,为了安全性,默认不开启 + smartLists: true, + smartypants: true + } + markdownDoc = new String(markdownDoc) + var markdownParsed = marked(markdownDoc, markedOptions) + markdownParsed = editormd.filterHTMLTags( + markdownParsed, + settings.htmlDecode + ) + if (settings.markdownSourceCode) { + saveTo.text(markdownDoc) + } else { + saveTo.remove() + } + div + .addClass("markdown-body " + this.classPrefix + "html-preview") + .append(markdownParsed) + var tocContainer = + settings.tocContainer !== "" ? $(settings.tocContainer) : div + if (settings.tocContainer !== "") { + tocContainer.attr("previewContainer", false) + } + if (settings.toc) { + div.tocContainer = this.markdownToCRenderer( + markdownToC, + tocContainer, + settings.tocDropdown, + settings.tocStartLevel + ) + if ( + settings.tocDropdown || + div.find("." + this.classPrefix + "toc-menu").length > 0 + ) { + this.tocDropdownMenu(div, settings.tocTitle) + } + if (settings.tocContainer !== "") { + div.find(".editormd-toc-menu, .editormd-markdown-toc").remove() + } + } + if (settings.previewCodeHighlight) { + div.find("pre").addClass("prettyprint linenums") + prettyPrint() + } + if (!editormd.isIE8) { + if (settings.flowChart) { + div.find(".flowchart").flowChart() + } + if (settings.sequenceDiagram) { + div.find(".sequence-diagram").sequenceDiagram({ theme: "simple" }) + } + } + if (settings.tex) { + var katexHandle = function() { + div.find("." + editormd.classNames.tex).each(function() { + var tex = $(this) + katex.render( + tex + .html() + .replace(/</g, "<") + .replace(/>/g, ">"), + tex[0] + ) + tex.find(".katex").css("font-size", "1.6em") + }) + } + if (settings.autoLoadKaTeX && !editormd.$katex && !editormd.kaTeXLoaded) { + this.loadKaTeX(function() { + editormd.$katex = katex + editormd.kaTeXLoaded = true + katexHandle() + }) + } else { + katexHandle() + } + } + div.getMarkdown = function() { + return saveTo.val() + } + return div + } + + // Editor.md themes, change toolbar themes etc. + // added @1.5.0 + editormd.themes = ["default", "dark"] + // Preview area themes + // added @1.5.0 + editormd.previewThemes = ["default", "dark"] + // CodeMirror / editor area themes + // @1.5.0 rename -> editorThemes, old version -> themes + editormd.editorThemes = [ + "default", + "3024-day", + "3024-night", + "ambiance", + "ambiance-mobile", + "base16-dark", + "base16-light", + "blackboard", + "cobalt", + "eclipse", + "elegant", + "erlang-dark", + "lesser-dark", + "mbo", + "mdn-like", + "midnight", + "monokai", + "neat", + "neo", + "night", + "paraiso-dark", + "paraiso-light", + "pastel-on-dark", + "rubyblue", + "solarized", + "the-matrix", + "tomorrow-night-eighties", + "twilight", + "vibrant-ink", + "xq-dark", + "xq-light" + ] + editormd.loadPlugins = {} + editormd.loadFiles = { + js: [], + css: [], + plugin: [] + } + /** + * 动态加载Editor.md插件,但不立即执行 + * Load editor.md plugins + * + * @param {String} fileName 插件文件路径 + * @param {Function} [callback=function()] 加载成功后执行的回调函数 + * @param {String} [into="head"] 嵌入页面的位置 + */ + + editormd.loadPlugin = function(fileName, callback, into) { + callback = callback || function() {} + this.loadScript( + fileName, + function() { + editormd.loadFiles.plugin.push(fileName) + callback() + }, + into + ) + } + + /** + * 动态加载CSS文件的方法 + * Load css file method + * + * @param {String} fileName CSS文件名 + * @param {Function} [callback=function()] 加载成功后执行的回调函数 + * @param {String} [into="head"] 嵌入页面的位置 + */ + + editormd.loadCSS = function(fileName, callback, into) { + into = into || "head" + callback = callback || function() {} + + var css = document.createElement("link") + css.type = "text/css" + css.rel = "stylesheet" + css.onload = css.onreadystatechange = function() { + editormd.loadFiles.css.push(fileName) + callback() + } + css.href = fileName + ".css" + if (into === "head") { + document.getElementsByTagName("head")[0].appendChild(css) + } else { + document.body.appendChild(css) + } + } + + editormd.isIE = navigator.appName === "Microsoft Internet Explorer" + editormd.isIE8 = editormd.isIE && navigator.appVersion.match(/8./i) === "8." + + /** + * 动态加载JS文件的方法 + * Load javascript file method + * + * @param {String} fileName JS文件名 + * @param {Function} [callback=function()] 加载成功后执行的回调函数 + * @param {String} [into="head"] 嵌入页面的位置 + */ + + editormd.loadScript = function(fileName, callback, into) { + into = into || "head" + callback = callback || function() {} + var script = null + script = document.createElement("script") + script.id = fileName.replace(/[\./]+/g, "-") + script.type = "text/javascript" + script.src = fileName + ".js" + if (editormd.isIE8) { + script.onreadystatechange = function() { + if (script.readyState) { + if ( + script.readyState === "loaded" || + script.readyState === "complete" + ) { + script.onreadystatechange = null + editormd.loadFiles.js.push(fileName) + callback() + } + } + } + } else { + script.onload = function() { + editormd.loadFiles.js.push(fileName) + callback() + } + } + if (into === "head") { + document.getElementsByTagName("head")[0].appendChild(script) + } else { + document.body.appendChild(script) + } + } + + // 使用国外的CDN,加载速度有时会很慢,或者自定义URL + // You can custom KaTeX load url. + editormd.katexURL = { + css: "//cdnjs.cloudflare.com/ajax/libs/KaTeX/0.9.0/katex.min", + jsmain: "//cdnjs.cloudflare.com/ajax/libs/KaTeX/0.9.0/katex.min", + jsauto: "//cdnjs.cloudflare.com/ajax/libs/KaTeX/0.9.0/contrib/auto-render.min" + } + editormd.kaTeXLoaded = false + /** + * 加载KaTeX文件 + * load KaTeX files + * + * @param {Function} [callback=function()] 加载成功后执行的回调函数 + */ + editormd.loadKaTeX = function(callback) { + editormd.loadCSS(editormd.katexURL.css, function() { + editormd.loadScript(editormd.katexURL.jsmain, function() { + editormd.loadScript(editormd.katexURL.jsauto, callback || function() {}) + }) + }) + } + + /** + * 锁屏 + * lock screen + * + * @param {Boolean} lock Boolean 布尔值,是否锁屏 + * @returns {void} + */ + + editormd.lockScreen = function(lock) { + $("html,body").css("overflow", lock ? "hidden" : "") + } + + /** + * 动态创建对话框 + * Creating custom dialogs + * + * @param {Object} options 配置项键值对 Key/Value + * @returns {dialog} 返回创建的dialog的jQuery实例对象 + */ + + editormd.createDialog = function(options) { + var defaults = { + name: "", + width: 420, + height: 240, + title: "", + drag: true, + closed: true, + content: "", + mask: true, + maskStyle: { + backgroundColor: "#fff", + opacity: 0.1 + }, + lockScreen: true, + footer: true, + buttons: false + } + + options = $.extend(true, defaults, options) + + var $this = this + var editor = this.editor + var classPrefix = editormd.classPrefix + var guid = new Date().getTime() + var dialogName = + options.name === "" ? classPrefix + "dialog-" + guid : options.name + var mouseOrTouch = editormd.mouseOrTouch + var html = '
                ' + if (options.title !== "") { + html += + '
                " + html += + '' + + options.title + + "" + html += "
                " + } + if (options.closed) { + html += + '
                ' + } + html += + '
                ' + options.content + if (options.footer || typeof options.footer === "string") { + html += + '" + } + html += "
                " + html += + '
                ' + html += + '
                ' + html += "
                " + editor.append(html) + var dialog = editor.find("." + dialogName) + dialog.lockScreen = function(lock) { + if (options.lockScreen) { + $("html,body").css("overflow", lock ? "hidden" : "") + $this.resize() + } + return dialog + } + dialog.showMask = function() { + if (options.mask) { + editor + .find("." + classPrefix + "mask") + .css(options.maskStyle) + .css("z-index", editormd.dialogZindex - 1) + .show() + } + return dialog + } + dialog.hideMask = function() { + if (options.mask) { + editor.find("." + classPrefix + "mask").hide() + } + return dialog + } + dialog.loading = function(show) { + var loading = dialog.find("." + classPrefix + "dialog-mask") + loading[show ? "show" : "hide"]() + return dialog + } + dialog.lockScreen(true).showMask() + dialog.show().css({ + zIndex: editormd.dialogZindex, + border: editormd.isIE8 ? "1px solid #ddd" : "", + width: + typeof options.width === "number" + ? options.width + "px" + : options.width, + height: + typeof options.height === "number" + ? options.height + "px" + : options.height + }) + var dialogPosition = function() { + dialog.css({ + top: ($(window).height() - dialog.height()) / 2 + "px", + left: ($(window).width() - dialog.width()) / 2 + "px" + }) + } + dialogPosition() + $(window).resize(dialogPosition) + dialog + .children("." + classPrefix + "dialog-close") + .bind(mouseOrTouch("click", "touchend"), function() { + dialog + .hide() + .lockScreen(false) + .hideMask() + }) + if (typeof options.buttons === "object") { + var footer = (dialog.footer = dialog.find( + "." + classPrefix + "dialog-footer" + )) + for (var key in options.buttons) { + var btn = options.buttons[key] + var btnClassName = classPrefix + key + "-btn" + footer.append( + '" + ) + btn[1] = $.proxy(btn[1], dialog) + footer + .children("." + btnClassName) + .bind(mouseOrTouch("click", "touchend"), btn[1]) + } + } + if (options.title !== "" && options.drag) { + var posX, posY + var dialogHeader = dialog.children("." + classPrefix + "dialog-header") + if (!options.mask) { + dialogHeader.bind(mouseOrTouch("click", "touchend"), function() { + editormd.dialogZindex += 2 + dialog.css("z-index", editormd.dialogZindex) + }) + } + var userCanSelect = function(obj) { + obj.removeClass(classPrefix + "user-unselect").off("selectstart") + } + var userUnselect = function(obj) { + obj + .addClass(classPrefix + "user-unselect") + .on("selectstart", function(event) { + // selectstart for IE + return false + }) + } + var moveAction = function(e) { + e = e || window.event //IE + var left, + top, + nowLeft = parseInt(dialog[0].style.left), + nowTop = parseInt(dialog[0].style.top) + if (nowLeft >= 0) { + if (nowLeft + dialog.width() <= $(window).width()) { + left = e.clientX - posX + } else { + left = $(window).width() - dialog.width() + document.onmousemove = null + } + } else { + left = 0 + document.onmousemove = null + } + if (nowTop >= 0) { + top = e.clientY - posY + } else { + top = 0 + document.onmousemove = null + } + document.onselectstart = function() { + return false + } + userUnselect($("body")) + userUnselect(dialog) + dialog[0].style.left = left + "px" + dialog[0].style.top = top + "px" + } + dialogHeader.mousedown(function(e) { + e = e || window.event //IE + posX = e.clientX - parseInt(dialog[0].style.left) + posY = e.clientY - parseInt(dialog[0].style.top) + document.onmousemove = moveAction + }) + document.onmouseup = function() { + userCanSelect($("body")) + userCanSelect(dialog) + document.onselectstart = null + document.onmousemove = null + } + dialogHeader.touchDraggable = function() { + var offset = null + var start = function(e) { + var orig = e.originalEvent + var pos = $(this) + .parent() + .position() + offset = { + x: orig.changedTouches[0].pageX - pos.left, + y: orig.changedTouches[0].pageY - pos.top + } + } + var move = function(e) { + e.preventDefault() + var orig = e.originalEvent + $(this) + .parent() + .css({ + top: orig.changedTouches[0].pageY - offset.y, + left: orig.changedTouches[0].pageX - offset.x + }) + } + this.bind("touchstart", start).bind("touchmove", move) + } + dialogHeader.touchDraggable() + } + editormd.dialogZindex += 2 + return dialog + } + + /** + * 鼠标和触摸事件的判断/选择方法 + * MouseEvent or TouchEvent type switch + * + * @param {String} [mouseEventType="click"] 供选择的鼠标事件 + * @param {String} [touchEventType="touchend"] 供选择的触摸事件 + * @returns {String} EventType 返回事件类型名称 + */ + + editormd.mouseOrTouch = function(mouseEventType, touchEventType) { + mouseEventType = mouseEventType || "click" + touchEventType = touchEventType || "touchend" + var eventType = mouseEventType + try { + document.createEvent("TouchEvent") + eventType = touchEventType + } catch (e) {} + return eventType + } + + /** + * 日期时间的格式化方法 + * Datetime format method + * + * @param {String} [format=""] 日期时间的格式,类似PHP的格式 + * @returns {String} datefmt 返回格式化后的日期时间字符串 + */ + + editormd.dateFormat = function(format) { + format = format || "" + var addZero = function(d) { + return d < 10 ? "0" + d : d + } + var date = new Date() + var year = date.getFullYear() + var year2 = year.toString().slice(2, 4) + var month = addZero(date.getMonth() + 1) + var day = addZero(date.getDate()) + var weekDay = date.getDay() + var hour = addZero(date.getHours()) + var min = addZero(date.getMinutes()) + var second = addZero(date.getSeconds()) + var ms = addZero(date.getMilliseconds()) + var datefmt = "" + var ymd = year2 + "-" + month + "-" + day + var fymd = year + "-" + month + "-" + day + var hms = hour + ":" + min + ":" + second + switch (format) { + case "UNIX Time": + datefmt = date.getTime() + break + case "UTC": + datefmt = date.toUTCString() + break + case "yy": + datefmt = year2 + break + case "year": + case "yyyy": + datefmt = year + break + case "month": + case "mm": + datefmt = month + break + case "cn-week-day": + case "cn-wd": + var cnWeekDays = ["日", "一", "二", "三", "四", "五", "六"] + datefmt = "星期" + cnWeekDays[weekDay] + break + case "week-day": + case "wd": + var weekDays = [ + "Sunday", + "Monday", + "Tuesday", + "Wednesday", + "Thursday", + "Friday", + "Saturday" + ] + datefmt = weekDays[weekDay] + break + case "day": + case "dd": + datefmt = day + break + case "hour": + case "hh": + datefmt = hour + break + case "min": + case "ii": + datefmt = min + break + case "second": + case "ss": + datefmt = second + break + case "ms": + datefmt = ms + break + case "yy-mm-dd": + datefmt = ymd + break + case "yyyy-mm-dd": + datefmt = fymd + break + case "yyyy-mm-dd h:i:s ms": + case "full + ms": + datefmt = fymd + " " + hms + " " + ms + break + case "full": + case "yyyy-mm-dd h:i:s": + break + default: + datefmt = fymd + " " + hms + break + } + return datefmt + } + return editormd +}) diff --git a/system/admin/editor.md/editormd.min.js b/system/admin/editor.md/editormd.min.js new file mode 100644 index 00000000..ab2f30ef --- /dev/null +++ b/system/admin/editor.md/editormd.min.js @@ -0,0 +1,2 @@ +/*! Editor.md v2.0.1 | editormd.min.js | Open source online markdown editor. | MIT License | By: Pandao | https://github.com/pandao/editor.md | 2018-03-29 */ +"use strict";function _defineProperty(e,t,i){return t in e?Object.defineProperty(e,t,{value:i,enumerable:!0,configurable:!0,writable:!0}):e[t]=i,e}function _typeof(e){return(_typeof="function"==typeof Symbol&&"symbol"==typeof Symbol.iterator?function(e){return typeof e}:function(e){return e&&"function"==typeof Symbol&&e.constructor===Symbol&&e!==Symbol.prototype?"symbol":typeof e})(e)}!function(e){"function"==typeof require&&"object"===("undefined"==typeof exports?"undefined":_typeof(exports))&&"object"===("undefined"==typeof module?"undefined":_typeof(module))?module.exports=e:"function"==typeof define?define.amd||define(["jquery"],e):window.editormd=e()}(function(){var e,t="undefined"!=typeof jQuery?jQuery:Zepto;if(void 0!==t){var i=function e(t,i){return new e.fn.init(t,i)};i.title=i.$name="Editor.md",i.version="1.5.0",i.homePage="https://pandao.github.io/editor.md/",i.classPrefix="editormd-",i.toolbarModes={full:["undo","redo","|","bold","del","italic","quote","ucwords","uppercase","lowercase","|","h1","h2","h3","h4","h5","h6","|","list-ul","list-ol","hr","|","link","reference-link","image","code","preformatted-text","code-block","table","datetime","emoji","html-entities","pagebreak","|","goto-line","watch","preview","fullscreen","clear","search","|","help","info"],simple:["undo","redo","|","bold","del","italic","quote","uppercase","lowercase","|","h1","h2","h3","h4","h5","h6","|","list-ul","list-ol","hr","|","watch","preview","fullscreen","|","help","info"],mini:["undo","redo","|","watch","preview","|","help","info"]},i.defaults={mode:"gfm",name:"",value:"",theme:"",editorTheme:"default",previewTheme:"",markdown:"",appendMarkdown:"",width:"100%",height:"100%",path:"./lib/",pluginPath:"",delay:300,autoLoadModules:!0,watch:!0,placeholder:"Enjoy Markdown! coding now...",gotoLine:!0,codeFold:!1,autoHeight:!1,minHeight:"150px",maxHeight:"none",autoFocus:!0,autoCloseTags:!0,searchReplace:!0,syncScrolling:!0,readOnly:!1,tabSize:4,indentUnit:4,lineNumbers:!0,lineWrapping:!0,autoCloseBrackets:!0,showTrailingSpace:!0,matchBrackets:!0,indentWithTabs:!0,styleSelectedText:!0,matchWordHighlight:!0,styleActiveLine:!0,dialogLockScreen:!0,dialogShowMask:!0,dialogDraggable:!0,dialogMaskBgColor:"#fff",dialogMaskOpacity:.1,fontSize:"13px",saveHTMLToTextarea:!1,disabledKeyMaps:[],onload:function(){},onresize:function(){},onchange:function(){},onwatch:null,onunwatch:null,onpreviewing:function(){},onpreviewed:function(){},onfullscreen:function(){},onfullscreenExit:function(){},onscroll:function(){},onpreviewscroll:function(){},imageUpload:!1,imageFormats:["jpg","jpeg","gif","png","bmp","webp"],imageUploadURL:"",crossDomainUpload:!1,uploadCallbackURL:"",toc:!0,tocm:!1,tocTitle:"",tocDropdown:!1,tocContainer:"",tocStartLevel:1,htmlDecode:!1,pageBreak:!0,atLink:!0,emailLink:!0,taskList:!1,emoji:!1,tex:!1,flowChart:!1,sequenceDiagram:!1,previewCodeHighlight:!0,toolbar:!0,toolbarAutoFixed:!0,toolbarIcons:"full",toolbarTitles:{},toolbarHandlers:{ucwords:function(){return i.toolbarHandlers.ucwords},lowercase:function(){return i.toolbarHandlers.lowercase}},toolbarCustomIcons:{lowercase:'a',ucwords:'Aa'},toolbarIconsClass:{undo:"fa-undo",redo:"fa-repeat",bold:"fa-bold",del:"fa-strikethrough",italic:"fa-italic",quote:"fa-quote-left",uppercase:"fa-font",h1:i.classPrefix+"bold",h2:i.classPrefix+"bold",h3:i.classPrefix+"bold",h4:i.classPrefix+"bold",h5:i.classPrefix+"bold",h6:i.classPrefix+"bold","list-ul":"fa-list-ul","list-ol":"fa-list-ol",hr:"fa-minus",link:"fa-link","reference-link":"fa-anchor",image:"fa-picture-o",code:"fa-code","preformatted-text":"fa-file-code-o","code-block":"fa-file-code-o",table:"fa-table",datetime:"fa-clock-o",emoji:"fa-smile-o","html-entities":"fa-copyright",pagebreak:"fa-newspaper-o","goto-line":"fa-terminal",watch:"fa-eye-slash",unwatch:"fa-eye",preview:"fa-desktop",search:"fa-search",fullscreen:"fa-arrows-alt",clear:"fa-eraser",help:"fa-question-circle",info:"fa-info-circle"},toolbarIconTexts:{},lang:{name:"zh-cn",description:"开源在线Markdown编辑器
                Open source online Markdown editor.",tocTitle:"目录",toolbar:{undo:"撤销(Ctrl+Z)",redo:"重做(Ctrl+Y)",bold:"粗体",del:"删除线",italic:"斜体",quote:"引用",ucwords:"将每个单词首字母转成大写",uppercase:"将所选转换成大写",lowercase:"将所选转换成小写",h1:"标题1",h2:"标题2",h3:"标题3",h4:"标题4",h5:"标题5",h6:"标题6","list-ul":"无序列表","list-ol":"有序列表",hr:"横线",link:"链接","reference-link":"引用链接",image:"添加图片",code:"行内代码","preformatted-text":"预格式文本 / 代码块(缩进风格)","code-block":"代码块(多语言风格)",table:"添加表格",datetime:"日期时间",emoji:"Emoji表情","html-entities":"HTML实体字符",pagebreak:"插入分页符","goto-line":"跳转到行",watch:"关闭实时预览",unwatch:"开启实时预览",preview:"全窗口预览HTML(按 Shift + ESC还原)",fullscreen:"全屏(按ESC还原)",clear:"清空",search:"搜索",help:"使用帮助",info:"关于"+i.title},buttons:{enter:"确定",cancel:"取消",close:"关闭"},dialog:{link:{title:"添加链接",url:"链接地址",urlTitle:"链接标题",urlEmpty:"错误:请填写链接地址。"},referenceLink:{title:"添加引用链接",name:"引用名称",url:"链接地址",urlId:"链接ID",urlTitle:"链接标题",nameEmpty:"错误:引用链接的名称不能为空。",idEmpty:"错误:请填写引用链接的ID。",urlEmpty:"错误:请填写引用链接的URL地址。"},image:{title:"添加图片",url:"图片地址",link:"图片链接",alt:"图片描述",uploadButton:"本地上传",imageURLEmpty:"错误:图片地址不能为空。",uploadFileEmpty:"错误:上传的图片不能为空。",formatNotAllowed:"错误:只允许上传图片文件,允许上传的图片文件格式有:"},preformattedText:{title:"添加预格式文本或代码块",emptyAlert:"错误:请填写预格式文本或代码的内容。"},codeBlock:{title:"添加代码块",selectLabel:"代码语言:",selectDefaultText:"请选择代码语言",otherLanguage:"其他语言",unselectedLanguageAlert:"错误:请选择代码所属的语言类型。",codeEmptyAlert:"错误:请填写代码内容。"},htmlEntities:{title:"HTML 实体字符"},help:{title:"使用帮助"}}}},i.classNames={tex:i.classPrefix+"tex"},i.dialogZindex=99999,i.$katex=null,i.$marked=null,i.$CodeMirror=null,i.$prettyPrint=null;var o,r;i.prototype=i.fn={state:{watching:!1,loaded:!1,preview:!1,fullscreen:!1},init:function(e,o){o=o||{},"object"===_typeof(e)&&(o=e);var r=this.classPrefix=i.classPrefix,n=this.settings=t.extend(!0,i.defaults,o);e="object"===_typeof(e)?n.id:e;var a=this.editor=t("#"+e);this.id=e,this.lang=n.lang;var s=this.classNames={textarea:{html:r+"html-textarea",markdown:r+"markdown-textarea"}};n.pluginPath=""===n.pluginPath?n.path+"../plugins/":n.pluginPath,this.state.watching=!!n.watch,a.hasClass("editormd")||a.addClass("editormd"),a.css({width:"number"==typeof n.width?n.width+"px":n.width,height:"number"==typeof n.height?n.height+"px":n.height}),n.autoHeight&&a.css({height:"auto",minHeight:n.minHeight});var l=this.markdownTextarea=a.children("textarea");l.length<1&&(a.append(""),l=this.markdownTextarea=a.children("textarea")),l.addClass(s.textarea.markdown).attr("placeholder",n.placeholder),void 0!==l.attr("name")&&""!==l.attr("name")||l.attr("name",""!==n.name?n.name:e+"-markdown-doc");var c=[n.readOnly?"":'',n.saveHTMLToTextarea?'':"",'
                ','
                ','
                '].join("\n");return a.append(c).addClass(r+"vertical"),""!==n.theme&&a.addClass(r+"theme-"+n.theme),this.mask=a.children("."+r+"mask"),this.containerMask=a.children("."+r+"container-mask"),""!==n.markdown&&l.val(n.markdown),""!==n.appendMarkdown&&l.val(l.val()+n.appendMarkdown),this.htmlTextarea=a.children("."+s.textarea.html),this.preview=a.children("."+r+"preview"),this.previewContainer=this.preview.children("."+r+"preview-container"),""!==n.previewTheme&&this.preview.addClass(r+"preview-theme-"+n.previewTheme),"function"==typeof define&&define.amd&&("undefined"!=typeof katex&&(i.$katex=katex),n.searchReplace&&!n.readOnly&&(i.loadCSS(n.path+"codemirror/addon/dialog/dialog"),i.loadCSS(n.path+"codemirror/addon/search/matchesonscrollbar"))),"function"==typeof define&&define.amd||!n.autoLoadModules?("undefined"!=typeof CodeMirror&&(i.$CodeMirror=CodeMirror),"undefined"!=typeof marked&&(i.$marked=marked),this.setCodeMirror().setToolbar().loadedDisplay()):this.loadQueues(),this},loadQueues:function(){var e=this,t=this.settings,o=t.path,r=function(){if(i.isIE8)return void e.loadedDisplay();t.flowChart||t.sequenceDiagram?i.loadScript(o+"raphael.min",function(){i.loadScript(o+"underscore.min",function(){!t.flowChart&&t.sequenceDiagram?i.loadScript(o+"sequence-diagram.min",function(){e.loadedDisplay()}):t.flowChart&&!t.sequenceDiagram?i.loadScript(o+"flowchart.min",function(){i.loadScript(o+"jquery.flowchart.min",function(){e.loadedDisplay()})}):t.flowChart&&t.sequenceDiagram&&i.loadScript(o+"flowchart.min",function(){i.loadScript(o+"jquery.flowchart.min",function(){i.loadScript(o+"sequence-diagram.min",function(){e.loadedDisplay()})})})})}):e.loadedDisplay()};return i.loadCSS(o+"codemirror/codemirror.min"),t.searchReplace&&!t.readOnly&&(i.loadCSS(o+"codemirror/addon/dialog/dialog"),i.loadCSS(o+"codemirror/addon/search/matchesonscrollbar")),t.codeFold&&i.loadCSS(o+"codemirror/addon/fold/foldgutter"),i.loadScript(o+"codemirror/codemirror.min",function(){i.$CodeMirror=CodeMirror,i.loadScript(o+"codemirror/modes.min",function(){i.loadScript(o+"codemirror/addons.min",function(){if(e.setCodeMirror(),"gfm"!==t.mode&&"markdown"!==t.mode)return e.loadedDisplay(),!1;e.setToolbar(),i.loadScript(o+"marked.min",function(){i.$marked=marked,t.previewCodeHighlight?i.loadScript(o+"prettify.min",function(){r()}):r()})})})}),this},setTheme:function(e){var t=this.editor,i=this.settings.theme,o=this.classPrefix+"theme-";return t.removeClass(o+i).addClass(o+e),this.settings.theme=e,this},setEditorTheme:function(e){var t=this.settings;return t.editorTheme=e,"default"!==e&&i.loadCSS(t.path+"codemirror/theme/"+t.editorTheme),this.cm.setOption("theme",e),this},setCodeMirrorTheme:function(e){return this.setEditorTheme(e),this},setPreviewTheme:function(e){var t=this.preview,i=this.settings.previewTheme,o=this.classPrefix+"preview-theme-";return t.removeClass(o+i).addClass(o+e),this.settings.previewTheme=e,this},setCodeMirror:function(){var e=this.settings,t=this.editor;"default"!==e.editorTheme&&i.loadCSS(e.path+"codemirror/theme/"+e.editorTheme);var o={mode:e.mode,theme:e.editorTheme,tabSize:e.tabSize,dragDrop:!1,autofocus:e.autoFocus,autoCloseTags:e.autoCloseTags,readOnly:!!e.readOnly&&"nocursor",indentUnit:e.indentUnit,lineNumbers:e.lineNumbers,lineWrapping:e.lineWrapping,extraKeys:{"Ctrl-Q":function(e){e.foldCode(e.getCursor())}},foldGutter:e.codeFold,gutters:["CodeMirror-linenumbers","CodeMirror-foldgutter"],matchBrackets:e.matchBrackets,indentWithTabs:e.indentWithTabs,styleActiveLine:e.styleActiveLine,styleSelectedText:e.styleSelectedText,autoCloseBrackets:e.autoCloseBrackets,showTrailingSpace:e.showTrailingSpace,highlightSelectionMatches:!!e.matchWordHighlight&&{showToken:"onselected"!==e.matchWordHighlight&&/\w/}};return this.codeEditor=this.cm=i.$CodeMirror.fromTextArea(this.markdownTextarea[0],o),this.codeMirror=this.cmElement=t.children(".CodeMirror"),""!==e.value&&this.cm.setValue(e.value),this.codeMirror.css({fontSize:e.fontSize,width:e.watch?"50%":"100%"}),e.autoHeight&&(this.codeMirror.css({height:"auto",minHeight:e.minHeight,maxHeight:e.maxHeight}),this.codeMirror.find(".CodeMirror-gutters").css({minHeight:e.minHeight}),this.codeMirror.find(".CodeMirror-scroll").css({minHeight:e.minHeight,maxHeight:e.maxHeight}),this.cm.setOption("viewportMargin",1/0)),e.lineNumbers||this.codeMirror.find(".CodeMirror-gutters").css("border-right","none"),this},getCodeMirrorOption:function(e){return this.cm.getOption(e)},setCodeMirrorOption:function(e,t){return this.cm.setOption(e,t),this},addKeyMap:function(e,t){return this.cm.addKeyMap(e,t),this},removeKeyMap:function(e){return this.cm.removeKeyMap(e),this},gotoLine:function(e){var i=this.settings;if(!i.gotoLine)return this;var o=this.cm,r=(this.editor,o.lineCount()),n=this.preview;if("string"==typeof e&&("last"===e&&(e=r),"first"===e&&(e=1)),"number"!=typeof e)return alert("Error: The line number must be an integer."),this;if((e=parseInt(e)-1)>r)return alert("Error: The line number range 1-"+r),this;o.setCursor({line:e,ch:0});var a=o.getScrollInfo(),s=a.clientHeight,l=o.charCoords({line:e,ch:0},"local");if(o.scrollTo(null,(l.top+l.bottom-s)/2),i.watch){var c=this.codeMirror.find(".CodeMirror-scroll")[0],h=t(c).height(),d=c.scrollTop,u=d/c.scrollHeight;0===d?n.scrollTop(0):d+h>=c.scrollHeight-16?n.scrollTop(n[0].scrollHeight):n.scrollTop(n[0].scrollHeight*u)}return o.focus(),this},extend:function(){return void 0!==arguments[1]&&("function"==typeof arguments[1]&&(arguments[1]=t.proxy(arguments[1],this)),this[arguments[0]]=arguments[1]),"object"===_typeof(arguments[0])&&void 0===arguments[0].length&&t.extend(!0,this,arguments[0]),this},set:function(e,i){return void 0!==i&&"function"==typeof i&&(i=t.proxy(i,this)),this[e]=i,this},config:function(e,i){var o=this.settings;return"object"===_typeof(e)&&(o=t.extend(!0,o,e)),"string"==typeof e&&(o[e]=i),this.settings=o,this.recreate(),this},on:function(e,i){var o=this.settings;return void 0!==o["on"+e]&&(o["on"+e]=t.proxy(i,this)),this},off:function(e){var t=this.settings;return void 0!==t["on"+e]&&(t["on"+e]=function(){}),this},showToolbar:function(e){var i=this.settings;return i.readOnly?this:(i.toolbar&&(this.toolbar.length<1||""===this.toolbar.find("."+this.classPrefix+"menu").html())&&this.setToolbar(),i.toolbar=!0,this.toolbar.show(),this.resize(),t.proxy(e||function(){},this)(),this)},hideToolbar:function(e){return this.settings.toolbar=!1,this.toolbar.hide(),this.resize(),t.proxy(e||function(){},this)(),this},setToolbarAutoFixed:function(e){var i=this.state,o=this.editor,r=this.toolbar,n=this.settings;void 0!==e&&(n.toolbarAutoFixed=e);var a=function(){var e=t(window),i=e.scrollTop();if(!n.toolbarAutoFixed)return!1;i-o.offset().top>10&&i
                  ';t.append(n),r=this.toolbar=t.children("."+o+"toolbar")}if(!e.toolbar)return r.hide(),this;r.show();for(var a="function"==typeof e.toolbarIcons?e.toolbarIcons():"string"==typeof e.toolbarIcons?i.toolbarModes[e.toolbarIcons]:e.toolbarIcons,s=r.find("."+this.classPrefix+"menu"),l="",c=!1,h=0,d=a.length;h|';else{var f=/h(\d)/.test(u),p=u;"watch"!==u||e.watch||(p="unwatch");var g=e.lang.toolbar[p],m=e.toolbarIconTexts[p],w=e.toolbarIconsClass[p];g=void 0===g?"":g,m=void 0===m?"":m,w=void 0===w?"":w;var v=c?'
                • ':"
                • ";void 0!==e.toolbarCustomIcons[u]&&"function"!=typeof e.toolbarCustomIcons[u]?v+=e.toolbarCustomIcons[u]:(v+='',v+=''+(f?u.toUpperCase():""===w?m:"")+"",v+=""),v+="
                • ",l=c?v+l:l+v}}return s.html(l),s.find('[title="Lowercase"]').attr("title",e.lang.toolbar.lowercase),s.find('[title="ucwords"]').attr("title",e.lang.toolbar.ucwords),this.setToolbarHandler(),this.setToolbarAutoFixed(),this},dialogLockScreen:function(){return t.proxy(i.dialogLockScreen,this)(),this},dialogShowMask:function(e){return t.proxy(i.dialogShowMask,this)(e),this},getToolbarHandles:function(e){var t=this.toolbarHandlers=i.toolbarHandlers;return e&&void 0!==toolbarIconHandlers[e]?t[e]:t},setToolbarHandler:function(){var e=this,o=this.settings;if(!o.toolbar||o.readOnly)return this;var r=this.toolbar,n=this.cm,a=this.classPrefix,s=this.toolbarIcons=r.find("."+a+"menu > li > a"),l=this.getToolbarHandles();return s.bind(i.mouseOrTouch("click","touchend"),function(i){var r=t(this).children(".fa"),a=r.attr("name"),s=n.getCursor(),c=n.getSelection();if(""!==a)return e.activeIcon=r,void 0!==l[a]?t.proxy(l[a],e)(n):void 0!==o.toolbarHandlers[a]&&t.proxy(o.toolbarHandlers[a],e)(n,r,s,c),"link"!==a&&"reference-link"!==a&&"image"!==a&&"code-block"!==a&&"preformatted-text"!==a&&"watch"!==a&&"preview"!==a&&"search"!==a&&"fullscreen"!==a&&"info"!==a&&n.focus(),!1}),this},createDialog:function(e){return t.proxy(i.createDialog,this)(e)},createInfoDialog:function(){var e=this,t=this.editor,o=this.classPrefix,r=['
                  ','
                  ','

                  '+i.title+"v"+i.version+"

                  ","

                  "+this.lang.description+"

                  ",'

                  '+i.homePage+'

                  ','

                  Copyright © 2015 Pandao, The MIT License.

                  ',"
                  ",'',"
                  "].join("\n");t.append(r);var n=this.infoDialog=t.children("."+o+"dialog-info");return n.find("."+o+"dialog-close").bind(i.mouseOrTouch("click","touchend"),function(){e.hideInfoDialog()}),n.css("border",i.isIE8?"1px solid #ddd":"").css("z-index",i.dialogZindex).show(),this.infoDialogPosition(),this},infoDialogPosition:function(){var e=this.infoDialog,i=function(){e.css({top:(t(window).height()-e.height())/2+"px",left:(t(window).width()-e.width())/2+"px"})};return i(),t(window).resize(i),this},showInfoDialog:function(){t("html,body").css("overflow-x","hidden");var e=this.editor,o=this.settings,r=this.infoDialog=e.children("."+this.classPrefix+"dialog-info");return r.length<1&&this.createInfoDialog(),this.lockScreen(!0),this.mask.css({opacity:o.dialogMaskOpacity,backgroundColor:o.dialogMaskBgColor}).show(),r.css("z-index",i.dialogZindex).show(),this.infoDialogPosition(),this},hideInfoDialog:function(){return t("html,body").css("overflow-x",""),this.infoDialog.hide(),this.mask.hide(),this.lockScreen(!1),this},lockScreen:function(e){return i.lockScreen(e),this.resize(),this},recreate:function(){var e=this.editor,t=this.settings;return this.codeMirror.remove(),this.setCodeMirror(),t.readOnly||(e.find(".editormd-dialog").length>0&&e.find(".editormd-dialog").remove(),t.toolbar&&(this.getToolbarHandles(),this.setToolbar())),this.loadedDisplay(!0),this},previewCodeHighlight:function(){var e=this.settings,t=this.previewContainer;return e.previewCodeHighlight&&(t.find("pre").addClass("prettyprint linenums"),"undefined"!=typeof prettyPrint&&prettyPrint()),this},katexRender:function(){return null===o?this:(this.previewContainer.find("."+i.classNames.tex).each(function(){var e=t(this);i.$katex.render(e.text(),e[0]),e.find(".katex").css("font-size","1.6em")}),this)},flowChartAndSequenceDiagramRender:function(){var e=this,o=this.settings,n=this.previewContainer;if(i.isIE8)return this;if(o.flowChart){if(null===r)return this;n.find(".flowchart").flowChart()}o.sequenceDiagram&&n.find(".sequence-diagram").sequenceDiagram({theme:"simple"});var a=e.preview,s=e.codeMirror,l=s.find(".CodeMirror-scroll"),c=l.height(),h=l.scrollTop(),d=h/l[0].scrollHeight,u=0;a.find(".markdown-toc-list").each(function(){u+=t(this).height()});var f=a.find(".editormd-toc-menu").height();return f=f||0,o.syncScrolling&&(0===h?a.scrollTop(0):h+c>=l[0].scrollHeight-16?a.scrollTop(a[0].scrollHeight):a.scrollTop((a[0].scrollHeight+u+f)*d)),this},registerKeyMaps:function(e){var o=this,r=this.cm,n=this.settings,a=i.toolbarHandlers,s=n.disabledKeyMaps;if(e=e||null){for(var l in e)if(t.inArray(l,s)<0){var c={};c[l]=e[l],r.addKeyMap(e)}}else{for(var h in i.keyMaps){var d=i.keyMaps[h],u="string"==typeof d?t.proxy(a[d],o):t.proxy(d,o);if(t.inArray(h,["F9","F10","F11"])<0&&t.inArray(h,s)<0){var f={};f[h]=u,r.addKeyMap(f)}}t(window).keydown(function(e){var i={120:"F9",121:"F10",122:"F11"};if(t.inArray(i[e.keyCode],s)<0)switch(e.keyCode){case 120:return t.proxy(a.watch,o)(),!1;case 121:return t.proxy(a.preview,o)(),!1;case 122:return t.proxy(a.fullscreen,o)(),!1}})}return this},bindScrollEvent:function(){var e=this,o=this.preview,r=this.settings,n=this.codeMirror,a=i.mouseOrTouch;if(!r.syncScrolling)return this;var s=function(){n.find(".CodeMirror-scroll").bind(a("scroll","touchmove"),function(i){var n=t(this).height(),a=t(this).scrollTop(),s=a/t(this)[0].scrollHeight,l=0;o.find(".markdown-toc-list").each(function(){l+=t(this).height()});var c=o.find(".editormd-toc-menu").height();c=c||0,0===a?o.scrollTop(0):a+n>=t(this)[0].scrollHeight-16?o.scrollTop(o[0].scrollHeight):o.scrollTop((o[0].scrollHeight+l+c)*s),t.proxy(r.onscroll,e)(i)})},l=function(){n.find(".CodeMirror-scroll").unbind(a("scroll","touchmove"))},c=function(){o.bind(a("scroll","touchmove"),function(i){var o=t(this).height(),a=t(this).scrollTop(),s=a/t(this)[0].scrollHeight,l=n.find(".CodeMirror-scroll");0===a?l.scrollTop(0):a+o>=t(this)[0].scrollHeight?l.scrollTop(l[0].scrollHeight):l.scrollTop(l[0].scrollHeight*s),t.proxy(r.onpreviewscroll,e)(i)})},h=function(){o.unbind(a("scroll","touchmove"))};return n.bind({mouseover:s,mouseout:l,touchstart:s,touchend:l}),"single"===r.syncScrolling?this:(o.bind({mouseover:c,mouseout:h,touchstart:c,touchend:h}),this)},bindChangeEvent:function(){var e=this,t=this.cm,i=this.settings;return t.on("change",function(t,r){i.watch&&e.previewContainer.css("padding",i.autoHeight?"20px 20px 50px 40px":"20px"),o=setTimeout(function(){clearTimeout(o),e.save(),o=null},i.delay)}),this},loadedDisplay:function(e){e=e||!1;var i=this,o=this.editor,r=this.preview,n=this.settings;return this.containerMask.hide(),this.save(),n.watch&&r.show(),o.data("oldWidth",o.width()).data("oldHeight",o.height()),this.resize(),this.registerKeyMaps(),t(window).resize(function(){i.resize()}),this.bindScrollEvent().bindChangeEvent(),e||t.proxy(n.onload,this)(),this.state.loaded=!0,this},width:function(e){return this.editor.css("width","number"==typeof e?e+"px":e),this.resize(),this},height:function(e){return this.editor.css("height","number"==typeof e?e+"px":e),this.resize(),this},resize:function(e,i){e=e||null,i=i||null;var o=this.state,r=this.editor,n=this.preview,a=this.toolbar,s=this.settings,l=this.codeMirror;if(e&&r.css("width","number"==typeof e?e+"px":e),!s.autoHeight||o.fullscreen||o.preview?(i&&r.css("height","number"==typeof i?i+"px":i),o.fullscreen&&r.height(t(window).height()),s.toolbar&&!s.readOnly?l.css("margin-top",a.height()+1).height(r.height()-a.height()):l.css("margin-top",0).height(r.height())):(r.css({height:"auto",minHeight:s.minHeight}),l.css({height:"auto",minHeight:s.minHeight,maxHeight:s.maxHeight})),s.watch)if(l.width(r.width()/2),n.width(o.preview?r.width():r.width()/2),this.previewContainer.css("padding",s.autoHeight?"20px 20px 50px 40px":"20px"),s.toolbar&&!s.readOnly?n.css("top",a.height()+1):n.css("top",0),!s.autoHeight||o.fullscreen||o.preview){var c=s.toolbar&&!s.readOnly?r.height()-a.height():r.height();n.height(c)}else n.height("");else l.width(r.width()),n.hide();return o.loaded&&t.proxy(s.onresize,this)(),this},save:function(){if(null===o)return this;var e=this,n=this.state,a=this.settings,s=this.cm,l=s.getValue(),c=this.previewContainer;if("gfm"!==a.mode&&"markdown"!==a.mode)return this.markdownTextarea.val(l),this;var h=i.$marked,d=this.markdownToC=[],u=this.markedRendererOptions={toc:a.toc,tocm:a.tocm,tocStartLevel:a.tocStartLevel,pageBreak:a.pageBreak,taskList:a.taskList,emoji:a.emoji,tex:a.tex,atLink:a.atLink,emailLink:a.emailLink,flowChart:a.flowChart,sequenceDiagram:a.sequenceDiagram,previewCodeHighlight:a.previewCodeHighlight},f=this.markedOptions={renderer:i.markedRenderer(d,u),gfm:!0,tables:!0,breaks:!0,pedantic:!1,sanitize:!a.htmlDecode,smartLists:!0,smartypants:!0};h.setOptions(f);var p=i.$marked(l,f);if(p=i.filterHTMLTags(p,a.htmlDecode),this.markdownTextarea.text(l),s.save(),a.saveHTMLToTextarea&&this.htmlTextarea.text(p),a.watch||!a.watch&&n.preview){if(c.html(p),this.previewCodeHighlight(),a.toc){var g=""===a.tocContainer?c:t(a.tocContainer),m=g.find("."+this.classPrefix+"toc-menu");g.attr("previewContainer",""===a.tocContainer?"true":"false"),""!==a.tocContainer&&m.length>0&&m.remove(),i.markdownToCRenderer(d,g,a.tocDropdown,a.tocStartLevel),(a.tocDropdown||g.find("."+this.classPrefix+"toc-menu").length>0)&&i.tocDropdownMenu(g,""!==a.tocTitle?a.tocTitle:this.lang.tocTitle),""!==a.tocContainer&&c.find(".markdown-toc").css("border","none")}a.tex&&(!i.kaTeXLoaded&&a.autoLoadModules?i.loadKaTeX(function(){i.$katex=katex,i.kaTeXLoaded=!0,e.katexRender()}):(i.$katex=katex,this.katexRender())),(a.flowChart||a.sequenceDiagram)&&(r=setTimeout(function(){clearTimeout(r),e.flowChartAndSequenceDiagramRender(),r=null},10)),n.loaded&&t.proxy(a.onchange,this)()}return this},focus:function(){return this.cm.focus(),this},setCursor:function(e){return this.cm.setCursor(e),this},getCursor:function(){return this.cm.getCursor()},setSelection:function(e,t){return this.cm.setSelection(e,t),this},getSelection:function(){return this.cm.getSelection()},setSelections:function(e){return this.cm.setSelections(e),this},getSelections:function(){return this.cm.getSelections()},replaceSelection:function(e){return this.cm.replaceSelection(e),this},insertValue:function(e){return this.replaceSelection(e),this},appendMarkdown:function(e){var t=(this.settings,this.cm);return t.setValue(t.getValue()+e),this},setMarkdown:function(e){return this.cm.setValue(e||this.settings.markdown),this},getMarkdown:function(){return this.cm.getValue()},getValue:function(){return this.cm.getValue()},setValue:function(e){return this.cm.setValue(e),this},clear:function(){return this.cm.setValue(""),this},getHTML:function(){return this.settings.saveHTMLToTextarea?this.htmlTextarea.val():(alert("Error: settings.saveHTMLToTextarea == false"),!1)},getTextareaSavedHTML:function(){return this.getHTML()},getPreviewedHTML:function(){return this.settings.watch?this.previewContainer.html():(alert("Error: settings.watch == false"),!1)},watch:function(e){var i=this.settings;if(t.inArray(i.mode,["gfm","markdown"])<0)return this;if(this.state.watching=i.watch=!0,this.preview.show(),this.toolbar){var r=i.toolbarIconsClass.watch,n=i.toolbarIconsClass.unwatch,a=this.toolbar.find(".fa[name=watch]");a.parent().attr("title",i.lang.toolbar.watch),a.removeClass(n).addClass(r)}return this.codeMirror.css("border-right","1px solid #ddd").width(this.editor.width()/2),o=0,this.save().resize(),i.onwatch||(i.onwatch=e||function(){}),t.proxy(i.onwatch,this)(),this},unwatch:function(e){var i=this.settings;if(this.state.watching=i.watch=!1,this.preview.hide(),this.toolbar){var o=i.toolbarIconsClass.watch,r=i.toolbarIconsClass.unwatch,n=this.toolbar.find(".fa[name=watch]");n.parent().attr("title",i.lang.toolbar.unwatch),n.removeClass(o).addClass(r)}return this.codeMirror.css("border-right","none").width(this.editor.width()),this.resize(),i.onunwatch||(i.onunwatch=e||function(){}),t.proxy(i.onunwatch,this)(),this},show:function(e){e=e||function(){};var i=this;return this.editor.show(0,function(){t.proxy(e,i)()}),this},hide:function(e){e=e||function(){};var i=this;return this.editor.hide(0,function(){t.proxy(e,i)()}),this},previewing:function(){var e=this,o=this.editor,r=this.preview,n=this.toolbar,a=this.settings,s=this.codeMirror,l=this.previewContainer;if(t.inArray(a.mode,["gfm","markdown"])<0)return this;a.toolbar&&n&&(n.toggle(),n.find(".fa[name=preview]").toggleClass("active")),s.toggle();var c=function(t){t.shiftKey&&27===t.keyCode&&e.previewed()};"none"===s.css("display")?(this.state.preview=!0,this.state.fullscreen&&r.css("background","#fff"),o.find("."+this.classPrefix+"preview-close-btn").show().bind(i.mouseOrTouch("click","touchend"),function(){e.previewed()}),a.watch?l.css("padding",""):this.save(),l.addClass(this.classPrefix+"preview-active"),r.show().css({position:"",top:0,width:o.width(),height:a.autoHeight&&!this.state.fullscreen?"auto":o.height()}),this.state.loaded&&t.proxy(a.onpreviewing,this)(),t(window).bind("keyup",c)):(t(window).unbind("keyup",c),this.previewed())},previewed:function(){var e=this.editor,o=this.preview,r=this.toolbar,n=this.settings,a=this.previewContainer,s=e.find("."+this.classPrefix+"preview-close-btn");return this.state.preview=!1,this.codeMirror.show(),n.toolbar&&r.show(),o[n.watch?"show":"hide"](),s.hide().unbind(i.mouseOrTouch("click","touchend")),a.removeClass(this.classPrefix+"preview-active"),n.watch&&a.css("padding","20px"),o.css({background:null,position:"absolute",width:e.width()/2,height:n.autoHeight&&!this.state.fullscreen?"auto":e.height()-r.height(),top:n.toolbar?r.height():0}),this.state.loaded&&t.proxy(n.onpreviewed,this)(),this},fullscreen:function(){var e=this,i=this.state,o=this.editor,r=(this.preview,this.toolbar),n=this.settings,a=this.classPrefix+"fullscreen";r&&r.find(".fa[name=fullscreen]").parent().toggleClass("active");var s=function(t){t.shiftKey||27!==t.keyCode||i.fullscreen&&e.fullscreenExit()};return o.hasClass(a)?(t(window).unbind("keyup",s),this.fullscreenExit()):(i.fullscreen=!0,t("html,body").css("overflow","hidden"),o.css({width:t(window).width(),height:t(window).height()}).addClass(a),this.resize(),t.proxy(n.onfullscreen,this)(),t(window).bind("keyup",s)),this},fullscreenExit:function(){var e=this.editor,i=this.settings,o=this.toolbar,r=this.classPrefix+"fullscreen";return this.state.fullscreen=!1,o&&o.find(".fa[name=fullscreen]").parent().removeClass("active"),t("html,body").css("overflow",""),e.css({width:e.data("oldWidth"),height:e.data("oldHeight")}).removeClass(r),this.resize(),t.proxy(i.onfullscreenExit,this)(),this},executePlugin:function(e,o){var r=this,n=this.cm;return o=this.settings.pluginPath+o,"function"==typeof define?void 0===this[e]?(alert("Error: "+e+" plugin is not found, you are not load this plugin."),this):(this[e](n),this):(t.inArray(o,i.loadFiles.plugin)<0?i.loadPlugin(o,function(){i.loadPlugins[e]=r[e],r[e](n)}):t.proxy(i.loadPlugins[e],this)(n),this)},search:function(e){var t=this.settings;return t.searchReplace?(t.readOnly||this.cm.execCommand(e||"find"),this):(alert("Error: settings.searchReplace == false"),this)},searchReplace:function(){return this.search("replace"),this},searchReplaceAll:function(){return this.search("replaceAll"),this}},i.fn.init.prototype=i.fn,i.dialogLockScreen=function(){(this.settings||{dialogLockScreen:!0}).dialogLockScreen&&(t("html,body").css("overflow","hidden"),this.resize())},i.dialogShowMask=function(e){var i=this.editor,o=this.settings||{dialogShowMask:!0};e.css({top:(t(window).height()-e.height())/2+"px",left:(t(window).width()-e.width())/2+"px"}),o.dialogShowMask&&i.children("."+this.classPrefix+"mask").css("z-index",parseInt(e.css("z-index"))-1).show()},i.toolbarHandlers={undo:function(){this.cm.undo()},redo:function(){this.cm.redo()},bold:function(){var e=this.cm,t=e.getCursor(),i=e.getSelection();e.replaceSelection("**"+i+"**"),""===i&&e.setCursor(t.line,t.ch+2)},del:function(){var e=this.cm,t=e.getCursor(),i=e.getSelection();e.replaceSelection("~~"+i+"~~"),""===i&&e.setCursor(t.line,t.ch+2)},italic:function(){var e=this.cm,t=e.getCursor(),i=e.getSelection();e.replaceSelection("*"+i+"*"),""===i&&e.setCursor(t.line,t.ch+1)},quote:function(){var e=this.cm,t=e.getCursor(),i=e.getSelection();0!==t.ch?(e.setCursor(t.line,0),e.replaceSelection("> "+i),e.setCursor(t.line,t.ch+2)):e.replaceSelection("> "+i)},ucfirst:function(){var e=this.cm,t=e.getSelection(),o=e.listSelections();e.replaceSelection(i.firstUpperCase(t)),e.setSelections(o)},ucwords:function(){var e=this.cm,t=e.getSelection(),o=e.listSelections();e.replaceSelection(i.wordsFirstUpperCase(t)),e.setSelections(o)},uppercase:function(){var e=this.cm,t=e.getSelection(),i=e.listSelections();e.replaceSelection(t.toUpperCase()),e.setSelections(i)},lowercase:function(){var e=this.cm,t=(e.getCursor(),e.getSelection()),i=e.listSelections();e.replaceSelection(t.toLowerCase()),e.setSelections(i)},h1:function(){var e=this.cm,t=e.getCursor(),i=e.getSelection();0!==t.ch?(e.setCursor(t.line,0),e.replaceSelection("# "+i),e.setCursor(t.line,t.ch+2)):e.replaceSelection("# "+i)},h2:function(){var e=this.cm,t=e.getCursor(),i=e.getSelection();0!==t.ch?(e.setCursor(t.line,0),e.replaceSelection("## "+i),e.setCursor(t.line,t.ch+3)):e.replaceSelection("## "+i)},h3:function(){var e=this.cm,t=e.getCursor(),i=e.getSelection();0!==t.ch?(e.setCursor(t.line,0),e.replaceSelection("### "+i),e.setCursor(t.line,t.ch+4)):e.replaceSelection("### "+i)},h4:function(){var e=this.cm,t=e.getCursor(),i=e.getSelection();0!==t.ch?(e.setCursor(t.line,0),e.replaceSelection("#### "+i),e.setCursor(t.line,t.ch+5)):e.replaceSelection("#### "+i)},h5:function(){var e=this.cm,t=e.getCursor(),i=e.getSelection();0!==t.ch?(e.setCursor(t.line,0),e.replaceSelection("##### "+i),e.setCursor(t.line,t.ch+6)):e.replaceSelection("##### "+i)},h6:function(){var e=this.cm,t=e.getCursor(),i=e.getSelection();0!==t.ch?(e.setCursor(t.line,0),e.replaceSelection("###### "+i),e.setCursor(t.line,t.ch+7)):e.replaceSelection("###### "+i)},"list-ul":function(){var e=this.cm,t=(e.getCursor(),e.getSelection());if(""===t)e.replaceSelection("- "+t);else{for(var i=t.split("\n"),o=0,r=i.length;o=0,a=n?"Cmd":"Ctrl";i.keyMaps=(e={},_defineProperty(e,a+"-1","h1"),_defineProperty(e,a+"-2","h2"),_defineProperty(e,a+"-3","h3"),_defineProperty(e,a+"-4","h4"),_defineProperty(e,a+"-5","h5"),_defineProperty(e,a+"-6","h6"),_defineProperty(e,a+"-B","bold"),_defineProperty(e,a+"-D","datetime"),_defineProperty(e,a+"Ctrl-E",function(){var e=this.cm,t=e.getCursor(),i=e.getSelection();if(!this.settings.emoji)return void alert("Error: settings.emoji == false");e.replaceSelection(":"+i+":"),""===i&&e.setCursor(t.line,t.ch+1)}),_defineProperty(e,a+"-Alt-G","goto-line"),_defineProperty(e,a+"-H","hr"),_defineProperty(e,a+"-I","italic"),_defineProperty(e,a+"-K","code"),_defineProperty(e,"Ctrl-L",function(){var e=this.cm,t=e.getCursor(),i=e.getSelection(),o=""===i?"":' "'+i+'"';e.replaceSelection("["+i+"]("+o+")"),""===i&&e.setCursor(t.line,t.ch+1)}),_defineProperty(e,a+"-U","list-ul"),_defineProperty(e,"Shift-Ctrl-A",function(){var e=this.cm,t=e.getCursor(),i=e.getSelection();if(!this.settings.atLink)return void alert("Error: settings.atLink == false");e.replaceSelection("@"+i),""===i&&e.setCursor(t.line,t.ch+1)}),_defineProperty(e,"Shift"+a+"-C","code"),_defineProperty(e,"Shift"+a+"Q","quote"),_defineProperty(e,"Shift"+a+"S","del"),_defineProperty(e,"Shift"+a+"K","tex"),_defineProperty(e,"Shift-Alt-C",function(){var e=this.cm,t=e.getCursor(),i=e.getSelection();e.replaceSelection(["```",i,"```"].join("\n")),""===i&&e.setCursor(t.line,t.ch+3)}),_defineProperty(e,"Shift-"+a+"-Alt-C","code-block"),_defineProperty(e,"Shift-"+a+"-H","html-entities"),_defineProperty(e,"Shift-Alt-H","help"),_defineProperty(e,"Shift-"+a+"-E","emoji"),_defineProperty(e,"Shift-"+a+"-U","uppercase"),_defineProperty(e,"Shift-Alt-U","ucwords"),_defineProperty(e,"Shift-"+a+"-Alt-U","ucfirst"),_defineProperty(e,"Shift-Alt-L","lowercase"),_defineProperty(e,"Shift-"+a+"-I",function(){var e=this.cm,t=e.getCursor(),i=e.getSelection(),o=""===i?"":' "'+i+'"';e.replaceSelection("!["+i+"]("+o+")"),""===i&&e.setCursor(t.line,t.ch+4)}),_defineProperty(e,"Shift-"+a+"-Alt-I","image"),_defineProperty(e,"Shift-"+a+"-L","link"),_defineProperty(e,"Shift-"+a+"-O","list-ol"),_defineProperty(e,"Shift-"+a+"-P","preformatted-text"),_defineProperty(e,"Shift-"+a+"-T","table"),_defineProperty(e,"Shift-Alt-P","pagebreak"),_defineProperty(e,"F9","watch"),_defineProperty(e,"F10","preview"),_defineProperty(e,"F11","fullscreen"),e);var s=function(e){return String.prototype.trim?e.trim():e.replace(/^[\s\uFEFF\xA0]+|[\s\uFEFF\xA0]+$/g,"")};i.trim=s;var l=function(e){return e.toLowerCase().replace(/\b(\w)|\s(\w)/g,function(e){return e.toUpperCase()})};i.ucwords=i.wordsFirstUpperCase=l;var c=function(e){return e.toLowerCase().replace(/\b(\w)/,function(e){return e.toUpperCase()})};return i.firstUpperCase=i.ucfirst=c,i.urls={atLinkBase:"https://github.com/"},i.regexs={atLink:/@(\w+)/g,email:/(\w+)@(\w+)\.(\w+)\.?(\w+)?/g,emailLink:/(mailto:)?([\w\.\_]+)@(\w+)\.(\w+)\.?(\w+)?/g,emoji:/:([\w\+-]+):/g,emojiDatetime:/(\d{1,2}:\d{1,2}:\d{1,2})/g,twemoji:/:(tw-([\w]+)-?(\w+)?):/g,fontAwesome:/:(fa-([\w]+)(-(\w+)){0,}):/g,editormdLogo:/:(editormd-logo-?(\w+)?):/g,pageBreak:/^\[[=]{8,}\]$/},i.emoji={path:"http://www.emoji-cheat-sheet.com/graphics/emojis/",ext:".png"},i.twemoji={path:"http://twemoji.maxcdn.com/36x36/",ext:".png"},i.markedRenderer=function(e,o){var r={toc:!0,tocm:!1,tocStartLevel:1,pageBreak:!0,atLink:!0,emailLink:!0,taskList:!1,emoji:!1,tex:!1,flowChart:!1,sequenceDiagram:!1},n=t.extend(r,o||{}),a=i.$marked,l=new a.Renderer;e=e||[];var c=i.regexs,h=c.atLink,d=c.emoji,u=c.email,f=c.emailLink,p=c.twemoji,g=c.fontAwesome,m=c.editormdLogo,w=c.pageBreak;return l.emoji=function(e){e=e.replace(i.regexs.emojiDatetime,function(e){return e.replace(/:/g,":")});var t=e.match(d);if(!t||!n.emoji)return e;for(var o=0,r=t.length;o'}else{var l=e.match(m),c=e.match(p);if(l)for(var h=0,d=l.length;h'}else{if(!c){var f="+1"===r?"plus1":r;return f="black_large_square"===f?"black_square":f,f="moon"===f?"waxing_gibbous_moon":f,':'+r+':'}for(var w=0,v=c.length;w'}}}});return e},l.atLink=function(e){return h.test(e)?(n.atLink&&(e=e.replace(u,function(e,t,i,o){return e.replace(/@/g,"_#_@_#_")}),e=e.replace(h,function(e,t){return''+e+""}).replace(/_#_@_#_/g,"@")),n.emailLink&&(e=e.replace(f,function(e,i,o,r,n){return!i&&t.inArray(n,"jpg|jpeg|png|gif|webp|ico|icon|pdf".split("|"))<0?''+e+"":e})),e):e},l.link=function(e,t,i){if(this.options.sanitize)try{if(0===decodeURIComponent(unescape(e)).replace(/[^\w:]/g,"").toLowerCase().indexOf("javascript:"))return""}catch(e){return""}var o=''+i.replace(/@/g,"@")+""):(t&&(o+=' title="'+t+'"'),o+=">"+i+"")},l.heading=function(t,i,o){var r=t,n=/\s*]*)\>(.*)<\/a\>\s*/;if(n.test(t)){var a=[];t=t.split(/]+)\>([^\>]*)<\/a\>/);for(var l=0,c=t.length;l';return p+='',p+='',p+=n?this.atLink(this.emoji(r)):this.atLink(this.emoji(t)),p+=""},l.pageBreak=function(e){return w.test(e)&&n.pageBreak&&(e='
                  '),e},l.paragraph=function(e){var t=/\$\$(.*)\$\$/g.test(e),o=/^\$\$(.*)\$\$$/.test(e),r=o?' class="'+i.classNames.tex+'"':"",a=n.tocm?/^(\[TOC\]|\[TOCM\])$/.test(e):/^\[TOC\]$/.test(e),s=/^\[TOCM\]$/.test(e);e=!o&&t?e.replace(/(\$\$([^\$]*)\$\$)+/g,function(e,t){return''+t.replace(/\$/g,"")+""}):o?e.replace(/\$/g,""):e;var l='
                  '+e+"
                  ";return a?s?'
                  '+l+"

                  ":l:w.test(e)?this.pageBreak(e):""+this.atLink(this.emoji(e))+"

                  \n"},l.code=function(e,t,o){return"seq"===t||"sequence"===t?'
                  '+e+"
                  ":"flow"===t?'
                  '+e+"
                  ":"math"===t||"latex"===t||"katex"===t?'

                  '+e+"

                  ":a.Renderer.prototype.code.apply(this,arguments)},l.tablecell=function(e,t){var i=t.header?"th":"td";return(t.align?"<"+i+' style="text-align:'+t.align+'">':"<"+i+">")+this.atLink(this.emoji(e))+"\n"},l.listitem=function(e){return n.taskList&&/^\s*\[[x\s]\]\s*/.test(e)?(e=e.replace(/^\s*\[\s\]\s*/,' ').replace(/^\s*\[x\]\s*/,' '),'
                • '+this.atLink(this.emoji(e))+"
                • "):"
                • "+this.atLink(this.emoji(e))+"
                • "},l},i.markdownToCRenderer=function(e,t,i,o){var r="",n=[0],a=this.classPrefix;o=o||1;for(var s=0,l=e.length;su)r+="",n.push(h);else if(h=0;p--){if(!(h<=n[p])){n.push(h);break}f++,n.pop()}r+=new Array(f+1).join("
              • ")}else r+="";r+='
              • '+c+"
                  "}}var g=t.find(".markdown-toc");if(g.length<1&&"false"===t.attr("previewContainer")){var m='
                  ';m=i?'
                  '+m+"
                  ":m,t.html(m),g=t.find(".markdown-toc")}return i&&g.wrap('

                  '),g.html('
                    ').children(".markdown-toc-list").html(r.replace(/\r?\n?<\/ul\>/g,"")),g},i.tocDropdownMenu=function(e,i){i=i||"Table of Contents";var o=400,r=e.find("."+this.classPrefix+"toc-menu");return r.each(function(){var e=t(this),r=e.children(".markdown-toc"),n='',a=''+n+i+"",s=r.children("ul"),l=s.find("li");r.append(a),l.first().before("
                  • "+i+" "+n+"

                  • "),e.mouseover(function(){s.show(),l.each(function(){var e=t(this),i=e.children("ul");if(""===i.html()&&i.remove(),i.length>0&&""!==i.html()){var r=e.children("a").first();r.children(".fa").length<1&&r.append(t(n).css({float:"right",paddingTop:"4px"}))}e.mouseover(function(){i.css("z-index",o).show(),o+=1}).mouseleave(function(){i.hide()})})}).mouseleave(function(){s.hide()})}),r},i.filterHTMLTags=function(e,i){if("string"!=typeof e&&(e=new String(e)),"string"!=typeof i)return e;for(var o=i.split("|"),r=o[0].split(","),n=o[1],a=0,s=r.length;a]*)>([^>]*)","igm"),"")}if(void 0!==n){var c=/<(\w+)\s*([^\>]*)\>([^\>]*)<\/(\w+)\>/gi;e="*"===n?e.replace(c,function(e,t,i,o,r){return"<"+t+">"+o+""}):"on*"===n?e.replace(c,function(e,i,o,r,n){var a=t("<"+i+">"+r+""),s=t(e)[0].attributes,l={};t.each(s,function(e,t){'"'!==t.nodeName&&(l[t.nodeName]=t.nodeValue,"href"===t.nodeName&&t.nodeValue.toLowerCase().indexOf("javascript:")>=0&&(l[t.nodeName]="javascript:;"))}),t.each(l,function(e){0===e.indexOf("on")&&delete l[e]}),a.attr(l);var c=void 0!==a[1]?t(a[1]).text():"";if(!i||isNaN(i)||i===n)return i&&isNaN(i)&&i!==n?(c+="",a[0].outerHTML+c):a[0].outerHTML+c;c+="<"+i+">"+r+""}):e.replace(c,function(e,i,o,r){var a=n.split(","),s=t(e);return s.html(r),t.each(a,function(e){s.attr(a[e],null)}),s[0].outerHTML})}return e},i.markdownToHTML=function(e,o){var r={gfm:!0,toc:!0,tocm:!1,tocStartLevel:1,tocTitle:"目录",tocDropdown:!1,tocContainer:"",markdown:"",markdownSourceCode:!1,htmlDecode:!1,autoLoadKaTeX:!0,pageBreak:!0,atLink:!0,emailLink:!0,tex:!1,taskList:!1,emoji:!1,flowChart:!1,sequenceDiagram:!1,previewCodeHighlight:!0};i.$marked=marked;var n=t("#"+e),a=n.settings=t.extend(!0,r,o||{}),s=n.find("textarea");s.length<1&&(n.append(""),s=n.find("textarea"));var l=""===a.markdown?s.val():a.markdown,c=[],h={toc:a.toc,tocm:a.tocm,tocStartLevel:a.tocStartLevel,taskList:a.taskList,emoji:a.emoji,tex:a.tex,pageBreak:a.pageBreak,atLink:a.atLink,emailLink:a.emailLink,flowChart:a.flowChart,sequenceDiagram:a.sequenceDiagram,previewCodeHighlight:a.previewCodeHighlight},d={renderer:i.markedRenderer(c,h),gfm:a.gfm,tables:!0,breaks:!0,pedantic:!1,sanitize:!a.htmlDecode,smartLists:!0,smartypants:!0};l=new String(l);var u=marked(l,d);u=i.filterHTMLTags(u,a.htmlDecode),a.markdownSourceCode?s.text(l):s.remove(),n.addClass("markdown-body "+this.classPrefix+"html-preview").append(u);var f=""!==a.tocContainer?t(a.tocContainer):n;if(""!==a.tocContainer&&f.attr("previewContainer",!1),a.toc&&(n.tocContainer=this.markdownToCRenderer(c,f,a.tocDropdown,a.tocStartLevel),(a.tocDropdown||n.find("."+this.classPrefix+"toc-menu").length>0)&&this.tocDropdownMenu(n,a.tocTitle),""!==a.tocContainer&&n.find(".editormd-toc-menu, .editormd-markdown-toc").remove()),a.previewCodeHighlight&&(n.find("pre").addClass("prettyprint linenums"),prettyPrint()),i.isIE8||(a.flowChart&&n.find(".flowchart").flowChart(),a.sequenceDiagram&&n.find(".sequence-diagram").sequenceDiagram({theme:"simple"})),a.tex){var p=function(){n.find("."+i.classNames.tex).each(function(){var e=t(this);katex.render(e.html().replace(/</g,"<").replace(/>/g,">"),e[0]),e.find(".katex").css("font-size","1.6em")})};!a.autoLoadKaTeX||i.$katex||i.kaTeXLoaded?p():this.loadKaTeX(function(){i.$katex=katex,i.kaTeXLoaded=!0,p()})}return n.getMarkdown=function(){return s.val()},n},i.themes=["default","dark"],i.previewThemes=["default","dark"],i.editorThemes=["default","3024-day","3024-night","ambiance","ambiance-mobile","base16-dark","base16-light","blackboard","cobalt","eclipse","elegant","erlang-dark","lesser-dark","mbo","mdn-like","midnight","monokai","neat","neo","night","paraiso-dark","paraiso-light","pastel-on-dark","rubyblue","solarized","the-matrix","tomorrow-night-eighties","twilight","vibrant-ink","xq-dark","xq-light"],i.loadPlugins={},i.loadFiles={js:[],css:[],plugin:[]},i.loadPlugin=function(e,t,o){t=t||function(){},this.loadScript(e,function(){i.loadFiles.plugin.push(e),t()},o)},i.loadCSS=function(e,t,o){o=o||"head",t=t||function(){};var r=document.createElement("link");r.type="text/css",r.rel="stylesheet",r.onload=r.onreadystatechange=function(){i.loadFiles.css.push(e),t()},r.href=e+".css","head"===o?document.getElementsByTagName("head")[0].appendChild(r):document.body.appendChild(r)},i.isIE="Microsoft Internet Explorer"===navigator.appName,i.isIE8=i.isIE&&"8."===navigator.appVersion.match(/8./i),i.loadScript=function(e,t,o){o=o||"head",t=t||function(){};var r=null;r=document.createElement("script"),r.id=e.replace(/[\.\/]+/g,"-"),r.type="text/javascript",r.src=e+".js",i.isIE8?r.onreadystatechange=function(){r.readyState&&("loaded"!==r.readyState&&"complete"!==r.readyState||(r.onreadystatechange=null,i.loadFiles.js.push(e),t()))}:r.onload=function(){i.loadFiles.js.push(e),t()},"head"===o?document.getElementsByTagName("head")[0].appendChild(r):document.body.appendChild(r)},i.katexURL={css:"//cdnjs.cloudflare.com/ajax/libs/KaTeX/0.3.0/katex.min",js:"//cdnjs.cloudflare.com/ajax/libs/KaTeX/0.3.0/katex.min"},i.kaTeXLoaded=!1,i.loadKaTeX=function(e){i.loadCSS(i.katexURL.css,function(){i.loadScript(i.katexURL.js,e||function(){})})},i.lockScreen=function(e){t("html,body").css("overflow",e?"hidden":"")},i.createDialog=function(e){var o={name:"",width:420,height:240,title:"",drag:!0,closed:!0,content:"",mask:!0,maskStyle:{backgroundColor:"#fff",opacity:.1},lockScreen:!0,footer:!0,buttons:!1};e=t.extend(!0,o,e);var r=this,n=this.editor,a=i.classPrefix,s=(new Date).getTime(),l=""===e.name?a+"dialog-"+s:e.name,c=i.mouseOrTouch,h='
                    ';""!==e.title&&(h+='
                    ",h+=''+e.title+"",h+="
                    "),e.closed&&(h+=''),h+='
                    '+e.content,(e.footer||"string"==typeof e.footer)&&(h+='"),h+="
                    ",h+='
                    ',h+='
                    ',h+="
                    ",n.append(h);var d=n.find("."+l);d.lockScreen=function(i){return e.lockScreen&&(t("html,body").css("overflow",i?"hidden":""),r.resize()),d},d.showMask=function(){return e.mask&&n.find("."+a+"mask").css(e.maskStyle).css("z-index",i.dialogZindex-1).show(),d},d.hideMask=function(){return e.mask&&n.find("."+a+"mask").hide(),d},d.loading=function(e){return d.find("."+a+"dialog-mask")[e?"show":"hide"](),d},d.lockScreen(!0).showMask(),d.show().css({zIndex:i.dialogZindex,border:i.isIE8?"1px solid #ddd":"",width:"number"==typeof e.width?e.width+"px":e.width,height:"number"==typeof e.height?e.height+"px":e.height});var u=function(){d.css({top:(t(window).height()-d.height())/2+"px",left:(t(window).width()-d.width())/2+"px"})};if(u(),t(window).resize(u),d.children("."+a+"dialog-close").bind(c("click","touchend"),function(){d.hide().lockScreen(!1).hideMask()}),"object"===_typeof(e.buttons)){var f=d.footer=d.find("."+a+"dialog-footer");for(var p in e.buttons){var g=e.buttons[p],m=a+p+"-btn";f.append('"),g[1]=t.proxy(g[1],d),f.children("."+m).bind(c("click","touchend"),g[1])}}if(""!==e.title&&e.drag){var w,v,k=d.children("."+a+"dialog-header");e.mask||k.bind(c("click","touchend"),function(){i.dialogZindex+=2,d.css("z-index",i.dialogZindex)});var y=function(e){e.removeClass(a+"user-unselect").off("selectstart")},b=function(e){e.addClass(a+"user-unselect").on("selectstart",function(e){return!1})},x=function(e){e=e||window.event;var i,o,r=parseInt(d[0].style.left),n=parseInt(d[0].style.top);r>=0?r+d.width()<=t(window).width()?i=e.clientX-w:(i=t(window).width()-d.width(),document.onmousemove=null):(i=0,document.onmousemove=null),n>=0?o=e.clientY-v:(o=0,document.onmousemove=null),document.onselectstart=function(){return!1},b(t("body")),b(d),d[0].style.left=i+"px",d[0].style.top=o+"px"};k.mousedown(function(e){e=e||window.event,w=e.clientX-parseInt(d[0].style.left),v=e.clientY-parseInt(d[0].style.top),document.onmousemove=x}),document.onmouseup=function(){y(t("body")),y(d),document.onselectstart=null,document.onmousemove=null},k.touchDraggable=function(){var e=null,i=function(i){var o=i.originalEvent,r=t(this).parent().position();e={x:o.changedTouches[0].pageX-r.left,y:o.changedTouches[0].pageY-r.top}},o=function(i){i.preventDefault();var o=i.originalEvent;t(this).parent().css({top:o.changedTouches[0].pageY-e.y,left:o.changedTouches[0].pageX-e.x})};this.bind("touchstart",i).bind("touchmove",o)},k.touchDraggable()}return i.dialogZindex+=2,d},i.mouseOrTouch=function(e,t){e=e||"click",t=t||"touchend";var i=e;try{document.createEvent("TouchEvent"),i=t}catch(e){}return i},i.dateFormat=function(e){e=e||"";var t=function(e){return e<10?"0"+e:e},i=new Date,o=i.getFullYear(),r=o.toString().slice(2,4),n=t(i.getMonth()+1),a=t(i.getDate()),s=i.getDay(),l=t(i.getHours()),c=t(i.getMinutes()),h=t(i.getSeconds()),d=t(i.getMilliseconds()),u="",f=r+"-"+n+"-"+a,p=o+"-"+n+"-"+a,g=l+":"+c+":"+h;switch(e){case"UNIX Time":u=i.getTime();break;case"UTC":u=i.toUTCString();break;case"yy":u=r;break;case"year":case"yyyy":u=o;break;case"month":case"mm":u=n;break;case"cn-week-day":case"cn-wd":u="星期"+["日","一","二","三","四","五","六"][s];break;case"week-day":case"wd":u=["Sunday","Monday","Tuesday","Wednesday","Thursday","Friday","Saturday"][s];break;case"day":case"dd":u=a;break;case"hour":case"hh":u=l;break;case"min":case"ii":u=c;break;case"second":case"ss":u=h;break;case"ms":u=d;break;case"yy-mm-dd":u=f;break;case"yyyy-mm-dd":u=p;break;case"yyyy-mm-dd h:i:s ms":case"full + ms":u=p+" "+g+" "+d;break;case"full":case"yyyy-mm-dd h:i:s":break;default:u=p+" "+g}return u},i}}); \ No newline at end of file diff --git a/system/admin/editor.md/examples/@links.html b/system/admin/editor.md/examples/@links.html new file mode 100644 index 00000000..2cc6a106 --- /dev/null +++ b/system/admin/editor.md/examples/@links.html @@ -0,0 +1,135 @@ + + + + + @links - Editor.md examples + + + + + +
                    +
                    +

                    @links

                    +

                    Github Flavored Markdown extras syntax

                    +
                    +
                    + +
                    +
                    + + + + + \ No newline at end of file diff --git a/system/admin/editor.md/examples/auto-height.html b/system/admin/editor.md/examples/auto-height.html new file mode 100644 index 00000000..4a37c431 --- /dev/null +++ b/system/admin/editor.md/examples/auto-height.html @@ -0,0 +1,55 @@ + + + + + Auto height - Editor.md examples + + + + + +
                    +
                    +

                    Auto height test

                    +
                    +
                    + +
                    +
                    + +
                    +
                    + + + + + \ No newline at end of file diff --git a/system/admin/editor.md/examples/change-mode.html b/system/admin/editor.md/examples/change-mode.html new file mode 100644 index 00000000..e25798af --- /dev/null +++ b/system/admin/editor.md/examples/change-mode.html @@ -0,0 +1,508 @@ + + + + + Chnage mode - Editor.md examples + + + + + + +
                    +
                    +

                    Chnage mode

                    +

                    Become to the code editor

                    +

                    Modes :   Themes : + +

                    +
                    +
                    + + +
                    +
                    + + + + + + + + + + +
                    +
                    + +
                    +
                    + + + + + \ No newline at end of file diff --git a/system/admin/editor.md/examples/code-fold.html b/system/admin/editor.md/examples/code-fold.html new file mode 100644 index 00000000..e2774bcc --- /dev/null +++ b/system/admin/editor.md/examples/code-fold.html @@ -0,0 +1,44 @@ + + + + + Code folding - Editor.md examples + + + + + +
                    +
                    +

                    Code folding

                    +

                    Switch code folding : Press Ctrl + Q / Command + Q

                    +
                    +
                    + +
                    +
                    + + + + + \ No newline at end of file diff --git a/system/admin/editor.md/examples/css/style.css b/system/admin/editor.md/examples/css/style.css new file mode 100644 index 00000000..0150e3b9 --- /dev/null +++ b/system/admin/editor.md/examples/css/style.css @@ -0,0 +1,94 @@ +* { + padding: 0; + margin: 0; +} + +*, *:before, *:after { + -webkit-box-sizing: border-box; + -moz-box-sizing: border-box; + box-sizing: border-box; +} + body,div,dl,dt,dd,ul,ol,li,h1,h2,h3,h4,h5,h6,pre,code,form,fieldset,legend,input,textarea,p,blockquote,th,td,hr,button,article,aside,details,figcaption,figure,footer,header,hgroup,menu,nav,section{ + margin: 0; + padding: 0; +} + +article, aside, details, figcaption, figure, footer, header, hgroup, menu, nav, section, summary { + display: block; +} + +audio, canvas, video { + display: inline-block; +} + +img { + border: none; + vertical-align: middle; +} + +ul, ol { + /*list-style: none;*/ +} + +.clear { + *zoom: 1; /* for IE 6/7 */ +} + +.clear:before, .clear:after { + height: 0; + content: ""; + font-size: 0; + display: table; + line-height: 0; /* for Opera */ + visibility: hidden; +} + +.clear:after { + clear: both; +} + +body { + font-size: 14px; + color: #666; + font-family: "Microsoft YaHei", "微软雅黑", Helvetica, Tahoma, STXihei, "华文细黑", STHeiti, "Helvetica Neue", Helvetica, Tahoma, "Droid Sans", "wenquanyi micro hei", FreeSans, Arimo, Arial, SimSun, "宋体", Heiti, "黑体", sans-serif; + background: #fff; + text-align: center; +} + +#layout { + text-align: left; +} + +#layout > header, .btns { + padding: 15px 0; + width: 90%; + margin: 0 auto; +} + +.btns { + padding-top: 0; +} + +.btns button { + padding: 2px 8px; +} + +#layout > header > h1 { + font-size: 20px; + margin-bottom: 10px; +} + +.btns button, .btn { + padding: 8px 10px; + background: #fff; + border: 1px solid #ddd; + -webkit-border-radius: 3px; + border-radius: 3px; + cursor: pointer; + -webkit-transition: background 300ms ease-out; + transition: background 300ms ease-out; +} + +.btns button:hover, .btn:hover { + background: #f6f6f6; +} \ No newline at end of file diff --git a/system/admin/editor.md/examples/custom-keyboard-shortcuts.html b/system/admin/editor.md/examples/custom-keyboard-shortcuts.html new file mode 100644 index 00000000..3afc27ba --- /dev/null +++ b/system/admin/editor.md/examples/custom-keyboard-shortcuts.html @@ -0,0 +1,118 @@ + + + + + Custom keyboard shortcuts - Editor.md examples + + + + + +
                    +
                    +

                    Custom keyboard shortcuts

                    +
                    +
                    + +
                    +
                    + + + + + \ No newline at end of file diff --git a/system/admin/editor.md/examples/custom-toolbar.html b/system/admin/editor.md/examples/custom-toolbar.html new file mode 100644 index 00000000..89177dae --- /dev/null +++ b/system/admin/editor.md/examples/custom-toolbar.html @@ -0,0 +1,178 @@ + + + + + 自定义工具栏 - Editor.md examples + + + + + +
                    +
                    +

                    自定义工具栏

                    +

                    Custom toolbar (icons handler)

                    +
                    +
                    + +
                    +
                    + + + + + + + + \ No newline at end of file diff --git a/system/admin/editor.md/examples/define-plugin.html b/system/admin/editor.md/examples/define-plugin.html new file mode 100644 index 00000000..8c867e72 --- /dev/null +++ b/system/admin/editor.md/examples/define-plugin.html @@ -0,0 +1,151 @@ + + + + + Define extention plugins for Editor.md - Editor.md examples + + + + + +
                    +
                    +

                    Define extention plugins for Editor.md

                    +
                    +
                    + +
                    +
                    + + + + + \ No newline at end of file diff --git a/system/admin/editor.md/examples/delay-renderer-preview.html b/system/admin/editor.md/examples/delay-renderer-preview.html new file mode 100644 index 00000000..ad343a2a --- /dev/null +++ b/system/admin/editor.md/examples/delay-renderer-preview.html @@ -0,0 +1,56 @@ + + + + + Delay Rerender & Preview - Editor.md examples + + + + + +
                    +
                    +

                    Delay Rerender & Preview

                    +

                    P.S. If you input the content too much and too fast, You can setting the delay value.

                    +

                    P.S. 适用于输入内容太多太快的情形,但要是一个合理的值,不然会显得预览太慢。打字慢会相对显得慢,打字快时则相对显得快。

                    +
                    +
                    + +
                    +
                    + + + + + \ No newline at end of file diff --git a/system/admin/editor.md/examples/dynamic-create-editormd.html b/system/admin/editor.md/examples/dynamic-create-editormd.html new file mode 100644 index 00000000..5644e098 --- /dev/null +++ b/system/admin/editor.md/examples/dynamic-create-editormd.html @@ -0,0 +1,47 @@ + + + + + 动态创建 Editor.md - Editor.md examples + + + + + +
                    +
                    +

                    动态创建 Editor.md

                    +

                    Dynamic create Editor.md

                    +
                    +
                    + + +
                    +
                    +
                    + + + + + \ No newline at end of file diff --git a/system/admin/editor.md/examples/emoji.html b/system/admin/editor.md/examples/emoji.html new file mode 100644 index 00000000..a5a6ea64 --- /dev/null +++ b/system/admin/editor.md/examples/emoji.html @@ -0,0 +1,191 @@ + + + + + Emoji - Editor.md examples + + + + + + +
                    +
                    +

                    Emoji 表情

                    +

                    Supports:

                    + +
                    +
                    + +
                    +
                    + + + + + \ No newline at end of file diff --git a/system/admin/editor.md/examples/extends.html b/system/admin/editor.md/examples/extends.html new file mode 100644 index 00000000..96018603 --- /dev/null +++ b/system/admin/editor.md/examples/extends.html @@ -0,0 +1,153 @@ + + + + + Expanded Editor.md - Editor.md examples + + + + + +
                    +
                    +

                    Expanded Editor.md

                    +

                    Expanded of member methods and properties

                    +
                    +
                    + +
                    +
                    + + + + + \ No newline at end of file diff --git a/system/admin/editor.md/examples/external-use.html b/system/admin/editor.md/examples/external-use.html new file mode 100644 index 00000000..32e02e26 --- /dev/null +++ b/system/admin/editor.md/examples/external-use.html @@ -0,0 +1,119 @@ + + + + + External use - Editor.md examples + + + + + +
                    +
                    +

                    External use

                    +

                    External use of toolbar handlers / modal dialog

                    +
                    +
                    + + + + + + + + +
                    +
                    + +
                    +
                    + + + + + + + + \ No newline at end of file diff --git a/system/admin/editor.md/examples/flowchart.html b/system/admin/editor.md/examples/flowchart.html new file mode 100644 index 00000000..5149cb7f --- /dev/null +++ b/system/admin/editor.md/examples/flowchart.html @@ -0,0 +1,53 @@ + + + + + FlowChart - Editor.md examples + + + + + +
                    +
                    +

                    FlowChart 流程图

                    +

                    Based on flowchart.js:http://adrai.github.io/flowchart.js/

                    +
                    +
                    + +
                    +
                    + + + + + + \ No newline at end of file diff --git a/system/admin/editor.md/examples/form-get-value.html b/system/admin/editor.md/examples/form-get-value.html new file mode 100644 index 00000000..5433d453 --- /dev/null +++ b/system/admin/editor.md/examples/form-get-value.html @@ -0,0 +1,92 @@ + + + + + Form get textarea value - Editor.md examples + + + + + +
                    +
                    +

                    表单取值

                    +

                    Form get textarea value.

                    +
                    +
                    +
                    + + +
                    +
                    + +
                    +
                    +
                    + + + + + \ No newline at end of file diff --git a/system/admin/editor.md/examples/full.html b/system/admin/editor.md/examples/full.html new file mode 100644 index 00000000..6fe08183 --- /dev/null +++ b/system/admin/editor.md/examples/full.html @@ -0,0 +1,231 @@ + + + + + Full example - Editor.md examples + + + + + + +
                    +
                    +

                    完整示例

                    +

                    Full example

                    +
                      +
                    • Enable HTML tags decode
                    • +
                    • Enable TeX, Flowchart, Sequence Diagram, Emoji, FontAwesome, Task lists
                    • +
                    • Enable Image upload
                    • +
                    • Enable [TOCM], Search Replace, Code fold
                    • +
                    +
                    +
                    + + + + + + + + + + + + + +
                    +
                    +
                    + + + + + \ No newline at end of file diff --git a/system/admin/editor.md/examples/goto-line.html b/system/admin/editor.md/examples/goto-line.html new file mode 100644 index 00000000..7eba47b0 --- /dev/null +++ b/system/admin/editor.md/examples/goto-line.html @@ -0,0 +1,84 @@ + + + + + Goto line - Editor.md examples + + + + + +
                    +
                    +

                    Goto line

                    +
                    +
                    + + + + + + + +
                    +
                    +
                    + + + + + \ No newline at end of file diff --git a/system/admin/editor.md/examples/html-preview-markdown-to-html-custom-toc-container.html b/system/admin/editor.md/examples/html-preview-markdown-to-html-custom-toc-container.html new file mode 100644 index 00000000..bdf1bd6e --- /dev/null +++ b/system/admin/editor.md/examples/html-preview-markdown-to-html-custom-toc-container.html @@ -0,0 +1,180 @@ + + + + + HTML Preview (markdown to html) - Editor.md examples + + + + + + +
                    +
                    +

                    Markdown转HTML的显示处理之自定义 ToC 容器

                    +

                    即:非编辑情况下的HTML预览

                    +

                    HTML Preview (markdown to html and custom ToC container)

                    +
                    +
                    + +
                    + +
                    + +
                    +
                    + +
                    +
                    + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/system/admin/editor.md/examples/html-preview-markdown-to-html.html b/system/admin/editor.md/examples/html-preview-markdown-to-html.html new file mode 100644 index 00000000..ad1cf590 --- /dev/null +++ b/system/admin/editor.md/examples/html-preview-markdown-to-html.html @@ -0,0 +1,142 @@ + + + + + HTML Preview(markdown to html) - Editor.md examples + + + + + + +
                    +
                    +

                    Markdown转HTML的显示处理

                    +

                    即:非编辑情况下的HTML预览

                    +

                    HTML Preview(markdown to html)

                    +
                    +
                    + +
                    +
                    + +
                    +
                    + +
                    +
                    + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/system/admin/editor.md/examples/html-tags-decode.html b/system/admin/editor.md/examples/html-tags-decode.html new file mode 100644 index 00000000..34de0d32 --- /dev/null +++ b/system/admin/editor.md/examples/html-tags-decode.html @@ -0,0 +1,119 @@ + + + + + 识别和解析 HTML 标签 - Editor.md examples + + + + + +
                    +
                    +

                    识别和解析HTML标签

                    +

                    HTML tags (filter) decode, You can increase safety by filtering the danger label.

                    +

                    注:虽然此功能能极大地扩展 Markdown 语法,但也面临着安全上的风险,所以默认是不开启的。

                    +

                    Update: 可以通过设置 `settings.htmlDecode = "style,script,iframe|on*"`来实现过滤指定标签及属性的解析,提高安全性;

                    +
                    +
                    + + + + +
                    +
                    + +
                    +
                    + + + + + + \ No newline at end of file diff --git a/system/admin/editor.md/examples/image-cross-domain-upload.html b/system/admin/editor.md/examples/image-cross-domain-upload.html new file mode 100644 index 00000000..5a635454 --- /dev/null +++ b/system/admin/editor.md/examples/image-cross-domain-upload.html @@ -0,0 +1,109 @@ + + + + + 图片跨域上传示例 - Editor.md examples + + + + + +
                    +
                    +

                    图片跨域上传示例

                    +

                    Image cross-domain upload example.

                    +
                    +
                    + +
                    +
                    + + + + + \ No newline at end of file diff --git a/system/admin/editor.md/examples/image-upload.html b/system/admin/editor.md/examples/image-upload.html new file mode 100644 index 00000000..e6fa69bc --- /dev/null +++ b/system/admin/editor.md/examples/image-upload.html @@ -0,0 +1,68 @@ + + + + + 图片上传示例 - Editor.md examples + + + + + +
                    +
                    +

                    图片上传示例

                    +

                    Image upload example

                    +
                    +
                    + +
                    +
                    + + + + + \ No newline at end of file diff --git a/system/admin/editor.md/examples/images/4.jpg b/system/admin/editor.md/examples/images/4.jpg new file mode 100644 index 00000000..948f88c0 Binary files /dev/null and b/system/admin/editor.md/examples/images/4.jpg differ diff --git a/system/admin/editor.md/examples/images/7.jpg b/system/admin/editor.md/examples/images/7.jpg new file mode 100644 index 00000000..c1806731 Binary files /dev/null and b/system/admin/editor.md/examples/images/7.jpg differ diff --git a/system/admin/editor.md/examples/images/8.jpg b/system/admin/editor.md/examples/images/8.jpg new file mode 100644 index 00000000..f56e66eb Binary files /dev/null and b/system/admin/editor.md/examples/images/8.jpg differ diff --git a/system/admin/editor.md/examples/images/editormd-screenshot.png b/system/admin/editor.md/examples/images/editormd-screenshot.png new file mode 100644 index 00000000..f63f633b Binary files /dev/null and b/system/admin/editor.md/examples/images/editormd-screenshot.png differ diff --git a/system/admin/editor.md/examples/index.html b/system/admin/editor.md/examples/index.html new file mode 100644 index 00000000..1d717e9f --- /dev/null +++ b/system/admin/editor.md/examples/index.html @@ -0,0 +1,356 @@ + + + + + Editor.md examples + + + + + + + +
                    + +

                    Basic

                    + +

                    + TOP + 自定义 Customs +

                    + +

                    + TOP + Markdown Extras +

                    + +

                    + TOP + Image Upload +

                    + +

                    + TOP + 事件处理 Events handle +

                    + +
                    + +
                    + + + + \ No newline at end of file diff --git a/system/admin/editor.md/examples/js/jquery.min.js b/system/admin/editor.md/examples/js/jquery.min.js new file mode 100644 index 00000000..b36821be --- /dev/null +++ b/system/admin/editor.md/examples/js/jquery.min.js @@ -0,0 +1,4 @@ +/*! jQuery v1.11.1 | (c) 2005, 2014 jQuery Foundation, Inc. | jquery.org/license */ +!function(a,b){"object"==typeof module&&"object"==typeof module.exports?module.exports=a.document?b(a,!0):function(a){if(!a.document)throw new Error("jQuery requires a window with a document");return b(a)}:b(a)}("undefined"!=typeof window?window:this,function(a,b){var c=[],d=c.slice,e=c.concat,f=c.push,g=c.indexOf,h={},i=h.toString,j=h.hasOwnProperty,k={},l="1.11.1",m=function(a,b){return new m.fn.init(a,b)},n=/^[\s\uFEFF\xA0]+|[\s\uFEFF\xA0]+$/g,o=/^-ms-/,p=/-([\da-z])/gi,q=function(a,b){return b.toUpperCase()};m.fn=m.prototype={jquery:l,constructor:m,selector:"",length:0,toArray:function(){return d.call(this)},get:function(a){return null!=a?0>a?this[a+this.length]:this[a]:d.call(this)},pushStack:function(a){var b=m.merge(this.constructor(),a);return b.prevObject=this,b.context=this.context,b},each:function(a,b){return m.each(this,a,b)},map:function(a){return this.pushStack(m.map(this,function(b,c){return a.call(b,c,b)}))},slice:function(){return this.pushStack(d.apply(this,arguments))},first:function(){return this.eq(0)},last:function(){return this.eq(-1)},eq:function(a){var b=this.length,c=+a+(0>a?b:0);return this.pushStack(c>=0&&b>c?[this[c]]:[])},end:function(){return this.prevObject||this.constructor(null)},push:f,sort:c.sort,splice:c.splice},m.extend=m.fn.extend=function(){var a,b,c,d,e,f,g=arguments[0]||{},h=1,i=arguments.length,j=!1;for("boolean"==typeof g&&(j=g,g=arguments[h]||{},h++),"object"==typeof g||m.isFunction(g)||(g={}),h===i&&(g=this,h--);i>h;h++)if(null!=(e=arguments[h]))for(d in e)a=g[d],c=e[d],g!==c&&(j&&c&&(m.isPlainObject(c)||(b=m.isArray(c)))?(b?(b=!1,f=a&&m.isArray(a)?a:[]):f=a&&m.isPlainObject(a)?a:{},g[d]=m.extend(j,f,c)):void 0!==c&&(g[d]=c));return g},m.extend({expando:"jQuery"+(l+Math.random()).replace(/\D/g,""),isReady:!0,error:function(a){throw new Error(a)},noop:function(){},isFunction:function(a){return"function"===m.type(a)},isArray:Array.isArray||function(a){return"array"===m.type(a)},isWindow:function(a){return null!=a&&a==a.window},isNumeric:function(a){return!m.isArray(a)&&a-parseFloat(a)>=0},isEmptyObject:function(a){var b;for(b in a)return!1;return!0},isPlainObject:function(a){var b;if(!a||"object"!==m.type(a)||a.nodeType||m.isWindow(a))return!1;try{if(a.constructor&&!j.call(a,"constructor")&&!j.call(a.constructor.prototype,"isPrototypeOf"))return!1}catch(c){return!1}if(k.ownLast)for(b in a)return j.call(a,b);for(b in a);return void 0===b||j.call(a,b)},type:function(a){return null==a?a+"":"object"==typeof a||"function"==typeof a?h[i.call(a)]||"object":typeof a},globalEval:function(b){b&&m.trim(b)&&(a.execScript||function(b){a.eval.call(a,b)})(b)},camelCase:function(a){return a.replace(o,"ms-").replace(p,q)},nodeName:function(a,b){return a.nodeName&&a.nodeName.toLowerCase()===b.toLowerCase()},each:function(a,b,c){var d,e=0,f=a.length,g=r(a);if(c){if(g){for(;f>e;e++)if(d=b.apply(a[e],c),d===!1)break}else for(e in a)if(d=b.apply(a[e],c),d===!1)break}else if(g){for(;f>e;e++)if(d=b.call(a[e],e,a[e]),d===!1)break}else for(e in a)if(d=b.call(a[e],e,a[e]),d===!1)break;return a},trim:function(a){return null==a?"":(a+"").replace(n,"")},makeArray:function(a,b){var c=b||[];return null!=a&&(r(Object(a))?m.merge(c,"string"==typeof a?[a]:a):f.call(c,a)),c},inArray:function(a,b,c){var d;if(b){if(g)return g.call(b,a,c);for(d=b.length,c=c?0>c?Math.max(0,d+c):c:0;d>c;c++)if(c in b&&b[c]===a)return c}return-1},merge:function(a,b){var c=+b.length,d=0,e=a.length;while(c>d)a[e++]=b[d++];if(c!==c)while(void 0!==b[d])a[e++]=b[d++];return a.length=e,a},grep:function(a,b,c){for(var d,e=[],f=0,g=a.length,h=!c;g>f;f++)d=!b(a[f],f),d!==h&&e.push(a[f]);return e},map:function(a,b,c){var d,f=0,g=a.length,h=r(a),i=[];if(h)for(;g>f;f++)d=b(a[f],f,c),null!=d&&i.push(d);else for(f in a)d=b(a[f],f,c),null!=d&&i.push(d);return e.apply([],i)},guid:1,proxy:function(a,b){var c,e,f;return"string"==typeof b&&(f=a[b],b=a,a=f),m.isFunction(a)?(c=d.call(arguments,2),e=function(){return a.apply(b||this,c.concat(d.call(arguments)))},e.guid=a.guid=a.guid||m.guid++,e):void 0},now:function(){return+new Date},support:k}),m.each("Boolean Number String Function Array Date RegExp Object Error".split(" "),function(a,b){h["[object "+b+"]"]=b.toLowerCase()});function r(a){var b=a.length,c=m.type(a);return"function"===c||m.isWindow(a)?!1:1===a.nodeType&&b?!0:"array"===c||0===b||"number"==typeof b&&b>0&&b-1 in a}var s=function(a){var b,c,d,e,f,g,h,i,j,k,l,m,n,o,p,q,r,s,t,u="sizzle"+-new Date,v=a.document,w=0,x=0,y=gb(),z=gb(),A=gb(),B=function(a,b){return a===b&&(l=!0),0},C="undefined",D=1<<31,E={}.hasOwnProperty,F=[],G=F.pop,H=F.push,I=F.push,J=F.slice,K=F.indexOf||function(a){for(var b=0,c=this.length;c>b;b++)if(this[b]===a)return b;return-1},L="checked|selected|async|autofocus|autoplay|controls|defer|disabled|hidden|ismap|loop|multiple|open|readonly|required|scoped",M="[\\x20\\t\\r\\n\\f]",N="(?:\\\\.|[\\w-]|[^\\x00-\\xa0])+",O=N.replace("w","w#"),P="\\["+M+"*("+N+")(?:"+M+"*([*^$|!~]?=)"+M+"*(?:'((?:\\\\.|[^\\\\'])*)'|\"((?:\\\\.|[^\\\\\"])*)\"|("+O+"))|)"+M+"*\\]",Q=":("+N+")(?:\\((('((?:\\\\.|[^\\\\'])*)'|\"((?:\\\\.|[^\\\\\"])*)\")|((?:\\\\.|[^\\\\()[\\]]|"+P+")*)|.*)\\)|)",R=new RegExp("^"+M+"+|((?:^|[^\\\\])(?:\\\\.)*)"+M+"+$","g"),S=new RegExp("^"+M+"*,"+M+"*"),T=new RegExp("^"+M+"*([>+~]|"+M+")"+M+"*"),U=new RegExp("="+M+"*([^\\]'\"]*?)"+M+"*\\]","g"),V=new RegExp(Q),W=new RegExp("^"+O+"$"),X={ID:new RegExp("^#("+N+")"),CLASS:new RegExp("^\\.("+N+")"),TAG:new RegExp("^("+N.replace("w","w*")+")"),ATTR:new RegExp("^"+P),PSEUDO:new RegExp("^"+Q),CHILD:new RegExp("^:(only|first|last|nth|nth-last)-(child|of-type)(?:\\("+M+"*(even|odd|(([+-]|)(\\d*)n|)"+M+"*(?:([+-]|)"+M+"*(\\d+)|))"+M+"*\\)|)","i"),bool:new RegExp("^(?:"+L+")$","i"),needsContext:new RegExp("^"+M+"*[>+~]|:(even|odd|eq|gt|lt|nth|first|last)(?:\\("+M+"*((?:-\\d)?\\d*)"+M+"*\\)|)(?=[^-]|$)","i")},Y=/^(?:input|select|textarea|button)$/i,Z=/^h\d$/i,$=/^[^{]+\{\s*\[native \w/,_=/^(?:#([\w-]+)|(\w+)|\.([\w-]+))$/,ab=/[+~]/,bb=/'|\\/g,cb=new RegExp("\\\\([\\da-f]{1,6}"+M+"?|("+M+")|.)","ig"),db=function(a,b,c){var d="0x"+b-65536;return d!==d||c?b:0>d?String.fromCharCode(d+65536):String.fromCharCode(d>>10|55296,1023&d|56320)};try{I.apply(F=J.call(v.childNodes),v.childNodes),F[v.childNodes.length].nodeType}catch(eb){I={apply:F.length?function(a,b){H.apply(a,J.call(b))}:function(a,b){var c=a.length,d=0;while(a[c++]=b[d++]);a.length=c-1}}}function fb(a,b,d,e){var f,h,j,k,l,o,r,s,w,x;if((b?b.ownerDocument||b:v)!==n&&m(b),b=b||n,d=d||[],!a||"string"!=typeof a)return d;if(1!==(k=b.nodeType)&&9!==k)return[];if(p&&!e){if(f=_.exec(a))if(j=f[1]){if(9===k){if(h=b.getElementById(j),!h||!h.parentNode)return d;if(h.id===j)return d.push(h),d}else if(b.ownerDocument&&(h=b.ownerDocument.getElementById(j))&&t(b,h)&&h.id===j)return d.push(h),d}else{if(f[2])return I.apply(d,b.getElementsByTagName(a)),d;if((j=f[3])&&c.getElementsByClassName&&b.getElementsByClassName)return I.apply(d,b.getElementsByClassName(j)),d}if(c.qsa&&(!q||!q.test(a))){if(s=r=u,w=b,x=9===k&&a,1===k&&"object"!==b.nodeName.toLowerCase()){o=g(a),(r=b.getAttribute("id"))?s=r.replace(bb,"\\$&"):b.setAttribute("id",s),s="[id='"+s+"'] ",l=o.length;while(l--)o[l]=s+qb(o[l]);w=ab.test(a)&&ob(b.parentNode)||b,x=o.join(",")}if(x)try{return I.apply(d,w.querySelectorAll(x)),d}catch(y){}finally{r||b.removeAttribute("id")}}}return i(a.replace(R,"$1"),b,d,e)}function gb(){var a=[];function b(c,e){return a.push(c+" ")>d.cacheLength&&delete b[a.shift()],b[c+" "]=e}return b}function hb(a){return a[u]=!0,a}function ib(a){var b=n.createElement("div");try{return!!a(b)}catch(c){return!1}finally{b.parentNode&&b.parentNode.removeChild(b),b=null}}function jb(a,b){var c=a.split("|"),e=a.length;while(e--)d.attrHandle[c[e]]=b}function kb(a,b){var c=b&&a,d=c&&1===a.nodeType&&1===b.nodeType&&(~b.sourceIndex||D)-(~a.sourceIndex||D);if(d)return d;if(c)while(c=c.nextSibling)if(c===b)return-1;return a?1:-1}function lb(a){return function(b){var c=b.nodeName.toLowerCase();return"input"===c&&b.type===a}}function mb(a){return function(b){var c=b.nodeName.toLowerCase();return("input"===c||"button"===c)&&b.type===a}}function nb(a){return hb(function(b){return b=+b,hb(function(c,d){var e,f=a([],c.length,b),g=f.length;while(g--)c[e=f[g]]&&(c[e]=!(d[e]=c[e]))})})}function ob(a){return a&&typeof a.getElementsByTagName!==C&&a}c=fb.support={},f=fb.isXML=function(a){var b=a&&(a.ownerDocument||a).documentElement;return b?"HTML"!==b.nodeName:!1},m=fb.setDocument=function(a){var b,e=a?a.ownerDocument||a:v,g=e.defaultView;return e!==n&&9===e.nodeType&&e.documentElement?(n=e,o=e.documentElement,p=!f(e),g&&g!==g.top&&(g.addEventListener?g.addEventListener("unload",function(){m()},!1):g.attachEvent&&g.attachEvent("onunload",function(){m()})),c.attributes=ib(function(a){return a.className="i",!a.getAttribute("className")}),c.getElementsByTagName=ib(function(a){return a.appendChild(e.createComment("")),!a.getElementsByTagName("*").length}),c.getElementsByClassName=$.test(e.getElementsByClassName)&&ib(function(a){return a.innerHTML="
                    ",a.firstChild.className="i",2===a.getElementsByClassName("i").length}),c.getById=ib(function(a){return o.appendChild(a).id=u,!e.getElementsByName||!e.getElementsByName(u).length}),c.getById?(d.find.ID=function(a,b){if(typeof b.getElementById!==C&&p){var c=b.getElementById(a);return c&&c.parentNode?[c]:[]}},d.filter.ID=function(a){var b=a.replace(cb,db);return function(a){return a.getAttribute("id")===b}}):(delete d.find.ID,d.filter.ID=function(a){var b=a.replace(cb,db);return function(a){var c=typeof a.getAttributeNode!==C&&a.getAttributeNode("id");return c&&c.value===b}}),d.find.TAG=c.getElementsByTagName?function(a,b){return typeof b.getElementsByTagName!==C?b.getElementsByTagName(a):void 0}:function(a,b){var c,d=[],e=0,f=b.getElementsByTagName(a);if("*"===a){while(c=f[e++])1===c.nodeType&&d.push(c);return d}return f},d.find.CLASS=c.getElementsByClassName&&function(a,b){return typeof b.getElementsByClassName!==C&&p?b.getElementsByClassName(a):void 0},r=[],q=[],(c.qsa=$.test(e.querySelectorAll))&&(ib(function(a){a.innerHTML="",a.querySelectorAll("[msallowclip^='']").length&&q.push("[*^$]="+M+"*(?:''|\"\")"),a.querySelectorAll("[selected]").length||q.push("\\["+M+"*(?:value|"+L+")"),a.querySelectorAll(":checked").length||q.push(":checked")}),ib(function(a){var b=e.createElement("input");b.setAttribute("type","hidden"),a.appendChild(b).setAttribute("name","D"),a.querySelectorAll("[name=d]").length&&q.push("name"+M+"*[*^$|!~]?="),a.querySelectorAll(":enabled").length||q.push(":enabled",":disabled"),a.querySelectorAll("*,:x"),q.push(",.*:")})),(c.matchesSelector=$.test(s=o.matches||o.webkitMatchesSelector||o.mozMatchesSelector||o.oMatchesSelector||o.msMatchesSelector))&&ib(function(a){c.disconnectedMatch=s.call(a,"div"),s.call(a,"[s!='']:x"),r.push("!=",Q)}),q=q.length&&new RegExp(q.join("|")),r=r.length&&new RegExp(r.join("|")),b=$.test(o.compareDocumentPosition),t=b||$.test(o.contains)?function(a,b){var c=9===a.nodeType?a.documentElement:a,d=b&&b.parentNode;return a===d||!(!d||1!==d.nodeType||!(c.contains?c.contains(d):a.compareDocumentPosition&&16&a.compareDocumentPosition(d)))}:function(a,b){if(b)while(b=b.parentNode)if(b===a)return!0;return!1},B=b?function(a,b){if(a===b)return l=!0,0;var d=!a.compareDocumentPosition-!b.compareDocumentPosition;return d?d:(d=(a.ownerDocument||a)===(b.ownerDocument||b)?a.compareDocumentPosition(b):1,1&d||!c.sortDetached&&b.compareDocumentPosition(a)===d?a===e||a.ownerDocument===v&&t(v,a)?-1:b===e||b.ownerDocument===v&&t(v,b)?1:k?K.call(k,a)-K.call(k,b):0:4&d?-1:1)}:function(a,b){if(a===b)return l=!0,0;var c,d=0,f=a.parentNode,g=b.parentNode,h=[a],i=[b];if(!f||!g)return a===e?-1:b===e?1:f?-1:g?1:k?K.call(k,a)-K.call(k,b):0;if(f===g)return kb(a,b);c=a;while(c=c.parentNode)h.unshift(c);c=b;while(c=c.parentNode)i.unshift(c);while(h[d]===i[d])d++;return d?kb(h[d],i[d]):h[d]===v?-1:i[d]===v?1:0},e):n},fb.matches=function(a,b){return fb(a,null,null,b)},fb.matchesSelector=function(a,b){if((a.ownerDocument||a)!==n&&m(a),b=b.replace(U,"='$1']"),!(!c.matchesSelector||!p||r&&r.test(b)||q&&q.test(b)))try{var d=s.call(a,b);if(d||c.disconnectedMatch||a.document&&11!==a.document.nodeType)return d}catch(e){}return fb(b,n,null,[a]).length>0},fb.contains=function(a,b){return(a.ownerDocument||a)!==n&&m(a),t(a,b)},fb.attr=function(a,b){(a.ownerDocument||a)!==n&&m(a);var e=d.attrHandle[b.toLowerCase()],f=e&&E.call(d.attrHandle,b.toLowerCase())?e(a,b,!p):void 0;return void 0!==f?f:c.attributes||!p?a.getAttribute(b):(f=a.getAttributeNode(b))&&f.specified?f.value:null},fb.error=function(a){throw new Error("Syntax error, unrecognized expression: "+a)},fb.uniqueSort=function(a){var b,d=[],e=0,f=0;if(l=!c.detectDuplicates,k=!c.sortStable&&a.slice(0),a.sort(B),l){while(b=a[f++])b===a[f]&&(e=d.push(f));while(e--)a.splice(d[e],1)}return k=null,a},e=fb.getText=function(a){var b,c="",d=0,f=a.nodeType;if(f){if(1===f||9===f||11===f){if("string"==typeof a.textContent)return a.textContent;for(a=a.firstChild;a;a=a.nextSibling)c+=e(a)}else if(3===f||4===f)return a.nodeValue}else while(b=a[d++])c+=e(b);return c},d=fb.selectors={cacheLength:50,createPseudo:hb,match:X,attrHandle:{},find:{},relative:{">":{dir:"parentNode",first:!0}," ":{dir:"parentNode"},"+":{dir:"previousSibling",first:!0},"~":{dir:"previousSibling"}},preFilter:{ATTR:function(a){return a[1]=a[1].replace(cb,db),a[3]=(a[3]||a[4]||a[5]||"").replace(cb,db),"~="===a[2]&&(a[3]=" "+a[3]+" "),a.slice(0,4)},CHILD:function(a){return a[1]=a[1].toLowerCase(),"nth"===a[1].slice(0,3)?(a[3]||fb.error(a[0]),a[4]=+(a[4]?a[5]+(a[6]||1):2*("even"===a[3]||"odd"===a[3])),a[5]=+(a[7]+a[8]||"odd"===a[3])):a[3]&&fb.error(a[0]),a},PSEUDO:function(a){var b,c=!a[6]&&a[2];return X.CHILD.test(a[0])?null:(a[3]?a[2]=a[4]||a[5]||"":c&&V.test(c)&&(b=g(c,!0))&&(b=c.indexOf(")",c.length-b)-c.length)&&(a[0]=a[0].slice(0,b),a[2]=c.slice(0,b)),a.slice(0,3))}},filter:{TAG:function(a){var b=a.replace(cb,db).toLowerCase();return"*"===a?function(){return!0}:function(a){return a.nodeName&&a.nodeName.toLowerCase()===b}},CLASS:function(a){var b=y[a+" "];return b||(b=new RegExp("(^|"+M+")"+a+"("+M+"|$)"))&&y(a,function(a){return b.test("string"==typeof a.className&&a.className||typeof a.getAttribute!==C&&a.getAttribute("class")||"")})},ATTR:function(a,b,c){return function(d){var e=fb.attr(d,a);return null==e?"!="===b:b?(e+="","="===b?e===c:"!="===b?e!==c:"^="===b?c&&0===e.indexOf(c):"*="===b?c&&e.indexOf(c)>-1:"$="===b?c&&e.slice(-c.length)===c:"~="===b?(" "+e+" ").indexOf(c)>-1:"|="===b?e===c||e.slice(0,c.length+1)===c+"-":!1):!0}},CHILD:function(a,b,c,d,e){var f="nth"!==a.slice(0,3),g="last"!==a.slice(-4),h="of-type"===b;return 1===d&&0===e?function(a){return!!a.parentNode}:function(b,c,i){var j,k,l,m,n,o,p=f!==g?"nextSibling":"previousSibling",q=b.parentNode,r=h&&b.nodeName.toLowerCase(),s=!i&&!h;if(q){if(f){while(p){l=b;while(l=l[p])if(h?l.nodeName.toLowerCase()===r:1===l.nodeType)return!1;o=p="only"===a&&!o&&"nextSibling"}return!0}if(o=[g?q.firstChild:q.lastChild],g&&s){k=q[u]||(q[u]={}),j=k[a]||[],n=j[0]===w&&j[1],m=j[0]===w&&j[2],l=n&&q.childNodes[n];while(l=++n&&l&&l[p]||(m=n=0)||o.pop())if(1===l.nodeType&&++m&&l===b){k[a]=[w,n,m];break}}else if(s&&(j=(b[u]||(b[u]={}))[a])&&j[0]===w)m=j[1];else while(l=++n&&l&&l[p]||(m=n=0)||o.pop())if((h?l.nodeName.toLowerCase()===r:1===l.nodeType)&&++m&&(s&&((l[u]||(l[u]={}))[a]=[w,m]),l===b))break;return m-=e,m===d||m%d===0&&m/d>=0}}},PSEUDO:function(a,b){var c,e=d.pseudos[a]||d.setFilters[a.toLowerCase()]||fb.error("unsupported pseudo: "+a);return e[u]?e(b):e.length>1?(c=[a,a,"",b],d.setFilters.hasOwnProperty(a.toLowerCase())?hb(function(a,c){var d,f=e(a,b),g=f.length;while(g--)d=K.call(a,f[g]),a[d]=!(c[d]=f[g])}):function(a){return e(a,0,c)}):e}},pseudos:{not:hb(function(a){var b=[],c=[],d=h(a.replace(R,"$1"));return d[u]?hb(function(a,b,c,e){var f,g=d(a,null,e,[]),h=a.length;while(h--)(f=g[h])&&(a[h]=!(b[h]=f))}):function(a,e,f){return b[0]=a,d(b,null,f,c),!c.pop()}}),has:hb(function(a){return function(b){return fb(a,b).length>0}}),contains:hb(function(a){return function(b){return(b.textContent||b.innerText||e(b)).indexOf(a)>-1}}),lang:hb(function(a){return W.test(a||"")||fb.error("unsupported lang: "+a),a=a.replace(cb,db).toLowerCase(),function(b){var c;do if(c=p?b.lang:b.getAttribute("xml:lang")||b.getAttribute("lang"))return c=c.toLowerCase(),c===a||0===c.indexOf(a+"-");while((b=b.parentNode)&&1===b.nodeType);return!1}}),target:function(b){var c=a.location&&a.location.hash;return c&&c.slice(1)===b.id},root:function(a){return a===o},focus:function(a){return a===n.activeElement&&(!n.hasFocus||n.hasFocus())&&!!(a.type||a.href||~a.tabIndex)},enabled:function(a){return a.disabled===!1},disabled:function(a){return a.disabled===!0},checked:function(a){var b=a.nodeName.toLowerCase();return"input"===b&&!!a.checked||"option"===b&&!!a.selected},selected:function(a){return a.parentNode&&a.parentNode.selectedIndex,a.selected===!0},empty:function(a){for(a=a.firstChild;a;a=a.nextSibling)if(a.nodeType<6)return!1;return!0},parent:function(a){return!d.pseudos.empty(a)},header:function(a){return Z.test(a.nodeName)},input:function(a){return Y.test(a.nodeName)},button:function(a){var b=a.nodeName.toLowerCase();return"input"===b&&"button"===a.type||"button"===b},text:function(a){var b;return"input"===a.nodeName.toLowerCase()&&"text"===a.type&&(null==(b=a.getAttribute("type"))||"text"===b.toLowerCase())},first:nb(function(){return[0]}),last:nb(function(a,b){return[b-1]}),eq:nb(function(a,b,c){return[0>c?c+b:c]}),even:nb(function(a,b){for(var c=0;b>c;c+=2)a.push(c);return a}),odd:nb(function(a,b){for(var c=1;b>c;c+=2)a.push(c);return a}),lt:nb(function(a,b,c){for(var d=0>c?c+b:c;--d>=0;)a.push(d);return a}),gt:nb(function(a,b,c){for(var d=0>c?c+b:c;++db;b++)d+=a[b].value;return d}function rb(a,b,c){var d=b.dir,e=c&&"parentNode"===d,f=x++;return b.first?function(b,c,f){while(b=b[d])if(1===b.nodeType||e)return a(b,c,f)}:function(b,c,g){var h,i,j=[w,f];if(g){while(b=b[d])if((1===b.nodeType||e)&&a(b,c,g))return!0}else while(b=b[d])if(1===b.nodeType||e){if(i=b[u]||(b[u]={}),(h=i[d])&&h[0]===w&&h[1]===f)return j[2]=h[2];if(i[d]=j,j[2]=a(b,c,g))return!0}}}function sb(a){return a.length>1?function(b,c,d){var e=a.length;while(e--)if(!a[e](b,c,d))return!1;return!0}:a[0]}function tb(a,b,c){for(var d=0,e=b.length;e>d;d++)fb(a,b[d],c);return c}function ub(a,b,c,d,e){for(var f,g=[],h=0,i=a.length,j=null!=b;i>h;h++)(f=a[h])&&(!c||c(f,d,e))&&(g.push(f),j&&b.push(h));return g}function vb(a,b,c,d,e,f){return d&&!d[u]&&(d=vb(d)),e&&!e[u]&&(e=vb(e,f)),hb(function(f,g,h,i){var j,k,l,m=[],n=[],o=g.length,p=f||tb(b||"*",h.nodeType?[h]:h,[]),q=!a||!f&&b?p:ub(p,m,a,h,i),r=c?e||(f?a:o||d)?[]:g:q;if(c&&c(q,r,h,i),d){j=ub(r,n),d(j,[],h,i),k=j.length;while(k--)(l=j[k])&&(r[n[k]]=!(q[n[k]]=l))}if(f){if(e||a){if(e){j=[],k=r.length;while(k--)(l=r[k])&&j.push(q[k]=l);e(null,r=[],j,i)}k=r.length;while(k--)(l=r[k])&&(j=e?K.call(f,l):m[k])>-1&&(f[j]=!(g[j]=l))}}else r=ub(r===g?r.splice(o,r.length):r),e?e(null,g,r,i):I.apply(g,r)})}function wb(a){for(var b,c,e,f=a.length,g=d.relative[a[0].type],h=g||d.relative[" "],i=g?1:0,k=rb(function(a){return a===b},h,!0),l=rb(function(a){return K.call(b,a)>-1},h,!0),m=[function(a,c,d){return!g&&(d||c!==j)||((b=c).nodeType?k(a,c,d):l(a,c,d))}];f>i;i++)if(c=d.relative[a[i].type])m=[rb(sb(m),c)];else{if(c=d.filter[a[i].type].apply(null,a[i].matches),c[u]){for(e=++i;f>e;e++)if(d.relative[a[e].type])break;return vb(i>1&&sb(m),i>1&&qb(a.slice(0,i-1).concat({value:" "===a[i-2].type?"*":""})).replace(R,"$1"),c,e>i&&wb(a.slice(i,e)),f>e&&wb(a=a.slice(e)),f>e&&qb(a))}m.push(c)}return sb(m)}function xb(a,b){var c=b.length>0,e=a.length>0,f=function(f,g,h,i,k){var l,m,o,p=0,q="0",r=f&&[],s=[],t=j,u=f||e&&d.find.TAG("*",k),v=w+=null==t?1:Math.random()||.1,x=u.length;for(k&&(j=g!==n&&g);q!==x&&null!=(l=u[q]);q++){if(e&&l){m=0;while(o=a[m++])if(o(l,g,h)){i.push(l);break}k&&(w=v)}c&&((l=!o&&l)&&p--,f&&r.push(l))}if(p+=q,c&&q!==p){m=0;while(o=b[m++])o(r,s,g,h);if(f){if(p>0)while(q--)r[q]||s[q]||(s[q]=G.call(i));s=ub(s)}I.apply(i,s),k&&!f&&s.length>0&&p+b.length>1&&fb.uniqueSort(i)}return k&&(w=v,j=t),r};return c?hb(f):f}return h=fb.compile=function(a,b){var c,d=[],e=[],f=A[a+" "];if(!f){b||(b=g(a)),c=b.length;while(c--)f=wb(b[c]),f[u]?d.push(f):e.push(f);f=A(a,xb(e,d)),f.selector=a}return f},i=fb.select=function(a,b,e,f){var i,j,k,l,m,n="function"==typeof a&&a,o=!f&&g(a=n.selector||a);if(e=e||[],1===o.length){if(j=o[0]=o[0].slice(0),j.length>2&&"ID"===(k=j[0]).type&&c.getById&&9===b.nodeType&&p&&d.relative[j[1].type]){if(b=(d.find.ID(k.matches[0].replace(cb,db),b)||[])[0],!b)return e;n&&(b=b.parentNode),a=a.slice(j.shift().value.length)}i=X.needsContext.test(a)?0:j.length;while(i--){if(k=j[i],d.relative[l=k.type])break;if((m=d.find[l])&&(f=m(k.matches[0].replace(cb,db),ab.test(j[0].type)&&ob(b.parentNode)||b))){if(j.splice(i,1),a=f.length&&qb(j),!a)return I.apply(e,f),e;break}}}return(n||h(a,o))(f,b,!p,e,ab.test(a)&&ob(b.parentNode)||b),e},c.sortStable=u.split("").sort(B).join("")===u,c.detectDuplicates=!!l,m(),c.sortDetached=ib(function(a){return 1&a.compareDocumentPosition(n.createElement("div"))}),ib(function(a){return a.innerHTML="","#"===a.firstChild.getAttribute("href")})||jb("type|href|height|width",function(a,b,c){return c?void 0:a.getAttribute(b,"type"===b.toLowerCase()?1:2)}),c.attributes&&ib(function(a){return a.innerHTML="",a.firstChild.setAttribute("value",""),""===a.firstChild.getAttribute("value")})||jb("value",function(a,b,c){return c||"input"!==a.nodeName.toLowerCase()?void 0:a.defaultValue}),ib(function(a){return null==a.getAttribute("disabled")})||jb(L,function(a,b,c){var d;return c?void 0:a[b]===!0?b.toLowerCase():(d=a.getAttributeNode(b))&&d.specified?d.value:null}),fb}(a);m.find=s,m.expr=s.selectors,m.expr[":"]=m.expr.pseudos,m.unique=s.uniqueSort,m.text=s.getText,m.isXMLDoc=s.isXML,m.contains=s.contains;var t=m.expr.match.needsContext,u=/^<(\w+)\s*\/?>(?:<\/\1>|)$/,v=/^.[^:#\[\.,]*$/;function w(a,b,c){if(m.isFunction(b))return m.grep(a,function(a,d){return!!b.call(a,d,a)!==c});if(b.nodeType)return m.grep(a,function(a){return a===b!==c});if("string"==typeof b){if(v.test(b))return m.filter(b,a,c);b=m.filter(b,a)}return m.grep(a,function(a){return m.inArray(a,b)>=0!==c})}m.filter=function(a,b,c){var d=b[0];return c&&(a=":not("+a+")"),1===b.length&&1===d.nodeType?m.find.matchesSelector(d,a)?[d]:[]:m.find.matches(a,m.grep(b,function(a){return 1===a.nodeType}))},m.fn.extend({find:function(a){var b,c=[],d=this,e=d.length;if("string"!=typeof a)return this.pushStack(m(a).filter(function(){for(b=0;e>b;b++)if(m.contains(d[b],this))return!0}));for(b=0;e>b;b++)m.find(a,d[b],c);return c=this.pushStack(e>1?m.unique(c):c),c.selector=this.selector?this.selector+" "+a:a,c},filter:function(a){return this.pushStack(w(this,a||[],!1))},not:function(a){return this.pushStack(w(this,a||[],!0))},is:function(a){return!!w(this,"string"==typeof a&&t.test(a)?m(a):a||[],!1).length}});var x,y=a.document,z=/^(?:\s*(<[\w\W]+>)[^>]*|#([\w-]*))$/,A=m.fn.init=function(a,b){var c,d;if(!a)return this;if("string"==typeof a){if(c="<"===a.charAt(0)&&">"===a.charAt(a.length-1)&&a.length>=3?[null,a,null]:z.exec(a),!c||!c[1]&&b)return!b||b.jquery?(b||x).find(a):this.constructor(b).find(a);if(c[1]){if(b=b instanceof m?b[0]:b,m.merge(this,m.parseHTML(c[1],b&&b.nodeType?b.ownerDocument||b:y,!0)),u.test(c[1])&&m.isPlainObject(b))for(c in b)m.isFunction(this[c])?this[c](b[c]):this.attr(c,b[c]);return this}if(d=y.getElementById(c[2]),d&&d.parentNode){if(d.id!==c[2])return x.find(a);this.length=1,this[0]=d}return this.context=y,this.selector=a,this}return a.nodeType?(this.context=this[0]=a,this.length=1,this):m.isFunction(a)?"undefined"!=typeof x.ready?x.ready(a):a(m):(void 0!==a.selector&&(this.selector=a.selector,this.context=a.context),m.makeArray(a,this))};A.prototype=m.fn,x=m(y);var B=/^(?:parents|prev(?:Until|All))/,C={children:!0,contents:!0,next:!0,prev:!0};m.extend({dir:function(a,b,c){var d=[],e=a[b];while(e&&9!==e.nodeType&&(void 0===c||1!==e.nodeType||!m(e).is(c)))1===e.nodeType&&d.push(e),e=e[b];return d},sibling:function(a,b){for(var c=[];a;a=a.nextSibling)1===a.nodeType&&a!==b&&c.push(a);return c}}),m.fn.extend({has:function(a){var b,c=m(a,this),d=c.length;return this.filter(function(){for(b=0;d>b;b++)if(m.contains(this,c[b]))return!0})},closest:function(a,b){for(var c,d=0,e=this.length,f=[],g=t.test(a)||"string"!=typeof a?m(a,b||this.context):0;e>d;d++)for(c=this[d];c&&c!==b;c=c.parentNode)if(c.nodeType<11&&(g?g.index(c)>-1:1===c.nodeType&&m.find.matchesSelector(c,a))){f.push(c);break}return this.pushStack(f.length>1?m.unique(f):f)},index:function(a){return a?"string"==typeof a?m.inArray(this[0],m(a)):m.inArray(a.jquery?a[0]:a,this):this[0]&&this[0].parentNode?this.first().prevAll().length:-1},add:function(a,b){return this.pushStack(m.unique(m.merge(this.get(),m(a,b))))},addBack:function(a){return this.add(null==a?this.prevObject:this.prevObject.filter(a))}});function D(a,b){do a=a[b];while(a&&1!==a.nodeType);return a}m.each({parent:function(a){var b=a.parentNode;return b&&11!==b.nodeType?b:null},parents:function(a){return m.dir(a,"parentNode")},parentsUntil:function(a,b,c){return m.dir(a,"parentNode",c)},next:function(a){return D(a,"nextSibling")},prev:function(a){return D(a,"previousSibling")},nextAll:function(a){return m.dir(a,"nextSibling")},prevAll:function(a){return m.dir(a,"previousSibling")},nextUntil:function(a,b,c){return m.dir(a,"nextSibling",c)},prevUntil:function(a,b,c){return m.dir(a,"previousSibling",c)},siblings:function(a){return m.sibling((a.parentNode||{}).firstChild,a)},children:function(a){return m.sibling(a.firstChild)},contents:function(a){return m.nodeName(a,"iframe")?a.contentDocument||a.contentWindow.document:m.merge([],a.childNodes)}},function(a,b){m.fn[a]=function(c,d){var e=m.map(this,b,c);return"Until"!==a.slice(-5)&&(d=c),d&&"string"==typeof d&&(e=m.filter(d,e)),this.length>1&&(C[a]||(e=m.unique(e)),B.test(a)&&(e=e.reverse())),this.pushStack(e)}});var E=/\S+/g,F={};function G(a){var b=F[a]={};return m.each(a.match(E)||[],function(a,c){b[c]=!0}),b}m.Callbacks=function(a){a="string"==typeof a?F[a]||G(a):m.extend({},a);var b,c,d,e,f,g,h=[],i=!a.once&&[],j=function(l){for(c=a.memory&&l,d=!0,f=g||0,g=0,e=h.length,b=!0;h&&e>f;f++)if(h[f].apply(l[0],l[1])===!1&&a.stopOnFalse){c=!1;break}b=!1,h&&(i?i.length&&j(i.shift()):c?h=[]:k.disable())},k={add:function(){if(h){var d=h.length;!function f(b){m.each(b,function(b,c){var d=m.type(c);"function"===d?a.unique&&k.has(c)||h.push(c):c&&c.length&&"string"!==d&&f(c)})}(arguments),b?e=h.length:c&&(g=d,j(c))}return this},remove:function(){return h&&m.each(arguments,function(a,c){var d;while((d=m.inArray(c,h,d))>-1)h.splice(d,1),b&&(e>=d&&e--,f>=d&&f--)}),this},has:function(a){return a?m.inArray(a,h)>-1:!(!h||!h.length)},empty:function(){return h=[],e=0,this},disable:function(){return h=i=c=void 0,this},disabled:function(){return!h},lock:function(){return i=void 0,c||k.disable(),this},locked:function(){return!i},fireWith:function(a,c){return!h||d&&!i||(c=c||[],c=[a,c.slice?c.slice():c],b?i.push(c):j(c)),this},fire:function(){return k.fireWith(this,arguments),this},fired:function(){return!!d}};return k},m.extend({Deferred:function(a){var b=[["resolve","done",m.Callbacks("once memory"),"resolved"],["reject","fail",m.Callbacks("once memory"),"rejected"],["notify","progress",m.Callbacks("memory")]],c="pending",d={state:function(){return c},always:function(){return e.done(arguments).fail(arguments),this},then:function(){var a=arguments;return m.Deferred(function(c){m.each(b,function(b,f){var g=m.isFunction(a[b])&&a[b];e[f[1]](function(){var a=g&&g.apply(this,arguments);a&&m.isFunction(a.promise)?a.promise().done(c.resolve).fail(c.reject).progress(c.notify):c[f[0]+"With"](this===d?c.promise():this,g?[a]:arguments)})}),a=null}).promise()},promise:function(a){return null!=a?m.extend(a,d):d}},e={};return d.pipe=d.then,m.each(b,function(a,f){var g=f[2],h=f[3];d[f[1]]=g.add,h&&g.add(function(){c=h},b[1^a][2].disable,b[2][2].lock),e[f[0]]=function(){return e[f[0]+"With"](this===e?d:this,arguments),this},e[f[0]+"With"]=g.fireWith}),d.promise(e),a&&a.call(e,e),e},when:function(a){var b=0,c=d.call(arguments),e=c.length,f=1!==e||a&&m.isFunction(a.promise)?e:0,g=1===f?a:m.Deferred(),h=function(a,b,c){return function(e){b[a]=this,c[a]=arguments.length>1?d.call(arguments):e,c===i?g.notifyWith(b,c):--f||g.resolveWith(b,c)}},i,j,k;if(e>1)for(i=new Array(e),j=new Array(e),k=new Array(e);e>b;b++)c[b]&&m.isFunction(c[b].promise)?c[b].promise().done(h(b,k,c)).fail(g.reject).progress(h(b,j,i)):--f;return f||g.resolveWith(k,c),g.promise()}});var H;m.fn.ready=function(a){return m.ready.promise().done(a),this},m.extend({isReady:!1,readyWait:1,holdReady:function(a){a?m.readyWait++:m.ready(!0)},ready:function(a){if(a===!0?!--m.readyWait:!m.isReady){if(!y.body)return setTimeout(m.ready);m.isReady=!0,a!==!0&&--m.readyWait>0||(H.resolveWith(y,[m]),m.fn.triggerHandler&&(m(y).triggerHandler("ready"),m(y).off("ready")))}}});function I(){y.addEventListener?(y.removeEventListener("DOMContentLoaded",J,!1),a.removeEventListener("load",J,!1)):(y.detachEvent("onreadystatechange",J),a.detachEvent("onload",J))}function J(){(y.addEventListener||"load"===event.type||"complete"===y.readyState)&&(I(),m.ready())}m.ready.promise=function(b){if(!H)if(H=m.Deferred(),"complete"===y.readyState)setTimeout(m.ready);else if(y.addEventListener)y.addEventListener("DOMContentLoaded",J,!1),a.addEventListener("load",J,!1);else{y.attachEvent("onreadystatechange",J),a.attachEvent("onload",J);var c=!1;try{c=null==a.frameElement&&y.documentElement}catch(d){}c&&c.doScroll&&!function e(){if(!m.isReady){try{c.doScroll("left")}catch(a){return setTimeout(e,50)}I(),m.ready()}}()}return H.promise(b)};var K="undefined",L;for(L in m(k))break;k.ownLast="0"!==L,k.inlineBlockNeedsLayout=!1,m(function(){var a,b,c,d;c=y.getElementsByTagName("body")[0],c&&c.style&&(b=y.createElement("div"),d=y.createElement("div"),d.style.cssText="position:absolute;border:0;width:0;height:0;top:0;left:-9999px",c.appendChild(d).appendChild(b),typeof b.style.zoom!==K&&(b.style.cssText="display:inline;margin:0;border:0;padding:1px;width:1px;zoom:1",k.inlineBlockNeedsLayout=a=3===b.offsetWidth,a&&(c.style.zoom=1)),c.removeChild(d))}),function(){var a=y.createElement("div");if(null==k.deleteExpando){k.deleteExpando=!0;try{delete a.test}catch(b){k.deleteExpando=!1}}a=null}(),m.acceptData=function(a){var b=m.noData[(a.nodeName+" ").toLowerCase()],c=+a.nodeType||1;return 1!==c&&9!==c?!1:!b||b!==!0&&a.getAttribute("classid")===b};var M=/^(?:\{[\w\W]*\}|\[[\w\W]*\])$/,N=/([A-Z])/g;function O(a,b,c){if(void 0===c&&1===a.nodeType){var d="data-"+b.replace(N,"-$1").toLowerCase();if(c=a.getAttribute(d),"string"==typeof c){try{c="true"===c?!0:"false"===c?!1:"null"===c?null:+c+""===c?+c:M.test(c)?m.parseJSON(c):c}catch(e){}m.data(a,b,c)}else c=void 0}return c}function P(a){var b;for(b in a)if(("data"!==b||!m.isEmptyObject(a[b]))&&"toJSON"!==b)return!1;return!0}function Q(a,b,d,e){if(m.acceptData(a)){var f,g,h=m.expando,i=a.nodeType,j=i?m.cache:a,k=i?a[h]:a[h]&&h; +if(k&&j[k]&&(e||j[k].data)||void 0!==d||"string"!=typeof b)return k||(k=i?a[h]=c.pop()||m.guid++:h),j[k]||(j[k]=i?{}:{toJSON:m.noop}),("object"==typeof b||"function"==typeof b)&&(e?j[k]=m.extend(j[k],b):j[k].data=m.extend(j[k].data,b)),g=j[k],e||(g.data||(g.data={}),g=g.data),void 0!==d&&(g[m.camelCase(b)]=d),"string"==typeof b?(f=g[b],null==f&&(f=g[m.camelCase(b)])):f=g,f}}function R(a,b,c){if(m.acceptData(a)){var d,e,f=a.nodeType,g=f?m.cache:a,h=f?a[m.expando]:m.expando;if(g[h]){if(b&&(d=c?g[h]:g[h].data)){m.isArray(b)?b=b.concat(m.map(b,m.camelCase)):b in d?b=[b]:(b=m.camelCase(b),b=b in d?[b]:b.split(" ")),e=b.length;while(e--)delete d[b[e]];if(c?!P(d):!m.isEmptyObject(d))return}(c||(delete g[h].data,P(g[h])))&&(f?m.cleanData([a],!0):k.deleteExpando||g!=g.window?delete g[h]:g[h]=null)}}}m.extend({cache:{},noData:{"applet ":!0,"embed ":!0,"object ":"clsid:D27CDB6E-AE6D-11cf-96B8-444553540000"},hasData:function(a){return a=a.nodeType?m.cache[a[m.expando]]:a[m.expando],!!a&&!P(a)},data:function(a,b,c){return Q(a,b,c)},removeData:function(a,b){return R(a,b)},_data:function(a,b,c){return Q(a,b,c,!0)},_removeData:function(a,b){return R(a,b,!0)}}),m.fn.extend({data:function(a,b){var c,d,e,f=this[0],g=f&&f.attributes;if(void 0===a){if(this.length&&(e=m.data(f),1===f.nodeType&&!m._data(f,"parsedAttrs"))){c=g.length;while(c--)g[c]&&(d=g[c].name,0===d.indexOf("data-")&&(d=m.camelCase(d.slice(5)),O(f,d,e[d])));m._data(f,"parsedAttrs",!0)}return e}return"object"==typeof a?this.each(function(){m.data(this,a)}):arguments.length>1?this.each(function(){m.data(this,a,b)}):f?O(f,a,m.data(f,a)):void 0},removeData:function(a){return this.each(function(){m.removeData(this,a)})}}),m.extend({queue:function(a,b,c){var d;return a?(b=(b||"fx")+"queue",d=m._data(a,b),c&&(!d||m.isArray(c)?d=m._data(a,b,m.makeArray(c)):d.push(c)),d||[]):void 0},dequeue:function(a,b){b=b||"fx";var c=m.queue(a,b),d=c.length,e=c.shift(),f=m._queueHooks(a,b),g=function(){m.dequeue(a,b)};"inprogress"===e&&(e=c.shift(),d--),e&&("fx"===b&&c.unshift("inprogress"),delete f.stop,e.call(a,g,f)),!d&&f&&f.empty.fire()},_queueHooks:function(a,b){var c=b+"queueHooks";return m._data(a,c)||m._data(a,c,{empty:m.Callbacks("once memory").add(function(){m._removeData(a,b+"queue"),m._removeData(a,c)})})}}),m.fn.extend({queue:function(a,b){var c=2;return"string"!=typeof a&&(b=a,a="fx",c--),arguments.lengthh;h++)b(a[h],c,g?d:d.call(a[h],h,b(a[h],c)));return e?a:j?b.call(a):i?b(a[0],c):f},W=/^(?:checkbox|radio)$/i;!function(){var a=y.createElement("input"),b=y.createElement("div"),c=y.createDocumentFragment();if(b.innerHTML="
                    a",k.leadingWhitespace=3===b.firstChild.nodeType,k.tbody=!b.getElementsByTagName("tbody").length,k.htmlSerialize=!!b.getElementsByTagName("link").length,k.html5Clone="<:nav>"!==y.createElement("nav").cloneNode(!0).outerHTML,a.type="checkbox",a.checked=!0,c.appendChild(a),k.appendChecked=a.checked,b.innerHTML="",k.noCloneChecked=!!b.cloneNode(!0).lastChild.defaultValue,c.appendChild(b),b.innerHTML="",k.checkClone=b.cloneNode(!0).cloneNode(!0).lastChild.checked,k.noCloneEvent=!0,b.attachEvent&&(b.attachEvent("onclick",function(){k.noCloneEvent=!1}),b.cloneNode(!0).click()),null==k.deleteExpando){k.deleteExpando=!0;try{delete b.test}catch(d){k.deleteExpando=!1}}}(),function(){var b,c,d=y.createElement("div");for(b in{submit:!0,change:!0,focusin:!0})c="on"+b,(k[b+"Bubbles"]=c in a)||(d.setAttribute(c,"t"),k[b+"Bubbles"]=d.attributes[c].expando===!1);d=null}();var X=/^(?:input|select|textarea)$/i,Y=/^key/,Z=/^(?:mouse|pointer|contextmenu)|click/,$=/^(?:focusinfocus|focusoutblur)$/,_=/^([^.]*)(?:\.(.+)|)$/;function ab(){return!0}function bb(){return!1}function cb(){try{return y.activeElement}catch(a){}}m.event={global:{},add:function(a,b,c,d,e){var f,g,h,i,j,k,l,n,o,p,q,r=m._data(a);if(r){c.handler&&(i=c,c=i.handler,e=i.selector),c.guid||(c.guid=m.guid++),(g=r.events)||(g=r.events={}),(k=r.handle)||(k=r.handle=function(a){return typeof m===K||a&&m.event.triggered===a.type?void 0:m.event.dispatch.apply(k.elem,arguments)},k.elem=a),b=(b||"").match(E)||[""],h=b.length;while(h--)f=_.exec(b[h])||[],o=q=f[1],p=(f[2]||"").split(".").sort(),o&&(j=m.event.special[o]||{},o=(e?j.delegateType:j.bindType)||o,j=m.event.special[o]||{},l=m.extend({type:o,origType:q,data:d,handler:c,guid:c.guid,selector:e,needsContext:e&&m.expr.match.needsContext.test(e),namespace:p.join(".")},i),(n=g[o])||(n=g[o]=[],n.delegateCount=0,j.setup&&j.setup.call(a,d,p,k)!==!1||(a.addEventListener?a.addEventListener(o,k,!1):a.attachEvent&&a.attachEvent("on"+o,k))),j.add&&(j.add.call(a,l),l.handler.guid||(l.handler.guid=c.guid)),e?n.splice(n.delegateCount++,0,l):n.push(l),m.event.global[o]=!0);a=null}},remove:function(a,b,c,d,e){var f,g,h,i,j,k,l,n,o,p,q,r=m.hasData(a)&&m._data(a);if(r&&(k=r.events)){b=(b||"").match(E)||[""],j=b.length;while(j--)if(h=_.exec(b[j])||[],o=q=h[1],p=(h[2]||"").split(".").sort(),o){l=m.event.special[o]||{},o=(d?l.delegateType:l.bindType)||o,n=k[o]||[],h=h[2]&&new RegExp("(^|\\.)"+p.join("\\.(?:.*\\.|)")+"(\\.|$)"),i=f=n.length;while(f--)g=n[f],!e&&q!==g.origType||c&&c.guid!==g.guid||h&&!h.test(g.namespace)||d&&d!==g.selector&&("**"!==d||!g.selector)||(n.splice(f,1),g.selector&&n.delegateCount--,l.remove&&l.remove.call(a,g));i&&!n.length&&(l.teardown&&l.teardown.call(a,p,r.handle)!==!1||m.removeEvent(a,o,r.handle),delete k[o])}else for(o in k)m.event.remove(a,o+b[j],c,d,!0);m.isEmptyObject(k)&&(delete r.handle,m._removeData(a,"events"))}},trigger:function(b,c,d,e){var f,g,h,i,k,l,n,o=[d||y],p=j.call(b,"type")?b.type:b,q=j.call(b,"namespace")?b.namespace.split("."):[];if(h=l=d=d||y,3!==d.nodeType&&8!==d.nodeType&&!$.test(p+m.event.triggered)&&(p.indexOf(".")>=0&&(q=p.split("."),p=q.shift(),q.sort()),g=p.indexOf(":")<0&&"on"+p,b=b[m.expando]?b:new m.Event(p,"object"==typeof b&&b),b.isTrigger=e?2:3,b.namespace=q.join("."),b.namespace_re=b.namespace?new RegExp("(^|\\.)"+q.join("\\.(?:.*\\.|)")+"(\\.|$)"):null,b.result=void 0,b.target||(b.target=d),c=null==c?[b]:m.makeArray(c,[b]),k=m.event.special[p]||{},e||!k.trigger||k.trigger.apply(d,c)!==!1)){if(!e&&!k.noBubble&&!m.isWindow(d)){for(i=k.delegateType||p,$.test(i+p)||(h=h.parentNode);h;h=h.parentNode)o.push(h),l=h;l===(d.ownerDocument||y)&&o.push(l.defaultView||l.parentWindow||a)}n=0;while((h=o[n++])&&!b.isPropagationStopped())b.type=n>1?i:k.bindType||p,f=(m._data(h,"events")||{})[b.type]&&m._data(h,"handle"),f&&f.apply(h,c),f=g&&h[g],f&&f.apply&&m.acceptData(h)&&(b.result=f.apply(h,c),b.result===!1&&b.preventDefault());if(b.type=p,!e&&!b.isDefaultPrevented()&&(!k._default||k._default.apply(o.pop(),c)===!1)&&m.acceptData(d)&&g&&d[p]&&!m.isWindow(d)){l=d[g],l&&(d[g]=null),m.event.triggered=p;try{d[p]()}catch(r){}m.event.triggered=void 0,l&&(d[g]=l)}return b.result}},dispatch:function(a){a=m.event.fix(a);var b,c,e,f,g,h=[],i=d.call(arguments),j=(m._data(this,"events")||{})[a.type]||[],k=m.event.special[a.type]||{};if(i[0]=a,a.delegateTarget=this,!k.preDispatch||k.preDispatch.call(this,a)!==!1){h=m.event.handlers.call(this,a,j),b=0;while((f=h[b++])&&!a.isPropagationStopped()){a.currentTarget=f.elem,g=0;while((e=f.handlers[g++])&&!a.isImmediatePropagationStopped())(!a.namespace_re||a.namespace_re.test(e.namespace))&&(a.handleObj=e,a.data=e.data,c=((m.event.special[e.origType]||{}).handle||e.handler).apply(f.elem,i),void 0!==c&&(a.result=c)===!1&&(a.preventDefault(),a.stopPropagation()))}return k.postDispatch&&k.postDispatch.call(this,a),a.result}},handlers:function(a,b){var c,d,e,f,g=[],h=b.delegateCount,i=a.target;if(h&&i.nodeType&&(!a.button||"click"!==a.type))for(;i!=this;i=i.parentNode||this)if(1===i.nodeType&&(i.disabled!==!0||"click"!==a.type)){for(e=[],f=0;h>f;f++)d=b[f],c=d.selector+" ",void 0===e[c]&&(e[c]=d.needsContext?m(c,this).index(i)>=0:m.find(c,this,null,[i]).length),e[c]&&e.push(d);e.length&&g.push({elem:i,handlers:e})}return h]","i"),hb=/^\s+/,ib=/<(?!area|br|col|embed|hr|img|input|link|meta|param)(([\w:]+)[^>]*)\/>/gi,jb=/<([\w:]+)/,kb=/\s*$/g,rb={option:[1,""],legend:[1,"
                    ","
                    "],area:[1,"",""],param:[1,"",""],thead:[1,"","
                    "],tr:[2,"","
                    "],col:[2,"","
                    "],td:[3,"","
                    "],_default:k.htmlSerialize?[0,"",""]:[1,"X
                    ","
                    "]},sb=db(y),tb=sb.appendChild(y.createElement("div"));rb.optgroup=rb.option,rb.tbody=rb.tfoot=rb.colgroup=rb.caption=rb.thead,rb.th=rb.td;function ub(a,b){var c,d,e=0,f=typeof a.getElementsByTagName!==K?a.getElementsByTagName(b||"*"):typeof a.querySelectorAll!==K?a.querySelectorAll(b||"*"):void 0;if(!f)for(f=[],c=a.childNodes||a;null!=(d=c[e]);e++)!b||m.nodeName(d,b)?f.push(d):m.merge(f,ub(d,b));return void 0===b||b&&m.nodeName(a,b)?m.merge([a],f):f}function vb(a){W.test(a.type)&&(a.defaultChecked=a.checked)}function wb(a,b){return m.nodeName(a,"table")&&m.nodeName(11!==b.nodeType?b:b.firstChild,"tr")?a.getElementsByTagName("tbody")[0]||a.appendChild(a.ownerDocument.createElement("tbody")):a}function xb(a){return a.type=(null!==m.find.attr(a,"type"))+"/"+a.type,a}function yb(a){var b=pb.exec(a.type);return b?a.type=b[1]:a.removeAttribute("type"),a}function zb(a,b){for(var c,d=0;null!=(c=a[d]);d++)m._data(c,"globalEval",!b||m._data(b[d],"globalEval"))}function Ab(a,b){if(1===b.nodeType&&m.hasData(a)){var c,d,e,f=m._data(a),g=m._data(b,f),h=f.events;if(h){delete g.handle,g.events={};for(c in h)for(d=0,e=h[c].length;e>d;d++)m.event.add(b,c,h[c][d])}g.data&&(g.data=m.extend({},g.data))}}function Bb(a,b){var c,d,e;if(1===b.nodeType){if(c=b.nodeName.toLowerCase(),!k.noCloneEvent&&b[m.expando]){e=m._data(b);for(d in e.events)m.removeEvent(b,d,e.handle);b.removeAttribute(m.expando)}"script"===c&&b.text!==a.text?(xb(b).text=a.text,yb(b)):"object"===c?(b.parentNode&&(b.outerHTML=a.outerHTML),k.html5Clone&&a.innerHTML&&!m.trim(b.innerHTML)&&(b.innerHTML=a.innerHTML)):"input"===c&&W.test(a.type)?(b.defaultChecked=b.checked=a.checked,b.value!==a.value&&(b.value=a.value)):"option"===c?b.defaultSelected=b.selected=a.defaultSelected:("input"===c||"textarea"===c)&&(b.defaultValue=a.defaultValue)}}m.extend({clone:function(a,b,c){var d,e,f,g,h,i=m.contains(a.ownerDocument,a);if(k.html5Clone||m.isXMLDoc(a)||!gb.test("<"+a.nodeName+">")?f=a.cloneNode(!0):(tb.innerHTML=a.outerHTML,tb.removeChild(f=tb.firstChild)),!(k.noCloneEvent&&k.noCloneChecked||1!==a.nodeType&&11!==a.nodeType||m.isXMLDoc(a)))for(d=ub(f),h=ub(a),g=0;null!=(e=h[g]);++g)d[g]&&Bb(e,d[g]);if(b)if(c)for(h=h||ub(a),d=d||ub(f),g=0;null!=(e=h[g]);g++)Ab(e,d[g]);else Ab(a,f);return d=ub(f,"script"),d.length>0&&zb(d,!i&&ub(a,"script")),d=h=e=null,f},buildFragment:function(a,b,c,d){for(var e,f,g,h,i,j,l,n=a.length,o=db(b),p=[],q=0;n>q;q++)if(f=a[q],f||0===f)if("object"===m.type(f))m.merge(p,f.nodeType?[f]:f);else if(lb.test(f)){h=h||o.appendChild(b.createElement("div")),i=(jb.exec(f)||["",""])[1].toLowerCase(),l=rb[i]||rb._default,h.innerHTML=l[1]+f.replace(ib,"<$1>")+l[2],e=l[0];while(e--)h=h.lastChild;if(!k.leadingWhitespace&&hb.test(f)&&p.push(b.createTextNode(hb.exec(f)[0])),!k.tbody){f="table"!==i||kb.test(f)?""!==l[1]||kb.test(f)?0:h:h.firstChild,e=f&&f.childNodes.length;while(e--)m.nodeName(j=f.childNodes[e],"tbody")&&!j.childNodes.length&&f.removeChild(j)}m.merge(p,h.childNodes),h.textContent="";while(h.firstChild)h.removeChild(h.firstChild);h=o.lastChild}else p.push(b.createTextNode(f));h&&o.removeChild(h),k.appendChecked||m.grep(ub(p,"input"),vb),q=0;while(f=p[q++])if((!d||-1===m.inArray(f,d))&&(g=m.contains(f.ownerDocument,f),h=ub(o.appendChild(f),"script"),g&&zb(h),c)){e=0;while(f=h[e++])ob.test(f.type||"")&&c.push(f)}return h=null,o},cleanData:function(a,b){for(var d,e,f,g,h=0,i=m.expando,j=m.cache,l=k.deleteExpando,n=m.event.special;null!=(d=a[h]);h++)if((b||m.acceptData(d))&&(f=d[i],g=f&&j[f])){if(g.events)for(e in g.events)n[e]?m.event.remove(d,e):m.removeEvent(d,e,g.handle);j[f]&&(delete j[f],l?delete d[i]:typeof d.removeAttribute!==K?d.removeAttribute(i):d[i]=null,c.push(f))}}}),m.fn.extend({text:function(a){return V(this,function(a){return void 0===a?m.text(this):this.empty().append((this[0]&&this[0].ownerDocument||y).createTextNode(a))},null,a,arguments.length)},append:function(){return this.domManip(arguments,function(a){if(1===this.nodeType||11===this.nodeType||9===this.nodeType){var b=wb(this,a);b.appendChild(a)}})},prepend:function(){return this.domManip(arguments,function(a){if(1===this.nodeType||11===this.nodeType||9===this.nodeType){var b=wb(this,a);b.insertBefore(a,b.firstChild)}})},before:function(){return this.domManip(arguments,function(a){this.parentNode&&this.parentNode.insertBefore(a,this)})},after:function(){return this.domManip(arguments,function(a){this.parentNode&&this.parentNode.insertBefore(a,this.nextSibling)})},remove:function(a,b){for(var c,d=a?m.filter(a,this):this,e=0;null!=(c=d[e]);e++)b||1!==c.nodeType||m.cleanData(ub(c)),c.parentNode&&(b&&m.contains(c.ownerDocument,c)&&zb(ub(c,"script")),c.parentNode.removeChild(c));return this},empty:function(){for(var a,b=0;null!=(a=this[b]);b++){1===a.nodeType&&m.cleanData(ub(a,!1));while(a.firstChild)a.removeChild(a.firstChild);a.options&&m.nodeName(a,"select")&&(a.options.length=0)}return this},clone:function(a,b){return a=null==a?!1:a,b=null==b?a:b,this.map(function(){return m.clone(this,a,b)})},html:function(a){return V(this,function(a){var b=this[0]||{},c=0,d=this.length;if(void 0===a)return 1===b.nodeType?b.innerHTML.replace(fb,""):void 0;if(!("string"!=typeof a||mb.test(a)||!k.htmlSerialize&&gb.test(a)||!k.leadingWhitespace&&hb.test(a)||rb[(jb.exec(a)||["",""])[1].toLowerCase()])){a=a.replace(ib,"<$1>");try{for(;d>c;c++)b=this[c]||{},1===b.nodeType&&(m.cleanData(ub(b,!1)),b.innerHTML=a);b=0}catch(e){}}b&&this.empty().append(a)},null,a,arguments.length)},replaceWith:function(){var a=arguments[0];return this.domManip(arguments,function(b){a=this.parentNode,m.cleanData(ub(this)),a&&a.replaceChild(b,this)}),a&&(a.length||a.nodeType)?this:this.remove()},detach:function(a){return this.remove(a,!0)},domManip:function(a,b){a=e.apply([],a);var c,d,f,g,h,i,j=0,l=this.length,n=this,o=l-1,p=a[0],q=m.isFunction(p);if(q||l>1&&"string"==typeof p&&!k.checkClone&&nb.test(p))return this.each(function(c){var d=n.eq(c);q&&(a[0]=p.call(this,c,d.html())),d.domManip(a,b)});if(l&&(i=m.buildFragment(a,this[0].ownerDocument,!1,this),c=i.firstChild,1===i.childNodes.length&&(i=c),c)){for(g=m.map(ub(i,"script"),xb),f=g.length;l>j;j++)d=i,j!==o&&(d=m.clone(d,!0,!0),f&&m.merge(g,ub(d,"script"))),b.call(this[j],d,j);if(f)for(h=g[g.length-1].ownerDocument,m.map(g,yb),j=0;f>j;j++)d=g[j],ob.test(d.type||"")&&!m._data(d,"globalEval")&&m.contains(h,d)&&(d.src?m._evalUrl&&m._evalUrl(d.src):m.globalEval((d.text||d.textContent||d.innerHTML||"").replace(qb,"")));i=c=null}return this}}),m.each({appendTo:"append",prependTo:"prepend",insertBefore:"before",insertAfter:"after",replaceAll:"replaceWith"},function(a,b){m.fn[a]=function(a){for(var c,d=0,e=[],g=m(a),h=g.length-1;h>=d;d++)c=d===h?this:this.clone(!0),m(g[d])[b](c),f.apply(e,c.get());return this.pushStack(e)}});var Cb,Db={};function Eb(b,c){var d,e=m(c.createElement(b)).appendTo(c.body),f=a.getDefaultComputedStyle&&(d=a.getDefaultComputedStyle(e[0]))?d.display:m.css(e[0],"display");return e.detach(),f}function Fb(a){var b=y,c=Db[a];return c||(c=Eb(a,b),"none"!==c&&c||(Cb=(Cb||m("" : "" ) + + "" + + "" + (function(){ + return (settings.imageUpload) ? "
                    " + + "" + + "" + + "
                    " : ""; + })() + + "
                    " + + "" + + "" + + "
                    " + + "" + + "" + + "
                    " + + ( (settings.imageUpload) ? "" : ""); + + //var imageFooterHTML = ""; + + dialog = this.createDialog({ + title : imageLang.title, + width : (settings.imageUpload) ? 465 : 380, + height : 254, + name : dialogName, + content : dialogContent, + mask : settings.dialogShowMask, + drag : settings.dialogDraggable, + lockScreen : settings.dialogLockScreen, + maskStyle : { + opacity : settings.dialogMaskOpacity, + backgroundColor : settings.dialogMaskBgColor + }, + buttons : { + enter : [lang.buttons.enter, function() { + var url = this.find("[data-url]").val(); + var alt = this.find("[data-alt]").val(); + var link = this.find("[data-link]").val(); + + if (url === "") + { + alert(imageLang.imageURLEmpty); + return false; + } + + var altAttr = (alt !== "") ? " \"" + alt + "\"" : ""; + + if (link === "" || link === "http://") + { + cm.replaceSelection("![" + alt + "](" + url + altAttr + ")"); + } + else + { + cm.replaceSelection("[![" + alt + "](" + url + altAttr + ")](" + link + altAttr + ")"); + } + + if (alt === "") { + cm.setCursor(cursor.line, cursor.ch + 2); + } + + this.hide().lockScreen(false).hideMask(); + + return false; + }], + + cancel : [lang.buttons.cancel, function() { + this.hide().lockScreen(false).hideMask(); + + return false; + }] + } + }); + + dialog.attr("id", classPrefix + "image-dialog-" + guid); + + if (!settings.imageUpload) { + return ; + } + + var fileInput = dialog.find("[name=\"" + classPrefix + "image-file\"]"); + + fileInput.bind("change", function() { + var fileName = fileInput.val(); + var isImage = new RegExp("(\\.(" + settings.imageFormats.join("|") + "))$"); // /(\.(webp|jpg|jpeg|gif|bmp|png))$/ + + if (fileName === "") + { + alert(imageLang.uploadFileEmpty); + + return false; + } + + if (!isImage.test(fileName)) + { + alert(imageLang.formatNotAllowed + settings.imageFormats.join(", ")); + + return false; + } + + loading(true); + + var submitHandler = function() { + + var uploadIframe = document.getElementById(iframeName); + + uploadIframe.onload = function() { + + loading(false); + + var body = (uploadIframe.contentWindow ? uploadIframe.contentWindow : uploadIframe.contentDocument).document.body; + var json = (body.innerText) ? body.innerText : ( (body.textContent) ? body.textContent : null); + + json = (typeof JSON.parse !== "undefined") ? JSON.parse(json) : eval("(" + json + ")"); + + if(!settings.crossDomainUpload) + { + if (json.success === 1) + { + dialog.find("[data-url]").val(json.url); + } + else + { + alert(json.message); + } + } + + return false; + }; + }; + + dialog.find("[type=\"submit\"]").bind("click", submitHandler).trigger("click"); + }); + } + + dialog = editor.find("." + dialogName); + dialog.find("[type=\"text\"]").val(""); + dialog.find("[type=\"file\"]").val(""); + dialog.find("[data-link]").val("http://"); + + this.dialogShowMask(dialog); + this.dialogLockScreen(); + dialog.show(); + + }; + + }; + + // CommonJS/Node.js + if (typeof require === "function" && typeof exports === "object" && typeof module === "object") + { + module.exports = factory; + } + else if (typeof define === "function") // AMD/CMD/Sea.js + { + if (define.amd) { // for Require.js + + define(["editormd"], function(editormd) { + factory(editormd); + }); + + } else { // for Sea.js + define(function(require) { + var editormd = require("./../../editormd"); + factory(editormd); + }); + } + } + else + { + factory(window.editormd); + } + +})(); diff --git a/system/admin/editor.md/plugins/link-dialog/link-dialog.js b/system/admin/editor.md/plugins/link-dialog/link-dialog.js new file mode 100644 index 00000000..3e1d0bf0 --- /dev/null +++ b/system/admin/editor.md/plugins/link-dialog/link-dialog.js @@ -0,0 +1,133 @@ +/*! + * Link dialog plugin for Editor.md + * + * @file link-dialog.js + * @author pandao + * @version 1.2.1 + * @updateTime 2015-06-09 + * {@link https://github.com/pandao/editor.md} + * @license MIT + */ + +(function() { + + var factory = function (exports) { + + var pluginName = "link-dialog"; + + exports.fn.linkDialog = function() { + + var _this = this; + var cm = this.cm; + var editor = this.editor; + var settings = this.settings; + var selection = cm.getSelection(); + var lang = this.lang; + var linkLang = lang.dialog.link; + var classPrefix = this.classPrefix; + var dialogName = classPrefix + pluginName, dialog; + + cm.focus(); + + if (editor.find("." + dialogName).length > 0) + { + dialog = editor.find("." + dialogName); + dialog.find("[data-url]").val("http://"); + dialog.find("[data-title]").val(selection); + + this.dialogShowMask(dialog); + this.dialogLockScreen(); + dialog.show(); + } + else + { + var dialogHTML = "
                    " + + "" + + "" + + "
                    " + + "" + + "" + + "
                    " + + "
                    "; + + dialog = this.createDialog({ + title : linkLang.title, + width : 380, + height : 211, + content : dialogHTML, + mask : settings.dialogShowMask, + drag : settings.dialogDraggable, + lockScreen : settings.dialogLockScreen, + maskStyle : { + opacity : settings.dialogMaskOpacity, + backgroundColor : settings.dialogMaskBgColor + }, + buttons : { + enter : [lang.buttons.enter, function() { + var url = this.find("[data-url]").val(); + var title = this.find("[data-title]").val(); + + if (url === "http://" || url === "") + { + alert(linkLang.urlEmpty); + return false; + } + + /*if (title === "") + { + alert(linkLang.titleEmpty); + return false; + }*/ + + var str = "[" + title + "](" + url + " \"" + title + "\")"; + + if (title == "") + { + str = "[" + url + "](" + url + ")"; + } + + cm.replaceSelection(str); + + this.hide().lockScreen(false).hideMask(); + + return false; + }], + + cancel : [lang.buttons.cancel, function() { + this.hide().lockScreen(false).hideMask(); + + return false; + }] + } + }); + } + }; + + }; + + // CommonJS/Node.js + if (typeof require === "function" && typeof exports === "object" && typeof module === "object") + { + module.exports = factory; + } + else if (typeof define === "function") // AMD/CMD/Sea.js + { + if (define.amd) { // for Require.js + + define(["editormd"], function(editormd) { + factory(editormd); + }); + + } else { // for Sea.js + define(function(require) { + var editormd = require("./../../editormd"); + factory(editormd); + }); + } + } + else + { + factory(window.editormd); + } + +})(); diff --git a/system/admin/editor.md/plugins/plugin-template.js b/system/admin/editor.md/plugins/plugin-template.js new file mode 100644 index 00000000..8e301697 --- /dev/null +++ b/system/admin/editor.md/plugins/plugin-template.js @@ -0,0 +1,111 @@ +/*! + * Link dialog plugin for Editor.md + * + * @file link-dialog.js + * @author pandao + * @version 1.2.0 + * @updateTime 2015-03-07 + * {@link https://github.com/pandao/editor.md} + * @license MIT + */ + +(function() { + + var factory = function (exports) { + + var $ = jQuery; // if using module loader(Require.js/Sea.js). + + var langs = { + "zh-cn" : { + toolbar : { + table : "表格" + }, + dialog : { + table : { + title : "添加表格", + cellsLabel : "单元格数", + alignLabel : "对齐方式", + rows : "行数", + cols : "列数", + aligns : ["默认", "左对齐", "居中对齐", "右对齐"] + } + } + }, + "zh-tw" : { + toolbar : { + table : "添加表格" + }, + dialog : { + table : { + title : "添加表格", + cellsLabel : "單元格數", + alignLabel : "對齊方式", + rows : "行數", + cols : "列數", + aligns : ["默認", "左對齊", "居中對齊", "右對齊"] + } + } + }, + "en" : { + toolbar : { + table : "Tables" + }, + dialog : { + table : { + title : "Tables", + cellsLabel : "Cells", + alignLabel : "Align", + rows : "Rows", + cols : "Cols", + aligns : ["Default", "Left align", "Center align", "Right align"] + } + } + } + }; + + exports.fn.htmlEntities = function() { + /* + var _this = this; // this == the current instance object of Editor.md + var lang = _this.lang; + var settings = _this.settings; + var editor = this.editor; + var cursor = cm.getCursor(); + var selection = cm.getSelection(); + var classPrefix = this.classPrefix; + + $.extend(true, this.lang, langs[this.lang.name]); // l18n + this.setToolbar(); + + cm.focus(); + */ + //.... + }; + + }; + + // CommonJS/Node.js + if (typeof require === "function" && typeof exports === "object" && typeof module === "object") + { + module.exports = factory; + } + else if (typeof define === "function") // AMD/CMD/Sea.js + { + if (define.amd) { // for Require.js + + define(["editormd"], function(editormd) { + factory(editormd); + }); + + } else { // for Sea.js + define(function(require) { + var editormd = require("./../../editormd"); + factory(editormd); + }); + } + } + else + { + factory(window.editormd); + } + +})(); diff --git a/system/admin/editor.md/plugins/preformatted-text-dialog/preformatted-text-dialog.js b/system/admin/editor.md/plugins/preformatted-text-dialog/preformatted-text-dialog.js new file mode 100644 index 00000000..c890adc6 --- /dev/null +++ b/system/admin/editor.md/plugins/preformatted-text-dialog/preformatted-text-dialog.js @@ -0,0 +1,172 @@ +/*! + * Preformatted text dialog plugin for Editor.md + * + * @file preformatted-text-dialog.js + * @author pandao + * @version 1.2.0 + * @updateTime 2015-03-07 + * {@link https://github.com/pandao/editor.md} + * @license MIT + */ + +(function() { + + var factory = function (exports) { + var cmEditor; + var pluginName = "preformatted-text-dialog"; + + exports.fn.preformattedTextDialog = function() { + + var _this = this; + var cm = this.cm; + var lang = this.lang; + var editor = this.editor; + var settings = this.settings; + var cursor = cm.getCursor(); + var selection = cm.getSelection(); + var classPrefix = this.classPrefix; + var dialogLang = lang.dialog.preformattedText; + var dialogName = classPrefix + pluginName, dialog; + + cm.focus(); + + if (editor.find("." + dialogName).length > 0) + { + dialog = editor.find("." + dialogName); + dialog.find("textarea").val(selection); + + this.dialogShowMask(dialog); + this.dialogLockScreen(); + dialog.show(); + } + else + { + var dialogContent = ""; + + dialog = this.createDialog({ + name : dialogName, + title : dialogLang.title, + width : 780, + height : 540, + mask : settings.dialogShowMask, + drag : settings.dialogDraggable, + content : dialogContent, + lockScreen : settings.dialogLockScreen, + maskStyle : { + opacity : settings.dialogMaskOpacity, + backgroundColor : settings.dialogMaskBgColor + }, + buttons : { + enter : [lang.buttons.enter, function() { + var codeTexts = this.find("textarea").val(); + + if (codeTexts === "") + { + alert(dialogLang.emptyAlert); + return false; + } + + codeTexts = codeTexts.split("\n"); + + for (var i in codeTexts) + { + codeTexts[i] = " " + codeTexts[i]; + } + + codeTexts = codeTexts.join("\n"); + + if (cursor.ch !== 0) { + codeTexts = "\r\n\r\n" + codeTexts; + } + + cm.replaceSelection(codeTexts); + + this.hide().lockScreen(false).hideMask(); + + return false; + }], + cancel : [lang.buttons.cancel, function() { + this.hide().lockScreen(false).hideMask(); + + return false; + }] + } + }); + } + + var cmConfig = { + mode : "text/html", + theme : settings.theme, + tabSize : 4, + autofocus : true, + autoCloseTags : true, + indentUnit : 4, + lineNumbers : true, + lineWrapping : true, + extraKeys : {"Ctrl-Q": function(cm){ cm.foldCode(cm.getCursor()); }}, + foldGutter : true, + gutters : ["CodeMirror-linenumbers", "CodeMirror-foldgutter"], + matchBrackets : true, + indentWithTabs : true, + styleActiveLine : true, + styleSelectedText : true, + autoCloseBrackets : true, + showTrailingSpace : true, + highlightSelectionMatches : true + }; + + var textarea = dialog.find("textarea"); + var cmObj = dialog.find(".CodeMirror"); + + if (dialog.find(".CodeMirror").length < 1) + { + cmEditor = exports.$CodeMirror.fromTextArea(textarea[0], cmConfig); + cmObj = dialog.find(".CodeMirror"); + + cmObj.css({ + "float" : "none", + margin : "0 0 5px", + border : "1px solid #ddd", + fontSize : settings.fontSize, + width : "100%", + height : "410px" + }); + + cmEditor.on("change", function(cm) { + textarea.val(cm.getValue()); + }); + } + else + { + cmEditor.setValue(cm.getSelection()); + } + }; + + }; + + // CommonJS/Node.js + if (typeof require === "function" && typeof exports === "object" && typeof module === "object") + { + module.exports = factory; + } + else if (typeof define === "function") // AMD/CMD/Sea.js + { + if (define.amd) { // for Require.js + + define(["editormd"], function(editormd) { + factory(editormd); + }); + + } else { // for Sea.js + define(function(require) { + var editormd = require("./../../editormd"); + factory(editormd); + }); + } + } + else + { + factory(window.editormd); + } + +})(); diff --git a/system/admin/editor.md/plugins/reference-link-dialog/reference-link-dialog.js b/system/admin/editor.md/plugins/reference-link-dialog/reference-link-dialog.js new file mode 100644 index 00000000..f1ad086b --- /dev/null +++ b/system/admin/editor.md/plugins/reference-link-dialog/reference-link-dialog.js @@ -0,0 +1,153 @@ +/*! + * Reference link dialog plugin for Editor.md + * + * @file reference-link-dialog.js + * @author pandao + * @version 1.2.1 + * @updateTime 2015-06-09 + * {@link https://github.com/pandao/editor.md} + * @license MIT + */ + +(function() { + + var factory = function (exports) { + + var pluginName = "reference-link-dialog"; + var ReLinkId = 1; + + exports.fn.referenceLinkDialog = function() { + + var _this = this; + var cm = this.cm; + var lang = this.lang; + var editor = this.editor; + var settings = this.settings; + var cursor = cm.getCursor(); + var selection = cm.getSelection(); + var dialogLang = lang.dialog.referenceLink; + var classPrefix = this.classPrefix; + var dialogName = classPrefix + pluginName, dialog; + + cm.focus(); + + if (editor.find("." + dialogName).length < 1) + { + var dialogHTML = "
                    " + + "" + + "" + + "
                    " + + "" + + "" + + "
                    " + + "" + + "" + + "
                    " + + "" + + "" + + "
                    " + + "
                    "; + + dialog = this.createDialog({ + name : dialogName, + title : dialogLang.title, + width : 380, + height : 296, + content : dialogHTML, + mask : settings.dialogShowMask, + drag : settings.dialogDraggable, + lockScreen : settings.dialogLockScreen, + maskStyle : { + opacity : settings.dialogMaskOpacity, + backgroundColor : settings.dialogMaskBgColor + }, + buttons : { + enter : [lang.buttons.enter, function() { + var name = this.find("[data-name]").val(); + var url = this.find("[data-url]").val(); + var rid = this.find("[data-url-id]").val(); + var title = this.find("[data-title]").val(); + + if (name === "") + { + alert(dialogLang.nameEmpty); + return false; + } + + if (rid === "") + { + alert(dialogLang.idEmpty); + return false; + } + + if (url === "http://" || url === "") + { + alert(dialogLang.urlEmpty); + return false; + } + + //cm.replaceSelection("[" + title + "][" + name + "]\n[" + name + "]: " + url + ""); + cm.replaceSelection("[" + name + "][" + rid + "]"); + + if (selection === "") { + cm.setCursor(cursor.line, cursor.ch + 1); + } + + title = (title === "") ? "" : " \"" + title + "\""; + + cm.setValue(cm.getValue() + "\n[" + rid + "]: " + url + title + ""); + + this.hide().lockScreen(false).hideMask(); + + return false; + }], + cancel : [lang.buttons.cancel, function() { + this.hide().lockScreen(false).hideMask(); + + return false; + }] + } + }); + } + + dialog = editor.find("." + dialogName); + dialog.find("[data-name]").val("[" + ReLinkId + "]"); + dialog.find("[data-url-id]").val(""); + dialog.find("[data-url]").val("http://"); + dialog.find("[data-title]").val(selection); + + this.dialogShowMask(dialog); + this.dialogLockScreen(); + dialog.show(); + + ReLinkId++; + }; + + }; + + // CommonJS/Node.js + if (typeof require === "function" && typeof exports === "object" && typeof module === "object") + { + module.exports = factory; + } + else if (typeof define === "function") // AMD/CMD/Sea.js + { + if (define.amd) { // for Require.js + + define(["editormd"], function(editormd) { + factory(editormd); + }); + + } else { // for Sea.js + define(function(require) { + var editormd = require("./../../editormd"); + factory(editormd); + }); + } + } + else + { + factory(window.editormd); + } + +})(); diff --git a/system/admin/editor.md/plugins/table-dialog/table-dialog.js b/system/admin/editor.md/plugins/table-dialog/table-dialog.js new file mode 100644 index 00000000..578adf25 --- /dev/null +++ b/system/admin/editor.md/plugins/table-dialog/table-dialog.js @@ -0,0 +1,218 @@ +/*! + * Table dialog plugin for Editor.md + * + * @file table-dialog.js + * @author pandao + * @version 1.2.1 + * @updateTime 2015-06-09 + * {@link https://github.com/pandao/editor.md} + * @license MIT + */ + +(function() { + + var factory = function (exports) { + + var $ = jQuery; + var pluginName = "table-dialog"; + + var langs = { + "zh-cn" : { + toolbar : { + table : "表格" + }, + dialog : { + table : { + title : "添加表格", + cellsLabel : "单元格数", + alignLabel : "对齐方式", + rows : "行数", + cols : "列数", + aligns : ["默认", "左对齐", "居中对齐", "右对齐"] + } + } + }, + "zh-tw" : { + toolbar : { + table : "添加表格" + }, + dialog : { + table : { + title : "添加表格", + cellsLabel : "單元格數", + alignLabel : "對齊方式", + rows : "行數", + cols : "列數", + aligns : ["默認", "左對齊", "居中對齊", "右對齊"] + } + } + }, + "en" : { + toolbar : { + table : "Tables" + }, + dialog : { + table : { + title : "Tables", + cellsLabel : "Cells", + alignLabel : "Align", + rows : "Rows", + cols : "Cols", + aligns : ["Default", "Left align", "Center align", "Right align"] + } + } + } + }; + + exports.fn.tableDialog = function() { + var _this = this; + var cm = this.cm; + var editor = this.editor; + var settings = this.settings; + var path = settings.path + "../plugins/" + pluginName +"/"; + var classPrefix = this.classPrefix; + var dialogName = classPrefix + pluginName, dialog; + + $.extend(true, this.lang, langs[this.lang.name]); + this.setToolbar(); + + var lang = this.lang; + var dialogLang = lang.dialog.table; + + var dialogContent = [ + "
                    ", + "", + dialogLang.rows + "   ", + dialogLang.cols + "
                    ", + "", + "
                    ", + "
                    " + ].join("\n"); + + if (editor.find("." + dialogName).length > 0) + { + dialog = editor.find("." + dialogName); + + this.dialogShowMask(dialog); + this.dialogLockScreen(); + dialog.show(); + } + else + { + dialog = this.createDialog({ + name : dialogName, + title : dialogLang.title, + width : 360, + height : 226, + mask : settings.dialogShowMask, + drag : settings.dialogDraggable, + content : dialogContent, + lockScreen : settings.dialogLockScreen, + maskStyle : { + opacity : settings.dialogMaskOpacity, + backgroundColor : settings.dialogMaskBgColor + }, + buttons : { + enter : [lang.buttons.enter, function() { + var rows = parseInt(this.find("[data-rows]").val()); + var cols = parseInt(this.find("[data-cols]").val()); + var align = this.find("[name=\"table-align\"]:checked").val(); + var table = ""; + var hrLine = "------------"; + + var alignSign = { + _default : hrLine, + left : ":" + hrLine, + center : ":" + hrLine + ":", + right : hrLine + ":" + }; + + if ( rows > 1 && cols > 0) + { + for (var r = 0, len = rows; r < len; r++) + { + var row = []; + var head = []; + + for (var c = 0, len2 = cols; c < len2; c++) + { + if (r === 1) { + head.push(alignSign[align]); + } + + row.push(" "); + } + + if (r === 1) { + table += "| " + head.join(" | ") + " |" + "\n"; + } + + table += "| " + row.join( (cols === 1) ? "" : " | " ) + " |" + "\n"; + } + } + + cm.replaceSelection(table); + + this.hide().lockScreen(false).hideMask(); + + return false; + }], + + cancel : [lang.buttons.cancel, function() { + this.hide().lockScreen(false).hideMask(); + + return false; + }] + } + }); + } + + var faBtns = dialog.find(".fa-btns"); + + if (faBtns.html() === "") + { + var icons = ["align-justify", "align-left", "align-center", "align-right"]; + var _lang = dialogLang.aligns; + var values = ["_default", "left", "center", "right"]; + + for (var i = 0, len = icons.length; i < len; i++) + { + var checked = (i === 0) ? " checked=\"checked\"" : ""; + var btn = ""; + + faBtns.append(btn); + } + } + }; + + }; + + // CommonJS/Node.js + if (typeof require === "function" && typeof exports === "object" && typeof module === "object") + { + module.exports = factory; + } + else if (typeof define === "function") // AMD/CMD/Sea.js + { + if (define.amd) { // for Require.js + + define(["editormd"], function(editormd) { + factory(editormd); + }); + + } else { // for Sea.js + define(function(require) { + var editormd = require("./../../editormd"); + factory(editormd); + }); + } + } + else + { + factory(window.editormd); + } + +})(); diff --git a/system/admin/editor.md/plugins/test-plugin/test-plugin.js b/system/admin/editor.md/plugins/test-plugin/test-plugin.js new file mode 100644 index 00000000..bc4da31d --- /dev/null +++ b/system/admin/editor.md/plugins/test-plugin/test-plugin.js @@ -0,0 +1,66 @@ +/*! + * Test plugin for Editor.md + * + * @file test-plugin.js + * @author pandao + * @version 1.2.0 + * @updateTime 2015-03-07 + * {@link https://github.com/pandao/editor.md} + * @license MIT + */ + +(function() { + + var factory = function (exports) { + + var $ = jQuery; // if using module loader(Require.js/Sea.js). + + exports.testPlugin = function(){ + alert("testPlugin"); + }; + + exports.fn.testPluginMethodA = function() { + /* + var _this = this; // this == the current instance object of Editor.md + var lang = _this.lang; + var settings = _this.settings; + var editor = this.editor; + var cursor = cm.getCursor(); + var selection = cm.getSelection(); + var classPrefix = this.classPrefix; + + cm.focus(); + */ + //.... + + alert("testPluginMethodA"); + }; + + }; + + // CommonJS/Node.js + if (typeof require === "function" && typeof exports === "object" && typeof module === "object") + { + module.exports = factory; + } + else if (typeof define === "function") // AMD/CMD/Sea.js + { + if (define.amd) { // for Require.js + + define(["editormd"], function(editormd) { + factory(editormd); + }); + + } else { // for Sea.js + define(function(require) { + var editormd = require("./../../editormd"); + factory(editormd); + }); + } + } + else + { + factory(window.editormd); + } + +})(); diff --git a/system/admin/editor.md/scss/editormd.codemirror.scss b/system/admin/editor.md/scss/editormd.codemirror.scss new file mode 100644 index 00000000..3996bade --- /dev/null +++ b/system/admin/editor.md/scss/editormd.codemirror.scss @@ -0,0 +1,90 @@ +@charset "UTF-8"; + +.editormd .CodeMirror, #{$prefix}preview { + display: inline-block; + width: 50%; + height: 100%; + vertical-align: top; + @include box-sizing(border-box); + margin: 0; +} + +#{$prefix}preview { + position: absolute; + top: 35px; + right: 0; + right: -1px\0; + overflow: auto; + line-height: 1.6; + display: none; + background: #fff; + z-index:10; +} + +.editormd { + + .CodeMirror { + z-index: 10; + float: left; + border-right: 1px solid $borderColor; + font-size: 14px; + font-family: "YaHei Consolas Hybrid", Consolas, "微软雅黑", "Meiryo UI", "Malgun Gothic", "Segoe UI", "Trebuchet MS", Helvetica, "Monaco", courier, monospace; + line-height: 1.6; + margin-top: 35px; + + pre { + font-size: 14px; + padding: 0 12px; + } + } + + .CodeMirror-linenumbers { + padding: 0 5px; + } + + .CodeMirror-selected { + background: #70B7FF; + } + + .CodeMirror-focused .CodeMirror-selected { + background: #70B7FF; + } + + .CodeMirror, .CodeMirror-scroll, #{$prefix}preview { + -webkit-overflow-scrolling : touch; + } + + .styled-background { + background-color: #ff7; + } + + .CodeMirror-focused .cm-matchhighlight { + background-image: url(); + background-position: bottom; + background-repeat: repeat-x; + } + + .CodeMirror-empty { + //outline: 1px solid #c22; + + &.CodeMirror-focused { + outline: none; + } + } + + .CodeMirror pre.CodeMirror-placeholder { + color: #999; + } + + .cm-trailingspace { + background-image: url(); + background-position: bottom left; + background-repeat: repeat-x; + } + + .cm-tab { + background: url(); + background-position: right; + background-repeat: no-repeat; + } +} diff --git a/system/admin/editor.md/scss/editormd.dialog.scss b/system/admin/editor.md/scss/editormd.dialog.scss new file mode 100644 index 00000000..609f6a9a --- /dev/null +++ b/system/admin/editor.md/scss/editormd.dialog.scss @@ -0,0 +1,184 @@ +@charset "UTF-8"; + +#{$prefix}dialog { + color: $color; + position: fixed; + z-index: 99999; + display: none; + @include border-radius(3px); + @include box-shadow(0 0 10px rgba(0, 0, 0, 0.3)); + //@include user-select(none); + background: #fff; + font-size: 14px; +} + +#{$prefix}dialog-container { + position: relative; + padding: 20px; + line-height: 1.4; + + h1 { + font-size: 24px; + margin-bottom: 10px; + + .fa { + color: #2C7EEA; + padding-right: 5px; + } + + small { + padding-left: 5px; + font-weight: normal; + font-size: 12px; + color: #999; + } + } + + select { + color: #999; + padding: 3px 8px; + border: 1px solid $borderColor; + } +} + +#{$prefix}dialog-close { + position: absolute; + top: 12px; + right: 15px; + font-size: 18px; + color: #ccc; + @include transition(color 300ms ease-out); + + &:hover { + color: #999; + } +} + +#{$prefix}dialog-header { + padding: 11px 20px; + border-bottom: 1px solid #eee; + @include transition(background 300ms ease-out); + + &:hover { + background: #f6f6f6; + } +} + +#{$prefix}dialog-title { + font-size: 14px; +} + +#{$prefix}dialog-footer { + padding: 10px 0 0 0; + text-align: right; +} + +#{$prefix}dialog-info { + width: 420px; + + h1 { + font-weight: normal; + } + + #{$prefix}dialog-container { + padding: 20px 25px 25px; + } + + #{$prefix}dialog-close { + top: 10px; + right: 10px; + } + + p > a, .hover-link:hover { + color: #2196F3; + } + + .hover-link { + color: #666; + } + + a { + .fa-external-link { + display: none; + } + + &:hover { + color: #2196F3; + + .fa-external-link { + display: inline-block; + } + } + } +} + +#{$prefix}mask, +#{$prefix}container-mask, +#{$prefix}dialog-mask { + display: none; + width: 100%; + height: 100%; + position: absolute; + top: 0; + left: 0; +} + +#{$prefix}mask, +#{$prefix}dialog-mask-bg { + background: #fff; + opacity: 0.5; + filter: alpha(opacity=50); +} + +#{$prefix}mask { + position: fixed; + background: #000; + @include opacity(0.2); + z-index: 99998; +} + +#{$prefix}container-mask, +#{$prefix}dialog-mask-con { + background: url(../images/loading.gif) no-repeat center center; + @include background-size(32px 32px); +} + +#{$prefix}container-mask { + z-index: 20; + display: block; + background-color: #fff; +} + +@media only screen and (-webkit-min-device-pixel-ratio: 2), only screen and (min-device-pixel-ratio: 2) { + #{$prefix}container-mask, + #{$prefix}dialog-mask-con { + background-image: url(../images/loading@2x.gif); + } +} + +@media only screen and (-webkit-min-device-pixel-ratio: 3), only screen and (min-device-pixel-ratio: 3) { + #{$prefix}container-mask, + #{$prefix}dialog-mask-con { + background-image: url(../images/loading@3x.gif); + } +} + +#{$prefix}code-block-dialog, +#{$prefix}preformatted-text-dialog { + textarea { + width: 100%; + height: 400px; + margin-bottom: 6px; + overflow: auto; + border: 1px solid #eee; + background: #fff; + padding: 15px; + resize: none; + } +} + +#{$prefix}code-toolbar { + color: #999; + font-size: 14px; + margin: -5px 0 10px; +} \ No newline at end of file diff --git a/system/admin/editor.md/scss/editormd.form.scss b/system/admin/editor.md/scss/editormd.form.scss new file mode 100644 index 00000000..e91456da --- /dev/null +++ b/system/admin/editor.md/scss/editormd.form.scss @@ -0,0 +1,130 @@ +@charset "UTF-8"; + +// Form + +#{$prefix}form { + color: $color; + + label { + float: left; + display: block; + width: 75px; + text-align: left; + padding: 7px 0 15px 5px; + margin: 0 0 2px; + font-weight: normal; + } + + br { + clear: both; + } + + iframe { + display: none; + } + + input:focus { + outline: 0; + } + + input[type="text"], input[type="number"] { + color: #999; + padding: 8px; + border: 1px solid $borderColor; + } + + input[type="number"] { + width: 40px; + display: inline-block; + padding: 6px 8px; + } + + input[type="text"] { + display: inline-block; + width: 264px; + } + + .fa-btns { + display: inline-block; + + a { + color: #999; + padding: 7px 10px 0 0; + display: inline-block; + text-decoration: none; + text-align: center; + } + + .fa { + font-size: 1.3em; + } + + label { + float: none; + display: inline-block; + width: auto; + text-align: left; + padding: 0 0 0 5px; + cursor: pointer; + } + } +} + +#{$prefix}form, +#{$prefix}dialog-container, +#{$prefix}dialog-footer { + + input[type="submit"], #{$prefix}btn, button { + color: $color; + min-width: 75px; + cursor: pointer; + background: #fff; + padding: 7px 10px; + border: 1px solid #ddd; + @include border-radius(3px); + @include transition(background 300ms ease-out); + + &:hover { + background: #eee; + } + } + + #{$prefix}btn { + padding: 5px 8px 4px\0; + } + + #{$prefix}btn + #{$prefix}btn { + margin-left: 8px; + } +} + +#{$prefix}file-input { + width: 75px; + height: 32px; + margin-left: 8px; + position: relative; + display: inline-block; + + input[type="file"] { + width: 75px; + height: 32px; + opacity: 0; + cursor: pointer; + background: #000; + display: inline-block; + position: absolute; + top: 0; + right: 0; + + &::-webkit-file-upload-button { + visibility: hidden; + } + } + + input[type="submit"] { + } + + &:hover input[type="submit"] { + background: #eee; + } +} \ No newline at end of file diff --git a/system/admin/editor.md/scss/editormd.grid.scss b/system/admin/editor.md/scss/editormd.grid.scss new file mode 100644 index 00000000..d7b6558a --- /dev/null +++ b/system/admin/editor.md/scss/editormd.grid.scss @@ -0,0 +1,36 @@ +@charset "utf-8"; + +.editormd-grid-table { + width: 99%; + display: table; + border: 1px solid #ddd; + border-collapse: collapse; +} + +.editormd-grid-table-row { + width: 100%; + display: table-row; + + a { + font-size: 1.4em; + width: 5%; + height: 36px; + color: #999; + text-align: center; + display: table-cell; + vertical-align: middle; + border: 1px solid #ddd; + text-decoration: none; + @include transition(background-color 300ms ease-out, color 100ms ease-in); + + &.selected { + color: #666; + background-color: #eee; + } + + &:hover { + color: #777; + background-color: #f6f6f6; + } + } +} \ No newline at end of file diff --git a/system/admin/editor.md/scss/editormd.logo.scss b/system/admin/editor.md/scss/editormd.logo.scss new file mode 100644 index 00000000..bd7e556b --- /dev/null +++ b/system/admin/editor.md/scss/editormd.logo.scss @@ -0,0 +1,89 @@ +@charset "UTF-8"; + +@import "lib/variables"; +@import "lib/prefixes"; + +@font-face { + font-family: 'editormd-logo'; + src:url('../fonts/editormd-logo.eot?-5y8q6h'); + src:url('.../fonts/editormd-logo.eot?#iefix-5y8q6h') format('embedded-opentype'), + url('../fonts/editormd-logo.woff?-5y8q6h') format('woff'), + url('../fonts/editormd-logo.ttf?-5y8q6h') format('truetype'), + url('../fonts/editormd-logo.svg?-5y8q6h#icomoon') format('svg'); + font-weight: normal; + font-style: normal; +} + +#{$prefix}logo, +#{$prefix}logo-1x, +#{$prefix}logo-2x, +#{$prefix}logo-3x, +#{$prefix}logo-4x, +#{$prefix}logo-5x, +#{$prefix}logo-6x, +#{$prefix}logo-7x, +#{$prefix}logo-8x { + font-family: 'editormd-logo'; + speak: none; + font-style: normal; + font-weight: normal; + font-variant: normal; + text-transform: none; + font-size: inherit; + line-height: 1; + display: inline-block; + text-rendering: auto; + vertical-align: inherit; + + //Better Font Rendering + -webkit-font-smoothing: antialiased; + -moz-osx-font-smoothing: grayscale; + + &:before { + content: "\e1987"; + /* + HTML Entity 󡦇 + example: + */ + } +} + +#{$prefix}logo-1x { + font-size: 1em; +} + +#{$prefix}logo-lg { + font-size: 1.2em; +} + +#{$prefix}logo-2x { + font-size: 2em; +} + +#{$prefix}logo-3x { + font-size: 3em; +} + +#{$prefix}logo-4x { + font-size: 4em; +} + +#{$prefix}logo-5x { + font-size: 5em; +} + +#{$prefix}logo-6x { + font-size: 6em; +} + +#{$prefix}logo-7x { + font-size: 7em; +} + +#{$prefix}logo-8x { + font-size: 8em; +} + +#{$prefix}logo-color { + color: $mainColor; +} \ No newline at end of file diff --git a/system/admin/editor.md/scss/editormd.menu.scss b/system/admin/editor.md/scss/editormd.menu.scss new file mode 100644 index 00000000..766903f1 --- /dev/null +++ b/system/admin/editor.md/scss/editormd.menu.scss @@ -0,0 +1,113 @@ +@charset "UTF-8"; + +#{$prefix}menu { + margin: 0; + padding: 0; + list-style: none; + + > li { + margin: 0; + padding: 5px 1px; + display: inline-block; + position: relative; + + &.divider { + display: inline-block; + text-indent: -9999px; + margin: 0 5px; + height: 65%; + border-right: 1px solid $borderColor; + } + + > a { + outline: 0; + color: $color; + display: inline-block; + min-width: 24px; + font-size: 16px; + text-decoration: none; + text-align: center; + @include border-radius(2px); + border: 1px solid #fff; + @include transition(all 300ms ease-out); + + &:hover, &.active { + border: 1px solid $borderColor; + background: #eee; + } + + > .fa { + text-align: center; + display: block; + padding: 5px; + } + + > #{$prefix}bold { + padding: 5px 2px; + display: inline-block; + font-weight: bold; + } + } + + &:hover #{$prefix}dropdown-menu { + display: block; + } + } + + > li + li > a { + margin-left: 3px; + } +} + +#{$prefix}dropdown-menu { + display: none; + background: #fff; + border: 1px solid $borderColor; + width: 148px; + list-style: none; + position: absolute; + top: 33px; + left: 0; + z-index: 100; + @include box-shadow(1px 2px 6px rgba(0, 0, 0, 0.15)); + + &:before, &:after { + width: 0; + height: 0; + display: block; + content: ""; + position: absolute; + top: -11px; + left: 8px; + border: 5px solid transparent; + } + + &:before { + border-bottom-color: #ccc; + } + + &:after { + border-bottom-color: #ffffff; + top: -10px; + } + + + > li { + + > a { + color: $color; + display: block; + text-decoration: none; + padding: 8px 10px; + + &:hover { + background: #f6f6f6; + @include transition(all 300ms ease-out); + } + } + } + + > li + li { + border-top: 1px solid $borderColor; + } +} \ No newline at end of file diff --git a/system/admin/editor.md/scss/editormd.preview.scss b/system/admin/editor.md/scss/editormd.preview.scss new file mode 100644 index 00000000..5f9ce90a --- /dev/null +++ b/system/admin/editor.md/scss/editormd.preview.scss @@ -0,0 +1,322 @@ +@charset "UTF-8"; + +@import "lib/variables"; +@import "lib/prefixes"; +@import "font-awesome"; + +@import "editormd.logo"; + +// github-markdown.css +@import "github-markdown"; + +#{$prefix}preview-container, #{$prefix}html-preview { + text-align: left; + font-size: 14px; + line-height: 1.6; + padding: 20px; + overflow: auto; + width: 100%; + background-color: #fff; + + blockquote { + color: $color; + border-left: 4px solid $borderColor; + padding-left: 20px; + margin-left: 0; + font-size: 14px; + font-style: italic; + } + + p code { + margin-left: 5px; + margin-right: 4px; + } + + abbr { + background: #ffffdd; + } + + hr { + height: 1px; + border: none; + border-top: 1px solid $borderColor; + background: none; + } + + code { + border: 1px solid $borderColor; + background: #f6f6f6; + padding: 3px; + border-radius: 3px; + font-size: 14px; + } + + pre { + border: 1px solid $borderColor; + background: #f6f6f6; + padding: 10px; + @include border-radius(3px); + + code { + padding: 0; + } + } + + pre, code, kbd { + font-family: "YaHei Consolas Hybrid", Consolas, "Meiryo UI", "Malgun Gothic", "Segoe UI", "Trebuchet MS", Helvetica, monospace, monospace; + } + + table thead tr { + background-color: #F8F8F8; + } + + .markdown-toc { + } + + .markdown-toc-list { + } + + p#{$prefix}tex { + text-align: center; + } + + span#{$prefix}tex { + margin: 0 5px; + } + + .emoji { + width: 24px; + height: 24px; + } + + .katex { + font-size: 1.4em; + } + + .sequence-diagram, .flowchart { + margin: 0 auto; + text-align: center; + + svg { + margin: 0 auto; + } + + text { + font-size : 15px !important; + font-family: "YaHei Consolas Hybrid", Consolas, "Microsoft YaHei", "Malgun Gothic", "Segoe UI", Helvetica, Arial !important; + } + } +} + +//Pretty printing styles. Used with prettify.js. + +@import "prettify"; + +#{$prefix}preview-container, #{$prefix}html-preview { + pre.prettyprint { + padding: 10px; + border: 1px solid $borderColor; + white-space: pre-wrap; + word-wrap: break-word; + } + + ol.linenums { + color: #999; + padding-left: 2.5em; + + li { + list-style-type: decimal; + + code { + border: none; + background:none; + padding: 0; + } + } + } +} + +#{$prefix}preview-container, #{$prefix}html-preview { + + #{$prefix}toc-menu { + margin: 8px 0 12px 0; + display: inline-block; + + > .markdown-toc { + position: relative; + @include border-radius(4px); + border: 1px solid #ddd; + display: inline-block; + font-size: 1em; + + > ul { + width : 160%; + min-width: 180px; + position: absolute; + left: -1px; + top: -2px; + z-index: 100; + padding: 0 10px 10px; + display: none; + background: #fff; + border: 1px solid #ddd; + @include border-radius(4px); + @include box-shadow(0 3px 5px rgba(0, 0, 0, 0.2)); + + > li ul { + width: 100%; + min-width: 180px; + border: 1px solid #ddd; + display: none; + background: #fff; + @include border-radius(4px); + } + + > li a { + color: #666; + padding: 6px 10px; + display: block; + @include transition(background-color 500ms ease-out); + + &:hover { + background-color: #f6f6f6; + } + } + } + + li { + position: relative; + + > ul { + position: absolute; + top: 32px; + left: 10%; + display: none; + @include box-shadow(0 3px 5px rgba(0, 0, 0, 0.2)); + + &:before, &:after { + pointer-events: pointer-events; + position: absolute; + left: 15px; + top: -6px; + display: block; + content: ""; + width: 0; + height: 0; + border: 6px solid transparent; + border-width: 0 6px 6px; + z-index: 10; + } + + &:before { + border-bottom-color: #ccc; + } + + &:after { + border-bottom-color: #ffffff; + top: -5px; + } + } + } + } + + ul { + list-style: none; + } + + a { + text-decoration: none; + } + + h1 { + font-size: 16px; + padding: 5px 0 10px 10px; + line-height: 1; + border-bottom: 1px solid #eee; + + .fa { + padding-left: 10px; + } + } + + .toc-menu-btn { + color: #666; + min-width: 180px; + padding: 5px 10px; + border-radius: 4px; + display: inline-block; + @include transition(background-color 500ms ease-out); + + &:hover { + background-color: #f6f6f6; + } + + .fa { + float: right; + padding: 3px 0 0 10px; + font-size: 1.3em; + } + } + } +} + +.markdown-body { + #{$prefix}toc-menu { + ul { + padding-left: 0; + } + } + + .highlight pre, pre { + line-height: 1.6; + } +} + +hr.editormd-page-break { + border: 1px dotted #ccc; + font-size: 0; + height: 2px; +} + +@media only print { + hr.editormd-page-break { + background: none; + border: none; + height: 0; + } +} + +#{$prefix}html-preview { + textarea { + display : none; + } + + hr.editormd-page-break { + background: none; + border: none; + height: 0; + } +} + +#{$prefix}preview-close-btn { + color: #fff; + padding: 4px 6px; + font-size: 18px; + @include border-radius(500px); + display: none; + background-color: #ccc; + position: absolute; + top: 25px; + right: 35px; + z-index: 19; + @include transition(background-color 300ms ease-out); + + &:hover { + background-color: #999; + } +} + +.editormd-preview-active { + width: 100%; + padding: 40px; +} \ No newline at end of file diff --git a/system/admin/editor.md/scss/editormd.preview.themes.scss b/system/admin/editor.md/scss/editormd.preview.themes.scss new file mode 100644 index 00000000..6553c1e1 --- /dev/null +++ b/system/admin/editor.md/scss/editormd.preview.themes.scss @@ -0,0 +1,131 @@ +/* Preview dark theme */ + +#{$prefix}preview-theme-dark { + color: #777; + background:#2C2827; + + #{$prefix}preview-container { + color: #888; + background-color: #2C2827; + //font-family: "Meiryo UI", "Helvetica Neue", "Microsoft YaHei"; + + pre.prettyprint { + border: none; + } + + blockquote { + color: #555; + padding: 0.5em; + background: #222; + border-color: #333; + } + + abbr { + color: #fff; + padding: 1px 3px; + @include border-radius(3px); + background:#ff9900; + } + + code { + color: #fff; + border: none; + padding: 1px 3px; + @include border-radius(3px); + background: #5A9600; + } + + table { + border: none; + } + + .fa-emoji { + color: #B4BF42; + } + + .katex { + color: #FEC93F; + } + } + + .editormd-toc-menu { + > .markdown-toc { + background:#fff; + border:none; + + h1 { + border-color:#ddd; + } + } + } + + .markdown-body { + h1, h2, hr { + border-color: #222; + } + } + + pre { + color: #999; + background-color: #111; + background-color: rgba(0,0,0,.4); + + /* plain text */ + .pln { + color: #999; + } + } + + li.L1, li.L3, li.L5, li.L7, li.L9 { + background: none; + } + + [class*=editormd-logo] { + color: #2196F3; + } + + .sequence-diagram { + text { + fill: #fff; + } + + rect, path { + color:#fff; + fill : #64D1CB; + stroke : #64D1CB; + } + } + + .flowchart { + rect, path { + stroke : #A6C6FF; + } + + rect { + fill: #A6C6FF; + } + + text { + fill: #5879B4; + } + } +} + +@media screen { + + #{$prefix}preview-theme-dark { + .str { color: #080 } /* string content */ + .kwd { color: #ff9900; } /* a keyword */ + .com { color: #444444; } /* a comment */ + .typ { color: #606 } /* a type name */ + .lit { color: #066 } /* a literal value */ + /* punctuation, lisp open bracket, lisp close bracket */ + .pun, .opn, .clo { color: #660 } + .tag { color: #ff9900; } /* a markup tag name */ + .atn { color: #6C95F5; } /* a markup attribute name */ + .atv { color: #080 } /* a markup attribute value */ + .dec, .var { color: #008BA7; } /* a declaration; a variable name */ + .fun { color: red } /* a function name */ + } + +} \ No newline at end of file diff --git a/system/admin/editor.md/scss/editormd.scss b/system/admin/editor.md/scss/editormd.scss new file mode 100644 index 00000000..30abf382 --- /dev/null +++ b/system/admin/editor.md/scss/editormd.scss @@ -0,0 +1,137 @@ +@charset "UTF-8"; + +@import "lib/variables"; +@import "lib/prefixes"; + +.editormd { + width: 90%; + height: 640px; + margin: 0 auto; + text-align: left; + overflow: hidden; + position: relative; + margin-bottom: 15px; + border: 1px solid $borderColor; + font-family: "Meiryo UI", "Microsoft YaHei", "Malgun Gothic", "Segoe UI", "Trebuchet MS", Helvetica, "Monaco", monospace, Tahoma, STXihei, "华文细黑", STHeiti, "Helvetica Neue", "Droid Sans", "wenquanyi micro hei", FreeSans, Arimo, Arial, SimSun, "宋体", Heiti, "黑体", sans-serif; + + *, *:before, *:after { + @include box-sizing(border-box); + } + + a { + text-decoration: none; + } + + img { + border: none; + vertical-align: middle; + } + + > textarea, + #{$prefix}html-textarea, + #{$prefix}markdown-textarea { + width: 0; + height: 0; + outline: 0; + resize:none; + } + + #{$prefix}html-textarea, + #{$prefix}markdown-textarea { + display : none; + } + + input[type="text"], + input[type="button"], + input[type="submit"], + select, textarea, button { + @include appearance(none); + } + + ::-webkit-scrollbar { + height: 10px; + width: 7px; + background: rgba(0, 0, 0, .1); + + &:hover { + background: rgba(0, 0, 0, .2); + } + } + + ::-webkit-scrollbar-thumb { + background: rgba(0,0,0,0.3); + @include border-radius(6px); + + &:hover { + @include box-shadow(inset 1px 1px 1px rgba(0, 0, 0, .25)); + background-color: rgba(0, 0, 0, .4); + } + } +} + +#{$prefix}user-unselect { + @include user-select(none); +} + +#{$prefix}toolbar { + width: 100%; + min-height: 37px; + background: #fff; + display: none; + position: absolute; + top: 0; + left: 0; + z-index: 10; + border-bottom: 1px solid $borderColor; +} + +#{$prefix}toolbar-container { + padding: 0 8px; + min-height: 35px; + @include user-select(none); +} + +@import "editormd.menu"; + +#{$prefix}container { + margin: 0; + width: 100%; + height: 100%; + overflow: hidden; + padding: 35px 0 0; + position: relative; + background: #fff; + @include box-sizing(border-box); +} + +@import "editormd.dialog"; +@import "editormd.grid"; +@import "editormd.tab"; +@import "editormd.form"; +@import "editormd.codemirror"; +@import "editormd.preview"; +@import "editormd.preview.themes"; + +#{$prefix}onlyread { + #{$prefix}toolbar { + display: none; + } + + .CodeMirror { + margin-top: 0; + } + + #{$prefix}preview { + top: 0; + } +} + +#{$prefix}fullscreen { + position: fixed; + top : 0; + left : 0; + border: none; + margin: 0 auto; +} + +@import "editormd.themes"; \ No newline at end of file diff --git a/system/admin/editor.md/scss/editormd.tab.scss b/system/admin/editor.md/scss/editormd.tab.scss new file mode 100644 index 00000000..483b4208 --- /dev/null +++ b/system/admin/editor.md/scss/editormd.tab.scss @@ -0,0 +1,49 @@ +@charset "utf-8"; + +.editormd-tab { +} + +.editormd-tab-head { + list-style: none; + border-bottom: 1px solid #ddd; + + li { + display: inline-block; + + a { + color: #999; + display: block; + padding: 6px 12px 5px; + text-align: center; + text-decoration: none; + margin-bottom: -1px; + border: 1px solid #ddd; + @include border-top-left-radius(3px); + @include border-top-right-radius(3px); + background: #f6f6f6; + @include transition(all 300ms ease-out); + + &:hover { + color: #666; + background: #eee; + } + } + + &.active a { + color: #666; + background: #fff; + border-bottom-color: #fff; + } + } + + li + li { + margin-left: 3px; + } +} + +.editormd-tab-container { +} + +.editormd-tab-box { + padding: 20px 0; +} \ No newline at end of file diff --git a/system/admin/editor.md/scss/editormd.themes.scss b/system/admin/editor.md/scss/editormd.themes.scss new file mode 100644 index 00000000..257766a7 --- /dev/null +++ b/system/admin/editor.md/scss/editormd.themes.scss @@ -0,0 +1,28 @@ +/* Editor.md Dark theme */ + +#{$prefix}theme-dark { + border-color: #1a1a17; + + #{$prefix}toolbar { + background: #1A1A17; + border-color: #1a1a17; + } + + #{$prefix}menu > li > a { + color: #777; + border-color: #1a1a17; + + &:hover, &.active { + border-color: #333; + background: #333; + } + } + + #{$prefix}menu > li.divider { + border-right: 1px solid #111; + } + + .CodeMirror { + border-right: 1px solid rgba(0,0,0,0.1); + } +} \ No newline at end of file diff --git a/system/admin/editor.md/scss/font-awesome.scss b/system/admin/editor.md/scss/font-awesome.scss new file mode 100644 index 00000000..2dcdc220 --- /dev/null +++ b/system/admin/editor.md/scss/font-awesome.scss @@ -0,0 +1,1801 @@ +/*! + * Font Awesome 4.3.0 by @davegandy - http://fontawesome.io - @fontawesome + * License - http://fontawesome.io/license (Font: SIL OFL 1.1, CSS: MIT License) + */ +/* FONT PATH + * -------------------------- */ +@font-face { + font-family: 'FontAwesome'; + src: url('../fonts/fontawesome-webfont.eot?v=4.3.0'); + src: url('../fonts/fontawesome-webfont.eot?#iefix&v=4.3.0') format('embedded-opentype'), url('../fonts/fontawesome-webfont.woff2?v=4.3.0') format('woff2'), url('../fonts/fontawesome-webfont.woff?v=4.3.0') format('woff'), url('../fonts/fontawesome-webfont.ttf?v=4.3.0') format('truetype'), url('../fonts/fontawesome-webfont.svg?v=4.3.0#fontawesomeregular') format('svg'); + font-weight: normal; + font-style: normal; +} +.fa { + display: inline-block; + font: normal normal normal 14px/1 FontAwesome; + font-size: inherit; + text-rendering: auto; + -webkit-font-smoothing: antialiased; + -moz-osx-font-smoothing: grayscale; + transform: translate(0, 0); +} +/* makes the font 33% larger relative to the icon container */ +.fa-lg { + font-size: 1.33333333em; + line-height: 0.75em; + vertical-align: -15%; +} +.fa-2x { + font-size: 2em; +} +.fa-3x { + font-size: 3em; +} +.fa-4x { + font-size: 4em; +} +.fa-5x { + font-size: 5em; +} +.fa-fw { + width: 1.28571429em; + text-align: center; +} +.fa-ul { + padding-left: 0; + margin-left: 2.14285714em; + list-style-type: none; +} +.fa-ul > li { + position: relative; +} +.fa-li { + position: absolute; + left: -2.14285714em; + width: 2.14285714em; + top: 0.14285714em; + text-align: center; +} +.fa-li.fa-lg { + left: -1.85714286em; +} +.fa-border { + padding: .2em .25em .15em; + border: solid 0.08em #eeeeee; + border-radius: .1em; +} +.pull-right { + float: right; +} +.pull-left { + float: left; +} +.fa.pull-left { + margin-right: .3em; +} +.fa.pull-right { + margin-left: .3em; +} +.fa-spin { + -webkit-animation: fa-spin 2s infinite linear; + animation: fa-spin 2s infinite linear; +} +.fa-pulse { + -webkit-animation: fa-spin 1s infinite steps(8); + animation: fa-spin 1s infinite steps(8); +} +@-webkit-keyframes fa-spin { + 0% { + -webkit-transform: rotate(0deg); + transform: rotate(0deg); + } + 100% { + -webkit-transform: rotate(359deg); + transform: rotate(359deg); + } +} +@keyframes fa-spin { + 0% { + -webkit-transform: rotate(0deg); + transform: rotate(0deg); + } + 100% { + -webkit-transform: rotate(359deg); + transform: rotate(359deg); + } +} +.fa-rotate-90 { + filter: progid:DXImageTransform.Microsoft.BasicImage(rotation=1); + -webkit-transform: rotate(90deg); + -ms-transform: rotate(90deg); + transform: rotate(90deg); +} +.fa-rotate-180 { + filter: progid:DXImageTransform.Microsoft.BasicImage(rotation=2); + -webkit-transform: rotate(180deg); + -ms-transform: rotate(180deg); + transform: rotate(180deg); +} +.fa-rotate-270 { + filter: progid:DXImageTransform.Microsoft.BasicImage(rotation=3); + -webkit-transform: rotate(270deg); + -ms-transform: rotate(270deg); + transform: rotate(270deg); +} +.fa-flip-horizontal { + filter: progid:DXImageTransform.Microsoft.BasicImage(rotation=0, mirror=1); + -webkit-transform: scale(-1, 1); + -ms-transform: scale(-1, 1); + transform: scale(-1, 1); +} +.fa-flip-vertical { + filter: progid:DXImageTransform.Microsoft.BasicImage(rotation=2, mirror=1); + -webkit-transform: scale(1, -1); + -ms-transform: scale(1, -1); + transform: scale(1, -1); +} +:root .fa-rotate-90, +:root .fa-rotate-180, +:root .fa-rotate-270, +:root .fa-flip-horizontal, +:root .fa-flip-vertical { + filter: none; +} +.fa-stack { + position: relative; + display: inline-block; + width: 2em; + height: 2em; + line-height: 2em; + vertical-align: middle; +} +.fa-stack-1x, +.fa-stack-2x { + position: absolute; + left: 0; + width: 100%; + text-align: center; +} +.fa-stack-1x { + line-height: inherit; +} +.fa-stack-2x { + font-size: 2em; +} +.fa-inverse { + color: #ffffff; +} +/* Font Awesome uses the Unicode Private Use Area (PUA) to ensure screen + readers do not read off random characters that represent icons */ +.fa-glass:before { + content: "\f000"; +} +.fa-music:before { + content: "\f001"; +} +.fa-search:before { + content: "\f002"; +} +.fa-envelope-o:before { + content: "\f003"; +} +.fa-heart:before { + content: "\f004"; +} +.fa-star:before { + content: "\f005"; +} +.fa-star-o:before { + content: "\f006"; +} +.fa-user:before { + content: "\f007"; +} +.fa-film:before { + content: "\f008"; +} +.fa-th-large:before { + content: "\f009"; +} +.fa-th:before { + content: "\f00a"; +} +.fa-th-list:before { + content: "\f00b"; +} +.fa-check:before { + content: "\f00c"; +} +.fa-remove:before, +.fa-close:before, +.fa-times:before { + content: "\f00d"; +} +.fa-search-plus:before { + content: "\f00e"; +} +.fa-search-minus:before { + content: "\f010"; +} +.fa-power-off:before { + content: "\f011"; +} +.fa-signal:before { + content: "\f012"; +} +.fa-gear:before, +.fa-cog:before { + content: "\f013"; +} +.fa-trash-o:before { + content: "\f014"; +} +.fa-home:before { + content: "\f015"; +} +.fa-file-o:before { + content: "\f016"; +} +.fa-clock-o:before { + content: "\f017"; +} +.fa-road:before { + content: "\f018"; +} +.fa-download:before { + content: "\f019"; +} +.fa-arrow-circle-o-down:before { + content: "\f01a"; +} +.fa-arrow-circle-o-up:before { + content: "\f01b"; +} +.fa-inbox:before { + content: "\f01c"; +} +.fa-play-circle-o:before { + content: "\f01d"; +} +.fa-rotate-right:before, +.fa-repeat:before { + content: "\f01e"; +} +.fa-refresh:before { + content: "\f021"; +} +.fa-list-alt:before { + content: "\f022"; +} +.fa-lock:before { + content: "\f023"; +} +.fa-flag:before { + content: "\f024"; +} +.fa-headphones:before { + content: "\f025"; +} +.fa-volume-off:before { + content: "\f026"; +} +.fa-volume-down:before { + content: "\f027"; +} +.fa-volume-up:before { + content: "\f028"; +} +.fa-qrcode:before { + content: "\f029"; +} +.fa-barcode:before { + content: "\f02a"; +} +.fa-tag:before { + content: "\f02b"; +} +.fa-tags:before { + content: "\f02c"; +} +.fa-book:before { + content: "\f02d"; +} +.fa-bookmark:before { + content: "\f02e"; +} +.fa-print:before { + content: "\f02f"; +} +.fa-camera:before { + content: "\f030"; +} +.fa-font:before { + content: "\f031"; +} +.fa-bold:before { + content: "\f032"; +} +.fa-italic:before { + content: "\f033"; +} +.fa-text-height:before { + content: "\f034"; +} +.fa-text-width:before { + content: "\f035"; +} +.fa-align-left:before { + content: "\f036"; +} +.fa-align-center:before { + content: "\f037"; +} +.fa-align-right:before { + content: "\f038"; +} +.fa-align-justify:before { + content: "\f039"; +} +.fa-list:before { + content: "\f03a"; +} +.fa-dedent:before, +.fa-outdent:before { + content: "\f03b"; +} +.fa-indent:before { + content: "\f03c"; +} +.fa-video-camera:before { + content: "\f03d"; +} +.fa-photo:before, +.fa-image:before, +.fa-picture-o:before { + content: "\f03e"; +} +.fa-pencil:before { + content: "\f040"; +} +.fa-map-marker:before { + content: "\f041"; +} +.fa-adjust:before { + content: "\f042"; +} +.fa-tint:before { + content: "\f043"; +} +.fa-edit:before, +.fa-pencil-square-o:before { + content: "\f044"; +} +.fa-share-square-o:before { + content: "\f045"; +} +.fa-check-square-o:before { + content: "\f046"; +} +.fa-arrows:before { + content: "\f047"; +} +.fa-step-backward:before { + content: "\f048"; +} +.fa-fast-backward:before { + content: "\f049"; +} +.fa-backward:before { + content: "\f04a"; +} +.fa-play:before { + content: "\f04b"; +} +.fa-pause:before { + content: "\f04c"; +} +.fa-stop:before { + content: "\f04d"; +} +.fa-forward:before { + content: "\f04e"; +} +.fa-fast-forward:before { + content: "\f050"; +} +.fa-step-forward:before { + content: "\f051"; +} +.fa-eject:before { + content: "\f052"; +} +.fa-chevron-left:before { + content: "\f053"; +} +.fa-chevron-right:before { + content: "\f054"; +} +.fa-plus-circle:before { + content: "\f055"; +} +.fa-minus-circle:before { + content: "\f056"; +} +.fa-times-circle:before { + content: "\f057"; +} +.fa-check-circle:before { + content: "\f058"; +} +.fa-question-circle:before { + content: "\f059"; +} +.fa-info-circle:before { + content: "\f05a"; +} +.fa-crosshairs:before { + content: "\f05b"; +} +.fa-times-circle-o:before { + content: "\f05c"; +} +.fa-check-circle-o:before { + content: "\f05d"; +} +.fa-ban:before { + content: "\f05e"; +} +.fa-arrow-left:before { + content: "\f060"; +} +.fa-arrow-right:before { + content: "\f061"; +} +.fa-arrow-up:before { + content: "\f062"; +} +.fa-arrow-down:before { + content: "\f063"; +} +.fa-mail-forward:before, +.fa-share:before { + content: "\f064"; +} +.fa-expand:before { + content: "\f065"; +} +.fa-compress:before { + content: "\f066"; +} +.fa-plus:before { + content: "\f067"; +} +.fa-minus:before { + content: "\f068"; +} +.fa-asterisk:before { + content: "\f069"; +} +.fa-exclamation-circle:before { + content: "\f06a"; +} +.fa-gift:before { + content: "\f06b"; +} +.fa-leaf:before { + content: "\f06c"; +} +.fa-fire:before { + content: "\f06d"; +} +.fa-eye:before { + content: "\f06e"; +} +.fa-eye-slash:before { + content: "\f070"; +} +.fa-warning:before, +.fa-exclamation-triangle:before { + content: "\f071"; +} +.fa-plane:before { + content: "\f072"; +} +.fa-calendar:before { + content: "\f073"; +} +.fa-random:before { + content: "\f074"; +} +.fa-comment:before { + content: "\f075"; +} +.fa-magnet:before { + content: "\f076"; +} +.fa-chevron-up:before { + content: "\f077"; +} +.fa-chevron-down:before { + content: "\f078"; +} +.fa-retweet:before { + content: "\f079"; +} +.fa-shopping-cart:before { + content: "\f07a"; +} +.fa-folder:before { + content: "\f07b"; +} +.fa-folder-open:before { + content: "\f07c"; +} +.fa-arrows-v:before { + content: "\f07d"; +} +.fa-arrows-h:before { + content: "\f07e"; +} +.fa-bar-chart-o:before, +.fa-bar-chart:before { + content: "\f080"; +} +.fa-twitter-square:before { + content: "\f081"; +} +.fa-facebook-square:before { + content: "\f082"; +} +.fa-camera-retro:before { + content: "\f083"; +} +.fa-key:before { + content: "\f084"; +} +.fa-gears:before, +.fa-cogs:before { + content: "\f085"; +} +.fa-comments:before { + content: "\f086"; +} +.fa-thumbs-o-up:before { + content: "\f087"; +} +.fa-thumbs-o-down:before { + content: "\f088"; +} +.fa-star-half:before { + content: "\f089"; +} +.fa-heart-o:before { + content: "\f08a"; +} +.fa-sign-out:before { + content: "\f08b"; +} +.fa-linkedin-square:before { + content: "\f08c"; +} +.fa-thumb-tack:before { + content: "\f08d"; +} +.fa-external-link:before { + content: "\f08e"; +} +.fa-sign-in:before { + content: "\f090"; +} +.fa-trophy:before { + content: "\f091"; +} +.fa-github-square:before { + content: "\f092"; +} +.fa-upload:before { + content: "\f093"; +} +.fa-lemon-o:before { + content: "\f094"; +} +.fa-phone:before { + content: "\f095"; +} +.fa-square-o:before { + content: "\f096"; +} +.fa-bookmark-o:before { + content: "\f097"; +} +.fa-phone-square:before { + content: "\f098"; +} +.fa-twitter:before { + content: "\f099"; +} +.fa-facebook-f:before, +.fa-facebook:before { + content: "\f09a"; +} +.fa-github:before { + content: "\f09b"; +} +.fa-unlock:before { + content: "\f09c"; +} +.fa-credit-card:before { + content: "\f09d"; +} +.fa-rss:before { + content: "\f09e"; +} +.fa-hdd-o:before { + content: "\f0a0"; +} +.fa-bullhorn:before { + content: "\f0a1"; +} +.fa-bell:before { + content: "\f0f3"; +} +.fa-certificate:before { + content: "\f0a3"; +} +.fa-hand-o-right:before { + content: "\f0a4"; +} +.fa-hand-o-left:before { + content: "\f0a5"; +} +.fa-hand-o-up:before { + content: "\f0a6"; +} +.fa-hand-o-down:before { + content: "\f0a7"; +} +.fa-arrow-circle-left:before { + content: "\f0a8"; +} +.fa-arrow-circle-right:before { + content: "\f0a9"; +} +.fa-arrow-circle-up:before { + content: "\f0aa"; +} +.fa-arrow-circle-down:before { + content: "\f0ab"; +} +.fa-globe:before { + content: "\f0ac"; +} +.fa-wrench:before { + content: "\f0ad"; +} +.fa-tasks:before { + content: "\f0ae"; +} +.fa-filter:before { + content: "\f0b0"; +} +.fa-briefcase:before { + content: "\f0b1"; +} +.fa-arrows-alt:before { + content: "\f0b2"; +} +.fa-group:before, +.fa-users:before { + content: "\f0c0"; +} +.fa-chain:before, +.fa-link:before { + content: "\f0c1"; +} +.fa-cloud:before { + content: "\f0c2"; +} +.fa-flask:before { + content: "\f0c3"; +} +.fa-cut:before, +.fa-scissors:before { + content: "\f0c4"; +} +.fa-copy:before, +.fa-files-o:before { + content: "\f0c5"; +} +.fa-paperclip:before { + content: "\f0c6"; +} +.fa-save:before, +.fa-floppy-o:before { + content: "\f0c7"; +} +.fa-square:before { + content: "\f0c8"; +} +.fa-navicon:before, +.fa-reorder:before, +.fa-bars:before { + content: "\f0c9"; +} +.fa-list-ul:before { + content: "\f0ca"; +} +.fa-list-ol:before { + content: "\f0cb"; +} +.fa-strikethrough:before { + content: "\f0cc"; +} +.fa-underline:before { + content: "\f0cd"; +} +.fa-table:before { + content: "\f0ce"; +} +.fa-magic:before { + content: "\f0d0"; +} +.fa-truck:before { + content: "\f0d1"; +} +.fa-pinterest:before { + content: "\f0d2"; +} +.fa-pinterest-square:before { + content: "\f0d3"; +} +.fa-google-plus-square:before { + content: "\f0d4"; +} +.fa-google-plus:before { + content: "\f0d5"; +} +.fa-money:before { + content: "\f0d6"; +} +.fa-caret-down:before { + content: "\f0d7"; +} +.fa-caret-up:before { + content: "\f0d8"; +} +.fa-caret-left:before { + content: "\f0d9"; +} +.fa-caret-right:before { + content: "\f0da"; +} +.fa-columns:before { + content: "\f0db"; +} +.fa-unsorted:before, +.fa-sort:before { + content: "\f0dc"; +} +.fa-sort-down:before, +.fa-sort-desc:before { + content: "\f0dd"; +} +.fa-sort-up:before, +.fa-sort-asc:before { + content: "\f0de"; +} +.fa-envelope:before { + content: "\f0e0"; +} +.fa-linkedin:before { + content: "\f0e1"; +} +.fa-rotate-left:before, +.fa-undo:before { + content: "\f0e2"; +} +.fa-legal:before, +.fa-gavel:before { + content: "\f0e3"; +} +.fa-dashboard:before, +.fa-tachometer:before { + content: "\f0e4"; +} +.fa-comment-o:before { + content: "\f0e5"; +} +.fa-comments-o:before { + content: "\f0e6"; +} +.fa-flash:before, +.fa-bolt:before { + content: "\f0e7"; +} +.fa-sitemap:before { + content: "\f0e8"; +} +.fa-umbrella:before { + content: "\f0e9"; +} +.fa-paste:before, +.fa-clipboard:before { + content: "\f0ea"; +} +.fa-lightbulb-o:before { + content: "\f0eb"; +} +.fa-exchange:before { + content: "\f0ec"; +} +.fa-cloud-download:before { + content: "\f0ed"; +} +.fa-cloud-upload:before { + content: "\f0ee"; +} +.fa-user-md:before { + content: "\f0f0"; +} +.fa-stethoscope:before { + content: "\f0f1"; +} +.fa-suitcase:before { + content: "\f0f2"; +} +.fa-bell-o:before { + content: "\f0a2"; +} +.fa-coffee:before { + content: "\f0f4"; +} +.fa-cutlery:before { + content: "\f0f5"; +} +.fa-file-text-o:before { + content: "\f0f6"; +} +.fa-building-o:before { + content: "\f0f7"; +} +.fa-hospital-o:before { + content: "\f0f8"; +} +.fa-ambulance:before { + content: "\f0f9"; +} +.fa-medkit:before { + content: "\f0fa"; +} +.fa-fighter-jet:before { + content: "\f0fb"; +} +.fa-beer:before { + content: "\f0fc"; +} +.fa-h-square:before { + content: "\f0fd"; +} +.fa-plus-square:before { + content: "\f0fe"; +} +.fa-angle-double-left:before { + content: "\f100"; +} +.fa-angle-double-right:before { + content: "\f101"; +} +.fa-angle-double-up:before { + content: "\f102"; +} +.fa-angle-double-down:before { + content: "\f103"; +} +.fa-angle-left:before { + content: "\f104"; +} +.fa-angle-right:before { + content: "\f105"; +} +.fa-angle-up:before { + content: "\f106"; +} +.fa-angle-down:before { + content: "\f107"; +} +.fa-desktop:before { + content: "\f108"; +} +.fa-laptop:before { + content: "\f109"; +} +.fa-tablet:before { + content: "\f10a"; +} +.fa-mobile-phone:before, +.fa-mobile:before { + content: "\f10b"; +} +.fa-circle-o:before { + content: "\f10c"; +} +.fa-quote-left:before { + content: "\f10d"; +} +.fa-quote-right:before { + content: "\f10e"; +} +.fa-spinner:before { + content: "\f110"; +} +.fa-circle:before { + content: "\f111"; +} +.fa-mail-reply:before, +.fa-reply:before { + content: "\f112"; +} +.fa-github-alt:before { + content: "\f113"; +} +.fa-folder-o:before { + content: "\f114"; +} +.fa-folder-open-o:before { + content: "\f115"; +} +.fa-smile-o:before { + content: "\f118"; +} +.fa-frown-o:before { + content: "\f119"; +} +.fa-meh-o:before { + content: "\f11a"; +} +.fa-gamepad:before { + content: "\f11b"; +} +.fa-keyboard-o:before { + content: "\f11c"; +} +.fa-flag-o:before { + content: "\f11d"; +} +.fa-flag-checkered:before { + content: "\f11e"; +} +.fa-terminal:before { + content: "\f120"; +} +.fa-code:before { + content: "\f121"; +} +.fa-mail-reply-all:before, +.fa-reply-all:before { + content: "\f122"; +} +.fa-star-half-empty:before, +.fa-star-half-full:before, +.fa-star-half-o:before { + content: "\f123"; +} +.fa-location-arrow:before { + content: "\f124"; +} +.fa-crop:before { + content: "\f125"; +} +.fa-code-fork:before { + content: "\f126"; +} +.fa-unlink:before, +.fa-chain-broken:before { + content: "\f127"; +} +.fa-question:before { + content: "\f128"; +} +.fa-info:before { + content: "\f129"; +} +.fa-exclamation:before { + content: "\f12a"; +} +.fa-superscript:before { + content: "\f12b"; +} +.fa-subscript:before { + content: "\f12c"; +} +.fa-eraser:before { + content: "\f12d"; +} +.fa-puzzle-piece:before { + content: "\f12e"; +} +.fa-microphone:before { + content: "\f130"; +} +.fa-microphone-slash:before { + content: "\f131"; +} +.fa-shield:before { + content: "\f132"; +} +.fa-calendar-o:before { + content: "\f133"; +} +.fa-fire-extinguisher:before { + content: "\f134"; +} +.fa-rocket:before { + content: "\f135"; +} +.fa-maxcdn:before { + content: "\f136"; +} +.fa-chevron-circle-left:before { + content: "\f137"; +} +.fa-chevron-circle-right:before { + content: "\f138"; +} +.fa-chevron-circle-up:before { + content: "\f139"; +} +.fa-chevron-circle-down:before { + content: "\f13a"; +} +.fa-html5:before { + content: "\f13b"; +} +.fa-css3:before { + content: "\f13c"; +} +.fa-anchor:before { + content: "\f13d"; +} +.fa-unlock-alt:before { + content: "\f13e"; +} +.fa-bullseye:before { + content: "\f140"; +} +.fa-ellipsis-h:before { + content: "\f141"; +} +.fa-ellipsis-v:before { + content: "\f142"; +} +.fa-rss-square:before { + content: "\f143"; +} +.fa-play-circle:before { + content: "\f144"; +} +.fa-ticket:before { + content: "\f145"; +} +.fa-minus-square:before { + content: "\f146"; +} +.fa-minus-square-o:before { + content: "\f147"; +} +.fa-level-up:before { + content: "\f148"; +} +.fa-level-down:before { + content: "\f149"; +} +.fa-check-square:before { + content: "\f14a"; +} +.fa-pencil-square:before { + content: "\f14b"; +} +.fa-external-link-square:before { + content: "\f14c"; +} +.fa-share-square:before { + content: "\f14d"; +} +.fa-compass:before { + content: "\f14e"; +} +.fa-toggle-down:before, +.fa-caret-square-o-down:before { + content: "\f150"; +} +.fa-toggle-up:before, +.fa-caret-square-o-up:before { + content: "\f151"; +} +.fa-toggle-right:before, +.fa-caret-square-o-right:before { + content: "\f152"; +} +.fa-euro:before, +.fa-eur:before { + content: "\f153"; +} +.fa-gbp:before { + content: "\f154"; +} +.fa-dollar:before, +.fa-usd:before { + content: "\f155"; +} +.fa-rupee:before, +.fa-inr:before { + content: "\f156"; +} +.fa-cny:before, +.fa-rmb:before, +.fa-yen:before, +.fa-jpy:before { + content: "\f157"; +} +.fa-ruble:before, +.fa-rouble:before, +.fa-rub:before { + content: "\f158"; +} +.fa-won:before, +.fa-krw:before { + content: "\f159"; +} +.fa-bitcoin:before, +.fa-btc:before { + content: "\f15a"; +} +.fa-file:before { + content: "\f15b"; +} +.fa-file-text:before { + content: "\f15c"; +} +.fa-sort-alpha-asc:before { + content: "\f15d"; +} +.fa-sort-alpha-desc:before { + content: "\f15e"; +} +.fa-sort-amount-asc:before { + content: "\f160"; +} +.fa-sort-amount-desc:before { + content: "\f161"; +} +.fa-sort-numeric-asc:before { + content: "\f162"; +} +.fa-sort-numeric-desc:before { + content: "\f163"; +} +.fa-thumbs-up:before { + content: "\f164"; +} +.fa-thumbs-down:before { + content: "\f165"; +} +.fa-youtube-square:before { + content: "\f166"; +} +.fa-youtube:before { + content: "\f167"; +} +.fa-xing:before { + content: "\f168"; +} +.fa-xing-square:before { + content: "\f169"; +} +.fa-youtube-play:before { + content: "\f16a"; +} +.fa-dropbox:before { + content: "\f16b"; +} +.fa-stack-overflow:before { + content: "\f16c"; +} +.fa-instagram:before { + content: "\f16d"; +} +.fa-flickr:before { + content: "\f16e"; +} +.fa-adn:before { + content: "\f170"; +} +.fa-bitbucket:before { + content: "\f171"; +} +.fa-bitbucket-square:before { + content: "\f172"; +} +.fa-tumblr:before { + content: "\f173"; +} +.fa-tumblr-square:before { + content: "\f174"; +} +.fa-long-arrow-down:before { + content: "\f175"; +} +.fa-long-arrow-up:before { + content: "\f176"; +} +.fa-long-arrow-left:before { + content: "\f177"; +} +.fa-long-arrow-right:before { + content: "\f178"; +} +.fa-apple:before { + content: "\f179"; +} +.fa-windows:before { + content: "\f17a"; +} +.fa-android:before { + content: "\f17b"; +} +.fa-linux:before { + content: "\f17c"; +} +.fa-dribbble:before { + content: "\f17d"; +} +.fa-skype:before { + content: "\f17e"; +} +.fa-foursquare:before { + content: "\f180"; +} +.fa-trello:before { + content: "\f181"; +} +.fa-female:before { + content: "\f182"; +} +.fa-male:before { + content: "\f183"; +} +.fa-gittip:before, +.fa-gratipay:before { + content: "\f184"; +} +.fa-sun-o:before { + content: "\f185"; +} +.fa-moon-o:before { + content: "\f186"; +} +.fa-archive:before { + content: "\f187"; +} +.fa-bug:before { + content: "\f188"; +} +.fa-vk:before { + content: "\f189"; +} +.fa-weibo:before { + content: "\f18a"; +} +.fa-renren:before { + content: "\f18b"; +} +.fa-pagelines:before { + content: "\f18c"; +} +.fa-stack-exchange:before { + content: "\f18d"; +} +.fa-arrow-circle-o-right:before { + content: "\f18e"; +} +.fa-arrow-circle-o-left:before { + content: "\f190"; +} +.fa-toggle-left:before, +.fa-caret-square-o-left:before { + content: "\f191"; +} +.fa-dot-circle-o:before { + content: "\f192"; +} +.fa-wheelchair:before { + content: "\f193"; +} +.fa-vimeo-square:before { + content: "\f194"; +} +.fa-turkish-lira:before, +.fa-try:before { + content: "\f195"; +} +.fa-plus-square-o:before { + content: "\f196"; +} +.fa-space-shuttle:before { + content: "\f197"; +} +.fa-slack:before { + content: "\f198"; +} +.fa-envelope-square:before { + content: "\f199"; +} +.fa-wordpress:before { + content: "\f19a"; +} +.fa-openid:before { + content: "\f19b"; +} +.fa-institution:before, +.fa-bank:before, +.fa-university:before { + content: "\f19c"; +} +.fa-mortar-board:before, +.fa-graduation-cap:before { + content: "\f19d"; +} +.fa-yahoo:before { + content: "\f19e"; +} +.fa-google:before { + content: "\f1a0"; +} +.fa-reddit:before { + content: "\f1a1"; +} +.fa-reddit-square:before { + content: "\f1a2"; +} +.fa-stumbleupon-circle:before { + content: "\f1a3"; +} +.fa-stumbleupon:before { + content: "\f1a4"; +} +.fa-delicious:before { + content: "\f1a5"; +} +.fa-digg:before { + content: "\f1a6"; +} +.fa-pied-piper:before { + content: "\f1a7"; +} +.fa-pied-piper-alt:before { + content: "\f1a8"; +} +.fa-drupal:before { + content: "\f1a9"; +} +.fa-joomla:before { + content: "\f1aa"; +} +.fa-language:before { + content: "\f1ab"; +} +.fa-fax:before { + content: "\f1ac"; +} +.fa-building:before { + content: "\f1ad"; +} +.fa-child:before { + content: "\f1ae"; +} +.fa-paw:before { + content: "\f1b0"; +} +.fa-spoon:before { + content: "\f1b1"; +} +.fa-cube:before { + content: "\f1b2"; +} +.fa-cubes:before { + content: "\f1b3"; +} +.fa-behance:before { + content: "\f1b4"; +} +.fa-behance-square:before { + content: "\f1b5"; +} +.fa-steam:before { + content: "\f1b6"; +} +.fa-steam-square:before { + content: "\f1b7"; +} +.fa-recycle:before { + content: "\f1b8"; +} +.fa-automobile:before, +.fa-car:before { + content: "\f1b9"; +} +.fa-cab:before, +.fa-taxi:before { + content: "\f1ba"; +} +.fa-tree:before { + content: "\f1bb"; +} +.fa-spotify:before { + content: "\f1bc"; +} +.fa-deviantart:before { + content: "\f1bd"; +} +.fa-soundcloud:before { + content: "\f1be"; +} +.fa-database:before { + content: "\f1c0"; +} +.fa-file-pdf-o:before { + content: "\f1c1"; +} +.fa-file-word-o:before { + content: "\f1c2"; +} +.fa-file-excel-o:before { + content: "\f1c3"; +} +.fa-file-powerpoint-o:before { + content: "\f1c4"; +} +.fa-file-photo-o:before, +.fa-file-picture-o:before, +.fa-file-image-o:before { + content: "\f1c5"; +} +.fa-file-zip-o:before, +.fa-file-archive-o:before { + content: "\f1c6"; +} +.fa-file-sound-o:before, +.fa-file-audio-o:before { + content: "\f1c7"; +} +.fa-file-movie-o:before, +.fa-file-video-o:before { + content: "\f1c8"; +} +.fa-file-code-o:before { + content: "\f1c9"; +} +.fa-vine:before { + content: "\f1ca"; +} +.fa-codepen:before { + content: "\f1cb"; +} +.fa-jsfiddle:before { + content: "\f1cc"; +} +.fa-life-bouy:before, +.fa-life-buoy:before, +.fa-life-saver:before, +.fa-support:before, +.fa-life-ring:before { + content: "\f1cd"; +} +.fa-circle-o-notch:before { + content: "\f1ce"; +} +.fa-ra:before, +.fa-rebel:before { + content: "\f1d0"; +} +.fa-ge:before, +.fa-empire:before { + content: "\f1d1"; +} +.fa-git-square:before { + content: "\f1d2"; +} +.fa-git:before { + content: "\f1d3"; +} +.fa-hacker-news:before { + content: "\f1d4"; +} +.fa-tencent-weibo:before { + content: "\f1d5"; +} +.fa-qq:before { + content: "\f1d6"; +} +.fa-wechat:before, +.fa-weixin:before { + content: "\f1d7"; +} +.fa-send:before, +.fa-paper-plane:before { + content: "\f1d8"; +} +.fa-send-o:before, +.fa-paper-plane-o:before { + content: "\f1d9"; +} +.fa-history:before { + content: "\f1da"; +} +.fa-genderless:before, +.fa-circle-thin:before { + content: "\f1db"; +} +.fa-header:before { + content: "\f1dc"; +} +.fa-paragraph:before { + content: "\f1dd"; +} +.fa-sliders:before { + content: "\f1de"; +} +.fa-share-alt:before { + content: "\f1e0"; +} +.fa-share-alt-square:before { + content: "\f1e1"; +} +.fa-bomb:before { + content: "\f1e2"; +} +.fa-soccer-ball-o:before, +.fa-futbol-o:before { + content: "\f1e3"; +} +.fa-tty:before { + content: "\f1e4"; +} +.fa-binoculars:before { + content: "\f1e5"; +} +.fa-plug:before { + content: "\f1e6"; +} +.fa-slideshare:before { + content: "\f1e7"; +} +.fa-twitch:before { + content: "\f1e8"; +} +.fa-yelp:before { + content: "\f1e9"; +} +.fa-newspaper-o:before { + content: "\f1ea"; +} +.fa-wifi:before { + content: "\f1eb"; +} +.fa-calculator:before { + content: "\f1ec"; +} +.fa-paypal:before { + content: "\f1ed"; +} +.fa-google-wallet:before { + content: "\f1ee"; +} +.fa-cc-visa:before { + content: "\f1f0"; +} +.fa-cc-mastercard:before { + content: "\f1f1"; +} +.fa-cc-discover:before { + content: "\f1f2"; +} +.fa-cc-amex:before { + content: "\f1f3"; +} +.fa-cc-paypal:before { + content: "\f1f4"; +} +.fa-cc-stripe:before { + content: "\f1f5"; +} +.fa-bell-slash:before { + content: "\f1f6"; +} +.fa-bell-slash-o:before { + content: "\f1f7"; +} +.fa-trash:before { + content: "\f1f8"; +} +.fa-copyright:before { + content: "\f1f9"; +} +.fa-at:before { + content: "\f1fa"; +} +.fa-eyedropper:before { + content: "\f1fb"; +} +.fa-paint-brush:before { + content: "\f1fc"; +} +.fa-birthday-cake:before { + content: "\f1fd"; +} +.fa-area-chart:before { + content: "\f1fe"; +} +.fa-pie-chart:before { + content: "\f200"; +} +.fa-line-chart:before { + content: "\f201"; +} +.fa-lastfm:before { + content: "\f202"; +} +.fa-lastfm-square:before { + content: "\f203"; +} +.fa-toggle-off:before { + content: "\f204"; +} +.fa-toggle-on:before { + content: "\f205"; +} +.fa-bicycle:before { + content: "\f206"; +} +.fa-bus:before { + content: "\f207"; +} +.fa-ioxhost:before { + content: "\f208"; +} +.fa-angellist:before { + content: "\f209"; +} +.fa-cc:before { + content: "\f20a"; +} +.fa-shekel:before, +.fa-sheqel:before, +.fa-ils:before { + content: "\f20b"; +} +.fa-meanpath:before { + content: "\f20c"; +} +.fa-buysellads:before { + content: "\f20d"; +} +.fa-connectdevelop:before { + content: "\f20e"; +} +.fa-dashcube:before { + content: "\f210"; +} +.fa-forumbee:before { + content: "\f211"; +} +.fa-leanpub:before { + content: "\f212"; +} +.fa-sellsy:before { + content: "\f213"; +} +.fa-shirtsinbulk:before { + content: "\f214"; +} +.fa-simplybuilt:before { + content: "\f215"; +} +.fa-skyatlas:before { + content: "\f216"; +} +.fa-cart-plus:before { + content: "\f217"; +} +.fa-cart-arrow-down:before { + content: "\f218"; +} +.fa-diamond:before { + content: "\f219"; +} +.fa-ship:before { + content: "\f21a"; +} +.fa-user-secret:before { + content: "\f21b"; +} +.fa-motorcycle:before { + content: "\f21c"; +} +.fa-street-view:before { + content: "\f21d"; +} +.fa-heartbeat:before { + content: "\f21e"; +} +.fa-venus:before { + content: "\f221"; +} +.fa-mars:before { + content: "\f222"; +} +.fa-mercury:before { + content: "\f223"; +} +.fa-transgender:before { + content: "\f224"; +} +.fa-transgender-alt:before { + content: "\f225"; +} +.fa-venus-double:before { + content: "\f226"; +} +.fa-mars-double:before { + content: "\f227"; +} +.fa-venus-mars:before { + content: "\f228"; +} +.fa-mars-stroke:before { + content: "\f229"; +} +.fa-mars-stroke-v:before { + content: "\f22a"; +} +.fa-mars-stroke-h:before { + content: "\f22b"; +} +.fa-neuter:before { + content: "\f22c"; +} +.fa-facebook-official:before { + content: "\f230"; +} +.fa-pinterest-p:before { + content: "\f231"; +} +.fa-whatsapp:before { + content: "\f232"; +} +.fa-server:before { + content: "\f233"; +} +.fa-user-plus:before { + content: "\f234"; +} +.fa-user-times:before { + content: "\f235"; +} +.fa-hotel:before, +.fa-bed:before { + content: "\f236"; +} +.fa-viacoin:before { + content: "\f237"; +} +.fa-train:before { + content: "\f238"; +} +.fa-subway:before { + content: "\f239"; +} +.fa-medium:before { + content: "\f23a"; +} diff --git a/system/admin/editor.md/scss/github-markdown.scss b/system/admin/editor.md/scss/github-markdown.scss new file mode 100644 index 00000000..0d42686d --- /dev/null +++ b/system/admin/editor.md/scss/github-markdown.scss @@ -0,0 +1,665 @@ +@charset "UTF-8"; + +/*! github-markdown-css | The MIT License (MIT) | Copyright (c) Sindre Sorhus (sindresorhus.com) | https://github.com/sindresorhus/github-markdown-css */ +@font-face { + font-family: octicons-anchor; + src: url(data:font/woff;charset=utf-8;base64,d09GRgABAAAAAAYcAA0AAAAACjQAAQAAAAAAAAAAAAAAAAAAAAAAAAAAAABGRlRNAAABMAAAABwAAAAca8vGTk9TLzIAAAFMAAAARAAAAFZG1VHVY21hcAAAAZAAAAA+AAABQgAP9AdjdnQgAAAB0AAAAAQAAAAEACICiGdhc3AAAAHUAAAACAAAAAj//wADZ2x5ZgAAAdwAAADRAAABEKyikaNoZWFkAAACsAAAAC0AAAA2AtXoA2hoZWEAAALgAAAAHAAAACQHngNFaG10eAAAAvwAAAAQAAAAEAwAACJsb2NhAAADDAAAAAoAAAAKALIAVG1heHAAAAMYAAAAHwAAACABEAB2bmFtZQAAAzgAAALBAAAFu3I9x/Nwb3N0AAAF/AAAAB0AAAAvaoFvbwAAAAEAAAAAzBdyYwAAAADP2IQvAAAAAM/bz7t4nGNgZGFgnMDAysDB1Ml0hoGBoR9CM75mMGLkYGBgYmBlZsAKAtJcUxgcPsR8iGF2+O/AEMPsznAYKMwIkgMA5REMOXicY2BgYGaAYBkGRgYQsAHyGMF8FgYFIM0ChED+h5j//yEk/3KoSgZGNgYYk4GRCUgwMaACRoZhDwCs7QgGAAAAIgKIAAAAAf//AAJ4nHWMMQrCQBBF/0zWrCCIKUQsTDCL2EXMohYGSSmorScInsRGL2DOYJe0Ntp7BK+gJ1BxF1stZvjz/v8DRghQzEc4kIgKwiAppcA9LtzKLSkdNhKFY3HF4lK69ExKslx7Xa+vPRVS43G98vG1DnkDMIBUgFN0MDXflU8tbaZOUkXUH0+U27RoRpOIyCKjbMCVejwypzJJG4jIwb43rfl6wbwanocrJm9XFYfskuVC5K/TPyczNU7b84CXcbxks1Un6H6tLH9vf2LRnn8Ax7A5WQAAAHicY2BkYGAA4teL1+yI57f5ysDNwgAC529f0kOmWRiYVgEpDgYmEA8AUzEKsQAAAHicY2BkYGB2+O/AEMPCAAJAkpEBFbAAADgKAe0EAAAiAAAAAAQAAAAEAAAAAAAAKgAqACoAiAAAeJxjYGRgYGBhsGFgYgABEMkFhAwM/xn0QAIAD6YBhwB4nI1Ty07cMBS9QwKlQapQW3VXySvEqDCZGbGaHULiIQ1FKgjWMxknMfLEke2A+IJu+wntrt/QbVf9gG75jK577Lg8K1qQPCfnnnt8fX1NRC/pmjrk/zprC+8D7tBy9DHgBXoWfQ44Av8t4Bj4Z8CLtBL9CniJluPXASf0Lm4CXqFX8Q84dOLnMB17N4c7tBo1AS/Qi+hTwBH4rwHHwN8DXqQ30XXAS7QaLwSc0Gn8NuAVWou/gFmnjLrEaEh9GmDdDGgL3B4JsrRPDU2hTOiMSuJUIdKQQayiAth69r6akSSFqIJuA19TrzCIaY8sIoxyrNIrL//pw7A2iMygkX5vDj+G+kuoLdX4GlGK/8Lnlz6/h9MpmoO9rafrz7ILXEHHaAx95s9lsI7AHNMBWEZHULnfAXwG9/ZqdzLI08iuwRloXE8kfhXYAvE23+23DU3t626rbs8/8adv+9DWknsHp3E17oCf+Z48rvEQNZ78paYM38qfk3v/u3l3u3GXN2Dmvmvpf1Srwk3pB/VSsp512bA/GG5i2WJ7wu430yQ5K3nFGiOqgtmSB5pJVSizwaacmUZzZhXLlZTq8qGGFY2YcSkqbth6aW1tRmlaCFs2016m5qn36SbJrqosG4uMV4aP2PHBmB3tjtmgN2izkGQyLWprekbIntJFing32a5rKWCN/SdSoga45EJykyQ7asZvHQ8PTm6cslIpwyeyjbVltNikc2HTR7YKh9LBl9DADC0U/jLcBZDKrMhUBfQBvXRzLtFtjU9eNHKin0x5InTqb8lNpfKv1s1xHzTXRqgKzek/mb7nB8RZTCDhGEX3kK/8Q75AmUM/eLkfA+0Hi908Kx4eNsMgudg5GLdRD7a84npi+YxNr5i5KIbW5izXas7cHXIMAau1OueZhfj+cOcP3P8MNIWLyYOBuxL6DRylJ4cAAAB4nGNgYoAALjDJyIAOWMCiTIxMLDmZedkABtIBygAAAA==) format('woff'); +} + +.markdown-body { + -ms-text-size-adjust: 100%; + -webkit-text-size-adjust: 100%; + color: #333; + overflow: hidden; + //font-family: "Helvetica Neue", Helvetica, "Segoe UI", Arial, freesans, sans-serif; + font-family: "Microsoft YaHei", Helvetica, "Meiryo UI", "Malgun Gothic", "Segoe UI", "Trebuchet MS", "Monaco", monospace, Tahoma, STXihei, "华文细黑", STHeiti, "Helvetica Neue", "Droid Sans", "wenquanyi micro hei", FreeSans, Arimo, Arial, SimSun, "宋体", Heiti, "黑体", sans-serif; + font-size: 16px; + line-height: 1.6; + word-wrap: break-word; +} + +.markdown-body a { + background: transparent; +} + +.markdown-body a:active, +.markdown-body a:hover { + outline: 0; +} + +.markdown-body strong { + font-weight: bold; +} + +.markdown-body h1 { + font-size: 2em; + margin: 0.67em 0; +} + +.markdown-body img { + border: 0; +} + +.markdown-body hr { + -moz-box-sizing: content-box; + box-sizing: content-box; + height: 0; +} + +.markdown-body pre { + overflow: auto; +} + +.markdown-body code, +.markdown-body kbd, +.markdown-body pre { + font-family: "Meiryo UI", "YaHei Consolas Hybrid", Consolas, "Malgun Gothic", "Segoe UI", "Trebuchet MS", Helvetica, monospace, monospace; + font-size: 1em; +} + +.markdown-body input { + color: inherit; + font: inherit; + margin: 0; +} + +.markdown-body html input[disabled] { + cursor: default; +} + +.markdown-body input { + line-height: normal; +} + +.markdown-body input[type="checkbox"] { + -moz-box-sizing: border-box; + box-sizing: border-box; + padding: 0; +} + +.markdown-body table { + border-collapse: collapse; + border-spacing: 0; +} + +.markdown-body td, +.markdown-body th { + padding: 0; +} + +.markdown-body * { + -moz-box-sizing: border-box; + box-sizing: border-box; +} + +.markdown-body input { + font: 13px/1.4 Helvetica, arial, freesans, clean, sans-serif, "Segoe UI Emoji", "Segoe UI Symbol"; +} + +.markdown-body a { + color: #4183c4; + text-decoration: none; +} + +.markdown-body a:hover, +.markdown-body a:active { + text-decoration: underline; +} + +.markdown-body hr { + height: 0; + margin: 15px 0; + overflow: hidden; + background: transparent; + border: 0; + border-bottom: 1px solid #ddd; +} + +.markdown-body hr:before { + display: table; + content: ""; +} + +.markdown-body hr:after { + display: table; + clear: both; + content: ""; +} + +.markdown-body h1, +.markdown-body h2, +.markdown-body h3, +.markdown-body h4, +.markdown-body h5, +.markdown-body h6 { + margin-top: 15px; + margin-bottom: 15px; + line-height: 1.1; +} + +.markdown-body h1 { + font-size: 30px; +} + +.markdown-body h2 { + font-size: 21px; +} + +.markdown-body h3 { + font-size: 16px; +} + +.markdown-body h4 { + font-size: 14px; +} + +.markdown-body h5 { + font-size: 12px; +} + +.markdown-body h6 { + font-size: 11px; +} + +.markdown-body blockquote { + margin: 0; +} + +.markdown-body ul, +.markdown-body ol { + padding: 0; + margin-top: 0; + margin-bottom: 0; +} + +.markdown-body ol ol, +.markdown-body ul ol { + list-style-type: lower-roman; +} + +.markdown-body ul ul ol, +.markdown-body ul ol ol, +.markdown-body ol ul ol, +.markdown-body ol ol ol { + list-style-type: lower-alpha; +} + +.markdown-body dd { + margin-left: 0; +} + +.markdown-body code { + font-family: Consolas, "Liberation Mono", Menlo, Courier, monospace; + font-size: 12px; +} + +.markdown-body pre { + margin-top: 0; + margin-bottom: 0; + font: 12px Consolas, "Liberation Mono", Menlo, Courier, monospace; +} + +.markdown-body .octicon { + font: normal normal 16px octicons-anchor; + line-height: 1; + display: inline-block; + text-decoration: none; + -webkit-font-smoothing: antialiased; + -moz-osx-font-smoothing: grayscale; + -webkit-user-select: none; + -moz-user-select: none; + -ms-user-select: none; + user-select: none; +} + +.markdown-body .octicon-link:before { + content: '\f05c'; +} + +.markdown-body>*:first-child { + margin-top: 0 !important; +} + +.markdown-body>*:last-child { + margin-bottom: 0 !important; +} + +.markdown-body .anchor { + position: absolute; + top: 0; + left: 0; + display: block; + padding-right: 6px; + padding-left: 30px; + margin-left: -30px; +} + +.markdown-body .anchor:focus { + outline: none; +} + +.markdown-body h1, +.markdown-body h2, +.markdown-body h3, +.markdown-body h4, +.markdown-body h5, +.markdown-body h6 { + position: relative; + margin-top: 1em; + margin-bottom: 16px; + font-weight: bold; + line-height: 1.4; +} + +.markdown-body h1 .octicon-link, +.markdown-body h2 .octicon-link, +.markdown-body h3 .octicon-link, +.markdown-body h4 .octicon-link, +.markdown-body h5 .octicon-link, +.markdown-body h6 .octicon-link { + display: none; + color: #000; + vertical-align: middle; +} + +.markdown-body h1:hover .anchor, +.markdown-body h2:hover .anchor, +.markdown-body h3:hover .anchor, +.markdown-body h4:hover .anchor, +.markdown-body h5:hover .anchor, +.markdown-body h6:hover .anchor { + padding-left: 8px; + margin-left: -30px; + text-decoration: none; +} + +.markdown-body h1:hover .anchor .octicon-link, +.markdown-body h2:hover .anchor .octicon-link, +.markdown-body h3:hover .anchor .octicon-link, +.markdown-body h4:hover .anchor .octicon-link, +.markdown-body h5:hover .anchor .octicon-link, +.markdown-body h6:hover .anchor .octicon-link { + display: inline-block; +} + +.markdown-body h1 { + padding-bottom: 0.3em; + font-size: 2.25em; + line-height: 1.2; + border-bottom: 1px solid #eee; +} + +.markdown-body h1 .anchor { + line-height: 1; +} + +.markdown-body h2 { + padding-bottom: 0.3em; + font-size: 1.75em; + line-height: 1.225; + border-bottom: 1px solid #eee; +} + +.markdown-body h2 .anchor { + line-height: 1; +} + +.markdown-body h3 { + font-size: 1.5em; + line-height: 1.43; +} + +.markdown-body h3 .anchor { + line-height: 1.2; +} + +.markdown-body h4 { + font-size: 1.25em; +} + +.markdown-body h4 .anchor { + line-height: 1.2; +} + +.markdown-body h5 { + font-size: 1em; +} + +.markdown-body h5 .anchor { + line-height: 1.1; +} + +.markdown-body h6 { + font-size: 1em; + color: #777; +} + +.markdown-body h6 .anchor { + line-height: 1.1; +} + +.markdown-body p, +.markdown-body blockquote, +.markdown-body ul, +.markdown-body ol, +.markdown-body dl, +.markdown-body table, +.markdown-body pre { + margin-top: 0; + margin-bottom: 16px; +} +/* +.markdown-body hr { + height: 4px; + padding: 0; + margin: 16px 0; + background-color: #e7e7e7; + border: 0 none; +}*/ + +.markdown-body ul, +.markdown-body ol { + padding-left: 2em; +} + +.markdown-body ul ul, +.markdown-body ul ol, +.markdown-body ol ol, +.markdown-body ol ul { + margin-top: 0; + margin-bottom: 0; +} + +.markdown-body li>p { + margin-top: 16px; +} + +.markdown-body dl { + padding: 0; +} + +.markdown-body dl dt { + padding: 0; + margin-top: 16px; + font-size: 1em; + font-style: italic; + font-weight: bold; +} + +.markdown-body dl dd { + padding: 0 16px; + margin-bottom: 16px; +} + +.markdown-body blockquote { + padding: 0 15px; + color: #777; + border-left: 4px solid #ddd; +} + +.markdown-body blockquote>:first-child { + margin-top: 0; +} + +.markdown-body blockquote>:last-child { + margin-bottom: 0; +} + +.markdown-body table { + display: block; + width: 100%; + overflow: auto; + word-break: normal; + word-break: keep-all; +} + +.markdown-body table th { + font-weight: bold; +} + +.markdown-body table th, +.markdown-body table td { + padding: 6px 13px; + border: 1px solid #ddd; +} + +.markdown-body table tr { + background-color: #fff; + border-top: 1px solid #ccc; +} + +.markdown-body table tr:nth-child(2n) { + background-color: #f8f8f8; +} + +.markdown-body img { + max-width: 100%; + -moz-box-sizing: border-box; + box-sizing: border-box; +} + +.markdown-body code { + padding: 0; + padding-top: 0.2em; + padding-bottom: 0.2em; + margin: 0; + font-size: 85%; + background-color: rgba(0,0,0,0.04); + border-radius: 3px; +} + +.markdown-body code:before, +.markdown-body code:after { + letter-spacing: -0.2em; + content: "\00a0"; +} + +.markdown-body pre>code { + padding: 0; + margin: 0; + font-size: 100%; + word-break: normal; + white-space: pre; + background: transparent; + border: 0; +} + +.markdown-body .highlight { + margin-bottom: 16px; +} + +.markdown-body .highlight pre, +.markdown-body pre { + padding: 16px; + overflow: auto; + font-size: 85%; + line-height: 1.45; + background-color: #f7f7f7; + border-radius: 3px; +} + +.markdown-body .highlight pre { + margin-bottom: 0; + word-break: normal; +} + +.markdown-body pre { + word-wrap: normal; +} + +.markdown-body pre code { + display: inline; + max-width: initial; + padding: 0; + margin: 0; + overflow: initial; + line-height: inherit; + word-wrap: normal; + background-color: transparent; + border: 0; +} + +.markdown-body pre code:before, +.markdown-body pre code:after { + content: normal; +} + +.markdown-body kbd { + display: inline-block; + padding: 3px 5px; + font-size: 11px; + line-height: 10px; + color: #555; + vertical-align: middle; + background-color: #fcfcfc; + border: solid 1px #ccc; + border-bottom-color: #bbb; + border-radius: 3px; + box-shadow: inset 0 -1px 0 #bbb; +} + +.markdown-body .pl-c { + color: #969896; +} + +.markdown-body .pl-c1, +.markdown-body .pl-mdh, +.markdown-body .pl-mm, +.markdown-body .pl-mp, +.markdown-body .pl-mr, +.markdown-body .pl-s1 .pl-v, +.markdown-body .pl-s3, +.markdown-body .pl-sc, +.markdown-body .pl-sv { + color: #0086b3; +} + +.markdown-body .pl-e, +.markdown-body .pl-en { + color: #795da3; +} + +.markdown-body .pl-s1 .pl-s2, +.markdown-body .pl-smi, +.markdown-body .pl-smp, +.markdown-body .pl-stj, +.markdown-body .pl-vo, +.markdown-body .pl-vpf { + color: #333; +} + +.markdown-body .pl-ent { + color: #63a35c; +} + +.markdown-body .pl-k, +.markdown-body .pl-s, +.markdown-body .pl-st { + color: #a71d5d; +} + +.markdown-body .pl-pds, +.markdown-body .pl-s1, +.markdown-body .pl-s1 .pl-pse .pl-s2, +.markdown-body .pl-sr, +.markdown-body .pl-sr .pl-cce, +.markdown-body .pl-sr .pl-sra, +.markdown-body .pl-sr .pl-sre, +.markdown-body .pl-src { + color: #df5000; +} + +.markdown-body .pl-mo, +.markdown-body .pl-v { + color: #1d3e81; +} + +.markdown-body .pl-id { + color: #b52a1d; +} + +.markdown-body .pl-ii { + background-color: #b52a1d; + color: #f8f8f8; +} + +.markdown-body .pl-sr .pl-cce { + color: #63a35c; + font-weight: bold; +} + +.markdown-body .pl-ml { + color: #693a17; +} + +.markdown-body .pl-mh, +.markdown-body .pl-mh .pl-en, +.markdown-body .pl-ms { + color: #1d3e81; + font-weight: bold; +} + +.markdown-body .pl-mq { + color: #008080; +} + +.markdown-body .pl-mi { + color: #333; + font-style: italic; +} + +.markdown-body .pl-mb { + color: #333; + font-weight: bold; +} + +.markdown-body .pl-md, +.markdown-body .pl-mdhf { + background-color: #ffecec; + color: #bd2c00; +} + +.markdown-body .pl-mdht, +.markdown-body .pl-mi1 { + background-color: #eaffea; + color: #55a532; +} + +.markdown-body .pl-mdr { + color: #795da3; + font-weight: bold; +} + +.markdown-body kbd { + display: inline-block; + padding: 3px 5px; + font: 11px Consolas, "Liberation Mono", Menlo, Courier, monospace; + line-height: 10px; + color: #555; + vertical-align: middle; + background-color: #fcfcfc; + border: solid 1px #ccc; + border-bottom-color: #bbb; + border-radius: 3px; + box-shadow: inset 0 -1px 0 #bbb; +} + +.markdown-body .task-list-item { + list-style-type: none; +} + +.markdown-body .task-list-item+.task-list-item { + margin-top: 3px; +} + +.markdown-body .task-list-item input { + float: left; + margin: 0.3em 0 0.25em -1.6em; + vertical-align: middle; +} + +.markdown-body :checked+.radio-label { + z-index: 1; + position: relative; + border-color: #4183c4; +} diff --git a/system/admin/editor.md/scss/lib/prefixes.scss b/system/admin/editor.md/scss/lib/prefixes.scss new file mode 100644 index 00000000..70a785d2 --- /dev/null +++ b/system/admin/editor.md/scss/lib/prefixes.scss @@ -0,0 +1,784 @@ +@charset "UTF-8"; + +/*! prefixes.scss v0.1.0 | Author: Pandao | https://github.com/pandao/prefixes.scss | MIT license | Copyright (c) 2015 */ + +// appearance + +@mixin appearance($value) { + -webkit-appearance: $value; + -moz-appearance: $value; + -ms-appearance: $value; + appearance: $value; +} + +// clearfix + +@mixin clearfix() { + &:before, &:after { + content: " "; + display: table; + } + + &:after { + clear: both; + } +} + +// viewport + +@mixin viewport-device-width() { + width: device-width; + user-zoom: fixed; +} + +@mixin viewport() { + @-webkit-viewport { + @include viewport-device-width(); + } + @-moz-viewport { + @include viewport-device-width(); + } + + @-ms-viewport { + @include viewport-device-width(); + } + + @-o-viewport { + @include viewport-device-width(); + } + + @viewport { + @include viewport-device-width(); + } +} + +// Transform + +@mixin transform($transform) { + -webkit-transform: $transform; /* Safari, Chrome */ + -moz-transform: $transform; /* Firefox 3.5~16.0 */ + -ms-transform: $transform; /* IE9~10 */ + -o-transform: $transform; /* Opera 10.5~12.10 */ + transform: $transform; +} + +@mixin transform-origin($origin) { + -webkit-transform-origin: $origin; + -moz-transform-origin: $origin; /* Firefox 3.5~16.0 */ + -ms-transform-origin: $origin; /* IE9~10 */ + -o-transform-origin: $origin; /* Opera 10.5~12.10 */ + transform-origin: $origin; +} + +@mixin transform-origin-x($origin) { + -webkit-transform-origin-x: $origin; /* Blink, Webkit */ + transform-origin-x: $origin; /* IE11+ */ +} + +@mixin transform-origin-y($origin) { + -webkit-transform-origin-y: $origin; /* Blink, Webkit */ + transform-origin-y: $origin; /* IE11+ */ +} + +@mixin transform-origin-z($origin) { + -webkit-transform-origin-z: $origin; /* Blink, Webkit */ + transform-origin-z: $origin; /* IE11+ */ +} + +@mixin transform-style($style) { + -webkit-transform-style: $style; + -moz-transform-style: $style; /* Firefox 10~16.0 */ + -ms-transform-style: $style; /* IE9~10 */ + transform-style: $style; /* Firefox, Blink, IE11+ */ +} + +// perspective + +@mixin perspective($value) { + -webkit-perspective: $value; /* Safari, Chrome */ + perspective: $value; /* None yet / Non-standard */ +} + +@mixin perspective-origin($value) { + -webkit-perspective-origin: $value; /* Safari, Chrome 12+ */ + -moz-perspective-origin: $value; /* Firefox 10~16 */ + perspective-origin: $value; /* Opera 15+, IE10+ */ +} + +@mixin perspective-origin-x($value) { + -webkit-perspective-origin-x: $value; /* Safari, Chrome 12+ */ + perspective-origin-x: $value; /* IE10+ */ +} + +@mixin perspective-origin-y($value) { + -webkit-perspective-origin-y: $value; /* Safari, Chrome 12+ */ + perspective-origin-y: $value; /* IE10+ */ +} + +@mixin backface-visibility($value : hidden) { + -webkit-backface-visibility: $value; /* Chrome, Safari, Opera 15+ */ + -moz-backface-visibility: $value; /* Firefox */ + -ms-backface-visibility: $value; /* IE10 */ + backface-visibility: $value; +} + +// Transitions IE10+ + +@mixin transition($transition...) { + -webkit-transition: $transition; /* Safari, Chrome */ + -moz-transition: $transition; /* Firefox 4.0~16.0 */ + transition: $transition; /* IE >9, FF >15, Opera >12.0 */ +} + +@mixin transition-property($property) { + -webkit-transition-property: $property; + -moz-transition-property: $property; /* Firefox 4.0~16.0 */ + transition-property: $property; +} + +@mixin transition-duration($duration) { + -webkit-transition-duration: $duration; + -moz-transition-duration: $duration; /* Firefox 4.0~16.0 */ + transition-duration: $duration; +} + +@mixin transition-timing-function($easing) { + -webkit-transition-timing-function: $easing; + -moz-transition-timing-function: $easing; /* Firefox 4.0~16.0 */ + transition-timing-function: $easing; +} + +@mixin transition-delay($delay) { + -webkit-transition-delay: $delay; + -moz-transition-delay: $delay; /* Firefox 4.0~16.0 */ + transition-delay: $delay; +} + +// Flex align + +@mixin align-content($value) { + -webkit-align-content: $value; /* Chrome 21.0+, Safari Not supported. */ + align-content: $value; /* Firefox 28+, Opera 12.10, IE Not supported. */ +} + +@mixin align-items($value) { + -webkit-align-items: $value; /* Safari 7.0+, Chrome 21.0+ */ + align-items: $value; /* Firefox 20.0+, IE11+, Opera 12.10 */ +} + +@mixin align-self($value) { + -webkit-align-self: $value; /* Chrome 21~36, Safari Not supported. */ + align-self: $value; /* Firefox 28+, Opera 12.10, IE Not supported. */ +} + +// Animations IE10+ + +@mixin keyframes($name) { + @-webkit-keyframes #{$name} { + @content; + } + + @-moz-keyframes #{$name} { + @content; + } + + @keyframes #{$name} { + @content; + } +} + +@mixin animation($animation...) { + -webkit-animation: $animation; + -moz-animation: $animation; /* Firefox 5.0~16.0 */ + animation: $animation; /* IE10+ */ +} + +@mixin animation-name($name) { + -webkit-animation-name: $name; + -moz-animation-name: $name; /* Firefox 5.0~16.0 */ + animation-name: $name; +} + +@mixin animation-duration($time : 1s) { + -webkit-animation-duration: $time; + -moz-animation-duration: $time; /* Firefox 5.0~16.0 */ + animation-duration: $time; +} + +@mixin animation-timing-function($easing : ease) { + -webkit-animation-timing-function: $easing; + -moz-animation-timing-function: $easing; /* Firefox 5.0~16.0 */ + animation-timing-function: $easing; +} + +@mixin animation-delay($delay : 1s) { + -webkit-animation-delay: $delay; + -moz-animation-delay: $delay; /* Firefox 5.0~16.0 */ + animation-delay: $delay; +} + +@mixin animation-iteration-count($count : infinite) { + -webkit-animation-iteration-count: $count; + -moz-animation-iteration-count: $count; /* Firefox 5.0~16.0 */ + animation-iteration-count: $count; +} + +// normal or alternate +@mixin animation-direction($direction : normal) { + -webkit-animation-direction: $direction; + -moz-animation-direction: $direction; /* Firefox 5.0~16.0 */ + animation-direction: $direction; +} + +// paused or running + +@mixin animation-play-state($state) { + -webkit-animation-play-state: $state; + -moz-animation-play-state: $state; /* Firefox 5.0~16.0 */ + animation-play-state: $state; +} + +// animation-fill-mode + +@mixin animation-fill-mode($mode) { + -webkit-animation-fill-mode: $mode; + -moz-animation-fill-mode: $mode; + animation-fill-mode: $mode; +} + +// user-select + +@mixin user-select($type) { + -webkit-user-select: $type; + -moz-user-select: $type; + -ms-user-select: $type; + -o-user-select: $type; + user-select: $type; +} + +// border-radius + +@mixin border-radius($radius: 4px) { + -webkit-border-radius: $radius; + -moz-border-radius: $radius; + -ms-border-radius: $radius; + -o-border-radius: $radius; + border-radius: $radius; +} + +@mixin border-top-left-radius($radius: 4px) { + -webkit-border-top-left-radius: $radius; + -moz-border-top-left-radius: $radius; + -ms-border-top-left-radius: $radius; + -o-border-top-left-radius: $radius; + border-top-left-radius: $radius; +} + +@mixin border-top-right-radius($radius: 4px) { + -webkit-border-top-right-radius: $radius; + -moz-border-top-right-radius: $radius; + -ms-border-top-right-radius: $radius; + -o-border-top-right-radius: $radius; + border-top-right-radius: $radius; +} + +@mixin border-bottom-left-radius($radius: 4px) { + -webkit-border-bottom-left-radius: $radius; + -moz-border-bottom-left-radius: $radius; + -ms-border-bottom-left-radius: $radius; + -o-border-bottom-left-radius: $radius; + border-bottom-left-radius: $radius; +} + +@mixin border-bottom-right-radius($radius: 4px) { + -webkit-border-bottom-right-radius: $radius; + -moz-border-bottom-right-radius: $radius; + -ms-border-bottom-right-radius: $radius; + -o-border-bottom-right-radius: $radius; + border-bottom-right-radius: $radius; +} + +// border-image + +@mixin border-image($value) { + -webkit-border-image: $value; /* Safari 5, Chrome */ + -moz-border-image: $value; /* Firefox 3.5~15.0 */ + -o-border-image: $value; /* Opera */ + border-image: $value; /* Safari 6+, Chrome, New */ +} + +@mixin border-image-source($value) { + -webkit-border-image-source: $value; /* Safari 5, Chrome */ + border-image-source: $value; /* Safari 6+, Chrome, IE11+, Opera 15+ */ +} + +@mixin border-image-slice($value) { + -webkit-border-image-slice: $value; /* Safari 5, Chrome */ + border-image-slice: $value; /* Safari 6+, Chrome, IE11+, Opera 15+ */ +} + +@mixin border-image-width($value) { + -webkit-border-image-width: $value; /* Safari 5, Chrome */ + border-image-width: $value; /* Safari 6+, Chrome, IE11+, Opera 15+ */ +} + +@mixin border-image-outset($value) { + -webkit-border-image-outset: $value; /* Safari 5, Chrome */ + border-image-outset: $value; /* Safari 6+, Chrome, IE11+, Opera 15+ */ +} + +@mixin border-image-repeat($value) { + -webkit-border-image-repeat: $value; /* Safari 5, Chrome */ + border-image-repeat: $value; /* Safari 6+, Chrome, IE11+, Opera 15+ */ +} + +// box-shadow + +@mixin box-shadow($value) { + -webkit-box-shadow: $value; /* Webkit browsers */ + -moz-box-shadow: $value; /* Firefox */ + -ms-box-shadow: $value; /* IE9 */ + -o-box-shadow: $value; /* Opera(Old) */ + box-shadow: $value; /* IE9+, News */ +} + +//box-sizing + +@mixin box-sizing($value) { + -webkit-box-sizing: $value; + -moz-box-sizing: $value; + box-sizing: $value; +} + +// box-reflect + +@mixin box-reflect($value) { + -webkit-box-reflect: $value; /* Chrome, Safari, iOS, Blackberry */ + box-reflect: $value; /* None yet / Non-standard */ +} + +// background + +@mixin linear-gradient($start-color, $end-color, $position : top, $perStart : 0%, $perEnd : 100%) { + background: -webkit-linear-gradient($position, $start-color, $end-color); /* Webkit browsers */ + background: -moz-linear-gradient( $position, $start-color, $end-color); /* Firefox(old) */ + background: -o-linear-gradient( $position, $start-color, $end-color); /* Opera(old) */ + background: -ms-linear-gradient( $position, $start-color $perStart, $end-color $perEnd); /* IE10 */ + filter: progid:DXImageTransform.Microsoft.gradient(GradientType=0, start-colorstr=#{$start-color}, end-colorstr=#{$end-color}); /* IE9 */ + ms-filter: "progid:DXImageTransform.Microsoft.gradient (GradientType=0, start-colorstr=#{$start-color}, end-colorstr=#{$end-color})"; /* IE8 */ + background: linear-gradient( $position, $start-color $perStart, $end-color $perEnd); /* W3C */ +} + +@mixin background-clip($value) { + -webkit-background-clip: $value; + background-clip: $value; /* Firefox 4.0, IE9+, Opera 10.5+, Chrome, Safari 3.0+ */ +} + +@mixin background-origin($value) { + -webkit-background-origin: $value; + background-origin: $value; /* IE9+, Other */ +} + +@mixin background-size($value) { + -webkit-background-size: $value; /* Chrome, iOS, Safari */ + -moz-background-size: $value; /* Firefox 3.6~4.0 */ + -o-background-size: $value; /* Opera 9.5 */ + background-size: $value; /* IE9+, New */ +} + +// Column + +@mixin column-count($value) { + -webkit-column-count: $value; /* Chrome, Safari, Android, Blackberry */ + -moz-column-count: $value; /* Firefox 34+ */ + column-count: $value; /* IE 10+, Opera 11.1+, New */ +} + +@mixin column-gap($value) { + -webkit-column-gap: $value; /* Chrome, Safari, Android, Blackberry */ + -moz-column-gap: $value; /* Firefox 34+ */ + column-gap: $value; /* IE 10+, Opera 11.1+, New */ +} + +@mixin column-rule($value) { + -webkit-column-rule: $value; /* Chrome, Safari, Android, Blackberry */ + -moz-column-rule: $value; /* Firefox 34+ */ + column-rule: $value; /* IE 10+, Opera 11.1+, New */ +} + +@mixin column-rule-color($value) { + -webkit-column-rule-color: $value; /* Chrome, Safari, Android, Blackberry */ + -moz-column-rule-color: $value; /* Firefox 34+ */ + column-rule-color: $value; /* IE 10+, Opera 11.1+, New */ +} + +@mixin column-rule-style($value) { + -webkit-column-rule-style: $value; /* Chrome, Safari, Android, Blackberry */ + -moz-column-rule-style: $value; /* Firefox 34+ */ + column-rule-style: $value; /* IE 10+, Opera 11.1+, New */ +} + +@mixin column-rule-width($value) { + -webkit-column-rule-width: $value; /* Chrome, Safari, Android, Blackberry */ + -moz-column-rule-width: $value; /* Firefox 34+ */ + column-rule-width: $value; /* IE 10+, Opera 11.1+, New */ +} + +@mixin column-fill($value) { + -webkit-column-fill: $value; /* None yet */ + -moz-column-fill: $value; /* Firefox 13.0+ */ + column-fill: $value; /* None yet / Non-standard */ +} + +@mixin column-span($value) { + -webkit-column-span: $value; /* Safari, Chrome, iOS 7.0+, Android, Opera 26+ */ + -moz-column-span: $value; /* Firefox 34+ */ + column-span: $value; /* IE10+, Opera Mini */ +} + +@mixin column-width($value) { + -webkit-column-width: $value; /* Safari, Chrome, iOS 7.0+, Android, Opera 26+ */ + -moz-column-width: $value; /* Firefox */ + column-width: $value; /* IE10+, Opera */ +} + +// columns: column-width column-count; + +@mixin columns($value) { + -webkit-columns: $value; /* Safari, Chrome, iOS 7.0+, Android, Opera 26+ */ + -moz-columns: $value; /* Firefox */ + columns: $value; /* IE10+, Opera */ +} + +// clip-path + +@mixin clip-path($value) { + -webkit-clip-path: $value; /* Chrome, iOS, Safari */ + clip-path: $value; +} + +// display + +@mixin display-grid() { + display: -ms-grid; /* IE 10 */ + display: grid; /* None yet */ +} + +@mixin display-flex() { + display: -webkit-box; /* Old - iOS 6-, Safari 3.1~6, Blackberry 7 */ + display: -ms-flexbox; /* TWEENER - IE 10 */ + display: -webkit-flex; /* New - Safari 6.1+. iOS 7.1+, Blackberry 10 */ + display: flex; /* New, Spec - Firefox, Chrome, Opera */ +} + +@mixin inline-flex($value) { + -webkit-inline-flex: $value; /* Chrome 21.0+ */ + inline-flex: $value; /* Firefox 20+, Opera 12.5 */ +} + +@mixin flex($value) { + -webkit-box-flex: $value; /* Old - iOS 6-, Safari 3.1~6 */ + -webkit-flex: $value; /* Safari 6.1+. iOS 7.1+, Blackberry 10 */ + -ms-flex: $value; /* IE 10 */ + flex: $value; /* New, Spec - Firefox, Chrome, Opera */ +} + +@mixin flex-direction($value) { + -webkit-flex-direction: $value; /* Chrome 21.0+, But Safari & Android & iOS Not supported. */ + flex-direction: $value; /* Firefox 28+, IE11, Opera 12.10 */ +} + +@mixin flex-basis($value) { + -webkit-flex-basis: $value; /* Chrome 21.0+, But Safari & Android & iOS Not supported. */ + flex-basis: $value; /* Firefox 22+, IE11, Opera 12.10 */ +} + +@mixin flex-flow($value) { + -webkit-flex-flow: $value; /* Chrome 21.0+, But Safari & Android & iOS Not supported. */ + flex-flow: $value; /* Firefox 28+, IE11, Opera 12.10 */ +} + +@mixin flex-grow($value) { + -webkit-flex-grow: $value; /* Chrome 21.0+, But Safari & Android & iOS Not supported. */ + flex-grow: $value; /* Firefox 20+, Opera 12.10, IE Not supported. */ +} + +@mixin flex-shrink($value) { + -webkit-flex-shrink: $value; /* Chrome 21.0+, But Safari & Android & iOS Not supported. */ + flex-shrink: $value; /* Firefox 20+, Opera 12.10, IE Not supported. */ +} + +@mixin flex-wrap($value) { + -webkit-flex-wrap: $value; /* Safari 6.1+, Chrome 21.0+, Android 4.4+, iOS 7.0+ */ + flex-wrap: $value; /* Firefox 28+, IE11, Opera 12.10 */ +} + +@mixin order($value) { + -webkit-order: $value; /* Chrome 21+, Safari Not supported. */ + -ms-flex-order: $value; /* IE 10.0 */ + order: $value; /* Firefox 20+, Opera 12.10 */ +} + +// flow + +@mixin flow-into($value) { + -webkit-flow-into: $value; /* Safari 7.1+, iOS Safari 7.1+ */ + -ms-flow-into: $value; /* IE10+ */ + flow-into: $value; /* None yet */ +} + +@mixin flow-from($value) { + -webkit-flow-from: $value; /* Safari 7.1+, iOS Safari 7.1+ */ + -ms-flow-from: $value; /* IE10+ */ + flow-from: $value; /* None yet */ +} + +// filter + +@mixin filter($value) { + -webkit-filter: $value; /* Chrome 23+, Safari 6.0+, Blackberry 10.0+ */ + filter: $value; /* None yet */ +} + +// filter blur + +@mixin filter-blur($value : 10px, $ie-value : 10) { + filter: progid:DXImageTransform.Microsoft.Blur(PixelRadius=#{$ie-value}, MakeShadow=false); /* IE6~IE9 */ + -webkit-filter: blur($value); /* Chrome, Opera, iOS, Safari */ + -moz-filter: blur($value); /* Firefox(Old) */ + -ms-filter: blur($value); + filter: blur($value); +} + +@mixin font-kerning($value) { + -webkit-font-kerning: $value; + font-kerning: $value; +} + +// font-feature-settings + +@mixin font-feature-settings($value) { + -webkit-font-feature-settings: $value; /* Chrome 16-26, Blackberry 10 */ + -moz-font-feature-settings: $value; /* Firefox 4-21 */ + font-feature-settings: $value; /* IE 10, Safari 4.0-6.0 */ +} + +@mixin font-variant-ligatures($value) { + -webkit-font-variant-ligatures: $value; + font-variant-ligatures: $value; +} + +// hyphens + +@mixin hyphens($value : auto) { + // Chrome 29- and Android 4.0 Browser support "-webkit-hyphens: none", but not the "auto" property. + -webkit-hyphens: $value; /* Safari 5.1+, Chrome */ + -moz-hyphens: $value; /* Firefox 6.0+ */ + -ms-hyphens: $value; /* IE 10+ */ + hyphens: $value; /* None yet */ +} + +@mixin justify-content($value) { + -webkit-justify-content: $value; /* Chrome 21+, Safari Not supported. */ + justify-content: $value; /* Firefox 20+, Opera 12.10, IE Not supported. */ +} + +// line + +@mixin line-break($value) { + -webkit-line-break: $value; + line-break: $value; +} + +// margin + +@mixin margin-start($value) { + -webkit-margin-start: $value; /* Safari 3.0+, Chrome */ + -moz-margin-start: $value; /* Firefox 1.0+ */ + margin-start: $value; /* None yet / Non-standard */ +} + +@mixin margin-end($value) { + -webkit-margin-end: $value; /* Safari 3.0+, Chrome */ + -moz-margin-end: $value; /* Firefox 1.0+ */ + margin-end: $value; /* None yet / Non-standard */ +} + +// mask + +@mixin mask-image($value) { + -webkit-mask-image: $value; /* Chrome, iOS, Safari */ + mask-image: $value; /* None yet / Non-standard */ +} + +@mixin mask-size($value) { + -webkit-mask-size: $value; /* Chrome, iOS, Safari */ + mask-size: $value; /* None yet / Non-standard */ +} + +@mixin mask-clip($value) { + -webkit-mask-clip: $value; /* Chrome, iOS, Safari */ + mask-clip: $value; /* None yet / Non-standard */ +} + +@mixin mask-position($value) { + -webkit-mask-position: $value; /* Chrome, iOS, Safari */ + mask-position: $value; /* None yet / Non-standard */ +} + +@mixin mask-position-x($value) { + -webkit-mask-position-x: $value; /* Chrome, iOS, Safari */ + mask-position-x: $value; /* None yet / Non-standard */ +} + +@mixin mask-position-y($value) { + -webkit-mask-position-y: $value; /* Chrome, iOS, Safari */ + mask-position-y: $value; /* None yet / Non-standard */ +} + +@mixin mask-origin($value) { + -webkit-mask-origin: $value; /* Chrome, iOS, Safari */ + mask-origin: $value; /* None yet / Non-standard */ +} + +@mixin mask-repeat($value) { + -webkit-mask-repeat: $value; /* Chrome, iOS, Safari */ + mask-repeat: $value; /* None yet / Non-standard */ +} + +@mixin mask-attachment($value) { + -webkit-mask-attachment: $value; /* Chrome, iOS, Safari */ + mask-attachment: $value; /* None yet / Non-standard */ +} + +@mixin mask-composite($value) { + -webkit-mask-composite: $value; /* Chrome, iOS, Safari */ + mask-composite: $value; /* None yet / Non-standard */ +} + +@mixin mask-box-image($value) { + -webkit-mask-box-image: $value; /* Chrome, iOS, Safari */ + mask-box-image: $value; /* None yet / Non-standard */ +} + +// opacity + +@mixin opacity($opacity) { + opacity: $opacity; /* W3C */ + filter: alpha(opacity=($opacity * 100)); /* IE */ +} + +// padding + +@mixin padding-start($value) { + -webkit-padding-start: $value; /* Safari, Chrome, WebKit */ + -moz-padding-start: $value; /* Firefox 3+ */ + padding-start: $value; +} + +@mixin padding-end($value) { + -webkit-padding-end: $value; /* Safari, Chrome, WebKit */ + -moz-padding-end: $value; /* Firefox 3+ */ + padding-end: $value; +} + +// ruby-position + +@mixin ruby-position($value) { + -webkit-ruby-position: $value; /* Blink, Webkit */ + ruby-position: $value; /* Firefox, IE */ +} + +// Text + +@mixin text-size-adjust($value) { + -webkit-text-size-adjust: $value; /* Chrome 27+ */ + -moz-text-size-adjust: $value; /* Firefox */ + text-size-adjust: $value; /* None yet */ +} + +@mixin text-align-last($value) { + -webkit-text-align-last: $value; /* Chrome 35+, Safari Not supported. */ + -moz-text-align-last: $value; /* Firefox 12.0 */ + text-align-last: $value; /* IE 5.5+ */ +} + +@mixin text-justify($value) { + -webkit-text-justify: $value; + text-justify: $value; +} + +@mixin text-decoration-color($value) { + -webkit-text-decoration-color: $value; + text-decoration-color: $value; +} + +@mixin text-decoration-line($value) { + -webkit-text-decoration-line: $value; + text-decoration-line: $value; +} + +@mixin text-decoration-style($value) { + -webkit-text-decoration-style: $value; + text-decoration-style: $value; +} + +@mixin text-orientation($value) { + -webkit-text-orientation: $value; + text-orientation: $value; +} + +@mixin text-underline-position($value) { + -webkit-text-underline-position: $value; /* Chrome 33 not fully supported. */ + text-underline-position: $value; /* IE 5 not fully supported. */ +} + +@mixin text-emphasis($value) { + -webkit-text-emphasis: $value; /* Blink */ + text-emphasis: $value; /* WebKit */ +} + +@mixin text-emphasis-color($value) { + -webkit-text-emphasis-color: $value; /* Blink */ + text-emphasis-color: $value; /* WebKit */ +} + +@mixin text-emphasis-style($value) { + -webkit-text-emphasis-style: $value; /* Blink */ + text-emphasis-style: $value; /* WebKit */ +} + +@mixin text-emphasis-position($value) { + -webkit-text-emphasis-position: $value; /* Blink */ + text-emphasis-position: $value; /* WebKit */ +} + +// tab + +@mixin tab-size($value) { + -webkit-tab-size: $value; /* Chrome 21+, Safari 6.1+ */ + -moz-tab-size: $value; /* Firefox 4.0 */ + -o-tab-size: $value; /* Opera 10.6~15 */ + tab-size: $value; /* Blink & Webkit */ +} + +// input-placeholder + +@mixin input-placeholder($seletor) { + + #{$seletor}::-webkit-input-placeholder { + @content; + } + + #{$seletor}:-moz-placeholder { /* Firefox 4~18 */ + @content; + } + + #{$seletor}::-moz-placeholder { /* Firefox 19+ */ + @content; + } + + #{$seletor}:-ms-input-placeholder { /* IE10+ */ + @content; + } +} \ No newline at end of file diff --git a/system/admin/editor.md/scss/lib/variables.scss b/system/admin/editor.md/scss/lib/variables.scss new file mode 100644 index 00000000..4e21a7be --- /dev/null +++ b/system/admin/editor.md/scss/lib/variables.scss @@ -0,0 +1,11 @@ +@charset "UTF-8"; + +// Global Variables + +$prefix : ".editormd-"; +$color : #666; +$mainColor : #2196F3; +$primaryColor : $mainColor; +$secondColor : #33CC66; +$thirdColor : #999999; +$borderColor : #ddd; \ No newline at end of file diff --git a/system/admin/editor.md/scss/prettify.scss b/system/admin/editor.md/scss/prettify.scss new file mode 100644 index 00000000..c50bbe0f --- /dev/null +++ b/system/admin/editor.md/scss/prettify.scss @@ -0,0 +1,53 @@ +@charset "UTF-8"; + +/*! Pretty printing styles. Used with prettify.js. */ +/* SPAN elements with the classes below are added by prettyprint. */ +.pln { color: #000 } /* plain text */ + +@media screen { + .str { color: #080 } /* string content */ + .kwd { color: #008 } /* a keyword */ + .com { color: #800 } /* a comment */ + .typ { color: #606 } /* a type name */ + .lit { color: #066 } /* a literal value */ + /* punctuation, lisp open bracket, lisp close bracket */ + .pun, .opn, .clo { color: #660 } + .tag { color: #008 } /* a markup tag name */ + .atn { color: #606 } /* a markup attribute name */ + .atv { color: #080 } /* a markup attribute value */ + .dec, .var { color: #606 } /* a declaration; a variable name */ + .fun { color: red } /* a function name */ +} + +/* Use higher contrast and text-weight for printable form. */ +@media print, projection { + .str { color: #060 } + .kwd { color: #006; font-weight: bold } + .com { color: #600; font-style: italic } + .typ { color: #404; font-weight: bold } + .lit { color: #044 } + .pun, .opn, .clo { color: #440 } + .tag { color: #006; font-weight: bold } + .atn { color: #404 } + .atv { color: #060 } +} + +/* Put a border around prettyprinted code snippets. */ +pre.prettyprint { padding: 2px; border: 1px solid #888 } + +/* Specify class=linenums on a pre to get line numbering */ +ol.linenums { margin-top: 0; margin-bottom: 0 } /* IE indents via margin-left */ +li.L0, +li.L1, +li.L2, +li.L3, +li.L5, +li.L6, +li.L7, +li.L8 { list-style-type: none } +/* Alternate shading for lines */ +li.L1, +li.L3, +li.L5, +li.L7, +li.L9 { background: #eee } \ No newline at end of file diff --git a/system/admin/editor.md/src/editormd.js b/system/admin/editor.md/src/editormd.js new file mode 100644 index 00000000..9e9d708c --- /dev/null +++ b/system/admin/editor.md/src/editormd.js @@ -0,0 +1,4261 @@ +;(function(factory) { + "use strict" + + // CommonJS/Node.js + if ( + typeof require === "function" && + typeof exports === "object" && + typeof module === "object" + ) { + module.exports = factory + } else if (typeof define === "function") { + // AMD/CMD/Sea.js + if (define.amd) { + // for Require.js + /* Require.js define replace */ + } else { + define(["jquery"], factory) // for Sea.js + } + } else { + window.editormd = factory() + } +})(function() { + /* Require.js assignment replace */ + + "use strict" + + var $ = typeof jQuery !== "undefined" ? jQuery : Zepto + + if (typeof $ === "undefined") { + return + } + + /** + * editormd + * + * @param {String} id 编辑器的ID + * @param {Object} options 配置选项 Key/Value + * @returns {Object} editormd 返回editormd对象 + */ + + var editormd = function(id, options) { + return new editormd.fn.init(id, options) + } + + editormd.title = editormd.$name = "Editor.md" + editormd.version = "1.5.0" + editormd.homePage = "https://pandao.github.io/editor.md/" + editormd.classPrefix = "editormd-" + + editormd.toolbarModes = { + full: [ + "undo", + "redo", + "|", + "bold", + "del", + "italic", + "quote", + "ucwords", + "uppercase", + "lowercase", + "|", + "h1", + "h2", + "h3", + "h4", + "h5", + "h6", + "|", + "list-ul", + "list-ol", + "hr", + "|", + "link", + "reference-link", + "image", + "code", + "preformatted-text", + "code-block", + "table", + "datetime", + "emoji", + "html-entities", + "pagebreak", + "|", + "goto-line", + "watch", + "preview", + "fullscreen", + "clear", + "search", + "|", + "help", + "info" + ], + simple: [ + "undo", + "redo", + "|", + "bold", + "del", + "italic", + "quote", + "uppercase", + "lowercase", + "|", + "h1", + "h2", + "h3", + "h4", + "h5", + "h6", + "|", + "list-ul", + "list-ol", + "hr", + "|", + "watch", + "preview", + "fullscreen", + "|", + "help", + "info" + ], + mini: ["undo", "redo", "|", "watch", "preview", "|", "help", "info"] + } + + editormd.defaults = { + mode: "gfm", //gfm or markdown + name: "", // Form element name + value: "", // value for CodeMirror, if mode not gfm/markdown + theme: "", // Editor.md self themes, before v1.5.0 is CodeMirror theme, default empty + editorTheme: "default", // Editor area, this is CodeMirror theme at v1.5.0 + previewTheme: "", // Preview area theme, default empty + markdown: "", // Markdown source code + appendMarkdown: "", // if in init textarea value not empty, append markdown to textarea + width: "100%", + height: "100%", + path: "./lib/", // Dependents module file directory + pluginPath: "", // If this empty, default use settings.path + "../plugins/" + delay: 300, // Delay parse markdown to html, Uint : ms + autoLoadModules: true, // Automatic load dependent module files + watch: true, + placeholder: "Enjoy Markdown! coding now...", + gotoLine: true, + codeFold: false, + autoHeight: false, + minHeight: "150px", + maxHeight: "none", // default "none" + autoFocus: true, + autoCloseTags: true, + searchReplace: true, + syncScrolling: true, // true | false | "single", default true + readOnly: false, + tabSize: 4, + indentUnit: 4, + lineNumbers: true, + lineWrapping: true, + autoCloseBrackets: true, + showTrailingSpace: true, + matchBrackets: true, + indentWithTabs: true, + styleSelectedText: true, + matchWordHighlight: true, // options: true, false, "onselected" + styleActiveLine: true, // Highlight the current line + dialogLockScreen: true, + dialogShowMask: true, + dialogDraggable: true, + dialogMaskBgColor: "#fff", + dialogMaskOpacity: 0.1, + fontSize: "13px", + saveHTMLToTextarea: false, + disabledKeyMaps: [], + onload: function() {}, + onresize: function() {}, + onchange: function() {}, + onwatch: null, + onunwatch: null, + onpreviewing: function() {}, + onpreviewed: function() {}, + onfullscreen: function() {}, + onfullscreenExit: function() {}, + onscroll: function() {}, + onpreviewscroll: function() {}, + imageUpload: false, + imageFormats: ["jpg", "jpeg", "gif", "png", "bmp", "webp"], + imageUploadURL: "", + crossDomainUpload: false, + uploadCallbackURL: "", + toc: true, // Table of contents + tocm: false, // Using [TOCM], auto create ToC dropdown menu + tocTitle: "", // for ToC dropdown menu btn + tocDropdown: false, + tocContainer: "", + tocStartLevel: 1, // Said from H1 to create ToC + htmlDecode: false, // Open the HTML tag identification + pageBreak: true, // Enable parse page break [========] + atLink: true, // for @link + emailLink: true, // for email address auto link + taskList: false, // Enable Github Flavored Markdown task lists + emoji: false, // :emoji: , Support Github emoji, Twitter Emoji (Twemoji); + // Support FontAwesome icon emoji :fa-xxx: > Using fontAwesome icon web fonts; + // Support Editor.md logo icon emoji :editormd-logo: :editormd-logo-1x: > 1~8x; + tex: false, // TeX(LaTeX), based on KaTeX + flowChart: false, // flowChart.js only support IE9+ + sequenceDiagram: false, // sequenceDiagram.js only support IE9+ + previewCodeHighlight: true, + toolbar: true, // show/hide toolbar + toolbarAutoFixed: true, // on window scroll auto fixed position + toolbarIcons: "full", + toolbarTitles: {}, + toolbarHandlers: { + ucwords: function() { + return editormd.toolbarHandlers.ucwords + }, + lowercase: function() { + return editormd.toolbarHandlers.lowercase + } + }, + toolbarCustomIcons: { + // using html tag create toolbar icon, unused default tag. + lowercase: + 'a', + ucwords: + 'Aa' + }, + toolbarIconsClass: { + undo: "fa-undo", + redo: "fa-repeat", + bold: "fa-bold", + del: "fa-strikethrough", + italic: "fa-italic", + quote: "fa-quote-left", + uppercase: "fa-font", + h1: editormd.classPrefix + "bold", + h2: editormd.classPrefix + "bold", + h3: editormd.classPrefix + "bold", + h4: editormd.classPrefix + "bold", + h5: editormd.classPrefix + "bold", + h6: editormd.classPrefix + "bold", + "list-ul": "fa-list-ul", + "list-ol": "fa-list-ol", + hr: "fa-minus", + link: "fa-link", + "reference-link": "fa-anchor", + image: "fa-picture-o", + code: "fa-code", + "preformatted-text": "fa-file-code-o", + "code-block": "fa-file-code-o", + table: "fa-table", + datetime: "fa-clock-o", + emoji: "fa-smile-o", + "html-entities": "fa-copyright", + pagebreak: "fa-newspaper-o", + "goto-line": "fa-terminal", // fa-crosshairs + watch: "fa-eye-slash", + unwatch: "fa-eye", + preview: "fa-desktop", + search: "fa-search", + fullscreen: "fa-arrows-alt", + clear: "fa-eraser", + help: "fa-question-circle", + info: "fa-info-circle" + }, + toolbarIconTexts: {}, + lang: { + name: "zh-cn", + description: + "开源在线Markdown编辑器
                    Open source online Markdown editor.", + tocTitle: "目录", + toolbar: { + undo: "撤销(Ctrl+Z)", + redo: "重做(Ctrl+Y)", + bold: "粗体", + del: "删除线", + italic: "斜体", + quote: "引用", + ucwords: "将每个单词首字母转成大写", + uppercase: "将所选转换成大写", + lowercase: "将所选转换成小写", + h1: "标题1", + h2: "标题2", + h3: "标题3", + h4: "标题4", + h5: "标题5", + h6: "标题6", + "list-ul": "无序列表", + "list-ol": "有序列表", + hr: "横线", + link: "链接", + "reference-link": "引用链接", + image: "添加图片", + code: "行内代码", + "preformatted-text": "预格式文本 / 代码块(缩进风格)", + "code-block": "代码块(多语言风格)", + table: "添加表格", + datetime: "日期时间", + emoji: "Emoji表情", + "html-entities": "HTML实体字符", + pagebreak: "插入分页符", + "goto-line": "跳转到行", + watch: "关闭实时预览", + unwatch: "开启实时预览", + preview: "全窗口预览HTML(按 Shift + ESC还原)", + fullscreen: "全屏(按ESC还原)", + clear: "清空", + search: "搜索", + help: "使用帮助", + info: "关于" + editormd.title + }, + buttons: { + enter: "确定", + cancel: "取消", + close: "关闭" + }, + dialog: { + link: { + title: "添加链接", + url: "链接地址", + urlTitle: "链接标题", + urlEmpty: "错误:请填写链接地址。" + }, + referenceLink: { + title: "添加引用链接", + name: "引用名称", + url: "链接地址", + urlId: "链接ID", + urlTitle: "链接标题", + nameEmpty: "错误:引用链接的名称不能为空。", + idEmpty: "错误:请填写引用链接的ID。", + urlEmpty: "错误:请填写引用链接的URL地址。" + }, + image: { + title: "添加图片", + url: "图片地址", + link: "图片链接", + alt: "图片描述", + uploadButton: "本地上传", + imageURLEmpty: "错误:图片地址不能为空。", + uploadFileEmpty: "错误:上传的图片不能为空。", + formatNotAllowed: + "错误:只允许上传图片文件,允许上传的图片文件格式有:" + }, + preformattedText: { + title: "添加预格式文本或代码块", + emptyAlert: "错误:请填写预格式文本或代码的内容。" + }, + codeBlock: { + title: "添加代码块", + selectLabel: "代码语言:", + selectDefaultText: "请选择代码语言", + otherLanguage: "其他语言", + unselectedLanguageAlert: "错误:请选择代码所属的语言类型。", + codeEmptyAlert: "错误:请填写代码内容。" + }, + htmlEntities: { + title: "HTML 实体字符" + }, + help: { + title: "使用帮助" + } + } + } + } + editormd.classNames = { + tex: editormd.classPrefix + "tex" + } + editormd.dialogZindex = 99999 + editormd.$katex = null + editormd.$marked = null + editormd.$CodeMirror = null + editormd.$prettyPrint = null + + var timer, flowchartTimer + + editormd.prototype = editormd.fn = { + state: { + watching: false, + loaded: false, + preview: false, + fullscreen: false + }, + + /** + * 构造函数/实例初始化 + * Constructor / instance initialization + * + * @param {String} id 编辑器的ID + * @param {Object} [options={}] 配置选项 Key/Value + * @returns {editormd} 返回editormd的实例对象 + */ + + init: function(id, options) { + options = options || {} + if (typeof id === "object") { + options = id + } + var _this = this + var classPrefix = (this.classPrefix = editormd.classPrefix) + var settings = (this.settings = $.extend( + true, + editormd.defaults, + options + )) + id = typeof id === "object" ? settings.id : id + + var editor = (this.editor = $("#" + id)) + this.id = id + this.lang = settings.lang + var classNames = (this.classNames = { + textarea: { + html: classPrefix + "html-textarea", + markdown: classPrefix + "markdown-textarea" + } + }) + settings.pluginPath = + settings.pluginPath === "" + ? settings.path + "../plugins/" + : settings.pluginPath + + this.state.watching = settings.watch ? true : false + if (!editor.hasClass("editormd")) { + editor.addClass("editormd") + } + editor.css({ + width: + typeof settings.width === "number" + ? settings.width + "px" + : settings.width, + height: + typeof settings.height === "number" + ? settings.height + "px" + : settings.height + }) + if (settings.autoHeight) { + editor.css({ + height: "auto", + minHeight: settings.minHeight + }) + } + var markdownTextarea = (this.markdownTextarea = editor.children( + "textarea" + )) + if (markdownTextarea.length < 1) { + editor.append("") + markdownTextarea = this.markdownTextarea = editor.children("textarea") + } + markdownTextarea + .addClass(classNames.textarea.markdown) + .attr("placeholder", settings.placeholder) + if ( + typeof markdownTextarea.attr("name") === "undefined" || + markdownTextarea.attr("name") === "" + ) { + markdownTextarea.attr( + "name", + settings.name !== "" ? settings.name : id + "-markdown-doc" + ) + } + var appendElements = [ + !settings.readOnly + ? '' + : "", + settings.saveHTMLToTextarea + ? '' + : "", + '
                    ', + '
                    ', + '
                    ' + ].join("\n") + editor.append(appendElements).addClass(classPrefix + "vertical") + if (settings.theme !== "") { + editor.addClass(classPrefix + "theme-" + settings.theme) + } + this.mask = editor.children("." + classPrefix + "mask") + this.containerMask = editor.children("." + classPrefix + "container-mask") + if (settings.markdown !== "") { + markdownTextarea.val(settings.markdown) + } + if (settings.appendMarkdown !== "") { + markdownTextarea.val(markdownTextarea.val() + settings.appendMarkdown) + } + this.htmlTextarea = editor.children("." + classNames.textarea.html) + this.preview = editor.children("." + classPrefix + "preview") + this.previewContainer = this.preview.children( + "." + classPrefix + "preview-container" + ) + if (settings.previewTheme !== "") { + this.preview.addClass( + classPrefix + "preview-theme-" + settings.previewTheme + ) + } + if (typeof define === "function" && define.amd) { + if (typeof katex !== "undefined") { + editormd.$katex = katex + } + if (settings.searchReplace && !settings.readOnly) { + editormd.loadCSS(settings.path + "codemirror/addon/dialog/dialog") + editormd.loadCSS( + settings.path + "codemirror/addon/search/matchesonscrollbar" + ) + } + } + if ( + (typeof define === "function" && define.amd) || + !settings.autoLoadModules + ) { + if (typeof CodeMirror !== "undefined") { + editormd.$CodeMirror = CodeMirror + } + + if (typeof marked !== "undefined") { + editormd.$marked = marked + } + this.setCodeMirror() + .setToolbar() + .loadedDisplay() + } else { + this.loadQueues() + } + return this + }, + + /** + * 所需组件加载队列 + * Required components loading queue + * + * @returns {editormd} 返回editormd的实例对象 + */ + + loadQueues: function() { + var _this = this + var settings = this.settings + var loadPath = settings.path + var loadFlowChartOrSequenceDiagram = function() { + if (editormd.isIE8) { + _this.loadedDisplay() + return + } + if (settings.flowChart || settings.sequenceDiagram) { + editormd.loadScript(loadPath + "raphael.min", function() { + editormd.loadScript(loadPath + "underscore.min", function() { + if (!settings.flowChart && settings.sequenceDiagram) { + editormd.loadScript( + loadPath + "sequence-diagram.min", + function() { + _this.loadedDisplay() + } + ) + } else if (settings.flowChart && !settings.sequenceDiagram) { + editormd.loadScript(loadPath + "flowchart.min", function() { + editormd.loadScript( + loadPath + "jquery.flowchart.min", + function() { + _this.loadedDisplay() + } + ) + }) + } else if (settings.flowChart && settings.sequenceDiagram) { + editormd.loadScript(loadPath + "flowchart.min", function() { + editormd.loadScript( + loadPath + "jquery.flowchart.min", + function() { + editormd.loadScript( + loadPath + "sequence-diagram.min", + function() { + _this.loadedDisplay() + } + ) + } + ) + }) + } + }) + }) + } else { + _this.loadedDisplay() + } + } + + editormd.loadCSS(loadPath + "codemirror/codemirror.min") + if (settings.searchReplace && !settings.readOnly) { + editormd.loadCSS(loadPath + "codemirror/addon/dialog/dialog") + editormd.loadCSS( + loadPath + "codemirror/addon/search/matchesonscrollbar" + ) + } + if (settings.codeFold) { + editormd.loadCSS(loadPath + "codemirror/addon/fold/foldgutter") + } + editormd.loadScript(loadPath + "codemirror/codemirror.min", function() { + editormd.$CodeMirror = CodeMirror + editormd.loadScript(loadPath + "codemirror/modes.min", function() { + editormd.loadScript(loadPath + "codemirror/addons.min", function() { + _this.setCodeMirror() + if (settings.mode !== "gfm" && settings.mode !== "markdown") { + _this.loadedDisplay() + return false + } + _this.setToolbar() + editormd.loadScript(loadPath + "marked.min", function() { + editormd.$marked = marked + + if (settings.previewCodeHighlight) { + editormd.loadScript(loadPath + "prettify.min", function() { + loadFlowChartOrSequenceDiagram() + }) + } else { + loadFlowChartOrSequenceDiagram() + } + }) + }) + }) + }) + return this + }, + + /** + * 设置 Editor.md 的整体主题,主要是工具栏 + * Setting Editor.md theme + * + * @returns {editormd} 返回editormd的实例对象 + */ + + setTheme: function(theme) { + var editor = this.editor + var oldTheme = this.settings.theme + var themePrefix = this.classPrefix + "theme-" + editor.removeClass(themePrefix + oldTheme).addClass(themePrefix + theme) + this.settings.theme = theme + return this + }, + + /** + * 设置 CodeMirror(编辑区)的主题 + * Setting CodeMirror (Editor area) theme + * + * @returns {editormd} 返回editormd的实例对象 + */ + + setEditorTheme: function(theme) { + var settings = this.settings + settings.editorTheme = theme + if (theme !== "default") { + editormd.loadCSS( + settings.path + "codemirror/theme/" + settings.editorTheme + ) + } + this.cm.setOption("theme", theme) + return this + }, + + /** + * setEditorTheme() 的别名 + * setEditorTheme() alias + * + * @returns {editormd} 返回editormd的实例对象 + */ + + setCodeMirrorTheme: function(theme) { + this.setEditorTheme(theme) + return this + }, + + /** + * 设置 Editor.md 的主题 + * Setting Editor.md theme + * + * @returns {editormd} 返回editormd的实例对象 + */ + + setPreviewTheme: function(theme) { + var preview = this.preview + var oldTheme = this.settings.previewTheme + var themePrefix = this.classPrefix + "preview-theme-" + preview.removeClass(themePrefix + oldTheme).addClass(themePrefix + theme) + this.settings.previewTheme = theme + return this + }, + + /** + * 配置和初始化CodeMirror组件 + * CodeMirror initialization + * + * @returns {editormd} 返回editormd的实例对象 + */ + + setCodeMirror: function() { + var settings = this.settings + var editor = this.editor + if (settings.editorTheme !== "default") { + editormd.loadCSS( + settings.path + "codemirror/theme/" + settings.editorTheme + ) + } + var codeMirrorConfig = { + mode: settings.mode, + theme: settings.editorTheme, + tabSize: settings.tabSize, + dragDrop: false, + autofocus: settings.autoFocus, + autoCloseTags: settings.autoCloseTags, + readOnly: settings.readOnly ? "nocursor" : false, + indentUnit: settings.indentUnit, + lineNumbers: settings.lineNumbers, + lineWrapping: settings.lineWrapping, + extraKeys: { + "Ctrl-Q": function(cm) { + cm.foldCode(cm.getCursor()) + } + }, + foldGutter: settings.codeFold, + gutters: ["CodeMirror-linenumbers", "CodeMirror-foldgutter"], + matchBrackets: settings.matchBrackets, + indentWithTabs: settings.indentWithTabs, + styleActiveLine: settings.styleActiveLine, + styleSelectedText: settings.styleSelectedText, + autoCloseBrackets: settings.autoCloseBrackets, + showTrailingSpace: settings.showTrailingSpace, + highlightSelectionMatches: !settings.matchWordHighlight + ? false + : { + showToken: + settings.matchWordHighlight === "onselected" ? false : /\w/ + } + } + + this.codeEditor = this.cm = editormd.$CodeMirror.fromTextArea( + this.markdownTextarea[0], + codeMirrorConfig + ) + this.codeMirror = this.cmElement = editor.children(".CodeMirror") + + if (settings.value !== "") { + this.cm.setValue(settings.value) + } + + this.codeMirror.css({ + fontSize: settings.fontSize, + width: !settings.watch ? "100%" : "50%" + }) + + if (settings.autoHeight) { + this.codeMirror.css({ + height: "auto", + minHeight: settings.minHeight, + maxHeight: settings.maxHeight + }) + this.codeMirror.find(".CodeMirror-gutters").css({ + minHeight: settings.minHeight + }) + this.codeMirror.find(".CodeMirror-scroll").css({ + minHeight: settings.minHeight, + maxHeight: settings.maxHeight + }) + this.cm.setOption("viewportMargin", Infinity) + } + + if (!settings.lineNumbers) { + this.codeMirror.find(".CodeMirror-gutters").css("border-right", "none") + } + + return this + }, + + /** + * 获取CodeMirror的配置选项 + * Get CodeMirror setting options + * + * @returns {Mixed} return CodeMirror setting option value + */ + + getCodeMirrorOption: function(key) { + return this.cm.getOption(key) + }, + + /** + * 配置和重配置CodeMirror的选项 + * CodeMirror setting options / resettings + * + * @returns {editormd} 返回editormd的实例对象 + */ + + setCodeMirrorOption: function(key, value) { + this.cm.setOption(key, value) + + return this + }, + + /** + * 添加 CodeMirror 键盘快捷键 + * Add CodeMirror keyboard shortcuts key map + * + * @returns {editormd} 返回editormd的实例对象 + */ + + addKeyMap: function(map, bottom) { + this.cm.addKeyMap(map, bottom) + + return this + }, + + /** + * 移除 CodeMirror 键盘快捷键 + * Remove CodeMirror keyboard shortcuts key map + * + * @returns {editormd} 返回editormd的实例对象 + */ + + removeKeyMap: function(map) { + this.cm.removeKeyMap(map) + + return this + }, + + /** + * 跳转到指定的行 + * Goto CodeMirror line + * + * @param {String|Intiger} line line number or "first"|"last" + * @returns {editormd} 返回editormd的实例对象 + */ + + gotoLine: function(line) { + var settings = this.settings + if (!settings.gotoLine) { + return this + } + var cm = this.cm + var editor = this.editor + var count = cm.lineCount() + var preview = this.preview + if (typeof line === "string") { + if (line === "last") { + line = count + } + if (line === "first") { + line = 1 + } + } + if (typeof line !== "number") { + alert("Error: The line number must be an integer.") + return this + } + line = parseInt(line) - 1 + if (line > count) { + alert("Error: The line number range 1-" + count) + return this + } + cm.setCursor({ line: line, ch: 0 }) + var scrollInfo = cm.getScrollInfo() + var clientHeight = scrollInfo.clientHeight + var coords = cm.charCoords({ line: line, ch: 0 }, "local") + cm.scrollTo(null, (coords.top + coords.bottom - clientHeight) / 2) + if (settings.watch) { + var cmScroll = this.codeMirror.find(".CodeMirror-scroll")[0] + var height = $(cmScroll).height() + var scrollTop = cmScroll.scrollTop + var percent = scrollTop / cmScroll.scrollHeight + if (scrollTop === 0) { + preview.scrollTop(0) + } else if (scrollTop + height >= cmScroll.scrollHeight - 16) { + preview.scrollTop(preview[0].scrollHeight) + } else { + preview.scrollTop(preview[0].scrollHeight * percent) + } + } + cm.focus() + return this + }, + + /** + * 扩展当前实例对象,可同时设置多个或者只设置一个 + * Extend editormd instance object, can mutil setting. + * + * @returns {editormd} this(editormd instance object.) + */ + + extend: function() { + if (typeof arguments[1] !== "undefined") { + if (typeof arguments[1] === "function") { + arguments[1] = $.proxy(arguments[1], this) + } + this[arguments[0]] = arguments[1] + } + if ( + typeof arguments[0] === "object" && + typeof arguments[0].length === "undefined" + ) { + $.extend(true, this, arguments[0]) + } + return this + }, + + /** + * 设置或扩展当前实例对象,单个设置 + * Extend editormd instance object, one by one + * + * @param {String|Object} key option key + * @param {String|Object} value option value + * @returns {editormd} this(editormd instance object.) + */ + + set: function(key, value) { + if (typeof value !== "undefined" && typeof value === "function") { + value = $.proxy(value, this) + } + this[key] = value + return this + }, + + /** + * 重新配置 + * Resetting editor options + * + * @param {String|Object} key option key + * @param {String|Object} value option value + * @returns {editormd} this(editormd instance object.) + */ + + config: function(key, value) { + var settings = this.settings + if (typeof key === "object") { + settings = $.extend(true, settings, key) + } + if (typeof key === "string") { + settings[key] = value + } + this.settings = settings + this.recreate() + return this + }, + + /** + * 注册事件处理方法 + * Bind editor event handle + * + * @param {String} eventType event type + * @param {Function} callback 回调函数 + * @returns {editormd} this(editormd instance object.) + */ + + on: function(eventType, callback) { + var settings = this.settings + if (typeof settings["on" + eventType] !== "undefined") { + settings["on" + eventType] = $.proxy(callback, this) + } + return this + }, + + /** + * 解除事件处理方法 + * Unbind editor event handle + * + * @param {String} eventType event type + * @returns {editormd} this(editormd instance object.) + */ + + off: function(eventType) { + var settings = this.settings + if (typeof settings["on" + eventType] !== "undefined") { + settings["on" + eventType] = function() {} + } + return this + }, + + /** + * 显示工具栏 + * Display toolbar + * + * @param {Function} [callback=function(){}] 回调函数 + * @returns {editormd} 返回editormd的实例对象 + */ + + showToolbar: function(callback) { + var settings = this.settings + if (settings.readOnly) { + return this + } + if ( + settings.toolbar && + (this.toolbar.length < 1 || + this.toolbar.find("." + this.classPrefix + "menu").html() === "") + ) { + this.setToolbar() + } + settings.toolbar = true + this.toolbar.show() + this.resize() + $.proxy(callback || function() {}, this)() + return this + }, + + /** + * 隐藏工具栏 + * Hide toolbar + * + * @param {Function} [callback=function(){}] 回调函数 + * @returns {editormd} this(editormd instance object.) + */ + + hideToolbar: function(callback) { + var settings = this.settings + settings.toolbar = false + this.toolbar.hide() + this.resize() + $.proxy(callback || function() {}, this)() + return this + }, + + /** + * 页面滚动时工具栏的固定定位 + * Set toolbar in window scroll auto fixed position + * + * @returns {editormd} 返回editormd的实例对象 + */ + + setToolbarAutoFixed: function(fixed) { + var state = this.state + var editor = this.editor + var toolbar = this.toolbar + var settings = this.settings + if (typeof fixed !== "undefined") { + settings.toolbarAutoFixed = fixed + } + var autoFixedHandle = function() { + var $window = $(window) + var top = $window.scrollTop() + if (!settings.toolbarAutoFixed) { + return false + } + if (top - editor.offset().top > 10 && top < editor.height()) { + toolbar.css({ + position: "fixed", + width: editor.width() + "px", + left: ($window.width() - editor.width()) / 2 + "px" + }) + } else { + toolbar.css({ + position: "absolute", + width: "100%", + left: 0 + }) + } + } + if ( + !state.fullscreen && + !state.preview && + settings.toolbar && + settings.toolbarAutoFixed + ) { + $(window).bind("scroll", autoFixedHandle) + } + return this + }, + + /** + * 配置和初始化工具栏 + * Set toolbar and Initialization + * + * @returns {editormd} 返回editormd的实例对象 + */ + + setToolbar: function() { + var settings = this.settings + if (settings.readOnly) { + return this + } + var editor = this.editor + var preview = this.preview + var classPrefix = this.classPrefix + var toolbar = (this.toolbar = editor.children( + "." + classPrefix + "toolbar" + )) + if (settings.toolbar && toolbar.length < 1) { + var toolbarHTML = + '
                      ' + editor.append(toolbarHTML) + toolbar = this.toolbar = editor.children("." + classPrefix + "toolbar") + } + if (!settings.toolbar) { + toolbar.hide() + return this + } + toolbar.show() + var icons = + typeof settings.toolbarIcons === "function" + ? settings.toolbarIcons() + : typeof settings.toolbarIcons === "string" + ? editormd.toolbarModes[settings.toolbarIcons] + : settings.toolbarIcons + var toolbarMenu = toolbar.find("." + this.classPrefix + "menu"), + menu = "" + var pullRight = false + for (var i = 0, len = icons.length; i < len; i++) { + var name = icons[i] + if (name === "||") { + pullRight = true + } else if (name === "|") { + menu += '
                    • |
                    • ' + } else { + var isHeader = /h(\d)/.test(name) + var index = name + if (name === "watch" && !settings.watch) { + index = "unwatch" + } + var title = settings.lang.toolbar[index] + var iconTexts = settings.toolbarIconTexts[index] + var iconClass = settings.toolbarIconsClass[index] + title = typeof title === "undefined" ? "" : title + iconTexts = typeof iconTexts === "undefined" ? "" : iconTexts + iconClass = typeof iconClass === "undefined" ? "" : iconClass + var menuItem = pullRight ? '
                    • ' : "
                    • " + if ( + typeof settings.toolbarCustomIcons[name] !== "undefined" && + typeof settings.toolbarCustomIcons[name] !== "function" + ) { + menuItem += settings.toolbarCustomIcons[name] + } else { + menuItem += + '' + menuItem += + '' + + (isHeader + ? name.toUpperCase() + : iconClass === "" ? iconTexts : "") + + "" + menuItem += "" + } + menuItem += "
                    • " + menu = pullRight ? menuItem + menu : menu + menuItem + } + } + toolbarMenu.html(menu) + toolbarMenu + .find('[title="Lowercase"]') + .attr("title", settings.lang.toolbar.lowercase) + toolbarMenu + .find('[title="ucwords"]') + .attr("title", settings.lang.toolbar.ucwords) + this.setToolbarHandler() + this.setToolbarAutoFixed() + return this + }, + + /** + * 工具栏图标事件处理对象序列 + * Get toolbar icons event handlers + * + * @param {Object} cm CodeMirror的实例对象 + * @param {String} name 要获取的事件处理器名称 + * @returns {Object} 返回处理对象序列 + */ + + dialogLockScreen: function() { + $.proxy(editormd.dialogLockScreen, this)() + return this + }, + dialogShowMask: function(dialog) { + $.proxy(editormd.dialogShowMask, this)(dialog) + return this + }, + getToolbarHandles: function(name) { + var toolbarHandlers = (this.toolbarHandlers = editormd.toolbarHandlers) + return name && typeof toolbarIconHandlers[name] !== "undefined" + ? toolbarHandlers[name] + : toolbarHandlers + }, + + /** + * 工具栏图标事件处理器 + * Bind toolbar icons event handle + * + * @returns {editormd} 返回editormd的实例对象 + */ + + setToolbarHandler: function() { + var _this = this + var settings = this.settings + if (!settings.toolbar || settings.readOnly) { + return this + } + var toolbar = this.toolbar + var cm = this.cm + var classPrefix = this.classPrefix + var toolbarIcons = (this.toolbarIcons = toolbar.find( + "." + classPrefix + "menu > li > a" + )) + var toolbarIconHandlers = this.getToolbarHandles() + toolbarIcons.bind(editormd.mouseOrTouch("click", "touchend"), function( + event + ) { + var icon = $(this).children(".fa") + var name = icon.attr("name") + var cursor = cm.getCursor() + var selection = cm.getSelection() + if (name === "") { + return + } + _this.activeIcon = icon + if (typeof toolbarIconHandlers[name] !== "undefined") { + $.proxy(toolbarIconHandlers[name], _this)(cm) + } else { + if (typeof settings.toolbarHandlers[name] !== "undefined") { + $.proxy(settings.toolbarHandlers[name], _this)( + cm, + icon, + cursor, + selection + ) + } + } + if ( + name !== "link" && + name !== "reference-link" && + name !== "image" && + name !== "code-block" && + name !== "preformatted-text" && + name !== "watch" && + name !== "preview" && + name !== "search" && + name !== "fullscreen" && + name !== "info" + ) { + cm.focus() + } + return false + }) + + return this + }, + + /** + * 动态创建对话框 + * Creating custom dialogs + * + * @param {Object} options 配置项键值对 Key/Value + * @returns {dialog} 返回创建的dialog的jQuery实例对象 + */ + + createDialog: function(options) { + return $.proxy(editormd.createDialog, this)(options) + }, + + /** + * 创建关于Editor.md的对话框 + * Create about Editor.md dialog + * + * @returns {editormd} 返回editormd的实例对象 + */ + + createInfoDialog: function() { + var _this = this + var editor = this.editor + var classPrefix = this.classPrefix + var infoDialogHTML = [ + '
                      ', + '
                      ', + '

                      ' + + editormd.title + + "v" + + editormd.version + + "

                      ", + "

                      " + this.lang.description + "

                      ", + '

                      ' + + editormd.homePage + + '

                      ', + '

                      Copyright © 2015 Pandao, The MIT License.

                      ', + "
                      ", + '', + "
                      " + ].join("\n") + editor.append(infoDialogHTML) + var infoDialog = (this.infoDialog = editor.children( + "." + classPrefix + "dialog-info" + )) + infoDialog + .find("." + classPrefix + "dialog-close") + .bind(editormd.mouseOrTouch("click", "touchend"), function() { + _this.hideInfoDialog() + }) + infoDialog + .css("border", editormd.isIE8 ? "1px solid #ddd" : "") + .css("z-index", editormd.dialogZindex) + .show() + this.infoDialogPosition() + return this + }, + + /** + * 关于Editor.md对话居中定位 + * Editor.md dialog position handle + * + * @returns {editormd} 返回editormd的实例对象 + */ + + infoDialogPosition: function() { + var infoDialog = this.infoDialog + var _infoDialogPosition = function() { + infoDialog.css({ + top: ($(window).height() - infoDialog.height()) / 2 + "px", + left: ($(window).width() - infoDialog.width()) / 2 + "px" + }) + } + _infoDialogPosition() + $(window).resize(_infoDialogPosition) + return this + }, + + /** + * 显示关于Editor.md + * Display about Editor.md dialog + * + * @returns {editormd} 返回editormd的实例对象 + */ + + showInfoDialog: function() { + $("html,body").css("overflow-x", "hidden") + var _this = this + var editor = this.editor + var settings = this.settings + var infoDialog = (this.infoDialog = editor.children( + "." + this.classPrefix + "dialog-info" + )) + if (infoDialog.length < 1) { + this.createInfoDialog() + } + this.lockScreen(true) + this.mask + .css({ + opacity: settings.dialogMaskOpacity, + backgroundColor: settings.dialogMaskBgColor + }) + .show() + infoDialog.css("z-index", editormd.dialogZindex).show() + this.infoDialogPosition() + return this + }, + + /** + * 隐藏关于Editor.md + * Hide about Editor.md dialog + * + * @returns {editormd} 返回editormd的实例对象 + */ + + hideInfoDialog: function() { + $("html,body").css("overflow-x", "") + this.infoDialog.hide() + this.mask.hide() + this.lockScreen(false) + return this + }, + + /** + * 锁屏 + * lock screen + * + * @param {Boolean} lock Boolean 布尔值,是否锁屏 + * @returns {editormd} 返回editormd的实例对象 + */ + + lockScreen: function(lock) { + editormd.lockScreen(lock) + this.resize() + return this + }, + + /** + * 编辑器界面重建,用于动态语言包或模块加载等 + * Recreate editor + * + * @returns {editormd} 返回editormd的实例对象 + */ + + recreate: function() { + var _this = this + var editor = this.editor + var settings = this.settings + this.codeMirror.remove() + this.setCodeMirror() + if (!settings.readOnly) { + if (editor.find(".editormd-dialog").length > 0) { + editor.find(".editormd-dialog").remove() + } + if (settings.toolbar) { + this.getToolbarHandles() + this.setToolbar() + } + } + this.loadedDisplay(true) + return this + }, + + /** + * 高亮预览HTML的pre代码部分 + * highlight of preview codes + * + * @returns {editormd} 返回editormd的实例对象 + */ + + previewCodeHighlight: function() { + var settings = this.settings + var previewContainer = this.previewContainer + if (settings.previewCodeHighlight) { + previewContainer.find("pre").addClass("prettyprint linenums") + if (typeof prettyPrint !== "undefined") { + prettyPrint() + } + } + + return this + }, + + /** + * 解析TeX(KaTeX)科学公式 + * TeX(KaTeX) Renderer + * + * @returns {editormd} 返回editormd的实例对象 + */ + + katexRender: function() { + if (timer === null) { + return this + } + this.previewContainer + .find("." + editormd.classNames.tex) + .each(function() { + var tex = $(this) + editormd.$katex.render(tex.text(), tex[0]) + tex.find(".katex").css("font-size", "1.6em") + }) + return this + }, + + /** + * 解析和渲染流程图及时序图 + * FlowChart and SequenceDiagram Renderer + * + * @returns {editormd} 返回editormd的实例对象 + */ + + flowChartAndSequenceDiagramRender: function() { + var $this = this + var settings = this.settings + var previewContainer = this.previewContainer + if (editormd.isIE8) { + return this + } + if (settings.flowChart) { + if (flowchartTimer === null) { + return this + } + previewContainer.find(".flowchart").flowChart() + } + + if (settings.sequenceDiagram) { + previewContainer + .find(".sequence-diagram") + .sequenceDiagram({ theme: "simple" }) + } + var preview = $this.preview + var codeMirror = $this.codeMirror + var codeView = codeMirror.find(".CodeMirror-scroll") + var height = codeView.height() + var scrollTop = codeView.scrollTop() + var percent = scrollTop / codeView[0].scrollHeight + var tocHeight = 0 + preview.find(".markdown-toc-list").each(function() { + tocHeight += $(this).height() + }) + var tocMenuHeight = preview.find(".editormd-toc-menu").height() + tocMenuHeight = !tocMenuHeight ? 0 : tocMenuHeight + if (settings.syncScrolling) { + if (scrollTop === 0) { + preview.scrollTop(0) + } else if (scrollTop + height >= codeView[0].scrollHeight - 16) { + preview.scrollTop(preview[0].scrollHeight) + } else { + preview.scrollTop( + (preview[0].scrollHeight + tocHeight + tocMenuHeight) * percent + ) + } + } + return this + }, + + /** + * 注册键盘快捷键处理 + * Register CodeMirror keyMaps (keyboard shortcuts). + * + * @param {Object} keyMap KeyMap key/value {"(Ctrl/Shift/Alt)-Key" : function(){}} + * @returns {editormd} return this + */ + + registerKeyMaps: function(keyMap) { + var _this = this + var cm = this.cm + var settings = this.settings + var toolbarHandlers = editormd.toolbarHandlers + var disabledKeyMaps = settings.disabledKeyMaps + keyMap = keyMap || null + if (keyMap) { + for (var i in keyMap) { + if ($.inArray(i, disabledKeyMaps) < 0) { + var map = {} + map[i] = keyMap[i] + cm.addKeyMap(keyMap) + } + } + } else { + for (var k in editormd.keyMaps) { + var _keyMap = editormd.keyMaps[k] + var handle = + typeof _keyMap === "string" + ? $.proxy(toolbarHandlers[_keyMap], _this) + : $.proxy(_keyMap, _this) + if ( + $.inArray(k, ["F9", "F10", "F11"]) < 0 && + $.inArray(k, disabledKeyMaps) < 0 + ) { + var _map = {} + _map[k] = handle + cm.addKeyMap(_map) + } + } + $(window).keydown(function(event) { + var keymaps = { + "120": "F9", + "121": "F10", + "122": "F11" + } + if ($.inArray(keymaps[event.keyCode], disabledKeyMaps) < 0) { + switch (event.keyCode) { + case 120: + $.proxy(toolbarHandlers.watch, _this)() + return false + case 121: + $.proxy(toolbarHandlers.preview, _this)() + return false + case 122: + $.proxy(toolbarHandlers.fullscreen, _this)() + return false + default: + break + } + } + }) + } + return this + }, + + /** + * 绑定同步滚动 + * @returns {editormd} return this + */ + + bindScrollEvent: function() { + var _this = this + var preview = this.preview + var settings = this.settings + var codeMirror = this.codeMirror + var mouseOrTouch = editormd.mouseOrTouch + if (!settings.syncScrolling) { + return this + } + var cmBindScroll = function() { + codeMirror + .find(".CodeMirror-scroll") + .bind(mouseOrTouch("scroll", "touchmove"), function(event) { + var height = $(this).height() + var scrollTop = $(this).scrollTop() + var percent = scrollTop / $(this)[0].scrollHeight + var tocHeight = 0 + preview.find(".markdown-toc-list").each(function() { + tocHeight += $(this).height() + }) + var tocMenuHeight = preview.find(".editormd-toc-menu").height() + tocMenuHeight = !tocMenuHeight ? 0 : tocMenuHeight + if (scrollTop === 0) { + preview.scrollTop(0) + } else if (scrollTop + height >= $(this)[0].scrollHeight - 16) { + preview.scrollTop(preview[0].scrollHeight) + } else { + preview.scrollTop( + (preview[0].scrollHeight + tocHeight + tocMenuHeight) * percent + ) + } + $.proxy(settings.onscroll, _this)(event) + }) + } + var cmUnbindScroll = function() { + codeMirror + .find(".CodeMirror-scroll") + .unbind(mouseOrTouch("scroll", "touchmove")) + } + var previewBindScroll = function() { + preview.bind(mouseOrTouch("scroll", "touchmove"), function(event) { + var height = $(this).height() + var scrollTop = $(this).scrollTop() + var percent = scrollTop / $(this)[0].scrollHeight + var codeView = codeMirror.find(".CodeMirror-scroll") + if (scrollTop === 0) { + codeView.scrollTop(0) + } else if (scrollTop + height >= $(this)[0].scrollHeight) { + codeView.scrollTop(codeView[0].scrollHeight) + } else { + codeView.scrollTop(codeView[0].scrollHeight * percent) + } + $.proxy(settings.onpreviewscroll, _this)(event) + }) + } + + var previewUnbindScroll = function() { + preview.unbind(mouseOrTouch("scroll", "touchmove")) + } + codeMirror.bind({ + mouseover: cmBindScroll, + mouseout: cmUnbindScroll, + touchstart: cmBindScroll, + touchend: cmUnbindScroll + }) + if (settings.syncScrolling === "single") { + return this + } + preview.bind({ + mouseover: previewBindScroll, + mouseout: previewUnbindScroll, + touchstart: previewBindScroll, + touchend: previewUnbindScroll + }) + return this + }, + + bindChangeEvent: function() { + var _this = this + var cm = this.cm + var settings = this.settings + + // if (!settings.syncScrolling) { + // return this; + // } + + cm.on("change", function(_cm, changeObj) { + if (settings.watch) { + _this.previewContainer.css( + "padding", + settings.autoHeight ? "20px 20px 50px 40px" : "20px" + ) + } + timer = setTimeout(function() { + clearTimeout(timer) + _this.save() + timer = null + }, settings.delay) + }) + return this + }, + + /** + * 加载队列完成之后的显示处理 + * Display handle of the module queues loaded after. + * + * @param {Boolean} recreate 是否为重建编辑器 + * @returns {editormd} 返回editormd的实例对象 + */ + + loadedDisplay: function(recreate) { + recreate = recreate || false + var _this = this + var editor = this.editor + var preview = this.preview + var settings = this.settings + this.containerMask.hide() + this.save() + if (settings.watch) { + preview.show() + } + editor.data("oldWidth", editor.width()).data("oldHeight", editor.height()) // 为了兼容Zepto + this.resize() + this.registerKeyMaps() + $(window).resize(function() { + _this.resize() + }) + this.bindScrollEvent().bindChangeEvent() + if (!recreate) { + $.proxy(settings.onload, this)() + } + this.state.loaded = true + return this + }, + + /** + * 设置编辑器的宽度 + * Set editor width + * + * @param {Number|String} width 编辑器宽度值 + * @returns {editormd} 返回editormd的实例对象 + */ + + width: function(width) { + this.editor.css("width", typeof width === "number" ? width + "px" : width) + this.resize() + return this + }, + + /** + * 设置编辑器的高度 + * Set editor height + * + * @param {Number|String} height 编辑器高度值 + * @returns {editormd} 返回editormd的实例对象 + */ + + height: function(height) { + this.editor.css( + "height", + typeof height === "number" ? height + "px" : height + ) + this.resize() + return this + }, + + /** + * 调整编辑器的尺寸和布局 + * Resize editor layout + * + * @param {Number|String} [width=null] 编辑器宽度值 + * @param {Number|String} [height=null] 编辑器高度值 + * @returns {editormd} 返回editormd的实例对象 + */ + + resize: function(width, height) { + width = width || null + height = height || null + var state = this.state + var editor = this.editor + var preview = this.preview + var toolbar = this.toolbar + var settings = this.settings + var codeMirror = this.codeMirror + if (width) { + editor.css("width", typeof width === "number" ? width + "px" : width) + } + if (settings.autoHeight && !state.fullscreen && !state.preview) { + editor.css({ + height: "auto", + minHeight: settings.minHeight + }) + codeMirror.css({ + height: "auto", + minHeight: settings.minHeight, + maxHeight: settings.maxHeight + }) + } else { + if (height) { + editor.css( + "height", + typeof height === "number" ? height + "px" : height + ) + } + if (state.fullscreen) { + editor.height($(window).height()) + } + if (settings.toolbar && !settings.readOnly) { + codeMirror + .css("margin-top", toolbar.height() + 1) + .height(editor.height() - toolbar.height()) + } else { + codeMirror.css("margin-top", 0).height(editor.height()) + } + } + if (settings.watch) { + codeMirror.width(editor.width() / 2) + preview.width(!state.preview ? editor.width() / 2 : editor.width()) + this.previewContainer.css( + "padding", + settings.autoHeight ? "20px 20px 50px 40px" : "20px" + ) + if (settings.toolbar && !settings.readOnly) { + preview.css("top", toolbar.height() + 1) + } else { + preview.css("top", 0) + } + if (settings.autoHeight && !state.fullscreen && !state.preview) { + preview.height("") + } else { + var previewHeight = + settings.toolbar && !settings.readOnly + ? editor.height() - toolbar.height() + : editor.height() + preview.height(previewHeight) + } + } else { + codeMirror.width(editor.width()) + preview.hide() + } + if (state.loaded) { + $.proxy(settings.onresize, this)() + } + return this + }, + + /** + * 解析和保存Markdown代码 + * Parse & Saving Markdown source code + * + * @returns {editormd} 返回editormd的实例对象 + */ + + save: function() { + if (timer === null) { + return this + } + var _this = this + var state = this.state + var settings = this.settings + var cm = this.cm + var cmValue = cm.getValue() + var previewContainer = this.previewContainer + if (settings.mode !== "gfm" && settings.mode !== "markdown") { + this.markdownTextarea.val(cmValue) + return this + } + var marked = editormd.$marked + var markdownToC = (this.markdownToC = []) + var rendererOptions = (this.markedRendererOptions = { + toc: settings.toc, + tocm: settings.tocm, + tocStartLevel: settings.tocStartLevel, + pageBreak: settings.pageBreak, + taskList: settings.taskList, + emoji: settings.emoji, + tex: settings.tex, + atLink: settings.atLink, // for @link + emailLink: settings.emailLink, // for mail address auto link + flowChart: settings.flowChart, + sequenceDiagram: settings.sequenceDiagram, + previewCodeHighlight: settings.previewCodeHighlight + }) + var markedOptions = (this.markedOptions = { + renderer: editormd.markedRenderer(markdownToC, rendererOptions), + gfm: true, + tables: true, + breaks: true, + pedantic: false, + sanitize: settings.htmlDecode ? false : true, // 关闭忽略HTML标签,即开启识别HTML标签,默认为false + smartLists: true, + smartypants: true + }) + marked.setOptions(markedOptions) + var newMarkdownDoc = editormd.$marked(cmValue, markedOptions) + newMarkdownDoc = editormd.filterHTMLTags( + newMarkdownDoc, + settings.htmlDecode + ) + this.markdownTextarea.text(cmValue) + cm.save() + if (settings.saveHTMLToTextarea) { + this.htmlTextarea.text(newMarkdownDoc) + } + if (settings.watch || (!settings.watch && state.preview)) { + previewContainer.html(newMarkdownDoc) + this.previewCodeHighlight() + if (settings.toc) { + var tocContainer = + settings.tocContainer === "" + ? previewContainer + : $(settings.tocContainer) + var tocMenu = tocContainer.find("." + this.classPrefix + "toc-menu") + tocContainer.attr( + "previewContainer", + settings.tocContainer === "" ? "true" : "false" + ) + if (settings.tocContainer !== "" && tocMenu.length > 0) { + tocMenu.remove() + } + editormd.markdownToCRenderer( + markdownToC, + tocContainer, + settings.tocDropdown, + settings.tocStartLevel + ) + if ( + settings.tocDropdown || + tocContainer.find("." + this.classPrefix + "toc-menu").length > 0 + ) { + editormd.tocDropdownMenu( + tocContainer, + settings.tocTitle !== "" ? settings.tocTitle : this.lang.tocTitle + ) + } + if (settings.tocContainer !== "") { + previewContainer.find(".markdown-toc").css("border", "none") + } + } + if (settings.tex) { + if (!editormd.kaTeXLoaded && settings.autoLoadModules) { + editormd.loadKaTeX(function() { + editormd.$katex = katex + editormd.kaTeXLoaded = true + _this.katexRender() + }) + } else { + editormd.$katex = katex + this.katexRender() + } + } + if (settings.flowChart || settings.sequenceDiagram) { + flowchartTimer = setTimeout(function() { + clearTimeout(flowchartTimer) + _this.flowChartAndSequenceDiagramRender() + flowchartTimer = null + }, 10) + } + if (state.loaded) { + $.proxy(settings.onchange, this)() + } + } + return this + }, + + /** + * 聚焦光标位置 + * Focusing the cursor position + * + * @returns {editormd} 返回editormd的实例对象 + */ + + focus: function() { + this.cm.focus() + return this + }, + + /** + * 设置光标的位置 + * Set cursor position + * + * @param {Object} cursor 要设置的光标位置键值对象,例:{line:1, ch:0} + * @returns {editormd} 返回editormd的实例对象 + */ + + setCursor: function(cursor) { + this.cm.setCursor(cursor) + return this + }, + + /** + * 获取当前光标的位置 + * Get the current position of the cursor + * + * @returns {Cursor} 返回一个光标Cursor对象 + */ + + getCursor: function() { + return this.cm.getCursor() + }, + + /** + * 设置光标选中的范围 + * Set cursor selected ranges + * + * @param {Object} from 开始位置的光标键值对象,例:{line:1, ch:0} + * @param {Object} to 结束位置的光标键值对象,例:{line:1, ch:0} + * @returns {editormd} 返回editormd的实例对象 + */ + + setSelection: function(from, to) { + this.cm.setSelection(from, to) + return this + }, + + /** + * 获取光标选中的文本 + * Get the texts from cursor selected + * + * @returns {String} 返回选中文本的字符串形式 + */ + + getSelection: function() { + return this.cm.getSelection() + }, + + /** + * 设置光标选中的文本范围 + * Set the cursor selection ranges + * + * @param {Array} ranges cursor selection ranges array + * @returns {Array} return this + */ + + setSelections: function(ranges) { + this.cm.setSelections(ranges) + return this + }, + + /** + * 获取光标选中的文本范围 + * Get the cursor selection ranges + * + * @returns {Array} return selection ranges array + */ + + getSelections: function() { + return this.cm.getSelections() + }, + + /** + * 替换当前光标选中的文本或在当前光标处插入新字符 + * Replace the text at the current cursor selected or insert a new character at the current cursor position + * + * @param {String} value 要插入的字符值 + * @returns {editormd} 返回editormd的实例对象 + */ + + replaceSelection: function(value) { + this.cm.replaceSelection(value) + return this + }, + + /** + * 在当前光标处插入新字符 + * Insert a new character at the current cursor position + * + * 同replaceSelection()方法 + * With the replaceSelection() method + * + * @param {String} value 要插入的字符值 + * @returns {editormd} 返回editormd的实例对象 + */ + + insertValue: function(value) { + this.replaceSelection(value) + return this + }, + + /** + * 追加markdown + * append Markdown to editor + * + * @param {String} md 要追加的markdown源文档 + * @returns {editormd} 返回editormd的实例对象 + */ + + appendMarkdown: function(md) { + var settings = this.settings + var cm = this.cm + cm.setValue(cm.getValue() + md) + return this + }, + + /** + * 设置和传入编辑器的markdown源文档 + * Set Markdown source document + * + * @param {String} md 要传入的markdown源文档 + * @returns {editormd} 返回editormd的实例对象 + */ + + setMarkdown: function(md) { + this.cm.setValue(md || this.settings.markdown) + return this + }, + + /** + * 获取编辑器的markdown源文档 + * Set Editor.md markdown/CodeMirror value + * + * @returns {editormd} 返回editormd的实例对象 + */ + + getMarkdown: function() { + return this.cm.getValue() + }, + + /** + * 获取编辑器的源文档 + * Get CodeMirror value + * + * @returns {editormd} 返回editormd的实例对象 + */ + + getValue: function() { + return this.cm.getValue() + }, + + /** + * 设置编辑器的源文档 + * Set CodeMirror value + * + * @param {String} value set code/value/string/text + * @returns {editormd} 返回editormd的实例对象 + */ + + setValue: function(value) { + this.cm.setValue(value) + return this + }, + + /** + * 清空编辑器 + * Empty CodeMirror editor container + * + * @returns {editormd} 返回editormd的实例对象 + */ + + clear: function() { + this.cm.setValue("") + return this + }, + + /** + * 获取解析后存放在Textarea的HTML源码 + * Get parsed html code from Textarea + * + * @returns {String} 返回HTML源码 + */ + + getHTML: function() { + if (!this.settings.saveHTMLToTextarea) { + alert("Error: settings.saveHTMLToTextarea == false") + return false + } + return this.htmlTextarea.val() + }, + + /** + * getHTML()的别名 + * getHTML (alias) + * + * @returns {String} Return html code 返回HTML源码 + */ + + getTextareaSavedHTML: function() { + return this.getHTML() + }, + + /** + * 获取预览窗口的HTML源码 + * Get html from preview container + * + * @returns {editormd} 返回editormd的实例对象 + */ + + getPreviewedHTML: function() { + if (!this.settings.watch) { + alert("Error: settings.watch == false") + return false + } + return this.previewContainer.html() + }, + + /** + * 开启实时预览 + * Enable real-time watching + * + * @returns {editormd} 返回editormd的实例对象 + */ + + watch: function(callback) { + var settings = this.settings + if ($.inArray(settings.mode, ["gfm", "markdown"]) < 0) { + return this + } + this.state.watching = settings.watch = true + this.preview.show() + if (this.toolbar) { + var watchIcon = settings.toolbarIconsClass.watch + var unWatchIcon = settings.toolbarIconsClass.unwatch + var icon = this.toolbar.find(".fa[name=watch]") + icon.parent().attr("title", settings.lang.toolbar.watch) + icon.removeClass(unWatchIcon).addClass(watchIcon) + } + this.codeMirror + .css("border-right", "1px solid #ddd") + .width(this.editor.width() / 2) + timer = 0 + this.save().resize() + if (!settings.onwatch) { + settings.onwatch = callback || function() {} + } + $.proxy(settings.onwatch, this)() + return this + }, + + /** + * 关闭实时预览 + * Disable real-time watching + * + * @returns {editormd} 返回editormd的实例对象 + */ + + unwatch: function(callback) { + var settings = this.settings + this.state.watching = settings.watch = false + this.preview.hide() + if (this.toolbar) { + var watchIcon = settings.toolbarIconsClass.watch + var unWatchIcon = settings.toolbarIconsClass.unwatch + var icon = this.toolbar.find(".fa[name=watch]") + icon.parent().attr("title", settings.lang.toolbar.unwatch) + icon.removeClass(watchIcon).addClass(unWatchIcon) + } + this.codeMirror.css("border-right", "none").width(this.editor.width()) + this.resize() + if (!settings.onunwatch) { + settings.onunwatch = callback || function() {} + } + $.proxy(settings.onunwatch, this)() + return this + }, + + /** + * 显示编辑器 + * Show editor + * + * @param {Function} [callback=function()] 回调函数 + * @returns {editormd} 返回editormd的实例对象 + */ + + show: function(callback) { + callback = callback || function() {} + var _this = this + this.editor.show(0, function() { + $.proxy(callback, _this)() + }) + return this + }, + + /** + * 隐藏编辑器 + * Hide editor + * + * @param {Function} [callback=function()] 回调函数 + * @returns {editormd} 返回editormd的实例对象 + */ + + hide: function(callback) { + callback = callback || function() {} + var _this = this + this.editor.hide(0, function() { + $.proxy(callback, _this)() + }) + return this + }, + + /** + * 隐藏编辑器部分,只预览HTML + * Enter preview html state + * + * @returns {editormd} 返回editormd的实例对象 + */ + + previewing: function() { + var _this = this + var editor = this.editor + var preview = this.preview + var toolbar = this.toolbar + var settings = this.settings + var codeMirror = this.codeMirror + var previewContainer = this.previewContainer + if ($.inArray(settings.mode, ["gfm", "markdown"]) < 0) { + return this + } + if (settings.toolbar && toolbar) { + toolbar.toggle() + toolbar.find(".fa[name=preview]").toggleClass("active") + } + codeMirror.toggle() + var escHandle = function(event) { + if (event.shiftKey && event.keyCode === 27) { + _this.previewed() + } + } + if (codeMirror.css("display") === "none") { + // 为了兼容Zepto,而不使用codeMirror.is(":hidden") + this.state.preview = true + if (this.state.fullscreen) { + preview.css("background", "#fff") + } + editor + .find("." + this.classPrefix + "preview-close-btn") + .show() + .bind(editormd.mouseOrTouch("click", "touchend"), function() { + _this.previewed() + }) + if (!settings.watch) { + this.save() + } else { + previewContainer.css("padding", "") + } + previewContainer.addClass(this.classPrefix + "preview-active") + preview.show().css({ + position: "", + top: 0, + width: editor.width(), + height: + settings.autoHeight && !this.state.fullscreen + ? "auto" + : editor.height() + }) + if (this.state.loaded) { + $.proxy(settings.onpreviewing, this)() + } + $(window).bind("keyup", escHandle) + } else { + $(window).unbind("keyup", escHandle) + this.previewed() + } + }, + + /** + * 显示编辑器部分,退出只预览HTML + * Exit preview html state + * + * @returns {editormd} 返回editormd的实例对象 + */ + + previewed: function() { + var editor = this.editor + var preview = this.preview + var toolbar = this.toolbar + var settings = this.settings + var previewContainer = this.previewContainer + var previewCloseBtn = editor.find( + "." + this.classPrefix + "preview-close-btn" + ) + this.state.preview = false + this.codeMirror.show() + if (settings.toolbar) { + toolbar.show() + } + preview[settings.watch ? "show" : "hide"]() + previewCloseBtn.hide().unbind(editormd.mouseOrTouch("click", "touchend")) + previewContainer.removeClass(this.classPrefix + "preview-active") + if (settings.watch) { + previewContainer.css("padding", "20px") + } + preview.css({ + background: null, + position: "absolute", + width: editor.width() / 2, + height: + settings.autoHeight && !this.state.fullscreen + ? "auto" + : editor.height() - toolbar.height(), + top: settings.toolbar ? toolbar.height() : 0 + }) + if (this.state.loaded) { + $.proxy(settings.onpreviewed, this)() + } + return this + }, + + /** + * 编辑器全屏显示 + * Fullscreen show + * + * @returns {editormd} 返回editormd的实例对象 + */ + + fullscreen: function() { + var _this = this + var state = this.state + var editor = this.editor + var preview = this.preview + var toolbar = this.toolbar + var settings = this.settings + var fullscreenClass = this.classPrefix + "fullscreen" + if (toolbar) { + toolbar + .find(".fa[name=fullscreen]") + .parent() + .toggleClass("active") + } + var escHandle = function(event) { + if (!event.shiftKey && event.keyCode === 27) { + if (state.fullscreen) { + _this.fullscreenExit() + } + } + } + if (!editor.hasClass(fullscreenClass)) { + state.fullscreen = true + $("html,body").css("overflow", "hidden") + editor + .css({ + width: $(window).width(), + height: $(window).height() + }) + .addClass(fullscreenClass) + this.resize() + $.proxy(settings.onfullscreen, this)() + $(window).bind("keyup", escHandle) + } else { + $(window).unbind("keyup", escHandle) + this.fullscreenExit() + } + return this + }, + + /** + * 编辑器退出全屏显示 + * Exit fullscreen state + * + * @returns {editormd} 返回editormd的实例对象 + */ + + fullscreenExit: function() { + var editor = this.editor + var settings = this.settings + var toolbar = this.toolbar + var fullscreenClass = this.classPrefix + "fullscreen" + this.state.fullscreen = false + if (toolbar) { + toolbar + .find(".fa[name=fullscreen]") + .parent() + .removeClass("active") + } + $("html,body").css("overflow", "") + editor + .css({ + width: editor.data("oldWidth"), + height: editor.data("oldHeight") + }) + .removeClass(fullscreenClass) + this.resize() + $.proxy(settings.onfullscreenExit, this)() + return this + }, + + /** + * 加载并执行插件 + * Load and execute the plugin + * + * @param {String} name plugin name / function name + * @param {String} path plugin load path + * @returns {editormd} 返回editormd的实例对象 + */ + + executePlugin: function(name, path) { + var _this = this + var cm = this.cm + var settings = this.settings + path = settings.pluginPath + path + if (typeof define === "function") { + if (typeof this[name] === "undefined") { + alert( + "Error: " + + name + + " plugin is not found, you are not load this plugin." + ) + return this + } + this[name](cm) + return this + } + if ($.inArray(path, editormd.loadFiles.plugin) < 0) { + editormd.loadPlugin(path, function() { + editormd.loadPlugins[name] = _this[name] + _this[name](cm) + }) + } else { + $.proxy(editormd.loadPlugins[name], this)(cm) + } + return this + }, + + /** + * 搜索替换 + * Search & replace + * + * @param {String} command CodeMirror serach commands, "find, fintNext, fintPrev, clearSearch, replace, replaceAll" + * @returns {editormd} return this + */ + + search: function(command) { + var settings = this.settings + if (!settings.searchReplace) { + alert("Error: settings.searchReplace == false") + return this + } + if (!settings.readOnly) { + this.cm.execCommand(command || "find") + } + return this + }, + searchReplace: function() { + this.search("replace") + return this + }, + searchReplaceAll: function() { + this.search("replaceAll") + return this + } + } + + editormd.fn.init.prototype = editormd.fn + + /** + * 锁屏 + * lock screen when dialog opening + * + * @returns {void} + */ + + editormd.dialogLockScreen = function() { + var settings = this.settings || { dialogLockScreen: true } + if (settings.dialogLockScreen) { + $("html,body").css("overflow", "hidden") + this.resize() + } + } + + /** + * 显示透明背景层 + * Display mask layer when dialog opening + * + * @param {Object} dialog dialog jQuery object + * @returns {void} + */ + + editormd.dialogShowMask = function(dialog) { + var editor = this.editor + var settings = this.settings || { dialogShowMask: true } + dialog.css({ + top: ($(window).height() - dialog.height()) / 2 + "px", + left: ($(window).width() - dialog.width()) / 2 + "px" + }) + if (settings.dialogShowMask) { + editor + .children("." + this.classPrefix + "mask") + .css("z-index", parseInt(dialog.css("z-index")) - 1) + .show() + } + } + + editormd.toolbarHandlers = { + undo: function() { + this.cm.undo() + }, + redo: function() { + this.cm.redo() + }, + bold: function() { + var cm = this.cm + var cursor = cm.getCursor() + var selection = cm.getSelection() + cm.replaceSelection("**" + selection + "**") + if (selection === "") { + cm.setCursor(cursor.line, cursor.ch + 2) + } + }, + del: function() { + var cm = this.cm + var cursor = cm.getCursor() + var selection = cm.getSelection() + cm.replaceSelection("~~" + selection + "~~") + if (selection === "") { + cm.setCursor(cursor.line, cursor.ch + 2) + } + }, + italic: function() { + var cm = this.cm + var cursor = cm.getCursor() + var selection = cm.getSelection() + cm.replaceSelection("*" + selection + "*") + if (selection === "") { + cm.setCursor(cursor.line, cursor.ch + 1) + } + }, + quote: function() { + var cm = this.cm + var cursor = cm.getCursor() + var selection = cm.getSelection() + if (cursor.ch !== 0) { + cm.setCursor(cursor.line, 0) + cm.replaceSelection("> " + selection) + cm.setCursor(cursor.line, cursor.ch + 2) + } else { + cm.replaceSelection("> " + selection) + } + //cm.replaceSelection("> " + selection); + //cm.setCursor(cursor.line, (selection === "") ? cursor.ch + 2 : cursor.ch + selection.length + 2); + }, + ucfirst: function() { + var cm = this.cm + var selection = cm.getSelection() + var selections = cm.listSelections() + cm.replaceSelection(editormd.firstUpperCase(selection)) + cm.setSelections(selections) + }, + ucwords: function() { + var cm = this.cm + var selection = cm.getSelection() + var selections = cm.listSelections() + cm.replaceSelection(editormd.wordsFirstUpperCase(selection)) + cm.setSelections(selections) + }, + uppercase: function() { + var cm = this.cm + var selection = cm.getSelection() + var selections = cm.listSelections() + cm.replaceSelection(selection.toUpperCase()) + cm.setSelections(selections) + }, + lowercase: function() { + var cm = this.cm + var cursor = cm.getCursor() + var selection = cm.getSelection() + var selections = cm.listSelections() + cm.replaceSelection(selection.toLowerCase()) + cm.setSelections(selections) + }, + h1: function() { + var cm = this.cm + var cursor = cm.getCursor() + var selection = cm.getSelection() + if (cursor.ch !== 0) { + cm.setCursor(cursor.line, 0) + cm.replaceSelection("# " + selection) + cm.setCursor(cursor.line, cursor.ch + 2) + } else { + cm.replaceSelection("# " + selection) + } + }, + h2: function() { + var cm = this.cm + var cursor = cm.getCursor() + var selection = cm.getSelection() + if (cursor.ch !== 0) { + cm.setCursor(cursor.line, 0) + cm.replaceSelection("## " + selection) + cm.setCursor(cursor.line, cursor.ch + 3) + } else { + cm.replaceSelection("## " + selection) + } + }, + h3: function() { + var cm = this.cm + var cursor = cm.getCursor() + var selection = cm.getSelection() + if (cursor.ch !== 0) { + cm.setCursor(cursor.line, 0) + cm.replaceSelection("### " + selection) + cm.setCursor(cursor.line, cursor.ch + 4) + } else { + cm.replaceSelection("### " + selection) + } + }, + h4: function() { + var cm = this.cm + var cursor = cm.getCursor() + var selection = cm.getSelection() + if (cursor.ch !== 0) { + cm.setCursor(cursor.line, 0) + cm.replaceSelection("#### " + selection) + cm.setCursor(cursor.line, cursor.ch + 5) + } else { + cm.replaceSelection("#### " + selection) + } + }, + h5: function() { + var cm = this.cm + var cursor = cm.getCursor() + var selection = cm.getSelection() + if (cursor.ch !== 0) { + cm.setCursor(cursor.line, 0) + cm.replaceSelection("##### " + selection) + cm.setCursor(cursor.line, cursor.ch + 6) + } else { + cm.replaceSelection("##### " + selection) + } + }, + h6: function() { + var cm = this.cm + var cursor = cm.getCursor() + var selection = cm.getSelection() + if (cursor.ch !== 0) { + cm.setCursor(cursor.line, 0) + cm.replaceSelection("###### " + selection) + cm.setCursor(cursor.line, cursor.ch + 7) + } else { + cm.replaceSelection("###### " + selection) + } + }, + "list-ul": function() { + var cm = this.cm + var cursor = cm.getCursor() + var selection = cm.getSelection() + if (selection === "") { + cm.replaceSelection("- " + selection) + } else { + var selectionText = selection.split("\n") + for (var i = 0, len = selectionText.length; i < len; i++) { + selectionText[i] = + selectionText[i] === "" ? "" : "- " + selectionText[i] + } + cm.replaceSelection(selectionText.join("\n")) + } + }, + "list-ol": function() { + var cm = this.cm + var cursor = cm.getCursor() + var selection = cm.getSelection() + if (selection === "") { + cm.replaceSelection("1. " + selection) + } else { + var selectionText = selection.split("\n") + for (var i = 0, len = selectionText.length; i < len; i++) { + selectionText[i] = + selectionText[i] === "" ? "" : i + 1 + ". " + selectionText[i] + } + cm.replaceSelection(selectionText.join("\n")) + } + }, + hr: function() { + var cm = this.cm + var cursor = cm.getCursor() + var selection = cm.getSelection() + cm.replaceSelection( + (cursor.ch !== 0 ? "\n\n" : "\n") + "------------\n\n" + ) + }, + tex: function() { + if (!this.settings.tex) { + alert("settings.tex === false") + return this + } + var cm = this.cm + var cursor = cm.getCursor() + var selection = cm.getSelection() + cm.replaceSelection("$$" + selection + "$$") + if (selection === "") { + cm.setCursor(cursor.line, cursor.ch + 2) + } + }, + link: function() { + this.executePlugin("linkDialog", "link-dialog/link-dialog") + }, + "reference-link": function() { + this.executePlugin( + "referenceLinkDialog", + "reference-link-dialog/reference-link-dialog" + ) + }, + pagebreak: function() { + if (!this.settings.pageBreak) { + alert("settings.pageBreak === false") + return this + } + var cm = this.cm + var selection = cm.getSelection() + cm.replaceSelection("\r\n[========]\r\n") + }, + image: function() { + this.executePlugin("imageDialog", "image-dialog/image-dialog") + }, + code: function() { + var cm = this.cm + var cursor = cm.getCursor() + var selection = cm.getSelection() + cm.replaceSelection("`" + selection + "`") + if (selection === "") { + cm.setCursor(cursor.line, cursor.ch + 1) + } + }, + "code-block": function() { + this.executePlugin( + "codeBlockDialog", + "code-block-dialog/code-block-dialog" + ) + }, + "preformatted-text": function() { + this.executePlugin( + "preformattedTextDialog", + "preformatted-text-dialog/preformatted-text-dialog" + ) + }, + table: function() { + this.executePlugin("tableDialog", "table-dialog/table-dialog") + }, + datetime: function() { + var cm = this.cm + var selection = cm.getSelection() + var date = new Date() + var langName = this.settings.lang.name + var datefmt = + editormd.dateFormat() + + " " + + editormd.dateFormat( + langName === "zh-cn" || langName === "zh-tw" + ? "cn-week-day" + : "week-day" + ) + cm.replaceSelection(datefmt) + }, + emoji: function() { + this.executePlugin("emojiDialog", "emoji-dialog/emoji-dialog") + }, + "html-entities": function() { + this.executePlugin( + "htmlEntitiesDialog", + "html-entities-dialog/html-entities-dialog" + ) + }, + "goto-line": function() { + this.executePlugin("gotoLineDialog", "goto-line-dialog/goto-line-dialog") + }, + watch: function() { + this[this.settings.watch ? "unwatch" : "watch"]() + }, + preview: function() { + this.previewing() + }, + fullscreen: function() { + this.fullscreen() + }, + clear: function() { + this.clear() + }, + search: function() { + this.search() + }, + help: function() { + this.executePlugin("helpDialog", "help-dialog/help-dialog") + }, + info: function() { + this.showInfoDialog() + } + } + + var isMac = navigator.platform.toUpperCase().indexOf("MAC") >= 0 + var key = isMac ? "Cmd" : "Ctrl" + editormd.keyMaps = { + [key + "-1"]: "h1", + [key + "-2"]: "h2", + [key + "-3"]: "h3", + [key + "-4"]: "h4", + [key + "-5"]: "h5", + [key + "-6"]: "h6", + [key + "-B"]: "bold", // if this is string == editormd.toolbarHandlers.xxxx + [key + "-D"]: "datetime", + [key + "Ctrl-E"]: function() { + // emoji + var cm = this.cm + var cursor = cm.getCursor() + var selection = cm.getSelection() + if (!this.settings.emoji) { + alert("Error: settings.emoji == false") + return + } + cm.replaceSelection(":" + selection + ":") + if (selection === "") { + cm.setCursor(cursor.line, cursor.ch + 1) + } + }, + [key + "-Alt-G"]: "goto-line", + [key + "-H"]: "hr", + [key + "-I"]: "italic", + [key + "-K"]: "code", + "Ctrl-L": function() { + var cm = this.cm + var cursor = cm.getCursor() + var selection = cm.getSelection() + var title = selection === "" ? "" : ' "' + selection + '"' + cm.replaceSelection("[" + selection + "](" + title + ")") + if (selection === "") { + cm.setCursor(cursor.line, cursor.ch + 1) + } + }, + [key + "-U"]: "list-ul", + "Shift-Ctrl-A": function() { + var cm = this.cm + var cursor = cm.getCursor() + var selection = cm.getSelection() + if (!this.settings.atLink) { + alert("Error: settings.atLink == false") + return + } + cm.replaceSelection("@" + selection) + if (selection === "") { + cm.setCursor(cursor.line, cursor.ch + 1) + } + }, + ["Shift" + key + "-C"]: "code", + ["Shift" + key + "Q"]: "quote", + ["Shift" + key + "S"]: "del", + ["Shift" + key + "K"]: "tex", // KaTeX + "Shift-Alt-C": function() { + var cm = this.cm + var cursor = cm.getCursor() + var selection = cm.getSelection() + cm.replaceSelection(["```", selection, "```"].join("\n")) + if (selection === "") { + cm.setCursor(cursor.line, cursor.ch + 3) + } + }, + ["Shift-" + key + "-Alt-C"]: "code-block", + ["Shift-" + key + "-H"]: "html-entities", + "Shift-Alt-H": "help", + ["Shift-" + key + "-E"]: "emoji", + ["Shift-" + key + "-U"]: "uppercase", + "Shift-Alt-U": "ucwords", + ["Shift-" + key + "-Alt-U"]: "ucfirst", + "Shift-Alt-L": "lowercase", + ["Shift-" + key + "-I"]: function() { + var cm = this.cm + var cursor = cm.getCursor() + var selection = cm.getSelection() + var title = selection === "" ? "" : ' "' + selection + '"' + cm.replaceSelection("![" + selection + "](" + title + ")") + if (selection === "") { + cm.setCursor(cursor.line, cursor.ch + 4) + } + }, + ["Shift-" + key + "-Alt-I"]: "image", + ["Shift-" + key + "-L"]: "link", + ["Shift-" + key + "-O"]: "list-ol", + ["Shift-" + key + "-P"]: "preformatted-text", + ["Shift-" + key + "-T"]: "table", + "Shift-Alt-P": "pagebreak", + F9: "watch", + F10: "preview", + F11: "fullscreen" + } + + /** + * 清除字符串两边的空格 + * Clear the space of strings both sides. + * + * @param {String} str string + * @returns {String} trimed string + */ + + var trim = function(str) { + return !String.prototype.trim + ? str.replace(/^[\s\uFEFF\xA0]+|[\s\uFEFF\xA0]+$/g, "") + : str.trim() + } + + editormd.trim = trim + + /** + * 所有单词首字母大写 + * Words first to uppercase + * + * @param {String} str string + * @returns {String} string + */ + + var ucwords = function(str) { + return str.toLowerCase().replace(/\b(\w)|\s(\w)/g, function($1) { + return $1.toUpperCase() + }) + } + + editormd.ucwords = editormd.wordsFirstUpperCase = ucwords + + /** + * 字符串首字母大写 + * Only string first char to uppercase + * + * @param {String} str string + * @returns {String} string + */ + + var firstUpperCase = function(str) { + return str.toLowerCase().replace(/\b(\w)/, function($1) { + return $1.toUpperCase() + }) + } + + var ucfirst = firstUpperCase + editormd.firstUpperCase = editormd.ucfirst = firstUpperCase + editormd.urls = { + atLinkBase: "https://github.com/" + } + editormd.regexs = { + atLink: /@(\w+)/g, + email: /(\w+)@(\w+)\.(\w+)\.?(\w+)?/g, + emailLink: /(mailto:)?([\w\.\_]+)@(\w+)\.(\w+)\.?(\w+)?/g, + emoji: /:([\w\+-]+):/g, + emojiDatetime: /(\d{1,2}:\d{1,2}:\d{1,2})/g, + twemoji: /:(tw-([\w]+)-?(\w+)?):/g, + fontAwesome: /:(fa-([\w]+)(-(\w+)){0,}):/g, + editormdLogo: /:(editormd-logo-?(\w+)?):/g, + pageBreak: /^\[[=]{8,}\]$/ + } + // Emoji graphics files url path + editormd.emoji = { + path: "http://www.emoji-cheat-sheet.com/graphics/emojis/", + ext: ".png" + } + // Twitter Emoji (Twemoji) graphics files url path + editormd.twemoji = { + path: "http://twemoji.maxcdn.com/36x36/", + ext: ".png" + } + + /** + * 自定义marked的解析器 + * Custom Marked renderer rules + * + * @param {Array} markdownToC 传入用于接收TOC的数组 + * @returns {Renderer} markedRenderer 返回marked的Renderer自定义对象 + */ + + editormd.markedRenderer = function(markdownToC, options) { + var defaults = { + toc: true, // Table of contents + tocm: false, + tocStartLevel: 1, // Said from H1 to create ToC + pageBreak: true, + atLink: true, // for @link + emailLink: true, // for mail address auto link + taskList: false, // Enable Github Flavored Markdown task lists + emoji: false, // :emoji: , Support Twemoji, fontAwesome, Editor.md logo emojis. + tex: false, // TeX(LaTeX), based on KaTeX + flowChart: false, // flowChart.js only support IE9+ + sequenceDiagram: false // sequenceDiagram.js only support IE9+ + } + var settings = $.extend(defaults, options || {}) + var marked = editormd.$marked + var markedRenderer = new marked.Renderer() + markdownToC = markdownToC || [] + + var regexs = editormd.regexs + var atLinkReg = regexs.atLink + var emojiReg = regexs.emoji + var emailReg = regexs.email + var emailLinkReg = regexs.emailLink + var twemojiReg = regexs.twemoji + var faIconReg = regexs.fontAwesome + var editormdLogoReg = regexs.editormdLogo + var pageBreakReg = regexs.pageBreak + + markedRenderer.emoji = function(text) { + text = text.replace(editormd.regexs.emojiDatetime, function($1) { + return $1.replace(/:/g, ":") + }) + + var matchs = text.match(emojiReg) + + if (!matchs || !settings.emoji) { + return text + } + + for (var i = 0, len = matchs.length; i < len; i++) { + if (matchs[i] === ":+1:") { + matchs[i] = ":\\+1:" + } + + text = text.replace(new RegExp(matchs[i]), function($1, $2) { + var faMatchs = $1.match(faIconReg) + var name = $1.replace(/:/g, "") + if (faMatchs) { + for (var fa = 0, len1 = faMatchs.length; fa < len1; fa++) { + var faName = faMatchs[fa].replace(/:/g, "") + return ( + '' + ) + } + } else { + var emdlogoMathcs = $1.match(editormdLogoReg) + var twemojiMatchs = $1.match(twemojiReg) + if (emdlogoMathcs) { + for (var x = 0, len2 = emdlogoMathcs.length; x < len2; x++) { + var logoName = emdlogoMathcs[x].replace(/:/g, "") + return ( + '' + ) + } + } else if (twemojiMatchs) { + for (var t = 0, len3 = twemojiMatchs.length; t < len3; t++) { + var twe = twemojiMatchs[t].replace(/:/g, "").replace("tw-", "") + return ( + 'twemoji-' +
+                  twe +
+                  '' + ) + } + } else { + var src = name === "+1" ? "plus1" : name + src = src === "black_large_square" ? "black_square" : src + src = src === "moon" ? "waxing_gibbous_moon" : src + return ( + ':' +
+                name +
+                ':' + ) + } + } + }) + } + return text + } + + markedRenderer.atLink = function(text) { + if (atLinkReg.test(text)) { + if (settings.atLink) { + text = text.replace(emailReg, function($1, $2, $3, $4) { + return $1.replace(/@/g, "_#_@_#_") + }) + text = text + .replace(atLinkReg, function($1, $2) { + return ( + '' + + $1 + + "" + ) + }) + .replace(/_#_@_#_/g, "@") + } + if (settings.emailLink) { + text = text.replace(emailLinkReg, function($1, $2, $3, $4, $5) { + return !$2 && + $.inArray($5, "jpg|jpeg|png|gif|webp|ico|icon|pdf".split("|")) < 0 + ? '' + $1 + "" + : $1 + }) + } + return text + } + return text + } + + markedRenderer.link = function(href, title, text) { + if (this.options.sanitize) { + try { + var prot = decodeURIComponent(unescape(href)) + .replace(/[^\w:]/g, "") + .toLowerCase() + if (prot.indexOf("javascript:") === 0) { + return "" + } + } catch (e) { + return "" + } + } + var out = '' + text.replace(/@/g, "@") + "" + } + if (title) { + out += ' title="' + title + '"' + } + out += ">" + text + "" + return out + } + + markedRenderer.heading = function(text, level, raw) { + var linkText = text + var hasLinkReg = /\s*]*)\>(.*)<\/a\>\s*/ + var getLinkTextReg = /\s*]+)\>([^\>]*)<\/a\>\s*/g + if (hasLinkReg.test(text)) { + var tempText = [] + text = text.split(/]+)\>([^\>]*)<\/a\>/) + for (var i = 0, len = text.length; i < len; i++) { + tempText.push(text[i].replace(/\s*href\=\"(.*)\"\s*/g, "")) + } + text = tempText.join(" ") + } + text = trim(text) + var escapedText = text.toLowerCase().replace(/[^\w]+/g, "-") + var toc = { + text: text, + level: level, + slug: escapedText + } + var isChinese = /^[\u4e00-\u9fa5]+$/.test(text) + var id = isChinese + ? escape(text).replace(/\%/g, "") + : text.toLowerCase().replace(/[^\w]+/g, "-") + markdownToC.push(toc) + var headingHTML = + "' + headingHTML += '' + headingHTML += '' + headingHTML += hasLinkReg + ? this.atLink(this.emoji(linkText)) + : this.atLink(this.emoji(text)) + headingHTML += "" + return headingHTML + } + markedRenderer.pageBreak = function(text) { + if (pageBreakReg.test(text) && settings.pageBreak) { + text = + '
                      ' + } + return text + } + markedRenderer.paragraph = function(text) { + var isTeXInline = /\$\$(.*)\$\$/g.test(text) + var isTeXLine = /^\$\$(.*)\$\$$/.test(text) + var isTeXAddClass = isTeXLine + ? ' class="' + editormd.classNames.tex + '"' + : "" + var isToC = settings.tocm + ? /^(\[TOC\]|\[TOCM\])$/.test(text) + : /^\[TOC\]$/.test(text) + var isToCMenu = /^\[TOCM\]$/.test(text) + if (!isTeXLine && isTeXInline) { + text = text.replace(/(\$\$([^\$]*)\$\$)+/g, function($1, $2) { + return ( + '' + + $2.replace(/\$/g, "") + + "" + ) + }) + } else { + text = isTeXLine ? text.replace(/\$/g, "") : text + } + var tocHTML = + '
                      ' + text + "
                      " + return isToC + ? isToCMenu + ? '
                      ' + tocHTML + "

                      " + : tocHTML + : pageBreakReg.test(text) + ? this.pageBreak(text) + : "" + + this.atLink(this.emoji(text)) + + "

                      \n" + } + markedRenderer.code = function(code, lang, escaped) { + if (lang === "seq" || lang === "sequence") { + return '
                      ' + code + "
                      " + } else if (lang === "flow") { + return '
                      ' + code + "
                      " + } else if (lang === "math" || lang === "latex" || lang === "katex") { + return '

                      ' + code + "

                      " + } else { + return marked.Renderer.prototype.code.apply(this, arguments) + } + } + markedRenderer.tablecell = function(content, flags) { + var type = flags.header ? "th" : "td" + var tag = flags.align + ? "<" + type + ' style="text-align:' + flags.align + '">' + : "<" + type + ">" + return tag + this.atLink(this.emoji(content)) + "\n" + } + markedRenderer.listitem = function(text) { + if (settings.taskList && /^\s*\[[x\s]\]\s*/.test(text)) { + text = text + .replace( + /^\s*\[\s\]\s*/, + ' ' + ) + .replace( + /^\s*\[x\]\s*/, + ' ' + ) + return ( + '
                    • ' + + this.atLink(this.emoji(text)) + + "
                    • " + ) + } else { + return "
                    • " + this.atLink(this.emoji(text)) + "
                    • " + } + } + return markedRenderer + } + + /** + * + * 生成TOC(Table of Contents) + * Creating ToC (Table of Contents) + * + * @param {Array} toc 从marked获取的TOC数组列表 + * @param {Element} container 插入TOC的容器元素 + * @param {Integer} startLevel Hx 起始层级 + * @returns {Object} tocContainer 返回ToC列表容器层的jQuery对象元素 + */ + + editormd.markdownToCRenderer = function( + toc, + container, + tocDropdown, + startLevel + ) { + var html = "" + var lastLevelArr = [0] + // var lastLevel = 0 + var classPrefix = this.classPrefix + startLevel = startLevel || 1 + for (var i = 0, len = toc.length; i < len; i++) { + var text = toc[i].text + var level = toc[i].level + var stageLen = lastLevelArr.length + var lastLevel = lastLevelArr[stageLen - 1] + var goBack = 0 + if (level < startLevel) { + continue + } + if (level > lastLevel) { + html += "" + lastLevelArr.push(level) + } else if (level < lastLevel) { + for (var j = stageLen - 1; j >= 0; j--) { + if (level <= lastLevelArr[j]) { + goBack ++ + lastLevelArr.pop() + } else { + lastLevelArr.push(level) + break + } + } + html += new Array(goBack + 1).join("") + } else { + html += "" + } + html += + '
                    • ' + + text + + "
                        " + } + + var tocContainer = container.find(".markdown-toc") + + if ( + tocContainer.length < 1 && + container.attr("previewContainer") === "false" + ) { + var tocHTML = + '
                        ' + tocHTML = tocDropdown + ? '
                        ' + tocHTML + "
                        " + : tocHTML + container.html(tocHTML) + tocContainer = container.find(".markdown-toc") + } + if (tocDropdown) { + tocContainer.wrap('

                        ') + } + tocContainer + .html('
                          ') + .children(".markdown-toc-list") + .html(html.replace(/\r?\n?<\/ul\>/g, "")) + return tocContainer + } + + /** + * + * 生成TOC下拉菜单 + * Creating ToC dropdown menu + * + * @param {Object} container 插入TOC的容器jQuery对象元素 + * @param {String} tocTitle ToC title + * @returns {Object} return toc-menu object + */ + + editormd.tocDropdownMenu = function(container, tocTitle) { + tocTitle = tocTitle || "Table of Contents" + var zindex = 400 + var tocMenus = container.find("." + this.classPrefix + "toc-menu") + tocMenus.each(function() { + var $this = $(this) + var toc = $this.children(".markdown-toc") + var icon = '' + var btn = + '' + + icon + + tocTitle + + "" + var menu = toc.children("ul") + var list = menu.find("li") + toc.append(btn) + list.first().before("
                        • " + tocTitle + " " + icon + "

                        • ") + $this + .mouseover(function() { + menu.show() + list.each(function() { + var li = $(this) + var ul = li.children("ul") + if (ul.html() === "") { + ul.remove() + } + if (ul.length > 0 && ul.html() !== "") { + var firstA = li.children("a").first() + if (firstA.children(".fa").length < 1) { + firstA.append( + $(icon).css({ float: "right", paddingTop: "4px" }) + ) + } + } + li + .mouseover(function() { + ul.css("z-index", zindex).show() + zindex += 1 + }) + .mouseleave(function() { + ul.hide() + }) + }) + }) + .mouseleave(function() { + menu.hide() + }) + }) + + return tocMenus + } + + /** + * 简单地过滤指定的HTML标签 + * Filter custom html tags + * + * @param {String} html 要过滤HTML + * @param {String} filters 要过滤的标签 + * @returns {String} html 返回过滤的HTML + */ + + editormd.filterHTMLTags = function(html, filters) { + if (typeof html !== "string") { + html = new String(html) + } + if (typeof filters !== "string") { + return html + } + var expression = filters.split("|") + var filterTags = expression[0].split(",") + var attrs = expression[1] + for (var i = 0, len = filterTags.length; i < len; i++) { + var tag = filterTags[i] + html = html.replace( + new RegExp("]*)>([^>]*)", "igm"), + "" + ) + } + //return html; + if (typeof attrs !== "undefined") { + var htmlTagRegex = /<(\w+)\s*([^\>]*)\>([^\>]*)<\/(\w+)\>/gi + if (attrs === "*") { + html = html.replace(htmlTagRegex, function($1, $2, $3, $4, $5) { + return "<" + $2 + ">" + $4 + "" + }) + } else if (attrs === "on*") { + html = html.replace(htmlTagRegex, function($1, $2, $3, $4, $5) { + var el = $("<" + $2 + ">" + $4 + "") + var _attrs = $($1)[0].attributes + var $attrs = {} + $.each(_attrs, function(i, e) { + if (e.nodeName !== '"') { + $attrs[e.nodeName] = e.nodeValue + // fixed will cause xss problem + if (e.nodeName === "href" && e.nodeValue.toLowerCase().indexOf('javascript:') >= 0) { + $attrs[e.nodeName] = 'javascript:;'; + } + } + }) + $.each($attrs, function(i) { + if (i.indexOf("on") === 0) { + delete $attrs[i] + } + }) + el.attr($attrs) + var text = typeof el[1] !== "undefined" ? $(el[1]).text() : "" + // FIXED 使用 on* 过滤标签的属性,图片加链接的语法会出错的问题 + if ($2 && !isNaN($2) && $2 !== $5) { + text += "<" + $2 + ">" + $4 + "" + } else if ($2 && isNaN($2) && $2 !== $5) { + text += "" + return el[0].outerHTML + text + } else { + return el[0].outerHTML + text + } + }) + } else { + html = html.replace(htmlTagRegex, function($1, $2, $3, $4) { + var filterAttrs = attrs.split(",") + var el = $($1) + el.html($4) + $.each(filterAttrs, function(i) { + el.attr(filterAttrs[i], null) + }) + return el[0].outerHTML + }) + } + } + return html + } + + /** + * 将Markdown文档解析为HTML用于前台显示 + * Parse Markdown to HTML for Font-end preview. + * + * @param {String} id 用于显示HTML的对象ID + * @param {Object} [options={}] 配置选项,可选 + * @returns {Object} div 返回jQuery对象元素 + */ + + editormd.markdownToHTML = function(id, options) { + var defaults = { + gfm: true, + toc: true, + tocm: false, + tocStartLevel: 1, + tocTitle: "目录", + tocDropdown: false, + tocContainer: "", + markdown: "", + markdownSourceCode: false, + htmlDecode: false, + autoLoadKaTeX: true, + pageBreak: true, + atLink: true, // for @link + emailLink: true, // for mail address auto link + tex: false, + taskList: false, // Github Flavored Markdown task lists + emoji: false, + flowChart: false, + sequenceDiagram: false, + previewCodeHighlight: true + } + + editormd.$marked = marked + + var div = $("#" + id) + var settings = (div.settings = $.extend(true, defaults, options || {})) + var saveTo = div.find("textarea") + if (saveTo.length < 1) { + div.append("") + saveTo = div.find("textarea") + } + var markdownDoc = + settings.markdown === "" ? saveTo.val() : settings.markdown + var markdownToC = [] + var rendererOptions = { + toc: settings.toc, + tocm: settings.tocm, + tocStartLevel: settings.tocStartLevel, + taskList: settings.taskList, + emoji: settings.emoji, + tex: settings.tex, + pageBreak: settings.pageBreak, + atLink: settings.atLink, // for @link + emailLink: settings.emailLink, // for mail address auto link + flowChart: settings.flowChart, + sequenceDiagram: settings.sequenceDiagram, + previewCodeHighlight: settings.previewCodeHighlight + } + var markedOptions = { + renderer: editormd.markedRenderer(markdownToC, rendererOptions), + gfm: settings.gfm, + tables: true, + breaks: true, + pedantic: false, + sanitize: settings.htmlDecode ? false : true, // 是否忽略HTML标签,即是否开启HTML标签解析,为了安全性,默认不开启 + smartLists: true, + smartypants: true + } + markdownDoc = new String(markdownDoc) + var markdownParsed = marked(markdownDoc, markedOptions) + markdownParsed = editormd.filterHTMLTags( + markdownParsed, + settings.htmlDecode + ) + if (settings.markdownSourceCode) { + saveTo.text(markdownDoc) + } else { + saveTo.remove() + } + div + .addClass("markdown-body " + this.classPrefix + "html-preview") + .append(markdownParsed) + var tocContainer = + settings.tocContainer !== "" ? $(settings.tocContainer) : div + if (settings.tocContainer !== "") { + tocContainer.attr("previewContainer", false) + } + if (settings.toc) { + div.tocContainer = this.markdownToCRenderer( + markdownToC, + tocContainer, + settings.tocDropdown, + settings.tocStartLevel + ) + if ( + settings.tocDropdown || + div.find("." + this.classPrefix + "toc-menu").length > 0 + ) { + this.tocDropdownMenu(div, settings.tocTitle) + } + if (settings.tocContainer !== "") { + div.find(".editormd-toc-menu, .editormd-markdown-toc").remove() + } + } + if (settings.previewCodeHighlight) { + div.find("pre").addClass("prettyprint linenums") + prettyPrint() + } + if (!editormd.isIE8) { + if (settings.flowChart) { + div.find(".flowchart").flowChart() + } + if (settings.sequenceDiagram) { + div.find(".sequence-diagram").sequenceDiagram({ theme: "simple" }) + } + } + if (settings.tex) { + var katexHandle = function() { + div.find("." + editormd.classNames.tex).each(function() { + var tex = $(this) + katex.render( + tex + .html() + .replace(/</g, "<") + .replace(/>/g, ">"), + tex[0] + ) + tex.find(".katex").css("font-size", "1.6em") + }) + } + if (settings.autoLoadKaTeX && !editormd.$katex && !editormd.kaTeXLoaded) { + this.loadKaTeX(function() { + editormd.$katex = katex + editormd.kaTeXLoaded = true + katexHandle() + }) + } else { + katexHandle() + } + } + div.getMarkdown = function() { + return saveTo.val() + } + return div + } + + // Editor.md themes, change toolbar themes etc. + // added @1.5.0 + editormd.themes = ["default", "dark"] + // Preview area themes + // added @1.5.0 + editormd.previewThemes = ["default", "dark"] + // CodeMirror / editor area themes + // @1.5.0 rename -> editorThemes, old version -> themes + editormd.editorThemes = [ + "default", + "3024-day", + "3024-night", + "ambiance", + "ambiance-mobile", + "base16-dark", + "base16-light", + "blackboard", + "cobalt", + "eclipse", + "elegant", + "erlang-dark", + "lesser-dark", + "mbo", + "mdn-like", + "midnight", + "monokai", + "neat", + "neo", + "night", + "paraiso-dark", + "paraiso-light", + "pastel-on-dark", + "rubyblue", + "solarized", + "the-matrix", + "tomorrow-night-eighties", + "twilight", + "vibrant-ink", + "xq-dark", + "xq-light" + ] + editormd.loadPlugins = {} + editormd.loadFiles = { + js: [], + css: [], + plugin: [] + } + /** + * 动态加载Editor.md插件,但不立即执行 + * Load editor.md plugins + * + * @param {String} fileName 插件文件路径 + * @param {Function} [callback=function()] 加载成功后执行的回调函数 + * @param {String} [into="head"] 嵌入页面的位置 + */ + + editormd.loadPlugin = function(fileName, callback, into) { + callback = callback || function() {} + this.loadScript( + fileName, + function() { + editormd.loadFiles.plugin.push(fileName) + callback() + }, + into + ) + } + + /** + * 动态加载CSS文件的方法 + * Load css file method + * + * @param {String} fileName CSS文件名 + * @param {Function} [callback=function()] 加载成功后执行的回调函数 + * @param {String} [into="head"] 嵌入页面的位置 + */ + + editormd.loadCSS = function(fileName, callback, into) { + into = into || "head" + callback = callback || function() {} + + var css = document.createElement("link") + css.type = "text/css" + css.rel = "stylesheet" + css.onload = css.onreadystatechange = function() { + editormd.loadFiles.css.push(fileName) + callback() + } + css.href = fileName + ".css" + if (into === "head") { + document.getElementsByTagName("head")[0].appendChild(css) + } else { + document.body.appendChild(css) + } + } + + editormd.isIE = navigator.appName === "Microsoft Internet Explorer" + editormd.isIE8 = editormd.isIE && navigator.appVersion.match(/8./i) === "8." + + /** + * 动态加载JS文件的方法 + * Load javascript file method + * + * @param {String} fileName JS文件名 + * @param {Function} [callback=function()] 加载成功后执行的回调函数 + * @param {String} [into="head"] 嵌入页面的位置 + */ + + editormd.loadScript = function(fileName, callback, into) { + into = into || "head" + callback = callback || function() {} + var script = null + script = document.createElement("script") + script.id = fileName.replace(/[\./]+/g, "-") + script.type = "text/javascript" + script.src = fileName + ".js" + if (editormd.isIE8) { + script.onreadystatechange = function() { + if (script.readyState) { + if ( + script.readyState === "loaded" || + script.readyState === "complete" + ) { + script.onreadystatechange = null + editormd.loadFiles.js.push(fileName) + callback() + } + } + } + } else { + script.onload = function() { + editormd.loadFiles.js.push(fileName) + callback() + } + } + if (into === "head") { + document.getElementsByTagName("head")[0].appendChild(script) + } else { + document.body.appendChild(script) + } + } + + // 使用国外的CDN,加载速度有时会很慢,或者自定义URL + // You can custom KaTeX load url. + editormd.katexURL = { + css: "//cdnjs.cloudflare.com/ajax/libs/KaTeX/0.3.0/katex.min", + js: "//cdnjs.cloudflare.com/ajax/libs/KaTeX/0.3.0/katex.min" + } + editormd.kaTeXLoaded = false + /** + * 加载KaTeX文件 + * load KaTeX files + * + * @param {Function} [callback=function()] 加载成功后执行的回调函数 + */ + editormd.loadKaTeX = function(callback) { + editormd.loadCSS(editormd.katexURL.css, function() { + editormd.loadScript(editormd.katexURL.js, callback || function() {}) + }) + } + + /** + * 锁屏 + * lock screen + * + * @param {Boolean} lock Boolean 布尔值,是否锁屏 + * @returns {void} + */ + + editormd.lockScreen = function(lock) { + $("html,body").css("overflow", lock ? "hidden" : "") + } + + /** + * 动态创建对话框 + * Creating custom dialogs + * + * @param {Object} options 配置项键值对 Key/Value + * @returns {dialog} 返回创建的dialog的jQuery实例对象 + */ + + editormd.createDialog = function(options) { + var defaults = { + name: "", + width: 420, + height: 240, + title: "", + drag: true, + closed: true, + content: "", + mask: true, + maskStyle: { + backgroundColor: "#fff", + opacity: 0.1 + }, + lockScreen: true, + footer: true, + buttons: false + } + + options = $.extend(true, defaults, options) + + var $this = this + var editor = this.editor + var classPrefix = editormd.classPrefix + var guid = new Date().getTime() + var dialogName = + options.name === "" ? classPrefix + "dialog-" + guid : options.name + var mouseOrTouch = editormd.mouseOrTouch + var html = '
                          ' + if (options.title !== "") { + html += + '
                          " + html += + '' + + options.title + + "" + html += "
                          " + } + if (options.closed) { + html += + '
                          ' + } + html += + '
                          ' + options.content + if (options.footer || typeof options.footer === "string") { + html += + '" + } + html += "
                          " + html += + '
                          ' + html += + '
                          ' + html += "
                          " + editor.append(html) + var dialog = editor.find("." + dialogName) + dialog.lockScreen = function(lock) { + if (options.lockScreen) { + $("html,body").css("overflow", lock ? "hidden" : "") + $this.resize() + } + return dialog + } + dialog.showMask = function() { + if (options.mask) { + editor + .find("." + classPrefix + "mask") + .css(options.maskStyle) + .css("z-index", editormd.dialogZindex - 1) + .show() + } + return dialog + } + dialog.hideMask = function() { + if (options.mask) { + editor.find("." + classPrefix + "mask").hide() + } + return dialog + } + dialog.loading = function(show) { + var loading = dialog.find("." + classPrefix + "dialog-mask") + loading[show ? "show" : "hide"]() + return dialog + } + dialog.lockScreen(true).showMask() + dialog.show().css({ + zIndex: editormd.dialogZindex, + border: editormd.isIE8 ? "1px solid #ddd" : "", + width: + typeof options.width === "number" + ? options.width + "px" + : options.width, + height: + typeof options.height === "number" + ? options.height + "px" + : options.height + }) + var dialogPosition = function() { + dialog.css({ + top: ($(window).height() - dialog.height()) / 2 + "px", + left: ($(window).width() - dialog.width()) / 2 + "px" + }) + } + dialogPosition() + $(window).resize(dialogPosition) + dialog + .children("." + classPrefix + "dialog-close") + .bind(mouseOrTouch("click", "touchend"), function() { + dialog + .hide() + .lockScreen(false) + .hideMask() + }) + if (typeof options.buttons === "object") { + var footer = (dialog.footer = dialog.find( + "." + classPrefix + "dialog-footer" + )) + for (var key in options.buttons) { + var btn = options.buttons[key] + var btnClassName = classPrefix + key + "-btn" + footer.append( + '" + ) + btn[1] = $.proxy(btn[1], dialog) + footer + .children("." + btnClassName) + .bind(mouseOrTouch("click", "touchend"), btn[1]) + } + } + if (options.title !== "" && options.drag) { + var posX, posY + var dialogHeader = dialog.children("." + classPrefix + "dialog-header") + if (!options.mask) { + dialogHeader.bind(mouseOrTouch("click", "touchend"), function() { + editormd.dialogZindex += 2 + dialog.css("z-index", editormd.dialogZindex) + }) + } + var userCanSelect = function(obj) { + obj.removeClass(classPrefix + "user-unselect").off("selectstart") + } + var userUnselect = function(obj) { + obj + .addClass(classPrefix + "user-unselect") + .on("selectstart", function(event) { + // selectstart for IE + return false + }) + } + var moveAction = function(e) { + e = e || window.event //IE + var left, + top, + nowLeft = parseInt(dialog[0].style.left), + nowTop = parseInt(dialog[0].style.top) + if (nowLeft >= 0) { + if (nowLeft + dialog.width() <= $(window).width()) { + left = e.clientX - posX + } else { + left = $(window).width() - dialog.width() + document.onmousemove = null + } + } else { + left = 0 + document.onmousemove = null + } + if (nowTop >= 0) { + top = e.clientY - posY + } else { + top = 0 + document.onmousemove = null + } + document.onselectstart = function() { + return false + } + userUnselect($("body")) + userUnselect(dialog) + dialog[0].style.left = left + "px" + dialog[0].style.top = top + "px" + } + dialogHeader.mousedown(function(e) { + e = e || window.event //IE + posX = e.clientX - parseInt(dialog[0].style.left) + posY = e.clientY - parseInt(dialog[0].style.top) + document.onmousemove = moveAction + }) + document.onmouseup = function() { + userCanSelect($("body")) + userCanSelect(dialog) + document.onselectstart = null + document.onmousemove = null + } + dialogHeader.touchDraggable = function() { + var offset = null + var start = function(e) { + var orig = e.originalEvent + var pos = $(this) + .parent() + .position() + offset = { + x: orig.changedTouches[0].pageX - pos.left, + y: orig.changedTouches[0].pageY - pos.top + } + } + var move = function(e) { + e.preventDefault() + var orig = e.originalEvent + $(this) + .parent() + .css({ + top: orig.changedTouches[0].pageY - offset.y, + left: orig.changedTouches[0].pageX - offset.x + }) + } + this.bind("touchstart", start).bind("touchmove", move) + } + dialogHeader.touchDraggable() + } + editormd.dialogZindex += 2 + return dialog + } + + /** + * 鼠标和触摸事件的判断/选择方法 + * MouseEvent or TouchEvent type switch + * + * @param {String} [mouseEventType="click"] 供选择的鼠标事件 + * @param {String} [touchEventType="touchend"] 供选择的触摸事件 + * @returns {String} EventType 返回事件类型名称 + */ + + editormd.mouseOrTouch = function(mouseEventType, touchEventType) { + mouseEventType = mouseEventType || "click" + touchEventType = touchEventType || "touchend" + var eventType = mouseEventType + try { + document.createEvent("TouchEvent") + eventType = touchEventType + } catch (e) {} + return eventType + } + + /** + * 日期时间的格式化方法 + * Datetime format method + * + * @param {String} [format=""] 日期时间的格式,类似PHP的格式 + * @returns {String} datefmt 返回格式化后的日期时间字符串 + */ + + editormd.dateFormat = function(format) { + format = format || "" + var addZero = function(d) { + return d < 10 ? "0" + d : d + } + var date = new Date() + var year = date.getFullYear() + var year2 = year.toString().slice(2, 4) + var month = addZero(date.getMonth() + 1) + var day = addZero(date.getDate()) + var weekDay = date.getDay() + var hour = addZero(date.getHours()) + var min = addZero(date.getMinutes()) + var second = addZero(date.getSeconds()) + var ms = addZero(date.getMilliseconds()) + var datefmt = "" + var ymd = year2 + "-" + month + "-" + day + var fymd = year + "-" + month + "-" + day + var hms = hour + ":" + min + ":" + second + switch (format) { + case "UNIX Time": + datefmt = date.getTime() + break + case "UTC": + datefmt = date.toUTCString() + break + case "yy": + datefmt = year2 + break + case "year": + case "yyyy": + datefmt = year + break + case "month": + case "mm": + datefmt = month + break + case "cn-week-day": + case "cn-wd": + var cnWeekDays = ["日", "一", "二", "三", "四", "五", "六"] + datefmt = "星期" + cnWeekDays[weekDay] + break + case "week-day": + case "wd": + var weekDays = [ + "Sunday", + "Monday", + "Tuesday", + "Wednesday", + "Thursday", + "Friday", + "Saturday" + ] + datefmt = weekDays[weekDay] + break + case "day": + case "dd": + datefmt = day + break + case "hour": + case "hh": + datefmt = hour + break + case "min": + case "ii": + datefmt = min + break + case "second": + case "ss": + datefmt = second + break + case "ms": + datefmt = ms + break + case "yy-mm-dd": + datefmt = ymd + break + case "yyyy-mm-dd": + datefmt = fymd + break + case "yyyy-mm-dd h:i:s ms": + case "full + ms": + datefmt = fymd + " " + hms + " " + ms + break + case "full": + case "yyyy-mm-dd h:i:s": + break + default: + datefmt = fymd + " " + hms + break + } + return datefmt + } + return editormd +}) diff --git a/system/admin/editor.md/tests/bootstrap-test.html b/system/admin/editor.md/tests/bootstrap-test.html new file mode 100644 index 00000000..e5e0f687 --- /dev/null +++ b/system/admin/editor.md/tests/bootstrap-test.html @@ -0,0 +1,63 @@ + + + + + Bootstrap 兼容测试 - Editor.md tests + + + + + + + + + + +
                          +
                          +

                          Bootstrap 兼容测试

                          +
                          +
                          + +
                          +
                          + + + + + + + + \ No newline at end of file diff --git a/system/admin/editor.md/tests/codemirror-searchbox-test.html b/system/admin/editor.md/tests/codemirror-searchbox-test.html new file mode 100644 index 00000000..60102e6f --- /dev/null +++ b/system/admin/editor.md/tests/codemirror-searchbox-test.html @@ -0,0 +1,109 @@ + + + + CodeMirror searchbox Test + + + + + + +
                          + +
                          + + + + + + + + + + \ No newline at end of file diff --git a/system/admin/editor.md/tests/codemirror-test.html b/system/admin/editor.md/tests/codemirror-test.html new file mode 100644 index 00000000..382993df --- /dev/null +++ b/system/admin/editor.md/tests/codemirror-test.html @@ -0,0 +1,75 @@ + + + + CodeMirror Test + + + + + + +
                          + +
                          + + + + + + + + + + + \ No newline at end of file diff --git a/system/admin/editor.md/tests/css/bootstrap-theme.min.css b/system/admin/editor.md/tests/css/bootstrap-theme.min.css new file mode 100644 index 00000000..cefa3d1a --- /dev/null +++ b/system/admin/editor.md/tests/css/bootstrap-theme.min.css @@ -0,0 +1,5 @@ +/*! + * Bootstrap v3.3.4 (http://getbootstrap.com) + * Copyright 2011-2015 Twitter, Inc. + * Licensed under MIT (https://github.com/twbs/bootstrap/blob/master/LICENSE) + */.btn-danger,.btn-default,.btn-info,.btn-primary,.btn-success,.btn-warning{text-shadow:0 -1px 0 rgba(0,0,0,.2);-webkit-box-shadow:inset 0 1px 0 rgba(255,255,255,.15),0 1px 1px rgba(0,0,0,.075);box-shadow:inset 0 1px 0 rgba(255,255,255,.15),0 1px 1px rgba(0,0,0,.075)}.btn-danger.active,.btn-danger:active,.btn-default.active,.btn-default:active,.btn-info.active,.btn-info:active,.btn-primary.active,.btn-primary:active,.btn-success.active,.btn-success:active,.btn-warning.active,.btn-warning:active{-webkit-box-shadow:inset 0 3px 5px rgba(0,0,0,.125);box-shadow:inset 0 3px 5px rgba(0,0,0,.125)}.btn-danger .badge,.btn-default .badge,.btn-info .badge,.btn-primary .badge,.btn-success .badge,.btn-warning .badge{text-shadow:none}.btn.active,.btn:active{background-image:none}.btn-default{text-shadow:0 1px 0 #fff;background-image:-webkit-linear-gradient(top,#fff 0,#e0e0e0 100%);background-image:-o-linear-gradient(top,#fff 0,#e0e0e0 100%);background-image:-webkit-gradient(linear,left top,left bottom,from(#fff),to(#e0e0e0));background-image:linear-gradient(to bottom,#fff 0,#e0e0e0 100%);filter:progid:DXImageTransform.Microsoft.gradient(startColorstr='#ffffffff', endColorstr='#ffe0e0e0', GradientType=0);filter:progid:DXImageTransform.Microsoft.gradient(enabled=false);background-repeat:repeat-x;border-color:#dbdbdb;border-color:#ccc}.btn-default:focus,.btn-default:hover{background-color:#e0e0e0;background-position:0 -15px}.btn-default.active,.btn-default:active{background-color:#e0e0e0;border-color:#dbdbdb}.btn-default.disabled,.btn-default:disabled,.btn-default[disabled]{background-color:#e0e0e0;background-image:none}.btn-primary{background-image:-webkit-linear-gradient(top,#337ab7 0,#265a88 100%);background-image:-o-linear-gradient(top,#337ab7 0,#265a88 100%);background-image:-webkit-gradient(linear,left top,left bottom,from(#337ab7),to(#265a88));background-image:linear-gradient(to bottom,#337ab7 0,#265a88 100%);filter:progid:DXImageTransform.Microsoft.gradient(startColorstr='#ff337ab7', endColorstr='#ff265a88', GradientType=0);filter:progid:DXImageTransform.Microsoft.gradient(enabled=false);background-repeat:repeat-x;border-color:#245580}.btn-primary:focus,.btn-primary:hover{background-color:#265a88;background-position:0 -15px}.btn-primary.active,.btn-primary:active{background-color:#265a88;border-color:#245580}.btn-primary.disabled,.btn-primary:disabled,.btn-primary[disabled]{background-color:#265a88;background-image:none}.btn-success{background-image:-webkit-linear-gradient(top,#5cb85c 0,#419641 100%);background-image:-o-linear-gradient(top,#5cb85c 0,#419641 100%);background-image:-webkit-gradient(linear,left top,left bottom,from(#5cb85c),to(#419641));background-image:linear-gradient(to bottom,#5cb85c 0,#419641 100%);filter:progid:DXImageTransform.Microsoft.gradient(startColorstr='#ff5cb85c', endColorstr='#ff419641', GradientType=0);filter:progid:DXImageTransform.Microsoft.gradient(enabled=false);background-repeat:repeat-x;border-color:#3e8f3e}.btn-success:focus,.btn-success:hover{background-color:#419641;background-position:0 -15px}.btn-success.active,.btn-success:active{background-color:#419641;border-color:#3e8f3e}.btn-success.disabled,.btn-success:disabled,.btn-success[disabled]{background-color:#419641;background-image:none}.btn-info{background-image:-webkit-linear-gradient(top,#5bc0de 0,#2aabd2 100%);background-image:-o-linear-gradient(top,#5bc0de 0,#2aabd2 100%);background-image:-webkit-gradient(linear,left top,left bottom,from(#5bc0de),to(#2aabd2));background-image:linear-gradient(to bottom,#5bc0de 0,#2aabd2 100%);filter:progid:DXImageTransform.Microsoft.gradient(startColorstr='#ff5bc0de', endColorstr='#ff2aabd2', GradientType=0);filter:progid:DXImageTransform.Microsoft.gradient(enabled=false);background-repeat:repeat-x;border-color:#28a4c9}.btn-info:focus,.btn-info:hover{background-color:#2aabd2;background-position:0 -15px}.btn-info.active,.btn-info:active{background-color:#2aabd2;border-color:#28a4c9}.btn-info.disabled,.btn-info:disabled,.btn-info[disabled]{background-color:#2aabd2;background-image:none}.btn-warning{background-image:-webkit-linear-gradient(top,#f0ad4e 0,#eb9316 100%);background-image:-o-linear-gradient(top,#f0ad4e 0,#eb9316 100%);background-image:-webkit-gradient(linear,left top,left bottom,from(#f0ad4e),to(#eb9316));background-image:linear-gradient(to bottom,#f0ad4e 0,#eb9316 100%);filter:progid:DXImageTransform.Microsoft.gradient(startColorstr='#fff0ad4e', endColorstr='#ffeb9316', GradientType=0);filter:progid:DXImageTransform.Microsoft.gradient(enabled=false);background-repeat:repeat-x;border-color:#e38d13}.btn-warning:focus,.btn-warning:hover{background-color:#eb9316;background-position:0 -15px}.btn-warning.active,.btn-warning:active{background-color:#eb9316;border-color:#e38d13}.btn-warning.disabled,.btn-warning:disabled,.btn-warning[disabled]{background-color:#eb9316;background-image:none}.btn-danger{background-image:-webkit-linear-gradient(top,#d9534f 0,#c12e2a 100%);background-image:-o-linear-gradient(top,#d9534f 0,#c12e2a 100%);background-image:-webkit-gradient(linear,left top,left bottom,from(#d9534f),to(#c12e2a));background-image:linear-gradient(to bottom,#d9534f 0,#c12e2a 100%);filter:progid:DXImageTransform.Microsoft.gradient(startColorstr='#ffd9534f', endColorstr='#ffc12e2a', GradientType=0);filter:progid:DXImageTransform.Microsoft.gradient(enabled=false);background-repeat:repeat-x;border-color:#b92c28}.btn-danger:focus,.btn-danger:hover{background-color:#c12e2a;background-position:0 -15px}.btn-danger.active,.btn-danger:active{background-color:#c12e2a;border-color:#b92c28}.btn-danger.disabled,.btn-danger:disabled,.btn-danger[disabled]{background-color:#c12e2a;background-image:none}.img-thumbnail,.thumbnail{-webkit-box-shadow:0 1px 2px rgba(0,0,0,.075);box-shadow:0 1px 2px rgba(0,0,0,.075)}.dropdown-menu>li>a:focus,.dropdown-menu>li>a:hover{background-color:#e8e8e8;background-image:-webkit-linear-gradient(top,#f5f5f5 0,#e8e8e8 100%);background-image:-o-linear-gradient(top,#f5f5f5 0,#e8e8e8 100%);background-image:-webkit-gradient(linear,left top,left bottom,from(#f5f5f5),to(#e8e8e8));background-image:linear-gradient(to bottom,#f5f5f5 0,#e8e8e8 100%);filter:progid:DXImageTransform.Microsoft.gradient(startColorstr='#fff5f5f5', endColorstr='#ffe8e8e8', GradientType=0);background-repeat:repeat-x}.dropdown-menu>.active>a,.dropdown-menu>.active>a:focus,.dropdown-menu>.active>a:hover{background-color:#2e6da4;background-image:-webkit-linear-gradient(top,#337ab7 0,#2e6da4 100%);background-image:-o-linear-gradient(top,#337ab7 0,#2e6da4 100%);background-image:-webkit-gradient(linear,left top,left bottom,from(#337ab7),to(#2e6da4));background-image:linear-gradient(to bottom,#337ab7 0,#2e6da4 100%);filter:progid:DXImageTransform.Microsoft.gradient(startColorstr='#ff337ab7', endColorstr='#ff2e6da4', GradientType=0);background-repeat:repeat-x}.navbar-default{background-image:-webkit-linear-gradient(top,#fff 0,#f8f8f8 100%);background-image:-o-linear-gradient(top,#fff 0,#f8f8f8 100%);background-image:-webkit-gradient(linear,left top,left bottom,from(#fff),to(#f8f8f8));background-image:linear-gradient(to bottom,#fff 0,#f8f8f8 100%);filter:progid:DXImageTransform.Microsoft.gradient(startColorstr='#ffffffff', endColorstr='#fff8f8f8', GradientType=0);filter:progid:DXImageTransform.Microsoft.gradient(enabled=false);background-repeat:repeat-x;border-radius:4px;-webkit-box-shadow:inset 0 1px 0 rgba(255,255,255,.15),0 1px 5px rgba(0,0,0,.075);box-shadow:inset 0 1px 0 rgba(255,255,255,.15),0 1px 5px rgba(0,0,0,.075)}.navbar-default .navbar-nav>.active>a,.navbar-default .navbar-nav>.open>a{background-image:-webkit-linear-gradient(top,#dbdbdb 0,#e2e2e2 100%);background-image:-o-linear-gradient(top,#dbdbdb 0,#e2e2e2 100%);background-image:-webkit-gradient(linear,left top,left bottom,from(#dbdbdb),to(#e2e2e2));background-image:linear-gradient(to bottom,#dbdbdb 0,#e2e2e2 100%);filter:progid:DXImageTransform.Microsoft.gradient(startColorstr='#ffdbdbdb', endColorstr='#ffe2e2e2', GradientType=0);background-repeat:repeat-x;-webkit-box-shadow:inset 0 3px 9px rgba(0,0,0,.075);box-shadow:inset 0 3px 9px rgba(0,0,0,.075)}.navbar-brand,.navbar-nav>li>a{text-shadow:0 1px 0 rgba(255,255,255,.25)}.navbar-inverse{background-image:-webkit-linear-gradient(top,#3c3c3c 0,#222 100%);background-image:-o-linear-gradient(top,#3c3c3c 0,#222 100%);background-image:-webkit-gradient(linear,left top,left bottom,from(#3c3c3c),to(#222));background-image:linear-gradient(to bottom,#3c3c3c 0,#222 100%);filter:progid:DXImageTransform.Microsoft.gradient(startColorstr='#ff3c3c3c', endColorstr='#ff222222', GradientType=0);filter:progid:DXImageTransform.Microsoft.gradient(enabled=false);background-repeat:repeat-x}.navbar-inverse .navbar-nav>.active>a,.navbar-inverse .navbar-nav>.open>a{background-image:-webkit-linear-gradient(top,#080808 0,#0f0f0f 100%);background-image:-o-linear-gradient(top,#080808 0,#0f0f0f 100%);background-image:-webkit-gradient(linear,left top,left bottom,from(#080808),to(#0f0f0f));background-image:linear-gradient(to bottom,#080808 0,#0f0f0f 100%);filter:progid:DXImageTransform.Microsoft.gradient(startColorstr='#ff080808', endColorstr='#ff0f0f0f', GradientType=0);background-repeat:repeat-x;-webkit-box-shadow:inset 0 3px 9px rgba(0,0,0,.25);box-shadow:inset 0 3px 9px rgba(0,0,0,.25)}.navbar-inverse .navbar-brand,.navbar-inverse .navbar-nav>li>a{text-shadow:0 -1px 0 rgba(0,0,0,.25)}.navbar-fixed-bottom,.navbar-fixed-top,.navbar-static-top{border-radius:0}@media (max-width:767px){.navbar .navbar-nav .open .dropdown-menu>.active>a,.navbar .navbar-nav .open .dropdown-menu>.active>a:focus,.navbar .navbar-nav .open .dropdown-menu>.active>a:hover{color:#fff;background-image:-webkit-linear-gradient(top,#337ab7 0,#2e6da4 100%);background-image:-o-linear-gradient(top,#337ab7 0,#2e6da4 100%);background-image:-webkit-gradient(linear,left top,left bottom,from(#337ab7),to(#2e6da4));background-image:linear-gradient(to bottom,#337ab7 0,#2e6da4 100%);filter:progid:DXImageTransform.Microsoft.gradient(startColorstr='#ff337ab7', endColorstr='#ff2e6da4', GradientType=0);background-repeat:repeat-x}}.alert{text-shadow:0 1px 0 rgba(255,255,255,.2);-webkit-box-shadow:inset 0 1px 0 rgba(255,255,255,.25),0 1px 2px rgba(0,0,0,.05);box-shadow:inset 0 1px 0 rgba(255,255,255,.25),0 1px 2px rgba(0,0,0,.05)}.alert-success{background-image:-webkit-linear-gradient(top,#dff0d8 0,#c8e5bc 100%);background-image:-o-linear-gradient(top,#dff0d8 0,#c8e5bc 100%);background-image:-webkit-gradient(linear,left top,left bottom,from(#dff0d8),to(#c8e5bc));background-image:linear-gradient(to bottom,#dff0d8 0,#c8e5bc 100%);filter:progid:DXImageTransform.Microsoft.gradient(startColorstr='#ffdff0d8', endColorstr='#ffc8e5bc', GradientType=0);background-repeat:repeat-x;border-color:#b2dba1}.alert-info{background-image:-webkit-linear-gradient(top,#d9edf7 0,#b9def0 100%);background-image:-o-linear-gradient(top,#d9edf7 0,#b9def0 100%);background-image:-webkit-gradient(linear,left top,left bottom,from(#d9edf7),to(#b9def0));background-image:linear-gradient(to bottom,#d9edf7 0,#b9def0 100%);filter:progid:DXImageTransform.Microsoft.gradient(startColorstr='#ffd9edf7', endColorstr='#ffb9def0', GradientType=0);background-repeat:repeat-x;border-color:#9acfea}.alert-warning{background-image:-webkit-linear-gradient(top,#fcf8e3 0,#f8efc0 100%);background-image:-o-linear-gradient(top,#fcf8e3 0,#f8efc0 100%);background-image:-webkit-gradient(linear,left top,left bottom,from(#fcf8e3),to(#f8efc0));background-image:linear-gradient(to bottom,#fcf8e3 0,#f8efc0 100%);filter:progid:DXImageTransform.Microsoft.gradient(startColorstr='#fffcf8e3', endColorstr='#fff8efc0', GradientType=0);background-repeat:repeat-x;border-color:#f5e79e}.alert-danger{background-image:-webkit-linear-gradient(top,#f2dede 0,#e7c3c3 100%);background-image:-o-linear-gradient(top,#f2dede 0,#e7c3c3 100%);background-image:-webkit-gradient(linear,left top,left bottom,from(#f2dede),to(#e7c3c3));background-image:linear-gradient(to bottom,#f2dede 0,#e7c3c3 100%);filter:progid:DXImageTransform.Microsoft.gradient(startColorstr='#fff2dede', endColorstr='#ffe7c3c3', GradientType=0);background-repeat:repeat-x;border-color:#dca7a7}.progress{background-image:-webkit-linear-gradient(top,#ebebeb 0,#f5f5f5 100%);background-image:-o-linear-gradient(top,#ebebeb 0,#f5f5f5 100%);background-image:-webkit-gradient(linear,left top,left bottom,from(#ebebeb),to(#f5f5f5));background-image:linear-gradient(to bottom,#ebebeb 0,#f5f5f5 100%);filter:progid:DXImageTransform.Microsoft.gradient(startColorstr='#ffebebeb', endColorstr='#fff5f5f5', GradientType=0);background-repeat:repeat-x}.progress-bar{background-image:-webkit-linear-gradient(top,#337ab7 0,#286090 100%);background-image:-o-linear-gradient(top,#337ab7 0,#286090 100%);background-image:-webkit-gradient(linear,left top,left bottom,from(#337ab7),to(#286090));background-image:linear-gradient(to bottom,#337ab7 0,#286090 100%);filter:progid:DXImageTransform.Microsoft.gradient(startColorstr='#ff337ab7', endColorstr='#ff286090', GradientType=0);background-repeat:repeat-x}.progress-bar-success{background-image:-webkit-linear-gradient(top,#5cb85c 0,#449d44 100%);background-image:-o-linear-gradient(top,#5cb85c 0,#449d44 100%);background-image:-webkit-gradient(linear,left top,left bottom,from(#5cb85c),to(#449d44));background-image:linear-gradient(to bottom,#5cb85c 0,#449d44 100%);filter:progid:DXImageTransform.Microsoft.gradient(startColorstr='#ff5cb85c', endColorstr='#ff449d44', GradientType=0);background-repeat:repeat-x}.progress-bar-info{background-image:-webkit-linear-gradient(top,#5bc0de 0,#31b0d5 100%);background-image:-o-linear-gradient(top,#5bc0de 0,#31b0d5 100%);background-image:-webkit-gradient(linear,left top,left bottom,from(#5bc0de),to(#31b0d5));background-image:linear-gradient(to bottom,#5bc0de 0,#31b0d5 100%);filter:progid:DXImageTransform.Microsoft.gradient(startColorstr='#ff5bc0de', endColorstr='#ff31b0d5', GradientType=0);background-repeat:repeat-x}.progress-bar-warning{background-image:-webkit-linear-gradient(top,#f0ad4e 0,#ec971f 100%);background-image:-o-linear-gradient(top,#f0ad4e 0,#ec971f 100%);background-image:-webkit-gradient(linear,left top,left bottom,from(#f0ad4e),to(#ec971f));background-image:linear-gradient(to bottom,#f0ad4e 0,#ec971f 100%);filter:progid:DXImageTransform.Microsoft.gradient(startColorstr='#fff0ad4e', endColorstr='#ffec971f', GradientType=0);background-repeat:repeat-x}.progress-bar-danger{background-image:-webkit-linear-gradient(top,#d9534f 0,#c9302c 100%);background-image:-o-linear-gradient(top,#d9534f 0,#c9302c 100%);background-image:-webkit-gradient(linear,left top,left bottom,from(#d9534f),to(#c9302c));background-image:linear-gradient(to bottom,#d9534f 0,#c9302c 100%);filter:progid:DXImageTransform.Microsoft.gradient(startColorstr='#ffd9534f', endColorstr='#ffc9302c', GradientType=0);background-repeat:repeat-x}.progress-bar-striped{background-image:-webkit-linear-gradient(45deg,rgba(255,255,255,.15) 25%,transparent 25%,transparent 50%,rgba(255,255,255,.15) 50%,rgba(255,255,255,.15) 75%,transparent 75%,transparent);background-image:-o-linear-gradient(45deg,rgba(255,255,255,.15) 25%,transparent 25%,transparent 50%,rgba(255,255,255,.15) 50%,rgba(255,255,255,.15) 75%,transparent 75%,transparent);background-image:linear-gradient(45deg,rgba(255,255,255,.15) 25%,transparent 25%,transparent 50%,rgba(255,255,255,.15) 50%,rgba(255,255,255,.15) 75%,transparent 75%,transparent)}.list-group{border-radius:4px;-webkit-box-shadow:0 1px 2px rgba(0,0,0,.075);box-shadow:0 1px 2px rgba(0,0,0,.075)}.list-group-item.active,.list-group-item.active:focus,.list-group-item.active:hover{text-shadow:0 -1px 0 #286090;background-image:-webkit-linear-gradient(top,#337ab7 0,#2b669a 100%);background-image:-o-linear-gradient(top,#337ab7 0,#2b669a 100%);background-image:-webkit-gradient(linear,left top,left bottom,from(#337ab7),to(#2b669a));background-image:linear-gradient(to bottom,#337ab7 0,#2b669a 100%);filter:progid:DXImageTransform.Microsoft.gradient(startColorstr='#ff337ab7', endColorstr='#ff2b669a', GradientType=0);background-repeat:repeat-x;border-color:#2b669a}.list-group-item.active .badge,.list-group-item.active:focus .badge,.list-group-item.active:hover .badge{text-shadow:none}.panel{-webkit-box-shadow:0 1px 2px rgba(0,0,0,.05);box-shadow:0 1px 2px rgba(0,0,0,.05)}.panel-default>.panel-heading{background-image:-webkit-linear-gradient(top,#f5f5f5 0,#e8e8e8 100%);background-image:-o-linear-gradient(top,#f5f5f5 0,#e8e8e8 100%);background-image:-webkit-gradient(linear,left top,left bottom,from(#f5f5f5),to(#e8e8e8));background-image:linear-gradient(to bottom,#f5f5f5 0,#e8e8e8 100%);filter:progid:DXImageTransform.Microsoft.gradient(startColorstr='#fff5f5f5', endColorstr='#ffe8e8e8', GradientType=0);background-repeat:repeat-x}.panel-primary>.panel-heading{background-image:-webkit-linear-gradient(top,#337ab7 0,#2e6da4 100%);background-image:-o-linear-gradient(top,#337ab7 0,#2e6da4 100%);background-image:-webkit-gradient(linear,left top,left bottom,from(#337ab7),to(#2e6da4));background-image:linear-gradient(to bottom,#337ab7 0,#2e6da4 100%);filter:progid:DXImageTransform.Microsoft.gradient(startColorstr='#ff337ab7', endColorstr='#ff2e6da4', GradientType=0);background-repeat:repeat-x}.panel-success>.panel-heading{background-image:-webkit-linear-gradient(top,#dff0d8 0,#d0e9c6 100%);background-image:-o-linear-gradient(top,#dff0d8 0,#d0e9c6 100%);background-image:-webkit-gradient(linear,left top,left bottom,from(#dff0d8),to(#d0e9c6));background-image:linear-gradient(to bottom,#dff0d8 0,#d0e9c6 100%);filter:progid:DXImageTransform.Microsoft.gradient(startColorstr='#ffdff0d8', endColorstr='#ffd0e9c6', GradientType=0);background-repeat:repeat-x}.panel-info>.panel-heading{background-image:-webkit-linear-gradient(top,#d9edf7 0,#c4e3f3 100%);background-image:-o-linear-gradient(top,#d9edf7 0,#c4e3f3 100%);background-image:-webkit-gradient(linear,left top,left bottom,from(#d9edf7),to(#c4e3f3));background-image:linear-gradient(to bottom,#d9edf7 0,#c4e3f3 100%);filter:progid:DXImageTransform.Microsoft.gradient(startColorstr='#ffd9edf7', endColorstr='#ffc4e3f3', GradientType=0);background-repeat:repeat-x}.panel-warning>.panel-heading{background-image:-webkit-linear-gradient(top,#fcf8e3 0,#faf2cc 100%);background-image:-o-linear-gradient(top,#fcf8e3 0,#faf2cc 100%);background-image:-webkit-gradient(linear,left top,left bottom,from(#fcf8e3),to(#faf2cc));background-image:linear-gradient(to bottom,#fcf8e3 0,#faf2cc 100%);filter:progid:DXImageTransform.Microsoft.gradient(startColorstr='#fffcf8e3', endColorstr='#fffaf2cc', GradientType=0);background-repeat:repeat-x}.panel-danger>.panel-heading{background-image:-webkit-linear-gradient(top,#f2dede 0,#ebcccc 100%);background-image:-o-linear-gradient(top,#f2dede 0,#ebcccc 100%);background-image:-webkit-gradient(linear,left top,left bottom,from(#f2dede),to(#ebcccc));background-image:linear-gradient(to bottom,#f2dede 0,#ebcccc 100%);filter:progid:DXImageTransform.Microsoft.gradient(startColorstr='#fff2dede', endColorstr='#ffebcccc', GradientType=0);background-repeat:repeat-x}.well{background-image:-webkit-linear-gradient(top,#e8e8e8 0,#f5f5f5 100%);background-image:-o-linear-gradient(top,#e8e8e8 0,#f5f5f5 100%);background-image:-webkit-gradient(linear,left top,left bottom,from(#e8e8e8),to(#f5f5f5));background-image:linear-gradient(to bottom,#e8e8e8 0,#f5f5f5 100%);filter:progid:DXImageTransform.Microsoft.gradient(startColorstr='#ffe8e8e8', endColorstr='#fff5f5f5', GradientType=0);background-repeat:repeat-x;border-color:#dcdcdc;-webkit-box-shadow:inset 0 1px 3px rgba(0,0,0,.05),0 1px 0 rgba(255,255,255,.1);box-shadow:inset 0 1px 3px rgba(0,0,0,.05),0 1px 0 rgba(255,255,255,.1)} \ No newline at end of file diff --git a/system/admin/editor.md/tests/css/bootstrap.min.css b/system/admin/editor.md/tests/css/bootstrap.min.css new file mode 100644 index 00000000..cd1c616a --- /dev/null +++ b/system/admin/editor.md/tests/css/bootstrap.min.css @@ -0,0 +1,5 @@ +/*! + * Bootstrap v3.3.4 (http://getbootstrap.com) + * Copyright 2011-2015 Twitter, Inc. + * Licensed under MIT (https://github.com/twbs/bootstrap/blob/master/LICENSE) + *//*! normalize.css v3.0.2 | MIT License | git.io/normalize */html{font-family:sans-serif;-webkit-text-size-adjust:100%;-ms-text-size-adjust:100%}body{margin:0}article,aside,details,figcaption,figure,footer,header,hgroup,main,menu,nav,section,summary{display:block}audio,canvas,progress,video{display:inline-block;vertical-align:baseline}audio:not([controls]){display:none;height:0}[hidden],template{display:none}a{background-color:transparent}a:active,a:hover{outline:0}abbr[title]{border-bottom:1px dotted}b,strong{font-weight:700}dfn{font-style:italic}h1{margin:.67em 0;font-size:2em}mark{color:#000;background:#ff0}small{font-size:80%}sub,sup{position:relative;font-size:75%;line-height:0;vertical-align:baseline}sup{top:-.5em}sub{bottom:-.25em}img{border:0}svg:not(:root){overflow:hidden}figure{margin:1em 40px}hr{height:0;-webkit-box-sizing:content-box;-moz-box-sizing:content-box;box-sizing:content-box}pre{overflow:auto}code,kbd,pre,samp{font-family:monospace,monospace;font-size:1em}button,input,optgroup,select,textarea{margin:0;font:inherit;color:inherit}button{overflow:visible}button,select{text-transform:none}button,html input[type=button],input[type=reset],input[type=submit]{-webkit-appearance:button;cursor:pointer}button[disabled],html input[disabled]{cursor:default}button::-moz-focus-inner,input::-moz-focus-inner{padding:0;border:0}input{line-height:normal}input[type=checkbox],input[type=radio]{-webkit-box-sizing:border-box;-moz-box-sizing:border-box;box-sizing:border-box;padding:0}input[type=number]::-webkit-inner-spin-button,input[type=number]::-webkit-outer-spin-button{height:auto}input[type=search]{-webkit-box-sizing:content-box;-moz-box-sizing:content-box;box-sizing:content-box;-webkit-appearance:textfield}input[type=search]::-webkit-search-cancel-button,input[type=search]::-webkit-search-decoration{-webkit-appearance:none}fieldset{padding:.35em .625em .75em;margin:0 2px;border:1px solid silver}legend{padding:0;border:0}textarea{overflow:auto}optgroup{font-weight:700}table{border-spacing:0;border-collapse:collapse}td,th{padding:0}/*! Source: https://github.com/h5bp/html5-boilerplate/blob/master/src/css/main.css */@media print{*,:after,:before{color:#000!important;text-shadow:none!important;background:0 0!important;-webkit-box-shadow:none!important;box-shadow:none!important}a,a:visited{text-decoration:underline}a[href]:after{content:" (" attr(href) ")"}abbr[title]:after{content:" (" attr(title) ")"}a[href^="javascript:"]:after,a[href^="#"]:after{content:""}blockquote,pre{border:1px solid #999;page-break-inside:avoid}thead{display:table-header-group}img,tr{page-break-inside:avoid}img{max-width:100%!important}h2,h3,p{orphans:3;widows:3}h2,h3{page-break-after:avoid}select{background:#fff!important}.navbar{display:none}.btn>.caret,.dropup>.btn>.caret{border-top-color:#000!important}.label{border:1px solid #000}.table{border-collapse:collapse!important}.table td,.table th{background-color:#fff!important}.table-bordered td,.table-bordered th{border:1px solid #ddd!important}}@font-face{font-family:'Glyphicons Halflings';src:url(../fonts/glyphicons-halflings-regular.eot);src:url(../fonts/glyphicons-halflings-regular.eot?#iefix) format('embedded-opentype'),url(../fonts/glyphicons-halflings-regular.woff2) format('woff2'),url(../fonts/glyphicons-halflings-regular.woff) format('woff'),url(../fonts/glyphicons-halflings-regular.ttf) format('truetype'),url(../fonts/glyphicons-halflings-regular.svg#glyphicons_halflingsregular) format('svg')}.glyphicon{position:relative;top:1px;display:inline-block;font-family:'Glyphicons Halflings';font-style:normal;font-weight:400;line-height:1;-webkit-font-smoothing:antialiased;-moz-osx-font-smoothing:grayscale}.glyphicon-asterisk:before{content:"\2a"}.glyphicon-plus:before{content:"\2b"}.glyphicon-eur:before,.glyphicon-euro:before{content:"\20ac"}.glyphicon-minus:before{content:"\2212"}.glyphicon-cloud:before{content:"\2601"}.glyphicon-envelope:before{content:"\2709"}.glyphicon-pencil:before{content:"\270f"}.glyphicon-glass:before{content:"\e001"}.glyphicon-music:before{content:"\e002"}.glyphicon-search:before{content:"\e003"}.glyphicon-heart:before{content:"\e005"}.glyphicon-star:before{content:"\e006"}.glyphicon-star-empty:before{content:"\e007"}.glyphicon-user:before{content:"\e008"}.glyphicon-film:before{content:"\e009"}.glyphicon-th-large:before{content:"\e010"}.glyphicon-th:before{content:"\e011"}.glyphicon-th-list:before{content:"\e012"}.glyphicon-ok:before{content:"\e013"}.glyphicon-remove:before{content:"\e014"}.glyphicon-zoom-in:before{content:"\e015"}.glyphicon-zoom-out:before{content:"\e016"}.glyphicon-off:before{content:"\e017"}.glyphicon-signal:before{content:"\e018"}.glyphicon-cog:before{content:"\e019"}.glyphicon-trash:before{content:"\e020"}.glyphicon-home:before{content:"\e021"}.glyphicon-file:before{content:"\e022"}.glyphicon-time:before{content:"\e023"}.glyphicon-road:before{content:"\e024"}.glyphicon-download-alt:before{content:"\e025"}.glyphicon-download:before{content:"\e026"}.glyphicon-upload:before{content:"\e027"}.glyphicon-inbox:before{content:"\e028"}.glyphicon-play-circle:before{content:"\e029"}.glyphicon-repeat:before{content:"\e030"}.glyphicon-refresh:before{content:"\e031"}.glyphicon-list-alt:before{content:"\e032"}.glyphicon-lock:before{content:"\e033"}.glyphicon-flag:before{content:"\e034"}.glyphicon-headphones:before{content:"\e035"}.glyphicon-volume-off:before{content:"\e036"}.glyphicon-volume-down:before{content:"\e037"}.glyphicon-volume-up:before{content:"\e038"}.glyphicon-qrcode:before{content:"\e039"}.glyphicon-barcode:before{content:"\e040"}.glyphicon-tag:before{content:"\e041"}.glyphicon-tags:before{content:"\e042"}.glyphicon-book:before{content:"\e043"}.glyphicon-bookmark:before{content:"\e044"}.glyphicon-print:before{content:"\e045"}.glyphicon-camera:before{content:"\e046"}.glyphicon-font:before{content:"\e047"}.glyphicon-bold:before{content:"\e048"}.glyphicon-italic:before{content:"\e049"}.glyphicon-text-height:before{content:"\e050"}.glyphicon-text-width:before{content:"\e051"}.glyphicon-align-left:before{content:"\e052"}.glyphicon-align-center:before{content:"\e053"}.glyphicon-align-right:before{content:"\e054"}.glyphicon-align-justify:before{content:"\e055"}.glyphicon-list:before{content:"\e056"}.glyphicon-indent-left:before{content:"\e057"}.glyphicon-indent-right:before{content:"\e058"}.glyphicon-facetime-video:before{content:"\e059"}.glyphicon-picture:before{content:"\e060"}.glyphicon-map-marker:before{content:"\e062"}.glyphicon-adjust:before{content:"\e063"}.glyphicon-tint:before{content:"\e064"}.glyphicon-edit:before{content:"\e065"}.glyphicon-share:before{content:"\e066"}.glyphicon-check:before{content:"\e067"}.glyphicon-move:before{content:"\e068"}.glyphicon-step-backward:before{content:"\e069"}.glyphicon-fast-backward:before{content:"\e070"}.glyphicon-backward:before{content:"\e071"}.glyphicon-play:before{content:"\e072"}.glyphicon-pause:before{content:"\e073"}.glyphicon-stop:before{content:"\e074"}.glyphicon-forward:before{content:"\e075"}.glyphicon-fast-forward:before{content:"\e076"}.glyphicon-step-forward:before{content:"\e077"}.glyphicon-eject:before{content:"\e078"}.glyphicon-chevron-left:before{content:"\e079"}.glyphicon-chevron-right:before{content:"\e080"}.glyphicon-plus-sign:before{content:"\e081"}.glyphicon-minus-sign:before{content:"\e082"}.glyphicon-remove-sign:before{content:"\e083"}.glyphicon-ok-sign:before{content:"\e084"}.glyphicon-question-sign:before{content:"\e085"}.glyphicon-info-sign:before{content:"\e086"}.glyphicon-screenshot:before{content:"\e087"}.glyphicon-remove-circle:before{content:"\e088"}.glyphicon-ok-circle:before{content:"\e089"}.glyphicon-ban-circle:before{content:"\e090"}.glyphicon-arrow-left:before{content:"\e091"}.glyphicon-arrow-right:before{content:"\e092"}.glyphicon-arrow-up:before{content:"\e093"}.glyphicon-arrow-down:before{content:"\e094"}.glyphicon-share-alt:before{content:"\e095"}.glyphicon-resize-full:before{content:"\e096"}.glyphicon-resize-small:before{content:"\e097"}.glyphicon-exclamation-sign:before{content:"\e101"}.glyphicon-gift:before{content:"\e102"}.glyphicon-leaf:before{content:"\e103"}.glyphicon-fire:before{content:"\e104"}.glyphicon-eye-open:before{content:"\e105"}.glyphicon-eye-close:before{content:"\e106"}.glyphicon-warning-sign:before{content:"\e107"}.glyphicon-plane:before{content:"\e108"}.glyphicon-calendar:before{content:"\e109"}.glyphicon-random:before{content:"\e110"}.glyphicon-comment:before{content:"\e111"}.glyphicon-magnet:before{content:"\e112"}.glyphicon-chevron-up:before{content:"\e113"}.glyphicon-chevron-down:before{content:"\e114"}.glyphicon-retweet:before{content:"\e115"}.glyphicon-shopping-cart:before{content:"\e116"}.glyphicon-folder-close:before{content:"\e117"}.glyphicon-folder-open:before{content:"\e118"}.glyphicon-resize-vertical:before{content:"\e119"}.glyphicon-resize-horizontal:before{content:"\e120"}.glyphicon-hdd:before{content:"\e121"}.glyphicon-bullhorn:before{content:"\e122"}.glyphicon-bell:before{content:"\e123"}.glyphicon-certificate:before{content:"\e124"}.glyphicon-thumbs-up:before{content:"\e125"}.glyphicon-thumbs-down:before{content:"\e126"}.glyphicon-hand-right:before{content:"\e127"}.glyphicon-hand-left:before{content:"\e128"}.glyphicon-hand-up:before{content:"\e129"}.glyphicon-hand-down:before{content:"\e130"}.glyphicon-circle-arrow-right:before{content:"\e131"}.glyphicon-circle-arrow-left:before{content:"\e132"}.glyphicon-circle-arrow-up:before{content:"\e133"}.glyphicon-circle-arrow-down:before{content:"\e134"}.glyphicon-globe:before{content:"\e135"}.glyphicon-wrench:before{content:"\e136"}.glyphicon-tasks:before{content:"\e137"}.glyphicon-filter:before{content:"\e138"}.glyphicon-briefcase:before{content:"\e139"}.glyphicon-fullscreen:before{content:"\e140"}.glyphicon-dashboard:before{content:"\e141"}.glyphicon-paperclip:before{content:"\e142"}.glyphicon-heart-empty:before{content:"\e143"}.glyphicon-link:before{content:"\e144"}.glyphicon-phone:before{content:"\e145"}.glyphicon-pushpin:before{content:"\e146"}.glyphicon-usd:before{content:"\e148"}.glyphicon-gbp:before{content:"\e149"}.glyphicon-sort:before{content:"\e150"}.glyphicon-sort-by-alphabet:before{content:"\e151"}.glyphicon-sort-by-alphabet-alt:before{content:"\e152"}.glyphicon-sort-by-order:before{content:"\e153"}.glyphicon-sort-by-order-alt:before{content:"\e154"}.glyphicon-sort-by-attributes:before{content:"\e155"}.glyphicon-sort-by-attributes-alt:before{content:"\e156"}.glyphicon-unchecked:before{content:"\e157"}.glyphicon-expand:before{content:"\e158"}.glyphicon-collapse-down:before{content:"\e159"}.glyphicon-collapse-up:before{content:"\e160"}.glyphicon-log-in:before{content:"\e161"}.glyphicon-flash:before{content:"\e162"}.glyphicon-log-out:before{content:"\e163"}.glyphicon-new-window:before{content:"\e164"}.glyphicon-record:before{content:"\e165"}.glyphicon-save:before{content:"\e166"}.glyphicon-open:before{content:"\e167"}.glyphicon-saved:before{content:"\e168"}.glyphicon-import:before{content:"\e169"}.glyphicon-export:before{content:"\e170"}.glyphicon-send:before{content:"\e171"}.glyphicon-floppy-disk:before{content:"\e172"}.glyphicon-floppy-saved:before{content:"\e173"}.glyphicon-floppy-remove:before{content:"\e174"}.glyphicon-floppy-save:before{content:"\e175"}.glyphicon-floppy-open:before{content:"\e176"}.glyphicon-credit-card:before{content:"\e177"}.glyphicon-transfer:before{content:"\e178"}.glyphicon-cutlery:before{content:"\e179"}.glyphicon-header:before{content:"\e180"}.glyphicon-compressed:before{content:"\e181"}.glyphicon-earphone:before{content:"\e182"}.glyphicon-phone-alt:before{content:"\e183"}.glyphicon-tower:before{content:"\e184"}.glyphicon-stats:before{content:"\e185"}.glyphicon-sd-video:before{content:"\e186"}.glyphicon-hd-video:before{content:"\e187"}.glyphicon-subtitles:before{content:"\e188"}.glyphicon-sound-stereo:before{content:"\e189"}.glyphicon-sound-dolby:before{content:"\e190"}.glyphicon-sound-5-1:before{content:"\e191"}.glyphicon-sound-6-1:before{content:"\e192"}.glyphicon-sound-7-1:before{content:"\e193"}.glyphicon-copyright-mark:before{content:"\e194"}.glyphicon-registration-mark:before{content:"\e195"}.glyphicon-cloud-download:before{content:"\e197"}.glyphicon-cloud-upload:before{content:"\e198"}.glyphicon-tree-conifer:before{content:"\e199"}.glyphicon-tree-deciduous:before{content:"\e200"}.glyphicon-cd:before{content:"\e201"}.glyphicon-save-file:before{content:"\e202"}.glyphicon-open-file:before{content:"\e203"}.glyphicon-level-up:before{content:"\e204"}.glyphicon-copy:before{content:"\e205"}.glyphicon-paste:before{content:"\e206"}.glyphicon-alert:before{content:"\e209"}.glyphicon-equalizer:before{content:"\e210"}.glyphicon-king:before{content:"\e211"}.glyphicon-queen:before{content:"\e212"}.glyphicon-pawn:before{content:"\e213"}.glyphicon-bishop:before{content:"\e214"}.glyphicon-knight:before{content:"\e215"}.glyphicon-baby-formula:before{content:"\e216"}.glyphicon-tent:before{content:"\26fa"}.glyphicon-blackboard:before{content:"\e218"}.glyphicon-bed:before{content:"\e219"}.glyphicon-apple:before{content:"\f8ff"}.glyphicon-erase:before{content:"\e221"}.glyphicon-hourglass:before{content:"\231b"}.glyphicon-lamp:before{content:"\e223"}.glyphicon-duplicate:before{content:"\e224"}.glyphicon-piggy-bank:before{content:"\e225"}.glyphicon-scissors:before{content:"\e226"}.glyphicon-bitcoin:before{content:"\e227"}.glyphicon-btc:before{content:"\e227"}.glyphicon-xbt:before{content:"\e227"}.glyphicon-yen:before{content:"\00a5"}.glyphicon-jpy:before{content:"\00a5"}.glyphicon-ruble:before{content:"\20bd"}.glyphicon-rub:before{content:"\20bd"}.glyphicon-scale:before{content:"\e230"}.glyphicon-ice-lolly:before{content:"\e231"}.glyphicon-ice-lolly-tasted:before{content:"\e232"}.glyphicon-education:before{content:"\e233"}.glyphicon-option-horizontal:before{content:"\e234"}.glyphicon-option-vertical:before{content:"\e235"}.glyphicon-menu-hamburger:before{content:"\e236"}.glyphicon-modal-window:before{content:"\e237"}.glyphicon-oil:before{content:"\e238"}.glyphicon-grain:before{content:"\e239"}.glyphicon-sunglasses:before{content:"\e240"}.glyphicon-text-size:before{content:"\e241"}.glyphicon-text-color:before{content:"\e242"}.glyphicon-text-background:before{content:"\e243"}.glyphicon-object-align-top:before{content:"\e244"}.glyphicon-object-align-bottom:before{content:"\e245"}.glyphicon-object-align-horizontal:before{content:"\e246"}.glyphicon-object-align-left:before{content:"\e247"}.glyphicon-object-align-vertical:before{content:"\e248"}.glyphicon-object-align-right:before{content:"\e249"}.glyphicon-triangle-right:before{content:"\e250"}.glyphicon-triangle-left:before{content:"\e251"}.glyphicon-triangle-bottom:before{content:"\e252"}.glyphicon-triangle-top:before{content:"\e253"}.glyphicon-console:before{content:"\e254"}.glyphicon-superscript:before{content:"\e255"}.glyphicon-subscript:before{content:"\e256"}.glyphicon-menu-left:before{content:"\e257"}.glyphicon-menu-right:before{content:"\e258"}.glyphicon-menu-down:before{content:"\e259"}.glyphicon-menu-up:before{content:"\e260"}*{-webkit-box-sizing:border-box;-moz-box-sizing:border-box;box-sizing:border-box}:after,:before{-webkit-box-sizing:border-box;-moz-box-sizing:border-box;box-sizing:border-box}html{font-size:10px;-webkit-tap-highlight-color:rgba(0,0,0,0)}body{font-family:"Helvetica Neue",Helvetica,Arial,sans-serif;font-size:14px;line-height:1.42857143;color:#333;background-color:#fff}button,input,select,textarea{font-family:inherit;font-size:inherit;line-height:inherit}a{color:#337ab7;text-decoration:none}a:focus,a:hover{color:#23527c;text-decoration:underline}a:focus{outline:thin dotted;outline:5px auto -webkit-focus-ring-color;outline-offset:-2px}figure{margin:0}img{vertical-align:middle}.carousel-inner>.item>a>img,.carousel-inner>.item>img,.img-responsive,.thumbnail a>img,.thumbnail>img{display:block;max-width:100%;height:auto}.img-rounded{border-radius:6px}.img-thumbnail{display:inline-block;max-width:100%;height:auto;padding:4px;line-height:1.42857143;background-color:#fff;border:1px solid #ddd;border-radius:4px;-webkit-transition:all .2s ease-in-out;-o-transition:all .2s ease-in-out;transition:all .2s ease-in-out}.img-circle{border-radius:50%}hr{margin-top:20px;margin-bottom:20px;border:0;border-top:1px solid #eee}.sr-only{position:absolute;width:1px;height:1px;padding:0;margin:-1px;overflow:hidden;clip:rect(0,0,0,0);border:0}.sr-only-focusable:active,.sr-only-focusable:focus{position:static;width:auto;height:auto;margin:0;overflow:visible;clip:auto}[role=button]{cursor:pointer}.h1,.h2,.h3,.h4,.h5,.h6,h1,h2,h3,h4,h5,h6{font-family:inherit;font-weight:500;line-height:1.1;color:inherit}.h1 .small,.h1 small,.h2 .small,.h2 small,.h3 .small,.h3 small,.h4 .small,.h4 small,.h5 .small,.h5 small,.h6 .small,.h6 small,h1 .small,h1 small,h2 .small,h2 small,h3 .small,h3 small,h4 .small,h4 small,h5 .small,h5 small,h6 .small,h6 small{font-weight:400;line-height:1;color:#777}.h1,.h2,.h3,h1,h2,h3{margin-top:20px;margin-bottom:10px}.h1 .small,.h1 small,.h2 .small,.h2 small,.h3 .small,.h3 small,h1 .small,h1 small,h2 .small,h2 small,h3 .small,h3 small{font-size:65%}.h4,.h5,.h6,h4,h5,h6{margin-top:10px;margin-bottom:10px}.h4 .small,.h4 small,.h5 .small,.h5 small,.h6 .small,.h6 small,h4 .small,h4 small,h5 .small,h5 small,h6 .small,h6 small{font-size:75%}.h1,h1{font-size:36px}.h2,h2{font-size:30px}.h3,h3{font-size:24px}.h4,h4{font-size:18px}.h5,h5{font-size:14px}.h6,h6{font-size:12px}p{margin:0 0 10px}.lead{margin-bottom:20px;font-size:16px;font-weight:300;line-height:1.4}@media (min-width:768px){.lead{font-size:21px}}.small,small{font-size:85%}.mark,mark{padding:.2em;background-color:#fcf8e3}.text-left{text-align:left}.text-right{text-align:right}.text-center{text-align:center}.text-justify{text-align:justify}.text-nowrap{white-space:nowrap}.text-lowercase{text-transform:lowercase}.text-uppercase{text-transform:uppercase}.text-capitalize{text-transform:capitalize}.text-muted{color:#777}.text-primary{color:#337ab7}a.text-primary:hover{color:#286090}.text-success{color:#3c763d}a.text-success:hover{color:#2b542c}.text-info{color:#31708f}a.text-info:hover{color:#245269}.text-warning{color:#8a6d3b}a.text-warning:hover{color:#66512c}.text-danger{color:#a94442}a.text-danger:hover{color:#843534}.bg-primary{color:#fff;background-color:#337ab7}a.bg-primary:hover{background-color:#286090}.bg-success{background-color:#dff0d8}a.bg-success:hover{background-color:#c1e2b3}.bg-info{background-color:#d9edf7}a.bg-info:hover{background-color:#afd9ee}.bg-warning{background-color:#fcf8e3}a.bg-warning:hover{background-color:#f7ecb5}.bg-danger{background-color:#f2dede}a.bg-danger:hover{background-color:#e4b9b9}.page-header{padding-bottom:9px;margin:40px 0 20px;border-bottom:1px solid #eee}ol,ul{margin-top:0;margin-bottom:10px}ol ol,ol ul,ul ol,ul ul{margin-bottom:0}.list-unstyled{padding-left:0;list-style:none}.list-inline{padding-left:0;margin-left:-5px;list-style:none}.list-inline>li{display:inline-block;padding-right:5px;padding-left:5px}dl{margin-top:0;margin-bottom:20px}dd,dt{line-height:1.42857143}dt{font-weight:700}dd{margin-left:0}@media (min-width:768px){.dl-horizontal dt{float:left;width:160px;overflow:hidden;clear:left;text-align:right;text-overflow:ellipsis;white-space:nowrap}.dl-horizontal dd{margin-left:180px}}abbr[data-original-title],abbr[title]{cursor:help;border-bottom:1px dotted #777}.initialism{font-size:90%;text-transform:uppercase}blockquote{padding:10px 20px;margin:0 0 20px;font-size:17.5px;border-left:5px solid #eee}blockquote ol:last-child,blockquote p:last-child,blockquote ul:last-child{margin-bottom:0}blockquote .small,blockquote footer,blockquote small{display:block;font-size:80%;line-height:1.42857143;color:#777}blockquote .small:before,blockquote footer:before,blockquote small:before{content:'\2014 \00A0'}.blockquote-reverse,blockquote.pull-right{padding-right:15px;padding-left:0;text-align:right;border-right:5px solid #eee;border-left:0}.blockquote-reverse .small:before,.blockquote-reverse footer:before,.blockquote-reverse small:before,blockquote.pull-right .small:before,blockquote.pull-right footer:before,blockquote.pull-right small:before{content:''}.blockquote-reverse .small:after,.blockquote-reverse footer:after,.blockquote-reverse small:after,blockquote.pull-right .small:after,blockquote.pull-right footer:after,blockquote.pull-right small:after{content:'\00A0 \2014'}address{margin-bottom:20px;font-style:normal;line-height:1.42857143}code,kbd,pre,samp{font-family:Menlo,Monaco,Consolas,"Courier New",monospace}code{padding:2px 4px;font-size:90%;color:#c7254e;background-color:#f9f2f4;border-radius:4px}kbd{padding:2px 4px;font-size:90%;color:#fff;background-color:#333;border-radius:3px;-webkit-box-shadow:inset 0 -1px 0 rgba(0,0,0,.25);box-shadow:inset 0 -1px 0 rgba(0,0,0,.25)}kbd kbd{padding:0;font-size:100%;font-weight:700;-webkit-box-shadow:none;box-shadow:none}pre{display:block;padding:9.5px;margin:0 0 10px;font-size:13px;line-height:1.42857143;color:#333;word-break:break-all;word-wrap:break-word;background-color:#f5f5f5;border:1px solid #ccc;border-radius:4px}pre code{padding:0;font-size:inherit;color:inherit;white-space:pre-wrap;background-color:transparent;border-radius:0}.pre-scrollable{max-height:340px;overflow-y:scroll}.container{padding-right:15px;padding-left:15px;margin-right:auto;margin-left:auto}@media (min-width:768px){.container{width:750px}}@media (min-width:992px){.container{width:970px}}@media (min-width:1200px){.container{width:1170px}}.container-fluid{padding-right:15px;padding-left:15px;margin-right:auto;margin-left:auto}.row{margin-right:-15px;margin-left:-15px}.col-lg-1,.col-lg-10,.col-lg-11,.col-lg-12,.col-lg-2,.col-lg-3,.col-lg-4,.col-lg-5,.col-lg-6,.col-lg-7,.col-lg-8,.col-lg-9,.col-md-1,.col-md-10,.col-md-11,.col-md-12,.col-md-2,.col-md-3,.col-md-4,.col-md-5,.col-md-6,.col-md-7,.col-md-8,.col-md-9,.col-sm-1,.col-sm-10,.col-sm-11,.col-sm-12,.col-sm-2,.col-sm-3,.col-sm-4,.col-sm-5,.col-sm-6,.col-sm-7,.col-sm-8,.col-sm-9,.col-xs-1,.col-xs-10,.col-xs-11,.col-xs-12,.col-xs-2,.col-xs-3,.col-xs-4,.col-xs-5,.col-xs-6,.col-xs-7,.col-xs-8,.col-xs-9{position:relative;min-height:1px;padding-right:15px;padding-left:15px}.col-xs-1,.col-xs-10,.col-xs-11,.col-xs-12,.col-xs-2,.col-xs-3,.col-xs-4,.col-xs-5,.col-xs-6,.col-xs-7,.col-xs-8,.col-xs-9{float:left}.col-xs-12{width:100%}.col-xs-11{width:91.66666667%}.col-xs-10{width:83.33333333%}.col-xs-9{width:75%}.col-xs-8{width:66.66666667%}.col-xs-7{width:58.33333333%}.col-xs-6{width:50%}.col-xs-5{width:41.66666667%}.col-xs-4{width:33.33333333%}.col-xs-3{width:25%}.col-xs-2{width:16.66666667%}.col-xs-1{width:8.33333333%}.col-xs-pull-12{right:100%}.col-xs-pull-11{right:91.66666667%}.col-xs-pull-10{right:83.33333333%}.col-xs-pull-9{right:75%}.col-xs-pull-8{right:66.66666667%}.col-xs-pull-7{right:58.33333333%}.col-xs-pull-6{right:50%}.col-xs-pull-5{right:41.66666667%}.col-xs-pull-4{right:33.33333333%}.col-xs-pull-3{right:25%}.col-xs-pull-2{right:16.66666667%}.col-xs-pull-1{right:8.33333333%}.col-xs-pull-0{right:auto}.col-xs-push-12{left:100%}.col-xs-push-11{left:91.66666667%}.col-xs-push-10{left:83.33333333%}.col-xs-push-9{left:75%}.col-xs-push-8{left:66.66666667%}.col-xs-push-7{left:58.33333333%}.col-xs-push-6{left:50%}.col-xs-push-5{left:41.66666667%}.col-xs-push-4{left:33.33333333%}.col-xs-push-3{left:25%}.col-xs-push-2{left:16.66666667%}.col-xs-push-1{left:8.33333333%}.col-xs-push-0{left:auto}.col-xs-offset-12{margin-left:100%}.col-xs-offset-11{margin-left:91.66666667%}.col-xs-offset-10{margin-left:83.33333333%}.col-xs-offset-9{margin-left:75%}.col-xs-offset-8{margin-left:66.66666667%}.col-xs-offset-7{margin-left:58.33333333%}.col-xs-offset-6{margin-left:50%}.col-xs-offset-5{margin-left:41.66666667%}.col-xs-offset-4{margin-left:33.33333333%}.col-xs-offset-3{margin-left:25%}.col-xs-offset-2{margin-left:16.66666667%}.col-xs-offset-1{margin-left:8.33333333%}.col-xs-offset-0{margin-left:0}@media (min-width:768px){.col-sm-1,.col-sm-10,.col-sm-11,.col-sm-12,.col-sm-2,.col-sm-3,.col-sm-4,.col-sm-5,.col-sm-6,.col-sm-7,.col-sm-8,.col-sm-9{float:left}.col-sm-12{width:100%}.col-sm-11{width:91.66666667%}.col-sm-10{width:83.33333333%}.col-sm-9{width:75%}.col-sm-8{width:66.66666667%}.col-sm-7{width:58.33333333%}.col-sm-6{width:50%}.col-sm-5{width:41.66666667%}.col-sm-4{width:33.33333333%}.col-sm-3{width:25%}.col-sm-2{width:16.66666667%}.col-sm-1{width:8.33333333%}.col-sm-pull-12{right:100%}.col-sm-pull-11{right:91.66666667%}.col-sm-pull-10{right:83.33333333%}.col-sm-pull-9{right:75%}.col-sm-pull-8{right:66.66666667%}.col-sm-pull-7{right:58.33333333%}.col-sm-pull-6{right:50%}.col-sm-pull-5{right:41.66666667%}.col-sm-pull-4{right:33.33333333%}.col-sm-pull-3{right:25%}.col-sm-pull-2{right:16.66666667%}.col-sm-pull-1{right:8.33333333%}.col-sm-pull-0{right:auto}.col-sm-push-12{left:100%}.col-sm-push-11{left:91.66666667%}.col-sm-push-10{left:83.33333333%}.col-sm-push-9{left:75%}.col-sm-push-8{left:66.66666667%}.col-sm-push-7{left:58.33333333%}.col-sm-push-6{left:50%}.col-sm-push-5{left:41.66666667%}.col-sm-push-4{left:33.33333333%}.col-sm-push-3{left:25%}.col-sm-push-2{left:16.66666667%}.col-sm-push-1{left:8.33333333%}.col-sm-push-0{left:auto}.col-sm-offset-12{margin-left:100%}.col-sm-offset-11{margin-left:91.66666667%}.col-sm-offset-10{margin-left:83.33333333%}.col-sm-offset-9{margin-left:75%}.col-sm-offset-8{margin-left:66.66666667%}.col-sm-offset-7{margin-left:58.33333333%}.col-sm-offset-6{margin-left:50%}.col-sm-offset-5{margin-left:41.66666667%}.col-sm-offset-4{margin-left:33.33333333%}.col-sm-offset-3{margin-left:25%}.col-sm-offset-2{margin-left:16.66666667%}.col-sm-offset-1{margin-left:8.33333333%}.col-sm-offset-0{margin-left:0}}@media (min-width:992px){.col-md-1,.col-md-10,.col-md-11,.col-md-12,.col-md-2,.col-md-3,.col-md-4,.col-md-5,.col-md-6,.col-md-7,.col-md-8,.col-md-9{float:left}.col-md-12{width:100%}.col-md-11{width:91.66666667%}.col-md-10{width:83.33333333%}.col-md-9{width:75%}.col-md-8{width:66.66666667%}.col-md-7{width:58.33333333%}.col-md-6{width:50%}.col-md-5{width:41.66666667%}.col-md-4{width:33.33333333%}.col-md-3{width:25%}.col-md-2{width:16.66666667%}.col-md-1{width:8.33333333%}.col-md-pull-12{right:100%}.col-md-pull-11{right:91.66666667%}.col-md-pull-10{right:83.33333333%}.col-md-pull-9{right:75%}.col-md-pull-8{right:66.66666667%}.col-md-pull-7{right:58.33333333%}.col-md-pull-6{right:50%}.col-md-pull-5{right:41.66666667%}.col-md-pull-4{right:33.33333333%}.col-md-pull-3{right:25%}.col-md-pull-2{right:16.66666667%}.col-md-pull-1{right:8.33333333%}.col-md-pull-0{right:auto}.col-md-push-12{left:100%}.col-md-push-11{left:91.66666667%}.col-md-push-10{left:83.33333333%}.col-md-push-9{left:75%}.col-md-push-8{left:66.66666667%}.col-md-push-7{left:58.33333333%}.col-md-push-6{left:50%}.col-md-push-5{left:41.66666667%}.col-md-push-4{left:33.33333333%}.col-md-push-3{left:25%}.col-md-push-2{left:16.66666667%}.col-md-push-1{left:8.33333333%}.col-md-push-0{left:auto}.col-md-offset-12{margin-left:100%}.col-md-offset-11{margin-left:91.66666667%}.col-md-offset-10{margin-left:83.33333333%}.col-md-offset-9{margin-left:75%}.col-md-offset-8{margin-left:66.66666667%}.col-md-offset-7{margin-left:58.33333333%}.col-md-offset-6{margin-left:50%}.col-md-offset-5{margin-left:41.66666667%}.col-md-offset-4{margin-left:33.33333333%}.col-md-offset-3{margin-left:25%}.col-md-offset-2{margin-left:16.66666667%}.col-md-offset-1{margin-left:8.33333333%}.col-md-offset-0{margin-left:0}}@media (min-width:1200px){.col-lg-1,.col-lg-10,.col-lg-11,.col-lg-12,.col-lg-2,.col-lg-3,.col-lg-4,.col-lg-5,.col-lg-6,.col-lg-7,.col-lg-8,.col-lg-9{float:left}.col-lg-12{width:100%}.col-lg-11{width:91.66666667%}.col-lg-10{width:83.33333333%}.col-lg-9{width:75%}.col-lg-8{width:66.66666667%}.col-lg-7{width:58.33333333%}.col-lg-6{width:50%}.col-lg-5{width:41.66666667%}.col-lg-4{width:33.33333333%}.col-lg-3{width:25%}.col-lg-2{width:16.66666667%}.col-lg-1{width:8.33333333%}.col-lg-pull-12{right:100%}.col-lg-pull-11{right:91.66666667%}.col-lg-pull-10{right:83.33333333%}.col-lg-pull-9{right:75%}.col-lg-pull-8{right:66.66666667%}.col-lg-pull-7{right:58.33333333%}.col-lg-pull-6{right:50%}.col-lg-pull-5{right:41.66666667%}.col-lg-pull-4{right:33.33333333%}.col-lg-pull-3{right:25%}.col-lg-pull-2{right:16.66666667%}.col-lg-pull-1{right:8.33333333%}.col-lg-pull-0{right:auto}.col-lg-push-12{left:100%}.col-lg-push-11{left:91.66666667%}.col-lg-push-10{left:83.33333333%}.col-lg-push-9{left:75%}.col-lg-push-8{left:66.66666667%}.col-lg-push-7{left:58.33333333%}.col-lg-push-6{left:50%}.col-lg-push-5{left:41.66666667%}.col-lg-push-4{left:33.33333333%}.col-lg-push-3{left:25%}.col-lg-push-2{left:16.66666667%}.col-lg-push-1{left:8.33333333%}.col-lg-push-0{left:auto}.col-lg-offset-12{margin-left:100%}.col-lg-offset-11{margin-left:91.66666667%}.col-lg-offset-10{margin-left:83.33333333%}.col-lg-offset-9{margin-left:75%}.col-lg-offset-8{margin-left:66.66666667%}.col-lg-offset-7{margin-left:58.33333333%}.col-lg-offset-6{margin-left:50%}.col-lg-offset-5{margin-left:41.66666667%}.col-lg-offset-4{margin-left:33.33333333%}.col-lg-offset-3{margin-left:25%}.col-lg-offset-2{margin-left:16.66666667%}.col-lg-offset-1{margin-left:8.33333333%}.col-lg-offset-0{margin-left:0}}table{background-color:transparent}caption{padding-top:8px;padding-bottom:8px;color:#777;text-align:left}th{text-align:left}.table{width:100%;max-width:100%;margin-bottom:20px}.table>tbody>tr>td,.table>tbody>tr>th,.table>tfoot>tr>td,.table>tfoot>tr>th,.table>thead>tr>td,.table>thead>tr>th{padding:8px;line-height:1.42857143;vertical-align:top;border-top:1px solid #ddd}.table>thead>tr>th{vertical-align:bottom;border-bottom:2px solid #ddd}.table>caption+thead>tr:first-child>td,.table>caption+thead>tr:first-child>th,.table>colgroup+thead>tr:first-child>td,.table>colgroup+thead>tr:first-child>th,.table>thead:first-child>tr:first-child>td,.table>thead:first-child>tr:first-child>th{border-top:0}.table>tbody+tbody{border-top:2px solid #ddd}.table .table{background-color:#fff}.table-condensed>tbody>tr>td,.table-condensed>tbody>tr>th,.table-condensed>tfoot>tr>td,.table-condensed>tfoot>tr>th,.table-condensed>thead>tr>td,.table-condensed>thead>tr>th{padding:5px}.table-bordered{border:1px solid #ddd}.table-bordered>tbody>tr>td,.table-bordered>tbody>tr>th,.table-bordered>tfoot>tr>td,.table-bordered>tfoot>tr>th,.table-bordered>thead>tr>td,.table-bordered>thead>tr>th{border:1px solid #ddd}.table-bordered>thead>tr>td,.table-bordered>thead>tr>th{border-bottom-width:2px}.table-striped>tbody>tr:nth-of-type(odd){background-color:#f9f9f9}.table-hover>tbody>tr:hover{background-color:#f5f5f5}table col[class*=col-]{position:static;display:table-column;float:none}table td[class*=col-],table th[class*=col-]{position:static;display:table-cell;float:none}.table>tbody>tr.active>td,.table>tbody>tr.active>th,.table>tbody>tr>td.active,.table>tbody>tr>th.active,.table>tfoot>tr.active>td,.table>tfoot>tr.active>th,.table>tfoot>tr>td.active,.table>tfoot>tr>th.active,.table>thead>tr.active>td,.table>thead>tr.active>th,.table>thead>tr>td.active,.table>thead>tr>th.active{background-color:#f5f5f5}.table-hover>tbody>tr.active:hover>td,.table-hover>tbody>tr.active:hover>th,.table-hover>tbody>tr:hover>.active,.table-hover>tbody>tr>td.active:hover,.table-hover>tbody>tr>th.active:hover{background-color:#e8e8e8}.table>tbody>tr.success>td,.table>tbody>tr.success>th,.table>tbody>tr>td.success,.table>tbody>tr>th.success,.table>tfoot>tr.success>td,.table>tfoot>tr.success>th,.table>tfoot>tr>td.success,.table>tfoot>tr>th.success,.table>thead>tr.success>td,.table>thead>tr.success>th,.table>thead>tr>td.success,.table>thead>tr>th.success{background-color:#dff0d8}.table-hover>tbody>tr.success:hover>td,.table-hover>tbody>tr.success:hover>th,.table-hover>tbody>tr:hover>.success,.table-hover>tbody>tr>td.success:hover,.table-hover>tbody>tr>th.success:hover{background-color:#d0e9c6}.table>tbody>tr.info>td,.table>tbody>tr.info>th,.table>tbody>tr>td.info,.table>tbody>tr>th.info,.table>tfoot>tr.info>td,.table>tfoot>tr.info>th,.table>tfoot>tr>td.info,.table>tfoot>tr>th.info,.table>thead>tr.info>td,.table>thead>tr.info>th,.table>thead>tr>td.info,.table>thead>tr>th.info{background-color:#d9edf7}.table-hover>tbody>tr.info:hover>td,.table-hover>tbody>tr.info:hover>th,.table-hover>tbody>tr:hover>.info,.table-hover>tbody>tr>td.info:hover,.table-hover>tbody>tr>th.info:hover{background-color:#c4e3f3}.table>tbody>tr.warning>td,.table>tbody>tr.warning>th,.table>tbody>tr>td.warning,.table>tbody>tr>th.warning,.table>tfoot>tr.warning>td,.table>tfoot>tr.warning>th,.table>tfoot>tr>td.warning,.table>tfoot>tr>th.warning,.table>thead>tr.warning>td,.table>thead>tr.warning>th,.table>thead>tr>td.warning,.table>thead>tr>th.warning{background-color:#fcf8e3}.table-hover>tbody>tr.warning:hover>td,.table-hover>tbody>tr.warning:hover>th,.table-hover>tbody>tr:hover>.warning,.table-hover>tbody>tr>td.warning:hover,.table-hover>tbody>tr>th.warning:hover{background-color:#faf2cc}.table>tbody>tr.danger>td,.table>tbody>tr.danger>th,.table>tbody>tr>td.danger,.table>tbody>tr>th.danger,.table>tfoot>tr.danger>td,.table>tfoot>tr.danger>th,.table>tfoot>tr>td.danger,.table>tfoot>tr>th.danger,.table>thead>tr.danger>td,.table>thead>tr.danger>th,.table>thead>tr>td.danger,.table>thead>tr>th.danger{background-color:#f2dede}.table-hover>tbody>tr.danger:hover>td,.table-hover>tbody>tr.danger:hover>th,.table-hover>tbody>tr:hover>.danger,.table-hover>tbody>tr>td.danger:hover,.table-hover>tbody>tr>th.danger:hover{background-color:#ebcccc}.table-responsive{min-height:.01%;overflow-x:auto}@media screen and (max-width:767px){.table-responsive{width:100%;margin-bottom:15px;overflow-y:hidden;-ms-overflow-style:-ms-autohiding-scrollbar;border:1px solid #ddd}.table-responsive>.table{margin-bottom:0}.table-responsive>.table>tbody>tr>td,.table-responsive>.table>tbody>tr>th,.table-responsive>.table>tfoot>tr>td,.table-responsive>.table>tfoot>tr>th,.table-responsive>.table>thead>tr>td,.table-responsive>.table>thead>tr>th{white-space:nowrap}.table-responsive>.table-bordered{border:0}.table-responsive>.table-bordered>tbody>tr>td:first-child,.table-responsive>.table-bordered>tbody>tr>th:first-child,.table-responsive>.table-bordered>tfoot>tr>td:first-child,.table-responsive>.table-bordered>tfoot>tr>th:first-child,.table-responsive>.table-bordered>thead>tr>td:first-child,.table-responsive>.table-bordered>thead>tr>th:first-child{border-left:0}.table-responsive>.table-bordered>tbody>tr>td:last-child,.table-responsive>.table-bordered>tbody>tr>th:last-child,.table-responsive>.table-bordered>tfoot>tr>td:last-child,.table-responsive>.table-bordered>tfoot>tr>th:last-child,.table-responsive>.table-bordered>thead>tr>td:last-child,.table-responsive>.table-bordered>thead>tr>th:last-child{border-right:0}.table-responsive>.table-bordered>tbody>tr:last-child>td,.table-responsive>.table-bordered>tbody>tr:last-child>th,.table-responsive>.table-bordered>tfoot>tr:last-child>td,.table-responsive>.table-bordered>tfoot>tr:last-child>th{border-bottom:0}}fieldset{min-width:0;padding:0;margin:0;border:0}legend{display:block;width:100%;padding:0;margin-bottom:20px;font-size:21px;line-height:inherit;color:#333;border:0;border-bottom:1px solid #e5e5e5}label{display:inline-block;max-width:100%;margin-bottom:5px;font-weight:700}input[type=search]{-webkit-box-sizing:border-box;-moz-box-sizing:border-box;box-sizing:border-box}input[type=checkbox],input[type=radio]{margin:4px 0 0;margin-top:1px \9;line-height:normal}input[type=file]{display:block}input[type=range]{display:block;width:100%}select[multiple],select[size]{height:auto}input[type=file]:focus,input[type=checkbox]:focus,input[type=radio]:focus{outline:thin dotted;outline:5px auto -webkit-focus-ring-color;outline-offset:-2px}output{display:block;padding-top:7px;font-size:14px;line-height:1.42857143;color:#555}.form-control{display:block;width:100%;height:34px;padding:6px 12px;font-size:14px;line-height:1.42857143;color:#555;background-color:#fff;background-image:none;border:1px solid #ccc;border-radius:4px;-webkit-box-shadow:inset 0 1px 1px rgba(0,0,0,.075);box-shadow:inset 0 1px 1px rgba(0,0,0,.075);-webkit-transition:border-color ease-in-out .15s,-webkit-box-shadow ease-in-out .15s;-o-transition:border-color ease-in-out .15s,box-shadow ease-in-out .15s;transition:border-color ease-in-out .15s,box-shadow ease-in-out .15s}.form-control:focus{border-color:#66afe9;outline:0;-webkit-box-shadow:inset 0 1px 1px rgba(0,0,0,.075),0 0 8px rgba(102,175,233,.6);box-shadow:inset 0 1px 1px rgba(0,0,0,.075),0 0 8px rgba(102,175,233,.6)}.form-control::-moz-placeholder{color:#999;opacity:1}.form-control:-ms-input-placeholder{color:#999}.form-control::-webkit-input-placeholder{color:#999}.form-control[disabled],.form-control[readonly],fieldset[disabled] .form-control{background-color:#eee;opacity:1}.form-control[disabled],fieldset[disabled] .form-control{cursor:not-allowed}textarea.form-control{height:auto}input[type=search]{-webkit-appearance:none}@media screen and (-webkit-min-device-pixel-ratio:0){input[type=date],input[type=time],input[type=datetime-local],input[type=month]{line-height:34px}.input-group-sm input[type=date],.input-group-sm input[type=time],.input-group-sm input[type=datetime-local],.input-group-sm input[type=month],input[type=date].input-sm,input[type=time].input-sm,input[type=datetime-local].input-sm,input[type=month].input-sm{line-height:30px}.input-group-lg input[type=date],.input-group-lg input[type=time],.input-group-lg input[type=datetime-local],.input-group-lg input[type=month],input[type=date].input-lg,input[type=time].input-lg,input[type=datetime-local].input-lg,input[type=month].input-lg{line-height:46px}}.form-group{margin-bottom:15px}.checkbox,.radio{position:relative;display:block;margin-top:10px;margin-bottom:10px}.checkbox label,.radio label{min-height:20px;padding-left:20px;margin-bottom:0;font-weight:400;cursor:pointer}.checkbox input[type=checkbox],.checkbox-inline input[type=checkbox],.radio input[type=radio],.radio-inline input[type=radio]{position:absolute;margin-top:4px \9;margin-left:-20px}.checkbox+.checkbox,.radio+.radio{margin-top:-5px}.checkbox-inline,.radio-inline{position:relative;display:inline-block;padding-left:20px;margin-bottom:0;font-weight:400;vertical-align:middle;cursor:pointer}.checkbox-inline+.checkbox-inline,.radio-inline+.radio-inline{margin-top:0;margin-left:10px}fieldset[disabled] input[type=checkbox],fieldset[disabled] input[type=radio],input[type=checkbox].disabled,input[type=checkbox][disabled],input[type=radio].disabled,input[type=radio][disabled]{cursor:not-allowed}.checkbox-inline.disabled,.radio-inline.disabled,fieldset[disabled] .checkbox-inline,fieldset[disabled] .radio-inline{cursor:not-allowed}.checkbox.disabled label,.radio.disabled label,fieldset[disabled] .checkbox label,fieldset[disabled] .radio label{cursor:not-allowed}.form-control-static{min-height:34px;padding-top:7px;padding-bottom:7px;margin-bottom:0}.form-control-static.input-lg,.form-control-static.input-sm{padding-right:0;padding-left:0}.input-sm{height:30px;padding:5px 10px;font-size:12px;line-height:1.5;border-radius:3px}select.input-sm{height:30px;line-height:30px}select[multiple].input-sm,textarea.input-sm{height:auto}.form-group-sm .form-control{height:30px;padding:5px 10px;font-size:12px;line-height:1.5;border-radius:3px}select.form-group-sm .form-control{height:30px;line-height:30px}select[multiple].form-group-sm .form-control,textarea.form-group-sm .form-control{height:auto}.form-group-sm .form-control-static{height:30px;min-height:32px;padding:5px 10px;font-size:12px;line-height:1.5}.input-lg{height:46px;padding:10px 16px;font-size:18px;line-height:1.3333333;border-radius:6px}select.input-lg{height:46px;line-height:46px}select[multiple].input-lg,textarea.input-lg{height:auto}.form-group-lg .form-control{height:46px;padding:10px 16px;font-size:18px;line-height:1.3333333;border-radius:6px}select.form-group-lg .form-control{height:46px;line-height:46px}select[multiple].form-group-lg .form-control,textarea.form-group-lg .form-control{height:auto}.form-group-lg .form-control-static{height:46px;min-height:38px;padding:10px 16px;font-size:18px;line-height:1.3333333}.has-feedback{position:relative}.has-feedback .form-control{padding-right:42.5px}.form-control-feedback{position:absolute;top:0;right:0;z-index:2;display:block;width:34px;height:34px;line-height:34px;text-align:center;pointer-events:none}.input-lg+.form-control-feedback{width:46px;height:46px;line-height:46px}.input-sm+.form-control-feedback{width:30px;height:30px;line-height:30px}.has-success .checkbox,.has-success .checkbox-inline,.has-success .control-label,.has-success .help-block,.has-success .radio,.has-success .radio-inline,.has-success.checkbox label,.has-success.checkbox-inline label,.has-success.radio label,.has-success.radio-inline label{color:#3c763d}.has-success .form-control{border-color:#3c763d;-webkit-box-shadow:inset 0 1px 1px rgba(0,0,0,.075);box-shadow:inset 0 1px 1px rgba(0,0,0,.075)}.has-success .form-control:focus{border-color:#2b542c;-webkit-box-shadow:inset 0 1px 1px rgba(0,0,0,.075),0 0 6px #67b168;box-shadow:inset 0 1px 1px rgba(0,0,0,.075),0 0 6px #67b168}.has-success .input-group-addon{color:#3c763d;background-color:#dff0d8;border-color:#3c763d}.has-success .form-control-feedback{color:#3c763d}.has-warning .checkbox,.has-warning .checkbox-inline,.has-warning .control-label,.has-warning .help-block,.has-warning .radio,.has-warning .radio-inline,.has-warning.checkbox label,.has-warning.checkbox-inline label,.has-warning.radio label,.has-warning.radio-inline label{color:#8a6d3b}.has-warning .form-control{border-color:#8a6d3b;-webkit-box-shadow:inset 0 1px 1px rgba(0,0,0,.075);box-shadow:inset 0 1px 1px rgba(0,0,0,.075)}.has-warning .form-control:focus{border-color:#66512c;-webkit-box-shadow:inset 0 1px 1px rgba(0,0,0,.075),0 0 6px #c0a16b;box-shadow:inset 0 1px 1px rgba(0,0,0,.075),0 0 6px #c0a16b}.has-warning .input-group-addon{color:#8a6d3b;background-color:#fcf8e3;border-color:#8a6d3b}.has-warning .form-control-feedback{color:#8a6d3b}.has-error .checkbox,.has-error .checkbox-inline,.has-error .control-label,.has-error .help-block,.has-error .radio,.has-error .radio-inline,.has-error.checkbox label,.has-error.checkbox-inline label,.has-error.radio label,.has-error.radio-inline label{color:#a94442}.has-error .form-control{border-color:#a94442;-webkit-box-shadow:inset 0 1px 1px rgba(0,0,0,.075);box-shadow:inset 0 1px 1px rgba(0,0,0,.075)}.has-error .form-control:focus{border-color:#843534;-webkit-box-shadow:inset 0 1px 1px rgba(0,0,0,.075),0 0 6px #ce8483;box-shadow:inset 0 1px 1px rgba(0,0,0,.075),0 0 6px #ce8483}.has-error .input-group-addon{color:#a94442;background-color:#f2dede;border-color:#a94442}.has-error .form-control-feedback{color:#a94442}.has-feedback label~.form-control-feedback{top:25px}.has-feedback label.sr-only~.form-control-feedback{top:0}.help-block{display:block;margin-top:5px;margin-bottom:10px;color:#737373}@media (min-width:768px){.form-inline .form-group{display:inline-block;margin-bottom:0;vertical-align:middle}.form-inline .form-control{display:inline-block;width:auto;vertical-align:middle}.form-inline .form-control-static{display:inline-block}.form-inline .input-group{display:inline-table;vertical-align:middle}.form-inline .input-group .form-control,.form-inline .input-group .input-group-addon,.form-inline .input-group .input-group-btn{width:auto}.form-inline .input-group>.form-control{width:100%}.form-inline .control-label{margin-bottom:0;vertical-align:middle}.form-inline .checkbox,.form-inline .radio{display:inline-block;margin-top:0;margin-bottom:0;vertical-align:middle}.form-inline .checkbox label,.form-inline .radio label{padding-left:0}.form-inline .checkbox input[type=checkbox],.form-inline .radio input[type=radio]{position:relative;margin-left:0}.form-inline .has-feedback .form-control-feedback{top:0}}.form-horizontal .checkbox,.form-horizontal .checkbox-inline,.form-horizontal .radio,.form-horizontal .radio-inline{padding-top:7px;margin-top:0;margin-bottom:0}.form-horizontal .checkbox,.form-horizontal .radio{min-height:27px}.form-horizontal .form-group{margin-right:-15px;margin-left:-15px}@media (min-width:768px){.form-horizontal .control-label{padding-top:7px;margin-bottom:0;text-align:right}}.form-horizontal .has-feedback .form-control-feedback{right:15px}@media (min-width:768px){.form-horizontal .form-group-lg .control-label{padding-top:14.33px}}@media (min-width:768px){.form-horizontal .form-group-sm .control-label{padding-top:6px}}.btn{display:inline-block;padding:6px 12px;margin-bottom:0;font-size:14px;font-weight:400;line-height:1.42857143;text-align:center;white-space:nowrap;vertical-align:middle;-ms-touch-action:manipulation;touch-action:manipulation;cursor:pointer;-webkit-user-select:none;-moz-user-select:none;-ms-user-select:none;user-select:none;background-image:none;border:1px solid transparent;border-radius:4px}.btn.active.focus,.btn.active:focus,.btn.focus,.btn:active.focus,.btn:active:focus,.btn:focus{outline:thin dotted;outline:5px auto -webkit-focus-ring-color;outline-offset:-2px}.btn.focus,.btn:focus,.btn:hover{color:#333;text-decoration:none}.btn.active,.btn:active{background-image:none;outline:0;-webkit-box-shadow:inset 0 3px 5px rgba(0,0,0,.125);box-shadow:inset 0 3px 5px rgba(0,0,0,.125)}.btn.disabled,.btn[disabled],fieldset[disabled] .btn{pointer-events:none;cursor:not-allowed;filter:alpha(opacity=65);-webkit-box-shadow:none;box-shadow:none;opacity:.65}.btn-default{color:#333;background-color:#fff;border-color:#ccc}.btn-default.active,.btn-default.focus,.btn-default:active,.btn-default:focus,.btn-default:hover,.open>.dropdown-toggle.btn-default{color:#333;background-color:#e6e6e6;border-color:#adadad}.btn-default.active,.btn-default:active,.open>.dropdown-toggle.btn-default{background-image:none}.btn-default.disabled,.btn-default.disabled.active,.btn-default.disabled.focus,.btn-default.disabled:active,.btn-default.disabled:focus,.btn-default.disabled:hover,.btn-default[disabled],.btn-default[disabled].active,.btn-default[disabled].focus,.btn-default[disabled]:active,.btn-default[disabled]:focus,.btn-default[disabled]:hover,fieldset[disabled] .btn-default,fieldset[disabled] .btn-default.active,fieldset[disabled] .btn-default.focus,fieldset[disabled] .btn-default:active,fieldset[disabled] .btn-default:focus,fieldset[disabled] .btn-default:hover{background-color:#fff;border-color:#ccc}.btn-default .badge{color:#fff;background-color:#333}.btn-primary{color:#fff;background-color:#337ab7;border-color:#2e6da4}.btn-primary.active,.btn-primary.focus,.btn-primary:active,.btn-primary:focus,.btn-primary:hover,.open>.dropdown-toggle.btn-primary{color:#fff;background-color:#286090;border-color:#204d74}.btn-primary.active,.btn-primary:active,.open>.dropdown-toggle.btn-primary{background-image:none}.btn-primary.disabled,.btn-primary.disabled.active,.btn-primary.disabled.focus,.btn-primary.disabled:active,.btn-primary.disabled:focus,.btn-primary.disabled:hover,.btn-primary[disabled],.btn-primary[disabled].active,.btn-primary[disabled].focus,.btn-primary[disabled]:active,.btn-primary[disabled]:focus,.btn-primary[disabled]:hover,fieldset[disabled] .btn-primary,fieldset[disabled] .btn-primary.active,fieldset[disabled] .btn-primary.focus,fieldset[disabled] .btn-primary:active,fieldset[disabled] .btn-primary:focus,fieldset[disabled] .btn-primary:hover{background-color:#337ab7;border-color:#2e6da4}.btn-primary .badge{color:#337ab7;background-color:#fff}.btn-success{color:#fff;background-color:#5cb85c;border-color:#4cae4c}.btn-success.active,.btn-success.focus,.btn-success:active,.btn-success:focus,.btn-success:hover,.open>.dropdown-toggle.btn-success{color:#fff;background-color:#449d44;border-color:#398439}.btn-success.active,.btn-success:active,.open>.dropdown-toggle.btn-success{background-image:none}.btn-success.disabled,.btn-success.disabled.active,.btn-success.disabled.focus,.btn-success.disabled:active,.btn-success.disabled:focus,.btn-success.disabled:hover,.btn-success[disabled],.btn-success[disabled].active,.btn-success[disabled].focus,.btn-success[disabled]:active,.btn-success[disabled]:focus,.btn-success[disabled]:hover,fieldset[disabled] .btn-success,fieldset[disabled] .btn-success.active,fieldset[disabled] .btn-success.focus,fieldset[disabled] .btn-success:active,fieldset[disabled] .btn-success:focus,fieldset[disabled] .btn-success:hover{background-color:#5cb85c;border-color:#4cae4c}.btn-success .badge{color:#5cb85c;background-color:#fff}.btn-info{color:#fff;background-color:#5bc0de;border-color:#46b8da}.btn-info.active,.btn-info.focus,.btn-info:active,.btn-info:focus,.btn-info:hover,.open>.dropdown-toggle.btn-info{color:#fff;background-color:#31b0d5;border-color:#269abc}.btn-info.active,.btn-info:active,.open>.dropdown-toggle.btn-info{background-image:none}.btn-info.disabled,.btn-info.disabled.active,.btn-info.disabled.focus,.btn-info.disabled:active,.btn-info.disabled:focus,.btn-info.disabled:hover,.btn-info[disabled],.btn-info[disabled].active,.btn-info[disabled].focus,.btn-info[disabled]:active,.btn-info[disabled]:focus,.btn-info[disabled]:hover,fieldset[disabled] .btn-info,fieldset[disabled] .btn-info.active,fieldset[disabled] .btn-info.focus,fieldset[disabled] .btn-info:active,fieldset[disabled] .btn-info:focus,fieldset[disabled] .btn-info:hover{background-color:#5bc0de;border-color:#46b8da}.btn-info .badge{color:#5bc0de;background-color:#fff}.btn-warning{color:#fff;background-color:#f0ad4e;border-color:#eea236}.btn-warning.active,.btn-warning.focus,.btn-warning:active,.btn-warning:focus,.btn-warning:hover,.open>.dropdown-toggle.btn-warning{color:#fff;background-color:#ec971f;border-color:#d58512}.btn-warning.active,.btn-warning:active,.open>.dropdown-toggle.btn-warning{background-image:none}.btn-warning.disabled,.btn-warning.disabled.active,.btn-warning.disabled.focus,.btn-warning.disabled:active,.btn-warning.disabled:focus,.btn-warning.disabled:hover,.btn-warning[disabled],.btn-warning[disabled].active,.btn-warning[disabled].focus,.btn-warning[disabled]:active,.btn-warning[disabled]:focus,.btn-warning[disabled]:hover,fieldset[disabled] .btn-warning,fieldset[disabled] .btn-warning.active,fieldset[disabled] .btn-warning.focus,fieldset[disabled] .btn-warning:active,fieldset[disabled] .btn-warning:focus,fieldset[disabled] .btn-warning:hover{background-color:#f0ad4e;border-color:#eea236}.btn-warning .badge{color:#f0ad4e;background-color:#fff}.btn-danger{color:#fff;background-color:#d9534f;border-color:#d43f3a}.btn-danger.active,.btn-danger.focus,.btn-danger:active,.btn-danger:focus,.btn-danger:hover,.open>.dropdown-toggle.btn-danger{color:#fff;background-color:#c9302c;border-color:#ac2925}.btn-danger.active,.btn-danger:active,.open>.dropdown-toggle.btn-danger{background-image:none}.btn-danger.disabled,.btn-danger.disabled.active,.btn-danger.disabled.focus,.btn-danger.disabled:active,.btn-danger.disabled:focus,.btn-danger.disabled:hover,.btn-danger[disabled],.btn-danger[disabled].active,.btn-danger[disabled].focus,.btn-danger[disabled]:active,.btn-danger[disabled]:focus,.btn-danger[disabled]:hover,fieldset[disabled] .btn-danger,fieldset[disabled] .btn-danger.active,fieldset[disabled] .btn-danger.focus,fieldset[disabled] .btn-danger:active,fieldset[disabled] .btn-danger:focus,fieldset[disabled] .btn-danger:hover{background-color:#d9534f;border-color:#d43f3a}.btn-danger .badge{color:#d9534f;background-color:#fff}.btn-link{font-weight:400;color:#337ab7;border-radius:0}.btn-link,.btn-link.active,.btn-link:active,.btn-link[disabled],fieldset[disabled] .btn-link{background-color:transparent;-webkit-box-shadow:none;box-shadow:none}.btn-link,.btn-link:active,.btn-link:focus,.btn-link:hover{border-color:transparent}.btn-link:focus,.btn-link:hover{color:#23527c;text-decoration:underline;background-color:transparent}.btn-link[disabled]:focus,.btn-link[disabled]:hover,fieldset[disabled] .btn-link:focus,fieldset[disabled] .btn-link:hover{color:#777;text-decoration:none}.btn-group-lg>.btn,.btn-lg{padding:10px 16px;font-size:18px;line-height:1.3333333;border-radius:6px}.btn-group-sm>.btn,.btn-sm{padding:5px 10px;font-size:12px;line-height:1.5;border-radius:3px}.btn-group-xs>.btn,.btn-xs{padding:1px 5px;font-size:12px;line-height:1.5;border-radius:3px}.btn-block{display:block;width:100%}.btn-block+.btn-block{margin-top:5px}input[type=button].btn-block,input[type=reset].btn-block,input[type=submit].btn-block{width:100%}.fade{opacity:0;-webkit-transition:opacity .15s linear;-o-transition:opacity .15s linear;transition:opacity .15s linear}.fade.in{opacity:1}.collapse{display:none}.collapse.in{display:block}tr.collapse.in{display:table-row}tbody.collapse.in{display:table-row-group}.collapsing{position:relative;height:0;overflow:hidden;-webkit-transition-timing-function:ease;-o-transition-timing-function:ease;transition-timing-function:ease;-webkit-transition-duration:.35s;-o-transition-duration:.35s;transition-duration:.35s;-webkit-transition-property:height,visibility;-o-transition-property:height,visibility;transition-property:height,visibility}.caret{display:inline-block;width:0;height:0;margin-left:2px;vertical-align:middle;border-top:4px dashed;border-right:4px solid transparent;border-left:4px solid transparent}.dropdown,.dropup{position:relative}.dropdown-toggle:focus{outline:0}.dropdown-menu{position:absolute;top:100%;left:0;z-index:1000;display:none;float:left;min-width:160px;padding:5px 0;margin:2px 0 0;font-size:14px;text-align:left;list-style:none;background-color:#fff;-webkit-background-clip:padding-box;background-clip:padding-box;border:1px solid #ccc;border:1px solid rgba(0,0,0,.15);border-radius:4px;-webkit-box-shadow:0 6px 12px rgba(0,0,0,.175);box-shadow:0 6px 12px rgba(0,0,0,.175)}.dropdown-menu.pull-right{right:0;left:auto}.dropdown-menu .divider{height:1px;margin:9px 0;overflow:hidden;background-color:#e5e5e5}.dropdown-menu>li>a{display:block;padding:3px 20px;clear:both;font-weight:400;line-height:1.42857143;color:#333;white-space:nowrap}.dropdown-menu>li>a:focus,.dropdown-menu>li>a:hover{color:#262626;text-decoration:none;background-color:#f5f5f5}.dropdown-menu>.active>a,.dropdown-menu>.active>a:focus,.dropdown-menu>.active>a:hover{color:#fff;text-decoration:none;background-color:#337ab7;outline:0}.dropdown-menu>.disabled>a,.dropdown-menu>.disabled>a:focus,.dropdown-menu>.disabled>a:hover{color:#777}.dropdown-menu>.disabled>a:focus,.dropdown-menu>.disabled>a:hover{text-decoration:none;cursor:not-allowed;background-color:transparent;background-image:none;filter:progid:DXImageTransform.Microsoft.gradient(enabled=false)}.open>.dropdown-menu{display:block}.open>a{outline:0}.dropdown-menu-right{right:0;left:auto}.dropdown-menu-left{right:auto;left:0}.dropdown-header{display:block;padding:3px 20px;font-size:12px;line-height:1.42857143;color:#777;white-space:nowrap}.dropdown-backdrop{position:fixed;top:0;right:0;bottom:0;left:0;z-index:990}.pull-right>.dropdown-menu{right:0;left:auto}.dropup .caret,.navbar-fixed-bottom .dropdown .caret{content:"";border-top:0;border-bottom:4px solid}.dropup .dropdown-menu,.navbar-fixed-bottom .dropdown .dropdown-menu{top:auto;bottom:100%;margin-bottom:2px}@media (min-width:768px){.navbar-right .dropdown-menu{right:0;left:auto}.navbar-right .dropdown-menu-left{right:auto;left:0}}.btn-group,.btn-group-vertical{position:relative;display:inline-block;vertical-align:middle}.btn-group-vertical>.btn,.btn-group>.btn{position:relative;float:left}.btn-group-vertical>.btn.active,.btn-group-vertical>.btn:active,.btn-group-vertical>.btn:focus,.btn-group-vertical>.btn:hover,.btn-group>.btn.active,.btn-group>.btn:active,.btn-group>.btn:focus,.btn-group>.btn:hover{z-index:2}.btn-group .btn+.btn,.btn-group .btn+.btn-group,.btn-group .btn-group+.btn,.btn-group .btn-group+.btn-group{margin-left:-1px}.btn-toolbar{margin-left:-5px}.btn-toolbar .btn-group,.btn-toolbar .input-group{float:left}.btn-toolbar>.btn,.btn-toolbar>.btn-group,.btn-toolbar>.input-group{margin-left:5px}.btn-group>.btn:not(:first-child):not(:last-child):not(.dropdown-toggle){border-radius:0}.btn-group>.btn:first-child{margin-left:0}.btn-group>.btn:first-child:not(:last-child):not(.dropdown-toggle){border-top-right-radius:0;border-bottom-right-radius:0}.btn-group>.btn:last-child:not(:first-child),.btn-group>.dropdown-toggle:not(:first-child){border-top-left-radius:0;border-bottom-left-radius:0}.btn-group>.btn-group{float:left}.btn-group>.btn-group:not(:first-child):not(:last-child)>.btn{border-radius:0}.btn-group>.btn-group:first-child:not(:last-child)>.btn:last-child,.btn-group>.btn-group:first-child:not(:last-child)>.dropdown-toggle{border-top-right-radius:0;border-bottom-right-radius:0}.btn-group>.btn-group:last-child:not(:first-child)>.btn:first-child{border-top-left-radius:0;border-bottom-left-radius:0}.btn-group .dropdown-toggle:active,.btn-group.open .dropdown-toggle{outline:0}.btn-group>.btn+.dropdown-toggle{padding-right:8px;padding-left:8px}.btn-group>.btn-lg+.dropdown-toggle{padding-right:12px;padding-left:12px}.btn-group.open .dropdown-toggle{-webkit-box-shadow:inset 0 3px 5px rgba(0,0,0,.125);box-shadow:inset 0 3px 5px rgba(0,0,0,.125)}.btn-group.open .dropdown-toggle.btn-link{-webkit-box-shadow:none;box-shadow:none}.btn .caret{margin-left:0}.btn-lg .caret{border-width:5px 5px 0;border-bottom-width:0}.dropup .btn-lg .caret{border-width:0 5px 5px}.btn-group-vertical>.btn,.btn-group-vertical>.btn-group,.btn-group-vertical>.btn-group>.btn{display:block;float:none;width:100%;max-width:100%}.btn-group-vertical>.btn-group>.btn{float:none}.btn-group-vertical>.btn+.btn,.btn-group-vertical>.btn+.btn-group,.btn-group-vertical>.btn-group+.btn,.btn-group-vertical>.btn-group+.btn-group{margin-top:-1px;margin-left:0}.btn-group-vertical>.btn:not(:first-child):not(:last-child){border-radius:0}.btn-group-vertical>.btn:first-child:not(:last-child){border-top-right-radius:4px;border-bottom-right-radius:0;border-bottom-left-radius:0}.btn-group-vertical>.btn:last-child:not(:first-child){border-top-left-radius:0;border-top-right-radius:0;border-bottom-left-radius:4px}.btn-group-vertical>.btn-group:not(:first-child):not(:last-child)>.btn{border-radius:0}.btn-group-vertical>.btn-group:first-child:not(:last-child)>.btn:last-child,.btn-group-vertical>.btn-group:first-child:not(:last-child)>.dropdown-toggle{border-bottom-right-radius:0;border-bottom-left-radius:0}.btn-group-vertical>.btn-group:last-child:not(:first-child)>.btn:first-child{border-top-left-radius:0;border-top-right-radius:0}.btn-group-justified{display:table;width:100%;table-layout:fixed;border-collapse:separate}.btn-group-justified>.btn,.btn-group-justified>.btn-group{display:table-cell;float:none;width:1%}.btn-group-justified>.btn-group .btn{width:100%}.btn-group-justified>.btn-group .dropdown-menu{left:auto}[data-toggle=buttons]>.btn input[type=checkbox],[data-toggle=buttons]>.btn input[type=radio],[data-toggle=buttons]>.btn-group>.btn input[type=checkbox],[data-toggle=buttons]>.btn-group>.btn input[type=radio]{position:absolute;clip:rect(0,0,0,0);pointer-events:none}.input-group{position:relative;display:table;border-collapse:separate}.input-group[class*=col-]{float:none;padding-right:0;padding-left:0}.input-group .form-control{position:relative;z-index:2;float:left;width:100%;margin-bottom:0}.input-group-lg>.form-control,.input-group-lg>.input-group-addon,.input-group-lg>.input-group-btn>.btn{height:46px;padding:10px 16px;font-size:18px;line-height:1.3333333;border-radius:6px}select.input-group-lg>.form-control,select.input-group-lg>.input-group-addon,select.input-group-lg>.input-group-btn>.btn{height:46px;line-height:46px}select[multiple].input-group-lg>.form-control,select[multiple].input-group-lg>.input-group-addon,select[multiple].input-group-lg>.input-group-btn>.btn,textarea.input-group-lg>.form-control,textarea.input-group-lg>.input-group-addon,textarea.input-group-lg>.input-group-btn>.btn{height:auto}.input-group-sm>.form-control,.input-group-sm>.input-group-addon,.input-group-sm>.input-group-btn>.btn{height:30px;padding:5px 10px;font-size:12px;line-height:1.5;border-radius:3px}select.input-group-sm>.form-control,select.input-group-sm>.input-group-addon,select.input-group-sm>.input-group-btn>.btn{height:30px;line-height:30px}select[multiple].input-group-sm>.form-control,select[multiple].input-group-sm>.input-group-addon,select[multiple].input-group-sm>.input-group-btn>.btn,textarea.input-group-sm>.form-control,textarea.input-group-sm>.input-group-addon,textarea.input-group-sm>.input-group-btn>.btn{height:auto}.input-group .form-control,.input-group-addon,.input-group-btn{display:table-cell}.input-group .form-control:not(:first-child):not(:last-child),.input-group-addon:not(:first-child):not(:last-child),.input-group-btn:not(:first-child):not(:last-child){border-radius:0}.input-group-addon,.input-group-btn{width:1%;white-space:nowrap;vertical-align:middle}.input-group-addon{padding:6px 12px;font-size:14px;font-weight:400;line-height:1;color:#555;text-align:center;background-color:#eee;border:1px solid #ccc;border-radius:4px}.input-group-addon.input-sm{padding:5px 10px;font-size:12px;border-radius:3px}.input-group-addon.input-lg{padding:10px 16px;font-size:18px;border-radius:6px}.input-group-addon input[type=checkbox],.input-group-addon input[type=radio]{margin-top:0}.input-group .form-control:first-child,.input-group-addon:first-child,.input-group-btn:first-child>.btn,.input-group-btn:first-child>.btn-group>.btn,.input-group-btn:first-child>.dropdown-toggle,.input-group-btn:last-child>.btn-group:not(:last-child)>.btn,.input-group-btn:last-child>.btn:not(:last-child):not(.dropdown-toggle){border-top-right-radius:0;border-bottom-right-radius:0}.input-group-addon:first-child{border-right:0}.input-group .form-control:last-child,.input-group-addon:last-child,.input-group-btn:first-child>.btn-group:not(:first-child)>.btn,.input-group-btn:first-child>.btn:not(:first-child),.input-group-btn:last-child>.btn,.input-group-btn:last-child>.btn-group>.btn,.input-group-btn:last-child>.dropdown-toggle{border-top-left-radius:0;border-bottom-left-radius:0}.input-group-addon:last-child{border-left:0}.input-group-btn{position:relative;font-size:0;white-space:nowrap}.input-group-btn>.btn{position:relative}.input-group-btn>.btn+.btn{margin-left:-1px}.input-group-btn>.btn:active,.input-group-btn>.btn:focus,.input-group-btn>.btn:hover{z-index:2}.input-group-btn:first-child>.btn,.input-group-btn:first-child>.btn-group{margin-right:-1px}.input-group-btn:last-child>.btn,.input-group-btn:last-child>.btn-group{margin-left:-1px}.nav{padding-left:0;margin-bottom:0;list-style:none}.nav>li{position:relative;display:block}.nav>li>a{position:relative;display:block;padding:10px 15px}.nav>li>a:focus,.nav>li>a:hover{text-decoration:none;background-color:#eee}.nav>li.disabled>a{color:#777}.nav>li.disabled>a:focus,.nav>li.disabled>a:hover{color:#777;text-decoration:none;cursor:not-allowed;background-color:transparent}.nav .open>a,.nav .open>a:focus,.nav .open>a:hover{background-color:#eee;border-color:#337ab7}.nav .nav-divider{height:1px;margin:9px 0;overflow:hidden;background-color:#e5e5e5}.nav>li>a>img{max-width:none}.nav-tabs{border-bottom:1px solid #ddd}.nav-tabs>li{float:left;margin-bottom:-1px}.nav-tabs>li>a{margin-right:2px;line-height:1.42857143;border:1px solid transparent;border-radius:4px 4px 0 0}.nav-tabs>li>a:hover{border-color:#eee #eee #ddd}.nav-tabs>li.active>a,.nav-tabs>li.active>a:focus,.nav-tabs>li.active>a:hover{color:#555;cursor:default;background-color:#fff;border:1px solid #ddd;border-bottom-color:transparent}.nav-tabs.nav-justified{width:100%;border-bottom:0}.nav-tabs.nav-justified>li{float:none}.nav-tabs.nav-justified>li>a{margin-bottom:5px;text-align:center}.nav-tabs.nav-justified>.dropdown .dropdown-menu{top:auto;left:auto}@media (min-width:768px){.nav-tabs.nav-justified>li{display:table-cell;width:1%}.nav-tabs.nav-justified>li>a{margin-bottom:0}}.nav-tabs.nav-justified>li>a{margin-right:0;border-radius:4px}.nav-tabs.nav-justified>.active>a,.nav-tabs.nav-justified>.active>a:focus,.nav-tabs.nav-justified>.active>a:hover{border:1px solid #ddd}@media (min-width:768px){.nav-tabs.nav-justified>li>a{border-bottom:1px solid #ddd;border-radius:4px 4px 0 0}.nav-tabs.nav-justified>.active>a,.nav-tabs.nav-justified>.active>a:focus,.nav-tabs.nav-justified>.active>a:hover{border-bottom-color:#fff}}.nav-pills>li{float:left}.nav-pills>li>a{border-radius:4px}.nav-pills>li+li{margin-left:2px}.nav-pills>li.active>a,.nav-pills>li.active>a:focus,.nav-pills>li.active>a:hover{color:#fff;background-color:#337ab7}.nav-stacked>li{float:none}.nav-stacked>li+li{margin-top:2px;margin-left:0}.nav-justified{width:100%}.nav-justified>li{float:none}.nav-justified>li>a{margin-bottom:5px;text-align:center}.nav-justified>.dropdown .dropdown-menu{top:auto;left:auto}@media (min-width:768px){.nav-justified>li{display:table-cell;width:1%}.nav-justified>li>a{margin-bottom:0}}.nav-tabs-justified{border-bottom:0}.nav-tabs-justified>li>a{margin-right:0;border-radius:4px}.nav-tabs-justified>.active>a,.nav-tabs-justified>.active>a:focus,.nav-tabs-justified>.active>a:hover{border:1px solid #ddd}@media (min-width:768px){.nav-tabs-justified>li>a{border-bottom:1px solid #ddd;border-radius:4px 4px 0 0}.nav-tabs-justified>.active>a,.nav-tabs-justified>.active>a:focus,.nav-tabs-justified>.active>a:hover{border-bottom-color:#fff}}.tab-content>.tab-pane{display:none}.tab-content>.active{display:block}.nav-tabs .dropdown-menu{margin-top:-1px;border-top-left-radius:0;border-top-right-radius:0}.navbar{position:relative;min-height:50px;margin-bottom:20px;border:1px solid transparent}@media (min-width:768px){.navbar{border-radius:4px}}@media (min-width:768px){.navbar-header{float:left}}.navbar-collapse{padding-right:15px;padding-left:15px;overflow-x:visible;-webkit-overflow-scrolling:touch;border-top:1px solid transparent;-webkit-box-shadow:inset 0 1px 0 rgba(255,255,255,.1);box-shadow:inset 0 1px 0 rgba(255,255,255,.1)}.navbar-collapse.in{overflow-y:auto}@media (min-width:768px){.navbar-collapse{width:auto;border-top:0;-webkit-box-shadow:none;box-shadow:none}.navbar-collapse.collapse{display:block!important;height:auto!important;padding-bottom:0;overflow:visible!important}.navbar-collapse.in{overflow-y:visible}.navbar-fixed-bottom .navbar-collapse,.navbar-fixed-top .navbar-collapse,.navbar-static-top .navbar-collapse{padding-right:0;padding-left:0}}.navbar-fixed-bottom .navbar-collapse,.navbar-fixed-top .navbar-collapse{max-height:340px}@media (max-device-width:480px)and (orientation:landscape){.navbar-fixed-bottom .navbar-collapse,.navbar-fixed-top .navbar-collapse{max-height:200px}}.container-fluid>.navbar-collapse,.container-fluid>.navbar-header,.container>.navbar-collapse,.container>.navbar-header{margin-right:-15px;margin-left:-15px}@media (min-width:768px){.container-fluid>.navbar-collapse,.container-fluid>.navbar-header,.container>.navbar-collapse,.container>.navbar-header{margin-right:0;margin-left:0}}.navbar-static-top{z-index:1000;border-width:0 0 1px}@media (min-width:768px){.navbar-static-top{border-radius:0}}.navbar-fixed-bottom,.navbar-fixed-top{position:fixed;right:0;left:0;z-index:1030}@media (min-width:768px){.navbar-fixed-bottom,.navbar-fixed-top{border-radius:0}}.navbar-fixed-top{top:0;border-width:0 0 1px}.navbar-fixed-bottom{bottom:0;margin-bottom:0;border-width:1px 0 0}.navbar-brand{float:left;height:50px;padding:15px 15px;font-size:18px;line-height:20px}.navbar-brand:focus,.navbar-brand:hover{text-decoration:none}.navbar-brand>img{display:block}@media (min-width:768px){.navbar>.container .navbar-brand,.navbar>.container-fluid .navbar-brand{margin-left:-15px}}.navbar-toggle{position:relative;float:right;padding:9px 10px;margin-top:8px;margin-right:15px;margin-bottom:8px;background-color:transparent;background-image:none;border:1px solid transparent;border-radius:4px}.navbar-toggle:focus{outline:0}.navbar-toggle .icon-bar{display:block;width:22px;height:2px;border-radius:1px}.navbar-toggle .icon-bar+.icon-bar{margin-top:4px}@media (min-width:768px){.navbar-toggle{display:none}}.navbar-nav{margin:7.5px -15px}.navbar-nav>li>a{padding-top:10px;padding-bottom:10px;line-height:20px}@media (max-width:767px){.navbar-nav .open .dropdown-menu{position:static;float:none;width:auto;margin-top:0;background-color:transparent;border:0;-webkit-box-shadow:none;box-shadow:none}.navbar-nav .open .dropdown-menu .dropdown-header,.navbar-nav .open .dropdown-menu>li>a{padding:5px 15px 5px 25px}.navbar-nav .open .dropdown-menu>li>a{line-height:20px}.navbar-nav .open .dropdown-menu>li>a:focus,.navbar-nav .open .dropdown-menu>li>a:hover{background-image:none}}@media (min-width:768px){.navbar-nav{float:left;margin:0}.navbar-nav>li{float:left}.navbar-nav>li>a{padding-top:15px;padding-bottom:15px}}.navbar-form{padding:10px 15px;margin-top:8px;margin-right:-15px;margin-bottom:8px;margin-left:-15px;border-top:1px solid transparent;border-bottom:1px solid transparent;-webkit-box-shadow:inset 0 1px 0 rgba(255,255,255,.1),0 1px 0 rgba(255,255,255,.1);box-shadow:inset 0 1px 0 rgba(255,255,255,.1),0 1px 0 rgba(255,255,255,.1)}@media (min-width:768px){.navbar-form .form-group{display:inline-block;margin-bottom:0;vertical-align:middle}.navbar-form .form-control{display:inline-block;width:auto;vertical-align:middle}.navbar-form .form-control-static{display:inline-block}.navbar-form .input-group{display:inline-table;vertical-align:middle}.navbar-form .input-group .form-control,.navbar-form .input-group .input-group-addon,.navbar-form .input-group .input-group-btn{width:auto}.navbar-form .input-group>.form-control{width:100%}.navbar-form .control-label{margin-bottom:0;vertical-align:middle}.navbar-form .checkbox,.navbar-form .radio{display:inline-block;margin-top:0;margin-bottom:0;vertical-align:middle}.navbar-form .checkbox label,.navbar-form .radio label{padding-left:0}.navbar-form .checkbox input[type=checkbox],.navbar-form .radio input[type=radio]{position:relative;margin-left:0}.navbar-form .has-feedback .form-control-feedback{top:0}}@media (max-width:767px){.navbar-form .form-group{margin-bottom:5px}.navbar-form .form-group:last-child{margin-bottom:0}}@media (min-width:768px){.navbar-form{width:auto;padding-top:0;padding-bottom:0;margin-right:0;margin-left:0;border:0;-webkit-box-shadow:none;box-shadow:none}}.navbar-nav>li>.dropdown-menu{margin-top:0;border-top-left-radius:0;border-top-right-radius:0}.navbar-fixed-bottom .navbar-nav>li>.dropdown-menu{margin-bottom:0;border-top-left-radius:4px;border-top-right-radius:4px;border-bottom-right-radius:0;border-bottom-left-radius:0}.navbar-btn{margin-top:8px;margin-bottom:8px}.navbar-btn.btn-sm{margin-top:10px;margin-bottom:10px}.navbar-btn.btn-xs{margin-top:14px;margin-bottom:14px}.navbar-text{margin-top:15px;margin-bottom:15px}@media (min-width:768px){.navbar-text{float:left;margin-right:15px;margin-left:15px}}@media (min-width:768px){.navbar-left{float:left!important}.navbar-right{float:right!important;margin-right:-15px}.navbar-right~.navbar-right{margin-right:0}}.navbar-default{background-color:#f8f8f8;border-color:#e7e7e7}.navbar-default .navbar-brand{color:#777}.navbar-default .navbar-brand:focus,.navbar-default .navbar-brand:hover{color:#5e5e5e;background-color:transparent}.navbar-default .navbar-text{color:#777}.navbar-default .navbar-nav>li>a{color:#777}.navbar-default .navbar-nav>li>a:focus,.navbar-default .navbar-nav>li>a:hover{color:#333;background-color:transparent}.navbar-default .navbar-nav>.active>a,.navbar-default .navbar-nav>.active>a:focus,.navbar-default .navbar-nav>.active>a:hover{color:#555;background-color:#e7e7e7}.navbar-default .navbar-nav>.disabled>a,.navbar-default .navbar-nav>.disabled>a:focus,.navbar-default .navbar-nav>.disabled>a:hover{color:#ccc;background-color:transparent}.navbar-default .navbar-toggle{border-color:#ddd}.navbar-default .navbar-toggle:focus,.navbar-default .navbar-toggle:hover{background-color:#ddd}.navbar-default .navbar-toggle .icon-bar{background-color:#888}.navbar-default .navbar-collapse,.navbar-default .navbar-form{border-color:#e7e7e7}.navbar-default .navbar-nav>.open>a,.navbar-default .navbar-nav>.open>a:focus,.navbar-default .navbar-nav>.open>a:hover{color:#555;background-color:#e7e7e7}@media (max-width:767px){.navbar-default .navbar-nav .open .dropdown-menu>li>a{color:#777}.navbar-default .navbar-nav .open .dropdown-menu>li>a:focus,.navbar-default .navbar-nav .open .dropdown-menu>li>a:hover{color:#333;background-color:transparent}.navbar-default .navbar-nav .open .dropdown-menu>.active>a,.navbar-default .navbar-nav .open .dropdown-menu>.active>a:focus,.navbar-default .navbar-nav .open .dropdown-menu>.active>a:hover{color:#555;background-color:#e7e7e7}.navbar-default .navbar-nav .open .dropdown-menu>.disabled>a,.navbar-default .navbar-nav .open .dropdown-menu>.disabled>a:focus,.navbar-default .navbar-nav .open .dropdown-menu>.disabled>a:hover{color:#ccc;background-color:transparent}}.navbar-default .navbar-link{color:#777}.navbar-default .navbar-link:hover{color:#333}.navbar-default .btn-link{color:#777}.navbar-default .btn-link:focus,.navbar-default .btn-link:hover{color:#333}.navbar-default .btn-link[disabled]:focus,.navbar-default .btn-link[disabled]:hover,fieldset[disabled] .navbar-default .btn-link:focus,fieldset[disabled] .navbar-default .btn-link:hover{color:#ccc}.navbar-inverse{background-color:#222;border-color:#080808}.navbar-inverse .navbar-brand{color:#9d9d9d}.navbar-inverse .navbar-brand:focus,.navbar-inverse .navbar-brand:hover{color:#fff;background-color:transparent}.navbar-inverse .navbar-text{color:#9d9d9d}.navbar-inverse .navbar-nav>li>a{color:#9d9d9d}.navbar-inverse .navbar-nav>li>a:focus,.navbar-inverse .navbar-nav>li>a:hover{color:#fff;background-color:transparent}.navbar-inverse .navbar-nav>.active>a,.navbar-inverse .navbar-nav>.active>a:focus,.navbar-inverse .navbar-nav>.active>a:hover{color:#fff;background-color:#080808}.navbar-inverse .navbar-nav>.disabled>a,.navbar-inverse .navbar-nav>.disabled>a:focus,.navbar-inverse .navbar-nav>.disabled>a:hover{color:#444;background-color:transparent}.navbar-inverse .navbar-toggle{border-color:#333}.navbar-inverse .navbar-toggle:focus,.navbar-inverse .navbar-toggle:hover{background-color:#333}.navbar-inverse .navbar-toggle .icon-bar{background-color:#fff}.navbar-inverse .navbar-collapse,.navbar-inverse .navbar-form{border-color:#101010}.navbar-inverse .navbar-nav>.open>a,.navbar-inverse .navbar-nav>.open>a:focus,.navbar-inverse .navbar-nav>.open>a:hover{color:#fff;background-color:#080808}@media (max-width:767px){.navbar-inverse .navbar-nav .open .dropdown-menu>.dropdown-header{border-color:#080808}.navbar-inverse .navbar-nav .open .dropdown-menu .divider{background-color:#080808}.navbar-inverse .navbar-nav .open .dropdown-menu>li>a{color:#9d9d9d}.navbar-inverse .navbar-nav .open .dropdown-menu>li>a:focus,.navbar-inverse .navbar-nav .open .dropdown-menu>li>a:hover{color:#fff;background-color:transparent}.navbar-inverse .navbar-nav .open .dropdown-menu>.active>a,.navbar-inverse .navbar-nav .open .dropdown-menu>.active>a:focus,.navbar-inverse .navbar-nav .open .dropdown-menu>.active>a:hover{color:#fff;background-color:#080808}.navbar-inverse .navbar-nav .open .dropdown-menu>.disabled>a,.navbar-inverse .navbar-nav .open .dropdown-menu>.disabled>a:focus,.navbar-inverse .navbar-nav .open .dropdown-menu>.disabled>a:hover{color:#444;background-color:transparent}}.navbar-inverse .navbar-link{color:#9d9d9d}.navbar-inverse .navbar-link:hover{color:#fff}.navbar-inverse .btn-link{color:#9d9d9d}.navbar-inverse .btn-link:focus,.navbar-inverse .btn-link:hover{color:#fff}.navbar-inverse .btn-link[disabled]:focus,.navbar-inverse .btn-link[disabled]:hover,fieldset[disabled] .navbar-inverse .btn-link:focus,fieldset[disabled] .navbar-inverse .btn-link:hover{color:#444}.breadcrumb{padding:8px 15px;margin-bottom:20px;list-style:none;background-color:#f5f5f5;border-radius:4px}.breadcrumb>li{display:inline-block}.breadcrumb>li+li:before{padding:0 5px;color:#ccc;content:"/\00a0"}.breadcrumb>.active{color:#777}.pagination{display:inline-block;padding-left:0;margin:20px 0;border-radius:4px}.pagination>li{display:inline}.pagination>li>a,.pagination>li>span{position:relative;float:left;padding:6px 12px;margin-left:-1px;line-height:1.42857143;color:#337ab7;text-decoration:none;background-color:#fff;border:1px solid #ddd}.pagination>li:first-child>a,.pagination>li:first-child>span{margin-left:0;border-top-left-radius:4px;border-bottom-left-radius:4px}.pagination>li:last-child>a,.pagination>li:last-child>span{border-top-right-radius:4px;border-bottom-right-radius:4px}.pagination>li>a:focus,.pagination>li>a:hover,.pagination>li>span:focus,.pagination>li>span:hover{color:#23527c;background-color:#eee;border-color:#ddd}.pagination>.active>a,.pagination>.active>a:focus,.pagination>.active>a:hover,.pagination>.active>span,.pagination>.active>span:focus,.pagination>.active>span:hover{z-index:2;color:#fff;cursor:default;background-color:#337ab7;border-color:#337ab7}.pagination>.disabled>a,.pagination>.disabled>a:focus,.pagination>.disabled>a:hover,.pagination>.disabled>span,.pagination>.disabled>span:focus,.pagination>.disabled>span:hover{color:#777;cursor:not-allowed;background-color:#fff;border-color:#ddd}.pagination-lg>li>a,.pagination-lg>li>span{padding:10px 16px;font-size:18px}.pagination-lg>li:first-child>a,.pagination-lg>li:first-child>span{border-top-left-radius:6px;border-bottom-left-radius:6px}.pagination-lg>li:last-child>a,.pagination-lg>li:last-child>span{border-top-right-radius:6px;border-bottom-right-radius:6px}.pagination-sm>li>a,.pagination-sm>li>span{padding:5px 10px;font-size:12px}.pagination-sm>li:first-child>a,.pagination-sm>li:first-child>span{border-top-left-radius:3px;border-bottom-left-radius:3px}.pagination-sm>li:last-child>a,.pagination-sm>li:last-child>span{border-top-right-radius:3px;border-bottom-right-radius:3px}.pager{padding-left:0;margin:20px 0;text-align:center;list-style:none}.pager li{display:inline}.pager li>a,.pager li>span{display:inline-block;padding:5px 14px;background-color:#fff;border:1px solid #ddd;border-radius:15px}.pager li>a:focus,.pager li>a:hover{text-decoration:none;background-color:#eee}.pager .next>a,.pager .next>span{float:right}.pager .previous>a,.pager .previous>span{float:left}.pager .disabled>a,.pager .disabled>a:focus,.pager .disabled>a:hover,.pager .disabled>span{color:#777;cursor:not-allowed;background-color:#fff}.label{display:inline;padding:.2em .6em .3em;font-size:75%;font-weight:700;line-height:1;color:#fff;text-align:center;white-space:nowrap;vertical-align:baseline;border-radius:.25em}a.label:focus,a.label:hover{color:#fff;text-decoration:none;cursor:pointer}.label:empty{display:none}.btn .label{position:relative;top:-1px}.label-default{background-color:#777}.label-default[href]:focus,.label-default[href]:hover{background-color:#5e5e5e}.label-primary{background-color:#337ab7}.label-primary[href]:focus,.label-primary[href]:hover{background-color:#286090}.label-success{background-color:#5cb85c}.label-success[href]:focus,.label-success[href]:hover{background-color:#449d44}.label-info{background-color:#5bc0de}.label-info[href]:focus,.label-info[href]:hover{background-color:#31b0d5}.label-warning{background-color:#f0ad4e}.label-warning[href]:focus,.label-warning[href]:hover{background-color:#ec971f}.label-danger{background-color:#d9534f}.label-danger[href]:focus,.label-danger[href]:hover{background-color:#c9302c}.badge{display:inline-block;min-width:10px;padding:3px 7px;font-size:12px;font-weight:700;line-height:1;color:#fff;text-align:center;white-space:nowrap;vertical-align:baseline;background-color:#777;border-radius:10px}.badge:empty{display:none}.btn .badge{position:relative;top:-1px}.btn-group-xs>.btn .badge,.btn-xs .badge{top:0;padding:1px 5px}a.badge:focus,a.badge:hover{color:#fff;text-decoration:none;cursor:pointer}.list-group-item.active>.badge,.nav-pills>.active>a>.badge{color:#337ab7;background-color:#fff}.list-group-item>.badge{float:right}.list-group-item>.badge+.badge{margin-right:5px}.nav-pills>li>a>.badge{margin-left:3px}.jumbotron{padding:30px 15px;margin-bottom:30px;color:inherit;background-color:#eee}.jumbotron .h1,.jumbotron h1{color:inherit}.jumbotron p{margin-bottom:15px;font-size:21px;font-weight:200}.jumbotron>hr{border-top-color:#d5d5d5}.container .jumbotron,.container-fluid .jumbotron{border-radius:6px}.jumbotron .container{max-width:100%}@media screen and (min-width:768px){.jumbotron{padding:48px 0}.container .jumbotron,.container-fluid .jumbotron{padding-right:60px;padding-left:60px}.jumbotron .h1,.jumbotron h1{font-size:63px}}.thumbnail{display:block;padding:4px;margin-bottom:20px;line-height:1.42857143;background-color:#fff;border:1px solid #ddd;border-radius:4px;-webkit-transition:border .2s ease-in-out;-o-transition:border .2s ease-in-out;transition:border .2s ease-in-out}.thumbnail a>img,.thumbnail>img{margin-right:auto;margin-left:auto}a.thumbnail.active,a.thumbnail:focus,a.thumbnail:hover{border-color:#337ab7}.thumbnail .caption{padding:9px;color:#333}.alert{padding:15px;margin-bottom:20px;border:1px solid transparent;border-radius:4px}.alert h4{margin-top:0;color:inherit}.alert .alert-link{font-weight:700}.alert>p,.alert>ul{margin-bottom:0}.alert>p+p{margin-top:5px}.alert-dismissable,.alert-dismissible{padding-right:35px}.alert-dismissable .close,.alert-dismissible .close{position:relative;top:-2px;right:-21px;color:inherit}.alert-success{color:#3c763d;background-color:#dff0d8;border-color:#d6e9c6}.alert-success hr{border-top-color:#c9e2b3}.alert-success .alert-link{color:#2b542c}.alert-info{color:#31708f;background-color:#d9edf7;border-color:#bce8f1}.alert-info hr{border-top-color:#a6e1ec}.alert-info .alert-link{color:#245269}.alert-warning{color:#8a6d3b;background-color:#fcf8e3;border-color:#faebcc}.alert-warning hr{border-top-color:#f7e1b5}.alert-warning .alert-link{color:#66512c}.alert-danger{color:#a94442;background-color:#f2dede;border-color:#ebccd1}.alert-danger hr{border-top-color:#e4b9c0}.alert-danger .alert-link{color:#843534}@-webkit-keyframes progress-bar-stripes{from{background-position:40px 0}to{background-position:0 0}}@-o-keyframes progress-bar-stripes{from{background-position:40px 0}to{background-position:0 0}}@keyframes progress-bar-stripes{from{background-position:40px 0}to{background-position:0 0}}.progress{height:20px;margin-bottom:20px;overflow:hidden;background-color:#f5f5f5;border-radius:4px;-webkit-box-shadow:inset 0 1px 2px rgba(0,0,0,.1);box-shadow:inset 0 1px 2px rgba(0,0,0,.1)}.progress-bar{float:left;width:0;height:100%;font-size:12px;line-height:20px;color:#fff;text-align:center;background-color:#337ab7;-webkit-box-shadow:inset 0 -1px 0 rgba(0,0,0,.15);box-shadow:inset 0 -1px 0 rgba(0,0,0,.15);-webkit-transition:width .6s ease;-o-transition:width .6s ease;transition:width .6s ease}.progress-bar-striped,.progress-striped .progress-bar{background-image:-webkit-linear-gradient(45deg,rgba(255,255,255,.15) 25%,transparent 25%,transparent 50%,rgba(255,255,255,.15) 50%,rgba(255,255,255,.15) 75%,transparent 75%,transparent);background-image:-o-linear-gradient(45deg,rgba(255,255,255,.15) 25%,transparent 25%,transparent 50%,rgba(255,255,255,.15) 50%,rgba(255,255,255,.15) 75%,transparent 75%,transparent);background-image:linear-gradient(45deg,rgba(255,255,255,.15) 25%,transparent 25%,transparent 50%,rgba(255,255,255,.15) 50%,rgba(255,255,255,.15) 75%,transparent 75%,transparent);-webkit-background-size:40px 40px;background-size:40px 40px}.progress-bar.active,.progress.active .progress-bar{-webkit-animation:progress-bar-stripes 2s linear infinite;-o-animation:progress-bar-stripes 2s linear infinite;animation:progress-bar-stripes 2s linear infinite}.progress-bar-success{background-color:#5cb85c}.progress-striped .progress-bar-success{background-image:-webkit-linear-gradient(45deg,rgba(255,255,255,.15) 25%,transparent 25%,transparent 50%,rgba(255,255,255,.15) 50%,rgba(255,255,255,.15) 75%,transparent 75%,transparent);background-image:-o-linear-gradient(45deg,rgba(255,255,255,.15) 25%,transparent 25%,transparent 50%,rgba(255,255,255,.15) 50%,rgba(255,255,255,.15) 75%,transparent 75%,transparent);background-image:linear-gradient(45deg,rgba(255,255,255,.15) 25%,transparent 25%,transparent 50%,rgba(255,255,255,.15) 50%,rgba(255,255,255,.15) 75%,transparent 75%,transparent)}.progress-bar-info{background-color:#5bc0de}.progress-striped .progress-bar-info{background-image:-webkit-linear-gradient(45deg,rgba(255,255,255,.15) 25%,transparent 25%,transparent 50%,rgba(255,255,255,.15) 50%,rgba(255,255,255,.15) 75%,transparent 75%,transparent);background-image:-o-linear-gradient(45deg,rgba(255,255,255,.15) 25%,transparent 25%,transparent 50%,rgba(255,255,255,.15) 50%,rgba(255,255,255,.15) 75%,transparent 75%,transparent);background-image:linear-gradient(45deg,rgba(255,255,255,.15) 25%,transparent 25%,transparent 50%,rgba(255,255,255,.15) 50%,rgba(255,255,255,.15) 75%,transparent 75%,transparent)}.progress-bar-warning{background-color:#f0ad4e}.progress-striped .progress-bar-warning{background-image:-webkit-linear-gradient(45deg,rgba(255,255,255,.15) 25%,transparent 25%,transparent 50%,rgba(255,255,255,.15) 50%,rgba(255,255,255,.15) 75%,transparent 75%,transparent);background-image:-o-linear-gradient(45deg,rgba(255,255,255,.15) 25%,transparent 25%,transparent 50%,rgba(255,255,255,.15) 50%,rgba(255,255,255,.15) 75%,transparent 75%,transparent);background-image:linear-gradient(45deg,rgba(255,255,255,.15) 25%,transparent 25%,transparent 50%,rgba(255,255,255,.15) 50%,rgba(255,255,255,.15) 75%,transparent 75%,transparent)}.progress-bar-danger{background-color:#d9534f}.progress-striped .progress-bar-danger{background-image:-webkit-linear-gradient(45deg,rgba(255,255,255,.15) 25%,transparent 25%,transparent 50%,rgba(255,255,255,.15) 50%,rgba(255,255,255,.15) 75%,transparent 75%,transparent);background-image:-o-linear-gradient(45deg,rgba(255,255,255,.15) 25%,transparent 25%,transparent 50%,rgba(255,255,255,.15) 50%,rgba(255,255,255,.15) 75%,transparent 75%,transparent);background-image:linear-gradient(45deg,rgba(255,255,255,.15) 25%,transparent 25%,transparent 50%,rgba(255,255,255,.15) 50%,rgba(255,255,255,.15) 75%,transparent 75%,transparent)}.media{margin-top:15px}.media:first-child{margin-top:0}.media,.media-body{overflow:hidden;zoom:1}.media-body{width:10000px}.media-object{display:block}.media-right,.media>.pull-right{padding-left:10px}.media-left,.media>.pull-left{padding-right:10px}.media-body,.media-left,.media-right{display:table-cell;vertical-align:top}.media-middle{vertical-align:middle}.media-bottom{vertical-align:bottom}.media-heading{margin-top:0;margin-bottom:5px}.media-list{padding-left:0;list-style:none}.list-group{padding-left:0;margin-bottom:20px}.list-group-item{position:relative;display:block;padding:10px 15px;margin-bottom:-1px;background-color:#fff;border:1px solid #ddd}.list-group-item:first-child{border-top-left-radius:4px;border-top-right-radius:4px}.list-group-item:last-child{margin-bottom:0;border-bottom-right-radius:4px;border-bottom-left-radius:4px}a.list-group-item{color:#555}a.list-group-item .list-group-item-heading{color:#333}a.list-group-item:focus,a.list-group-item:hover{color:#555;text-decoration:none;background-color:#f5f5f5}.list-group-item.disabled,.list-group-item.disabled:focus,.list-group-item.disabled:hover{color:#777;cursor:not-allowed;background-color:#eee}.list-group-item.disabled .list-group-item-heading,.list-group-item.disabled:focus .list-group-item-heading,.list-group-item.disabled:hover .list-group-item-heading{color:inherit}.list-group-item.disabled .list-group-item-text,.list-group-item.disabled:focus .list-group-item-text,.list-group-item.disabled:hover .list-group-item-text{color:#777}.list-group-item.active,.list-group-item.active:focus,.list-group-item.active:hover{z-index:2;color:#fff;background-color:#337ab7;border-color:#337ab7}.list-group-item.active .list-group-item-heading,.list-group-item.active .list-group-item-heading>.small,.list-group-item.active .list-group-item-heading>small,.list-group-item.active:focus .list-group-item-heading,.list-group-item.active:focus .list-group-item-heading>.small,.list-group-item.active:focus .list-group-item-heading>small,.list-group-item.active:hover .list-group-item-heading,.list-group-item.active:hover .list-group-item-heading>.small,.list-group-item.active:hover .list-group-item-heading>small{color:inherit}.list-group-item.active .list-group-item-text,.list-group-item.active:focus .list-group-item-text,.list-group-item.active:hover .list-group-item-text{color:#c7ddef}.list-group-item-success{color:#3c763d;background-color:#dff0d8}a.list-group-item-success{color:#3c763d}a.list-group-item-success .list-group-item-heading{color:inherit}a.list-group-item-success:focus,a.list-group-item-success:hover{color:#3c763d;background-color:#d0e9c6}a.list-group-item-success.active,a.list-group-item-success.active:focus,a.list-group-item-success.active:hover{color:#fff;background-color:#3c763d;border-color:#3c763d}.list-group-item-info{color:#31708f;background-color:#d9edf7}a.list-group-item-info{color:#31708f}a.list-group-item-info .list-group-item-heading{color:inherit}a.list-group-item-info:focus,a.list-group-item-info:hover{color:#31708f;background-color:#c4e3f3}a.list-group-item-info.active,a.list-group-item-info.active:focus,a.list-group-item-info.active:hover{color:#fff;background-color:#31708f;border-color:#31708f}.list-group-item-warning{color:#8a6d3b;background-color:#fcf8e3}a.list-group-item-warning{color:#8a6d3b}a.list-group-item-warning .list-group-item-heading{color:inherit}a.list-group-item-warning:focus,a.list-group-item-warning:hover{color:#8a6d3b;background-color:#faf2cc}a.list-group-item-warning.active,a.list-group-item-warning.active:focus,a.list-group-item-warning.active:hover{color:#fff;background-color:#8a6d3b;border-color:#8a6d3b}.list-group-item-danger{color:#a94442;background-color:#f2dede}a.list-group-item-danger{color:#a94442}a.list-group-item-danger .list-group-item-heading{color:inherit}a.list-group-item-danger:focus,a.list-group-item-danger:hover{color:#a94442;background-color:#ebcccc}a.list-group-item-danger.active,a.list-group-item-danger.active:focus,a.list-group-item-danger.active:hover{color:#fff;background-color:#a94442;border-color:#a94442}.list-group-item-heading{margin-top:0;margin-bottom:5px}.list-group-item-text{margin-bottom:0;line-height:1.3}.panel{margin-bottom:20px;background-color:#fff;border:1px solid transparent;border-radius:4px;-webkit-box-shadow:0 1px 1px rgba(0,0,0,.05);box-shadow:0 1px 1px rgba(0,0,0,.05)}.panel-body{padding:15px}.panel-heading{padding:10px 15px;border-bottom:1px solid transparent;border-top-left-radius:3px;border-top-right-radius:3px}.panel-heading>.dropdown .dropdown-toggle{color:inherit}.panel-title{margin-top:0;margin-bottom:0;font-size:16px;color:inherit}.panel-title>.small,.panel-title>.small>a,.panel-title>a,.panel-title>small,.panel-title>small>a{color:inherit}.panel-footer{padding:10px 15px;background-color:#f5f5f5;border-top:1px solid #ddd;border-bottom-right-radius:3px;border-bottom-left-radius:3px}.panel>.list-group,.panel>.panel-collapse>.list-group{margin-bottom:0}.panel>.list-group .list-group-item,.panel>.panel-collapse>.list-group .list-group-item{border-width:1px 0;border-radius:0}.panel>.list-group:first-child .list-group-item:first-child,.panel>.panel-collapse>.list-group:first-child .list-group-item:first-child{border-top:0;border-top-left-radius:3px;border-top-right-radius:3px}.panel>.list-group:last-child .list-group-item:last-child,.panel>.panel-collapse>.list-group:last-child .list-group-item:last-child{border-bottom:0;border-bottom-right-radius:3px;border-bottom-left-radius:3px}.panel-heading+.list-group .list-group-item:first-child{border-top-width:0}.list-group+.panel-footer{border-top-width:0}.panel>.panel-collapse>.table,.panel>.table,.panel>.table-responsive>.table{margin-bottom:0}.panel>.panel-collapse>.table caption,.panel>.table caption,.panel>.table-responsive>.table caption{padding-right:15px;padding-left:15px}.panel>.table-responsive:first-child>.table:first-child,.panel>.table:first-child{border-top-left-radius:3px;border-top-right-radius:3px}.panel>.table-responsive:first-child>.table:first-child>tbody:first-child>tr:first-child,.panel>.table-responsive:first-child>.table:first-child>thead:first-child>tr:first-child,.panel>.table:first-child>tbody:first-child>tr:first-child,.panel>.table:first-child>thead:first-child>tr:first-child{border-top-left-radius:3px;border-top-right-radius:3px}.panel>.table-responsive:first-child>.table:first-child>tbody:first-child>tr:first-child td:first-child,.panel>.table-responsive:first-child>.table:first-child>tbody:first-child>tr:first-child th:first-child,.panel>.table-responsive:first-child>.table:first-child>thead:first-child>tr:first-child td:first-child,.panel>.table-responsive:first-child>.table:first-child>thead:first-child>tr:first-child th:first-child,.panel>.table:first-child>tbody:first-child>tr:first-child td:first-child,.panel>.table:first-child>tbody:first-child>tr:first-child th:first-child,.panel>.table:first-child>thead:first-child>tr:first-child td:first-child,.panel>.table:first-child>thead:first-child>tr:first-child th:first-child{border-top-left-radius:3px}.panel>.table-responsive:first-child>.table:first-child>tbody:first-child>tr:first-child td:last-child,.panel>.table-responsive:first-child>.table:first-child>tbody:first-child>tr:first-child th:last-child,.panel>.table-responsive:first-child>.table:first-child>thead:first-child>tr:first-child td:last-child,.panel>.table-responsive:first-child>.table:first-child>thead:first-child>tr:first-child th:last-child,.panel>.table:first-child>tbody:first-child>tr:first-child td:last-child,.panel>.table:first-child>tbody:first-child>tr:first-child th:last-child,.panel>.table:first-child>thead:first-child>tr:first-child td:last-child,.panel>.table:first-child>thead:first-child>tr:first-child th:last-child{border-top-right-radius:3px}.panel>.table-responsive:last-child>.table:last-child,.panel>.table:last-child{border-bottom-right-radius:3px;border-bottom-left-radius:3px}.panel>.table-responsive:last-child>.table:last-child>tbody:last-child>tr:last-child,.panel>.table-responsive:last-child>.table:last-child>tfoot:last-child>tr:last-child,.panel>.table:last-child>tbody:last-child>tr:last-child,.panel>.table:last-child>tfoot:last-child>tr:last-child{border-bottom-right-radius:3px;border-bottom-left-radius:3px}.panel>.table-responsive:last-child>.table:last-child>tbody:last-child>tr:last-child td:first-child,.panel>.table-responsive:last-child>.table:last-child>tbody:last-child>tr:last-child th:first-child,.panel>.table-responsive:last-child>.table:last-child>tfoot:last-child>tr:last-child td:first-child,.panel>.table-responsive:last-child>.table:last-child>tfoot:last-child>tr:last-child th:first-child,.panel>.table:last-child>tbody:last-child>tr:last-child td:first-child,.panel>.table:last-child>tbody:last-child>tr:last-child th:first-child,.panel>.table:last-child>tfoot:last-child>tr:last-child td:first-child,.panel>.table:last-child>tfoot:last-child>tr:last-child th:first-child{border-bottom-left-radius:3px}.panel>.table-responsive:last-child>.table:last-child>tbody:last-child>tr:last-child td:last-child,.panel>.table-responsive:last-child>.table:last-child>tbody:last-child>tr:last-child th:last-child,.panel>.table-responsive:last-child>.table:last-child>tfoot:last-child>tr:last-child td:last-child,.panel>.table-responsive:last-child>.table:last-child>tfoot:last-child>tr:last-child th:last-child,.panel>.table:last-child>tbody:last-child>tr:last-child td:last-child,.panel>.table:last-child>tbody:last-child>tr:last-child th:last-child,.panel>.table:last-child>tfoot:last-child>tr:last-child td:last-child,.panel>.table:last-child>tfoot:last-child>tr:last-child th:last-child{border-bottom-right-radius:3px}.panel>.panel-body+.table,.panel>.panel-body+.table-responsive,.panel>.table+.panel-body,.panel>.table-responsive+.panel-body{border-top:1px solid #ddd}.panel>.table>tbody:first-child>tr:first-child td,.panel>.table>tbody:first-child>tr:first-child th{border-top:0}.panel>.table-bordered,.panel>.table-responsive>.table-bordered{border:0}.panel>.table-bordered>tbody>tr>td:first-child,.panel>.table-bordered>tbody>tr>th:first-child,.panel>.table-bordered>tfoot>tr>td:first-child,.panel>.table-bordered>tfoot>tr>th:first-child,.panel>.table-bordered>thead>tr>td:first-child,.panel>.table-bordered>thead>tr>th:first-child,.panel>.table-responsive>.table-bordered>tbody>tr>td:first-child,.panel>.table-responsive>.table-bordered>tbody>tr>th:first-child,.panel>.table-responsive>.table-bordered>tfoot>tr>td:first-child,.panel>.table-responsive>.table-bordered>tfoot>tr>th:first-child,.panel>.table-responsive>.table-bordered>thead>tr>td:first-child,.panel>.table-responsive>.table-bordered>thead>tr>th:first-child{border-left:0}.panel>.table-bordered>tbody>tr>td:last-child,.panel>.table-bordered>tbody>tr>th:last-child,.panel>.table-bordered>tfoot>tr>td:last-child,.panel>.table-bordered>tfoot>tr>th:last-child,.panel>.table-bordered>thead>tr>td:last-child,.panel>.table-bordered>thead>tr>th:last-child,.panel>.table-responsive>.table-bordered>tbody>tr>td:last-child,.panel>.table-responsive>.table-bordered>tbody>tr>th:last-child,.panel>.table-responsive>.table-bordered>tfoot>tr>td:last-child,.panel>.table-responsive>.table-bordered>tfoot>tr>th:last-child,.panel>.table-responsive>.table-bordered>thead>tr>td:last-child,.panel>.table-responsive>.table-bordered>thead>tr>th:last-child{border-right:0}.panel>.table-bordered>tbody>tr:first-child>td,.panel>.table-bordered>tbody>tr:first-child>th,.panel>.table-bordered>thead>tr:first-child>td,.panel>.table-bordered>thead>tr:first-child>th,.panel>.table-responsive>.table-bordered>tbody>tr:first-child>td,.panel>.table-responsive>.table-bordered>tbody>tr:first-child>th,.panel>.table-responsive>.table-bordered>thead>tr:first-child>td,.panel>.table-responsive>.table-bordered>thead>tr:first-child>th{border-bottom:0}.panel>.table-bordered>tbody>tr:last-child>td,.panel>.table-bordered>tbody>tr:last-child>th,.panel>.table-bordered>tfoot>tr:last-child>td,.panel>.table-bordered>tfoot>tr:last-child>th,.panel>.table-responsive>.table-bordered>tbody>tr:last-child>td,.panel>.table-responsive>.table-bordered>tbody>tr:last-child>th,.panel>.table-responsive>.table-bordered>tfoot>tr:last-child>td,.panel>.table-responsive>.table-bordered>tfoot>tr:last-child>th{border-bottom:0}.panel>.table-responsive{margin-bottom:0;border:0}.panel-group{margin-bottom:20px}.panel-group .panel{margin-bottom:0;border-radius:4px}.panel-group .panel+.panel{margin-top:5px}.panel-group .panel-heading{border-bottom:0}.panel-group .panel-heading+.panel-collapse>.list-group,.panel-group .panel-heading+.panel-collapse>.panel-body{border-top:1px solid #ddd}.panel-group .panel-footer{border-top:0}.panel-group .panel-footer+.panel-collapse .panel-body{border-bottom:1px solid #ddd}.panel-default{border-color:#ddd}.panel-default>.panel-heading{color:#333;background-color:#f5f5f5;border-color:#ddd}.panel-default>.panel-heading+.panel-collapse>.panel-body{border-top-color:#ddd}.panel-default>.panel-heading .badge{color:#f5f5f5;background-color:#333}.panel-default>.panel-footer+.panel-collapse>.panel-body{border-bottom-color:#ddd}.panel-primary{border-color:#337ab7}.panel-primary>.panel-heading{color:#fff;background-color:#337ab7;border-color:#337ab7}.panel-primary>.panel-heading+.panel-collapse>.panel-body{border-top-color:#337ab7}.panel-primary>.panel-heading .badge{color:#337ab7;background-color:#fff}.panel-primary>.panel-footer+.panel-collapse>.panel-body{border-bottom-color:#337ab7}.panel-success{border-color:#d6e9c6}.panel-success>.panel-heading{color:#3c763d;background-color:#dff0d8;border-color:#d6e9c6}.panel-success>.panel-heading+.panel-collapse>.panel-body{border-top-color:#d6e9c6}.panel-success>.panel-heading .badge{color:#dff0d8;background-color:#3c763d}.panel-success>.panel-footer+.panel-collapse>.panel-body{border-bottom-color:#d6e9c6}.panel-info{border-color:#bce8f1}.panel-info>.panel-heading{color:#31708f;background-color:#d9edf7;border-color:#bce8f1}.panel-info>.panel-heading+.panel-collapse>.panel-body{border-top-color:#bce8f1}.panel-info>.panel-heading .badge{color:#d9edf7;background-color:#31708f}.panel-info>.panel-footer+.panel-collapse>.panel-body{border-bottom-color:#bce8f1}.panel-warning{border-color:#faebcc}.panel-warning>.panel-heading{color:#8a6d3b;background-color:#fcf8e3;border-color:#faebcc}.panel-warning>.panel-heading+.panel-collapse>.panel-body{border-top-color:#faebcc}.panel-warning>.panel-heading .badge{color:#fcf8e3;background-color:#8a6d3b}.panel-warning>.panel-footer+.panel-collapse>.panel-body{border-bottom-color:#faebcc}.panel-danger{border-color:#ebccd1}.panel-danger>.panel-heading{color:#a94442;background-color:#f2dede;border-color:#ebccd1}.panel-danger>.panel-heading+.panel-collapse>.panel-body{border-top-color:#ebccd1}.panel-danger>.panel-heading .badge{color:#f2dede;background-color:#a94442}.panel-danger>.panel-footer+.panel-collapse>.panel-body{border-bottom-color:#ebccd1}.embed-responsive{position:relative;display:block;height:0;padding:0;overflow:hidden}.embed-responsive .embed-responsive-item,.embed-responsive embed,.embed-responsive iframe,.embed-responsive object,.embed-responsive video{position:absolute;top:0;bottom:0;left:0;width:100%;height:100%;border:0}.embed-responsive-16by9{padding-bottom:56.25%}.embed-responsive-4by3{padding-bottom:75%}.well{min-height:20px;padding:19px;margin-bottom:20px;background-color:#f5f5f5;border:1px solid #e3e3e3;border-radius:4px;-webkit-box-shadow:inset 0 1px 1px rgba(0,0,0,.05);box-shadow:inset 0 1px 1px rgba(0,0,0,.05)}.well blockquote{border-color:#ddd;border-color:rgba(0,0,0,.15)}.well-lg{padding:24px;border-radius:6px}.well-sm{padding:9px;border-radius:3px}.close{float:right;font-size:21px;font-weight:700;line-height:1;color:#000;text-shadow:0 1px 0 #fff;filter:alpha(opacity=20);opacity:.2}.close:focus,.close:hover{color:#000;text-decoration:none;cursor:pointer;filter:alpha(opacity=50);opacity:.5}button.close{-webkit-appearance:none;padding:0;cursor:pointer;background:0 0;border:0}.modal-open{overflow:hidden}.modal{position:fixed;top:0;right:0;bottom:0;left:0;z-index:1050;display:none;overflow:hidden;-webkit-overflow-scrolling:touch;outline:0}.modal.fade .modal-dialog{-webkit-transition:-webkit-transform .3s ease-out;-o-transition:-o-transform .3s ease-out;transition:transform .3s ease-out;-webkit-transform:translate(0,-25%);-ms-transform:translate(0,-25%);-o-transform:translate(0,-25%);transform:translate(0,-25%)}.modal.in .modal-dialog{-webkit-transform:translate(0,0);-ms-transform:translate(0,0);-o-transform:translate(0,0);transform:translate(0,0)}.modal-open .modal{overflow-x:hidden;overflow-y:auto}.modal-dialog{position:relative;width:auto;margin:10px}.modal-content{position:relative;background-color:#fff;-webkit-background-clip:padding-box;background-clip:padding-box;border:1px solid #999;border:1px solid rgba(0,0,0,.2);border-radius:6px;outline:0;-webkit-box-shadow:0 3px 9px rgba(0,0,0,.5);box-shadow:0 3px 9px rgba(0,0,0,.5)}.modal-backdrop{position:fixed;top:0;right:0;bottom:0;left:0;z-index:1040;background-color:#000}.modal-backdrop.fade{filter:alpha(opacity=0);opacity:0}.modal-backdrop.in{filter:alpha(opacity=50);opacity:.5}.modal-header{min-height:16.43px;padding:15px;border-bottom:1px solid #e5e5e5}.modal-header .close{margin-top:-2px}.modal-title{margin:0;line-height:1.42857143}.modal-body{position:relative;padding:15px}.modal-footer{padding:15px;text-align:right;border-top:1px solid #e5e5e5}.modal-footer .btn+.btn{margin-bottom:0;margin-left:5px}.modal-footer .btn-group .btn+.btn{margin-left:-1px}.modal-footer .btn-block+.btn-block{margin-left:0}.modal-scrollbar-measure{position:absolute;top:-9999px;width:50px;height:50px;overflow:scroll}@media (min-width:768px){.modal-dialog{width:600px;margin:30px auto}.modal-content{-webkit-box-shadow:0 5px 15px rgba(0,0,0,.5);box-shadow:0 5px 15px rgba(0,0,0,.5)}.modal-sm{width:300px}}@media (min-width:992px){.modal-lg{width:900px}}.tooltip{position:absolute;z-index:1070;display:block;font-family:"Helvetica Neue",Helvetica,Arial,sans-serif;font-size:12px;font-weight:400;line-height:1.4;filter:alpha(opacity=0);opacity:0}.tooltip.in{filter:alpha(opacity=90);opacity:.9}.tooltip.top{padding:5px 0;margin-top:-3px}.tooltip.right{padding:0 5px;margin-left:3px}.tooltip.bottom{padding:5px 0;margin-top:3px}.tooltip.left{padding:0 5px;margin-left:-3px}.tooltip-inner{max-width:200px;padding:3px 8px;color:#fff;text-align:center;text-decoration:none;background-color:#000;border-radius:4px}.tooltip-arrow{position:absolute;width:0;height:0;border-color:transparent;border-style:solid}.tooltip.top .tooltip-arrow{bottom:0;left:50%;margin-left:-5px;border-width:5px 5px 0;border-top-color:#000}.tooltip.top-left .tooltip-arrow{right:5px;bottom:0;margin-bottom:-5px;border-width:5px 5px 0;border-top-color:#000}.tooltip.top-right .tooltip-arrow{bottom:0;left:5px;margin-bottom:-5px;border-width:5px 5px 0;border-top-color:#000}.tooltip.right .tooltip-arrow{top:50%;left:0;margin-top:-5px;border-width:5px 5px 5px 0;border-right-color:#000}.tooltip.left .tooltip-arrow{top:50%;right:0;margin-top:-5px;border-width:5px 0 5px 5px;border-left-color:#000}.tooltip.bottom .tooltip-arrow{top:0;left:50%;margin-left:-5px;border-width:0 5px 5px;border-bottom-color:#000}.tooltip.bottom-left .tooltip-arrow{top:0;right:5px;margin-top:-5px;border-width:0 5px 5px;border-bottom-color:#000}.tooltip.bottom-right .tooltip-arrow{top:0;left:5px;margin-top:-5px;border-width:0 5px 5px;border-bottom-color:#000}.popover{position:absolute;top:0;left:0;z-index:1060;display:none;max-width:276px;padding:1px;font-family:"Helvetica Neue",Helvetica,Arial,sans-serif;font-size:14px;font-weight:400;line-height:1.42857143;text-align:left;white-space:normal;background-color:#fff;-webkit-background-clip:padding-box;background-clip:padding-box;border:1px solid #ccc;border:1px solid rgba(0,0,0,.2);border-radius:6px;-webkit-box-shadow:0 5px 10px rgba(0,0,0,.2);box-shadow:0 5px 10px rgba(0,0,0,.2)}.popover.top{margin-top:-10px}.popover.right{margin-left:10px}.popover.bottom{margin-top:10px}.popover.left{margin-left:-10px}.popover-title{padding:8px 14px;margin:0;font-size:14px;background-color:#f7f7f7;border-bottom:1px solid #ebebeb;border-radius:5px 5px 0 0}.popover-content{padding:9px 14px}.popover>.arrow,.popover>.arrow:after{position:absolute;display:block;width:0;height:0;border-color:transparent;border-style:solid}.popover>.arrow{border-width:11px}.popover>.arrow:after{content:"";border-width:10px}.popover.top>.arrow{bottom:-11px;left:50%;margin-left:-11px;border-top-color:#999;border-top-color:rgba(0,0,0,.25);border-bottom-width:0}.popover.top>.arrow:after{bottom:1px;margin-left:-10px;content:" ";border-top-color:#fff;border-bottom-width:0}.popover.right>.arrow{top:50%;left:-11px;margin-top:-11px;border-right-color:#999;border-right-color:rgba(0,0,0,.25);border-left-width:0}.popover.right>.arrow:after{bottom:-10px;left:1px;content:" ";border-right-color:#fff;border-left-width:0}.popover.bottom>.arrow{top:-11px;left:50%;margin-left:-11px;border-top-width:0;border-bottom-color:#999;border-bottom-color:rgba(0,0,0,.25)}.popover.bottom>.arrow:after{top:1px;margin-left:-10px;content:" ";border-top-width:0;border-bottom-color:#fff}.popover.left>.arrow{top:50%;right:-11px;margin-top:-11px;border-right-width:0;border-left-color:#999;border-left-color:rgba(0,0,0,.25)}.popover.left>.arrow:after{right:1px;bottom:-10px;content:" ";border-right-width:0;border-left-color:#fff}.carousel{position:relative}.carousel-inner{position:relative;width:100%;overflow:hidden}.carousel-inner>.item{position:relative;display:none;-webkit-transition:.6s ease-in-out left;-o-transition:.6s ease-in-out left;transition:.6s ease-in-out left}.carousel-inner>.item>a>img,.carousel-inner>.item>img{line-height:1}@media all and (transform-3d),(-webkit-transform-3d){.carousel-inner>.item{-webkit-transition:-webkit-transform .6s ease-in-out;-o-transition:-o-transform .6s ease-in-out;transition:transform .6s ease-in-out;-webkit-backface-visibility:hidden;backface-visibility:hidden;-webkit-perspective:1000;perspective:1000}.carousel-inner>.item.active.right,.carousel-inner>.item.next{left:0;-webkit-transform:translate3d(100%,0,0);transform:translate3d(100%,0,0)}.carousel-inner>.item.active.left,.carousel-inner>.item.prev{left:0;-webkit-transform:translate3d(-100%,0,0);transform:translate3d(-100%,0,0)}.carousel-inner>.item.active,.carousel-inner>.item.next.left,.carousel-inner>.item.prev.right{left:0;-webkit-transform:translate3d(0,0,0);transform:translate3d(0,0,0)}}.carousel-inner>.active,.carousel-inner>.next,.carousel-inner>.prev{display:block}.carousel-inner>.active{left:0}.carousel-inner>.next,.carousel-inner>.prev{position:absolute;top:0;width:100%}.carousel-inner>.next{left:100%}.carousel-inner>.prev{left:-100%}.carousel-inner>.next.left,.carousel-inner>.prev.right{left:0}.carousel-inner>.active.left{left:-100%}.carousel-inner>.active.right{left:100%}.carousel-control{position:absolute;top:0;bottom:0;left:0;width:15%;font-size:20px;color:#fff;text-align:center;text-shadow:0 1px 2px rgba(0,0,0,.6);filter:alpha(opacity=50);opacity:.5}.carousel-control.left{background-image:-webkit-linear-gradient(left,rgba(0,0,0,.5) 0,rgba(0,0,0,.0001) 100%);background-image:-o-linear-gradient(left,rgba(0,0,0,.5) 0,rgba(0,0,0,.0001) 100%);background-image:-webkit-gradient(linear,left top,right top,from(rgba(0,0,0,.5)),to(rgba(0,0,0,.0001)));background-image:linear-gradient(to right,rgba(0,0,0,.5) 0,rgba(0,0,0,.0001) 100%);filter:progid:DXImageTransform.Microsoft.gradient(startColorstr='#80000000', endColorstr='#00000000', GradientType=1);background-repeat:repeat-x}.carousel-control.right{right:0;left:auto;background-image:-webkit-linear-gradient(left,rgba(0,0,0,.0001) 0,rgba(0,0,0,.5) 100%);background-image:-o-linear-gradient(left,rgba(0,0,0,.0001) 0,rgba(0,0,0,.5) 100%);background-image:-webkit-gradient(linear,left top,right top,from(rgba(0,0,0,.0001)),to(rgba(0,0,0,.5)));background-image:linear-gradient(to right,rgba(0,0,0,.0001) 0,rgba(0,0,0,.5) 100%);filter:progid:DXImageTransform.Microsoft.gradient(startColorstr='#00000000', endColorstr='#80000000', GradientType=1);background-repeat:repeat-x}.carousel-control:focus,.carousel-control:hover{color:#fff;text-decoration:none;filter:alpha(opacity=90);outline:0;opacity:.9}.carousel-control .glyphicon-chevron-left,.carousel-control .glyphicon-chevron-right,.carousel-control .icon-next,.carousel-control .icon-prev{position:absolute;top:50%;z-index:5;display:inline-block}.carousel-control .glyphicon-chevron-left,.carousel-control .icon-prev{left:50%;margin-left:-10px}.carousel-control .glyphicon-chevron-right,.carousel-control .icon-next{right:50%;margin-right:-10px}.carousel-control .icon-next,.carousel-control .icon-prev{width:20px;height:20px;margin-top:-10px;font-family:serif;line-height:1}.carousel-control .icon-prev:before{content:'\2039'}.carousel-control .icon-next:before{content:'\203a'}.carousel-indicators{position:absolute;bottom:10px;left:50%;z-index:15;width:60%;padding-left:0;margin-left:-30%;text-align:center;list-style:none}.carousel-indicators li{display:inline-block;width:10px;height:10px;margin:1px;text-indent:-999px;cursor:pointer;background-color:#000 \9;background-color:rgba(0,0,0,0);border:1px solid #fff;border-radius:10px}.carousel-indicators .active{width:12px;height:12px;margin:0;background-color:#fff}.carousel-caption{position:absolute;right:15%;bottom:20px;left:15%;z-index:10;padding-top:20px;padding-bottom:20px;color:#fff;text-align:center;text-shadow:0 1px 2px rgba(0,0,0,.6)}.carousel-caption .btn{text-shadow:none}@media screen and (min-width:768px){.carousel-control .glyphicon-chevron-left,.carousel-control .glyphicon-chevron-right,.carousel-control .icon-next,.carousel-control .icon-prev{width:30px;height:30px;margin-top:-15px;font-size:30px}.carousel-control .glyphicon-chevron-left,.carousel-control .icon-prev{margin-left:-15px}.carousel-control .glyphicon-chevron-right,.carousel-control .icon-next{margin-right:-15px}.carousel-caption{right:20%;left:20%;padding-bottom:30px}.carousel-indicators{bottom:20px}}.btn-group-vertical>.btn-group:after,.btn-group-vertical>.btn-group:before,.btn-toolbar:after,.btn-toolbar:before,.clearfix:after,.clearfix:before,.container-fluid:after,.container-fluid:before,.container:after,.container:before,.dl-horizontal dd:after,.dl-horizontal dd:before,.form-horizontal .form-group:after,.form-horizontal .form-group:before,.modal-footer:after,.modal-footer:before,.nav:after,.nav:before,.navbar-collapse:after,.navbar-collapse:before,.navbar-header:after,.navbar-header:before,.navbar:after,.navbar:before,.pager:after,.pager:before,.panel-body:after,.panel-body:before,.row:after,.row:before{display:table;content:" "}.btn-group-vertical>.btn-group:after,.btn-toolbar:after,.clearfix:after,.container-fluid:after,.container:after,.dl-horizontal dd:after,.form-horizontal .form-group:after,.modal-footer:after,.nav:after,.navbar-collapse:after,.navbar-header:after,.navbar:after,.pager:after,.panel-body:after,.row:after{clear:both}.center-block{display:block;margin-right:auto;margin-left:auto}.pull-right{float:right!important}.pull-left{float:left!important}.hide{display:none!important}.show{display:block!important}.invisible{visibility:hidden}.text-hide{font:0/0 a;color:transparent;text-shadow:none;background-color:transparent;border:0}.hidden{display:none!important}.affix{position:fixed}@-ms-viewport{width:device-width}.visible-lg,.visible-md,.visible-sm,.visible-xs{display:none!important}.visible-lg-block,.visible-lg-inline,.visible-lg-inline-block,.visible-md-block,.visible-md-inline,.visible-md-inline-block,.visible-sm-block,.visible-sm-inline,.visible-sm-inline-block,.visible-xs-block,.visible-xs-inline,.visible-xs-inline-block{display:none!important}@media (max-width:767px){.visible-xs{display:block!important}table.visible-xs{display:table}tr.visible-xs{display:table-row!important}td.visible-xs,th.visible-xs{display:table-cell!important}}@media (max-width:767px){.visible-xs-block{display:block!important}}@media (max-width:767px){.visible-xs-inline{display:inline!important}}@media (max-width:767px){.visible-xs-inline-block{display:inline-block!important}}@media (min-width:768px)and (max-width:991px){.visible-sm{display:block!important}table.visible-sm{display:table}tr.visible-sm{display:table-row!important}td.visible-sm,th.visible-sm{display:table-cell!important}}@media (min-width:768px)and (max-width:991px){.visible-sm-block{display:block!important}}@media (min-width:768px)and (max-width:991px){.visible-sm-inline{display:inline!important}}@media (min-width:768px)and (max-width:991px){.visible-sm-inline-block{display:inline-block!important}}@media (min-width:992px)and (max-width:1199px){.visible-md{display:block!important}table.visible-md{display:table}tr.visible-md{display:table-row!important}td.visible-md,th.visible-md{display:table-cell!important}}@media (min-width:992px)and (max-width:1199px){.visible-md-block{display:block!important}}@media (min-width:992px)and (max-width:1199px){.visible-md-inline{display:inline!important}}@media (min-width:992px)and (max-width:1199px){.visible-md-inline-block{display:inline-block!important}}@media (min-width:1200px){.visible-lg{display:block!important}table.visible-lg{display:table}tr.visible-lg{display:table-row!important}td.visible-lg,th.visible-lg{display:table-cell!important}}@media (min-width:1200px){.visible-lg-block{display:block!important}}@media (min-width:1200px){.visible-lg-inline{display:inline!important}}@media (min-width:1200px){.visible-lg-inline-block{display:inline-block!important}}@media (max-width:767px){.hidden-xs{display:none!important}}@media (min-width:768px)and (max-width:991px){.hidden-sm{display:none!important}}@media (min-width:992px)and (max-width:1199px){.hidden-md{display:none!important}}@media (min-width:1200px){.hidden-lg{display:none!important}}.visible-print{display:none!important}@media print{.visible-print{display:block!important}table.visible-print{display:table}tr.visible-print{display:table-row!important}td.visible-print,th.visible-print{display:table-cell!important}}.visible-print-block{display:none!important}@media print{.visible-print-block{display:block!important}}.visible-print-inline{display:none!important}@media print{.visible-print-inline{display:inline!important}}.visible-print-inline-block{display:none!important}@media print{.visible-print-inline-block{display:inline-block!important}}@media print{.hidden-print{display:none!important}} \ No newline at end of file diff --git a/system/admin/editor.md/tests/js/bootstrap.min.js b/system/admin/editor.md/tests/js/bootstrap.min.js new file mode 100644 index 00000000..c8f82e59 --- /dev/null +++ b/system/admin/editor.md/tests/js/bootstrap.min.js @@ -0,0 +1,7 @@ +/*! + * Bootstrap v3.3.4 (http://getbootstrap.com) + * Copyright 2011-2015 Twitter, Inc. + * Licensed under MIT (https://github.com/twbs/bootstrap/blob/master/LICENSE) + */ +if("undefined"==typeof jQuery)throw new Error("Bootstrap's JavaScript requires jQuery");+function(a){"use strict";var b=a.fn.jquery.split(" ")[0].split(".");if(b[0]<2&&b[1]<9||1==b[0]&&9==b[1]&&b[2]<1)throw new Error("Bootstrap's JavaScript requires jQuery version 1.9.1 or higher")}(jQuery),+function(a){"use strict";function b(){var a=document.createElement("bootstrap"),b={WebkitTransition:"webkitTransitionEnd",MozTransition:"transitionend",OTransition:"oTransitionEnd otransitionend",transition:"transitionend"};for(var c in b)if(void 0!==a.style[c])return{end:b[c]};return!1}a.fn.emulateTransitionEnd=function(b){var c=!1,d=this;a(this).one("bsTransitionEnd",function(){c=!0});var e=function(){c||a(d).trigger(a.support.transition.end)};return setTimeout(e,b),this},a(function(){a.support.transition=b(),a.support.transition&&(a.event.special.bsTransitionEnd={bindType:a.support.transition.end,delegateType:a.support.transition.end,handle:function(b){return a(b.target).is(this)?b.handleObj.handler.apply(this,arguments):void 0}})})}(jQuery),+function(a){"use strict";function b(b){return this.each(function(){var c=a(this),e=c.data("bs.alert");e||c.data("bs.alert",e=new d(this)),"string"==typeof b&&e[b].call(c)})}var c='[data-dismiss="alert"]',d=function(b){a(b).on("click",c,this.close)};d.VERSION="3.3.4",d.TRANSITION_DURATION=150,d.prototype.close=function(b){function c(){g.detach().trigger("closed.bs.alert").remove()}var e=a(this),f=e.attr("data-target");f||(f=e.attr("href"),f=f&&f.replace(/.*(?=#[^\s]*$)/,""));var g=a(f);b&&b.preventDefault(),g.length||(g=e.closest(".alert")),g.trigger(b=a.Event("close.bs.alert")),b.isDefaultPrevented()||(g.removeClass("in"),a.support.transition&&g.hasClass("fade")?g.one("bsTransitionEnd",c).emulateTransitionEnd(d.TRANSITION_DURATION):c())};var e=a.fn.alert;a.fn.alert=b,a.fn.alert.Constructor=d,a.fn.alert.noConflict=function(){return a.fn.alert=e,this},a(document).on("click.bs.alert.data-api",c,d.prototype.close)}(jQuery),+function(a){"use strict";function b(b){return this.each(function(){var d=a(this),e=d.data("bs.button"),f="object"==typeof b&&b;e||d.data("bs.button",e=new c(this,f)),"toggle"==b?e.toggle():b&&e.setState(b)})}var c=function(b,d){this.$element=a(b),this.options=a.extend({},c.DEFAULTS,d),this.isLoading=!1};c.VERSION="3.3.4",c.DEFAULTS={loadingText:"loading..."},c.prototype.setState=function(b){var c="disabled",d=this.$element,e=d.is("input")?"val":"html",f=d.data();b+="Text",null==f.resetText&&d.data("resetText",d[e]()),setTimeout(a.proxy(function(){d[e](null==f[b]?this.options[b]:f[b]),"loadingText"==b?(this.isLoading=!0,d.addClass(c).attr(c,c)):this.isLoading&&(this.isLoading=!1,d.removeClass(c).removeAttr(c))},this),0)},c.prototype.toggle=function(){var a=!0,b=this.$element.closest('[data-toggle="buttons"]');if(b.length){var c=this.$element.find("input");"radio"==c.prop("type")&&(c.prop("checked")&&this.$element.hasClass("active")?a=!1:b.find(".active").removeClass("active")),a&&c.prop("checked",!this.$element.hasClass("active")).trigger("change")}else this.$element.attr("aria-pressed",!this.$element.hasClass("active"));a&&this.$element.toggleClass("active")};var d=a.fn.button;a.fn.button=b,a.fn.button.Constructor=c,a.fn.button.noConflict=function(){return a.fn.button=d,this},a(document).on("click.bs.button.data-api",'[data-toggle^="button"]',function(c){var d=a(c.target);d.hasClass("btn")||(d=d.closest(".btn")),b.call(d,"toggle"),c.preventDefault()}).on("focus.bs.button.data-api blur.bs.button.data-api",'[data-toggle^="button"]',function(b){a(b.target).closest(".btn").toggleClass("focus",/^focus(in)?$/.test(b.type))})}(jQuery),+function(a){"use strict";function b(b){return this.each(function(){var d=a(this),e=d.data("bs.carousel"),f=a.extend({},c.DEFAULTS,d.data(),"object"==typeof b&&b),g="string"==typeof b?b:f.slide;e||d.data("bs.carousel",e=new c(this,f)),"number"==typeof b?e.to(b):g?e[g]():f.interval&&e.pause().cycle()})}var c=function(b,c){this.$element=a(b),this.$indicators=this.$element.find(".carousel-indicators"),this.options=c,this.paused=null,this.sliding=null,this.interval=null,this.$active=null,this.$items=null,this.options.keyboard&&this.$element.on("keydown.bs.carousel",a.proxy(this.keydown,this)),"hover"==this.options.pause&&!("ontouchstart"in document.documentElement)&&this.$element.on("mouseenter.bs.carousel",a.proxy(this.pause,this)).on("mouseleave.bs.carousel",a.proxy(this.cycle,this))};c.VERSION="3.3.4",c.TRANSITION_DURATION=600,c.DEFAULTS={interval:5e3,pause:"hover",wrap:!0,keyboard:!0},c.prototype.keydown=function(a){if(!/input|textarea/i.test(a.target.tagName)){switch(a.which){case 37:this.prev();break;case 39:this.next();break;default:return}a.preventDefault()}},c.prototype.cycle=function(b){return b||(this.paused=!1),this.interval&&clearInterval(this.interval),this.options.interval&&!this.paused&&(this.interval=setInterval(a.proxy(this.next,this),this.options.interval)),this},c.prototype.getItemIndex=function(a){return this.$items=a.parent().children(".item"),this.$items.index(a||this.$active)},c.prototype.getItemForDirection=function(a,b){var c=this.getItemIndex(b),d="prev"==a&&0===c||"next"==a&&c==this.$items.length-1;if(d&&!this.options.wrap)return b;var e="prev"==a?-1:1,f=(c+e)%this.$items.length;return this.$items.eq(f)},c.prototype.to=function(a){var b=this,c=this.getItemIndex(this.$active=this.$element.find(".item.active"));return a>this.$items.length-1||0>a?void 0:this.sliding?this.$element.one("slid.bs.carousel",function(){b.to(a)}):c==a?this.pause().cycle():this.slide(a>c?"next":"prev",this.$items.eq(a))},c.prototype.pause=function(b){return b||(this.paused=!0),this.$element.find(".next, .prev").length&&a.support.transition&&(this.$element.trigger(a.support.transition.end),this.cycle(!0)),this.interval=clearInterval(this.interval),this},c.prototype.next=function(){return this.sliding?void 0:this.slide("next")},c.prototype.prev=function(){return this.sliding?void 0:this.slide("prev")},c.prototype.slide=function(b,d){var e=this.$element.find(".item.active"),f=d||this.getItemForDirection(b,e),g=this.interval,h="next"==b?"left":"right",i=this;if(f.hasClass("active"))return this.sliding=!1;var j=f[0],k=a.Event("slide.bs.carousel",{relatedTarget:j,direction:h});if(this.$element.trigger(k),!k.isDefaultPrevented()){if(this.sliding=!0,g&&this.pause(),this.$indicators.length){this.$indicators.find(".active").removeClass("active");var l=a(this.$indicators.children()[this.getItemIndex(f)]);l&&l.addClass("active")}var m=a.Event("slid.bs.carousel",{relatedTarget:j,direction:h});return a.support.transition&&this.$element.hasClass("slide")?(f.addClass(b),f[0].offsetWidth,e.addClass(h),f.addClass(h),e.one("bsTransitionEnd",function(){f.removeClass([b,h].join(" ")).addClass("active"),e.removeClass(["active",h].join(" ")),i.sliding=!1,setTimeout(function(){i.$element.trigger(m)},0)}).emulateTransitionEnd(c.TRANSITION_DURATION)):(e.removeClass("active"),f.addClass("active"),this.sliding=!1,this.$element.trigger(m)),g&&this.cycle(),this}};var d=a.fn.carousel;a.fn.carousel=b,a.fn.carousel.Constructor=c,a.fn.carousel.noConflict=function(){return a.fn.carousel=d,this};var e=function(c){var d,e=a(this),f=a(e.attr("data-target")||(d=e.attr("href"))&&d.replace(/.*(?=#[^\s]+$)/,""));if(f.hasClass("carousel")){var g=a.extend({},f.data(),e.data()),h=e.attr("data-slide-to");h&&(g.interval=!1),b.call(f,g),h&&f.data("bs.carousel").to(h),c.preventDefault()}};a(document).on("click.bs.carousel.data-api","[data-slide]",e).on("click.bs.carousel.data-api","[data-slide-to]",e),a(window).on("load",function(){a('[data-ride="carousel"]').each(function(){var c=a(this);b.call(c,c.data())})})}(jQuery),+function(a){"use strict";function b(b){var c,d=b.attr("data-target")||(c=b.attr("href"))&&c.replace(/.*(?=#[^\s]+$)/,"");return a(d)}function c(b){return this.each(function(){var c=a(this),e=c.data("bs.collapse"),f=a.extend({},d.DEFAULTS,c.data(),"object"==typeof b&&b);!e&&f.toggle&&/show|hide/.test(b)&&(f.toggle=!1),e||c.data("bs.collapse",e=new d(this,f)),"string"==typeof b&&e[b]()})}var d=function(b,c){this.$element=a(b),this.options=a.extend({},d.DEFAULTS,c),this.$trigger=a('[data-toggle="collapse"][href="#'+b.id+'"],[data-toggle="collapse"][data-target="#'+b.id+'"]'),this.transitioning=null,this.options.parent?this.$parent=this.getParent():this.addAriaAndCollapsedClass(this.$element,this.$trigger),this.options.toggle&&this.toggle()};d.VERSION="3.3.4",d.TRANSITION_DURATION=350,d.DEFAULTS={toggle:!0},d.prototype.dimension=function(){var a=this.$element.hasClass("width");return a?"width":"height"},d.prototype.show=function(){if(!this.transitioning&&!this.$element.hasClass("in")){var b,e=this.$parent&&this.$parent.children(".panel").children(".in, .collapsing");if(!(e&&e.length&&(b=e.data("bs.collapse"),b&&b.transitioning))){var f=a.Event("show.bs.collapse");if(this.$element.trigger(f),!f.isDefaultPrevented()){e&&e.length&&(c.call(e,"hide"),b||e.data("bs.collapse",null));var g=this.dimension();this.$element.removeClass("collapse").addClass("collapsing")[g](0).attr("aria-expanded",!0),this.$trigger.removeClass("collapsed").attr("aria-expanded",!0),this.transitioning=1;var h=function(){this.$element.removeClass("collapsing").addClass("collapse in")[g](""),this.transitioning=0,this.$element.trigger("shown.bs.collapse")};if(!a.support.transition)return h.call(this);var i=a.camelCase(["scroll",g].join("-"));this.$element.one("bsTransitionEnd",a.proxy(h,this)).emulateTransitionEnd(d.TRANSITION_DURATION)[g](this.$element[0][i])}}}},d.prototype.hide=function(){if(!this.transitioning&&this.$element.hasClass("in")){var b=a.Event("hide.bs.collapse");if(this.$element.trigger(b),!b.isDefaultPrevented()){var c=this.dimension();this.$element[c](this.$element[c]())[0].offsetHeight,this.$element.addClass("collapsing").removeClass("collapse in").attr("aria-expanded",!1),this.$trigger.addClass("collapsed").attr("aria-expanded",!1),this.transitioning=1;var e=function(){this.transitioning=0,this.$element.removeClass("collapsing").addClass("collapse").trigger("hidden.bs.collapse")};return a.support.transition?void this.$element[c](0).one("bsTransitionEnd",a.proxy(e,this)).emulateTransitionEnd(d.TRANSITION_DURATION):e.call(this)}}},d.prototype.toggle=function(){this[this.$element.hasClass("in")?"hide":"show"]()},d.prototype.getParent=function(){return a(this.options.parent).find('[data-toggle="collapse"][data-parent="'+this.options.parent+'"]').each(a.proxy(function(c,d){var e=a(d);this.addAriaAndCollapsedClass(b(e),e)},this)).end()},d.prototype.addAriaAndCollapsedClass=function(a,b){var c=a.hasClass("in");a.attr("aria-expanded",c),b.toggleClass("collapsed",!c).attr("aria-expanded",c)};var e=a.fn.collapse;a.fn.collapse=c,a.fn.collapse.Constructor=d,a.fn.collapse.noConflict=function(){return a.fn.collapse=e,this},a(document).on("click.bs.collapse.data-api",'[data-toggle="collapse"]',function(d){var e=a(this);e.attr("data-target")||d.preventDefault();var f=b(e),g=f.data("bs.collapse"),h=g?"toggle":e.data();c.call(f,h)})}(jQuery),+function(a){"use strict";function b(b){b&&3===b.which||(a(e).remove(),a(f).each(function(){var d=a(this),e=c(d),f={relatedTarget:this};e.hasClass("open")&&(e.trigger(b=a.Event("hide.bs.dropdown",f)),b.isDefaultPrevented()||(d.attr("aria-expanded","false"),e.removeClass("open").trigger("hidden.bs.dropdown",f)))}))}function c(b){var c=b.attr("data-target");c||(c=b.attr("href"),c=c&&/#[A-Za-z]/.test(c)&&c.replace(/.*(?=#[^\s]*$)/,""));var d=c&&a(c);return d&&d.length?d:b.parent()}function d(b){return this.each(function(){var c=a(this),d=c.data("bs.dropdown");d||c.data("bs.dropdown",d=new g(this)),"string"==typeof b&&d[b].call(c)})}var e=".dropdown-backdrop",f='[data-toggle="dropdown"]',g=function(b){a(b).on("click.bs.dropdown",this.toggle)};g.VERSION="3.3.4",g.prototype.toggle=function(d){var e=a(this);if(!e.is(".disabled, :disabled")){var f=c(e),g=f.hasClass("open");if(b(),!g){"ontouchstart"in document.documentElement&&!f.closest(".navbar-nav").length&&a('',trigger:"hover focus",title:"",delay:0,html:!1,container:!1,viewport:{selector:"body",padding:0}},c.prototype.init=function(b,c,d){if(this.enabled=!0,this.type=b,this.$element=a(c),this.options=this.getOptions(d),this.$viewport=this.options.viewport&&a(this.options.viewport.selector||this.options.viewport),this.$element[0]instanceof document.constructor&&!this.options.selector)throw new Error("`selector` option must be specified when initializing "+this.type+" on the window.document object!");for(var e=this.options.trigger.split(" "),f=e.length;f--;){var g=e[f];if("click"==g)this.$element.on("click."+this.type,this.options.selector,a.proxy(this.toggle,this));else if("manual"!=g){var h="hover"==g?"mouseenter":"focusin",i="hover"==g?"mouseleave":"focusout";this.$element.on(h+"."+this.type,this.options.selector,a.proxy(this.enter,this)),this.$element.on(i+"."+this.type,this.options.selector,a.proxy(this.leave,this))}}this.options.selector?this._options=a.extend({},this.options,{trigger:"manual",selector:""}):this.fixTitle()},c.prototype.getDefaults=function(){return c.DEFAULTS},c.prototype.getOptions=function(b){return b=a.extend({},this.getDefaults(),this.$element.data(),b),b.delay&&"number"==typeof b.delay&&(b.delay={show:b.delay,hide:b.delay}),b},c.prototype.getDelegateOptions=function(){var b={},c=this.getDefaults();return this._options&&a.each(this._options,function(a,d){c[a]!=d&&(b[a]=d)}),b},c.prototype.enter=function(b){var c=b instanceof this.constructor?b:a(b.currentTarget).data("bs."+this.type);return c&&c.$tip&&c.$tip.is(":visible")?void(c.hoverState="in"):(c||(c=new this.constructor(b.currentTarget,this.getDelegateOptions()),a(b.currentTarget).data("bs."+this.type,c)),clearTimeout(c.timeout),c.hoverState="in",c.options.delay&&c.options.delay.show?void(c.timeout=setTimeout(function(){"in"==c.hoverState&&c.show()},c.options.delay.show)):c.show())},c.prototype.leave=function(b){var c=b instanceof this.constructor?b:a(b.currentTarget).data("bs."+this.type);return c||(c=new this.constructor(b.currentTarget,this.getDelegateOptions()),a(b.currentTarget).data("bs."+this.type,c)),clearTimeout(c.timeout),c.hoverState="out",c.options.delay&&c.options.delay.hide?void(c.timeout=setTimeout(function(){"out"==c.hoverState&&c.hide()},c.options.delay.hide)):c.hide()},c.prototype.show=function(){var b=a.Event("show.bs."+this.type);if(this.hasContent()&&this.enabled){this.$element.trigger(b);var d=a.contains(this.$element[0].ownerDocument.documentElement,this.$element[0]);if(b.isDefaultPrevented()||!d)return;var e=this,f=this.tip(),g=this.getUID(this.type);this.setContent(),f.attr("id",g),this.$element.attr("aria-describedby",g),this.options.animation&&f.addClass("fade");var h="function"==typeof this.options.placement?this.options.placement.call(this,f[0],this.$element[0]):this.options.placement,i=/\s?auto?\s?/i,j=i.test(h);j&&(h=h.replace(i,"")||"top"),f.detach().css({top:0,left:0,display:"block"}).addClass(h).data("bs."+this.type,this),this.options.container?f.appendTo(this.options.container):f.insertAfter(this.$element);var k=this.getPosition(),l=f[0].offsetWidth,m=f[0].offsetHeight;if(j){var n=h,o=this.options.container?a(this.options.container):this.$element.parent(),p=this.getPosition(o);h="bottom"==h&&k.bottom+m>p.bottom?"top":"top"==h&&k.top-mp.width?"left":"left"==h&&k.left-lg.top+g.height&&(e.top=g.top+g.height-i)}else{var j=b.left-f,k=b.left+f+c;jg.width&&(e.left=g.left+g.width-k)}return e},c.prototype.getTitle=function(){var a,b=this.$element,c=this.options;return a=b.attr("data-original-title")||("function"==typeof c.title?c.title.call(b[0]):c.title)},c.prototype.getUID=function(a){do a+=~~(1e6*Math.random());while(document.getElementById(a));return a},c.prototype.tip=function(){return this.$tip=this.$tip||a(this.options.template)},c.prototype.arrow=function(){return this.$arrow=this.$arrow||this.tip().find(".tooltip-arrow")},c.prototype.enable=function(){this.enabled=!0},c.prototype.disable=function(){this.enabled=!1},c.prototype.toggleEnabled=function(){this.enabled=!this.enabled},c.prototype.toggle=function(b){var c=this;b&&(c=a(b.currentTarget).data("bs."+this.type),c||(c=new this.constructor(b.currentTarget,this.getDelegateOptions()),a(b.currentTarget).data("bs."+this.type,c))),c.tip().hasClass("in")?c.leave(c):c.enter(c)},c.prototype.destroy=function(){var a=this;clearTimeout(this.timeout),this.hide(function(){a.$element.off("."+a.type).removeData("bs."+a.type)})};var d=a.fn.tooltip;a.fn.tooltip=b,a.fn.tooltip.Constructor=c,a.fn.tooltip.noConflict=function(){return a.fn.tooltip=d,this}}(jQuery),+function(a){"use strict";function b(b){return this.each(function(){var d=a(this),e=d.data("bs.popover"),f="object"==typeof b&&b;(e||!/destroy|hide/.test(b))&&(e||d.data("bs.popover",e=new c(this,f)),"string"==typeof b&&e[b]())})}var c=function(a,b){this.init("popover",a,b)};if(!a.fn.tooltip)throw new Error("Popover requires tooltip.js");c.VERSION="3.3.4",c.DEFAULTS=a.extend({},a.fn.tooltip.Constructor.DEFAULTS,{placement:"right",trigger:"click",content:"",template:''}),c.prototype=a.extend({},a.fn.tooltip.Constructor.prototype),c.prototype.constructor=c,c.prototype.getDefaults=function(){return c.DEFAULTS},c.prototype.setContent=function(){var a=this.tip(),b=this.getTitle(),c=this.getContent();a.find(".popover-title")[this.options.html?"html":"text"](b),a.find(".popover-content").children().detach().end()[this.options.html?"string"==typeof c?"html":"append":"text"](c),a.removeClass("fade top bottom left right in"),a.find(".popover-title").html()||a.find(".popover-title").hide()},c.prototype.hasContent=function(){return this.getTitle()||this.getContent()},c.prototype.getContent=function(){var a=this.$element,b=this.options;return a.attr("data-content")||("function"==typeof b.content?b.content.call(a[0]):b.content)},c.prototype.arrow=function(){return this.$arrow=this.$arrow||this.tip().find(".arrow")};var d=a.fn.popover;a.fn.popover=b,a.fn.popover.Constructor=c,a.fn.popover.noConflict=function(){return a.fn.popover=d,this}}(jQuery),+function(a){"use strict";function b(c,d){this.$body=a(document.body),this.$scrollElement=a(a(c).is(document.body)?window:c),this.options=a.extend({},b.DEFAULTS,d),this.selector=(this.options.target||"")+" .nav li > a",this.offsets=[],this.targets=[],this.activeTarget=null,this.scrollHeight=0,this.$scrollElement.on("scroll.bs.scrollspy",a.proxy(this.process,this)),this.refresh(),this.process()}function c(c){return this.each(function(){var d=a(this),e=d.data("bs.scrollspy"),f="object"==typeof c&&c;e||d.data("bs.scrollspy",e=new b(this,f)),"string"==typeof c&&e[c]()})}b.VERSION="3.3.4",b.DEFAULTS={offset:10},b.prototype.getScrollHeight=function(){return this.$scrollElement[0].scrollHeight||Math.max(this.$body[0].scrollHeight,document.documentElement.scrollHeight)},b.prototype.refresh=function(){var b=this,c="offset",d=0;this.offsets=[],this.targets=[],this.scrollHeight=this.getScrollHeight(),a.isWindow(this.$scrollElement[0])||(c="position",d=this.$scrollElement.scrollTop()),this.$body.find(this.selector).map(function(){var b=a(this),e=b.data("target")||b.attr("href"),f=/^#./.test(e)&&a(e);return f&&f.length&&f.is(":visible")&&[[f[c]().top+d,e]]||null}).sort(function(a,b){return a[0]-b[0]}).each(function(){b.offsets.push(this[0]),b.targets.push(this[1])})},b.prototype.process=function(){var a,b=this.$scrollElement.scrollTop()+this.options.offset,c=this.getScrollHeight(),d=this.options.offset+c-this.$scrollElement.height(),e=this.offsets,f=this.targets,g=this.activeTarget;if(this.scrollHeight!=c&&this.refresh(),b>=d)return g!=(a=f[f.length-1])&&this.activate(a);if(g&&b=e[a]&&(void 0===e[a+1]||b .dropdown-menu > .active").removeClass("active").end().find('[data-toggle="tab"]').attr("aria-expanded",!1),b.addClass("active").find('[data-toggle="tab"]').attr("aria-expanded",!0),h?(b[0].offsetWidth,b.addClass("in")):b.removeClass("fade"),b.parent(".dropdown-menu").length&&b.closest("li.dropdown").addClass("active").end().find('[data-toggle="tab"]').attr("aria-expanded",!0),e&&e()}var g=d.find("> .active"),h=e&&a.support.transition&&(g.length&&g.hasClass("fade")||!!d.find("> .fade").length);g.length&&h?g.one("bsTransitionEnd",f).emulateTransitionEnd(c.TRANSITION_DURATION):f(),g.removeClass("in")};var d=a.fn.tab;a.fn.tab=b,a.fn.tab.Constructor=c,a.fn.tab.noConflict=function(){return a.fn.tab=d,this};var e=function(c){c.preventDefault(),b.call(a(this),"show")};a(document).on("click.bs.tab.data-api",'[data-toggle="tab"]',e).on("click.bs.tab.data-api",'[data-toggle="pill"]',e)}(jQuery),+function(a){"use strict";function b(b){return this.each(function(){var d=a(this),e=d.data("bs.affix"),f="object"==typeof b&&b;e||d.data("bs.affix",e=new c(this,f)),"string"==typeof b&&e[b]()})}var c=function(b,d){this.options=a.extend({},c.DEFAULTS,d),this.$target=a(this.options.target).on("scroll.bs.affix.data-api",a.proxy(this.checkPosition,this)).on("click.bs.affix.data-api",a.proxy(this.checkPositionWithEventLoop,this)),this.$element=a(b),this.affixed=null,this.unpin=null,this.pinnedOffset=null,this.checkPosition()};c.VERSION="3.3.4",c.RESET="affix affix-top affix-bottom",c.DEFAULTS={offset:0,target:window},c.prototype.getState=function(a,b,c,d){var e=this.$target.scrollTop(),f=this.$element.offset(),g=this.$target.height();if(null!=c&&"top"==this.affixed)return c>e?"top":!1;if("bottom"==this.affixed)return null!=c?e+this.unpin<=f.top?!1:"bottom":a-d>=e+g?!1:"bottom";var h=null==this.affixed,i=h?e:f.top,j=h?g:b;return null!=c&&c>=e?"top":null!=d&&i+j>=a-d?"bottom":!1},c.prototype.getPinnedOffset=function(){if(this.pinnedOffset)return this.pinnedOffset;this.$element.removeClass(c.RESET).addClass("affix");var a=this.$target.scrollTop(),b=this.$element.offset();return this.pinnedOffset=b.top-a},c.prototype.checkPositionWithEventLoop=function(){setTimeout(a.proxy(this.checkPosition,this),1)},c.prototype.checkPosition=function(){if(this.$element.is(":visible")){var b=this.$element.height(),d=this.options.offset,e=d.top,f=d.bottom,g=a(document.body).height();"object"!=typeof d&&(f=e=d),"function"==typeof e&&(e=d.top(this.$element)),"function"==typeof f&&(f=d.bottom(this.$element));var h=this.getState(g,b,e,f);if(this.affixed!=h){null!=this.unpin&&this.$element.css("top","");var i="affix"+(h?"-"+h:""),j=a.Event(i+".bs.affix");if(this.$element.trigger(j),j.isDefaultPrevented())return;this.affixed=h,this.unpin="bottom"==h?this.getPinnedOffset():null,this.$element.removeClass(c.RESET).addClass(i).trigger(i.replace("affix","affixed")+".bs.affix")}"bottom"==h&&this.$element.offset({top:g-b-f})}};var d=a.fn.affix;a.fn.affix=b,a.fn.affix.Constructor=c,a.fn.affix.noConflict=function(){return a.fn.affix=d,this},a(window).on("load",function(){a('[data-spy="affix"]').each(function(){var c=a(this),d=c.data();d.offset=d.offset||{},null!=d.offsetBottom&&(d.offset.bottom=d.offsetBottom),null!=d.offsetTop&&(d.offset.top=d.offsetTop),b.call(c,d)})})}(jQuery); \ No newline at end of file diff --git a/system/admin/editor.md/tests/js/searchbox.js b/system/admin/editor.md/tests/js/searchbox.js new file mode 100644 index 00000000..ff95f563 --- /dev/null +++ b/system/admin/editor.md/tests/js/searchbox.js @@ -0,0 +1,674 @@ +/* global CodeMirror */ +/* global define */ + +(function(mod) { + 'use strict'; + + if (typeof exports === 'object' && typeof module === 'object') // CommonJS + mod(require('../../lib/codemirror')); + else if (typeof define === 'function' && define.amd) // AMD + define(['../../lib/codemirror'], mod); + else + mod(CodeMirror); +})(function(CodeMirror) { + 'use strict'; + + var Search; + + CodeMirror.defineOption('searchbox', false, function(cm) { + cm.addKeyMap({ + 'Ctrl-F': function() { + if (!Search) + Search = new SearchBox(cm); + + Search.show(); + }, + + 'Esc': function() { + if (Search && Search.isVisible()) { + Search.hide(); + + if (typeof event !== 'undefined') + event.stopPropagation(); + } + + return false; + }, + + 'Cmd-F': function() { + if (!Search) + Search = new SearchBox(cm); + + Search.show(); + } + }); + }); + + function SearchBox(cm) { + var self = this; + + init(); + + function initElements(el) { + self.searchBox = el.querySelector('.ace_search_form'); + self.replaceBox = el.querySelector('.ace_replace_form'); + self.searchOptions = el.querySelector('.ace_search_options'); + + self.regExpOption = el.querySelector('[action=toggleRegexpMode]'); + self.caseSensitiveOption = el.querySelector('[action=toggleCaseSensitive]'); + self.wholeWordOption = el.querySelector('[action=toggleWholeWords]'); + + self.searchInput = self.searchBox.querySelector('.ace_search_field'); + self.replaceInput = self.replaceBox.querySelector('.ace_search_field'); + } + + function init() { + var el = self.element = addHtml(); + + addStyle(); + + initElements(el); + bindKeys(); + + el.addEventListener('mousedown', function(e) { + setTimeout(function(){ + self.activeInput.focus(); + }, 0); + + e.stopPropagation(); + }); + + el.addEventListener('click', function(e) { + var t = e.target || e.srcElement; + var action = t.getAttribute('action'); + if (action && self[action]) + self[action](); + else if (self.commands[action]) + self.commands[action](); + + e.stopPropagation(); + }); + + self.searchInput.addEventListener('input', function() { + self.$onChange.schedule(20); + }); + + self.searchInput.addEventListener('focus', function() { + self.activeInput = self.searchInput; + }); + + self.replaceInput.addEventListener('focus', function() { + self.activeInput = self.replaceInput; + }); + + self.$onChange = delayedCall(function() { + self.find(false, false); + }); + } + + function bindKeys() { + var sb = self, + obj = { + 'Ctrl-F|Cmd-F|Ctrl-H|Command-Alt-F': function() { + var isReplace = sb.isReplace = !sb.isReplace; + sb.replaceBox.style.display = isReplace ? '' : 'none'; + sb[isReplace ? 'replaceInput' : 'searchInput'].focus(); + }, + 'Ctrl-G|Cmd-G': function() { + sb.findNext(); + }, + 'Ctrl-Shift-G|Cmd-Shift-G': function() { + sb.findPrev(); + }, + 'Esc': function() { + setTimeout(function() { sb.hide();}); + }, + 'Enter': function() { + if (sb.activeInput === sb.replaceInput) + sb.replace(); + sb.findNext(); + }, + 'Shift-Enter': function() { + if (sb.activeInput === sb.replaceInput) + sb.replace(); + sb.findPrev(); + }, + 'Alt-Enter': function() { + if (sb.activeInput === sb.replaceInput) + sb.replaceAll(); + sb.findAll(); + }, + 'Tab': function() { + if (self.activeInput === self.replaceInput) + self.searchInput.focus(); + else + self.replaceInput.focus(); + } + }; + + self.element.addEventListener('keydown', function(event) { + Object.keys(obj).some(function(name) { + var is = key(name, event); + + if (is) { + event.stopPropagation(); + event.preventDefault(); + obj[name](event); + } + + return is; + }); + }); + } + + this.commands = { + toggleRegexpMode: function() { + self.regExpOption.checked = !self.regExpOption.checked; + self.$syncOptions(); + }, + + toggleCaseSensitive: function() { + self.caseSensitiveOption.checked = !self.caseSensitiveOption.checked; + self.$syncOptions(); + }, + + toggleWholeWords: function() { + self.wholeWordOption.checked = !self.wholeWordOption.checked; + self.$syncOptions(); + } + }; + + this.$syncOptions = function() { + setCssClass(this.regExpOption, 'checked', this.regExpOption.checked); + setCssClass(this.wholeWordOption, 'checked', this.wholeWordOption.checked); + setCssClass(this.caseSensitiveOption, 'checked', this.caseSensitiveOption.checked); + + this.find(false, false); + }; + + this.find = function(skipCurrent, backwards) { + var value = this.searchInput.value, + options = { + skipCurrent: skipCurrent, + backwards: backwards, + regExp: this.regExpOption.checked, + caseSensitive: this.caseSensitiveOption.checked, + wholeWord: this.wholeWordOption.checked + }; + + find(value, options, function(searchCursor) { + var current = searchCursor.matches(false, searchCursor.from()); + cm.setSelection(current.from, current.to); + }); + }; + + function find(value, options, callback) { + var done, + noMatch, searchCursor, next, prev, matches, cursor, + position, + o = options, + is = true, + caseSensitive = o.caseSensitive, + regExp = o.regExp, + wholeWord = o.wholeWord; + + if (regExp || wholeWord) { + if (options.wholeWord) + value = '\\b' + value + '\\b'; + + value = RegExp(value); + } + + if (o.backwards) + position = o.skipCurrent ? 'from': 'to'; + else + position = o.skipCurrent ? 'to' : 'from'; + + cursor = cm.getCursor(position); + searchCursor = cm.getSearchCursor(value, cursor, !caseSensitive); + + next = searchCursor.findNext.bind(searchCursor), + prev = searchCursor.findPrevious.bind(searchCursor), + matches = searchCursor.matches.bind(searchCursor); + + if (o.backwards && !prev()) { + is = next(); + + if (is) { + cm.setCursor(cm.doc.size - 1, 0); + find(true, true, callback); + done = true; + } + } else if (!o.backwards && !next()) { + is = prev(); + + if (is) { + cm.setCursor(0, 0); + find(true, false, callback); + done = true; + } + } + + noMatch = !is && self.searchInput.value; + setCssClass(self.searchBox, 'ace_nomatch', noMatch); + + if (!done && is) + callback(searchCursor); + } + + this.findNext = function() { + this.find(true, false); + }; + + this.findPrev = function() { + this.find(true, true); + }; + + this.findAll = function(){ + /* + var range = this.editor.findAll(this.searchInput.value, { + regExp: this.regExpOption.checked, + caseSensitive: this.caseSensitiveOption.checked, + wholeWord: this.wholeWordOption.checked + }); + */ + + var value = this.searchInput.value, + range, + noMatch = !range && this.searchInput.value; + + setCssClass(this.searchBox, 'ace_nomatch', noMatch); + + if (cm.showMatchesOnScrollbar) + cm.showMatchesOnScrollbar(value); + + this.hide(); + }; + + this.replace = function() { + if (!cm.getOption('readOnly')) + cm.replaceSelection(this.replaceInput.value, 'start'); + }; + + this.replaceAndFindNext = function() { + if (!cm.getOption('readOnly')) { + this.editor.replace(this.replaceInput.value); + this.findNext(); + } + }; + + this.replaceAll = function() { + var value, + cursor, + from = this.searchInput.value, + to = this.replaceInput.value, + reg = RegExp(from, 'g'); + + if (!cm.getOption('readOnly')) { + cursor = cm.getCursor(); + value = cm.getValue(); + value = value.replace(reg, to); + + cm.setValue(value); + cm.setCursor(cursor); + } + }; + + this.hide = function() { + this.element.style.display = 'none'; + cm.focus(); + }; + + this.isVisible = function() { + var is = this.element.style.display === ''; + + return is; + }; + + this.show = function(value, isReplace) { + this.element.style.display = ''; + this.replaceBox.style.display = isReplace ? '' : 'none'; + + this.isReplace = isReplace; + + if (value) + this.searchInput.value = value; + + this.searchInput.focus(); + this.searchInput.select(); + }; + + this.isFocused = function() { + var el = document.activeElement; + return el === this.searchInput || el === this.replaceInput; + }; + + function addStyle() { + var style = document.createElement('style'), + css = [ + '.ace_search {', + 'background-color: #ddd;', + 'border: 1px solid #cbcbcb;', + 'border-top: 0 none;', + 'max-width: 325px;', + 'overflow: hidden;', + 'margin: 0;', + 'padding: 4px;', + 'padding-right: 6px;', + 'padding-bottom: 0;', + 'position: absolute;', + 'top: 0px;', + 'z-index: 99;', + 'white-space: normal;', + '}', + '.ace_search.left {', + 'border-left: 0 none;', + 'border-radius: 0px 0px 5px 0px;', + 'left: 0;', + '}', + '.ace_search.right {', + 'border-radius: 0px 0px 0px 5px;', + 'border-right: 0 none;', + 'right: 0;', + '}', + '.ace_search_form, .ace_replace_form {', + 'border-radius: 3px;', + 'border: 1px solid #cbcbcb;', + 'float: left;', + 'margin-bottom: 4px;', + 'overflow: hidden;', + '}', + '.ace_search_form.ace_nomatch {', + 'outline: 1px solid red;', + '}', + '.ace_search_field {', + 'background-color: white;', + 'border-right: 1px solid #cbcbcb;', + 'border: 0 none;', + '-webkit-box-sizing: border-box;', + '-moz-box-sizing: border-box;', + 'box-sizing: border-box;', + 'float: left;', + 'height: 22px;', + 'outline: 0;', + 'padding: 0 7px;', + 'width: 214px;', + 'margin: 0;', + '}', + '.ace_searchbtn,', + '.ace_replacebtn {', + 'background: #fff;', + 'border: 0 none;', + 'border-left: 1px solid #dcdcdc;', + 'cursor: pointer;', + 'float: left;', + 'height: 22px;', + 'margin: 0;', + 'padding: 0;', + 'position: relative;', + '}', + '.ace_searchbtn:last-child,', + '.ace_replacebtn:last-child {', + 'border-top-right-radius: 3px;', + 'border-bottom-right-radius: 3px;', + '}', + '.ace_searchbtn:disabled {', + 'background: none;', + 'cursor: default;', + '}', + '.ace_searchbtn {', + 'background-position: 50% 50%;', + 'background-repeat: no-repeat;', + 'width: 27px;', + '}', + '.ace_searchbtn.prev {', + 'background-image: url(); ', + '}', + '.ace_searchbtn.next {', + 'background-image: url(); ', + '}', + '.ace_searchbtn_close {', + 'background: url() no-repeat 50% 0;', + 'border-radius: 50%;', + 'border: 0 none;', + 'color: #656565;', + 'cursor: pointer;', + 'float: right;', + 'font: 16px/16px Arial;', + 'height: 14px;', + 'margin: 5px 1px 9px 5px;', + 'padding: 0;', + 'text-align: center;', + 'width: 14px;', + '}', + '.ace_searchbtn_close:hover {', + 'background-color: #656565;', + 'background-position: 50% 100%;', + 'color: white;', + '}', + '.ace_replacebtn.prev {', + 'width: 54px', + '}', + '.ace_replacebtn.next {', + 'width: 27px', + '}', + '.ace_button {', + 'margin-left: 2px;', + 'cursor: pointer;', + '-webkit-user-select: none;', + '-moz-user-select: none;', + '-o-user-select: none;', + '-ms-user-select: none;', + 'user-select: none;', + 'overflow: hidden;', + 'opacity: 0.7;', + 'border: 1px solid rgba(100,100,100,0.23);', + 'padding: 1px;', + '-moz-box-sizing: border-box;', + 'box-sizing: border-box;', + 'color: black;', + '}', + '.ace_button:hover {', + 'background-color: #eee;', + 'opacity:1;', + '}', + '.ace_button:active {', + 'background-color: #ddd;', + '}', + '.ace_button.checked {', + 'border-color: #3399ff;', + 'opacity:1;', + '}', + '.ace_search_options{', + 'margin-bottom: 3px;', + 'text-align: right;', + '-webkit-user-select: none;', + '-moz-user-select: none;', + '-o-user-select: none;', + '-ms-user-select: none;', + 'user-select: none;', + '}' + ].join(''); + + style.setAttribute('data-name', 'js-searchbox'); + + style.textContent = css; + + document.head.appendChild(style); + } + + function addHtml() { + var elSearch, + el = document.querySelector('.CodeMirror'), + div = document.createElement('div'), + html = [ + '' + ].join(''); + + div.innerHTML = html; + + elSearch = div.firstChild; + + el.parentElement.appendChild(elSearch); + + return elSearch; + } + } + + function setCssClass(el, className, condition) { + var list = el.classList; + + list[condition ? 'add' : 'remove'](className); + } + + function delayedCall(fcn, defaultTimeout) { + var timer, + callback = function() { + timer = null; + fcn(); + }, + + _self = function(timeout) { + if (!timer) + timer = setTimeout(callback, timeout || defaultTimeout); + }; + + _self.delay = function(timeout) { + timer && clearTimeout(timer); + timer = setTimeout(callback, timeout || defaultTimeout); + }; + _self.schedule = _self; + + _self.call = function() { + this.cancel(); + fcn(); + }; + + _self.cancel = function() { + timer && clearTimeout(timer); + timer = null; + }; + + _self.isPending = function() { + return timer; + }; + + return _self; + } + + /* https://github.com/coderaiser/key */ + function key(str, event) { + var right, + KEY = { + BACKSPACE : 8, + TAB : 9, + ENTER : 13, + ESC : 27, + + SPACE : 32, + PAGE_UP : 33, + PAGE_DOWN : 34, + END : 35, + HOME : 36, + UP : 38, + DOWN : 40, + + INSERT : 45, + DELETE : 46, + + INSERT_MAC : 96, + + ASTERISK : 106, + PLUS : 107, + MINUS : 109, + + F1 : 112, + F2 : 113, + F3 : 114, + F4 : 115, + F5 : 116, + F6 : 117, + F7 : 118, + F8 : 119, + F9 : 120, + F10 : 121, + + SLASH : 191, + TRA : 192, /* Typewritten Reverse Apostrophe (`) */ + BACKSLASH : 220 + }; + + keyCheck(str, event); + + right = str.split('|').some(function(combination) { + var wrong; + + wrong = combination.split('-').some(function(key) { + var right; + + switch(key) { + case 'Ctrl': + right = event.ctrlKey; + break; + + case 'Shift': + right = event.shiftKey; + break; + + case 'Alt': + right = event.altKey; + break; + + case 'Cmd': + right = event.metaKey; + break; + + default: + if (key.length === 1) + right = event.keyCode === key.charCodeAt(0); + else + Object.keys(KEY).some(function(name) { + var up = key.toUpperCase(); + + if (up === name) + right = event.keyCode === KEY[name]; + }); + break; + } + + return !right; + }); + + return !wrong; + }); + + return right; + } + + function keyCheck(str, event) { + if (typeof str !== 'string') + throw(Error('str should be string!')); + + if (typeof event !== 'object') + throw(Error('event should be object!')); + } + +}); diff --git a/system/admin/editor.md/tests/katex-tests.html b/system/admin/editor.md/tests/katex-tests.html new file mode 100644 index 00000000..f6c4f401 --- /dev/null +++ b/system/admin/editor.md/tests/katex-tests.html @@ -0,0 +1,132 @@ + + + + KaTeX Tests + + + + + + + http://khan.github.io/KaTeX/

                          + http://meta.wikimedia.org/wiki/Help:Displaying_a_formula +

                          + a^2 + a^{2+2} + a_2 + {x_2}^3 + x_2^3 + 10^{10^{8}} + a_{i,j} + _nP_k + E=MC^2 + \left \{ \frac{a}{b} \right \} \quad \left \lbrace \frac{a}{b} \right \rbrace + \left [ \frac{a}{b} \right ] \quad \left \lbrack \frac{a}{b} \right \rbrack + \left ( \frac{a}{b} \right ) + \left \langle \frac{a}{b} \right \rangle + x > y = 100 + c = \pm\sqrt{a^2 + b^2} + \left . \frac{A}{B} \right \} \to X + \left / \frac{a}{b} \right \backslash + \left \lfloor \frac{a}{b} \right \rfloor \left \lceil \frac{c}{d} \right \rceil + \frac{1}{2}=0.5 + \dfrac{k}{k-1} = 0.5 + \dbinom{n}{k} \binom{n}{k} + \oint_C x^3\, dx + 4y^2\, dy + \bigcap_1^n p \bigcup_1^k p + \phi_n(\kappa) = + \frac{1}{4\pi^2\kappa^2} \int_0^\infty + \frac{\sin(\kappa R)}{\kappa R} + \frac{\partial}{\partial R} + \left[R^2\frac{\partial D_n(R)}{\partial R}\right]\,dR + \sum_{m=1}^\infty\sum_{n=1}^\infty\frac{m^2\,n} + {3^m\left(m\,3^n+n\,3^m\right)} + e^{i \pi} + 1 = 0 + \left ( \frac{1}{2} \right ) + x_{1,2}=\frac{-b\pm\sqrt{\color{Red}b^2-4ac}}{2a} + {\color{Blue}x^2}+{\color{YellowOrange}2x}-{\color{OliveGreen}1} + \textstyle \sum_{k=1}^N k^2 + \dfrac{ \tfrac{1}{2}[1-(\tfrac{1}{2})^n] }{ 1-\tfrac{1}{2} } = s_n + \binom{n}{k} + 0+1+2+3+4+5+6+7+8+9+10+11+12+13+14+15+16+17+18+19+20+\cdots + f(x) = \int_{-\infty}^\infty + \hat f(\xi)\,e^{2 \pi i \xi x} + \,d\xi + \displaystyle \frac{1}{\Bigl(\sqrt{\phi \sqrt{5}}-\phi\Bigr) e^{\frac25 \pi}} = 1+\frac{e^{-2\pi}} {1+\frac{e^{-4\pi}} {1+\frac{e^{-6\pi}} {1+\frac{e^{-8\pi}} {1+\cdots} } } } + \displaystyle \left( \sum_{k=1}^n a_k b_k \right)^2 \leq \left( \sum_{k=1}^n a_k^2 \right) \left( \sum_{k=1}^n b_k^2 \right) + \displaystyle 1 + \frac{q^2}{(1-q)}+\frac{q^6}{(1-q)(1-q^2)}+\cdots = \prod_{j=0}^{\infty}\frac{1}{(1-q^{5j+2})(1-q^{5j+3})}, \quad\quad \text{for }\lvert q\rvert<1. + 2 = \left( + \frac{\left(3-x\right) \times 2}{3-x} + \right) + S_{\text{new}} = S_{\text{old}} - \frac{ \left( 5-T \right) ^2} {2} + x=\frac{-b\pm\sqrt{b^2-4ac}}{2a} + ax^2 + bx + c = 0\, + \int_a^x \!\!\!\int_a^s f(y)\,dy\,ds + = \int_a^x f(y)(x-y)\,dy + \sum_{m=1}^\infty\sum_{n=1}^\infty\frac{m^2\,n} + {3^m\left(m\,3^n+n\,3^m\right)} + u'' + p(x)u' + q(x)u=f(x),\quad x>a + |\bar{z}| = |z|, + |(\bar{z})^n| = |z|^n, + \arg(z^n) = n \arg(z) + \lim_{z\rightarrow z_0} f(z)=f(z_0) + \phi_n(\kappa) = + 0.033C_n^2\kappa^{-11/3},\quad + \frac{1}{L_0}\ll\kappa\ll\frac{1}{l_0} + \sum_{k=1}^N k^2 + \textstyle \sum_{k=1}^N k^2 + \prod_{i=1}^N x_i + \textstyle \prod_{i=1}^N x_i + \coprod_{i=1}^N x_i + \textstyle \coprod_{i=1}^N x_i + \int_{1}^{3}\frac{e^3/x}{x^2}\, dx + \int_C x^3\, dx + 4y^2\, dy + {}_1^2\!\Omega_3^4 + x', y'', f', f'' + \dot{x}, \ddot{x} + \hat a \ \bar b \ \vec c + \lessapprox \lesssim \eqslantless \leqslant \leqq \geqq \geqslant \eqslantgtr \gtrsim \gtrapprox + \smile \frown \wr \triangleleft \triangleright \infty \bot \top + \leftarrow \gets \rightarrow \to \nleftarrow \nrightarrow \leftrightarrow \nleftrightarrow \longleftarrow \longrightarrow \longleftrightarrow + \uparrow \downarrow \updownarrow \Uparrow \Downarrow \Updownarrow \nearrow \searrow \swarrow \nwarrow + \rightharpoonup \rightharpoondown \leftharpoonup \leftharpoondown \upharpoonleft \upharpoonright \downharpoonleft \downharpoonright \rightleftharpoons \leftrightharpoons + \curvearrowleft \circlearrowleft \Lsh \upuparrows \rightrightarrows \rightleftarrows \Rrightarrow \rightarrowtail \looparrowright + \curvearrowright \circlearrowright \Rsh \downdownarrows \leftleftarrows \leftrightarrows \Lleftarrow \leftarrowtail \looparrowleft + \mapsto \longmapsto \hookrightarrow \hookleftarrow \multimap \leftrightsquigarrow \rightsquigarrow + \Diamond \Box \triangle \angle \perp \mid \nmid \| 45^\circ + +\sim \approx \simeq \cong \dot= \overset{\underset{\mathrm{def}}{}}{=} + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/system/admin/editor.md/tests/marked-@at-test.html b/system/admin/editor.md/tests/marked-@at-test.html new file mode 100644 index 00000000..821f308f --- /dev/null +++ b/system/admin/editor.md/tests/marked-@at-test.html @@ -0,0 +1,221 @@ + + + + Marked @ Test + + + + + + + +
                           ~~@mentions~~, #refs @tylerlong `inline code @tylerlong`, [links](), **formatting**, and tags supported @pandao;
                          +list syntax required (any unordered or ordered list supported) @pandao;
                          +this is @pandao a complete item @pandao;
                          +link [@pandao](https://github.com/pandao "@pandao") @
                          +link [@pandao](https://github.com/pandao "@pandao") 
                          +this is an incomplete item **@pandao**;
                          +*@pandao* this is an incomplete item ___@pandao___;
                          +# Github: @pandao
                          +## Github: @pandao
                          +### Github: @tylerlong
                          +#### Github: @tylerlong
                          +##### Github: @tylerlong
                          +###### Github: @tylerlong
                          + 
                          +- dafssdfsdaf@chjj dfsdfsdf
                          +- dafssdfsdaf@chjj dfsdfsdf     
                          +    - dafssdfsdaf@chjj dfsdfsdf
                          +    - dafss@pandao dfsdaf@chjj dfsdfsdf
                          +- dafssd:  @pandao fsdaf@chjj dfsdfsdf @codemirror  @pandao
                          +    + dafssdfsdaf@chjj dfsdfsdf
                          +    + dafss@pandaodfsdaf@chjj dfsdfsdf
                          +
                          +
                          +1. @chjj 第一行@pandao fsdaf@chjj dfsdfsdf :fa-save::  @pandao
                          +    - dafssdfsdaf@chjj dfsdfsdf
                          +    - dafss@pandao dfsdaf@chjj dfsdfsdf
                          +2. @chjj 第二行@pandao fsdaf@chjj dfsdfsdf @codemirror  @pandao
                          +3. 第三行@pandao fsdaf@chjj dfsdfsdf :fa-save::  @pandao
                          +
                          +> Blockquotes @pandao
                          +
                          +> dd@pandao引用文本(Blockquotes @pandao)fdasfad @_pandao fdasfad @xxx454xxx fdasfad @xx_x454xxx454
                          +
                          +|@pandao First Header  | Second@pandao Header@pandao |
                          +| ------------- | ------------- |
                          +| Content@pandao Cell  | @pandao Content Cell @pandao|
                          +| Con@pandao tent Cell@pandao  | Content@pan-dao Cell dfsdfsdf @pan_dao |
                          +
                          +dsfdf@pandao fasdfsdfsfddffd@pandao
                          +
                          +    dfasfasdfasdf:bangbang:
                          +
                          +This is an H1 @pandao
                          +=============
                          +
                          +This @pandao an H2 @pandao
                          +-------------
                          +
                          + + + + + \ No newline at end of file diff --git a/system/admin/editor.md/tests/marked-emoji-test.html b/system/admin/editor.md/tests/marked-emoji-test.html new file mode 100644 index 00000000..4b60c2eb --- /dev/null +++ b/system/admin/editor.md/tests/marked-emoji-test.html @@ -0,0 +1,231 @@ + + + + Marked Emoji Test + + + + + + + +
                          +
                          > Blockquotes
                          +dasfsadfasdf:fa-edit: :warning: :smiley:dsafsdfsad\:fdsfdf\:f dfdf:   :fa-save::fa-star:  :fa-truck:
                          +**fdfasd:smiley:dsfsdfsfd** ~~fsdfds:smiley:dfsdfsdf :fa-info:~~
                          +*dsfdfsfd:smiley:dsfsfdsfd:smiley:dsfdf*
                          +___Emphasis :fa-gear: Italic:smiley:___  __Emphasis:smiley:__
                          +# H1 dsfdfsfd:smiley:dsfsfdsfd:smiley:dsfdf
                          +## H2 dsfdfsfd:smiley:dsfsfdsfd:smiley:dsfdf
                          +### H3 dsfdfsfd:smiley:dsfsfdsfd:smiley:dsfdf:fa-edit: fdsfsdf:fa-save:dsfsdf
                          +#### H4 dsfdfsfd:smiley:dsfsfdsfd:smiley:dsfdf:fa-edit: fdsfsdf:fa-save:dsfsdf
                          +##### H5 dsfdfsfd:smiley:dsfsfdsfd:smiley:dsfdf:fa-edit: fdsfsdf:fa-save:dsfsdf
                          +###### H6 dsfdfsfd:smiley:dsfsfdsfd:smiley:dsfdf:fa-edit: fdsfsdf:fa-star:dsfsdf
                          +[:smiley:](http://www.emoji-cheat-sheet.com/ "link + emoji") link + emoji
                          +- dafssdfsdaf:smiley:dfsdfsdf
                          +- dafssdfsdaf:smiley:dfsdfsdf     
                          +    - dafssdfsdaf:smiley:dfsdfsdf
                          +    - dafss:fa-truck:dfsdaf:smiley:dfsdfsdf
                          +- dafssd:  :fa-truck:fsdaf:smiley:dfsdfsdf :fa-star::  :fa-truck:
                          +    + dafssdfsdaf:smiley:dfsdfsdf
                          +    + dafss:fa-truck:dfsdaf:smiley:dfsdfsdf
                          +
                          +
                          +1. :smiley:第一行:fa-truck:fsdaf:smiley:dfsdfsdf :fa-save::  :fa-truck:
                          +    - dafssdfsdaf:smiley:dfsdfsdf
                          +    - dafss:fa-truck:dfsdaf:smiley:dfsdfsdf
                          +2. :smiley:第二行:fa-truck:fsdaf:smiley:dfsdfsdf :fa-star::  :fa-truck:
                          +3. 第三行:fa-truck:fsdaf:smiley:dfsdfsdf :fa-save::  :fa-truck:
                          +
                          +> Blockquotes
                          +
                          +> dd:smiley:引用文本(Blockquotes:smiley:)fdasfad :fa-star:: dfd :fa-truck:
                          +
                          +|:100: First Header  | Second:smiley: Header:smiley: |
                          +| ------------- | ------------- |
                          +| Content:fa-truck: Cell  | :smiley:Content Cell :smiley: |
                          +| Con:fa-truck:tent Cell:bangbang:  | Content Cell dfsdfsdf :fa-star: :dfdf  :fa-truck: |
                          +
                          +:fa-heart:fasdfsdfsfddffd:editormd-logo: :editormd-logo: :editormd-logo-4x:
                          +
                          +    dfasfasdfasdf:bangbang:
                          +
                          +This is an H1 :editormd-logo-4x:
                          +=============
                          +
                          +This  :fa-save::  :fa-truck:is an H2 :100:
                          +-------------
                          +
                          + + + + + \ No newline at end of file diff --git a/system/admin/editor.md/tests/marked-heading-link-test.html b/system/admin/editor.md/tests/marked-heading-link-test.html new file mode 100644 index 00000000..108db879 --- /dev/null +++ b/system/admin/editor.md/tests/marked-heading-link-test.html @@ -0,0 +1,138 @@ + + + + Marked heading link Test + + + + + + + + + + + \ No newline at end of file diff --git a/system/admin/editor.md/tests/marked-todo-list-test.html b/system/admin/editor.md/tests/marked-todo-list-test.html new file mode 100644 index 00000000..50a465e0 --- /dev/null +++ b/system/admin/editor.md/tests/marked-todo-list-test.html @@ -0,0 +1,61 @@ + + + + Marked Emoji Test + + + + + + + + + + + \ No newline at end of file diff --git a/system/admin/editor.md/tests/qunit/qunit-1.16.0.css b/system/admin/editor.md/tests/qunit/qunit-1.16.0.css new file mode 100644 index 00000000..385a1ce0 --- /dev/null +++ b/system/admin/editor.md/tests/qunit/qunit-1.16.0.css @@ -0,0 +1,264 @@ +/*! + * QUnit 1.16.0 + * http://qunitjs.com/ + * + * Copyright 2006, 2014 jQuery Foundation and other contributors + * Released under the MIT license + * http://jquery.org/license + * + * Date: 2014-12-03T16:32Z + */ + +/** Font Family and Sizes */ + +#qunit-tests, #qunit-header, #qunit-banner, #qunit-testrunner-toolbar, #qunit-userAgent, #qunit-testresult { + font-family: "Helvetica Neue Light", "HelveticaNeue-Light", "Helvetica Neue", Calibri, Helvetica, Arial, sans-serif; +} + +#qunit-testrunner-toolbar, #qunit-userAgent, #qunit-testresult, #qunit-tests li { font-size: small; } +#qunit-tests { font-size: smaller; } + + +/** Resets */ + +#qunit-tests, #qunit-header, #qunit-banner, #qunit-userAgent, #qunit-testresult, #qunit-modulefilter { + margin: 0; + padding: 0; +} + + +/** Header */ + +#qunit-header { + padding: 0.5em 0 0.5em 1em; + + color: #8699A4; + background-color: #0D3349; + + font-size: 1.5em; + line-height: 1em; + font-weight: 400; + + border-radius: 5px 5px 0 0; +} + +#qunit-header a { + text-decoration: none; + color: #C2CCD1; +} + +#qunit-header a:hover, +#qunit-header a:focus { + color: #FFF; +} + +#qunit-testrunner-toolbar label { + display: inline-block; + padding: 0 0.5em 0 0.1em; +} + +#qunit-banner { + height: 5px; +} + +#qunit-testrunner-toolbar { + padding: 0.5em 1em 0.5em 1em; + color: #5E740B; + background-color: #EEE; + overflow: hidden; +} + +#qunit-userAgent { + padding: 0.5em 1em 0.5em 1em; + background-color: #2B81AF; + color: #FFF; + text-shadow: rgba(0, 0, 0, 0.5) 2px 2px 1px; +} + +#qunit-modulefilter-container { + float: right; +} + +/** Tests: Pass/Fail */ + +#qunit-tests { + list-style-position: inside; +} + +#qunit-tests li { + padding: 0.4em 1em 0.4em 1em; + border-bottom: 1px solid #FFF; + list-style-position: inside; +} + +#qunit-tests > li { + display: none; +} + +#qunit-tests li.pass, #qunit-tests li.running, #qunit-tests li.fail { + display: list-item; +} + +#qunit-tests.hidepass li.pass, #qunit-tests.hidepass li.running { + display: none; +} + +#qunit-tests li strong { + cursor: pointer; +} + +#qunit-tests li.skipped strong { + cursor: default; +} + +#qunit-tests li a { + padding: 0.5em; + color: #C2CCD1; + text-decoration: none; +} +#qunit-tests li a:hover, +#qunit-tests li a:focus { + color: #000; +} + +#qunit-tests li .runtime { + float: right; + font-size: smaller; +} + +.qunit-assert-list { + margin-top: 0.5em; + padding: 0.5em; + + background-color: #FFF; + + border-radius: 5px; +} + +.qunit-collapsed { + display: none; +} + +#qunit-tests table { + border-collapse: collapse; + margin-top: 0.2em; +} + +#qunit-tests th { + text-align: right; + vertical-align: top; + padding: 0 0.5em 0 0; +} + +#qunit-tests td { + vertical-align: top; +} + +#qunit-tests pre { + margin: 0; + white-space: pre-wrap; + word-wrap: break-word; +} + +#qunit-tests del { + background-color: #E0F2BE; + color: #374E0C; + text-decoration: none; +} + +#qunit-tests ins { + background-color: #FFCACA; + color: #500; + text-decoration: none; +} + +/*** Test Counts */ + +#qunit-tests b.counts { color: #000; } +#qunit-tests b.passed { color: #5E740B; } +#qunit-tests b.failed { color: #710909; } + +#qunit-tests li li { + padding: 5px; + background-color: #FFF; + border-bottom: none; + list-style-position: inside; +} + +/*** Passing Styles */ + +#qunit-tests li li.pass { + color: #3C510C; + background-color: #FFF; + border-left: 10px solid #C6E746; +} + +#qunit-tests .pass { color: #528CE0; background-color: #D2E0E6; } +#qunit-tests .pass .test-name { color: #366097; } + +#qunit-tests .pass .test-actual, +#qunit-tests .pass .test-expected { color: #999; } + +#qunit-banner.qunit-pass { background-color: #C6E746; } + +/*** Failing Styles */ + +#qunit-tests li li.fail { + color: #710909; + background-color: #FFF; + border-left: 10px solid #EE5757; + white-space: pre; +} + +#qunit-tests > li:last-child { + border-radius: 0 0 5px 5px; +} + +#qunit-tests .fail { color: #000; background-color: #EE5757; } +#qunit-tests .fail .test-name, +#qunit-tests .fail .module-name { color: #000; } + +#qunit-tests .fail .test-actual { color: #EE5757; } +#qunit-tests .fail .test-expected { color: #008000; } + +#qunit-banner.qunit-fail { background-color: #EE5757; } + +/*** Skipped tests */ + +#qunit-tests .skipped { + background-color: #EBECE9; +} + +#qunit-tests .qunit-skipped-label { + background-color: #F4FF77; + display: inline-block; + font-style: normal; + color: #366097; + line-height: 1.8em; + padding: 0 0.5em; + margin: -0.4em 0.4em -0.4em 0; +} + +/** Result */ + +#qunit-testresult { + padding: 0.5em 1em 0.5em 1em; + + color: #2B81AF; + background-color: #D2E0E6; + + border-bottom: 1px solid #FFF; +} +#qunit-testresult .module-name { + font-weight: 700; +} + +/** Fixture */ + +#qunit-fixture { + position: absolute; + top: -10000px; + left: -10000px; + width: 1000px; + height: 1000px; +} diff --git a/system/admin/editor.md/tests/qunit/qunit-1.16.0.js b/system/admin/editor.md/tests/qunit/qunit-1.16.0.js new file mode 100644 index 00000000..82020d40 --- /dev/null +++ b/system/admin/editor.md/tests/qunit/qunit-1.16.0.js @@ -0,0 +1,2819 @@ +/*! + * QUnit 1.16.0 + * http://qunitjs.com/ + * + * Copyright 2006, 2014 jQuery Foundation and other contributors + * Released under the MIT license + * http://jquery.org/license + * + * Date: 2014-12-03T16:32Z + */ + +(function( window ) { + +var QUnit, + config, + onErrorFnPrev, + loggingCallbacks = {}, + fileName = ( sourceFromStacktrace( 0 ) || "" ).replace( /(:\d+)+\)?/, "" ).replace( /.+\//, "" ), + toString = Object.prototype.toString, + hasOwn = Object.prototype.hasOwnProperty, + // Keep a local reference to Date (GH-283) + Date = window.Date, + now = Date.now || function() { + return new Date().getTime(); + }, + globalStartCalled = false, + runStarted = false, + setTimeout = window.setTimeout, + clearTimeout = window.clearTimeout, + defined = { + document: window.document !== undefined, + setTimeout: window.setTimeout !== undefined, + sessionStorage: (function() { + var x = "qunit-test-string"; + try { + sessionStorage.setItem( x, x ); + sessionStorage.removeItem( x ); + return true; + } catch ( e ) { + return false; + } + }()) + }, + /** + * Provides a normalized error string, correcting an issue + * with IE 7 (and prior) where Error.prototype.toString is + * not properly implemented + * + * Based on http://es5.github.com/#x15.11.4.4 + * + * @param {String|Error} error + * @return {String} error message + */ + errorString = function( error ) { + var name, message, + errorString = error.toString(); + if ( errorString.substring( 0, 7 ) === "[object" ) { + name = error.name ? error.name.toString() : "Error"; + message = error.message ? error.message.toString() : ""; + if ( name && message ) { + return name + ": " + message; + } else if ( name ) { + return name; + } else if ( message ) { + return message; + } else { + return "Error"; + } + } else { + return errorString; + } + }, + /** + * Makes a clone of an object using only Array or Object as base, + * and copies over the own enumerable properties. + * + * @param {Object} obj + * @return {Object} New object with only the own properties (recursively). + */ + objectValues = function( obj ) { + var key, val, + vals = QUnit.is( "array", obj ) ? [] : {}; + for ( key in obj ) { + if ( hasOwn.call( obj, key ) ) { + val = obj[ key ]; + vals[ key ] = val === Object( val ) ? objectValues( val ) : val; + } + } + return vals; + }; + +QUnit = {}; + +/** + * Config object: Maintain internal state + * Later exposed as QUnit.config + * `config` initialized at top of scope + */ +config = { + // The queue of tests to run + queue: [], + + // block until document ready + blocking: true, + + // when enabled, show only failing tests + // gets persisted through sessionStorage and can be changed in UI via checkbox + hidepassed: false, + + // by default, run previously failed tests first + // very useful in combination with "Hide passed tests" checked + reorder: true, + + // by default, modify document.title when suite is done + altertitle: true, + + // by default, scroll to top of the page when suite is done + scrolltop: true, + + // when enabled, all tests must call expect() + requireExpects: false, + + // add checkboxes that are persisted in the query-string + // when enabled, the id is set to `true` as a `QUnit.config` property + urlConfig: [ + { + id: "hidepassed", + label: "Hide passed tests", + tooltip: "Only show tests and assertions that fail. Stored as query-strings." + }, + { + id: "noglobals", + label: "Check for Globals", + tooltip: "Enabling this will test if any test introduces new properties on the " + + "`window` object. Stored as query-strings." + }, + { + id: "notrycatch", + label: "No try-catch", + tooltip: "Enabling this will run tests outside of a try-catch block. Makes debugging " + + "exceptions in IE reasonable. Stored as query-strings." + } + ], + + // Set of all modules. + modules: [], + + // The first unnamed module + currentModule: { + name: "", + tests: [] + }, + + callbacks: {} +}; + +// Push a loose unnamed module to the modules collection +config.modules.push( config.currentModule ); + +// Initialize more QUnit.config and QUnit.urlParams +(function() { + var i, current, + location = window.location || { search: "", protocol: "file:" }, + params = location.search.slice( 1 ).split( "&" ), + length = params.length, + urlParams = {}; + + if ( params[ 0 ] ) { + for ( i = 0; i < length; i++ ) { + current = params[ i ].split( "=" ); + current[ 0 ] = decodeURIComponent( current[ 0 ] ); + + // allow just a key to turn on a flag, e.g., test.html?noglobals + current[ 1 ] = current[ 1 ] ? decodeURIComponent( current[ 1 ] ) : true; + if ( urlParams[ current[ 0 ] ] ) { + urlParams[ current[ 0 ] ] = [].concat( urlParams[ current[ 0 ] ], current[ 1 ] ); + } else { + urlParams[ current[ 0 ] ] = current[ 1 ]; + } + } + } + + QUnit.urlParams = urlParams; + + // String search anywhere in moduleName+testName + config.filter = urlParams.filter; + + config.testId = []; + if ( urlParams.testId ) { + + // Ensure that urlParams.testId is an array + urlParams.testId = [].concat( urlParams.testId ); + for ( i = 0; i < urlParams.testId.length; i++ ) { + config.testId.push( urlParams.testId[ i ] ); + } + } + + // Figure out if we're running the tests from a server or not + QUnit.isLocal = location.protocol === "file:"; +}()); + +// Root QUnit object. +// `QUnit` initialized at top of scope +extend( QUnit, { + + // call on start of module test to prepend name to all tests + module: function( name, testEnvironment ) { + var currentModule = { + name: name, + testEnvironment: testEnvironment, + tests: [] + }; + + // DEPRECATED: handles setup/teardown functions, + // beforeEach and afterEach should be used instead + if ( testEnvironment && testEnvironment.setup ) { + testEnvironment.beforeEach = testEnvironment.setup; + delete testEnvironment.setup; + } + if ( testEnvironment && testEnvironment.teardown ) { + testEnvironment.afterEach = testEnvironment.teardown; + delete testEnvironment.teardown; + } + + config.modules.push( currentModule ); + config.currentModule = currentModule; + }, + + // DEPRECATED: QUnit.asyncTest() will be removed in QUnit 2.0. + asyncTest: function( testName, expected, callback ) { + if ( arguments.length === 2 ) { + callback = expected; + expected = null; + } + + QUnit.test( testName, expected, callback, true ); + }, + + test: function( testName, expected, callback, async ) { + var test; + + if ( arguments.length === 2 ) { + callback = expected; + expected = null; + } + + test = new Test({ + testName: testName, + expected: expected, + async: async, + callback: callback + }); + + test.queue(); + }, + + skip: function( testName ) { + var test = new Test({ + testName: testName, + skip: true + }); + + test.queue(); + }, + + // DEPRECATED: The functionality of QUnit.start() will be altered in QUnit 2.0. + // In QUnit 2.0, invoking it will ONLY affect the `QUnit.config.autostart` blocking behavior. + start: function( count ) { + var globalStartAlreadyCalled = globalStartCalled; + + if ( !config.current ) { + globalStartCalled = true; + + if ( runStarted ) { + throw new Error( "Called start() outside of a test context while already started" ); + } else if ( globalStartAlreadyCalled || count > 1 ) { + throw new Error( "Called start() outside of a test context too many times" ); + } else if ( config.autostart ) { + throw new Error( "Called start() outside of a test context when " + + "QUnit.config.autostart was true" ); + } else if ( !config.pageLoaded ) { + + // The page isn't completely loaded yet, so bail out and let `QUnit.load` handle it + config.autostart = true; + return; + } + } else { + + // If a test is running, adjust its semaphore + config.current.semaphore -= count || 1; + + // Don't start until equal number of stop-calls + if ( config.current.semaphore > 0 ) { + return; + } + + // throw an Error if start is called more often than stop + if ( config.current.semaphore < 0 ) { + config.current.semaphore = 0; + + QUnit.pushFailure( + "Called start() while already started (test's semaphore was 0 already)", + sourceFromStacktrace( 2 ) + ); + return; + } + } + + resumeProcessing(); + }, + + // DEPRECATED: QUnit.stop() will be removed in QUnit 2.0. + stop: function( count ) { + + // If there isn't a test running, don't allow QUnit.stop() to be called + if ( !config.current ) { + throw new Error( "Called stop() outside of a test context" ); + } + + // If a test is running, adjust its semaphore + config.current.semaphore += count || 1; + + pauseProcessing(); + }, + + config: config, + + // Safe object type checking + is: function( type, obj ) { + return QUnit.objectType( obj ) === type; + }, + + objectType: function( obj ) { + if ( typeof obj === "undefined" ) { + return "undefined"; + } + + // Consider: typeof null === object + if ( obj === null ) { + return "null"; + } + + var match = toString.call( obj ).match( /^\[object\s(.*)\]$/ ), + type = match && match[ 1 ] || ""; + + switch ( type ) { + case "Number": + if ( isNaN( obj ) ) { + return "nan"; + } + return "number"; + case "String": + case "Boolean": + case "Array": + case "Date": + case "RegExp": + case "Function": + return type.toLowerCase(); + } + if ( typeof obj === "object" ) { + return "object"; + } + return undefined; + }, + + url: function( params ) { + params = extend( extend( {}, QUnit.urlParams ), params ); + var key, + querystring = "?"; + + for ( key in params ) { + if ( hasOwn.call( params, key ) ) { + querystring += encodeURIComponent( key ); + if ( params[ key ] !== true ) { + querystring += "=" + encodeURIComponent( params[ key ] ); + } + querystring += "&"; + } + } + return location.protocol + "//" + location.host + + location.pathname + querystring.slice( 0, -1 ); + }, + + extend: extend, + + load: function() { + config.pageLoaded = true; + + // Initialize the configuration options + extend( config, { + stats: { all: 0, bad: 0 }, + moduleStats: { all: 0, bad: 0 }, + started: 0, + updateRate: 1000, + autostart: true, + filter: "" + }, true ); + + config.blocking = false; + + if ( config.autostart ) { + resumeProcessing(); + } + } +}); + +// Register logging callbacks +(function() { + var i, l, key, + callbacks = [ "begin", "done", "log", "testStart", "testDone", + "moduleStart", "moduleDone" ]; + + function registerLoggingCallback( key ) { + var loggingCallback = function( callback ) { + if ( QUnit.objectType( callback ) !== "function" ) { + throw new Error( + "QUnit logging methods require a callback function as their first parameters." + ); + } + + config.callbacks[ key ].push( callback ); + }; + + // DEPRECATED: This will be removed on QUnit 2.0.0+ + // Stores the registered functions allowing restoring + // at verifyLoggingCallbacks() if modified + loggingCallbacks[ key ] = loggingCallback; + + return loggingCallback; + } + + for ( i = 0, l = callbacks.length; i < l; i++ ) { + key = callbacks[ i ]; + + // Initialize key collection of logging callback + if ( QUnit.objectType( config.callbacks[ key ] ) === "undefined" ) { + config.callbacks[ key ] = []; + } + + QUnit[ key ] = registerLoggingCallback( key ); + } +})(); + +// `onErrorFnPrev` initialized at top of scope +// Preserve other handlers +onErrorFnPrev = window.onerror; + +// Cover uncaught exceptions +// Returning true will suppress the default browser handler, +// returning false will let it run. +window.onerror = function( error, filePath, linerNr ) { + var ret = false; + if ( onErrorFnPrev ) { + ret = onErrorFnPrev( error, filePath, linerNr ); + } + + // Treat return value as window.onerror itself does, + // Only do our handling if not suppressed. + if ( ret !== true ) { + if ( QUnit.config.current ) { + if ( QUnit.config.current.ignoreGlobalErrors ) { + return true; + } + QUnit.pushFailure( error, filePath + ":" + linerNr ); + } else { + QUnit.test( "global failure", extend(function() { + QUnit.pushFailure( error, filePath + ":" + linerNr ); + }, { validTest: true } ) ); + } + return false; + } + + return ret; +}; + +function done() { + var runtime, passed; + + config.autorun = true; + + // Log the last module results + if ( config.previousModule ) { + runLoggingCallbacks( "moduleDone", { + name: config.previousModule.name, + tests: config.previousModule.tests, + failed: config.moduleStats.bad, + passed: config.moduleStats.all - config.moduleStats.bad, + total: config.moduleStats.all, + runtime: now() - config.moduleStats.started + }); + } + delete config.previousModule; + + runtime = now() - config.started; + passed = config.stats.all - config.stats.bad; + + runLoggingCallbacks( "done", { + failed: config.stats.bad, + passed: passed, + total: config.stats.all, + runtime: runtime + }); +} + +// Doesn't support IE6 to IE9 +// See also https://developer.mozilla.org/en/JavaScript/Reference/Global_Objects/Error/Stack +function extractStacktrace( e, offset ) { + offset = offset === undefined ? 4 : offset; + + var stack, include, i; + + if ( e.stacktrace ) { + + // Opera 12.x + return e.stacktrace.split( "\n" )[ offset + 3 ]; + } else if ( e.stack ) { + + // Firefox, Chrome, Safari 6+, IE10+, PhantomJS and Node + stack = e.stack.split( "\n" ); + if ( /^error$/i.test( stack[ 0 ] ) ) { + stack.shift(); + } + if ( fileName ) { + include = []; + for ( i = offset; i < stack.length; i++ ) { + if ( stack[ i ].indexOf( fileName ) !== -1 ) { + break; + } + include.push( stack[ i ] ); + } + if ( include.length ) { + return include.join( "\n" ); + } + } + return stack[ offset ]; + } else if ( e.sourceURL ) { + + // Safari < 6 + // exclude useless self-reference for generated Error objects + if ( /qunit.js$/.test( e.sourceURL ) ) { + return; + } + + // for actual exceptions, this is useful + return e.sourceURL + ":" + e.line; + } +} + +function sourceFromStacktrace( offset ) { + var e = new Error(); + if ( !e.stack ) { + try { + throw e; + } catch ( err ) { + // This should already be true in most browsers + e = err; + } + } + return extractStacktrace( e, offset ); +} + +function synchronize( callback, last ) { + if ( QUnit.objectType( callback ) === "array" ) { + while ( callback.length ) { + synchronize( callback.shift() ); + } + return; + } + config.queue.push( callback ); + + if ( config.autorun && !config.blocking ) { + process( last ); + } +} + +function process( last ) { + function next() { + process( last ); + } + var start = now(); + config.depth = config.depth ? config.depth + 1 : 1; + + while ( config.queue.length && !config.blocking ) { + if ( !defined.setTimeout || config.updateRate <= 0 || + ( ( now() - start ) < config.updateRate ) ) { + if ( config.current ) { + + // Reset async tracking for each phase of the Test lifecycle + config.current.usedAsync = false; + } + config.queue.shift()(); + } else { + setTimeout( next, 13 ); + break; + } + } + config.depth--; + if ( last && !config.blocking && !config.queue.length && config.depth === 0 ) { + done(); + } +} + +function begin() { + var i, l, + modulesLog = []; + + // If the test run hasn't officially begun yet + if ( !config.started ) { + + // Record the time of the test run's beginning + config.started = now(); + + verifyLoggingCallbacks(); + + // Delete the loose unnamed module if unused. + if ( config.modules[ 0 ].name === "" && config.modules[ 0 ].tests.length === 0 ) { + config.modules.shift(); + } + + // Avoid unnecessary information by not logging modules' test environments + for ( i = 0, l = config.modules.length; i < l; i++ ) { + modulesLog.push({ + name: config.modules[ i ].name, + tests: config.modules[ i ].tests + }); + } + + // The test run is officially beginning now + runLoggingCallbacks( "begin", { + totalTests: Test.count, + modules: modulesLog + }); + } + + config.blocking = false; + process( true ); +} + +function resumeProcessing() { + runStarted = true; + + // A slight delay to allow this iteration of the event loop to finish (more assertions, etc.) + if ( defined.setTimeout ) { + setTimeout(function() { + if ( config.current && config.current.semaphore > 0 ) { + return; + } + if ( config.timeout ) { + clearTimeout( config.timeout ); + } + + begin(); + }, 13 ); + } else { + begin(); + } +} + +function pauseProcessing() { + config.blocking = true; + + if ( config.testTimeout && defined.setTimeout ) { + clearTimeout( config.timeout ); + config.timeout = setTimeout(function() { + if ( config.current ) { + config.current.semaphore = 0; + QUnit.pushFailure( "Test timed out", sourceFromStacktrace( 2 ) ); + } else { + throw new Error( "Test timed out" ); + } + resumeProcessing(); + }, config.testTimeout ); + } +} + +function saveGlobal() { + config.pollution = []; + + if ( config.noglobals ) { + for ( var key in window ) { + if ( hasOwn.call( window, key ) ) { + // in Opera sometimes DOM element ids show up here, ignore them + if ( /^qunit-test-output/.test( key ) ) { + continue; + } + config.pollution.push( key ); + } + } + } +} + +function checkPollution() { + var newGlobals, + deletedGlobals, + old = config.pollution; + + saveGlobal(); + + newGlobals = diff( config.pollution, old ); + if ( newGlobals.length > 0 ) { + QUnit.pushFailure( "Introduced global variable(s): " + newGlobals.join( ", " ) ); + } + + deletedGlobals = diff( old, config.pollution ); + if ( deletedGlobals.length > 0 ) { + QUnit.pushFailure( "Deleted global variable(s): " + deletedGlobals.join( ", " ) ); + } +} + +// returns a new Array with the elements that are in a but not in b +function diff( a, b ) { + var i, j, + result = a.slice(); + + for ( i = 0; i < result.length; i++ ) { + for ( j = 0; j < b.length; j++ ) { + if ( result[ i ] === b[ j ] ) { + result.splice( i, 1 ); + i--; + break; + } + } + } + return result; +} + +function extend( a, b, undefOnly ) { + for ( var prop in b ) { + if ( hasOwn.call( b, prop ) ) { + + // Avoid "Member not found" error in IE8 caused by messing with window.constructor + if ( !( prop === "constructor" && a === window ) ) { + if ( b[ prop ] === undefined ) { + delete a[ prop ]; + } else if ( !( undefOnly && typeof a[ prop ] !== "undefined" ) ) { + a[ prop ] = b[ prop ]; + } + } + } + } + + return a; +} + +function runLoggingCallbacks( key, args ) { + var i, l, callbacks; + + callbacks = config.callbacks[ key ]; + for ( i = 0, l = callbacks.length; i < l; i++ ) { + callbacks[ i ]( args ); + } +} + +// DEPRECATED: This will be removed on 2.0.0+ +// This function verifies if the loggingCallbacks were modified by the user +// If so, it will restore it, assign the given callback and print a console warning +function verifyLoggingCallbacks() { + var loggingCallback, userCallback; + + for ( loggingCallback in loggingCallbacks ) { + if ( QUnit[ loggingCallback ] !== loggingCallbacks[ loggingCallback ] ) { + + userCallback = QUnit[ loggingCallback ]; + + // Restore the callback function + QUnit[ loggingCallback ] = loggingCallbacks[ loggingCallback ]; + + // Assign the deprecated given callback + QUnit[ loggingCallback ]( userCallback ); + + if ( window.console && window.console.warn ) { + window.console.warn( + "QUnit." + loggingCallback + " was replaced with a new value.\n" + + "Please, check out the documentation on how to apply logging callbacks.\n" + + "Reference: http://api.qunitjs.com/category/callbacks/" + ); + } + } + } +} + +// from jquery.js +function inArray( elem, array ) { + if ( array.indexOf ) { + return array.indexOf( elem ); + } + + for ( var i = 0, length = array.length; i < length; i++ ) { + if ( array[ i ] === elem ) { + return i; + } + } + + return -1; +} + +function Test( settings ) { + var i, l; + + ++Test.count; + + extend( this, settings ); + this.assertions = []; + this.semaphore = 0; + this.usedAsync = false; + this.module = config.currentModule; + this.stack = sourceFromStacktrace( 3 ); + + // Register unique strings + for ( i = 0, l = this.module.tests; i < l.length; i++ ) { + if ( this.module.tests[ i ].name === this.testName ) { + this.testName += " "; + } + } + + this.testId = generateHash( this.module.name, this.testName ); + + this.module.tests.push({ + name: this.testName, + testId: this.testId + }); + + if ( settings.skip ) { + + // Skipped tests will fully ignore any sent callback + this.callback = function() {}; + this.async = false; + this.expected = 0; + } else { + this.assert = new Assert( this ); + } +} + +Test.count = 0; + +Test.prototype = { + before: function() { + if ( + + // Emit moduleStart when we're switching from one module to another + this.module !== config.previousModule || + + // They could be equal (both undefined) but if the previousModule property doesn't + // yet exist it means this is the first test in a suite that isn't wrapped in a + // module, in which case we'll just emit a moduleStart event for 'undefined'. + // Without this, reporters can get testStart before moduleStart which is a problem. + !hasOwn.call( config, "previousModule" ) + ) { + if ( hasOwn.call( config, "previousModule" ) ) { + runLoggingCallbacks( "moduleDone", { + name: config.previousModule.name, + tests: config.previousModule.tests, + failed: config.moduleStats.bad, + passed: config.moduleStats.all - config.moduleStats.bad, + total: config.moduleStats.all, + runtime: now() - config.moduleStats.started + }); + } + config.previousModule = this.module; + config.moduleStats = { all: 0, bad: 0, started: now() }; + runLoggingCallbacks( "moduleStart", { + name: this.module.name, + tests: this.module.tests + }); + } + + config.current = this; + + this.testEnvironment = extend( {}, this.module.testEnvironment ); + delete this.testEnvironment.beforeEach; + delete this.testEnvironment.afterEach; + + this.started = now(); + runLoggingCallbacks( "testStart", { + name: this.testName, + module: this.module.name, + testId: this.testId + }); + + if ( !config.pollution ) { + saveGlobal(); + } + }, + + run: function() { + var promise; + + config.current = this; + + if ( this.async ) { + QUnit.stop(); + } + + this.callbackStarted = now(); + + if ( config.notrycatch ) { + promise = this.callback.call( this.testEnvironment, this.assert ); + this.resolvePromise( promise ); + return; + } + + try { + promise = this.callback.call( this.testEnvironment, this.assert ); + this.resolvePromise( promise ); + } catch ( e ) { + this.pushFailure( "Died on test #" + ( this.assertions.length + 1 ) + " " + + this.stack + ": " + ( e.message || e ), extractStacktrace( e, 0 ) ); + + // else next test will carry the responsibility + saveGlobal(); + + // Restart the tests if they're blocking + if ( config.blocking ) { + QUnit.start(); + } + } + }, + + after: function() { + checkPollution(); + }, + + queueHook: function( hook, hookName ) { + var promise, + test = this; + return function runHook() { + config.current = test; + if ( config.notrycatch ) { + promise = hook.call( test.testEnvironment, test.assert ); + test.resolvePromise( promise, hookName ); + return; + } + try { + promise = hook.call( test.testEnvironment, test.assert ); + test.resolvePromise( promise, hookName ); + } catch ( error ) { + test.pushFailure( hookName + " failed on " + test.testName + ": " + + ( error.message || error ), extractStacktrace( error, 0 ) ); + } + }; + }, + + // Currently only used for module level hooks, can be used to add global level ones + hooks: function( handler ) { + var hooks = []; + + // Hooks are ignored on skipped tests + if ( this.skip ) { + return hooks; + } + + if ( this.module.testEnvironment && + QUnit.objectType( this.module.testEnvironment[ handler ] ) === "function" ) { + hooks.push( this.queueHook( this.module.testEnvironment[ handler ], handler ) ); + } + + return hooks; + }, + + finish: function() { + config.current = this; + if ( config.requireExpects && this.expected === null ) { + this.pushFailure( "Expected number of assertions to be defined, but expect() was " + + "not called.", this.stack ); + } else if ( this.expected !== null && this.expected !== this.assertions.length ) { + this.pushFailure( "Expected " + this.expected + " assertions, but " + + this.assertions.length + " were run", this.stack ); + } else if ( this.expected === null && !this.assertions.length ) { + this.pushFailure( "Expected at least one assertion, but none were run - call " + + "expect(0) to accept zero assertions.", this.stack ); + } + + var i, + bad = 0; + + this.runtime = now() - this.started; + config.stats.all += this.assertions.length; + config.moduleStats.all += this.assertions.length; + + for ( i = 0; i < this.assertions.length; i++ ) { + if ( !this.assertions[ i ].result ) { + bad++; + config.stats.bad++; + config.moduleStats.bad++; + } + } + + runLoggingCallbacks( "testDone", { + name: this.testName, + module: this.module.name, + skipped: !!this.skip, + failed: bad, + passed: this.assertions.length - bad, + total: this.assertions.length, + runtime: this.runtime, + + // HTML Reporter use + assertions: this.assertions, + testId: this.testId, + + // DEPRECATED: this property will be removed in 2.0.0, use runtime instead + duration: this.runtime + }); + + // QUnit.reset() is deprecated and will be replaced for a new + // fixture reset function on QUnit 2.0/2.1. + // It's still called here for backwards compatibility handling + QUnit.reset(); + + config.current = undefined; + }, + + queue: function() { + var bad, + test = this; + + if ( !this.valid() ) { + return; + } + + function run() { + + // each of these can by async + synchronize([ + function() { + test.before(); + }, + + test.hooks( "beforeEach" ), + + function() { + test.run(); + }, + + test.hooks( "afterEach" ).reverse(), + + function() { + test.after(); + }, + function() { + test.finish(); + } + ]); + } + + // `bad` initialized at top of scope + // defer when previous test run passed, if storage is available + bad = QUnit.config.reorder && defined.sessionStorage && + +sessionStorage.getItem( "qunit-test-" + this.module.name + "-" + this.testName ); + + if ( bad ) { + run(); + } else { + synchronize( run, true ); + } + }, + + push: function( result, actual, expected, message ) { + var source, + details = { + module: this.module.name, + name: this.testName, + result: result, + message: message, + actual: actual, + expected: expected, + testId: this.testId, + runtime: now() - this.started + }; + + if ( !result ) { + source = sourceFromStacktrace(); + + if ( source ) { + details.source = source; + } + } + + runLoggingCallbacks( "log", details ); + + this.assertions.push({ + result: !!result, + message: message + }); + }, + + pushFailure: function( message, source, actual ) { + if ( !this instanceof Test ) { + throw new Error( "pushFailure() assertion outside test context, was " + + sourceFromStacktrace( 2 ) ); + } + + var details = { + module: this.module.name, + name: this.testName, + result: false, + message: message || "error", + actual: actual || null, + testId: this.testId, + runtime: now() - this.started + }; + + if ( source ) { + details.source = source; + } + + runLoggingCallbacks( "log", details ); + + this.assertions.push({ + result: false, + message: message + }); + }, + + resolvePromise: function( promise, phase ) { + var then, message, + test = this; + if ( promise != null ) { + then = promise.then; + if ( QUnit.objectType( then ) === "function" ) { + QUnit.stop(); + then.call( + promise, + QUnit.start, + function( error ) { + message = "Promise rejected " + + ( !phase ? "during" : phase.replace( /Each$/, "" ) ) + + " " + test.testName + ": " + ( error.message || error ); + test.pushFailure( message, extractStacktrace( error, 0 ) ); + + // else next test will carry the responsibility + saveGlobal(); + + // Unblock + QUnit.start(); + } + ); + } + } + }, + + valid: function() { + var include, + filter = config.filter && config.filter.toLowerCase(), + module = QUnit.urlParams.module && QUnit.urlParams.module.toLowerCase(), + fullName = ( this.module.name + ": " + this.testName ).toLowerCase(); + + // Internally-generated tests are always valid + if ( this.callback && this.callback.validTest ) { + return true; + } + + if ( config.testId.length > 0 && inArray( this.testId, config.testId ) < 0 ) { + return false; + } + + if ( module && ( !this.module.name || this.module.name.toLowerCase() !== module ) ) { + return false; + } + + if ( !filter ) { + return true; + } + + include = filter.charAt( 0 ) !== "!"; + if ( !include ) { + filter = filter.slice( 1 ); + } + + // If the filter matches, we need to honour include + if ( fullName.indexOf( filter ) !== -1 ) { + return include; + } + + // Otherwise, do the opposite + return !include; + } + +}; + +// Resets the test setup. Useful for tests that modify the DOM. +/* +DEPRECATED: Use multiple tests instead of resetting inside a test. +Use testStart or testDone for custom cleanup. +This method will throw an error in 2.0, and will be removed in 2.1 +*/ +QUnit.reset = function() { + + // Return on non-browser environments + // This is necessary to not break on node tests + if ( typeof window === "undefined" ) { + return; + } + + var fixture = defined.document && document.getElementById && + document.getElementById( "qunit-fixture" ); + + if ( fixture ) { + fixture.innerHTML = config.fixture; + } +}; + +QUnit.pushFailure = function() { + if ( !QUnit.config.current ) { + throw new Error( "pushFailure() assertion outside test context, in " + + sourceFromStacktrace( 2 ) ); + } + + // Gets current test obj + var currentTest = QUnit.config.current; + + return currentTest.pushFailure.apply( currentTest, arguments ); +}; + +// Based on Java's String.hashCode, a simple but not +// rigorously collision resistant hashing function +function generateHash( module, testName ) { + var hex, + i = 0, + hash = 0, + str = module + "\x1C" + testName, + len = str.length; + + for ( ; i < len; i++ ) { + hash = ( ( hash << 5 ) - hash ) + str.charCodeAt( i ); + hash |= 0; + } + + // Convert the possibly negative integer hash code into an 8 character hex string, which isn't + // strictly necessary but increases user understanding that the id is a SHA-like hash + hex = ( 0x100000000 + hash ).toString( 16 ); + if ( hex.length < 8 ) { + hex = "0000000" + hex; + } + + return hex.slice( -8 ); +} + +function Assert( testContext ) { + this.test = testContext; +} + +// Assert helpers +QUnit.assert = Assert.prototype = { + + // Specify the number of expected assertions to guarantee that failed test + // (no assertions are run at all) don't slip through. + expect: function( asserts ) { + if ( arguments.length === 1 ) { + this.test.expected = asserts; + } else { + return this.test.expected; + } + }, + + // Increment this Test's semaphore counter, then return a single-use function that + // decrements that counter a maximum of once. + async: function() { + var test = this.test, + popped = false; + + test.semaphore += 1; + test.usedAsync = true; + pauseProcessing(); + + return function done() { + if ( !popped ) { + test.semaphore -= 1; + popped = true; + resumeProcessing(); + } else { + test.pushFailure( "Called the callback returned from `assert.async` more than once", + sourceFromStacktrace( 2 ) ); + } + }; + }, + + // Exports test.push() to the user API + push: function( /* result, actual, expected, message */ ) { + var assert = this, + currentTest = ( assert instanceof Assert && assert.test ) || QUnit.config.current; + + // Backwards compatibility fix. + // Allows the direct use of global exported assertions and QUnit.assert.* + // Although, it's use is not recommended as it can leak assertions + // to other tests from async tests, because we only get a reference to the current test, + // not exactly the test where assertion were intended to be called. + if ( !currentTest ) { + throw new Error( "assertion outside test context, in " + sourceFromStacktrace( 2 ) ); + } + + if ( currentTest.usedAsync === true && currentTest.semaphore === 0 ) { + currentTest.pushFailure( "Assertion after the final `assert.async` was resolved", + sourceFromStacktrace( 2 ) ); + + // Allow this assertion to continue running anyway... + } + + if ( !( assert instanceof Assert ) ) { + assert = currentTest.assert; + } + return assert.test.push.apply( assert.test, arguments ); + }, + + /** + * Asserts rough true-ish result. + * @name ok + * @function + * @example ok( "asdfasdf".length > 5, "There must be at least 5 chars" ); + */ + ok: function( result, message ) { + message = message || ( result ? "okay" : "failed, expected argument to be truthy, was: " + + QUnit.dump.parse( result ) ); + this.push( !!result, result, true, message ); + }, + + /** + * Assert that the first two arguments are equal, with an optional message. + * Prints out both actual and expected values. + * @name equal + * @function + * @example equal( format( "{0} bytes.", 2), "2 bytes.", "replaces {0} with next argument" ); + */ + equal: function( actual, expected, message ) { + /*jshint eqeqeq:false */ + this.push( expected == actual, actual, expected, message ); + }, + + /** + * @name notEqual + * @function + */ + notEqual: function( actual, expected, message ) { + /*jshint eqeqeq:false */ + this.push( expected != actual, actual, expected, message ); + }, + + /** + * @name propEqual + * @function + */ + propEqual: function( actual, expected, message ) { + actual = objectValues( actual ); + expected = objectValues( expected ); + this.push( QUnit.equiv( actual, expected ), actual, expected, message ); + }, + + /** + * @name notPropEqual + * @function + */ + notPropEqual: function( actual, expected, message ) { + actual = objectValues( actual ); + expected = objectValues( expected ); + this.push( !QUnit.equiv( actual, expected ), actual, expected, message ); + }, + + /** + * @name deepEqual + * @function + */ + deepEqual: function( actual, expected, message ) { + this.push( QUnit.equiv( actual, expected ), actual, expected, message ); + }, + + /** + * @name notDeepEqual + * @function + */ + notDeepEqual: function( actual, expected, message ) { + this.push( !QUnit.equiv( actual, expected ), actual, expected, message ); + }, + + /** + * @name strictEqual + * @function + */ + strictEqual: function( actual, expected, message ) { + this.push( expected === actual, actual, expected, message ); + }, + + /** + * @name notStrictEqual + * @function + */ + notStrictEqual: function( actual, expected, message ) { + this.push( expected !== actual, actual, expected, message ); + }, + + "throws": function( block, expected, message ) { + var actual, expectedType, + expectedOutput = expected, + ok = false; + + // 'expected' is optional unless doing string comparison + if ( message == null && typeof expected === "string" ) { + message = expected; + expected = null; + } + + this.test.ignoreGlobalErrors = true; + try { + block.call( this.test.testEnvironment ); + } catch (e) { + actual = e; + } + this.test.ignoreGlobalErrors = false; + + if ( actual ) { + expectedType = QUnit.objectType( expected ); + + // we don't want to validate thrown error + if ( !expected ) { + ok = true; + expectedOutput = null; + + // expected is a regexp + } else if ( expectedType === "regexp" ) { + ok = expected.test( errorString( actual ) ); + + // expected is a string + } else if ( expectedType === "string" ) { + ok = expected === errorString( actual ); + + // expected is a constructor, maybe an Error constructor + } else if ( expectedType === "function" && actual instanceof expected ) { + ok = true; + + // expected is an Error object + } else if ( expectedType === "object" ) { + ok = actual instanceof expected.constructor && + actual.name === expected.name && + actual.message === expected.message; + + // expected is a validation function which returns true if validation passed + } else if ( expectedType === "function" && expected.call( {}, actual ) === true ) { + expectedOutput = null; + ok = true; + } + + this.push( ok, actual, expectedOutput, message ); + } else { + this.test.pushFailure( message, null, "No exception was thrown." ); + } + } +}; + +// Provide an alternative to assert.throws(), for enviroments that consider throws a reserved word +// Known to us are: Closure Compiler, Narwhal +(function() { + /*jshint sub:true */ + Assert.prototype.raises = Assert.prototype[ "throws" ]; +}()); + +// Test for equality any JavaScript type. +// Author: Philippe Rathé +QUnit.equiv = (function() { + + // Call the o related callback with the given arguments. + function bindCallbacks( o, callbacks, args ) { + var prop = QUnit.objectType( o ); + if ( prop ) { + if ( QUnit.objectType( callbacks[ prop ] ) === "function" ) { + return callbacks[ prop ].apply( callbacks, args ); + } else { + return callbacks[ prop ]; // or undefined + } + } + } + + // the real equiv function + var innerEquiv, + + // stack to decide between skip/abort functions + callers = [], + + // stack to avoiding loops from circular referencing + parents = [], + parentsB = [], + + getProto = Object.getPrototypeOf || function( obj ) { + /* jshint camelcase: false, proto: true */ + return obj.__proto__; + }, + callbacks = (function() { + + // for string, boolean, number and null + function useStrictEquality( b, a ) { + + /*jshint eqeqeq:false */ + if ( b instanceof a.constructor || a instanceof b.constructor ) { + + // to catch short annotation VS 'new' annotation of a + // declaration + // e.g. var i = 1; + // var j = new Number(1); + return a == b; + } else { + return a === b; + } + } + + return { + "string": useStrictEquality, + "boolean": useStrictEquality, + "number": useStrictEquality, + "null": useStrictEquality, + "undefined": useStrictEquality, + + "nan": function( b ) { + return isNaN( b ); + }, + + "date": function( b, a ) { + return QUnit.objectType( b ) === "date" && a.valueOf() === b.valueOf(); + }, + + "regexp": function( b, a ) { + return QUnit.objectType( b ) === "regexp" && + + // the regex itself + a.source === b.source && + + // and its modifiers + a.global === b.global && + + // (gmi) ... + a.ignoreCase === b.ignoreCase && + a.multiline === b.multiline && + a.sticky === b.sticky; + }, + + // - skip when the property is a method of an instance (OOP) + // - abort otherwise, + // initial === would have catch identical references anyway + "function": function() { + var caller = callers[ callers.length - 1 ]; + return caller !== Object && typeof caller !== "undefined"; + }, + + "array": function( b, a ) { + var i, j, len, loop, aCircular, bCircular; + + // b could be an object literal here + if ( QUnit.objectType( b ) !== "array" ) { + return false; + } + + len = a.length; + if ( len !== b.length ) { + // safe and faster + return false; + } + + // track reference to avoid circular references + parents.push( a ); + parentsB.push( b ); + for ( i = 0; i < len; i++ ) { + loop = false; + for ( j = 0; j < parents.length; j++ ) { + aCircular = parents[ j ] === a[ i ]; + bCircular = parentsB[ j ] === b[ i ]; + if ( aCircular || bCircular ) { + if ( a[ i ] === b[ i ] || aCircular && bCircular ) { + loop = true; + } else { + parents.pop(); + parentsB.pop(); + return false; + } + } + } + if ( !loop && !innerEquiv( a[ i ], b[ i ] ) ) { + parents.pop(); + parentsB.pop(); + return false; + } + } + parents.pop(); + parentsB.pop(); + return true; + }, + + "object": function( b, a ) { + + /*jshint forin:false */ + var i, j, loop, aCircular, bCircular, + // Default to true + eq = true, + aProperties = [], + bProperties = []; + + // comparing constructors is more strict than using + // instanceof + if ( a.constructor !== b.constructor ) { + + // Allow objects with no prototype to be equivalent to + // objects with Object as their constructor. + if ( !( ( getProto( a ) === null && getProto( b ) === Object.prototype ) || + ( getProto( b ) === null && getProto( a ) === Object.prototype ) ) ) { + return false; + } + } + + // stack constructor before traversing properties + callers.push( a.constructor ); + + // track reference to avoid circular references + parents.push( a ); + parentsB.push( b ); + + // be strict: don't ensure hasOwnProperty and go deep + for ( i in a ) { + loop = false; + for ( j = 0; j < parents.length; j++ ) { + aCircular = parents[ j ] === a[ i ]; + bCircular = parentsB[ j ] === b[ i ]; + if ( aCircular || bCircular ) { + if ( a[ i ] === b[ i ] || aCircular && bCircular ) { + loop = true; + } else { + eq = false; + break; + } + } + } + aProperties.push( i ); + if ( !loop && !innerEquiv( a[ i ], b[ i ] ) ) { + eq = false; + break; + } + } + + parents.pop(); + parentsB.pop(); + callers.pop(); // unstack, we are done + + for ( i in b ) { + bProperties.push( i ); // collect b's properties + } + + // Ensures identical properties name + return eq && innerEquiv( aProperties.sort(), bProperties.sort() ); + } + }; + }()); + + innerEquiv = function() { // can take multiple arguments + var args = [].slice.apply( arguments ); + if ( args.length < 2 ) { + return true; // end transition + } + + return ( (function( a, b ) { + if ( a === b ) { + return true; // catch the most you can + } else if ( a === null || b === null || typeof a === "undefined" || + typeof b === "undefined" || + QUnit.objectType( a ) !== QUnit.objectType( b ) ) { + + // don't lose time with error prone cases + return false; + } else { + return bindCallbacks( a, callbacks, [ b, a ] ); + } + + // apply transition with (1..n) arguments + }( args[ 0 ], args[ 1 ] ) ) && + innerEquiv.apply( this, args.splice( 1, args.length - 1 ) ) ); + }; + + return innerEquiv; +}()); + +// Based on jsDump by Ariel Flesler +// http://flesler.blogspot.com/2008/05/jsdump-pretty-dump-of-any-javascript.html +QUnit.dump = (function() { + function quote( str ) { + return "\"" + str.toString().replace( /"/g, "\\\"" ) + "\""; + } + function literal( o ) { + return o + ""; + } + function join( pre, arr, post ) { + var s = dump.separator(), + base = dump.indent(), + inner = dump.indent( 1 ); + if ( arr.join ) { + arr = arr.join( "," + s + inner ); + } + if ( !arr ) { + return pre + post; + } + return [ pre, inner + arr, base + post ].join( s ); + } + function array( arr, stack ) { + var i = arr.length, + ret = new Array( i ); + + if ( dump.maxDepth && dump.depth > dump.maxDepth ) { + return "[object Array]"; + } + + this.up(); + while ( i-- ) { + ret[ i ] = this.parse( arr[ i ], undefined, stack ); + } + this.down(); + return join( "[", ret, "]" ); + } + + var reName = /^function (\w+)/, + dump = { + + // objType is used mostly internally, you can fix a (custom) type in advance + parse: function( obj, objType, stack ) { + stack = stack || []; + var res, parser, parserType, + inStack = inArray( obj, stack ); + + if ( inStack !== -1 ) { + return "recursion(" + ( inStack - stack.length ) + ")"; + } + + objType = objType || this.typeOf( obj ); + parser = this.parsers[ objType ]; + parserType = typeof parser; + + if ( parserType === "function" ) { + stack.push( obj ); + res = parser.call( this, obj, stack ); + stack.pop(); + return res; + } + return ( parserType === "string" ) ? parser : this.parsers.error; + }, + typeOf: function( obj ) { + var type; + if ( obj === null ) { + type = "null"; + } else if ( typeof obj === "undefined" ) { + type = "undefined"; + } else if ( QUnit.is( "regexp", obj ) ) { + type = "regexp"; + } else if ( QUnit.is( "date", obj ) ) { + type = "date"; + } else if ( QUnit.is( "function", obj ) ) { + type = "function"; + } else if ( obj.setInterval !== undefined && + obj.document !== undefined && + obj.nodeType === undefined ) { + type = "window"; + } else if ( obj.nodeType === 9 ) { + type = "document"; + } else if ( obj.nodeType ) { + type = "node"; + } else if ( + + // native arrays + toString.call( obj ) === "[object Array]" || + + // NodeList objects + ( typeof obj.length === "number" && obj.item !== undefined && + ( obj.length ? obj.item( 0 ) === obj[ 0 ] : ( obj.item( 0 ) === null && + obj[ 0 ] === undefined ) ) ) + ) { + type = "array"; + } else if ( obj.constructor === Error.prototype.constructor ) { + type = "error"; + } else { + type = typeof obj; + } + return type; + }, + separator: function() { + return this.multiline ? this.HTML ? "
                          " : "\n" : this.HTML ? " " : " "; + }, + // extra can be a number, shortcut for increasing-calling-decreasing + indent: function( extra ) { + if ( !this.multiline ) { + return ""; + } + var chr = this.indentChar; + if ( this.HTML ) { + chr = chr.replace( /\t/g, " " ).replace( / /g, " " ); + } + return new Array( this.depth + ( extra || 0 ) ).join( chr ); + }, + up: function( a ) { + this.depth += a || 1; + }, + down: function( a ) { + this.depth -= a || 1; + }, + setParser: function( name, parser ) { + this.parsers[ name ] = parser; + }, + // The next 3 are exposed so you can use them + quote: quote, + literal: literal, + join: join, + // + depth: 1, + maxDepth: 5, + + // This is the list of parsers, to modify them, use dump.setParser + parsers: { + window: "[Window]", + document: "[Document]", + error: function( error ) { + return "Error(\"" + error.message + "\")"; + }, + unknown: "[Unknown]", + "null": "null", + "undefined": "undefined", + "function": function( fn ) { + var ret = "function", + + // functions never have name in IE + name = "name" in fn ? fn.name : ( reName.exec( fn ) || [] )[ 1 ]; + + if ( name ) { + ret += " " + name; + } + ret += "( "; + + ret = [ ret, dump.parse( fn, "functionArgs" ), "){" ].join( "" ); + return join( ret, dump.parse( fn, "functionCode" ), "}" ); + }, + array: array, + nodelist: array, + "arguments": array, + object: function( map, stack ) { + var keys, key, val, i, nonEnumerableProperties, + ret = []; + + if ( dump.maxDepth && dump.depth > dump.maxDepth ) { + return "[object Object]"; + } + + dump.up(); + keys = []; + for ( key in map ) { + keys.push( key ); + } + + // Some properties are not always enumerable on Error objects. + nonEnumerableProperties = [ "message", "name" ]; + for ( i in nonEnumerableProperties ) { + key = nonEnumerableProperties[ i ]; + if ( key in map && !( key in keys ) ) { + keys.push( key ); + } + } + keys.sort(); + for ( i = 0; i < keys.length; i++ ) { + key = keys[ i ]; + val = map[ key ]; + ret.push( dump.parse( key, "key" ) + ": " + + dump.parse( val, undefined, stack ) ); + } + dump.down(); + return join( "{", ret, "}" ); + }, + node: function( node ) { + var len, i, val, + open = dump.HTML ? "<" : "<", + close = dump.HTML ? ">" : ">", + tag = node.nodeName.toLowerCase(), + ret = open + tag, + attrs = node.attributes; + + if ( attrs ) { + for ( i = 0, len = attrs.length; i < len; i++ ) { + val = attrs[ i ].nodeValue; + + // IE6 includes all attributes in .attributes, even ones not explicitly + // set. Those have values like undefined, null, 0, false, "" or + // "inherit". + if ( val && val !== "inherit" ) { + ret += " " + attrs[ i ].nodeName + "=" + + dump.parse( val, "attribute" ); + } + } + } + ret += close; + + // Show content of TextNode or CDATASection + if ( node.nodeType === 3 || node.nodeType === 4 ) { + ret += node.nodeValue; + } + + return ret + open + "/" + tag + close; + }, + + // function calls it internally, it's the arguments part of the function + functionArgs: function( fn ) { + var args, + l = fn.length; + + if ( !l ) { + return ""; + } + + args = new Array( l ); + while ( l-- ) { + + // 97 is 'a' + args[ l ] = String.fromCharCode( 97 + l ); + } + return " " + args.join( ", " ) + " "; + }, + // object calls it internally, the key part of an item in a map + key: quote, + // function calls it internally, it's the content of the function + functionCode: "[code]", + // node calls it internally, it's an html attribute value + attribute: quote, + string: quote, + date: quote, + regexp: literal, + number: literal, + "boolean": literal + }, + // if true, entities are escaped ( <, >, \t, space and \n ) + HTML: false, + // indentation unit + indentChar: " ", + // if true, items in a collection, are separated by a \n, else just a space. + multiline: true + }; + + return dump; +}()); + +// back compat +QUnit.jsDump = QUnit.dump; + +// For browser, export only select globals +if ( typeof window !== "undefined" ) { + + // Deprecated + // Extend assert methods to QUnit and Global scope through Backwards compatibility + (function() { + var i, + assertions = Assert.prototype; + + function applyCurrent( current ) { + return function() { + var assert = new Assert( QUnit.config.current ); + current.apply( assert, arguments ); + }; + } + + for ( i in assertions ) { + QUnit[ i ] = applyCurrent( assertions[ i ] ); + } + })(); + + (function() { + var i, l, + keys = [ + "test", + "module", + "expect", + "asyncTest", + "start", + "stop", + "ok", + "equal", + "notEqual", + "propEqual", + "notPropEqual", + "deepEqual", + "notDeepEqual", + "strictEqual", + "notStrictEqual", + "throws" + ]; + + for ( i = 0, l = keys.length; i < l; i++ ) { + window[ keys[ i ] ] = QUnit[ keys[ i ] ]; + } + })(); + + window.QUnit = QUnit; +} + +// For nodejs +if ( typeof module !== "undefined" && module.exports ) { + module.exports = QUnit; +} + +// For CommonJS with exports, but without module.exports, like Rhino +if ( typeof exports !== "undefined" ) { + exports.QUnit = QUnit; +} + +// Get a reference to the global object, like window in browsers +}( (function() { + return this; +})() )); + +/*istanbul ignore next */ +// jscs:disable maximumLineLength +/* + * Javascript Diff Algorithm + * By John Resig (http://ejohn.org/) + * Modified by Chu Alan "sprite" + * + * Released under the MIT license. + * + * More Info: + * http://ejohn.org/projects/javascript-diff-algorithm/ + * + * Usage: QUnit.diff(expected, actual) + * + * QUnit.diff( "the quick brown fox jumped over", "the quick fox jumps over" ) == "the quick brown fox jumped jumps over" + */ +QUnit.diff = (function() { + var hasOwn = Object.prototype.hasOwnProperty; + + /*jshint eqeqeq:false, eqnull:true */ + function diff( o, n ) { + var i, + ns = {}, + os = {}; + + for ( i = 0; i < n.length; i++ ) { + if ( !hasOwn.call( ns, n[ i ] ) ) { + ns[ n[ i ] ] = { + rows: [], + o: null + }; + } + ns[ n[ i ] ].rows.push( i ); + } + + for ( i = 0; i < o.length; i++ ) { + if ( !hasOwn.call( os, o[ i ] ) ) { + os[ o[ i ] ] = { + rows: [], + n: null + }; + } + os[ o[ i ] ].rows.push( i ); + } + + for ( i in ns ) { + if ( hasOwn.call( ns, i ) ) { + if ( ns[ i ].rows.length === 1 && hasOwn.call( os, i ) && os[ i ].rows.length === 1 ) { + n[ ns[ i ].rows[ 0 ] ] = { + text: n[ ns[ i ].rows[ 0 ] ], + row: os[ i ].rows[ 0 ] + }; + o[ os[ i ].rows[ 0 ] ] = { + text: o[ os[ i ].rows[ 0 ] ], + row: ns[ i ].rows[ 0 ] + }; + } + } + } + + for ( i = 0; i < n.length - 1; i++ ) { + if ( n[ i ].text != null && n[ i + 1 ].text == null && n[ i ].row + 1 < o.length && o[ n[ i ].row + 1 ].text == null && + n[ i + 1 ] == o[ n[ i ].row + 1 ] ) { + + n[ i + 1 ] = { + text: n[ i + 1 ], + row: n[ i ].row + 1 + }; + o[ n[ i ].row + 1 ] = { + text: o[ n[ i ].row + 1 ], + row: i + 1 + }; + } + } + + for ( i = n.length - 1; i > 0; i-- ) { + if ( n[ i ].text != null && n[ i - 1 ].text == null && n[ i ].row > 0 && o[ n[ i ].row - 1 ].text == null && + n[ i - 1 ] == o[ n[ i ].row - 1 ] ) { + + n[ i - 1 ] = { + text: n[ i - 1 ], + row: n[ i ].row - 1 + }; + o[ n[ i ].row - 1 ] = { + text: o[ n[ i ].row - 1 ], + row: i - 1 + }; + } + } + + return { + o: o, + n: n + }; + } + + return function( o, n ) { + o = o.replace( /\s+$/, "" ); + n = n.replace( /\s+$/, "" ); + + var i, pre, + str = "", + out = diff( o === "" ? [] : o.split( /\s+/ ), n === "" ? [] : n.split( /\s+/ ) ), + oSpace = o.match( /\s+/g ), + nSpace = n.match( /\s+/g ); + + if ( oSpace == null ) { + oSpace = [ " " ]; + } else { + oSpace.push( " " ); + } + + if ( nSpace == null ) { + nSpace = [ " " ]; + } else { + nSpace.push( " " ); + } + + if ( out.n.length === 0 ) { + for ( i = 0; i < out.o.length; i++ ) { + str += "" + out.o[ i ] + oSpace[ i ] + ""; + } + } else { + if ( out.n[ 0 ].text == null ) { + for ( n = 0; n < out.o.length && out.o[ n ].text == null; n++ ) { + str += "" + out.o[ n ] + oSpace[ n ] + ""; + } + } + + for ( i = 0; i < out.n.length; i++ ) { + if ( out.n[ i ].text == null ) { + str += "" + out.n[ i ] + nSpace[ i ] + ""; + } else { + + // `pre` initialized at top of scope + pre = ""; + + for ( n = out.n[ i ].row + 1; n < out.o.length && out.o[ n ].text == null; n++ ) { + pre += "" + out.o[ n ] + oSpace[ n ] + ""; + } + str += " " + out.n[ i ].text + nSpace[ i ] + pre; + } + } + } + + return str; + }; +}()); +// jscs:enable + +(function() { + +// Deprecated QUnit.init - Ref #530 +// Re-initialize the configuration options +QUnit.init = function() { + var tests, banner, result, qunit, + config = QUnit.config; + + config.stats = { all: 0, bad: 0 }; + config.moduleStats = { all: 0, bad: 0 }; + config.started = 0; + config.updateRate = 1000; + config.blocking = false; + config.autostart = true; + config.autorun = false; + config.filter = ""; + config.queue = []; + + // Return on non-browser environments + // This is necessary to not break on node tests + if ( typeof window === "undefined" ) { + return; + } + + qunit = id( "qunit" ); + if ( qunit ) { + qunit.innerHTML = + "

                          " + escapeText( document.title ) + "

                          " + + "

                          " + + "
                          " + + "

                          " + + "
                            "; + } + + tests = id( "qunit-tests" ); + banner = id( "qunit-banner" ); + result = id( "qunit-testresult" ); + + if ( tests ) { + tests.innerHTML = ""; + } + + if ( banner ) { + banner.className = ""; + } + + if ( result ) { + result.parentNode.removeChild( result ); + } + + if ( tests ) { + result = document.createElement( "p" ); + result.id = "qunit-testresult"; + result.className = "result"; + tests.parentNode.insertBefore( result, tests ); + result.innerHTML = "Running...
                             "; + } +}; + +// Don't load the HTML Reporter on non-Browser environments +if ( typeof window === "undefined" ) { + return; +} + +var config = QUnit.config, + hasOwn = Object.prototype.hasOwnProperty, + defined = { + document: window.document !== undefined, + sessionStorage: (function() { + var x = "qunit-test-string"; + try { + sessionStorage.setItem( x, x ); + sessionStorage.removeItem( x ); + return true; + } catch ( e ) { + return false; + } + }()) + }, + modulesList = []; + +/** +* Escape text for attribute or text content. +*/ +function escapeText( s ) { + if ( !s ) { + return ""; + } + s = s + ""; + + // Both single quotes and double quotes (for attributes) + return s.replace( /['"<>&]/g, function( s ) { + switch ( s ) { + case "'": + return "'"; + case "\"": + return """; + case "<": + return "<"; + case ">": + return ">"; + case "&": + return "&"; + } + }); +} + +/** + * @param {HTMLElement} elem + * @param {string} type + * @param {Function} fn + */ +function addEvent( elem, type, fn ) { + if ( elem.addEventListener ) { + + // Standards-based browsers + elem.addEventListener( type, fn, false ); + } else if ( elem.attachEvent ) { + + // support: IE <9 + elem.attachEvent( "on" + type, fn ); + } +} + +/** + * @param {Array|NodeList} elems + * @param {string} type + * @param {Function} fn + */ +function addEvents( elems, type, fn ) { + var i = elems.length; + while ( i-- ) { + addEvent( elems[ i ], type, fn ); + } +} + +function hasClass( elem, name ) { + return ( " " + elem.className + " " ).indexOf( " " + name + " " ) >= 0; +} + +function addClass( elem, name ) { + if ( !hasClass( elem, name ) ) { + elem.className += ( elem.className ? " " : "" ) + name; + } +} + +function toggleClass( elem, name ) { + if ( hasClass( elem, name ) ) { + removeClass( elem, name ); + } else { + addClass( elem, name ); + } +} + +function removeClass( elem, name ) { + var set = " " + elem.className + " "; + + // Class name may appear multiple times + while ( set.indexOf( " " + name + " " ) >= 0 ) { + set = set.replace( " " + name + " ", " " ); + } + + // trim for prettiness + elem.className = typeof set.trim === "function" ? set.trim() : set.replace( /^\s+|\s+$/g, "" ); +} + +function id( name ) { + return defined.document && document.getElementById && document.getElementById( name ); +} + +function getUrlConfigHtml() { + var i, j, val, + escaped, escapedTooltip, + selection = false, + len = config.urlConfig.length, + urlConfigHtml = ""; + + for ( i = 0; i < len; i++ ) { + val = config.urlConfig[ i ]; + if ( typeof val === "string" ) { + val = { + id: val, + label: val + }; + } + + escaped = escapeText( val.id ); + escapedTooltip = escapeText( val.tooltip ); + + config[ val.id ] = QUnit.urlParams[ val.id ]; + if ( !val.value || typeof val.value === "string" ) { + urlConfigHtml += ""; + } else { + urlConfigHtml += ""; + } + } + + return urlConfigHtml; +} + +// Handle "click" events on toolbar checkboxes and "change" for select menus. +// Updates the URL with the new state of `config.urlConfig` values. +function toolbarChanged() { + var updatedUrl, value, + field = this, + params = {}; + + // Detect if field is a select menu or a checkbox + if ( "selectedIndex" in field ) { + value = field.options[ field.selectedIndex ].value || undefined; + } else { + value = field.checked ? ( field.defaultValue || true ) : undefined; + } + + params[ field.name ] = value; + updatedUrl = QUnit.url( params ); + + if ( "hidepassed" === field.name && "replaceState" in window.history ) { + config[ field.name ] = value || false; + if ( value ) { + addClass( id( "qunit-tests" ), "hidepass" ); + } else { + removeClass( id( "qunit-tests" ), "hidepass" ); + } + + // It is not necessary to refresh the whole page + window.history.replaceState( null, "", updatedUrl ); + } else { + window.location = updatedUrl; + } +} + +function toolbarUrlConfigContainer() { + var urlConfigContainer = document.createElement( "span" ); + + urlConfigContainer.innerHTML = getUrlConfigHtml(); + + // For oldIE support: + // * Add handlers to the individual elements instead of the container + // * Use "click" instead of "change" for checkboxes + addEvents( urlConfigContainer.getElementsByTagName( "input" ), "click", toolbarChanged ); + addEvents( urlConfigContainer.getElementsByTagName( "select" ), "change", toolbarChanged ); + + return urlConfigContainer; +} + +function toolbarModuleFilterHtml() { + var i, + moduleFilterHtml = ""; + + if ( !modulesList.length ) { + return false; + } + + modulesList.sort(function( a, b ) { + return a.localeCompare( b ); + }); + + moduleFilterHtml += "" + + ""; + + return moduleFilterHtml; +} + +function toolbarModuleFilter() { + var toolbar = id( "qunit-testrunner-toolbar" ), + moduleFilter = document.createElement( "span" ), + moduleFilterHtml = toolbarModuleFilterHtml(); + + if ( !moduleFilterHtml ) { + return false; + } + + moduleFilter.setAttribute( "id", "qunit-modulefilter-container" ); + moduleFilter.innerHTML = moduleFilterHtml; + + addEvent( moduleFilter.lastChild, "change", function() { + var selectBox = moduleFilter.getElementsByTagName( "select" )[ 0 ], + selection = decodeURIComponent( selectBox.options[ selectBox.selectedIndex ].value ); + + window.location = QUnit.url({ + module: ( selection === "" ) ? undefined : selection, + + // Remove any existing filters + filter: undefined, + testId: undefined + }); + }); + + toolbar.appendChild( moduleFilter ); +} + +function appendToolbar() { + var toolbar = id( "qunit-testrunner-toolbar" ); + + if ( toolbar ) { + toolbar.appendChild( toolbarUrlConfigContainer() ); + } +} + +function appendBanner() { + var banner = id( "qunit-banner" ); + + if ( banner ) { + banner.className = ""; + banner.innerHTML = "" + banner.innerHTML + " "; + } +} + +function appendTestResults() { + var tests = id( "qunit-tests" ), + result = id( "qunit-testresult" ); + + if ( result ) { + result.parentNode.removeChild( result ); + } + + if ( tests ) { + tests.innerHTML = ""; + result = document.createElement( "p" ); + result.id = "qunit-testresult"; + result.className = "result"; + tests.parentNode.insertBefore( result, tests ); + result.innerHTML = "Running...
                             "; + } +} + +function storeFixture() { + var fixture = id( "qunit-fixture" ); + if ( fixture ) { + config.fixture = fixture.innerHTML; + } +} + +function appendUserAgent() { + var userAgent = id( "qunit-userAgent" ); + if ( userAgent ) { + userAgent.innerHTML = navigator.userAgent; + } +} + +function appendTestsList( modules ) { + var i, l, x, z, test, moduleObj; + + for ( i = 0, l = modules.length; i < l; i++ ) { + moduleObj = modules[ i ]; + + if ( moduleObj.name ) { + modulesList.push( moduleObj.name ); + } + + for ( x = 0, z = moduleObj.tests.length; x < z; x++ ) { + test = moduleObj.tests[ x ]; + + appendTest( test.name, test.testId, moduleObj.name ); + } + } +} + +function appendTest( name, testId, moduleName ) { + var title, rerunTrigger, testBlock, assertList, + tests = id( "qunit-tests" ); + + if ( !tests ) { + return; + } + + title = document.createElement( "strong" ); + title.innerHTML = getNameHtml( name, moduleName ); + + rerunTrigger = document.createElement( "a" ); + rerunTrigger.innerHTML = "Rerun"; + rerunTrigger.href = QUnit.url({ testId: testId }); + + testBlock = document.createElement( "li" ); + testBlock.appendChild( title ); + testBlock.appendChild( rerunTrigger ); + testBlock.id = "qunit-test-output-" + testId; + + assertList = document.createElement( "ol" ); + assertList.className = "qunit-assert-list"; + + testBlock.appendChild( assertList ); + + tests.appendChild( testBlock ); +} + +// HTML Reporter initialization and load +QUnit.begin(function( details ) { + var qunit = id( "qunit" ); + + // Fixture is the only one necessary to run without the #qunit element + storeFixture(); + + if ( !qunit ) { + return; + } + + qunit.innerHTML = + "

                            " + escapeText( document.title ) + "

                            " + + "

                            " + + "
                            " + + "

                            " + + "
                              "; + + appendBanner(); + appendTestResults(); + appendUserAgent(); + appendToolbar(); + appendTestsList( details.modules ); + toolbarModuleFilter(); + + if ( config.hidepassed ) { + addClass( qunit.lastChild, "hidepass" ); + } +}); + +QUnit.done(function( details ) { + var i, key, + banner = id( "qunit-banner" ), + tests = id( "qunit-tests" ), + html = [ + "Tests completed in ", + details.runtime, + " milliseconds.
                              ", + "", + details.passed, + " assertions of ", + details.total, + " passed, ", + details.failed, + " failed." + ].join( "" ); + + if ( banner ) { + banner.className = details.failed ? "qunit-fail" : "qunit-pass"; + } + + if ( tests ) { + id( "qunit-testresult" ).innerHTML = html; + } + + if ( config.altertitle && defined.document && document.title ) { + + // show ✖ for good, ✔ for bad suite result in title + // use escape sequences in case file gets loaded with non-utf-8-charset + document.title = [ + ( details.failed ? "\u2716" : "\u2714" ), + document.title.replace( /^[\u2714\u2716] /i, "" ) + ].join( " " ); + } + + // clear own sessionStorage items if all tests passed + if ( config.reorder && defined.sessionStorage && details.failed === 0 ) { + for ( i = 0; i < sessionStorage.length; i++ ) { + key = sessionStorage.key( i++ ); + if ( key.indexOf( "qunit-test-" ) === 0 ) { + sessionStorage.removeItem( key ); + } + } + } + + // scroll back to top to show results + if ( config.scrolltop && window.scrollTo ) { + window.scrollTo( 0, 0 ); + } +}); + +function getNameHtml( name, module ) { + var nameHtml = ""; + + if ( module ) { + nameHtml = "" + escapeText( module ) + ": "; + } + + nameHtml += "" + escapeText( name ) + ""; + + return nameHtml; +} + +QUnit.testStart(function( details ) { + var running, testBlock; + + testBlock = id( "qunit-test-output-" + details.testId ); + if ( testBlock ) { + testBlock.className = "running"; + } else { + + // Report later registered tests + appendTest( details.name, details.testId, details.module ); + } + + running = id( "qunit-testresult" ); + if ( running ) { + running.innerHTML = "Running:
                              " + getNameHtml( details.name, details.module ); + } + +}); + +QUnit.log(function( details ) { + var assertList, assertLi, + message, expected, actual, + testItem = id( "qunit-test-output-" + details.testId ); + + if ( !testItem ) { + return; + } + + message = escapeText( details.message ) || ( details.result ? "okay" : "failed" ); + message = "" + message + ""; + message += "@ " + details.runtime + " ms"; + + // pushFailure doesn't provide details.expected + // when it calls, it's implicit to also not show expected and diff stuff + // Also, we need to check details.expected existence, as it can exist and be undefined + if ( !details.result && hasOwn.call( details, "expected" ) ) { + expected = escapeText( QUnit.dump.parse( details.expected ) ); + actual = escapeText( QUnit.dump.parse( details.actual ) ); + message += "
                          1. "; + + if ( actual !== expected ) { + message += "" + + ""; + } + + if ( details.source ) { + message += ""; + } + + message += "
                            Expected:
                            " +
                            +			expected +
                            +			"
                            Result:
                            " +
                            +				actual + "
                            Diff:
                            " +
                            +				QUnit.diff( expected, actual ) + "
                            Source:
                            " +
                            +				escapeText( details.source ) + "
                            "; + + // this occours when pushFailure is set and we have an extracted stack trace + } else if ( !details.result && details.source ) { + message += "" + + "" + + "
                            Source:
                            " +
                            +			escapeText( details.source ) + "
                            "; + } + + assertList = testItem.getElementsByTagName( "ol" )[ 0 ]; + + assertLi = document.createElement( "li" ); + assertLi.className = details.result ? "pass" : "fail"; + assertLi.innerHTML = message; + assertList.appendChild( assertLi ); +}); + +QUnit.testDone(function( details ) { + var testTitle, time, testItem, assertList, + good, bad, testCounts, skipped, + tests = id( "qunit-tests" ); + + if ( !tests ) { + return; + } + + testItem = id( "qunit-test-output-" + details.testId ); + + assertList = testItem.getElementsByTagName( "ol" )[ 0 ]; + + good = details.passed; + bad = details.failed; + + // store result when possible + if ( config.reorder && defined.sessionStorage ) { + if ( bad ) { + sessionStorage.setItem( "qunit-test-" + details.module + "-" + details.name, bad ); + } else { + sessionStorage.removeItem( "qunit-test-" + details.module + "-" + details.name ); + } + } + + if ( bad === 0 ) { + addClass( assertList, "qunit-collapsed" ); + } + + // testItem.firstChild is the test name + testTitle = testItem.firstChild; + + testCounts = bad ? + "" + bad + ", " + "" + good + ", " : + ""; + + testTitle.innerHTML += " (" + testCounts + + details.assertions.length + ")"; + + if ( details.skipped ) { + addClass( testItem, "skipped" ); + skipped = document.createElement( "em" ); + skipped.className = "qunit-skipped-label"; + skipped.innerHTML = "skipped"; + testItem.insertBefore( skipped, testTitle ); + } else { + addEvent( testTitle, "click", function() { + toggleClass( assertList, "qunit-collapsed" ); + }); + + testItem.className = bad ? "fail" : "pass"; + + time = document.createElement( "span" ); + time.className = "runtime"; + time.innerHTML = details.runtime + " ms"; + testItem.insertBefore( time, assertList ); + } +}); + +if ( !defined.document || document.readyState === "complete" ) { + config.pageLoaded = true; + config.autorun = true; +} + +if ( defined.document ) { + addEvent( window, "load", QUnit.load ); +} + +})(); diff --git a/system/admin/editor.md/upload.php b/system/admin/editor.md/upload.php new file mode 100644 index 00000000..a3d99a10 --- /dev/null +++ b/system/admin/editor.md/upload.php @@ -0,0 +1,62 @@ + $success, + 'message' => $message, + 'url' => site_url() . 'content/images/' . $timestamp . '-' . $name + )); +} else { + $login = site_url() . 'login'; + header("location: $login"); +} \ No newline at end of file diff --git a/system/admin/editor/css/editor.css b/system/admin/editor/css/editor.css deleted file mode 100644 index cb750731..00000000 --- a/system/admin/editor/css/editor.css +++ /dev/null @@ -1,154 +0,0 @@ -body { - font-family: Georgia, sans-serif; -} - -blockquote { - border-left: 2px dotted #888; - padding-left: 5px; - background: #F6F7F9; -} - -.wmd-panel { - float: left; -} - -.wmd-button-bar { - background-color: #F6F7F9; - padding: 5px 0; - margin-bottom: 10px; - width: 100%; - border: 1px solid #CFDAE5; - display: block; - float: left; -} - -.wmd-button-bar:hover { - border: 1px solid #CCCCCC; -} - -.wmd-input { - min-height: 300px; -} - -.wmd-preview { - background-color: #E4EBF1; - float: left; - padding: 2%; -} - -.wmd-preview table { - margin: 20px 0; -} - -.wmd-preview td, .wmd-preview th { - border-right: 1px solid #ccc; - padding: 8px 12px; -} - -.wmd-preview td:last-child, .wmd-preview th:last-child { - border-right: none; -} - -.wmd-preview td { - border-top: 1px solid #ccc; -} - -.wmd-button-row { - position: relative; - margin: 0px; - padding: 0px; -} - -.wmd-spacer { - width: 1px; - height: 20px; - background-color: Silver; - list-style: none; - float: left; - margin: 5px; -} - -.wmd-button { - list-style: none; - cursor: pointer; - float: left; -} - -.wmd-button > span { - background-repeat: no-repeat; - background-position: 0px 0px; - - display: inline-block; -} - -.wmd-spacer1 { - left: 50px; -} - -.wmd-spacer2 { - left: 175px; -} - -.wmd-spacer3 { - left: 300px; -} - -.wmd-prompt-background { - background-color: Black; -} - -.wmd-prompt-dialog { - border: 1px solid #999999; - background-color: #F5F5F5; -} - -.wmd-prompt-dialog > div { - font-size: 0.8em; - font-family: arial, helvetica, sans-serif; -} - -.wmd-prompt-dialog > form > input[type="text"] { - border: 1px solid #999999; - color: black; -} - -.wmd-prompt-dialog > form > input[type="button"] { - border: 1px solid #888888; - font-family: trebuchet MS, helvetica, sans-serif; - font-size: 0.8em; - font-weight: bold; -} - -pre { - margin: 1em 0; - overflow: auto; - background: #F1F1FF; -} - -pre code { - color: #333333; - display: block; - font-family: Consolas, Menlo, Monaco, Lucida Console, Liberation Mono, DejaVu Sans Mono, Bitstream Vera Sans Mono, Courier New, monospace, serif; - font-size: 14px; - padding: 5px 10px; -} - -@media all and (max-width: 980px) { - - .nav { - width: 100%; - padding: 2%; - } - - .wmd-panel, .wmd-preview { - width: 96%; - float: left; - padding: 2%; - } - - pre { - white-space: pre-wrap; - word-wrap: break-word; - } - -} \ No newline at end of file diff --git a/system/admin/editor/js/Markdown.Converter.js b/system/admin/editor/js/Markdown.Converter.js deleted file mode 100644 index 60e0bb54..00000000 --- a/system/admin/editor/js/Markdown.Converter.js +++ /dev/null @@ -1,1428 +0,0 @@ -var Markdown; - -if (typeof exports === "object" && typeof require === "function") // we're in a CommonJS (e.g. Node.js) module - Markdown = exports; -else - Markdown = {}; - -// The following text is included for historical reasons, but should -// be taken with a pinch of salt; it's not all true anymore. - -// -// Wherever possible, Showdown is a straight, line-by-line port -// of the Perl version of Markdown. -// -// This is not a normal parser design; it's basically just a -// series of string substitutions. It's hard to read and -// maintain this way, but keeping Showdown close to the original -// design makes it easier to port new features. -// -// More importantly, Showdown behaves like markdown.pl in most -// edge cases. So web applications can do client-side preview -// in Javascript, and then build identical HTML on the server. -// -// This port needs the new RegExp functionality of ECMA 262, -// 3rd Edition (i.e. Javascript 1.5). Most modern web browsers -// should do fine. Even with the new regular expression features, -// We do a lot of work to emulate Perl's regex functionality. -// The tricky changes in this file mostly have the "attacklab:" -// label. Major or self-explanatory changes don't. -// -// Smart diff tools like Araxis Merge will be able to match up -// this file with markdown.pl in a useful way. A little tweaking -// helps: in a copy of markdown.pl, replace "#" with "//" and -// replace "$text" with "text". Be sure to ignore whitespace -// and line endings. -// - - -// -// Usage: -// -// var text = "Markdown *rocks*."; -// -// var converter = new Markdown.Converter(); -// var html = converter.makeHtml(text); -// -// alert(html); -// -// Note: move the sample code to the bottom of this -// file before uncommenting it. -// - -(function () { - - function identity(x) { - return x; - } - - function returnFalse(x) { - return false; - } - - function HookCollection() { - } - - HookCollection.prototype = { - - chain: function (hookname, func) { - var original = this[hookname]; - if (!original) - throw new Error("unknown hook " + hookname); - - if (original === identity) - this[hookname] = func; - else - this[hookname] = function (text) { - var args = Array.prototype.slice.call(arguments, 0); - args[0] = original.apply(null, args); - return func.apply(null, args); - }; - }, - set: function (hookname, func) { - if (!this[hookname]) - throw new Error("unknown hook " + hookname); - this[hookname] = func; - }, - addNoop: function (hookname) { - this[hookname] = identity; - }, - addFalse: function (hookname) { - this[hookname] = returnFalse; - } - }; - - Markdown.HookCollection = HookCollection; - - // g_urls and g_titles allow arbitrary user-entered strings as keys. This - // caused an exception (and hence stopped the rendering) when the user entered - // e.g. [push] or [__proto__]. Adding a prefix to the actual key prevents this - // (since no builtin property starts with "s_"). See - // http://meta.stackoverflow.com/questions/64655/strange-wmd-bug - // (granted, switching from Array() to Object() alone would have left only __proto__ - // to be a problem) - function SaveHash() { - } - - SaveHash.prototype = { - set: function (key, value) { - this["s_" + key] = value; - }, - get: function (key) { - return this["s_" + key]; - } - }; - - Markdown.Converter = function () { - var pluginHooks = this.hooks = new HookCollection(); - - // given a URL that was encountered by itself (without markup), should return the link text that's to be given to this link - pluginHooks.addNoop("plainLinkText"); - - // called with the orignal text as given to makeHtml. The result of this plugin hook is the actual markdown source that will be cooked - pluginHooks.addNoop("preConversion"); - - // called with the text once all normalizations have been completed (tabs to spaces, line endings, etc.), but before any conversions have - pluginHooks.addNoop("postNormalization"); - - // Called with the text before / after creating block elements like code blocks and lists. Note that this is called recursively - // with inner content, e.g. it's called with the full text, and then only with the content of a blockquote. The inner - // call will receive outdented text. - pluginHooks.addNoop("preBlockGamut"); - pluginHooks.addNoop("postBlockGamut"); - - // called with the text of a single block element before / after the span-level conversions (bold, code spans, etc.) have been made - pluginHooks.addNoop("preSpanGamut"); - pluginHooks.addNoop("postSpanGamut"); - - // called with the final cooked HTML code. The result of this plugin hook is the actual output of makeHtml - pluginHooks.addNoop("postConversion"); - - // - // Private state of the converter instance: - // - - // Global hashes, used by various utility routines - var g_urls; - var g_titles; - var g_html_blocks; - - // Used to track when we're inside an ordered or unordered list - // (see _ProcessListItems() for details): - var g_list_level; - - this.makeHtml = function (text) { - - // - // Main function. The order in which other subs are called here is - // essential. Link and image substitutions need to happen before - // _EscapeSpecialCharsWithinTagAttributes(), so that any *'s or _'s in the - // and tags get encoded. - // - - // This will only happen if makeHtml on the same converter instance is called from a plugin hook. - // Don't do that. - if (g_urls) - throw new Error("Recursive call to converter.makeHtml"); - - // Create the private state objects. - g_urls = new SaveHash(); - g_titles = new SaveHash(); - g_html_blocks = []; - g_list_level = 0; - - text = pluginHooks.preConversion(text); - - // attacklab: Replace ~ with ~T - // This lets us use tilde as an escape char to avoid md5 hashes - // The choice of character is arbitray; anything that isn't - // magic in Markdown will work. - text = text.replace(/~/g, "~T"); - - // attacklab: Replace $ with ~D - // RegExp interprets $ as a special character - // when it's in a replacement string - text = text.replace(/\$/g, "~D"); - - // Standardize line endings - text = text.replace(/\r\n/g, "\n"); // DOS to Unix - text = text.replace(/\r/g, "\n"); // Mac to Unix - - // Make sure text begins and ends with a couple of newlines: - text = "\n\n" + text + "\n\n"; - - // Convert all tabs to spaces. - text = _Detab(text); - - // Strip any lines consisting only of spaces and tabs. - // This makes subsequent regexen easier to write, because we can - // match consecutive blank lines with /\n+/ instead of something - // contorted like /[ \t]*\n+/ . - text = text.replace(/^[ \t]+$/mg, ""); - - text = pluginHooks.postNormalization(text); - - // Turn block-level HTML blocks into hash entries - text = _HashHTMLBlocks(text); - - // Strip link definitions, store in hashes. - text = _StripLinkDefinitions(text); - - text = _RunBlockGamut(text); - - text = _UnescapeSpecialChars(text); - - // attacklab: Restore dollar signs - text = text.replace(/~D/g, "$$"); - - // attacklab: Restore tildes - text = text.replace(/~T/g, "~"); - - text = pluginHooks.postConversion(text); - - g_html_blocks = g_titles = g_urls = null; - - return text; - }; - - function _StripLinkDefinitions(text) { - // - // Strips link definitions from text, stores the URLs and titles in - // hash references. - // - - // Link defs are in the form: ^[id]: url "optional title" - - /* - text = text.replace(/ - ^[ ]{0,3}\[(.+)\]: // id = $1 attacklab: g_tab_width - 1 - [ \t]* - \n? // maybe *one* newline - [ \t]* - ? // url = $2 - (?=\s|$) // lookahead for whitespace instead of the lookbehind removed below - [ \t]* - \n? // maybe one newline - [ \t]* - ( // (potential) title = $3 - (\n*) // any lines skipped = $4 attacklab: lookbehind removed - [ \t]+ - ["(] - (.+?) // title = $5 - [")] - [ \t]* - )? // title is optional - (?:\n+|$) - /gm, function(){...}); - */ - - text = text.replace(/^[ ]{0,3}\[(.+)\]:[ \t]*\n?[ \t]*?(?=\s|$)[ \t]*\n?[ \t]*((\n*)["(](.+?)[")][ \t]*)?(?:\n+)/gm, - function (wholeMatch, m1, m2, m3, m4, m5) { - m1 = m1.toLowerCase(); - g_urls.set(m1, _EncodeAmpsAndAngles(m2)); // Link IDs are case-insensitive - if (m4) { - // Oops, found blank lines, so it's not a title. - // Put back the parenthetical statement we stole. - return m3; - } else if (m5) { - g_titles.set(m1, m5.replace(/"/g, """)); - } - - // Completely remove the definition from the text - return ""; - } - ); - - return text; - } - - function _HashHTMLBlocks(text) { - - // Hashify HTML blocks: - // We only want to do this for block-level HTML tags, such as headers, - // lists, and tables. That's because we still want to wrap

                            s around - // "paragraphs" that are wrapped in non-block-level tags, such as anchors, - // phrase emphasis, and spans. The list of tags we're looking for is - // hard-coded: - var block_tags_a = "p|div|h[1-6]|blockquote|pre|table|dl|ol|ul|script|noscript|form|fieldset|iframe|math|ins|del" - var block_tags_b = "p|div|h[1-6]|blockquote|pre|table|dl|ol|ul|script|noscript|form|fieldset|iframe|math" - - // First, look for nested blocks, e.g.: - //

                            - //
                            - // tags for inner block must be indented. - //
                            - //
                            - // - // The outermost tags must start at the left margin for this to match, and - // the inner nested divs must be indented. - // We need to do this before the next, more liberal match, because the next - // match will start at the first `
                            ` and stop at the first `
                            `. - - // attacklab: This regex can be expensive when it fails. - - /* - text = text.replace(/ - ( // save in $1 - ^ // start of line (with /m) - <($block_tags_a) // start tag = $2 - \b // word break - // attacklab: hack around khtml/pcre bug... - [^\r]*?\n // any number of lines, minimally matching - // the matching end tag - [ \t]* // trailing spaces/tabs - (?=\n+) // followed by a newline - ) // attacklab: there are sentinel newlines at end of document - /gm,function(){...}}; - */ - text = text.replace(/^(<(p|div|h[1-6]|blockquote|pre|table|dl|ol|ul|script|noscript|form|fieldset|iframe|math|ins|del)\b[^\r]*?\n<\/\2>[ \t]*(?=\n+))/gm, hashElement); - - // - // Now match more liberally, simply from `\n` to `\n` - // - - /* - text = text.replace(/ - ( // save in $1 - ^ // start of line (with /m) - <($block_tags_b) // start tag = $2 - \b // word break - // attacklab: hack around khtml/pcre bug... - [^\r]*? // any number of lines, minimally matching - .* // the matching end tag - [ \t]* // trailing spaces/tabs - (?=\n+) // followed by a newline - ) // attacklab: there are sentinel newlines at end of document - /gm,function(){...}}; - */ - text = text.replace(/^(<(p|div|h[1-6]|blockquote|pre|table|dl|ol|ul|script|noscript|form|fieldset|iframe|math)\b[^\r]*?.*<\/\2>[ \t]*(?=\n+)\n)/gm, hashElement); - - // Special case just for
                            . It was easier to make a special case than - // to make the other regex more complicated. - - /* - text = text.replace(/ - \n // Starting after a blank line - [ ]{0,3} - ( // save in $1 - (<(hr) // start tag = $2 - \b // word break - ([^<>])*? - \/?>) // the matching end tag - [ \t]* - (?=\n{2,}) // followed by a blank line - ) - /g,hashElement); - */ - text = text.replace(/\n[ ]{0,3}((<(hr)\b([^<>])*?\/?>)[ \t]*(?=\n{2,}))/g, hashElement); - - // Special case for standalone HTML comments: - - /* - text = text.replace(/ - \n\n // Starting after a blank line - [ ]{0,3} // attacklab: g_tab_width - 1 - ( // save in $1 - -]|-[^>])(?:[^-]|-[^-])*)--) // see http://www.w3.org/TR/html-markup/syntax.html#comments and http://meta.stackoverflow.com/q/95256 - > - [ \t]* - (?=\n{2,}) // followed by a blank line - ) - /g,hashElement); - */ - text = text.replace(/\n\n[ ]{0,3}(-]|-[^>])(?:[^-]|-[^-])*)--)>[ \t]*(?=\n{2,}))/g, hashElement); - - // PHP and ASP-style processor instructions ( and <%...%>) - - /* - text = text.replace(/ - (?: - \n\n // Starting after a blank line - ) - ( // save in $1 - [ ]{0,3} // attacklab: g_tab_width - 1 - (?: - <([?%]) // $2 - [^\r]*? - \2> - ) - [ \t]* - (?=\n{2,}) // followed by a blank line - ) - /g,hashElement); - */ - text = text.replace(/(?:\n\n)([ ]{0,3}(?:<([?%])[^\r]*?\2>)[ \t]*(?=\n{2,}))/g, hashElement); - - return text; - } - - function hashElement(wholeMatch, m1) { - var blockText = m1; - - // Undo double lines - blockText = blockText.replace(/^\n+/, ""); - - // strip trailing blank lines - blockText = blockText.replace(/\n+$/g, ""); - - // Replace the element text with a marker ("~KxK" where x is its key) - blockText = "\n\n~K" + (g_html_blocks.push(blockText) - 1) + "K\n\n"; - - return blockText; - } - - var blockGamutHookCallback = function (t) { - return _RunBlockGamut(t); - } - - function _RunBlockGamut(text, doNotUnhash) { - // - // These are all the transformations that form block-level - // tags like paragraphs, headers, and list items. - // - - text = pluginHooks.preBlockGamut(text, blockGamutHookCallback); - - text = _DoHeaders(text); - - // Do Horizontal Rules: - var replacement = "
                            \n"; - text = text.replace(/^[ ]{0,2}([ ]?\*[ ]?){3,}[ \t]*$/gm, replacement); - text = text.replace(/^[ ]{0,2}([ ]?-[ ]?){3,}[ \t]*$/gm, replacement); - text = text.replace(/^[ ]{0,2}([ ]?_[ ]?){3,}[ \t]*$/gm, replacement); - - text = _DoLists(text); - text = _DoCodeBlocks(text); - text = _DoBlockQuotes(text); - - text = pluginHooks.postBlockGamut(text, blockGamutHookCallback); - - // We already ran _HashHTMLBlocks() before, in Markdown(), but that - // was to escape raw HTML in the original Markdown source. This time, - // we're escaping the markup we've just created, so that we don't wrap - //

                            tags around block-level tags. - text = _HashHTMLBlocks(text); - text = _FormParagraphs(text, doNotUnhash); - - return text; - } - - function _RunSpanGamut(text) { - // - // These are all the transformations that occur *within* block-level - // tags like paragraphs, headers, and list items. - // - - text = pluginHooks.preSpanGamut(text); - - text = _DoCodeSpans(text); - text = _EscapeSpecialCharsWithinTagAttributes(text); - text = _EncodeBackslashEscapes(text); - - // Process anchor and image tags. Images must come first, - // because ![foo][f] looks like an anchor. - text = _DoImages(text); - text = _DoAnchors(text); - - // Make links out of things like `` - // Must come after _DoAnchors(), because you can use < and > - // delimiters in inline links like [this](). - text = _DoAutoLinks(text); - - text = text.replace(/~P/g, "://"); // put in place to prevent autolinking; reset now - - text = _EncodeAmpsAndAngles(text); - text = _DoItalicsAndBold(text); - - // Do hard breaks: - text = text.replace(/ +\n/g, "
                            \n"); - - text = pluginHooks.postSpanGamut(text); - - return text; - } - - function _EscapeSpecialCharsWithinTagAttributes(text) { - // - // Within tags -- meaning between < and > -- encode [\ ` * _] so they - // don't conflict with their use in Markdown for code, italics and strong. - // - - // Build a regex to find HTML tags and comments. See Friedl's - // "Mastering Regular Expressions", 2nd Ed., pp. 200-201. - - // SE: changed the comment part of the regex - - var regex = /(<[a-z\/!$]("[^"]*"|'[^']*'|[^'">])*>|-]|-[^>])(?:[^-]|-[^-])*)--)>)/gi; - - text = text.replace(regex, function (wholeMatch) { - var tag = wholeMatch.replace(/(.)<\/?code>(?=.)/g, "$1`"); - tag = escapeCharacters(tag, wholeMatch.charAt(1) == "!" ? "\\`*_/" : "\\`*_"); // also escape slashes in comments to prevent autolinking there -- http://meta.stackoverflow.com/questions/95987 - return tag; - }); - - return text; - } - - function _DoAnchors(text) { - // - // Turn Markdown link shortcuts into XHTML
                            tags. - // - // - // First, handle reference-style links: [link text] [id] - // - - /* - text = text.replace(/ - ( // wrap whole match in $1 - \[ - ( - (?: - \[[^\]]*\] // allow brackets nested one level - | - [^\[] // or anything else - )* - ) - \] - - [ ]? // one optional space - (?:\n[ ]*)? // one optional newline followed by spaces - - \[ - (.*?) // id = $3 - \] - ) - ()()()() // pad remaining backreferences - /g, writeAnchorTag); - */ - text = text.replace(/(\[((?:\[[^\]]*\]|[^\[\]])*)\][ ]?(?:\n[ ]*)?\[(.*?)\])()()()()/g, writeAnchorTag); - - // - // Next, inline-style links: [link text](url "optional title") - // - - /* - text = text.replace(/ - ( // wrap whole match in $1 - \[ - ( - (?: - \[[^\]]*\] // allow brackets nested one level - | - [^\[\]] // or anything else - )* - ) - \] - \( // literal paren - [ \t]* - () // no id, so leave $3 empty - ? - [ \t]* - ( // $5 - (['"]) // quote char = $6 - (.*?) // Title = $7 - \6 // matching quote - [ \t]* // ignore any spaces/tabs between closing quote and ) - )? // title is optional - \) - ) - /g, writeAnchorTag); - */ - - text = text.replace(/(\[((?:\[[^\]]*\]|[^\[\]])*)\]\([ \t]*()?[ \t]*((['"])(.*?)\6[ \t]*)?\))/g, writeAnchorTag); - - // - // Last, handle reference-style shortcuts: [link text] - // These must come last in case you've also got [link test][1] - // or [link test](/foo) - // - - /* - text = text.replace(/ - ( // wrap whole match in $1 - \[ - ([^\[\]]+) // link text = $2; can't contain '[' or ']' - \] - ) - ()()()()() // pad rest of backreferences - /g, writeAnchorTag); - */ - text = text.replace(/(\[([^\[\]]+)\])()()()()()/g, writeAnchorTag); - - return text; - } - - function writeAnchorTag(wholeMatch, m1, m2, m3, m4, m5, m6, m7) { - if (m7 == undefined) m7 = ""; - var whole_match = m1; - var link_text = m2.replace(/:\/\//g, "~P"); // to prevent auto-linking withing the link. will be converted back after the auto-linker runs - var link_id = m3.toLowerCase(); - var url = m4; - var title = m7; - - if (url == "") { - if (link_id == "") { - // lower-case and turn embedded newlines into spaces - link_id = link_text.toLowerCase().replace(/ ?\n/g, " "); - } - url = "#" + link_id; - - if (g_urls.get(link_id) != undefined) { - url = g_urls.get(link_id); - if (g_titles.get(link_id) != undefined) { - title = g_titles.get(link_id); - } - } - else { - if (whole_match.search(/\(\s*\)$/m) > -1) { - // Special case for explicit empty url - url = ""; - } else { - return whole_match; - } - } - } - url = encodeProblemUrlChars(url); - url = escapeCharacters(url, "*_"); - var result = ""; - - return result; - } - - function _DoImages(text) { - // - // Turn Markdown image shortcuts into tags. - // - - // - // First, handle reference-style labeled images: ![alt text][id] - // - - /* - text = text.replace(/ - ( // wrap whole match in $1 - !\[ - (.*?) // alt text = $2 - \] - - [ ]? // one optional space - (?:\n[ ]*)? // one optional newline followed by spaces - - \[ - (.*?) // id = $3 - \] - ) - ()()()() // pad rest of backreferences - /g, writeImageTag); - */ - text = text.replace(/(!\[(.*?)\][ ]?(?:\n[ ]*)?\[(.*?)\])()()()()/g, writeImageTag); - - // - // Next, handle inline images: ![alt text](url "optional title") - // Don't forget: encode * and _ - - /* - text = text.replace(/ - ( // wrap whole match in $1 - !\[ - (.*?) // alt text = $2 - \] - \s? // One optional whitespace character - \( // literal paren - [ \t]* - () // no id, so leave $3 empty - ? // src url = $4 - [ \t]* - ( // $5 - (['"]) // quote char = $6 - (.*?) // title = $7 - \6 // matching quote - [ \t]* - )? // title is optional - \) - ) - /g, writeImageTag); - */ - text = text.replace(/(!\[(.*?)\]\s?\([ \t]*()?[ \t]*((['"])(.*?)\6[ \t]*)?\))/g, writeImageTag); - - return text; - } - - function attributeEncode(text) { - // unconditionally replace angle brackets here -- what ends up in an attribute (e.g. alt or title) - // never makes sense to have verbatim HTML in it (and the sanitizer would totally break it) - return text.replace(/>/g, ">").replace(/" + _RunSpanGamut(m1) + "\n\n"; - } - ); - - text = text.replace(/^(.+)[ \t]*\n-+[ \t]*\n+/gm, - function (matchFound, m1) { - return "

                            " + _RunSpanGamut(m1) + "

                            \n\n"; - } - ); - - // atx-style headers: - // # Header 1 - // ## Header 2 - // ## Header 2 with closing hashes ## - // ... - // ###### Header 6 - // - - /* - text = text.replace(/ - ^(\#{1,6}) // $1 = string of #'s - [ \t]* - (.+?) // $2 = Header text - [ \t]* - \#* // optional closing #'s (not counted) - \n+ - /gm, function() {...}); - */ - - text = text.replace(/^(\#{1,6})[ \t]*(.+?)[ \t]*\#*\n+/gm, - function (wholeMatch, m1, m2) { - var h_level = m1.length; - return "" + _RunSpanGamut(m2) + "\n\n"; - } - ); - - return text; - } - - function _DoLists(text, isInsideParagraphlessListItem) { - // - // Form HTML ordered (numbered) and unordered (bulleted) lists. - // - - // attacklab: add sentinel to hack around khtml/safari bug: - // http://bugs.webkit.org/show_bug.cgi?id=11231 - text += "~0"; - - // Re-usable pattern to match any entirel ul or ol list: - - /* - var whole_list = / - ( // $1 = whole list - ( // $2 - [ ]{0,3} // attacklab: g_tab_width - 1 - ([*+-]|\d+[.]) // $3 = first list item marker - [ \t]+ - ) - [^\r]+? - ( // $4 - ~0 // sentinel for workaround; should be $ - | - \n{2,} - (?=\S) - (?! // Negative lookahead for another list item marker - [ \t]* - (?:[*+-]|\d+[.])[ \t]+ - ) - ) - ) - /g - */ - var whole_list = /^(([ ]{0,3}([*+-]|\d+[.])[ \t]+)[^\r]+?(~0|\n{2,}(?=\S)(?![ \t]*(?:[*+-]|\d+[.])[ \t]+)))/gm; - - if (g_list_level) { - text = text.replace(whole_list, function (wholeMatch, m1, m2) { - var list = m1; - var list_type = (m2.search(/[*+-]/g) > -1) ? "ul" : "ol"; - - var result = _ProcessListItems(list, list_type, isInsideParagraphlessListItem); - - // Trim any trailing whitespace, to put the closing `` - // up on the preceding line, to get it past the current stupid - // HTML block parser. This is a hack to work around the terrible - // hack that is the HTML block parser. - result = result.replace(/\s+$/, ""); - result = "<" + list_type + ">" + result + "\n"; - return result; - }); - } else { - whole_list = /(\n\n|^\n?)(([ ]{0,3}([*+-]|\d+[.])[ \t]+)[^\r]+?(~0|\n{2,}(?=\S)(?![ \t]*(?:[*+-]|\d+[.])[ \t]+)))/g; - text = text.replace(whole_list, function (wholeMatch, m1, m2, m3) { - var runup = m1; - var list = m2; - - var list_type = (m3.search(/[*+-]/g) > -1) ? "ul" : "ol"; - var result = _ProcessListItems(list, list_type); - result = runup + "<" + list_type + ">\n" + result + "\n"; - return result; - }); - } - - // attacklab: strip sentinel - text = text.replace(/~0/, ""); - - return text; - } - - var _listItemMarkers = {ol: "\\d+[.]", ul: "[*+-]"}; - - function _ProcessListItems(list_str, list_type, isInsideParagraphlessListItem) { - // - // Process the contents of a single ordered or unordered list, splitting it - // into individual list items. - // - // list_type is either "ul" or "ol". - - // The $g_list_level global keeps track of when we're inside a list. - // Each time we enter a list, we increment it; when we leave a list, - // we decrement. If it's zero, we're not in a list anymore. - // - // We do this because when we're not inside a list, we want to treat - // something like this: - // - // I recommend upgrading to version - // 8. Oops, now this line is treated - // as a sub-list. - // - // As a single paragraph, despite the fact that the second line starts - // with a digit-period-space sequence. - // - // Whereas when we're inside a list (or sub-list), that line will be - // treated as the start of a sub-list. What a kludge, huh? This is - // an aspect of Markdown's syntax that's hard to parse perfectly - // without resorting to mind-reading. Perhaps the solution is to - // change the syntax rules such that sub-lists must start with a - // starting cardinal number; e.g. "1." or "a.". - - g_list_level++; - - // trim trailing blank lines: - list_str = list_str.replace(/\n{2,}$/, "\n"); - - // attacklab: add sentinel to emulate \z - list_str += "~0"; - - // In the original attacklab showdown, list_type was not given to this function, and anything - // that matched /[*+-]|\d+[.]/ would just create the next
                          2. , causing this mismatch: - // - // Markdown rendered by WMD rendered by MarkdownSharp - // ------------------------------------------------------------------ - // 1. first 1. first 1. first - // 2. second 2. second 2. second - // - third 3. third * third - // - // We changed this to behave identical to MarkdownSharp. This is the constructed RegEx, - // with {MARKER} being one of \d+[.] or [*+-], depending on list_type: - - /* - list_str = list_str.replace(/ - (^[ \t]*) // leading whitespace = $1 - ({MARKER}) [ \t]+ // list marker = $2 - ([^\r]+? // list item text = $3 - (\n+) - ) - (?= - (~0 | \2 ({MARKER}) [ \t]+) - ) - /gm, function(){...}); - */ - - var marker = _listItemMarkers[list_type]; - var re = new RegExp("(^[ \\t]*)(" + marker + ")[ \\t]+([^\\r]+?(\\n+))(?=(~0|\\1(" + marker + ")[ \\t]+))", "gm"); - var last_item_had_a_double_newline = false; - list_str = list_str.replace(re, - function (wholeMatch, m1, m2, m3) { - var item = m3; - var leading_space = m1; - var ends_with_double_newline = /\n\n$/.test(item); - var contains_double_newline = ends_with_double_newline || item.search(/\n{2,}/) > -1; - - if (contains_double_newline || last_item_had_a_double_newline) { - item = _RunBlockGamut(_Outdent(item), /* doNotUnhash = */true); - } - else { - // Recursion for sub-lists: - item = _DoLists(_Outdent(item), /* isInsideParagraphlessListItem= */ true); - item = item.replace(/\n$/, ""); // chomp(item) - if (!isInsideParagraphlessListItem) // only the outer-most item should run this, otherwise it's run multiple times for the inner ones - item = _RunSpanGamut(item); - } - last_item_had_a_double_newline = ends_with_double_newline; - return "
                          3. " + item + "
                          4. \n"; - } - ); - - // attacklab: strip sentinel - list_str = list_str.replace(/~0/g, ""); - - g_list_level--; - return list_str; - } - - function _DoCodeBlocks(text) { - // - // Process Markdown `
                            ` blocks.
                            -            //  
                            -
                            -            /*
                            -             text = text.replace(/
                            -             (?:\n\n|^)
                            -             (                               // $1 = the code block -- one or more lines, starting with a space/tab
                            -             (?:
                            -             (?:[ ]{4}|\t)           // Lines must start with a tab or a tab-width of spaces - attacklab: g_tab_width
                            -             .*\n+
                            -             )+
                            -             )
                            -             (\n*[ ]{0,3}[^ \t\n]|(?=~0))    // attacklab: g_tab_width
                            -             /g ,function(){...});
                            -             */
                            -
                            -            // attacklab: sentinel workarounds for lack of \A and \Z, safari\khtml bug
                            -            text += "~0";
                            -
                            -            text = text.replace(/(?:\n\n|^\n?)((?:(?:[ ]{4}|\t).*\n+)+)(\n*[ ]{0,3}[^ \t\n]|(?=~0))/g,
                            -                function (wholeMatch, m1, m2) {
                            -                    var codeblock = m1;
                            -                    var nextChar = m2;
                            -
                            -                    codeblock = _EncodeCode(_Outdent(codeblock));
                            -                    codeblock = _Detab(codeblock);
                            -                    codeblock = codeblock.replace(/^\n+/g, ""); // trim leading newlines
                            -                    codeblock = codeblock.replace(/\n+$/g, ""); // trim trailing whitespace
                            -
                            -                    codeblock = "
                            " + codeblock + "\n
                            "; - - return "\n\n" + codeblock + "\n\n" + nextChar; - } - ); - - // attacklab: strip sentinel - text = text.replace(/~0/, ""); - - return text; - } - - function hashBlock(text) { - text = text.replace(/(^\n+|\n+$)/g, ""); - return "\n\n~K" + (g_html_blocks.push(text) - 1) + "K\n\n"; - } - - function _DoCodeSpans(text) { - // - // * Backtick quotes are used for spans. - // - // * You can use multiple backticks as the delimiters if you want to - // include literal backticks in the code span. So, this input: - // - // Just type ``foo `bar` baz`` at the prompt. - // - // Will translate to: - // - //

                            Just type foo `bar` baz at the prompt.

                            - // - // There's no arbitrary limit to the number of backticks you - // can use as delimters. If you need three consecutive backticks - // in your code, use four for delimiters, etc. - // - // * You can use spaces to get literal backticks at the edges: - // - // ... type `` `bar` `` ... - // - // Turns to: - // - // ... type `bar` ... - // - - /* - text = text.replace(/ - (^|[^\\]) // Character before opening ` can't be a backslash - (`+) // $2 = Opening run of ` - ( // $3 = The code block - [^\r]*? - [^`] // attacklab: work around lack of lookbehind - ) - \2 // Matching closer - (?!`) - /gm, function(){...}); - */ - - text = text.replace(/(^|[^\\])(`+)([^\r]*?[^`])\2(?!`)/gm, - function (wholeMatch, m1, m2, m3, m4) { - var c = m3; - c = c.replace(/^([ \t]*)/g, ""); // leading whitespace - c = c.replace(/[ \t]*$/g, ""); // trailing whitespace - c = _EncodeCode(c); - c = c.replace(/:\/\//g, "~P"); // to prevent auto-linking. Not necessary in code *blocks*, but in code spans. Will be converted back after the auto-linker runs. - return m1 + "" + c + ""; - } - ); - - return text; - } - - function _EncodeCode(text) { - // - // Encode/escape certain characters inside Markdown code runs. - // The point is that in code, these characters are literals, - // and lose their special Markdown meanings. - // - // Encode all ampersands; HTML entities are not - // entities within a Markdown code span. - text = text.replace(/&/g, "&"); - - // Do the angle bracket song and dance: - text = text.replace(//g, ">"); - - // Now, escape characters that are magic in Markdown: - text = escapeCharacters(text, "\*_{}[]\\", false); - - // jj the line above breaks this: - //--- - - //* Item - - // 1. Subitem - - // special char: * - //--- - - return text; - } - - function _DoItalicsAndBold(text) { - - // must go first: - text = text.replace(/([\W_]|^)(\*\*|__)(?=\S)([^\r]*?\S[\*_]*)\2([\W_]|$)/g, - "$1$3$4"); - - text = text.replace(/([\W_]|^)(\*|_)(?=\S)([^\r\*_]*?\S)\2([\W_]|$)/g, - "$1$3$4"); - - return text; - } - - function _DoBlockQuotes(text) { - - /* - text = text.replace(/ - ( // Wrap whole match in $1 - ( - ^[ \t]*>[ \t]? // '>' at the start of a line - .+\n // rest of the first line - (.+\n)* // subsequent consecutive lines - \n* // blanks - )+ - ) - /gm, function(){...}); - */ - - text = text.replace(/((^[ \t]*>[ \t]?.+\n(.+\n)*\n*)+)/gm, - function (wholeMatch, m1) { - var bq = m1; - - // attacklab: hack around Konqueror 3.5.4 bug: - // "----------bug".replace(/^-/g,"") == "bug" - - bq = bq.replace(/^[ \t]*>[ \t]?/gm, "~0"); // trim one level of quoting - - // attacklab: clean up hack - bq = bq.replace(/~0/g, ""); - - bq = bq.replace(/^[ \t]+$/gm, ""); // trim whitespace-only lines - bq = _RunBlockGamut(bq); // recurse - - bq = bq.replace(/(^|\n)/g, "$1 "); - // These leading spaces screw with
                             content, so we need to fix that:
                            -                    bq = bq.replace(
                            -                        /(\s*
                            [^\r]+?<\/pre>)/gm,
                            -                        function (wholeMatch, m1) {
                            -                            var pre = m1;
                            -                            // attacklab: hack around Konqueror 3.5.4 bug:
                            -                            pre = pre.replace(/^  /mg, "~0");
                            -                            pre = pre.replace(/~0/g, "");
                            -                            return pre;
                            -                        });
                            -
                            -                    return hashBlock("
                            \n" + bq + "\n
                            "); - } - ); - return text; - } - - function _FormParagraphs(text, doNotUnhash) { - // - // Params: - // $text - string to process with html

                            tags - // - - // Strip leading and trailing lines: - text = text.replace(/^\n+/g, ""); - text = text.replace(/\n+$/g, ""); - - var grafs = text.split(/\n{2,}/g); - var grafsOut = []; - - var markerRe = /~K(\d+)K/; - - // - // Wrap

                            tags. - // - var end = grafs.length; - for (var i = 0; i < end; i++) { - var str = grafs[i]; - - // if this is an HTML marker, copy it - if (markerRe.test(str)) { - grafsOut.push(str); - } - else if (/\S/.test(str)) { - str = _RunSpanGamut(str); - str = str.replace(/^([ \t]*)/g, "

                            "); - str += "

                            " - grafsOut.push(str); - } - - } - // - // Unhashify HTML blocks - // - if (!doNotUnhash) { - end = grafsOut.length; - for (var i = 0; i < end; i++) { - var foundAny = true; - while (foundAny) { // we may need several runs, since the data may be nested - foundAny = false; - grafsOut[i] = grafsOut[i].replace(/~K(\d+)K/g, function (wholeMatch, id) { - foundAny = true; - return g_html_blocks[id]; - }); - } - } - } - return grafsOut.join("\n\n"); - } - - function _EncodeAmpsAndAngles(text) { - // Smart processing for ampersands and angle brackets that need to be encoded. - - // Ampersand-encoding based entirely on Nat Irons's Amputator MT plugin: - // http://bumppo.net/projects/amputator/ - text = text.replace(/&(?!#?[xX]?(?:[0-9a-fA-F]+|\w+);)/g, "&"); - - // Encode naked <'s - text = text.replace(/<(?![a-z\/?!]|~D)/gi, "<"); - - return text; - } - - function _EncodeBackslashEscapes(text) { - // - // Parameter: String. - // Returns: The string, with after processing the following backslash - // escape sequences. - // - - // attacklab: The polite way to do this is with the new - // escapeCharacters() function: - // - // text = escapeCharacters(text,"\\",true); - // text = escapeCharacters(text,"`*_{}[]()>#+-.!",true); - // - // ...but we're sidestepping its use of the (slow) RegExp constructor - // as an optimization for Firefox. This function gets called a LOT. - - text = text.replace(/\\(\\)/g, escapeCharacters_callback); - text = text.replace(/\\([`*_{}\[\]()>#+-.!])/g, escapeCharacters_callback); - return text; - } - - var charInsideUrl = "[-A-Z0-9+&@#/%?=~_|[\\]()!:,.;]", - charEndingUrl = "[-A-Z0-9+&@#/%=~_|[\\])]", - autoLinkRegex = new RegExp("(=\"|<)?\\b(https?|ftp)(://" + charInsideUrl + "*" + charEndingUrl + ")(?=$|\\W)", "gi"), - endCharRegex = new RegExp(charEndingUrl, "i"); - - function handleTrailingParens(wholeMatch, lookbehind, protocol, link) { - if (lookbehind) - return wholeMatch; - if (link.charAt(link.length - 1) !== ")") - return "<" + protocol + link + ">"; - var parens = link.match(/[()]/g); - var level = 0; - for (var i = 0; i < parens.length; i++) { - if (parens[i] === "(") { - if (level <= 0) - level = 1; - else - level++; - } - else { - level--; - } - } - var tail = ""; - if (level < 0) { - var re = new RegExp("\\){1," + (-level) + "}$"); - link = link.replace(re, function (trailingParens) { - tail = trailingParens; - return ""; - }); - } - if (tail) { - var lastChar = link.charAt(link.length - 1); - if (!endCharRegex.test(lastChar)) { - tail = lastChar + tail; - link = link.substr(0, link.length - 1); - } - } - return "<" + protocol + link + ">" + tail; - } - - function _DoAutoLinks(text) { - - // note that at this point, all other URL in the text are already hyperlinked as
                            - // *except* for the case - - // automatically add < and > around unadorned raw hyperlinks - // must be preceded by a non-word character (and not by =" or <) and followed by non-word/EOF character - // simulating the lookbehind in a consuming way is okay here, since a URL can neither and with a " nor - // with a <, so there is no risk of overlapping matches. - text = text.replace(autoLinkRegex, handleTrailingParens); - - // autolink anything like - - var replacer = function (wholematch, m1) { - return "" + pluginHooks.plainLinkText(m1) + ""; - } - text = text.replace(/<((https?|ftp):[^'">\s]+)>/gi, replacer); - - // Email addresses: - /* - text = text.replace(/ - < - (?:mailto:)? - ( - [-.\w]+ - \@ - [-a-z0-9]+(\.[-a-z0-9]+)*\.[a-z]+ - ) - > - /gi, _DoAutoLinks_callback()); - */ - - /* disabling email autolinking, since we don't do that on the server, either - text = text.replace(/<(?:mailto:)?([-.\w]+\@[-a-z0-9]+(\.[-a-z0-9]+)*\.[a-z]+)>/gi, - function(wholeMatch,m1) { - return _EncodeEmailAddress( _UnescapeSpecialChars(m1) ); - } - ); - */ - return text; - } - - function _UnescapeSpecialChars(text) { - // - // Swap back in all the special characters we've hidden. - // - text = text.replace(/~E(\d+)E/g, - function (wholeMatch, m1) { - var charCodeToReplace = parseInt(m1); - return String.fromCharCode(charCodeToReplace); - } - ); - return text; - } - - function _Outdent(text) { - // - // Remove one level of line-leading tabs or spaces - // - - // attacklab: hack around Konqueror 3.5.4 bug: - // "----------bug".replace(/^-/g,"") == "bug" - - text = text.replace(/^(\t|[ ]{1,4})/gm, "~0"); // attacklab: g_tab_width - - // attacklab: clean up hack - text = text.replace(/~0/g, "") - - return text; - } - - function _Detab(text) { - if (!/\t/.test(text)) - return text; - - var spaces = [" ", " ", " ", " "], - skew = 0, - v; - - return text.replace(/[\n\t]/g, function (match, offset) { - if (match === "\n") { - skew = offset + 1; - return match; - } - v = (offset - skew) % 4; - skew = offset + 1; - return spaces[v]; - }); - } - - // - // attacklab: Utility functions - // - - var _problemUrlChars = /(?:["'*()[\]:]|~D)/g; - - // hex-encodes some unusual "problem" chars in URLs to avoid URL detection problems - function encodeProblemUrlChars(url) { - if (!url) - return ""; - - var len = url.length; - - return url.replace(_problemUrlChars, function (match, offset) { - if (match == "~D") // escape for dollar - return "%24"; - if (match == ":") { - if (offset == len - 1 || /[0-9\/]/.test(url.charAt(offset + 1))) - return ":" - } - return "%" + match.charCodeAt(0).toString(16); - }); - } - - - function escapeCharacters(text, charsToEscape, afterBackslash) { - // First we have to escape the escape characters so that - // we can build a character class out of them - var regexString = "([" + charsToEscape.replace(/([\[\]\\])/g, "\\$1") + "])"; - - if (afterBackslash) { - regexString = "\\\\" + regexString; - } - - var regex = new RegExp(regexString, "g"); - text = text.replace(regex, escapeCharacters_callback); - - return text; - } - - - function escapeCharacters_callback(wholeMatch, m1) { - var charCodeToEscape = m1.charCodeAt(0); - return "~E" + charCodeToEscape + "E"; - } - - }; // end of the Markdown.Converter constructor - -})(); diff --git a/system/admin/editor/js/Markdown.Editor.js b/system/admin/editor/js/Markdown.Editor.js deleted file mode 100644 index 57fdd556..00000000 --- a/system/admin/editor/js/Markdown.Editor.js +++ /dev/null @@ -1,2468 +0,0 @@ -// needs Markdown.Converter.js at the moment - -(function () { - - var util = {}, - position = {}, - ui = {}, - doc = window.document, - re = window.RegExp, - nav = window.navigator, - SETTINGS = {lineLength: 72}, - - // Used to work around some browser bugs where we can't use feature testing. - uaSniffed = { - isIE: /msie/.test(nav.userAgent.toLowerCase()), - isIE_5or6: /msie 6/.test(nav.userAgent.toLowerCase()) || /msie 5/.test(nav.userAgent.toLowerCase()), - isOpera: /opera/.test(nav.userAgent.toLowerCase()) - }; - - var defaultsStrings = { - bold: "Strong Ctrl+B", - boldexample: "strong text", - - italic: "Emphasis Ctrl+I", - italicexample: "emphasized text", - - strikethrough: "Strikethrough Ctrl+X", - strikethroughexample: "strikethrough text", - - link: "Hyperlink Ctrl+L", - linkdescription: "enter link description here", - linkdialog: "

                            Insert Hyperlink

                            http://example.com/ \"optional title\"

                            ", - - quote: "Blockquote
                            Ctrl+Q", - quoteexample: "Blockquote", - - code: "Code Sample
                             Ctrl+K",
                            -        codeexample: "enter code here",
                            -
                            -        image: "Image  Ctrl+G",
                            -        imagedescription: "enter image description here",
                            -        imagedialog: "

                            Insert Image

                            http://example.com/images/diagram.jpg \"optional title\"

                            Need
                            free image hosting?

                            ", - - olist: "Numbered List
                              Ctrl+O", - ulist: "Bulleted List
                                Ctrl+U", - litem: "List item", - - heading: "Heading

                                /

                                Ctrl+H", - headingexample: "Heading", - - hr: "Horizontal Rule
                                Ctrl+R", - - readmore: "Read More Ctrl+M", - - table: "Table - Ctrl+J", - - undo: "Undo - Ctrl+Z", - redo: "Redo - Ctrl+Y", - redomac: "Redo - Ctrl+Shift+Z", - - help: "Markdown Editing Help" - }; - - - // ------------------------------------------------------------------- - // YOUR CHANGES GO HERE - // - // I've tried to localize the things you are likely to change to - // this area. - // ------------------------------------------------------------------- - - // The default text that appears in the dialog input box when entering - // links. - var imageDefaultText = "http://"; - var linkDefaultText = "http://"; - - //Polyfill for node.remove() from MDN - // from:https://github.com/jserz/js_piece/blob/master/DOM/ChildNode/remove()/remove().md - (function (arr) { - arr.forEach(function (item) { - if (item.hasOwnProperty('remove')) { - return; - } - Object.defineProperty(item, 'remove', { - configurable: true, - enumerable: true, - writable: true, - value: function remove() { - this.parentNode.removeChild(this); - } - }); - }); - })([Element.prototype, CharacterData.prototype, DocumentType.prototype]); - // ------------------------------------------------------------------- - // END OF YOUR CHANGES - // ------------------------------------------------------------------- - - // options, if given, can have the following properties: - // options.helpButton = { handler: yourEventHandler } - // options.strings = { italicexample: "slanted text" } - // `yourEventHandler` is the click handler for the help button. - // If `options.helpButton` isn't given, not help button is created. - // `options.strings` can have any or all of the same properties as - // `defaultStrings` above, so you can just override some string displayed - // to the user on a case-by-case basis, or translate all strings to - // a different language. - // - // For backwards compatibility reasons, the `options` argument can also - // be just the `helpButton` object, and `strings.help` can also be set via - // `helpButton.title`. This should be considered legacy. - // - // The constructed editor object has the methods: - // - getConverter() returns the markdown converter object that was passed to the constructor - // - run() actually starts the editor; should be called after all necessary plugins are registered. Calling this more than once is a no-op. - // - refreshPreview() forces the preview to be updated. This method is only available after run() was called. - Markdown.Editor = function (markdownConverter, idPostfix, options) { - - options = options || {}; - - if (typeof options.handler === "function") { //backwards compatible behavior - options = {helpButton: options}; - } - options.strings = options.strings || {}; - if (options.helpButton) { - options.strings.help = options.strings.help || options.helpButton.title; - } - var getString = function (identifier) { - return options.strings[identifier] || defaultsStrings[identifier]; - } - - idPostfix = idPostfix || ""; - - var hooks = this.hooks = new Markdown.HookCollection(); - hooks.addNoop("onPreviewRefresh"); // called with no arguments after the preview has been refreshed - hooks.addNoop("postBlockquoteCreation"); // called with the user's selection *after* the blockquote was created; should return the actual to-be-inserted text - hooks.addFalse("insertImageDialog"); - /* called with one parameter: a callback to be called with the URL of the image. If the application creates - * its own image insertion dialog, this hook should return true, and the callback should be called with the chosen - * image url (or null if the user cancelled). If this hook returns false, the default dialog will be used. - */ - - this.getConverter = function () { - return markdownConverter; - } - - var that = this, - panels; - - this.run = function () { - if (panels) - return; // already initialized - - panels = new PanelCollection(idPostfix); - var commandManager = new CommandManager(hooks, getString); - var previewManager = new PreviewManager(markdownConverter, panels, function () { - hooks.onPreviewRefresh(); - }); - var undoManager, uiManager; - - if (!/\?noundo/.test(doc.location.href)) { - undoManager = new UndoManager(function () { - previewManager.refresh(); - if (uiManager) // not available on the first call - uiManager.setUndoRedoButtonStates(); - }, panels); - this.textOperation = function (f) { - undoManager.setCommandMode(); - f(); - that.refreshPreview(); - } - } - - uiManager = new UIManager(idPostfix, panels, undoManager, previewManager, commandManager, options.helpButton, getString); - uiManager.setUndoRedoButtonStates(); - - var forceRefresh = that.refreshPreview = function () { - previewManager.refresh(true); - }; - - forceRefresh(); - }; - - } - - // before: contains all the text in the input box BEFORE the selection. - // after: contains all the text in the input box AFTER the selection. - function Chunks() { - } - - // startRegex: a regular expression to find the start tag - // endRegex: a regular expresssion to find the end tag - Chunks.prototype.findTags = function (startRegex, endRegex) { - - var chunkObj = this; - var regex; - - if (startRegex) { - - regex = util.extendRegExp(startRegex, "", "$"); - - this.before = this.before.replace(regex, - function (match) { - chunkObj.startTag = chunkObj.startTag + match; - return ""; - }); - - regex = util.extendRegExp(startRegex, "^", ""); - - this.selection = this.selection.replace(regex, - function (match) { - chunkObj.startTag = chunkObj.startTag + match; - return ""; - }); - } - - if (endRegex) { - - regex = util.extendRegExp(endRegex, "", "$"); - - this.selection = this.selection.replace(regex, - function (match) { - chunkObj.endTag = match + chunkObj.endTag; - return ""; - }); - - regex = util.extendRegExp(endRegex, "^", ""); - - this.after = this.after.replace(regex, - function (match) { - chunkObj.endTag = match + chunkObj.endTag; - return ""; - }); - } - }; - - // If remove is false, the whitespace is transferred - // to the before/after regions. - // - // If remove is true, the whitespace disappears. - Chunks.prototype.trimWhitespace = function (remove) { - var beforeReplacer, afterReplacer, that = this; - if (remove) { - beforeReplacer = afterReplacer = ""; - } else { - beforeReplacer = function (s) { - that.before += s; - return ""; - } - afterReplacer = function (s) { - that.after = s + that.after; - return ""; - } - } - - this.selection = this.selection.replace(/^(\s*)/, beforeReplacer).replace(/(\s*)$/, afterReplacer); - }; - - - Chunks.prototype.skipLines = function (nLinesBefore, nLinesAfter, findExtraNewlines) { - - if (nLinesBefore === undefined) { - nLinesBefore = 1; - } - - if (nLinesAfter === undefined) { - nLinesAfter = 1; - } - - nLinesBefore++; - nLinesAfter++; - - var regexText; - var replacementText; - - // chrome bug ... documented at: http://meta.stackoverflow.com/questions/63307/blockquote-glitch-in-editor-in-chrome-6-and-7/65985#65985 - if (navigator.userAgent.match(/Chrome/)) { - "X".match(/()./); - } - - this.selection = this.selection.replace(/(^\n*)/, ""); - - this.startTag = this.startTag + re.$1; - - this.selection = this.selection.replace(/(\n*$)/, ""); - this.endTag = this.endTag + re.$1; - this.startTag = this.startTag.replace(/(^\n*)/, ""); - this.before = this.before + re.$1; - this.endTag = this.endTag.replace(/(\n*$)/, ""); - this.after = this.after + re.$1; - - if (this.before) { - - regexText = replacementText = ""; - - while (nLinesBefore--) { - regexText += "\\n?"; - replacementText += "\n"; - } - - if (findExtraNewlines) { - regexText = "\\n*"; - } - this.before = this.before.replace(new re(regexText + "$", ""), replacementText); - } - - if (this.after) { - - regexText = replacementText = ""; - - while (nLinesAfter--) { - regexText += "\\n?"; - replacementText += "\n"; - } - if (findExtraNewlines) { - regexText = "\\n*"; - } - - this.after = this.after.replace(new re(regexText, ""), replacementText); - } - }; - - // end of Chunks - - // A collection of the important regions on the page. - // Cached so we don't have to keep traversing the DOM. - // Also holds ieCachedRange and ieCachedScrollTop, where necessary; working around - // this issue: - // Internet explorer has problems with CSS sprite buttons that use HTML - // lists. When you click on the background image "button", IE will - // select the non-existent link text and discard the selection in the - // textarea. The solution to this is to cache the textarea selection - // on the button's mousedown event and set a flag. In the part of the - // code where we need to grab the selection, we check for the flag - // and, if it's set, use the cached area instead of querying the - // textarea. - // - // This ONLY affects Internet Explorer (tested on versions 6, 7 - // and 8) and ONLY on button clicks. Keyboard shortcuts work - // normally since the focus never leaves the textarea. - function PanelCollection(postfix) { - this.buttonBar = doc.getElementById("wmd-button-bar" + postfix); - this.preview = doc.getElementById("wmd-preview" + postfix); - this.input = doc.getElementById("wmd-input" + postfix); - }; - - // Returns true if the DOM element is visible, false if it's hidden. - // Checks if display is anything other than none. - util.isVisible = function (elem) { - - if (window.getComputedStyle) { - // Most browsers - return window.getComputedStyle(elem, null).getPropertyValue("display") !== "none"; - } - else if (elem.currentStyle) { - // IE - return elem.currentStyle["display"] !== "none"; - } - }; - - - // Adds a listener callback to a DOM element which is fired on a specified - // event. - util.addEvent = function (elem, event, listener) { - if (elem.attachEvent) { - // IE only. The "on" is mandatory. - elem.attachEvent("on" + event, listener); - } - else { - // Other browsers. - elem.addEventListener(event, listener, false); - } - }; - - - // Removes a listener callback from a DOM element which is fired on a specified - // event. - util.removeEvent = function (elem, event, listener) { - if (elem.detachEvent) { - // IE only. The "on" is mandatory. - elem.detachEvent("on" + event, listener); - } - else { - // Other browsers. - elem.removeEventListener(event, listener, false); - } - }; - - // Converts \r\n and \r to \n. - util.fixEolChars = function (text) { - text = text.replace(/\r\n/g, "\n"); - text = text.replace(/\r/g, "\n"); - return text; - }; - - // Extends a regular expression. Returns a new RegExp - // using pre + regex + post as the expression. - // Used in a few functions where we have a base - // expression and we want to pre- or append some - // conditions to it (e.g. adding "$" to the end). - // The flags are unchanged. - // - // regex is a RegExp, pre and post are strings. - util.extendRegExp = function (regex, pre, post) { - - if (pre === null || pre === undefined) { - pre = ""; - } - if (post === null || post === undefined) { - post = ""; - } - - var pattern = regex.toString(); - var flags; - - // Replace the flags with empty space and store them. - pattern = pattern.replace(/\/([gim]*)$/, function (wholeMatch, flagsPart) { - flags = flagsPart; - return ""; - }); - - // Remove the slash delimiters on the regular expression. - pattern = pattern.replace(/(^\/|\/$)/g, ""); - pattern = pre + pattern + post; - - return new re(pattern, flags); - } - - // UNFINISHED - // The assignment in the while loop makes jslint cranky. - // I'll change it to a better loop later. - position.getTop = function (elem, isInner) { - var result = elem.offsetTop; - if (!isInner) { - while (elem = elem.offsetParent) { - result += elem.offsetTop; - } - } - return result; - }; - - position.getHeight = function (elem) { - return elem.offsetHeight || elem.scrollHeight; - }; - - position.getWidth = function (elem) { - return elem.offsetWidth || elem.scrollWidth; - }; - - position.getPageSize = function () { - - var scrollWidth, scrollHeight; - var innerWidth, innerHeight; - - // It's not very clear which blocks work with which browsers. - if (self.innerHeight && self.scrollMaxY) { - scrollWidth = doc.body.scrollWidth; - scrollHeight = self.innerHeight + self.scrollMaxY; - } - else if (doc.body.scrollHeight > doc.body.offsetHeight) { - scrollWidth = doc.body.scrollWidth; - scrollHeight = doc.body.scrollHeight; - } - else { - scrollWidth = doc.body.offsetWidth; - scrollHeight = doc.body.offsetHeight; - } - - if (self.innerHeight) { - // Non-IE browser - innerWidth = self.innerWidth; - innerHeight = self.innerHeight; - } - else if (doc.documentElement && doc.documentElement.clientHeight) { - // Some versions of IE (IE 6 w/ a DOCTYPE declaration) - innerWidth = doc.documentElement.clientWidth; - innerHeight = doc.documentElement.clientHeight; - } - else if (doc.body) { - // Other versions of IE - innerWidth = doc.body.clientWidth; - innerHeight = doc.body.clientHeight; - } - - var maxWidth = Math.max(scrollWidth, innerWidth); - var maxHeight = Math.max(scrollHeight, innerHeight); - return [maxWidth, maxHeight, innerWidth, innerHeight]; - }; - - // Handles pushing and popping TextareaStates for undo/redo commands. - // I should rename the stack variables to list. - function UndoManager(callback, panels) { - - var undoObj = this; - var undoStack = []; // A stack of undo states - var stackPtr = 0; // The index of the current state - var mode = "none"; - var lastState; // The last state - var timer; // The setTimeout handle for cancelling the timer - var inputStateObj; - - // Set the mode for later logic steps. - var setMode = function (newMode, noSave) { - if (mode != newMode) { - mode = newMode; - if (!noSave) { - saveState(); - } - } - - if (!uaSniffed.isIE || mode != "moving") { - timer = setTimeout(refreshState, 1); - } - else { - inputStateObj = null; - } - }; - - var refreshState = function (isInitialState) { - inputStateObj = new TextareaState(panels, isInitialState); - timer = undefined; - }; - - this.setCommandMode = function () { - mode = "command"; - saveState(); - timer = setTimeout(refreshState, 0); - }; - - this.canUndo = function () { - return stackPtr > 1; - }; - - this.canRedo = function () { - if (undoStack[stackPtr + 1]) { - return true; - } - return false; - }; - - // Removes the last state and restores it. - this.undo = function () { - - if (undoObj.canUndo()) { - if (lastState) { - // What about setting state -1 to null or checking for undefined? - lastState.restore(); - lastState = null; - } - else { - undoStack[stackPtr] = new TextareaState(panels); - undoStack[--stackPtr].restore(); - - if (callback) { - callback(); - } - } - } - - mode = "none"; - panels.input.focus(); - refreshState(); - }; - - // Redo an action. - this.redo = function () { - - if (undoObj.canRedo()) { - - undoStack[++stackPtr].restore(); - - if (callback) { - callback(); - } - } - - mode = "none"; - panels.input.focus(); - refreshState(); - }; - - // Push the input area state to the stack. - var saveState = function () { - var currState = inputStateObj || new TextareaState(panels); - - if (!currState) { - return false; - } - if (mode == "moving") { - if (!lastState) { - lastState = currState; - } - return; - } - if (lastState) { - if (undoStack[stackPtr - 1].text != lastState.text) { - undoStack[stackPtr++] = lastState; - } - lastState = null; - } - undoStack[stackPtr++] = currState; - undoStack[stackPtr + 1] = null; - if (callback) { - callback(); - } - }; - - var handleCtrlYZ = function (event) { - - var handled = false; - - if ((event.ctrlKey || event.metaKey) && !event.altKey) { - - // IE and Opera do not support charCode. - var keyCode = event.charCode || event.keyCode; - var keyCodeChar = String.fromCharCode(keyCode); - - switch (keyCodeChar.toLowerCase()) { - - case "y": - undoObj.redo(); - handled = true; - break; - - case "z": - if (!event.shiftKey) { - undoObj.undo(); - } - else { - undoObj.redo(); - } - handled = true; - break; - } - } - - if (handled) { - if (event.preventDefault) { - event.preventDefault(); - } - if (window.event) { - window.event.returnValue = false; - } - return; - } - }; - - // Set the mode depending on what is going on in the input area. - var handleModeChange = function (event) { - - if (!event.ctrlKey && !event.metaKey) { - - var keyCode = event.keyCode; - - if ((keyCode >= 33 && keyCode <= 40) || (keyCode >= 63232 && keyCode <= 63235)) { - // 33 - 40: page up/dn and arrow keys - // 63232 - 63235: page up/dn and arrow keys on safari - setMode("moving"); - } - else if (keyCode == 8 || keyCode == 46 || keyCode == 127) { - // 8: backspace - // 46: delete - // 127: delete - setMode("deleting"); - } - else if (keyCode == 13) { - // 13: Enter - setMode("newlines"); - } - else if (keyCode == 27) { - // 27: escape - setMode("escape"); - } - else if ((keyCode < 16 || keyCode > 20) && keyCode != 91) { - // 16-20 are shift, etc. - // 91: left window key - // I think this might be a little messed up since there are - // a lot of nonprinting keys above 20. - setMode("typing"); - } - } - }; - - var setEventHandlers = function () { - util.addEvent(panels.input, "keypress", function (event) { - // keyCode 89: y - // keyCode 90: z - if ((event.ctrlKey || event.metaKey) && !event.altKey && (event.keyCode == 89 || event.keyCode == 90)) { - event.preventDefault(); - } - }); - - var handlePaste = function () { - if (uaSniffed.isIE || (inputStateObj && inputStateObj.text != panels.input.value)) { - if (timer == undefined) { - mode = "paste"; - saveState(); - refreshState(); - } - } - }; - - util.addEvent(panels.input, "keydown", handleCtrlYZ); - util.addEvent(panels.input, "keydown", handleModeChange); - util.addEvent(panels.input, "mousedown", function () { - setMode("moving"); - }); - - panels.input.onpaste = handlePaste; - panels.input.ondrop = handlePaste; - }; - - var init = function () { - setEventHandlers(); - refreshState(true); - saveState(); - }; - - init(); - } - - // end of UndoManager - - // The input textarea state/contents. - // This is used to implement undo/redo by the undo manager. - function TextareaState(panels, isInitialState) { - - // Aliases - var stateObj = this; - var inputArea = panels.input; - this.init = function () { - if (!util.isVisible(inputArea)) { - return; - } - if (!isInitialState && doc.activeElement && doc.activeElement !== inputArea) { // this happens when tabbing out of the input box - return; - } - - this.setInputAreaSelectionStartEnd(); - this.scrollTop = inputArea.scrollTop; - if (!this.text && inputArea.selectionStart || inputArea.selectionStart === 0) { - this.text = inputArea.value; - } - - } - - // Sets the selected text in the input box after we've performed an - // operation. - this.setInputAreaSelection = function () { - - if (!util.isVisible(inputArea)) { - return; - } - - if (inputArea.selectionStart !== undefined && !uaSniffed.isOpera) { - - inputArea.focus(); - inputArea.selectionStart = stateObj.start; - inputArea.selectionEnd = stateObj.end; - inputArea.scrollTop = stateObj.scrollTop; - } - else if (doc.selection) { - - if (doc.activeElement && doc.activeElement !== inputArea) { - return; - } - - inputArea.focus(); - var range = inputArea.createTextRange(); - range.moveStart("character", -inputArea.value.length); - range.moveEnd("character", -inputArea.value.length); - range.moveEnd("character", stateObj.end); - range.moveStart("character", stateObj.start); - range.select(); - } - }; - - this.setInputAreaSelectionStartEnd = function () { - - if (!panels.ieCachedRange && (inputArea.selectionStart || inputArea.selectionStart === 0)) { - - stateObj.start = inputArea.selectionStart; - stateObj.end = inputArea.selectionEnd; - } - else if (doc.selection) { - - stateObj.text = util.fixEolChars(inputArea.value); - - // IE loses the selection in the textarea when buttons are - // clicked. On IE we cache the selection. Here, if something is cached, - // we take it. - var range = panels.ieCachedRange || doc.selection.createRange(); - - var fixedRange = util.fixEolChars(range.text); - var marker = "\x07"; - var markedRange = marker + fixedRange + marker; - range.text = markedRange; - var inputText = util.fixEolChars(inputArea.value); - - range.moveStart("character", -markedRange.length); - range.text = fixedRange; - - stateObj.start = inputText.indexOf(marker); - stateObj.end = inputText.lastIndexOf(marker) - marker.length; - - var len = stateObj.text.length - util.fixEolChars(inputArea.value).length; - - if (len) { - range.moveStart("character", -fixedRange.length); - while (len--) { - fixedRange += "\n"; - stateObj.end += 1; - } - range.text = fixedRange; - } - - if (panels.ieCachedRange) - stateObj.scrollTop = panels.ieCachedScrollTop; // this is set alongside with ieCachedRange - - panels.ieCachedRange = null; - - this.setInputAreaSelection(); - } - }; - - // Restore this state into the input area. - this.restore = function () { - - if (stateObj.text != undefined && stateObj.text != inputArea.value) { - inputArea.value = stateObj.text; - } - this.setInputAreaSelection(); - inputArea.scrollTop = stateObj.scrollTop; - }; - - // Gets a collection of HTML chunks from the inptut textarea. - this.getChunks = function () { - - var chunk = new Chunks(); - chunk.before = util.fixEolChars(stateObj.text.substring(0, stateObj.start)); - chunk.startTag = ""; - chunk.selection = util.fixEolChars(stateObj.text.substring(stateObj.start, stateObj.end)); - chunk.endTag = ""; - chunk.after = util.fixEolChars(stateObj.text.substring(stateObj.end)); - chunk.scrollTop = stateObj.scrollTop; - - return chunk; - }; - - // Sets the TextareaState properties given a chunk of markdown. - this.setChunks = function (chunk) { - - chunk.before = chunk.before + chunk.startTag; - chunk.after = chunk.endTag + chunk.after; - - this.start = chunk.before.length; - this.end = chunk.before.length + chunk.selection.length; - this.text = chunk.before + chunk.selection + chunk.after; - this.scrollTop = chunk.scrollTop; - }; - this.init(); - }; - - function PreviewManager(converter, panels, previewRefreshCallback) { - - var managerObj = this; - var timeout; - var elapsedTime; - var oldInputText; - var maxDelay = 3000; - var startType = "delayed"; // The other legal value is "manual" - - // Adds event listeners to elements - var setupEvents = function (inputElem, listener) { - - util.addEvent(inputElem, "input", listener); - inputElem.onpaste = listener; - inputElem.ondrop = listener; - - util.addEvent(inputElem, "keypress", listener); - util.addEvent(inputElem, "keydown", listener); - }; - - var getDocScrollTop = function () { - - var result = 0; - - if (window.innerHeight) { - result = window.pageYOffset; - } - else if (doc.documentElement && doc.documentElement.scrollTop) { - result = doc.documentElement.scrollTop; - } - else if (doc.body) { - result = doc.body.scrollTop; - } - - return result; - }; - - var makePreviewHtml = function () { - - // If there is no registered preview panel - // there is nothing to do. - if (!panels.preview) - return; - - - var text = panels.input.value; - if (text && text == oldInputText) { - return; // Input text hasn't changed. - } - else { - oldInputText = text; - } - - var prevTime = new Date().getTime(); - - text = converter.makeHtml(text); - - // Calculate the processing time of the HTML creation. - // It's used as the delay time in the event listener. - var currTime = new Date().getTime(); - elapsedTime = currTime - prevTime; - - pushPreviewHtml(text); - }; - - // setTimeout is already used. Used as an event listener. - var applyTimeout = function () { - - if (timeout) { - clearTimeout(timeout); - timeout = undefined; - } - - if (startType !== "manual") { - - var delay = 0; - - if (startType === "delayed") { - delay = elapsedTime; - } - - if (delay > maxDelay) { - delay = maxDelay; - } - timeout = setTimeout(makePreviewHtml, delay); - } - }; - - var getScaleFactor = function (panel) { - if (panel.scrollHeight <= panel.clientHeight) { - return 1; - } - return panel.scrollTop / (panel.scrollHeight - panel.clientHeight); - }; - - var setPanelScrollTops = function () { - if (panels.preview) { - panels.preview.scrollTop = (panels.preview.scrollHeight - panels.preview.clientHeight) * getScaleFactor(panels.preview); - } - }; - - this.refresh = function (requiresRefresh) { - - if (requiresRefresh) { - oldInputText = ""; - makePreviewHtml(); - } - else { - applyTimeout(); - } - }; - - this.processingTime = function () { - return elapsedTime; - }; - - var isFirstTimeFilled = true; - - // IE doesn't let you use innerHTML if the element is contained somewhere in a table - // (which is the case for inline editing) -- in that case, detach the element, set the - // value, and reattach. Yes, that *is* ridiculous. - var ieSafePreviewSet = function (text) { - var preview = panels.preview; - var parent = preview.parentNode; - var sibling = preview.nextSibling; - parent.removeChild(preview); - preview.innerHTML = text; - if (!sibling) - parent.appendChild(preview); - else - parent.insertBefore(preview, sibling); - } - - var nonSuckyBrowserPreviewSet = function (text) { - panels.preview.innerHTML = text; - } - - var previewSetter; - - var previewSet = function (text) { - if (previewSetter) - return previewSetter(text); - - try { - nonSuckyBrowserPreviewSet(text); - previewSetter = nonSuckyBrowserPreviewSet; - } catch (e) { - previewSetter = ieSafePreviewSet; - previewSetter(text); - } - }; - - var pushPreviewHtml = function (text) { - - var emptyTop = position.getTop(panels.input) - getDocScrollTop(); - - if (panels.preview) { - previewSet(text); - previewRefreshCallback(); - } - - setPanelScrollTops(); - - if (isFirstTimeFilled) { - isFirstTimeFilled = false; - return; - } - - var fullTop = position.getTop(panels.input) - getDocScrollTop(); - - if (uaSniffed.isIE) { - setTimeout(function () { - window.scrollBy(0, fullTop - emptyTop); - }, 0); - } - else { - window.scrollBy(0, fullTop - emptyTop); - } - }; - - var init = function () { - - setupEvents(panels.input, applyTimeout); - makePreviewHtml(); - - if (panels.preview) { - panels.preview.scrollTop = 0; - } - }; - - init(); - }; - - // Creates the background behind the hyperlink text entry box. - // And download dialog - // Most of this has been moved to CSS but the div creation and - // browser-specific hacks remain here. - ui.createBackground = function () { - - var background = doc.createElement("div"), - style = background.style; - - background.className = "wmd-prompt-background"; - - style.position = "absolute"; - style.top = "0"; - - style.zIndex = "1000"; - - if (uaSniffed.isIE) { - style.filter = "alpha(opacity=50)"; - } - else { - style.opacity = "0.5"; - } - - var pageSize = position.getPageSize(); - style.height = pageSize[1] + "px"; - - if (uaSniffed.isIE) { - style.left = doc.documentElement.scrollLeft; - style.width = doc.documentElement.clientWidth; - } - else { - style.left = "0"; - style.width = "100%"; - } - - doc.body.appendChild(background); - return background; - }; - - // This simulates a modal dialog box and asks for the URL when you - // click the hyperlink or image buttons. - // - // text: The html for the input box. - // defaultInputText: The default value that appears in the input box. - // callback: The function which is executed when the prompt is dismissed, either via OK or Cancel. - // It receives a single argument; either the entered text (if OK was chosen) or null (if Cancel - // was chosen). - ui.prompt = function (text, defaultInputText, callback) { - - // These variables need to be declared at this level since they are used - // in multiple functions. - var dialog; // The dialog box. - var input; // The text box where you enter the hyperlink. - - - if (defaultInputText === undefined) { - defaultInputText = ""; - } - - // Used as a keydown event handler. Esc dismisses the prompt. - // Key code 27 is ESC. - var checkEscape = function (key) { - var code = (key.charCode || key.keyCode); - if (code === 27) { - close(true); - } - }; - - // Dismisses the hyperlink input box. - // isCancel is true if we don't care about the input text. - // isCancel is false if we are going to keep the text. - var close = function (isCancel) { - util.removeEvent(doc.body, "keydown", checkEscape); - var text = input.value; - - if (isCancel) { - text = null; - } - else { - // Fixes common pasting errors. - text = text.replace(/^http:\/\/(https?|ftp):\/\//, '$1://'); - if (!/^(?:https?|ftp):\/\//.test(text)) - text = 'http://' + text; - } - - dialog.parentNode.removeChild(dialog); - - callback(text); - return false; - }; - - - // Create the text input box form/window. - var createDialog = function () { - - // The main dialog box. - dialog = doc.createElement("div"); - dialog.className = "wmd-prompt-dialog"; - dialog.style.padding = "10px;"; - dialog.style.position = "fixed"; - dialog.style.width = "400px"; - dialog.style.zIndex = "1001"; - - // The dialog text. - var question = doc.createElement("div"); - question.innerHTML = text; - question.style.padding = "5px"; - dialog.appendChild(question); - - // The web form container for the text box and buttons. - var form = doc.createElement("form"), - style = form.style; - form.onsubmit = function () { - return close(false); - }; - style.padding = "0"; - style.margin = "0"; - style.cssFloat = "left"; - style.width = "100%"; - style.textAlign = "center"; - style.position = "relative"; - dialog.appendChild(form); - - // The input text box - input = doc.createElement("input"); - input.type = "text"; - input.value = defaultInputText; - style = input.style; - style.display = "block"; - style.width = "80%"; - style.marginLeft = style.marginRight = "auto"; - form.appendChild(input); - - // The ok button - var okButton = doc.createElement("input"); - okButton.type = "button"; - okButton.onclick = function () { - return close(false); - }; - okButton.value = "OK"; - style = okButton.style; - style.margin = "10px"; - style.display = "inline"; - style.width = "7em"; - - - // The cancel button - var cancelButton = doc.createElement("input"); - cancelButton.type = "button"; - cancelButton.onclick = function () { - return close(true); - }; - cancelButton.value = "Cancel"; - style = cancelButton.style; - style.margin = "10px"; - style.display = "inline"; - style.width = "7em"; - - form.appendChild(okButton); - form.appendChild(cancelButton); - - util.addEvent(doc.body, "keydown", checkEscape); - dialog.style.top = "50%"; - dialog.style.left = "50%"; - dialog.style.display = "block"; - if (uaSniffed.isIE_5or6) { - dialog.style.position = "absolute"; - dialog.style.top = doc.documentElement.scrollTop + 200 + "px"; - dialog.style.left = "50%"; - } - doc.body.appendChild(dialog); - - // This has to be done AFTER adding the dialog to the form if you - // want it to be centered. - dialog.style.marginTop = -(position.getHeight(dialog) / 2) + "px"; - dialog.style.marginLeft = -(position.getWidth(dialog) / 2) + "px"; - - }; - - // Why is this in a zero-length timeout? - // Is it working around a browser bug? - setTimeout(function () { - - createDialog(); - - var defTextLen = defaultInputText.length; - if (input.selectionStart !== undefined) { - input.selectionStart = 0; - input.selectionEnd = defTextLen; - } - else if (input.createTextRange) { - var range = input.createTextRange(); - range.collapse(false); - range.moveStart("character", -defTextLen); - range.moveEnd("character", defTextLen); - range.select(); - } - - input.focus(); - }, 0); - }; - - function UIManager(postfix, panels, undoManager, previewManager, commandManager, helpOptions, getString) { - - var inputBox = panels.input, - buttons = {}; // buttons.undo, buttons.link, etc. The actual DOM elements. - - makeSpritedButtonRow(); - - var keyEvent = "keydown"; - if (uaSniffed.isOpera) { - keyEvent = "keypress"; - } - - util.addEvent(inputBox, keyEvent, function (key) { - - // Check to see if we have a button key and, if so execute the callback. - if ((key.ctrlKey || key.metaKey) && !key.altKey && !key.shiftKey) { - - var keyCode = key.charCode || key.keyCode; - var keyCodeStr = String.fromCharCode(keyCode).toLowerCase(); - - switch (keyCodeStr) { - case "b": - doClick(buttons.bold); - break; - case "i": - doClick(buttons.italic); - break; - case "l": - doClick(buttons.link); - break; - case "q": - doClick(buttons.quote); - break; - case "k": - doClick(buttons.code); - break; - case "g": - doClick(buttons.image); - break; - case "o": - doClick(buttons.olist); - break; - case "u": - doClick(buttons.ulist); - break; - case "h": - doClick(buttons.heading); - break; - case "r": - doClick(buttons.hr); - break; - case "m": - doClick(buttons.readmore); - break; - // case "x": - // doClick(buttons.strikethrough); - // break; - case "j": - doClick(buttons.table); - break; - // case "y": - // doClick(buttons.redo); - // break; - // case "z": - // if (key.shiftKey) { - // doClick(buttons.redo); - // } - // else { - // doClick(buttons.undo); - // } - // break; - default: - return; - } - - - if (key.preventDefault) { - key.preventDefault(); - } - - if (window.event) { - window.event.returnValue = false; - } - } - }); - - // Auto-indent on shift-enter - util.addEvent(inputBox, "keyup", function (key) { - if (key.shiftKey && !key.ctrlKey && !key.metaKey) { - var keyCode = key.charCode || key.keyCode; - // Character 13 is Enter - if (keyCode === 13) { - var fakeButton = {}; - fakeButton.textOp = bindCommand("doAutoindent"); - doClick(fakeButton); - } - } - }); - - // special handler because IE clears the context of the textbox on ESC - if (uaSniffed.isIE) { - util.addEvent(inputBox, "keydown", function (key) { - var code = key.keyCode; - if (code === 27) { - return false; - } - }); - } - - - // Perform the button's action. - function doClick(button) { - - inputBox.focus(); - - if (button.textOp) { - - if (undoManager) { - undoManager.setCommandMode(); - } - - var state = new TextareaState(panels); - - if (!state) { - return; - } - - var chunks = state.getChunks(); - - // Some commands launch a "modal" prompt dialog. Javascript - // can't really make a modal dialog box and the WMD code - // will continue to execute while the dialog is displayed. - // This prevents the dialog pattern I'm used to and means - // I can't do something like this: - // - // var link = CreateLinkDialog(); - // makeMarkdownLink(link); - // - // Instead of this straightforward method of handling a - // dialog I have to pass any code which would execute - // after the dialog is dismissed (e.g. link creation) - // in a function parameter. - // - // Yes this is awkward and I think it sucks, but there's - // no real workaround. Only the image and link code - // create dialogs and require the function pointers. - var fixupInputArea = function () { - - inputBox.focus(); - - if (chunks) { - state.setChunks(chunks); - } - - state.restore(); - previewManager.refresh(); - }; - - var noCleanup = button.textOp(chunks, fixupInputArea); - - if (!noCleanup) { - fixupInputArea(); - } - - } - - if (button.execute) { - button.execute(undoManager); - } - }; - - function setupButton(button, isEnabled) { - - var normalYShift = "note-btn btn btn-light btn-sm"; - var disabledYShift = "note-btn btn btn-light btn-sm disabled"; - var highlightYShift = ""; - var image = button.getElementsByTagName("span")[0]; - if (isEnabled) { - image.className = normalYShift; - if (!button.isHelp) { - button.onclick = function () { - if (this.onmouseout) { - this.onmouseout(); - } - doClick(this); - return false; - } - } - } - else { - image.className = disabledYShift; - button.onmouseover = button.onmouseout = button.onclick = function () { - }; - } - } - - function bindCommand(method) { - if (typeof method === "string") - method = commandManager[method]; - return function () { - method.apply(commandManager, arguments); - } - } - - function makeSpritedButtonRow() { - - var buttonBar = panels.buttonBar; - - var normalYShift = "note-btn btn btn-light btn-sm"; - var disabledYShift = "note-btn btn btn-light btn-sm disabled"; - var highlightYShift = ""; - - var buttonRow = document.createElement("ul"); - buttonRow.style.padding = "0px 10px"; - buttonRow.id = "wmd-button-row" + postfix; - buttonRow.className = 'wmd-button-row'; - buttonRow = buttonBar.appendChild(buttonRow); - var xPosition = 0; - var makeButton = function (id, title, XClass, textOp) { - var button = document.createElement("li"); - button.className = "wmd-button"; - var buttonImage = document.createElement("span"); - var buttonIcon = document.createElement("i"); - buttonImage.className = "note-btn btn btn-light btn-sm"; - buttonIcon.className = XClass; - buttonImage.appendChild(buttonIcon); - button.id = id + postfix; - button.appendChild(buttonImage); - button.title = title; - if (textOp) - button.textOp = textOp; - setupButton(button, true); - buttonRow.appendChild(button); - return button; - }; - var makeSpacer = function (num) { - var spacer = document.createElement("li"); - spacer.className = "wmd-spacer wmd-spacer" + num; - spacer.id = "wmd-spacer" + num + postfix; - buttonRow.appendChild(spacer); - } - - buttons.bold = makeButton("wmd-bold-button", getString("bold"), "fa fa-bold", bindCommand("doBold")); - buttons.italic = makeButton("wmd-italic-button", getString("italic"), "fa fa-italic", bindCommand("doItalic")); - buttons.heading = makeButton("wmd-heading-button", getString("heading"), "fa fa-header", bindCommand("doHeading")); - buttons.strikethrough = makeButton("wmd-strikethrough-button", getString("strikethrough"), "fa fa-strikethrough", bindCommand("doStrikethrough")); - //makeSpacer(1); - buttons.olist = makeButton("wmd-olist-button", getString("olist"), "fa fa-list-ol", bindCommand(function (chunk, postProcessing) { - this.doList(chunk, postProcessing, true); - })); - buttons.ulist = makeButton("wmd-ulist-button", getString("ulist"), "fa fa-list-ul", bindCommand(function (chunk, postProcessing) { - this.doList(chunk, postProcessing, false); - })); - buttons.quote = makeButton("wmd-quote-button", getString("quote"), "fa fa-quote-right", bindCommand("doBlockquote")); - buttons.code = makeButton("wmd-code-button", getString("code"), "fa fa-code", bindCommand("doCode")); - buttons.table = makeButton("wmd-table-button", getString("table"), "fa fa-table", bindCommand("doTable")); - //makeSpacer(2); - buttons.link = makeButton("wmd-link-button", getString("link"), "fa fa-link", bindCommand(function (chunk, postProcessing) { - return this.doLinkOrImage(chunk, postProcessing, false); - })); - buttons.image = makeButton("wmd-image-button", getString("image"), "fa fa-image", bindCommand(function (chunk, postProcessing) { - return this.doLinkOrImage(chunk, postProcessing, true); - })); - buttons.hr = makeButton("wmd-hr-button", getString("hr"), "fa fa-ellipsis-h", bindCommand("doHorizontalRule")); - buttons.readmore = makeButton("wmd-readmore-button", getString("readmore"), "fa fa-arrow-right", bindCommand("doReadMore")); - //makeSpacer(3); - buttons.undo = makeButton("wmd-undo-button", getString("undo"), "fa fa-undo", null); - buttons.undo.execute = function (manager) { - if (manager) manager.undo(); - }; - - var redoTitle = /win/.test(nav.platform.toLowerCase()) ? - getString("redo") : - getString("redomac"); // mac and other non-Windows platforms - - buttons.redo = makeButton("wmd-redo-button", redoTitle, "fa fa-repeat", null); - buttons.redo.execute = function (manager) { - if (manager) manager.redo(); - }; - - if (helpOptions) { - var helpButton = document.createElement("li"); - var helpButtonImage = document.createElement("span"); - helpButton.appendChild(helpButtonImage); - helpButton.className = "wmd-button wmd-help-button"; - helpButton.id = "wmd-help-button" + postfix; - helpButton.XClass = "note-btn btn btn-light btn-sm"; - helpButton.isHelp = true; - helpButton.style.right = "0px"; - helpButton.title = getString("help"); - helpButton.onclick = helpOptions.handler; - - setupButton(helpButton, true); - buttonRow.appendChild(helpButton); - buttons.help = helpButton; - } - - setUndoRedoButtonStates(); - } - - function setUndoRedoButtonStates() { - if (undoManager) { - setupButton(buttons.undo, undoManager.canUndo()); - setupButton(buttons.redo, undoManager.canRedo()); - } - }; - - this.setUndoRedoButtonStates = setUndoRedoButtonStates; - - } - - function CommandManager(pluginHooks, getString) { - this.hooks = pluginHooks; - this.getString = getString; - } - - var commandProto = CommandManager.prototype; - - // The markdown symbols - 4 spaces = code, > = blockquote, etc. - commandProto.prefixes = "(?:\\s{4,}|\\s*>|\\s*-\\s+|\\s*\\d+\\.|=|\\+|-|_|\\*|#|\\s*\\[[^\n]]+\\]:)"; - - // Remove markdown symbols from the chunk selection. - commandProto.unwrap = function (chunk) { - var txt = new re("([^\\n])\\n(?!(\\n|" + this.prefixes + "))", "g"); - chunk.selection = chunk.selection.replace(txt, "$1 $2"); - }; - - commandProto.wrap = function (chunk, len) { - this.unwrap(chunk); - var regex = new re("(.{1," + len + "})( +|$\\n?)", "gm"), - that = this; - - chunk.selection = chunk.selection.replace(regex, function (line, marked) { - if (new re("^" + that.prefixes, "").test(line)) { - return line; - } - return marked + "\n"; - }); - - chunk.selection = chunk.selection.replace(/\s+$/, ""); - }; - - commandProto.doBold = function (chunk, postProcessing) { - return this.doBorI(chunk, postProcessing, 2, this.getString("boldexample")); - }; - - commandProto.doItalic = function (chunk, postProcessing) { - return this.doBorI(chunk, postProcessing, 1, this.getString("italicexample")); - }; - - // chunk: The selected region that will be enclosed with */** - // nStars: 1 for italics, 2 for bold - // insertText: If you just click the button without highlighting text, this gets inserted - commandProto.doBorI = function (chunk, postProcessing, nStars, insertText) { - - // Get rid of whitespace and fixup newlines. - chunk.trimWhitespace(); - chunk.selection = chunk.selection.replace(/\n{2,}/g, "\n"); - - // Look for stars before and after. Is the chunk already marked up? - // note that these regex matches cannot fail - var starsBefore = /(\**$)/.exec(chunk.before)[0]; - var starsAfter = /(^\**)/.exec(chunk.after)[0]; - - var prevStars = Math.min(starsBefore.length, starsAfter.length); - - // Remove stars if we have to since the button acts as a toggle. - if ((prevStars >= nStars) && (prevStars != 2 || nStars != 1)) { - chunk.before = chunk.before.replace(re("[*]{" + nStars + "}$", ""), ""); - chunk.after = chunk.after.replace(re("^[*]{" + nStars + "}", ""), ""); - } - else if (!chunk.selection && starsAfter) { - // It's not really clear why this code is necessary. It just moves - // some arbitrary stuff around. - chunk.after = chunk.after.replace(/^([*_]*)/, ""); - chunk.before = chunk.before.replace(/(\s?)$/, ""); - var whitespace = re.$1; - chunk.before = chunk.before + starsAfter + whitespace; - } - else { - - // In most cases, if you don't have any selected text and click the button - // you'll get a selected, marked up region with the default text inserted. - if (!chunk.selection && !starsAfter) { - chunk.selection = insertText; - } - - // Add the true markup. - var markup = nStars <= 1 ? "*" : "**"; // shouldn't the test be = ? - chunk.before = chunk.before + markup; - chunk.after = markup + chunk.after; - } - - return; - }; - - commandProto.stripLinkDefs = function (text, defsToAdd) { - - text = text.replace(/^[ ]{0,3}\[(\d+)\]:[ \t]*\n?[ \t]*?[ \t]*\n?[ \t]*(?:(\n*)["(](.+?)[")][ \t]*)?(?:\n+|$)/gm, - function (totalMatch, id, link, newlines, title) { - defsToAdd[id] = totalMatch.replace(/\s*$/, ""); - if (newlines) { - // Strip the title and return that separately. - defsToAdd[id] = totalMatch.replace(/["(](.+?)[")]$/, ""); - return newlines + title; - } - return ""; - }); - - return text; - }; - - commandProto.addLinkDef = function (chunk, linkDef) { - - var refNumber = 0; // The current reference number - var defsToAdd = {}; // - // Start with a clean slate by removing all previous link definitions. - chunk.before = this.stripLinkDefs(chunk.before, defsToAdd); - chunk.selection = this.stripLinkDefs(chunk.selection, defsToAdd); - chunk.after = this.stripLinkDefs(chunk.after, defsToAdd); - - var defs = ""; - var regex = /(\[)((?:\[[^\]]*\]|[^\[\]])*)(\][ ]?(?:\n[ ]*)?\[)(\d+)(\])/g; - - var addDefNumber = function (def) { - refNumber++; - def = def.replace(/^[ ]{0,3}\[(\d+)\]:/, " [" + refNumber + "]:"); - defs += "\n" + def; - }; - - // note that - // a) the recursive call to getLink cannot go infinite, because by definition - // of regex, inner is always a proper substring of wholeMatch, and - // b) more than one level of nesting is neither supported by the regex - // nor making a lot of sense (the only use case for nesting is a linked image) - var getLink = function (wholeMatch, before, inner, afterInner, id, end) { - inner = inner.replace(regex, getLink); - if (defsToAdd[id]) { - addDefNumber(defsToAdd[id]); - return before + inner + afterInner + refNumber + end; - } - return wholeMatch; - }; - - chunk.before = chunk.before.replace(regex, getLink); - - if (linkDef) { - addDefNumber(linkDef); - } - else { - chunk.selection = chunk.selection.replace(regex, getLink); - } - - var refOut = refNumber; - - chunk.after = chunk.after.replace(regex, getLink); - - if (chunk.after) { - chunk.after = chunk.after.replace(/\n*$/, ""); - } - if (!chunk.after) { - chunk.selection = chunk.selection.replace(/\n*$/, ""); - } - - chunk.after += "\n\n" + defs; - - return refOut; - }; - - // takes the line as entered into the add link/as image dialog and makes - // sure the URL and the optinal title are "nice". - function properlyEncoded(linkdef) { - return linkdef.replace(/^\s*(.*?)(?:\s+"(.+)")?\s*$/, function (wholematch, link, title) { - link = link.replace(/\?.*$/, function (querypart) { - return querypart.replace(/\+/g, " "); // in the query string, a plus and a space are identical - }); - link = decodeURIComponent(link); // unencode first, to prevent double encoding - link = encodeURI(link).replace(/'/g, '%27').replace(/\(/g, '%28').replace(/\)/g, '%29'); - link = link.replace(/\?.*$/, function (querypart) { - return querypart.replace(/\+/g, "%2b"); // since we replaced plus with spaces in the query part, all pluses that now appear where originally encoded - }); - if (title) { - title = title.trim ? title.trim() : title.replace(/^\s*/, "").replace(/\s*$/, ""); - title = title.replace(/"/g, "quot;").replace(/\(/g, "(").replace(/\)/g, ")").replace(//g, ">"); - } - return title ? link + ' "' + title + '"' : link; - }); - } - - commandProto.doLinkOrImage = function (chunk, postProcessing, isImage) { - - chunk.trimWhitespace(); - chunk.findTags(/\s*!?\[/, /\][ ]?(?:\n[ ]*)?(\[.*?\])?/); - var background; - - if (chunk.endTag.length > 1 && chunk.startTag.length > 0) { - - chunk.startTag = chunk.startTag.replace(/!?\[/, ""); - chunk.endTag = ""; - this.addLinkDef(chunk, null); - - } - else { - - // We're moving start and end tag back into the selection, since (as we're in the else block) we're not - // *removing* a link, but *adding* one, so whatever findTags() found is now back to being part of the - // link text. linkEnteredCallback takes care of escaping any brackets. - chunk.selection = chunk.startTag + chunk.selection + chunk.endTag; - chunk.startTag = chunk.endTag = ""; - - if (/\n\n/.test(chunk.selection)) { - this.addLinkDef(chunk, null); - return; - } - var that = this; - // The function to be executed when you enter a link and press OK or Cancel. - // Marks up the link and adds the ref. - var linkEnteredCallback = function (link) { - - background.remove(); - - if (link !== null) { - // ( $1 - // [^\\] anything that's not a backslash - // (?:\\\\)* an even number (this includes zero) of backslashes - // ) - // (?= followed by - // [[\]] an opening or closing bracket - // ) - // - // In other words, a non-escaped bracket. These have to be escaped now to make sure they - // don't count as the end of the link or similar. - // Note that the actual bracket has to be a lookahead, because (in case of to subsequent brackets), - // the bracket in one match may be the "not a backslash" character in the next match, so it - // should not be consumed by the first match. - // The "prepend a space and finally remove it" steps makes sure there is a "not a backslash" at the - // start of the string, so this also works if the selection begins with a bracket. We cannot solve - // this by anchoring with ^, because in the case that the selection starts with two brackets, this - // would mean a zero-width match at the start. Since zero-width matches advance the string position, - // the first bracket could then not act as the "not a backslash" for the second. - chunk.selection = (" " + chunk.selection).replace(/([^\\](?:\\\\)*)(?=[[\]])/g, "$1\\").substr(1); - - var linkDef = " [999]: " + properlyEncoded(link); - - var num = that.addLinkDef(chunk, linkDef); - chunk.startTag = isImage ? "![" : "["; - chunk.endTag = "][" + num + "]"; - - if (!chunk.selection) { - if (isImage) { - chunk.selection = that.getString("imagedescription"); - } - else { - chunk.selection = that.getString("linkdescription"); - } - } - } - postProcessing(); - }; - - background = ui.createBackground(); - - if (isImage) { - if (!this.hooks.insertImageDialog(linkEnteredCallback)) - ui.prompt(this.getString("imagedialog"), imageDefaultText, linkEnteredCallback); - } - else { - ui.prompt(this.getString("linkdialog"), linkDefaultText, linkEnteredCallback); - } - return true; - } - }; - - // When making a list, hitting shift-enter will put your cursor on the next line - // at the current indent level. - commandProto.doAutoindent = function (chunk, postProcessing) { - - var commandMgr = this, - fakeSelection = false; - - chunk.before = chunk.before.replace(/(\n|^)[ ]{0,3}([*+-]|\d+[.])[ \t]*\n$/, "\n\n"); - chunk.before = chunk.before.replace(/(\n|^)[ ]{0,3}>[ \t]*\n$/, "\n\n"); - chunk.before = chunk.before.replace(/(\n|^)[ \t]+\n$/, "\n\n"); - - // There's no selection, end the cursor wasn't at the end of the line: - // The user wants to split the current list item / code line / blockquote line - // (for the latter it doesn't really matter) in two. Temporarily select the - // (rest of the) line to achieve this. - if (!chunk.selection && !/^[ \t]*(?:\n|$)/.test(chunk.after)) { - chunk.after = chunk.after.replace(/^[^\n]*/, function (wholeMatch) { - chunk.selection = wholeMatch; - return ""; - }); - fakeSelection = true; - } - - if (/(\n|^)[ ]{0,3}([*+-]|\d+[.])[ \t]+.*\n$/.test(chunk.before)) { - if (commandMgr.doList) { - commandMgr.doList(chunk); - } - } - if (/(\n|^)[ ]{0,3}>[ \t]+.*\n$/.test(chunk.before)) { - if (commandMgr.doBlockquote) { - commandMgr.doBlockquote(chunk); - } - } - if (/(\n|^)(\t|[ ]{4,}).*\n$/.test(chunk.before)) { - if (commandMgr.doCode) { - commandMgr.doCode(chunk); - } - } - - if (fakeSelection) { - chunk.after = chunk.selection + chunk.after; - chunk.selection = ""; - } - }; - - commandProto.doBlockquote = function (chunk, postProcessing) { - - chunk.selection = chunk.selection.replace(/^(\n*)([^\r]+?)(\n*)$/, - function (totalMatch, newlinesBefore, text, newlinesAfter) { - chunk.before += newlinesBefore; - chunk.after = newlinesAfter + chunk.after; - return text; - }); - - chunk.before = chunk.before.replace(/(>[ \t]*)$/, - function (totalMatch, blankLine) { - chunk.selection = blankLine + chunk.selection; - return ""; - }); - - chunk.selection = chunk.selection.replace(/^(\s|>)+$/, ""); - chunk.selection = chunk.selection || this.getString("quoteexample"); - - // The original code uses a regular expression to find out how much of the - // text *directly before* the selection already was a blockquote: - - /* - if (chunk.before) { - chunk.before = chunk.before.replace(/\n?$/, "\n"); - } - chunk.before = chunk.before.replace(/(((\n|^)(\n[ \t]*)*>(.+\n)*.*)+(\n[ \t]*)*$)/, - function (totalMatch) { - chunk.startTag = totalMatch; - return ""; - }); - */ - - // This comes down to: - // Go backwards as many lines a possible, such that each line - // a) starts with ">", or - // b) is almost empty, except for whitespace, or - // c) is preceeded by an unbroken chain of non-empty lines - // leading up to a line that starts with ">" and at least one more character - // and in addition - // d) at least one line fulfills a) - // - // Since this is essentially a backwards-moving regex, it's susceptible to - // catstrophic backtracking and can cause the browser to hang; - // see e.g. http://meta.stackoverflow.com/questions/9807. - // - // Hence we replaced this by a simple state machine that just goes through the - // lines and checks for a), b), and c). - - var match = "", - leftOver = "", - line; - if (chunk.before) { - var lines = chunk.before.replace(/\n$/, "").split("\n"); - var inChain = false; - for (var i = 0; i < lines.length; i++) { - var good = false; - line = lines[i]; - inChain = inChain && line.length > 0; // c) any non-empty line continues the chain - if (/^>/.test(line)) { // a) - good = true; - if (!inChain && line.length > 1) // c) any line that starts with ">" and has at least one more character starts the chain - inChain = true; - } else if (/^[ \t]*$/.test(line)) { // b) - good = true; - } else { - good = inChain; // c) the line is not empty and does not start with ">", so it matches if and only if we're in the chain - } - if (good) { - match += line + "\n"; - } else { - leftOver += match + line; - match = "\n"; - } - } - if (!/(^|\n)>/.test(match)) { // d) - leftOver += match; - match = ""; - } - } - - chunk.startTag = match; - chunk.before = leftOver; - - // end of change - - if (chunk.after) { - chunk.after = chunk.after.replace(/^\n?/, "\n"); - } - - chunk.after = chunk.after.replace(/^(((\n|^)(\n[ \t]*)*>(.+\n)*.*)+(\n[ \t]*)*)/, - function (totalMatch) { - chunk.endTag = totalMatch; - return ""; - } - ); - - var replaceBlanksInTags = function (useBracket) { - - var replacement = useBracket ? "> " : ""; - - if (chunk.startTag) { - chunk.startTag = chunk.startTag.replace(/\n((>|\s)*)\n$/, - function (totalMatch, markdown) { - return "\n" + markdown.replace(/^[ ]{0,3}>?[ \t]*$/gm, replacement) + "\n"; - }); - } - if (chunk.endTag) { - chunk.endTag = chunk.endTag.replace(/^\n((>|\s)*)\n/, - function (totalMatch, markdown) { - return "\n" + markdown.replace(/^[ ]{0,3}>?[ \t]*$/gm, replacement) + "\n"; - }); - } - }; - - if (/^(?![ ]{0,3}>)/m.test(chunk.selection)) { - this.wrap(chunk, SETTINGS.lineLength - 2); - chunk.selection = chunk.selection.replace(/^/gm, "> "); - replaceBlanksInTags(true); - chunk.skipLines(); - } else { - chunk.selection = chunk.selection.replace(/^[ ]{0,3}> ?/gm, ""); - this.unwrap(chunk); - replaceBlanksInTags(false); - - if (!/^(\n|^)[ ]{0,3}>/.test(chunk.selection) && chunk.startTag) { - chunk.startTag = chunk.startTag.replace(/\n{0,2}$/, "\n\n"); - } - - if (!/(\n|^)[ ]{0,3}>.*$/.test(chunk.selection) && chunk.endTag) { - chunk.endTag = chunk.endTag.replace(/^\n{0,2}/, "\n\n"); - } - } - - chunk.selection = this.hooks.postBlockquoteCreation(chunk.selection); - - if (!/\n/.test(chunk.selection)) { - chunk.selection = chunk.selection.replace(/^(> *)/, - function (wholeMatch, blanks) { - chunk.startTag += blanks; - return ""; - }); - } - }; - - commandProto.doCode = function (chunk, postProcessing) { - - var hasTextBefore = /\S[ ]*$/.test(chunk.before); - var hasTextAfter = /^[ ]*\S/.test(chunk.after); - - // Use 'four space' markdown if the selection is on its own - // line or is multiline. - if ((!hasTextAfter && !hasTextBefore) || /\n/.test(chunk.selection)) { - - chunk.before = chunk.before.replace(/[ ]{4}$/, - function (totalMatch) { - chunk.selection = totalMatch + chunk.selection; - return ""; - }); - - var nLinesBack = 1; - var nLinesForward = 1; - - if (/(\n|^)(\t|[ ]{4,}).*\n$/.test(chunk.before)) { - nLinesBack = 0; - } - if (/^\n(\t|[ ]{4,})/.test(chunk.after)) { - nLinesForward = 0; - } - - chunk.skipLines(nLinesBack, nLinesForward); - - if (!chunk.selection) { - chunk.startTag = " "; - chunk.selection = this.getString("codeexample"); - } - else { - if (/^[ ]{0,3}\S/m.test(chunk.selection)) { - if (/\n/.test(chunk.selection)) - chunk.selection = chunk.selection.replace(/^/gm, " "); - else // if it's not multiline, do not select the four added spaces; this is more consistent with the doList behavior - chunk.before += " "; - } - else { - chunk.selection = chunk.selection.replace(/^(?:[ ]{4}|[ ]{0,3}\t)/gm, ""); - } - } - } - else { - // Use backticks (`) to delimit the code block. - - chunk.trimWhitespace(); - chunk.findTags(/`/, /`/); - - if (!chunk.startTag && !chunk.endTag) { - chunk.startTag = chunk.endTag = "`"; - if (!chunk.selection) { - chunk.selection = this.getString("codeexample"); - } - } - else if (chunk.endTag && !chunk.startTag) { - chunk.before += chunk.endTag; - chunk.endTag = ""; - } - else { - chunk.startTag = chunk.endTag = ""; - } - } - }; - - commandProto.doList = function (chunk, postProcessing, isNumberedList) { - - // These are identical except at the very beginning and end. - // Should probably use the regex extension function to make this clearer. - var previousItemsRegex = /(\n|^)(([ ]{0,3}([*+-]|\d+[.])[ \t]+.*)(\n.+|\n{2,}([*+-].*|\d+[.])[ \t]+.*|\n{2,}[ \t]+\S.*)*)\n*$/; - var nextItemsRegex = /^\n*(([ ]{0,3}([*+-]|\d+[.])[ \t]+.*)(\n.+|\n{2,}([*+-].*|\d+[.])[ \t]+.*|\n{2,}[ \t]+\S.*)*)\n*/; - - // The default bullet is a dash but others are possible. - // This has nothing to do with the particular HTML bullet, - // it's just a markdown bullet. - var bullet = "-"; - - // The number in a numbered list. - var num = 1; - - // Get the item prefix - e.g. " 1. " for a numbered list, " - " for a bulleted list. - var getItemPrefix = function () { - var prefix; - if (isNumberedList) { - prefix = " " + num + ". "; - num++; - } - else { - prefix = " " + bullet + " "; - } - return prefix; - }; - - // Fixes the prefixes of the other list items. - var getPrefixedItem = function (itemText) { - - // The numbering flag is unset when called by autoindent. - if (isNumberedList === undefined) { - isNumberedList = /^\s*\d/.test(itemText); - } - - // Renumber/bullet the list element. - itemText = itemText.replace(/^[ ]{0,3}([*+-]|\d+[.])\s/gm, - function (_) { - return getItemPrefix(); - }); - - return itemText; - }; - - chunk.findTags(/(\n|^)*[ ]{0,3}([*+-]|\d+[.])\s+/, null); - - if (chunk.before && !/\n$/.test(chunk.before) && !/^\n/.test(chunk.startTag)) { - chunk.before += chunk.startTag; - chunk.startTag = ""; - } - - if (chunk.startTag) { - - var hasDigits = /\d+[.]/.test(chunk.startTag); - chunk.startTag = ""; - chunk.selection = chunk.selection.replace(/\n[ ]{4}/g, "\n"); - this.unwrap(chunk); - chunk.skipLines(); - - if (hasDigits) { - // Have to renumber the bullet points if this is a numbered list. - chunk.after = chunk.after.replace(nextItemsRegex, getPrefixedItem); - } - if (isNumberedList == hasDigits) { - return; - } - } - - var nLinesUp = 1; - - chunk.before = chunk.before.replace(previousItemsRegex, - function (itemText) { - if (/^\s*([*+-])/.test(itemText)) { - bullet = re.$1; - } - nLinesUp = /[^\n]\n\n[^\n]/.test(itemText) ? 1 : 0; - return getPrefixedItem(itemText); - }); - - if (!chunk.selection) { - chunk.selection = this.getString("litem"); - } - - var prefix = getItemPrefix(); - - var nLinesDown = 1; - - chunk.after = chunk.after.replace(nextItemsRegex, - function (itemText) { - nLinesDown = /[^\n]\n\n[^\n]/.test(itemText) ? 1 : 0; - return getPrefixedItem(itemText); - }); - - chunk.trimWhitespace(true); - chunk.skipLines(nLinesUp, nLinesDown, true); - chunk.startTag = prefix; - var spaces = prefix.replace(/./g, " "); - this.wrap(chunk, SETTINGS.lineLength - spaces.length); - chunk.selection = chunk.selection.replace(/\n/g, "\n" + spaces); - - }; - - commandProto.doHeading = function (chunk, postProcessing) { - - // Remove leading/trailing whitespace and reduce internal spaces to single spaces. - chunk.selection = chunk.selection.replace(/\s+/g, " "); - chunk.selection = chunk.selection.replace(/(^\s+|\s+$)/g, ""); - - // If we clicked the button with no selected text, we just - // make a level 2 hash header around some default text. - if (!chunk.selection) { - chunk.startTag = "## "; - chunk.selection = this.getString("headingexample"); - chunk.endTag = " ##"; - return; - } - - var headerLevel = 0; // The existing header level of the selected text. - - // Remove any existing hash heading markdown and save the header level. - chunk.findTags(/#+[ ]*/, /[ ]*#+/); - if (/#+/.test(chunk.startTag)) { - headerLevel = re.lastMatch.length; - } - chunk.startTag = chunk.endTag = ""; - - // Try to get the current header level by looking for - and = in the line - // below the selection. - chunk.findTags(null, /\s?(-+|=+)/); - if (/=+/.test(chunk.endTag)) { - headerLevel = 1; - } - if (/-+/.test(chunk.endTag)) { - headerLevel = 2; - } - - // Skip to the next line so we can create the header markdown. - chunk.startTag = chunk.endTag = ""; - chunk.skipLines(1, 1); - - // We make a level 2 header if there is no current header. - // If there is a header level, we substract one from the header level. - // If it's already a level 1 header, it's removed. - var headerLevelToCreate = headerLevel == 0 ? 2 : headerLevel - 1; - - if (headerLevelToCreate > 0) { - - // The button only creates level 1 and 2 underline headers. - // Why not have it iterate over hash header levels? Wouldn't that be easier and cleaner? - var headerChar = headerLevelToCreate >= 2 ? "-" : "="; - var len = chunk.selection.length; - if (len > SETTINGS.lineLength) { - len = SETTINGS.lineLength; - } - chunk.endTag = "\n"; - while (len--) { - chunk.endTag += headerChar; - } - } - }; - - commandProto.doHorizontalRule = function (chunk, postProcessing) { - chunk.startTag = "----------\n"; - chunk.selection = ""; - chunk.skipLines(2, 1, true); - } - - commandProto.doReadMore = function (chunk, postProcessing) { - chunk.startTag = ""; - chunk.selection = ""; - chunk.skipLines(0, 1, true); - } - - commandProto.doStrikethrough = function (chunk, postProcessing) { - - // Get rid of whitespace and fixup newlines. - chunk.trimWhitespace(); - chunk.selection = chunk.selection.replace(/\n{2,}/g, "\n"); - - // Look for stars before and after. Is the chunk already marked up? - // note that these regex matches cannot fail - var starsBefore = /(~*$)/.exec(chunk.before)[0]; - var starsAfter = /(^~*)/.exec(chunk.after)[0]; - - var prevStars = Math.min(starsBefore.length, starsAfter.length); - - var nStars = 2; - - // Remove stars if we have to since the button acts as a toggle. - if ((prevStars >= nStars) && (prevStars != 2 || nStars != 1)) { - chunk.before = chunk.before.replace(re("[~]{" + nStars + "}$", ""), ""); - chunk.after = chunk.after.replace(re("^[~]{" + nStars + "}", ""), ""); - } else if (!chunk.selection && starsAfter) { - // It's not really clear why this code is necessary. It just moves - // some arbitrary stuff around. - chunk.after = chunk.after.replace(/^(~*)/, ""); - chunk.before = chunk.before.replace(/(\s?)$/, ""); - var whitespace = re.$1; - chunk.before = chunk.before + starsAfter + whitespace; - } else { - - // In most cases, if you don't have any selected text and click the button - // you'll get a selected, marked up region with the default text inserted. - if (!chunk.selection && !starsAfter) { - chunk.selection = this.getString("strikethroughexample"); - } - - // Add the true markup. - var markup = "~~"; // shouldn't the test be = ? - chunk.before = chunk.before + markup; - chunk.after = markup + chunk.after; - } - - return; - }; - - commandProto.doTable = function (chunk) { - // Credit: https://github.com/fcrespo82/atom-markdown-table-formatter - - var keepFirstAndLastPipes = true, - /* - ( # header capture - (?: - (?:[^\n]*?\|[^\n]*) # line w/ at least one pipe - \ * # maybe trailing whitespace - )? # maybe header - (?:\n|^) # newline - ) - ( # format capture - (?: - \|\ *:?-+:?\ * # format starting w/pipe - |\|?(?:\ *:?-+:?\ *\|)+ # or separated by pipe - ) - (?:\ *:?-+:?\ *)? # maybe w/o trailing pipe - \ * # maybe trailing whitespace - \n # newline - ) - ( # body capture - (?: - (?:[^\n]*?\|[^\n]*) # line w/ at least one pipe - \ * # maybe trailing whitespace - (?:\n|$) # newline - )+ # at least one - ) - */ - regex = /((?:(?:[^\n]*?\|[^\n]*) *)?(?:\r?\n|^))((?:\| *:?-+:? *|\|?(?: *:?-+:? *\|)+)(?: *:?-+:? *)? *\r?\n)((?:(?:[^\n]*?\|[^\n]*) *(?:\r?\n|$))+)/; - - - function padding(len, str) { - var result = ''; - str = str || ' '; - len = Math.floor(len); - for (var i = 0; i < len; i++) { - result += str; - } - return result; - } - - function stripTailPipes(str) { - return str.trim().replace(/(^\||\|$)/g, ""); - } - - function splitCells(str) { - return str.split('|'); - } - - function addTailPipes(str) { - if (keepFirstAndLastPipes) { - return "|" + str + "|"; - } else { - return str; - } - } - - function joinCells(arr) { - return arr.join('|'); - } - - function formatTable(text, appendNewline) { - var i, j, len1, ref1, ref2, ref3, k, len2, results, formatline, headerline, just, formatrow, data, line, lines, justify, cell, cells, first, last, ends, columns, content, widths, formatted, front, back; - formatline = text[2].trim(); - headerline = text[1].trim(); - ref1 = headerline.length === 0 ? [0, text[3]] : [1, text[1] + text[3]], formatrow = ref1[0], data = ref1[1]; - lines = data.trim().split('\n'); - justify = []; - ref2 = splitCells(stripTailPipes(formatline)); - for (j = 0, len1 = ref2.length; j < len1; j++) { - cell = ref2[j]; - ref3 = cell.trim(), first = ref3[0], last = ref3[ref3.length - 1]; - switch ((ends = (first ? first : ':') + (last ? last : ''))) { - case '::': - case '-:': - case ':-': - justify.push(ends); - break; - default: - justify.push('--'); - } - } - columns = justify.length; - content = []; - for (j = 0, len1 = lines.length; j < len1; j++) { - line = lines[j]; - cells = splitCells(stripTailPipes(line)); - cells[columns - 1] = joinCells(cells.slice(columns - 1)); - results = []; - for (k = 0, len2 = cells.length; k < len2; k++) { - cell = cells[k]; - results.push(padding(' ') + ((ref2 = cell ? typeof cell.trim === "function" ? cell.trim() : void 0 : void 0) ? ref2 : '') + padding(' ')); - } - content.push(results); - } - widths = []; - for (i = j = 0, ref2 = columns - 1; 0 <= ref2 ? j <= ref2 : j >= ref2; i = 0 <= ref2 ? ++j : --j) { - results = []; - for (k = 0, len1 = content.length; k < len1; k++) { - cells = content[k]; - results.push(cells[i].length); - } - widths.push(Math.max.apply(Math, [2].concat(results))); - } - just = function (string, col) { - var back, front, length; - length = widths[col] - string.length; - switch (justify[col]) { - case '::': - front = padding[0], back = padding[1]; - return padding(length / 2) + string + padding((length + 1) / 2); - case '-:': - return padding(length) + string; - default: - return string + padding(length); - } - }; - formatted = []; - for (j = 0, len1 = content.length; j < len1; j++) { - cells = content[j]; - results = []; - for (i = k = 0, ref2 = columns - 1; 0 <= ref2 ? k <= ref2 : k >= ref2; i = 0 <= ref2 ? ++k : --k) { - results.push(just(cells[i], i)); - } - formatted.push(addTailPipes(joinCells(results))); - } - formatline = addTailPipes(joinCells((function () { - var j, ref2, ref3, results; - results = []; - for (i = j = 0, ref2 = columns - 1; 0 <= ref2 ? j <= ref2 : j >= ref2; i = 0 <= ref2 ? ++j : --j) { - ref3 = justify[i], front = ref3[0], back = ref3[1]; - results.push(front + padding(widths[i] - 2, '-') + back); - } - return results; - })())); - formatted.splice(formatrow, 0, formatline); - var result = (headerline.length === 0 && text[1] !== '' ? '\n' : '') + formatted.join('\n'); - if (appendNewline !== false) { - result += '\n' - } - return result; - } - - if (chunk.before.slice(-1) !== '\n') { - chunk.before += '\n'; - } - var match = chunk.selection.match(regex); - if (match) { - chunk.selection = formatTable(match, chunk.selection.slice(-1) === '\n'); - } else { - var table = chunk.selection + '|\n-|-\n|'; - match = table.match(regex); - if (!match || match[0].slice(0, table.length) !== table) { - return; - } - table = formatTable(match); - var selectionOffset = keepFirstAndLastPipes ? 1 : 0; - var pipePos = table.indexOf('|', selectionOffset); - chunk.before += table.slice(0, selectionOffset); - chunk.selection = table.slice(selectionOffset, pipePos); - chunk.after = table.slice(pipePos) + chunk.after; - } - }; - - -})(); \ No newline at end of file diff --git a/system/admin/editor/js/Markdown.Extra.js b/system/admin/editor/js/Markdown.Extra.js deleted file mode 100644 index d3b859a5..00000000 --- a/system/admin/editor/js/Markdown.Extra.js +++ /dev/null @@ -1,874 +0,0 @@ -(function () { - // A quick way to make sure we're only keeping span-level tags when we need to. - // This isn't supposed to be foolproof. It's just a quick way to make sure we - // keep all span-level tags returned by a pagedown converter. It should allow - // all span-level tags through, with or without attributes. - var inlineTags = new RegExp(['^(<\\/?(a|abbr|acronym|applet|area|b|basefont|', - 'bdo|big|button|cite|code|del|dfn|em|figcaption|', - 'font|i|iframe|img|input|ins|kbd|label|map|', - 'mark|meter|object|param|progress|q|ruby|rp|rt|s|', - 'samp|script|select|small|span|strike|strong|', - 'sub|sup|textarea|time|tt|u|var|wbr)[^>]*>|', - '<(br)\\s?\\/?>)$'].join(''), 'i'); - - /****************************************************************** - * Utility Functions * - *****************************************************************/ - - // patch for ie7 - if (!Array.indexOf) { - Array.prototype.indexOf = function(obj) { - for (var i = 0; i < this.length; i++) { - if (this[i] == obj) { - return i; - } - } - return -1; - }; - } - - function trim(str) { - return str.replace(/^\s+|\s+$/g, ''); - } - - function rtrim(str) { - return str.replace(/\s+$/g, ''); - } - - // Remove one level of indentation from text. Indent is 4 spaces. - function outdent(text) { - return text.replace(new RegExp('^(\\t|[ ]{1,4})', 'gm'), ''); - } - - function contains(str, substr) { - return str.indexOf(substr) != -1; - } - - // Sanitize html, removing tags that aren't in the whitelist - function sanitizeHtml(html, whitelist) { - return html.replace(/<[^>]*>?/gi, function(tag) { - return tag.match(whitelist) ? tag : ''; - }); - } - - // Merge two arrays, keeping only unique elements. - function union(x, y) { - var obj = {}; - for (var i = 0; i < x.length; i++) - obj[x[i]] = x[i]; - for (i = 0; i < y.length; i++) - obj[y[i]] = y[i]; - var res = []; - for (var k in obj) { - if (obj.hasOwnProperty(k)) - res.push(obj[k]); - } - return res; - } - - // JS regexes don't support \A or \Z, so we add sentinels, as Pagedown - // does. In this case, we add the ascii codes for start of text (STX) and - // end of text (ETX), an idea borrowed from: - // https://github.com/tanakahisateru/js-markdown-extra - function addAnchors(text) { - if(text.charAt(0) != '\x02') - text = '\x02' + text; - if(text.charAt(text.length - 1) != '\x03') - text = text + '\x03'; - return text; - } - - // Remove STX and ETX sentinels. - function removeAnchors(text) { - if(text.charAt(0) == '\x02') - text = text.substr(1); - if(text.charAt(text.length - 1) == '\x03') - text = text.substr(0, text.length - 1); - return text; - } - - // Convert markdown within an element, retaining only span-level tags - function convertSpans(text, extra) { - return sanitizeHtml(convertAll(text, extra), inlineTags); - } - - // Convert internal markdown using the stock pagedown converter - function convertAll(text, extra) { - var result = extra.blockGamutHookCallback(text); - // We need to perform these operations since we skip the steps in the converter - result = unescapeSpecialChars(result); - result = result.replace(/~D/g, "$$").replace(/~T/g, "~"); - result = extra.previousPostConversion(result); - return result; - } - - // Convert escaped special characters - function processEscapesStep1(text) { - // Markdown extra adds two escapable characters, `:` and `|` - return text.replace(/\\\|/g, '~I').replace(/\\:/g, '~i'); - } - function processEscapesStep2(text) { - return text.replace(/~I/g, '|').replace(/~i/g, ':'); - } - - // Duplicated from PageDown converter - function unescapeSpecialChars(text) { - // Swap back in all the special characters we've hidden. - text = text.replace(/~E(\d+)E/g, function(wholeMatch, m1) { - var charCodeToReplace = parseInt(m1); - return String.fromCharCode(charCodeToReplace); - }); - return text; - } - - function slugify(text) { - return text.toLowerCase() - .replace(/\s+/g, '-') // Replace spaces with - - .replace(/[^\w\-]+/g, '') // Remove all non-word chars - .replace(/\-\-+/g, '-') // Replace multiple - with single - - .replace(/^-+/, '') // Trim - from start of text - .replace(/-+$/, ''); // Trim - from end of text - } - - /***************************************************************************** - * Markdown.Extra * - ****************************************************************************/ - - Markdown.Extra = function() { - // For converting internal markdown (in tables for instance). - // This is necessary since these methods are meant to be called as - // preConversion hooks, and the Markdown converter passed to init() - // won't convert any markdown contained in the html tags we return. - this.converter = null; - - // Stores html blocks we generate in hooks so that - // they're not destroyed if the user is using a sanitizing converter - this.hashBlocks = []; - - // Stores footnotes - this.footnotes = {}; - this.usedFootnotes = []; - - // Special attribute blocks for fenced code blocks and headers enabled. - this.attributeBlocks = false; - - // Fenced code block options - this.googleCodePrettify = false; - this.highlightJs = false; - - // Table options - this.tableClass = ''; - - this.tabWidth = 4; - }; - - Markdown.Extra.init = function(converter, options) { - // Each call to init creates a new instance of Markdown.Extra so it's - // safe to have multiple converters, with different options, on a single page - var extra = new Markdown.Extra(); - var postNormalizationTransformations = []; - var preBlockGamutTransformations = []; - var postSpanGamutTransformations = []; - var postConversionTransformations = ["unHashExtraBlocks"]; - - options = options || {}; - options.extensions = options.extensions || ["all"]; - if (contains(options.extensions, "all")) { - options.extensions = ["tables", "fenced_code_gfm", "def_list", "attr_list", "footnotes", "smartypants", "strikethrough", "newlines"]; - } - preBlockGamutTransformations.push("wrapHeaders"); - if (contains(options.extensions, "attr_list")) { - postNormalizationTransformations.push("hashFcbAttributeBlocks"); - preBlockGamutTransformations.push("hashHeaderAttributeBlocks"); - postConversionTransformations.push("applyAttributeBlocks"); - extra.attributeBlocks = true; - } - if (contains(options.extensions, "fenced_code_gfm")) { - // This step will convert fcb inside list items and blockquotes - preBlockGamutTransformations.push("fencedCodeBlocks"); - // This extra step is to prevent html blocks hashing and link definition/footnotes stripping inside fcb - postNormalizationTransformations.push("fencedCodeBlocks"); - } - if (contains(options.extensions, "tables")) { - preBlockGamutTransformations.push("tables"); - } - if (contains(options.extensions, "def_list")) { - preBlockGamutTransformations.push("definitionLists"); - } - if (contains(options.extensions, "footnotes")) { - postNormalizationTransformations.push("stripFootnoteDefinitions"); - preBlockGamutTransformations.push("doFootnotes"); - postConversionTransformations.push("printFootnotes"); - } - if (contains(options.extensions, "smartypants")) { - postConversionTransformations.push("runSmartyPants"); - } - if (contains(options.extensions, "strikethrough")) { - postSpanGamutTransformations.push("strikethrough"); - } - if (contains(options.extensions, "newlines")) { - postSpanGamutTransformations.push("newlines"); - } - - converter.hooks.chain("postNormalization", function(text) { - return extra.doTransform(postNormalizationTransformations, text) + '\n'; - }); - - converter.hooks.chain("preBlockGamut", function(text, blockGamutHookCallback) { - // Keep a reference to the block gamut callback to run recursively - extra.blockGamutHookCallback = blockGamutHookCallback; - text = processEscapesStep1(text); - text = extra.doTransform(preBlockGamutTransformations, text) + '\n'; - text = processEscapesStep2(text); - return text; - }); - - converter.hooks.chain("postSpanGamut", function(text) { - return extra.doTransform(postSpanGamutTransformations, text); - }); - - // Keep a reference to the hook chain running before doPostConversion to apply on hashed extra blocks - extra.previousPostConversion = converter.hooks.postConversion; - converter.hooks.chain("postConversion", function(text) { - text = extra.doTransform(postConversionTransformations, text); - // Clear state vars that may use unnecessary memory - extra.hashBlocks = []; - extra.footnotes = {}; - extra.usedFootnotes = []; - return text; - }); - - if ("highlighter" in options) { - extra.googleCodePrettify = options.highlighter === 'prettify'; - extra.highlightJs = options.highlighter === 'highlight'; - } - - if ("table_class" in options) { - extra.tableClass = options.table_class; - } - - extra.converter = converter; - - // Caller usually won't need this, but it's handy for testing. - return extra; - }; - - // Do transformations - Markdown.Extra.prototype.doTransform = function(transformations, text) { - for(var i = 0; i < transformations.length; i++) - text = this[transformations[i]](text); - return text; - }; - - // Return a placeholder containing a key, which is the block's index in the - // hashBlocks array. We wrap our output in a

                                tag here so Pagedown won't. - Markdown.Extra.prototype.hashExtraBlock = function(block) { - return '\n

                                ~X' + (this.hashBlocks.push(block) - 1) + 'X

                                \n'; - }; - Markdown.Extra.prototype.hashExtraInline = function(block) { - return '~X' + (this.hashBlocks.push(block) - 1) + 'X'; - }; - - // Replace placeholder blocks in `text` with their corresponding - // html blocks in the hashBlocks array. - Markdown.Extra.prototype.unHashExtraBlocks = function(text) { - var self = this; - function recursiveUnHash() { - var hasHash = false; - text = text.replace(/(?:

                                )?~X(\d+)X(?:<\/p>)?/g, function(wholeMatch, m1) { - hasHash = true; - var key = parseInt(m1, 10); - return self.hashBlocks[key]; - }); - if(hasHash === true) { - recursiveUnHash(); - } - } - recursiveUnHash(); - return text; - }; - - // Wrap headers to make sure they won't be in def lists - Markdown.Extra.prototype.wrapHeaders = function(text) { - function wrap(text) { - return '\n' + text + '\n'; - } - text = text.replace(/^.+[ \t]*\n=+[ \t]*\n+/gm, wrap); - text = text.replace(/^.+[ \t]*\n-+[ \t]*\n+/gm, wrap); - text = text.replace(/^\#{1,6}[ \t]*.+?[ \t]*\#*\n+/gm, wrap); - return text; - }; - - - /****************************************************************** - * Attribute Blocks * - *****************************************************************/ - - // TODO: use sentinels. Should we just add/remove them in doConversion? - // TODO: better matches for id / class attributes - var attrBlock = "\\{[ \\t]*((?:[#.][-_:a-zA-Z0-9]+[ \\t]*)+)\\}"; - var hdrAttributesA = new RegExp("^(#{1,6}.*#{0,6})[ \\t]+" + attrBlock + "[ \\t]*(?:\\n|0x03)", "gm"); - var hdrAttributesB = new RegExp("^(.*)[ \\t]+" + attrBlock + "[ \\t]*\\n" + - "(?=[\\-|=]+\\s*(?:\\n|0x03))", "gm"); // underline lookahead - var fcbAttributes = new RegExp("^(```[^`\\n]*)[ \\t]+" + attrBlock + "[ \\t]*\\n" + - "(?=([\\s\\S]*?)\\n```[ \\t]*(\\n|0x03))", "gm"); - - // Extract headers attribute blocks, move them above the element they will be - // applied to, and hash them for later. - Markdown.Extra.prototype.hashHeaderAttributeBlocks = function(text) { - - var self = this; - function attributeCallback(wholeMatch, pre, attr) { - return '

                                ~XX' + (self.hashBlocks.push(attr) - 1) + 'XX

                                \n' + pre + "\n"; - } - - text = text.replace(hdrAttributesA, attributeCallback); // ## headers - text = text.replace(hdrAttributesB, attributeCallback); // underline headers - return text; - }; - - // Extract FCB attribute blocks, move them above the element they will be - // applied to, and hash them for later. - Markdown.Extra.prototype.hashFcbAttributeBlocks = function(text) { - // TODO: use sentinels. Should we just add/remove them in doConversion? - // TODO: better matches for id / class attributes - - var self = this; - function attributeCallback(wholeMatch, pre, attr) { - return '

                                ~XX' + (self.hashBlocks.push(attr) - 1) + 'XX

                                \n' + pre + "\n"; - } - - return text.replace(fcbAttributes, attributeCallback); - }; - - Markdown.Extra.prototype.applyAttributeBlocks = function(text) { - var self = this; - var blockRe = new RegExp('

                                ~XX(\\d+)XX

                                [\\s]*' + - '(?:<(h[1-6]|pre)(?: +class="(\\S+)")?(>[\\s\\S]*?))', "gm"); - text = text.replace(blockRe, function(wholeMatch, k, tag, cls, rest) { - if (!tag) // no following header or fenced code block. - return ''; - - // get attributes list from hash - var key = parseInt(k, 10); - var attributes = self.hashBlocks[key]; - - // get id - var id = attributes.match(/#[^\s#.]+/g) || []; - var idStr = id[0] ? ' id="' + id[0].substr(1, id[0].length - 1) + '"' : ''; - - // get classes and merge with existing classes - var classes = attributes.match(/\.[^\s#.]+/g) || []; - for (var i = 0; i < classes.length; i++) // Remove leading dot - classes[i] = classes[i].substr(1, classes[i].length - 1); - - var classStr = ''; - if (cls) - classes = union(classes, [cls]); - - if (classes.length > 0) - classStr = ' class="' + classes.join(' ') + '"'; - - return "<" + tag + idStr + classStr + rest; - }); - - return text; - }; - - /****************************************************************** - * Tables * - *****************************************************************/ - - // Find and convert Markdown Extra tables into html. - Markdown.Extra.prototype.tables = function(text) { - var self = this; - - var leadingPipe = new RegExp( - ['^' , - '[ ]{0,3}' , // Allowed whitespace - '[|]' , // Initial pipe - '(.+)\\n' , // $1: Header Row - - '[ ]{0,3}' , // Allowed whitespace - '[|]([ ]*[-:]+[-| :]*)\\n' , // $2: Separator - - '(' , // $3: Table Body - '(?:[ ]*[|].*\\n?)*' , // Table rows - ')', - '(?:\\n|$)' // Stop at final newline - ].join(''), - 'gm' - ); - - var noLeadingPipe = new RegExp( - ['^' , - '[ ]{0,3}' , // Allowed whitespace - '(\\S.*[|].*)\\n' , // $1: Header Row - - '[ ]{0,3}' , // Allowed whitespace - '([-:]+[ ]*[|][-| :]*)\\n' , // $2: Separator - - '(' , // $3: Table Body - '(?:.*[|].*\\n?)*' , // Table rows - ')' , - '(?:\\n|$)' // Stop at final newline - ].join(''), - 'gm' - ); - - text = text.replace(leadingPipe, doTable); - text = text.replace(noLeadingPipe, doTable); - - // $1 = header, $2 = separator, $3 = body - function doTable(match, header, separator, body, offset, string) { - // remove any leading pipes and whitespace - header = header.replace(/^ *[|]/m, ''); - separator = separator.replace(/^ *[|]/m, ''); - body = body.replace(/^ *[|]/gm, ''); - - // remove trailing pipes and whitespace - header = header.replace(/[|] *$/m, ''); - separator = separator.replace(/[|] *$/m, ''); - body = body.replace(/[|] *$/gm, ''); - - // determine column alignments - var alignspecs = separator.split(/ *[|] */); - var align = []; - for (var i = 0; i < alignspecs.length; i++) { - var spec = alignspecs[i]; - if (spec.match(/^ *-+: *$/m)) - align[i] = ' align="right"'; - else if (spec.match(/^ *:-+: *$/m)) - align[i] = ' align="center"'; - else if (spec.match(/^ *:-+ *$/m)) - align[i] = ' align="left"'; - else align[i] = ''; - } - - // TODO: parse spans in header and rows before splitting, so that pipes - // inside of tags are not interpreted as separators - var headers = header.split(/ *[|] */); - var colCount = headers.length; - - // build html - var cls = self.tableClass ? ' class="' + self.tableClass + '"' : ''; - var html = ['\n', '\n', '\n'].join(''); - - // build column headers. - for (i = 0; i < colCount; i++) { - var headerHtml = convertSpans(trim(headers[i]), self); - html += [" ", headerHtml, "\n"].join(''); - } - html += "\n\n"; - - // build rows - var rows = body.split('\n'); - for (i = 0; i < rows.length; i++) { - if (rows[i].match(/^\s*$/)) // can apply to final row - continue; - - // ensure number of rowCells matches colCount - var rowCells = rows[i].split(/ *[|] */); - var lenDiff = colCount - rowCells.length; - for (var j = 0; j < lenDiff; j++) - rowCells.push(''); - - html += "\n"; - for (j = 0; j < colCount; j++) { - var colHtml = convertSpans(trim(rowCells[j]), self); - html += [" ", colHtml, "\n"].join(''); - } - html += "\n"; - } - - html += "\n"; - - // replace html with placeholder until postConversion step - return self.hashExtraBlock(html); - } - - return text; - }; - - - /****************************************************************** - * Footnotes * - *****************************************************************/ - - // Strip footnote, store in hashes. - Markdown.Extra.prototype.stripFootnoteDefinitions = function(text) { - var self = this; - - text = text.replace( - /\n[ ]{0,3}\[\^(.+?)\]\:[ \t]*\n?([\s\S]*?)\n{1,2}((?=\n[ ]{0,3}\S)|$)/g, - function(wholeMatch, m1, m2) { - m1 = slugify(m1); - m2 += "\n"; - m2 = m2.replace(/^[ ]{0,3}/g, ""); - self.footnotes[m1] = m2; - return "\n"; - }); - - return text; - }; - - - // Find and convert footnotes references. - Markdown.Extra.prototype.doFootnotes = function(text) { - var self = this; - if(self.isConvertingFootnote === true) { - return text; - } - - var footnoteCounter = 0; - text = text.replace(/\[\^(.+?)\]/g, function(wholeMatch, m1) { - var id = slugify(m1); - var footnote = self.footnotes[id]; - if (footnote === undefined) { - return wholeMatch; - } - footnoteCounter++; - self.usedFootnotes.push(id); - var html = '' + footnoteCounter - + ''; - return self.hashExtraInline(html); - }); - - return text; - }; - - // Print footnotes at the end of the document - Markdown.Extra.prototype.printFootnotes = function(text) { - var self = this; - - if (self.usedFootnotes.length === 0) { - return text; - } - - text += '\n\n
                                \n
                                \n
                                  \n\n'; - for(var i=0; i' - + formattedfootnote - + ' \n\n'; - } - text += '
                                \n
                                '; - return text; - }; - - - /****************************************************************** - * Fenced Code Blocks (gfm) * - ******************************************************************/ - - // Find and convert gfm-inspired fenced code blocks into html. - Markdown.Extra.prototype.fencedCodeBlocks = function(text) { - function encodeCode(code) { - code = code.replace(/&/g, "&"); - code = code.replace(//g, ">"); - // These were escaped by PageDown before postNormalization - code = code.replace(/~D/g, "$$"); - code = code.replace(/~T/g, "~"); - return code; - } - - var self = this; - text = text.replace(/(?:^|\n)```([^`\n]*)\n([\s\S]*?)\n```[ \t]*(?=\n)/g, function(match, m1, m2) { - var language = trim(m1), codeblock = m2; - - // adhere to specified options - var preclass = self.googleCodePrettify ? ' class="prettyprint"' : ''; - var codeclass = ''; - if (language) { - if (self.googleCodePrettify || self.highlightJs) { - // use html5 language- class names. supported by both prettify and highlight.js - codeclass = ' class="language-' + language + '"'; - } else { - codeclass = ' class="' + language + '"'; - } - } - - var html = ['', - encodeCode(codeblock), '

                            '].join(''); - - // replace codeblock with placeholder until postConversion step - return self.hashExtraBlock(html); - }); - - return text; - }; - - - /****************************************************************** - * SmartyPants * - ******************************************************************/ - - Markdown.Extra.prototype.educatePants = function(text) { - var self = this; - var result = ''; - var blockOffset = 0; - // Here we parse HTML in a very bad manner - text.replace(/(?:)|(<)([a-zA-Z1-6]+)([^\n]*?>)([\s\S]*?)(<\/\2>)/g, function(wholeMatch, m1, m2, m3, m4, m5, offset) { - var token = text.substring(blockOffset, offset); - result += self.applyPants(token); - self.smartyPantsLastChar = result.substring(result.length - 1); - blockOffset = offset + wholeMatch.length; - if(!m1) { - // Skip commentary - result += wholeMatch; - return; - } - // Skip special tags - if(!/code|kbd|pre|script|noscript|iframe|math|ins|del|pre/i.test(m2)) { - m4 = self.educatePants(m4); - } - else { - self.smartyPantsLastChar = m4.substring(m4.length - 1); - } - result += m1 + m2 + m3 + m4 + m5; - }); - var lastToken = text.substring(blockOffset); - result += self.applyPants(lastToken); - self.smartyPantsLastChar = result.substring(result.length - 1); - return result; - }; - - function revertPants(wholeMatch, m1) { - var blockText = m1; - blockText = blockText.replace(/&\#8220;/g, "\""); - blockText = blockText.replace(/&\#8221;/g, "\""); - blockText = blockText.replace(/&\#8216;/g, "'"); - blockText = blockText.replace(/&\#8217;/g, "'"); - blockText = blockText.replace(/&\#8212;/g, "---"); - blockText = blockText.replace(/&\#8211;/g, "--"); - blockText = blockText.replace(/&\#8230;/g, "..."); - return blockText; - } - - Markdown.Extra.prototype.applyPants = function(text) { - // Dashes - text = text.replace(/---/g, "—").replace(/--/g, "–"); - // Ellipses - text = text.replace(/\.\.\./g, "…").replace(/\.\s\.\s\./g, "…"); - // Backticks - text = text.replace(/``/g, "“").replace (/''/g, "”"); - - if(/^'$/.test(text)) { - // Special case: single-character ' token - if(/\S/.test(this.smartyPantsLastChar)) { - return "’"; - } - return "‘"; - } - if(/^"$/.test(text)) { - // Special case: single-character " token - if(/\S/.test(this.smartyPantsLastChar)) { - return "”"; - } - return "“"; - } - - // Special case if the very first character is a quote - // followed by punctuation at a non-word-break. Close the quotes by brute force: - text = text.replace (/^'(?=[!"#\$\%'()*+,\-.\/:;<=>?\@\[\\]\^_`{|}~]\B)/, "’"); - text = text.replace (/^"(?=[!"#\$\%'()*+,\-.\/:;<=>?\@\[\\]\^_`{|}~]\B)/, "”"); - - // Special case for double sets of quotes, e.g.: - //

                            He said, "'Quoted' words in a larger quote."

                            - text = text.replace(/"'(?=\w)/g, "“‘"); - text = text.replace(/'"(?=\w)/g, "‘“"); - - // Special case for decade abbreviations (the '80s): - text = text.replace(/'(?=\d{2}s)/g, "’"); - - // Get most opening single quotes: - text = text.replace(/(\s| |--|&[mn]dash;|&\#8211;|&\#8212;|&\#x201[34];)'(?=\w)/g, "$1‘"); - - // Single closing quotes: - text = text.replace(/([^\s\[\{\(\-])'/g, "$1’"); - text = text.replace(/'(?=\s|s\b)/g, "’"); - - // Any remaining single quotes should be opening ones: - text = text.replace(/'/g, "‘"); - - // Get most opening double quotes: - text = text.replace(/(\s| |--|&[mn]dash;|&\#8211;|&\#8212;|&\#x201[34];)"(?=\w)/g, "$1“"); - - // Double closing quotes: - text = text.replace(/([^\s\[\{\(\-])"/g, "$1”"); - text = text.replace(/"(?=\s)/g, "”"); - - // Any remaining quotes should be opening ones. - text = text.replace(/"/ig, "“"); - return text; - }; - - // Find and convert markdown extra definition lists into html. - Markdown.Extra.prototype.runSmartyPants = function(text) { - this.smartyPantsLastChar = ''; - text = this.educatePants(text); - // Clean everything inside html tags (some of them may have been converted due to our rough html parsing) - text = text.replace(/(<([a-zA-Z1-6]+)\b([^\n>]*?)(\/)?>)/g, revertPants); - return text; - }; - - /****************************************************************** - * Definition Lists * - ******************************************************************/ - - // Find and convert markdown extra definition lists into html. - Markdown.Extra.prototype.definitionLists = function(text) { - var wholeList = new RegExp( - ['(\\x02\\n?|\\n\\n)' , - '(?:' , - '(' , // $1 = whole list - '(' , // $2 - '[ ]{0,3}' , - '((?:[ \\t]*\\S.*\\n)+)', // $3 = defined term - '\\n?' , - '[ ]{0,3}:[ ]+' , // colon starting definition - ')' , - '([\\s\\S]+?)' , - '(' , // $4 - '(?=\\0x03)' , // \z - '|' , - '(?=' , - '\\n{2,}' , - '(?=\\S)' , - '(?!' , // Negative lookahead for another term - '[ ]{0,3}' , - '(?:\\S.*\\n)+?' , // defined term - '\\n?' , - '[ ]{0,3}:[ ]+' , // colon starting definition - ')' , - '(?!' , // Negative lookahead for another definition - '[ ]{0,3}:[ ]+' , // colon starting definition - ')' , - ')' , - ')' , - ')' , - ')' - ].join(''), - 'gm' - ); - - var self = this; - text = addAnchors(text); - - text = text.replace(wholeList, function(match, pre, list) { - var result = trim(self.processDefListItems(list)); - result = "
                            \n" + result + "\n
                            "; - return pre + self.hashExtraBlock(result) + "\n\n"; - }); - - return removeAnchors(text); - }; - - // Process the contents of a single definition list, splitting it - // into individual term and definition list items. - Markdown.Extra.prototype.processDefListItems = function(listStr) { - var self = this; - - var dt = new RegExp( - ['(\\x02\\n?|\\n\\n+)' , // leading line - '(' , // definition terms = $1 - '[ ]{0,3}' , // leading whitespace - '(?![:][ ]|[ ])' , // negative lookahead for a definition - // mark (colon) or more whitespace - '(?:\\S.*\\n)+?' , // actual term (not whitespace) - ')' , - '(?=\\n?[ ]{0,3}:[ ])' // lookahead for following line feed - ].join(''), // with a definition mark - 'gm' - ); - - var dd = new RegExp( - ['\\n(\\n+)?' , // leading line = $1 - '(' , // marker space = $2 - '[ ]{0,3}' , // whitespace before colon - '[:][ ]+' , // definition mark (colon) - ')' , - '([\\s\\S]+?)' , // definition text = $3 - '(?=\\n*' , // stop at next definition mark, - '(?:' , // next term or end of text - '\\n[ ]{0,3}[:][ ]|' , - '
                            |\\x03' , // \z - ')' , - ')' - ].join(''), - 'gm' - ); - - listStr = addAnchors(listStr); - // trim trailing blank lines: - listStr = listStr.replace(/\n{2,}(?=\\x03)/, "\n"); - - // Process definition terms. - listStr = listStr.replace(dt, function(match, pre, termsStr) { - var terms = trim(termsStr).split("\n"); - var text = ''; - for (var i = 0; i < terms.length; i++) { - var term = terms[i]; - // process spans inside dt - term = convertSpans(trim(term), self); - text += "\n
                            " + term + "
                            "; - } - return text + "\n"; - }); - - // Process actual definitions. - listStr = listStr.replace(dd, function(match, leadingLine, markerSpace, def) { - if (leadingLine || def.match(/\n{2,}/)) { - // replace marker with the appropriate whitespace indentation - def = Array(markerSpace.length + 1).join(' ') + def; - // process markdown inside definition - // TODO?: currently doesn't apply extensions - def = outdent(def) + "\n\n"; - def = "\n" + convertAll(def, self) + "\n"; - } else { - // convert span-level markdown inside definition - def = rtrim(def); - def = convertSpans(outdent(def), self); - } - - return "\n
                            " + def + "
                            \n"; - }); - - return removeAnchors(listStr); - }; - - - /*********************************************************** - * Strikethrough * - ************************************************************/ - - Markdown.Extra.prototype.strikethrough = function(text) { - // Pretty much duplicated from _DoItalicsAndBold - return text.replace(/([\W_]|^)~T~T(?=\S)([^\r]*?\S[\*_]*)~T~T([\W_]|$)/g, - "$1$2$3"); - }; - - - /*********************************************************** - * New lines * - ************************************************************/ - - Markdown.Extra.prototype.newlines = function(text) { - // We have to ignore already converted newlines and line breaks in sub-list items - return text.replace(/(<(?:br|\/li)>)?\n/g, function(wholeMatch, previousTag) { - return previousTag ? wholeMatch : "
                            \n"; - }); - }; - -})(); - diff --git a/system/admin/editor/js/Markdown.Sanitizer.js b/system/admin/editor/js/Markdown.Sanitizer.js deleted file mode 100644 index bfced16f..00000000 --- a/system/admin/editor/js/Markdown.Sanitizer.js +++ /dev/null @@ -1,108 +0,0 @@ -(function () { - var output, Converter; - if (typeof exports === "object" && typeof require === "function") { // we're in a CommonJS (e.g. Node.js) module - output = exports; - Converter = require("./Markdown.Converter").Converter; - } else { - output = window.Markdown; - Converter = output.Converter; - } - - output.getSanitizingConverter = function () { - var converter = new Converter(); - converter.hooks.chain("postConversion", sanitizeHtml); - converter.hooks.chain("postConversion", balanceTags); - return converter; - } - - function sanitizeHtml(html) { - return html.replace(/<[^>]*>?/gi, sanitizeTag); - } - - // (tags that can be opened/closed) | (tags that stand alone) - var basic_tag_whitelist = /^(<\/?(b|blockquote|code|del|dd|dl|dt|em|h1|h2|h3|i|kbd|li|ol|p|pre|s|sup|sub|strong|strike|ul)>|<(br|hr)\s?\/?>)$/i; - // | - var a_white = /^(]+")?\s?>|<\/a>)$/i; - - // ]*")?(\stitle="[^"<>]*")?\s?\/?>)$/i; - - function sanitizeTag(tag) { - if (tag.match(basic_tag_whitelist) || tag.match(a_white) || tag.match(img_white)) - return tag; - else - return ""; - } - - /// - /// attempt to balance HTML tags in the html string - /// by removing any unmatched opening or closing tags - /// IMPORTANT: we *assume* HTML has *already* been - /// sanitized and is safe/sane before balancing! - /// - /// adapted from CODESNIPPET: A8591DBA-D1D3-11DE-947C-BA5556D89593 - /// - function balanceTags(html) { - - if (html == "") - return ""; - - var re = /<\/?\w+[^>]*(\s|$|>)/g; - // convert everything to lower case; this makes - // our case insensitive comparisons easier - var tags = html.toLowerCase().match(re); - - // no HTML tags present? nothing to do; exit now - var tagcount = (tags || []).length; - if (tagcount == 0) - return html; - - var tagname, tag; - var ignoredtags = "



                          5. "; - var match; - var tagpaired = []; - var tagremove = []; - var needsRemoval = false; - - // loop through matched tags in forward order - for (var ctag = 0; ctag < tagcount; ctag++) { - tagname = tags[ctag].replace(/<\/?(\w+).*/, "$1"); - // skip any already paired tags - // and skip tags in our ignore list; assume they're self-closed - if (tagpaired[ctag] || ignoredtags.search("<" + tagname + ">") > -1) - continue; - - tag = tags[ctag]; - match = -1; - - if (!/^<\//.test(tag)) { - // this is an opening tag - // search forwards (next tags), look for closing tags - for (var ntag = ctag + 1; ntag < tagcount; ntag++) { - if (!tagpaired[ntag] && tags[ntag] == "") { - match = ntag; - break; - } - } - } - - if (match == -1) - needsRemoval = tagremove[ctag] = true; // mark for removal - else - tagpaired[match] = true; // mark paired - } - - if (!needsRemoval) - return html; - - // delete all orphaned tags from the string - - var ctag = 0; - html = html.replace(re, function (match) { - var res = tagremove[ctag] ? "" : match; - ctag++; - return res; - }); - return html; - } -})(); diff --git a/system/admin/editor/js/editor.js b/system/admin/editor/js/editor.js deleted file mode 100644 index 3a192988..00000000 --- a/system/admin/editor/js/editor.js +++ /dev/null @@ -1,71 +0,0 @@ -(function () { - var converter = new Markdown.Converter(); - Markdown.Extra.init(converter); - var editor = new Markdown.Editor(converter); - - //======Image Uploader===== - var callbackFunc; - var dialogClose = function() { - $('#insertImageDialog').modal('hide'); - $('#insertImageDialogURL').val(''); - $('#insertImageDialogFile').val(''); - $('#insertMediaDialogURL').val(''); - $('#insertMediaDialogFile').val(''); - }; - $('#insertImageDialogInsert').click( function() { - callbackFunc( $('#insertImageDialogURL').val().length > 0 ? $('#insertImageDialogURL').val() : null ); - dialogClose(); - }); - $('#insertImageDialogClose').click( function() { - callbackFunc(null); - dialogClose(); - }); - $('#insertImageDialogCancel').click( function() { - callbackFunc(null); - dialogClose(); - }); - $('#insertImageDialogFile').on('input', function(){ - var file = $("#insertImageDialogFile").prop("files"); - var formData = new FormData(); - formData.append('file', file[0], file[0].name); - // Set up the request. - $.ajax({ - type: "POST", - url: base_path + 'upload.php', - data: formData, - processData: false, - contentType: false, - success: function (response) { - if (response.error == '0') - { - callbackFunc(base_path + response.path); - dialogClose(); - } - else - { - if (response.error !== '') alert(response.error); - else alert("An unknown error has occurred"); - console.error("Bad Response"); - console.error(response); - $('#insertImageDialogFile').val(''); - } - }, - failure: function (response) { - if (response.error !== '') alert(response.error); - else alert("An unknown error has occurred"); - console.error("Unable to Upload"); - console.error(response); - $('#insertImageDialogFile').val(''); - } - });//ajax - });//oninput - editor.hooks.set('insertImageDialog', function(callback) { - $('#insertImageDialog').modal('show'); - callbackFunc = callback; - - return true; // tell the editor that we'll take care of getting the image url - }); - //=====end image uploader===== - editor.run(); - -})(); \ No newline at end of file diff --git a/system/admin/editor/js/local/Markdown.local.fr.js b/system/admin/editor/js/local/Markdown.local.fr.js deleted file mode 100644 index febd0806..00000000 --- a/system/admin/editor/js/local/Markdown.local.fr.js +++ /dev/null @@ -1,43 +0,0 @@ -// Usage: -// -// var myConverter = new Markdown.Editor(myConverter, null, { strings: Markdown.local.fr }); - -(function () { - Markdown.local = Markdown.local || {}; - Markdown.local.fr = { - bold: "Gras Ctrl+B", - boldexample: "texte en gras", - - italic: "Italique Ctrl+I", - italicexample: "texte en italique", - - link: "Hyperlien Ctrl+L", - linkdescription: "description de l'hyperlien", - linkdialog: "

                            Insérer un hyperlien

                            http://example.com/ \"titre optionnel\"

                            ", - - quote: "Citation
                            Ctrl+Q", - quoteexample: "Citation", - - code: "Extrait de code
                             Ctrl+K",
                            -        codeexample: "votre extrait de code",
                            -
                            -        image: "Image  Ctrl+G",
                            -        imagedescription: "description de l'image",
                            -        imagedialog: "

                            Insérer une image

                            http://example.com/images/diagram.jpg \"titre optionnel\"

                            Vous chercher un hébergement d'image grauit ?

                            ", - - olist: "Liste numérotée
                              Ctrl+O", - ulist: "Liste à point
                                Ctrl+U", - litem: "Elément de liste", - - heading: "Titre

                                /

                                Ctrl+H", - headingexample: "Titre", - - hr: "Trait horizontal
                                Ctrl+R", - - undo: "Annuler - Ctrl+Z", - redo: "Refaire - Ctrl+Y", - redomac: "Refaire - Ctrl+Shift+Z", - - help: "Aide sur Markdown" - }; -})(); \ No newline at end of file diff --git a/system/admin/editor/js/node-pagedown-extra.js b/system/admin/editor/js/node-pagedown-extra.js deleted file mode 100644 index a19f1d97..00000000 --- a/system/admin/editor/js/node-pagedown-extra.js +++ /dev/null @@ -1,3 +0,0 @@ -GLOBAL.Markdown = {}; -require('./Markdown.Extra.js'); -exports.Extra = Markdown.Extra; diff --git a/system/admin/editor/js/node-pagedown.js b/system/admin/editor/js/node-pagedown.js deleted file mode 100644 index e214548d..00000000 --- a/system/admin/editor/js/node-pagedown.js +++ /dev/null @@ -1,2 +0,0 @@ -exports.Converter = require("./Markdown.Converter").Converter; -exports.getSanitizingConverter = require("./Markdown.Sanitizer").getSanitizingConverter; diff --git a/system/admin/editor/package.json b/system/admin/editor/package.json deleted file mode 100644 index 57b06896..00000000 --- a/system/admin/editor/package.json +++ /dev/null @@ -1,19 +0,0 @@ -{ - "name": "pagedown", - "version": "1.1.0", - "description": "markdown converter, based on showdown", - "repository": { - "type": "hg", - "url": "https://code.google.com/p/pagedown/" - }, - "keywords": ["markdown"], - "license": "MIT", - "files": [ - "Markdown.Converter.js", - "Markdown.Sanitizer.js", - "node-pagedown.js" - ], - "main": "node-pagedown.js", - "bugs": "http://code.google.com/p/pagedown/issues/list", - "homepage": "http://code.google.com/p/pagedown/wiki/PageDown" -} diff --git a/system/admin/views/add-content.html.php b/system/admin/views/add-content.html.php index 25aa1f30..e0517fc4 100644 --- a/system/admin/views/add-content.html.php +++ b/system/admin/views/add-content.html.php @@ -29,12 +29,8 @@ ?> - - - - - + - + + + \ No newline at end of file diff --git a/system/admin/views/add-page.html.php b/system/admin/views/add-page.html.php index 656282a0..bbd61ed8 100644 --- a/system/admin/views/add-page.html.php +++ b/system/admin/views/add-page.html.php @@ -1,12 +1,8 @@ - - - - - + @@ -33,13 +29,25 @@ -
                                - -
                                - -
                                +
                                + +
                                +
                                +
                                + +
                                +
                                +
                                +
                                +
                                + +
                                +
                                +
                                +
                                +
                                @@ -48,15 +56,10 @@
                                -
                                - -
                                -
                                -
                                - +