.
+ */
+ static async discoverEndpoints(siteUrl) {
+ const resp = await (0, import_obsidian2.requestUrl)({ url: siteUrl, method: "GET" });
+ const html = resp.text;
+ const authorizationEndpoint = _IndieAuth.extractLinkRel(html, "authorization_endpoint");
+ const tokenEndpoint = _IndieAuth.extractLinkRel(html, "token_endpoint");
+ const micropubEndpoint = _IndieAuth.extractLinkRel(html, "micropub");
+ if (!authorizationEndpoint) {
+ throw new Error(
+ `No found at ${siteUrl}. Make sure Indiekit is running and SITE_URL is set correctly.`
+ );
+ }
+ if (!tokenEndpoint) {
+ throw new Error(`No found at ${siteUrl}.`);
+ }
+ return { authorizationEndpoint, tokenEndpoint, micropubEndpoint };
+ }
+ /**
+ * Run the full IndieAuth PKCE sign-in flow.
+ *
+ * Opens the browser at the user's IndieAuth login page. After login the
+ * browser is redirected to the GitHub Pages callback, which triggers
+ * the obsidian://micropub-auth protocol, which resolves the Promise here.
+ *
+ * Requires handleProtocolCallback() to be wired up in main.ts via
+ * this.registerObsidianProtocolHandler("micropub-auth", handleProtocolCallback)
+ */
+ static async signIn(siteUrl) {
+ var _a, _b, _c, _d, _e, _f;
+ const { authorizationEndpoint, tokenEndpoint, micropubEndpoint } = await _IndieAuth.discoverEndpoints(siteUrl);
+ const state = _IndieAuth.base64url(crypto.randomBytes(16));
+ const codeVerifier = _IndieAuth.base64url(crypto.randomBytes(64));
+ const codeChallenge = _IndieAuth.base64url(
+ crypto.createHash("sha256").update(codeVerifier).digest()
+ );
+ const callbackPromise = new Promise(
+ (resolve, reject) => {
+ const timeout = setTimeout(() => {
+ pendingCallback = null;
+ reject(new Error("Sign-in timed out (5 min). Please try again."));
+ }, AUTH_TIMEOUT_MS);
+ pendingCallback = {
+ state,
+ resolve: (params) => {
+ clearTimeout(timeout);
+ resolve(params);
+ }
+ };
+ }
+ );
+ const authUrl = new URL(authorizationEndpoint);
+ authUrl.searchParams.set("response_type", "code");
+ authUrl.searchParams.set("client_id", CLIENT_ID);
+ authUrl.searchParams.set("redirect_uri", REDIRECT_URI);
+ authUrl.searchParams.set("state", state);
+ authUrl.searchParams.set("code_challenge", codeChallenge);
+ authUrl.searchParams.set("code_challenge_method", "S256");
+ authUrl.searchParams.set("scope", SCOPE);
+ authUrl.searchParams.set("me", siteUrl);
+ window.open(authUrl.toString());
+ const callbackParams = await callbackPromise;
+ if (callbackParams.state !== state) {
+ throw new Error("State mismatch \u2014 possible CSRF attack. Please try again.");
+ }
+ const code = callbackParams.code;
+ if (!code) {
+ throw new Error(
+ (_b = (_a = callbackParams.error_description) != null ? _a : callbackParams.error) != null ? _b : "No authorization code received."
+ );
+ }
+ const tokenResp = await (0, import_obsidian2.requestUrl)({
+ url: tokenEndpoint,
+ method: "POST",
+ headers: {
+ "Content-Type": "application/x-www-form-urlencoded",
+ Accept: "application/json"
+ },
+ body: new URLSearchParams({
+ grant_type: "authorization_code",
+ code,
+ client_id: CLIENT_ID,
+ redirect_uri: REDIRECT_URI,
+ code_verifier: codeVerifier
+ }).toString(),
+ throw: false
+ });
+ const data = tokenResp.json;
+ if (!data.access_token) {
+ throw new Error(
+ (_d = (_c = data.error_description) != null ? _c : data.error) != null ? _d : `Token exchange failed (HTTP ${tokenResp.status})`
+ );
+ }
+ return {
+ accessToken: data.access_token,
+ scope: (_e = data.scope) != null ? _e : SCOPE,
+ me: (_f = data.me) != null ? _f : siteUrl,
+ authorizationEndpoint,
+ tokenEndpoint,
+ micropubEndpoint
+ };
+ }
+ // ── Helpers ───────────────────────────────────────────────────────────────
+ static base64url(buf) {
+ return buf.toString("base64").replace(/\+/g, "-").replace(/\//g, "_").replace(/=/g, "");
+ }
+ static extractLinkRel(html, rel) {
+ var _a;
+ const re = new RegExp(
+ `]+rel=["'][^"']*\\b${rel}\\b[^"']*["'][^>]+href=["']([^"']+)["']|]+href=["']([^"']+)["'][^>]+rel=["'][^"']*\\b${rel}\\b[^"']*["']`,
+ "i"
+ );
+ const m = html.match(re);
+ return (_a = m == null ? void 0 : m[1]) != null ? _a : m == null ? void 0 : m[2];
+ }
+};
+
+// src/SettingsTab.ts
+var MicropubSettingsTab = class extends import_obsidian3.PluginSettingTab {
+ constructor(app, plugin) {
+ super(app, plugin);
+ this.plugin = plugin;
+ }
+ display() {
+ const { containerEl } = this;
+ containerEl.empty();
+ containerEl.createEl("h2", { text: "Micropub Publisher" });
+ containerEl.createEl("h3", { text: "Account" });
+ if (this.plugin.settings.me && this.plugin.settings.accessToken) {
+ this.renderSignedIn(containerEl);
+ } else {
+ this.renderSignedOut(containerEl);
+ }
+ containerEl.createEl("h3", { text: "Endpoints" });
+ containerEl.createEl("p", {
+ text: "These are filled automatically when you sign in. Only edit them manually if your server uses non-standard paths.",
+ cls: "setting-item-description"
+ });
+ new import_obsidian3.Setting(containerEl).setName("Micropub endpoint").setDesc("e.g. https://blog.giersig.eu/micropub").addText(
+ (text) => text.setPlaceholder("https://example.com/micropub").setValue(this.plugin.settings.micropubEndpoint).onChange(async (value) => {
+ this.plugin.settings.micropubEndpoint = value.trim();
+ await this.plugin.saveSettings();
+ })
+ );
+ new import_obsidian3.Setting(containerEl).setName("Media endpoint").setDesc("For image uploads. Auto-discovered if blank.").addText(
+ (text) => text.setPlaceholder("https://example.com/micropub/media").setValue(this.plugin.settings.mediaEndpoint).onChange(async (value) => {
+ this.plugin.settings.mediaEndpoint = value.trim();
+ await this.plugin.saveSettings();
+ })
+ );
+ containerEl.createEl("h3", { text: "Publish Behaviour" });
+ new import_obsidian3.Setting(containerEl).setName("Default visibility").setDesc("Applies when the note has no explicit visibility property.").addDropdown(
+ (drop) => drop.addOption("public", "Public").addOption("unlisted", "Unlisted").addOption("private", "Private").setValue(this.plugin.settings.defaultVisibility).onChange(async (value) => {
+ this.plugin.settings.defaultVisibility = value;
+ await this.plugin.saveSettings();
+ })
+ );
+ new import_obsidian3.Setting(containerEl).setName("Write URL back to note").setDesc(
+ "After publishing, store the post URL as `mp-url` in frontmatter. Subsequent publishes will update the existing post instead of creating a new one."
+ ).addToggle(
+ (toggle) => toggle.setValue(this.plugin.settings.writeUrlToFrontmatter).onChange(async (value) => {
+ this.plugin.settings.writeUrlToFrontmatter = value;
+ await this.plugin.saveSettings();
+ })
+ );
+ containerEl.createEl("h3", { text: "Digital Garden" });
+ new import_obsidian3.Setting(containerEl).setName("Map #garden/* tags to gardenStage").setDesc(
+ "Obsidian tags like #garden/plant become a `garden-stage: plant` Micropub property. The blog renders these as growth stage badges at /garden/."
+ ).addToggle(
+ (toggle) => toggle.setValue(this.plugin.settings.mapGardenTags).onChange(async (value) => {
+ this.plugin.settings.mapGardenTags = value;
+ await this.plugin.saveSettings();
+ })
+ );
+ containerEl.createEl("p", {
+ text: "Stages: plant \u{1F331} \xB7 cultivate \u{1F33F} \xB7 question \u2753 \xB7 repot \u{1FAB4} \xB7 revitalize \u2728 \xB7 revisit \u{1F504}",
+ cls: "setting-item-description"
+ });
+ }
+ // ── Signed-out state ─────────────────────────────────────────────────────
+ renderSignedOut(containerEl) {
+ new import_obsidian3.Setting(containerEl).setName("Site URL").setDesc(
+ "Your site's home page. Clicking Sign in opens your blog's login page in the browser \u2014 the same flow iA Writer uses."
+ ).addText(
+ (text) => text.setPlaceholder("https://blog.giersig.eu").setValue(this.plugin.settings.siteUrl).onChange(async (value) => {
+ this.plugin.settings.siteUrl = value.trim();
+ await this.plugin.saveSettings();
+ })
+ ).addButton((btn) => {
+ btn.setButtonText("Sign in").setCta().onClick(async () => {
+ const siteUrl = this.plugin.settings.siteUrl.trim();
+ if (!siteUrl) {
+ new import_obsidian3.Notice("Enter your site URL first.");
+ return;
+ }
+ btn.setDisabled(true);
+ btn.setButtonText("Opening browser\u2026");
+ try {
+ const result = await IndieAuth.signIn(siteUrl);
+ this.plugin.settings.accessToken = result.accessToken;
+ this.plugin.settings.me = result.me;
+ this.plugin.settings.authorizationEndpoint = result.authorizationEndpoint;
+ this.plugin.settings.tokenEndpoint = result.tokenEndpoint;
+ if (result.micropubEndpoint) {
+ this.plugin.settings.micropubEndpoint = result.micropubEndpoint;
+ }
+ if (result.mediaEndpoint) {
+ this.plugin.settings.mediaEndpoint = result.mediaEndpoint;
+ }
+ await this.plugin.saveSettings();
+ if (!this.plugin.settings.mediaEndpoint) {
+ try {
+ const client = new MicropubClient(
+ () => this.plugin.settings.micropubEndpoint,
+ () => this.plugin.settings.mediaEndpoint,
+ () => this.plugin.settings.accessToken
+ );
+ const cfg = await client.fetchConfig();
+ if (cfg["media-endpoint"]) {
+ this.plugin.settings.mediaEndpoint = cfg["media-endpoint"];
+ await this.plugin.saveSettings();
+ }
+ } catch (e) {
+ }
+ }
+ new import_obsidian3.Notice(`\u2705 Signed in as ${result.me}`);
+ this.display();
+ } catch (err) {
+ new import_obsidian3.Notice(`Sign-in failed: ${String(err)}`, 8e3);
+ btn.setDisabled(false);
+ btn.setButtonText("Sign in");
+ }
+ });
+ });
+ const details = containerEl.createEl("details");
+ details.createEl("summary", {
+ text: "Or paste a token manually",
+ cls: "setting-item-description"
+ });
+ details.style.marginTop = "8px";
+ details.style.marginBottom = "8px";
+ new import_obsidian3.Setting(details).setName("Access token").setDesc("Bearer token from your Indiekit admin panel.").addText((text) => {
+ text.setPlaceholder("your-bearer-token").setValue(this.plugin.settings.accessToken).onChange(async (value) => {
+ this.plugin.settings.accessToken = value.trim();
+ await this.plugin.saveSettings();
+ });
+ text.inputEl.type = "password";
+ }).addButton(
+ (btn) => btn.setButtonText("Verify").onClick(async () => {
+ if (!this.plugin.settings.micropubEndpoint || !this.plugin.settings.accessToken) {
+ new import_obsidian3.Notice("Set the Micropub endpoint and token first.");
+ return;
+ }
+ btn.setDisabled(true);
+ try {
+ const client = new MicropubClient(
+ () => this.plugin.settings.micropubEndpoint,
+ () => this.plugin.settings.mediaEndpoint,
+ () => this.plugin.settings.accessToken
+ );
+ await client.fetchConfig();
+ new import_obsidian3.Notice("\u2705 Token is valid!");
+ } catch (err) {
+ new import_obsidian3.Notice(`Token check failed: ${String(err)}`);
+ } finally {
+ btn.setDisabled(false);
+ }
+ })
+ );
+ }
+ // ── Signed-in state ──────────────────────────────────────────────────────
+ renderSignedIn(containerEl) {
+ const me = this.plugin.settings.me;
+ const banner = containerEl.createDiv({
+ cls: "micropub-auth-banner"
+ });
+ banner.style.cssText = "display:flex;align-items:center;gap:12px;padding:12px 16px;border:1px solid var(--background-modifier-border);border-radius:8px;margin-bottom:16px;background:var(--background-secondary);";
+ const icon = banner.createDiv();
+ icon.style.cssText = "width:40px;height:40px;border-radius:50%;background:var(--interactive-accent);display:flex;align-items:center;justify-content:center;font-size:1.2rem;flex-shrink:0;";
+ icon.textContent = "\u{1F310}";
+ const info = banner.createDiv();
+ info.createEl("div", {
+ text: "Signed in",
+ attr: { style: "font-size:.75rem;color:var(--text-muted);margin-bottom:2px" }
+ });
+ info.createEl("div", {
+ text: me,
+ attr: { style: "font-weight:500;word-break:break-all" }
+ });
+ new import_obsidian3.Setting(containerEl).setName("Site URL").addText(
+ (text) => text.setValue(this.plugin.settings.siteUrl).setDisabled(true)
+ ).addButton(
+ (btn) => btn.setButtonText("Sign out").setWarning().onClick(async () => {
+ this.plugin.settings.accessToken = "";
+ this.plugin.settings.me = "";
+ this.plugin.settings.authorizationEndpoint = "";
+ this.plugin.settings.tokenEndpoint = "";
+ await this.plugin.saveSettings();
+ this.display();
+ })
+ );
+ }
+};
+
+// src/Publisher.ts
+var import_obsidian4 = require("obsidian");
+var GARDEN_TAG_PREFIX = "garden/";
+var Publisher = class {
+ constructor(app, settings) {
+ this.app = app;
+ this.settings = settings;
+ this.client = new MicropubClient(
+ () => settings.micropubEndpoint,
+ () => settings.mediaEndpoint,
+ () => settings.accessToken
+ );
+ }
+ /** Publish the given file. Returns PublishResult. */
+ async publish(file) {
+ var _a, _b;
+ const raw = await this.app.vault.read(file);
+ const { frontmatter, body } = this.parseFrontmatter(raw);
+ const existingUrl = (_b = (_a = frontmatter["mp-url"]) != null ? _a : frontmatter["url"]) != null ? _b : void 0;
+ const { content: processedBody, uploadedUrls } = await this.processImages(body);
+ const properties = this.buildProperties(frontmatter, processedBody, uploadedUrls);
+ let result;
+ if (existingUrl) {
+ const replace = {};
+ for (const [k, v] of Object.entries(properties)) {
+ replace[k] = Array.isArray(v) ? v : [v];
+ }
+ result = await this.client.updatePost(existingUrl, replace);
+ } else {
+ result = await this.client.createPost(properties);
+ }
+ if (result.success && result.url && this.settings.writeUrlToFrontmatter) {
+ await this.writeUrlToNote(file, raw, result.url);
+ }
+ return result;
+ }
+ // ── Property builder ─────────────────────────────────────────────────────
+ buildProperties(fm, body, uploadedUrls) {
+ var _a, _b, _c, _d, _e, _f, _g, _h, _i;
+ const props = {};
+ const trimmedBody = body.trim();
+ const bookmarkOf = (_a = fm["bookmarkOf"]) != null ? _a : fm["bookmark-of"];
+ const likeOf = (_b = fm["likeOf"]) != null ? _b : fm["like-of"];
+ const inReplyTo = (_c = fm["inReplyTo"]) != null ? _c : fm["in-reply-to"];
+ const repostOf = (_d = fm["repostOf"]) != null ? _d : fm["repost-of"];
+ if (bookmarkOf) props["bookmark-of"] = [String(bookmarkOf)];
+ if (likeOf) props["like-of"] = [String(likeOf)];
+ if (inReplyTo) props["in-reply-to"] = [String(inReplyTo)];
+ if (repostOf) props["repost-of"] = [String(repostOf)];
+ const isInteractionWithoutBody = (likeOf || repostOf) && !trimmedBody;
+ if (!isInteractionWithoutBody) {
+ props["content"] = trimmedBody ? [{ html: trimmedBody }] : [{ html: "" }];
+ }
+ if (fm["title"]) {
+ props["name"] = [String(fm["title"])];
+ }
+ if ((_e = fm["summary"]) != null ? _e : fm["excerpt"]) {
+ props["summary"] = [String((_f = fm["summary"]) != null ? _f : fm["excerpt"])];
+ }
+ if (fm["date"]) {
+ props["published"] = [new Date(String(fm["date"])).toISOString()];
+ }
+ const rawTags = this.resolveArray((_g = fm["tags"]) != null ? _g : fm["category"]);
+ const gardenStage = this.extractGardenStage(rawTags);
+ const normalTags = rawTags.filter(
+ (t) => !t.startsWith(GARDEN_TAG_PREFIX) && t !== "garden"
+ );
+ if (normalTags.length > 0) {
+ props["category"] = normalTags;
+ }
+ if (this.settings.mapGardenTags && gardenStage) {
+ props["garden-stage"] = [gardenStage];
+ }
+ const syndicateTo = this.resolveArray(
+ (_h = fm["mp-syndicate-to"]) != null ? _h : fm["mpSyndicateTo"]
+ );
+ const allSyndicateTo = [
+ .../* @__PURE__ */ new Set([...this.settings.defaultSyndicateTo, ...syndicateTo])
+ ];
+ if (allSyndicateTo.length > 0) {
+ props["mp-syndicate-to"] = allSyndicateTo;
+ }
+ const visibility = (_i = fm["visibility"]) != null ? _i : this.settings.defaultVisibility;
+ if (visibility && visibility !== "public") {
+ props["visibility"] = [visibility];
+ }
+ if (fm["ai"] && typeof fm["ai"] === "object") {
+ props["ai"] = [fm["ai"]];
+ }
+ const fmPhotos = this.resolvePhotoArray(fm["photo"]);
+ if (fmPhotos.length > 0) {
+ props["photo"] = fmPhotos;
+ } else if (uploadedUrls.length > 0) {
+ props["photo"] = uploadedUrls.map((url) => ({ value: url }));
+ }
+ for (const [k, v] of Object.entries(fm)) {
+ if (k.startsWith("mp-") && k !== "mp-url" && k !== "mp-syndicate-to") {
+ props[k] = this.resolveArray(v);
+ }
+ }
+ return props;
+ }
+ /**
+ * Normalise the `photo` frontmatter field into Micropub photo objects.
+ * Handles three formats:
+ * - string URL: "https://..."
+ * - array of strings: ["https://..."]
+ * - array of objects: [{url: "https://...", alt: "..."}]
+ */
+ resolvePhotoArray(value) {
+ if (!value) return [];
+ const items = Array.isArray(value) ? value : [value];
+ return items.map((item) => {
+ var _a, _b;
+ if (typeof item === "string") return { value: item };
+ if (typeof item === "object" && item !== null) {
+ const obj = item;
+ const url = String((_b = (_a = obj["url"]) != null ? _a : obj["value"]) != null ? _b : "");
+ if (!url) return null;
+ return obj["alt"] ? { value: url, alt: String(obj["alt"]) } : { value: url };
+ }
+ return null;
+ }).filter((x) => x !== null);
+ }
+ // ── Garden tag extraction ────────────────────────────────────────────────
+ /**
+ * Find the first #garden/ tag and return the stage name.
+ * Supports both "garden/plant" (Obsidian array) and "#garden/plant" (inline).
+ */
+ extractGardenStage(tags) {
+ for (const tag of tags) {
+ const clean = tag.replace(/^#/, "");
+ if (clean.startsWith(GARDEN_TAG_PREFIX)) {
+ const stage = clean.slice(GARDEN_TAG_PREFIX.length);
+ const valid = [
+ "plant",
+ "cultivate",
+ "question",
+ "repot",
+ "revitalize",
+ "revisit"
+ ];
+ if (valid.includes(stage)) return stage;
+ }
+ }
+ return void 0;
+ }
+ // ── Image processing ─────────────────────────────────────────────────────
+ /**
+ * Find all `![[local-image.png]]` or `` in the body,
+ * upload them to the media endpoint, and replace the references with remote URLs.
+ */
+ async processImages(body) {
+ const uploadedUrls = [];
+ const wikiPattern = /!\[\[([^\]]+\.(png|jpg|jpeg|gif|webp|svg))\]\]/gi;
+ const mdPattern = /!\[([^\]]*)\]\(([^)]+\.(png|jpg|jpeg|gif|webp|svg))\)/gi;
+ let content = body;
+ const wikiMatches = [...body.matchAll(wikiPattern)];
+ for (const match of wikiMatches) {
+ const filename = match[1];
+ try {
+ const remoteUrl = await this.uploadLocalFile(filename);
+ if (remoteUrl) {
+ uploadedUrls.push(remoteUrl);
+ content = content.replace(match[0], ``);
+ }
+ } catch (err) {
+ console.warn(`[micropub] Failed to upload ${filename}:`, err);
+ }
+ }
+ const mdMatches = [...content.matchAll(mdPattern)];
+ for (const match of mdMatches) {
+ const alt = match[1];
+ const path = match[2];
+ if (path.startsWith("http")) continue;
+ try {
+ const remoteUrl = await this.uploadLocalFile(path);
+ if (remoteUrl) {
+ uploadedUrls.push(remoteUrl);
+ content = content.replace(match[0], ``);
+ }
+ } catch (err) {
+ console.warn(`[micropub] Failed to upload ${path}:`, err);
+ }
+ }
+ return { content, uploadedUrls };
+ }
+ async uploadLocalFile(path) {
+ const file = this.app.vault.getFiles().find(
+ (f) => f.name === path || f.path === path
+ );
+ if (!file) return void 0;
+ const buffer = await this.app.vault.readBinary(file);
+ const mimeType = this.guessMimeType(file.extension);
+ return this.client.uploadMedia(buffer, file.name, mimeType);
+ }
+ // ── Frontmatter helpers ──────────────────────────────────────────────────
+ parseFrontmatter(raw) {
+ var _a;
+ const fmMatch = raw.match(/^---\r?\n([\s\S]*?)\r?\n---\r?\n([\s\S]*)$/);
+ if (!fmMatch) return { frontmatter: {}, body: raw };
+ let frontmatter = {};
+ try {
+ frontmatter = (_a = (0, import_obsidian4.parseYaml)(fmMatch[1])) != null ? _a : {};
+ } catch (e) {
+ }
+ return { frontmatter, body: fmMatch[2] };
+ }
+ async writeUrlToNote(file, originalContent, url) {
+ const fmMatch = originalContent.match(
+ /^(---\r?\n[\s\S]*?\r?\n---\r?\n)([\s\S]*)$/
+ );
+ if (!fmMatch) {
+ const newFm = `---
+mp-url: "${url}"
+---
+`;
+ await this.app.vault.modify(file, newFm + originalContent);
+ return;
+ }
+ const fmBlock = fmMatch[1];
+ const body = fmMatch[2];
+ if (fmBlock.includes("mp-url:")) {
+ const updated = fmBlock.replace(
+ /mp-url:.*(\r?\n)/,
+ `mp-url: "${url}"$1`
+ );
+ await this.app.vault.modify(file, updated + body);
+ } else {
+ const updated = fmBlock.replace(
+ /(\r?\n---\r?\n)$/,
+ `
+mp-url: "${url}"$1`
+ );
+ await this.app.vault.modify(file, updated + body);
+ }
+ }
+ resolveArray(value) {
+ if (!value) return [];
+ if (Array.isArray(value)) return value.map(String);
+ return [String(value)];
+ }
+ guessMimeType(ext) {
+ var _a;
+ const map = {
+ png: "image/png",
+ jpg: "image/jpeg",
+ jpeg: "image/jpeg",
+ gif: "image/gif",
+ webp: "image/webp",
+ svg: "image/svg+xml"
+ };
+ return (_a = map[ext.toLowerCase()]) != null ? _a : "application/octet-stream";
+ }
+};
+
+// src/main.ts
+var MicropubPlugin = class extends import_obsidian5.Plugin {
+ async onload() {
+ await this.loadSettings();
+ this.addCommand({
+ id: "publish-to-micropub",
+ name: "Publish to Micropub",
+ checkCallback: (checking) => {
+ const file = this.app.workspace.getActiveFile();
+ if (!file || file.extension !== "md") return false;
+ if (checking) return true;
+ this.publishActiveNote(file);
+ return true;
+ }
+ });
+ this.addCommand({
+ id: "publish-to-micropub-update",
+ name: "Update existing Micropub post",
+ checkCallback: (checking) => {
+ const file = this.app.workspace.getActiveFile();
+ if (!file || file.extension !== "md") return false;
+ if (checking) return true;
+ this.publishActiveNote(file);
+ return true;
+ }
+ });
+ this.registerObsidianProtocolHandler("micropub-auth", (params) => {
+ handleProtocolCallback(params);
+ });
+ this.addSettingTab(new MicropubSettingsTab(this.app, this));
+ this.addRibbonIcon("send", "Publish to Micropub", () => {
+ const file = this.app.workspace.getActiveFile();
+ if (!file || file.extension !== "md") {
+ new import_obsidian5.Notice("Open a Markdown note to publish.");
+ return;
+ }
+ this.publishActiveNote(file);
+ });
+ }
+ onunload() {
+ }
+ // ── Publish flow ──────────────────────────────────────────────────────────
+ async publishActiveNote(file) {
+ if (!this.settings.micropubEndpoint) {
+ new import_obsidian5.Notice(
+ "\u26A0\uFE0F Micropub endpoint not configured. Open plugin settings to add it."
+ );
+ return;
+ }
+ if (!this.settings.accessToken) {
+ new import_obsidian5.Notice(
+ "\u26A0\uFE0F Access token not configured. Open plugin settings to add it."
+ );
+ return;
+ }
+ const notice = new import_obsidian5.Notice(
+ "Publishing\u2026",
+ 0
+ /* persist until dismissed */
+ );
+ try {
+ const publisher = new Publisher(this.app, this.settings);
+ const result = await publisher.publish(file);
+ notice.hide();
+ if (result.success) {
+ const urlDisplay = result.url ? `
+${result.url}` : "";
+ new import_obsidian5.Notice(`\u2705 Published!${urlDisplay}`, 8e3);
+ } else {
+ new import_obsidian5.Notice(`\u274C Publish failed: ${result.error}`, 1e4);
+ console.error("[micropub] Publish failed:", result.error);
+ }
+ } catch (err) {
+ notice.hide();
+ const msg = err instanceof Error ? err.message : String(err);
+ new import_obsidian5.Notice(`\u274C Error: ${msg}`, 1e4);
+ console.error("[micropub] Unexpected error:", err);
+ }
+ }
+ // ── Settings persistence ──────────────────────────────────────────────────
+ async loadSettings() {
+ this.settings = Object.assign(
+ {},
+ DEFAULT_SETTINGS,
+ await this.loadData()
+ );
+ }
+ async saveSettings() {
+ await this.saveData(this.settings);
+ }
+};
+//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["src/main.ts", "src/types.ts", "src/SettingsTab.ts", "src/MicropubClient.ts", "src/IndieAuth.ts", "src/Publisher.ts"],
  "sourcesContent": ["/**\n * main.ts \u2014 obsidian-micropub plugin entry point\n *\n * Publishes the active note to any Micropub-compatible endpoint.\n * Designed to work with Indiekit (https://getindiekit.com) but compatible\n * with any server that implements the Micropub spec (W3C).\n *\n * Key features vs. the original obsidian-microblog:\n *   - Configurable endpoint URL (not hardcoded to Micro.blog)\n *   - Auto-discovery of micropub/media endpoints from <link rel> headers\n *   - #garden/* tag \u2192 gardenStage property mapping for Digital Garden\n *   - Writes returned post URL back to note frontmatter for future updates\n *   - Supports create + update flows\n *\n * Based on: https://github.com/svemagie/obsidian-microblog (MIT)\n */\n\nimport { Notice, Plugin, TFile } from \"obsidian\";\nimport { DEFAULT_SETTINGS, type MicropubSettings } from \"./types\";\nimport { MicropubSettingsTab } from \"./SettingsTab\";\nimport { Publisher } from \"./Publisher\";\nimport { handleProtocolCallback } from \"./IndieAuth\";\n\nexport default class MicropubPlugin extends Plugin {\n  settings!: MicropubSettings;\n\n  async onload(): Promise<void> {\n    await this.loadSettings();\n\n    // \u2500\u2500 Commands \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\n\n    this.addCommand({\n      id: \"publish-to-micropub\",\n      name: \"Publish to Micropub\",\n      checkCallback: (checking: boolean) => {\n        const file = this.app.workspace.getActiveFile();\n        if (!file || file.extension !== \"md\") return false;\n        if (checking) return true;\n\n        this.publishActiveNote(file);\n        return true;\n      },\n    });\n\n    this.addCommand({\n      id: \"publish-to-micropub-update\",\n      name: \"Update existing Micropub post\",\n      checkCallback: (checking: boolean) => {\n        const file = this.app.workspace.getActiveFile();\n        if (!file || file.extension !== \"md\") return false;\n        if (checking) return true;\n\n        // Update uses the same publish flow \u2014 Publisher detects mp-url and routes to update\n        this.publishActiveNote(file);\n        return true;\n      },\n    });\n\n    // \u2500\u2500 IndieAuth protocol handler \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\n    // Receives obsidian://micropub-auth?code=...&state=... after the user\n    // approves on their IndieAuth login page. The GitHub Pages callback page\n    // at svemagie.github.io/obsidian-micropub/callback redirects here.\n    this.registerObsidianProtocolHandler(\"micropub-auth\", (params) => {\n      handleProtocolCallback(params as Record<string, string>);\n    });\n\n    // \u2500\u2500 Settings tab \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\n\n    this.addSettingTab(new MicropubSettingsTab(this.app, this));\n\n    // \u2500\u2500 Ribbon icon \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\n\n    this.addRibbonIcon(\"send\", \"Publish to Micropub\", () => {\n      const file = this.app.workspace.getActiveFile();\n      if (!file || file.extension !== \"md\") {\n        new Notice(\"Open a Markdown note to publish.\");\n        return;\n      }\n      this.publishActiveNote(file);\n    });\n  }\n\n  onunload(): void {\n    // Nothing to clean up\n  }\n\n  // \u2500\u2500 Publish flow \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\n\n  private async publishActiveNote(file: TFile): Promise<void> {\n    if (!this.settings.micropubEndpoint) {\n      new Notice(\n        \"\u26A0\uFE0F Micropub endpoint not configured. Open plugin settings to add it.\",\n      );\n      return;\n    }\n\n    if (!this.settings.accessToken) {\n      new Notice(\n        \"\u26A0\uFE0F Access token not configured. Open plugin settings to add it.\",\n      );\n      return;\n    }\n\n    const notice = new Notice(\"Publishing\u2026\", 0 /* persist until dismissed */);\n\n    try {\n      const publisher = new Publisher(this.app, this.settings);\n      const result = await publisher.publish(file);\n\n      notice.hide();\n\n      if (result.success) {\n        const urlDisplay = result.url\n          ? `\\n${result.url}`\n          : \"\";\n        new Notice(`\u2705 Published!${urlDisplay}`, 8000);\n      } else {\n        new Notice(`\u274C Publish failed: ${result.error}`, 10000);\n        console.error(\"[micropub] Publish failed:\", result.error);\n      }\n    } catch (err: unknown) {\n      notice.hide();\n      const msg = err instanceof Error ? err.message : String(err);\n      new Notice(`\u274C Error: ${msg}`, 10000);\n      console.error(\"[micropub] Unexpected error:\", err);\n    }\n  }\n\n  // \u2500\u2500 Settings persistence \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\n\n  async loadSettings(): Promise<void> {\n    this.settings = Object.assign(\n      {},\n      DEFAULT_SETTINGS,\n      await this.loadData(),\n    ) as MicropubSettings;\n  }\n\n  async saveSettings(): Promise<void> {\n    await this.saveData(this.settings);\n  }\n}\n", "/**\n * types.ts \u2014 shared interfaces for obsidian-micropub\n */\n\n/** Plugin settings stored in data.json */\nexport interface MicropubSettings {\n  /** Full URL of the Micropub endpoint, e.g. https://example.com/micropub */\n  micropubEndpoint: string;\n\n  /**\n   * Full URL of the media endpoint for image uploads.\n   * If empty, discovered automatically from the Micropub config query,\n   * or derived from the micropubEndpoint (some servers use /micropub/media).\n   */\n  mediaEndpoint: string;\n\n  /**\n   * Bearer token for Authorization: Bearer <token>.\n   * Obtain from your IndieAuth token endpoint or server admin panel.\n   */\n  accessToken: string;\n\n  /**\n   * The syndication targets to pre-tick in the publish dialog.\n   * Values are uid strings returned by the Micropub config ?q=config.\n   */\n  defaultSyndicateTo: string[];\n\n  /**\n   * When true, perform a discovery fetch against the site URL to auto-detect\n   * the micropub and token endpoints from <link rel=\"micropub\"> headers.\n   */\n  autoDiscover: boolean;\n\n  /** Your site's homepage URL \u2014 used for endpoint discovery and IndieAuth. */\n  siteUrl: string;\n\n  /**\n   * The authorization_endpoint discovered from the site.\n   * Populated automatically by the IndieAuth sign-in flow.\n   */\n  authorizationEndpoint: string;\n\n  /**\n   * The token_endpoint discovered from the site.\n   * Populated automatically by the IndieAuth sign-in flow.\n   */\n  tokenEndpoint: string;\n\n  /**\n   * The canonical \"me\" URL returned by the token endpoint after sign-in.\n   * Used to show who is currently logged in.\n   */\n  me: string;\n\n  /**\n   * When true, after a successful publish the post URL returned by the server\n   * is written back to the note's frontmatter as `mp-url`.\n   */\n  writeUrlToFrontmatter: boolean;\n\n  /**\n   * Map Obsidian #garden/* tags to a `gardenStage` Micropub property.\n   * When enabled, a tag like #garden/plant becomes { \"garden-stage\": \"plant\" }\n   * in the Micropub request (and gardenStage: plant in the server's front matter).\n   */\n  mapGardenTags: boolean;\n\n  /** Visibility default for new posts: \"public\" | \"unlisted\" | \"private\" */\n  defaultVisibility: \"public\" | \"unlisted\" | \"private\";\n}\n\nexport const DEFAULT_SETTINGS: MicropubSettings = {\n  micropubEndpoint: \"\",\n  mediaEndpoint: \"\",\n  accessToken: \"\",\n  defaultSyndicateTo: [],\n  autoDiscover: false,\n  siteUrl: \"\",\n  authorizationEndpoint: \"\",\n  tokenEndpoint: \"\",\n  me: \"\",\n  writeUrlToFrontmatter: true,\n  mapGardenTags: true,\n  defaultVisibility: \"public\",\n};\n\n/** A syndication target as returned by Micropub config query */\nexport interface SyndicationTarget {\n  uid: string;\n  name: string;\n}\n\n/** Micropub config response (?q=config) */\nexport interface MicropubConfig {\n  \"media-endpoint\"?: string;\n  \"syndicate-to\"?: SyndicationTarget[];\n  \"post-types\"?: Array<{ type: string; name: string }>;\n}\n\n/**\n * Garden stages \u2014 matches Obsidian #garden/* tags and blog gardenStage values.\n * The Micropub property name is \"garden-stage\" (hyphenated, Micropub convention).\n */\nexport type GardenStage =\n  | \"plant\"\n  | \"cultivate\"\n  | \"question\"\n  | \"repot\"\n  | \"revitalize\"\n  | \"revisit\";\n\nexport const GARDEN_STAGE_LABELS: Record<GardenStage, string> = {\n  plant:      \"\uD83C\uDF31 Seedling\",\n  cultivate:  \"\uD83C\uDF3F Growing\",\n  question:   \"\u2753 Open Question\",\n  repot:      \"\uD83E\uDEB4 Repotting\",\n  revitalize: \"\u2728 Revitalizing\",\n  revisit:    \"\uD83D\uDD04 Revisit\",\n};\n\n/** Result returned by Publisher.publish() */\nexport interface PublishResult {\n  success: boolean;\n  /** URL of the published post (from Location response header) */\n  url?: string;\n  error?: string;\n}\n", "/**\n * SettingsTab.ts \u2014 Obsidian settings UI for obsidian-micropub\n *\n * Authentication section works like iA Writer:\n *   1. User enters their site URL\n *   2. Clicks \"Sign in\" \u2014 browser opens at their IndieAuth login page\n *   3. They log in with their blog password\n *   4. Browser redirects back; plugin receives the token automatically\n *   5. Settings show \"Signed in as <me>\" + a Sign Out button\n *\n * Advanced users can still paste a token manually if they prefer.\n */\n\nimport { App, Notice, PluginSettingTab, Setting } from \"obsidian\";\nimport type MicropubPlugin from \"./main\";\nimport { MicropubClient } from \"./MicropubClient\";\nimport { IndieAuth } from \"./IndieAuth\";\n\nexport class MicropubSettingsTab extends PluginSettingTab {\n  constructor(\n    app: App,\n    private readonly plugin: MicropubPlugin,\n  ) {\n    super(app, plugin);\n  }\n\n  display(): void {\n    const { containerEl } = this;\n    containerEl.empty();\n\n    containerEl.createEl(\"h2\", { text: \"Micropub Publisher\" });\n\n    // \u2500\u2500 Site URL + Sign In \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\n    containerEl.createEl(\"h3\", { text: \"Account\" });\n\n    // Show current sign-in status\n    if (this.plugin.settings.me && this.plugin.settings.accessToken) {\n      this.renderSignedIn(containerEl);\n    } else {\n      this.renderSignedOut(containerEl);\n    }\n\n    // \u2500\u2500 Endpoints (collapsed / advanced) \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\n    containerEl.createEl(\"h3\", { text: \"Endpoints\" });\n\n    containerEl.createEl(\"p\", {\n      text: \"These are filled automatically when you sign in. Only edit them manually if your server uses non-standard paths.\",\n      cls: \"setting-item-description\",\n    });\n\n    new Setting(containerEl)\n      .setName(\"Micropub endpoint\")\n      .setDesc(\"e.g. https://blog.giersig.eu/micropub\")\n      .addText((text) =>\n        text\n          .setPlaceholder(\"https://example.com/micropub\")\n          .setValue(this.plugin.settings.micropubEndpoint)\n          .onChange(async (value) => {\n            this.plugin.settings.micropubEndpoint = value.trim();\n            await this.plugin.saveSettings();\n          }),\n      );\n\n    new Setting(containerEl)\n      .setName(\"Media endpoint\")\n      .setDesc(\"For image uploads. Auto-discovered if blank.\")\n      .addText((text) =>\n        text\n          .setPlaceholder(\"https://example.com/micropub/media\")\n          .setValue(this.plugin.settings.mediaEndpoint)\n          .onChange(async (value) => {\n            this.plugin.settings.mediaEndpoint = value.trim();\n            await this.plugin.saveSettings();\n          }),\n      );\n\n    // \u2500\u2500 Publish behaviour \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\n    containerEl.createEl(\"h3\", { text: \"Publish Behaviour\" });\n\n    new Setting(containerEl)\n      .setName(\"Default visibility\")\n      .setDesc(\"Applies when the note has no explicit visibility property.\")\n      .addDropdown((drop) =>\n        drop\n          .addOption(\"public\", \"Public\")\n          .addOption(\"unlisted\", \"Unlisted\")\n          .addOption(\"private\", \"Private\")\n          .setValue(this.plugin.settings.defaultVisibility)\n          .onChange(async (value) => {\n            this.plugin.settings.defaultVisibility = value as\n              | \"public\"\n              | \"unlisted\"\n              | \"private\";\n            await this.plugin.saveSettings();\n          }),\n      );\n\n    new Setting(containerEl)\n      .setName(\"Write URL back to note\")\n      .setDesc(\n        \"After publishing, store the post URL as `mp-url` in frontmatter. \" +\n        \"Subsequent publishes will update the existing post instead of creating a new one.\",\n      )\n      .addToggle((toggle) =>\n        toggle\n          .setValue(this.plugin.settings.writeUrlToFrontmatter)\n          .onChange(async (value) => {\n            this.plugin.settings.writeUrlToFrontmatter = value;\n            await this.plugin.saveSettings();\n          }),\n      );\n\n    // \u2500\u2500 Digital Garden \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\n    containerEl.createEl(\"h3\", { text: \"Digital Garden\" });\n\n    new Setting(containerEl)\n      .setName(\"Map #garden/* tags to gardenStage\")\n      .setDesc(\n        \"Obsidian tags like #garden/plant become a `garden-stage: plant` Micropub \" +\n        \"property. The blog renders these as growth stage badges at /garden/.\",\n      )\n      .addToggle((toggle) =>\n        toggle\n          .setValue(this.plugin.settings.mapGardenTags)\n          .onChange(async (value) => {\n            this.plugin.settings.mapGardenTags = value;\n            await this.plugin.saveSettings();\n          }),\n      );\n\n    containerEl.createEl(\"p\", {\n      text: \"Stages: plant \uD83C\uDF31 \u00B7 cultivate \uD83C\uDF3F \u00B7 question \u2753 \u00B7 repot \uD83E\uDEB4 \u00B7 revitalize \u2728 \u00B7 revisit \uD83D\uDD04\",\n      cls: \"setting-item-description\",\n    });\n  }\n\n  // \u2500\u2500 Signed-out state \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\n\n  private renderSignedOut(containerEl: HTMLElement): void {\n    // Site URL input + Sign In button on the same row\n    new Setting(containerEl)\n      .setName(\"Site URL\")\n      .setDesc(\n        \"Your site's home page. Clicking Sign in opens your blog's login page \" +\n        \"in the browser \u2014 the same flow iA Writer uses.\",\n      )\n      .addText((text) =>\n        text\n          .setPlaceholder(\"https://blog.giersig.eu\")\n          .setValue(this.plugin.settings.siteUrl)\n          .onChange(async (value) => {\n            this.plugin.settings.siteUrl = value.trim();\n            await this.plugin.saveSettings();\n          }),\n      )\n      .addButton((btn) => {\n        btn\n          .setButtonText(\"Sign in\")\n          .setCta()\n          .onClick(async () => {\n            const siteUrl = this.plugin.settings.siteUrl.trim();\n            if (!siteUrl) {\n              new Notice(\"Enter your site URL first.\");\n              return;\n            }\n\n            btn.setDisabled(true);\n            btn.setButtonText(\"Opening browser\u2026\");\n\n            try {\n              const result = await IndieAuth.signIn(siteUrl);\n\n              // Save everything returned by the auth flow\n              this.plugin.settings.accessToken      = result.accessToken;\n              this.plugin.settings.me               = result.me;\n              this.plugin.settings.authorizationEndpoint = result.authorizationEndpoint;\n              this.plugin.settings.tokenEndpoint    = result.tokenEndpoint;\n              if (result.micropubEndpoint) {\n                this.plugin.settings.micropubEndpoint = result.micropubEndpoint;\n              }\n              if (result.mediaEndpoint) {\n                this.plugin.settings.mediaEndpoint = result.mediaEndpoint;\n              }\n\n              await this.plugin.saveSettings();\n\n              // Try to fetch the Micropub config to pick up media endpoint\n              if (!this.plugin.settings.mediaEndpoint) {\n                try {\n                  const client = new MicropubClient(\n                    () => this.plugin.settings.micropubEndpoint,\n                    () => this.plugin.settings.mediaEndpoint,\n                    () => this.plugin.settings.accessToken,\n                  );\n                  const cfg = await client.fetchConfig();\n                  if (cfg[\"media-endpoint\"]) {\n                    this.plugin.settings.mediaEndpoint = cfg[\"media-endpoint\"];\n                    await this.plugin.saveSettings();\n                  }\n                } catch {\n                  // Non-fatal\n                }\n              }\n\n              new Notice(`\u2705 Signed in as ${result.me}`);\n              this.display(); // Refresh to show signed-in state\n            } catch (err: unknown) {\n              new Notice(`Sign-in failed: ${String(err)}`, 8000);\n              btn.setDisabled(false);\n              btn.setButtonText(\"Sign in\");\n            }\n          });\n      });\n\n    // Divider + manual token fallback (collapsed by default)\n    const details = containerEl.createEl(\"details\");\n    details.createEl(\"summary\", {\n      text: \"Or paste a token manually\",\n      cls: \"setting-item-description\",\n    });\n    details.style.marginTop = \"8px\";\n    details.style.marginBottom = \"8px\";\n\n    new Setting(details)\n      .setName(\"Access token\")\n      .setDesc(\"Bearer token from your Indiekit admin panel.\")\n      .addText((text) => {\n        text\n          .setPlaceholder(\"your-bearer-token\")\n          .setValue(this.plugin.settings.accessToken)\n          .onChange(async (value) => {\n            this.plugin.settings.accessToken = value.trim();\n            await this.plugin.saveSettings();\n          });\n        text.inputEl.type = \"password\";\n      })\n      .addButton((btn) =>\n        btn.setButtonText(\"Verify\").onClick(async () => {\n          if (\n            !this.plugin.settings.micropubEndpoint ||\n            !this.plugin.settings.accessToken\n          ) {\n            new Notice(\"Set the Micropub endpoint and token first.\");\n            return;\n          }\n          btn.setDisabled(true);\n          try {\n            const client = new MicropubClient(\n              () => this.plugin.settings.micropubEndpoint,\n              () => this.plugin.settings.mediaEndpoint,\n              () => this.plugin.settings.accessToken,\n            );\n            await client.fetchConfig();\n            new Notice(\"\u2705 Token is valid!\");\n          } catch (err: unknown) {\n            new Notice(`Token check failed: ${String(err)}`);\n          } finally {\n            btn.setDisabled(false);\n          }\n        }),\n      );\n  }\n\n  // \u2500\u2500 Signed-in state \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\n\n  private renderSignedIn(containerEl: HTMLElement): void {\n    const me = this.plugin.settings.me;\n\n    // Avatar + \"Signed in as\" banner\n    const banner = containerEl.createDiv({\n      cls: \"micropub-auth-banner\",\n    });\n    banner.style.cssText =\n      \"display:flex;align-items:center;gap:12px;padding:12px 16px;\" +\n      \"border:1px solid var(--background-modifier-border);\" +\n      \"border-radius:8px;margin-bottom:16px;background:var(--background-secondary);\";\n\n    const icon = banner.createDiv();\n    icon.style.cssText =\n      \"width:40px;height:40px;border-radius:50%;background:var(--interactive-accent);\" +\n      \"display:flex;align-items:center;justify-content:center;\" +\n      \"font-size:1.2rem;flex-shrink:0;\";\n    icon.textContent = \"\uD83C\uDF10\";\n\n    const info = banner.createDiv();\n    info.createEl(\"div\", {\n      text: \"Signed in\",\n      attr: { style: \"font-size:.75rem;color:var(--text-muted);margin-bottom:2px\" },\n    });\n    info.createEl(\"div\", {\n      text: me,\n      attr: { style: \"font-weight:500;word-break:break-all\" },\n    });\n\n    new Setting(containerEl)\n      .setName(\"Site URL\")\n      .addText((text) =>\n        text\n          .setValue(this.plugin.settings.siteUrl)\n          .setDisabled(true),\n      )\n      .addButton((btn) =>\n        btn\n          .setButtonText(\"Sign out\")\n          .setWarning()\n          .onClick(async () => {\n            this.plugin.settings.accessToken = \"\";\n            this.plugin.settings.me = \"\";\n            this.plugin.settings.authorizationEndpoint = \"\";\n            this.plugin.settings.tokenEndpoint = \"\";\n            await this.plugin.saveSettings();\n            this.display();\n          }),\n      );\n  }\n}\n", "/**\n * MicropubClient.ts\n *\n * Low-level HTTP client for Micropub and Media endpoint requests.\n * Uses Obsidian's requestUrl() so requests are made from the desktop app\n * (no CORS issues) rather than a browser fetch.\n */\n\nimport { requestUrl, RequestUrlParam } from \"obsidian\";\nimport type { MicropubConfig, PublishResult } from \"./types\";\n\nexport class MicropubClient {\n  constructor(\n    private readonly getEndpoint: () => string,\n    private readonly getMediaEndpoint: () => string,\n    private readonly getToken: () => string,\n  ) {}\n\n  // \u2500\u2500 Config discovery \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\n\n  /** Fetch Micropub server config (syndication targets, media endpoint, etc.) */\n  async fetchConfig(): Promise<MicropubConfig> {\n    const url = `${this.getEndpoint()}?q=config`;\n    const resp = await requestUrl({\n      url,\n      method: \"GET\",\n      headers: this.authHeaders(),\n    });\n    return resp.json as MicropubConfig;\n  }\n\n  /**\n   * Discover micropub + token endpoint URLs from a site's home page\n   * by reading <link rel=\"micropub\"> and <link rel=\"token_endpoint\"> tags.\n   */\n  async discoverEndpoints(siteUrl: string): Promise<{\n    micropubEndpoint?: string;\n    tokenEndpoint?: string;\n    mediaEndpoint?: string;\n  }> {\n    const resp = await requestUrl({ url: siteUrl, method: \"GET\" });\n    const html = resp.text;\n\n    const micropub = this.extractLinkRel(html, \"micropub\");\n    const tokenEndpoint = this.extractLinkRel(html, \"token_endpoint\");\n\n    // After discovering the Micropub endpoint, fetch its config for the media URL\n    let mediaEndpoint: string | undefined;\n    if (micropub) {\n      try {\n        const cfg = await this.fetchConfigFrom(micropub);\n        mediaEndpoint = cfg[\"media-endpoint\"];\n      } catch {\n        // Non-fatal \u2014 media endpoint stays undefined\n      }\n    }\n\n    return { micropubEndpoint: micropub, tokenEndpoint, mediaEndpoint };\n  }\n\n  // \u2500\u2500 Post publishing \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\n\n  /**\n   * Create a new post via Micropub.\n   * Sends a JSON body with h-entry properties.\n   * Returns the Location header URL on success.\n   */\n  async createPost(properties: Record<string, unknown>): Promise<PublishResult> {\n    const body = {\n      type: [\"h-entry\"],\n      properties,\n    };\n\n    try {\n      const resp = await requestUrl({\n        url: this.getEndpoint(),\n        method: \"POST\",\n        headers: {\n          ...this.authHeaders(),\n          \"Content-Type\": \"application/json\",\n        },\n        body: JSON.stringify(body),\n        throw: false,\n      });\n\n      if (resp.status === 201 || resp.status === 202) {\n        const location =\n          resp.headers?.[\"location\"] ||\n          resp.headers?.[\"Location\"] ||\n          (resp.json as { url?: string })?.url;\n        return { success: true, url: location };\n      }\n\n      const detail = this.extractError(resp.text);\n      return { success: false, error: `HTTP ${resp.status}: ${detail}` };\n    } catch (err: unknown) {\n      return { success: false, error: String(err) };\n    }\n  }\n\n  /**\n   * Update an existing post.\n   * @param postUrl  The canonical URL of the post to update\n   * @param replace  Properties to replace (will overwrite existing values)\n   */\n  async updatePost(\n    postUrl: string,\n    replace: Record<string, unknown[]>,\n  ): Promise<PublishResult> {\n    const body = { action: \"update\", url: postUrl, replace };\n\n    try {\n      const resp = await requestUrl({\n        url: this.getEndpoint(),\n        method: \"POST\",\n        headers: {\n          ...this.authHeaders(),\n          \"Content-Type\": \"application/json\",\n        },\n        body: JSON.stringify(body),\n        throw: false,\n      });\n\n      if (resp.status >= 200 && resp.status < 300) {\n        return { success: true, url: postUrl };\n      }\n\n      return {\n        success: false,\n        error: `HTTP ${resp.status}: ${this.extractError(resp.text)}`,\n      };\n    } catch (err: unknown) {\n      return { success: false, error: String(err) };\n    }\n  }\n\n  // \u2500\u2500 Media upload \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\n\n  /**\n   * Upload a binary file to the media endpoint.\n   * @returns The URL of the uploaded media, or throws on failure.\n   */\n  async uploadMedia(\n    fileBuffer: ArrayBuffer,\n    fileName: string,\n    mimeType: string,\n  ): Promise<string> {\n    const endpoint = this.getMediaEndpoint() || `${this.getEndpoint()}/media`;\n\n    // Build multipart/form-data manually \u2014 Obsidian's requestUrl doesn't\n    // support FormData directly, so we encode the boundary ourselves.\n    const boundary = `----MicropubBoundary${Date.now()}`;\n    const header =\n      `--${boundary}\\r\\n` +\n      `Content-Disposition: form-data; name=\"file\"; filename=\"${fileName}\"\\r\\n` +\n      `Content-Type: ${mimeType}\\r\\n\\r\\n`;\n    const footer = `\\r\\n--${boundary}--\\r\\n`;\n\n    const headerBuf = new TextEncoder().encode(header);\n    const footerBuf = new TextEncoder().encode(footer);\n    const fileBuf = new Uint8Array(fileBuffer);\n\n    const combined = new Uint8Array(\n      headerBuf.length + fileBuf.length + footerBuf.length,\n    );\n    combined.set(headerBuf, 0);\n    combined.set(fileBuf, headerBuf.length);\n    combined.set(footerBuf, headerBuf.length + fileBuf.length);\n\n    const resp = await requestUrl({\n      url: endpoint,\n      method: \"POST\",\n      headers: {\n        ...this.authHeaders(),\n        \"Content-Type\": `multipart/form-data; boundary=${boundary}`,\n      },\n      body: combined.buffer,\n      throw: false,\n    });\n\n    if (resp.status === 201 || resp.status === 202) {\n      const location =\n        resp.headers?.[\"location\"] ||\n        resp.headers?.[\"Location\"] ||\n        (resp.json as { url?: string })?.url;\n      if (location) return location;\n    }\n\n    throw new Error(\n      `Media upload failed (HTTP ${resp.status}): ${this.extractError(resp.text)}`,\n    );\n  }\n\n  // \u2500\u2500 Helpers \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\n\n  private authHeaders(): Record<string, string> {\n    return { Authorization: `Bearer ${this.getToken()}` };\n  }\n\n  private extractLinkRel(html: string, rel: string): string | undefined {\n    // Match both <link> and HTTP Link headers embedded in HTML\n    const re = new RegExp(\n      `<link[^>]+rel=[\"']${rel}[\"'][^>]+href=[\"']([^\"']+)[\"']|<link[^>]+href=[\"']([^\"']+)[\"'][^>]+rel=[\"']${rel}[\"']`,\n      \"i\",\n    );\n    const m = html.match(re);\n    return m?.[1] ?? m?.[2];\n  }\n\n  private async fetchConfigFrom(endpoint: string): Promise<MicropubConfig> {\n    const resp = await requestUrl({\n      url: `${endpoint}?q=config`,\n      method: \"GET\",\n      headers: this.authHeaders(),\n    });\n    return resp.json as MicropubConfig;\n  }\n\n  private extractError(text: string): string {\n    try {\n      const obj = JSON.parse(text) as { error_description?: string; error?: string };\n      return obj.error_description ?? obj.error ?? text.slice(0, 200);\n    } catch {\n      return text.slice(0, 200);\n    }\n  }\n}\n", "/**\n * IndieAuth.ts \u2014 IndieAuth PKCE sign-in flow for obsidian-micropub\n *\n * Why no local HTTP server:\n *   IndieKit (and most IndieAuth servers) fetch the client_id URL server-side\n *   to retrieve app metadata. A local 127.0.0.1 address is unreachable from a\n *   remote server, so that approach always fails with \"fetch failed\".\n *\n * The solution \u2014 GitHub Pages relay:\n *   client_id  = https://svemagie.github.io/obsidian-micropub/\n *   redirect_uri = https://svemagie.github.io/obsidian-micropub/callback\n *\n *   Both are on the same host \u2192 IndieKit's host-matching check passes \u2713\n *   The callback page is a static HTML file that immediately redirects to\n *   obsidian://micropub-auth?code=CODE&state=STATE\n *   Obsidian's protocol handler (registered in main.ts) receives the code.\n *\n * Flow:\n *   1. Discover authorization_endpoint + token_endpoint from site HTML\n *   2. Generate PKCE code_verifier + code_challenge (SHA-256)\n *   3. Open browser \u2192 user's IndieAuth login page\n *   4. User logs in \u2192 server redirects to GitHub Pages callback\n *   5. Callback page redirects to obsidian://micropub-auth?code=...\n *   6. Plugin protocol handler resolves the pending Promise\n *   7. Exchange code for token at token_endpoint\n */\n\nimport * as crypto from \"crypto\";\nimport { requestUrl } from \"obsidian\";\n\nexport const CLIENT_ID   = \"https://svemagie.github.io/obsidian-micropub/\";\nexport const REDIRECT_URI = \"https://svemagie.github.io/obsidian-micropub/callback\";\n\nconst SCOPE = \"create update media\";\nconst AUTH_TIMEOUT_MS = 5 * 60 * 1000; // 5 minutes\n\nexport interface IndieAuthResult {\n  accessToken: string;\n  scope: string;\n  /** Canonical \"me\" URL returned by the token endpoint */\n  me: string;\n  authorizationEndpoint: string;\n  tokenEndpoint: string;\n  micropubEndpoint?: string;\n  mediaEndpoint?: string;\n}\n\nexport interface DiscoveredEndpoints {\n  authorizationEndpoint: string;\n  tokenEndpoint: string;\n  micropubEndpoint?: string;\n}\n\n/** Pending callback set by main.ts protocol handler */\nlet pendingCallback:\n  | { resolve: (params: Record<string, string>) => void; state: string }\n  | null = null;\n\n/**\n * Called by the Obsidian protocol handler in main.ts when\n * obsidian://micropub-auth is opened by the browser.\n */\nexport function handleProtocolCallback(params: Record<string, string>): void {\n  if (!pendingCallback) return;\n\n  const { resolve, state: expectedState } = pendingCallback;\n  pendingCallback = null;\n  resolve(params); // let signIn() validate state + extract code\n}\n\nexport class IndieAuth {\n  // \u2500\u2500 Public API \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\n\n  /**\n   * Discover IndieAuth + Micropub endpoint URLs from the site's home page\n   * by reading <link rel=\"...\"> tags in the HTML <head>.\n   */\n  static async discoverEndpoints(siteUrl: string): Promise<DiscoveredEndpoints> {\n    const resp = await requestUrl({ url: siteUrl, method: \"GET\" });\n    const html = resp.text;\n\n    const authorizationEndpoint = IndieAuth.extractLinkRel(html, \"authorization_endpoint\");\n    const tokenEndpoint         = IndieAuth.extractLinkRel(html, \"token_endpoint\");\n    const micropubEndpoint      = IndieAuth.extractLinkRel(html, \"micropub\");\n\n    if (!authorizationEndpoint) {\n      throw new Error(\n        `No <link rel=\"authorization_endpoint\"> found at ${siteUrl}. ` +\n        \"Make sure Indiekit is running and SITE_URL is set correctly.\",\n      );\n    }\n    if (!tokenEndpoint) {\n      throw new Error(`No <link rel=\"token_endpoint\"> found at ${siteUrl}.`);\n    }\n\n    return { authorizationEndpoint, tokenEndpoint, micropubEndpoint };\n  }\n\n  /**\n   * Run the full IndieAuth PKCE sign-in flow.\n   *\n   * Opens the browser at the user's IndieAuth login page. After login the\n   * browser is redirected to the GitHub Pages callback, which triggers\n   * the obsidian://micropub-auth protocol, which resolves the Promise here.\n   *\n   * Requires handleProtocolCallback() to be wired up in main.ts via\n   * this.registerObsidianProtocolHandler(\"micropub-auth\", handleProtocolCallback)\n   */\n  static async signIn(siteUrl: string): Promise<IndieAuthResult> {\n    // 1. Discover endpoints\n    const { authorizationEndpoint, tokenEndpoint, micropubEndpoint } =\n      await IndieAuth.discoverEndpoints(siteUrl);\n\n    // 2. Generate PKCE + state\n    const state        = IndieAuth.base64url(crypto.randomBytes(16));\n    const codeVerifier = IndieAuth.base64url(crypto.randomBytes(64));\n    const codeChallenge = IndieAuth.base64url(\n      crypto.createHash(\"sha256\").update(codeVerifier).digest(),\n    );\n\n    // 3. Register pending callback \u2014 resolved by handleProtocolCallback()\n    const callbackPromise = new Promise<Record<string, string>>(\n      (resolve, reject) => {\n        const timeout = setTimeout(() => {\n          pendingCallback = null;\n          reject(new Error(\"Sign-in timed out (5 min). Please try again.\"));\n        }, AUTH_TIMEOUT_MS);\n\n        pendingCallback = {\n          state,\n          resolve: (params) => {\n            clearTimeout(timeout);\n            resolve(params);\n          },\n        };\n      },\n    );\n\n    // 4. Build the authorization URL and open the browser\n    const authUrl = new URL(authorizationEndpoint);\n    authUrl.searchParams.set(\"response_type\",        \"code\");\n    authUrl.searchParams.set(\"client_id\",            CLIENT_ID);\n    authUrl.searchParams.set(\"redirect_uri\",         REDIRECT_URI);\n    authUrl.searchParams.set(\"state\",                state);\n    authUrl.searchParams.set(\"code_challenge\",       codeChallenge);\n    authUrl.searchParams.set(\"code_challenge_method\",\"S256\");\n    authUrl.searchParams.set(\"scope\",                SCOPE);\n    authUrl.searchParams.set(\"me\",                   siteUrl);\n\n    window.open(authUrl.toString());\n\n    // 5. Wait for obsidian://micropub-auth to be called\n    const callbackParams = await callbackPromise;\n\n    // 6. Validate state (CSRF protection)\n    if (callbackParams.state !== state) {\n      throw new Error(\"State mismatch \u2014 possible CSRF attack. Please try again.\");\n    }\n\n    const code = callbackParams.code;\n    if (!code) {\n      throw new Error(\n        callbackParams.error_description ??\n        callbackParams.error ??\n        \"No authorization code received.\",\n      );\n    }\n\n    // 7. Exchange code for token\n    const tokenResp = await requestUrl({\n      url: tokenEndpoint,\n      method: \"POST\",\n      headers: {\n        \"Content-Type\": \"application/x-www-form-urlencoded\",\n        Accept: \"application/json\",\n      },\n      body: new URLSearchParams({\n        grant_type:    \"authorization_code\",\n        code,\n        client_id:     CLIENT_ID,\n        redirect_uri:  REDIRECT_URI,\n        code_verifier: codeVerifier,\n      }).toString(),\n      throw: false,\n    });\n\n    const data = tokenResp.json as {\n      access_token?: string;\n      scope?: string;\n      me?: string;\n      error?: string;\n      error_description?: string;\n    };\n\n    if (!data.access_token) {\n      throw new Error(\n        data.error_description ??\n        data.error ??\n        `Token exchange failed (HTTP ${tokenResp.status})`,\n      );\n    }\n\n    return {\n      accessToken:           data.access_token,\n      scope:                 data.scope ?? SCOPE,\n      me:                    data.me ?? siteUrl,\n      authorizationEndpoint,\n      tokenEndpoint,\n      micropubEndpoint,\n    };\n  }\n\n  // \u2500\u2500 Helpers \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\n\n  private static base64url(buf: Buffer): string {\n    return buf.toString(\"base64\")\n      .replace(/\\+/g, \"-\")\n      .replace(/\\//g, \"_\")\n      .replace(/=/g, \"\");\n  }\n\n  static extractLinkRel(html: string, rel: string): string | undefined {\n    const re = new RegExp(\n      `<link[^>]+rel=[\"'][^\"']*\\\\b${rel}\\\\b[^\"']*[\"'][^>]+href=[\"']([^\"']+)[\"']` +\n      `|<link[^>]+href=[\"']([^\"']+)[\"'][^>]+rel=[\"'][^\"']*\\\\b${rel}\\\\b[^\"']*[\"']`,\n      \"i\",\n    );\n    const m = html.match(re);\n    return m?.[1] ?? m?.[2];\n  }\n}\n", "/**\n * Publisher.ts\n *\n * Orchestrates a full publish flow:\n *   1. Parse the active note's frontmatter + body\n *   2. Upload any local images to the media endpoint\n *   3. Build the Micropub properties object\n *   4. POST to the Micropub endpoint\n *   5. Optionally write the returned URL back to frontmatter\n *\n * Garden tag mapping:\n *   Obsidian tags #garden/plant \u2192 gardenStage: \"plant\" in properties\n *   The blog reads this as `gardenStage` frontmatter, so the Indiekit\n *   Micropub server must be configured to pass through unknown properties.\n */\n\nimport { App, TFile, parseFrontMatterAliases, parseYaml, stringifyYaml } from \"obsidian\";\nimport type { MicropubSettings, GardenStage, PublishResult } from \"./types\";\nimport { MicropubClient } from \"./MicropubClient\";\n\nconst GARDEN_TAG_PREFIX = \"garden/\";\n\nexport class Publisher {\n  private client: MicropubClient;\n\n  constructor(\n    private readonly app: App,\n    private readonly settings: MicropubSettings,\n  ) {\n    this.client = new MicropubClient(\n      () => settings.micropubEndpoint,\n      () => settings.mediaEndpoint,\n      () => settings.accessToken,\n    );\n  }\n\n  /** Publish the given file. Returns PublishResult. */\n  async publish(file: TFile): Promise<PublishResult> {\n    const raw = await this.app.vault.read(file);\n    const { frontmatter, body } = this.parseFrontmatter(raw);\n\n    // Determine if this is an update (post already has a URL) or new post\n    const existingUrl: string | undefined =\n      frontmatter[\"mp-url\"] ?? frontmatter[\"url\"] ?? undefined;\n\n    // Upload local images and rewrite markdown references\n    const { content: processedBody, uploadedUrls } =\n      await this.processImages(body);\n\n    // Build Micropub properties\n    const properties = this.buildProperties(frontmatter, processedBody, uploadedUrls);\n\n    let result: PublishResult;\n\n    if (existingUrl) {\n      // Update existing post\n      const replace: Record<string, unknown[]> = {};\n      for (const [k, v] of Object.entries(properties)) {\n        replace[k] = Array.isArray(v) ? v : [v];\n      }\n      result = await this.client.updatePost(existingUrl, replace);\n    } else {\n      // Create new post\n      result = await this.client.createPost(properties);\n    }\n\n    // Write URL back to frontmatter\n    if (result.success && result.url && this.settings.writeUrlToFrontmatter) {\n      await this.writeUrlToNote(file, raw, result.url);\n    }\n\n    return result;\n  }\n\n  // \u2500\u2500 Property builder \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\n\n  private buildProperties(\n    fm: Record<string, unknown>,\n    body: string,\n    uploadedUrls: string[],\n  ): Record<string, unknown> {\n    const props: Record<string, unknown> = {};\n\n    // \u2500\u2500 Post type detection \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\n    // Interaction posts (bookmark, like, reply, repost) have no body content.\n    // For those, only include content if the note body is non-empty (i.e. a comment/quote).\n    const trimmedBody = body.trim();\n\n    // \u2500\u2500 Interaction URL properties \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\n    // Support both camelCase (Obsidian-friendly) and hyphenated (Micropub-spec).\n    const bookmarkOf = fm[\"bookmarkOf\"] ?? fm[\"bookmark-of\"];\n    const likeOf     = fm[\"likeOf\"]     ?? fm[\"like-of\"];\n    const inReplyTo  = fm[\"inReplyTo\"]  ?? fm[\"in-reply-to\"];\n    const repostOf   = fm[\"repostOf\"]   ?? fm[\"repost-of\"];\n\n    if (bookmarkOf) props[\"bookmark-of\"] = [String(bookmarkOf)];\n    if (likeOf)     props[\"like-of\"]     = [String(likeOf)];\n    if (inReplyTo)  props[\"in-reply-to\"] = [String(inReplyTo)];\n    if (repostOf)   props[\"repost-of\"]   = [String(repostOf)];\n\n    // Content \u2014 omit for bare likes/reposts with no body text\n    const isInteractionWithoutBody =\n      (likeOf || repostOf) && !trimmedBody;\n    if (!isInteractionWithoutBody) {\n      props[\"content\"] = trimmedBody ? [{ html: trimmedBody }] : [{ html: \"\" }];\n    }\n\n    // \u2500\u2500 Standard properties \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\n\n    // Title (articles have titles; notes/micro-posts don't)\n    if (fm[\"title\"]) {\n      props[\"name\"] = [String(fm[\"title\"])];\n    }\n\n    // Summary / excerpt\n    if (fm[\"summary\"] ?? fm[\"excerpt\"]) {\n      props[\"summary\"] = [String(fm[\"summary\"] ?? fm[\"excerpt\"])];\n    }\n\n    // Published date\n    if (fm[\"date\"]) {\n      props[\"published\"] = [new Date(String(fm[\"date\"])).toISOString()];\n    }\n\n    // Categories from frontmatter `category` or `tags` (excluding garden/* tags)\n    const rawTags = this.resolveArray(fm[\"tags\"] ?? fm[\"category\"]);\n    const gardenStage = this.extractGardenStage(rawTags);\n    const normalTags = rawTags.filter(\n      (t) => !t.startsWith(GARDEN_TAG_PREFIX) && t !== \"garden\",\n    );\n    if (normalTags.length > 0) {\n      props[\"category\"] = normalTags;\n    }\n\n    // Garden stage \u2192 dedicated property\n    if (this.settings.mapGardenTags && gardenStage) {\n      props[\"garden-stage\"] = [gardenStage];\n    }\n\n    // Syndication targets\n    // Support both camelCase (mpSyndicateTo) used in existing blog posts and mp-syndicate-to\n    const syndicateTo = this.resolveArray(\n      fm[\"mp-syndicate-to\"] ?? fm[\"mpSyndicateTo\"],\n    );\n    const allSyndicateTo = [\n      ...new Set([...this.settings.defaultSyndicateTo, ...syndicateTo]),\n    ];\n    if (allSyndicateTo.length > 0) {\n      props[\"mp-syndicate-to\"] = allSyndicateTo;\n    }\n\n    // Visibility\n    const visibility =\n      (fm[\"visibility\"] as string) ?? this.settings.defaultVisibility;\n    if (visibility && visibility !== \"public\") {\n      props[\"visibility\"] = [visibility];\n    }\n\n    // AI disclosure (custom property passed through to Indiekit)\n    if (fm[\"ai\"] && typeof fm[\"ai\"] === \"object\") {\n      props[\"ai\"] = [fm[\"ai\"]];\n    }\n\n    // Photos: prefer structured photo array from frontmatter (with alt text),\n    // fall back to uploaded local images.\n    const fmPhotos = this.resolvePhotoArray(fm[\"photo\"]);\n    if (fmPhotos.length > 0) {\n      props[\"photo\"] = fmPhotos;\n    } else if (uploadedUrls.length > 0) {\n      props[\"photo\"] = uploadedUrls.map((url) => ({ value: url }));\n    }\n\n    // Pass through any `mp-*` properties from frontmatter verbatim\n    for (const [k, v] of Object.entries(fm)) {\n      if (k.startsWith(\"mp-\") && k !== \"mp-url\" && k !== \"mp-syndicate-to\") {\n        props[k] = this.resolveArray(v);\n      }\n    }\n\n    return props;\n  }\n\n  /**\n   * Normalise the `photo` frontmatter field into Micropub photo objects.\n   * Handles three formats:\n   *   - string URL: \"https://...\"\n   *   - array of strings: [\"https://...\"]\n   *   - array of objects: [{url: \"https://...\", alt: \"...\"}]\n   */\n  private resolvePhotoArray(\n    value: unknown,\n  ): Array<{ value: string; alt?: string }> {\n    if (!value) return [];\n    const items = Array.isArray(value) ? value : [value];\n    return items\n      .map((item) => {\n        if (typeof item === \"string\") return { value: item };\n        if (typeof item === \"object\" && item !== null) {\n          const obj = item as Record<string, unknown>;\n          const url = String(obj[\"url\"] ?? obj[\"value\"] ?? \"\");\n          if (!url) return null;\n          return obj[\"alt\"]\n            ? { value: url, alt: String(obj[\"alt\"]) }\n            : { value: url };\n        }\n        return null;\n      })\n      .filter((x): x is { value: string; alt?: string } => x !== null);\n  }\n\n  // \u2500\u2500 Garden tag extraction \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\n\n  /**\n   * Find the first #garden/<stage> tag and return the stage name.\n   * Supports both \"garden/plant\" (Obsidian array) and \"#garden/plant\" (inline).\n   */\n  private extractGardenStage(tags: string[]): GardenStage | undefined {\n    for (const tag of tags) {\n      const clean = tag.replace(/^#/, \"\");\n      if (clean.startsWith(GARDEN_TAG_PREFIX)) {\n        const stage = clean.slice(GARDEN_TAG_PREFIX.length) as GardenStage;\n        const valid: GardenStage[] = [\n          \"plant\", \"cultivate\", \"question\", \"repot\", \"revitalize\", \"revisit\",\n        ];\n        if (valid.includes(stage)) return stage;\n      }\n    }\n    return undefined;\n  }\n\n  // \u2500\u2500 Image processing \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\n\n  /**\n   * Find all `![[local-image.png]]` or `![alt](relative/path.jpg)` in the body,\n   * upload them to the media endpoint, and replace the references with remote URLs.\n   */\n  private async processImages(\n    body: string,\n  ): Promise<{ content: string; uploadedUrls: string[] }> {\n    const uploadedUrls: string[] = [];\n\n    // Match wiki-style embeds: ![[filename.ext]]\n    const wikiPattern = /!\\[\\[([^\\]]+\\.(png|jpg|jpeg|gif|webp|svg))\\]\\]/gi;\n    // Match markdown images: ![alt](path)\n    const mdPattern = /!\\[([^\\]]*)\\]\\(([^)]+\\.(png|jpg|jpeg|gif|webp|svg))\\)/gi;\n\n    let content = body;\n\n    // Process wiki-style embeds\n    const wikiMatches = [...body.matchAll(wikiPattern)];\n    for (const match of wikiMatches) {\n      const filename = match[1];\n      try {\n        const remoteUrl = await this.uploadLocalFile(filename);\n        if (remoteUrl) {\n          uploadedUrls.push(remoteUrl);\n          content = content.replace(match[0], `![${filename}](${remoteUrl})`);\n        }\n      } catch (err) {\n        console.warn(`[micropub] Failed to upload ${filename}:`, err);\n      }\n    }\n\n    // Process markdown image references\n    const mdMatches = [...content.matchAll(mdPattern)];\n    for (const match of mdMatches) {\n      const alt = match[1];\n      const path = match[2];\n      if (path.startsWith(\"http\")) continue; // already remote\n      try {\n        const remoteUrl = await this.uploadLocalFile(path);\n        if (remoteUrl) {\n          uploadedUrls.push(remoteUrl);\n          content = content.replace(match[0], `![${alt}](${remoteUrl})`);\n        }\n      } catch (err) {\n        console.warn(`[micropub] Failed to upload ${path}:`, err);\n      }\n    }\n\n    return { content, uploadedUrls };\n  }\n\n  private async uploadLocalFile(path: string): Promise<string | undefined> {\n    const file = this.app.vault.getFiles().find(\n      (f) => f.name === path || f.path === path,\n    );\n    if (!file) return undefined;\n\n    const buffer = await this.app.vault.readBinary(file);\n    const mimeType = this.guessMimeType(file.extension);\n\n    return this.client.uploadMedia(buffer, file.name, mimeType);\n  }\n\n  // \u2500\u2500 Frontmatter helpers \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\n\n  private parseFrontmatter(raw: string): {\n    frontmatter: Record<string, unknown>;\n    body: string;\n  } {\n    const fmMatch = raw.match(/^---\\r?\\n([\\s\\S]*?)\\r?\\n---\\r?\\n([\\s\\S]*)$/);\n    if (!fmMatch) return { frontmatter: {}, body: raw };\n\n    let frontmatter: Record<string, unknown> = {};\n    try {\n      frontmatter = (parseYaml(fmMatch[1]) ?? {}) as Record<string, unknown>;\n    } catch {\n      // Malformed frontmatter \u2014 treat as empty\n    }\n\n    return { frontmatter, body: fmMatch[2] };\n  }\n\n  private async writeUrlToNote(\n    file: TFile,\n    originalContent: string,\n    url: string,\n  ): Promise<void> {\n    const fmMatch = originalContent.match(\n      /^(---\\r?\\n[\\s\\S]*?\\r?\\n---\\r?\\n)([\\s\\S]*)$/,\n    );\n\n    if (!fmMatch) {\n      // No existing frontmatter \u2014 prepend it\n      const newFm = `---\\nmp-url: \"${url}\"\\n---\\n`;\n      await this.app.vault.modify(file, newFm + originalContent);\n      return;\n    }\n\n    // Inject mp-url into existing frontmatter block\n    const fmBlock = fmMatch[1];\n    const body = fmMatch[2];\n\n    if (fmBlock.includes(\"mp-url:\")) {\n      // Replace existing mp-url line\n      const updated = fmBlock.replace(\n        /mp-url:.*(\\r?\\n)/,\n        `mp-url: \"${url}\"$1`,\n      );\n      await this.app.vault.modify(file, updated + body);\n    } else {\n      // Insert mp-url before closing ---\n      const updated = fmBlock.replace(\n        /(\\r?\\n---\\r?\\n)$/,\n        `\\nmp-url: \"${url}\"$1`,\n      );\n      await this.app.vault.modify(file, updated + body);\n    }\n  }\n\n  private resolveArray(value: unknown): string[] {\n    if (!value) return [];\n    if (Array.isArray(value)) return value.map(String);\n    return [String(value)];\n  }\n\n  private guessMimeType(ext: string): string {\n    const map: Record<string, string> = {\n      png: \"image/png\",\n      jpg: \"image/jpeg\",\n      jpeg: \"image/jpeg\",\n      gif: \"image/gif\",\n      webp: \"image/webp\",\n      svg: \"image/svg+xml\",\n    };\n    return map[ext.toLowerCase()] ?? \"application/octet-stream\";\n  }\n}\n"],
  "mappings": "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",
  "names": ["import_obsidian", "import_obsidian", "import_obsidian", "import_obsidian"]
}

diff --git a/node_modules/.bin/esbuild b/node_modules/.bin/esbuild
new file mode 120000
index 0000000..c83ac07
--- /dev/null
+++ b/node_modules/.bin/esbuild
@@ -0,0 +1 @@
+../esbuild/bin/esbuild
\ No newline at end of file
diff --git a/node_modules/.bin/tsc b/node_modules/.bin/tsc
new file mode 120000
index 0000000..0863208
--- /dev/null
+++ b/node_modules/.bin/tsc
@@ -0,0 +1 @@
+../typescript/bin/tsc
\ No newline at end of file
diff --git a/node_modules/.bin/tsserver b/node_modules/.bin/tsserver
new file mode 120000
index 0000000..f8f8f1a
--- /dev/null
+++ b/node_modules/.bin/tsserver
@@ -0,0 +1 @@
+../typescript/bin/tsserver
\ No newline at end of file
diff --git a/node_modules/.package-lock.json b/node_modules/.package-lock.json
new file mode 100644
index 0000000..b85d85c
--- /dev/null
+++ b/node_modules/.package-lock.json
@@ -0,0 +1,237 @@
+{
+ "name": "obsidian-micropub",
+ "version": "0.1.0",
+ "lockfileVersion": 3,
+ "requires": true,
+ "packages": {
+ "node_modules/@codemirror/state": {
+ "version": "6.5.0",
+ "resolved": "https://registry.npmjs.org/@codemirror/state/-/state-6.5.0.tgz",
+ "integrity": "sha512-MwBHVK60IiIHDcoMet78lxt6iw5gJOGSbNbOIVBHWVXIH4/Nq1+GQgLLGgI1KlnN86WDXsPudVaqYHKBIx7Eyw==",
+ "dev": true,
+ "license": "MIT",
+ "peer": true,
+ "dependencies": {
+ "@marijn/find-cluster-break": "^1.0.0"
+ }
+ },
+ "node_modules/@codemirror/view": {
+ "version": "6.38.6",
+ "resolved": "https://registry.npmjs.org/@codemirror/view/-/view-6.38.6.tgz",
+ "integrity": "sha512-qiS0z1bKs5WOvHIAC0Cybmv4AJSkAXgX5aD6Mqd2epSLlVJsQl8NG23jCVouIgkh4All/mrbdsf2UOLFnJw0tw==",
+ "dev": true,
+ "license": "MIT",
+ "peer": true,
+ "dependencies": {
+ "@codemirror/state": "^6.5.0",
+ "crelt": "^1.0.6",
+ "style-mod": "^4.1.0",
+ "w3c-keyname": "^2.2.4"
+ }
+ },
+ "node_modules/@esbuild/darwin-arm64": {
+ "version": "0.25.12",
+ "resolved": "https://registry.npmjs.org/@esbuild/darwin-arm64/-/darwin-arm64-0.25.12.tgz",
+ "integrity": "sha512-N3zl+lxHCifgIlcMUP5016ESkeQjLj/959RxxNYIthIg+CQHInujFuXeWbWMgnTo4cp5XVHqFPmpyu9J65C1Yg==",
+ "cpu": [
+ "arm64"
+ ],
+ "dev": true,
+ "license": "MIT",
+ "optional": true,
+ "os": [
+ "darwin"
+ ],
+ "engines": {
+ "node": ">=18"
+ }
+ },
+ "node_modules/@marijn/find-cluster-break": {
+ "version": "1.0.2",
+ "resolved": "https://registry.npmjs.org/@marijn/find-cluster-break/-/find-cluster-break-1.0.2.tgz",
+ "integrity": "sha512-l0h88YhZFyKdXIFNfSWpyjStDjGHwZ/U7iobcK1cQQD8sejsONdQtTVU+1wVN1PBw40PiiHB1vA5S7VTfQiP9g==",
+ "dev": true,
+ "license": "MIT",
+ "peer": true
+ },
+ "node_modules/@types/codemirror": {
+ "version": "5.60.8",
+ "resolved": "https://registry.npmjs.org/@types/codemirror/-/codemirror-5.60.8.tgz",
+ "integrity": "sha512-VjFgDF/eB+Aklcy15TtOTLQeMjTo07k7KAjql8OK5Dirr7a6sJY4T1uVBDuTVG9VEmn1uUsohOpYnVfgC6/jyw==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "@types/tern": "*"
+ }
+ },
+ "node_modules/@types/estree": {
+ "version": "1.0.8",
+ "resolved": "https://registry.npmjs.org/@types/estree/-/estree-1.0.8.tgz",
+ "integrity": "sha512-dWHzHa2WqEXI/O1E9OjrocMTKJl2mSrEolh1Iomrv6U+JuNwaHXsXx9bLu5gG7BUWFIN0skIQJQ/L1rIex4X6w==",
+ "dev": true,
+ "license": "MIT"
+ },
+ "node_modules/@types/node": {
+ "version": "20.19.37",
+ "resolved": "https://registry.npmjs.org/@types/node/-/node-20.19.37.tgz",
+ "integrity": "sha512-8kzdPJ3FsNsVIurqBs7oodNnCEVbni9yUEkaHbgptDACOPW04jimGagZ51E6+lXUwJjgnBw+hyko/lkFWCldqw==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "undici-types": "~6.21.0"
+ }
+ },
+ "node_modules/@types/qrcode": {
+ "version": "1.5.6",
+ "resolved": "https://registry.npmjs.org/@types/qrcode/-/qrcode-1.5.6.tgz",
+ "integrity": "sha512-te7NQcV2BOvdj2b1hCAHzAoMNuj65kNBMz0KBaxM6c3VGBOhU0dURQKOtH8CFNI/dsKkwlv32p26qYQTWoB5bw==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "@types/node": "*"
+ }
+ },
+ "node_modules/@types/tern": {
+ "version": "0.23.9",
+ "resolved": "https://registry.npmjs.org/@types/tern/-/tern-0.23.9.tgz",
+ "integrity": "sha512-ypzHFE/wBzh+BlH6rrBgS5I/Z7RD21pGhZ2rltb/+ZrVM1awdZwjx7hE5XfuYgHWk9uvV5HLZN3SloevCAp3Bw==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "@types/estree": "*"
+ }
+ },
+ "node_modules/builtin-modules": {
+ "version": "4.0.0",
+ "resolved": "https://registry.npmjs.org/builtin-modules/-/builtin-modules-4.0.0.tgz",
+ "integrity": "sha512-p1n8zyCkt1BVrKNFymOHjcDSAl7oq/gUvfgULv2EblgpPVQlQr9yHnWjg9IJ2MhfwPqiYqMMrr01OY7yQoK2yA==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": ">=18.20"
+ },
+ "funding": {
+ "url": "https://github.com/sponsors/sindresorhus"
+ }
+ },
+ "node_modules/crelt": {
+ "version": "1.0.6",
+ "resolved": "https://registry.npmjs.org/crelt/-/crelt-1.0.6.tgz",
+ "integrity": "sha512-VQ2MBenTq1fWZUH9DJNGti7kKv6EeAuYr3cLwxUWhIu1baTaXh4Ib5W2CqHVqib4/MqbYGJqiL3Zb8GJZr3l4g==",
+ "dev": true,
+ "license": "MIT",
+ "peer": true
+ },
+ "node_modules/esbuild": {
+ "version": "0.25.12",
+ "resolved": "https://registry.npmjs.org/esbuild/-/esbuild-0.25.12.tgz",
+ "integrity": "sha512-bbPBYYrtZbkt6Os6FiTLCTFxvq4tt3JKall1vRwshA3fdVztsLAatFaZobhkBC8/BrPetoa0oksYoKXoG4ryJg==",
+ "dev": true,
+ "hasInstallScript": true,
+ "license": "MIT",
+ "bin": {
+ "esbuild": "bin/esbuild"
+ },
+ "engines": {
+ "node": ">=18"
+ },
+ "optionalDependencies": {
+ "@esbuild/aix-ppc64": "0.25.12",
+ "@esbuild/android-arm": "0.25.12",
+ "@esbuild/android-arm64": "0.25.12",
+ "@esbuild/android-x64": "0.25.12",
+ "@esbuild/darwin-arm64": "0.25.12",
+ "@esbuild/darwin-x64": "0.25.12",
+ "@esbuild/freebsd-arm64": "0.25.12",
+ "@esbuild/freebsd-x64": "0.25.12",
+ "@esbuild/linux-arm": "0.25.12",
+ "@esbuild/linux-arm64": "0.25.12",
+ "@esbuild/linux-ia32": "0.25.12",
+ "@esbuild/linux-loong64": "0.25.12",
+ "@esbuild/linux-mips64el": "0.25.12",
+ "@esbuild/linux-ppc64": "0.25.12",
+ "@esbuild/linux-riscv64": "0.25.12",
+ "@esbuild/linux-s390x": "0.25.12",
+ "@esbuild/linux-x64": "0.25.12",
+ "@esbuild/netbsd-arm64": "0.25.12",
+ "@esbuild/netbsd-x64": "0.25.12",
+ "@esbuild/openbsd-arm64": "0.25.12",
+ "@esbuild/openbsd-x64": "0.25.12",
+ "@esbuild/openharmony-arm64": "0.25.12",
+ "@esbuild/sunos-x64": "0.25.12",
+ "@esbuild/win32-arm64": "0.25.12",
+ "@esbuild/win32-ia32": "0.25.12",
+ "@esbuild/win32-x64": "0.25.12"
+ }
+ },
+ "node_modules/moment": {
+ "version": "2.29.4",
+ "resolved": "https://registry.npmjs.org/moment/-/moment-2.29.4.tgz",
+ "integrity": "sha512-5LC9SOxjSc2HF6vO2CyuTDNivEdoz2IvyJJGj6X8DJ0eFyfszE0QiEd+iXmBvUP3WHxSjFH/vIsA0EN00cgr8w==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": "*"
+ }
+ },
+ "node_modules/obsidian": {
+ "version": "1.12.3",
+ "resolved": "https://registry.npmjs.org/obsidian/-/obsidian-1.12.3.tgz",
+ "integrity": "sha512-HxWqe763dOqzXjnNiHmAJTRERN8KILBSqxDSEqbeSr7W8R8Jxezzbca+nz1LiiqXnMpM8lV2jzAezw3CZ4xNUw==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "@types/codemirror": "5.60.8",
+ "moment": "2.29.4"
+ },
+ "peerDependencies": {
+ "@codemirror/state": "6.5.0",
+ "@codemirror/view": "6.38.6"
+ }
+ },
+ "node_modules/style-mod": {
+ "version": "4.1.3",
+ "resolved": "https://registry.npmjs.org/style-mod/-/style-mod-4.1.3.tgz",
+ "integrity": "sha512-i/n8VsZydrugj3Iuzll8+x/00GH2vnYsk1eomD8QiRrSAeW6ItbCQDtfXCeJHd0iwiNagqjQkvpvREEPtW3IoQ==",
+ "dev": true,
+ "license": "MIT",
+ "peer": true
+ },
+ "node_modules/tslib": {
+ "version": "2.8.1",
+ "resolved": "https://registry.npmjs.org/tslib/-/tslib-2.8.1.tgz",
+ "integrity": "sha512-oJFu94HQb+KVduSUQL7wnpmqnfmLsOA/nAh6b6EH0wCEoK0/mPeXU6c3wKDV83MkOuHPRHtSXKKU99IBazS/2w==",
+ "dev": true,
+ "license": "0BSD"
+ },
+ "node_modules/typescript": {
+ "version": "5.9.3",
+ "resolved": "https://registry.npmjs.org/typescript/-/typescript-5.9.3.tgz",
+ "integrity": "sha512-jl1vZzPDinLr9eUt3J/t7V6FgNEw9QjvBPdysz9KfQDD41fQrC2Y4vKQdiaUpFT4bXlb1RHhLpp8wtm6M5TgSw==",
+ "dev": true,
+ "license": "Apache-2.0",
+ "bin": {
+ "tsc": "bin/tsc",
+ "tsserver": "bin/tsserver"
+ },
+ "engines": {
+ "node": ">=14.17"
+ }
+ },
+ "node_modules/undici-types": {
+ "version": "6.21.0",
+ "resolved": "https://registry.npmjs.org/undici-types/-/undici-types-6.21.0.tgz",
+ "integrity": "sha512-iwDZqg0QAGrg9Rav5H4n0M64c3mkR59cJ6wQp+7C4nI0gsmExaedaYLNO44eT4AtBBwjbTiGPMlt2Md0T9H9JQ==",
+ "dev": true,
+ "license": "MIT"
+ },
+ "node_modules/w3c-keyname": {
+ "version": "2.2.8",
+ "resolved": "https://registry.npmjs.org/w3c-keyname/-/w3c-keyname-2.2.8.tgz",
+ "integrity": "sha512-dpojBhNsCNN7T82Tm7k26A6G9ML3NkhDsnw9n/eoxSRlVBB4CEtIQ/KTCLI2Fwf3ataSXRhYFkQi3SlnFwPvPQ==",
+ "dev": true,
+ "license": "MIT",
+ "peer": true
+ }
+ }
+}
diff --git a/node_modules/@codemirror/state/.github/workflows/dispatch.yml b/node_modules/@codemirror/state/.github/workflows/dispatch.yml
new file mode 100644
index 0000000..d050072
--- /dev/null
+++ b/node_modules/@codemirror/state/.github/workflows/dispatch.yml
@@ -0,0 +1,16 @@
+name: Trigger CI
+on: push
+
+jobs:
+ build:
+ name: Dispatch to main repo
+ runs-on: ubuntu-latest
+ steps:
+ - name: Emit repository_dispatch
+ uses: mvasigh/dispatch-action@main
+ with:
+ # You should create a personal access token and store it in your repository
+ token: ${{ secrets.DISPATCH_AUTH }}
+ repo: dev
+ owner: codemirror
+ event_type: push
diff --git a/node_modules/@codemirror/state/CHANGELOG.md b/node_modules/@codemirror/state/CHANGELOG.md
new file mode 100644
index 0000000..636d52b
--- /dev/null
+++ b/node_modules/@codemirror/state/CHANGELOG.md
@@ -0,0 +1,274 @@
+## 6.5.0 (2024-12-09)
+
+### New features
+
+`RangeSet.compare` now supports a `boundChange` callback that is called when there's a change in the way ranges are split.
+
+## 6.4.1 (2024-02-19)
+
+### Bug fixes
+
+Fix an issue that caused widgets at the end of a mark decoration to be rendered in their own separate mark DOM element.
+
+## 6.4.0 (2023-12-28)
+
+### Bug fixes
+
+When multiple ranges in a single range set overlap, put the smaller ones inside the bigger ones, so that overlapping decorations don't break up each other's elements when coming from the same source.
+
+### New features
+
+Selection and selection range `eq` methods now support an optional argument that makes them also compare by cursor associativity.
+
+The `RangeSet.join` function can be used to join multiple range sets together.
+
+## 6.3.3 (2023-12-06)
+
+### Bug fixes
+
+Fix an issue where `Text.slice` and `Text.replace` could return objects with incorrect `length` when the given `from`/`to` values were out of range for the text.
+
+## 6.3.2 (2023-11-27)
+
+### Bug fixes
+
+Make sure transactions cannot add multiple selections when `allowMultipleSelections` is false.
+
+Fix a bug that caused `Text.iterLines` to not return empty lines at the end of the iterated ranges.
+
+## 6.3.1 (2023-10-18)
+
+### Bug fixes
+
+Give the tag property on `FacetReader` the type of the output type parameter to force TypeScript to infer the proper type when converting from `Facet` to `FacetReader`.
+
+## 6.3.0 (2023-10-12)
+
+### New features
+
+The new `FacetReader` type provides a way to export a read-only handle to a `Facet`.
+
+## 6.2.1 (2023-05-23)
+
+### Bug fixes
+
+Fix an issue that could cause `RangeSet.compare` to miss changes in the set of active ranges around a point range.
+
+## 6.2.0 (2022-12-26)
+
+### New features
+
+`EditorSelection.range` now accepts an optional 4th argument to specify the bidi level of the range's head position.
+
+## 6.1.4 (2022-11-15)
+
+### Bug fixes
+
+Fix a bug that caused the `openStart` value passed to span iterators to be incorrect around widgets in some circumstances.
+
+## 6.1.3 (2022-11-10)
+
+### Bug fixes
+
+Avoid unnecessary calls to computed facet getters when a state is reconfigured but no dependencies of the computed facet change.
+
+Fix an infinite loop in `RangeSet.eq` when the `to` parameter isn't given.
+
+## 6.1.2 (2022-09-21)
+
+### Bug fixes
+
+Fix an issue where, when multiple transaction extenders took effect, only the highest-precedence one was actually included in the transaction.
+
+## 6.1.1 (2022-08-03)
+
+### Bug fixes
+
+Fix a bug in range set span iteration that would cause decorations to be inappropriately split in some situations.
+
+## 6.1.0 (2022-06-30)
+
+### Bug fixes
+
+Refine change mapping to preserve insertions made by concurrent changes.
+
+### New features
+
+The `enables` option to `Facet.define` may now take a function, which will be called with the facet value to create the extensions.
+
+## 6.0.1 (2022-06-17)
+
+### Bug fixes
+
+Fix a problem that caused effects' `map` methods to be called with an incorrect change set when filtering changes.
+
+## 6.0.0 (2022-06-08)
+
+### Breaking changes
+
+Update dependencies to 6.0.0
+
+## 0.20.1 (2022-06-02)
+
+### New features
+
+`EditorView.phrase` now accepts additional arguments, which it will interpolate into the phrase in the place of `$` markers.
+
+## 0.20.0 (2022-04-20)
+
+### Breaking changes
+
+The deprecated precedence names `fallback`, `extend`, and `override` were removed from the library.
+
+### Bug fixes
+
+Fix a bug where, if an extension value occurs multiple times, its lowest, rather than highest precedence is used.
+
+Fix an issue where facets with computed inputs would unneccesarily have their outputs recreated on state reconfiguration.
+
+Fix a bug in the order in which new values for state fields and facets were computed, which could cause dynamic facets to hold the wrong value in some situations.
+
+### New features
+
+The exports from @codemirror/rangeset now live in this package.
+
+The exports from @codemirror/text now live in this package.
+
+## 0.19.9 (2022-02-16)
+
+### Bug fixes
+
+Mapping a non-empty selection range now always puts any newly inserted text on the sides of the range outside of the mapped version.
+
+## 0.19.8 (2022-02-15)
+
+### Bug fixes
+
+Fix a bug where facet values with computed inputs could incorrectly retain their old value on reconfiguration.
+
+## 0.19.7 (2022-02-11)
+
+### Bug fixes
+
+Avoid recomputing facets on state reconfiguration if that facet's inputs stayed precisely the same.
+
+Selection ranges created with `EditorSelection.range` will now have an assoc pointing at their anchor, when non-empty.
+
+## 0.19.6 (2021-11-19)
+
+### Bug fixes
+
+Fix a bug that caused facet compare functions to be called with an invalid value in some situations.
+
+Fix a bug that caused dynamic facet values to be incorrectly kept unchanged when reconfiguration changed one of their dependencies.
+
+## 0.19.5 (2021-11-10)
+
+### Bug fixes
+
+Fix a bug that would cause dynamic facet values influenced by a state reconfiguration to not properly recompute.
+
+## 0.19.4 (2021-11-05)
+
+### Bug fixes
+
+When reconfiguring a state, effects from the reconfiguring transaction can now be seen by newly added state fields.
+
+## 0.19.3 (2021-11-03)
+
+### New features
+
+The precedence levels (under `Prec`) now have more generic names, because their 'meaningful' names were entirely inappropriate in many situations.
+
+## 0.19.2 (2021-09-13)
+
+### New features
+
+The editor state now has a `readOnly` property with a matching facet to control its value.
+
+## 0.19.1 (2021-08-15)
+
+### Bug fixes
+
+Fix a bug where `wordAt` never returned a useful result.
+
+## 0.19.0 (2021-08-11)
+
+### Breaking changes
+
+User event strings now work differently—the events emitted by the core packages follow a different system, and hierarchical event tags can be created by separating the words with dots.
+
+### New features
+
+`languageDataAt` now takes an optional `side` argument to specificy which side of the position you're interested in.
+
+It is now possible to add a user event annotation with a direct `userEvent` property on a transaction spec.
+
+Transactions now have an `isUserEvent` method that can be used to check if it is (a subtype of) some user event type.
+
+## 0.18.7 (2021-05-04)
+
+### Bug fixes
+
+Fix an issue where state fields might be initialized with a state that they aren't actually part of during reconfiguration.
+
+## 0.18.6 (2021-04-12)
+
+### New features
+
+The new `EditorState.wordAt` method finds the word at a given position.
+
+## 0.18.5 (2021-04-08)
+
+### Bug fixes
+
+Fix an issue in the compiled output that would break the code when minified with terser.
+
+## 0.18.4 (2021-04-06)
+
+### New features
+
+The new `Transaction.remote` annotation can be used to mark and recognize transactions created by other actors.
+
+## 0.18.3 (2021-03-23)
+
+### New features
+
+The `ChangeDesc` class now has `toJSON` and `fromJSON` methods.
+
+## 0.18.2 (2021-03-14)
+
+### Bug fixes
+
+Fix unintended ES2020 output (the package contains ES6 code again).
+
+## 0.18.1 (2021-03-10)
+
+### New features
+
+The new `Compartment.get` method can be used to get the content of a compartment in a given state.
+
+## 0.18.0 (2021-03-03)
+
+### Breaking changes
+
+`tagExtension` and the `reconfigure` transaction spec property have been replaced with the concept of configuration compartments and reconfiguration effects (see `Compartment`, `StateEffect.reconfigure`, and `StateEffect.appendConfig`).
+
+## 0.17.2 (2021-02-19)
+
+### New features
+
+`EditorSelection.map` and `SelectionRange.map` now take an optional second argument to indicate which direction to map to.
+
+## 0.17.1 (2021-01-06)
+
+### New features
+
+The package now also exports a CommonJS module.
+
+## 0.17.0 (2020-12-29)
+
+### Breaking changes
+
+First numbered release.
+
diff --git a/node_modules/@codemirror/state/LICENSE b/node_modules/@codemirror/state/LICENSE
new file mode 100644
index 0000000..9a91f48
--- /dev/null
+++ b/node_modules/@codemirror/state/LICENSE
@@ -0,0 +1,21 @@
+MIT License
+
+Copyright (C) 2018-2021 by Marijn Haverbeke and others
+
+Permission is hereby granted, free of charge, to any person obtaining a copy
+of this software and associated documentation files (the "Software"), to deal
+in the Software without restriction, including without limitation the rights
+to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
+copies of the Software, and to permit persons to whom the Software is
+furnished to do so, subject to the following conditions:
+
+The above copyright notice and this permission notice shall be included in
+all copies or substantial portions of the Software.
+
+THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
+IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
+FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
+AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
+LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
+OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
+THE SOFTWARE.
diff --git a/node_modules/@codemirror/state/README.md b/node_modules/@codemirror/state/README.md
new file mode 100644
index 0000000..5f1a325
--- /dev/null
+++ b/node_modules/@codemirror/state/README.md
@@ -0,0 +1,18 @@
+# @codemirror/state [](https://www.npmjs.org/package/@codemirror/state)
+
+[ [**WEBSITE**](https://codemirror.net/) | [**DOCS**](https://codemirror.net/docs/ref/#state) | [**ISSUES**](https://github.com/codemirror/dev/issues) | [**FORUM**](https://discuss.codemirror.net/c/next/) | [**CHANGELOG**](https://github.com/codemirror/state/blob/main/CHANGELOG.md) ]
+
+This package implements the editor state data structures for the
+[CodeMirror](https://codemirror.net/) code editor.
+
+The [project page](https://codemirror.net/) has more information, a
+number of [examples](https://codemirror.net/examples/) and the
+[documentation](https://codemirror.net/docs/).
+
+This code is released under an
+[MIT license](https://github.com/codemirror/state/tree/main/LICENSE).
+
+We aim to be an inclusive, welcoming community. To make that explicit,
+we have a [code of
+conduct](http://contributor-covenant.org/version/1/1/0/) that applies
+to communication around the project.
diff --git a/node_modules/@codemirror/state/dist/index.cjs b/node_modules/@codemirror/state/dist/index.cjs
new file mode 100644
index 0000000..fbf4421
--- /dev/null
+++ b/node_modules/@codemirror/state/dist/index.cjs
@@ -0,0 +1,3906 @@
+'use strict';
+
+var findClusterBreak$1 = require('@marijn/find-cluster-break');
+
+/**
+The data structure for documents. @nonabstract
+*/
+class Text {
+ /**
+ Get the line description around the given position.
+ */
+ lineAt(pos) {
+ if (pos < 0 || pos > this.length)
+ throw new RangeError(`Invalid position ${pos} in document of length ${this.length}`);
+ return this.lineInner(pos, false, 1, 0);
+ }
+ /**
+ Get the description for the given (1-based) line number.
+ */
+ line(n) {
+ if (n < 1 || n > this.lines)
+ throw new RangeError(`Invalid line number ${n} in ${this.lines}-line document`);
+ return this.lineInner(n, true, 1, 0);
+ }
+ /**
+ Replace a range of the text with the given content.
+ */
+ replace(from, to, text) {
+ [from, to] = clip(this, from, to);
+ let parts = [];
+ this.decompose(0, from, parts, 2 /* Open.To */);
+ if (text.length)
+ text.decompose(0, text.length, parts, 1 /* Open.From */ | 2 /* Open.To */);
+ this.decompose(to, this.length, parts, 1 /* Open.From */);
+ return TextNode.from(parts, this.length - (to - from) + text.length);
+ }
+ /**
+ Append another document to this one.
+ */
+ append(other) {
+ return this.replace(this.length, this.length, other);
+ }
+ /**
+ Retrieve the text between the given points.
+ */
+ slice(from, to = this.length) {
+ [from, to] = clip(this, from, to);
+ let parts = [];
+ this.decompose(from, to, parts, 0);
+ return TextNode.from(parts, to - from);
+ }
+ /**
+ Test whether this text is equal to another instance.
+ */
+ eq(other) {
+ if (other == this)
+ return true;
+ if (other.length != this.length || other.lines != this.lines)
+ return false;
+ let start = this.scanIdentical(other, 1), end = this.length - this.scanIdentical(other, -1);
+ let a = new RawTextCursor(this), b = new RawTextCursor(other);
+ for (let skip = start, pos = start;;) {
+ a.next(skip);
+ b.next(skip);
+ skip = 0;
+ if (a.lineBreak != b.lineBreak || a.done != b.done || a.value != b.value)
+ return false;
+ pos += a.value.length;
+ if (a.done || pos >= end)
+ return true;
+ }
+ }
+ /**
+ Iterate over the text. When `dir` is `-1`, iteration happens
+ from end to start. This will return lines and the breaks between
+ them as separate strings.
+ */
+ iter(dir = 1) { return new RawTextCursor(this, dir); }
+ /**
+ Iterate over a range of the text. When `from` > `to`, the
+ iterator will run in reverse.
+ */
+ iterRange(from, to = this.length) { return new PartialTextCursor(this, from, to); }
+ /**
+ Return a cursor that iterates over the given range of lines,
+ _without_ returning the line breaks between, and yielding empty
+ strings for empty lines.
+
+ When `from` and `to` are given, they should be 1-based line numbers.
+ */
+ iterLines(from, to) {
+ let inner;
+ if (from == null) {
+ inner = this.iter();
+ }
+ else {
+ if (to == null)
+ to = this.lines + 1;
+ let start = this.line(from).from;
+ inner = this.iterRange(start, Math.max(start, to == this.lines + 1 ? this.length : to <= 1 ? 0 : this.line(to - 1).to));
+ }
+ return new LineCursor(inner);
+ }
+ /**
+ Return the document as a string, using newline characters to
+ separate lines.
+ */
+ toString() { return this.sliceString(0); }
+ /**
+ Convert the document to an array of lines (which can be
+ deserialized again via [`Text.of`](https://codemirror.net/6/docs/ref/#state.Text^of)).
+ */
+ toJSON() {
+ let lines = [];
+ this.flatten(lines);
+ return lines;
+ }
+ /**
+ @internal
+ */
+ constructor() { }
+ /**
+ Create a `Text` instance for the given array of lines.
+ */
+ static of(text) {
+ if (text.length == 0)
+ throw new RangeError("A document must have at least one line");
+ if (text.length == 1 && !text[0])
+ return Text.empty;
+ return text.length <= 32 /* Tree.Branch */ ? new TextLeaf(text) : TextNode.from(TextLeaf.split(text, []));
+ }
+}
+// Leaves store an array of line strings. There are always line breaks
+// between these strings. Leaves are limited in size and have to be
+// contained in TextNode instances for bigger documents.
+class TextLeaf extends Text {
+ constructor(text, length = textLength(text)) {
+ super();
+ this.text = text;
+ this.length = length;
+ }
+ get lines() { return this.text.length; }
+ get children() { return null; }
+ lineInner(target, isLine, line, offset) {
+ for (let i = 0;; i++) {
+ let string = this.text[i], end = offset + string.length;
+ if ((isLine ? line : end) >= target)
+ return new Line(offset, end, line, string);
+ offset = end + 1;
+ line++;
+ }
+ }
+ decompose(from, to, target, open) {
+ let text = from <= 0 && to >= this.length ? this
+ : new TextLeaf(sliceText(this.text, from, to), Math.min(to, this.length) - Math.max(0, from));
+ if (open & 1 /* Open.From */) {
+ let prev = target.pop();
+ let joined = appendText(text.text, prev.text.slice(), 0, text.length);
+ if (joined.length <= 32 /* Tree.Branch */) {
+ target.push(new TextLeaf(joined, prev.length + text.length));
+ }
+ else {
+ let mid = joined.length >> 1;
+ target.push(new TextLeaf(joined.slice(0, mid)), new TextLeaf(joined.slice(mid)));
+ }
+ }
+ else {
+ target.push(text);
+ }
+ }
+ replace(from, to, text) {
+ if (!(text instanceof TextLeaf))
+ return super.replace(from, to, text);
+ [from, to] = clip(this, from, to);
+ let lines = appendText(this.text, appendText(text.text, sliceText(this.text, 0, from)), to);
+ let newLen = this.length + text.length - (to - from);
+ if (lines.length <= 32 /* Tree.Branch */)
+ return new TextLeaf(lines, newLen);
+ return TextNode.from(TextLeaf.split(lines, []), newLen);
+ }
+ sliceString(from, to = this.length, lineSep = "\n") {
+ [from, to] = clip(this, from, to);
+ let result = "";
+ for (let pos = 0, i = 0; pos <= to && i < this.text.length; i++) {
+ let line = this.text[i], end = pos + line.length;
+ if (pos > from && i)
+ result += lineSep;
+ if (from < end && to > pos)
+ result += line.slice(Math.max(0, from - pos), to - pos);
+ pos = end + 1;
+ }
+ return result;
+ }
+ flatten(target) {
+ for (let line of this.text)
+ target.push(line);
+ }
+ scanIdentical() { return 0; }
+ static split(text, target) {
+ let part = [], len = -1;
+ for (let line of text) {
+ part.push(line);
+ len += line.length + 1;
+ if (part.length == 32 /* Tree.Branch */) {
+ target.push(new TextLeaf(part, len));
+ part = [];
+ len = -1;
+ }
+ }
+ if (len > -1)
+ target.push(new TextLeaf(part, len));
+ return target;
+ }
+}
+// Nodes provide the tree structure of the `Text` type. They store a
+// number of other nodes or leaves, taking care to balance themselves
+// on changes. There are implied line breaks _between_ the children of
+// a node (but not before the first or after the last child).
+class TextNode extends Text {
+ constructor(children, length) {
+ super();
+ this.children = children;
+ this.length = length;
+ this.lines = 0;
+ for (let child of children)
+ this.lines += child.lines;
+ }
+ lineInner(target, isLine, line, offset) {
+ for (let i = 0;; i++) {
+ let child = this.children[i], end = offset + child.length, endLine = line + child.lines - 1;
+ if ((isLine ? endLine : end) >= target)
+ return child.lineInner(target, isLine, line, offset);
+ offset = end + 1;
+ line = endLine + 1;
+ }
+ }
+ decompose(from, to, target, open) {
+ for (let i = 0, pos = 0; pos <= to && i < this.children.length; i++) {
+ let child = this.children[i], end = pos + child.length;
+ if (from <= end && to >= pos) {
+ let childOpen = open & ((pos <= from ? 1 /* Open.From */ : 0) | (end >= to ? 2 /* Open.To */ : 0));
+ if (pos >= from && end <= to && !childOpen)
+ target.push(child);
+ else
+ child.decompose(from - pos, to - pos, target, childOpen);
+ }
+ pos = end + 1;
+ }
+ }
+ replace(from, to, text) {
+ [from, to] = clip(this, from, to);
+ if (text.lines < this.lines)
+ for (let i = 0, pos = 0; i < this.children.length; i++) {
+ let child = this.children[i], end = pos + child.length;
+ // Fast path: if the change only affects one child and the
+ // child's size remains in the acceptable range, only update
+ // that child
+ if (from >= pos && to <= end) {
+ let updated = child.replace(from - pos, to - pos, text);
+ let totalLines = this.lines - child.lines + updated.lines;
+ if (updated.lines < (totalLines >> (5 /* Tree.BranchShift */ - 1)) &&
+ updated.lines > (totalLines >> (5 /* Tree.BranchShift */ + 1))) {
+ let copy = this.children.slice();
+ copy[i] = updated;
+ return new TextNode(copy, this.length - (to - from) + text.length);
+ }
+ return super.replace(pos, end, updated);
+ }
+ pos = end + 1;
+ }
+ return super.replace(from, to, text);
+ }
+ sliceString(from, to = this.length, lineSep = "\n") {
+ [from, to] = clip(this, from, to);
+ let result = "";
+ for (let i = 0, pos = 0; i < this.children.length && pos <= to; i++) {
+ let child = this.children[i], end = pos + child.length;
+ if (pos > from && i)
+ result += lineSep;
+ if (from < end && to > pos)
+ result += child.sliceString(from - pos, to - pos, lineSep);
+ pos = end + 1;
+ }
+ return result;
+ }
+ flatten(target) {
+ for (let child of this.children)
+ child.flatten(target);
+ }
+ scanIdentical(other, dir) {
+ if (!(other instanceof TextNode))
+ return 0;
+ let length = 0;
+ let [iA, iB, eA, eB] = dir > 0 ? [0, 0, this.children.length, other.children.length]
+ : [this.children.length - 1, other.children.length - 1, -1, -1];
+ for (;; iA += dir, iB += dir) {
+ if (iA == eA || iB == eB)
+ return length;
+ let chA = this.children[iA], chB = other.children[iB];
+ if (chA != chB)
+ return length + chA.scanIdentical(chB, dir);
+ length += chA.length + 1;
+ }
+ }
+ static from(children, length = children.reduce((l, ch) => l + ch.length + 1, -1)) {
+ let lines = 0;
+ for (let ch of children)
+ lines += ch.lines;
+ if (lines < 32 /* Tree.Branch */) {
+ let flat = [];
+ for (let ch of children)
+ ch.flatten(flat);
+ return new TextLeaf(flat, length);
+ }
+ let chunk = Math.max(32 /* Tree.Branch */, lines >> 5 /* Tree.BranchShift */), maxChunk = chunk << 1, minChunk = chunk >> 1;
+ let chunked = [], currentLines = 0, currentLen = -1, currentChunk = [];
+ function add(child) {
+ let last;
+ if (child.lines > maxChunk && child instanceof TextNode) {
+ for (let node of child.children)
+ add(node);
+ }
+ else if (child.lines > minChunk && (currentLines > minChunk || !currentLines)) {
+ flush();
+ chunked.push(child);
+ }
+ else if (child instanceof TextLeaf && currentLines &&
+ (last = currentChunk[currentChunk.length - 1]) instanceof TextLeaf &&
+ child.lines + last.lines <= 32 /* Tree.Branch */) {
+ currentLines += child.lines;
+ currentLen += child.length + 1;
+ currentChunk[currentChunk.length - 1] = new TextLeaf(last.text.concat(child.text), last.length + 1 + child.length);
+ }
+ else {
+ if (currentLines + child.lines > chunk)
+ flush();
+ currentLines += child.lines;
+ currentLen += child.length + 1;
+ currentChunk.push(child);
+ }
+ }
+ function flush() {
+ if (currentLines == 0)
+ return;
+ chunked.push(currentChunk.length == 1 ? currentChunk[0] : TextNode.from(currentChunk, currentLen));
+ currentLen = -1;
+ currentLines = currentChunk.length = 0;
+ }
+ for (let child of children)
+ add(child);
+ flush();
+ return chunked.length == 1 ? chunked[0] : new TextNode(chunked, length);
+ }
+}
+Text.empty = new TextLeaf([""], 0);
+function textLength(text) {
+ let length = -1;
+ for (let line of text)
+ length += line.length + 1;
+ return length;
+}
+function appendText(text, target, from = 0, to = 1e9) {
+ for (let pos = 0, i = 0, first = true; i < text.length && pos <= to; i++) {
+ let line = text[i], end = pos + line.length;
+ if (end >= from) {
+ if (end > to)
+ line = line.slice(0, to - pos);
+ if (pos < from)
+ line = line.slice(from - pos);
+ if (first) {
+ target[target.length - 1] += line;
+ first = false;
+ }
+ else
+ target.push(line);
+ }
+ pos = end + 1;
+ }
+ return target;
+}
+function sliceText(text, from, to) {
+ return appendText(text, [""], from, to);
+}
+class RawTextCursor {
+ constructor(text, dir = 1) {
+ this.dir = dir;
+ this.done = false;
+ this.lineBreak = false;
+ this.value = "";
+ this.nodes = [text];
+ this.offsets = [dir > 0 ? 1 : (text instanceof TextLeaf ? text.text.length : text.children.length) << 1];
+ }
+ nextInner(skip, dir) {
+ this.done = this.lineBreak = false;
+ for (;;) {
+ let last = this.nodes.length - 1;
+ let top = this.nodes[last], offsetValue = this.offsets[last], offset = offsetValue >> 1;
+ let size = top instanceof TextLeaf ? top.text.length : top.children.length;
+ if (offset == (dir > 0 ? size : 0)) {
+ if (last == 0) {
+ this.done = true;
+ this.value = "";
+ return this;
+ }
+ if (dir > 0)
+ this.offsets[last - 1]++;
+ this.nodes.pop();
+ this.offsets.pop();
+ }
+ else if ((offsetValue & 1) == (dir > 0 ? 0 : 1)) {
+ this.offsets[last] += dir;
+ if (skip == 0) {
+ this.lineBreak = true;
+ this.value = "\n";
+ return this;
+ }
+ skip--;
+ }
+ else if (top instanceof TextLeaf) {
+ // Move to the next string
+ let next = top.text[offset + (dir < 0 ? -1 : 0)];
+ this.offsets[last] += dir;
+ if (next.length > Math.max(0, skip)) {
+ this.value = skip == 0 ? next : dir > 0 ? next.slice(skip) : next.slice(0, next.length - skip);
+ return this;
+ }
+ skip -= next.length;
+ }
+ else {
+ let next = top.children[offset + (dir < 0 ? -1 : 0)];
+ if (skip > next.length) {
+ skip -= next.length;
+ this.offsets[last] += dir;
+ }
+ else {
+ if (dir < 0)
+ this.offsets[last]--;
+ this.nodes.push(next);
+ this.offsets.push(dir > 0 ? 1 : (next instanceof TextLeaf ? next.text.length : next.children.length) << 1);
+ }
+ }
+ }
+ }
+ next(skip = 0) {
+ if (skip < 0) {
+ this.nextInner(-skip, (-this.dir));
+ skip = this.value.length;
+ }
+ return this.nextInner(skip, this.dir);
+ }
+}
+class PartialTextCursor {
+ constructor(text, start, end) {
+ this.value = "";
+ this.done = false;
+ this.cursor = new RawTextCursor(text, start > end ? -1 : 1);
+ this.pos = start > end ? text.length : 0;
+ this.from = Math.min(start, end);
+ this.to = Math.max(start, end);
+ }
+ nextInner(skip, dir) {
+ if (dir < 0 ? this.pos <= this.from : this.pos >= this.to) {
+ this.value = "";
+ this.done = true;
+ return this;
+ }
+ skip += Math.max(0, dir < 0 ? this.pos - this.to : this.from - this.pos);
+ let limit = dir < 0 ? this.pos - this.from : this.to - this.pos;
+ if (skip > limit)
+ skip = limit;
+ limit -= skip;
+ let { value } = this.cursor.next(skip);
+ this.pos += (value.length + skip) * dir;
+ this.value = value.length <= limit ? value : dir < 0 ? value.slice(value.length - limit) : value.slice(0, limit);
+ this.done = !this.value;
+ return this;
+ }
+ next(skip = 0) {
+ if (skip < 0)
+ skip = Math.max(skip, this.from - this.pos);
+ else if (skip > 0)
+ skip = Math.min(skip, this.to - this.pos);
+ return this.nextInner(skip, this.cursor.dir);
+ }
+ get lineBreak() { return this.cursor.lineBreak && this.value != ""; }
+}
+class LineCursor {
+ constructor(inner) {
+ this.inner = inner;
+ this.afterBreak = true;
+ this.value = "";
+ this.done = false;
+ }
+ next(skip = 0) {
+ let { done, lineBreak, value } = this.inner.next(skip);
+ if (done && this.afterBreak) {
+ this.value = "";
+ this.afterBreak = false;
+ }
+ else if (done) {
+ this.done = true;
+ this.value = "";
+ }
+ else if (lineBreak) {
+ if (this.afterBreak) {
+ this.value = "";
+ }
+ else {
+ this.afterBreak = true;
+ this.next();
+ }
+ }
+ else {
+ this.value = value;
+ this.afterBreak = false;
+ }
+ return this;
+ }
+ get lineBreak() { return false; }
+}
+if (typeof Symbol != "undefined") {
+ Text.prototype[Symbol.iterator] = function () { return this.iter(); };
+ RawTextCursor.prototype[Symbol.iterator] = PartialTextCursor.prototype[Symbol.iterator] =
+ LineCursor.prototype[Symbol.iterator] = function () { return this; };
+}
+/**
+This type describes a line in the document. It is created
+on-demand when lines are [queried](https://codemirror.net/6/docs/ref/#state.Text.lineAt).
+*/
+class Line {
+ /**
+ @internal
+ */
+ constructor(
+ /**
+ The position of the start of the line.
+ */
+ from,
+ /**
+ The position at the end of the line (_before_ the line break,
+ or at the end of document for the last line).
+ */
+ to,
+ /**
+ This line's line number (1-based).
+ */
+ number,
+ /**
+ The line's content.
+ */
+ text) {
+ this.from = from;
+ this.to = to;
+ this.number = number;
+ this.text = text;
+ }
+ /**
+ The length of the line (not including any line break after it).
+ */
+ get length() { return this.to - this.from; }
+}
+function clip(text, from, to) {
+ from = Math.max(0, Math.min(text.length, from));
+ return [from, Math.max(from, Math.min(text.length, to))];
+}
+
+/**
+Returns a next grapheme cluster break _after_ (not equal to)
+`pos`, if `forward` is true, or before otherwise. Returns `pos`
+itself if no further cluster break is available in the string.
+Moves across surrogate pairs, extending characters (when
+`includeExtending` is true), characters joined with zero-width
+joiners, and flag emoji.
+*/
+function findClusterBreak(str, pos, forward = true, includeExtending = true) {
+ return findClusterBreak$1.findClusterBreak(str, pos, forward, includeExtending);
+}
+function surrogateLow(ch) { return ch >= 0xDC00 && ch < 0xE000; }
+function surrogateHigh(ch) { return ch >= 0xD800 && ch < 0xDC00; }
+/**
+Find the code point at the given position in a string (like the
+[`codePointAt`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/String/codePointAt)
+string method).
+*/
+function codePointAt(str, pos) {
+ let code0 = str.charCodeAt(pos);
+ if (!surrogateHigh(code0) || pos + 1 == str.length)
+ return code0;
+ let code1 = str.charCodeAt(pos + 1);
+ if (!surrogateLow(code1))
+ return code0;
+ return ((code0 - 0xd800) << 10) + (code1 - 0xdc00) + 0x10000;
+}
+/**
+Given a Unicode codepoint, return the JavaScript string that
+respresents it (like
+[`String.fromCodePoint`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/String/fromCodePoint)).
+*/
+function fromCodePoint(code) {
+ if (code <= 0xffff)
+ return String.fromCharCode(code);
+ code -= 0x10000;
+ return String.fromCharCode((code >> 10) + 0xd800, (code & 1023) + 0xdc00);
+}
+/**
+The amount of positions a character takes up in a JavaScript string.
+*/
+function codePointSize(code) { return code < 0x10000 ? 1 : 2; }
+
+const DefaultSplit = /\r\n?|\n/;
+/**
+Distinguishes different ways in which positions can be mapped.
+*/
+exports.MapMode = void 0;
+(function (MapMode) {
+ /**
+ Map a position to a valid new position, even when its context
+ was deleted.
+ */
+ MapMode[MapMode["Simple"] = 0] = "Simple";
+ /**
+ Return null if deletion happens across the position.
+ */
+ MapMode[MapMode["TrackDel"] = 1] = "TrackDel";
+ /**
+ Return null if the character _before_ the position is deleted.
+ */
+ MapMode[MapMode["TrackBefore"] = 2] = "TrackBefore";
+ /**
+ Return null if the character _after_ the position is deleted.
+ */
+ MapMode[MapMode["TrackAfter"] = 3] = "TrackAfter";
+})(exports.MapMode || (exports.MapMode = {}));
+/**
+A change description is a variant of [change set](https://codemirror.net/6/docs/ref/#state.ChangeSet)
+that doesn't store the inserted text. As such, it can't be
+applied, but is cheaper to store and manipulate.
+*/
+class ChangeDesc {
+ // Sections are encoded as pairs of integers. The first is the
+ // length in the current document, and the second is -1 for
+ // unaffected sections, and the length of the replacement content
+ // otherwise. So an insertion would be (0, n>0), a deletion (n>0,
+ // 0), and a replacement two positive numbers.
+ /**
+ @internal
+ */
+ constructor(
+ /**
+ @internal
+ */
+ sections) {
+ this.sections = sections;
+ }
+ /**
+ The length of the document before the change.
+ */
+ get length() {
+ let result = 0;
+ for (let i = 0; i < this.sections.length; i += 2)
+ result += this.sections[i];
+ return result;
+ }
+ /**
+ The length of the document after the change.
+ */
+ get newLength() {
+ let result = 0;
+ for (let i = 0; i < this.sections.length; i += 2) {
+ let ins = this.sections[i + 1];
+ result += ins < 0 ? this.sections[i] : ins;
+ }
+ return result;
+ }
+ /**
+ False when there are actual changes in this set.
+ */
+ get empty() { return this.sections.length == 0 || this.sections.length == 2 && this.sections[1] < 0; }
+ /**
+ Iterate over the unchanged parts left by these changes. `posA`
+ provides the position of the range in the old document, `posB`
+ the new position in the changed document.
+ */
+ iterGaps(f) {
+ for (let i = 0, posA = 0, posB = 0; i < this.sections.length;) {
+ let len = this.sections[i++], ins = this.sections[i++];
+ if (ins < 0) {
+ f(posA, posB, len);
+ posB += len;
+ }
+ else {
+ posB += ins;
+ }
+ posA += len;
+ }
+ }
+ /**
+ Iterate over the ranges changed by these changes. (See
+ [`ChangeSet.iterChanges`](https://codemirror.net/6/docs/ref/#state.ChangeSet.iterChanges) for a
+ variant that also provides you with the inserted text.)
+ `fromA`/`toA` provides the extent of the change in the starting
+ document, `fromB`/`toB` the extent of the replacement in the
+ changed document.
+
+ When `individual` is true, adjacent changes (which are kept
+ separate for [position mapping](https://codemirror.net/6/docs/ref/#state.ChangeDesc.mapPos)) are
+ reported separately.
+ */
+ iterChangedRanges(f, individual = false) {
+ iterChanges(this, f, individual);
+ }
+ /**
+ Get a description of the inverted form of these changes.
+ */
+ get invertedDesc() {
+ let sections = [];
+ for (let i = 0; i < this.sections.length;) {
+ let len = this.sections[i++], ins = this.sections[i++];
+ if (ins < 0)
+ sections.push(len, ins);
+ else
+ sections.push(ins, len);
+ }
+ return new ChangeDesc(sections);
+ }
+ /**
+ Compute the combined effect of applying another set of changes
+ after this one. The length of the document after this set should
+ match the length before `other`.
+ */
+ composeDesc(other) { return this.empty ? other : other.empty ? this : composeSets(this, other); }
+ /**
+ Map this description, which should start with the same document
+ as `other`, over another set of changes, so that it can be
+ applied after it. When `before` is true, map as if the changes
+ in `this` happened before the ones in `other`.
+ */
+ mapDesc(other, before = false) { return other.empty ? this : mapSet(this, other, before); }
+ mapPos(pos, assoc = -1, mode = exports.MapMode.Simple) {
+ let posA = 0, posB = 0;
+ for (let i = 0; i < this.sections.length;) {
+ let len = this.sections[i++], ins = this.sections[i++], endA = posA + len;
+ if (ins < 0) {
+ if (endA > pos)
+ return posB + (pos - posA);
+ posB += len;
+ }
+ else {
+ if (mode != exports.MapMode.Simple && endA >= pos &&
+ (mode == exports.MapMode.TrackDel && posA < pos && endA > pos ||
+ mode == exports.MapMode.TrackBefore && posA < pos ||
+ mode == exports.MapMode.TrackAfter && endA > pos))
+ return null;
+ if (endA > pos || endA == pos && assoc < 0 && !len)
+ return pos == posA || assoc < 0 ? posB : posB + ins;
+ posB += ins;
+ }
+ posA = endA;
+ }
+ if (pos > posA)
+ throw new RangeError(`Position ${pos} is out of range for changeset of length ${posA}`);
+ return posB;
+ }
+ /**
+ Check whether these changes touch a given range. When one of the
+ changes entirely covers the range, the string `"cover"` is
+ returned.
+ */
+ touchesRange(from, to = from) {
+ for (let i = 0, pos = 0; i < this.sections.length && pos <= to;) {
+ let len = this.sections[i++], ins = this.sections[i++], end = pos + len;
+ if (ins >= 0 && pos <= to && end >= from)
+ return pos < from && end > to ? "cover" : true;
+ pos = end;
+ }
+ return false;
+ }
+ /**
+ @internal
+ */
+ toString() {
+ let result = "";
+ for (let i = 0; i < this.sections.length;) {
+ let len = this.sections[i++], ins = this.sections[i++];
+ result += (result ? " " : "") + len + (ins >= 0 ? ":" + ins : "");
+ }
+ return result;
+ }
+ /**
+ Serialize this change desc to a JSON-representable value.
+ */
+ toJSON() { return this.sections; }
+ /**
+ Create a change desc from its JSON representation (as produced
+ by [`toJSON`](https://codemirror.net/6/docs/ref/#state.ChangeDesc.toJSON).
+ */
+ static fromJSON(json) {
+ if (!Array.isArray(json) || json.length % 2 || json.some(a => typeof a != "number"))
+ throw new RangeError("Invalid JSON representation of ChangeDesc");
+ return new ChangeDesc(json);
+ }
+ /**
+ @internal
+ */
+ static create(sections) { return new ChangeDesc(sections); }
+}
+/**
+A change set represents a group of modifications to a document. It
+stores the document length, and can only be applied to documents
+with exactly that length.
+*/
+class ChangeSet extends ChangeDesc {
+ constructor(sections,
+ /**
+ @internal
+ */
+ inserted) {
+ super(sections);
+ this.inserted = inserted;
+ }
+ /**
+ Apply the changes to a document, returning the modified
+ document.
+ */
+ apply(doc) {
+ if (this.length != doc.length)
+ throw new RangeError("Applying change set to a document with the wrong length");
+ iterChanges(this, (fromA, toA, fromB, _toB, text) => doc = doc.replace(fromB, fromB + (toA - fromA), text), false);
+ return doc;
+ }
+ mapDesc(other, before = false) { return mapSet(this, other, before, true); }
+ /**
+ Given the document as it existed _before_ the changes, return a
+ change set that represents the inverse of this set, which could
+ be used to go from the document created by the changes back to
+ the document as it existed before the changes.
+ */
+ invert(doc) {
+ let sections = this.sections.slice(), inserted = [];
+ for (let i = 0, pos = 0; i < sections.length; i += 2) {
+ let len = sections[i], ins = sections[i + 1];
+ if (ins >= 0) {
+ sections[i] = ins;
+ sections[i + 1] = len;
+ let index = i >> 1;
+ while (inserted.length < index)
+ inserted.push(Text.empty);
+ inserted.push(len ? doc.slice(pos, pos + len) : Text.empty);
+ }
+ pos += len;
+ }
+ return new ChangeSet(sections, inserted);
+ }
+ /**
+ Combine two subsequent change sets into a single set. `other`
+ must start in the document produced by `this`. If `this` goes
+ `docA` → `docB` and `other` represents `docB` → `docC`, the
+ returned value will represent the change `docA` → `docC`.
+ */
+ compose(other) { return this.empty ? other : other.empty ? this : composeSets(this, other, true); }
+ /**
+ Given another change set starting in the same document, maps this
+ change set over the other, producing a new change set that can be
+ applied to the document produced by applying `other`. When
+ `before` is `true`, order changes as if `this` comes before
+ `other`, otherwise (the default) treat `other` as coming first.
+
+ Given two changes `A` and `B`, `A.compose(B.map(A))` and
+ `B.compose(A.map(B, true))` will produce the same document. This
+ provides a basic form of [operational
+ transformation](https://en.wikipedia.org/wiki/Operational_transformation),
+ and can be used for collaborative editing.
+ */
+ map(other, before = false) { return other.empty ? this : mapSet(this, other, before, true); }
+ /**
+ Iterate over the changed ranges in the document, calling `f` for
+ each, with the range in the original document (`fromA`-`toA`)
+ and the range that replaces it in the new document
+ (`fromB`-`toB`).
+
+ When `individual` is true, adjacent changes are reported
+ separately.
+ */
+ iterChanges(f, individual = false) {
+ iterChanges(this, f, individual);
+ }
+ /**
+ Get a [change description](https://codemirror.net/6/docs/ref/#state.ChangeDesc) for this change
+ set.
+ */
+ get desc() { return ChangeDesc.create(this.sections); }
+ /**
+ @internal
+ */
+ filter(ranges) {
+ let resultSections = [], resultInserted = [], filteredSections = [];
+ let iter = new SectionIter(this);
+ done: for (let i = 0, pos = 0;;) {
+ let next = i == ranges.length ? 1e9 : ranges[i++];
+ while (pos < next || pos == next && iter.len == 0) {
+ if (iter.done)
+ break done;
+ let len = Math.min(iter.len, next - pos);
+ addSection(filteredSections, len, -1);
+ let ins = iter.ins == -1 ? -1 : iter.off == 0 ? iter.ins : 0;
+ addSection(resultSections, len, ins);
+ if (ins > 0)
+ addInsert(resultInserted, resultSections, iter.text);
+ iter.forward(len);
+ pos += len;
+ }
+ let end = ranges[i++];
+ while (pos < end) {
+ if (iter.done)
+ break done;
+ let len = Math.min(iter.len, end - pos);
+ addSection(resultSections, len, -1);
+ addSection(filteredSections, len, iter.ins == -1 ? -1 : iter.off == 0 ? iter.ins : 0);
+ iter.forward(len);
+ pos += len;
+ }
+ }
+ return { changes: new ChangeSet(resultSections, resultInserted),
+ filtered: ChangeDesc.create(filteredSections) };
+ }
+ /**
+ Serialize this change set to a JSON-representable value.
+ */
+ toJSON() {
+ let parts = [];
+ for (let i = 0; i < this.sections.length; i += 2) {
+ let len = this.sections[i], ins = this.sections[i + 1];
+ if (ins < 0)
+ parts.push(len);
+ else if (ins == 0)
+ parts.push([len]);
+ else
+ parts.push([len].concat(this.inserted[i >> 1].toJSON()));
+ }
+ return parts;
+ }
+ /**
+ Create a change set for the given changes, for a document of the
+ given length, using `lineSep` as line separator.
+ */
+ static of(changes, length, lineSep) {
+ let sections = [], inserted = [], pos = 0;
+ let total = null;
+ function flush(force = false) {
+ if (!force && !sections.length)
+ return;
+ if (pos < length)
+ addSection(sections, length - pos, -1);
+ let set = new ChangeSet(sections, inserted);
+ total = total ? total.compose(set.map(total)) : set;
+ sections = [];
+ inserted = [];
+ pos = 0;
+ }
+ function process(spec) {
+ if (Array.isArray(spec)) {
+ for (let sub of spec)
+ process(sub);
+ }
+ else if (spec instanceof ChangeSet) {
+ if (spec.length != length)
+ throw new RangeError(`Mismatched change set length (got ${spec.length}, expected ${length})`);
+ flush();
+ total = total ? total.compose(spec.map(total)) : spec;
+ }
+ else {
+ let { from, to = from, insert } = spec;
+ if (from > to || from < 0 || to > length)
+ throw new RangeError(`Invalid change range ${from} to ${to} (in doc of length ${length})`);
+ let insText = !insert ? Text.empty : typeof insert == "string" ? Text.of(insert.split(lineSep || DefaultSplit)) : insert;
+ let insLen = insText.length;
+ if (from == to && insLen == 0)
+ return;
+ if (from < pos)
+ flush();
+ if (from > pos)
+ addSection(sections, from - pos, -1);
+ addSection(sections, to - from, insLen);
+ addInsert(inserted, sections, insText);
+ pos = to;
+ }
+ }
+ process(changes);
+ flush(!total);
+ return total;
+ }
+ /**
+ Create an empty changeset of the given length.
+ */
+ static empty(length) {
+ return new ChangeSet(length ? [length, -1] : [], []);
+ }
+ /**
+ Create a changeset from its JSON representation (as produced by
+ [`toJSON`](https://codemirror.net/6/docs/ref/#state.ChangeSet.toJSON).
+ */
+ static fromJSON(json) {
+ if (!Array.isArray(json))
+ throw new RangeError("Invalid JSON representation of ChangeSet");
+ let sections = [], inserted = [];
+ for (let i = 0; i < json.length; i++) {
+ let part = json[i];
+ if (typeof part == "number") {
+ sections.push(part, -1);
+ }
+ else if (!Array.isArray(part) || typeof part[0] != "number" || part.some((e, i) => i && typeof e != "string")) {
+ throw new RangeError("Invalid JSON representation of ChangeSet");
+ }
+ else if (part.length == 1) {
+ sections.push(part[0], 0);
+ }
+ else {
+ while (inserted.length < i)
+ inserted.push(Text.empty);
+ inserted[i] = Text.of(part.slice(1));
+ sections.push(part[0], inserted[i].length);
+ }
+ }
+ return new ChangeSet(sections, inserted);
+ }
+ /**
+ @internal
+ */
+ static createSet(sections, inserted) {
+ return new ChangeSet(sections, inserted);
+ }
+}
+function addSection(sections, len, ins, forceJoin = false) {
+ if (len == 0 && ins <= 0)
+ return;
+ let last = sections.length - 2;
+ if (last >= 0 && ins <= 0 && ins == sections[last + 1])
+ sections[last] += len;
+ else if (last >= 0 && len == 0 && sections[last] == 0)
+ sections[last + 1] += ins;
+ else if (forceJoin) {
+ sections[last] += len;
+ sections[last + 1] += ins;
+ }
+ else
+ sections.push(len, ins);
+}
+function addInsert(values, sections, value) {
+ if (value.length == 0)
+ return;
+ let index = (sections.length - 2) >> 1;
+ if (index < values.length) {
+ values[values.length - 1] = values[values.length - 1].append(value);
+ }
+ else {
+ while (values.length < index)
+ values.push(Text.empty);
+ values.push(value);
+ }
+}
+function iterChanges(desc, f, individual) {
+ let inserted = desc.inserted;
+ for (let posA = 0, posB = 0, i = 0; i < desc.sections.length;) {
+ let len = desc.sections[i++], ins = desc.sections[i++];
+ if (ins < 0) {
+ posA += len;
+ posB += len;
+ }
+ else {
+ let endA = posA, endB = posB, text = Text.empty;
+ for (;;) {
+ endA += len;
+ endB += ins;
+ if (ins && inserted)
+ text = text.append(inserted[(i - 2) >> 1]);
+ if (individual || i == desc.sections.length || desc.sections[i + 1] < 0)
+ break;
+ len = desc.sections[i++];
+ ins = desc.sections[i++];
+ }
+ f(posA, endA, posB, endB, text);
+ posA = endA;
+ posB = endB;
+ }
+ }
+}
+function mapSet(setA, setB, before, mkSet = false) {
+ // Produce a copy of setA that applies to the document after setB
+ // has been applied (assuming both start at the same document).
+ let sections = [], insert = mkSet ? [] : null;
+ let a = new SectionIter(setA), b = new SectionIter(setB);
+ // Iterate over both sets in parallel. inserted tracks, for changes
+ // in A that have to be processed piece-by-piece, whether their
+ // content has been inserted already, and refers to the section
+ // index.
+ for (let inserted = -1;;) {
+ if (a.done && b.len || b.done && a.len) {
+ throw new Error("Mismatched change set lengths");
+ }
+ else if (a.ins == -1 && b.ins == -1) {
+ // Move across ranges skipped by both sets.
+ let len = Math.min(a.len, b.len);
+ addSection(sections, len, -1);
+ a.forward(len);
+ b.forward(len);
+ }
+ else if (b.ins >= 0 && (a.ins < 0 || inserted == a.i || a.off == 0 && (b.len < a.len || b.len == a.len && !before))) {
+ // If there's a change in B that comes before the next change in
+ // A (ordered by start pos, then len, then before flag), skip
+ // that (and process any changes in A it covers).
+ let len = b.len;
+ addSection(sections, b.ins, -1);
+ while (len) {
+ let piece = Math.min(a.len, len);
+ if (a.ins >= 0 && inserted < a.i && a.len <= piece) {
+ addSection(sections, 0, a.ins);
+ if (insert)
+ addInsert(insert, sections, a.text);
+ inserted = a.i;
+ }
+ a.forward(piece);
+ len -= piece;
+ }
+ b.next();
+ }
+ else if (a.ins >= 0) {
+ // Process the part of a change in A up to the start of the next
+ // non-deletion change in B (if overlapping).
+ let len = 0, left = a.len;
+ while (left) {
+ if (b.ins == -1) {
+ let piece = Math.min(left, b.len);
+ len += piece;
+ left -= piece;
+ b.forward(piece);
+ }
+ else if (b.ins == 0 && b.len < left) {
+ left -= b.len;
+ b.next();
+ }
+ else {
+ break;
+ }
+ }
+ addSection(sections, len, inserted < a.i ? a.ins : 0);
+ if (insert && inserted < a.i)
+ addInsert(insert, sections, a.text);
+ inserted = a.i;
+ a.forward(a.len - left);
+ }
+ else if (a.done && b.done) {
+ return insert ? ChangeSet.createSet(sections, insert) : ChangeDesc.create(sections);
+ }
+ else {
+ throw new Error("Mismatched change set lengths");
+ }
+ }
+}
+function composeSets(setA, setB, mkSet = false) {
+ let sections = [];
+ let insert = mkSet ? [] : null;
+ let a = new SectionIter(setA), b = new SectionIter(setB);
+ for (let open = false;;) {
+ if (a.done && b.done) {
+ return insert ? ChangeSet.createSet(sections, insert) : ChangeDesc.create(sections);
+ }
+ else if (a.ins == 0) { // Deletion in A
+ addSection(sections, a.len, 0, open);
+ a.next();
+ }
+ else if (b.len == 0 && !b.done) { // Insertion in B
+ addSection(sections, 0, b.ins, open);
+ if (insert)
+ addInsert(insert, sections, b.text);
+ b.next();
+ }
+ else if (a.done || b.done) {
+ throw new Error("Mismatched change set lengths");
+ }
+ else {
+ let len = Math.min(a.len2, b.len), sectionLen = sections.length;
+ if (a.ins == -1) {
+ let insB = b.ins == -1 ? -1 : b.off ? 0 : b.ins;
+ addSection(sections, len, insB, open);
+ if (insert && insB)
+ addInsert(insert, sections, b.text);
+ }
+ else if (b.ins == -1) {
+ addSection(sections, a.off ? 0 : a.len, len, open);
+ if (insert)
+ addInsert(insert, sections, a.textBit(len));
+ }
+ else {
+ addSection(sections, a.off ? 0 : a.len, b.off ? 0 : b.ins, open);
+ if (insert && !b.off)
+ addInsert(insert, sections, b.text);
+ }
+ open = (a.ins > len || b.ins >= 0 && b.len > len) && (open || sections.length > sectionLen);
+ a.forward2(len);
+ b.forward(len);
+ }
+ }
+}
+class SectionIter {
+ constructor(set) {
+ this.set = set;
+ this.i = 0;
+ this.next();
+ }
+ next() {
+ let { sections } = this.set;
+ if (this.i < sections.length) {
+ this.len = sections[this.i++];
+ this.ins = sections[this.i++];
+ }
+ else {
+ this.len = 0;
+ this.ins = -2;
+ }
+ this.off = 0;
+ }
+ get done() { return this.ins == -2; }
+ get len2() { return this.ins < 0 ? this.len : this.ins; }
+ get text() {
+ let { inserted } = this.set, index = (this.i - 2) >> 1;
+ return index >= inserted.length ? Text.empty : inserted[index];
+ }
+ textBit(len) {
+ let { inserted } = this.set, index = (this.i - 2) >> 1;
+ return index >= inserted.length && !len ? Text.empty
+ : inserted[index].slice(this.off, len == null ? undefined : this.off + len);
+ }
+ forward(len) {
+ if (len == this.len)
+ this.next();
+ else {
+ this.len -= len;
+ this.off += len;
+ }
+ }
+ forward2(len) {
+ if (this.ins == -1)
+ this.forward(len);
+ else if (len == this.ins)
+ this.next();
+ else {
+ this.ins -= len;
+ this.off += len;
+ }
+ }
+}
+
+/**
+A single selection range. When
+[`allowMultipleSelections`](https://codemirror.net/6/docs/ref/#state.EditorState^allowMultipleSelections)
+is enabled, a [selection](https://codemirror.net/6/docs/ref/#state.EditorSelection) may hold
+multiple ranges. By default, selections hold exactly one range.
+*/
+class SelectionRange {
+ constructor(
+ /**
+ The lower boundary of the range.
+ */
+ from,
+ /**
+ The upper boundary of the range.
+ */
+ to, flags) {
+ this.from = from;
+ this.to = to;
+ this.flags = flags;
+ }
+ /**
+ The anchor of the range—the side that doesn't move when you
+ extend it.
+ */
+ get anchor() { return this.flags & 32 /* RangeFlag.Inverted */ ? this.to : this.from; }
+ /**
+ The head of the range, which is moved when the range is
+ [extended](https://codemirror.net/6/docs/ref/#state.SelectionRange.extend).
+ */
+ get head() { return this.flags & 32 /* RangeFlag.Inverted */ ? this.from : this.to; }
+ /**
+ True when `anchor` and `head` are at the same position.
+ */
+ get empty() { return this.from == this.to; }
+ /**
+ If this is a cursor that is explicitly associated with the
+ character on one of its sides, this returns the side. -1 means
+ the character before its position, 1 the character after, and 0
+ means no association.
+ */
+ get assoc() { return this.flags & 8 /* RangeFlag.AssocBefore */ ? -1 : this.flags & 16 /* RangeFlag.AssocAfter */ ? 1 : 0; }
+ /**
+ The bidirectional text level associated with this cursor, if
+ any.
+ */
+ get bidiLevel() {
+ let level = this.flags & 7 /* RangeFlag.BidiLevelMask */;
+ return level == 7 ? null : level;
+ }
+ /**
+ The goal column (stored vertical offset) associated with a
+ cursor. This is used to preserve the vertical position when
+ [moving](https://codemirror.net/6/docs/ref/#view.EditorView.moveVertically) across
+ lines of different length.
+ */
+ get goalColumn() {
+ let value = this.flags >> 6 /* RangeFlag.GoalColumnOffset */;
+ return value == 16777215 /* RangeFlag.NoGoalColumn */ ? undefined : value;
+ }
+ /**
+ Map this range through a change, producing a valid range in the
+ updated document.
+ */
+ map(change, assoc = -1) {
+ let from, to;
+ if (this.empty) {
+ from = to = change.mapPos(this.from, assoc);
+ }
+ else {
+ from = change.mapPos(this.from, 1);
+ to = change.mapPos(this.to, -1);
+ }
+ return from == this.from && to == this.to ? this : new SelectionRange(from, to, this.flags);
+ }
+ /**
+ Extend this range to cover at least `from` to `to`.
+ */
+ extend(from, to = from) {
+ if (from <= this.anchor && to >= this.anchor)
+ return EditorSelection.range(from, to);
+ let head = Math.abs(from - this.anchor) > Math.abs(to - this.anchor) ? from : to;
+ return EditorSelection.range(this.anchor, head);
+ }
+ /**
+ Compare this range to another range.
+ */
+ eq(other, includeAssoc = false) {
+ return this.anchor == other.anchor && this.head == other.head &&
+ (!includeAssoc || !this.empty || this.assoc == other.assoc);
+ }
+ /**
+ Return a JSON-serializable object representing the range.
+ */
+ toJSON() { return { anchor: this.anchor, head: this.head }; }
+ /**
+ Convert a JSON representation of a range to a `SelectionRange`
+ instance.
+ */
+ static fromJSON(json) {
+ if (!json || typeof json.anchor != "number" || typeof json.head != "number")
+ throw new RangeError("Invalid JSON representation for SelectionRange");
+ return EditorSelection.range(json.anchor, json.head);
+ }
+ /**
+ @internal
+ */
+ static create(from, to, flags) {
+ return new SelectionRange(from, to, flags);
+ }
+}
+/**
+An editor selection holds one or more selection ranges.
+*/
+class EditorSelection {
+ constructor(
+ /**
+ The ranges in the selection, sorted by position. Ranges cannot
+ overlap (but they may touch, if they aren't empty).
+ */
+ ranges,
+ /**
+ The index of the _main_ range in the selection (which is
+ usually the range that was added last).
+ */
+ mainIndex) {
+ this.ranges = ranges;
+ this.mainIndex = mainIndex;
+ }
+ /**
+ Map a selection through a change. Used to adjust the selection
+ position for changes.
+ */
+ map(change, assoc = -1) {
+ if (change.empty)
+ return this;
+ return EditorSelection.create(this.ranges.map(r => r.map(change, assoc)), this.mainIndex);
+ }
+ /**
+ Compare this selection to another selection. By default, ranges
+ are compared only by position. When `includeAssoc` is true,
+ cursor ranges must also have the same
+ [`assoc`](https://codemirror.net/6/docs/ref/#state.SelectionRange.assoc) value.
+ */
+ eq(other, includeAssoc = false) {
+ if (this.ranges.length != other.ranges.length ||
+ this.mainIndex != other.mainIndex)
+ return false;
+ for (let i = 0; i < this.ranges.length; i++)
+ if (!this.ranges[i].eq(other.ranges[i], includeAssoc))
+ return false;
+ return true;
+ }
+ /**
+ Get the primary selection range. Usually, you should make sure
+ your code applies to _all_ ranges, by using methods like
+ [`changeByRange`](https://codemirror.net/6/docs/ref/#state.EditorState.changeByRange).
+ */
+ get main() { return this.ranges[this.mainIndex]; }
+ /**
+ Make sure the selection only has one range. Returns a selection
+ holding only the main range from this selection.
+ */
+ asSingle() {
+ return this.ranges.length == 1 ? this : new EditorSelection([this.main], 0);
+ }
+ /**
+ Extend this selection with an extra range.
+ */
+ addRange(range, main = true) {
+ return EditorSelection.create([range].concat(this.ranges), main ? 0 : this.mainIndex + 1);
+ }
+ /**
+ Replace a given range with another range, and then normalize the
+ selection to merge and sort ranges if necessary.
+ */
+ replaceRange(range, which = this.mainIndex) {
+ let ranges = this.ranges.slice();
+ ranges[which] = range;
+ return EditorSelection.create(ranges, this.mainIndex);
+ }
+ /**
+ Convert this selection to an object that can be serialized to
+ JSON.
+ */
+ toJSON() {
+ return { ranges: this.ranges.map(r => r.toJSON()), main: this.mainIndex };
+ }
+ /**
+ Create a selection from a JSON representation.
+ */
+ static fromJSON(json) {
+ if (!json || !Array.isArray(json.ranges) || typeof json.main != "number" || json.main >= json.ranges.length)
+ throw new RangeError("Invalid JSON representation for EditorSelection");
+ return new EditorSelection(json.ranges.map((r) => SelectionRange.fromJSON(r)), json.main);
+ }
+ /**
+ Create a selection holding a single range.
+ */
+ static single(anchor, head = anchor) {
+ return new EditorSelection([EditorSelection.range(anchor, head)], 0);
+ }
+ /**
+ Sort and merge the given set of ranges, creating a valid
+ selection.
+ */
+ static create(ranges, mainIndex = 0) {
+ if (ranges.length == 0)
+ throw new RangeError("A selection needs at least one range");
+ for (let pos = 0, i = 0; i < ranges.length; i++) {
+ let range = ranges[i];
+ if (range.empty ? range.from <= pos : range.from < pos)
+ return EditorSelection.normalized(ranges.slice(), mainIndex);
+ pos = range.to;
+ }
+ return new EditorSelection(ranges, mainIndex);
+ }
+ /**
+ Create a cursor selection range at the given position. You can
+ safely ignore the optional arguments in most situations.
+ */
+ static cursor(pos, assoc = 0, bidiLevel, goalColumn) {
+ return SelectionRange.create(pos, pos, (assoc == 0 ? 0 : assoc < 0 ? 8 /* RangeFlag.AssocBefore */ : 16 /* RangeFlag.AssocAfter */) |
+ (bidiLevel == null ? 7 : Math.min(6, bidiLevel)) |
+ ((goalColumn !== null && goalColumn !== void 0 ? goalColumn : 16777215 /* RangeFlag.NoGoalColumn */) << 6 /* RangeFlag.GoalColumnOffset */));
+ }
+ /**
+ Create a selection range.
+ */
+ static range(anchor, head, goalColumn, bidiLevel) {
+ let flags = ((goalColumn !== null && goalColumn !== void 0 ? goalColumn : 16777215 /* RangeFlag.NoGoalColumn */) << 6 /* RangeFlag.GoalColumnOffset */) |
+ (bidiLevel == null ? 7 : Math.min(6, bidiLevel));
+ return head < anchor ? SelectionRange.create(head, anchor, 32 /* RangeFlag.Inverted */ | 16 /* RangeFlag.AssocAfter */ | flags)
+ : SelectionRange.create(anchor, head, (head > anchor ? 8 /* RangeFlag.AssocBefore */ : 0) | flags);
+ }
+ /**
+ @internal
+ */
+ static normalized(ranges, mainIndex = 0) {
+ let main = ranges[mainIndex];
+ ranges.sort((a, b) => a.from - b.from);
+ mainIndex = ranges.indexOf(main);
+ for (let i = 1; i < ranges.length; i++) {
+ let range = ranges[i], prev = ranges[i - 1];
+ if (range.empty ? range.from <= prev.to : range.from < prev.to) {
+ let from = prev.from, to = Math.max(range.to, prev.to);
+ if (i <= mainIndex)
+ mainIndex--;
+ ranges.splice(--i, 2, range.anchor > range.head ? EditorSelection.range(to, from) : EditorSelection.range(from, to));
+ }
+ }
+ return new EditorSelection(ranges, mainIndex);
+ }
+}
+function checkSelection(selection, docLength) {
+ for (let range of selection.ranges)
+ if (range.to > docLength)
+ throw new RangeError("Selection points outside of document");
+}
+
+let nextID = 0;
+/**
+A facet is a labeled value that is associated with an editor
+state. It takes inputs from any number of extensions, and combines
+those into a single output value.
+
+Examples of uses of facets are the [tab
+size](https://codemirror.net/6/docs/ref/#state.EditorState^tabSize), [editor
+attributes](https://codemirror.net/6/docs/ref/#view.EditorView^editorAttributes), and [update
+listeners](https://codemirror.net/6/docs/ref/#view.EditorView^updateListener).
+
+Note that `Facet` instances can be used anywhere where
+[`FacetReader`](https://codemirror.net/6/docs/ref/#state.FacetReader) is expected.
+*/
+class Facet {
+ constructor(
+ /**
+ @internal
+ */
+ combine,
+ /**
+ @internal
+ */
+ compareInput,
+ /**
+ @internal
+ */
+ compare, isStatic, enables) {
+ this.combine = combine;
+ this.compareInput = compareInput;
+ this.compare = compare;
+ this.isStatic = isStatic;
+ /**
+ @internal
+ */
+ this.id = nextID++;
+ this.default = combine([]);
+ this.extensions = typeof enables == "function" ? enables(this) : enables;
+ }
+ /**
+ Returns a facet reader for this facet, which can be used to
+ [read](https://codemirror.net/6/docs/ref/#state.EditorState.facet) it but not to define values for it.
+ */
+ get reader() { return this; }
+ /**
+ Define a new facet.
+ */
+ static define(config = {}) {
+ return new Facet(config.combine || ((a) => a), config.compareInput || ((a, b) => a === b), config.compare || (!config.combine ? sameArray : (a, b) => a === b), !!config.static, config.enables);
+ }
+ /**
+ Returns an extension that adds the given value to this facet.
+ */
+ of(value) {
+ return new FacetProvider([], this, 0 /* Provider.Static */, value);
+ }
+ /**
+ Create an extension that computes a value for the facet from a
+ state. You must take care to declare the parts of the state that
+ this value depends on, since your function is only called again
+ for a new state when one of those parts changed.
+
+ In cases where your value depends only on a single field, you'll
+ want to use the [`from`](https://codemirror.net/6/docs/ref/#state.Facet.from) method instead.
+ */
+ compute(deps, get) {
+ if (this.isStatic)
+ throw new Error("Can't compute a static facet");
+ return new FacetProvider(deps, this, 1 /* Provider.Single */, get);
+ }
+ /**
+ Create an extension that computes zero or more values for this
+ facet from a state.
+ */
+ computeN(deps, get) {
+ if (this.isStatic)
+ throw new Error("Can't compute a static facet");
+ return new FacetProvider(deps, this, 2 /* Provider.Multi */, get);
+ }
+ from(field, get) {
+ if (!get)
+ get = x => x;
+ return this.compute([field], state => get(state.field(field)));
+ }
+}
+function sameArray(a, b) {
+ return a == b || a.length == b.length && a.every((e, i) => e === b[i]);
+}
+class FacetProvider {
+ constructor(dependencies, facet, type, value) {
+ this.dependencies = dependencies;
+ this.facet = facet;
+ this.type = type;
+ this.value = value;
+ this.id = nextID++;
+ }
+ dynamicSlot(addresses) {
+ var _a;
+ let getter = this.value;
+ let compare = this.facet.compareInput;
+ let id = this.id, idx = addresses[id] >> 1, multi = this.type == 2 /* Provider.Multi */;
+ let depDoc = false, depSel = false, depAddrs = [];
+ for (let dep of this.dependencies) {
+ if (dep == "doc")
+ depDoc = true;
+ else if (dep == "selection")
+ depSel = true;
+ else if ((((_a = addresses[dep.id]) !== null && _a !== void 0 ? _a : 1) & 1) == 0)
+ depAddrs.push(addresses[dep.id]);
+ }
+ return {
+ create(state) {
+ state.values[idx] = getter(state);
+ return 1 /* SlotStatus.Changed */;
+ },
+ update(state, tr) {
+ if ((depDoc && tr.docChanged) || (depSel && (tr.docChanged || tr.selection)) || ensureAll(state, depAddrs)) {
+ let newVal = getter(state);
+ if (multi ? !compareArray(newVal, state.values[idx], compare) : !compare(newVal, state.values[idx])) {
+ state.values[idx] = newVal;
+ return 1 /* SlotStatus.Changed */;
+ }
+ }
+ return 0;
+ },
+ reconfigure: (state, oldState) => {
+ let newVal, oldAddr = oldState.config.address[id];
+ if (oldAddr != null) {
+ let oldVal = getAddr(oldState, oldAddr);
+ if (this.dependencies.every(dep => {
+ return dep instanceof Facet ? oldState.facet(dep) === state.facet(dep) :
+ dep instanceof StateField ? oldState.field(dep, false) == state.field(dep, false) : true;
+ }) || (multi ? compareArray(newVal = getter(state), oldVal, compare) : compare(newVal = getter(state), oldVal))) {
+ state.values[idx] = oldVal;
+ return 0;
+ }
+ }
+ else {
+ newVal = getter(state);
+ }
+ state.values[idx] = newVal;
+ return 1 /* SlotStatus.Changed */;
+ }
+ };
+ }
+}
+function compareArray(a, b, compare) {
+ if (a.length != b.length)
+ return false;
+ for (let i = 0; i < a.length; i++)
+ if (!compare(a[i], b[i]))
+ return false;
+ return true;
+}
+function ensureAll(state, addrs) {
+ let changed = false;
+ for (let addr of addrs)
+ if (ensureAddr(state, addr) & 1 /* SlotStatus.Changed */)
+ changed = true;
+ return changed;
+}
+function dynamicFacetSlot(addresses, facet, providers) {
+ let providerAddrs = providers.map(p => addresses[p.id]);
+ let providerTypes = providers.map(p => p.type);
+ let dynamic = providerAddrs.filter(p => !(p & 1));
+ let idx = addresses[facet.id] >> 1;
+ function get(state) {
+ let values = [];
+ for (let i = 0; i < providerAddrs.length; i++) {
+ let value = getAddr(state, providerAddrs[i]);
+ if (providerTypes[i] == 2 /* Provider.Multi */)
+ for (let val of value)
+ values.push(val);
+ else
+ values.push(value);
+ }
+ return facet.combine(values);
+ }
+ return {
+ create(state) {
+ for (let addr of providerAddrs)
+ ensureAddr(state, addr);
+ state.values[idx] = get(state);
+ return 1 /* SlotStatus.Changed */;
+ },
+ update(state, tr) {
+ if (!ensureAll(state, dynamic))
+ return 0;
+ let value = get(state);
+ if (facet.compare(value, state.values[idx]))
+ return 0;
+ state.values[idx] = value;
+ return 1 /* SlotStatus.Changed */;
+ },
+ reconfigure(state, oldState) {
+ let depChanged = ensureAll(state, providerAddrs);
+ let oldProviders = oldState.config.facets[facet.id], oldValue = oldState.facet(facet);
+ if (oldProviders && !depChanged && sameArray(providers, oldProviders)) {
+ state.values[idx] = oldValue;
+ return 0;
+ }
+ let value = get(state);
+ if (facet.compare(value, oldValue)) {
+ state.values[idx] = oldValue;
+ return 0;
+ }
+ state.values[idx] = value;
+ return 1 /* SlotStatus.Changed */;
+ }
+ };
+}
+const initField = Facet.define({ static: true });
+/**
+Fields can store additional information in an editor state, and
+keep it in sync with the rest of the state.
+*/
+class StateField {
+ constructor(
+ /**
+ @internal
+ */
+ id, createF, updateF, compareF,
+ /**
+ @internal
+ */
+ spec) {
+ this.id = id;
+ this.createF = createF;
+ this.updateF = updateF;
+ this.compareF = compareF;
+ this.spec = spec;
+ /**
+ @internal
+ */
+ this.provides = undefined;
+ }
+ /**
+ Define a state field.
+ */
+ static define(config) {
+ let field = new StateField(nextID++, config.create, config.update, config.compare || ((a, b) => a === b), config);
+ if (config.provide)
+ field.provides = config.provide(field);
+ return field;
+ }
+ create(state) {
+ let init = state.facet(initField).find(i => i.field == this);
+ return ((init === null || init === void 0 ? void 0 : init.create) || this.createF)(state);
+ }
+ /**
+ @internal
+ */
+ slot(addresses) {
+ let idx = addresses[this.id] >> 1;
+ return {
+ create: (state) => {
+ state.values[idx] = this.create(state);
+ return 1 /* SlotStatus.Changed */;
+ },
+ update: (state, tr) => {
+ let oldVal = state.values[idx];
+ let value = this.updateF(oldVal, tr);
+ if (this.compareF(oldVal, value))
+ return 0;
+ state.values[idx] = value;
+ return 1 /* SlotStatus.Changed */;
+ },
+ reconfigure: (state, oldState) => {
+ if (oldState.config.address[this.id] != null) {
+ state.values[idx] = oldState.field(this);
+ return 0;
+ }
+ state.values[idx] = this.create(state);
+ return 1 /* SlotStatus.Changed */;
+ }
+ };
+ }
+ /**
+ Returns an extension that enables this field and overrides the
+ way it is initialized. Can be useful when you need to provide a
+ non-default starting value for the field.
+ */
+ init(create) {
+ return [this, initField.of({ field: this, create })];
+ }
+ /**
+ State field instances can be used as
+ [`Extension`](https://codemirror.net/6/docs/ref/#state.Extension) values to enable the field in a
+ given state.
+ */
+ get extension() { return this; }
+}
+const Prec_ = { lowest: 4, low: 3, default: 2, high: 1, highest: 0 };
+function prec(value) {
+ return (ext) => new PrecExtension(ext, value);
+}
+/**
+By default extensions are registered in the order they are found
+in the flattened form of nested array that was provided.
+Individual extension values can be assigned a precedence to
+override this. Extensions that do not have a precedence set get
+the precedence of the nearest parent with a precedence, or
+[`default`](https://codemirror.net/6/docs/ref/#state.Prec.default) if there is no such parent. The
+final ordering of extensions is determined by first sorting by
+precedence and then by order within each precedence.
+*/
+const Prec = {
+ /**
+ The highest precedence level, for extensions that should end up
+ near the start of the precedence ordering.
+ */
+ highest: prec(Prec_.highest),
+ /**
+ A higher-than-default precedence, for extensions that should
+ come before those with default precedence.
+ */
+ high: prec(Prec_.high),
+ /**
+ The default precedence, which is also used for extensions
+ without an explicit precedence.
+ */
+ default: prec(Prec_.default),
+ /**
+ A lower-than-default precedence.
+ */
+ low: prec(Prec_.low),
+ /**
+ The lowest precedence level. Meant for things that should end up
+ near the end of the extension order.
+ */
+ lowest: prec(Prec_.lowest)
+};
+class PrecExtension {
+ constructor(inner, prec) {
+ this.inner = inner;
+ this.prec = prec;
+ }
+}
+/**
+Extension compartments can be used to make a configuration
+dynamic. By [wrapping](https://codemirror.net/6/docs/ref/#state.Compartment.of) part of your
+configuration in a compartment, you can later
+[replace](https://codemirror.net/6/docs/ref/#state.Compartment.reconfigure) that part through a
+transaction.
+*/
+class Compartment {
+ /**
+ Create an instance of this compartment to add to your [state
+ configuration](https://codemirror.net/6/docs/ref/#state.EditorStateConfig.extensions).
+ */
+ of(ext) { return new CompartmentInstance(this, ext); }
+ /**
+ Create an [effect](https://codemirror.net/6/docs/ref/#state.TransactionSpec.effects) that
+ reconfigures this compartment.
+ */
+ reconfigure(content) {
+ return Compartment.reconfigure.of({ compartment: this, extension: content });
+ }
+ /**
+ Get the current content of the compartment in the state, or
+ `undefined` if it isn't present.
+ */
+ get(state) {
+ return state.config.compartments.get(this);
+ }
+}
+class CompartmentInstance {
+ constructor(compartment, inner) {
+ this.compartment = compartment;
+ this.inner = inner;
+ }
+}
+class Configuration {
+ constructor(base, compartments, dynamicSlots, address, staticValues, facets) {
+ this.base = base;
+ this.compartments = compartments;
+ this.dynamicSlots = dynamicSlots;
+ this.address = address;
+ this.staticValues = staticValues;
+ this.facets = facets;
+ this.statusTemplate = [];
+ while (this.statusTemplate.length < dynamicSlots.length)
+ this.statusTemplate.push(0 /* SlotStatus.Unresolved */);
+ }
+ staticFacet(facet) {
+ let addr = this.address[facet.id];
+ return addr == null ? facet.default : this.staticValues[addr >> 1];
+ }
+ static resolve(base, compartments, oldState) {
+ let fields = [];
+ let facets = Object.create(null);
+ let newCompartments = new Map();
+ for (let ext of flatten(base, compartments, newCompartments)) {
+ if (ext instanceof StateField)
+ fields.push(ext);
+ else
+ (facets[ext.facet.id] || (facets[ext.facet.id] = [])).push(ext);
+ }
+ let address = Object.create(null);
+ let staticValues = [];
+ let dynamicSlots = [];
+ for (let field of fields) {
+ address[field.id] = dynamicSlots.length << 1;
+ dynamicSlots.push(a => field.slot(a));
+ }
+ let oldFacets = oldState === null || oldState === void 0 ? void 0 : oldState.config.facets;
+ for (let id in facets) {
+ let providers = facets[id], facet = providers[0].facet;
+ let oldProviders = oldFacets && oldFacets[id] || [];
+ if (providers.every(p => p.type == 0 /* Provider.Static */)) {
+ address[facet.id] = (staticValues.length << 1) | 1;
+ if (sameArray(oldProviders, providers)) {
+ staticValues.push(oldState.facet(facet));
+ }
+ else {
+ let value = facet.combine(providers.map(p => p.value));
+ staticValues.push(oldState && facet.compare(value, oldState.facet(facet)) ? oldState.facet(facet) : value);
+ }
+ }
+ else {
+ for (let p of providers) {
+ if (p.type == 0 /* Provider.Static */) {
+ address[p.id] = (staticValues.length << 1) | 1;
+ staticValues.push(p.value);
+ }
+ else {
+ address[p.id] = dynamicSlots.length << 1;
+ dynamicSlots.push(a => p.dynamicSlot(a));
+ }
+ }
+ address[facet.id] = dynamicSlots.length << 1;
+ dynamicSlots.push(a => dynamicFacetSlot(a, facet, providers));
+ }
+ }
+ let dynamic = dynamicSlots.map(f => f(address));
+ return new Configuration(base, newCompartments, dynamic, address, staticValues, facets);
+ }
+}
+function flatten(extension, compartments, newCompartments) {
+ let result = [[], [], [], [], []];
+ let seen = new Map();
+ function inner(ext, prec) {
+ let known = seen.get(ext);
+ if (known != null) {
+ if (known <= prec)
+ return;
+ let found = result[known].indexOf(ext);
+ if (found > -1)
+ result[known].splice(found, 1);
+ if (ext instanceof CompartmentInstance)
+ newCompartments.delete(ext.compartment);
+ }
+ seen.set(ext, prec);
+ if (Array.isArray(ext)) {
+ for (let e of ext)
+ inner(e, prec);
+ }
+ else if (ext instanceof CompartmentInstance) {
+ if (newCompartments.has(ext.compartment))
+ throw new RangeError(`Duplicate use of compartment in extensions`);
+ let content = compartments.get(ext.compartment) || ext.inner;
+ newCompartments.set(ext.compartment, content);
+ inner(content, prec);
+ }
+ else if (ext instanceof PrecExtension) {
+ inner(ext.inner, ext.prec);
+ }
+ else if (ext instanceof StateField) {
+ result[prec].push(ext);
+ if (ext.provides)
+ inner(ext.provides, prec);
+ }
+ else if (ext instanceof FacetProvider) {
+ result[prec].push(ext);
+ if (ext.facet.extensions)
+ inner(ext.facet.extensions, Prec_.default);
+ }
+ else {
+ let content = ext.extension;
+ if (!content)
+ throw new Error(`Unrecognized extension value in extension set (${ext}). This sometimes happens because multiple instances of @codemirror/state are loaded, breaking instanceof checks.`);
+ inner(content, prec);
+ }
+ }
+ inner(extension, Prec_.default);
+ return result.reduce((a, b) => a.concat(b));
+}
+function ensureAddr(state, addr) {
+ if (addr & 1)
+ return 2 /* SlotStatus.Computed */;
+ let idx = addr >> 1;
+ let status = state.status[idx];
+ if (status == 4 /* SlotStatus.Computing */)
+ throw new Error("Cyclic dependency between fields and/or facets");
+ if (status & 2 /* SlotStatus.Computed */)
+ return status;
+ state.status[idx] = 4 /* SlotStatus.Computing */;
+ let changed = state.computeSlot(state, state.config.dynamicSlots[idx]);
+ return state.status[idx] = 2 /* SlotStatus.Computed */ | changed;
+}
+function getAddr(state, addr) {
+ return addr & 1 ? state.config.staticValues[addr >> 1] : state.values[addr >> 1];
+}
+
+const languageData = Facet.define();
+const allowMultipleSelections = Facet.define({
+ combine: values => values.some(v => v),
+ static: true
+});
+const lineSeparator = Facet.define({
+ combine: values => values.length ? values[0] : undefined,
+ static: true
+});
+const changeFilter = Facet.define();
+const transactionFilter = Facet.define();
+const transactionExtender = Facet.define();
+const readOnly = Facet.define({
+ combine: values => values.length ? values[0] : false
+});
+
+/**
+Annotations are tagged values that are used to add metadata to
+transactions in an extensible way. They should be used to model
+things that effect the entire transaction (such as its [time
+stamp](https://codemirror.net/6/docs/ref/#state.Transaction^time) or information about its
+[origin](https://codemirror.net/6/docs/ref/#state.Transaction^userEvent)). For effects that happen
+_alongside_ the other changes made by the transaction, [state
+effects](https://codemirror.net/6/docs/ref/#state.StateEffect) are more appropriate.
+*/
+class Annotation {
+ /**
+ @internal
+ */
+ constructor(
+ /**
+ The annotation type.
+ */
+ type,
+ /**
+ The value of this annotation.
+ */
+ value) {
+ this.type = type;
+ this.value = value;
+ }
+ /**
+ Define a new type of annotation.
+ */
+ static define() { return new AnnotationType(); }
+}
+/**
+Marker that identifies a type of [annotation](https://codemirror.net/6/docs/ref/#state.Annotation).
+*/
+class AnnotationType {
+ /**
+ Create an instance of this annotation.
+ */
+ of(value) { return new Annotation(this, value); }
+}
+/**
+Representation of a type of state effect. Defined with
+[`StateEffect.define`](https://codemirror.net/6/docs/ref/#state.StateEffect^define).
+*/
+class StateEffectType {
+ /**
+ @internal
+ */
+ constructor(
+ // The `any` types in these function types are there to work
+ // around TypeScript issue #37631, where the type guard on
+ // `StateEffect.is` mysteriously stops working when these properly
+ // have type `Value`.
+ /**
+ @internal
+ */
+ map) {
+ this.map = map;
+ }
+ /**
+ Create a [state effect](https://codemirror.net/6/docs/ref/#state.StateEffect) instance of this
+ type.
+ */
+ of(value) { return new StateEffect(this, value); }
+}
+/**
+State effects can be used to represent additional effects
+associated with a [transaction](https://codemirror.net/6/docs/ref/#state.Transaction.effects). They
+are often useful to model changes to custom [state
+fields](https://codemirror.net/6/docs/ref/#state.StateField), when those changes aren't implicit in
+document or selection changes.
+*/
+class StateEffect {
+ /**
+ @internal
+ */
+ constructor(
+ /**
+ @internal
+ */
+ type,
+ /**
+ The value of this effect.
+ */
+ value) {
+ this.type = type;
+ this.value = value;
+ }
+ /**
+ Map this effect through a position mapping. Will return
+ `undefined` when that ends up deleting the effect.
+ */
+ map(mapping) {
+ let mapped = this.type.map(this.value, mapping);
+ return mapped === undefined ? undefined : mapped == this.value ? this : new StateEffect(this.type, mapped);
+ }
+ /**
+ Tells you whether this effect object is of a given
+ [type](https://codemirror.net/6/docs/ref/#state.StateEffectType).
+ */
+ is(type) { return this.type == type; }
+ /**
+ Define a new effect type. The type parameter indicates the type
+ of values that his effect holds. It should be a type that
+ doesn't include `undefined`, since that is used in
+ [mapping](https://codemirror.net/6/docs/ref/#state.StateEffect.map) to indicate that an effect is
+ removed.
+ */
+ static define(spec = {}) {
+ return new StateEffectType(spec.map || (v => v));
+ }
+ /**
+ Map an array of effects through a change set.
+ */
+ static mapEffects(effects, mapping) {
+ if (!effects.length)
+ return effects;
+ let result = [];
+ for (let effect of effects) {
+ let mapped = effect.map(mapping);
+ if (mapped)
+ result.push(mapped);
+ }
+ return result;
+ }
+}
+/**
+This effect can be used to reconfigure the root extensions of
+the editor. Doing this will discard any extensions
+[appended](https://codemirror.net/6/docs/ref/#state.StateEffect^appendConfig), but does not reset
+the content of [reconfigured](https://codemirror.net/6/docs/ref/#state.Compartment.reconfigure)
+compartments.
+*/
+StateEffect.reconfigure = StateEffect.define();
+/**
+Append extensions to the top-level configuration of the editor.
+*/
+StateEffect.appendConfig = StateEffect.define();
+/**
+Changes to the editor state are grouped into transactions.
+Typically, a user action creates a single transaction, which may
+contain any number of document changes, may change the selection,
+or have other effects. Create a transaction by calling
+[`EditorState.update`](https://codemirror.net/6/docs/ref/#state.EditorState.update), or immediately
+dispatch one by calling
+[`EditorView.dispatch`](https://codemirror.net/6/docs/ref/#view.EditorView.dispatch).
+*/
+class Transaction {
+ constructor(
+ /**
+ The state from which the transaction starts.
+ */
+ startState,
+ /**
+ The document changes made by this transaction.
+ */
+ changes,
+ /**
+ The selection set by this transaction, or undefined if it
+ doesn't explicitly set a selection.
+ */
+ selection,
+ /**
+ The effects added to the transaction.
+ */
+ effects,
+ /**
+ @internal
+ */
+ annotations,
+ /**
+ Whether the selection should be scrolled into view after this
+ transaction is dispatched.
+ */
+ scrollIntoView) {
+ this.startState = startState;
+ this.changes = changes;
+ this.selection = selection;
+ this.effects = effects;
+ this.annotations = annotations;
+ this.scrollIntoView = scrollIntoView;
+ /**
+ @internal
+ */
+ this._doc = null;
+ /**
+ @internal
+ */
+ this._state = null;
+ if (selection)
+ checkSelection(selection, changes.newLength);
+ if (!annotations.some((a) => a.type == Transaction.time))
+ this.annotations = annotations.concat(Transaction.time.of(Date.now()));
+ }
+ /**
+ @internal
+ */
+ static create(startState, changes, selection, effects, annotations, scrollIntoView) {
+ return new Transaction(startState, changes, selection, effects, annotations, scrollIntoView);
+ }
+ /**
+ The new document produced by the transaction. Contrary to
+ [`.state`](https://codemirror.net/6/docs/ref/#state.Transaction.state)`.doc`, accessing this won't
+ force the entire new state to be computed right away, so it is
+ recommended that [transaction
+ filters](https://codemirror.net/6/docs/ref/#state.EditorState^transactionFilter) use this getter
+ when they need to look at the new document.
+ */
+ get newDoc() {
+ return this._doc || (this._doc = this.changes.apply(this.startState.doc));
+ }
+ /**
+ The new selection produced by the transaction. If
+ [`this.selection`](https://codemirror.net/6/docs/ref/#state.Transaction.selection) is undefined,
+ this will [map](https://codemirror.net/6/docs/ref/#state.EditorSelection.map) the start state's
+ current selection through the changes made by the transaction.
+ */
+ get newSelection() {
+ return this.selection || this.startState.selection.map(this.changes);
+ }
+ /**
+ The new state created by the transaction. Computed on demand
+ (but retained for subsequent access), so it is recommended not to
+ access it in [transaction
+ filters](https://codemirror.net/6/docs/ref/#state.EditorState^transactionFilter) when possible.
+ */
+ get state() {
+ if (!this._state)
+ this.startState.applyTransaction(this);
+ return this._state;
+ }
+ /**
+ Get the value of the given annotation type, if any.
+ */
+ annotation(type) {
+ for (let ann of this.annotations)
+ if (ann.type == type)
+ return ann.value;
+ return undefined;
+ }
+ /**
+ Indicates whether the transaction changed the document.
+ */
+ get docChanged() { return !this.changes.empty; }
+ /**
+ Indicates whether this transaction reconfigures the state
+ (through a [configuration compartment](https://codemirror.net/6/docs/ref/#state.Compartment) or
+ with a top-level configuration
+ [effect](https://codemirror.net/6/docs/ref/#state.StateEffect^reconfigure).
+ */
+ get reconfigured() { return this.startState.config != this.state.config; }
+ /**
+ Returns true if the transaction has a [user
+ event](https://codemirror.net/6/docs/ref/#state.Transaction^userEvent) annotation that is equal to
+ or more specific than `event`. For example, if the transaction
+ has `"select.pointer"` as user event, `"select"` and
+ `"select.pointer"` will match it.
+ */
+ isUserEvent(event) {
+ let e = this.annotation(Transaction.userEvent);
+ return !!(e && (e == event || e.length > event.length && e.slice(0, event.length) == event && e[event.length] == "."));
+ }
+}
+/**
+Annotation used to store transaction timestamps. Automatically
+added to every transaction, holding `Date.now()`.
+*/
+Transaction.time = Annotation.define();
+/**
+Annotation used to associate a transaction with a user interface
+event. Holds a string identifying the event, using a
+dot-separated format to support attaching more specific
+information. The events used by the core libraries are:
+
+ - `"input"` when content is entered
+ - `"input.type"` for typed input
+ - `"input.type.compose"` for composition
+ - `"input.paste"` for pasted input
+ - `"input.drop"` when adding content with drag-and-drop
+ - `"input.complete"` when autocompleting
+ - `"delete"` when the user deletes content
+ - `"delete.selection"` when deleting the selection
+ - `"delete.forward"` when deleting forward from the selection
+ - `"delete.backward"` when deleting backward from the selection
+ - `"delete.cut"` when cutting to the clipboard
+ - `"move"` when content is moved
+ - `"move.drop"` when content is moved within the editor through drag-and-drop
+ - `"select"` when explicitly changing the selection
+ - `"select.pointer"` when selecting with a mouse or other pointing device
+ - `"undo"` and `"redo"` for history actions
+
+Use [`isUserEvent`](https://codemirror.net/6/docs/ref/#state.Transaction.isUserEvent) to check
+whether the annotation matches a given event.
+*/
+Transaction.userEvent = Annotation.define();
+/**
+Annotation indicating whether a transaction should be added to
+the undo history or not.
+*/
+Transaction.addToHistory = Annotation.define();
+/**
+Annotation indicating (when present and true) that a transaction
+represents a change made by some other actor, not the user. This
+is used, for example, to tag other people's changes in
+collaborative editing.
+*/
+Transaction.remote = Annotation.define();
+function joinRanges(a, b) {
+ let result = [];
+ for (let iA = 0, iB = 0;;) {
+ let from, to;
+ if (iA < a.length && (iB == b.length || b[iB] >= a[iA])) {
+ from = a[iA++];
+ to = a[iA++];
+ }
+ else if (iB < b.length) {
+ from = b[iB++];
+ to = b[iB++];
+ }
+ else
+ return result;
+ if (!result.length || result[result.length - 1] < from)
+ result.push(from, to);
+ else if (result[result.length - 1] < to)
+ result[result.length - 1] = to;
+ }
+}
+function mergeTransaction(a, b, sequential) {
+ var _a;
+ let mapForA, mapForB, changes;
+ if (sequential) {
+ mapForA = b.changes;
+ mapForB = ChangeSet.empty(b.changes.length);
+ changes = a.changes.compose(b.changes);
+ }
+ else {
+ mapForA = b.changes.map(a.changes);
+ mapForB = a.changes.mapDesc(b.changes, true);
+ changes = a.changes.compose(mapForA);
+ }
+ return {
+ changes,
+ selection: b.selection ? b.selection.map(mapForB) : (_a = a.selection) === null || _a === void 0 ? void 0 : _a.map(mapForA),
+ effects: StateEffect.mapEffects(a.effects, mapForA).concat(StateEffect.mapEffects(b.effects, mapForB)),
+ annotations: a.annotations.length ? a.annotations.concat(b.annotations) : b.annotations,
+ scrollIntoView: a.scrollIntoView || b.scrollIntoView
+ };
+}
+function resolveTransactionInner(state, spec, docSize) {
+ let sel = spec.selection, annotations = asArray(spec.annotations);
+ if (spec.userEvent)
+ annotations = annotations.concat(Transaction.userEvent.of(spec.userEvent));
+ return {
+ changes: spec.changes instanceof ChangeSet ? spec.changes
+ : ChangeSet.of(spec.changes || [], docSize, state.facet(lineSeparator)),
+ selection: sel && (sel instanceof EditorSelection ? sel : EditorSelection.single(sel.anchor, sel.head)),
+ effects: asArray(spec.effects),
+ annotations,
+ scrollIntoView: !!spec.scrollIntoView
+ };
+}
+function resolveTransaction(state, specs, filter) {
+ let s = resolveTransactionInner(state, specs.length ? specs[0] : {}, state.doc.length);
+ if (specs.length && specs[0].filter === false)
+ filter = false;
+ for (let i = 1; i < specs.length; i++) {
+ if (specs[i].filter === false)
+ filter = false;
+ let seq = !!specs[i].sequential;
+ s = mergeTransaction(s, resolveTransactionInner(state, specs[i], seq ? s.changes.newLength : state.doc.length), seq);
+ }
+ let tr = Transaction.create(state, s.changes, s.selection, s.effects, s.annotations, s.scrollIntoView);
+ return extendTransaction(filter ? filterTransaction(tr) : tr);
+}
+// Finish a transaction by applying filters if necessary.
+function filterTransaction(tr) {
+ let state = tr.startState;
+ // Change filters
+ let result = true;
+ for (let filter of state.facet(changeFilter)) {
+ let value = filter(tr);
+ if (value === false) {
+ result = false;
+ break;
+ }
+ if (Array.isArray(value))
+ result = result === true ? value : joinRanges(result, value);
+ }
+ if (result !== true) {
+ let changes, back;
+ if (result === false) {
+ back = tr.changes.invertedDesc;
+ changes = ChangeSet.empty(state.doc.length);
+ }
+ else {
+ let filtered = tr.changes.filter(result);
+ changes = filtered.changes;
+ back = filtered.filtered.mapDesc(filtered.changes).invertedDesc;
+ }
+ tr = Transaction.create(state, changes, tr.selection && tr.selection.map(back), StateEffect.mapEffects(tr.effects, back), tr.annotations, tr.scrollIntoView);
+ }
+ // Transaction filters
+ let filters = state.facet(transactionFilter);
+ for (let i = filters.length - 1; i >= 0; i--) {
+ let filtered = filters[i](tr);
+ if (filtered instanceof Transaction)
+ tr = filtered;
+ else if (Array.isArray(filtered) && filtered.length == 1 && filtered[0] instanceof Transaction)
+ tr = filtered[0];
+ else
+ tr = resolveTransaction(state, asArray(filtered), false);
+ }
+ return tr;
+}
+function extendTransaction(tr) {
+ let state = tr.startState, extenders = state.facet(transactionExtender), spec = tr;
+ for (let i = extenders.length - 1; i >= 0; i--) {
+ let extension = extenders[i](tr);
+ if (extension && Object.keys(extension).length)
+ spec = mergeTransaction(spec, resolveTransactionInner(state, extension, tr.changes.newLength), true);
+ }
+ return spec == tr ? tr : Transaction.create(state, tr.changes, tr.selection, spec.effects, spec.annotations, spec.scrollIntoView);
+}
+const none = [];
+function asArray(value) {
+ return value == null ? none : Array.isArray(value) ? value : [value];
+}
+
+/**
+The categories produced by a [character
+categorizer](https://codemirror.net/6/docs/ref/#state.EditorState.charCategorizer). These are used
+do things like selecting by word.
+*/
+exports.CharCategory = void 0;
+(function (CharCategory) {
+ /**
+ Word characters.
+ */
+ CharCategory[CharCategory["Word"] = 0] = "Word";
+ /**
+ Whitespace.
+ */
+ CharCategory[CharCategory["Space"] = 1] = "Space";
+ /**
+ Anything else.
+ */
+ CharCategory[CharCategory["Other"] = 2] = "Other";
+})(exports.CharCategory || (exports.CharCategory = {}));
+const nonASCIISingleCaseWordChar = /[\u00df\u0587\u0590-\u05f4\u0600-\u06ff\u3040-\u309f\u30a0-\u30ff\u3400-\u4db5\u4e00-\u9fcc\uac00-\ud7af]/;
+let wordChar;
+try {
+ wordChar = new RegExp("[\\p{Alphabetic}\\p{Number}_]", "u");
+}
+catch (_) { }
+function hasWordChar(str) {
+ if (wordChar)
+ return wordChar.test(str);
+ for (let i = 0; i < str.length; i++) {
+ let ch = str[i];
+ if (/\w/.test(ch) || ch > "\x80" && (ch.toUpperCase() != ch.toLowerCase() || nonASCIISingleCaseWordChar.test(ch)))
+ return true;
+ }
+ return false;
+}
+function makeCategorizer(wordChars) {
+ return (char) => {
+ if (!/\S/.test(char))
+ return exports.CharCategory.Space;
+ if (hasWordChar(char))
+ return exports.CharCategory.Word;
+ for (let i = 0; i < wordChars.length; i++)
+ if (char.indexOf(wordChars[i]) > -1)
+ return exports.CharCategory.Word;
+ return exports.CharCategory.Other;
+ };
+}
+
+/**
+The editor state class is a persistent (immutable) data structure.
+To update a state, you [create](https://codemirror.net/6/docs/ref/#state.EditorState.update) a
+[transaction](https://codemirror.net/6/docs/ref/#state.Transaction), which produces a _new_ state
+instance, without modifying the original object.
+
+As such, _never_ mutate properties of a state directly. That'll
+just break things.
+*/
+class EditorState {
+ constructor(
+ /**
+ @internal
+ */
+ config,
+ /**
+ The current document.
+ */
+ doc,
+ /**
+ The current selection.
+ */
+ selection,
+ /**
+ @internal
+ */
+ values, computeSlot, tr) {
+ this.config = config;
+ this.doc = doc;
+ this.selection = selection;
+ this.values = values;
+ this.status = config.statusTemplate.slice();
+ this.computeSlot = computeSlot;
+ // Fill in the computed state immediately, so that further queries
+ // for it made during the update return this state
+ if (tr)
+ tr._state = this;
+ for (let i = 0; i < this.config.dynamicSlots.length; i++)
+ ensureAddr(this, i << 1);
+ this.computeSlot = null;
+ }
+ field(field, require = true) {
+ let addr = this.config.address[field.id];
+ if (addr == null) {
+ if (require)
+ throw new RangeError("Field is not present in this state");
+ return undefined;
+ }
+ ensureAddr(this, addr);
+ return getAddr(this, addr);
+ }
+ /**
+ Create a [transaction](https://codemirror.net/6/docs/ref/#state.Transaction) that updates this
+ state. Any number of [transaction specs](https://codemirror.net/6/docs/ref/#state.TransactionSpec)
+ can be passed. Unless
+ [`sequential`](https://codemirror.net/6/docs/ref/#state.TransactionSpec.sequential) is set, the
+ [changes](https://codemirror.net/6/docs/ref/#state.TransactionSpec.changes) (if any) of each spec
+ are assumed to start in the _current_ document (not the document
+ produced by previous specs), and its
+ [selection](https://codemirror.net/6/docs/ref/#state.TransactionSpec.selection) and
+ [effects](https://codemirror.net/6/docs/ref/#state.TransactionSpec.effects) are assumed to refer
+ to the document created by its _own_ changes. The resulting
+ transaction contains the combined effect of all the different
+ specs. For [selection](https://codemirror.net/6/docs/ref/#state.TransactionSpec.selection), later
+ specs take precedence over earlier ones.
+ */
+ update(...specs) {
+ return resolveTransaction(this, specs, true);
+ }
+ /**
+ @internal
+ */
+ applyTransaction(tr) {
+ let conf = this.config, { base, compartments } = conf;
+ for (let effect of tr.effects) {
+ if (effect.is(Compartment.reconfigure)) {
+ if (conf) {
+ compartments = new Map;
+ conf.compartments.forEach((val, key) => compartments.set(key, val));
+ conf = null;
+ }
+ compartments.set(effect.value.compartment, effect.value.extension);
+ }
+ else if (effect.is(StateEffect.reconfigure)) {
+ conf = null;
+ base = effect.value;
+ }
+ else if (effect.is(StateEffect.appendConfig)) {
+ conf = null;
+ base = asArray(base).concat(effect.value);
+ }
+ }
+ let startValues;
+ if (!conf) {
+ conf = Configuration.resolve(base, compartments, this);
+ let intermediateState = new EditorState(conf, this.doc, this.selection, conf.dynamicSlots.map(() => null), (state, slot) => slot.reconfigure(state, this), null);
+ startValues = intermediateState.values;
+ }
+ else {
+ startValues = tr.startState.values.slice();
+ }
+ let selection = tr.startState.facet(allowMultipleSelections) ? tr.newSelection : tr.newSelection.asSingle();
+ new EditorState(conf, tr.newDoc, selection, startValues, (state, slot) => slot.update(state, tr), tr);
+ }
+ /**
+ Create a [transaction spec](https://codemirror.net/6/docs/ref/#state.TransactionSpec) that
+ replaces every selection range with the given content.
+ */
+ replaceSelection(text) {
+ if (typeof text == "string")
+ text = this.toText(text);
+ return this.changeByRange(range => ({ changes: { from: range.from, to: range.to, insert: text },
+ range: EditorSelection.cursor(range.from + text.length) }));
+ }
+ /**
+ Create a set of changes and a new selection by running the given
+ function for each range in the active selection. The function
+ can return an optional set of changes (in the coordinate space
+ of the start document), plus an updated range (in the coordinate
+ space of the document produced by the call's own changes). This
+ method will merge all the changes and ranges into a single
+ changeset and selection, and return it as a [transaction
+ spec](https://codemirror.net/6/docs/ref/#state.TransactionSpec), which can be passed to
+ [`update`](https://codemirror.net/6/docs/ref/#state.EditorState.update).
+ */
+ changeByRange(f) {
+ let sel = this.selection;
+ let result1 = f(sel.ranges[0]);
+ let changes = this.changes(result1.changes), ranges = [result1.range];
+ let effects = asArray(result1.effects);
+ for (let i = 1; i < sel.ranges.length; i++) {
+ let result = f(sel.ranges[i]);
+ let newChanges = this.changes(result.changes), newMapped = newChanges.map(changes);
+ for (let j = 0; j < i; j++)
+ ranges[j] = ranges[j].map(newMapped);
+ let mapBy = changes.mapDesc(newChanges, true);
+ ranges.push(result.range.map(mapBy));
+ changes = changes.compose(newMapped);
+ effects = StateEffect.mapEffects(effects, newMapped).concat(StateEffect.mapEffects(asArray(result.effects), mapBy));
+ }
+ return {
+ changes,
+ selection: EditorSelection.create(ranges, sel.mainIndex),
+ effects
+ };
+ }
+ /**
+ Create a [change set](https://codemirror.net/6/docs/ref/#state.ChangeSet) from the given change
+ description, taking the state's document length and line
+ separator into account.
+ */
+ changes(spec = []) {
+ if (spec instanceof ChangeSet)
+ return spec;
+ return ChangeSet.of(spec, this.doc.length, this.facet(EditorState.lineSeparator));
+ }
+ /**
+ Using the state's [line
+ separator](https://codemirror.net/6/docs/ref/#state.EditorState^lineSeparator), create a
+ [`Text`](https://codemirror.net/6/docs/ref/#state.Text) instance from the given string.
+ */
+ toText(string) {
+ return Text.of(string.split(this.facet(EditorState.lineSeparator) || DefaultSplit));
+ }
+ /**
+ Return the given range of the document as a string.
+ */
+ sliceDoc(from = 0, to = this.doc.length) {
+ return this.doc.sliceString(from, to, this.lineBreak);
+ }
+ /**
+ Get the value of a state [facet](https://codemirror.net/6/docs/ref/#state.Facet).
+ */
+ facet(facet) {
+ let addr = this.config.address[facet.id];
+ if (addr == null)
+ return facet.default;
+ ensureAddr(this, addr);
+ return getAddr(this, addr);
+ }
+ /**
+ Convert this state to a JSON-serializable object. When custom
+ fields should be serialized, you can pass them in as an object
+ mapping property names (in the resulting object, which should
+ not use `doc` or `selection`) to fields.
+ */
+ toJSON(fields) {
+ let result = {
+ doc: this.sliceDoc(),
+ selection: this.selection.toJSON()
+ };
+ if (fields)
+ for (let prop in fields) {
+ let value = fields[prop];
+ if (value instanceof StateField && this.config.address[value.id] != null)
+ result[prop] = value.spec.toJSON(this.field(fields[prop]), this);
+ }
+ return result;
+ }
+ /**
+ Deserialize a state from its JSON representation. When custom
+ fields should be deserialized, pass the same object you passed
+ to [`toJSON`](https://codemirror.net/6/docs/ref/#state.EditorState.toJSON) when serializing as
+ third argument.
+ */
+ static fromJSON(json, config = {}, fields) {
+ if (!json || typeof json.doc != "string")
+ throw new RangeError("Invalid JSON representation for EditorState");
+ let fieldInit = [];
+ if (fields)
+ for (let prop in fields) {
+ if (Object.prototype.hasOwnProperty.call(json, prop)) {
+ let field = fields[prop], value = json[prop];
+ fieldInit.push(field.init(state => field.spec.fromJSON(value, state)));
+ }
+ }
+ return EditorState.create({
+ doc: json.doc,
+ selection: EditorSelection.fromJSON(json.selection),
+ extensions: config.extensions ? fieldInit.concat([config.extensions]) : fieldInit
+ });
+ }
+ /**
+ Create a new state. You'll usually only need this when
+ initializing an editor—updated states are created by applying
+ transactions.
+ */
+ static create(config = {}) {
+ let configuration = Configuration.resolve(config.extensions || [], new Map);
+ let doc = config.doc instanceof Text ? config.doc
+ : Text.of((config.doc || "").split(configuration.staticFacet(EditorState.lineSeparator) || DefaultSplit));
+ let selection = !config.selection ? EditorSelection.single(0)
+ : config.selection instanceof EditorSelection ? config.selection
+ : EditorSelection.single(config.selection.anchor, config.selection.head);
+ checkSelection(selection, doc.length);
+ if (!configuration.staticFacet(allowMultipleSelections))
+ selection = selection.asSingle();
+ return new EditorState(configuration, doc, selection, configuration.dynamicSlots.map(() => null), (state, slot) => slot.create(state), null);
+ }
+ /**
+ The size (in columns) of a tab in the document, determined by
+ the [`tabSize`](https://codemirror.net/6/docs/ref/#state.EditorState^tabSize) facet.
+ */
+ get tabSize() { return this.facet(EditorState.tabSize); }
+ /**
+ Get the proper [line-break](https://codemirror.net/6/docs/ref/#state.EditorState^lineSeparator)
+ string for this state.
+ */
+ get lineBreak() { return this.facet(EditorState.lineSeparator) || "\n"; }
+ /**
+ Returns true when the editor is
+ [configured](https://codemirror.net/6/docs/ref/#state.EditorState^readOnly) to be read-only.
+ */
+ get readOnly() { return this.facet(readOnly); }
+ /**
+ Look up a translation for the given phrase (via the
+ [`phrases`](https://codemirror.net/6/docs/ref/#state.EditorState^phrases) facet), or return the
+ original string if no translation is found.
+
+ If additional arguments are passed, they will be inserted in
+ place of markers like `$1` (for the first value) and `$2`, etc.
+ A single `$` is equivalent to `$1`, and `$$` will produce a
+ literal dollar sign.
+ */
+ phrase(phrase, ...insert) {
+ for (let map of this.facet(EditorState.phrases))
+ if (Object.prototype.hasOwnProperty.call(map, phrase)) {
+ phrase = map[phrase];
+ break;
+ }
+ if (insert.length)
+ phrase = phrase.replace(/\$(\$|\d*)/g, (m, i) => {
+ if (i == "$")
+ return "$";
+ let n = +(i || 1);
+ return !n || n > insert.length ? m : insert[n - 1];
+ });
+ return phrase;
+ }
+ /**
+ Find the values for a given language data field, provided by the
+ the [`languageData`](https://codemirror.net/6/docs/ref/#state.EditorState^languageData) facet.
+
+ Examples of language data fields are...
+
+ - [`"commentTokens"`](https://codemirror.net/6/docs/ref/#commands.CommentTokens) for specifying
+ comment syntax.
+ - [`"autocomplete"`](https://codemirror.net/6/docs/ref/#autocomplete.autocompletion^config.override)
+ for providing language-specific completion sources.
+ - [`"wordChars"`](https://codemirror.net/6/docs/ref/#state.EditorState.charCategorizer) for adding
+ characters that should be considered part of words in this
+ language.
+ - [`"closeBrackets"`](https://codemirror.net/6/docs/ref/#autocomplete.CloseBracketConfig) controls
+ bracket closing behavior.
+ */
+ languageDataAt(name, pos, side = -1) {
+ let values = [];
+ for (let provider of this.facet(languageData)) {
+ for (let result of provider(this, pos, side)) {
+ if (Object.prototype.hasOwnProperty.call(result, name))
+ values.push(result[name]);
+ }
+ }
+ return values;
+ }
+ /**
+ Return a function that can categorize strings (expected to
+ represent a single [grapheme cluster](https://codemirror.net/6/docs/ref/#state.findClusterBreak))
+ into one of:
+
+ - Word (contains an alphanumeric character or a character
+ explicitly listed in the local language's `"wordChars"`
+ language data, which should be a string)
+ - Space (contains only whitespace)
+ - Other (anything else)
+ */
+ charCategorizer(at) {
+ return makeCategorizer(this.languageDataAt("wordChars", at).join(""));
+ }
+ /**
+ Find the word at the given position, meaning the range
+ containing all [word](https://codemirror.net/6/docs/ref/#state.CharCategory.Word) characters
+ around it. If no word characters are adjacent to the position,
+ this returns null.
+ */
+ wordAt(pos) {
+ let { text, from, length } = this.doc.lineAt(pos);
+ let cat = this.charCategorizer(pos);
+ let start = pos - from, end = pos - from;
+ while (start > 0) {
+ let prev = findClusterBreak(text, start, false);
+ if (cat(text.slice(prev, start)) != exports.CharCategory.Word)
+ break;
+ start = prev;
+ }
+ while (end < length) {
+ let next = findClusterBreak(text, end);
+ if (cat(text.slice(end, next)) != exports.CharCategory.Word)
+ break;
+ end = next;
+ }
+ return start == end ? null : EditorSelection.range(start + from, end + from);
+ }
+}
+/**
+A facet that, when enabled, causes the editor to allow multiple
+ranges to be selected. Be careful though, because by default the
+editor relies on the native DOM selection, which cannot handle
+multiple selections. An extension like
+[`drawSelection`](https://codemirror.net/6/docs/ref/#view.drawSelection) can be used to make
+secondary selections visible to the user.
+*/
+EditorState.allowMultipleSelections = allowMultipleSelections;
+/**
+Configures the tab size to use in this state. The first
+(highest-precedence) value of the facet is used. If no value is
+given, this defaults to 4.
+*/
+EditorState.tabSize = Facet.define({
+ combine: values => values.length ? values[0] : 4
+});
+/**
+The line separator to use. By default, any of `"\n"`, `"\r\n"`
+and `"\r"` is treated as a separator when splitting lines, and
+lines are joined with `"\n"`.
+
+When you configure a value here, only that precise separator
+will be used, allowing you to round-trip documents through the
+editor without normalizing line separators.
+*/
+EditorState.lineSeparator = lineSeparator;
+/**
+This facet controls the value of the
+[`readOnly`](https://codemirror.net/6/docs/ref/#state.EditorState.readOnly) getter, which is
+consulted by commands and extensions that implement editing
+functionality to determine whether they should apply. It
+defaults to false, but when its highest-precedence value is
+`true`, such functionality disables itself.
+
+Not to be confused with
+[`EditorView.editable`](https://codemirror.net/6/docs/ref/#view.EditorView^editable), which
+controls whether the editor's DOM is set to be editable (and
+thus focusable).
+*/
+EditorState.readOnly = readOnly;
+/**
+Registers translation phrases. The
+[`phrase`](https://codemirror.net/6/docs/ref/#state.EditorState.phrase) method will look through
+all objects registered with this facet to find translations for
+its argument.
+*/
+EditorState.phrases = Facet.define({
+ compare(a, b) {
+ let kA = Object.keys(a), kB = Object.keys(b);
+ return kA.length == kB.length && kA.every(k => a[k] == b[k]);
+ }
+});
+/**
+A facet used to register [language
+data](https://codemirror.net/6/docs/ref/#state.EditorState.languageDataAt) providers.
+*/
+EditorState.languageData = languageData;
+/**
+Facet used to register change filters, which are called for each
+transaction (unless explicitly
+[disabled](https://codemirror.net/6/docs/ref/#state.TransactionSpec.filter)), and can suppress
+part of the transaction's changes.
+
+Such a function can return `true` to indicate that it doesn't
+want to do anything, `false` to completely stop the changes in
+the transaction, or a set of ranges in which changes should be
+suppressed. Such ranges are represented as an array of numbers,
+with each pair of two numbers indicating the start and end of a
+range. So for example `[10, 20, 100, 110]` suppresses changes
+between 10 and 20, and between 100 and 110.
+*/
+EditorState.changeFilter = changeFilter;
+/**
+Facet used to register a hook that gets a chance to update or
+replace transaction specs before they are applied. This will
+only be applied for transactions that don't have
+[`filter`](https://codemirror.net/6/docs/ref/#state.TransactionSpec.filter) set to `false`. You
+can either return a single transaction spec (possibly the input
+transaction), or an array of specs (which will be combined in
+the same way as the arguments to
+[`EditorState.update`](https://codemirror.net/6/docs/ref/#state.EditorState.update)).
+
+When possible, it is recommended to avoid accessing
+[`Transaction.state`](https://codemirror.net/6/docs/ref/#state.Transaction.state) in a filter,
+since it will force creation of a state that will then be
+discarded again, if the transaction is actually filtered.
+
+(This functionality should be used with care. Indiscriminately
+modifying transaction is likely to break something or degrade
+the user experience.)
+*/
+EditorState.transactionFilter = transactionFilter;
+/**
+This is a more limited form of
+[`transactionFilter`](https://codemirror.net/6/docs/ref/#state.EditorState^transactionFilter),
+which can only add
+[annotations](https://codemirror.net/6/docs/ref/#state.TransactionSpec.annotations) and
+[effects](https://codemirror.net/6/docs/ref/#state.TransactionSpec.effects). _But_, this type
+of filter runs even if the transaction has disabled regular
+[filtering](https://codemirror.net/6/docs/ref/#state.TransactionSpec.filter), making it suitable
+for effects that don't need to touch the changes or selection,
+but do want to process every transaction.
+
+Extenders run _after_ filters, when both are present.
+*/
+EditorState.transactionExtender = transactionExtender;
+Compartment.reconfigure = StateEffect.define();
+
+/**
+Utility function for combining behaviors to fill in a config
+object from an array of provided configs. `defaults` should hold
+default values for all optional fields in `Config`.
+
+The function will, by default, error
+when a field gets two values that aren't `===`-equal, but you can
+provide combine functions per field to do something else.
+*/
+function combineConfig(configs, defaults, // Should hold only the optional properties of Config, but I haven't managed to express that
+combine = {}) {
+ let result = {};
+ for (let config of configs)
+ for (let key of Object.keys(config)) {
+ let value = config[key], current = result[key];
+ if (current === undefined)
+ result[key] = value;
+ else if (current === value || value === undefined) ; // No conflict
+ else if (Object.hasOwnProperty.call(combine, key))
+ result[key] = combine[key](current, value);
+ else
+ throw new Error("Config merge conflict for field " + key);
+ }
+ for (let key in defaults)
+ if (result[key] === undefined)
+ result[key] = defaults[key];
+ return result;
+}
+
+/**
+Each range is associated with a value, which must inherit from
+this class.
+*/
+class RangeValue {
+ /**
+ Compare this value with another value. Used when comparing
+ rangesets. The default implementation compares by identity.
+ Unless you are only creating a fixed number of unique instances
+ of your value type, it is a good idea to implement this
+ properly.
+ */
+ eq(other) { return this == other; }
+ /**
+ Create a [range](https://codemirror.net/6/docs/ref/#state.Range) with this value.
+ */
+ range(from, to = from) { return Range.create(from, to, this); }
+}
+RangeValue.prototype.startSide = RangeValue.prototype.endSide = 0;
+RangeValue.prototype.point = false;
+RangeValue.prototype.mapMode = exports.MapMode.TrackDel;
+/**
+A range associates a value with a range of positions.
+*/
+class Range {
+ constructor(
+ /**
+ The range's start position.
+ */
+ from,
+ /**
+ Its end position.
+ */
+ to,
+ /**
+ The value associated with this range.
+ */
+ value) {
+ this.from = from;
+ this.to = to;
+ this.value = value;
+ }
+ /**
+ @internal
+ */
+ static create(from, to, value) {
+ return new Range(from, to, value);
+ }
+}
+function cmpRange(a, b) {
+ return a.from - b.from || a.value.startSide - b.value.startSide;
+}
+class Chunk {
+ constructor(from, to, value,
+ // Chunks are marked with the largest point that occurs
+ // in them (or -1 for no points), so that scans that are
+ // only interested in points (such as the
+ // heightmap-related logic) can skip range-only chunks.
+ maxPoint) {
+ this.from = from;
+ this.to = to;
+ this.value = value;
+ this.maxPoint = maxPoint;
+ }
+ get length() { return this.to[this.to.length - 1]; }
+ // Find the index of the given position and side. Use the ranges'
+ // `from` pos when `end == false`, `to` when `end == true`.
+ findIndex(pos, side, end, startAt = 0) {
+ let arr = end ? this.to : this.from;
+ for (let lo = startAt, hi = arr.length;;) {
+ if (lo == hi)
+ return lo;
+ let mid = (lo + hi) >> 1;
+ let diff = arr[mid] - pos || (end ? this.value[mid].endSide : this.value[mid].startSide) - side;
+ if (mid == lo)
+ return diff >= 0 ? lo : hi;
+ if (diff >= 0)
+ hi = mid;
+ else
+ lo = mid + 1;
+ }
+ }
+ between(offset, from, to, f) {
+ for (let i = this.findIndex(from, -1000000000 /* C.Far */, true), e = this.findIndex(to, 1000000000 /* C.Far */, false, i); i < e; i++)
+ if (f(this.from[i] + offset, this.to[i] + offset, this.value[i]) === false)
+ return false;
+ }
+ map(offset, changes) {
+ let value = [], from = [], to = [], newPos = -1, maxPoint = -1;
+ for (let i = 0; i < this.value.length; i++) {
+ let val = this.value[i], curFrom = this.from[i] + offset, curTo = this.to[i] + offset, newFrom, newTo;
+ if (curFrom == curTo) {
+ let mapped = changes.mapPos(curFrom, val.startSide, val.mapMode);
+ if (mapped == null)
+ continue;
+ newFrom = newTo = mapped;
+ if (val.startSide != val.endSide) {
+ newTo = changes.mapPos(curFrom, val.endSide);
+ if (newTo < newFrom)
+ continue;
+ }
+ }
+ else {
+ newFrom = changes.mapPos(curFrom, val.startSide);
+ newTo = changes.mapPos(curTo, val.endSide);
+ if (newFrom > newTo || newFrom == newTo && val.startSide > 0 && val.endSide <= 0)
+ continue;
+ }
+ if ((newTo - newFrom || val.endSide - val.startSide) < 0)
+ continue;
+ if (newPos < 0)
+ newPos = newFrom;
+ if (val.point)
+ maxPoint = Math.max(maxPoint, newTo - newFrom);
+ value.push(val);
+ from.push(newFrom - newPos);
+ to.push(newTo - newPos);
+ }
+ return { mapped: value.length ? new Chunk(from, to, value, maxPoint) : null, pos: newPos };
+ }
+}
+/**
+A range set stores a collection of [ranges](https://codemirror.net/6/docs/ref/#state.Range) in a
+way that makes them efficient to [map](https://codemirror.net/6/docs/ref/#state.RangeSet.map) and
+[update](https://codemirror.net/6/docs/ref/#state.RangeSet.update). This is an immutable data
+structure.
+*/
+class RangeSet {
+ constructor(
+ /**
+ @internal
+ */
+ chunkPos,
+ /**
+ @internal
+ */
+ chunk,
+ /**
+ @internal
+ */
+ nextLayer,
+ /**
+ @internal
+ */
+ maxPoint) {
+ this.chunkPos = chunkPos;
+ this.chunk = chunk;
+ this.nextLayer = nextLayer;
+ this.maxPoint = maxPoint;
+ }
+ /**
+ @internal
+ */
+ static create(chunkPos, chunk, nextLayer, maxPoint) {
+ return new RangeSet(chunkPos, chunk, nextLayer, maxPoint);
+ }
+ /**
+ @internal
+ */
+ get length() {
+ let last = this.chunk.length - 1;
+ return last < 0 ? 0 : Math.max(this.chunkEnd(last), this.nextLayer.length);
+ }
+ /**
+ The number of ranges in the set.
+ */
+ get size() {
+ if (this.isEmpty)
+ return 0;
+ let size = this.nextLayer.size;
+ for (let chunk of this.chunk)
+ size += chunk.value.length;
+ return size;
+ }
+ /**
+ @internal
+ */
+ chunkEnd(index) {
+ return this.chunkPos[index] + this.chunk[index].length;
+ }
+ /**
+ Update the range set, optionally adding new ranges or filtering
+ out existing ones.
+
+ (Note: The type parameter is just there as a kludge to work
+ around TypeScript variance issues that prevented `RangeSet`
+ from being a subtype of `RangeSet` when `X` is a subtype of
+ `Y`.)
+ */
+ update(updateSpec) {
+ let { add = [], sort = false, filterFrom = 0, filterTo = this.length } = updateSpec;
+ let filter = updateSpec.filter;
+ if (add.length == 0 && !filter)
+ return this;
+ if (sort)
+ add = add.slice().sort(cmpRange);
+ if (this.isEmpty)
+ return add.length ? RangeSet.of(add) : this;
+ let cur = new LayerCursor(this, null, -1).goto(0), i = 0, spill = [];
+ let builder = new RangeSetBuilder();
+ while (cur.value || i < add.length) {
+ if (i < add.length && (cur.from - add[i].from || cur.startSide - add[i].value.startSide) >= 0) {
+ let range = add[i++];
+ if (!builder.addInner(range.from, range.to, range.value))
+ spill.push(range);
+ }
+ else if (cur.rangeIndex == 1 && cur.chunkIndex < this.chunk.length &&
+ (i == add.length || this.chunkEnd(cur.chunkIndex) < add[i].from) &&
+ (!filter || filterFrom > this.chunkEnd(cur.chunkIndex) || filterTo < this.chunkPos[cur.chunkIndex]) &&
+ builder.addChunk(this.chunkPos[cur.chunkIndex], this.chunk[cur.chunkIndex])) {
+ cur.nextChunk();
+ }
+ else {
+ if (!filter || filterFrom > cur.to || filterTo < cur.from || filter(cur.from, cur.to, cur.value)) {
+ if (!builder.addInner(cur.from, cur.to, cur.value))
+ spill.push(Range.create(cur.from, cur.to, cur.value));
+ }
+ cur.next();
+ }
+ }
+ return builder.finishInner(this.nextLayer.isEmpty && !spill.length ? RangeSet.empty
+ : this.nextLayer.update({ add: spill, filter, filterFrom, filterTo }));
+ }
+ /**
+ Map this range set through a set of changes, return the new set.
+ */
+ map(changes) {
+ if (changes.empty || this.isEmpty)
+ return this;
+ let chunks = [], chunkPos = [], maxPoint = -1;
+ for (let i = 0; i < this.chunk.length; i++) {
+ let start = this.chunkPos[i], chunk = this.chunk[i];
+ let touch = changes.touchesRange(start, start + chunk.length);
+ if (touch === false) {
+ maxPoint = Math.max(maxPoint, chunk.maxPoint);
+ chunks.push(chunk);
+ chunkPos.push(changes.mapPos(start));
+ }
+ else if (touch === true) {
+ let { mapped, pos } = chunk.map(start, changes);
+ if (mapped) {
+ maxPoint = Math.max(maxPoint, mapped.maxPoint);
+ chunks.push(mapped);
+ chunkPos.push(pos);
+ }
+ }
+ }
+ let next = this.nextLayer.map(changes);
+ return chunks.length == 0 ? next : new RangeSet(chunkPos, chunks, next || RangeSet.empty, maxPoint);
+ }
+ /**
+ Iterate over the ranges that touch the region `from` to `to`,
+ calling `f` for each. There is no guarantee that the ranges will
+ be reported in any specific order. When the callback returns
+ `false`, iteration stops.
+ */
+ between(from, to, f) {
+ if (this.isEmpty)
+ return;
+ for (let i = 0; i < this.chunk.length; i++) {
+ let start = this.chunkPos[i], chunk = this.chunk[i];
+ if (to >= start && from <= start + chunk.length &&
+ chunk.between(start, from - start, to - start, f) === false)
+ return;
+ }
+ this.nextLayer.between(from, to, f);
+ }
+ /**
+ Iterate over the ranges in this set, in order, including all
+ ranges that end at or after `from`.
+ */
+ iter(from = 0) {
+ return HeapCursor.from([this]).goto(from);
+ }
+ /**
+ @internal
+ */
+ get isEmpty() { return this.nextLayer == this; }
+ /**
+ Iterate over the ranges in a collection of sets, in order,
+ starting from `from`.
+ */
+ static iter(sets, from = 0) {
+ return HeapCursor.from(sets).goto(from);
+ }
+ /**
+ Iterate over two groups of sets, calling methods on `comparator`
+ to notify it of possible differences.
+ */
+ static compare(oldSets, newSets,
+ /**
+ This indicates how the underlying data changed between these
+ ranges, and is needed to synchronize the iteration.
+ */
+ textDiff, comparator,
+ /**
+ Can be used to ignore all non-point ranges, and points below
+ the given size. When -1, all ranges are compared.
+ */
+ minPointSize = -1) {
+ let a = oldSets.filter(set => set.maxPoint > 0 || !set.isEmpty && set.maxPoint >= minPointSize);
+ let b = newSets.filter(set => set.maxPoint > 0 || !set.isEmpty && set.maxPoint >= minPointSize);
+ let sharedChunks = findSharedChunks(a, b, textDiff);
+ let sideA = new SpanCursor(a, sharedChunks, minPointSize);
+ let sideB = new SpanCursor(b, sharedChunks, minPointSize);
+ textDiff.iterGaps((fromA, fromB, length) => compare(sideA, fromA, sideB, fromB, length, comparator));
+ if (textDiff.empty && textDiff.length == 0)
+ compare(sideA, 0, sideB, 0, 0, comparator);
+ }
+ /**
+ Compare the contents of two groups of range sets, returning true
+ if they are equivalent in the given range.
+ */
+ static eq(oldSets, newSets, from = 0, to) {
+ if (to == null)
+ to = 1000000000 /* C.Far */ - 1;
+ let a = oldSets.filter(set => !set.isEmpty && newSets.indexOf(set) < 0);
+ let b = newSets.filter(set => !set.isEmpty && oldSets.indexOf(set) < 0);
+ if (a.length != b.length)
+ return false;
+ if (!a.length)
+ return true;
+ let sharedChunks = findSharedChunks(a, b);
+ let sideA = new SpanCursor(a, sharedChunks, 0).goto(from), sideB = new SpanCursor(b, sharedChunks, 0).goto(from);
+ for (;;) {
+ if (sideA.to != sideB.to ||
+ !sameValues(sideA.active, sideB.active) ||
+ sideA.point && (!sideB.point || !sideA.point.eq(sideB.point)))
+ return false;
+ if (sideA.to > to)
+ return true;
+ sideA.next();
+ sideB.next();
+ }
+ }
+ /**
+ Iterate over a group of range sets at the same time, notifying
+ the iterator about the ranges covering every given piece of
+ content. Returns the open count (see
+ [`SpanIterator.span`](https://codemirror.net/6/docs/ref/#state.SpanIterator.span)) at the end
+ of the iteration.
+ */
+ static spans(sets, from, to, iterator,
+ /**
+ When given and greater than -1, only points of at least this
+ size are taken into account.
+ */
+ minPointSize = -1) {
+ let cursor = new SpanCursor(sets, null, minPointSize).goto(from), pos = from;
+ let openRanges = cursor.openStart;
+ for (;;) {
+ let curTo = Math.min(cursor.to, to);
+ if (cursor.point) {
+ let active = cursor.activeForPoint(cursor.to);
+ let openCount = cursor.pointFrom < from ? active.length + 1
+ : cursor.point.startSide < 0 ? active.length
+ : Math.min(active.length, openRanges);
+ iterator.point(pos, curTo, cursor.point, active, openCount, cursor.pointRank);
+ openRanges = Math.min(cursor.openEnd(curTo), active.length);
+ }
+ else if (curTo > pos) {
+ iterator.span(pos, curTo, cursor.active, openRanges);
+ openRanges = cursor.openEnd(curTo);
+ }
+ if (cursor.to > to)
+ return openRanges + (cursor.point && cursor.to > to ? 1 : 0);
+ pos = cursor.to;
+ cursor.next();
+ }
+ }
+ /**
+ Create a range set for the given range or array of ranges. By
+ default, this expects the ranges to be _sorted_ (by start
+ position and, if two start at the same position,
+ `value.startSide`). You can pass `true` as second argument to
+ cause the method to sort them.
+ */
+ static of(ranges, sort = false) {
+ let build = new RangeSetBuilder();
+ for (let range of ranges instanceof Range ? [ranges] : sort ? lazySort(ranges) : ranges)
+ build.add(range.from, range.to, range.value);
+ return build.finish();
+ }
+ /**
+ Join an array of range sets into a single set.
+ */
+ static join(sets) {
+ if (!sets.length)
+ return RangeSet.empty;
+ let result = sets[sets.length - 1];
+ for (let i = sets.length - 2; i >= 0; i--) {
+ for (let layer = sets[i]; layer != RangeSet.empty; layer = layer.nextLayer)
+ result = new RangeSet(layer.chunkPos, layer.chunk, result, Math.max(layer.maxPoint, result.maxPoint));
+ }
+ return result;
+ }
+}
+/**
+The empty set of ranges.
+*/
+RangeSet.empty = new RangeSet([], [], null, -1);
+function lazySort(ranges) {
+ if (ranges.length > 1)
+ for (let prev = ranges[0], i = 1; i < ranges.length; i++) {
+ let cur = ranges[i];
+ if (cmpRange(prev, cur) > 0)
+ return ranges.slice().sort(cmpRange);
+ prev = cur;
+ }
+ return ranges;
+}
+RangeSet.empty.nextLayer = RangeSet.empty;
+/**
+A range set builder is a data structure that helps build up a
+[range set](https://codemirror.net/6/docs/ref/#state.RangeSet) directly, without first allocating
+an array of [`Range`](https://codemirror.net/6/docs/ref/#state.Range) objects.
+*/
+class RangeSetBuilder {
+ finishChunk(newArrays) {
+ this.chunks.push(new Chunk(this.from, this.to, this.value, this.maxPoint));
+ this.chunkPos.push(this.chunkStart);
+ this.chunkStart = -1;
+ this.setMaxPoint = Math.max(this.setMaxPoint, this.maxPoint);
+ this.maxPoint = -1;
+ if (newArrays) {
+ this.from = [];
+ this.to = [];
+ this.value = [];
+ }
+ }
+ /**
+ Create an empty builder.
+ */
+ constructor() {
+ this.chunks = [];
+ this.chunkPos = [];
+ this.chunkStart = -1;
+ this.last = null;
+ this.lastFrom = -1000000000 /* C.Far */;
+ this.lastTo = -1000000000 /* C.Far */;
+ this.from = [];
+ this.to = [];
+ this.value = [];
+ this.maxPoint = -1;
+ this.setMaxPoint = -1;
+ this.nextLayer = null;
+ }
+ /**
+ Add a range. Ranges should be added in sorted (by `from` and
+ `value.startSide`) order.
+ */
+ add(from, to, value) {
+ if (!this.addInner(from, to, value))
+ (this.nextLayer || (this.nextLayer = new RangeSetBuilder)).add(from, to, value);
+ }
+ /**
+ @internal
+ */
+ addInner(from, to, value) {
+ let diff = from - this.lastTo || value.startSide - this.last.endSide;
+ if (diff <= 0 && (from - this.lastFrom || value.startSide - this.last.startSide) < 0)
+ throw new Error("Ranges must be added sorted by `from` position and `startSide`");
+ if (diff < 0)
+ return false;
+ if (this.from.length == 250 /* C.ChunkSize */)
+ this.finishChunk(true);
+ if (this.chunkStart < 0)
+ this.chunkStart = from;
+ this.from.push(from - this.chunkStart);
+ this.to.push(to - this.chunkStart);
+ this.last = value;
+ this.lastFrom = from;
+ this.lastTo = to;
+ this.value.push(value);
+ if (value.point)
+ this.maxPoint = Math.max(this.maxPoint, to - from);
+ return true;
+ }
+ /**
+ @internal
+ */
+ addChunk(from, chunk) {
+ if ((from - this.lastTo || chunk.value[0].startSide - this.last.endSide) < 0)
+ return false;
+ if (this.from.length)
+ this.finishChunk(true);
+ this.setMaxPoint = Math.max(this.setMaxPoint, chunk.maxPoint);
+ this.chunks.push(chunk);
+ this.chunkPos.push(from);
+ let last = chunk.value.length - 1;
+ this.last = chunk.value[last];
+ this.lastFrom = chunk.from[last] + from;
+ this.lastTo = chunk.to[last] + from;
+ return true;
+ }
+ /**
+ Finish the range set. Returns the new set. The builder can't be
+ used anymore after this has been called.
+ */
+ finish() { return this.finishInner(RangeSet.empty); }
+ /**
+ @internal
+ */
+ finishInner(next) {
+ if (this.from.length)
+ this.finishChunk(false);
+ if (this.chunks.length == 0)
+ return next;
+ let result = RangeSet.create(this.chunkPos, this.chunks, this.nextLayer ? this.nextLayer.finishInner(next) : next, this.setMaxPoint);
+ this.from = null; // Make sure further `add` calls produce errors
+ return result;
+ }
+}
+function findSharedChunks(a, b, textDiff) {
+ let inA = new Map();
+ for (let set of a)
+ for (let i = 0; i < set.chunk.length; i++)
+ if (set.chunk[i].maxPoint <= 0)
+ inA.set(set.chunk[i], set.chunkPos[i]);
+ let shared = new Set();
+ for (let set of b)
+ for (let i = 0; i < set.chunk.length; i++) {
+ let known = inA.get(set.chunk[i]);
+ if (known != null && (textDiff ? textDiff.mapPos(known) : known) == set.chunkPos[i] &&
+ !(textDiff === null || textDiff === void 0 ? void 0 : textDiff.touchesRange(known, known + set.chunk[i].length)))
+ shared.add(set.chunk[i]);
+ }
+ return shared;
+}
+class LayerCursor {
+ constructor(layer, skip, minPoint, rank = 0) {
+ this.layer = layer;
+ this.skip = skip;
+ this.minPoint = minPoint;
+ this.rank = rank;
+ }
+ get startSide() { return this.value ? this.value.startSide : 0; }
+ get endSide() { return this.value ? this.value.endSide : 0; }
+ goto(pos, side = -1000000000 /* C.Far */) {
+ this.chunkIndex = this.rangeIndex = 0;
+ this.gotoInner(pos, side, false);
+ return this;
+ }
+ gotoInner(pos, side, forward) {
+ while (this.chunkIndex < this.layer.chunk.length) {
+ let next = this.layer.chunk[this.chunkIndex];
+ if (!(this.skip && this.skip.has(next) ||
+ this.layer.chunkEnd(this.chunkIndex) < pos ||
+ next.maxPoint < this.minPoint))
+ break;
+ this.chunkIndex++;
+ forward = false;
+ }
+ if (this.chunkIndex < this.layer.chunk.length) {
+ let rangeIndex = this.layer.chunk[this.chunkIndex].findIndex(pos - this.layer.chunkPos[this.chunkIndex], side, true);
+ if (!forward || this.rangeIndex < rangeIndex)
+ this.setRangeIndex(rangeIndex);
+ }
+ this.next();
+ }
+ forward(pos, side) {
+ if ((this.to - pos || this.endSide - side) < 0)
+ this.gotoInner(pos, side, true);
+ }
+ next() {
+ for (;;) {
+ if (this.chunkIndex == this.layer.chunk.length) {
+ this.from = this.to = 1000000000 /* C.Far */;
+ this.value = null;
+ break;
+ }
+ else {
+ let chunkPos = this.layer.chunkPos[this.chunkIndex], chunk = this.layer.chunk[this.chunkIndex];
+ let from = chunkPos + chunk.from[this.rangeIndex];
+ this.from = from;
+ this.to = chunkPos + chunk.to[this.rangeIndex];
+ this.value = chunk.value[this.rangeIndex];
+ this.setRangeIndex(this.rangeIndex + 1);
+ if (this.minPoint < 0 || this.value.point && this.to - this.from >= this.minPoint)
+ break;
+ }
+ }
+ }
+ setRangeIndex(index) {
+ if (index == this.layer.chunk[this.chunkIndex].value.length) {
+ this.chunkIndex++;
+ if (this.skip) {
+ while (this.chunkIndex < this.layer.chunk.length && this.skip.has(this.layer.chunk[this.chunkIndex]))
+ this.chunkIndex++;
+ }
+ this.rangeIndex = 0;
+ }
+ else {
+ this.rangeIndex = index;
+ }
+ }
+ nextChunk() {
+ this.chunkIndex++;
+ this.rangeIndex = 0;
+ this.next();
+ }
+ compare(other) {
+ return this.from - other.from || this.startSide - other.startSide || this.rank - other.rank ||
+ this.to - other.to || this.endSide - other.endSide;
+ }
+}
+class HeapCursor {
+ constructor(heap) {
+ this.heap = heap;
+ }
+ static from(sets, skip = null, minPoint = -1) {
+ let heap = [];
+ for (let i = 0; i < sets.length; i++) {
+ for (let cur = sets[i]; !cur.isEmpty; cur = cur.nextLayer) {
+ if (cur.maxPoint >= minPoint)
+ heap.push(new LayerCursor(cur, skip, minPoint, i));
+ }
+ }
+ return heap.length == 1 ? heap[0] : new HeapCursor(heap);
+ }
+ get startSide() { return this.value ? this.value.startSide : 0; }
+ goto(pos, side = -1000000000 /* C.Far */) {
+ for (let cur of this.heap)
+ cur.goto(pos, side);
+ for (let i = this.heap.length >> 1; i >= 0; i--)
+ heapBubble(this.heap, i);
+ this.next();
+ return this;
+ }
+ forward(pos, side) {
+ for (let cur of this.heap)
+ cur.forward(pos, side);
+ for (let i = this.heap.length >> 1; i >= 0; i--)
+ heapBubble(this.heap, i);
+ if ((this.to - pos || this.value.endSide - side) < 0)
+ this.next();
+ }
+ next() {
+ if (this.heap.length == 0) {
+ this.from = this.to = 1000000000 /* C.Far */;
+ this.value = null;
+ this.rank = -1;
+ }
+ else {
+ let top = this.heap[0];
+ this.from = top.from;
+ this.to = top.to;
+ this.value = top.value;
+ this.rank = top.rank;
+ if (top.value)
+ top.next();
+ heapBubble(this.heap, 0);
+ }
+ }
+}
+function heapBubble(heap, index) {
+ for (let cur = heap[index];;) {
+ let childIndex = (index << 1) + 1;
+ if (childIndex >= heap.length)
+ break;
+ let child = heap[childIndex];
+ if (childIndex + 1 < heap.length && child.compare(heap[childIndex + 1]) >= 0) {
+ child = heap[childIndex + 1];
+ childIndex++;
+ }
+ if (cur.compare(child) < 0)
+ break;
+ heap[childIndex] = cur;
+ heap[index] = child;
+ index = childIndex;
+ }
+}
+class SpanCursor {
+ constructor(sets, skip, minPoint) {
+ this.minPoint = minPoint;
+ this.active = [];
+ this.activeTo = [];
+ this.activeRank = [];
+ this.minActive = -1;
+ // A currently active point range, if any
+ this.point = null;
+ this.pointFrom = 0;
+ this.pointRank = 0;
+ this.to = -1000000000 /* C.Far */;
+ this.endSide = 0;
+ // The amount of open active ranges at the start of the iterator.
+ // Not including points.
+ this.openStart = -1;
+ this.cursor = HeapCursor.from(sets, skip, minPoint);
+ }
+ goto(pos, side = -1000000000 /* C.Far */) {
+ this.cursor.goto(pos, side);
+ this.active.length = this.activeTo.length = this.activeRank.length = 0;
+ this.minActive = -1;
+ this.to = pos;
+ this.endSide = side;
+ this.openStart = -1;
+ this.next();
+ return this;
+ }
+ forward(pos, side) {
+ while (this.minActive > -1 && (this.activeTo[this.minActive] - pos || this.active[this.minActive].endSide - side) < 0)
+ this.removeActive(this.minActive);
+ this.cursor.forward(pos, side);
+ }
+ removeActive(index) {
+ remove(this.active, index);
+ remove(this.activeTo, index);
+ remove(this.activeRank, index);
+ this.minActive = findMinIndex(this.active, this.activeTo);
+ }
+ addActive(trackOpen) {
+ let i = 0, { value, to, rank } = this.cursor;
+ // Organize active marks by rank first, then by size
+ while (i < this.activeRank.length && (rank - this.activeRank[i] || to - this.activeTo[i]) > 0)
+ i++;
+ insert(this.active, i, value);
+ insert(this.activeTo, i, to);
+ insert(this.activeRank, i, rank);
+ if (trackOpen)
+ insert(trackOpen, i, this.cursor.from);
+ this.minActive = findMinIndex(this.active, this.activeTo);
+ }
+ // After calling this, if `this.point` != null, the next range is a
+ // point. Otherwise, it's a regular range, covered by `this.active`.
+ next() {
+ let from = this.to, wasPoint = this.point;
+ this.point = null;
+ let trackOpen = this.openStart < 0 ? [] : null;
+ for (;;) {
+ let a = this.minActive;
+ if (a > -1 && (this.activeTo[a] - this.cursor.from || this.active[a].endSide - this.cursor.startSide) < 0) {
+ if (this.activeTo[a] > from) {
+ this.to = this.activeTo[a];
+ this.endSide = this.active[a].endSide;
+ break;
+ }
+ this.removeActive(a);
+ if (trackOpen)
+ remove(trackOpen, a);
+ }
+ else if (!this.cursor.value) {
+ this.to = this.endSide = 1000000000 /* C.Far */;
+ break;
+ }
+ else if (this.cursor.from > from) {
+ this.to = this.cursor.from;
+ this.endSide = this.cursor.startSide;
+ break;
+ }
+ else {
+ let nextVal = this.cursor.value;
+ if (!nextVal.point) { // Opening a range
+ this.addActive(trackOpen);
+ this.cursor.next();
+ }
+ else if (wasPoint && this.cursor.to == this.to && this.cursor.from < this.cursor.to) {
+ // Ignore any non-empty points that end precisely at the end of the prev point
+ this.cursor.next();
+ }
+ else { // New point
+ this.point = nextVal;
+ this.pointFrom = this.cursor.from;
+ this.pointRank = this.cursor.rank;
+ this.to = this.cursor.to;
+ this.endSide = nextVal.endSide;
+ this.cursor.next();
+ this.forward(this.to, this.endSide);
+ break;
+ }
+ }
+ }
+ if (trackOpen) {
+ this.openStart = 0;
+ for (let i = trackOpen.length - 1; i >= 0 && trackOpen[i] < from; i--)
+ this.openStart++;
+ }
+ }
+ activeForPoint(to) {
+ if (!this.active.length)
+ return this.active;
+ let active = [];
+ for (let i = this.active.length - 1; i >= 0; i--) {
+ if (this.activeRank[i] < this.pointRank)
+ break;
+ if (this.activeTo[i] > to || this.activeTo[i] == to && this.active[i].endSide >= this.point.endSide)
+ active.push(this.active[i]);
+ }
+ return active.reverse();
+ }
+ openEnd(to) {
+ let open = 0;
+ for (let i = this.activeTo.length - 1; i >= 0 && this.activeTo[i] > to; i--)
+ open++;
+ return open;
+ }
+}
+function compare(a, startA, b, startB, length, comparator) {
+ a.goto(startA);
+ b.goto(startB);
+ let endB = startB + length;
+ let pos = startB, dPos = startB - startA;
+ for (;;) {
+ let dEnd = (a.to + dPos) - b.to, diff = dEnd || a.endSide - b.endSide;
+ let end = diff < 0 ? a.to + dPos : b.to, clipEnd = Math.min(end, endB);
+ if (a.point || b.point) {
+ if (!(a.point && b.point && (a.point == b.point || a.point.eq(b.point)) &&
+ sameValues(a.activeForPoint(a.to), b.activeForPoint(b.to))))
+ comparator.comparePoint(pos, clipEnd, a.point, b.point);
+ }
+ else {
+ if (clipEnd > pos && !sameValues(a.active, b.active))
+ comparator.compareRange(pos, clipEnd, a.active, b.active);
+ }
+ if (end > endB)
+ break;
+ if ((dEnd || a.openEnd != b.openEnd) && comparator.boundChange)
+ comparator.boundChange(end);
+ pos = end;
+ if (diff <= 0)
+ a.next();
+ if (diff >= 0)
+ b.next();
+ }
+}
+function sameValues(a, b) {
+ if (a.length != b.length)
+ return false;
+ for (let i = 0; i < a.length; i++)
+ if (a[i] != b[i] && !a[i].eq(b[i]))
+ return false;
+ return true;
+}
+function remove(array, index) {
+ for (let i = index, e = array.length - 1; i < e; i++)
+ array[i] = array[i + 1];
+ array.pop();
+}
+function insert(array, index, value) {
+ for (let i = array.length - 1; i >= index; i--)
+ array[i + 1] = array[i];
+ array[index] = value;
+}
+function findMinIndex(value, array) {
+ let found = -1, foundPos = 1000000000 /* C.Far */;
+ for (let i = 0; i < array.length; i++)
+ if ((array[i] - foundPos || value[i].endSide - value[found].endSide) < 0) {
+ found = i;
+ foundPos = array[i];
+ }
+ return found;
+}
+
+/**
+Count the column position at the given offset into the string,
+taking extending characters and tab size into account.
+*/
+function countColumn(string, tabSize, to = string.length) {
+ let n = 0;
+ for (let i = 0; i < to;) {
+ if (string.charCodeAt(i) == 9) {
+ n += tabSize - (n % tabSize);
+ i++;
+ }
+ else {
+ n++;
+ i = findClusterBreak(string, i);
+ }
+ }
+ return n;
+}
+/**
+Find the offset that corresponds to the given column position in a
+string, taking extending characters and tab size into account. By
+default, the string length is returned when it is too short to
+reach the column. Pass `strict` true to make it return -1 in that
+situation.
+*/
+function findColumn(string, col, tabSize, strict) {
+ for (let i = 0, n = 0;;) {
+ if (n >= col)
+ return i;
+ if (i == string.length)
+ break;
+ n += string.charCodeAt(i) == 9 ? tabSize - (n % tabSize) : 1;
+ i = findClusterBreak(string, i);
+ }
+ return strict === true ? -1 : string.length;
+}
+
+exports.Annotation = Annotation;
+exports.AnnotationType = AnnotationType;
+exports.ChangeDesc = ChangeDesc;
+exports.ChangeSet = ChangeSet;
+exports.Compartment = Compartment;
+exports.EditorSelection = EditorSelection;
+exports.EditorState = EditorState;
+exports.Facet = Facet;
+exports.Line = Line;
+exports.Prec = Prec;
+exports.Range = Range;
+exports.RangeSet = RangeSet;
+exports.RangeSetBuilder = RangeSetBuilder;
+exports.RangeValue = RangeValue;
+exports.SelectionRange = SelectionRange;
+exports.StateEffect = StateEffect;
+exports.StateEffectType = StateEffectType;
+exports.StateField = StateField;
+exports.Text = Text;
+exports.Transaction = Transaction;
+exports.codePointAt = codePointAt;
+exports.codePointSize = codePointSize;
+exports.combineConfig = combineConfig;
+exports.countColumn = countColumn;
+exports.findClusterBreak = findClusterBreak;
+exports.findColumn = findColumn;
+exports.fromCodePoint = fromCodePoint;
diff --git a/node_modules/@codemirror/state/dist/index.d.cts b/node_modules/@codemirror/state/dist/index.d.cts
new file mode 100644
index 0000000..f3adc66
--- /dev/null
+++ b/node_modules/@codemirror/state/dist/index.d.cts
@@ -0,0 +1,1700 @@
+/**
+A text iterator iterates over a sequence of strings. When
+iterating over a [`Text`](https://codemirror.net/6/docs/ref/#state.Text) document, result values will
+either be lines or line breaks.
+*/
+interface TextIterator extends Iterator, Iterable {
+ /**
+ Retrieve the next string. Optionally skip a given number of
+ positions after the current position. Always returns the object
+ itself.
+ */
+ next(skip?: number): this;
+ /**
+ The current string. Will be the empty string when the cursor is
+ at its end or `next` hasn't been called on it yet.
+ */
+ value: string;
+ /**
+ Whether the end of the iteration has been reached. You should
+ probably check this right after calling `next`.
+ */
+ done: boolean;
+ /**
+ Whether the current string represents a line break.
+ */
+ lineBreak: boolean;
+}
+/**
+The data structure for documents. @nonabstract
+*/
+declare abstract class Text implements Iterable {
+ /**
+ The length of the string.
+ */
+ abstract readonly length: number;
+ /**
+ The number of lines in the string (always >= 1).
+ */
+ abstract readonly lines: number;
+ /**
+ Get the line description around the given position.
+ */
+ lineAt(pos: number): Line;
+ /**
+ Get the description for the given (1-based) line number.
+ */
+ line(n: number): Line;
+ /**
+ Replace a range of the text with the given content.
+ */
+ replace(from: number, to: number, text: Text): Text;
+ /**
+ Append another document to this one.
+ */
+ append(other: Text): Text;
+ /**
+ Retrieve the text between the given points.
+ */
+ slice(from: number, to?: number): Text;
+ /**
+ Retrieve a part of the document as a string
+ */
+ abstract sliceString(from: number, to?: number, lineSep?: string): string;
+ /**
+ Test whether this text is equal to another instance.
+ */
+ eq(other: Text): boolean;
+ /**
+ Iterate over the text. When `dir` is `-1`, iteration happens
+ from end to start. This will return lines and the breaks between
+ them as separate strings.
+ */
+ iter(dir?: 1 | -1): TextIterator;
+ /**
+ Iterate over a range of the text. When `from` > `to`, the
+ iterator will run in reverse.
+ */
+ iterRange(from: number, to?: number): TextIterator;
+ /**
+ Return a cursor that iterates over the given range of lines,
+ _without_ returning the line breaks between, and yielding empty
+ strings for empty lines.
+
+ When `from` and `to` are given, they should be 1-based line numbers.
+ */
+ iterLines(from?: number, to?: number): TextIterator;
+ /**
+ Return the document as a string, using newline characters to
+ separate lines.
+ */
+ toString(): string;
+ /**
+ Convert the document to an array of lines (which can be
+ deserialized again via [`Text.of`](https://codemirror.net/6/docs/ref/#state.Text^of)).
+ */
+ toJSON(): string[];
+ /**
+ If this is a branch node, `children` will hold the `Text`
+ objects that it is made up of. For leaf nodes, this holds null.
+ */
+ abstract readonly children: readonly Text[] | null;
+ [Symbol.iterator]: () => Iterator;
+ /**
+ Create a `Text` instance for the given array of lines.
+ */
+ static of(text: readonly string[]): Text;
+ /**
+ The empty document.
+ */
+ static empty: Text;
+}
+/**
+This type describes a line in the document. It is created
+on-demand when lines are [queried](https://codemirror.net/6/docs/ref/#state.Text.lineAt).
+*/
+declare class Line {
+ /**
+ The position of the start of the line.
+ */
+ readonly from: number;
+ /**
+ The position at the end of the line (_before_ the line break,
+ or at the end of document for the last line).
+ */
+ readonly to: number;
+ /**
+ This line's line number (1-based).
+ */
+ readonly number: number;
+ /**
+ The line's content.
+ */
+ readonly text: string;
+ /**
+ The length of the line (not including any line break after it).
+ */
+ get length(): number;
+}
+
+/**
+Distinguishes different ways in which positions can be mapped.
+*/
+declare enum MapMode {
+ /**
+ Map a position to a valid new position, even when its context
+ was deleted.
+ */
+ Simple = 0,
+ /**
+ Return null if deletion happens across the position.
+ */
+ TrackDel = 1,
+ /**
+ Return null if the character _before_ the position is deleted.
+ */
+ TrackBefore = 2,
+ /**
+ Return null if the character _after_ the position is deleted.
+ */
+ TrackAfter = 3
+}
+/**
+A change description is a variant of [change set](https://codemirror.net/6/docs/ref/#state.ChangeSet)
+that doesn't store the inserted text. As such, it can't be
+applied, but is cheaper to store and manipulate.
+*/
+declare class ChangeDesc {
+ /**
+ The length of the document before the change.
+ */
+ get length(): number;
+ /**
+ The length of the document after the change.
+ */
+ get newLength(): number;
+ /**
+ False when there are actual changes in this set.
+ */
+ get empty(): boolean;
+ /**
+ Iterate over the unchanged parts left by these changes. `posA`
+ provides the position of the range in the old document, `posB`
+ the new position in the changed document.
+ */
+ iterGaps(f: (posA: number, posB: number, length: number) => void): void;
+ /**
+ Iterate over the ranges changed by these changes. (See
+ [`ChangeSet.iterChanges`](https://codemirror.net/6/docs/ref/#state.ChangeSet.iterChanges) for a
+ variant that also provides you with the inserted text.)
+ `fromA`/`toA` provides the extent of the change in the starting
+ document, `fromB`/`toB` the extent of the replacement in the
+ changed document.
+
+ When `individual` is true, adjacent changes (which are kept
+ separate for [position mapping](https://codemirror.net/6/docs/ref/#state.ChangeDesc.mapPos)) are
+ reported separately.
+ */
+ iterChangedRanges(f: (fromA: number, toA: number, fromB: number, toB: number) => void, individual?: boolean): void;
+ /**
+ Get a description of the inverted form of these changes.
+ */
+ get invertedDesc(): ChangeDesc;
+ /**
+ Compute the combined effect of applying another set of changes
+ after this one. The length of the document after this set should
+ match the length before `other`.
+ */
+ composeDesc(other: ChangeDesc): ChangeDesc;
+ /**
+ Map this description, which should start with the same document
+ as `other`, over another set of changes, so that it can be
+ applied after it. When `before` is true, map as if the changes
+ in `this` happened before the ones in `other`.
+ */
+ mapDesc(other: ChangeDesc, before?: boolean): ChangeDesc;
+ /**
+ Map a given position through these changes, to produce a
+ position pointing into the new document.
+
+ `assoc` indicates which side the position should be associated
+ with. When it is negative or zero, the mapping will try to keep
+ the position close to the character before it (if any), and will
+ move it before insertions at that point or replacements across
+ that point. When it is positive, the position is associated with
+ the character after it, and will be moved forward for insertions
+ at or replacements across the position. Defaults to -1.
+
+ `mode` determines whether deletions should be
+ [reported](https://codemirror.net/6/docs/ref/#state.MapMode). It defaults to
+ [`MapMode.Simple`](https://codemirror.net/6/docs/ref/#state.MapMode.Simple) (don't report
+ deletions).
+ */
+ mapPos(pos: number, assoc?: number): number;
+ mapPos(pos: number, assoc: number, mode: MapMode): number | null;
+ /**
+ Check whether these changes touch a given range. When one of the
+ changes entirely covers the range, the string `"cover"` is
+ returned.
+ */
+ touchesRange(from: number, to?: number): boolean | "cover";
+ /**
+ Serialize this change desc to a JSON-representable value.
+ */
+ toJSON(): readonly number[];
+ /**
+ Create a change desc from its JSON representation (as produced
+ by [`toJSON`](https://codemirror.net/6/docs/ref/#state.ChangeDesc.toJSON).
+ */
+ static fromJSON(json: any): ChangeDesc;
+}
+/**
+This type is used as argument to
+[`EditorState.changes`](https://codemirror.net/6/docs/ref/#state.EditorState.changes) and in the
+[`changes` field](https://codemirror.net/6/docs/ref/#state.TransactionSpec.changes) of transaction
+specs to succinctly describe document changes. It may either be a
+plain object describing a change (a deletion, insertion, or
+replacement, depending on which fields are present), a [change
+set](https://codemirror.net/6/docs/ref/#state.ChangeSet), or an array of change specs.
+*/
+type ChangeSpec = {
+ from: number;
+ to?: number;
+ insert?: string | Text;
+} | ChangeSet | readonly ChangeSpec[];
+/**
+A change set represents a group of modifications to a document. It
+stores the document length, and can only be applied to documents
+with exactly that length.
+*/
+declare class ChangeSet extends ChangeDesc {
+ private constructor();
+ /**
+ Apply the changes to a document, returning the modified
+ document.
+ */
+ apply(doc: Text): Text;
+ mapDesc(other: ChangeDesc, before?: boolean): ChangeDesc;
+ /**
+ Given the document as it existed _before_ the changes, return a
+ change set that represents the inverse of this set, which could
+ be used to go from the document created by the changes back to
+ the document as it existed before the changes.
+ */
+ invert(doc: Text): ChangeSet;
+ /**
+ Combine two subsequent change sets into a single set. `other`
+ must start in the document produced by `this`. If `this` goes
+ `docA` → `docB` and `other` represents `docB` → `docC`, the
+ returned value will represent the change `docA` → `docC`.
+ */
+ compose(other: ChangeSet): ChangeSet;
+ /**
+ Given another change set starting in the same document, maps this
+ change set over the other, producing a new change set that can be
+ applied to the document produced by applying `other`. When
+ `before` is `true`, order changes as if `this` comes before
+ `other`, otherwise (the default) treat `other` as coming first.
+
+ Given two changes `A` and `B`, `A.compose(B.map(A))` and
+ `B.compose(A.map(B, true))` will produce the same document. This
+ provides a basic form of [operational
+ transformation](https://en.wikipedia.org/wiki/Operational_transformation),
+ and can be used for collaborative editing.
+ */
+ map(other: ChangeDesc, before?: boolean): ChangeSet;
+ /**
+ Iterate over the changed ranges in the document, calling `f` for
+ each, with the range in the original document (`fromA`-`toA`)
+ and the range that replaces it in the new document
+ (`fromB`-`toB`).
+
+ When `individual` is true, adjacent changes are reported
+ separately.
+ */
+ iterChanges(f: (fromA: number, toA: number, fromB: number, toB: number, inserted: Text) => void, individual?: boolean): void;
+ /**
+ Get a [change description](https://codemirror.net/6/docs/ref/#state.ChangeDesc) for this change
+ set.
+ */
+ get desc(): ChangeDesc;
+ /**
+ Serialize this change set to a JSON-representable value.
+ */
+ toJSON(): any;
+ /**
+ Create a change set for the given changes, for a document of the
+ given length, using `lineSep` as line separator.
+ */
+ static of(changes: ChangeSpec, length: number, lineSep?: string): ChangeSet;
+ /**
+ Create an empty changeset of the given length.
+ */
+ static empty(length: number): ChangeSet;
+ /**
+ Create a changeset from its JSON representation (as produced by
+ [`toJSON`](https://codemirror.net/6/docs/ref/#state.ChangeSet.toJSON).
+ */
+ static fromJSON(json: any): ChangeSet;
+}
+
+/**
+A single selection range. When
+[`allowMultipleSelections`](https://codemirror.net/6/docs/ref/#state.EditorState^allowMultipleSelections)
+is enabled, a [selection](https://codemirror.net/6/docs/ref/#state.EditorSelection) may hold
+multiple ranges. By default, selections hold exactly one range.
+*/
+declare class SelectionRange {
+ /**
+ The lower boundary of the range.
+ */
+ readonly from: number;
+ /**
+ The upper boundary of the range.
+ */
+ readonly to: number;
+ private flags;
+ private constructor();
+ /**
+ The anchor of the range—the side that doesn't move when you
+ extend it.
+ */
+ get anchor(): number;
+ /**
+ The head of the range, which is moved when the range is
+ [extended](https://codemirror.net/6/docs/ref/#state.SelectionRange.extend).
+ */
+ get head(): number;
+ /**
+ True when `anchor` and `head` are at the same position.
+ */
+ get empty(): boolean;
+ /**
+ If this is a cursor that is explicitly associated with the
+ character on one of its sides, this returns the side. -1 means
+ the character before its position, 1 the character after, and 0
+ means no association.
+ */
+ get assoc(): -1 | 0 | 1;
+ /**
+ The bidirectional text level associated with this cursor, if
+ any.
+ */
+ get bidiLevel(): number | null;
+ /**
+ The goal column (stored vertical offset) associated with a
+ cursor. This is used to preserve the vertical position when
+ [moving](https://codemirror.net/6/docs/ref/#view.EditorView.moveVertically) across
+ lines of different length.
+ */
+ get goalColumn(): number | undefined;
+ /**
+ Map this range through a change, producing a valid range in the
+ updated document.
+ */
+ map(change: ChangeDesc, assoc?: number): SelectionRange;
+ /**
+ Extend this range to cover at least `from` to `to`.
+ */
+ extend(from: number, to?: number): SelectionRange;
+ /**
+ Compare this range to another range.
+ */
+ eq(other: SelectionRange, includeAssoc?: boolean): boolean;
+ /**
+ Return a JSON-serializable object representing the range.
+ */
+ toJSON(): any;
+ /**
+ Convert a JSON representation of a range to a `SelectionRange`
+ instance.
+ */
+ static fromJSON(json: any): SelectionRange;
+}
+/**
+An editor selection holds one or more selection ranges.
+*/
+declare class EditorSelection {
+ /**
+ The ranges in the selection, sorted by position. Ranges cannot
+ overlap (but they may touch, if they aren't empty).
+ */
+ readonly ranges: readonly SelectionRange[];
+ /**
+ The index of the _main_ range in the selection (which is
+ usually the range that was added last).
+ */
+ readonly mainIndex: number;
+ private constructor();
+ /**
+ Map a selection through a change. Used to adjust the selection
+ position for changes.
+ */
+ map(change: ChangeDesc, assoc?: number): EditorSelection;
+ /**
+ Compare this selection to another selection. By default, ranges
+ are compared only by position. When `includeAssoc` is true,
+ cursor ranges must also have the same
+ [`assoc`](https://codemirror.net/6/docs/ref/#state.SelectionRange.assoc) value.
+ */
+ eq(other: EditorSelection, includeAssoc?: boolean): boolean;
+ /**
+ Get the primary selection range. Usually, you should make sure
+ your code applies to _all_ ranges, by using methods like
+ [`changeByRange`](https://codemirror.net/6/docs/ref/#state.EditorState.changeByRange).
+ */
+ get main(): SelectionRange;
+ /**
+ Make sure the selection only has one range. Returns a selection
+ holding only the main range from this selection.
+ */
+ asSingle(): EditorSelection;
+ /**
+ Extend this selection with an extra range.
+ */
+ addRange(range: SelectionRange, main?: boolean): EditorSelection;
+ /**
+ Replace a given range with another range, and then normalize the
+ selection to merge and sort ranges if necessary.
+ */
+ replaceRange(range: SelectionRange, which?: number): EditorSelection;
+ /**
+ Convert this selection to an object that can be serialized to
+ JSON.
+ */
+ toJSON(): any;
+ /**
+ Create a selection from a JSON representation.
+ */
+ static fromJSON(json: any): EditorSelection;
+ /**
+ Create a selection holding a single range.
+ */
+ static single(anchor: number, head?: number): EditorSelection;
+ /**
+ Sort and merge the given set of ranges, creating a valid
+ selection.
+ */
+ static create(ranges: readonly SelectionRange[], mainIndex?: number): EditorSelection;
+ /**
+ Create a cursor selection range at the given position. You can
+ safely ignore the optional arguments in most situations.
+ */
+ static cursor(pos: number, assoc?: number, bidiLevel?: number, goalColumn?: number): SelectionRange;
+ /**
+ Create a selection range.
+ */
+ static range(anchor: number, head: number, goalColumn?: number, bidiLevel?: number): SelectionRange;
+}
+
+type FacetConfig = {
+ /**
+ How to combine the input values into a single output value. When
+ not given, the array of input values becomes the output. This
+ function will immediately be called on creating the facet, with
+ an empty array, to compute the facet's default value when no
+ inputs are present.
+ */
+ combine?: (value: readonly Input[]) => Output;
+ /**
+ How to compare output values to determine whether the value of
+ the facet changed. Defaults to comparing by `===` or, if no
+ `combine` function was given, comparing each element of the
+ array with `===`.
+ */
+ compare?: (a: Output, b: Output) => boolean;
+ /**
+ How to compare input values to avoid recomputing the output
+ value when no inputs changed. Defaults to comparing with `===`.
+ */
+ compareInput?: (a: Input, b: Input) => boolean;
+ /**
+ Forbids dynamic inputs to this facet.
+ */
+ static?: boolean;
+ /**
+ If given, these extension(s) (or the result of calling the given
+ function with the facet) will be added to any state where this
+ facet is provided. (Note that, while a facet's default value can
+ be read from a state even if the facet wasn't present in the
+ state at all, these extensions won't be added in that
+ situation.)
+ */
+ enables?: Extension | ((self: Facet) => Extension);
+};
+/**
+A facet is a labeled value that is associated with an editor
+state. It takes inputs from any number of extensions, and combines
+those into a single output value.
+
+Examples of uses of facets are the [tab
+size](https://codemirror.net/6/docs/ref/#state.EditorState^tabSize), [editor
+attributes](https://codemirror.net/6/docs/ref/#view.EditorView^editorAttributes), and [update
+listeners](https://codemirror.net/6/docs/ref/#view.EditorView^updateListener).
+
+Note that `Facet` instances can be used anywhere where
+[`FacetReader`](https://codemirror.net/6/docs/ref/#state.FacetReader) is expected.
+*/
+declare class Facet implements FacetReader