obsidian-dev-utils 56.0.2 → 57.0.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (93) hide show
  1. package/CHANGELOG.md +14 -0
  2. package/dist/lib/cjs/async.cjs +10 -2
  3. package/dist/lib/cjs/async.d.cts +1 -1
  4. package/dist/lib/cjs/enum.cjs +1 -1
  5. package/dist/lib/cjs/library.cjs +1 -1
  6. package/dist/lib/cjs/object-utils.cjs +1 -1
  7. package/dist/lib/cjs/obsidian/command-handlers/abstract-file-command-handler.cjs +1 -1
  8. package/dist/lib/cjs/obsidian/command-handlers/abstract-file-command-handler.d.cts +2 -1
  9. package/dist/lib/cjs/obsidian/command-handlers/editor-command-handler.cjs +1 -1
  10. package/dist/lib/cjs/obsidian/command-handlers/editor-command-handler.d.cts +2 -1
  11. package/dist/lib/cjs/obsidian/command-handlers/file-command-handler.cjs +1 -1
  12. package/dist/lib/cjs/obsidian/command-handlers/file-command-handler.d.cts +2 -1
  13. package/dist/lib/cjs/obsidian/command-handlers/folder-command-handler.cjs +1 -1
  14. package/dist/lib/cjs/obsidian/command-handlers/folder-command-handler.d.cts +2 -1
  15. package/dist/lib/cjs/obsidian/command-handlers/global-command-handler.cjs +1 -1
  16. package/dist/lib/cjs/obsidian/command-handlers/global-command-handler.d.cts +2 -1
  17. package/dist/lib/cjs/obsidian/command-handlers/index.cjs +6 -3
  18. package/dist/lib/cjs/obsidian/command-handlers/index.d.cts +1 -0
  19. package/dist/lib/cjs/obsidian/command-handlers/open-settings-command-handler.cjs +160 -0
  20. package/dist/lib/cjs/obsidian/command-handlers/open-settings-command-handler.d.cts +24 -0
  21. package/dist/lib/cjs/obsidian/file-change.cjs +1 -1
  22. package/dist/lib/cjs/obsidian/file-system.cjs +1 -1
  23. package/dist/lib/cjs/obsidian/plugin/components/abort-signal-component.cjs +5 -1
  24. package/dist/lib/cjs/obsidian/plugin/components/abort-signal-component.d.cts +4 -0
  25. package/dist/lib/cjs/obsidian/plugin/components/async-error-handler-component.cjs +5 -1
  26. package/dist/lib/cjs/obsidian/plugin/components/async-error-handler-component.d.cts +4 -0
  27. package/dist/lib/cjs/obsidian/plugin/components/console-debug-component.cjs +5 -1
  28. package/dist/lib/cjs/obsidian/plugin/components/console-debug-component.d.cts +4 -0
  29. package/dist/lib/cjs/obsidian/plugin/components/i18n-component.cjs +5 -1
  30. package/dist/lib/cjs/obsidian/plugin/components/i18n-component.d.cts +4 -0
  31. package/dist/lib/cjs/obsidian/plugin/components/lifecycle-events-component.cjs +5 -1
  32. package/dist/lib/cjs/obsidian/plugin/components/lifecycle-events-component.d.cts +4 -0
  33. package/dist/lib/cjs/obsidian/plugin/components/plugin-context-component.cjs +5 -1
  34. package/dist/lib/cjs/obsidian/plugin/components/plugin-context-component.d.cts +4 -0
  35. package/dist/lib/cjs/obsidian/plugin/components/plugin-notice-component.cjs +5 -1
  36. package/dist/lib/cjs/obsidian/plugin/components/plugin-notice-component.d.cts +4 -0
  37. package/dist/lib/cjs/obsidian/plugin/components/plugin-settings-component.cjs +2 -5
  38. package/dist/lib/cjs/obsidian/plugin/components/plugin-settings-component.d.cts +1 -5
  39. package/dist/lib/cjs/obsidian/plugin/components/plugin-settings-tab-component.cjs +4 -10
  40. package/dist/lib/cjs/obsidian/plugin/plugin-settings-tab.cjs +1 -1
  41. package/dist/lib/cjs/obsidian/plugin/plugin.cjs +21 -17
  42. package/dist/lib/cjs/obsidian/plugin/plugin.d.cts +7 -8
  43. package/dist/lib/cjs/script-utils/exec.cjs +4 -4
  44. package/dist/lib/cjs/script-utils/exec.d.cts +7 -7
  45. package/dist/lib/cjs/script-utils/linters/eslint-config.cjs +1 -1
  46. package/dist/lib/cjs/script-utils/linters/eslint-rules/no-async-callback-to-unsafe-return.cjs +1 -1
  47. package/dist/lib/esm/async.d.mts +1 -1
  48. package/dist/lib/esm/async.mjs +10 -2
  49. package/dist/lib/esm/enum.mjs +1 -1
  50. package/dist/lib/esm/library.mjs +1 -1
  51. package/dist/lib/esm/object-utils.mjs +1 -1
  52. package/dist/lib/esm/obsidian/command-handlers/abstract-file-command-handler.d.mts +2 -1
  53. package/dist/lib/esm/obsidian/command-handlers/abstract-file-command-handler.mjs +1 -1
  54. package/dist/lib/esm/obsidian/command-handlers/editor-command-handler.d.mts +2 -1
  55. package/dist/lib/esm/obsidian/command-handlers/editor-command-handler.mjs +1 -1
  56. package/dist/lib/esm/obsidian/command-handlers/file-command-handler.d.mts +2 -1
  57. package/dist/lib/esm/obsidian/command-handlers/file-command-handler.mjs +1 -1
  58. package/dist/lib/esm/obsidian/command-handlers/folder-command-handler.d.mts +2 -1
  59. package/dist/lib/esm/obsidian/command-handlers/folder-command-handler.mjs +1 -1
  60. package/dist/lib/esm/obsidian/command-handlers/global-command-handler.d.mts +2 -1
  61. package/dist/lib/esm/obsidian/command-handlers/global-command-handler.mjs +1 -1
  62. package/dist/lib/esm/obsidian/command-handlers/index.d.mts +1 -0
  63. package/dist/lib/esm/obsidian/command-handlers/index.mjs +4 -2
  64. package/dist/lib/esm/obsidian/command-handlers/open-settings-command-handler.d.mts +24 -0
  65. package/dist/lib/esm/obsidian/command-handlers/open-settings-command-handler.mjs +52 -0
  66. package/dist/lib/esm/obsidian/file-change.mjs +1 -1
  67. package/dist/lib/esm/obsidian/file-system.mjs +1 -1
  68. package/dist/lib/esm/obsidian/plugin/components/abort-signal-component.d.mts +4 -0
  69. package/dist/lib/esm/obsidian/plugin/components/abort-signal-component.mjs +5 -1
  70. package/dist/lib/esm/obsidian/plugin/components/async-error-handler-component.d.mts +4 -0
  71. package/dist/lib/esm/obsidian/plugin/components/async-error-handler-component.mjs +5 -1
  72. package/dist/lib/esm/obsidian/plugin/components/console-debug-component.d.mts +4 -0
  73. package/dist/lib/esm/obsidian/plugin/components/console-debug-component.mjs +5 -1
  74. package/dist/lib/esm/obsidian/plugin/components/i18n-component.d.mts +4 -0
  75. package/dist/lib/esm/obsidian/plugin/components/i18n-component.mjs +5 -1
  76. package/dist/lib/esm/obsidian/plugin/components/lifecycle-events-component.d.mts +4 -0
  77. package/dist/lib/esm/obsidian/plugin/components/lifecycle-events-component.mjs +5 -1
  78. package/dist/lib/esm/obsidian/plugin/components/plugin-context-component.d.mts +4 -0
  79. package/dist/lib/esm/obsidian/plugin/components/plugin-context-component.mjs +5 -1
  80. package/dist/lib/esm/obsidian/plugin/components/plugin-notice-component.d.mts +4 -0
  81. package/dist/lib/esm/obsidian/plugin/components/plugin-notice-component.mjs +5 -1
  82. package/dist/lib/esm/obsidian/plugin/components/plugin-settings-component.d.mts +1 -5
  83. package/dist/lib/esm/obsidian/plugin/components/plugin-settings-component.mjs +2 -4
  84. package/dist/lib/esm/obsidian/plugin/components/plugin-settings-tab-component.mjs +4 -10
  85. package/dist/lib/esm/obsidian/plugin/plugin-settings-tab.mjs +1 -1
  86. package/dist/lib/esm/obsidian/plugin/plugin.d.mts +7 -8
  87. package/dist/lib/esm/obsidian/plugin/plugin.mjs +21 -17
  88. package/dist/lib/esm/script-utils/exec.d.mts +7 -7
  89. package/dist/lib/esm/script-utils/exec.mjs +4 -4
  90. package/dist/lib/esm/script-utils/linters/eslint-config.mjs +1 -1
  91. package/dist/lib/esm/script-utils/linters/eslint-rules/no-async-callback-to-unsafe-return.mjs +1 -1
  92. package/obsidian/command-handlers/open-settings-command-handler/package.json +6 -0
  93. package/package.json +4 -1
@@ -9,6 +9,10 @@ import { Component } from 'obsidian';
9
9
  */
10
10
  export declare class PluginNoticeComponent extends Component {
11
11
  private readonly pluginName;
12
+ /**
13
+ * The singleton key for the {@link PluginNoticeComponent} class.
14
+ */
15
+ static readonly COMPONENT_KEY: unique symbol;
12
16
  private notice?;
13
17
  /**
14
18
  * Creates a new plugin notice component.
@@ -126,7 +126,6 @@ var __toCommonJS = (mod) => __copyProps(__defProp({}, "__esModule", { value: tru
126
126
  var plugin_settings_component_exports = {};
127
127
  __export(plugin_settings_component_exports, {
128
128
  EmptyPluginSettingsComponent: () => EmptyPluginSettingsComponent,
129
- PLUGIN_SETTINGS_COMPONENT_KEY: () => PLUGIN_SETTINGS_COMPONENT_KEY,
130
129
  PluginSettingsComponentBase: () => PluginSettingsComponentBase
131
130
  });
132
131
  module.exports = __toCommonJS(plugin_settings_component_exports);
@@ -151,12 +150,11 @@ const defaultTransformer = new import_group_transformer.GroupTransformer([
151
150
  new import_set_transformer.SetTransformer(),
152
151
  new import_two_way_map_transformer.TwoWayMapTransformer()
153
152
  ]);
154
- const PLUGIN_SETTINGS_COMPONENT_KEY = "PluginSettingsComponent";
155
153
  class PluginSettingsComponentBase extends import_async_component.AsyncComponentBase {
156
154
  /**
157
155
  * Component key for {@link registerComponent} replacement.
158
156
  */
159
- static COMPONENT_KEY = PLUGIN_SETTINGS_COMPONENT_KEY;
157
+ static COMPONENT_KEY = Symbol(PluginSettingsComponentBase.name);
160
158
  /**
161
159
  * Gets the readonly default settings.
162
160
  *
@@ -560,7 +558,6 @@ class EmptyPluginSettingsComponent extends PluginSettingsComponentBase {
560
558
  // Annotate the CommonJS export names for ESM import in node:
561
559
  0 && (module.exports = {
562
560
  EmptyPluginSettingsComponent,
563
- PLUGIN_SETTINGS_COMPONENT_KEY,
564
561
  PluginSettingsComponentBase
565
562
  });
566
- //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../../../../src/obsidian/plugin/components/plugin-settings-component.ts"],
  "sourcesContent": ["/**\n * @file\n *\n * Component that manages plugin settings: data, persistence, validation, and events.\n *\n * This is the single public-facing settings API for plugin authors.\n * All settings logic (load, save, validate, transform, clone) is handled directly\n * by this component \u2014 no separate manager class is needed.\n */\n\nimport type {\n  Promisable,\n  ReadonlyDeep\n} from 'type-fest';\n\nimport type { AsyncEventRef } from '../../../async-events.ts';\nimport type { Transformer } from '../../../transformers/transformer.ts';\nimport type { GenericObject } from '../../../type-guards.ts';\nimport type {\n  MaybeReturn,\n  StringKeys\n} from '../../../type.ts';\n\nimport { AsyncEvents } from '../../../async-events.ts';\nimport { getLibDebugger } from '../../../debug.ts';\nimport {\n  noop,\n  noopAsync\n} from '../../../function.ts';\nimport {\n  castTo,\n  deepEqual,\n  getAllKeys\n} from '../../../object-utils.ts';\nimport { DateTransformer } from '../../../transformers/date-transformer.ts';\nimport { DurationTransformer } from '../../../transformers/duration-transformer.ts';\nimport { GroupTransformer } from '../../../transformers/group-transformer.ts';\nimport { MapTransformer } from '../../../transformers/map-transformer.ts';\nimport { SetTransformer } from '../../../transformers/set-transformer.ts';\nimport { SkipPrivatePropertyTransformer } from '../../../transformers/skip-private-property-transformer.ts';\nimport { TwoWayMapTransformer } from '../../../transformers/two-way-map-transformer.ts';\nimport { AsyncComponentBase } from '../../components/async-component.ts';\nimport { registerAsyncEvent } from '../../components/async-events-component.ts';\n\nconst defaultTransformer = new GroupTransformer([\n  new SkipPrivatePropertyTransformer(),\n  new DateTransformer(),\n  new DurationTransformer(),\n  new MapTransformer(),\n  new SetTransformer(),\n  new TwoWayMapTransformer()\n]);\n\n/**\n * Component key used by {@link registerComponent} to identify settings components.\n */\nexport const PLUGIN_SETTINGS_COMPONENT_KEY = 'PluginSettingsComponent';\n\n/**\n * Params for creating a {@link PluginSettingsComponentBase}.\n */\nexport interface PluginSettingsComponentParams {\n  /**\n   * A function to load data from the plugin's data file.\n   *\n   * @returns The loaded data.\n   */\n  readonly loadData: () => Promise<unknown>;\n\n  /**\n   * A function to save data to the plugin's data file.\n   *\n   * @param data - The data to save.\n   */\n  readonly saveData: (data: unknown) => Promise<void>;\n}\n\n/**\n * A snapshot of plugin settings state, including raw input values, effective (validated) values,\n * and per-property validation messages.\n *\n * @typeParam PluginSettings - The type of the plugin settings.\n */\nexport interface PluginSettingsState<PluginSettings extends object> {\n  /**\n   * The effective settings values used by the plugin. Invalid properties are replaced with defaults.\n   */\n  effectiveValues: PluginSettings;\n\n  /**\n   * The raw input values as entered by the user. May contain invalid values.\n   */\n  inputValues: PluginSettings;\n\n  /**\n   * Per-property validation messages. Empty string means valid.\n   */\n  validationMessages: Record<StringKeys<PluginSettings>, string>;\n}\n\n/**\n * Readonly version of {@link PluginSettings}.\n *\n * @typeParam PluginSettings - The type of the plugin settings.\n */\nexport type ReadonlyPluginSettings<PluginSettings extends object> = ReadonlyDeep<PluginSettings>;\n\n/**\n * Readonly version of {@link PluginSettingsState} for use in event callbacks and public getters.\n *\n * @typeParam PluginSettings - The type of the plugin settings.\n */\nexport type ReadonlyPluginSettingsState<PluginSettings extends object> = ReadonlyDeep<PluginSettingsState<PluginSettings>>;\n\n/**\n * A validator function for a settings property.\n *\n * @typeParam PluginSettings - The plugin settings type.\n * @typeParam PropertyName - The property name.\n */\nexport type SettingsValidator<PluginSettings extends object, PropertyName extends StringKeys<PluginSettings> = StringKeys<PluginSettings>> = (\n  value: PluginSettings[PropertyName],\n  settings: PluginSettings\n) => Promisable<MaybeReturn<string>>;\n\ntype PropertyNames<PluginSettings extends object> = StringKeys<PluginSettings>;\n\ntype PropertyValues<PluginSettings extends object> = PluginSettings[PropertyNames<PluginSettings>];\n\ntype ValidationResult<PluginSettings extends object> = Partial<Record<StringKeys<PluginSettings>, string>>;\n\n/**\n * Base class for plugin settings components.\n *\n * Manages settings data, persistence, validation, and events.\n * Plugin authors extend this class and implement {@link createDefaultSettings}.\n *\n * @typeParam PluginSettings - The plugin settings type.\n */\nexport abstract class PluginSettingsComponentBase<PluginSettings extends object> extends AsyncComponentBase {\n  /**\n   * Component key for {@link registerComponent} replacement.\n   */\n  public static readonly COMPONENT_KEY = PLUGIN_SETTINGS_COMPONENT_KEY;\n\n  /**\n   * Gets the readonly default settings.\n   *\n   * @returns The default settings (as a readonly object).\n   */\n  public readonly defaultSettings: ReadonlyDeep<PluginSettings>;\n\n  /**\n   * Gets the readonly effective settings (validated, with defaults substituted for invalid values).\n   *\n   * @returns The readonly effective settings.\n   */\n  public get settings(): ReadonlyDeep<PluginSettings> {\n    return this.settingsState.effectiveValues as ReadonlyDeep<PluginSettings>;\n  }\n\n  /**\n   * Gets the current settings state snapshot.\n   *\n   * @returns The current settings state.\n   */\n  public get settingsState(): ReadonlyPluginSettingsState<PluginSettings> {\n    return this.currentState as ReadonlyPluginSettingsState<PluginSettings>;\n  }\n\n  private readonly asyncEvents = new AsyncEvents();\n  private currentState: PluginSettingsState<PluginSettings>;\n  private lastSavedState: PluginSettingsState<PluginSettings>;\n  private readonly legacySettingsConverters: ((record: GenericObject) => void)[] = [];\n  private readonly loadDataFn: () => Promise<unknown>;\n  private readonly propertyNames: PropertyNames<PluginSettings>[];\n  private readonly saveDataFn: (data: unknown) => Promise<void>;\n  private readonly validators = new Map<PropertyNames<PluginSettings>, SettingsValidator<PluginSettings>>();\n\n  /**\n   * Creates a new plugin settings component.\n   *\n   * @param params - The params for loading/saving data.\n   */\n  public constructor(params: PluginSettingsComponentParams) {\n    super();\n    this.loadDataFn = params.loadData.bind(params);\n    this.saveDataFn = params.saveData.bind(params);\n    this.defaultSettings = this.createDefaultSettings() as ReadonlyDeep<PluginSettings>;\n    this.currentState = this.createDefaultState();\n    this.lastSavedState = this.createDefaultState();\n    this.propertyNames = getAllKeys(this.currentState.inputValues);\n    this.registerValidators();\n    this.registerLegacySettingsConverters();\n  }\n\n  /**\n   * Edits the plugin settings and saves them.\n   *\n   * @param settingsEditor - The editor.\n   * @param context - The context.\n   * @returns A {@link Promise} that resolves when the settings are saved.\n   */\n  public async editAndSave(settingsEditor: (settings: PluginSettings) => Promisable<void>, context?: unknown): Promise<void> {\n    await this.edit(settingsEditor);\n    await this.saveToFile(context);\n  }\n\n  /**\n   * Ensures the settings are safe.\n   *\n   * It runs validation for each property and sets the default value if the validation fails.\n   *\n   * @param settings - The settings.\n   * @returns A {@link Promise} that resolves when the settings are safe.\n   */\n  public async ensureSafe(settings: PluginSettings): Promise<void> {\n    const validationResult = await this.validate(settings);\n    const defaults = this.defaultSettings as PluginSettings;\n    for (const propertyName of this.propertyNames) {\n      if (validationResult[propertyName]) {\n        settings[propertyName] = defaults[propertyName];\n      }\n    }\n  }\n\n  /**\n   * Gets a safe copy of the settings.\n   *\n   * @param settings - The settings.\n   * @returns A {@link Promise} that resolves to the safe copy of the settings.\n   */\n  public async getSafeCopy(settings: PluginSettings): Promise<PluginSettings> {\n    const safeSettings = await this.cloneSettings(settings);\n    await this.ensureSafe(safeSettings);\n    return safeSettings;\n  }\n\n  /**\n   * Loads the plugin settings from the file.\n   *\n   * @param isInitialLoad - Whether the settings are being loaded for the first time.\n   * @returns A {@link Promise} that resolves when the settings are loaded.\n   */\n  public async loadFromFile(isInitialLoad: boolean): Promise<void> {\n    const data = await this.loadDataFn();\n    this.lastSavedState = this.createDefaultState();\n    this.currentState = this.createDefaultState();\n\n    try {\n      if (data === undefined || data === null) {\n        return;\n      }\n\n      if (typeof data !== 'object') {\n        console.error(`Invalid settings from data.json. Expected Object, got: ${typeof data}`);\n        return;\n      }\n\n      const rawRecord = data as GenericObject;\n      const parsedSettings = await this.rawRecordToSettings(rawRecord);\n      const validationResult = await this.validate(parsedSettings);\n\n      for (const propertyName of this.propertyNames) {\n        this.setPropertyImpl(propertyName, parsedSettings[propertyName], validationResult[propertyName]);\n      }\n\n      this.lastSavedState = await this.cloneState(this.currentState);\n\n      const newRecord = await this.settingsToRawRecord(this.currentState.inputValues);\n\n      if (!deepEqual(newRecord, data)) {\n        await this.saveToFileImpl();\n      }\n    } finally {\n      await this.asyncEvents.triggerAsync('loadSettings', this.currentState, isInitialLoad);\n    }\n  }\n\n  /**\n   * Subscribes to the `loadSettings` event.\n   *\n   * @param name - Always `loadSettings`.\n   * @param callback - The callback.\n   * @param thisArg - The `this` context.\n   * @returns A reference to the event listener.\n   */\n  public on(\n    name: 'loadSettings',\n    callback: (\n      loadedState: ReadonlyPluginSettingsState<PluginSettings>,\n      isInitialLoad: boolean\n    ) => Promisable<void>,\n    thisArg?: unknown\n  ): AsyncEventRef;\n  /**\n   * Subscribes to the `saveSettings` event.\n   *\n   * @param name - Always `saveSettings`.\n   * @param callback - The callback.\n   * @param thisArg - The `this` context.\n   * @returns A reference to the event listener.\n   */\n  public on(\n    name: 'saveSettings',\n    callback: (\n      newState: ReadonlyPluginSettingsState<PluginSettings>,\n      oldState: ReadonlyPluginSettingsState<PluginSettings>,\n      context: unknown\n    ) => Promisable<void>,\n    thisArg?: unknown\n  ): AsyncEventRef;\n  /**\n   * Subscribes to an event.\n   *\n   * @param name - The name of the event.\n   * @param callback - The callback.\n   * @param thisArg - The `this` context.\n   * @returns A reference to the event listener.\n   */\n  public on<Args extends unknown[]>(\n    name: string,\n    callback: (...args: Args) => Promisable<void>,\n    thisArg?: unknown\n  ): AsyncEventRef {\n    return this.asyncEvents.on(name, callback, thisArg);\n  }\n\n  /**\n   * Called when the external settings change.\n   *\n   * @returns A {@link Promise} that resolves when the settings are reloaded.\n   */\n  public async onExternalSettingsChange(): Promise<void> {\n    await this.loadFromFile(false);\n  }\n\n  /**\n   * Loads settings from file and registers event handlers.\n   */\n  public override async onload(): Promise<void> {\n    registerAsyncEvent(this, this.on('loadSettings', this.onLoadSettings.bind(this)));\n    registerAsyncEvent(this, this.on('saveSettings', this.onSaveSettings.bind(this)));\n    await this.loadFromFile(true);\n  }\n\n  /**\n   * Registers a legacy settings converter.\n   *\n   * @typeParam LegacySettings - The legacy settings class.\n   * @param legacySettingsClass - The legacy settings class.\n   * @param converter - The converter.\n   */\n  public registerLegacySettingsConverter<LegacySettings extends object>(\n    legacySettingsClass: new () => LegacySettings,\n    converter: (legacySettings: Partial<LegacySettings> & Partial<PluginSettings>) => void\n  ): void {\n    const that = this;\n    this.legacySettingsConverters.push(legacySettingsConverter);\n\n    function legacySettingsConverter(record: GenericObject): void {\n      const legacySettingsKeys = new Set<string>(Object.keys(new legacySettingsClass()));\n      const pluginSettingKeys = new Set<string>(that.propertyNames);\n      const legacySettings = record as Partial<LegacySettings> & Partial<PluginSettings>;\n      converter(legacySettings);\n      for (const key of Object.keys(legacySettings)) {\n        if (pluginSettingKeys.has(key)) {\n          continue;\n        }\n\n        if (!legacySettingsKeys.has(key)) {\n          continue;\n        }\n\n        // eslint-disable-next-line @typescript-eslint/no-dynamic-delete -- We have no other way to delete the property.\n        delete record[key];\n      }\n    }\n  }\n\n  /**\n   * Registers a validator for a property.\n   *\n   * @param propertyName - The name of the property.\n   * @param validator - The validator.\n   */\n  public registerValidator<PropertyName extends PropertyNames<PluginSettings>>(\n    propertyName: PropertyName,\n    validator: SettingsValidator<PluginSettings, PropertyName>\n  ): void {\n    this.validators.set(propertyName, validator as SettingsValidator<PluginSettings>);\n  }\n\n  /**\n   * Revalidates the settings.\n   *\n   * @returns The validation messages.\n   */\n  public async revalidate(): Promise<Record<PropertyNames<PluginSettings>, string>> {\n    await this.edit(noop);\n    return this.currentState.validationMessages;\n  }\n\n  /**\n   * Saves the new plugin settings.\n   *\n   * @param context - The context of the save to file operation.\n   * @returns A {@link Promise} that resolves when the settings are saved.\n   */\n  public async saveToFile(context?: unknown): Promise<void> {\n    if (deepEqual(this.lastSavedState.inputValues, this.currentState.inputValues)) {\n      return;\n    }\n\n    await this.saveToFileImpl();\n    await this.asyncEvents.triggerAsync('saveSettings', this.currentState, this.lastSavedState, context);\n    this.lastSavedState = await this.cloneState(this.currentState);\n  }\n\n  /**\n   * Sets the value of a property.\n   *\n   * @typeParam PropertyName - The name of the property.\n   * @param propertyName - The name of the property.\n   * @param value - The value to set.\n   * @returns A {@link Promise} that resolves to the validation message.\n   */\n  public async setProperty<PropertyName extends PropertyNames<PluginSettings>>(\n    propertyName: PropertyName,\n    value: PluginSettings[PropertyName]\n  ): Promise<string> {\n    await this.edit((settings) => {\n      settings[propertyName] = value;\n    });\n    return this.currentState.validationMessages[propertyName];\n  }\n\n  /**\n   * Validates the settings.\n   *\n   * @param settings - The settings.\n   * @returns A {@link Promise} that resolves to the validation result.\n   */\n  public async validate(settings: PluginSettings): Promise<ValidationResult<PluginSettings>> {\n    const result: ValidationResult<PluginSettings> = {};\n    for (const [propertyName, validator] of this.validators.entries()) {\n      const validationMessage = await validator(settings[propertyName], settings);\n      if (validationMessage) {\n        result[propertyName] = validationMessage;\n      }\n    }\n\n    return result;\n  }\n\n  /**\n   * Creates the default settings. Must be implemented by subclasses.\n   *\n   * @returns The default settings.\n   */\n  protected abstract createDefaultSettings(): PluginSettings;\n\n  /**\n   * Gets the transformer.\n   *\n   * @returns The transformer.\n   */\n  protected getTransformer(): Transformer {\n    return defaultTransformer;\n  }\n\n  /**\n   * Called when the plugin settings record is loaded from disk.\n   *\n   * @param record - The record.\n   */\n  protected async onLoadRecord(record: GenericObject): Promise<void> {\n    for (const converter of this.legacySettingsConverters) {\n      converter(record);\n    }\n    await Promise.resolve();\n  }\n\n  /**\n   * Called when settings are loaded or reloaded.\n   *\n   * @param _loadedState - The loaded settings state.\n   * @param _isInitialLoad - Whether this is the initial load.\n   */\n  protected async onLoadSettings(\n    _loadedState: ReadonlyPluginSettingsState<PluginSettings>,\n    _isInitialLoad: boolean\n  ): Promise<void> {\n    await noopAsync();\n  }\n\n  /**\n   * Called when settings are saved.\n   *\n   * @param _newState - The new settings state.\n   * @param _oldState - The old settings state.\n   * @param _context - The save context.\n   */\n  protected async onSaveSettings(\n    _newState: ReadonlyPluginSettingsState<PluginSettings>,\n    _oldState: ReadonlyPluginSettingsState<PluginSettings>,\n    _context: unknown\n  ): Promise<void> {\n    await noopAsync();\n  }\n\n  /**\n   * Called when the plugin settings record is about to be saved to disk.\n   *\n   * @param _record - The record.\n   */\n  protected async onSavingRecord(_record: GenericObject): Promise<void> {\n    await noopAsync();\n  }\n\n  /**\n   * Registers the legacy settings converters.\n   * Override to register legacy settings converters.\n   */\n  protected registerLegacySettingsConverters(): void {\n    noop();\n  }\n\n  /**\n   * Registers the validators.\n   * Override to register validators for properties.\n   */\n  protected registerValidators(): void {\n    noop();\n  }\n\n  private async cloneSettings(settings: PluginSettings): Promise<PluginSettings> {\n    const record = await this.settingsToRawRecord(settings);\n    const json = JSON.stringify(record);\n    const cloneRecord = JSON.parse(json) as GenericObject;\n    return await this.rawRecordToSettings(cloneRecord);\n  }\n\n  private async cloneState(\n    state: PluginSettingsState<PluginSettings>\n  ): Promise<PluginSettingsState<PluginSettings>> {\n    return {\n      effectiveValues: await this.cloneSettings(state.effectiveValues),\n      inputValues: await this.cloneSettings(state.inputValues),\n      validationMessages: { ...state.validationMessages }\n    };\n  }\n\n  private createDefaultState(): PluginSettingsState<PluginSettings> {\n    return {\n      effectiveValues: this.createDefaultSettings(),\n      inputValues: this.createDefaultSettings(),\n      validationMessages: castTo<Record<PropertyNames<PluginSettings>, string>>({})\n    };\n  }\n\n  private async edit(settingsEditor: (settings: PluginSettings) => Promisable<void>): Promise<void> {\n    try {\n      await settingsEditor(this.currentState.inputValues);\n    } finally {\n      const validationResult = await this.validate(this.currentState.inputValues);\n      for (const propertyName of this.propertyNames) {\n        const validationMessage = validationResult[propertyName] ?? '';\n        this.currentState.validationMessages[propertyName] = validationMessage;\n        const defaults = this.defaultSettings as PluginSettings;\n        this.currentState.effectiveValues[propertyName] = validationMessage\n          ? defaults[propertyName]\n          : this.currentState.inputValues[propertyName];\n      }\n    }\n  }\n\n  private isValidPropertyName(prop: unknown): prop is PropertyNames<PluginSettings> {\n    if (typeof prop !== 'string') {\n      return false;\n    }\n\n    return (this.propertyNames as string[]).includes(prop);\n  }\n\n  private async rawRecordToSettings(rawRecord: GenericObject): Promise<PluginSettings> {\n    rawRecord = this.getTransformer().transformObjectRecursively(rawRecord);\n    await this.onLoadRecord(rawRecord);\n\n    const settings = this.createDefaultSettings();\n    const defaults = this.defaultSettings as PluginSettings;\n\n    for (const [propertyName, value] of Object.entries(rawRecord)) {\n      if (!this.isValidPropertyName(propertyName)) {\n        getLibDebugger('PluginSettingsComponentBase:rawRecordToSettings')(`Unknown property: ${propertyName}`);\n        continue;\n      }\n\n      if (typeof value !== typeof defaults[propertyName]) {\n        getLibDebugger('PluginSettingsComponentBase:rawRecordToSettings')(\n          'Possible invalid value type. It might lead to an unexpected behavior of the plugin. There is also a chance it is a false-negative warning, as we are unable to determine the exact type of the value in runtime.',\n          {\n            defaultValue: defaults[propertyName],\n            propertyName,\n            value\n          }\n        );\n      }\n\n      settings[propertyName] = value as PropertyValues<PluginSettings>;\n    }\n\n    return settings;\n  }\n\n  private async saveToFileImpl(): Promise<void> {\n    await this.saveDataFn(await this.settingsToRawRecord(this.currentState.inputValues));\n  }\n\n  private setPropertyImpl(\n    propertyName: PropertyNames<PluginSettings>,\n    value: PropertyValues<PluginSettings>,\n    validationMessage?: string\n  ): void {\n    const defaults = this.defaultSettings as PluginSettings;\n    this.currentState.inputValues[propertyName] = value;\n    this.currentState.validationMessages[propertyName] = validationMessage ?? '';\n    this.currentState.effectiveValues[propertyName] = validationMessage ? defaults[propertyName] : value;\n  }\n\n  private async settingsToRawRecord(settings: PluginSettings): Promise<GenericObject> {\n    const rawRecord: GenericObject = {};\n\n    for (const propertyName of this.propertyNames) {\n      rawRecord[propertyName] = settings[propertyName];\n    }\n\n    await this.onSavingRecord(rawRecord);\n\n    return this.getTransformer().transformObjectRecursively(rawRecord);\n  }\n}\n\n/**\n * A no-op settings component for plugins without settings.\n */\nexport class EmptyPluginSettingsComponent extends PluginSettingsComponentBase<object> {\n  /**\n   * Creates a new empty plugin settings component.\n   *\n   * @param params - The params. Uses no-op load/save by default.\n   */\n  public constructor(params?: PluginSettingsComponentParams) {\n    super(\n      params ?? {\n        loadData: async (): Promise<object> => {\n          await noopAsync();\n          return {};\n        },\n        saveData: noopAsync\n      }\n    );\n  }\n\n  /**\n   * Creates empty default settings.\n   *\n   * @returns An empty object.\n   */\n  protected override createDefaultSettings(): object {\n    return {};\n  }\n}\n"],
  "mappings": "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",
  "names": []
}

563
+ //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../../../../src/obsidian/plugin/components/plugin-settings-component.ts"],
  "sourcesContent": ["/**\n * @file\n *\n * Component that manages plugin settings: data, persistence, validation, and events.\n *\n * This is the single public-facing settings API for plugin authors.\n * All settings logic (load, save, validate, transform, clone) is handled directly\n * by this component \u2014 no separate manager class is needed.\n */\n\nimport type {\n  Promisable,\n  ReadonlyDeep\n} from 'type-fest';\n\nimport type { AsyncEventRef } from '../../../async-events.ts';\nimport type { Transformer } from '../../../transformers/transformer.ts';\nimport type { GenericObject } from '../../../type-guards.ts';\nimport type {\n  MaybeReturn,\n  StringKeys\n} from '../../../type.ts';\n\nimport { AsyncEvents } from '../../../async-events.ts';\nimport { getLibDebugger } from '../../../debug.ts';\nimport {\n  noop,\n  noopAsync\n} from '../../../function.ts';\nimport {\n  castTo,\n  deepEqual,\n  getAllKeys\n} from '../../../object-utils.ts';\nimport { DateTransformer } from '../../../transformers/date-transformer.ts';\nimport { DurationTransformer } from '../../../transformers/duration-transformer.ts';\nimport { GroupTransformer } from '../../../transformers/group-transformer.ts';\nimport { MapTransformer } from '../../../transformers/map-transformer.ts';\nimport { SetTransformer } from '../../../transformers/set-transformer.ts';\nimport { SkipPrivatePropertyTransformer } from '../../../transformers/skip-private-property-transformer.ts';\nimport { TwoWayMapTransformer } from '../../../transformers/two-way-map-transformer.ts';\nimport { AsyncComponentBase } from '../../components/async-component.ts';\nimport { registerAsyncEvent } from '../../components/async-events-component.ts';\n\nconst defaultTransformer = new GroupTransformer([\n  new SkipPrivatePropertyTransformer(),\n  new DateTransformer(),\n  new DurationTransformer(),\n  new MapTransformer(),\n  new SetTransformer(),\n  new TwoWayMapTransformer()\n]);\n\n/**\n * Params for creating a {@link PluginSettingsComponentBase}.\n */\nexport interface PluginSettingsComponentParams {\n  /**\n   * A function to load data from the plugin's data file.\n   *\n   * @returns The loaded data.\n   */\n  readonly loadData: () => Promise<unknown>;\n\n  /**\n   * A function to save data to the plugin's data file.\n   *\n   * @param data - The data to save.\n   */\n  readonly saveData: (data: unknown) => Promise<void>;\n}\n\n/**\n * A snapshot of plugin settings state, including raw input values, effective (validated) values,\n * and per-property validation messages.\n *\n * @typeParam PluginSettings - The type of the plugin settings.\n */\nexport interface PluginSettingsState<PluginSettings extends object> {\n  /**\n   * The effective settings values used by the plugin. Invalid properties are replaced with defaults.\n   */\n  effectiveValues: PluginSettings;\n\n  /**\n   * The raw input values as entered by the user. May contain invalid values.\n   */\n  inputValues: PluginSettings;\n\n  /**\n   * Per-property validation messages. Empty string means valid.\n   */\n  validationMessages: Record<StringKeys<PluginSettings>, string>;\n}\n\n/**\n * Readonly version of {@link PluginSettings}.\n *\n * @typeParam PluginSettings - The type of the plugin settings.\n */\nexport type ReadonlyPluginSettings<PluginSettings extends object> = ReadonlyDeep<PluginSettings>;\n\n/**\n * Readonly version of {@link PluginSettingsState} for use in event callbacks and public getters.\n *\n * @typeParam PluginSettings - The type of the plugin settings.\n */\nexport type ReadonlyPluginSettingsState<PluginSettings extends object> = ReadonlyDeep<PluginSettingsState<PluginSettings>>;\n\n/**\n * A validator function for a settings property.\n *\n * @typeParam PluginSettings - The plugin settings type.\n * @typeParam PropertyName - The property name.\n */\nexport type SettingsValidator<PluginSettings extends object, PropertyName extends StringKeys<PluginSettings> = StringKeys<PluginSettings>> = (\n  value: PluginSettings[PropertyName],\n  settings: PluginSettings\n) => Promisable<MaybeReturn<string>>;\n\ntype PropertyNames<PluginSettings extends object> = StringKeys<PluginSettings>;\n\ntype PropertyValues<PluginSettings extends object> = PluginSettings[PropertyNames<PluginSettings>];\n\ntype ValidationResult<PluginSettings extends object> = Partial<Record<StringKeys<PluginSettings>, string>>;\n\n/**\n * Base class for plugin settings components.\n *\n * Manages settings data, persistence, validation, and events.\n * Plugin authors extend this class and implement {@link createDefaultSettings}.\n *\n * @typeParam PluginSettings - The plugin settings type.\n */\nexport abstract class PluginSettingsComponentBase<PluginSettings extends object> extends AsyncComponentBase {\n  /**\n   * Component key for {@link registerComponent} replacement.\n   */\n  public static readonly COMPONENT_KEY = Symbol(PluginSettingsComponentBase.name);\n\n  /**\n   * Gets the readonly default settings.\n   *\n   * @returns The default settings (as a readonly object).\n   */\n  public readonly defaultSettings: ReadonlyDeep<PluginSettings>;\n\n  /**\n   * Gets the readonly effective settings (validated, with defaults substituted for invalid values).\n   *\n   * @returns The readonly effective settings.\n   */\n  public get settings(): ReadonlyDeep<PluginSettings> {\n    return this.settingsState.effectiveValues;\n  }\n\n  /**\n   * Gets the current settings state snapshot.\n   *\n   * @returns The current settings state.\n   */\n  public get settingsState(): ReadonlyPluginSettingsState<PluginSettings> {\n    return this.currentState as ReadonlyPluginSettingsState<PluginSettings>;\n  }\n\n  private readonly asyncEvents = new AsyncEvents();\n  private currentState: PluginSettingsState<PluginSettings>;\n  private lastSavedState: PluginSettingsState<PluginSettings>;\n  private readonly legacySettingsConverters: ((record: GenericObject) => void)[] = [];\n  private readonly loadDataFn: () => Promise<unknown>;\n  private readonly propertyNames: PropertyNames<PluginSettings>[];\n  private readonly saveDataFn: (data: unknown) => Promise<void>;\n  private readonly validators = new Map<PropertyNames<PluginSettings>, SettingsValidator<PluginSettings>>();\n\n  /**\n   * Creates a new plugin settings component.\n   *\n   * @param params - The params for loading/saving data.\n   */\n  public constructor(params: PluginSettingsComponentParams) {\n    super();\n    this.loadDataFn = params.loadData.bind(params);\n    this.saveDataFn = params.saveData.bind(params);\n    this.defaultSettings = this.createDefaultSettings() as ReadonlyDeep<PluginSettings>;\n    this.currentState = this.createDefaultState();\n    this.lastSavedState = this.createDefaultState();\n    this.propertyNames = getAllKeys(this.currentState.inputValues);\n    this.registerValidators();\n    this.registerLegacySettingsConverters();\n  }\n\n  /**\n   * Edits the plugin settings and saves them.\n   *\n   * @param settingsEditor - The editor.\n   * @param context - The context.\n   * @returns A {@link Promise} that resolves when the settings are saved.\n   */\n  public async editAndSave(settingsEditor: (settings: PluginSettings) => Promisable<void>, context?: unknown): Promise<void> {\n    await this.edit(settingsEditor);\n    await this.saveToFile(context);\n  }\n\n  /**\n   * Ensures the settings are safe.\n   *\n   * It runs validation for each property and sets the default value if the validation fails.\n   *\n   * @param settings - The settings.\n   * @returns A {@link Promise} that resolves when the settings are safe.\n   */\n  public async ensureSafe(settings: PluginSettings): Promise<void> {\n    const validationResult = await this.validate(settings);\n    const defaults = this.defaultSettings as PluginSettings;\n    for (const propertyName of this.propertyNames) {\n      if (validationResult[propertyName]) {\n        settings[propertyName] = defaults[propertyName];\n      }\n    }\n  }\n\n  /**\n   * Gets a safe copy of the settings.\n   *\n   * @param settings - The settings.\n   * @returns A {@link Promise} that resolves to the safe copy of the settings.\n   */\n  public async getSafeCopy(settings: PluginSettings): Promise<PluginSettings> {\n    const safeSettings = await this.cloneSettings(settings);\n    await this.ensureSafe(safeSettings);\n    return safeSettings;\n  }\n\n  /**\n   * Loads the plugin settings from the file.\n   *\n   * @param isInitialLoad - Whether the settings are being loaded for the first time.\n   * @returns A {@link Promise} that resolves when the settings are loaded.\n   */\n  public async loadFromFile(isInitialLoad: boolean): Promise<void> {\n    const data = await this.loadDataFn();\n    this.lastSavedState = this.createDefaultState();\n    this.currentState = this.createDefaultState();\n\n    try {\n      if (data === undefined || data === null) {\n        return;\n      }\n\n      if (typeof data !== 'object') {\n        console.error(`Invalid settings from data.json. Expected Object, got: ${typeof data}`);\n        return;\n      }\n\n      const rawRecord = data as GenericObject;\n      const parsedSettings = await this.rawRecordToSettings(rawRecord);\n      const validationResult = await this.validate(parsedSettings);\n\n      for (const propertyName of this.propertyNames) {\n        this.setPropertyImpl(propertyName, parsedSettings[propertyName], validationResult[propertyName]);\n      }\n\n      this.lastSavedState = await this.cloneState(this.currentState);\n\n      const newRecord = await this.settingsToRawRecord(this.currentState.inputValues);\n\n      if (!deepEqual(newRecord, data)) {\n        await this.saveToFileImpl();\n      }\n    } finally {\n      await this.asyncEvents.triggerAsync('loadSettings', this.currentState, isInitialLoad);\n    }\n  }\n\n  /**\n   * Subscribes to the `loadSettings` event.\n   *\n   * @param name - Always `loadSettings`.\n   * @param callback - The callback.\n   * @param thisArg - The `this` context.\n   * @returns A reference to the event listener.\n   */\n  public on(\n    name: 'loadSettings',\n    callback: (\n      loadedState: ReadonlyPluginSettingsState<PluginSettings>,\n      isInitialLoad: boolean\n    ) => Promisable<void>,\n    thisArg?: unknown\n  ): AsyncEventRef;\n  /**\n   * Subscribes to the `saveSettings` event.\n   *\n   * @param name - Always `saveSettings`.\n   * @param callback - The callback.\n   * @param thisArg - The `this` context.\n   * @returns A reference to the event listener.\n   */\n  public on(\n    name: 'saveSettings',\n    callback: (\n      newState: ReadonlyPluginSettingsState<PluginSettings>,\n      oldState: ReadonlyPluginSettingsState<PluginSettings>,\n      context: unknown\n    ) => Promisable<void>,\n    thisArg?: unknown\n  ): AsyncEventRef;\n  /**\n   * Subscribes to an event.\n   *\n   * @param name - The name of the event.\n   * @param callback - The callback.\n   * @param thisArg - The `this` context.\n   * @returns A reference to the event listener.\n   */\n  public on<Args extends unknown[]>(\n    name: string,\n    callback: (...args: Args) => Promisable<void>,\n    thisArg?: unknown\n  ): AsyncEventRef {\n    return this.asyncEvents.on(name, callback, thisArg);\n  }\n\n  /**\n   * Called when the external settings change.\n   *\n   * @returns A {@link Promise} that resolves when the settings are reloaded.\n   */\n  public async onExternalSettingsChange(): Promise<void> {\n    await this.loadFromFile(false);\n  }\n\n  /**\n   * Loads settings from file and registers event handlers.\n   */\n  public override async onload(): Promise<void> {\n    registerAsyncEvent(this, this.on('loadSettings', this.onLoadSettings.bind(this)));\n    registerAsyncEvent(this, this.on('saveSettings', this.onSaveSettings.bind(this)));\n    await this.loadFromFile(true);\n  }\n\n  /**\n   * Registers a legacy settings converter.\n   *\n   * @typeParam LegacySettings - The legacy settings class.\n   * @param legacySettingsClass - The legacy settings class.\n   * @param converter - The converter.\n   */\n  public registerLegacySettingsConverter<LegacySettings extends object>(\n    legacySettingsClass: new () => LegacySettings,\n    converter: (legacySettings: Partial<LegacySettings> & Partial<PluginSettings>) => void\n  ): void {\n    const that = this;\n    this.legacySettingsConverters.push(legacySettingsConverter);\n\n    function legacySettingsConverter(record: GenericObject): void {\n      const legacySettingsKeys = new Set<string>(Object.keys(new legacySettingsClass()));\n      const pluginSettingKeys = new Set<string>(that.propertyNames);\n      const legacySettings = record as Partial<LegacySettings> & Partial<PluginSettings>;\n      converter(legacySettings);\n      for (const key of Object.keys(legacySettings)) {\n        if (pluginSettingKeys.has(key)) {\n          continue;\n        }\n\n        if (!legacySettingsKeys.has(key)) {\n          continue;\n        }\n\n        // eslint-disable-next-line @typescript-eslint/no-dynamic-delete -- We have no other way to delete the property.\n        delete record[key];\n      }\n    }\n  }\n\n  /**\n   * Registers a validator for a property.\n   *\n   * @param propertyName - The name of the property.\n   * @param validator - The validator.\n   */\n  public registerValidator<PropertyName extends PropertyNames<PluginSettings>>(\n    propertyName: PropertyName,\n    validator: SettingsValidator<PluginSettings, PropertyName>\n  ): void {\n    this.validators.set(propertyName, validator as SettingsValidator<PluginSettings>);\n  }\n\n  /**\n   * Revalidates the settings.\n   *\n   * @returns The validation messages.\n   */\n  public async revalidate(): Promise<Record<PropertyNames<PluginSettings>, string>> {\n    await this.edit(noop);\n    return this.currentState.validationMessages;\n  }\n\n  /**\n   * Saves the new plugin settings.\n   *\n   * @param context - The context of the save to file operation.\n   * @returns A {@link Promise} that resolves when the settings are saved.\n   */\n  public async saveToFile(context?: unknown): Promise<void> {\n    if (deepEqual(this.lastSavedState.inputValues, this.currentState.inputValues)) {\n      return;\n    }\n\n    await this.saveToFileImpl();\n    await this.asyncEvents.triggerAsync('saveSettings', this.currentState, this.lastSavedState, context);\n    this.lastSavedState = await this.cloneState(this.currentState);\n  }\n\n  /**\n   * Sets the value of a property.\n   *\n   * @typeParam PropertyName - The name of the property.\n   * @param propertyName - The name of the property.\n   * @param value - The value to set.\n   * @returns A {@link Promise} that resolves to the validation message.\n   */\n  public async setProperty<PropertyName extends PropertyNames<PluginSettings>>(\n    propertyName: PropertyName,\n    value: PluginSettings[PropertyName]\n  ): Promise<string> {\n    await this.edit((settings) => {\n      settings[propertyName] = value;\n    });\n    return this.currentState.validationMessages[propertyName];\n  }\n\n  /**\n   * Validates the settings.\n   *\n   * @param settings - The settings.\n   * @returns A {@link Promise} that resolves to the validation result.\n   */\n  public async validate(settings: PluginSettings): Promise<ValidationResult<PluginSettings>> {\n    const result: ValidationResult<PluginSettings> = {};\n    for (const [propertyName, validator] of this.validators.entries()) {\n      const validationMessage = await validator(settings[propertyName], settings);\n      if (validationMessage) {\n        result[propertyName] = validationMessage;\n      }\n    }\n\n    return result;\n  }\n\n  /**\n   * Creates the default settings. Must be implemented by subclasses.\n   *\n   * @returns The default settings.\n   */\n  protected abstract createDefaultSettings(): PluginSettings;\n\n  /**\n   * Gets the transformer.\n   *\n   * @returns The transformer.\n   */\n  protected getTransformer(): Transformer {\n    return defaultTransformer;\n  }\n\n  /**\n   * Called when the plugin settings record is loaded from disk.\n   *\n   * @param record - The record.\n   */\n  protected async onLoadRecord(record: GenericObject): Promise<void> {\n    for (const converter of this.legacySettingsConverters) {\n      converter(record);\n    }\n    await Promise.resolve();\n  }\n\n  /**\n   * Called when settings are loaded or reloaded.\n   *\n   * @param _loadedState - The loaded settings state.\n   * @param _isInitialLoad - Whether this is the initial load.\n   */\n  protected async onLoadSettings(\n    _loadedState: ReadonlyPluginSettingsState<PluginSettings>,\n    _isInitialLoad: boolean\n  ): Promise<void> {\n    await noopAsync();\n  }\n\n  /**\n   * Called when settings are saved.\n   *\n   * @param _newState - The new settings state.\n   * @param _oldState - The old settings state.\n   * @param _context - The save context.\n   */\n  protected async onSaveSettings(\n    _newState: ReadonlyPluginSettingsState<PluginSettings>,\n    _oldState: ReadonlyPluginSettingsState<PluginSettings>,\n    _context: unknown\n  ): Promise<void> {\n    await noopAsync();\n  }\n\n  /**\n   * Called when the plugin settings record is about to be saved to disk.\n   *\n   * @param _record - The record.\n   */\n  protected async onSavingRecord(_record: GenericObject): Promise<void> {\n    await noopAsync();\n  }\n\n  /**\n   * Registers the legacy settings converters.\n   * Override to register legacy settings converters.\n   */\n  protected registerLegacySettingsConverters(): void {\n    noop();\n  }\n\n  /**\n   * Registers the validators.\n   * Override to register validators for properties.\n   */\n  protected registerValidators(): void {\n    noop();\n  }\n\n  private async cloneSettings(settings: PluginSettings): Promise<PluginSettings> {\n    const record = await this.settingsToRawRecord(settings);\n    const json = JSON.stringify(record);\n    const cloneRecord = JSON.parse(json) as GenericObject;\n    return await this.rawRecordToSettings(cloneRecord);\n  }\n\n  private async cloneState(\n    state: PluginSettingsState<PluginSettings>\n  ): Promise<PluginSettingsState<PluginSettings>> {\n    return {\n      effectiveValues: await this.cloneSettings(state.effectiveValues),\n      inputValues: await this.cloneSettings(state.inputValues),\n      validationMessages: { ...state.validationMessages }\n    };\n  }\n\n  private createDefaultState(): PluginSettingsState<PluginSettings> {\n    return {\n      effectiveValues: this.createDefaultSettings(),\n      inputValues: this.createDefaultSettings(),\n      validationMessages: castTo<Record<PropertyNames<PluginSettings>, string>>({})\n    };\n  }\n\n  private async edit(settingsEditor: (settings: PluginSettings) => Promisable<void>): Promise<void> {\n    try {\n      await settingsEditor(this.currentState.inputValues);\n    } finally {\n      const validationResult = await this.validate(this.currentState.inputValues);\n      for (const propertyName of this.propertyNames) {\n        const validationMessage = validationResult[propertyName] ?? '';\n        this.currentState.validationMessages[propertyName] = validationMessage;\n        const defaults = this.defaultSettings as PluginSettings;\n        this.currentState.effectiveValues[propertyName] = validationMessage\n          ? defaults[propertyName]\n          : this.currentState.inputValues[propertyName];\n      }\n    }\n  }\n\n  private isValidPropertyName(prop: unknown): prop is PropertyNames<PluginSettings> {\n    if (typeof prop !== 'string') {\n      return false;\n    }\n\n    return (this.propertyNames as string[]).includes(prop);\n  }\n\n  private async rawRecordToSettings(rawRecord: GenericObject): Promise<PluginSettings> {\n    rawRecord = this.getTransformer().transformObjectRecursively(rawRecord);\n    await this.onLoadRecord(rawRecord);\n\n    const settings = this.createDefaultSettings();\n    const defaults = this.defaultSettings as PluginSettings;\n\n    for (const [propertyName, value] of Object.entries(rawRecord)) {\n      if (!this.isValidPropertyName(propertyName)) {\n        getLibDebugger('PluginSettingsComponentBase:rawRecordToSettings')(`Unknown property: ${propertyName}`);\n        continue;\n      }\n\n      if (typeof value !== typeof defaults[propertyName]) {\n        getLibDebugger('PluginSettingsComponentBase:rawRecordToSettings')(\n          'Possible invalid value type. It might lead to an unexpected behavior of the plugin. There is also a chance it is a false-negative warning, as we are unable to determine the exact type of the value in runtime.',\n          {\n            defaultValue: defaults[propertyName],\n            propertyName,\n            value\n          }\n        );\n      }\n\n      settings[propertyName] = value as PropertyValues<PluginSettings>;\n    }\n\n    return settings;\n  }\n\n  private async saveToFileImpl(): Promise<void> {\n    await this.saveDataFn(await this.settingsToRawRecord(this.currentState.inputValues));\n  }\n\n  private setPropertyImpl(\n    propertyName: PropertyNames<PluginSettings>,\n    value: PropertyValues<PluginSettings>,\n    validationMessage?: string\n  ): void {\n    const defaults = this.defaultSettings as PluginSettings;\n    this.currentState.inputValues[propertyName] = value;\n    this.currentState.validationMessages[propertyName] = validationMessage ?? '';\n    this.currentState.effectiveValues[propertyName] = validationMessage ? defaults[propertyName] : value;\n  }\n\n  private async settingsToRawRecord(settings: PluginSettings): Promise<GenericObject> {\n    const rawRecord: GenericObject = {};\n\n    for (const propertyName of this.propertyNames) {\n      rawRecord[propertyName] = settings[propertyName];\n    }\n\n    await this.onSavingRecord(rawRecord);\n\n    return this.getTransformer().transformObjectRecursively(rawRecord);\n  }\n}\n\n/**\n * A no-op settings component for plugins without settings.\n */\nexport class EmptyPluginSettingsComponent extends PluginSettingsComponentBase<object> {\n  /**\n   * Creates a new empty plugin settings component.\n   *\n   * @param params - The params. Uses no-op load/save by default.\n   */\n  public constructor(params?: PluginSettingsComponentParams) {\n    super(\n      params ?? {\n        loadData: async (): Promise<object> => {\n          await noopAsync();\n          return {};\n        },\n        saveData: noopAsync\n      }\n    );\n  }\n\n  /**\n   * Creates empty default settings.\n   *\n   * @returns An empty object.\n   */\n  protected override createDefaultSettings(): object {\n    return {};\n  }\n}\n"],
  "mappings": ";;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAuBA,0BAA4B;AAC5B,mBAA+B;AAC/B,sBAGO;AACP,0BAIO;AACP,8BAAgC;AAChC,kCAAoC;AACpC,+BAAiC;AACjC,6BAA+B;AAC/B,6BAA+B;AAC/B,+CAA+C;AAC/C,qCAAqC;AACrC,6BAAmC;AACnC,oCAAmC;AAEnC,MAAM,qBAAqB,IAAI,0CAAiB;AAAA,EAC9C,IAAI,wEAA+B;AAAA,EACnC,IAAI,wCAAgB;AAAA,EACpB,IAAI,gDAAoB;AAAA,EACxB,IAAI,sCAAe;AAAA,EACnB,IAAI,sCAAe;AAAA,EACnB,IAAI,oDAAqB;AAC3B,CAAC;AAmFM,MAAe,oCAAmE,0CAAmB;AAAA;AAAA;AAAA;AAAA,EAI1G,OAAuB,gBAAgB,OAAO,4BAA4B,IAAI;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAO9D;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAOhB,IAAW,WAAyC;AAClD,WAAO,KAAK,cAAc;AAAA,EAC5B;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAOA,IAAW,gBAA6D;AACtE,WAAO,KAAK;AAAA,EACd;AAAA,EAEiB,cAAc,IAAI,gCAAY;AAAA,EACvC;AAAA,EACA;AAAA,EACS,2BAAgE,CAAC;AAAA,EACjE;AAAA,EACA;AAAA,EACA;AAAA,EACA,aAAa,oBAAI,IAAsE;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAOjG,YAAY,QAAuC;AACxD,UAAM;AACN,SAAK,aAAa,OAAO,SAAS,KAAK,MAAM;AAC7C,SAAK,aAAa,OAAO,SAAS,KAAK,MAAM;AAC7C,SAAK,kBAAkB,KAAK,sBAAsB;AAClD,SAAK,eAAe,KAAK,mBAAmB;AAC5C,SAAK,iBAAiB,KAAK,mBAAmB;AAC9C,SAAK,oBAAgB,gCAAW,KAAK,aAAa,WAAW;AAC7D,SAAK,mBAAmB;AACxB,SAAK,iCAAiC;AAAA,EACxC;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EASA,MAAa,YAAY,gBAAgE,SAAkC;AACzH,UAAM,KAAK,KAAK,cAAc;AAC9B,UAAM,KAAK,WAAW,OAAO;AAAA,EAC/B;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAUA,MAAa,WAAW,UAAyC;AAC/D,UAAM,mBAAmB,MAAM,KAAK,SAAS,QAAQ;AACrD,UAAM,WAAW,KAAK;AACtB,eAAW,gBAAgB,KAAK,eAAe;AAC7C,UAAI,iBAAiB,YAAY,GAAG;AAClC,iBAAS,YAAY,IAAI,SAAS,YAAY;AAAA,MAChD;AAAA,IACF;AAAA,EACF;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAQA,MAAa,YAAY,UAAmD;AAC1E,UAAM,eAAe,MAAM,KAAK,cAAc,QAAQ;AACtD,UAAM,KAAK,WAAW,YAAY;AAClC,WAAO;AAAA,EACT;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAQA,MAAa,aAAa,eAAuC;AAC/D,UAAM,OAAO,MAAM,KAAK,WAAW;AACnC,SAAK,iBAAiB,KAAK,mBAAmB;AAC9C,SAAK,eAAe,KAAK,mBAAmB;AAE5C,QAAI;AACF,UAAI,SAAS,UAAa,SAAS,MAAM;AACvC;AAAA,MACF;AAEA,UAAI,OAAO,SAAS,UAAU;AAC5B,gBAAQ,MAAM,0DAA0D,OAAO,IAAI,EAAE;AACrF;AAAA,MACF;AAEA,YAAM,YAAY;AAClB,YAAM,iBAAiB,MAAM,KAAK,oBAAoB,SAAS;AAC/D,YAAM,mBAAmB,MAAM,KAAK,SAAS,cAAc;AAE3D,iBAAW,gBAAgB,KAAK,eAAe;AAC7C,aAAK,gBAAgB,cAAc,eAAe,YAAY,GAAG,iBAAiB,YAAY,CAAC;AAAA,MACjG;AAEA,WAAK,iBAAiB,MAAM,KAAK,WAAW,KAAK,YAAY;AAE7D,YAAM,YAAY,MAAM,KAAK,oBAAoB,KAAK,aAAa,WAAW;AAE9E,UAAI,KAAC,+BAAU,WAAW,IAAI,GAAG;AAC/B,cAAM,KAAK,eAAe;AAAA,MAC5B;AAAA,IACF,UAAE;AACA,YAAM,KAAK,YAAY,aAAa,gBAAgB,KAAK,cAAc,aAAa;AAAA,IACtF;AAAA,EACF;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EA2CO,GACL,MACA,UACA,SACe;AACf,WAAO,KAAK,YAAY,GAAG,MAAM,UAAU,OAAO;AAAA,EACpD;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAOA,MAAa,2BAA0C;AACrD,UAAM,KAAK,aAAa,KAAK;AAAA,EAC/B;AAAA;AAAA;AAAA;AAAA,EAKA,MAAsB,SAAwB;AAC5C,0DAAmB,MAAM,KAAK,GAAG,gBAAgB,KAAK,eAAe,KAAK,IAAI,CAAC,CAAC;AAChF,0DAAmB,MAAM,KAAK,GAAG,gBAAgB,KAAK,eAAe,KAAK,IAAI,CAAC,CAAC;AAChF,UAAM,KAAK,aAAa,IAAI;AAAA,EAC9B;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EASO,gCACL,qBACA,WACM;AACN,UAAM,OAAO;AACb,SAAK,yBAAyB,KAAK,uBAAuB;AAE1D,aAAS,wBAAwB,QAA6B;AAC5D,YAAM,qBAAqB,IAAI,IAAY,OAAO,KAAK,IAAI,oBAAoB,CAAC,CAAC;AACjF,YAAM,oBAAoB,IAAI,IAAY,KAAK,aAAa;AAC5D,YAAM,iBAAiB;AACvB,gBAAU,cAAc;AACxB,iBAAW,OAAO,OAAO,KAAK,cAAc,GAAG;AAC7C,YAAI,kBAAkB,IAAI,GAAG,GAAG;AAC9B;AAAA,QACF;AAEA,YAAI,CAAC,mBAAmB,IAAI,GAAG,GAAG;AAChC;AAAA,QACF;AAGA,eAAO,OAAO,GAAG;AAAA,MACnB;AAAA,IACF;AAAA,EACF;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAQO,kBACL,cACA,WACM;AACN,SAAK,WAAW,IAAI,cAAc,SAA8C;AAAA,EAClF;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAOA,MAAa,aAAqE;AAChF,UAAM,KAAK,KAAK,oBAAI;AACpB,WAAO,KAAK,aAAa;AAAA,EAC3B;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAQA,MAAa,WAAW,SAAkC;AACxD,YAAI,+BAAU,KAAK,eAAe,aAAa,KAAK,aAAa,WAAW,GAAG;AAC7E;AAAA,IACF;AAEA,UAAM,KAAK,eAAe;AAC1B,UAAM,KAAK,YAAY,aAAa,gBAAgB,KAAK,cAAc,KAAK,gBAAgB,OAAO;AACnG,SAAK,iBAAiB,MAAM,KAAK,WAAW,KAAK,YAAY;AAAA,EAC/D;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAUA,MAAa,YACX,cACA,OACiB;AACjB,UAAM,KAAK,KAAK,CAAC,aAAa;AAC5B,eAAS,YAAY,IAAI;AAAA,IAC3B,CAAC;AACD,WAAO,KAAK,aAAa,mBAAmB,YAAY;AAAA,EAC1D;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAQA,MAAa,SAAS,UAAqE;AACzF,UAAM,SAA2C,CAAC;AAClD,eAAW,CAAC,cAAc,SAAS,KAAK,KAAK,WAAW,QAAQ,GAAG;AACjE,YAAM,oBAAoB,MAAM,UAAU,SAAS,YAAY,GAAG,QAAQ;AAC1E,UAAI,mBAAmB;AACrB,eAAO,YAAY,IAAI;AAAA,MACzB;AAAA,IACF;AAEA,WAAO;AAAA,EACT;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAcU,iBAA8B;AACtC,WAAO;AAAA,EACT;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAOA,MAAgB,aAAa,QAAsC;AACjE,eAAW,aAAa,KAAK,0BAA0B;AACrD,gBAAU,MAAM;AAAA,IAClB;AACA,UAAM,QAAQ,QAAQ;AAAA,EACxB;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAQA,MAAgB,eACd,cACA,gBACe;AACf,cAAM,2BAAU;AAAA,EAClB;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EASA,MAAgB,eACd,WACA,WACA,UACe;AACf,cAAM,2BAAU;AAAA,EAClB;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAOA,MAAgB,eAAe,SAAuC;AACpE,cAAM,2BAAU;AAAA,EAClB;AAAA;AAAA;AAAA;AAAA;AAAA,EAMU,mCAAyC;AACjD,8BAAK;AAAA,EACP;AAAA;AAAA;AAAA;AAAA;AAAA,EAMU,qBAA2B;AACnC,8BAAK;AAAA,EACP;AAAA,EAEA,MAAc,cAAc,UAAmD;AAC7E,UAAM,SAAS,MAAM,KAAK,oBAAoB,QAAQ;AACtD,UAAM,OAAO,KAAK,UAAU,MAAM;AAClC,UAAM,cAAc,KAAK,MAAM,IAAI;AACnC,WAAO,MAAM,KAAK,oBAAoB,WAAW;AAAA,EACnD;AAAA,EAEA,MAAc,WACZ,OAC8C;AAC9C,WAAO;AAAA,MACL,iBAAiB,MAAM,KAAK,cAAc,MAAM,eAAe;AAAA,MAC/D,aAAa,MAAM,KAAK,cAAc,MAAM,WAAW;AAAA,MACvD,oBAAoB,EAAE,GAAG,MAAM,mBAAmB;AAAA,IACpD;AAAA,EACF;AAAA,EAEQ,qBAA0D;AAChE,WAAO;AAAA,MACL,iBAAiB,KAAK,sBAAsB;AAAA,MAC5C,aAAa,KAAK,sBAAsB;AAAA,MACxC,wBAAoB,4BAAsD,CAAC,CAAC;AAAA,IAC9E;AAAA,EACF;AAAA,EAEA,MAAc,KAAK,gBAA+E;AAChG,QAAI;AACF,YAAM,eAAe,KAAK,aAAa,WAAW;AAAA,IACpD,UAAE;AACA,YAAM,mBAAmB,MAAM,KAAK,SAAS,KAAK,aAAa,WAAW;AAC1E,iBAAW,gBAAgB,KAAK,eAAe;AAC7C,cAAM,oBAAoB,iBAAiB,YAAY,KAAK;AAC5D,aAAK,aAAa,mBAAmB,YAAY,IAAI;AACrD,cAAM,WAAW,KAAK;AACtB,aAAK,aAAa,gBAAgB,YAAY,IAAI,oBAC9C,SAAS,YAAY,IACrB,KAAK,aAAa,YAAY,YAAY;AAAA,MAChD;AAAA,IACF;AAAA,EACF;AAAA,EAEQ,oBAAoB,MAAsD;AAChF,QAAI,OAAO,SAAS,UAAU;AAC5B,aAAO;AAAA,IACT;AAEA,WAAQ,KAAK,cAA2B,SAAS,IAAI;AAAA,EACvD;AAAA,EAEA,MAAc,oBAAoB,WAAmD;AACnF,gBAAY,KAAK,eAAe,EAAE,2BAA2B,SAAS;AACtE,UAAM,KAAK,aAAa,SAAS;AAEjC,UAAM,WAAW,KAAK,sBAAsB;AAC5C,UAAM,WAAW,KAAK;AAEtB,eAAW,CAAC,cAAc,KAAK,KAAK,OAAO,QAAQ,SAAS,GAAG;AAC7D,UAAI,CAAC,KAAK,oBAAoB,YAAY,GAAG;AAC3C,yCAAe,iDAAiD,EAAE,qBAAqB,YAAY,EAAE;AACrG;AAAA,MACF;AAEA,UAAI,OAAO,UAAU,OAAO,SAAS,YAAY,GAAG;AAClD,yCAAe,iDAAiD;AAAA,UAC9D;AAAA,UACA;AAAA,YACE,cAAc,SAAS,YAAY;AAAA,YACnC;AAAA,YACA;AAAA,UACF;AAAA,QACF;AAAA,MACF;AAEA,eAAS,YAAY,IAAI;AAAA,IAC3B;AAEA,WAAO;AAAA,EACT;AAAA,EAEA,MAAc,iBAAgC;AAC5C,UAAM,KAAK,WAAW,MAAM,KAAK,oBAAoB,KAAK,aAAa,WAAW,CAAC;AAAA,EACrF;AAAA,EAEQ,gBACN,cACA,OACA,mBACM;AACN,UAAM,WAAW,KAAK;AACtB,SAAK,aAAa,YAAY,YAAY,IAAI;AAC9C,SAAK,aAAa,mBAAmB,YAAY,IAAI,qBAAqB;AAC1E,SAAK,aAAa,gBAAgB,YAAY,IAAI,oBAAoB,SAAS,YAAY,IAAI;AAAA,EACjG;AAAA,EAEA,MAAc,oBAAoB,UAAkD;AAClF,UAAM,YAA2B,CAAC;AAElC,eAAW,gBAAgB,KAAK,eAAe;AAC7C,gBAAU,YAAY,IAAI,SAAS,YAAY;AAAA,IACjD;AAEA,UAAM,KAAK,eAAe,SAAS;AAEnC,WAAO,KAAK,eAAe,EAAE,2BAA2B,SAAS;AAAA,EACnE;AACF;AAKO,MAAM,qCAAqC,4BAAoC;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAM7E,YAAY,QAAwC;AACzD;AAAA,MACE,UAAU;AAAA,QACR,UAAU,YAA6B;AACrC,oBAAM,2BAAU;AAChB,iBAAO,CAAC;AAAA,QACV;AAAA,QACA,UAAU;AAAA,MACZ;AAAA,IACF;AAAA,EACF;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAOmB,wBAAgC;AACjD,WAAO,CAAC;AAAA,EACV;AACF;",
  "names": []
}

@@ -13,10 +13,6 @@ import type { Transformer } from '../../../transformers/transformer.cjs';
13
13
  import type { GenericObject } from '../../../type-guards.cjs';
14
14
  import type { MaybeReturn, StringKeys } from '../../../type.cjs';
15
15
  import { AsyncComponentBase } from '../../components/async-component.cjs';
16
- /**
17
- * Component key used by {@link registerComponent} to identify settings components.
18
- */
19
- export declare const PLUGIN_SETTINGS_COMPONENT_KEY = "PluginSettingsComponent";
20
16
  /**
21
17
  * Params for creating a {@link PluginSettingsComponentBase}.
22
18
  */
@@ -87,7 +83,7 @@ export declare abstract class PluginSettingsComponentBase<PluginSettings extends
87
83
  /**
88
84
  * Component key for {@link registerComponent} replacement.
89
85
  */
90
- static readonly COMPONENT_KEY = "PluginSettingsComponent";
86
+ static readonly COMPONENT_KEY: unique symbol;
91
87
  /**
92
88
  * Gets the readonly default settings.
93
89
  *
@@ -129,6 +129,8 @@ __export(plugin_settings_tab_component_exports, {
129
129
  });
130
130
  module.exports = __toCommonJS(plugin_settings_tab_component_exports);
131
131
  var import_obsidian = require('obsidian');
132
+ var import_command_handler_component = require('../../command-handlers/command-handler-component.cjs');
133
+ var import_open_settings_command_handler = require('../../command-handlers/open-settings-command-handler.cjs');
132
134
  class PluginSettingsTabComponent extends import_obsidian.Component {
133
135
  /**
134
136
  * Creates a new plugin settings tab component.
@@ -148,19 +150,11 @@ class PluginSettingsTabComponent extends import_obsidian.Component {
148
150
  */
149
151
  onload() {
150
152
  this.plugin.addSettingTab(this.settingsTab);
151
- this.plugin.addCommand({
152
- callback: () => {
153
- this.plugin.app.setting.openTabById(this.plugin.manifest.id);
154
- this.plugin.app.setting.open();
155
- },
156
- icon: "settings",
157
- id: "open-settings",
158
- name: "Open settings"
159
- });
153
+ this.addChild(new import_command_handler_component.CommandHandlerComponent(this.plugin, new import_open_settings_command_handler.OpenSettingsCommandHandler(this.plugin.manifest.name, this.settingsTab)));
160
154
  }
161
155
  }
162
156
  // Annotate the CommonJS export names for ESM import in node:
163
157
  0 && (module.exports = {
164
158
  PluginSettingsTabComponent
165
159
  });
166
- //# sourceMappingURL=data:application/json;base64,ewogICJ2ZXJzaW9uIjogMywKICAic291cmNlcyI6IFsiLi4vLi4vLi4vLi4vLi4vLi4vc3JjL29ic2lkaWFuL3BsdWdpbi9jb21wb25lbnRzL3BsdWdpbi1zZXR0aW5ncy10YWItY29tcG9uZW50LnRzIl0sCiAgInNvdXJjZXNDb250ZW50IjogWyIvKipcbiAqIEBmaWxlXG4gKlxuICogQ29tcG9uZW50IHRoYXQgcmVnaXN0ZXJzIGEgcGx1Z2luIHNldHRpbmdzIHRhYiB3aXRoIE9ic2lkaWFuLlxuICogQWxzbyByZWdpc3RlcnMgYW4gXCJPcGVuIFNldHRpbmdzXCIgY29tbWFuZCBhdXRvbWF0aWNhbGx5LlxuICovXG5cbmltcG9ydCB0eXBlIHsgUGx1Z2luIH0gZnJvbSAnb2JzaWRpYW4nO1xuXG5pbXBvcnQgeyBDb21wb25lbnQgfSBmcm9tICdvYnNpZGlhbic7XG5cbmltcG9ydCB0eXBlIHsgUGx1Z2luU2V0dGluZ3NUYWJCYXNlIH0gZnJvbSAnLi4vcGx1Z2luLXNldHRpbmdzLXRhYi50cyc7XG5cbi8qKlxuICogV3JhcHMgYSB7QGxpbmsgUGx1Z2luU2V0dGluZ3NUYWJCYXNlfSBhbmQgcmVnaXN0ZXJzIGl0IHdpdGggT2JzaWRpYW4gb24gbG9hZC5cbiAqIEFsc28gcmVnaXN0ZXJzIGFuIFwiT3BlbiBTZXR0aW5nc1wiIGNvbW1hbmQgdG8gb3BlbiB0aGUgc2V0dGluZ3MgdGFiIGZyb20gdGhlIGNvbW1hbmQgcGFsZXR0ZS5cbiAqL1xuZXhwb3J0IGNsYXNzIFBsdWdpblNldHRpbmdzVGFiQ29tcG9uZW50IGV4dGVuZHMgQ29tcG9uZW50IHtcbiAgLyoqXG4gICAqIENyZWF0ZXMgYSBuZXcgcGx1Z2luIHNldHRpbmdzIHRhYiBjb21wb25lbnQuXG4gICAqXG4gICAqIEBwYXJhbSBwbHVnaW4gLSBUaGUgT2JzaWRpYW4gcGx1Z2luIGluc3RhbmNlLlxuICAgKiBAcGFyYW0gc2V0dGluZ3NUYWIgLSBUaGUgc2V0dGluZ3MgdGFiIHRvIHJlZ2lzdGVyLlxuICAgKi9cbiAgcHVibGljIGNvbnN0cnVjdG9yKFxuICAgIHByaXZhdGUgcmVhZG9ubHkgcGx1Z2luOiBQbHVnaW4sXG4gICAgcHVibGljIHJlYWRvbmx5IHNldHRpbmdzVGFiOiBQbHVnaW5TZXR0aW5nc1RhYkJhc2U8b2JqZWN0PlxuICApIHtcbiAgICBzdXBlcigpO1xuICB9XG5cbiAgLyoqXG4gICAqIFJlZ2lzdGVycyB0aGUgc2V0dGluZ3MgdGFiIGFuZCBhbiBcIk9wZW4gU2V0dGluZ3NcIiBjb21tYW5kIHdpdGggT2JzaWRpYW4uXG4gICAqL1xuICBwdWJsaWMgb3ZlcnJpZGUgb25sb2FkKCk6IHZvaWQge1xuICAgIHRoaXMucGx1Z2luLmFkZFNldHRpbmdUYWIodGhpcy5zZXR0aW5nc1RhYik7XG4gICAgdGhpcy5wbHVnaW4uYWRkQ29tbWFuZCh7XG4gICAgICBjYWxsYmFjazogKCkgPT4ge1xuICAgICAgICB0aGlzLnBsdWdpbi5hcHAuc2V0dGluZy5vcGVuVGFiQnlJZCh0aGlzLnBsdWdpbi5tYW5pZmVzdC5pZCk7XG4gICAgICAgIHRoaXMucGx1Z2luLmFwcC5zZXR0aW5nLm9wZW4oKTtcbiAgICAgIH0sXG4gICAgICBpY29uOiAnc2V0dGluZ3MnLFxuICAgICAgaWQ6ICdvcGVuLXNldHRpbmdzJyxcbiAgICAgIG5hbWU6ICdPcGVuIHNldHRpbmdzJ1xuICAgIH0pO1xuICB9XG59XG4iXSwKICAibWFwcGluZ3MiOiAiOzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7OztBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFTQSxzQkFBMEI7QUFRbkIsTUFBTSxtQ0FBbUMsMEJBQVU7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQSxFQU9qRCxZQUNZLFFBQ0QsYUFDaEI7QUFDQSxVQUFNO0FBSFc7QUFDRDtBQUFBLEVBR2xCO0FBQUEsRUFKbUI7QUFBQSxFQUNEO0FBQUE7QUFBQTtBQUFBO0FBQUEsRUFRRixTQUFlO0FBQzdCLFNBQUssT0FBTyxjQUFjLEtBQUssV0FBVztBQUMxQyxTQUFLLE9BQU8sV0FBVztBQUFBLE1BQ3JCLFVBQVUsTUFBTTtBQUNkLGFBQUssT0FBTyxJQUFJLFFBQVEsWUFBWSxLQUFLLE9BQU8sU0FBUyxFQUFFO0FBQzNELGFBQUssT0FBTyxJQUFJLFFBQVEsS0FBSztBQUFBLE1BQy9CO0FBQUEsTUFDQSxNQUFNO0FBQUEsTUFDTixJQUFJO0FBQUEsTUFDSixNQUFNO0FBQUEsSUFDUixDQUFDO0FBQUEsRUFDSDtBQUNGOyIsCiAgIm5hbWVzIjogW10KfQo=
160
+ //# sourceMappingURL=data:application/json;base64,ewogICJ2ZXJzaW9uIjogMywKICAic291cmNlcyI6IFsiLi4vLi4vLi4vLi4vLi4vLi4vc3JjL29ic2lkaWFuL3BsdWdpbi9jb21wb25lbnRzL3BsdWdpbi1zZXR0aW5ncy10YWItY29tcG9uZW50LnRzIl0sCiAgInNvdXJjZXNDb250ZW50IjogWyIvKipcbiAqIEBmaWxlXG4gKlxuICogQ29tcG9uZW50IHRoYXQgcmVnaXN0ZXJzIGEgcGx1Z2luIHNldHRpbmdzIHRhYiB3aXRoIE9ic2lkaWFuLlxuICogQWxzbyByZWdpc3RlcnMgYW4gXCJPcGVuIFNldHRpbmdzXCIgY29tbWFuZCBhdXRvbWF0aWNhbGx5LlxuICovXG5cbmltcG9ydCB0eXBlIHsgUGx1Z2luIH0gZnJvbSAnb2JzaWRpYW4nO1xuXG5pbXBvcnQgeyBDb21wb25lbnQgfSBmcm9tICdvYnNpZGlhbic7XG5cbmltcG9ydCB0eXBlIHsgUGx1Z2luU2V0dGluZ3NUYWJCYXNlIH0gZnJvbSAnLi4vcGx1Z2luLXNldHRpbmdzLXRhYi50cyc7XG5cbmltcG9ydCB7IENvbW1hbmRIYW5kbGVyQ29tcG9uZW50IH0gZnJvbSAnLi4vLi4vY29tbWFuZC1oYW5kbGVycy9jb21tYW5kLWhhbmRsZXItY29tcG9uZW50LnRzJztcbmltcG9ydCB7IE9wZW5TZXR0aW5nc0NvbW1hbmRIYW5kbGVyIH0gZnJvbSAnLi4vLi4vY29tbWFuZC1oYW5kbGVycy9vcGVuLXNldHRpbmdzLWNvbW1hbmQtaGFuZGxlci50cyc7XG5cbi8qKlxuICogV3JhcHMgYSB7QGxpbmsgUGx1Z2luU2V0dGluZ3NUYWJCYXNlfSBhbmQgcmVnaXN0ZXJzIGl0IHdpdGggT2JzaWRpYW4gb24gbG9hZC5cbiAqIEFsc28gcmVnaXN0ZXJzIGFuIFwiT3BlbiBTZXR0aW5nc1wiIGNvbW1hbmQgdG8gb3BlbiB0aGUgc2V0dGluZ3MgdGFiIGZyb20gdGhlIGNvbW1hbmQgcGFsZXR0ZS5cbiAqL1xuZXhwb3J0IGNsYXNzIFBsdWdpblNldHRpbmdzVGFiQ29tcG9uZW50IGV4dGVuZHMgQ29tcG9uZW50IHtcbiAgLyoqXG4gICAqIENyZWF0ZXMgYSBuZXcgcGx1Z2luIHNldHRpbmdzIHRhYiBjb21wb25lbnQuXG4gICAqXG4gICAqIEBwYXJhbSBwbHVnaW4gLSBUaGUgT2JzaWRpYW4gcGx1Z2luIGluc3RhbmNlLlxuICAgKiBAcGFyYW0gc2V0dGluZ3NUYWIgLSBUaGUgc2V0dGluZ3MgdGFiIHRvIHJlZ2lzdGVyLlxuICAgKi9cbiAgcHVibGljIGNvbnN0cnVjdG9yKFxuICAgIHByaXZhdGUgcmVhZG9ubHkgcGx1Z2luOiBQbHVnaW4sXG4gICAgcHVibGljIHJlYWRvbmx5IHNldHRpbmdzVGFiOiBQbHVnaW5TZXR0aW5nc1RhYkJhc2U8b2JqZWN0PlxuICApIHtcbiAgICBzdXBlcigpO1xuICB9XG5cbiAgLyoqXG4gICAqIFJlZ2lzdGVycyB0aGUgc2V0dGluZ3MgdGFiIGFuZCBhbiBcIk9wZW4gU2V0dGluZ3NcIiBjb21tYW5kIHdpdGggT2JzaWRpYW4uXG4gICAqL1xuICBwdWJsaWMgb3ZlcnJpZGUgb25sb2FkKCk6IHZvaWQge1xuICAgIHRoaXMucGx1Z2luLmFkZFNldHRpbmdUYWIodGhpcy5zZXR0aW5nc1RhYik7XG4gICAgdGhpcy5hZGRDaGlsZChuZXcgQ29tbWFuZEhhbmRsZXJDb21wb25lbnQodGhpcy5wbHVnaW4sIG5ldyBPcGVuU2V0dGluZ3NDb21tYW5kSGFuZGxlcih0aGlzLnBsdWdpbi5tYW5pZmVzdC5uYW1lLCB0aGlzLnNldHRpbmdzVGFiKSkpO1xuICB9XG59XG4iXSwKICAibWFwcGluZ3MiOiAiOzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7OztBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFTQSxzQkFBMEI7QUFJMUIsdUNBQXdDO0FBQ3hDLDJDQUEyQztBQU1wQyxNQUFNLG1DQUFtQywwQkFBVTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBLEVBT2pELFlBQ1ksUUFDRCxhQUNoQjtBQUNBLFVBQU07QUFIVztBQUNEO0FBQUEsRUFHbEI7QUFBQSxFQUptQjtBQUFBLEVBQ0Q7QUFBQTtBQUFBO0FBQUE7QUFBQSxFQVFGLFNBQWU7QUFDN0IsU0FBSyxPQUFPLGNBQWMsS0FBSyxXQUFXO0FBQzFDLFNBQUssU0FBUyxJQUFJLHlEQUF3QixLQUFLLFFBQVEsSUFBSSxnRUFBMkIsS0FBSyxPQUFPLFNBQVMsTUFBTSxLQUFLLFdBQVcsQ0FBQyxDQUFDO0FBQUEsRUFDckk7QUFDRjsiLAogICJuYW1lcyI6IFtdCn0K
@@ -409,4 +409,4 @@ class PluginSettingsTabBase extends import_obsidian.PluginSettingTab {
409
409
  PluginSettingsTabBase,
410
410
  SAVE_TO_FILE_CONTEXT
411
411
  });
412
- //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../../../src/obsidian/plugin/plugin-settings-tab.ts"],
  "sourcesContent": ["/**\n * @file\n *\n * This module defines a base class for creating plugin setting tabs in Obsidian.\n * It provides a utility method to bind value components to plugin settings and handle changes.\n */\n\nimport type {\n  Debouncer,\n  Plugin\n} from 'obsidian';\nimport type {\n  ConditionalKeys,\n  Promisable,\n  ReadonlyDeep\n} from 'type-fest';\n\nimport {\n  debounce,\n  PluginSettingTab,\n  setTooltip\n} from 'obsidian';\n\nimport type { AsyncEventRef } from '../../async-events.ts';\nimport type { StringKeys } from '../../type.ts';\nimport type { ValueComponentWithChangeTracking } from '../components/setting-components/value-component-with-change-tracking.ts';\nimport type { ValidationMessageHolder } from '../validation.ts';\nimport type {\n  PluginSettingsComponentBase,\n  ReadonlyPluginSettings,\n  ReadonlyPluginSettingsState\n} from './components/plugin-settings-component.ts';\n\nimport {\n  convertAsyncToSync,\n  invokeAsyncSafely\n} from '../../async.ts';\nimport { CssClass } from '../../css-class.ts';\nimport {\n  noop,\n  noopAsync\n} from '../../function.ts';\nimport { deepEqual } from '../../object-utils.ts';\nimport { assertNonNullable } from '../../type-guards.ts';\nimport { AsyncEventsComponent } from '../components/async-events-component.ts';\nimport { ensureWrapped } from '../components/setting-components/setting-component-wrapper.ts';\nimport { getTextBasedComponentValue } from '../components/setting-components/text-based-component.ts';\nimport { getValidatorComponent } from '../components/setting-components/validator-component.ts';\nimport { isValidationMessageHolder } from '../validation.ts';\nimport { addPluginCssClasses } from './plugin-context.ts';\n\n/**\n * A context passed to the {@link PluginSettingsComponentBase.saveToFile} method.\n */\nexport const SAVE_TO_FILE_CONTEXT = 'PluginSettingsTab';\n\n/**\n * Options for `PluginSettingsTabBase.bind`.\n */\nexport interface BindOptions<T> {\n  /**\n   * A callback function that is called when the value of the component changes.\n   */\n  readonly onChanged?: (newValue: ReadonlyDeep<T>, oldValue: ReadonlyDeep<T>) => Promisable<void>;\n\n  /**\n   * Whether to reset the setting when the component value is empty. Default is `true`.\n   * Applicable only to text-based components.\n   */\n  readonly shouldResetSettingWhenComponentIsEmpty?: boolean;\n\n  /**\n   * Whether to show the placeholder for default values. Default is `true`.\n   * Applicable only to text-based components.\n   */\n  readonly shouldShowPlaceholderForDefaultValues?: boolean;\n\n  /**\n   * Whether to show the validation message when the component value is invalid. Default is `true`.\n   */\n  readonly shouldShowValidationMessage?: boolean;\n}\n\n/**\n * Extended options for `PluginSettingsTabBase.bind`.\n */\nexport interface BindOptionsExtended<\n  PluginSettings extends object,\n  UIValue,\n  PropertyName extends StringKeys<PluginSettings>\n> extends BindOptions<PluginSettings[PropertyName]> {\n  /**\n   * Converts the UI component's value back to the plugin settings value.\n   *\n   * @param uiValue - The value of the UI component.\n   * @returns The value to set on the plugin settings.\n   */\n  readonly componentToPluginSettingsValueConverter: (uiValue: UIValue) => PluginSettings[PropertyName] | ValidationMessageHolder;\n\n  /**\n   * Converts the plugin settings value to the value used by the UI component.\n   *\n   * @param pluginSettingsValue - The value of the property in the plugin settings.\n   * @returns The value to set on the UI component.\n   */\n  readonly pluginSettingsToComponentValueConverter: (pluginSettingsValue: ReadonlyDeep<PluginSettings[PropertyName]>) => UIValue;\n}\n\n/**\n * Params for creating a {@link PluginSettingsTabBase}.\n */\nexport interface PluginSettingsTabBaseParams<PluginSettings extends object> {\n  /**\n   * The plugin instance (needed by Obsidian's PluginSettingTab).\n   */\n  readonly plugin: Plugin;\n\n  /**\n   * The settings component.\n   */\n  readonly pluginSettingsComponent: PluginSettingsComponentBase<PluginSettings>;\n}\n\n/**\n * Base class for creating plugin settings tabs in Obsidian.\n * Provides a method for binding value components to plugin settings and handling changes.\n *\n * @typeParam PluginSettings - The plugin settings type.\n */\nexport abstract class PluginSettingsTabBase<PluginSettings extends object> extends PluginSettingTab {\n  /**\n   * Whether the plugin settings tab is open.\n   *\n   * @returns Whether the plugin settings tab is open.\n   */\n  public get isOpen(): boolean {\n    return this._isOpen;\n  }\n\n  /**\n   * The settings manager.\n   */\n  protected readonly pluginSettingsComponent: PluginSettingsComponentBase<PluginSettings>;\n\n  /**\n   * A debounce timeout for saving settings.\n   *\n   * @returns The debounce timeout for saving settings.\n   */\n  protected get saveSettingsDebounceTimeoutInMilliseconds(): number {\n    const DEFAULT = 2_000;\n    return DEFAULT;\n  }\n\n  private _isOpen = false;\n  private readonly asyncEventsComponent: AsyncEventsComponent;\n  private readonly saveSettingsDebounced: Debouncer<[], void>;\n\n  private get pluginSettings(): ReadonlyPluginSettings<PluginSettings> {\n    return this.pluginSettingsComponent.settingsState.inputValues;\n  }\n\n  /**\n   * Creates a new plugin settings tab.\n   *\n   * @param params - The params.\n   */\n  public constructor(params: PluginSettingsTabBaseParams<PluginSettings>) {\n    super(params.plugin.app, params.plugin);\n    this.pluginSettingsComponent = params.pluginSettingsComponent;\n    addPluginCssClasses(this.containerEl, CssClass.PluginSettingsTab);\n    this.saveSettingsDebounced = debounce(\n      convertAsyncToSync(() => this.pluginSettingsComponent.saveToFile(SAVE_TO_FILE_CONTEXT)),\n      this.saveSettingsDebounceTimeoutInMilliseconds\n    );\n    this.asyncEventsComponent = new AsyncEventsComponent();\n  }\n\n  /**\n   * Binds a value component to a plugin setting.\n   *\n   * @typeParam UIValue - The type of the value of the UI component.\n   * @typeParam TValueComponent - The type of the value component.\n   * @param valueComponent - The value component to bind.\n   * @param propertyName - The property of the plugin settings to bind to.\n   * @param options - The options for binding the value component.\n   * @returns The value component.\n   */\n  public bind<\n    UIValue,\n    TValueComponent\n  >(\n    valueComponent: TValueComponent & ValueComponentWithChangeTracking<UIValue>,\n    propertyName: ConditionalKeys<PluginSettings, UIValue>,\n    options?: BindOptions<UIValue>\n  ): TValueComponent;\n  /**\n   * Binds a value component to a plugin setting.\n   *\n   * @typeParam UIValue - The type of the value of the UI component.\n   * @typeParam TValueComponent - The type of the value component.\n   * @typeParam PropertyName - The property name of the plugin settings to bind to.\n   * @param valueComponent - The value component to bind.\n   * @param propertyName - The property name of the plugin settings to bind to.\n   * @param options - The options for binding the value component.\n   * @returns The value component.\n   */\n  public bind<\n    UIValue,\n    TValueComponent,\n    PropertyName extends StringKeys<PluginSettings>\n  >(\n    valueComponent: TValueComponent & ValueComponentWithChangeTracking<UIValue>,\n    propertyName: PropertyName,\n    options: BindOptionsExtended<PluginSettings, UIValue, PropertyName>\n  ): TValueComponent;\n  /**\n   * Binds a value component to a plugin setting.\n   *\n   * @typeParam UIValue - The type of the value of the UI component.\n   * @typeParam TValueComponent - The type of the value component.\n   * @typeParam PropertyName - The property name of the plugin settings to bind to.\n   * @param valueComponent - The value component to bind.\n   * @param propertyName - The property name of the plugin settings to bind to.\n   * @param options - The options for binding the value component.\n   * @returns The value component.\n   */\n  public bind<\n    UIValue,\n    TValueComponent,\n    PropertyName extends StringKeys<PluginSettings>\n  >(\n    valueComponent: TValueComponent & ValueComponentWithChangeTracking<UIValue>,\n    propertyName: PropertyName,\n    options?: BindOptions<PluginSettings[PropertyName]>\n  ): TValueComponent {\n    type PropertyType = PluginSettings[PropertyName];\n    const DEFAULT_OPTIONS: Required<BindOptionsExtended<PluginSettings, UIValue, PropertyName>> = {\n      componentToPluginSettingsValueConverter: (value: UIValue): PropertyType => value as PropertyType,\n      onChanged: noop,\n      pluginSettingsToComponentValueConverter: (value: ReadonlyDeep<PropertyType>): UIValue => value as UIValue,\n      shouldResetSettingWhenComponentIsEmpty: true,\n      shouldShowPlaceholderForDefaultValues: true,\n      shouldShowValidationMessage: true\n    };\n\n    const optionsExt: Required<BindOptionsExtended<PluginSettings, UIValue, PropertyName>> = { ...DEFAULT_OPTIONS, ...options };\n\n    const validatorEl = getValidatorComponent(valueComponent)?.validatorEl;\n\n    const textBasedComponent = getTextBasedComponentValue(valueComponent);\n\n    const readonlyValue = this.getPluginSettingsProperty(propertyName);\n    const defaults = this.pluginSettingsComponent.defaultSettings as PluginSettings;\n    const defaultValue = defaults[propertyName] as PropertyType;\n    const defaultComponentValue = optionsExt.pluginSettingsToComponentValueConverter(defaultValue as ReadonlyDeep<PropertyType>);\n    textBasedComponent?.setPlaceholderValue(defaultComponentValue);\n\n    let validationMessage: string;\n    let tooltipEl: HTMLElement | null = null;\n    let tooltipContentEl: HTMLElement | null = null;\n    if (validatorEl) {\n      const wrapper = ensureWrapped(validatorEl);\n      tooltipEl = wrapper.createDiv();\n      addPluginCssClasses(tooltipEl, CssClass.Tooltip, CssClass.TooltipValidator);\n      tooltipContentEl = tooltipEl.createSpan();\n      const tooltipArrowEl = tooltipEl.createDiv();\n      addPluginCssClasses(tooltipArrowEl, CssClass.TooltipArrow);\n      tooltipEl.hide();\n      wrapper.appendChild(tooltipEl);\n    }\n\n    this.asyncEventsComponent.registerAsyncEvent(this.on('validationMessageChanged', (anotherPropertyName, anotherValidationMessage) => {\n      if (propertyName !== anotherPropertyName) {\n        return;\n      }\n\n      validationMessage = anotherValidationMessage;\n      updateValidatorElDebounced();\n    }));\n\n    let shouldEmptyOnBlur = false;\n    let shouldRevertToDefaultValueOnBlur = false;\n\n    if (textBasedComponent && optionsExt.shouldShowPlaceholderForDefaultValues && deepEqual(readonlyValue, defaultValue)) {\n      textBasedComponent.empty();\n    } else {\n      valueComponent.setValue(optionsExt.pluginSettingsToComponentValueConverter(readonlyValue));\n    }\n\n    let shouldSkipOnChange = false;\n    const UPDATE_VALIDATOR_EL_TIMEOUT_IN_MILLISECONDS = 100;\n    const updateValidatorElDebounced = debounce(() => {\n      requestAnimationFrame(() => {\n        updateValidatorEl();\n      });\n    }, UPDATE_VALIDATOR_EL_TIMEOUT_IN_MILLISECONDS);\n\n    valueComponent.onChange(async (uiValue) => {\n      if (shouldSkipOnChange) {\n        shouldSkipOnChange = false;\n        return;\n      }\n\n      shouldEmptyOnBlur = false;\n\n      const oldValue = this.getPluginSettingsProperty(propertyName);\n      let newValue: PropertyType | undefined = undefined;\n      let shouldSetProperty = true;\n      shouldRevertToDefaultValueOnBlur = !!textBasedComponent?.isEmpty() && optionsExt.shouldResetSettingWhenComponentIsEmpty;\n      if (shouldRevertToDefaultValueOnBlur) {\n        newValue = defaultValue;\n      } else {\n        const convertedValue = optionsExt.componentToPluginSettingsValueConverter(uiValue);\n        if (isValidationMessageHolder(convertedValue)) {\n          validationMessage = convertedValue.validationMessage;\n          shouldSetProperty = false;\n        } else {\n          newValue = convertedValue;\n        }\n      }\n\n      if (shouldSetProperty) {\n        validationMessage = await this.pluginSettingsComponent.setProperty(propertyName, newValue as PluginSettings[PropertyName]);\n        if (textBasedComponent && optionsExt.shouldShowPlaceholderForDefaultValues && !textBasedComponent.isEmpty() && deepEqual(newValue, defaultValue)) {\n          shouldEmptyOnBlur = true;\n        }\n      }\n\n      updateValidatorElDebounced();\n      if (shouldSetProperty) {\n        await optionsExt.onChanged(newValue as ReadonlyDeep<PropertyType>, oldValue as ReadonlyDeep<PropertyType>);\n      }\n      this.saveSettingsDebounced();\n    });\n\n    validatorEl?.addEventListener('focus', () => {\n      updateValidatorElDebounced();\n    });\n    validatorEl?.addEventListener('blur', () => {\n      updateValidatorElDebounced();\n    });\n    validatorEl?.addEventListener('click', () => {\n      requestAnimationFrame(() => {\n        updateValidatorElDebounced();\n      });\n    });\n\n    const validationMessages = this.pluginSettingsComponent.settingsState.validationMessages as Record<string, string>;\n    validationMessage = validationMessages[propertyName] ?? '';\n    updateValidatorElDebounced();\n\n    return valueComponent;\n\n    function updateValidatorEl(): void {\n      if (!validatorEl?.isActiveElement()) {\n        if (shouldEmptyOnBlur) {\n          shouldEmptyOnBlur = false;\n\n          if (!textBasedComponent?.isEmpty()) {\n            shouldSkipOnChange = true;\n            textBasedComponent?.empty();\n          }\n        } else if (shouldRevertToDefaultValueOnBlur) {\n          shouldRevertToDefaultValueOnBlur = false;\n\n          if (textBasedComponent?.isEmpty()) {\n            shouldSkipOnChange = true;\n            valueComponent.setValue(defaultComponentValue);\n          }\n        }\n      }\n\n      if (!validatorEl) {\n        return;\n      }\n\n      assertNonNullable(tooltipContentEl);\n\n      if (validationMessage === '') {\n        validatorEl.setCustomValidity('');\n        validatorEl.checkValidity();\n        validationMessage = validatorEl.validationMessage;\n      }\n\n      validatorEl.setCustomValidity(validationMessage);\n      if (optionsExt.shouldShowValidationMessage) {\n        tooltipContentEl.textContent = validationMessage;\n        tooltipEl?.toggle(!!validationMessage);\n      } else if (validationMessage) {\n        setTooltip(validatorEl, validationMessage);\n      }\n    }\n  }\n\n  /**\n   * Renders the plugin settings tab.\n   */\n  public override display(): void {\n    this.containerEl.empty();\n    this._isOpen = true;\n    this.asyncEventsComponent.load();\n    this.asyncEventsComponent.registerAsyncEvent(this.pluginSettingsComponent.on('loadSettings', this.onLoadSettings.bind(this)));\n    this.asyncEventsComponent.registerAsyncEvent(this.pluginSettingsComponent.on('saveSettings', this.onSaveSettings.bind(this)));\n  }\n\n  /**\n   * Hides the plugin settings tab.\n   */\n  public override hide(): void {\n    super.hide();\n    this.saveSettingsDebounced.cancel();\n    this._isOpen = false;\n    this.asyncEventsComponent.unload();\n    this.asyncEventsComponent.load();\n    invokeAsyncSafely(() => this.hideAsync());\n  }\n\n  /**\n   * Async actions to perform when the settings tab is being hidden.\n   *\n   * @returns A {@link Promise} that resolves when the settings tab is hidden.\n   */\n  public async hideAsync(): Promise<void> {\n    await this.pluginSettingsComponent.saveToFile(SAVE_TO_FILE_CONTEXT);\n  }\n\n  /**\n   * Shows the plugin settings tab.\n   */\n  public show(): void {\n    this.app.setting.openTab(this);\n  }\n\n  /**\n   * Called when the plugin settings are loaded.\n   *\n   * @param _loadedState - The loaded settings state.\n   * @param _isInitialLoad - Whether the settings are being loaded for the first time.\n   * @returns A {@link Promise} that resolves when the settings are loaded.\n   */\n  protected async onLoadSettings(_loadedState: ReadonlyPluginSettingsState<PluginSettings>, _isInitialLoad: boolean): Promise<void> {\n    this.display();\n    await noopAsync();\n  }\n\n  /**\n   * Revalidates the settings.\n   *\n   * @returns A {@link Promise} that resolves when the settings are revalidated.\n   */\n  protected async revalidate(): Promise<void> {\n    const validationMessages = await this.pluginSettingsComponent.revalidate();\n    await this.updateValidations(validationMessages);\n  }\n\n  private getPluginSettingsProperty<PropertyName extends StringKeys<PluginSettings>>(\n    propertyName: PropertyName\n  ): ReadonlyDeep<PluginSettings[PropertyName]> {\n    const settings = this.pluginSettings as PluginSettings;\n    return settings[propertyName] as ReadonlyDeep<PluginSettings[PropertyName]>;\n  }\n\n  private on(\n    name: 'validationMessageChanged',\n    callback: (\n      propertyName: string,\n      validationMessage: string\n    ) => Promisable<void>,\n    thisArg?: unknown\n  ): AsyncEventRef;\n  private on<Args extends unknown[]>(\n    name: string,\n    callback: (...args: Args) => Promisable<void>,\n    thisArg?: unknown\n  ): AsyncEventRef {\n    return this.asyncEventsComponent.asyncEvents.on(name, callback, thisArg);\n  }\n\n  private async onSaveSettings(\n    newState: ReadonlyPluginSettingsState<PluginSettings>,\n    _oldState: ReadonlyPluginSettingsState<PluginSettings>,\n    context: unknown\n  ): Promise<void> {\n    if (context === SAVE_TO_FILE_CONTEXT) {\n      await this.updateValidations(newState.validationMessages as Record<StringKeys<PluginSettings>, string>);\n      return;\n    }\n\n    this.display();\n  }\n\n  private async updateValidations(validationMessages: Record<StringKeys<PluginSettings>, string>): Promise<void> {\n    for (const [propertyName, validationMessage] of Object.entries(validationMessages)) {\n      await this.asyncEventsComponent.asyncEvents.triggerAsync('validationMessageChanged', propertyName, validationMessage);\n    }\n  }\n}\n"],
  "mappings": "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",
  "names": []
}

412
+ //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../../../src/obsidian/plugin/plugin-settings-tab.ts"],
  "sourcesContent": ["/**\n * @file\n *\n * This module defines a base class for creating plugin setting tabs in Obsidian.\n * It provides a utility method to bind value components to plugin settings and handle changes.\n */\n\nimport type {\n  Debouncer,\n  Plugin\n} from 'obsidian';\nimport type {\n  ConditionalKeys,\n  Promisable,\n  ReadonlyDeep\n} from 'type-fest';\n\nimport {\n  debounce,\n  PluginSettingTab,\n  setTooltip\n} from 'obsidian';\n\nimport type { AsyncEventRef } from '../../async-events.ts';\nimport type { StringKeys } from '../../type.ts';\nimport type { ValueComponentWithChangeTracking } from '../components/setting-components/value-component-with-change-tracking.ts';\nimport type { ValidationMessageHolder } from '../validation.ts';\nimport type {\n  PluginSettingsComponentBase,\n  ReadonlyPluginSettings,\n  ReadonlyPluginSettingsState\n} from './components/plugin-settings-component.ts';\n\nimport {\n  convertAsyncToSync,\n  invokeAsyncSafely\n} from '../../async.ts';\nimport { CssClass } from '../../css-class.ts';\nimport {\n  noop,\n  noopAsync\n} from '../../function.ts';\nimport { deepEqual } from '../../object-utils.ts';\nimport { assertNonNullable } from '../../type-guards.ts';\nimport { AsyncEventsComponent } from '../components/async-events-component.ts';\nimport { ensureWrapped } from '../components/setting-components/setting-component-wrapper.ts';\nimport { getTextBasedComponentValue } from '../components/setting-components/text-based-component.ts';\nimport { getValidatorComponent } from '../components/setting-components/validator-component.ts';\nimport { isValidationMessageHolder } from '../validation.ts';\nimport { addPluginCssClasses } from './plugin-context.ts';\n\n/**\n * A context passed to the {@link PluginSettingsComponentBase.saveToFile} method.\n */\nexport const SAVE_TO_FILE_CONTEXT = 'PluginSettingsTab';\n\n/**\n * Options for `PluginSettingsTabBase.bind`.\n */\nexport interface BindOptions<T> {\n  /**\n   * A callback function that is called when the value of the component changes.\n   */\n  readonly onChanged?: (newValue: ReadonlyDeep<T>, oldValue: ReadonlyDeep<T>) => Promisable<void>;\n\n  /**\n   * Whether to reset the setting when the component value is empty. Default is `true`.\n   * Applicable only to text-based components.\n   */\n  readonly shouldResetSettingWhenComponentIsEmpty?: boolean;\n\n  /**\n   * Whether to show the placeholder for default values. Default is `true`.\n   * Applicable only to text-based components.\n   */\n  readonly shouldShowPlaceholderForDefaultValues?: boolean;\n\n  /**\n   * Whether to show the validation message when the component value is invalid. Default is `true`.\n   */\n  readonly shouldShowValidationMessage?: boolean;\n}\n\n/**\n * Extended options for `PluginSettingsTabBase.bind`.\n */\nexport interface BindOptionsExtended<\n  PluginSettings extends object,\n  UIValue,\n  PropertyName extends StringKeys<PluginSettings>\n> extends BindOptions<PluginSettings[PropertyName]> {\n  /**\n   * Converts the UI component's value back to the plugin settings value.\n   *\n   * @param uiValue - The value of the UI component.\n   * @returns The value to set on the plugin settings.\n   */\n  readonly componentToPluginSettingsValueConverter: (uiValue: UIValue) => PluginSettings[PropertyName] | ValidationMessageHolder;\n\n  /**\n   * Converts the plugin settings value to the value used by the UI component.\n   *\n   * @param pluginSettingsValue - The value of the property in the plugin settings.\n   * @returns The value to set on the UI component.\n   */\n  readonly pluginSettingsToComponentValueConverter: (pluginSettingsValue: ReadonlyDeep<PluginSettings[PropertyName]>) => UIValue;\n}\n\n/**\n * Params for creating a {@link PluginSettingsTabBase}.\n */\nexport interface PluginSettingsTabBaseParams<PluginSettings extends object> {\n  /**\n   * The plugin instance (needed by Obsidian's PluginSettingTab).\n   */\n  readonly plugin: Plugin;\n\n  /**\n   * The settings component.\n   */\n  readonly pluginSettingsComponent: PluginSettingsComponentBase<PluginSettings>;\n}\n\n/**\n * Base class for creating plugin settings tabs in Obsidian.\n * Provides a method for binding value components to plugin settings and handling changes.\n *\n * @typeParam PluginSettings - The plugin settings type.\n */\nexport abstract class PluginSettingsTabBase<PluginSettings extends object> extends PluginSettingTab {\n  /**\n   * Whether the plugin settings tab is open.\n   *\n   * @returns Whether the plugin settings tab is open.\n   */\n  public get isOpen(): boolean {\n    return this._isOpen;\n  }\n\n  /**\n   * The settings manager.\n   */\n  protected readonly pluginSettingsComponent: PluginSettingsComponentBase<PluginSettings>;\n\n  /**\n   * A debounce timeout for saving settings.\n   *\n   * @returns The debounce timeout for saving settings.\n   */\n  protected get saveSettingsDebounceTimeoutInMilliseconds(): number {\n    const DEFAULT = 2_000;\n    return DEFAULT;\n  }\n\n  private _isOpen = false;\n  private readonly asyncEventsComponent: AsyncEventsComponent;\n  private readonly saveSettingsDebounced: Debouncer<[], void>;\n\n  private get pluginSettings(): ReadonlyPluginSettings<PluginSettings> {\n    return this.pluginSettingsComponent.settingsState.inputValues;\n  }\n\n  /**\n   * Creates a new plugin settings tab.\n   *\n   * @param params - The params.\n   */\n  public constructor(params: PluginSettingsTabBaseParams<PluginSettings>) {\n    super(params.plugin.app, params.plugin);\n    this.pluginSettingsComponent = params.pluginSettingsComponent;\n    addPluginCssClasses(this.containerEl, CssClass.PluginSettingsTab);\n    this.saveSettingsDebounced = debounce(\n      convertAsyncToSync(() => this.pluginSettingsComponent.saveToFile(SAVE_TO_FILE_CONTEXT)),\n      this.saveSettingsDebounceTimeoutInMilliseconds\n    );\n    this.asyncEventsComponent = new AsyncEventsComponent();\n  }\n\n  /**\n   * Binds a value component to a plugin setting.\n   *\n   * @typeParam UIValue - The type of the value of the UI component.\n   * @typeParam TValueComponent - The type of the value component.\n   * @param valueComponent - The value component to bind.\n   * @param propertyName - The property of the plugin settings to bind to.\n   * @param options - The options for binding the value component.\n   * @returns The value component.\n   */\n  public bind<\n    UIValue,\n    TValueComponent\n  >(\n    valueComponent: TValueComponent & ValueComponentWithChangeTracking<UIValue>,\n    propertyName: ConditionalKeys<PluginSettings, UIValue>,\n    options?: BindOptions<UIValue>\n  ): TValueComponent;\n  /**\n   * Binds a value component to a plugin setting.\n   *\n   * @typeParam UIValue - The type of the value of the UI component.\n   * @typeParam TValueComponent - The type of the value component.\n   * @typeParam PropertyName - The property name of the plugin settings to bind to.\n   * @param valueComponent - The value component to bind.\n   * @param propertyName - The property name of the plugin settings to bind to.\n   * @param options - The options for binding the value component.\n   * @returns The value component.\n   */\n  public bind<\n    UIValue,\n    TValueComponent,\n    PropertyName extends StringKeys<PluginSettings>\n  >(\n    valueComponent: TValueComponent & ValueComponentWithChangeTracking<UIValue>,\n    propertyName: PropertyName,\n    options: BindOptionsExtended<PluginSettings, UIValue, PropertyName>\n  ): TValueComponent;\n  /**\n   * Binds a value component to a plugin setting.\n   *\n   * @typeParam UIValue - The type of the value of the UI component.\n   * @typeParam TValueComponent - The type of the value component.\n   * @typeParam PropertyName - The property name of the plugin settings to bind to.\n   * @param valueComponent - The value component to bind.\n   * @param propertyName - The property name of the plugin settings to bind to.\n   * @param options - The options for binding the value component.\n   * @returns The value component.\n   */\n  public bind<\n    UIValue,\n    TValueComponent,\n    PropertyName extends StringKeys<PluginSettings>\n  >(\n    valueComponent: TValueComponent & ValueComponentWithChangeTracking<UIValue>,\n    propertyName: PropertyName,\n    options?: BindOptions<PluginSettings[PropertyName]>\n  ): TValueComponent {\n    type PropertyType = PluginSettings[PropertyName];\n    const DEFAULT_OPTIONS: Required<BindOptionsExtended<PluginSettings, UIValue, PropertyName>> = {\n      componentToPluginSettingsValueConverter: (value: UIValue): PropertyType => value as PropertyType,\n      onChanged: noop,\n      pluginSettingsToComponentValueConverter: (value: ReadonlyDeep<PropertyType>): UIValue => value as UIValue,\n      shouldResetSettingWhenComponentIsEmpty: true,\n      shouldShowPlaceholderForDefaultValues: true,\n      shouldShowValidationMessage: true\n    };\n\n    const optionsExt: Required<BindOptionsExtended<PluginSettings, UIValue, PropertyName>> = { ...DEFAULT_OPTIONS, ...options };\n\n    const validatorEl = getValidatorComponent(valueComponent)?.validatorEl;\n\n    const textBasedComponent = getTextBasedComponentValue(valueComponent);\n\n    const readonlyValue = this.getPluginSettingsProperty(propertyName);\n    const defaults = this.pluginSettingsComponent.defaultSettings as PluginSettings;\n    const defaultValue = defaults[propertyName];\n    const defaultComponentValue = optionsExt.pluginSettingsToComponentValueConverter(defaultValue as ReadonlyDeep<PropertyType>);\n    textBasedComponent?.setPlaceholderValue(defaultComponentValue);\n\n    let validationMessage: string;\n    let tooltipEl: HTMLElement | null = null;\n    let tooltipContentEl: HTMLElement | null = null;\n    if (validatorEl) {\n      const wrapper = ensureWrapped(validatorEl);\n      tooltipEl = wrapper.createDiv();\n      addPluginCssClasses(tooltipEl, CssClass.Tooltip, CssClass.TooltipValidator);\n      tooltipContentEl = tooltipEl.createSpan();\n      const tooltipArrowEl = tooltipEl.createDiv();\n      addPluginCssClasses(tooltipArrowEl, CssClass.TooltipArrow);\n      tooltipEl.hide();\n      wrapper.appendChild(tooltipEl);\n    }\n\n    this.asyncEventsComponent.registerAsyncEvent(this.on('validationMessageChanged', (anotherPropertyName, anotherValidationMessage) => {\n      if (propertyName !== anotherPropertyName) {\n        return;\n      }\n\n      validationMessage = anotherValidationMessage;\n      updateValidatorElDebounced();\n    }));\n\n    let shouldEmptyOnBlur = false;\n    let shouldRevertToDefaultValueOnBlur = false;\n\n    if (textBasedComponent && optionsExt.shouldShowPlaceholderForDefaultValues && deepEqual(readonlyValue, defaultValue)) {\n      textBasedComponent.empty();\n    } else {\n      valueComponent.setValue(optionsExt.pluginSettingsToComponentValueConverter(readonlyValue));\n    }\n\n    let shouldSkipOnChange = false;\n    const UPDATE_VALIDATOR_EL_TIMEOUT_IN_MILLISECONDS = 100;\n    const updateValidatorElDebounced = debounce(() => {\n      requestAnimationFrame(() => {\n        updateValidatorEl();\n      });\n    }, UPDATE_VALIDATOR_EL_TIMEOUT_IN_MILLISECONDS);\n\n    valueComponent.onChange(async (uiValue) => {\n      if (shouldSkipOnChange) {\n        shouldSkipOnChange = false;\n        return;\n      }\n\n      shouldEmptyOnBlur = false;\n\n      const oldValue = this.getPluginSettingsProperty(propertyName);\n      let newValue: PropertyType | undefined = undefined;\n      let shouldSetProperty = true;\n      shouldRevertToDefaultValueOnBlur = !!textBasedComponent?.isEmpty() && optionsExt.shouldResetSettingWhenComponentIsEmpty;\n      if (shouldRevertToDefaultValueOnBlur) {\n        newValue = defaultValue;\n      } else {\n        const convertedValue = optionsExt.componentToPluginSettingsValueConverter(uiValue);\n        if (isValidationMessageHolder(convertedValue)) {\n          validationMessage = convertedValue.validationMessage;\n          shouldSetProperty = false;\n        } else {\n          newValue = convertedValue;\n        }\n      }\n\n      if (shouldSetProperty) {\n        validationMessage = await this.pluginSettingsComponent.setProperty(propertyName, newValue as PluginSettings[PropertyName]);\n        if (textBasedComponent && optionsExt.shouldShowPlaceholderForDefaultValues && !textBasedComponent.isEmpty() && deepEqual(newValue, defaultValue)) {\n          shouldEmptyOnBlur = true;\n        }\n      }\n\n      updateValidatorElDebounced();\n      if (shouldSetProperty) {\n        await optionsExt.onChanged(newValue as ReadonlyDeep<PropertyType>, oldValue);\n      }\n      this.saveSettingsDebounced();\n    });\n\n    validatorEl?.addEventListener('focus', () => {\n      updateValidatorElDebounced();\n    });\n    validatorEl?.addEventListener('blur', () => {\n      updateValidatorElDebounced();\n    });\n    validatorEl?.addEventListener('click', () => {\n      requestAnimationFrame(() => {\n        updateValidatorElDebounced();\n      });\n    });\n\n    const validationMessages = this.pluginSettingsComponent.settingsState.validationMessages as Record<string, string>;\n    validationMessage = validationMessages[propertyName] ?? '';\n    updateValidatorElDebounced();\n\n    return valueComponent;\n\n    function updateValidatorEl(): void {\n      if (!validatorEl?.isActiveElement()) {\n        if (shouldEmptyOnBlur) {\n          shouldEmptyOnBlur = false;\n\n          if (!textBasedComponent?.isEmpty()) {\n            shouldSkipOnChange = true;\n            textBasedComponent?.empty();\n          }\n        } else if (shouldRevertToDefaultValueOnBlur) {\n          shouldRevertToDefaultValueOnBlur = false;\n\n          if (textBasedComponent?.isEmpty()) {\n            shouldSkipOnChange = true;\n            valueComponent.setValue(defaultComponentValue);\n          }\n        }\n      }\n\n      if (!validatorEl) {\n        return;\n      }\n\n      assertNonNullable(tooltipContentEl);\n\n      if (validationMessage === '') {\n        validatorEl.setCustomValidity('');\n        validatorEl.checkValidity();\n        validationMessage = validatorEl.validationMessage;\n      }\n\n      validatorEl.setCustomValidity(validationMessage);\n      if (optionsExt.shouldShowValidationMessage) {\n        tooltipContentEl.textContent = validationMessage;\n        tooltipEl?.toggle(!!validationMessage);\n      } else if (validationMessage) {\n        setTooltip(validatorEl, validationMessage);\n      }\n    }\n  }\n\n  /**\n   * Renders the plugin settings tab.\n   */\n  public override display(): void {\n    this.containerEl.empty();\n    this._isOpen = true;\n    this.asyncEventsComponent.load();\n    this.asyncEventsComponent.registerAsyncEvent(this.pluginSettingsComponent.on('loadSettings', this.onLoadSettings.bind(this)));\n    this.asyncEventsComponent.registerAsyncEvent(this.pluginSettingsComponent.on('saveSettings', this.onSaveSettings.bind(this)));\n  }\n\n  /**\n   * Hides the plugin settings tab.\n   */\n  public override hide(): void {\n    super.hide();\n    this.saveSettingsDebounced.cancel();\n    this._isOpen = false;\n    this.asyncEventsComponent.unload();\n    this.asyncEventsComponent.load();\n    invokeAsyncSafely(() => this.hideAsync());\n  }\n\n  /**\n   * Async actions to perform when the settings tab is being hidden.\n   *\n   * @returns A {@link Promise} that resolves when the settings tab is hidden.\n   */\n  public async hideAsync(): Promise<void> {\n    await this.pluginSettingsComponent.saveToFile(SAVE_TO_FILE_CONTEXT);\n  }\n\n  /**\n   * Shows the plugin settings tab.\n   */\n  public show(): void {\n    this.app.setting.openTab(this);\n  }\n\n  /**\n   * Called when the plugin settings are loaded.\n   *\n   * @param _loadedState - The loaded settings state.\n   * @param _isInitialLoad - Whether the settings are being loaded for the first time.\n   * @returns A {@link Promise} that resolves when the settings are loaded.\n   */\n  protected async onLoadSettings(_loadedState: ReadonlyPluginSettingsState<PluginSettings>, _isInitialLoad: boolean): Promise<void> {\n    this.display();\n    await noopAsync();\n  }\n\n  /**\n   * Revalidates the settings.\n   *\n   * @returns A {@link Promise} that resolves when the settings are revalidated.\n   */\n  protected async revalidate(): Promise<void> {\n    const validationMessages = await this.pluginSettingsComponent.revalidate();\n    await this.updateValidations(validationMessages);\n  }\n\n  private getPluginSettingsProperty<PropertyName extends StringKeys<PluginSettings>>(\n    propertyName: PropertyName\n  ): ReadonlyDeep<PluginSettings[PropertyName]> {\n    const settings = this.pluginSettings as PluginSettings;\n    return settings[propertyName] as ReadonlyDeep<PluginSettings[PropertyName]>;\n  }\n\n  private on(\n    name: 'validationMessageChanged',\n    callback: (\n      propertyName: string,\n      validationMessage: string\n    ) => Promisable<void>,\n    thisArg?: unknown\n  ): AsyncEventRef;\n  private on<Args extends unknown[]>(\n    name: string,\n    callback: (...args: Args) => Promisable<void>,\n    thisArg?: unknown\n  ): AsyncEventRef {\n    return this.asyncEventsComponent.asyncEvents.on(name, callback, thisArg);\n  }\n\n  private async onSaveSettings(\n    newState: ReadonlyPluginSettingsState<PluginSettings>,\n    _oldState: ReadonlyPluginSettingsState<PluginSettings>,\n    context: unknown\n  ): Promise<void> {\n    if (context === SAVE_TO_FILE_CONTEXT) {\n      await this.updateValidations(newState.validationMessages as Record<StringKeys<PluginSettings>, string>);\n      return;\n    }\n\n    this.display();\n  }\n\n  private async updateValidations(validationMessages: Record<StringKeys<PluginSettings>, string>): Promise<void> {\n    for (const [propertyName, validationMessage] of Object.entries(validationMessages)) {\n      await this.asyncEventsComponent.asyncEvents.triggerAsync('validationMessageChanged', propertyName, validationMessage);\n    }\n  }\n}\n"],
  "mappings": "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",
  "names": []
}

@@ -134,7 +134,6 @@ var import_obsidian = require('obsidian');
134
134
  var import_async = require('../../async.cjs');
135
135
  var import_error = require('../../error.cjs');
136
136
  var import_function = require('../../function.cjs');
137
- var import_type_guards = require('../../type-guards.cjs');
138
137
  var import_abort_signal_component = require('./components/abort-signal-component.cjs');
139
138
  var import_async_error_handler_component = require('./components/async-error-handler-component.cjs');
140
139
  var import_console_debug_component = require('./components/console-debug-component.cjs');
@@ -164,8 +163,8 @@ class PluginBase extends import_obsidian.Plugin {
164
163
  * The settings component. Manages plugin settings lifecycle.
165
164
  */
166
165
  settingsComponent;
167
- componentsMap = /* @__PURE__ */ new Map();
168
- preloadKeys = /* @__PURE__ */ new Set();
166
+ preloadComponents = [];
167
+ singletonComponents = /* @__PURE__ */ new Map();
169
168
  /**
170
169
  * Creates a new PluginBase.
171
170
  *
@@ -200,9 +199,7 @@ class PluginBase extends import_obsidian.Plugin {
200
199
  */
201
200
  async onload() {
202
201
  await super.onload();
203
- for (const key of this.preloadKeys) {
204
- const component = this.componentsMap.get(key);
205
- (0, import_type_guards.assertNonNullable)(component, `Component with key '${key}' not found`);
202
+ for (const component of this.preloadComponents) {
206
203
  await component.load();
207
204
  }
208
205
  await this.onloadImpl();
@@ -223,25 +220,32 @@ class PluginBase extends import_obsidian.Plugin {
223
220
  await (0, import_function.noopAsync)();
224
221
  }
225
222
  /**
226
- * Registers a component with the plugin. If a component with the same key is already registered,
227
- * the old component is removed and replaced.
223
+ * Registers a component with the plugin.
224
+ *
225
+ * If the component's class defines a static `COMPONENT_KEY` symbol, it is treated as a singleton —
226
+ * registering another component with the same key replaces the previous one.
227
+ * Components without a `COMPONENT_KEY` are multi-instance and simply added.
228
228
  *
229
229
  * @typeParam T - The component type.
230
230
  * @param params - The registration params.
231
231
  * @returns The registered component.
232
232
  */
233
233
  registerComponent(params) {
234
- const resolvedKey = params.key ?? params.component.constructor.name;
235
- const oldComponent = this.componentsMap.get(resolvedKey);
236
- if (oldComponent) {
237
- this.removeChild(oldComponent);
234
+ const singletonKey = params.component.constructor.COMPONENT_KEY;
235
+ if (singletonKey) {
236
+ const oldComponent = this.singletonComponents.get(singletonKey);
237
+ if (oldComponent) {
238
+ this.removeChild(oldComponent);
239
+ const preloadIndex = this.preloadComponents.indexOf(oldComponent);
240
+ if (preloadIndex !== -1) {
241
+ this.preloadComponents.splice(preloadIndex, 1);
242
+ }
243
+ }
244
+ this.singletonComponents.set(singletonKey, params.component);
238
245
  }
239
- this.componentsMap.set(resolvedKey, params.component);
240
246
  this.addChild(params.component);
241
247
  if (params.shouldPreload) {
242
- this.preloadKeys.add(resolvedKey);
243
- } else {
244
- this.preloadKeys.delete(resolvedKey);
248
+ this.preloadComponents.push(params.component);
245
249
  }
246
250
  return params.component;
247
251
  }
@@ -277,4 +281,4 @@ async function showErrorAndDisablePlugin(plugin, message) {
277
281
  reloadPlugin,
278
282
  showErrorAndDisablePlugin
279
283
  });
280
- //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../../../src/obsidian/plugin/plugin.ts"],
  "sourcesContent": ["/**\n * @file\n *\n * Base class for Obsidian plugins using a component-based architecture.\n *\n * PluginBase registers universal components (context, i18n, error handling, abort signal, lifecycle events, debug).\n * Subclasses add their own components via {@link registerComponent} in their constructor.\n * Any universal component can be replaced by calling {@link registerComponent} with a new instance of the same class.\n */\n\nimport type {\n  App,\n  PluginManifest\n} from 'obsidian';\nimport type { Promisable } from 'type-fest';\n\nimport {\n  Component,\n  Notice,\n  Plugin as ObsidianPlugin\n} from 'obsidian';\n\nimport type { PluginSettingsComponentBase } from './components/plugin-settings-component.ts';\n\nimport {\n  convertAsyncToSync,\n  invokeAsyncSafelyAfterDelay\n} from '../../async.ts';\nimport { printError } from '../../error.ts';\nimport { noopAsync } from '../../function.ts';\nimport { assertNonNullable } from '../../type-guards.ts';\nimport { AbortSignalComponent } from './components/abort-signal-component.ts';\nimport { AsyncErrorHandlerComponent } from './components/async-error-handler-component.ts';\nimport { ConsoleDebugComponent } from './components/console-debug-component.ts';\nimport { I18nComponent } from './components/i18n-component.ts';\nimport { LifecycleEventsComponent } from './components/lifecycle-events-component.ts';\nimport { PluginContextComponent } from './components/plugin-context-component.ts';\nimport { PluginNoticeComponent } from './components/plugin-notice-component.ts';\nimport { EmptyPluginSettingsComponent } from './components/plugin-settings-component.ts';\n\n/**\n * Params for {@link PluginBase.registerComponent}.\n *\n * @typeParam T - The component type.\n */\nexport interface RegisterComponentParams<T extends Component = Component> {\n  /**\n   * The component to register.\n   */\n  readonly component: T;\n\n  /**\n   * Optional key. Defaults to the component's constructor name.\n   */\n  readonly key?: string;\n\n  /**\n   * Whether this component should be loaded before the plugin's `onloadImpl()` runs.\n   * Components marked with this flag are force-loaded during `onload()`, ensuring they are\n   * fully initialized before plugin logic executes.\n   */\n  readonly shouldPreload?: boolean;\n}\n\n/**\n * Base class for creating Obsidian plugins with a component-based architecture.\n *\n * Registers universal components automatically. Subclasses add or replace components\n * via {@link registerComponent} in their constructor.\n */\nexport abstract class PluginBase extends ObsidianPlugin {\n  /**\n   * The abort signal component. Aborted when the plugin is unloaded.\n   */\n  protected readonly abortSignalComponent: AbortSignalComponent;\n\n  /**\n   * The console debug component. Provides namespaced debug logging.\n   */\n  protected readonly consoleDebugComponent: ConsoleDebugComponent;\n\n  /**\n   * The lifecycle events component. Provides load, layoutReady, and unload events.\n   */\n  protected readonly lifecycleEventsComponent: LifecycleEventsComponent;\n\n  /**\n   * The notice component. Displays notices to the user.\n   */\n  protected readonly noticeComponent: PluginNoticeComponent;\n\n  /**\n   * The settings component. Manages plugin settings lifecycle.\n   */\n  protected readonly settingsComponent: PluginSettingsComponentBase<object>;\n\n  private readonly componentsMap = new Map<string, Component>();\n  private readonly preloadKeys = new Set<string>();\n\n  /**\n   * Creates a new PluginBase.\n   *\n   * @param app - The Obsidian app instance.\n   * @param manifest - The plugin manifest.\n   */\n  public constructor(app: App, manifest: PluginManifest) {\n    super(app, manifest);\n\n    this.registerComponent({ component: new PluginContextComponent(app, manifest.id), shouldPreload: true });\n    this.registerComponent({ component: new I18nComponent(), shouldPreload: true });\n    this.noticeComponent = this.registerComponent({ component: new PluginNoticeComponent(manifest.name) });\n    this.registerComponent({ component: new AsyncErrorHandlerComponent(this.noticeComponent) });\n    this.abortSignalComponent = this.registerComponent({ component: new AbortSignalComponent(manifest.id) });\n    this.consoleDebugComponent = this.registerComponent({ component: new ConsoleDebugComponent(manifest.id) });\n    this.lifecycleEventsComponent = this.registerComponent({ component: new LifecycleEventsComponent(app) });\n    this.settingsComponent = this.registerComponent({ component: new EmptyPluginSettingsComponent(), shouldPreload: true });\n  }\n\n  /**\n   * Called when the external settings change.\n   *\n   * Override in subclass if needed. Make sure to call `await super.onExternalSettingsChange()` first.\n   */\n  public override async onExternalSettingsChange(): Promise<void> {\n    await super.onExternalSettingsChange?.();\n    await this.settingsComponent.onExternalSettingsChange();\n  }\n\n  /**\n   * Called when the plugin is loaded. Force-loads components that must be ready\n   * before plugin logic, then calls `onloadImpl()`.\n   *\n   * Usually, you don't need to override this method. Override {@link onloadImpl} instead.\n   */\n  public override async onload(): Promise<void> {\n    await super.onload();\n\n    for (const key of this.preloadKeys) {\n      const component = this.componentsMap.get(key);\n      assertNonNullable(component, `Component with key '${key}' not found`);\n      // eslint-disable-next-line @typescript-eslint/no-confusing-void-expression -- Component.load() returns void|Promise at runtime despite void typing.\n      await (component.load() as Promisable<void>);\n    }\n\n    await this.onloadImpl();\n    invokeAsyncSafelyAfterDelay(this.afterLoad.bind(this));\n  }\n\n  /**\n   * Called when the layout is ready.\n   */\n  protected async onLayoutReady(): Promise<void> {\n    await noopAsync();\n  }\n\n  /**\n   * Called after all pre-load components are initialized. Override this to add plugin-specific logic.\n   *\n   * @remarks It is important to call `super.onloadImpl()` in overridden method.\n   */\n  protected async onloadImpl(): Promise<void> {\n    await noopAsync();\n  }\n\n  /**\n   * Registers a component with the plugin. If a component with the same key is already registered,\n   * the old component is removed and replaced.\n   *\n   * @typeParam T - The component type.\n   * @param params - The registration params.\n   * @returns The registered component.\n   */\n  protected registerComponent<T extends Component>(params: RegisterComponentParams<T>): T {\n    const resolvedKey = params.key ?? params.component.constructor.name;\n    const oldComponent = this.componentsMap.get(resolvedKey);\n    if (oldComponent) {\n      this.removeChild(oldComponent);\n    }\n    this.componentsMap.set(resolvedKey, params.component);\n    this.addChild(params.component);\n\n    if (params.shouldPreload) {\n      this.preloadKeys.add(resolvedKey);\n    } else {\n      this.preloadKeys.delete(resolvedKey);\n    }\n\n    return params.component;\n  }\n\n  private async afterLoad(): Promise<void> {\n    if (this.abortSignalComponent.abortSignal.aborted) {\n      return;\n    }\n    await this.lifecycleEventsComponent.triggerLifecycleEvent('load');\n    this.app.workspace.onLayoutReady(convertAsyncToSync(this.onLayoutReadyBase.bind(this)));\n  }\n\n  private async onLayoutReadyBase(): Promise<void> {\n    try {\n      await this.onLayoutReady();\n    } finally {\n      await this.lifecycleEventsComponent.triggerLifecycleEvent('layoutReady');\n    }\n  }\n}\n\n/**\n * Reloads the specified plugin by disabling and then re-enabling it.\n *\n * @param plugin - The plugin to reload.\n * @returns A {@link Promise} that resolves when the plugin is reloaded.\n */\nexport async function reloadPlugin(plugin: ObsidianPlugin): Promise<void> {\n  const plugins = plugin.app.plugins;\n  const pluginId = plugin.manifest.id;\n  await plugins.disablePlugin(pluginId);\n  await plugins.enablePlugin(pluginId);\n}\n\n/**\n * Displays an error message as a notice, logs it to the console, and disables the specified plugin.\n *\n * @param plugin - The plugin to disable.\n * @param message - The error message to display and log.\n * @returns A {@link Promise} that resolves when the plugin is disabled.\n */\nexport async function showErrorAndDisablePlugin(plugin: ObsidianPlugin, message: string): Promise<void> {\n  new Notice(message);\n  printError(new Error(message));\n  await plugin.app.plugins.disablePlugin(plugin.manifest.id);\n}\n"],
  "mappings": "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",
  "names": ["ObsidianPlugin"]
}

284
+ //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../../../src/obsidian/plugin/plugin.ts"],
  "sourcesContent": ["/**\n * @file\n *\n * Base class for Obsidian plugins using a component-based architecture.\n *\n * PluginBase registers universal components (context, i18n, error handling, abort signal, lifecycle events, debug).\n * Subclasses add their own components via {@link registerComponent} in their constructor.\n * Any universal component can be replaced by calling {@link registerComponent} with a new instance of the same class.\n */\n\nimport type {\n  App,\n  PluginManifest\n} from 'obsidian';\nimport type { Promisable } from 'type-fest';\n\nimport {\n  Component,\n  Notice,\n  Plugin as ObsidianPlugin\n} from 'obsidian';\n\nimport type { PluginSettingsComponentBase } from './components/plugin-settings-component.ts';\n\nimport {\n  convertAsyncToSync,\n  invokeAsyncSafelyAfterDelay\n} from '../../async.ts';\nimport { printError } from '../../error.ts';\nimport { noopAsync } from '../../function.ts';\nimport { AbortSignalComponent } from './components/abort-signal-component.ts';\nimport { AsyncErrorHandlerComponent } from './components/async-error-handler-component.ts';\nimport { ConsoleDebugComponent } from './components/console-debug-component.ts';\nimport { I18nComponent } from './components/i18n-component.ts';\nimport { LifecycleEventsComponent } from './components/lifecycle-events-component.ts';\nimport { PluginContextComponent } from './components/plugin-context-component.ts';\nimport { PluginNoticeComponent } from './components/plugin-notice-component.ts';\nimport { EmptyPluginSettingsComponent } from './components/plugin-settings-component.ts';\n\n/**\n * Params for {@link PluginBase.registerComponent}.\n *\n * @typeParam T - The component type.\n */\nexport interface RegisterComponentParams<T extends Component = Component> {\n  /**\n   * The component to register.\n   */\n  readonly component: T;\n\n  /**\n   * Whether this component should be loaded before the plugin's `onloadImpl()` runs.\n   * Components marked with this flag are force-loaded during `onload()`, ensuring they are\n   * fully initialized before plugin logic executes.\n   */\n  readonly shouldPreload?: boolean;\n}\n\ninterface ComponentClassWithKey {\n  COMPONENT_KEY: symbol;\n}\n\n/**\n * Base class for creating Obsidian plugins with a component-based architecture.\n *\n * Registers universal components automatically. Subclasses add or replace components\n * via {@link registerComponent} in their constructor.\n */\nexport abstract class PluginBase extends ObsidianPlugin {\n  /**\n   * The abort signal component. Aborted when the plugin is unloaded.\n   */\n  protected readonly abortSignalComponent: AbortSignalComponent;\n\n  /**\n   * The console debug component. Provides namespaced debug logging.\n   */\n  protected readonly consoleDebugComponent: ConsoleDebugComponent;\n\n  /**\n   * The lifecycle events component. Provides load, layoutReady, and unload events.\n   */\n  protected readonly lifecycleEventsComponent: LifecycleEventsComponent;\n\n  /**\n   * The notice component. Displays notices to the user.\n   */\n  protected readonly noticeComponent: PluginNoticeComponent;\n\n  /**\n   * The settings component. Manages plugin settings lifecycle.\n   */\n  protected readonly settingsComponent: PluginSettingsComponentBase<object>;\n\n  private readonly preloadComponents: Component[] = [];\n  private readonly singletonComponents = new Map<symbol, Component>();\n\n  /**\n   * Creates a new PluginBase.\n   *\n   * @param app - The Obsidian app instance.\n   * @param manifest - The plugin manifest.\n   */\n  public constructor(app: App, manifest: PluginManifest) {\n    super(app, manifest);\n\n    this.registerComponent({ component: new PluginContextComponent(app, manifest.id), shouldPreload: true });\n    this.registerComponent({ component: new I18nComponent(), shouldPreload: true });\n    this.noticeComponent = this.registerComponent({ component: new PluginNoticeComponent(manifest.name) });\n    this.registerComponent({ component: new AsyncErrorHandlerComponent(this.noticeComponent) });\n    this.abortSignalComponent = this.registerComponent({ component: new AbortSignalComponent(manifest.id) });\n    this.consoleDebugComponent = this.registerComponent({ component: new ConsoleDebugComponent(manifest.id) });\n    this.lifecycleEventsComponent = this.registerComponent({ component: new LifecycleEventsComponent(app) });\n    this.settingsComponent = this.registerComponent({ component: new EmptyPluginSettingsComponent(), shouldPreload: true });\n  }\n\n  /**\n   * Called when the external settings change.\n   *\n   * Override in subclass if needed. Make sure to call `await super.onExternalSettingsChange()` first.\n   */\n  public override async onExternalSettingsChange(): Promise<void> {\n    await super.onExternalSettingsChange?.();\n    await this.settingsComponent.onExternalSettingsChange();\n  }\n\n  /**\n   * Called when the plugin is loaded. Force-loads components that must be ready\n   * before plugin logic, then calls `onloadImpl()`.\n   *\n   * Usually, you don't need to override this method. Override {@link onloadImpl} instead.\n   */\n  public override async onload(): Promise<void> {\n    await super.onload();\n\n    for (const component of this.preloadComponents) {\n      // eslint-disable-next-line @typescript-eslint/no-confusing-void-expression -- Component.load() returns void|Promise at runtime despite void typing.\n      await (component.load() as Promisable<void>);\n    }\n\n    await this.onloadImpl();\n    invokeAsyncSafelyAfterDelay(this.afterLoad.bind(this));\n  }\n\n  /**\n   * Called when the layout is ready.\n   */\n  protected async onLayoutReady(): Promise<void> {\n    await noopAsync();\n  }\n\n  /**\n   * Called after all pre-load components are initialized. Override this to add plugin-specific logic.\n   *\n   * @remarks It is important to call `super.onloadImpl()` in overridden method.\n   */\n  protected async onloadImpl(): Promise<void> {\n    await noopAsync();\n  }\n\n  /**\n   * Registers a component with the plugin.\n   *\n   * If the component's class defines a static `COMPONENT_KEY` symbol, it is treated as a singleton \u2014\n   * registering another component with the same key replaces the previous one.\n   * Components without a `COMPONENT_KEY` are multi-instance and simply added.\n   *\n   * @typeParam T - The component type.\n   * @param params - The registration params.\n   * @returns The registered component.\n   */\n  protected registerComponent<T extends Component>(params: RegisterComponentParams<T>): T {\n    const singletonKey = (params.component.constructor as Partial<ComponentClassWithKey>).COMPONENT_KEY;\n\n    if (singletonKey) {\n      const oldComponent = this.singletonComponents.get(singletonKey);\n      if (oldComponent) {\n        this.removeChild(oldComponent);\n        const preloadIndex = this.preloadComponents.indexOf(oldComponent);\n        if (preloadIndex !== -1) {\n          this.preloadComponents.splice(preloadIndex, 1);\n        }\n      }\n      this.singletonComponents.set(singletonKey, params.component);\n    }\n\n    this.addChild(params.component);\n\n    if (params.shouldPreload) {\n      this.preloadComponents.push(params.component);\n    }\n\n    return params.component;\n  }\n\n  private async afterLoad(): Promise<void> {\n    if (this.abortSignalComponent.abortSignal.aborted) {\n      return;\n    }\n    await this.lifecycleEventsComponent.triggerLifecycleEvent('load');\n    this.app.workspace.onLayoutReady(convertAsyncToSync(this.onLayoutReadyBase.bind(this)));\n  }\n\n  private async onLayoutReadyBase(): Promise<void> {\n    try {\n      await this.onLayoutReady();\n    } finally {\n      await this.lifecycleEventsComponent.triggerLifecycleEvent('layoutReady');\n    }\n  }\n}\n\n/**\n * Reloads the specified plugin by disabling and then re-enabling it.\n *\n * @param plugin - The plugin to reload.\n * @returns A {@link Promise} that resolves when the plugin is reloaded.\n */\nexport async function reloadPlugin(plugin: ObsidianPlugin): Promise<void> {\n  const plugins = plugin.app.plugins;\n  const pluginId = plugin.manifest.id;\n  await plugins.disablePlugin(pluginId);\n  await plugins.enablePlugin(pluginId);\n}\n\n/**\n * Displays an error message as a notice, logs it to the console, and disables the specified plugin.\n *\n * @param plugin - The plugin to disable.\n * @param message - The error message to display and log.\n * @returns A {@link Promise} that resolves when the plugin is disabled.\n */\nexport async function showErrorAndDisablePlugin(plugin: ObsidianPlugin, message: string): Promise<void> {\n  new Notice(message);\n  printError(new Error(message));\n  await plugin.app.plugins.disablePlugin(plugin.manifest.id);\n}\n"],
  "mappings": "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",
  "names": ["ObsidianPlugin"]
}
