obsidian-dev-utils 13.10.0 → 13.11.0

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 CHANGED
@@ -1,5 +1,14 @@
1
1
  # CHANGELOG
2
2
 
3
+ ## 13.11.0
4
+
5
+ - Improve stack trace printer
6
+
7
+ ## 13.10.1
8
+
9
+ - Update libs
10
+ - Don't show `consoleDebug()` in call stack
11
+
3
12
  ## 13.10.0
4
13
 
5
14
  - Refactor and document debugging
package/README.md CHANGED
@@ -175,6 +175,10 @@ window.DEBUG.enable('*'); // show all debug messages
175
175
 
176
176
  See full documentation of [`window.DEBUG`](https://github.com/mnaoumov/obsidian-dev-utils/blob/main/src/DebugController.ts).
177
177
 
178
+ > [!INFO]
179
+ >
180
+ > You will see `StackTraceFakeError` in the debug messages. They are not actual errors. It's just a workaround to make stack trace links clickable.
181
+
178
182
  ## Support
179
183
 
180
184
  <a href="https://www.buymeacoffee.com/mnaoumov" target="_blank"><img src="https://cdn.buymeacoffee.com/buttons/v2/default-yellow.png" alt="Buy Me A Coffee" style="height: 60px !important;width: 217px !important;"></a>
@@ -46,10 +46,10 @@ var import_debug = __toESM(__extractDefault(require('debug')), 1);
46
46
  let currentPluginId = "";
47
47
  const NAMESPACE_SEPARATOR = ",";
48
48
  const NEGATED_NAMESPACE_PREFIX = "-";
49
- function getDebugger(namespace) {
49
+ function getDebugger(namespace, framesToSkip = 0) {
50
50
  const debugInstance = (0, import_debug.default)(namespace);
51
51
  debugInstance.log = (message, ...args) => {
52
- logWithCaller(namespace, message, ...args);
52
+ logWithCaller(namespace, framesToSkip, message, ...args);
53
53
  };
54
54
  debugInstance.printStackTrace = (stackTrace, title) => {
55
55
  printStackTrace(namespace, stackTrace, title);
@@ -99,18 +99,19 @@ function enableNamespaces(namespaces) {
99
99
  function getNamespaces() {
100
100
  return toArray(import_debug.default.load() ?? "");
101
101
  }
102
- function logWithCaller(namespace, message, ...args) {
102
+ function logWithCaller(namespace, framesToSkip, message, ...args) {
103
103
  if (!import_debug.default.enabled(namespace)) {
104
104
  return;
105
105
  }
106
106
  const CALLER_LINE_INDEX = 4;
107
107
  const stackLines = new Error().stack?.split("\n") ?? [];
108
- const callerLine = stackLines[CALLER_LINE_INDEX] ?? "";
108
+ const callerLine = stackLines[CALLER_LINE_INDEX + framesToSkip] ?? "";
109
109
  console.debug(message, ...args);
110
- printStackTrace(namespace, callerLine, "Original debug message caller");
110
+ printStackTrace(namespace, callerLine, "Debug message caller");
111
111
  }
112
112
  function printStackTrace(namespace, stackTrace, title) {
113
- if (!import_debug.default.enabled(namespace)) {
113
+ const _debugger = (0, import_debug.default)(namespace);
114
+ if (!_debugger.enabled) {
114
115
  return;
115
116
  }
116
117
  if (!stackTrace) {
@@ -119,7 +120,8 @@ function printStackTrace(namespace, stackTrace, title) {
119
120
  if (!title) {
120
121
  title = "Caller stack trace";
121
122
  }
122
- console.debug(`NotError:${namespace}:${title}
123
+ _debugger.log(title);
124
+ console.debug(`StackTraceFakeError
123
125
  ${stackTrace}`);
124
126
  }
125
127
  function setNamespaces(namespaces) {
@@ -134,4 +136,4 @@ function toArray(namespaces) {
134
136
  getLibDebugger,
135
137
  initDebugHelpers
136
138
  });
137
- //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../src/Debug.ts"],
  "sourcesContent": ["/**\n * @packageDocumentation Debug\n * Contains utility functions for debugging.\n */\n\nimport type { Debugger } from 'debug';\n\nimport debug from 'debug';\n\nimport type { DebugController } from './DebugController.ts';\n\ninterface DebuggerEx extends Debugger {\n  printStackTrace(stackTrace: string, title?: string): void;\n}\n\ninterface DebugWrapper {\n  DEBUG: DebugController;\n}\n\nlet currentPluginId = '';\nconst NAMESPACE_SEPARATOR = ',';\nconst NEGATED_NAMESPACE_PREFIX = '-';\n\n/**\n * Returns a debugger instance with a log function that includes the caller's file name and line number.\n *\n * @param namespace - The namespace for the debugger instance.\n * @returns A debugger instance with a log function that includes the caller's file name and line number.\n */\nexport function getDebugger(namespace: string): DebuggerEx {\n  const debugInstance = debug(namespace) as DebuggerEx;\n  debugInstance.log = (message: string, ...args: unknown[]): void => {\n    logWithCaller(namespace, message, ...args);\n  };\n  debugInstance.printStackTrace = (stackTrace, title): void => {\n    printStackTrace(namespace, stackTrace, title);\n  };\n  return debugInstance;\n}\n\n/**\n * Returns a debugger instance for the `obsidian-dev-utils` library.\n *\n * @param namespace - The namespace for the debugger instance.\n * @returns A debugger instance for the `obsidian-dev-utils` library.\n */\nexport function getLibDebugger(namespace: string): DebuggerEx {\n  const prefix = currentPluginId ? `${currentPluginId}:` : '';\n  return getDebugger(`${prefix}obsidian-dev-utils:${namespace}`);\n}\n\n/**\n * Adds the DEBUG variable to the window object.\n *\n * @param pluginId - The plugin ID.\n */\nexport function initDebugHelpers(pluginId: string): void {\n  currentPluginId = pluginId;\n  (window as Partial<DebugWrapper>).DEBUG = {\n    disable: disableNamespaces,\n    enable: enableNamespaces,\n    get: getNamespaces,\n    set: setNamespaces\n  };\n}\n\nfunction disableNamespaces(namespaces: string | string[]): void {\n  const set = new Set(getNamespaces());\n  for (const namespace of toArray(namespaces)) {\n    if (namespace.startsWith(NEGATED_NAMESPACE_PREFIX)) {\n      continue;\n    }\n    const negatedNamespace = NEGATED_NAMESPACE_PREFIX + namespace;\n    if (set.has(namespace)) {\n      set.delete(namespace);\n    }\n    set.add(negatedNamespace);\n  }\n  setNamespaces(Array.from(set));\n}\n\nfunction enableNamespaces(namespaces: string | string[]): void {\n  const set = new Set(getNamespaces());\n  for (const namespace of toArray(namespaces)) {\n    if (!namespace.startsWith(NEGATED_NAMESPACE_PREFIX)) {\n      const negatedNamespace = NEGATED_NAMESPACE_PREFIX + namespace;\n      if (set.has(negatedNamespace)) {\n        set.delete(negatedNamespace);\n      }\n    }\n    set.add(namespace);\n  }\n  setNamespaces(Array.from(set));\n}\n\nfunction getNamespaces(): string[] {\n  return toArray(debug.load() ?? '');\n}\n\nfunction logWithCaller(namespace: string, message: string, ...args: unknown[]): void {\n  if (!debug.enabled(namespace)) {\n    return;\n  }\n\n  /**\n   * The caller line index is 4 because the call stack is as follows:\n   *\n   * 0: Error\n   * 1:     at logWithCaller (?:?:?)\n   * 2:     at debugInstance.log (?:?:?)\n   * 3:     at debug (?:?:?)\n   * 4:     at functionName (path/to/caller.js:?:?)\n   */\n  const CALLER_LINE_INDEX = 4;\n\n  const stackLines = new Error().stack?.split('\\n') ?? [];\n  const callerLine = stackLines[CALLER_LINE_INDEX] ?? '';\n  console.debug(message, ...args);\n  printStackTrace(namespace, callerLine, 'Original debug message caller');\n}\n\nfunction printStackTrace(namespace: string, stackTrace: string, title?: string): void {\n  if (!debug.enabled(namespace)) {\n    return;\n  }\n\n  if (!stackTrace) {\n    stackTrace = '(unavailable)';\n  }\n  if (!title) {\n    title = 'Caller stack trace';\n  }\n  console.debug(`NotError:${namespace}:${title}\\n${stackTrace}`);\n}\n\nfunction setNamespaces(namespaces: string | string[]): void {\n  debug.enable(toArray(namespaces).join(NAMESPACE_SEPARATOR));\n}\n\nfunction toArray(namespaces: string | string[]): string[] {\n  return typeof namespaces === 'string' ? namespaces.split(NAMESPACE_SEPARATOR).filter(Boolean) : namespaces.flatMap(toArray);\n}\n"],
  "mappings": ";;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAOA,mBAAkB;AAYlB,IAAI,kBAAkB;AACtB,MAAM,sBAAsB;AAC5B,MAAM,2BAA2B;AAQ1B,SAAS,YAAY,WAA+B;AACzD,QAAM,oBAAgB,aAAAA,SAAM,SAAS;AACrC,gBAAc,MAAM,CAAC,YAAoB,SAA0B;AACjE,kBAAc,WAAW,SAAS,GAAG,IAAI;AAAA,EAC3C;AACA,gBAAc,kBAAkB,CAAC,YAAY,UAAgB;AAC3D,oBAAgB,WAAW,YAAY,KAAK;AAAA,EAC9C;AACA,SAAO;AACT;AAQO,SAAS,eAAe,WAA+B;AAC5D,QAAM,SAAS,kBAAkB,GAAG,eAAe,MAAM;AACzD,SAAO,YAAY,GAAG,MAAM,sBAAsB,SAAS,EAAE;AAC/D;AAOO,SAAS,iBAAiB,UAAwB;AACvD,oBAAkB;AAClB,EAAC,OAAiC,QAAQ;AAAA,IACxC,SAAS;AAAA,IACT,QAAQ;AAAA,IACR,KAAK;AAAA,IACL,KAAK;AAAA,EACP;AACF;AAEA,SAAS,kBAAkB,YAAqC;AAC9D,QAAM,MAAM,IAAI,IAAI,cAAc,CAAC;AACnC,aAAW,aAAa,QAAQ,UAAU,GAAG;AAC3C,QAAI,UAAU,WAAW,wBAAwB,GAAG;AAClD;AAAA,IACF;AACA,UAAM,mBAAmB,2BAA2B;AACpD,QAAI,IAAI,IAAI,SAAS,GAAG;AACtB,UAAI,OAAO,SAAS;AAAA,IACtB;AACA,QAAI,IAAI,gBAAgB;AAAA,EAC1B;AACA,gBAAc,MAAM,KAAK,GAAG,CAAC;AAC/B;AAEA,SAAS,iBAAiB,YAAqC;AAC7D,QAAM,MAAM,IAAI,IAAI,cAAc,CAAC;AACnC,aAAW,aAAa,QAAQ,UAAU,GAAG;AAC3C,QAAI,CAAC,UAAU,WAAW,wBAAwB,GAAG;AACnD,YAAM,mBAAmB,2BAA2B;AACpD,UAAI,IAAI,IAAI,gBAAgB,GAAG;AAC7B,YAAI,OAAO,gBAAgB;AAAA,MAC7B;AAAA,IACF;AACA,QAAI,IAAI,SAAS;AAAA,EACnB;AACA,gBAAc,MAAM,KAAK,GAAG,CAAC;AAC/B;AAEA,SAAS,gBAA0B;AACjC,SAAO,QAAQ,aAAAA,QAAM,KAAK,KAAK,EAAE;AACnC;AAEA,SAAS,cAAc,WAAmB,YAAoB,MAAuB;AACnF,MAAI,CAAC,aAAAA,QAAM,QAAQ,SAAS,GAAG;AAC7B;AAAA,EACF;AAWA,QAAM,oBAAoB;AAE1B,QAAM,aAAa,IAAI,MAAM,EAAE,OAAO,MAAM,IAAI,KAAK,CAAC;AACtD,QAAM,aAAa,WAAW,iBAAiB,KAAK;AACpD,UAAQ,MAAM,SAAS,GAAG,IAAI;AAC9B,kBAAgB,WAAW,YAAY,+BAA+B;AACxE;AAEA,SAAS,gBAAgB,WAAmB,YAAoB,OAAsB;AACpF,MAAI,CAAC,aAAAA,QAAM,QAAQ,SAAS,GAAG;AAC7B;AAAA,EACF;AAEA,MAAI,CAAC,YAAY;AACf,iBAAa;AAAA,EACf;AACA,MAAI,CAAC,OAAO;AACV,YAAQ;AAAA,EACV;AACA,UAAQ,MAAM,YAAY,SAAS,IAAI,KAAK;AAAA,EAAK,UAAU,EAAE;AAC/D;AAEA,SAAS,cAAc,YAAqC;AAC1D,eAAAA,QAAM,OAAO,QAAQ,UAAU,EAAE,KAAK,mBAAmB,CAAC;AAC5D;AAEA,SAAS,QAAQ,YAAyC;AACxD,SAAO,OAAO,eAAe,WAAW,WAAW,MAAM,mBAAmB,EAAE,OAAO,OAAO,IAAI,WAAW,QAAQ,OAAO;AAC5H;",
  "names": ["debug"]
}

139
+ //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../src/Debug.ts"],
  "sourcesContent": ["/**\n * @packageDocumentation Debug\n * Contains utility functions for debugging.\n */\n\nimport type { Debugger } from 'debug';\n\nimport debug from 'debug';\n\nimport type { DebugController } from './DebugController.ts';\n\ninterface DebuggerEx extends Debugger {\n  printStackTrace(stackTrace: string, title?: string): void;\n}\n\ninterface DebugWrapper {\n  DEBUG: DebugController;\n}\n\nlet currentPluginId = '';\nconst NAMESPACE_SEPARATOR = ',';\nconst NEGATED_NAMESPACE_PREFIX = '-';\n\n/**\n * Returns a debugger instance with a log function that includes the caller's file name and line number.\n *\n * @param namespace - The namespace for the debugger instance.\n * @param framesToSkip - The number of frames to skip in the stack trace.\n * @returns A debugger instance with a log function that includes the caller's file name and line number.\n */\nexport function getDebugger(namespace: string, framesToSkip = 0): DebuggerEx {\n  const debugInstance = debug(namespace) as DebuggerEx;\n  debugInstance.log = (message: string, ...args: unknown[]): void => {\n    logWithCaller(namespace, framesToSkip, message, ...args);\n  };\n  debugInstance.printStackTrace = (stackTrace, title): void => {\n    printStackTrace(namespace, stackTrace, title);\n  };\n  return debugInstance;\n}\n\n/**\n * Returns a debugger instance for the `obsidian-dev-utils` library.\n *\n * @param namespace - The namespace for the debugger instance.\n * @returns A debugger instance for the `obsidian-dev-utils` library.\n */\nexport function getLibDebugger(namespace: string): DebuggerEx {\n  const prefix = currentPluginId ? `${currentPluginId}:` : '';\n  return getDebugger(`${prefix}obsidian-dev-utils:${namespace}`);\n}\n\n/**\n * Adds the DEBUG variable to the window object.\n *\n * @param pluginId - The plugin ID.\n */\nexport function initDebugHelpers(pluginId: string): void {\n  currentPluginId = pluginId;\n  (window as Partial<DebugWrapper>).DEBUG = {\n    disable: disableNamespaces,\n    enable: enableNamespaces,\n    get: getNamespaces,\n    set: setNamespaces\n  };\n}\n\nfunction disableNamespaces(namespaces: string | string[]): void {\n  const set = new Set(getNamespaces());\n  for (const namespace of toArray(namespaces)) {\n    if (namespace.startsWith(NEGATED_NAMESPACE_PREFIX)) {\n      continue;\n    }\n    const negatedNamespace = NEGATED_NAMESPACE_PREFIX + namespace;\n    if (set.has(namespace)) {\n      set.delete(namespace);\n    }\n    set.add(negatedNamespace);\n  }\n  setNamespaces(Array.from(set));\n}\n\nfunction enableNamespaces(namespaces: string | string[]): void {\n  const set = new Set(getNamespaces());\n  for (const namespace of toArray(namespaces)) {\n    if (!namespace.startsWith(NEGATED_NAMESPACE_PREFIX)) {\n      const negatedNamespace = NEGATED_NAMESPACE_PREFIX + namespace;\n      if (set.has(negatedNamespace)) {\n        set.delete(negatedNamespace);\n      }\n    }\n    set.add(namespace);\n  }\n  setNamespaces(Array.from(set));\n}\n\nfunction getNamespaces(): string[] {\n  return toArray(debug.load() ?? '');\n}\n\nfunction logWithCaller(namespace: string, framesToSkip: number, message: string, ...args: unknown[]): void {\n  if (!debug.enabled(namespace)) {\n    return;\n  }\n\n  /**\n   * The caller line index is 4 because the call stack is as follows:\n   *\n   * 0: Error\n   * 1:     at logWithCaller (?:?:?)\n   * 2:     at debugInstance.log (?:?:?)\n   * 3:     at debug (?:?:?)\n   * 4:     at functionName (path/to/caller.js:?:?)\n   */\n  const CALLER_LINE_INDEX = 4;\n\n  const stackLines = new Error().stack?.split('\\n') ?? [];\n  const callerLine = stackLines[CALLER_LINE_INDEX + framesToSkip] ?? '';\n  console.debug(message, ...args);\n  printStackTrace(namespace, callerLine, 'Debug message caller');\n}\n\nfunction printStackTrace(namespace: string, stackTrace: string, title?: string): void {\n  const _debugger = debug(namespace);\n\n  if (!_debugger.enabled) {\n    return;\n  }\n\n  if (!stackTrace) {\n    stackTrace = '(unavailable)';\n  }\n  if (!title) {\n    title = 'Caller stack trace';\n  }\n\n  _debugger.log(title);\n  console.debug(`StackTraceFakeError\\n${stackTrace}`);\n}\n\nfunction setNamespaces(namespaces: string | string[]): void {\n  debug.enable(toArray(namespaces).join(NAMESPACE_SEPARATOR));\n}\n\nfunction toArray(namespaces: string | string[]): string[] {\n  return typeof namespaces === 'string' ? namespaces.split(NAMESPACE_SEPARATOR).filter(Boolean) : namespaces.flatMap(toArray);\n}\n"],
  "mappings": ";;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAOA,mBAAkB;AAYlB,IAAI,kBAAkB;AACtB,MAAM,sBAAsB;AAC5B,MAAM,2BAA2B;AAS1B,SAAS,YAAY,WAAmB,eAAe,GAAe;AAC3E,QAAM,oBAAgB,aAAAA,SAAM,SAAS;AACrC,gBAAc,MAAM,CAAC,YAAoB,SAA0B;AACjE,kBAAc,WAAW,cAAc,SAAS,GAAG,IAAI;AAAA,EACzD;AACA,gBAAc,kBAAkB,CAAC,YAAY,UAAgB;AAC3D,oBAAgB,WAAW,YAAY,KAAK;AAAA,EAC9C;AACA,SAAO;AACT;AAQO,SAAS,eAAe,WAA+B;AAC5D,QAAM,SAAS,kBAAkB,GAAG,eAAe,MAAM;AACzD,SAAO,YAAY,GAAG,MAAM,sBAAsB,SAAS,EAAE;AAC/D;AAOO,SAAS,iBAAiB,UAAwB;AACvD,oBAAkB;AAClB,EAAC,OAAiC,QAAQ;AAAA,IACxC,SAAS;AAAA,IACT,QAAQ;AAAA,IACR,KAAK;AAAA,IACL,KAAK;AAAA,EACP;AACF;AAEA,SAAS,kBAAkB,YAAqC;AAC9D,QAAM,MAAM,IAAI,IAAI,cAAc,CAAC;AACnC,aAAW,aAAa,QAAQ,UAAU,GAAG;AAC3C,QAAI,UAAU,WAAW,wBAAwB,GAAG;AAClD;AAAA,IACF;AACA,UAAM,mBAAmB,2BAA2B;AACpD,QAAI,IAAI,IAAI,SAAS,GAAG;AACtB,UAAI,OAAO,SAAS;AAAA,IACtB;AACA,QAAI,IAAI,gBAAgB;AAAA,EAC1B;AACA,gBAAc,MAAM,KAAK,GAAG,CAAC;AAC/B;AAEA,SAAS,iBAAiB,YAAqC;AAC7D,QAAM,MAAM,IAAI,IAAI,cAAc,CAAC;AACnC,aAAW,aAAa,QAAQ,UAAU,GAAG;AAC3C,QAAI,CAAC,UAAU,WAAW,wBAAwB,GAAG;AACnD,YAAM,mBAAmB,2BAA2B;AACpD,UAAI,IAAI,IAAI,gBAAgB,GAAG;AAC7B,YAAI,OAAO,gBAAgB;AAAA,MAC7B;AAAA,IACF;AACA,QAAI,IAAI,SAAS;AAAA,EACnB;AACA,gBAAc,MAAM,KAAK,GAAG,CAAC;AAC/B;AAEA,SAAS,gBAA0B;AACjC,SAAO,QAAQ,aAAAA,QAAM,KAAK,KAAK,EAAE;AACnC;AAEA,SAAS,cAAc,WAAmB,cAAsB,YAAoB,MAAuB;AACzG,MAAI,CAAC,aAAAA,QAAM,QAAQ,SAAS,GAAG;AAC7B;AAAA,EACF;AAWA,QAAM,oBAAoB;AAE1B,QAAM,aAAa,IAAI,MAAM,EAAE,OAAO,MAAM,IAAI,KAAK,CAAC;AACtD,QAAM,aAAa,WAAW,oBAAoB,YAAY,KAAK;AACnE,UAAQ,MAAM,SAAS,GAAG,IAAI;AAC9B,kBAAgB,WAAW,YAAY,sBAAsB;AAC/D;AAEA,SAAS,gBAAgB,WAAmB,YAAoB,OAAsB;AACpF,QAAM,gBAAY,aAAAA,SAAM,SAAS;AAEjC,MAAI,CAAC,UAAU,SAAS;AACtB;AAAA,EACF;AAEA,MAAI,CAAC,YAAY;AACf,iBAAa;AAAA,EACf;AACA,MAAI,CAAC,OAAO;AACV,YAAQ;AAAA,EACV;AAEA,YAAU,IAAI,KAAK;AACnB,UAAQ,MAAM;AAAA,EAAwB,UAAU,EAAE;AACpD;AAEA,SAAS,cAAc,YAAqC;AAC1D,eAAAA,QAAM,OAAO,QAAQ,UAAU,EAAE,KAAK,mBAAmB,CAAC;AAC5D;AAEA,SAAS,QAAQ,YAAyC;AACxD,SAAO,OAAO,eAAe,WAAW,WAAW,MAAM,mBAAmB,EAAE,OAAO,OAAO,IAAI,WAAW,QAAQ,OAAO;AAC5H;",
  "names": ["debug"]
}

@@ -10,9 +10,10 @@ interface DebuggerEx extends Debugger {
10
10
  * Returns a debugger instance with a log function that includes the caller's file name and line number.
11
11
  *
12
12
  * @param namespace - The namespace for the debugger instance.
13
+ * @param framesToSkip - The number of frames to skip in the stack trace.
13
14
  * @returns A debugger instance with a log function that includes the caller's file name and line number.
14
15
  */
15
- export declare function getDebugger(namespace: string): DebuggerEx;
16
+ export declare function getDebugger(namespace: string, framesToSkip?: number): DebuggerEx;
16
17
  /**
17
18
  * Returns a debugger instance for the `obsidian-dev-utils` library.
18
19
  *
@@ -92,7 +92,8 @@ class PluginBase extends import_obsidian.Plugin {
92
92
  * @param args - The arguments to log.
93
93
  */
94
94
  consoleDebug(message, ...args) {
95
- const _debugger = (0, import_Debug.getDebugger)(this.manifest.id);
95
+ const FRAMES_TO_SKIP = 1;
96
+ const _debugger = (0, import_Debug.getDebugger)(this.manifest.id, FRAMES_TO_SKIP);
96
97
  _debugger(message, ...args);
97
98
  }
98
99
  /**
@@ -173,4 +174,4 @@ ${message}`);
173
174
  0 && (module.exports = {
174
175
  PluginBase
175
176
  });
176
- //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../../src/obsidian/Plugin/PluginBase.ts"],
  "sourcesContent": ["var __process = globalThis['process'] ?? {\n  \"browser\": true,\n  \"cwd\": ()=>\"/\",\n  \"env\": {},\n  \"platform\": \"android\"\n};\n/**\n * @packageDocumentation PluginBase\n * Base class for Obsidian plugins providing utility methods for settings management, error handling, and notifications.\n *\n * This class simplifies the process of managing plugin settings, displaying notifications, and handling errors.\n * Subclasses should implement methods to create default settings and settings tabs, and complete plugin-specific\n * loading tasks.\n */\n\nimport type { PluginManifest } from 'obsidian';\n\nimport {\n  App,\n  Notice,\n  Plugin,\n  PluginSettingTab\n} from 'obsidian';\n\nimport type { MaybePromise } from '../../Async.ts';\nimport type { EmptySettings } from './EmptySettings.ts';\nimport type { PluginSettingsBase } from './PluginSettingsBase.ts';\n\nimport {\n  getDebugger,\n  initDebugHelpers\n} from '../../Debug.ts';\nimport { registerAsyncErrorEventHandler } from '../../Error.ts';\nimport { noop } from '../../Function.ts';\n\n/**\n * Base class for creating Obsidian plugins with built-in support for settings management, error handling, and notifications.\n *\n * @typeParam PluginSettings - The type representing the plugin settings object.\n */\nexport abstract class PluginBase<PluginSettings extends PluginSettingsBase = EmptySettings> extends Plugin {\n  /**\n   * Gets the AbortSignal used for aborting long-running operations.\n   *\n   * @returns The abort signal.\n   */\n  public get abortSignal(): AbortSignal {\n    return this._abortSignal;\n  }\n\n  /**\n   * Gets a copy of the current plugin settings.\n   *\n   * @returns A copy of the plugin settings.\n   */\n  public get settingsCopy(): PluginSettings {\n    return this.createPluginSettings(this.settings.toJSON());\n  }\n\n  /**\n   * Gets the plugin settings.\n   *\n   * @returns The plugin settings.\n   */\n  protected get settings(): PluginSettings {\n    return this._settings;\n  }\n\n  private _abortSignal!: AbortSignal;\n\n  private _settings!: PluginSettings;\n\n  private notice?: Notice;\n\n  /**\n   * Constructs a new PluginBase instance.\n   *\n   * @param app - The Obsidian app instance.\n   * @param manifest - The plugin manifest.\n   */\n  public constructor(app: App, manifest: PluginManifest) {\n    super(app, manifest);\n    initDebugHelpers(manifest.id);\n    console.debug(`Debug messages for plugin ${manifest.id} are not shown by default. Set https://github.com/mnaoumov/obsidian-dev-utils/?tab=readme-ov-file#debugging for more information`);\n  }\n\n  /**\n   * Logs a message to the console.\n   *\n   * Use instead of `console.debug()`.\n   *\n   * Those messages are not shown by default, but they can be shown by enabling `your-plugin-id` debugger namespace.\n   *\n   * @see {@link https://github.com/mnaoumov/obsidian-dev-utils/?tab=readme-ov-file#debugging} for more information.\n   *\n   * @param message - The message to log.\n   * @param args - The arguments to log.\n   */\n  public consoleDebug(message: string, ...args: unknown[]): void {\n    const _debugger = getDebugger(this.manifest.id);\n    _debugger(message, ...args);\n  }\n\n  /**\n   * Called when the plugin is loaded\n   */\n  public override async onload(): Promise<void> {\n    this.register(registerAsyncErrorEventHandler(() => {\n      this.showNotice('An unhandled error occurred. Please check the console for more information.');\n    }));\n\n    await this.loadSettings();\n    const pluginSettingsTab = this.createPluginSettingsTab();\n    if (pluginSettingsTab) {\n      this.addSettingTab(pluginSettingsTab);\n    }\n\n    const abortController = new AbortController();\n    this._abortSignal = abortController.signal;\n    this.register(() => {\n      abortController.abort();\n    });\n    await this.onloadComplete();\n    this.app.workspace.onLayoutReady(this.onLayoutReady.bind(this));\n  }\n\n  /**\n   * Saves the new plugin settings.\n   *\n   * @param newSettings - The new settings to save.\n   * @returns A promise that resolves when the settings are saved.\n   */\n  public async saveSettings(newSettings: PluginSettings): Promise<void> {\n    const json = newSettings.toJSON();\n    this._settings = this.createPluginSettings(json);\n    await this.saveData(json);\n  }\n\n  /**\n   * Creates the plugin settings. This method must be implemented by subclasses.\n   *\n   * @param data - The data to create the plugin settings from.\n   * @returns The plugin settings.\n   */\n  protected abstract createPluginSettings(data: unknown): PluginSettings;\n\n  /**\n   * Creates a plugin settings tab. This method must be implemented by subclasses.\n   *\n   * @returns The settings tab or null if not applicable.\n   */\n  protected abstract createPluginSettingsTab(): null | PluginSettingTab;\n\n  /**\n   * Called when the layout is ready. This method can be overridden by subclasses to perform actions once\n   * the layout is ready.\n   *\n   * @returns A promise or void indicating the completion of the layout setup.\n   */\n  protected onLayoutReady(): MaybePromise<void> {\n    noop();\n  }\n\n  /**\n   * Called when the plugin loading is complete. This method must be implemented by subclasses to perform\n   * any additional setup required after loading is complete.\n   *\n   * @returns A promise or void indicating the completion of the load process.\n   */\n  protected onloadComplete(): MaybePromise<void> {\n    noop();\n  }\n\n  /**\n   * Displays a notice message to the user.\n   *\n   * @param message - The message to display.\n   */\n  protected showNotice(message: string): void {\n    if (this.notice) {\n      this.notice.hide();\n    }\n\n    this.notice = new Notice(`${this.manifest.name}\\n${message}`);\n  }\n\n  /**\n   * Loads the plugin settings from the saved data.\n   *\n   * @returns A promise that resolves when the settings are loaded.\n   */\n  private async loadSettings(): Promise<void> {\n    const data = await this.loadData() as unknown;\n    this._settings = this.createPluginSettings(data);\n    if (this._settings.shouldSaveAfterLoad()) {\n      await this.saveSettings(this._settings);\n    }\n  }\n}\n"],
  "mappings": ";;;;;;;;;;;;;;;;;;;;;;;;;;;AAAA;AAAA;AAAA;AAAA;AAAA;AAiBA,sBAKO;AAMP,mBAGO;AACP,mBAA+C;AAC/C,sBAAqB;AAjCrB,IAAI,YAAY,WAAW,SAAS,KAAK;AAAA,EACvC,WAAW;AAAA,EACX,OAAO,MAAI;AAAA,EACX,OAAO,CAAC;AAAA,EACR,YAAY;AACd;AAmCO,MAAe,mBAA8E,uBAAO;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAMzG,IAAW,cAA2B;AACpC,WAAO,KAAK;AAAA,EACd;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAOA,IAAW,eAA+B;AACxC,WAAO,KAAK,qBAAqB,KAAK,SAAS,OAAO,CAAC;AAAA,EACzD;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAOA,IAAc,WAA2B;AACvC,WAAO,KAAK;AAAA,EACd;AAAA,EAEQ;AAAA,EAEA;AAAA,EAEA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAQD,YAAY,KAAU,UAA0B;AACrD,UAAM,KAAK,QAAQ;AACnB,uCAAiB,SAAS,EAAE;AAC5B,YAAQ,MAAM,6BAA6B,SAAS,EAAE,kIAAkI;AAAA,EAC1L;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAcO,aAAa,YAAoB,MAAuB;AAC7D,UAAM,gBAAY,0BAAY,KAAK,SAAS,EAAE;AAC9C,cAAU,SAAS,GAAG,IAAI;AAAA,EAC5B;AAAA;AAAA;AAAA;AAAA,EAKA,MAAsB,SAAwB;AAC5C,SAAK,aAAS,6CAA+B,MAAM;AACjD,WAAK,WAAW,6EAA6E;AAAA,IAC/F,CAAC,CAAC;AAEF,UAAM,KAAK,aAAa;AACxB,UAAM,oBAAoB,KAAK,wBAAwB;AACvD,QAAI,mBAAmB;AACrB,WAAK,cAAc,iBAAiB;AAAA,IACtC;AAEA,UAAM,kBAAkB,IAAI,gBAAgB;AAC5C,SAAK,eAAe,gBAAgB;AACpC,SAAK,SAAS,MAAM;AAClB,sBAAgB,MAAM;AAAA,IACxB,CAAC;AACD,UAAM,KAAK,eAAe;AAC1B,SAAK,IAAI,UAAU,cAAc,KAAK,cAAc,KAAK,IAAI,CAAC;AAAA,EAChE;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAQA,MAAa,aAAa,aAA4C;AACpE,UAAM,OAAO,YAAY,OAAO;AAChC,SAAK,YAAY,KAAK,qBAAqB,IAAI;AAC/C,UAAM,KAAK,SAAS,IAAI;AAAA,EAC1B;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAuBU,gBAAoC;AAC5C,8BAAK;AAAA,EACP;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAQU,iBAAqC;AAC7C,8BAAK;AAAA,EACP;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAOU,WAAW,SAAuB;AAC1C,QAAI,KAAK,QAAQ;AACf,WAAK,OAAO,KAAK;AAAA,IACnB;AAEA,SAAK,SAAS,IAAI,uBAAO,GAAG,KAAK,SAAS,IAAI;AAAA,EAAK,OAAO,EAAE;AAAA,EAC9D;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAOA,MAAc,eAA8B;AAC1C,UAAM,OAAO,MAAM,KAAK,SAAS;AACjC,SAAK,YAAY,KAAK,qBAAqB,IAAI;AAC/C,QAAI,KAAK,UAAU,oBAAoB,GAAG;AACxC,YAAM,KAAK,aAAa,KAAK,SAAS;AAAA,IACxC;AAAA,EACF;AACF;",
  "names": []
}

177
+ //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../../src/obsidian/Plugin/PluginBase.ts"],
  "sourcesContent": ["var __process = globalThis['process'] ?? {\n  \"browser\": true,\n  \"cwd\": ()=>\"/\",\n  \"env\": {},\n  \"platform\": \"android\"\n};\n/**\n * @packageDocumentation PluginBase\n * Base class for Obsidian plugins providing utility methods for settings management, error handling, and notifications.\n *\n * This class simplifies the process of managing plugin settings, displaying notifications, and handling errors.\n * Subclasses should implement methods to create default settings and settings tabs, and complete plugin-specific\n * loading tasks.\n */\n\nimport type { PluginManifest } from 'obsidian';\n\nimport {\n  App,\n  Notice,\n  Plugin,\n  PluginSettingTab\n} from 'obsidian';\n\nimport type { MaybePromise } from '../../Async.ts';\nimport type { EmptySettings } from './EmptySettings.ts';\nimport type { PluginSettingsBase } from './PluginSettingsBase.ts';\n\nimport {\n  getDebugger,\n  initDebugHelpers\n} from '../../Debug.ts';\nimport { registerAsyncErrorEventHandler } from '../../Error.ts';\nimport { noop } from '../../Function.ts';\n\n/**\n * Base class for creating Obsidian plugins with built-in support for settings management, error handling, and notifications.\n *\n * @typeParam PluginSettings - The type representing the plugin settings object.\n */\nexport abstract class PluginBase<PluginSettings extends PluginSettingsBase = EmptySettings> extends Plugin {\n  /**\n   * Gets the AbortSignal used for aborting long-running operations.\n   *\n   * @returns The abort signal.\n   */\n  public get abortSignal(): AbortSignal {\n    return this._abortSignal;\n  }\n\n  /**\n   * Gets a copy of the current plugin settings.\n   *\n   * @returns A copy of the plugin settings.\n   */\n  public get settingsCopy(): PluginSettings {\n    return this.createPluginSettings(this.settings.toJSON());\n  }\n\n  /**\n   * Gets the plugin settings.\n   *\n   * @returns The plugin settings.\n   */\n  protected get settings(): PluginSettings {\n    return this._settings;\n  }\n\n  private _abortSignal!: AbortSignal;\n\n  private _settings!: PluginSettings;\n\n  private notice?: Notice;\n\n  /**\n   * Constructs a new PluginBase instance.\n   *\n   * @param app - The Obsidian app instance.\n   * @param manifest - The plugin manifest.\n   */\n  public constructor(app: App, manifest: PluginManifest) {\n    super(app, manifest);\n    initDebugHelpers(manifest.id);\n    console.debug(`Debug messages for plugin ${manifest.id} are not shown by default. Set https://github.com/mnaoumov/obsidian-dev-utils/?tab=readme-ov-file#debugging for more information`);\n  }\n\n  /**\n   * Logs a message to the console.\n   *\n   * Use instead of `console.debug()`.\n   *\n   * Those messages are not shown by default, but they can be shown by enabling `your-plugin-id` debugger namespace.\n   *\n   * @see {@link https://github.com/mnaoumov/obsidian-dev-utils/?tab=readme-ov-file#debugging} for more information.\n   *\n   * @param message - The message to log.\n   * @param args - The arguments to log.\n   */\n  public consoleDebug(message: string, ...args: unknown[]): void {\n    // Skip the `consoleDebug()` call itself\n    const FRAMES_TO_SKIP = 1;\n    const _debugger = getDebugger(this.manifest.id, FRAMES_TO_SKIP);\n    _debugger(message, ...args);\n  }\n\n  /**\n   * Called when the plugin is loaded\n   */\n  public override async onload(): Promise<void> {\n    this.register(registerAsyncErrorEventHandler(() => {\n      this.showNotice('An unhandled error occurred. Please check the console for more information.');\n    }));\n\n    await this.loadSettings();\n    const pluginSettingsTab = this.createPluginSettingsTab();\n    if (pluginSettingsTab) {\n      this.addSettingTab(pluginSettingsTab);\n    }\n\n    const abortController = new AbortController();\n    this._abortSignal = abortController.signal;\n    this.register(() => {\n      abortController.abort();\n    });\n    await this.onloadComplete();\n    this.app.workspace.onLayoutReady(this.onLayoutReady.bind(this));\n  }\n\n  /**\n   * Saves the new plugin settings.\n   *\n   * @param newSettings - The new settings to save.\n   * @returns A promise that resolves when the settings are saved.\n   */\n  public async saveSettings(newSettings: PluginSettings): Promise<void> {\n    const json = newSettings.toJSON();\n    this._settings = this.createPluginSettings(json);\n    await this.saveData(json);\n  }\n\n  /**\n   * Creates the plugin settings. This method must be implemented by subclasses.\n   *\n   * @param data - The data to create the plugin settings from.\n   * @returns The plugin settings.\n   */\n  protected abstract createPluginSettings(data: unknown): PluginSettings;\n\n  /**\n   * Creates a plugin settings tab. This method must be implemented by subclasses.\n   *\n   * @returns The settings tab or null if not applicable.\n   */\n  protected abstract createPluginSettingsTab(): null | PluginSettingTab;\n\n  /**\n   * Called when the layout is ready. This method can be overridden by subclasses to perform actions once\n   * the layout is ready.\n   *\n   * @returns A promise or void indicating the completion of the layout setup.\n   */\n  protected onLayoutReady(): MaybePromise<void> {\n    noop();\n  }\n\n  /**\n   * Called when the plugin loading is complete. This method must be implemented by subclasses to perform\n   * any additional setup required after loading is complete.\n   *\n   * @returns A promise or void indicating the completion of the load process.\n   */\n  protected onloadComplete(): MaybePromise<void> {\n    noop();\n  }\n\n  /**\n   * Displays a notice message to the user.\n   *\n   * @param message - The message to display.\n   */\n  protected showNotice(message: string): void {\n    if (this.notice) {\n      this.notice.hide();\n    }\n\n    this.notice = new Notice(`${this.manifest.name}\\n${message}`);\n  }\n\n  /**\n   * Loads the plugin settings from the saved data.\n   *\n   * @returns A promise that resolves when the settings are loaded.\n   */\n  private async loadSettings(): Promise<void> {\n    const data = await this.loadData() as unknown;\n    this._settings = this.createPluginSettings(data);\n    if (this._settings.shouldSaveAfterLoad()) {\n      await this.saveSettings(this._settings);\n    }\n  }\n}\n"],
  "mappings": ";;;;;;;;;;;;;;;;;;;;;;;;;;;AAAA;AAAA;AAAA;AAAA;AAAA;AAiBA,sBAKO;AAMP,mBAGO;AACP,mBAA+C;AAC/C,sBAAqB;AAjCrB,IAAI,YAAY,WAAW,SAAS,KAAK;AAAA,EACvC,WAAW;AAAA,EACX,OAAO,MAAI;AAAA,EACX,OAAO,CAAC;AAAA,EACR,YAAY;AACd;AAmCO,MAAe,mBAA8E,uBAAO;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAMzG,IAAW,cAA2B;AACpC,WAAO,KAAK;AAAA,EACd;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAOA,IAAW,eAA+B;AACxC,WAAO,KAAK,qBAAqB,KAAK,SAAS,OAAO,CAAC;AAAA,EACzD;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAOA,IAAc,WAA2B;AACvC,WAAO,KAAK;AAAA,EACd;AAAA,EAEQ;AAAA,EAEA;AAAA,EAEA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAQD,YAAY,KAAU,UAA0B;AACrD,UAAM,KAAK,QAAQ;AACnB,uCAAiB,SAAS,EAAE;AAC5B,YAAQ,MAAM,6BAA6B,SAAS,EAAE,kIAAkI;AAAA,EAC1L;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAcO,aAAa,YAAoB,MAAuB;AAE7D,UAAM,iBAAiB;AACvB,UAAM,gBAAY,0BAAY,KAAK,SAAS,IAAI,cAAc;AAC9D,cAAU,SAAS,GAAG,IAAI;AAAA,EAC5B;AAAA;AAAA;AAAA;AAAA,EAKA,MAAsB,SAAwB;AAC5C,SAAK,aAAS,6CAA+B,MAAM;AACjD,WAAK,WAAW,6EAA6E;AAAA,IAC/F,CAAC,CAAC;AAEF,UAAM,KAAK,aAAa;AACxB,UAAM,oBAAoB,KAAK,wBAAwB;AACvD,QAAI,mBAAmB;AACrB,WAAK,cAAc,iBAAiB;AAAA,IACtC;AAEA,UAAM,kBAAkB,IAAI,gBAAgB;AAC5C,SAAK,eAAe,gBAAgB;AACpC,SAAK,SAAS,MAAM;AAClB,sBAAgB,MAAM;AAAA,IACxB,CAAC;AACD,UAAM,KAAK,eAAe;AAC1B,SAAK,IAAI,UAAU,cAAc,KAAK,cAAc,KAAK,IAAI,CAAC;AAAA,EAChE;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAQA,MAAa,aAAa,aAA4C;AACpE,UAAM,OAAO,YAAY,OAAO;AAChC,SAAK,YAAY,KAAK,qBAAqB,IAAI;AAC/C,UAAM,KAAK,SAAS,IAAI;AAAA,EAC1B;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAuBU,gBAAoC;AAC5C,8BAAK;AAAA,EACP;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAQU,iBAAqC;AAC7C,8BAAK;AAAA,EACP;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAOU,WAAW,SAAuB;AAC1C,QAAI,KAAK,QAAQ;AACf,WAAK,OAAO,KAAK;AAAA,IACnB;AAEA,SAAK,SAAS,IAAI,uBAAO,GAAG,KAAK,SAAS,IAAI;AAAA,EAAK,OAAO,EAAE;AAAA,EAC9D;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAOA,MAAc,eAA8B;AAC1C,UAAM,OAAO,MAAM,KAAK,SAAS;AACjC,SAAK,YAAY,KAAK,qBAAqB,IAAI;AAC/C,QAAI,KAAK,UAAU,oBAAoB,GAAG;AACxC,YAAM,KAAK,aAAa,KAAK,SAAS;AAAA,IACxC;AAAA,EACF;AACF;",
  "names": []
}

package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "obsidian-dev-utils",
3
- "version": "13.10.0",
3
+ "version": "13.11.0",
4
4
  "description": "This is the collection of useful functions that you can use for your Obsidian plugin development",
5
5
  "main": "./dist/lib/index.cjs",
6
6
  "types": "./dist/lib/index.d.ts",
@@ -51,7 +51,7 @@
51
51
  "@typescript-eslint/eslint-plugin": "^8.20.0",
52
52
  "@typescript-eslint/parser": "^8.20.0",
53
53
  "adm-zip": "^0.5.16",
54
- "better-typescript-lib": "^2.10.0",
54
+ "better-typescript-lib": "^2.10.1",
55
55
  "commander": "^13.0.0",
56
56
  "cspell": "^8.17.2",
57
57
  "debug": "^4.4.0",