obsidian-dev-utils 22.1.1-beta.2 → 22.1.1-beta.21

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 (135) hide show
  1. package/CHANGELOG.md +113 -0
  2. package/dist/dprint.json +1 -0
  3. package/dist/lib/cjs/Async.cjs +1 -1
  4. package/dist/lib/cjs/Async.d.cts +17 -17
  5. package/dist/lib/cjs/Library.cjs +1 -1
  6. package/dist/lib/cjs/Object.cjs +38 -3
  7. package/dist/lib/cjs/Object.d.cts +16 -6
  8. package/dist/lib/cjs/ScriptUtils/CliUtils.cjs +7 -8
  9. package/dist/lib/cjs/ScriptUtils/CliUtils.d.cts +11 -10
  10. package/dist/lib/cjs/ScriptUtils/Exec.cjs +1 -1
  11. package/dist/lib/cjs/ScriptUtils/Exec.d.cts +2 -2
  12. package/dist/lib/cjs/ScriptUtils/Root.cjs +1 -1
  13. package/dist/lib/cjs/ScriptUtils/Root.d.cts +2 -2
  14. package/dist/lib/cjs/ScriptUtils/cli.cjs +1 -1
  15. package/dist/lib/cjs/ScriptUtils/esbuild/Dependency.cjs +1 -1
  16. package/dist/lib/cjs/ScriptUtils/esbuild/Dependency.d.cts +2 -2
  17. package/dist/lib/cjs/ScriptUtils/esbuild/ObsidianPluginBuilder.cjs +1 -1
  18. package/dist/lib/cjs/ScriptUtils/esbuild/ObsidianPluginBuilder.d.cts +2 -2
  19. package/dist/lib/cjs/ScriptUtils/spellcheck.cjs +1 -1
  20. package/dist/lib/cjs/ScriptUtils/spellcheck.d.cts +2 -2
  21. package/dist/lib/cjs/ScriptUtils/version.cjs +1 -1
  22. package/dist/lib/cjs/ScriptUtils/version.d.cts +2 -2
  23. package/dist/lib/cjs/String.cjs +1 -1
  24. package/dist/lib/cjs/String.d.cts +2 -1
  25. package/dist/lib/cjs/Transformers/Transformer.cjs +9 -2
  26. package/dist/lib/cjs/Type.cjs +24 -0
  27. package/dist/lib/cjs/Type.d.cts +12 -0
  28. package/dist/lib/cjs/index.cjs +4 -1
  29. package/dist/lib/cjs/index.d.cts +1 -0
  30. package/dist/lib/cjs/obsidian/Backlink.cjs +1 -1
  31. package/dist/lib/cjs/obsidian/Backlink.d.cts +1 -1
  32. package/dist/lib/cjs/obsidian/Callout.cjs +1 -1
  33. package/dist/lib/cjs/obsidian/Callout.d.cts +2 -1
  34. package/dist/lib/cjs/obsidian/Components/MultipleTextComponent.cjs +29 -2
  35. package/dist/lib/cjs/obsidian/Components/MultipleTextComponent.d.cts +20 -1
  36. package/dist/lib/cjs/obsidian/Components/TextBasedComponent.cjs +60 -0
  37. package/dist/lib/cjs/obsidian/Components/TextBasedComponent.d.cts +30 -0
  38. package/dist/lib/cjs/obsidian/Components/TypedTextComponent.cjs +27 -3
  39. package/dist/lib/cjs/obsidian/Components/TypedTextComponent.d.cts +19 -1
  40. package/dist/lib/cjs/obsidian/Components/index.cjs +4 -1
  41. package/dist/lib/cjs/obsidian/Components/index.d.cts +1 -0
  42. package/dist/lib/cjs/obsidian/Dataview.cjs +1 -1
  43. package/dist/lib/cjs/obsidian/Dataview.d.cts +3 -3
  44. package/dist/lib/cjs/obsidian/DataviewLink.cjs +1 -1
  45. package/dist/lib/cjs/obsidian/DataviewLink.d.cts +2 -2
  46. package/dist/lib/cjs/obsidian/FileManager.cjs +1 -1
  47. package/dist/lib/cjs/obsidian/FileManager.d.cts +2 -1
  48. package/dist/lib/cjs/obsidian/Link.cjs +1 -1
  49. package/dist/lib/cjs/obsidian/Link.d.cts +4 -3
  50. package/dist/lib/cjs/obsidian/Markdown.cjs +5 -8
  51. package/dist/lib/cjs/obsidian/Modals/Prompt.cjs +1 -1
  52. package/dist/lib/cjs/obsidian/Modals/Prompt.d.cts +2 -1
  53. package/dist/lib/cjs/obsidian/MonkeyAround.cjs +1 -1
  54. package/dist/lib/cjs/obsidian/MonkeyAround.d.cts +2 -2
  55. package/dist/lib/cjs/obsidian/Plugin/PluginBase.cjs +42 -4
  56. package/dist/lib/cjs/obsidian/Plugin/PluginBase.d.cts +24 -5
  57. package/dist/lib/cjs/obsidian/Plugin/PluginSettingsManagerBase.cjs +203 -62
  58. package/dist/lib/cjs/obsidian/Plugin/PluginSettingsManagerBase.d.cts +55 -21
  59. package/dist/lib/cjs/obsidian/Plugin/PluginSettingsTabBase.cjs +33 -20
  60. package/dist/lib/cjs/obsidian/Plugin/PluginSettingsTabBase.d.cts +20 -17
  61. package/dist/lib/cjs/obsidian/RenameDeleteHandler.cjs +75 -75
  62. package/dist/lib/cjs/obsidian/ValidationMessage.cjs +38 -0
  63. package/dist/lib/cjs/obsidian/ValidationMessage.d.cts +16 -0
  64. package/dist/lib/cjs/obsidian/Vault.cjs +1 -1
  65. package/dist/lib/cjs/obsidian/Vault.d.cts +1 -1
  66. package/dist/lib/cjs/obsidian/index.cjs +4 -1
  67. package/dist/lib/cjs/obsidian/index.d.cts +1 -0
  68. package/dist/lib/esm/Async.d.mts +17 -17
  69. package/dist/lib/esm/Async.mjs +1 -1
  70. package/dist/lib/esm/Library.mjs +1 -1
  71. package/dist/lib/esm/Object.d.mts +16 -6
  72. package/dist/lib/esm/Object.mjs +36 -3
  73. package/dist/lib/esm/ScriptUtils/CliUtils.d.mts +11 -10
  74. package/dist/lib/esm/ScriptUtils/CliUtils.mjs +7 -8
  75. package/dist/lib/esm/ScriptUtils/Exec.d.mts +2 -2
  76. package/dist/lib/esm/ScriptUtils/Exec.mjs +1 -1
  77. package/dist/lib/esm/ScriptUtils/Root.d.mts +2 -2
  78. package/dist/lib/esm/ScriptUtils/Root.mjs +1 -1
  79. package/dist/lib/esm/ScriptUtils/cli.mjs +1 -1
  80. package/dist/lib/esm/ScriptUtils/esbuild/Dependency.d.mts +2 -2
  81. package/dist/lib/esm/ScriptUtils/esbuild/Dependency.mjs +1 -1
  82. package/dist/lib/esm/ScriptUtils/esbuild/ObsidianPluginBuilder.d.mts +2 -2
  83. package/dist/lib/esm/ScriptUtils/esbuild/ObsidianPluginBuilder.mjs +1 -1
  84. package/dist/lib/esm/ScriptUtils/spellcheck.d.mts +2 -2
  85. package/dist/lib/esm/ScriptUtils/spellcheck.mjs +1 -1
  86. package/dist/lib/esm/ScriptUtils/version.d.mts +2 -2
  87. package/dist/lib/esm/ScriptUtils/version.mjs +1 -1
  88. package/dist/lib/esm/String.d.mts +2 -1
  89. package/dist/lib/esm/String.mjs +1 -1
  90. package/dist/lib/esm/Transformers/Transformer.mjs +9 -2
  91. package/dist/lib/esm/Type.d.mts +12 -0
  92. package/dist/lib/esm/Type.mjs +8 -0
  93. package/dist/lib/esm/index.d.mts +1 -0
  94. package/dist/lib/esm/index.mjs +3 -1
  95. package/dist/lib/esm/obsidian/Backlink.d.mts +1 -1
  96. package/dist/lib/esm/obsidian/Backlink.mjs +1 -1
  97. package/dist/lib/esm/obsidian/Callout.d.mts +2 -1
  98. package/dist/lib/esm/obsidian/Callout.mjs +1 -1
  99. package/dist/lib/esm/obsidian/Components/MultipleTextComponent.d.mts +20 -1
  100. package/dist/lib/esm/obsidian/Components/MultipleTextComponent.mjs +29 -2
  101. package/dist/lib/esm/obsidian/Components/TextBasedComponent.d.mts +30 -0
  102. package/dist/lib/esm/obsidian/Components/TextBasedComponent.mjs +36 -0
  103. package/dist/lib/esm/obsidian/Components/TypedTextComponent.d.mts +19 -1
  104. package/dist/lib/esm/obsidian/Components/TypedTextComponent.mjs +27 -3
  105. package/dist/lib/esm/obsidian/Components/index.d.mts +1 -0
  106. package/dist/lib/esm/obsidian/Components/index.mjs +3 -1
  107. package/dist/lib/esm/obsidian/Dataview.d.mts +3 -3
  108. package/dist/lib/esm/obsidian/Dataview.mjs +1 -1
  109. package/dist/lib/esm/obsidian/DataviewLink.d.mts +2 -2
  110. package/dist/lib/esm/obsidian/DataviewLink.mjs +1 -1
  111. package/dist/lib/esm/obsidian/FileManager.d.mts +2 -1
  112. package/dist/lib/esm/obsidian/FileManager.mjs +1 -1
  113. package/dist/lib/esm/obsidian/Link.d.mts +4 -3
  114. package/dist/lib/esm/obsidian/Link.mjs +1 -1
  115. package/dist/lib/esm/obsidian/Markdown.mjs +5 -8
  116. package/dist/lib/esm/obsidian/Modals/Prompt.d.mts +2 -1
  117. package/dist/lib/esm/obsidian/Modals/Prompt.mjs +1 -1
  118. package/dist/lib/esm/obsidian/MonkeyAround.d.mts +2 -2
  119. package/dist/lib/esm/obsidian/MonkeyAround.mjs +1 -1
  120. package/dist/lib/esm/obsidian/Plugin/PluginBase.d.mts +24 -5
  121. package/dist/lib/esm/obsidian/Plugin/PluginBase.mjs +42 -4
  122. package/dist/lib/esm/obsidian/Plugin/PluginSettingsManagerBase.d.mts +55 -21
  123. package/dist/lib/esm/obsidian/Plugin/PluginSettingsManagerBase.mjs +203 -63
  124. package/dist/lib/esm/obsidian/Plugin/PluginSettingsTabBase.d.mts +20 -17
  125. package/dist/lib/esm/obsidian/Plugin/PluginSettingsTabBase.mjs +36 -20
  126. package/dist/lib/esm/obsidian/RenameDeleteHandler.mjs +75 -75
  127. package/dist/lib/esm/obsidian/ValidationMessage.d.mts +16 -0
  128. package/dist/lib/esm/obsidian/ValidationMessage.mjs +14 -0
  129. package/dist/lib/esm/obsidian/Vault.d.mts +1 -1
  130. package/dist/lib/esm/obsidian/Vault.mjs +1 -1
  131. package/dist/lib/esm/obsidian/index.d.mts +1 -0
  132. package/dist/lib/esm/obsidian/index.mjs +3 -1
  133. package/obsidian/Components/TextBasedComponent/package.json +6 -0
  134. package/obsidian/ValidationMessage/package.json +6 -0
  135. package/package.json +4 -4
@@ -52,10 +52,13 @@ class PluginBase extends import_obsidian.Plugin {
52
52
  return this.settingsManager.safeSettings;
53
53
  }
54
54
  get settingsManager() {
55
+ if (!this._settingsManager) {
56
+ throw new Error("Settings manager not defined");
57
+ }
55
58
  return this._settingsManager;
56
59
  }
57
60
  _abortSignal;
58
- _settingsManager;
61
+ _settingsManager = null;
59
62
  notice;
60
63
  /**
61
64
  * Logs a message to the console.
@@ -104,11 +107,46 @@ class PluginBase extends import_obsidian.Plugin {
104
107
  this.app.workspace.onLayoutReady(this.onLayoutReady.bind(this));
105
108
  }, 0);
106
109
  }
110
+ /**
111
+ * Called when the plugin settings are loaded or reloaded.
112
+ *
113
+ * @param _settings - The settings.
114
+ * @returns A promise or `void` indicating the completion of the save process
115
+ */
116
+ onLoadSettings(_settings) {
117
+ (0, import_Function.noop)();
118
+ }
119
+ /**
120
+ * Called when the plugin settings are saved.
121
+ *
122
+ * @param _newSettings - The new settings.
123
+ * @param _oldSettings - The old settings.
124
+ * @returns A promise or `void` indicating the completion of the save process
125
+ */
126
+ onSaveSettings(_newSettings, _oldSettings) {
127
+ (0, import_Function.noop)();
128
+ }
129
+ /**
130
+ * Creates a plugin settings tab.
131
+ *
132
+ * @returns The settings tab or null if not applicable.
133
+ */
134
+ createPluginSettingsTab() {
135
+ return null;
136
+ }
137
+ /**
138
+ * Creates the plugin settings manager. This method must be implemented by subclasses.
139
+ *
140
+ * @returns The plugin settings manager.
141
+ */
142
+ createSettingsManager() {
143
+ return null;
144
+ }
107
145
  /**
108
146
  * Called when the layout is ready. This method can be overridden by subclasses to perform actions once
109
147
  * the layout is ready.
110
148
  *
111
- * @returns A promise or void indicating the completion of the layout setup.
149
+ * @returns A promise or `void` indicating the completion of the layout setup.
112
150
  */
113
151
  onLayoutReady() {
114
152
  (0, import_Function.noop)();
@@ -117,7 +155,7 @@ class PluginBase extends import_obsidian.Plugin {
117
155
  * Called when the plugin loading is complete. This method must be implemented by subclasses to perform
118
156
  * any additional setup required after loading is complete.
119
157
  *
120
- * @returns A promise or void indicating the completion of the load process.
158
+ * @returns A promise or `void` indicating the completion of the load process.
121
159
  */
122
160
  onloadComplete() {
123
161
  (0, import_Function.noop)();
@@ -139,4 +177,4 @@ ${message}`);
139
177
  0 && (module.exports = {
140
178
  PluginBase
141
179
  });
142
- //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../../../src/obsidian/Plugin/PluginBase.ts"],
  "sourcesContent": ["/**\n * @packageDocumentation PluginBase\n * Base class for Obsidian plugins providing utility methods for settings management, error handling, and notifications.\n *\n * This class simplifies the process of managing plugin settings, displaying notifications, and handling errors.\n * Subclasses should implement methods to create default settings and settings tabs, and complete plugin-specific\n * loading tasks.\n */\n\nimport type { PluginSettingTab } from 'obsidian';\nimport type {\n  Promisable,\n  ReadonlyDeep\n} from 'type-fest';\n\nimport {\n  Notice,\n  Plugin\n} from 'obsidian';\n\nimport { getDebugger } from '../../Debug.ts';\nimport { registerAsyncErrorEventHandler } from '../../Error.ts';\nimport { noop } from '../../Function.ts';\nimport { initPluginContext } from './PluginContext.ts';\nimport { PluginSettingsManagerBase } from './PluginSettingsManagerBase.ts';\n\n/**\n * Base class for creating Obsidian plugins with built-in support for settings management, error handling, and notifications.\n *\n * @typeParam PluginSettings - The type representing the plugin settings object.\n */\nexport abstract class PluginBase<PluginSettings extends object = object> extends Plugin {\n  /**\n   * Gets the AbortSignal used for aborting long-running operations.\n   *\n   * @returns The abort signal.\n   */\n  public get abortSignal(): AbortSignal {\n    return this._abortSignal;\n  }\n\n  /**\n   * Gets the readonly plugin settings.\n   *\n   * @returns The readonly plugin settings.\n   */\n  public get settings(): ReadonlyDeep<PluginSettings> {\n    return this.settingsManager.safeSettings;\n  }\n\n  public get settingsManager(): PluginSettingsManagerBase<PluginSettings> {\n    return this._settingsManager;\n  }\n\n  private _abortSignal!: AbortSignal;\n\n  private _settingsManager!: PluginSettingsManagerBase<PluginSettings>;\n\n  private notice?: Notice;\n\n  /**\n   * Logs a message to the console.\n   *\n   * Use instead of `console.debug()`.\n   *\n   * Those messages are not shown by default, but they can be shown by enabling `your-plugin-id` debugger namespace.\n   *\n   * @see {@link https://github.com/mnaoumov/obsidian-dev-utils/?tab=readme-ov-file#debugging} for more information.\n   *\n   * @param message - The message to log.\n   * @param args - The arguments to log.\n   */\n  public consoleDebug(message: string, ...args: unknown[]): void {\n    // Skip the `consoleDebug()` call itself\n    const FRAMES_TO_SKIP = 1;\n    const _debugger = getDebugger(this.manifest.id, FRAMES_TO_SKIP);\n    _debugger(message, ...args);\n  }\n\n  /**\n   * Called when the external settings change.\n   */\n  public override async onExternalSettingsChange(): Promise<void> {\n    await this.settingsManager.loadFromFile();\n  }\n\n  /**\n   * Called when the plugin is loaded\n   */\n  public override async onload(): Promise<void> {\n    initPluginContext(this.app, this.manifest.id);\n\n    this.register(registerAsyncErrorEventHandler(() => {\n      this.showNotice('An unhandled error occurred. Please check the console for more information.');\n    }));\n\n    this._settingsManager = this.createSettingsManager();\n\n    await this.onExternalSettingsChange();\n    const pluginSettingsTab = this.createPluginSettingsTab();\n    if (pluginSettingsTab) {\n      this.addSettingTab(pluginSettingsTab);\n    }\n\n    const abortController = new AbortController();\n    this._abortSignal = abortController.signal;\n    this.register(() => {\n      abortController.abort();\n    });\n    await this.onloadComplete();\n    setTimeout(() => {\n      this.app.workspace.onLayoutReady(this.onLayoutReady.bind(this));\n    }, 0);\n  }\n\n  /**\n   * Creates a plugin settings tab. This method must be implemented by subclasses.\n   *\n   * @returns The settings tab or null if not applicable.\n   */\n  protected abstract createPluginSettingsTab(): null | PluginSettingTab;\n\n  /**\n   * Creates the plugin settings manager. This method must be implemented by subclasses.\n   *\n   * @returns The plugin settings manager.\n   */\n  protected abstract createSettingsManager(): PluginSettingsManagerBase<PluginSettings>;\n\n  /**\n   * Called when the layout is ready. This method can be overridden by subclasses to perform actions once\n   * the layout is ready.\n   *\n   * @returns A promise or void indicating the completion of the layout setup.\n   */\n  protected onLayoutReady(): Promisable<void> {\n    noop();\n  }\n\n  /**\n   * Called when the plugin loading is complete. This method must be implemented by subclasses to perform\n   * any additional setup required after loading is complete.\n   *\n   * @returns A promise or void indicating the completion of the load process.\n   */\n  protected onloadComplete(): Promisable<void> {\n    noop();\n  }\n\n  /**\n   * Displays a notice message to the user.\n   *\n   * @param message - The message to display.\n   */\n  protected showNotice(message: string): void {\n    if (this.notice) {\n      this.notice.hide();\n    }\n\n    this.notice = new Notice(`${this.manifest.name}\\n${message}`);\n  }\n}\n"],
  "mappings": ";;;;;;;;;;;;;;;;;;;;;;;;;AAAA;AAAA;AAAA;AAAA;AAAA;AAeA,sBAGO;AAEP,mBAA4B;AAC5B,mBAA+C;AAC/C,sBAAqB;AACrB,2BAAkC;AAClC,uCAA0C;AAOnC,MAAe,mBAA2D,uBAAO;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAMtF,IAAW,cAA2B;AACpC,WAAO,KAAK;AAAA,EACd;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAOA,IAAW,WAAyC;AAClD,WAAO,KAAK,gBAAgB;AAAA,EAC9B;AAAA,EAEA,IAAW,kBAA6D;AACtE,WAAO,KAAK;AAAA,EACd;AAAA,EAEQ;AAAA,EAEA;AAAA,EAEA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAcD,aAAa,YAAoB,MAAuB;AAE7D,UAAM,iBAAiB;AACvB,UAAM,gBAAY,0BAAY,KAAK,SAAS,IAAI,cAAc;AAC9D,cAAU,SAAS,GAAG,IAAI;AAAA,EAC5B;AAAA;AAAA;AAAA;AAAA,EAKA,MAAsB,2BAA0C;AAC9D,UAAM,KAAK,gBAAgB,aAAa;AAAA,EAC1C;AAAA;AAAA;AAAA;AAAA,EAKA,MAAsB,SAAwB;AAC5C,gDAAkB,KAAK,KAAK,KAAK,SAAS,EAAE;AAE5C,SAAK,aAAS,6CAA+B,MAAM;AACjD,WAAK,WAAW,6EAA6E;AAAA,IAC/F,CAAC,CAAC;AAEF,SAAK,mBAAmB,KAAK,sBAAsB;AAEnD,UAAM,KAAK,yBAAyB;AACpC,UAAM,oBAAoB,KAAK,wBAAwB;AACvD,QAAI,mBAAmB;AACrB,WAAK,cAAc,iBAAiB;AAAA,IACtC;AAEA,UAAM,kBAAkB,IAAI,gBAAgB;AAC5C,SAAK,eAAe,gBAAgB;AACpC,SAAK,SAAS,MAAM;AAClB,sBAAgB,MAAM;AAAA,IACxB,CAAC;AACD,UAAM,KAAK,eAAe;AAC1B,eAAW,MAAM;AACf,WAAK,IAAI,UAAU,cAAc,KAAK,cAAc,KAAK,IAAI,CAAC;AAAA,IAChE,GAAG,CAAC;AAAA,EACN;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAsBU,gBAAkC;AAC1C,8BAAK;AAAA,EACP;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAQU,iBAAmC;AAC3C,8BAAK;AAAA,EACP;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAOU,WAAW,SAAuB;AAC1C,QAAI,KAAK,QAAQ;AACf,WAAK,OAAO,KAAK;AAAA,IACnB;AAEA,SAAK,SAAS,IAAI,uBAAO,GAAG,KAAK,SAAS,IAAI;AAAA,EAAK,OAAO,EAAE;AAAA,EAC9D;AACF;",
  "names": []
}

180
+ //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../../../src/obsidian/Plugin/PluginBase.ts"],
  "sourcesContent": ["/**\n * @packageDocumentation PluginBase\n * Base class for Obsidian plugins providing utility methods for settings management, error handling, and notifications.\n *\n * This class simplifies the process of managing plugin settings, displaying notifications, and handling errors.\n * Subclasses should implement methods to create default settings and settings tabs, and complete plugin-specific\n * loading tasks.\n */\n\nimport type { PluginSettingTab } from 'obsidian';\nimport type {\n  Promisable,\n  ReadonlyDeep\n} from 'type-fest';\n\nimport {\n  Notice,\n  Plugin\n} from 'obsidian';\n\nimport { getDebugger } from '../../Debug.ts';\nimport { registerAsyncErrorEventHandler } from '../../Error.ts';\nimport { noop } from '../../Function.ts';\nimport { initPluginContext } from './PluginContext.ts';\nimport { PluginSettingsManagerBase } from './PluginSettingsManagerBase.ts';\n\n/**\n * Base class for creating Obsidian plugins with built-in support for settings management, error handling, and notifications.\n *\n * @typeParam PluginSettings - The type representing the plugin settings object.\n */\nexport abstract class PluginBase<PluginSettings extends object = object> extends Plugin {\n  /**\n   * @deprecated Used only for type inference. Don't use it directly.\n   */\n  declare public __pluginSettingsType: PluginSettings;\n\n  /**\n   * Gets the AbortSignal used for aborting long-running operations.\n   *\n   * @returns The abort signal.\n   */\n  public get abortSignal(): AbortSignal {\n    return this._abortSignal;\n  }\n\n  /**\n   * Gets the readonly plugin settings.\n   *\n   * @returns The readonly plugin settings.\n   */\n  public get settings(): ReadonlyDeep<PluginSettings> {\n    return this.settingsManager.safeSettings;\n  }\n\n  public get settingsManager(): PluginSettingsManagerBase<PluginSettings> {\n    if (!this._settingsManager) {\n      throw new Error('Settings manager not defined');\n    }\n\n    return this._settingsManager;\n  }\n\n  private _abortSignal!: AbortSignal;\n\n  private _settingsManager: null | PluginSettingsManagerBase<PluginSettings> = null;\n\n  private notice?: Notice;\n\n  /**\n   * Logs a message to the console.\n   *\n   * Use instead of `console.debug()`.\n   *\n   * Those messages are not shown by default, but they can be shown by enabling `your-plugin-id` debugger namespace.\n   *\n   * @see {@link https://github.com/mnaoumov/obsidian-dev-utils/?tab=readme-ov-file#debugging} for more information.\n   *\n   * @param message - The message to log.\n   * @param args - The arguments to log.\n   */\n  public consoleDebug(message: string, ...args: unknown[]): void {\n    // Skip the `consoleDebug()` call itself\n    const FRAMES_TO_SKIP = 1;\n    const _debugger = getDebugger(this.manifest.id, FRAMES_TO_SKIP);\n    _debugger(message, ...args);\n  }\n\n  /**\n   * Called when the external settings change.\n   */\n  public override async onExternalSettingsChange(): Promise<void> {\n    await this.settingsManager.loadFromFile();\n  }\n\n  /**\n   * Called when the plugin is loaded\n   */\n  public override async onload(): Promise<void> {\n    initPluginContext(this.app, this.manifest.id);\n\n    this.register(registerAsyncErrorEventHandler(() => {\n      this.showNotice('An unhandled error occurred. Please check the console for more information.');\n    }));\n\n    this._settingsManager = this.createSettingsManager();\n\n    await this.onExternalSettingsChange();\n    const pluginSettingsTab = this.createPluginSettingsTab();\n    if (pluginSettingsTab) {\n      this.addSettingTab(pluginSettingsTab);\n    }\n\n    const abortController = new AbortController();\n    this._abortSignal = abortController.signal;\n    this.register(() => {\n      abortController.abort();\n    });\n    await this.onloadComplete();\n    setTimeout(() => {\n      this.app.workspace.onLayoutReady(this.onLayoutReady.bind(this));\n    }, 0);\n  }\n\n  /**\n   * Called when the plugin settings are loaded or reloaded.\n   *\n   * @param _settings - The settings.\n   * @returns A promise or `void` indicating the completion of the save process\n   */\n  public onLoadSettings(_settings: PluginSettings): Promisable<void> {\n    noop();\n  }\n\n  /**\n   * Called when the plugin settings are saved.\n   *\n   * @param _newSettings - The new settings.\n   * @param _oldSettings - The old settings.\n   * @returns A promise or `void` indicating the completion of the save process\n   */\n  public onSaveSettings(_newSettings: PluginSettings, _oldSettings: PluginSettings): Promisable<void> {\n    noop();\n  }\n\n  /**\n   * Creates a plugin settings tab.\n   *\n   * @returns The settings tab or null if not applicable.\n   */\n  protected createPluginSettingsTab(): null | PluginSettingTab {\n    return null;\n  }\n\n  /**\n   * Creates the plugin settings manager. This method must be implemented by subclasses.\n   *\n   * @returns The plugin settings manager.\n   */\n  protected createSettingsManager(): null | PluginSettingsManagerBase<PluginSettings> {\n    return null;\n  }\n\n  /**\n   * Called when the layout is ready. This method can be overridden by subclasses to perform actions once\n   * the layout is ready.\n   *\n   * @returns A promise or `void` indicating the completion of the layout setup.\n   */\n  protected onLayoutReady(): Promisable<void> {\n    noop();\n  }\n\n  /**\n   * Called when the plugin loading is complete. This method must be implemented by subclasses to perform\n   * any additional setup required after loading is complete.\n   *\n   * @returns A promise or `void` indicating the completion of the load process.\n   */\n  protected onloadComplete(): Promisable<void> {\n    noop();\n  }\n\n  /**\n   * Displays a notice message to the user.\n   *\n   * @param message - The message to display.\n   */\n  protected showNotice(message: string): void {\n    if (this.notice) {\n      this.notice.hide();\n    }\n\n    this.notice = new Notice(`${this.manifest.name}\\n${message}`);\n  }\n}\n"],
  "mappings": ";;;;;;;;;;;;;;;;;;;;;;;;;AAAA;AAAA;AAAA;AAAA;AAAA;AAeA,sBAGO;AAEP,mBAA4B;AAC5B,mBAA+C;AAC/C,sBAAqB;AACrB,2BAAkC;AAClC,uCAA0C;AAOnC,MAAe,mBAA2D,uBAAO;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAWtF,IAAW,cAA2B;AACpC,WAAO,KAAK;AAAA,EACd;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAOA,IAAW,WAAyC;AAClD,WAAO,KAAK,gBAAgB;AAAA,EAC9B;AAAA,EAEA,IAAW,kBAA6D;AACtE,QAAI,CAAC,KAAK,kBAAkB;AAC1B,YAAM,IAAI,MAAM,8BAA8B;AAAA,IAChD;AAEA,WAAO,KAAK;AAAA,EACd;AAAA,EAEQ;AAAA,EAEA,mBAAqE;AAAA,EAErE;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAcD,aAAa,YAAoB,MAAuB;AAE7D,UAAM,iBAAiB;AACvB,UAAM,gBAAY,0BAAY,KAAK,SAAS,IAAI,cAAc;AAC9D,cAAU,SAAS,GAAG,IAAI;AAAA,EAC5B;AAAA;AAAA;AAAA;AAAA,EAKA,MAAsB,2BAA0C;AAC9D,UAAM,KAAK,gBAAgB,aAAa;AAAA,EAC1C;AAAA;AAAA;AAAA;AAAA,EAKA,MAAsB,SAAwB;AAC5C,gDAAkB,KAAK,KAAK,KAAK,SAAS,EAAE;AAE5C,SAAK,aAAS,6CAA+B,MAAM;AACjD,WAAK,WAAW,6EAA6E;AAAA,IAC/F,CAAC,CAAC;AAEF,SAAK,mBAAmB,KAAK,sBAAsB;AAEnD,UAAM,KAAK,yBAAyB;AACpC,UAAM,oBAAoB,KAAK,wBAAwB;AACvD,QAAI,mBAAmB;AACrB,WAAK,cAAc,iBAAiB;AAAA,IACtC;AAEA,UAAM,kBAAkB,IAAI,gBAAgB;AAC5C,SAAK,eAAe,gBAAgB;AACpC,SAAK,SAAS,MAAM;AAClB,sBAAgB,MAAM;AAAA,IACxB,CAAC;AACD,UAAM,KAAK,eAAe;AAC1B,eAAW,MAAM;AACf,WAAK,IAAI,UAAU,cAAc,KAAK,cAAc,KAAK,IAAI,CAAC;AAAA,IAChE,GAAG,CAAC;AAAA,EACN;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAQO,eAAe,WAA6C;AACjE,8BAAK;AAAA,EACP;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EASO,eAAe,cAA8B,cAAgD;AAClG,8BAAK;AAAA,EACP;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAOU,0BAAmD;AAC3D,WAAO;AAAA,EACT;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAOU,wBAA0E;AAClF,WAAO;AAAA,EACT;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAQU,gBAAkC;AAC1C,8BAAK;AAAA,EACP;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAQU,iBAAmC;AAC3C,8BAAK;AAAA,EACP;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAOU,WAAW,SAAuB;AAC1C,QAAI,KAAK,QAAQ;AACf,WAAK,OAAO,KAAK;AAAA,IACnB;AAEA,SAAK,SAAS,IAAI,uBAAO,GAAG,KAAK,SAAS,IAAI;AAAA,EAAK,OAAO,EAAE;AAAA,EAC9D;AACF;",
  "names": []
}

@@ -16,6 +16,10 @@ import { PluginSettingsManagerBase } from './PluginSettingsManagerBase.cjs';
16
16
  * @typeParam PluginSettings - The type representing the plugin settings object.
17
17
  */
18
18
  export declare abstract class PluginBase<PluginSettings extends object = object> extends Plugin {
19
+ /**
20
+ * @deprecated Used only for type inference. Don't use it directly.
21
+ */
22
+ __pluginSettingsType: PluginSettings;
19
23
  /**
20
24
  * Gets the AbortSignal used for aborting long-running operations.
21
25
  *
@@ -54,29 +58,44 @@ export declare abstract class PluginBase<PluginSettings extends object = object>
54
58
  */
55
59
  onload(): Promise<void>;
56
60
  /**
57
- * Creates a plugin settings tab. This method must be implemented by subclasses.
61
+ * Called when the plugin settings are loaded or reloaded.
62
+ *
63
+ * @param _settings - The settings.
64
+ * @returns A promise or `void` indicating the completion of the save process
65
+ */
66
+ onLoadSettings(_settings: PluginSettings): Promisable<void>;
67
+ /**
68
+ * Called when the plugin settings are saved.
69
+ *
70
+ * @param _newSettings - The new settings.
71
+ * @param _oldSettings - The old settings.
72
+ * @returns A promise or `void` indicating the completion of the save process
73
+ */
74
+ onSaveSettings(_newSettings: PluginSettings, _oldSettings: PluginSettings): Promisable<void>;
75
+ /**
76
+ * Creates a plugin settings tab.
58
77
  *
59
78
  * @returns The settings tab or null if not applicable.
60
79
  */
61
- protected abstract createPluginSettingsTab(): null | PluginSettingTab;
80
+ protected createPluginSettingsTab(): null | PluginSettingTab;
62
81
  /**
63
82
  * Creates the plugin settings manager. This method must be implemented by subclasses.
64
83
  *
65
84
  * @returns The plugin settings manager.
66
85
  */
67
- protected abstract createSettingsManager(): PluginSettingsManagerBase<PluginSettings>;
86
+ protected createSettingsManager(): null | PluginSettingsManagerBase<PluginSettings>;
68
87
  /**
69
88
  * Called when the layout is ready. This method can be overridden by subclasses to perform actions once
70
89
  * the layout is ready.
71
90
  *
72
- * @returns A promise or void indicating the completion of the layout setup.
91
+ * @returns A promise or `void` indicating the completion of the layout setup.
73
92
  */
74
93
  protected onLayoutReady(): Promisable<void>;
75
94
  /**
76
95
  * Called when the plugin loading is complete. This method must be implemented by subclasses to perform
77
96
  * any additional setup required after loading is complete.
78
97
  *
79
- * @returns A promise or void indicating the completion of the load process.
98
+ * @returns A promise or `void` indicating the completion of the load process.
80
99
  */
81
100
  protected onloadComplete(): Promisable<void>;
82
101
  /**
@@ -25,11 +25,13 @@ var __copyProps = (to, from, except, desc) => {
25
25
  var __toCommonJS = (mod) => __copyProps(__defProp({}, "__esModule", { value: true }), mod);
26
26
  var PluginSettingsManagerBase_exports = {};
27
27
  __export(PluginSettingsManagerBase_exports, {
28
- PluginSettingsManagerBase: () => PluginSettingsManagerBase
28
+ PluginSettingsManagerBase: () => PluginSettingsManagerBase,
29
+ PluginSettingsProperty: () => PluginSettingsProperty
29
30
  });
30
31
  module.exports = __toCommonJS(PluginSettingsManagerBase_exports);
31
32
  var import_obsidian = require('obsidian');
32
33
  var import_Function = require('../../Function.cjs');
34
+ var import_Object = require('../../Object.cjs');
33
35
  var import_DateTransformer = require('../../Transformers/DateTransformer.cjs');
34
36
  var import_DurationTransformer = require('../../Transformers/DurationTransformer.cjs');
35
37
  var import_GroupTransformer = require('../../Transformers/GroupTransformer.cjs');
@@ -39,68 +41,94 @@ const defaultTransformer = new import_GroupTransformer.GroupTransformer([
39
41
  new import_DateTransformer.DateTransformer(),
40
42
  new import_DurationTransformer.DurationTransformer()
41
43
  ]);
42
- class PluginSettingsProperty {
43
- constructor(manager, property, defaultValue) {
44
- this.manager = manager;
45
- this.property = property;
46
- this.defaultValue = defaultValue;
47
- }
48
- validationMessage = "";
49
- value;
50
- clear() {
51
- this.value = void 0;
52
- this.validationMessage = "";
53
- }
54
- get() {
55
- return this.value ?? this.defaultValue;
44
+ class ProxyHandlerBase {
45
+ constructor(properties) {
46
+ this.properties = properties;
56
47
  }
57
- getSafe() {
58
- return this.validationMessage ? this.defaultValue : this.get();
48
+ get(target, prop) {
49
+ const record = target;
50
+ if (typeof prop !== "string") {
51
+ return record[prop];
52
+ }
53
+ const property = this.properties.get(prop);
54
+ if (!property) {
55
+ return record[prop];
56
+ }
57
+ return this.getPropertyValue(property);
59
58
  }
60
- async set(value) {
61
- this.value = value;
62
- if (this.value !== void 0) {
63
- this.validationMessage = await this.manager.validate(this.property, this.value) ?? "";
59
+ }
60
+ class EditableSettingsProxyHandler extends ProxyHandlerBase {
61
+ validationPromise = Promise.resolve();
62
+ set(target, prop, value) {
63
+ const record = target;
64
+ if (typeof prop !== "string") {
65
+ record[prop] = value;
66
+ return true;
67
+ }
68
+ const property = this.properties.get(prop);
69
+ if (!property) {
70
+ record[prop] = value;
71
+ return true;
64
72
  }
73
+ property.setValue(value);
74
+ this.validationPromise = this.validationPromise.then(() => property.setValueAndValidate(value));
75
+ return true;
76
+ }
77
+ getPropertyValue(property) {
78
+ return property.currentValue;
65
79
  }
66
80
  }
67
81
  class PropertiesMap extends Map {
68
- getTyped(key) {
69
- const property = super.get(key);
82
+ getTyped(propertyName) {
83
+ const property = super.get(propertyName);
70
84
  if (!property) {
71
- throw new Error(`Property ${String(key)} not found`);
85
+ throw new Error(`Property ${String(propertyName)} not found`);
72
86
  }
73
87
  return property;
74
88
  }
75
- setTyped(key, value) {
76
- return super.set(key, value);
89
+ setTyped(propertyName, value) {
90
+ return super.set(propertyName, value);
91
+ }
92
+ }
93
+ class SafeSettingsProxyHandler extends ProxyHandlerBase {
94
+ getPropertyValue(property) {
95
+ return property.safeValue;
77
96
  }
78
97
  }
79
98
  class PluginSettingsManagerBase {
80
99
  constructor(plugin) {
81
100
  this.plugin = plugin;
82
- const defaultSettings = this.createDefaultSettings();
101
+ this.app = plugin.app;
102
+ this.defaultSettings = this.createDefaultSettings();
103
+ this.addValidators();
83
104
  this.properties = new PropertiesMap();
84
- for (const key of Object.keys(defaultSettings)) {
85
- this.properties.set(key, new PluginSettingsProperty(this, key, defaultSettings[key]));
86
- }
87
- this.safeSettings = new Proxy(defaultSettings, {
88
- get: (_target, prop) => {
89
- if (typeof prop !== "string") {
90
- return void 0;
91
- }
92
- return this.properties.get(prop);
93
- }
94
- });
105
+ for (const propertyName of (0, import_Object.getAllKeys)(this.defaultSettings)) {
106
+ this.properties.set(
107
+ propertyName,
108
+ new PluginSettingsProperty(propertyName, this.defaultSettings[propertyName], this.validators.get(propertyName) ?? import_Function.noop)
109
+ );
110
+ }
111
+ this.validators.clear();
112
+ this.safeSettings = new Proxy(this.defaultSettings, new SafeSettingsProxyHandler(this.properties));
95
113
  }
114
+ app;
96
115
  safeSettings;
116
+ defaultSettings;
97
117
  properties;
98
- getProperty(property) {
99
- return this.properties.getTyped(property);
118
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
119
+ validators = /* @__PURE__ */ new Map();
120
+ async editAndSave(editor) {
121
+ const editableSettings = new Proxy(this.defaultSettings, new EditableSettingsProxyHandler(this.properties));
122
+ await editor(editableSettings);
123
+ await editableSettings.validationPromise;
124
+ await this.saveToFile();
125
+ }
126
+ getProperty(propertyName) {
127
+ return this.properties.getTyped(propertyName);
100
128
  }
101
129
  async loadFromFile() {
102
130
  for (const property of this.properties.values()) {
103
- property.clear();
131
+ property.reset();
104
132
  }
105
133
  const data = await this.plugin.loadData();
106
134
  if (data === void 0 || data === null) {
@@ -112,20 +140,31 @@ class PluginSettingsManagerBase {
112
140
  return;
113
141
  }
114
142
  let record = data;
143
+ const originalJson = JSON.stringify(record);
115
144
  record = this.getTransformer().transformObjectRecursively(record);
116
- await this.prepareRecord(record);
117
- for (const [key, value] of Object.entries(record)) {
118
- const propertyObj = this.properties.get(key);
119
- if (!propertyObj) {
120
- console.warn(`Unknown property: ${key}`);
145
+ await this.onLoadRecord(record);
146
+ for (const [propertyName, value] of Object.entries(record)) {
147
+ const property = this.properties.get(propertyName);
148
+ if (!property) {
149
+ console.warn(`Unknown property: ${propertyName}`);
121
150
  continue;
122
151
  }
123
- if (typeof value !== typeof propertyObj.defaultValue) {
124
- console.warn(`Invalid value type. Expected ${typeof propertyObj.defaultValue}, got: ${typeof value}`);
152
+ if (typeof value !== typeof property.defaultValue) {
153
+ console.warn("Invalid value type", {
154
+ propertyName,
155
+ propertyType: typeof property.defaultValue,
156
+ value
157
+ });
125
158
  continue;
126
159
  }
127
- await propertyObj.set(value);
160
+ await property.setValueAndValidate(value);
161
+ property.save();
162
+ }
163
+ const newJson = JSON.stringify(await this.prepareRecordToSave());
164
+ if (newJson !== originalJson) {
165
+ await this.saveToFileImpl();
128
166
  }
167
+ await this.plugin.onLoadSettings(this.getSavedSettings());
129
168
  }
130
169
  /**
131
170
  * Saves the new plugin settings.
@@ -133,29 +172,131 @@ class PluginSettingsManagerBase {
133
172
  * @returns A promise that resolves when the settings are saved.
134
173
  */
135
174
  async saveToFile() {
136
- const record = this.getTransformer().transformObjectRecursively(this.getSettings());
137
- await this.plugin.saveData(record);
175
+ const oldSettings = this.getSavedSettings();
176
+ let hasChanges = false;
177
+ for (const property of this.properties.values()) {
178
+ hasChanges ||= property.save();
179
+ }
180
+ if (!hasChanges) {
181
+ return;
182
+ }
183
+ await this.saveToFileImpl();
184
+ await this.plugin.onSaveSettings(this.getSavedSettings(), oldSettings);
185
+ }
186
+ addValidator(propertyName, validator) {
187
+ this.validators.set(propertyName, validator);
138
188
  }
139
- // eslint-disable-next-line @typescript-eslint/no-invalid-void-type
140
- async validate(_property, _value) {
141
- await (0, import_Function.noopAsync)();
189
+ addValidators() {
190
+ (0, import_Function.noop)();
142
191
  }
143
192
  getTransformer() {
144
193
  return defaultTransformer;
145
194
  }
146
- async prepareRecord(_record) {
147
- await (0, import_Function.noopAsync)();
195
+ /**
196
+ * Called when the plugin settings are loaded.
197
+ *
198
+ * @param _record - The record.
199
+ * @returns A promise or `void` indicating the completion of the load process
200
+ */
201
+ onLoadRecord(_record) {
202
+ (0, import_Function.noop)();
148
203
  }
149
- getSettings() {
204
+ /**
205
+ * Called when the plugin settings are saving.
206
+ *
207
+ * @param _record - The record.
208
+ * @returns A promise or `void` indicating the completion of the save process
209
+ */
210
+ onSavingRecord(_record) {
211
+ (0, import_Function.noop)();
212
+ }
213
+ getSavedSettings() {
214
+ const savedSettings = {};
215
+ for (const [propertyName, property] of this.properties.entries()) {
216
+ savedSettings[propertyName] = property.lastSavedValue;
217
+ }
218
+ const proto = Object.getPrototypeOf(this.defaultSettings);
219
+ Object.setPrototypeOf(savedSettings, proto);
220
+ return savedSettings;
221
+ }
222
+ async prepareRecordToSave() {
150
223
  const settings = {};
151
- for (const [key, property] of this.properties.entries()) {
152
- settings[key] = property.get();
224
+ for (const [propertyName, property] of this.properties.entries()) {
225
+ settings[propertyName] = property.currentValue;
226
+ }
227
+ await this.onSavingRecord(settings);
228
+ return this.getTransformer().transformObjectRecursively(settings);
229
+ }
230
+ async saveToFileImpl() {
231
+ await this.plugin.saveData(await this.prepareRecordToSave());
232
+ }
233
+ }
234
+ class PluginSettingsProperty {
235
+ constructor(propertyName, defaultValue, validator) {
236
+ this.propertyName = propertyName;
237
+ this.defaultValue = defaultValue;
238
+ this.validator = validator;
239
+ this._lastSavedValue = defaultValue;
240
+ this._currentValue = defaultValue;
241
+ }
242
+ get currentValue() {
243
+ return this._currentValue;
244
+ }
245
+ get lastSavedValue() {
246
+ return this._lastSavedValue;
247
+ }
248
+ get safeValue() {
249
+ return this._validationMessage ? this.defaultValue : this._currentValue;
250
+ }
251
+ get validationMessage() {
252
+ return this._validationMessage;
253
+ }
254
+ _currentValue;
255
+ _lastSavedValue;
256
+ _validationMessage = "";
257
+ reset() {
258
+ this._currentValue = this.defaultValue;
259
+ this._validationMessage = "";
260
+ }
261
+ save() {
262
+ if (this._lastSavedValue === this._currentValue) {
263
+ return false;
153
264
  }
154
- return settings;
265
+ this._lastSavedValue = this._currentValue;
266
+ return true;
267
+ }
268
+ setValidationMessage(validationMessage) {
269
+ this._validationMessage = validationMessage;
270
+ this.showWarning();
271
+ }
272
+ setValue(value) {
273
+ this._currentValue = value;
274
+ }
275
+ async setValueAndValidate(value) {
276
+ this.setValue(value);
277
+ if (this._currentValue === void 0) {
278
+ return;
279
+ }
280
+ this._validationMessage = await this.validator(this._currentValue) ?? "";
281
+ this.showWarning(value);
282
+ }
283
+ showWarning(value) {
284
+ if (!this._validationMessage) {
285
+ return;
286
+ }
287
+ const warningMessage = `Could not set plugin setting: ${this.propertyName}. Using default value instead.`;
288
+ new import_obsidian.Notice(warningMessage);
289
+ console.warn(warningMessage, {
290
+ defaultValue: this.defaultValue,
291
+ propertyName: this.propertyName,
292
+ validationMessage: this._validationMessage,
293
+ value
294
+ });
155
295
  }
156
296
  }
157
297
  // Annotate the CommonJS export names for ESM import in node:
158
298
  0 && (module.exports = {
159
- PluginSettingsManagerBase
299
+ PluginSettingsManagerBase,
300
+ PluginSettingsProperty
160
301
  });
161
- //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../../../src/obsidian/Plugin/PluginSettingsManagerBase.ts"],
  "sourcesContent": ["import type { ReadonlyDeep } from 'type-fest';\n\nimport { Plugin } from 'obsidian';\n\nimport type { StringKeys } from '../../Object.ts';\nimport type { Transformer } from '../../Transformers/Transformer.ts';\n\nimport { noopAsync } from '../../Function.ts';\nimport { DateTransformer } from '../../Transformers/DateTransformer.ts';\nimport { DurationTransformer } from '../../Transformers/DurationTransformer.ts';\nimport { GroupTransformer } from '../../Transformers/GroupTransformer.ts';\nimport { SkipPrivatePropertyTransformer } from '../../Transformers/SkipPrivatePropertyTransformer.ts';\n\nconst defaultTransformer = new GroupTransformer([\n  new SkipPrivatePropertyTransformer(),\n  new DateTransformer(),\n  new DurationTransformer()\n]);\n\nclass PluginSettingsProperty<PluginSettings extends object, Property extends StringKeys<PluginSettings>> {\n  public validationMessage = '';\n  private value: PluginSettings[Property] | undefined;\n\n  public constructor(\n    private readonly manager: PluginSettingsManagerBase<PluginSettings>,\n    private readonly property: Property,\n    public readonly defaultValue: PluginSettings[Property]\n  ) {}\n\n  public clear(): void {\n    this.value = undefined;\n    this.validationMessage = '';\n  }\n\n  public get(): PluginSettings[Property] {\n    return this.value ?? this.defaultValue;\n  }\n\n  public getSafe(): PluginSettings[Property] {\n    return this.validationMessage ? this.defaultValue : this.get();\n  }\n\n  public async set(value: PluginSettings[Property] | undefined): Promise<void> {\n    this.value = value;\n    if (this.value !== undefined) {\n      this.validationMessage = (await this.manager.validate(this.property, this.value) as string | undefined) ?? '';\n    }\n  }\n}\n\nclass PropertiesMap<PluginSettings extends object> extends Map<string, PluginSettingsProperty<PluginSettings, StringKeys<PluginSettings>>> {\n  public getTyped<Property extends StringKeys<PluginSettings>>(key: Property): PluginSettingsProperty<PluginSettings, Property> {\n    const property = super.get(key);\n    if (!property) {\n      throw new Error(`Property ${String(key)} not found`);\n    }\n\n    return property as PluginSettingsProperty<PluginSettings, Property>;\n  }\n\n  public setTyped<Property extends StringKeys<PluginSettings>>(key: Property, value: PluginSettingsProperty<PluginSettings, Property>): this {\n    return super.set(key, value);\n  }\n}\n\n/**\n * Base class for managing plugin settings.\n *\n * @typeParam PluginSettings - The type representing the plugin settings object.\n */\nexport abstract class PluginSettingsManagerBase<PluginSettings extends object> {\n  public readonly safeSettings: ReadonlyDeep<PluginSettings>;\n\n  private properties: PropertiesMap<PluginSettings>;\n\n  public constructor(private plugin: Plugin) {\n    const defaultSettings = this.createDefaultSettings();\n    this.properties = new PropertiesMap<PluginSettings>();\n\n    for (const key of Object.keys(defaultSettings) as StringKeys<PluginSettings>[]) {\n      this.properties.set(key, new PluginSettingsProperty(this, key, defaultSettings[key]));\n    }\n\n    this.safeSettings = new Proxy(defaultSettings, {\n      get: (_target, prop): unknown => {\n        if (typeof prop !== 'string') {\n          return undefined;\n        }\n\n        return this.properties.get(prop);\n      }\n    }) as ReadonlyDeep<PluginSettings>;\n  }\n\n  public getProperty<Property extends StringKeys<PluginSettings>>(property: Property): PluginSettingsProperty<PluginSettings, Property> {\n    return this.properties.getTyped(property);\n  }\n\n  public async loadFromFile(): Promise<void> {\n    for (const property of this.properties.values()) {\n      property.clear();\n    }\n\n    const data = await this.plugin.loadData() as unknown;\n\n    if (data === undefined || data === null) {\n      return;\n    }\n\n    if (typeof data !== 'object' || Array.isArray(data)) {\n      const type = Array.isArray(data) ? 'Array' : typeof data;\n      console.error(`Invalid data type. Expected Object, got: ${type}`);\n      return;\n    }\n\n    let record = data as Record<string, unknown>;\n    record = this.getTransformer().transformObjectRecursively(record);\n    await this.prepareRecord(record);\n\n    for (const [key, value] of Object.entries(record)) {\n      const propertyObj = this.properties.get(key);\n      if (!propertyObj) {\n        console.warn(`Unknown property: ${key}`);\n        continue;\n      }\n\n      if (typeof value !== typeof propertyObj.defaultValue) {\n        console.warn(`Invalid value type. Expected ${typeof propertyObj.defaultValue}, got: ${typeof value}`);\n        continue;\n      }\n\n      await propertyObj.set(value as PluginSettings[StringKeys<PluginSettings>]);\n    }\n  }\n\n  /**\n   * Saves the new plugin settings.\n   *\n   * @returns A promise that resolves when the settings are saved.\n   */\n  public async saveToFile(): Promise<void> {\n    const record = this.getTransformer().transformObjectRecursively(this.getSettings());\n    await this.plugin.saveData(record);\n  }\n\n  // eslint-disable-next-line @typescript-eslint/no-invalid-void-type\n  public async validate<Property extends StringKeys<PluginSettings>>(_property: Property, _value: PluginSettings[Property]): Promise<string | void> {\n    await noopAsync();\n  }\n\n  protected abstract createDefaultSettings(): PluginSettings;\n\n  protected getTransformer(): Transformer {\n    return defaultTransformer;\n  }\n\n  protected async prepareRecord(_record: Record<string, unknown>): Promise<void> {\n    await noopAsync();\n  }\n\n  private getSettings(): Record<StringKeys<PluginSettings>, unknown> {\n    const settings: Record<StringKeys<PluginSettings>, unknown> = {} as Record<StringKeys<PluginSettings>, unknown>;\n    for (const [key, property] of this.properties.entries()) {\n      settings[key as StringKeys<PluginSettings>] = property.get();\n    }\n\n    return settings;\n  }\n}\n"],
  "mappings": "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",
  "names": []
}

302
+ //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../../../src/obsidian/Plugin/PluginSettingsManagerBase.ts"],
  "sourcesContent": ["import type { App } from 'obsidian';\nimport type {\n  Promisable,\n  ReadonlyDeep\n} from 'type-fest';\n\nimport { Notice } from 'obsidian';\n\nimport type { Transformer } from '../../Transformers/Transformer.ts';\nimport type {\n  MaybeReturn,\n  StringKeys\n} from '../../Type.ts';\nimport type { PluginBase } from './PluginBase.ts';\n\nimport { noop } from '../../Function.ts';\nimport { getAllKeys } from '../../Object.ts';\nimport { DateTransformer } from '../../Transformers/DateTransformer.ts';\nimport { DurationTransformer } from '../../Transformers/DurationTransformer.ts';\nimport { GroupTransformer } from '../../Transformers/GroupTransformer.ts';\nimport { SkipPrivatePropertyTransformer } from '../../Transformers/SkipPrivatePropertyTransformer.ts';\n\nconst defaultTransformer = new GroupTransformer([\n  new SkipPrivatePropertyTransformer(),\n  new DateTransformer(),\n  new DurationTransformer()\n]);\n\ntype Validator<T> = (value: T) => Promisable<MaybeReturn<string>>;\n\nabstract class ProxyHandlerBase<PluginSettings extends object> implements ProxyHandler<PluginSettings> {\n  public constructor(protected readonly properties: PropertiesMap<PluginSettings>) {}\n\n  public get(target: PluginSettings, prop: string | symbol): unknown {\n    const record = target as Record<string | symbol, unknown>;\n    if (typeof prop !== 'string') {\n      return record[prop];\n    }\n\n    const property = this.properties.get(prop);\n    if (!property) {\n      return record[prop];\n    }\n\n    return this.getPropertyValue(property);\n  }\n\n  protected abstract getPropertyValue(property: PluginSettingsProperty<unknown>): unknown;\n}\n\nclass EditableSettingsProxyHandler<PluginSettings extends object> extends ProxyHandlerBase<PluginSettings> {\n  private validationPromise = Promise.resolve();\n  public set(target: PluginSettings, prop: string | symbol, value: unknown): boolean {\n    const record = target as Record<string | symbol, unknown>;\n\n    if (typeof prop !== 'string') {\n      record[prop] = value;\n      return true;\n    }\n\n    const property = this.properties.get(prop);\n    if (!property) {\n      record[prop] = value;\n      return true;\n    }\n\n    property.setValue(value);\n    this.validationPromise = this.validationPromise.then(() => property.setValueAndValidate(value));\n\n    return true;\n  }\n\n  protected override getPropertyValue(property: PluginSettingsProperty<unknown>): unknown {\n    return property.currentValue;\n  }\n}\n\n// eslint-disable-next-line @typescript-eslint/no-explicit-any\nclass PropertiesMap<PluginSettings extends object> extends Map<string, PluginSettingsProperty<any>> {\n  public getTyped<PropertyName extends StringKeys<PluginSettings>>(propertyName: PropertyName): PluginSettingsProperty<PluginSettings[PropertyName]> {\n    const property = super.get(propertyName);\n    if (!property) {\n      throw new Error(`Property ${String(propertyName)} not found`);\n    }\n\n    return property as PluginSettingsProperty<PluginSettings[PropertyName]>;\n  }\n\n  public setTyped<PropertyName extends StringKeys<PluginSettings>>(\n    propertyName: PropertyName,\n    value: PluginSettingsProperty<PluginSettings[PropertyName]>\n  ): this {\n    return super.set(propertyName, value);\n  }\n}\n\nclass SafeSettingsProxyHandler<PluginSettings extends object> extends ProxyHandlerBase<PluginSettings> {\n  protected override getPropertyValue(property: PluginSettingsProperty<unknown>): unknown {\n    return property.safeValue;\n  }\n}\n\n/**\n * Base class for managing plugin settings.\n *\n * @typeParam PluginSettings - The type representing the plugin settings object.\n */\nexport abstract class PluginSettingsManagerBase<PluginSettings extends object> {\n  public readonly app: App;\n  public readonly safeSettings: ReadonlyDeep<PluginSettings>;\n\n  private defaultSettings: PluginSettings;\n  private properties: PropertiesMap<PluginSettings>;\n  // eslint-disable-next-line @typescript-eslint/no-explicit-any\n  private validators: Map<string, Validator<any>> = new Map<string, Validator<any>>();\n\n  public constructor(public readonly plugin: PluginBase<PluginSettings>) {\n    this.app = plugin.app;\n    this.defaultSettings = this.createDefaultSettings();\n\n    this.addValidators();\n\n    this.properties = new PropertiesMap<PluginSettings>();\n\n    for (const propertyName of getAllKeys(this.defaultSettings)) {\n      this.properties.set(\n        propertyName,\n        new PluginSettingsProperty(propertyName, this.defaultSettings[propertyName], this.validators.get(propertyName) ?? noop)\n      );\n    }\n\n    this.validators.clear();\n\n    this.safeSettings = new Proxy(this.defaultSettings, new SafeSettingsProxyHandler<PluginSettings>(this.properties)) as ReadonlyDeep<PluginSettings>;\n  }\n\n  public async editAndSave(editor: (settings: PluginSettings) => Promisable<void>): Promise<void> {\n    const editableSettings = new Proxy(this.defaultSettings, new EditableSettingsProxyHandler<PluginSettings>(this.properties)) as {\n      validationPromise: Promise<void>;\n    } & PluginSettings;\n    await editor(editableSettings);\n    await editableSettings.validationPromise;\n    await this.saveToFile();\n  }\n\n  public getProperty<PropertyName extends StringKeys<PluginSettings>>(propertyName: PropertyName): PluginSettingsProperty<PluginSettings[PropertyName]> {\n    return this.properties.getTyped(propertyName);\n  }\n\n  public async loadFromFile(): Promise<void> {\n    for (const property of this.properties.values()) {\n      property.reset();\n    }\n\n    const data = await this.plugin.loadData() as unknown;\n\n    if (data === undefined || data === null) {\n      return;\n    }\n\n    if (typeof data !== 'object' || Array.isArray(data)) {\n      const type = Array.isArray(data) ? 'Array' : typeof data;\n      console.error(`Invalid data type. Expected Object, got: ${type}`);\n      return;\n    }\n\n    let record = data as Record<string, unknown>;\n    const originalJson = JSON.stringify(record);\n    record = this.getTransformer().transformObjectRecursively(record);\n    await this.onLoadRecord(record);\n\n    for (const [propertyName, value] of Object.entries(record)) {\n      const property = this.properties.get(propertyName);\n      if (!property) {\n        console.warn(`Unknown property: ${propertyName}`);\n        continue;\n      }\n\n      if (typeof value !== typeof property.defaultValue) {\n        console.warn('Invalid value type', {\n          propertyName,\n          propertyType: typeof property.defaultValue,\n          value\n        });\n        continue;\n      }\n\n      await property.setValueAndValidate(value);\n      property.save();\n    }\n\n    const newJson = JSON.stringify(await this.prepareRecordToSave());\n\n    if (newJson !== originalJson) {\n      await this.saveToFileImpl();\n    }\n\n    await this.plugin.onLoadSettings(this.getSavedSettings());\n  }\n\n  /**\n   * Saves the new plugin settings.\n   *\n   * @returns A promise that resolves when the settings are saved.\n   */\n  public async saveToFile(): Promise<void> {\n    const oldSettings = this.getSavedSettings();\n\n    let hasChanges = false;\n\n    for (const property of this.properties.values()) {\n      hasChanges ||= property.save();\n    }\n\n    if (!hasChanges) {\n      return;\n    }\n\n    await this.saveToFileImpl();\n    await this.plugin.onSaveSettings(this.getSavedSettings(), oldSettings);\n  }\n\n  protected addValidator<PropertyName extends StringKeys<PluginSettings>>(\n    propertyName: PropertyName,\n    validator: Validator<PluginSettings[PropertyName]>\n  ): void {\n    this.validators.set(propertyName, validator);\n  }\n\n  protected addValidators(): void {\n    noop();\n  }\n\n  protected abstract createDefaultSettings(): PluginSettings;\n\n  protected getTransformer(): Transformer {\n    return defaultTransformer;\n  }\n\n  /**\n   * Called when the plugin settings are loaded.\n   *\n   * @param _record - The record.\n   * @returns A promise or `void` indicating the completion of the load process\n   */\n  protected onLoadRecord(_record: Record<string, unknown>): Promisable<void> {\n    noop();\n  }\n\n  /**\n   * Called when the plugin settings are saving.\n   *\n   * @param _record - The record.\n   * @returns A promise or `void` indicating the completion of the save process\n   */\n  protected onSavingRecord(_record: Record<string, unknown>): Promisable<void> {\n    noop();\n  }\n\n  private getSavedSettings(): PluginSettings {\n    const savedSettings: Partial<PluginSettings> = {};\n    for (const [propertyName, property] of this.properties.entries()) {\n      savedSettings[propertyName as StringKeys<PluginSettings>] = property.lastSavedValue as PluginSettings[StringKeys<PluginSettings>] | undefined;\n    }\n    const proto = Object.getPrototypeOf(this.defaultSettings) as object;\n    Object.setPrototypeOf(savedSettings, proto);\n    return savedSettings as PluginSettings;\n  }\n\n  private async prepareRecordToSave(): Promise<Record<StringKeys<PluginSettings>, unknown>> {\n    const settings: Record<StringKeys<PluginSettings>, unknown> = {} as Record<StringKeys<PluginSettings>, unknown>;\n    for (const [propertyName, property] of this.properties.entries()) {\n      settings[propertyName as StringKeys<PluginSettings>] = property.currentValue as unknown;\n    }\n\n    await this.onSavingRecord(settings);\n\n    return this.getTransformer().transformObjectRecursively(settings);\n  }\n\n  private async saveToFileImpl(): Promise<void> {\n    await this.plugin.saveData(await this.prepareRecordToSave());\n  }\n}\n\n/**\n * A property of a plugin settings.\n *\n * @typeParam T - The type of the property.\n */\nexport class PluginSettingsProperty<T> {\n  public get currentValue(): T {\n    return this._currentValue;\n  }\n\n  public get lastSavedValue(): T {\n    return this._lastSavedValue;\n  }\n\n  public get safeValue(): T {\n    return this._validationMessage ? this.defaultValue : this._currentValue;\n  }\n\n  public get validationMessage(): string {\n    return this._validationMessage;\n  }\n\n  private _currentValue: T;\n\n  private _lastSavedValue: T;\n\n  private _validationMessage = '';\n\n  public constructor(private readonly propertyName: string, public readonly defaultValue: T, private readonly validator: Validator<T>) {\n    this._lastSavedValue = defaultValue;\n    this._currentValue = defaultValue;\n  }\n\n  public reset(): void {\n    this._currentValue = this.defaultValue;\n    this._validationMessage = '';\n  }\n\n  public save(): boolean {\n    if (this._lastSavedValue === this._currentValue) {\n      return false;\n    }\n\n    this._lastSavedValue = this._currentValue;\n    return true;\n  }\n\n  public setValidationMessage(validationMessage: string): void {\n    this._validationMessage = validationMessage;\n    this.showWarning();\n  }\n\n  public setValue(value: T): void {\n    this._currentValue = value;\n  }\n\n  public async setValueAndValidate(value: T): Promise<void> {\n    this.setValue(value);\n    if (this._currentValue === undefined) {\n      return;\n    }\n\n    this._validationMessage = (await this.validator(this._currentValue) as string | undefined) ?? '';\n    this.showWarning(value);\n  }\n\n  private showWarning(value?: T): void {\n    if (!this._validationMessage) {\n      return;\n    }\n\n    const warningMessage = `Could not set plugin setting: ${this.propertyName}. Using default value instead.`;\n    new Notice(warningMessage);\n    console.warn(warningMessage, {\n      defaultValue: this.defaultValue,\n      propertyName: this.propertyName,\n      validationMessage: this._validationMessage,\n      value\n    });\n  }\n}\n"],
  "mappings": "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",
  "names": []
}
