annotate .cms/lib/codemirror/addon/mode/simple.js @ 1:1d486627aa1e draft default tip

24.10
author Coffee CMS <info@coffee-cms.ru>
date Sat, 12 Oct 2024 02:51:39 +0000
parents 78edf6b517a0
children
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
rev   line source
0
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
1 // CodeMirror, copyright (c) by Marijn Haverbeke and others
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
2 // Distributed under an MIT license: https://codemirror.net/5/LICENSE
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
3
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
4 (function(mod) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
5 if (typeof exports == "object" && typeof module == "object") // CommonJS
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
6 mod(require("../../lib/codemirror"));
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
7 else if (typeof define == "function" && define.amd) // AMD
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
8 define(["../../lib/codemirror"], mod);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
9 else // Plain browser env
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
10 mod(CodeMirror);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
11 })(function(CodeMirror) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
12 "use strict";
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
13
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
14 CodeMirror.defineSimpleMode = function(name, states) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
15 CodeMirror.defineMode(name, function(config) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
16 return CodeMirror.simpleMode(config, states);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
17 });
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
18 };
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
19
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
20 CodeMirror.simpleMode = function(config, states) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
21 ensureState(states, "start");
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
22 var states_ = {}, meta = states.meta || {}, hasIndentation = false;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
23 for (var state in states) if (state != meta && states.hasOwnProperty(state)) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
24 var list = states_[state] = [], orig = states[state];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
25 for (var i = 0; i < orig.length; i++) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
26 var data = orig[i];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
27 list.push(new Rule(data, states));
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
28 if (data.indent || data.dedent) hasIndentation = true;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
29 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
30 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
31 var mode = {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
32 startState: function() {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
33 return {state: "start", pending: null,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
34 local: null, localState: null,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
35 indent: hasIndentation ? [] : null};
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
36 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
37 copyState: function(state) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
38 var s = {state: state.state, pending: state.pending,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
39 local: state.local, localState: null,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
40 indent: state.indent && state.indent.slice(0)};
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
41 if (state.localState)
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
42 s.localState = CodeMirror.copyState(state.local.mode, state.localState);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
43 if (state.stack)
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
44 s.stack = state.stack.slice(0);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
45 for (var pers = state.persistentStates; pers; pers = pers.next)
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
46 s.persistentStates = {mode: pers.mode,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
47 spec: pers.spec,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
48 state: pers.state == state.localState ? s.localState : CodeMirror.copyState(pers.mode, pers.state),
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
49 next: s.persistentStates};
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
50 return s;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
51 },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
52 token: tokenFunction(states_, config),
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
53 innerMode: function(state) { return state.local && {mode: state.local.mode, state: state.localState}; },
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
54 indent: indentFunction(states_, meta)
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
55 };
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
56 if (meta) for (var prop in meta) if (meta.hasOwnProperty(prop))
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
57 mode[prop] = meta[prop];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
58 return mode;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
59 };
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
60
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
61 function ensureState(states, name) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
62 if (!states.hasOwnProperty(name))
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
63 throw new Error("Undefined state " + name + " in simple mode");
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
64 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
65
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
66 function toRegex(val, caret) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
67 if (!val) return /(?:)/;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
68 var flags = "";
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
69 if (val instanceof RegExp) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
70 if (val.ignoreCase) flags = "i";
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
71 if (val.unicode) flags += "u"
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
72 val = val.source;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
73 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
74 val = String(val);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
75 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
76 return new RegExp((caret === false ? "" : "^") + "(?:" + val + ")", flags);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
77 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
78
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
79 function asToken(val) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
80 if (!val) return null;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
81 if (val.apply) return val
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
82 if (typeof val == "string") return val.replace(/\./g, " ");
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
83 var result = [];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
84 for (var i = 0; i < val.length; i++)
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
85 result.push(val[i] && val[i].replace(/\./g, " "));
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
86 return result;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
87 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
88
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
89 function Rule(data, states) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
90 if (data.next || data.push) ensureState(states, data.next || data.push);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
91 this.regex = toRegex(data.regex);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
92 this.token = asToken(data.token);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
93 this.data = data;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
94 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
95
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
96 function tokenFunction(states, config) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
97 return function(stream, state) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
98 if (state.pending) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
99 var pend = state.pending.shift();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
100 if (state.pending.length == 0) state.pending = null;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
101 stream.pos += pend.text.length;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
102 return pend.token;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
103 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
104
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
105 if (state.local) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
106 if (state.local.end && stream.match(state.local.end)) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
107 var tok = state.local.endToken || null;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
108 state.local = state.localState = null;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
109 return tok;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
110 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
111 var tok = state.local.mode.token(stream, state.localState), m;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
112 if (state.local.endScan && (m = state.local.endScan.exec(stream.current())))
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
113 stream.pos = stream.start + m.index;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
114 return tok;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
115 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
116 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
117
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
118 var curState = states[state.state];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
119 for (var i = 0; i < curState.length; i++) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
120 var rule = curState[i];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
121 var matches = (!rule.data.sol || stream.sol()) && stream.match(rule.regex);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
122 if (matches) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
123 if (rule.data.next) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
124 state.state = rule.data.next;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
125 } else if (rule.data.push) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
126 (state.stack || (state.stack = [])).push(state.state);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
127 state.state = rule.data.push;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
128 } else if (rule.data.pop && state.stack && state.stack.length) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
129 state.state = state.stack.pop();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
130 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
131
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
132 if (rule.data.mode)
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
133 enterLocalMode(config, state, rule.data.mode, rule.token);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
134 if (rule.data.indent)
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
135 state.indent.push(stream.indentation() + config.indentUnit);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
136 if (rule.data.dedent)
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
137 state.indent.pop();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
138 var token = rule.token
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
139 if (token && token.apply) token = token(matches)
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
140 if (matches.length > 2 && rule.token && typeof rule.token != "string") {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
141 for (var j = 2; j < matches.length; j++)
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
142 if (matches[j])
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
143 (state.pending || (state.pending = [])).push({text: matches[j], token: rule.token[j - 1]});
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
144 stream.backUp(matches[0].length - (matches[1] ? matches[1].length : 0));
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
145 return token[0];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
146 } else if (token && token.join) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
147 return token[0];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
148 } else {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
149 return token;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
150 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
151 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
152 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
153 stream.next();
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
154 return null;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
155 };
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
156 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
157
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
158 function cmp(a, b) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
159 if (a === b) return true;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
160 if (!a || typeof a != "object" || !b || typeof b != "object") return false;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
161 var props = 0;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
162 for (var prop in a) if (a.hasOwnProperty(prop)) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
163 if (!b.hasOwnProperty(prop) || !cmp(a[prop], b[prop])) return false;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
164 props++;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
165 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
166 for (var prop in b) if (b.hasOwnProperty(prop)) props--;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
167 return props == 0;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
168 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
169
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
170 function enterLocalMode(config, state, spec, token) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
171 var pers;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
172 if (spec.persistent) for (var p = state.persistentStates; p && !pers; p = p.next)
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
173 if (spec.spec ? cmp(spec.spec, p.spec) : spec.mode == p.mode) pers = p;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
174 var mode = pers ? pers.mode : spec.mode || CodeMirror.getMode(config, spec.spec);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
175 var lState = pers ? pers.state : CodeMirror.startState(mode);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
176 if (spec.persistent && !pers)
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
177 state.persistentStates = {mode: mode, spec: spec.spec, state: lState, next: state.persistentStates};
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
178
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
179 state.localState = lState;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
180 state.local = {mode: mode,
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
181 end: spec.end && toRegex(spec.end),
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
182 endScan: spec.end && spec.forceEnd !== false && toRegex(spec.end, false),
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
183 endToken: token && token.join ? token[token.length - 1] : token};
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
184 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
185
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
186 function indexOf(val, arr) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
187 for (var i = 0; i < arr.length; i++) if (arr[i] === val) return true;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
188 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
189
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
190 function indentFunction(states, meta) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
191 return function(state, textAfter, line) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
192 if (state.local && state.local.mode.indent)
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
193 return state.local.mode.indent(state.localState, textAfter, line);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
194 if (state.indent == null || state.local || meta.dontIndentStates && indexOf(state.state, meta.dontIndentStates) > -1)
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
195 return CodeMirror.Pass;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
196
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
197 var pos = state.indent.length - 1, rules = states[state.state];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
198 scan: for (;;) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
199 for (var i = 0; i < rules.length; i++) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
200 var rule = rules[i];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
201 if (rule.data.dedent && rule.data.dedentIfLineStart !== false) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
202 var m = rule.regex.exec(textAfter);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
203 if (m && m[0]) {
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
204 pos--;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
205 if (rule.next || rule.push) rules = states[rule.next || rule.push];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
206 textAfter = textAfter.slice(m[0].length);
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
207 continue scan;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
208 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
209 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
210 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
211 break;
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
212 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
213 return pos < 0 ? 0 : state.indent[pos];
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
214 };
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
215 }
Coffee CMS <info@coffee-cms.ru>
parents:
diff changeset
216 });