| // CodeMirror, copyright (c) by Marijn Haverbeke and others |
| // Distributed under an MIT license: http://codemirror.net/LICENSE |
| |
| (function(mod) { |
| if (typeof exports == "object" && typeof module == "object") // CommonJS |
| mod(require("../../lib/codemirror"), require("../xml/xml"), require("../javascript/javascript"), require("../css/css")); |
| else if (typeof define == "function" && define.amd) // AMD |
| define(["../../lib/codemirror", "../xml/xml", "../javascript/javascript", "../css/css"], mod); |
| else // Plain browser env |
| mod(CodeMirror); |
| })(function(CodeMirror) { |
| "use strict"; |
| |
| var defaultTags = { |
| script: [ |
| ["lang", /(javascript|babel)/i, "javascript"], |
| ["type", /^(?:text|application)\/(?:x-)?(?:java|ecma)script$|^$/i, "javascript"], |
| ["type", /./, "text/plain"], |
| [null, null, "javascript"] |
| ], |
| style: [ |
| ["lang", /^css$/i, "css"], |
| ["type", /^(text\/)?(x-)?(stylesheet|css)$/i, "css"], |
| ["type", /./, "text/plain"], |
| [null, null, "css"] |
| ] |
| }; |
| |
| function maybeBackup(stream, pat, style) { |
| var cur = stream.current(), close = cur.search(pat); |
| if (close > -1) { |
| stream.backUp(cur.length - close); |
| } else if (cur.match(/<\/?$/)) { |
| stream.backUp(cur.length); |
| if (!stream.match(pat, false)) stream.match(cur); |
| } |
| return style; |
| } |
| |
| var attrRegexpCache = {}; |
| function getAttrRegexp(attr) { |
| var regexp = attrRegexpCache[attr]; |
| if (regexp) return regexp; |
| return attrRegexpCache[attr] = new RegExp("\\s+" + attr + "\\s*=\\s*('|\")?([^'\"]+)('|\")?\\s*"); |
| } |
| |
| function getAttrValue(text, attr) { |
| var match = text.match(getAttrRegexp(attr)) |
| return match ? /^\s*(.*?)\s*$/.exec(match[2])[1] : "" |
| } |
| |
| function getTagRegexp(tagName, anchored) { |
| return new RegExp((anchored ? "^" : "") + "<\/\s*" + tagName + "\s*>", "i"); |
| } |
| |
| function addTags(from, to) { |
| for (var tag in from) { |
| var dest = to[tag] || (to[tag] = []); |
| var source = from[tag]; |
| for (var i = source.length - 1; i >= 0; i--) |
| dest.unshift(source[i]) |
| } |
| } |
| |
| function findMatchingMode(tagInfo, tagText) { |
| for (var i = 0; i < tagInfo.length; i++) { |
| var spec = tagInfo[i]; |
| if (!spec[0] || spec[1].test(getAttrValue(tagText, spec[0]))) return spec[2]; |
| } |
| } |
| |
| CodeMirror.defineMode("htmlmixed", function (config, parserConfig) { |
| var htmlMode = CodeMirror.getMode(config, { |
| name: "xml", |
| htmlMode: true, |
| multilineTagIndentFactor: parserConfig.multilineTagIndentFactor, |
| multilineTagIndentPastTag: parserConfig.multilineTagIndentPastTag |
| }); |
| |
| var tags = {}; |
| var configTags = parserConfig && parserConfig.tags, configScript = parserConfig && parserConfig.scriptTypes; |
| addTags(defaultTags, tags); |
| if (configTags) addTags(configTags, tags); |
| if (configScript) for (var i = configScript.length - 1; i >= 0; i--) |
| tags.script.unshift(["type", configScript[i].matches, configScript[i].mode]) |
| |
| function html(stream, state) { |
| var style = htmlMode.token(stream, state.htmlState), tag = /\btag\b/.test(style), tagName |
| if (tag && !/[<>\s\/]/.test(stream.current()) && |
| (tagName = state.htmlState.tagName && state.htmlState.tagName.toLowerCase()) && |
| tags.hasOwnProperty(tagName)) { |
| state.inTag = tagName + " " |
| } else if (state.inTag && tag && />$/.test(stream.current())) { |
| var inTag = /^([\S]+) (.*)/.exec(state.inTag) |
| state.inTag = null |
| var modeSpec = stream.current() == ">" && findMatchingMode(tags[inTag[1]], inTag[2]) |
| var mode = CodeMirror.getMode(config, modeSpec) |
| var endTagA = getTagRegexp(inTag[1], true), endTag = getTagRegexp(inTag[1], false); |
| state.token = function (stream, state) { |
| if (stream.match(endTagA, false)) { |
| state.token = html; |
| state.localState = state.localMode = null; |
| return null; |
| } |
| return maybeBackup(stream, endTag, state.localMode.token(stream, state.localState)); |
| }; |
| state.localMode = mode; |
| state.localState = CodeMirror.startState(mode, htmlMode.indent(state.htmlState, "")); |
| } else if (state.inTag) { |
| state.inTag += stream.current() |
| if (stream.eol()) state.inTag += " " |
| } |
| return style; |
| }; |
| |
| return { |
| startState: function () { |
| var state = CodeMirror.startState(htmlMode); |
| return {token: html, inTag: null, localMode: null, localState: null, htmlState: state}; |
| }, |
| |
| copyState: function (state) { |
| var local; |
| if (state.localState) { |
| local = CodeMirror.copyState(state.localMode, state.localState); |
| } |
| return {token: state.token, inTag: state.inTag, |
| localMode: state.localMode, localState: local, |
| htmlState: CodeMirror.copyState(htmlMode, state.htmlState)}; |
| }, |
| |
| token: function (stream, state) { |
| return state.token(stream, state); |
| }, |
| |
| indent: function (state, textAfter) { |
| if (!state.localMode || /^\s*<\//.test(textAfter)) |
| return htmlMode.indent(state.htmlState, textAfter); |
| else if (state.localMode.indent) |
| return state.localMode.indent(state.localState, textAfter); |
| else |
| return CodeMirror.Pass; |
| }, |
| |
| innerMode: function (state) { |
| return {state: state.localState || state.htmlState, mode: state.localMode || htmlMode}; |
| } |
| }; |
| }, "xml", "javascript", "css"); |
| |
| CodeMirror.defineMIME("text/html", "htmlmixed"); |
| }); |