obsidian-dev-utils 22.1.0 → 22.1.1-beta.10
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 +59 -0
- package/dist/dprint.json +1 -0
- package/dist/lib/cjs/Async.cjs +2 -2
- package/dist/lib/cjs/Library.cjs +1 -1
- package/dist/lib/cjs/Object.cjs +1 -1
- package/dist/lib/cjs/Object.d.cts +12 -0
- package/dist/lib/cjs/ScriptUtils/CliUtils.cjs +1 -2
- package/dist/lib/cjs/ScriptUtils/CliUtils.d.cts +3 -2
- package/dist/lib/cjs/ScriptUtils/cli.cjs +1 -1
- package/dist/lib/cjs/String.cjs +1 -1
- package/dist/lib/cjs/String.d.cts +2 -1
- package/dist/lib/cjs/codemirror/StateFieldSpec.cjs +1 -1
- package/dist/lib/cjs/codemirror/StateFieldSpec.d.cts +4 -4
- package/dist/lib/cjs/obsidian/Callout.cjs +1 -1
- package/dist/lib/cjs/obsidian/Callout.d.cts +2 -1
- package/dist/lib/cjs/obsidian/Components/MultipleTextComponent.cjs +1 -1
- package/dist/lib/cjs/obsidian/Components/MultipleTextComponent.d.cts +2 -1
- package/dist/lib/cjs/obsidian/Components/PlaceholderComponent.cjs +38 -0
- package/dist/lib/cjs/obsidian/Components/PlaceholderComponent.d.cts +19 -0
- package/dist/lib/cjs/obsidian/Components/TypedTextComponent.cjs +1 -1
- package/dist/lib/cjs/obsidian/Components/TypedTextComponent.d.cts +2 -1
- package/dist/lib/cjs/obsidian/Components/index.cjs +4 -1
- package/dist/lib/cjs/obsidian/Components/index.d.cts +1 -0
- package/dist/lib/cjs/obsidian/FileManager.cjs +1 -1
- package/dist/lib/cjs/obsidian/FileManager.d.cts +2 -1
- package/dist/lib/cjs/obsidian/Link.cjs +1 -1
- package/dist/lib/cjs/obsidian/Link.d.cts +4 -3
- package/dist/lib/cjs/obsidian/Modals/Prompt.cjs +1 -1
- package/dist/lib/cjs/obsidian/Modals/Prompt.d.cts +2 -1
- package/dist/lib/cjs/obsidian/Plugin/PluginBase.cjs +15 -31
- package/dist/lib/cjs/obsidian/Plugin/PluginBase.d.cts +15 -27
- package/dist/lib/cjs/obsidian/Plugin/PluginSettingsManagerBase.cjs +279 -0
- package/dist/lib/cjs/obsidian/Plugin/PluginSettingsManagerBase.d.cts +60 -0
- package/dist/lib/cjs/obsidian/Plugin/PluginSettingsTabBase.cjs +50 -53
- package/dist/lib/cjs/obsidian/Plugin/PluginSettingsTabBase.d.cts +14 -35
- package/dist/lib/cjs/obsidian/Plugin/index.cjs +4 -7
- package/dist/lib/cjs/obsidian/Plugin/index.d.cts +1 -2
- package/dist/lib/cjs/obsidian/{Plugin/EmptySettings.cjs → ValidationMessage.cjs} +8 -8
- package/dist/lib/cjs/obsidian/ValidationMessage.d.cts +16 -0
- package/dist/lib/cjs/obsidian/Vault.cjs +2 -2
- package/dist/lib/cjs/obsidian/index.cjs +4 -1
- package/dist/lib/cjs/obsidian/index.d.cts +1 -0
- package/dist/lib/esm/Async.mjs +2 -2
- package/dist/lib/esm/Library.mjs +1 -1
- package/dist/lib/esm/Object.d.mts +12 -0
- package/dist/lib/esm/Object.mjs +1 -1
- package/dist/lib/esm/ScriptUtils/CliUtils.d.mts +3 -2
- package/dist/lib/esm/ScriptUtils/CliUtils.mjs +1 -2
- package/dist/lib/esm/ScriptUtils/cli.mjs +1 -1
- package/dist/lib/esm/String.d.mts +2 -1
- package/dist/lib/esm/String.mjs +1 -1
- package/dist/lib/esm/codemirror/StateFieldSpec.d.mts +4 -4
- package/dist/lib/esm/obsidian/Callout.d.mts +2 -1
- package/dist/lib/esm/obsidian/Callout.mjs +1 -1
- package/dist/lib/esm/obsidian/Components/MultipleTextComponent.d.mts +2 -1
- package/dist/lib/esm/obsidian/Components/MultipleTextComponent.mjs +1 -1
- package/dist/lib/esm/obsidian/Components/PlaceholderComponent.d.mts +19 -0
- package/dist/lib/esm/obsidian/Components/PlaceholderComponent.mjs +14 -0
- package/dist/lib/esm/obsidian/Components/TypedTextComponent.d.mts +2 -1
- package/dist/lib/esm/obsidian/Components/TypedTextComponent.mjs +1 -1
- package/dist/lib/esm/obsidian/Components/index.d.mts +1 -0
- package/dist/lib/esm/obsidian/Components/index.mjs +3 -1
- package/dist/lib/esm/obsidian/FileManager.d.mts +2 -1
- package/dist/lib/esm/obsidian/FileManager.mjs +1 -1
- package/dist/lib/esm/obsidian/Link.d.mts +4 -3
- package/dist/lib/esm/obsidian/Link.mjs +1 -1
- package/dist/lib/esm/obsidian/Modals/Prompt.d.mts +2 -1
- package/dist/lib/esm/obsidian/Modals/Prompt.mjs +1 -1
- package/dist/lib/esm/obsidian/Plugin/PluginBase.d.mts +15 -27
- package/dist/lib/esm/obsidian/Plugin/PluginBase.mjs +19 -32
- package/dist/lib/esm/obsidian/Plugin/PluginSettingsManagerBase.d.mts +60 -0
- package/dist/lib/esm/obsidian/Plugin/PluginSettingsManagerBase.mjs +254 -0
- package/dist/lib/esm/obsidian/Plugin/PluginSettingsTabBase.d.mts +14 -35
- package/dist/lib/esm/obsidian/Plugin/PluginSettingsTabBase.mjs +51 -57
- package/dist/lib/esm/obsidian/Plugin/index.d.mts +1 -2
- package/dist/lib/esm/obsidian/Plugin/index.mjs +3 -5
- package/dist/lib/esm/obsidian/ValidationMessage.d.mts +16 -0
- package/dist/lib/esm/obsidian/ValidationMessage.mjs +14 -0
- package/dist/lib/esm/obsidian/Vault.mjs +2 -2
- package/dist/lib/esm/obsidian/index.d.mts +1 -0
- package/dist/lib/esm/obsidian/index.mjs +3 -1
- package/obsidian/Components/PlaceholderComponent/package.json +6 -0
- package/obsidian/Plugin/{PluginSettingsBase → PluginSettingsManagerBase}/package.json +3 -3
- package/obsidian/ValidationMessage/package.json +6 -0
- package/package.json +7 -7
- package/dist/lib/cjs/obsidian/Plugin/EmptySettings.d.cts +0 -14
- package/dist/lib/cjs/obsidian/Plugin/PluginSettingsBase.cjs +0 -92
- package/dist/lib/cjs/obsidian/Plugin/PluginSettingsBase.d.cts +0 -31
- package/dist/lib/esm/obsidian/Plugin/EmptySettings.d.mts +0 -14
- package/dist/lib/esm/obsidian/Plugin/EmptySettings.mjs +0 -14
- package/dist/lib/esm/obsidian/Plugin/PluginSettingsBase.d.mts +0 -31
- package/dist/lib/esm/obsidian/Plugin/PluginSettingsBase.mjs +0 -68
- package/obsidian/Plugin/EmptySettings/package.json +0 -6
@@ -0,0 +1,279 @@
|
|
1
|
+
/*
|
2
|
+
THIS IS A GENERATED/BUNDLED FILE BY ESBUILD
|
3
|
+
if you want to view the source, please visit the github repository of this plugin
|
4
|
+
*/
|
5
|
+
|
6
|
+
(function initCjs(){const globalThisRecord=globalThis;globalThisRecord["__name"]??=name;const originalRequire=require;if(originalRequire&&!originalRequire.__isPatched){require=Object.assign(id=>requirePatched(id),originalRequire,{__isPatched:true})}const newFuncs={__extractDefault:__name(()=>extractDefault,"__extractDefault"),process:__name(()=>{const browserProcess={browser:true,cwd:__name(()=>"/","cwd"),env:{},platform:"android"};return browserProcess},"process")};for(const key of Object.keys(newFuncs)){globalThisRecord[key]??=newFuncs[key]?.()}function name(obj){return obj}__name(name,"name");function extractDefault(module){return module&&module.__esModule&&module.default?module.default:module}__name(extractDefault,"extractDefault");function requirePatched(id){const module=originalRequire?.(id);if(module){return extractDefault(module)}if(id==="process"||id==="node:process"){console.error(`Module not found: ${id}. Fake process object is returned instead.`);return globalThis.process}console.error(`Module not found: ${id}. Empty object is returned instead.`);return{}}__name(requirePatched,"requirePatched")})();
|
7
|
+
|
8
|
+
"use strict";
|
9
|
+
var __defProp = Object.defineProperty;
|
10
|
+
var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
|
11
|
+
var __getOwnPropNames = Object.getOwnPropertyNames;
|
12
|
+
var __hasOwnProp = Object.prototype.hasOwnProperty;
|
13
|
+
var __export = (target, all) => {
|
14
|
+
for (var name in all)
|
15
|
+
__defProp(target, name, { get: all[name], enumerable: true });
|
16
|
+
};
|
17
|
+
var __copyProps = (to, from, except, desc) => {
|
18
|
+
if (from && typeof from === "object" || typeof from === "function") {
|
19
|
+
for (let key of __getOwnPropNames(from))
|
20
|
+
if (!__hasOwnProp.call(to, key) && key !== except)
|
21
|
+
__defProp(to, key, { get: () => from[key], enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable });
|
22
|
+
}
|
23
|
+
return to;
|
24
|
+
};
|
25
|
+
var __toCommonJS = (mod) => __copyProps(__defProp({}, "__esModule", { value: true }), mod);
|
26
|
+
var PluginSettingsManagerBase_exports = {};
|
27
|
+
__export(PluginSettingsManagerBase_exports, {
|
28
|
+
PluginSettingsManagerBase: () => PluginSettingsManagerBase,
|
29
|
+
PluginSettingsProperty: () => PluginSettingsProperty
|
30
|
+
});
|
31
|
+
module.exports = __toCommonJS(PluginSettingsManagerBase_exports);
|
32
|
+
var import_obsidian = require('obsidian');
|
33
|
+
var import_Function = require('../../Function.cjs');
|
34
|
+
var import_DateTransformer = require('../../Transformers/DateTransformer.cjs');
|
35
|
+
var import_DurationTransformer = require('../../Transformers/DurationTransformer.cjs');
|
36
|
+
var import_GroupTransformer = require('../../Transformers/GroupTransformer.cjs');
|
37
|
+
var import_SkipPrivatePropertyTransformer = require('../../Transformers/SkipPrivatePropertyTransformer.cjs');
|
38
|
+
const defaultTransformer = new import_GroupTransformer.GroupTransformer([
|
39
|
+
new import_SkipPrivatePropertyTransformer.SkipPrivatePropertyTransformer(),
|
40
|
+
new import_DateTransformer.DateTransformer(),
|
41
|
+
new import_DurationTransformer.DurationTransformer()
|
42
|
+
]);
|
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
|
+
if (typeof prop !== "string") {
|
64
|
+
record[prop] = value;
|
65
|
+
return true;
|
66
|
+
}
|
67
|
+
const property = this.properties.get(prop);
|
68
|
+
if (!property) {
|
69
|
+
record[prop] = value;
|
70
|
+
return true;
|
71
|
+
}
|
72
|
+
property.setValue(value);
|
73
|
+
this.validationPromise = this.validationPromise.then(() => property.setValueAndValidate(value));
|
74
|
+
return true;
|
75
|
+
}
|
76
|
+
getPropertyValue(property) {
|
77
|
+
return property.getModifiedOrDefaultValue();
|
78
|
+
}
|
79
|
+
}
|
80
|
+
class PropertiesMap extends Map {
|
81
|
+
getTyped(propertyName) {
|
82
|
+
const property = super.get(propertyName);
|
83
|
+
if (!property) {
|
84
|
+
throw new Error(`Property ${String(propertyName)} not found`);
|
85
|
+
}
|
86
|
+
return property;
|
87
|
+
}
|
88
|
+
setTyped(propertyName, value) {
|
89
|
+
return super.set(propertyName, value);
|
90
|
+
}
|
91
|
+
}
|
92
|
+
class SafeSettingsProxyHandler extends ProxyHandlerBase {
|
93
|
+
getPropertyValue(property) {
|
94
|
+
return property.getSafeValue();
|
95
|
+
}
|
96
|
+
}
|
97
|
+
class PluginSettingsManagerBase {
|
98
|
+
constructor(plugin) {
|
99
|
+
this.plugin = plugin;
|
100
|
+
this.defaultSettings = this.createDefaultSettings();
|
101
|
+
this.addValidators();
|
102
|
+
this.properties = new PropertiesMap();
|
103
|
+
for (const propertyName of Object.keys(this.defaultSettings)) {
|
104
|
+
this.properties.set(
|
105
|
+
propertyName,
|
106
|
+
new PluginSettingsProperty(propertyName, this.defaultSettings[propertyName], this.validators.get(propertyName) ?? import_Function.noop)
|
107
|
+
);
|
108
|
+
}
|
109
|
+
this.validators.clear();
|
110
|
+
this.safeSettings = new Proxy(this.defaultSettings, new SafeSettingsProxyHandler(this.properties));
|
111
|
+
}
|
112
|
+
safeSettings;
|
113
|
+
defaultSettings;
|
114
|
+
properties;
|
115
|
+
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
116
|
+
validators = /* @__PURE__ */ new Map();
|
117
|
+
async editAndSave(editor) {
|
118
|
+
const editableSettings = new Proxy(this.defaultSettings, new EditableSettingsProxyHandler(this.properties));
|
119
|
+
await editor(editableSettings);
|
120
|
+
await editableSettings.validationPromise;
|
121
|
+
await this.saveToFile();
|
122
|
+
}
|
123
|
+
getProperty(propertyName) {
|
124
|
+
return this.properties.getTyped(propertyName);
|
125
|
+
}
|
126
|
+
async loadFromFile() {
|
127
|
+
for (const property of this.properties.values()) {
|
128
|
+
property.clear();
|
129
|
+
}
|
130
|
+
const data = await this.plugin.loadData();
|
131
|
+
if (data === void 0 || data === null) {
|
132
|
+
return;
|
133
|
+
}
|
134
|
+
if (typeof data !== "object" || Array.isArray(data)) {
|
135
|
+
const type = Array.isArray(data) ? "Array" : typeof data;
|
136
|
+
console.error(`Invalid data type. Expected Object, got: ${type}`);
|
137
|
+
return;
|
138
|
+
}
|
139
|
+
let record = data;
|
140
|
+
const originalJson = JSON.stringify(record);
|
141
|
+
record = this.getTransformer().transformObjectRecursively(record);
|
142
|
+
await this.prepareRecord(record);
|
143
|
+
for (const [propertyName, value] of Object.entries(record)) {
|
144
|
+
const property = this.properties.get(propertyName);
|
145
|
+
if (!property) {
|
146
|
+
console.warn(`Unknown property: ${propertyName}`);
|
147
|
+
continue;
|
148
|
+
}
|
149
|
+
if (typeof value !== typeof property.defaultValue) {
|
150
|
+
console.warn("Invalid value type", {
|
151
|
+
propertyName,
|
152
|
+
propertyType: typeof property.defaultValue,
|
153
|
+
value
|
154
|
+
});
|
155
|
+
continue;
|
156
|
+
}
|
157
|
+
await property.setValueAndValidate(value);
|
158
|
+
}
|
159
|
+
const newJson = JSON.stringify(this.prepareRecordToSave());
|
160
|
+
if (newJson !== originalJson) {
|
161
|
+
await this.saveToFile();
|
162
|
+
}
|
163
|
+
}
|
164
|
+
/**
|
165
|
+
* Saves the new plugin settings.
|
166
|
+
*
|
167
|
+
* @returns A promise that resolves when the settings are saved.
|
168
|
+
*/
|
169
|
+
async saveToFile() {
|
170
|
+
const oldSettings = this.getSavedSettings();
|
171
|
+
let hasChanges = false;
|
172
|
+
for (const property of this.properties.values()) {
|
173
|
+
hasChanges ||= property.save();
|
174
|
+
}
|
175
|
+
if (!hasChanges) {
|
176
|
+
return;
|
177
|
+
}
|
178
|
+
await this.plugin.saveData(this.prepareRecordToSave());
|
179
|
+
await this.plugin.onSaveSettings(this.getSavedSettings(), oldSettings);
|
180
|
+
}
|
181
|
+
addValidator(propertyName, validator) {
|
182
|
+
this.validators.set(propertyName, validator);
|
183
|
+
}
|
184
|
+
addValidators() {
|
185
|
+
(0, import_Function.noop)();
|
186
|
+
}
|
187
|
+
getTransformer() {
|
188
|
+
return defaultTransformer;
|
189
|
+
}
|
190
|
+
prepareRecord(_record) {
|
191
|
+
(0, import_Function.noop)();
|
192
|
+
}
|
193
|
+
getSavedSettings() {
|
194
|
+
const savedSettings = {};
|
195
|
+
for (const [propertyName, property] of this.properties.entries()) {
|
196
|
+
savedSettings[propertyName] = property.getLastSavedValue();
|
197
|
+
}
|
198
|
+
const proto = Object.getPrototypeOf(this.defaultSettings);
|
199
|
+
Object.setPrototypeOf(savedSettings, proto);
|
200
|
+
return savedSettings;
|
201
|
+
}
|
202
|
+
prepareRecordToSave() {
|
203
|
+
const settings = {};
|
204
|
+
for (const [propertyName, property] of this.properties.entries()) {
|
205
|
+
settings[propertyName] = property.getModifiedValue();
|
206
|
+
}
|
207
|
+
return this.getTransformer().transformObjectRecursively(settings);
|
208
|
+
}
|
209
|
+
}
|
210
|
+
class PluginSettingsProperty {
|
211
|
+
constructor(propertyName, defaultValue, validator) {
|
212
|
+
this.propertyName = propertyName;
|
213
|
+
this.defaultValue = defaultValue;
|
214
|
+
this.validator = validator;
|
215
|
+
}
|
216
|
+
get validationMessage() {
|
217
|
+
return this._validationMessage;
|
218
|
+
}
|
219
|
+
_validationMessage = "";
|
220
|
+
lastSavedValue;
|
221
|
+
modifiedValue;
|
222
|
+
clear() {
|
223
|
+
this.modifiedValue = void 0;
|
224
|
+
this._validationMessage = "";
|
225
|
+
}
|
226
|
+
getLastSavedValue() {
|
227
|
+
return this.lastSavedValue;
|
228
|
+
}
|
229
|
+
getModifiedOrDefaultValue() {
|
230
|
+
return this.modifiedValue ?? this.defaultValue;
|
231
|
+
}
|
232
|
+
getModifiedValue() {
|
233
|
+
return this.modifiedValue;
|
234
|
+
}
|
235
|
+
getSafeValue() {
|
236
|
+
return this._validationMessage ? this.defaultValue : this.getModifiedOrDefaultValue();
|
237
|
+
}
|
238
|
+
save() {
|
239
|
+
if (this.lastSavedValue === this.modifiedValue) {
|
240
|
+
return false;
|
241
|
+
}
|
242
|
+
this.lastSavedValue = this.modifiedValue;
|
243
|
+
return true;
|
244
|
+
}
|
245
|
+
setValidationMessage(validationMessage) {
|
246
|
+
this._validationMessage = validationMessage;
|
247
|
+
this.showWarning();
|
248
|
+
}
|
249
|
+
setValue(value) {
|
250
|
+
this.modifiedValue = value;
|
251
|
+
}
|
252
|
+
async setValueAndValidate(value) {
|
253
|
+
this.setValue(value);
|
254
|
+
if (this.modifiedValue === void 0) {
|
255
|
+
return;
|
256
|
+
}
|
257
|
+
this._validationMessage = await this.validator(this.modifiedValue) ?? "";
|
258
|
+
this.showWarning(value);
|
259
|
+
}
|
260
|
+
showWarning(value) {
|
261
|
+
if (!this._validationMessage) {
|
262
|
+
return;
|
263
|
+
}
|
264
|
+
const warningMessage = `Could not set plugin setting: ${this.propertyName}. Using default value instead.`;
|
265
|
+
new import_obsidian.Notice(warningMessage);
|
266
|
+
console.warn(warningMessage, {
|
267
|
+
defaultValue: this.defaultValue,
|
268
|
+
propertyName: this.propertyName,
|
269
|
+
validationMessage: this._validationMessage,
|
270
|
+
value
|
271
|
+
});
|
272
|
+
}
|
273
|
+
}
|
274
|
+
// Annotate the CommonJS export names for ESM import in node:
|
275
|
+
0 && (module.exports = {
|
276
|
+
PluginSettingsManagerBase,
|
277
|
+
PluginSettingsProperty
|
278
|
+
});
|
279
|
+
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../../../src/obsidian/Plugin/PluginSettingsManagerBase.ts"],
  "sourcesContent": ["import type {\n  Promisable,\n  ReadonlyDeep\n} from 'type-fest';\n\nimport { Notice } from 'obsidian';\n\nimport type {\n  MaybeReturn,\n  StringKeys\n} from '../../Object.ts';\nimport type { Transformer } from '../../Transformers/Transformer.ts';\nimport type { PluginBase } from './PluginBase.ts';\n\nimport { noop } 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\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.getModifiedOrDefaultValue();\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.getSafeValue();\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 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(private plugin: PluginBase<PluginSettings>) {\n    this.defaultSettings = this.createDefaultSettings();\n\n    this.addValidators();\n\n    this.properties = new PropertiesMap<PluginSettings>();\n\n    for (const propertyName of Object.keys(this.defaultSettings) as StringKeys<PluginSettings>[]) {\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.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    const originalJson = JSON.stringify(record);\n    record = this.getTransformer().transformObjectRecursively(record);\n    await this.prepareRecord(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(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(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 prepareRecord(_record: Record<string, unknown>): Promisable<void> {\n    noop();\n  }\n\n  private getSavedSettings(): Partial<PluginSettings> {\n    const savedSettings: Partial<PluginSettings> = {};\n    for (const [propertyName, property] of this.properties.entries()) {\n      savedSettings[propertyName as StringKeys<PluginSettings>] = property.getLastSavedValue() as PluginSettings[StringKeys<PluginSettings>] | undefined;\n    }\n    const proto = Object.getPrototypeOf(this.defaultSettings) as object;\n    Object.setPrototypeOf(savedSettings, proto);\n    return savedSettings;\n  }\n\n  private prepareRecordToSave(): 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.getModifiedValue() as unknown;\n    }\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 validationMessage(): string {\n    return this._validationMessage;\n  }\n\n  private _validationMessage = '';\n  private lastSavedValue: T | undefined;\n  private modifiedValue: T | undefined;\n\n  public constructor(private readonly propertyName: string, public readonly defaultValue: T, private readonly validator: Validator<T>) {}\n\n  public clear(): void {\n    this.modifiedValue = undefined;\n    this._validationMessage = '';\n  }\n\n  public getLastSavedValue(): T | undefined {\n    return this.lastSavedValue;\n  }\n\n  public getModifiedOrDefaultValue(): T {\n    return this.modifiedValue ?? this.defaultValue;\n  }\n\n  public getModifiedValue(): T | undefined {\n    return this.modifiedValue;\n  }\n\n  public getSafeValue(): T {\n    return this._validationMessage ? this.defaultValue : this.getModifiedOrDefaultValue();\n  }\n\n  public save(): boolean {\n    if (this.lastSavedValue === this.modifiedValue) {\n      return false;\n    }\n\n    this.lastSavedValue = this.modifiedValue;\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 | undefined): void {\n    this.modifiedValue = value;\n  }\n\n  public async setValueAndValidate(value: T): Promise<void> {\n    this.setValue(value);\n    if (this.modifiedValue === undefined) {\n      return;\n    }\n\n    this._validationMessage = (await this.validator(this.modifiedValue) 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": []
}

|
@@ -0,0 +1,60 @@
|
|
1
|
+
import type { Promisable, ReadonlyDeep } from 'type-fest';
|
2
|
+
import type { MaybeReturn, StringKeys } from '../../Object.cjs';
|
3
|
+
import type { Transformer } from '../../Transformers/Transformer.cjs';
|
4
|
+
import type { PluginBase } from './PluginBase.cjs';
|
5
|
+
type Validator<T> = (value: T) => Promisable<MaybeReturn<string>>;
|
6
|
+
/**
|
7
|
+
* Base class for managing plugin settings.
|
8
|
+
*
|
9
|
+
* @typeParam PluginSettings - The type representing the plugin settings object.
|
10
|
+
*/
|
11
|
+
export declare abstract class PluginSettingsManagerBase<PluginSettings extends object> {
|
12
|
+
private plugin;
|
13
|
+
readonly safeSettings: ReadonlyDeep<PluginSettings>;
|
14
|
+
private defaultSettings;
|
15
|
+
private properties;
|
16
|
+
private validators;
|
17
|
+
constructor(plugin: PluginBase<PluginSettings>);
|
18
|
+
editAndSave(editor: (settings: PluginSettings) => Promisable<void>): Promise<void>;
|
19
|
+
getProperty<PropertyName extends StringKeys<PluginSettings>>(propertyName: PropertyName): PluginSettingsProperty<PluginSettings[PropertyName]>;
|
20
|
+
loadFromFile(): Promise<void>;
|
21
|
+
/**
|
22
|
+
* Saves the new plugin settings.
|
23
|
+
*
|
24
|
+
* @returns A promise that resolves when the settings are saved.
|
25
|
+
*/
|
26
|
+
saveToFile(): Promise<void>;
|
27
|
+
protected addValidator<PropertyName extends StringKeys<PluginSettings>>(propertyName: PropertyName, validator: Validator<PluginSettings[PropertyName]>): void;
|
28
|
+
protected addValidators(): void;
|
29
|
+
protected abstract createDefaultSettings(): PluginSettings;
|
30
|
+
protected getTransformer(): Transformer;
|
31
|
+
protected prepareRecord(_record: Record<string, unknown>): Promisable<void>;
|
32
|
+
private getSavedSettings;
|
33
|
+
private prepareRecordToSave;
|
34
|
+
}
|
35
|
+
/**
|
36
|
+
* A property of a plugin settings.
|
37
|
+
*
|
38
|
+
* @typeParam T - The type of the property.
|
39
|
+
*/
|
40
|
+
export declare class PluginSettingsProperty<T> {
|
41
|
+
private readonly propertyName;
|
42
|
+
readonly defaultValue: T;
|
43
|
+
private readonly validator;
|
44
|
+
get validationMessage(): string;
|
45
|
+
private _validationMessage;
|
46
|
+
private lastSavedValue;
|
47
|
+
private modifiedValue;
|
48
|
+
constructor(propertyName: string, defaultValue: T, validator: Validator<T>);
|
49
|
+
clear(): void;
|
50
|
+
getLastSavedValue(): T | undefined;
|
51
|
+
getModifiedOrDefaultValue(): T;
|
52
|
+
getModifiedValue(): T | undefined;
|
53
|
+
getSafeValue(): T;
|
54
|
+
save(): boolean;
|
55
|
+
setValidationMessage(validationMessage: string): void;
|
56
|
+
setValue(value: T | undefined): void;
|
57
|
+
setValueAndValidate(value: T): Promise<void>;
|
58
|
+
private showWarning;
|
59
|
+
}
|
60
|
+
export {};
|
@@ -32,7 +32,9 @@ var import_obsidian = require('obsidian');
|
|
32
32
|
var import_Async = require('../../Async.cjs');
|
33
33
|
var import_CssClass = require('../../CssClass.cjs');
|
34
34
|
var import_Function = require('../../Function.cjs');
|
35
|
+
var import_PlaceholderComponent = require('../Components/PlaceholderComponent.cjs');
|
35
36
|
var import_ValidatorComponent = require('../Components/ValidatorComponent.cjs');
|
37
|
+
var import_ValidationMessage = require('../ValidationMessage.cjs');
|
36
38
|
var import_PluginId = require('./PluginId.cjs');
|
37
39
|
class PluginSettingsTabBase extends import_obsidian.PluginSettingTab {
|
38
40
|
constructor(plugin) {
|
@@ -40,82 +42,77 @@ class PluginSettingsTabBase extends import_obsidian.PluginSettingTab {
|
|
40
42
|
this.plugin = plugin;
|
41
43
|
this.containerEl.addClass(import_CssClass.CssClass.LibraryName, (0, import_PluginId.getPluginId)(), import_CssClass.CssClass.PluginSettingsTab);
|
42
44
|
}
|
43
|
-
validatorsMap = /* @__PURE__ */ new WeakMap();
|
44
45
|
/**
|
45
46
|
* Binds a value component to a plugin setting.
|
46
47
|
*
|
47
48
|
* @typeParam UIValue - The type of the value of the UI component.
|
48
49
|
* @typeParam TValueComponent - The type of the value component.
|
49
|
-
* @typeParam
|
50
|
+
* @typeParam PropertyName - The property name of the plugin settings to bind to.
|
50
51
|
* @param valueComponent - The value component to bind.
|
51
|
-
* @param
|
52
|
+
* @param propertyName - The property name of the plugin settings to bind to.
|
52
53
|
* @param options - The options for binding the value component.
|
53
54
|
* @returns The value component.
|
54
55
|
*/
|
55
|
-
bind(valueComponent,
|
56
|
+
bind(valueComponent, propertyName, options) {
|
56
57
|
const DEFAULT_OPTIONS = {
|
57
58
|
// eslint-disable-next-line @typescript-eslint/no-unsafe-return
|
58
|
-
componentToPluginSettingsValueConverter: (
|
59
|
+
componentToPluginSettingsValueConverter: (value2) => value2,
|
59
60
|
onChanged: import_Function.noop,
|
60
|
-
|
61
|
-
|
62
|
-
shouldAutoSave: true,
|
63
|
-
shouldShowValidationMessage: true,
|
64
|
-
valueValidator: import_Function.noop
|
61
|
+
pluginSettingsToComponentValueConverter: (value2) => value2,
|
62
|
+
shouldShowValidationMessage: true
|
65
63
|
};
|
66
64
|
const optionsExt = { ...DEFAULT_OPTIONS, ...options };
|
67
|
-
const pluginSettingsFn = () => optionsExt.pluginSettings ?? this.plugin.settingsClone;
|
68
65
|
const validatorElement = (0, import_ValidatorComponent.getValidatorComponent)(valueComponent)?.validatorEl;
|
69
|
-
const
|
70
|
-
|
71
|
-
|
72
|
-
|
73
|
-
|
74
|
-
|
75
|
-
|
76
|
-
|
77
|
-
|
78
|
-
|
79
|
-
|
80
|
-
|
81
|
-
|
82
|
-
|
66
|
+
const property = this.plugin.settingsManager.getProperty(propertyName);
|
67
|
+
let value = property.getModifiedValue();
|
68
|
+
if (value === void 0 && !(0, import_PlaceholderComponent.isPlaceholderComponent)(valueComponent)) {
|
69
|
+
value = property.defaultValue;
|
70
|
+
property.setValue(value);
|
71
|
+
}
|
72
|
+
if (value !== void 0) {
|
73
|
+
valueComponent.setValue(optionsExt.pluginSettingsToComponentValueConverter(value));
|
74
|
+
}
|
75
|
+
valueComponent.onChange(async (uiValue) => {
|
76
|
+
const oldValue = property.getModifiedOrDefaultValue();
|
77
|
+
const convertedValue = optionsExt.componentToPluginSettingsValueConverter(uiValue);
|
78
|
+
if ((0, import_ValidationMessage.isValidationMessageHolder)(convertedValue)) {
|
79
|
+
property.setValidationMessage(convertedValue.validationMessage);
|
80
|
+
} else {
|
81
|
+
await property.setValueAndValidate(convertedValue);
|
83
82
|
}
|
84
|
-
|
85
|
-
|
86
|
-
|
87
|
-
|
83
|
+
const newValue = (0, import_ValidationMessage.isValidationMessageHolder)(convertedValue) ? void 0 : convertedValue;
|
84
|
+
await optionsExt.onChanged(newValue, oldValue);
|
85
|
+
});
|
86
|
+
if ((0, import_PlaceholderComponent.isPlaceholderComponent)(valueComponent)) {
|
87
|
+
valueComponent.setPlaceholder(optionsExt.pluginSettingsToComponentValueConverter(property.defaultValue));
|
88
|
+
}
|
89
|
+
validatorElement?.addEventListener("focus", validate);
|
90
|
+
validatorElement?.addEventListener("blur", validate);
|
91
|
+
validate();
|
92
|
+
return valueComponent;
|
93
|
+
function validate() {
|
94
|
+
if (!validatorElement) {
|
88
95
|
return;
|
89
96
|
}
|
90
|
-
|
91
|
-
|
92
|
-
|
93
|
-
|
97
|
+
if (!property.validationMessage) {
|
98
|
+
validatorElement.setCustomValidity("");
|
99
|
+
validatorElement.checkValidity();
|
100
|
+
property.setValidationMessage(validatorElement.validationMessage);
|
101
|
+
}
|
102
|
+
validatorElement.setCustomValidity(property.validationMessage);
|
103
|
+
validatorElement.title = property.validationMessage;
|
104
|
+
if (validatorElement.isActiveElement() && optionsExt.shouldShowValidationMessage) {
|
105
|
+
validatorElement.reportValidity();
|
94
106
|
}
|
95
|
-
await optionsExt.onChanged();
|
96
|
-
});
|
97
|
-
validatorElement?.addEventListener("focus", (0, import_Async.convertAsyncToSync)(() => validate()));
|
98
|
-
validatorElement?.addEventListener("blur", (0, import_Async.convertAsyncToSync)(() => validate()));
|
99
|
-
this.validatorsMap.set(valueComponent, () => validate());
|
100
|
-
(0, import_Async.invokeAsyncSafely)(() => validate());
|
101
|
-
return valueComponent;
|
102
|
-
}
|
103
|
-
/**
|
104
|
-
* Revalidates the value component.
|
105
|
-
*
|
106
|
-
* @param baseComponent - The base component to revalidate.
|
107
|
-
* @returns A promise that resolves to a boolean indicating whether the value component is valid.
|
108
|
-
*/
|
109
|
-
async revalidate(baseComponent) {
|
110
|
-
const validator = this.validatorsMap.get(baseComponent);
|
111
|
-
if (validator) {
|
112
|
-
return await validator();
|
113
107
|
}
|
114
|
-
|
108
|
+
}
|
109
|
+
hide() {
|
110
|
+
super.hide();
|
111
|
+
(0, import_Async.invokeAsyncSafely)(() => this.plugin.settingsManager.saveToFile());
|
115
112
|
}
|
116
113
|
}
|
117
114
|
// Annotate the CommonJS export names for ESM import in node:
|
118
115
|
0 && (module.exports = {
|
119
116
|
PluginSettingsTabBase
|
120
117
|
});
|
121
|
-
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../../../src/obsidian/Plugin/PluginSettingsTabBase.ts"],
  "sourcesContent": ["/**\n * @packageDocumentation PluginSettingsTabBase\n * This module defines a base class for creating plugin setting tabs in Obsidian.\n * It provides a utility method to bind value components to plugin settings and handle changes.\n */\n\nimport type { BaseComponent } from 'obsidian';\nimport type {\n  ConditionalKeys,\n  Promisable\n} from 'type-fest';\n\nimport { PluginSettingTab } from 'obsidian';\n\nimport type { ValueComponentWithChangeTracking } from '../Components/ValueComponentWithChangeTracking.ts';\nimport type { PluginBase } from './PluginBase.ts';\nimport type { PluginSettingsBase } from './PluginSettingsBase.ts';\n\nimport {\n  convertAsyncToSync,\n  invokeAsyncSafely\n} from '../../Async.ts';\nimport { CssClass } from '../../CssClass.ts';\nimport { noop } from '../../Function.ts';\nimport { getValidatorComponent } from '../Components/ValidatorComponent.ts';\nimport { getPluginId } from './PluginId.ts';\n\n/**\n * Options for binding a value component to a plugin setting.\n */\nexport interface BindOptions<PluginSettings, UIValue> {\n  /**\n   * A callback function that is called when the value of the component changes.\n   */\n  onChanged?: () => Promisable<void>;\n\n  /**\n   * The plugin settings object to bind the component to. Default is the plugin's current settings.\n   */\n  pluginSettings?: PluginSettings;\n\n  /**\n   * If true, saves the plugin settings automatically after the component value changes. Default is `true`.\n   */\n  shouldAutoSave?: boolean;\n\n  /**\n   * If true, shows the validation message when the component value is invalid. Default is `true`.\n   */\n  shouldShowValidationMessage?: boolean;\n\n  /**\n   * Validates the UI value before setting it on the plugin settings.\n   * @param uiValue - The value of the UI component.\n   * @returns An error message if the value is invalid, or `(empty string)` or `void` if it is valid.\n   */\n  // eslint-disable-next-line @typescript-eslint/no-invalid-void-type\n  valueValidator?: (uiValue: UIValue) => Promisable<string | void>;\n}\n\n/**\n * Extended options for binding a value component to a plugin setting.\n */\nexport interface BindOptionsExtended<PluginSettings, UIValue, Property extends keyof PluginSettings> extends BindOptions<PluginSettings, UIValue> {\n  /**\n   * Converts the UI component's value back to the plugin settings value.\n   * @param uiValue - The value of the UI component.\n   * @returns The value to set on the plugin settings.\n   */\n  componentToPluginSettingsValueConverter: (uiValue: UIValue) => PluginSettings[Property];\n\n  /**\n   * Converts the plugin settings value to the value used by the UI component.\n   * @param pluginSettingsValue - The value of the property in the plugin settings.\n   * @returns The value to set on the UI component.\n   */\n  pluginSettingsToComponentValueConverter: (pluginSettingsValue: PluginSettings[Property]) => UIValue;\n}\n\n// eslint-disable-next-line @typescript-eslint/no-explicit-any\ntype ExtractPluginSettings<T extends PluginBase<any>> = PluginSettingsBase & T['settingsClone'];\n\n/**\n * Base class for creating plugin settings tabs in Obsidian.\n * Provides a method for binding value components to plugin settings and handling changes.\n *\n * @typeParam TPlugin - The type of the plugin that extends PluginBase.\n */\n// eslint-disable-next-line @typescript-eslint/no-explicit-any\nexport abstract class PluginSettingsTabBase<TPlugin extends PluginBase<any>> extends PluginSettingTab {\n  private validatorsMap = new WeakMap<BaseComponent, () => Promise<boolean>>();\n\n  public constructor(public override plugin: TPlugin) {\n    super(plugin.app, plugin);\n    this.containerEl.addClass(CssClass.LibraryName, getPluginId(), CssClass.PluginSettingsTab);\n  }\n\n  /**\n   * Binds a value component to a plugin setting.\n   *\n   * @typeParam UIValue - The type of the value of the UI component.\n   * @typeParam TValueComponent - The type of the value component.\n   * @param valueComponent - The value component to bind.\n   * @param property - The property of the plugin settings to bind to.\n   * @param options - The options for binding the value component.\n   * @returns The value component.\n   */\n  public bind<\n    UIValue,\n    TValueComponent\n  >(\n    valueComponent: TValueComponent & ValueComponentWithChangeTracking<UIValue>,\n    property: ConditionalKeys<ExtractPluginSettings<TPlugin>, UIValue>,\n    options?: BindOptions<ExtractPluginSettings<TPlugin>, UIValue>\n  ): TValueComponent;\n  /**\n   * Binds a value component to a plugin setting.\n   *\n   * @typeParam UIValue - The type of the value of the UI component.\n   * @typeParam TValueComponent - The type of the value component.\n   * @typeParam Property - The property of the plugin settings to bind to.\n   * @param valueComponent - The value component to bind.\n   * @param property - The property of the plugin settings to bind to.\n   * @param options - The options for binding the value component.\n   * @returns The value component.\n   */\n  public bind<\n    UIValue,\n    TValueComponent,\n    Property extends keyof ExtractPluginSettings<TPlugin>\n  >(\n    valueComponent: TValueComponent & ValueComponentWithChangeTracking<UIValue>,\n    property: Property,\n    options: BindOptionsExtended<ExtractPluginSettings<TPlugin>, UIValue, Property>\n  ): TValueComponent;\n  /**\n   * Binds a value component to a plugin setting.\n   *\n   * @typeParam UIValue - The type of the value of the UI component.\n   * @typeParam TValueComponent - The type of the value component.\n   * @typeParam Property - The property of the plugin settings to bind to.\n   * @param valueComponent - The value component to bind.\n   * @param property - The property of the plugin settings to bind to.\n   * @param options - The options for binding the value component.\n   * @returns The value component.\n   */\n  public bind<\n    UIValue,\n    TValueComponent,\n    Property extends keyof ExtractPluginSettings<TPlugin>\n  >(\n    valueComponent: TValueComponent & ValueComponentWithChangeTracking<UIValue>,\n    property: Property,\n    options?: BindOptions<ExtractPluginSettings<TPlugin>, UIValue>\n  ): TValueComponent {\n    type PluginSettings = ExtractPluginSettings<TPlugin>;\n    type PropertyType = PluginSettings[Property];\n    const DEFAULT_OPTIONS: Required<BindOptionsExtended<PluginSettings, UIValue, Property>> = {\n      // eslint-disable-next-line @typescript-eslint/no-unsafe-return\n      componentToPluginSettingsValueConverter: (value: UIValue): PropertyType => value as PropertyType,\n      onChanged: noop,\n      pluginSettings: undefined as PluginSettings,\n      pluginSettingsToComponentValueConverter: (value: PropertyType): UIValue => value as UIValue,\n      shouldAutoSave: true,\n      shouldShowValidationMessage: true,\n      valueValidator: noop\n    };\n\n    const optionsExt: Required<BindOptionsExtended<PluginSettings, UIValue, Property>> = { ...DEFAULT_OPTIONS, ...options };\n\n    // eslint-disable-next-line @typescript-eslint/no-unsafe-return\n    const pluginSettingsFn = (): PluginSettings => optionsExt.pluginSettings ?? this.plugin.settingsClone;\n    const validatorElement = getValidatorComponent(valueComponent)?.validatorEl;\n\n    const validate = async (uiValue?: UIValue): Promise<boolean> => {\n      uiValue ??= valueComponent.getValue();\n      let errorMessage = await optionsExt.valueValidator(uiValue) as string | undefined;\n      if (validatorElement) {\n        if (!errorMessage) {\n          validatorElement.setCustomValidity('');\n          validatorElement.checkValidity();\n          errorMessage = validatorElement.validationMessage;\n        }\n\n        validatorElement.setCustomValidity(errorMessage);\n        validatorElement.title = errorMessage;\n        if (validatorElement.isActiveElement() && optionsExt.shouldShowValidationMessage) {\n          validatorElement.reportValidity();\n        }\n      }\n\n      return !errorMessage;\n    };\n\n    valueComponent\n      .setValue(optionsExt.pluginSettingsToComponentValueConverter(pluginSettingsFn()[property]))\n      .onChange(async (uiValue) => {\n        if (!await validate(uiValue)) {\n          return;\n        }\n        const pluginSettings = pluginSettingsFn();\n        pluginSettings[property] = optionsExt.componentToPluginSettingsValueConverter(uiValue);\n        if (optionsExt.shouldAutoSave) {\n          await this.plugin.saveSettings(pluginSettings);\n        }\n\n        await optionsExt.onChanged();\n      });\n\n    validatorElement?.addEventListener('focus', convertAsyncToSync(() => validate()));\n    validatorElement?.addEventListener('blur', convertAsyncToSync(() => validate()));\n    this.validatorsMap.set(valueComponent, () => validate());\n\n    invokeAsyncSafely(() => validate());\n    return valueComponent;\n  }\n\n  /**\n   * Revalidates the value component.\n   *\n   * @param baseComponent - The base component to revalidate.\n   * @returns A promise that resolves to a boolean indicating whether the value component is valid.\n   */\n  protected async revalidate(baseComponent: BaseComponent): Promise<boolean> {\n    const validator = this.validatorsMap.get(baseComponent);\n    if (validator) {\n      return await validator();\n    }\n\n    return true;\n  }\n}\n"],
  "mappings": ";;;;;;;;;;;;;;;;;;;;;;;;;AAAA;AAAA;AAAA;AAAA;AAAA;AAYA,sBAAiC;AAMjC,mBAGO;AACP,sBAAyB;AACzB,sBAAqB;AACrB,gCAAsC;AACtC,sBAA4B;AAgErB,MAAe,8BAA+D,iCAAiB;AAAA,EAG7F,YAA4B,QAAiB;AAClD,UAAM,OAAO,KAAK,MAAM;AADS;AAEjC,SAAK,YAAY,SAAS,yBAAS,iBAAa,6BAAY,GAAG,yBAAS,iBAAiB;AAAA,EAC3F;AAAA,EALQ,gBAAgB,oBAAI,QAA+C;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAwDpE,KAKL,gBACA,UACA,SACiB;AAGjB,UAAM,kBAAoF;AAAA;AAAA,MAExF,yCAAyC,CAAC,UAAiC;AAAA,MAC3E,WAAW;AAAA,MACX,gBAAgB;AAAA,MAChB,yCAAyC,CAAC,UAAiC;AAAA,MAC3E,gBAAgB;AAAA,MAChB,6BAA6B;AAAA,MAC7B,gBAAgB;AAAA,IAClB;AAEA,UAAM,aAA+E,EAAE,GAAG,iBAAiB,GAAG,QAAQ;AAGtH,UAAM,mBAAmB,MAAsB,WAAW,kBAAkB,KAAK,OAAO;AACxF,UAAM,uBAAmB,iDAAsB,cAAc,GAAG;AAEhE,UAAM,WAAW,OAAO,YAAwC;AAC9D,kBAAY,eAAe,SAAS;AACpC,UAAI,eAAe,MAAM,WAAW,eAAe,OAAO;AAC1D,UAAI,kBAAkB;AACpB,YAAI,CAAC,cAAc;AACjB,2BAAiB,kBAAkB,EAAE;AACrC,2BAAiB,cAAc;AAC/B,yBAAe,iBAAiB;AAAA,QAClC;AAEA,yBAAiB,kBAAkB,YAAY;AAC/C,yBAAiB,QAAQ;AACzB,YAAI,iBAAiB,gBAAgB,KAAK,WAAW,6BAA6B;AAChF,2BAAiB,eAAe;AAAA,QAClC;AAAA,MACF;AAEA,aAAO,CAAC;AAAA,IACV;AAEA,mBACG,SAAS,WAAW,wCAAwC,iBAAiB,EAAE,QAAQ,CAAC,CAAC,EACzF,SAAS,OAAO,YAAY;AAC3B,UAAI,CAAC,MAAM,SAAS,OAAO,GAAG;AAC5B;AAAA,MACF;AACA,YAAM,iBAAiB,iBAAiB;AACxC,qBAAe,QAAQ,IAAI,WAAW,wCAAwC,OAAO;AACrF,UAAI,WAAW,gBAAgB;AAC7B,cAAM,KAAK,OAAO,aAAa,cAAc;AAAA,MAC/C;AAEA,YAAM,WAAW,UAAU;AAAA,IAC7B,CAAC;AAEH,sBAAkB,iBAAiB,aAAS,iCAAmB,MAAM,SAAS,CAAC,CAAC;AAChF,sBAAkB,iBAAiB,YAAQ,iCAAmB,MAAM,SAAS,CAAC,CAAC;AAC/E,SAAK,cAAc,IAAI,gBAAgB,MAAM,SAAS,CAAC;AAEvD,wCAAkB,MAAM,SAAS,CAAC;AAClC,WAAO;AAAA,EACT;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAQA,MAAgB,WAAW,eAAgD;AACzE,UAAM,YAAY,KAAK,cAAc,IAAI,aAAa;AACtD,QAAI,WAAW;AACb,aAAO,MAAM,UAAU;AAAA,IACzB;AAEA,WAAO;AAAA,EACT;AACF;",
  "names": []
}

|
118
|
+
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../../../src/obsidian/Plugin/PluginSettingsTabBase.ts"],
  "sourcesContent": ["/**\n * @packageDocumentation PluginSettingsTabBase\n * This module defines a base class for creating plugin setting tabs in Obsidian.\n * It provides a utility method to bind value components to plugin settings and handle changes.\n */\n\nimport type {\n  ConditionalKeys,\n  Promisable\n} from 'type-fest';\n\nimport { PluginSettingTab } from 'obsidian';\n\nimport type { StringKeys } from '../../Object.ts';\nimport type { ValueComponentWithChangeTracking } from '../Components/ValueComponentWithChangeTracking.ts';\nimport type { ValidationMessageHolder } from '../ValidationMessage.ts';\nimport type { PluginBase } from './PluginBase.ts';\nimport type { PluginSettingsProperty } from './PluginSettingsManagerBase.ts';\n\nimport { invokeAsyncSafely } from '../../Async.ts';\nimport { CssClass } from '../../CssClass.ts';\nimport { noop } from '../../Function.ts';\nimport { isPlaceholderComponent } from '../Components/PlaceholderComponent.ts';\nimport { getValidatorComponent } from '../Components/ValidatorComponent.ts';\nimport { isValidationMessageHolder } from '../ValidationMessage.ts';\nimport { getPluginId } from './PluginId.ts';\n\n/**\n * Options for binding a value component to a plugin setting.\n */\nexport interface BindOptions<T> {\n  /**\n   * A callback function that is called when the value of the component changes.\n   */\n  onChanged?(newValue: T | undefined, oldValue: T): Promisable<void>;\n\n  /**\n   * If true, shows the validation message when the component value is invalid. Default is `true`.\n   */\n  shouldShowValidationMessage?: boolean;\n}\n\n/**\n * Extended options for binding a value component to a plugin setting.\n */\nexport interface BindOptionsExtended<\n  PluginSettings extends object,\n  UIValue,\n  PropertyName extends StringKeys<PluginSettings>\n> extends BindOptions<PluginSettings[PropertyName]> {\n  /**\n   * Converts the UI component's value back to the plugin settings value.\n   * @param uiValue - The value of the UI component.\n   * @returns The value to set on the plugin settings.\n   */\n  componentToPluginSettingsValueConverter: (uiValue: UIValue) => PluginSettings[PropertyName] | ValidationMessageHolder;\n\n  /**\n   * Converts the plugin settings value to the value used by the UI component.\n   * @param pluginSettingsValue - The value of the property in the plugin settings.\n   * @returns The value to set on the UI component.\n   */\n  pluginSettingsToComponentValueConverter: (pluginSettingsValue: PluginSettings[PropertyName]) => UIValue;\n}\n\n// eslint-disable-next-line @typescript-eslint/no-explicit-any\ntype ExtractPluginSettings<Plugin extends PluginBase<any>> = Plugin extends PluginBase<infer PluginSettings> ? PluginSettings : never;\n\n/**\n * Base class for creating plugin settings tabs in Obsidian.\n * Provides a method for binding value components to plugin settings and handling changes.\n *\n * @typeParam TPlugin - The type of the plugin that extends PluginBase.\n */\n// eslint-disable-next-line @typescript-eslint/no-explicit-any\nexport abstract class PluginSettingsTabBase<TPlugin extends PluginBase<any>> extends PluginSettingTab {\n  public constructor(public override plugin: TPlugin) {\n    super(plugin.app, plugin);\n    this.containerEl.addClass(CssClass.LibraryName, getPluginId(), CssClass.PluginSettingsTab);\n  }\n\n  /**\n   * Binds a value component to a plugin setting.\n   *\n   * @typeParam UIValue - The type of the value of the UI component.\n   * @typeParam TValueComponent - The type of the value component.\n   * @param valueComponent - The value component to bind.\n   * @param propertyName - The property of the plugin settings to bind to.\n   * @param options - The options for binding the value component.\n   * @returns The value component.\n   */\n  public bind<\n    UIValue,\n    TValueComponent\n  >(\n    valueComponent: TValueComponent & ValueComponentWithChangeTracking<UIValue>,\n    propertyName: ConditionalKeys<ExtractPluginSettings<TPlugin>, UIValue>,\n    options?: BindOptions<UIValue>\n  ): TValueComponent;\n  /**\n   * Binds a value component to a plugin setting.\n   *\n   * @typeParam UIValue - The type of the value of the UI component.\n   * @typeParam TValueComponent - The type of the value component.\n   * @typeParam PropertyName - The property name of the plugin settings to bind to.\n   * @param valueComponent - The value component to bind.\n   * @param propertyName - The property name of the plugin settings to bind to.\n   * @param options - The options for binding the value component.\n   * @returns The value component.\n   */\n  public bind<\n    UIValue,\n    TValueComponent,\n    PropertyName extends StringKeys<ExtractPluginSettings<TPlugin>>\n  >(\n    valueComponent: TValueComponent & ValueComponentWithChangeTracking<UIValue>,\n    propertyName: PropertyName,\n    options: BindOptionsExtended<ExtractPluginSettings<TPlugin>, UIValue, PropertyName>\n  ): TValueComponent;\n  /**\n   * Binds a value component to a plugin setting.\n   *\n   * @typeParam UIValue - The type of the value of the UI component.\n   * @typeParam TValueComponent - The type of the value component.\n   * @typeParam PropertyName - The property name of the plugin settings to bind to.\n   * @param valueComponent - The value component to bind.\n   * @param propertyName - The property name of the plugin settings to bind to.\n   * @param options - The options for binding the value component.\n   * @returns The value component.\n   */\n  public bind<\n    UIValue,\n    TValueComponent,\n    PropertyName extends StringKeys<ExtractPluginSettings<TPlugin>>\n  >(\n    valueComponent: TValueComponent & ValueComponentWithChangeTracking<UIValue>,\n    propertyName: PropertyName,\n    options?: BindOptions<ExtractPluginSettings<TPlugin>[PropertyName]>\n  ): TValueComponent {\n    type PluginSettings = ExtractPluginSettings<TPlugin>;\n    type PropertyType = PluginSettings[PropertyName];\n    const DEFAULT_OPTIONS: Required<BindOptionsExtended<PluginSettings, UIValue, PropertyName>> = {\n      // eslint-disable-next-line @typescript-eslint/no-unsafe-return\n      componentToPluginSettingsValueConverter: (value: UIValue): PropertyType => value as PropertyType,\n      onChanged: noop,\n      pluginSettingsToComponentValueConverter: (value: PropertyType): UIValue => value as UIValue,\n      shouldShowValidationMessage: true\n    };\n\n    const optionsExt: Required<BindOptionsExtended<PluginSettings, UIValue, PropertyName>> = { ...DEFAULT_OPTIONS, ...options };\n\n    const validatorElement = getValidatorComponent(valueComponent)?.validatorEl;\n\n    const property = this.plugin.settingsManager.getProperty(propertyName) as PluginSettingsProperty<PropertyType>;\n\n    let value = property.getModifiedValue();\n\n    if (value === undefined && !isPlaceholderComponent(valueComponent)) {\n      value = property.defaultValue;\n      property.setValue(value);\n    }\n\n    if (value !== undefined) {\n      valueComponent.setValue(optionsExt.pluginSettingsToComponentValueConverter(value));\n    }\n\n    valueComponent.onChange(async (uiValue) => {\n      const oldValue = property.getModifiedOrDefaultValue();\n      const convertedValue = optionsExt.componentToPluginSettingsValueConverter(uiValue);\n      if (isValidationMessageHolder(convertedValue)) {\n        property.setValidationMessage(convertedValue.validationMessage);\n      } else {\n        await property.setValueAndValidate(convertedValue);\n      }\n      const newValue = isValidationMessageHolder(convertedValue) ? undefined : convertedValue;\n      await optionsExt.onChanged(newValue, oldValue);\n    });\n\n    if (isPlaceholderComponent(valueComponent)) {\n      valueComponent.setPlaceholder(optionsExt.pluginSettingsToComponentValueConverter(property.defaultValue) as string);\n    }\n\n    validatorElement?.addEventListener('focus', validate);\n    validatorElement?.addEventListener('blur', validate);\n\n    validate();\n    return valueComponent;\n\n    function validate(): void {\n      if (!validatorElement) {\n        return;\n      }\n\n      if (!property.validationMessage) {\n        validatorElement.setCustomValidity('');\n        validatorElement.checkValidity();\n        property.setValidationMessage(validatorElement.validationMessage);\n      }\n\n      validatorElement.setCustomValidity(property.validationMessage);\n      validatorElement.title = property.validationMessage;\n      if (validatorElement.isActiveElement() && optionsExt.shouldShowValidationMessage) {\n        validatorElement.reportValidity();\n      }\n    }\n  }\n\n  public override hide(): void {\n    super.hide();\n    invokeAsyncSafely(() => this.plugin.settingsManager.saveToFile());\n  }\n}\n"],
  "mappings": ";;;;;;;;;;;;;;;;;;;;;;;;;AAAA;AAAA;AAAA;AAAA;AAAA;AAWA,sBAAiC;AAQjC,mBAAkC;AAClC,sBAAyB;AACzB,sBAAqB;AACrB,kCAAuC;AACvC,gCAAsC;AACtC,+BAA0C;AAC1C,sBAA4B;AAkDrB,MAAe,8BAA+D,iCAAiB;AAAA,EAC7F,YAA4B,QAAiB;AAClD,UAAM,OAAO,KAAK,MAAM;AADS;AAEjC,SAAK,YAAY,SAAS,yBAAS,iBAAa,6BAAY,GAAG,yBAAS,iBAAiB;AAAA,EAC3F;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAmDO,KAKL,gBACA,cACA,SACiB;AAGjB,UAAM,kBAAwF;AAAA;AAAA,MAE5F,yCAAyC,CAACA,WAAiCA;AAAA,MAC3E,WAAW;AAAA,MACX,yCAAyC,CAACA,WAAiCA;AAAA,MAC3E,6BAA6B;AAAA,IAC/B;AAEA,UAAM,aAAmF,EAAE,GAAG,iBAAiB,GAAG,QAAQ;AAE1H,UAAM,uBAAmB,iDAAsB,cAAc,GAAG;AAEhE,UAAM,WAAW,KAAK,OAAO,gBAAgB,YAAY,YAAY;AAErE,QAAI,QAAQ,SAAS,iBAAiB;AAEtC,QAAI,UAAU,UAAa,KAAC,oDAAuB,cAAc,GAAG;AAClE,cAAQ,SAAS;AACjB,eAAS,SAAS,KAAK;AAAA,IACzB;AAEA,QAAI,UAAU,QAAW;AACvB,qBAAe,SAAS,WAAW,wCAAwC,KAAK,CAAC;AAAA,IACnF;AAEA,mBAAe,SAAS,OAAO,YAAY;AACzC,YAAM,WAAW,SAAS,0BAA0B;AACpD,YAAM,iBAAiB,WAAW,wCAAwC,OAAO;AACjF,cAAI,oDAA0B,cAAc,GAAG;AAC7C,iBAAS,qBAAqB,eAAe,iBAAiB;AAAA,MAChE,OAAO;AACL,cAAM,SAAS,oBAAoB,cAAc;AAAA,MACnD;AACA,YAAM,eAAW,oDAA0B,cAAc,IAAI,SAAY;AACzE,YAAM,WAAW,UAAU,UAAU,QAAQ;AAAA,IAC/C,CAAC;AAED,YAAI,oDAAuB,cAAc,GAAG;AAC1C,qBAAe,eAAe,WAAW,wCAAwC,SAAS,YAAY,CAAW;AAAA,IACnH;AAEA,sBAAkB,iBAAiB,SAAS,QAAQ;AACpD,sBAAkB,iBAAiB,QAAQ,QAAQ;AAEnD,aAAS;AACT,WAAO;AAEP,aAAS,WAAiB;AACxB,UAAI,CAAC,kBAAkB;AACrB;AAAA,MACF;AAEA,UAAI,CAAC,SAAS,mBAAmB;AAC/B,yBAAiB,kBAAkB,EAAE;AACrC,yBAAiB,cAAc;AAC/B,iBAAS,qBAAqB,iBAAiB,iBAAiB;AAAA,MAClE;AAEA,uBAAiB,kBAAkB,SAAS,iBAAiB;AAC7D,uBAAiB,QAAQ,SAAS;AAClC,UAAI,iBAAiB,gBAAgB,KAAK,WAAW,6BAA6B;AAChF,yBAAiB,eAAe;AAAA,MAClC;AAAA,IACF;AAAA,EACF;AAAA,EAEgB,OAAa;AAC3B,UAAM,KAAK;AACX,wCAAkB,MAAM,KAAK,OAAO,gBAAgB,WAAW,CAAC;AAAA,EAClE;AACF;",
  "names": ["value"]
}

|