obsidian-dev-utils 60.0.0 → 61.0.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/CHANGELOG.md +8 -0
- package/dist/lib/cjs/library.cjs +1 -1
- package/dist/lib/cjs/obsidian/data-handler.cjs +163 -0
- package/dist/lib/cjs/obsidian/data-handler.d.cts +48 -0
- package/dist/lib/cjs/obsidian/index.cjs +4 -1
- package/dist/lib/cjs/obsidian/index.d.cts +1 -0
- package/dist/lib/cjs/obsidian/plugin/components/plugin-settings-component.cjs +32 -33
- package/dist/lib/cjs/obsidian/plugin/components/plugin-settings-component.d.cts +5 -24
- package/dist/lib/esm/library.mjs +1 -1
- package/dist/lib/esm/obsidian/data-handler.d.mts +48 -0
- package/dist/lib/esm/obsidian/data-handler.mjs +55 -0
- package/dist/lib/esm/obsidian/index.d.mts +1 -0
- package/dist/lib/esm/obsidian/index.mjs +3 -1
- package/dist/lib/esm/obsidian/plugin/components/plugin-settings-component.d.mts +5 -24
- package/dist/lib/esm/obsidian/plugin/components/plugin-settings-component.mjs +32 -33
- package/obsidian/data-handler/package.json +6 -0
- package/package.json +1 -1
package/CHANGELOG.md
CHANGED
package/dist/lib/cjs/library.cjs
CHANGED
|
@@ -130,7 +130,7 @@ __export(library_exports, {
|
|
|
130
130
|
LIBRARY_VERSION: () => LIBRARY_VERSION
|
|
131
131
|
});
|
|
132
132
|
module.exports = __toCommonJS(library_exports);
|
|
133
|
-
const LIBRARY_VERSION = "
|
|
133
|
+
const LIBRARY_VERSION = "61.0.1";
|
|
134
134
|
const LIBRARY_NAME = "obsidian-dev-utils";
|
|
135
135
|
const LIBRARY_STYLES = ".obsidian-dev-utils.code-highlighter-component textarea, .obsidian-dev-utils.code-highlighter-component pre, .obsidian-dev-utils.code-highlighter-component code {\n font-family: var(--font-monospace);\n line-height: var(--line-height-normal);\n margin: 0;\n}\n.obsidian-dev-utils.code-highlighter-component textarea, .obsidian-dev-utils.code-highlighter-component code {\n font-size: var(--code-size);\n}\n.obsidian-dev-utils.code-highlighter-component textarea {\n background: transparent;\n color: transparent;\n z-index: 2;\n width: 20em;\n height: 10em;\n}\n.obsidian-dev-utils.code-highlighter-component pre {\n position: absolute;\n pointer-events: none;\n border: var(--input-border-width) solid transparent;\n overflow: auto;\n inset: 0;\n padding: var(--size-4-1) var(--size-4-2);\n z-index: 1;\n}\n.obsidian-dev-utils.code-highlighter-component pre::after {\n content: \"\";\n display: block;\n height: var(--bottom-gap, 0);\n}\n.obsidian-dev-utils.code-highlighter-component pre.is-placeholder {\n opacity: 0.6;\n}\n.obsidian-dev-utils.code-highlighter-component code {\n display: block;\n padding: 0;\n}\n\n.obsidian-dev-utils input[type=url] {\n height: var(--input-height);\n}\n.obsidian-dev-utils input[type=month],\n.obsidian-dev-utils input[type=tel],\n.obsidian-dev-utils input[type=time],\n.obsidian-dev-utils input[type=url],\n.obsidian-dev-utils input[type=week] {\n -webkit-app-region: no-drag;\n background: var(--background-modifier-form-field);\n border: var(--input-border-width) solid var(--background-modifier-border);\n color: var(--text-normal);\n font-family: inherit;\n padding: var(--size-4-1) var(--size-4-2);\n font-size: var(--font-ui-small);\n border-radius: var(--input-radius);\n outline: none;\n}\n@media (hover: hover) {\n .obsidian-dev-utils input[type=month]:hover,\n .obsidian-dev-utils input[type=tel]:hover,\n .obsidian-dev-utils input[type=time]:hover,\n .obsidian-dev-utils input[type=url]:hover,\n .obsidian-dev-utils input[type=week]:hover {\n border-color: var(--background-modifier-border-hover);\n transition: box-shadow 0.15s ease-in-out, border 0.15s ease-in-out;\n }\n}\n.obsidian-dev-utils input[type=month]:active, .obsidian-dev-utils input[type=month]:focus,\n.obsidian-dev-utils input[type=tel]:active,\n.obsidian-dev-utils input[type=tel]:focus,\n.obsidian-dev-utils input[type=time]:active,\n.obsidian-dev-utils input[type=time]:focus,\n.obsidian-dev-utils input[type=url]:active,\n.obsidian-dev-utils input[type=url]:focus,\n.obsidian-dev-utils input[type=week]:active,\n.obsidian-dev-utils input[type=week]:focus {\n border-color: var(--background-modifier-border-focus);\n transition: box-shadow 0.15s ease-in-out, border 0.15s ease-in-out;\n}\n.obsidian-dev-utils input[type=month]:active, .obsidian-dev-utils input[type=month]:focus, .obsidian-dev-utils input[type=month]:focus-visible,\n.obsidian-dev-utils input[type=tel]:active,\n.obsidian-dev-utils input[type=tel]:focus,\n.obsidian-dev-utils input[type=tel]:focus-visible,\n.obsidian-dev-utils input[type=time]:active,\n.obsidian-dev-utils input[type=time]:focus,\n.obsidian-dev-utils input[type=time]:focus-visible,\n.obsidian-dev-utils input[type=url]:active,\n.obsidian-dev-utils input[type=url]:focus,\n.obsidian-dev-utils input[type=url]:focus-visible,\n.obsidian-dev-utils input[type=week]:active,\n.obsidian-dev-utils input[type=week]:focus,\n.obsidian-dev-utils input[type=week]:focus-visible {\n box-shadow: 0 0 0 2px var(--background-modifier-border-focus);\n}\n.obsidian-dev-utils input[type=month]::placeholder,\n.obsidian-dev-utils input[type=tel]::placeholder,\n.obsidian-dev-utils input[type=time]::placeholder,\n.obsidian-dev-utils input[type=url]::placeholder,\n.obsidian-dev-utils input[type=week]::placeholder {\n color: var(--text-faint);\n}\n.mod-rtl input[type=month],\n.mod-rtl input[type=time],\n.mod-rtl input[type=week],\n.is-rtl input[type=month],\n.is-rtl input[type=time],\n.is-rtl input[type=week],\n.rtl input[type=month],\n.rtl input[type=time],\n.rtl input[type=week] {\n direction: rtl;\n}\n.mod-rtl input[type=month]::-webkit-calendar-picker-indicator,\n.mod-rtl input[type=time]::-webkit-calendar-picker-indicator,\n.mod-rtl input[type=week]::-webkit-calendar-picker-indicator,\n.is-rtl input[type=month]::-webkit-calendar-picker-indicator,\n.is-rtl input[type=time]::-webkit-calendar-picker-indicator,\n.is-rtl input[type=week]::-webkit-calendar-picker-indicator,\n.rtl input[type=month]::-webkit-calendar-picker-indicator,\n.rtl input[type=time]::-webkit-calendar-picker-indicator,\n.rtl input[type=week]::-webkit-calendar-picker-indicator {\n right: var(--size-4-1);\n left: auto;\n}\n\n.obsidian-dev-utils input[type=month],\n.obsidian-dev-utils input[type=time],\n.obsidian-dev-utils input[type=week] {\n font-variant-numeric: tabular-nums;\n position: relative;\n}\n.obsidian-dev-utils input[type=month]::-webkit-datetime-edit-text,\n.obsidian-dev-utils input[type=time]::-webkit-datetime-edit-text,\n.obsidian-dev-utils input[type=week]::-webkit-datetime-edit-text {\n color: var(--text-faint);\n padding-inline-end: 0;\n}\n.obsidian-dev-utils input[type=month]::-webkit-calendar-picker-indicator,\n.obsidian-dev-utils input[type=time]::-webkit-calendar-picker-indicator,\n.obsidian-dev-utils input[type=week]::-webkit-calendar-picker-indicator {\n position: absolute;\n left: var(--size-4-1);\n right: auto;\n opacity: 0.5;\n}\n.obsidian-dev-utils input[type=month]::-webkit-datetime-edit-month-field:active, .obsidian-dev-utils input[type=month]::-webkit-datetime-edit-month-field:focus, .obsidian-dev-utils input[type=month]::-webkit-datetime-edit-day-field:active, .obsidian-dev-utils input[type=month]::-webkit-datetime-edit-day-field:focus, .obsidian-dev-utils input[type=month]::-webkit-datetime-edit-year-field:active, .obsidian-dev-utils input[type=month]::-webkit-datetime-edit-year-field:focus,\n.obsidian-dev-utils input[type=time]::-webkit-datetime-edit-month-field:active,\n.obsidian-dev-utils input[type=time]::-webkit-datetime-edit-month-field:focus,\n.obsidian-dev-utils input[type=time]::-webkit-datetime-edit-day-field:active,\n.obsidian-dev-utils input[type=time]::-webkit-datetime-edit-day-field:focus,\n.obsidian-dev-utils input[type=time]::-webkit-datetime-edit-year-field:active,\n.obsidian-dev-utils input[type=time]::-webkit-datetime-edit-year-field:focus,\n.obsidian-dev-utils input[type=week]::-webkit-datetime-edit-month-field:active,\n.obsidian-dev-utils input[type=week]::-webkit-datetime-edit-month-field:focus,\n.obsidian-dev-utils input[type=week]::-webkit-datetime-edit-day-field:active,\n.obsidian-dev-utils input[type=week]::-webkit-datetime-edit-day-field:focus,\n.obsidian-dev-utils input[type=week]::-webkit-datetime-edit-year-field:active,\n.obsidian-dev-utils input[type=week]::-webkit-datetime-edit-year-field:focus {\n background-color: var(--text-selection);\n color: var(--text-normal);\n cursor: text;\n}\n.mod-rtl .obsidian-dev-utils input[type=month], .is-rtl .obsidian-dev-utils input[type=month], .rtl .obsidian-dev-utils input[type=month],\n.mod-rtl .obsidian-dev-utils input[type=time],\n.is-rtl .obsidian-dev-utils input[type=time],\n.rtl .obsidian-dev-utils input[type=time],\n.mod-rtl .obsidian-dev-utils input[type=week],\n.is-rtl .obsidian-dev-utils input[type=week],\n.rtl .obsidian-dev-utils input[type=week] {\n direction: rtl;\n}\n.mod-rtl .obsidian-dev-utils input[type=month]::-webkit-calendar-picker-indicator, .is-rtl .obsidian-dev-utils input[type=month]::-webkit-calendar-picker-indicator, .rtl .obsidian-dev-utils input[type=month]::-webkit-calendar-picker-indicator,\n.mod-rtl .obsidian-dev-utils input[type=time]::-webkit-calendar-picker-indicator,\n.is-rtl .obsidian-dev-utils input[type=time]::-webkit-calendar-picker-indicator,\n.rtl .obsidian-dev-utils input[type=time]::-webkit-calendar-picker-indicator,\n.mod-rtl .obsidian-dev-utils input[type=week]::-webkit-calendar-picker-indicator,\n.is-rtl .obsidian-dev-utils input[type=week]::-webkit-calendar-picker-indicator,\n.rtl .obsidian-dev-utils input[type=week]::-webkit-calendar-picker-indicator {\n left: auto;\n right: var(--size-4-1);\n}\n\nbody:not(.is-ios):not(.is-android) .obsidian-dev-utils input[type=month],\nbody:not(.is-ios):not(.is-android) .obsidian-dev-utils input[type=time],\nbody:not(.is-ios):not(.is-android) .obsidian-dev-utils input[type=week] {\n padding-inline-start: var(--size-4-6);\n}\n\n.obsidian-dev-utils input[type=time]::-webkit-calendar-picker-indicator {\n margin-inline-start: 0;\n}\n\n.obsidian-dev-utilsprogress.loop {\n min-width: 200px;\n}\n\n.obsidian-dev-utils.modal-container .ok-button {\n margin-right: 10px;\n margin-top: 20px;\n}\n\n.obsidian-dev-utils .multiple-dropdown-component select,\n.obsidian-dev-utils .multiple-dropdown-component select:focus,\n.obsidian-dev-utils .multiple-dropdown-component .dropdown {\n height: auto;\n padding-top: 3px;\n}\n.obsidian-dev-utils .multiple-dropdown-component select option:checked,\n.obsidian-dev-utils .multiple-dropdown-component select:focus option:checked,\n.obsidian-dev-utils .multiple-dropdown-component .dropdown option:checked {\n background-color: #1967d2;\n color: #fff;\n}\n\n.obsidian-dev-utils.plugin-settings-tab a:focus {\n outline: 2px solid var(--link-color);\n}\n\n.obsidian-dev-utils.prompt-modal .text-box {\n width: 100%;\n}\n\n.obsidian-dev-utils.tri-state-checkbox-component input[type=checkbox]:indeterminate {\n appearance: checkbox;\n}\n\n.obsidian-dev-utils :invalid {\n box-shadow: 0 0 0 2px var(--text-error);\n}\n.obsidian-dev-utils input.metadata-input-text:active:invalid, .obsidian-dev-utils input.metadata-input-text:focus-visible:invalid, .obsidian-dev-utils input.metadata-input-text:focus:invalid,\n.obsidian-dev-utils input[type=date]:active:invalid,\n.obsidian-dev-utils input[type=date]:focus-visible:invalid,\n.obsidian-dev-utils input[type=date]:focus:invalid,\n.obsidian-dev-utils input[type=datetime-local]:active:invalid,\n.obsidian-dev-utils input[type=datetime-local]:focus-visible:invalid,\n.obsidian-dev-utils input[type=datetime-local]:focus:invalid,\n.obsidian-dev-utils input[type=email]:active:invalid,\n.obsidian-dev-utils input[type=email]:focus-visible:invalid,\n.obsidian-dev-utils input[type=email]:focus:invalid,\n.obsidian-dev-utils input[type=number]:active:invalid,\n.obsidian-dev-utils input[type=number]:focus-visible:invalid,\n.obsidian-dev-utils input[type=number]:focus:invalid,\n.obsidian-dev-utils input[type=password]:active:invalid,\n.obsidian-dev-utils input[type=password]:focus-visible:invalid,\n.obsidian-dev-utils input[type=password]:focus:invalid,\n.obsidian-dev-utils input[type=search]:active:invalid,\n.obsidian-dev-utils input[type=search]:focus-visible:invalid,\n.obsidian-dev-utils input[type=search]:focus:invalid,\n.obsidian-dev-utils input[type=text]:active:invalid,\n.obsidian-dev-utils input[type=text]:focus-visible:invalid,\n.obsidian-dev-utils input[type=text]:focus:invalid,\n.obsidian-dev-utils textarea:active:invalid,\n.obsidian-dev-utils textarea:focus-visible:invalid,\n.obsidian-dev-utils textarea:focus:invalid {\n box-shadow: 0 0 0 2px var(--text-error);\n}\n.obsidian-dev-utils.setting-component-wrapper {\n position: relative;\n display: inline-flex;\n}\n.obsidian-dev-utils.overlay-validator {\n caret-color: transparent;\n cursor: default;\n position: absolute;\n background-color: transparent;\n border: none;\n outline: none;\n pointer-events: none;\n z-index: 9999;\n left: 0;\n top: 0;\n width: 100%;\n height: 100%;\n}\n.obsidian-dev-utils.tooltip.tooltip-validator {\n position: absolute;\n top: calc(100% + 8px);\n width: max-content;\n}\n\n/*# sourceMappingURL=data:application/json;charset=utf-8,%7B%22version%22:3,%22sourceRoot%22:%22%22,%22sources%22:%5B%22../src/styles/code-highlighter-component.scss%22,%22../src/styles/input.scss%22,%22../src/styles/input-time.scss%22,%22../src/styles/loop.scss%22,%22../src/styles/modal-container.scss%22,%22../src/styles/multiple-dropdown-component.scss%22,%22../src/styles/plugin-settings-tab.scss%22,%22../src/styles/prompt-modal.scss%22,%22../src/styles/tri-state-checkbox-component.scss%22,%22../src/styles/validation.scss%22%5D,%22names%22:%5B%5D,%22mappings%22:%22AAEI;EACE;EACA;EACA;;AAGF;EACE;;AAGF;EACE;EACA;EACA;EACA;EACA;;AAGF;EACE;EACA;EACA;EACA;EACA;EACA;EACA;;AAGF;EACE;EACA;EACA;;AAGF;EACE;;AAGF;EACE;EACA;;;ACzCJ;EACE;;AAGF;AAAA;AAAA;AAAA;AAAA;EAKE;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;;AAGE;EACE;AAAA;AAAA;AAAA;AAAA;IACE;IACA,YACE;;;AAMR;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;EAEE;EACA,YACE;;AAIJ;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;EAGE;;AAGF;AAAA;AAAA;AAAA;AAAA;EACE;;AASE;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;EAGE;;AAEA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;EACE;EACA;;;AC7DV;AAAA;AAAA;EAGE;EACA;;AAEA;AAAA;AAAA;EACE;EACA;;AAGF;AAAA;AAAA;EACE;EACA;EACA;EACA;;AAMA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;EAEE;EACA;EACA;;AAIK;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;EAGP;;AAEA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;EACE;EACA;;;AAKF;AAAA;AAAA;EACE;;;AAMJ;EACE;;;ACjDJ;EACE;;;ACAA;EACE;EACA;;;ACFF;AAAA;AAAA;EAGE;EACA;;AAEA;AAAA;AAAA;EACE;EACA;;;ACRJ;EACE;;;ACDF;EACE;;;ACDF;EACE;;;ACEJ;EAJA;;AAoBI;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;EApBJ;;AA0BA;EACE;EACA;;AAGF;EACE;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;;AAGF;EACE;EACA;EACA%22,%22file%22:%22styles.css%22,%22sourcesContent%22:%5B%22.obsidian-dev-utils%20%7B%5Cn%20%20&.code-highlighter-component%20%7B%5Cn%20%20%20%20textarea,%20pre,%20code%20%7B%5Cn%20%20%20%20%20%20font-family:%20var(--font-monospace);%5Cn%20%20%20%20%20%20line-height:%20var(--line-height-normal);%5Cn%20%20%20%20%20%20margin:%200;%5Cn%20%20%20%20%7D%5Cn%5Cn%20%20%20%20textarea,%20code%20%7B%5Cn%20%20%20%20%20%20font-size:%20var(--code-size);%5Cn%20%20%20%20%7D%5Cn%5Cn%20%20%20%20textarea%20%7B%5Cn%20%20%20%20%20%20background:%20transparent;%5Cn%20%20%20%20%20%20color:%20transparent;%5Cn%20%20%20%20%20%20z-index:%202;%5Cn%20%20%20%20%20%20width:%2020em;%5Cn%20%20%20%20%20%20height:%2010em;%5Cn%20%20%20%20%7D%5Cn%5Cn%20%20%20%20pre%20%7B%5Cn%20%20%20%20%20%20position:%20absolute;%5Cn%20%20%20%20%20%20pointer-events:%20none;%5Cn%20%20%20%20%20%20border:%20var(--input-border-width)%20solid%20transparent;%5Cn%20%20%20%20%20%20overflow:%20auto;%5Cn%20%20%20%20%20%20inset:%200;%5Cn%20%20%20%20%20%20padding:%20var(--size-4-1)%20var(--size-4-2);%5Cn%20%20%20%20%20%20z-index:%201;%5Cn%20%20%20%20%7D%5Cn%5Cn%20%20%20%20pre::after%20%7B%5Cn%20%20%20%20%20%20content:%20%5C%22%5C%22;%5Cn%20%20%20%20%20%20display:%20block;%5Cn%20%20%20%20%20%20height:%20var(--bottom-gap,%200);%5Cn%20%20%20%20%7D%5Cn%5Cn%20%20%20%20pre.is-placeholder%20%7B%5Cn%20%20%20%20%20%20opacity:%200.6;%5Cn%20%20%20%20%7D%5Cn%5Cn%20%20%20%20code%20%7B%5Cn%20%20%20%20%20%20display:%20block;%5Cn%20%20%20%20%20%20padding:%200;%5Cn%20%20%20%20%7D%5Cn%20%20%7D%5Cn%7D%5Cn%22,%22.obsidian-dev-utils%20%7B%5Cn%20%20input%5Btype='url'%5D%20%7B%5Cn%20%20%20%20height:%20var(--input-height)%5Cn%20%20%7D%5Cn%5Cn%20%20input%5Btype='month'%5D,%5Cn%20%20input%5Btype='tel'%5D,%5Cn%20%20input%5Btype='time'%5D,%5Cn%20%20input%5Btype='url'%5D,%5Cn%20%20input%5Btype='week'%5D%20%7B%5Cn%20%20%20%20-webkit-app-region:%20no-drag;%5Cn%20%20%20%20background:%20var(--background-modifier-form-field);%5Cn%20%20%20%20border:%20var(--input-border-width)%20solid%20var(--background-modifier-border);%5Cn%20%20%20%20color:%20var(--text-normal);%5Cn%20%20%20%20font-family:%20inherit;%5Cn%20%20%20%20padding:%20var(--size-4-1)%20var(--size-4-2);%5Cn%20%20%20%20font-size:%20var(--font-ui-small);%5Cn%20%20%20%20border-radius:%20var(--input-radius);%5Cn%20%20%20%20outline:%20none;%5Cn%5Cn%20%20%20%20@at-root%20%7B%5Cn%20%20%20%20%20%20@media%20(hover:%20hover)%20%7B%5Cn%20%20%20%20%20%20%20%20&:hover%20%7B%5Cn%20%20%20%20%20%20%20%20%20%20border-color:%20var(--background-modifier-border-hover);%5Cn%20%20%20%20%20%20%20%20%20%20transition:%5Cn%20%20%20%20%20%20%20%20%20%20%20%20box-shadow%200.15s%20ease-in-out,%5Cn%20%20%20%20%20%20%20%20%20%20%20%20border%200.15s%20ease-in-out;%5Cn%20%20%20%20%20%20%20%20%7D%5Cn%20%20%20%20%20%20%7D%5Cn%20%20%20%20%7D%5Cn%5Cn%20%20%20%20&:active,%5Cn%20%20%20%20&:focus%20%7B%5Cn%20%20%20%20%20%20border-color:%20var(--background-modifier-border-focus);%5Cn%20%20%20%20%20%20transition:%5Cn%20%20%20%20%20%20%20%20box-shadow%200.15s%20ease-in-out,%5Cn%20%20%20%20%20%20%20%20border%200.15s%20ease-in-out;%5Cn%20%20%20%20%7D%5Cn%5Cn%20%20%20%20&:active,%5Cn%20%20%20%20&:focus,%5Cn%20%20%20%20&:focus-visible%20%7B%5Cn%20%20%20%20%20%20box-shadow:%200%200%200%202px%20var(--background-modifier-border-focus);%5Cn%20%20%20%20%7D%5Cn%5Cn%20%20%20%20&::placeholder%20%7B%5Cn%20%20%20%20%20%20color:%20var(--text-faint);%5Cn%20%20%20%20%7D%5Cn%20%20%7D%5Cn%5Cn%20%20@at-root%20%7B%5Cn%20%20%20%20.mod-rtl,%5Cn%20%20%20%20.is-rtl,%5Cn%20%20%20%20.rtl%20%7B%5Cn%20%20%20%20%20%20&%20%7B%5Cn%20%20%20%20%20%20%20%20input%5Btype='month'%5D,%5Cn%20%20%20%20%20%20%20%20input%5Btype='time'%5D,%5Cn%20%20%20%20%20%20%20%20input%5Btype='week'%5D%20%7B%5Cn%20%20%20%20%20%20%20%20%20%20direction:%20rtl;%5Cn%5Cn%20%20%20%20%20%20%20%20%20%20&::-webkit-calendar-picker-indicator%20%7B%5Cn%20%20%20%20%20%20%20%20%20%20%20%20right:%20var(--size-4-1);%5Cn%20%20%20%20%20%20%20%20%20%20%20%20left:%20auto;%5Cn%20%20%20%20%20%20%20%20%20%20%7D%5Cn%20%20%20%20%20%20%20%20%7D%5Cn%20%20%20%20%20%20%7D%5Cn%20%20%20%20%7D%5Cn%20%20%7D%5Cn%7D%5Cn%22,%22.obsidian-dev-utils%20%7B%5Cn%20%20input%5Btype='month'%5D,%5Cn%20%20input%5Btype='time'%5D,%5Cn%20%20input%5Btype='week'%5D%20%7B%5Cn%20%20%20%20font-variant-numeric:%20tabular-nums;%5Cn%20%20%20%20position:%20relative;%5Cn%5Cn%20%20%20%20&::-webkit-datetime-edit-text%20%7B%5Cn%20%20%20%20%20%20color:%20var(--text-faint);%5Cn%20%20%20%20%20%20padding-inline-end:%200;%5Cn%20%20%20%20%7D%5Cn%5Cn%20%20%20%20&::-webkit-calendar-picker-indicator%20%7B%5Cn%20%20%20%20%20%20position:%20absolute;%5Cn%20%20%20%20%20%20left:%20var(--size-4-1);%5Cn%20%20%20%20%20%20right:%20auto;%5Cn%20%20%20%20%20%20opacity:%200.5;%5Cn%20%20%20%20%7D%5Cn%5Cn%20%20%20%20&::-webkit-datetime-edit-month-field,%5Cn%20%20%20%20&::-webkit-datetime-edit-day-field,%5Cn%20%20%20%20&::-webkit-datetime-edit-year-field%20%7B%5Cn%20%20%20%20%20%20&:active,%5Cn%20%20%20%20%20%20&:focus%20%7B%5Cn%20%20%20%20%20%20%20%20background-color:%20var(--text-selection);%5Cn%20%20%20%20%20%20%20%20color:%20var(--text-normal);%5Cn%20%20%20%20%20%20%20%20cursor:%20text;%5Cn%20%20%20%20%20%20%7D%5Cn%20%20%20%20%7D%5Cn%5Cn%20%20%20%20@at-root%20.mod-rtl%20&,%5Cn%20%20%20%20%20%20.is-rtl%20&,%5Cn%20%20%20%20%20%20.rtl%20&%20%7B%5Cn%20%20%20%20%20%20direction:%20rtl;%5Cn%5Cn%20%20%20%20%20%20&::-webkit-calendar-picker-indicator%20%7B%5Cn%20%20%20%20%20%20%20%20left:%20auto;%5Cn%20%20%20%20%20%20%20%20right:%20var(--size-4-1);%5Cn%20%20%20%20%20%20%7D%5Cn%20%20%20%20%7D%5Cn%5Cn%20%20%20%20@at-root%20%7B%5Cn%20%20%20%20%20%20body:not(.is-ios):not(.is-android)%20&%20%7B%5Cn%20%20%20%20%20%20%20%20padding-inline-start:%20var(--size-4-6);%5Cn%20%20%20%20%20%20%7D%5Cn%20%20%20%20%7D%5Cn%20%20%7D%5Cn%5Cn%20%20input%5Btype='time'%5D%20%7B%5Cn%20%20%20%20&::-webkit-calendar-picker-indicator%20%7B%5Cn%20%20%20%20%20%20margin-inline-start:%200;%5Cn%20%20%20%20%7D%5Cn%20%20%7D%5Cn%7D%5Cn%22,%22.obsidian-dev-utils%20%7B%5Cn%20%20&progress.loop%20%7B%5Cn%20%20%20%20min-width:%20200px;%5Cn%20%20%7D%5Cn%7D%5Cn%22,%22.obsidian-dev-utils%20%7B%5Cn%20%20&.modal-container%20%7B%5Cn%20%20%20%20.ok-button%20%7B%5Cn%20%20%20%20%20%20margin-right:%2010px;%5Cn%20%20%20%20%20%20margin-top:%2020px;%5Cn%20%20%20%20%7D%5Cn%20%20%7D%5Cn%7D%5Cn%22,%22.obsidian-dev-utils%20%7B%5Cn%20%20.multiple-dropdown-component%20%7B%5Cn%20%20%20%20select,%5Cn%20%20%20%20select:focus,%5Cn%20%20%20%20.dropdown%20%7B%5Cn%20%20%20%20%20%20height:%20auto;%5Cn%20%20%20%20%20%20padding-top:%203px;%5Cn%5Cn%20%20%20%20%20%20option:checked%20%7B%5Cn%20%20%20%20%20%20%20%20background-color:%20%231967d2;%5Cn%20%20%20%20%20%20%20%20color:%20%23fff;%5Cn%20%20%20%20%20%20%7D%5Cn%20%20%20%20%7D%5Cn%20%20%7D%5Cn%7D%5Cn%22,%22.obsidian-dev-utils%20%7B%5Cn%20%20&.plugin-settings-tab%20%7B%5Cn%20%20%20%20a:focus%20%7B%5Cn%20%20%20%20%20%20outline:%202px%20solid%20var(--link-color);%5Cn%20%20%20%20%7D%5Cn%20%20%7D%5Cn%7D%5Cn%22,%22.obsidian-dev-utils%20%7B%5Cn%20%20&.prompt-modal%20%7B%5Cn%20%20%20%20.text-box%20%7B%5Cn%20%20%20%20%20%20width:%20100%25;%5Cn%20%20%20%20%7D%5Cn%20%20%7D%5Cn%7D%5Cn%22,%22.obsidian-dev-utils%20%7B%5Cr%5Cn%20%20&.tri-state-checkbox-component%20%7B%5Cr%5Cn%20%20%20%20input%5Btype='checkbox'%5D:indeterminate%20%7B%5Cr%5Cn%20%20%20%20%20%20appearance:%20checkbox;%5Cr%5Cn%20%20%20%20%7D%5Cr%5Cn%20%20%7D%5Cr%5Cn%7D%5Cr%5Cn%22,%22@mixin%20invalid%20%7B%5Cn%20%20box-shadow:%200%200%200%202px%20var(--text-error);%5Cn%7D%5Cn%5Cn.obsidian-dev-utils%20%7B%5Cn%20%20:invalid%20%7B%5Cn%20%20%20%20@include%20invalid;%5Cn%20%20%7D%5Cn%5Cn%20%20input.metadata-input-text,%5Cn%20%20input%5Btype='date'%5D,%5Cn%20%20input%5Btype='datetime-local'%5D,%5Cn%20%20input%5Btype='email'%5D,%5Cn%20%20input%5Btype='number'%5D,%5Cn%20%20input%5Btype='password'%5D,%5Cn%20%20input%5Btype='search'%5D,%5Cn%20%20input%5Btype='text'%5D,%5Cn%20%20textarea%20%7B%5Cn%20%20%20%20&:active,%5Cn%20%20%20%20&:focus-visible,%5Cn%20%20%20%20&:focus%20%7B%5Cn%20%20%20%20%20%20&:invalid%20%7B%5Cn%20%20%20%20%20%20%20%20@include%20invalid;%5Cn%20%20%20%20%20%20%7D%5Cn%20%20%20%20%7D%5Cn%20%20%7D%5Cn%5Cn%20%20&.setting-component-wrapper%20%7B%5Cn%20%20%20%20position:%20relative;%5Cn%20%20%20%20display:%20inline-flex;%5Cn%20%20%7D%5Cn%5Cn%20%20&.overlay-validator%20%7B%5Cn%20%20%20%20caret-color:%20transparent;%5Cn%20%20%20%20cursor:%20default;%5Cn%20%20%20%20position:%20absolute;%5Cn%20%20%20%20background-color:%20transparent;%5Cn%20%20%20%20border:%20none;%5Cn%20%20%20%20outline:%20none;%5Cn%20%20%20%20pointer-events:%20none;%5Cn%20%20%20%20z-index:%209999;%5Cn%20%20%20%20left:%200;%5Cn%20%20%20%20top:%200;%5Cn%20%20%20%20width:%20100%25;%5Cn%20%20%20%20height:%20100%25;%5Cn%20%20%7D%5Cn%5Cn%20%20&.tooltip.tooltip-validator%20%7B%5Cn%20%20%20%20position:%20absolute;%5Cn%20%20%20%20top:%20calc(100%25%20+%208px);%5Cn%20%20%20%20width:%20max-content;%5Cn%20%20%7D%5Cn%7D%5Cn%22%5D%7D */\n";
|
|
136
136
|
// Annotate the CommonJS export names for ESM import in node:
|
|
@@ -0,0 +1,163 @@
|
|
|
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() {
|
|
7
|
+
// eslint-disable-next-line obsidianmd/no-global-this -- Actively use globalThis.
|
|
8
|
+
const globalThisRecord = globalThis;
|
|
9
|
+
globalThisRecord['__name'] ??= name;
|
|
10
|
+
const originalRequire = require;
|
|
11
|
+
if (originalRequire && !originalRequire.__isPatched) {
|
|
12
|
+
// eslint-disable-next-line no-global-assign, no-implicit-globals -- We need to patch the `require()` function.
|
|
13
|
+
require = Object.assign(
|
|
14
|
+
(id) => requirePatched(id),
|
|
15
|
+
originalRequire,
|
|
16
|
+
{
|
|
17
|
+
__isPatched: true
|
|
18
|
+
}
|
|
19
|
+
);
|
|
20
|
+
}
|
|
21
|
+
|
|
22
|
+
const newFuncs = {
|
|
23
|
+
__extractDefault() {
|
|
24
|
+
return extractDefault;
|
|
25
|
+
},
|
|
26
|
+
process() {
|
|
27
|
+
const browserProcess = {
|
|
28
|
+
browser: true,
|
|
29
|
+
cwd() {
|
|
30
|
+
return '/';
|
|
31
|
+
},
|
|
32
|
+
env: {},
|
|
33
|
+
platform: 'android'
|
|
34
|
+
};
|
|
35
|
+
return browserProcess;
|
|
36
|
+
}
|
|
37
|
+
};
|
|
38
|
+
|
|
39
|
+
for (const key of Object.keys(newFuncs)) {
|
|
40
|
+
globalThisRecord[key] ??= newFuncs[key]?.();
|
|
41
|
+
}
|
|
42
|
+
|
|
43
|
+
function name(obj) {
|
|
44
|
+
return obj;
|
|
45
|
+
}
|
|
46
|
+
|
|
47
|
+
function extractDefault(module) {
|
|
48
|
+
return module && module.__esModule && 'default' in module ? module.default : module;
|
|
49
|
+
}
|
|
50
|
+
|
|
51
|
+
const OBSIDIAN_BUILT_IN_MODULE_NAMES = [
|
|
52
|
+
'obsidian',
|
|
53
|
+
'@codemirror/autocomplete',
|
|
54
|
+
'@codemirror/collab',
|
|
55
|
+
'@codemirror/commands',
|
|
56
|
+
'@codemirror/language',
|
|
57
|
+
'@codemirror/lint',
|
|
58
|
+
'@codemirror/search',
|
|
59
|
+
'@codemirror/state',
|
|
60
|
+
'@codemirror/text',
|
|
61
|
+
'@codemirror/view',
|
|
62
|
+
'@lezer/common',
|
|
63
|
+
'@lezer/lr',
|
|
64
|
+
'@lezer/highlight'];
|
|
65
|
+
|
|
66
|
+
|
|
67
|
+
const DEPRECATED_OBSIDIAN_BUILT_IN_MODULE_NAMES = [
|
|
68
|
+
'@codemirror/closebrackets',
|
|
69
|
+
'@codemirror/comment',
|
|
70
|
+
'@codemirror/fold',
|
|
71
|
+
'@codemirror/gutter',
|
|
72
|
+
'@codemirror/highlight',
|
|
73
|
+
'@codemirror/history',
|
|
74
|
+
'@codemirror/matchbrackets',
|
|
75
|
+
'@codemirror/panel',
|
|
76
|
+
'@codemirror/rangeset',
|
|
77
|
+
'@codemirror/rectangular-selection',
|
|
78
|
+
'@codemirror/stream-parser',
|
|
79
|
+
'@codemirror/tooltip'];
|
|
80
|
+
|
|
81
|
+
|
|
82
|
+
function requirePatched(id) {
|
|
83
|
+
if (OBSIDIAN_BUILT_IN_MODULE_NAMES.includes(id) || DEPRECATED_OBSIDIAN_BUILT_IN_MODULE_NAMES.includes(id)) {
|
|
84
|
+
return originalRequire?.(id);
|
|
85
|
+
}
|
|
86
|
+
|
|
87
|
+
// eslint-disable-next-line @typescript-eslint/no-deprecated, obsidianmd/no-global-this -- Need access to app. Actively use globalThis.
|
|
88
|
+
if (globalThis.app.isMobile) {
|
|
89
|
+
if (id === 'process' || id === 'node:process') {
|
|
90
|
+
// eslint-disable-next-line no-console -- Valid usage.
|
|
91
|
+
console.debug(`The most likely you can safely ignore this error. Module not found: ${id}. Fake process object is returned instead.`);
|
|
92
|
+
// eslint-disable-next-line obsidianmd/no-global-this -- Actively use globalThis.
|
|
93
|
+
return globalThis.process;
|
|
94
|
+
}
|
|
95
|
+
} else {
|
|
96
|
+
const module = originalRequire?.(id);
|
|
97
|
+
if (module) {
|
|
98
|
+
return extractDefault(module);
|
|
99
|
+
}
|
|
100
|
+
}
|
|
101
|
+
|
|
102
|
+
// eslint-disable-next-line no-console -- Valid usage.
|
|
103
|
+
console.debug(`The most likely you can safely ignore this error. Module not found: ${id}. Empty object is returned instead.`);
|
|
104
|
+
return {};
|
|
105
|
+
}
|
|
106
|
+
})();
|
|
107
|
+
|
|
108
|
+
"use strict";
|
|
109
|
+
var __defProp = Object.defineProperty;
|
|
110
|
+
var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
|
|
111
|
+
var __getOwnPropNames = Object.getOwnPropertyNames;
|
|
112
|
+
var __hasOwnProp = Object.prototype.hasOwnProperty;
|
|
113
|
+
var __export = (target, all) => {
|
|
114
|
+
for (var name in all)
|
|
115
|
+
__defProp(target, name, { get: all[name], enumerable: true });
|
|
116
|
+
};
|
|
117
|
+
var __copyProps = (to, from, except, desc) => {
|
|
118
|
+
if (from && typeof from === "object" || typeof from === "function") {
|
|
119
|
+
for (let key of __getOwnPropNames(from))
|
|
120
|
+
if (!__hasOwnProp.call(to, key) && key !== except)
|
|
121
|
+
__defProp(to, key, { get: () => from[key], enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable });
|
|
122
|
+
}
|
|
123
|
+
return to;
|
|
124
|
+
};
|
|
125
|
+
var __toCommonJS = (mod) => __copyProps(__defProp({}, "__esModule", { value: true }), mod);
|
|
126
|
+
var data_handler_exports = {};
|
|
127
|
+
__export(data_handler_exports, {
|
|
128
|
+
PluginDataHandler: () => PluginDataHandler
|
|
129
|
+
});
|
|
130
|
+
module.exports = __toCommonJS(data_handler_exports);
|
|
131
|
+
class PluginDataHandler {
|
|
132
|
+
/**
|
|
133
|
+
* Creates a new plugin data handler.
|
|
134
|
+
*
|
|
135
|
+
* @param plugin - The plugin instance.
|
|
136
|
+
*/
|
|
137
|
+
constructor(plugin) {
|
|
138
|
+
this.plugin = plugin;
|
|
139
|
+
}
|
|
140
|
+
plugin;
|
|
141
|
+
/**
|
|
142
|
+
* Loads data from the plugin's data file.
|
|
143
|
+
*
|
|
144
|
+
* @returns The loaded data.
|
|
145
|
+
*/
|
|
146
|
+
async loadData() {
|
|
147
|
+
return this.plugin.loadData();
|
|
148
|
+
}
|
|
149
|
+
/**
|
|
150
|
+
* Saves data to the plugin's data file.
|
|
151
|
+
*
|
|
152
|
+
* @param data - The data to save.
|
|
153
|
+
* @returns A promise that resolves when the data is saved.
|
|
154
|
+
*/
|
|
155
|
+
async saveData(data) {
|
|
156
|
+
return this.plugin.saveData(data);
|
|
157
|
+
}
|
|
158
|
+
}
|
|
159
|
+
// Annotate the CommonJS export names for ESM import in node:
|
|
160
|
+
0 && (module.exports = {
|
|
161
|
+
PluginDataHandler
|
|
162
|
+
});
|
|
163
|
+
//# sourceMappingURL=data:application/json;base64,ewogICJ2ZXJzaW9uIjogMywKICAic291cmNlcyI6IFsiLi4vLi4vLi4vLi4vc3JjL29ic2lkaWFuL2RhdGEtaGFuZGxlci50cyJdLAogICJzb3VyY2VzQ29udGVudCI6IFsiLyoqXG4gKiBAZmlsZVxuICpcbiAqIEEgbW9kdWxlIGZvciBoYW5kbGluZyBkYXRhIGxvYWRpbmcgYW5kIHNhdmluZyB0byB0aGUgcGx1Z2luJ3MgZGF0YSBmaWxlLlxuICovXG5cbmltcG9ydCB0eXBlIHsgUGx1Z2luIH0gZnJvbSAnb2JzaWRpYW4nO1xuXG4vKipcbiAqIEEgaGFuZGxlciBmb3IgbG9hZGluZyBhbmQgc2F2aW5nIGRhdGEgdG8gdGhlIHBsdWdpbidzIGRhdGEgZmlsZS5cbiAqL1xuZXhwb3J0IGludGVyZmFjZSBEYXRhSGFuZGxlciB7XG4gIC8qKlxuICAgKiBBIGZ1bmN0aW9uIHRvIGxvYWQgZGF0YSBmcm9tIHRoZSBwbHVnaW4ncyBkYXRhIGZpbGUuXG4gICAqXG4gICAqIEByZXR1cm5zIFRoZSBsb2FkZWQgZGF0YS5cbiAgICovXG4gIGxvYWREYXRhKCk6IFByb21pc2U8dW5rbm93bj47XG5cbiAgLyoqXG4gICAqIEEgZnVuY3Rpb24gdG8gc2F2ZSBkYXRhIHRvIHRoZSBwbHVnaW4ncyBkYXRhIGZpbGUuXG4gICAqXG4gICAqIEBwYXJhbSBkYXRhIC0gVGhlIGRhdGEgdG8gc2F2ZS5cbiAgICovXG4gIHNhdmVEYXRhKGRhdGE6IHVua25vd24pOiBQcm9taXNlPHZvaWQ+O1xufVxuXG4vKipcbiAqIEEgaGFuZGxlciBmb3IgbG9hZGluZyBhbmQgc2F2aW5nIGRhdGEgdG8gdGhlIHBsdWdpbidzIGRhdGEgZmlsZS5cbiAqL1xuZXhwb3J0IGNsYXNzIFBsdWdpbkRhdGFIYW5kbGVyIGltcGxlbWVudHMgRGF0YUhhbmRsZXIge1xuICAvKipcbiAgICogQ3JlYXRlcyBhIG5ldyBwbHVnaW4gZGF0YSBoYW5kbGVyLlxuICAgKlxuICAgKiBAcGFyYW0gcGx1Z2luIC0gVGhlIHBsdWdpbiBpbnN0YW5jZS5cbiAgICovXG4gIHB1YmxpYyBjb25zdHJ1Y3Rvcihwcml2YXRlIHJlYWRvbmx5IHBsdWdpbjogUGx1Z2luKSB7XG4gIH1cblxuICAvKipcbiAgICogTG9hZHMgZGF0YSBmcm9tIHRoZSBwbHVnaW4ncyBkYXRhIGZpbGUuXG4gICAqXG4gICAqIEByZXR1cm5zIFRoZSBsb2FkZWQgZGF0YS5cbiAgICovXG4gIHB1YmxpYyBhc3luYyBsb2FkRGF0YSgpOiBQcm9taXNlPHVua25vd24+IHtcbiAgICByZXR1cm4gdGhpcy5wbHVnaW4ubG9hZERhdGEoKTtcbiAgfVxuXG4gIC8qKlxuICAgKiBTYXZlcyBkYXRhIHRvIHRoZSBwbHVnaW4ncyBkYXRhIGZpbGUuXG4gICAqXG4gICAqIEBwYXJhbSBkYXRhIC0gVGhlIGRhdGEgdG8gc2F2ZS5cbiAgICogQHJldHVybnMgQSBwcm9taXNlIHRoYXQgcmVzb2x2ZXMgd2hlbiB0aGUgZGF0YSBpcyBzYXZlZC5cbiAgICovXG4gIHB1YmxpYyBhc3luYyBzYXZlRGF0YShkYXRhOiB1bmtub3duKTogUHJvbWlzZTx2b2lkPiB7XG4gICAgcmV0dXJuIHRoaXMucGx1Z2luLnNhdmVEYXRhKGRhdGEpO1xuICB9XG59XG4iXSwKICAibWFwcGluZ3MiOiAiOzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7OztBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUE4Qk8sTUFBTSxrQkFBeUM7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUEsRUFNN0MsWUFBNkIsUUFBZ0I7QUFBaEI7QUFBQSxFQUNwQztBQUFBLEVBRG9DO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBLEVBUXBDLE1BQWEsV0FBNkI7QUFDeEMsV0FBTyxLQUFLLE9BQU8sU0FBUztBQUFBLEVBQzlCO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUEsRUFRQSxNQUFhLFNBQVMsTUFBOEI7QUFDbEQsV0FBTyxLQUFLLE9BQU8sU0FBUyxJQUFJO0FBQUEsRUFDbEM7QUFDRjsiLAogICJuYW1lcyI6IFtdCn0K
|
|
@@ -0,0 +1,48 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* @file
|
|
3
|
+
*
|
|
4
|
+
* A module for handling data loading and saving to the plugin's data file.
|
|
5
|
+
*/
|
|
6
|
+
import type { Plugin } from 'obsidian';
|
|
7
|
+
/**
|
|
8
|
+
* A handler for loading and saving data to the plugin's data file.
|
|
9
|
+
*/
|
|
10
|
+
export interface DataHandler {
|
|
11
|
+
/**
|
|
12
|
+
* A function to load data from the plugin's data file.
|
|
13
|
+
*
|
|
14
|
+
* @returns The loaded data.
|
|
15
|
+
*/
|
|
16
|
+
loadData(): Promise<unknown>;
|
|
17
|
+
/**
|
|
18
|
+
* A function to save data to the plugin's data file.
|
|
19
|
+
*
|
|
20
|
+
* @param data - The data to save.
|
|
21
|
+
*/
|
|
22
|
+
saveData(data: unknown): Promise<void>;
|
|
23
|
+
}
|
|
24
|
+
/**
|
|
25
|
+
* A handler for loading and saving data to the plugin's data file.
|
|
26
|
+
*/
|
|
27
|
+
export declare class PluginDataHandler implements DataHandler {
|
|
28
|
+
private readonly plugin;
|
|
29
|
+
/**
|
|
30
|
+
* Creates a new plugin data handler.
|
|
31
|
+
*
|
|
32
|
+
* @param plugin - The plugin instance.
|
|
33
|
+
*/
|
|
34
|
+
constructor(plugin: Plugin);
|
|
35
|
+
/**
|
|
36
|
+
* Loads data from the plugin's data file.
|
|
37
|
+
*
|
|
38
|
+
* @returns The loaded data.
|
|
39
|
+
*/
|
|
40
|
+
loadData(): Promise<unknown>;
|
|
41
|
+
/**
|
|
42
|
+
* Saves data to the plugin's data file.
|
|
43
|
+
*
|
|
44
|
+
* @param data - The data to save.
|
|
45
|
+
* @returns A promise that resolves when the data is saved.
|
|
46
|
+
*/
|
|
47
|
+
saveData(data: unknown): Promise<void>;
|
|
48
|
+
}
|
|
@@ -146,6 +146,7 @@ __export(obsidian_exports, {
|
|
|
146
146
|
command_registrar: () => command_registrar,
|
|
147
147
|
components: () => components,
|
|
148
148
|
constructors: () => constructors,
|
|
149
|
+
data_handler: () => data_handler,
|
|
149
150
|
dataview: () => dataview,
|
|
150
151
|
dataview_link: () => dataview_link,
|
|
151
152
|
editor: () => editor,
|
|
@@ -193,6 +194,7 @@ var command_handlers = __toESM(__extractDefault(require('./command-handlers/inde
|
|
|
193
194
|
var command_registrar = __toESM(__extractDefault(require('./command-registrar.cjs')), 1);
|
|
194
195
|
var components = __toESM(__extractDefault(require('./components/index.cjs')), 1);
|
|
195
196
|
var constructors = __toESM(__extractDefault(require('./constructors/index.cjs')), 1);
|
|
197
|
+
var data_handler = __toESM(__extractDefault(require('./data-handler.cjs')), 1);
|
|
196
198
|
var dataview_link = __toESM(__extractDefault(require('./dataview-link.cjs')), 1);
|
|
197
199
|
var dataview = __toESM(__extractDefault(require('./dataview.cjs')), 1);
|
|
198
200
|
var editor = __toESM(__extractDefault(require('./editor.cjs')), 1);
|
|
@@ -240,6 +242,7 @@ var workspace = __toESM(__extractDefault(require('./workspace.cjs')), 1);
|
|
|
240
242
|
command_registrar,
|
|
241
243
|
components,
|
|
242
244
|
constructors,
|
|
245
|
+
data_handler,
|
|
243
246
|
dataview,
|
|
244
247
|
dataview_link,
|
|
245
248
|
editor,
|
|
@@ -275,4 +278,4 @@ var workspace = __toESM(__extractDefault(require('./workspace.cjs')), 1);
|
|
|
275
278
|
vault_delete,
|
|
276
279
|
workspace
|
|
277
280
|
});
|
|
278
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
281
|
+
//# sourceMappingURL=data:application/json;base64,ewogICJ2ZXJzaW9uIjogMywKICAic291cmNlcyI6IFsiLi4vLi4vLi4vLi4vc3JjL29ic2lkaWFuL2luZGV4LnRzIl0sCiAgInNvdXJjZXNDb250ZW50IjogWyIvKiBUSElTIElTIEEgR0VORVJBVEVEL0JVTkRMRUQgRklMRSBCWSBCVUlMRCBTQ1JJUFQgKi9cblxuZXhwb3J0ICogYXMgYWN0aXZlX2ZpbGVfcHJvdmlkZXIgZnJvbSAnLi9hY3RpdmUtZmlsZS1wcm92aWRlci50cyc7XG5leHBvcnQgKiBhcyBhcHAgZnJvbSAnLi9hcHAudHMnO1xuZXhwb3J0ICogYXMgYXN5bmNfd2l0aF9ub3RpY2UgZnJvbSAnLi9hc3luYy13aXRoLW5vdGljZS50cyc7XG5leHBvcnQgKiBhcyBhdHRhY2htZW50X3BhdGggZnJvbSAnLi9hdHRhY2htZW50LXBhdGgudHMnO1xuZXhwb3J0ICogYXMgYmFja2xpbmsgZnJvbSAnLi9iYWNrbGluay50cyc7XG5leHBvcnQgKiBhcyBjYWxsb3V0IGZyb20gJy4vY2FsbG91dC50cyc7XG5leHBvcnQgKiBhcyBjb2RlX2Jsb2NrX21hcmtkb3duX2luZm9ybWF0aW9uIGZyb20gJy4vY29kZS1ibG9jay1tYXJrZG93bi1pbmZvcm1hdGlvbi50cyc7XG5leHBvcnQgKiBhcyBjb21tYW5kX2hhbmRsZXJzIGZyb20gJy4vY29tbWFuZC1oYW5kbGVycy9pbmRleC50cyc7XG5leHBvcnQgKiBhcyBjb21tYW5kX3JlZ2lzdHJhciBmcm9tICcuL2NvbW1hbmQtcmVnaXN0cmFyLnRzJztcbmV4cG9ydCAqIGFzIGNvbXBvbmVudHMgZnJvbSAnLi9jb21wb25lbnRzL2luZGV4LnRzJztcbmV4cG9ydCAqIGFzIGNvbnN0cnVjdG9ycyBmcm9tICcuL2NvbnN0cnVjdG9ycy9pbmRleC50cyc7XG5leHBvcnQgKiBhcyBkYXRhX2hhbmRsZXIgZnJvbSAnLi9kYXRhLWhhbmRsZXIudHMnO1xuZXhwb3J0ICogYXMgZGF0YXZpZXdfbGluayBmcm9tICcuL2RhdGF2aWV3LWxpbmsudHMnO1xuZXhwb3J0ICogYXMgZGF0YXZpZXcgZnJvbSAnLi9kYXRhdmlldy50cyc7XG5leHBvcnQgKiBhcyBlZGl0b3IgZnJvbSAnLi9lZGl0b3IudHMnO1xuZXhwb3J0ICogYXMgZmlsZV9jaGFuZ2UgZnJvbSAnLi9maWxlLWNoYW5nZS50cyc7XG5leHBvcnQgKiBhcyBmaWxlX21hbmFnZXIgZnJvbSAnLi9maWxlLW1hbmFnZXIudHMnO1xuZXhwb3J0ICogYXMgZmlsZV9zeXN0ZW0gZnJvbSAnLi9maWxlLXN5c3RlbS50cyc7XG5leHBvcnQgKiBhcyBmcm9udG1hdHRlcl9saW5rX2NhY2hlX3dpdGhfb2Zmc2V0cyBmcm9tICcuL2Zyb250bWF0dGVyLWxpbmstY2FjaGUtd2l0aC1vZmZzZXRzLnRzJztcbmV4cG9ydCAqIGFzIGZyb250bWF0dGVyIGZyb20gJy4vZnJvbnRtYXR0ZXIudHMnO1xuZXhwb3J0ICogYXMgaTE4biBmcm9tICcuL2kxOG4vaW5kZXgudHMnO1xuZXhwb3J0ICogYXMgaXNfaW5fb2JzaWRpYW4gZnJvbSAnLi9pcy1pbi1vYnNpZGlhbi50cyc7XG5leHBvcnQgKiBhcyBsaW5rIGZyb20gJy4vbGluay50cyc7XG5leHBvcnQgKiBhcyBsb2dnZXIgZnJvbSAnLi9sb2dnZXIudHMnO1xuZXhwb3J0ICogYXMgbG9vcCBmcm9tICcuL2xvb3AudHMnO1xuZXhwb3J0ICogYXMgbWFya2Rvd25fY29kZV9ibG9ja19wcm9jZXNzb3IgZnJvbSAnLi9tYXJrZG93bi1jb2RlLWJsb2NrLXByb2Nlc3Nvci50cyc7XG5leHBvcnQgKiBhcyBtYXJrZG93bl92aWV3IGZyb20gJy4vbWFya2Rvd24tdmlldy50cyc7XG5leHBvcnQgKiBhcyBtYXJrZG93biBmcm9tICcuL21hcmtkb3duLnRzJztcbmV4cG9ydCAqIGFzIG1lbnVfZXZlbnRfcmVnaXN0cmFyIGZyb20gJy4vbWVudS1ldmVudC1yZWdpc3RyYXIudHMnO1xuZXhwb3J0ICogYXMgbWV0YWRhdGFfY2FjaGUgZnJvbSAnLi9tZXRhZGF0YS1jYWNoZS50cyc7XG5leHBvcnQgKiBhcyBtb2RhbHMgZnJvbSAnLi9tb2RhbHMvaW5kZXgudHMnO1xuZXhwb3J0ICogYXMgbW9ua2V5X2Fyb3VuZCBmcm9tICcuL21vbmtleS1hcm91bmQudHMnO1xuZXhwb3J0ICogYXMgb2JzaWRpYW5fc2V0dGluZ3MgZnJvbSAnLi9vYnNpZGlhbi1zZXR0aW5ncy50cyc7XG5leHBvcnQgKiBhcyBwZGYgZnJvbSAnLi9wZGYudHMnO1xuZXhwb3J0ICogYXMgcGx1Z2luIGZyb20gJy4vcGx1Z2luL2luZGV4LnRzJztcbmV4cG9ydCAqIGFzIHF1ZXVlIGZyb20gJy4vcXVldWUudHMnO1xuZXhwb3J0ICogYXMgcmVhY3QgZnJvbSAnLi9yZWFjdC9pbmRleC50cyc7XG5leHBvcnQgKiBhcyByZWZlcmVuY2UgZnJvbSAnLi9yZWZlcmVuY2UudHMnO1xuZXhwb3J0ICogYXMgcmVuYW1lX2RlbGV0ZV9oYW5kbGVyIGZyb20gJy4vcmVuYW1lLWRlbGV0ZS1oYW5kbGVyLnRzJztcbmV4cG9ydCAqIGFzIHJlc291cmNlX3VybCBmcm9tICcuL3Jlc291cmNlLXVybC50cyc7XG5leHBvcnQgKiBhcyBzZXR0aW5nX2V4IGZyb20gJy4vc2V0dGluZy1leC50cyc7XG5leHBvcnQgKiBhcyBzZXR0aW5nX2dyb3VwX2V4IGZyb20gJy4vc2V0dGluZy1ncm91cC1leC50cyc7XG5leHBvcnQgKiBhcyB2YWxpZGF0aW9uIGZyb20gJy4vdmFsaWRhdGlvbi50cyc7XG5leHBvcnQgKiBhcyB2YXVsdF9kZWxldGUgZnJvbSAnLi92YXVsdC1kZWxldGUudHMnO1xuZXhwb3J0ICogYXMgdmF1bHQgZnJvbSAnLi92YXVsdC50cyc7XG5leHBvcnQgKiBhcyB3b3Jrc3BhY2UgZnJvbSAnLi93b3Jrc3BhY2UudHMnO1xuIl0sCiAgIm1hcHBpbmdzIjogIjs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7O0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUVBLDJCQUFzQztBQUN0QyxVQUFxQjtBQUNyQix3QkFBbUM7QUFDbkMsc0JBQWlDO0FBQ2pDLGVBQTBCO0FBQzFCLGNBQXlCO0FBQ3pCLHNDQUFpRDtBQUNqRCx1QkFBa0M7QUFDbEMsd0JBQW1DO0FBQ25DLGlCQUE0QjtBQUM1QixtQkFBOEI7QUFDOUIsbUJBQThCO0FBQzlCLG9CQUErQjtBQUMvQixlQUEwQjtBQUMxQixhQUF3QjtBQUN4QixrQkFBNkI7QUFDN0IsbUJBQThCO0FBQzlCLGtCQUE2QjtBQUM3QiwwQ0FBcUQ7QUFDckQsa0JBQTZCO0FBQzdCLFdBQXNCO0FBQ3RCLHFCQUFnQztBQUNoQyxXQUFzQjtBQUN0QixhQUF3QjtBQUN4QixXQUFzQjtBQUN0QixvQ0FBK0M7QUFDL0Msb0JBQStCO0FBQy9CLGVBQTBCO0FBQzFCLDJCQUFzQztBQUN0QyxxQkFBZ0M7QUFDaEMsYUFBd0I7QUFDeEIsb0JBQStCO0FBQy9CLHdCQUFtQztBQUNuQyxVQUFxQjtBQUNyQixhQUF3QjtBQUN4QixZQUF1QjtBQUN2QixZQUF1QjtBQUN2QixnQkFBMkI7QUFDM0IsNEJBQXVDO0FBQ3ZDLG1CQUE4QjtBQUM5QixpQkFBNEI7QUFDNUIsdUJBQWtDO0FBQ2xDLGlCQUE0QjtBQUM1QixtQkFBOEI7QUFDOUIsWUFBdUI7QUFDdkIsZ0JBQTJCOyIsCiAgIm5hbWVzIjogW10KfQo=
|
|
@@ -9,6 +9,7 @@ export * as command_handlers from './command-handlers/index.cjs';
|
|
|
9
9
|
export * as command_registrar from './command-registrar.cjs';
|
|
10
10
|
export * as components from './components/index.cjs';
|
|
11
11
|
export * as constructors from './constructors/index.cjs';
|
|
12
|
+
export * as data_handler from './data-handler.cjs';
|
|
12
13
|
export * as dataview_link from './dataview-link.cjs';
|
|
13
14
|
export * as dataview from './dataview.cjs';
|
|
14
15
|
export * as editor from './editor.cjs';
|
|
@@ -150,7 +150,34 @@ const defaultTransformer = new import_group_transformer.GroupTransformer([
|
|
|
150
150
|
new import_set_transformer.SetTransformer(),
|
|
151
151
|
new import_two_way_map_transformer.TwoWayMapTransformer()
|
|
152
152
|
]);
|
|
153
|
+
class EmptyDataHandler {
|
|
154
|
+
async loadData() {
|
|
155
|
+
await (0, import_function.noopAsync)();
|
|
156
|
+
return {};
|
|
157
|
+
}
|
|
158
|
+
/* v8 ignore start -- unreachable because EmptyPluginSettingsComponent has no properties to change, so saveToFile always short-circuits. */
|
|
159
|
+
async saveData() {
|
|
160
|
+
await (0, import_function.noopAsync)();
|
|
161
|
+
}
|
|
162
|
+
/* v8 ignore stop */
|
|
163
|
+
}
|
|
153
164
|
class PluginSettingsComponentBase extends import_async_component.AsyncComponentBase {
|
|
165
|
+
/**
|
|
166
|
+
* Creates a new plugin settings component.
|
|
167
|
+
*
|
|
168
|
+
* @param dataHandler - The data handler.
|
|
169
|
+
*/
|
|
170
|
+
constructor(dataHandler) {
|
|
171
|
+
super();
|
|
172
|
+
this.dataHandler = dataHandler;
|
|
173
|
+
this.defaultSettings = this.createDefaultSettings();
|
|
174
|
+
this.currentState = this.createDefaultState();
|
|
175
|
+
this.lastSavedState = this.createDefaultState();
|
|
176
|
+
this.propertyNames = (0, import_object_utils.getAllKeys)(this.currentState.inputValues);
|
|
177
|
+
this.registerValidators();
|
|
178
|
+
this.registerLegacySettingsConverters();
|
|
179
|
+
}
|
|
180
|
+
dataHandler;
|
|
154
181
|
/**
|
|
155
182
|
* Component key for {@link registerComponent} replacement.
|
|
156
183
|
*/
|
|
@@ -181,26 +208,8 @@ class PluginSettingsComponentBase extends import_async_component.AsyncComponentB
|
|
|
181
208
|
currentState;
|
|
182
209
|
lastSavedState;
|
|
183
210
|
legacySettingsConverters = [];
|
|
184
|
-
loadDataFn;
|
|
185
211
|
propertyNames;
|
|
186
|
-
saveDataFn;
|
|
187
212
|
validators = /* @__PURE__ */ new Map();
|
|
188
|
-
/**
|
|
189
|
-
* Creates a new plugin settings component.
|
|
190
|
-
*
|
|
191
|
-
* @param params - The params for loading/saving data.
|
|
192
|
-
*/
|
|
193
|
-
constructor(params) {
|
|
194
|
-
super();
|
|
195
|
-
this.loadDataFn = params.loadData.bind(params);
|
|
196
|
-
this.saveDataFn = params.saveData.bind(params);
|
|
197
|
-
this.defaultSettings = this.createDefaultSettings();
|
|
198
|
-
this.currentState = this.createDefaultState();
|
|
199
|
-
this.lastSavedState = this.createDefaultState();
|
|
200
|
-
this.propertyNames = (0, import_object_utils.getAllKeys)(this.currentState.inputValues);
|
|
201
|
-
this.registerValidators();
|
|
202
|
-
this.registerLegacySettingsConverters();
|
|
203
|
-
}
|
|
204
213
|
/**
|
|
205
214
|
* Edits the plugin settings and saves them.
|
|
206
215
|
*
|
|
@@ -247,7 +256,7 @@ class PluginSettingsComponentBase extends import_async_component.AsyncComponentB
|
|
|
247
256
|
* @returns A {@link Promise} that resolves when the settings are loaded.
|
|
248
257
|
*/
|
|
249
258
|
async loadFromFile(isInitialLoad) {
|
|
250
|
-
const data = await this.
|
|
259
|
+
const data = await this.dataHandler.loadData();
|
|
251
260
|
this.lastSavedState = this.createDefaultState();
|
|
252
261
|
this.currentState = this.createDefaultState();
|
|
253
262
|
try {
|
|
@@ -512,7 +521,7 @@ class PluginSettingsComponentBase extends import_async_component.AsyncComponentB
|
|
|
512
521
|
return settings;
|
|
513
522
|
}
|
|
514
523
|
async saveToFileImpl() {
|
|
515
|
-
await this.
|
|
524
|
+
await this.dataHandler.saveData(await this.settingsToRawRecord(this.currentState.inputValues));
|
|
516
525
|
}
|
|
517
526
|
setPropertyImpl(propertyName, value, validationMessage) {
|
|
518
527
|
const defaults = this.defaultSettings;
|
|
@@ -532,19 +541,9 @@ class PluginSettingsComponentBase extends import_async_component.AsyncComponentB
|
|
|
532
541
|
class EmptyPluginSettingsComponent extends PluginSettingsComponentBase {
|
|
533
542
|
/**
|
|
534
543
|
* Creates a new empty plugin settings component.
|
|
535
|
-
*
|
|
536
|
-
* @param params - The params. Uses no-op load/save by default.
|
|
537
544
|
*/
|
|
538
|
-
constructor(
|
|
539
|
-
super(
|
|
540
|
-
params ?? {
|
|
541
|
-
loadData: async () => {
|
|
542
|
-
await (0, import_function.noopAsync)();
|
|
543
|
-
return {};
|
|
544
|
-
},
|
|
545
|
-
saveData: import_function.noopAsync
|
|
546
|
-
}
|
|
547
|
-
);
|
|
545
|
+
constructor() {
|
|
546
|
+
super(new EmptyDataHandler());
|
|
548
547
|
}
|
|
549
548
|
/**
|
|
550
549
|
* Creates empty default settings.
|
|
@@ -560,4 +559,4 @@ class EmptyPluginSettingsComponent extends PluginSettingsComponentBase {
|
|
|
560
559
|
EmptyPluginSettingsComponent,
|
|
561
560
|
PluginSettingsComponentBase
|
|
562
561
|
});
|
|
563
|
-
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../../../../src/obsidian/plugin/components/plugin-settings-component.ts"],
  "sourcesContent": ["/**\n * @file\n *\n * Component that manages plugin settings: data, persistence, validation, and events.\n *\n * This is the single public-facing settings API for plugin authors.\n * All settings logic (load, save, validate, transform, clone) is handled directly\n * by this component \u2014 no separate manager class is needed.\n */\n\nimport type {\n  Promisable,\n  ReadonlyDeep\n} from 'type-fest';\n\nimport type { AsyncEventRef } from '../../../async-events.ts';\nimport type { Transformer } from '../../../transformers/transformer.ts';\nimport type { GenericObject } from '../../../type-guards.ts';\nimport type {\n  MaybeReturn,\n  StringKeys\n} from '../../../type.ts';\n\nimport { AsyncEvents } from '../../../async-events.ts';\nimport { getLibDebugger } from '../../../debug.ts';\nimport {\n  noop,\n  noopAsync\n} from '../../../function.ts';\nimport {\n  castTo,\n  deepEqual,\n  getAllKeys\n} from '../../../object-utils.ts';\nimport { DateTransformer } from '../../../transformers/date-transformer.ts';\nimport { DurationTransformer } from '../../../transformers/duration-transformer.ts';\nimport { GroupTransformer } from '../../../transformers/group-transformer.ts';\nimport { MapTransformer } from '../../../transformers/map-transformer.ts';\nimport { SetTransformer } from '../../../transformers/set-transformer.ts';\nimport { SkipPrivatePropertyTransformer } from '../../../transformers/skip-private-property-transformer.ts';\nimport { TwoWayMapTransformer } from '../../../transformers/two-way-map-transformer.ts';\nimport { AsyncComponentBase } from '../../components/async-component.ts';\nimport { registerAsyncEvent } from '../../components/async-events-component.ts';\n\nconst defaultTransformer = new GroupTransformer([\n  new SkipPrivatePropertyTransformer(),\n  new DateTransformer(),\n  new DurationTransformer(),\n  new MapTransformer(),\n  new SetTransformer(),\n  new TwoWayMapTransformer()\n]);\n\n/**\n * Params for creating a {@link PluginSettingsComponentBase}.\n */\nexport interface PluginSettingsComponentConstructorParams {\n  /**\n   * A function to load data from the plugin's data file.\n   *\n   * @returns The loaded data.\n   */\n  readonly loadData: () => Promise<unknown>;\n\n  /**\n   * A function to save data to the plugin's data file.\n   *\n   * @param data - The data to save.\n   */\n  readonly saveData: (data: unknown) => Promise<void>;\n}\n\n/**\n * A snapshot of plugin settings state, including raw input values, effective (validated) values,\n * and per-property validation messages.\n *\n * @typeParam PluginSettings - The type of the plugin settings.\n */\nexport interface PluginSettingsState<PluginSettings extends object> {\n  /**\n   * The effective settings values used by the plugin. Invalid properties are replaced with defaults.\n   */\n  effectiveValues: PluginSettings;\n\n  /**\n   * The raw input values as entered by the user. May contain invalid values.\n   */\n  inputValues: PluginSettings;\n\n  /**\n   * Per-property validation messages. Empty string means valid.\n   */\n  validationMessages: Record<StringKeys<PluginSettings>, string>;\n}\n\n/**\n * Readonly version of {@link PluginSettings}.\n *\n * @typeParam PluginSettings - The type of the plugin settings.\n */\nexport type ReadonlyPluginSettings<PluginSettings extends object> = ReadonlyDeep<PluginSettings>;\n\n/**\n * Readonly version of {@link PluginSettingsState} for use in event callbacks and public getters.\n *\n * @typeParam PluginSettings - The type of the plugin settings.\n */\nexport type ReadonlyPluginSettingsState<PluginSettings extends object> = ReadonlyDeep<PluginSettingsState<PluginSettings>>;\n\n/**\n * A validator function for a settings property.\n *\n * @typeParam PluginSettings - The plugin settings type.\n * @typeParam PropertyName - The property name.\n */\nexport type SettingsValidator<PluginSettings extends object, PropertyName extends StringKeys<PluginSettings> = StringKeys<PluginSettings>> = (\n  value: PluginSettings[PropertyName],\n  settings: PluginSettings\n) => Promisable<MaybeReturn<string>>;\n\ntype PropertyNames<PluginSettings extends object> = StringKeys<PluginSettings>;\n\ntype PropertyValues<PluginSettings extends object> = PluginSettings[PropertyNames<PluginSettings>];\n\ntype ValidationResult<PluginSettings extends object> = Partial<Record<StringKeys<PluginSettings>, string>>;\n\n/**\n * Base class for plugin settings components.\n *\n * Manages settings data, persistence, validation, and events.\n * Plugin authors extend this class and implement {@link createDefaultSettings}.\n *\n * @typeParam PluginSettings - The plugin settings type.\n */\nexport abstract class PluginSettingsComponentBase<PluginSettings extends object> extends AsyncComponentBase {\n  /**\n   * Component key for {@link registerComponent} replacement.\n   */\n  public static readonly COMPONENT_KEY = Symbol(PluginSettingsComponentBase.name);\n\n  /**\n   * Gets the readonly default settings.\n   *\n   * @returns The default settings (as a readonly object).\n   */\n  public readonly defaultSettings: ReadonlyDeep<PluginSettings>;\n\n  /**\n   * Gets the readonly effective settings (validated, with defaults substituted for invalid values).\n   *\n   * @returns The readonly effective settings.\n   */\n  public get settings(): ReadonlyDeep<PluginSettings> {\n    return this.settingsState.effectiveValues;\n  }\n\n  /**\n   * Gets the current settings state snapshot.\n   *\n   * @returns The current settings state.\n   */\n  public get settingsState(): ReadonlyPluginSettingsState<PluginSettings> {\n    return this.currentState as ReadonlyPluginSettingsState<PluginSettings>;\n  }\n\n  private readonly asyncEvents = new AsyncEvents();\n  private currentState: PluginSettingsState<PluginSettings>;\n  private lastSavedState: PluginSettingsState<PluginSettings>;\n  private readonly legacySettingsConverters: ((record: GenericObject) => void)[] = [];\n  private readonly loadDataFn: () => Promise<unknown>;\n  private readonly propertyNames: PropertyNames<PluginSettings>[];\n  private readonly saveDataFn: (data: unknown) => Promise<void>;\n  private readonly validators = new Map<PropertyNames<PluginSettings>, SettingsValidator<PluginSettings>>();\n\n  /**\n   * Creates a new plugin settings component.\n   *\n   * @param params - The params for loading/saving data.\n   */\n  public constructor(params: PluginSettingsComponentConstructorParams) {\n    super();\n    this.loadDataFn = params.loadData.bind(params);\n    this.saveDataFn = params.saveData.bind(params);\n    this.defaultSettings = this.createDefaultSettings() as ReadonlyDeep<PluginSettings>;\n    this.currentState = this.createDefaultState();\n    this.lastSavedState = this.createDefaultState();\n    this.propertyNames = getAllKeys(this.currentState.inputValues);\n    this.registerValidators();\n    this.registerLegacySettingsConverters();\n  }\n\n  /**\n   * Edits the plugin settings and saves them.\n   *\n   * @param settingsEditor - The editor.\n   * @param context - The context.\n   * @returns A {@link Promise} that resolves when the settings are saved.\n   */\n  public async editAndSave(settingsEditor: (settings: PluginSettings) => Promisable<void>, context?: unknown): Promise<void> {\n    await this.edit(settingsEditor);\n    await this.saveToFile(context);\n  }\n\n  /**\n   * Ensures the settings are safe.\n   *\n   * It runs validation for each property and sets the default value if the validation fails.\n   *\n   * @param settings - The settings.\n   * @returns A {@link Promise} that resolves when the settings are safe.\n   */\n  public async ensureSafe(settings: PluginSettings): Promise<void> {\n    const validationResult = await this.validate(settings);\n    const defaults = this.defaultSettings as PluginSettings;\n    for (const propertyName of this.propertyNames) {\n      if (validationResult[propertyName]) {\n        settings[propertyName] = defaults[propertyName];\n      }\n    }\n  }\n\n  /**\n   * Gets a safe copy of the settings.\n   *\n   * @param settings - The settings.\n   * @returns A {@link Promise} that resolves to the safe copy of the settings.\n   */\n  public async getSafeCopy(settings: PluginSettings): Promise<PluginSettings> {\n    const safeSettings = await this.cloneSettings(settings);\n    await this.ensureSafe(safeSettings);\n    return safeSettings;\n  }\n\n  /**\n   * Loads the plugin settings from the file.\n   *\n   * @param isInitialLoad - Whether the settings are being loaded for the first time.\n   * @returns A {@link Promise} that resolves when the settings are loaded.\n   */\n  public async loadFromFile(isInitialLoad: boolean): Promise<void> {\n    const data = await this.loadDataFn();\n    this.lastSavedState = this.createDefaultState();\n    this.currentState = this.createDefaultState();\n\n    try {\n      if (data === undefined || data === null) {\n        return;\n      }\n\n      if (typeof data !== 'object') {\n        console.error(`Invalid settings from data.json. Expected Object, got: ${typeof data}`);\n        return;\n      }\n\n      const rawRecord = data as GenericObject;\n      const parsedSettings = await this.rawRecordToSettings(rawRecord);\n      const validationResult = await this.validate(parsedSettings);\n\n      for (const propertyName of this.propertyNames) {\n        this.setPropertyImpl(propertyName, parsedSettings[propertyName], validationResult[propertyName]);\n      }\n\n      this.lastSavedState = await this.cloneState(this.currentState);\n\n      const newRecord = await this.settingsToRawRecord(this.currentState.inputValues);\n\n      if (!deepEqual(newRecord, data)) {\n        await this.saveToFileImpl();\n      }\n    } finally {\n      await this.asyncEvents.triggerAsync('loadSettings', this.currentState, isInitialLoad);\n    }\n  }\n\n  /**\n   * Subscribes to the `loadSettings` event.\n   *\n   * @param name - Always `loadSettings`.\n   * @param callback - The callback.\n   * @param thisArg - The `this` context.\n   * @returns A reference to the event listener.\n   */\n  public on(\n    name: 'loadSettings',\n    callback: (\n      loadedState: ReadonlyPluginSettingsState<PluginSettings>,\n      isInitialLoad: boolean\n    ) => Promisable<void>,\n    thisArg?: unknown\n  ): AsyncEventRef;\n  /**\n   * Subscribes to the `saveSettings` event.\n   *\n   * @param name - Always `saveSettings`.\n   * @param callback - The callback.\n   * @param thisArg - The `this` context.\n   * @returns A reference to the event listener.\n   */\n  public on(\n    name: 'saveSettings',\n    callback: (\n      newState: ReadonlyPluginSettingsState<PluginSettings>,\n      oldState: ReadonlyPluginSettingsState<PluginSettings>,\n      context: unknown\n    ) => Promisable<void>,\n    thisArg?: unknown\n  ): AsyncEventRef;\n  /**\n   * Subscribes to an event.\n   *\n   * @param name - The name of the event.\n   * @param callback - The callback.\n   * @param thisArg - The `this` context.\n   * @returns A reference to the event listener.\n   */\n  public on<Args extends unknown[]>(\n    name: string,\n    callback: (...args: Args) => Promisable<void>,\n    thisArg?: unknown\n  ): AsyncEventRef {\n    return this.asyncEvents.on(name, callback, thisArg);\n  }\n\n  /**\n   * Called when the external settings change.\n   *\n   * @returns A {@link Promise} that resolves when the settings are reloaded.\n   */\n  public async onExternalSettingsChange(): Promise<void> {\n    await this.loadFromFile(false);\n  }\n\n  /**\n   * Loads settings from file and registers event handlers.\n   */\n  public override async onload(): Promise<void> {\n    registerAsyncEvent(this, this.on('loadSettings', this.onLoadSettings.bind(this)));\n    registerAsyncEvent(this, this.on('saveSettings', this.onSaveSettings.bind(this)));\n    await this.loadFromFile(true);\n  }\n\n  /**\n   * Registers a legacy settings converter.\n   *\n   * @typeParam LegacySettings - The legacy settings class.\n   * @param legacySettingsClass - The legacy settings class.\n   * @param converter - The converter.\n   */\n  public registerLegacySettingsConverter<LegacySettings extends object>(\n    legacySettingsClass: new () => LegacySettings,\n    converter: (legacySettings: Partial<LegacySettings> & Partial<PluginSettings>) => void\n  ): void {\n    const that = this;\n    this.legacySettingsConverters.push(legacySettingsConverter);\n\n    function legacySettingsConverter(record: GenericObject): void {\n      const legacySettingsKeys = new Set<string>(Object.keys(new legacySettingsClass()));\n      const pluginSettingKeys = new Set<string>(that.propertyNames);\n      const legacySettings = record as Partial<LegacySettings> & Partial<PluginSettings>;\n      converter(legacySettings);\n      for (const key of Object.keys(legacySettings)) {\n        if (pluginSettingKeys.has(key)) {\n          continue;\n        }\n\n        if (!legacySettingsKeys.has(key)) {\n          continue;\n        }\n\n        // eslint-disable-next-line @typescript-eslint/no-dynamic-delete -- We have no other way to delete the property.\n        delete record[key];\n      }\n    }\n  }\n\n  /**\n   * Registers a validator for a property.\n   *\n   * @param propertyName - The name of the property.\n   * @param validator - The validator.\n   */\n  public registerValidator<PropertyName extends PropertyNames<PluginSettings>>(\n    propertyName: PropertyName,\n    validator: SettingsValidator<PluginSettings, PropertyName>\n  ): void {\n    this.validators.set(propertyName, validator as SettingsValidator<PluginSettings>);\n  }\n\n  /**\n   * Revalidates the settings.\n   *\n   * @returns The validation messages.\n   */\n  public async revalidate(): Promise<Record<PropertyNames<PluginSettings>, string>> {\n    await this.edit(noop);\n    return this.currentState.validationMessages;\n  }\n\n  /**\n   * Saves the new plugin settings.\n   *\n   * @param context - The context of the save to file operation.\n   * @returns A {@link Promise} that resolves when the settings are saved.\n   */\n  public async saveToFile(context?: unknown): Promise<void> {\n    if (deepEqual(this.lastSavedState.inputValues, this.currentState.inputValues)) {\n      return;\n    }\n\n    await this.saveToFileImpl();\n    await this.asyncEvents.triggerAsync('saveSettings', this.currentState, this.lastSavedState, context);\n    this.lastSavedState = await this.cloneState(this.currentState);\n  }\n\n  /**\n   * Sets the value of a property.\n   *\n   * @typeParam PropertyName - The name of the property.\n   * @param propertyName - The name of the property.\n   * @param value - The value to set.\n   * @returns A {@link Promise} that resolves to the validation message.\n   */\n  public async setProperty<PropertyName extends PropertyNames<PluginSettings>>(\n    propertyName: PropertyName,\n    value: PluginSettings[PropertyName]\n  ): Promise<string> {\n    await this.edit((settings) => {\n      settings[propertyName] = value;\n    });\n    return this.currentState.validationMessages[propertyName];\n  }\n\n  /**\n   * Validates the settings.\n   *\n   * @param settings - The settings.\n   * @returns A {@link Promise} that resolves to the validation result.\n   */\n  public async validate(settings: PluginSettings): Promise<ValidationResult<PluginSettings>> {\n    const result: ValidationResult<PluginSettings> = {};\n    for (const [propertyName, validator] of this.validators.entries()) {\n      const validationMessage = await validator(settings[propertyName], settings);\n      if (validationMessage) {\n        result[propertyName] = validationMessage;\n      }\n    }\n\n    return result;\n  }\n\n  /**\n   * Creates the default settings. Must be implemented by subclasses.\n   *\n   * @returns The default settings.\n   */\n  protected abstract createDefaultSettings(): PluginSettings;\n\n  /**\n   * Gets the transformer.\n   *\n   * @returns The transformer.\n   */\n  protected getTransformer(): Transformer {\n    return defaultTransformer;\n  }\n\n  /**\n   * Called when the plugin settings record is loaded from disk.\n   *\n   * @param record - The record.\n   */\n  protected async onLoadRecord(record: GenericObject): Promise<void> {\n    for (const converter of this.legacySettingsConverters) {\n      converter(record);\n    }\n    await Promise.resolve();\n  }\n\n  /**\n   * Called when settings are loaded or reloaded.\n   *\n   * @param _loadedState - The loaded settings state.\n   * @param _isInitialLoad - Whether this is the initial load.\n   */\n  protected async onLoadSettings(\n    _loadedState: ReadonlyPluginSettingsState<PluginSettings>,\n    _isInitialLoad: boolean\n  ): Promise<void> {\n    await noopAsync();\n  }\n\n  /**\n   * Called when settings are saved.\n   *\n   * @param _newState - The new settings state.\n   * @param _oldState - The old settings state.\n   * @param _context - The save context.\n   */\n  protected async onSaveSettings(\n    _newState: ReadonlyPluginSettingsState<PluginSettings>,\n    _oldState: ReadonlyPluginSettingsState<PluginSettings>,\n    _context: unknown\n  ): Promise<void> {\n    await noopAsync();\n  }\n\n  /**\n   * Called when the plugin settings record is about to be saved to disk.\n   *\n   * @param _record - The record.\n   */\n  protected async onSavingRecord(_record: GenericObject): Promise<void> {\n    await noopAsync();\n  }\n\n  /**\n   * Registers the legacy settings converters.\n   * Override to register legacy settings converters.\n   */\n  protected registerLegacySettingsConverters(): void {\n    noop();\n  }\n\n  /**\n   * Registers the validators.\n   * Override to register validators for properties.\n   */\n  protected registerValidators(): void {\n    noop();\n  }\n\n  private async cloneSettings(settings: PluginSettings): Promise<PluginSettings> {\n    const record = await this.settingsToRawRecord(settings);\n    const json = JSON.stringify(record);\n    const cloneRecord = JSON.parse(json) as GenericObject;\n    return await this.rawRecordToSettings(cloneRecord);\n  }\n\n  private async cloneState(\n    state: PluginSettingsState<PluginSettings>\n  ): Promise<PluginSettingsState<PluginSettings>> {\n    return {\n      effectiveValues: await this.cloneSettings(state.effectiveValues),\n      inputValues: await this.cloneSettings(state.inputValues),\n      validationMessages: { ...state.validationMessages }\n    };\n  }\n\n  private createDefaultState(): PluginSettingsState<PluginSettings> {\n    return {\n      effectiveValues: this.createDefaultSettings(),\n      inputValues: this.createDefaultSettings(),\n      validationMessages: castTo<Record<PropertyNames<PluginSettings>, string>>({})\n    };\n  }\n\n  private async edit(settingsEditor: (settings: PluginSettings) => Promisable<void>): Promise<void> {\n    try {\n      await settingsEditor(this.currentState.inputValues);\n    } finally {\n      const validationResult = await this.validate(this.currentState.inputValues);\n      for (const propertyName of this.propertyNames) {\n        const validationMessage = validationResult[propertyName] ?? '';\n        this.currentState.validationMessages[propertyName] = validationMessage;\n        const defaults = this.defaultSettings as PluginSettings;\n        this.currentState.effectiveValues[propertyName] = validationMessage\n          ? defaults[propertyName]\n          : this.currentState.inputValues[propertyName];\n      }\n    }\n  }\n\n  private isValidPropertyName(prop: unknown): prop is PropertyNames<PluginSettings> {\n    if (typeof prop !== 'string') {\n      return false;\n    }\n\n    return (this.propertyNames as string[]).includes(prop);\n  }\n\n  private async rawRecordToSettings(rawRecord: GenericObject): Promise<PluginSettings> {\n    rawRecord = this.getTransformer().transformObjectRecursively(rawRecord);\n    await this.onLoadRecord(rawRecord);\n\n    const settings = this.createDefaultSettings();\n    const defaults = this.defaultSettings as PluginSettings;\n\n    for (const [propertyName, value] of Object.entries(rawRecord)) {\n      if (!this.isValidPropertyName(propertyName)) {\n        getLibDebugger('PluginSettingsComponentBase:rawRecordToSettings')(`Unknown property: ${propertyName}`);\n        continue;\n      }\n\n      if (typeof value !== typeof defaults[propertyName]) {\n        getLibDebugger('PluginSettingsComponentBase:rawRecordToSettings')(\n          'Possible invalid value type. It might lead to an unexpected behavior of the plugin. There is also a chance it is a false-negative warning, as we are unable to determine the exact type of the value in runtime.',\n          {\n            defaultValue: defaults[propertyName],\n            propertyName,\n            value\n          }\n        );\n      }\n\n      settings[propertyName] = value as PropertyValues<PluginSettings>;\n    }\n\n    return settings;\n  }\n\n  private async saveToFileImpl(): Promise<void> {\n    await this.saveDataFn(await this.settingsToRawRecord(this.currentState.inputValues));\n  }\n\n  private setPropertyImpl(\n    propertyName: PropertyNames<PluginSettings>,\n    value: PropertyValues<PluginSettings>,\n    validationMessage?: string\n  ): void {\n    const defaults = this.defaultSettings as PluginSettings;\n    this.currentState.inputValues[propertyName] = value;\n    this.currentState.validationMessages[propertyName] = validationMessage ?? '';\n    this.currentState.effectiveValues[propertyName] = validationMessage ? defaults[propertyName] : value;\n  }\n\n  private async settingsToRawRecord(settings: PluginSettings): Promise<GenericObject> {\n    const rawRecord: GenericObject = {};\n\n    for (const propertyName of this.propertyNames) {\n      rawRecord[propertyName] = settings[propertyName];\n    }\n\n    await this.onSavingRecord(rawRecord);\n\n    return this.getTransformer().transformObjectRecursively(rawRecord);\n  }\n}\n\n/**\n * A no-op settings component for plugins without settings.\n */\nexport class EmptyPluginSettingsComponent extends PluginSettingsComponentBase<object> {\n  /**\n   * Creates a new empty plugin settings component.\n   *\n   * @param params - The params. Uses no-op load/save by default.\n   */\n  public constructor(params?: PluginSettingsComponentConstructorParams) {\n    super(\n      params ?? {\n        loadData: async (): Promise<object> => {\n          await noopAsync();\n          return {};\n        },\n        saveData: noopAsync\n      }\n    );\n  }\n\n  /**\n   * Creates empty default settings.\n   *\n   * @returns An empty object.\n   */\n  protected override createDefaultSettings(): object {\n    return {};\n  }\n}\n"],
  "mappings": "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",
  "names": []
}

|
|
562
|
+
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../../../../src/obsidian/plugin/components/plugin-settings-component.ts"],
  "sourcesContent": ["/**\n * @file\n *\n * Component that manages plugin settings: data, persistence, validation, and events.\n *\n * This is the single public-facing settings API for plugin authors.\n * All settings logic (load, save, validate, transform, clone) is handled directly\n * by this component \u2014 no separate manager class is needed.\n */\n\nimport type {\n  Promisable,\n  ReadonlyDeep\n} from 'type-fest';\n\nimport type { AsyncEventRef } from '../../../async-events.ts';\nimport type { Transformer } from '../../../transformers/transformer.ts';\nimport type { GenericObject } from '../../../type-guards.ts';\nimport type {\n  MaybeReturn,\n  StringKeys\n} from '../../../type.ts';\nimport type { DataHandler } from '../../data-handler.ts';\n\nimport { AsyncEvents } from '../../../async-events.ts';\nimport { getLibDebugger } from '../../../debug.ts';\nimport {\n  noop,\n  noopAsync\n} from '../../../function.ts';\nimport {\n  castTo,\n  deepEqual,\n  getAllKeys\n} from '../../../object-utils.ts';\nimport { DateTransformer } from '../../../transformers/date-transformer.ts';\nimport { DurationTransformer } from '../../../transformers/duration-transformer.ts';\nimport { GroupTransformer } from '../../../transformers/group-transformer.ts';\nimport { MapTransformer } from '../../../transformers/map-transformer.ts';\nimport { SetTransformer } from '../../../transformers/set-transformer.ts';\nimport { SkipPrivatePropertyTransformer } from '../../../transformers/skip-private-property-transformer.ts';\nimport { TwoWayMapTransformer } from '../../../transformers/two-way-map-transformer.ts';\nimport { AsyncComponentBase } from '../../components/async-component.ts';\nimport { registerAsyncEvent } from '../../components/async-events-component.ts';\n\nconst defaultTransformer = new GroupTransformer([\n  new SkipPrivatePropertyTransformer(),\n  new DateTransformer(),\n  new DurationTransformer(),\n  new MapTransformer(),\n  new SetTransformer(),\n  new TwoWayMapTransformer()\n]);\n\n/**\n * A snapshot of plugin settings state, including raw input values, effective (validated) values,\n * and per-property validation messages.\n *\n * @typeParam PluginSettings - The type of the plugin settings.\n */\nexport interface PluginSettingsState<PluginSettings extends object> {\n  /**\n   * The effective settings values used by the plugin. Invalid properties are replaced with defaults.\n   */\n  effectiveValues: PluginSettings;\n\n  /**\n   * The raw input values as entered by the user. May contain invalid values.\n   */\n  inputValues: PluginSettings;\n\n  /**\n   * Per-property validation messages. Empty string means valid.\n   */\n  validationMessages: Record<StringKeys<PluginSettings>, string>;\n}\n\n/**\n * Readonly version of {@link PluginSettings}.\n *\n * @typeParam PluginSettings - The type of the plugin settings.\n */\nexport type ReadonlyPluginSettings<PluginSettings extends object> = ReadonlyDeep<PluginSettings>;\n\n/**\n * Readonly version of {@link PluginSettingsState} for use in event callbacks and public getters.\n *\n * @typeParam PluginSettings - The type of the plugin settings.\n */\nexport type ReadonlyPluginSettingsState<PluginSettings extends object> = ReadonlyDeep<PluginSettingsState<PluginSettings>>;\n\n/**\n * A validator function for a settings property.\n *\n * @typeParam PluginSettings - The plugin settings type.\n * @typeParam PropertyName - The property name.\n */\nexport type SettingsValidator<PluginSettings extends object, PropertyName extends StringKeys<PluginSettings> = StringKeys<PluginSettings>> = (\n  value: PluginSettings[PropertyName],\n  settings: PluginSettings\n) => Promisable<MaybeReturn<string>>;\n\ntype PropertyNames<PluginSettings extends object> = StringKeys<PluginSettings>;\n\ntype PropertyValues<PluginSettings extends object> = PluginSettings[PropertyNames<PluginSettings>];\n\ntype ValidationResult<PluginSettings extends object> = Partial<Record<StringKeys<PluginSettings>, string>>;\n\nclass EmptyDataHandler implements DataHandler {\n  public async loadData(): Promise<unknown> {\n    await noopAsync();\n    return {};\n  }\n\n  /* v8 ignore start -- unreachable because EmptyPluginSettingsComponent has no properties to change, so saveToFile always short-circuits. */\n  public async saveData(): Promise<void> {\n    await noopAsync();\n  }\n  /* v8 ignore stop */\n}\n\n/**\n * Base class for plugin settings components.\n *\n * Manages settings data, persistence, validation, and events.\n * Plugin authors extend this class and implement {@link createDefaultSettings}.\n *\n * @typeParam PluginSettings - The plugin settings type.\n */\nexport abstract class PluginSettingsComponentBase<PluginSettings extends object> extends AsyncComponentBase {\n  /**\n   * Component key for {@link registerComponent} replacement.\n   */\n  public static readonly COMPONENT_KEY = Symbol(PluginSettingsComponentBase.name);\n\n  /**\n   * Gets the readonly default settings.\n   *\n   * @returns The default settings (as a readonly object).\n   */\n  public readonly defaultSettings: ReadonlyDeep<PluginSettings>;\n\n  /**\n   * Gets the readonly effective settings (validated, with defaults substituted for invalid values).\n   *\n   * @returns The readonly effective settings.\n   */\n  public get settings(): ReadonlyDeep<PluginSettings> {\n    return this.settingsState.effectiveValues;\n  }\n\n  /**\n   * Gets the current settings state snapshot.\n   *\n   * @returns The current settings state.\n   */\n  public get settingsState(): ReadonlyPluginSettingsState<PluginSettings> {\n    return this.currentState as ReadonlyPluginSettingsState<PluginSettings>;\n  }\n\n  private readonly asyncEvents = new AsyncEvents();\n  private currentState: PluginSettingsState<PluginSettings>;\n  private lastSavedState: PluginSettingsState<PluginSettings>;\n  private readonly legacySettingsConverters: ((record: GenericObject) => void)[] = [];\n  private readonly propertyNames: PropertyNames<PluginSettings>[];\n  private readonly validators = new Map<PropertyNames<PluginSettings>, SettingsValidator<PluginSettings>>();\n\n  /**\n   * Creates a new plugin settings component.\n   *\n   * @param dataHandler - The data handler.\n   */\n  public constructor(private readonly dataHandler: DataHandler) {\n    super();\n    this.defaultSettings = this.createDefaultSettings() as ReadonlyDeep<PluginSettings>;\n    this.currentState = this.createDefaultState();\n    this.lastSavedState = this.createDefaultState();\n    this.propertyNames = getAllKeys(this.currentState.inputValues);\n    this.registerValidators();\n    this.registerLegacySettingsConverters();\n  }\n\n  /**\n   * Edits the plugin settings and saves them.\n   *\n   * @param settingsEditor - The editor.\n   * @param context - The context.\n   * @returns A {@link Promise} that resolves when the settings are saved.\n   */\n  public async editAndSave(settingsEditor: (settings: PluginSettings) => Promisable<void>, context?: unknown): Promise<void> {\n    await this.edit(settingsEditor);\n    await this.saveToFile(context);\n  }\n\n  /**\n   * Ensures the settings are safe.\n   *\n   * It runs validation for each property and sets the default value if the validation fails.\n   *\n   * @param settings - The settings.\n   * @returns A {@link Promise} that resolves when the settings are safe.\n   */\n  public async ensureSafe(settings: PluginSettings): Promise<void> {\n    const validationResult = await this.validate(settings);\n    const defaults = this.defaultSettings as PluginSettings;\n    for (const propertyName of this.propertyNames) {\n      if (validationResult[propertyName]) {\n        settings[propertyName] = defaults[propertyName];\n      }\n    }\n  }\n\n  /**\n   * Gets a safe copy of the settings.\n   *\n   * @param settings - The settings.\n   * @returns A {@link Promise} that resolves to the safe copy of the settings.\n   */\n  public async getSafeCopy(settings: PluginSettings): Promise<PluginSettings> {\n    const safeSettings = await this.cloneSettings(settings);\n    await this.ensureSafe(safeSettings);\n    return safeSettings;\n  }\n\n  /**\n   * Loads the plugin settings from the file.\n   *\n   * @param isInitialLoad - Whether the settings are being loaded for the first time.\n   * @returns A {@link Promise} that resolves when the settings are loaded.\n   */\n  public async loadFromFile(isInitialLoad: boolean): Promise<void> {\n    const data = await this.dataHandler.loadData();\n    this.lastSavedState = this.createDefaultState();\n    this.currentState = this.createDefaultState();\n\n    try {\n      if (data === undefined || data === null) {\n        return;\n      }\n\n      if (typeof data !== 'object') {\n        console.error(`Invalid settings from data.json. Expected Object, got: ${typeof data}`);\n        return;\n      }\n\n      const rawRecord = data as GenericObject;\n      const parsedSettings = await this.rawRecordToSettings(rawRecord);\n      const validationResult = await this.validate(parsedSettings);\n\n      for (const propertyName of this.propertyNames) {\n        this.setPropertyImpl(propertyName, parsedSettings[propertyName], validationResult[propertyName]);\n      }\n\n      this.lastSavedState = await this.cloneState(this.currentState);\n\n      const newRecord = await this.settingsToRawRecord(this.currentState.inputValues);\n\n      if (!deepEqual(newRecord, data)) {\n        await this.saveToFileImpl();\n      }\n    } finally {\n      await this.asyncEvents.triggerAsync('loadSettings', this.currentState, isInitialLoad);\n    }\n  }\n\n  /**\n   * Subscribes to the `loadSettings` event.\n   *\n   * @param name - Always `loadSettings`.\n   * @param callback - The callback.\n   * @param thisArg - The `this` context.\n   * @returns A reference to the event listener.\n   */\n  public on(\n    name: 'loadSettings',\n    callback: (\n      loadedState: ReadonlyPluginSettingsState<PluginSettings>,\n      isInitialLoad: boolean\n    ) => Promisable<void>,\n    thisArg?: unknown\n  ): AsyncEventRef;\n  /**\n   * Subscribes to the `saveSettings` event.\n   *\n   * @param name - Always `saveSettings`.\n   * @param callback - The callback.\n   * @param thisArg - The `this` context.\n   * @returns A reference to the event listener.\n   */\n  public on(\n    name: 'saveSettings',\n    callback: (\n      newState: ReadonlyPluginSettingsState<PluginSettings>,\n      oldState: ReadonlyPluginSettingsState<PluginSettings>,\n      context: unknown\n    ) => Promisable<void>,\n    thisArg?: unknown\n  ): AsyncEventRef;\n  /**\n   * Subscribes to an event.\n   *\n   * @param name - The name of the event.\n   * @param callback - The callback.\n   * @param thisArg - The `this` context.\n   * @returns A reference to the event listener.\n   */\n  public on<Args extends unknown[]>(\n    name: string,\n    callback: (...args: Args) => Promisable<void>,\n    thisArg?: unknown\n  ): AsyncEventRef {\n    return this.asyncEvents.on(name, callback, thisArg);\n  }\n\n  /**\n   * Called when the external settings change.\n   *\n   * @returns A {@link Promise} that resolves when the settings are reloaded.\n   */\n  public async onExternalSettingsChange(): Promise<void> {\n    await this.loadFromFile(false);\n  }\n\n  /**\n   * Loads settings from file and registers event handlers.\n   */\n  public override async onload(): Promise<void> {\n    registerAsyncEvent(this, this.on('loadSettings', this.onLoadSettings.bind(this)));\n    registerAsyncEvent(this, this.on('saveSettings', this.onSaveSettings.bind(this)));\n    await this.loadFromFile(true);\n  }\n\n  /**\n   * Registers a legacy settings converter.\n   *\n   * @typeParam LegacySettings - The legacy settings class.\n   * @param legacySettingsClass - The legacy settings class.\n   * @param converter - The converter.\n   */\n  public registerLegacySettingsConverter<LegacySettings extends object>(\n    legacySettingsClass: new () => LegacySettings,\n    converter: (legacySettings: Partial<LegacySettings> & Partial<PluginSettings>) => void\n  ): void {\n    const that = this;\n    this.legacySettingsConverters.push(legacySettingsConverter);\n\n    function legacySettingsConverter(record: GenericObject): void {\n      const legacySettingsKeys = new Set<string>(Object.keys(new legacySettingsClass()));\n      const pluginSettingKeys = new Set<string>(that.propertyNames);\n      const legacySettings = record as Partial<LegacySettings> & Partial<PluginSettings>;\n      converter(legacySettings);\n      for (const key of Object.keys(legacySettings)) {\n        if (pluginSettingKeys.has(key)) {\n          continue;\n        }\n\n        if (!legacySettingsKeys.has(key)) {\n          continue;\n        }\n\n        // eslint-disable-next-line @typescript-eslint/no-dynamic-delete -- We have no other way to delete the property.\n        delete record[key];\n      }\n    }\n  }\n\n  /**\n   * Registers a validator for a property.\n   *\n   * @param propertyName - The name of the property.\n   * @param validator - The validator.\n   */\n  public registerValidator<PropertyName extends PropertyNames<PluginSettings>>(\n    propertyName: PropertyName,\n    validator: SettingsValidator<PluginSettings, PropertyName>\n  ): void {\n    this.validators.set(propertyName, validator as SettingsValidator<PluginSettings>);\n  }\n\n  /**\n   * Revalidates the settings.\n   *\n   * @returns The validation messages.\n   */\n  public async revalidate(): Promise<Record<PropertyNames<PluginSettings>, string>> {\n    await this.edit(noop);\n    return this.currentState.validationMessages;\n  }\n\n  /**\n   * Saves the new plugin settings.\n   *\n   * @param context - The context of the save to file operation.\n   * @returns A {@link Promise} that resolves when the settings are saved.\n   */\n  public async saveToFile(context?: unknown): Promise<void> {\n    if (deepEqual(this.lastSavedState.inputValues, this.currentState.inputValues)) {\n      return;\n    }\n\n    await this.saveToFileImpl();\n    await this.asyncEvents.triggerAsync('saveSettings', this.currentState, this.lastSavedState, context);\n    this.lastSavedState = await this.cloneState(this.currentState);\n  }\n\n  /**\n   * Sets the value of a property.\n   *\n   * @typeParam PropertyName - The name of the property.\n   * @param propertyName - The name of the property.\n   * @param value - The value to set.\n   * @returns A {@link Promise} that resolves to the validation message.\n   */\n  public async setProperty<PropertyName extends PropertyNames<PluginSettings>>(\n    propertyName: PropertyName,\n    value: PluginSettings[PropertyName]\n  ): Promise<string> {\n    await this.edit((settings) => {\n      settings[propertyName] = value;\n    });\n    return this.currentState.validationMessages[propertyName];\n  }\n\n  /**\n   * Validates the settings.\n   *\n   * @param settings - The settings.\n   * @returns A {@link Promise} that resolves to the validation result.\n   */\n  public async validate(settings: PluginSettings): Promise<ValidationResult<PluginSettings>> {\n    const result: ValidationResult<PluginSettings> = {};\n    for (const [propertyName, validator] of this.validators.entries()) {\n      const validationMessage = await validator(settings[propertyName], settings);\n      if (validationMessage) {\n        result[propertyName] = validationMessage;\n      }\n    }\n\n    return result;\n  }\n\n  /**\n   * Creates the default settings. Must be implemented by subclasses.\n   *\n   * @returns The default settings.\n   */\n  protected abstract createDefaultSettings(): PluginSettings;\n\n  /**\n   * Gets the transformer.\n   *\n   * @returns The transformer.\n   */\n  protected getTransformer(): Transformer {\n    return defaultTransformer;\n  }\n\n  /**\n   * Called when the plugin settings record is loaded from disk.\n   *\n   * @param record - The record.\n   */\n  protected async onLoadRecord(record: GenericObject): Promise<void> {\n    for (const converter of this.legacySettingsConverters) {\n      converter(record);\n    }\n    await Promise.resolve();\n  }\n\n  /**\n   * Called when settings are loaded or reloaded.\n   *\n   * @param _loadedState - The loaded settings state.\n   * @param _isInitialLoad - Whether this is the initial load.\n   */\n  protected async onLoadSettings(\n    _loadedState: ReadonlyPluginSettingsState<PluginSettings>,\n    _isInitialLoad: boolean\n  ): Promise<void> {\n    await noopAsync();\n  }\n\n  /**\n   * Called when settings are saved.\n   *\n   * @param _newState - The new settings state.\n   * @param _oldState - The old settings state.\n   * @param _context - The save context.\n   */\n  protected async onSaveSettings(\n    _newState: ReadonlyPluginSettingsState<PluginSettings>,\n    _oldState: ReadonlyPluginSettingsState<PluginSettings>,\n    _context: unknown\n  ): Promise<void> {\n    await noopAsync();\n  }\n\n  /**\n   * Called when the plugin settings record is about to be saved to disk.\n   *\n   * @param _record - The record.\n   */\n  protected async onSavingRecord(_record: GenericObject): Promise<void> {\n    await noopAsync();\n  }\n\n  /**\n   * Registers the legacy settings converters.\n   * Override to register legacy settings converters.\n   */\n  protected registerLegacySettingsConverters(): void {\n    noop();\n  }\n\n  /**\n   * Registers the validators.\n   * Override to register validators for properties.\n   */\n  protected registerValidators(): void {\n    noop();\n  }\n\n  private async cloneSettings(settings: PluginSettings): Promise<PluginSettings> {\n    const record = await this.settingsToRawRecord(settings);\n    const json = JSON.stringify(record);\n    const cloneRecord = JSON.parse(json) as GenericObject;\n    return await this.rawRecordToSettings(cloneRecord);\n  }\n\n  private async cloneState(\n    state: PluginSettingsState<PluginSettings>\n  ): Promise<PluginSettingsState<PluginSettings>> {\n    return {\n      effectiveValues: await this.cloneSettings(state.effectiveValues),\n      inputValues: await this.cloneSettings(state.inputValues),\n      validationMessages: { ...state.validationMessages }\n    };\n  }\n\n  private createDefaultState(): PluginSettingsState<PluginSettings> {\n    return {\n      effectiveValues: this.createDefaultSettings(),\n      inputValues: this.createDefaultSettings(),\n      validationMessages: castTo<Record<PropertyNames<PluginSettings>, string>>({})\n    };\n  }\n\n  private async edit(settingsEditor: (settings: PluginSettings) => Promisable<void>): Promise<void> {\n    try {\n      await settingsEditor(this.currentState.inputValues);\n    } finally {\n      const validationResult = await this.validate(this.currentState.inputValues);\n      for (const propertyName of this.propertyNames) {\n        const validationMessage = validationResult[propertyName] ?? '';\n        this.currentState.validationMessages[propertyName] = validationMessage;\n        const defaults = this.defaultSettings as PluginSettings;\n        this.currentState.effectiveValues[propertyName] = validationMessage\n          ? defaults[propertyName]\n          : this.currentState.inputValues[propertyName];\n      }\n    }\n  }\n\n  private isValidPropertyName(prop: unknown): prop is PropertyNames<PluginSettings> {\n    if (typeof prop !== 'string') {\n      return false;\n    }\n\n    return (this.propertyNames as string[]).includes(prop);\n  }\n\n  private async rawRecordToSettings(rawRecord: GenericObject): Promise<PluginSettings> {\n    rawRecord = this.getTransformer().transformObjectRecursively(rawRecord);\n    await this.onLoadRecord(rawRecord);\n\n    const settings = this.createDefaultSettings();\n    const defaults = this.defaultSettings as PluginSettings;\n\n    for (const [propertyName, value] of Object.entries(rawRecord)) {\n      if (!this.isValidPropertyName(propertyName)) {\n        getLibDebugger('PluginSettingsComponentBase:rawRecordToSettings')(`Unknown property: ${propertyName}`);\n        continue;\n      }\n\n      if (typeof value !== typeof defaults[propertyName]) {\n        getLibDebugger('PluginSettingsComponentBase:rawRecordToSettings')(\n          'Possible invalid value type. It might lead to an unexpected behavior of the plugin. There is also a chance it is a false-negative warning, as we are unable to determine the exact type of the value in runtime.',\n          {\n            defaultValue: defaults[propertyName],\n            propertyName,\n            value\n          }\n        );\n      }\n\n      settings[propertyName] = value as PropertyValues<PluginSettings>;\n    }\n\n    return settings;\n  }\n\n  private async saveToFileImpl(): Promise<void> {\n    await this.dataHandler.saveData(await this.settingsToRawRecord(this.currentState.inputValues));\n  }\n\n  private setPropertyImpl(\n    propertyName: PropertyNames<PluginSettings>,\n    value: PropertyValues<PluginSettings>,\n    validationMessage?: string\n  ): void {\n    const defaults = this.defaultSettings as PluginSettings;\n    this.currentState.inputValues[propertyName] = value;\n    this.currentState.validationMessages[propertyName] = validationMessage ?? '';\n    this.currentState.effectiveValues[propertyName] = validationMessage ? defaults[propertyName] : value;\n  }\n\n  private async settingsToRawRecord(settings: PluginSettings): Promise<GenericObject> {\n    const rawRecord: GenericObject = {};\n\n    for (const propertyName of this.propertyNames) {\n      rawRecord[propertyName] = settings[propertyName];\n    }\n\n    await this.onSavingRecord(rawRecord);\n\n    return this.getTransformer().transformObjectRecursively(rawRecord);\n  }\n}\n\n/**\n * A no-op settings component for plugins without settings.\n */\nexport class EmptyPluginSettingsComponent extends PluginSettingsComponentBase<object> {\n  /**\n   * Creates a new empty plugin settings component.\n   */\n  public constructor() {\n    super(new EmptyDataHandler());\n  }\n\n  /**\n   * Creates empty default settings.\n   *\n   * @returns An empty object.\n   */\n  protected override createDefaultSettings(): object {\n    return {};\n  }\n}\n"],
  "mappings": "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",
  "names": []
}

|