|
@@ -0,0 +1,205 @@
|
|
|
+/*---------------------------------------------------------------------------------------------
|
|
|
+ * Copyright (c) Microsoft Corporation. All rights reserved.
|
|
|
+ * Licensed under the MIT License. See License.txt in the project root for license information.
|
|
|
+ *--------------------------------------------------------------------------------------------*/
|
|
|
+
|
|
|
+'use strict';
|
|
|
+
|
|
|
+import IRichLanguageConfiguration = monaco.languages.LanguageConfiguration;
|
|
|
+import ILanguage = monaco.languages.IMonarchLanguage;
|
|
|
+
|
|
|
+const EMPTY_ELEMENTS:string[] = ['area', 'base', 'br', 'col', 'embed', 'hr', 'img', 'input', 'keygen', 'link', 'menuitem', 'meta', 'param', 'source', 'track', 'wbr'];
|
|
|
+
|
|
|
+export var conf:IRichLanguageConfiguration = {
|
|
|
+ wordPattern: /(-?\d*\.\d\w*)|([^\`\~\!\@\$\^\&\*\(\)\=\+\[\{\]\}\\\|\;\:\'\"\,\.\<\>\/\s]+)/g,
|
|
|
+
|
|
|
+ comments: {
|
|
|
+ blockComment: ['<!--', '-->']
|
|
|
+ },
|
|
|
+
|
|
|
+ brackets: [
|
|
|
+ ['<!--', '-->'],
|
|
|
+ ['<', '>'],
|
|
|
+ ],
|
|
|
+
|
|
|
+ __electricCharacterSupport: {
|
|
|
+ embeddedElectricCharacters: ['*', '}', ']', ')']
|
|
|
+ },
|
|
|
+
|
|
|
+ autoClosingPairs: [
|
|
|
+ { open: '{', close: '}' },
|
|
|
+ { open: '[', close: ']' },
|
|
|
+ { open: '(', close: ')' },
|
|
|
+ { open: '"', close: '"' },
|
|
|
+ { open: '\'', close: '\'' }
|
|
|
+ ],
|
|
|
+
|
|
|
+ surroundingPairs: [
|
|
|
+ { open: '"', close: '"' },
|
|
|
+ { open: '\'', close: '\'' }
|
|
|
+ ],
|
|
|
+
|
|
|
+ onEnterRules: [
|
|
|
+ {
|
|
|
+ beforeText: new RegExp(`<(?!(?:${EMPTY_ELEMENTS.join('|')}))([_:\\w][_:\\w-.\\d]*)([^/>]*(?!/)>)[^<]*$`, 'i'),
|
|
|
+ afterText: /^<\/([_:\w][_:\w-.\d]*)\s*>$/i,
|
|
|
+ action: { indentAction: monaco.languages.IndentAction.IndentOutdent }
|
|
|
+ },
|
|
|
+ {
|
|
|
+ beforeText: new RegExp(`<(?!(?:${EMPTY_ELEMENTS.join('|')}))(\\w[\\w\\d]*)([^/>]*(?!/)>)[^<]*$`, 'i'),
|
|
|
+ action: { indentAction: monaco.languages.IndentAction.Indent }
|
|
|
+ }
|
|
|
+ ],
|
|
|
+};
|
|
|
+
|
|
|
+export var language = <ILanguage> {
|
|
|
+ defaultToken: '',
|
|
|
+ tokenPostfix: '.html',
|
|
|
+
|
|
|
+ // The main tokenizer for our languages
|
|
|
+ tokenizer: {
|
|
|
+ root: [
|
|
|
+ [/<!DOCTYPE/, 'metatag', '@doctype'],
|
|
|
+ [/<!--/, 'comment', '@comment'],
|
|
|
+ [/<\w+\/>/, 'tag'],
|
|
|
+ [/<script/, 'tag', '@script'],
|
|
|
+ [/<style/, 'tag', '@style'],
|
|
|
+ [/<\w+/, 'tag', '@otherTag'],
|
|
|
+ [/<\/\w+/, 'tag', '@otherTag']
|
|
|
+ ],
|
|
|
+
|
|
|
+ doctype: [
|
|
|
+ [/[^>]+/, 'metatag.content' ],
|
|
|
+ [/>/, 'metatag', '@pop' ],
|
|
|
+ ],
|
|
|
+
|
|
|
+ comment: [
|
|
|
+ [/-->/, 'comment', '@pop'],
|
|
|
+ [/[^-]+/, 'comment'],
|
|
|
+ [/./, 'comment']
|
|
|
+ ],
|
|
|
+
|
|
|
+ otherTag: [
|
|
|
+ [/\/?>/, 'tag', '@pop'],
|
|
|
+ [/"([^"]*)"/, 'attribute.value'],
|
|
|
+ [/'([^']*)'/, 'attribute.value'],
|
|
|
+ [/[\w\-]+/, 'attribute.name'],
|
|
|
+ [/=/, 'delimiter']
|
|
|
+ ],
|
|
|
+
|
|
|
+ // -- BEGIN <script> tags handling
|
|
|
+
|
|
|
+ // After <script
|
|
|
+ script: [
|
|
|
+ [/type/, 'attribute.name', '@scriptAfterType'],
|
|
|
+ [/"([^"]*)"/, 'attribute.value'],
|
|
|
+ [/'([^']*)'/, 'attribute.value'],
|
|
|
+ [/[\w\-]+/, 'attribute.name'],
|
|
|
+ [/=/, 'delimiter'],
|
|
|
+ [/>/, { token: 'tag', next: '@scriptEmbedded', nextEmbedded: 'text/javascript'} ],
|
|
|
+ [/<\/script\s*>/, 'tag', '@pop']
|
|
|
+ ],
|
|
|
+
|
|
|
+ // After <script ... type
|
|
|
+ scriptAfterType: [
|
|
|
+ [/=/,'delimiter', '@scriptAfterTypeEquals'],
|
|
|
+ [/<\/script\s*>/, { token: '@rematch', next: '@pop' }]
|
|
|
+ ],
|
|
|
+
|
|
|
+ // After <script ... type =
|
|
|
+ scriptAfterTypeEquals: [
|
|
|
+ [/"([^"]*)"/, { token: 'attribute.value', switchTo: '@scriptWithCustomType.$1' } ],
|
|
|
+ [/'([^']*)'/, { token: 'attribute.value', switchTo: '@scriptWithCustomType.$1' } ],
|
|
|
+ [/<\/script\s*>/, { token: '@rematch', next: '@pop' }]
|
|
|
+ ],
|
|
|
+
|
|
|
+ // After <script ... type = $S2
|
|
|
+ scriptWithCustomType: [
|
|
|
+ [/>/, { token: 'tag', next: '@scriptEmbedded', nextEmbedded: '$S2'}],
|
|
|
+ [/<\/script\s*>/, { token: '@rematch', next: '@pop' }]
|
|
|
+ ],
|
|
|
+
|
|
|
+ scriptEmbedded: [
|
|
|
+ [/<\/script/, { token: '@rematch', next: '@pop', nextEmbedded: '@pop' }]
|
|
|
+ ],
|
|
|
+
|
|
|
+ // -- END <script> tags handling
|
|
|
+
|
|
|
+
|
|
|
+ // -- BEGIN <style> tags handling
|
|
|
+
|
|
|
+ // After <style
|
|
|
+ style: [
|
|
|
+ [/type/, 'attribute.name', '@styleAfterType'],
|
|
|
+ [/"([^"]*)"/, 'attribute.value'],
|
|
|
+ [/'([^']*)'/, 'attribute.value'],
|
|
|
+ [/[\w\-]+/, 'attribute.name'],
|
|
|
+ [/=/, 'delimiter'],
|
|
|
+ [/>/, { token: 'tag', next: '@styleEmbedded', nextEmbedded: 'text/css'} ],
|
|
|
+ [/<\/style\s*>/, 'tag', '@pop']
|
|
|
+ ],
|
|
|
+
|
|
|
+ // After <style ... type
|
|
|
+ styleAfterType: [
|
|
|
+ [/=/,'delimiter', '@styleAfterTypeEquals'],
|
|
|
+ [/<\/style\s*>/, { token: '@rematch', next: '@pop' }]
|
|
|
+ ],
|
|
|
+
|
|
|
+ // After <style ... type =
|
|
|
+ styleAfterTypeEquals: [
|
|
|
+ [/"([^"]*)"/, { token: 'attribute.value', switchTo: '@styleWithCustomType.$1' } ],
|
|
|
+ [/'([^']*)'/, { token: 'attribute.value', switchTo: '@styleWithCustomType.$1' } ],
|
|
|
+ [/<\/style\s*>/, { token: '@rematch', next: '@pop' }]
|
|
|
+ ],
|
|
|
+
|
|
|
+ // After <style ... type = $S2
|
|
|
+ styleWithCustomType: [
|
|
|
+ [/>/, { token: 'tag', next: '@styleEmbedded', nextEmbedded: '$S2'}],
|
|
|
+ [/<\/style\s*>/, { token: '@rematch', next: '@pop' }]
|
|
|
+ ],
|
|
|
+
|
|
|
+ styleEmbedded: [
|
|
|
+ [/<\/style/, { token: '@rematch', next: '@pop', nextEmbedded: '@pop' }]
|
|
|
+ ],
|
|
|
+
|
|
|
+ // -- END <style> tags handling
|
|
|
+ },
|
|
|
+};
|
|
|
+
|
|
|
+// TESTED WITH:
|
|
|
+
|
|
|
+// <!DOCTYPE html>
|
|
|
+// <html>
|
|
|
+// <head>
|
|
|
+// <title>Monarch Workbench</title>
|
|
|
+// <meta http-equiv="X-UA-Compatible" content="IE=edge" />
|
|
|
+// <!----
|
|
|
+// -- -- -- a comment -- -- --
|
|
|
+// ---->
|
|
|
+// <style bah="bah">
|
|
|
+// body { font-family: Consolas; } /* nice */
|
|
|
+// </style>
|
|
|
+// </head
|
|
|
+// >
|
|
|
+// a = "asd"
|
|
|
+// <body>
|
|
|
+// <br/>
|
|
|
+// <div
|
|
|
+// class
|
|
|
+// =
|
|
|
+// "test"
|
|
|
+// >
|
|
|
+// <script>
|
|
|
+// function() {
|
|
|
+// alert("hi </ script>"); // javascript
|
|
|
+// };
|
|
|
+// </script>
|
|
|
+// <script
|
|
|
+// bah="asdfg"
|
|
|
+// type="text/css"
|
|
|
+// >
|
|
|
+// .bar { text-decoration: underline; }
|
|
|
+// </script>
|
|
|
+// </div>
|
|
|
+// </body>
|
|
|
+// </html>
|