226 lines
5.9 KiB
JavaScript
226 lines
5.9 KiB
JavaScript
/**
|
|
* @param {string} value
|
|
* @returns {RegExp}
|
|
* */
|
|
|
|
/**
|
|
* @param {RegExp | string } re
|
|
* @returns {string}
|
|
*/
|
|
function source(re) {
|
|
if (!re) return null;
|
|
if (typeof re === "string") return re;
|
|
|
|
return re.source;
|
|
}
|
|
|
|
/**
|
|
* @param {...(RegExp | string) } args
|
|
* @returns {string}
|
|
*/
|
|
function concat(...args) {
|
|
const joined = args.map((x) => source(x)).join("");
|
|
return joined;
|
|
}
|
|
|
|
/*
|
|
Language: QML
|
|
Requires: javascript.js, xml.js
|
|
Author: John Foster <jfoster@esri.com>
|
|
Description: Syntax highlighting for the Qt Quick QML scripting language, based mostly off
|
|
the JavaScript parser.
|
|
Website: https://doc.qt.io/qt-5/qmlapplications.html
|
|
Category: scripting
|
|
*/
|
|
|
|
function qml(hljs) {
|
|
const KEYWORDS = {
|
|
keyword:
|
|
'in of on if for while finally var new function do return void else break catch ' +
|
|
'instanceof with throw case default try this switch continue typeof delete ' +
|
|
'let yield const export super debugger as async await import',
|
|
literal:
|
|
'true false null undefined NaN Infinity',
|
|
built_in:
|
|
'eval isFinite isNaN parseFloat parseInt decodeURI decodeURIComponent ' +
|
|
'encodeURI encodeURIComponent escape unescape Object Function Boolean Error ' +
|
|
'EvalError InternalError RangeError ReferenceError StopIteration SyntaxError ' +
|
|
'TypeError URIError Number Math Date String RegExp Array Float32Array ' +
|
|
'Float64Array Int16Array Int32Array Int8Array Uint16Array Uint32Array ' +
|
|
'Uint8Array Uint8ClampedArray ArrayBuffer DataView JSON Intl arguments require ' +
|
|
'module console window document Symbol Set Map WeakSet WeakMap Proxy Reflect ' +
|
|
'Behavior bool color coordinate date double enumeration font geocircle georectangle ' +
|
|
'geoshape int list matrix4x4 parent point quaternion real rect ' +
|
|
'size string url variant vector2d vector3d vector4d ' +
|
|
'Promise'
|
|
};
|
|
|
|
const QML_IDENT_RE = '[a-zA-Z_][a-zA-Z0-9\\._]*';
|
|
|
|
// Isolate property statements. Ends at a :, =, ;, ,, a comment or end of line.
|
|
// Use property class.
|
|
const PROPERTY = {
|
|
className: 'keyword',
|
|
begin: '\\bproperty\\b',
|
|
starts: {
|
|
className: 'string',
|
|
end: '(:|=|;|,|//|/\\*|$)',
|
|
returnEnd: true
|
|
}
|
|
};
|
|
|
|
// Isolate signal statements. Ends at a ) a comment or end of line.
|
|
// Use property class.
|
|
const SIGNAL = {
|
|
className: 'keyword',
|
|
begin: '\\bsignal\\b',
|
|
starts: {
|
|
className: 'string',
|
|
end: '(\\(|:|=|;|,|//|/\\*|$)',
|
|
returnEnd: true
|
|
}
|
|
};
|
|
|
|
// id: is special in QML. When we see id: we want to mark the id: as attribute and
|
|
// emphasize the token following.
|
|
const ID_ID = {
|
|
className: 'attribute',
|
|
begin: '\\bid\\s*:',
|
|
starts: {
|
|
className: 'string',
|
|
end: QML_IDENT_RE,
|
|
returnEnd: false
|
|
}
|
|
};
|
|
|
|
// Find QML object attribute. An attribute is a QML identifier followed by :.
|
|
// Unfortunately it's hard to know where it ends, as it may contain scalars,
|
|
// objects, object definitions, or javascript. The true end is either when the parent
|
|
// ends or the next attribute is detected.
|
|
const QML_ATTRIBUTE = {
|
|
begin: QML_IDENT_RE + '\\s*:',
|
|
returnBegin: true,
|
|
contains: [
|
|
{
|
|
className: 'attribute',
|
|
begin: QML_IDENT_RE,
|
|
end: '\\s*:',
|
|
excludeEnd: true,
|
|
relevance: 0
|
|
}
|
|
],
|
|
relevance: 0
|
|
};
|
|
|
|
// Find QML object. A QML object is a QML identifier followed by { and ends at the matching }.
|
|
// All we really care about is finding IDENT followed by { and just mark up the IDENT and ignore the {.
|
|
const QML_OBJECT = {
|
|
begin: concat(QML_IDENT_RE, /\s*\{/),
|
|
end: /\{/,
|
|
returnBegin: true,
|
|
relevance: 0,
|
|
contains: [
|
|
hljs.inherit(hljs.TITLE_MODE, {
|
|
begin: QML_IDENT_RE
|
|
})
|
|
]
|
|
};
|
|
|
|
return {
|
|
name: 'QML',
|
|
aliases: [ 'qt' ],
|
|
case_insensitive: false,
|
|
keywords: KEYWORDS,
|
|
contains: [
|
|
{
|
|
className: 'meta',
|
|
begin: /^\s*['"]use (strict|asm)['"]/
|
|
},
|
|
hljs.APOS_STRING_MODE,
|
|
hljs.QUOTE_STRING_MODE,
|
|
{ // template string
|
|
className: 'string',
|
|
begin: '`',
|
|
end: '`',
|
|
contains: [
|
|
hljs.BACKSLASH_ESCAPE,
|
|
{
|
|
className: 'subst',
|
|
begin: '\\$\\{',
|
|
end: '\\}'
|
|
}
|
|
]
|
|
},
|
|
hljs.C_LINE_COMMENT_MODE,
|
|
hljs.C_BLOCK_COMMENT_MODE,
|
|
{
|
|
className: 'number',
|
|
variants: [
|
|
{
|
|
begin: '\\b(0[bB][01]+)'
|
|
},
|
|
{
|
|
begin: '\\b(0[oO][0-7]+)'
|
|
},
|
|
{
|
|
begin: hljs.C_NUMBER_RE
|
|
}
|
|
],
|
|
relevance: 0
|
|
},
|
|
{ // "value" container
|
|
begin: '(' + hljs.RE_STARTERS_RE + '|\\b(case|return|throw)\\b)\\s*',
|
|
keywords: 'return throw case',
|
|
contains: [
|
|
hljs.C_LINE_COMMENT_MODE,
|
|
hljs.C_BLOCK_COMMENT_MODE,
|
|
hljs.REGEXP_MODE,
|
|
{ // E4X / JSX
|
|
begin: /</,
|
|
end: />\s*[);\]]/,
|
|
relevance: 0,
|
|
subLanguage: 'xml'
|
|
}
|
|
],
|
|
relevance: 0
|
|
},
|
|
SIGNAL,
|
|
PROPERTY,
|
|
{
|
|
className: 'function',
|
|
beginKeywords: 'function',
|
|
end: /\{/,
|
|
excludeEnd: true,
|
|
contains: [
|
|
hljs.inherit(hljs.TITLE_MODE, {
|
|
begin: /[A-Za-z$_][0-9A-Za-z$_]*/
|
|
}),
|
|
{
|
|
className: 'params',
|
|
begin: /\(/,
|
|
end: /\)/,
|
|
excludeBegin: true,
|
|
excludeEnd: true,
|
|
contains: [
|
|
hljs.C_LINE_COMMENT_MODE,
|
|
hljs.C_BLOCK_COMMENT_MODE
|
|
]
|
|
}
|
|
],
|
|
illegal: /\[|%/
|
|
},
|
|
{
|
|
// hack: prevents detection of keywords after dots
|
|
begin: '\\.' + hljs.IDENT_RE,
|
|
relevance: 0
|
|
},
|
|
ID_ID,
|
|
QML_ATTRIBUTE,
|
|
QML_OBJECT
|
|
],
|
|
illegal: /#/
|
|
};
|
|
}
|
|
|
|
module.exports = qml;
|