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

|
305
|
+
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../../../src/obsidian/Plugin/PluginSettingsManagerBase.ts"],
  "sourcesContent": ["/**\n * @packageDocumentation\n *\n * Plugin settings manager base class.\n */\n\nimport type { App } from 'obsidian';\nimport type {\n  Promisable,\n  ReadonlyDeep\n} from 'type-fest';\n\nimport type { GenericObject } from '../../Object.ts';\nimport type { Transformer } from '../../Transformers/Transformer.ts';\nimport type {\n  MaybeReturn,\n  StringKeys\n} from '../../Type.ts';\nimport type { PluginSettingsWrapper } from './PluginSettingsWrapper.ts';\nimport type {\n  ExtractPlugin,\n  ExtractPluginSettings,\n  ExtractPluginSettingsPropertyNames,\n  ExtractPluginSettingsPropertyValues,\n  ExtractPluginSettingsWrapper,\n  ExtractReadonlyPluginSettingsWrapper,\n  PluginTypesBase\n} from './PluginTypesBase.ts';\n\nimport {\n  noop,\n  noopAsync\n} from '../../Function.ts';\nimport {\n  deepEqual,\n  getAllKeys\n} from '../../Object.ts';\nimport { DateTransformer } from '../../Transformers/DateTransformer.ts';\nimport { DurationTransformer } from '../../Transformers/DurationTransformer.ts';\nimport { GroupTransformer } from '../../Transformers/GroupTransformer.ts';\nimport { SkipPrivatePropertyTransformer } from '../../Transformers/SkipPrivatePropertyTransformer.ts';\n\nconst defaultTransformer = new GroupTransformer([\n  new SkipPrivatePropertyTransformer(),\n  new DateTransformer(),\n  new DurationTransformer()\n]);\n\ntype ValidationResult<PluginSettings extends object> = Partial<Record<StringKeys<PluginSettings>, string>>;\n\ntype Validator<PluginSettings extends object, PropertyName extends StringKeys<PluginSettings> = StringKeys<PluginSettings>> = (\n  value: PluginSettings[PropertyName],\n  settings: PluginSettings\n) => Promisable<MaybeReturn<string>>;\n\n/**\n * Base class for managing plugin settings.\n *\n * @typeParam PluginTypes - Plugin-specific types.\n */\nexport abstract class PluginSettingsManagerBase<PluginTypes extends PluginTypesBase> {\n  public readonly app: App;\n\n  public readonly defaultSettings: ReadonlyDeep<ExtractPluginSettings<PluginTypes>>;\n\n  /**\n   * Gets the current settings wrapper.\n   *\n   * @returns The current settings wrapper.\n   */\n  public get settingsWrapper(): ExtractReadonlyPluginSettingsWrapper<PluginTypes> {\n    return this.currentSettingsWrapper as ExtractReadonlyPluginSettingsWrapper<PluginTypes>;\n  }\n\n  private currentSettingsWrapper: ExtractPluginSettingsWrapper<PluginTypes>;\n\n  private lastSavedSettingsWrapper: ExtractPluginSettingsWrapper<PluginTypes>;\n  private readonly propertyNames: ExtractPluginSettingsPropertyNames<PluginTypes>[];\n  private readonly validators = new Map<ExtractPluginSettingsPropertyNames<PluginTypes>, Validator<ExtractPluginSettings<PluginTypes>>>();\n  /**\n   * Creates a new plugin settings manager.\n   *\n   * @param plugin - The plugin.\n   */\n  public constructor(public readonly plugin: ExtractPlugin<PluginTypes>) {\n    this.app = plugin.app;\n    this.defaultSettings = this.createDefaultSettings() as ReadonlyDeep<ExtractPluginSettings<PluginTypes>>;\n    this.currentSettingsWrapper = this.createDefaultSettingsWrapper();\n    this.lastSavedSettingsWrapper = this.createDefaultSettingsWrapper();\n    this.propertyNames = getAllKeys(this.currentSettingsWrapper.settings);\n    this.registerValidators();\n  }\n\n  /**\n   * Edits the plugin settings and saves them.\n   *\n   * @param settingsEditor - The editor.\n   * @param context - The context.\n   * @returns A {@link Promise} that resolves when the settings are saved.\n   */\n  public async editAndSave(settingsEditor: (settings: ExtractPluginSettings<PluginTypes>) => Promisable<void>, context?: unknown): Promise<void> {\n    await this.edit(settingsEditor);\n    await this.saveToFile(context);\n  }\n\n  /**\n   * Ensures the settings are safe.\n   *\n   * It runs validation for each property and sets the default value if the validation fails.\n   *\n   * @param settings - The settings.\n   * @returns A {@link Promise} that resolves when the settings are safe.\n   */\n  public async ensureSafe(settings: ExtractPluginSettings<PluginTypes>): Promise<void> {\n    const validationResult = await this.validate(settings);\n    for (const propertyName of this.propertyNames) {\n      if (validationResult[propertyName]) {\n        settings[propertyName] = this.defaultSettings[propertyName];\n      }\n    }\n  }\n\n  /**\n   * Gets a safe copy of the settings.\n   *\n   * @param settings - The settings.\n   * @returns A {@link Promise} that resolves to the safe copy of the settings.\n   */\n  public async getSafeCopy(settings: ExtractPluginSettings<PluginTypes>): Promise<ExtractPluginSettings<PluginTypes>> {\n    const safeSettings = this.cloneSettings(settings);\n    await this.ensureSafe(safeSettings);\n    return safeSettings;\n  }\n\n  /**\n   * Loads the plugin settings from the file.\n   *\n   * @param isInitialLoad - Whether the settings are being loaded for the first time.\n   * @returns A {@link Promise} that resolves when the settings are loaded.\n   */\n  public async loadFromFile(isInitialLoad: boolean): Promise<void> {\n    const data = await this.plugin.loadData() as unknown;\n    this.lastSavedSettingsWrapper = this.createDefaultSettingsWrapper();\n    this.currentSettingsWrapper = this.createDefaultSettingsWrapper();\n\n    if (data === undefined || data === null) {\n      return;\n    }\n\n    if (typeof data !== 'object') {\n      console.error(`Invalid settings from data.json. Expected Object, got: ${typeof data}`);\n      return;\n    }\n\n    const rawRecord = data as GenericObject;\n    const parsedSettings = await this.rawRecordToSettings(rawRecord);\n    const validationResult = await this.validate(parsedSettings);\n\n    for (const propertyName of this.propertyNames) {\n      this.setPropertyImpl(propertyName, parsedSettings[propertyName], validationResult[propertyName]);\n    }\n\n    this.lastSavedSettingsWrapper = this.cloneSettingsWrapper(this.currentSettingsWrapper);\n\n    const newRecord = await this.settingsToRawRecord(this.currentSettingsWrapper.settings);\n\n    if (!deepEqual(newRecord, data)) {\n      await this.saveToFileImpl();\n    }\n\n    await this.plugin.onLoadSettings(this.currentSettingsWrapper, isInitialLoad);\n  }\n\n  /**\n   * Saves the new plugin settings.\n   *\n   * @param context - The context of the save to file operation.\n   * @returns A {@link Promise} that resolves when the settings are saved.\n   */\n  public async saveToFile(context?: unknown): Promise<void> {\n    if (deepEqual(this.lastSavedSettingsWrapper.settings, this.currentSettingsWrapper.settings)) {\n      return;\n    }\n\n    await this.saveToFileImpl();\n    await this.plugin.onSaveSettings(this.currentSettingsWrapper, this.lastSavedSettingsWrapper, context);\n    this.lastSavedSettingsWrapper = this.cloneSettingsWrapper(this.currentSettingsWrapper);\n  }\n\n  /**\n   * Sets the value of a property.\n   *\n   * @typeParam PropertyName - The name of the property.\n   * @param propertyName - The name of the property.\n   * @param value - The value to set.\n   * @returns A {@link Promise} that resolves to the validation message.\n   */\n  public async setProperty<PropertyName extends ExtractPluginSettingsPropertyNames<PluginTypes>>(\n    propertyName: PropertyName,\n    value: ExtractPluginSettings<PluginTypes>[PropertyName]\n  ): Promise<string> {\n    await this.edit((settings) => {\n      settings[propertyName] = value;\n    });\n    return this.currentSettingsWrapper.validationMessages[propertyName];\n  }\n\n  /**\n   * Validates the settings.\n   *\n   * @param settings - The settings.\n   * @returns A {@link Promise} that resolves to the validation result.\n   */\n  public async validate(settings: ExtractPluginSettings<PluginTypes>): Promise<ValidationResult<ExtractPluginSettings<PluginTypes>>> {\n    const result: ValidationResult<ExtractPluginSettings<PluginTypes>> = {};\n    for (const [propertyName, validator] of this.validators.entries()) {\n      const validationMessage = await validator(settings[propertyName], settings);\n      if (validationMessage) {\n        result[propertyName] = validationMessage;\n      }\n    }\n\n    return result;\n  }\n\n  protected abstract createDefaultSettings(): ExtractPluginSettings<PluginTypes>;\n\n  /**\n   * Gets the transformer.\n   *\n   * @returns The transformer.\n   */\n  protected getTransformer(): Transformer {\n    return defaultTransformer;\n  }\n\n  /**\n   * Called when the plugin settings are loaded.\n   *\n   * @param _record - The record.\n   */\n  protected async onLoadRecord(_record: GenericObject): Promise<void> {\n    await noopAsync();\n  }\n\n  /**\n   * Called when the plugin settings are saving.\n   *\n   * @param _record - The record.\n   */\n  protected async onSavingRecord(_record: GenericObject): Promise<void> {\n    await noopAsync();\n  }\n\n  /**\n   * Registers a validator for a property.\n   *\n   * @param propertyName - The name of the property.\n   * @param validator - The validator.\n   */\n  protected registerValidator<PropertyName extends ExtractPluginSettingsPropertyNames<PluginTypes>>(\n    propertyName: PropertyName,\n    validator: Validator<ExtractPluginSettings<PluginTypes>, PropertyName>\n  ): void {\n    this.validators.set(propertyName, validator as Validator<ExtractPluginSettings<PluginTypes>>);\n  }\n\n  /**\n   * Registers the validators.\n   *\n   * This method can be overridden by subclasses to register validators for properties.\n   */\n  protected registerValidators(): void {\n    noop();\n  }\n\n  private cloneSettings(settings: ExtractPluginSettings<PluginTypes>): ExtractPluginSettings<PluginTypes> {\n    const record = this.settingsToRawRecord(settings);\n    const json = JSON.stringify(record);\n    const cloneRecord = JSON.parse(json) as GenericObject;\n    return this.rawRecordToSettings(cloneRecord);\n  }\n\n  private cloneSettingsWrapper(\n    settingsWrapper: PluginSettingsWrapper<ExtractPluginSettings<PluginTypes>>\n  ): PluginSettingsWrapper<ExtractPluginSettings<PluginTypes>> {\n    return {\n      safeSettings: this.cloneSettings(settingsWrapper.safeSettings),\n      settings: this.cloneSettings(settingsWrapper.settings),\n      validationMessages: { ...settingsWrapper.validationMessages }\n    };\n  }\n\n  private createDefaultSettingsWrapper(): PluginSettingsWrapper<ExtractPluginSettings<PluginTypes>> {\n    return {\n      safeSettings: this.createDefaultSettings(),\n      settings: this.createDefaultSettings(),\n      validationMessages: {} as Record<ExtractPluginSettingsPropertyNames<PluginTypes>, string>\n    };\n  }\n\n  private async edit(settingsEditor: (settings: ExtractPluginSettings<PluginTypes>) => Promisable<void>): Promise<void> {\n    try {\n      await settingsEditor(this.currentSettingsWrapper.settings);\n    } finally {\n      const validationResult = await this.validate(this.currentSettingsWrapper.settings);\n      for (const propertyName of this.propertyNames) {\n        const validationMessage = validationResult[propertyName] ?? '';\n        this.currentSettingsWrapper.validationMessages[propertyName] = validationMessage;\n        this.currentSettingsWrapper.safeSettings[propertyName] = validationMessage\n          ? this.defaultSettings[propertyName]\n          : this.currentSettingsWrapper.settings[propertyName];\n      }\n    }\n  }\n\n  private isValidPropertyName(prop: unknown): prop is ExtractPluginSettingsPropertyNames<PluginTypes> {\n    if (typeof prop !== 'string') {\n      return false;\n    }\n\n    return (this.propertyNames as string[]).includes(prop);\n  }\n\n  private async rawRecordToSettings(rawRecord: GenericObject): Promise<ExtractPluginSettings<PluginTypes>> {\n    await this.onLoadRecord(rawRecord);\n\n    const settings = this.createDefaultSettings();\n\n    for (const [propertyName, value] of Object.entries(rawRecord)) {\n      if (!this.isValidPropertyName(propertyName)) {\n        console.warn(`Unknown property: ${propertyName}`);\n        continue;\n      }\n\n      if (typeof value !== typeof this.defaultSettings[propertyName]) {\n        console.warn(\n          'Possible invalid value type. It might lead to an unexpected behavior of the plugin. There is also a chance it is a false-negative warning, as we are unable to determine the exact type of the value in runtime.',\n          {\n            defaultValue: this.defaultSettings[propertyName],\n            propertyName,\n            value\n          }\n        );\n      }\n\n      settings[propertyName] = value as ExtractPluginSettingsPropertyValues<PluginTypes>;\n    }\n\n    return settings;\n  }\n\n  private async saveToFileImpl(): Promise<void> {\n    await this.plugin.saveData(await this.settingsToRawRecord(this.currentSettingsWrapper));\n  }\n\n  private setPropertyImpl(\n    propertyName: ExtractPluginSettingsPropertyNames<PluginTypes>,\n    value: ExtractPluginSettingsPropertyValues<PluginTypes>,\n    validationMessage?: string\n  ): void {\n    this.currentSettingsWrapper.settings[propertyName] = value;\n    this.currentSettingsWrapper.validationMessages[propertyName] = validationMessage ?? '';\n    this.currentSettingsWrapper.safeSettings[propertyName] = validationMessage ? this.defaultSettings[propertyName] : value;\n  }\n\n  private async settingsToRawRecord(settings: ExtractPluginSettings<PluginTypes>): Promise<GenericObject> {\n    const rawRecord: GenericObject = {};\n\n    for (const propertyName of this.propertyNames) {\n      rawRecord[propertyName] = settings[propertyName];\n    }\n\n    await this.onSavingRecord(rawRecord);\n\n    return this.getTransformer().transformObjectRecursively(rawRecord);\n  }\n}\n"],
  "mappings": ";;;;;;;;;;;;;;;;;;;;;;;;;AAAA;AAAA;AAAA;AAAA;AAAA;AA6BA,sBAGO;AACP,oBAGO;AACP,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;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,oBAAgB,0BAAW,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,KAAC,yBAAU,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,YAAI,yBAAU,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,cAAM,2BAAU;AAAA,EAClB;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAOA,MAAgB,eAAe,SAAuC;AACpE,cAAM,2BAAU;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,8BAAK;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": []
}

|