blob: e1390cd719f4338686a7e3d735af449d9eedc1a4 [file] [log] [blame]
/*
* Copyright (C) 2013 Apple Inc. All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
*
* THIS SOFTWARE IS PROVIDED BY APPLE INC. AND ITS CONTRIBUTORS ``AS IS''
* AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO,
* THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
* PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR ITS CONTRIBUTORS
* BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
* CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
* SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
* INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
* CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
* ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF
* THE POSSIBILITY OF SUCH DAMAGE.
*/
(function () {
// By default CodeMirror defines syntax highlighting styles based on token
// only and shared styles between modes. This limiting and does not match
// what we have done in the Web Inspector. So this modifies the XML, CSS
// and JavaScript modes to supply two styles for each token. One for the
// token and one with the mode name.
function tokenizeLinkString(stream, state)
{
console.assert(state._linkQuoteCharacter !== undefined);
// Eat the string until the same quote is found that started the string.
// If this is unquoted, then eat until whitespace or common parse errors.
if (state._linkQuoteCharacter)
stream.eatWhile(new RegExp("[^" + state._linkQuoteCharacter + "]"));
else
stream.eatWhile(/[^\s\u00a0=<>\"\']/);
// If the stream isn't at the end of line then we found the end quote.
// In the case, change _linkTokenize to parse the end of the link next.
// Otherwise _linkTokenize will stay as-is to parse more of the link.
if (!stream.eol())
state._linkTokenize = tokenizeEndOfLinkString;
return "link";
}
function tokenizeEndOfLinkString(stream, state)
{
console.assert(state._linkQuoteCharacter !== undefined);
console.assert(state._linkBaseStyle);
// Eat the quote character to style it with the base style.
if (state._linkQuoteCharacter)
stream.eat(state._linkQuoteCharacter);
var style = state._linkBaseStyle;
// Clean up the state.
delete state._linkTokenize;
delete state._linkQuoteCharacter;
delete state._linkBaseStyle;
delete state._srcSetTokenizeState;
return style;
}
function tokenizeSrcSetString(stream, state)
{
console.assert(state._linkQuoteCharacter !== undefined);
if (state._srcSetTokenizeState === "link") {
// Eat the string until a space, comma, or ending quote.
// If this is unquoted, then eat until whitespace or common parse errors.
if (state._linkQuoteCharacter)
stream.eatWhile(new RegExp("[^\\s," + state._linkQuoteCharacter + "]"));
else
stream.eatWhile(/[^\s,\u00a0=<>\"\']/);
} else {
// Eat the string until a comma, or ending quote.
// If this is unquoted, then eat until whitespace or common parse errors.
stream.eatSpace();
if (state._linkQuoteCharacter)
stream.eatWhile(new RegExp("[^," + state._linkQuoteCharacter + "]"));
else
stream.eatWhile(/[^\s\u00a0=<>\"\']/);
stream.eatWhile(/[\s,]/);
}
// If the stream isn't at the end of line and we found the end quote
// change _linkTokenize to parse the end of the link next. Otherwise
// _linkTokenize will stay as-is to parse more of the srcset.
if (stream.eol() || (!state._linkQuoteCharacter || stream.peek() === state._linkQuoteCharacter))
state._linkTokenize = tokenizeEndOfLinkString;
// Link portion.
if (state._srcSetTokenizeState === "link") {
state._srcSetTokenizeState = "descriptor";
return "link";
}
// Descriptor portion.
state._srcSetTokenizeState = "link";
return state._linkBaseStyle;
}
function extendedXMLToken(stream, state)
{
if (state._linkTokenize) {
// Call the link tokenizer instead.
var style = state._linkTokenize(stream, state);
return style && (style + " m-" + this.name);
}
// Remember the start position so we can rewind if needed.
var startPosition = stream.pos;
var style = this._token(stream, state);
if (style === "attribute") {
// Look for "href" or "src" attributes. If found then we should
// expect a string later that should get the "link" style instead.
var text = stream.current().toLowerCase();
if (text === "src" || /\bhref\b/.test(text))
state._expectLink = true;
else if (text === "srcset")
state._expectSrcSet = true;
else {
delete state._expectLink;
delete state._expectSrcSet;
}
} else if (state._expectLink && style === "string") {
var current = stream.current();
// Unless current token is empty quotes, consume quote character
// and tokenize link next.
if (current !== "\"\"" && current !== "''") {
delete state._expectLink;
// This is a link, so setup the state to process it next.
state._linkTokenize = tokenizeLinkString;
state._linkBaseStyle = style;
// The attribute may or may not be quoted.
var quote = current[0];
state._linkQuoteCharacter = quote === "'" || quote === "\"" ? quote : null;
// Rewind the stream to the start of this token.
stream.pos = startPosition;
// Eat the open quote of the string so the string style
// will be used for the quote character.
if (state._linkQuoteCharacter)
stream.eat(state._linkQuoteCharacter);
}
} else if (state._expectSrcSet && style === "string") {
var current = stream.current();
// Unless current token is empty quotes, consume quote character
// and tokenize link next.
if (current !== "\"\"" && current !== "''") {
delete state._expectSrcSet;
// This is a link, so setup the state to process it next.
state._srcSetTokenizeState = "link";
state._linkTokenize = tokenizeSrcSetString;
state._linkBaseStyle = style;
// The attribute may or may not be quoted.
var quote = current[0];
state._linkQuoteCharacter = quote === "'" || quote === "\"" ? quote : null;
// Rewind the stream to the start of this token.
stream.pos = startPosition;
// Eat the open quote of the string so the string style
// will be used for the quote character.
if (state._linkQuoteCharacter)
stream.eat(state._linkQuoteCharacter);
}
} else if (style) {
// We don't expect other tokens between attribute and string since
// spaces and the equal character are not tokenized. So if we get
// another token before a string then we stop expecting a link.
delete state._expectLink;
delete state._expectSrcSet;
}
return style && (style + " m-" + this.name);
}
function tokenizeCSSURLString(stream, state)
{
console.assert(state._urlQuoteCharacter);
// If we are an unquoted url string, return whitespace blocks as a whitespace token (null).
if (state._unquotedURLString && stream.eatSpace())
return null;
var ch = null;
var escaped = false;
var reachedEndOfURL = false;
var lastNonWhitespace = stream.pos;
var quote = state._urlQuoteCharacter;
// Parse characters until the end of the stream/line or a proper end quote character.
while ((ch = stream.next()) != null) {
if (ch === quote && !escaped) {
reachedEndOfURL = true;
break;
}
escaped = !escaped && ch === "\\";
if (!/[\s\u00a0]/.test(ch))
lastNonWhitespace = stream.pos;
}
// If we are an unquoted url string, do not include trailing whitespace, rewind to the last real character.
if (state._unquotedURLString)
stream.pos = lastNonWhitespace;
// If we have reached the proper the end of the url string, switch to the end tokenizer to reset the state.
if (reachedEndOfURL) {
if (!state._unquotedURLString)
stream.backUp(1);
this._urlTokenize = tokenizeEndOfCSSURLString;
}
return "link";
}
function tokenizeEndOfCSSURLString(stream, state)
{
console.assert(state._urlQuoteCharacter);
console.assert(state._urlBaseStyle);
// Eat the quote character to style it with the base style.
if (!state._unquotedURLString)
stream.eat(state._urlQuoteCharacter);
var style = state._urlBaseStyle;
delete state._urlTokenize;
delete state._urlQuoteCharacter;
delete state._urlBaseStyle;
return style;
}
function extendedCSSToken(stream, state)
{
var hexColorRegex = /#(?:[0-9a-fA-F]{8}|[0-9a-fA-F]{6}|[0-9a-fA-F]{3,4})\b/g;
if (state._urlTokenize) {
// Call the link tokenizer instead.
var style = state._urlTokenize(stream, state);
return style && (style + " m-" + (this.alternateName || this.name));
}
// Remember the start position so we can rewind if needed.
var startPosition = stream.pos;
var style = this._token(stream, state);
if (style) {
if (style === "atom") {
if (stream.current() === "url") {
// If the current text is "url" then we should expect the next string token to be a link.
state._expectLink = true;
} else if (hexColorRegex.test(stream.current()))
style = style + " hex-color";
} else if (style === "error") {
if (state.state=== "atBlock" || state.state === "atBlock_parens") {
switch (stream.current()) {
case "prefers-color-scheme":
case "light":
case "dark":
case "prefers-reduced-motion":
case "reduce":
case "no-preference":
case "inverted-colors":
case "inverted":
case "color-gamut":
case "p3":
case "rec2020":
case "display-mode":
case "fullscreen":
case "standalone":
case "minimal-ui":
case "browser":
case /*-webkit-*/"video-playable-inline":
case /*-webkit-*/"transform-2d":
case /*-webkit-*/"transform-3d":
style = "property";
break;
}
}
} else if (state._expectLink) {
delete state._expectLink;
if (style === "string") {
// This is a link, so setup the state to process it next.
state._urlTokenize = tokenizeCSSURLString;
state._urlBaseStyle = style;
// The url may or may not be quoted.
var quote = stream.current()[0];
state._urlQuoteCharacter = quote === "'" || quote === "\"" ? quote : ")";
state._unquotedURLString = state._urlQuoteCharacter === ")";
// Rewind the stream to the start of this token.
stream.pos = startPosition;
// Eat the open quote of the string so the string style
// will be used for the quote character.
if (!state._unquotedURLString)
stream.eat(state._urlQuoteCharacter);
}
}
}
return style && (style + " m-" + (this.alternateName || this.name));
}
function extendedJavaScriptToken(stream, state)
{
// CodeMirror moves the original token function to _token when we extended it.
// So call it to get the style that we will add an additional class name to.
var style = this._token(stream, state);
if (style === "number" && stream.current().endsWith("n"))
style += " bigint";
return style && (style + " m-" + (this.alternateName || this.name));
}
function scrollCursorIntoView(codeMirror, event)
{
// We don't want to use the default implementation since it can cause massive jumping
// when the editor is contained inside overflow elements.
event.preventDefault();
function delayedWork()
{
// Don't try to scroll unless the editor is focused.
if (!codeMirror.getWrapperElement().classList.contains("CodeMirror-focused"))
return;
// The cursor element can contain multiple cursors. The first one is the blinky cursor,
// which is the one we want to scroll into view. It can be missing, so check first.
var cursorElement = codeMirror.getScrollerElement().getElementsByClassName("CodeMirror-cursor")[0];
if (cursorElement)
cursorElement.scrollIntoViewIfNeeded(false);
}
// We need to delay this because CodeMirror can fire scrollCursorIntoView as a view is being blurred
// and another is being focused. The blurred editor still has the focused state when this event fires.
// We don't want to scroll the blurred editor into view, only the focused editor.
setTimeout(delayedWork, 0);
}
CodeMirror.extendMode("css", {token: extendedCSSToken});
CodeMirror.extendMode("xml", {token: extendedXMLToken});
CodeMirror.extendMode("javascript", {token: extendedJavaScriptToken});
CodeMirror.defineInitHook(function(codeMirror) {
codeMirror.on("scrollCursorIntoView", scrollCursorIntoView);
});
let whitespaceStyleElement = null;
let whitespaceCountsWithStyling = new Set;
CodeMirror.defineOption("showWhitespaceCharacters", false, function(cm, value, old) {
if (!value || (old && old !== CodeMirror.Init)) {
cm.removeOverlay("whitespace");
return;
}
cm.addOverlay({
name: "whitespace",
token(stream) {
if (stream.peek() === " ") {
let count = 0;
while (stream.peek() === " ") {
++count;
stream.next();
}
if (!whitespaceCountsWithStyling.has(count)) {
whitespaceCountsWithStyling.add(count);
if (!whitespaceStyleElement)
whitespaceStyleElement = document.head.appendChild(document.createElement("style"));
const middleDot = "\\00B7";
let styleText = whitespaceStyleElement.textContent;
styleText += `.show-whitespace-characters .CodeMirror .cm-whitespace-${count}::before {`;
styleText += `content: "${middleDot.repeat(count)}";`;
styleText += `}`;
whitespaceStyleElement.textContent = styleText;
}
return `whitespace whitespace-${count}`;
}
while (!stream.eol() && stream.peek() !== " ")
stream.next();
return null;
}
});
});
CodeMirror.defineExtension("hasLineClass", function(line, where, className) {
// This matches the arguments to addLineClass and removeLineClass.
var classProperty = where === "text" ? "textClass" : (where === "background" ? "bgClass" : "wrapClass");
var lineInfo = this.lineInfo(line);
if (!lineInfo)
return false;
if (!lineInfo[classProperty])
return false;
// Test for the simple case.
if (lineInfo[classProperty] === className)
return true;
// Do a quick check for the substring. This is faster than a regex, which requires escaping the input first.
var index = lineInfo[classProperty].indexOf(className);
if (index === -1)
return false;
// Check that it is surrounded by spaces. Add padding spaces first to work with beginning and end of string cases.
var paddedClass = " " + lineInfo[classProperty] + " ";
return paddedClass.indexOf(" " + className + " ", index) !== -1;
});
CodeMirror.defineExtension("setUniqueBookmark", function(position, options) {
var marks = this.findMarksAt(position);
for (var i = 0; i < marks.length; ++i) {
if (marks[i].__uniqueBookmark) {
marks[i].clear();
break;
}
}
var uniqueBookmark = this.setBookmark(position, options);
uniqueBookmark.__uniqueBookmark = true;
return uniqueBookmark;
});
CodeMirror.defineExtension("toggleLineClass", function(line, where, className) {
if (this.hasLineClass(line, where, className)) {
this.removeLineClass(line, where, className);
return false;
}
this.addLineClass(line, where, className);
return true;
});
CodeMirror.defineExtension("alterNumberInRange", function(amount, startPosition, endPosition, updateSelection) {
// We don't try if the range is multiline, pass to another key handler.
if (startPosition.line !== endPosition.line)
return false;
if (updateSelection) {
// Remember the cursor position/selection.
var selectionStart = this.getCursor("start");
var selectionEnd = this.getCursor("end");
}
var line = this.getLine(startPosition.line);
var foundPeriod = false;
var start = NaN;
var end = NaN;
for (var i = startPosition.ch; i >= 0; --i) {
var character = line.charAt(i);
if (character === ".") {
if (foundPeriod)
break;
foundPeriod = true;
} else if (character !== "-" && character !== "+" && isNaN(parseInt(character))) {
// Found the end already, just scan backwards.
if (i === startPosition.ch) {
end = i;
continue;
}
break;
}
start = i;
}
if (isNaN(end)) {
for (var i = startPosition.ch + 1; i < line.length; ++i) {
var character = line.charAt(i);
if (character === ".") {
if (foundPeriod) {
end = i;
break;
}
foundPeriod = true;
} else if (isNaN(parseInt(character))) {
end = i;
break;
}
end = i + 1;
}
}
// No number range found, pass to another key handler.
if (isNaN(start) || isNaN(end))
return false;
var number = parseFloat(line.substring(start, end));
// Make the new number and constrain it to a precision of 6, this matches numbers the engine returns.
// Use the Number constructor to forget the fixed precision, so 1.100000 will print as 1.1.
var alteredNumber = Number((number + amount).toFixed(6));
var alteredNumberString = alteredNumber.toString();
var from = {line: startPosition.line, ch: start};
var to = {line: startPosition.line, ch: end};
this.replaceRange(alteredNumberString, from, to);
if (updateSelection) {
var previousLength = to.ch - from.ch;
var newLength = alteredNumberString.length;
// Fix up the selection so it follows the increase or decrease in the replacement length.
// selectionStart/End may the same object if there is no selection. If that is the case
// make only one modification to prevent a double adjustment, and keep it a single object
// to avoid CodeMirror inadvertently creating an actual selection range.
let diff = newLength - previousLength;
if (selectionStart === selectionEnd)
selectionStart.ch += diff;
else {
if (selectionStart.ch > from.ch)
selectionStart.ch += diff;
if (selectionEnd.ch > from.ch)
selectionEnd.ch += diff;
}
this.setSelection(selectionStart, selectionEnd);
}
return true;
});
function alterNumber(amount, codeMirror)
{
function findNumberToken(position)
{
// CodeMirror includes the unit in the number token, so searching for
// number tokens is the best way to get both the number and unit.
var token = codeMirror.getTokenAt(position);
if (token && token.type && /\bnumber\b/.test(token.type))
return token;
return null;
}
var position = codeMirror.getCursor("head");
var token = findNumberToken(position);
if (!token) {
// If the cursor is at the outside beginning of the token, the previous
// findNumberToken wont find it. So check the next column for a number too.
position.ch += 1;
token = findNumberToken(position);
}
if (!token)
return CodeMirror.Pass;
var foundNumber = codeMirror.alterNumberInRange(amount, {ch: token.start, line: position.line}, {ch: token.end, line: position.line}, true);
if (!foundNumber)
return CodeMirror.Pass;
}
CodeMirror.defineExtension("rectsForRange", function(range) {
var lineRects = [];
for (var line = range.start.line; line <= range.end.line; ++line) {
var lineContent = this.getLine(line);
var startChar = line === range.start.line ? range.start.ch : (lineContent.length - lineContent.trimLeft().length);
var endChar = line === range.end.line ? range.end.ch : lineContent.length;
var firstCharCoords = this.cursorCoords({ch: startChar, line});
var endCharCoords = this.cursorCoords({ch: endChar, line});
// Handle line wrapping.
if (firstCharCoords.bottom !== endCharCoords.bottom) {
var maxY = -Number.MAX_VALUE;
for (var ch = startChar; ch <= endChar; ++ch) {
var coords = this.cursorCoords({ch, line});
if (coords.bottom > maxY) {
if (ch > startChar) {
var maxX = Math.ceil(this.cursorCoords({ch: ch - 1, line}).right);
lineRects.push(new WI.Rect(minX, minY, maxX - minX, maxY - minY));
}
var minX = Math.floor(coords.left);
var minY = Math.floor(coords.top);
maxY = Math.ceil(coords.bottom);
}
}
maxX = Math.ceil(coords.right);
lineRects.push(new WI.Rect(minX, minY, maxX - minX, maxY - minY));
} else {
var minX = Math.floor(firstCharCoords.left);
var minY = Math.floor(firstCharCoords.top);
var maxX = Math.ceil(endCharCoords.right);
var maxY = Math.ceil(endCharCoords.bottom);
lineRects.push(new WI.Rect(minX, minY, maxX - minX, maxY - minY));
}
}
return lineRects;
});
let mac = WI.Platform.name === "mac";
CodeMirror.keyMap["default"] = {
"Alt-Up": alterNumber.bind(null, 1),
"Ctrl-Alt-Up": alterNumber.bind(null, 0.1),
"Shift-Alt-Up": alterNumber.bind(null, 10),
"Alt-PageUp": alterNumber.bind(null, 10),
"Shift-Alt-PageUp": alterNumber.bind(null, 100),
"Alt-Down": alterNumber.bind(null, -1),
"Ctrl-Alt-Down": alterNumber.bind(null, -0.1),
"Shift-Alt-Down": alterNumber.bind(null, -10),
"Alt-PageDown": alterNumber.bind(null, -10),
"Shift-Alt-PageDown": alterNumber.bind(null, -100),
"Cmd-/": "toggleComment",
"Cmd-D": "selectNextOccurrence",
"Shift-Tab": "indentLess",
fallthrough: mac ? "macDefault" : "pcDefault"
};
{
// CodeMirror's default behavior is to always insert a tab ("\t") regardless of `indentWithTabs`.
let original = CodeMirror.commands.insertTab;
CodeMirror.commands.insertTab = function(cm) {
if (cm.options.indentWithTabs)
original(cm);
else
CodeMirror.commands.insertSoftTab(cm);
};
}
// Register some extra MIME-types for CodeMirror. These are in addition to the
// ones CodeMirror already registers, like text/html, text/javascript, etc.
var extraXMLTypes = ["text/xml", "text/xsl"];
extraXMLTypes.forEach(function(type) {
CodeMirror.defineMIME(type, "xml");
});
var extraHTMLTypes = ["application/xhtml+xml", "image/svg+xml"];
extraHTMLTypes.forEach(function(type) {
CodeMirror.defineMIME(type, "htmlmixed");
});
var extraJavaScriptTypes = ["text/ecmascript", "application/javascript", "application/ecmascript", "application/x-javascript",
"text/x-javascript", "text/javascript1.1", "text/javascript1.2", "text/javascript1.3", "text/jscript", "text/livescript"];
extraJavaScriptTypes.forEach(function(type) {
CodeMirror.defineMIME(type, "javascript");
});
var extraJSONTypes = ["application/x-json", "text/x-json", "application/vnd.api+json"];
extraJSONTypes.forEach(function(type) {
CodeMirror.defineMIME(type, {name: "javascript", json: true});
});
// FIXME: Add WHLSL specific modes.
CodeMirror.defineMIME("x-pipeline/x-compute", CodeMirror.resolveMode("x-shader/x-vertex"));
CodeMirror.defineMIME("x-pipeline/x-render", CodeMirror.resolveMode("x-shader/x-vertex"));
})();
WI.compareCodeMirrorPositions = function(a, b)
{
var lineCompare = a.line - b.line;
if (lineCompare !== 0)
return lineCompare;
var aColumn = "ch" in a ? a.ch : Number.MAX_VALUE;
var bColumn = "ch" in b ? b.ch : Number.MAX_VALUE;
return aColumn - bColumn;
};
WI.walkTokens = function(cm, mode, initialPosition, callback)
{
let state = CodeMirror.copyState(mode, cm.getTokenAt(initialPosition).state);
if (state.localState)
state = state.localState;
let lineCount = cm.lineCount();
let abort = false;
for (let lineNumber = initialPosition.line; !abort && lineNumber < lineCount; ++lineNumber) {
let line = cm.getLine(lineNumber);
let stream = new CodeMirror.StringStream(line);
if (lineNumber === initialPosition.line)
stream.start = stream.pos = initialPosition.ch;
while (!stream.eol()) {
let tokenType = mode.token(stream, state);
if (!callback(tokenType, stream.current())) {
abort = true;
break;
}
stream.start = stream.pos;
}
}
if (!abort)
callback(null);
};
WI.tokenizeCSSValue = function(cssValue)
{
const rulePrefix = "*{X:";
let cssRule = rulePrefix + cssValue + "}";
let tokens = [];
let mode = CodeMirror.getMode({indentUnit: 0}, "text/css");
let state = CodeMirror.startState(mode);
let stream = new CodeMirror.StringStream(cssRule);
function processToken(token, tokenType, column) {
if (column < rulePrefix.length)
return;
if (token === "}" && !tokenType)
return;
tokens.push({value: token, type: tokenType});
}
while (!stream.eol()) {
let style = mode.token(stream, state);
let value = stream.current();
processToken(value, style, stream.start);
stream.start = stream.pos;
}
return tokens;
};