php.js
150 lines
| 6.0 KiB
| application/javascript
|
JavascriptLexer
Matthias BUSSONNIER
|
r8053 | (function() { | ||
function keywords(str) { | ||||
var obj = {}, words = str.split(" "); | ||||
for (var i = 0; i < words.length; ++i) obj[words[i]] = true; | ||||
return obj; | ||||
} | ||||
function heredoc(delim) { | ||||
return function(stream, state) { | ||||
if (stream.match(delim)) state.tokenize = null; | ||||
else stream.skipToEnd(); | ||||
return "string"; | ||||
} | ||||
} | ||||
var phpConfig = { | ||||
name: "clike", | ||||
keywords: keywords("abstract and array as break case catch class clone const continue declare default " + | ||||
"do else elseif enddeclare endfor endforeach endif endswitch endwhile extends final " + | ||||
"for foreach function global goto if implements interface instanceof namespace " + | ||||
"new or private protected public static switch throw trait try use var while xor " + | ||||
"die echo empty exit eval include include_once isset list require require_once return " + | ||||
"print unset __halt_compiler self static parent"), | ||||
blockKeywords: keywords("catch do else elseif for foreach if switch try while"), | ||||
atoms: keywords("true false null TRUE FALSE NULL"), | ||||
multiLineStrings: true, | ||||
hooks: { | ||||
"$": function(stream, state) { | ||||
stream.eatWhile(/[\w\$_]/); | ||||
return "variable-2"; | ||||
}, | ||||
"<": function(stream, state) { | ||||
if (stream.match(/<</)) { | ||||
stream.eatWhile(/[\w\.]/); | ||||
state.tokenize = heredoc(stream.current().slice(3)); | ||||
return state.tokenize(stream, state); | ||||
} | ||||
return false; | ||||
}, | ||||
"#": function(stream, state) { | ||||
while (!stream.eol() && !stream.match("?>", false)) stream.next(); | ||||
return "comment"; | ||||
}, | ||||
"/": function(stream, state) { | ||||
if (stream.eat("/")) { | ||||
while (!stream.eol() && !stream.match("?>", false)) stream.next(); | ||||
return "comment"; | ||||
} | ||||
return false; | ||||
} | ||||
} | ||||
}; | ||||
CodeMirror.defineMode("php", function(config, parserConfig) { | ||||
var htmlMode = CodeMirror.getMode(config, {name: "xml", htmlMode: true}); | ||||
var jsMode = CodeMirror.getMode(config, "javascript"); | ||||
var cssMode = CodeMirror.getMode(config, "css"); | ||||
var phpMode = CodeMirror.getMode(config, phpConfig); | ||||
function dispatch(stream, state) { // TODO open PHP inside text/css | ||||
var isPHP = state.mode == "php"; | ||||
if (stream.sol() && state.pending != '"') state.pending = null; | ||||
if (state.curMode == htmlMode) { | ||||
if (stream.match(/^<\?\w*/)) { | ||||
state.curMode = phpMode; | ||||
state.curState = state.php; | ||||
state.curClose = "?>"; | ||||
state.mode = "php"; | ||||
return "meta"; | ||||
} | ||||
if (state.pending == '"') { | ||||
while (!stream.eol() && stream.next() != '"') {} | ||||
var style = "string"; | ||||
} else if (state.pending && stream.pos < state.pending.end) { | ||||
stream.pos = state.pending.end; | ||||
var style = state.pending.style; | ||||
} else { | ||||
var style = htmlMode.token(stream, state.curState); | ||||
} | ||||
state.pending = null; | ||||
var cur = stream.current(), openPHP = cur.search(/<\?/); | ||||
if (openPHP != -1) { | ||||
if (style == "string" && /\"$/.test(cur) && !/\?>/.test(cur)) state.pending = '"'; | ||||
else state.pending = {end: stream.pos, style: style}; | ||||
stream.backUp(cur.length - openPHP); | ||||
} else if (style == "tag" && stream.current() == ">" && state.curState.context) { | ||||
if (/^script$/i.test(state.curState.context.tagName)) { | ||||
state.curMode = jsMode; | ||||
state.curState = jsMode.startState(htmlMode.indent(state.curState, "")); | ||||
state.curClose = /^<\/\s*script\s*>/i; | ||||
state.mode = "javascript"; | ||||
} | ||||
else if (/^style$/i.test(state.curState.context.tagName)) { | ||||
state.curMode = cssMode; | ||||
state.curState = cssMode.startState(htmlMode.indent(state.curState, "")); | ||||
state.curClose = /^<\/\s*style\s*>/i; | ||||
state.mode = "css"; | ||||
} | ||||
} | ||||
return style; | ||||
} else if ((!isPHP || state.php.tokenize == null) && | ||||
stream.match(state.curClose, isPHP)) { | ||||
state.curMode = htmlMode; | ||||
state.curState = state.html; | ||||
state.curClose = null; | ||||
state.mode = "html"; | ||||
if (isPHP) return "meta"; | ||||
else return dispatch(stream, state); | ||||
} else { | ||||
return state.curMode.token(stream, state.curState); | ||||
} | ||||
} | ||||
return { | ||||
startState: function() { | ||||
var html = htmlMode.startState(); | ||||
return {html: html, | ||||
php: phpMode.startState(), | ||||
curMode: parserConfig.startOpen ? phpMode : htmlMode, | ||||
curState: parserConfig.startOpen ? phpMode.startState() : html, | ||||
curClose: parserConfig.startOpen ? /^\?>/ : null, | ||||
mode: parserConfig.startOpen ? "php" : "html", | ||||
pending: null} | ||||
}, | ||||
copyState: function(state) { | ||||
var html = state.html, htmlNew = CodeMirror.copyState(htmlMode, html), | ||||
php = state.php, phpNew = CodeMirror.copyState(phpMode, php), cur; | ||||
if (state.curState == html) cur = htmlNew; | ||||
else if (state.curState == php) cur = phpNew; | ||||
else cur = CodeMirror.copyState(state.curMode, state.curState); | ||||
return {html: htmlNew, php: phpNew, curMode: state.curMode, curState: cur, | ||||
curClose: state.curClose, mode: state.mode, | ||||
pending: state.pending}; | ||||
}, | ||||
token: dispatch, | ||||
indent: function(state, textAfter) { | ||||
if ((state.curMode != phpMode && /^\s*<\//.test(textAfter)) || | ||||
(state.curMode == phpMode && /^\?>/.test(textAfter))) | ||||
return htmlMode.indent(state.html, textAfter); | ||||
return state.curMode.indent(state.curState, textAfter); | ||||
}, | ||||
electricChars: "/{}:" | ||||
} | ||||
}, "xml", "clike", "javascript", "css"); | ||||
CodeMirror.defineMIME("application/x-httpd-php", "php"); | ||||
CodeMirror.defineMIME("application/x-httpd-php-open", {name: "php", startOpen: true}); | ||||
CodeMirror.defineMIME("text/x-php", phpConfig); | ||||
})(); | ||||