/* THIS IS A GENERATED/BUNDLED FILE BY ESBUILD if you want to view the source, please visit the github repository of this plugin */ 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, { default: () => FileNameHistoryPlugin }); module.exports = __toCommonJS(main_exports); var import_obsidian3 = require("obsidian"); // src/settings.ts var DEFAULT_SETTINGS = { historyPropertyName: "aliases", ignoreRegexes: ["^_", "^Untitled$", "^Untitled \\d+$"], timeoutSeconds: 5, caseSensitive: false, autoCreateFrontmatter: true, includeFolders: [], excludeFolders: [], fileExtensions: ["md"], trackFolderRenames: "", excludePropertyName: "" }; // src/ui/settings-tab.ts var import_obsidian = require("obsidian"); var FileNameHistorySettingTab = class extends import_obsidian.PluginSettingTab { constructor(app, plugin) { super(app, plugin); this.icon = "lucide-forward"; this.plugin = plugin; } display() { const { containerEl } = this; containerEl.empty(); const saveSettings = () => { void this.plugin.saveSettings(); }; const generalGroup = new import_obsidian.SettingGroup(containerEl); generalGroup.addSetting((setting) => { setting.setName("History property name").setDesc("The list property to store file name history.").addText( (text) => text.setPlaceholder("aliases").setValue(this.plugin.settings.historyPropertyName).onChange((value) => { this.plugin.settings.historyPropertyName = value || "aliases"; saveSettings(); }) ); }); generalGroup.addSetting((setting) => { setting.setName("Timeout seconds").setDesc("Time in seconds the name must be stable before adding to the configured property.").addSlider( (slider) => slider.setLimits(1, 20, 1).setValue(this.plugin.settings.timeoutSeconds).setDynamicTooltip().onChange((value) => { this.plugin.settings.timeoutSeconds = value; saveSettings(); }) ); }); generalGroup.addSetting((setting) => { setting.setName("Case-sensitive uniqueness").setDesc("If enabled, treat case differences as unique values in the configured property.").addToggle( (toggle) => toggle.setValue(this.plugin.settings.caseSensitive).onChange((value) => { this.plugin.settings.caseSensitive = value; saveSettings(); }) ); }); generalGroup.addSetting((setting) => { setting.setName("Auto-create history property").setDesc("Automatically create the configured property if missing.").addToggle( (toggle) => toggle.setValue(this.plugin.settings.autoCreateFrontmatter).onChange((value) => { this.plugin.settings.autoCreateFrontmatter = value; saveSettings(); }) ); }); generalGroup.addSetting((setting) => { setting.setName("File extensions").setDesc("Comma-separated list of file extensions to track.").addText( (text) => text.setPlaceholder("Md, txt").setValue(this.plugin.settings.fileExtensions.join(",")).onChange((value) => { this.plugin.settings.fileExtensions = value.split(",").map((s) => s.trim()).filter((s) => s); saveSettings(); }) ); }); const filteringGroup = new import_obsidian.SettingGroup(containerEl).setHeading("Filtering"); const foldersGroup = new import_obsidian.SettingGroup(containerEl).setHeading("Folders"); const advancedGroup = new import_obsidian.SettingGroup(containerEl).setHeading("Advanced"); filteringGroup.addSetting((setting) => { setting.setName("Ignore regex patterns").setDesc( "Comma-separated regex patterns for file names or immediate parent folder names to ignore (e.g., ^_ for underscore prefixes, ^untitled$ for untitled). Leave empty to disable." ).addText( (text) => text.setPlaceholder("^_, ^untitled$, ^untitled \\d+$").setValue(this.plugin.settings.ignoreRegexes.join(",")).onChange((value) => { this.plugin.settings.ignoreRegexes = value.split(",").map((s) => s.trim()).filter((s) => s); saveSettings(); }) ); }); filteringGroup.addSetting((setting) => { setting.setName("Exclude property name").setDesc( "Name of a boolean property to check in files. Files with this property set to true will be excluded from tracking. Takes priority over folder filtering." ).addText( (text) => text.setPlaceholder("Skip-rename-tracking").setValue(this.plugin.settings.excludePropertyName).onChange((value) => { this.plugin.settings.excludePropertyName = value; saveSettings(); }) ); }); foldersGroup.addSetting((setting) => { setting.setName("Include folders").setDesc( "Comma-separated list of folder paths to include. If empty, all folders are included. Use {vault} or {root} to include only files directly in the vault root (no subfolders)." ).addText( (text) => text.setValue(this.plugin.settings.includeFolders.join(",")).onChange((value) => { this.plugin.settings.includeFolders = value.split(",").map((s) => s.trim()).filter((s) => s); saveSettings(); }) ); }); foldersGroup.addSetting((setting) => { setting.setName("Exclude folders").setDesc( 'Comma-separated list of folder paths to exclude. Supports wildcards: use "folder/*" to exclude direct children, "folder/**" to exclude all descendants. Use {vault} or {root} to exclude files directly in the vault root.' ).addText( (text) => text.setValue(this.plugin.settings.excludeFolders.join(",")).onChange((value) => { this.plugin.settings.excludeFolders = value.split(",").map((s) => s.trim()).filter((s) => s); saveSettings(); }) ); }); advancedGroup.addSetting((setting) => { setting.setName("Track folder renames for specific file name").setDesc( "If a Markdown file matches this file name, store old immediate parent folder names in the configured property when parent folders are renamed." ).addText( (text) => text.setPlaceholder("Index").setValue(this.plugin.settings.trackFolderRenames).onChange((value) => { this.plugin.settings.trackFolderRenames = value; saveSettings(); }) ); }); } }; // src/utils/history-processor.ts var import_obsidian2 = require("obsidian"); var HistoryProcessor = class { constructor(app, settings) { this.app = app; this.settings = settings; } async processAliasesManually(path, queue) { const file = this.app.vault.getFileByPath(path); if (!file) { return; } const regexes = []; for (const regexStr of this.settings.ignoreRegexes) { try { regexes.push(new RegExp(regexStr)); } catch (e) { console.error(`Invalid ignore regex: ${regexStr}`, e); } } const toAdd = []; const currentBasename = file.basename; const currentBasenameLower = currentBasename.toLowerCase(); for (const name of queue) { if (regexes.some((re) => re.test(name))) { continue; } const nameLower = name.toLowerCase(); if (this.settings.caseSensitive && name === currentBasename || !this.settings.caseSensitive && nameLower === currentBasenameLower) { continue; } toAdd.push(name); } if (toAdd.length === 0) { return; } let content = await this.app.vault.read(file); const frontmatterRegex = /^---\n([\s\S]*?)\n---\n/; const match = content.match(frontmatterRegex); let frontmatter = {}; let frontmatterText = ""; let bodyContent = content; if (match) { frontmatterText = match[1]; bodyContent = content.slice(match[0].length); try { const parsed = (0, import_obsidian2.parseYaml)(frontmatterText); frontmatter = parsed && typeof parsed === "object" ? parsed : {}; } catch (e) { console.error(`Error parsing properties:`, e); frontmatter = {}; } } else { bodyContent = content; } let aliases = frontmatter[this.settings.historyPropertyName]; if (!Array.isArray(aliases)) { const hasFrontmatter = Object.keys(frontmatter).length > 0; if (hasFrontmatter && !this.settings.autoCreateFrontmatter) { return; } aliases = []; } const aliasesArray = aliases; const existing = new Set( this.settings.caseSensitive ? aliasesArray : aliasesArray.map((a) => a.toLowerCase()) ); let added = false; for (const name of toAdd) { const checkName = this.settings.caseSensitive ? name : name.toLowerCase(); if (!existing.has(checkName)) { aliasesArray.push(name); existing.add(checkName); added = true; } } if (!added) { return; } frontmatter[this.settings.historyPropertyName] = aliasesArray; const newFrontmatterText = (0, import_obsidian2.stringifyYaml)(frontmatter).trim(); const newContent = `--- ${newFrontmatterText} --- ${bodyContent}`; await this.app.vault.modify(file, newContent); } async processAliases(path, queue) { const file = this.app.vault.getFileByPath(path); if (!file) return; const regexes = []; for (const regexStr of this.settings.ignoreRegexes) { try { regexes.push(new RegExp(regexStr)); } catch (e) { console.error(`Invalid ignore regex: ${regexStr}`, e); } } const toAdd = []; const currentBasename = file.basename; const currentBasenameLower = currentBasename.toLowerCase(); for (const name of queue) { if (regexes.some((re) => re.test(name))) { continue; } const nameLower = name.toLowerCase(); if (this.settings.caseSensitive && name === currentBasename || !this.settings.caseSensitive && nameLower === currentBasenameLower) { continue; } toAdd.push(name); } if (toAdd.length === 0) { return; } if (file.extension !== "md") { await this.processAliasesManually(path, queue); return; } await this.app.fileManager.processFrontMatter(file, (fm) => { let aliases = fm[this.settings.historyPropertyName]; if (!Array.isArray(aliases)) { const hasFrontmatter = Object.keys(fm).length > 0; if (hasFrontmatter && !this.settings.autoCreateFrontmatter) { return; } aliases = []; fm[this.settings.historyPropertyName] = aliases; } const aliasesArray = aliases; const existing = new Set( this.settings.caseSensitive ? aliasesArray : aliasesArray.map((a) => a.toLowerCase()) ); for (const name of toAdd) { const checkName = this.settings.caseSensitive ? name : name.toLowerCase(); if (!existing.has(checkName)) { aliasesArray.push(name); existing.add(checkName); } } }); } }; // src/utils/path-utils.ts function getBasename(path) { const name = path.split("/").pop() || ""; return name.replace(/\.[^/.]+$/, ""); } function getImmediateParentName(path) { const parts = path.split("/"); parts.pop(); return parts.pop() || ""; } // src/main.ts var FileNameHistoryPlugin = class extends import_obsidian3.Plugin { constructor() { super(...arguments); this.debounceMap = /* @__PURE__ */ new Map(); } async onload() { await this.loadSettings(); this.historyProcessor = new HistoryProcessor(this.app, this.settings); this.addSettingTab(new FileNameHistorySettingTab(this.app, this)); this.registerEvent( this.app.vault.on("rename", (file, oldPath) => { this.handleRename(file, oldPath); }) ); } onunload() { for (const entry of this.debounceMap.values()) { if (entry.timeoutId !== 0) { window.clearTimeout(entry.timeoutId); } } this.debounceMap.clear(); } async loadSettings() { const loadedData = await this.loadData(); this.settings = Object.assign({}, DEFAULT_SETTINGS, loadedData); } async saveSettings() { await this.saveData(this.settings); } isPathInFolder(path, folder) { if (folder.includes("{vault}") || folder.includes("{root}")) { const resolvedFolder = folder.replace(/\{vault\}|\{root\}/g, ""); if (resolvedFolder === "" || resolvedFolder === "/") { const isVaultRoot = !path.includes("/"); return isVaultRoot; } return path.startsWith(resolvedFolder + "/") || path === resolvedFolder; } return path.startsWith(folder + "/") || path === folder; } isPathExcluded(path, excludePattern) { if (excludePattern.includes("{vault}") || excludePattern.includes("{root}")) { const resolvedPattern = excludePattern.replace(/\{vault\}|\{root\}/g, ""); if (resolvedPattern === "" || resolvedPattern === "/") { return !path.includes("/"); } excludePattern = resolvedPattern; } if (excludePattern.endsWith("/**")) { const baseFolder = excludePattern.slice(0, -3); return path.startsWith(baseFolder + "/") || path === baseFolder; } else if (excludePattern.endsWith("/*")) { const baseFolder = excludePattern.slice(0, -2); if (!path.startsWith(baseFolder + "/")) { return path === baseFolder; } const pathAfterBase = path.slice(baseFolder.length + 1); return pathAfterBase.includes("/"); } return path.startsWith(excludePattern + "/") || path === excludePattern; } handleRename(newFile, oldPath) { if (!(newFile instanceof import_obsidian3.TFile)) return; if (!this.settings.fileExtensions.includes(newFile.extension)) return; const oldBasename = getBasename(oldPath); const newBasename = newFile.basename; const oldImmediateParentName = getImmediateParentName(oldPath); const newImmediateParentName = getImmediateParentName(newFile.path); const isNameChange = this.settings.caseSensitive ? oldBasename !== newBasename : oldBasename.toLowerCase() !== newBasename.toLowerCase(); const isFolderChange = oldImmediateParentName !== newImmediateParentName && !isNameChange; if (!isNameChange && !isFolderChange) { return; } const path = newFile.path; if (this.settings.excludePropertyName && this.settings.excludePropertyName.trim() !== "") { const cache = this.app.metadataCache.getFileCache(newFile); const frontmatter = cache == null ? void 0 : cache.frontmatter; if (frontmatter && frontmatter[this.settings.excludePropertyName] === true) { return; } } if (this.settings.includeFolders.length > 0) { if (!this.settings.includeFolders.some((f) => this.isPathInFolder(path, f))) { return; } } const isIndexFileForFolderRename = isFolderChange && this.settings.trackFolderRenames && this.settings.trackFolderRenames.trim() !== "" && (this.settings.caseSensitive ? newFile.basename === this.settings.trackFolderRenames : newFile.basename.toLowerCase() === this.settings.trackFolderRenames.toLowerCase()); for (const excludePattern of this.settings.excludeFolders) { if (this.isPathExcluded(path, excludePattern)) { if (isIndexFileForFolderRename && excludePattern.endsWith("/*") && !excludePattern.endsWith("/**")) { const baseFolder = excludePattern.slice(0, -2); if (path.startsWith(baseFolder + "/")) { const pathAfterBase = path.slice(baseFolder.length + 1); const pathParts = pathAfterBase.split("/"); if (pathParts.length === 2) { continue; } } } return; } } const regexes = []; for (const regexStr of this.settings.ignoreRegexes) { try { regexes.push(new RegExp(regexStr)); } catch (e) { console.error(`Invalid ignore regex: ${regexStr}`, e); } } let toQueue = null; if (isNameChange) { if (regexes.some((re) => re.test(oldBasename) || re.test(newBasename))) { return; } toQueue = oldBasename; } else if (isFolderChange && this.settings.trackFolderRenames && this.settings.trackFolderRenames.trim() !== "") { const currentBasename = newFile.basename; const matchesFilename = this.settings.caseSensitive ? currentBasename === this.settings.trackFolderRenames : currentBasename.toLowerCase() === this.settings.trackFolderRenames.toLowerCase(); if (!matchesFilename) { return; } if (oldImmediateParentName === "" || newImmediateParentName === "") { return; } if (regexes.some((re) => re.test(oldImmediateParentName) || re.test(newImmediateParentName))) { return; } toQueue = oldImmediateParentName; } if (!toQueue) return; let existingEntry = this.debounceMap.get(newFile.path); if (!existingEntry) { existingEntry = this.debounceMap.get(oldPath); if (existingEntry) { this.debounceMap.delete(oldPath); } } if (existingEntry) { if (existingEntry.timeoutId !== 0) { window.clearTimeout(existingEntry.timeoutId); } toQueue = Array.from(existingEntry.queue)[0]; } const entry = { queue: /* @__PURE__ */ new Set([toQueue]), timeoutId: 0, currentPath: newFile.path }; entry.timeoutId = window.setTimeout(() => { void (async () => { try { await this.historyProcessor.processAliases(entry.currentPath, entry.queue); } catch (error) { console.error("Error processing aliases:", error); } this.debounceMap.delete(entry.currentPath); })(); }, this.settings.timeoutSeconds * 1e3); this.debounceMap.set(newFile.path, entry); } }; //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["src/main.ts", "src/settings.ts", "src/ui/settings-tab.ts", "src/utils/history-processor.ts", "src/utils/path-utils.ts"],
  "sourcesContent": ["import { Plugin, TAbstractFile, TFile } from 'obsidian';\nimport { FileNameHistorySettings, DEFAULT_SETTINGS } from './settings';\nimport { FileNameHistorySettingTab } from './ui/settings-tab';\nimport { HistoryProcessor } from './utils/history-processor';\nimport { getBasename, getImmediateParentName } from './utils/path-utils';\n\nexport default class FileNameHistoryPlugin extends Plugin {\n  settings: FileNameHistorySettings;\n  private debounceMap: Map<string, { queue: Set<string>; timeoutId: number; currentPath: string }> = new Map();\n  private historyProcessor: HistoryProcessor;\n\n  async onload() {\n    await this.loadSettings();\n    this.historyProcessor = new HistoryProcessor(this.app, this.settings);\n    this.addSettingTab(new FileNameHistorySettingTab(this.app, this));\n    this.registerEvent(\n      this.app.vault.on('rename', (file: TAbstractFile, oldPath: string) => {\n        this.handleRename(file, oldPath);\n      })\n    );\n  }\n\n  onunload() {\n    // Clear any pending timeouts\n    for (const entry of this.debounceMap.values()) {\n      if (entry.timeoutId !== 0) {\n        window.clearTimeout(entry.timeoutId);\n      }\n    }\n    this.debounceMap.clear();\n  }\n\n  async loadSettings() {\n    const loadedData = (await this.loadData()) as Partial<FileNameHistorySettings> | null;\n    this.settings = Object.assign({}, DEFAULT_SETTINGS, loadedData);\n  }\n\n  async saveSettings() {\n    await this.saveData(this.settings);\n  }\n\n  private isPathInFolder(path: string, folder: string): boolean {\n    // Handle vault root variable\n    if (folder.includes('{vault}') || folder.includes('{root}')) {\n      const resolvedFolder = folder.replace(/\\{vault\\}|\\{root\\}/g, '');\n      // If the folder is just the variable, it means include only vault root files\n      if (resolvedFolder === '' || resolvedFolder === '/') {\n        // Include only files directly in the vault root (no subfolders)\n        const isVaultRoot = !path.includes('/');\n        return isVaultRoot;\n      }\n      // Otherwise, replace the variable and check normally\n      return path.startsWith(resolvedFolder + '/') || path === resolvedFolder;\n    }\n\n    // Normal folder matching\n    return path.startsWith(folder + '/') || path === folder;\n  }\n\n  private isPathExcluded(path: string, excludePattern: string): boolean {\n    // Handle vault root variable\n    if (excludePattern.includes('{vault}') || excludePattern.includes('{root}')) {\n      const resolvedPattern = excludePattern.replace(/\\{vault\\}|\\{root\\}/g, '');\n      if (resolvedPattern === '' || resolvedPattern === '/') {\n        // Exclude only files directly in the vault root (no subfolders)\n        return !path.includes('/');\n      }\n      excludePattern = resolvedPattern;\n    }\n\n    // Handle wildcards\n    if (excludePattern.endsWith('/**')) {\n      // Recursive exclusion: docs/** matches docs and all subfolders\n      const baseFolder = excludePattern.slice(0, -3);\n      return path.startsWith(baseFolder + '/') || path === baseFolder;\n    } else if (excludePattern.endsWith('/*')) {\n      // Direct children only: docs/* matches docs/subfolder but not docs/subfolder/nested\n      // Should exclude files in subfolders, but NOT files directly in the base folder\n      const baseFolder = excludePattern.slice(0, -2);\n      if (!path.startsWith(baseFolder + '/')) {\n        return path === baseFolder;\n      }\n      const pathAfterBase = path.slice(baseFolder.length + 1);\n      // Exclude if path is in a subfolder (has at least one slash)\n      // Don't exclude files directly in the base folder (no slash)\n      return pathAfterBase.includes('/');\n    }\n\n    // Normal folder matching\n    return path.startsWith(excludePattern + '/') || path === excludePattern;\n  }\n\n  private handleRename(newFile: TAbstractFile, oldPath: string) {\n    if (!(newFile instanceof TFile)) return;\n    if (!this.settings.fileExtensions.includes(newFile.extension)) return;\n\n    const oldBasename = getBasename(oldPath);\n    const newBasename = newFile.basename;\n    const oldImmediateParentName = getImmediateParentName(oldPath);\n    const newImmediateParentName = getImmediateParentName(newFile.path);\n\n    const isNameChange = this.settings.caseSensitive\n      ? oldBasename !== newBasename\n      : oldBasename.toLowerCase() !== newBasename.toLowerCase();\n    const isFolderChange = oldImmediateParentName !== newImmediateParentName && !isNameChange;\n\n    if (!isNameChange && !isFolderChange) {\n      return;\n    }\n\n    const path = newFile.path;\n\n    // Apply filtering checks to both file name changes and folder renames\n    // Priority: Property exclusion -> Include folders -> Exclude folders\n\n    // 1. Check property-based exclusion first (highest priority)\n    if (this.settings.excludePropertyName && this.settings.excludePropertyName.trim() !== '') {\n      const cache = this.app.metadataCache.getFileCache(newFile);\n      const frontmatter = cache?.frontmatter;\n      if (frontmatter && frontmatter[this.settings.excludePropertyName] === true) {\n        return; // Exclude this file\n      }\n    }\n\n    // 2. Check include folders (if includeFolders is not empty, only include those)\n    if (this.settings.includeFolders.length > 0) {\n      if (!this.settings.includeFolders.some(f => this.isPathInFolder(path, f))) {\n        return; // Not in any included folder\n      }\n    }\n\n    // 3. Check exclude folders (with wildcard support)\n    // Special case: if tracking folder renames for a specific file name, allow it even in excluded subfolders\n    // unless using recursive exclusion (**) or the file is nested deeper than one level\n    const isIndexFileForFolderRename = isFolderChange &&\n      this.settings.trackFolderRenames &&\n      this.settings.trackFolderRenames.trim() !== '' &&\n      (this.settings.caseSensitive\n        ? newFile.basename === this.settings.trackFolderRenames\n        : newFile.basename.toLowerCase() === this.settings.trackFolderRenames.toLowerCase());\n\n    for (const excludePattern of this.settings.excludeFolders) {\n      if (this.isPathExcluded(path, excludePattern)) {\n        // If this is an index file for folder rename tracking, and the pattern is /* (not /**),\n        // allow it through only if it's in a direct child folder (one level deep)\n        if (isIndexFileForFolderRename && excludePattern.endsWith('/*') && !excludePattern.endsWith('/**')) {\n          // Check if the file is in a direct child (only one level deep)\n          const baseFolder = excludePattern.slice(0, -2);\n          if (path.startsWith(baseFolder + '/')) {\n            const pathAfterBase = path.slice(baseFolder.length + 1);\n            const pathParts = pathAfterBase.split('/');\n            // If there's only one path part before the filename, it's a direct child\n            // pathParts will be like ['subfolder', 'filename.md'] - we want exactly 2 parts\n            if (pathParts.length === 2) {\n              continue; // Skip this exclusion - it's a direct child index file\n            }\n          }\n        }\n        return; // Excluded by folder pattern\n      }\n    }\n\n    // Check ignore regexes\n    const regexes: RegExp[] = [];\n    for (const regexStr of this.settings.ignoreRegexes) {\n      try {\n        regexes.push(new RegExp(regexStr));\n      } catch (e) {\n        console.error(`Invalid ignore regex: ${regexStr}`, e);\n      }\n    }\n\n    let toQueue: string | null = null;\n    if (isNameChange) {\n      if (regexes.some(re => re.test(oldBasename) || re.test(newBasename))) {\n        return;\n      }\n      toQueue = oldBasename;\n    } else if (isFolderChange && this.settings.trackFolderRenames && this.settings.trackFolderRenames.trim() !== '') {\n      // Check if the current file name matches the specified name (without extension)\n      const currentBasename = newFile.basename;\n      const matchesFilename = this.settings.caseSensitive\n        ? currentBasename === this.settings.trackFolderRenames\n        : currentBasename.toLowerCase() === this.settings.trackFolderRenames.toLowerCase();\n\n      if (!matchesFilename) {\n        return;\n      }\n\n      if (oldImmediateParentName === '' || newImmediateParentName === '') {\n        return;\n      }\n      if (regexes.some(re => re.test(oldImmediateParentName) || re.test(newImmediateParentName))) {\n        return;\n      }\n      toQueue = oldImmediateParentName;\n    }\n\n    if (!toQueue) return;\n\n    // Check if there's already a pending timeout for this file\n    // We need to check both the new path and the old path since the file was just renamed\n    let existingEntry = this.debounceMap.get(newFile.path);\n    if (!existingEntry) {\n      // Check if there's a timeout for the old path (the file was just renamed from there)\n      existingEntry = this.debounceMap.get(oldPath);\n      if (existingEntry) {\n        // Remove the old entry since we're updating it with the new path\n        this.debounceMap.delete(oldPath);\n      }\n    }\n\n    if (existingEntry) {\n      // File was renamed again before timeout expired - cancel the previous timeout\n      if (existingEntry.timeoutId !== 0) {\n        window.clearTimeout(existingEntry.timeoutId);\n      }\n\n      // Use the original stable name from the previous timeout, not the temporary name\n      toQueue = Array.from(existingEntry.queue)[0]; // Use the original stable name\n    }\n\n    // Create entry to track the timeout\n    const entry = {\n      queue: new Set<string>([toQueue]),\n      timeoutId: 0,\n      currentPath: newFile.path\n    };\n\n    // Set timeout to actually store the alias after the debounce period\n    entry.timeoutId = window.setTimeout(() => {\n      void (async () => {\n        try {\n          await this.historyProcessor.processAliases(entry.currentPath, entry.queue);\n        } catch (error) {\n          console.error('Error processing aliases:', error);\n        }\n        this.debounceMap.delete(entry.currentPath);\n      })();\n    }, this.settings.timeoutSeconds * 1000);\n\n    this.debounceMap.set(newFile.path, entry);\n  }\n}\n", "export interface FileNameHistorySettings {\r\n  historyPropertyName: string;\r\n  ignoreRegexes: string[];\r\n  timeoutSeconds: number;\r\n  caseSensitive: boolean;\r\n  autoCreateFrontmatter: boolean;\r\n  includeFolders: string[];\r\n  excludeFolders: string[];\r\n  fileExtensions: string[];\r\n  trackFolderRenames: string;\r\n  excludePropertyName: string;\r\n}\r\n\r\nexport const DEFAULT_SETTINGS: FileNameHistorySettings = {\r\n  historyPropertyName: 'aliases',\r\n  ignoreRegexes: ['^_', '^Untitled$', '^Untitled \\\\d+$'],\r\n  timeoutSeconds: 5,\r\n  caseSensitive: false,\r\n  autoCreateFrontmatter: true,\r\n  includeFolders: [],\r\n  excludeFolders: [],\r\n  fileExtensions: ['md'],\r\n  trackFolderRenames: '',\r\n  excludePropertyName: '',\r\n};", "import { App, Plugin, PluginSettingTab , SettingGroup} from 'obsidian';\r\nimport { FileNameHistorySettings } from '../settings';\r\n\r\n\r\ninterface FileNameHistoryPlugin extends Plugin {\r\n  settings: FileNameHistorySettings;\r\n  saveSettings(): Promise<void>;\r\n}\r\n\r\nexport class FileNameHistorySettingTab extends PluginSettingTab {\r\n  plugin: FileNameHistoryPlugin;\r\n  public icon = 'lucide-forward';\r\n\r\n  constructor(app: App, plugin: FileNameHistoryPlugin) {\r\n    super(app, plugin);\r\n    this.plugin = plugin;\r\n  }\r\n\r\n  display(): void {\r\n    const { containerEl } = this;\r\n    containerEl.empty();\r\n\r\n    // Helper to save settings without returning a promise\r\n    const saveSettings = (): void => {\r\n      void this.plugin.saveSettings();\r\n    };\r\n\r\n    const generalGroup = new SettingGroup(containerEl);\r\n\r\n    // General behavior settings (grouped, no heading)\r\n    generalGroup.addSetting(setting => {\r\n      setting\r\n        .setName('History property name')\r\n        .setDesc('The list property to store file name history.')\r\n        .addText(text =>\r\n          text\r\n            .setPlaceholder('aliases')\r\n            .setValue(this.plugin.settings.historyPropertyName)\r\n            .onChange(value => {\r\n              this.plugin.settings.historyPropertyName = value || 'aliases';\r\n              saveSettings();\r\n            })\r\n        );\r\n    });\r\n\r\n    generalGroup.addSetting(setting => {\r\n      setting\r\n        .setName('Timeout seconds')\r\n        .setDesc('Time in seconds the name must be stable before adding to the configured property.')\r\n        .addSlider(slider =>\r\n          slider\r\n            .setLimits(1, 20, 1)\r\n            .setValue(this.plugin.settings.timeoutSeconds)\r\n            .setDynamicTooltip()\r\n            .onChange(value => {\r\n              this.plugin.settings.timeoutSeconds = value;\r\n              saveSettings();\r\n            })\r\n        );\r\n    });\r\n\r\n    generalGroup.addSetting(setting => {\r\n      setting\r\n        .setName('Case-sensitive uniqueness')\r\n\r\n        .setDesc('If enabled, treat case differences as unique values in the configured property.')\r\n        .addToggle(toggle =>\r\n          toggle\r\n            .setValue(this.plugin.settings.caseSensitive)\r\n            .onChange(value => {\r\n              this.plugin.settings.caseSensitive = value;\r\n              saveSettings();\r\n            })\r\n        );\r\n    });\r\n\r\n    generalGroup.addSetting(setting => {\r\n      setting\r\n        .setName('Auto-create history property')\r\n        .setDesc('Automatically create the configured property if missing.')\r\n        .addToggle(toggle =>\r\n          toggle\r\n            .setValue(this.plugin.settings.autoCreateFrontmatter)\r\n            .onChange(value => {\r\n              this.plugin.settings.autoCreateFrontmatter = value;\r\n              saveSettings();\r\n            })\r\n        );\r\n    });\r\n\r\n    generalGroup.addSetting(setting => {\r\n      setting\r\n        .setName('File extensions')\r\n        .setDesc('Comma-separated list of file extensions to track.')\r\n        .addText(text =>\r\n          text\r\n\r\n            .setPlaceholder('Md, txt')\r\n            .setValue(this.plugin.settings.fileExtensions.join(','))\r\n            .onChange(value => {\r\n              this.plugin.settings.fileExtensions = value\r\n                .split(',')\r\n                .map(s => s.trim())\r\n                .filter(s => s);\r\n              saveSettings();\r\n            })\r\n        );\r\n    });\r\n\r\n    const filteringGroup = new SettingGroup(containerEl).setHeading('Filtering');\r\n    const foldersGroup = new SettingGroup(containerEl).setHeading('Folders');\r\n    const advancedGroup = new SettingGroup(containerEl).setHeading('Advanced');\r\n\r\n    // Filtering settings\r\n    filteringGroup.addSetting(setting => {\r\n      setting\r\n        .setName('Ignore regex patterns')\r\n        .setDesc(\r\n          'Comma-separated regex patterns for file names or immediate parent folder names to ignore (e.g., ^_ for underscore prefixes, ^untitled$ for untitled). Leave empty to disable.'\r\n        )\r\n        .addText(text =>\r\n          text\r\n            .setPlaceholder('^_, ^untitled$, ^untitled \\\\d+$')\r\n            .setValue(this.plugin.settings.ignoreRegexes.join(','))\r\n            .onChange(value => {\r\n              this.plugin.settings.ignoreRegexes = value\r\n                .split(',')\r\n                .map(s => s.trim())\r\n                .filter(s => s);\r\n              saveSettings();\r\n            })\r\n        );\r\n    });\r\n\r\n    filteringGroup.addSetting(setting => {\r\n      setting\r\n        .setName('Exclude property name')\r\n        .setDesc(\r\n          'Name of a boolean property to check in files. Files with this property set to true will be excluded from tracking. Takes priority over folder filtering.'\r\n        )\r\n        .addText(text =>\r\n          text\r\n\r\n            .setPlaceholder('Skip-rename-tracking')\r\n            .setValue(this.plugin.settings.excludePropertyName)\r\n            .onChange(value => {\r\n              this.plugin.settings.excludePropertyName = value;\r\n              saveSettings();\r\n            })\r\n        );\r\n    });\r\n\r\n    // Folder include/exclude settings\r\n    foldersGroup.addSetting(setting => {\r\n      setting\r\n        .setName('Include folders')\r\n        .setDesc(\r\n          'Comma-separated list of folder paths to include. If empty, all folders are included. Use {vault} or {root} to include only files directly in the vault root (no subfolders).'\r\n        )\r\n        .addText(text =>\r\n          text\r\n            .setValue(this.plugin.settings.includeFolders.join(','))\r\n            .onChange(value => {\r\n              this.plugin.settings.includeFolders = value\r\n                .split(',')\r\n                .map(s => s.trim())\r\n                .filter(s => s);\r\n              saveSettings();\r\n            })\r\n        );\r\n    });\r\n\r\n    foldersGroup.addSetting(setting => {\r\n      setting\r\n        .setName('Exclude folders')\r\n        .setDesc(\r\n          'Comma-separated list of folder paths to exclude. Supports wildcards: use \"folder/*\" to exclude direct children, \"folder/**\" to exclude all descendants. Use {vault} or {root} to exclude files directly in the vault root.'\r\n        )\r\n        .addText(text =>\r\n          text\r\n            .setValue(this.plugin.settings.excludeFolders.join(','))\r\n            .onChange(value => {\r\n              this.plugin.settings.excludeFolders = value\r\n                .split(',')\r\n                .map(s => s.trim())\r\n                .filter(s => s);\r\n              saveSettings();\r\n            })\r\n        );\r\n    });\r\n\r\n    // Advanced / niche options\r\n    advancedGroup.addSetting(setting => {\r\n      setting\r\n        .setName('Track folder renames for specific file name')\r\n        .setDesc(\r\n\r\n          'If a Markdown file matches this file name, store old immediate parent folder names in the configured property when parent folders are renamed.'\r\n        )\r\n        .addText(text =>\r\n          text\r\n\r\n            .setPlaceholder('Index')\r\n            .setValue(this.plugin.settings.trackFolderRenames)\r\n            .onChange(value => {\r\n              this.plugin.settings.trackFolderRenames = value;\r\n              saveSettings();\r\n            })\r\n        );\r\n    });\r\n\r\n  }\r\n}\r\n", "import { App, stringifyYaml, parseYaml } from 'obsidian';\r\nimport { FileNameHistorySettings } from '../settings';\r\n\r\nexport class HistoryProcessor {\r\n  constructor(\r\n    private app: App,\r\n    private settings: FileNameHistorySettings\r\n  ) { }\r\n\r\n  private async processAliasesManually(path: string, queue: Set<string>): Promise<void> {\r\n    const file = this.app.vault.getFileByPath(path);\r\n    if (!file) {\r\n      return;\r\n    }\r\n\r\n    const regexes: RegExp[] = [];\r\n    for (const regexStr of this.settings.ignoreRegexes) {\r\n      try {\r\n        regexes.push(new RegExp(regexStr));\r\n      } catch (e) {\r\n        console.error(`Invalid ignore regex: ${regexStr}`, e);\r\n      }\r\n    }\r\n\r\n    const toAdd: string[] = [];\r\n    const currentBasename = file.basename;\r\n    const currentBasenameLower = currentBasename.toLowerCase();\r\n\r\n    for (const name of queue) {\r\n      if (regexes.some(re => re.test(name))) {\r\n        continue;\r\n      }\r\n      const nameLower = name.toLowerCase();\r\n      if (\r\n        (this.settings.caseSensitive && name === currentBasename) ||\r\n        (!this.settings.caseSensitive && nameLower === currentBasenameLower)\r\n      ) {\r\n        continue;\r\n      }\r\n      toAdd.push(name);\r\n    }\r\n\r\n    if (toAdd.length === 0) {\r\n      return;\r\n    }\r\n\r\n    // Read file content\r\n    let content = await this.app.vault.read(file);\r\n\r\n    // Parse frontmatter\r\n    const frontmatterRegex = /^---\\n([\\s\\S]*?)\\n---\\n/;\r\n    const match = content.match(frontmatterRegex);\r\n\r\n    let frontmatter: Record<string, unknown> = {};\r\n    let frontmatterText = '';\r\n    let bodyContent = content;\r\n\r\n    if (match) {\r\n      frontmatterText = match[1];\r\n      bodyContent = content.slice(match[0].length);\r\n      try {\r\n        const parsed = parseYaml(frontmatterText) as Record<string, unknown> | null | undefined;\r\n        frontmatter = parsed && typeof parsed === 'object' ? parsed : {};\r\n      } catch (e) {\r\n        console.error(`Error parsing properties:`, e);\r\n        frontmatter = {};\r\n      }\r\n    } else {\r\n      bodyContent = content;\r\n    }\r\n\r\n    // Get or create aliases array\r\n    let aliases = frontmatter[this.settings.historyPropertyName];\r\n    if (!Array.isArray(aliases)) {\r\n      const hasFrontmatter = Object.keys(frontmatter).length > 0;\r\n      if (hasFrontmatter && !this.settings.autoCreateFrontmatter) {\r\n        return;\r\n      }\r\n      aliases = [];\r\n    }\r\n\r\n    const aliasesArray = aliases as string[];\r\n    const existing = new Set<string>(\r\n      this.settings.caseSensitive ? aliasesArray : aliasesArray.map((a: string) => a.toLowerCase())\r\n    );\r\n\r\n    let added = false;\r\n    for (const name of toAdd) {\r\n      const checkName = this.settings.caseSensitive ? name : name.toLowerCase();\r\n      if (!existing.has(checkName)) {\r\n        aliasesArray.push(name);\r\n        existing.add(checkName);\r\n        added = true;\r\n      }\r\n    }\r\n\r\n    if (!added) {\r\n      return;\r\n    }\r\n\r\n    // Update frontmatter\r\n    frontmatter[this.settings.historyPropertyName] = aliasesArray;\r\n\r\n    // Stringify frontmatter\r\n    const newFrontmatterText = stringifyYaml(frontmatter).trim();\r\n\r\n    // Reconstruct file content\r\n    const newContent = `---\\n${newFrontmatterText}\\n---\\n${bodyContent}`;\r\n\r\n    // Write back to file\r\n    await this.app.vault.modify(file, newContent);\r\n  }\r\n\r\n  async processAliases(path: string, queue: Set<string>): Promise<void> {\r\n    const file = this.app.vault.getFileByPath(path);\r\n    if (!file) return;\r\n\r\n    const regexes: RegExp[] = [];\r\n    for (const regexStr of this.settings.ignoreRegexes) {\r\n      try {\r\n        regexes.push(new RegExp(regexStr));\r\n      } catch (e) {\r\n        console.error(`Invalid ignore regex: ${regexStr}`, e);\r\n      }\r\n    }\r\n\r\n    const toAdd: string[] = [];\r\n    const currentBasename = file.basename;\r\n    const currentBasenameLower = currentBasename.toLowerCase();\r\n\r\n    for (const name of queue) {\r\n      if (regexes.some(re => re.test(name))) {\r\n        continue;\r\n      }\r\n      const nameLower = name.toLowerCase();\r\n      if (\r\n        (this.settings.caseSensitive && name === currentBasename) ||\r\n        (!this.settings.caseSensitive && nameLower === currentBasenameLower)\r\n      ) {\r\n        continue;\r\n      }\r\n      toAdd.push(name);\r\n    }\r\n\r\n    if (toAdd.length === 0) {\r\n      return;\r\n    }\r\n\r\n    // Use manual processing for non-md files, or as fallback\r\n    if (file.extension !== 'md') {\r\n      await this.processAliasesManually(path, queue);\r\n      return;\r\n    }\r\n\r\n    await this.app.fileManager.processFrontMatter(file, (fm: Record<string, unknown>) => {\r\n      let aliases = fm[this.settings.historyPropertyName];\r\n      if (!Array.isArray(aliases)) {\r\n        // If there's no frontmatter at all, we need to create it to add aliases\r\n        // If there's frontmatter but no aliases property, respect the autoCreateFrontmatter setting\r\n        const hasFrontmatter = Object.keys(fm).length > 0;\r\n        if (hasFrontmatter && !this.settings.autoCreateFrontmatter) {\r\n          return;\r\n        }\r\n        aliases = [];\r\n        fm[this.settings.historyPropertyName] = aliases;\r\n      }\r\n\r\n      const aliasesArray = aliases as string[];\r\n      const existing = new Set<string>(\r\n        this.settings.caseSensitive ? aliasesArray : aliasesArray.map((a: string) => a.toLowerCase())\r\n      );\r\n\r\n      for (const name of toAdd) {\r\n        const checkName = this.settings.caseSensitive ? name : name.toLowerCase();\r\n        if (!existing.has(checkName)) {\r\n          aliasesArray.push(name);\r\n          existing.add(checkName);\r\n        }\r\n      }\r\n    });\r\n  }\r\n}\r\n", "export function getBasename(path: string): string {\r\n  const name = path.split('/').pop() || '';\r\n  return name.replace(/\\.[^/.]+$/, '');\r\n}\r\n\r\nexport function getImmediateParentName(path: string): string {\r\n  const parts = path.split('/');\r\n  parts.pop(); // Remove file name\r\n  return parts.pop() || ''; // Get immediate parent folder name or '' if root\r\n}\r\n"],
  "mappings": "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",
  "names": ["import_obsidian", "import_obsidian"]
}
