obsidian-dev-utils 22.1.1-beta.2 → 22.1.1-beta.20

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (91) hide show
  1. package/CHANGELOG.md +102 -0
  2. package/dist/dprint.json +1 -0
  3. package/dist/lib/cjs/Library.cjs +1 -1
  4. package/dist/lib/cjs/Object.cjs +38 -3
  5. package/dist/lib/cjs/Object.d.cts +16 -6
  6. package/dist/lib/cjs/ScriptUtils/CliUtils.cjs +1 -2
  7. package/dist/lib/cjs/ScriptUtils/CliUtils.d.cts +3 -2
  8. package/dist/lib/cjs/ScriptUtils/cli.cjs +1 -1
  9. package/dist/lib/cjs/String.cjs +1 -1
  10. package/dist/lib/cjs/String.d.cts +2 -1
  11. package/dist/lib/cjs/Transformers/Transformer.cjs +9 -2
  12. package/dist/lib/cjs/Type.cjs +24 -0
  13. package/dist/lib/cjs/Type.d.cts +12 -0
  14. package/dist/lib/cjs/index.cjs +4 -1
  15. package/dist/lib/cjs/index.d.cts +1 -0
  16. package/dist/lib/cjs/obsidian/Callout.cjs +1 -1
  17. package/dist/lib/cjs/obsidian/Callout.d.cts +2 -1
  18. package/dist/lib/cjs/obsidian/Components/MultipleTextComponent.cjs +29 -2
  19. package/dist/lib/cjs/obsidian/Components/MultipleTextComponent.d.cts +20 -1
  20. package/dist/lib/cjs/obsidian/Components/TextBasedComponent.cjs +60 -0
  21. package/dist/lib/cjs/obsidian/Components/TextBasedComponent.d.cts +30 -0
  22. package/dist/lib/cjs/obsidian/Components/TypedTextComponent.cjs +27 -3
  23. package/dist/lib/cjs/obsidian/Components/TypedTextComponent.d.cts +19 -1
  24. package/dist/lib/cjs/obsidian/Components/index.cjs +4 -1
  25. package/dist/lib/cjs/obsidian/Components/index.d.cts +1 -0
  26. package/dist/lib/cjs/obsidian/FileManager.cjs +1 -1
  27. package/dist/lib/cjs/obsidian/FileManager.d.cts +2 -1
  28. package/dist/lib/cjs/obsidian/Link.cjs +1 -1
  29. package/dist/lib/cjs/obsidian/Link.d.cts +4 -3
  30. package/dist/lib/cjs/obsidian/Markdown.cjs +5 -8
  31. package/dist/lib/cjs/obsidian/Modals/Prompt.cjs +1 -1
  32. package/dist/lib/cjs/obsidian/Modals/Prompt.d.cts +2 -1
  33. package/dist/lib/cjs/obsidian/MonkeyAround.cjs +1 -1
  34. package/dist/lib/cjs/obsidian/MonkeyAround.d.cts +2 -2
  35. package/dist/lib/cjs/obsidian/Plugin/PluginBase.cjs +31 -2
  36. package/dist/lib/cjs/obsidian/Plugin/PluginBase.d.cts +15 -3
  37. package/dist/lib/cjs/obsidian/Plugin/PluginSettingsManagerBase.cjs +186 -62
  38. package/dist/lib/cjs/obsidian/Plugin/PluginSettingsManagerBase.d.cts +42 -21
  39. package/dist/lib/cjs/obsidian/Plugin/PluginSettingsTabBase.cjs +33 -20
  40. package/dist/lib/cjs/obsidian/Plugin/PluginSettingsTabBase.d.cts +20 -17
  41. package/dist/lib/cjs/obsidian/RenameDeleteHandler.cjs +75 -75
  42. package/dist/lib/cjs/obsidian/ValidationMessage.cjs +38 -0
  43. package/dist/lib/cjs/obsidian/ValidationMessage.d.cts +16 -0
  44. package/dist/lib/cjs/obsidian/index.cjs +4 -1
  45. package/dist/lib/cjs/obsidian/index.d.cts +1 -0
  46. package/dist/lib/esm/Library.mjs +1 -1
  47. package/dist/lib/esm/Object.d.mts +16 -6
  48. package/dist/lib/esm/Object.mjs +36 -3
  49. package/dist/lib/esm/ScriptUtils/CliUtils.d.mts +3 -2
  50. package/dist/lib/esm/ScriptUtils/CliUtils.mjs +1 -2
  51. package/dist/lib/esm/ScriptUtils/cli.mjs +1 -1
  52. package/dist/lib/esm/String.d.mts +2 -1
  53. package/dist/lib/esm/String.mjs +1 -1
  54. package/dist/lib/esm/Transformers/Transformer.mjs +9 -2
  55. package/dist/lib/esm/Type.d.mts +12 -0
  56. package/dist/lib/esm/Type.mjs +8 -0
  57. package/dist/lib/esm/index.d.mts +1 -0
  58. package/dist/lib/esm/index.mjs +3 -1
  59. package/dist/lib/esm/obsidian/Callout.d.mts +2 -1
  60. package/dist/lib/esm/obsidian/Callout.mjs +1 -1
  61. package/dist/lib/esm/obsidian/Components/MultipleTextComponent.d.mts +20 -1
  62. package/dist/lib/esm/obsidian/Components/MultipleTextComponent.mjs +29 -2
  63. package/dist/lib/esm/obsidian/Components/TextBasedComponent.d.mts +30 -0
  64. package/dist/lib/esm/obsidian/Components/TextBasedComponent.mjs +36 -0
  65. package/dist/lib/esm/obsidian/Components/TypedTextComponent.d.mts +19 -1
  66. package/dist/lib/esm/obsidian/Components/TypedTextComponent.mjs +27 -3
  67. package/dist/lib/esm/obsidian/Components/index.d.mts +1 -0
  68. package/dist/lib/esm/obsidian/Components/index.mjs +3 -1
  69. package/dist/lib/esm/obsidian/FileManager.d.mts +2 -1
  70. package/dist/lib/esm/obsidian/FileManager.mjs +1 -1
  71. package/dist/lib/esm/obsidian/Link.d.mts +4 -3
  72. package/dist/lib/esm/obsidian/Link.mjs +1 -1
  73. package/dist/lib/esm/obsidian/Markdown.mjs +5 -8
  74. package/dist/lib/esm/obsidian/Modals/Prompt.d.mts +2 -1
  75. package/dist/lib/esm/obsidian/Modals/Prompt.mjs +1 -1
  76. package/dist/lib/esm/obsidian/MonkeyAround.d.mts +2 -2
  77. package/dist/lib/esm/obsidian/MonkeyAround.mjs +1 -1
  78. package/dist/lib/esm/obsidian/Plugin/PluginBase.d.mts +15 -3
  79. package/dist/lib/esm/obsidian/Plugin/PluginBase.mjs +35 -3
  80. package/dist/lib/esm/obsidian/Plugin/PluginSettingsManagerBase.d.mts +42 -21
  81. package/dist/lib/esm/obsidian/Plugin/PluginSettingsManagerBase.mjs +186 -63
  82. package/dist/lib/esm/obsidian/Plugin/PluginSettingsTabBase.d.mts +20 -17
  83. package/dist/lib/esm/obsidian/Plugin/PluginSettingsTabBase.mjs +36 -20
  84. package/dist/lib/esm/obsidian/RenameDeleteHandler.mjs +75 -75
  85. package/dist/lib/esm/obsidian/ValidationMessage.d.mts +16 -0
  86. package/dist/lib/esm/obsidian/ValidationMessage.mjs +14 -0
  87. package/dist/lib/esm/obsidian/index.d.mts +1 -0
  88. package/dist/lib/esm/obsidian/index.mjs +3 -1
  89. package/obsidian/Components/TextBasedComponent/package.json +6 -0
  90. package/obsidian/ValidationMessage/package.json +6 -0
  91. package/package.json +4 -4
@@ -1,30 +1,24 @@
1
- import type { ReadonlyDeep } from 'type-fest';
2
- import { Plugin } from 'obsidian';
3
- import type { StringKeys } from '../../Object.mjs';
1
+ import type { App } from 'obsidian';
2
+ import type { Promisable, ReadonlyDeep } from 'type-fest';
4
3
  import type { Transformer } from '../../Transformers/Transformer.mjs';
5
- declare class PluginSettingsProperty<PluginSettings extends object, Property extends StringKeys<PluginSettings>> {
6
- private readonly manager;
7
- private readonly property;
8
- readonly defaultValue: PluginSettings[Property];
9
- validationMessage: string;
10
- private value;
11
- constructor(manager: PluginSettingsManagerBase<PluginSettings>, property: Property, defaultValue: PluginSettings[Property]);
12
- clear(): void;
13
- get(): PluginSettings[Property];
14
- getSafe(): PluginSettings[Property];
15
- set(value: PluginSettings[Property] | undefined): Promise<void>;
16
- }
4
+ import type { MaybeReturn, StringKeys } from '../../Type.mjs';
5
+ import type { PluginBase } from './PluginBase.mjs';
6
+ type Validator<T> = (value: T) => Promisable<MaybeReturn<string>>;
17
7
  /**
18
8
  * Base class for managing plugin settings.
19
9
  *
20
10
  * @typeParam PluginSettings - The type representing the plugin settings object.
21
11
  */
22
12
  export declare abstract class PluginSettingsManagerBase<PluginSettings extends object> {
23
- private plugin;
13
+ readonly plugin: PluginBase<PluginSettings>;
14
+ readonly app: App;
24
15
  readonly safeSettings: ReadonlyDeep<PluginSettings>;
16
+ private defaultSettings;
25
17
  private properties;
26
- constructor(plugin: Plugin);
27
- getProperty<Property extends StringKeys<PluginSettings>>(property: Property): PluginSettingsProperty<PluginSettings, Property>;
18
+ private validators;
19
+ constructor(plugin: PluginBase<PluginSettings>);
20
+ editAndSave(editor: (settings: PluginSettings) => Promisable<void>): Promise<void>;
21
+ getProperty<PropertyName extends StringKeys<PluginSettings>>(propertyName: PropertyName): PluginSettingsProperty<PluginSettings[PropertyName]>;
28
22
  loadFromFile(): Promise<void>;
29
23
  /**
30
24
  * Saves the new plugin settings.
@@ -32,10 +26,37 @@ export declare abstract class PluginSettingsManagerBase<PluginSettings extends o
32
26
  * @returns A promise that resolves when the settings are saved.
33
27
  */
34
28
  saveToFile(): Promise<void>;
35
- validate<Property extends StringKeys<PluginSettings>>(_property: Property, _value: PluginSettings[Property]): Promise<string | void>;
29
+ protected addValidator<PropertyName extends StringKeys<PluginSettings>>(propertyName: PropertyName, validator: Validator<PluginSettings[PropertyName]>): void;
30
+ protected addValidators(): void;
36
31
  protected abstract createDefaultSettings(): PluginSettings;
37
32
  protected getTransformer(): Transformer;
38
- protected prepareRecord(_record: Record<string, unknown>): Promise<void>;
39
- private getSettings;
33
+ protected onLoadRecord(_record: Record<string, unknown>): Promisable<void>;
34
+ protected onSavingRecord(_record: Record<string, unknown>): Promisable<void>;
35
+ private getSavedSettings;
36
+ private prepareRecordToSave;
37
+ }
38
+ /**
39
+ * A property of a plugin settings.
40
+ *
41
+ * @typeParam T - The type of the property.
42
+ */
43
+ export declare class PluginSettingsProperty<T> {
44
+ private readonly propertyName;
45
+ readonly defaultValue: T;
46
+ private readonly validator;
47
+ get currentValue(): T;
48
+ get lastSavedValue(): T;
49
+ get safeValue(): T;
50
+ get validationMessage(): string;
51
+ private _currentValue;
52
+ private _lastSavedValue;
53
+ private _validationMessage;
54
+ constructor(propertyName: string, defaultValue: T, validator: Validator<T>);
55
+ reset(): void;
56
+ save(): boolean;
57
+ setValidationMessage(validationMessage: string): void;
58
+ setValue(value: T): void;
59
+ setValueAndValidate(value: T): Promise<void>;
60
+ private showWarning;
40
61
  }
41
62
  export {};
@@ -5,8 +5,9 @@ if you want to view the source, please visit the github repository of this plugi
5
5
 
6
6
  (function initEsm(){if(globalThis.process){return}const browserProcess={browser:true,cwd:__name(()=>"/","cwd"),env:{},platform:"android"};globalThis.process=browserProcess})();
7
7
 
8
- import { Plugin } from "obsidian";
9
- import { noopAsync } from "../../Function.mjs";
8
+ import { Notice } from "obsidian";
9
+ import { noop } from "../../Function.mjs";
10
+ import { getAllKeys } from "../../Object.mjs";
10
11
  import { DateTransformer } from "../../Transformers/DateTransformer.mjs";
11
12
  import { DurationTransformer } from "../../Transformers/DurationTransformer.mjs";
12
13
  import { GroupTransformer } from "../../Transformers/GroupTransformer.mjs";
@@ -16,68 +17,94 @@ const defaultTransformer = new GroupTransformer([
16
17
  new DateTransformer(),
17
18
  new DurationTransformer()
18
19
  ]);
19
- class PluginSettingsProperty {
20
- constructor(manager, property, defaultValue) {
21
- this.manager = manager;
22
- this.property = property;
23
- this.defaultValue = defaultValue;
24
- }
25
- validationMessage = "";
26
- value;
27
- clear() {
28
- this.value = void 0;
29
- this.validationMessage = "";
20
+ class ProxyHandlerBase {
21
+ constructor(properties) {
22
+ this.properties = properties;
30
23
  }
31
- get() {
32
- return this.value ?? this.defaultValue;
33
- }
34
- getSafe() {
35
- return this.validationMessage ? this.defaultValue : this.get();
24
+ get(target, prop) {
25
+ const record = target;
26
+ if (typeof prop !== "string") {
27
+ return record[prop];
28
+ }
29
+ const property = this.properties.get(prop);
30
+ if (!property) {
31
+ return record[prop];
32
+ }
33
+ return this.getPropertyValue(property);
36
34
  }
37
- async set(value) {
38
- this.value = value;
39
- if (this.value !== void 0) {
40
- this.validationMessage = await this.manager.validate(this.property, this.value) ?? "";
35
+ }
36
+ class EditableSettingsProxyHandler extends ProxyHandlerBase {
37
+ validationPromise = Promise.resolve();
38
+ set(target, prop, value) {
39
+ const record = target;
40
+ if (typeof prop !== "string") {
41
+ record[prop] = value;
42
+ return true;
43
+ }
44
+ const property = this.properties.get(prop);
45
+ if (!property) {
46
+ record[prop] = value;
47
+ return true;
41
48
  }
49
+ property.setValue(value);
50
+ this.validationPromise = this.validationPromise.then(() => property.setValueAndValidate(value));
51
+ return true;
52
+ }
53
+ getPropertyValue(property) {
54
+ return property.currentValue;
42
55
  }
43
56
  }
44
57
  class PropertiesMap extends Map {
45
- getTyped(key) {
46
- const property = super.get(key);
58
+ getTyped(propertyName) {
59
+ const property = super.get(propertyName);
47
60
  if (!property) {
48
- throw new Error(`Property ${String(key)} not found`);
61
+ throw new Error(`Property ${String(propertyName)} not found`);
49
62
  }
50
63
  return property;
51
64
  }
52
- setTyped(key, value) {
53
- return super.set(key, value);
65
+ setTyped(propertyName, value) {
66
+ return super.set(propertyName, value);
67
+ }
68
+ }
69
+ class SafeSettingsProxyHandler extends ProxyHandlerBase {
70
+ getPropertyValue(property) {
71
+ return property.safeValue;
54
72
  }
55
73
  }
56
74
  class PluginSettingsManagerBase {
57
75
  constructor(plugin) {
58
76
  this.plugin = plugin;
59
- const defaultSettings = this.createDefaultSettings();
77
+ this.app = plugin.app;
78
+ this.defaultSettings = this.createDefaultSettings();
79
+ this.addValidators();
60
80
  this.properties = new PropertiesMap();
61
- for (const key of Object.keys(defaultSettings)) {
62
- this.properties.set(key, new PluginSettingsProperty(this, key, defaultSettings[key]));
63
- }
64
- this.safeSettings = new Proxy(defaultSettings, {
65
- get: (_target, prop) => {
66
- if (typeof prop !== "string") {
67
- return void 0;
68
- }
69
- return this.properties.get(prop);
70
- }
71
- });
81
+ for (const propertyName of getAllKeys(this.defaultSettings)) {
82
+ this.properties.set(
83
+ propertyName,
84
+ new PluginSettingsProperty(propertyName, this.defaultSettings[propertyName], this.validators.get(propertyName) ?? noop)
85
+ );
86
+ }
87
+ this.validators.clear();
88
+ this.safeSettings = new Proxy(this.defaultSettings, new SafeSettingsProxyHandler(this.properties));
72
89
  }
90
+ app;
73
91
  safeSettings;
92
+ defaultSettings;
74
93
  properties;
75
- getProperty(property) {
76
- return this.properties.getTyped(property);
94
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
95
+ validators = /* @__PURE__ */ new Map();
96
+ async editAndSave(editor) {
97
+ const editableSettings = new Proxy(this.defaultSettings, new EditableSettingsProxyHandler(this.properties));
98
+ await editor(editableSettings);
99
+ await editableSettings.validationPromise;
100
+ await this.saveToFile();
101
+ }
102
+ getProperty(propertyName) {
103
+ return this.properties.getTyped(propertyName);
77
104
  }
78
105
  async loadFromFile() {
79
106
  for (const property of this.properties.values()) {
80
- property.clear();
107
+ property.reset();
81
108
  }
82
109
  const data = await this.plugin.loadData();
83
110
  if (data === void 0 || data === null) {
@@ -89,19 +116,28 @@ class PluginSettingsManagerBase {
89
116
  return;
90
117
  }
91
118
  let record = data;
119
+ const originalJson = JSON.stringify(record);
92
120
  record = this.getTransformer().transformObjectRecursively(record);
93
- await this.prepareRecord(record);
94
- for (const [key, value] of Object.entries(record)) {
95
- const propertyObj = this.properties.get(key);
96
- if (!propertyObj) {
97
- console.warn(`Unknown property: ${key}`);
121
+ await this.onLoadRecord(record);
122
+ for (const [propertyName, value] of Object.entries(record)) {
123
+ const property = this.properties.get(propertyName);
124
+ if (!property) {
125
+ console.warn(`Unknown property: ${propertyName}`);
98
126
  continue;
99
127
  }
100
- if (typeof value !== typeof propertyObj.defaultValue) {
101
- console.warn(`Invalid value type. Expected ${typeof propertyObj.defaultValue}, got: ${typeof value}`);
128
+ if (typeof value !== typeof property.defaultValue) {
129
+ console.warn("Invalid value type", {
130
+ propertyName,
131
+ propertyType: typeof property.defaultValue,
132
+ value
133
+ });
102
134
  continue;
103
135
  }
104
- await propertyObj.set(value);
136
+ await property.setValueAndValidate(value);
137
+ }
138
+ const newJson = JSON.stringify(await this.prepareRecordToSave());
139
+ if (newJson !== originalJson) {
140
+ await this.saveToFile();
105
141
  }
106
142
  }
107
143
  /**
@@ -110,28 +146,115 @@ class PluginSettingsManagerBase {
110
146
  * @returns A promise that resolves when the settings are saved.
111
147
  */
112
148
  async saveToFile() {
113
- const record = this.getTransformer().transformObjectRecursively(this.getSettings());
114
- await this.plugin.saveData(record);
149
+ const oldSettings = this.getSavedSettings();
150
+ let hasChanges = false;
151
+ for (const property of this.properties.values()) {
152
+ hasChanges ||= property.save();
153
+ }
154
+ if (!hasChanges) {
155
+ return;
156
+ }
157
+ await this.plugin.saveData(await this.prepareRecordToSave());
158
+ await this.plugin.onSaveSettings(this.getSavedSettings(), oldSettings);
159
+ }
160
+ addValidator(propertyName, validator) {
161
+ this.validators.set(propertyName, validator);
115
162
  }
116
- // eslint-disable-next-line @typescript-eslint/no-invalid-void-type
117
- async validate(_property, _value) {
118
- await noopAsync();
163
+ addValidators() {
164
+ noop();
119
165
  }
120
166
  getTransformer() {
121
167
  return defaultTransformer;
122
168
  }
123
- async prepareRecord(_record) {
124
- await noopAsync();
169
+ onLoadRecord(_record) {
170
+ noop();
125
171
  }
126
- getSettings() {
172
+ onSavingRecord(_record) {
173
+ noop();
174
+ }
175
+ getSavedSettings() {
176
+ const savedSettings = {};
177
+ for (const [propertyName, property] of this.properties.entries()) {
178
+ savedSettings[propertyName] = property.lastSavedValue;
179
+ }
180
+ const proto = Object.getPrototypeOf(this.defaultSettings);
181
+ Object.setPrototypeOf(savedSettings, proto);
182
+ return savedSettings;
183
+ }
184
+ async prepareRecordToSave() {
127
185
  const settings = {};
128
- for (const [key, property] of this.properties.entries()) {
129
- settings[key] = property.get();
186
+ for (const [propertyName, property] of this.properties.entries()) {
187
+ settings[propertyName] = property.currentValue;
130
188
  }
131
- return settings;
189
+ await this.onSavingRecord(settings);
190
+ return this.getTransformer().transformObjectRecursively(settings);
191
+ }
192
+ }
193
+ class PluginSettingsProperty {
194
+ constructor(propertyName, defaultValue, validator) {
195
+ this.propertyName = propertyName;
196
+ this.defaultValue = defaultValue;
197
+ this.validator = validator;
198
+ this._lastSavedValue = defaultValue;
199
+ this._currentValue = defaultValue;
200
+ }
201
+ get currentValue() {
202
+ return this._currentValue;
203
+ }
204
+ get lastSavedValue() {
205
+ return this._lastSavedValue;
206
+ }
207
+ get safeValue() {
208
+ return this._validationMessage ? this.defaultValue : this._currentValue;
209
+ }
210
+ get validationMessage() {
211
+ return this._validationMessage;
212
+ }
213
+ _currentValue;
214
+ _lastSavedValue;
215
+ _validationMessage = "";
216
+ reset() {
217
+ this._currentValue = this.defaultValue;
218
+ this._validationMessage = "";
219
+ }
220
+ save() {
221
+ if (this._lastSavedValue === this._currentValue) {
222
+ return false;
223
+ }
224
+ this._lastSavedValue = this._currentValue;
225
+ return true;
226
+ }
227
+ setValidationMessage(validationMessage) {
228
+ this._validationMessage = validationMessage;
229
+ this.showWarning();
230
+ }
231
+ setValue(value) {
232
+ this._currentValue = value;
233
+ }
234
+ async setValueAndValidate(value) {
235
+ this.setValue(value);
236
+ if (this._currentValue === void 0) {
237
+ return;
238
+ }
239
+ this._validationMessage = await this.validator(this._currentValue) ?? "";
240
+ this.showWarning(value);
241
+ }
242
+ showWarning(value) {
243
+ if (!this._validationMessage) {
244
+ return;
245
+ }
246
+ const warningMessage = `Could not set plugin setting: ${this.propertyName}. Using default value instead.`;
247
+ new Notice(warningMessage);
248
+ console.warn(warningMessage, {
249
+ defaultValue: this.defaultValue,
250
+ propertyName: this.propertyName,
251
+ validationMessage: this._validationMessage,
252
+ value
253
+ });
132
254
  }
133
255
  }
134
256
  export {
135
- PluginSettingsManagerBase
257
+ PluginSettingsManagerBase,
258
+ PluginSettingsProperty
136
259
  };
137
- //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../../../src/obsidian/Plugin/PluginSettingsManagerBase.ts"],
  "sourcesContent": ["import type { ReadonlyDeep } from 'type-fest';\n\nimport { Plugin } from 'obsidian';\n\nimport type { StringKeys } from '../../Object.ts';\nimport type { Transformer } from '../../Transformers/Transformer.ts';\n\nimport { noopAsync } from '../../Function.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\nclass PluginSettingsProperty<PluginSettings extends object, Property extends StringKeys<PluginSettings>> {\n  public validationMessage = '';\n  private value: PluginSettings[Property] | undefined;\n\n  public constructor(\n    private readonly manager: PluginSettingsManagerBase<PluginSettings>,\n    private readonly property: Property,\n    public readonly defaultValue: PluginSettings[Property]\n  ) {}\n\n  public clear(): void {\n    this.value = undefined;\n    this.validationMessage = '';\n  }\n\n  public get(): PluginSettings[Property] {\n    return this.value ?? this.defaultValue;\n  }\n\n  public getSafe(): PluginSettings[Property] {\n    return this.validationMessage ? this.defaultValue : this.get();\n  }\n\n  public async set(value: PluginSettings[Property] | undefined): Promise<void> {\n    this.value = value;\n    if (this.value !== undefined) {\n      this.validationMessage = (await this.manager.validate(this.property, this.value) as string | undefined) ?? '';\n    }\n  }\n}\n\nclass PropertiesMap<PluginSettings extends object> extends Map<string, PluginSettingsProperty<PluginSettings, StringKeys<PluginSettings>>> {\n  public getTyped<Property extends StringKeys<PluginSettings>>(key: Property): PluginSettingsProperty<PluginSettings, Property> {\n    const property = super.get(key);\n    if (!property) {\n      throw new Error(`Property ${String(key)} not found`);\n    }\n\n    return property as PluginSettingsProperty<PluginSettings, Property>;\n  }\n\n  public setTyped<Property extends StringKeys<PluginSettings>>(key: Property, value: PluginSettingsProperty<PluginSettings, Property>): this {\n    return super.set(key, value);\n  }\n}\n\n/**\n * Base class for managing plugin settings.\n *\n * @typeParam PluginSettings - The type representing the plugin settings object.\n */\nexport abstract class PluginSettingsManagerBase<PluginSettings extends object> {\n  public readonly safeSettings: ReadonlyDeep<PluginSettings>;\n\n  private properties: PropertiesMap<PluginSettings>;\n\n  public constructor(private plugin: Plugin) {\n    const defaultSettings = this.createDefaultSettings();\n    this.properties = new PropertiesMap<PluginSettings>();\n\n    for (const key of Object.keys(defaultSettings) as StringKeys<PluginSettings>[]) {\n      this.properties.set(key, new PluginSettingsProperty(this, key, defaultSettings[key]));\n    }\n\n    this.safeSettings = new Proxy(defaultSettings, {\n      get: (_target, prop): unknown => {\n        if (typeof prop !== 'string') {\n          return undefined;\n        }\n\n        return this.properties.get(prop);\n      }\n    }) as ReadonlyDeep<PluginSettings>;\n  }\n\n  public getProperty<Property extends StringKeys<PluginSettings>>(property: Property): PluginSettingsProperty<PluginSettings, Property> {\n    return this.properties.getTyped(property);\n  }\n\n  public async loadFromFile(): Promise<void> {\n    for (const property of this.properties.values()) {\n      property.clear();\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    record = this.getTransformer().transformObjectRecursively(record);\n    await this.prepareRecord(record);\n\n    for (const [key, value] of Object.entries(record)) {\n      const propertyObj = this.properties.get(key);\n      if (!propertyObj) {\n        console.warn(`Unknown property: ${key}`);\n        continue;\n      }\n\n      if (typeof value !== typeof propertyObj.defaultValue) {\n        console.warn(`Invalid value type. Expected ${typeof propertyObj.defaultValue}, got: ${typeof value}`);\n        continue;\n      }\n\n      await propertyObj.set(value as PluginSettings[StringKeys<PluginSettings>]);\n    }\n  }\n\n  /**\n   * Saves the new plugin settings.\n   *\n   * @returns A promise that resolves when the settings are saved.\n   */\n  public async saveToFile(): Promise<void> {\n    const record = this.getTransformer().transformObjectRecursively(this.getSettings());\n    await this.plugin.saveData(record);\n  }\n\n  // eslint-disable-next-line @typescript-eslint/no-invalid-void-type\n  public async validate<Property extends StringKeys<PluginSettings>>(_property: Property, _value: PluginSettings[Property]): Promise<string | void> {\n    await noopAsync();\n  }\n\n  protected abstract createDefaultSettings(): PluginSettings;\n\n  protected getTransformer(): Transformer {\n    return defaultTransformer;\n  }\n\n  protected async prepareRecord(_record: Record<string, unknown>): Promise<void> {\n    await noopAsync();\n  }\n\n  private getSettings(): Record<StringKeys<PluginSettings>, unknown> {\n    const settings: Record<StringKeys<PluginSettings>, unknown> = {} as Record<StringKeys<PluginSettings>, unknown>;\n    for (const [key, property] of this.properties.entries()) {\n      settings[key as StringKeys<PluginSettings>] = property.get();\n    }\n\n    return settings;\n  }\n}\n"],
  "mappings": "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",
  "names": []
}

260
+ //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../../../src/obsidian/Plugin/PluginSettingsManagerBase.ts"],
  "sourcesContent": ["import type { App } from 'obsidian';\nimport type {\n  Promisable,\n  ReadonlyDeep\n} from 'type-fest';\n\nimport { Notice } from 'obsidian';\n\nimport type { Transformer } from '../../Transformers/Transformer.ts';\nimport type {\n  MaybeReturn,\n  StringKeys\n} from '../../Type.ts';\nimport type { PluginBase } from './PluginBase.ts';\n\nimport { noop } from '../../Function.ts';\nimport { getAllKeys } from '../../Object.ts';\nimport { DateTransformer } from '../../Transformers/DateTransformer.ts';\nimport { DurationTransformer } from '../../Transformers/DurationTransformer.ts';\nimport { GroupTransformer } from '../../Transformers/GroupTransformer.ts';\nimport { SkipPrivatePropertyTransformer } from '../../Transformers/SkipPrivatePropertyTransformer.ts';\n\nconst defaultTransformer = new GroupTransformer([\n  new SkipPrivatePropertyTransformer(),\n  new DateTransformer(),\n  new DurationTransformer()\n]);\n\ntype Validator<T> = (value: T) => Promisable<MaybeReturn<string>>;\n\nabstract class ProxyHandlerBase<PluginSettings extends object> implements ProxyHandler<PluginSettings> {\n  public constructor(protected readonly properties: PropertiesMap<PluginSettings>) {}\n\n  public get(target: PluginSettings, prop: string | symbol): unknown {\n    const record = target as Record<string | symbol, unknown>;\n    if (typeof prop !== 'string') {\n      return record[prop];\n    }\n\n    const property = this.properties.get(prop);\n    if (!property) {\n      return record[prop];\n    }\n\n    return this.getPropertyValue(property);\n  }\n\n  protected abstract getPropertyValue(property: PluginSettingsProperty<unknown>): unknown;\n}\n\nclass EditableSettingsProxyHandler<PluginSettings extends object> extends ProxyHandlerBase<PluginSettings> {\n  private validationPromise = Promise.resolve();\n  public set(target: PluginSettings, prop: string | symbol, value: unknown): boolean {\n    const record = target as Record<string | symbol, unknown>;\n\n    if (typeof prop !== 'string') {\n      record[prop] = value;\n      return true;\n    }\n\n    const property = this.properties.get(prop);\n    if (!property) {\n      record[prop] = value;\n      return true;\n    }\n\n    property.setValue(value);\n    this.validationPromise = this.validationPromise.then(() => property.setValueAndValidate(value));\n\n    return true;\n  }\n\n  protected override getPropertyValue(property: PluginSettingsProperty<unknown>): unknown {\n    return property.currentValue;\n  }\n}\n\n// eslint-disable-next-line @typescript-eslint/no-explicit-any\nclass PropertiesMap<PluginSettings extends object> extends Map<string, PluginSettingsProperty<any>> {\n  public getTyped<PropertyName extends StringKeys<PluginSettings>>(propertyName: PropertyName): PluginSettingsProperty<PluginSettings[PropertyName]> {\n    const property = super.get(propertyName);\n    if (!property) {\n      throw new Error(`Property ${String(propertyName)} not found`);\n    }\n\n    return property as PluginSettingsProperty<PluginSettings[PropertyName]>;\n  }\n\n  public setTyped<PropertyName extends StringKeys<PluginSettings>>(\n    propertyName: PropertyName,\n    value: PluginSettingsProperty<PluginSettings[PropertyName]>\n  ): this {\n    return super.set(propertyName, value);\n  }\n}\n\nclass SafeSettingsProxyHandler<PluginSettings extends object> extends ProxyHandlerBase<PluginSettings> {\n  protected override getPropertyValue(property: PluginSettingsProperty<unknown>): unknown {\n    return property.safeValue;\n  }\n}\n\n/**\n * Base class for managing plugin settings.\n *\n * @typeParam PluginSettings - The type representing the plugin settings object.\n */\nexport abstract class PluginSettingsManagerBase<PluginSettings extends object> {\n  public readonly app: App;\n  public readonly safeSettings: ReadonlyDeep<PluginSettings>;\n\n  private defaultSettings: PluginSettings;\n  private properties: PropertiesMap<PluginSettings>;\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: PluginBase<PluginSettings>) {\n    this.app = plugin.app;\n    this.defaultSettings = this.createDefaultSettings();\n\n    this.addValidators();\n\n    this.properties = new PropertiesMap<PluginSettings>();\n\n    for (const propertyName of getAllKeys(this.defaultSettings)) {\n      this.properties.set(\n        propertyName,\n        new PluginSettingsProperty(propertyName, this.defaultSettings[propertyName], this.validators.get(propertyName) ?? noop)\n      );\n    }\n\n    this.validators.clear();\n\n    this.safeSettings = new Proxy(this.defaultSettings, new SafeSettingsProxyHandler<PluginSettings>(this.properties)) as ReadonlyDeep<PluginSettings>;\n  }\n\n  public async editAndSave(editor: (settings: PluginSettings) => Promisable<void>): Promise<void> {\n    const editableSettings = new Proxy(this.defaultSettings, new EditableSettingsProxyHandler<PluginSettings>(this.properties)) as {\n      validationPromise: Promise<void>;\n    } & PluginSettings;\n    await editor(editableSettings);\n    await editableSettings.validationPromise;\n    await this.saveToFile();\n  }\n\n  public getProperty<PropertyName extends StringKeys<PluginSettings>>(propertyName: PropertyName): PluginSettingsProperty<PluginSettings[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    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('Invalid value type', {\n          propertyName,\n          propertyType: typeof property.defaultValue,\n          value\n        });\n        continue;\n      }\n\n      await property.setValueAndValidate(value);\n    }\n\n    const newJson = JSON.stringify(await this.prepareRecordToSave());\n\n    if (newJson !== originalJson) {\n      await this.saveToFile();\n    }\n  }\n\n  /**\n   * Saves the new plugin settings.\n   *\n   * @returns A 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.plugin.saveData(await this.prepareRecordToSave());\n    await this.plugin.onSaveSettings(this.getSavedSettings(), oldSettings);\n  }\n\n  protected addValidator<PropertyName extends StringKeys<PluginSettings>>(\n    propertyName: PropertyName,\n    validator: Validator<PluginSettings[PropertyName]>\n  ): void {\n    this.validators.set(propertyName, validator);\n  }\n\n  protected addValidators(): void {\n    noop();\n  }\n\n  protected abstract createDefaultSettings(): PluginSettings;\n\n  protected getTransformer(): Transformer {\n    return defaultTransformer;\n  }\n\n  protected onLoadRecord(_record: Record<string, unknown>): Promisable<void> {\n    noop();\n  }\n\n  protected onSavingRecord(_record: Record<string, unknown>): Promisable<void> {\n    noop();\n  }\n\n  private getSavedSettings(): PluginSettings {\n    const savedSettings: Partial<PluginSettings> = {};\n    for (const [propertyName, property] of this.properties.entries()) {\n      savedSettings[propertyName as StringKeys<PluginSettings>] = property.lastSavedValue as PluginSettings[StringKeys<PluginSettings>] | undefined;\n    }\n    const proto = Object.getPrototypeOf(this.defaultSettings) as object;\n    Object.setPrototypeOf(savedSettings, proto);\n    return savedSettings as PluginSettings;\n  }\n\n  private async prepareRecordToSave(): Promise<Record<StringKeys<PluginSettings>, unknown>> {\n    const settings: Record<StringKeys<PluginSettings>, unknown> = {} as Record<StringKeys<PluginSettings>, unknown>;\n    for (const [propertyName, property] of this.properties.entries()) {\n      settings[propertyName as StringKeys<PluginSettings>] = property.currentValue as unknown;\n    }\n\n    await this.onSavingRecord(settings);\n\n    return this.getTransformer().transformObjectRecursively(settings);\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    if (this._currentValue === undefined) {\n      return;\n    }\n\n    this._validationMessage = (await this.validator(this._currentValue) as string | undefined) ?? '';\n    this.showWarning(value);\n  }\n\n  private showWarning(value?: T): void {\n    if (!this._validationMessage) {\n      return;\n    }\n\n    const warningMessage = `Could not set plugin setting: ${this.propertyName}. Using default value instead.`;\n    new Notice(warningMessage);\n    console.warn(warningMessage, {\n      defaultValue: this.defaultValue,\n      propertyName: this.propertyName,\n      validationMessage: this._validationMessage,\n      value\n    });\n  }\n}\n"],
  "mappings": "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",
  "names": []
}

@@ -3,45 +3,48 @@
3
3
  * This module defines a base class for creating plugin setting tabs in Obsidian.
4
4
  * It provides a utility method to bind value components to plugin settings and handle changes.
5
5
  */
6
- import type { ConditionalKeys, Promisable, WritableDeep } from 'type-fest';
6
+ import type { ConditionalKeys, Promisable } from 'type-fest';
7
7
  import { PluginSettingTab } from 'obsidian';
8
- import type { StringKeys } from '../../Object.mjs';
8
+ import type { StringKeys } from '../../Type.mjs';
9
9
  import type { ValueComponentWithChangeTracking } from '../Components/ValueComponentWithChangeTracking.mjs';
10
+ import type { ValidationMessageHolder } from '../ValidationMessage.mjs';
10
11
  import type { PluginBase } from './PluginBase.mjs';
11
12
  /**
12
13
  * Options for binding a value component to a plugin setting.
13
14
  */
14
- export interface BindOptions {
15
+ export interface BindOptions<T> {
15
16
  /**
16
17
  * A callback function that is called when the value of the component changes.
17
18
  */
18
- onChanged?(): Promisable<void>;
19
+ onChanged?(newValue: T | undefined, oldValue: T): Promisable<void>;
19
20
  /**
20
- * If true, shows the validation message when the component value is invalid. Default is `true`.
21
+ * Whether to reset the setting when the component value is empty. Default is `true`.
22
+ * Applicable only to text-based components.
23
+ */
24
+ shouldResetSettingWhenComponentIsEmpty?: boolean;
25
+ /**
26
+ * Whether to show the validation message when the component value is invalid. Default is `true`.
21
27
  */
22
28
  shouldShowValidationMessage?: boolean;
23
29
  }
24
30
  /**
25
31
  * Extended options for binding a value component to a plugin setting.
26
32
  */
27
- export interface BindOptionsExtended<PluginSettings extends object, UIValue, Property extends StringKeys<PluginSettings>> extends BindOptions {
33
+ export interface BindOptionsExtended<PluginSettings extends object, UIValue, PropertyName extends StringKeys<PluginSettings>> extends BindOptions<PluginSettings[PropertyName]> {
28
34
  /**
29
35
  * Converts the UI component's value back to the plugin settings value.
30
36
  * @param uiValue - The value of the UI component.
31
37
  * @returns The value to set on the plugin settings.
32
38
  */
33
- componentToPluginSettingsValueConverter: (uiValue: UIValue) => PluginSettings[Property] | ValidationMessageHolder;
39
+ componentToPluginSettingsValueConverter: (uiValue: UIValue) => PluginSettings[PropertyName] | ValidationMessageHolder;
34
40
  /**
35
41
  * Converts the plugin settings value to the value used by the UI component.
36
42
  * @param pluginSettingsValue - The value of the property in the plugin settings.
37
43
  * @returns The value to set on the UI component.
38
44
  */
39
- pluginSettingsToComponentValueConverter: (pluginSettingsValue: PluginSettings[Property]) => UIValue;
40
- }
41
- type ExtractPluginSettings<T extends PluginBase<any>> = WritableDeep<T['settings']>;
42
- interface ValidationMessageHolder {
43
- validationMessage: string;
45
+ pluginSettingsToComponentValueConverter: (pluginSettingsValue: PluginSettings[PropertyName]) => UIValue;
44
46
  }
47
+ type ExtractPluginSettings<Plugin extends PluginBase<any>> = Plugin['__pluginSettingsType'];
45
48
  /**
46
49
  * Base class for creating plugin settings tabs in Obsidian.
47
50
  * Provides a method for binding value components to plugin settings and handling changes.
@@ -57,23 +60,23 @@ export declare abstract class PluginSettingsTabBase<TPlugin extends PluginBase<a
57
60
  * @typeParam UIValue - The type of the value of the UI component.
58
61
  * @typeParam TValueComponent - The type of the value component.
59
62
  * @param valueComponent - The value component to bind.
60
- * @param property - The property of the plugin settings to bind to.
63
+ * @param propertyName - The property of the plugin settings to bind to.
61
64
  * @param options - The options for binding the value component.
62
65
  * @returns The value component.
63
66
  */
64
- bind<UIValue, TValueComponent>(valueComponent: TValueComponent & ValueComponentWithChangeTracking<UIValue>, property: ConditionalKeys<ExtractPluginSettings<TPlugin>, UIValue>, options?: BindOptions): TValueComponent;
67
+ bind<UIValue, TValueComponent>(valueComponent: TValueComponent & ValueComponentWithChangeTracking<UIValue>, propertyName: ConditionalKeys<ExtractPluginSettings<TPlugin>, UIValue>, options?: BindOptions<UIValue>): TValueComponent;
65
68
  /**
66
69
  * Binds a value component to a plugin setting.
67
70
  *
68
71
  * @typeParam UIValue - The type of the value of the UI component.
69
72
  * @typeParam TValueComponent - The type of the value component.
70
- * @typeParam Property - The property of the plugin settings to bind to.
73
+ * @typeParam PropertyName - The property name of the plugin settings to bind to.
71
74
  * @param valueComponent - The value component to bind.
72
- * @param property - The property of the plugin settings to bind to.
75
+ * @param propertyName - The property name of the plugin settings to bind to.
73
76
  * @param options - The options for binding the value component.
74
77
  * @returns The value component.
75
78
  */
76
- bind<UIValue, TValueComponent, Property extends StringKeys<ExtractPluginSettings<TPlugin>>>(valueComponent: TValueComponent & ValueComponentWithChangeTracking<UIValue>, property: Property, options: BindOptionsExtended<ExtractPluginSettings<TPlugin>, UIValue, Property>): TValueComponent;
79
+ bind<UIValue, TValueComponent, PropertyName extends StringKeys<ExtractPluginSettings<TPlugin>>>(valueComponent: TValueComponent & ValueComponentWithChangeTracking<UIValue>, propertyName: PropertyName, options: BindOptionsExtended<ExtractPluginSettings<TPlugin>, UIValue, PropertyName>): TValueComponent;
77
80
  hide(): void;
78
81
  }
79
82
  export {};