prism-code-editor
Version:
Lightweight, extensible code editor component for the web using Prism
457 lines (456 loc) • 15.8 kB
JavaScript
import { b as createTemplate, a as addListener, p as preventDefault } from "./index-CKRNGLIi.js";
import { addTooltip } from "./tooltips.js";
import { u as updateNode, b as getLanguage, e as getLineBefore, p as prevSelection, c as getModifierCode, g as getStyleValue, i as insertText, a as addOverlay, s as setSelection } from "./index-BvZmi6ce.js";
import { m as matchTemplate, s as searchTemplate } from "./search-Dk5fdw4x.js";
const optionsFromKeys = (obj, icon) => Object.keys(obj).map((tag) => ({ label: tag, icon }));
const updateMatched = (container, matched, text) => {
let nodes = container.childNodes;
let nodeCount = nodes.length - 1;
let pos = 0;
let i = 0;
let l = matched.length;
for (; i < l; ) {
if (i >= nodeCount) {
nodes[i].before("", matchTemplate());
}
updateNode(nodes[i], text.slice(pos, pos = matched[i++]));
updateNode(nodes[i].firstChild, text.slice(pos, pos = matched[i++]));
}
for (; nodeCount > i; ) {
nodes[--nodeCount].remove();
}
updateNode(nodes[l], text.slice(pos));
};
const completeFromList = (options) => {
return ({ path, explicit, pos }) => {
if (path?.length == 1 && (path[0] || explicit)) {
return {
from: pos - path[0].length,
options
};
}
};
};
const findWords = (context, editor, filter, pattern, init, tokensOnly) => {
const cursorPos = context.pos;
const definition = map[context.language];
const result = new Set(init);
const search = (tokens, pos, isCorrectLang) => {
let i = 0;
let token;
for (; token = tokens[i++]; ) {
if (typeof token == "string") {
if (!tokensOnly && isCorrectLang) match(token, pos);
} else {
const type = token.type;
const content = token.content;
const aliasType = token.alias || type;
if (Array.isArray(content)) {
if (!isCorrectLang || filter(type, pos)) {
search(
content,
pos,
aliasType.slice(0, 9) == "language-" ? definition == map[aliasType.slice(9)] : isCorrectLang
);
}
} else if (isCorrectLang && filter(type, pos)) match(content, pos);
}
pos += token.length;
}
};
const match = (token, pos) => {
let match2;
while (match2 = pattern.exec(token)) {
let start = pos + match2.index;
let str = match2[0];
if (start > cursorPos || start + str.length < cursorPos) result.add(str);
}
};
search(editor.tokens, 0, definition == map[editor.options.language]);
return result;
};
const attrSnippet = (name, quotes, icon, boost) => ({
label: name,
icon,
insert: name + "=" + quotes,
tabStops: [name.length + 2, name.length + 2, name.length + 3],
boost
});
const completionsFromRecords = (records, icon) => {
const names = /* @__PURE__ */ new Set();
records.forEach((tags) => {
for (let key in tags) names.add(key);
});
return Array.from(names, (name) => ({
label: name,
icon
}));
};
let count = 0;
const template = /* @__PURE__ */ createTemplate("<div class=pce-ac-tooltip><ul role=listbox>");
const rowTemplate = /* @__PURE__ */ createTemplate(
"<li class=pce-ac-row role=option><div></div><div> </div><div class=pce-ac-details> "
);
const map = {};
const registerCompletions = (langs, definition) => {
langs.forEach((lang) => map[lang] = definition);
};
const autoComplete = (config) => (editor, options) => {
let isOpen;
let isTyping;
let shouldOpen;
let currentOptions;
let numOptions;
let activeIndex;
let active;
let pos;
let offset;
let rowHeight;
let cursor;
let stops;
let activeStop;
let currentSelection;
let prevLength;
let isDeleteForwards;
const windowSize = 13;
const textarea = editor.textarea;
const getSelection = editor.getSelection;
const tooltip = template();
const tabStopsContainer = searchTemplate();
const [show, _hide] = addTooltip(editor, tooltip);
const list = tooltip.firstChild;
const id = list.id = "pce-ac-" + count++;
const rows = list.children;
const prevIcons = [];
const hide = () => {
if (isOpen) {
_hide();
textarea.removeAttribute("aria-controls");
textarea.removeAttribute("aria-haspopup");
textarea.removeAttribute("aria-activedescendant");
isOpen = false;
}
};
const setRowHeight = () => {
rowHeight = getStyleValue(rows[0], "height");
};
const updateRow = (index) => {
const option = currentOptions[index + offset];
const [iconEl, labelEl, detailsEl] = rows[index].children;
const completion = option[4];
const icon = completion.icon || "variable";
updateMatched(labelEl, option[1], completion.label);
updateNode(detailsEl.firstChild, completion.detail || "");
if (prevIcons[index] != icon) {
iconEl.className = `pce-ac-icon pce-ac-icon-${prevIcons[index] = icon}`;
iconEl.style.color = `var(--pce-ac-icon-${icon})`;
}
};
const scrollActiveIntoView = () => {
setRowHeight();
const scrollTop = tooltip.scrollTop;
const lower = rowHeight * activeIndex;
const upper = rowHeight * (activeIndex + 1) - tooltip.clientHeight;
tooltip.scrollTop = scrollTop > lower ? lower : scrollTop < upper ? upper : scrollTop;
};
const updateActive = () => {
const newActive = rows[activeIndex - offset];
if (newActive != active) {
active?.removeAttribute("aria-selected");
if (newActive) {
textarea.setAttribute("aria-activedescendant", newActive.id);
newActive.setAttribute("aria-selected", true);
} else {
textarea.removeAttribute("aria-activedescendant");
}
active = newActive;
}
};
const move = (decrement) => {
if (decrement) activeIndex = activeIndex ? activeIndex - 1 : numOptions - 1;
else activeIndex = activeIndex + 1 < numOptions ? activeIndex + 1 : 0;
scrollActiveIntoView();
updateActive();
};
const insertOption = (index) => {
if (options.readOnly) return;
let [, , start, end, completion] = currentOptions[index];
let { label, tabStops = [], insert } = completion;
let l = tabStops.length;
tabStops = tabStops.map((stop) => stop + start);
if (insert) {
let indent = "\n" + getLineBefore(editor.value, pos).match(/\s*/)[0];
let tab = options.insertSpaces == false ? " " : " ".repeat(options.tabSize || 2);
let temp = tabStops.slice();
insert = insert.replace(/\n| /g, (match, index2) => {
let replacement = match == " " ? tab : indent;
let diff = replacement.length - 1;
let i = 0;
while (i < l) {
if (temp[i] > index2 + start) tabStops[i] += diff;
i++;
}
return replacement;
});
} else insert = label;
if (l % 2) tabStops[l] = tabStops[l - 1];
insertText(editor, insert, start, end, tabStops[0], tabStops[1]);
if (l > 2) {
if (!stops) addOverlay(editor, tabStopsContainer);
stops = tabStops;
activeStop = 0;
prevLength = editor.value.length;
updateStops();
currentSelection = getSelection();
}
cursor.scrollIntoView();
};
const moveActiveStop = (offset2) => {
activeStop += offset2;
setSelection(editor, stops[activeStop], stops[activeStop + 1]);
tabStopsContainer.children[activeStop / 2].scrollIntoView({ block: "nearest" });
};
const clearStops = () => {
tabStopsContainer.remove();
stops = null;
};
const updateStops = () => {
let sorted = [];
let i = 0;
for (; i < stops.length; ) sorted[i / 2] = [stops[i++], stops[i++]];
sorted.sort((a, b) => a[0] - b[0]);
updateMatched(tabStopsContainer, sorted.flat(), editor.value);
};
const startQuery = (explicit = false) => {
const [start, end, dir] = getSelection();
const language = getLanguage(editor, pos = dir < "f" ? start : end);
const definition = map[language];
if (definition && (explicit || start == end) && !options.readOnly) {
const value = editor.value;
const lineBefore = getLineBefore(value, pos);
const before = value.slice(0, pos);
const context = {
before,
lineBefore,
language,
explicit,
pos
};
const newContext = Object.assign(context, definition.context?.(context, editor));
const filter = config.filter;
currentOptions = [];
definition.sources.forEach((source) => {
const result = source(newContext, editor);
if (result) {
const from = result.from;
const query = before.slice(from);
result.options.forEach((option) => {
const filterResult = filter(query, option.label);
if (filterResult) {
filterResult[0] += option.boost || 0;
filterResult.push(from, result.to ?? end, option);
currentOptions.push(filterResult);
}
});
}
});
if (currentOptions[0]) {
currentOptions.sort((a, b) => b[0] - a[0] || a[4].label.localeCompare(b[4].label));
numOptions = currentOptions.length;
activeIndex = offset = 0;
for (let i = 0, l = numOptions < windowSize ? numOptions : windowSize; i < l; ) {
updateRow(i++);
}
if (!isOpen) {
const { clientHeight, clientWidth } = editor.container;
const pos2 = cursor.getPosition();
const max = Math.max(pos2.bottom, pos2.top);
tooltip.style.width = `min(25em, ${clientWidth}px - var(--padding-left) - 1em)`;
tooltip.style.maxHeight = `min(17em, ${max}px + .25em, ${clientHeight}px - 2em)`;
}
list.style.paddingTop = "";
list.style.height = rowHeight ? rowHeight * numOptions + "px" : 1.4 * numOptions + "em";
tooltip.scrollTop = 0;
isOpen = true;
show(config.preferAbove);
textarea.setAttribute("aria-controls", id);
textarea.setAttribute("aria-haspopup", "listbox");
updateActive();
} else hide();
} else hide();
};
const addSelectionHandler = () => {
if (!cursor && (cursor = editor.extensions.cursor)) {
editor.on("selectionChange", (selection) => {
if (stops && (selection[0] < stops[activeStop] || selection[1] > stops[activeStop + 1])) {
clearStops();
}
if (shouldOpen) {
shouldOpen = false;
startQuery();
} else hide();
isTyping = false;
});
}
};
tabStopsContainer.className = "pce-tabstops";
textarea.setAttribute("aria-autocomplete", "list");
for (let i = 0; i < windowSize; ) {
list.append(rowTemplate());
rows[i].id = id + "-" + i++;
}
addListener(tooltip, "scroll", () => {
setRowHeight();
const newOffset = Math.min(Math.floor(tooltip.scrollTop / rowHeight), numOptions - windowSize);
if (newOffset == offset || newOffset < 0) return;
offset = newOffset;
for (let i = 0; i < windowSize; i) {
updateRow(i++);
}
list.style.paddingTop = offset * rowHeight + "px";
updateActive();
});
editor.on("update", () => {
addSelectionHandler();
if (stops) {
let value = editor.value;
let diff = prevLength - (prevLength = value.length);
let [start, end] = currentSelection;
let i = 0;
let l = stops.length;
let activeStart = stops[activeStop];
let activeEnd = stops[activeStop + 1];
if (start < stops[activeStop] || end > activeEnd) {
clearStops();
} else {
if (isDeleteForwards) end++;
if (end <= activeEnd) stops[activeStop + 1] -= diff;
if (end <= activeStart && diff > 0) stops[activeStop] -= diff;
for (; i < l; i += 2) {
if (i != activeStop && stops[i] >= activeEnd) {
stops[i] = Math.max(stops[i] - diff, stops[activeStop + 1]);
stops[i + 1] = Math.max(stops[i + 1] - diff, stops[activeStop + 1]);
}
}
updateStops();
}
isDeleteForwards = false;
currentSelection = getSelection();
}
shouldOpen = isTyping;
});
addListener(textarea, "mousedown", () => {
if (stops) {
setTimeout(() => {
const [start, end] = getSelection();
if (stops && (start < stops[activeStop] || end > stops[activeStop + 1])) {
for (let i = 0, l = stops.length; i < stops.length; i += 2) {
if (start >= stops[i] && end <= stops[i + 1]) {
if (i + 3 > l) clearStops();
else activeStop = i;
break;
}
}
}
});
}
});
addListener(
textarea,
"beforeinput",
(e) => {
let inputType = e.inputType;
let isDelete = inputType[0] == "d";
let isInsert = inputType == "insertText";
let data = e.data;
if (isOpen && isInsert && !prevSelection && data && !data[1] && currentOptions[activeIndex][4].commitChars?.includes(data)) {
insertOption(activeIndex);
}
if (stops) {
currentSelection = getSelection();
isDeleteForwards = isDelete && inputType[13] == "F" && currentSelection[0] == currentSelection[1];
}
isTyping = !config.explicitOnly && (isTyping || isInsert && !prevSelection || isDelete && isOpen);
},
true
);
addListener(textarea, "blur", (e) => {
if (config.closeOnBlur != false && !tooltip.contains(e.relatedTarget)) hide();
});
addListener(
textarea,
"keydown",
(e) => {
let key = e.key;
let code = getModifierCode(e);
let top;
let height;
let newActive;
if (key == " " && code == 2) {
addSelectionHandler();
if (cursor) startQuery(true);
preventDefault(e);
} else if (isOpen) {
if (!code) {
if (/^Arrow[UD]/.test(key)) {
move(key[5] == "U");
preventDefault(e);
} else if (key == "Tab" || key == "Enter") {
insertOption(activeIndex);
preventDefault(e);
} else if (key == "Escape") {
hide();
preventDefault(e);
} else if (key.slice(0, 4) == "Page") {
setRowHeight();
top = tooltip.scrollTop;
height = tooltip.clientHeight;
if (key[4] == "U") {
newActive = Math.ceil(top / rowHeight);
activeIndex = activeIndex == newActive || newActive - 1 == activeIndex ? Math.ceil(Math.max(0, (top - height) / rowHeight + 1)) : newActive;
} else {
top += height + 1;
newActive = Math.ceil(top / rowHeight - 2);
activeIndex = activeIndex == newActive || newActive + 1 == activeIndex ? Math.ceil(Math.min(numOptions - 1, (top + height) / rowHeight - 3)) : newActive;
}
scrollActiveIntoView();
updateActive();
preventDefault(e);
}
}
} else if (stops) {
if (!(code & 7) && key == "Tab") {
if (!code) {
moveActiveStop(2);
if (activeStop + 3 > stops.length) clearStops();
preventDefault(e);
} else if (activeStop) {
moveActiveStop(-2);
preventDefault(e);
}
} else if (!code && key == "Escape") {
clearStops();
preventDefault(e);
}
}
},
true
);
addListener(list, "mousedown", (e) => {
insertOption([].indexOf.call(rows, e.target.closest("li")) + offset);
preventDefault(e);
});
addListener(tooltip, "focusout", (e) => {
if (config.closeOnBlur != false && e.relatedTarget != textarea) hide();
});
};
export {
autoComplete as a,
completionsFromRecords as b,
completeFromList as c,
attrSnippet as d,
findWords as f,
optionsFromKeys as o,
registerCompletions as r
};
//# sourceMappingURL=tooltip-DhaeURAZ.js.map