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

287
+ //# 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": ";;;;;;;AA6BA;AAAA,EACE;AAAA,EACA;AAAA,OACK;AACP;AAAA,EACE;AAAA,EACA;AAAA,OACK;AACP,SAAS,uBAAuB;AAChC,SAAS,2BAA2B;AACpC,SAAS,wBAAwB;AACjC,SAAS,sCAAsC;AAE/C,MAAM,qBAAqB,IAAI,iBAAiB;AAAA,EAC9C,IAAI,+BAA+B;AAAA,EACnC,IAAI,gBAAgB;AAAA,EACpB,IAAI,oBAAoB;AAC1B,CAAC;AAcM,MAAe,0BAA+D;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAwB5E,YAA4B,QAAoC;AAApC;AACjC,SAAK,MAAM,OAAO;AAClB,SAAK,kBAAkB,KAAK,sBAAsB;AAClD,SAAK,yBAAyB,KAAK,6BAA6B;AAChE,SAAK,2BAA2B,KAAK,6BAA6B;AAClE,SAAK,gBAAgB,WAAW,KAAK,uBAAuB,QAAQ;AACpE,SAAK,mBAAmB;AAAA,EAC1B;AAAA,EA9BgB;AAAA,EAEA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAOhB,IAAW,kBAAqE;AAC9E,WAAO,KAAK;AAAA,EACd;AAAA,EAEQ;AAAA,EAEA;AAAA,EACS;AAAA,EACA,aAAa,oBAAI,IAAoG;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAsBtI,MAAa,YAAY,gBAAoF,SAAkC;AAC7I,UAAM,KAAK,KAAK,cAAc;AAC9B,UAAM,KAAK,WAAW,OAAO;AAAA,EAC/B;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAUA,MAAa,WAAW,UAA6D;AACnF,UAAM,mBAAmB,MAAM,KAAK,SAAS,QAAQ;AACrD,eAAW,gBAAgB,KAAK,eAAe;AAC7C,UAAI,iBAAiB,YAAY,GAAG;AAClC,iBAAS,YAAY,IAAI,KAAK,gBAAgB,YAAY;AAAA,MAC5D;AAAA,IACF;AAAA,EACF;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAQA,MAAa,YAAY,UAA2F;AAClH,UAAM,eAAe,KAAK,cAAc,QAAQ;AAChD,UAAM,KAAK,WAAW,YAAY;AAClC,WAAO;AAAA,EACT;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAQA,MAAa,aAAa,eAAuC;AAC/D,UAAM,OAAO,MAAM,KAAK,OAAO,SAAS;AACxC,SAAK,2BAA2B,KAAK,6BAA6B;AAClE,SAAK,yBAAyB,KAAK,6BAA6B;AAEhE,QAAI,SAAS,UAAa,SAAS,MAAM;AACvC;AAAA,IACF;AAEA,QAAI,OAAO,SAAS,UAAU;AAC5B,cAAQ,MAAM,0DAA0D,OAAO,IAAI,EAAE;AACrF;AAAA,IACF;AAEA,UAAM,YAAY;AAClB,UAAM,iBAAiB,MAAM,KAAK,oBAAoB,SAAS;AAC/D,UAAM,mBAAmB,MAAM,KAAK,SAAS,cAAc;AAE3D,eAAW,gBAAgB,KAAK,eAAe;AAC7C,WAAK,gBAAgB,cAAc,eAAe,YAAY,GAAG,iBAAiB,YAAY,CAAC;AAAA,IACjG;AAEA,SAAK,2BAA2B,KAAK,qBAAqB,KAAK,sBAAsB;AAErF,UAAM,YAAY,MAAM,KAAK,oBAAoB,KAAK,uBAAuB,QAAQ;AAErF,QAAI,CAAC,UAAU,WAAW,IAAI,GAAG;AAC/B,YAAM,KAAK,eAAe;AAAA,IAC5B;AAEA,UAAM,KAAK,OAAO,eAAe,KAAK,wBAAwB,aAAa;AAAA,EAC7E;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAQA,MAAa,WAAW,SAAkC;AACxD,QAAI,UAAU,KAAK,yBAAyB,UAAU,KAAK,uBAAuB,QAAQ,GAAG;AAC3F;AAAA,IACF;AAEA,UAAM,KAAK,eAAe;AAC1B,UAAM,KAAK,OAAO,eAAe,KAAK,wBAAwB,KAAK,0BAA0B,OAAO;AACpG,SAAK,2BAA2B,KAAK,qBAAqB,KAAK,sBAAsB;AAAA,EACvF;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAUA,MAAa,YACX,cACA,OACiB;AACjB,UAAM,KAAK,KAAK,CAAC,aAAa;AAC5B,eAAS,YAAY,IAAI;AAAA,IAC3B,CAAC;AACD,WAAO,KAAK,uBAAuB,mBAAmB,YAAY;AAAA,EACpE;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAQA,MAAa,SAAS,UAA6G;AACjI,UAAM,SAA+D,CAAC;AACtE,eAAW,CAAC,cAAc,SAAS,KAAK,KAAK,WAAW,QAAQ,GAAG;AACjE,YAAM,oBAAoB,MAAM,UAAU,SAAS,YAAY,GAAG,QAAQ;AAC1E,UAAI,mBAAmB;AACrB,eAAO,YAAY,IAAI;AAAA,MACzB;AAAA,IACF;AAEA,WAAO;AAAA,EACT;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EASU,iBAA8B;AACtC,WAAO;AAAA,EACT;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAOA,MAAgB,aAAa,SAAuC;AAClE,UAAM,UAAU;AAAA,EAClB;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAOA,MAAgB,eAAe,SAAuC;AACpE,UAAM,UAAU;AAAA,EAClB;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAQU,kBACR,cACA,WACM;AACN,SAAK,WAAW,IAAI,cAAc,SAA0D;AAAA,EAC9F;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAOU,qBAA2B;AACnC,SAAK;AAAA,EACP;AAAA,EAEQ,cAAc,UAAkF;AACtG,UAAM,SAAS,KAAK,oBAAoB,QAAQ;AAChD,UAAM,OAAO,KAAK,UAAU,MAAM;AAClC,UAAM,cAAc,KAAK,MAAM,IAAI;AACnC,WAAO,KAAK,oBAAoB,WAAW;AAAA,EAC7C;AAAA,EAEQ,qBACN,iBAC2D;AAC3D,WAAO;AAAA,MACL,cAAc,KAAK,cAAc,gBAAgB,YAAY;AAAA,MAC7D,UAAU,KAAK,cAAc,gBAAgB,QAAQ;AAAA,MACrD,oBAAoB,EAAE,GAAG,gBAAgB,mBAAmB;AAAA,IAC9D;AAAA,EACF;AAAA,EAEQ,+BAA0F;AAChG,WAAO;AAAA,MACL,cAAc,KAAK,sBAAsB;AAAA,MACzC,UAAU,KAAK,sBAAsB;AAAA,MACrC,oBAAoB,CAAC;AAAA,IACvB;AAAA,EACF;AAAA,EAEA,MAAc,KAAK,gBAAmG;AACpH,QAAI;AACF,YAAM,eAAe,KAAK,uBAAuB,QAAQ;AAAA,IAC3D,UAAE;AACA,YAAM,mBAAmB,MAAM,KAAK,SAAS,KAAK,uBAAuB,QAAQ;AACjF,iBAAW,gBAAgB,KAAK,eAAe;AAC7C,cAAM,oBAAoB,iBAAiB,YAAY,KAAK;AAC5D,aAAK,uBAAuB,mBAAmB,YAAY,IAAI;AAC/D,aAAK,uBAAuB,aAAa,YAAY,IAAI,oBACrD,KAAK,gBAAgB,YAAY,IACjC,KAAK,uBAAuB,SAAS,YAAY;AAAA,MACvD;AAAA,IACF;AAAA,EACF;AAAA,EAEQ,oBAAoB,MAAwE;AAClG,QAAI,OAAO,SAAS,UAAU;AAC5B,aAAO;AAAA,IACT;AAEA,WAAQ,KAAK,cAA2B,SAAS,IAAI;AAAA,EACvD;AAAA,EAEA,MAAc,oBAAoB,WAAuE;AACvG,UAAM,KAAK,aAAa,SAAS;AAEjC,UAAM,WAAW,KAAK,sBAAsB;AAE5C,eAAW,CAAC,cAAc,KAAK,KAAK,OAAO,QAAQ,SAAS,GAAG;AAC7D,UAAI,CAAC,KAAK,oBAAoB,YAAY,GAAG;AAC3C,gBAAQ,KAAK,qBAAqB,YAAY,EAAE;AAChD;AAAA,MACF;AAEA,UAAI,OAAO,UAAU,OAAO,KAAK,gBAAgB,YAAY,GAAG;AAC9D,gBAAQ;AAAA,UACN;AAAA,UACA;AAAA,YACE,cAAc,KAAK,gBAAgB,YAAY;AAAA,YAC/C;AAAA,YACA;AAAA,UACF;AAAA,QACF;AAAA,MACF;AAEA,eAAS,YAAY,IAAI;AAAA,IAC3B;AAEA,WAAO;AAAA,EACT;AAAA,EAEA,MAAc,iBAAgC;AAC5C,UAAM,KAAK,OAAO,SAAS,MAAM,KAAK,oBAAoB,KAAK,sBAAsB,CAAC;AAAA,EACxF;AAAA,EAEQ,gBACN,cACA,OACA,mBACM;AACN,SAAK,uBAAuB,SAAS,YAAY,IAAI;AACrD,SAAK,uBAAuB,mBAAmB,YAAY,IAAI,qBAAqB;AACpF,SAAK,uBAAuB,aAAa,YAAY,IAAI,oBAAoB,KAAK,gBAAgB,YAAY,IAAI;AAAA,EACpH;AAAA,EAEA,MAAc,oBAAoB,UAAsE;AACtG,UAAM,YAA2B,CAAC;AAElC,eAAW,gBAAgB,KAAK,eAAe;AAC7C,gBAAU,YAAY,IAAI,SAAS,YAAY;AAAA,IACjD;AAEA,UAAM,KAAK,eAAe,SAAS;AAEnC,WAAO,KAAK,eAAe,EAAE,2BAA2B,SAAS;AAAA,EACnE;AACF;",
  "names": []
}

@@ -69,6 +69,7 @@ export declare abstract class PluginSettingsTabBase<PluginTypes extends PluginTy
69
69
  protected get saveSettingsDebounceTimeoutInMilliseconds(): number;
70
70
  private _isOpen;
71
71
  private saveSettingsDebounced;
72
+ private get pluginSettings();
72
73
  /**
73
74
  * Creates a new plugin settings tab.
74
75
  *