obsidian-dev-utils 24.0.1 → 24.0.2-beta.2

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 (32) hide show
  1. package/CHANGELOG.md +15 -0
  2. package/dist/lib/cjs/Library.cjs +1 -1
  3. package/dist/lib/cjs/obsidian/App.cjs +6 -1
  4. package/dist/lib/cjs/obsidian/App.d.cts +5 -0
  5. package/dist/lib/cjs/obsidian/Components/CheckboxComponent.cjs +11 -1
  6. package/dist/lib/cjs/obsidian/Components/CheckboxComponent.d.cts +10 -0
  7. package/dist/lib/cjs/obsidian/Components/TriStateCheckboxComponent.cjs +11 -1
  8. package/dist/lib/cjs/obsidian/Components/TriStateCheckboxComponent.d.cts +10 -0
  9. package/dist/lib/cjs/obsidian/Modals/ModalBase.cjs +8 -1
  10. package/dist/lib/cjs/obsidian/Modals/ModalBase.d.cts +7 -0
  11. package/dist/lib/cjs/obsidian/Plugin/PluginBase.cjs +23 -5
  12. package/dist/lib/cjs/obsidian/Plugin/PluginBase.d.cts +20 -2
  13. package/dist/lib/cjs/obsidian/Plugin/PluginSettingsManagerBase.cjs +120 -23
  14. package/dist/lib/cjs/obsidian/Plugin/PluginSettingsManagerBase.d.cts +102 -8
  15. package/dist/lib/cjs/obsidian/Plugin/PluginSettingsTabBase.cjs +39 -2
  16. package/dist/lib/cjs/obsidian/Plugin/PluginSettingsTabBase.d.cts +23 -0
  17. package/dist/lib/esm/Library.mjs +1 -1
  18. package/dist/lib/esm/obsidian/App.d.mts +5 -0
  19. package/dist/lib/esm/obsidian/App.mjs +6 -1
  20. package/dist/lib/esm/obsidian/Components/CheckboxComponent.d.mts +10 -0
  21. package/dist/lib/esm/obsidian/Components/CheckboxComponent.mjs +11 -1
  22. package/dist/lib/esm/obsidian/Components/TriStateCheckboxComponent.d.mts +10 -0
  23. package/dist/lib/esm/obsidian/Components/TriStateCheckboxComponent.mjs +11 -1
  24. package/dist/lib/esm/obsidian/Modals/ModalBase.d.mts +7 -0
  25. package/dist/lib/esm/obsidian/Modals/ModalBase.mjs +8 -1
  26. package/dist/lib/esm/obsidian/Plugin/PluginBase.d.mts +20 -2
  27. package/dist/lib/esm/obsidian/Plugin/PluginBase.mjs +23 -5
  28. package/dist/lib/esm/obsidian/Plugin/PluginSettingsManagerBase.d.mts +102 -8
  29. package/dist/lib/esm/obsidian/Plugin/PluginSettingsManagerBase.mjs +120 -23
  30. package/dist/lib/esm/obsidian/Plugin/PluginSettingsTabBase.d.mts +23 -0
  31. package/dist/lib/esm/obsidian/Plugin/PluginSettingsTabBase.mjs +39 -2
  32. package/package.json +1 -1
@@ -60,17 +60,16 @@ class EditableSettingsProxyHandler extends ProxyHandlerBase {
60
60
  validationPromise = Promise.resolve();
61
61
  set(target, prop, value) {
62
62
  const record = target;
63
+ record[prop] = value;
63
64
  if (typeof prop !== "string") {
64
- record[prop] = value;
65
65
  return true;
66
66
  }
67
67
  const property = this.properties.get(prop);
68
68
  if (!property) {
69
- record[prop] = value;
70
69
  return true;
71
70
  }
72
71
  property.setValue(value);
73
- this.validationPromise = this.validationPromise.then(() => property.setValueAndValidate(value));
72
+ this.validationPromise = this.validationPromise.then(() => property.validate());
74
73
  return true;
75
74
  }
76
75
  getPropertyValue(property) {
@@ -95,37 +94,66 @@ class SafeSettingsProxyHandler extends ProxyHandlerBase {
95
94
  }
96
95
  }
97
96
  class PluginSettingsManagerBase {
97
+ /**
98
+ * Creates a new plugin settings manager.
99
+ *
100
+ * @param plugin - The plugin.
101
+ */
98
102
  constructor(plugin) {
99
103
  this.plugin = plugin;
100
104
  this.app = plugin.app;
101
- this.defaultSettings = this.createDefaultSettings();
105
+ this.currentSettings = this.createDefaultSettings();
102
106
  this.registerValidators();
103
107
  this.properties = new PropertiesMap();
104
- for (const propertyName of (0, import_Object.getAllKeys)(this.defaultSettings)) {
108
+ const record = this.currentSettings;
109
+ for (const propertyName of (0, import_Object.getAllKeys)(this.currentSettings)) {
110
+ let propertySetter2 = function(value) {
111
+ record[propertyName] = value;
112
+ };
113
+ var propertySetter = propertySetter2;
105
114
  this.properties.set(
106
115
  propertyName,
107
- new PluginSettingsProperty(propertyName, this.defaultSettings[propertyName], this.validators.get(propertyName) ?? import_Function.noop)
116
+ new PluginSettingsProperty(propertyName, this.currentSettings[propertyName], this.validators.get(propertyName) ?? import_Function.noop, propertySetter2)
108
117
  );
109
118
  }
110
119
  this.validators.clear();
111
- this.safeSettings = new Proxy(this.defaultSettings, new SafeSettingsProxyHandler(this.properties));
120
+ this.safeSettings = new Proxy(this.currentSettings, new SafeSettingsProxyHandler(this.properties));
112
121
  }
113
122
  app;
114
123
  safeSettings;
115
- defaultSettings;
124
+ currentSettings;
116
125
  properties;
117
126
  // eslint-disable-next-line @typescript-eslint/no-explicit-any
118
127
  validators = /* @__PURE__ */ new Map();
119
- async editAndSave(editor) {
120
- const editableSettings = new Proxy(this.defaultSettings, new EditableSettingsProxyHandler(this.properties));
128
+ /**
129
+ * Edits the plugin settings and saves them.
130
+ *
131
+ * @param editor - The editor.
132
+ * @param context - The context.
133
+ * @returns A {@link Promise} that resolves when the settings are saved.
134
+ */
135
+ async editAndSave(editor, context) {
136
+ const editableSettings = new Proxy(this.currentSettings, new EditableSettingsProxyHandler(this.properties));
121
137
  await editor(editableSettings);
122
138
  await editableSettings.validationPromise;
123
- await this.saveToFile();
139
+ await this.saveToFile(context);
124
140
  }
141
+ /**
142
+ * Gets a property of the plugin settings.
143
+ *
144
+ * @param propertyName - The name of the property.
145
+ * @returns The property.
146
+ */
125
147
  getProperty(propertyName) {
126
148
  return this.properties.getTyped(propertyName);
127
149
  }
128
- async loadFromFile() {
150
+ /**
151
+ * Loads the plugin settings from the file.
152
+ *
153
+ * @param isInitialLoad - Whether the settings are being loaded for the first time.
154
+ * @returns A {@link Promise} that resolves when the settings are loaded.
155
+ */
156
+ async loadFromFile(isInitialLoad) {
129
157
  for (const property of this.properties.values()) {
130
158
  property.reset();
131
159
  }
@@ -163,21 +191,22 @@ class PluginSettingsManagerBase {
163
191
  propertiesToSave.push(property);
164
192
  }
165
193
  for (const property of propertiesToSave) {
166
- await property.setValueAndValidate(property.currentValue);
194
+ await property.validate();
167
195
  property.save();
168
196
  }
169
197
  const newJson = JSON.stringify(await this.prepareRecordToSave());
170
198
  if (newJson !== originalJson) {
171
199
  await this.saveToFileImpl();
172
200
  }
173
- await this.plugin.onLoadSettings(this.getSavedSettings());
201
+ await this.plugin.onLoadSettings(this.getSavedSettings(), isInitialLoad);
174
202
  }
175
203
  /**
176
204
  * Saves the new plugin settings.
177
205
  *
206
+ * @param context - The context of the save to file operation.
178
207
  * @returns A {@link Promise} that resolves when the settings are saved.
179
208
  */
180
- async saveToFile() {
209
+ async saveToFile(context) {
181
210
  const oldSettings = this.getSavedSettings();
182
211
  let hasChanges = false;
183
212
  for (const property of this.properties.values()) {
@@ -187,8 +216,13 @@ class PluginSettingsManagerBase {
187
216
  return;
188
217
  }
189
218
  await this.saveToFileImpl();
190
- await this.plugin.onSaveSettings(this.getSavedSettings(), oldSettings);
219
+ await this.plugin.onSaveSettings(this.getSavedSettings(), oldSettings, context);
191
220
  }
221
+ /**
222
+ * Gets the transformer.
223
+ *
224
+ * @returns The transformer.
225
+ */
192
226
  getTransformer() {
193
227
  return defaultTransformer;
194
228
  }
@@ -208,9 +242,20 @@ class PluginSettingsManagerBase {
208
242
  async onSavingRecord(_record) {
209
243
  await (0, import_Function.noopAsync)();
210
244
  }
245
+ /**
246
+ * Registers a validator for a property.
247
+ *
248
+ * @param propertyName - The name of the property.
249
+ * @param validator - The validator.
250
+ */
211
251
  registerValidator(propertyName, validator) {
212
252
  this.validators.set(propertyName, validator);
213
253
  }
254
+ /**
255
+ * Registers the validators.
256
+ *
257
+ * This method can be overridden by subclasses to register validators for properties.
258
+ */
214
259
  registerValidators() {
215
260
  (0, import_Function.noop)();
216
261
  }
@@ -219,7 +264,7 @@ class PluginSettingsManagerBase {
219
264
  for (const [propertyName, property] of this.properties.entries()) {
220
265
  savedSettings[propertyName] = property.lastSavedValue;
221
266
  }
222
- const proto = Object.getPrototypeOf(this.defaultSettings);
267
+ const proto = Object.getPrototypeOf(this.currentSettings);
223
268
  Object.setPrototypeOf(savedSettings, proto);
224
269
  return savedSettings;
225
270
  }
@@ -236,32 +281,69 @@ class PluginSettingsManagerBase {
236
281
  }
237
282
  }
238
283
  class PluginSettingsProperty {
239
- constructor(propertyName, defaultValue, validator) {
284
+ /**
285
+ * Creates a new plugin settings property.
286
+ *
287
+ * @param propertyName - The name of the property.
288
+ * @param defaultValue - The default value of the property.
289
+ * @param validator - The validator of the property.
290
+ * @param propertySetter - The property setter of the property.
291
+ */
292
+ constructor(propertyName, defaultValue, validator, propertySetter) {
240
293
  this.propertyName = propertyName;
241
294
  this.defaultValue = defaultValue;
242
295
  this.validator = validator;
296
+ this.propertySetter = propertySetter;
243
297
  this._lastSavedValue = defaultValue;
244
298
  this._currentValue = defaultValue;
245
299
  }
300
+ /**
301
+ * The current value of the property.
302
+ *
303
+ * @returns The current value.
304
+ */
246
305
  get currentValue() {
247
306
  return this._currentValue;
248
307
  }
308
+ /**
309
+ * The last saved value of the property.
310
+ *
311
+ * @returns The last saved value.
312
+ */
249
313
  get lastSavedValue() {
250
314
  return this._lastSavedValue;
251
315
  }
316
+ /**
317
+ * The safe value of the property.
318
+ *
319
+ * @returns The safe value.
320
+ */
252
321
  get safeValue() {
253
322
  return this._validationMessage ? this.defaultValue : this._currentValue;
254
323
  }
324
+ /**
325
+ * The validation message of the property.
326
+ *
327
+ * @returns The validation message.
328
+ */
255
329
  get validationMessage() {
256
330
  return this._validationMessage;
257
331
  }
258
332
  _currentValue;
259
333
  _lastSavedValue;
260
334
  _validationMessage = "";
335
+ /**
336
+ * Resets the current value of the property to the default value.
337
+ */
261
338
  reset() {
262
339
  this._currentValue = this.defaultValue;
263
340
  this._validationMessage = "";
264
341
  }
342
+ /**
343
+ * Saves the current value of the property.
344
+ *
345
+ * @returns `true` if the value was changed, `false` otherwise.
346
+ */
265
347
  save() {
266
348
  if (this._lastSavedValue === this._currentValue) {
267
349
  return false;
@@ -269,25 +351,40 @@ class PluginSettingsProperty {
269
351
  this._lastSavedValue = this._currentValue;
270
352
  return true;
271
353
  }
354
+ /**
355
+ * Sets the validation message of the property.
356
+ *
357
+ * @param validationMessage - The validation message.
358
+ */
272
359
  setValidationMessage(validationMessage) {
273
360
  this._validationMessage = validationMessage;
274
361
  this.showWarning();
275
362
  }
363
+ /**
364
+ * Sets the current value of the property.
365
+ *
366
+ * @param value - The value to set.
367
+ */
276
368
  setValue(value) {
277
369
  this._currentValue = value;
370
+ this.propertySetter(value);
278
371
  }
279
- async setValueAndValidate(value) {
280
- this.setValue(value);
372
+ /**
373
+ * Validates the current value of the property.
374
+ *
375
+ * @returns A {@link Promise} that resolves when the validation is complete.
376
+ */
377
+ async validate() {
281
378
  try {
282
379
  this._validationMessage = await this.validator(this._currentValue) ?? "";
283
380
  } catch (error) {
284
381
  console.error("Validation failed", {
285
382
  propertyName: this.propertyName,
286
- value
383
+ value: this._currentValue
287
384
  }, error);
288
385
  this._validationMessage = "Validation failed";
289
386
  }
290
- this.showWarning(value);
387
+ this.showWarning(this._currentValue);
291
388
  }
292
389
  showWarning(value) {
293
390
  if (!this._validationMessage) {
@@ -306,4 +403,4 @@ class PluginSettingsProperty {
306
403
  PluginSettingsManagerBase,
307
404
  PluginSettingsProperty
308
405
  });
309
- //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../../../src/obsidian/Plugin/PluginSettingsManagerBase.ts"],
  "sourcesContent": ["/**\n * @packageDocumentation\n *\n * Plugin settings manager base class.\n */\n\nimport type { App } from 'obsidian';\nimport type {\n  Promisable,\n  ReadonlyDeep\n} from 'type-fest';\n\nimport type { Transformer } from '../../Transformers/Transformer.ts';\nimport type {\n  MaybeReturn,\n  StringKeys\n} from '../../Type.ts';\nimport type {\n  ExtractPlugin,\n  ExtractPluginSettings,\n  PluginTypesBase\n} from './PluginTypesBase.ts';\n\nimport {\n  noop,\n  noopAsync\n} 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 PluginTypes - Plugin-specific types.\n */\nexport abstract class PluginSettingsManagerBase<PluginTypes extends PluginTypesBase> {\n  public readonly app: App;\n  public readonly safeSettings: ReadonlyDeep<ExtractPluginSettings<PluginTypes>>;\n\n  private defaultSettings: ExtractPluginSettings<PluginTypes>;\n  private properties: PropertiesMap<ExtractPluginSettings<PluginTypes>>;\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: ExtractPlugin<PluginTypes>) {\n    this.app = plugin.app;\n    this.defaultSettings = this.createDefaultSettings();\n\n    this.registerValidators();\n\n    this.properties = new PropertiesMap<ExtractPluginSettings<PluginTypes>>();\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<ExtractPluginSettings<PluginTypes>>(this.properties)) as ReadonlyDeep<\n      ExtractPluginSettings<PluginTypes>\n    >;\n  }\n\n  public async editAndSave(editor: (settings: ExtractPluginSettings<PluginTypes>) => Promisable<void>): Promise<void> {\n    const editableSettings = new Proxy(this.defaultSettings, new EditableSettingsProxyHandler<ExtractPluginSettings<PluginTypes>>(this.properties)) as {\n      validationPromise: Promise<void>;\n    } & ExtractPluginSettings<PluginTypes>;\n    await editor(editableSettings);\n    await editableSettings.validationPromise;\n    await this.saveToFile();\n  }\n\n  public getProperty<PropertyName extends StringKeys<ExtractPluginSettings<PluginTypes>>>(\n    propertyName: PropertyName\n  ): PluginSettingsProperty<ExtractPluginSettings<PluginTypes>[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    const propertiesToSave: PluginSettingsProperty<unknown>[] = [];\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(\n          'Possible invalid value type read from config file. 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: property.defaultValue as unknown,\n            propertyName,\n            value\n          }\n        );\n      }\n\n      property.setValue(value);\n      propertiesToSave.push(property);\n    }\n\n    for (const property of propertiesToSave) {\n      await property.setValueAndValidate(property.currentValue);\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 {@link 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 abstract createDefaultSettings(): ExtractPluginSettings<PluginTypes>;\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   */\n  protected async onLoadRecord(_record: Record<string, unknown>): Promise<void> {\n    await noopAsync();\n  }\n\n  /**\n   * Called when the plugin settings are saving.\n   *\n   * @param _record - The record.\n   */\n  protected async onSavingRecord(_record: Record<string, unknown>): Promise<void> {\n    await noopAsync();\n  }\n\n  protected registerValidator<PropertyName extends StringKeys<ExtractPluginSettings<PluginTypes>>>(\n    propertyName: PropertyName,\n    validator: Validator<ExtractPluginSettings<PluginTypes>[PropertyName]>\n  ): void {\n    this.validators.set(propertyName, validator);\n  }\n\n  protected registerValidators(): void {\n    noop();\n  }\n\n  private getSavedSettings(): ExtractPluginSettings<PluginTypes> {\n    const savedSettings: Record<string, unknown> = {};\n    for (const [propertyName, property] of this.properties.entries()) {\n      savedSettings[propertyName] = property.lastSavedValue as\n        | ExtractPluginSettings<PluginTypes>[StringKeys<ExtractPluginSettings<PluginTypes>>]\n        | undefined;\n    }\n    const proto = Object.getPrototypeOf(this.defaultSettings) as object;\n    Object.setPrototypeOf(savedSettings, proto);\n\n    return savedSettings as ExtractPluginSettings<PluginTypes>;\n  }\n\n  private async prepareRecordToSave(): Promise<Record<string, unknown>> {\n    const settings: Record<string, unknown> = {};\n    for (const [propertyName, property] of this.properties.entries()) {\n      settings[propertyName] = 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    try {\n      this._validationMessage = (await this.validator(this._currentValue) as string | undefined) ?? '';\n    } catch (error) {\n      console.error('Validation failed', {\n        propertyName: this.propertyName,\n        value\n      }, error);\n      this._validationMessage = 'Validation failed';\n    }\n    this.showWarning(value);\n  }\n\n  private showWarning(value?: T): void {\n    if (!this._validationMessage) {\n      return;\n    }\n\n    console.warn(`Could not set plugin setting: ${this.propertyName}. Using default value instead.`, {\n      defaultValue: this.defaultValue,\n      propertyName: this.propertyName,\n      validationMessage: this._validationMessage,\n      value\n    });\n  }\n}\n"],
  "mappings": "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",
  "names": []
}

406
+ //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../../../src/obsidian/Plugin/PluginSettingsManagerBase.ts"],
  "sourcesContent": ["/**\n * @packageDocumentation\n *\n * Plugin settings manager base class.\n */\n\nimport type { App } from 'obsidian';\nimport type {\n  Promisable,\n  ReadonlyDeep\n} from 'type-fest';\n\nimport type { Transformer } from '../../Transformers/Transformer.ts';\nimport type {\n  MaybeReturn,\n  StringKeys\n} from '../../Type.ts';\nimport type {\n  ExtractPlugin,\n  ExtractPluginSettings,\n  PluginTypesBase\n} from './PluginTypesBase.ts';\n\nimport {\n  noop,\n  noopAsync\n} 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 PropertySetter = (value: unknown) => void;\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    record[prop] = value;\n\n    if (typeof prop !== 'string') {\n      return true;\n    }\n\n    const property = this.properties.get(prop);\n    if (!property) {\n      return true;\n    }\n\n    property.setValue(value);\n    this.validationPromise = this.validationPromise.then(() => property.validate());\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 PluginTypes - Plugin-specific types.\n */\nexport abstract class PluginSettingsManagerBase<PluginTypes extends PluginTypesBase> {\n  public readonly app: App;\n  public readonly safeSettings: ReadonlyDeep<ExtractPluginSettings<PluginTypes>>;\n\n  private readonly currentSettings: ExtractPluginSettings<PluginTypes>;\n  private readonly properties: PropertiesMap<ExtractPluginSettings<PluginTypes>>;\n  // eslint-disable-next-line @typescript-eslint/no-explicit-any\n  private readonly validators: Map<string, Validator<any>> = new Map<string, Validator<any>>();\n\n  /**\n   * Creates a new plugin settings manager.\n   *\n   * @param plugin - The plugin.\n   */\n  public constructor(public readonly plugin: ExtractPlugin<PluginTypes>) {\n    this.app = plugin.app;\n    this.currentSettings = this.createDefaultSettings();\n\n    this.registerValidators();\n\n    this.properties = new PropertiesMap<ExtractPluginSettings<PluginTypes>>();\n\n    const record = this.currentSettings as Record<string, unknown>;\n\n    for (const propertyName of getAllKeys(this.currentSettings)) {\n      function propertySetter(value: unknown): void {\n        record[propertyName] = value;\n      }\n\n      this.properties.set(\n        propertyName,\n        new PluginSettingsProperty(propertyName, this.currentSettings[propertyName], this.validators.get(propertyName) ?? noop, propertySetter)\n      );\n    }\n\n    this.validators.clear();\n\n    this.safeSettings = new Proxy(this.currentSettings, new SafeSettingsProxyHandler<ExtractPluginSettings<PluginTypes>>(this.properties)) as ReadonlyDeep<\n      ExtractPluginSettings<PluginTypes>\n    >;\n  }\n\n  /**\n   * Edits the plugin settings and saves them.\n   *\n   * @param editor - The editor.\n   * @param context - The context.\n   * @returns A {@link Promise} that resolves when the settings are saved.\n   */\n  public async editAndSave(editor: (settings: ExtractPluginSettings<PluginTypes>) => Promisable<void>, context?: unknown): Promise<void> {\n    const editableSettings = new Proxy(this.currentSettings, new EditableSettingsProxyHandler<ExtractPluginSettings<PluginTypes>>(this.properties)) as {\n      validationPromise: Promise<void>;\n    } & ExtractPluginSettings<PluginTypes>;\n    await editor(editableSettings);\n    await editableSettings.validationPromise;\n    await this.saveToFile(context);\n  }\n\n  /**\n   * Gets a property of the plugin settings.\n   *\n   * @param propertyName - The name of the property.\n   * @returns The property.\n   */\n  public getProperty<PropertyName extends StringKeys<ExtractPluginSettings<PluginTypes>>>(\n    propertyName: PropertyName\n  ): PluginSettingsProperty<ExtractPluginSettings<PluginTypes>[PropertyName]> {\n    return this.properties.getTyped(propertyName);\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    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    const propertiesToSave: PluginSettingsProperty<unknown>[] = [];\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(\n          'Possible invalid value type read from config file. 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: property.defaultValue as unknown,\n            propertyName,\n            value\n          }\n        );\n      }\n\n      property.setValue(value);\n      propertiesToSave.push(property);\n    }\n\n    for (const property of propertiesToSave) {\n      await property.validate();\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(), isInitialLoad);\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    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, context);\n  }\n\n  protected abstract createDefaultSettings(): ExtractPluginSettings<PluginTypes>;\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 are loaded.\n   *\n   * @param _record - The record.\n   */\n  protected async onLoadRecord(_record: Record<string, unknown>): Promise<void> {\n    await noopAsync();\n  }\n\n  /**\n   * Called when the plugin settings are saving.\n   *\n   * @param _record - The record.\n   */\n  protected async onSavingRecord(_record: Record<string, unknown>): Promise<void> {\n    await noopAsync();\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  protected registerValidator<PropertyName extends StringKeys<ExtractPluginSettings<PluginTypes>>>(\n    propertyName: PropertyName,\n    validator: Validator<ExtractPluginSettings<PluginTypes>[PropertyName]>\n  ): void {\n    this.validators.set(propertyName, validator);\n  }\n\n  /**\n   * Registers the validators.\n   *\n   * This method can be overridden by subclasses to register validators for properties.\n   */\n  protected registerValidators(): void {\n    noop();\n  }\n\n  private getSavedSettings(): ExtractPluginSettings<PluginTypes> {\n    const savedSettings: Record<string, unknown> = {};\n    for (const [propertyName, property] of this.properties.entries()) {\n      savedSettings[propertyName] = property.lastSavedValue as\n        | ExtractPluginSettings<PluginTypes>[StringKeys<ExtractPluginSettings<PluginTypes>>]\n        | undefined;\n    }\n    const proto = Object.getPrototypeOf(this.currentSettings) as object;\n    Object.setPrototypeOf(savedSettings, proto);\n\n    return savedSettings as ExtractPluginSettings<PluginTypes>;\n  }\n\n  private async prepareRecordToSave(): Promise<Record<string, unknown>> {\n    const settings: Record<string, unknown> = {};\n    for (const [propertyName, property] of this.properties.entries()) {\n      settings[propertyName] = 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  /**\n   * The current value of the property.\n   *\n   * @returns The current value.\n   */\n  public get currentValue(): T {\n    return this._currentValue;\n  }\n\n  /**\n   * The last saved value of the property.\n   *\n   * @returns The last saved value.\n   */\n  public get lastSavedValue(): T {\n    return this._lastSavedValue;\n  }\n\n  /**\n   * The safe value of the property.\n   *\n   * @returns The safe value.\n   */\n  public get safeValue(): T {\n    return this._validationMessage ? this.defaultValue : this._currentValue;\n  }\n\n  /**\n   * The validation message of the property.\n   *\n   * @returns The validation message.\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  /**\n   * Creates a new plugin settings property.\n   *\n   * @param propertyName - The name of the property.\n   * @param defaultValue - The default value of the property.\n   * @param validator - The validator of the property.\n   * @param propertySetter - The property setter of the property.\n   */\n  public constructor(\n    private readonly propertyName: string,\n    public readonly defaultValue: T,\n    private readonly validator: Validator<T>,\n    private readonly propertySetter: PropertySetter\n  ) {\n    this._lastSavedValue = defaultValue;\n    this._currentValue = defaultValue;\n  }\n\n  /**\n   * Resets the current value of the property to the default value.\n   */\n  public reset(): void {\n    this._currentValue = this.defaultValue;\n    this._validationMessage = '';\n  }\n\n  /**\n   * Saves the current value of the property.\n   *\n   * @returns `true` if the value was changed, `false` otherwise.\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  /**\n   * Sets the validation message of the property.\n   *\n   * @param validationMessage - The validation message.\n   */\n  public setValidationMessage(validationMessage: string): void {\n    this._validationMessage = validationMessage;\n    this.showWarning();\n  }\n\n  /**\n   * Sets the current value of the property.\n   *\n   * @param value - The value to set.\n   */\n  public setValue(value: T): void {\n    this._currentValue = value;\n    this.propertySetter(value);\n  }\n\n  /**\n   * Validates the current value of the property.\n   *\n   * @returns A {@link Promise} that resolves when the validation is complete.\n   */\n  public async validate(): Promise<void> {\n    try {\n      this._validationMessage = (await this.validator(this._currentValue) as string | undefined) ?? '';\n    } catch (error) {\n      console.error('Validation failed', {\n        propertyName: this.propertyName,\n        value: this._currentValue\n      }, error);\n      this._validationMessage = 'Validation failed';\n    }\n    this.showWarning(this._currentValue);\n  }\n\n  private showWarning(value?: T): void {\n    if (!this._validationMessage) {\n      return;\n    }\n\n    console.warn(`Could not set plugin setting: ${this.propertyName}. Using default value instead.`, {\n      defaultValue: this.defaultValue,\n      propertyName: this.propertyName,\n      validationMessage: this._validationMessage,\n      value\n    });\n  }\n}\n"],
  "mappings": "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",
  "names": ["propertySetter"]
}

@@ -8,6 +8,7 @@ import type { Promisable, ReadonlyDeep } from 'type-fest';
8
8
  import type { Transformer } from '../../Transformers/Transformer.cjs';
9
9
  import type { MaybeReturn, StringKeys } from '../../Type.cjs';
10
10
  import type { ExtractPlugin, ExtractPluginSettings, PluginTypesBase } from './PluginTypesBase.cjs';
11
+ type PropertySetter = (value: unknown) => void;
11
12
  type Validator<T> = (value: T) => Promisable<MaybeReturn<string>>;
12
13
  /**
13
14
  * Base class for managing plugin settings.
@@ -18,20 +19,50 @@ export declare abstract class PluginSettingsManagerBase<PluginTypes extends Plug
18
19
  readonly plugin: ExtractPlugin<PluginTypes>;
19
20
  readonly app: App;
20
21
  readonly safeSettings: ReadonlyDeep<ExtractPluginSettings<PluginTypes>>;
21
- private defaultSettings;
22
- private properties;
23
- private validators;
22
+ private readonly currentSettings;
23
+ private readonly properties;
24
+ private readonly validators;
25
+ /**
26
+ * Creates a new plugin settings manager.
27
+ *
28
+ * @param plugin - The plugin.
29
+ */
24
30
  constructor(plugin: ExtractPlugin<PluginTypes>);
25
- editAndSave(editor: (settings: ExtractPluginSettings<PluginTypes>) => Promisable<void>): Promise<void>;
31
+ /**
32
+ * Edits the plugin settings and saves them.
33
+ *
34
+ * @param editor - The editor.
35
+ * @param context - The context.
36
+ * @returns A {@link Promise} that resolves when the settings are saved.
37
+ */
38
+ editAndSave(editor: (settings: ExtractPluginSettings<PluginTypes>) => Promisable<void>, context?: unknown): Promise<void>;
39
+ /**
40
+ * Gets a property of the plugin settings.
41
+ *
42
+ * @param propertyName - The name of the property.
43
+ * @returns The property.
44
+ */
26
45
  getProperty<PropertyName extends StringKeys<ExtractPluginSettings<PluginTypes>>>(propertyName: PropertyName): PluginSettingsProperty<ExtractPluginSettings<PluginTypes>[PropertyName]>;
27
- loadFromFile(): Promise<void>;
46
+ /**
47
+ * Loads the plugin settings from the file.
48
+ *
49
+ * @param isInitialLoad - Whether the settings are being loaded for the first time.
50
+ * @returns A {@link Promise} that resolves when the settings are loaded.
51
+ */
52
+ loadFromFile(isInitialLoad: boolean): Promise<void>;
28
53
  /**
29
54
  * Saves the new plugin settings.
30
55
  *
56
+ * @param context - The context of the save to file operation.
31
57
  * @returns A {@link Promise} that resolves when the settings are saved.
32
58
  */
33
- saveToFile(): Promise<void>;
59
+ saveToFile(context?: unknown): Promise<void>;
34
60
  protected abstract createDefaultSettings(): ExtractPluginSettings<PluginTypes>;
61
+ /**
62
+ * Gets the transformer.
63
+ *
64
+ * @returns The transformer.
65
+ */
35
66
  protected getTransformer(): Transformer;
36
67
  /**
37
68
  * Called when the plugin settings are loaded.
@@ -45,7 +76,18 @@ export declare abstract class PluginSettingsManagerBase<PluginTypes extends Plug
45
76
  * @param _record - The record.
46
77
  */
47
78
  protected onSavingRecord(_record: Record<string, unknown>): Promise<void>;
79
+ /**
80
+ * Registers a validator for a property.
81
+ *
82
+ * @param propertyName - The name of the property.
83
+ * @param validator - The validator.
84
+ */
48
85
  protected registerValidator<PropertyName extends StringKeys<ExtractPluginSettings<PluginTypes>>>(propertyName: PropertyName, validator: Validator<ExtractPluginSettings<PluginTypes>[PropertyName]>): void;
86
+ /**
87
+ * Registers the validators.
88
+ *
89
+ * This method can be overridden by subclasses to register validators for properties.
90
+ */
49
91
  protected registerValidators(): void;
50
92
  private getSavedSettings;
51
93
  private prepareRecordToSave;
@@ -60,19 +102,71 @@ export declare class PluginSettingsProperty<T> {
60
102
  private readonly propertyName;
61
103
  readonly defaultValue: T;
62
104
  private readonly validator;
105
+ private readonly propertySetter;
106
+ /**
107
+ * The current value of the property.
108
+ *
109
+ * @returns The current value.
110
+ */
63
111
  get currentValue(): T;
112
+ /**
113
+ * The last saved value of the property.
114
+ *
115
+ * @returns The last saved value.
116
+ */
64
117
  get lastSavedValue(): T;
118
+ /**
119
+ * The safe value of the property.
120
+ *
121
+ * @returns The safe value.
122
+ */
65
123
  get safeValue(): T;
124
+ /**
125
+ * The validation message of the property.
126
+ *
127
+ * @returns The validation message.
128
+ */
66
129
  get validationMessage(): string;
67
130
  private _currentValue;
68
131
  private _lastSavedValue;
69
132
  private _validationMessage;
70
- constructor(propertyName: string, defaultValue: T, validator: Validator<T>);
133
+ /**
134
+ * Creates a new plugin settings property.
135
+ *
136
+ * @param propertyName - The name of the property.
137
+ * @param defaultValue - The default value of the property.
138
+ * @param validator - The validator of the property.
139
+ * @param propertySetter - The property setter of the property.
140
+ */
141
+ constructor(propertyName: string, defaultValue: T, validator: Validator<T>, propertySetter: PropertySetter);
142
+ /**
143
+ * Resets the current value of the property to the default value.
144
+ */
71
145
  reset(): void;
146
+ /**
147
+ * Saves the current value of the property.
148
+ *
149
+ * @returns `true` if the value was changed, `false` otherwise.
150
+ */
72
151
  save(): boolean;
152
+ /**
153
+ * Sets the validation message of the property.
154
+ *
155
+ * @param validationMessage - The validation message.
156
+ */
73
157
  setValidationMessage(validationMessage: string): void;
158
+ /**
159
+ * Sets the current value of the property.
160
+ *
161
+ * @param value - The value to set.
162
+ */
74
163
  setValue(value: T): void;
75
- setValueAndValidate(value: T): Promise<void>;
164
+ /**
165
+ * Validates the current value of the property.
166
+ *
167
+ * @returns A {@link Promise} that resolves when the validation is complete.
168
+ */
169
+ validate(): Promise<void>;
76
170
  private showWarning;
77
171
  }
78
172
  export {};