obsidian-dev-utils 24.2.0 → 24.2.1-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 (28) hide show
  1. package/CHANGELOG.md +8 -0
  2. package/dist/lib/cjs/Library.cjs +1 -1
  3. package/dist/lib/cjs/obsidian/Plugin/PluginBase.cjs +4 -4
  4. package/dist/lib/cjs/obsidian/Plugin/PluginBase.d.cts +4 -4
  5. package/dist/lib/cjs/obsidian/Plugin/PluginSettingsManagerBase.cjs +145 -273
  6. package/dist/lib/cjs/obsidian/Plugin/PluginSettingsManagerBase.d.cts +42 -90
  7. package/dist/lib/cjs/obsidian/Plugin/PluginSettingsTabBase.cjs +28 -19
  8. package/dist/lib/cjs/obsidian/Plugin/PluginSettingsTabBase.d.cts +1 -0
  9. package/dist/lib/cjs/obsidian/Plugin/PluginSettingsWrapper.cjs +24 -0
  10. package/dist/lib/cjs/obsidian/Plugin/PluginSettingsWrapper.d.cts +25 -0
  11. package/dist/lib/cjs/obsidian/Plugin/PluginTypesBase.cjs +1 -1
  12. package/dist/lib/cjs/obsidian/Plugin/PluginTypesBase.d.cts +14 -0
  13. package/dist/lib/cjs/obsidian/Plugin/index.cjs +4 -1
  14. package/dist/lib/cjs/obsidian/Plugin/index.d.cts +1 -0
  15. package/dist/lib/esm/Library.mjs +1 -1
  16. package/dist/lib/esm/obsidian/Plugin/PluginBase.d.mts +4 -4
  17. package/dist/lib/esm/obsidian/Plugin/PluginBase.mjs +4 -4
  18. package/dist/lib/esm/obsidian/Plugin/PluginSettingsManagerBase.d.mts +42 -90
  19. package/dist/lib/esm/obsidian/Plugin/PluginSettingsManagerBase.mjs +148 -272
  20. package/dist/lib/esm/obsidian/Plugin/PluginSettingsTabBase.d.mts +1 -0
  21. package/dist/lib/esm/obsidian/Plugin/PluginSettingsTabBase.mjs +28 -19
  22. package/dist/lib/esm/obsidian/Plugin/PluginSettingsWrapper.d.mts +25 -0
  23. package/dist/lib/esm/obsidian/Plugin/PluginSettingsWrapper.mjs +8 -0
  24. package/dist/lib/esm/obsidian/Plugin/PluginTypesBase.d.mts +14 -0
  25. package/dist/lib/esm/obsidian/Plugin/index.d.mts +1 -0
  26. package/dist/lib/esm/obsidian/Plugin/index.mjs +3 -1
  27. package/obsidian/Plugin/PluginSettingsWrapper/package.json +6 -0
  28. package/package.json +1 -1
@@ -25,8 +25,7 @@ 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,
29
- PluginSettingsProperty: () => PluginSettingsProperty
28
+ PluginSettingsManagerBase: () => PluginSettingsManagerBase
30
29
  });
31
30
  module.exports = __toCommonJS(PluginSettingsManagerBase_exports);
32
31
  var import_Function = require('../../Function.cjs');
@@ -40,61 +39,6 @@ const defaultTransformer = new import_GroupTransformer.GroupTransformer([
40
39
  new import_DateTransformer.DateTransformer(),
41
40
  new import_DurationTransformer.DurationTransformer()
42
41
  ]);
43
- class ProxyHandlerBase {
44
- constructor(properties) {
45
- this.properties = properties;
46
- }
47
- get(target, prop) {
48
- const record = target;
49
- if (typeof prop !== "string") {
50
- return record[prop];
51
- }
52
- const property = this.properties.get(prop);
53
- if (!property) {
54
- return record[prop];
55
- }
56
- return this.getPropertyValue(property);
57
- }
58
- }
59
- class EditableSettingsProxyHandler extends ProxyHandlerBase {
60
- validationPromise = Promise.resolve();
61
- set(target, prop, value) {
62
- const record = target;
63
- record[prop] = value;
64
- if (typeof prop !== "string") {
65
- return true;
66
- }
67
- const property = this.properties.get(prop);
68
- if (!property) {
69
- return true;
70
- }
71
- property.setValue(value);
72
- this.validationPromise = this.validationPromise.then(async () => {
73
- await property.validate();
74
- });
75
- return true;
76
- }
77
- getPropertyValue(property) {
78
- return property.currentValue;
79
- }
80
- }
81
- class PropertiesMap extends Map {
82
- getTyped(propertyName) {
83
- const property = super.get(propertyName);
84
- if (!property) {
85
- throw new Error(`Property ${String(propertyName)} not found`);
86
- }
87
- return property;
88
- }
89
- setTyped(propertyName, value) {
90
- return super.set(propertyName, value);
91
- }
92
- }
93
- class SafeSettingsProxyHandler extends ProxyHandlerBase {
94
- getPropertyValue(property) {
95
- return property.safeValue;
96
- }
97
- }
98
42
  class PluginSettingsManagerBase {
99
43
  /**
100
44
  * Creates a new plugin settings manager.
@@ -104,34 +48,35 @@ class PluginSettingsManagerBase {
104
48
  constructor(plugin) {
105
49
  this.plugin = plugin;
106
50
  this.app = plugin.app;
107
- this.currentSettings = this.createDefaultSettings();
51
+ this.defaultSettings = this.createDefaultSettings();
52
+ this.currentSettingsWrapper = this.createDefaultSettingsWrapper();
53
+ this.lastSavedSettingsWrapper = this.createDefaultSettingsWrapper();
54
+ this.propertyNames = (0, import_Object.getAllKeys)(this.currentSettingsWrapper.settings);
108
55
  this.registerValidators();
109
- this.properties = new PropertiesMap();
110
- for (const propertyName of (0, import_Object.getAllKeys)(this.currentSettings)) {
111
- this.properties.set(
112
- propertyName,
113
- new PluginSettingsProperty(propertyName, this.currentSettings, this.validators.get(propertyName) ?? import_Function.noop)
114
- );
115
- }
116
- this.validators.clear();
117
- this.safeSettings = new Proxy(this.currentSettings, new SafeSettingsProxyHandler(this.properties));
118
56
  }
119
57
  app;
120
- safeSettings;
121
- currentSettings;
122
- properties;
58
+ defaultSettings;
59
+ /**
60
+ * Gets the current settings wrapper.
61
+ *
62
+ * @returns The current settings wrapper.
63
+ */
64
+ get settingsWrapper() {
65
+ return this.currentSettingsWrapper;
66
+ }
67
+ currentSettingsWrapper;
68
+ lastSavedSettingsWrapper;
69
+ propertyNames;
123
70
  validators = /* @__PURE__ */ new Map();
124
71
  /**
125
72
  * Edits the plugin settings and saves them.
126
73
  *
127
- * @param editor - The editor.
74
+ * @param settingsEditor - The editor.
128
75
  * @param context - The context.
129
76
  * @returns A {@link Promise} that resolves when the settings are saved.
130
77
  */
131
- async editAndSave(editor, context) {
132
- const editableSettings = new Proxy(this.currentSettings, new EditableSettingsProxyHandler(this.properties));
133
- await editor(editableSettings);
134
- await editableSettings.validationPromise;
78
+ async editAndSave(settingsEditor, context) {
79
+ await this.edit(settingsEditor);
135
80
  await this.saveToFile(context);
136
81
  }
137
82
  /**
@@ -143,23 +88,23 @@ class PluginSettingsManagerBase {
143
88
  * @returns A {@link Promise} that resolves when the settings are safe.
144
89
  */
145
90
  async ensureSafe(settings) {
146
- const record = settings;
147
- for (const propertyName of (0, import_Object.getAllKeys)(settings)) {
148
- const property = this.getProperty(propertyName);
149
- const validationMessage = await property.validate(settings);
150
- if (validationMessage) {
151
- record[propertyName] = property.defaultValue;
91
+ const validationResult = await this.validate(settings);
92
+ for (const propertyName of this.propertyNames) {
93
+ if (validationResult[propertyName]) {
94
+ settings[propertyName] = this.defaultSettings[propertyName];
152
95
  }
153
96
  }
154
97
  }
155
98
  /**
156
- * Gets a property of the plugin settings.
99
+ * Gets a safe copy of the settings.
157
100
  *
158
- * @param propertyName - The name of the property.
159
- * @returns The property.
101
+ * @param settings - The settings.
102
+ * @returns A {@link Promise} that resolves to the safe copy of the settings.
160
103
  */
161
- getProperty(propertyName) {
162
- return this.properties.getTyped(propertyName);
104
+ async getSafeCopy(settings) {
105
+ const safeSettings = this.cloneSettings(settings);
106
+ await this.ensureSafe(safeSettings);
107
+ return safeSettings;
163
108
  }
164
109
  /**
165
110
  * Loads the plugin settings from the file.
@@ -168,51 +113,28 @@ class PluginSettingsManagerBase {
168
113
  * @returns A {@link Promise} that resolves when the settings are loaded.
169
114
  */
170
115
  async loadFromFile(isInitialLoad) {
171
- for (const property of this.properties.values()) {
172
- property.reset();
173
- }
174
116
  const data = await this.plugin.loadData();
117
+ this.lastSavedSettingsWrapper = this.createDefaultSettingsWrapper();
118
+ this.currentSettingsWrapper = this.createDefaultSettingsWrapper();
175
119
  if (data === void 0 || data === null) {
176
120
  return;
177
121
  }
178
- if (typeof data !== "object" || Array.isArray(data)) {
179
- const type = Array.isArray(data) ? "Array" : typeof data;
180
- console.error(`Invalid data type. Expected Object, got: ${type}`);
122
+ if (typeof data !== "object") {
123
+ console.error(`Invalid settings from data.json. Expected Object, got: ${typeof data}`);
181
124
  return;
182
125
  }
183
- let record = data;
184
- const originalJson = JSON.stringify(record);
185
- record = this.getTransformer().transformObjectRecursively(record);
186
- await this.onLoadRecord(record);
187
- const propertiesToSave = [];
188
- for (const [propertyName, value] of Object.entries(record)) {
189
- const property = this.properties.get(propertyName);
190
- if (!property) {
191
- console.warn(`Unknown property: ${propertyName}`);
192
- continue;
193
- }
194
- if (typeof value !== typeof property.defaultValue) {
195
- console.warn(
196
- "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.",
197
- {
198
- defaultValue: property.defaultValue,
199
- propertyName,
200
- value
201
- }
202
- );
203
- }
204
- property.setValue(value);
205
- propertiesToSave.push(property);
206
- }
207
- for (const property of propertiesToSave) {
208
- await property.validate();
209
- property.save();
126
+ const rawRecord = data;
127
+ const parsedSettings = await this.rawRecordToSettings(rawRecord);
128
+ const validationResult = await this.validate(parsedSettings);
129
+ for (const propertyName of this.propertyNames) {
130
+ this.setPropertyImpl(propertyName, parsedSettings[propertyName], validationResult[propertyName]);
210
131
  }
211
- const newJson = JSON.stringify(await this.prepareRecordToSave());
212
- if (newJson !== originalJson) {
132
+ this.lastSavedSettingsWrapper = this.cloneSettingsWrapper(this.currentSettingsWrapper);
133
+ const newRecord = await this.settingsToRawRecord(this.currentSettingsWrapper.settings);
134
+ if (!(0, import_Object.deepEqual)(newRecord, data)) {
213
135
  await this.saveToFileImpl();
214
136
  }
215
- await this.plugin.onLoadSettings(this.getSavedSettings(), isInitialLoad);
137
+ await this.plugin.onLoadSettings(this.currentSettingsWrapper, isInitialLoad);
216
138
  }
217
139
  /**
218
140
  * Saves the new plugin settings.
@@ -221,16 +143,42 @@ class PluginSettingsManagerBase {
221
143
  * @returns A {@link Promise} that resolves when the settings are saved.
222
144
  */
223
145
  async saveToFile(context) {
224
- const oldSettings = this.getSavedSettings();
225
- let hasChanges = false;
226
- for (const property of this.properties.values()) {
227
- hasChanges ||= property.save();
228
- }
229
- if (!hasChanges) {
146
+ if ((0, import_Object.deepEqual)(this.lastSavedSettingsWrapper.settings, this.currentSettingsWrapper.settings)) {
230
147
  return;
231
148
  }
232
149
  await this.saveToFileImpl();
233
- await this.plugin.onSaveSettings(this.getSavedSettings(), oldSettings, context);
150
+ await this.plugin.onSaveSettings(this.currentSettingsWrapper, this.lastSavedSettingsWrapper, context);
151
+ this.lastSavedSettingsWrapper = this.cloneSettingsWrapper(this.currentSettingsWrapper);
152
+ }
153
+ /**
154
+ * Sets the value of a property.
155
+ *
156
+ * @typeParam PropertyName - The name of the property.
157
+ * @param propertyName - The name of the property.
158
+ * @param value - The value to set.
159
+ * @returns A {@link Promise} that resolves to the validation message.
160
+ */
161
+ async setProperty(propertyName, value) {
162
+ await this.edit((settings) => {
163
+ settings[propertyName] = value;
164
+ });
165
+ return this.currentSettingsWrapper.validationMessages[propertyName];
166
+ }
167
+ /**
168
+ * Validates the settings.
169
+ *
170
+ * @param settings - The settings.
171
+ * @returns A {@link Promise} that resolves to the validation result.
172
+ */
173
+ async validate(settings) {
174
+ const result = {};
175
+ for (const [propertyName, validator] of this.validators.entries()) {
176
+ const validationMessage = await validator(settings[propertyName], settings);
177
+ if (validationMessage) {
178
+ result[propertyName] = validationMessage;
179
+ }
180
+ }
181
+ return result;
234
182
  }
235
183
  /**
236
184
  * Gets the transformer.
@@ -273,161 +221,85 @@ class PluginSettingsManagerBase {
273
221
  registerValidators() {
274
222
  (0, import_Function.noop)();
275
223
  }
276
- getSavedSettings() {
277
- const savedSettings = {};
278
- for (const [propertyName, property] of this.properties.entries()) {
279
- savedSettings[propertyName] = property.lastSavedValue;
280
- }
281
- const proto = Object.getPrototypeOf(this.currentSettings);
282
- Object.setPrototypeOf(savedSettings, proto);
283
- return savedSettings;
284
- }
285
- async prepareRecordToSave() {
286
- const settings = {};
287
- for (const [propertyName, property] of this.properties.entries()) {
288
- settings[propertyName] = property.currentValue;
224
+ cloneSettings(settings) {
225
+ const record = this.settingsToRawRecord(settings);
226
+ const json = JSON.stringify(record);
227
+ const cloneRecord = JSON.parse(json);
228
+ return this.rawRecordToSettings(cloneRecord);
229
+ }
230
+ cloneSettingsWrapper(settingsWrapper) {
231
+ return {
232
+ safeSettings: this.cloneSettings(settingsWrapper.safeSettings),
233
+ settings: this.cloneSettings(settingsWrapper.settings),
234
+ validationMessages: { ...settingsWrapper.validationMessages }
235
+ };
236
+ }
237
+ createDefaultSettingsWrapper() {
238
+ return {
239
+ safeSettings: this.createDefaultSettings(),
240
+ settings: this.createDefaultSettings(),
241
+ validationMessages: {}
242
+ };
243
+ }
244
+ async edit(settingsEditor) {
245
+ try {
246
+ await settingsEditor(this.currentSettingsWrapper.settings);
247
+ } finally {
248
+ const validationResult = await this.validate(this.currentSettingsWrapper.settings);
249
+ for (const propertyName of this.propertyNames) {
250
+ const validationMessage = validationResult[propertyName] ?? "";
251
+ this.currentSettingsWrapper.validationMessages[propertyName] = validationMessage;
252
+ this.currentSettingsWrapper.safeSettings[propertyName] = validationMessage ? this.defaultSettings[propertyName] : this.currentSettingsWrapper.settings[propertyName];
253
+ }
289
254
  }
290
- await this.onSavingRecord(settings);
291
- return this.getTransformer().transformObjectRecursively(settings);
292
255
  }
293
- async saveToFileImpl() {
294
- await this.plugin.saveData(await this.prepareRecordToSave());
295
- }
296
- }
297
- class PluginSettingsProperty {
298
- /**
299
- * Creates a new plugin settings property.
300
- *
301
- * @param propertyName - The name of the property.
302
- * @param currentSettings - The current settings.
303
- * @param validator - The validator of the property.
304
- */
305
- constructor(propertyName, currentSettings, validator) {
306
- this.propertyName = propertyName;
307
- this.currentSettings = currentSettings;
308
- this.validator = validator;
309
- const record = currentSettings;
310
- this.defaultValue = record[propertyName];
311
- this._lastSavedValue = this.defaultValue;
312
- this._currentValue = this.defaultValue;
313
- }
314
- defaultValue;
315
- /**
316
- * The current value of the property.
317
- *
318
- * @returns The current value.
319
- */
320
- get currentValue() {
321
- return this._currentValue;
322
- }
323
- /**
324
- * The last saved value of the property.
325
- *
326
- * @returns The last saved value.
327
- */
328
- get lastSavedValue() {
329
- return this._lastSavedValue;
330
- }
331
- /**
332
- * The safe value of the property.
333
- *
334
- * @returns The safe value.
335
- */
336
- get safeValue() {
337
- return this._validationMessage ? this.defaultValue : this._currentValue;
338
- }
339
- /**
340
- * The validation message of the property.
341
- *
342
- * @returns The validation message.
343
- */
344
- get validationMessage() {
345
- return this._validationMessage;
346
- }
347
- _currentValue;
348
- _lastSavedValue;
349
- _validationMessage = "";
350
- get currentSettingsRecord() {
351
- return this.currentSettings;
352
- }
353
- /**
354
- * Resets the current value of the property to the default value.
355
- */
356
- reset() {
357
- this._currentValue = this.defaultValue;
358
- this._validationMessage = "";
359
- }
360
- /**
361
- * Saves the current value of the property.
362
- *
363
- * @returns `true` if the value was changed, `false` otherwise.
364
- */
365
- save() {
366
- if (this._lastSavedValue === this._currentValue) {
256
+ isValidPropertyName(prop) {
257
+ if (typeof prop !== "string") {
367
258
  return false;
368
259
  }
369
- this._lastSavedValue = this._currentValue;
370
- return true;
260
+ return this.propertyNames.includes(prop);
371
261
  }
372
- /**
373
- * Sets the validation message of the property.
374
- *
375
- * @param validationMessage - The validation message.
376
- */
377
- setValidationMessage(validationMessage) {
378
- this._validationMessage = validationMessage;
379
- this.showWarning();
262
+ async rawRecordToSettings(rawRecord) {
263
+ await this.onLoadRecord(rawRecord);
264
+ const settings = this.createDefaultSettings();
265
+ for (const [propertyName, value] of Object.entries(rawRecord)) {
266
+ if (!this.isValidPropertyName(propertyName)) {
267
+ console.warn(`Unknown property: ${propertyName}`);
268
+ continue;
269
+ }
270
+ if (typeof value !== typeof this.defaultSettings[propertyName]) {
271
+ console.warn(
272
+ "Possible invalid value type. It might lead to an unexpected behavior of the plugin. There is also a chance it is a false-negative warning, as we are unable to determine the exact type of the value in runtime.",
273
+ {
274
+ defaultValue: this.defaultSettings[propertyName],
275
+ propertyName,
276
+ value
277
+ }
278
+ );
279
+ }
280
+ settings[propertyName] = value;
281
+ }
282
+ return settings;
380
283
  }
381
- /**
382
- * Sets the current value of the property.
383
- *
384
- * @param value - The value to set.
385
- */
386
- setValue(value) {
387
- this._currentValue = value;
388
- this.currentSettingsRecord[this.propertyName] = value;
284
+ async saveToFileImpl() {
285
+ await this.plugin.saveData(await this.settingsToRawRecord(this.currentSettingsWrapper));
389
286
  }
390
- /**
391
- * Validates the current value of the property.
392
- *
393
- * @param settings - The settings to validate. If not provided, the current settings will be used.
394
- * @returns A {@link Promise} that resolves to the validation message.
395
- */
396
- async validate(settings) {
397
- const isCurrent = settings === void 0;
398
- settings ??= this.currentSettings;
399
- const value = isCurrent ? this._currentValue : settings[this.propertyName];
400
- let validationMessage;
401
- try {
402
- validationMessage = await this.validator(value, settings) ?? "";
403
- } catch (error) {
404
- console.error("Validation failed", {
405
- propertyName: this.propertyName,
406
- value
407
- }, error);
408
- validationMessage = "Validation failed";
409
- }
410
- if (isCurrent) {
411
- this._validationMessage = validationMessage;
412
- this.showWarning(this._currentValue);
413
- }
414
- return validationMessage;
287
+ setPropertyImpl(propertyName, value, validationMessage) {
288
+ this.currentSettingsWrapper.settings[propertyName] = value;
289
+ this.currentSettingsWrapper.validationMessages[propertyName] = validationMessage ?? "";
290
+ this.currentSettingsWrapper.safeSettings[propertyName] = validationMessage ? this.defaultSettings[propertyName] : value;
415
291
  }
416
- showWarning(value) {
417
- if (!this._validationMessage) {
418
- return;
292
+ async settingsToRawRecord(settings) {
293
+ const rawRecord = {};
294
+ for (const propertyName of this.propertyNames) {
295
+ rawRecord[propertyName] = settings[propertyName];
419
296
  }
420
- console.warn(`Could not set plugin setting: ${this.propertyName}. Using default value instead.`, {
421
- defaultValue: this.defaultValue,
422
- propertyName: this.propertyName,
423
- validationMessage: this._validationMessage,
424
- value
425
- });
297
+ await this.onSavingRecord(rawRecord);
298
+ return this.getTransformer().transformObjectRecursively(rawRecord);
426
299
  }
427
300
  }
428
301
  // Annotate the CommonJS export names for ESM import in node:
429
302
  0 && (module.exports = {
430
- PluginSettingsManagerBase,
431
- PluginSettingsProperty
303
+ PluginSettingsManagerBase
432
304
  });
433
- //# 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 { GenericObject } from '../../Object.ts';\nimport type { Transformer } from '../../Transformers/Transformer.ts';\nimport type {\n  MaybeReturn,\n  PropertyValues,\n  StringKeys\n} from '../../Type.ts';\nimport type {\n  ExtractPlugin,\n  ExtractPluginSettings,\n  ExtractPluginSettingsPropertyNames,\n  ExtractPluginSettingsPropertyValues,\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<PluginSettings extends object, PropertyName extends StringKeys<PluginSettings> = StringKeys<PluginSettings>> = (\n  value: PluginSettings[PropertyName],\n  settings: PluginSettings\n) => 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<PluginSettings>): 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 as PropertyValues<PluginSettings>);\n    this.validationPromise = this.validationPromise.then(async () => {\n      await property.validate();\n    });\n    return true;\n  }\n\n  protected override getPropertyValue(property: PluginSettingsProperty<PluginSettings>): unknown {\n    return property.currentValue;\n  }\n}\n\nclass PropertiesMap<PluginSettings extends object> extends Map<string, PluginSettingsProperty<PluginSettings, StringKeys<PluginSettings>>> {\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 unknown 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 as unknown as PluginSettingsProperty<PluginSettings, StringKeys<PluginSettings>>);\n  }\n}\n\nclass SafeSettingsProxyHandler<PluginSettings extends object> extends ProxyHandlerBase<PluginSettings> {\n  protected override getPropertyValue(property: PluginSettingsProperty<PluginSettings>): 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  private readonly validators: Map<string, Validator<ExtractPluginSettings<PluginTypes>>> = new Map<\n    string,\n    Validator<ExtractPluginSettings<PluginTypes>>\n  >();\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    for (const propertyName of getAllKeys(this.currentSettings)) {\n      this.properties.set(\n        propertyName,\n        new PluginSettingsProperty(propertyName, this.currentSettings, this.validators.get(propertyName) ?? noop)\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   * Ensures the settings are safe.\n   *\n   * It runs validation for each property and sets the default value if the validation fails.\n   *\n   * @param settings - The settings.\n   * @returns A {@link Promise} that resolves when the settings are safe.\n   */\n  public async ensureSafe(settings: ExtractPluginSettings<PluginTypes>): Promise<void> {\n    const record = settings as GenericObject;\n    for (const propertyName of getAllKeys(settings)) {\n      const property = this.getProperty(propertyName);\n      const validationMessage = await property.validate(settings);\n      if (validationMessage) {\n        record[propertyName] = property.defaultValue;\n      }\n    }\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 ExtractPluginSettingsPropertyNames<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    type PluginSettings = ExtractPluginSettings<PluginTypes>;\n\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 GenericObject;\n    const originalJson = JSON.stringify(record);\n    record = this.getTransformer().transformObjectRecursively(record);\n    await this.onLoadRecord(record);\n\n    const propertiesToSave: PluginSettingsProperty<PluginSettings>[] = [];\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,\n            propertyName,\n            value\n          }\n        );\n      }\n\n      property.setValue(value as PropertyValues<PluginSettings>);\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: GenericObject): 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: GenericObject): 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 ExtractPluginSettingsPropertyNames<PluginTypes>>(\n    propertyName: PropertyName,\n    validator: Validator<ExtractPluginSettings<PluginTypes>, PropertyName>\n  ): void {\n    this.validators.set(propertyName, validator as Validator<ExtractPluginSettings<PluginTypes>>);\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: GenericObject = {};\n    for (const [propertyName, property] of this.properties.entries()) {\n      savedSettings[propertyName] = property.lastSavedValue as\n        | ExtractPluginSettingsPropertyValues<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<GenericObject> {\n    const settings: GenericObject = {};\n    for (const [propertyName, property] of this.properties.entries()) {\n      settings[propertyName] = property.currentValue;\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<PluginSettings extends object, PropertyName extends StringKeys<PluginSettings> = StringKeys<PluginSettings>> {\n  public readonly defaultValue: typeof this.PropertyType;\n\n  /**\n   * The current value of the property.\n   *\n   * @returns The current value.\n   */\n  public get currentValue(): typeof this.PropertyType {\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(): typeof this.PropertyType {\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(): typeof this.PropertyType {\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: typeof this.PropertyType;\n\n  private _lastSavedValue: typeof this.PropertyType;\n\n  private _validationMessage = '';\n\n  declare private PropertyType: PluginSettings[PropertyName];\n\n  private get currentSettingsRecord(): GenericObject {\n    return this.currentSettings as GenericObject;\n  }\n\n  /**\n   * Creates a new plugin settings property.\n   *\n   * @param propertyName - The name of the property.\n   * @param currentSettings - The current settings.\n   * @param validator - The validator of the property.\n   */\n  public constructor(\n    private readonly propertyName: StringKeys<PluginSettings>,\n    private readonly currentSettings: PluginSettings,\n    private readonly validator: Validator<PluginSettings, PropertyName>\n  ) {\n    const record = currentSettings as GenericObject;\n    this.defaultValue = record[propertyName] as typeof this.PropertyType;\n    this._lastSavedValue = this.defaultValue;\n    this._currentValue = this.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: typeof this.PropertyType): void {\n    this._currentValue = value;\n    this.currentSettingsRecord[this.propertyName] = value;\n  }\n\n  /**\n   * Validates the current value of the property.\n   *\n   * @param settings - The settings to validate. If not provided, the current settings will be used.\n   * @returns A {@link Promise} that resolves to the validation message.\n   */\n  public async validate(settings?: PluginSettings): Promise<string> {\n    const isCurrent = settings === undefined;\n    settings ??= this.currentSettings;\n    const value = isCurrent ? this._currentValue : settings[this.propertyName] as typeof this.PropertyType;\n\n    let validationMessage: string;\n    try {\n      validationMessage = (await this.validator(value, settings) as string | undefined) ?? '';\n    } catch (error) {\n      console.error('Validation failed', {\n        propertyName: this.propertyName,\n        value\n      }, error);\n      validationMessage = 'Validation failed';\n    }\n\n    if (isCurrent) {\n      this._validationMessage = validationMessage;\n      this.showWarning(this._currentValue);\n    }\n\n    return validationMessage;\n  }\n\n  private showWarning(value?: typeof this.PropertyType): 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": []
}

305
+ //# 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 { GenericObject } from '../../Object.ts';\nimport type { Transformer } from '../../Transformers/Transformer.ts';\nimport type {\n  MaybeReturn,\n  StringKeys\n} from '../../Type.ts';\nimport type { PluginSettingsWrapper } from './PluginSettingsWrapper.ts';\nimport type {\n  ExtractPlugin,\n  ExtractPluginSettings,\n  ExtractPluginSettingsPropertyNames,\n  ExtractPluginSettingsPropertyValues,\n  ExtractPluginSettingsWrapper,\n  ExtractReadonlyPluginSettingsWrapper,\n  PluginTypesBase\n} from './PluginTypesBase.ts';\n\nimport {\n  noop,\n  noopAsync\n} from '../../Function.ts';\nimport {\n  deepEqual,\n  getAllKeys\n} 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 ValidationResult<PluginSettings extends object> = Partial<Record<StringKeys<PluginSettings>, string>>;\n\ntype Validator<PluginSettings extends object, PropertyName extends StringKeys<PluginSettings> = StringKeys<PluginSettings>> = (\n  value: PluginSettings[PropertyName],\n  settings: PluginSettings\n) => Promisable<MaybeReturn<string>>;\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\n  public readonly defaultSettings: ReadonlyDeep<ExtractPluginSettings<PluginTypes>>;\n\n  /**\n   * Gets the current settings wrapper.\n   *\n   * @returns The current settings wrapper.\n   */\n  public get settingsWrapper(): ExtractReadonlyPluginSettingsWrapper<PluginTypes> {\n    return this.currentSettingsWrapper as ExtractReadonlyPluginSettingsWrapper<PluginTypes>;\n  }\n\n  private currentSettingsWrapper: ExtractPluginSettingsWrapper<PluginTypes>;\n\n  private lastSavedSettingsWrapper: ExtractPluginSettingsWrapper<PluginTypes>;\n  private readonly propertyNames: ExtractPluginSettingsPropertyNames<PluginTypes>[];\n  private readonly validators = new Map<ExtractPluginSettingsPropertyNames<PluginTypes>, Validator<ExtractPluginSettings<PluginTypes>>>();\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.defaultSettings = this.createDefaultSettings() as ReadonlyDeep<ExtractPluginSettings<PluginTypes>>;\n    this.currentSettingsWrapper = this.createDefaultSettingsWrapper();\n    this.lastSavedSettingsWrapper = this.createDefaultSettingsWrapper();\n    this.propertyNames = getAllKeys(this.currentSettingsWrapper.settings);\n    this.registerValidators();\n  }\n\n  /**\n   * Edits the plugin settings and saves them.\n   *\n   * @param settingsEditor - The editor.\n   * @param context - The context.\n   * @returns A {@link Promise} that resolves when the settings are saved.\n   */\n  public async editAndSave(settingsEditor: (settings: ExtractPluginSettings<PluginTypes>) => Promisable<void>, context?: unknown): Promise<void> {\n    await this.edit(settingsEditor);\n    await this.saveToFile(context);\n  }\n\n  /**\n   * Ensures the settings are safe.\n   *\n   * It runs validation for each property and sets the default value if the validation fails.\n   *\n   * @param settings - The settings.\n   * @returns A {@link Promise} that resolves when the settings are safe.\n   */\n  public async ensureSafe(settings: ExtractPluginSettings<PluginTypes>): Promise<void> {\n    const validationResult = await this.validate(settings);\n    for (const propertyName of this.propertyNames) {\n      if (validationResult[propertyName]) {\n        settings[propertyName] = this.defaultSettings[propertyName];\n      }\n    }\n  }\n\n  /**\n   * Gets a safe copy of the settings.\n   *\n   * @param settings - The settings.\n   * @returns A {@link Promise} that resolves to the safe copy of the settings.\n   */\n  public async getSafeCopy(settings: ExtractPluginSettings<PluginTypes>): Promise<ExtractPluginSettings<PluginTypes>> {\n    const safeSettings = this.cloneSettings(settings);\n    await this.ensureSafe(safeSettings);\n    return safeSettings;\n  }\n\n  /**\n   * Loads the plugin settings from the file.\n   *\n   * @param isInitialLoad - Whether the settings are being loaded for the first time.\n   * @returns A {@link Promise} that resolves when the settings are loaded.\n   */\n  public async loadFromFile(isInitialLoad: boolean): Promise<void> {\n    const data = await this.plugin.loadData() as unknown;\n    this.lastSavedSettingsWrapper = this.createDefaultSettingsWrapper();\n    this.currentSettingsWrapper = this.createDefaultSettingsWrapper();\n\n    if (data === undefined || data === null) {\n      return;\n    }\n\n    if (typeof data !== 'object') {\n      console.error(`Invalid settings from data.json. Expected Object, got: ${typeof data}`);\n      return;\n    }\n\n    const rawRecord = data as GenericObject;\n    const parsedSettings = await this.rawRecordToSettings(rawRecord);\n    const validationResult = await this.validate(parsedSettings);\n\n    for (const propertyName of this.propertyNames) {\n      this.setPropertyImpl(propertyName, parsedSettings[propertyName], validationResult[propertyName]);\n    }\n\n    this.lastSavedSettingsWrapper = this.cloneSettingsWrapper(this.currentSettingsWrapper);\n\n    const newRecord = await this.settingsToRawRecord(this.currentSettingsWrapper.settings);\n\n    if (!deepEqual(newRecord, data)) {\n      await this.saveToFileImpl();\n    }\n\n    await this.plugin.onLoadSettings(this.currentSettingsWrapper, 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    if (deepEqual(this.lastSavedSettingsWrapper.settings, this.currentSettingsWrapper.settings)) {\n      return;\n    }\n\n    await this.saveToFileImpl();\n    await this.plugin.onSaveSettings(this.currentSettingsWrapper, this.lastSavedSettingsWrapper, context);\n    this.lastSavedSettingsWrapper = this.cloneSettingsWrapper(this.currentSettingsWrapper);\n  }\n\n  /**\n   * Sets the value of a property.\n   *\n   * @typeParam PropertyName - The name of the property.\n   * @param propertyName - The name of the property.\n   * @param value - The value to set.\n   * @returns A {@link Promise} that resolves to the validation message.\n   */\n  public async setProperty<PropertyName extends ExtractPluginSettingsPropertyNames<PluginTypes>>(\n    propertyName: PropertyName,\n    value: ExtractPluginSettings<PluginTypes>[PropertyName]\n  ): Promise<string> {\n    await this.edit((settings) => {\n      settings[propertyName] = value;\n    });\n    return this.currentSettingsWrapper.validationMessages[propertyName];\n  }\n\n  /**\n   * Validates the settings.\n   *\n   * @param settings - The settings.\n   * @returns A {@link Promise} that resolves to the validation result.\n   */\n  public async validate(settings: ExtractPluginSettings<PluginTypes>): Promise<ValidationResult<ExtractPluginSettings<PluginTypes>>> {\n    const result: ValidationResult<ExtractPluginSettings<PluginTypes>> = {};\n    for (const [propertyName, validator] of this.validators.entries()) {\n      const validationMessage = await validator(settings[propertyName], settings);\n      if (validationMessage) {\n        result[propertyName] = validationMessage;\n      }\n    }\n\n    return result;\n  }\n\n  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: GenericObject): 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: GenericObject): 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 ExtractPluginSettingsPropertyNames<PluginTypes>>(\n    propertyName: PropertyName,\n    validator: Validator<ExtractPluginSettings<PluginTypes>, PropertyName>\n  ): void {\n    this.validators.set(propertyName, validator as Validator<ExtractPluginSettings<PluginTypes>>);\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 cloneSettings(settings: ExtractPluginSettings<PluginTypes>): ExtractPluginSettings<PluginTypes> {\n    const record = this.settingsToRawRecord(settings);\n    const json = JSON.stringify(record);\n    const cloneRecord = JSON.parse(json) as GenericObject;\n    return this.rawRecordToSettings(cloneRecord);\n  }\n\n  private cloneSettingsWrapper(\n    settingsWrapper: PluginSettingsWrapper<ExtractPluginSettings<PluginTypes>>\n  ): PluginSettingsWrapper<ExtractPluginSettings<PluginTypes>> {\n    return {\n      safeSettings: this.cloneSettings(settingsWrapper.safeSettings),\n      settings: this.cloneSettings(settingsWrapper.settings),\n      validationMessages: { ...settingsWrapper.validationMessages }\n    };\n  }\n\n  private createDefaultSettingsWrapper(): PluginSettingsWrapper<ExtractPluginSettings<PluginTypes>> {\n    return {\n      safeSettings: this.createDefaultSettings(),\n      settings: this.createDefaultSettings(),\n      validationMessages: {} as Record<ExtractPluginSettingsPropertyNames<PluginTypes>, string>\n    };\n  }\n\n  private async edit(settingsEditor: (settings: ExtractPluginSettings<PluginTypes>) => Promisable<void>): Promise<void> {\n    try {\n      await settingsEditor(this.currentSettingsWrapper.settings);\n    } finally {\n      const validationResult = await this.validate(this.currentSettingsWrapper.settings);\n      for (const propertyName of this.propertyNames) {\n        const validationMessage = validationResult[propertyName] ?? '';\n        this.currentSettingsWrapper.validationMessages[propertyName] = validationMessage;\n        this.currentSettingsWrapper.safeSettings[propertyName] = validationMessage\n          ? this.defaultSettings[propertyName]\n          : this.currentSettingsWrapper.settings[propertyName];\n      }\n    }\n  }\n\n  private isValidPropertyName(prop: unknown): prop is ExtractPluginSettingsPropertyNames<PluginTypes> {\n    if (typeof prop !== 'string') {\n      return false;\n    }\n\n    return (this.propertyNames as string[]).includes(prop);\n  }\n\n  private async rawRecordToSettings(rawRecord: GenericObject): Promise<ExtractPluginSettings<PluginTypes>> {\n    await this.onLoadRecord(rawRecord);\n\n    const settings = this.createDefaultSettings();\n\n    for (const [propertyName, value] of Object.entries(rawRecord)) {\n      if (!this.isValidPropertyName(propertyName)) {\n        console.warn(`Unknown property: ${propertyName}`);\n        continue;\n      }\n\n      if (typeof value !== typeof this.defaultSettings[propertyName]) {\n        console.warn(\n          'Possible invalid value type. It might lead to an unexpected behavior of the plugin. There is also a chance it is a false-negative warning, as we are unable to determine the exact type of the value in runtime.',\n          {\n            defaultValue: this.defaultSettings[propertyName],\n            propertyName,\n            value\n          }\n        );\n      }\n\n      settings[propertyName] = value as ExtractPluginSettingsPropertyValues<PluginTypes>;\n    }\n\n    return settings;\n  }\n\n  private async saveToFileImpl(): Promise<void> {\n    await this.plugin.saveData(await this.settingsToRawRecord(this.currentSettingsWrapper));\n  }\n\n  private setPropertyImpl(\n    propertyName: ExtractPluginSettingsPropertyNames<PluginTypes>,\n    value: ExtractPluginSettingsPropertyValues<PluginTypes>,\n    validationMessage?: string\n  ): void {\n    this.currentSettingsWrapper.settings[propertyName] = value;\n    this.currentSettingsWrapper.validationMessages[propertyName] = validationMessage ?? '';\n    this.currentSettingsWrapper.safeSettings[propertyName] = validationMessage ? this.defaultSettings[propertyName] : value;\n  }\n\n  private async settingsToRawRecord(settings: ExtractPluginSettings<PluginTypes>): Promise<GenericObject> {\n    const rawRecord: GenericObject = {};\n\n    for (const propertyName of this.propertyNames) {\n      rawRecord[propertyName] = settings[propertyName];\n    }\n\n    await this.onSavingRecord(rawRecord);\n\n    return this.getTransformer().transformObjectRecursively(rawRecord);\n  }\n}\n"],
  "mappings": "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",
  "names": []
}
