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

|
406
|
+
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../../../src/obsidian/Plugin/PluginSettingsManagerBase.ts"],
  "sourcesContent": ["/**\n * @packageDocumentation\n *\n * Plugin settings manager base class.\n */\n\nimport type { App } from 'obsidian';\nimport type {\n  Promisable,\n  ReadonlyDeep\n} from 'type-fest';\n\nimport type { Transformer } from '../../Transformers/Transformer.ts';\nimport type {\n  MaybeReturn,\n  StringKeys\n} from '../../Type.ts';\nimport type {\n  ExtractPlugin,\n  ExtractPluginSettings,\n  PluginTypesBase\n} from './PluginTypesBase.ts';\n\nimport {\n  noop,\n  noopAsync\n} from '../../Function.ts';\nimport { getAllKeys } from '../../Object.ts';\nimport { DateTransformer } from '../../Transformers/DateTransformer.ts';\nimport { DurationTransformer } from '../../Transformers/DurationTransformer.ts';\nimport { GroupTransformer } from '../../Transformers/GroupTransformer.ts';\nimport { SkipPrivatePropertyTransformer } from '../../Transformers/SkipPrivatePropertyTransformer.ts';\n\nconst defaultTransformer = new GroupTransformer([\n  new SkipPrivatePropertyTransformer(),\n  new DateTransformer(),\n  new DurationTransformer()\n]);\n\ntype PropertySetter = (value: unknown) => void;\n\ntype Validator<T> = (value: T) => Promisable<MaybeReturn<string>>;\n\nabstract class ProxyHandlerBase<PluginSettings extends object> implements ProxyHandler<PluginSettings> {\n  public constructor(protected readonly properties: PropertiesMap<PluginSettings>) {}\n\n  public get(target: PluginSettings, prop: string | symbol): unknown {\n    const record = target as Record<string | symbol, unknown>;\n    if (typeof prop !== 'string') {\n      return record[prop];\n    }\n\n    const property = this.properties.get(prop);\n    if (!property) {\n      return record[prop];\n    }\n\n    return this.getPropertyValue(property);\n  }\n\n  protected abstract getPropertyValue(property: PluginSettingsProperty<unknown>): unknown;\n}\n\nclass EditableSettingsProxyHandler<PluginSettings extends object> extends ProxyHandlerBase<PluginSettings> {\n  private validationPromise = Promise.resolve();\n  public set(target: PluginSettings, prop: string | symbol, value: unknown): boolean {\n    const record = target as Record<string | symbol, unknown>;\n    record[prop] = value;\n\n    if (typeof prop !== 'string') {\n      return true;\n    }\n\n    const property = this.properties.get(prop);\n    if (!property) {\n      return true;\n    }\n\n    property.setValue(value);\n    this.validationPromise = this.validationPromise.then(() => property.validate());\n    return true;\n  }\n\n  protected override getPropertyValue(property: PluginSettingsProperty<unknown>): unknown {\n    return property.currentValue;\n  }\n}\n\n// eslint-disable-next-line @typescript-eslint/no-explicit-any\nclass PropertiesMap<PluginSettings extends object> extends Map<string, PluginSettingsProperty<any>> {\n  public getTyped<PropertyName extends StringKeys<PluginSettings>>(propertyName: PropertyName): PluginSettingsProperty<PluginSettings[PropertyName]> {\n    const property = super.get(propertyName);\n    if (!property) {\n      throw new Error(`Property ${String(propertyName)} not found`);\n    }\n\n    return property as PluginSettingsProperty<PluginSettings[PropertyName]>;\n  }\n\n  public setTyped<PropertyName extends StringKeys<PluginSettings>>(\n    propertyName: PropertyName,\n    value: PluginSettingsProperty<PluginSettings[PropertyName]>\n  ): this {\n    return super.set(propertyName, value);\n  }\n}\n\nclass SafeSettingsProxyHandler<PluginSettings extends object> extends ProxyHandlerBase<PluginSettings> {\n  protected override getPropertyValue(property: PluginSettingsProperty<unknown>): unknown {\n    return property.safeValue;\n  }\n}\n\n/**\n * Base class for managing plugin settings.\n *\n * @typeParam PluginTypes - Plugin-specific types.\n */\nexport abstract class PluginSettingsManagerBase<PluginTypes extends PluginTypesBase> {\n  public readonly app: App;\n  public readonly safeSettings: ReadonlyDeep<ExtractPluginSettings<PluginTypes>>;\n\n  private readonly currentSettings: ExtractPluginSettings<PluginTypes>;\n  private readonly properties: PropertiesMap<ExtractPluginSettings<PluginTypes>>;\n  // eslint-disable-next-line @typescript-eslint/no-explicit-any\n  private readonly validators: Map<string, Validator<any>> = new Map<string, Validator<any>>();\n\n  /**\n   * Creates a new plugin settings manager.\n   *\n   * @param plugin - The plugin.\n   */\n  public constructor(public readonly plugin: ExtractPlugin<PluginTypes>) {\n    this.app = plugin.app;\n    this.currentSettings = this.createDefaultSettings();\n\n    this.registerValidators();\n\n    this.properties = new PropertiesMap<ExtractPluginSettings<PluginTypes>>();\n\n    const record = this.currentSettings as Record<string, unknown>;\n\n    for (const propertyName of getAllKeys(this.currentSettings)) {\n      function propertySetter(value: unknown): void {\n        record[propertyName] = value;\n      }\n\n      this.properties.set(\n        propertyName,\n        new PluginSettingsProperty(propertyName, this.currentSettings[propertyName], this.validators.get(propertyName) ?? noop, propertySetter)\n      );\n    }\n\n    this.validators.clear();\n\n    this.safeSettings = new Proxy(this.currentSettings, new SafeSettingsProxyHandler<ExtractPluginSettings<PluginTypes>>(this.properties)) as ReadonlyDeep<\n      ExtractPluginSettings<PluginTypes>\n    >;\n  }\n\n  /**\n   * Edits the plugin settings and saves them.\n   *\n   * @param editor - The editor.\n   * @param context - The context.\n   * @returns A {@link Promise} that resolves when the settings are saved.\n   */\n  public async editAndSave(editor: (settings: ExtractPluginSettings<PluginTypes>) => Promisable<void>, context?: unknown): Promise<void> {\n    const editableSettings = new Proxy(this.currentSettings, new EditableSettingsProxyHandler<ExtractPluginSettings<PluginTypes>>(this.properties)) as {\n      validationPromise: Promise<void>;\n    } & ExtractPluginSettings<PluginTypes>;\n    await editor(editableSettings);\n    await editableSettings.validationPromise;\n    await this.saveToFile(context);\n  }\n\n  /**\n   * Gets a property of the plugin settings.\n   *\n   * @param propertyName - The name of the property.\n   * @returns The property.\n   */\n  public getProperty<PropertyName extends StringKeys<ExtractPluginSettings<PluginTypes>>>(\n    propertyName: PropertyName\n  ): PluginSettingsProperty<ExtractPluginSettings<PluginTypes>[PropertyName]> {\n    return this.properties.getTyped(propertyName);\n  }\n\n  /**\n   * Loads the plugin settings from the file.\n   *\n   * @param isInitialLoad - Whether the settings are being loaded for the first time.\n   * @returns A {@link Promise} that resolves when the settings are loaded.\n   */\n  public async loadFromFile(isInitialLoad: boolean): Promise<void> {\n    for (const property of this.properties.values()) {\n      property.reset();\n    }\n\n    const data = await this.plugin.loadData() as unknown;\n\n    if (data === undefined || data === null) {\n      return;\n    }\n\n    if (typeof data !== 'object' || Array.isArray(data)) {\n      const type = Array.isArray(data) ? 'Array' : typeof data;\n      console.error(`Invalid data type. Expected Object, got: ${type}`);\n      return;\n    }\n\n    let record = data as Record<string, unknown>;\n    const originalJson = JSON.stringify(record);\n    record = this.getTransformer().transformObjectRecursively(record);\n    await this.onLoadRecord(record);\n\n    const propertiesToSave: PluginSettingsProperty<unknown>[] = [];\n\n    for (const [propertyName, value] of Object.entries(record)) {\n      const property = this.properties.get(propertyName);\n      if (!property) {\n        console.warn(`Unknown property: ${propertyName}`);\n        continue;\n      }\n\n      if (typeof value !== typeof property.defaultValue) {\n        console.warn(\n          'Possible invalid value type read from config file. It might lead to an unexpected behavior of the plugin. There is also a chance it is a false-negative warning, as we are unable to determine the exact type of the value in runtime.',\n          {\n            defaultValue: property.defaultValue as unknown,\n            propertyName,\n            value\n          }\n        );\n      }\n\n      property.setValue(value);\n      propertiesToSave.push(property);\n    }\n\n    for (const property of propertiesToSave) {\n      await property.validate();\n      property.save();\n    }\n\n    const newJson = JSON.stringify(await this.prepareRecordToSave());\n\n    if (newJson !== originalJson) {\n      await this.saveToFileImpl();\n    }\n\n    await this.plugin.onLoadSettings(this.getSavedSettings(), isInitialLoad);\n  }\n\n  /**\n   * Saves the new plugin settings.\n   *\n   * @param context - The context of the save to file operation.\n   * @returns A {@link Promise} that resolves when the settings are saved.\n   */\n  public async saveToFile(context?: unknown): Promise<void> {\n    const oldSettings = this.getSavedSettings();\n\n    let hasChanges = false;\n\n    for (const property of this.properties.values()) {\n      hasChanges ||= property.save();\n    }\n\n    if (!hasChanges) {\n      return;\n    }\n\n    await this.saveToFileImpl();\n    await this.plugin.onSaveSettings(this.getSavedSettings(), oldSettings, context);\n  }\n\n  protected abstract createDefaultSettings(): ExtractPluginSettings<PluginTypes>;\n\n  /**\n   * Gets the transformer.\n   *\n   * @returns The transformer.\n   */\n  protected getTransformer(): Transformer {\n    return defaultTransformer;\n  }\n\n  /**\n   * Called when the plugin settings are loaded.\n   *\n   * @param _record - The record.\n   */\n  protected async onLoadRecord(_record: Record<string, unknown>): Promise<void> {\n    await noopAsync();\n  }\n\n  /**\n   * Called when the plugin settings are saving.\n   *\n   * @param _record - The record.\n   */\n  protected async onSavingRecord(_record: Record<string, unknown>): Promise<void> {\n    await noopAsync();\n  }\n\n  /**\n   * Registers a validator for a property.\n   *\n   * @param propertyName - The name of the property.\n   * @param validator - The validator.\n   */\n  protected registerValidator<PropertyName extends StringKeys<ExtractPluginSettings<PluginTypes>>>(\n    propertyName: PropertyName,\n    validator: Validator<ExtractPluginSettings<PluginTypes>[PropertyName]>\n  ): void {\n    this.validators.set(propertyName, validator);\n  }\n\n  /**\n   * Registers the validators.\n   *\n   * This method can be overridden by subclasses to register validators for properties.\n   */\n  protected registerValidators(): void {\n    noop();\n  }\n\n  private getSavedSettings(): ExtractPluginSettings<PluginTypes> {\n    const savedSettings: Record<string, unknown> = {};\n    for (const [propertyName, property] of this.properties.entries()) {\n      savedSettings[propertyName] = property.lastSavedValue as\n        | ExtractPluginSettings<PluginTypes>[StringKeys<ExtractPluginSettings<PluginTypes>>]\n        | undefined;\n    }\n    const proto = Object.getPrototypeOf(this.currentSettings) as object;\n    Object.setPrototypeOf(savedSettings, proto);\n\n    return savedSettings as ExtractPluginSettings<PluginTypes>;\n  }\n\n  private async prepareRecordToSave(): Promise<Record<string, unknown>> {\n    const settings: Record<string, unknown> = {};\n    for (const [propertyName, property] of this.properties.entries()) {\n      settings[propertyName] = property.currentValue as unknown;\n    }\n\n    await this.onSavingRecord(settings);\n\n    return this.getTransformer().transformObjectRecursively(settings);\n  }\n\n  private async saveToFileImpl(): Promise<void> {\n    await this.plugin.saveData(await this.prepareRecordToSave());\n  }\n}\n\n/**\n * A property of a plugin settings.\n *\n * @typeParam T - The type of the property.\n */\nexport class PluginSettingsProperty<T> {\n  /**\n   * The current value of the property.\n   *\n   * @returns The current value.\n   */\n  public get currentValue(): T {\n    return this._currentValue;\n  }\n\n  /**\n   * The last saved value of the property.\n   *\n   * @returns The last saved value.\n   */\n  public get lastSavedValue(): T {\n    return this._lastSavedValue;\n  }\n\n  /**\n   * The safe value of the property.\n   *\n   * @returns The safe value.\n   */\n  public get safeValue(): T {\n    return this._validationMessage ? this.defaultValue : this._currentValue;\n  }\n\n  /**\n   * The validation message of the property.\n   *\n   * @returns The validation message.\n   */\n  public get validationMessage(): string {\n    return this._validationMessage;\n  }\n\n  private _currentValue: T;\n\n  private _lastSavedValue: T;\n\n  private _validationMessage = '';\n\n  /**\n   * Creates a new plugin settings property.\n   *\n   * @param propertyName - The name of the property.\n   * @param defaultValue - The default value of the property.\n   * @param validator - The validator of the property.\n   * @param propertySetter - The property setter of the property.\n   */\n  public constructor(\n    private readonly propertyName: string,\n    public readonly defaultValue: T,\n    private readonly validator: Validator<T>,\n    private readonly propertySetter: PropertySetter\n  ) {\n    this._lastSavedValue = defaultValue;\n    this._currentValue = defaultValue;\n  }\n\n  /**\n   * Resets the current value of the property to the default value.\n   */\n  public reset(): void {\n    this._currentValue = this.defaultValue;\n    this._validationMessage = '';\n  }\n\n  /**\n   * Saves the current value of the property.\n   *\n   * @returns `true` if the value was changed, `false` otherwise.\n   */\n  public save(): boolean {\n    if (this._lastSavedValue === this._currentValue) {\n      return false;\n    }\n\n    this._lastSavedValue = this._currentValue;\n    return true;\n  }\n\n  /**\n   * Sets the validation message of the property.\n   *\n   * @param validationMessage - The validation message.\n   */\n  public setValidationMessage(validationMessage: string): void {\n    this._validationMessage = validationMessage;\n    this.showWarning();\n  }\n\n  /**\n   * Sets the current value of the property.\n   *\n   * @param value - The value to set.\n   */\n  public setValue(value: T): void {\n    this._currentValue = value;\n    this.propertySetter(value);\n  }\n\n  /**\n   * Validates the current value of the property.\n   *\n   * @returns A {@link Promise} that resolves when the validation is complete.\n   */\n  public async validate(): Promise<void> {\n    try {\n      this._validationMessage = (await this.validator(this._currentValue) as string | undefined) ?? '';\n    } catch (error) {\n      console.error('Validation failed', {\n        propertyName: this.propertyName,\n        value: this._currentValue\n      }, error);\n      this._validationMessage = 'Validation failed';\n    }\n    this.showWarning(this._currentValue);\n  }\n\n  private showWarning(value?: T): void {\n    if (!this._validationMessage) {\n      return;\n    }\n\n    console.warn(`Could not set plugin setting: ${this.propertyName}. Using default value instead.`, {\n      defaultValue: this.defaultValue,\n      propertyName: this.propertyName,\n      validationMessage: this._validationMessage,\n      value\n    });\n  }\n}\n"],
  "mappings": ";;;;;;;;;;;;;;;;;;;;;;;;;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAuBA,sBAGO;AACP,oBAA2B;AAC3B,6BAAgC;AAChC,iCAAoC;AACpC,8BAAiC;AACjC,4CAA+C;AAE/C,MAAM,qBAAqB,IAAI,yCAAiB;AAAA,EAC9C,IAAI,qEAA+B;AAAA,EACnC,IAAI,uCAAgB;AAAA,EACpB,IAAI,+CAAoB;AAC1B,CAAC;AAMD,MAAe,iBAAwF;AAAA,EAC9F,YAA+B,YAA2C;AAA3C;AAAA,EAA4C;AAAA,EAE3E,IAAI,QAAwB,MAAgC;AACjE,UAAM,SAAS;AACf,QAAI,OAAO,SAAS,UAAU;AAC5B,aAAO,OAAO,IAAI;AAAA,IACpB;AAEA,UAAM,WAAW,KAAK,WAAW,IAAI,IAAI;AACzC,QAAI,CAAC,UAAU;AACb,aAAO,OAAO,IAAI;AAAA,IACpB;AAEA,WAAO,KAAK,iBAAiB,QAAQ;AAAA,EACvC;AAGF;AAEA,MAAM,qCAAoE,iBAAiC;AAAA,EACjG,oBAAoB,QAAQ,QAAQ;AAAA,EACrC,IAAI,QAAwB,MAAuB,OAAyB;AACjF,UAAM,SAAS;AACf,WAAO,IAAI,IAAI;AAEf,QAAI,OAAO,SAAS,UAAU;AAC5B,aAAO;AAAA,IACT;AAEA,UAAM,WAAW,KAAK,WAAW,IAAI,IAAI;AACzC,QAAI,CAAC,UAAU;AACb,aAAO;AAAA,IACT;AAEA,aAAS,SAAS,KAAK;AACvB,SAAK,oBAAoB,KAAK,kBAAkB,KAAK,MAAM,SAAS,SAAS,CAAC;AAC9E,WAAO;AAAA,EACT;AAAA,EAEmB,iBAAiB,UAAoD;AACtF,WAAO,SAAS;AAAA,EAClB;AACF;AAGA,MAAM,sBAAqD,IAAyC;AAAA,EAC3F,SAA0D,cAAkF;AACjJ,UAAM,WAAW,MAAM,IAAI,YAAY;AACvC,QAAI,CAAC,UAAU;AACb,YAAM,IAAI,MAAM,YAAY,OAAO,YAAY,CAAC,YAAY;AAAA,IAC9D;AAEA,WAAO;AAAA,EACT;AAAA,EAEO,SACL,cACA,OACM;AACN,WAAO,MAAM,IAAI,cAAc,KAAK;AAAA,EACtC;AACF;AAEA,MAAM,iCAAgE,iBAAiC;AAAA,EAClF,iBAAiB,UAAoD;AACtF,WAAO,SAAS;AAAA,EAClB;AACF;AAOO,MAAe,0BAA+D;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAc5E,YAA4B,QAAoC;AAApC;AACjC,SAAK,MAAM,OAAO;AAClB,SAAK,kBAAkB,KAAK,sBAAsB;AAElD,SAAK,mBAAmB;AAExB,SAAK,aAAa,IAAI,cAAkD;AAExE,UAAM,SAAS,KAAK;AAEpB,eAAW,oBAAgB,0BAAW,KAAK,eAAe,GAAG;AAC3D,UAASA,kBAAT,SAAwB,OAAsB;AAC5C,eAAO,YAAY,IAAI;AAAA,MACzB;AAFS,2BAAAA;AAIT,WAAK,WAAW;AAAA,QACd;AAAA,QACA,IAAI,uBAAuB,cAAc,KAAK,gBAAgB,YAAY,GAAG,KAAK,WAAW,IAAI,YAAY,KAAK,sBAAMA,eAAc;AAAA,MACxI;AAAA,IACF;AAEA,SAAK,WAAW,MAAM;AAEtB,SAAK,eAAe,IAAI,MAAM,KAAK,iBAAiB,IAAI,yBAA6D,KAAK,UAAU,CAAC;AAAA,EAGvI;AAAA,EAvCgB;AAAA,EACA;AAAA,EAEC;AAAA,EACA;AAAA;AAAA,EAEA,aAA0C,oBAAI,IAA4B;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EA0C3F,MAAa,YAAY,QAA4E,SAAkC;AACrI,UAAM,mBAAmB,IAAI,MAAM,KAAK,iBAAiB,IAAI,6BAAiE,KAAK,UAAU,CAAC;AAG9I,UAAM,OAAO,gBAAgB;AAC7B,UAAM,iBAAiB;AACvB,UAAM,KAAK,WAAW,OAAO;AAAA,EAC/B;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAQO,YACL,cAC0E;AAC1E,WAAO,KAAK,WAAW,SAAS,YAAY;AAAA,EAC9C;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAQA,MAAa,aAAa,eAAuC;AAC/D,eAAW,YAAY,KAAK,WAAW,OAAO,GAAG;AAC/C,eAAS,MAAM;AAAA,IACjB;AAEA,UAAM,OAAO,MAAM,KAAK,OAAO,SAAS;AAExC,QAAI,SAAS,UAAa,SAAS,MAAM;AACvC;AAAA,IACF;AAEA,QAAI,OAAO,SAAS,YAAY,MAAM,QAAQ,IAAI,GAAG;AACnD,YAAM,OAAO,MAAM,QAAQ,IAAI,IAAI,UAAU,OAAO;AACpD,cAAQ,MAAM,4CAA4C,IAAI,EAAE;AAChE;AAAA,IACF;AAEA,QAAI,SAAS;AACb,UAAM,eAAe,KAAK,UAAU,MAAM;AAC1C,aAAS,KAAK,eAAe,EAAE,2BAA2B,MAAM;AAChE,UAAM,KAAK,aAAa,MAAM;AAE9B,UAAM,mBAAsD,CAAC;AAE7D,eAAW,CAAC,cAAc,KAAK,KAAK,OAAO,QAAQ,MAAM,GAAG;AAC1D,YAAM,WAAW,KAAK,WAAW,IAAI,YAAY;AACjD,UAAI,CAAC,UAAU;AACb,gBAAQ,KAAK,qBAAqB,YAAY,EAAE;AAChD;AAAA,MACF;AAEA,UAAI,OAAO,UAAU,OAAO,SAAS,cAAc;AACjD,gBAAQ;AAAA,UACN;AAAA,UACA;AAAA,YACE,cAAc,SAAS;AAAA,YACvB;AAAA,YACA;AAAA,UACF;AAAA,QACF;AAAA,MACF;AAEA,eAAS,SAAS,KAAK;AACvB,uBAAiB,KAAK,QAAQ;AAAA,IAChC;AAEA,eAAW,YAAY,kBAAkB;AACvC,YAAM,SAAS,SAAS;AACxB,eAAS,KAAK;AAAA,IAChB;AAEA,UAAM,UAAU,KAAK,UAAU,MAAM,KAAK,oBAAoB,CAAC;AAE/D,QAAI,YAAY,cAAc;AAC5B,YAAM,KAAK,eAAe;AAAA,IAC5B;AAEA,UAAM,KAAK,OAAO,eAAe,KAAK,iBAAiB,GAAG,aAAa;AAAA,EACzE;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAQA,MAAa,WAAW,SAAkC;AACxD,UAAM,cAAc,KAAK,iBAAiB;AAE1C,QAAI,aAAa;AAEjB,eAAW,YAAY,KAAK,WAAW,OAAO,GAAG;AAC/C,qBAAe,SAAS,KAAK;AAAA,IAC/B;AAEA,QAAI,CAAC,YAAY;AACf;AAAA,IACF;AAEA,UAAM,KAAK,eAAe;AAC1B,UAAM,KAAK,OAAO,eAAe,KAAK,iBAAiB,GAAG,aAAa,OAAO;AAAA,EAChF;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EASU,iBAA8B;AACtC,WAAO;AAAA,EACT;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAOA,MAAgB,aAAa,SAAiD;AAC5E,cAAM,2BAAU;AAAA,EAClB;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAOA,MAAgB,eAAe,SAAiD;AAC9E,cAAM,2BAAU;AAAA,EAClB;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAQU,kBACR,cACA,WACM;AACN,SAAK,WAAW,IAAI,cAAc,SAAS;AAAA,EAC7C;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAOU,qBAA2B;AACnC,8BAAK;AAAA,EACP;AAAA,EAEQ,mBAAuD;AAC7D,UAAM,gBAAyC,CAAC;AAChD,eAAW,CAAC,cAAc,QAAQ,KAAK,KAAK,WAAW,QAAQ,GAAG;AAChE,oBAAc,YAAY,IAAI,SAAS;AAAA,IAGzC;AACA,UAAM,QAAQ,OAAO,eAAe,KAAK,eAAe;AACxD,WAAO,eAAe,eAAe,KAAK;AAE1C,WAAO;AAAA,EACT;AAAA,EAEA,MAAc,sBAAwD;AACpE,UAAM,WAAoC,CAAC;AAC3C,eAAW,CAAC,cAAc,QAAQ,KAAK,KAAK,WAAW,QAAQ,GAAG;AAChE,eAAS,YAAY,IAAI,SAAS;AAAA,IACpC;AAEA,UAAM,KAAK,eAAe,QAAQ;AAElC,WAAO,KAAK,eAAe,EAAE,2BAA2B,QAAQ;AAAA,EAClE;AAAA,EAEA,MAAc,iBAAgC;AAC5C,UAAM,KAAK,OAAO,SAAS,MAAM,KAAK,oBAAoB,CAAC;AAAA,EAC7D;AACF;AAOO,MAAM,uBAA0B;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAmD9B,YACY,cACD,cACC,WACA,gBACjB;AAJiB;AACD;AACC;AACA;AAEjB,SAAK,kBAAkB;AACvB,SAAK,gBAAgB;AAAA,EACvB;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EArDA,IAAW,eAAkB;AAC3B,WAAO,KAAK;AAAA,EACd;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAOA,IAAW,iBAAoB;AAC7B,WAAO,KAAK;AAAA,EACd;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAOA,IAAW,YAAe;AACxB,WAAO,KAAK,qBAAqB,KAAK,eAAe,KAAK;AAAA,EAC5D;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAOA,IAAW,oBAA4B;AACrC,WAAO,KAAK;AAAA,EACd;AAAA,EAEQ;AAAA,EAEA;AAAA,EAEA,qBAAqB;AAAA;AAAA;AAAA;AAAA,EAuBtB,QAAc;AACnB,SAAK,gBAAgB,KAAK;AAC1B,SAAK,qBAAqB;AAAA,EAC5B;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAOO,OAAgB;AACrB,QAAI,KAAK,oBAAoB,KAAK,eAAe;AAC/C,aAAO;AAAA,IACT;AAEA,SAAK,kBAAkB,KAAK;AAC5B,WAAO;AAAA,EACT;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAOO,qBAAqB,mBAAiC;AAC3D,SAAK,qBAAqB;AAC1B,SAAK,YAAY;AAAA,EACnB;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAOO,SAAS,OAAgB;AAC9B,SAAK,gBAAgB;AACrB,SAAK,eAAe,KAAK;AAAA,EAC3B;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAOA,MAAa,WAA0B;AACrC,QAAI;AACF,WAAK,qBAAsB,MAAM,KAAK,UAAU,KAAK,aAAa,KAA4B;AAAA,IAChG,SAAS,OAAO;AACd,cAAQ,MAAM,qBAAqB;AAAA,QACjC,cAAc,KAAK;AAAA,QACnB,OAAO,KAAK;AAAA,MACd,GAAG,KAAK;AACR,WAAK,qBAAqB;AAAA,IAC5B;AACA,SAAK,YAAY,KAAK,aAAa;AAAA,EACrC;AAAA,EAEQ,YAAY,OAAiB;AACnC,QAAI,CAAC,KAAK,oBAAoB;AAC5B;AAAA,IACF;AAEA,YAAQ,KAAK,iCAAiC,KAAK,YAAY,kCAAkC;AAAA,MAC/F,cAAc,KAAK;AAAA,MACnB,cAAc,KAAK;AAAA,MACnB,mBAAmB,KAAK;AAAA,MACxB;AAAA,IACF,CAAC;AAAA,EACH;AACF;",
  "names": ["propertySetter"]
}

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