mirror of
https://github.com/microsoft/monaco-editor.git
synced 2025-12-22 22:02:55 +01:00
Finish implementing Scala
This commit is contained in:
parent
a467bb0774
commit
b2a10419ce
3 changed files with 187 additions and 42 deletions
15
src/scala/scala.contribution.ts
Normal file
15
src/scala/scala.contribution.ts
Normal file
|
|
@ -0,0 +1,15 @@
|
||||||
|
/*---------------------------------------------------------------------------------------------
|
||||||
|
* 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 { registerLanguage } from '../_.contribution';
|
||||||
|
|
||||||
|
registerLanguage({
|
||||||
|
id: 'scala',
|
||||||
|
extensions: ['.scala', '.sbt', '.sc'],
|
||||||
|
aliases: ['Scala', 'scala', 'SBT', 'Sbt', 'sbt', 'Dotty', 'dotty'],
|
||||||
|
mimetypes: ['text/x-scala', 'text/x-sbt', 'text/x-dotty'],
|
||||||
|
loader: () => import('./scala')
|
||||||
|
});
|
||||||
38
src/scala/scala.test.ts
Normal file
38
src/scala/scala.test.ts
Normal file
|
|
@ -0,0 +1,38 @@
|
||||||
|
/*---------------------------------------------------------------------------------------------
|
||||||
|
* 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 { testTokenization } from '../test/testRunner';
|
||||||
|
|
||||||
|
testTokenization('scala', [
|
||||||
|
[{
|
||||||
|
line: '//',
|
||||||
|
tokens: [
|
||||||
|
{startIndex: 0, type: 'comment.scala'}
|
||||||
|
]
|
||||||
|
}],
|
||||||
|
|
||||||
|
[{
|
||||||
|
line: ' // a comment',
|
||||||
|
tokens: [
|
||||||
|
{startIndex: 0, type: 'white.scala'},
|
||||||
|
{startIndex: 4, type: 'comment.scala'}
|
||||||
|
]
|
||||||
|
}],
|
||||||
|
|
||||||
|
[{
|
||||||
|
line: 'var a = 1',
|
||||||
|
tokens: [
|
||||||
|
{startIndex: 0, type: 'keyword.scala'},
|
||||||
|
{startIndex: 3, type: 'white.scala'},
|
||||||
|
{startIndex: 4, type: 'variable.scala'},
|
||||||
|
{startIndex: 5, type: 'white.scala'},
|
||||||
|
{startIndex: 6, type: 'operator.scala'},
|
||||||
|
{startIndex: 7, type: 'white.scala'},
|
||||||
|
{startIndex: 8, type: 'number.scala'}
|
||||||
|
]
|
||||||
|
}]
|
||||||
|
]);
|
||||||
|
|
@ -9,8 +9,48 @@ import IRichLanguageConfiguration = monaco.languages.LanguageConfiguration;
|
||||||
import ILanguage = monaco.languages.IMonarchLanguage;
|
import ILanguage = monaco.languages.IMonarchLanguage;
|
||||||
|
|
||||||
export const conf: IRichLanguageConfiguration = {
|
export const conf: IRichLanguageConfiguration = {
|
||||||
|
/*
|
||||||
|
* `...` is allowed as an identifier.
|
||||||
|
* $ is allowed in identifiers.
|
||||||
|
* unary_<op> is allowed as an identifier.
|
||||||
|
* <name>_= is allowed as an identifier.
|
||||||
|
*/
|
||||||
|
wordPattern: /(unary_[@~!#%^&*()\-=+\\|:<>\/?]+)|([a-zA-Z_$][\w$]*?_=)|(`[^`]+`)|([a-zA-Z_$][\w$]*)/g,
|
||||||
|
comments: {
|
||||||
|
lineComment: '//',
|
||||||
|
blockComment: ['/*', '*/'],
|
||||||
|
},
|
||||||
|
brackets: [
|
||||||
|
['{', '}'],
|
||||||
|
['[', ']'],
|
||||||
|
['(', ')'],
|
||||||
|
],
|
||||||
|
autoClosingPairs: [
|
||||||
|
{ open: '{', close: '}' },
|
||||||
|
{ open: '[', close: ']' },
|
||||||
|
{ open: '(', close: ')' },
|
||||||
|
{ open: '"', close: '"' },
|
||||||
|
{ open: '\'', close: '\'' },
|
||||||
|
],
|
||||||
|
surroundingPairs: [
|
||||||
|
{ open: '{', close: '}' },
|
||||||
|
{ open: '[', close: ']' },
|
||||||
|
{ open: '(', close: ')' },
|
||||||
|
{ open: '"', close: '"' },
|
||||||
|
{ open: '\'', close: '\'' },
|
||||||
|
],
|
||||||
|
folding: {
|
||||||
|
markers: {
|
||||||
|
start: new RegExp("^\\s*//\\s*(?:(?:#?region\\b)|(?:<editor-fold\\b))"),
|
||||||
|
end: new RegExp("^\\s*//\\s*(?:(?:#?endregion\\b)|(?:</editor-fold>))")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
export const language = <ILanguage>{
|
||||||
tokenPostfix: '.scala',
|
tokenPostfix: '.scala',
|
||||||
|
|
||||||
|
// We can't easily add everything from Dotty, but we can at least add some of its keywords
|
||||||
keywords: [
|
keywords: [
|
||||||
'asInstanceOf',
|
'asInstanceOf',
|
||||||
'catch', 'class', 'classOf',
|
'catch', 'class', 'classOf',
|
||||||
|
|
@ -18,7 +58,7 @@ export const conf: IRichLanguageConfiguration = {
|
||||||
'else', 'extends',
|
'else', 'extends',
|
||||||
'finally', 'for', 'foreach', 'forSome',
|
'finally', 'for', 'foreach', 'forSome',
|
||||||
'if', 'import', 'isInstanceOf',
|
'if', 'import', 'isInstanceOf',
|
||||||
'match',
|
'macro', 'match',
|
||||||
'new',
|
'new',
|
||||||
'object',
|
'object',
|
||||||
'package',
|
'package',
|
||||||
|
|
@ -27,7 +67,16 @@ export const conf: IRichLanguageConfiguration = {
|
||||||
'until',
|
'until',
|
||||||
'val', 'var',
|
'val', 'var',
|
||||||
'while', 'with',
|
'while', 'with',
|
||||||
'yield'
|
'yield',
|
||||||
|
|
||||||
|
// Dotty-specific:
|
||||||
|
'given', 'enum', 'then'
|
||||||
|
],
|
||||||
|
|
||||||
|
// Dotty-specific:
|
||||||
|
softKeywords: [
|
||||||
|
'as', 'export', 'extension', 'end',
|
||||||
|
'derives', 'on'
|
||||||
],
|
],
|
||||||
|
|
||||||
constants: [
|
constants: [
|
||||||
|
|
@ -40,10 +89,16 @@ export const conf: IRichLanguageConfiguration = {
|
||||||
'private', 'protected', 'sealed'
|
'private', 'protected', 'sealed'
|
||||||
],
|
],
|
||||||
|
|
||||||
name: /[a-z_$][\w$]*/,
|
// Dotty-specific:
|
||||||
|
softModifiers: [
|
||||||
|
'inline', 'opaque', 'open', 'transparent', 'using'
|
||||||
|
],
|
||||||
|
|
||||||
|
name: /(?:[a-z_$][\w$]*|`[^`]+`)/,
|
||||||
|
type: /(?:[A-Z][\w$]*)/,
|
||||||
|
|
||||||
// we include these common regular expressions
|
// we include these common regular expressions
|
||||||
symbols: /[=><!~?:&|+\-*\/\^%@#]+/,
|
symbols: /[=><!~?:&|+\-*\/^\\%@#]+/,
|
||||||
|
|
||||||
// C# style strings
|
// C# style strings
|
||||||
escapes: /\\(?:[btnfr\\"']|x[0-9A-Fa-f]{1,4}|u[0-9A-Fa-f]{4}|U[0-9A-Fa-f]{8})/,
|
escapes: /\\(?:[btnfr\\"']|x[0-9A-Fa-f]{1,4}|u[0-9A-Fa-f]{4}|U[0-9A-Fa-f]{8})/,
|
||||||
|
|
@ -67,32 +122,31 @@ export const conf: IRichLanguageConfiguration = {
|
||||||
[/"/, {token: 'string.quote', bracket: '@open', next: '@string'}],
|
[/"/, {token: 'string.quote', bracket: '@open', next: '@string'}],
|
||||||
|
|
||||||
// numbers
|
// numbers
|
||||||
[/[+\-]?(?:\d[_\d])*\.\d+[dDfFlL]?([eE][\-+]?\d+)?/, 'number.float'],
|
[/[+\-]?(?:\d[_\d])*\.\d+[dDfFlL]?([eE][\-+]?\d+)?/, 'number.float', '@allowMethod'],
|
||||||
[/0[xX][0-9a-fA-F]+/, 'number.hex'],
|
[/0[xX][0-9a-fA-F]+/, 'number.hex', '@allowMethod'],
|
||||||
[/[+\-]?\d[_\d]*[dDfFlL]?/, 'number'],
|
[/[+\-]?\d[_\d]*[dDfFlL]?/, 'number', '@allowMethod'],
|
||||||
|
|
||||||
[/\b_\*/, 'key'],
|
[/\b_\*/, 'key'],
|
||||||
[/\b(_)(\b)/, 'keyword'],
|
[/\b(_)\b/, 'keyword', '@allowMethod'],
|
||||||
|
|
||||||
// identifiers and keywords
|
// identifiers and keywords
|
||||||
|
[/\bimport\b/, 'keyword', '@import'],
|
||||||
[/\b(case)([ \t]+)(class)\b/, ['tag.id.pug', 'white', 'keyword']],
|
[/\b(case)([ \t]+)(class)\b/, ['tag.id.pug', 'white', 'keyword']],
|
||||||
[/\bcase\b/, 'keyword', '@case'],
|
[/\bcase\b/, 'keyword', '@case'],
|
||||||
[/\bva[lr]\b/, 'keyword', '@vardef'],
|
[/\bva[lr]\b/, 'keyword', '@vardef'],
|
||||||
[/\b(def[ \t]+)(@name)/, ['keyword', 'keyword.flow']],
|
[/\b(def[ \t]+)((?:unary_)?@symbols|@name(?:_=)|@name)/, ['keyword', 'keyword.flow']],
|
||||||
[/@name(?=:(?!:))/, 'variable'],
|
[/@name(?=[ \t]*:(?!:))/, 'variable'],
|
||||||
[/(\.)(@name)(?=[ \t]*[({])/, ['operator', 'keyword.flow']],
|
[/(\.)(@name|@symbols)/, ['operator', {token: 'keyword.flow', next: '@allowMethod'}]],
|
||||||
[/@name(?=[ \t]*[({])/, {cases: {
|
[/([{(])(\s*)(@name(?=\s*=>))/, ['@brackets', 'white', 'variable']],
|
||||||
'@keywords': 'keyword',
|
|
||||||
'@default': 'keyword.flow'
|
|
||||||
}}],
|
|
||||||
[/(\{)(\s*)(@name(?=\s*=>))/, ['@brackets', 'white', 'variable']],
|
|
||||||
[/@name/, {cases: {
|
[/@name/, {cases: {
|
||||||
'@keywords': 'keyword',
|
'@keywords': 'keyword',
|
||||||
|
'@softKeywords': 'keyword',
|
||||||
'@modifiers': 'tag.id.pug',
|
'@modifiers': 'tag.id.pug',
|
||||||
'@constants': 'constant',
|
'@softModifiers': 'tag.id.pug',
|
||||||
'@default': 'identifier'
|
'@constants': {token: 'constant', next: '@allowMethod'},
|
||||||
|
'@default': {token: 'identifier', next: '@allowMethod'}
|
||||||
}}],
|
}}],
|
||||||
[/[A-Z]\w*/, 'type.identifier'],
|
[/@type/, 'type', '@allowMethod'],
|
||||||
|
|
||||||
// whitespace
|
// whitespace
|
||||||
{include: '@whitespace'},
|
{include: '@whitespace'},
|
||||||
|
|
@ -101,33 +155,67 @@ export const conf: IRichLanguageConfiguration = {
|
||||||
[/@[a-zA-Z_$][\w$]*(?:\.[a-zA-Z_$][\w$]*)*/, 'annotation'],
|
[/@[a-zA-Z_$][\w$]*(?:\.[a-zA-Z_$][\w$]*)*/, 'annotation'],
|
||||||
|
|
||||||
// delimiters and operators
|
// delimiters and operators
|
||||||
[/[{}()]/, '@brackets'],
|
[/[{(]/, '@brackets'],
|
||||||
[/[\[\]]/, 'operator.scss'],
|
[/[})]/, '@brackets', '@allowMethod'],
|
||||||
[/[=-]>|<-|>:|<:|<%/, 'keyword'],
|
[/\[/, 'operator.scss'],
|
||||||
|
[/](?!\s*(?:va[rl]|def|type)\b)/, 'operator.scss', '@allowMethod'],
|
||||||
|
[/]/, 'operator.scss'],
|
||||||
|
[/([=-]>|<-|>:|<:|:>|<%)(?=[\s\w()[\]{},\."'`])/, 'keyword'],
|
||||||
[/@symbols/, 'operator'],
|
[/@symbols/, 'operator'],
|
||||||
|
|
||||||
// delimiter: after number because of .\d floats
|
// delimiter: after number because of .\d floats
|
||||||
[/[;,.]/, 'delimiter'],
|
[/[;,\.]/, 'delimiter'],
|
||||||
|
|
||||||
|
// symbols
|
||||||
|
[/'[a-zA-Z$][\w$]*(?!')/, 'attribute.name'],
|
||||||
|
|
||||||
// characters
|
// characters
|
||||||
[/'[^\\']'/, 'string'],
|
[/'[^\\']'/, 'string', '@allowMethod'],
|
||||||
[/(')(@escapes)(')/, ['string', 'string.escape', 'string']],
|
[/(')(@escapes)(')/, ['string', 'string.escape', {token: 'string', next: '@allowMethod'}]],
|
||||||
[/'/, 'string.invalid']
|
[/'/, 'string.invalid']
|
||||||
],
|
],
|
||||||
|
|
||||||
|
import: [
|
||||||
|
[/;/, 'delimiter', '@pop'],
|
||||||
|
[/^|$/, '', '@pop'],
|
||||||
|
[/[ \t]+/, 'white'],
|
||||||
|
[/[\n\r]+/, 'white', '@pop'],
|
||||||
|
[/\/\*/, 'comment', '@comment'],
|
||||||
|
[/@name|@type/, 'type'],
|
||||||
|
[/[(){}]/, '@brackets'],
|
||||||
|
[/[[\]]/, 'operator.scss'],
|
||||||
|
[/[\.,]/, 'delimiter'],
|
||||||
|
],
|
||||||
|
|
||||||
|
allowMethod: [
|
||||||
|
[/^|$/, '', '@pop'],
|
||||||
|
[/[ \t]+/, 'white'],
|
||||||
|
[/[\n\r]+/, 'white', '@pop'],
|
||||||
|
[/\/\*/, 'comment', '@comment'],
|
||||||
|
[/(?==>[\s\w([{])/, 'keyword', '@pop'],
|
||||||
|
[/(@name|@symbols)(?=[ \t]*[[({"'`]|[ \t]+(?:[+-]?\.?\d|\w))/, {
|
||||||
|
cases: {
|
||||||
|
'@keywords': {token: 'keyword', next: '@pop'},
|
||||||
|
'->|<-|>:|<:|<%': {token: 'keyword', next: '@pop'},
|
||||||
|
'@default': {token: 'keyword.flow', next: '@pop'}
|
||||||
|
}
|
||||||
|
}],
|
||||||
|
["", "", "@pop"]
|
||||||
|
],
|
||||||
|
|
||||||
comment: [
|
comment: [
|
||||||
[/[^\/*]+/, 'comment'],
|
[/[^\/*]+/, 'comment'],
|
||||||
[/\/\*/, 'comment', '@push'], // nested comment
|
[/\/\*/, 'comment', '@push'], // nested comment
|
||||||
["\\*/", 'comment', '@pop'],
|
[/\*\//, 'comment', '@pop'],
|
||||||
[/[\/*]/, 'comment']
|
[/[\/*]/, 'comment']
|
||||||
],
|
],
|
||||||
|
|
||||||
case: [
|
case: [
|
||||||
[/\b_\*/, 'key'],
|
[/\b_\*/, 'key'],
|
||||||
[/\b(_|true|false|null|this|super)\b/, 'keyword'],
|
[/\b(_|true|false|null|this|super)\b/, 'keyword', '@allowMethod'],
|
||||||
[/\bif\b|=>/, 'keyword', '@pop'],
|
[/\bif\b|=>/, 'keyword', '@pop'],
|
||||||
[/`@name`/, 'identifier'],
|
[/`[^`]+`/, 'identifier', '@allowMethod'],
|
||||||
[/@name/, 'variable'],
|
[/@name/, 'variable', '@allowMethod'],
|
||||||
[/:::?|\||@(?![a-z_$])/, 'keyword'],
|
[/:::?|\||@(?![a-z_$])/, 'keyword'],
|
||||||
{include: '@root'}
|
{include: '@root'}
|
||||||
],
|
],
|
||||||
|
|
@ -137,7 +225,7 @@ export const conf: IRichLanguageConfiguration = {
|
||||||
[/\b(_|true|false|null|this|super)\b/, 'keyword'],
|
[/\b(_|true|false|null|this|super)\b/, 'keyword'],
|
||||||
[/@name/, 'variable'],
|
[/@name/, 'variable'],
|
||||||
[/:::?|\||@(?![a-z_$])/, 'keyword'],
|
[/:::?|\||@(?![a-z_$])/, 'keyword'],
|
||||||
[/[=:]/, 'operator', '@pop'],
|
[/=|:(?!:)/, 'operator', '@pop'],
|
||||||
[/$/, 'white', '@pop'],
|
[/$/, 'white', '@pop'],
|
||||||
{include: '@root'}
|
{include: '@root'}
|
||||||
],
|
],
|
||||||
|
|
@ -146,20 +234,21 @@ export const conf: IRichLanguageConfiguration = {
|
||||||
[/[^\\"\n\r]+/, 'string'],
|
[/[^\\"\n\r]+/, 'string'],
|
||||||
[/@escapes/, 'string.escape'],
|
[/@escapes/, 'string.escape'],
|
||||||
[/\\./, 'string.escape.invalid'],
|
[/\\./, 'string.escape.invalid'],
|
||||||
[/"/, {token: 'string.quote', bracket: '@close', next: '@pop'}],
|
[/"/, {token: 'string.quote', bracket: '@close', switchTo: '@allowMethod'}],
|
||||||
],
|
],
|
||||||
|
|
||||||
stringt: [
|
stringt: [
|
||||||
[/[^\\"\n\r]+/, 'string'],
|
[/[^\\"\n\r]+/, 'string'],
|
||||||
[/@escapes/, 'string.escape'],
|
[/@escapes/, 'string.escape'],
|
||||||
[/\\./, 'string.escape.invalid'],
|
[/\\./, 'string.escape.invalid'],
|
||||||
[/"""/, {token: 'string.quote', bracket: '@close', next: '@pop'}],
|
[/"(?=""")/, 'string'],
|
||||||
|
[/"""/, {token: 'string.quote', bracket: '@close', switchTo: '@allowMethod'}],
|
||||||
[/"/, 'string']
|
[/"/, 'string']
|
||||||
],
|
],
|
||||||
|
|
||||||
fstring: [
|
fstring: [
|
||||||
[/@escapes/, 'string.escape'],
|
[/@escapes/, 'string.escape'],
|
||||||
[/"/, {token: 'string.quote', bracket: '@close', next: '@pop'}],
|
[/"/, {token: 'string.quote', bracket: '@close', switchTo: '@allowMethod'}],
|
||||||
[/\$\$/, 'string'],
|
[/\$\$/, 'string'],
|
||||||
[/(\$)([a-z_]\w*)/, ['operator', 'identifier']],
|
[/(\$)([a-z_]\w*)/, ['operator', 'identifier']],
|
||||||
[/\$\{/, 'operator', '@interp'],
|
[/\$\{/, 'operator', '@interp'],
|
||||||
|
|
@ -173,7 +262,8 @@ export const conf: IRichLanguageConfiguration = {
|
||||||
|
|
||||||
fstringt: [
|
fstringt: [
|
||||||
[/@escapes/, 'string.escape'],
|
[/@escapes/, 'string.escape'],
|
||||||
[/"""/, {token: 'string.quote', bracket: '@close', next: '@pop'}],
|
[/"(?=""")/, 'string'],
|
||||||
|
[/"""/, {token: 'string.quote', bracket: '@close', switchTo: '@allowMethod'}],
|
||||||
[/\$\$/, 'string'],
|
[/\$\$/, 'string'],
|
||||||
[/(\$)([a-z_]\w*)/, ['operator', 'identifier']],
|
[/(\$)([a-z_]\w*)/, ['operator', 'identifier']],
|
||||||
[/\$\{/, 'operator', '@interp'],
|
[/\$\{/, 'operator', '@interp'],
|
||||||
|
|
@ -187,7 +277,7 @@ export const conf: IRichLanguageConfiguration = {
|
||||||
|
|
||||||
sstring: [
|
sstring: [
|
||||||
[/@escapes/, 'string.escape'],
|
[/@escapes/, 'string.escape'],
|
||||||
[/"/, {token: 'string.quote', bracket: '@close', next: '@pop'}],
|
[/"/, {token: 'string.quote', bracket: '@close', switchTo: '@allowMethod'}],
|
||||||
[/\$\$/, 'string'],
|
[/\$\$/, 'string'],
|
||||||
[/(\$)([a-z_]\w*)/, ['operator', 'identifier']],
|
[/(\$)([a-z_]\w*)/, ['operator', 'identifier']],
|
||||||
[/\$\{/, 'operator', '@interp'],
|
[/\$\{/, 'operator', '@interp'],
|
||||||
|
|
@ -196,7 +286,8 @@ export const conf: IRichLanguageConfiguration = {
|
||||||
|
|
||||||
sstringt: [
|
sstringt: [
|
||||||
[/@escapes/, 'string.escape'],
|
[/@escapes/, 'string.escape'],
|
||||||
[/"""/, {token: 'string.quote', bracket: '@close', next: '@pop'}],
|
[/"(?=""")/, 'string'],
|
||||||
|
[/"""/, {token: 'string.quote', bracket: '@close', switchTo: '@allowMethod'}],
|
||||||
[/\$\$/, 'string'],
|
[/\$\$/, 'string'],
|
||||||
[/(\$)([a-z_]\w*)/, ['operator', 'identifier']],
|
[/(\$)([a-z_]\w*)/, ['operator', 'identifier']],
|
||||||
[/\$\{/, 'operator', '@interp'],
|
[/\$\{/, 'operator', '@interp'],
|
||||||
|
|
@ -204,19 +295,20 @@ export const conf: IRichLanguageConfiguration = {
|
||||||
],
|
],
|
||||||
|
|
||||||
interp: [
|
interp: [
|
||||||
[/\{/, 'operator', '@push'],
|
[/{/, 'operator', '@push'],
|
||||||
[/\}/, 'operator', '@pop'],
|
[/}/, 'operator', '@pop'],
|
||||||
{include: '@root'}
|
{include: '@root'}
|
||||||
],
|
],
|
||||||
|
|
||||||
rawstring: [
|
rawstring: [
|
||||||
[/[^"]/, 'string'],
|
[/[^"]/, 'string'],
|
||||||
[/"/, {token: 'string.quote', bracket: '@close', next: '@pop'}]
|
[/"/, {token: 'string.quote', bracket: '@close', switchTo: '@allowMethod'}]
|
||||||
],
|
],
|
||||||
|
|
||||||
rawstringt: [
|
rawstringt: [
|
||||||
[/[^"]/, 'string'],
|
[/[^"]/, 'string'],
|
||||||
[/"""/, {token: 'string.quote', bracket: '@close', next: '@pop'}],
|
[/"(?=""")/, 'string'],
|
||||||
|
[/"""/, {token: 'string.quote', bracket: '@close', switchTo: '@allowMethod'}],
|
||||||
[/"/, 'string']
|
[/"/, 'string']
|
||||||
],
|
],
|
||||||
|
|
||||||
|
|
|
||||||
Loading…
Add table
Add a link
Reference in a new issue