{"version":3,"file":"main.js","sources":["../node_modules/tslib/tslib.es6.js","../src-client/oembed.ts","../src-client/render.ts","../src-client/api.ts","../src-client/navigation.ts","../node_modules/svelte/src/runtime/internal/utils.js","../node_modules/svelte/src/runtime/internal/dom.js","../node_modules/svelte/src/runtime/internal/lifecycle.js","../node_modules/svelte/src/runtime/internal/scheduler.js","../node_modules/svelte/src/runtime/internal/transitions.js","../node_modules/svelte/src/runtime/internal/Component.js","../src-client/components/AuthButton.svelte","../node_modules/svelte/src/runtime/internal/disclose-version/index.js","../node_modules/svelte/src/shared/version.js","../src-client/gtag.ts","../src-client/components/CookieConsentPopup.svelte","../src-client/main.ts"],"sourcesContent":["/******************************************************************************\r\nCopyright (c) Microsoft Corporation.\r\n\r\nPermission to use, copy, modify, and/or distribute this software for any\r\npurpose with or without fee is hereby granted.\r\n\r\nTHE SOFTWARE IS PROVIDED \"AS IS\" AND THE AUTHOR DISCLAIMS ALL WARRANTIES WITH\r\nREGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY\r\nAND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT,\r\nINDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM\r\nLOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR\r\nOTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR\r\nPERFORMANCE OF THIS SOFTWARE.\r\n***************************************************************************** */\r\n/* global Reflect, Promise, SuppressedError, Symbol */\r\n\r\nvar extendStatics = function(d, b) {\r\n extendStatics = Object.setPrototypeOf ||\r\n ({ __proto__: [] } instanceof Array && function (d, b) { d.__proto__ = b; }) ||\r\n function (d, b) { for (var p in b) if (Object.prototype.hasOwnProperty.call(b, p)) d[p] = b[p]; };\r\n return extendStatics(d, b);\r\n};\r\n\r\nexport function __extends(d, b) {\r\n if (typeof b !== \"function\" && b !== null)\r\n throw new TypeError(\"Class extends value \" + String(b) + \" is not a constructor or null\");\r\n extendStatics(d, b);\r\n function __() { this.constructor = d; }\r\n d.prototype = b === null ? Object.create(b) : (__.prototype = b.prototype, new __());\r\n}\r\n\r\nexport var __assign = function() {\r\n __assign = Object.assign || function __assign(t) {\r\n for (var s, i = 1, n = arguments.length; i < n; i++) {\r\n s = arguments[i];\r\n for (var p in s) if (Object.prototype.hasOwnProperty.call(s, p)) t[p] = s[p];\r\n }\r\n return t;\r\n }\r\n return __assign.apply(this, arguments);\r\n}\r\n\r\nexport function __rest(s, e) {\r\n var t = {};\r\n for (var p in s) if (Object.prototype.hasOwnProperty.call(s, p) && e.indexOf(p) < 0)\r\n t[p] = s[p];\r\n if (s != null && typeof Object.getOwnPropertySymbols === \"function\")\r\n for (var i = 0, p = Object.getOwnPropertySymbols(s); i < p.length; i++) {\r\n if (e.indexOf(p[i]) < 0 && Object.prototype.propertyIsEnumerable.call(s, p[i]))\r\n t[p[i]] = s[p[i]];\r\n }\r\n return t;\r\n}\r\n\r\nexport function __decorate(decorators, target, key, desc) {\r\n var c = arguments.length, r = c < 3 ? target : desc === null ? desc = Object.getOwnPropertyDescriptor(target, key) : desc, d;\r\n if (typeof Reflect === \"object\" && typeof Reflect.decorate === \"function\") r = Reflect.decorate(decorators, target, key, desc);\r\n else for (var i = decorators.length - 1; i >= 0; i--) if (d = decorators[i]) r = (c < 3 ? d(r) : c > 3 ? d(target, key, r) : d(target, key)) || r;\r\n return c > 3 && r && Object.defineProperty(target, key, r), r;\r\n}\r\n\r\nexport function __param(paramIndex, decorator) {\r\n return function (target, key) { decorator(target, key, paramIndex); }\r\n}\r\n\r\nexport function __esDecorate(ctor, descriptorIn, decorators, contextIn, initializers, extraInitializers) {\r\n function accept(f) { if (f !== void 0 && typeof f !== \"function\") throw new TypeError(\"Function expected\"); return f; }\r\n var kind = contextIn.kind, key = kind === \"getter\" ? \"get\" : kind === \"setter\" ? \"set\" : \"value\";\r\n var target = !descriptorIn && ctor ? contextIn[\"static\"] ? ctor : ctor.prototype : null;\r\n var descriptor = descriptorIn || (target ? Object.getOwnPropertyDescriptor(target, contextIn.name) : {});\r\n var _, done = false;\r\n for (var i = decorators.length - 1; i >= 0; i--) {\r\n var context = {};\r\n for (var p in contextIn) context[p] = p === \"access\" ? {} : contextIn[p];\r\n for (var p in contextIn.access) context.access[p] = contextIn.access[p];\r\n context.addInitializer = function (f) { if (done) throw new TypeError(\"Cannot add initializers after decoration has completed\"); extraInitializers.push(accept(f || null)); };\r\n var result = (0, decorators[i])(kind === \"accessor\" ? { get: descriptor.get, set: descriptor.set } : descriptor[key], context);\r\n if (kind === \"accessor\") {\r\n if (result === void 0) continue;\r\n if (result === null || typeof result !== \"object\") throw new TypeError(\"Object expected\");\r\n if (_ = accept(result.get)) descriptor.get = _;\r\n if (_ = accept(result.set)) descriptor.set = _;\r\n if (_ = accept(result.init)) initializers.unshift(_);\r\n }\r\n else if (_ = accept(result)) {\r\n if (kind === \"field\") initializers.unshift(_);\r\n else descriptor[key] = _;\r\n }\r\n }\r\n if (target) Object.defineProperty(target, contextIn.name, descriptor);\r\n done = true;\r\n};\r\n\r\nexport function __runInitializers(thisArg, initializers, value) {\r\n var useValue = arguments.length > 2;\r\n for (var i = 0; i < initializers.length; i++) {\r\n value = useValue ? initializers[i].call(thisArg, value) : initializers[i].call(thisArg);\r\n }\r\n return useValue ? value : void 0;\r\n};\r\n\r\nexport function __propKey(x) {\r\n return typeof x === \"symbol\" ? x : \"\".concat(x);\r\n};\r\n\r\nexport function __setFunctionName(f, name, prefix) {\r\n if (typeof name === \"symbol\") name = name.description ? \"[\".concat(name.description, \"]\") : \"\";\r\n return Object.defineProperty(f, \"name\", { configurable: true, value: prefix ? \"\".concat(prefix, \" \", name) : name });\r\n};\r\n\r\nexport function __metadata(metadataKey, metadataValue) {\r\n if (typeof Reflect === \"object\" && typeof Reflect.metadata === \"function\") return Reflect.metadata(metadataKey, metadataValue);\r\n}\r\n\r\nexport function __awaiter(thisArg, _arguments, P, generator) {\r\n function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }\r\n return new (P || (P = Promise))(function (resolve, reject) {\r\n function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }\r\n function rejected(value) { try { step(generator[\"throw\"](value)); } catch (e) { reject(e); } }\r\n function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }\r\n step((generator = generator.apply(thisArg, _arguments || [])).next());\r\n });\r\n}\r\n\r\nexport function __generator(thisArg, body) {\r\n var _ = { label: 0, sent: function() { if (t[0] & 1) throw t[1]; return t[1]; }, trys: [], ops: [] }, f, y, t, g;\r\n return g = { next: verb(0), \"throw\": verb(1), \"return\": verb(2) }, typeof Symbol === \"function\" && (g[Symbol.iterator] = function() { return this; }), g;\r\n function verb(n) { return function (v) { return step([n, v]); }; }\r\n function step(op) {\r\n if (f) throw new TypeError(\"Generator is already executing.\");\r\n while (g && (g = 0, op[0] && (_ = 0)), _) try {\r\n if (f = 1, y && (t = op[0] & 2 ? y[\"return\"] : op[0] ? y[\"throw\"] || ((t = y[\"return\"]) && t.call(y), 0) : y.next) && !(t = t.call(y, op[1])).done) return t;\r\n if (y = 0, t) op = [op[0] & 2, t.value];\r\n switch (op[0]) {\r\n case 0: case 1: t = op; break;\r\n case 4: _.label++; return { value: op[1], done: false };\r\n case 5: _.label++; y = op[1]; op = [0]; continue;\r\n case 7: op = _.ops.pop(); _.trys.pop(); continue;\r\n default:\r\n if (!(t = _.trys, t = t.length > 0 && t[t.length - 1]) && (op[0] === 6 || op[0] === 2)) { _ = 0; continue; }\r\n if (op[0] === 3 && (!t || (op[1] > t[0] && op[1] < t[3]))) { _.label = op[1]; break; }\r\n if (op[0] === 6 && _.label < t[1]) { _.label = t[1]; t = op; break; }\r\n if (t && _.label < t[2]) { _.label = t[2]; _.ops.push(op); break; }\r\n if (t[2]) _.ops.pop();\r\n _.trys.pop(); continue;\r\n }\r\n op = body.call(thisArg, _);\r\n } catch (e) { op = [6, e]; y = 0; } finally { f = t = 0; }\r\n if (op[0] & 5) throw op[1]; return { value: op[0] ? op[1] : void 0, done: true };\r\n }\r\n}\r\n\r\nexport var __createBinding = Object.create ? (function(o, m, k, k2) {\r\n if (k2 === undefined) k2 = k;\r\n var desc = Object.getOwnPropertyDescriptor(m, k);\r\n if (!desc || (\"get\" in desc ? !m.__esModule : desc.writable || desc.configurable)) {\r\n desc = { enumerable: true, get: function() { return m[k]; } };\r\n }\r\n Object.defineProperty(o, k2, desc);\r\n}) : (function(o, m, k, k2) {\r\n if (k2 === undefined) k2 = k;\r\n o[k2] = m[k];\r\n});\r\n\r\nexport function __exportStar(m, o) {\r\n for (var p in m) if (p !== \"default\" && !Object.prototype.hasOwnProperty.call(o, p)) __createBinding(o, m, p);\r\n}\r\n\r\nexport function __values(o) {\r\n var s = typeof Symbol === \"function\" && Symbol.iterator, m = s && o[s], i = 0;\r\n if (m) return m.call(o);\r\n if (o && typeof o.length === \"number\") return {\r\n next: function () {\r\n if (o && i >= o.length) o = void 0;\r\n return { value: o && o[i++], done: !o };\r\n }\r\n };\r\n throw new TypeError(s ? \"Object is not iterable.\" : \"Symbol.iterator is not defined.\");\r\n}\r\n\r\nexport function __read(o, n) {\r\n var m = typeof Symbol === \"function\" && o[Symbol.iterator];\r\n if (!m) return o;\r\n var i = m.call(o), r, ar = [], e;\r\n try {\r\n while ((n === void 0 || n-- > 0) && !(r = i.next()).done) ar.push(r.value);\r\n }\r\n catch (error) { e = { error: error }; }\r\n finally {\r\n try {\r\n if (r && !r.done && (m = i[\"return\"])) m.call(i);\r\n }\r\n finally { if (e) throw e.error; }\r\n }\r\n return ar;\r\n}\r\n\r\n/** @deprecated */\r\nexport function __spread() {\r\n for (var ar = [], i = 0; i < arguments.length; i++)\r\n ar = ar.concat(__read(arguments[i]));\r\n return ar;\r\n}\r\n\r\n/** @deprecated */\r\nexport function __spreadArrays() {\r\n for (var s = 0, i = 0, il = arguments.length; i < il; i++) s += arguments[i].length;\r\n for (var r = Array(s), k = 0, i = 0; i < il; i++)\r\n for (var a = arguments[i], j = 0, jl = a.length; j < jl; j++, k++)\r\n r[k] = a[j];\r\n return r;\r\n}\r\n\r\nexport function __spreadArray(to, from, pack) {\r\n if (pack || arguments.length === 2) for (var i = 0, l = from.length, ar; i < l; i++) {\r\n if (ar || !(i in from)) {\r\n if (!ar) ar = Array.prototype.slice.call(from, 0, i);\r\n ar[i] = from[i];\r\n }\r\n }\r\n return to.concat(ar || Array.prototype.slice.call(from));\r\n}\r\n\r\nexport function __await(v) {\r\n return this instanceof __await ? (this.v = v, this) : new __await(v);\r\n}\r\n\r\nexport function __asyncGenerator(thisArg, _arguments, generator) {\r\n if (!Symbol.asyncIterator) throw new TypeError(\"Symbol.asyncIterator is not defined.\");\r\n var g = generator.apply(thisArg, _arguments || []), i, q = [];\r\n return i = {}, verb(\"next\"), verb(\"throw\"), verb(\"return\"), i[Symbol.asyncIterator] = function () { return this; }, i;\r\n function verb(n) { if (g[n]) i[n] = function (v) { return new Promise(function (a, b) { q.push([n, v, a, b]) > 1 || resume(n, v); }); }; }\r\n function resume(n, v) { try { step(g[n](v)); } catch (e) { settle(q[0][3], e); } }\r\n function step(r) { r.value instanceof __await ? Promise.resolve(r.value.v).then(fulfill, reject) : settle(q[0][2], r); }\r\n function fulfill(value) { resume(\"next\", value); }\r\n function reject(value) { resume(\"throw\", value); }\r\n function settle(f, v) { if (f(v), q.shift(), q.length) resume(q[0][0], q[0][1]); }\r\n}\r\n\r\nexport function __asyncDelegator(o) {\r\n var i, p;\r\n return i = {}, verb(\"next\"), verb(\"throw\", function (e) { throw e; }), verb(\"return\"), i[Symbol.iterator] = function () { return this; }, i;\r\n function verb(n, f) { i[n] = o[n] ? function (v) { return (p = !p) ? { value: __await(o[n](v)), done: false } : f ? f(v) : v; } : f; }\r\n}\r\n\r\nexport function __asyncValues(o) {\r\n if (!Symbol.asyncIterator) throw new TypeError(\"Symbol.asyncIterator is not defined.\");\r\n var m = o[Symbol.asyncIterator], i;\r\n return m ? m.call(o) : (o = typeof __values === \"function\" ? __values(o) : o[Symbol.iterator](), i = {}, verb(\"next\"), verb(\"throw\"), verb(\"return\"), i[Symbol.asyncIterator] = function () { return this; }, i);\r\n function verb(n) { i[n] = o[n] && function (v) { return new Promise(function (resolve, reject) { v = o[n](v), settle(resolve, reject, v.done, v.value); }); }; }\r\n function settle(resolve, reject, d, v) { Promise.resolve(v).then(function(v) { resolve({ value: v, done: d }); }, reject); }\r\n}\r\n\r\nexport function __makeTemplateObject(cooked, raw) {\r\n if (Object.defineProperty) { Object.defineProperty(cooked, \"raw\", { value: raw }); } else { cooked.raw = raw; }\r\n return cooked;\r\n};\r\n\r\nvar __setModuleDefault = Object.create ? (function(o, v) {\r\n Object.defineProperty(o, \"default\", { enumerable: true, value: v });\r\n}) : function(o, v) {\r\n o[\"default\"] = v;\r\n};\r\n\r\nexport function __importStar(mod) {\r\n if (mod && mod.__esModule) return mod;\r\n var result = {};\r\n if (mod != null) for (var k in mod) if (k !== \"default\" && Object.prototype.hasOwnProperty.call(mod, k)) __createBinding(result, mod, k);\r\n __setModuleDefault(result, mod);\r\n return result;\r\n}\r\n\r\nexport function __importDefault(mod) {\r\n return (mod && mod.__esModule) ? mod : { default: mod };\r\n}\r\n\r\nexport function __classPrivateFieldGet(receiver, state, kind, f) {\r\n if (kind === \"a\" && !f) throw new TypeError(\"Private accessor was defined without a getter\");\r\n if (typeof state === \"function\" ? receiver !== state || !f : !state.has(receiver)) throw new TypeError(\"Cannot read private member from an object whose class did not declare it\");\r\n return kind === \"m\" ? f : kind === \"a\" ? f.call(receiver) : f ? f.value : state.get(receiver);\r\n}\r\n\r\nexport function __classPrivateFieldSet(receiver, state, value, kind, f) {\r\n if (kind === \"m\") throw new TypeError(\"Private method is not writable\");\r\n if (kind === \"a\" && !f) throw new TypeError(\"Private accessor was defined without a setter\");\r\n if (typeof state === \"function\" ? receiver !== state || !f : !state.has(receiver)) throw new TypeError(\"Cannot write private member to an object whose class did not declare it\");\r\n return (kind === \"a\" ? f.call(receiver, value) : f ? f.value = value : state.set(receiver, value)), value;\r\n}\r\n\r\nexport function __classPrivateFieldIn(state, receiver) {\r\n if (receiver === null || (typeof receiver !== \"object\" && typeof receiver !== \"function\")) throw new TypeError(\"Cannot use 'in' operator on non-object\");\r\n return typeof state === \"function\" ? receiver === state : state.has(receiver);\r\n}\r\n\r\nexport function __addDisposableResource(env, value, async) {\r\n if (value !== null && value !== void 0) {\r\n if (typeof value !== \"object\" && typeof value !== \"function\") throw new TypeError(\"Object expected.\");\r\n var dispose;\r\n if (async) {\r\n if (!Symbol.asyncDispose) throw new TypeError(\"Symbol.asyncDispose is not defined.\");\r\n dispose = value[Symbol.asyncDispose];\r\n }\r\n if (dispose === void 0) {\r\n if (!Symbol.dispose) throw new TypeError(\"Symbol.dispose is not defined.\");\r\n dispose = value[Symbol.dispose];\r\n }\r\n if (typeof dispose !== \"function\") throw new TypeError(\"Object not disposable.\");\r\n env.stack.push({ value: value, dispose: dispose, async: async });\r\n }\r\n else if (async) {\r\n env.stack.push({ async: true });\r\n }\r\n return value;\r\n}\r\n\r\nvar _SuppressedError = typeof SuppressedError === \"function\" ? SuppressedError : function (error, suppressed, message) {\r\n var e = new Error(message);\r\n return e.name = \"SuppressedError\", e.error = error, e.suppressed = suppressed, e;\r\n};\r\n\r\nexport function __disposeResources(env) {\r\n function fail(e) {\r\n env.error = env.hasError ? new _SuppressedError(e, env.error, \"An error was suppressed during disposal.\") : e;\r\n env.hasError = true;\r\n }\r\n function next() {\r\n while (env.stack.length) {\r\n var rec = env.stack.pop();\r\n try {\r\n var result = rec.dispose && rec.dispose.call(rec.value);\r\n if (rec.async) return Promise.resolve(result).then(next, function(e) { fail(e); return next(); });\r\n }\r\n catch (e) {\r\n fail(e);\r\n }\r\n }\r\n if (env.hasError) throw env.error;\r\n }\r\n return next();\r\n}\r\n\r\nexport default {\r\n __extends: __extends,\r\n __assign: __assign,\r\n __rest: __rest,\r\n __decorate: __decorate,\r\n __param: __param,\r\n __metadata: __metadata,\r\n __awaiter: __awaiter,\r\n __generator: __generator,\r\n __createBinding: __createBinding,\r\n __exportStar: __exportStar,\r\n __values: __values,\r\n __read: __read,\r\n __spread: __spread,\r\n __spreadArrays: __spreadArrays,\r\n __spreadArray: __spreadArray,\r\n __await: __await,\r\n __asyncGenerator: __asyncGenerator,\r\n __asyncDelegator: __asyncDelegator,\r\n __asyncValues: __asyncValues,\r\n __makeTemplateObject: __makeTemplateObject,\r\n __importStar: __importStar,\r\n __importDefault: __importDefault,\r\n __classPrivateFieldGet: __classPrivateFieldGet,\r\n __classPrivateFieldSet: __classPrivateFieldSet,\r\n __classPrivateFieldIn: __classPrivateFieldIn,\r\n __addDisposableResource: __addDisposableResource,\r\n __disposeResources: __disposeResources,\r\n};\r\n","export function renderOEmbed() {\r\n\tdocument.querySelectorAll(\"oembed[url]\").forEach( element => {\r\n\t\tlet vCode = (element as any).attributes.url.value.split(\"?v=\")[1];\r\n\t\t(element as any).parentElement.innerHTML = `\r\n\t\t\t\r\n\t\t\t`;\r\n\t});\r\n}\r\nwindow.addEventListener('load', renderOEmbed);\r\n","import {ChildPage, Page, PageGroup} from \"./dto\";\r\nimport {renderOEmbed} from \"./oembed\";\r\n\r\nexport const PAGE_IS_ABOUT_TO_CHANGE_EVENT = \"page-is-about-to-change\";\r\n\r\nexport const container = document.getElementById(\"container\")!;\r\nexport const pageLangDropDown = document.getElementById(\"page-lang-dropdown\")!;\r\nexport const pageLangButton = document.getElementById(\"page-lang-button\")!;\r\nexport const pageTitle = document.getElementById(\"page-title\")!;\r\nexport const pageText = document.getElementById(\"page-text\")!;\r\nexport const pageLangList = document.getElementById(\"page-lang-list\")!;\r\nexport const pageGroups = document.getElementById(\"page-groups\")!;\r\nexport const pageChildren = document.getElementById(\"page-children\")!;\r\nexport const pagePublicationDate = document.getElementById(\"page-publication-date\")!;\r\nexport const pageAuthor = document.getElementById(\"page-author\")!;\r\nexport const homeLink = document.getElementById(\"home-link\")!;\r\n\r\nexport function currentPageLangList(): (string | null)[] {\r\n\treturn Array.from(pageLangList.querySelectorAll(\"li\")).map(elem =>\r\n\t\telem.dataset.langCode !== undefined && elem.dataset.langCode !== \"\" ? elem.dataset.langCode : null\r\n\t);\r\n}\r\n\r\nexport function currentPageGroups(): PageGroup[] {\r\n\treturn Array.from(pageGroups.querySelectorAll(\"li\")).map(elem => {\r\n\t\treturn {\r\n\t\t\tid: parseInt(elem.dataset.id!),\r\n\t\t\tslug: elem.dataset.slug!,\r\n\t\t\ttitle: elem.querySelector(\"a\")!.innerText,\r\n\t\t\tlang: elem.dataset.langCode !== \"\" ? elem.dataset.langCode! : null,\r\n\t\t\tpinned: elem.dataset.pinned !== \"false\",\r\n\t\t\thidden: false\r\n\t\t};\r\n\t});\r\n}\r\n\r\nexport function currentPageChildren(): ChildPage[] {\r\n\treturn Array.from(pageChildren.children).map(elem => {\r\n\t\tconst htmlElem = elem as HTMLElement;\r\n\t\treturn {\r\n\t\t\tid: parseInt(htmlElem.dataset.id!),\r\n\t\t\tslug: htmlElem.dataset.slug!,\r\n\t\t\ttitle: (htmlElem.querySelector(\".child-page-title a\") as HTMLElement).innerText,\r\n\t\t\ttext: htmlElem.querySelector(\".child-page-text\")!.innerHTML,\r\n\t\t\tlang: htmlElem.dataset.langCode !== \"\" ? htmlElem.dataset.langCode! : null,\r\n\t\t\tisGroup: htmlElem.dataset.isGroup !== \"false\",\r\n\t\t\tpublishedAt: htmlElem.dataset.publishedAt !== \"\" ? htmlElem.dataset.publishedAt! : null,\r\n\t\t\tpinned: htmlElem.dataset.pinned !== \"false\",\r\n\t\t\tshowAuthor: htmlElem.dataset.showAuthor !== \"false\",\r\n\t\t\tauthorEmail: null,\r\n\t\t\tauthorDisplayName: null,\r\n\t\t\tshowPublishedAt: htmlElem.dataset.showPublishedAt !== \"false\",\r\n\t\t\thasMore: htmlElem.dataset.hasMore !== \"false\"\r\n\t\t};\r\n\t});\r\n}\r\n\r\nexport function currentPageId(): number {\r\n\treturn parseInt(container.dataset.id!);\r\n}\r\n\r\nexport function currentPageSlug(): string {\r\n\treturn container.dataset.slug!;\r\n}\r\n\r\nexport function currentPageLang(): string | null {\r\n\treturn container.dataset.lang !== \"\" ? container.dataset.lang! : null;\r\n}\r\n\r\nexport function currentPageIsGroup(): boolean {\r\n\treturn container.dataset.isGroup !== \"false\";\r\n}\r\n\r\nexport function currentPageIndex(): number {\r\n\treturn parseInt(pageChildren.dataset.page!);\r\n}\r\n\r\nexport function currentPagePublishedAt(): string | null {\r\n\treturn container.dataset.publishedAt !== \"\" ? container.dataset.publishedAt! : null;\r\n}\r\n\r\nexport function currentPageShowAuthor(): boolean {\r\n\treturn container.dataset.showAuthor !== \"false\";\r\n}\r\n\r\nexport function currentPageShowPublishedAt(): boolean {\r\n\treturn container.dataset.showPublishedAt !== \"false\";\r\n}\r\n\r\nexport function currentPage(): Page {\r\n\treturn {\r\n\t\tid: currentPageId(),\r\n\t\tslug: currentPageSlug(),\r\n\t\ttitle: pageTitle.innerText,\r\n\t\ttext: pageText.innerHTML,\r\n\t\tlang: currentPageLang(),\r\n\t\tavailableLangList: currentPageLangList(),\r\n\t\tisGroup: currentPageIsGroup(),\r\n\t\tgroups: currentPageGroups(),\r\n\t\tchildren: currentPageChildren(),\r\n\t\tpage: currentPageIndex(),\r\n\t\thasMore: pageChildren.dataset.hasMore! !== \"false\",\r\n\t\tpublishedAt: currentPagePublishedAt(),\r\n\t\tshowAuthor: currentPageShowAuthor(),\r\n\t\tauthorEmail: null,\r\n\t\tauthorDisplayName: null,\r\n\t\tshowPublishedAt: currentPageShowPublishedAt()\r\n\t};\r\n}\r\n\r\nconst getFlagEmoji = (countryCode: string) => {\r\n\tconst codePoints = (countryCode !== \"en\" ? countryCode : \"us\")\r\n\t\t.toUpperCase()\r\n\t\t.split(\"\")\r\n\t\t.map(char => 127397 + char.charCodeAt(0));\r\n\treturn String.fromCodePoint(...codePoints);\r\n}\r\n\r\nexport function renderPage(page: Page) {\r\n\tif (page.id !== currentPageId()) {\r\n\t\tdocument.dispatchEvent(new CustomEvent(PAGE_IS_ABOUT_TO_CHANGE_EVENT, {detail: page}));\r\n\t}\r\n\tif (page.lang !== null) {\r\n\t\tdocument.documentElement.setAttribute(\"lang\", page.lang);\r\n\t} else {\r\n\t\tdocument.documentElement.removeAttribute(\"lang\");\r\n\t}\r\n\tdocument.title = page.title;\r\n\tcontainer.dataset.id = page.id.toString();\r\n\tcontainer.dataset.slug = page.slug;\r\n\tcontainer.dataset.lang = page.lang !== null ? page.lang : \"\";\r\n\tif (page.lang !== null) {\r\n\t\tpageLangButton.innerText = getFlagEmoji(page.lang);\r\n\t} else {\r\n\t\tpageLangButton.innerHTML = \"🌎\";\r\n\t}\r\n\tcontainer.dataset.isGroup = page.isGroup ? \"true\" : \"false\";\r\n\tcontainer.dataset.publishedAt = page.publishedAt !== null ? page.publishedAt : \"\";\r\n\tcontainer.dataset.showAuthor = page.showAuthor ? \"true\" : \"false\";\r\n\tcontainer.dataset.showPublishedAt = page.showPublishedAt ? \"true\" : \"false\";\r\n\tpageTitle.innerText = page.title;\r\n\tpageText.innerHTML = page.text;\r\n\tpageLangDropDown.style.display = page.availableLangList.length <= 1 ? \"none\" : \"\";\r\n\tpageLangButton.title = localeStr(\"language\");\r\n\tpageLangList.innerHTML = \"\";\r\n\tfor (let lang of page.availableLangList) {\r\n\t\tif (lang === page.lang) {\r\n\t\t\tcontinue;\r\n\t\t}\r\n\t\tconst elem = document.createElement(\"li\");\r\n\t\telem.dataset.langCode = lang !== null ? lang : \"\";\r\n\t\tif (lang !== null) {\r\n\t\t\telem.innerHTML = ``;\r\n\t\t} else {\r\n\t\t\telem.innerHTML = ``;\r\n\t\t}\r\n\t\tpageLangList.appendChild(elem);\r\n\t}\r\n\tpageGroups.innerHTML = \"\";\r\n\tfor (let group of page.groups) {\r\n\t\tconst elem = document.createElement(\"li\");\r\n\t\telem.dataset.id = group.id.toString();\r\n\t\telem.dataset.slug = group.slug;\r\n\t\telem.dataset.langCode = group.lang !== null ? group.lang : \"\";\r\n\t\telem.dataset.pinned = group.pinned ? \"true\" : \"false\";\r\n\t\tconst linkElem = document.createElement(\"a\");\r\n\t\tlinkElem.href = makePageUrl(group);\r\n\t\tlinkElem.innerText = group.title;\r\n\t\telem.appendChild(linkElem);\r\n\t\tpageGroups.appendChild(elem);\r\n\t}\r\n\tpageAuthor.innerHTML = \"\";\r\n\tif (page.showAuthor && page.authorDisplayName !== null) {\r\n\t\tif (page.authorEmail !== null) {\r\n\t\t\tconst linkElem = document.createElement(\"a\");\r\n\t\t\tlinkElem.href = `mailto:${page.authorEmail}`;\r\n\t\t\tlinkElem.innerText = page.authorDisplayName;\r\n\t\t\tpageAuthor.appendChild(linkElem);\r\n\t\t} else {\r\n\t\t\tpageAuthor.innerText = page.authorDisplayName;\r\n\t\t}\r\n\t}\r\n\tpageChildren.innerHTML = \"\";\r\n\tfor (let child of page.children) {\r\n\t\tconst elem = document.createElement(\"div\");\r\n\t\telem.className = \"child-page\";\r\n\t\telem.dataset.id = child.id.toString();\r\n\t\telem.dataset.slug = child.slug;\r\n\t\telem.dataset.langCode = child.lang !== null ? child.lang : \"\";\r\n\t\telem.dataset.isGroup = child.isGroup ? \"true\" : \"false\";\r\n\t\telem.dataset.publishedAt = child.publishedAt !== null ? child.publishedAt : \"\";\r\n\t\telem.dataset.pinned = child.pinned ? \"true\" : \"false\";\r\n\t\telem.dataset.showAuthor = child.showAuthor ? \"true\" : \"false\";\r\n\t\telem.dataset.showPublishedAt = child.showPublishedAt ? \"true\" : \"false\";\r\n\t\telem.dataset.hasMore = child.hasMore ? \"true\" : \"false\";\r\n\t\tconst titleElem = document.createElement(\"h2\");\r\n\t\ttitleElem.className = \"child-page-title\";\r\n\t\tconst titleLink = document.createElement(\"a\");\r\n\t\ttitleLink.href = makePageUrl(child);\r\n\t\ttitleLink.innerText = child.title;\r\n\t\ttitleElem.appendChild(titleLink);\r\n\t\tif (child.isGroup) {\r\n\t\t\ttitleElem.append(\" 🖿\");\r\n\t\t}\r\n\t\tif (child.pinned) {\r\n\t\t\ttitleElem.append(\" 📌\");\r\n\t\t}\r\n\t\telem.appendChild(titleElem);\r\n\t\tconst textElem = document.createElement(\"div\");\r\n\t\ttextElem.className = \"child-page-text\";\r\n\t\ttextElem.innerHTML = child.text;\r\n\t\telem.appendChild(textElem);\r\n\t\tif (child.hasMore) {\r\n\t\t\tconst readMoreElem = document.createElement(\"div\");\r\n\t\t\treadMoreElem.className = \"child-page-read-more\";\r\n\t\t\tconst linkElem = document.createElement(\"a\");\r\n\t\t\tlinkElem.href = makePageUrl(child);\r\n\t\t\tlinkElem.innerText = localeStr(\"readMore\");\r\n\t\t\treadMoreElem.appendChild(linkElem);\r\n\t\t\telem.appendChild(readMoreElem);\r\n\t\t}\r\n\t\tconst footerElem = document.createElement(\"div\");\r\n\t\tfooterElem.className = \"child-page-footer\";\r\n\t\tconst authorElem = document.createElement(\"div\");\r\n\t\tauthorElem.className = \"child-page-author\";\r\n\t\tif (child.showAuthor && child.authorDisplayName !== null) {\r\n\t\t\tif (child.authorEmail !== null) {\r\n\t\t\t\tconst linkElem = document.createElement(\"a\");\r\n\t\t\t\tlinkElem.href = `mailto:${child.authorEmail}`;\r\n\t\t\t\tlinkElem.innerText = child.authorDisplayName;\r\n\t\t\t\tauthorElem.appendChild(linkElem);\r\n\t\t\t} else {\r\n\t\t\t\tauthorElem.innerText = child.authorDisplayName;\r\n\t\t\t}\r\n\t\t}\r\n\t\tfooterElem.appendChild(authorElem);\r\n\t\tconst publicationDateElem = document.createElement(\"div\");\r\n\t\tpublicationDateElem.className = \"child-page-publication-date\";\r\n\t\tfooterElem.append(publicationDateElem);\r\n\t\telem.appendChild(footerElem);\r\n\t\tpageChildren.appendChild(elem);\r\n\t}\r\n\tpageChildren.dataset.page = page.page.toString();\r\n\tpageChildren.dataset.hasMore = page.hasMore ? \"true\" : \"false\";\r\n\tArray.from(document.getElementsByClassName(\"page-group-nav\")).forEach(pageGroupNav => {\r\n\t\tpageGroupNav.innerHTML = \"\";\r\n\t\tif (page.page > 0) {\r\n\t\t\tconst elem = document.createElement(\"a\");\r\n\t\t\telem.href = `?page=${page.page - 1}`;\r\n\t\t\telem.style.float = \"left\";\r\n\t\t\telem.innerText = localeStr(\"prevPage\");\r\n\t\t\tpageGroupNav.appendChild(elem);\r\n\t\t}\r\n\t\tif (page.hasMore) {\r\n\t\t\tconst elem = document.createElement(\"a\");\r\n\t\t\telem.href = `?page=${page.page + 1}`;\r\n\t\t\telem.style.float = \"right\";\r\n\t\t\telem.innerText = localeStr(\"nextPage\");\r\n\t\t\tpageGroupNav.appendChild(elem);\r\n\t\t}\r\n\t});\r\n\tif (page.slug !== \"\") {\r\n\t\tconst url = page.lang !== null ? `/${page.lang}/` : \"/\";\r\n\t\thomeLink.innerHTML = `${localeStr(\"home\")}`;\r\n\t} else {\r\n\t\thomeLink.innerHTML = \"\";\r\n\t}\r\n\trenderPagePublicationDate();\r\n\tif (window.hljs !== undefined) {\r\n\t\twindow.hljs.highlightAll();\r\n\t}\r\n\trenderOEmbed();\r\n}\r\n\r\nexport function makePageUrl(page: Page | PageGroup | ChildPage, hash?: string) {\r\n\tlet url = page.lang !== null ? `/${page.lang}/${page.slug}` : `/${page.slug}`;\r\n\tif (\"page\" in page && page.page > 0) {\r\n\t\turl += `?page=${page.page}`;\r\n\t}\r\n\tif (hash !== undefined) {\r\n\t\turl += hash;\r\n\t}\r\n\treturn url;\r\n}\r\n\r\nexport function replaceCurrentPage(page: Page, hash?: string) {\r\n\thistory.replaceState(\r\n\t\t\"navigation\" in window ? null : page,\r\n\t\tpage.title,\r\n\t\tmakePageUrl(page, hash)\r\n\t);\r\n\trenderPage(page);\r\n}\r\n\r\nexport function pushCurrentPage(page: Page, hash?: string) {\r\n\thistory.pushState(\r\n\t\t\"navigation\" in window ? null : page,\r\n\t\tpage.title,\r\n\t\tmakePageUrl(page, hash)\r\n\t);\r\n\trenderPage(page);\r\n}\r\n\r\nexport function renderPagePublicationDate() {\r\n\tif (!currentPageIsGroup()) {\r\n\t\tconst publishedAt = currentPagePublishedAt();\r\n\t\tif (publishedAt !== null && currentPageShowPublishedAt()) {\r\n\t\t\tconst date = new Date(publishedAt);\r\n\t\t\tpagePublicationDate.innerText = date.toLocaleString();\r\n\t\t} else {\r\n\t\t\tpagePublicationDate.innerText = \"\";\r\n\t\t}\r\n\t} else {\r\n\t\tpagePublicationDate.innerText = \"\";\r\n\t\tArray.from(pageChildren.children).forEach(childPageElem => {\r\n\t\t\tconst childPageHtmlElem = childPageElem as HTMLElement;\r\n\t\t\tArray.from(childPageHtmlElem.getElementsByClassName(\"child-page-publication-date\")).forEach(\r\n\t\t\t\tdateElem => {\r\n\t\t\t\t\tconst dateHtmlElem = dateElem as HTMLElement;\r\n\t\t\t\t\tif (\r\n\t\t\t\t\t\tchildPageHtmlElem.dataset.showPublishedAt !== \"false\" &&\r\n\t\t\t\t\t\tchildPageHtmlElem.dataset.publishedAt !== \"\"\r\n\t\t\t\t\t) {\r\n\t\t\t\t\t\tconst date = new Date(childPageHtmlElem.dataset.publishedAt!);\r\n\t\t\t\t\t\tdateHtmlElem.innerText = date.toLocaleString();\r\n\t\t\t\t\t} else {\r\n\t\t\t\t\t\tdateHtmlElem.innerText = \"\";\r\n\t\t\t\t\t}\r\n\t\t\t\t}\r\n\t\t\t);\r\n\t\t});\r\n\t}\r\n}\r\n\r\nconst localesLoadCallbacks: (() => void)[] = [];\r\nif (window.APP_LOCALES === undefined) {\r\n\twindow.APP_LOCALES = {};\r\n\tfetch((document.getElementById(\"locales-json\") as HTMLInputElement).value)\r\n\t\t.then(response => {\r\n\t\t\tif (response.status < 200 || response.status >= 299) {\r\n\t\t\t\tconsole.error(\"Unable to load locales\", response);\r\n\t\t\t\treturn;\r\n\t\t\t}\r\n\t\t\treturn response.json();\r\n\t\t})\r\n\t\t.then(response => {\r\n\t\t\twindow.APP_LOCALES = response;\r\n\t\t\tlocalesLoadCallbacks.splice(0, localesLoadCallbacks.length).forEach(callback => callback());\r\n\t\t});\r\n}\r\n\r\nexport function waitLocalesLoaded(callback: () => void) {\r\n\tif (Object.keys(window.APP_LOCALES!).length === 0) {\r\n\t\tlocalesLoadCallbacks.push(callback);\r\n\t} else {\r\n\t\tcallback();\r\n\t}\r\n}\r\n\r\nexport function localeStr(key: string): string {\r\n\tif (window.APP_LOCALES !== undefined) {\r\n\t\t// TODO: Use user lang if page lang is missing\r\n\t\tconst localeCode = currentPageLang() || \"\";\r\n\t\t// TODO: Use fallback lang instead of English\r\n\t\tconst locale = window.APP_LOCALES[localeCode] || window.APP_LOCALES[\"en\"];\r\n\t\tconst value = locale[key];\r\n\t\tif (value !== undefined) {\r\n\t\t\treturn value;\r\n\t\t}\r\n\t}\r\n\treturn key;\r\n}\r\n","import {\r\n\tPage, PageContent,\r\n\tPageGroup,\r\n\tPageGroupUpdate,\r\n\tPageInfo,\r\n\tSignInResponse,\r\n\tThisUserInfo, UploadResponse,\r\n\tUploadsGCReport,\r\n\tUserInfo\r\n} from \"./dto\";\r\nimport {currentPageIndex} from \"./render\";\r\n\r\nexport const AUTH_TOKEN_KEY = \"AUTH_TOKEN\";\r\n\r\nexport async function fetchPage(slug: string): Promise {\r\n\tconst response = await apiCall(`/api/pages/by-slug/${slug}`);\r\n\treturn await response.json();\r\n}\r\n\r\nexport function setAuthToken(token: string) {\r\n\tlocalStorage.setItem(AUTH_TOKEN_KEY, token);\r\n}\r\n\r\nexport function clearAuthToken() {\r\n\tlocalStorage.removeItem(AUTH_TOKEN_KEY);\r\n}\r\n\r\nexport function hasAuthToken(): boolean {\r\n\treturn localStorage.getItem(AUTH_TOKEN_KEY) !== null;\r\n}\r\n\r\nexport function apiCall(url: string, init?: RequestInit) {\r\n\tconst authToken = localStorage.getItem(AUTH_TOKEN_KEY);\r\n\tconst req = init !== undefined ? Object.assign({}, init) : {};\r\n\tif (authToken !== null) {\r\n\t\treq.headers = req.headers !== undefined ? Object.assign({}, req.headers) : {};\r\n\t\t(req.headers as any)[\"Authorization\"] = `Bearer ${authToken}`;\r\n\t}\r\n\treturn fetch(url, req);\r\n}\r\n\r\nexport async function callSignIn(login: string, password: string): Promise {\r\n\tconst data = new URLSearchParams();\r\n\tdata.append(\"login\", login);\r\n\tdata.append(\"password\", password);\r\n\tconst response = await fetch(\"/api/users/sign-in\", {\r\n\t\tmethod: \"post\",\r\n\t\tbody: data\r\n\t});\r\n\tif (response.status >= 200 && response.status <= 299) {\r\n\t\treturn await response.json();\r\n\t} else if (response.status === 401) {\r\n\t\treturn \"invalid\";\r\n\t} else {\r\n\t\treturn \"unknown\";\r\n\t}\r\n}\r\n\r\nexport async function callSignOut(): Promise {\r\n\tawait fetch(\"/api/users/sign-out\", {\r\n\t\tmethod: \"post\"\r\n\t});\r\n}\r\n\r\nexport async function callCheckToken(): Promise {\r\n\tconst response = await apiCall(\"/api/users/check\", {\r\n\t\tmethod: \"post\"\r\n\t});\r\n\tif (response.status >= 200 && response.status <= 299) {\r\n\t\treturn await response.json();\r\n\t} else if (response.status >= 400 && response.status <= 499) {\r\n\t\treturn \"invalid\";\r\n\t} else {\r\n\t\treturn \"unknown\";\r\n\t}\r\n}\r\n\r\nexport async function callChangePassword(\r\n\toldPassword: string,\r\n\tnewPassword: string\r\n): Promise<\"ok\" | \"invalid-old-password\" | \"unknown\"> {\r\n\tconst data = new URLSearchParams();\r\n\tdata.append(\"oldPassword\", oldPassword);\r\n\tdata.append(\"newPassword\", newPassword);\r\n\tconst response = await apiCall(\"/api/users/password\", {\r\n\t\tmethod: \"put\",\r\n\t\tbody: data\r\n\t});\r\n\tif (response.status >= 200 && response.status <= 299) {\r\n\t\treturn \"ok\";\r\n\t} else if (response.status >= 400 && response.status <= 499) {\r\n\t\treturn \"invalid-old-password\";\r\n\t} else {\r\n\t\treturn \"unknown\";\r\n\t}\r\n}\r\n\r\nexport async function callUpdateProfile(\r\n\temail: string | null,\r\n\tdisplayName: string | null\r\n): Promise {\r\n\tconst data = new URLSearchParams();\r\n\tif (email !== null) {\r\n\t\tdata.append(\"email\", email);\r\n\t}\r\n\tif (displayName !== null) {\r\n\t\tdata.append(\"displayName\", displayName);\r\n\t}\r\n\tconst response = await apiCall(\"/api/users/profile\", {\r\n\t\tmethod: \"put\",\r\n\t\tbody: data\r\n\t});\r\n\tif (response.status >= 200 && response.status <= 299) {\r\n\t\treturn await response.json();\r\n\t} else {\r\n\t\treturn \"unknown\";\r\n\t}\r\n}\r\n\r\nfunction getPageIndexOrCurrent(index: number | null) {\r\n\treturn index === null ? currentPageIndex() : index;\r\n}\r\n\r\nexport async function callUpdatePageText(id: number, text: string, page: number | null): Promise {\r\n\tconst response = await apiCall(\r\n\t\t`/api/pages/${id}?page=${getPageIndexOrCurrent(page)}`,\r\n\t\t{\r\n\t\t\tmethod: \"put\",\r\n\t\t\tbody: text\r\n\t\t}\r\n\t);\r\n\treturn await response.json();\r\n}\r\n\r\nexport async function callUpdatePageProps(\r\n\tid: number,\r\n\ttitle: string,\r\n\tslug: string,\r\n\tlang: string | null,\r\n\tisGroup: boolean,\r\n\tshowAuthor: boolean,\r\n\tshowPublishedAt: boolean,\r\n\tpage: number | null\r\n): Promise {\r\n\tconst data = new URLSearchParams();\r\n\tdata.append(\"title\", title);\r\n\tdata.append(\"slug\", slug);\r\n\tif (lang !== null) {\r\n\t\tdata.append(\"lang\", lang);\r\n\t}\r\n\tdata.append(\"isGroup\", isGroup ? \"true\" : \"false\");\r\n\tdata.append(\"showAuthor\", showAuthor ? \"true\" : \"false\");\r\n\tdata.append(\"showPublishedAt\", showPublishedAt ? \"true\" : \"false\");\r\n\tconst response = await apiCall(`/api/pages/${id}/props?page=${getPageIndexOrCurrent(page)}`, {\r\n\t\tmethod: \"put\",\r\n\t\tbody: data\r\n\t});\r\n\tif (response.status >= 200 && response.status <= 299) {\r\n\t\treturn await response.json();\r\n\t} else if (response.status === 404) {\r\n\t\treturn \"not-found\";\r\n\t} else if (response.status === 409) {\r\n\t\treturn \"conflict\";\r\n\t} else {\r\n\t\treturn \"unknown\";\r\n\t}\r\n}\r\n\r\nexport async function callCreatePage(\r\n\ttitle: string,\r\n\tslug: string,\r\n\ttext: string,\r\n\tteaser: string | null,\r\n\tlang: string | null,\r\n\tisGroup: boolean,\r\n\tpublish: boolean,\r\n\tshowAuthor: boolean,\r\n\tshowPublishedAt: boolean\r\n): Promise {\r\n\tconst data = new URLSearchParams();\r\n\tdata.append(\"title\", title);\r\n\tdata.append(\"slug\", slug);\r\n\tdata.append(\"text\", text);\r\n\tif (teaser !== null) {\r\n\t\tdata.append(\"teaser\", teaser);\r\n\t}\r\n\tif (lang !== null) {\r\n\t\tdata.append(\"lang\", lang);\r\n\t}\r\n\tdata.append(\"isGroup\", isGroup ? \"true\" : \"false\");\r\n\tdata.append(\"publish\", publish ? \"true\" : \"false\");\r\n\tdata.append(\"showAuthor\", showAuthor ? \"true\" : \"false\");\r\n\tdata.append(\"showPublishedAt\", showPublishedAt ? \"true\" : \"false\");\r\n\tconst response = await apiCall(`/api/pages`, {\r\n\t\tmethod: \"post\",\r\n\t\tbody: data\r\n\t});\r\n\tif (response.status >= 200 && response.status <= 299) {\r\n\t\treturn await response.json();\r\n\t} else if (response.status === 409) {\r\n\t\treturn \"conflict\";\r\n\t} else {\r\n\t\treturn \"unknown\";\r\n\t}\r\n}\r\n\r\nexport async function callDeletePage(id: number): Promise<\"ok\" | \"not-found\" | \"unknown\"> {\r\n\tconst response = await apiCall(`/api/pages/${id}`, {\r\n\t\tmethod: \"delete\"\r\n\t});\r\n\tif (response.status >= 200 && response.status <= 299) {\r\n\t\treturn \"ok\";\r\n\t} else if (response.status === 404) {\r\n\t\treturn \"not-found\";\r\n\t} else {\r\n\t\treturn \"unknown\";\r\n\t}\r\n}\r\n\r\nexport async function callListPages(): Promise {\r\n\tconst response = await apiCall(\"/api/pages/all\");\r\n\tif (response.status >= 200 && response.status <= 299) {\r\n\t\treturn await response.json();\r\n\t} else {\r\n\t\treturn [];\r\n\t}\r\n}\r\n\r\nexport async function callPublishPage(id: number, page: number | null): Promise {\r\n\tconst response = await apiCall(`/api/pages/${id}/publish?page=${getPageIndexOrCurrent(page)}`, {\r\n\t\tmethod: \"post\"\r\n\t});\r\n\tif (response.status >= 200 && response.status <= 299) {\r\n\t\treturn await response.json();\r\n\t} else if (response.status === 404) {\r\n\t\treturn \"not-found\";\r\n\t} else {\r\n\t\treturn \"unknown\";\r\n\t}\r\n}\r\n\r\nexport async function callUnpublishPage(id: number, page: number | null): Promise {\r\n\tconst response = await apiCall(`/api/pages/${id}/unpublish?page=${getPageIndexOrCurrent(page)}`, {\r\n\t\tmethod: \"post\"\r\n\t});\r\n\tif (response.status >= 200 && response.status <= 299) {\r\n\t\treturn await response.json();\r\n\t} else if (response.status === 404) {\r\n\t\treturn \"not-found\";\r\n\t} else {\r\n\t\treturn \"unknown\";\r\n\t}\r\n}\r\n\r\nexport async function callListGroups(): Promise {\r\n\tconst response = await apiCall(\"/api/pages/groups\");\r\n\tif (response.status >= 200 && response.status <= 299) {\r\n\t\treturn await response.json();\r\n\t} else {\r\n\t\treturn [];\r\n\t}\r\n}\r\n\r\nexport async function callSetPageGroups(\r\n\tpageId: number,\r\n\tgroups: PageGroupUpdate[],\r\n\tpage: number | null\r\n): Promise {\r\n\tconst response = await apiCall(\r\n\t\t`/api/pages/${pageId}/groups?page=${getPageIndexOrCurrent(page)}`,\r\n\t\t{\r\n\t\t\tmethod: \"put\",\r\n\t\t\theaders: {\r\n\t\t\t\t\"Content-Type\": \"application/json\"\r\n\t\t\t},\r\n\t\t\tbody: JSON.stringify(groups)\r\n\t\t}\r\n\t);\r\n\tif (response.status >= 200 && response.status <= 299) {\r\n\t\treturn await response.json();\r\n\t} else if (response.status === 404) {\r\n\t\treturn \"not-found\";\r\n\t} else {\r\n\t\treturn \"unknown\";\r\n\t}\r\n}\r\n\r\nexport async function callListUsers(): Promise {\r\n\tconst response = await apiCall(\"/api/users/all\");\r\n\tif (response.status >= 200 && response.status <= 299) {\r\n\t\treturn await response.json();\r\n\t} else if (response.status === 403) {\r\n\t\treturn \"forbidden\";\r\n\t} else {\r\n\t\treturn \"unknown\";\r\n\t}\r\n}\r\n\r\nexport async function callSetIsAdmin(id: number): Promise<\"ok\" | \"not-found\" | \"forbidden\" | \"unknown\"> {\r\n\tconst response = await apiCall(`/api/users/${id}/is-admin`, {\r\n\t\tmethod: \"post\"\r\n\t});\r\n\tif (response.status >= 200 && response.status <= 299) {\r\n\t\treturn \"ok\";\r\n\t} else if (response.status === 403) {\r\n\t\treturn \"forbidden\";\r\n\t} else if (response.status === 404) {\r\n\t\treturn \"not-found\";\r\n\t} else {\r\n\t\treturn \"unknown\";\r\n\t}\r\n}\r\n\r\nexport async function callClearIsAdmin(id: number): Promise<\"ok\" | \"not-found\" | \"forbidden\" | \"unknown\"> {\r\n\tconst response = await apiCall(`/api/users/${id}/is-admin`, {\r\n\t\tmethod: \"delete\"\r\n\t});\r\n\tif (response.status >= 200 && response.status <= 299) {\r\n\t\treturn \"ok\";\r\n\t} else if (response.status === 403) {\r\n\t\treturn \"forbidden\";\r\n\t} if (response.status === 404) {\r\n\t\treturn \"not-found\";\r\n\t} else {\r\n\t\treturn \"unknown\";\r\n\t}\r\n}\r\n\r\nexport async function callCreateUser(\r\n\tlogin: string,\r\n\tpassword: string | null,\r\n\tisAdmin: boolean\r\n): Promise {\r\n\tconst data = new URLSearchParams();\r\n\tdata.append(\"login\", login);\r\n\tif (password !== null) {\r\n\t\tdata.append(\"password\", password);\r\n\t}\r\n\tdata.append(\"isAdmin\", isAdmin ? \"true\" : \"false\");\r\n\tconst response = await apiCall(`/api/users`, {\r\n\t\tmethod: \"post\",\r\n\t\tbody: data\r\n\t});\r\n\tif (response.status >= 200 && response.status <= 299) {\r\n\t\treturn await response.json();\r\n\t} else if (response.status === 403) {\r\n\t\treturn \"forbidden\";\r\n\t} else if (response.status === 409) {\r\n\t\treturn \"conflict\";\r\n\t} else {\r\n\t\treturn \"unknown\";\r\n\t}\r\n}\r\n\r\nexport async function callDeleteUser(id: number): Promise<\"ok\" | \"forbidden\" | \"not-found\" | \"unknown\"> {\r\n\tconst response = await apiCall(`/api/users/${id}`, {\r\n\t\tmethod: \"delete\"\r\n\t});\r\n\tif (response.status >= 200 && response.status <= 299) {\r\n\t\treturn \"ok\";\r\n\t} else if (response.status === 403) {\r\n\t\treturn \"forbidden\";\r\n\t} else if (response.status === 404) {\r\n\t\treturn \"not-found\";\r\n\t} else {\r\n\t\treturn \"unknown\";\r\n\t}\r\n}\r\n\r\nexport async function callSetUserPassword(\r\n\tid: number,\r\n\tpassword: string\r\n): Promise<\"ok\" | \"forbidden\" | \"not-found\" | \"unknown\"> {\r\n\tconst response = await apiCall(`/api/users/${id}/password`, {\r\n\t\tmethod: \"put\",\r\n\t\tbody: password\r\n\t});\r\n\tif (response.status >= 200 && response.status <= 299) {\r\n\t\treturn await response.json();\r\n\t} else if (response.status === 403) {\r\n\t\treturn \"forbidden\";\r\n\t} else if (response.status === 404) {\r\n\t\treturn \"not-found\";\r\n\t} else {\r\n\t\treturn \"unknown\";\r\n\t}\r\n}\r\n\r\nexport async function callClearUserPassword(id: number): Promise<\"ok\" | \"forbidden\" | \"not-found\" | \"unknown\"> {\r\n\tconst response = await apiCall(`/api/users/${id}/password`, {\r\n\t\tmethod: \"delete\"\r\n\t});\r\n\tif (response.status >= 200 && response.status <= 299) {\r\n\t\treturn await response.json();\r\n\t} else if (response.status === 403) {\r\n\t\treturn \"forbidden\";\r\n\t} else if (response.status === 404) {\r\n\t\treturn \"not-found\";\r\n\t} else {\r\n\t\treturn \"unknown\";\r\n\t}\r\n}\r\n\r\nexport async function callSetUserLogin(\r\n\tid: number,\r\n\tlogin: string\r\n): Promise<\"ok\" | \"forbidden\" | \"conflict\" | \"not-found\" | \"unknown\"> {\r\n\tconst response = await apiCall(`/api/users/${id}/login`, {\r\n\t\tmethod: \"put\",\r\n\t\tbody: login\r\n\t});\r\n\tif (response.status >= 200 && response.status <= 299) {\r\n\t\treturn await response.json();\r\n\t} else if (response.status === 403) {\r\n\t\treturn \"forbidden\";\r\n\t} else if (response.status === 409) {\r\n\t\treturn \"conflict\";\r\n\t} else if (response.status === 404) {\r\n\t\treturn \"not-found\";\r\n\t} else {\r\n\t\treturn \"unknown\";\r\n\t}\r\n}\r\n\r\nexport async function callListPageGroups(id: number): Promise {\r\n\tconst response = await apiCall(`/api/pages/${id}/groups`);\r\n\tif (response.status >= 200 && response.status <= 299) {\r\n\t\treturn await response.json();\r\n\t} else if (response.status === 404) {\r\n\t\treturn \"not-found\";\r\n\t} else {\r\n\t\treturn \"unknown\";\r\n\t}\r\n}\r\n\r\nexport async function callGetContent(id: number): Promise {\r\n\tconst response = await apiCall(`/api/pages/${id}/content`);\r\n\tif (response.status >= 200 && response.status <= 299) {\r\n\t\treturn await response.json();\r\n\t} else if (response.status === 404) {\r\n\t\treturn \"not-found\";\r\n\t} else {\r\n\t\treturn \"unknown\";\r\n\t}\r\n}\r\n\r\nexport async function callSetTeaser(id: number, teaser: string): Promise {\r\n\tawait apiCall(`/api/pages/${id}/teaser`, {\r\n\t\tmethod: \"put\",\r\n\t\tbody: teaser\r\n\t});\r\n}\r\n\r\nexport async function callDeleteTeaser(id: number): Promise {\r\n\tawait apiCall(`/api/pages/${id}/teaser`, {\r\n\t\tmethod: \"delete\"\r\n\t});\r\n}\r\n\r\nexport async function callUploadFile(file: Blob, filename?: string): Promise {\r\n\tconst data = new FormData();\r\n\tdata.append(\"upload\", file, filename);\r\n\tconst response = await apiCall(`/api/uploads`, {\r\n\t\tmethod: \"post\",\r\n\t\tbody: data\r\n\t});\r\n\tif (response.status >= 200 && response.status <= 299) {\r\n\t\treturn await response.json();\r\n\t} else {\r\n\t\treturn \"unknown\";\r\n\t}\r\n}\r\n\r\nexport async function callUploadsGC(): Promise {\r\n\tconst response = await apiCall(`/api/uploads/gc`, {\r\n\t\tmethod: \"post\"\r\n\t});\r\n\tif (response.status >= 200 && response.status <= 299) {\r\n\t\treturn await response.json();\r\n\t} else {\r\n\t\treturn \"unknown\";\r\n\t}\r\n}\r\n","import {fetchPage} from \"./api\";\r\nimport {\r\n\tcurrentPage,\r\n\tmakePageUrl,\r\n\tpushCurrentPage,\r\n\trenderPage,\r\n\trenderPagePublicationDate\r\n} from \"./render\";\r\n\r\nfunction initState() {\r\n\tconst page = currentPage();\r\n\tconst hash = window.location.hash;\r\n\thistory.replaceState(\r\n\t\t\"navigation\" in window ? null : page,\r\n\t\tpage.title,\r\n\t\tmakePageUrl(page, hash !== \"\" ? hash : undefined)\r\n\t);\r\n}\r\ninitState();\r\n\r\nwindow.CHECK_CONFIRM_NAVIGATION = [];\r\n\r\nwindow.addEventListener(\"beforeunload\", evt => {\r\n\tfor (let callback of window.CHECK_CONFIRM_NAVIGATION) {\r\n\t\tif (callback()) {\r\n\t\t\tevt.returnValue = true;\r\n\t\t\tbreak;\r\n\t\t}\r\n\t}\r\n});\r\n\r\nfunction confirmNavigation(): boolean {\r\n\tfor (let callback of window.CHECK_CONFIRM_NAVIGATION) {\r\n\t\tif (callback()) {\r\n\t\t\tif (!confirm(\r\n\t\t\t\t\"Are you sure that you want to leave current page? \" +\r\n\t\t\t\t\"You might lose unsaved data.\"\r\n\t\t\t)) {\r\n\t\t\t\treturn false;\r\n\t\t\t}\r\n\t\t\tbreak;\r\n\t\t}\r\n\t}\r\n\treturn true;\r\n}\r\n\r\nfunction isPageSlug(url: string): boolean {\r\n\treturn !url.startsWith(\"api/\") && !url.startsWith(\"uploads/\");\r\n}\r\n\r\nif (\"navigation\" in window) {\r\n\tconsole.log(\"Using Navigation API\");\r\n\t(window.navigation as any).addEventListener(\"navigate\", (evt: Event) => {\r\n\t\tif (!(evt as any).userInitiated) {\r\n\t\t\treturn;\r\n\t\t}\r\n\t\tif (evt.cancelable) {\r\n\t\t\tif (!confirmNavigation()) {\r\n\t\t\t\tevt.preventDefault();\r\n\t\t\t\treturn;\r\n\t\t\t}\r\n\t\t}\r\n\t\tif ((evt as any).canIntercept) {\r\n\t\t\tconst url = new URL((evt as any).destination.url);\r\n\t\t\tif (!isPageSlug(url.pathname.substring(1))) {\r\n\t\t\t\treturn;\r\n\t\t\t}\r\n\t\t\tif (url.pathname !== location.pathname || url.search !== location.search) {\r\n\t\t\t\t(evt as any).intercept({\r\n\t\t\t\t\thandler: async () => {\r\n\t\t\t\t\t\tconst page = await fetchPage(url.pathname.substring(1) + url.search);\r\n\t\t\t\t\t\trenderPage(page);\r\n\t\t\t\t\t\tif (\r\n\t\t\t\t\t\t\t(evt as any).navigationType === \"push\" &&\r\n\t\t\t\t\t\t\t(url.hash.length === 0 || document.getElementById(url.hash.substring(1)) === null)\r\n\t\t\t\t\t\t) {\r\n\t\t\t\t\t\t\twindow.scrollTo(0, 0);\r\n\t\t\t\t\t\t} else {\r\n\t\t\t\t\t\t\t(evt as any).scroll();\r\n\t\t\t\t\t\t}\r\n\t\t\t\t\t\tif (makePageUrl(page) !== url.pathname + url.search) {\r\n\t\t\t\t\t\t\thistory.replaceState(\r\n\t\t\t\t\t\t\t\tnull,\r\n\t\t\t\t\t\t\t\tpage.title,\r\n\t\t\t\t\t\t\t\tmakePageUrl(page, url.hash !== \"\" ? url.hash : undefined)\r\n\t\t\t\t\t\t\t);\r\n\t\t\t\t\t\t}\r\n\t\t\t\t\t}\r\n\t\t\t\t});\r\n\t\t\t}\r\n\t\t}\r\n\t});\r\n} else {\r\n\tconsole.log(\"Using History API\");\r\n\tinitState();\r\n\tdocument.addEventListener(\"click\", evt => {\r\n\t\tconst origin = (evt.target as HTMLElement).closest(\"a\");\r\n\t\tif (origin) {\r\n\t\t\tif (origin.target === \"\" || origin.target === \"_self\") {\r\n\t\t\t\tconst href = origin.href;\r\n\t\t\t\tif (href.startsWith(window.location.origin)) {\r\n\t\t\t\t\tconst urlAndHash = href.substring(window.location.origin.length + 1);\r\n\t\t\t\t\tif (!isPageSlug(urlAndHash)) {\r\n\t\t\t\t\t\treturn;\r\n\t\t\t\t\t}\r\n\t\t\t\t\tconst hashOffset = urlAndHash.indexOf(\"#\");\r\n\t\t\t\t\tconst url = hashOffset < 0 ? urlAndHash : urlAndHash.substring(0, hashOffset);\r\n\t\t\t\t\tconst hash = hashOffset < 0 ? \"\" : urlAndHash.substring(hashOffset);\r\n\t\t\t\t\tif (url !== window.location.pathname.substring(1)) {\r\n\t\t\t\t\t\tevt.preventDefault();\r\n\t\t\t\t\t\tif (confirmNavigation()) {\r\n\t\t\t\t\t\t\tfetchPage(url).then(response => {\r\n\t\t\t\t\t\t\t\tpushCurrentPage(response, hash !== \"\" ? hash : undefined);\r\n\t\t\t\t\t\t\t\tif (hash !== \"\") {\r\n\t\t\t\t\t\t\t\t\tsetTimeout(\r\n\t\t\t\t\t\t\t\t\t\t() => {\r\n\t\t\t\t\t\t\t\t\t\t\tconst elem = document.getElementById(hash.substring(1));\r\n\t\t\t\t\t\t\t\t\t\t\tif (elem !== null) {\r\n\t\t\t\t\t\t\t\t\t\t\t\telem.scrollIntoView();\r\n\t\t\t\t\t\t\t\t\t\t\t} else {\r\n\t\t\t\t\t\t\t\t\t\t\t\twindow.scrollTo(0, 0);\r\n\t\t\t\t\t\t\t\t\t\t\t}\r\n\t\t\t\t\t\t\t\t\t\t},\r\n\t\t\t\t\t\t\t\t\t\t0\r\n\t\t\t\t\t\t\t\t\t);\r\n\t\t\t\t\t\t\t\t} else {\r\n\t\t\t\t\t\t\t\t\twindow.scrollTo(0, 0);\r\n\t\t\t\t\t\t\t\t}\r\n\t\t\t\t\t\t\t});\r\n\t\t\t\t\t\t}\r\n\t\t\t\t\t} else if (hash !== \"\") {\r\n\t\t\t\t\t\tsetTimeout(initState, 1 /* Firefox work-around */);\r\n\t\t\t\t\t} else {\r\n\t\t\t\t\t\tevt.preventDefault();\r\n\t\t\t\t\t}\r\n\t\t\t\t}\r\n\t\t\t}\r\n\t\t}\r\n\t});\r\n\twindow.addEventListener(\"popstate\", evt => {\r\n\t\tif (evt.state !== null) {\r\n\t\t\trenderPage(evt.state);\r\n\t\t}\r\n\t});\r\n}\r\n\r\nrenderPagePublicationDate();\r\n","/** @returns {void} */\nexport function noop() {}\n\nexport const identity = (x) => x;\n\n/**\n * @template T\n * @template S\n * @param {T} tar\n * @param {S} src\n * @returns {T & S}\n */\nexport function assign(tar, src) {\n\t// @ts-ignore\n\tfor (const k in src) tar[k] = src[k];\n\treturn /** @type {T & S} */ (tar);\n}\n\n// Adapted from https://github.com/then/is-promise/blob/master/index.js\n// Distributed under MIT License https://github.com/then/is-promise/blob/master/LICENSE\n/**\n * @param {any} value\n * @returns {value is PromiseLike}\n */\nexport function is_promise(value) {\n\treturn (\n\t\t!!value &&\n\t\t(typeof value === 'object' || typeof value === 'function') &&\n\t\ttypeof (/** @type {any} */ (value).then) === 'function'\n\t);\n}\n\n/** @returns {void} */\nexport function add_location(element, file, line, column, char) {\n\telement.__svelte_meta = {\n\t\tloc: { file, line, column, char }\n\t};\n}\n\nexport function run(fn) {\n\treturn fn();\n}\n\nexport function blank_object() {\n\treturn Object.create(null);\n}\n\n/**\n * @param {Function[]} fns\n * @returns {void}\n */\nexport function run_all(fns) {\n\tfns.forEach(run);\n}\n\n/**\n * @param {any} thing\n * @returns {thing is Function}\n */\nexport function is_function(thing) {\n\treturn typeof thing === 'function';\n}\n\n/** @returns {boolean} */\nexport function safe_not_equal(a, b) {\n\treturn a != a ? b == b : a !== b || (a && typeof a === 'object') || typeof a === 'function';\n}\n\nlet src_url_equal_anchor;\n\n/**\n * @param {string} element_src\n * @param {string} url\n * @returns {boolean}\n */\nexport function src_url_equal(element_src, url) {\n\tif (element_src === url) return true;\n\tif (!src_url_equal_anchor) {\n\t\tsrc_url_equal_anchor = document.createElement('a');\n\t}\n\t// This is actually faster than doing URL(..).href\n\tsrc_url_equal_anchor.href = url;\n\treturn element_src === src_url_equal_anchor.href;\n}\n\n/** @param {string} srcset */\nfunction split_srcset(srcset) {\n\treturn srcset.split(',').map((src) => src.trim().split(' ').filter(Boolean));\n}\n\n/**\n * @param {HTMLSourceElement | HTMLImageElement} element_srcset\n * @param {string | undefined | null} srcset\n * @returns {boolean}\n */\nexport function srcset_url_equal(element_srcset, srcset) {\n\tconst element_urls = split_srcset(element_srcset.srcset);\n\tconst urls = split_srcset(srcset || '');\n\n\treturn (\n\t\turls.length === element_urls.length &&\n\t\turls.every(\n\t\t\t([url, width], i) =>\n\t\t\t\twidth === element_urls[i][1] &&\n\t\t\t\t// We need to test both ways because Vite will create an a full URL with\n\t\t\t\t// `new URL(asset, import.meta.url).href` for the client when `base: './'`, and the\n\t\t\t\t// relative URLs inside srcset are not automatically resolved to absolute URLs by\n\t\t\t\t// browsers (in contrast to img.src). This means both SSR and DOM code could\n\t\t\t\t// contain relative or absolute URLs.\n\t\t\t\t(src_url_equal(element_urls[i][0], url) || src_url_equal(url, element_urls[i][0]))\n\t\t)\n\t);\n}\n\n/** @returns {boolean} */\nexport function not_equal(a, b) {\n\treturn a != a ? b == b : a !== b;\n}\n\n/** @returns {boolean} */\nexport function is_empty(obj) {\n\treturn Object.keys(obj).length === 0;\n}\n\n/** @returns {void} */\nexport function validate_store(store, name) {\n\tif (store != null && typeof store.subscribe !== 'function') {\n\t\tthrow new Error(`'${name}' is not a store with a 'subscribe' method`);\n\t}\n}\n\nexport function subscribe(store, ...callbacks) {\n\tif (store == null) {\n\t\tfor (const callback of callbacks) {\n\t\t\tcallback(undefined);\n\t\t}\n\t\treturn noop;\n\t}\n\tconst unsub = store.subscribe(...callbacks);\n\treturn unsub.unsubscribe ? () => unsub.unsubscribe() : unsub;\n}\n\n/**\n * Get the current value from a store by subscribing and immediately unsubscribing.\n *\n * https://svelte.dev/docs/svelte-store#get\n * @template T\n * @param {import('../store/public.js').Readable} store\n * @returns {T}\n */\nexport function get_store_value(store) {\n\tlet value;\n\tsubscribe(store, (_) => (value = _))();\n\treturn value;\n}\n\n/** @returns {void} */\nexport function component_subscribe(component, store, callback) {\n\tcomponent.$$.on_destroy.push(subscribe(store, callback));\n}\n\nexport function create_slot(definition, ctx, $$scope, fn) {\n\tif (definition) {\n\t\tconst slot_ctx = get_slot_context(definition, ctx, $$scope, fn);\n\t\treturn definition[0](slot_ctx);\n\t}\n}\n\nfunction get_slot_context(definition, ctx, $$scope, fn) {\n\treturn definition[1] && fn ? assign($$scope.ctx.slice(), definition[1](fn(ctx))) : $$scope.ctx;\n}\n\nexport function get_slot_changes(definition, $$scope, dirty, fn) {\n\tif (definition[2] && fn) {\n\t\tconst lets = definition[2](fn(dirty));\n\t\tif ($$scope.dirty === undefined) {\n\t\t\treturn lets;\n\t\t}\n\t\tif (typeof lets === 'object') {\n\t\t\tconst merged = [];\n\t\t\tconst len = Math.max($$scope.dirty.length, lets.length);\n\t\t\tfor (let i = 0; i < len; i += 1) {\n\t\t\t\tmerged[i] = $$scope.dirty[i] | lets[i];\n\t\t\t}\n\t\t\treturn merged;\n\t\t}\n\t\treturn $$scope.dirty | lets;\n\t}\n\treturn $$scope.dirty;\n}\n\n/** @returns {void} */\nexport function update_slot_base(\n\tslot,\n\tslot_definition,\n\tctx,\n\t$$scope,\n\tslot_changes,\n\tget_slot_context_fn\n) {\n\tif (slot_changes) {\n\t\tconst slot_context = get_slot_context(slot_definition, ctx, $$scope, get_slot_context_fn);\n\t\tslot.p(slot_context, slot_changes);\n\t}\n}\n\n/** @returns {void} */\nexport function update_slot(\n\tslot,\n\tslot_definition,\n\tctx,\n\t$$scope,\n\tdirty,\n\tget_slot_changes_fn,\n\tget_slot_context_fn\n) {\n\tconst slot_changes = get_slot_changes(slot_definition, $$scope, dirty, get_slot_changes_fn);\n\tupdate_slot_base(slot, slot_definition, ctx, $$scope, slot_changes, get_slot_context_fn);\n}\n\n/** @returns {any[] | -1} */\nexport function get_all_dirty_from_scope($$scope) {\n\tif ($$scope.ctx.length > 32) {\n\t\tconst dirty = [];\n\t\tconst length = $$scope.ctx.length / 32;\n\t\tfor (let i = 0; i < length; i++) {\n\t\t\tdirty[i] = -1;\n\t\t}\n\t\treturn dirty;\n\t}\n\treturn -1;\n}\n\n/** @returns {{}} */\nexport function exclude_internal_props(props) {\n\tconst result = {};\n\tfor (const k in props) if (k[0] !== '$') result[k] = props[k];\n\treturn result;\n}\n\n/** @returns {{}} */\nexport function compute_rest_props(props, keys) {\n\tconst rest = {};\n\tkeys = new Set(keys);\n\tfor (const k in props) if (!keys.has(k) && k[0] !== '$') rest[k] = props[k];\n\treturn rest;\n}\n\n/** @returns {{}} */\nexport function compute_slots(slots) {\n\tconst result = {};\n\tfor (const key in slots) {\n\t\tresult[key] = true;\n\t}\n\treturn result;\n}\n\n/** @returns {(this: any, ...args: any[]) => void} */\nexport function once(fn) {\n\tlet ran = false;\n\treturn function (...args) {\n\t\tif (ran) return;\n\t\tran = true;\n\t\tfn.call(this, ...args);\n\t};\n}\n\nexport function null_to_empty(value) {\n\treturn value == null ? '' : value;\n}\n\nexport function set_store_value(store, ret, value) {\n\tstore.set(value);\n\treturn ret;\n}\n\nexport const has_prop = (obj, prop) => Object.prototype.hasOwnProperty.call(obj, prop);\n\nexport function action_destroyer(action_result) {\n\treturn action_result && is_function(action_result.destroy) ? action_result.destroy : noop;\n}\n\n/** @param {number | string} value\n * @returns {[number, string]}\n */\nexport function split_css_unit(value) {\n\tconst split = typeof value === 'string' && value.match(/^\\s*(-?[\\d.]+)([^\\s]*)\\s*$/);\n\treturn split ? [parseFloat(split[1]), split[2] || 'px'] : [/** @type {number} */ (value), 'px'];\n}\n\nexport const contenteditable_truthy_values = ['', true, 1, 'true', 'contenteditable'];\n","import { contenteditable_truthy_values, has_prop } from './utils.js';\n\nimport { ResizeObserverSingleton } from './ResizeObserverSingleton.js';\n\n// Track which nodes are claimed during hydration. Unclaimed nodes can then be removed from the DOM\n// at the end of hydration without touching the remaining nodes.\nlet is_hydrating = false;\n\n/**\n * @returns {void}\n */\nexport function start_hydrating() {\n\tis_hydrating = true;\n}\n\n/**\n * @returns {void}\n */\nexport function end_hydrating() {\n\tis_hydrating = false;\n}\n\n/**\n * @param {number} low\n * @param {number} high\n * @param {(index: number) => number} key\n * @param {number} value\n * @returns {number}\n */\nfunction upper_bound(low, high, key, value) {\n\t// Return first index of value larger than input value in the range [low, high)\n\twhile (low < high) {\n\t\tconst mid = low + ((high - low) >> 1);\n\t\tif (key(mid) <= value) {\n\t\t\tlow = mid + 1;\n\t\t} else {\n\t\t\thigh = mid;\n\t\t}\n\t}\n\treturn low;\n}\n\n/**\n * @param {NodeEx} target\n * @returns {void}\n */\nfunction init_hydrate(target) {\n\tif (target.hydrate_init) return;\n\ttarget.hydrate_init = true;\n\t// We know that all children have claim_order values since the unclaimed have been detached if target is not \n\n\tlet children = /** @type {ArrayLike} */ (target.childNodes);\n\t// If target is , there may be children without claim_order\n\tif (target.nodeName === 'HEAD') {\n\t\tconst my_children = [];\n\t\tfor (let i = 0; i < children.length; i++) {\n\t\t\tconst node = children[i];\n\t\t\tif (node.claim_order !== undefined) {\n\t\t\t\tmy_children.push(node);\n\t\t\t}\n\t\t}\n\t\tchildren = my_children;\n\t}\n\t/*\n\t * Reorder claimed children optimally.\n\t * We can reorder claimed children optimally by finding the longest subsequence of\n\t * nodes that are already claimed in order and only moving the rest. The longest\n\t * subsequence of nodes that are claimed in order can be found by\n\t * computing the longest increasing subsequence of .claim_order values.\n\t *\n\t * This algorithm is optimal in generating the least amount of reorder operations\n\t * possible.\n\t *\n\t * Proof:\n\t * We know that, given a set of reordering operations, the nodes that do not move\n\t * always form an increasing subsequence, since they do not move among each other\n\t * meaning that they must be already ordered among each other. Thus, the maximal\n\t * set of nodes that do not move form a longest increasing subsequence.\n\t */\n\t// Compute longest increasing subsequence\n\t// m: subsequence length j => index k of smallest value that ends an increasing subsequence of length j\n\tconst m = new Int32Array(children.length + 1);\n\t// Predecessor indices + 1\n\tconst p = new Int32Array(children.length);\n\tm[0] = -1;\n\tlet longest = 0;\n\tfor (let i = 0; i < children.length; i++) {\n\t\tconst current = children[i].claim_order;\n\t\t// Find the largest subsequence length such that it ends in a value less than our current value\n\t\t// upper_bound returns first greater value, so we subtract one\n\t\t// with fast path for when we are on the current longest subsequence\n\t\tconst seq_len =\n\t\t\t(longest > 0 && children[m[longest]].claim_order <= current\n\t\t\t\t? longest + 1\n\t\t\t\t: upper_bound(1, longest, (idx) => children[m[idx]].claim_order, current)) - 1;\n\t\tp[i] = m[seq_len] + 1;\n\t\tconst new_len = seq_len + 1;\n\t\t// We can guarantee that current is the smallest value. Otherwise, we would have generated a longer sequence.\n\t\tm[new_len] = i;\n\t\tlongest = Math.max(new_len, longest);\n\t}\n\t// The longest increasing subsequence of nodes (initially reversed)\n\n\t/**\n\t * @type {NodeEx2[]}\n\t */\n\tconst lis = [];\n\t// The rest of the nodes, nodes that will be moved\n\n\t/**\n\t * @type {NodeEx2[]}\n\t */\n\tconst to_move = [];\n\tlet last = children.length - 1;\n\tfor (let cur = m[longest] + 1; cur != 0; cur = p[cur - 1]) {\n\t\tlis.push(children[cur - 1]);\n\t\tfor (; last >= cur; last--) {\n\t\t\tto_move.push(children[last]);\n\t\t}\n\t\tlast--;\n\t}\n\tfor (; last >= 0; last--) {\n\t\tto_move.push(children[last]);\n\t}\n\tlis.reverse();\n\t// We sort the nodes being moved to guarantee that their insertion order matches the claim order\n\tto_move.sort((a, b) => a.claim_order - b.claim_order);\n\t// Finally, we move the nodes\n\tfor (let i = 0, j = 0; i < to_move.length; i++) {\n\t\twhile (j < lis.length && to_move[i].claim_order >= lis[j].claim_order) {\n\t\t\tj++;\n\t\t}\n\t\tconst anchor = j < lis.length ? lis[j] : null;\n\t\ttarget.insertBefore(to_move[i], anchor);\n\t}\n}\n\n/**\n * @param {Node} target\n * @param {Node} node\n * @returns {void}\n */\nexport function append(target, node) {\n\ttarget.appendChild(node);\n}\n\n/**\n * @param {Node} target\n * @param {string} style_sheet_id\n * @param {string} styles\n * @returns {void}\n */\nexport function append_styles(target, style_sheet_id, styles) {\n\tconst append_styles_to = get_root_for_style(target);\n\tif (!append_styles_to.getElementById(style_sheet_id)) {\n\t\tconst style = element('style');\n\t\tstyle.id = style_sheet_id;\n\t\tstyle.textContent = styles;\n\t\tappend_stylesheet(append_styles_to, style);\n\t}\n}\n\n/**\n * @param {Node} node\n * @returns {ShadowRoot | Document}\n */\nexport function get_root_for_style(node) {\n\tif (!node) return document;\n\tconst root = node.getRootNode ? node.getRootNode() : node.ownerDocument;\n\tif (root && /** @type {ShadowRoot} */ (root).host) {\n\t\treturn /** @type {ShadowRoot} */ (root);\n\t}\n\treturn node.ownerDocument;\n}\n\n/**\n * @param {Node} node\n * @returns {CSSStyleSheet}\n */\nexport function append_empty_stylesheet(node) {\n\tconst style_element = element('style');\n\t// For transitions to work without 'style-src: unsafe-inline' Content Security Policy,\n\t// these empty tags need to be allowed with a hash as a workaround until we move to the Web Animations API.\n\t// Using the hash for the empty string (for an empty tag) works in all browsers except Safari.\n\t// So as a workaround for the workaround, when we append empty style tags we set their content to /* empty */.\n\t// The hash 'sha256-9OlNO0DNEeaVzHL4RZwCLsBHA8WBQ8toBp/4F5XV2nc=' will then work even in Safari.\n\tstyle_element.textContent = '/* empty */';\n\tappend_stylesheet(get_root_for_style(node), style_element);\n\treturn style_element.sheet;\n}\n\n/**\n * @param {ShadowRoot | Document} node\n * @param {HTMLStyleElement} style\n * @returns {CSSStyleSheet}\n */\nfunction append_stylesheet(node, style) {\n\tappend(/** @type {Document} */ (node).head || node, style);\n\treturn style.sheet;\n}\n\n/**\n * @param {NodeEx} target\n * @param {NodeEx} node\n * @returns {void}\n */\nexport function append_hydration(target, node) {\n\tif (is_hydrating) {\n\t\tinit_hydrate(target);\n\t\tif (\n\t\t\ttarget.actual_end_child === undefined ||\n\t\t\t(target.actual_end_child !== null && target.actual_end_child.parentNode !== target)\n\t\t) {\n\t\t\ttarget.actual_end_child = target.firstChild;\n\t\t}\n\t\t// Skip nodes of undefined ordering\n\t\twhile (target.actual_end_child !== null && target.actual_end_child.claim_order === undefined) {\n\t\t\ttarget.actual_end_child = target.actual_end_child.nextSibling;\n\t\t}\n\t\tif (node !== target.actual_end_child) {\n\t\t\t// We only insert if the ordering of this node should be modified or the parent node is not target\n\t\t\tif (node.claim_order !== undefined || node.parentNode !== target) {\n\t\t\t\ttarget.insertBefore(node, target.actual_end_child);\n\t\t\t}\n\t\t} else {\n\t\t\ttarget.actual_end_child = node.nextSibling;\n\t\t}\n\t} else if (node.parentNode !== target || node.nextSibling !== null) {\n\t\ttarget.appendChild(node);\n\t}\n}\n\n/**\n * @param {Node} target\n * @param {Node} node\n * @param {Node} [anchor]\n * @returns {void}\n */\nexport function insert(target, node, anchor) {\n\ttarget.insertBefore(node, anchor || null);\n}\n\n/**\n * @param {NodeEx} target\n * @param {NodeEx} node\n * @param {NodeEx} [anchor]\n * @returns {void}\n */\nexport function insert_hydration(target, node, anchor) {\n\tif (is_hydrating && !anchor) {\n\t\tappend_hydration(target, node);\n\t} else if (node.parentNode !== target || node.nextSibling != anchor) {\n\t\ttarget.insertBefore(node, anchor || null);\n\t}\n}\n\n/**\n * @param {Node} node\n * @returns {void}\n */\nexport function detach(node) {\n\tif (node.parentNode) {\n\t\tnode.parentNode.removeChild(node);\n\t}\n}\n\n/**\n * @returns {void} */\nexport function destroy_each(iterations, detaching) {\n\tfor (let i = 0; i < iterations.length; i += 1) {\n\t\tif (iterations[i]) iterations[i].d(detaching);\n\t}\n}\n\n/**\n * @template {keyof HTMLElementTagNameMap} K\n * @param {K} name\n * @returns {HTMLElementTagNameMap[K]}\n */\nexport function element(name) {\n\treturn document.createElement(name);\n}\n\n/**\n * @template {keyof HTMLElementTagNameMap} K\n * @param {K} name\n * @param {string} is\n * @returns {HTMLElementTagNameMap[K]}\n */\nexport function element_is(name, is) {\n\treturn document.createElement(name, { is });\n}\n\n/**\n * @template T\n * @template {keyof T} K\n * @param {T} obj\n * @param {K[]} exclude\n * @returns {Pick>}\n */\nexport function object_without_properties(obj, exclude) {\n\tconst target = /** @type {Pick>} */ ({});\n\tfor (const k in obj) {\n\t\tif (\n\t\t\thas_prop(obj, k) &&\n\t\t\t// @ts-ignore\n\t\t\texclude.indexOf(k) === -1\n\t\t) {\n\t\t\t// @ts-ignore\n\t\t\ttarget[k] = obj[k];\n\t\t}\n\t}\n\treturn target;\n}\n\n/**\n * @template {keyof SVGElementTagNameMap} K\n * @param {K} name\n * @returns {SVGElement}\n */\nexport function svg_element(name) {\n\treturn document.createElementNS('http://www.w3.org/2000/svg', name);\n}\n\n/**\n * @param {string} data\n * @returns {Text}\n */\nexport function text(data) {\n\treturn document.createTextNode(data);\n}\n\n/**\n * @returns {Text} */\nexport function space() {\n\treturn text(' ');\n}\n\n/**\n * @returns {Text} */\nexport function empty() {\n\treturn text('');\n}\n\n/**\n * @param {string} content\n * @returns {Comment}\n */\nexport function comment(content) {\n\treturn document.createComment(content);\n}\n\n/**\n * @param {EventTarget} node\n * @param {string} event\n * @param {EventListenerOrEventListenerObject} handler\n * @param {boolean | AddEventListenerOptions | EventListenerOptions} [options]\n * @returns {() => void}\n */\nexport function listen(node, event, handler, options) {\n\tnode.addEventListener(event, handler, options);\n\treturn () => node.removeEventListener(event, handler, options);\n}\n\n/**\n * @returns {(event: any) => any} */\nexport function prevent_default(fn) {\n\treturn function (event) {\n\t\tevent.preventDefault();\n\t\t// @ts-ignore\n\t\treturn fn.call(this, event);\n\t};\n}\n\n/**\n * @returns {(event: any) => any} */\nexport function stop_propagation(fn) {\n\treturn function (event) {\n\t\tevent.stopPropagation();\n\t\t// @ts-ignore\n\t\treturn fn.call(this, event);\n\t};\n}\n\n/**\n * @returns {(event: any) => any} */\nexport function stop_immediate_propagation(fn) {\n\treturn function (event) {\n\t\tevent.stopImmediatePropagation();\n\t\t// @ts-ignore\n\t\treturn fn.call(this, event);\n\t};\n}\n\n/**\n * @returns {(event: any) => void} */\nexport function self(fn) {\n\treturn function (event) {\n\t\t// @ts-ignore\n\t\tif (event.target === this) fn.call(this, event);\n\t};\n}\n\n/**\n * @returns {(event: any) => void} */\nexport function trusted(fn) {\n\treturn function (event) {\n\t\t// @ts-ignore\n\t\tif (event.isTrusted) fn.call(this, event);\n\t};\n}\n\n/**\n * @param {Element} node\n * @param {string} attribute\n * @param {string} [value]\n * @returns {void}\n */\nexport function attr(node, attribute, value) {\n\tif (value == null) node.removeAttribute(attribute);\n\telse if (node.getAttribute(attribute) !== value) node.setAttribute(attribute, value);\n}\n/**\n * List of attributes that should always be set through the attr method,\n * because updating them through the property setter doesn't work reliably.\n * In the example of `width`/`height`, the problem is that the setter only\n * accepts numeric values, but the attribute can also be set to a string like `50%`.\n * If this list becomes too big, rethink this approach.\n */\nconst always_set_through_set_attribute = ['width', 'height'];\n\n/**\n * @param {Element & ElementCSSInlineStyle} node\n * @param {{ [x: string]: string }} attributes\n * @returns {void}\n */\nexport function set_attributes(node, attributes) {\n\t// @ts-ignore\n\tconst descriptors = Object.getOwnPropertyDescriptors(node.__proto__);\n\tfor (const key in attributes) {\n\t\tif (attributes[key] == null) {\n\t\t\tnode.removeAttribute(key);\n\t\t} else if (key === 'style') {\n\t\t\tnode.style.cssText = attributes[key];\n\t\t} else if (key === '__value') {\n\t\t\t/** @type {any} */ (node).value = node[key] = attributes[key];\n\t\t} else if (\n\t\t\tdescriptors[key] &&\n\t\t\tdescriptors[key].set &&\n\t\t\talways_set_through_set_attribute.indexOf(key) === -1\n\t\t) {\n\t\t\tnode[key] = attributes[key];\n\t\t} else {\n\t\t\tattr(node, key, attributes[key]);\n\t\t}\n\t}\n}\n\n/**\n * @param {Element & ElementCSSInlineStyle} node\n * @param {{ [x: string]: string }} attributes\n * @returns {void}\n */\nexport function set_svg_attributes(node, attributes) {\n\tfor (const key in attributes) {\n\t\tattr(node, key, attributes[key]);\n\t}\n}\n\n/**\n * @param {Record} data_map\n * @returns {void}\n */\nexport function set_custom_element_data_map(node, data_map) {\n\tObject.keys(data_map).forEach((key) => {\n\t\tset_custom_element_data(node, key, data_map[key]);\n\t});\n}\n\n/**\n * @returns {void} */\nexport function set_custom_element_data(node, prop, value) {\n\tif (prop in node) {\n\t\tnode[prop] = typeof node[prop] === 'boolean' && value === '' ? true : value;\n\t} else {\n\t\tattr(node, prop, value);\n\t}\n}\n\n/**\n * @param {string} tag\n */\nexport function set_dynamic_element_data(tag) {\n\treturn /-/.test(tag) ? set_custom_element_data_map : set_attributes;\n}\n\n/**\n * @returns {void}\n */\nexport function xlink_attr(node, attribute, value) {\n\tnode.setAttributeNS('http://www.w3.org/1999/xlink', attribute, value);\n}\n\n/**\n * @param {HTMLElement} node\n * @returns {string}\n */\nexport function get_svelte_dataset(node) {\n\treturn node.dataset.svelteH;\n}\n\n/**\n * @returns {unknown[]} */\nexport function get_binding_group_value(group, __value, checked) {\n\tconst value = new Set();\n\tfor (let i = 0; i < group.length; i += 1) {\n\t\tif (group[i].checked) value.add(group[i].__value);\n\t}\n\tif (!checked) {\n\t\tvalue.delete(__value);\n\t}\n\treturn Array.from(value);\n}\n\n/**\n * @param {HTMLInputElement[]} group\n * @returns {{ p(...inputs: HTMLInputElement[]): void; r(): void; }}\n */\nexport function init_binding_group(group) {\n\t/**\n\t * @type {HTMLInputElement[]} */\n\tlet _inputs;\n\treturn {\n\t\t/* push */ p(...inputs) {\n\t\t\t_inputs = inputs;\n\t\t\t_inputs.forEach((input) => group.push(input));\n\t\t},\n\t\t/* remove */ r() {\n\t\t\t_inputs.forEach((input) => group.splice(group.indexOf(input), 1));\n\t\t}\n\t};\n}\n\n/**\n * @param {number[]} indexes\n * @returns {{ u(new_indexes: number[]): void; p(...inputs: HTMLInputElement[]): void; r: () => void; }}\n */\nexport function init_binding_group_dynamic(group, indexes) {\n\t/**\n\t * @type {HTMLInputElement[]} */\n\tlet _group = get_binding_group(group);\n\n\t/**\n\t * @type {HTMLInputElement[]} */\n\tlet _inputs;\n\n\tfunction get_binding_group(group) {\n\t\tfor (let i = 0; i < indexes.length; i++) {\n\t\t\tgroup = group[indexes[i]] = group[indexes[i]] || [];\n\t\t}\n\t\treturn group;\n\t}\n\n\t/**\n\t * @returns {void} */\n\tfunction push() {\n\t\t_inputs.forEach((input) => _group.push(input));\n\t}\n\n\t/**\n\t * @returns {void} */\n\tfunction remove() {\n\t\t_inputs.forEach((input) => _group.splice(_group.indexOf(input), 1));\n\t}\n\treturn {\n\t\t/* update */ u(new_indexes) {\n\t\t\tindexes = new_indexes;\n\t\t\tconst new_group = get_binding_group(group);\n\t\t\tif (new_group !== _group) {\n\t\t\t\tremove();\n\t\t\t\t_group = new_group;\n\t\t\t\tpush();\n\t\t\t}\n\t\t},\n\t\t/* push */ p(...inputs) {\n\t\t\t_inputs = inputs;\n\t\t\tpush();\n\t\t},\n\t\t/* remove */ r: remove\n\t};\n}\n\n/** @returns {number} */\nexport function to_number(value) {\n\treturn value === '' ? null : +value;\n}\n\n/** @returns {any[]} */\nexport function time_ranges_to_array(ranges) {\n\tconst array = [];\n\tfor (let i = 0; i < ranges.length; i += 1) {\n\t\tarray.push({ start: ranges.start(i), end: ranges.end(i) });\n\t}\n\treturn array;\n}\n\n/**\n * @param {Element} element\n * @returns {ChildNode[]}\n */\nexport function children(element) {\n\treturn Array.from(element.childNodes);\n}\n\n/**\n * @param {ChildNodeArray} nodes\n * @returns {void}\n */\nfunction init_claim_info(nodes) {\n\tif (nodes.claim_info === undefined) {\n\t\tnodes.claim_info = { last_index: 0, total_claimed: 0 };\n\t}\n}\n\n/**\n * @template {ChildNodeEx} R\n * @param {ChildNodeArray} nodes\n * @param {(node: ChildNodeEx) => node is R} predicate\n * @param {(node: ChildNodeEx) => ChildNodeEx | undefined} process_node\n * @param {() => R} create_node\n * @param {boolean} dont_update_last_index\n * @returns {R}\n */\nfunction claim_node(nodes, predicate, process_node, create_node, dont_update_last_index = false) {\n\t// Try to find nodes in an order such that we lengthen the longest increasing subsequence\n\tinit_claim_info(nodes);\n\tconst result_node = (() => {\n\t\t// We first try to find an element after the previous one\n\t\tfor (let i = nodes.claim_info.last_index; i < nodes.length; i++) {\n\t\t\tconst node = nodes[i];\n\t\t\tif (predicate(node)) {\n\t\t\t\tconst replacement = process_node(node);\n\t\t\t\tif (replacement === undefined) {\n\t\t\t\t\tnodes.splice(i, 1);\n\t\t\t\t} else {\n\t\t\t\t\tnodes[i] = replacement;\n\t\t\t\t}\n\t\t\t\tif (!dont_update_last_index) {\n\t\t\t\t\tnodes.claim_info.last_index = i;\n\t\t\t\t}\n\t\t\t\treturn node;\n\t\t\t}\n\t\t}\n\t\t// Otherwise, we try to find one before\n\t\t// We iterate in reverse so that we don't go too far back\n\t\tfor (let i = nodes.claim_info.last_index - 1; i >= 0; i--) {\n\t\t\tconst node = nodes[i];\n\t\t\tif (predicate(node)) {\n\t\t\t\tconst replacement = process_node(node);\n\t\t\t\tif (replacement === undefined) {\n\t\t\t\t\tnodes.splice(i, 1);\n\t\t\t\t} else {\n\t\t\t\t\tnodes[i] = replacement;\n\t\t\t\t}\n\t\t\t\tif (!dont_update_last_index) {\n\t\t\t\t\tnodes.claim_info.last_index = i;\n\t\t\t\t} else if (replacement === undefined) {\n\t\t\t\t\t// Since we spliced before the last_index, we decrease it\n\t\t\t\t\tnodes.claim_info.last_index--;\n\t\t\t\t}\n\t\t\t\treturn node;\n\t\t\t}\n\t\t}\n\t\t// If we can't find any matching node, we create a new one\n\t\treturn create_node();\n\t})();\n\tresult_node.claim_order = nodes.claim_info.total_claimed;\n\tnodes.claim_info.total_claimed += 1;\n\treturn result_node;\n}\n\n/**\n * @param {ChildNodeArray} nodes\n * @param {string} name\n * @param {{ [key: string]: boolean }} attributes\n * @param {(name: string) => Element | SVGElement} create_element\n * @returns {Element | SVGElement}\n */\nfunction claim_element_base(nodes, name, attributes, create_element) {\n\treturn claim_node(\n\t\tnodes,\n\t\t/** @returns {node is Element | SVGElement} */\n\t\t(node) => node.nodeName === name,\n\t\t/** @param {Element} node */\n\t\t(node) => {\n\t\t\tconst remove = [];\n\t\t\tfor (let j = 0; j < node.attributes.length; j++) {\n\t\t\t\tconst attribute = node.attributes[j];\n\t\t\t\tif (!attributes[attribute.name]) {\n\t\t\t\t\tremove.push(attribute.name);\n\t\t\t\t}\n\t\t\t}\n\t\t\tremove.forEach((v) => node.removeAttribute(v));\n\t\t\treturn undefined;\n\t\t},\n\t\t() => create_element(name)\n\t);\n}\n\n/**\n * @param {ChildNodeArray} nodes\n * @param {string} name\n * @param {{ [key: string]: boolean }} attributes\n * @returns {Element | SVGElement}\n */\nexport function claim_element(nodes, name, attributes) {\n\treturn claim_element_base(nodes, name, attributes, element);\n}\n\n/**\n * @param {ChildNodeArray} nodes\n * @param {string} name\n * @param {{ [key: string]: boolean }} attributes\n * @returns {Element | SVGElement}\n */\nexport function claim_svg_element(nodes, name, attributes) {\n\treturn claim_element_base(nodes, name, attributes, svg_element);\n}\n\n/**\n * @param {ChildNodeArray} nodes\n * @returns {Text}\n */\nexport function claim_text(nodes, data) {\n\treturn claim_node(\n\t\tnodes,\n\t\t/** @returns {node is Text} */\n\t\t(node) => node.nodeType === 3,\n\t\t/** @param {Text} node */\n\t\t(node) => {\n\t\t\tconst data_str = '' + data;\n\t\t\tif (node.data.startsWith(data_str)) {\n\t\t\t\tif (node.data.length !== data_str.length) {\n\t\t\t\t\treturn node.splitText(data_str.length);\n\t\t\t\t}\n\t\t\t} else {\n\t\t\t\tnode.data = data_str;\n\t\t\t}\n\t\t},\n\t\t() => text(data),\n\t\ttrue // Text nodes should not update last index since it is likely not worth it to eliminate an increasing subsequence of actual elements\n\t);\n}\n\n/**\n * @returns {Text} */\nexport function claim_space(nodes) {\n\treturn claim_text(nodes, ' ');\n}\n\n/**\n * @param {ChildNodeArray} nodes\n * @returns {Comment}\n */\nexport function claim_comment(nodes, data) {\n\treturn claim_node(\n\t\tnodes,\n\t\t/** @returns {node is Comment} */\n\t\t(node) => node.nodeType === 8,\n\t\t/** @param {Comment} node */\n\t\t(node) => {\n\t\t\tnode.data = '' + data;\n\t\t\treturn undefined;\n\t\t},\n\t\t() => comment(data),\n\t\ttrue\n\t);\n}\n\nfunction get_comment_idx(nodes, text, start) {\n\tfor (let i = start; i < nodes.length; i += 1) {\n\t\tconst node = nodes[i];\n\t\tif (node.nodeType === 8 /* comment node */ && node.textContent.trim() === text) {\n\t\t\treturn i;\n\t\t}\n\t}\n\treturn -1;\n}\n\n/**\n * @param {boolean} is_svg\n * @returns {HtmlTagHydration}\n */\nexport function claim_html_tag(nodes, is_svg) {\n\t// find html opening tag\n\tconst start_index = get_comment_idx(nodes, 'HTML_TAG_START', 0);\n\tconst end_index = get_comment_idx(nodes, 'HTML_TAG_END', start_index + 1);\n\tif (start_index === -1 || end_index === -1) {\n\t\treturn new HtmlTagHydration(is_svg);\n\t}\n\n\tinit_claim_info(nodes);\n\tconst html_tag_nodes = nodes.splice(start_index, end_index - start_index + 1);\n\tdetach(html_tag_nodes[0]);\n\tdetach(html_tag_nodes[html_tag_nodes.length - 1]);\n\tconst claimed_nodes = html_tag_nodes.slice(1, html_tag_nodes.length - 1);\n\tfor (const n of claimed_nodes) {\n\t\tn.claim_order = nodes.claim_info.total_claimed;\n\t\tnodes.claim_info.total_claimed += 1;\n\t}\n\treturn new HtmlTagHydration(is_svg, claimed_nodes);\n}\n\n/**\n * @param {Text} text\n * @param {unknown} data\n * @returns {void}\n */\nexport function set_data(text, data) {\n\tdata = '' + data;\n\tif (text.data === data) return;\n\ttext.data = /** @type {string} */ (data);\n}\n\n/**\n * @param {Text} text\n * @param {unknown} data\n * @returns {void}\n */\nexport function set_data_contenteditable(text, data) {\n\tdata = '' + data;\n\tif (text.wholeText === data) return;\n\ttext.data = /** @type {string} */ (data);\n}\n\n/**\n * @param {Text} text\n * @param {unknown} data\n * @param {string} attr_value\n * @returns {void}\n */\nexport function set_data_maybe_contenteditable(text, data, attr_value) {\n\tif (~contenteditable_truthy_values.indexOf(attr_value)) {\n\t\tset_data_contenteditable(text, data);\n\t} else {\n\t\tset_data(text, data);\n\t}\n}\n\n/**\n * @returns {void} */\nexport function set_input_value(input, value) {\n\tinput.value = value == null ? '' : value;\n}\n\n/**\n * @returns {void} */\nexport function set_input_type(input, type) {\n\ttry {\n\t\tinput.type = type;\n\t} catch (e) {\n\t\t// do nothing\n\t}\n}\n\n/**\n * @returns {void} */\nexport function set_style(node, key, value, important) {\n\tif (value == null) {\n\t\tnode.style.removeProperty(key);\n\t} else {\n\t\tnode.style.setProperty(key, value, important ? 'important' : '');\n\t}\n}\n\n/**\n * @returns {void} */\nexport function select_option(select, value, mounting) {\n\tfor (let i = 0; i < select.options.length; i += 1) {\n\t\tconst option = select.options[i];\n\t\tif (option.__value === value) {\n\t\t\toption.selected = true;\n\t\t\treturn;\n\t\t}\n\t}\n\tif (!mounting || value !== undefined) {\n\t\tselect.selectedIndex = -1; // no option should be selected\n\t}\n}\n\n/**\n * @returns {void} */\nexport function select_options(select, value) {\n\tfor (let i = 0; i < select.options.length; i += 1) {\n\t\tconst option = select.options[i];\n\t\toption.selected = ~value.indexOf(option.__value);\n\t}\n}\n\nexport function select_value(select) {\n\tconst selected_option = select.querySelector(':checked');\n\treturn selected_option && selected_option.__value;\n}\n\nexport function select_multiple_value(select) {\n\treturn [].map.call(select.querySelectorAll(':checked'), (option) => option.__value);\n}\n// unfortunately this can't be a constant as that wouldn't be tree-shakeable\n// so we cache the result instead\n\n/**\n * @type {boolean} */\nlet crossorigin;\n\n/**\n * @returns {boolean} */\nexport function is_crossorigin() {\n\tif (crossorigin === undefined) {\n\t\tcrossorigin = false;\n\t\ttry {\n\t\t\tif (typeof window !== 'undefined' && window.parent) {\n\t\t\t\tvoid window.parent.document;\n\t\t\t}\n\t\t} catch (error) {\n\t\t\tcrossorigin = true;\n\t\t}\n\t}\n\treturn crossorigin;\n}\n\n/**\n * @param {HTMLElement} node\n * @param {() => void} fn\n * @returns {() => void}\n */\nexport function add_iframe_resize_listener(node, fn) {\n\tconst computed_style = getComputedStyle(node);\n\tif (computed_style.position === 'static') {\n\t\tnode.style.position = 'relative';\n\t}\n\tconst iframe = element('iframe');\n\tiframe.setAttribute(\n\t\t'style',\n\t\t'display: block; position: absolute; top: 0; left: 0; width: 100%; height: 100%; ' +\n\t\t\t'overflow: hidden; border: 0; opacity: 0; pointer-events: none; z-index: -1;'\n\t);\n\tiframe.setAttribute('aria-hidden', 'true');\n\tiframe.tabIndex = -1;\n\tconst crossorigin = is_crossorigin();\n\n\t/**\n\t * @type {() => void}\n\t */\n\tlet unsubscribe;\n\tif (crossorigin) {\n\t\tiframe.src = \"data:text/html,\";\n\t\tunsubscribe = listen(\n\t\t\twindow,\n\t\t\t'message',\n\t\t\t/** @param {MessageEvent} event */ (event) => {\n\t\t\t\tif (event.source === iframe.contentWindow) fn();\n\t\t\t}\n\t\t);\n\t} else {\n\t\tiframe.src = 'about:blank';\n\t\tiframe.onload = () => {\n\t\t\tunsubscribe = listen(iframe.contentWindow, 'resize', fn);\n\t\t\t// make sure an initial resize event is fired _after_ the iframe is loaded (which is asynchronous)\n\t\t\t// see https://github.com/sveltejs/svelte/issues/4233\n\t\t\tfn();\n\t\t};\n\t}\n\tappend(node, iframe);\n\treturn () => {\n\t\tif (crossorigin) {\n\t\t\tunsubscribe();\n\t\t} else if (unsubscribe && iframe.contentWindow) {\n\t\t\tunsubscribe();\n\t\t}\n\t\tdetach(iframe);\n\t};\n}\nexport const resize_observer_content_box = /* @__PURE__ */ new ResizeObserverSingleton({\n\tbox: 'content-box'\n});\nexport const resize_observer_border_box = /* @__PURE__ */ new ResizeObserverSingleton({\n\tbox: 'border-box'\n});\nexport const resize_observer_device_pixel_content_box = /* @__PURE__ */ new ResizeObserverSingleton(\n\t{ box: 'device-pixel-content-box' }\n);\nexport { ResizeObserverSingleton };\n\n/**\n * @returns {void} */\nexport function toggle_class(element, name, toggle) {\n\t// The `!!` is required because an `undefined` flag means flipping the current state.\n\telement.classList.toggle(name, !!toggle);\n}\n\n/**\n * @template T\n * @param {string} type\n * @param {T} [detail]\n * @param {{ bubbles?: boolean, cancelable?: boolean }} [options]\n * @returns {CustomEvent}\n */\nexport function custom_event(type, detail, { bubbles = false, cancelable = false } = {}) {\n\treturn new CustomEvent(type, { detail, bubbles, cancelable });\n}\n\n/**\n * @param {string} selector\n * @param {HTMLElement} parent\n * @returns {ChildNodeArray}\n */\nexport function query_selector_all(selector, parent = document.body) {\n\treturn Array.from(parent.querySelectorAll(selector));\n}\n\n/**\n * @param {string} nodeId\n * @param {HTMLElement} head\n * @returns {any[]}\n */\nexport function head_selector(nodeId, head) {\n\tconst result = [];\n\tlet started = 0;\n\tfor (const node of head.childNodes) {\n\t\tif (node.nodeType === 8 /* comment node */) {\n\t\t\tconst comment = node.textContent.trim();\n\t\t\tif (comment === `HEAD_${nodeId}_END`) {\n\t\t\t\tstarted -= 1;\n\t\t\t\tresult.push(node);\n\t\t\t} else if (comment === `HEAD_${nodeId}_START`) {\n\t\t\t\tstarted += 1;\n\t\t\t\tresult.push(node);\n\t\t\t}\n\t\t} else if (started > 0) {\n\t\t\tresult.push(node);\n\t\t}\n\t}\n\treturn result;\n}\n/** */\nexport class HtmlTag {\n\t/**\n\t * @private\n\t * @default false\n\t */\n\tis_svg = false;\n\t/** parent for creating node */\n\te = undefined;\n\t/** html tag nodes */\n\tn = undefined;\n\t/** target */\n\tt = undefined;\n\t/** anchor */\n\ta = undefined;\n\tconstructor(is_svg = false) {\n\t\tthis.is_svg = is_svg;\n\t\tthis.e = this.n = null;\n\t}\n\n\t/**\n\t * @param {string} html\n\t * @returns {void}\n\t */\n\tc(html) {\n\t\tthis.h(html);\n\t}\n\n\t/**\n\t * @param {string} html\n\t * @param {HTMLElement | SVGElement} target\n\t * @param {HTMLElement | SVGElement} anchor\n\t * @returns {void}\n\t */\n\tm(html, target, anchor = null) {\n\t\tif (!this.e) {\n\t\t\tif (this.is_svg)\n\t\t\t\tthis.e = svg_element(/** @type {keyof SVGElementTagNameMap} */ (target.nodeName));\n\t\t\t/** #7364 target for