{"version":3,"sources":["@webhare/env/src/envbackend.ts","@webhare/tsrun/src/polyfills.ts","@webhare/env/src/navigation.ts","@webhare/std/src/quacks.ts","@webhare/std/src/datetime.ts","@webhare/std/src/promises.ts","@webhare/std/src/finmath.ts","@webhare/std/src/money.ts","@webhare/std/src/types.ts","@webhare/std/src/strings.ts","@webhare/std/src/platformbased.ts","@webhare/std/src/collections.ts","@webhare/std/src/utils.ts","@webhare/std/src/localmutex.ts","@webhare/std/src/std.ts","@webhare/env/src/fetchdebug.ts","@webhare/env/src/env.ts","mod::system/js/dompack/src/events.ts","@webhare/dompack/src/busy.ts","@webhare/dompack/src/tree.ts","mod::system/js/dompack/src/tree.ts","mod::system/js/dompack/src/create.ts","mod::system/js/dompack/src/components.ts","@webhare/dompack/src/cookiebuilder.ts","@webhare/dompack/src/storage.ts","@webhare/dompack/src/preload.ts","@webhare/dompack/src/browser.ts","@webhare/dompack/src/dompack.ts","mod::system/js/dompack/index.ts","mod::system/js/dompack/extra/browser.ts","node_modules/stacktrace-parser/dist/stack-trace-parser.esm.js","@webhare/js-api-tools/src/stacktracing.ts","@webhare/js-api-tools/src/levenshtein.ts","@webhare/js-api-tools/src/js-api-tools.ts","@webhare/jsonrpc-client/src/jsonrpc-client.ts","@webhare/forms/src/formservice.ts","mod::system/js/dompack/src/debug.ts","mod::system/js/dompack/extra/keyboard.ts","@webhare/gettid/src/hooks.ts","@webhare/gettid/src/internal.ts","@webhare/gettid/src/gettid.ts","@webhare/upload/src/upload.ts"],"sourcesContent":["//We implement the backend version of getWHDebugFlags so bridge can access us without going through a recursive dep\n\nimport type { DTAPStage } from \"./concepts\";\n\n/// List global polyfills currently active. This may be needed to align TypeScript hosts configuration\nexport const globalPolyfills: string[] = [];\n\n/// An object with string keys and typed values\nexport interface WellKnownFlags {\n  /** Log RPcs */\n  rpc?: boolean;\n  /** Log web traffic */\n  wrq?: boolean;\n  /** Autoprofile */\n  apr?: boolean;\n  /** IPC */\n  ipc?: boolean;\n  /** async */\n  async?: boolean;\n}\nexport type DebugFlags = WellKnownFlags & { [key: string]: boolean | undefined };\n\nexport type DebugConfig = {\n  tags: readonly string[];\n  outputsession: string;\n  context: string;\n};\n\nconst settingschangedcallbacks = new Array<() => void>;\n\n/* Global debug flags are set by `wh debug`, local flags are prefilled by environment variables.\n   Edits to the debug flags will be applied to the local flags (unless setDebugFlagsOverrideCB returns\n   override records).\n*/\nconst globalDebugFlags: DebugFlags = {}, localDebugFlags: DebugFlags = {};\n\n/** Returns the flag override records. The first record is examined first, then the second, etc. */\nlet debugFlagsOverridesCB: undefined | (() => DebugFlags[]);\n\n/** Override the debug flag override getter. */\nexport function setDebugFlagsOverrideCB(cb: undefined | (() => DebugFlags[])) {\n  debugFlagsOverridesCB = cb;\n}\n\n/* Proxy handler for the published debug flags. Uses the override records (with the first record\n   examined first, then the second, etc.) to override the flags from baseDebugFlags. Setting to `true` sets\n   the flag, `false` disables it and setting to `undefined` has the same effect as deleting the property in the\n   current top record (exposing any value from a lower record, like the debug settings from `wh debug`).\n*/\nclass DebugFlagsProxyHandler implements ProxyHandler<DebugFlags> {\n  private getRecordList(): DebugFlags[] {\n    return [...debugFlagsOverridesCB?.() ?? [], localDebugFlags, globalDebugFlags];\n  }\n\n  get(target: DebugFlags, p: string) {\n    for (const record of this.getRecordList())\n      if (p in record && typeof record[p] !== \"undefined\")\n        return record[p];\n    return undefined;\n  }\n  has(target: DebugFlags, p: string): boolean {\n    return this.get(target, p) !== undefined;\n  }\n  ownKeys(target: DebugFlags): Array<string | symbol> {\n    const keys = new Array<string | symbol>;\n    for (const record of this.getRecordList())\n      for (const key of Reflect.ownKeys(record))\n        if (typeof key === \"string\" && record[key] !== undefined && !keys.includes(key))\n          keys.push(key);\n    return keys;\n  }\n  set(target: DebugFlags, p: string, newValue: boolean | undefined): boolean {\n    const toModify = this.getRecordList()[0];\n    if (typeof newValue === \"boolean\")\n      toModify[p] = newValue;\n    else\n      delete toModify[p];\n\n    runSettingsCallbacks();\n    return true;\n  }\n  deleteProperty(target: DebugFlags, p: string): boolean {\n    const toModify = this.getRecordList()[0];\n    delete toModify[p];\n\n    runSettingsCallbacks();\n    return true;\n  }\n  getOwnPropertyDescriptor(target: DebugFlags, p: string): PropertyDescriptor | undefined {\n    const value = this.get(target, p);\n    return value !== undefined ? { enumerable: true, value, configurable: true } : undefined;\n  }\n}\n\nexport const debugFlags = new Proxy<DebugFlags>({\n  [Symbol.for('nodejs.util.inspect.custom')]: formatForConsoleLogs\n}, new DebugFlagsProxyHandler());\n\nfunction formatForConsoleLogs() {\n  return `DebugFlags [${[...Object.keys(debugFlags)].filter(key => debugFlags[key]).join(\", \")}]`;\n}\n\nfunction runSettingsCallbacks() {\n  for (const cb of [...settingschangedcallbacks]) {\n    try {\n      cb();\n    } catch (e) {\n      console.error(\"Error invoking settings change callback\", cb?.name, e);\n      //ignore, debugFlags can change due to external updates so they're basically signal handlers and there's no real point in crashing whatever was really running\n    }\n  }\n}\n\n/** Update the debugconfig as present in the system configuration record\n    @param settings - debugconfig cell of the system configuration record\n*/\nexport function updateDebugConfig(settings: DebugConfig | null) {\n  const oldenabledflags = Object.keys(globalDebugFlags).sort();\n  const newenabledflags = settings?.tags ? [...settings.tags].sort() : [];\n\n  if (oldenabledflags.join(\",\") !== newenabledflags.join(\",\")) {\n    for (const flag of newenabledflags)\n      globalDebugFlags[flag] = true;\n    for (const flag of oldenabledflags)\n      if (!newenabledflags.includes(flag))\n        delete globalDebugFlags[flag];\n\n    runSettingsCallbacks();\n  }\n  if (debugFlags.async && Error.stackTraceLimit < 100)\n    Error.stackTraceLimit = 100;\n}\n\nexport function registerDebugConfigChangedCallback(cb: () => void) {\n  settingschangedcallbacks.push(cb);\n}\n\n/** DTAP stage set for this WebHare */\nlet dtapStage: DTAPStage = \"production\";\n\n/** @deprecated We now recommend checking dtapStage against production/acceptance Explicitly */\nlet isLive: boolean = dtapStage === \"production\" || dtapStage === \"acceptance\";\n\n/** The backend base URL. Used for eg. autoconfiguring JSON/RPC */\nlet backendBase = \"\";\n\n//deprecated variants\n/** @deprecated For WH5.4 and up use 'dtapStage' */\nlet dtapstage: DTAPStage = dtapStage;\n/** @deprecated For WH5.4 and up compare 'dtapStage' for production or acceptance */\nlet islive: boolean = isLive;\n\nexport function initEnv(setDtapStage: DTAPStage, setBackendBase: string) {\n  dtapStage = setDtapStage;\n  isLive = dtapStage === \"production\" || dtapStage === \"acceptance\";\n  backendBase = setBackendBase;\n\n  dtapstage = dtapStage;\n  islive = isLive;\n}\n\nexport { dtapStage, isLive, backendBase };\nexport { dtapstage, islive }; //deprecated variants\n","// Set up dispose symbols - https://github.com/evanw/esbuild/pull/3192 \"you'll need to polyfill Symbol.dispose if it's not present before you use it.\n\n//@ts-ignore -- It's marked readonly\nSymbol.dispose ||= Symbol.for('Symbol.dispose');\n//@ts-ignore -- It's marked readonly\nSymbol.asyncDispose ||= Symbol.for('Symbol.asyncDispose');\n\n// Set up promise resolvers (ES2024)\nPromise.withResolvers ||= function <T>(): PromiseWithResolvers<T> {\n  let resolve: (value: T | PromiseLike<T>) => void;\n  let reject: (reason: Error) => void;\n  const promise = new Promise<T>((_resolve, _reject) => { resolve = _resolve; reject = _reject; });\n  // @ts-ignore `resolve` and `reject` are assigned synchronously, which isn't picked up by the TypeScript compiler (see\n  // https://github.com/Microsoft/TypeScript/issues/30053)\n  return { promise, resolve, reject };\n};\n\n// Polyfills for https://caniuse.com/?search=groupby - needed until we can assume safari 17.4 as global baseline\nif (!Map.groupBy)\n  Map.groupBy = function <Item, Key>(items: Iterable<Item>, callbackfn: (item: Item, idx: number) => Key): Map<Key, Item[]> {\n    const retval = new Map<Key, Item[]>;\n    let idx = 0;\n    for (const item of items) {\n      const key = callbackfn(item, idx++);\n      let list = retval.get(key);\n      if (!list)\n        retval.set(key, list = []);\n      list.push(item);\n    }\n    return retval;\n  };\n\nif (!Object.groupBy)\n  Object.groupBy = function <Item, Key extends string | number | symbol>(items: Iterable<Item>, callbackfn: (item: Item, idx: number) => Key): Partial<Record<Key, Item[]>> {\n    return Object.fromEntries(Map.groupBy(items, callbackfn).entries()) as Partial<Record<Key, Item[]>>;\n  };\n","type FormValueList = Array<{ name: string; value: string }>;\n\n/** A navigation instruction encapsulates the next step that a client has to take in eg. a login, payment or wizard flow.\n * Use navigateTo to execute a navigation in a browser\n*/\n\nexport type NavigateInstruction =\n  {\n    type: \"redirect\";\n    url: string;\n  } | {\n    type: \"form\";\n    form: {\n      action: string;\n      vars: FormValueList;\n      method?: \"POST\";\n    };\n  } | {\n    type: \"reload\";\n  } | {\n    type: \"postmessage\";\n    message: unknown;\n    target?: \"parent\" | \"opener\";\n  } | {\n    type: \"close\";\n  };\n\nfunction submitForm(action: string, values: FormValueList, method = \"POST\") {\n  const form = document.createElement(\"form\");\n  form.method = method;\n  form.action = action;\n  form.append(...values.map(item => {\n    const input = document.createElement(\"input\");\n    input.type = \"hidden\";\n    input.name = item.name;\n    input.value = item.value;\n    return input;\n  }));\n  document.body.appendChild(form);\n  form.submit();\n}\n\n/** Navigate to a specific URL (may be hookable by other scripts)\n* @param navigation - Target URL or NavigateInstruction\n*/\nexport function navigateTo(navigation: string | NavigateInstruction) {\n  if (typeof window === \"undefined\")\n    throw new Error(`navigateTo() is not available in this environment`);\n\n  if (typeof navigation === \"string\")\n    navigation = { type: \"redirect\", url: navigation };\n\n  switch (navigation.type) {\n    case \"redirect\":\n      {\n        location.href = navigation.url;\n      } break;\n\n    case \"form\":\n      {\n        submitForm(navigation.form.action, navigation.form.vars, navigation.form.method);\n      } break;\n\n    case \"reload\":\n      {\n        window.location.reload();\n      } break;\n\n    case \"postmessage\":\n      {\n        if (!navigation.target || navigation.target === \"parent\")\n          window.parent.postMessage(navigation.message, \"*\");\n        else if (navigation.target === \"opener\") {\n          window.opener.postMessage(navigation.message, \"*\");\n          window.close();\n        } else\n          throw Error(\"Unknown postmessage target '\" + navigation.target + \"' received\");\n      } break;\n\n    case \"close\":\n      {\n        window.close();\n      } break;\n\n    default:\n      throw new Error(`Unknown navigation type '${(navigation as { type: string }).type}'`);\n  }\n}\n","//TODO can we replace ourselves with nodejs util.types in a pure NodeJS environment?\n\nimport type { Temporal } from \"temporal-polyfill\";\nimport type { Money } from \"./money\";\n\nexport function getWHType(obj: unknown): string | null { //not exporting this directly as we hope JS comes up with better solutions for instanceOf not being compatible with realms/reloading\n  // Ideally we'd use symbol but it breaks tree shaking in esbuild, so this is our workaround. Reported as https://github.com/evanw/esbuild/issues/3940\n  // eslint-disable-next-line @typescript-eslint/no-explicit-any -- Can't cleanly type this it seems and it's low level anyway\n  return (obj as any)?.constructor?.[\"__ $whTypeSymbol\"] ?? null;\n}\n\nfunction isCrossRealm(value: unknown): value is object {\n  return Boolean(value // it's not null\n    && typeof value === \"object\" //and it's an object\n    && !(value instanceof Object)); //and it's not an instance of our Object\n}\n\n/** Test whether a value looks like an instance of Money */\nexport function isMoney(value: unknown): value is Money {\n  return getWHType(value) === \"Money\";\n}\n\n/** Test whether a value looks like an instance of Temporal.Instant */\nexport function isTemporalInstant(value: unknown): value is Temporal.Instant {\n  return Boolean(value && typeof value === \"object\" && Symbol.toStringTag in value && value?.[Symbol.toStringTag] === 'Temporal.Instant');\n}\n\n/** Test whether a value looks like an instance of Temporal.PlainDateTime */\nexport function isTemporalPlainDateTime(value: unknown): value is Temporal.PlainDateTime {\n  return Boolean(value && typeof value === \"object\" && Symbol.toStringTag in value && value?.[Symbol.toStringTag] === 'Temporal.PlainDateTime');\n}\n\n/** Test whether a value looks like an instance of Temporal.PlainDate */\nexport function isTemporalPlainDate(value: unknown): value is Temporal.PlainDate {\n  return Boolean(value && typeof value === \"object\" && Symbol.toStringTag in value && value?.[Symbol.toStringTag] === 'Temporal.PlainDate');\n}\n\n/** Test whether a value looks like an instance of Temporal.ZonedDateTime */\nexport function isTemporalZonedDateTime(value: unknown): value is Temporal.ZonedDateTime {\n  return Boolean(value && typeof value === \"object\" && Symbol.toStringTag in value && value?.[Symbol.toStringTag] === 'Temporal.ZonedDateTime');\n}\n\n/** Test whether a value looks like an instance of Date (assumes no subclasses) */\nexport function isDate(value: unknown): value is Date {\n  return value instanceof Date || (isCrossRealm(value) && value.constructor.name === \"Date\");\n}\n\n/** Test whether a value appears to implement the Blob interface */\nexport function isBlob(value: unknown): value is Blob {\n  return value instanceof Blob || Boolean(value && typeof value === \"object\" && \"size\" in value && \"type\" in value && \"slice\" in value && typeof value.slice === \"function\" && \"stream\" in value && typeof value.stream === \"function\");\n}\n\n/** Test whether a value appears to implement the File interface */\nexport function isFile(value: unknown): value is File {\n  return isBlob(value) && \"name\" in value;\n}\n\n/** Check if the object is probably an Error object. Can't use 'instanceof Error' as an Error might come from a different frame */\nexport function isError(e: unknown): e is Error {\n  return e instanceof Error || (isCrossRealm(e) && \"name\" in e && \"stack\" in e && \"message\" in e);\n}\n\n/** Check if the object looks like a promise */\nexport function isPromise<T>(e: unknown): e is Promise<T> {\n  return Boolean(e && typeof (e as Promise<unknown>).then === \"function\" && typeof (e as Promise<unknown>).catch === \"function\");\n}\n\n/** Check the type of a value, return its JS or STD type\n * @param value - The value to check\n * @returns The type of the value. If the value is an object but recognized as any of Money, Date, Blob, Temporal.Instant/PlainDate/PlainDateTime, that type is returned.\n */\nexport function stdTypeOf(value: unknown): \"string\" | \"number\" | \"boolean\" | \"null\" | \"symbol\" | \"bigint\" | \"function\" | \"object\" | \"undefined\" | \"Date\" | \"Money\" | \"Array\" | \"Instant\" | \"PlainDate\" | \"PlainDateTime\" | \"ZonedDateTime\" | \"File\" | \"Blob\" {\n  const t = typeof value;\n  if (t === \"object\") {\n    if (!value)\n      return \"null\";\n    if (Array.isArray(value))\n      return \"Array\";\n    if (isMoney(value))\n      return \"Money\";\n    if (isBlob(value))\n      return isFile(value) ? \"File\" : \"Blob\";\n    if (isDate(value))\n      return \"Date\";\n    if (isTemporalInstant(value))\n      return \"Instant\";\n    if (isTemporalPlainDate(value))\n      return \"PlainDate\";\n    if (isTemporalPlainDateTime(value))\n      return \"PlainDateTime\";\n    if (isTemporalZonedDateTime(value))\n      return \"ZonedDateTime\";\n  }\n  return t;\n}\n","import { isDate, isTemporalInstant } from \"./quacks\";\nimport type { Temporal } from \"temporal-polyfill\"; //we need an explicit include for separate @webhare/std publication\n\n/** A tolerant JS date parameter */\nexport type FlexibleInstant = Date | Temporal.Instant | Temporal.ZonedDateTime;\n\n/** A relative (up to a week) or absolute wait period. Use 0 for 'polling' and Infinity to indicate an endless waits. Numbers are interpreted to be in milliseconds, a string is interpreted as a ISO8601 duration */\nexport type WaitPeriod = 0 | number | string | FlexibleInstant;\nexport interface Duration {\n  sign: \"+\" | \"-\";\n  years: number;\n  months: number;\n  weeks: number;\n  days: number;\n  hours: number;\n  minutes: number;\n  seconds: number;\n  milliseconds: number;\n}\n\nfunction parseMS(ms: string): number {\n  if (ms.length === 2) //.5\n    return parseInt(ms.substring(1)) * 100;\n  if (ms.length === 3) //.54\n    return parseInt(ms.substring(1)) * 10;\n  return parseInt(ms.substring(1, 4));\n}\n\n/** Parse an ISO8601 duration\n * @param duration - ISO8601 duration string (e.g. \"P1Y2M3DT4H5M6S\")\n */\nexport function parseDuration(duration: string): Duration {\n  // const matches = duration.match(/(-)?P(?:([.,\\d]+)Y)?(?:([.,\\d]+)M)?(?:([.,\\d]+)W)?(?:([.,\\d]+)D)?T(?:([.,\\d]+)H)?(?:([.,\\d]+)M)?(?:([.,\\d]+)S)?/);\n  const matches = duration.match(/^(-)?P(?:(\\d+)Y)?(?:(\\d+)M)?(?:(\\d+)W)?(?:(\\d+)D)?(?:T(?:(\\d+)H)?(?:(\\d+)M)?(?:(\\d+)(\\.\\d+)?S)?)?$/);\n  if (!matches ||\n    (matches[4] && (matches[2] || matches[3] || matches[5])) || //can't have weeks and years/months/days\n    !(matches[2] || matches[3] || matches[4] || matches[5] || matches[6] || matches[7] || matches[8])) //one component must be present\n    throw new Error(`Invalid ISO8601 duration '${duration}'`);\n\n  return {\n    sign: matches[1] ? '-' : '+',\n    years: parseInt(matches[2]) || 0,\n    months: parseInt(matches[3]) || 0,\n    weeks: parseInt(matches[4]) || 0,\n    days: parseInt(matches[5]) || 0,\n    hours: parseInt(matches[6]) || 0,\n    minutes: parseInt(matches[7]) || 0,\n    seconds: parseInt(matches[8]) || 0,\n    milliseconds: matches[9] ? parseMS(matches[9]) : 0\n  };\n}\n\n/** Add a duration (time) to a date\n * @param startingdate - Date to start from\n * @param duration - Duration to add (as object or as ISO8601 duration string, eg \"P1Y2M3DT4H5M6S\")\n */\nexport function addDuration(startingdate: Date, duration: Partial<Duration> | string): Date;\nexport function addDuration(startingdate: Temporal.Instant, duration: Partial<Duration> | string): Temporal.Instant;\n\nexport function addDuration(startingdate: Date | Temporal.Instant, duration: Partial<Duration> | string): Date | Temporal.Instant {\n  if (isTemporalInstant(startingdate))\n    // eslint-disable-next-line @typescript-eslint/no-explicit-any -- if you pass a Temporal.Instant, we should be able to create new ones from Temporal.\n    return (globalThis as any).Temporal.Instant.fromEpochMilliseconds(addDuration(new Date(startingdate.epochMilliseconds), duration).getTime());\n\n  //we cannot take timezones (flexible instant) yet? as we do not timezone correct day/week/month calculations\n  if (typeof duration === \"string\")\n    duration = parseDuration(duration);\n\n  const direction = duration.sign === \"-\" ? -1 : 1;\n  const date = new Date(startingdate.getTime());\n  if (duration.years)\n    date.setUTCFullYear(date.getUTCFullYear() + direction * duration.years);\n  if (duration.months)\n    date.setUTCMonth(date.getUTCMonth() + direction * duration.months);\n\n  const modifydays = (duration.days ?? 0) + (duration.weeks ?? 0) * 7;\n  if (modifydays)\n    date.setUTCDate(date.getUTCDate() + direction * modifydays);\n\n  const timeoffset = (duration.hours ?? 0) * 60 * 60 * 1000 + (duration.minutes ?? 0) * 60 * 1000 + (duration.seconds ?? 0) * 1000 + (duration.milliseconds ?? 0);\n  date.setTime(date.getTime() + direction * timeoffset);\n\n  return date;\n}\n\n/** Subtract a duration (time) from a date\n * @param startingdate - Date to start from\n * @param duration - Duration to subtract (as object or as ISO8601 duration string, eg \"P1Y2M3DT4H5M6S\")\n */\nexport function subtractDuration(startingdate: Date, duration: Partial<Duration> | string): Date;\nexport function subtractDuration(startingdate: Temporal.Instant, duration: Partial<Duration> | string): Temporal.Instant;\n\nexport function subtractDuration(startingdate: Date | Temporal.Instant, duration: Partial<Duration> | string): Date | Temporal.Instant {\n  if (isTemporalInstant(startingdate))\n    // eslint-disable-next-line @typescript-eslint/no-explicit-any -- if you pass a Temporal.Instant, we should be able to create new ones from Temporal.\n    return (globalThis as any).Temporal.Instant.fromEpochMilliseconds(subtractDuration(new Date(startingdate.epochMilliseconds), duration).getTime());\n\n  if (typeof duration === \"string\")\n    duration = parseDuration(duration);\n\n  return addDuration(startingdate, { ...duration, sign: duration?.sign === \"-\" ? \"+\" : \"-\" });\n}\n\n/** Convert a flexible instant to a date\n * @param input - Date, Temporal.Instant or Temporal.ZonedDateTime value to convert\n * @returns The converted date (or null if the input is null)\n */\nexport function convertFlexibleInstantToDate(input: FlexibleInstant): Date;\nexport function convertFlexibleInstantToDate(input: null | undefined): null;\nexport function convertFlexibleInstantToDate(input: FlexibleInstant | null | undefined): Date | null;\n\nexport function convertFlexibleInstantToDate(input: FlexibleInstant | null | undefined): Date | null {\n  return input ? \"epochMilliseconds\" in input ? new Date(input.epochMilliseconds) : input : null;\n}\n\nexport function convertWaitPeriodToDate(wait: WaitPeriod, options: { relativeTo: Temporal.Instant }): Temporal.Instant;\nexport function convertWaitPeriodToDate(wait: WaitPeriod, options?: { relativeTo?: Date }): Date;\n\n/** Convert a WaitPeriod parameter to a Date\n * @param wait - Wait time as milliseconds or a Date\n * @param options - Options for conversion\n * @param options.relativeTo - Date to use as a reference for relative waits\n*/\nexport function convertWaitPeriodToDate(wait: WaitPeriod, options?: { relativeTo?: Date | Temporal.Instant }): Date | Temporal.Instant {\n  if (options?.relativeTo && isTemporalInstant(options?.relativeTo)) { //then our caller expects a Temporal.Instant\n    // eslint-disable-next-line @typescript-eslint/no-explicit-any -- if you pass a Temporal.Instant, we should be able to create new ones from Temporal.\n    return (globalThis as any).Temporal.Instant.fromEpochMilliseconds(\n      convertWaitPeriodToDate(wait, { relativeTo: new Date(options?.relativeTo.epochMilliseconds) }).getTime()\n    );\n  }\n\n  if (isDate(wait)) {\n    return wait;\n  } else if (typeof wait === \"string\") {\n    return addDuration(convertFlexibleInstantToDate(options?.relativeTo) ?? new Date(), wait); //NOTE that \"P1D\" here adds \"24 hours\" which is *not* correct for a TimeZonedDate - which is why convertWaitPeriodToDate cannot yet accept these\n  } else if (typeof wait === \"number\") {\n    if (wait === 0)\n      return new Date(-864000 * 1000 * 10000000);\n    if (wait === Infinity)\n      return new Date(864000 * 1000 * 10000000);\n    if (wait > 0) //note: options?.relativeTo can't be instant, we already eliminated that above\n      return new Date(((options?.relativeTo as Date | undefined)?.getTime() ?? Date.now()) + wait);\n  } else if (wait && \"epochMilliseconds\" in wait) { //Instant or ZonedDateTime\n    return new Date(wait.epochMilliseconds);\n  }\n\n  throw new Error(\"Invalid wait duration - it must either be an absolute date, 0, a number of milliseconds or Infinity\");\n}\n\n/** Check whether the specified parts make a valid and reasonable (usually 1600+) date */\nexport function isValidDate(year: number, month: number, day: number, { minYear = 1601 } = {}) {\n  if (!(year >= minYear && year <= 9999 && month >= 1 && month <= 12 && day >= 1 && day <= 31\n    && Number.isSafeInteger(year) && Number.isSafeInteger(month) && Number.isSafeInteger(day)))\n    return false;\n  if ([4, 6, 9, 11].includes(month) && day > 30) //handle april, june, sep, nov\n    return false;\n  const isleapyear = (year % 400) === 0 || ((year % 100) !== 0 && (year % 4) === 0);\n  if (month === 2 && day > (isleapyear ? 29 : 28))\n    return false;\n  return true;\n}\n\n/** Check whether the specified parts make a reasonable time (does not consider the leap second a valid time) */\nexport function isValidTime(hour: number, minute: number, second: number, msec: number) {\n  return hour >= 0 && hour <= 23 && minute >= 0 && minute <= 59 && second >= 0 && second <= 59 && msec >= 0 && msec <= 999\n    && Number.isSafeInteger(hour) && Number.isSafeInteger(minute) && Number.isSafeInteger(second) && Number.isSafeInteger(msec);\n}\n\n/** Returns a string containing a formatted date and time.\n * @param format  A string that specifies the format of the output string. The string can contain the following:\n%%  Character %\n%a  Abbreviated weekday name\n%A  Full weekday name\n%b  Abbreviated month name\n%B  Full month name\n%d  Two-digit day of month (01 - 31)\n%#d Two-digit day of month, remove any leading zeros\n%H  Hour of the day, 24 hour day\n%#H Hour of the day, 24 hour day, remove any leading zeros\n%I  Two-digit hour, 12 hour day (01 - 12)\n%#I Two-digit hour, 12 hour day, remove any leading zeros\n%j  Three-digit day of year (001 - 366)\n%#j Three-digit day of year, remove any leading zeros\n%m  Two-digit month number (01 - 12)\n%#m Two-digit month number, remove any leading zeros\n%M  2-digit minute (00 - 59)\n%#M 2-digit minute, remove leading any zeros\n%Q  Three-digit millisecond (000-999)\n%#Q Three-digit millisecond, remove any leading zeros\n%p  AM or PM\n%S  Two-digit second (00 - 59)\n%#S Two-digit second, remove leading any zeros\n%V  Two-digot week number (00 - 52)\n%#V One-digit week number (0 - 52)\n%y  Two-digit year without century (00 to 99)\n%#y Two-digit year without century, remove any leading zeros\n%Y  Year with century\n%#Y Year with century, remove any leading zeros\n   @param zdt          The date/time value to format.\n   @example\n// The current date and time in German in the format: 01 Januar 2005\nconst example1 = formatDateTime(\"%d %B %Y\", Temporal.Instant.now(), { locale: \"de-DE\" });\n\n// The current day of the week in the Dutch language, using the\n// parameter datetexts as language switch.\n// The date format is <abbreviated> - <full>.\nconst example2 = formatDateTime(\"%a - %B \", new Date, { locale: \"nl-NL\" });\n*/\n\nexport function formatDateTime(format: string, zdt: Temporal.ZonedDateTime, options?: { locale: string }): string {\n  //TODO consider allowing Date & Instant too? might avoid users going for things like toString().replace(\"T\", \" \").replace(\"Z\", \"\")\n  //const zdt: Temporal.ZonedDateTime = date instanceof Date ? date.toTemporalInstant().toZonedDateTimeISO('UTC') : \"year\" in date ? date : date.toZonedDateTimeISO('UTC');\n  const locale = options?.locale || 'en-US';\n  let stripZeroes = false;\n\n  function formatNumber(num: number, length: number): string {\n    return stripZeroes ? num.toString() : num.toString().padStart(length, '0');\n  }\n\n  function formatPart(spec: string): string {\n    switch (spec) {\n      case 'a': //dayofweek name, abbreviated\n        return zdt.toLocaleString(locale, { weekday: 'short' });\n      case 'A': //dayofweek name, full\n        return zdt.toLocaleString(locale, { weekday: 'long' });\n      case 'b':  //month name, abbreviated\n        return zdt.toLocaleString(locale, { month: 'short' });\n      case 'B': //month name, abbreviated\n        return zdt.toLocaleString(locale, { month: 'long' });\n      case 'C': //century number\n        return formatNumber(Math.floor(zdt.year / 100), 2);\n      case 'd':  //day of month\n        return formatNumber(zdt.day, 2);\n      case 'H':  //hour of the day (24 hours format)\n        return formatNumber(zdt.hour, 2);\n      case 'I'://hour of the day (12 hours format)\n        return formatNumber((zdt.hour % 12) || 12, 2);\n      case 'j'://day of the year (three-digit)\n        return formatNumber(zdt.dayOfYear, 3);\n      case 'M'://two digit minute\n        return formatNumber(zdt.minute, 2);\n      case 'm'://two digit month\n        return formatNumber(zdt.month, 2);\n      case 'p'://am/pm\n        //Unfortunately this doesn't return am/pm: return zdt.toLocaleString(locale, { dayPeriod: 'narrow', hourCycle: \"h12\" });\n        return zdt.hour < 12 ? \"am\" : \"pm\";\n      case 'Q'://three digit millisecond\n        return formatNumber(Math.floor(zdt.millisecond), 3);\n      case 'S'://two digit second\n        return formatNumber(zdt.second, 2);\n      case 'V'://week number\n        if (zdt.weekOfYear === undefined)\n          throw new Error(\"Week of year is not defined for this calendar\");\n        return formatNumber(zdt.weekOfYear, 2);\n      case 'Y'://year, with century\n        return formatNumber(zdt.year, 4);\n      case 'y'://year, without century\n        return formatNumber(zdt.year % 100, 2);\n      default:\n        throw new Error(`Invalid date format specifier '%${spec}'`);\n    }\n  }\n\n  let out = '', inSpecifier = false;\n  for (const spec of [...format]) {\n    if (inSpecifier) {\n      if (spec === '#') {\n        stripZeroes = true;\n        continue; //get the next char\n      }\n\n      inSpecifier = false;\n      if (spec === '%') {\n        out += '%';\n        continue;\n      }\n      out += formatPart(spec);\n    } else {\n      if (spec === '%') {\n        inSpecifier = true;\n        stripZeroes = false; //reset it\n        continue;\n      }\n      out += spec;\n    }\n  }\n  if (inSpecifier)\n    throw new Error(\"Invalid date format string, ends with single %\");\n\n  return out;\n}\n","import { type WaitPeriod, convertWaitPeriodToDate } from \"./datetime\";\n\n//TODO Deprecate once everyone is WH5.5+\nexport type DeferredPromise<T> = PromiseWithResolvers<T>;\n\n/** A promise that sleeps for the specified number of milliseconds\n *  @param duration - Relative (milliseconds, but not Infinity) or absolute (Date) wait duration. May not be Infinity\n *  @param options - Options - `signal` to set an AbortSignal which will canel this sleep\n*/\nexport function sleep(duration: WaitPeriod, options?: { signal?: AbortSignal }): Promise<void> {\n  if (duration === Infinity)\n    throw new Error(`A sleep may not be infinite`);\n\n  const until = convertWaitPeriodToDate(duration);\n  return new Promise(resolve => {\n    const timeoutid = setTimeout(resolve, Math.max(0, until.getTime() - Date.now()));\n    if (options?.signal)\n      options.signal.addEventListener(\"abort\", () => clearTimeout(timeoutid));\n  });\n}\n\n/** @deprecated Use Promise.withResolvers instead */\nexport function createDeferred<T>(): PromiseWithResolvers<T> {\n  return Promise.withResolvers<T>();\n}\n\n/** Wrap a promise in a timeout\n * @param promise - Promise to wrap\n * @param timeout - Timeout in milliseconds or as a Date\n * @param rejectWith - Error to reject with if the timeout expires (string, Error, or a callback returning one)\n*/\nexport function wrapInTimeout<T>(promise: Promise<T>, timeout: WaitPeriod, rejectWith: string | Error | (() => string | Error)): Promise<T> {\n  if (timeout === Infinity)\n    return promise;\n\n  const until = convertWaitPeriodToDate(timeout);\n  const timeoutpromise = new Promise<T>((resolve, reject) => {\n    const timer = setTimeout(() => {\n      //this approach allows to delay actual Error object construction\n      const error: string | Error = typeof rejectWith === \"function\" ? rejectWith() : rejectWith;\n      reject(typeof error === \"object\" ? error : new Error(error));\n    }, until.getTime() - Date.now());\n\n    //ensure timer cancellation. No need to wait on this promise\n    promise = promise.finally(() => clearTimeout(timer));\n  });\n\n  return Promise.race([promise, timeoutpromise]);\n}\n\n// eslint-disable-next-line @typescript-eslint/no-explicit-any -- we need to accept any callback.\ntype CallbackFunctionVariadic<RetType> = (...args: any[]) => Promise<RetType>;\n\nexport interface SerializeOptions {\n  ///Coalescing will ignore intermediate calls if the original call has not completed yet. Eg when invoking functions 1,2,3 in succession, only 3 will run, calls 1 and 2 will be merged into that call\n  coalesce?: boolean;\n}\n\nclass Coalescer<RetVal> {\n  private queue: Array<PromiseWithResolvers<RetVal>> = [];\n  private call: CallbackFunctionVariadic<RetVal> | null = null;\n  private readonly fn: CallbackFunctionVariadic<RetVal>;\n\n  //invoke is the actual entry point for incoming calls (ie C++'s operator() )\n  readonly invoke: CallbackFunctionVariadic<RetVal>;\n\n  constructor(fn: CallbackFunctionVariadic<RetVal>) {\n    this.invoke = this.onInvoke.bind(this);\n    this.fn = fn;\n  }\n  private onInvoke(...args: unknown[]): Promise<RetVal> {\n    //overwrite the call to execute\n    this.call = () => this.fn(...args); //TODO just queueing the parameters might do?\n\n    if (!this.queue.length) //schedule a flush of the queue\n      setTimeout(() => void this.flush(), 0);\n\n    this.queue.push(Promise.withResolvers<RetVal>());\n    return this.queue[this.queue.length - 1].promise;\n  }\n\n  private async flush() {\n    for (let startpos = 0; this.queue && startpos < this.queue?.length;) {\n      const currentpos = this.queue?.length;\n\n      try {\n        //invoke the last known call. *NOTE* new calls may be added to the queue now\n        const result = await this.call!();\n        for (let i = startpos; i < currentpos; i++)\n          this.queue[i].resolve(result);\n      } catch (e) {\n        for (let i = startpos; i < currentpos; i++)\n          this.queue[i].reject(e as Error);\n      }\n      startpos = currentpos;\n    }\n    //empty the queue\n    this.queue.splice(0, this.queue.length);\n  }\n}\n\n// export function<Fn extends (...args: any[]) => Promise<any>>(fn: Fn, options?: SerializeOptions): Fn {\n// type RetVal = ReturnType<Fn>;\n\n/** Wrap a function in a serializer */\n// eslint-disable-next-line @typescript-eslint/no-explicit-any -- we need to accept any callback.\nexport function wrapSerialized<Fn extends (...args: any[]) => Promise<any>>(fn: Fn, options?: SerializeOptions): Fn {\n  type RetVal = ReturnType<Fn>;\n  if (options?.coalesce)\n    return (new Coalescer<RetVal>(fn)).invoke as Fn;\n\n  let queue = Promise.resolve() as Promise<unknown>;\n  return ((...args: unknown[]): Promise<RetVal> => {\n    const res = queue.then(() => fn(...args));\n    queue = res.catch(() => { /* ignore errors */ });\n    return res;\n  }) as Fn;\n}\n\n//TODO abandon the name 'serialize' for now so it's free and safe to use once JS Decorators land (so we can use @serialize there)\n/** @deprecated Use wrapSerialized instead */\nexport function serialize<RetVal>(fn: (...args: unknown[]) => Promise<RetVal>, context: unknown) {\n  return wrapSerialized(fn, undefined);\n}\n","/* A math library for safe calculation with money amounts in JS. If you don't\n   know why we need this, try to predict what the following code would print:\n   var cents=0;for(var i=0;i<100;++i)cents+=0.01;cents-=1;console.log(cents);\n   and think twice about whether you want to code anything financial in JS.\n\n   Money values supplied from HareScript should be formatted using FormatJSFinmathMoney\n*/\n\nexport type FinmathInput = string | number;\n\nexport type RoundMode = \"none\" | \"toward-zero\" | \"down\" | \"up\" | \"half-toward-zero\" | \"half-down\" | \"half-up\" | \"toward-infinity\" | \"half-toward-infinity\";\n\nexport interface SplitNumber {\n  num: number;\n  decimals: number;\n}\n\n//a 'price' is a string of the form \"nnn[.NNNN]\"\n\nfunction stripUnneededDecimals(num: number, decimals: number) {\n  //we have a maximum of 5 digits of external precision\n  if (decimals > 5) {\n    // math.round rounds toward positive infinity\n    const isneg = num < 0;\n    if (isneg)\n      num = -num;\n\n    while (decimals > 6) { //truncate excess digits\n      num = Math.floor(num / 10);\n      --decimals;\n    }\n    //round up if 6th decimal >= 5\n    num = Math.round(num / 10);\n    decimals = 5;\n\n    if (isneg)\n      num = -num;\n  }\n\n  //strip unneeded decimals\n  while (decimals > 0 && !(num % 10)) {\n    num /= 10;\n    --decimals;\n  }\n\n  return { num, decimals };\n}\n\nfunction toText(amount: SplitNumber, decimalpoint: string, mindecimals: number) {\n  if (!Number.isSafeInteger(amount.num))\n    throw new Error(\"Result would overflow the safe value range\");\n\n  let num, decimals;\n  ({ num, decimals } = stripUnneededDecimals(amount.num, amount.decimals));\n\n  // Strip sign from number, may need to prefix it\n  const isnegative = num < 0;\n  if (isnegative)\n    num = -num;\n\n  let astext = String(num);\n\n  // Ensure we have enough leading 0's to render the first integer digit\n  if (astext.length <= decimals)\n    astext = '00000000000000000000'.substr(0, decimals + 1 - astext.length) + astext;\n  // make sure we have enough 0's to show mindecimals\n  if (decimals < mindecimals) {\n    astext += '00000000000000000000'.substr(0, mindecimals - decimals);\n    decimals = mindecimals;\n  }\n\n  // insert decimal point if needed\n  if (decimals)\n    astext = astext.substr(0, astext.length - decimals) + decimalpoint + astext.substr(-decimals);\n\n  // add sign if needed\n  astext = (isnegative ? \"-\" : \"\") + astext;\n  return astext;\n}\n\n/** Convert a money of any format to a split number object\n    @param money - Either an integer number or string with a number\n    @returns Split number object\n*/\nfunction splitValue(money: FinmathInput): SplitNumber {\n  if (typeof money === 'number') {\n    if (money !== Math.floor(money))\n      throw new Error(\"Non-integer number passed\");\n    if (!Number.isSafeInteger(money))\n      throw new Error(`The value ${money} is outside the safe value range`);\n    return { num: money, decimals: 0 };\n  }\n  if (typeof money !== 'string')\n    throw new Error(\"Number or string expected, got \" + money);\n\n  const split = money.match(/^(-)?([0-9]*)(\\.([0-9]{0,5})([0-9]*))?$/);\n  if (!split)\n    throw new Error(`Illegal money value received: '${money}'`);\n\n  const sign = split[1] === '-' ? -1 : 1;\n  const decimals = split[3] ? split[4].length : 0;\n  // If there are more than 5 decimals, round up (using halfExpand strategy)\n  const num = sign * (parseInt(split[2] || \"0\") * Math.pow(10, decimals) + (parseInt((split[4] || '')) || 0) + (split[5]?.[0] >= '5' ? 1 : 0));\n  if (!Number.isSafeInteger(num))\n    throw new Error(`The value '${money}' is outside the safe value range`);\n\n  return stripUnneededDecimals(num, decimals);\n}\n\n/** Convert split number into a string\n*/\nfunction joinPrice(parts: SplitNumber): string {\n  return toText(parts, '.', 0);\n}\n\nfunction adjustDecimals(amount: SplitNumber, requiredecimals: number) {\n  const toadd = requiredecimals - amount.decimals;\n  if (toadd <= 0)\n    return;\n\n  let multiplier = 1;\n  for (let idx = 0; idx < toadd; ++idx)\n    multiplier *= 10;\n\n  if (!Number.isSafeInteger(amount.num * multiplier))\n    throw new Error(\"adjustDecimals would overflow the safe value range\");\n\n  amount.num *= multiplier;\n  amount.decimals += toadd;\n}\n\n/** Returns if a price string is valid\n*/\nexport function isValidPrice(money: string) {\n  if (typeof money !== 'string' || !money.match(/[0-9]+(\\.[0-9]{0,5})?$/))\n    return false;\n  return true;\n}\n\nfunction __add(lhs: SplitNumber, rhs: SplitNumber) {\n  //equalize # of decimals, and then it's a simple addition\n  const requiredecimals = Math.max(lhs.decimals, rhs.decimals);\n  adjustDecimals(lhs, requiredecimals);\n  adjustDecimals(rhs, requiredecimals);\n  return { num: lhs.num + rhs.num, decimals: requiredecimals };\n}\n\n/** Adds two numbers together\n*/\nexport function add(amount1: FinmathInput, amount2: FinmathInput): string {\n  return joinPrice(__add(splitValue(amount1), splitValue(amount2)));\n}\n\n/** Subtracts a number from another number\n*/\nexport function subtract(amount: FinmathInput, tosubtract: FinmathInput) {\n  const lhs = splitValue(amount), rhs = splitValue(tosubtract);\n  rhs.num = -rhs.num;\n  return joinPrice(__add(lhs, rhs));\n}\n\nfunction __multiply(lhs: SplitNumber, rhs: SplitNumber) {\n  //ADDME the naive 'add decimals, multiple nums' approach gets you out of the safe range real fast. needs tests\n  return { num: lhs.num * rhs.num, decimals: lhs.decimals + rhs.decimals };\n}\n\n/** Multiplies two numbers together\n*/\nexport function multiply(amount1: FinmathInput, amount2: FinmathInput): string {\n  const lhs = splitValue(amount1), rhs = splitValue(amount2);\n  return joinPrice(__multiply(lhs, rhs));\n}\n\n/** Compares two numbers\n    @param amount1 - Left hand value\n    @param amount2 - Right hand value\n    @returns Returns 0 if amount1 === amount2, -1 if amount1 \\< amount2, 1 if amount1 \\> amount2\n*/\nexport function cmp(amount1: FinmathInput, amount2: FinmathInput) {\n  const diff = __add(splitValue(amount1), __multiply(splitValue(amount2), { num: -1, decimals: 0 }));\n  return diff.num < 0 ? -1 : diff.num === 0 ? 0 : 1;\n}\n\n/** Returns a percentage of an amount\n    @param amount - Original amount\n    @param perc - Percentage of the amount to return\n    @returns Percentage of the amount\n*/\nexport function getPercentageOfAmount(amount: FinmathInput, perc: FinmathInput) {\n  const lhs = splitValue(amount), rhs = splitValue(perc);\n  const result = __multiply(lhs, rhs);\n  result.decimals += 2;\n  return joinPrice(normalize(result));\n}\n\nfunction normalize(amount: SplitNumber) {\n  while (amount.decimals && (amount.num % 10) === 0) {\n    amount.num /= 10;\n    --amount.decimals;\n  }\n  return amount;\n}\n\n/// format a price amount. extend # of decimals to specified # if not enough\nexport function formatPrice(money: FinmathInput, decimalpoint: string, decimals: number) {\n  return toText(splitValue(money), decimalpoint, decimals);\n}\n\n/** Rounds integer to multiple, exposed for testing only\n*/\nexport function __roundIntegerToMultiple(value: number, roundunit: number, mode: RoundMode) {\n  switch (mode) {\n    case \"none\":\n      {\n        // no rounding\n      } break;\n    case \"toward-zero\":\n      {\n        value = Math.trunc(value / roundunit) * roundunit;\n      } break;\n    case \"toward-infinity\":\n      {\n        if (value > 0)\n          value = Math.trunc((value + roundunit - 1) / roundunit) * roundunit;\n        else\n          value = Math.trunc((value - roundunit + 1) / roundunit) * roundunit;\n      } break;\n    case \"down\":\n      {\n        if (value > 0)\n          value = Math.trunc(value / roundunit) * roundunit;\n        else\n          value = Math.trunc((value - roundunit + 1) / roundunit) * roundunit;\n      } break;\n    case \"up\":\n      {\n        if (value > 0)\n          value = Math.trunc((value + roundunit - 1) / roundunit) * roundunit;\n        else\n          value = Math.trunc(value / roundunit) * roundunit;\n      } break;\n    case \"half-toward-zero\":\n      {\n        if (value > 0)\n          value = Math.trunc((value + (roundunit - 1) / 2) / roundunit) * roundunit;\n        else\n          value = Math.trunc((value - (roundunit - 1) / 2) / roundunit) * roundunit;\n      } break;\n    case \"half-toward-infinity\":\n      {\n        if (value > 0)\n          value = Math.trunc((value + roundunit / 2) / roundunit) * roundunit;\n        else\n          value = Math.trunc((value - roundunit / 2) / roundunit) * roundunit;\n      } break;\n    case \"half-down\":\n      {\n        if (value > 0)\n          value = Math.trunc((value + (roundunit - 1) / 2) / roundunit) * roundunit;\n        else\n          value = Math.trunc((value - roundunit / 2) / roundunit) * roundunit;\n      } break;\n    case \"half-up\":\n      {\n        if (value > 0)\n          value = Math.trunc((value + roundunit / 2) / roundunit) * roundunit;\n        else\n          value = Math.trunc((value - (roundunit - 1) / 2) / roundunit) * roundunit;\n      } break;\n    default:\n      {\n        throw new Error(`Unknown rounding mode ${mode}`);\n      }\n  }\n  return value;\n}\n\n/** Rounds a value to a multiple of a unit, with a specific rounding mode\n    @param value - Value to round\n    @param unit - The value will be rounded to a mulitple of this unit (except when rounding mode is 'none')\n    @param mode - Rounding mode. Possible values:\n      <ul>\n        <li>none: No rounding</li>\n        <li>toward-zero: Round toward zero</li>\n        <li>down: Round toward negative infity</li>\n        <li>up: Round toward positive infity</li>\n        <li>half-toward-zero: Round nearest multiple, round half of a multiple toward zero</li>\n        <li>half-down: Round nearest multiple, round half of a multiple toward negative infinity</li>\n        <li>half-up: Round nearest multiple, round half of a multiple toward positive infity</li>\n      </ul>\n    @returns The rounded value\n*/\nexport function roundToMultiple(value: FinmathInput, unit: FinmathInput, mode: RoundMode) {\n  const split_value = splitValue(value);\n  const split_unit = splitValue(unit);\n\n  const requiredecimals = Math.max(split_value.decimals, split_unit.decimals);\n  adjustDecimals(split_value, requiredecimals);\n  adjustDecimals(split_unit, requiredecimals);\n\n  const result = { num: __roundIntegerToMultiple(split_value.num, split_unit.num, mode), decimals: split_value.decimals };\n  return joinPrice(result);\n}\n\n/** Returns the minimum of all the arguments\n    @param amount - First value\n    @param amounts - Rest of the values\n    @returns The lowest value among amount and amounts\n*/\nexport function min(amount: FinmathInput, ...amounts: FinmathInput[]) {\n  for (const val of amounts)\n    if (cmp(amount, val) > 0)\n      amount = val;\n  return joinPrice(splitValue(amount));\n}\n\n/** Returns the maximum of all the arguments\n    @param amount - First value\n    @param amounts - Rest of the values\n    @returns The highest value among amount and amounts\n*/\nexport function max(amount: FinmathInput, ...amounts: FinmathInput[]) {\n  for (const val of amounts)\n    if (cmp(amount, val) < 0)\n      amount = val;\n  return joinPrice(splitValue(amount));\n}\n\n/** Returns a power of 10\n    @param exp - Integer power, must be 0 or bigger\n    @returns Requested power of 10\n*/\nfunction getNonNegativePowerOf10(exp: number) {\n  let retval = 1, running_exp = 10;\n  while (exp) {\n    if (exp & 1)\n      retval *= running_exp;\n    running_exp *= running_exp;\n    exp = exp >> 1;\n  }\n  return retval;\n}\n\n/** Divides two values, with 5 decimals of precision\n    @param value - Value To divide\n    @param divisor - Divisor\n    @returns Divided value, with 5 decimals of precision\n*/\nexport function divide(value: FinmathInput, divisor: FinmathInput) {\n  const lhs = splitValue(value), rhs = splitValue(divisor);\n  const mul10exp = lhs.decimals - rhs.decimals - 5;\n  const mulfactor = mul10exp < 0 ? getNonNegativePowerOf10(-mul10exp) : 1;\n  const roundunit = mul10exp > 0 ? getNonNegativePowerOf10(mul10exp) : 1;\n  return joinPrice({ num: __roundIntegerToMultiple(mulfactor * lhs.num / rhs.num, roundunit, \"half-toward-infinity\"), decimals: 5 });\n}\n\nexport function moneyDivide(value: FinmathInput, divisor: FinmathInput) { //divide was added in 5.0. remove this old name eventually\n  return divide(value, divisor);\n}\n\n/** Test whether two values have a specific relation\n    @param lhs - Left hand value\n    @param relation - One of '\\<', '\\<=', '==', '!=', '\\>', '\\>=''\n    @param rhs - Right hand value\n    @returns TRUE if the relation holds\n    @example\n      console.log(finmath.test(1, '\\<', 2)); // prints 'true'\n*/\nexport function test(lhs: FinmathInput, relation: \"<\" | \"<=\" | \"==\" | \"!=\" | \">\" | \">=\", rhs: FinmathInput) {\n  const compareresult = cmp(lhs, rhs);\n  switch (relation) {\n    case '<': return compareresult < 0;\n    case '<=': return compareresult <= 0;\n    case '==': return compareresult === 0;\n    case '!=': return compareresult !== 0;\n    case '>': return compareresult > 0;\n    case '>=': return compareresult >= 0;\n  }\n  throw Error(`Cannot test for unknown relation '${relation}'`);\n}\n","/* A Money type library for safe calculation with money amounts in JS. If you don't\n   know why we need this, try to predict what the following code would print:\n   var cents=0;for(var i=0;i<100;++i)cents+=0.01;cents-=1;console.log(cents);\n   and think twice about whether you want to code anything financial in JS.\n\n   We haven't decideded yet what range Money should support. To keep our options open and keep HS compatbility for now\n   we will fix ourselves to 5 digits (HS compat) and not allow values bigger than 900_000_000 (2**53 / 100000 'number' compatbiility clamped to a more readable number)\n\n  Users should not rely on JS Money keeping this smaller range in the future. Money values supplied from HareScript should be formatted using FormatJSFinmathMoney\n*/\n\nimport * as finmath from \"./finmath\"; //TODO absorb into us as soon as noone externally depends on finmath anymore\nimport { isMoney } from \"./quacks\";\n\nexport type MoneyRoundingMode = \"none\" | \"toward-zero\" | \"down\" | \"up\" | \"half-toward-zero\" | \"half-down\" | \"half-up\" | \"toward-infinity\" | \"half-toward-infinity\";\nexport type MoneyTestTypes = \"<\" | \"<=\" | \"==\" | \"!=\" | \">\" | \">=\";\n\ntype MoneyParameter = Money | string;\n\ninterface SplitNumber {\n  num: number;\n  decimals: number;\n}\n\nexport interface MoneyFormatOptions {\n  //thousand separator. defaults to \"\"\n  thousandsSeparator?: string;\n  //decimal separator. defaults to \".\"\n  decimalSeparator?: string;\n  //minimum # of decimals. defaults to 2\n  minDecimals?: number;\n}\n\nfunction stripUnneededDecimals(num: number, decimals: number) {\n  //we have a maximum of 5 digits of external precision\n  if (decimals > 5) {\n    // math.round rounds toward positive infinity\n    const isneg = num < 0;\n    if (isneg)\n      num = -num;\n\n    while (decimals > 6) { //truncate excess digits\n      num = Math.floor(num / 10);\n      --decimals;\n    }\n    //round up if 6th decimal >= 5\n    num = Math.round(num / 10);\n    decimals = 5;\n\n    if (isneg)\n      num = -num;\n  }\n\n  //strip unneeded decimals\n  while (decimals > 0 && !(num % 10)) {\n    num /= 10;\n    --decimals;\n  }\n\n  return { num, decimals };\n}\n\n/** Convert a money of any format to a split number object\n    @param money - Either an integer number or string with a number\n    @returns Split number object\n*/\nfunction splitValue(money: MoneyParameter): SplitNumber {\n  if (typeof money === 'number') {\n    if (money !== Math.floor(money))\n      throw new Error(\"Non-integer number passed\");\n    if (!Number.isSafeInteger(money))\n      throw new Error(`The value ${money} is outside the safe value range`);\n    return { num: money, decimals: 0 };\n  }\n  if (typeof money !== 'string')\n    throw new Error(\"Number or string expected, got \" + money);\n\n  const split = money.match(/^(-)?([0-9]*)(\\.([0-9]{0,5})([0-9]*))?$/);\n  if (!split)\n    throw new Error(`Illegal money value received: '${money}'`);\n\n  const sign = split[1] === '-' ? -1 : 1;\n  const decimals = split[3] ? split[4].length : 0;\n  // If there are more than 5 decimals, round up (using halfExpand strategy)\n  const num = sign * (parseInt(split[2] || \"0\") * Math.pow(10, decimals) + (parseInt((split[4] || '')) || 0) + (split[5]?.[0] >= '5' ? 1 : 0));\n  if (!Number.isSafeInteger(num))\n    throw new Error(`The value '${money}' is outside the safe value range`);\n\n  return stripUnneededDecimals(num, decimals);\n}\n\nfunction toText(amount: SplitNumber, decimalpoint: string, mindecimals: number, thousandpoint: string) {\n  if (!Number.isSafeInteger(amount.num))\n    throw new Error(\"Result would overflow the safe value range\");\n\n  let { num, decimals } = stripUnneededDecimals(amount.num, amount.decimals);\n\n  // Strip sign from number, may need to prefix it\n  const isnegative = num < 0;\n  if (isnegative)\n    num = -num;\n\n  let astext = String(num);\n\n  // Ensure we have enough leading 0's to render the first integer digit\n  if (astext.length <= decimals)\n    astext = '00000000000000000000'.substring(0, decimals + 1 - astext.length) + astext;\n  // make sure we have enough 0's to show mindecimals\n  if (decimals < mindecimals) {\n    astext += '00000000000000000000'.substring(0, mindecimals - decimals);\n    decimals = mindecimals;\n  }\n\n  let beforepoint = astext.substring(0, astext.length - decimals);\n  const afterpoint = astext.substring(astext.length - decimals);\n\n  // Add thouands points if neeed\n  if (thousandpoint)\n    beforepoint = beforepoint.replaceAll(/\\B(?=(\\d{3})+(?!\\d))/g, thousandpoint);\n\n  return (isnegative ? \"-\" : \"\") + beforepoint + (afterpoint.length ? decimalpoint + afterpoint : \"\");\n}\n\n/** A decimal based JS money type*/\nexport class Money {\n  /** finmath-compatible value */\n  readonly value: string;\n  private static \"__ $whTypeSymbol\" = \"Money\";\n\n  constructor(value: MoneyParameter = \"0\") {\n    this.value = Money.parseParameter(value);\n\n    const intvalue = parseInt(this.value);\n    //We need the number to be in the safe range even after adding 5 decimals\n    if (intvalue <= (Number.MIN_SAFE_INTEGER / 100000) || intvalue >= (Number.MAX_SAFE_INTEGER / 100000))\n      throw new TypeError(`Money value '${value}' is out of range`);\n  }\n\n  private static parseParameter(param: MoneyParameter): string {\n    if (typeof param === \"string\")\n      return toText(splitValue(param), \".\", 0, \"\");\n    if (Money.isMoney(param))\n      return (param as unknown as { value: string }).value;\n\n    throw new TypeError(`Money cannot be constructed out of a value of type ${typeof param}`);\n  }\n\n  static isMoney(value: unknown): value is Money {\n    return isMoney(value);\n  }\n\n  static fromNumber(value: number): Money {\n    return new Money(String(value));\n  }\n\n  /** Adds numbers together\n  */\n  static add(left: MoneyParameter, ...right: MoneyParameter[]): Money {\n    let sum = Money.parseParameter(left);\n    for (const item of right)\n      sum = finmath.add(sum, Money.parseParameter(item));\n    return new Money(sum);\n  }\n\n  /** Subtracts a number from another number\n  */\n  static subtract(left: MoneyParameter, right: MoneyParameter): Money {\n    return new Money(finmath.subtract(Money.parseParameter(left), Money.parseParameter(right)));\n  }\n\n  /** Rounds a value to a multiple of a unit, with a specific rounding mode\n      @param amount - Value to round\n      @param roundto - The value will be rounded to a mulitple of this unit (except when rounding mode is 'none')\n      @param mode - Rounding mode. Possible values:<br>\n        <ul>\n          <li>none: No rounding</li>\n          <li>toward-zero: Round toward zero</li>\n          <li>down: Round toward negative infity</li>\n          <li>up: Round toward positive infity</li>\n          <li>half-toward-zero: Round nearest multiple, round half of a multiple toward zero</li>\n          <li>half-down: Round nearest multiple, round half of a multiple toward negative infinity</li>\n          <li>half-up: Round nearest multiple, round half of a multiple toward positive infity</li>\n        </ul>\n      @returns The rounded value\n  */\n  static roundToMultiple(amount: MoneyParameter, roundto: MoneyParameter, mode: MoneyRoundingMode): Money {\n    return new Money(finmath.roundToMultiple(Money.parseParameter(amount), Money.parseParameter(roundto), mode));\n  }\n\n  /** Compares two numbers\n      @param lhs - Left hand value\n      @param rhs - Right hand value\n      @returns Returns 0 if amount1 === amount2, -1 if amount1 \\< amount2, 1 if amount1 \\> amount2\n  */\n  static cmp(lhs: MoneyParameter, rhs: MoneyParameter): -1 | 0 | 1 {\n    return finmath.cmp(Money.parseParameter(lhs), Money.parseParameter(rhs));\n  }\n\n  /** Test whether two values have a specific relation\n      @param lhs - Left hand value\n      @param relation - One of '\\<', '\\<=', '==', '!=', '\\>', '\\>=''\n      @param rhs - Right hand value\n      @returns TRUE if the relation holds\n      @example\n        console.log(Money.check(1, '\\<', 2)); // prints 'true'\n  */\n  static check(lhs: MoneyParameter, relation: MoneyTestTypes, rhs: MoneyParameter): boolean {\n    return finmath.test(Money.parseParameter(lhs), relation, Money.parseParameter(rhs));\n  }\n\n  /** Multiplies two numbers together\n  */\n  static multiply(left: MoneyParameter, right: MoneyParameter): Money {\n    return new Money(finmath.multiply(Money.parseParameter(left), Money.parseParameter(right)));\n  }\n\n  /** Returns a percentage of an amount\n      @param amount - Original amount\n      @param percentage - Percentage of the amount to return\n      @returns Percentage of the amount\n  */\n  static getPercentage(amount: MoneyParameter, percentage: MoneyParameter): Money {\n    return new Money(finmath.getPercentageOfAmount(Money.parseParameter(amount), Money.parseParameter(percentage)));\n  }\n\n  /** Divides two values, (currently with up to 5 decimals of precision)\n      @param numerator - Value to divide\n      @param divisor - Divisor\n      @returns Divided value\n  */\n  static divide(numerator: MoneyParameter, divisor: MoneyParameter): Money {\n    return new Money(finmath.divide(Money.parseParameter(numerator), Money.parseParameter(divisor)));\n  }\n  /** Returns the minimum of all the arguments\n      @param amount - First value\n      @param amounts - Rest of the values\n      @returns The lowest value among amount and amounts\n  */\n  static min(amount: MoneyParameter, ...amounts: MoneyParameter[]): Money {\n    return new Money(finmath.min(Money.parseParameter(amount), ...(amounts.map(Money.parseParameter))));\n\n  }\n\n  /** Returns the maximum of all the arguments\n      @param amount - First value\n      @param amounts - Rest of the values\n      @returns The highest value among amount and amounts\n  */\n  static max(amount: MoneyParameter, ...amounts: MoneyParameter[]): Money {\n    return new Money(finmath.max(Money.parseParameter(amount), ...(amounts.map(Money.parseParameter))));\n  }\n\n  /** format a price amount. extend # of decimals to specified # if not enough */\n  format(format?: MoneyFormatOptions): string {\n    return toText(splitValue(this.value),\n      format?.decimalSeparator ?? \".\",\n      format?.minDecimals ?? 2,\n      format?.thousandsSeparator ?? \"\");\n  }\n\n  toJSON(): string {\n    return this.value;\n  }\n\n  toString(): string {\n    return this.value;\n  }\n\n  toNumber(): number {\n    return parseFloat(this.value);\n  }\n}\n","import { Money } from \"./money\";\nimport { isBlob, isDate, isMoney, isTemporalInstant, isTemporalPlainDate, isTemporalPlainDateTime, isTemporalZonedDateTime, stdTypeOf } from \"./quacks\";\nimport type { Temporal } from \"temporal-polyfill\";\n\n/// Returns T or a promise resolving to T\nexport type MaybePromise<T> = Promise<T> | T;\n\nexport type ComparableType = number | null | bigint | string | Date | Money | boolean | Uint8Array | Temporal.Instant | Temporal.PlainDate | Temporal.ZonedDateTime;\n\ntype CamelCase<S extends string> = S extends `${infer P1}_${infer P2}${infer P3}`\n  ? `${Lowercase<P1>}${Uppercase<P2>}${CamelCase<P3>}`\n  : Lowercase<S>;\n\ntype KeysToCamelCase<T> = {\n  [K in keyof T as CamelCase<string & K>]: ToCamelCase<T[K]>;\n};\n\ntype CamelToSnakeCase<S extends string> = S extends `${infer T}${infer U}` ?\n  `${T extends Capitalize<T> ? \"_\" : \"\"}${Lowercase<T>}${CamelToSnakeCase<U>}` :\n  S;\n\ntype KeysToSnakeCase<T> = {\n  [K in keyof T as CamelToSnakeCase<string & K>]: ToSnakeCase<T[K]>\n};\n\ntype NonRecordTypes = Date | Money | Blob | Temporal.Instant | Temporal.PlainDate | Temporal.PlainDateTime | Temporal.ZonedDateTime;\n\nexport type ToSnakeCase<T> =\n  T extends unknown[] ? Array<ToSnakeCase<T[number]>> :\n  T extends NonRecordTypes ? T :\n  T extends object ? KeysToSnakeCase<T> :\n  T;\n\nexport type ToCamelCase<T> =\n  T extends unknown[] ? Array<ToCamelCase<T[number]>> :\n  T extends NonRecordTypes ? T :\n  T extends object ? KeysToCamelCase<T> :\n  T;\n\n/** Convert a snake_case_name to its corresponding camelCaseName\n * @param name - Name to convert\n * @returns Converted name\n*/\nexport function nameToCamelCase(name: string) {\n  return name.replaceAll(/_[a-z]/g, c => c[1].toUpperCase());\n}\n\n/** Convert a camelCaseName to corresponding snake_case_name\n * @param name - Name to convert\n * @returns Converted name\n*/\nexport function nameToSnakeCase(name: string) {\n  return name.replaceAll(/[A-Z]/g, c => '_' + c.toLowerCase());\n}\n\n/// Should we rewrite the key/values in this object?\nfunction recodeObject(inp: object) {\n  return !isDate(inp) && !isMoney(inp) && !isBlob(inp) && !isTemporalInstant(inp) && !isTemporalPlainDate(inp) && !isTemporalPlainDateTime(inp) && !isTemporalZonedDateTime(inp);\n}\n\nfunction recursiveToSnakeCase<T>(inp: T): ToSnakeCase<T> {\n  if (Array.isArray(inp))\n    return inp.map(toSnakeCase) as ToSnakeCase<T>;\n  if (inp && typeof inp === \"object\" && recodeObject(inp))\n    return Object.fromEntries(Object.entries(inp).map(([key, value]) => [nameToSnakeCase(key), recursiveToSnakeCase(value)])) as ToSnakeCase<T>;\n  return inp as ToSnakeCase<T>;\n}\n\nfunction recursiveToCamelCase<T>(inp: T): ToCamelCase<T> {\n  if (Array.isArray(inp))\n    return inp.map(toCamelCase) as ToCamelCase<T>;\n  if (inp && typeof inp === \"object\" && recodeObject(inp))\n    return Object.fromEntries(Object.entries(inp).map(([key, value]) => [nameToCamelCase(key), recursiveToCamelCase(value)])) as ToCamelCase<T>;\n  return inp as ToCamelCase<T>;\n}\n\n/** Convert all keys to camel case recursively\n * @param inp - Array or object to convert\n * @returns Converted object\n*/\nexport function toSnakeCase<T extends object | object[] | null | undefined>(inp: T): ToSnakeCase<T> {\n  return recursiveToSnakeCase(inp);\n}\n\n/** Convert all keys to snake case recursively\n * @param inp - Array or object to convert\n * @returns Converted object\n*/\nexport function toCamelCase<T extends object | object[] | null | undefined>(inp: T): ToCamelCase<T> {\n  return recursiveToCamelCase(inp);\n}\n\n// TODO make this a quack if we're sure Uint8array is what we want to support - this is here basically to give Buffer support to be compatible with hscompat' compare\nfunction isUInt8Array(value: unknown): value is Uint8Array {\n  return value !== null && typeof value === \"object\" && \"length\" in value && (value as Uint8Array).BYTES_PER_ELEMENT === 1;\n}\n\n/** Compare two values of std-supported types\n * @param left - First value\n * @param right - Second value\n * @returns -1 if left \\< right, 0 if equal, 1 if left \\> right\n */\nexport function compare(left: ComparableType, right: ComparableType): -1 | 0 | 1 {\n  if (left === null)\n    return right === null ? 0 : -1;\n  else if (right === null)\n    return 1;\n\n  switch (typeof left) {\n    case \"boolean\": {\n      if (typeof right === \"boolean\")\n        return left !== right ? left < right ? -1 : 1 : 0;\n    } break;\n    case \"number\": {\n      switch (typeof right) {\n        case \"bigint\": {\n          const right_number = Number(right);\n          return left !== right_number ? left < right_number ? -1 : 1 : 0;\n        }\n        case \"number\": {\n          return left !== right ? left < right ? -1 : 1 : 0;\n        }\n        case \"object\": {\n          if (Money.isMoney(right))\n            return Money.cmp(left.toString(), right);\n        }\n      }\n    } break;\n    case \"bigint\": {\n      switch (typeof right) {\n        case \"bigint\": {\n          return left !== right ? left < right ? -1 : 1 : 0;\n        }\n        case \"number\": {\n          const left_number = Number(left);\n          return left_number !== right ? left_number < right ? -1 : 1 : 0;\n        }\n        case \"object\": {\n          if (Money.isMoney(right)) {\n            return Money.cmp(left.toString(), right);\n          }\n        }\n      }\n    } break;\n    case \"string\": {\n      if (typeof right === \"string\")\n        return left === right ? 0 : left < right ? -1 : 1;\n    } break;\n    case \"object\": {\n      if (Money.isMoney(left)) {\n        switch (typeof right) {\n          case \"number\":\n          case \"bigint\":\n            return Money.cmp(left, right.toString());\n          case \"object\": {\n            if (right === null) {\n              return 1;\n            } else if (Money.isMoney(right))\n              return Money.cmp(left, right);\n          }\n        }\n      } else if (isDate(left) && isDate(right)) {\n        const left_value = Number(left);\n        const right_value = Number(right);\n        return left_value !== right_value ? left_value < right_value ? -1 : 1 : 0;\n      } else if (isUInt8Array(left) && isUInt8Array(right)) {\n        const compareLength = Math.min(left.length, right.length);\n        for (let i = 0; i < compareLength; i++) {\n          if (left[i] !== right[i])\n            return left[i] < right[i] ? -1 : 1;\n        }\n        return left.length !== right.length ? left.length < right.length ? -1 : 1 : 0;\n      } else if (isTemporalInstant(left) && isTemporalInstant(right)) {\n        const left_value = left.epochMilliseconds;\n        const right_value = right.epochMilliseconds;\n        return left_value !== right_value ? left_value < right_value ? -1 : 1 : 0;\n      } else if (isTemporalPlainDate(left) && isTemporalPlainDate(right)) {\n        // eslint-disable-next-line @typescript-eslint/no-explicit-any -- we're not loading Temporal types when packaging @webhare/std separately\n        return (globalThis as any).Temporal.PlainDate.compare(left, right);\n      } else if (isTemporalZonedDateTime(left) && isTemporalZonedDateTime(right)) {\n        // eslint-disable-next-line @typescript-eslint/no-explicit-any -- we're not loading Temporal types when packaging @webhare/std separately\n        return (globalThis as any).Temporal.ZonedDateTime.compare(left, right);\n      }\n    } break;\n  }\n  throw new Error(`Cannot compare a ${stdTypeOf(left)} with a ${stdTypeOf(right)}`);\n}\n\n/** Returns all property keys of an object type whose value is comparable */\ntype ComparableFields<E extends object, K extends keyof E = keyof E> = K extends keyof E ? E[K] extends ComparableType ? K : never : never;\n\n/** Type of array elements for keys in `compareProperties` */\ntype ComparePropertiesArrayElement<E extends object> = ComparableFields<E> | [ComparableFields<E>, \"asc\" | \"desc\"];\n\n/** All possible types of object keys */\ntype AnyKey = string | symbol | number;\n\n/** Extracts the property name from an array element of CompareProperties */\ntype GetPropertyFromComparePropertiesArrayElement<T> = T extends AnyKey ? T : T extends [infer A, \"asc\" | \"desc\"] ? A : never;\n\n/** Get the list of properties referenced in the `compareProperties` keys list */\ntype GetPropertiesFromComparePropertiesKeys<K extends AnyKey | Array<AnyKey | [AnyKey, \"asc\" | \"desc\"]>> = K extends AnyKey ? K : K extends Array<infer T> ? GetPropertyFromComparePropertiesArrayElement<T> : never;\n\n/** Get all proper prefixes of a tuple, excluding the empty tuple and the source tuple */\n// eslint-disable-next-line @typescript-eslint/no-explicit-any\ntype TuplePrefixes<K extends AnyKey | [...any[]]> = K extends [any] ? never : K extends [infer A, ...infer B] ? [A] | [A, ...TuplePrefixes<B>] : never;\n\n/** Template for possible keys value (needed to make TS understand that a tuple prefix of the keys is also a valid keys list */\ntype PossibleFields<E extends object> = ComparableFields<E> | [ComparePropertiesArrayElement<E>, ...Array<ComparePropertiesArrayElement<E>>];\n\n\nexport function compareProperties<const K extends ComparableFields<E> | [...Array<ComparePropertiesArrayElement<E>>], E extends object = Record<GetPropertiesFromComparePropertiesKeys<K>, ComparableType>>(fields: K) {\n  const compareList = (Array.isArray(fields) ? fields : [fields]).map((field): [keyof E, 1 | -1] => Array.isArray(field) ? [field[0], field[1] === \"desc\" ? -1 : 1] : [field, 1]);\n  const retval: (lhs: E, rhs: E) => number = (a, b) => {\n    for (const [field, negate] of compareList) {\n      if (a[field] === undefined || b[field] === undefined)\n        throw new Error(`Property '${String(field)}' does not exist`);\n      const res = negate * compare(a[field] as ComparableType, b[field] as ComparableType);\n      if (res)\n        return res;\n    }\n    return 0;\n  };\n  return Object.assign(retval, {\n    partialCompare: <T extends TuplePrefixes<K> & PossibleFields<E>>(partialFields: T) => compareProperties<T, E>(partialFields)\n  });\n}\n","import { stdTypeOf } from \"./quacks\";\nimport { Money } from \"./money\";\n\nexport type WildcardTypes = \"?*\";\n\n/** Encode string for use in a regexp\n * @param text - Text to encode\n * @param options - Options for encoding\n * @param options.wildcards - Type of wildcards to encode (defaults to none)\n * @returns Encoded for safe use in a RegExp\n*/\nexport function escapeRegExp(text: string, options?: { wildcards?: WildcardTypes }): string {\n  let mask = text.replaceAll(/[/\\-\\\\^$*+?.()|[\\]{}]/g, '\\\\$&'); // $& means the whole matched string\n  if (options?.wildcards === \"?*\") {\n    mask = mask.replaceAll(\"\\\\*\", \".*\");\n    mask = mask.replaceAll(\"\\\\?\", \".\");\n  }\n  return mask;\n}\n\n/** Create a regular expression from one or more wildcard masks\n * @param masks - One or more masks with '?' and/or '*' wildcards\n * @returns Regular expression string\n*/\nexport function regExpFromWildcards(masks: string | string[], options?: { wildcards?: WildcardTypes; caseInsensitive?: boolean }): RegExp {\n  if (Array.isArray(masks) && masks.length === 0)\n    throw new Error(\"Empty mask list\");\n\n  const code = Array.isArray(masks)\n    ? `^(${masks.map(mask => escapeRegExp(mask, { wildcards: options?.wildcards || \"?*\" })).join('|')})$`\n    : `^${escapeRegExp(masks, { wildcards: options?.wildcards || \"?*\" })}$`;\n  return new RegExp(code, options?.caseInsensitive ? \"i\" : undefined);\n}\n\nfunction isHTMLUnrepresentableChar(curch: number) {\n  return (curch < 32 && curch !== 9 && curch !== 10 && curch !== 13)\n    || (curch >= 128 && curch <= 159);\n}\n\nfunction encodeEntities(str: string, html: boolean) {\n  let s = \"\";\n  for (const char of str) {\n    const curch = char.codePointAt(0);\n    if (curch === undefined || isHTMLUnrepresentableChar(curch))\n      continue;\n    if (curch >= 32 && curch < 128 && curch !== 38 && curch !== 60 && curch !== 62 && (html || curch !== 34 && curch !== 39)) {\n      s += String.fromCodePoint(curch);\n      continue;\n    }\n\n    switch (curch) {\n      case 10:\n        {\n          if (html) {\n            s += \"<br>\";\n            continue;\n          }\n          break;\n        }\n      case 13:\n        {\n          if (html)\n            continue;\n          break;\n        }\n      case 34:\n        {\n          s += \"&quot;\";\n          continue;\n        }\n      case 38:\n        {\n          s += \"&amp;\";\n          continue;\n        }\n      case 39:\n        {\n          s += \"&apos;\";\n          continue;\n        }\n      case 60:\n        {\n          s += \"&lt;\";\n          continue;\n        }\n      case 62:\n        {\n          s += \"&gt;\";\n          continue;\n        }\n    }\n\n    s += \"&#\" + curch + \";\";\n  }\n  return s;\n}\n\nfunction decodeEntities(str: string, html: boolean) {\n  if (html)\n    str = str.replace(/<br *\\/?>/g, \"\\n\");\n\n  str = str.replace(/&#(\\d+);/g, (_, dec) => String.fromCharCode(dec))\n    .replace(/&lt;/g, \"<\")\n    .replace(/&gt;/g, \">\")\n    .replace(/&quot;/g, '\"')\n    .replace(/&apos;/g, \"'\")\n    .replace(/&amp;/g, \"&\");\n\n  return str;\n}\n\nexport type StringEncodings = \"base64url\" | \"attribute\" | \"html\";\n\nexport function encodeString(str: string, encoding: StringEncodings): string {\n  if (encoding === \"base64url\")\n    return btoa(str).replaceAll(\"=\", \"\").replaceAll(\"+\", \"-\").replaceAll(\"/\", \"_\");\n  if (encoding === \"attribute\")\n    return encodeEntities(str, false);\n  if (encoding === \"html\")\n    return encodeEntities(str, true);\n\n  throw new Error(`Invalid encoding '${encoding}'`);\n}\n\nexport function decodeString(str: string, encoding: StringEncodings): string {\n  if (encoding === \"base64url\")\n    return atob(str.replaceAll(\"-\", \"+\").replaceAll(\"_\", \"/\"));\n  if (encoding === \"attribute\")\n    return decodeEntities(str, false);\n  if (encoding === \"html\")\n    return decodeEntities(str, true);\n\n  throw new Error(`Invalid encoding '${encoding}'`);\n}\n\ntype JSONReplacerArgument = ((this: unknown, key: string, value: unknown) => unknown) | undefined;\n\nexport interface StringifyOptions {\n  replacer?: JSONReplacerArgument;\n  space?: string | number;\n  stable?: boolean;\n  ///Encode with types (preserve Money, Date, BigInt). Needs std.parseTyped\n  typed?: boolean;\n  ///What to target: string (like JSON.stringify), script (escapes '/') or attribute (escapes '/' and applies attribute encoding)\n  target?: \"string\" | \"script\" | \"attribute\";\n}\n\n/** Improved JSON encoder\n * @param arg - Object to encode\n * @param options - Encoding options\n*/\nexport function stringify(arg: unknown, options?: StringifyOptions) {\n  const usereplacer: JSONReplacerArgument = options?.stable || options?.typed ? (function (this: unknown, key: string, value: unknown) {\n    if (options.typed) {\n      const origvalue = (this as Record<string, unknown>)[key]; //We can't use 'value' as .toJSON() will already have been invoked\n      const type = stdTypeOf(origvalue);\n      switch (type) {\n        case \"function\":\n          throw new Error(`Cannot stringify property '${key}' of type \"${type}'`);\n        case \"Date\":\n          value = { \"$stdType\": \"Date\", date: (origvalue as Date).toISOString() };\n          break;\n        case \"Money\":\n        case \"bigint\":\n        case \"Instant\":\n        case \"PlainDate\":\n        case \"PlainDateTime\":\n        case \"ZonedDateTime\":\n          value = { \"$stdType\": type, [type.toLowerCase()]: (origvalue as { toString: () => string }).toString() };\n          break;\n        case \"object\":\n          if (\"$stdType\" in (origvalue as { \"$stdType\": string }))\n            throw new Error(`Cannot encode objects with already embedded '$stdType's`);\n        //fallthrough\n      }\n    }\n    if (options.stable && value && typeof value === \"object\" && !Array.isArray(value))\n      value = Object.fromEntries(Object.entries(value).sort((lhs, rhs) => lhs < rhs ? -1 : lhs === rhs ? 0 : 1));\n    if (options.replacer)\n      value = options.replacer.call(this, key, value);\n    return value;\n  }) : options?.replacer ?? undefined;\n\n  let result = JSON.stringify(arg, usereplacer, options?.space);\n  if (options?.target && [\"script\", \"attribute\"].includes(options.target)) {\n    result = result.replaceAll(\"/\", \"\\\\/\");\n    if (options.target === \"attribute\")\n      result = encodeEntities(result, false);\n  }\n  return result;\n}\n\n/** Decode JSON with types (Generated using stringify with typed:true ) */\nexport function parseTyped(input: string) {\n  return JSON.parse(input, (key, value) => {\n    switch (value?.[\"$stdType\"]) {\n      case \"Money\":\n        return new Money(value.money);\n      case \"Date\":\n        return new Date(value.date);\n      case \"bigint\":\n      case \"BigInt\": //pre wh5.7 spelling\n        return BigInt(value.bigint as string);\n      case \"Instant\":\n      case \"PlainDate\":\n      case \"PlainDateTime\":\n      case \"ZonedDateTime\":\n        // eslint-disable-next-line @typescript-eslint/no-explicit-any -- we just assume/require you to have Temporal installed if you expect to receive/decode Temporal types. browsers should catch up eventually\n        if (!(globalThis as any).Temporal)\n          throw new Error(`Temporal is not available in this environment, cannot deserialize value of type Temporal.${value[\"$stdType\"]}. Load eg. @webhare/deps/temporal-polyfill to use Temporal types in browsers`);\n        // eslint-disable-next-line @typescript-eslint/no-explicit-any -- we just assume/require you to have Temporal installed if you expect to receive/decode Temporal types. browsers should catch up eventually\n        return (globalThis as any).Temporal[value[\"$stdType\"]].from(value[value[\"$stdType\"].toLowerCase()]);\n      case undefined:\n        return value;\n      default:\n        throw new Error(`Unrecognized type '${value[\"$stdType\"]}'`);\n    }\n  });\n}\n\n/** Generate a slug from a (suggested) (file)name\n * @param text - Text to convert\n * @param options - Options for slugification\n * @param options.separator - Separator to use between words (defaults to '-')\n * @param options.keep - Set of characters to keep in addition to a-z0-9\n * @returns Slugified text or null if we couldn't generate anything reeadable\n */\nexport function slugify(text: string, { separator = \"-\", keep = \"\" }: {\n  separator?: string;\n  keep?: string;\n} = {}): string | null {\n  //This function mixes HS getSafeName with a few more modern approaches\n  const keepclass = `[^a-z0-9${escapeRegExp(keep)}]`;\n  text = text\n    .normalize('NFD')                   // split an accented letter in the base letter and the acent\n    // eslint-disable-next-line no-control-regex\n    .replaceAll(/[\\u0000-\\u001F]/g, '')\n    .replaceAll(/[\\u0300-\\u036f]/g, '')   // remove all previously split accents\n    .replaceAll(/ß/g, 'ss')               // german ss\n    .toLowerCase()\n    .replace(new RegExp(`^${keepclass}+`), \"\") //replace bad characters at the start\n    .replace(new RegExp(`${keepclass}+$`), \"\") //.. and end\n    .replaceAll(new RegExp(`${keepclass}+`, \"g\"), separator); // replace all non alphanumeric/space with a single dash\n\n  return text || null; //we return 'null' on purpose so callers realize we won't necessarily give them a string!\n}\n\n/** Check if an email address is valid in modern times (an emailcheck much closer to what a browser would do, with additional sanity checks. No attempt to allow all legacy styles supported by the RFCs but 99.9%+ sure to be an error if seen submitted in a form\n    @returns True if the email address appears to be a well-formed email address to a non-greybeard\n*/\nexport function isValidEmail(email: string): boolean {\n  if (email.length > 254) //TODO count bytes instead of characters\n    return false;\n\n  return /^[a-zA-Z0-9.!#$%&'*+/=?^_`{|}~-]{1,64}@[a-zA-Z0-9-]+(?:\\.[a-zA-Z0-9-]+)*(?:\\.[a-zA-Z0-9-]{2,})$/.test(email);\n}\n\n/** Joins a path to a URL, eliminating any double slash (similar to NodeJS path.join)\n * @param baseurl - URL to join to\n * @param path - Path to join. May not contain '.' or '..' segments in the path\n * @returns Joined URL, with a single slash between both parts\n*/\nexport function joinURL(baseurl: string, path: string) {\n  const pathtoks = path.match(/([^?#]*)(.*)/)!; //this will always return something\n  if (pathtoks[1].match(/(\\/\\/)|:/))\n    throw new Error(`Invalid path to merge with: ${path}`);\n  //look for any . or .. path segments\n  if ((\"/\" + path + \"/\").match(/\\/\\.\\.?\\//))\n    throw new Error(`Invalid path to merge with: ${path}`);\n\n  if (baseurl.endsWith(\"/\"))\n    baseurl = baseurl.substring(0, baseurl.length - 1);\n  return baseurl + (path.startsWith(\"/\") ? path : \"/\" + path);\n}\n\nfunction unpackUrl(inUrl: string) {\n  const retval = {\n    inUrl,\n    scheme: \"\",\n    schemeSpecificPart: \"\",\n    user: \"\",\n    password: \"\",\n    host: \"\",\n    port: 0,\n    urlPathSlash: false,\n    urlPath: \"\",\n    secure: false,\n    specifiedPort: false,\n    isDefaultPort: false,\n    origin: \"\",\n  };\n  let missingSlash = false;\n  const firstColon = inUrl.indexOf(\":\");\n  if (firstColon < 0) {\n    retval.scheme = inUrl;\n    return retval;\n  }\n\n  retval.scheme = decodeURIComponent(inUrl.substring(0, firstColon)).toLowerCase();\n  inUrl = inUrl.substring(firstColon + 1);\n  retval.schemeSpecificPart = inUrl;\n\n  if (!retval.schemeSpecificPart.startsWith(\"//\")) //not an internet host format\n    return retval;\n\n  inUrl = inUrl.substring(2);\n\n  //Find the first slash, it terminates the user..port part\n  let firstSlash = inUrl.indexOf(\"/\");\n  if (firstSlash < 0)\n    firstSlash = inUrl.length;\n\n  const firstAt = inUrl.indexOf(\"@\");\n  if (firstAt >= 0 && firstAt < firstSlash) {\n    //Username and possible password are present\n    const pwdColon = inUrl.indexOf(\":\");\n    if (pwdColon >= 0 && pwdColon < firstAt) {\n      retval.user = decodeURIComponent(inUrl.substring(0, pwdColon));\n      retval.password = decodeURIComponent(inUrl.substring(pwdColon + 1, firstAt));\n    } else {\n      retval.user = decodeURIComponent(inUrl.substring(0, firstAt));\n    }\n    inUrl = inUrl.substring(firstAt + 1);\n    firstSlash = inUrl.indexOf(\"/\");\n    if (firstSlash < 0)\n      firstSlash = inUrl.length;\n  }\n\n  //Common mistake, at least in http* urls, specifying a ? right after the URL (ADDME perhaps this shoud apply for all URLs? check rfc)\n  if (retval.scheme === \"http\" || retval.scheme === \"https\") {\n    const firstQuestionMark = inUrl.indexOf(\"?\");\n    if (firstQuestionMark >= 0 && firstQuestionMark < firstSlash) {\n      firstSlash = firstQuestionMark;\n      missingSlash = true;\n    }\n  }\n\n  let closingBracket = 0;\n  if (inUrl.startsWith(\"[\")) //RFC3986 IP-literal\n    closingBracket = inUrl.indexOf(\"]\");\n\n  if (closingBracket > 0) { //Indeed an IP-literal\n    retval.host = decodeURIComponent(inUrl.substring(1, closingBracket));\n    if (inUrl[closingBracket + 1] === ':') //port follows\n      retval.port = parseInt(decodeURIComponent(inUrl.substring(closingBracket + 2, firstSlash - 1)), 0) || 0;\n    else\n      retval.port = getSchemeDefaultPort(retval.scheme);\n  } else {\n    const portColon = inUrl.indexOf(\":\");\n    if (portColon >= 0 && portColon < firstSlash) {\n      retval.host = decodeURIComponent(inUrl.substring(0, portColon));\n      retval.port = parseInt(decodeURIComponent(inUrl.substring(portColon + 1, firstSlash)), 0) || 0;\n      retval.specifiedPort = true;\n      retval.isDefaultPort = retval.port === getSchemeDefaultPort(retval.scheme);\n    } else {\n      retval.host = decodeURIComponent(inUrl.substring(0, firstSlash));\n      retval.port = getSchemeDefaultPort(retval.scheme);\n      retval.isDefaultPort = true;\n    }\n  }\n\n  retval.urlPathSlash = !(missingSlash || firstSlash === inUrl.length);\n  retval.urlPath = inUrl.substring(firstSlash + (missingSlash ? 0 : 1));\n  retval.secure = [\"aas\", \"ftps\", \"https\", \"imaps\", \"ldaps\", \"shttp\", \"sips\", \"ssh\", \"sftp\"].includes(retval.scheme);\n  if (retval.host)\n    retval.origin = retval.scheme + \"://\" + retval.host + (retval.isDefaultPort ? \"\" : \":\" + retval.port);\n  return retval;\n}\n\nfunction getSchemeDefaultPort(scheme: string) {\n  switch (scheme) {//port numbers from RFC1738\n    case \"ftp\": return 21;\n    case \"http\": return 80;\n    case \"ws\": return 80;\n    case \"gopher\": return 70;\n    case \"nntp\": return 119;\n    case \"telnet\": return 23;\n    case \"wais\": return 210;\n    case \"prospero\": return 1525;\n    //well known protocols\n    case \"https\": return 443;\n    case \"wss\": return 443;\n    case \"ldap\": return 389;\n    case \"ldaps\": return 636;\n    case \"smtp\": return 25;\n    default: return 0;\n  }\n}\n\n\n/** Test whether an URL is a valid url. This function verifies that an URL looks\n * like an URL. If the scheme is recognized, more  stringent checks are performed\n * @returns True if the url appears to be a well-formed url\n */\nexport function isValidUrl(url: string): boolean {\n  // test: no control characters or spaces (0x00 - 0x20)\n  // test: starts with 'scheme' ':' anychar +\n  // test: scheme only has letters, numbers, '-', '.' or '+' (and is not empty)\n  // Unpack url\n  // test: for http/https schema, hostname is not empty and port is in range 1-65535\n\n  // eslint-disable-next-line no-control-regex\n  if (!url.match(/^[-.+a-zA-Z0-9]+:[^\\x00-\\x20]+$/))\n    return false;\n  const unpacked = unpackUrl(url);\n  if (unpacked.scheme !== \"http\" && unpacked.scheme !== \"https\")\n    return true;\n  if (unpacked.port < 1 || unpacked.port > 65535 || !unpacked.host)\n    return false;\n  return true;\n}\n\n/** Uppercases a string using the C-locale (so only the ASCII characters a-z are uppercased)\n * @param str - String to uppercase\n * @returns Uppercased string\n */\nexport function toCLocaleUppercase(str: string) {\n  return str.replaceAll(/[a-z]+/g, part => part.toUpperCase());\n}\n\n/** Lowercases a string using the C-locale (so only the ASCII characters A-Z are lowercased)\n * @param str - String to lowercase\n * @returns Lowercased string\n */\nexport function toCLocaleLowercase(str: string) {\n  return str.replaceAll(/[A-Z]+/g, part => part.toLowerCase());\n}\n\n/** Calculate the levenshtein distance between two strings */\nexport function levenshteinDistance(a: string, b: string): number {\n  /* calculateLevenshteinDistance is adopted from https://github.com/gustf/js-levenshtein/blob/master/index.js, licensed MIT © Gustaf Andersson\n    Picked over fastest-levenshtein because a permanent 256KB memory area for a bit more speed doesn't seem worth it\n  */\n  function _min(d0: number, d1: number, d2: number, bx: number, ay: number) {\n    return d0 < d1 || d2 < d1\n      ? d0 > d2\n        ? d2 + 1\n        : d0 + 1\n      : bx === ay\n        ? d1\n        : d1 + 1;\n  }\n\n  if (a === b) {\n    return 0;\n  }\n\n  if (a.length > b.length) {\n    const tmp = a;\n    a = b;\n    b = tmp;\n  }\n\n  let la = a.length;\n  let lb = b.length;\n\n  while (la > 0 && (a.charCodeAt(la - 1) === b.charCodeAt(lb - 1))) {\n    la--;\n    lb--;\n  }\n\n  let offset = 0;\n\n  while (offset < la && (a.charCodeAt(offset) === b.charCodeAt(offset))) {\n    offset++;\n  }\n\n  la -= offset;\n  lb -= offset;\n\n  if (la === 0 || lb < 3) {\n    return lb;\n  }\n\n  let x = 0;\n  let y, d0, d1, d2, d3, dd = 0, dy, ay, bx0, bx1, bx2, bx3;\n\n  const vector: number[] = [];\n\n  for (y = 0; y < la; y++) {\n    vector.push(y + 1);\n    vector.push(a.charCodeAt(offset + y));\n  }\n\n  const len = vector.length - 1;\n\n  for (; x < lb - 3;) {\n    bx0 = b.charCodeAt(offset + (d0 = x));\n    bx1 = b.charCodeAt(offset + (d1 = x + 1));\n    bx2 = b.charCodeAt(offset + (d2 = x + 2));\n    bx3 = b.charCodeAt(offset + (d3 = x + 3));\n    dd = (x += 4);\n    for (y = 0; y < len; y += 2) {\n      dy = vector[y];\n      ay = vector[y + 1];\n      d0 = _min(dy, d0, d1, bx0, ay);\n      d1 = _min(d0, d1, d2, bx1, ay);\n      d2 = _min(d1, d2, d3, bx2, ay);\n      dd = _min(d2, d3, dd, bx3, ay);\n      vector[y] = dd;\n      d3 = d2;\n      d2 = d1;\n      d1 = d0;\n      d0 = dy;\n    }\n  }\n\n  for (; x < lb;) {\n    bx0 = b.charCodeAt(offset + (d0 = x));\n    dd = ++x;\n    for (y = 0; y < len; y += 2) {\n      dy = vector[y];\n      vector[y] = dd = _min(dy, d0, dd, bx0, vector[y + 1]);\n      d0 = dy;\n    }\n  }\n\n  return dd;\n}\n\n/** Return the length of the string in bytes when UTF-8 encoded */\nexport function getUTF8Length(str: string) {\n  return new TextEncoder().encode(str).length;\n}\n\n/** Truncate the string so it's at most len UTF-8 bytes long */\nexport function limitUTF8Length(str: string, len: number) {\n  // TextEncoder.encodeInto writes into the given array until there's no more room to add more valid UTF-8 sequences from the\n  // source. We can use this property to limit the number of bytes by creating an array that has room for at most the given\n  // number of bytes.\n  const utf8array = new Uint8Array(len);\n  // This function returns the actual number of bytes written to the array\n  const { written } = new TextEncoder().encodeInto(str, utf8array);\n  // Decode only the actually written bytes\n  return new TextDecoder().decode(utf8array.subarray(0, written));\n}\n","/** Test whether a string looks like a valid random id */\nexport function isLikeRandomId(input: string): boolean { //TODO add encoding parameter/bytes support ?\n  return Boolean(input.match(/^[A-Za-z0-9_-]{22}$/));\n}\n\n/** Test whether a string is a valid hex UUID\n * @param input - uuid in hex format to verify. you may need to lowercase the uuid first\n * @param format - optional, if set to \"v4\" it will only accept UUIDv4 format\n*/\nexport function isValidUUID(input: string, format?: \"v4\"): boolean {\n  //we'll require you to lowercase the uuid first to increase the chance that you'll actually be storing/matching it in lowercase too\n\n  if (format === \"v4\")\n    return Boolean(input.match(/^[0-9a-f]{8}-[0-9a-f]{4}-4[0-9a-f]{3}-[89ab][0-9a-f]{3}-[0-9a-f]{12}$/));\n  else if (format !== undefined)\n    throw new Error(`Unsupported format '${format}'`);\n  else\n    return Boolean(input.match(/^[0-9a-f]{8}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{12}$/));\n}\n\n/** Generate a configurable random id (the default settings, base64url and 16 bytes, match HareScript's GenerateUFS128BitId)\n * @param encoding - Encoding to use, base64url, hex or uuidv4. Default is base64url\n * @param bytes - Number of bytes to generate. Default is 16\n * @returns An encoded ID. With the default settings it will be a url (and filename) safe string of 22 characters in length\n */\nexport function generateRandomId(encoding: \"base64url\" | \"hex\" | \"uuidv4\" = \"base64url\", bytes: number = 16): string {\n  if (encoding === \"uuidv4\") {\n    if (bytes !== 16)\n      throw new Error(\"UUIDv4 encoding only supports 16 bytes\");\n\n    if (crypto.randomUUID) //NOTE: not available in non-secure contexts so the fallback can never go away\n      return crypto.randomUUID();\n  }\n\n  const u8array = new Uint8Array(bytes);\n  crypto.getRandomValues(u8array);\n\n  if (encoding === \"uuidv4\") {\n    u8array[6] = (u8array[6] & 0x0f) | 0x40;\n    u8array[8] = (u8array[8] & 0x3f) | 0x80;\n    return [...u8array.values()].map(x => x.toString(16).padStart(2, \"0\")).join(\"\").replace(/^(.{8})(.{4})(.{4})(.{4})(.{12})$/, \"$1-$2-$3-$4-$5\");\n  }\n\n  if (encoding === \"base64url\")\n    return btoa(String.fromCharCode.apply(null, u8array as unknown as number[])).replaceAll(\"=\", \"\").replaceAll(\"+\", \"-\").replaceAll(\"/\", \"_\");\n\n  if (encoding === \"hex\")\n    return [...u8array.values()].map(x => x.toString(16).padStart(2, \"0\")).join(\"\");\n\n  throw new Error(`Invalid encoding '${encoding}'`);\n}\n","/* eslint-disable @typescript-eslint/no-explicit-any -- Cannot specify this without using any */\n\nexport interface EmplaceHandler<ValueType> {\n  insert?: () => ValueType;\n  update?: (current: ValueType) => ValueType;\n}\n\nfunction binarySearchImpl<A, K>(compareFn: (element: A, key: K) => number, contents: A[], searchfor: K, upper_bound: boolean): { present: boolean; index: number } {\n  let first = 0;\n  let len = contents.length;\n  let present = false;\n\n  const cmpbound = upper_bound ? 1 : 0;\n  let unsorted_cmp = 0; // if this is non-0 and cmp is this value, we have an unsorted list\n  while (len > 0) {\n    const half = Math.floor(len / 2);\n    const middle = first + half;\n    const cmp = compareFn(contents[middle], searchfor);\n    if (cmp === 0) {\n      present = true;\n      unsorted_cmp = upper_bound ? -1 : 1;\n    } else if (cmp === unsorted_cmp)\n      throw new Error(`The provided array was not properly sorted!`);\n\n    if (cmp < cmpbound) {\n      first = middle + 1;\n      len -= half;\n      --len;\n    } else {\n      len = half;\n    }\n  }\n  return { present, index: first };\n}\n\n//emplace is based on https://github.com/tc39/proposal-upsert (a Map.prototype.emplace)\n/** Place a value into a Map\n * @param map - The map to place the value into\n * @param key - The key to add/replace\n * @param handler - Callbacks for inserting or updating the value\n * @returns The value that was placed into the map.\n * @throws If the key is not found and no insert handler is provided\n */\nexport function emplace<T extends Map<any, any> | WeakMap<any, any>>(\n  map: T,\n  key: T extends Map<infer K, any> ? K : T extends WeakMap<infer K, any> ? K : never,\n  handler?: T extends Map<any, infer V> ? EmplaceHandler<V> : T extends WeakMap<any, infer V> ? EmplaceHandler<V> : never):\n  T extends Map<any, infer V> ? V : T extends WeakMap<any, infer V> ? V : never {\n  let current = map.get(key);\n  if (current !== undefined) {\n    if (handler?.update) {\n      current = handler.update(current);\n      map.set(key, current);\n    }\n    return current;\n  }\n\n  if (!handler?.insert)\n    throw new Error(\"Key not found and no insert handler provided\");\n\n  const setvalue = handler.insert();\n  map.set(key, setvalue);\n  return setvalue;\n}\n\n// the `| keyof X` is needed to be able to use a `keyof X` type as key parameter in DistributedPick or DistributedOmit in generics\nexport type DistributedKeys<X extends object> = (X extends object ? keyof X : never) | keyof X;\n\n/** Applies Pick to all types in a union. Allows all keys that are present in any object in the union. Warning: You might not be\n    able to use all keys of the union if TypeScript has narrowed the union to a specific type. Eg:\n```typescript\ntype A = { x: number; t: \"a\"; a: number } | { x: number; t: \"b\"; b: number };\nconst a: A = { t: \"a\", a: 1 };\nconst b = pick(a, [\"t\", \"a\", \"b\"]); // No overload matches this call. <snip> Type '\"b\"' is not assignable to type '\"a\" | \"t\" | \"d\"'.\n```\n    @typeParam T - Type of the supplied object\n    @typeParam K - Type of the property keys to return\n    @returns Type with only the specified keys (distributed over the union if present)\n*/\nexport type DistributedPick<X extends object, Y extends DistributedKeys<X>> = X extends object ? Pick<X, keyof X & Y> : never;\n\n/** Applies Omit to all types in a union. Allows all keys that are not present in any object in the union. Warning: You might not be\n    able to use all keys of the union if TypeScript has narrowed the union to a specific type. Eg:\n```typescript\ntype A = { x: number; t: \"a\"; a: number } | { x: number; t: \"b\"; b: number };\nconst a: A = { t: \"a\", a: 1 };\nconst b = omit(a, [\"b\"]); // No overload matches this call. <snip> Type '\"b\"' is not assignable to type '\"a\" | \"t\" | \"d\"'.\n```\n    @typeParam T - Type of the supplied object\n    @typeParam K - Type of the property keys to leave out\n    @returns Type with only the specified keys left out (distributed over the union if present)\n*/\nexport type DistributedOmit<X extends object, Y extends DistributedKeys<X>> = X extends object ? Omit<X, keyof X & Y> : never;\n\n/** Returns an object with a selection of properties\n    @typeParam T - Type of the supplied object\n    @typeParam K - Type of the property keys to return\n    @param obj - Object to pick properties out of\n    @param keys - Names of the properties to pick\n    @returns Resulting object\n*/\nexport function pick<T extends object, K extends string & NoInfer<DistributedKeys<T>>>(obj: T, keys: readonly K[]): DistributedPick<T, K>;\n\n/** Returns an array with a selection of properties\n    @typeParam T - Type of the supplied array\n    @typeParam K - Type of the property keys to return\n    @param arr - Array to pick properties out of\n    @param keys - Names of the properties to pick\n    @returns Resulting array\n*/\nexport function pick<T extends object, K extends string & NoInfer<DistributedKeys<T>>>(arr: T[], keys: readonly K[]): Array<DistributedPick<T, K>>;\n\nexport function pick<T extends object, K extends string & NoInfer<DistributedKeys<T>>>(value: T | T[], keys: readonly K[]): DistributedPick<T, K> | Array<DistributedPick<T, K>> {\n  if (Array.isArray(value))\n    return value.map((elt: T) => pick(elt, keys));\n  const ret = {} as T;\n  keys.forEach((key: K) => {\n    if (key in value)\n      ret[key] = value[key];\n  });\n  return ret as object as DistributedPick<T, K>;\n}\n\n/** Returns an object with a selection of properties left out\n    @typeParam T - Type of the supplied object\n    @typeParam K - Type of the property keys to leave out\n    @param obj - Object to leave properties out of\n    @param keys - Names of the properties to remove\n    @returns Resulting object\n*/\nexport function omit<T extends object, K extends string & NoInfer<DistributedKeys<T>>>(obj: T, keys: readonly K[]): DistributedOmit<T, K>;\n\n/** Returns an array with a selection of properties left out\n    @typeParam T - Type of the supplied array\n    @typeParam K - Type of the property keys to leave out\n    @param arr - Array to leave properties out of\n    @param keys - Names of the properties to leave out\n    @returns Resulting array\n*/\nexport function omit<T extends object, K extends string & NoInfer<DistributedKeys<T>>>(arr: T[], keys: readonly K[]): Array<DistributedOmit<T, K>>;\n\nexport function omit<T extends object, K extends string & NoInfer<DistributedKeys<T>>>(value: T | T[], keys: readonly K[]): DistributedOmit<T, K> | Array<DistributedOmit<T, K>> {\n  if (Array.isArray(value))\n    return value.map((elt: T) => omit(elt, keys));\n  const ret = {} as T;\n  for (const [key, val] of Object.entries(value)) {\n    if (!keys.includes(key as K))\n      ret[key as K] = val;\n  }\n  return ret as object as DistributedOmit<T, K>;\n}\n\n/** Shuffle an array in-place\n * @param array - Array to shuffle\n * @returns The shuffled array\n*/\nexport function shuffle<T>(array: T[]): T[] {\n  for (let i = array.length - 1; i > 0; i--) {\n    const j = Math.floor(Math.random() * (i + 1));\n    [array[i], array[j]] = [array[j], array[i]];\n  }\n  return array;\n}\n\n/** Returns (and TypeScript-asserts) that the specified function is truthy. Handy for adjusting the type of an array when\n * filtering out falsy values\n * @example\n * const myarray: Array&lt;number | null&gt; = [0, null];\n * const filtered: number[] = myarray.filter(isTruthy);\n **/\nexport function isTruthy<T>(a: T): a is (T & {}) {\n  return Boolean(a);\n}\n\n/** Append to array, without overflowing the stack (eg V8 overflows at more than 32K entries)\n * @param array - Array to append to\n * @param values - Values to append\n*/\nexport function appendToArray<T>(array: T[], values: ReadonlyArray<NoInfer<T>>): void {\n  if (values.length < 1000)\n    array.push(...values); //push should be safe enough\n  else for (const value of values) //performance wise this appears just as fast as tricks with pushing blocks of slices\n    array.push(value);\n\n  //not returning the original array to make it clear we're not creating a new one\n}\n\ntype TypedEntriesInternal<T extends object, K extends keyof T> = K extends keyof T ? [K, T[K]] : never;\n\n/** Returns a union of [ key, value ] types for an object */\nexport type TypedEntries<T extends object> = T extends object ? TypedEntriesInternal<T, keyof T & string> : never;\n\n/** Returns an array of key/values of the enumerable own properties of an object, with exact types. Use when\n * no extra properties are expected on the object. Exists because Object.entries returns always string as key type.\n * @param obj - Object that contains the properties and methods\n */\nexport function typedEntries<T extends object>(obj: T): Array<TypedEntries<T>> {\n  return Object.entries(obj) as Array<TypedEntries<T>>;\n}\n\n/** Returns an array of keys of the enumerable own properties of an object, with exact types. Use when\n * no extra properties are expected on the object. Exists because Object.keys returns always string as key type.\n * @param obj - Object that contains the properties and methods\n */\nexport function typedKeys<T extends object>(obj: T): Array<keyof T & string> {\n  return Object.keys(obj) as Array<keyof T & string>;\n}\n\n/** Given a union of [key, value] types, returns the object constructed from those pairs. Exists because\n *  Object.fromEntries does not take the type of the keys and values into account.\n * @typeParam T - Union of [key, value] pairs\n */\nexport type TypedFromEntries<T extends [string, unknown]> = { [C in T as C[0]]: C[1] };\n\n/** Returns an object from an iterable of [key, value] pairs, with exact types. Exists because Object.fromEntries\n * doesn't take the type of the keys and values into account.\n * @param entries - Iterable of [key, value] pairs\n * @returns An object with the same keys and values as the input\n */\nexport function typedFromEntries<T extends [string, unknown]>(entries: Iterable<T>): TypedFromEntries<T> {\n  return Object.fromEntries(entries) as TypedFromEntries<T>;\n}\n\n\nexport class SortedMultiSet<V> {\n  private contents: V[] = [];\n\n  constructor(private compareFn: (lhs: V, rhs: V) => number, contents?: V[]) {\n    if (contents)\n      this.addMultiple(contents);\n  }\n\n  get size() {\n    return this.contents.length;\n  }\n\n  at(index: number): V | undefined {\n    return this.contents.at(index);\n  }\n\n  add(value: V): number {\n    const pos = this.upperBound(value);\n    this.contents.splice(pos, 0, value);\n    return pos;\n  }\n\n  addMultiple(values: V[]): void {\n    for (const value of values) {\n      this.add(value);\n    }\n  }\n\n  delete(key: V): void {\n    const [start, limit] = this.range(key);\n    this.contents.splice(start, limit - start);\n  }\n\n  clear() {\n    this.contents.splice(0, this.contents.length);\n  }\n\n  lowerBound(key: Readonly<V>): { present: boolean; index: number } {\n    return binarySearchImpl(this.compareFn, this.contents, key, false);\n  }\n\n  upperBound(key: Readonly<V>): number {\n    return binarySearchImpl(this.compareFn, this.contents, key, true).index;\n  }\n\n  range(key: Readonly<V>): [number, number] {\n    return [this.lowerBound(key).index, this.upperBound(key)];\n  }\n\n  slice(start: number, limit: number): V[] {\n    return this.contents.slice(start, limit);\n  }\n\n  *#sliceIterator(start: number, limit: number): Generator<V, void> {\n    for (let idx = start; idx < limit; ++idx)\n      yield this.contents[idx];\n  }\n\n  sliceRange(key: Readonly<V>): V[] {\n    const [start, limit] = this.range(key);\n    return this.slice(start, limit);\n  }\n\n  rangeIterator(key: Readonly<V>): Iterable<V> {\n    const [start, limit] = this.range(key);\n    return this.#sliceIterator(start, limit);\n  }\n}\n\nexport class SortedMultiMap<K, V> {\n  private compareFn;\n  private contents: Array<[K, V]> = [];\n\n  constructor(compareFn: (lhs: K, rhs: K) => number, contents?: Array<[K, V]>) {\n    this.compareFn = (element: [K, V], key: K) => compareFn(element[0], key);\n    if (contents)\n      this.addMultiple(contents);\n  }\n\n  get size() {\n    return this.contents.length;\n  }\n\n  at(index: number): [K, V] | undefined {\n    return this.contents.at(index);\n  }\n\n  add(key: K, value: V): number {\n    const pos = this.upperBound(key);\n    this.contents.splice(pos, 0, [key, value]);\n    return pos;\n  }\n\n  addMultiple(values: Array<[K, V]>) {\n    for (const [key, value] of values) {\n      this.add(key, value);\n    }\n  }\n\n  delete(key: K): void {\n    const [start, limit] = this.range(key);\n    this.contents.splice(start, limit - start);\n  }\n\n  clear() {\n    this.contents.splice(0, this.contents.length);\n  }\n\n  lowerBound(key: Readonly<K>): { present: boolean; index: number } {\n    return binarySearchImpl(this.compareFn, this.contents, key, false);\n  }\n\n  upperBound(key: Readonly<K>): number {\n    return binarySearchImpl(this.compareFn, this.contents, key, true).index;\n  }\n\n  range(key: Readonly<K>): [number, number] {\n    return [this.lowerBound(key).index, this.upperBound(key)];\n  }\n\n  slice(start: number, limit: number): Array<[K, V]> {\n    return this.contents.slice(start, limit);\n  }\n\n  *#sliceIterator(start: number, limit: number): Generator<[K, V], void> {\n    for (let idx = start; idx < limit; ++idx)\n      yield this.contents[idx];\n  }\n\n  sliceRange(key: Readonly<K>): Array<[K, V]> {\n    const [start, limit] = this.range(key);\n    return this.slice(start, limit);\n  }\n\n  rangeIterator(key: Readonly<K>): Iterable<[K, V]> {\n    const [start, limit] = this.range(key);\n    return this.#sliceIterator(start, limit);\n  }\n}\n","function propagateAbort(signal: AbortSignal, cb: ((reason: typeof signal[\"reason\"]) => void) | AbortController) {\n  if (signal.aborted) {\n    if (\"abort\" in cb)\n      cb.abort(signal.reason);\n    else\n      cb(signal.reason);\n  } else signal.addEventListener(\"abort\", () => {\n    if (\"abort\" in cb)\n      cb.abort(signal.reason);\n    else\n      cb(signal.reason);\n  });\n}\n\nexport type AbortSignals = Iterable<AbortSignal | Promise<AbortSignal> | undefined | null> | AbortSignal | Promise<AbortSignal> | null | undefined;\n\n/** Calls the callback when any of the passed abort signals is (or already was) aborted. Use\n * this instead of addEventListener when it is possible that any of the abortsignals was already aborted.\n*/\nexport function whenAborted(signals: AbortSignals, cb: ((reason: AbortSignal[\"reason\"]) => void) | AbortController): void {\n  propagateAbort(combineAbortSignals(signals), cb);\n}\n\n/** Combines abort signals (or promises for abort signals) into one abort signal. More generic version of AbortSignal.any() */\nexport function combineAbortSignals(signals: AbortSignals): AbortSignal {\n  if (!signals || !(Symbol.iterator in signals))\n    signals = [signals];\n  const abortController = new AbortController;\n  for (const signal of signals) {\n    if (signal) {\n      if (\"then\" in signal) {\n        // Convert promise errors to abort\n        void signal.then(s => propagateAbort(s, abortController), e => abortController.abort(e));\n      } else\n        propagateAbort(signal, abortController);\n    }\n  }\n  return abortController.signal;\n}\n","const unlock = Symbol(\"unlock\");\n\n/** A mutex which is only visible in the current JavaScript VM\n\n    @example\n```ts\nconst mutex = new LocalMutex;\n\nusing lock = await mutex.lock();\nlock.release();\n\n// If you'll be relying on automatic release\nusing lock = await mutex.lock();\nvoid(lock);\n```\n\n*/\nexport class LocalMutex {\n  private locked = false;\n  private unlockList = new Array<() => void>;\n\n  /** Lock this mutex\n   * @returns Lock object. Use with `using` to ensure automatic unlock\n   */\n  async lock() {\n    if (this.locked) {\n      const defer = Promise.withResolvers<void>();\n      this.unlockList.push(() => defer.resolve());\n      await defer.promise;\n    } else {\n      this.locked = true;\n    }\n    return new LocalLock(this);\n  }\n\n  [unlock](): void {\n    if (this.unlockList.length > 0)\n      (this.unlockList.shift()!)(); //we're fair... unlock longest waiter\n    else\n      this.locked = false;\n  }\n}\n\n/** Instance of a LocalMutex lock */\nclass LocalLock implements Disposable {\n  private mutex: LocalMutex | null;\n\n  constructor(mutex: LocalMutex) {\n    this.mutex = mutex;\n  }\n  /** Explicitly release the lock\n   * @throws Error if the lock is already released\n  */\n  release() {\n    if (!this.mutex)\n      throw new Error(\"Lock already released\");\n\n    this.mutex[unlock]();\n    this.mutex = null;\n  }\n  [Symbol.dispose]() {\n    this.mutex?.[unlock]();\n  }\n}\n\nexport type { LocalLock };\n","// This gets TypeScript to refer to us by our @webhare/... name in auto imports:\ndeclare module \"@webhare/std\" {\n}\n\n// Adding APIs may also require an update to https://www.webhare.dev/manuals/typescript/harescript-conversion-guide/\nexport { sleep, createDeferred, wrapInTimeout, serialize, wrapSerialized, type DeferredPromise, type SerializeOptions } from \"./promises\";\n\nexport { compare, compareProperties, nameToCamelCase, nameToSnakeCase, toSnakeCase, toCamelCase } from \"./types\";\nexport type { ToSnakeCase, ToCamelCase, ComparableType, MaybePromise } from \"./types\";\n\nexport { encodeString, decodeString, escapeRegExp, regExpFromWildcards, stringify, parseTyped, slugify, isValidEmail, isValidUrl, joinURL, type StringEncodings, toCLocaleLowercase, toCLocaleUppercase, levenshteinDistance, getUTF8Length, limitUTF8Length } from \"./strings\";\nexport { generateRandomId, isLikeRandomId, isValidUUID } from \"./platformbased\";\n\nexport { shuffle, emplace, pick, omit, isTruthy, appendToArray, typedEntries, typedFromEntries, typedKeys, SortedMultiSet, SortedMultiMap, type TypedEntries, type TypedFromEntries } from \"./collections\";\nexport type { EmplaceHandler, DistributedKeys, DistributedOmit, DistributedPick } from \"./collections\";\n\nexport { Money, type MoneyRoundingMode, type MoneyFormatOptions } from \"./money\";\n\nexport { addDuration, parseDuration, subtractDuration, convertWaitPeriodToDate, isValidDate, isValidTime, convertFlexibleInstantToDate, formatDateTime } from \"./datetime\";\nexport type { Duration, WaitPeriod, FlexibleInstant } from \"./datetime\";\n\nexport { stdTypeOf, isDate, isBlob, isFile, isError, isPromise, isMoney, isTemporalInstant, isTemporalPlainDate, isTemporalPlainDateTime, isTemporalZonedDateTime } from \"./quacks\";\n\nexport { combineAbortSignals, whenAborted } from \"./utils\";\n\nexport { LocalMutex, type LocalLock } from \"./localmutex\";\n\n/** Throw an error with the specified message. This function allows you to throw inside expressions\n * @param err - The error message\n * @returns This function never returns\n * @throws An error with the specified message\n */\nexport function throwError(err: string): never {\n  //TODO remove ourselves from the stack ?\n  throw new Error(err);\n}\n","import { debugFlags, registerDebugConfigChangedCallback } from \"./envbackend\";\nimport { generateRandomId } from \"@webhare/std\";\n\nlet hookedfetch = false;\n\nfunction sanitizeBody(body: unknown) {\n  if (body instanceof ArrayBuffer) {\n    const view = new Uint8Array(body.slice(0, 5000)); //show the bytes but filter unprintable as '.'\n    return Array.from(view).map(v => v >= 32 && v < 127 ? String.fromCharCode(v) : \".\").join(\"\");\n  }\n  if (body instanceof URLSearchParams)\n    return body.toString();\n  if (typeof body !== 'string') {\n    return `[${typeof body}]`;\n  }\n  body = body.substring(0, 5000).replaceAll(\"\\r\", \" \").replaceAll(\"\\n\", \" \");\n  return body;\n}\n\nfunction getResponseSummary(response: Pick<Response, \"headers\">) {\n  const toks = [];\n  if (response.headers.get(\"Content-Type\"))\n    toks.push(response.headers.get(\"Content-Type\"));\n  if (response.headers.get(\"Content-Length\"))\n    toks.push(response.headers.get(\"Content-Length\") + \" bytes\");\n  if (response.headers.get(\"Transfer-Encoding\"))\n    toks.push(response.headers.get(\"Transfer-Encoding\"));\n  if (response.headers.get(\"Content-Encoding\"))\n    toks.push(response.headers.get(\"Content-Encoding\"));\n\n  return toks.length ? `(${toks.join(\", \")})` : \"\";\n}\n\nfunction headersToString(headers: Headers) {\n  return JSON.stringify(Object.fromEntries(headers.entries()));\n}\n\nexport function extractRequestInfo(input: RequestInfo | URL, init?: RequestInit) {\n  const method = (init?.method || (input as Request)?.method || \"GET\");\n  const url = input instanceof URL ? input.toString() : typeof input === \"string\" ? input : input.url;\n  const headers = new Headers(typeof input === \"object\" && !(input instanceof URL) ? input.headers : init?.headers);\n  const body = typeof input === \"object\" && !(input instanceof URL) ? input.body : init?.body;\n  return { method, url, headers, body };\n}\n\nexport async function extractRequestBody(body: BodyInit): Promise<XMLHttpRequestBodyInit> {\n  if (!(body instanceof ReadableStream))\n    return body;\n\n  //convert the ReadableStream to ArrayBuffer so we can print AND send it (TODO ideally 'tee' the stream and just grab the first 5K bytes)\n  const reader = body.getReader();\n  const chunks = [];\n  for (; ;) {\n    const { done, value } = await reader.read();\n    if (done)\n      break;\n    chunks.push(value);\n  }\n  return await new Blob(chunks).arrayBuffer();\n}\n\nexport function logWrqRequest(method: string, url: string, headers: Headers, body: XMLHttpRequestBodyInit | undefined | null) {\n  const debugrequestid = generateRandomId();\n\n  console.log(`[wrq] ${debugrequestid} ${method.padEnd(7)} ${url}`);\n  console.log(`[wrq] ${debugrequestid} headers ${headersToString(headers)}`);\n  if (body) {\n    console.log(`[wrq] ${debugrequestid} body    ${sanitizeBody(body)}`);\n  }\n\n  return debugrequestid;\n}\n\nexport async function logWrqResponse(debugrequestid: string, fetchresult: Pick<Response, \"ok\" | \"status\" | \"statusText\" | \"headers\" | \"json\" | \"text\" | \"arrayBuffer\" | \"clone\">) {\n  console.log(`[wrq] ${debugrequestid} result  ${fetchresult.status} ${fetchresult.statusText} ${getResponseSummary(fetchresult)}`);\n  console.log(`[wrq] ${debugrequestid} headers ${headersToString(fetchresult.headers)}`);\n\n  const ct = fetchresult.headers.get(\"Content-Type\");\n  if (ct && [\"text/html\", \"text/plain\", \"application/json\"].includes(ct?.split(';')[0])) {\n    const responsebody = await fetchresult.clone().text();\n    console.log(`[wrq] ${debugrequestid} body    ${sanitizeBody(responsebody)}`);\n  }\n}\n\nasync function debuggableFetch(originalfetch: typeof fetch, input: RequestInfo | URL, init?: RequestInit) {\n  if (!debugFlags.wrq)\n    return originalfetch(input, init);\n\n  let { method, url, headers, body } = extractRequestInfo(input, init);\n  if (body) {\n    body = await extractRequestBody(body);\n    init = { ...init, body };\n  }\n\n  const debugrequestid = logWrqRequest(method, url, headers, body);\n\n  const fetchresult = await originalfetch(input, init); //TODO log responses as well (if safe/applicable, eg not binary or Very Long... and we probably should wait for the first json()/text()/body() call? but at least log the status and time!)\n  await logWrqResponse(debugrequestid, fetchresult);\n\n  return fetchresult;\n}\n\nexport function hookFetch() {\n  if (hookedfetch)\n    return;\n\n  globalThis.fetch = debuggableFetch.bind(null, globalThis.fetch);\n  hookedfetch = true;\n}\n\nexport function enableFetchDebugging() {\n  // Hook global fetch if requested\n  if (globalThis[\"fetch\"]) {\n    if (debugFlags.wrq)\n      hookFetch();\n\n    registerDebugConfigChangedCallback(() => {\n      if (debugFlags.wrq)\n        hookFetch();\n    });\n  }\n}\n","// This gets TypeScript to refer to us by our @webhare/... name in auto imports:\ndeclare module \"@webhare/env\" {\n}\n\nimport { debugFlags, dtapStage, isLive, backendBase } from \"./envbackend\";\nimport type { DTAPStage } from \"./concepts\";\n\nexport { type NavigateInstruction, navigateTo } from \"./navigation\";\nexport { enableFetchDebugging } from \"./fetchdebug\";\nexport { type DTAPStage, dtapStage, debugFlags, isLive, backendBase };\n\n//user locale *which Tollium and sites should start to use to define localization, and perhaps gettid too?)\nexport type UserLocale = {\n  /** The language code, eg 'en' including optional country, eg en-US */\n  lang: string;\n  //TODO regional settings\n};\n\n//export deprecated variants\nexport { dtapstage, islive } from \"./envbackend\";\n\n/** @deprecated For WH5.4 and up use 'debugFlags' */\nexport const flags = debugFlags;\n","import type { FormControlElement } from \"@webhare/dompack\";\n\nexport const CustomEvent = globalThis.CustomEvent;\n/** Wrap an event to ensure it's target is a HTMLElement\n * @typeParam EventType - The expected event type\n * @typeParam CurrentTargetType - The type of the elemnt you're binding the event to. Optional, defaults to HTMLElement\n * @typeParam RelatedTargetType - The type of the elemnt you're binding the event to. Optional, defaults to HTMLElement\n*/\nexport type DocEvent<EventType extends Event, CurrentTargetType extends HTMLElement = HTMLElement, RelatedTargetType extends HTMLElement = HTMLElement> = EventType & {\n  target: HTMLElement;\n  currentTarget: CurrentTargetType;\n  relatedTarget: RelatedTargetType | null;\n};\n\nexport type DomEventOptions =\n  {\n    bubbles?: boolean;\n    cancelable?: boolean;\n    relatedTarget?: EventTarget;\n    detail?: object;\n  };\n\nexport function dispatchDomEvent(element: EventTarget, eventtype: string, options?: DomEventOptions) {\n  //see here https://developer.mozilla.org/en-US/docs/Web/Events whether an event is bubbles/cancelabel\n  options = {\n    bubbles: [\n      \"input\", \"change\", \"click\", \"contextmenu\", \"dblclick\",\n      \"reset\", \"submit\"\n    ].includes(eventtype),\n    cancelable: [\n      \"animationstart\", \"animationcancel\", \"animationend\", \"animationiteration\",\n      \"beforeunload\",\n      \"click\", \"contextmenu\", \"dblclick\",\n      \"reset\", \"submit\",\n      \"transitionstart\", \"transitioncancel\", \"transitionend\", \"transitionrun\"\n    ].includes(eventtype),\n    ...options\n  };\n\n  if (!element || !(element as Node).ownerDocument)\n    return true; //the element has left the dom... so there's no more bubbling. just drop it\n\n  //FIXME the load/scroll is buggy and we should be probably be using new Event (but an earlier attempt at that triggered quite a few test failures)\n  const createtype = /*[\"load\",\"scroll\"].includes(eventtype) === \"load\" ? \"UIEvents\" :*/[\"focus\", \"blur\", \"focusin\", \"focusout\"].includes(eventtype) ? \"FocusEvent\" : eventtype === \"click\" ? \"MouseEvents\" : \"HTMLEvents\";\n  //we verified its non-null ness above but TS doesn't really understand that\n  const evt = (element as Node).ownerDocument!.createEvent(createtype);\n  evt.initEvent(eventtype, options.bubbles, options.cancelable);\n  if (options.detail)\n    // eslint-disable-next-line @typescript-eslint/no-explicit-any -- we should just rewrite to new Event\n    (evt as any).detail = options.detail;\n  if (options.relatedTarget) //its a readonly prop, so redefine it\n    Object.defineProperty(evt, 'relatedTarget', { value: options.relatedTarget, writable: false });\n\n  //TODO: Should we keep this code?\n  // @ts-ignore IScroll is a custom Window property set by IScroll\n  if (eventtype === 'click' && window.IScroll)\n    // @ts-ignore _constructed is a custom Event property used by IScroll\n    evt._constructed = true; //ensure IScroll doesn't blindly cancel our synthetic clicks\n\n  return element.dispatchEvent(evt);\n}\n\n/** Fire the proper modified events (input and/or change) on the element after changing its value\n * @param element - Element to receive event\n * @param options - Event options\n * @deprecated Use changeValue so we can figure out the proper events to fire\n */\nexport function fireModifiedEvents(element: FormControlElement, options?: DomEventOptions) {\n  dispatchDomEvent(element, 'input', options);\n  dispatchDomEvent(element, 'change', options);\n}\n\n//manually fire 'onchange' events. needed for event simulation - DEPRECATED\n/**\n * @param element - Element to receive event\n * @param type - Event type\n * @param options - Event options\n * @deprecated Use dispatchDomEvent instead\n */\nexport function fireHTMLEvent(element: EventTarget, type: string, options?: DomEventOptions) {\n  return dispatchDomEvent(element, type, options);\n}\n\ntype CustomEventParams =\n  {\n    /** Whether this event should bubble up in the DOM */\n    bubbles: boolean;\n    /** Whether this event can be cancelled */\n    cancelable: boolean;\n    /** Custom event information */\n    detail?: unknown;\n    /** Handler to execute if the default isn't prevented by a event listener */\n    defaulthandler?: (evt: CustomEvent) => void;\n  };\n\n/** Fire a custom event\n    @param node - node to fire the event on\n    @param event - event type. You should add this event to the GlobalEventHandlersEventMap for validation in dispatchCustomEvent calls and addEventListener callbacks.\n    @param params - Event options\n    @returns true if the default wasn't prevented\n */\n\nexport function dispatchCustomEvent<K extends string>(\n  node: EventTarget,\n  event: K,\n  params: CustomEventParams & (K extends keyof GlobalEventHandlersEventMap ?\n    GlobalEventHandlersEventMap[K] extends CustomEvent ?\n    { detail: GlobalEventHandlersEventMap[K][\"detail\"] } : unknown : unknown)) {\n\n  if (!params)\n    throw new Error(`Missing dispatchCustomEvent params`);\n  ['bubbles', 'cancelable'].forEach(prop => {\n    if (!(prop in params))\n      throw new Error(`Missing '${prop}' in dispatchCustomEvent params`);\n  });\n\n  const evt = new CustomEvent(event, {\n    bubbles: params.bubbles,\n    cancelable: params.cancelable,\n    detail: params.detail\n  });\n  let defaultaction = true;\n  try {\n    if (!node.dispatchEvent(evt))\n      defaultaction = false; //defaultPrevented is unreliable on IE11, so double check\n  } finally {\n    if (!evt.defaultPrevented && params.defaulthandler && defaultaction) {\n      params.defaulthandler(evt);\n    }\n  }\n  return defaultaction && !evt.defaultPrevented;\n}\n\n/**\n     Change the value of a form element, and fire the correct events as if it were a user change\n *\n    @param element - Element to change\n    @param newvalue - New value\n */\nexport function changeValue(element: FormControlElement, newvalue: string | number | boolean) {\n  if (element.matches(`input[type=radio], input[type=checkbox]`)) {\n    if (Boolean((element as HTMLInputElement).checked) === Boolean(newvalue))\n      return;\n    (element as HTMLInputElement).checked = Boolean(newvalue);\n  } else {\n    //FIXME it's not really clean to assume that this element is changeable - throw for non input/select..\n    const asString = String(newvalue);\n    if ((element as HTMLInputElement).value === asString)\n      return;\n\n    (element as HTMLInputElement).value = String(newvalue);\n  }\n  dispatchDomEvent(element as EventTarget, 'input');\n  dispatchDomEvent(element as EventTarget, 'change');\n}\n\nlet keydata: { mapping: { [key: string]: string } };\n\nfunction initKeyMapping() {\n  keydata =\n  {\n    // Mapping from keyIdentifier/key to key. If not found, translate U+XXXX to the unicode char and return that.\n\n    // List of all current key mappings (and current inconsistencies\n    // at https://developer.mozilla.org/en-US/docs/Web/API/KeyboardEvent/key/Key_Values\n    mapping:\n    {\n      \"Del\": \"Delete\", // IE/edge use 'Del' instead of 'Delete'\n      \"Esc\": \"Escape\", // IE/edge use 'Esc' instead of 'Escape'\n      \"OS\": \"Meta\", // Meta key is called OS on IE 9 and Firefox (tested v50)\n      \"Win\": \"Meta\", // Meta key is called Win on IE/Edge\n      \"Scroll\": \"ScrollLock\", // More IE/Edge, from tests\n      \"PrintScreen\": \"Print\", // More IE/Edge, from tests\n      \"MozHomeScreen\": \"GoHome\", // Prior to Firefox 37, the Home button generated a key code of \"Exit\". Starting in Firefox 37, the button generates the key code \"MozHomeScreen\".\n\n      // Internet Explorer 9 and Firefox 36 and earlier return \"Left\", \"Right\", \"Up\", and \"Down\" for the arrow keys, instead of \"ArrowLeft\", \"ArrowRight\", \"ArrowUp\", and \"ArrowDown\".\n      \"Left\": \"ArrowLeft\",\n      \"Right\": \"ArrowRight\",\n      \"Up\": \"ArrowUp\",\n      \"Down\": \"ArrowDown\",\n      // More IE and old Firefox stuff\n      \"Crsel\": \"CrSel\",\n      \"Exsel\": \"ExSel\",\n      \"Nonconvert\": \"NonConvert\",\n      // Internet Explorer 9 and Firefox 36 and earlier report \"Apps\" instead of \"ContextMenu\" for the context menu key.\n      \"Apps\": \"ContextMenu\",\n      // Internet Explorer 9 and Firefox 36 and earlier use \"MediaNextTrack\" and \"MediaPreviousTrack\" instead of \"MediaTrackNext\" and \"MediaTrackPrevious\".\n      \"MediaNextTrack\": \"MediaTrackNext\",\n      \"MediaPreviousTrack\": \"MediaTrackPrevious\",\n      // In Internet Explorer 9, and prior to Firefox 49, \"AudioVolumeUp\", \"AudioVolumeDown\", and \"AudioVolumeMute\" were \"VolumeUp\", \"VolumeDown\", and \"VolumeMute\".\n      // In Firefox 49 they were updated to match the latest specification. The old names are still used on Boot to Gecko.\n      \"VolumeUp\": \"AudioVolumeUp\",\n      \"VolumeDown\": \"AudioVolumeDown\",\n      \"VolumeMute\": \"AudioVolumeMute\",\n      // Firefox added proper support for the \"TV\" key in Firefox 37; before that, this key generated the key code \"Live\".\n      \"Live\": \"TV\",\n      // Internet Explorer 9 and Firefox 36 and earlier identify the zoom toggle button as \"Zoom\". Firefox 37 corrects this to \"ZoomToggle\".\n      \"Zoom\": \"ZoomToggle\",\n      // Internet Explorer 9 and Firefox 36 and earlier use \"SelectMedia\" instead of \"LaunchMediaPlayer\". Firefox 37 through Firefox 48 use \"MediaSelect\". Firefox 49 has been updated to match the latest specification, and to return \"LaunchMediaPlayer\".\n      \"SelectMedia\": \"LaunchMediaPlayer\",\n      \"MediaSelect\": \"LaunchMediaPlayer\",\n      // Google Chrome returns \"LaunchCalculator\" instead of \"LaunchApplication1\". See Chromium bug 612743 for more information.\n      // Google Chrome returns \"LaunchMyComputer\" instead of \"LaunchApplication2\". See Chromium bug 612743 for more information.\n      // (LaunchCalculator and LaunchMyComputer are valid too, so no translation)\n\n      // While older browsers used words like \"Add\", \"Decimal\", \"Multiply\", and so forth modern browsers identify these using the actual character (\"+\", \".\", \"*\", and so forth).\n      \"Multiply\": \"*\",\n      \"Add\": \"+\",\n      \"Divide\": \"/\",\n      \"Subtract\": \"-\",\n      \"Decimal\": \".\", // (mozilla Key_Values doc says depends on region)\n      \"Separator\": \".\", // (mozilla Key_Values doc says depends on region)\n\n      // keyIdenfier spec mapping of non-printable keys: https://www.w3.org/TR/2009/WD-DOM-Level-3-Events-20090908/#keyset-keyidentifiers\n      // Used in safari\n      \"U+0008\": \"Backspace\",\n      \"U+0009\": \"Tab\",\n      \"U+000D\": \"Enter\",\n      \"U+0018\": \"Cancel\",\n      \"U+001B\": \"Escape\",\n      \"U+007F\": \"Delete\",\n      \"U+0300\": \"DeadGrave\",\n      \"U+0301\": \"DeadEacute\",\n      \"U+0302\": \"DeadCircumflex\",\n      \"U+0303\": \"DeadTilde\",\n      \"U+0304\": \"DeadMacron\",\n      \"U+0306\": \"DeadBreve\",\n      \"U+0307\": \"DeadAboveDot\",\n      \"U+0308\": \"DeadUmlaut\",\n      \"U+030A\": \"DeadAboveRing\",\n      \"U+030B\": \"DeadDoubleacute\",\n      \"U+030C\": \"DeadCaron\",\n      \"U+0327\": \"DeadCedilla\",\n      \"U+0328\": \"DeadOgonek\",\n      \"U+0345\": \"DeadIota\",\n      \"U+3099\": \"DeadVoicedSound\",\n      \"U+309A\": \"DeadSemivoicedSound\",\n\n      // keyIdenfier spec mapping to characters: https://www.w3.org/TR/2009/WD-DOM-Level-3-Events-20090908/#keyset-keyidentifiers\n      \"Exclamation\": \"!\",\n      \"DoubleQuote\": \"\\\"\",\n      \"Hash\": \"#\",\n      \"Dollar\": \"$\",\n      \"Ampersand\": \"&\",\n      \"LeftParen\": \"(\",\n      \"RightParen\": \")\",\n      \"Asterisk\": \"*\",\n      \"Plus\": \"+\",\n      \"Percent\": \"%\",\n      \"Comma\": \",\",\n      \"HyphenMinus\": \"-\",\n      \"Period\": \".\",\n      \"Solidus\": \"/\",\n      \"Colon\": \":\",\n      \"Semicolon\": \";\",\n      \"LessThan\": \"<\",\n      \"Equals\": \"=\",\n      \"GreaterThan\": \">\",\n      \"QuestionMark\": \"?\",\n      \"At\": \"@\",\n      \"LeftSquareBracket\": \"[\",\n      \"Backslash\": \"\\\\\",\n      \"RightSquareBracket\": \"]\",\n      \"Circumflex\": \"^\",\n      \"Underscore\": \"_\",\n      \"Grave\": \"`\",\n      \"LeftCurlyBracket\": \"{\",\n      \"Pipe\": \"|\",\n      \"RightCurlyBracket\": \"}\",\n      \"Euro\": \"€\",\n      \"InvertedExclamation\": \"¡\",\n\n      // Safari fixes: viewed in local tests\n      \"U+0010\": \"ContextMenu\"\n    }\n  };\n}\n\nexport type NormalizedKeyboardEvent =\n  {\n    type: string;\n    target: EventTarget | null;\n    key: string;\n    code: string;\n    ctrlKey: boolean;\n    altKey: boolean;\n    location: number;\n    shiftKey: boolean;\n    metaKey: boolean;\n    repeat: boolean;\n    isComposing: boolean;\n  };\n\n/**\n     Returns normalized keyboard event properties, following the current W3C UI Events spec\n *\n    @param evt - Keyboard event\n    @returns Normalized keyboard event data\n */\nexport function normalizeKeyboardEventData(evt: KeyboardEvent): NormalizedKeyboardEvent {\n  // event.key is supported from chrome:51, edge, ff: 29, ie: 9, not in safari\n  // event.keyIdentifier in chrome 26-54, opera 15-41, safara: 5.1\n  // safari doesn't provide either in keypress events, use U+evt.keyCode (uppercase 4-byte hex)\n\n  let key = evt.key;\n  if (!keydata)\n    initKeyMapping();\n  if (keydata.mapping[key])\n    key = keydata.mapping[key];\n  else if (key.startsWith(\"U+\")) // U+xxxx code\n    key = String.fromCodePoint(parseInt(key.substring(2), 16));\n\n  // Seen in chrome 56.0.2924.76 on linux, numpad '.' without numlock returns key \"\\u0000\"\n  if (evt.key === \"\\u0000\" && evt.code === \"NumpadDecimal\")\n    key = \".\";\n\n  return (\n    {\n      type: evt.type,\n      target: evt.target,\n      key: key,\n      code: evt.code || \"Unidentified\",\n      ctrlKey: evt.ctrlKey,\n      altKey: evt.altKey,\n      location: evt.location,\n      shiftKey: evt.shiftKey,\n      metaKey: evt.metaKey,\n      repeat: evt.repeat,\n      isComposing: evt.isComposing\n    });\n}\n\n/**\n * Stop, fully, an event\n *\n * @param event - Event to stop\n */\nexport function stop(event: Event) {\n  event.preventDefault();\n  event.stopImmediatePropagation();\n}\n\n/** Add an event listener to HTMLElements inside a document (which allows us to ensure that 'target' is a HTMLElement for easier typings) */\nexport function addDocEventListener<CurrentTargetType extends HTMLElement, Type extends keyof HTMLElementEventMap>(node: CurrentTargetType, type: Type, listener: (this: CurrentTargetType, ev: DocEvent<HTMLElementEventMap[Type], CurrentTargetType>) => void | Promise<void>, options?: AddEventListenerOptions): void;\nexport function addDocEventListener<CurrentTargetType extends HTMLElement>(node: CurrentTargetType, type: string, listener: (this: CurrentTargetType, evt: DocEvent<Event, CurrentTargetType>) => void | Promise<void>, options?: AddEventListenerOptions): void;\n\nexport function addDocEventListener<CurrentTargetType extends HTMLElement>(node: CurrentTargetType, type: string, listener: (this: CurrentTargetType, evt: DocEvent<Event, CurrentTargetType>) => void | Promise<void>, options?: AddEventListenerOptions): void {\n  node.addEventListener(type, listener as EventListener, options);\n}\n","import { debugFlags } from '@webhare/env';\nimport * as domevents from '../../../modules/system/js/dompack/src/events';\nimport \"@webhare/tsrun/src/polyfills\";\n\nlet locallocks: BusyLock[] = [];\nlet modallocked = false;\nlet uiwatcher: NodeJS.Timeout | null = null;\nlet installedanticancelhandler = false;\n\nlet currentbusymodaldialog: HTMLDialogElement | null = null;\nlet currentbusymodaluserdialog: HTMLDialogElement | null = null;\nlet busymodalcontent: string | HTMLElement | HTMLDialogElement | undefined;\nlet mylockmgr: LockManager | undefined;\n\nexport type BusyModalEvent = CustomEvent<{ show: boolean }>;\n\n// As 'Lock' already exists on the Web (https://developer.mozilla.org/docs/Web/API/Lock) and it's seldom used just use a longer name. Available since WH 5.4\nexport interface UIBusyLock extends Disposable {\n  release(): void;\n  [Symbol.dispose](): void;\n}\n\ndeclare global {\n  interface GlobalEventHandlersEventMap {\n    \"dompack:busymodal\": BusyModalEvent;\n  }\n  interface Window {\n    __dompack_busylockmanager?: LockManager;\n  }\n\n  // __dompack_busylockmanager: LockManager | undefined;  //\n}\n\nfunction getLockmgr(): LockManager {\n  if (!mylockmgr) {\n    mylockmgr = getParentLockManager() || new LockManager;\n    if (typeof window !== \"undefined\")\n      window.__dompack_busylockmanager = mylockmgr;\n  }\n  return mylockmgr;\n}\n\nfunction anyModalLocks() {\n  return locallocks.some(l => l.modal);\n}\n\n/* scheduleCheckUIFree is invoked by release() or when waitUIFree is explicitly called. the call from release() should\n   be a 'fast' path. We schedule a full check for the next tick if there's a chance it might actually find a free UI */\nfunction scheduleCheckUIFree() {\n  if (!uiwatcher && locallocks.length === 0)\n    uiwatcher = setTimeout(() => checkUIFree(), 0);\n}\n\n/* check if the UI is actually free. if so, remove busymodals and resolve waitUIFrees for the benefit of testfw  */\nfunction checkUIFree() {\n  uiwatcher = null;\n\n  if (modallocked && !anyModalLocks()) { //did the last frame-level *modal* lock just get released?\n    modallocked = false;\n    toggleBusyModal(false);\n  }\n\n  if (locallocks.length === 0) {\n    const lockmgr = getLockmgr();\n    lockmgr.busyframes.delete(window); //we won't release our block in the lockmanager until we've had a chance to remove our modal layer\n    lockmgr.checkUIFree(); //to resolve any waitUIFrees. runs in the top-level frame. note that lockmgr cares about ALL ui locks, not just modals\n  }\n}\n\nfunction isDialogElement(el: unknown): boolean {\n  return typeof el === \"object\" && (el as HTMLElement).matches?.(\"dialog\") || false;\n}\n\nfunction checkCancelEvent(evt: Event) {\n  if (modallocked)\n    evt.preventDefault();\n}\n\nfunction toggleBusyModal(show: boolean) {\n  //'islock' is legacy non-camel version. TypeScript typing should help us transition (since 5.3). 'as' shuts up the warning TODO remove 'islock' and the 'as'\n  if (!domevents.dispatchCustomEvent(window, 'dompack:busymodal', { bubbles: true, cancelable: true, detail: { show: show, islock: show } as BusyModalEvent['detail'] }))\n    return; //cancelled!\n\n  if (!installedanticancelhandler) {\n    //capture cancel, as it doesn't bubble up\n    addEventListener(\"cancel\", evt => checkCancelEvent(evt), { capture: true });\n    installedanticancelhandler = true;\n  }\n\n  if (show) {\n    if (isDialogElement(busymodalcontent)) { //the user provided us with an element\n      currentbusymodaluserdialog = busymodalcontent as HTMLDialogElement;\n      currentbusymodaluserdialog.showModal();\n    } else if (busymodalcontent) { //we'll create our own dialog\n      const dialog = document.createElement('dialog');\n      const toembed = typeof busymodalcontent === \"string\" ? document.createTextNode(busymodalcontent) : busymodalcontent.cloneNode(true);\n      dialog.className = \"dompack-busydialog\";\n      dialog.role = \"status\";\n      dialog.ariaLive = \"off\";\n      dialog.append(toembed);\n      document.body.appendChild(dialog);\n      currentbusymodaldialog = dialog;\n      dialog.showModal();\n      return;\n    }\n\n    document.documentElement.classList.add(\"dompack--busymodal\");\n    return;\n  }\n\n  //hiding\n  document.documentElement.classList.remove(\"dompack--busymodal\");\n  if (currentbusymodaluserdialog)\n    currentbusymodaluserdialog.close();\n  if (currentbusymodaldialog) { //we added a dialog to the dom\n    currentbusymodaldialog.close();\n    document.body.removeChild(currentbusymodaldialog);\n    currentbusymodaldialog = null;\n  }\n}\n\nclass LockManager {\n  locks: BusyLock[];\n  busyframes: Set<Window> = new Set;\n  busycounter: number;\n  deferreduipromise: PromiseWithResolvers<boolean> | null;\n\n  //this object is not for external consumption\n  constructor() {\n    this.locks = [];\n    this.busycounter = 0;\n    this.deferreduipromise = null;\n  }\n  add(lock: BusyLock) {\n    this.locks.push(lock);\n    const returnvalue = this.busycounter++;\n    return returnvalue;\n  }\n  release(lock: BusyLock) {\n    const pos = this.locks.indexOf(lock);\n    if (pos === -1) {\n      if (debugFlags.bus) {\n        console.error(\"Duplicate release of busy lock #\" + lock.locknum);\n        console.log(\"Lock allocated:\");\n        console.log(lock.acquirestack);\n        console.log(\"Lock first released:\");\n        console.log(lock.releasestack);\n      }\n      throw new Error(\"Duplicate release of busy lock\");\n    }\n\n    this.locks.splice(pos, 1);\n  }\n  getNumLocks() {\n    return this.locks.length;\n  }\n  //used by child windows to schedule a check in *our* frame (eg before they themselves are unloaded)\n  scheduleCheckUIFree() {\n    scheduleCheckUIFree();\n  }\n  checkUIFree() {\n    if (this.locks.length === 0 && this.busyframes.size === 0 && this.deferreduipromise) {\n      this.deferreduipromise.resolve(true);\n      this.deferreduipromise = null;\n    }\n  }\n  waitUIFree() {\n    if (!this.deferreduipromise)\n      this.deferreduipromise = Promise.withResolvers();\n\n    scheduleCheckUIFree(); //ensures uiwait is released at next tick if no locks are present at all\n    return this.deferreduipromise.promise;\n  }\n  logLocks() {\n    this.locks.forEach(lock => console.log('[bus] lock #' + lock.locknum, lock.acquirestack, lock));\n    console.log(\"[bus] total \" + this.locks.length + \" locks\");\n  }\n  getLockIds() {\n    return this.locks.map(l => \"#\" + l.locknum).join(\", \");\n  }\n}\n\n/** Configure an (accessible) modal dialog\n * @param bmc - What to show in the dialog: either a text or DOM fragment to clone.\n *              If a <dialog> element is passed, this dialog will be used instead of creating a new one.\n */\nexport function setupBusyModal(bmc: NonNullable<typeof busymodalcontent>) {\n  busymodalcontent = bmc;\n}\n\ninterface LockOptions {\n  ///Whether this lock should enable a modality laayer\n  modal?: boolean;\n}\n\nclass BusyLock implements UIBusyLock {\n  modal: boolean;\n  locknum: number;\n  acquirestack: string | undefined;\n  releasestack: string | undefined;\n\n  constructor(options?: LockOptions) {\n    //legacy non-camel name is 'ismodal'\n    this.modal = options?.modal ?? (options as { ismodal?: boolean })?.ismodal ?? false;\n\n    const lockmgr = getLockmgr();\n    this.locknum = lockmgr.add(this);\n    lockmgr.busyframes.add(window);\n    locallocks.push(this);\n\n    if (this.modal && !modallocked) {\n      modallocked = true;\n      toggleBusyModal(true);\n    }\n\n    if (debugFlags.bus) {\n      this.acquirestack = (new Error).stack;\n      console.trace('[bus] Busy lock #' + this.locknum + ' taken. ' + lockmgr.getNumLocks() + \" locks active now: \" + lockmgr.getLockIds());\n    }\n  }\n\n  [Symbol.dispose]() {\n    if (debugFlags.bus)\n      this.releasestack = (new Error).stack;\n\n    const lockmgr = getLockmgr();\n    lockmgr.release(this);\n    const lockpos = locallocks.indexOf(this);\n    locallocks.splice(lockpos, 1);\n\n    if (debugFlags.bus) {\n      console.trace('[bus] Busy lock #' + this.locknum + ' released. ' + lockmgr.getNumLocks() + \" locks active now: \" + lockmgr.getLockIds());\n    }\n\n    scheduleCheckUIFree();\n  }\n\n  release() {\n    this[Symbol.dispose]();\n  }\n}\n\n/** Return a promise resolving as soon as the UI (any accessible frame) is free for at least one tick */\nexport function waitUIFree() {\n  return getLockmgr().waitUIFree();\n}\n\n/**\n     flag userinterface as busy. tests then know not to interact with the UI until the busy flag is released\n *\n    @param options - Options.<br>\n                   - modal: true/false - Whether the lock is a modal lock\n */\nexport function flagUIBusy(options?: LockOptions): UIBusyLock {\n  return new BusyLock(options);\n}\n\nexport function getUIBusyCounter() {\n  return getLockmgr().busycounter;\n}\n\nfunction getParentLockManager(): LockManager | null {\n  try { //we're accessing a parent window, so we may hit security exceptions\n    if (!(window.parent?.__dompack_busylockmanager))\n      return null;\n\n    //if we connected to a parent...  deregister our locks, eg. if parent navigated our frame away\n    window.addEventListener(\"pagehide\", () => {\n      if (debugFlags.bus)\n        console.log(\"[bus] Frame unloading, \" + locallocks.length + \" locks pending.\", locallocks.map(l => \"#\" + l.locknum).join(\", \"), locallocks);\n\n      //switch to local instance in case anyone still tries to touch these locks during unload\n      const lockmgr = getLockmgr();\n      const locallockmgr = new LockManager;\n      locallocks.forEach(lock => { lockmgr.release(lock); locallockmgr.add(lock); });\n      locallocks = [];\n\n      lockmgr.busyframes.delete(window); //explicitly remove us so we won't be waited upon\n      lockmgr.scheduleCheckUIFree();\n      mylockmgr = locallockmgr;\n    });\n\n    return window.parent.__dompack_busylockmanager;\n  } catch (e) {\n    return null;\n  }\n}\n","/** Rect describing an elements position */\nexport type Rect = {\n  top: number;\n  bottom: number;\n  left: number;\n  right: number;\n  width: number;\n  height: number;\n};\n\n/** Elements you can set a value on and would have to trigger change and/or input events */\nexport type FormControlElement = HTMLInputElement | HTMLSelectElement | HTMLTextAreaElement;\n\n/** Types that support querySelector(All) */\ntype QuerySelectorAble = ParentNode | ShadowRoot;\n\nexport function qS<E extends Element = HTMLElement>(startnode: QuerySelectorAble, selector: string): E | null;\nexport function qS<E extends Element = HTMLElement>(selector: string): E | null;\n\n/** Match the first element using a CSS selector\n * @param node_or_selector - The starting node or selector\n * @param selector - The selector to match if the starting node was specified\n * @returns The first matching element or null\n */\nexport function qS<E extends Element>(node_or_selector: QuerySelectorAble | string, selector?: string): E | null {\n  if (typeof node_or_selector === 'string')\n    return document.querySelector<E>(node_or_selector);\n  else if (selector)\n    return node_or_selector.querySelector<E>(selector);\n  return null;\n}\n\n\nexport function qR<E extends Element = HTMLElement>(startnode: QuerySelectorAble, selector: string): E;\nexport function qR<E extends Element = HTMLElement>(selector: string): E;\n\n/** Match a specific element using a CSS selector, requiring it to exist and be unique\n * @param node_or_selector - The starting node or selector\n * @param selector - The selector to match if the starting node was specified\n * @returns The requested element. Throw it the selector doesn't match exactly one element\n */\nexport function qR<E extends Element>(node_or_selector: QuerySelectorAble | string, selector?: string): E {\n  const matches = qSA<E>(node_or_selector as QuerySelectorAble, selector as string);\n  if (matches.length === 1)\n    return matches[0];\n\n  if (typeof node_or_selector !== 'string') {\n    console.error(`${matches.length} elements match selector \\`${selector}\\` with startingpoint`, node_or_selector, matches);\n    throw new Error(`${matches.length} elements match selector \\`${selector}\\` given startingpoint '${node_or_selector.nodeName}'`);\n  } else {\n    console.error(`${matches.length} elements match selector \\`${node_or_selector}\\` in the document`, matches);\n    throw new Error(`${matches.length} elements match selector \\`${node_or_selector}\\` in the document`);\n  }\n}\n\nexport function qSA<E extends Element = HTMLElement>(startnode: QuerySelectorAble, selector: string): E[];\nexport function qSA<E extends Element = HTMLElement>(selector: string): E[];\n\n/** Find multiple elements using a CSS selector\n * @param node_or_selector - The starting node or selector\n * @param selector - The selector to match if the starting node was specified\n * @returns The requested elements.\n */\nexport function qSA<E extends Element>(node_or_selector: QuerySelectorAble | string, selector?: string): E[] {\n  if (typeof node_or_selector === 'string')\n    return Array.from(document.querySelectorAll(node_or_selector));\n  else if (selector)\n    return Array.from(node_or_selector.querySelectorAll(selector));\n\n  return [];\n}\n\n/** Return whether the passed element is a FormControlElement */\nexport function isFormControl(field: Element): field is FormControlElement {\n  return isHTMLElement(field) && [\"INPUT\", \"SELECT\", \"TEXTAREA\"].includes(field.tagName);\n}\n\n/** Return whether the passed element is an editable text field */\nexport function isEditControl(field: Element): field is HTMLElement {\n  return isHTMLElement(field) && ([\"INPUT\", \"TEXTAREA\"].includes(field.tagName) || field.isContentEditable);\n}\n\n/** Test whether node is an Element, even if it's in a different iframe */\nexport function isElement(node: unknown): node is Element {\n  //TODO What is actually going on if defaultView is missing?\n  if (!node || typeof node !== \"object\")\n    return false;\n\n  /* Getting the proto doesn't always work:\n  const proto = (node as Element).ownerDocument.defaultView?.Element;\n  (because our iframes derive off about? not sure) so until someone finds the real answer, we'll do a heuristic\n  */\n\n  /** Analyzing node.constructor.name doesn't work because custom elements don't necessarily have their constructor's name ending in Element\n  return Boolean(\"ownerDocument\" in node && node.constructor.name.match(/Element$/));\n  */\n\n  return Boolean(\"ownerDocument\" in node && (node as Element).nodeType === 1);\n\n}\n\n/** Test whether node is a HTMLElement, even if it's in a different iframe */\nexport function isHTMLElement(node: unknown): node is HTMLElement {\n  return isElement(node) && \"accessKey\" in node;\n}\n\n/**\n * get the relative bound difference between two elements, and return a writable copy\n *\n * @param node - The node for which you need coordinates\n * @param relativeto - Optional reference point. If not set, you just get a 'normal' coordinate object\n */\nexport function getRelativeBounds(node: Element, relativeto?: Element): Rect {\n  if (!relativeto)\n    relativeto = node.ownerDocument.documentElement;\n\n  const nodecoords = node.getBoundingClientRect();\n  const relcoords = relativeto.getBoundingClientRect();\n  return {\n    top: nodecoords.top - relcoords.top,\n    left: nodecoords.left - relcoords.left,\n    right: nodecoords.right - relcoords.left,\n    bottom: nodecoords.bottom - relcoords.top,\n    width: nodecoords.width,\n    height: nodecoords.height\n  };\n}\n","\n/* Regex to identify dimensionless style sttributes. copied from old version of preact/src/constants.js (MIT)\n   meant to capture:\n  { boxFlex:1, boxFlexGroup:1, columnCount:1, fillOpacity:1, flex:1, flexGrow:1,\n    flexPositive:1, flexShrink:1, flexNegative:1, fontWeight:1, lineClamp:1, lineHeight:1,\n    opacity:1, order:1, orphans:1, strokeOpacity:1, widows:1, zIndex:1, zoom:1\n*/\nconst IS_NON_DIMENSIONAL = /acit|ex(?:s|g|n|p|$)|rph|ows|mnc|ntw|ine[ch]|zoo|^ord/i;\n\nexport { qS, qSA } from '@webhare/dompack/src/tree.ts';\n\nfunction generateInsertList(nodes: Array<string | Node>) {\n  if (nodes.length === 1)\n    return typeof nodes[0] === 'string' ? document.createTextNode(nodes[0]) : nodes[0];\n\n  const frag = document.createDocumentFragment();\n  nodes.forEach(node => frag.appendChild(typeof node === 'string' ? document.createTextNode(node) : node));\n  return frag;\n}\n\n/** @deprecated Use node.matches() */\nexport function matches(node: Element, selector: string): boolean {\n  //only invoke 'matches' if it exists. it *should* but past versions of dompack.matches would check for it too (and thus not fail if you passed in a string instead of a Node)\n  return node.matches?.(selector);\n}\n/** @deprecated Use node.closest() */\nexport function closest(node: Element, selector: string) {\n  if (node.closest)\n    return node.closest(selector);\n  //TODO: Warn about out-of-date browser?\n  let testNode: Element | null = node;\n  for (; testNode && !matches(testNode, selector); testNode = testNode.parentElement)\n    /*iterate*/;\n  return testNode;\n}\n//implements contains. TODO we only really need this on IE11, which doesn't consider a text node a child, we can probably fall back to native elsewhere ?\n/** @deprecated Use node.contains() */\nexport function contains(ancestor: Node, child: Node) {\n  if (ancestor.contains)\n    return ancestor.contains(child);\n  //TODO: Warn about out-of-date browser?\n  for (let testNode: Node | null = child; testNode; testNode = testNode.parentNode)\n    if (testNode === ancestor)\n      return true;\n  return false;\n}\n//insert a range of nodes before a node: https://dom.spec.whatwg.org/#dom-childnode-before\n/** @deprecated Use node.before() */\nexport function before(node: ChildNode, ...nodes: Array<string | Node>) {\n  if (node.before) {\n    node.before(...nodes);\n    return;\n  }\n  //TODO: Warn about out-of-date browser?\n  if (node.parentNode)\n    node.parentNode.insertBefore(generateInsertList(nodes), node);\n}\n//insert a range of nodes after a node: https://dom.spec.whatwg.org/#dom-childnode-after\n/** @deprecated Use node.after() */\nexport function after(node: ChildNode, ...nodes: Array<string | Node>) {\n  if (node.after) {\n    node.after(...nodes);\n    return;\n  }\n  //TODO: Warn about out-of-date browser?\n  if (node.parentNode)\n    node.parentNode.insertBefore(generateInsertList(nodes), node.nextSibling);\n}\n//replace node with a set of nodes : https://dom.spec.whatwg.org/#dom-childnode-replacewith\n/** @deprecated Use node.replaceWith() */\nexport function replaceWith(node: ChildNode, ...nodes: Array<string | Node>) {\n  if (node.replaceWith) {\n    node.replaceWith(...nodes);\n    return;\n  }\n  //TODO: Warn about out-of-date browser?\n  if (node.parentNode)\n    node.parentNode.replaceChild(generateInsertList(nodes), node);\n}\n//remove node with a set of nodes : https://dom.spec.whatwg.org/#dom-childnode-remove\n/** @deprecated Use node.remove() */\nexport function remove(node: ChildNode) {\n  if (node.remove) {\n    node.remove();\n    return;\n  }\n  //TODO: Warn about out-of-date browser?\n  if (node.parentNode)\n    node.parentNode.removeChild(node);\n}\n//insert nodes at start: https://dom.spec.whatwg.org/#dom-parentnode-prepend\n/** @deprecated Use node.prepend() */\nexport function prepend(node: ParentNode, ...nodes: Array<string | Node>) {\n  if (node.prepend) {\n    node.prepend(...nodes);\n    return;\n  }\n  //TODO: Warn about out-of-date browser?\n  node.insertBefore(generateInsertList(nodes), node.firstChild);\n}\n//insert nodes at end: https://dom.spec.whatwg.org/#dom-parentnode-append\n/** @deprecated Use node.append() */\nexport function append(node: ParentNode, ...nodes: Array<string | Node>) {\n  if (node.append) {\n    node.append(...nodes);\n    return;\n  }\n  //TODO: Warn about out-of-date browser?\n  node.appendChild(generateInsertList(nodes));\n}\n\n/**\n * Toggle a single class\n *\n * @param node - Node to modify\n * @param classname - Class to toggle\n * @param settoggle - true to enable, false to disable, undefined to toggle\n * @deprecated Just use classList.toggle on the node itself\n */\n/** @deprecated Use classList.toggle() */\nexport function toggleClass(node: Element, classname: string, settoggle?: boolean) {\n  if (arguments.length === 2) //in old dompack, 2 argument version toggled and 3 argument version toggled off. match that behavior\n    node.classList.toggle(classname);\n  else\n    node.classList.toggle(classname, settoggle);\n}\n\n/**\n     Toggle classes in a node\n *\n    @param node - Node which classes to toggle\n    @param toggles - Object, all keys will be added/removed based on the truthyness of their values\n */\nexport function toggleClasses(node: Element, toggles: { [key: string]: boolean }) {\n  if (typeof (toggles) !== \"object\")\n    throw new Error(\"Expected an object with keys as classnames\");\n  Object.keys(toggles).forEach(key => node.classList[toggles[key] ? \"add\" : \"remove\"](key));\n}\n\n/* remove the contents of an existing node */\n/** @deprecated Use node.replaceChildren() */\nexport function empty(node: Element) {\n  if (node.replaceChildren) {\n    node.replaceChildren();\n    return;\n  }\n  //TODO: Warn about out-of-date browser?\n  while (node.lastChild)\n    node.removeChild(node.lastChild);\n}\n\nexport function isDomReady() {\n  //ensure no domready events can run if there will never be a dom\n  return typeof document !== \"undefined\" && (document.readyState === \"interactive\" || document.readyState === \"complete\");\n}\n\n/* run the specified function 'on ready'. adds to DOMContentLoaded if dom is not ready yet. Exceptions/rejections from the ready handler will not be fatal to the rest of code execution */\nexport function onDomReady(callback: () => unknown | Promise<unknown>) {\n  if (isDomReady()) {\n    try {\n      void callback(); //Ignore unhandled rejections\n    } catch (e) { // We don't want our caller to 'stop' due to a domready exception as its usually the toplevel initialization code, so log the exception ourselves\n      void Promise.reject(new Error(\"Synchronous onDomReady handler failed\", { cause: e })); //Let it propagate as a standard unhandled rejection\n    }\n  } else\n    document.addEventListener(\"DOMContentLoaded\", () => void callback());\n}\n\n//parse JSON data, throw with more info on parse failure\nexport function getJSONAttribute<T>(node: Element, attributename: string): T | null {\n  try {\n    if (node.hasAttribute(attributename))\n      return JSON.parse(node.getAttribute(attributename) as string);\n  } catch (e) {\n    console.error(\"JSON parse failure on attribute '\" + attributename + \"' of node\", node);\n    throw e;\n  }\n  return null;\n}\n\n/**\n     Get the base URI of the current document. IE11 doesn't implement document.baseURI\n *\n    @param doc - Document to query. Defaults to window.document\n    @deprecated Use document.baseURI\n */\nexport function getBaseURI(doc: Document | undefined) {\n  if (!doc)\n    doc = window.document;\n  if (doc.baseURI)\n    return doc.baseURI;\n\n  const base = doc.querySelector('base');\n  if (base && base.href)\n    return base.href;\n  return doc.URL;\n}\n\n/**\n     Sets multiple styles on a node, automatically adding 'px' to numbers when appropriate\n    (can be used as replacement for Mootools .setStyles)\n *\n * @param node - Node to update\n * @param value - Styles to set\n */\nexport function setStyles(node: HTMLElement, value?: string | { [key: string]: string | number }) {\n  if (!value)\n    node.style.cssText = '';\n  else if (typeof value === 'string')\n    node.style.cssText = value || '';\n  else {\n    for (const [key, propvalue] of Object.entries(value)) {\n      // for numbers, add 'px' if the constant isn't dimensionless (eg zIndex)\n      // eslint-disable-next-line @typescript-eslint/no-explicit-any -- we don't know which keys will be set\n      (node.style as any)[key] = typeof value[key] === 'number' && IS_NON_DIMENSIONAL.test(key) === false\n        ? propvalue + 'px'\n        : propvalue;\n    }\n  }\n}\n","import { append, setStyles } from './tree';\n\n// This will assign the `any` type to the list of accepted attributes for elements constructed via JSX syntax.\ndeclare global {\n  // eslint-disable-next-line @typescript-eslint/no-namespace\n  namespace JSX {\n    interface IntrinsicElements {\n      // eslint-disable-next-line @typescript-eslint/no-explicit-any\n      [eleName: string]: any;\n    }\n  }\n}\n\ntype CreateAttributes = {\n  // eslint-disable-next-line @typescript-eslint/no-explicit-any\n  [key: string]: any;\n  on?: {\n    [key in keyof GlobalEventHandlersEventMap]?: (evt: GlobalEventHandlersEventMap[key]) => void;\n  };\n};\n\ntype CreateElementFunction = (attributes: CreateAttributes, _1?: null, _2?: null) => HTMLElement;\n\nfunction flattenArray<T>(list: T[]): T[] {\n  return list.reduce((acc: T[], elt) => acc.concat(Array.isArray(elt) ? flattenArray(elt) : elt), []);\n}\n\nfunction setClassName(node: Element, value?: string | string[] | { [key: string]: boolean }) {\n  if (!value)\n    node.className = '';\n  else if (typeof value === 'string')\n    node.className = value || '';\n  else if (Array.isArray(value))\n    node.className = value.filter(elt => elt && typeof elt === 'string').join(\" \");\n  else if (typeof value === \"object\") {\n    let str = \"\";\n    Object.keys(value).forEach((key, idx) => { if (value[key]) str += (idx ? \" \" : \"\") + key; });\n    node.className = str;\n  }\n}\n\n/**\n     Matches non-first uppercase characters\n    (when the second char is uppercases, the first char is passed too)\n */\nconst MATCH_UPCASE = /([A-Z])/g;\nconst MATCH_DASH_AND_CHAR = /-([a-zA-Z])/g;\n\n/**\n Convert a camelCased identifier to a dashed string\n *\n * @param value - identifier to convert\n */\nexport function toDashed(value: string) {\n  if (value)\n    return (value.substring(0, 1) + value.substring(1).replace(MATCH_UPCASE, \"-$1\")).toLowerCase();\n  return \"\";\n}\n\n/**\n Convert a dashed string to a camelCase identifier\n *\n * @param value - identifier to convert\n */\nexport function toCamel(value: string) {\n  return value.replace(MATCH_DASH_AND_CHAR, (_, match_1) => match_1.toUpperCase());\n}\n\nfunction attrHasBooleanValue(propname: string) {\n  return ['disabled', 'checked', 'selected', 'readonly', 'multiple', 'ismap'].includes(propname);\n}\n\nfunction createElement(elementname: string, attributes?: CreateAttributes, toattrs?: boolean) {\n  const node = document.createElement(elementname);\n  if (attributes) {\n    Object.keys(attributes).forEach(attrname => {\n      if (attrname === 'events')\n        throw new Error(\"Use 'on' instead of 'events' in dompack.create\");\n      if (attrname === 'styles')\n        throw new Error(\"Use 'style' instead of 'styles' in dompack.create\");\n      if (attrname === 'children') {\n        // allow null 'children' property for jsxcreate, property delete is detrimental to performance.\n        if (attributes[attrname])\n          throw new Error(\"Use 'childNodes' instead of 'children' in dompack.create\");\n        return;\n      }\n\n      const value = attributes[attrname];\n\n      if (attrname === 'on') //create event listeners\n        return void Object.keys(value).forEach(eventname => node.addEventListener(eventname, value[eventname], false));\n      else if (attrname.startsWith(\"on\"))\n        return void node.addEventListener(toDashed(attrname.substring(2)), value, false);\n\n      if (attrname === \"className\" || attrname === \"class\") {\n        if (node.className) // already modified the class?\n          throw new Error(\"Specify either 'className' or 'class' to dompack.create, but not both\");\n        setClassName(node, value);\n        return;\n      }\n\n      if (attrname === 'style')\n        return void setStyles(node, value);\n\n      if (attrname === 'dataset') //explicitly assign\n        return void Object.assign(node[attrname], value);\n\n      if (attrname === 'childNodes') //append as children\n        return void append(node, ...attributes.childNodes.filter((child: Node | string | number | boolean | null) => child !== null && child !== true && child !== false && child !== undefined));\n\n      if (toattrs && attrHasBooleanValue(attrname)) {\n        if (value)\n          node.setAttribute(attrname, \"\");\n        else\n          node.removeAttribute(attrname);\n        return;\n      }\n\n      if (toattrs && !attrname.startsWith(\"prop\")) {\n        if (value !== null && value !== undefined) {\n          if (value && typeof value === \"object\")\n            throw new Error(\"Cannot store non-null objects in attributes, use a property starting with 'prop'\");\n          node.setAttribute(attrname, attributes[attrname]);\n        }\n      } else {\n        // eslint-disable-next-line @typescript-eslint/no-explicit-any -- we don't know which keys will be set\n        (node as any)[attrname] = attributes[attrname];\n      }\n    });\n  }\n  return node;\n}\n\n/* create elements. sets properties (not attributes!) immediately.\n   everything inside 'on' is added as an addEventListener without capture\n   everything inside 'childNodes' is appended to the node. nulls are ignored inside childNodes\n\n   Examples:\n   domtools.create(\"input\", { type:\"file\", className: \"myupload\", style: { display: \"none\" }));\n\n*/\nexport function create<K extends keyof HTMLElementTagNameMap>(elementname: K, attributes?: CreateAttributes, children?: Array<Node | string>): HTMLElementTagNameMap[K];\nexport function create(elementname: string, attributes?: CreateAttributes, children?: Array<Node | string>): HTMLElement;\n\nexport function create(elementname: string, attributes?: CreateAttributes, children?: Array<Node | string>) {\n  const el = createElement(elementname, attributes, false);\n  if (children?.length)\n    el.append(...children);\n  return el;\n}\n\n/** Function to create for jsx, create elements directly (instead of virtual dom nodes).\n\n    import * as dompack from 'dompack';\n\n    /* \\@jsx dompack.jsxcreate *\\/\n    /* \\@jsxFrag dompack.jsxfragment *\\/\n    your code\n\n    @param element - Name of the element to create\n    @param childNodes - Any child nodes to add immediately\n*/\nexport function jsxcreate<K extends keyof HTMLElementTagNameMap>(element: K, attributes: CreateAttributes, ...childNodes: Array<Node | string | number>): HTMLElementTagNameMap[K];\nexport function jsxcreate(element: string | CreateElementFunction, attributes: CreateAttributes, ...childNodes: Array<Node | string | number>): HTMLElement;\n\nexport function jsxcreate(element: string | CreateElementFunction, attributes: CreateAttributes, ...childNodes: Array<Node | string | number>): HTMLElement {\n  // Ensure attributes\n  attributes = attributes || {};\n  // Flatten childnodes arrays, convert numbers to strings. Also support children property (React uses that)\n  let parts: Array<Node | string | number> = (attributes.childNodes || []).concat(attributes.children || []).concat(childNodes);\n  if (attributes.children)\n    attributes.children = null;\n  parts = flattenArray(parts);\n  parts = parts.map((elt) => typeof elt === \"number\" ? String(elt) : elt);\n  // Create the element\n  attributes.childNodes = parts;\n  if (typeof element === \"function\")\n    return element(attributes, null, null);\n  return createElement(element, attributes, true);\n}\n\nexport function jsxfragment(inp: Node): DocumentFragment {\n  const frag = document.createDocumentFragment();\n  frag.append(...inp.childNodes);\n  return frag;\n}\n","import * as domtree from './tree';\nimport * as domevents from './events';\n\nexport type TakeFocusEvent = CustomEvent<{ options?: FocusOptions }>;\n\ndeclare global {\n  interface GlobalEventHandlersEventMap {\n    \"dompack:takefocus\": TakeFocusEvent;\n  }\n}\n\ntype RegistrationHandler<E extends Element = HTMLElement> = (node: E, index?: number) => void;\ntype ComponentRegistration<E extends Element = HTMLElement> =\n  {\n    selector: string;\n    handler: RegistrationHandler<E>;\n    index: number;\n    num: number;\n    afterdomready: boolean;\n  };\n\nconst components: Array<ComponentRegistration<Element>> = [];\nconst map = new WeakMap();\n\n//is a node completely in the dom? if we can find a sibling anywhere, it must be closed\nfunction isNodeCompletelyInDom(node: Element | null) {\n  for (; node; node = node.parentElement)\n    if (node.nextSibling)\n      return true;\n  return false;\n}\nfunction processRegistration<E extends Element>(item: E, reg: ComponentRegistration<E>, domready: boolean) {\n  if (!domready && !isNodeCompletelyInDom(item))\n    return; //not safe to register\n\n  if (!map.has(item))\n    map.set(item, [reg.num]);\n  else {\n    const list = map.get(item);\n    if (list.includes(reg.num))\n      return;\n    list.push(reg.num);\n  }\n  reg.handler(item, reg.index++); //note: if an exception is reported from Object.handler,\n}\nfunction applyRegistration<E extends Element>(reg: ComponentRegistration<E>, startnode?: Element) {\n  const domready = domtree.isDomReady();\n  if (reg.afterdomready && !domready)\n    return;\n\n  const items = Array.from((startnode || document).querySelectorAll(reg.selector)) as E[];\n  if (startnode?.matches(reg.selector))\n    items.unshift(startnode as E); //if startnode matches it has to be an E\n\n  items.forEach(item => {\n    try {\n      processRegistration<E>(item, reg, domready);\n    } catch (e) {\n      console.error(\"Exception handling registration of\", item, \"for rule\", reg.selector);\n      console.log(\"Registration\", reg);\n      if (e instanceof Error) {\n        console.log(e, e.stack);\n        if (window.onerror) {\n          // Send to onerror to trigger exception reporting\n          try {\n            // @ts-ignore: fileName, lineNumber and columnNumber are non-standard\n            window.onerror(e.message, e.fileName || \"\", e.lineNumber || 1, e.columNumber || 1, e);\n          } catch (e2) { }\n        }\n      } else\n        console.log(e);\n    }\n  });\n}\n\n/* A focus implementation that allows the node to intercept focused, allowing eg\n   radio/checkbox replacements to redirect focus but also explicitly preventing\n   focus of a disabled element\n   Returns true when the focus operation was successfull or handled by an event handler.\n   @param node Node to focus\n   @param options.preventScroll Prevent scroll to focused element\n*/\nexport function focus(node: Element, options?: FocusOptions) {\n  if (!domevents.dispatchCustomEvent(node, 'dompack:takefocus', { bubbles: true, cancelable: true, detail: { options } }))\n    return true;\n\n  if (typeof (node as HTMLElement).focus !== \"function\" || (node as HTMLInputElement).disabled)\n    return false;\n\n  (node as HTMLInputElement).focus(options);\n  return true;\n}\n\n/**\n * @param node - Node to show\n * @param options - Scroll options\n * @deprecated invoke scrollIntoView directly  on the nodes\n */\nexport function scrollIntoView(node: Element, options?: ScrollIntoViewOptions) {\n  node.scrollIntoView(options);\n  return true;\n}\n\n/**\n    Register a component for auto-initialization.\n\n    @param selector - Selector the component must match\n    @param handler - Handler\n    @param options - Any unrecognized options are passed to the handler\n\n    The handler will be invoked with two parameters\n    - the node to register\n    - the index of the node (a unique counter for this selector - first is 0)\n */\n\nexport function register<E extends Element = HTMLElement>(selector: string, handler: RegistrationHandler<E>, options?: { afterdomready: boolean }) {\n  const newreg: ComponentRegistration<E> =\n  {\n    selector: selector,\n    handler: handler,\n    index: 0,\n    num: components.length,\n    afterdomready: !options || options.afterdomready\n  };\n  if (components.length === 0 && !domtree.isDomReady()) //first component... we'll need a ready handler\n    domtree.onDomReady(() => registerMissed());\n\n  components.push(newreg as ComponentRegistration<Element>);\n  applyRegistration(newreg);\n}\n\n// register any components we missed on previous scans\nexport function registerMissed(startnode?: Element) {\n  const todo = components.slice(0);\n  todo.forEach(item => applyRegistration(item, startnode));\n}\n","export type CookieOptions = {\n  path?: string;\n  domain?: string | null;\n  expires?: Date | Temporal.Instant | null;\n  duration?: number | null;\n  secure?: boolean;\n  sameSite?: \"Strict\" | \"Lax\" | \"None\";\n};\n\nexport type ServersideCookieOptions = CookieOptions & {\n  httpOnly?: boolean;\n};\n\nexport function buildCookieHeader(name: string, value: string, options?: ServersideCookieOptions) {\n  let header = `${name}=${encodeURIComponent(value)}`;\n  if (options?.domain)\n    header += ';domain=' + options?.domain;\n  header += ';path=' + (options?.path || '/');\n  if (!value) //clearing a cookie, so ignore expires/duration and just set 1970\n    header += ';expires=Thu, 01 Jan 1970 00:00:00 GMT';\n  else if (options?.expires) //we need toUTCString to give us the proper Date formatting\n    header += ';expires=' + (\"epochMilliseconds\" in options.expires ? new Date(options.expires.epochMilliseconds) : options.expires).toUTCString();\n  else if (options?.duration) {\n    const date = new Date();\n    date.setTime(date.getTime() + options?.duration * 24 * 60 * 60 * 1000);\n    header += ';expires=' + date.toUTCString();\n  }\n  if (options?.secure)\n    header += ';secure';\n  if (options?.sameSite)\n    header += ';SameSite=' + options?.sameSite;\n  if (options?.httpOnly) //NOTE browsers cannot effectively set this\n    header += ';HttpOnly';\n  return header;\n}\n","import { buildCookieHeader, type CookieOptions } from '@webhare/dompack/src/cookiebuilder';\nimport { escapeRegExp, parseTyped, stringify } from '@webhare/std';\n\nconst backup: {\n  sessionStorage?: Record<string, string>;\n  localStorage?: Record<string, string>;\n} = {};\ntype BrowserStorage = keyof typeof backup;\n\nexport function listCookies() {\n  return document.cookie.split(';').map(cookie => {\n    const parts = cookie.split('=');\n    return { name: decodeURIComponent(parts[0].trim()), value: decodeURIComponent(parts[1] || '') };\n  });\n}\n\nexport function setCookie(key: string, value: string, options?: CookieOptions) {\n  document.cookie = buildCookieHeader(key, value, options);\n}\n\nexport function getCookie(key: string): string | null {\n  const value = document.cookie.match('(?:^|;)\\\\s*' + escapeRegExp(key) + '=([^;]*)');\n  return (value) ? decodeURIComponent(value[1]) : null;\n}\n\nexport function deleteCookie(key: string, options?: CookieOptions) {\n  setCookie(key, '', { ...options, duration: -1, expires: undefined });\n}\n\nfunction get(storage: BrowserStorage, key: string): unknown | null {\n  let foundvalue: string | undefined | null = backup[storage]?.[key];\n  if (foundvalue) //if it's in the backups storage encoding can't fail either\n    return parseTyped(foundvalue);\n\n  try {\n    foundvalue = window[storage].getItem(key);\n    if (foundvalue)\n      return parseTyped(foundvalue);\n  } catch (e) {\n    //we ignore parse failures\n  }\n  return null;\n}\n\nfunction set(storage: BrowserStorage, key: string, value: unknown) {\n  const tostore = value !== null && value !== undefined ? stringify(value, { typed: true }) : null;\n  try {\n    if (tostore !== null)\n      window[storage].setItem(key, tostore);\n    else\n      window[storage].removeItem(key);\n    return;\n  } catch (e) {\n    //ignore\n  }\n\n  const store = (backup[storage] ||= {});\n  if (tostore !== null)\n    store[key] = tostore;\n  else\n    delete store[key];\n}\n\nexport function getSession<T>(key: string): T | null {\n  return get(\"sessionStorage\", key) as T | null;\n}\nexport function setSession<T>(key: string, value: T | null) {\n  set(\"sessionStorage\", key, value);\n}\nexport function getLocal<T>(key: string): T | null {\n  return get(\"localStorage\", key) as T | null;\n}\nexport function setLocal<T>(key: string, value: T | null) {\n  set(\"localStorage\", key, value);\n}\n","/** Load an image\n * @param imgsrc - The image source URL\n * @returns A promise resolving to the image node\n */\nexport function loadImage(imgsrc: string): Promise<HTMLImageElement> {\n  return new Promise((resolve, reject) => {\n    const img = new Image;\n    img.onload = () => {\n      resolve(img);\n    };\n    img.onerror = reject;\n    img.src = imgsrc;\n  });\n}\n\n/** Load a JavaScript file and add it to the DOM\n * @param scriptsrc - The script source URL\n * @param module - Load as module\n * @returns A promise resolving to the script node\n */\nexport function loadScript(scriptsrc: string, { module = false } = {}): Promise<HTMLScriptElement> {\n  return new Promise((resolve, reject) => {\n    const scripttag = document.createElement('script');\n    scripttag.onload = () => {\n      resolve(scripttag);\n    };\n    scripttag.onerror = reject;\n    scripttag.src = scriptsrc;\n    if (module)\n      scripttag.type = \"module\";\n\n    document.querySelector('head,body')?.appendChild(scripttag);\n  });\n}\n\n/** Load a CSS file and add it to the DOM\n * @param src - The CSS source URL\n * @returns A promise resolving to the link node\n*/\nexport function loadCSS(src: string): Promise<HTMLLinkElement> {\n  const element = document.createElement('link');\n  element.type = 'text/css';\n  element.rel = 'stylesheet';\n  element.href = src;\n\n  const retval = new Promise<HTMLLinkElement>((resolve, reject) => {\n    element.onload = () => resolve(element);\n    element.onerror = reject;\n  });\n\n  document.querySelector('head,body')?.appendChild(element);\n  return retval;\n}\n","/* Identify devices for the purpose of analytics/tracing\n   NOT a library for feature detection!\n   Originally based on Mootools.Browser\n*/\n\nexport type Platform = \"windows\" | \"ios\" | \"webos\" | \"android\" | \"linux\" | \"mac\" | \"other\";\nexport type Device = \"desktop\" | \"mobile\" | \"tablet\" | \"\";\n\n/** Valid KeyboardEvent.key values (next to 'plain' keys)\n    This list was built from https://www.w3.org/TR/uievents-key/\n    using `console.log([...new Set([...document.querySelectorAll(\".key-table-key .key\")].map(_ => _.textContent))].toSorted().join(' | '))`\n*/\nexport type KeyAttributeValue = \"AVRInput\" | \"AVRPower\" | \"Accept\" | \"Again\" | \"AllCandidates\" | \"Alphanumeric\" | \"Alt\" | \"AltGraph\" | \"AppSwitch\" | \"ArrowDown\" | \"ArrowLeft\" | \"ArrowRight\" | \"ArrowUp\" | \"Attn\" | \"AudioBalanceLeft\" | \"AudioBalanceRight\" | \"AudioBassBoostDown\" | \"AudioBassBoostToggle\" | \"AudioBassBoostUp\" | \"AudioFaderFront\" | \"AudioFaderRear\" | \"AudioSurroundModeNext\" | \"AudioTrebleDown\" | \"AudioTrebleUp\" | \"AudioVolumeDown\" | \"AudioVolumeMute\" | \"AudioVolumeUp\" | \"Backspace\" | \"BrightnessDown\" | \"BrightnessUp\" | \"BrowserBack\" | \"BrowserFavorites\" | \"BrowserForward\" | \"BrowserHome\" | \"BrowserRefresh\" | \"BrowserSearch\" | \"BrowserStop\" | \"Call\" | \"Camera\" | \"CameraFocus\" | \"Cancel\" | \"CapsLock\" | \"ChannelDown\" | \"ChannelUp\" | \"Clear\" | \"Close\" | \"ClosedCaptionToggle\" | \"CodeInput\" | \"ColorF0Red\" | \"ColorF1Green\" | \"ColorF2Yellow\" | \"ColorF3Blue\" | \"ColorF4Grey\" | \"ColorF5Brown\" | \"Compose\" | \"ContextMenu\" | \"Control\" | \"Convert\" | \"Copy\" | \"CrSel\" | \"Cut\" | \"DVR\" | \"Dead\" | \"Delete\" | \"Dimmer\" | \"DisplaySwap\" | \"Eisu\" | \"Eject\" | \"End\" | \"EndCall\" | \"Enter\" | \"EraseEof\" | \"Escape\" | \"ExSel\" | \"Execute\" | \"Exit\" | \"F1\" | \"F10\" | \"F11\" | \"F12\" | \"F2\" | \"F3\" | \"F4\" | \"F5\" | \"F6\" | \"F7\" | \"F8\" | \"F9\" | \"FavoriteClear0\" | \"FavoriteClear1\" | \"FavoriteClear2\" | \"FavoriteClear3\" | \"FavoriteRecall0\" | \"FavoriteRecall1\" | \"FavoriteRecall2\" | \"FavoriteRecall3\" | \"FavoriteStore0\" | \"FavoriteStore1\" | \"FavoriteStore2\" | \"FavoriteStore3\" | \"FinalMode\" | \"Find\" | \"Fn\" | \"FnLock\" | \"GoBack\" | \"GoHome\" | \"GroupFirst\" | \"GroupLast\" | \"GroupNext\" | \"GroupPrevious\" | \"Guide\" | \"GuideNextDay\" | \"GuidePreviousDay\" | \"HangulMode\" | \"HanjaMode\" | \"Hankaku\" | \"HeadsetHook\" | \"Help\" | \"Hibernate\" | \"Hiragana\" | \"HiraganaKatakana\" | \"Home\" | \"Hyper\" | \"Info\" | \"Insert\" | \"InstantReplay\" | \"JunjaMode\" | \"KanaMode\" | \"KanjiMode\" | \"Katakana\" | \"Key11\" | \"Key12\" | \"LastNumberRedial\" | \"LaunchApplication1\" | \"LaunchApplication2\" | \"LaunchCalendar\" | \"LaunchContacts\" | \"LaunchMail\" | \"LaunchMediaPlayer\" | \"LaunchMusicPlayer\" | \"LaunchPhone\" | \"LaunchScreenSaver\" | \"LaunchSpreadsheet\" | \"LaunchWebBrowser\" | \"LaunchWebCam\" | \"LaunchWordProcessor\" | \"Link\" | \"ListProgram\" | \"LiveContent\" | \"Lock\" | \"LogOff\" | \"MailForward\" | \"MailReply\" | \"MailSend\" | \"MannerMode\" | \"MediaApps\" | \"MediaAudioTrack\" | \"MediaClose\" | \"MediaFastForward\" | \"MediaLast\" | \"MediaNextTrack\" | \"MediaPause\" | \"MediaPlay\" | \"MediaPlayPause\" | \"MediaPreviousTrack\" | \"MediaRecord\" | \"MediaRewind\" | \"MediaSkipBackward\" | \"MediaSkipForward\" | \"MediaStepBackward\" | \"MediaStepForward\" | \"MediaStop\" | \"MediaTopMenu\" | \"MediaTrackNext\" | \"MediaTrackPrevious\" | \"Meta\" | \"MicrophoneToggle\" | \"MicrophoneVolumeDown\" | \"MicrophoneVolumeMute\" | \"MicrophoneVolumeUp\" | \"ModeChange\" | \"NavigateIn\" | \"NavigateNext\" | \"NavigateOut\" | \"NavigatePrevious\" | \"New\" | \"NextCandidate\" | \"NextFavoriteChannel\" | \"NextUserProfile\" | \"NonConvert\" | \"Notification\" | \"NumLock\" | \"OnDemand\" | \"Open\" | \"PageDown\" | \"PageUp\" | \"Pairing\" | \"Paste\" | \"Pause\" | \"PinPDown\" | \"PinPMove\" | \"PinPToggle\" | \"PinPUp\" | \"Play\" | \"PlaySpeedDown\" | \"PlaySpeedReset\" | \"PlaySpeedUp\" | \"Power\" | \"PowerOff\" | \"PreviousCandidate\" | \"Print\" | \"PrintScreen\" | \"Process\" | \"Props\" | \"RandomToggle\" | \"RcLowBattery\" | \"RecordSpeedNext\" | \"Redo\" | \"RfBypass\" | \"Romaji\" | \"STBInput\" | \"STBPower\" | \"Save\" | \"ScanChannelsToggle\" | \"ScreenModeNext\" | \"ScrollLock\" | \"Select\" | \"Settings\" | \"Shift\" | \"SingleCandidate\" | \"Soft1\" | \"Soft2\" | \"Soft3\" | \"Soft4\" | \"SpeechCorrectionList\" | \"SpeechInputToggle\" | \"SpellCheck\" | \"SplitScreenToggle\" | \"Standby\" | \"Subtitle\" | \"Super\" | \"Symbol\" | \"SymbolLock\" | \"TV\" | \"TV3DMode\" | \"TVAntennaCable\" | \"TVAudioDescription\" | \"TVAudioDescriptionMixDown\" | \"TVAudioDescriptionMixUp\" | \"TVContentsMenu\" | \"TVDataService\" | \"TVInput\" | \"TVInputComponent1\" | \"TVInputComponent2\" | \"TVInputComposite1\" | \"TVInputComposite2\" | \"TVInputHDMI1\" | \"TVInputHDMI2\" | \"TVInputHDMI3\" | \"TVInputHDMI4\" | \"TVInputVGA1\" | \"TVMediaContext\" | \"TVNetwork\" | \"TVNumberEntry\" | \"TVPower\" | \"TVRadioService\" | \"TVSatellite\" | \"TVSatelliteBS\" | \"TVSatelliteCS\" | \"TVSatelliteToggle\" | \"TVTerrestrialAnalog\" | \"TVTerrestrialDigital\" | \"TVTimer\" | \"Tab\" | \"Teletext\" | \"Undo\" | \"Unidentified\" | \"VideoModeNext\" | \"VoiceDial\" | \"WakeUp\" | \"Wink\" | \"Zenkaku\" | \"ZenkakuHankaku\" | \"ZoomIn\" | \"ZoomOut\" | \"ZoomToggle\";\n\nexport type UserAgentInfo =\n  {\n    /** Browser name, eg 'chrome' or 'firefox' */\n    name: string;\n    /** Browser numeric version (eg 97) */\n    version: number;\n    /** Platform the browser is running on (eg 'windows') */\n    platform: Platform;\n    /** Type of device (eg 'desktop' or 'tablet' for an iPad) */\n    device: Device;\n    /** platform-browsername-version eg ios-safari-11 */\n    triplet: string;\n  };\n\nexport function parseUserAgent(ua: string): UserAgentInfo {\n  ua = ua.toLowerCase();\n\n  // chrome is included in the edge UA, so need to check for edge first, before checking if it's chrome.\n  // safari is included in the miuibrowser UA, so need to check for miuibrowser first, before checking if it's safari.\n  let UA: RegExpMatchArray | null = ua.match(/(edge|miuibrowser)[\\s/:]([\\w\\d.]+)/);\n  if (!UA)\n    UA = ua.match(/(opera|ie|firefox|chrome|trident|crios|version)[\\s/:]([\\w\\d.]+)?.*?(safari|(?:rv[\\s/:]|version[\\s/:])([\\w\\d.]+)|$)/);\n  if (!UA) { //try ios 11.4.1\n    UA = ua.match(/; cpu os ([\\d]+)/);\n    if (UA)\n      UA = ['', 'safari', UA[1]];\n  }\n  if (!UA)\n    UA = ['', 'unknown', \"0\"];\n\n  if (UA[1] === 'trident') {\n    UA[1] = 'ie';\n    if (UA[4]) UA[2] = UA[4];\n  } else if (UA[1] === 'crios') {\n    UA[1] = 'chrome';\n  }\n\n  let platform = ua.match(/ip(?:ad|od|hone)/) ? 'ios' : (ua.match(/(?:webos|android)/) || ua.match(/mac|win|linux/) || ['other'])[0];\n  if (platform === 'win')\n    platform = 'windows';\n\n  const name = (UA[1] === 'version') ? UA[3] : UA[1];\n  const version = parseInt((UA[1] === 'opera' && UA[4]) ? UA[4] : UA[2]);\n  const device = ua.match(/ipad/) ? 'tablet' : ['ios', 'webos', 'android'].includes(platform) ? 'mobile' : ['mac', 'windows', 'linux'].includes(platform) ? 'desktop' : '';\n\n  return {\n    name,\n    version,\n    platform: platform as Platform,\n    device,\n    triplet: platform + '-' + name + '-' + version\n  };\n}\n\n/**\n * Is the native 'multiselect' modifier for this platform pressed? (Cmd for Mac, Ctrl for Windows/Linux)\n *\n * @param event - Event to check\n */\nexport function isMultiSelectKey(event: KeyboardEvent | MouseEvent): boolean {\n  return browser.platform === 'mac' ? event.metaKey : event.ctrlKey;\n}\n\n/**\n * Is the native 'copy' modifier for this platform pressed?\n *\n * @param event - Event to check\n */\nexport function isCopyKey(event: KeyboardEvent | MouseEvent): boolean {\n  return browser.platform === 'mac' ? event.altKey : event.ctrlKey;\n}\n\n/** @deprecated Use getBrowser() instead, available since WH5.6.3. We will remove `browser` in the future to improve tree shaking */\nexport const browser: Readonly<UserAgentInfo> = Object.freeze(parseUserAgent(globalThis.navigator?.userAgent || \"\"));\n\n/** Get browser information */\nexport function getBrowser(): Readonly<UserAgentInfo> {\n  //Offer this as a function to improve treeshakability\n  return browser;\n}\n\n//CSS Query used to find valid submittors\nexport const submitselector = 'input[type=submit],input[type=image],button[type=submit],button:not([type])';\n//Must match possible types returned by the submitselector\nexport type SubmitSelectorType = HTMLInputElement | HTMLButtonElement;\n","// This gets TypeScript to refer to us by our @webhare/... name in auto imports:\ndeclare module \"@webhare/dompack\" {\n}\n\n/* This is our public api */\nexport { flagUIBusy, setupBusyModal } from './busy';\nexport type { UIBusyLock, BusyModalEvent } from './busy';\nexport { dispatchCustomEvent, dispatchDomEvent, fireModifiedEvents, changeValue, stop, addDocEventListener } from '../../../modules/system/js/dompack/src/events';\nexport type { DocEvent, DomEventOptions } from '../../../modules/system/js/dompack/src/events';\nexport { qS, qR, qSA, isFormControl, isEditControl, isElement, isHTMLElement, getRelativeBounds } from './tree';\nexport type { FormControlElement, Rect } from './tree';\nexport { isDomReady, onDomReady } from '../../../modules/system/js/dompack/src/tree';\nexport { create, jsxcreate, jsxfragment } from '../../../modules/system/js/dompack/src/create';\nexport { focus, register, registerMissed } from '../../../modules/system/js/dompack/src/components';\nexport type { TakeFocusEvent } from '../../../modules/system/js/dompack/src/components';\nexport { getLocal, setLocal, getSession, setSession, listCookies, getCookie, setCookie, deleteCookie } from './storage';\nexport { loadImage, loadScript, loadCSS } from \"./preload\";\nexport { browser, getBrowser, isMultiSelectKey, isCopyKey } from './browser';\nexport type { KeyAttributeValue, UserAgentInfo, Platform, Device } from './browser';\n\n/** @deprecated As 'Lock' conflicts, in WH5.4+ you can safely switch to UIBusyLock */\nimport type { UIBusyLock } from './busy';\nexport type Lock = UIBusyLock;\n","/* This is our public api. Any direct inclusions from src/.... not mentioned here are not a stable API\n*/\n\nexport { createDeferred } from '@webhare/std';\nexport { flagUIBusy, getRelativeBounds, type Rect } from '@webhare/dompack';\nexport {\n  dispatchCustomEvent, dispatchDomEvent, fireModifiedEvents, changeValue, normalizeKeyboardEventData,\n  stop\n} from './src/events';\nexport {\n  qS, qSA, contains, closest, matches,\n  empty, isDomReady, onDomReady, getJSONAttribute,\n  before, after, replaceWith, remove, prepend, append,\n  toggleClass, toggleClasses,\n  setStyles,\n  getBaseURI,\n} from './src/tree';\nexport { create, jsxcreate, jsxfragment } from './src/create';\nexport { focus, register, registerMissed, scrollIntoView } from './src/components';\nexport type { TakeFocusEvent } from './src/components';\n\nimport { debugFlags } from \"@webhare/env\";\n\n/** @deprecated You should use debugFlags from \\@webhare/env */\nexport const debugflags = debugFlags;\n","/* @recommendedimport: import * as browser from 'dompack/extra/browser';\n\n   Identify devices for the purpose of analytics/tracing\n   NOT a library for feature detection!\n\n   Based on Mootools.Browser\n*/\n/* eslint no-useless-escape: off */\n\ntype Platform = \"windows\" | \"ios\" | \"webos\" | \"android\" | \"linux\" | \"mac\" | \"other\";\n\nexport type UserAgentInfo =\n  {\n    name: string;\n    version: number;\n    platform: Platform;\n    device: \"desktop\" | \"mobile\" | \"tablet\" | \"\";\n  };\n\nexport function parseUserAgent(ua: string): UserAgentInfo {\n  ua = ua.toLowerCase();\n\n  // chrome is included in the edge UA, so need to check for edge first, before checking if it's chrome.\n  // safari is included in the miuibrowser UA, so need to check for miuibrowser first, before checking if it's safari.\n  let UA: RegExpMatchArray | null = ua.match(/(edge|miuibrowser)[\\s\\/:]([\\w\\d\\.]+)/);\n  if (!UA)\n    UA = ua.match(/(opera|ie|firefox|chrome|trident|crios|version)[\\s\\/:]([\\w\\d\\.]+)?.*?(safari|(?:rv[\\s\\/:]|version[\\s\\/:])([\\w\\d\\.]+)|$)/);\n  if (!UA) { //try ios 11.4.1\n    UA = ua.match(/; cpu os ([\\d]+)/);\n    if (UA)\n      UA = ['', 'safari', UA[1]];\n  }\n  if (!UA)\n    UA = ['', 'unknown', \"0\"];\n\n  if (UA[1] === 'trident') {\n    UA[1] = 'ie';\n    if (UA[4]) UA[2] = UA[4];\n  } else if (UA[1] === 'crios') {\n    UA[1] = 'chrome';\n  }\n\n  let platform = ua.match(/ip(?:ad|od|hone)/) ? 'ios' : (ua.match(/(?:webos|android)/) || ua.match(/mac|win|linux/) || ['other'])[0];\n  if (platform === 'win') platform = 'windows';\n\n  const ret: UserAgentInfo =\n  {\n    name: (UA[1] === 'version') ? UA[3] : UA[1],\n    version: parseInt((UA[1] === 'opera' && UA[4]) ? UA[4] : UA[2]),\n    platform: platform as Platform,\n    device: ua.match(/ipad/) ? 'tablet' : ['ios', 'webos', 'android'].includes(platform) ? 'mobile' : ['mac', 'windows', 'linux'].includes(platform) ? 'desktop' : ''\n  };\n\n  return ret;\n}\n\n//module.exports =\nconst browser = parseUserAgent(globalThis.navigator?.userAgent || \"\");\n\nexport function getName() {\n  return browser.name;\n}\nexport function getVersion() {\n  return browser.version;\n}\nexport function getPlatform() {\n  return browser.platform;\n}\nexport function getDevice() {\n  return browser.device;\n}\nexport function getTriplet() {\n  return browser.platform + '-' + browser.name + '-' + browser.version;\n}\n","var UNKNOWN_FUNCTION = '<unknown>';\n/**\n * This parses the different stack traces and puts them into one format\n * This borrows heavily from TraceKit (https://github.com/csnover/TraceKit)\n */\n\nfunction parse(stackString) {\n  var lines = stackString.split('\\n');\n  return lines.reduce(function (stack, line) {\n    var parseResult = parseChrome(line) || parseWinjs(line) || parseGecko(line) || parseNode(line) || parseJSC(line);\n\n    if (parseResult) {\n      stack.push(parseResult);\n    }\n\n    return stack;\n  }, []);\n}\nvar chromeRe = /^\\s*at (.*?) ?\\(((?:file|https?|blob|chrome-extension|native|eval|webpack|rsc|<anonymous>|\\/|[a-z]:\\\\|\\\\\\\\).*?)(?::(\\d+))?(?::(\\d+))?\\)?\\s*$/i;\nvar chromeEvalRe = /\\((\\S*)(?::(\\d+))(?::(\\d+))\\)/;\n\nfunction parseChrome(line) {\n  var parts = chromeRe.exec(line);\n\n  if (!parts) {\n    return null;\n  }\n\n  var isNative = parts[2] && parts[2].indexOf('native') === 0; // start of line\n\n  var isEval = parts[2] && parts[2].indexOf('eval') === 0; // start of line\n\n  var submatch = chromeEvalRe.exec(parts[2]);\n\n  if (isEval && submatch != null) {\n    // throw out eval line/column and use top-most line/column number\n    parts[2] = submatch[1]; // url\n\n    parts[3] = submatch[2]; // line\n\n    parts[4] = submatch[3]; // column\n  }\n\n  return {\n    file: !isNative ? parts[2] : null,\n    methodName: parts[1] || UNKNOWN_FUNCTION,\n    arguments: isNative ? [parts[2]] : [],\n    lineNumber: parts[3] ? +parts[3] : null,\n    column: parts[4] ? +parts[4] : null\n  };\n}\n\nvar winjsRe = /^\\s*at (?:((?:\\[object object\\])?.+) )?\\(?((?:file|ms-appx|https?|webpack|rsc|blob):.*?):(\\d+)(?::(\\d+))?\\)?\\s*$/i;\n\nfunction parseWinjs(line) {\n  var parts = winjsRe.exec(line);\n\n  if (!parts) {\n    return null;\n  }\n\n  return {\n    file: parts[2],\n    methodName: parts[1] || UNKNOWN_FUNCTION,\n    arguments: [],\n    lineNumber: +parts[3],\n    column: parts[4] ? +parts[4] : null\n  };\n}\n\nvar geckoRe = /^\\s*(.*?)(?:\\((.*?)\\))?(?:^|@)((?:file|https?|blob|chrome|webpack|rsc|resource|\\[native).*?|[^@]*bundle)(?::(\\d+))?(?::(\\d+))?\\s*$/i;\nvar geckoEvalRe = /(\\S+) line (\\d+)(?: > eval line \\d+)* > eval/i;\n\nfunction parseGecko(line) {\n  var parts = geckoRe.exec(line);\n\n  if (!parts) {\n    return null;\n  }\n\n  var isEval = parts[3] && parts[3].indexOf(' > eval') > -1;\n  var submatch = geckoEvalRe.exec(parts[3]);\n\n  if (isEval && submatch != null) {\n    // throw out eval line/column and use top-most line number\n    parts[3] = submatch[1];\n    parts[4] = submatch[2];\n    parts[5] = null; // no column when eval\n  }\n\n  return {\n    file: parts[3],\n    methodName: parts[1] || UNKNOWN_FUNCTION,\n    arguments: parts[2] ? parts[2].split(',') : [],\n    lineNumber: parts[4] ? +parts[4] : null,\n    column: parts[5] ? +parts[5] : null\n  };\n}\n\nvar javaScriptCoreRe = /^\\s*(?:([^@]*)(?:\\((.*?)\\))?@)?(\\S.*?):(\\d+)(?::(\\d+))?\\s*$/i;\n\nfunction parseJSC(line) {\n  var parts = javaScriptCoreRe.exec(line);\n\n  if (!parts) {\n    return null;\n  }\n\n  return {\n    file: parts[3],\n    methodName: parts[1] || UNKNOWN_FUNCTION,\n    arguments: [],\n    lineNumber: +parts[4],\n    column: parts[5] ? +parts[5] : null\n  };\n}\n\nvar nodeRe = /^\\s*at (?:((?:\\[object object\\])?[^\\\\/]+(?: \\[as \\S+\\])?) )?\\(?(.*?):(\\d+)(?::(\\d+))?\\)?\\s*$/i;\n\nfunction parseNode(line) {\n  var parts = nodeRe.exec(line);\n\n  if (!parts) {\n    return null;\n  }\n\n  return {\n    file: parts[2],\n    methodName: parts[1] || UNKNOWN_FUNCTION,\n    arguments: [],\n    lineNumber: +parts[3],\n    column: parts[4] ? +parts[4] : null\n  };\n}\n\nexport { parse };\n","import * as stacktrace_parser from \"stacktrace-parser\";\n\nexport type StackTraceItem = {\n  filename: string;\n  line: number;\n  col: number;\n  func: string;\n};\n\nexport type StackTrace = StackTraceItem[];\n\nexport function parseTrace(e: Error): StackTrace {\n  if (!e?.stack)\n    return [];\n\n  const trace = stacktrace_parser.parse(e.stack);\n  return trace.map(i => ({ filename: i.file || \"\", line: i.lineNumber || 1, col: i.column || 1, func: (i.methodName || \"\") }));\n}\n\nexport function getStackTrace(): StackTrace {\n  return parseTrace(new Error()).slice(1);\n}\n\nexport function formatTrace(trace: StackTraceItem[]): string {\n  return trace.map(e =>\n    `    at ${e.func} (${e.filename}:${e.line}:${e.col})`).join(\"\\n\");\n}\n\nexport function prependStackTrace(error: Error, trace: StackTrace) {\n  if (!error.stack)\n    return; //no stack to manipulate\n\n  const stacklines = error.stack?.split(\"\\n\") || [];\n  const tracelines = formatTrace(trace);\n  error.stack = (stacklines[0] ? stacklines[0] + \"\\n\" : \"\") + tracelines + '\\n' + (stacklines.slice(1).join(\"\\n\"));\n  return error;\n}\n","import { levenshteinDistance } from \"@webhare/std\";\n\n/** Find the best match for a string fiven a list of alternatives. Defaults to case insensitive match.\n    @param name - Name to search\n    @param alternatives - Alternatives to choose from\n    @param matchCase - Do a case sensitive compare (defaults to FALSE)\n    @returns Best match or null if none found\n*/\nexport function getBestMatch(name: string, alternatives: string[], { matchCase = false } = {}): string | null {\n  if (!matchCase)\n    name = name.toLowerCase();\n\n  const threshold = name.length < 3 ? 1 : 2;\n  const matches = alternatives.\n    map((value) => ({ value, distance: levenshteinDistance(name, matchCase ? value : value.toLowerCase()) })).\n    filter((match) => match.distance <= threshold).\n    sort((a, b) => a.distance - b.distance);\n\n  return matches[0]?.value || null;\n}\n\n/** Return match info as text to append to the end of a message\n    @param name - Name to search\n    @param alternatives - Alternatives to choose from\n    @param matchCase - Do a case sensitive compare (defaults to FALSE)\n    @returns Returns \", did you mean 'XXX'?\" text if a better match was found, otherwise an empty string\n    @example throw new Error(`${libname} does not export '${name}'${addBestMatch(name, Object.keys(lib))}`);\n\n*/\nexport function addBestMatch(name: string, alternatives: string[], { matchCase = false } = {}): string {\n  const match = getBestMatch(name, alternatives, { matchCase });\n  return match ? `, did you mean '${match}'?` : \"\";\n}\n","// This gets TypeScript to refer to us by our @webhare/... name in auto imports:\ndeclare module \"@webhare/js-api-tools\" {\n}\n\nexport { parseTrace, getStackTrace, prependStackTrace } from \"./stacktracing\";\nexport { getBestMatch, addBestMatch } from \"./levenshtein\";\nexport type { StackTrace, StackTraceItem } from \"./stacktracing\";\nexport type { DisallowExtraPropsRecursive, RecursiveReadonly, PromisifyFunctionReturnType, RecursivePartial } from \"./utility-types\";\n\nimport type { RecursiveReadonly } from \"./utility-types\";\n/** @deprecated Switch to RecursiveReadonly in WH5.6 (matches TS Readonly casing) */\nexport type RecursiveReadOnly<T> = RecursiveReadonly<T>;\n","// This gets TypeScript to refer to us by our @webhare/... name in auto imports:\ndeclare module \"@webhare/jsonrpc-client\" {\n}\n\nimport { debugFlags, backendBase } from \"@webhare/env\";\nimport { type StackTrace, parseTrace, prependStackTrace, type PromisifyFunctionReturnType } from \"@webhare/js-api-tools\";\n\n//just number RPCs globally instead of per server, makes debug ouput more useful\nlet globalseqnr = 0;\n\nexport interface RPCCallOptions {\n  /** Append function name to URLs. Purely for better logging/visibility as the WebHare JSONRPC server will ignore it */\n  addfunctionname?: boolean;\n  /** Custom URL marker */\n  urlappend?: string;\n  /** Call timeout */\n  timeout?: number;\n  /** Abort signal to cancel the RPC */\n  signal?: AbortSignal;\n  /** Retry on 429 */\n  retry429?: boolean;\n  /** Silent - do not log errors */\n  silent?: boolean;\n  /** Debug (Follows 'rpc' debugflag if not explicity specified) */\n  debug?: boolean;\n  /** Wrap result with response info */\n  wrapresult?: boolean;\n  keepalive?: boolean;\n  /** Headers to submit (Eg Authorization) */\n  headers?: Record<string, string>;\n  /** Base URL for service paths */\n  baseUrl?: string;\n  /** Overide fetch credentials option (defaults to same-origin) */\n  credentials?: RequestCredentials;\n}\n\nexport type RequestID = number | string | null;\n\nexport interface JSONRPCBaseResponse {\n  id: RequestID;\n  debug?: {\n    consoleLog: Array<{\n      //TODO this is Serialized<ConsoleLogItem[]> - should probably rename it to SerializedToJSON and move to std or env ?\n      /** Date when console function was called */\n      when: string;\n      /** `console` method that was called (eg 'log') */\n      method: string;\n      /** Logged data */\n      data: string;\n      /** Location of caller */\n      location?: {\n        filename: string;\n        line: number;\n        col: number;\n        func: string;\n      };\n      /** Codecontext */\n      codeContextId?: string;\n    }>;\n  };\n}\n\nexport interface JSONRPCSuccesfulResponse extends JSONRPCBaseResponse {\n  error: null;\n  result: unknown;\n}\n\nexport interface JSONRPCErrorResponse extends JSONRPCBaseResponse {\n  result: null;\n  error: {\n    code: number;\n    message: string;\n    data?: {\n      trace?: StackTrace;\n    };\n  };\n}\n\nexport type JSONRPCResponse = JSONRPCSuccesfulResponse | JSONRPCErrorResponse;\n\nfunction getDebugAppend() {\n  if (typeof window !== \"undefined\" && typeof window.location !== \"undefined\") {\n    const urldebugvar = new URL(window.location.href).searchParams.get(\"wh-debug\");\n    if (urldebugvar)\n      return '?wh-debug=' + encodeURIComponent(urldebugvar);\n  }\n  return '';\n}\n\nclass ControlledCall {\n  client: RPCClient;\n  options: RPCCallOptions;\n  abortcontroller: AbortController;\n  timeout?: number;\n  _callurl: string;\n  _fetchoptions: RequestInit;\n  promise: Promise<unknown>;\n  timedout?: boolean;\n  aborted?: boolean;\n\n  constructor(client: RPCClient, method: string, stack: StackTrace | null, id: number, options: RPCCallOptions, callurl: string, fetchoptions: RequestInit) {\n    this.client = client;\n    this.options = options;\n\n    // if(options.timeout || options.signal) //as long as rpcResolve exists, we'll ALWAYS need to setup a controller\n    {\n      this.abortcontroller = new AbortController;\n      fetchoptions.signal = this.abortcontroller.signal;\n\n      if (options.timeout && options.timeout > 0) {\n        this.timeout = options.timeout;\n        setTimeout(() => this._handleTimeout(), options.timeout);\n      }\n      if (options.signal) {\n        options.signal.addEventListener(\"abort\", () => this._abort());\n      }\n    }\n\n    this._callurl = callurl;\n    this._fetchoptions = fetchoptions;\n\n    const fetchpromise = fetch(this._callurl, this._fetchoptions);\n    this.promise = this._completeCall(method, stack, id, fetchpromise) as Promise<unknown>;\n  }\n  _handleTimeout() {\n    this.timedout = true;\n    this.abortcontroller.abort();\n  }\n  _abort() {\n    this.aborted = true;\n    this.abortcontroller.abort();\n  }\n\n  async _completeCall(method: string, requestStack: StackTrace | null, id: number, fetchpromise: Promise<Response>) {\n    let response;\n    try {\n      for (; ;) { //loop to handle \"429 Conflict\"s\n        response = await fetchpromise;\n        if (response.status === 429 && !(\"retry429\" in this.options && !this.options.retry429) && response.headers.get(\"Retry-After\")) {\n          const retryafter = parseInt(response.headers.get(\"Retry-After\") || \"\");\n          if (this.client.debug)\n            console.warn(`[rpc] We are being throttled (429 Too Many Requests) - retrying after ${retryafter} seconds`);\n\n          await new Promise(resolve => setTimeout(resolve, retryafter * 1000));\n          fetchpromise = fetch(this._callurl, this._fetchoptions);\n          continue;\n        }\n        break;\n      }\n    } catch (exception) {\n      if (this.client.debug)\n        console.log(`[rpc] #${id} Exception invoking '${method}'`, exception);\n\n      if (this.aborted)\n        throw new Error(`RPC Aborted`);\n      else if (this.timedout)\n        throw new Error(`RPC Timeout: timeout was set to ${this.timeout} milliseconds`);\n      else\n        throw new Error(`RPC Failed: exception: ` + exception);\n    }\n\n    let jsonresponse;\n    try {\n      jsonresponse = await response.json() as JSONRPCResponse;\n      if (jsonresponse?.debug) {\n        if (jsonresponse.debug.consoleLog) {\n          for (const logitem of jsonresponse.debug.consoleLog) {\n            //should we log 'when'? it's getting more and more noisy then though....\n            //TODO should we match the remote's method (after validating) or just keep eveyrthing at 'log' ?\n            console.log(`[remote:${logitem.method}] ${logitem.location ? `${logitem.location.filename.split(\"/\").at(-1)}:${logitem.location.line}: ` : ''}${logitem.data}`);\n          }\n        }\n\n        delete jsonresponse.debug; //reduce noise in later logs\n      }\n\n      if (this.client.debug)\n        console.log(`[rpc] #${id} Received response to '${method}'`, jsonresponse);\n    } catch (exception) {\n      if (this.client.debug)\n        console.warn(`[rpc] #${id} Response was not valid JSON`, exception);\n    }\n\n    if (!jsonresponse)\n      throw new Error(\"RPC Failed: Invalid JSON/RPC response received\");\n\n    if (jsonresponse?.error) {\n      const err = new Error(\"RPC Error: \" + (jsonresponse.error.message || \"Unknown error\"));\n      if (jsonresponse.error.data?.trace) {\n        try {\n          prependStackTrace(err, jsonresponse.error.data.trace);\n        } catch (err2) {\n          //ignore stacktrace manipulation error\n        }\n      }\n\n      if (!this.options.silent)\n        this.client._tryLogError(requestStack, err);\n      throw err;\n    }\n\n    if (response.status === 200 && jsonresponse && jsonresponse.id !== id)\n      throw new Error(\"RPC Failed: Invalid JSON/RPC response received\");\n\n    if (this.options.wrapresult) {\n      return {\n        status: response.status,\n        result: jsonresponse.result || null,\n        error: jsonresponse.error || null,\n        retryafter: response.headers.get(\"Retry-After\") ? parseInt(response.headers.get(\"Retry-After\") || \"\") : null\n      };\n    }\n\n    return jsonresponse.result;\n  }\n}\n\n/** Invokes (WebHare) JSON/RPC\n    @param url - URL to invoke (leave empty or pass no parameters at all to callback to the current page)\n    */\nclass RPCClient {\n  url: string;\n  addfunctionname: boolean;\n  urlappend: string;\n  options: RPCCallOptions;\n  whservicematch: RegExpMatchArray | null;\n\n  constructor(url: string, options?: RPCCallOptions) {\n    this.options = {\n      timeout: 0,\n      debug: false,\n      ...options\n    };\n\n    if (!url)\n      throw new Error(`You must specify either a WebHare service name or a full URL`);\n\n    this.url = url;\n    this.whservicematch = this.url.match(/^([a-z0-9_]+):([a-z0-9_]+)$/);\n    //if shorthand syntax is used, we know we're talking to our local webhare. add function names and the profiling flag if needed\n    this.addfunctionname = this.options.addfunctionname !== undefined ? this.options.addfunctionname : Boolean(this.whservicematch);\n    this.urlappend = this.options.urlappend !== undefined ? this.options.urlappend : this.whservicematch ? getDebugAppend() : \"\";\n  }\n\n  get debug() {\n    return this.options.debug || debugFlags.rpc;\n  }\n\n  _tryLogError(requestStack: StackTrace | null, error: Error) {\n    console.group();\n    console.warn(\"RPC failed:\", error);\n    if (requestStack) {\n      console.warn(\"Stack at calling point\");\n      console.log(requestStack);\n    }\n    console.groupEnd();\n  }\n\n  //calculate the final URL. delayed here so services can be created on import (getDefaultRPCBase may require waiting for service.ready)\n  private getURL() {\n    //wh_services are always in the root (and soon inside /.wh)\n    const url = this.whservicematch ? `/wh_services/${this.whservicematch[1]}/${this.whservicematch[2]}` : this.url;\n    if (url.match(/^https?:\\/\\//))\n      return url;\n\n    if (this.options.baseUrl)\n      return new URL(url, this.options.baseUrl).toString();\n    else if (backendBase)\n      return backendBase + (backendBase.endsWith('/') && url.startsWith('/') ? url.substring(1) : url);\n    else\n      throw new Error(`You must set the baseUrl option when using the JSONRPC Client without a full URL outside Webhare`);\n  }\n\n  invoke(method: string, params: unknown[]) {\n    //build the URL, add profiling and function parameters where needed\n    let callurl = this.getURL();\n    if (this.addfunctionname) //simplifies log analysis, ignored by the server\n      callurl += `/${method}`;\n    callurl += this.urlappend;\n\n    const id = ++globalseqnr;\n    let requestStack: StackTrace | null = null;\n\n    if (this.debug) {\n      requestStack = parseTrace(new Error);\n      console.log(`[rpc] #${id} Invoking '${method}'`, params, callurl);\n    }\n\n    const fetchoptions: RequestInit = {\n      method: \"POST\",\n      credentials: this.options.credentials || 'same-origin', //this is the default since 2017-08-25, but Edge pre-18 is still around and will fail here\n      headers: {\n        \"Accept\": \"application/json\",\n        \"Content-Type\": \"application/json; charset=utf-8\",\n        ...this.options.headers\n      },\n      body: JSON.stringify(\n        {\n          id: id,\n          method: method,\n          params: params || []\n        }),\n      keepalive: Boolean(this.options.keepalive)\n    };\n\n    return new ControlledCall(this, method, requestStack, id, this.options, callurl, fetchoptions).promise;\n  }\n}\n\ntype ServiceBase<T> = {\n  withOptions(options: RPCCallOptions): T & ServiceBase<T>;\n};\n\nclass ServiceProxy<T> {\n  client: RPCClient;\n\n  constructor(client: RPCClient) {\n    this.client = client;\n  }\n\n  get(target: object, prop: string, receiver: unknown) {\n    if (prop === 'withOptions') { //create a withOptions function\n      return (options: RPCCallOptions) => {\n        const newoptions = {\n          ...this.client.options,\n          ...options,\n          headers: { ...this.client.options.headers, ...options.headers }\n        };\n        return createClient<T>(this.client.url, newoptions);\n      };\n    }\n\n    return (...args: unknown[]) => this.client.invoke(prop, args);\n  }\n}\n\n\n/** Creates an async version of the functions in a class\n * @typeParam ServiceType - Type definition of the service class that implements this service.\n*/\ntype ConvertToRPCInterface<ServiceType> = {\n  // eslint-disable-next-line @typescript-eslint/no-explicit-any -- using any is needed for this type definition\n  [K in Exclude<keyof ServiceType, `_${string}` | \"close\" | \"emit\"> as ServiceType[K] extends (...a: any) => any ? K : never]: ServiceType[K] extends (...a: any[]) => void ? PromisifyFunctionReturnType<ServiceType[K]> : never;\n};\n\n/** Get the client interface type as would be returned by createClient */\nexport type GetClientInterface<Client> = ConvertToRPCInterface<Client> & ServiceBase<ConvertToRPCInterface<Client>>;\n\n// eslint-disable-next-line @typescript-eslint/no-explicit-any -- If you don't care to type the service, just assume it can do anything\nexport function createClient<Client = any>(servicename: string, options?: RPCCallOptions): GetClientInterface<Client> {\n  const rpcclient = new RPCClient(servicename, options);\n  return new Proxy({}, new ServiceProxy<Client>(rpcclient)) as GetClientInterface<Client>;\n}\n","import { createClient, type GetClientInterface } from \"@webhare/jsonrpc-client\";\nimport type { AddressValidationOptions, AddressValidationStatus } from \"./address\";\nimport type { FormSubmitResult } from \"@mod-publisher/js/forms/formbase\";\nimport type { EmailValidationResult, RPCFormTarget, RPCFormInvokeRPC, RPCFormSubmission, RPCFormMessage } from \"./types\";\nimport type { FormService } from \"@mod-publisher/lib/internal/forms/service\";\nimport type { AddressValue } from \"@webhare/address\";\nimport type { CaptchaProvider } from \"@mod-publisher/js/captcha/api\";\n\n\n/// HareScript uses 'nr_detail' instead of 'houseNumber'\nexport type HareScriptAddressValue = Omit<AddressValue, \"houseNumber\"> & { nr_detail?: string };\n\nexport interface HareScriptAddressValidationResult {\n  status: AddressValidationStatus;\n  errors: Array<{\n    ///Fields affected by the error\n    fields: string[];\n    ///Error message in the requested language\n    message: string;\n  }>;\n  corrections: Partial<Record<keyof HareScriptAddressValue, string>> | null;\n}\n\nexport interface PublisherFormService {\n  /** Validate an email address for validity (including against server configured correction/blacklists)\n   *\n   * @param langcode - Language code for messages\n   * @param emailaddress - Address to validate\n   * @returns Validation result\n   */\n  validateEmail(langcode: string, emailaddress: string): Promise<EmailValidationResult>;\n\n  formValidateEmail(target: RPCFormTarget & { field: string }, emailaddress: string): Promise<EmailValidationResult>;\n\n  /** Get an image from a remote URL */\n  getImgFromRemoteURL(imageurl: string): Promise<string>;\n\n  /** Get the final image URL to use */\n  getUploadedFileFinalURL(uploadurl: string): Promise<string>;\n\n  validateEmbeddedObjects(objrefs: string[]): Promise<{ tokill: string[] }>;\n\n  /** Verify address */\n  verifyAddress(url: string, address: HareScriptAddressValue, options: AddressValidationOptions): Promise<HareScriptAddressValidationResult>;\n\n  formSubmit(submitinfo: RPCFormSubmission): Promise<FormSubmitResult>;\n\n  formInvoke(submitinfo: RPCFormInvokeRPC): Promise<{\n    messages: RPCFormMessage[];\n    result: unknown;\n  }>;\n\n  requestBuiltinForm(submitinfo: RPCFormTarget, filename: string, formname: string): Promise<{ html: string }>;\n\n  /** Get captcha configuration. This is needed when the form will preactivate captchas (if form submission requests a captcha it will transmit provider setting in the error metadata) */\n  getCaptchaConfiguration(submitinfo: RPCFormTarget): Promise<CaptchaProvider | null>;\n}\n\nlet hsformservice: GetClientInterface<PublisherFormService> | undefined;\nlet tsformservice: GetClientInterface<FormService> | undefined;\n\nexport function getFormService(): GetClientInterface<PublisherFormService> {\n  hsformservice ||= createClient<PublisherFormService>(\"publisher:forms\");\n  return hsformservice;\n}\n\nexport function getTSFormService(): GetClientInterface<FormService> {\n  tsformservice ||= createClient<FormService>(\"publisher:formsts\");\n  return tsformservice;\n}\n","import { debugFlags } from \"@webhare/env\";\n\n/** @deprecated  Use debugFlags from \\@webhare/env */\nconst debugflags = debugFlags;\nexport { debugflags };\n\n//Add all flags as a dompack-debug-- class\nif (typeof document !== 'undefined')\n  document.documentElement.classList.add(...Object.keys(debugFlags).map(flagname => \"dompack--debug-\" + flagname));\nif (debugFlags.dompack)\n  console.log('[dompack] debugging flags: ' + Object.keys(debugFlags).join(', '));\n","/* All keynames should be mixed-cased, as done here: http://www.w3.org/TR/DOM-Level-3-Events-key/\n\n   Modifiers should be in the ordering Alt+Control+Meta+Shift+key (ie alphabetical)\n\n   To simply prevent keys from propagating up (ie to keep Enter inside its textarea)\n   new KeyboardHandler(this.textarea, {}, { dontpropagate: ['Enter']});\n*/\n\nimport { type NormalizedKeyboardEvent, normalizeKeyboardEventData } from '../src/events';\nimport { debugflags } from '../src/debug';\nimport { getPlatform } from './browser';\nconst IS_MAC_PLATFORM = getPlatform() === \"mac\";\n\nconst propnames = {\n  shiftKey: \"Shift\",\n  ctrlKey: IS_MAC_PLATFORM ? \"Control\" : [\"Accel\", \"Control\"],\n  metaKey: IS_MAC_PLATFORM ? [\"Accel\", \"Meta\"] : \"Meta\",\n  altKey: \"Alt\"\n};\n\nfunction getFinalKey(event: NormalizedKeyboardEvent) { //get the name for the 'final' key, eg the 'D' in 'alt+control+d'\n  if (event.code?.startsWith('Key') && event.code.length === 4)\n    return event.code.substring(3, 4).toUpperCase();\n  if (event.code?.startsWith('Digit') && event.code.length === 6)\n    return event.code.substring(5, 6).toUpperCase();\n  return event.key.length === 1 ? event.key.toUpperCase() : event.key;\n}\n\nfunction getKeyNames(event: NormalizedKeyboardEvent) {\n  let names: string[][] = [[]];\n\n  /*\n    // Firefix under selenium on linux always says 'Unidentified' as key. Backup for some keys.\n    if (basekey === \"Unidentified\")\n      basekey = selenium_backup[event.keyCode];\n  */\n  // Create the modifiers in the names array (omit the basekey, so we can sort on modifier first)\n  (Object.keys(propnames) as Array<keyof typeof propnames>).forEach(propname => {\n    if (event[propname]) {\n      // The key is pressed. Add the modifier name to all current names.\n      const modifier: string | string[] = propnames[propname];\n      if (!Array.isArray(modifier))\n        names.forEach(function (arr) { arr.push(modifier); });\n      else {\n        // Multiple modifiers map to this key, duplicate all result sequences for every modifier\n        const newkeys: string[][] = [];\n        modifier.forEach(function (singlemodifier) {\n          names.forEach(function (arr) {\n            newkeys.push(arr.concat([singlemodifier]));\n          });\n        });\n        names = newkeys;\n      }\n    }\n  });\n\n  return names.map(function (arr) {\n    // Sort the modifier names\n    arr = arr.sort();\n    arr.push(getFinalKey(event));\n    return arr.join(\"+\");\n  });\n}\n\nfunction validateKeyName(key: string) {\n  const modifiers = key.split(\"+\");\n  modifiers.pop();\n\n  // Check for allowed modifiers\n  modifiers.forEach(function (mod) {\n    if (![\"Accel\", \"Alt\", \"Control\", \"Meta\", \"Shift\"].includes(mod))\n      throw new Error(\"Illegal modifier name '\" + mod + \"' in key '\" + key + \"'\");\n  });\n\n  const original_order = modifiers.join('+');\n  modifiers.sort();\n  if (modifiers.join('+') !== original_order)\n    throw new Error(\"Illegal key name \" + key + \", modifiers must be sorted alphabetically\");\n}\n\ntype KeyboardMappingHandler = (event: KeyboardEvent) => boolean | void;\ntype KeyboardMapping = { [key: string]: KeyboardMappingHandler };\ntype KeyPressHandler = (event: KeyboardEvent, key: string) => boolean | void;\ntype KeyboardEventHandler = (event: Event) => void;\ntype KeyboardHandlerOptions =\n  {\n    stopmapped?: boolean;\n    dontpropagate?: string[];\n    onkeypress?: KeyPressHandler;\n    captureunsafekeys?: boolean;\n    listenoptions?: AddEventListenerOptions;\n  };\n\n/**\n     node: The node to attach to\n    keymap: Keymap\n    options.stopmapped - preventDefault and stopPropagation on any key we have in our map\n    options.dontpropagate - string array of keys not to propagate out of this object\n    options.onkeypress - when set, call for all keypresses. signature: function (event, key). Should always return true and preventDefault (and/or stop) the event to cancel its handling\n    options.listenoptions - addEventListener options (eg \\{capture:true\\})\n */\nexport default class KeyboardHandler {\n  node: EventTarget;\n  keymap: KeyboardMapping;\n  stopmapped: boolean;\n  dontpropagate: string[];\n  onkeypress?: KeyPressHandler;\n  captureunsafekeys: boolean;\n  private _listenoptions?: AddEventListenerOptions;\n  private _onkeydown: KeyboardEventHandler;\n  private _onkeypress: KeyboardEventHandler;\n\n  constructor(node: EventTarget, keymap: KeyboardMapping, options?: KeyboardHandlerOptions) {\n    this.node = node;\n    this.keymap = {};\n    this.stopmapped = options?.stopmapped ?? false;\n    this.dontpropagate = options?.dontpropagate ? [...options.dontpropagate].map(name => name.toUpperCase()) : [];\n    this.onkeypress = options?.onkeypress;\n    this.captureunsafekeys = options?.captureunsafekeys ?? false;\n    this._listenoptions = options?.listenoptions;\n\n    Object.keys(keymap).forEach(keyname => {\n      if (debugflags.key)\n        validateKeyName(keyname);\n      this.keymap[keyname.toUpperCase()] = keymap[keyname];\n    });\n\n    this._onkeydown = (event) => this._onKeyDown(event);\n    this._onkeypress = (event) => this._onKeyPress(event);\n    node.addEventListener('keydown', this._onkeydown, this._listenoptions);\n    node.addEventListener('keypress', this._onkeypress, this._listenoptions);\n  }\n\n  destroy() {\n    this.node.removeEventListener('keydown', this._onkeydown, this._listenoptions);\n    this.node.removeEventListener('keypress', this._onkeypress, this._listenoptions);\n  }\n\n  /**\n       Returns thether the current pressed special key should be ignored for the current target node\n      Used to detect input/textarea/rte's\n   *\n      @param target - Current target node for keyboard event\n      @param key - Parsed key\n      @param keynames - Potential names for the keys (as returned by GetKeyNames)\n      @returns Whether the key must be ignored by KeyboardHandler, default browser behaviour should be triggered.\n   */\n  private _mustIgnoreKey(target: EventTarget | null, key: string, keynames: string[]) {\n    if (!(target instanceof Node))\n      return false;\n    const tag = target.nodeName.toLowerCase();\n    if (tag === \"select\") {\n      if ([\"ArrowUp\", \"ArrowDown\", \"Home\", \"End\", \"PageUp\", \"PageDown\"].indexOf(key) !== -1)\n        return true;\n    } else if (tag === \"input\" || tag === \"textarea\" || (target instanceof HTMLElement && target.isContentEditable)) {\n      // These keys we ignore, regardless of the modifier\n      if ([\n        \"ArrowLeft\", \"ArrowRight\", \"ArrowUp\", \"ArrowDown\",\n        \"PageUp\", \"PageDown\",\n        \"Home\", \"End\",\n        \"Insert\", \"Delete\", \"Backspace\"\n      ].indexOf(key) !== -1)\n        return true;\n\n      let is_special_combo = false;\n\n      // only input doesn't want exact combo 'Enter', the rest does\n      if (tag !== \"input\" && keynames.indexOf(\"Enter\") !== -1)\n        is_special_combo = true;\n\n      // Only contenteditable wants \"Shift+Enter\"\n      if (target instanceof HTMLElement && target.isContentEditable && keynames.indexOf(\"Shift+Enter\") !== -1)\n        is_special_combo = true;\n\n      // These exact combo's are wanted by all inputs\n      [\"Accel+A\", \"Accel+V\", \"Accel+C\", \"Accel+X\"].forEach(function (name) {\n        is_special_combo = is_special_combo || keynames.indexOf(name) !== -1;\n      });\n      return is_special_combo;\n    }\n    return false;\n  }\n\n  addKey(keybinding: string, handler: KeyboardMappingHandler) {\n    if (debugflags.key) {\n      validateKeyName(keybinding);\n      console.log(\"[key] KeyDown handler registered for \" + keybinding);\n    }\n    this.keymap[keybinding.toUpperCase()] = handler;\n  }\n  removeKey(keybinding: string) {\n    delete this.keymap[keybinding.toUpperCase()];\n  }\n  _onKeyDown(event: Event) { // We're a key event handler, so we know the event is a KeyboardEvent\n    const keydata = normalizeKeyboardEventData(event as KeyboardEvent);\n\n    // Get all possible names for this key\n    const keynames = getKeyNames(keydata);\n    if (!keydata.key || !keynames.length) {\n      if (debugflags.key)\n        console.log(\"[key] KeyDown handler for \", this.node, \" did not recognize key from event\", event);\n      return true;\n    }\n\n    if (debugflags.key)\n      console.log(\"[key] KeyDown handler for \", this.node, \" got key \", keydata.key, \" with target \", event.target, \" keynames:\", keynames);\n\n    /* Some keys we ignore, unless we're explicitly bound to a node, so we don't inadvertly break eg a <input> node inside\n       a listview we're handling or otherwise break a user's expectation. Set the option 'captureunsafekeys' if you explicitly\n       want to be able to capture any key */\n\n    if (!this.captureunsafekeys && this._mustIgnoreKey(event.target, keydata.key, keynames)) {\n      if (debugflags.key)\n        console.log(\"[key] KeyDown event will not be intercepted, it's an unsafe key to intercept\");\n      return true;\n    }\n\n    if (this.dontpropagate) {\n      keynames.forEach(keyname => {\n        if (this.dontpropagate.includes(keyname)) {\n          if (debugflags.key)\n            console.log(\"[key] KeyDown event will not bubbleup because of our dontpropagate option (but may still trigger a default action)\");\n          event.stopPropagation();\n        }\n      });\n    }\n\n    for (let i = 0; i < keynames.length; ++i) {\n      const mapping = this.keymap[keynames[i].toUpperCase()];\n      if (!mapping)\n        continue;\n\n      if (this.stopmapped) {\n        if (debugflags.key)\n          console.log(\"[key] KeyDown event will not bubbleup or trigger default, because we're configured to block any mapped key\");\n        event.stopPropagation();\n        event.preventDefault();\n      }\n\n      const ishandled = mapping.apply(this.node, [event as KeyboardEvent]);\n      if (ishandled && !event.defaultPrevented) {\n        console.warn(`The key handler for '${keynames[i]}' should preventDefault (or dompack.stop) the event to block fruther propagation`);\n        event.stopPropagation();\n        event.preventDefault();\n        if (debugflags.key)\n          console.log(\"[key] KeyDown event will not bubbleup or trigger default, because the keyhandler indicated the key was handled\");\n      }\n\n      if (!event.defaultPrevented && debugflags.key)\n        console.log(\"[key] KeyDown event was not blocked by its explicitly configured handler\");\n    }\n    return true;\n  }\n  _onKeyPress(event: Event) { // We're a key event handler, so we know the event is a KeyboardEvent\n    const keydata = normalizeKeyboardEventData(event as KeyboardEvent);\n\n    if (this.onkeypress) {\n      if (!this.onkeypress.apply(this.node, [event as KeyboardEvent, keydata.key])) {\n        if (!event.defaultPrevented)\n          console.warn(\"The onkeypress handler should preventDefault (or dompack.stop) the event to block fruther propagation\");\n        event.stopPropagation();\n        event.preventDefault();\n      }\n    }\n  }\n}\n\nexport function getEventKeyNames(event: KeyboardEvent) {\n  const keydata = normalizeKeyboardEventData(event);\n  return getKeyNames(keydata);\n}\n","import type { GetTidHooks } from \"./types.js\";\n\nlet hookFactory: (() => GetTidHooks) | undefined;\nlet gotHooks = false;\n\nexport function setGetTidHooksFactory(newHookFactory: () => GetTidHooks) {\n  if (gotHooks)\n    throw new Error(`The call to setGetTidHooksFactory was too late - hooks have already been retrieved once`);\n  hookFactory = newHookFactory;\n}\n\nexport function getGetTidHooks() {\n  gotHooks = true;\n  return hookFactory?.();\n}\n","import { decodeString, encodeString } from '@webhare/std';\nimport type { CompiledLanguageFile, GetTidHooks, LanguagePart, LanguageText, RecursiveLanguageTexts, TidAllowedHTMLTags, TidParam } from './types';\nimport { getGetTidHooks } from './hooks';\nimport { debugFlags, dtapStage } from '@webhare/env';\n\n\n\nlet hooks: GetTidHooks | undefined;\nlet browserSetLanguage = (\"document\" in globalThis && document?.documentElement?.lang?.substring(0, 2).toLowerCase()) || \"\";\nconst defaultLanguage = \"en\";\n\nfunction ensureHooks(): GetTidHooks {\n  return hooks ??= getGetTidHooks?.() ?? {\n    loader: null,\n    currentLanguage: (lang) => {\n      if (lang)\n        browserSetLanguage = lang;\n      return browserSetLanguage;\n    }\n  };\n}\n\nexport function getTidLanguage(): string {\n  return ensureHooks().currentLanguage() || defaultLanguage;\n}\n\nexport function setTidLanguage(newLanguage: string): void {\n  ensureHooks().currentLanguage(newLanguage.toLowerCase());\n}\n\nconst langFileCache: Record<string, CompiledLanguageFile> = {};\n\nfunction getLanguageFile(module: string, langCode: string): CompiledLanguageFile | undefined {\n  const langkey = `${module}:${langCode}`;\n  let compiled = langFileCache[langkey];\n  if (!compiled || !compiled.resource) {\n    const loader = ensureHooks().loader;\n    if (loader) {\n      try {\n        compiled = loader(module, langCode, compiled?.registered ?? new Map);\n      } catch (e) {\n        // loading failed, erase the loaded texts (keep the registered texts, though)\n        if (dtapStage !== \"production\" || debugFlags.gtd) {\n          console.error(e);\n        }\n        const texts = compiled?.registered ?? new Map;\n        compiled = { resource: \"\", fallbackLanguage: \"\", langCode, texts, registered: texts };\n      }\n      langFileCache[langkey] = compiled;\n    }\n  }\n  if (!compiled && (dtapStage !== \"production\" || debugFlags.gtd)) {\n    console.warn(`No language texts found for module '${module}' and language '${langCode}'`);\n  }\n  return compiled;\n}\n\n\n// eslint-disable-next-line @typescript-eslint/no-explicit-any -- needed for generics\nexport type GetTidRenderFunc<Result extends object = object> = ((tag: TidAllowedHTMLTags, props: { children?: any[]; href?: string }, ...childNodes: any[]) => Result);\nexport type GetTidOptions = { langCode?: string; render?: GetTidRenderFunc | \"fragment\" };\n\ntype GetTidArgumentTypes = [GetTidOptions?] | [TidParam[], GetTidOptions?] | [TidParam?, TidParam?, TidParam?, TidParam?];\n\nfunction toHTMLNode(tag: string, props: { href?: string; children?: Array<string | Node> }): Node {\n  const node = document.createElement(tag);\n  if (props.href)\n    node.setAttribute(\"href\", props.href);\n  if (props.children)\n    node.append(...props.children);\n  return node;\n}\n\nfunction wrapInFragment(children: Array<string | Node>): DocumentFragment {\n  const fragment = document.createDocumentFragment();\n  fragment.append(...children);\n  return fragment;\n}\n\n\nfunction getTidInternal(tid: string, params: TidParam[], options?: GetTidOptions): string | Array<string | object> | DocumentFragment {\n  if (tid === \"\") {\n    return \"\";\n  }\n\n  if (tid.startsWith(\":\")) {\n    return tid.substring(1);\n  } else if (tid.startsWith(\"<>\")) {\n    return decodeString(tid.substring(2), \"html\");\n  }\n\n  const langCode = options?.langCode?.toLowerCase() || getTidLanguage();\n  if (options?.render) {\n    if (options.render === \"fragment\") {\n      const nodes = calcTIDForLanguage(langCode, tid, params, true, toHTMLNode) as Array<string | Node>;\n      return wrapInFragment(nodes);\n    }\n    return calcTIDForLanguage(langCode, tid, params, true, options.render);\n  } else\n    return calcTIDForLanguage(langCode, tid, params, false, null).join(\"\");\n}\n\nexport function getTid(tid: string, options?: GetTidOptions & { render?: undefined }): string;\nexport function getTid<VNode extends object>(tid: string, options?: GetTidOptions & { render?: GetTidRenderFunc<VNode> }): string | Array<string | VNode>;\nexport function getTid(tid: string, options?: GetTidOptions & { render?: \"fragment\" }): DocumentFragment;\nexport function getTid(tid: string, param: TidParam[], options?: GetTidOptions & { render?: undefined }): string;\nexport function getTid<VNode extends object>(tid: string, param: TidParam[], options?: GetTidOptions & { render?: GetTidRenderFunc<VNode> }): string | Array<string | VNode>;\nexport function getTid(tid: string, param: TidParam[], options?: GetTidOptions & { render?: \"fragment\" }): DocumentFragment;\nexport function getTid(tid: string, p1?: TidParam, p2?: TidParam, p3?: TidParam, p4?: TidParam): string;\n\nexport function getTid(tid: string, ...p: GetTidArgumentTypes): string | Array<string | object> | DocumentFragment {\n  if (Array.isArray(p[0])) // case: param: TidParam[], options?: GetTidOptions)\n    return getTidInternal(tid, p[0], p[1] as GetTidOptions);\n  if (p[0] && typeof p[0] === \"object\") // case: options: GetTidOptions\n    return getTidInternal(tid, [], p[0]);\n  return getTidInternal(tid, p as Array<TidParam | undefined>, {});\n}\n\nexport function getTidForLanguage(langCode: string, tid: string, p1: TidParam = null, p2: TidParam = null, p3: TidParam = null, p4: TidParam = null): string {\n  return getTid(tid, [p1, p2, p3, p4], { langCode });\n}\n\nexport function getHTMLTid(tid: string, ...params: TidParam[]): string {\n  return getHTMLTidForLanguage(getTidLanguage(), tid, ...params);\n}\n\nexport function getHTMLTidForLanguage(langCode: string, tid: string, ...params: TidParam[]): string {\n  if (tid === \"\") {\n    return \"\";\n  }\n\n  if (tid.startsWith(\":\")) {\n    return encodeString(tid.substring(1), \"html\");\n  } else if (tid.startsWith(\"<>\")) {\n    return tid.substring(2);\n  }\n\n  return calcTIDForLanguage(langCode, tid, params, true, null).join(\"\");\n}\n\nexport function getTIDListForLanguage(langcode: string, gid: string) {\n  gid = getCanonicalTid(gid.toLowerCase());\n  langcode = langcode.toLowerCase();\n\n  const modsep = gid.indexOf(':');\n  if (modsep === -1) {\n    if (dtapStage !== \"production\" || debugFlags.gtd)\n      console.warn(`Missing module name in call for gid '${gid}'`);\n    return `(cannot find textnode: ${gid})`;\n  }\n\n  const module = gid.substring(0, modsep);\n  const lookup = gid.substring(modsep + 1).replace(/ /g, \"_\") + \".\";\n\n  const compiled = getLanguageFile(module, langcode);\n  const result: string[] = [];\n  if (compiled) {\n    for (const key of compiled.texts.keys()) {\n      if (key.startsWith(lookup) && key.indexOf(\".\", lookup.length) === -1)\n        result.push(`${module}:${key}`);\n    }\n    if (compiled.fallbackLanguage) {\n      const fallback = getLanguageFile(module, compiled.fallbackLanguage);\n      if (fallback) {\n        for (const key of fallback.texts.keys()) {\n          if (key.startsWith(lookup) && key.indexOf(\".\", lookup.length) === -1 && !result.includes(`${module}:${key}`))\n            result.push(`${module}:${key}`);\n        }\n      }\n    }\n  }\n\n  return result;\n}\n\nfunction cannotFind(tid: string) {\n  if (debugFlags.sut)\n    return [tid.substring(tid.lastIndexOf(\".\"))];\n\n  return [`(cannot find text: ${tid})`];\n}\nfunction calcTIDForLanguage(langcode: string, tid: string, rawParams: TidParam[], rich: boolean, render: null): string[];\nfunction calcTIDForLanguage(langcode: string, tid: string, rawParams: TidParam[], rich: boolean, render: GetTidRenderFunc): Array<string | object>;\n\n\nfunction calcTIDForLanguage(langcode: string, tid: string, rawParams: TidParam[], rich: boolean, render: GetTidRenderFunc | null): Array<string | object> {\n  tid = getCanonicalTid(tid.toLowerCase());\n  if (langcode.match(/^[a-z]{2}-/)) //truncate country codes from language identifiers (eg en-GB) until we have actual support for those, so we don't break when en-GB is passed straight from eg. document.documentElement.lang\n    langcode = langcode.substring(0, 2);\n\n  if (tid === \"tollium:tilde.locale.datetimestrings\") {\n    return [getLanguageDatetimeStrings(langcode)];\n  }\n\n  // convert null to \"\", numbers to strings\n  const params = rawParams.map(p => (typeof p === \"number\" ? p.toString() : (p ?? \"\")));\n\n  const modsep = tid.indexOf(':');\n  if (modsep === -1) {\n    if (dtapStage !== \"production\" || debugFlags.gtd)\n      console.warn(`Missing module name in call for tid '${tid}'`);\n    return [`(missing module name in tid: ${tid})`];\n  }\n\n  const module = tid.substring(0, modsep);\n  const lookup = tid.substring(modsep + 1).replace(/ /g, \"_\");\n  tid = `${module}:${lookup}`;\n\n  if (langcode === \"debug\") {\n    const debugtid = `{${[tid, ...params].join(\"|\")}}`;\n    return [rich ? encodeString(debugtid, \"html\") : debugtid];\n  }\n\n  const compiled = getLanguageFile(module, langcode);\n  const match = compiled?.texts.get(lookup);\n  if (!match && compiled?.fallbackLanguage) {\n    const fallback = getLanguageFile(module, compiled.fallbackLanguage);\n    const fallbackMatch = fallback?.texts.get(lookup);\n    if (!fallbackMatch && (dtapStage !== \"production\" || debugFlags.gtd)) {\n      console.warn(`Cannot find text ${tid} for language ${langcode}, also tried fallback language '${compiled.fallbackLanguage}'`);\n    }\n    return fallbackMatch ? executeCompiledTidText(fallbackMatch, params, rich, render) : cannotFind(tid);\n  }\n\n  if (!match && (dtapStage !== \"production\" || debugFlags.gtd)) {\n    console.warn(`Cannot find text ${tid} for language ${langcode}, no fallback language`);\n  }\n  return match ? executeCompiledTidText(match, params, rich, render) : cannotFind(tid);\n}\n\nfunction getCanonicalTid(tid: string) {\n  if (tid.startsWith(\"tollium:common.buttons.\") || tid.startsWith(\"tollium:common.actions.\") || tid.startsWith(\"tollium:common.labels.\")) {\n    return `tollium:tilde.${tid.substring(tid.indexOf(\".\") + 1)}`;\n  } else if (tid.startsWith(\"~\")) {\n    return `tollium:tilde.${tid.substring(1)}`;\n  }\n\n  return tid;\n}\n\nfunction renderString(tok: string, rich: boolean, render: GetTidRenderFunc | null) {\n  return rich ?\n    render ?\n      tok.split(/(\\n)/).map(part => part !== \"\\n\" ? part : render(\"br\", {})).filter(_ => _) :\n      [encodeString(tok, \"html\")] :\n    [tok];\n}\n\nfunction executeCompiledTidText(text: string | LanguagePart[], params: string[], rich: boolean, render: GetTidRenderFunc | null): Array<object | string> {\n  if (typeof text === \"string\")\n    return renderString(text, rich, render);\n\n  const parts = new Array<object | string>;\n  for (const tok of text) {\n    if (typeof tok === \"string\") {\n      parts.push(...renderString(tok, rich, render));\n    } else if (typeof tok === \"number\") {\n      if (tok >= 1) {\n        const get_param = params[tok - 1];\n        if (get_param) {\n          parts.push(...renderString(get_param, rich, render));\n        }\n      }\n    } else if (tok.t === \"tag\") {\n      const sub = executeCompiledTidText(tok.subs, params, rich, render);\n      if (rich && render)\n        parts.push(render(tok.tag, { children: sub }));\n      else\n        parts.push(...(rich ? [`<${tok.tag}>`, ...sub, `</${tok.tag}>`] : sub));\n    } else if (tok.t === \"ifparam\") {\n      const get_param = params[tok.p - 1] || \"\";\n      parts.push(...executeCompiledTidText(get_param.toUpperCase() === tok.value.toUpperCase() ? tok.subs : tok.subselse, params, rich, render));\n    } else if (tok.t === \"a\") {\n      const sub = executeCompiledTidText(tok.subs, params, rich, render);\n      if (rich) {\n        let link = tok.link;\n        if (tok.linkparam > 0 && tok.linkparam <= params.length) {\n          link = params[tok.linkparam - 1] || link;\n        }\n        if (link) {\n          if (render)\n            parts.push(render(\"a\", { href: link, children: sub }));\n          else\n            parts.push(`<a href=\"${encodeString(link, \"attribute\")}\">`, ...sub, `</a>`);\n        } else {\n          parts.push(...sub);\n        }\n      } else {\n        parts.push(...sub);\n      }\n    }\n  }\n  return parts;\n}\n\nfunction flattenRecursiveLanguageTexts(text: RecursiveLanguageTexts, pathsofar: string, result: Map<string, LanguageText>): void {\n  if (typeof text === \"string\" || Array.isArray(text))\n    result.set(pathsofar, text);\n  else\n    for (const [name, value] of Object.entries(text))\n      flattenRecursiveLanguageTexts(value, `${pathsofar}${pathsofar && name ? \".\" : \"\"}${name}`, result);\n}\n\nexport function registerTexts(module: string, langCode: string, tids: RecursiveLanguageTexts) {\n  const langkey = `${module}:${langCode}`;\n  let compiledFile = langFileCache[langkey];\n  if (!compiledFile) {\n    const texts = new Map;\n    langFileCache[langkey] = compiledFile = { resource: \"\", fallbackLanguage: \"\", langCode, texts, registered: texts };\n  }\n  flattenRecursiveLanguageTexts(tids, \"\", compiledFile.registered);\n  if (compiledFile.texts !== compiledFile.registered) {\n    for (const [key, value] of compiledFile.registered)\n      compiledFile.texts.set(key, value);\n  }\n}\n\nfunction getLanguageDatetimeStrings(langCode: string) {\n  switch (langCode.substring(0, 2)) {\n    case \"nl\": return \"am;pm;januari;februari;maart;april;mei;juni;juli;augustus;september;oktober;november;december;maandag;dinsdag;woensdag;donderdag;vrijdag;zaterdag;zondag;jan;feb;mrt;apr;mei;jun;jul;aug;sep;okt;nov;dec;ma;di;wo;do;vr;za;zo\";\n    case \"de\": return \"am;pm;Januar;Februar;März;April;Mai;Juni;Juli;August;September;Oktober;November;Dezember;Montag;Dienstag;Mittwoch;Donnerstag;Freitag;Samstag;Sonntag;Jan.;Febr.;März;Apr.;Mai;Juni;Juli;Aug.;Sept.;Okt.;Nov.;Dez.;Mo;Di;Mi;Do;Fr;Sa;So\";\n    case \"fr\": return \"am;pm;Janvier;Février;Mars;Avril;Mai;Juin;Juillet;Août;Septembre;Octobre;Novembre;Décembre;Lundi;Mardi;Mercredi;Jeudi;Vendredi;Samedi;Dimanche;Janv;Févr;Mars;Avril;Mai;Juin;Juil;Août;Sept;Oct;Nov;Déc;Lun;Mar;Mer;Jeu;Ven;Sam;Dim\";\n    case \"jp\": {\n      return Buffer.from(\"5Y2I5YmNO+WNiOW+jDvvvJHmnIg777yS5pyIO++8k+aciDvvvJTmnIg777yV5pyIO++8luaciDvvvJfmnIg777yY5pyIO++8meaciDvvvJHvvJDmnIg777yR77yR5pyIO++8ke+8kuaciDvmnIjmm5zml6U754Gr5puc5pelO+awtOabnOaXpTvmnKjmm5zml6U76YeR5puc5pelO+Wcn+abnOaXpTvml6Xmm5zml6U777yR5pyIO++8kuaciDvvvJPmnIg777yU5pyIO++8leaciDvvvJbmnIg777yX5pyIO++8mOaciDvvvJnmnIg777yR77yQ5pyIO++8ke+8keaciDvvvJHvvJLmnIg75pyIO+eBqzvmsLQ75pyoO+mHkTvlnJ875pel\", \"base64\").toString();\n    }\n  }\n\n  // \"en\" is the default\n  return \"am;pm;January;February;March;April;May;June;July;August;September;October;November;December;Monday;Tuesday;Wednesday;Thursday;Friday;Saturday;Sunday;Jan;Feb;Mar;Apr;May;Jun;Jul;Aug;Sep;Oct;Nov;Dec;Mon;Tue;Wed;Thu;Fri;Sat;Sun\";\n}\n","// This gets TypeScript to refer to us by our @webhare/... name in auto imports:\ndeclare module \"@webhare/gettid\" {\n}\n\nexport {\n  getTidLanguage,\n  setTidLanguage,\n  getTid,\n  getTidForLanguage,\n  getHTMLTid,\n  getHTMLTidForLanguage,\n  getTIDListForLanguage\n} from \"./internal\";\n","// This gets TypeScript to refer to us by our @webhare/... name in auto imports:\ndeclare module \"@webhare/upload\" {\n}\n\nimport * as dompack from \"@webhare/dompack\";\n\n/* Note: you can't really build a FileList yourself, but FileList doesn't satisfy File[] and neither the reverse works. (noone is really happy with that though)\n   So we'll just accept both types */\n\ndeclare global {\n  interface GlobalEventHandlersEventMap {\n    \"wh:requestfiles\": CustomEvent<{\n      /** Callback to invoke with the list of files to upload */\n      resolve: (files: File[]) => void;\n    } & Required<UploadRequestOptions>>;\n  }\n}\n\nexport interface UploadRequestOptions {\n  ///List of mimetypes to accept, supports wildcards (eg `image/*`)\n  accept?: string[];\n  ///Allow multiple files to be selected. By default determined by whether you use requestFiles or requestFile, but can be set to false to force requestFiles to only accept one file\n  multiple?: boolean;\n}\n\nexport interface UploadProgressStatus {\n  uploadedBytes: number;\n  totalBytes: number;\n  uploadedFiles: number;\n  totalFiles: number;\n  //Upload speed in bytes/sec.\n  uploadSpeed: number;\n}\n\nexport interface UploadOptions {\n  onProgress?: (progress: UploadProgressStatus) => void;\n  signal?: AbortSignal;\n}\n\nexport interface UploadResult {\n  name: string;\n  size: number;\n  type: string;\n  token: string;\n}\n\nexport interface UploadManifest {\n  files: Array<{\n    name: string;\n    size: number;\n    type: string;\n  }>;\n}\n\nexport interface UploadInstructions {\n  baseUrl: string;\n  sessionId: string;\n  chunkSize: number;\n  signal?: AbortSignal;\n}\n\nexport type ResourceDescriptorCompatible = {\n  resource: Blob;\n  mediaType: string;\n  fileName: string | null;\n};\n\nexport interface UploaderBase {\n  readonly manifest: UploadManifest;\n  upload(instructions: UploadInstructions, options?: UploadOptions): Promise<UploadResult | UploadResult[]>;\n}\n\nfunction isRealBlob(data: Blob) { return data instanceof Blob; }\nasync function convertToRealBlob(data: Blob): Promise<Blob> { return new Blob([await data.arrayBuffer()]); }\n\nexport class MultiFileUploader implements UploaderBase {\n  readonly #files: Array<{ name: string; size: number; type: string; data: Blob }>;\n  readonly manifest: UploadManifest;\n\n  constructor(files: Array<File | Blob | ResourceDescriptorCompatible>, signal?: AbortSignal) {\n    if (!files.length)\n      throw new Error(\"No files to upload\");\n\n    this.#files = files.map(item => (\n      \"resource\" in item ? {\n        name: item.fileName ?? \"upload\",\n        size: item.resource.size,\n        type: item.mediaType || \"application/octet-stream\",\n        data: item.resource,\n      } : {\n        name: \"name\" in item ? item.name : \"upload\",\n        size: item.size,\n        type: item.type || \"application/octet-stream\",\n        data: item\n      }));\n    this.manifest = {\n      files: this.#files.map(file => ({ name: file.name, size: file.size, type: file.type }))\n    };\n  }\n\n  async upload(instructions: UploadInstructions, options?: UploadOptions): Promise<UploadResult[]> {\n    const outfiles: UploadResult[] = [];\n    let uploadedBytes = 0, uploadedFiles = 0;\n    const totalBytes = this.#files.reduce((acc, file) => acc + file.size, 0);\n    const totalFiles = this.#files.length;\n    const start = Date.now();\n\n    function fireProgressEvent(partialbytes: number) {\n      const curUploaded = uploadedBytes + partialbytes;\n      const timeElapsed = Date.now() - start;\n      options?.onProgress?.({ uploadedBytes: curUploaded, totalBytes, uploadedFiles, totalFiles, uploadSpeed: timeElapsed ? curUploaded / (timeElapsed / 1000) : 0 });\n    }\n\n    fireProgressEvent(0);\n\n    for (const [idx, file] of this.#files.entries()) {\n      for (let offset = 0; offset < file.size; offset += instructions.chunkSize) {\n        let data = file.data.slice(offset, offset + instructions.chunkSize);\n        // Data needs to be a real Blob for XMLHttpRequest to work\n        if (!isRealBlob(data))\n          data = await convertToRealBlob(data);\n        const uploadurl = `${instructions.baseUrl}&offset=${offset}&file=${idx}`;\n\n\n        /* we can't use fetch as we can't do progress tracking there!\n           there's https://stackoverflow.com/questions/35711724/upload-progress-indicators-for-fetch but it requires duplex: half which requires QUIC/H2\n\n           https://fetch.spec.whatwg.org/#fetch-api says\n           The fetch() method is relatively low-level API for fetching resources. It covers slightly more ground than XMLHttpRequest, although it is currently lacking when it comes to request progression (not response progression).\n\n        let bytesUploaded = 0;\n        const progressTrackingStream = new TransformStream<Uint8Array, Uint8Array>({\n          transform(chunk: Uint8Array, controller: TransformStreamDefaultController<Uint8Array>) {\n            controller.enqueue(chunk);\n            bytesUploaded += chunk.byteLength;\n            console.log(\"upload progress:\", bytesUploaded / data.size);\n            // uploadProgress.value = bytesUploaded / totalBytes;\n          },\n          flush(controller: TransformStreamDefaultController<Uint8Array>) {\n            console.log(\"completed stream\");\n          },\n        });\n           */\n        if (options?.signal?.aborted)\n          throw new Error(\"Upload has been aborted\");\n\n        if (globalThis.XMLHttpRequest) { //let's hope by the time browsers drop XMLHttpRequest, fetch finally has proper progress\n          const defer = Promise.withResolvers<void>();\n          const xmlhttp = new globalThis.XMLHttpRequest;\n          xmlhttp.overrideMimeType(\"application/octet-stream\");\n          xmlhttp.upload.addEventListener('progress', ev => fireProgressEvent(ev.loaded));\n          xmlhttp.addEventListener('abort', (ev: ProgressEvent<XMLHttpRequestEventTarget>) => defer.reject(new Error(\"Upload has been aborted\")));\n          xmlhttp.addEventListener('error', (ev: ProgressEvent<XMLHttpRequestEventTarget>) => defer.reject(new Error(\"Error\")));\n          xmlhttp.addEventListener('load', () => defer.resolve()); //invoked on success\n          xmlhttp.addEventListener('loadend', (ev: ProgressEvent<XMLHttpRequestEventTarget>) => { //invoked after either abort/error/load\n          });\n          xmlhttp.open(\"POST\", uploadurl, true);\n          xmlhttp.send(data);\n\n          const doAbort = () => xmlhttp.abort();\n          options?.signal?.addEventListener(\"abort\", doAbort);\n          await defer.promise;\n          options?.signal?.removeEventListener(\"abort\", doAbort);\n        } else { //fallback to fetch(). needed on Node, but it'll break progress reporting\n          const uploadresult = await fetch(uploadurl, {\n            method: \"POST\",\n            headers: { \"Content-Type\": \"application/octet-stream\" },\n            body: data // At some point... .stream().pipeThrough(progressTrackingStream) - but see above why it won't work yet\n          });\n\n          if (!uploadresult.ok)\n            throw new Error(`Upload failed`);\n        }\n\n        uploadedBytes += data.size;\n      }\n\n      outfiles.push({ ...this.manifest.files[idx], token: instructions.sessionId + '#' + idx });\n      ++uploadedFiles;\n      fireProgressEvent(0);\n    }\n\n    return outfiles;\n  }\n}\n\nexport class SingleFileUploader implements UploaderBase {\n  uploader;\n\n  get manifest() {\n    return this.uploader.manifest;\n  }\n\n  constructor(file: File | Blob | ResourceDescriptorCompatible) {\n    this.uploader = new MultiFileUploader([file]);\n  }\n\n  async upload(instructions: UploadInstructions, options?: UploadOptions): Promise<UploadResult> {\n    return (await this.uploader.upload(instructions, options))[0];\n  }\n}\n\nasync function getFilelistFromUser(multiple: boolean, accept: string[]): Promise<File[]> {\n  const defer = Promise.withResolvers<File[]>();\n  if (dompack.dispatchCustomEvent(window, \"wh:requestfiles\", { bubbles: true, cancelable: true, detail: { resolve: defer.resolve, multiple, accept } })) {\n    const input = document.createElement('input');\n    input.type = \"file\";\n    input.multiple = multiple;\n\n    if (accept.length)\n      input.accept = accept.join(\",\");\n    input.addEventListener(\"change\", () => defer.resolve([...input.files || []]));\n    input.addEventListener(\"cancel\", () => defer.resolve([]));\n    input.showPicker();\n  }\n\n  const list = await defer.promise;\n  if (!multiple && list.length > 1)\n    throw new Error(`wh:requestfiles intercepter selected multiple files, but only one was requested`);\n  return list;\n}\n\nexport async function requestFiles(options?: UploadRequestOptions): Promise<File[] | null> {\n  const files = await getFilelistFromUser(!(options?.multiple === false), options?.accept || []);\n  if (!files.length)\n    return null;\n\n  return files;\n}\n\n// We're adding a separate interface for single-file uploads as it's quite annoying to have to deal with interfaces generalized for multiple files if you really, really know you only ever wanted one file anyway\nexport async function requestFile(options?: UploadRequestOptions): Promise<File | null> {\n  const files = await getFilelistFromUser(false, options?.accept || []);\n  if (files.length !== 1)\n    return null;\n\n  return files[0];\n}\n\n/** Convert a file to a data: URL\n * @param file - The file to convert\n * @returns A promise that resolves to the data: URL\n*/\nexport function getFileAsDataURL(file: File): Promise<string> {\n  return new Promise((resolve, reject) => {\n    const reader = new FileReader;\n    // MDN: When the read operation is finished… the loadend event is triggered.\n    reader.addEventListener(\"loadend\", () => {\n      //At that time, the result attribute contains the data as a data: URL representing the file's data as a base64 encoded string.\n      resolve(reader.result as string);\n    });\n    reader.addEventListener(\"error\", () => reject(new Error(\"Failed to load file\")));\n    reader.readAsDataURL(file);\n  });\n}\n"],"mappings":"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","names":["formatForConsoleLogs","debugFlags","key","runSettingsCallbacks","cb","settingschangedcallbacks","initEnv","setDtapStage","setBackendBase","dtapStage","isLive","backendBase","dtapstage","islive","globalDebugFlags","localDebugFlags","debugFlagsOverridesCB","DebugFlagsProxyHandler","init_envbackend","__esmMin","target","p","record","keys","newValue","toModify","value","init_polyfills","__esmMin","resolve","reject","_resolve","_reject","items","callbackfn","retval","idx","item","key","list","submitForm","action","values","method","form","item","input","navigateTo","navigation","init_navigation","__esmMin","getWHType","obj","isCrossRealm","value","isMoney","isTemporalInstant","isTemporalPlainDateTime","isTemporalPlainDate","isTemporalZonedDateTime","isDate","isBlob","isFile","isPromise","e","stdTypeOf","t","init_quacks","__esmMin","parseMS","ms","parseDuration","duration","matches","addDuration","startingdate","isTemporalInstant","direction","date","modifydays","timeoffset","convertFlexibleInstantToDate","input","convertWaitPeriodToDate","wait","options","isDate","isValidDate","year","month","day","minYear","isleapyear","init_datetime","__esmMin","init_quacks","sleep","duration","options","until","convertWaitPeriodToDate","resolve","timeoutid","createDeferred","wrapSerialized","fn","Coalescer","queue","args","res","init_promises","__esmMin","init_datetime","startpos","currentpos","result","i","e","stripUnneededDecimals","num","decimals","isneg","toText","amount","decimalpoint","mindecimals","isnegative","astext","splitValue","money","split","sign","joinPrice","parts","adjustDecimals","requiredecimals","toadd","multiplier","idx","__add","lhs","rhs","add","amount1","amount2","subtract","tosubtract","__multiply","multiply","cmp","diff","getPercentageOfAmount","perc","result","normalize","__roundIntegerToMultiple","value","roundunit","mode","roundToMultiple","unit","split_value","split_unit","min","amounts","val","max","getNonNegativePowerOf10","exp","retval","running_exp","divide","divisor","mul10exp","mulfactor","test","relation","compareresult","init_finmath","__esmMin","stripUnneededDecimals","num","decimals","isneg","splitValue","money","split","sign","toText","amount","decimalpoint","mindecimals","thousandpoint","isnegative","astext","beforepoint","afterpoint","Money","init_money","__esmMin","init_finmath","init_quacks","_Money","value","intvalue","param","isMoney","left","right","sum","item","add","subtract","roundto","mode","roundToMultiple","lhs","rhs","cmp","relation","test","multiply","percentage","getPercentageOfAmount","numerator","divisor","divide","amounts","min","max","format","nameToCamelCase","name","c","nameToSnakeCase","isUInt8Array","value","compare","left","right","right_number","Money","left_number","isDate","left_value","right_value","compareLength","i","isTemporalInstant","isTemporalPlainDate","isTemporalZonedDateTime","stdTypeOf","compareProperties","fields","compareList","field","a","b","negate","res","partialFields","init_types","__esmMin","init_money","init_quacks","escapeRegExp","text","options","mask","isHTMLUnrepresentableChar","curch","encodeEntities","str","html","s","char","decodeEntities","_","dec","encodeString","encoding","decodeString","stringify","arg","usereplacer","key","value","origvalue","type","stdTypeOf","lhs","rhs","result","parseTyped","input","Money","isValidEmail","email","levenshteinDistance","a","b","_min","d0","d1","d2","bx","ay","tmp","la","lb","offset","x","y","d3","dd","dy","bx0","bx1","bx2","bx3","vector","len","init_strings","__esmMin","init_quacks","init_money","generateRandomId","encoding","bytes","u8array","x","init_platformbased","__esmMin","emplace","map","key","handler","current","setvalue","pick","value","keys","elt","ret","omit","val","init_collections","__esmMin","init_utils","__esmMin","unlock","LocalMutex","LocalLock","init_localmutex","__esmMin","defer","mutex","throwError","err","init_std","__esmMin","init_promises","init_types","init_strings","init_platformbased","init_collections","init_money","init_datetime","init_quacks","init_utils","init_localmutex","init_fetchdebug","__esmMin","init_envbackend","init_std","init_env","__esmMin","init_envbackend","init_navigation","init_fetchdebug","dispatchDomEvent","element","eventtype","options","createtype","evt","fireModifiedEvents","fireHTMLEvent","type","dispatchCustomEvent","node","event","params","prop","CustomEvent","defaultaction","changeValue","newvalue","asString","initKeyMapping","keydata","normalizeKeyboardEventData","key","stop","addDocEventListener","listener","init_events","__esmMin","getLockmgr","mylockmgr","getParentLockManager","LockManager","anyModalLocks","locallocks","l","scheduleCheckUIFree","uiwatcher","checkUIFree","modallocked","toggleBusyModal","lockmgr","isDialogElement","el","checkCancelEvent","evt","show","dispatchCustomEvent","installedanticancelhandler","busymodalcontent","currentbusymodaluserdialog","dialog","toembed","currentbusymodaldialog","flagUIBusy","options","BusyLock","debugFlags","locallockmgr","lock","init_busy","__esmMin","init_env","init_events","init_polyfills","pos","lockpos","qS","node_or_selector","selector","qR","matches","qSA","isFormControl","field","isHTMLElement","isElement","node","getRelativeBounds","relativeto","nodecoords","relcoords","init_tree","__esmMin","generateInsertList","nodes","frag","node","matches","selector","closest","testNode","contains","ancestor","child","before","after","replaceWith","remove","prepend","append","toggleClass","classname","settoggle","toggleClasses","toggles","key","empty","isDomReady","onDomReady","callback","getJSONAttribute","attributename","e","getBaseURI","doc","base","setStyles","value","propvalue","IS_NON_DIMENSIONAL","init_tree","__esmMin","flattenArray","list","acc","elt","setClassName","node","value","str","key","idx","toDashed","MATCH_UPCASE","attrHasBooleanValue","propname","createElement","elementname","attributes","toattrs","attrname","eventname","setStyles","append","child","create","children","el","jsxcreate","element","childNodes","parts","jsxfragment","inp","frag","init_create","__esmMin","init_tree","isNodeCompletelyInDom","node","processRegistration","item","reg","domready","map","list","applyRegistration","startnode","isDomReady","items","e","focus","options","dispatchCustomEvent","scrollIntoView","register","selector","handler","newreg","components","onDomReady","registerMissed","init_components","__esmMin","init_tree","init_events","buildCookieHeader","name","value","options","header","date","init_cookiebuilder","__esmMin","listCookies","cookie","parts","setCookie","key","value","options","buildCookieHeader","getCookie","escapeRegExp","deleteCookie","get","storage","foundvalue","backup","parseTyped","set","tostore","stringify","store","getSession","setSession","getLocal","setLocal","init_storage","__esmMin","init_cookiebuilder","init_std","loadImage","imgsrc","resolve","reject","img","loadScript","scriptsrc","module","scripttag","loadCSS","src","element","retval","init_preload","__esmMin","parseUserAgent","ua","UA","platform","name","version","device","isMultiSelectKey","event","browser","getBrowser","submitselector","init_browser","__esmMin","init_dompack","__esmMin","init_busy","init_events","init_tree","init_create","init_components","init_storage","init_preload","init_browser","dompack_exports","__export","after","append","before","changeValue","closest","contains","create","createDeferred","debugflags","dispatchCustomEvent","dispatchDomEvent","empty","fireModifiedEvents","flagUIBusy","focus","getBaseURI","getJSONAttribute","getRelativeBounds","isDomReady","jsxcreate","jsxfragment","matches","normalizeKeyboardEventData","onDomReady","prepend","qS","qSA","register","registerMissed","remove","replaceWith","scrollIntoView","setStyles","stop","toggleClass","toggleClasses","init_dompack","__esmMin","init_std","init_events","init_tree","init_create","init_components","init_env","debugFlags","parseUserAgent","ua","UA","platform","getName","browser","getVersion","getPlatform","init_browser","__esmMin","parse","stackString","lines","stack","line","parseResult","parseChrome","parseWinjs","parseGecko","parseNode","parseJSC","parts","chromeRe","isNative","isEval","submatch","chromeEvalRe","UNKNOWN_FUNCTION","winjsRe","geckoRe","geckoEvalRe","javaScriptCoreRe","nodeRe","init_stack_trace_parser_esm","__esmMin","parseTrace","e","parse","i","formatTrace","trace","prependStackTrace","error","stacklines","tracelines","init_stacktracing","__esmMin","init_stack_trace_parser_esm","init_levenshtein","__esmMin","init_std","init_js_api_tools","__esmMin","init_stacktracing","init_levenshtein","getDebugAppend","urldebugvar","createClient","servicename","options","rpcclient","RPCClient","ServiceProxy","globalseqnr","ControlledCall","init_jsonrpc_client","__esmMin","init_env","init_js_api_tools","client","method","stack","id","callurl","fetchoptions","fetchpromise","requestStack","response","retryafter","resolve","exception","jsonresponse","logitem","err","prependStackTrace","url","debugFlags","error","backendBase","params","parseTrace","target","prop","receiver","newoptions","args","getFormService","hsformservice","createClient","getTSFormService","tsformservice","init_formservice","__esmMin","init_jsonrpc_client","debugflags","init_debug","__esmMin","init_env","debugFlags","flagname","getFinalKey","event","getKeyNames","names","propnames","propname","modifier","arr","newkeys","singlemodifier","validateKeyName","key","modifiers","mod","original_order","getEventKeyNames","keydata","normalizeKeyboardEventData","IS_MAC_PLATFORM","KeyboardHandler","init_keyboard","__esmMin","init_events","init_debug","init_browser","getPlatform","node","keymap","options","name","keyname","debugflags","target","keynames","tag","is_special_combo","keybinding","handler","i","mapping","getGetTidHooks","gotHooks","hookFactory","init_hooks","__esmMin","internal_exports","__export","getHTMLTid","getHTMLTidForLanguage","getTIDListForLanguage","getTid","getTidForLanguage","getTidLanguage","registerTexts","setTidLanguage","ensureHooks","hooks","getGetTidHooks","lang","browserSetLanguage","defaultLanguage","newLanguage","getLanguageFile","module","langCode","langkey","compiled","langFileCache","loader","e","dtapStage","debugFlags","texts","toHTMLNode","tag","props","node","wrapInFragment","children","fragment","getTidInternal","tid","params","options","decodeString","nodes","calcTIDForLanguage","p","p1","p2","p3","p4","encodeString","langcode","gid","getCanonicalTid","modsep","lookup","result","key","fallback","cannotFind","rawParams","rich","render","getLanguageDatetimeStrings","debugtid","match","fallbackMatch","executeCompiledTidText","renderString","tok","part","_","text","parts","get_param","sub","link","flattenRecursiveLanguageTexts","pathsofar","name","value","tids","compiledFile","init_internal","__esmMin","init_std","init_hooks","init_env","init_gettid","__esmMin","init_internal","isRealBlob","data","convertToRealBlob","getFilelistFromUser","multiple","accept","defer","dispatchCustomEvent","input","list","requestFile","options","files","getFileAsDataURL","file","resolve","reject","reader","MultiFileUploader","SingleFileUploader","init_upload","__esmMin","init_dompack","#files","signal","item","instructions","outfiles","uploadedBytes","uploadedFiles","totalBytes","acc","totalFiles","start","fireProgressEvent","partialbytes","curUploaded","timeElapsed","idx","offset","uploadurl","xmlhttp","ev","doAbort"]}