@nitra/zebra 8.0.3 → 8.2.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.
@@ -11,7 +11,7 @@ Pod::Spec.new do |s|
11
11
  s.author = package['author']
12
12
  s.source = { :git => package['repository']['url'], :tag => s.version.to_s }
13
13
  s.source_files = 'ios/Sources/**/*.{swift,h,m,c,cc,mm,cpp}'
14
- s.ios.deployment_target = '15.0'
14
+ s.ios.deployment_target = '16.0'
15
15
  s.dependency 'Capacitor'
16
16
  s.frameworks = 'ExternalAccessory'
17
17
  s.swift_version = '5.1'
package/Package.swift CHANGED
@@ -3,7 +3,7 @@ import PackageDescription
3
3
 
4
4
  let package = Package(
5
5
  name: "NitraZebra",
6
- platforms: [.iOS(.v15)],
6
+ platforms: [.iOS(.v16)],
7
7
  products: [
8
8
  .library(
9
9
  name: "NitraZebra",
@@ -20,6 +20,9 @@ let package = Package(
20
20
  .product(name: "Cordova", package: "capacitor-swift-pm")
21
21
  ],
22
22
  path: "ios/Sources/ZebraPlugin",
23
+ swiftSettings: [
24
+ .enableUpcomingFeature("StrictConcurrency")
25
+ ],
23
26
  linkerSettings: [
24
27
  .linkedFramework("ExternalAccessory")
25
28
  ]),
package/README.md CHANGED
@@ -7,7 +7,7 @@ Capacitor-плагін для друку ZPL на принтерах Zebra з п
7
7
  - **Capacitor** 7.0.0 або новіший
8
8
  - **Веб**: браузер з підтримкою [Web Serial API](https://developer.mozilla.org/en-US/docs/Web/API/Web_Serial_API) (Chrome, Edge, Opera)
9
9
  - **Android**: minSdk 23, targetSdk 35
10
- - **iOS**: 14.0+
10
+ - **iOS**: 16.0+
11
11
 
12
12
  ## Встановлення
13
13
 
package/dist/plugin.js CHANGED
@@ -18,7 +18,7 @@ var __exportAll = (all, no_symbols) => {
18
18
  };
19
19
 
20
20
  //#endregion
21
- //#region ../node_modules/.bun/@capacitor+preferences@7.0.3+2476a4e6bb24aa03/node_modules/@capacitor/preferences/dist/esm/web.js
21
+ //#region ../node_modules/.bun/@capacitor+preferences@8.0.0+2476a4e6bb24aa03/node_modules/@capacitor/preferences/dist/esm/web.js
22
22
  var web_exports$1 = /* @__PURE__ */ __exportAll({ PreferencesWeb: () => PreferencesWeb });
23
23
  var PreferencesWeb;
24
24
  var init_web$1 = __esmMin((() => {
@@ -90,11 +90,18 @@ var init_web$1 = __esmMin((() => {
90
90
  }));
91
91
 
92
92
  //#endregion
93
- //#region ../node_modules/.bun/@capacitor+preferences@7.0.3+2476a4e6bb24aa03/node_modules/@capacitor/preferences/dist/esm/index.js
93
+ //#region ../node_modules/.bun/@capacitor+preferences@8.0.0+2476a4e6bb24aa03/node_modules/@capacitor/preferences/dist/esm/index.js
94
94
  const Preferences = registerPlugin("Preferences", { web: () => Promise.resolve().then(() => (init_web$1(), web_exports$1)).then((m) => new m.PreferencesWeb()) });
95
95
 
96
96
  //#endregion
97
97
  //#region src/utils/validate.js
98
+ /**
99
+ * Нормалізує аргумент ZPL для друку: перевіряє рядок і повертає обрізану команду.
100
+ * @param {string} zpl - рядок ZPL-команд
101
+ * @returns {string} обрізаний рядок ZPL
102
+ * @throws {TypeError} якщо zpl не рядок
103
+ * @throws {Error} якщо команда порожня
104
+ */
98
105
  function normalizePrintArg(zpl) {
99
106
  if (typeof zpl !== "string") throw new TypeError("Expected \"zpl\" to be a string");
100
107
  const command = zpl.trim();
@@ -1 +1 @@
1
- {"version":3,"file":"plugin.js","names":["createConsola","getViteEnvVar"],"sources":["../../node_modules/.bun/@capacitor+preferences@7.0.3+2476a4e6bb24aa03/node_modules/@capacitor/preferences/dist/esm/web.js","../../node_modules/.bun/@capacitor+preferences@7.0.3+2476a4e6bb24aa03/node_modules/@capacitor/preferences/dist/esm/index.js","../src/utils/validate.js","../../node_modules/.bun/consola@3.4.2/node_modules/consola/dist/core.mjs","../../node_modules/.bun/consola@3.4.2/node_modules/consola/dist/browser.mjs","../../node_modules/.bun/@nitra+consola@2.4.1/node_modules/@nitra/consola/src/otel-reporter.js","../../node_modules/.bun/@nitra+consola@2.4.1/node_modules/@nitra/consola/src/browser.js","../src/web.js","../src/index.js"],"sourcesContent":["import { WebPlugin } from '@capacitor/core';\nexport class PreferencesWeb extends WebPlugin {\n constructor() {\n super(...arguments);\n this.group = 'CapacitorStorage';\n }\n async configure({ group }) {\n if (typeof group === 'string') {\n this.group = group;\n }\n }\n async get(options) {\n const value = this.impl.getItem(this.applyPrefix(options.key));\n return { value };\n }\n async set(options) {\n this.impl.setItem(this.applyPrefix(options.key), options.value);\n }\n async remove(options) {\n this.impl.removeItem(this.applyPrefix(options.key));\n }\n async keys() {\n const keys = this.rawKeys().map(k => k.substring(this.prefix.length));\n return { keys };\n }\n async clear() {\n for (const key of this.rawKeys()) {\n this.impl.removeItem(key);\n }\n }\n async migrate() {\n var _a;\n const migrated = [];\n const existing = [];\n const oldprefix = '_cap_';\n const keys = Object.keys(this.impl).filter(k => k.indexOf(oldprefix) === 0);\n for (const oldkey of keys) {\n const key = oldkey.substring(oldprefix.length);\n const value = (_a = this.impl.getItem(oldkey)) !== null && _a !== void 0 ? _a : '';\n const { value: currentValue } = await this.get({ key });\n if (typeof currentValue === 'string') {\n existing.push(key);\n }\n else {\n await this.set({ key, value });\n migrated.push(key);\n }\n }\n return { migrated, existing };\n }\n async removeOld() {\n const oldprefix = '_cap_';\n const keys = Object.keys(this.impl).filter(k => k.indexOf(oldprefix) === 0);\n for (const oldkey of keys) {\n this.impl.removeItem(oldkey);\n }\n }\n get impl() {\n return window.localStorage;\n }\n get prefix() {\n return this.group === 'NativeStorage' ? '' : `${this.group}.`;\n }\n rawKeys() {\n return Object.keys(this.impl).filter(k => k.indexOf(this.prefix) === 0);\n }\n applyPrefix(key) {\n return this.prefix + key;\n }\n}\n//# sourceMappingURL=web.js.map","import { registerPlugin } from '@capacitor/core';\nconst Preferences = registerPlugin('Preferences', {\n web: () => import('./web').then(m => new m.PreferencesWeb()),\n});\nexport * from './definitions';\nexport { Preferences };\n//# sourceMappingURL=index.js.map","export function normalizePrintArg(zpl) {\n if (typeof zpl !== 'string') {\n throw new TypeError('Expected \"zpl\" to be a string')\n }\n\n const command = zpl.trim()\n if (!command) {\n throw new Error('ZPL команда порожня')\n }\n\n return command\n}\n","const LogLevels = {\n silent: Number.NEGATIVE_INFINITY,\n fatal: 0,\n error: 0,\n warn: 1,\n log: 2,\n info: 3,\n success: 3,\n fail: 3,\n ready: 3,\n start: 3,\n box: 3,\n debug: 4,\n trace: 5,\n verbose: Number.POSITIVE_INFINITY\n};\nconst LogTypes = {\n // Silent\n silent: {\n level: -1\n },\n // Level 0\n fatal: {\n level: LogLevels.fatal\n },\n error: {\n level: LogLevels.error\n },\n // Level 1\n warn: {\n level: LogLevels.warn\n },\n // Level 2\n log: {\n level: LogLevels.log\n },\n // Level 3\n info: {\n level: LogLevels.info\n },\n success: {\n level: LogLevels.success\n },\n fail: {\n level: LogLevels.fail\n },\n ready: {\n level: LogLevels.info\n },\n start: {\n level: LogLevels.info\n },\n box: {\n level: LogLevels.info\n },\n // Level 4\n debug: {\n level: LogLevels.debug\n },\n // Level 5\n trace: {\n level: LogLevels.trace\n },\n // Verbose\n verbose: {\n level: LogLevels.verbose\n }\n};\n\nfunction isPlainObject$1(value) {\n if (value === null || typeof value !== \"object\") {\n return false;\n }\n const prototype = Object.getPrototypeOf(value);\n if (prototype !== null && prototype !== Object.prototype && Object.getPrototypeOf(prototype) !== null) {\n return false;\n }\n if (Symbol.iterator in value) {\n return false;\n }\n if (Symbol.toStringTag in value) {\n return Object.prototype.toString.call(value) === \"[object Module]\";\n }\n return true;\n}\n\nfunction _defu(baseObject, defaults, namespace = \".\", merger) {\n if (!isPlainObject$1(defaults)) {\n return _defu(baseObject, {}, namespace, merger);\n }\n const object = Object.assign({}, defaults);\n for (const key in baseObject) {\n if (key === \"__proto__\" || key === \"constructor\") {\n continue;\n }\n const value = baseObject[key];\n if (value === null || value === void 0) {\n continue;\n }\n if (merger && merger(object, key, value, namespace)) {\n continue;\n }\n if (Array.isArray(value) && Array.isArray(object[key])) {\n object[key] = [...value, ...object[key]];\n } else if (isPlainObject$1(value) && isPlainObject$1(object[key])) {\n object[key] = _defu(\n value,\n object[key],\n (namespace ? `${namespace}.` : \"\") + key.toString(),\n merger\n );\n } else {\n object[key] = value;\n }\n }\n return object;\n}\nfunction createDefu(merger) {\n return (...arguments_) => (\n // eslint-disable-next-line unicorn/no-array-reduce\n arguments_.reduce((p, c) => _defu(p, c, \"\", merger), {})\n );\n}\nconst defu = createDefu();\n\nfunction isPlainObject(obj) {\n return Object.prototype.toString.call(obj) === \"[object Object]\";\n}\nfunction isLogObj(arg) {\n if (!isPlainObject(arg)) {\n return false;\n }\n if (!arg.message && !arg.args) {\n return false;\n }\n if (arg.stack) {\n return false;\n }\n return true;\n}\n\nlet paused = false;\nconst queue = [];\nclass Consola {\n options;\n _lastLog;\n _mockFn;\n /**\n * Creates an instance of Consola with specified options or defaults.\n *\n * @param {Partial<ConsolaOptions>} [options={}] - Configuration options for the Consola instance.\n */\n constructor(options = {}) {\n const types = options.types || LogTypes;\n this.options = defu(\n {\n ...options,\n defaults: { ...options.defaults },\n level: _normalizeLogLevel(options.level, types),\n reporters: [...options.reporters || []]\n },\n {\n types: LogTypes,\n throttle: 1e3,\n throttleMin: 5,\n formatOptions: {\n date: true,\n colors: false,\n compact: true\n }\n }\n );\n for (const type in types) {\n const defaults = {\n type,\n ...this.options.defaults,\n ...types[type]\n };\n this[type] = this._wrapLogFn(defaults);\n this[type].raw = this._wrapLogFn(\n defaults,\n true\n );\n }\n if (this.options.mockFn) {\n this.mockTypes();\n }\n this._lastLog = {};\n }\n /**\n * Gets the current log level of the Consola instance.\n *\n * @returns {number} The current log level.\n */\n get level() {\n return this.options.level;\n }\n /**\n * Sets the minimum log level that will be output by the instance.\n *\n * @param {number} level - The new log level to set.\n */\n set level(level) {\n this.options.level = _normalizeLogLevel(\n level,\n this.options.types,\n this.options.level\n );\n }\n /**\n * Displays a prompt to the user and returns the response.\n * Throw an error if `prompt` is not supported by the current configuration.\n *\n * @template T\n * @param {string} message - The message to display in the prompt.\n * @param {T} [opts] - Optional options for the prompt. See {@link PromptOptions}.\n * @returns {promise<T>} A promise that infer with the prompt options. See {@link PromptOptions}.\n */\n prompt(message, opts) {\n if (!this.options.prompt) {\n throw new Error(\"prompt is not supported!\");\n }\n return this.options.prompt(message, opts);\n }\n /**\n * Creates a new instance of Consola, inheriting options from the current instance, with possible overrides.\n *\n * @param {Partial<ConsolaOptions>} options - Optional overrides for the new instance. See {@link ConsolaOptions}.\n * @returns {ConsolaInstance} A new Consola instance. See {@link ConsolaInstance}.\n */\n create(options) {\n const instance = new Consola({\n ...this.options,\n ...options\n });\n if (this._mockFn) {\n instance.mockTypes(this._mockFn);\n }\n return instance;\n }\n /**\n * Creates a new Consola instance with the specified default log object properties.\n *\n * @param {InputLogObject} defaults - Default properties to include in any log from the new instance. See {@link InputLogObject}.\n * @returns {ConsolaInstance} A new Consola instance. See {@link ConsolaInstance}.\n */\n withDefaults(defaults) {\n return this.create({\n ...this.options,\n defaults: {\n ...this.options.defaults,\n ...defaults\n }\n });\n }\n /**\n * Creates a new Consola instance with a specified tag, which will be included in every log.\n *\n * @param {string} tag - The tag to include in each log of the new instance.\n * @returns {ConsolaInstance} A new Consola instance. See {@link ConsolaInstance}.\n */\n withTag(tag) {\n return this.withDefaults({\n tag: this.options.defaults.tag ? this.options.defaults.tag + \":\" + tag : tag\n });\n }\n /**\n * Adds a custom reporter to the Consola instance.\n * Reporters will be called for each log message, depending on their implementation and log level.\n *\n * @param {ConsolaReporter} reporter - The reporter to add. See {@link ConsolaReporter}.\n * @returns {Consola} The current Consola instance.\n */\n addReporter(reporter) {\n this.options.reporters.push(reporter);\n return this;\n }\n /**\n * Removes a custom reporter from the Consola instance.\n * If no reporter is specified, all reporters will be removed.\n *\n * @param {ConsolaReporter} reporter - The reporter to remove. See {@link ConsolaReporter}.\n * @returns {Consola} The current Consola instance.\n */\n removeReporter(reporter) {\n if (reporter) {\n const i = this.options.reporters.indexOf(reporter);\n if (i !== -1) {\n return this.options.reporters.splice(i, 1);\n }\n } else {\n this.options.reporters.splice(0);\n }\n return this;\n }\n /**\n * Replaces all reporters of the Consola instance with the specified array of reporters.\n *\n * @param {ConsolaReporter[]} reporters - The new reporters to set. See {@link ConsolaReporter}.\n * @returns {Consola} The current Consola instance.\n */\n setReporters(reporters) {\n this.options.reporters = Array.isArray(reporters) ? reporters : [reporters];\n return this;\n }\n wrapAll() {\n this.wrapConsole();\n this.wrapStd();\n }\n restoreAll() {\n this.restoreConsole();\n this.restoreStd();\n }\n /**\n * Overrides console methods with Consola logging methods for consistent logging.\n */\n wrapConsole() {\n for (const type in this.options.types) {\n if (!console[\"__\" + type]) {\n console[\"__\" + type] = console[type];\n }\n console[type] = this[type].raw;\n }\n }\n /**\n * Restores the original console methods, removing Consola overrides.\n */\n restoreConsole() {\n for (const type in this.options.types) {\n if (console[\"__\" + type]) {\n console[type] = console[\"__\" + type];\n delete console[\"__\" + type];\n }\n }\n }\n /**\n * Overrides standard output and error streams to redirect them through Consola.\n */\n wrapStd() {\n this._wrapStream(this.options.stdout, \"log\");\n this._wrapStream(this.options.stderr, \"log\");\n }\n _wrapStream(stream, type) {\n if (!stream) {\n return;\n }\n if (!stream.__write) {\n stream.__write = stream.write;\n }\n stream.write = (data) => {\n this[type].raw(String(data).trim());\n };\n }\n /**\n * Restores the original standard output and error streams, removing the Consola redirection.\n */\n restoreStd() {\n this._restoreStream(this.options.stdout);\n this._restoreStream(this.options.stderr);\n }\n _restoreStream(stream) {\n if (!stream) {\n return;\n }\n if (stream.__write) {\n stream.write = stream.__write;\n delete stream.__write;\n }\n }\n /**\n * Pauses logging, queues incoming logs until resumed.\n */\n pauseLogs() {\n paused = true;\n }\n /**\n * Resumes logging, processing any queued logs.\n */\n resumeLogs() {\n paused = false;\n const _queue = queue.splice(0);\n for (const item of _queue) {\n item[0]._logFn(item[1], item[2]);\n }\n }\n /**\n * Replaces logging methods with mocks if a mock function is provided.\n *\n * @param {ConsolaOptions[\"mockFn\"]} mockFn - The function to use for mocking logging methods. See {@link ConsolaOptions[\"mockFn\"]}.\n */\n mockTypes(mockFn) {\n const _mockFn = mockFn || this.options.mockFn;\n this._mockFn = _mockFn;\n if (typeof _mockFn !== \"function\") {\n return;\n }\n for (const type in this.options.types) {\n this[type] = _mockFn(type, this.options.types[type]) || this[type];\n this[type].raw = this[type];\n }\n }\n _wrapLogFn(defaults, isRaw) {\n return (...args) => {\n if (paused) {\n queue.push([this, defaults, args, isRaw]);\n return;\n }\n return this._logFn(defaults, args, isRaw);\n };\n }\n _logFn(defaults, args, isRaw) {\n if ((defaults.level || 0) > this.level) {\n return false;\n }\n const logObj = {\n date: /* @__PURE__ */ new Date(),\n args: [],\n ...defaults,\n level: _normalizeLogLevel(defaults.level, this.options.types)\n };\n if (!isRaw && args.length === 1 && isLogObj(args[0])) {\n Object.assign(logObj, args[0]);\n } else {\n logObj.args = [...args];\n }\n if (logObj.message) {\n logObj.args.unshift(logObj.message);\n delete logObj.message;\n }\n if (logObj.additional) {\n if (!Array.isArray(logObj.additional)) {\n logObj.additional = logObj.additional.split(\"\\n\");\n }\n logObj.args.push(\"\\n\" + logObj.additional.join(\"\\n\"));\n delete logObj.additional;\n }\n logObj.type = typeof logObj.type === \"string\" ? logObj.type.toLowerCase() : \"log\";\n logObj.tag = typeof logObj.tag === \"string\" ? logObj.tag : \"\";\n const resolveLog = (newLog = false) => {\n const repeated = (this._lastLog.count || 0) - this.options.throttleMin;\n if (this._lastLog.object && repeated > 0) {\n const args2 = [...this._lastLog.object.args];\n if (repeated > 1) {\n args2.push(`(repeated ${repeated} times)`);\n }\n this._log({ ...this._lastLog.object, args: args2 });\n this._lastLog.count = 1;\n }\n if (newLog) {\n this._lastLog.object = logObj;\n this._log(logObj);\n }\n };\n clearTimeout(this._lastLog.timeout);\n const diffTime = this._lastLog.time && logObj.date ? logObj.date.getTime() - this._lastLog.time.getTime() : 0;\n this._lastLog.time = logObj.date;\n if (diffTime < this.options.throttle) {\n try {\n const serializedLog = JSON.stringify([\n logObj.type,\n logObj.tag,\n logObj.args\n ]);\n const isSameLog = this._lastLog.serialized === serializedLog;\n this._lastLog.serialized = serializedLog;\n if (isSameLog) {\n this._lastLog.count = (this._lastLog.count || 0) + 1;\n if (this._lastLog.count > this.options.throttleMin) {\n this._lastLog.timeout = setTimeout(\n resolveLog,\n this.options.throttle\n );\n return;\n }\n }\n } catch {\n }\n }\n resolveLog(true);\n }\n _log(logObj) {\n for (const reporter of this.options.reporters) {\n reporter.log(logObj, {\n options: this.options\n });\n }\n }\n}\nfunction _normalizeLogLevel(input, types = {}, defaultLevel = 3) {\n if (input === void 0) {\n return defaultLevel;\n }\n if (typeof input === \"number\") {\n return input;\n }\n if (types[input] && types[input].level !== void 0) {\n return types[input].level;\n }\n return defaultLevel;\n}\nConsola.prototype.add = Consola.prototype.addReporter;\nConsola.prototype.remove = Consola.prototype.removeReporter;\nConsola.prototype.clear = Consola.prototype.removeReporter;\nConsola.prototype.withScope = Consola.prototype.withTag;\nConsola.prototype.mock = Consola.prototype.mockTypes;\nConsola.prototype.pause = Consola.prototype.pauseLogs;\nConsola.prototype.resume = Consola.prototype.resumeLogs;\nfunction createConsola(options = {}) {\n return new Consola(options);\n}\n\nexport { Consola, LogLevels, LogTypes, createConsola };\n","import { createConsola as createConsola$1 } from './core.mjs';\nexport { Consola, LogLevels, LogTypes } from './core.mjs';\n\nclass BrowserReporter {\n options;\n defaultColor;\n levelColorMap;\n typeColorMap;\n constructor(options) {\n this.options = { ...options };\n this.defaultColor = \"#7f8c8d\";\n this.levelColorMap = {\n 0: \"#c0392b\",\n // Red\n 1: \"#f39c12\",\n // Yellow\n 3: \"#00BCD4\"\n // Cyan\n };\n this.typeColorMap = {\n success: \"#2ecc71\"\n // Green\n };\n }\n _getLogFn(level) {\n if (level < 1) {\n return console.__error || console.error;\n }\n if (level === 1) {\n return console.__warn || console.warn;\n }\n return console.__log || console.log;\n }\n log(logObj) {\n const consoleLogFn = this._getLogFn(logObj.level);\n const type = logObj.type === \"log\" ? \"\" : logObj.type;\n const tag = logObj.tag || \"\";\n const color = this.typeColorMap[logObj.type] || this.levelColorMap[logObj.level] || this.defaultColor;\n const style = `\n background: ${color};\n border-radius: 0.5em;\n color: white;\n font-weight: bold;\n padding: 2px 0.5em;\n `;\n const badge = `%c${[tag, type].filter(Boolean).join(\":\")}`;\n if (typeof logObj.args[0] === \"string\") {\n consoleLogFn(\n `${badge}%c ${logObj.args[0]}`,\n style,\n // Empty string as style resets to default console style\n \"\",\n ...logObj.args.slice(1)\n );\n } else {\n consoleLogFn(badge, style, ...logObj.args);\n }\n }\n}\n\nfunction createConsola(options = {}) {\n const consola2 = createConsola$1({\n reporters: options.reporters || [new BrowserReporter({})],\n prompt(message, options2 = {}) {\n if (options2.type === \"confirm\") {\n return Promise.resolve(confirm(message));\n }\n return Promise.resolve(prompt(message));\n },\n ...options\n });\n return consola2;\n}\nconst consola = createConsola();\n\nexport { consola, createConsola, consola as default };\n","// Отримання змінних середовища з Vite (import.meta.env)\nconst getViteEnvVar = name => {\n try {\n // eslint-disable-next-line no-undef\n const viteName = `VITE_${name}`\n // eslint-disable-next-line no-undef\n return import.meta?.env?.[viteName]\n } catch {\n return null\n }\n}\n\n// OpenTelemetry Reporter для експорту логів до OpenTelemetry Collector через HTTP\nexport class OpenTelemetryReporter {\n constructor(options = {}) {\n this.endpoint =\n options.endpoint || getViteEnvVar('OTEL_EXPORTER_OTLP_LOGS_ENDPOINT') || 'http://localhost:4318/v1/logs'\n this.serviceName = options.serviceName || getViteEnvVar('OTEL_SERVICE_NAME') || 'consola-service'\n this.serviceVersion = options.serviceVersion || getViteEnvVar('OTEL_SERVICE_VERSION') || '1.0.0'\n this.batchSize = options.batchSize || parseInt(getViteEnvVar('OTEL_BATCH_SIZE') || '10', 10)\n this.flushInterval = options.flushInterval || parseInt(getViteEnvVar('OTEL_FLUSH_INTERVAL') || '5000', 10)\n this.buffer = []\n this.flushTimer = null\n this.headers = {\n 'Content-Type': 'application/json',\n ...options.headers\n }\n\n // Запускаємо таймер для періодичної відправки\n if (this.flushInterval > 0) {\n this.startFlushTimer()\n }\n }\n\n formatMessage(logObj) {\n const args = logObj.args || []\n if (args.length === 0) return ''\n\n return args\n .map(arg => {\n if (arg instanceof Error) {\n let msg = `${arg.name || 'Error'}: ${arg.message || ''}`\n if (arg.stack && arg.stack !== arg.message) {\n msg += '\\n' + arg.stack\n }\n return msg\n }\n if (arg === null) return 'null'\n if (arg === undefined) return 'undefined'\n if (typeof arg === 'object') {\n try {\n return JSON.stringify(arg, null, 2)\n } catch {\n return String(arg)\n }\n }\n return String(arg)\n })\n .join(' ')\n }\n\n getSeverityNumber(type) {\n // Маппінг типів consola до OpenTelemetry severity numbers\n const severityMap = {\n trace: 1, // TRACE\n debug: 5, // DEBUG\n info: 9, // INFO\n log: 9, // INFO\n warn: 13, // WARN\n error: 17, // ERROR\n fatal: 21, // FATAL\n success: 9, // INFO\n start: 9, // INFO\n ready: 9, // INFO\n fail: 17 // ERROR\n }\n return severityMap[type] || 9\n }\n\n getSeverityText(type) {\n const textMap = {\n trace: 'TRACE',\n debug: 'DEBUG',\n info: 'INFO',\n log: 'INFO',\n warn: 'WARN',\n error: 'ERROR',\n fatal: 'FATAL',\n success: 'INFO',\n start: 'INFO',\n ready: 'INFO',\n fail: 'ERROR'\n }\n return textMap[type] || 'INFO'\n }\n\n getFileInfo() {\n try {\n const stack = new Error('Stack trace').stack\n if (!stack) return null\n\n const skipPatterns = ['OpenTelemetryReporter', 'otel-reporter.js', 'browser.js', 'consola', 'createLogger']\n const lines = stack.split('\\n').slice(4, 15)\n\n for (const line of lines) {\n const trimmed = line.trim()\n const shouldSkip = skipPatterns.some(pattern => trimmed.includes(pattern))\n\n if (!shouldSkip) {\n const match =\n trimmed.match(/\\(?([^()]+):(\\d+):(\\d+)\\)?/) || trimmed.match(/at\\s+[^(]*\\(?([^:]+):(\\d+):(\\d+)\\)?/)\n\n if (match) {\n let file = match[1].trim()\n const lineNum = match[2]\n\n if (file.includes('://')) {\n file = file.slice(file.indexOf('://') + 3).slice(file.indexOf('/'))\n }\n if (file.includes('/src/')) {\n file = file.slice(file.indexOf('/src/') + 5)\n }\n const queryIndex = file.indexOf('?')\n if (queryIndex > 0) file = file.slice(0, queryIndex)\n const hashIndex = file.indexOf('#')\n if (hashIndex > 0) file = file.slice(0, hashIndex)\n if (file.startsWith('/')) file = file.slice(1)\n\n if (file && lineNum) {\n return { file, line: parseInt(lineNum, 10), column: parseInt(match[3], 10) }\n }\n }\n }\n }\n } catch {\n // Ігноруємо помилки\n }\n return null\n }\n\n createLogRecord(logObj) {\n const type = logObj.type || 'log'\n const message = this.formatMessage(logObj)\n const fileInfo = this.getFileInfo()\n const timestamp = Date.now() * 1000000 // наносекунди\n\n const logRecord = {\n timeUnixNano: timestamp.toString(),\n severityNumber: this.getSeverityNumber(type),\n severityText: this.getSeverityText(type),\n body: {\n stringValue: message\n },\n attributes: [\n {\n key: 'log.type',\n value: { stringValue: type }\n }\n ]\n }\n\n // Додаємо інформацію про файл, якщо доступна\n if (fileInfo) {\n logRecord.attributes.push(\n {\n key: 'code.filepath',\n value: { stringValue: fileInfo.file }\n },\n {\n key: 'code.lineno',\n value: { intValue: fileInfo.line }\n }\n )\n if (fileInfo.column) {\n logRecord.attributes.push({\n key: 'code.colno',\n value: { intValue: fileInfo.column }\n })\n }\n }\n\n // Додаємо додаткові атрибути з logObj, якщо вони є\n if (logObj.extra && typeof logObj.extra === 'object') {\n for (const [key, value] of Object.entries(logObj.extra)) {\n logRecord.attributes.push({\n key: `extra.${key}`,\n value: { stringValue: String(value) }\n })\n }\n }\n\n return logRecord\n }\n\n async sendLogs(logs) {\n if (logs.length === 0) return\n\n const resourceLogs = {\n resource: {\n attributes: [\n {\n key: 'service.name',\n value: { stringValue: this.serviceName }\n },\n {\n key: 'service.version',\n value: { stringValue: this.serviceVersion }\n }\n ]\n },\n scopeLogs: [\n {\n scope: {\n name: 'consola',\n version: '1.0.0'\n },\n logRecords: logs\n }\n ]\n }\n\n const payload = {\n resourceLogs: [resourceLogs]\n }\n\n try {\n const response = await fetch(this.endpoint, {\n method: 'POST',\n mode: 'cors',\n headers: this.headers,\n body: JSON.stringify(payload)\n })\n\n if (!response.ok) {\n console.warn(`OpenTelemetry export failed: ${response.status} ${response.statusText}`)\n }\n } catch (error) {\n // Тихо ігноруємо помилки, щоб не порушити роботу додатку\n console.warn('OpenTelemetry export error:', error?.message || String(error))\n }\n }\n\n async flush() {\n if (this.buffer.length === 0) return\n\n const logsToSend = [...this.buffer]\n this.buffer = []\n await this.sendLogs(logsToSend)\n }\n\n startFlushTimer() {\n if (this.flushTimer) {\n clearInterval(this.flushTimer)\n }\n this.flushTimer = setInterval(() => {\n this.flush().catch(error => {\n console.warn('OpenTelemetry flush error:', error?.message || String(error))\n })\n }, this.flushInterval)\n }\n\n log(logObj) {\n try {\n const logRecord = this.createLogRecord(logObj)\n this.buffer.push(logRecord)\n\n // Відправляємо одразу, якщо буфер досяг максимального розміру\n if (this.buffer.length >= this.batchSize) {\n this.flush().catch(error => {\n console.warn('OpenTelemetry flush error:', error?.message || String(error))\n })\n }\n } catch (error) {\n // Тихо ігноруємо помилки\n console.warn('OpenTelemetry log processing error:', error?.message || String(error))\n }\n }\n\n destroy() {\n if (this.flushTimer) {\n clearInterval(this.flushTimer)\n this.flushTimer = null\n }\n // Відправляємо залишкові логи перед знищенням\n return this.flush()\n }\n}\n\n/**\n * Створює OpenTelemetry репортер для експорту логів до OpenTelemetry Collector\n *\n * @param {Object} options - Опції репортера\n * @param {String} options.endpoint - URL endpoint OpenTelemetry Collector (за замовчуванням: http://localhost:4318/v1/logs)\n * @param {String} options.serviceName - Ім'я сервісу (за замовчуванням: consola-service)\n * @param {String} options.serviceVersion - Версія сервісу (за замовчуванням: 1.0.0)\n * @param {Number} options.batchSize - Розмір батча для відправки (за замовчуванням: 10)\n * @param {Number} options.flushInterval - Інтервал автоматичної відправки в мс (за замовчуванням: 5000)\n * @param {Object} options.headers - Додаткові HTTP заголовки\n * @returns {OpenTelemetryReporter} Екземпляр OpenTelemetry репортера\n *\n * @example\n * import { createOpenTelemetryReporter } from '@nitra/consola'\n * const otelReporter = createOpenTelemetryReporter({\n * endpoint: 'http://localhost:4318/v1/logs',\n * serviceName: 'my-app',\n * serviceVersion: '1.0.0'\n * })\n * consola.addReporter(otelReporter)\n */\nexport const createOpenTelemetryReporter = (options = {}) => {\n return new OpenTelemetryReporter(options)\n}\n","/* global location, process, import */\nimport { consola } from 'consola'\nimport { OpenTelemetryReporter } from './otel-reporter.js'\n\nconst options = {}\n\n// Vite Debug mode\nif (typeof __CONSOLA_LEVEL_DEBUG__ !== 'undefined') {\n options.level = 4\n} else if (typeof location !== 'undefined' && location.protocol === 'http:') {\n options.level = 4\n} else if (typeof process !== 'undefined' && (process.env.DEV || process.env.CONSOLA_LEVEL_DEBUG)) {\n // Quasar Debug mode\n options.level = 4\n}\n\n// HTML Popup Reporter — дублювання повідомлень у вигляді спливаючих блоків (без canvas)\nconst POPUP_PADDING = 12\nconst POPUP_LABEL_PADDING = 8\nconst POPUP_LINE_HEIGHT = 20\nconst POPUP_BUTTON_SIZE = 18\nconst POPUP_BUTTON_PADDING = 6\nconst POPUP_MIN_HEIGHT = 36\nconst POPUP_MAX_MESSAGE_LENGTH = 500\n\nconst POPUP_STYLES = `\n#consola-popup-container { position: fixed; inset: 0; pointer-events: none; z-index: 999999; }\n.consola-popup { pointer-events: auto; position: fixed; left: 20px; max-width: calc(100vw - 40px); min-height: ${POPUP_MIN_HEIGHT}px;\n display: flex; align-items: stretch; background: #fff; box-shadow: 0 2px 8px rgba(0,0,0,0.1); border: 1px solid #e0e0e0;\n font: 13px -apple-system, BlinkMacSystemFont, \"Segoe UI\", Roboto, \"Helvetica Neue\", Arial, sans-serif; }\n.consola-popup-label { flex-shrink: 0; padding: 0 ${POPUP_LABEL_PADDING}px; display: flex; align-items: center; justify-content: center;\n color: #fff; font: 11px monospace; font-weight: bold; text-transform: lowercase; }\n.consola-popup-content { flex: 1; min-width: 0; padding: ${POPUP_PADDING}px; display: flex; flex-wrap: wrap; align-items: flex-start; gap: 4px 8px; }\n.consola-popup-text { flex: 1 1 auto; min-width: 0; color: #212121; line-height: ${POPUP_LINE_HEIGHT}px; white-space: pre-wrap; word-break: break-word; }\n.consola-popup-filelink { flex: 0 0 auto; color: #1976D2; font: 11px monospace; text-decoration: underline; cursor: pointer; }\n.consola-popup-filelink:hover { text-decoration: underline; }\n.consola-popup-close { flex-shrink: 0; width: ${POPUP_BUTTON_SIZE}px; height: ${POPUP_BUTTON_SIZE}px; margin: ${POPUP_BUTTON_PADDING}px; padding: 0; border: none; background: transparent;\n color: #999; font-size: 16px; line-height: 1; cursor: pointer; display: flex; align-items: center; justify-content: center; border-radius: 2px; }\n.consola-popup-close:hover { background: rgba(0,0,0,0.06); color: #666; }\n`\n\nclass HtmlPopupReporter {\n constructor() {\n this.container = null\n this.messages = []\n this.messageId = 0\n this.init()\n }\n\n init() {\n if (typeof document === 'undefined') return\n\n const doInit = () => {\n if (!document.body) {\n setTimeout(doInit, 10)\n return\n }\n\n let container = document.querySelector('#consola-popup-container')\n if (!container) {\n const style = document.createElement('style')\n style.textContent = POPUP_STYLES\n document.head.append(style)\n container = document.createElement('div')\n container.id = 'consola-popup-container'\n document.body.append(container)\n }\n\n this.container = container\n this.animate()\n }\n\n if (document.readyState === 'loading') {\n document.addEventListener('DOMContentLoaded', doInit)\n } else {\n doInit()\n }\n }\n\n getColorForType(type) {\n const colors = {\n trace: '#9E9E9E',\n debug: '#607D8B',\n info: '#2196F3',\n log: '#2196F3',\n warn: '#FFC107',\n error: '#D32F2F',\n fatal: '#7B1FA2',\n success: '#4CAF50',\n start: '#00BCD4',\n box: '#9E9E9E',\n ready: '#4CAF50',\n fail: '#D32F2F'\n }\n return colors[type] || colors.log || '#616161'\n }\n\n getTypeLabel(type) {\n return type || 'log'\n }\n\n getFileInfo() {\n try {\n const stack = new Error('Stack trace').stack\n if (!stack) return null\n\n const skipPatterns = ['HtmlPopupReporter', 'CanvasPopupReporter', 'browser.js', 'consola', 'createLogger']\n const lines = stack.split('\\n').slice(4, 15)\n\n for (const line of lines) {\n const trimmed = line.trim()\n const shouldSkip = skipPatterns.some(pattern => trimmed.includes(pattern))\n\n if (!shouldSkip) {\n const match =\n trimmed.match(/\\(?([^()]+):(\\d+):(\\d+)\\)?/) || trimmed.match(/at\\s+[^(]*\\(?([^:]+):(\\d+):(\\d+)\\)?/)\n\n if (match) {\n let file = match[1].trim()\n const lineNum = match[2]\n\n // Очищаємо ім'я файлу\n if (file.includes('://')) {\n file = file.slice(file.indexOf('://') + 3).slice(file.indexOf('/'))\n }\n if (file.includes('/src/')) {\n file = file.slice(file.indexOf('/src/') + 5)\n }\n const queryIndex = file.indexOf('?')\n if (queryIndex > 0) file = file.slice(0, queryIndex)\n const hashIndex = file.indexOf('#')\n if (hashIndex > 0) file = file.slice(0, hashIndex)\n if (file.startsWith('/')) file = file.slice(1)\n\n if (file && lineNum) {\n return { file, line: parseInt(lineNum, 10), column: parseInt(match[3], 10), fullPath: match[0] }\n }\n }\n }\n }\n } catch {\n // Ігноруємо помилки\n }\n return null\n }\n\n formatMessage(logObj) {\n const args = logObj.args || []\n if (args.length === 0) return ''\n\n const message = args\n .map(arg => {\n if (arg instanceof Error) {\n let msg = `${arg.name || 'Error'}: ${arg.message || ''}`\n if (arg.stack && arg.stack !== arg.message) {\n msg += '\\n' + arg.stack.split('\\n').slice(0, 3).join('\\n')\n }\n return msg\n }\n if (arg === null) return 'null'\n if (arg === undefined) return 'undefined'\n if (typeof arg === 'object') {\n try {\n const str = arg.toString?.()\n if (str && str !== '[object Object]') return str\n return JSON.stringify(arg, null, 2)\n } catch {\n return String(arg)\n }\n }\n return String(arg)\n })\n .join(' ')\n\n return message.length > POPUP_MAX_MESSAGE_LENGTH ? message.slice(0, POPUP_MAX_MESSAGE_LENGTH) + '...' : message\n }\n\n log(logObj) {\n if (!this.container) return\n\n const text = this.formatMessage(logObj)\n if (!text) return\n\n const type = logObj.type || 'log'\n const color = this.getColorForType(type)\n const typeLabel = this.getTypeLabel(type)\n const id = this.messageId++\n const fileInfo = this.getFileInfo()\n\n const popup = document.createElement('div')\n popup.className = 'consola-popup'\n popup.dataset.id = String(id)\n popup.style.bottom = `${window.innerHeight}px`\n popup.style.backgroundColor = '#fff'\n\n const label = document.createElement('span')\n label.className = 'consola-popup-label'\n label.style.backgroundColor = color\n label.textContent = typeLabel\n\n const content = document.createElement('div')\n content.className = 'consola-popup-content'\n\n const textEl = document.createElement('div')\n textEl.className = 'consola-popup-text'\n textEl.textContent = text\n\n content.append(textEl)\n\n if (fileInfo) {\n const fileLink = document.createElement('span')\n fileLink.className = 'consola-popup-filelink'\n fileLink.textContent = `${fileInfo.file}:${fileInfo.line}`\n fileLink.role = 'button'\n fileLink.tabIndex = 0\n fileLink.addEventListener('click', e => {\n e.preventDefault()\n console.log(`%c${fileInfo.file}:${fileInfo.line}`, 'color: #1976D2; text-decoration: underline;')\n })\n content.append(fileLink)\n }\n\n const closeBtn = document.createElement('button')\n closeBtn.type = 'button'\n closeBtn.className = 'consola-popup-close'\n closeBtn.setAttribute('aria-label', 'Close')\n closeBtn.textContent = '×'\n closeBtn.addEventListener('click', () => {\n const idx = this.messages.findIndex(m => m.id === id)\n if (idx !== -1) {\n const entry = this.messages[idx]\n if (entry.el && entry.el.parentNode) entry.el.remove()\n this.messages.splice(idx, 1)\n }\n })\n\n popup.append(label, content, closeBtn)\n this.container.append(popup)\n\n const height = popup.offsetHeight\n this.messages.push({\n id,\n el: popup,\n targetBottom: 20,\n currentBottom: window.innerHeight,\n height\n })\n }\n\n animate() {\n if (!this.container) return\n\n let offset = 20\n for (let i = this.messages.length - 1; i >= 0; i--) {\n const msg = this.messages[i]\n if (msg.el && msg.el.parentNode) {\n const h = msg.el.offsetHeight\n msg.height = h\n msg.targetBottom = offset\n msg.currentBottom += (msg.targetBottom - msg.currentBottom) * 0.1\n msg.el.style.bottom = `${Math.round(msg.currentBottom)}px`\n offset += h + 10\n }\n }\n\n requestAnimationFrame(() => this.animate())\n }\n}\n\n// Отримання змінних середовища з Vite (import.meta.env)\nconst getViteEnvVar = name => {\n try {\n // eslint-disable-next-line no-undef\n const viteName = `VITE_${name}`\n // eslint-disable-next-line no-undef\n return import.meta?.env?.[viteName]\n } catch {\n return null\n }\n}\n\n// Перевіряємо змінну середовища VITE_CONSOLA_POPUP_DEBUG\nlet isPopupDebugEnabled = false\n\n// Vite замінює import.meta.env.VITE_* під час збірки\n// Використовуємо прямий доступ до import.meta.env, оскільки Vite обробляє це під час збірки\nif (typeof document !== 'undefined') {\n try {\n // Vite замінює цей код під час збірки\n // eslint-disable-next-line no-undef\n const envValue = import.meta.env?.VITE_CONSOLA_POPUP_DEBUG\n if (envValue !== undefined && envValue !== null && envValue !== '') {\n // Vite завжди повертає рядки для змінних середовища\n isPopupDebugEnabled = String(envValue).toLowerCase() === 'true' || envValue === true\n }\n } catch {\n // Ігноруємо помилки, якщо import.meta недоступний\n }\n}\n\nif (!isPopupDebugEnabled && typeof process !== 'undefined' && process.env) {\n const processValue = process.env.VITE_CONSOLA_POPUP_DEBUG\n if (processValue) {\n isPopupDebugEnabled = String(processValue).toLowerCase() === 'true'\n }\n}\n\n// Створюємо один екземпляр репортера, якщо потрібно\n// НЕ встановлюємо options.reporters, щоб стандартний BrowserReporter працював\n// Додаємо наш репортер після створення екземпляра через addReporter\nlet htmlPopupReporter = null\nif (isPopupDebugEnabled && typeof document !== 'undefined') {\n htmlPopupReporter = new HtmlPopupReporter()\n}\n\nconst defaultConsola = consola.create(options)\n\n// Додаємо HTML репортер до default екземпляра, якщо потрібно (не замінює стандартний)\nif (isPopupDebugEnabled && typeof document !== 'undefined' && htmlPopupReporter) {\n defaultConsola.addReporter(htmlPopupReporter)\n}\n\n// Перевіряємо змінні середовища Vite для OpenTelemetry експорту\nlet openTelemetryReporter = null\nconst otelEndpoint = getViteEnvVar('OTEL_EXPORTER_OTLP_LOGS_ENDPOINT')\n\nif (otelEndpoint) {\n openTelemetryReporter = new OpenTelemetryReporter()\n defaultConsola.addReporter(openTelemetryReporter)\n}\n\nexport default defaultConsola\n\n// Експортуємо consola з нашими опціями, а не базовий\nexport { defaultConsola as consola }\n\n// Експортуємо OpenTelemetryReporter та createOpenTelemetryReporter для прямого використання\nexport { createOpenTelemetryReporter, OpenTelemetryReporter } from './otel-reporter.js'\n\n/**\n * pass import.meta.url\n * example: const consola = createLogger(import.meta.url)\n *\n * @param {String} _url - The import.meta.url or file URL to use for logger creation\n * @returns {Consola} A consola logger instance\n */\nexport const createLogger = _url => {\n return defaultConsola\n}\n","/// <reference path=\"./web-serial.d.ts\" />\nimport { WebPlugin } from '@capacitor/core'\nimport { consola } from '@nitra/consola'\n\nexport class ZebraWeb extends WebPlugin {\n async print({ zpl }) {\n try {\n consola.debug('Запит на підключення до принтера...', 'info')\n\n // Запитуємо порт (Web Serial API)\n const serial = navigator.serial\n if (!serial) throw new Error('Web Serial API не підтримується в цьому браузері')\n const port = await serial.requestPort()\n consola.debug(\"Порт вибрано, відкриваємо з'єднання...\", 'info')\n\n // Відкриваємо з'єднання з налаштуваннями для Zebra принтера\n await port.open({\n baudRate: 9600, // Можна спробувати 115200 якщо не працює\n dataBits: 8,\n parity: 'none',\n stopBits: 1,\n flowControl: 'none'\n })\n\n consola.debug(\"З'єднання відкрито!\", 'success')\n\n // Отримуємо writer для відправки даних\n const writer = port.writable.getWriter()\n\n // Оновлюємо стан\n consola.debug('Принтер готовий до друку', 'success')\n\n if (!port || !writer) {\n consola.debug('Помилка: Принтер не підключено', 'error')\n return\n }\n\n consola.debug(`Відправка ZPL команди: ${zpl.slice(0, 50)}...`, 'info')\n\n // Конвертуємо текст в Uint8Array\n const encoder = new TextEncoder()\n const data = encoder.encode(zpl)\n\n // Відправляємо дані\n await writer.write(data)\n\n // Чекаємо трохи для обробки принтером\n // await new Promise(resolve => setTimeout(resolve, 500))\n\n consola.debug(`✓ Команда відправлена (${data.length} байт)`, 'success')\n consola.debug('print from web capacitor plugin', zpl)\n return {\n success: true\n }\n } catch (error) {\n const message = error instanceof Error ? error.message : String(error)\n consola.debug(`Помилка підключення: ${message}`, 'error')\n\n // Спробуємо перепідключитися\n // await disconnect()\n throw error\n }\n }\n\n getPairedDevices() {\n return {\n devices: [\n { address: '00:00:00:00:00:00', name: 'Virtual Printer' },\n { address: '11:11:11:11:11:11', name: 'Virtual Headphone' }\n ]\n }\n }\n}\n","import { registerPlugin } from '@capacitor/core'\nimport { Preferences } from '@capacitor/preferences'\nimport { normalizePrintArg } from './utils/validate.js'\n\nconst ZebraPlugin = registerPlugin('Zebra', {\n web: () => import('./web.js').then(m => new m.ZebraWeb())\n})\n\nconst Zebra = {\n /**\n * Приймає лише zpl як рядок; address для нативних платформ береться з Preferences (після setPrinterAddress або вибору пристрою).\n * @param {string} zpl - ZPL-рядок для друку\n * @returns {Promise<{success?: boolean, devices?: Array<{address: string, name: string}>, message?: string}>} Результат друку або список пристроїв\n */\n async print(zpl) {\n const zplNormalized = normalizePrintArg(zpl)\n\n const { value: address } = await Preferences.get({ key: 'printer_address' })\n const addressTrimmed = typeof address === 'string' ? address.trim() : ''\n if (!addressTrimmed) {\n const { devices } = await ZebraPlugin.getPairedDevices()\n if (devices?.length > 0) {\n return { success: false, message: 'device list', devices }\n }\n return { success: false, message: 'No paired devices found' }\n }\n return ZebraPlugin.print({ zpl: zplNormalized, address: addressTrimmed })\n },\n getPairedDevices() {\n return ZebraPlugin.getPairedDevices()\n }\n}\n\nexport { Zebra }\n"],"x_google_ignoreList":[0,1,3,4,5,6],"mappings":"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"}
1
+ {"version":3,"file":"plugin.js","names":["createConsola","getViteEnvVar"],"sources":["../../node_modules/.bun/@capacitor+preferences@8.0.0+2476a4e6bb24aa03/node_modules/@capacitor/preferences/dist/esm/web.js","../../node_modules/.bun/@capacitor+preferences@8.0.0+2476a4e6bb24aa03/node_modules/@capacitor/preferences/dist/esm/index.js","../src/utils/validate.js","../../node_modules/.bun/consola@3.4.2/node_modules/consola/dist/core.mjs","../../node_modules/.bun/consola@3.4.2/node_modules/consola/dist/browser.mjs","../../node_modules/.bun/@nitra+consola@2.4.1/node_modules/@nitra/consola/src/otel-reporter.js","../../node_modules/.bun/@nitra+consola@2.4.1/node_modules/@nitra/consola/src/browser.js","../src/web.js","../src/index.js"],"sourcesContent":["import { WebPlugin } from '@capacitor/core';\nexport class PreferencesWeb extends WebPlugin {\n constructor() {\n super(...arguments);\n this.group = 'CapacitorStorage';\n }\n async configure({ group }) {\n if (typeof group === 'string') {\n this.group = group;\n }\n }\n async get(options) {\n const value = this.impl.getItem(this.applyPrefix(options.key));\n return { value };\n }\n async set(options) {\n this.impl.setItem(this.applyPrefix(options.key), options.value);\n }\n async remove(options) {\n this.impl.removeItem(this.applyPrefix(options.key));\n }\n async keys() {\n const keys = this.rawKeys().map((k) => k.substring(this.prefix.length));\n return { keys };\n }\n async clear() {\n for (const key of this.rawKeys()) {\n this.impl.removeItem(key);\n }\n }\n async migrate() {\n var _a;\n const migrated = [];\n const existing = [];\n const oldprefix = '_cap_';\n const keys = Object.keys(this.impl).filter((k) => k.indexOf(oldprefix) === 0);\n for (const oldkey of keys) {\n const key = oldkey.substring(oldprefix.length);\n const value = (_a = this.impl.getItem(oldkey)) !== null && _a !== void 0 ? _a : '';\n const { value: currentValue } = await this.get({ key });\n if (typeof currentValue === 'string') {\n existing.push(key);\n }\n else {\n await this.set({ key, value });\n migrated.push(key);\n }\n }\n return { migrated, existing };\n }\n async removeOld() {\n const oldprefix = '_cap_';\n const keys = Object.keys(this.impl).filter((k) => k.indexOf(oldprefix) === 0);\n for (const oldkey of keys) {\n this.impl.removeItem(oldkey);\n }\n }\n get impl() {\n return window.localStorage;\n }\n get prefix() {\n return this.group === 'NativeStorage' ? '' : `${this.group}.`;\n }\n rawKeys() {\n return Object.keys(this.impl).filter((k) => k.indexOf(this.prefix) === 0);\n }\n applyPrefix(key) {\n return this.prefix + key;\n }\n}\n//# sourceMappingURL=web.js.map","import { registerPlugin } from '@capacitor/core';\nconst Preferences = registerPlugin('Preferences', {\n web: () => import('./web').then((m) => new m.PreferencesWeb()),\n});\nexport * from './definitions';\nexport { Preferences };\n//# sourceMappingURL=index.js.map","/**\n * Нормалізує аргумент ZPL для друку: перевіряє рядок і повертає обрізану команду.\n * @param {string} zpl - рядок ZPL-команд\n * @returns {string} обрізаний рядок ZPL\n * @throws {TypeError} якщо zpl не рядок\n * @throws {Error} якщо команда порожня\n */\nexport function normalizePrintArg(zpl) {\n if (typeof zpl !== 'string') {\n throw new TypeError('Expected \"zpl\" to be a string')\n }\n\n const command = zpl.trim()\n if (!command) {\n throw new Error('ZPL команда порожня')\n }\n\n return command\n}\n","const LogLevels = {\n silent: Number.NEGATIVE_INFINITY,\n fatal: 0,\n error: 0,\n warn: 1,\n log: 2,\n info: 3,\n success: 3,\n fail: 3,\n ready: 3,\n start: 3,\n box: 3,\n debug: 4,\n trace: 5,\n verbose: Number.POSITIVE_INFINITY\n};\nconst LogTypes = {\n // Silent\n silent: {\n level: -1\n },\n // Level 0\n fatal: {\n level: LogLevels.fatal\n },\n error: {\n level: LogLevels.error\n },\n // Level 1\n warn: {\n level: LogLevels.warn\n },\n // Level 2\n log: {\n level: LogLevels.log\n },\n // Level 3\n info: {\n level: LogLevels.info\n },\n success: {\n level: LogLevels.success\n },\n fail: {\n level: LogLevels.fail\n },\n ready: {\n level: LogLevels.info\n },\n start: {\n level: LogLevels.info\n },\n box: {\n level: LogLevels.info\n },\n // Level 4\n debug: {\n level: LogLevels.debug\n },\n // Level 5\n trace: {\n level: LogLevels.trace\n },\n // Verbose\n verbose: {\n level: LogLevels.verbose\n }\n};\n\nfunction isPlainObject$1(value) {\n if (value === null || typeof value !== \"object\") {\n return false;\n }\n const prototype = Object.getPrototypeOf(value);\n if (prototype !== null && prototype !== Object.prototype && Object.getPrototypeOf(prototype) !== null) {\n return false;\n }\n if (Symbol.iterator in value) {\n return false;\n }\n if (Symbol.toStringTag in value) {\n return Object.prototype.toString.call(value) === \"[object Module]\";\n }\n return true;\n}\n\nfunction _defu(baseObject, defaults, namespace = \".\", merger) {\n if (!isPlainObject$1(defaults)) {\n return _defu(baseObject, {}, namespace, merger);\n }\n const object = Object.assign({}, defaults);\n for (const key in baseObject) {\n if (key === \"__proto__\" || key === \"constructor\") {\n continue;\n }\n const value = baseObject[key];\n if (value === null || value === void 0) {\n continue;\n }\n if (merger && merger(object, key, value, namespace)) {\n continue;\n }\n if (Array.isArray(value) && Array.isArray(object[key])) {\n object[key] = [...value, ...object[key]];\n } else if (isPlainObject$1(value) && isPlainObject$1(object[key])) {\n object[key] = _defu(\n value,\n object[key],\n (namespace ? `${namespace}.` : \"\") + key.toString(),\n merger\n );\n } else {\n object[key] = value;\n }\n }\n return object;\n}\nfunction createDefu(merger) {\n return (...arguments_) => (\n // eslint-disable-next-line unicorn/no-array-reduce\n arguments_.reduce((p, c) => _defu(p, c, \"\", merger), {})\n );\n}\nconst defu = createDefu();\n\nfunction isPlainObject(obj) {\n return Object.prototype.toString.call(obj) === \"[object Object]\";\n}\nfunction isLogObj(arg) {\n if (!isPlainObject(arg)) {\n return false;\n }\n if (!arg.message && !arg.args) {\n return false;\n }\n if (arg.stack) {\n return false;\n }\n return true;\n}\n\nlet paused = false;\nconst queue = [];\nclass Consola {\n options;\n _lastLog;\n _mockFn;\n /**\n * Creates an instance of Consola with specified options or defaults.\n *\n * @param {Partial<ConsolaOptions>} [options={}] - Configuration options for the Consola instance.\n */\n constructor(options = {}) {\n const types = options.types || LogTypes;\n this.options = defu(\n {\n ...options,\n defaults: { ...options.defaults },\n level: _normalizeLogLevel(options.level, types),\n reporters: [...options.reporters || []]\n },\n {\n types: LogTypes,\n throttle: 1e3,\n throttleMin: 5,\n formatOptions: {\n date: true,\n colors: false,\n compact: true\n }\n }\n );\n for (const type in types) {\n const defaults = {\n type,\n ...this.options.defaults,\n ...types[type]\n };\n this[type] = this._wrapLogFn(defaults);\n this[type].raw = this._wrapLogFn(\n defaults,\n true\n );\n }\n if (this.options.mockFn) {\n this.mockTypes();\n }\n this._lastLog = {};\n }\n /**\n * Gets the current log level of the Consola instance.\n *\n * @returns {number} The current log level.\n */\n get level() {\n return this.options.level;\n }\n /**\n * Sets the minimum log level that will be output by the instance.\n *\n * @param {number} level - The new log level to set.\n */\n set level(level) {\n this.options.level = _normalizeLogLevel(\n level,\n this.options.types,\n this.options.level\n );\n }\n /**\n * Displays a prompt to the user and returns the response.\n * Throw an error if `prompt` is not supported by the current configuration.\n *\n * @template T\n * @param {string} message - The message to display in the prompt.\n * @param {T} [opts] - Optional options for the prompt. See {@link PromptOptions}.\n * @returns {promise<T>} A promise that infer with the prompt options. See {@link PromptOptions}.\n */\n prompt(message, opts) {\n if (!this.options.prompt) {\n throw new Error(\"prompt is not supported!\");\n }\n return this.options.prompt(message, opts);\n }\n /**\n * Creates a new instance of Consola, inheriting options from the current instance, with possible overrides.\n *\n * @param {Partial<ConsolaOptions>} options - Optional overrides for the new instance. See {@link ConsolaOptions}.\n * @returns {ConsolaInstance} A new Consola instance. See {@link ConsolaInstance}.\n */\n create(options) {\n const instance = new Consola({\n ...this.options,\n ...options\n });\n if (this._mockFn) {\n instance.mockTypes(this._mockFn);\n }\n return instance;\n }\n /**\n * Creates a new Consola instance with the specified default log object properties.\n *\n * @param {InputLogObject} defaults - Default properties to include in any log from the new instance. See {@link InputLogObject}.\n * @returns {ConsolaInstance} A new Consola instance. See {@link ConsolaInstance}.\n */\n withDefaults(defaults) {\n return this.create({\n ...this.options,\n defaults: {\n ...this.options.defaults,\n ...defaults\n }\n });\n }\n /**\n * Creates a new Consola instance with a specified tag, which will be included in every log.\n *\n * @param {string} tag - The tag to include in each log of the new instance.\n * @returns {ConsolaInstance} A new Consola instance. See {@link ConsolaInstance}.\n */\n withTag(tag) {\n return this.withDefaults({\n tag: this.options.defaults.tag ? this.options.defaults.tag + \":\" + tag : tag\n });\n }\n /**\n * Adds a custom reporter to the Consola instance.\n * Reporters will be called for each log message, depending on their implementation and log level.\n *\n * @param {ConsolaReporter} reporter - The reporter to add. See {@link ConsolaReporter}.\n * @returns {Consola} The current Consola instance.\n */\n addReporter(reporter) {\n this.options.reporters.push(reporter);\n return this;\n }\n /**\n * Removes a custom reporter from the Consola instance.\n * If no reporter is specified, all reporters will be removed.\n *\n * @param {ConsolaReporter} reporter - The reporter to remove. See {@link ConsolaReporter}.\n * @returns {Consola} The current Consola instance.\n */\n removeReporter(reporter) {\n if (reporter) {\n const i = this.options.reporters.indexOf(reporter);\n if (i !== -1) {\n return this.options.reporters.splice(i, 1);\n }\n } else {\n this.options.reporters.splice(0);\n }\n return this;\n }\n /**\n * Replaces all reporters of the Consola instance with the specified array of reporters.\n *\n * @param {ConsolaReporter[]} reporters - The new reporters to set. See {@link ConsolaReporter}.\n * @returns {Consola} The current Consola instance.\n */\n setReporters(reporters) {\n this.options.reporters = Array.isArray(reporters) ? reporters : [reporters];\n return this;\n }\n wrapAll() {\n this.wrapConsole();\n this.wrapStd();\n }\n restoreAll() {\n this.restoreConsole();\n this.restoreStd();\n }\n /**\n * Overrides console methods with Consola logging methods for consistent logging.\n */\n wrapConsole() {\n for (const type in this.options.types) {\n if (!console[\"__\" + type]) {\n console[\"__\" + type] = console[type];\n }\n console[type] = this[type].raw;\n }\n }\n /**\n * Restores the original console methods, removing Consola overrides.\n */\n restoreConsole() {\n for (const type in this.options.types) {\n if (console[\"__\" + type]) {\n console[type] = console[\"__\" + type];\n delete console[\"__\" + type];\n }\n }\n }\n /**\n * Overrides standard output and error streams to redirect them through Consola.\n */\n wrapStd() {\n this._wrapStream(this.options.stdout, \"log\");\n this._wrapStream(this.options.stderr, \"log\");\n }\n _wrapStream(stream, type) {\n if (!stream) {\n return;\n }\n if (!stream.__write) {\n stream.__write = stream.write;\n }\n stream.write = (data) => {\n this[type].raw(String(data).trim());\n };\n }\n /**\n * Restores the original standard output and error streams, removing the Consola redirection.\n */\n restoreStd() {\n this._restoreStream(this.options.stdout);\n this._restoreStream(this.options.stderr);\n }\n _restoreStream(stream) {\n if (!stream) {\n return;\n }\n if (stream.__write) {\n stream.write = stream.__write;\n delete stream.__write;\n }\n }\n /**\n * Pauses logging, queues incoming logs until resumed.\n */\n pauseLogs() {\n paused = true;\n }\n /**\n * Resumes logging, processing any queued logs.\n */\n resumeLogs() {\n paused = false;\n const _queue = queue.splice(0);\n for (const item of _queue) {\n item[0]._logFn(item[1], item[2]);\n }\n }\n /**\n * Replaces logging methods with mocks if a mock function is provided.\n *\n * @param {ConsolaOptions[\"mockFn\"]} mockFn - The function to use for mocking logging methods. See {@link ConsolaOptions[\"mockFn\"]}.\n */\n mockTypes(mockFn) {\n const _mockFn = mockFn || this.options.mockFn;\n this._mockFn = _mockFn;\n if (typeof _mockFn !== \"function\") {\n return;\n }\n for (const type in this.options.types) {\n this[type] = _mockFn(type, this.options.types[type]) || this[type];\n this[type].raw = this[type];\n }\n }\n _wrapLogFn(defaults, isRaw) {\n return (...args) => {\n if (paused) {\n queue.push([this, defaults, args, isRaw]);\n return;\n }\n return this._logFn(defaults, args, isRaw);\n };\n }\n _logFn(defaults, args, isRaw) {\n if ((defaults.level || 0) > this.level) {\n return false;\n }\n const logObj = {\n date: /* @__PURE__ */ new Date(),\n args: [],\n ...defaults,\n level: _normalizeLogLevel(defaults.level, this.options.types)\n };\n if (!isRaw && args.length === 1 && isLogObj(args[0])) {\n Object.assign(logObj, args[0]);\n } else {\n logObj.args = [...args];\n }\n if (logObj.message) {\n logObj.args.unshift(logObj.message);\n delete logObj.message;\n }\n if (logObj.additional) {\n if (!Array.isArray(logObj.additional)) {\n logObj.additional = logObj.additional.split(\"\\n\");\n }\n logObj.args.push(\"\\n\" + logObj.additional.join(\"\\n\"));\n delete logObj.additional;\n }\n logObj.type = typeof logObj.type === \"string\" ? logObj.type.toLowerCase() : \"log\";\n logObj.tag = typeof logObj.tag === \"string\" ? logObj.tag : \"\";\n const resolveLog = (newLog = false) => {\n const repeated = (this._lastLog.count || 0) - this.options.throttleMin;\n if (this._lastLog.object && repeated > 0) {\n const args2 = [...this._lastLog.object.args];\n if (repeated > 1) {\n args2.push(`(repeated ${repeated} times)`);\n }\n this._log({ ...this._lastLog.object, args: args2 });\n this._lastLog.count = 1;\n }\n if (newLog) {\n this._lastLog.object = logObj;\n this._log(logObj);\n }\n };\n clearTimeout(this._lastLog.timeout);\n const diffTime = this._lastLog.time && logObj.date ? logObj.date.getTime() - this._lastLog.time.getTime() : 0;\n this._lastLog.time = logObj.date;\n if (diffTime < this.options.throttle) {\n try {\n const serializedLog = JSON.stringify([\n logObj.type,\n logObj.tag,\n logObj.args\n ]);\n const isSameLog = this._lastLog.serialized === serializedLog;\n this._lastLog.serialized = serializedLog;\n if (isSameLog) {\n this._lastLog.count = (this._lastLog.count || 0) + 1;\n if (this._lastLog.count > this.options.throttleMin) {\n this._lastLog.timeout = setTimeout(\n resolveLog,\n this.options.throttle\n );\n return;\n }\n }\n } catch {\n }\n }\n resolveLog(true);\n }\n _log(logObj) {\n for (const reporter of this.options.reporters) {\n reporter.log(logObj, {\n options: this.options\n });\n }\n }\n}\nfunction _normalizeLogLevel(input, types = {}, defaultLevel = 3) {\n if (input === void 0) {\n return defaultLevel;\n }\n if (typeof input === \"number\") {\n return input;\n }\n if (types[input] && types[input].level !== void 0) {\n return types[input].level;\n }\n return defaultLevel;\n}\nConsola.prototype.add = Consola.prototype.addReporter;\nConsola.prototype.remove = Consola.prototype.removeReporter;\nConsola.prototype.clear = Consola.prototype.removeReporter;\nConsola.prototype.withScope = Consola.prototype.withTag;\nConsola.prototype.mock = Consola.prototype.mockTypes;\nConsola.prototype.pause = Consola.prototype.pauseLogs;\nConsola.prototype.resume = Consola.prototype.resumeLogs;\nfunction createConsola(options = {}) {\n return new Consola(options);\n}\n\nexport { Consola, LogLevels, LogTypes, createConsola };\n","import { createConsola as createConsola$1 } from './core.mjs';\nexport { Consola, LogLevels, LogTypes } from './core.mjs';\n\nclass BrowserReporter {\n options;\n defaultColor;\n levelColorMap;\n typeColorMap;\n constructor(options) {\n this.options = { ...options };\n this.defaultColor = \"#7f8c8d\";\n this.levelColorMap = {\n 0: \"#c0392b\",\n // Red\n 1: \"#f39c12\",\n // Yellow\n 3: \"#00BCD4\"\n // Cyan\n };\n this.typeColorMap = {\n success: \"#2ecc71\"\n // Green\n };\n }\n _getLogFn(level) {\n if (level < 1) {\n return console.__error || console.error;\n }\n if (level === 1) {\n return console.__warn || console.warn;\n }\n return console.__log || console.log;\n }\n log(logObj) {\n const consoleLogFn = this._getLogFn(logObj.level);\n const type = logObj.type === \"log\" ? \"\" : logObj.type;\n const tag = logObj.tag || \"\";\n const color = this.typeColorMap[logObj.type] || this.levelColorMap[logObj.level] || this.defaultColor;\n const style = `\n background: ${color};\n border-radius: 0.5em;\n color: white;\n font-weight: bold;\n padding: 2px 0.5em;\n `;\n const badge = `%c${[tag, type].filter(Boolean).join(\":\")}`;\n if (typeof logObj.args[0] === \"string\") {\n consoleLogFn(\n `${badge}%c ${logObj.args[0]}`,\n style,\n // Empty string as style resets to default console style\n \"\",\n ...logObj.args.slice(1)\n );\n } else {\n consoleLogFn(badge, style, ...logObj.args);\n }\n }\n}\n\nfunction createConsola(options = {}) {\n const consola2 = createConsola$1({\n reporters: options.reporters || [new BrowserReporter({})],\n prompt(message, options2 = {}) {\n if (options2.type === \"confirm\") {\n return Promise.resolve(confirm(message));\n }\n return Promise.resolve(prompt(message));\n },\n ...options\n });\n return consola2;\n}\nconst consola = createConsola();\n\nexport { consola, createConsola, consola as default };\n","// Отримання змінних середовища з Vite (import.meta.env)\nconst getViteEnvVar = name => {\n try {\n // eslint-disable-next-line no-undef\n const viteName = `VITE_${name}`\n // eslint-disable-next-line no-undef\n return import.meta?.env?.[viteName]\n } catch {\n return null\n }\n}\n\n// OpenTelemetry Reporter для експорту логів до OpenTelemetry Collector через HTTP\nexport class OpenTelemetryReporter {\n constructor(options = {}) {\n this.endpoint =\n options.endpoint || getViteEnvVar('OTEL_EXPORTER_OTLP_LOGS_ENDPOINT') || 'http://localhost:4318/v1/logs'\n this.serviceName = options.serviceName || getViteEnvVar('OTEL_SERVICE_NAME') || 'consola-service'\n this.serviceVersion = options.serviceVersion || getViteEnvVar('OTEL_SERVICE_VERSION') || '1.0.0'\n this.batchSize = options.batchSize || parseInt(getViteEnvVar('OTEL_BATCH_SIZE') || '10', 10)\n this.flushInterval = options.flushInterval || parseInt(getViteEnvVar('OTEL_FLUSH_INTERVAL') || '5000', 10)\n this.buffer = []\n this.flushTimer = null\n this.headers = {\n 'Content-Type': 'application/json',\n ...options.headers\n }\n\n // Запускаємо таймер для періодичної відправки\n if (this.flushInterval > 0) {\n this.startFlushTimer()\n }\n }\n\n formatMessage(logObj) {\n const args = logObj.args || []\n if (args.length === 0) return ''\n\n return args\n .map(arg => {\n if (arg instanceof Error) {\n let msg = `${arg.name || 'Error'}: ${arg.message || ''}`\n if (arg.stack && arg.stack !== arg.message) {\n msg += '\\n' + arg.stack\n }\n return msg\n }\n if (arg === null) return 'null'\n if (arg === undefined) return 'undefined'\n if (typeof arg === 'object') {\n try {\n return JSON.stringify(arg, null, 2)\n } catch {\n return String(arg)\n }\n }\n return String(arg)\n })\n .join(' ')\n }\n\n getSeverityNumber(type) {\n // Маппінг типів consola до OpenTelemetry severity numbers\n const severityMap = {\n trace: 1, // TRACE\n debug: 5, // DEBUG\n info: 9, // INFO\n log: 9, // INFO\n warn: 13, // WARN\n error: 17, // ERROR\n fatal: 21, // FATAL\n success: 9, // INFO\n start: 9, // INFO\n ready: 9, // INFO\n fail: 17 // ERROR\n }\n return severityMap[type] || 9\n }\n\n getSeverityText(type) {\n const textMap = {\n trace: 'TRACE',\n debug: 'DEBUG',\n info: 'INFO',\n log: 'INFO',\n warn: 'WARN',\n error: 'ERROR',\n fatal: 'FATAL',\n success: 'INFO',\n start: 'INFO',\n ready: 'INFO',\n fail: 'ERROR'\n }\n return textMap[type] || 'INFO'\n }\n\n getFileInfo() {\n try {\n const stack = new Error('Stack trace').stack\n if (!stack) return null\n\n const skipPatterns = ['OpenTelemetryReporter', 'otel-reporter.js', 'browser.js', 'consola', 'createLogger']\n const lines = stack.split('\\n').slice(4, 15)\n\n for (const line of lines) {\n const trimmed = line.trim()\n const shouldSkip = skipPatterns.some(pattern => trimmed.includes(pattern))\n\n if (!shouldSkip) {\n const match =\n trimmed.match(/\\(?([^()]+):(\\d+):(\\d+)\\)?/) || trimmed.match(/at\\s+[^(]*\\(?([^:]+):(\\d+):(\\d+)\\)?/)\n\n if (match) {\n let file = match[1].trim()\n const lineNum = match[2]\n\n if (file.includes('://')) {\n file = file.slice(file.indexOf('://') + 3).slice(file.indexOf('/'))\n }\n if (file.includes('/src/')) {\n file = file.slice(file.indexOf('/src/') + 5)\n }\n const queryIndex = file.indexOf('?')\n if (queryIndex > 0) file = file.slice(0, queryIndex)\n const hashIndex = file.indexOf('#')\n if (hashIndex > 0) file = file.slice(0, hashIndex)\n if (file.startsWith('/')) file = file.slice(1)\n\n if (file && lineNum) {\n return { file, line: parseInt(lineNum, 10), column: parseInt(match[3], 10) }\n }\n }\n }\n }\n } catch {\n // Ігноруємо помилки\n }\n return null\n }\n\n createLogRecord(logObj) {\n const type = logObj.type || 'log'\n const message = this.formatMessage(logObj)\n const fileInfo = this.getFileInfo()\n const timestamp = Date.now() * 1000000 // наносекунди\n\n const logRecord = {\n timeUnixNano: timestamp.toString(),\n severityNumber: this.getSeverityNumber(type),\n severityText: this.getSeverityText(type),\n body: {\n stringValue: message\n },\n attributes: [\n {\n key: 'log.type',\n value: { stringValue: type }\n }\n ]\n }\n\n // Додаємо інформацію про файл, якщо доступна\n if (fileInfo) {\n logRecord.attributes.push(\n {\n key: 'code.filepath',\n value: { stringValue: fileInfo.file }\n },\n {\n key: 'code.lineno',\n value: { intValue: fileInfo.line }\n }\n )\n if (fileInfo.column) {\n logRecord.attributes.push({\n key: 'code.colno',\n value: { intValue: fileInfo.column }\n })\n }\n }\n\n // Додаємо додаткові атрибути з logObj, якщо вони є\n if (logObj.extra && typeof logObj.extra === 'object') {\n for (const [key, value] of Object.entries(logObj.extra)) {\n logRecord.attributes.push({\n key: `extra.${key}`,\n value: { stringValue: String(value) }\n })\n }\n }\n\n return logRecord\n }\n\n async sendLogs(logs) {\n if (logs.length === 0) return\n\n const resourceLogs = {\n resource: {\n attributes: [\n {\n key: 'service.name',\n value: { stringValue: this.serviceName }\n },\n {\n key: 'service.version',\n value: { stringValue: this.serviceVersion }\n }\n ]\n },\n scopeLogs: [\n {\n scope: {\n name: 'consola',\n version: '1.0.0'\n },\n logRecords: logs\n }\n ]\n }\n\n const payload = {\n resourceLogs: [resourceLogs]\n }\n\n try {\n const response = await fetch(this.endpoint, {\n method: 'POST',\n mode: 'cors',\n headers: this.headers,\n body: JSON.stringify(payload)\n })\n\n if (!response.ok) {\n console.warn(`OpenTelemetry export failed: ${response.status} ${response.statusText}`)\n }\n } catch (error) {\n // Тихо ігноруємо помилки, щоб не порушити роботу додатку\n console.warn('OpenTelemetry export error:', error?.message || String(error))\n }\n }\n\n async flush() {\n if (this.buffer.length === 0) return\n\n const logsToSend = [...this.buffer]\n this.buffer = []\n await this.sendLogs(logsToSend)\n }\n\n startFlushTimer() {\n if (this.flushTimer) {\n clearInterval(this.flushTimer)\n }\n this.flushTimer = setInterval(() => {\n this.flush().catch(error => {\n console.warn('OpenTelemetry flush error:', error?.message || String(error))\n })\n }, this.flushInterval)\n }\n\n log(logObj) {\n try {\n const logRecord = this.createLogRecord(logObj)\n this.buffer.push(logRecord)\n\n // Відправляємо одразу, якщо буфер досяг максимального розміру\n if (this.buffer.length >= this.batchSize) {\n this.flush().catch(error => {\n console.warn('OpenTelemetry flush error:', error?.message || String(error))\n })\n }\n } catch (error) {\n // Тихо ігноруємо помилки\n console.warn('OpenTelemetry log processing error:', error?.message || String(error))\n }\n }\n\n destroy() {\n if (this.flushTimer) {\n clearInterval(this.flushTimer)\n this.flushTimer = null\n }\n // Відправляємо залишкові логи перед знищенням\n return this.flush()\n }\n}\n\n/**\n * Створює OpenTelemetry репортер для експорту логів до OpenTelemetry Collector\n *\n * @param {Object} options - Опції репортера\n * @param {String} options.endpoint - URL endpoint OpenTelemetry Collector (за замовчуванням: http://localhost:4318/v1/logs)\n * @param {String} options.serviceName - Ім'я сервісу (за замовчуванням: consola-service)\n * @param {String} options.serviceVersion - Версія сервісу (за замовчуванням: 1.0.0)\n * @param {Number} options.batchSize - Розмір батча для відправки (за замовчуванням: 10)\n * @param {Number} options.flushInterval - Інтервал автоматичної відправки в мс (за замовчуванням: 5000)\n * @param {Object} options.headers - Додаткові HTTP заголовки\n * @returns {OpenTelemetryReporter} Екземпляр OpenTelemetry репортера\n *\n * @example\n * import { createOpenTelemetryReporter } from '@nitra/consola'\n * const otelReporter = createOpenTelemetryReporter({\n * endpoint: 'http://localhost:4318/v1/logs',\n * serviceName: 'my-app',\n * serviceVersion: '1.0.0'\n * })\n * consola.addReporter(otelReporter)\n */\nexport const createOpenTelemetryReporter = (options = {}) => {\n return new OpenTelemetryReporter(options)\n}\n","/* global location, process, import */\nimport { consola } from 'consola'\nimport { OpenTelemetryReporter } from './otel-reporter.js'\n\nconst options = {}\n\n// Vite Debug mode\nif (typeof __CONSOLA_LEVEL_DEBUG__ !== 'undefined') {\n options.level = 4\n} else if (typeof location !== 'undefined' && location.protocol === 'http:') {\n options.level = 4\n} else if (typeof process !== 'undefined' && (process.env.DEV || process.env.CONSOLA_LEVEL_DEBUG)) {\n // Quasar Debug mode\n options.level = 4\n}\n\n// HTML Popup Reporter — дублювання повідомлень у вигляді спливаючих блоків (без canvas)\nconst POPUP_PADDING = 12\nconst POPUP_LABEL_PADDING = 8\nconst POPUP_LINE_HEIGHT = 20\nconst POPUP_BUTTON_SIZE = 18\nconst POPUP_BUTTON_PADDING = 6\nconst POPUP_MIN_HEIGHT = 36\nconst POPUP_MAX_MESSAGE_LENGTH = 500\n\nconst POPUP_STYLES = `\n#consola-popup-container { position: fixed; inset: 0; pointer-events: none; z-index: 999999; }\n.consola-popup { pointer-events: auto; position: fixed; left: 20px; max-width: calc(100vw - 40px); min-height: ${POPUP_MIN_HEIGHT}px;\n display: flex; align-items: stretch; background: #fff; box-shadow: 0 2px 8px rgba(0,0,0,0.1); border: 1px solid #e0e0e0;\n font: 13px -apple-system, BlinkMacSystemFont, \"Segoe UI\", Roboto, \"Helvetica Neue\", Arial, sans-serif; }\n.consola-popup-label { flex-shrink: 0; padding: 0 ${POPUP_LABEL_PADDING}px; display: flex; align-items: center; justify-content: center;\n color: #fff; font: 11px monospace; font-weight: bold; text-transform: lowercase; }\n.consola-popup-content { flex: 1; min-width: 0; padding: ${POPUP_PADDING}px; display: flex; flex-wrap: wrap; align-items: flex-start; gap: 4px 8px; }\n.consola-popup-text { flex: 1 1 auto; min-width: 0; color: #212121; line-height: ${POPUP_LINE_HEIGHT}px; white-space: pre-wrap; word-break: break-word; }\n.consola-popup-filelink { flex: 0 0 auto; color: #1976D2; font: 11px monospace; text-decoration: underline; cursor: pointer; }\n.consola-popup-filelink:hover { text-decoration: underline; }\n.consola-popup-close { flex-shrink: 0; width: ${POPUP_BUTTON_SIZE}px; height: ${POPUP_BUTTON_SIZE}px; margin: ${POPUP_BUTTON_PADDING}px; padding: 0; border: none; background: transparent;\n color: #999; font-size: 16px; line-height: 1; cursor: pointer; display: flex; align-items: center; justify-content: center; border-radius: 2px; }\n.consola-popup-close:hover { background: rgba(0,0,0,0.06); color: #666; }\n`\n\nclass HtmlPopupReporter {\n constructor() {\n this.container = null\n this.messages = []\n this.messageId = 0\n this.init()\n }\n\n init() {\n if (typeof document === 'undefined') return\n\n const doInit = () => {\n if (!document.body) {\n setTimeout(doInit, 10)\n return\n }\n\n let container = document.querySelector('#consola-popup-container')\n if (!container) {\n const style = document.createElement('style')\n style.textContent = POPUP_STYLES\n document.head.append(style)\n container = document.createElement('div')\n container.id = 'consola-popup-container'\n document.body.append(container)\n }\n\n this.container = container\n this.animate()\n }\n\n if (document.readyState === 'loading') {\n document.addEventListener('DOMContentLoaded', doInit)\n } else {\n doInit()\n }\n }\n\n getColorForType(type) {\n const colors = {\n trace: '#9E9E9E',\n debug: '#607D8B',\n info: '#2196F3',\n log: '#2196F3',\n warn: '#FFC107',\n error: '#D32F2F',\n fatal: '#7B1FA2',\n success: '#4CAF50',\n start: '#00BCD4',\n box: '#9E9E9E',\n ready: '#4CAF50',\n fail: '#D32F2F'\n }\n return colors[type] || colors.log || '#616161'\n }\n\n getTypeLabel(type) {\n return type || 'log'\n }\n\n getFileInfo() {\n try {\n const stack = new Error('Stack trace').stack\n if (!stack) return null\n\n const skipPatterns = ['HtmlPopupReporter', 'CanvasPopupReporter', 'browser.js', 'consola', 'createLogger']\n const lines = stack.split('\\n').slice(4, 15)\n\n for (const line of lines) {\n const trimmed = line.trim()\n const shouldSkip = skipPatterns.some(pattern => trimmed.includes(pattern))\n\n if (!shouldSkip) {\n const match =\n trimmed.match(/\\(?([^()]+):(\\d+):(\\d+)\\)?/) || trimmed.match(/at\\s+[^(]*\\(?([^:]+):(\\d+):(\\d+)\\)?/)\n\n if (match) {\n let file = match[1].trim()\n const lineNum = match[2]\n\n // Очищаємо ім'я файлу\n if (file.includes('://')) {\n file = file.slice(file.indexOf('://') + 3).slice(file.indexOf('/'))\n }\n if (file.includes('/src/')) {\n file = file.slice(file.indexOf('/src/') + 5)\n }\n const queryIndex = file.indexOf('?')\n if (queryIndex > 0) file = file.slice(0, queryIndex)\n const hashIndex = file.indexOf('#')\n if (hashIndex > 0) file = file.slice(0, hashIndex)\n if (file.startsWith('/')) file = file.slice(1)\n\n if (file && lineNum) {\n return { file, line: parseInt(lineNum, 10), column: parseInt(match[3], 10), fullPath: match[0] }\n }\n }\n }\n }\n } catch {\n // Ігноруємо помилки\n }\n return null\n }\n\n formatMessage(logObj) {\n const args = logObj.args || []\n if (args.length === 0) return ''\n\n const message = args\n .map(arg => {\n if (arg instanceof Error) {\n let msg = `${arg.name || 'Error'}: ${arg.message || ''}`\n if (arg.stack && arg.stack !== arg.message) {\n msg += '\\n' + arg.stack.split('\\n').slice(0, 3).join('\\n')\n }\n return msg\n }\n if (arg === null) return 'null'\n if (arg === undefined) return 'undefined'\n if (typeof arg === 'object') {\n try {\n const str = arg.toString?.()\n if (str && str !== '[object Object]') return str\n return JSON.stringify(arg, null, 2)\n } catch {\n return String(arg)\n }\n }\n return String(arg)\n })\n .join(' ')\n\n return message.length > POPUP_MAX_MESSAGE_LENGTH ? message.slice(0, POPUP_MAX_MESSAGE_LENGTH) + '...' : message\n }\n\n log(logObj) {\n if (!this.container) return\n\n const text = this.formatMessage(logObj)\n if (!text) return\n\n const type = logObj.type || 'log'\n const color = this.getColorForType(type)\n const typeLabel = this.getTypeLabel(type)\n const id = this.messageId++\n const fileInfo = this.getFileInfo()\n\n const popup = document.createElement('div')\n popup.className = 'consola-popup'\n popup.dataset.id = String(id)\n popup.style.bottom = `${window.innerHeight}px`\n popup.style.backgroundColor = '#fff'\n\n const label = document.createElement('span')\n label.className = 'consola-popup-label'\n label.style.backgroundColor = color\n label.textContent = typeLabel\n\n const content = document.createElement('div')\n content.className = 'consola-popup-content'\n\n const textEl = document.createElement('div')\n textEl.className = 'consola-popup-text'\n textEl.textContent = text\n\n content.append(textEl)\n\n if (fileInfo) {\n const fileLink = document.createElement('span')\n fileLink.className = 'consola-popup-filelink'\n fileLink.textContent = `${fileInfo.file}:${fileInfo.line}`\n fileLink.role = 'button'\n fileLink.tabIndex = 0\n fileLink.addEventListener('click', e => {\n e.preventDefault()\n console.log(`%c${fileInfo.file}:${fileInfo.line}`, 'color: #1976D2; text-decoration: underline;')\n })\n content.append(fileLink)\n }\n\n const closeBtn = document.createElement('button')\n closeBtn.type = 'button'\n closeBtn.className = 'consola-popup-close'\n closeBtn.setAttribute('aria-label', 'Close')\n closeBtn.textContent = '×'\n closeBtn.addEventListener('click', () => {\n const idx = this.messages.findIndex(m => m.id === id)\n if (idx !== -1) {\n const entry = this.messages[idx]\n if (entry.el && entry.el.parentNode) entry.el.remove()\n this.messages.splice(idx, 1)\n }\n })\n\n popup.append(label, content, closeBtn)\n this.container.append(popup)\n\n const height = popup.offsetHeight\n this.messages.push({\n id,\n el: popup,\n targetBottom: 20,\n currentBottom: window.innerHeight,\n height\n })\n }\n\n animate() {\n if (!this.container) return\n\n let offset = 20\n for (let i = this.messages.length - 1; i >= 0; i--) {\n const msg = this.messages[i]\n if (msg.el && msg.el.parentNode) {\n const h = msg.el.offsetHeight\n msg.height = h\n msg.targetBottom = offset\n msg.currentBottom += (msg.targetBottom - msg.currentBottom) * 0.1\n msg.el.style.bottom = `${Math.round(msg.currentBottom)}px`\n offset += h + 10\n }\n }\n\n requestAnimationFrame(() => this.animate())\n }\n}\n\n// Отримання змінних середовища з Vite (import.meta.env)\nconst getViteEnvVar = name => {\n try {\n // eslint-disable-next-line no-undef\n const viteName = `VITE_${name}`\n // eslint-disable-next-line no-undef\n return import.meta?.env?.[viteName]\n } catch {\n return null\n }\n}\n\n// Перевіряємо змінну середовища VITE_CONSOLA_POPUP_DEBUG\nlet isPopupDebugEnabled = false\n\n// Vite замінює import.meta.env.VITE_* під час збірки\n// Використовуємо прямий доступ до import.meta.env, оскільки Vite обробляє це під час збірки\nif (typeof document !== 'undefined') {\n try {\n // Vite замінює цей код під час збірки\n // eslint-disable-next-line no-undef\n const envValue = import.meta.env?.VITE_CONSOLA_POPUP_DEBUG\n if (envValue !== undefined && envValue !== null && envValue !== '') {\n // Vite завжди повертає рядки для змінних середовища\n isPopupDebugEnabled = String(envValue).toLowerCase() === 'true' || envValue === true\n }\n } catch {\n // Ігноруємо помилки, якщо import.meta недоступний\n }\n}\n\nif (!isPopupDebugEnabled && typeof process !== 'undefined' && process.env) {\n const processValue = process.env.VITE_CONSOLA_POPUP_DEBUG\n if (processValue) {\n isPopupDebugEnabled = String(processValue).toLowerCase() === 'true'\n }\n}\n\n// Створюємо один екземпляр репортера, якщо потрібно\n// НЕ встановлюємо options.reporters, щоб стандартний BrowserReporter працював\n// Додаємо наш репортер після створення екземпляра через addReporter\nlet htmlPopupReporter = null\nif (isPopupDebugEnabled && typeof document !== 'undefined') {\n htmlPopupReporter = new HtmlPopupReporter()\n}\n\nconst defaultConsola = consola.create(options)\n\n// Додаємо HTML репортер до default екземпляра, якщо потрібно (не замінює стандартний)\nif (isPopupDebugEnabled && typeof document !== 'undefined' && htmlPopupReporter) {\n defaultConsola.addReporter(htmlPopupReporter)\n}\n\n// Перевіряємо змінні середовища Vite для OpenTelemetry експорту\nlet openTelemetryReporter = null\nconst otelEndpoint = getViteEnvVar('OTEL_EXPORTER_OTLP_LOGS_ENDPOINT')\n\nif (otelEndpoint) {\n openTelemetryReporter = new OpenTelemetryReporter()\n defaultConsola.addReporter(openTelemetryReporter)\n}\n\nexport default defaultConsola\n\n// Експортуємо consola з нашими опціями, а не базовий\nexport { defaultConsola as consola }\n\n// Експортуємо OpenTelemetryReporter та createOpenTelemetryReporter для прямого використання\nexport { createOpenTelemetryReporter, OpenTelemetryReporter } from './otel-reporter.js'\n\n/**\n * pass import.meta.url\n * example: const consola = createLogger(import.meta.url)\n *\n * @param {String} _url - The import.meta.url or file URL to use for logger creation\n * @returns {Consola} A consola logger instance\n */\nexport const createLogger = _url => {\n return defaultConsola\n}\n","/// <reference path=\"./web-serial.d.ts\" />\nimport { WebPlugin } from '@capacitor/core'\nimport { consola } from '@nitra/consola'\n\nexport class ZebraWeb extends WebPlugin {\n async print({ zpl }) {\n try {\n consola.debug('Запит на підключення до принтера...', 'info')\n\n // Запитуємо порт (Web Serial API)\n const serial = navigator.serial\n if (!serial) throw new Error('Web Serial API не підтримується в цьому браузері')\n const port = await serial.requestPort()\n consola.debug(\"Порт вибрано, відкриваємо з'єднання...\", 'info')\n\n // Відкриваємо з'єднання з налаштуваннями для Zebra принтера\n await port.open({\n baudRate: 9600, // Можна спробувати 115200 якщо не працює\n dataBits: 8,\n parity: 'none',\n stopBits: 1,\n flowControl: 'none'\n })\n\n consola.debug(\"З'єднання відкрито!\", 'success')\n\n // Отримуємо writer для відправки даних\n const writer = port.writable.getWriter()\n\n // Оновлюємо стан\n consola.debug('Принтер готовий до друку', 'success')\n\n if (!port || !writer) {\n consola.debug('Помилка: Принтер не підключено', 'error')\n return\n }\n\n consola.debug(`Відправка ZPL команди: ${zpl.slice(0, 50)}...`, 'info')\n\n // Конвертуємо текст в Uint8Array\n const encoder = new TextEncoder()\n const data = encoder.encode(zpl)\n\n // Відправляємо дані\n await writer.write(data)\n\n // Чекаємо трохи для обробки принтером\n // await new Promise(resolve => setTimeout(resolve, 500))\n\n consola.debug(`✓ Команда відправлена (${data.length} байт)`, 'success')\n consola.debug('print from web capacitor plugin', zpl)\n return {\n success: true\n }\n } catch (error) {\n const message = error instanceof Error ? error.message : String(error)\n consola.debug(`Помилка підключення: ${message}`, 'error')\n\n // Спробуємо перепідключитися\n // await disconnect()\n throw error\n }\n }\n\n getPairedDevices() {\n return {\n devices: [\n { address: '00:00:00:00:00:00', name: 'Virtual Printer' },\n { address: '11:11:11:11:11:11', name: 'Virtual Headphone' }\n ]\n }\n }\n}\n","import { registerPlugin } from '@capacitor/core'\nimport { Preferences } from '@capacitor/preferences'\nimport { normalizePrintArg } from './utils/validate.js'\n\nconst ZebraPlugin = registerPlugin('Zebra', {\n web: () => import('./web.js').then(m => new m.ZebraWeb())\n})\n\nconst Zebra = {\n /**\n * Приймає лише zpl як рядок; address для нативних платформ береться з Preferences (після setPrinterAddress або вибору пристрою).\n * @param {string} zpl - ZPL-рядок для друку\n * @returns {Promise<{success?: boolean, devices?: Array<{address: string, name: string}>, message?: string}>} Результат друку або список пристроїв\n */\n async print(zpl) {\n const zplNormalized = normalizePrintArg(zpl)\n\n const { value: address } = await Preferences.get({ key: 'printer_address' })\n const addressTrimmed = typeof address === 'string' ? address.trim() : ''\n if (!addressTrimmed) {\n const { devices } = await ZebraPlugin.getPairedDevices()\n if (devices?.length > 0) {\n return { success: false, message: 'device list', devices }\n }\n return { success: false, message: 'No paired devices found' }\n }\n return ZebraPlugin.print({ zpl: zplNormalized, address: addressTrimmed })\n },\n getPairedDevices() {\n return ZebraPlugin.getPairedDevices()\n }\n}\n\nexport { Zebra }\n"],"x_google_ignoreList":[0,1,3,4,5,6],"mappings":"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"}
@@ -23,16 +23,20 @@ import ExternalAccessory
23
23
  /// Невелика затримка між черговими завданнями, щоб дати прошивці час стабілізуватися
24
24
  private let cooldownInterval: TimeInterval = 1.0
25
25
 
26
- /// Перемикач debug-логування
27
- private static let debugLogging = true
28
- /// Функція логування з префіксом [Zebra]
29
- private func log(_ items: Any...) {
30
- guard Zebra.debugLogging else { return }
31
- print("[Zebra]", items.map { String(describing: $0) }.joined(separator: " "))
26
+ /// Один раз реєструємося для нотифікацій EA — потрібно для коректного доступу до аксесуарів при холодному запуску (без Xcode).
27
+ private static var didRegisterForAccessoryNotifications = false
28
+ private static let registerLock = NSLock()
29
+ private func ensureRegisteredForAccessoryNotifications() {
30
+ Zebra.registerLock.lock()
31
+ defer { Zebra.registerLock.unlock() }
32
+ guard !Zebra.didRegisterForAccessoryNotifications else { return }
33
+ EAAccessoryManager.shared().registerForLocalNotifications()
34
+ Zebra.didRegisterForAccessoryNotifications = true
35
+ logDebug("EAAccessoryManager.registerForLocalNotifications() called")
32
36
  }
33
37
 
34
38
  /// Представляє інформацію про підключений аксесуар, яку використовує плагін
35
- public struct DeviceInfo {
39
+ public struct DeviceInfo: Sendable {
36
40
  public let address: String
37
41
  public let name: String
38
42
  public init(address: String, name: String) {
@@ -42,7 +46,7 @@ import ExternalAccessory
42
46
  }
43
47
 
44
48
  /// Представляє успішний результат друку
45
- public struct PrintResult {
49
+ public struct PrintResult: Sendable {
46
50
  public let address: String
47
51
  public init(address: String) {
48
52
  self.address = address
@@ -50,7 +54,7 @@ import ExternalAccessory
50
54
  }
51
55
 
52
56
  /// Можливі помилки під час друку
53
- public enum ZebraError: LocalizedError {
57
+ public enum ZebraError: LocalizedError, Sendable {
54
58
  case addressMissing // Адреса принтера не вказана
55
59
  case zplEmpty // ZPL команда порожня
56
60
  /// Пристрій не знайдено: запитувана адреса та список підключених аксесуарів (назва + адреса), щоб показувати їх у помилці
@@ -90,12 +94,10 @@ import ExternalAccessory
90
94
  private let timeout: TimeInterval
91
95
  private var totalWritten = 0
92
96
  private var completion: ((Result<PrintResult, ZebraError>) -> Void)?
93
- private weak var owner: Zebra?
94
97
  private var startTime = Date()
95
98
  private let session: EASession
96
99
 
97
- init(owner: Zebra, session: EASession, output: OutputStream, data: [UInt8], address: String, timeout: TimeInterval = 5.0, completion: @escaping (Result<PrintResult, ZebraError>) -> Void) {
98
- self.owner = owner
100
+ init(owner _: Zebra, session: EASession, output: OutputStream, data: [UInt8], address: String, timeout: TimeInterval = 5.0, completion: @escaping (Result<PrintResult, ZebraError>) -> Void) {
99
101
  self.session = session
100
102
  self.output = output
101
103
  self.data = data
@@ -117,18 +119,18 @@ import ExternalAccessory
117
119
  output.delegate = self
118
120
  output.schedule(in: runLoop, forMode: mode)
119
121
  output.open()
120
- owner?.log("StreamWriter started, bytes=", data.count)
122
+ logDebug("StreamWriter started, bytes= \(self.data.count)")
121
123
  startTime = Date()
122
124
  }
123
125
 
124
- /// Після запису всіх байтів дає час на флаш буфера (iOS може не встигнути відправити на пристрій при миттєвому закритті)
126
+ /// Після запису всіх байтів дає час на флаш буфера (iOS може не встигнути відправити на пристрій при миттєвому закритті, особливо при запуску без Xcode).
125
127
  private func flushThenClose() {
126
- let flushDuration: TimeInterval = 0.35
128
+ let flushDuration: TimeInterval = 0.75
127
129
  let deadline = Date().addingTimeInterval(flushDuration)
128
130
  while Date() < deadline {
129
- RunLoop.current.run(mode: .default, before: Date().addingTimeInterval(0.05))
131
+ RunLoop.current.run(mode: .default, before: Date().addingTimeInterval(0.1))
130
132
  }
131
- owner?.log("Flush done, closing stream")
133
+ logDebug("Flush done, closing stream")
132
134
  close(with: .success(PrintResult(address: address)))
133
135
  }
134
136
 
@@ -159,26 +161,26 @@ import ExternalAccessory
159
161
  completion(result)
160
162
  }
161
163
  completion = nil
162
- owner?.log("StreamWriter finished with", result)
164
+ logDebug("StreamWriter finished with \(String(describing: result))")
163
165
  }
164
166
 
165
167
  func stream(_ aStream: Stream, handle eventCode: Stream.Event) {
166
168
  // Події input-потоку ігноруємо (потрібен лише для коректного життєвого циклу сесії)
167
169
  guard aStream === output else { return }
168
170
  if Date().timeIntervalSince(startTime) > timeout {
169
- owner?.log("StreamWriter timeout")
171
+ logDebug("StreamWriter timeout")
170
172
  close(with: .failure(.writeFailed("Таймаут запису")))
171
173
  return
172
174
  }
173
175
  switch eventCode {
174
176
  case .openCompleted:
175
177
  // Потік відкрито
176
- owner?.log("Stream open completed")
178
+ logDebug("Stream open completed")
177
179
  case .hasSpaceAvailable:
178
180
  // Є простір для запису, пишемо дані порціями
179
181
  if totalWritten >= data.count {
180
182
  // Усі байти записані — даємо час на флаш буфера перед закриттям (iOS часто не встигає відправити на пристрій)
181
- owner?.log("All bytes written, flushing before close...")
183
+ logDebug("All bytes written, flushing before close...")
182
184
  flushThenClose()
183
185
  return
184
186
  }
@@ -187,7 +189,7 @@ import ExternalAccessory
187
189
  let wrote = output.write(Array(data[totalWritten..<(totalWritten + chunkSize)]), maxLength: chunkSize)
188
190
  if wrote <= 0 {
189
191
  let reason = output.streamError?.localizedDescription ?? "Невідома помилка потоку"
190
- owner?.log("StreamWriter write failed:", reason)
192
+ logDebug("StreamWriter write failed: \(reason)")
191
193
  close(with: .failure(.writeFailed(reason)))
192
194
  return
193
195
  }
@@ -195,11 +197,11 @@ import ExternalAccessory
195
197
  case .errorOccurred:
196
198
  // Сталася помилка потоку
197
199
  let reason = output.streamError?.localizedDescription ?? "Невідома помилка потоку"
198
- owner?.log("Stream error:", reason)
200
+ logDebug("Stream error: \(reason)")
199
201
  close(with: .failure(.writeFailed(reason)))
200
202
  case .endEncountered:
201
203
  // Потік закінчився, перевіряємо чи всі дані записані
202
- owner?.log("Stream end encountered, bytes=", totalWritten)
204
+ logDebug("Stream end encountered, bytes= \(self.totalWritten)")
203
205
  if totalWritten >= data.count {
204
206
  close(with: .success(PrintResult(address: address)))
205
207
  } else {
@@ -225,7 +227,19 @@ import ExternalAccessory
225
227
 
226
228
  /// Повертає список підключених EA аксесуарів у вигляді DeviceInfo, вибираючи серійний номер або ім'я як адресу
227
229
  public func getConnectedAccessories() -> [DeviceInfo] {
228
- let accessories = EAAccessoryManager.shared().connectedAccessories
230
+ ensureRegisteredForAccessoryNotifications()
231
+ // Якщо аксесуарів немає, чекаємо 0.6 с і повторюємо спробу до 3 разів (щоб дати системі час підхопити нове підключення)
232
+ let maxAttempts = 3
233
+ let retryDelay: TimeInterval = 0.6
234
+ var accessories = EAAccessoryManager.shared().connectedAccessories
235
+ for attempt in 1...maxAttempts {
236
+ if !accessories.isEmpty { break }
237
+ if attempt < maxAttempts {
238
+ logDebug("connectedAccessories attempt: \(attempt)")
239
+ Thread.sleep(forTimeInterval: retryDelay)
240
+ accessories = EAAccessoryManager.shared().connectedAccessories
241
+ }
242
+ }
229
243
  return accessories.compactMap { accessory in
230
244
  let address = accessory.serialNumber.isEmpty ? accessory.name : accessory.serialNumber
231
245
  if address.isEmpty {
@@ -238,14 +252,14 @@ import ExternalAccessory
238
252
  /// Виводить діагностичну інформацію щодо підключених аксесуарів та протоколів у Info.plist
239
253
  public func diagnoseEnvironment() {
240
254
  let accessories = EAAccessoryManager.shared().connectedAccessories
241
- log("Connected accessories count:", accessories.count)
255
+ logDebug("Connected accessories count: \(accessories.count)")
242
256
  for a in accessories {
243
- log("Accessory:", "name=\(a.name)", "serial=\(a.serialNumber)", "protocols=\(a.protocolStrings)")
257
+ logDebug("Accessory: name=\(a.name) serial=\(a.serialNumber) protocols=\(a.protocolStrings)")
244
258
  }
245
259
  if let supported = Bundle.main.object(forInfoDictionaryKey: "UISupportedExternalAccessoryProtocols") as? [String] {
246
- log("Info.plist UISupportedExternalAccessoryProtocols:", supported)
260
+ logDebug("Info.plist UISupportedExternalAccessoryProtocols: \(supported)")
247
261
  } else {
248
- log("Info.plist UISupportedExternalAccessoryProtocols: MISSING or not an array")
262
+ logDebug("Info.plist UISupportedExternalAccessoryProtocols: MISSING or not an array")
249
263
  }
250
264
  }
251
265
 
@@ -256,10 +270,10 @@ import ExternalAccessory
256
270
  for proto in protocolsToTry {
257
271
  for attempt in 1...retries {
258
272
  if let session = EASession(accessory: accessory, forProtocol: proto) {
259
- self.log("EASession open success on attempt", attempt, "protocol:", proto)
273
+ logDebug("EASession open success on attempt \(attempt) protocol: \(proto)")
260
274
  return (session, proto)
261
275
  } else {
262
- self.log("EASession open failed attempt", attempt, "protocol:", proto)
276
+ logDebug("EASession open failed attempt \(attempt) protocol: \(proto)")
263
277
  Thread.sleep(forTimeInterval: delay)
264
278
  }
265
279
  }
@@ -285,17 +299,25 @@ import ExternalAccessory
285
299
  payload += "\r\n"
286
300
  }
287
301
  let zplBytes = Array(payload.utf8)
288
- log("printZpl called with address=\(trimmedAddress)", "zplLength=\(zplBytes.count)")
302
+ logDebug("printZpl called with address=\(trimmedAddress) zplLength=\(zplBytes.count)")
289
303
 
290
304
  DispatchQueue.main.async { [weak self] in
291
305
  guard let self = self else { return }
306
+ self.ensureRegisteredForAccessoryNotifications()
292
307
  let accessories = EAAccessoryManager.shared().connectedAccessories
293
- self.log("EA connectedAccessories count=", accessories.count)
308
+ logDebug("EA connectedAccessories count= \(accessories.count)")
294
309
 
295
310
  // Шукаємо аксесуар, що відповідає адресі (serialNumber або name)
296
311
  let availableList = self.getConnectedAccessories()
312
+
313
+ if availableList.isEmpty {
314
+ logDebug("No accessories found")
315
+ completion(.failure(.accessoryNotFound(requestedAddress: trimmedAddress, availableAccessories: availableList)))
316
+ return
317
+ }
318
+
297
319
  guard let accessory = accessories.first(where: { self.accessoryMatches($0, address: trimmedAddress) }) else {
298
- self.log("No accessory matched address:", trimmedAddress)
320
+ logDebug("No accessory matched address: \(trimmedAddress)")
299
321
  // У failure передаємо не лише адресу, а й список доступних аксесуарів (назви та адреси) для показу в помилці
300
322
  completion(.failure(.accessoryNotFound(requestedAddress: trimmedAddress, availableAccessories: availableList)))
301
323
  return
@@ -303,30 +325,30 @@ import ExternalAccessory
303
325
 
304
326
  // Обираємо протокол для з’єднання
305
327
  let protocolString = self.preferredProtocol(for: accessory)
306
- self.log("Chosen accessory:", accessory.name, "serial=\(accessory.serialNumber)")
307
- self.log("Selected protocol:", protocolString)
328
+ logDebug("Chosen accessory: \(accessory.name) serial=\(accessory.serialNumber)")
329
+ logDebug("Selected protocol: \(protocolString)")
308
330
  if let supported = Bundle.main.object(forInfoDictionaryKey: "UISupportedExternalAccessoryProtocols") as? [String] {
309
331
  if !supported.contains(protocolString) {
310
- self.log("WARNING: Selected protocol not present in Info.plist UISupportedExternalAccessoryProtocols")
332
+ logDebug("WARNING: Selected protocol not present in Info.plist UISupportedExternalAccessoryProtocols")
311
333
  }
312
334
  } else {
313
- self.log("WARNING: UISupportedExternalAccessoryProtocols missing in Info.plist")
335
+ logDebug("WARNING: UISupportedExternalAccessoryProtocols missing in Info.plist")
314
336
  }
315
337
 
316
- self.log("Attempting to open EASession with retries and fallback protocols...")
338
+ logDebug("Attempting to open EASession with retries and fallback protocols...")
317
339
  Thread.sleep(forTimeInterval: 0.1)
318
340
  let fallback = accessory.protocolStrings
319
341
  // Відкриваємо сесію, пробуємо ретраї і fallback протоколи
320
342
  guard let result = self.openSessionWithRetries(accessory: accessory, preferredProtocol: protocolString, fallbackProtocols: fallback) else {
321
- self.log("EASession failed after retries and fallback")
343
+ logDebug("EASession failed after retries and fallback")
322
344
  completion(.failure(.sessionFailed("EASession повернув nil")))
323
345
  return
324
346
  }
325
347
  let session = result.session
326
348
  let protocolUsed = result.protocolUsed
327
- self.log("Using protocol for session:", protocolUsed)
349
+ logDebug("Using protocol for session: \(protocolUsed)")
328
350
  guard let outputStream = session.outputStream else {
329
- self.log("Session opened but outputStream is nil")
351
+ logDebug("Session opened but outputStream is nil")
330
352
  completion(.failure(.sessionFailed("Відсутній outputStream")))
331
353
  return
332
354
  }
@@ -342,9 +364,9 @@ import ExternalAccessory
342
364
  }
343
365
  writer.start()
344
366
 
345
- // Запускаємо RunLoop, щоб обробляти події стріму
367
+ // Запускаємо RunLoop, щоб обробляти події стріму (довші інтервали — надійніший flush при запуску без Xcode)
346
368
  while group.wait(timeout: .now()) == .timedOut {
347
- RunLoop.current.run(mode: .default, before: Date().addingTimeInterval(0.15))
369
+ RunLoop.current.run(mode: .default, before: Date().addingTimeInterval(0.25))
348
370
  }
349
371
  }
350
372
  }
@@ -370,3 +392,35 @@ import ExternalAccessory
370
392
  }
371
393
  }
372
394
 
395
+ // MARK: - Actor-based API (Swift Concurrency)
396
+
397
+ /// Actor для безпечної роботи з принтером Zebra з підтримкою Swift Concurrency.
398
+ /// Усі виклики серіалізуються через actor isolation.
399
+ public actor ZebraActor {
400
+ private let zebra = Zebra()
401
+
402
+ /// Зберігає адресу принтера у UserDefaults
403
+ public func setPrinterAddress(_ address: String) {
404
+ zebra.setPrinterAddress(address)
405
+ }
406
+
407
+ /// Повертає збережену адресу принтера, якщо є
408
+ public func getStoredAddress() -> String? {
409
+ zebra.getStoredAddress()
410
+ }
411
+
412
+ /// Повертає список підключених EA аксесуарів
413
+ public func getConnectedAccessories() -> [Zebra.DeviceInfo] {
414
+ zebra.getConnectedAccessories()
415
+ }
416
+
417
+ /// Друкує ZPL на принтері за вказаною адресою (async/await API)
418
+ public func printZpl(address: String, zpl: String) async throws -> Zebra.PrintResult {
419
+ try await withCheckedThrowingContinuation { continuation in
420
+ zebra.printZpl(address: address, zpl: zpl) { result in
421
+ continuation.resume(with: result)
422
+ }
423
+ }
424
+ }
425
+ }
426
+
@@ -0,0 +1,11 @@
1
+ import os
2
+
3
+ /// Спільна функція дебаг-логування для модуля Zebra.
4
+ /// У Release не виконується й не обчислює повідомлення (zero cost).
5
+ #if DEBUG
6
+ func logDebug(_ message: @autoclosure @escaping () -> String) {
7
+ Logger(subsystem: "com.nitra.zebra", category: "Zebra").debug("\(message(), privacy: .public)")
8
+ }
9
+ #else
10
+ func logDebug(_ message: @autoclosure @escaping () -> String) {}
11
+ #endif
@@ -19,8 +19,8 @@ public class ZebraPlugin: CAPPlugin, CAPBridgedPlugin {
19
19
  CAPPluginMethod(name: "print", returnType: CAPPluginReturnPromise)
20
20
  ]
21
21
 
22
- /// Екземпляр класу для роботи з принтером Zebra
23
- private let implementation = Zebra()
22
+ /// Actor для роботи з принтером Zebra (Swift Concurrency)
23
+ private let implementation = ZebraActor()
24
24
 
25
25
  // MARK: - Методи плагіна
26
26
 
@@ -35,29 +35,26 @@ public class ZebraPlugin: CAPPlugin, CAPBridgedPlugin {
35
35
  call.reject("Параметр 'address' обов'язковий для iOS (серійний номер або ім'я пристрою).")
36
36
  return
37
37
  }
38
-
39
- // Зберігаємо адресу в UserDefaults
40
- implementation.setPrinterAddress(address)
41
-
42
- // Повертаємо успішний результат
43
- call.resolve([
44
- "address": address
45
- ])
38
+ Task {
39
+ await implementation.setPrinterAddress(address)
40
+ logDebug("setPrinterAddress: \(address)")
41
+ await MainActor.run {
42
+ call.resolve(["address": address])
43
+ }
44
+ }
46
45
  }
47
46
 
48
47
  /// Отримує список підключених пристроїв Zebra
49
48
  /// - Parameter call: Виклик без параметрів
50
49
  @objc func getPairedDevices(_ call: CAPPluginCall) {
51
- // Отримуємо список підключених аксесуарів
52
- let devices = implementation.getConnectedAccessories()
53
-
54
- // Перетворюємо масив DeviceInfo в масив словників
55
- let list = devices.map { ["address": $0.address, "name": $0.name] }
56
-
57
- // Повертаємо результат
58
- call.resolve([
59
- "devices": list
60
- ])
50
+ Task {
51
+ let devices = await implementation.getConnectedAccessories()
52
+ logDebug("getPairedDevices: \(devices.count) device(s)")
53
+ let list = devices.map { ["address": $0.address, "name": $0.name] }
54
+ await MainActor.run {
55
+ call.resolve(["devices": list])
56
+ }
57
+ }
61
58
  }
62
59
 
63
60
  /// Друкує ZPL команди на принтері Zebra
@@ -69,47 +66,54 @@ public class ZebraPlugin: CAPPlugin, CAPBridgedPlugin {
69
66
  call.reject("Параметр 'address' обов'язковий (серійний номер або ім'я пристрою).")
70
67
  return
71
68
  }
72
-
73
- implementation.printZpl(address: address, zpl: zpl, completion: { result in
74
- // Переключаємося на головний потік для виклику Capacitor колбеків
75
- DispatchQueue.main.async {
76
- switch result {
77
- case .success:
78
- // Успішний друк - повертаємо лише success
69
+ Task {
70
+ logDebug("print address=\(address) zplLength=\(zpl.utf8.count)")
71
+ do {
72
+ _ = try await implementation.printZpl(address: address, zpl: zpl)
73
+ logDebug("print success")
74
+ await MainActor.run {
79
75
  call.resolve(["success": true])
80
-
81
- case .failure(let error):
82
- // Помилка друку - визначаємо код помилки
76
+ }
77
+ } catch let error as Zebra.ZebraError {
78
+ logDebug("print failed: \(error.localizedDescription)")
79
+ // Помилка друку - визначаємо код помилки
80
+ await MainActor.run {
83
81
  let code: String
84
82
  switch error {
85
- case .addressMissing:
86
- code = "ADDRESS_MISSING"
87
- case .zplEmpty:
88
- code = "ZPL_EMPTY"
89
- case .accessoryNotFound(let requestedAddress, let availableAccessories):
90
- code = "ACCESSORY_NOT_FOUND"
91
- // Передаємо в JS запитувану адресу та список доступних пристроїв (name, address), щоб показувати їх у UI
92
- let accessoriesPayload = availableAccessories.map { ["name": $0.name, "address": $0.address] }
93
- call.reject(
94
- error.localizedDescription,
95
- code,
96
- error,
97
- ["requestedAddress": requestedAddress, "availableAccessories": accessoriesPayload]
98
- )
99
- return
100
- case .sessionFailed:
101
- code = "SESSION_FAILED"
102
- case .writeFailed:
103
- code = "CONNECTION_FAILED"
104
- @unknown default:
105
- code = "UNKNOWN"
83
+ case .addressMissing:
84
+ code = "ADDRESS_MISSING"
85
+ case .zplEmpty:
86
+ code = "ZPL_EMPTY"
87
+ case .accessoryNotFound(let requestedAddress, let availableAccessories):
88
+ code = "ACCESSORY_NOT_FOUND"
89
+ // Передаємо в JS запитувану адресу та список доступних пристроїв (name, address), щоб показувати їх у UI
90
+ let accessoriesPayload = availableAccessories.map { ["name": $0.name, "address": $0.address] }
91
+ call.reject(
92
+ error.localizedDescription,
93
+ code,
94
+ error,
95
+ ["requestedAddress": requestedAddress, "availableAccessories": accessoriesPayload]
96
+ )
97
+ return
98
+ case .sessionFailed:
99
+ code = "SESSION_FAILED"
100
+ case .writeFailed:
101
+ code = "CONNECTION_FAILED"
102
+ case .protocolMissing:
103
+ code = "PROTOCOL_MISSING"
104
+ @unknown default:
105
+ code = "UNKNOWN"
106
106
  }
107
-
108
107
  // Повертаємо помилку з описом
109
108
  call.reject(error.localizedDescription, code, error)
110
109
  }
110
+ } catch {
111
+ logDebug("print failed (unknown): \(error.localizedDescription)")
112
+ await MainActor.run {
113
+ call.reject(error.localizedDescription, "UNKNOWN", error)
114
+ }
111
115
  }
112
- })
116
+ }
113
117
  }
114
118
  }
115
119
 
@@ -2,14 +2,17 @@ import XCTest
2
2
  @testable import ZebraPlugin
3
3
 
4
4
  class ZebraTests: XCTestCase {
5
- func testEcho() {
6
- // This is an example of a functional test case for a plugin.
7
- // Use XCTAssert and related functions to verify your tests produce the correct results.
8
-
9
- let implementation = Zebra()
10
- let value = "Hello, World!"
11
- let result = implementation.echo(value)
5
+ func testZebraActorSetAndGetAddress() async {
6
+ let actor = ZebraActor()
7
+ await actor.setPrinterAddress("test-address")
8
+ let stored = await actor.getStoredAddress()
9
+ XCTAssertEqual(stored, "test-address")
10
+ }
12
11
 
13
- XCTAssertEqual(value, result)
12
+ func testZebraActorGetConnectedAccessories() async {
13
+ let actor = ZebraActor()
14
+ let devices = await actor.getConnectedAccessories()
15
+ XCTAssertNotNil(devices)
16
+ XCTAssertTrue(devices is [Zebra.DeviceInfo])
14
17
  }
15
18
  }
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@nitra/zebra",
3
- "version": "8.0.3",
3
+ "version": "8.2.0",
4
4
  "description": "Zebra printer",
5
5
  "keywords": [
6
6
  "capacitor",
@@ -34,6 +34,7 @@
34
34
  "scripts": {
35
35
  "verify": "npm run verify:ios && npm run verify:android && npm run verify:web",
36
36
  "verify:ios": "xcodebuild -scheme NitraZebra -destination generic/platform=iOS",
37
+ "test:ios": "swift test --sdk $(xcrun --sdk iphonesimulator --show-sdk-path) -Xswiftc -target -Xswiftc arm64-apple-ios16.0-simulator",
37
38
  "verify:android": "cd android && ./gradlew clean build test && cd ..",
38
39
  "verify:web": "npm run build",
39
40
  "lint": "npm run eslint && npm run prettier -- --check && npm run swiftlint -- lint",