04295d4e4ab66991946700a8b6739bb2.js 108 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658165916601661166216631664166516661667166816691670167116721673167416751676167716781679168016811682168316841685168616871688168916901691169216931694169516961697169816991700170117021703170417051706170717081709171017111712171317141715171617171718171917201721172217231724172517261727172817291730173117321733173417351736173717381739174017411742174317441745174617471748174917501751175217531754175517561757175817591760176117621763176417651766176717681769177017711772177317741775177617771778177917801781178217831784178517861787178817891790179117921793179417951796179717981799180018011802180318041805180618071808180918101811181218131814181518161817181818191820182118221823182418251826182718281829183018311832183318341835183618371838183918401841184218431844184518461847184818491850185118521853185418551856185718581859186018611862186318641865186618671868186918701871187218731874187518761877187818791880188118821883188418851886188718881889189018911892189318941895189618971898189919001901190219031904190519061907190819091910191119121913191419151916191719181919192019211922192319241925192619271928192919301931193219331934193519361937193819391940194119421943194419451946194719481949195019511952195319541955195619571958195919601961196219631964196519661967196819691970197119721973197419751976197719781979198019811982198319841985198619871988198919901991199219931994199519961997199819992000200120022003200420052006200720082009201020112012201320142015201620172018201920202021202220232024202520262027202820292030203120322033203420352036203720382039204020412042204320442045204620472048204920502051205220532054205520562057205820592060206120622063206420652066206720682069207020712072207320742075207620772078207920802081208220832084208520862087208820892090209120922093209420952096209720982099210021012102210321042105210621072108210921102111211221132114211521162117211821192120212121222123212421252126212721282129213021312132213321342135213621372138213921402141214221432144214521462147214821492150215121522153215421552156215721582159216021612162216321642165216621672168216921702171217221732174217521762177217821792180218121822183218421852186218721882189219021912192219321942195219621972198219922002201220222032204220522062207220822092210221122122213221422152216221722182219222022212222222322242225222622272228222922302231223222332234223522362237223822392240224122422243224422452246224722482249225022512252225322542255225622572258225922602261226222632264226522662267226822692270227122722273227422752276227722782279228022812282228322842285228622872288228922902291229222932294229522962297229822992300230123022303230423052306230723082309231023112312231323142315231623172318231923202321232223232324232523262327232823292330233123322333233423352336233723382339234023412342234323442345234623472348234923502351235223532354235523562357235823592360236123622363236423652366236723682369237023712372237323742375237623772378237923802381238223832384238523862387238823892390239123922393239423952396239723982399240024012402240324042405240624072408240924102411241224132414241524162417241824192420242124222423242424252426242724282429243024312432243324342435243624372438243924402441244224432444244524462447244824492450245124522453245424552456245724582459246024612462246324642465246624672468246924702471247224732474247524762477247824792480248124822483248424852486248724882489249024912492249324942495249624972498249925002501250225032504250525062507250825092510251125122513251425152516251725182519252025212522252325242525252625272528252925302531253225332534253525362537253825392540254125422543254425452546254725482549255025512552255325542555255625572558255925602561256225632564256525662567256825692570257125722573257425752576257725782579258025812582258325842585258625872588258925902591259225932594259525962597259825992600260126022603260426052606260726082609261026112612261326142615261626172618
  1. ace.define("ace/snippets",["require","exports","module","ace/lib/dom","ace/lib/oop","ace/lib/event_emitter","ace/lib/lang","ace/range","ace/range_list","ace/keyboard/hash_handler","ace/tokenizer","ace/clipboard","ace/editor"], function(require, exports, module){"use strict";
  2. var dom = require("./lib/dom");
  3. var oop = require("./lib/oop");
  4. var EventEmitter = require("./lib/event_emitter").EventEmitter;
  5. var lang = require("./lib/lang");
  6. var Range = require("./range").Range;
  7. var RangeList = require("./range_list").RangeList;
  8. var HashHandler = require("./keyboard/hash_handler").HashHandler;
  9. var Tokenizer = require("./tokenizer").Tokenizer;
  10. var clipboard = require("./clipboard");
  11. var VARIABLES = {
  12. CURRENT_WORD: function (editor) {
  13. return editor.session.getTextRange(editor.session.getWordRange());
  14. },
  15. SELECTION: function (editor, name, indentation) {
  16. var text = editor.session.getTextRange();
  17. if (indentation)
  18. return text.replace(/\n\r?([ \t]*\S)/g, "\n" + indentation + "$1");
  19. return text;
  20. },
  21. CURRENT_LINE: function (editor) {
  22. return editor.session.getLine(editor.getCursorPosition().row);
  23. },
  24. PREV_LINE: function (editor) {
  25. return editor.session.getLine(editor.getCursorPosition().row - 1);
  26. },
  27. LINE_INDEX: function (editor) {
  28. return editor.getCursorPosition().row;
  29. },
  30. LINE_NUMBER: function (editor) {
  31. return editor.getCursorPosition().row + 1;
  32. },
  33. SOFT_TABS: function (editor) {
  34. return editor.session.getUseSoftTabs() ? "YES" : "NO";
  35. },
  36. TAB_SIZE: function (editor) {
  37. return editor.session.getTabSize();
  38. },
  39. CLIPBOARD: function (editor) {
  40. return clipboard.getText && clipboard.getText();
  41. },
  42. FILENAME: function (editor) {
  43. return /[^/\\]*$/.exec(this.FILEPATH(editor))[0];
  44. },
  45. FILENAME_BASE: function (editor) {
  46. return /[^/\\]*$/.exec(this.FILEPATH(editor))[0].replace(/\.[^.]*$/, "");
  47. },
  48. DIRECTORY: function (editor) {
  49. return this.FILEPATH(editor).replace(/[^/\\]*$/, "");
  50. },
  51. FILEPATH: function (editor) { return "/not implemented.txt"; },
  52. WORKSPACE_NAME: function () { return "Unknown"; },
  53. FULLNAME: function () { return "Unknown"; },
  54. BLOCK_COMMENT_START: function (editor) {
  55. var mode = editor.session.$mode || {};
  56. return mode.blockComment && mode.blockComment.start || "";
  57. },
  58. BLOCK_COMMENT_END: function (editor) {
  59. var mode = editor.session.$mode || {};
  60. return mode.blockComment && mode.blockComment.end || "";
  61. },
  62. LINE_COMMENT: function (editor) {
  63. var mode = editor.session.$mode || {};
  64. return mode.lineCommentStart || "";
  65. },
  66. CURRENT_YEAR: date.bind(null, { year: "numeric" }),
  67. CURRENT_YEAR_SHORT: date.bind(null, { year: "2-digit" }),
  68. CURRENT_MONTH: date.bind(null, { month: "numeric" }),
  69. CURRENT_MONTH_NAME: date.bind(null, { month: "long" }),
  70. CURRENT_MONTH_NAME_SHORT: date.bind(null, { month: "short" }),
  71. CURRENT_DATE: date.bind(null, { day: "2-digit" }),
  72. CURRENT_DAY_NAME: date.bind(null, { weekday: "long" }),
  73. CURRENT_DAY_NAME_SHORT: date.bind(null, { weekday: "short" }),
  74. CURRENT_HOUR: date.bind(null, { hour: "2-digit", hour12: false }),
  75. CURRENT_MINUTE: date.bind(null, { minute: "2-digit" }),
  76. CURRENT_SECOND: date.bind(null, { second: "2-digit" })
  77. };
  78. VARIABLES.SELECTED_TEXT = VARIABLES.SELECTION;
  79. function date(dateFormat) {
  80. var str = new Date().toLocaleString("en-us", dateFormat);
  81. return str.length == 1 ? "0" + str : str;
  82. }
  83. var SnippetManager = /** @class */ (function () {
  84. function SnippetManager() {
  85. this.snippetMap = {};
  86. this.snippetNameMap = {};
  87. this.variables = VARIABLES;
  88. }
  89. SnippetManager.prototype.getTokenizer = function () {
  90. return SnippetManager["$tokenizer"] || this.createTokenizer();
  91. };
  92. SnippetManager.prototype.createTokenizer = function () {
  93. function TabstopToken(str) {
  94. str = str.substr(1);
  95. if (/^\d+$/.test(str))
  96. return [{ tabstopId: parseInt(str, 10) }];
  97. return [{ text: str }];
  98. }
  99. function escape(ch) {
  100. return "(?:[^\\\\" + ch + "]|\\\\.)";
  101. }
  102. var formatMatcher = {
  103. regex: "/(" + escape("/") + "+)/",
  104. onMatch: function (val, state, stack) {
  105. var ts = stack[0];
  106. ts.fmtString = true;
  107. ts.guard = val.slice(1, -1);
  108. ts.flag = "";
  109. return "";
  110. },
  111. next: "formatString"
  112. };
  113. SnippetManager["$tokenizer"] = new Tokenizer({
  114. start: [
  115. { regex: /\\./, onMatch: function (val, state, stack) {
  116. var ch = val[1];
  117. if (ch == "}" && stack.length) {
  118. val = ch;
  119. }
  120. else if ("`$\\".indexOf(ch) != -1) {
  121. val = ch;
  122. }
  123. return [val];
  124. } },
  125. { regex: /}/, onMatch: function (val, state, stack) {
  126. return [stack.length ? stack.shift() : val];
  127. } },
  128. { regex: /\$(?:\d+|\w+)/, onMatch: TabstopToken },
  129. { regex: /\$\{[\dA-Z_a-z]+/, onMatch: function (str, state, stack) {
  130. var t = TabstopToken(str.substr(1));
  131. stack.unshift(t[0]);
  132. return t;
  133. }, next: "snippetVar" },
  134. { regex: /\n/, token: "newline", merge: false }
  135. ],
  136. snippetVar: [
  137. { regex: "\\|" + escape("\\|") + "*\\|", onMatch: function (val, state, stack) {
  138. var choices = val.slice(1, -1).replace(/\\[,|\\]|,/g, function (operator) {
  139. return operator.length == 2 ? operator[1] : "\x00";
  140. }).split("\x00").map(function (value) {
  141. return { value: value };
  142. });
  143. stack[0].choices = choices;
  144. return [choices[0]];
  145. }, next: "start" },
  146. formatMatcher,
  147. { regex: "([^:}\\\\]|\\\\.)*:?", token: "", next: "start" }
  148. ],
  149. formatString: [
  150. { regex: /:/, onMatch: function (val, state, stack) {
  151. if (stack.length && stack[0].expectElse) {
  152. stack[0].expectElse = false;
  153. stack[0].ifEnd = { elseEnd: stack[0] };
  154. return [stack[0].ifEnd];
  155. }
  156. return ":";
  157. } },
  158. { regex: /\\./, onMatch: function (val, state, stack) {
  159. var ch = val[1];
  160. if (ch == "}" && stack.length)
  161. val = ch;
  162. else if ("`$\\".indexOf(ch) != -1)
  163. val = ch;
  164. else if (ch == "n")
  165. val = "\n";
  166. else if (ch == "t")
  167. val = "\t";
  168. else if ("ulULE".indexOf(ch) != -1)
  169. val = { changeCase: ch, local: ch > "a" };
  170. return [val];
  171. } },
  172. { regex: "/\\w*}", onMatch: function (val, state, stack) {
  173. var next = stack.shift();
  174. if (next)
  175. next.flag = val.slice(1, -1);
  176. this.next = next && next.tabstopId ? "start" : "";
  177. return [next || val];
  178. }, next: "start" },
  179. { regex: /\$(?:\d+|\w+)/, onMatch: function (val, state, stack) {
  180. return [{ text: val.slice(1) }];
  181. } },
  182. { regex: /\${\w+/, onMatch: function (val, state, stack) {
  183. var token = { text: val.slice(2) };
  184. stack.unshift(token);
  185. return [token];
  186. }, next: "formatStringVar" },
  187. { regex: /\n/, token: "newline", merge: false },
  188. { regex: /}/, onMatch: function (val, state, stack) {
  189. var next = stack.shift();
  190. this.next = next && next.tabstopId ? "start" : "";
  191. return [next || val];
  192. }, next: "start" }
  193. ],
  194. formatStringVar: [
  195. { regex: /:\/\w+}/, onMatch: function (val, state, stack) {
  196. var ts = stack[0];
  197. ts.formatFunction = val.slice(2, -1);
  198. return [stack.shift()];
  199. }, next: "formatString" },
  200. formatMatcher,
  201. { regex: /:[\?\-+]?/, onMatch: function (val, state, stack) {
  202. if (val[1] == "+")
  203. stack[0].ifEnd = stack[0];
  204. if (val[1] == "?")
  205. stack[0].expectElse = true;
  206. }, next: "formatString" },
  207. { regex: "([^:}\\\\]|\\\\.)*:?", token: "", next: "formatString" }
  208. ]
  209. });
  210. return SnippetManager["$tokenizer"];
  211. };
  212. SnippetManager.prototype.tokenizeTmSnippet = function (str, startState) {
  213. return this.getTokenizer().getLineTokens(str, startState).tokens.map(function (x) {
  214. return x.value || x;
  215. });
  216. };
  217. SnippetManager.prototype.getVariableValue = function (editor, name, indentation) {
  218. if (/^\d+$/.test(name))
  219. return (this.variables.__ || {})[name] || "";
  220. if (/^[A-Z]\d+$/.test(name))
  221. return (this.variables[name[0] + "__"] || {})[name.substr(1)] || "";
  222. name = name.replace(/^TM_/, "");
  223. if (!this.variables.hasOwnProperty(name))
  224. return "";
  225. var value = this.variables[name];
  226. if (typeof value == "function")
  227. value = this.variables[name](editor, name, indentation);
  228. return value == null ? "" : value;
  229. };
  230. SnippetManager.prototype.tmStrFormat = function (str, ch, editor) {
  231. if (!ch.fmt)
  232. return str;
  233. var flag = ch.flag || "";
  234. var re = ch.guard;
  235. re = new RegExp(re, flag.replace(/[^gim]/g, ""));
  236. var fmtTokens = typeof ch.fmt == "string" ? this.tokenizeTmSnippet(ch.fmt, "formatString") : ch.fmt;
  237. var _self = this;
  238. var formatted = str.replace(re, function () {
  239. var oldArgs = _self.variables.__;
  240. _self.variables.__ = [].slice.call(arguments);
  241. var fmtParts = _self.resolveVariables(fmtTokens, editor);
  242. var gChangeCase = "E";
  243. for (var i = 0; i < fmtParts.length; i++) {
  244. var ch = fmtParts[i];
  245. if (typeof ch == "object") {
  246. fmtParts[i] = "";
  247. if (ch.changeCase && ch.local) {
  248. var next = fmtParts[i + 1];
  249. if (next && typeof next == "string") {
  250. if (ch.changeCase == "u")
  251. fmtParts[i] = next[0].toUpperCase();
  252. else
  253. fmtParts[i] = next[0].toLowerCase();
  254. fmtParts[i + 1] = next.substr(1);
  255. }
  256. }
  257. else if (ch.changeCase) {
  258. gChangeCase = ch.changeCase;
  259. }
  260. }
  261. else if (gChangeCase == "U") {
  262. fmtParts[i] = ch.toUpperCase();
  263. }
  264. else if (gChangeCase == "L") {
  265. fmtParts[i] = ch.toLowerCase();
  266. }
  267. }
  268. _self.variables.__ = oldArgs;
  269. return fmtParts.join("");
  270. });
  271. return formatted;
  272. };
  273. SnippetManager.prototype.tmFormatFunction = function (str, ch, editor) {
  274. if (ch.formatFunction == "upcase")
  275. return str.toUpperCase();
  276. if (ch.formatFunction == "downcase")
  277. return str.toLowerCase();
  278. return str;
  279. };
  280. SnippetManager.prototype.resolveVariables = function (snippet, editor) {
  281. var result = [];
  282. var indentation = "";
  283. var afterNewLine = true;
  284. for (var i = 0; i < snippet.length; i++) {
  285. var ch = snippet[i];
  286. if (typeof ch == "string") {
  287. result.push(ch);
  288. if (ch == "\n") {
  289. afterNewLine = true;
  290. indentation = "";
  291. }
  292. else if (afterNewLine) {
  293. indentation = /^\t*/.exec(ch)[0];
  294. afterNewLine = /\S/.test(ch);
  295. }
  296. continue;
  297. }
  298. if (!ch)
  299. continue;
  300. afterNewLine = false;
  301. if (ch.fmtString) {
  302. var j = snippet.indexOf(ch, i + 1);
  303. if (j == -1)
  304. j = snippet.length;
  305. ch.fmt = snippet.slice(i + 1, j);
  306. i = j;
  307. }
  308. if (ch.text) {
  309. var value = this.getVariableValue(editor, ch.text, indentation) + "";
  310. if (ch.fmtString)
  311. value = this.tmStrFormat(value, ch, editor);
  312. if (ch.formatFunction)
  313. value = this.tmFormatFunction(value, ch, editor);
  314. if (value && !ch.ifEnd) {
  315. result.push(value);
  316. gotoNext(ch);
  317. }
  318. else if (!value && ch.ifEnd) {
  319. gotoNext(ch.ifEnd);
  320. }
  321. }
  322. else if (ch.elseEnd) {
  323. gotoNext(ch.elseEnd);
  324. }
  325. else if (ch.tabstopId != null) {
  326. result.push(ch);
  327. }
  328. else if (ch.changeCase != null) {
  329. result.push(ch);
  330. }
  331. }
  332. function gotoNext(ch) {
  333. var i1 = snippet.indexOf(ch, i + 1);
  334. if (i1 != -1)
  335. i = i1;
  336. }
  337. return result;
  338. };
  339. SnippetManager.prototype.getDisplayTextForSnippet = function (editor, snippetText) {
  340. var processedSnippet = processSnippetText.call(this, editor, snippetText);
  341. return processedSnippet.text;
  342. };
  343. SnippetManager.prototype.insertSnippetForSelection = function (editor, snippetText, options) {
  344. if (options === void 0) { options = {}; }
  345. var processedSnippet = processSnippetText.call(this, editor, snippetText, options);
  346. var range = editor.getSelectionRange();
  347. var end = editor.session.replace(range, processedSnippet.text);
  348. var tabstopManager = new TabstopManager(editor);
  349. var selectionId = editor.inVirtualSelectionMode && editor.selection.index;
  350. tabstopManager.addTabstops(processedSnippet.tabstops, range.start, end, selectionId);
  351. };
  352. SnippetManager.prototype.insertSnippet = function (editor, snippetText, options) {
  353. if (options === void 0) { options = {}; }
  354. var self = this;
  355. if (editor.inVirtualSelectionMode)
  356. return self.insertSnippetForSelection(editor, snippetText, options);
  357. editor.forEachSelection(function () {
  358. self.insertSnippetForSelection(editor, snippetText, options);
  359. }, null, { keepOrder: true });
  360. if (editor.tabstopManager)
  361. editor.tabstopManager.tabNext();
  362. };
  363. SnippetManager.prototype.$getScope = function (editor) {
  364. var scope = editor.session.$mode.$id || "";
  365. scope = scope.split("/").pop();
  366. if (scope === "html" || scope === "php") {
  367. if (scope === "php" && !editor.session.$mode.inlinePhp)
  368. scope = "html";
  369. var c = editor.getCursorPosition();
  370. var state = editor.session.getState(c.row);
  371. if (typeof state === "object") {
  372. state = state[0];
  373. }
  374. if (state.substring) {
  375. if (state.substring(0, 3) == "js-")
  376. scope = "javascript";
  377. else if (state.substring(0, 4) == "css-")
  378. scope = "css";
  379. else if (state.substring(0, 4) == "php-")
  380. scope = "php";
  381. }
  382. }
  383. return scope;
  384. };
  385. SnippetManager.prototype.getActiveScopes = function (editor) {
  386. var scope = this.$getScope(editor);
  387. var scopes = [scope];
  388. var snippetMap = this.snippetMap;
  389. if (snippetMap[scope] && snippetMap[scope].includeScopes) {
  390. scopes.push.apply(scopes, snippetMap[scope].includeScopes);
  391. }
  392. scopes.push("_");
  393. return scopes;
  394. };
  395. SnippetManager.prototype.expandWithTab = function (editor, options) {
  396. var self = this;
  397. var result = editor.forEachSelection(function () {
  398. return self.expandSnippetForSelection(editor, options);
  399. }, null, { keepOrder: true });
  400. if (result && editor.tabstopManager)
  401. editor.tabstopManager.tabNext();
  402. return result;
  403. };
  404. SnippetManager.prototype.expandSnippetForSelection = function (editor, options) {
  405. var cursor = editor.getCursorPosition();
  406. var line = editor.session.getLine(cursor.row);
  407. var before = line.substring(0, cursor.column);
  408. var after = line.substr(cursor.column);
  409. var snippetMap = this.snippetMap;
  410. var snippet;
  411. this.getActiveScopes(editor).some(function (scope) {
  412. var snippets = snippetMap[scope];
  413. if (snippets)
  414. snippet = this.findMatchingSnippet(snippets, before, after);
  415. return !!snippet;
  416. }, this);
  417. if (!snippet)
  418. return false;
  419. if (options && options.dryRun)
  420. return true;
  421. editor.session.doc.removeInLine(cursor.row, cursor.column - snippet.replaceBefore.length, cursor.column + snippet.replaceAfter.length);
  422. this.variables.M__ = snippet.matchBefore;
  423. this.variables.T__ = snippet.matchAfter;
  424. this.insertSnippetForSelection(editor, snippet.content);
  425. this.variables.M__ = this.variables.T__ = null;
  426. return true;
  427. };
  428. SnippetManager.prototype.findMatchingSnippet = function (snippetList, before, after) {
  429. for (var i = snippetList.length; i--;) {
  430. var s = snippetList[i];
  431. if (s.startRe && !s.startRe.test(before))
  432. continue;
  433. if (s.endRe && !s.endRe.test(after))
  434. continue;
  435. if (!s.startRe && !s.endRe)
  436. continue;
  437. s.matchBefore = s.startRe ? s.startRe.exec(before) : [""];
  438. s.matchAfter = s.endRe ? s.endRe.exec(after) : [""];
  439. s.replaceBefore = s.triggerRe ? s.triggerRe.exec(before)[0] : "";
  440. s.replaceAfter = s.endTriggerRe ? s.endTriggerRe.exec(after)[0] : "";
  441. return s;
  442. }
  443. };
  444. SnippetManager.prototype.register = function (snippets, scope) {
  445. var snippetMap = this.snippetMap;
  446. var snippetNameMap = this.snippetNameMap;
  447. var self = this;
  448. if (!snippets)
  449. snippets = [];
  450. function wrapRegexp(src) {
  451. if (src && !/^\^?\(.*\)\$?$|^\\b$/.test(src))
  452. src = "(?:" + src + ")";
  453. return src || "";
  454. }
  455. function guardedRegexp(re, guard, opening) {
  456. re = wrapRegexp(re);
  457. guard = wrapRegexp(guard);
  458. if (opening) {
  459. re = guard + re;
  460. if (re && re[re.length - 1] != "$")
  461. re = re + "$";
  462. }
  463. else {
  464. re = re + guard;
  465. if (re && re[0] != "^")
  466. re = "^" + re;
  467. }
  468. return new RegExp(re);
  469. }
  470. function addSnippet(s) {
  471. if (!s.scope)
  472. s.scope = scope || "_";
  473. scope = s.scope;
  474. if (!snippetMap[scope]) {
  475. snippetMap[scope] = [];
  476. snippetNameMap[scope] = {};
  477. }
  478. var map = snippetNameMap[scope];
  479. if (s.name) {
  480. var old = map[s.name];
  481. if (old)
  482. self.unregister(old);
  483. map[s.name] = s;
  484. }
  485. snippetMap[scope].push(s);
  486. if (s.prefix)
  487. s.tabTrigger = s.prefix;
  488. if (!s.content && s.body)
  489. s.content = Array.isArray(s.body) ? s.body.join("\n") : s.body;
  490. if (s.tabTrigger && !s.trigger) {
  491. if (!s.guard && /^\w/.test(s.tabTrigger))
  492. s.guard = "\\b";
  493. s.trigger = lang.escapeRegExp(s.tabTrigger);
  494. }
  495. if (!s.trigger && !s.guard && !s.endTrigger && !s.endGuard)
  496. return;
  497. s.startRe = guardedRegexp(s.trigger, s.guard, true);
  498. s.triggerRe = new RegExp(s.trigger);
  499. s.endRe = guardedRegexp(s.endTrigger, s.endGuard, true);
  500. s.endTriggerRe = new RegExp(s.endTrigger);
  501. }
  502. if (Array.isArray(snippets)) {
  503. snippets.forEach(addSnippet);
  504. }
  505. else {
  506. Object.keys(snippets).forEach(function (key) {
  507. addSnippet(snippets[key]);
  508. });
  509. }
  510. this._signal("registerSnippets", { scope: scope });
  511. };
  512. SnippetManager.prototype.unregister = function (snippets, scope) {
  513. var snippetMap = this.snippetMap;
  514. var snippetNameMap = this.snippetNameMap;
  515. function removeSnippet(s) {
  516. var nameMap = snippetNameMap[s.scope || scope];
  517. if (nameMap && nameMap[s.name]) {
  518. delete nameMap[s.name];
  519. var map = snippetMap[s.scope || scope];
  520. var i = map && map.indexOf(s);
  521. if (i >= 0)
  522. map.splice(i, 1);
  523. }
  524. }
  525. if (snippets.content)
  526. removeSnippet(snippets);
  527. else if (Array.isArray(snippets))
  528. snippets.forEach(removeSnippet);
  529. };
  530. SnippetManager.prototype.parseSnippetFile = function (str) {
  531. str = str.replace(/\r/g, "");
  532. var list = [], /**@type{Snippet}*/ snippet = {};
  533. var re = /^#.*|^({[\s\S]*})\s*$|^(\S+) (.*)$|^((?:\n*\t.*)+)/gm;
  534. var m;
  535. while (m = re.exec(str)) {
  536. if (m[1]) {
  537. try {
  538. snippet = JSON.parse(m[1]);
  539. list.push(snippet);
  540. }
  541. catch (e) { }
  542. }
  543. if (m[4]) {
  544. snippet.content = m[4].replace(/^\t/gm, "");
  545. list.push(snippet);
  546. snippet = {};
  547. }
  548. else {
  549. var key = m[2], val = m[3];
  550. if (key == "regex") {
  551. var guardRe = /\/((?:[^\/\\]|\\.)*)|$/g;
  552. snippet.guard = guardRe.exec(val)[1];
  553. snippet.trigger = guardRe.exec(val)[1];
  554. snippet.endTrigger = guardRe.exec(val)[1];
  555. snippet.endGuard = guardRe.exec(val)[1];
  556. }
  557. else if (key == "snippet") {
  558. snippet.tabTrigger = val.match(/^\S*/)[0];
  559. if (!snippet.name)
  560. snippet.name = val;
  561. }
  562. else if (key) {
  563. snippet[key] = val;
  564. }
  565. }
  566. }
  567. return list;
  568. };
  569. SnippetManager.prototype.getSnippetByName = function (name, editor) {
  570. var snippetMap = this.snippetNameMap;
  571. var snippet;
  572. this.getActiveScopes(editor).some(function (scope) {
  573. var snippets = snippetMap[scope];
  574. if (snippets)
  575. snippet = snippets[name];
  576. return !!snippet;
  577. }, this);
  578. return snippet;
  579. };
  580. return SnippetManager;
  581. }());
  582. oop.implement(SnippetManager.prototype, EventEmitter);
  583. var processSnippetText = function (editor, snippetText, options) {
  584. if (options === void 0) { options = {}; }
  585. var cursor = editor.getCursorPosition();
  586. var line = editor.session.getLine(cursor.row);
  587. var tabString = editor.session.getTabString();
  588. var indentString = line.match(/^\s*/)[0];
  589. if (cursor.column < indentString.length)
  590. indentString = indentString.slice(0, cursor.column);
  591. snippetText = snippetText.replace(/\r/g, "");
  592. var tokens = this.tokenizeTmSnippet(snippetText);
  593. tokens = this.resolveVariables(tokens, editor);
  594. tokens = tokens.map(function (x) {
  595. if (x == "\n" && !options.excludeExtraIndent)
  596. return x + indentString;
  597. if (typeof x == "string")
  598. return x.replace(/\t/g, tabString);
  599. return x;
  600. });
  601. var tabstops = [];
  602. tokens.forEach(function (p, i) {
  603. if (typeof p != "object")
  604. return;
  605. var id = p.tabstopId;
  606. var ts = tabstops[id];
  607. if (!ts) {
  608. ts = tabstops[id] = [];
  609. ts.index = id;
  610. ts.value = "";
  611. ts.parents = {};
  612. }
  613. if (ts.indexOf(p) !== -1)
  614. return;
  615. if (p.choices && !ts.choices)
  616. ts.choices = p.choices;
  617. ts.push(p);
  618. var i1 = tokens.indexOf(p, i + 1);
  619. if (i1 === -1)
  620. return;
  621. var value = tokens.slice(i + 1, i1);
  622. var isNested = value.some(function (t) { return typeof t === "object"; });
  623. if (isNested && !ts.value) {
  624. ts.value = value;
  625. }
  626. else if (value.length && (!ts.value || typeof ts.value !== "string")) {
  627. ts.value = value.join("");
  628. }
  629. });
  630. tabstops.forEach(function (ts) { ts.length = 0; });
  631. var expanding = {};
  632. function copyValue(val) {
  633. var copy = [];
  634. for (var i = 0; i < val.length; i++) {
  635. var p = val[i];
  636. if (typeof p == "object") {
  637. if (expanding[p.tabstopId])
  638. continue;
  639. var j = val.lastIndexOf(p, i - 1);
  640. p = copy[j] || { tabstopId: p.tabstopId };
  641. }
  642. copy[i] = p;
  643. }
  644. return copy;
  645. }
  646. for (var i = 0; i < tokens.length; i++) {
  647. var p = tokens[i];
  648. if (typeof p != "object")
  649. continue;
  650. var id = p.tabstopId;
  651. var ts = tabstops[id];
  652. var i1 = tokens.indexOf(p, i + 1);
  653. if (expanding[id]) {
  654. if (expanding[id] === p) {
  655. delete expanding[id];
  656. Object.keys(expanding).forEach(function (parentId) {
  657. ts.parents[parentId] = true;
  658. });
  659. }
  660. continue;
  661. }
  662. expanding[id] = p;
  663. var value = ts.value;
  664. if (typeof value !== "string")
  665. value = copyValue(value);
  666. else if (p.fmt)
  667. value = this.tmStrFormat(value, p, editor);
  668. tokens.splice.apply(tokens, [i + 1, Math.max(0, i1 - i)].concat(value, p));
  669. if (ts.indexOf(p) === -1)
  670. ts.push(p);
  671. }
  672. var row = 0, column = 0;
  673. var text = "";
  674. tokens.forEach(function (t) {
  675. if (typeof t === "string") {
  676. var lines = t.split("\n");
  677. if (lines.length > 1) {
  678. column = lines[lines.length - 1].length;
  679. row += lines.length - 1;
  680. }
  681. else
  682. column += t.length;
  683. text += t;
  684. }
  685. else if (t) {
  686. if (!t.start)
  687. t.start = { row: row, column: column };
  688. else
  689. t.end = { row: row, column: column };
  690. }
  691. });
  692. return {
  693. text: text,
  694. tabstops: tabstops,
  695. tokens: tokens
  696. };
  697. };
  698. var TabstopManager = /** @class */ (function () {
  699. function TabstopManager(editor) {
  700. this.index = 0;
  701. this.ranges = [];
  702. this.tabstops = [];
  703. if (editor.tabstopManager)
  704. return editor.tabstopManager;
  705. editor.tabstopManager = this;
  706. this.$onChange = this.onChange.bind(this);
  707. this.$onChangeSelection = lang.delayedCall(this.onChangeSelection.bind(this)).schedule;
  708. this.$onChangeSession = this.onChangeSession.bind(this);
  709. this.$onAfterExec = this.onAfterExec.bind(this);
  710. this.attach(editor);
  711. }
  712. TabstopManager.prototype.attach = function (editor) {
  713. this.$openTabstops = null;
  714. this.selectedTabstop = null;
  715. this.editor = editor;
  716. this.session = editor.session;
  717. this.editor.on("change", this.$onChange);
  718. this.editor.on("changeSelection", this.$onChangeSelection);
  719. this.editor.on("changeSession", this.$onChangeSession);
  720. this.editor.commands.on("afterExec", this.$onAfterExec);
  721. this.editor.keyBinding.addKeyboardHandler(this.keyboardHandler);
  722. };
  723. TabstopManager.prototype.detach = function () {
  724. this.tabstops.forEach(this.removeTabstopMarkers, this);
  725. this.ranges.length = 0;
  726. this.tabstops.length = 0;
  727. this.selectedTabstop = null;
  728. this.editor.off("change", this.$onChange);
  729. this.editor.off("changeSelection", this.$onChangeSelection);
  730. this.editor.off("changeSession", this.$onChangeSession);
  731. this.editor.commands.off("afterExec", this.$onAfterExec);
  732. this.editor.keyBinding.removeKeyboardHandler(this.keyboardHandler);
  733. this.editor.tabstopManager = null;
  734. this.session = null;
  735. this.editor = null;
  736. };
  737. TabstopManager.prototype.onChange = function (delta) {
  738. var isRemove = delta.action[0] == "r";
  739. var selectedTabstop = this.selectedTabstop || {};
  740. var parents = selectedTabstop.parents || {};
  741. var tabstops = this.tabstops.slice();
  742. for (var i = 0; i < tabstops.length; i++) {
  743. var ts = tabstops[i];
  744. var active = ts == selectedTabstop || parents[ts.index];
  745. ts.rangeList.$bias = active ? 0 : 1;
  746. if (delta.action == "remove" && ts !== selectedTabstop) {
  747. var parentActive = ts.parents && ts.parents[selectedTabstop.index];
  748. var startIndex = ts.rangeList.pointIndex(delta.start, parentActive);
  749. startIndex = startIndex < 0 ? -startIndex - 1 : startIndex + 1;
  750. var endIndex = ts.rangeList.pointIndex(delta.end, parentActive);
  751. endIndex = endIndex < 0 ? -endIndex - 1 : endIndex - 1;
  752. var toRemove = ts.rangeList.ranges.slice(startIndex, endIndex);
  753. for (var j = 0; j < toRemove.length; j++)
  754. this.removeRange(toRemove[j]);
  755. }
  756. ts.rangeList.$onChange(delta);
  757. }
  758. var session = this.session;
  759. if (!this.$inChange && isRemove && session.getLength() == 1 && !session.getValue())
  760. this.detach();
  761. };
  762. TabstopManager.prototype.updateLinkedFields = function () {
  763. var ts = this.selectedTabstop;
  764. if (!ts || !ts.hasLinkedRanges || !ts.firstNonLinked)
  765. return;
  766. this.$inChange = true;
  767. var session = this.session;
  768. var text = session.getTextRange(ts.firstNonLinked);
  769. for (var i = 0; i < ts.length; i++) {
  770. var range = ts[i];
  771. if (!range.linked)
  772. continue;
  773. var original = range.original;
  774. var fmt = exports.snippetManager.tmStrFormat(text, original, this.editor);
  775. session.replace(range, fmt);
  776. }
  777. this.$inChange = false;
  778. };
  779. TabstopManager.prototype.onAfterExec = function (e) {
  780. if (e.command && !e.command.readOnly)
  781. this.updateLinkedFields();
  782. };
  783. TabstopManager.prototype.onChangeSelection = function () {
  784. if (!this.editor)
  785. return;
  786. var lead = this.editor.selection.lead;
  787. var anchor = this.editor.selection.anchor;
  788. var isEmpty = this.editor.selection.isEmpty();
  789. for (var i = 0; i < this.ranges.length; i++) {
  790. if (this.ranges[i].linked)
  791. continue;
  792. var containsLead = this.ranges[i].contains(lead.row, lead.column);
  793. var containsAnchor = isEmpty || this.ranges[i].contains(anchor.row, anchor.column);
  794. if (containsLead && containsAnchor)
  795. return;
  796. }
  797. this.detach();
  798. };
  799. TabstopManager.prototype.onChangeSession = function () {
  800. this.detach();
  801. };
  802. TabstopManager.prototype.tabNext = function (dir) {
  803. var max = this.tabstops.length;
  804. var index = this.index + (dir || 1);
  805. index = Math.min(Math.max(index, 1), max);
  806. if (index == max)
  807. index = 0;
  808. this.selectTabstop(index);
  809. this.updateTabstopMarkers();
  810. if (index === 0) {
  811. this.detach();
  812. }
  813. };
  814. TabstopManager.prototype.selectTabstop = function (index) {
  815. this.$openTabstops = null;
  816. var ts = this.tabstops[this.index];
  817. if (ts)
  818. this.addTabstopMarkers(ts);
  819. this.index = index;
  820. ts = this.tabstops[this.index];
  821. if (!ts || !ts.length)
  822. return;
  823. this.selectedTabstop = ts;
  824. var range = ts.firstNonLinked || ts;
  825. if (ts.choices)
  826. range.cursor = range.start;
  827. if (!this.editor.inVirtualSelectionMode) {
  828. var sel = this.editor.multiSelect;
  829. sel.toSingleRange(range);
  830. for (var i = 0; i < ts.length; i++) {
  831. if (ts.hasLinkedRanges && ts[i].linked)
  832. continue;
  833. sel.addRange(ts[i].clone(), true);
  834. }
  835. }
  836. else {
  837. this.editor.selection.fromOrientedRange(range);
  838. }
  839. this.editor.keyBinding.addKeyboardHandler(this.keyboardHandler);
  840. if (this.selectedTabstop && this.selectedTabstop.choices)
  841. this.editor.execCommand("startAutocomplete", { matches: this.selectedTabstop.choices });
  842. };
  843. TabstopManager.prototype.addTabstops = function (tabstops, start, end) {
  844. var useLink = this.useLink || !this.editor.getOption("enableMultiselect");
  845. if (!this.$openTabstops)
  846. this.$openTabstops = [];
  847. if (!tabstops[0]) {
  848. var p = Range.fromPoints(end, end);
  849. moveRelative(p.start, start);
  850. moveRelative(p.end, start);
  851. tabstops[0] = [p];
  852. tabstops[0].index = 0;
  853. }
  854. var i = this.index;
  855. var arg = [i + 1, 0];
  856. var ranges = this.ranges;
  857. var snippetId = this.snippetId = (this.snippetId || 0) + 1;
  858. tabstops.forEach(function (ts, index) {
  859. var dest = this.$openTabstops[index] || ts;
  860. dest.snippetId = snippetId;
  861. for (var i = 0; i < ts.length; i++) {
  862. var p = ts[i];
  863. var range = Range.fromPoints(p.start, p.end || p.start);
  864. movePoint(range.start, start);
  865. movePoint(range.end, start);
  866. range.original = p;
  867. range.tabstop = dest;
  868. ranges.push(range);
  869. if (dest != ts)
  870. dest.unshift(range);
  871. else
  872. dest[i] = range;
  873. if (p.fmtString || (dest.firstNonLinked && useLink)) {
  874. range.linked = true;
  875. dest.hasLinkedRanges = true;
  876. }
  877. else if (!dest.firstNonLinked)
  878. dest.firstNonLinked = range;
  879. }
  880. if (!dest.firstNonLinked)
  881. dest.hasLinkedRanges = false;
  882. if (dest === ts) {
  883. arg.push(dest);
  884. this.$openTabstops[index] = dest;
  885. }
  886. this.addTabstopMarkers(dest);
  887. dest.rangeList = dest.rangeList || new RangeList();
  888. dest.rangeList.$bias = 0;
  889. dest.rangeList.addList(dest);
  890. }, this);
  891. if (arg.length > 2) {
  892. if (this.tabstops.length)
  893. arg.push(arg.splice(2, 1)[0]);
  894. this.tabstops.splice.apply(this.tabstops, arg);
  895. }
  896. };
  897. TabstopManager.prototype.addTabstopMarkers = function (ts) {
  898. var session = this.session;
  899. ts.forEach(function (range) {
  900. if (!range.markerId)
  901. range.markerId = session.addMarker(range, "ace_snippet-marker", "text");
  902. });
  903. };
  904. TabstopManager.prototype.removeTabstopMarkers = function (ts) {
  905. var session = this.session;
  906. ts.forEach(function (range) {
  907. session.removeMarker(range.markerId);
  908. range.markerId = null;
  909. });
  910. };
  911. TabstopManager.prototype.updateTabstopMarkers = function () {
  912. if (!this.selectedTabstop)
  913. return;
  914. var currentSnippetId = this.selectedTabstop.snippetId;
  915. if (this.selectedTabstop.index === 0) {
  916. currentSnippetId--;
  917. }
  918. this.tabstops.forEach(function (ts) {
  919. if (ts.snippetId === currentSnippetId)
  920. this.addTabstopMarkers(ts);
  921. else
  922. this.removeTabstopMarkers(ts);
  923. }, this);
  924. };
  925. TabstopManager.prototype.removeRange = function (range) {
  926. var i = range.tabstop.indexOf(range);
  927. if (i != -1)
  928. range.tabstop.splice(i, 1);
  929. i = this.ranges.indexOf(range);
  930. if (i != -1)
  931. this.ranges.splice(i, 1);
  932. i = range.tabstop.rangeList.ranges.indexOf(range);
  933. if (i != -1)
  934. range.tabstop.splice(i, 1);
  935. this.session.removeMarker(range.markerId);
  936. if (!range.tabstop.length) {
  937. i = this.tabstops.indexOf(range.tabstop);
  938. if (i != -1)
  939. this.tabstops.splice(i, 1);
  940. if (!this.tabstops.length)
  941. this.detach();
  942. }
  943. };
  944. return TabstopManager;
  945. }());
  946. TabstopManager.prototype.keyboardHandler = new HashHandler();
  947. TabstopManager.prototype.keyboardHandler.bindKeys({
  948. "Tab": function (editor) {
  949. if (exports.snippetManager && exports.snippetManager.expandWithTab(editor))
  950. return;
  951. editor.tabstopManager.tabNext(1);
  952. editor.renderer.scrollCursorIntoView();
  953. },
  954. "Shift-Tab": function (editor) {
  955. editor.tabstopManager.tabNext(-1);
  956. editor.renderer.scrollCursorIntoView();
  957. },
  958. "Esc": function (editor) {
  959. editor.tabstopManager.detach();
  960. }
  961. });
  962. var movePoint = function (point, diff) {
  963. if (point.row == 0)
  964. point.column += diff.column;
  965. point.row += diff.row;
  966. };
  967. var moveRelative = function (point, start) {
  968. if (point.row == start.row)
  969. point.column -= start.column;
  970. point.row -= start.row;
  971. };
  972. dom.importCssString("\n.ace_snippet-marker {\n -moz-box-sizing: border-box;\n box-sizing: border-box;\n background: rgba(194, 193, 208, 0.09);\n border: 1px dotted rgba(211, 208, 235, 0.62);\n position: absolute;\n}", "snippets.css", false);
  973. exports.snippetManager = new SnippetManager();
  974. var Editor = require("./editor").Editor;
  975. (function () {
  976. this.insertSnippet = function (content, options) {
  977. return exports.snippetManager.insertSnippet(this, content, options);
  978. };
  979. this.expandSnippet = function (options) {
  980. return exports.snippetManager.expandWithTab(this, options);
  981. };
  982. }).call(Editor.prototype);
  983. });
  984. ace.define("ace/autocomplete/popup",["require","exports","module","ace/virtual_renderer","ace/editor","ace/range","ace/lib/event","ace/lib/lang","ace/lib/dom","ace/config","ace/lib/useragent"], function(require, exports, module){"use strict";
  985. var Renderer = require("../virtual_renderer").VirtualRenderer;
  986. var Editor = require("../editor").Editor;
  987. var Range = require("../range").Range;
  988. var event = require("../lib/event");
  989. var lang = require("../lib/lang");
  990. var dom = require("../lib/dom");
  991. var nls = require("../config").nls;
  992. var userAgent = require("./../lib/useragent");
  993. var getAriaId = function (index) {
  994. return "suggest-aria-id:".concat(index);
  995. };
  996. var popupAriaRole = userAgent.isSafari ? "menu" : "listbox";
  997. var optionAriaRole = userAgent.isSafari ? "menuitem" : "option";
  998. var ariaActiveState = userAgent.isSafari ? "aria-current" : "aria-selected";
  999. var $singleLineEditor = function (el) {
  1000. var renderer = new Renderer(el);
  1001. renderer.$maxLines = 4;
  1002. var editor = new Editor(renderer);
  1003. editor.setHighlightActiveLine(false);
  1004. editor.setShowPrintMargin(false);
  1005. editor.renderer.setShowGutter(false);
  1006. editor.renderer.setHighlightGutterLine(false);
  1007. editor.$mouseHandler.$focusTimeout = 0;
  1008. editor.$highlightTagPending = true;
  1009. return editor;
  1010. };
  1011. var AcePopup = /** @class */ (function () {
  1012. function AcePopup(parentNode) {
  1013. var el = dom.createElement("div");
  1014. var popup = $singleLineEditor(el);
  1015. if (parentNode) {
  1016. parentNode.appendChild(el);
  1017. }
  1018. el.style.display = "none";
  1019. popup.renderer.content.style.cursor = "default";
  1020. popup.renderer.setStyle("ace_autocomplete");
  1021. popup.renderer.$textLayer.element.setAttribute("role", popupAriaRole);
  1022. popup.renderer.$textLayer.element.setAttribute("aria-roledescription", nls("autocomplete.popup.aria-roledescription", "Autocomplete suggestions"));
  1023. popup.renderer.$textLayer.element.setAttribute("aria-label", nls("autocomplete.popup.aria-label", "Autocomplete suggestions"));
  1024. popup.renderer.textarea.setAttribute("aria-hidden", "true");
  1025. popup.setOption("displayIndentGuides", false);
  1026. popup.setOption("dragDelay", 150);
  1027. var noop = function () { };
  1028. popup.focus = noop;
  1029. popup.$isFocused = true;
  1030. popup.renderer.$cursorLayer.restartTimer = noop;
  1031. popup.renderer.$cursorLayer.element.style.opacity = "0";
  1032. popup.renderer.$maxLines = 8;
  1033. popup.renderer.$keepTextAreaAtCursor = false;
  1034. popup.setHighlightActiveLine(false);
  1035. popup.session.highlight("");
  1036. popup.session.$searchHighlight.clazz = "ace_highlight-marker";
  1037. popup.on("mousedown", function (e) {
  1038. var pos = e.getDocumentPosition();
  1039. popup.selection.moveToPosition(pos);
  1040. selectionMarker.start.row = selectionMarker.end.row = pos.row;
  1041. e.stop();
  1042. });
  1043. var lastMouseEvent;
  1044. var hoverMarker = new Range(-1, 0, -1, Infinity);
  1045. var selectionMarker = new Range(-1, 0, -1, Infinity);
  1046. selectionMarker.id = popup.session.addMarker(selectionMarker, "ace_active-line", "fullLine");
  1047. popup.setSelectOnHover = function (val) {
  1048. if (!val) {
  1049. hoverMarker.id = popup.session.addMarker(hoverMarker, "ace_line-hover", "fullLine");
  1050. }
  1051. else if (hoverMarker.id) {
  1052. popup.session.removeMarker(hoverMarker.id);
  1053. hoverMarker.id = null;
  1054. }
  1055. };
  1056. popup.setSelectOnHover(false);
  1057. popup.on("mousemove", function (e) {
  1058. if (!lastMouseEvent) {
  1059. lastMouseEvent = e;
  1060. return;
  1061. }
  1062. if (lastMouseEvent.x == e.x && lastMouseEvent.y == e.y) {
  1063. return;
  1064. }
  1065. lastMouseEvent = e;
  1066. lastMouseEvent.scrollTop = popup.renderer.scrollTop;
  1067. popup.isMouseOver = true;
  1068. var row = lastMouseEvent.getDocumentPosition().row;
  1069. if (hoverMarker.start.row != row) {
  1070. if (!hoverMarker.id)
  1071. popup.setRow(row);
  1072. setHoverMarker(row);
  1073. }
  1074. });
  1075. popup.renderer.on("beforeRender", function () {
  1076. if (lastMouseEvent && hoverMarker.start.row != -1) {
  1077. lastMouseEvent.$pos = null;
  1078. var row = lastMouseEvent.getDocumentPosition().row;
  1079. if (!hoverMarker.id)
  1080. popup.setRow(row);
  1081. setHoverMarker(row, true);
  1082. }
  1083. });
  1084. popup.renderer.on("afterRender", function () {
  1085. var t = popup.renderer.$textLayer;
  1086. for (var row = t.config.firstRow, l = t.config.lastRow; row <= l; row++) {
  1087. var popupRowElement = /** @type {HTMLElement|null} */ (t.element.childNodes[row - t.config.firstRow]);
  1088. popupRowElement.setAttribute("role", optionAriaRole);
  1089. popupRowElement.setAttribute("aria-roledescription", nls("autocomplete.popup.item.aria-roledescription", "item"));
  1090. popupRowElement.setAttribute("aria-setsize", popup.data.length);
  1091. popupRowElement.setAttribute("aria-describedby", "doc-tooltip");
  1092. popupRowElement.setAttribute("aria-posinset", row + 1);
  1093. var rowData = popup.getData(row);
  1094. if (rowData) {
  1095. var ariaLabel = "".concat(rowData.caption || rowData.value).concat(rowData.meta ? ", ".concat(rowData.meta) : '');
  1096. popupRowElement.setAttribute("aria-label", ariaLabel);
  1097. }
  1098. var highlightedSpans = popupRowElement.querySelectorAll(".ace_completion-highlight");
  1099. highlightedSpans.forEach(function (span) {
  1100. span.setAttribute("role", "mark");
  1101. });
  1102. }
  1103. });
  1104. popup.renderer.on("afterRender", function () {
  1105. var row = popup.getRow();
  1106. var t = popup.renderer.$textLayer;
  1107. var selected = /** @type {HTMLElement|null} */ (t.element.childNodes[row - t.config.firstRow]);
  1108. var el = document.activeElement; // Active element is textarea of main editor
  1109. if (selected !== popup.selectedNode && popup.selectedNode) {
  1110. dom.removeCssClass(popup.selectedNode, "ace_selected");
  1111. popup.selectedNode.removeAttribute(ariaActiveState);
  1112. popup.selectedNode.removeAttribute("id");
  1113. }
  1114. el.removeAttribute("aria-activedescendant");
  1115. popup.selectedNode = selected;
  1116. if (selected) {
  1117. var ariaId = getAriaId(row);
  1118. dom.addCssClass(selected, "ace_selected");
  1119. selected.id = ariaId;
  1120. t.element.setAttribute("aria-activedescendant", ariaId);
  1121. el.setAttribute("aria-activedescendant", ariaId);
  1122. selected.setAttribute(ariaActiveState, "true");
  1123. }
  1124. });
  1125. var hideHoverMarker = function () { setHoverMarker(-1); };
  1126. var setHoverMarker = function (row, suppressRedraw) {
  1127. if (row !== hoverMarker.start.row) {
  1128. hoverMarker.start.row = hoverMarker.end.row = row;
  1129. if (!suppressRedraw)
  1130. popup.session._emit("changeBackMarker");
  1131. popup._emit("changeHoverMarker");
  1132. }
  1133. };
  1134. popup.getHoveredRow = function () {
  1135. return hoverMarker.start.row;
  1136. };
  1137. event.addListener(popup.container, "mouseout", function () {
  1138. popup.isMouseOver = false;
  1139. hideHoverMarker();
  1140. });
  1141. popup.on("hide", hideHoverMarker);
  1142. popup.on("changeSelection", hideHoverMarker);
  1143. popup.session.doc.getLength = function () {
  1144. return popup.data.length;
  1145. };
  1146. popup.session.doc.getLine = function (i) {
  1147. var data = popup.data[i];
  1148. if (typeof data == "string")
  1149. return data;
  1150. return (data && data.value) || "";
  1151. };
  1152. var bgTokenizer = popup.session.bgTokenizer;
  1153. bgTokenizer.$tokenizeRow = function (row) {
  1154. var data = popup.data[row];
  1155. var tokens = [];
  1156. if (!data)
  1157. return tokens;
  1158. if (typeof data == "string")
  1159. data = { value: data };
  1160. var caption = data.caption || data.value || data.name;
  1161. function addToken(value, className) {
  1162. value && tokens.push({
  1163. type: (data.className || "") + (className || ""),
  1164. value: value
  1165. });
  1166. }
  1167. var lower = caption.toLowerCase();
  1168. var filterText = (popup.filterText || "").toLowerCase();
  1169. var lastIndex = 0;
  1170. var lastI = 0;
  1171. for (var i = 0; i <= filterText.length; i++) {
  1172. if (i != lastI && (data.matchMask & (1 << i) || i == filterText.length)) {
  1173. var sub = filterText.slice(lastI, i);
  1174. lastI = i;
  1175. var index = lower.indexOf(sub, lastIndex);
  1176. if (index == -1)
  1177. continue;
  1178. addToken(caption.slice(lastIndex, index), "");
  1179. lastIndex = index + sub.length;
  1180. addToken(caption.slice(index, lastIndex), "completion-highlight");
  1181. }
  1182. }
  1183. addToken(caption.slice(lastIndex, caption.length), "");
  1184. tokens.push({ type: "completion-spacer", value: " " });
  1185. if (data.meta)
  1186. tokens.push({ type: "completion-meta", value: data.meta });
  1187. if (data.message)
  1188. tokens.push({ type: "completion-message", value: data.message });
  1189. return tokens;
  1190. };
  1191. bgTokenizer.$updateOnChange = noop;
  1192. bgTokenizer.start = noop;
  1193. popup.session.$computeWidth = function () {
  1194. return this.screenWidth = 0;
  1195. };
  1196. popup.isOpen = false;
  1197. popup.isTopdown = false;
  1198. popup.autoSelect = true;
  1199. popup.filterText = "";
  1200. popup.isMouseOver = false;
  1201. popup.data = [];
  1202. popup.setData = function (list, filterText) {
  1203. popup.filterText = filterText || "";
  1204. popup.setValue(lang.stringRepeat("\n", list.length), -1);
  1205. popup.data = list || [];
  1206. popup.setRow(0);
  1207. };
  1208. popup.getData = function (row) {
  1209. return popup.data[row];
  1210. };
  1211. popup.getRow = function () {
  1212. return selectionMarker.start.row;
  1213. };
  1214. popup.setRow = function (line) {
  1215. line = Math.max(this.autoSelect ? 0 : -1, Math.min(this.data.length - 1, line));
  1216. if (selectionMarker.start.row != line) {
  1217. popup.selection.clearSelection();
  1218. selectionMarker.start.row = selectionMarker.end.row = line || 0;
  1219. popup.session._emit("changeBackMarker");
  1220. popup.moveCursorTo(line || 0, 0);
  1221. if (popup.isOpen)
  1222. popup._signal("select");
  1223. }
  1224. };
  1225. popup.on("changeSelection", function () {
  1226. if (popup.isOpen)
  1227. popup.setRow(popup.selection.lead.row);
  1228. popup.renderer.scrollCursorIntoView();
  1229. });
  1230. popup.hide = function () {
  1231. this.container.style.display = "none";
  1232. popup.anchorPos = null;
  1233. popup.anchor = null;
  1234. if (popup.isOpen) {
  1235. popup.isOpen = false;
  1236. this._signal("hide");
  1237. }
  1238. };
  1239. popup.tryShow = function (pos, lineHeight, anchor, forceShow) {
  1240. if (!forceShow && popup.isOpen && popup.anchorPos && popup.anchor &&
  1241. popup.anchorPos.top === pos.top && popup.anchorPos.left === pos.left &&
  1242. popup.anchor === anchor) {
  1243. return true;
  1244. }
  1245. var el = this.container;
  1246. var scrollBarSize = this.renderer.scrollBar.width || 10;
  1247. var screenHeight = window.innerHeight - scrollBarSize;
  1248. var screenWidth = window.innerWidth - scrollBarSize;
  1249. var renderer = this.renderer;
  1250. var maxH = renderer.$maxLines * lineHeight * 1.4;
  1251. var dims = { top: 0, bottom: 0, left: 0 };
  1252. var spaceBelow = screenHeight - pos.top - 3 * this.$borderSize - lineHeight;
  1253. var spaceAbove = pos.top - 3 * this.$borderSize;
  1254. if (!anchor) {
  1255. if (spaceAbove <= spaceBelow || spaceBelow >= maxH) {
  1256. anchor = "bottom";
  1257. }
  1258. else {
  1259. anchor = "top";
  1260. }
  1261. }
  1262. if (anchor === "top") {
  1263. dims.bottom = pos.top - this.$borderSize;
  1264. dims.top = dims.bottom - maxH;
  1265. }
  1266. else if (anchor === "bottom") {
  1267. dims.top = pos.top + lineHeight + this.$borderSize;
  1268. dims.bottom = dims.top + maxH;
  1269. }
  1270. var fitsX = dims.top >= 0 && dims.bottom <= screenHeight;
  1271. if (!forceShow && !fitsX) {
  1272. return false;
  1273. }
  1274. if (!fitsX) {
  1275. if (anchor === "top") {
  1276. renderer.$maxPixelHeight = spaceAbove;
  1277. }
  1278. else {
  1279. renderer.$maxPixelHeight = spaceBelow;
  1280. }
  1281. }
  1282. else {
  1283. renderer.$maxPixelHeight = null;
  1284. }
  1285. if (anchor === "top") {
  1286. el.style.top = "";
  1287. el.style.bottom = (screenHeight + scrollBarSize - dims.bottom) + "px";
  1288. popup.isTopdown = false;
  1289. }
  1290. else {
  1291. el.style.top = dims.top + "px";
  1292. el.style.bottom = "";
  1293. popup.isTopdown = true;
  1294. }
  1295. el.style.display = "";
  1296. var left = pos.left;
  1297. if (left + el.offsetWidth > screenWidth)
  1298. left = screenWidth - el.offsetWidth;
  1299. el.style.left = left + "px";
  1300. el.style.right = "";
  1301. if (!popup.isOpen) {
  1302. popup.isOpen = true;
  1303. this._signal("show");
  1304. lastMouseEvent = null;
  1305. }
  1306. popup.anchorPos = pos;
  1307. popup.anchor = anchor;
  1308. return true;
  1309. };
  1310. popup.show = function (pos, lineHeight, topdownOnly) {
  1311. this.tryShow(pos, lineHeight, topdownOnly ? "bottom" : undefined, true);
  1312. };
  1313. popup.goTo = function (where) {
  1314. var row = this.getRow();
  1315. var max = this.session.getLength() - 1;
  1316. switch (where) {
  1317. case "up":
  1318. row = row <= 0 ? max : row - 1;
  1319. break;
  1320. case "down":
  1321. row = row >= max ? -1 : row + 1;
  1322. break;
  1323. case "start":
  1324. row = 0;
  1325. break;
  1326. case "end":
  1327. row = max;
  1328. break;
  1329. }
  1330. this.setRow(row);
  1331. };
  1332. popup.getTextLeftOffset = function () {
  1333. return this.$borderSize + this.renderer.$padding + this.$imageSize;
  1334. };
  1335. popup.$imageSize = 0;
  1336. popup.$borderSize = 1;
  1337. return popup;
  1338. }
  1339. return AcePopup;
  1340. }());
  1341. dom.importCssString("\n.ace_editor.ace_autocomplete .ace_marker-layer .ace_active-line {\n background-color: #CAD6FA;\n z-index: 1;\n}\n.ace_dark.ace_editor.ace_autocomplete .ace_marker-layer .ace_active-line {\n background-color: #3a674e;\n}\n.ace_editor.ace_autocomplete .ace_line-hover {\n border: 1px solid #abbffe;\n margin-top: -1px;\n background: rgba(233,233,253,0.4);\n position: absolute;\n z-index: 2;\n}\n.ace_dark.ace_editor.ace_autocomplete .ace_line-hover {\n border: 1px solid rgba(109, 150, 13, 0.8);\n background: rgba(58, 103, 78, 0.62);\n}\n.ace_completion-meta {\n opacity: 0.5;\n margin-left: 0.9em;\n}\n.ace_completion-message {\n margin-left: 0.9em;\n color: blue;\n}\n.ace_editor.ace_autocomplete .ace_completion-highlight{\n color: #2d69c7;\n}\n.ace_dark.ace_editor.ace_autocomplete .ace_completion-highlight{\n color: #93ca12;\n}\n.ace_editor.ace_autocomplete {\n width: 300px;\n z-index: 200000;\n border: 1px lightgray solid;\n position: fixed;\n box-shadow: 2px 3px 5px rgba(0,0,0,.2);\n line-height: 1.4;\n background: #fefefe;\n color: #111;\n}\n.ace_dark.ace_editor.ace_autocomplete {\n border: 1px #484747 solid;\n box-shadow: 2px 3px 5px rgba(0, 0, 0, 0.51);\n line-height: 1.4;\n background: #25282c;\n color: #c1c1c1;\n}\n.ace_autocomplete .ace_text-layer {\n width: calc(100% - 8px);\n}\n.ace_autocomplete .ace_line {\n display: flex;\n align-items: center;\n}\n.ace_autocomplete .ace_line > * {\n min-width: 0;\n flex: 0 0 auto;\n}\n.ace_autocomplete .ace_line .ace_ {\n flex: 0 1 auto;\n overflow: hidden;\n text-overflow: ellipsis;\n}\n.ace_autocomplete .ace_completion-spacer {\n flex: 1;\n}\n.ace_autocomplete.ace_loading:after {\n content: \"\";\n position: absolute;\n top: 0px;\n height: 2px;\n width: 8%;\n background: blue;\n z-index: 100;\n animation: ace_progress 3s infinite linear;\n animation-delay: 300ms;\n transform: translateX(-100%) scaleX(1);\n}\n@keyframes ace_progress {\n 0% { transform: translateX(-100%) scaleX(1) }\n 50% { transform: translateX(625%) scaleX(2) } \n 100% { transform: translateX(1500%) scaleX(3) } \n}\n@media (prefers-reduced-motion) {\n .ace_autocomplete.ace_loading:after {\n transform: translateX(625%) scaleX(2);\n animation: none;\n }\n}\n", "autocompletion.css", false);
  1342. exports.AcePopup = AcePopup;
  1343. exports.$singleLineEditor = $singleLineEditor;
  1344. exports.getAriaId = getAriaId;
  1345. });
  1346. ace.define("ace/autocomplete/inline_screenreader",["require","exports","module"], function(require, exports, module){"use strict";
  1347. var AceInlineScreenReader = /** @class */ (function () {
  1348. function AceInlineScreenReader(editor) {
  1349. this.editor = editor;
  1350. this.screenReaderDiv = document.createElement("div");
  1351. this.screenReaderDiv.classList.add("ace_screenreader-only");
  1352. this.editor.container.appendChild(this.screenReaderDiv);
  1353. }
  1354. AceInlineScreenReader.prototype.setScreenReaderContent = function (content) {
  1355. if (!this.popup && this.editor.completer && /**@type{import("../autocomplete").Autocomplete}*/ (this.editor.completer).popup) {
  1356. this.popup = /**@type{import("../autocomplete").Autocomplete}*/ (this.editor.completer).popup;
  1357. this.popup.renderer.on("afterRender", function () {
  1358. var row = this.popup.getRow();
  1359. var t = this.popup.renderer.$textLayer;
  1360. var selected = t.element.childNodes[row - t.config.firstRow];
  1361. if (selected) {
  1362. var idString = "doc-tooltip ";
  1363. for (var lineIndex = 0; lineIndex < this._lines.length; lineIndex++) {
  1364. idString += "ace-inline-screenreader-line-".concat(lineIndex, " ");
  1365. }
  1366. selected.setAttribute("aria-describedby", idString);
  1367. }
  1368. }.bind(this));
  1369. }
  1370. while (this.screenReaderDiv.firstChild) {
  1371. this.screenReaderDiv.removeChild(this.screenReaderDiv.firstChild);
  1372. }
  1373. this._lines = content.split(/\r\n|\r|\n/);
  1374. var codeElement = this.createCodeBlock();
  1375. this.screenReaderDiv.appendChild(codeElement);
  1376. };
  1377. AceInlineScreenReader.prototype.destroy = function () {
  1378. this.screenReaderDiv.remove();
  1379. };
  1380. AceInlineScreenReader.prototype.createCodeBlock = function () {
  1381. var container = document.createElement("pre");
  1382. container.setAttribute("id", "ace-inline-screenreader");
  1383. for (var lineIndex = 0; lineIndex < this._lines.length; lineIndex++) {
  1384. var codeElement = document.createElement("code");
  1385. codeElement.setAttribute("id", "ace-inline-screenreader-line-".concat(lineIndex));
  1386. var line = document.createTextNode(this._lines[lineIndex]);
  1387. codeElement.appendChild(line);
  1388. container.appendChild(codeElement);
  1389. }
  1390. return container;
  1391. };
  1392. return AceInlineScreenReader;
  1393. }());
  1394. exports.AceInlineScreenReader = AceInlineScreenReader;
  1395. });
  1396. ace.define("ace/autocomplete/inline",["require","exports","module","ace/snippets","ace/autocomplete/inline_screenreader"], function(require, exports, module){"use strict";
  1397. var snippetManager = require("../snippets").snippetManager;
  1398. var AceInlineScreenReader = require("./inline_screenreader").AceInlineScreenReader;
  1399. var AceInline = /** @class */ (function () {
  1400. function AceInline() {
  1401. this.editor = null;
  1402. }
  1403. AceInline.prototype.show = function (editor, completion, prefix) {
  1404. prefix = prefix || "";
  1405. if (editor && this.editor && this.editor !== editor) {
  1406. this.hide();
  1407. this.editor = null;
  1408. this.inlineScreenReader = null;
  1409. }
  1410. if (!editor || !completion) {
  1411. return false;
  1412. }
  1413. if (!this.inlineScreenReader) {
  1414. this.inlineScreenReader = new AceInlineScreenReader(editor);
  1415. }
  1416. var displayText = completion.snippet ? snippetManager.getDisplayTextForSnippet(editor, completion.snippet) : completion.value;
  1417. if (completion.hideInlinePreview || !displayText || !displayText.startsWith(prefix)) {
  1418. return false;
  1419. }
  1420. this.editor = editor;
  1421. this.inlineScreenReader.setScreenReaderContent(displayText);
  1422. displayText = displayText.slice(prefix.length);
  1423. if (displayText === "") {
  1424. editor.removeGhostText();
  1425. }
  1426. else {
  1427. editor.setGhostText(displayText);
  1428. }
  1429. return true;
  1430. };
  1431. AceInline.prototype.isOpen = function () {
  1432. if (!this.editor) {
  1433. return false;
  1434. }
  1435. return !!this.editor.renderer.$ghostText;
  1436. };
  1437. AceInline.prototype.hide = function () {
  1438. if (!this.editor) {
  1439. return false;
  1440. }
  1441. this.editor.removeGhostText();
  1442. return true;
  1443. };
  1444. AceInline.prototype.destroy = function () {
  1445. this.hide();
  1446. this.editor = null;
  1447. if (this.inlineScreenReader) {
  1448. this.inlineScreenReader.destroy();
  1449. this.inlineScreenReader = null;
  1450. }
  1451. };
  1452. return AceInline;
  1453. }());
  1454. exports.AceInline = AceInline;
  1455. });
  1456. ace.define("ace/autocomplete/util",["require","exports","module"], function(require, exports, module){"use strict";
  1457. exports.parForEach = function (array, fn, callback) {
  1458. var completed = 0;
  1459. var arLength = array.length;
  1460. if (arLength === 0)
  1461. callback();
  1462. for (var i = 0; i < arLength; i++) {
  1463. fn(array[i], function (result, err) {
  1464. completed++;
  1465. if (completed === arLength)
  1466. callback(result, err);
  1467. });
  1468. }
  1469. };
  1470. var ID_REGEX = /[a-zA-Z_0-9\$\-\u00A2-\u2000\u2070-\uFFFF]/;
  1471. exports.retrievePrecedingIdentifier = function (text, pos, regex) {
  1472. regex = regex || ID_REGEX;
  1473. var buf = [];
  1474. for (var i = pos - 1; i >= 0; i--) {
  1475. if (regex.test(text[i]))
  1476. buf.push(text[i]);
  1477. else
  1478. break;
  1479. }
  1480. return buf.reverse().join("");
  1481. };
  1482. exports.retrieveFollowingIdentifier = function (text, pos, regex) {
  1483. regex = regex || ID_REGEX;
  1484. var buf = [];
  1485. for (var i = pos; i < text.length; i++) {
  1486. if (regex.test(text[i]))
  1487. buf.push(text[i]);
  1488. else
  1489. break;
  1490. }
  1491. return buf;
  1492. };
  1493. exports.getCompletionPrefix = function (editor) {
  1494. var pos = editor.getCursorPosition();
  1495. var line = editor.session.getLine(pos.row);
  1496. var prefix;
  1497. editor.completers.forEach(function (completer) {
  1498. if (completer.identifierRegexps) {
  1499. completer.identifierRegexps.forEach(function (identifierRegex) {
  1500. if (!prefix && identifierRegex)
  1501. prefix = this.retrievePrecedingIdentifier(line, pos.column, identifierRegex);
  1502. }.bind(this));
  1503. }
  1504. }.bind(this));
  1505. return prefix || this.retrievePrecedingIdentifier(line, pos.column);
  1506. };
  1507. exports.triggerAutocomplete = function (editor, previousChar) {
  1508. var previousChar = previousChar == null
  1509. ? editor.session.getPrecedingCharacter()
  1510. : previousChar;
  1511. return editor.completers.some(function (completer) {
  1512. if (completer.triggerCharacters && Array.isArray(completer.triggerCharacters)) {
  1513. return completer.triggerCharacters.includes(previousChar);
  1514. }
  1515. });
  1516. };
  1517. });
  1518. ace.define("ace/autocomplete",["require","exports","module","ace/keyboard/hash_handler","ace/autocomplete/popup","ace/autocomplete/inline","ace/autocomplete/popup","ace/autocomplete/util","ace/lib/lang","ace/lib/dom","ace/snippets","ace/config","ace/lib/event","ace/lib/scroll"], function(require, exports, module){"use strict";
  1519. var HashHandler = require("./keyboard/hash_handler").HashHandler;
  1520. var AcePopup = require("./autocomplete/popup").AcePopup;
  1521. var AceInline = require("./autocomplete/inline").AceInline;
  1522. var getAriaId = require("./autocomplete/popup").getAriaId;
  1523. var util = require("./autocomplete/util");
  1524. var lang = require("./lib/lang");
  1525. var dom = require("./lib/dom");
  1526. var snippetManager = require("./snippets").snippetManager;
  1527. var config = require("./config");
  1528. var event = require("./lib/event");
  1529. var preventParentScroll = require("./lib/scroll").preventParentScroll;
  1530. var destroyCompleter = function (e, editor) {
  1531. editor.completer && editor.completer.destroy();
  1532. };
  1533. var Autocomplete = /** @class */ (function () {
  1534. function Autocomplete() {
  1535. this.autoInsert = false;
  1536. this.autoSelect = true;
  1537. this.autoShown = false;
  1538. this.exactMatch = false;
  1539. this.inlineEnabled = false;
  1540. this.keyboardHandler = new HashHandler();
  1541. this.keyboardHandler.bindKeys(this.commands);
  1542. this.parentNode = null;
  1543. this.setSelectOnHover = false;
  1544. this.hasSeen = new Set();
  1545. this.showLoadingState = false;
  1546. this.stickySelectionDelay = 500;
  1547. this.blurListener = this.blurListener.bind(this);
  1548. this.changeListener = this.changeListener.bind(this);
  1549. this.mousedownListener = this.mousedownListener.bind(this);
  1550. this.mousewheelListener = this.mousewheelListener.bind(this);
  1551. this.onLayoutChange = this.onLayoutChange.bind(this);
  1552. this.changeTimer = lang.delayedCall(function () {
  1553. this.updateCompletions(true);
  1554. }.bind(this));
  1555. this.tooltipTimer = lang.delayedCall(this.updateDocTooltip.bind(this), 50);
  1556. this.popupTimer = lang.delayedCall(this.$updatePopupPosition.bind(this), 50);
  1557. this.stickySelectionTimer = lang.delayedCall(function () {
  1558. this.stickySelection = true;
  1559. }.bind(this), this.stickySelectionDelay);
  1560. this.$firstOpenTimer = lang.delayedCall(/**@this{Autocomplete}*/ function () {
  1561. var initialPosition = this.completionProvider && this.completionProvider.initialPosition;
  1562. if (this.autoShown || (this.popup && this.popup.isOpen) || !initialPosition || this.editor.completers.length === 0)
  1563. return;
  1564. this.completions = new FilteredList(Autocomplete.completionsForLoading);
  1565. this.openPopup(this.editor, initialPosition.prefix, false);
  1566. this.popup.renderer.setStyle("ace_loading", true);
  1567. }.bind(this), this.stickySelectionDelay);
  1568. }
  1569. Object.defineProperty(Autocomplete, "completionsForLoading", {
  1570. get: function () {
  1571. return [{
  1572. caption: config.nls("autocomplete.loading", "Loading..."),
  1573. value: ""
  1574. }];
  1575. },
  1576. enumerable: false,
  1577. configurable: true
  1578. });
  1579. Autocomplete.prototype.$init = function () {
  1580. this.popup = new AcePopup(this.parentNode || document.body || document.documentElement);
  1581. this.popup.on("click", function (e) {
  1582. this.insertMatch();
  1583. e.stop();
  1584. }.bind(this));
  1585. this.popup.focus = this.editor.focus.bind(this.editor);
  1586. this.popup.on("show", this.$onPopupShow.bind(this));
  1587. this.popup.on("hide", this.$onHidePopup.bind(this));
  1588. this.popup.on("select", this.$onPopupChange.bind(this));
  1589. event.addListener(this.popup.container, "mouseout", this.mouseOutListener.bind(this));
  1590. this.popup.on("changeHoverMarker", this.tooltipTimer.bind(null, null));
  1591. this.popup.renderer.on("afterRender", this.$onPopupRender.bind(this));
  1592. return this.popup;
  1593. };
  1594. Autocomplete.prototype.$initInline = function () {
  1595. if (!this.inlineEnabled || this.inlineRenderer)
  1596. return;
  1597. this.inlineRenderer = new AceInline();
  1598. return this.inlineRenderer;
  1599. };
  1600. Autocomplete.prototype.getPopup = function () {
  1601. return this.popup || this.$init();
  1602. };
  1603. Autocomplete.prototype.$onHidePopup = function () {
  1604. if (this.inlineRenderer) {
  1605. this.inlineRenderer.hide();
  1606. }
  1607. this.hideDocTooltip();
  1608. this.stickySelectionTimer.cancel();
  1609. this.popupTimer.cancel();
  1610. this.stickySelection = false;
  1611. };
  1612. Autocomplete.prototype.$seen = function (completion) {
  1613. if (!this.hasSeen.has(completion) && completion && completion.completer && completion.completer.onSeen && typeof completion.completer.onSeen === "function") {
  1614. completion.completer.onSeen(this.editor, completion);
  1615. this.hasSeen.add(completion);
  1616. }
  1617. };
  1618. Autocomplete.prototype.$onPopupChange = function (hide) {
  1619. if (this.inlineRenderer && this.inlineEnabled) {
  1620. var completion = hide ? null : this.popup.getData(this.popup.getRow());
  1621. this.$updateGhostText(completion);
  1622. if (this.popup.isMouseOver && this.setSelectOnHover) {
  1623. this.tooltipTimer.call(null, null);
  1624. return;
  1625. }
  1626. this.popupTimer.schedule();
  1627. this.tooltipTimer.schedule();
  1628. }
  1629. else {
  1630. this.popupTimer.call(null, null);
  1631. this.tooltipTimer.call(null, null);
  1632. }
  1633. };
  1634. Autocomplete.prototype.$updateGhostText = function (completion) {
  1635. var row = this.base.row;
  1636. var column = this.base.column;
  1637. var cursorColumn = this.editor.getCursorPosition().column;
  1638. var prefix = this.editor.session.getLine(row).slice(column, cursorColumn);
  1639. if (!this.inlineRenderer.show(this.editor, completion, prefix)) {
  1640. this.inlineRenderer.hide();
  1641. }
  1642. else {
  1643. this.$seen(completion);
  1644. }
  1645. };
  1646. Autocomplete.prototype.$onPopupRender = function () {
  1647. var inlineEnabled = this.inlineRenderer && this.inlineEnabled;
  1648. if (this.completions && this.completions.filtered && this.completions.filtered.length > 0) {
  1649. for (var i = this.popup.getFirstVisibleRow(); i <= this.popup.getLastVisibleRow(); i++) {
  1650. var completion = this.popup.getData(i);
  1651. if (completion && (!inlineEnabled || completion.hideInlinePreview)) {
  1652. this.$seen(completion);
  1653. }
  1654. }
  1655. }
  1656. };
  1657. Autocomplete.prototype.$onPopupShow = function (hide) {
  1658. this.$onPopupChange(hide);
  1659. this.stickySelection = false;
  1660. if (this.stickySelectionDelay >= 0)
  1661. this.stickySelectionTimer.schedule(this.stickySelectionDelay);
  1662. };
  1663. Autocomplete.prototype.observeLayoutChanges = function () {
  1664. if (this.$elements || !this.editor)
  1665. return;
  1666. window.addEventListener("resize", this.onLayoutChange, { passive: true });
  1667. window.addEventListener("wheel", this.mousewheelListener);
  1668. var el = this.editor.container.parentNode;
  1669. var elements = [];
  1670. while (el) {
  1671. elements.push(el);
  1672. el.addEventListener("scroll", this.onLayoutChange, { passive: true });
  1673. el = el.parentNode;
  1674. }
  1675. this.$elements = elements;
  1676. };
  1677. Autocomplete.prototype.unObserveLayoutChanges = function () {
  1678. var _this = this;
  1679. window.removeEventListener("resize", this.onLayoutChange, { passive: true });
  1680. window.removeEventListener("wheel", this.mousewheelListener);
  1681. this.$elements && this.$elements.forEach(function (el) {
  1682. el.removeEventListener("scroll", _this.onLayoutChange, { passive: true });
  1683. });
  1684. this.$elements = null;
  1685. };
  1686. Autocomplete.prototype.onLayoutChange = function () {
  1687. if (!this.popup.isOpen)
  1688. return this.unObserveLayoutChanges();
  1689. this.$updatePopupPosition();
  1690. this.updateDocTooltip();
  1691. };
  1692. Autocomplete.prototype.$updatePopupPosition = function () {
  1693. var editor = this.editor;
  1694. var renderer = editor.renderer;
  1695. var lineHeight = renderer.layerConfig.lineHeight;
  1696. var pos = renderer.$cursorLayer.getPixelPosition(this.base, true);
  1697. pos.left -= this.popup.getTextLeftOffset();
  1698. var rect = editor.container.getBoundingClientRect();
  1699. pos.top += rect.top - renderer.layerConfig.offset;
  1700. pos.left += rect.left - editor.renderer.scrollLeft;
  1701. pos.left += renderer.gutterWidth;
  1702. var posGhostText = {
  1703. top: pos.top,
  1704. left: pos.left
  1705. };
  1706. if (renderer.$ghostText && renderer.$ghostTextWidget) {
  1707. if (this.base.row === renderer.$ghostText.position.row) {
  1708. posGhostText.top += renderer.$ghostTextWidget.el.offsetHeight;
  1709. }
  1710. }
  1711. var editorContainerBottom = editor.container.getBoundingClientRect().bottom - lineHeight;
  1712. var lowestPosition = editorContainerBottom < posGhostText.top ?
  1713. { top: editorContainerBottom, left: posGhostText.left } :
  1714. posGhostText;
  1715. if (this.popup.tryShow(lowestPosition, lineHeight, "bottom")) {
  1716. return;
  1717. }
  1718. if (this.popup.tryShow(pos, lineHeight, "top")) {
  1719. return;
  1720. }
  1721. this.popup.show(pos, lineHeight);
  1722. };
  1723. Autocomplete.prototype.openPopup = function (editor, prefix, keepPopupPosition) {
  1724. this.$firstOpenTimer.cancel();
  1725. if (!this.popup)
  1726. this.$init();
  1727. if (this.inlineEnabled && !this.inlineRenderer)
  1728. this.$initInline();
  1729. this.popup.autoSelect = this.autoSelect;
  1730. this.popup.setSelectOnHover(this.setSelectOnHover);
  1731. var oldRow = this.popup.getRow();
  1732. var previousSelectedItem = this.popup.data[oldRow];
  1733. this.popup.setData(this.completions.filtered, this.completions.filterText);
  1734. if (this.editor.textInput.setAriaOptions) {
  1735. this.editor.textInput.setAriaOptions({
  1736. activeDescendant: getAriaId(this.popup.getRow()),
  1737. inline: this.inlineEnabled
  1738. });
  1739. }
  1740. editor.keyBinding.addKeyboardHandler(this.keyboardHandler);
  1741. var newRow;
  1742. if (this.stickySelection)
  1743. newRow = this.popup.data.indexOf(previousSelectedItem);
  1744. if (!newRow || newRow === -1)
  1745. newRow = 0;
  1746. this.popup.setRow(this.autoSelect ? newRow : -1);
  1747. if (newRow === oldRow && previousSelectedItem !== this.completions.filtered[newRow])
  1748. this.$onPopupChange();
  1749. var inlineEnabled = this.inlineRenderer && this.inlineEnabled;
  1750. if (newRow === oldRow && inlineEnabled) {
  1751. var completion = this.popup.getData(this.popup.getRow());
  1752. this.$updateGhostText(completion);
  1753. }
  1754. if (!keepPopupPosition) {
  1755. this.popup.setTheme(editor.getTheme());
  1756. this.popup.setFontSize(editor.getFontSize());
  1757. this.$updatePopupPosition();
  1758. if (this.tooltipNode) {
  1759. this.updateDocTooltip();
  1760. }
  1761. }
  1762. this.changeTimer.cancel();
  1763. this.observeLayoutChanges();
  1764. };
  1765. Autocomplete.prototype.detach = function () {
  1766. if (this.editor) {
  1767. this.editor.keyBinding.removeKeyboardHandler(this.keyboardHandler);
  1768. this.editor.off("changeSelection", this.changeListener);
  1769. this.editor.off("blur", this.blurListener);
  1770. this.editor.off("mousedown", this.mousedownListener);
  1771. this.editor.off("mousewheel", this.mousewheelListener);
  1772. }
  1773. this.$firstOpenTimer.cancel();
  1774. this.changeTimer.cancel();
  1775. this.hideDocTooltip();
  1776. if (this.completionProvider) {
  1777. this.completionProvider.detach();
  1778. }
  1779. if (this.popup && this.popup.isOpen)
  1780. this.popup.hide();
  1781. if (this.popup && this.popup.renderer) {
  1782. this.popup.renderer.off("afterRender", this.$onPopupRender);
  1783. }
  1784. if (this.base)
  1785. this.base.detach();
  1786. this.activated = false;
  1787. this.completionProvider = this.completions = this.base = null;
  1788. this.unObserveLayoutChanges();
  1789. };
  1790. Autocomplete.prototype.changeListener = function (e) {
  1791. var cursor = this.editor.selection.lead;
  1792. if (cursor.row != this.base.row || cursor.column < this.base.column) {
  1793. this.detach();
  1794. }
  1795. if (this.activated)
  1796. this.changeTimer.schedule();
  1797. else
  1798. this.detach();
  1799. };
  1800. Autocomplete.prototype.blurListener = function (e) {
  1801. var el = document.activeElement;
  1802. var text = this.editor.textInput.getElement();
  1803. var fromTooltip = e.relatedTarget && this.tooltipNode && this.tooltipNode.contains(e.relatedTarget);
  1804. var container = this.popup && this.popup.container;
  1805. if (el != text && el.parentNode != container && !fromTooltip
  1806. && el != this.tooltipNode && e.relatedTarget != text) {
  1807. this.detach();
  1808. }
  1809. };
  1810. Autocomplete.prototype.mousedownListener = function (e) {
  1811. this.detach();
  1812. };
  1813. Autocomplete.prototype.mousewheelListener = function (e) {
  1814. if (this.popup && !this.popup.isMouseOver)
  1815. this.detach();
  1816. };
  1817. Autocomplete.prototype.mouseOutListener = function (e) {
  1818. if (this.popup.isOpen)
  1819. this.$updatePopupPosition();
  1820. };
  1821. Autocomplete.prototype.goTo = function (where) {
  1822. this.popup.goTo(where);
  1823. };
  1824. Autocomplete.prototype.insertMatch = function (data, options) {
  1825. if (!data)
  1826. data = this.popup.getData(this.popup.getRow());
  1827. if (!data)
  1828. return false;
  1829. if (data.value === "") // Explicitly given nothing to insert, e.g. "No suggestion state"
  1830. return this.detach();
  1831. var completions = this.completions;
  1832. var result = this.getCompletionProvider().insertMatch(this.editor, data, completions.filterText, options);
  1833. if (this.completions == completions)
  1834. this.detach();
  1835. return result;
  1836. };
  1837. Autocomplete.prototype.showPopup = function (editor, options) {
  1838. if (this.editor)
  1839. this.detach();
  1840. this.activated = true;
  1841. this.editor = editor;
  1842. if (editor.completer != this) {
  1843. if (editor.completer)
  1844. editor.completer.detach();
  1845. editor.completer = this;
  1846. }
  1847. editor.on("changeSelection", this.changeListener);
  1848. editor.on("blur", this.blurListener);
  1849. editor.on("mousedown", this.mousedownListener);
  1850. editor.on("mousewheel", this.mousewheelListener);
  1851. this.updateCompletions(false, options);
  1852. };
  1853. Autocomplete.prototype.getCompletionProvider = function (initialPosition) {
  1854. if (!this.completionProvider)
  1855. this.completionProvider = new CompletionProvider(initialPosition);
  1856. return this.completionProvider;
  1857. };
  1858. Autocomplete.prototype.gatherCompletions = function (editor, callback) {
  1859. return this.getCompletionProvider().gatherCompletions(editor, callback);
  1860. };
  1861. Autocomplete.prototype.updateCompletions = function (keepPopupPosition, options) {
  1862. if (keepPopupPosition && this.base && this.completions) {
  1863. var pos = this.editor.getCursorPosition();
  1864. var prefix = this.editor.session.getTextRange({ start: this.base, end: pos });
  1865. if (prefix == this.completions.filterText)
  1866. return;
  1867. this.completions.setFilter(prefix);
  1868. if (!this.completions.filtered.length)
  1869. return this.detach();
  1870. if (this.completions.filtered.length == 1
  1871. && this.completions.filtered[0].value == prefix
  1872. && !this.completions.filtered[0].snippet)
  1873. return this.detach();
  1874. this.openPopup(this.editor, prefix, keepPopupPosition);
  1875. return;
  1876. }
  1877. if (options && options.matches) {
  1878. var pos = this.editor.getSelectionRange().start;
  1879. this.base = this.editor.session.doc.createAnchor(pos.row, pos.column);
  1880. this.base.$insertRight = true;
  1881. this.completions = new FilteredList(options.matches);
  1882. this.getCompletionProvider().completions = this.completions;
  1883. return this.openPopup(this.editor, "", keepPopupPosition);
  1884. }
  1885. var session = this.editor.getSession();
  1886. var pos = this.editor.getCursorPosition();
  1887. var prefix = util.getCompletionPrefix(this.editor);
  1888. this.base = session.doc.createAnchor(pos.row, pos.column - prefix.length);
  1889. this.base.$insertRight = true;
  1890. var completionOptions = {
  1891. exactMatch: this.exactMatch,
  1892. ignoreCaption: this.ignoreCaption
  1893. };
  1894. this.getCompletionProvider({
  1895. prefix: prefix,
  1896. pos: pos
  1897. }).provideCompletions(this.editor, completionOptions,
  1898. function (err, completions, finished) {
  1899. var filtered = completions.filtered;
  1900. var prefix = util.getCompletionPrefix(this.editor);
  1901. this.$firstOpenTimer.cancel();
  1902. if (finished) {
  1903. if (!filtered.length) {
  1904. var emptyMessage = !this.autoShown && this.emptyMessage;
  1905. if (typeof emptyMessage == "function")
  1906. emptyMessage = this.emptyMessage(prefix);
  1907. if (emptyMessage) {
  1908. var completionsForEmpty = [{
  1909. caption: emptyMessage,
  1910. value: ""
  1911. }
  1912. ];
  1913. this.completions = new FilteredList(completionsForEmpty);
  1914. this.openPopup(this.editor, prefix, keepPopupPosition);
  1915. this.popup.renderer.setStyle("ace_loading", false);
  1916. this.popup.renderer.setStyle("ace_empty-message", true);
  1917. return;
  1918. }
  1919. return this.detach();
  1920. }
  1921. if (filtered.length == 1 && filtered[0].value == prefix
  1922. && !filtered[0].snippet)
  1923. return this.detach();
  1924. if (this.autoInsert && !this.autoShown && filtered.length == 1)
  1925. return this.insertMatch(filtered[0]);
  1926. }
  1927. this.completions = !finished && this.showLoadingState ?
  1928. new FilteredList(Autocomplete.completionsForLoading.concat(filtered), completions.filterText) :
  1929. completions;
  1930. this.openPopup(this.editor, prefix, keepPopupPosition);
  1931. this.popup.renderer.setStyle("ace_empty-message", false);
  1932. this.popup.renderer.setStyle("ace_loading", !finished);
  1933. }.bind(this));
  1934. if (this.showLoadingState && !this.autoShown && !(this.popup && this.popup.isOpen)) {
  1935. this.$firstOpenTimer.delay(this.stickySelectionDelay / 2);
  1936. }
  1937. };
  1938. Autocomplete.prototype.cancelContextMenu = function () {
  1939. this.editor.$mouseHandler.cancelContextMenu();
  1940. };
  1941. Autocomplete.prototype.updateDocTooltip = function () {
  1942. var popup = this.popup;
  1943. var all = this.completions && this.completions.filtered;
  1944. var selected = all && (all[popup.getHoveredRow()] || all[popup.getRow()]);
  1945. var doc = null;
  1946. if (!selected || !this.editor || !this.popup.isOpen)
  1947. return this.hideDocTooltip();
  1948. var completersLength = this.editor.completers.length;
  1949. for (var i = 0; i < completersLength; i++) {
  1950. var completer = this.editor.completers[i];
  1951. if (completer.getDocTooltip && selected.completerId === completer.id) {
  1952. doc = completer.getDocTooltip(selected);
  1953. break;
  1954. }
  1955. }
  1956. if (!doc && typeof selected != "string")
  1957. doc = selected;
  1958. if (typeof doc == "string")
  1959. doc = { docText: doc };
  1960. if (!doc || !(doc.docHTML || doc.docText))
  1961. return this.hideDocTooltip();
  1962. this.showDocTooltip(doc);
  1963. };
  1964. Autocomplete.prototype.showDocTooltip = function (item) {
  1965. if (!this.tooltipNode) {
  1966. this.tooltipNode = dom.createElement("div");
  1967. this.tooltipNode.style.margin = "0";
  1968. this.tooltipNode.style.pointerEvents = "auto";
  1969. this.tooltipNode.style.overscrollBehavior = "contain";
  1970. this.tooltipNode.tabIndex = -1;
  1971. this.tooltipNode.onblur = this.blurListener.bind(this);
  1972. this.tooltipNode.onclick = this.onTooltipClick.bind(this);
  1973. this.tooltipNode.id = "doc-tooltip";
  1974. this.tooltipNode.setAttribute("role", "tooltip");
  1975. this.tooltipNode.addEventListener("wheel", preventParentScroll);
  1976. }
  1977. var theme = this.editor.renderer.theme;
  1978. this.tooltipNode.className = "ace_tooltip ace_doc-tooltip " +
  1979. (theme.isDark ? "ace_dark " : "") + (theme.cssClass || "");
  1980. var tooltipNode = this.tooltipNode;
  1981. if (item.docHTML) {
  1982. tooltipNode.innerHTML = item.docHTML;
  1983. }
  1984. else if (item.docText) {
  1985. tooltipNode.textContent = item.docText;
  1986. }
  1987. if (!tooltipNode.parentNode)
  1988. this.popup.container.appendChild(this.tooltipNode);
  1989. var popup = this.popup;
  1990. var rect = popup.container.getBoundingClientRect();
  1991. var targetWidth = 400;
  1992. var targetHeight = 300;
  1993. var scrollBarSize = popup.renderer.scrollBar.width || 10;
  1994. var leftSize = rect.left;
  1995. var rightSize = window.innerWidth - rect.right - scrollBarSize;
  1996. var topSize = popup.isTopdown ? rect.top : window.innerHeight - scrollBarSize - rect.bottom;
  1997. var scores = [
  1998. Math.min(rightSize / targetWidth, 1),
  1999. Math.min(leftSize / targetWidth, 1),
  2000. Math.min(topSize / targetHeight * 0.9),
  2001. ];
  2002. var max = Math.max.apply(Math, scores);
  2003. var tooltipStyle = tooltipNode.style;
  2004. tooltipStyle.display = "block";
  2005. if (max == scores[0]) {
  2006. tooltipStyle.left = (rect.right + 1) + "px";
  2007. tooltipStyle.right = "";
  2008. tooltipStyle.maxWidth = targetWidth * max + "px";
  2009. tooltipStyle.top = rect.top + "px";
  2010. tooltipStyle.bottom = "";
  2011. tooltipStyle.maxHeight = Math.min(window.innerHeight - scrollBarSize - rect.top, targetHeight) + "px";
  2012. }
  2013. else if (max == scores[1]) {
  2014. tooltipStyle.right = window.innerWidth - rect.left + "px";
  2015. tooltipStyle.left = "";
  2016. tooltipStyle.maxWidth = targetWidth * max + "px";
  2017. tooltipStyle.top = rect.top + "px";
  2018. tooltipStyle.bottom = "";
  2019. tooltipStyle.maxHeight = Math.min(window.innerHeight - scrollBarSize - rect.top, targetHeight) + "px";
  2020. }
  2021. else if (max == scores[2]) {
  2022. tooltipStyle.left = window.innerWidth - rect.left + "px";
  2023. tooltipStyle.maxWidth = Math.min(targetWidth, window.innerWidth) + "px";
  2024. if (popup.isTopdown) {
  2025. tooltipStyle.top = rect.bottom + "px";
  2026. tooltipStyle.left = rect.left + "px";
  2027. tooltipStyle.right = "";
  2028. tooltipStyle.bottom = "";
  2029. tooltipStyle.maxHeight = Math.min(window.innerHeight - scrollBarSize - rect.bottom, targetHeight) + "px";
  2030. }
  2031. else {
  2032. tooltipStyle.top = popup.container.offsetTop - tooltipNode.offsetHeight + "px";
  2033. tooltipStyle.left = rect.left + "px";
  2034. tooltipStyle.right = "";
  2035. tooltipStyle.bottom = "";
  2036. tooltipStyle.maxHeight = Math.min(popup.container.offsetTop, targetHeight) + "px";
  2037. }
  2038. }
  2039. };
  2040. Autocomplete.prototype.hideDocTooltip = function () {
  2041. this.tooltipTimer.cancel();
  2042. if (!this.tooltipNode)
  2043. return;
  2044. var el = this.tooltipNode;
  2045. if (!this.editor.isFocused() && document.activeElement == el)
  2046. this.editor.focus();
  2047. this.tooltipNode = null;
  2048. if (el.parentNode)
  2049. el.parentNode.removeChild(el);
  2050. };
  2051. Autocomplete.prototype.onTooltipClick = function (e) {
  2052. var a = e.target;
  2053. while (a && a != this.tooltipNode) {
  2054. if (a.nodeName == "A" && a.href) {
  2055. a.rel = "noreferrer";
  2056. a.target = "_blank";
  2057. break;
  2058. }
  2059. a = a.parentNode;
  2060. }
  2061. };
  2062. Autocomplete.prototype.destroy = function () {
  2063. this.detach();
  2064. if (this.popup) {
  2065. this.popup.destroy();
  2066. var el = this.popup.container;
  2067. if (el && el.parentNode)
  2068. el.parentNode.removeChild(el);
  2069. }
  2070. if (this.editor && this.editor.completer == this) {
  2071. this.editor.off("destroy", destroyCompleter);
  2072. this.editor.completer = null;
  2073. }
  2074. this.inlineRenderer = this.popup = this.editor = null;
  2075. };
  2076. Autocomplete.for = function (editor) {
  2077. if (editor.completer instanceof Autocomplete) {
  2078. return editor.completer;
  2079. }
  2080. if (editor.completer) {
  2081. editor.completer.destroy();
  2082. editor.completer = null;
  2083. }
  2084. if (config.get("sharedPopups")) {
  2085. if (!Autocomplete["$sharedInstance"])
  2086. Autocomplete["$sharedInstance"] = new Autocomplete();
  2087. editor.completer = Autocomplete["$sharedInstance"];
  2088. }
  2089. else {
  2090. editor.completer = new Autocomplete();
  2091. editor.once("destroy", destroyCompleter);
  2092. }
  2093. return editor.completer;
  2094. };
  2095. return Autocomplete;
  2096. }());
  2097. Autocomplete.prototype.commands = {
  2098. "Up": function (editor) { editor.completer.goTo("up"); },
  2099. "Down": function (editor) { editor.completer.goTo("down"); },
  2100. "Ctrl-Up|Ctrl-Home": function (editor) { editor.completer.goTo("start"); },
  2101. "Ctrl-Down|Ctrl-End": function (editor) { editor.completer.goTo("end"); },
  2102. "Esc": function (editor) { editor.completer.detach(); },
  2103. "Return": function (editor) { return editor.completer.insertMatch(); },
  2104. "Shift-Return": function (editor) { editor.completer.insertMatch(null, { deleteSuffix: true }); },
  2105. "Tab": function (editor) {
  2106. var result = editor.completer.insertMatch();
  2107. if (!result && !editor.tabstopManager)
  2108. editor.completer.goTo("down");
  2109. else
  2110. return result;
  2111. },
  2112. "Backspace": function (editor) {
  2113. editor.execCommand("backspace");
  2114. var prefix = util.getCompletionPrefix(editor);
  2115. if (!prefix && editor.completer)
  2116. editor.completer.detach();
  2117. },
  2118. "PageUp": function (editor) { editor.completer.popup.gotoPageUp(); },
  2119. "PageDown": function (editor) { editor.completer.popup.gotoPageDown(); }
  2120. };
  2121. Autocomplete.startCommand = {
  2122. name: "startAutocomplete",
  2123. exec: function (editor, options) {
  2124. var completer = Autocomplete.for(editor);
  2125. completer.autoInsert = false;
  2126. completer.autoSelect = true;
  2127. completer.autoShown = false;
  2128. completer.showPopup(editor, options);
  2129. completer.cancelContextMenu();
  2130. },
  2131. bindKey: "Ctrl-Space|Ctrl-Shift-Space|Alt-Space"
  2132. };
  2133. var CompletionProvider = /** @class */ (function () {
  2134. function CompletionProvider(initialPosition) {
  2135. this.initialPosition = initialPosition;
  2136. this.active = true;
  2137. }
  2138. CompletionProvider.prototype.insertByIndex = function (editor, index, options) {
  2139. if (!this.completions || !this.completions.filtered) {
  2140. return false;
  2141. }
  2142. return this.insertMatch(editor, this.completions.filtered[index], options);
  2143. };
  2144. CompletionProvider.prototype.insertMatch = function (editor, data, options) {
  2145. if (!data)
  2146. return false;
  2147. editor.startOperation({ command: { name: "insertMatch" } });
  2148. if (data.completer && data.completer.insertMatch) {
  2149. data.completer.insertMatch(editor, data);
  2150. }
  2151. else {
  2152. if (!this.completions)
  2153. return false;
  2154. var replaceBefore = this.completions.filterText.length;
  2155. var replaceAfter = 0;
  2156. if (data.range && data.range.start.row === data.range.end.row) {
  2157. replaceBefore -= this.initialPosition.prefix.length;
  2158. replaceBefore += this.initialPosition.pos.column - data.range.start.column;
  2159. replaceAfter += data.range.end.column - this.initialPosition.pos.column;
  2160. }
  2161. if (replaceBefore || replaceAfter) {
  2162. var ranges;
  2163. if (editor.selection.getAllRanges) {
  2164. ranges = editor.selection.getAllRanges();
  2165. }
  2166. else {
  2167. ranges = [editor.getSelectionRange()];
  2168. }
  2169. for (var i = 0, range; range = ranges[i]; i++) {
  2170. range.start.column -= replaceBefore;
  2171. range.end.column += replaceAfter;
  2172. editor.session.remove(range);
  2173. }
  2174. }
  2175. if (data.snippet) {
  2176. snippetManager.insertSnippet(editor, data.snippet);
  2177. }
  2178. else {
  2179. this.$insertString(editor, data);
  2180. }
  2181. if (data.completer && data.completer.onInsert && typeof data.completer.onInsert == "function") {
  2182. data.completer.onInsert(editor, data);
  2183. }
  2184. if (data.command && data.command === "startAutocomplete") {
  2185. editor.execCommand(data.command);
  2186. }
  2187. }
  2188. editor.endOperation();
  2189. return true;
  2190. };
  2191. CompletionProvider.prototype.$insertString = function (editor, data) {
  2192. var text = data.value || data;
  2193. editor.execCommand("insertstring", text);
  2194. };
  2195. CompletionProvider.prototype.gatherCompletions = function (editor, callback) {
  2196. var session = editor.getSession();
  2197. var pos = editor.getCursorPosition();
  2198. var prefix = util.getCompletionPrefix(editor);
  2199. var matches = [];
  2200. this.completers = editor.completers;
  2201. var total = editor.completers.length;
  2202. editor.completers.forEach(function (completer, i) {
  2203. completer.getCompletions(editor, session, pos, prefix, function (err, results) {
  2204. if (completer.hideInlinePreview)
  2205. results = results.map(function (result) {
  2206. return Object.assign(result, { hideInlinePreview: completer.hideInlinePreview });
  2207. });
  2208. if (!err && results)
  2209. matches = matches.concat(results);
  2210. callback(null, {
  2211. prefix: util.getCompletionPrefix(editor),
  2212. matches: matches,
  2213. finished: (--total === 0)
  2214. });
  2215. });
  2216. });
  2217. return true;
  2218. };
  2219. CompletionProvider.prototype.provideCompletions = function (editor, options, callback) {
  2220. var processResults = function (results) {
  2221. var prefix = results.prefix;
  2222. var matches = results.matches;
  2223. this.completions = new FilteredList(matches);
  2224. if (options.exactMatch)
  2225. this.completions.exactMatch = true;
  2226. if (options.ignoreCaption)
  2227. this.completions.ignoreCaption = true;
  2228. this.completions.setFilter(prefix);
  2229. if (results.finished || this.completions.filtered.length)
  2230. callback(null, this.completions, results.finished);
  2231. }.bind(this);
  2232. var isImmediate = true;
  2233. var immediateResults = null;
  2234. this.gatherCompletions(editor, function (err, results) {
  2235. if (!this.active) {
  2236. return;
  2237. }
  2238. if (err) {
  2239. callback(err, [], true);
  2240. this.detach();
  2241. }
  2242. var prefix = results.prefix;
  2243. if (prefix.indexOf(results.prefix) !== 0)
  2244. return;
  2245. if (isImmediate) {
  2246. immediateResults = results;
  2247. return;
  2248. }
  2249. processResults(results);
  2250. }.bind(this));
  2251. isImmediate = false;
  2252. if (immediateResults) {
  2253. var results = immediateResults;
  2254. immediateResults = null;
  2255. processResults(results);
  2256. }
  2257. };
  2258. CompletionProvider.prototype.detach = function () {
  2259. this.active = false;
  2260. this.completers && this.completers.forEach(function (completer) {
  2261. if (typeof completer.cancel === "function") {
  2262. completer.cancel();
  2263. }
  2264. });
  2265. };
  2266. return CompletionProvider;
  2267. }());
  2268. var FilteredList = /** @class */ (function () {
  2269. function FilteredList(array, filterText) {
  2270. this.all = array;
  2271. this.filtered = array;
  2272. this.filterText = filterText || "";
  2273. this.exactMatch = false;
  2274. this.ignoreCaption = false;
  2275. }
  2276. FilteredList.prototype.setFilter = function (str) {
  2277. if (str.length > this.filterText && str.lastIndexOf(this.filterText, 0) === 0)
  2278. var matches = this.filtered;
  2279. else
  2280. var matches = this.all;
  2281. this.filterText = str;
  2282. matches = this.filterCompletions(matches, this.filterText);
  2283. matches = matches.sort(function (a, b) {
  2284. return b.exactMatch - a.exactMatch || b.$score - a.$score
  2285. || (a.caption || a.value).localeCompare(b.caption || b.value);
  2286. });
  2287. var prev = null;
  2288. matches = matches.filter(function (item) {
  2289. var caption = item.snippet || item.caption || item.value;
  2290. if (caption === prev)
  2291. return false;
  2292. prev = caption;
  2293. return true;
  2294. });
  2295. this.filtered = matches;
  2296. };
  2297. FilteredList.prototype.filterCompletions = function (items, needle) {
  2298. var results = [];
  2299. var upper = needle.toUpperCase();
  2300. var lower = needle.toLowerCase();
  2301. loop: for (var i = 0, item; item = items[i]; i++) {
  2302. if (item.skipFilter) {
  2303. item.$score = item.score;
  2304. results.push(item);
  2305. continue;
  2306. }
  2307. var caption = (!this.ignoreCaption && item.caption) || item.value || item.snippet;
  2308. if (!caption)
  2309. continue;
  2310. var lastIndex = -1;
  2311. var matchMask = 0;
  2312. var penalty = 0;
  2313. var index, distance;
  2314. if (this.exactMatch) {
  2315. if (needle !== caption.substr(0, needle.length))
  2316. continue loop;
  2317. }
  2318. else {
  2319. var fullMatchIndex = caption.toLowerCase().indexOf(lower);
  2320. if (fullMatchIndex > -1) {
  2321. penalty = fullMatchIndex;
  2322. }
  2323. else {
  2324. for (var j = 0; j < needle.length; j++) {
  2325. var i1 = caption.indexOf(lower[j], lastIndex + 1);
  2326. var i2 = caption.indexOf(upper[j], lastIndex + 1);
  2327. index = (i1 >= 0) ? ((i2 < 0 || i1 < i2) ? i1 : i2) : i2;
  2328. if (index < 0)
  2329. continue loop;
  2330. distance = index - lastIndex - 1;
  2331. if (distance > 0) {
  2332. if (lastIndex === -1)
  2333. penalty += 10;
  2334. penalty += distance;
  2335. matchMask = matchMask | (1 << j);
  2336. }
  2337. lastIndex = index;
  2338. }
  2339. }
  2340. }
  2341. item.matchMask = matchMask;
  2342. item.exactMatch = penalty ? 0 : 1;
  2343. item.$score = (item.score || 0) - penalty;
  2344. results.push(item);
  2345. }
  2346. return results;
  2347. };
  2348. return FilteredList;
  2349. }());
  2350. exports.Autocomplete = Autocomplete;
  2351. exports.CompletionProvider = CompletionProvider;
  2352. exports.FilteredList = FilteredList;
  2353. });
  2354. ace.define("ace/autocomplete/text_completer",["require","exports","module","ace/range"], function(require, exports, module){var Range = require("../range").Range;
  2355. var splitRegex = /[^a-zA-Z_0-9\$\-\u00C0-\u1FFF\u2C00-\uD7FF\w]+/;
  2356. function getWordIndex(doc, pos) {
  2357. var textBefore = doc.getTextRange(Range.fromPoints({
  2358. row: 0,
  2359. column: 0
  2360. }, pos));
  2361. return textBefore.split(splitRegex).length - 1;
  2362. }
  2363. function wordDistance(doc, pos) {
  2364. var prefixPos = getWordIndex(doc, pos);
  2365. var words = doc.getValue().split(splitRegex);
  2366. var wordScores = Object.create(null);
  2367. var currentWord = words[prefixPos];
  2368. words.forEach(function (word, idx) {
  2369. if (!word || word === currentWord)
  2370. return;
  2371. var distance = Math.abs(prefixPos - idx);
  2372. var score = words.length - distance;
  2373. if (wordScores[word]) {
  2374. wordScores[word] = Math.max(score, wordScores[word]);
  2375. }
  2376. else {
  2377. wordScores[word] = score;
  2378. }
  2379. });
  2380. return wordScores;
  2381. }
  2382. exports.getCompletions = function (editor, session, pos, prefix, callback) {
  2383. var wordScore = wordDistance(session, pos);
  2384. var wordList = Object.keys(wordScore);
  2385. callback(null, wordList.map(function (word) {
  2386. return {
  2387. caption: word,
  2388. value: word,
  2389. score: wordScore[word],
  2390. meta: "local"
  2391. };
  2392. }));
  2393. };
  2394. });
  2395. ace.define("ace/ext/language_tools",["require","exports","module","ace/snippets","ace/autocomplete","ace/config","ace/lib/lang","ace/autocomplete/util","ace/autocomplete/text_completer","ace/editor","ace/config"], function(require, exports, module){"use strict";
  2396. var snippetManager = require("../snippets").snippetManager;
  2397. var Autocomplete = require("../autocomplete").Autocomplete;
  2398. var config = require("../config");
  2399. var lang = require("../lib/lang");
  2400. var util = require("../autocomplete/util");
  2401. var textCompleter = require("../autocomplete/text_completer");
  2402. var keyWordCompleter = {
  2403. getCompletions: function (editor, session, pos, prefix, callback) {
  2404. if (session.$mode.completer) {
  2405. return session.$mode.completer.getCompletions(editor, session, pos, prefix, callback);
  2406. }
  2407. var state = editor.session.getState(pos.row);
  2408. var completions = session.$mode.getCompletions(state, session, pos, prefix);
  2409. completions = completions.map(function (el) {
  2410. el.completerId = keyWordCompleter.id;
  2411. return el;
  2412. });
  2413. callback(null, completions);
  2414. },
  2415. id: "keywordCompleter"
  2416. };
  2417. var transformSnippetTooltip = function (str) {
  2418. var record = {};
  2419. return str.replace(/\${(\d+)(:(.*?))?}/g, function (_, p1, p2, p3) {
  2420. return (record[p1] = p3 || '');
  2421. }).replace(/\$(\d+?)/g, function (_, p1) {
  2422. return record[p1];
  2423. });
  2424. };
  2425. var snippetCompleter = {
  2426. getCompletions: function (editor, session, pos, prefix, callback) {
  2427. var scopes = [];
  2428. var token = session.getTokenAt(pos.row, pos.column);
  2429. if (token && token.type.match(/(tag-name|tag-open|tag-whitespace|attribute-name|attribute-value)\.xml$/))
  2430. scopes.push('html-tag');
  2431. else
  2432. scopes = snippetManager.getActiveScopes(editor);
  2433. var snippetMap = snippetManager.snippetMap;
  2434. var completions = [];
  2435. scopes.forEach(function (scope) {
  2436. var snippets = snippetMap[scope] || [];
  2437. for (var i = snippets.length; i--;) {
  2438. var s = snippets[i];
  2439. var caption = s.name || s.tabTrigger;
  2440. if (!caption)
  2441. continue;
  2442. completions.push({
  2443. caption: caption,
  2444. snippet: s.content,
  2445. meta: s.tabTrigger && !s.name ? s.tabTrigger + "\u21E5 " : "snippet",
  2446. completerId: snippetCompleter.id
  2447. });
  2448. }
  2449. }, this);
  2450. callback(null, completions);
  2451. },
  2452. getDocTooltip: function (item) {
  2453. if (item.snippet && !item.docHTML) {
  2454. item.docHTML = [
  2455. "<b>", lang.escapeHTML(item.caption), "</b>", "<hr></hr>",
  2456. lang.escapeHTML(transformSnippetTooltip(item.snippet))
  2457. ].join("");
  2458. }
  2459. },
  2460. id: "snippetCompleter"
  2461. };
  2462. var completers = [snippetCompleter, textCompleter, keyWordCompleter];
  2463. exports.setCompleters = function (val) {
  2464. completers.length = 0;
  2465. if (val)
  2466. completers.push.apply(completers, val);
  2467. };
  2468. exports.addCompleter = function (completer) {
  2469. completers.push(completer);
  2470. };
  2471. exports.textCompleter = textCompleter;
  2472. exports.keyWordCompleter = keyWordCompleter;
  2473. exports.snippetCompleter = snippetCompleter;
  2474. var expandSnippet = {
  2475. name: "expandSnippet",
  2476. exec: function (editor) {
  2477. return snippetManager.expandWithTab(editor);
  2478. },
  2479. bindKey: "Tab"
  2480. };
  2481. var onChangeMode = function (e, editor) {
  2482. loadSnippetsForMode(editor.session.$mode);
  2483. };
  2484. var loadSnippetsForMode = function (mode) {
  2485. if (typeof mode == "string")
  2486. mode = config.$modes[mode];
  2487. if (!mode)
  2488. return;
  2489. if (!snippetManager.files)
  2490. snippetManager.files = {};
  2491. loadSnippetFile(mode.$id, mode.snippetFileId);
  2492. if (mode.modes)
  2493. mode.modes.forEach(loadSnippetsForMode);
  2494. };
  2495. var loadSnippetFile = function (id, snippetFilePath) {
  2496. if (!snippetFilePath || !id || snippetManager.files[id])
  2497. return;
  2498. snippetManager.files[id] = {};
  2499. config.loadModule(snippetFilePath, function (m) {
  2500. if (!m)
  2501. return;
  2502. snippetManager.files[id] = m;
  2503. if (!m.snippets && m.snippetText)
  2504. m.snippets = snippetManager.parseSnippetFile(m.snippetText);
  2505. snippetManager.register(m.snippets || [], m.scope);
  2506. if (m.includeScopes) {
  2507. snippetManager.snippetMap[m.scope].includeScopes = m.includeScopes;
  2508. m.includeScopes.forEach(function (x) {
  2509. loadSnippetsForMode("ace/mode/" + x);
  2510. });
  2511. }
  2512. });
  2513. };
  2514. var doLiveAutocomplete = function (e) {
  2515. var editor = e.editor;
  2516. var hasCompleter = editor.completer && editor.completer.activated;
  2517. if (e.command.name === "backspace") {
  2518. if (hasCompleter && !util.getCompletionPrefix(editor))
  2519. editor.completer.detach();
  2520. }
  2521. else if (e.command.name === "insertstring" && !hasCompleter) {
  2522. lastExecEvent = e;
  2523. var delay = e.editor.$liveAutocompletionDelay;
  2524. if (delay) {
  2525. liveAutocompleteTimer.delay(delay);
  2526. }
  2527. else {
  2528. showLiveAutocomplete(e);
  2529. }
  2530. }
  2531. };
  2532. var lastExecEvent;
  2533. var liveAutocompleteTimer = lang.delayedCall(function () {
  2534. showLiveAutocomplete(lastExecEvent);
  2535. }, 0);
  2536. var showLiveAutocomplete = function (e) {
  2537. var editor = e.editor;
  2538. var prefix = util.getCompletionPrefix(editor);
  2539. var previousChar = e.args;
  2540. var triggerAutocomplete = util.triggerAutocomplete(editor, previousChar);
  2541. if (prefix && prefix.length >= editor.$liveAutocompletionThreshold || triggerAutocomplete) {
  2542. var completer = Autocomplete.for(editor);
  2543. completer.autoShown = true;
  2544. completer.showPopup(editor);
  2545. }
  2546. };
  2547. var Editor = require("../editor").Editor;
  2548. require("../config").defineOptions(Editor.prototype, "editor", {
  2549. enableBasicAutocompletion: {
  2550. set: function (val) {
  2551. if (val) {
  2552. if (!this.completers)
  2553. this.completers = Array.isArray(val) ? val : completers;
  2554. this.commands.addCommand(Autocomplete.startCommand);
  2555. }
  2556. else {
  2557. this.commands.removeCommand(Autocomplete.startCommand);
  2558. }
  2559. },
  2560. value: false
  2561. },
  2562. enableLiveAutocompletion: {
  2563. set: function (val) {
  2564. if (val) {
  2565. if (!this.completers)
  2566. this.completers = Array.isArray(val) ? val : completers;
  2567. this.commands.on('afterExec', doLiveAutocomplete);
  2568. }
  2569. else {
  2570. this.commands.off('afterExec', doLiveAutocomplete);
  2571. }
  2572. },
  2573. value: false
  2574. },
  2575. liveAutocompletionDelay: {
  2576. initialValue: 0
  2577. },
  2578. liveAutocompletionThreshold: {
  2579. initialValue: 0
  2580. },
  2581. enableSnippets: {
  2582. set: function (val) {
  2583. if (val) {
  2584. this.commands.addCommand(expandSnippet);
  2585. this.on("changeMode", onChangeMode);
  2586. onChangeMode(null, this);
  2587. }
  2588. else {
  2589. this.commands.removeCommand(expandSnippet);
  2590. this.off("changeMode", onChangeMode);
  2591. }
  2592. },
  2593. value: false
  2594. }
  2595. });
  2596. }); (function() {
  2597. ace.require(["ace/ext/language_tools"], function(m) {
  2598. if (typeof module == "object" && typeof exports == "object" && module) {
  2599. module.exports = m;
  2600. }
  2601. });
  2602. })();