Files
pantosite/src/content/.obsidian/plugins/explorer-focus/main.js
2026-04-11 00:41:28 +02:00

819 lines
106 KiB
JavaScript

/*
THIS IS A GENERATED/BUNDLED FILE BY ESBUILD
if you want to view the source, please visit the github repository of this plugin
*/
"use strict";
var __defProp = Object.defineProperty;
var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
var __getOwnPropNames = Object.getOwnPropertyNames;
var __hasOwnProp = Object.prototype.hasOwnProperty;
var __export = (target, all) => {
for (var name in all)
__defProp(target, name, { get: all[name], enumerable: true });
};
var __copyProps = (to, from, except, desc) => {
if (from && typeof from === "object" || typeof from === "function") {
for (let key of __getOwnPropNames(from))
if (!__hasOwnProp.call(to, key) && key !== except)
__defProp(to, key, { get: () => from[key], enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable });
}
return to;
};
var __toCommonJS = (mod) => __copyProps(__defProp({}, "__esModule", { value: true }), mod);
// src/main.ts
var main_exports = {};
__export(main_exports, {
ExplorerFocusPlugin: () => ExplorerFocusPlugin,
default: () => main_default
});
module.exports = __toCommonJS(main_exports);
var import_obsidian5 = require("obsidian");
// src/types.ts
var DEFAULT_SETTINGS = {
showRightClickMenu: true,
showFileExplorerIcon: true,
focusLevel: "parent",
customFolderPath: "",
hideAncestorFolders: false,
autoHidePaths: []
};
// src/ui/settings-tab.ts
var import_obsidian4 = require("obsidian");
// src/utils/file-explorer-patch.ts
var import_obsidian = require("obsidian");
// node_modules/.pnpm/monkey-around@3.0.0/node_modules/monkey-around/dist/index.mjs
function around(obj, factories) {
const removers = Object.keys(factories).map((key) => around1(obj, key, factories[key]));
return removers.length === 1 ? removers[0] : function() {
removers.forEach((r) => r());
};
}
function around1(obj, method, createWrapper) {
const inherited = obj[method], hadOwn = obj.hasOwnProperty(method), original = hadOwn ? inherited : function() {
return Object.getPrototypeOf(obj)[method].apply(this, arguments);
};
let current = createWrapper(original);
if (inherited)
Object.setPrototypeOf(current, inherited);
Object.setPrototypeOf(wrapper, current);
obj[method] = wrapper;
return remove;
function wrapper(...args) {
if (current === original && obj[method] === wrapper)
remove();
return current.apply(this, args);
}
function remove() {
if (obj[method] === wrapper) {
if (hadOwn)
obj[method] = original;
else
delete obj[method];
}
if (current === original)
return;
current = original;
Object.setPrototypeOf(wrapper, inherited || Function);
}
}
// src/utils/file-explorer-patch.ts
var prototypePatched = false;
var patchingFailed = false;
function patchFileExplorer(plugin) {
if (patchingFailed) {
return;
}
if (import_obsidian.Platform.isMobile) {
patchingFailed = true;
return;
}
const fileExplorer = getFileExplorer(plugin);
if (!fileExplorer) {
return;
}
if (!prototypePatched) {
try {
const prototype = Object.getPrototypeOf(fileExplorer);
if (typeof prototype.getSortedFolderItems !== "function") {
throw new Error("getSortedFolderItems method not found on file explorer prototype");
}
plugin.register(
around(prototype, {
getSortedFolderItems(old) {
return function(folder) {
let sortedChildren = old.call(this, folder);
if (plugin.isFocus && plugin.focusedPath) {
const focusedPath = plugin.focusedPath;
sortedChildren = sortedChildren.filter((vEl) => {
const filePath = vEl.file.path;
if (filePath === focusedPath) {
vEl.info.hidden = false;
return true;
}
if (filePath.startsWith(focusedPath + "/")) {
vEl.info.hidden = false;
return true;
}
if (focusedPath.startsWith(filePath + "/")) {
vEl.info.hidden = false;
return true;
}
vEl.info.hidden = true;
return false;
});
} else {
sortedChildren.forEach((vEl) => {
vEl.info.hidden = false;
});
}
return sortedChildren;
};
}
})
);
prototypePatched = true;
} catch (error) {
patchingFailed = true;
console.warn(
"[Explorer Focus] Failed to patch file explorer. The plugin will use CSS-based hiding as a fallback, which may be less precise. This usually happens after an Obsidian update - please check for plugin updates.",
error
);
new import_obsidian.Notice(
"Explorer Focus: File explorer patching failed. The plugin will still work but may need an update for full functionality.",
8e3
);
return;
}
}
fileExplorer.fileExplorerPlusPatched = true;
}
function getFileExplorer(plugin) {
var _a;
const fileExplorerContainer = (_a = plugin.app.workspace.getLeavesOfType("file-explorer")) == null ? void 0 : _a.first();
return fileExplorerContainer == null ? void 0 : fileExplorerContainer.view;
}
function getAllFileExplorers(plugin) {
const fileExplorerLeaves = plugin.app.workspace.getLeavesOfType("file-explorer");
return fileExplorerLeaves.map((leaf) => leaf.view);
}
// src/ui/folder-suggest.ts
var import_obsidian2 = require("obsidian");
var FolderSuggest = class extends import_obsidian2.AbstractInputSuggest {
constructor(app, inputEl) {
super(app, inputEl);
this.inputEl = inputEl;
}
getSuggestions(query) {
const lowerQuery = query.toLowerCase();
const folders = [];
this.app.vault.getAllLoadedFiles().forEach((file) => {
if (file instanceof import_obsidian2.TFolder) {
if (file.path.toLowerCase().includes(lowerQuery)) {
folders.push(file);
}
}
});
folders.sort((a, b) => a.path.localeCompare(b.path));
return folders.slice(0, 20);
}
renderSuggestion(folder, el) {
el.createEl("div", { text: folder.path || "/" });
}
selectSuggestion(folder) {
this.inputEl.value = folder.path;
this.inputEl.trigger("input");
this.close();
}
};
// src/ui/auto-hide-modal.ts
var import_obsidian3 = require("obsidian");
var AutoHideModal = class extends import_obsidian3.Modal {
constructor(app, plugin, onSave) {
var _a;
super(app);
this.addInput = null;
this.plugin = plugin;
this.paths = [...(_a = plugin.settings.autoHidePaths) != null ? _a : []];
this.onSave = onSave;
}
onOpen() {
this.setTitle("Auto-hide folders");
this.renderContent();
}
renderContent() {
var _a;
const { contentEl } = this;
contentEl.empty();
contentEl.createEl("p", {
text: "The following folders are currently hidden from the file explorer."
});
const listWrapper = contentEl.createDiv();
this.paths.forEach((path, index) => {
const row = listWrapper.createDiv({ cls: "mobile-option-setting-item" });
row.createSpan({
text: path || "(empty)",
cls: "mobile-option-setting-item-name"
});
const removeBtn = row.createDiv({
cls: "clickable-icon mobile-option-setting-item-option-icon",
attr: { "aria-label": "Delete" }
});
(0, import_obsidian3.setIcon)(removeBtn, "x");
removeBtn.addEventListener("click", () => {
this.paths.splice(index, 1);
this.renderContent();
});
});
new import_obsidian3.Setting(contentEl).setName("Folder").addText((text) => {
new FolderSuggest(this.app, text.inputEl);
text.setPlaceholder("Enter folder path");
this.addInput = text;
}).addButton((button) => button.setButtonText("Add").onClick(() => {
var _a2, _b, _c;
const value = (_b = (_a2 = this.addInput) == null ? void 0 : _a2.getValue()) == null ? void 0 : _b.trim();
if (value) {
this.paths.push(value);
(_c = this.addInput) == null ? void 0 : _c.setValue("");
this.renderContent();
}
}));
(_a = this.modalEl.querySelector(".modal-button-container")) == null ? void 0 : _a.remove();
const buttonContainer = this.modalEl.createDiv({ cls: "modal-button-container" });
const saveBtn = buttonContainer.createEl("button", {
text: "Save",
cls: "mod-cta"
});
saveBtn.addEventListener("click", () => {
this.plugin.settings.autoHidePaths = this.paths;
void this.plugin.saveSettings();
this.plugin.updateAutoHideStyles();
this.onSave();
this.close();
});
const cancelBtn = buttonContainer.createEl("button", {
text: "Cancel",
cls: "mod-cancel"
});
cancelBtn.addEventListener("click", () => {
this.close();
});
}
};
// src/ui/settings-tab.ts
var ExplorerFocusSettingTab = class extends import_obsidian4.PluginSettingTab {
constructor(app, plugin) {
super(app, plugin);
this.icon = "lucide-focus";
this.plugin = plugin;
}
display() {
const { containerEl } = this;
containerEl.empty();
const generalGroup = new import_obsidian4.SettingGroup(containerEl);
generalGroup.addSetting((setting) => {
setting.setName("Show right-click menu option").addToggle((toggle) => toggle.setValue(this.plugin.settings.showRightClickMenu).onChange(async (value) => {
this.plugin.settings.showRightClickMenu = value;
await this.plugin.saveSettings();
}));
});
generalGroup.addSetting((setting) => {
setting.setName("Show file explorer icon").addToggle((toggle) => toggle.setValue(this.plugin.settings.showFileExplorerIcon).onChange(async (value) => {
this.plugin.settings.showFileExplorerIcon = value;
await this.plugin.saveSettings();
this.plugin.updateFileExplorerIcon();
}));
});
generalGroup.addSetting((setting) => {
setting.setName("Command focus level").setDesc("Determines what to focus when using the toggle command or file explorer icon. Right-click menu always focuses the clicked file/folder.").addDropdown((dropdown) => dropdown.addOption("file", "Current file only").addOption("parent", "Parent folder").addOption("grandparent", "Grandparent folder").addOption("greatgrandparent", "Great grandparent folder").addOption("custom", "Specific folder").setValue(this.plugin.settings.focusLevel).onChange(async (value) => {
this.plugin.settings.focusLevel = value;
await this.plugin.saveSettings();
if (this.plugin.isFocus) {
const fileExplorers = getAllFileExplorers(this.plugin);
fileExplorers.forEach((fileExplorer) => {
if (fileExplorer == null ? void 0 : fileExplorer.requestSort) {
fileExplorer.requestSort();
}
});
}
this.display();
}));
});
if (this.plugin.settings.focusLevel === "custom") {
generalGroup.addSetting((setting) => {
setting.setName("Custom folder path").setDesc("Enter a folder path (folder/subfolder). This folder will be focused regardless of which file is open.").addText((text) => {
new FolderSuggest(this.app, text.inputEl);
text.setPlaceholder("Folder/subfolder").setValue(this.plugin.settings.customFolderPath).onChange(async (value) => {
this.plugin.settings.customFolderPath = value;
await this.plugin.saveSettings();
});
});
});
}
const autoHideGroup = new import_obsidian4.SettingGroup(containerEl).setHeading("Auto-hide folders");
autoHideGroup.addSetting((setting) => {
var _a;
const autoHidePaths = ((_a = this.plugin.settings.autoHidePaths) != null ? _a : []).filter((p) => p.trim().length > 0);
const descFragment = document.createDocumentFragment();
descFragment.appendText("These folders are always hidden from the file explorer.");
if (autoHidePaths.length > 0) {
const list = descFragment.createEl("ul");
autoHidePaths.forEach((p) => {
list.createEl("li", { text: p });
});
}
setting.setName("Hidden folders").setDesc(descFragment).addButton((button) => button.setButtonText("Manage").onClick(() => {
new AutoHideModal(this.app, this.plugin, () => {
this.display();
}).open();
}));
});
}
};
// src/utils/focus.ts
function getFocusPath(path, level, settings) {
if (level === "custom") {
return settings.customFolderPath || path;
}
if (level === "file") {
return path;
}
const parts = path.split("/");
if (level === "parent") {
if (parts.length === 1) {
return path;
}
return parts.slice(0, -1).join("/");
}
if (level === "grandparent") {
if (parts.length <= 2) {
return parts[0] || path;
}
return parts.slice(0, -2).join("/");
}
if (parts.length <= 3) {
return parts[0] || path;
}
return parts.slice(0, -3).join("/");
}
// src/commands/index.ts
function registerCommands(plugin) {
plugin.addCommand({
id: "toggle",
name: "Toggle focus",
callback: () => {
if (plugin.isFocus) {
plugin.exitFocus();
} else {
if (plugin.settings.focusLevel === "custom") {
if (plugin.settings.customFolderPath) {
plugin.enterFocus(plugin.settings.customFolderPath);
}
} else {
const file = plugin.app.workspace.getActiveFile();
if (file == null ? void 0 : file.path) {
const focusPath = getFocusPath(file.path, plugin.settings.focusLevel, plugin.settings);
plugin.enterFocus(focusPath);
}
}
}
}
});
if (plugin.settings.showRightClickMenu) {
plugin.registerEvent(
plugin.app.workspace.on("file-menu", (menu, file) => {
if (!plugin.settings.showRightClickMenu) {
return;
}
menu.addItem((item) => {
item.setTitle(plugin.isFocus ? "Exit focus" : "Focus").setIcon(plugin.isFocus ? "log-out" : "focus").onClick(() => {
plugin.toggleFocus(file == null ? void 0 : file.path);
});
});
})
);
}
}
// src/utils/file-explorer.ts
function findNavButtonsContainer(fileExplorerView) {
var _a;
let container = fileExplorerView.querySelector(".nav-buttons-container");
if (container) {
return container;
}
container = fileExplorerView.querySelector(".nav-header-button-container");
if (container) {
return container;
}
const existingButtons = fileExplorerView.querySelectorAll(".nav-action-button");
if (existingButtons.length > 0) {
const firstButton = existingButtons[0];
const parent = firstButton.parentElement;
if (parent && (parent.classList.contains("nav-buttons-container") || parent.classList.contains("nav-header-button-container") || Array.from(parent.children).some((el) => el.classList.contains("nav-action-button")))) {
return parent;
}
}
const viewHeader = fileExplorerView.querySelector(".view-header");
if (viewHeader) {
const headerContainer = viewHeader.querySelector(".nav-buttons-container") || viewHeader.querySelector(".nav-header-button-container") || viewHeader.querySelector(".view-header-title-container");
if (headerContainer) {
return headerContainer;
}
return viewHeader;
}
const navFilesContainer = fileExplorerView.querySelector(".nav-files-container");
if (navFilesContainer) {
let mobileButtonContainer = fileExplorerView.querySelector(".explorer-focus-mobile-buttons");
if (!mobileButtonContainer) {
mobileButtonContainer = document.createElement("div");
mobileButtonContainer.className = "nav-buttons-container explorer-focus-mobile-buttons";
mobileButtonContainer.setCssProps({
display: "flex",
alignItems: "center",
gap: "4px",
padding: "8px",
borderBottom: "1px solid var(--background-modifier-border)"
});
(_a = navFilesContainer.parentElement) == null ? void 0 : _a.insertBefore(mobileButtonContainer, navFilesContainer);
}
return mobileButtonContainer;
}
let topContainer = fileExplorerView.querySelector(".explorer-focus-mobile-buttons");
if (!topContainer) {
topContainer = document.createElement("div");
topContainer.className = "nav-buttons-container explorer-focus-mobile-buttons";
topContainer.setCssProps({
display: "flex",
alignItems: "center",
gap: "4px",
padding: "8px",
borderBottom: "1px solid var(--background-modifier-border)"
});
fileExplorerView.insertBefore(topContainer, fileExplorerView.firstChild);
}
return topContainer;
}
function createFileExplorerIcon(plugin) {
const icon = document.createElement("div");
icon.className = "clickable-icon nav-action-button";
icon.setAttribute("aria-label", "Toggle focus");
const svg = document.createElementNS("http://www.w3.org/2000/svg", "svg");
svg.setAttribute("xmlns", "http://www.w3.org/2000/svg");
svg.setAttribute("width", "24");
svg.setAttribute("height", "24");
svg.setAttribute("viewBox", "0 0 24 24");
svg.setAttribute("fill", "none");
svg.setAttribute("stroke", "currentColor");
svg.setAttribute("stroke-width", "2");
svg.setAttribute("stroke-linecap", "round");
svg.setAttribute("stroke-linejoin", "round");
svg.setAttribute("class", "svg-icon lucide-focus");
const circle = document.createElementNS("http://www.w3.org/2000/svg", "circle");
circle.setAttribute("cx", "12");
circle.setAttribute("cy", "12");
circle.setAttribute("r", "3");
svg.appendChild(circle);
const path1 = document.createElementNS("http://www.w3.org/2000/svg", "path");
path1.setAttribute("d", "M3 7V5a2 2 0 0 1 2-2h2");
svg.appendChild(path1);
const path2 = document.createElementNS("http://www.w3.org/2000/svg", "path");
path2.setAttribute("d", "M17 3h2a2 2 0 0 1 2 2v2");
svg.appendChild(path2);
const path3 = document.createElementNS("http://www.w3.org/2000/svg", "path");
path3.setAttribute("d", "M21 17v2a2 2 0 0 1-2 2h-2");
svg.appendChild(path3);
const path4 = document.createElementNS("http://www.w3.org/2000/svg", "path");
path4.setAttribute("d", "M7 21H5a2 2 0 0 1-2-2v-2");
svg.appendChild(path4);
icon.appendChild(svg);
return icon;
}
function insertFileExplorerIcon(icon, navButtonsContainer) {
if (navButtonsContainer.contains(icon)) {
return;
}
const allIcons = Array.from(navButtonsContainer.querySelectorAll(".clickable-icon.nav-action-button"));
const defaultIcons = allIcons.filter((el) => !el.classList.contains("cmdr") && el !== icon);
const cmdrIcons = Array.from(navButtonsContainer.querySelectorAll(".cmdr"));
if (cmdrIcons.length > 0) {
navButtonsContainer.insertBefore(icon, cmdrIcons[0]);
} else if (defaultIcons.length > 0) {
const lastIcon = defaultIcons[defaultIcons.length - 1];
if (lastIcon.nextSibling) {
navButtonsContainer.insertBefore(icon, lastIcon.nextSibling);
} else {
navButtonsContainer.appendChild(icon);
}
} else {
navButtonsContainer.appendChild(icon);
}
}
// src/main.ts
var ExplorerFocusPlugin = class extends import_obsidian5.Plugin {
constructor(app, manifest) {
super(app, manifest);
this.isFocus = false;
this.focusedPath = null;
this.fileExplorerIcon = null;
}
async onload() {
await this.loadSettings();
this.addSettingTab(new ExplorerFocusSettingTab(this.app, this));
registerCommands(this);
if (this.settings.showFileExplorerIcon) {
this.addFileExplorerIcon();
}
this.app.workspace.onLayoutReady(() => {
this.patchAllFileExplorers();
this.updateFocusModeClasses();
this.updateAutoHideStyles();
});
this.app.workspace.on("layout-change", () => {
this.patchAllFileExplorers();
this.updateFocusModeClasses();
this.updateAutoHideStyles();
});
}
async loadSettings() {
const loadedData = await this.loadData();
this.settings = Object.assign({}, DEFAULT_SETTINGS, loadedData);
}
async saveSettings() {
await this.saveData(this.settings);
}
toggleFocus(path) {
if (this.isFocus) {
this.exitFocus();
} else if (path) {
this.enterFocus(path);
}
}
enterFocus(path) {
this.isFocus = true;
this.focusedPath = path;
this.updateFileExplorerIcon();
this.updateFocusModeClasses();
const fileExplorers = getAllFileExplorers(this);
fileExplorers.forEach((fileExplorer) => {
if (fileExplorer == null ? void 0 : fileExplorer.requestSort) {
fileExplorer.requestSort();
}
});
setTimeout(() => {
fileExplorers.forEach((fileExplorer) => {
this.refreshFileExplorerVisibility(fileExplorer);
});
}, 0);
}
patchAllFileExplorers() {
patchFileExplorer(this);
const fileExplorers = getAllFileExplorers(this);
fileExplorers.forEach((fileExplorer) => {
fileExplorer.fileExplorerPlusPatched = true;
});
}
exitFocus() {
this.isFocus = false;
this.focusedPath = null;
this.updateFileExplorerIcon();
this.updateFocusModeClasses();
const fileExplorers = getAllFileExplorers(this);
fileExplorers.forEach((fileExplorer) => {
if (fileExplorer == null ? void 0 : fileExplorer.requestSort) {
fileExplorer.requestSort();
}
});
setTimeout(() => {
fileExplorers.forEach((fileExplorer) => {
this.refreshFileExplorerVisibility(fileExplorer);
});
this.updateAutoHideStyles();
}, 0);
}
refreshFileExplorerVisibility(fileExplorer) {
if (!(fileExplorer == null ? void 0 : fileExplorer.fileItems)) {
return;
}
if (!this.isFocus || !this.focusedPath) {
Object.values(fileExplorer.fileItems).forEach((vEl) => {
if (!(vEl == null ? void 0 : vEl.el)) return;
if (vEl.el.hasAttribute("data-explorer-focus-hidden")) {
if (vEl.info) vEl.info.hidden = false;
vEl.el.setCssProps({ display: "" });
vEl.el.removeAttribute("data-explorer-focus-hidden");
}
});
return;
}
const focusedPath = this.focusedPath;
const ancestorPaths = /* @__PURE__ */ new Set();
const pathParts = focusedPath.split("/");
for (let i = 1; i < pathParts.length; i++) {
ancestorPaths.add(pathParts.slice(0, i).join("/"));
}
const topLevelFolders = /* @__PURE__ */ new Map();
const rootItems = [];
for (const path of Object.keys(fileExplorer.fileItems)) {
const firstSlash = path.indexOf("/");
if (firstSlash === -1) {
rootItems.push(path);
} else {
const topLevel = path.substring(0, firstSlash);
if (!topLevelFolders.has(topLevel)) {
topLevelFolders.set(topLevel, []);
}
topLevelFolders.get(topLevel).push(path);
}
}
const focusedTopLevel = focusedPath.split("/")[0];
for (const path of rootItems) {
const vEl = fileExplorer.fileItems[path];
if (!(vEl == null ? void 0 : vEl.el)) continue;
const shouldShow = path === focusedPath || focusedPath.startsWith(path + "/") || path.startsWith(focusedPath + "/");
this.updateItemVisibility(vEl, shouldShow);
}
for (const [topLevel, paths] of topLevelFolders) {
const isInFocusPath = topLevel === focusedTopLevel || ancestorPaths.has(topLevel) || focusedPath === topLevel;
if (!isInFocusPath) {
for (const path of paths) {
const vEl = fileExplorer.fileItems[path];
if (!(vEl == null ? void 0 : vEl.el)) continue;
this.updateItemVisibility(vEl, false);
}
} else {
for (const path of paths) {
const vEl = fileExplorer.fileItems[path];
if (!(vEl == null ? void 0 : vEl.el)) continue;
const shouldShow = path === focusedPath || path.startsWith(focusedPath + "/") || ancestorPaths.has(path);
this.updateItemVisibility(vEl, shouldShow);
}
}
}
}
updateItemVisibility(vEl, shouldShow) {
const currentlyHidden = vEl.el.hasAttribute("data-explorer-focus-hidden");
if (shouldShow && currentlyHidden) {
if (vEl.info) vEl.info.hidden = false;
vEl.el.setCssProps({ display: "" });
vEl.el.removeAttribute("data-explorer-focus-hidden");
} else if (!shouldShow && !currentlyHidden) {
if (vEl.info) vEl.info.hidden = true;
vEl.el.setCssProps({ display: "none" });
vEl.el.setAttribute("data-explorer-focus-hidden", "true");
}
}
updateFocusModeClasses() {
const fileExplorers = getAllFileExplorers(this);
fileExplorers.forEach((fileExplorer) => {
const containerEl = fileExplorer.containerEl;
if (this.isFocus) {
containerEl.addClass("explorer-focus-mode");
} else {
containerEl.removeClass("explorer-focus-mode");
}
});
}
addFileExplorerIcon() {
const handleIconClick = () => {
if (this.isFocus) {
this.exitFocus();
} else {
if (this.settings.focusLevel === "custom") {
if (this.settings.customFolderPath) {
this.enterFocus(this.settings.customFolderPath);
}
} else {
const file = this.app.workspace.getActiveFile();
if (file == null ? void 0 : file.path) {
const focusPath = getFocusPath(file.path, this.settings.focusLevel, this.settings);
this.enterFocus(focusPath);
}
}
}
};
const addIconToFileExplorer = () => {
const fileExplorerLeaves = this.app.workspace.getLeavesOfType("file-explorer");
if (fileExplorerLeaves.length === 0) {
return;
}
const fileExplorerView = fileExplorerLeaves[0].view.containerEl;
const navButtonsContainer = findNavButtonsContainer(fileExplorerView);
if (!navButtonsContainer) {
return;
}
if (this.fileExplorerIcon && navButtonsContainer.contains(this.fileExplorerIcon)) {
return;
}
if (this.fileExplorerIcon && this.fileExplorerIcon.parentElement) {
this.fileExplorerIcon.remove();
}
if (!this.fileExplorerIcon) {
this.fileExplorerIcon = createFileExplorerIcon(this);
this.fileExplorerIcon.setCssProps({ touchAction: "manipulation" });
let touchHandled = false;
this.registerDomEvent(this.fileExplorerIcon, "touchstart", (evt) => {
touchHandled = true;
evt.preventDefault();
evt.stopPropagation();
handleIconClick();
setTimeout(() => {
touchHandled = false;
}, 300);
});
this.registerDomEvent(this.fileExplorerIcon, "click", (evt) => {
if (touchHandled) {
evt.preventDefault();
evt.stopPropagation();
return;
}
evt.preventDefault();
evt.stopPropagation();
handleIconClick();
});
}
this.updateFileExplorerIcon();
insertFileExplorerIcon(this.fileExplorerIcon, navButtonsContainer);
};
addIconToFileExplorer();
this.registerEvent(
this.app.workspace.on("layout-change", () => {
if (this.settings.showFileExplorerIcon) {
addIconToFileExplorer();
}
})
);
}
updateFileExplorerIcon() {
if (!this.fileExplorerIcon) {
return;
}
if (!this.settings.showFileExplorerIcon) {
if (this.fileExplorerIcon.parentElement) {
this.fileExplorerIcon.remove();
}
return;
}
if (!this.fileExplorerIcon.parentElement) {
const fileExplorerLeaves = this.app.workspace.getLeavesOfType("file-explorer");
if (fileExplorerLeaves.length > 0) {
const fileExplorerView = fileExplorerLeaves[0].view.containerEl;
const navButtonsContainer = findNavButtonsContainer(fileExplorerView);
if (navButtonsContainer) {
insertFileExplorerIcon(this.fileExplorerIcon, navButtonsContainer);
}
}
}
if (this.fileExplorerIcon.style.cursor) {
this.fileExplorerIcon.style.removeProperty("cursor");
}
if (this.isFocus) {
this.fileExplorerIcon.addClass("is-active");
} else {
this.fileExplorerIcon.removeClass("is-active");
}
}
updateAutoHideStyles() {
var _a;
const paths = new Set(
((_a = this.settings.autoHidePaths) != null ? _a : []).map((p) => p.trim()).filter((p) => p.length > 0)
);
const fileExplorers = getAllFileExplorers(this);
fileExplorers.forEach((fileExplorer) => {
if (!(fileExplorer == null ? void 0 : fileExplorer.fileItems)) return;
for (const [filePath, vEl] of Object.entries(fileExplorer.fileItems)) {
if (!(vEl == null ? void 0 : vEl.el)) continue;
const shouldHide = paths.has(filePath);
const isHidden = vEl.el.hasAttribute("data-explorer-focus-auto-hidden");
if (shouldHide && !isHidden) {
vEl.el.setCssProps({ display: "none" });
vEl.el.setAttribute("data-explorer-focus-auto-hidden", "true");
} else if (!shouldHide && isHidden) {
vEl.el.setCssProps({ display: "" });
vEl.el.removeAttribute("data-explorer-focus-auto-hidden");
}
}
});
}
onunload() {
if (this.fileExplorerIcon) {
this.fileExplorerIcon.remove();
this.fileExplorerIcon = null;
}
}
};
var main_default = ExplorerFocusPlugin;
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["src/main.ts", "src/types.ts", "src/ui/settings-tab.ts", "src/utils/file-explorer-patch.ts", "node_modules/.pnpm/monkey-around@3.0.0/node_modules/monkey-around/dist/index.mjs", "src/ui/folder-suggest.ts", "src/ui/auto-hide-modal.ts", "src/utils/focus.ts", "src/commands/index.ts", "src/utils/file-explorer.ts"],
  "sourcesContent": ["import { App, Plugin, PluginManifest } from \"obsidian\";\r\nimport { ExplorerFocusSettings, DEFAULT_SETTINGS } from './types';\r\nimport { ExplorerFocusSettingTab } from './ui/settings-tab';\r\nimport { registerCommands } from './commands';\r\nimport { createFileExplorerIcon, insertFileExplorerIcon, findNavButtonsContainer } from './utils/file-explorer';\r\nimport { patchFileExplorer as patchFileExplorerUtil, getAllFileExplorers, FileExplorerView } from './utils/file-explorer-patch';\r\nimport { getFocusPath } from './utils/focus';\r\n\r\nexport class ExplorerFocusPlugin extends Plugin {\r\n\tisFocus: boolean;\r\n\tfocusedPath: string | null;\r\n\tsettings!: ExplorerFocusSettings;\r\n\tfileExplorerIcon: HTMLElement | null;\r\n\r\n\tconstructor(app: App, manifest: PluginManifest) {\r\n\t\tsuper(app, manifest);\r\n\t\tthis.isFocus = false;\r\n\t\tthis.focusedPath = null;\r\n\t\tthis.fileExplorerIcon = null;\r\n\t}\r\n\r\n\tasync onload() {\r\n\t\tawait this.loadSettings();\r\n\r\n\t\tthis.addSettingTab(new ExplorerFocusSettingTab(this.app, this));\r\n\r\n\t\tregisterCommands(this);\r\n\r\n\t\tif (this.settings.showFileExplorerIcon) {\r\n\t\t\tthis.addFileExplorerIcon();\r\n\t\t}\r\n\r\n\t\tthis.app.workspace.onLayoutReady(() => {\r\n\t\t\tthis.patchAllFileExplorers();\r\n\t\t\tthis.updateFocusModeClasses();\r\n\t\t\tthis.updateAutoHideStyles();\r\n\t\t});\r\n\r\n\t\tthis.app.workspace.on(\"layout-change\", () => {\r\n\t\t\tthis.patchAllFileExplorers();\r\n\t\t\tthis.updateFocusModeClasses();\r\n\t\t\tthis.updateAutoHideStyles();\r\n\t\t});\r\n\t}\r\n\r\n\tasync loadSettings() {\r\n\t\tconst loadedData = await this.loadData() as Partial<ExplorerFocusSettings> | null;\r\n\t\tthis.settings = Object.assign({}, DEFAULT_SETTINGS, loadedData);\r\n\t}\r\n\r\n\tasync saveSettings() {\r\n\t\tawait this.saveData(this.settings);\r\n\t}\r\n\r\n\ttoggleFocus(path: string | undefined) {\r\n\t\tif (this.isFocus) {\r\n\t\t\tthis.exitFocus();\r\n\t\t} else if (path) {\r\n\t\t\tthis.enterFocus(path);\r\n\t\t}\r\n\t}\r\n\r\n\tenterFocus(path: string) {\r\n\t\tthis.isFocus = true;\r\n\t\tthis.focusedPath = path;\r\n\t\t\r\n\t\t// Update icon if it exists\r\n\t\tthis.updateFileExplorerIcon();\r\n\t\t\r\n\t\t// Update CSS classes\r\n\t\tthis.updateFocusModeClasses();\r\n\t\t\r\n\t\t// Trigger file explorer refresh on all file explorer instances\r\n\t\tconst fileExplorers = getAllFileExplorers(this);\r\n\t\tfileExplorers.forEach(fileExplorer => {\r\n\t\t\tif (fileExplorer?.requestSort) {\r\n\t\t\t\tfileExplorer.requestSort();\r\n\t\t\t}\r\n\t\t});\r\n\t\t\r\n\t\t// Force refresh by manually updating visibility (especially important on mobile)\r\n\t\t// Use setTimeout to ensure DOM is ready after requestSort\r\n\t\tsetTimeout(() => {\r\n\t\t\tfileExplorers.forEach(fileExplorer => {\r\n\t\t\t\tthis.refreshFileExplorerVisibility(fileExplorer);\r\n\t\t\t});\r\n\t\t}, 0);\r\n\t}\r\n\r\n\tpatchAllFileExplorers(): void {\r\n\t\t// Patch the prototype (once) and mark all instances\r\n\t\tpatchFileExplorerUtil(this);\r\n\t\tconst fileExplorers = getAllFileExplorers(this);\r\n\t\tfileExplorers.forEach(fileExplorer => {\r\n\t\t\tfileExplorer.fileExplorerPlusPatched = true;\r\n\t\t});\r\n\t}\r\n\r\n\texitFocus() {\r\n\t\tthis.isFocus = false;\r\n\t\tthis.focusedPath = null;\r\n\r\n\t\t// Update icon if it exists\r\n\t\tthis.updateFileExplorerIcon();\r\n\r\n\t\t// Update CSS classes\r\n\t\tthis.updateFocusModeClasses();\r\n\r\n\t\t// Trigger file explorer refresh on all file explorer instances\r\n\t\tconst fileExplorers = getAllFileExplorers(this);\r\n\t\tfileExplorers.forEach(fileExplorer => {\r\n\t\t\tif (fileExplorer?.requestSort) {\r\n\t\t\t\tfileExplorer.requestSort();\r\n\t\t\t}\r\n\t\t});\r\n\r\n\t\t// Force refresh by manually updating visibility (especially important on mobile)\r\n\t\t// Use setTimeout to ensure DOM is ready after requestSort\r\n\t\tsetTimeout(() => {\r\n\t\t\tfileExplorers.forEach(fileExplorer => {\r\n\t\t\t\tthis.refreshFileExplorerVisibility(fileExplorer);\r\n\t\t\t});\r\n\t\t\t// Re-apply auto-hide after focus mode clears visibility\r\n\t\t\tthis.updateAutoHideStyles();\r\n\t\t}, 0);\r\n\t}\r\n\r\n\trefreshFileExplorerVisibility(fileExplorer: FileExplorerView): void {\r\n\t\tif (!fileExplorer?.fileItems) {\r\n\t\t\treturn;\r\n\t\t}\r\n\r\n\t\t// When exiting focus mode, the patch handles visibility reset via requestSort.\r\n\t\t// We only need to clear any stale hidden attributes.\r\n\t\tif (!this.isFocus || !this.focusedPath) {\r\n\t\t\tObject.values(fileExplorer.fileItems).forEach((vEl) => {\r\n\t\t\t\tif (!vEl?.el) return;\r\n\t\t\t\t// Only update if currently marked hidden to minimize DOM operations\r\n\t\t\t\tif (vEl.el.hasAttribute('data-explorer-focus-hidden')) {\r\n\t\t\t\t\tif (vEl.info) vEl.info.hidden = false;\r\n\t\t\t\t\tvEl.el.setCssProps({ display: '' });\r\n\t\t\t\t\tvEl.el.removeAttribute('data-explorer-focus-hidden');\r\n\t\t\t\t}\r\n\t\t\t});\r\n\t\t\treturn;\r\n\t\t}\r\n\r\n\t\tconst focusedPath = this.focusedPath;\r\n\r\n\t\t// Pre-compute ancestor paths as a Set for O(1) lookup\r\n\t\tconst ancestorPaths = new Set<string>();\r\n\t\tconst pathParts = focusedPath.split('/');\r\n\t\tfor (let i = 1; i < pathParts.length; i++) {\r\n\t\t\tancestorPaths.add(pathParts.slice(0, i).join('/'));\r\n\t\t}\r\n\r\n\t\t// Group items by their top-level folder for efficient subtree skipping\r\n\t\tconst topLevelFolders = new Map<string, string[]>();\r\n\t\tconst rootItems: string[] = [];\r\n\r\n\t\tfor (const path of Object.keys(fileExplorer.fileItems)) {\r\n\t\t\tconst firstSlash = path.indexOf('/');\r\n\t\t\tif (firstSlash === -1) {\r\n\t\t\t\trootItems.push(path);\r\n\t\t\t} else {\r\n\t\t\t\tconst topLevel = path.substring(0, firstSlash);\r\n\t\t\t\tif (!topLevelFolders.has(topLevel)) {\r\n\t\t\t\t\ttopLevelFolders.set(topLevel, []);\r\n\t\t\t\t}\r\n\t\t\t\ttopLevelFolders.get(topLevel)!.push(path);\r\n\t\t\t}\r\n\t\t}\r\n\r\n\t\t// Determine which top-level folders need full processing\r\n\t\tconst focusedTopLevel = focusedPath.split('/')[0];\r\n\r\n\t\t// Process root-level items\r\n\t\tfor (const path of rootItems) {\r\n\t\t\tconst vEl = fileExplorer.fileItems[path];\r\n\t\t\tif (!vEl?.el) continue;\r\n\r\n\t\t\tconst shouldShow = path === focusedPath ||\r\n\t\t\t\tfocusedPath.startsWith(path + '/') ||\r\n\t\t\t\tpath.startsWith(focusedPath + '/');\r\n\t\t\tthis.updateItemVisibility(vEl, shouldShow);\r\n\t\t}\r\n\r\n\t\t// Process each top-level folder\r\n\t\tfor (const [topLevel, paths] of topLevelFolders) {\r\n\t\t\t// Check if this top-level folder needs detailed processing\r\n\t\t\tconst isInFocusPath = topLevel === focusedTopLevel ||\r\n\t\t\t\tancestorPaths.has(topLevel) ||\r\n\t\t\t\tfocusedPath === topLevel;\r\n\r\n\t\t\tif (!isInFocusPath) {\r\n\t\t\t\t// This entire subtree should be hidden - batch hide all items\r\n\t\t\t\tfor (const path of paths) {\r\n\t\t\t\t\tconst vEl = fileExplorer.fileItems[path];\r\n\t\t\t\t\tif (!vEl?.el) continue;\r\n\t\t\t\t\tthis.updateItemVisibility(vEl, false);\r\n\t\t\t\t}\r\n\t\t\t} else {\r\n\t\t\t\t// This subtree needs individual evaluation\r\n\t\t\t\tfor (const path of paths) {\r\n\t\t\t\t\tconst vEl = fileExplorer.fileItems[path];\r\n\t\t\t\t\tif (!vEl?.el) continue;\r\n\r\n\t\t\t\t\tconst shouldShow = path === focusedPath ||\r\n\t\t\t\t\t\tpath.startsWith(focusedPath + '/') ||\r\n\t\t\t\t\t\tancestorPaths.has(path);\r\n\t\t\t\t\tthis.updateItemVisibility(vEl, shouldShow);\r\n\t\t\t\t}\r\n\t\t\t}\r\n\t\t}\r\n\t}\r\n\r\n\tprivate updateItemVisibility(vEl: { info?: { hidden: boolean }; el: HTMLElement }, shouldShow: boolean): void {\r\n\t\tconst currentlyHidden = vEl.el.hasAttribute('data-explorer-focus-hidden');\r\n\r\n\t\t// Only update DOM if state actually changes\r\n\t\tif (shouldShow && currentlyHidden) {\r\n\t\t\tif (vEl.info) vEl.info.hidden = false;\r\n\t\t\tvEl.el.setCssProps({ display: '' });\r\n\t\t\tvEl.el.removeAttribute('data-explorer-focus-hidden');\r\n\t\t} else if (!shouldShow && !currentlyHidden) {\r\n\t\t\tif (vEl.info) vEl.info.hidden = true;\r\n\t\t\tvEl.el.setCssProps({ display: 'none' });\r\n\t\t\tvEl.el.setAttribute('data-explorer-focus-hidden', 'true');\r\n\t\t}\r\n\t}\r\n\r\n\tupdateFocusModeClasses(): void {\r\n\t\tconst fileExplorers = getAllFileExplorers(this);\r\n\t\tfileExplorers.forEach(fileExplorer => {\r\n\t\t\tconst containerEl = fileExplorer.containerEl;\r\n\t\t\tif (this.isFocus) {\r\n\t\t\t\tcontainerEl.addClass(\"explorer-focus-mode\");\r\n\t\t\t} else {\r\n\t\t\t\tcontainerEl.removeClass(\"explorer-focus-mode\");\r\n\t\t\t}\r\n\t\t});\r\n\t}\r\n\r\n\r\n\taddFileExplorerIcon() {\r\n\t\tconst handleIconClick = () => {\r\n\t\t\tif (this.isFocus) {\r\n\t\t\t\tthis.exitFocus();\r\n\t\t\t} else {\r\n\t\t\t\tif (this.settings.focusLevel === 'custom') {\r\n\t\t\t\t\t// For custom folder, use the custom path directly\r\n\t\t\t\t\tif (this.settings.customFolderPath) {\r\n\t\t\t\t\t\tthis.enterFocus(this.settings.customFolderPath);\r\n\t\t\t\t\t}\r\n\t\t\t\t} else {\r\n\t\t\t\t\t// For other levels, need the current file\r\n\t\t\t\t\tconst file = this.app.workspace.getActiveFile();\r\n\t\t\t\t\tif (file?.path) {\r\n\t\t\t\t\t\tconst focusPath = getFocusPath(file.path, this.settings.focusLevel, this.settings);\r\n\t\t\t\t\t\tthis.enterFocus(focusPath);\r\n\t\t\t\t\t}\r\n\t\t\t\t}\r\n\t\t\t}\r\n\t\t};\r\n\r\n\t\tconst addIconToFileExplorer = () => {\r\n\t\t\tconst fileExplorerLeaves = this.app.workspace.getLeavesOfType('file-explorer');\r\n\t\t\tif (fileExplorerLeaves.length === 0) {\r\n\t\t\t\treturn;\r\n\t\t\t}\r\n\r\n\t\t\tconst fileExplorerView = fileExplorerLeaves[0].view.containerEl;\r\n\t\t\tconst navButtonsContainer = findNavButtonsContainer(fileExplorerView);\r\n\t\t\t\r\n\t\t\tif (!navButtonsContainer) {\r\n\t\t\t\treturn;\r\n\t\t\t}\r\n\r\n\t\t\t// Check if icon already exists in this container\r\n\t\t\tif (this.fileExplorerIcon && navButtonsContainer.contains(this.fileExplorerIcon)) {\r\n\t\t\t\treturn;\r\n\t\t\t}\r\n\r\n\t\t\t// Remove old icon if it exists elsewhere\r\n\t\t\tif (this.fileExplorerIcon && this.fileExplorerIcon.parentElement) {\r\n\t\t\t\tthis.fileExplorerIcon.remove();\r\n\t\t\t}\r\n\r\n\t\t\t// Create the icon button if it doesn't exist\r\n\t\t\tif (!this.fileExplorerIcon) {\r\n\t\t\t\tthis.fileExplorerIcon = createFileExplorerIcon(this);\r\n\r\n\t\t\t\t// Don't set cursor - let it inherit from .clickable-icon class (cursor: var(--cursor))\r\n\t\t\t\t// This matches other file explorer icons which use the CSS variable\r\n\t\t\t\tthis.fileExplorerIcon.setCssProps({ touchAction: 'manipulation' });\r\n\r\n\t\t\t\t// Use a unified handler that works for both click and touch\r\n\t\t\t\t// On mobile, touch events typically trigger click, but we handle both to be safe\r\n\t\t\t\tlet touchHandled = false;\r\n\t\t\t\t\r\n\t\t\t\tthis.registerDomEvent(this.fileExplorerIcon, 'touchstart', (evt) => {\r\n\t\t\t\t\ttouchHandled = true;\r\n\t\t\t\t\tevt.preventDefault();\r\n\t\t\t\t\tevt.stopPropagation();\r\n\t\t\t\t\thandleIconClick();\r\n\t\t\t\t\t// Reset flag after a short delay\r\n\t\t\t\t\tsetTimeout(() => { touchHandled = false; }, 300);\r\n\t\t\t\t});\r\n\r\n\t\t\t\tthis.registerDomEvent(this.fileExplorerIcon, 'click', (evt) => {\r\n\t\t\t\t\t// If touchstart already handled it, skip click to avoid double-firing\r\n\t\t\t\t\tif (touchHandled) {\r\n\t\t\t\t\t\tevt.preventDefault();\r\n\t\t\t\t\t\tevt.stopPropagation();\r\n\t\t\t\t\t\treturn;\r\n\t\t\t\t\t}\r\n\t\t\t\t\tevt.preventDefault();\r\n\t\t\t\t\tevt.stopPropagation();\r\n\t\t\t\t\thandleIconClick();\r\n\t\t\t\t});\r\n\t\t\t}\r\n\t\t\t\r\n\t\t\t// Set/update icon\r\n\t\t\tthis.updateFileExplorerIcon();\r\n\r\n\t\t\t// Insert icon in correct position\r\n\t\t\tinsertFileExplorerIcon(this.fileExplorerIcon, navButtonsContainer);\r\n\t\t};\r\n\r\n\t\t// Try to add immediately\r\n\t\taddIconToFileExplorer();\r\n\r\n\t\t// Also listen for workspace changes in case file explorer is created later\r\n\t\tthis.registerEvent(\r\n\t\t\tthis.app.workspace.on('layout-change', () => {\r\n\t\t\t\tif (this.settings.showFileExplorerIcon) {\r\n\t\t\t\t\taddIconToFileExplorer();\r\n\t\t\t\t}\r\n\t\t\t})\r\n\t\t);\r\n\t}\r\n\r\n\tupdateFileExplorerIcon() {\r\n\t\tif (!this.fileExplorerIcon) {\r\n\t\t\treturn;\r\n\t\t}\r\n\r\n\t\tif (!this.settings.showFileExplorerIcon) {\r\n\t\t\tif (this.fileExplorerIcon.parentElement) {\r\n\t\t\t\tthis.fileExplorerIcon.remove();\r\n\t\t\t}\r\n\t\t\treturn;\r\n\t\t}\r\n\r\n\t\t// Ensure icon is in DOM if setting is enabled\r\n\t\tif (!this.fileExplorerIcon.parentElement) {\r\n\t\t\t// Re-insert the icon into the file explorer\r\n\t\t\tconst fileExplorerLeaves = this.app.workspace.getLeavesOfType('file-explorer');\r\n\t\t\tif (fileExplorerLeaves.length > 0) {\r\n\t\t\t\tconst fileExplorerView = fileExplorerLeaves[0].view.containerEl;\r\n\t\t\t\tconst navButtonsContainer = findNavButtonsContainer(fileExplorerView);\r\n\t\t\t\t\r\n\t\t\t\tif (navButtonsContainer) {\r\n\t\t\t\t\tinsertFileExplorerIcon(this.fileExplorerIcon, navButtonsContainer);\r\n\t\t\t\t}\r\n\t\t\t}\r\n\t\t}\r\n\r\n\t\t// Don't set cursor - let it inherit from .clickable-icon class (cursor: var(--cursor))\r\n\t\t// This matches other file explorer icons which use the CSS variable\r\n\t\tif (this.fileExplorerIcon.style.cursor) {\r\n\t\t\tthis.fileExplorerIcon.style.removeProperty('cursor');\r\n\t\t}\r\n\r\n\t\t// Toggle is-active class based on focus state\r\n\t\tif (this.isFocus) {\r\n\t\t\tthis.fileExplorerIcon.addClass('is-active');\r\n\t\t} else {\r\n\t\t\tthis.fileExplorerIcon.removeClass('is-active');\r\n\t\t}\r\n\t}\r\n\r\n\tupdateAutoHideStyles(): void {\r\n\t\tconst paths = new Set(\r\n\t\t\t(this.settings.autoHidePaths ?? [])\r\n\t\t\t\t.map(p => p.trim())\r\n\t\t\t\t.filter(p => p.length > 0)\r\n\t\t);\r\n\r\n\t\tconst fileExplorers = getAllFileExplorers(this);\r\n\t\tfileExplorers.forEach(fileExplorer => {\r\n\t\t\tif (!fileExplorer?.fileItems) return;\r\n\r\n\t\t\tfor (const [filePath, vEl] of Object.entries(fileExplorer.fileItems)) {\r\n\t\t\t\tif (!vEl?.el) continue;\r\n\t\t\t\tconst shouldHide = paths.has(filePath);\r\n\t\t\t\tconst isHidden = vEl.el.hasAttribute('data-explorer-focus-auto-hidden');\r\n\r\n\t\t\t\tif (shouldHide && !isHidden) {\r\n\t\t\t\t\tvEl.el.setCssProps({ display: 'none' });\r\n\t\t\t\t\tvEl.el.setAttribute('data-explorer-focus-auto-hidden', 'true');\r\n\t\t\t\t} else if (!shouldHide && isHidden) {\r\n\t\t\t\t\tvEl.el.setCssProps({ display: '' });\r\n\t\t\t\t\tvEl.el.removeAttribute('data-explorer-focus-auto-hidden');\r\n\t\t\t\t}\r\n\t\t\t}\r\n\t\t});\r\n\t}\r\n\r\n\tonunload() {\r\n\t\tif (this.fileExplorerIcon) {\r\n\t\t\tthis.fileExplorerIcon.remove();\r\n\t\t\tthis.fileExplorerIcon = null;\r\n\t\t}\r\n\t}\r\n}\r\n\r\nexport default ExplorerFocusPlugin;\r\n\r\n", "export interface ExplorerFocusSettings {\n\tshowRightClickMenu: boolean;\n\tshowFileExplorerIcon: boolean;\n\tfocusLevel: 'file' | 'parent' | 'grandparent' | 'greatgrandparent' | 'custom';\n\tcustomFolderPath: string;\n\thideAncestorFolders: boolean;\n\tautoHidePaths: string[];\n}\n\nexport const DEFAULT_SETTINGS: ExplorerFocusSettings = {\n\tshowRightClickMenu: true,\n\tshowFileExplorerIcon: true,\n\tfocusLevel: 'parent',\n\tcustomFolderPath: '',\n\thideAncestorFolders: false,\n\tautoHidePaths: []\n};\n\n", "import { App, PluginSettingTab, SettingGroup } from \"obsidian\";\r\nimport { ExplorerFocusPlugin } from \"../main\";\r\nimport { getAllFileExplorers } from \"../utils/file-explorer-patch\";\r\nimport { FolderSuggest } from \"./folder-suggest\";\r\nimport { AutoHideModal } from \"./auto-hide-modal\";\r\n\r\n\r\nexport class ExplorerFocusSettingTab extends PluginSettingTab {\r\n\tplugin: ExplorerFocusPlugin;\r\n\tpublic icon = 'lucide-focus';\r\n\r\n\tconstructor(app: App, plugin: ExplorerFocusPlugin) {\r\n\t\tsuper(app, plugin);\r\n\t\tthis.plugin = plugin;\r\n\t}\r\n\r\n\tdisplay(): void {\r\n\t\tconst { containerEl } = this;\r\n\r\n\t\tcontainerEl.empty();\r\n\r\n\t\t// First group (no heading)\r\n\t\tconst generalGroup = new SettingGroup(containerEl);\r\n\r\n\t\tgeneralGroup.addSetting(setting => {\r\n\t\t\tsetting\r\n\t\t\t\t.setName(\"Show right-click menu option\")\r\n\t\t\t\t.addToggle(toggle => toggle\r\n\t\t\t\t\t.setValue(this.plugin.settings.showRightClickMenu)\r\n\t\t\t\t\t.onChange(async value => {\r\n\t\t\t\t\t\tthis.plugin.settings.showRightClickMenu = value;\r\n\t\t\t\t\t\tawait this.plugin.saveSettings();\r\n\t\t\t\t\t}));\r\n\t\t});\r\n\r\n\t\tgeneralGroup.addSetting(setting => {\r\n\t\t\tsetting\r\n\t\t\t\t.setName(\"Show file explorer icon\")\r\n\t\t\t\t.addToggle(toggle => toggle\r\n\t\t\t\t\t.setValue(this.plugin.settings.showFileExplorerIcon)\r\n\t\t\t\t\t.onChange(async value => {\r\n\t\t\t\t\t\tthis.plugin.settings.showFileExplorerIcon = value;\r\n\t\t\t\t\t\tawait this.plugin.saveSettings();\r\n\t\t\t\t\t\tthis.plugin.updateFileExplorerIcon();\r\n\t\t\t\t\t}));\r\n\t\t});\r\n\r\n\t\tgeneralGroup.addSetting(setting => {\r\n\t\t\tsetting\r\n\t\t\t\t.setName(\"Command focus level\")\r\n\t\t\t\t.setDesc(\"Determines what to focus when using the toggle command or file explorer icon. Right-click menu always focuses the clicked file/folder.\")\r\n\t\t\t\t.addDropdown(dropdown => dropdown\r\n\t\t\t\t\t.addOption('file', \"Current file only\")\r\n\t\t\t\t\t.addOption('parent', \"Parent folder\")\r\n\t\t\t\t\t.addOption('grandparent', \"Grandparent folder\")\r\n\t\t\t\t\t.addOption('greatgrandparent', \"Great grandparent folder\")\r\n\t\t\t\t\t.addOption('custom', \"Specific folder\")\r\n\t\t\t\t\t.setValue(this.plugin.settings.focusLevel)\r\n\t\t\t\t\t.onChange(async (value: string) => {\r\n\t\t\t\t\t\tthis.plugin.settings.focusLevel = value as 'file' | 'parent' | 'grandparent' | 'greatgrandparent' | 'custom';\r\n\t\t\t\t\t\tawait this.plugin.saveSettings();\r\n\t\t\t\t\t\t// Refresh file explorer if in focus mode\r\n\t\t\t\t\t\tif (this.plugin.isFocus) {\r\n\t\t\t\t\t\t\tconst fileExplorers = getAllFileExplorers(this.plugin);\r\n\t\t\t\t\t\t\tfileExplorers.forEach((fileExplorer) => {\r\n\t\t\t\t\t\t\t\tif (fileExplorer?.requestSort) {\r\n\t\t\t\t\t\t\t\t\tfileExplorer.requestSort();\r\n\t\t\t\t\t\t\t\t}\r\n\t\t\t\t\t\t\t});\r\n\t\t\t\t\t\t}\r\n\t\t\t\t\t\tthis.display(); // Refresh to show/hide custom folder input\r\n\t\t\t\t\t}));\r\n\t\t});\r\n\r\n\t\t// Add custom folder path input that only shows when custom is selected\r\n\t\tif (this.plugin.settings.focusLevel === 'custom') {\r\n\t\t\tgeneralGroup.addSetting(setting => {\r\n\t\t\t\tsetting\r\n\t\t\t\t\t.setName(\"Custom folder path\")\r\n\t\t\t\t\t.setDesc(\"Enter a folder path (folder/subfolder). This folder will be focused regardless of which file is open.\")\r\n\t\t\t\t\t.addText(text => {\r\n\t\t\t\t\t\tnew FolderSuggest(this.app, text.inputEl);\r\n\t\t\t\t\t\ttext\r\n\t\t\t\t\t\t\t.setPlaceholder('Folder/subfolder')\r\n\t\t\t\t\t\t\t.setValue(this.plugin.settings.customFolderPath)\r\n\t\t\t\t\t\t\t.onChange(async value => {\r\n\t\t\t\t\t\t\t\tthis.plugin.settings.customFolderPath = value;\r\n\t\t\t\t\t\t\t\tawait this.plugin.saveSettings();\r\n\t\t\t\t\t\t\t});\r\n\t\t\t\t\t});\r\n\t\t\t});\r\n\t\t}\r\n\r\n\t\t// Auto-hide folders section\r\n\t\tconst autoHideGroup = new SettingGroup(containerEl)\r\n\t\t\t.setHeading(\"Auto-hide folders\");\r\n\r\n\t\tautoHideGroup.addSetting(setting => {\r\n\t\t\tconst autoHidePaths = (this.plugin.settings.autoHidePaths ?? []).filter(p => p.trim().length > 0);\r\n\r\n\t\t\tconst descFragment = document.createDocumentFragment();\r\n\t\t\tdescFragment.appendText(\"These folders are always hidden from the file explorer.\");\r\n\r\n\t\t\tif (autoHidePaths.length > 0) {\r\n\t\t\t\tconst list = descFragment.createEl(\"ul\");\r\n\t\t\t\tautoHidePaths.forEach(p => {\r\n\t\t\t\t\tlist.createEl(\"li\", { text: p });\r\n\t\t\t\t});\r\n\t\t\t}\r\n\r\n\t\t\tsetting\r\n\t\t\t\t.setName(\"Hidden folders\")\r\n\t\t\t\t.setDesc(descFragment)\r\n\t\t\t\t.addButton(button => button\r\n\t\t\t\t\t.setButtonText(\"Manage\")\r\n\t\t\t\t\t.onClick(() => {\r\n\t\t\t\t\t\tnew AutoHideModal(this.app, this.plugin, () => {\r\n\t\t\t\t\t\t\tthis.display();\r\n\t\t\t\t\t\t}).open();\r\n\t\t\t\t\t}));\r\n\t\t});\r\n\r\n\t}\r\n}\r\n\r\n", "import { Notice, View, TAbstractFile, Platform } from \"obsidian\";\r\nimport { around } from \"monkey-around\";\r\nimport { ExplorerFocusPlugin } from \"../main\";\r\nimport \"../types.d\";\r\n\r\n// Type definitions from module augmentation\r\ntype PathVirtualElement = {\r\n\tchildrenEl: HTMLElement;\r\n\tcollapseEl: HTMLElement;\r\n\tcollapsed: boolean;\r\n\tcollapsible: boolean;\r\n\tcoverEl: HTMLElement;\r\n\tel: HTMLElement;\r\n\tfile: TAbstractFile;\r\n\tinfo: {\r\n\t\tchildLeft: number;\r\n\t\tchildLeftPadding: number;\r\n\t\tchildTop: number;\r\n\t\tcomputed: boolean;\r\n\t\theight: number;\r\n\t\thidden: boolean;\r\n\t\tpinned?: boolean;\r\n\t\tnext: boolean;\r\n\t\tqueued: boolean;\r\n\t\twidth: number;\r\n\t};\r\n\tinnerEl: HTMLElement;\r\n\tparent?: PathVirtualElement;\r\n\tpusherEl: HTMLElement;\r\n\tselfEl: HTMLElement;\r\n\tvChildren: {\r\n\t\tchildren: PathVirtualElement;\r\n\t};\r\n};\r\n\r\nexport type FileExplorerView = View & {\r\n\trequestSort(): void;\r\n\tfileExplorerPlusPatched?: boolean;\r\n\tfileItems: {\r\n\t\t[key: string]: PathVirtualElement;\r\n\t};\r\n\tcontainerEl: HTMLElement;\r\n};\r\n\r\ntype GetSortedFolderItemsFunction = (folder: TAbstractFile) => PathVirtualElement[];\r\n\r\n// Global flags to track patching state\r\nlet prototypePatched = false;\r\nlet patchingFailed = false;\r\n\r\nexport function isPatchingFailed(): boolean {\r\n\treturn patchingFailed;\r\n}\r\n\r\nexport function patchFileExplorer(plugin: ExplorerFocusPlugin): void {\r\n\t// If patching previously failed, don't retry - rely on CSS fallback\r\n\tif (patchingFailed) {\r\n\t\treturn;\r\n\t}\r\n\r\n\t// For mobile, we explicitly skip prototype patching as it often fails\r\n\t// and rely on our robust DOM-based fallback implemented in main.ts\r\n\tif (Platform.isMobile) {\r\n\t\tpatchingFailed = true; // Set to true to avoid retries, but don't show notice\r\n\t\treturn;\r\n\t}\r\n\r\n\tconst fileExplorer = getFileExplorer(plugin);\r\n\r\n\tif (!fileExplorer) {\r\n\t\treturn; // File explorer not available yet, will be patched on layout-change\r\n\t}\r\n\r\n\t// Patch the prototype only once (affects all instances)\r\n\tif (!prototypePatched) {\r\n\t\ttry {\r\n\t\t\tconst prototype = Object.getPrototypeOf(fileExplorer);\r\n\r\n\t\t\t// Verify the method we're patching exists\r\n\t\t\tif (typeof prototype.getSortedFolderItems !== 'function') {\r\n\t\t\t\tthrow new Error('getSortedFolderItems method not found on file explorer prototype');\r\n\t\t\t}\r\n\r\n\t\t\tplugin.register(\r\n\t\t\t\taround(prototype, {\r\n\t\t\t\t\tgetSortedFolderItems(old: GetSortedFolderItemsFunction) {\r\n\t\t\t\t\t\treturn function (this: FileExplorerView, folder: TAbstractFile): PathVirtualElement[] {\r\n\t\t\t\t\t\t\tlet sortedChildren: PathVirtualElement[] = old.call(this, folder);\r\n\r\n\t\t\t\t\t\t\t// Apply focus hiding if focus mode is active\r\n\t\t\t\t\t\t\tif (plugin.isFocus && plugin.focusedPath) {\r\n\t\t\t\t\t\t\t\tconst focusedPath = plugin.focusedPath;\r\n\r\n\t\t\t\t\t\t\t\tsortedChildren = sortedChildren.filter((vEl) => {\r\n\t\t\t\t\t\t\t\t\tconst filePath = vEl.file.path;\r\n\r\n\t\t\t\t\t\t\t\t\t// Show items that match the focused path exactly\r\n\t\t\t\t\t\t\t\t\tif (filePath === focusedPath) {\r\n\t\t\t\t\t\t\t\t\t\tvEl.info.hidden = false;\r\n\t\t\t\t\t\t\t\t\t\treturn true;\r\n\t\t\t\t\t\t\t\t\t}\r\n\r\n\t\t\t\t\t\t\t\t\t// Show items that are children of the focused path\r\n\t\t\t\t\t\t\t\t\tif (filePath.startsWith(focusedPath + \"/\")) {\r\n\t\t\t\t\t\t\t\t\t\tvEl.info.hidden = false;\r\n\t\t\t\t\t\t\t\t\t\treturn true;\r\n\t\t\t\t\t\t\t\t\t}\r\n\r\n\t\t\t\t\t\t\t\t\t// Show items that are ancestors of the focused path (parent chain)\r\n\t\t\t\t\t\t\t\t\tif (focusedPath.startsWith(filePath + \"/\")) {\r\n\t\t\t\t\t\t\t\t\t\tvEl.info.hidden = false;\r\n\t\t\t\t\t\t\t\t\t\treturn true;\r\n\t\t\t\t\t\t\t\t\t}\r\n\r\n\t\t\t\t\t\t\t\t\t// Hide everything else\r\n\t\t\t\t\t\t\t\t\tvEl.info.hidden = true;\r\n\t\t\t\t\t\t\t\t\treturn false;\r\n\t\t\t\t\t\t\t\t});\r\n\t\t\t\t\t\t\t} else {\r\n\t\t\t\t\t\t\t\t// Ensure all items are visible when not in focus mode\r\n\t\t\t\t\t\t\t\tsortedChildren.forEach((vEl) => {\r\n\t\t\t\t\t\t\t\t\tvEl.info.hidden = false;\r\n\t\t\t\t\t\t\t\t});\r\n\t\t\t\t\t\t\t}\r\n\r\n\t\t\t\t\t\t\treturn sortedChildren;\r\n\t\t\t\t\t\t};\r\n\t\t\t\t\t},\r\n\t\t\t\t}),\r\n\t\t\t);\r\n\r\n\t\t\tprototypePatched = true;\r\n\t\t} catch (error) {\r\n\t\t\tpatchingFailed = true;\r\n\t\t\tconsole.warn(\r\n\t\t\t\t'[Explorer Focus] Failed to patch file explorer. ' +\r\n\t\t\t\t'The plugin will use CSS-based hiding as a fallback, which may be less precise. ' +\r\n\t\t\t\t'This usually happens after an Obsidian update - please check for plugin updates.',\r\n\t\t\t\terror\r\n\t\t\t);\r\n\t\t\tnew Notice(\r\n\t\t\t\t'Explorer Focus: File explorer patching failed. ' +\r\n\t\t\t\t'The plugin will still work but may need an update for full functionality.',\r\n\t\t\t\t8000\r\n\t\t\t);\r\n\t\t\treturn;\r\n\t\t}\r\n\t}\r\n\r\n\t// Mark this instance as patched\r\n\tfileExplorer.fileExplorerPlusPatched = true;\r\n}\r\n\r\nexport function getFileExplorer(plugin: ExplorerFocusPlugin): FileExplorerView | undefined {\r\n\tconst fileExplorerContainer = plugin.app.workspace.getLeavesOfType(\"file-explorer\")?.first();\r\n\treturn fileExplorerContainer?.view as FileExplorerView;\r\n}\r\n\r\nexport function getAllFileExplorers(plugin: ExplorerFocusPlugin): FileExplorerView[] {\r\n\tconst fileExplorerLeaves = plugin.app.workspace.getLeavesOfType(\"file-explorer\");\r\n\treturn fileExplorerLeaves.map(leaf => leaf.view as FileExplorerView);\r\n}\r\n\r\n", "function around(obj, factories) {\n  const removers = Object.keys(factories).map((key) => around1(obj, key, factories[key]));\n  return removers.length === 1 ? removers[0] : function() {\n    removers.forEach((r) => r());\n  };\n}\nfunction around1(obj, method, createWrapper) {\n  const inherited = obj[method], hadOwn = obj.hasOwnProperty(method), original = hadOwn ? inherited : function() {\n    return Object.getPrototypeOf(obj)[method].apply(this, arguments);\n  };\n  let current = createWrapper(original);\n  if (inherited)\n    Object.setPrototypeOf(current, inherited);\n  Object.setPrototypeOf(wrapper, current);\n  obj[method] = wrapper;\n  return remove;\n  function wrapper(...args) {\n    if (current === original && obj[method] === wrapper)\n      remove();\n    return current.apply(this, args);\n  }\n  function remove() {\n    if (obj[method] === wrapper) {\n      if (hadOwn)\n        obj[method] = original;\n      else\n        delete obj[method];\n    }\n    if (current === original)\n      return;\n    current = original;\n    Object.setPrototypeOf(wrapper, inherited || Function);\n  }\n}\nfunction dedupe(key, oldFn, newFn) {\n  check[key] = key;\n  return check;\n  function check(...args) {\n    return (oldFn[key] === key ? oldFn : newFn).apply(this, args);\n  }\n}\nfunction after(promise, cb) {\n  return promise.then(cb, cb);\n}\nfunction serialize(asyncFunction) {\n  let lastRun = Promise.resolve();\n  function wrapper(...args) {\n    return lastRun = new Promise((res, rej) => {\n      after(lastRun, () => {\n        asyncFunction.apply(this, args).then(res, rej);\n      });\n    });\n  }\n  wrapper.after = function() {\n    return lastRun = new Promise((res, rej) => {\n      after(lastRun, res);\n    });\n  };\n  return wrapper;\n}\n\nexport { after, around, dedupe, serialize };\n", "import { AbstractInputSuggest, App, TFolder } from 'obsidian';\n\nexport class FolderSuggest extends AbstractInputSuggest<TFolder> {\n\tprivate inputEl: HTMLInputElement;\n\n\tconstructor(app: App, inputEl: HTMLInputElement) {\n\t\tsuper(app, inputEl);\n\t\tthis.inputEl = inputEl;\n\t}\n\n\tgetSuggestions(query: string): TFolder[] {\n\t\tconst lowerQuery = query.toLowerCase();\n\t\tconst folders: TFolder[] = [];\n\n\t\tthis.app.vault.getAllLoadedFiles().forEach((file) => {\n\t\t\tif (file instanceof TFolder) {\n\t\t\t\tif (file.path.toLowerCase().includes(lowerQuery)) {\n\t\t\t\t\tfolders.push(file);\n\t\t\t\t}\n\t\t\t}\n\t\t});\n\n\t\tfolders.sort((a, b) => a.path.localeCompare(b.path));\n\t\treturn folders.slice(0, 20);\n\t}\n\n\trenderSuggestion(folder: TFolder, el: HTMLElement): void {\n\t\tel.createEl('div', { text: folder.path || '/' });\n\t}\n\n\tselectSuggestion(folder: TFolder): void {\n\t\tthis.inputEl.value = folder.path;\n\t\tthis.inputEl.trigger('input');\n\t\tthis.close();\n\t}\n}\n", "import { App, Modal, setIcon, Setting, TextComponent } from \"obsidian\";\nimport { ExplorerFocusPlugin } from \"../main\";\nimport { FolderSuggest } from \"./folder-suggest\";\n\nexport class AutoHideModal extends Modal {\n\tplugin: ExplorerFocusPlugin;\n\tprivate paths: string[];\n\tprivate onSave: () => void;\n\tprivate addInput: TextComponent | null = null;\n\n\tconstructor(app: App, plugin: ExplorerFocusPlugin, onSave: () => void) {\n\t\tsuper(app);\n\t\tthis.plugin = plugin;\n\t\tthis.paths = [...(plugin.settings.autoHidePaths ?? [])];\n\t\tthis.onSave = onSave;\n\t}\n\n\tonOpen(): void {\n\t\tthis.setTitle(\"Auto-hide folders\");\n\t\tthis.renderContent();\n\t}\n\n\tprivate renderContent(): void {\n\t\tconst { contentEl } = this;\n\t\tcontentEl.empty();\n\n\t\tcontentEl.createEl(\"p\", {\n\t\t\ttext: \"The following folders are currently hidden from the file explorer.\"\n\t\t});\n\n\t\t// Wrapper div for the list items (matches native structure)\n\t\tconst listWrapper = contentEl.createDiv();\n\n\t\t// List existing paths using native Obsidian classes\n\t\tthis.paths.forEach((path, index) => {\n\t\t\tconst row = listWrapper.createDiv({ cls: \"mobile-option-setting-item\" });\n\n\t\t\trow.createSpan({\n\t\t\t\ttext: path || \"(empty)\",\n\t\t\t\tcls: \"mobile-option-setting-item-name\"\n\t\t\t});\n\n\t\t\tconst removeBtn = row.createDiv({\n\t\t\t\tcls: \"clickable-icon mobile-option-setting-item-option-icon\",\n\t\t\t\tattr: { \"aria-label\": \"Delete\" }\n\t\t\t});\n\t\t\tsetIcon(removeBtn, \"x\");\n\t\t\tremoveBtn.addEventListener(\"click\", () => {\n\t\t\t\tthis.paths.splice(index, 1);\n\t\t\t\tthis.renderContent();\n\t\t\t});\n\t\t});\n\n\t\t// Add new folder input\n\t\tnew Setting(contentEl)\n\t\t\t.setName(\"Folder\")\n\t\t\t.addText(text => {\n\t\t\t\tnew FolderSuggest(this.app, text.inputEl);\n\t\t\t\ttext.setPlaceholder(\"Enter folder path\");\n\t\t\t\tthis.addInput = text;\n\t\t\t})\n\t\t\t.addButton(button => button\n\t\t\t\t.setButtonText(\"Add\")\n\t\t\t\t.onClick(() => {\n\t\t\t\t\tconst value = this.addInput?.getValue()?.trim();\n\t\t\t\t\tif (value) {\n\t\t\t\t\t\tthis.paths.push(value);\n\t\t\t\t\t\tthis.addInput?.setValue(\"\");\n\t\t\t\t\t\tthis.renderContent();\n\t\t\t\t\t}\n\t\t\t\t}));\n\n\t\t// Save / Cancel buttons using native modal-button-container\n\t\tthis.modalEl.querySelector('.modal-button-container')?.remove();\n\t\tconst buttonContainer = this.modalEl.createDiv({ cls: \"modal-button-container\" });\n\n\t\tconst saveBtn = buttonContainer.createEl(\"button\", {\n\t\t\ttext: \"Save\",\n\t\t\tcls: \"mod-cta\"\n\t\t});\n\t\tsaveBtn.addEventListener(\"click\", () => {\n\t\t\tthis.plugin.settings.autoHidePaths = this.paths;\n\t\t\tvoid this.plugin.saveSettings();\n\t\t\tthis.plugin.updateAutoHideStyles();\n\t\t\tthis.onSave();\n\t\t\tthis.close();\n\t\t});\n\n\t\tconst cancelBtn = buttonContainer.createEl(\"button\", {\n\t\t\ttext: \"Cancel\",\n\t\t\tcls: \"mod-cancel\"\n\t\t});\n\t\tcancelBtn.addEventListener(\"click\", () => {\n\t\t\tthis.close();\n\t\t});\n\t}\n}\n", "import { ExplorerFocusSettings } from '../types';\n\nexport function getFocusPath(\n\tpath: string,\n\tlevel: 'file' | 'parent' | 'grandparent' | 'greatgrandparent' | 'custom',\n\tsettings: ExplorerFocusSettings\n): string {\n\tif (level === 'custom') {\n\t\t// Use the custom folder path from settings, regardless of current file\n\t\treturn settings.customFolderPath || path;\n\t}\n\n\tif (level === 'file') {\n\t\treturn path;\n\t}\n\n\tconst parts = path.split('/');\n\t\n\tif (level === 'parent') {\n\t\tif (parts.length === 1) {\n\t\t\treturn path; // Already at root\n\t\t}\n\t\treturn parts.slice(0, -1).join('/');\n\t}\n\n\tif (level === 'grandparent') {\n\t\tif (parts.length <= 2) {\n\t\t\treturn parts[0] || path; // Return root or first part\n\t\t}\n\t\treturn parts.slice(0, -2).join('/');\n\t}\n\n\t// greatgrandparent\n\tif (parts.length <= 3) {\n\t\treturn parts[0] || path; // Return root or first part\n\t}\n\treturn parts.slice(0, -3).join('/');\n}\n\n", "import { ExplorerFocusPlugin } from \"../main\";\nimport { getFocusPath } from \"../utils/focus\";\n\nexport function registerCommands(plugin: ExplorerFocusPlugin): void {\n\t// Toggle focus command\n\tplugin.addCommand({\n\t\tid: \"toggle\",\n\t\tname: \"Toggle focus\",\n\t\tcallback: () => {\n\t\t\tif (plugin.isFocus) {\n\t\t\t\tplugin.exitFocus();\n\t\t\t} else {\n\t\t\t\tif (plugin.settings.focusLevel === 'custom') {\n\t\t\t\t\t// For custom folder, use the custom path directly\n\t\t\t\t\tif (plugin.settings.customFolderPath) {\n\t\t\t\t\t\tplugin.enterFocus(plugin.settings.customFolderPath);\n\t\t\t\t\t}\n\t\t\t\t} else {\n\t\t\t\t\t// For other levels, need the current file\n\t\t\t\t\tconst file = plugin.app.workspace.getActiveFile();\n\t\t\t\t\tif (file?.path) {\n\t\t\t\t\t\tconst focusPath = getFocusPath(file.path, plugin.settings.focusLevel, plugin.settings);\n\t\t\t\t\t\tplugin.enterFocus(focusPath);\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t}\n\t\t},\n\t});\n\n\t// File menu item\n\tif (plugin.settings.showRightClickMenu) {\n\t\tplugin.registerEvent(\n\t\t\tplugin.app.workspace.on(\"file-menu\", (menu, file) => {\n\t\t\t\tif (!plugin.settings.showRightClickMenu) {\n\t\t\t\t\treturn;\n\t\t\t\t}\n\n\t\t\t\tmenu.addItem((item) => {\n\t\t\t\t\titem\n\t\t\t\t\t\t.setTitle(plugin.isFocus ? \"Exit focus\" : \"Focus\")\n\t\t\t\t\t\t.setIcon(plugin.isFocus ? \"log-out\" : \"focus\")\n\t\t\t\t\t.onClick(() => {\n\t\t\t\t\t\tplugin.toggleFocus(file?.path);\n\t\t\t\t\t});\n\t\t\t\t});\n\t\t\t})\n\t\t);\n\t}\n}\n\n", "import { ExplorerFocusPlugin } from \"../main\";\n\n/**\n * Find or create the navigation buttons container in the file explorer view.\n * Tries multiple selectors to support both desktop and mobile.\n * On mobile, creates a container if none exists.\n */\nexport function findNavButtonsContainer(fileExplorerView: HTMLElement): HTMLElement | null {\n\t// Try desktop selector first\n\tlet container = fileExplorerView.querySelector('.nav-buttons-container');\n\tif (container) {\n\t\treturn container as HTMLElement;\n\t}\n\n\t// Try alternative selectors for mobile\n\t// Mobile might have the buttons in a different structure\n\tcontainer = fileExplorerView.querySelector('.nav-header-button-container');\n\tif (container) {\n\t\treturn container as HTMLElement;\n\t}\n\n\t// Try finding by looking for existing nav-action-button elements\n\tconst existingButtons = fileExplorerView.querySelectorAll('.nav-action-button');\n\tif (existingButtons.length > 0) {\n\t\tconst firstButton = existingButtons[0];\n\t\tconst parent = firstButton.parentElement;\n\t\tif (parent && (parent.classList.contains('nav-buttons-container') || \n\t\t               parent.classList.contains('nav-header-button-container') ||\n\t\t               Array.from(parent.children).some(el => el.classList.contains('nav-action-button')))) {\n\t\t\treturn parent;\n\t\t}\n\t}\n\n\t// Try finding the view header and appending to it\n\tconst viewHeader = fileExplorerView.querySelector('.view-header') as HTMLElement;\n\tif (viewHeader) {\n\t\t// Look for any container with nav buttons inside the header\n\t\tconst headerContainer = viewHeader.querySelector('.nav-buttons-container') as HTMLElement ||\n\t\t                       viewHeader.querySelector('.nav-header-button-container') as HTMLElement ||\n\t\t                       viewHeader.querySelector('.view-header-title-container') as HTMLElement;\n\t\tif (headerContainer) {\n\t\t\treturn headerContainer;\n\t\t}\n\t\t// If no container found, return the header itself as fallback\n\t\treturn viewHeader;\n\t}\n\n\t// Mobile fallback: Look for nav-files-container and create a button bar above it\n\tconst navFilesContainer = fileExplorerView.querySelector('.nav-files-container') as HTMLElement;\n\tif (navFilesContainer) {\n\t\t// Check if we already created a mobile button container\n\t\tlet mobileButtonContainer = fileExplorerView.querySelector('.explorer-focus-mobile-buttons') as HTMLElement;\n\t\tif (!mobileButtonContainer) {\n\t\t\t// Create a mobile button container\n\t\t\tmobileButtonContainer = document.createElement('div');\n\t\t\tmobileButtonContainer.className = 'nav-buttons-container explorer-focus-mobile-buttons';\n\t\t\tmobileButtonContainer.setCssProps({\n\t\t\t\tdisplay: 'flex',\n\t\t\t\talignItems: 'center',\n\t\t\t\tgap: '4px',\n\t\t\t\tpadding: '8px',\n\t\t\t\tborderBottom: '1px solid var(--background-modifier-border)'\n\t\t\t});\n\t\t\t\n\t\t\t// Insert it before the nav-files-container\n\t\t\tnavFilesContainer.parentElement?.insertBefore(mobileButtonContainer, navFilesContainer);\n\t\t}\n\t\treturn mobileButtonContainer;\n\t}\n\n\t// Last resort: create a container at the top of the view\n\tlet topContainer = fileExplorerView.querySelector('.explorer-focus-mobile-buttons') as HTMLElement;\n\tif (!topContainer) {\n\t\ttopContainer = document.createElement('div');\n\t\ttopContainer.className = 'nav-buttons-container explorer-focus-mobile-buttons';\n\t\ttopContainer.setCssProps({\n\t\t\tdisplay: 'flex',\n\t\t\talignItems: 'center',\n\t\t\tgap: '4px',\n\t\t\tpadding: '8px',\n\t\t\tborderBottom: '1px solid var(--background-modifier-border)'\n\t\t});\n\t\t\n\t\t// Insert at the beginning of the view\n\t\tfileExplorerView.insertBefore(topContainer, fileExplorerView.firstChild);\n\t}\n\treturn topContainer;\n}\n\nexport function createFileExplorerIcon(plugin: ExplorerFocusPlugin): HTMLElement {\n\tconst icon = document.createElement('div');\n\ticon.className = 'clickable-icon nav-action-button';\n\ticon.setAttribute('aria-label', \"Toggle focus\");\n\n\t// Create the focus icon SVG\n\tconst svg = document.createElementNS('http://www.w3.org/2000/svg', 'svg');\n\tsvg.setAttribute('xmlns', 'http://www.w3.org/2000/svg');\n\tsvg.setAttribute('width', '24');\n\tsvg.setAttribute('height', '24');\n\tsvg.setAttribute('viewBox', '0 0 24 24');\n\tsvg.setAttribute('fill', 'none');\n\tsvg.setAttribute('stroke', 'currentColor');\n\tsvg.setAttribute('stroke-width', '2');\n\tsvg.setAttribute('stroke-linecap', 'round');\n\tsvg.setAttribute('stroke-linejoin', 'round');\n\tsvg.setAttribute('class', 'svg-icon lucide-focus');\n\t\n\t// Lucide focus icon - circle with corner brackets\n\tconst circle = document.createElementNS('http://www.w3.org/2000/svg', 'circle');\n\tcircle.setAttribute('cx', '12');\n\tcircle.setAttribute('cy', '12');\n\tcircle.setAttribute('r', '3');\n\tsvg.appendChild(circle);\n\t\n\tconst path1 = document.createElementNS('http://www.w3.org/2000/svg', 'path');\n\tpath1.setAttribute('d', 'M3 7V5a2 2 0 0 1 2-2h2');\n\tsvg.appendChild(path1);\n\t\n\tconst path2 = document.createElementNS('http://www.w3.org/2000/svg', 'path');\n\tpath2.setAttribute('d', 'M17 3h2a2 2 0 0 1 2 2v2');\n\tsvg.appendChild(path2);\n\t\n\tconst path3 = document.createElementNS('http://www.w3.org/2000/svg', 'path');\n\tpath3.setAttribute('d', 'M21 17v2a2 2 0 0 1-2 2h-2');\n\tsvg.appendChild(path3);\n\t\n\tconst path4 = document.createElementNS('http://www.w3.org/2000/svg', 'path');\n\tpath4.setAttribute('d', 'M7 21H5a2 2 0 0 1-2-2v-2');\n\tsvg.appendChild(path4);\n\t\n\ticon.appendChild(svg);\n\treturn icon;\n}\n\nexport function insertFileExplorerIcon(icon: HTMLElement, navButtonsContainer: HTMLElement): void {\n\t// Don't insert if already in the container\n\tif (navButtonsContainer.contains(icon)) {\n\t\treturn;\n\t}\n\n\t// Find the position to insert: after all default Obsidian icons, before any .cmdr elements\n\tconst allIcons = Array.from(navButtonsContainer.querySelectorAll('.clickable-icon.nav-action-button'));\n\tconst defaultIcons = allIcons.filter(el => !el.classList.contains('cmdr') && el !== icon);\n\tconst cmdrIcons = Array.from(navButtonsContainer.querySelectorAll('.cmdr'));\n\t\n\tif (cmdrIcons.length > 0) {\n\t\t// Insert before first cmdr icon\n\t\tnavButtonsContainer.insertBefore(icon, cmdrIcons[0]);\n\t} else if (defaultIcons.length > 0) {\n\t\t// Insert after last default icon\n\t\tconst lastIcon = defaultIcons[defaultIcons.length - 1];\n\t\tif (lastIcon.nextSibling) {\n\t\t\tnavButtonsContainer.insertBefore(icon, lastIcon.nextSibling);\n\t\t} else {\n\t\t\tnavButtonsContainer.appendChild(icon);\n\t\t}\n\t} else {\n\t\t// Just append if no other icons\n\t\tnavButtonsContainer.appendChild(icon);\n\t}\n}\n\n"],
  "mappings": ";;;;;;;;;;;;;;;;;;;;;;;;;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,IAAAA,mBAA4C;;;ACSrC,IAAM,mBAA0C;AAAA,EACtD,oBAAoB;AAAA,EACpB,sBAAsB;AAAA,EACtB,YAAY;AAAA,EACZ,kBAAkB;AAAA,EAClB,qBAAqB;AAAA,EACrB,eAAe,CAAC;AACjB;;;AChBA,IAAAC,mBAAoD;;;ACApD,sBAAsD;;;ACAtD,SAAS,OAAO,KAAK,WAAW;AAC9B,QAAM,WAAW,OAAO,KAAK,SAAS,EAAE,IAAI,CAAC,QAAQ,QAAQ,KAAK,KAAK,UAAU,GAAG,CAAC,CAAC;AACtF,SAAO,SAAS,WAAW,IAAI,SAAS,CAAC,IAAI,WAAW;AACtD,aAAS,QAAQ,CAAC,MAAM,EAAE,CAAC;AAAA,EAC7B;AACF;AACA,SAAS,QAAQ,KAAK,QAAQ,eAAe;AAC3C,QAAM,YAAY,IAAI,MAAM,GAAG,SAAS,IAAI,eAAe,MAAM,GAAG,WAAW,SAAS,YAAY,WAAW;AAC7G,WAAO,OAAO,eAAe,GAAG,EAAE,MAAM,EAAE,MAAM,MAAM,SAAS;AAAA,EACjE;AACA,MAAI,UAAU,cAAc,QAAQ;AACpC,MAAI;AACF,WAAO,eAAe,SAAS,SAAS;AAC1C,SAAO,eAAe,SAAS,OAAO;AACtC,MAAI,MAAM,IAAI;AACd,SAAO;AACP,WAAS,WAAW,MAAM;AACxB,QAAI,YAAY,YAAY,IAAI,MAAM,MAAM;AAC1C,aAAO;AACT,WAAO,QAAQ,MAAM,MAAM,IAAI;AAAA,EACjC;AACA,WAAS,SAAS;AAChB,QAAI,IAAI,MAAM,MAAM,SAAS;AAC3B,UAAI;AACF,YAAI,MAAM,IAAI;AAAA;AAEd,eAAO,IAAI,MAAM;AAAA,IACrB;AACA,QAAI,YAAY;AACd;AACF,cAAU;AACV,WAAO,eAAe,SAAS,aAAa,QAAQ;AAAA,EACtD;AACF;;;ADcA,IAAI,mBAAmB;AACvB,IAAI,iBAAiB;AAMd,SAAS,kBAAkB,QAAmC;AAEpE,MAAI,gBAAgB;AACnB;AAAA,EACD;AAIA,MAAI,yBAAS,UAAU;AACtB,qBAAiB;AACjB;AAAA,EACD;AAEA,QAAM,eAAe,gBAAgB,MAAM;AAE3C,MAAI,CAAC,cAAc;AAClB;AAAA,EACD;AAGA,MAAI,CAAC,kBAAkB;AACtB,QAAI;AACH,YAAM,YAAY,OAAO,eAAe,YAAY;AAGpD,UAAI,OAAO,UAAU,yBAAyB,YAAY;AACzD,cAAM,IAAI,MAAM,kEAAkE;AAAA,MACnF;AAEA,aAAO;AAAA,QACN,OAAO,WAAW;AAAA,UACjB,qBAAqB,KAAmC;AACvD,mBAAO,SAAkC,QAA6C;AACrF,kBAAI,iBAAuC,IAAI,KAAK,MAAM,MAAM;AAGhE,kBAAI,OAAO,WAAW,OAAO,aAAa;AACzC,sBAAM,cAAc,OAAO;AAE3B,iCAAiB,eAAe,OAAO,CAAC,QAAQ;AAC/C,wBAAM,WAAW,IAAI,KAAK;AAG1B,sBAAI,aAAa,aAAa;AAC7B,wBAAI,KAAK,SAAS;AAClB,2BAAO;AAAA,kBACR;AAGA,sBAAI,SAAS,WAAW,cAAc,GAAG,GAAG;AAC3C,wBAAI,KAAK,SAAS;AAClB,2BAAO;AAAA,kBACR;AAGA,sBAAI,YAAY,WAAW,WAAW,GAAG,GAAG;AAC3C,wBAAI,KAAK,SAAS;AAClB,2BAAO;AAAA,kBACR;AAGA,sBAAI,KAAK,SAAS;AAClB,yBAAO;AAAA,gBACR,CAAC;AAAA,cACF,OAAO;AAEN,+BAAe,QAAQ,CAAC,QAAQ;AAC/B,sBAAI,KAAK,SAAS;AAAA,gBACnB,CAAC;AAAA,cACF;AAEA,qBAAO;AAAA,YACR;AAAA,UACD;AAAA,QACD,CAAC;AAAA,MACF;AAEA,yBAAmB;AAAA,IACpB,SAAS,OAAO;AACf,uBAAiB;AACjB,cAAQ;AAAA,QACP;AAAA,QAGA;AAAA,MACD;AACA,UAAI;AAAA,QACH;AAAA,QAEA;AAAA,MACD;AACA;AAAA,IACD;AAAA,EACD;AAGA,eAAa,0BAA0B;AACxC;AAEO,SAAS,gBAAgB,QAA2D;AAzJ3F;AA0JC,QAAM,yBAAwB,YAAO,IAAI,UAAU,gBAAgB,eAAe,MAApD,mBAAuD;AACrF,SAAO,+DAAuB;AAC/B;AAEO,SAAS,oBAAoB,QAAiD;AACpF,QAAM,qBAAqB,OAAO,IAAI,UAAU,gBAAgB,eAAe;AAC/E,SAAO,mBAAmB,IAAI,UAAQ,KAAK,IAAwB;AACpE;;;AEjKA,IAAAC,mBAAmD;AAE5C,IAAM,gBAAN,cAA4B,sCAA8B;AAAA,EAGhE,YAAY,KAAU,SAA2B;AAChD,UAAM,KAAK,OAAO;AAClB,SAAK,UAAU;AAAA,EAChB;AAAA,EAEA,eAAe,OAA0B;AACxC,UAAM,aAAa,MAAM,YAAY;AACrC,UAAM,UAAqB,CAAC;AAE5B,SAAK,IAAI,MAAM,kBAAkB,EAAE,QAAQ,CAAC,SAAS;AACpD,UAAI,gBAAgB,0BAAS;AAC5B,YAAI,KAAK,KAAK,YAAY,EAAE,SAAS,UAAU,GAAG;AACjD,kBAAQ,KAAK,IAAI;AAAA,QAClB;AAAA,MACD;AAAA,IACD,CAAC;AAED,YAAQ,KAAK,CAAC,GAAG,MAAM,EAAE,KAAK,cAAc,EAAE,IAAI,CAAC;AACnD,WAAO,QAAQ,MAAM,GAAG,EAAE;AAAA,EAC3B;AAAA,EAEA,iBAAiB,QAAiB,IAAuB;AACxD,OAAG,SAAS,OAAO,EAAE,MAAM,OAAO,QAAQ,IAAI,CAAC;AAAA,EAChD;AAAA,EAEA,iBAAiB,QAAuB;AACvC,SAAK,QAAQ,QAAQ,OAAO;AAC5B,SAAK,QAAQ,QAAQ,OAAO;AAC5B,SAAK,MAAM;AAAA,EACZ;AACD;;;ACnCA,IAAAC,mBAA4D;AAIrD,IAAM,gBAAN,cAA4B,uBAAM;AAAA,EAMxC,YAAY,KAAU,QAA6B,QAAoB;AAVxE;AAWE,UAAM,GAAG;AAHV,SAAQ,WAAiC;AAIxC,SAAK,SAAS;AACd,SAAK,QAAQ,CAAC,IAAI,YAAO,SAAS,kBAAhB,YAAiC,CAAC,CAAE;AACtD,SAAK,SAAS;AAAA,EACf;AAAA,EAEA,SAAe;AACd,SAAK,SAAS,mBAAmB;AACjC,SAAK,cAAc;AAAA,EACpB;AAAA,EAEQ,gBAAsB;AAtB/B;AAuBE,UAAM,EAAE,UAAU,IAAI;AACtB,cAAU,MAAM;AAEhB,cAAU,SAAS,KAAK;AAAA,MACvB,MAAM;AAAA,IACP,CAAC;AAGD,UAAM,cAAc,UAAU,UAAU;AAGxC,SAAK,MAAM,QAAQ,CAAC,MAAM,UAAU;AACnC,YAAM,MAAM,YAAY,UAAU,EAAE,KAAK,6BAA6B,CAAC;AAEvE,UAAI,WAAW;AAAA,QACd,MAAM,QAAQ;AAAA,QACd,KAAK;AAAA,MACN,CAAC;AAED,YAAM,YAAY,IAAI,UAAU;AAAA,QAC/B,KAAK;AAAA,QACL,MAAM,EAAE,cAAc,SAAS;AAAA,MAChC,CAAC;AACD,oCAAQ,WAAW,GAAG;AACtB,gBAAU,iBAAiB,SAAS,MAAM;AACzC,aAAK,MAAM,OAAO,OAAO,CAAC;AAC1B,aAAK,cAAc;AAAA,MACpB,CAAC;AAAA,IACF,CAAC;AAGD,QAAI,yBAAQ,SAAS,EACnB,QAAQ,QAAQ,EAChB,QAAQ,UAAQ;AAChB,UAAI,cAAc,KAAK,KAAK,KAAK,OAAO;AACxC,WAAK,eAAe,mBAAmB;AACvC,WAAK,WAAW;AAAA,IACjB,CAAC,EACA,UAAU,YAAU,OACnB,cAAc,KAAK,EACnB,QAAQ,MAAM;AA/DnB,UAAAC,KAAA;AAgEK,YAAM,SAAQ,MAAAA,MAAA,KAAK,aAAL,gBAAAA,IAAe,eAAf,mBAA2B;AACzC,UAAI,OAAO;AACV,aAAK,MAAM,KAAK,KAAK;AACrB,mBAAK,aAAL,mBAAe,SAAS;AACxB,aAAK,cAAc;AAAA,MACpB;AAAA,IACD,CAAC,CAAC;AAGJ,eAAK,QAAQ,cAAc,yBAAyB,MAApD,mBAAuD;AACvD,UAAM,kBAAkB,KAAK,QAAQ,UAAU,EAAE,KAAK,yBAAyB,CAAC;AAEhF,UAAM,UAAU,gBAAgB,SAAS,UAAU;AAAA,MAClD,MAAM;AAAA,MACN,KAAK;AAAA,IACN,CAAC;AACD,YAAQ,iBAAiB,SAAS,MAAM;AACvC,WAAK,OAAO,SAAS,gBAAgB,KAAK;AAC1C,WAAK,KAAK,OAAO,aAAa;AAC9B,WAAK,OAAO,qBAAqB;AACjC,WAAK,OAAO;AACZ,WAAK,MAAM;AAAA,IACZ,CAAC;AAED,UAAM,YAAY,gBAAgB,SAAS,UAAU;AAAA,MACpD,MAAM;AAAA,MACN,KAAK;AAAA,IACN,CAAC;AACD,cAAU,iBAAiB,SAAS,MAAM;AACzC,WAAK,MAAM;AAAA,IACZ,CAAC;AAAA,EACF;AACD;;;AJzFO,IAAM,0BAAN,cAAsC,kCAAiB;AAAA,EAI7D,YAAY,KAAU,QAA6B;AAClD,UAAM,KAAK,MAAM;AAHlB,SAAO,OAAO;AAIb,SAAK,SAAS;AAAA,EACf;AAAA,EAEA,UAAgB;AACf,UAAM,EAAE,YAAY,IAAI;AAExB,gBAAY,MAAM;AAGlB,UAAM,eAAe,IAAI,8BAAa,WAAW;AAEjD,iBAAa,WAAW,aAAW;AAClC,cACE,QAAQ,8BAA8B,EACtC,UAAU,YAAU,OACnB,SAAS,KAAK,OAAO,SAAS,kBAAkB,EAChD,SAAS,OAAM,UAAS;AACxB,aAAK,OAAO,SAAS,qBAAqB;AAC1C,cAAM,KAAK,OAAO,aAAa;AAAA,MAChC,CAAC,CAAC;AAAA,IACL,CAAC;AAED,iBAAa,WAAW,aAAW;AAClC,cACE,QAAQ,yBAAyB,EACjC,UAAU,YAAU,OACnB,SAAS,KAAK,OAAO,SAAS,oBAAoB,EAClD,SAAS,OAAM,UAAS;AACxB,aAAK,OAAO,SAAS,uBAAuB;AAC5C,cAAM,KAAK,OAAO,aAAa;AAC/B,aAAK,OAAO,uBAAuB;AAAA,MACpC,CAAC,CAAC;AAAA,IACL,CAAC;AAED,iBAAa,WAAW,aAAW;AAClC,cACE,QAAQ,qBAAqB,EAC7B,QAAQ,wIAAwI,EAChJ,YAAY,cAAY,SACvB,UAAU,QAAQ,mBAAmB,EACrC,UAAU,UAAU,eAAe,EACnC,UAAU,eAAe,oBAAoB,EAC7C,UAAU,oBAAoB,0BAA0B,EACxD,UAAU,UAAU,iBAAiB,EACrC,SAAS,KAAK,OAAO,SAAS,UAAU,EACxC,SAAS,OAAO,UAAkB;AAClC,aAAK,OAAO,SAAS,aAAa;AAClC,cAAM,KAAK,OAAO,aAAa;AAE/B,YAAI,KAAK,OAAO,SAAS;AACxB,gBAAM,gBAAgB,oBAAoB,KAAK,MAAM;AACrD,wBAAc,QAAQ,CAAC,iBAAiB;AACvC,gBAAI,6CAAc,aAAa;AAC9B,2BAAa,YAAY;AAAA,YAC1B;AAAA,UACD,CAAC;AAAA,QACF;AACA,aAAK,QAAQ;AAAA,MACd,CAAC,CAAC;AAAA,IACL,CAAC;AAGD,QAAI,KAAK,OAAO,SAAS,eAAe,UAAU;AACjD,mBAAa,WAAW,aAAW;AAClC,gBACE,QAAQ,oBAAoB,EAC5B,QAAQ,uGAAuG,EAC/G,QAAQ,UAAQ;AAChB,cAAI,cAAc,KAAK,KAAK,KAAK,OAAO;AACxC,eACE,eAAe,kBAAkB,EACjC,SAAS,KAAK,OAAO,SAAS,gBAAgB,EAC9C,SAAS,OAAM,UAAS;AACxB,iBAAK,OAAO,SAAS,mBAAmB;AACxC,kBAAM,KAAK,OAAO,aAAa;AAAA,UAChC,CAAC;AAAA,QACH,CAAC;AAAA,MACH,CAAC;AAAA,IACF;AAGA,UAAM,gBAAgB,IAAI,8BAAa,WAAW,EAChD,WAAW,mBAAmB;AAEhC,kBAAc,WAAW,aAAW;AAjGtC;AAkGG,YAAM,kBAAiB,UAAK,OAAO,SAAS,kBAArB,YAAsC,CAAC,GAAG,OAAO,OAAK,EAAE,KAAK,EAAE,SAAS,CAAC;AAEhG,YAAM,eAAe,SAAS,uBAAuB;AACrD,mBAAa,WAAW,yDAAyD;AAEjF,UAAI,cAAc,SAAS,GAAG;AAC7B,cAAM,OAAO,aAAa,SAAS,IAAI;AACvC,sBAAc,QAAQ,OAAK;AAC1B,eAAK,SAAS,MAAM,EAAE,MAAM,EAAE,CAAC;AAAA,QAChC,CAAC;AAAA,MACF;AAEA,cACE,QAAQ,gBAAgB,EACxB,QAAQ,YAAY,EACpB,UAAU,YAAU,OACnB,cAAc,QAAQ,EACtB,QAAQ,MAAM;AACd,YAAI,cAAc,KAAK,KAAK,KAAK,QAAQ,MAAM;AAC9C,eAAK,QAAQ;AAAA,QACd,CAAC,EAAE,KAAK;AAAA,MACT,CAAC,CAAC;AAAA,IACL,CAAC;AAAA,EAEF;AACD;;;AKzHO,SAAS,aACf,MACA,OACA,UACS;AACT,MAAI,UAAU,UAAU;AAEvB,WAAO,SAAS,oBAAoB;AAAA,EACrC;AAEA,MAAI,UAAU,QAAQ;AACrB,WAAO;AAAA,EACR;AAEA,QAAM,QAAQ,KAAK,MAAM,GAAG;AAE5B,MAAI,UAAU,UAAU;AACvB,QAAI,MAAM,WAAW,GAAG;AACvB,aAAO;AAAA,IACR;AACA,WAAO,MAAM,MAAM,GAAG,EAAE,EAAE,KAAK,GAAG;AAAA,EACnC;AAEA,MAAI,UAAU,eAAe;AAC5B,QAAI,MAAM,UAAU,GAAG;AACtB,aAAO,MAAM,CAAC,KAAK;AAAA,IACpB;AACA,WAAO,MAAM,MAAM,GAAG,EAAE,EAAE,KAAK,GAAG;AAAA,EACnC;AAGA,MAAI,MAAM,UAAU,GAAG;AACtB,WAAO,MAAM,CAAC,KAAK;AAAA,EACpB;AACA,SAAO,MAAM,MAAM,GAAG,EAAE,EAAE,KAAK,GAAG;AACnC;;;AClCO,SAAS,iBAAiB,QAAmC;AAEnE,SAAO,WAAW;AAAA,IACjB,IAAI;AAAA,IACJ,MAAM;AAAA,IACN,UAAU,MAAM;AACf,UAAI,OAAO,SAAS;AACnB,eAAO,UAAU;AAAA,MAClB,OAAO;AACN,YAAI,OAAO,SAAS,eAAe,UAAU;AAE5C,cAAI,OAAO,SAAS,kBAAkB;AACrC,mBAAO,WAAW,OAAO,SAAS,gBAAgB;AAAA,UACnD;AAAA,QACD,OAAO;AAEN,gBAAM,OAAO,OAAO,IAAI,UAAU,cAAc;AAChD,cAAI,6BAAM,MAAM;AACf,kBAAM,YAAY,aAAa,KAAK,MAAM,OAAO,SAAS,YAAY,OAAO,QAAQ;AACrF,mBAAO,WAAW,SAAS;AAAA,UAC5B;AAAA,QACD;AAAA,MACD;AAAA,IACD;AAAA,EACD,CAAC;AAGD,MAAI,OAAO,SAAS,oBAAoB;AACvC,WAAO;AAAA,MACN,OAAO,IAAI,UAAU,GAAG,aAAa,CAAC,MAAM,SAAS;AACpD,YAAI,CAAC,OAAO,SAAS,oBAAoB;AACxC;AAAA,QACD;AAEA,aAAK,QAAQ,CAAC,SAAS;AACtB,eACE,SAAS,OAAO,UAAU,eAAe,OAAO,EAChD,QAAQ,OAAO,UAAU,YAAY,OAAO,EAC7C,QAAQ,MAAM;AACd,mBAAO,YAAY,6BAAM,IAAI;AAAA,UAC9B,CAAC;AAAA,QACF,CAAC;AAAA,MACF,CAAC;AAAA,IACF;AAAA,EACD;AACD;;;ACzCO,SAAS,wBAAwB,kBAAmD;AAP3F;AASC,MAAI,YAAY,iBAAiB,cAAc,wBAAwB;AACvE,MAAI,WAAW;AACd,WAAO;AAAA,EACR;AAIA,cAAY,iBAAiB,cAAc,8BAA8B;AACzE,MAAI,WAAW;AACd,WAAO;AAAA,EACR;AAGA,QAAM,kBAAkB,iBAAiB,iBAAiB,oBAAoB;AAC9E,MAAI,gBAAgB,SAAS,GAAG;AAC/B,UAAM,cAAc,gBAAgB,CAAC;AACrC,UAAM,SAAS,YAAY;AAC3B,QAAI,WAAW,OAAO,UAAU,SAAS,uBAAuB,KACjD,OAAO,UAAU,SAAS,6BAA6B,KACvD,MAAM,KAAK,OAAO,QAAQ,EAAE,KAAK,QAAM,GAAG,UAAU,SAAS,mBAAmB,CAAC,IAAI;AACnG,aAAO;AAAA,IACR;AAAA,EACD;AAGA,QAAM,aAAa,iBAAiB,cAAc,cAAc;AAChE,MAAI,YAAY;AAEf,UAAM,kBAAkB,WAAW,cAAc,wBAAwB,KAClD,WAAW,cAAc,8BAA8B,KACvD,WAAW,cAAc,8BAA8B;AAC9E,QAAI,iBAAiB;AACpB,aAAO;AAAA,IACR;AAEA,WAAO;AAAA,EACR;AAGA,QAAM,oBAAoB,iBAAiB,cAAc,sBAAsB;AAC/E,MAAI,mBAAmB;AAEtB,QAAI,wBAAwB,iBAAiB,cAAc,gCAAgC;AAC3F,QAAI,CAAC,uBAAuB;AAE3B,8BAAwB,SAAS,cAAc,KAAK;AACpD,4BAAsB,YAAY;AAClC,4BAAsB,YAAY;AAAA,QACjC,SAAS;AAAA,QACT,YAAY;AAAA,QACZ,KAAK;AAAA,QACL,SAAS;AAAA,QACT,cAAc;AAAA,MACf,CAAC;AAGD,8BAAkB,kBAAlB,mBAAiC,aAAa,uBAAuB;AAAA,IACtE;AACA,WAAO;AAAA,EACR;AAGA,MAAI,eAAe,iBAAiB,cAAc,gCAAgC;AAClF,MAAI,CAAC,cAAc;AAClB,mBAAe,SAAS,cAAc,KAAK;AAC3C,iBAAa,YAAY;AACzB,iBAAa,YAAY;AAAA,MACxB,SAAS;AAAA,MACT,YAAY;AAAA,MACZ,KAAK;AAAA,MACL,SAAS;AAAA,MACT,cAAc;AAAA,IACf,CAAC;AAGD,qBAAiB,aAAa,cAAc,iBAAiB,UAAU;AAAA,EACxE;AACA,SAAO;AACR;AAEO,SAAS,uBAAuB,QAA0C;AAChF,QAAM,OAAO,SAAS,cAAc,KAAK;AACzC,OAAK,YAAY;AACjB,OAAK,aAAa,cAAc,cAAc;AAG9C,QAAM,MAAM,SAAS,gBAAgB,8BAA8B,KAAK;AACxE,MAAI,aAAa,SAAS,4BAA4B;AACtD,MAAI,aAAa,SAAS,IAAI;AAC9B,MAAI,aAAa,UAAU,IAAI;AAC/B,MAAI,aAAa,WAAW,WAAW;AACvC,MAAI,aAAa,QAAQ,MAAM;AAC/B,MAAI,aAAa,UAAU,cAAc;AACzC,MAAI,aAAa,gBAAgB,GAAG;AACpC,MAAI,aAAa,kBAAkB,OAAO;AAC1C,MAAI,aAAa,mBAAmB,OAAO;AAC3C,MAAI,aAAa,SAAS,uBAAuB;AAGjD,QAAM,SAAS,SAAS,gBAAgB,8BAA8B,QAAQ;AAC9E,SAAO,aAAa,MAAM,IAAI;AAC9B,SAAO,aAAa,MAAM,IAAI;AAC9B,SAAO,aAAa,KAAK,GAAG;AAC5B,MAAI,YAAY,MAAM;AAEtB,QAAM,QAAQ,SAAS,gBAAgB,8BAA8B,MAAM;AAC3E,QAAM,aAAa,KAAK,wBAAwB;AAChD,MAAI,YAAY,KAAK;AAErB,QAAM,QAAQ,SAAS,gBAAgB,8BAA8B,MAAM;AAC3E,QAAM,aAAa,KAAK,yBAAyB;AACjD,MAAI,YAAY,KAAK;AAErB,QAAM,QAAQ,SAAS,gBAAgB,8BAA8B,MAAM;AAC3E,QAAM,aAAa,KAAK,2BAA2B;AACnD,MAAI,YAAY,KAAK;AAErB,QAAM,QAAQ,SAAS,gBAAgB,8BAA8B,MAAM;AAC3E,QAAM,aAAa,KAAK,0BAA0B;AAClD,MAAI,YAAY,KAAK;AAErB,OAAK,YAAY,GAAG;AACpB,SAAO;AACR;AAEO,SAAS,uBAAuB,MAAmB,qBAAwC;AAEjG,MAAI,oBAAoB,SAAS,IAAI,GAAG;AACvC;AAAA,EACD;AAGA,QAAM,WAAW,MAAM,KAAK,oBAAoB,iBAAiB,mCAAmC,CAAC;AACrG,QAAM,eAAe,SAAS,OAAO,QAAM,CAAC,GAAG,UAAU,SAAS,MAAM,KAAK,OAAO,IAAI;AACxF,QAAM,YAAY,MAAM,KAAK,oBAAoB,iBAAiB,OAAO,CAAC;AAE1E,MAAI,UAAU,SAAS,GAAG;AAEzB,wBAAoB,aAAa,MAAM,UAAU,CAAC,CAAC;AAAA,EACpD,WAAW,aAAa,SAAS,GAAG;AAEnC,UAAM,WAAW,aAAa,aAAa,SAAS,CAAC;AACrD,QAAI,SAAS,aAAa;AACzB,0BAAoB,aAAa,MAAM,SAAS,WAAW;AAAA,IAC5D,OAAO;AACN,0BAAoB,YAAY,IAAI;AAAA,IACrC;AAAA,EACD,OAAO;AAEN,wBAAoB,YAAY,IAAI;AAAA,EACrC;AACD;;;ATxJO,IAAM,sBAAN,cAAkC,wBAAO;AAAA,EAM/C,YAAY,KAAU,UAA0B;AAC/C,UAAM,KAAK,QAAQ;AACnB,SAAK,UAAU;AACf,SAAK,cAAc;AACnB,SAAK,mBAAmB;AAAA,EACzB;AAAA,EAEA,MAAM,SAAS;AACd,UAAM,KAAK,aAAa;AAExB,SAAK,cAAc,IAAI,wBAAwB,KAAK,KAAK,IAAI,CAAC;AAE9D,qBAAiB,IAAI;AAErB,QAAI,KAAK,SAAS,sBAAsB;AACvC,WAAK,oBAAoB;AAAA,IAC1B;AAEA,SAAK,IAAI,UAAU,cAAc,MAAM;AACtC,WAAK,sBAAsB;AAC3B,WAAK,uBAAuB;AAC5B,WAAK,qBAAqB;AAAA,IAC3B,CAAC;AAED,SAAK,IAAI,UAAU,GAAG,iBAAiB,MAAM;AAC5C,WAAK,sBAAsB;AAC3B,WAAK,uBAAuB;AAC5B,WAAK,qBAAqB;AAAA,IAC3B,CAAC;AAAA,EACF;AAAA,EAEA,MAAM,eAAe;AACpB,UAAM,aAAa,MAAM,KAAK,SAAS;AACvC,SAAK,WAAW,OAAO,OAAO,CAAC,GAAG,kBAAkB,UAAU;AAAA,EAC/D;AAAA,EAEA,MAAM,eAAe;AACpB,UAAM,KAAK,SAAS,KAAK,QAAQ;AAAA,EAClC;AAAA,EAEA,YAAY,MAA0B;AACrC,QAAI,KAAK,SAAS;AACjB,WAAK,UAAU;AAAA,IAChB,WAAW,MAAM;AAChB,WAAK,WAAW,IAAI;AAAA,IACrB;AAAA,EACD;AAAA,EAEA,WAAW,MAAc;AACxB,SAAK,UAAU;AACf,SAAK,cAAc;AAGnB,SAAK,uBAAuB;AAG5B,SAAK,uBAAuB;AAG5B,UAAM,gBAAgB,oBAAoB,IAAI;AAC9C,kBAAc,QAAQ,kBAAgB;AACrC,UAAI,6CAAc,aAAa;AAC9B,qBAAa,YAAY;AAAA,MAC1B;AAAA,IACD,CAAC;AAID,eAAW,MAAM;AAChB,oBAAc,QAAQ,kBAAgB;AACrC,aAAK,8BAA8B,YAAY;AAAA,MAChD,CAAC;AAAA,IACF,GAAG,CAAC;AAAA,EACL;AAAA,EAEA,wBAA8B;AAE7B,sBAAsB,IAAI;AAC1B,UAAM,gBAAgB,oBAAoB,IAAI;AAC9C,kBAAc,QAAQ,kBAAgB;AACrC,mBAAa,0BAA0B;AAAA,IACxC,CAAC;AAAA,EACF;AAAA,EAEA,YAAY;AACX,SAAK,UAAU;AACf,SAAK,cAAc;AAGnB,SAAK,uBAAuB;AAG5B,SAAK,uBAAuB;AAG5B,UAAM,gBAAgB,oBAAoB,IAAI;AAC9C,kBAAc,QAAQ,kBAAgB;AACrC,UAAI,6CAAc,aAAa;AAC9B,qBAAa,YAAY;AAAA,MAC1B;AAAA,IACD,CAAC;AAID,eAAW,MAAM;AAChB,oBAAc,QAAQ,kBAAgB;AACrC,aAAK,8BAA8B,YAAY;AAAA,MAChD,CAAC;AAED,WAAK,qBAAqB;AAAA,IAC3B,GAAG,CAAC;AAAA,EACL;AAAA,EAEA,8BAA8B,cAAsC;AACnE,QAAI,EAAC,6CAAc,YAAW;AAC7B;AAAA,IACD;AAIA,QAAI,CAAC,KAAK,WAAW,CAAC,KAAK,aAAa;AACvC,aAAO,OAAO,aAAa,SAAS,EAAE,QAAQ,CAAC,QAAQ;AACtD,YAAI,EAAC,2BAAK,IAAI;AAEd,YAAI,IAAI,GAAG,aAAa,4BAA4B,GAAG;AACtD,cAAI,IAAI,KAAM,KAAI,KAAK,SAAS;AAChC,cAAI,GAAG,YAAY,EAAE,SAAS,GAAG,CAAC;AAClC,cAAI,GAAG,gBAAgB,4BAA4B;AAAA,QACpD;AAAA,MACD,CAAC;AACD;AAAA,IACD;AAEA,UAAM,cAAc,KAAK;AAGzB,UAAM,gBAAgB,oBAAI,IAAY;AACtC,UAAM,YAAY,YAAY,MAAM,GAAG;AACvC,aAAS,IAAI,GAAG,IAAI,UAAU,QAAQ,KAAK;AAC1C,oBAAc,IAAI,UAAU,MAAM,GAAG,CAAC,EAAE,KAAK,GAAG,CAAC;AAAA,IAClD;AAGA,UAAM,kBAAkB,oBAAI,IAAsB;AAClD,UAAM,YAAsB,CAAC;AAE7B,eAAW,QAAQ,OAAO,KAAK,aAAa,SAAS,GAAG;AACvD,YAAM,aAAa,KAAK,QAAQ,GAAG;AACnC,UAAI,eAAe,IAAI;AACtB,kBAAU,KAAK,IAAI;AAAA,MACpB,OAAO;AACN,cAAM,WAAW,KAAK,UAAU,GAAG,UAAU;AAC7C,YAAI,CAAC,gBAAgB,IAAI,QAAQ,GAAG;AACnC,0BAAgB,IAAI,UAAU,CAAC,CAAC;AAAA,QACjC;AACA,wBAAgB,IAAI,QAAQ,EAAG,KAAK,IAAI;AAAA,MACzC;AAAA,IACD;AAGA,UAAM,kBAAkB,YAAY,MAAM,GAAG,EAAE,CAAC;AAGhD,eAAW,QAAQ,WAAW;AAC7B,YAAM,MAAM,aAAa,UAAU,IAAI;AACvC,UAAI,EAAC,2BAAK,IAAI;AAEd,YAAM,aAAa,SAAS,eAC3B,YAAY,WAAW,OAAO,GAAG,KACjC,KAAK,WAAW,cAAc,GAAG;AAClC,WAAK,qBAAqB,KAAK,UAAU;AAAA,IAC1C;AAGA,eAAW,CAAC,UAAU,KAAK,KAAK,iBAAiB;AAEhD,YAAM,gBAAgB,aAAa,mBAClC,cAAc,IAAI,QAAQ,KAC1B,gBAAgB;AAEjB,UAAI,CAAC,eAAe;AAEnB,mBAAW,QAAQ,OAAO;AACzB,gBAAM,MAAM,aAAa,UAAU,IAAI;AACvC,cAAI,EAAC,2BAAK,IAAI;AACd,eAAK,qBAAqB,KAAK,KAAK;AAAA,QACrC;AAAA,MACD,OAAO;AAEN,mBAAW,QAAQ,OAAO;AACzB,gBAAM,MAAM,aAAa,UAAU,IAAI;AACvC,cAAI,EAAC,2BAAK,IAAI;AAEd,gBAAM,aAAa,SAAS,eAC3B,KAAK,WAAW,cAAc,GAAG,KACjC,cAAc,IAAI,IAAI;AACvB,eAAK,qBAAqB,KAAK,UAAU;AAAA,QAC1C;AAAA,MACD;AAAA,IACD;AAAA,EACD;AAAA,EAEQ,qBAAqB,KAAsD,YAA2B;AAC7G,UAAM,kBAAkB,IAAI,GAAG,aAAa,4BAA4B;AAGxE,QAAI,cAAc,iBAAiB;AAClC,UAAI,IAAI,KAAM,KAAI,KAAK,SAAS;AAChC,UAAI,GAAG,YAAY,EAAE,SAAS,GAAG,CAAC;AAClC,UAAI,GAAG,gBAAgB,4BAA4B;AAAA,IACpD,WAAW,CAAC,cAAc,CAAC,iBAAiB;AAC3C,UAAI,IAAI,KAAM,KAAI,KAAK,SAAS;AAChC,UAAI,GAAG,YAAY,EAAE,SAAS,OAAO,CAAC;AACtC,UAAI,GAAG,aAAa,8BAA8B,MAAM;AAAA,IACzD;AAAA,EACD;AAAA,EAEA,yBAA+B;AAC9B,UAAM,gBAAgB,oBAAoB,IAAI;AAC9C,kBAAc,QAAQ,kBAAgB;AACrC,YAAM,cAAc,aAAa;AACjC,UAAI,KAAK,SAAS;AACjB,oBAAY,SAAS,qBAAqB;AAAA,MAC3C,OAAO;AACN,oBAAY,YAAY,qBAAqB;AAAA,MAC9C;AAAA,IACD,CAAC;AAAA,EACF;AAAA,EAGA,sBAAsB;AACrB,UAAM,kBAAkB,MAAM;AAC7B,UAAI,KAAK,SAAS;AACjB,aAAK,UAAU;AAAA,MAChB,OAAO;AACN,YAAI,KAAK,SAAS,eAAe,UAAU;AAE1C,cAAI,KAAK,SAAS,kBAAkB;AACnC,iBAAK,WAAW,KAAK,SAAS,gBAAgB;AAAA,UAC/C;AAAA,QACD,OAAO;AAEN,gBAAM,OAAO,KAAK,IAAI,UAAU,cAAc;AAC9C,cAAI,6BAAM,MAAM;AACf,kBAAM,YAAY,aAAa,KAAK,MAAM,KAAK,SAAS,YAAY,KAAK,QAAQ;AACjF,iBAAK,WAAW,SAAS;AAAA,UAC1B;AAAA,QACD;AAAA,MACD;AAAA,IACD;AAEA,UAAM,wBAAwB,MAAM;AACnC,YAAM,qBAAqB,KAAK,IAAI,UAAU,gBAAgB,eAAe;AAC7E,UAAI,mBAAmB,WAAW,GAAG;AACpC;AAAA,MACD;AAEA,YAAM,mBAAmB,mBAAmB,CAAC,EAAE,KAAK;AACpD,YAAM,sBAAsB,wBAAwB,gBAAgB;AAEpE,UAAI,CAAC,qBAAqB;AACzB;AAAA,MACD;AAGA,UAAI,KAAK,oBAAoB,oBAAoB,SAAS,KAAK,gBAAgB,GAAG;AACjF;AAAA,MACD;AAGA,UAAI,KAAK,oBAAoB,KAAK,iBAAiB,eAAe;AACjE,aAAK,iBAAiB,OAAO;AAAA,MAC9B;AAGA,UAAI,CAAC,KAAK,kBAAkB;AAC3B,aAAK,mBAAmB,uBAAuB,IAAI;AAInD,aAAK,iBAAiB,YAAY,EAAE,aAAa,eAAe,CAAC;AAIjE,YAAI,eAAe;AAEnB,aAAK,iBAAiB,KAAK,kBAAkB,cAAc,CAAC,QAAQ;AACnE,yBAAe;AACf,cAAI,eAAe;AACnB,cAAI,gBAAgB;AACpB,0BAAgB;AAEhB,qBAAW,MAAM;AAAE,2BAAe;AAAA,UAAO,GAAG,GAAG;AAAA,QAChD,CAAC;AAED,aAAK,iBAAiB,KAAK,kBAAkB,SAAS,CAAC,QAAQ;AAE9D,cAAI,cAAc;AACjB,gBAAI,eAAe;AACnB,gBAAI,gBAAgB;AACpB;AAAA,UACD;AACA,cAAI,eAAe;AACnB,cAAI,gBAAgB;AACpB,0BAAgB;AAAA,QACjB,CAAC;AAAA,MACF;AAGA,WAAK,uBAAuB;AAG5B,6BAAuB,KAAK,kBAAkB,mBAAmB;AAAA,IAClE;AAGA,0BAAsB;AAGtB,SAAK;AAAA,MACJ,KAAK,IAAI,UAAU,GAAG,iBAAiB,MAAM;AAC5C,YAAI,KAAK,SAAS,sBAAsB;AACvC,gCAAsB;AAAA,QACvB;AAAA,MACD,CAAC;AAAA,IACF;AAAA,EACD;AAAA,EAEA,yBAAyB;AACxB,QAAI,CAAC,KAAK,kBAAkB;AAC3B;AAAA,IACD;AAEA,QAAI,CAAC,KAAK,SAAS,sBAAsB;AACxC,UAAI,KAAK,iBAAiB,eAAe;AACxC,aAAK,iBAAiB,OAAO;AAAA,MAC9B;AACA;AAAA,IACD;AAGA,QAAI,CAAC,KAAK,iBAAiB,eAAe;AAEzC,YAAM,qBAAqB,KAAK,IAAI,UAAU,gBAAgB,eAAe;AAC7E,UAAI,mBAAmB,SAAS,GAAG;AAClC,cAAM,mBAAmB,mBAAmB,CAAC,EAAE,KAAK;AACpD,cAAM,sBAAsB,wBAAwB,gBAAgB;AAEpE,YAAI,qBAAqB;AACxB,iCAAuB,KAAK,kBAAkB,mBAAmB;AAAA,QAClE;AAAA,MACD;AAAA,IACD;AAIA,QAAI,KAAK,iBAAiB,MAAM,QAAQ;AACvC,WAAK,iBAAiB,MAAM,eAAe,QAAQ;AAAA,IACpD;AAGA,QAAI,KAAK,SAAS;AACjB,WAAK,iBAAiB,SAAS,WAAW;AAAA,IAC3C,OAAO;AACN,WAAK,iBAAiB,YAAY,WAAW;AAAA,IAC9C;AAAA,EACD;AAAA,EAEA,uBAA6B;AA9X9B;AA+XE,UAAM,QAAQ,IAAI;AAAA,QAChB,UAAK,SAAS,kBAAd,YAA+B,CAAC,GAC/B,IAAI,OAAK,EAAE,KAAK,CAAC,EACjB,OAAO,OAAK,EAAE,SAAS,CAAC;AAAA,IAC3B;AAEA,UAAM,gBAAgB,oBAAoB,IAAI;AAC9C,kBAAc,QAAQ,kBAAgB;AACrC,UAAI,EAAC,6CAAc,WAAW;AAE9B,iBAAW,CAAC,UAAU,GAAG,KAAK,OAAO,QAAQ,aAAa,SAAS,GAAG;AACrE,YAAI,EAAC,2BAAK,IAAI;AACd,cAAM,aAAa,MAAM,IAAI,QAAQ;AACrC,cAAM,WAAW,IAAI,GAAG,aAAa,iCAAiC;AAEtE,YAAI,cAAc,CAAC,UAAU;AAC5B,cAAI,GAAG,YAAY,EAAE,SAAS,OAAO,CAAC;AACtC,cAAI,GAAG,aAAa,mCAAmC,MAAM;AAAA,QAC9D,WAAW,CAAC,cAAc,UAAU;AACnC,cAAI,GAAG,YAAY,EAAE,SAAS,GAAG,CAAC;AAClC,cAAI,GAAG,gBAAgB,iCAAiC;AAAA,QACzD;AAAA,MACD;AAAA,IACD,CAAC;AAAA,EACF;AAAA,EAEA,WAAW;AACV,QAAI,KAAK,kBAAkB;AAC1B,WAAK,iBAAiB,OAAO;AAC7B,WAAK,mBAAmB;AAAA,IACzB;AAAA,EACD;AACD;AAEA,IAAO,eAAQ;",
  "names": ["import_obsidian", "import_obsidian", "import_obsidian", "import_obsidian", "_a"]
}
