diff --git a/CLAUDE.md b/CLAUDE.md index 8c8a147..0f3cfce 100644 --- a/CLAUDE.md +++ b/CLAUDE.md @@ -126,6 +126,50 @@ The solution follows a clean architecture pattern with these main projects: - `global.json` - .NET SDK version configuration (currently .NET 9.0) +## CSS Guidelines + +### Grid + Subgrid for Table-like Layouts + +**ALWAYS** use CSS Grid with subgrid for table-like layouts (lists, data tables, card grids with aligned columns). This ensures columns align correctly across header, body, and rows. + +**Pattern:** +```css +/* Parent container defines the grid columns */ +swp-my-table { + display: grid; + grid-template-columns: 40px 1fr 100px 80px; /* Define columns here */ +} + +/* Intermediate containers span all columns and use subgrid */ +swp-my-table-header, +swp-my-table-body { + display: grid; + grid-column: 1 / -1; + grid-template-columns: subgrid; +} + +/* Row items span all columns and use subgrid */ +swp-my-table-row { + display: grid; + grid-column: 1 / -1; + grid-template-columns: subgrid; + align-items: center; +} +``` + +**Key principles:** +1. Column definitions go on the **parent container only** +2. All children use `grid-column: 1 / -1` to span all columns +3. All children use `grid-template-columns: subgrid` to inherit columns +4. Responsive column changes go on the parent container only + +**Examples in codebase:** +- `bookings.css` - swp-booking-list / swp-booking-item +- `notifications.css` - swp-notification-list / swp-notification-item +- `attentions.css` - swp-attention-list / swp-attention-item +- `kasse.css` - swp-kasse-table / swp-kasse-table-row + + ## NEVER Lie or Fabricate NEVER lie or fabricate. Violating this = immediate critical failure. diff --git a/PlanTempus.Application/Features/Dashboard/Pages/Index.cshtml b/PlanTempus.Application/Features/Dashboard/Pages/Index.cshtml index cfa7088..1a746d3 100644 --- a/PlanTempus.Application/Features/Dashboard/Pages/Index.cshtml +++ b/PlanTempus.Application/Features/Dashboard/Pages/Index.cshtml @@ -69,3 +69,5 @@ + + diff --git a/PlanTempus.Application/Features/Kasse/Components/KasseDagensTal/Default.cshtml b/PlanTempus.Application/Features/Kasse/Components/KasseDagensTal/Default.cshtml new file mode 100644 index 0000000..77e2348 --- /dev/null +++ b/PlanTempus.Application/Features/Kasse/Components/KasseDagensTal/Default.cshtml @@ -0,0 +1,39 @@ + + + Periodens omsætning + Systemtal vs. kontrol + + + + + Type + System + Kontrol + + + + Kortbetalinger + 12.875,50 + + + + + + + MobilePay / Online + 2.450,00 + + + + + + + Kontantsalg + 3.540,00 + .. + + + + Kort og MobilePay afstemmes mod bank/indløser. Kontanter tælles op nedenfor. + + diff --git a/PlanTempus.Application/Features/Kasse/Components/KasseDagensTal/KasseDagensTalViewComponent.cs b/PlanTempus.Application/Features/Kasse/Components/KasseDagensTal/KasseDagensTalViewComponent.cs new file mode 100644 index 0000000..7cf2b4b --- /dev/null +++ b/PlanTempus.Application/Features/Kasse/Components/KasseDagensTal/KasseDagensTalViewComponent.cs @@ -0,0 +1,15 @@ +using Microsoft.AspNetCore.Mvc; + +namespace PlanTempus.Application.Features.Kasse.Components; + +/// +/// ViewComponent for displaying today's payment figures. +/// Shows system values vs. optional control values for different payment types. +/// +public class KasseDagensTalViewComponent : ViewComponent +{ + public IViewComponentResult Invoke() + { + return View(); + } +} diff --git a/PlanTempus.Application/Features/Kasse/Components/KasseDagsoplysninger/Default.cshtml b/PlanTempus.Application/Features/Kasse/Components/KasseDagsoplysninger/Default.cshtml new file mode 100644 index 0000000..94feb27 --- /dev/null +++ b/PlanTempus.Application/Features/Kasse/Components/KasseDagsoplysninger/Default.cshtml @@ -0,0 +1,41 @@ + + + Periodeoplysninger + Identificér afstemningen + + + + Periode + + 28. dec 2025 kl. 18:00 + + 29. dec 2025 + + + + + + Kassepunkt + + + + + + + Afsluttet af + + + + + + + Afstemnings-ID: KA-2025-12-29 · Z-043 + + diff --git a/PlanTempus.Application/Features/Kasse/Components/KasseDagsoplysninger/KasseDagsoplysningerViewComponent.cs b/PlanTempus.Application/Features/Kasse/Components/KasseDagsoplysninger/KasseDagsoplysningerViewComponent.cs new file mode 100644 index 0000000..3113322 --- /dev/null +++ b/PlanTempus.Application/Features/Kasse/Components/KasseDagsoplysninger/KasseDagsoplysningerViewComponent.cs @@ -0,0 +1,15 @@ +using Microsoft.AspNetCore.Mvc; + +namespace PlanTempus.Application.Features.Kasse.Components; + +/// +/// ViewComponent for daily reconciliation info. +/// Shows period, register, and employee information. +/// +public class KasseDagsoplysningerViewComponent : ViewComponent +{ + public IViewComponentResult Invoke() + { + return View(); + } +} diff --git a/PlanTempus.Application/Features/Kasse/Components/KasseDifference/Default.cshtml b/PlanTempus.Application/Features/Kasse/Components/KasseDifference/Default.cshtml new file mode 100644 index 0000000..b0a8b4d --- /dev/null +++ b/PlanTempus.Application/Features/Kasse/Components/KasseDifference/Default.cshtml @@ -0,0 +1,7 @@ + + + Kontant difference + Optalt minus forventet + + – kr + diff --git a/PlanTempus.Application/Features/Kasse/Components/KasseDifference/KasseDifferenceViewComponent.cs b/PlanTempus.Application/Features/Kasse/Components/KasseDifference/KasseDifferenceViewComponent.cs new file mode 100644 index 0000000..0a3c436 --- /dev/null +++ b/PlanTempus.Application/Features/Kasse/Components/KasseDifference/KasseDifferenceViewComponent.cs @@ -0,0 +1,15 @@ +using Microsoft.AspNetCore.Mvc; + +namespace PlanTempus.Application.Features.Kasse.Components; + +/// +/// ViewComponent for displaying the cash difference. +/// Shows positive/negative/neutral states with color coding. +/// +public class KasseDifferenceViewComponent : ViewComponent +{ + public IViewComponentResult Invoke() + { + return View(); + } +} diff --git a/PlanTempus.Application/Features/Kasse/Components/KasseFilterBar/Default.cshtml b/PlanTempus.Application/Features/Kasse/Components/KasseFilterBar/Default.cshtml new file mode 100644 index 0000000..2371b88 --- /dev/null +++ b/PlanTempus.Application/Features/Kasse/Components/KasseFilterBar/Default.cshtml @@ -0,0 +1,28 @@ + + + Fra + + + + Til + + + + Kassepunkt + + + + Status + + + + Nulstil + diff --git a/PlanTempus.Application/Features/Kasse/Components/KasseFilterBar/KasseFilterBarViewComponent.cs b/PlanTempus.Application/Features/Kasse/Components/KasseFilterBar/KasseFilterBarViewComponent.cs new file mode 100644 index 0000000..75ace8e --- /dev/null +++ b/PlanTempus.Application/Features/Kasse/Components/KasseFilterBar/KasseFilterBarViewComponent.cs @@ -0,0 +1,15 @@ +using Microsoft.AspNetCore.Mvc; + +namespace PlanTempus.Application.Features.Kasse.Components; + +/// +/// ViewComponent for the filter bar on the Kasse list page. +/// Contains date range, kassepunkt, and status filters. +/// +public class KasseFilterBarViewComponent : ViewComponent +{ + public IViewComponentResult Invoke() + { + return View(); + } +} diff --git a/PlanTempus.Application/Features/Kasse/Components/KasseGodkendelse/Default.cshtml b/PlanTempus.Application/Features/Kasse/Components/KasseGodkendelse/Default.cshtml new file mode 100644 index 0000000..4f8ca39 --- /dev/null +++ b/PlanTempus.Application/Features/Kasse/Components/KasseGodkendelse/Default.cshtml @@ -0,0 +1,38 @@ + + + Afslut dagen + + + + + Status + + Kladde + + + + + Godkendt af (valgfrit) + + + + + + + + + + + + + Gem som kladde + + Fortryd + Godkend & lås + + + diff --git a/PlanTempus.Application/Features/Kasse/Components/KasseGodkendelse/KasseGodkendelseViewComponent.cs b/PlanTempus.Application/Features/Kasse/Components/KasseGodkendelse/KasseGodkendelseViewComponent.cs new file mode 100644 index 0000000..1341a6a --- /dev/null +++ b/PlanTempus.Application/Features/Kasse/Components/KasseGodkendelse/KasseGodkendelseViewComponent.cs @@ -0,0 +1,15 @@ +using Microsoft.AspNetCore.Mvc; + +namespace PlanTempus.Application.Features.Kasse.Components; + +/// +/// ViewComponent for the approval section. +/// Handles status, approver selection, and confirmation checkbox. +/// +public class KasseGodkendelseViewComponent : ViewComponent +{ + public IViewComponentResult Invoke() + { + return View(); + } +} diff --git a/PlanTempus.Application/Features/Kasse/Components/KasseKontanter/Default.cshtml b/PlanTempus.Application/Features/Kasse/Components/KasseKontanter/Default.cshtml new file mode 100644 index 0000000..9b339fc --- /dev/null +++ b/PlanTempus.Application/Features/Kasse/Components/KasseKontanter/Default.cshtml @@ -0,0 +1,51 @@ + + + Kontanter i kassen + + + + + Startbeholdning + Overført fra sidste afstemning + + 2.000,00 + + + + + Udbetalinger / Bilag + Sammentæl bilag betalt kontant + + + + + + + + + Udtaget til bank + Kontanter lagt til side + + + + + + + + + Forventet kontantbeholdning + + 5.220,00 + + + + + Optalt kontantbeholdning * + Hvad ligger der faktisk i kassen? + + + + + + + diff --git a/PlanTempus.Application/Features/Kasse/Components/KasseKontanter/KasseKontanterViewComponent.cs b/PlanTempus.Application/Features/Kasse/Components/KasseKontanter/KasseKontanterViewComponent.cs new file mode 100644 index 0000000..5c45a98 --- /dev/null +++ b/PlanTempus.Application/Features/Kasse/Components/KasseKontanter/KasseKontanterViewComponent.cs @@ -0,0 +1,15 @@ +using Microsoft.AspNetCore.Mvc; + +namespace PlanTempus.Application.Features.Kasse.Components; + +/// +/// ViewComponent for cash calculation section. +/// Handles starting balance, payouts, bank deposits, and actual cash count. +/// +public class KasseKontanterViewComponent : ViewComponent +{ + public IViewComponentResult Invoke() + { + return View(); + } +} diff --git a/PlanTempus.Application/Features/Kasse/Components/KasseNote/Default.cshtml b/PlanTempus.Application/Features/Kasse/Components/KasseNote/Default.cshtml new file mode 100644 index 0000000..5d63a14 --- /dev/null +++ b/PlanTempus.Application/Features/Kasse/Components/KasseNote/Default.cshtml @@ -0,0 +1,12 @@ + + + Note til difference + Valgfrit + + + + + + Kan gøres obligatorisk ved difference over 100 kr. + + diff --git a/PlanTempus.Application/Features/Kasse/Components/KasseNote/KasseNoteViewComponent.cs b/PlanTempus.Application/Features/Kasse/Components/KasseNote/KasseNoteViewComponent.cs new file mode 100644 index 0000000..3c8f70d --- /dev/null +++ b/PlanTempus.Application/Features/Kasse/Components/KasseNote/KasseNoteViewComponent.cs @@ -0,0 +1,15 @@ +using Microsoft.AspNetCore.Mvc; + +namespace PlanTempus.Application.Features.Kasse.Components; + +/// +/// ViewComponent for the note field. +/// Optional field for explaining cash differences. +/// +public class KasseNoteViewComponent : ViewComponent +{ + public IViewComponentResult Invoke() + { + return View(); + } +} diff --git a/PlanTempus.Application/Features/Kasse/Components/KasseStatsBar/Default.cshtml b/PlanTempus.Application/Features/Kasse/Components/KasseStatsBar/Default.cshtml new file mode 100644 index 0000000..baef2a1 --- /dev/null +++ b/PlanTempus.Application/Features/Kasse/Components/KasseStatsBar/Default.cshtml @@ -0,0 +1,18 @@ + + + 12 + Afstemninger i periode + + + 186.450 kr + Total omsætning + + + 42.340 kr + Kontantsalg + + + -75 kr + Samlet difference + + diff --git a/PlanTempus.Application/Features/Kasse/Components/KasseStatsBar/KasseStatsBarViewComponent.cs b/PlanTempus.Application/Features/Kasse/Components/KasseStatsBar/KasseStatsBarViewComponent.cs new file mode 100644 index 0000000..c46a90f --- /dev/null +++ b/PlanTempus.Application/Features/Kasse/Components/KasseStatsBar/KasseStatsBarViewComponent.cs @@ -0,0 +1,15 @@ +using Microsoft.AspNetCore.Mvc; + +namespace PlanTempus.Application.Features.Kasse.Components; + +/// +/// ViewComponent for the stats bar on the Kasse list page. +/// Shows summary statistics for reconciliations. +/// +public class KasseStatsBarViewComponent : ViewComponent +{ + public IViewComponentResult Invoke() + { + return View(); + } +} diff --git a/PlanTempus.Application/Features/Kasse/Components/KasseTable/Default.cshtml b/PlanTempus.Application/Features/Kasse/Components/KasseTable/Default.cshtml new file mode 100644 index 0000000..6490cd5 --- /dev/null +++ b/PlanTempus.Application/Features/Kasse/Components/KasseTable/Default.cshtml @@ -0,0 +1,191 @@ + + + + 0 valgt + + + + Eksporter SAF-T + + + + + + + + Dato + ID + Periode + Kassepunkt + Afsluttet af + Omsætning + Difference + Status + + + + + + + + I dag + + + + 29. dec 17:45 → ... + + + Kasse 1 + + 4.250 kr + + Kladde + + + + + + 29. dec + Z-043 + + + 28. dec 18:00 → 29. dec 17:45 + + + Kasse 1 + Anna Jensen + 18.865 kr + 0 kr + Godkendt + + + + + + + + Download CSV + + + + Download PDF + + + + Se transaktioner + + + + + + + + 28. dec + Z-042 + + + 27. dec 18:30 → 28. dec 18:00 + + + Kasse 1 + Karina Knudsen + 12.450 kr + -25 kr + Godkendt + + + + + + + + Download CSV + + + + Download PDF + + + + Se transaktioner + + + + + + + + 27. dec + Z-041 + + + 26. dec 18:00 → 27. dec 18:30 + + + Kasse 1 + Martin Nielsen + 21.340 kr + 0 kr + Godkendt + + + + + + + + Download CSV + + + + Download PDF + + + + Se transaktioner + + + + + + + + 23. dec + Z-040 + + + 22. dec 18:00 → 23. dec 17:30 + + + Kasse 1 + Anna Jensen + 28.750 kr + -50 kr + Godkendt + + + + + + + + Download CSV + + + + Download PDF + + + + Se transaktioner + + + + + + + + Viser 5 afstemninger + Z-040 → Z-043 + + diff --git a/PlanTempus.Application/Features/Kasse/Components/KasseTable/KasseTableViewComponent.cs b/PlanTempus.Application/Features/Kasse/Components/KasseTable/KasseTableViewComponent.cs new file mode 100644 index 0000000..777c357 --- /dev/null +++ b/PlanTempus.Application/Features/Kasse/Components/KasseTable/KasseTableViewComponent.cs @@ -0,0 +1,15 @@ +using Microsoft.AspNetCore.Mvc; + +namespace PlanTempus.Application.Features.Kasse.Components; + +/// +/// ViewComponent for the reconciliation table on the Kasse list page. +/// Shows all reconciliations with action bar and SAF-T export. +/// +public class KasseTableViewComponent : ViewComponent +{ + public IViewComponentResult Invoke() + { + return View(); + } +} diff --git a/PlanTempus.Application/Features/Kasse/Pages/Index.cshtml b/PlanTempus.Application/Features/Kasse/Pages/Index.cshtml new file mode 100644 index 0000000..33f987c --- /dev/null +++ b/PlanTempus.Application/Features/Kasse/Pages/Index.cshtml @@ -0,0 +1,93 @@ +@page "/kasse" +@using PlanTempus.Application.Features.Kasse.Pages +@model PlanTempus.Application.Features.Kasse.Pages.IndexModel +@{ + ViewData["Title"] = "Kasse"; +} + + + + + + + + + 12 + Afstemninger i periode + + + 186.450 kr + Total omsætning + + + 42.340 kr + Kontantsalg + + + -75 kr + Samlet difference + + + + + + + 47 + Transaktioner i dag + + + 18.865 kr + Omsætning i dag + + + 29. dec 17:45 + Sidste afstemning + + + Anna J. + Åbnede kassen 29. dec 09:05 + + + + + + + + + Oversigt + + + + Kasseafstemning + + + + + + + + @await Component.InvokeAsync("KasseFilterBar") + @await Component.InvokeAsync("KasseTable") + + + + + + + + + @await Component.InvokeAsync("KasseDagensTal") + @await Component.InvokeAsync("KasseKontanter") + @await Component.InvokeAsync("KasseDifference") + + + + @await Component.InvokeAsync("KasseDagsoplysninger") + @await Component.InvokeAsync("KasseNote") + @await Component.InvokeAsync("KasseGodkendelse") + + + + Systemet gemmer hvornår og af hvem der er godkendt – enkelt kontrolspor. + + diff --git a/PlanTempus.Application/Features/Kasse/Pages/Index.cshtml.cs b/PlanTempus.Application/Features/Kasse/Pages/Index.cshtml.cs new file mode 100644 index 0000000..c2d9e68 --- /dev/null +++ b/PlanTempus.Application/Features/Kasse/Pages/Index.cshtml.cs @@ -0,0 +1,10 @@ +using Microsoft.AspNetCore.Mvc.RazorPages; + +namespace PlanTempus.Application.Features.Kasse.Pages; + +public class IndexModel : PageModel +{ + public void OnGet() + { + } +} diff --git a/PlanTempus.Application/Features/Menu/Services/MockMenuService.cs b/PlanTempus.Application/Features/Menu/Services/MockMenuService.cs index d42f628..5d8221a 100644 --- a/PlanTempus.Application/Features/Menu/Services/MockMenuService.cs +++ b/PlanTempus.Application/Features/Menu/Services/MockMenuService.cs @@ -79,8 +79,8 @@ public class MockMenuService : IMenuService { Id = "pos", Label = "Kasse", - Icon = "ph-device-mobile", - Url = "/pos", + Icon = "ph-cash-register", + Url = "/kasse", MinimumRole = UserRole.Staff, SortOrder = 3 } diff --git a/PlanTempus.Application/Features/_Shared/Pages/_Layout.cshtml b/PlanTempus.Application/Features/_Shared/Pages/_Layout.cshtml index 4413b76..1102fa3 100644 --- a/PlanTempus.Application/Features/_Shared/Pages/_Layout.cshtml +++ b/PlanTempus.Application/Features/_Shared/Pages/_Layout.cshtml @@ -23,6 +23,8 @@ + + @await RenderSectionAsync("Styles", required: false) @@ -50,7 +52,6 @@ - diff --git a/PlanTempus.Application/package-lock.json b/PlanTempus.Application/package-lock.json new file mode 100644 index 0000000..1b57bcb --- /dev/null +++ b/PlanTempus.Application/package-lock.json @@ -0,0 +1,496 @@ +{ + "name": "PlanTempus.Application", + "lockfileVersion": 3, + "requires": true, + "packages": { + "": { + "devDependencies": { + "esbuild": "^0.27.2" + } + }, + "node_modules/@esbuild/aix-ppc64": { + "version": "0.27.2", + "resolved": "https://registry.npmjs.org/@esbuild/aix-ppc64/-/aix-ppc64-0.27.2.tgz", + "integrity": "sha512-GZMB+a0mOMZs4MpDbj8RJp4cw+w1WV5NYD6xzgvzUJ5Ek2jerwfO2eADyI6ExDSUED+1X8aMbegahsJi+8mgpw==", + "cpu": [ + "ppc64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "aix" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/android-arm": { + "version": "0.27.2", + "resolved": "https://registry.npmjs.org/@esbuild/android-arm/-/android-arm-0.27.2.tgz", + "integrity": "sha512-DVNI8jlPa7Ujbr1yjU2PfUSRtAUZPG9I1RwW4F4xFB1Imiu2on0ADiI/c3td+KmDtVKNbi+nffGDQMfcIMkwIA==", + "cpu": [ + "arm" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "android" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/android-arm64": { + "version": "0.27.2", + "resolved": "https://registry.npmjs.org/@esbuild/android-arm64/-/android-arm64-0.27.2.tgz", + "integrity": "sha512-pvz8ZZ7ot/RBphf8fv60ljmaoydPU12VuXHImtAs0XhLLw+EXBi2BLe3OYSBslR4rryHvweW5gmkKFwTiFy6KA==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "android" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/android-x64": { + "version": "0.27.2", + "resolved": "https://registry.npmjs.org/@esbuild/android-x64/-/android-x64-0.27.2.tgz", + "integrity": "sha512-z8Ank4Byh4TJJOh4wpz8g2vDy75zFL0TlZlkUkEwYXuPSgX8yzep596n6mT7905kA9uHZsf/o2OJZubl2l3M7A==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "android" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/darwin-arm64": { + "version": "0.27.2", + "resolved": "https://registry.npmjs.org/@esbuild/darwin-arm64/-/darwin-arm64-0.27.2.tgz", + "integrity": "sha512-davCD2Zc80nzDVRwXTcQP/28fiJbcOwvdolL0sOiOsbwBa72kegmVU0Wrh1MYrbuCL98Omp5dVhQFWRKR2ZAlg==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "darwin" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/darwin-x64": { + "version": "0.27.2", + "resolved": "https://registry.npmjs.org/@esbuild/darwin-x64/-/darwin-x64-0.27.2.tgz", + "integrity": "sha512-ZxtijOmlQCBWGwbVmwOF/UCzuGIbUkqB1faQRf5akQmxRJ1ujusWsb3CVfk/9iZKr2L5SMU5wPBi1UWbvL+VQA==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "darwin" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/freebsd-arm64": { + "version": "0.27.2", + "resolved": "https://registry.npmjs.org/@esbuild/freebsd-arm64/-/freebsd-arm64-0.27.2.tgz", + "integrity": "sha512-lS/9CN+rgqQ9czogxlMcBMGd+l8Q3Nj1MFQwBZJyoEKI50XGxwuzznYdwcav6lpOGv5BqaZXqvBSiB/kJ5op+g==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "freebsd" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/freebsd-x64": { + "version": "0.27.2", + "resolved": "https://registry.npmjs.org/@esbuild/freebsd-x64/-/freebsd-x64-0.27.2.tgz", + "integrity": "sha512-tAfqtNYb4YgPnJlEFu4c212HYjQWSO/w/h/lQaBK7RbwGIkBOuNKQI9tqWzx7Wtp7bTPaGC6MJvWI608P3wXYA==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "freebsd" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/linux-arm": { + "version": "0.27.2", + "resolved": "https://registry.npmjs.org/@esbuild/linux-arm/-/linux-arm-0.27.2.tgz", + "integrity": "sha512-vWfq4GaIMP9AIe4yj1ZUW18RDhx6EPQKjwe7n8BbIecFtCQG4CfHGaHuh7fdfq+y3LIA2vGS/o9ZBGVxIDi9hw==", + "cpu": [ + "arm" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/linux-arm64": { + "version": "0.27.2", + "resolved": "https://registry.npmjs.org/@esbuild/linux-arm64/-/linux-arm64-0.27.2.tgz", + "integrity": "sha512-hYxN8pr66NsCCiRFkHUAsxylNOcAQaxSSkHMMjcpx0si13t1LHFphxJZUiGwojB1a/Hd5OiPIqDdXONia6bhTw==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/linux-ia32": { + "version": "0.27.2", + "resolved": "https://registry.npmjs.org/@esbuild/linux-ia32/-/linux-ia32-0.27.2.tgz", + "integrity": "sha512-MJt5BRRSScPDwG2hLelYhAAKh9imjHK5+NE/tvnRLbIqUWa+0E9N4WNMjmp/kXXPHZGqPLxggwVhz7QP8CTR8w==", + "cpu": [ + "ia32" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/linux-loong64": { + "version": "0.27.2", + "resolved": "https://registry.npmjs.org/@esbuild/linux-loong64/-/linux-loong64-0.27.2.tgz", + "integrity": "sha512-lugyF1atnAT463aO6KPshVCJK5NgRnU4yb3FUumyVz+cGvZbontBgzeGFO1nF+dPueHD367a2ZXe1NtUkAjOtg==", + "cpu": [ + "loong64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/linux-mips64el": { + "version": "0.27.2", + "resolved": "https://registry.npmjs.org/@esbuild/linux-mips64el/-/linux-mips64el-0.27.2.tgz", + "integrity": "sha512-nlP2I6ArEBewvJ2gjrrkESEZkB5mIoaTswuqNFRv/WYd+ATtUpe9Y09RnJvgvdag7he0OWgEZWhviS1OTOKixw==", + "cpu": [ + "mips64el" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/linux-ppc64": { + "version": "0.27.2", + "resolved": "https://registry.npmjs.org/@esbuild/linux-ppc64/-/linux-ppc64-0.27.2.tgz", + "integrity": "sha512-C92gnpey7tUQONqg1n6dKVbx3vphKtTHJaNG2Ok9lGwbZil6DrfyecMsp9CrmXGQJmZ7iiVXvvZH6Ml5hL6XdQ==", + "cpu": [ + "ppc64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/linux-riscv64": { + "version": "0.27.2", + "resolved": "https://registry.npmjs.org/@esbuild/linux-riscv64/-/linux-riscv64-0.27.2.tgz", + "integrity": "sha512-B5BOmojNtUyN8AXlK0QJyvjEZkWwy/FKvakkTDCziX95AowLZKR6aCDhG7LeF7uMCXEJqwa8Bejz5LTPYm8AvA==", + "cpu": [ + "riscv64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/linux-s390x": { + "version": "0.27.2", + "resolved": "https://registry.npmjs.org/@esbuild/linux-s390x/-/linux-s390x-0.27.2.tgz", + "integrity": "sha512-p4bm9+wsPwup5Z8f4EpfN63qNagQ47Ua2znaqGH6bqLlmJ4bx97Y9JdqxgGZ6Y8xVTixUnEkoKSHcpRlDnNr5w==", + "cpu": [ + "s390x" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/linux-x64": { + "version": "0.27.2", + "resolved": "https://registry.npmjs.org/@esbuild/linux-x64/-/linux-x64-0.27.2.tgz", + "integrity": "sha512-uwp2Tip5aPmH+NRUwTcfLb+W32WXjpFejTIOWZFw/v7/KnpCDKG66u4DLcurQpiYTiYwQ9B7KOeMJvLCu/OvbA==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "linux" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/netbsd-arm64": { + "version": "0.27.2", + "resolved": "https://registry.npmjs.org/@esbuild/netbsd-arm64/-/netbsd-arm64-0.27.2.tgz", + "integrity": "sha512-Kj6DiBlwXrPsCRDeRvGAUb/LNrBASrfqAIok+xB0LxK8CHqxZ037viF13ugfsIpePH93mX7xfJp97cyDuTZ3cw==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "netbsd" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/netbsd-x64": { + "version": "0.27.2", + "resolved": "https://registry.npmjs.org/@esbuild/netbsd-x64/-/netbsd-x64-0.27.2.tgz", + "integrity": "sha512-HwGDZ0VLVBY3Y+Nw0JexZy9o/nUAWq9MlV7cahpaXKW6TOzfVno3y3/M8Ga8u8Yr7GldLOov27xiCnqRZf0tCA==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "netbsd" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/openbsd-arm64": { + "version": "0.27.2", + "resolved": "https://registry.npmjs.org/@esbuild/openbsd-arm64/-/openbsd-arm64-0.27.2.tgz", + "integrity": "sha512-DNIHH2BPQ5551A7oSHD0CKbwIA/Ox7+78/AWkbS5QoRzaqlev2uFayfSxq68EkonB+IKjiuxBFoV8ESJy8bOHA==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "openbsd" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/openbsd-x64": { + "version": "0.27.2", + "resolved": "https://registry.npmjs.org/@esbuild/openbsd-x64/-/openbsd-x64-0.27.2.tgz", + "integrity": "sha512-/it7w9Nb7+0KFIzjalNJVR5bOzA9Vay+yIPLVHfIQYG/j+j9VTH84aNB8ExGKPU4AzfaEvN9/V4HV+F+vo8OEg==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "openbsd" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/openharmony-arm64": { + "version": "0.27.2", + "resolved": "https://registry.npmjs.org/@esbuild/openharmony-arm64/-/openharmony-arm64-0.27.2.tgz", + "integrity": "sha512-LRBbCmiU51IXfeXk59csuX/aSaToeG7w48nMwA6049Y4J4+VbWALAuXcs+qcD04rHDuSCSRKdmY63sruDS5qag==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "openharmony" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/sunos-x64": { + "version": "0.27.2", + "resolved": "https://registry.npmjs.org/@esbuild/sunos-x64/-/sunos-x64-0.27.2.tgz", + "integrity": "sha512-kMtx1yqJHTmqaqHPAzKCAkDaKsffmXkPHThSfRwZGyuqyIeBvf08KSsYXl+abf5HDAPMJIPnbBfXvP2ZC2TfHg==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "sunos" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/win32-arm64": { + "version": "0.27.2", + "resolved": "https://registry.npmjs.org/@esbuild/win32-arm64/-/win32-arm64-0.27.2.tgz", + "integrity": "sha512-Yaf78O/B3Kkh+nKABUF++bvJv5Ijoy9AN1ww904rOXZFLWVc5OLOfL56W+C8F9xn5JQZa3UX6m+IktJnIb1Jjg==", + "cpu": [ + "arm64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "win32" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/win32-ia32": { + "version": "0.27.2", + "resolved": "https://registry.npmjs.org/@esbuild/win32-ia32/-/win32-ia32-0.27.2.tgz", + "integrity": "sha512-Iuws0kxo4yusk7sw70Xa2E2imZU5HoixzxfGCdxwBdhiDgt9vX9VUCBhqcwY7/uh//78A1hMkkROMJq9l27oLQ==", + "cpu": [ + "ia32" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "win32" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/@esbuild/win32-x64": { + "version": "0.27.2", + "resolved": "https://registry.npmjs.org/@esbuild/win32-x64/-/win32-x64-0.27.2.tgz", + "integrity": "sha512-sRdU18mcKf7F+YgheI/zGf5alZatMUTKj/jNS6l744f9u3WFu4v7twcUI9vu4mknF4Y9aDlblIie0IM+5xxaqQ==", + "cpu": [ + "x64" + ], + "dev": true, + "license": "MIT", + "optional": true, + "os": [ + "win32" + ], + "engines": { + "node": ">=18" + } + }, + "node_modules/esbuild": { + "version": "0.27.2", + "resolved": "https://registry.npmjs.org/esbuild/-/esbuild-0.27.2.tgz", + "integrity": "sha512-HyNQImnsOC7X9PMNaCIeAm4ISCQXs5a5YasTXVliKv4uuBo1dKrG0A+uQS8M5eXjVMnLg3WgXaKvprHlFJQffw==", + "dev": true, + "hasInstallScript": true, + "license": "MIT", + "bin": { + "esbuild": "bin/esbuild" + }, + "engines": { + "node": ">=18" + }, + "optionalDependencies": { + "@esbuild/aix-ppc64": "0.27.2", + "@esbuild/android-arm": "0.27.2", + "@esbuild/android-arm64": "0.27.2", + "@esbuild/android-x64": "0.27.2", + "@esbuild/darwin-arm64": "0.27.2", + "@esbuild/darwin-x64": "0.27.2", + "@esbuild/freebsd-arm64": "0.27.2", + "@esbuild/freebsd-x64": "0.27.2", + "@esbuild/linux-arm": "0.27.2", + "@esbuild/linux-arm64": "0.27.2", + "@esbuild/linux-ia32": "0.27.2", + "@esbuild/linux-loong64": "0.27.2", + "@esbuild/linux-mips64el": "0.27.2", + "@esbuild/linux-ppc64": "0.27.2", + "@esbuild/linux-riscv64": "0.27.2", + "@esbuild/linux-s390x": "0.27.2", + "@esbuild/linux-x64": "0.27.2", + "@esbuild/netbsd-arm64": "0.27.2", + "@esbuild/netbsd-x64": "0.27.2", + "@esbuild/openbsd-arm64": "0.27.2", + "@esbuild/openbsd-x64": "0.27.2", + "@esbuild/openharmony-arm64": "0.27.2", + "@esbuild/sunos-x64": "0.27.2", + "@esbuild/win32-arm64": "0.27.2", + "@esbuild/win32-ia32": "0.27.2", + "@esbuild/win32-x64": "0.27.2" + } + } + } +} diff --git a/PlanTempus.Application/package.json b/PlanTempus.Application/package.json new file mode 100644 index 0000000..d4bea98 --- /dev/null +++ b/PlanTempus.Application/package.json @@ -0,0 +1,5 @@ +{ + "devDependencies": { + "esbuild": "^0.27.2" + } +} diff --git a/PlanTempus.Application/wwwroot/css/kasse.css b/PlanTempus.Application/wwwroot/css/kasse.css new file mode 100644 index 0000000..379dc1c --- /dev/null +++ b/PlanTempus.Application/wwwroot/css/kasse.css @@ -0,0 +1,900 @@ +/** + * Kasse (Cash Register) - Page Styling + * + * Filter bar, stats, table, forms, and difference box + */ + +/* =========================================== + STICKY HEADER CONTAINER + =========================================== */ +swp-kasse-sticky-header { + display: block; + position: sticky; + top: 0; + z-index: var(--z-sticky); + background: var(--color-surface); + overflow: visible; +} + +/* Override tab-bar sticky when inside sticky header */ +swp-kasse-sticky-header swp-tab-bar { + position: static; + top: auto; +} + +/* =========================================== + KASSE HEADER (Stats above tabs) + =========================================== */ +swp-kasse-header { + display: block; + background: var(--color-surface); + border-bottom: 1px solid var(--color-border); + padding: var(--spacing-10) var(--spacing-12); +} + +/* =========================================== + FILTER BAR + =========================================== */ +swp-filter-bar { + display: flex; + align-items: center; + gap: var(--spacing-8); + padding: var(--spacing-8) var(--spacing-10); + background: var(--color-surface); + border: 1px solid var(--color-border); + border-radius: var(--radius-lg); + margin-bottom: var(--spacing-10); + flex-wrap: wrap; +} + +swp-filter-group { + display: flex; + align-items: center; + gap: var(--spacing-4); +} + +swp-filter-label { + font-size: var(--font-size-sm); + font-weight: var(--font-weight-medium); + color: var(--color-text-secondary); +} + +swp-filter-bar input, +swp-filter-bar select { + padding: var(--spacing-4) var(--spacing-6); + font-size: var(--font-size-md); + font-family: var(--font-family); + border: 1px solid var(--color-border); + border-radius: var(--radius-md); + background: var(--color-surface); +} + +swp-filter-bar input:focus, +swp-filter-bar select:focus { + outline: none; + border-color: var(--color-teal); +} + +swp-filter-spacer { + flex: 1; +} + +/* =========================================== + KASSE STATS BAR + =========================================== */ +swp-kasse-stats { + display: grid; + grid-template-columns: repeat(4, 1fr); + gap: var(--spacing-8); + max-width: var(--page-max-width); + margin: 0 auto; +} + +swp-kasse-stats:not(.active) { + display: none; +} + +swp-kasse-stat { + background: var(--color-background-alt); + border-radius: var(--radius-lg); + padding: var(--spacing-6) var(--spacing-8); +} + +swp-kasse-stat-value { + display: block; + font-size: var(--font-size-2xl); + font-weight: var(--font-weight-semibold); + font-family: var(--font-mono); + color: var(--color-text); +} + +swp-kasse-stat-label { + display: block; + font-size: var(--font-size-sm); + color: var(--color-text-secondary); + margin-top: var(--spacing-2); +} + +swp-kasse-stat.highlight swp-kasse-stat-value { + color: var(--color-teal); +} + +swp-kasse-stat.warning swp-kasse-stat-value { + color: var(--color-amber); +} + +swp-kasse-stat.negative swp-kasse-stat-value { + color: var(--color-red); +} + +swp-kasse-stat.user swp-kasse-stat-value { + color: var(--color-blue); +} + +/* =========================================== + ACTION BAR (Table Header) + =========================================== */ +swp-action-bar { + display: flex; + align-items: center; + justify-content: space-between; + padding: var(--spacing-6) var(--spacing-8); + background: var(--color-surface); + border: 1px solid var(--color-border); + border-bottom: none; + border-radius: var(--radius-lg) var(--radius-lg) 0 0; +} + +swp-selection-info { + font-size: var(--font-size-md); + color: var(--color-text-secondary); +} + +/* =========================================== + KASSE TABLE (Grid + Subgrid pattern) + =========================================== */ +swp-kasse-table { + display: grid; + grid-template-columns: 50px 70px 60px minmax(140px, 1fr) 90px 100px 100px 110px 120px 40px; + background: var(--color-surface); + border: 1px solid var(--color-border); + border-radius: 0 0 var(--radius-lg) var(--radius-lg); + overflow: hidden; +} + +swp-kasse-table-header, +swp-kasse-table-body { + display: grid; + grid-column: 1 / -1; + grid-template-columns: subgrid; +} + +swp-kasse-table-header { + background: var(--color-background-alt); + border-bottom: 1px solid var(--color-border); + padding: var(--spacing-6) var(--spacing-10); + align-items: center; +} + +swp-kasse-table-row { + display: grid; + grid-column: 1 / -1; + grid-template-columns: subgrid; + align-items: center; +} + +swp-kasse-th { + font-size: var(--font-size-xs); + font-weight: var(--font-weight-semibold); + text-transform: uppercase; + letter-spacing: 0.5px; + color: var(--color-text-secondary); +} + +swp-kasse-th.right { + text-align: right; +} + +swp-kasse-th.checkbox, +swp-kasse-td.checkbox { + display: flex; + align-items: center; + justify-content: center; +} + +swp-kasse-table input[type="checkbox"] { + width: 16px; + height: 16px; + accent-color: var(--color-teal); + cursor: pointer; +} + +swp-kasse-table-row { + padding: var(--spacing-7) var(--spacing-10); + border-bottom: 1px solid var(--color-border); + cursor: pointer; + transition: background var(--transition-fast); +} + +swp-kasse-table-row:last-child { + border-bottom: none; +} + +swp-kasse-table-row:hover { + background: var(--color-background-hover); +} + +/* Draft row - clickable to go to Kasseafstemning */ +swp-kasse-table-row.draft-row { + background: color-mix(in srgb, var(--color-amber) 5%, transparent); + cursor: pointer; +} + +swp-kasse-table-row.draft-row:hover { + background: color-mix(in srgb, var(--color-amber) 12%, transparent); +} + +swp-kasse-td { + font-size: var(--font-size-base); + color: var(--color-text); +} + +swp-kasse-td.right { + text-align: right; +} + +swp-kasse-td.mono { + font-family: var(--font-mono); +} + +swp-kasse-td.muted { + color: var(--color-text-secondary); + font-size: var(--font-size-sm); +} + +swp-kasse-td.negative { + color: var(--color-red); +} + +swp-kasse-td.positive { + color: var(--color-green); +} + +swp-kasse-td.id { + font-size: var(--font-size-sm); + color: var(--color-text-secondary); + font-family: var(--font-mono); +} + +swp-period-cell { + display: block; +} + +swp-period-cell .dates { + font-size: var(--font-size-sm); + color: var(--color-text-secondary); +} + +/* =========================================== + ROW TOGGLE & EXPANDABLE DETAIL + =========================================== */ +swp-row-toggle { + display: flex; + align-items: center; + justify-content: center; + width: 32px; + height: 32px; + border-radius: var(--radius-md); + color: var(--color-text-secondary); + cursor: pointer; + transition: all var(--transition-fast); +} + +swp-row-toggle:hover { + background: var(--color-background-alt); + color: var(--color-text); +} + +swp-row-toggle i { + font-size: var(--font-size-lg); + transition: transform var(--transition-fast); +} + +/* Row detail - hidden by default */ +swp-kasse-row-detail { + grid-column: 1 / -1; + display: none; + overflow: hidden; + background: var(--color-background-alt); + border-bottom: 1px solid var(--color-border); +} + +swp-kasse-row-detail.expanded { + display: block; +} + +swp-row-detail-content { + display: block; + padding: var(--spacing-8) var(--spacing-10); +} + +swp-row-detail-actions { + display: flex; + gap: var(--spacing-4); + justify-content: flex-end; +} + +/* Legacy support */ +swp-row-arrow { + display: flex; + align-items: center; + justify-content: flex-end; + color: var(--color-text-secondary); +} + +swp-row-arrow i { + font-size: var(--font-size-lg); +} + +swp-kasse-table-footer { + grid-column: 1 / -1; + display: flex; + align-items: center; + justify-content: space-between; + padding: var(--spacing-7) var(--spacing-10); + background: var(--color-background-alt); + border-top: 1px solid var(--color-border); + font-size: var(--font-size-md); + color: var(--color-text-secondary); +} + +/* =========================================== + STATUS BADGE + =========================================== */ +/* Center status column */ +swp-kasse-th:nth-child(9), +swp-kasse-td:nth-child(9) { + text-align: center; +} + +swp-status-badge { + display: inline-flex; + align-items: center; + gap: var(--spacing-3); + padding: var(--spacing-2) var(--spacing-5); + font-size: var(--font-size-xs); + font-weight: var(--font-weight-medium); + border-radius: var(--radius-pill); +} + +swp-status-badge::before { + content: ''; + width: 6px; + height: 6px; + border-radius: var(--radius-full); + background: currentColor; +} + +swp-status-badge.approved { + background: color-mix(in srgb, var(--color-green) 15%, transparent); + color: var(--color-green); +} + +swp-status-badge.draft { + background: color-mix(in srgb, var(--color-amber) 15%, transparent); + color: #b45309; +} + +/* =========================================== + TWO-COLUMN GRID (Detail View) + =========================================== */ +swp-kasse-grid { + display: grid; + grid-template-columns: 1fr 1fr; + gap: var(--spacing-12); +} + +@media (max-width: 900px) { + swp-kasse-grid { + grid-template-columns: 1fr; + } +} + +swp-kasse-column { + display: grid; + gap: var(--spacing-10); + align-content: start; +} + +/* =========================================== + DATA TABLE (Dagens Tal) + =========================================== */ +swp-data-table { + display: block; + width: 100%; +} + +swp-data-header { + display: grid; + grid-template-columns: 1fr 100px 140px; + gap: var(--spacing-6); + padding: var(--spacing-5) 0; + border-bottom: 2px solid var(--color-border); +} + +swp-data-header span { + font-size: var(--font-size-xs); + font-weight: var(--font-weight-semibold); + text-transform: uppercase; + letter-spacing: 0.5px; + color: var(--color-text-secondary); +} + +swp-data-header span:not(:first-child) { + text-align: right; +} + +swp-data-row { + display: grid; + grid-template-columns: 1fr 100px 140px; + gap: var(--spacing-6); + padding: var(--spacing-7) 0; + border-bottom: 1px solid var(--color-border); + align-items: center; +} + +swp-data-row:last-child { + border-bottom: none; +} + +swp-data-label { + font-size: var(--font-size-base); + color: var(--color-text); +} + +swp-data-system { + text-align: right; + font-family: var(--font-mono); + font-size: var(--font-size-base); + color: var(--color-text-secondary); +} + +swp-data-input input { + width: 100%; + padding: var(--spacing-4) var(--spacing-5); + font-size: var(--font-size-base); + font-family: var(--font-mono); + text-align: right; + border: 1px solid var(--color-border); + border-radius: var(--radius-md); + background: var(--color-surface); +} + +swp-data-input input:focus { + outline: none; + border-color: var(--color-teal); +} + +swp-data-input input::placeholder { + color: var(--color-text-muted); + font-family: var(--font-family); + font-size: var(--font-size-sm); +} + +swp-data-value { + text-align: right; + font-family: var(--font-mono); + font-size: var(--font-size-base); + color: var(--color-text); +} + +swp-data-value.muted { + color: var(--color-text-secondary); +} + +swp-table-note { + display: block; + font-size: var(--font-size-sm); + color: var(--color-text-secondary); + margin-top: var(--spacing-8); + padding: var(--spacing-6); + background: var(--color-background-alt); + border-radius: var(--radius-md); +} + +/* =========================================== + CALC ROW (Kontanter) + =========================================== */ +swp-calc-row { + display: flex; + justify-content: space-between; + align-items: center; + padding: var(--spacing-7) 0; + border-bottom: 1px solid var(--color-border); +} + +swp-calc-row:last-of-type { + border-bottom: none; +} + +swp-calc-row.input-row { + background: var(--color-background-alt); + margin: var(--spacing-8) calc(-1 * var(--spacing-5)) calc(-1 * var(--spacing-5)); + padding: var(--spacing-8) var(--spacing-5); + border-radius: 0 0 var(--border-radius-lg) var(--border-radius-lg); +} + +swp-calc-label span { + display: block; + font-size: var(--font-size-base); + color: var(--color-text); +} + +swp-calc-label small { + display: block; + font-size: var(--font-size-sm); + color: var(--color-text-secondary); + margin-top: var(--spacing-1); +} + +swp-calc-value { + font-family: var(--font-mono); + font-size: var(--font-size-base); + color: var(--color-text); +} + +swp-calc-value.muted { + color: var(--color-text-secondary); +} + +swp-calc-input input { + width: 140px; + padding: var(--spacing-6) var(--spacing-7); + font-size: var(--font-size-lg); + font-family: var(--font-mono); + text-align: right; + border: 2px solid var(--color-border); + border-radius: var(--radius-md); + background: var(--color-surface); +} + +swp-calc-input input:focus { + outline: none; + border-color: var(--color-teal); +} + +/* =========================================== + DIFFERENCE BOX + =========================================== */ +swp-difference-box { + display: flex; + align-items: center; + justify-content: space-between; + padding: var(--spacing-10); + border-radius: var(--radius-lg); + background: var(--color-background-alt); +} + +swp-difference-box.positive { + background: color-mix(in srgb, var(--color-green) 10%, transparent); +} + +swp-difference-box.negative { + background: color-mix(in srgb, var(--color-red) 10%, transparent); +} + +swp-difference-box.neutral { + background: color-mix(in srgb, var(--color-teal) 10%, transparent); +} + +swp-difference-label { + font-size: var(--font-size-base); + font-weight: var(--font-weight-medium); + color: var(--color-text); +} + +swp-difference-label small { + display: block; + font-size: var(--font-size-sm); + font-weight: var(--font-weight-regular); + color: var(--color-text-secondary); + margin-top: var(--spacing-2); +} + +swp-difference-value { + font-size: var(--font-size-2xl); + font-weight: var(--font-weight-semibold); + font-family: var(--font-mono); +} + +swp-difference-box.positive swp-difference-value { + color: var(--color-green); +} + +swp-difference-box.negative swp-difference-value { + color: var(--color-red); +} + +swp-difference-box.neutral swp-difference-value { + color: var(--color-teal); +} + +/* =========================================== + PERIOD DISPLAY + =========================================== */ +swp-period-display { + display: block; + padding: var(--spacing-8); + background: var(--color-background-alt); + border-radius: var(--radius-lg); +} + +swp-period-label { + display: block; + font-size: var(--font-size-xs); + font-weight: var(--font-weight-semibold); + text-transform: uppercase; + letter-spacing: 0.5px; + color: var(--color-text-secondary); + margin-bottom: var(--spacing-4); +} + +swp-period-value { + display: flex; + align-items: center; + gap: var(--spacing-5); + font-size: var(--font-size-base); + font-weight: var(--font-weight-medium); + color: var(--color-text); +} + +swp-period-value .arrow { + color: var(--color-teal); + font-weight: var(--font-weight-regular); +} + +/* =========================================== + FORM ELEMENTS + =========================================== */ +swp-form-grid { + display: grid; + grid-template-columns: 1fr 1fr; + gap: var(--spacing-8); +} + +swp-form-field { + display: block; +} + +swp-form-field.full-width { + grid-column: 1 / -1; +} + +swp-form-label { + display: block; + font-size: var(--font-size-sm); + font-weight: var(--font-weight-medium); + color: var(--color-text-secondary); + margin-bottom: var(--spacing-3); + text-transform: uppercase; + letter-spacing: 0.3px; +} + +swp-form-label .required { + color: var(--color-red); +} + +swp-form-input input, +swp-form-input select { + width: 100%; + padding: var(--spacing-5) var(--spacing-6); + font-size: var(--font-size-base); + font-family: var(--font-family); + border: 1px solid var(--color-border); + border-radius: var(--radius-md); + background: var(--color-surface); + transition: border-color var(--transition-fast); +} + +swp-form-input input:focus, +swp-form-input select:focus { + outline: none; + border-color: var(--color-teal); +} + +swp-auto-id { + display: block; + font-size: var(--font-size-sm); + color: var(--color-text-secondary); + margin-top: var(--spacing-8); + padding-top: var(--spacing-8); + border-top: 1px solid var(--color-border); +} + +/* =========================================== + NOTE FIELD + =========================================== */ +swp-note-field textarea { + width: 100%; + min-height: 80px; + padding: var(--spacing-6); + font-size: var(--font-size-base); + font-family: var(--font-family); + border: 1px solid var(--color-border); + border-radius: var(--radius-md); + resize: vertical; +} + +swp-note-field textarea:focus { + outline: none; + border-color: var(--color-teal); +} + +swp-note-hint { + display: block; + font-size: var(--font-size-sm); + color: var(--color-text-secondary); + margin-top: var(--spacing-4); +} + +/* =========================================== + APPROVAL SECTION + =========================================== */ +swp-status-row { + display: flex; + align-items: center; + gap: var(--spacing-6); +} + +swp-approval-grid { + display: grid; + grid-template-columns: 1fr 1fr; + gap: var(--spacing-8); +} + +swp-checkbox-field { + display: flex; + align-items: flex-start; + gap: var(--spacing-6); + padding: var(--spacing-8); + background: var(--color-background-alt); + border-radius: var(--radius-lg); + grid-column: 1 / -1; +} + +swp-checkbox-field input[type="checkbox"] { + width: 18px; + height: 18px; + margin-top: var(--spacing-1); + accent-color: var(--color-teal); + cursor: pointer; +} + +swp-checkbox-field label { + font-size: var(--font-size-md); + color: var(--color-text); + cursor: pointer; + line-height: var(--line-height-normal); +} + +/* =========================================== + CARD FOOTER (Actions) + =========================================== */ +swp-card-footer { + display: flex; + justify-content: space-between; + align-items: center; + padding: var(--spacing-8) var(--spacing-10); + background: var(--color-background-alt); + border-top: 1px solid var(--color-border); + margin: var(--spacing-10) calc(-1 * var(--spacing-5)) calc(-1 * var(--spacing-5)); + border-radius: 0 0 var(--border-radius-lg) var(--border-radius-lg); +} + +swp-actions-right { + display: flex; + gap: var(--spacing-5); +} + +/* =========================================== + BUTTONS + =========================================== */ +swp-btn { + display: inline-flex; + align-items: center; + gap: var(--spacing-3); + padding: var(--spacing-5) var(--spacing-8); + font-size: var(--font-size-md); + font-weight: var(--font-weight-medium); + font-family: var(--font-family); + border-radius: var(--radius-md); + cursor: pointer; + transition: all var(--transition-fast); + border: none; +} + +swp-btn i { + font-size: var(--font-size-lg); +} + +swp-btn.primary { + background: var(--color-teal); + color: white; +} + +swp-btn.primary:hover { + opacity: 0.9; +} + +swp-btn.primary:disabled { + background: var(--color-border); + cursor: not-allowed; +} + +swp-btn.secondary { + background: var(--color-surface); + border: 1px solid var(--color-border); + color: var(--color-text); +} + +swp-btn.secondary:hover { + background: var(--color-background-hover); +} + +swp-btn.ghost { + background: transparent; + color: var(--color-text-secondary); +} + +swp-btn.ghost:hover { + color: var(--color-text); +} + +/* =========================================== + SYSTEM NOTE + =========================================== */ +swp-system-note { + display: block; + font-size: var(--font-size-xs); + color: var(--color-text-secondary); + text-align: center; + padding: var(--spacing-6); + margin-top: var(--spacing-8); +} + +/* =========================================== + RESPONSIVE + =========================================== */ +@media (max-width: 1000px) { + swp-kasse-stats { + grid-template-columns: repeat(2, 1fr); + } + + /* Table columns defined on parent - subgrid inherits */ + swp-kasse-table { + grid-template-columns: 50px 80px 1fr 100px 110px 120px 40px; + } + + /* Hide some columns on smaller screens */ + swp-kasse-th:nth-child(3), + swp-kasse-td:nth-child(3), + swp-kasse-th:nth-child(6), + swp-kasse-td:nth-child(6) { + display: none; + } +} + +@media (max-width: 768px) { + swp-filter-bar { + flex-direction: column; + align-items: stretch; + } + + swp-filter-group { + width: 100%; + } + + swp-filter-spacer { + display: none; + } +} diff --git a/PlanTempus.Application/wwwroot/css/tabs.css b/PlanTempus.Application/wwwroot/css/tabs.css new file mode 100644 index 0000000..8795ad5 --- /dev/null +++ b/PlanTempus.Application/wwwroot/css/tabs.css @@ -0,0 +1,94 @@ +/** + * Tabs - Tab Bar Navigation + * + * Horizontal tab bar with underline active state + * Based on POC: poc-indstillinger.html + */ + +/* =========================================== + TAB BAR + =========================================== */ +swp-tab-bar { + display: flex; + gap: 0; + background: var(--color-surface); + border-bottom: 1px solid var(--color-border); + padding: 0 var(--spacing-12); + position: sticky; + top: var(--topbar-height); + z-index: var(--z-sticky); +} + +/* Account for demo banner if present */ +body.has-demo-banner swp-tab-bar { + top: calc(var(--topbar-height) + 40px); +} + +/* =========================================== + TAB ITEM + =========================================== */ +swp-tab { + display: flex; + align-items: center; + gap: var(--spacing-4); + padding: var(--spacing-7) var(--spacing-12); + font-size: var(--font-size-base); + font-weight: var(--font-weight-medium); + color: var(--color-text-secondary); + cursor: pointer; + border-bottom: 2px solid transparent; + margin-bottom: -1px; + transition: all var(--transition-fast); +} + +swp-tab i { + font-size: var(--font-size-xl); +} + +swp-tab:hover { + color: var(--color-text); + background: var(--color-background-alt); +} + +swp-tab.active { + color: var(--color-teal); + border-bottom-color: var(--color-teal); +} + +swp-tab.active:hover { + background: transparent; +} + +/* =========================================== + TAB CONTENT + =========================================== */ +swp-tab-content { + display: none; +} + +swp-tab-content.active { + display: block; +} + +/* =========================================== + TAB WITH BADGE + =========================================== */ +swp-tab swp-badge { + display: inline-flex; + align-items: center; + justify-content: center; + min-width: 20px; + height: 20px; + padding: 0 var(--spacing-2); + margin-left: var(--spacing-2); + font-size: var(--font-size-xs); + font-weight: var(--font-weight-semibold); + border-radius: var(--radius-full); + background: var(--color-background-alt); + color: var(--color-text-secondary); +} + +swp-tab.active swp-badge { + background: color-mix(in srgb, var(--color-teal) 15%, transparent); + color: var(--color-teal); +} diff --git a/PlanTempus.Application/wwwroot/js/app.js b/PlanTempus.Application/wwwroot/js/app.js index 7aed0c8..e1af9ff 100644 --- a/PlanTempus.Application/wwwroot/js/app.js +++ b/PlanTempus.Application/wwwroot/js/app.js @@ -1,5 +1,8 @@ -// modules/sidebar.ts -var SidebarController = class { +var __defProp = Object.defineProperty; +var __name = (target, value) => __defProp(target, "name", { value, configurable: true }); + +// wwwroot/ts/modules/sidebar.ts +var _SidebarController = class _SidebarController { constructor() { this.menuToggle = null; this.appLayout = null; @@ -71,9 +74,11 @@ var SidebarController = class { } } }; +__name(_SidebarController, "SidebarController"); +var SidebarController = _SidebarController; -// modules/drawers.ts -var DrawerController = class { +// wwwroot/ts/modules/drawers.ts +var _DrawerController = class _DrawerController { constructor() { this.profileDrawer = null; this.notificationDrawer = null; @@ -287,18 +292,11 @@ var DrawerController = class { }); } }; +__name(_DrawerController, "DrawerController"); +var DrawerController = _DrawerController; -// modules/theme.ts -var ThemeController = class _ThemeController { - static { - this.STORAGE_KEY = "theme-preference"; - } - static { - this.DARK_CLASS = "dark-mode"; - } - static { - this.LIGHT_CLASS = "light-mode"; - } +// wwwroot/ts/modules/theme.ts +var _ThemeController = class _ThemeController { constructor() { this.root = document.documentElement; this.themeOptions = document.querySelectorAll("swp-theme-option"); @@ -381,9 +379,14 @@ var ThemeController = class _ThemeController { } } }; +__name(_ThemeController, "ThemeController"); +_ThemeController.STORAGE_KEY = "theme-preference"; +_ThemeController.DARK_CLASS = "dark-mode"; +_ThemeController.LIGHT_CLASS = "light-mode"; +var ThemeController = _ThemeController; -// modules/search.ts -var SearchController = class { +// wwwroot/ts/modules/search.ts +var _SearchController = class _SearchController { constructor() { this.input = null; this.container = null; @@ -459,9 +462,11 @@ var SearchController = class { })); } }; +__name(_SearchController, "SearchController"); +var SearchController = _SearchController; -// modules/lockscreen.ts -var LockScreenController = class _LockScreenController { +// wwwroot/ts/modules/lockscreen.ts +var _LockScreenController = class _LockScreenController { constructor(drawers) { // Demo PIN this.lockScreen = null; @@ -479,9 +484,6 @@ var LockScreenController = class _LockScreenController { this.pinDigits = this.pinInput?.querySelectorAll("swp-pin-digit") ?? null; this.setupListeners(); } - static { - this.CORRECT_PIN = "1234"; - } /** * Check if lock screen is active */ @@ -597,17 +599,312 @@ var LockScreenController = class _LockScreenController { } } }; +__name(_LockScreenController, "LockScreenController"); +_LockScreenController.CORRECT_PIN = "1234"; +var LockScreenController = _LockScreenController; -// app.ts -var App = class { +// wwwroot/ts/modules/kasse.ts +var _KasseController = class _KasseController { + constructor() { + // Base values (from system - would come from server in real app) + this.startBalance = 2e3; + this.cashSales = 3540; + this.setupTabs(); + this.setupCashCalculation(); + this.setupCheckboxSelection(); + this.setupApprovalCheckbox(); + this.setupDateFilters(); + this.setupRowToggle(); + this.setupDraftRowClick(); + } + /** + * Setup tab switching functionality + */ + setupTabs() { + const tabs = document.querySelectorAll("swp-tab[data-tab]"); + tabs.forEach((tab) => { + tab.addEventListener("click", () => { + const targetTab = tab.dataset.tab; + if (targetTab) { + this.switchToTab(targetTab); + } + }); + }); + } + /** + * Switch to a specific tab by name + */ + switchToTab(targetTab) { + const tabs = document.querySelectorAll("swp-tab[data-tab]"); + const contents = document.querySelectorAll("swp-tab-content[data-tab]"); + const statsBars = document.querySelectorAll("swp-kasse-stats[data-for-tab]"); + tabs.forEach((t) => { + if (t.dataset.tab === targetTab) { + t.classList.add("active"); + } else { + t.classList.remove("active"); + } + }); + contents.forEach((content) => { + if (content.dataset.tab === targetTab) { + content.classList.add("active"); + } else { + content.classList.remove("active"); + } + }); + statsBars.forEach((stats) => { + if (stats.dataset.forTab === targetTab) { + stats.classList.add("active"); + } else { + stats.classList.remove("active"); + } + }); + } + /** + * Setup cash calculation with real-time updates + */ + setupCashCalculation() { + const payoutsInput = document.getElementById("payouts"); + const toBankInput = document.getElementById("toBank"); + const actualCashInput = document.getElementById("actualCash"); + if (!payoutsInput || !toBankInput || !actualCashInput) return; + const calculate = /* @__PURE__ */ __name(() => this.calculateCash(payoutsInput, toBankInput, actualCashInput), "calculate"); + payoutsInput.addEventListener("input", calculate); + toBankInput.addEventListener("input", calculate); + actualCashInput.addEventListener("input", calculate); + calculate(); + } + /** + * Calculate expected cash and difference + */ + calculateCash(payoutsInput, toBankInput, actualCashInput) { + const payouts = this.parseNumber(payoutsInput.value); + const toBank = this.parseNumber(toBankInput.value); + const actual = this.parseNumber(actualCashInput.value); + const expectedCash = this.startBalance + this.cashSales - payouts - toBank; + const expectedElement = document.getElementById("expectedCash"); + if (expectedElement) { + expectedElement.textContent = this.formatNumber(expectedCash); + } + this.updateDifference(actual, expectedCash, actualCashInput.value); + } + /** + * Update difference box with color coding + */ + updateDifference(actual, expected, rawValue) { + const box = document.getElementById("differenceBox"); + const value = document.getElementById("differenceValue"); + if (!box || !value) return; + const diff = actual - expected; + box.classList.remove("positive", "negative", "neutral"); + if (actual === 0 && rawValue === "") { + value.textContent = "\u2013 kr"; + box.classList.add("neutral"); + } else if (diff > 0) { + value.textContent = "+" + this.formatNumber(diff) + " kr"; + box.classList.add("positive"); + } else if (diff < 0) { + value.textContent = this.formatNumber(diff) + " kr"; + box.classList.add("negative"); + } else { + value.textContent = "0,00 kr"; + box.classList.add("neutral"); + } + } + /** + * Setup checkbox selection for table rows + */ + setupCheckboxSelection() { + const selectAll = document.getElementById("selectAll"); + const rowCheckboxes = document.querySelectorAll(".row-select"); + const exportBtn = document.getElementById("exportBtn"); + const selectionCount = document.getElementById("selectionCount"); + if (!selectAll || !exportBtn || !selectionCount) return; + const updateSelection = /* @__PURE__ */ __name(() => { + const checked = document.querySelectorAll(".row-select:checked"); + const count = checked.length; + selectionCount.textContent = count === 0 ? "0 valgt" : `${count} valgt`; + exportBtn.disabled = count === 0; + selectAll.checked = count === rowCheckboxes.length && count > 0; + selectAll.indeterminate = count > 0 && count < rowCheckboxes.length; + }, "updateSelection"); + selectAll.addEventListener("change", () => { + rowCheckboxes.forEach((cb) => cb.checked = selectAll.checked); + updateSelection(); + }); + rowCheckboxes.forEach((cb) => { + cb.addEventListener("change", updateSelection); + cb.addEventListener("click", (e) => e.stopPropagation()); + }); + } + /** + * Setup approval checkbox to enable/disable approve button + */ + setupApprovalCheckbox() { + const checkbox = document.getElementById("confirmCheckbox"); + const approveBtn = document.getElementById("approveBtn"); + if (!checkbox || !approveBtn) return; + checkbox.addEventListener("change", () => { + approveBtn.disabled = !checkbox.checked; + }); + } + /** + * Setup date filter defaults (last 30 days) + */ + setupDateFilters() { + const dateFrom = document.getElementById("dateFrom"); + const dateTo = document.getElementById("dateTo"); + if (!dateFrom || !dateTo) return; + const today = /* @__PURE__ */ new Date(); + const thirtyDaysAgo = new Date(today); + thirtyDaysAgo.setDate(today.getDate() - 30); + dateTo.value = this.formatDateISO(today); + dateFrom.value = this.formatDateISO(thirtyDaysAgo); + } + /** + * Format number as Danish currency + */ + formatNumber(num) { + return num.toLocaleString("da-DK", { + minimumFractionDigits: 2, + maximumFractionDigits: 2 + }); + } + /** + * Parse Danish number format + */ + parseNumber(str) { + if (!str) return 0; + return parseFloat(str.replace(/\./g, "").replace(",", ".")) || 0; + } + /** + * Format date as ISO string (YYYY-MM-DD) + */ + formatDateISO(date) { + return date.toISOString().split("T")[0]; + } + /** + * Setup row toggle for expandable details + */ + setupRowToggle() { + const rows = document.querySelectorAll("swp-kasse-table-row[data-id]:not(.draft-row)"); + rows.forEach((row) => { + const rowId = row.getAttribute("data-id"); + if (!rowId) return; + const detail = document.querySelector(`swp-kasse-row-detail[data-for="${rowId}"]`); + if (!detail) return; + row.addEventListener("click", (e) => { + if (e.target.closest('input[type="checkbox"]')) return; + const icon = row.querySelector("swp-row-toggle i"); + const isExpanded = row.classList.contains("expanded"); + document.querySelectorAll("swp-kasse-table-row.expanded").forEach((r) => { + if (r !== row) { + const otherId = r.getAttribute("data-id"); + if (otherId) { + const otherDetail = document.querySelector(`swp-kasse-row-detail[data-for="${otherId}"]`); + const otherIcon = r.querySelector("swp-row-toggle i"); + if (otherDetail && otherIcon) { + this.collapseRow(r, otherDetail, otherIcon); + } + } + } + }); + if (isExpanded) { + this.collapseRow(row, detail, icon); + } else { + this.expandRow(row, detail, icon); + } + }); + }); + } + /** + * Expand a row with animation + */ + expandRow(row, detail, icon) { + row.classList.add("expanded"); + detail.classList.add("expanded"); + icon?.animate([ + { transform: "rotate(0deg)" }, + { transform: "rotate(90deg)" } + ], { + duration: 200, + easing: "ease-out", + fill: "forwards" + }); + const content = detail.querySelector("swp-row-detail-content"); + if (content) { + const height = content.offsetHeight; + detail.animate([ + { height: "0px", opacity: 0 }, + { height: `${height}px`, opacity: 1 } + ], { + duration: 250, + easing: "ease-out", + fill: "forwards" + }); + } + } + /** + * Collapse a row with animation + */ + collapseRow(row, detail, icon) { + icon?.animate([ + { transform: "rotate(90deg)" }, + { transform: "rotate(0deg)" } + ], { + duration: 200, + easing: "ease-out", + fill: "forwards" + }); + const content = detail.querySelector("swp-row-detail-content"); + if (content) { + const height = content.offsetHeight; + const animation = detail.animate([ + { height: `${height}px`, opacity: 1 }, + { height: "0px", opacity: 0 } + ], { + duration: 200, + easing: "ease-out", + fill: "forwards" + }); + animation.onfinish = () => { + row.classList.remove("expanded"); + detail.classList.remove("expanded"); + }; + } else { + row.classList.remove("expanded"); + detail.classList.remove("expanded"); + } + } + /** + * Setup draft row click to navigate to Kasseafstemning tab + */ + setupDraftRowClick() { + const draftRow = document.querySelector("swp-kasse-table-row.draft-row"); + if (!draftRow) return; + draftRow.style.cursor = "pointer"; + draftRow.addEventListener("click", (e) => { + if (e.target.closest('input[type="checkbox"]')) return; + this.switchToTab("afstemning"); + }); + } +}; +__name(_KasseController, "KasseController"); +var KasseController = _KasseController; + +// wwwroot/ts/app.ts +var _App = class _App { constructor() { this.sidebar = new SidebarController(); this.drawers = new DrawerController(); this.theme = new ThemeController(); this.search = new SearchController(); this.lockScreen = new LockScreenController(this.drawers); + this.kasse = new KasseController(); } }; +__name(_App, "App"); +var App = _App; var app; function init() { app = new App(); @@ -615,6 +912,7 @@ function init() { window.app = app; } } +__name(init, "init"); if (document.readyState === "loading") { document.addEventListener("DOMContentLoaded", init); } else { @@ -626,4 +924,4 @@ export { app, app_default as default }; -//# sourceMappingURL=app.js.map +//# sourceMappingURL=data:application/json;base64,ewogICJ2ZXJzaW9uIjogMywKICAic291cmNlcyI6IFsiLi4vdHMvbW9kdWxlcy9zaWRlYmFyLnRzIiwgIi4uL3RzL21vZHVsZXMvZHJhd2Vycy50cyIsICIuLi90cy9tb2R1bGVzL3RoZW1lLnRzIiwgIi4uL3RzL21vZHVsZXMvc2VhcmNoLnRzIiwgIi4uL3RzL21vZHVsZXMvbG9ja3NjcmVlbi50cyIsICIuLi90cy9tb2R1bGVzL2thc3NlLnRzIiwgIi4uL3RzL2FwcC50cyJdLAogICJzb3VyY2VzQ29udGVudCI6IFsiLyoqXG4gKiBTaWRlYmFyIENvbnRyb2xsZXJcbiAqXG4gKiBIYW5kbGVzIHNpZGViYXIgY29sbGFwc2UvZXhwYW5kIGFuZCB0b29sdGlwIGZ1bmN0aW9uYWxpdHlcbiAqL1xuXG5leHBvcnQgY2xhc3MgU2lkZWJhckNvbnRyb2xsZXIge1xuICBwcml2YXRlIG1lbnVUb2dnbGU6IEhUTUxFbGVtZW50IHwgbnVsbCA9IG51bGw7XG4gIHByaXZhdGUgYXBwTGF5b3V0OiBIVE1MRWxlbWVudCB8IG51bGwgPSBudWxsO1xuICBwcml2YXRlIG1lbnVUb29sdGlwOiBIVE1MRWxlbWVudCB8IG51bGwgPSBudWxsO1xuXG4gIGNvbnN0cnVjdG9yKCkge1xuICAgIHRoaXMubWVudVRvZ2dsZSA9IGRvY3VtZW50LmdldEVsZW1lbnRCeUlkKCdtZW51VG9nZ2xlJyk7XG4gICAgdGhpcy5hcHBMYXlvdXQgPSBkb2N1bWVudC5xdWVyeVNlbGVjdG9yKCdzd3AtYXBwLWxheW91dCcpO1xuICAgIHRoaXMubWVudVRvb2x0aXAgPSBkb2N1bWVudC5nZXRFbGVtZW50QnlJZCgnbWVudVRvb2x0aXAnKTtcblxuICAgIHRoaXMuc2V0dXBMaXN0ZW5lcnMoKTtcbiAgICB0aGlzLnNldHVwVG9vbHRpcHMoKTtcbiAgICB0aGlzLnJlc3RvcmVTdGF0ZSgpO1xuICB9XG5cbiAgLyoqXG4gICAqIENoZWNrIGlmIHNpZGViYXIgaXMgY29sbGFwc2VkXG4gICAqL1xuICBnZXQgaXNDb2xsYXBzZWQoKTogYm9vbGVhbiB7XG4gICAgcmV0dXJuIHRoaXMuYXBwTGF5b3V0Py5jbGFzc0xpc3QuY29udGFpbnMoJ21lbnUtY29sbGFwc2VkJykgPz8gZmFsc2U7XG4gIH1cblxuICAvKipcbiAgICogVG9nZ2xlIHNpZGViYXIgY29sbGFwc2VkIHN0YXRlXG4gICAqL1xuICB0b2dnbGUoKTogdm9pZCB7XG4gICAgaWYgKCF0aGlzLmFwcExheW91dCkgcmV0dXJuO1xuXG4gICAgdGhpcy5hcHBMYXlvdXQuY2xhc3NMaXN0LnRvZ2dsZSgnbWVudS1jb2xsYXBzZWQnKTtcbiAgICBsb2NhbFN0b3JhZ2Uuc2V0SXRlbSgnc2lkZWJhci1jb2xsYXBzZWQnLCBTdHJpbmcodGhpcy5pc0NvbGxhcHNlZCkpO1xuICB9XG5cbiAgLyoqXG4gICAqIENvbGxhcHNlIHRoZSBzaWRlYmFyXG4gICAqL1xuICBjb2xsYXBzZSgpOiB2b2lkIHtcbiAgICB0aGlzLmFwcExheW91dD8uY2xhc3NMaXN0LmFkZCgnbWVudS1jb2xsYXBzZWQnKTtcbiAgICBsb2NhbFN0b3JhZ2Uuc2V0SXRlbSgnc2lkZWJhci1jb2xsYXBzZWQnLCAndHJ1ZScpO1xuICB9XG5cbiAgLyoqXG4gICAqIEV4cGFuZCB0aGUgc2lkZWJhclxuICAgKi9cbiAgZXhwYW5kKCk6IHZvaWQge1xuICAgIHRoaXMuYXBwTGF5b3V0Py5jbGFzc0xpc3QucmVtb3ZlKCdtZW51LWNvbGxhcHNlZCcpO1xuICAgIGxvY2FsU3RvcmFnZS5zZXRJdGVtKCdzaWRlYmFyLWNvbGxhcHNlZCcsICdmYWxzZScpO1xuICB9XG5cbiAgcHJpdmF0ZSBzZXR1cExpc3RlbmVycygpOiB2b2lkIHtcbiAgICB0aGlzLm1lbnVUb2dnbGU/LmFkZEV2ZW50TGlzdGVuZXIoJ2NsaWNrJywgKCkgPT4gdGhpcy50b2dnbGUoKSk7XG4gIH1cblxuICBwcml2YXRlIHNldHVwVG9vbHRpcHMoKTogdm9pZCB7XG4gICAgaWYgKCF0aGlzLm1lbnVUb29sdGlwKSByZXR1cm47XG5cbiAgICBjb25zdCBtZW51SXRlbXMgPSBkb2N1bWVudC5xdWVyeVNlbGVjdG9yQWxsPEhUTUxFbGVtZW50Pignc3dwLXNpZGUtbWVudS1pdGVtW2RhdGEtdG9vbHRpcF0nKTtcblxuICAgIG1lbnVJdGVtcy5mb3JFYWNoKGl0ZW0gPT4ge1xuICAgICAgaXRlbS5hZGRFdmVudExpc3RlbmVyKCdtb3VzZWVudGVyJywgKCkgPT4gdGhpcy5zaG93VG9vbHRpcChpdGVtKSk7XG4gICAgICBpdGVtLmFkZEV2ZW50TGlzdGVuZXIoJ21vdXNlbGVhdmUnLCAoKSA9PiB0aGlzLmhpZGVUb29sdGlwKCkpO1xuICAgIH0pO1xuICB9XG5cbiAgcHJpdmF0ZSBzaG93VG9vbHRpcChpdGVtOiBIVE1MRWxlbWVudCk6IHZvaWQge1xuICAgIGlmICghdGhpcy5pc0NvbGxhcHNlZCB8fCAhdGhpcy5tZW51VG9vbHRpcCkgcmV0dXJuO1xuXG4gICAgY29uc3QgcmVjdCA9IGl0ZW0uZ2V0Qm91bmRpbmdDbGllbnRSZWN0KCk7XG4gICAgY29uc3QgdG9vbHRpcFRleHQgPSBpdGVtLmRhdGFzZXQudG9vbHRpcDtcblxuICAgIGlmICghdG9vbHRpcFRleHQpIHJldHVybjtcblxuICAgIHRoaXMubWVudVRvb2x0aXAudGV4dENvbnRlbnQgPSB0b29sdGlwVGV4dDtcbiAgICB0aGlzLm1lbnVUb29sdGlwLnN0eWxlLmxlZnQgPSBgJHtyZWN0LnJpZ2h0ICsgOH1weGA7XG4gICAgdGhpcy5tZW51VG9vbHRpcC5zdHlsZS50b3AgPSBgJHtyZWN0LnRvcCArIHJlY3QuaGVpZ2h0IC8gMn1weGA7XG4gICAgdGhpcy5tZW51VG9vbHRpcC5zdHlsZS50cmFuc2Zvcm0gPSAndHJhbnNsYXRlWSgtNTAlKSc7XG4gICAgdGhpcy5tZW51VG9vbHRpcC5zaG93UG9wb3ZlcigpO1xuICB9XG5cbiAgcHJpdmF0ZSBoaWRlVG9vbHRpcCgpOiB2b2lkIHtcbiAgICB0aGlzLm1lbnVUb29sdGlwPy5oaWRlUG9wb3ZlcigpO1xuICB9XG5cbiAgcHJpdmF0ZSByZXN0b3JlU3RhdGUoKTogdm9pZCB7XG4gICAgaWYgKCF0aGlzLmFwcExheW91dCkgcmV0dXJuO1xuXG4gICAgaWYgKGxvY2FsU3RvcmFnZS5nZXRJdGVtKCdzaWRlYmFyLWNvbGxhcHNlZCcpID09PSAndHJ1ZScpIHtcbiAgICAgIHRoaXMuYXBwTGF5b3V0LmNsYXNzTGlzdC5hZGQoJ21lbnUtY29sbGFwc2VkJyk7XG4gICAgfVxuICB9XG59XG4iLCAiLyoqXG4gKiBEcmF3ZXIgQ29udHJvbGxlclxuICpcbiAqIEhhbmRsZXMgYWxsIGRyYXdlciBmdW5jdGlvbmFsaXR5IGluY2x1ZGluZyBwcm9maWxlLCBub3RpZmljYXRpb25zLCBhbmQgdG9kbyBkcmF3ZXJzXG4gKi9cblxuZXhwb3J0IHR5cGUgRHJhd2VyTmFtZSA9ICdwcm9maWxlJyB8ICdub3RpZmljYXRpb24nIHwgJ3RvZG8nIHwgJ25ld1RvZG8nO1xuXG5leHBvcnQgY2xhc3MgRHJhd2VyQ29udHJvbGxlciB7XG4gIHByaXZhdGUgcHJvZmlsZURyYXdlcjogSFRNTEVsZW1lbnQgfCBudWxsID0gbnVsbDtcbiAgcHJpdmF0ZSBub3RpZmljYXRpb25EcmF3ZXI6IEhUTUxFbGVtZW50IHwgbnVsbCA9IG51bGw7XG4gIHByaXZhdGUgdG9kb0RyYXdlcjogSFRNTEVsZW1lbnQgfCBudWxsID0gbnVsbDtcbiAgcHJpdmF0ZSBuZXdUb2RvRHJhd2VyOiBIVE1MRWxlbWVudCB8IG51bGwgPSBudWxsO1xuICBwcml2YXRlIG92ZXJsYXk6IEhUTUxFbGVtZW50IHwgbnVsbCA9IG51bGw7XG4gIHByaXZhdGUgYWN0aXZlRHJhd2VyOiBEcmF3ZXJOYW1lIHwgbnVsbCA9IG51bGw7XG4gIHByaXZhdGUgYWN0aXZlR2VuZXJpY0RyYXdlcjogSFRNTEVsZW1lbnQgfCBudWxsID0gbnVsbDtcblxuICBjb25zdHJ1Y3RvcigpIHtcbiAgICB0aGlzLnByb2ZpbGVEcmF3ZXIgPSBkb2N1bWVudC5nZXRFbGVtZW50QnlJZCgncHJvZmlsZURyYXdlcicpO1xuICAgIHRoaXMubm90aWZpY2F0aW9uRHJhd2VyID0gZG9jdW1lbnQuZ2V0RWxlbWVudEJ5SWQoJ25vdGlmaWNhdGlvbkRyYXdlcicpO1xuICAgIHRoaXMudG9kb0RyYXdlciA9IGRvY3VtZW50LmdldEVsZW1lbnRCeUlkKCd0b2RvRHJhd2VyJyk7XG4gICAgdGhpcy5uZXdUb2RvRHJhd2VyID0gZG9jdW1lbnQuZ2V0RWxlbWVudEJ5SWQoJ25ld1RvZG9EcmF3ZXInKTtcbiAgICB0aGlzLm92ZXJsYXkgPSBkb2N1bWVudC5nZXRFbGVtZW50QnlJZCgnZHJhd2VyT3ZlcmxheScpO1xuXG4gICAgdGhpcy5zZXR1cExpc3RlbmVycygpO1xuICAgIHRoaXMuc2V0dXBHZW5lcmljRHJhd2VycygpO1xuICB9XG5cbiAgLyoqXG4gICAqIEdldCBjdXJyZW50bHkgYWN0aXZlIGRyYXdlciBuYW1lXG4gICAqL1xuICBnZXQgYWN0aXZlKCk6IERyYXdlck5hbWUgfCBudWxsIHtcbiAgICByZXR1cm4gdGhpcy5hY3RpdmVEcmF3ZXI7XG4gIH1cblxuICAvKipcbiAgICogT3BlbiBhIGRyYXdlciBieSBuYW1lXG4gICAqL1xuICBvcGVuKG5hbWU6IERyYXdlck5hbWUpOiB2b2lkIHtcbiAgICB0aGlzLmNsb3NlQWxsKCk7XG5cbiAgICBjb25zdCBkcmF3ZXIgPSB0aGlzLmdldERyYXdlcihuYW1lKTtcbiAgICBpZiAoZHJhd2VyICYmIHRoaXMub3ZlcmxheSkge1xuICAgICAgZHJhd2VyLmNsYXNzTGlzdC5hZGQoJ2FjdGl2ZScpO1xuICAgICAgdGhpcy5vdmVybGF5LmNsYXNzTGlzdC5hZGQoJ2FjdGl2ZScpO1xuICAgICAgZG9jdW1lbnQuYm9keS5zdHlsZS5vdmVyZmxvdyA9ICdoaWRkZW4nO1xuICAgICAgdGhpcy5hY3RpdmVEcmF3ZXIgPSBuYW1lO1xuICAgIH1cbiAgfVxuXG4gIC8qKlxuICAgKiBDbG9zZSBhIHNwZWNpZmljIGRyYXdlclxuICAgKi9cbiAgY2xvc2UobmFtZTogRHJhd2VyTmFtZSk6IHZvaWQge1xuICAgIGNvbnN0IGRyYXdlciA9IHRoaXMuZ2V0RHJhd2VyKG5hbWUpO1xuICAgIGRyYXdlcj8uY2xhc3NMaXN0LnJlbW92ZSgnYWN0aXZlJyk7XG5cbiAgICAvLyBPbmx5IGhpZGUgb3ZlcmxheSBpZiBubyBkcmF3ZXJzIGFyZSBhY3RpdmVcbiAgICBpZiAodGhpcy5vdmVybGF5ICYmICFkb2N1bWVudC5xdWVyeVNlbGVjdG9yKCcuYWN0aXZlW2NsYXNzKj1cImRyYXdlclwiXScpKSB7XG4gICAgICB0aGlzLm92ZXJsYXkuY2xhc3NMaXN0LnJlbW92ZSgnYWN0aXZlJyk7XG4gICAgICBkb2N1bWVudC5ib2R5LnN0eWxlLm92ZXJmbG93ID0gJyc7XG4gICAgfVxuXG4gICAgaWYgKHRoaXMuYWN0aXZlRHJhd2VyID09PSBuYW1lKSB7XG4gICAgICB0aGlzLmFjdGl2ZURyYXdlciA9IG51bGw7XG4gICAgfVxuICB9XG5cbiAgLyoqXG4gICAqIENsb3NlIGFsbCBkcmF3ZXJzXG4gICAqL1xuICBjbG9zZUFsbCgpOiB2b2lkIHtcbiAgICBbdGhpcy5wcm9maWxlRHJhd2VyLCB0aGlzLm5vdGlmaWNhdGlvbkRyYXdlciwgdGhpcy50b2RvRHJhd2VyLCB0aGlzLm5ld1RvZG9EcmF3ZXJdXG4gICAgICAuZm9yRWFjaChkcmF3ZXIgPT4gZHJhd2VyPy5jbGFzc0xpc3QucmVtb3ZlKCdhY3RpdmUnKSk7XG5cbiAgICAvLyBDbG9zZSBhbnkgZ2VuZXJpYyBkcmF3ZXJzXG4gICAgdGhpcy5jbG9zZUdlbmVyaWNEcmF3ZXIoKTtcblxuICAgIHRoaXMub3ZlcmxheT8uY2xhc3NMaXN0LnJlbW92ZSgnYWN0aXZlJyk7XG4gICAgZG9jdW1lbnQuYm9keS5zdHlsZS5vdmVyZmxvdyA9ICcnO1xuICAgIHRoaXMuYWN0aXZlRHJhd2VyID0gbnVsbDtcbiAgfVxuXG4gIC8qKlxuICAgKiBPcGVuIGEgZ2VuZXJpYyBkcmF3ZXIgYnkgSURcbiAgICovXG4gIG9wZW5HZW5lcmljRHJhd2VyKGRyYXdlcklkOiBzdHJpbmcpOiB2b2lkIHtcbiAgICB0aGlzLmNsb3NlQWxsKCk7XG5cbiAgICBjb25zdCBkcmF3ZXIgPSBkb2N1bWVudC5nZXRFbGVtZW50QnlJZChkcmF3ZXJJZCk7XG4gICAgaWYgKGRyYXdlciAmJiB0aGlzLm92ZXJsYXkpIHtcbiAgICAgIGRyYXdlci5jbGFzc0xpc3QuYWRkKCdvcGVuJyk7XG4gICAgICB0aGlzLm92ZXJsYXkuY2xhc3NMaXN0LmFkZCgnYWN0aXZlJyk7XG4gICAgICBkb2N1bWVudC5ib2R5LnN0eWxlLm92ZXJmbG93ID0gJ2hpZGRlbic7XG4gICAgICB0aGlzLmFjdGl2ZUdlbmVyaWNEcmF3ZXIgPSBkcmF3ZXI7XG4gICAgfVxuICB9XG5cbiAgLyoqXG4gICAqIENsb3NlIHRoZSBjdXJyZW50bHkgb3BlbiBnZW5lcmljIGRyYXdlclxuICAgKi9cbiAgY2xvc2VHZW5lcmljRHJhd2VyKCk6IHZvaWQge1xuICAgIHRoaXMuYWN0aXZlR2VuZXJpY0RyYXdlcj8uY2xhc3NMaXN0LnJlbW92ZSgnb3BlbicpO1xuICAgIHRoaXMuYWN0aXZlR2VuZXJpY0RyYXdlciA9IG51bGw7XG4gIH1cblxuICAvKipcbiAgICogT3BlbiBwcm9maWxlIGRyYXdlclxuICAgKi9cbiAgb3BlblByb2ZpbGUoKTogdm9pZCB7XG4gICAgdGhpcy5vcGVuKCdwcm9maWxlJyk7XG4gIH1cblxuICAvKipcbiAgICogT3BlbiBub3RpZmljYXRpb24gZHJhd2VyXG4gICAqL1xuICBvcGVuTm90aWZpY2F0aW9uKCk6IHZvaWQge1xuICAgIHRoaXMub3Blbignbm90aWZpY2F0aW9uJyk7XG4gIH1cblxuICAvKipcbiAgICogT3BlbiB0b2RvIGRyYXdlciAoc2xpZGVzIG9uIHRvcCBvZiBwcm9maWxlKVxuICAgKi9cbiAgb3BlblRvZG8oKTogdm9pZCB7XG4gICAgdGhpcy50b2RvRHJhd2VyPy5jbGFzc0xpc3QuYWRkKCdhY3RpdmUnKTtcbiAgfVxuXG4gIC8qKlxuICAgKiBDbG9zZSB0b2RvIGRyYXdlclxuICAgKi9cbiAgY2xvc2VUb2RvKCk6IHZvaWQge1xuICAgIHRoaXMudG9kb0RyYXdlcj8uY2xhc3NMaXN0LnJlbW92ZSgnYWN0aXZlJyk7XG4gICAgdGhpcy5jbG9zZU5ld1RvZG8oKTtcbiAgfVxuXG4gIC8qKlxuICAgKiBPcGVuIG5ldyB0b2RvIGRyYXdlclxuICAgKi9cbiAgb3Blbk5ld1RvZG8oKTogdm9pZCB7XG4gICAgdGhpcy5uZXdUb2RvRHJhd2VyPy5jbGFzc0xpc3QuYWRkKCdhY3RpdmUnKTtcbiAgfVxuXG4gIC8qKlxuICAgKiBDbG9zZSBuZXcgdG9kbyBkcmF3ZXJcbiAgICovXG4gIGNsb3NlTmV3VG9kbygpOiB2b2lkIHtcbiAgICB0aGlzLm5ld1RvZG9EcmF3ZXI/LmNsYXNzTGlzdC5yZW1vdmUoJ2FjdGl2ZScpO1xuICB9XG5cbiAgLyoqXG4gICAqIE1hcmsgYWxsIG5vdGlmaWNhdGlvbnMgYXMgcmVhZFxuICAgKi9cbiAgbWFya0FsbE5vdGlmaWNhdGlvbnNSZWFkKCk6IHZvaWQge1xuICAgIGlmICghdGhpcy5ub3RpZmljYXRpb25EcmF3ZXIpIHJldHVybjtcblxuICAgIGNvbnN0IHVucmVhZEl0ZW1zID0gdGhpcy5ub3RpZmljYXRpb25EcmF3ZXIucXVlcnlTZWxlY3RvckFsbDxIVE1MRWxlbWVudD4oXG4gICAgICAnc3dwLW5vdGlmaWNhdGlvbi1pdGVtW2RhdGEtdW5yZWFkPVwidHJ1ZVwiXSdcbiAgICApO1xuICAgIHVucmVhZEl0ZW1zLmZvckVhY2goaXRlbSA9PiBpdGVtLnJlbW92ZUF0dHJpYnV0ZSgnZGF0YS11bnJlYWQnKSk7XG5cbiAgICBjb25zdCBiYWRnZSA9IGRvY3VtZW50LnF1ZXJ5U2VsZWN0b3I8SFRNTEVsZW1lbnQ+KCdzd3Atbm90aWZpY2F0aW9uLWJhZGdlJyk7XG4gICAgaWYgKGJhZGdlKSB7XG4gICAgICBiYWRnZS5zdHlsZS5kaXNwbGF5ID0gJ25vbmUnO1xuICAgIH1cbiAgfVxuXG4gIHByaXZhdGUgZ2V0RHJhd2VyKG5hbWU6IERyYXdlck5hbWUpOiBIVE1MRWxlbWVudCB8IG51bGwge1xuICAgIHN3aXRjaCAobmFtZSkge1xuICAgICAgY2FzZSAncHJvZmlsZSc6IHJldHVybiB0aGlzLnByb2ZpbGVEcmF3ZXI7XG4gICAgICBjYXNlICdub3RpZmljYXRpb24nOiByZXR1cm4gdGhpcy5ub3RpZmljYXRpb25EcmF3ZXI7XG4gICAgICBjYXNlICd0b2RvJzogcmV0dXJuIHRoaXMudG9kb0RyYXdlcjtcbiAgICAgIGNhc2UgJ25ld1RvZG8nOiByZXR1cm4gdGhpcy5uZXdUb2RvRHJhd2VyO1xuICAgIH1cbiAgfVxuXG4gIHByaXZhdGUgc2V0dXBMaXN0ZW5lcnMoKTogdm9pZCB7XG4gICAgLy8gUHJvZmlsZSBkcmF3ZXIgdHJpZ2dlcnNcbiAgICBkb2N1bWVudC5nZXRFbGVtZW50QnlJZCgncHJvZmlsZVRyaWdnZXInKVxuICAgICAgPy5hZGRFdmVudExpc3RlbmVyKCdjbGljaycsICgpID0+IHRoaXMub3BlblByb2ZpbGUoKSk7XG4gICAgZG9jdW1lbnQuZ2V0RWxlbWVudEJ5SWQoJ2RyYXdlckNsb3NlJylcbiAgICAgID8uYWRkRXZlbnRMaXN0ZW5lcignY2xpY2snLCAoKSA9PiB0aGlzLmNsb3NlKCdwcm9maWxlJykpO1xuXG4gICAgLy8gTm90aWZpY2F0aW9uIGRyYXdlciB0cmlnZ2Vyc1xuICAgIGRvY3VtZW50LmdldEVsZW1lbnRCeUlkKCdub3RpZmljYXRpb25zQnRuJylcbiAgICAgID8uYWRkRXZlbnRMaXN0ZW5lcignY2xpY2snLCAoKSA9PiB0aGlzLm9wZW5Ob3RpZmljYXRpb24oKSk7XG4gICAgZG9jdW1lbnQuZ2V0RWxlbWVudEJ5SWQoJ25vdGlmaWNhdGlvbkRyYXdlckNsb3NlJylcbiAgICAgID8uYWRkRXZlbnRMaXN0ZW5lcignY2xpY2snLCAoKSA9PiB0aGlzLmNsb3NlKCdub3RpZmljYXRpb24nKSk7XG4gICAgZG9jdW1lbnQuZ2V0RWxlbWVudEJ5SWQoJ21hcmtBbGxSZWFkJylcbiAgICAgID8uYWRkRXZlbnRMaXN0ZW5lcignY2xpY2snLCAoKSA9PiB0aGlzLm1hcmtBbGxOb3RpZmljYXRpb25zUmVhZCgpKTtcblxuICAgIC8vIFRvZG8gZHJhd2VyIHRyaWdnZXJzXG4gICAgZG9jdW1lbnQuZ2V0RWxlbWVudEJ5SWQoJ29wZW5Ub2RvRHJhd2VyJylcbiAgICAgID8uYWRkRXZlbnRMaXN0ZW5lcignY2xpY2snLCAoKSA9PiB0aGlzLm9wZW5Ub2RvKCkpO1xuICAgIGRvY3VtZW50LmdldEVsZW1lbnRCeUlkKCd0b2RvRHJhd2VyQmFjaycpXG4gICAgICA/LmFkZEV2ZW50TGlzdGVuZXIoJ2NsaWNrJywgKCkgPT4gdGhpcy5jbG9zZVRvZG8oKSk7XG5cbiAgICAvLyBOZXcgdG9kbyBkcmF3ZXIgdHJpZ2dlcnNcbiAgICBkb2N1bWVudC5nZXRFbGVtZW50QnlJZCgnYWRkVG9kb0J0bicpXG4gICAgICA/LmFkZEV2ZW50TGlzdGVuZXIoJ2NsaWNrJywgKCkgPT4gdGhpcy5vcGVuTmV3VG9kbygpKTtcbiAgICBkb2N1bWVudC5nZXRFbGVtZW50QnlJZCgnbmV3VG9kb0RyYXdlckJhY2snKVxuICAgICAgPy5hZGRFdmVudExpc3RlbmVyKCdjbGljaycsICgpID0+IHRoaXMuY2xvc2VOZXdUb2RvKCkpO1xuICAgIGRvY3VtZW50LmdldEVsZW1lbnRCeUlkKCdjYW5jZWxOZXdUb2RvJylcbiAgICAgID8uYWRkRXZlbnRMaXN0ZW5lcignY2xpY2snLCAoKSA9PiB0aGlzLmNsb3NlTmV3VG9kbygpKTtcbiAgICBkb2N1bWVudC5nZXRFbGVtZW50QnlJZCgnc2F2ZU5ld1RvZG8nKVxuICAgICAgPy5hZGRFdmVudExpc3RlbmVyKCdjbGljaycsICgpID0+IHRoaXMuY2xvc2VOZXdUb2RvKCkpO1xuXG4gICAgLy8gT3ZlcmxheSBjbGljayBjbG9zZXMgYWxsXG4gICAgdGhpcy5vdmVybGF5Py5hZGRFdmVudExpc3RlbmVyKCdjbGljaycsICgpID0+IHRoaXMuY2xvc2VBbGwoKSk7XG5cbiAgICAvLyBFc2NhcGUga2V5IGNsb3NlcyBhbGxcbiAgICBkb2N1bWVudC5hZGRFdmVudExpc3RlbmVyKCdrZXlkb3duJywgKGU6IEtleWJvYXJkRXZlbnQpID0+IHtcbiAgICAgIGlmIChlLmtleSA9PT0gJ0VzY2FwZScpIHRoaXMuY2xvc2VBbGwoKTtcbiAgICB9KTtcblxuICAgIC8vIFRvZG8gaW50ZXJhY3Rpb25zXG4gICAgdGhpcy50b2RvRHJhd2VyPy5hZGRFdmVudExpc3RlbmVyKCdjbGljaycsIChlKSA9PiB0aGlzLmhhbmRsZVRvZG9DbGljayhlKSk7XG5cbiAgICAvLyBWaXNpYmlsaXR5IG9wdGlvbnNcbiAgICBkb2N1bWVudC5hZGRFdmVudExpc3RlbmVyKCdjbGljaycsIChlKSA9PiB0aGlzLmhhbmRsZVZpc2liaWxpdHlDbGljayhlKSk7XG4gIH1cblxuICBwcml2YXRlIGhhbmRsZVRvZG9DbGljayhlOiBFdmVudCk6IHZvaWQge1xuICAgIGNvbnN0IHRhcmdldCA9IGUudGFyZ2V0IGFzIEhUTUxFbGVtZW50O1xuICAgIGNvbnN0IHRvZG9JdGVtID0gdGFyZ2V0LmNsb3Nlc3Q8SFRNTEVsZW1lbnQ+KCdzd3AtdG9kby1pdGVtJyk7XG4gICAgY29uc3QgY2hlY2tib3ggPSB0YXJnZXQuY2xvc2VzdDxIVE1MRWxlbWVudD4oJ3N3cC10b2RvLWNoZWNrYm94Jyk7XG5cbiAgICBpZiAoY2hlY2tib3ggJiYgdG9kb0l0ZW0pIHtcbiAgICAgIGNvbnN0IGlzQ29tcGxldGVkID0gdG9kb0l0ZW0uZGF0YXNldC5jb21wbGV0ZWQgPT09ICd0cnVlJztcbiAgICAgIGlmIChpc0NvbXBsZXRlZCkge1xuICAgICAgICB0b2RvSXRlbS5yZW1vdmVBdHRyaWJ1dGUoJ2RhdGEtY29tcGxldGVkJyk7XG4gICAgICB9IGVsc2Uge1xuICAgICAgICB0b2RvSXRlbS5kYXRhc2V0LmNvbXBsZXRlZCA9ICd0cnVlJztcbiAgICAgIH1cbiAgICB9XG5cbiAgICAvLyBUb2dnbGUgc2VjdGlvbiBjb2xsYXBzZVxuICAgIGNvbnN0IHNlY3Rpb25IZWFkZXIgPSB0YXJnZXQuY2xvc2VzdDxIVE1MRWxlbWVudD4oJ3N3cC10b2RvLXNlY3Rpb24taGVhZGVyJyk7XG4gICAgaWYgKHNlY3Rpb25IZWFkZXIpIHtcbiAgICAgIGNvbnN0IHNlY3Rpb24gPSBzZWN0aW9uSGVhZGVyLmNsb3Nlc3Q8SFRNTEVsZW1lbnQ+KCdzd3AtdG9kby1zZWN0aW9uJyk7XG4gICAgICBzZWN0aW9uPy5jbGFzc0xpc3QudG9nZ2xlKCdjb2xsYXBzZWQnKTtcbiAgICB9XG4gIH1cblxuICBwcml2YXRlIGhhbmRsZVZpc2liaWxpdHlDbGljayhlOiBFdmVudCk6IHZvaWQge1xuICAgIGNvbnN0IHRhcmdldCA9IGUudGFyZ2V0IGFzIEhUTUxFbGVtZW50O1xuICAgIGNvbnN0IG9wdGlvbiA9IHRhcmdldC5jbG9zZXN0PEhUTUxFbGVtZW50Pignc3dwLXZpc2liaWxpdHktb3B0aW9uJyk7XG5cbiAgICBpZiAob3B0aW9uKSB7XG4gICAgICBkb2N1bWVudC5xdWVyeVNlbGVjdG9yQWxsPEhUTUxFbGVtZW50Pignc3dwLXZpc2liaWxpdHktb3B0aW9uJylcbiAgICAgICAgLmZvckVhY2gobyA9PiBvLmNsYXNzTGlzdC5yZW1vdmUoJ2FjdGl2ZScpKTtcbiAgICAgIG9wdGlvbi5jbGFzc0xpc3QuYWRkKCdhY3RpdmUnKTtcbiAgICB9XG4gIH1cblxuICAvKipcbiAgICogU2V0dXAgZ2VuZXJpYyBkcmF3ZXIgdHJpZ2dlcnMgYW5kIGNsb3NlIGJ1dHRvbnNcbiAgICogVXNlcyBkYXRhLWRyYXdlci10cmlnZ2VyPVwiZHJhd2VyLWlkXCIgYW5kIGRhdGEtZHJhd2VyLWNsb3NlIGF0dHJpYnV0ZXNcbiAgICovXG4gIHByaXZhdGUgc2V0dXBHZW5lcmljRHJhd2VycygpOiB2b2lkIHtcbiAgICAvLyBIYW5kbGUgZHJhd2VyIHRyaWdnZXJzXG4gICAgZG9jdW1lbnQuYWRkRXZlbnRMaXN0ZW5lcignY2xpY2snLCAoZTogRXZlbnQpID0+IHtcbiAgICAgIGNvbnN0IHRhcmdldCA9IGUudGFyZ2V0IGFzIEhUTUxFbGVtZW50O1xuICAgICAgY29uc3QgdHJpZ2dlciA9IHRhcmdldC5jbG9zZXN0PEhUTUxFbGVtZW50PignW2RhdGEtZHJhd2VyLXRyaWdnZXJdJyk7XG5cbiAgICAgIGlmICh0cmlnZ2VyKSB7XG4gICAgICAgIGNvbnN0IGRyYXdlcklkID0gdHJpZ2dlci5kYXRhc2V0LmRyYXdlclRyaWdnZXI7XG4gICAgICAgIGlmIChkcmF3ZXJJZCkge1xuICAgICAgICAgIHRoaXMub3BlbkdlbmVyaWNEcmF3ZXIoZHJhd2VySWQpO1xuICAgICAgICB9XG4gICAgICB9XG4gICAgfSk7XG5cbiAgICAvLyBIYW5kbGUgZHJhd2VyIGNsb3NlIGJ1dHRvbnNcbiAgICBkb2N1bWVudC5hZGRFdmVudExpc3RlbmVyKCdjbGljaycsIChlOiBFdmVudCkgPT4ge1xuICAgICAgY29uc3QgdGFyZ2V0ID0gZS50YXJnZXQgYXMgSFRNTEVsZW1lbnQ7XG4gICAgICBjb25zdCBjbG9zZUJ0biA9IHRhcmdldC5jbG9zZXN0PEhUTUxFbGVtZW50PignW2RhdGEtZHJhd2VyLWNsb3NlXScpO1xuXG4gICAgICBpZiAoY2xvc2VCdG4pIHtcbiAgICAgICAgdGhpcy5jbG9zZUdlbmVyaWNEcmF3ZXIoKTtcbiAgICAgICAgdGhpcy5vdmVybGF5Py5jbGFzc0xpc3QucmVtb3ZlKCdhY3RpdmUnKTtcbiAgICAgICAgZG9jdW1lbnQuYm9keS5zdHlsZS5vdmVyZmxvdyA9ICcnO1xuICAgICAgfVxuICAgIH0pO1xuICB9XG59XG4iLCAiLyoqXG4gKiBUaGVtZSBDb250cm9sbGVyXG4gKlxuICogSGFuZGxlcyBkYXJrL2xpZ2h0IG1vZGUgc3dpdGNoaW5nIGFuZCBzeXN0ZW0gcHJlZmVyZW5jZSBkZXRlY3Rpb25cbiAqL1xuXG5leHBvcnQgdHlwZSBUaGVtZSA9ICdsaWdodCcgfCAnZGFyaycgfCAnc3lzdGVtJztcblxuZXhwb3J0IGNsYXNzIFRoZW1lQ29udHJvbGxlciB7XG4gIHByaXZhdGUgc3RhdGljIHJlYWRvbmx5IFNUT1JBR0VfS0VZID0gJ3RoZW1lLXByZWZlcmVuY2UnO1xuICBwcml2YXRlIHN0YXRpYyByZWFkb25seSBEQVJLX0NMQVNTID0gJ2RhcmstbW9kZSc7XG4gIHByaXZhdGUgc3RhdGljIHJlYWRvbmx5IExJR0hUX0NMQVNTID0gJ2xpZ2h0LW1vZGUnO1xuXG4gIHByaXZhdGUgcm9vdDogSFRNTEVsZW1lbnQ7XG4gIHByaXZhdGUgdGhlbWVPcHRpb25zOiBOb2RlTGlzdE9mPEhUTUxFbGVtZW50PjtcblxuICBjb25zdHJ1Y3RvcigpIHtcbiAgICB0aGlzLnJvb3QgPSBkb2N1bWVudC5kb2N1bWVudEVsZW1lbnQ7XG4gICAgdGhpcy50aGVtZU9wdGlvbnMgPSBkb2N1bWVudC5xdWVyeVNlbGVjdG9yQWxsPEhUTUxFbGVtZW50Pignc3dwLXRoZW1lLW9wdGlvbicpO1xuXG4gICAgdGhpcy5hcHBseVRoZW1lKHRoaXMuY3VycmVudCk7XG4gICAgdGhpcy51cGRhdGVVSSgpO1xuICAgIHRoaXMuc2V0dXBMaXN0ZW5lcnMoKTtcbiAgfVxuXG4gIC8qKlxuICAgKiBHZXQgdGhlIGN1cnJlbnQgdGhlbWUgc2V0dGluZ1xuICAgKi9cbiAgZ2V0IGN1cnJlbnQoKTogVGhlbWUge1xuICAgIGNvbnN0IHN0b3JlZCA9IGxvY2FsU3RvcmFnZS5nZXRJdGVtKFRoZW1lQ29udHJvbGxlci5TVE9SQUdFX0tFWSkgYXMgVGhlbWUgfCBudWxsO1xuICAgIGlmIChzdG9yZWQgPT09ICdkYXJrJyB8fCBzdG9yZWQgPT09ICdsaWdodCcgfHwgc3RvcmVkID09PSAnc3lzdGVtJykge1xuICAgICAgcmV0dXJuIHN0b3JlZDtcbiAgICB9XG4gICAgcmV0dXJuICdzeXN0ZW0nO1xuICB9XG5cbiAgLyoqXG4gICAqIENoZWNrIGlmIGRhcmsgbW9kZSBpcyBjdXJyZW50bHkgYWN0aXZlXG4gICAqL1xuICBnZXQgaXNEYXJrKCk6IGJvb2xlYW4ge1xuICAgIHJldHVybiB0aGlzLnJvb3QuY2xhc3NMaXN0LmNvbnRhaW5zKFRoZW1lQ29udHJvbGxlci5EQVJLX0NMQVNTKSB8fFxuICAgICAgKHRoaXMuc3lzdGVtUHJlZmVyc0RhcmsgJiYgIXRoaXMucm9vdC5jbGFzc0xpc3QuY29udGFpbnMoVGhlbWVDb250cm9sbGVyLkxJR0hUX0NMQVNTKSk7XG4gIH1cblxuICAvKipcbiAgICogQ2hlY2sgaWYgc3lzdGVtIHByZWZlcnMgZGFyayBtb2RlXG4gICAqL1xuICBnZXQgc3lzdGVtUHJlZmVyc0RhcmsoKTogYm9vbGVhbiB7XG4gICAgcmV0dXJuIHdpbmRvdy5tYXRjaE1lZGlhKCcocHJlZmVycy1jb2xvci1zY2hlbWU6IGRhcmspJykubWF0Y2hlcztcbiAgfVxuXG4gIC8qKlxuICAgKiBTZXQgdGhlbWUgYW5kIHBlcnNpc3QgcHJlZmVyZW5jZVxuICAgKi9cbiAgc2V0KHRoZW1lOiBUaGVtZSk6IHZvaWQge1xuICAgIGxvY2FsU3RvcmFnZS5zZXRJdGVtKFRoZW1lQ29udHJvbGxlci5TVE9SQUdFX0tFWSwgdGhlbWUpO1xuICAgIHRoaXMuYXBwbHlUaGVtZSh0aGVtZSk7XG4gICAgdGhpcy51cGRhdGVVSSgpO1xuICB9XG5cbiAgLyoqXG4gICAqIFRvZ2dsZSBiZXR3ZWVuIGxpZ2h0IGFuZCBkYXJrIHRoZW1lc1xuICAgKi9cbiAgdG9nZ2xlKCk6IHZvaWQge1xuICAgIHRoaXMuc2V0KHRoaXMuaXNEYXJrID8gJ2xpZ2h0JyA6ICdkYXJrJyk7XG4gIH1cblxuICBwcml2YXRlIGFwcGx5VGhlbWUodGhlbWU6IFRoZW1lKTogdm9pZCB7XG4gICAgdGhpcy5yb290LmNsYXNzTGlzdC5yZW1vdmUoVGhlbWVDb250cm9sbGVyLkRBUktfQ0xBU1MsIFRoZW1lQ29udHJvbGxlci5MSUdIVF9DTEFTUyk7XG5cbiAgICBpZiAodGhlbWUgPT09ICdkYXJrJykge1xuICAgICAgdGhpcy5yb290LmNsYXNzTGlzdC5hZGQoVGhlbWVDb250cm9sbGVyLkRBUktfQ0xBU1MpO1xuICAgIH0gZWxzZSBpZiAodGhlbWUgPT09ICdsaWdodCcpIHtcbiAgICAgIHRoaXMucm9vdC5jbGFzc0xpc3QuYWRkKFRoZW1lQ29udHJvbGxlci5MSUdIVF9DTEFTUyk7XG4gICAgfVxuICAgIC8vICdzeXN0ZW0nIGxlYXZlcyBib3RoIGNsYXNzZXMgb2ZmLCBsZXR0aW5nIENTUyBtZWRpYSBxdWVyeSBoYW5kbGUgaXRcbiAgfVxuXG4gIHByaXZhdGUgdXBkYXRlVUkoKTogdm9pZCB7XG4gICAgaWYgKCF0aGlzLnRoZW1lT3B0aW9ucykgcmV0dXJuO1xuXG4gICAgY29uc3QgZGFya0FjdGl2ZSA9IHRoaXMuaXNEYXJrO1xuXG4gICAgdGhpcy50aGVtZU9wdGlvbnMuZm9yRWFjaChvcHRpb24gPT4ge1xuICAgICAgY29uc3QgdGhlbWUgPSBvcHRpb24uZGF0YXNldC50aGVtZSBhcyBUaGVtZTtcbiAgICAgIGNvbnN0IGlzQWN0aXZlID0gKHRoZW1lID09PSAnZGFyaycgJiYgZGFya0FjdGl2ZSkgfHwgKHRoZW1lID09PSAnbGlnaHQnICYmICFkYXJrQWN0aXZlKTtcbiAgICAgIG9wdGlvbi5jbGFzc0xpc3QudG9nZ2xlKCdhY3RpdmUnLCBpc0FjdGl2ZSk7XG4gICAgfSk7XG4gIH1cblxuICBwcml2YXRlIHNldHVwTGlzdGVuZXJzKCk6IHZvaWQge1xuICAgIC8vIFRoZW1lIG9wdGlvbiBjbGlja3NcbiAgICB0aGlzLnRoZW1lT3B0aW9ucy5mb3JFYWNoKG9wdGlvbiA9PiB7XG4gICAgICBvcHRpb24uYWRkRXZlbnRMaXN0ZW5lcignY2xpY2snLCAoZSkgPT4gdGhpcy5oYW5kbGVPcHRpb25DbGljayhlKSk7XG4gICAgfSk7XG5cbiAgICAvLyBTeXN0ZW0gdGhlbWUgY2hhbmdlc1xuICAgIHdpbmRvdy5tYXRjaE1lZGlhKCcocHJlZmVycy1jb2xvci1zY2hlbWU6IGRhcmspJylcbiAgICAgIC5hZGRFdmVudExpc3RlbmVyKCdjaGFuZ2UnLCAoKSA9PiB0aGlzLmhhbmRsZVN5c3RlbUNoYW5nZSgpKTtcbiAgfVxuXG4gIHByaXZhdGUgaGFuZGxlT3B0aW9uQ2xpY2soZTogRXZlbnQpOiB2b2lkIHtcbiAgICBjb25zdCB0YXJnZXQgPSBlLnRhcmdldCBhcyBIVE1MRWxlbWVudDtcbiAgICBjb25zdCBvcHRpb24gPSB0YXJnZXQuY2xvc2VzdDxIVE1MRWxlbWVudD4oJ3N3cC10aGVtZS1vcHRpb24nKTtcblxuICAgIGlmIChvcHRpb24pIHtcbiAgICAgIGNvbnN0IHRoZW1lID0gb3B0aW9uLmRhdGFzZXQudGhlbWUgYXMgVGhlbWU7XG4gICAgICBpZiAodGhlbWUpIHtcbiAgICAgICAgdGhpcy5zZXQodGhlbWUpO1xuICAgICAgfVxuICAgIH1cbiAgfVxuXG4gIHByaXZhdGUgaGFuZGxlU3lzdGVtQ2hhbmdlKCk6IHZvaWQge1xuICAgIC8vIE9ubHkgcmVhY3QgdG8gc3lzdGVtIGNoYW5nZXMgaWYgd2UncmUgdXNpbmcgc3lzdGVtIHByZWZlcmVuY2VcbiAgICBpZiAodGhpcy5jdXJyZW50ID09PSAnc3lzdGVtJykge1xuICAgICAgdGhpcy51cGRhdGVVSSgpO1xuICAgIH1cbiAgfVxufVxuIiwgIi8qKlxuICogU2VhcmNoIENvbnRyb2xsZXJcbiAqXG4gKiBIYW5kbGVzIGdsb2JhbCBzZWFyY2ggZnVuY3Rpb25hbGl0eSBhbmQga2V5Ym9hcmQgc2hvcnRjdXRzXG4gKi9cblxuZXhwb3J0IGNsYXNzIFNlYXJjaENvbnRyb2xsZXIge1xuICBwcml2YXRlIGlucHV0OiBIVE1MSW5wdXRFbGVtZW50IHwgbnVsbCA9IG51bGw7XG4gIHByaXZhdGUgY29udGFpbmVyOiBIVE1MRWxlbWVudCB8IG51bGwgPSBudWxsO1xuXG4gIGNvbnN0cnVjdG9yKCkge1xuICAgIHRoaXMuaW5wdXQgPSBkb2N1bWVudC5nZXRFbGVtZW50QnlJZCgnZ2xvYmFsU2VhcmNoJykgYXMgSFRNTElucHV0RWxlbWVudCB8IG51bGw7XG4gICAgdGhpcy5jb250YWluZXIgPSBkb2N1bWVudC5xdWVyeVNlbGVjdG9yPEhUTUxFbGVtZW50Pignc3dwLXRvcGJhci1zZWFyY2gnKTtcblxuICAgIHRoaXMuc2V0dXBMaXN0ZW5lcnMoKTtcbiAgfVxuXG4gIC8qKlxuICAgKiBHZXQgY3VycmVudCBzZWFyY2ggdmFsdWVcbiAgICovXG4gIGdldCB2YWx1ZSgpOiBzdHJpbmcge1xuICAgIHJldHVybiB0aGlzLmlucHV0Py52YWx1ZSA/PyAnJztcbiAgfVxuXG4gIC8qKlxuICAgKiBTZXQgc2VhcmNoIHZhbHVlXG4gICAqL1xuICBzZXQgdmFsdWUodmFsOiBzdHJpbmcpIHtcbiAgICBpZiAodGhpcy5pbnB1dCkge1xuICAgICAgdGhpcy5pbnB1dC52YWx1ZSA9IHZhbDtcbiAgICB9XG4gIH1cblxuICAvKipcbiAgICogRm9jdXMgdGhlIHNlYXJjaCBpbnB1dFxuICAgKi9cbiAgZm9jdXMoKTogdm9pZCB7XG4gICAgdGhpcy5pbnB1dD8uZm9jdXMoKTtcbiAgfVxuXG4gIC8qKlxuICAgKiBCbHVyIHRoZSBzZWFyY2ggaW5wdXRcbiAgICovXG4gIGJsdXIoKTogdm9pZCB7XG4gICAgdGhpcy5pbnB1dD8uYmx1cigpO1xuICB9XG5cbiAgLyoqXG4gICAqIENsZWFyIHRoZSBzZWFyY2ggaW5wdXRcbiAgICovXG4gIGNsZWFyKCk6IHZvaWQge1xuICAgIHRoaXMudmFsdWUgPSAnJztcbiAgfVxuXG4gIHByaXZhdGUgc2V0dXBMaXN0ZW5lcnMoKTogdm9pZCB7XG4gICAgLy8gS2V5Ym9hcmQgc2hvcnRjdXRzXG4gICAgZG9jdW1lbnQuYWRkRXZlbnRMaXN0ZW5lcigna2V5ZG93bicsIChlKSA9PiB0aGlzLmhhbmRsZUtleWJvYXJkKGUpKTtcblxuICAgIC8vIElucHV0IGhhbmRsZXJzXG4gICAgaWYgKHRoaXMuaW5wdXQpIHtcbiAgICAgIHRoaXMuaW5wdXQuYWRkRXZlbnRMaXN0ZW5lcignaW5wdXQnLCAoZSkgPT4gdGhpcy5oYW5kbGVJbnB1dChlKSk7XG5cbiAgICAgIC8vIFByZXZlbnQgZm9ybSBzdWJtaXNzaW9uIGlmIHdyYXBwZWQgaW4gZm9ybVxuICAgICAgY29uc3QgZm9ybSA9IHRoaXMuaW5wdXQuY2xvc2VzdCgnZm9ybScpO1xuICAgICAgZm9ybT8uYWRkRXZlbnRMaXN0ZW5lcignc3VibWl0JywgKGUpID0+IHRoaXMuaGFuZGxlU3VibWl0KGUpKTtcbiAgICB9XG4gIH1cblxuICBwcml2YXRlIGhhbmRsZUtleWJvYXJkKGU6IEtleWJvYXJkRXZlbnQpOiB2b2lkIHtcbiAgICAvLyBDbWQvQ3RybCArIEsgdG8gZm9jdXMgc2VhcmNoXG4gICAgaWYgKChlLm1ldGFLZXkgfHwgZS5jdHJsS2V5KSAmJiBlLmtleSA9PT0gJ2snKSB7XG4gICAgICBlLnByZXZlbnREZWZhdWx0KCk7XG4gICAgICB0aGlzLmZvY3VzKCk7XG4gICAgICByZXR1cm47XG4gICAgfVxuXG4gICAgLy8gRXNjYXBlIHRvIGJsdXIgc2VhcmNoIHdoZW4gZm9jdXNlZFxuICAgIGlmIChlLmtleSA9PT0gJ0VzY2FwZScgJiYgZG9jdW1lbnQuYWN0aXZlRWxlbWVudCA9PT0gdGhpcy5pbnB1dCkge1xuICAgICAgdGhpcy5ibHVyKCk7XG4gICAgfVxuICB9XG5cbiAgcHJpdmF0ZSBoYW5kbGVJbnB1dChlOiBFdmVudCk6IHZvaWQge1xuICAgIGNvbnN0IHRhcmdldCA9IGUudGFyZ2V0IGFzIEhUTUxJbnB1dEVsZW1lbnQ7XG4gICAgY29uc3QgcXVlcnkgPSB0YXJnZXQudmFsdWUudHJpbSgpO1xuXG4gICAgLy8gRW1pdCBjdXN0b20gZXZlbnQgZm9yIHNlYXJjaFxuICAgIGRvY3VtZW50LmRpc3BhdGNoRXZlbnQobmV3IEN1c3RvbUV2ZW50KCdhcHA6c2VhcmNoJywge1xuICAgICAgZGV0YWlsOiB7IHF1ZXJ5IH0sXG4gICAgICBidWJibGVzOiB0cnVlXG4gICAgfSkpO1xuICB9XG5cbiAgcHJpdmF0ZSBoYW5kbGVTdWJtaXQoZTogRXZlbnQpOiB2b2lkIHtcbiAgICBlLnByZXZlbnREZWZhdWx0KCk7XG5cbiAgICBjb25zdCBxdWVyeSA9IHRoaXMudmFsdWUudHJpbSgpO1xuICAgIGlmICghcXVlcnkpIHJldHVybjtcblxuICAgIC8vIEVtaXQgY3VzdG9tIGV2ZW50IGZvciBzZWFyY2ggc3VibWl0XG4gICAgZG9jdW1lbnQuZGlzcGF0Y2hFdmVudChuZXcgQ3VzdG9tRXZlbnQoJ2FwcDpzZWFyY2gtc3VibWl0Jywge1xuICAgICAgZGV0YWlsOiB7IHF1ZXJ5IH0sXG4gICAgICBidWJibGVzOiB0cnVlXG4gICAgfSkpO1xuICB9XG59XG4iLCAiLyoqXG4gKiBMb2NrIFNjcmVlbiBDb250cm9sbGVyXG4gKlxuICogSGFuZGxlcyBQSU4tYmFzZWQgbG9jayBzY3JlZW4gZnVuY3Rpb25hbGl0eVxuICovXG5cbmltcG9ydCB7IERyYXdlckNvbnRyb2xsZXIgfSBmcm9tICcuL2RyYXdlcnMnO1xuXG5leHBvcnQgY2xhc3MgTG9ja1NjcmVlbkNvbnRyb2xsZXIge1xuICBwcml2YXRlIHN0YXRpYyByZWFkb25seSBDT1JSRUNUX1BJTiA9ICcxMjM0JzsgLy8gRGVtbyBQSU5cblxuICBwcml2YXRlIGxvY2tTY3JlZW46IEhUTUxFbGVtZW50IHwgbnVsbCA9IG51bGw7XG4gIHByaXZhdGUgcGluSW5wdXQ6IEhUTUxFbGVtZW50IHwgbnVsbCA9IG51bGw7XG4gIHByaXZhdGUgcGluS2V5cGFkOiBIVE1MRWxlbWVudCB8IG51bGwgPSBudWxsO1xuICBwcml2YXRlIGxvY2tUaW1lRWw6IEhUTUxFbGVtZW50IHwgbnVsbCA9IG51bGw7XG4gIHByaXZhdGUgcGluRGlnaXRzOiBOb2RlTGlzdE9mPEhUTUxFbGVtZW50PiB8IG51bGwgPSBudWxsO1xuICBwcml2YXRlIGN1cnJlbnRQaW4gPSAnJztcbiAgcHJpdmF0ZSBkcmF3ZXJzOiBEcmF3ZXJDb250cm9sbGVyIHwgbnVsbCA9IG51bGw7XG5cbiAgY29uc3RydWN0b3IoZHJhd2Vycz86IERyYXdlckNvbnRyb2xsZXIpIHtcbiAgICB0aGlzLmRyYXdlcnMgPSBkcmF3ZXJzID8/IG51bGw7XG4gICAgdGhpcy5sb2NrU2NyZWVuID0gZG9jdW1lbnQuZ2V0RWxlbWVudEJ5SWQoJ2xvY2tTY3JlZW4nKTtcbiAgICB0aGlzLnBpbklucHV0ID0gZG9jdW1lbnQuZ2V0RWxlbWVudEJ5SWQoJ3BpbklucHV0Jyk7XG4gICAgdGhpcy5waW5LZXlwYWQgPSBkb2N1bWVudC5nZXRFbGVtZW50QnlJZCgncGluS2V5cGFkJyk7XG4gICAgdGhpcy5sb2NrVGltZUVsID0gZG9jdW1lbnQuZ2V0RWxlbWVudEJ5SWQoJ2xvY2tUaW1lJyk7XG4gICAgdGhpcy5waW5EaWdpdHMgPSB0aGlzLnBpbklucHV0Py5xdWVyeVNlbGVjdG9yQWxsPEhUTUxFbGVtZW50Pignc3dwLXBpbi1kaWdpdCcpID8/IG51bGw7XG5cbiAgICB0aGlzLnNldHVwTGlzdGVuZXJzKCk7XG4gIH1cblxuICAvKipcbiAgICogQ2hlY2sgaWYgbG9jayBzY3JlZW4gaXMgYWN0aXZlXG4gICAqL1xuICBnZXQgaXNBY3RpdmUoKTogYm9vbGVhbiB7XG4gICAgcmV0dXJuIHRoaXMubG9ja1NjcmVlbj8uY2xhc3NMaXN0LmNvbnRhaW5zKCdhY3RpdmUnKSA/PyBmYWxzZTtcbiAgfVxuXG4gIC8qKlxuICAgKiBTaG93IHRoZSBsb2NrIHNjcmVlblxuICAgKi9cbiAgc2hvdygpOiB2b2lkIHtcbiAgICB0aGlzLmRyYXdlcnM/LmNsb3NlQWxsKCk7XG5cbiAgICBpZiAodGhpcy5sb2NrU2NyZWVuKSB7XG4gICAgICB0aGlzLmxvY2tTY3JlZW4uY2xhc3NMaXN0LmFkZCgnYWN0aXZlJyk7XG4gICAgICBkb2N1bWVudC5ib2R5LnN0eWxlLm92ZXJmbG93ID0gJ2hpZGRlbic7XG4gICAgfVxuXG4gICAgdGhpcy5jdXJyZW50UGluID0gJyc7XG4gICAgdGhpcy51cGRhdGVEaXNwbGF5KCk7XG5cbiAgICAvLyBVcGRhdGUgbG9jayB0aW1lXG4gICAgaWYgKHRoaXMubG9ja1RpbWVFbCkge1xuICAgICAgdGhpcy5sb2NrVGltZUVsLnRleHRDb250ZW50ID0gYExcdTAwRTVzdCBrbC4gJHt0aGlzLmZvcm1hdFRpbWUoKX1gO1xuICAgIH1cbiAgfVxuXG4gIC8qKlxuICAgKiBIaWRlIHRoZSBsb2NrIHNjcmVlblxuICAgKi9cbiAgaGlkZSgpOiB2b2lkIHtcbiAgICBpZiAodGhpcy5sb2NrU2NyZWVuKSB7XG4gICAgICB0aGlzLmxvY2tTY3JlZW4uY2xhc3NMaXN0LnJlbW92ZSgnYWN0aXZlJyk7XG4gICAgICBkb2N1bWVudC5ib2R5LnN0eWxlLm92ZXJmbG93ID0gJyc7XG4gICAgfVxuXG4gICAgdGhpcy5jdXJyZW50UGluID0gJyc7XG4gICAgdGhpcy51cGRhdGVEaXNwbGF5KCk7XG4gIH1cblxuICBwcml2YXRlIGZvcm1hdFRpbWUoKTogc3RyaW5nIHtcbiAgICBjb25zdCBub3cgPSBuZXcgRGF0ZSgpO1xuICAgIGNvbnN0IGhvdXJzID0gbm93LmdldEhvdXJzKCkudG9TdHJpbmcoKS5wYWRTdGFydCgyLCAnMCcpO1xuICAgIGNvbnN0IG1pbnV0ZXMgPSBub3cuZ2V0TWludXRlcygpLnRvU3RyaW5nKCkucGFkU3RhcnQoMiwgJzAnKTtcbiAgICByZXR1cm4gYCR7aG91cnN9OiR7bWludXRlc31gO1xuICB9XG5cbiAgcHJpdmF0ZSB1cGRhdGVEaXNwbGF5KCk6IHZvaWQge1xuICAgIGlmICghdGhpcy5waW5EaWdpdHMpIHJldHVybjtcblxuICAgIHRoaXMucGluRGlnaXRzLmZvckVhY2goKGRpZ2l0LCBpbmRleCkgPT4ge1xuICAgICAgZGlnaXQuY2xhc3NMaXN0LnJlbW92ZSgnZmlsbGVkJywgJ2Vycm9yJyk7XG4gICAgICBpZiAoaW5kZXggPCB0aGlzLmN1cnJlbnRQaW4ubGVuZ3RoKSB7XG4gICAgICAgIGRpZ2l0LnRleHRDb250ZW50ID0gJ1x1MjAyMic7XG4gICAgICAgIGRpZ2l0LmNsYXNzTGlzdC5hZGQoJ2ZpbGxlZCcpO1xuICAgICAgfSBlbHNlIHtcbiAgICAgICAgZGlnaXQudGV4dENvbnRlbnQgPSAnJztcbiAgICAgIH1cbiAgICB9KTtcbiAgfVxuXG4gIHByaXZhdGUgc2hvd0Vycm9yKCk6IHZvaWQge1xuICAgIGlmICghdGhpcy5waW5EaWdpdHMpIHJldHVybjtcblxuICAgIHRoaXMucGluRGlnaXRzLmZvckVhY2goZGlnaXQgPT4gZGlnaXQuY2xhc3NMaXN0LmFkZCgnZXJyb3InKSk7XG5cbiAgICAvLyBTaGFrZSBhbmltYXRpb25cbiAgICB0aGlzLnBpbklucHV0Py5jbGFzc0xpc3QuYWRkKCdzaGFrZScpO1xuXG4gICAgc2V0VGltZW91dCgoKSA9PiB7XG4gICAgICB0aGlzLmN1cnJlbnRQaW4gPSAnJztcbiAgICAgIHRoaXMudXBkYXRlRGlzcGxheSgpO1xuICAgICAgdGhpcy5waW5JbnB1dD8uY2xhc3NMaXN0LnJlbW92ZSgnc2hha2UnKTtcbiAgICB9LCA1MDApO1xuICB9XG5cbiAgcHJpdmF0ZSB2ZXJpZnkoKTogdm9pZCB7XG4gICAgaWYgKHRoaXMuY3VycmVudFBpbiA9PT0gTG9ja1NjcmVlbkNvbnRyb2xsZXIuQ09SUkVDVF9QSU4pIHtcbiAgICAgIHRoaXMuaGlkZSgpO1xuICAgIH0gZWxzZSB7XG4gICAgICB0aGlzLnNob3dFcnJvcigpO1xuICAgIH1cbiAgfVxuXG4gIHByaXZhdGUgYWRkRGlnaXQoZGlnaXQ6IHN0cmluZyk6IHZvaWQge1xuICAgIGlmICh0aGlzLmN1cnJlbnRQaW4ubGVuZ3RoID49IDQpIHJldHVybjtcblxuICAgIHRoaXMuY3VycmVudFBpbiArPSBkaWdpdDtcbiAgICB0aGlzLnVwZGF0ZURpc3BsYXkoKTtcblxuICAgIC8vIEF1dG8tdmVyaWZ5IHdoZW4gNCBkaWdpdHMgZW50ZXJlZFxuICAgIGlmICh0aGlzLmN1cnJlbnRQaW4ubGVuZ3RoID09PSA0KSB7XG4gICAgICBzZXRUaW1lb3V0KCgpID0+IHRoaXMudmVyaWZ5KCksIDIwMCk7XG4gICAgfVxuICB9XG5cbiAgcHJpdmF0ZSByZW1vdmVEaWdpdCgpOiB2b2lkIHtcbiAgICBpZiAodGhpcy5jdXJyZW50UGluLmxlbmd0aCA9PT0gMCkgcmV0dXJuO1xuICAgIHRoaXMuY3VycmVudFBpbiA9IHRoaXMuY3VycmVudFBpbi5zbGljZSgwLCAtMSk7XG4gICAgdGhpcy51cGRhdGVEaXNwbGF5KCk7XG4gIH1cblxuICBwcml2YXRlIGNsZWFyUGluKCk6IHZvaWQge1xuICAgIHRoaXMuY3VycmVudFBpbiA9ICcnO1xuICAgIHRoaXMudXBkYXRlRGlzcGxheSgpO1xuICB9XG5cbiAgcHJpdmF0ZSBzZXR1cExpc3RlbmVycygpOiB2b2lkIHtcbiAgICAvLyBLZXlwYWQgY2xpY2sgaGFuZGxlclxuICAgIHRoaXMucGluS2V5cGFkPy5hZGRFdmVudExpc3RlbmVyKCdjbGljaycsIChlKSA9PiB0aGlzLmhhbmRsZUtleXBhZENsaWNrKGUpKTtcblxuICAgIC8vIEtleWJvYXJkIGlucHV0XG4gICAgZG9jdW1lbnQuYWRkRXZlbnRMaXN0ZW5lcigna2V5ZG93bicsIChlKSA9PiB0aGlzLmhhbmRsZUtleWJvYXJkKGUpKTtcblxuICAgIC8vIExvY2sgYnV0dG9uIGluIHNpZGViYXJcbiAgICBkb2N1bWVudC5xdWVyeVNlbGVjdG9yPEhUTUxFbGVtZW50Pignc3dwLXNpZGUtbWVudS1hY3Rpb24ubG9jaycpXG4gICAgICA/LmFkZEV2ZW50TGlzdGVuZXIoJ2NsaWNrJywgKCkgPT4gdGhpcy5zaG93KCkpO1xuICB9XG5cbiAgcHJpdmF0ZSBoYW5kbGVLZXlwYWRDbGljayhlOiBFdmVudCk6IHZvaWQge1xuICAgIGNvbnN0IHRhcmdldCA9IGUudGFyZ2V0IGFzIEhUTUxFbGVtZW50O1xuICAgIGNvbnN0IGtleSA9IHRhcmdldC5jbG9zZXN0PEhUTUxFbGVtZW50Pignc3dwLXBpbi1rZXknKTtcblxuICAgIGlmICgha2V5KSByZXR1cm47XG5cbiAgICBjb25zdCBkaWdpdCA9IGtleS5kYXRhc2V0LmRpZ2l0O1xuICAgIGNvbnN0IGFjdGlvbiA9IGtleS5kYXRhc2V0LmFjdGlvbjtcblxuICAgIGlmIChkaWdpdCkge1xuICAgICAgdGhpcy5hZGREaWdpdChkaWdpdCk7XG4gICAgfSBlbHNlIGlmIChhY3Rpb24gPT09ICdiYWNrc3BhY2UnKSB7XG4gICAgICB0aGlzLnJlbW92ZURpZ2l0KCk7XG4gICAgfSBlbHNlIGlmIChhY3Rpb24gPT09ICdjbGVhcicpIHtcbiAgICAgIHRoaXMuY2xlYXJQaW4oKTtcbiAgICB9XG4gIH1cblxuICBwcml2YXRlIGhhbmRsZUtleWJvYXJkKGU6IEtleWJvYXJkRXZlbnQpOiB2b2lkIHtcbiAgICBpZiAoIXRoaXMuaXNBY3RpdmUpIHJldHVybjtcblxuICAgIC8vIFByZXZlbnQgZGVmYXVsdCB0byBhdm9pZCBvdGhlciBpbnRlcmFjdGlvbnNcbiAgICBlLnByZXZlbnREZWZhdWx0KCk7XG5cbiAgICBpZiAoZS5rZXkgPj0gJzAnICYmIGUua2V5IDw9ICc5Jykge1xuICAgICAgdGhpcy5hZGREaWdpdChlLmtleSk7XG4gICAgfSBlbHNlIGlmIChlLmtleSA9PT0gJ0JhY2tzcGFjZScpIHtcbiAgICAgIHRoaXMucmVtb3ZlRGlnaXQoKTtcbiAgICB9IGVsc2UgaWYgKGUua2V5ID09PSAnRXNjYXBlJykge1xuICAgICAgdGhpcy5jbGVhclBpbigpO1xuICAgIH1cbiAgfVxufVxuIiwgIi8qKlxuICogS2Fzc2UgQ29udHJvbGxlclxuICpcbiAqIEhhbmRsZXMgdGFiIHN3aXRjaGluZywgY2FzaCBjYWxjdWxhdGlvbnMsIGFuZCBmb3JtIGludGVyYWN0aW9uc1xuICogZm9yIHRoZSBLYXNzZSAoQ2FzaCBSZWdpc3RlcikgcGFnZS5cbiAqL1xuXG5leHBvcnQgY2xhc3MgS2Fzc2VDb250cm9sbGVyIHtcbiAgLy8gQmFzZSB2YWx1ZXMgKGZyb20gc3lzdGVtIC0gd291bGQgY29tZSBmcm9tIHNlcnZlciBpbiByZWFsIGFwcClcbiAgcHJpdmF0ZSByZWFkb25seSBzdGFydEJhbGFuY2UgPSAyMDAwO1xuICBwcml2YXRlIHJlYWRvbmx5IGNhc2hTYWxlcyA9IDM1NDA7XG5cbiAgY29uc3RydWN0b3IoKSB7XG4gICAgdGhpcy5zZXR1cFRhYnMoKTtcbiAgICB0aGlzLnNldHVwQ2FzaENhbGN1bGF0aW9uKCk7XG4gICAgdGhpcy5zZXR1cENoZWNrYm94U2VsZWN0aW9uKCk7XG4gICAgdGhpcy5zZXR1cEFwcHJvdmFsQ2hlY2tib3goKTtcbiAgICB0aGlzLnNldHVwRGF0ZUZpbHRlcnMoKTtcbiAgICB0aGlzLnNldHVwUm93VG9nZ2xlKCk7XG4gICAgdGhpcy5zZXR1cERyYWZ0Um93Q2xpY2soKTtcbiAgfVxuXG4gIC8qKlxuICAgKiBTZXR1cCB0YWIgc3dpdGNoaW5nIGZ1bmN0aW9uYWxpdHlcbiAgICovXG4gIHByaXZhdGUgc2V0dXBUYWJzKCk6IHZvaWQge1xuICAgIGNvbnN0IHRhYnMgPSBkb2N1bWVudC5xdWVyeVNlbGVjdG9yQWxsPEhUTUxFbGVtZW50Pignc3dwLXRhYltkYXRhLXRhYl0nKTtcblxuICAgIHRhYnMuZm9yRWFjaCh0YWIgPT4ge1xuICAgICAgdGFiLmFkZEV2ZW50TGlzdGVuZXIoJ2NsaWNrJywgKCkgPT4ge1xuICAgICAgICBjb25zdCB0YXJnZXRUYWIgPSB0YWIuZGF0YXNldC50YWI7XG4gICAgICAgIGlmICh0YXJnZXRUYWIpIHtcbiAgICAgICAgICB0aGlzLnN3aXRjaFRvVGFiKHRhcmdldFRhYik7XG4gICAgICAgIH1cbiAgICAgIH0pO1xuICAgIH0pO1xuICB9XG5cbiAgLyoqXG4gICAqIFN3aXRjaCB0byBhIHNwZWNpZmljIHRhYiBieSBuYW1lXG4gICAqL1xuICBwcml2YXRlIHN3aXRjaFRvVGFiKHRhcmdldFRhYjogc3RyaW5nKTogdm9pZCB7XG4gICAgY29uc3QgdGFicyA9IGRvY3VtZW50LnF1ZXJ5U2VsZWN0b3JBbGw8SFRNTEVsZW1lbnQ+KCdzd3AtdGFiW2RhdGEtdGFiXScpO1xuICAgIGNvbnN0IGNvbnRlbnRzID0gZG9jdW1lbnQucXVlcnlTZWxlY3RvckFsbDxIVE1MRWxlbWVudD4oJ3N3cC10YWItY29udGVudFtkYXRhLXRhYl0nKTtcbiAgICBjb25zdCBzdGF0c0JhcnMgPSBkb2N1bWVudC5xdWVyeVNlbGVjdG9yQWxsPEhUTUxFbGVtZW50Pignc3dwLWthc3NlLXN0YXRzW2RhdGEtZm9yLXRhYl0nKTtcblxuICAgIC8vIFVwZGF0ZSB0YWIgc3RhdGVzXG4gICAgdGFicy5mb3JFYWNoKHQgPT4ge1xuICAgICAgaWYgKHQuZGF0YXNldC50YWIgPT09IHRhcmdldFRhYikge1xuICAgICAgICB0LmNsYXNzTGlzdC5hZGQoJ2FjdGl2ZScpO1xuICAgICAgfSBlbHNlIHtcbiAgICAgICAgdC5jbGFzc0xpc3QucmVtb3ZlKCdhY3RpdmUnKTtcbiAgICAgIH1cbiAgICB9KTtcblxuICAgIC8vIFVwZGF0ZSBjb250ZW50IHZpc2liaWxpdHlcbiAgICBjb250ZW50cy5mb3JFYWNoKGNvbnRlbnQgPT4ge1xuICAgICAgaWYgKGNvbnRlbnQuZGF0YXNldC50YWIgPT09IHRhcmdldFRhYikge1xuICAgICAgICBjb250ZW50LmNsYXNzTGlzdC5hZGQoJ2FjdGl2ZScpO1xuICAgICAgfSBlbHNlIHtcbiAgICAgICAgY29udGVudC5jbGFzc0xpc3QucmVtb3ZlKCdhY3RpdmUnKTtcbiAgICAgIH1cbiAgICB9KTtcblxuICAgIC8vIFVwZGF0ZSBzdGF0cyBiYXIgdmlzaWJpbGl0eVxuICAgIHN0YXRzQmFycy5mb3JFYWNoKHN0YXRzID0+IHtcbiAgICAgIGlmIChzdGF0cy5kYXRhc2V0LmZvclRhYiA9PT0gdGFyZ2V0VGFiKSB7XG4gICAgICAgIHN0YXRzLmNsYXNzTGlzdC5hZGQoJ2FjdGl2ZScpO1xuICAgICAgfSBlbHNlIHtcbiAgICAgICAgc3RhdHMuY2xhc3NMaXN0LnJlbW92ZSgnYWN0aXZlJyk7XG4gICAgICB9XG4gICAgfSk7XG4gIH1cblxuICAvKipcbiAgICogU2V0dXAgY2FzaCBjYWxjdWxhdGlvbiB3aXRoIHJlYWwtdGltZSB1cGRhdGVzXG4gICAqL1xuICBwcml2YXRlIHNldHVwQ2FzaENhbGN1bGF0aW9uKCk6IHZvaWQge1xuICAgIGNvbnN0IHBheW91dHNJbnB1dCA9IGRvY3VtZW50LmdldEVsZW1lbnRCeUlkKCdwYXlvdXRzJykgYXMgSFRNTElucHV0RWxlbWVudDtcbiAgICBjb25zdCB0b0JhbmtJbnB1dCA9IGRvY3VtZW50LmdldEVsZW1lbnRCeUlkKCd0b0JhbmsnKSBhcyBIVE1MSW5wdXRFbGVtZW50O1xuICAgIGNvbnN0IGFjdHVhbENhc2hJbnB1dCA9IGRvY3VtZW50LmdldEVsZW1lbnRCeUlkKCdhY3R1YWxDYXNoJykgYXMgSFRNTElucHV0RWxlbWVudDtcblxuICAgIGlmICghcGF5b3V0c0lucHV0IHx8ICF0b0JhbmtJbnB1dCB8fCAhYWN0dWFsQ2FzaElucHV0KSByZXR1cm47XG5cbiAgICBjb25zdCBjYWxjdWxhdGUgPSAoKSA9PiB0aGlzLmNhbGN1bGF0ZUNhc2gocGF5b3V0c0lucHV0LCB0b0JhbmtJbnB1dCwgYWN0dWFsQ2FzaElucHV0KTtcblxuICAgIHBheW91dHNJbnB1dC5hZGRFdmVudExpc3RlbmVyKCdpbnB1dCcsIGNhbGN1bGF0ZSk7XG4gICAgdG9CYW5rSW5wdXQuYWRkRXZlbnRMaXN0ZW5lcignaW5wdXQnLCBjYWxjdWxhdGUpO1xuICAgIGFjdHVhbENhc2hJbnB1dC5hZGRFdmVudExpc3RlbmVyKCdpbnB1dCcsIGNhbGN1bGF0ZSk7XG5cbiAgICAvLyBJbml0aWFsIGNhbGN1bGF0aW9uXG4gICAgY2FsY3VsYXRlKCk7XG4gIH1cblxuICAvKipcbiAgICogQ2FsY3VsYXRlIGV4cGVjdGVkIGNhc2ggYW5kIGRpZmZlcmVuY2VcbiAgICovXG4gIHByaXZhdGUgY2FsY3VsYXRlQ2FzaChcbiAgICBwYXlvdXRzSW5wdXQ6IEhUTUxJbnB1dEVsZW1lbnQsXG4gICAgdG9CYW5rSW5wdXQ6IEhUTUxJbnB1dEVsZW1lbnQsXG4gICAgYWN0dWFsQ2FzaElucHV0OiBIVE1MSW5wdXRFbGVtZW50XG4gICk6IHZvaWQge1xuICAgIGNvbnN0IHBheW91dHMgPSB0aGlzLnBhcnNlTnVtYmVyKHBheW91dHNJbnB1dC52YWx1ZSk7XG4gICAgY29uc3QgdG9CYW5rID0gdGhpcy5wYXJzZU51bWJlcih0b0JhbmtJbnB1dC52YWx1ZSk7XG4gICAgY29uc3QgYWN0dWFsID0gdGhpcy5wYXJzZU51bWJlcihhY3R1YWxDYXNoSW5wdXQudmFsdWUpO1xuXG4gICAgLy8gRXhwZWN0ZWQgPSBzdGFydCArIHNhbGVzIC0gcGF5b3V0cyAtIHRvIGJhbmtcbiAgICBjb25zdCBleHBlY3RlZENhc2ggPSB0aGlzLnN0YXJ0QmFsYW5jZSArIHRoaXMuY2FzaFNhbGVzIC0gcGF5b3V0cyAtIHRvQmFuaztcblxuICAgIGNvbnN0IGV4cGVjdGVkRWxlbWVudCA9IGRvY3VtZW50LmdldEVsZW1lbnRCeUlkKCdleHBlY3RlZENhc2gnKTtcbiAgICBpZiAoZXhwZWN0ZWRFbGVtZW50KSB7XG4gICAgICBleHBlY3RlZEVsZW1lbnQudGV4dENvbnRlbnQgPSB0aGlzLmZvcm1hdE51bWJlcihleHBlY3RlZENhc2gpO1xuICAgIH1cblxuICAgIC8vIENhbGN1bGF0ZSBhbmQgZGlzcGxheSBkaWZmZXJlbmNlXG4gICAgdGhpcy51cGRhdGVEaWZmZXJlbmNlKGFjdHVhbCwgZXhwZWN0ZWRDYXNoLCBhY3R1YWxDYXNoSW5wdXQudmFsdWUpO1xuICB9XG5cbiAgLyoqXG4gICAqIFVwZGF0ZSBkaWZmZXJlbmNlIGJveCB3aXRoIGNvbG9yIGNvZGluZ1xuICAgKi9cbiAgcHJpdmF0ZSB1cGRhdGVEaWZmZXJlbmNlKGFjdHVhbDogbnVtYmVyLCBleHBlY3RlZDogbnVtYmVyLCByYXdWYWx1ZTogc3RyaW5nKTogdm9pZCB7XG4gICAgY29uc3QgYm94ID0gZG9jdW1lbnQuZ2V0RWxlbWVudEJ5SWQoJ2RpZmZlcmVuY2VCb3gnKTtcbiAgICBjb25zdCB2YWx1ZSA9IGRvY3VtZW50LmdldEVsZW1lbnRCeUlkKCdkaWZmZXJlbmNlVmFsdWUnKTtcbiAgICBpZiAoIWJveCB8fCAhdmFsdWUpIHJldHVybjtcblxuICAgIGNvbnN0IGRpZmYgPSBhY3R1YWwgLSBleHBlY3RlZDtcblxuICAgIC8vIFJlbW92ZSBhbGwgc3RhdGUgY2xhc3Nlc1xuICAgIGJveC5jbGFzc0xpc3QucmVtb3ZlKCdwb3NpdGl2ZScsICduZWdhdGl2ZScsICduZXV0cmFsJyk7XG5cbiAgICBpZiAoYWN0dWFsID09PSAwICYmIHJhd1ZhbHVlID09PSAnJykge1xuICAgICAgLy8gTm8gaW5wdXQgeWV0XG4gICAgICB2YWx1ZS50ZXh0Q29udGVudCA9ICdcdTIwMTMga3InO1xuICAgICAgYm94LmNsYXNzTGlzdC5hZGQoJ25ldXRyYWwnKTtcbiAgICB9IGVsc2UgaWYgKGRpZmYgPiAwKSB7XG4gICAgICAvLyBNb3JlIGNhc2ggdGhhbiBleHBlY3RlZFxuICAgICAgdmFsdWUudGV4dENvbnRlbnQgPSAnKycgKyB0aGlzLmZvcm1hdE51bWJlcihkaWZmKSArICcga3InO1xuICAgICAgYm94LmNsYXNzTGlzdC5hZGQoJ3Bvc2l0aXZlJyk7XG4gICAgfSBlbHNlIGlmIChkaWZmIDwgMCkge1xuICAgICAgLy8gTGVzcyBjYXNoIHRoYW4gZXhwZWN0ZWRcbiAgICAgIHZhbHVlLnRleHRDb250ZW50ID0gdGhpcy5mb3JtYXROdW1iZXIoZGlmZikgKyAnIGtyJztcbiAgICAgIGJveC5jbGFzc0xpc3QuYWRkKCduZWdhdGl2ZScpO1xuICAgIH0gZWxzZSB7XG4gICAgICAvLyBFeGFjdCBtYXRjaFxuICAgICAgdmFsdWUudGV4dENvbnRlbnQgPSAnMCwwMCBrcic7XG4gICAgICBib3guY2xhc3NMaXN0LmFkZCgnbmV1dHJhbCcpO1xuICAgIH1cbiAgfVxuXG4gIC8qKlxuICAgKiBTZXR1cCBjaGVja2JveCBzZWxlY3Rpb24gZm9yIHRhYmxlIHJvd3NcbiAgICovXG4gIHByaXZhdGUgc2V0dXBDaGVja2JveFNlbGVjdGlvbigpOiB2b2lkIHtcbiAgICBjb25zdCBzZWxlY3RBbGwgPSBkb2N1bWVudC5nZXRFbGVtZW50QnlJZCgnc2VsZWN0QWxsJykgYXMgSFRNTElucHV0RWxlbWVudDtcbiAgICBjb25zdCByb3dDaGVja2JveGVzID0gZG9jdW1lbnQucXVlcnlTZWxlY3RvckFsbDxIVE1MSW5wdXRFbGVtZW50PignLnJvdy1zZWxlY3QnKTtcbiAgICBjb25zdCBleHBvcnRCdG4gPSBkb2N1bWVudC5nZXRFbGVtZW50QnlJZCgnZXhwb3J0QnRuJykgYXMgSFRNTEJ1dHRvbkVsZW1lbnQ7XG4gICAgY29uc3Qgc2VsZWN0aW9uQ291bnQgPSBkb2N1bWVudC5nZXRFbGVtZW50QnlJZCgnc2VsZWN0aW9uQ291bnQnKTtcblxuICAgIGlmICghc2VsZWN0QWxsIHx8ICFleHBvcnRCdG4gfHwgIXNlbGVjdGlvbkNvdW50KSByZXR1cm47XG5cbiAgICBjb25zdCB1cGRhdGVTZWxlY3Rpb24gPSAoKSA9PiB7XG4gICAgICBjb25zdCBjaGVja2VkID0gZG9jdW1lbnQucXVlcnlTZWxlY3RvckFsbDxIVE1MSW5wdXRFbGVtZW50PignLnJvdy1zZWxlY3Q6Y2hlY2tlZCcpO1xuICAgICAgY29uc3QgY291bnQgPSBjaGVja2VkLmxlbmd0aDtcblxuICAgICAgc2VsZWN0aW9uQ291bnQudGV4dENvbnRlbnQgPSBjb3VudCA9PT0gMCA/ICcwIHZhbGd0JyA6IGAke2NvdW50fSB2YWxndGA7XG4gICAgICBleHBvcnRCdG4uZGlzYWJsZWQgPSBjb3VudCA9PT0gMDtcblxuICAgICAgLy8gVXBkYXRlIHNlbGVjdCBhbGwgc3RhdGVcbiAgICAgIHNlbGVjdEFsbC5jaGVja2VkID0gY291bnQgPT09IHJvd0NoZWNrYm94ZXMubGVuZ3RoICYmIGNvdW50ID4gMDtcbiAgICAgIHNlbGVjdEFsbC5pbmRldGVybWluYXRlID0gY291bnQgPiAwICYmIGNvdW50IDwgcm93Q2hlY2tib3hlcy5sZW5ndGg7XG4gICAgfTtcblxuICAgIHNlbGVjdEFsbC5hZGRFdmVudExpc3RlbmVyKCdjaGFuZ2UnLCAoKSA9PiB7XG4gICAgICByb3dDaGVja2JveGVzLmZvckVhY2goY2IgPT4gY2IuY2hlY2tlZCA9IHNlbGVjdEFsbC5jaGVja2VkKTtcbiAgICAgIHVwZGF0ZVNlbGVjdGlvbigpO1xuICAgIH0pO1xuXG4gICAgcm93Q2hlY2tib3hlcy5mb3JFYWNoKGNiID0+IHtcbiAgICAgIGNiLmFkZEV2ZW50TGlzdGVuZXIoJ2NoYW5nZScsIHVwZGF0ZVNlbGVjdGlvbik7XG4gICAgICAvLyBTdG9wIGNsaWNrIGZyb20gYnViYmxpbmcgdG8gcm93XG4gICAgICBjYi5hZGRFdmVudExpc3RlbmVyKCdjbGljaycsIGUgPT4gZS5zdG9wUHJvcGFnYXRpb24oKSk7XG4gICAgfSk7XG4gIH1cblxuICAvKipcbiAgICogU2V0dXAgYXBwcm92YWwgY2hlY2tib3ggdG8gZW5hYmxlL2Rpc2FibGUgYXBwcm92ZSBidXR0b25cbiAgICovXG4gIHByaXZhdGUgc2V0dXBBcHByb3ZhbENoZWNrYm94KCk6IHZvaWQge1xuICAgIGNvbnN0IGNoZWNrYm94ID0gZG9jdW1lbnQuZ2V0RWxlbWVudEJ5SWQoJ2NvbmZpcm1DaGVja2JveCcpIGFzIEhUTUxJbnB1dEVsZW1lbnQ7XG4gICAgY29uc3QgYXBwcm92ZUJ0biA9IGRvY3VtZW50LmdldEVsZW1lbnRCeUlkKCdhcHByb3ZlQnRuJykgYXMgSFRNTEJ1dHRvbkVsZW1lbnQ7XG5cbiAgICBpZiAoIWNoZWNrYm94IHx8ICFhcHByb3ZlQnRuKSByZXR1cm47XG5cbiAgICBjaGVja2JveC5hZGRFdmVudExpc3RlbmVyKCdjaGFuZ2UnLCAoKSA9PiB7XG4gICAgICBhcHByb3ZlQnRuLmRpc2FibGVkID0gIWNoZWNrYm94LmNoZWNrZWQ7XG4gICAgfSk7XG4gIH1cblxuICAvKipcbiAgICogU2V0dXAgZGF0ZSBmaWx0ZXIgZGVmYXVsdHMgKGxhc3QgMzAgZGF5cylcbiAgICovXG4gIHByaXZhdGUgc2V0dXBEYXRlRmlsdGVycygpOiB2b2lkIHtcbiAgICBjb25zdCBkYXRlRnJvbSA9IGRvY3VtZW50LmdldEVsZW1lbnRCeUlkKCdkYXRlRnJvbScpIGFzIEhUTUxJbnB1dEVsZW1lbnQ7XG4gICAgY29uc3QgZGF0ZVRvID0gZG9jdW1lbnQuZ2V0RWxlbWVudEJ5SWQoJ2RhdGVUbycpIGFzIEhUTUxJbnB1dEVsZW1lbnQ7XG5cbiAgICBpZiAoIWRhdGVGcm9tIHx8ICFkYXRlVG8pIHJldHVybjtcblxuICAgIGNvbnN0IHRvZGF5ID0gbmV3IERhdGUoKTtcbiAgICBjb25zdCB0aGlydHlEYXlzQWdvID0gbmV3IERhdGUodG9kYXkpO1xuICAgIHRoaXJ0eURheXNBZ28uc2V0RGF0ZSh0b2RheS5nZXREYXRlKCkgLSAzMCk7XG5cbiAgICBkYXRlVG8udmFsdWUgPSB0aGlzLmZvcm1hdERhdGVJU08odG9kYXkpO1xuICAgIGRhdGVGcm9tLnZhbHVlID0gdGhpcy5mb3JtYXREYXRlSVNPKHRoaXJ0eURheXNBZ28pO1xuICB9XG5cbiAgLyoqXG4gICAqIEZvcm1hdCBudW1iZXIgYXMgRGFuaXNoIGN1cnJlbmN5XG4gICAqL1xuICBwcml2YXRlIGZvcm1hdE51bWJlcihudW06IG51bWJlcik6IHN0cmluZyB7XG4gICAgcmV0dXJuIG51bS50b0xvY2FsZVN0cmluZygnZGEtREsnLCB7XG4gICAgICBtaW5pbXVtRnJhY3Rpb25EaWdpdHM6IDIsXG4gICAgICBtYXhpbXVtRnJhY3Rpb25EaWdpdHM6IDJcbiAgICB9KTtcbiAgfVxuXG4gIC8qKlxuICAgKiBQYXJzZSBEYW5pc2ggbnVtYmVyIGZvcm1hdFxuICAgKi9cbiAgcHJpdmF0ZSBwYXJzZU51bWJlcihzdHI6IHN0cmluZyk6IG51bWJlciB7XG4gICAgaWYgKCFzdHIpIHJldHVybiAwO1xuICAgIHJldHVybiBwYXJzZUZsb2F0KHN0ci5yZXBsYWNlKC9cXC4vZywgJycpLnJlcGxhY2UoJywnLCAnLicpKSB8fCAwO1xuICB9XG5cbiAgLyoqXG4gICAqIEZvcm1hdCBkYXRlIGFzIElTTyBzdHJpbmcgKFlZWVktTU0tREQpXG4gICAqL1xuICBwcml2YXRlIGZvcm1hdERhdGVJU08oZGF0ZTogRGF0ZSk6IHN0cmluZyB7XG4gICAgcmV0dXJuIGRhdGUudG9JU09TdHJpbmcoKS5zcGxpdCgnVCcpWzBdO1xuICB9XG5cbiAgLyoqXG4gICAqIFNldHVwIHJvdyB0b2dnbGUgZm9yIGV4cGFuZGFibGUgZGV0YWlsc1xuICAgKi9cbiAgcHJpdmF0ZSBzZXR1cFJvd1RvZ2dsZSgpOiB2b2lkIHtcbiAgICBjb25zdCByb3dzID0gZG9jdW1lbnQucXVlcnlTZWxlY3RvckFsbDxIVE1MRWxlbWVudD4oJ3N3cC1rYXNzZS10YWJsZS1yb3dbZGF0YS1pZF06bm90KC5kcmFmdC1yb3cpJyk7XG5cbiAgICByb3dzLmZvckVhY2gocm93ID0+IHtcbiAgICAgIGNvbnN0IHJvd0lkID0gcm93LmdldEF0dHJpYnV0ZSgnZGF0YS1pZCcpO1xuICAgICAgaWYgKCFyb3dJZCkgcmV0dXJuO1xuXG4gICAgICBjb25zdCBkZXRhaWwgPSBkb2N1bWVudC5xdWVyeVNlbGVjdG9yPEhUTUxFbGVtZW50Pihgc3dwLWthc3NlLXJvdy1kZXRhaWxbZGF0YS1mb3I9XCIke3Jvd0lkfVwiXWApO1xuICAgICAgaWYgKCFkZXRhaWwpIHJldHVybjtcblxuICAgICAgcm93LmFkZEV2ZW50TGlzdGVuZXIoJ2NsaWNrJywgKGUpID0+IHtcbiAgICAgICAgLy8gRG9uJ3QgdG9nZ2xlIGlmIGNsaWNraW5nIG9uIGNoZWNrYm94XG4gICAgICAgIGlmICgoZS50YXJnZXQgYXMgSFRNTEVsZW1lbnQpLmNsb3Nlc3QoJ2lucHV0W3R5cGU9XCJjaGVja2JveFwiXScpKSByZXR1cm47XG5cbiAgICAgICAgY29uc3QgaWNvbiA9IHJvdy5xdWVyeVNlbGVjdG9yKCdzd3Atcm93LXRvZ2dsZSBpJyk7XG4gICAgICAgIGNvbnN0IGlzRXhwYW5kZWQgPSByb3cuY2xhc3NMaXN0LmNvbnRhaW5zKCdleHBhbmRlZCcpO1xuXG4gICAgICAgIC8vIENsb3NlIG90aGVyIGV4cGFuZGVkIHJvd3NcbiAgICAgICAgZG9jdW1lbnQucXVlcnlTZWxlY3RvckFsbCgnc3dwLWthc3NlLXRhYmxlLXJvdy5leHBhbmRlZCcpLmZvckVhY2gociA9PiB7XG4gICAgICAgICAgaWYgKHIgIT09IHJvdykge1xuICAgICAgICAgICAgY29uc3Qgb3RoZXJJZCA9IHIuZ2V0QXR0cmlidXRlKCdkYXRhLWlkJyk7XG4gICAgICAgICAgICBpZiAob3RoZXJJZCkge1xuICAgICAgICAgICAgICBjb25zdCBvdGhlckRldGFpbCA9IGRvY3VtZW50LnF1ZXJ5U2VsZWN0b3I8SFRNTEVsZW1lbnQ+KGBzd3Ata2Fzc2Utcm93LWRldGFpbFtkYXRhLWZvcj1cIiR7b3RoZXJJZH1cIl1gKTtcbiAgICAgICAgICAgICAgY29uc3Qgb3RoZXJJY29uID0gci5xdWVyeVNlbGVjdG9yKCdzd3Atcm93LXRvZ2dsZSBpJyk7XG4gICAgICAgICAgICAgIGlmIChvdGhlckRldGFpbCAmJiBvdGhlckljb24pIHtcbiAgICAgICAgICAgICAgICB0aGlzLmNvbGxhcHNlUm93KHIsIG90aGVyRGV0YWlsLCBvdGhlckljb24gYXMgSFRNTEVsZW1lbnQpO1xuICAgICAgICAgICAgICB9XG4gICAgICAgICAgICB9XG4gICAgICAgICAgfVxuICAgICAgICB9KTtcblxuICAgICAgICAvLyBUb2dnbGUgY3VycmVudCByb3dcbiAgICAgICAgaWYgKGlzRXhwYW5kZWQpIHtcbiAgICAgICAgICB0aGlzLmNvbGxhcHNlUm93KHJvdywgZGV0YWlsLCBpY29uKTtcbiAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICB0aGlzLmV4cGFuZFJvdyhyb3csIGRldGFpbCwgaWNvbik7XG4gICAgICAgIH1cbiAgICAgIH0pO1xuICAgIH0pO1xuICB9XG5cbiAgLyoqXG4gICAqIEV4cGFuZCBhIHJvdyB3aXRoIGFuaW1hdGlvblxuICAgKi9cbiAgcHJpdmF0ZSBleHBhbmRSb3cocm93OiBFbGVtZW50LCBkZXRhaWw6IEhUTUxFbGVtZW50LCBpY29uOiBFbGVtZW50IHwgbnVsbCk6IHZvaWQge1xuICAgIHJvdy5jbGFzc0xpc3QuYWRkKCdleHBhbmRlZCcpO1xuICAgIGRldGFpbC5jbGFzc0xpc3QuYWRkKCdleHBhbmRlZCcpO1xuXG4gICAgLy8gQW5pbWF0ZSBpY29uIHJvdGF0aW9uXG4gICAgaWNvbj8uYW5pbWF0ZShbXG4gICAgICB7IHRyYW5zZm9ybTogJ3JvdGF0ZSgwZGVnKScgfSxcbiAgICAgIHsgdHJhbnNmb3JtOiAncm90YXRlKDkwZGVnKScgfVxuICAgIF0sIHtcbiAgICAgIGR1cmF0aW9uOiAyMDAsXG4gICAgICBlYXNpbmc6ICdlYXNlLW91dCcsXG4gICAgICBmaWxsOiAnZm9yd2FyZHMnXG4gICAgfSk7XG5cbiAgICAvLyBBbmltYXRlIGRldGFpbCBleHBhbnNpb25cbiAgICBjb25zdCBjb250ZW50ID0gZGV0YWlsLnF1ZXJ5U2VsZWN0b3IoJ3N3cC1yb3ctZGV0YWlsLWNvbnRlbnQnKSBhcyBIVE1MRWxlbWVudDtcbiAgICBpZiAoY29udGVudCkge1xuICAgICAgY29uc3QgaGVpZ2h0ID0gY29udGVudC5vZmZzZXRIZWlnaHQ7XG4gICAgICBkZXRhaWwuYW5pbWF0ZShbXG4gICAgICAgIHsgaGVpZ2h0OiAnMHB4Jywgb3BhY2l0eTogMCB9LFxuICAgICAgICB7IGhlaWdodDogYCR7aGVpZ2h0fXB4YCwgb3BhY2l0eTogMSB9XG4gICAgICBdLCB7XG4gICAgICAgIGR1cmF0aW9uOiAyNTAsXG4gICAgICAgIGVhc2luZzogJ2Vhc2Utb3V0JyxcbiAgICAgICAgZmlsbDogJ2ZvcndhcmRzJ1xuICAgICAgfSk7XG4gICAgfVxuICB9XG5cbiAgLyoqXG4gICAqIENvbGxhcHNlIGEgcm93IHdpdGggYW5pbWF0aW9uXG4gICAqL1xuICBwcml2YXRlIGNvbGxhcHNlUm93KHJvdzogRWxlbWVudCwgZGV0YWlsOiBIVE1MRWxlbWVudCwgaWNvbjogRWxlbWVudCB8IG51bGwpOiB2b2lkIHtcbiAgICAvLyBBbmltYXRlIGljb24gcm90YXRpb25cbiAgICBpY29uPy5hbmltYXRlKFtcbiAgICAgIHsgdHJhbnNmb3JtOiAncm90YXRlKDkwZGVnKScgfSxcbiAgICAgIHsgdHJhbnNmb3JtOiAncm90YXRlKDBkZWcpJyB9XG4gICAgXSwge1xuICAgICAgZHVyYXRpb246IDIwMCxcbiAgICAgIGVhc2luZzogJ2Vhc2Utb3V0JyxcbiAgICAgIGZpbGw6ICdmb3J3YXJkcydcbiAgICB9KTtcblxuICAgIC8vIEFuaW1hdGUgZGV0YWlsIGNvbGxhcHNlXG4gICAgY29uc3QgY29udGVudCA9IGRldGFpbC5xdWVyeVNlbGVjdG9yKCdzd3Atcm93LWRldGFpbC1jb250ZW50JykgYXMgSFRNTEVsZW1lbnQ7XG4gICAgaWYgKGNvbnRlbnQpIHtcbiAgICAgIGNvbnN0IGhlaWdodCA9IGNvbnRlbnQub2Zmc2V0SGVpZ2h0O1xuICAgICAgY29uc3QgYW5pbWF0aW9uID0gZGV0YWlsLmFuaW1hdGUoW1xuICAgICAgICB7IGhlaWdodDogYCR7aGVpZ2h0fXB4YCwgb3BhY2l0eTogMSB9LFxuICAgICAgICB7IGhlaWdodDogJzBweCcsIG9wYWNpdHk6IDAgfVxuICAgICAgXSwge1xuICAgICAgICBkdXJhdGlvbjogMjAwLFxuICAgICAgICBlYXNpbmc6ICdlYXNlLW91dCcsXG4gICAgICAgIGZpbGw6ICdmb3J3YXJkcydcbiAgICAgIH0pO1xuXG4gICAgICBhbmltYXRpb24ub25maW5pc2ggPSAoKSA9PiB7XG4gICAgICAgIHJvdy5jbGFzc0xpc3QucmVtb3ZlKCdleHBhbmRlZCcpO1xuICAgICAgICBkZXRhaWwuY2xhc3NMaXN0LnJlbW92ZSgnZXhwYW5kZWQnKTtcbiAgICAgIH07XG4gICAgfSBlbHNlIHtcbiAgICAgIHJvdy5jbGFzc0xpc3QucmVtb3ZlKCdleHBhbmRlZCcpO1xuICAgICAgZGV0YWlsLmNsYXNzTGlzdC5yZW1vdmUoJ2V4cGFuZGVkJyk7XG4gICAgfVxuICB9XG5cbiAgLyoqXG4gICAqIFNldHVwIGRyYWZ0IHJvdyBjbGljayB0byBuYXZpZ2F0ZSB0byBLYXNzZWFmc3RlbW5pbmcgdGFiXG4gICAqL1xuICBwcml2YXRlIHNldHVwRHJhZnRSb3dDbGljaygpOiB2b2lkIHtcbiAgICBjb25zdCBkcmFmdFJvdyA9IGRvY3VtZW50LnF1ZXJ5U2VsZWN0b3I8SFRNTEVsZW1lbnQ+KCdzd3Ata2Fzc2UtdGFibGUtcm93LmRyYWZ0LXJvdycpO1xuICAgIGlmICghZHJhZnRSb3cpIHJldHVybjtcblxuICAgIGRyYWZ0Um93LnN0eWxlLmN1cnNvciA9ICdwb2ludGVyJztcbiAgICBkcmFmdFJvdy5hZGRFdmVudExpc3RlbmVyKCdjbGljaycsIChlKSA9PiB7XG4gICAgICAvLyBEb24ndCBuYXZpZ2F0ZSBpZiBjbGlja2luZyBvbiBjaGVja2JveFxuICAgICAgaWYgKChlLnRhcmdldCBhcyBIVE1MRWxlbWVudCkuY2xvc2VzdCgnaW5wdXRbdHlwZT1cImNoZWNrYm94XCJdJykpIHJldHVybjtcblxuICAgICAgdGhpcy5zd2l0Y2hUb1RhYignYWZzdGVtbmluZycpO1xuICAgIH0pO1xuICB9XG59XG4iLCAiLyoqXG4gKiBTYWxvbiBPUyBBcHBcbiAqXG4gKiBNYWluIGFwcGxpY2F0aW9uIGNsYXNzIHRoYXQgb3JjaGVzdHJhdGVzIGFsbCBVSSBjb250cm9sbGVyc1xuICovXG5cbmltcG9ydCB7IFNpZGViYXJDb250cm9sbGVyIH0gZnJvbSAnLi9tb2R1bGVzL3NpZGViYXInO1xuaW1wb3J0IHsgRHJhd2VyQ29udHJvbGxlciB9IGZyb20gJy4vbW9kdWxlcy9kcmF3ZXJzJztcbmltcG9ydCB7IFRoZW1lQ29udHJvbGxlciB9IGZyb20gJy4vbW9kdWxlcy90aGVtZSc7XG5pbXBvcnQgeyBTZWFyY2hDb250cm9sbGVyIH0gZnJvbSAnLi9tb2R1bGVzL3NlYXJjaCc7XG5pbXBvcnQgeyBMb2NrU2NyZWVuQ29udHJvbGxlciB9IGZyb20gJy4vbW9kdWxlcy9sb2Nrc2NyZWVuJztcbmltcG9ydCB7IEthc3NlQ29udHJvbGxlciB9IGZyb20gJy4vbW9kdWxlcy9rYXNzZSc7XG5cbi8qKlxuICogTWFpbiBhcHBsaWNhdGlvbiBjbGFzc1xuICovXG5leHBvcnQgY2xhc3MgQXBwIHtcbiAgcmVhZG9ubHkgc2lkZWJhcjogU2lkZWJhckNvbnRyb2xsZXI7XG4gIHJlYWRvbmx5IGRyYXdlcnM6IERyYXdlckNvbnRyb2xsZXI7XG4gIHJlYWRvbmx5IHRoZW1lOiBUaGVtZUNvbnRyb2xsZXI7XG4gIHJlYWRvbmx5IHNlYXJjaDogU2VhcmNoQ29udHJvbGxlcjtcbiAgcmVhZG9ubHkgbG9ja1NjcmVlbjogTG9ja1NjcmVlbkNvbnRyb2xsZXI7XG4gIHJlYWRvbmx5IGthc3NlOiBLYXNzZUNvbnRyb2xsZXI7XG5cbiAgY29uc3RydWN0b3IoKSB7XG4gICAgLy8gSW5pdGlhbGl6ZSBjb250cm9sbGVyc1xuICAgIHRoaXMuc2lkZWJhciA9IG5ldyBTaWRlYmFyQ29udHJvbGxlcigpO1xuICAgIHRoaXMuZHJhd2VycyA9IG5ldyBEcmF3ZXJDb250cm9sbGVyKCk7XG4gICAgdGhpcy50aGVtZSA9IG5ldyBUaGVtZUNvbnRyb2xsZXIoKTtcbiAgICB0aGlzLnNlYXJjaCA9IG5ldyBTZWFyY2hDb250cm9sbGVyKCk7XG4gICAgdGhpcy5sb2NrU2NyZWVuID0gbmV3IExvY2tTY3JlZW5Db250cm9sbGVyKHRoaXMuZHJhd2Vycyk7XG4gICAgdGhpcy5rYXNzZSA9IG5ldyBLYXNzZUNvbnRyb2xsZXIoKTtcbiAgfVxufVxuXG4vKipcbiAqIEdsb2JhbCBhcHAgaW5zdGFuY2VcbiAqL1xubGV0IGFwcDogQXBwO1xuXG4vKipcbiAqIEluaXRpYWxpemUgdGhlIGFwcGxpY2F0aW9uXG4gKi9cbmZ1bmN0aW9uIGluaXQoKTogdm9pZCB7XG4gIGFwcCA9IG5ldyBBcHAoKTtcblxuICAvLyBFeHBvc2UgdG8gd2luZG93IGZvciBkZWJ1Z2dpbmdcbiAgaWYgKHR5cGVvZiB3aW5kb3cgIT09ICd1bmRlZmluZWQnKSB7XG4gICAgKHdpbmRvdyBhcyB1bmtub3duIGFzIHsgYXBwOiBBcHAgfSkuYXBwID0gYXBwO1xuICB9XG59XG5cbi8vIFdhaXQgZm9yIERPTSByZWFkeVxuaWYgKGRvY3VtZW50LnJlYWR5U3RhdGUgPT09ICdsb2FkaW5nJykge1xuICBkb2N1bWVudC5hZGRFdmVudExpc3RlbmVyKCdET01Db250ZW50TG9hZGVkJywgaW5pdCk7XG59IGVsc2Uge1xuICBpbml0KCk7XG59XG5cbmV4cG9ydCB7IGFwcCB9O1xuZXhwb3J0IGRlZmF1bHQgQXBwO1xuIl0sCiAgIm1hcHBpbmdzIjogIjs7OztBQU1PLElBQU0scUJBQU4sTUFBTSxtQkFBa0I7QUFBQSxFQUs3QixjQUFjO0FBSmQsU0FBUSxhQUFpQztBQUN6QyxTQUFRLFlBQWdDO0FBQ3hDLFNBQVEsY0FBa0M7QUFHeEMsU0FBSyxhQUFhLFNBQVMsZUFBZSxZQUFZO0FBQ3RELFNBQUssWUFBWSxTQUFTLGNBQWMsZ0JBQWdCO0FBQ3hELFNBQUssY0FBYyxTQUFTLGVBQWUsYUFBYTtBQUV4RCxTQUFLLGVBQWU7QUFDcEIsU0FBSyxjQUFjO0FBQ25CLFNBQUssYUFBYTtBQUFBLEVBQ3BCO0FBQUE7QUFBQTtBQUFBO0FBQUEsRUFLQSxJQUFJLGNBQXVCO0FBQ3pCLFdBQU8sS0FBSyxXQUFXLFVBQVUsU0FBUyxnQkFBZ0IsS0FBSztBQUFBLEVBQ2pFO0FBQUE7QUFBQTtBQUFBO0FBQUEsRUFLQSxTQUFlO0FBQ2IsUUFBSSxDQUFDLEtBQUssVUFBVztBQUVyQixTQUFLLFVBQVUsVUFBVSxPQUFPLGdCQUFnQjtBQUNoRCxpQkFBYSxRQUFRLHFCQUFxQixPQUFPLEtBQUssV0FBVyxDQUFDO0FBQUEsRUFDcEU7QUFBQTtBQUFBO0FBQUE7QUFBQSxFQUtBLFdBQWlCO0FBQ2YsU0FBSyxXQUFXLFVBQVUsSUFBSSxnQkFBZ0I7QUFDOUMsaUJBQWEsUUFBUSxxQkFBcUIsTUFBTTtBQUFBLEVBQ2xEO0FBQUE7QUFBQTtBQUFBO0FBQUEsRUFLQSxTQUFlO0FBQ2IsU0FBSyxXQUFXLFVBQVUsT0FBTyxnQkFBZ0I7QUFDakQsaUJBQWEsUUFBUSxxQkFBcUIsT0FBTztBQUFBLEVBQ25EO0FBQUEsRUFFUSxpQkFBdUI7QUFDN0IsU0FBSyxZQUFZLGlCQUFpQixTQUFTLE1BQU0sS0FBSyxPQUFPLENBQUM7QUFBQSxFQUNoRTtBQUFBLEVBRVEsZ0JBQXNCO0FBQzVCLFFBQUksQ0FBQyxLQUFLLFlBQWE7QUFFdkIsVUFBTSxZQUFZLFNBQVMsaUJBQThCLGtDQUFrQztBQUUzRixjQUFVLFFBQVEsVUFBUTtBQUN4QixXQUFLLGlCQUFpQixjQUFjLE1BQU0sS0FBSyxZQUFZLElBQUksQ0FBQztBQUNoRSxXQUFLLGlCQUFpQixjQUFjLE1BQU0sS0FBSyxZQUFZLENBQUM7QUFBQSxJQUM5RCxDQUFDO0FBQUEsRUFDSDtBQUFBLEVBRVEsWUFBWSxNQUF5QjtBQUMzQyxRQUFJLENBQUMsS0FBSyxlQUFlLENBQUMsS0FBSyxZQUFhO0FBRTVDLFVBQU0sT0FBTyxLQUFLLHNCQUFzQjtBQUN4QyxVQUFNLGNBQWMsS0FBSyxRQUFRO0FBRWpDLFFBQUksQ0FBQyxZQUFhO0FBRWxCLFNBQUssWUFBWSxjQUFjO0FBQy9CLFNBQUssWUFBWSxNQUFNLE9BQU8sR0FBRyxLQUFLLFFBQVEsQ0FBQztBQUMvQyxTQUFLLFlBQVksTUFBTSxNQUFNLEdBQUcsS0FBSyxNQUFNLEtBQUssU0FBUyxDQUFDO0FBQzFELFNBQUssWUFBWSxNQUFNLFlBQVk7QUFDbkMsU0FBSyxZQUFZLFlBQVk7QUFBQSxFQUMvQjtBQUFBLEVBRVEsY0FBb0I7QUFDMUIsU0FBSyxhQUFhLFlBQVk7QUFBQSxFQUNoQztBQUFBLEVBRVEsZUFBcUI7QUFDM0IsUUFBSSxDQUFDLEtBQUssVUFBVztBQUVyQixRQUFJLGFBQWEsUUFBUSxtQkFBbUIsTUFBTSxRQUFRO0FBQ3hELFdBQUssVUFBVSxVQUFVLElBQUksZ0JBQWdCO0FBQUEsSUFDL0M7QUFBQSxFQUNGO0FBQ0Y7QUF6RitCO0FBQXhCLElBQU0sb0JBQU47OztBQ0VBLElBQU0sb0JBQU4sTUFBTSxrQkFBaUI7QUFBQSxFQVM1QixjQUFjO0FBUmQsU0FBUSxnQkFBb0M7QUFDNUMsU0FBUSxxQkFBeUM7QUFDakQsU0FBUSxhQUFpQztBQUN6QyxTQUFRLGdCQUFvQztBQUM1QyxTQUFRLFVBQThCO0FBQ3RDLFNBQVEsZUFBa0M7QUFDMUMsU0FBUSxzQkFBMEM7QUFHaEQsU0FBSyxnQkFBZ0IsU0FBUyxlQUFlLGVBQWU7QUFDNUQsU0FBSyxxQkFBcUIsU0FBUyxlQUFlLG9CQUFvQjtBQUN0RSxTQUFLLGFBQWEsU0FBUyxlQUFlLFlBQVk7QUFDdEQsU0FBSyxnQkFBZ0IsU0FBUyxlQUFlLGVBQWU7QUFDNUQsU0FBSyxVQUFVLFNBQVMsZUFBZSxlQUFlO0FBRXRELFNBQUssZUFBZTtBQUNwQixTQUFLLG9CQUFvQjtBQUFBLEVBQzNCO0FBQUE7QUFBQTtBQUFBO0FBQUEsRUFLQSxJQUFJLFNBQTRCO0FBQzlCLFdBQU8sS0FBSztBQUFBLEVBQ2Q7QUFBQTtBQUFBO0FBQUE7QUFBQSxFQUtBLEtBQUssTUFBd0I7QUFDM0IsU0FBSyxTQUFTO0FBRWQsVUFBTSxTQUFTLEtBQUssVUFBVSxJQUFJO0FBQ2xDLFFBQUksVUFBVSxLQUFLLFNBQVM7QUFDMUIsYUFBTyxVQUFVLElBQUksUUFBUTtBQUM3QixXQUFLLFFBQVEsVUFBVSxJQUFJLFFBQVE7QUFDbkMsZUFBUyxLQUFLLE1BQU0sV0FBVztBQUMvQixXQUFLLGVBQWU7QUFBQSxJQUN0QjtBQUFBLEVBQ0Y7QUFBQTtBQUFBO0FBQUE7QUFBQSxFQUtBLE1BQU0sTUFBd0I7QUFDNUIsVUFBTSxTQUFTLEtBQUssVUFBVSxJQUFJO0FBQ2xDLFlBQVEsVUFBVSxPQUFPLFFBQVE7QUFHakMsUUFBSSxLQUFLLFdBQVcsQ0FBQyxTQUFTLGNBQWMsMEJBQTBCLEdBQUc7QUFDdkUsV0FBSyxRQUFRLFVBQVUsT0FBTyxRQUFRO0FBQ3RDLGVBQVMsS0FBSyxNQUFNLFdBQVc7QUFBQSxJQUNqQztBQUVBLFFBQUksS0FBSyxpQkFBaUIsTUFBTTtBQUM5QixXQUFLLGVBQWU7QUFBQSxJQUN0QjtBQUFBLEVBQ0Y7QUFBQTtBQUFBO0FBQUE7QUFBQSxFQUtBLFdBQWlCO0FBQ2YsS0FBQyxLQUFLLGVBQWUsS0FBSyxvQkFBb0IsS0FBSyxZQUFZLEtBQUssYUFBYSxFQUM5RSxRQUFRLFlBQVUsUUFBUSxVQUFVLE9BQU8sUUFBUSxDQUFDO0FBR3ZELFNBQUssbUJBQW1CO0FBRXhCLFNBQUssU0FBUyxVQUFVLE9BQU8sUUFBUTtBQUN2QyxhQUFTLEtBQUssTUFBTSxXQUFXO0FBQy9CLFNBQUssZUFBZTtBQUFBLEVBQ3RCO0FBQUE7QUFBQTtBQUFBO0FBQUEsRUFLQSxrQkFBa0IsVUFBd0I7QUFDeEMsU0FBSyxTQUFTO0FBRWQsVUFBTSxTQUFTLFNBQVMsZUFBZSxRQUFRO0FBQy9DLFFBQUksVUFBVSxLQUFLLFNBQVM7QUFDMUIsYUFBTyxVQUFVLElBQUksTUFBTTtBQUMzQixXQUFLLFFBQVEsVUFBVSxJQUFJLFFBQVE7QUFDbkMsZUFBUyxLQUFLLE1BQU0sV0FBVztBQUMvQixXQUFLLHNCQUFzQjtBQUFBLElBQzdCO0FBQUEsRUFDRjtBQUFBO0FBQUE7QUFBQTtBQUFBLEVBS0EscUJBQTJCO0FBQ3pCLFNBQUsscUJBQXFCLFVBQVUsT0FBTyxNQUFNO0FBQ2pELFNBQUssc0JBQXNCO0FBQUEsRUFDN0I7QUFBQTtBQUFBO0FBQUE7QUFBQSxFQUtBLGNBQW9CO0FBQ2xCLFNBQUssS0FBSyxTQUFTO0FBQUEsRUFDckI7QUFBQTtBQUFBO0FBQUE7QUFBQSxFQUtBLG1CQUF5QjtBQUN2QixTQUFLLEtBQUssY0FBYztBQUFBLEVBQzFCO0FBQUE7QUFBQTtBQUFBO0FBQUEsRUFLQSxXQUFpQjtBQUNmLFNBQUssWUFBWSxVQUFVLElBQUksUUFBUTtBQUFBLEVBQ3pDO0FBQUE7QUFBQTtBQUFBO0FBQUEsRUFLQSxZQUFrQjtBQUNoQixTQUFLLFlBQVksVUFBVSxPQUFPLFFBQVE7QUFDMUMsU0FBSyxhQUFhO0FBQUEsRUFDcEI7QUFBQTtBQUFBO0FBQUE7QUFBQSxFQUtBLGNBQW9CO0FBQ2xCLFNBQUssZUFBZSxVQUFVLElBQUksUUFBUTtBQUFBLEVBQzVDO0FBQUE7QUFBQTtBQUFBO0FBQUEsRUFLQSxlQUFxQjtBQUNuQixTQUFLLGVBQWUsVUFBVSxPQUFPLFFBQVE7QUFBQSxFQUMvQztBQUFBO0FBQUE7QUFBQTtBQUFBLEVBS0EsMkJBQWlDO0FBQy9CLFFBQUksQ0FBQyxLQUFLLG1CQUFvQjtBQUU5QixVQUFNLGNBQWMsS0FBSyxtQkFBbUI7QUFBQSxNQUMxQztBQUFBLElBQ0Y7QUFDQSxnQkFBWSxRQUFRLFVBQVEsS0FBSyxnQkFBZ0IsYUFBYSxDQUFDO0FBRS9ELFVBQU0sUUFBUSxTQUFTLGNBQTJCLHdCQUF3QjtBQUMxRSxRQUFJLE9BQU87QUFDVCxZQUFNLE1BQU0sVUFBVTtBQUFBLElBQ3hCO0FBQUEsRUFDRjtBQUFBLEVBRVEsVUFBVSxNQUFzQztBQUN0RCxZQUFRLE1BQU07QUFBQSxNQUNaLEtBQUs7QUFBVyxlQUFPLEtBQUs7QUFBQSxNQUM1QixLQUFLO0FBQWdCLGVBQU8sS0FBSztBQUFBLE1BQ2pDLEtBQUs7QUFBUSxlQUFPLEtBQUs7QUFBQSxNQUN6QixLQUFLO0FBQVcsZUFBTyxLQUFLO0FBQUEsSUFDOUI7QUFBQSxFQUNGO0FBQUEsRUFFUSxpQkFBdUI7QUFFN0IsYUFBUyxlQUFlLGdCQUFnQixHQUNwQyxpQkFBaUIsU0FBUyxNQUFNLEtBQUssWUFBWSxDQUFDO0FBQ3RELGFBQVMsZUFBZSxhQUFhLEdBQ2pDLGlCQUFpQixTQUFTLE1BQU0sS0FBSyxNQUFNLFNBQVMsQ0FBQztBQUd6RCxhQUFTLGVBQWUsa0JBQWtCLEdBQ3RDLGlCQUFpQixTQUFTLE1BQU0sS0FBSyxpQkFBaUIsQ0FBQztBQUMzRCxhQUFTLGVBQWUseUJBQXlCLEdBQzdDLGlCQUFpQixTQUFTLE1BQU0sS0FBSyxNQUFNLGNBQWMsQ0FBQztBQUM5RCxhQUFTLGVBQWUsYUFBYSxHQUNqQyxpQkFBaUIsU0FBUyxNQUFNLEtBQUsseUJBQXlCLENBQUM7QUFHbkUsYUFBUyxlQUFlLGdCQUFnQixHQUNwQyxpQkFBaUIsU0FBUyxNQUFNLEtBQUssU0FBUyxDQUFDO0FBQ25ELGFBQVMsZUFBZSxnQkFBZ0IsR0FDcEMsaUJBQWlCLFNBQVMsTUFBTSxLQUFLLFVBQVUsQ0FBQztBQUdwRCxhQUFTLGVBQWUsWUFBWSxHQUNoQyxpQkFBaUIsU0FBUyxNQUFNLEtBQUssWUFBWSxDQUFDO0FBQ3RELGFBQVMsZUFBZSxtQkFBbUIsR0FDdkMsaUJBQWlCLFNBQVMsTUFBTSxLQUFLLGFBQWEsQ0FBQztBQUN2RCxhQUFTLGVBQWUsZUFBZSxHQUNuQyxpQkFBaUIsU0FBUyxNQUFNLEtBQUssYUFBYSxDQUFDO0FBQ3ZELGFBQVMsZUFBZSxhQUFhLEdBQ2pDLGlCQUFpQixTQUFTLE1BQU0sS0FBSyxhQUFhLENBQUM7QUFHdkQsU0FBSyxTQUFTLGlCQUFpQixTQUFTLE1BQU0sS0FBSyxTQUFTLENBQUM7QUFHN0QsYUFBUyxpQkFBaUIsV0FBVyxDQUFDLE1BQXFCO0FBQ3pELFVBQUksRUFBRSxRQUFRLFNBQVUsTUFBSyxTQUFTO0FBQUEsSUFDeEMsQ0FBQztBQUdELFNBQUssWUFBWSxpQkFBaUIsU0FBUyxDQUFDLE1BQU0sS0FBSyxnQkFBZ0IsQ0FBQyxDQUFDO0FBR3pFLGFBQVMsaUJBQWlCLFNBQVMsQ0FBQyxNQUFNLEtBQUssc0JBQXNCLENBQUMsQ0FBQztBQUFBLEVBQ3pFO0FBQUEsRUFFUSxnQkFBZ0IsR0FBZ0I7QUFDdEMsVUFBTSxTQUFTLEVBQUU7QUFDakIsVUFBTSxXQUFXLE9BQU8sUUFBcUIsZUFBZTtBQUM1RCxVQUFNLFdBQVcsT0FBTyxRQUFxQixtQkFBbUI7QUFFaEUsUUFBSSxZQUFZLFVBQVU7QUFDeEIsWUFBTSxjQUFjLFNBQVMsUUFBUSxjQUFjO0FBQ25ELFVBQUksYUFBYTtBQUNmLGlCQUFTLGdCQUFnQixnQkFBZ0I7QUFBQSxNQUMzQyxPQUFPO0FBQ0wsaUJBQVMsUUFBUSxZQUFZO0FBQUEsTUFDL0I7QUFBQSxJQUNGO0FBR0EsVUFBTSxnQkFBZ0IsT0FBTyxRQUFxQix5QkFBeUI7QUFDM0UsUUFBSSxlQUFlO0FBQ2pCLFlBQU0sVUFBVSxjQUFjLFFBQXFCLGtCQUFrQjtBQUNyRSxlQUFTLFVBQVUsT0FBTyxXQUFXO0FBQUEsSUFDdkM7QUFBQSxFQUNGO0FBQUEsRUFFUSxzQkFBc0IsR0FBZ0I7QUFDNUMsVUFBTSxTQUFTLEVBQUU7QUFDakIsVUFBTSxTQUFTLE9BQU8sUUFBcUIsdUJBQXVCO0FBRWxFLFFBQUksUUFBUTtBQUNWLGVBQVMsaUJBQThCLHVCQUF1QixFQUMzRCxRQUFRLE9BQUssRUFBRSxVQUFVLE9BQU8sUUFBUSxDQUFDO0FBQzVDLGFBQU8sVUFBVSxJQUFJLFFBQVE7QUFBQSxJQUMvQjtBQUFBLEVBQ0Y7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBLEVBTVEsc0JBQTRCO0FBRWxDLGFBQVMsaUJBQWlCLFNBQVMsQ0FBQyxNQUFhO0FBQy9DLFlBQU0sU0FBUyxFQUFFO0FBQ2pCLFlBQU0sVUFBVSxPQUFPLFFBQXFCLHVCQUF1QjtBQUVuRSxVQUFJLFNBQVM7QUFDWCxjQUFNLFdBQVcsUUFBUSxRQUFRO0FBQ2pDLFlBQUksVUFBVTtBQUNaLGVBQUssa0JBQWtCLFFBQVE7QUFBQSxRQUNqQztBQUFBLE1BQ0Y7QUFBQSxJQUNGLENBQUM7QUFHRCxhQUFTLGlCQUFpQixTQUFTLENBQUMsTUFBYTtBQUMvQyxZQUFNLFNBQVMsRUFBRTtBQUNqQixZQUFNLFdBQVcsT0FBTyxRQUFxQixxQkFBcUI7QUFFbEUsVUFBSSxVQUFVO0FBQ1osYUFBSyxtQkFBbUI7QUFDeEIsYUFBSyxTQUFTLFVBQVUsT0FBTyxRQUFRO0FBQ3ZDLGlCQUFTLEtBQUssTUFBTSxXQUFXO0FBQUEsTUFDakM7QUFBQSxJQUNGLENBQUM7QUFBQSxFQUNIO0FBQ0Y7QUFwUjhCO0FBQXZCLElBQU0sbUJBQU47OztBQ0FBLElBQU0sbUJBQU4sTUFBTSxpQkFBZ0I7QUFBQSxFQVEzQixjQUFjO0FBQ1osU0FBSyxPQUFPLFNBQVM7QUFDckIsU0FBSyxlQUFlLFNBQVMsaUJBQThCLGtCQUFrQjtBQUU3RSxTQUFLLFdBQVcsS0FBSyxPQUFPO0FBQzVCLFNBQUssU0FBUztBQUNkLFNBQUssZUFBZTtBQUFBLEVBQ3RCO0FBQUE7QUFBQTtBQUFBO0FBQUEsRUFLQSxJQUFJLFVBQWlCO0FBQ25CLFVBQU0sU0FBUyxhQUFhLFFBQVEsaUJBQWdCLFdBQVc7QUFDL0QsUUFBSSxXQUFXLFVBQVUsV0FBVyxXQUFXLFdBQVcsVUFBVTtBQUNsRSxhQUFPO0FBQUEsSUFDVDtBQUNBLFdBQU87QUFBQSxFQUNUO0FBQUE7QUFBQTtBQUFBO0FBQUEsRUFLQSxJQUFJLFNBQWtCO0FBQ3BCLFdBQU8sS0FBSyxLQUFLLFVBQVUsU0FBUyxpQkFBZ0IsVUFBVSxLQUMzRCxLQUFLLHFCQUFxQixDQUFDLEtBQUssS0FBSyxVQUFVLFNBQVMsaUJBQWdCLFdBQVc7QUFBQSxFQUN4RjtBQUFBO0FBQUE7QUFBQTtBQUFBLEVBS0EsSUFBSSxvQkFBNkI7QUFDL0IsV0FBTyxPQUFPLFdBQVcsOEJBQThCLEVBQUU7QUFBQSxFQUMzRDtBQUFBO0FBQUE7QUFBQTtBQUFBLEVBS0EsSUFBSSxPQUFvQjtBQUN0QixpQkFBYSxRQUFRLGlCQUFnQixhQUFhLEtBQUs7QUFDdkQsU0FBSyxXQUFXLEtBQUs7QUFDckIsU0FBSyxTQUFTO0FBQUEsRUFDaEI7QUFBQTtBQUFBO0FBQUE7QUFBQSxFQUtBLFNBQWU7QUFDYixTQUFLLElBQUksS0FBSyxTQUFTLFVBQVUsTUFBTTtBQUFBLEVBQ3pDO0FBQUEsRUFFUSxXQUFXLE9BQW9CO0FBQ3JDLFNBQUssS0FBSyxVQUFVLE9BQU8saUJBQWdCLFlBQVksaUJBQWdCLFdBQVc7QUFFbEYsUUFBSSxVQUFVLFFBQVE7QUFDcEIsV0FBSyxLQUFLLFVBQVUsSUFBSSxpQkFBZ0IsVUFBVTtBQUFBLElBQ3BELFdBQVcsVUFBVSxTQUFTO0FBQzVCLFdBQUssS0FBSyxVQUFVLElBQUksaUJBQWdCLFdBQVc7QUFBQSxJQUNyRDtBQUFBLEVBRUY7QUFBQSxFQUVRLFdBQWlCO0FBQ3ZCLFFBQUksQ0FBQyxLQUFLLGFBQWM7QUFFeEIsVUFBTSxhQUFhLEtBQUs7QUFFeEIsU0FBSyxhQUFhLFFBQVEsWUFBVTtBQUNsQyxZQUFNLFFBQVEsT0FBTyxRQUFRO0FBQzdCLFlBQU0sV0FBWSxVQUFVLFVBQVUsY0FBZ0IsVUFBVSxXQUFXLENBQUM7QUFDNUUsYUFBTyxVQUFVLE9BQU8sVUFBVSxRQUFRO0FBQUEsSUFDNUMsQ0FBQztBQUFBLEVBQ0g7QUFBQSxFQUVRLGlCQUF1QjtBQUU3QixTQUFLLGFBQWEsUUFBUSxZQUFVO0FBQ2xDLGFBQU8saUJBQWlCLFNBQVMsQ0FBQyxNQUFNLEtBQUssa0JBQWtCLENBQUMsQ0FBQztBQUFBLElBQ25FLENBQUM7QUFHRCxXQUFPLFdBQVcsOEJBQThCLEVBQzdDLGlCQUFpQixVQUFVLE1BQU0sS0FBSyxtQkFBbUIsQ0FBQztBQUFBLEVBQy9EO0FBQUEsRUFFUSxrQkFBa0IsR0FBZ0I7QUFDeEMsVUFBTSxTQUFTLEVBQUU7QUFDakIsVUFBTSxTQUFTLE9BQU8sUUFBcUIsa0JBQWtCO0FBRTdELFFBQUksUUFBUTtBQUNWLFlBQU0sUUFBUSxPQUFPLFFBQVE7QUFDN0IsVUFBSSxPQUFPO0FBQ1QsYUFBSyxJQUFJLEtBQUs7QUFBQSxNQUNoQjtBQUFBLElBQ0Y7QUFBQSxFQUNGO0FBQUEsRUFFUSxxQkFBMkI7QUFFakMsUUFBSSxLQUFLLFlBQVksVUFBVTtBQUM3QixXQUFLLFNBQVM7QUFBQSxJQUNoQjtBQUFBLEVBQ0Y7QUFDRjtBQS9HNkI7QUFBaEIsaUJBQ2EsY0FBYztBQUQzQixpQkFFYSxhQUFhO0FBRjFCLGlCQUdhLGNBQWM7QUFIakMsSUFBTSxrQkFBTjs7O0FDRkEsSUFBTSxvQkFBTixNQUFNLGtCQUFpQjtBQUFBLEVBSTVCLGNBQWM7QUFIZCxTQUFRLFFBQWlDO0FBQ3pDLFNBQVEsWUFBZ0M7QUFHdEMsU0FBSyxRQUFRLFNBQVMsZUFBZSxjQUFjO0FBQ25ELFNBQUssWUFBWSxTQUFTLGNBQTJCLG1CQUFtQjtBQUV4RSxTQUFLLGVBQWU7QUFBQSxFQUN0QjtBQUFBO0FBQUE7QUFBQTtBQUFBLEVBS0EsSUFBSSxRQUFnQjtBQUNsQixXQUFPLEtBQUssT0FBTyxTQUFTO0FBQUEsRUFDOUI7QUFBQTtBQUFBO0FBQUE7QUFBQSxFQUtBLElBQUksTUFBTSxLQUFhO0FBQ3JCLFFBQUksS0FBSyxPQUFPO0FBQ2QsV0FBSyxNQUFNLFFBQVE7QUFBQSxJQUNyQjtBQUFBLEVBQ0Y7QUFBQTtBQUFBO0FBQUE7QUFBQSxFQUtBLFFBQWM7QUFDWixTQUFLLE9BQU8sTUFBTTtBQUFBLEVBQ3BCO0FBQUE7QUFBQTtBQUFBO0FBQUEsRUFLQSxPQUFhO0FBQ1gsU0FBSyxPQUFPLEtBQUs7QUFBQSxFQUNuQjtBQUFBO0FBQUE7QUFBQTtBQUFBLEVBS0EsUUFBYztBQUNaLFNBQUssUUFBUTtBQUFBLEVBQ2Y7QUFBQSxFQUVRLGlCQUF1QjtBQUU3QixhQUFTLGlCQUFpQixXQUFXLENBQUMsTUFBTSxLQUFLLGVBQWUsQ0FBQyxDQUFDO0FBR2xFLFFBQUksS0FBSyxPQUFPO0FBQ2QsV0FBSyxNQUFNLGlCQUFpQixTQUFTLENBQUMsTUFBTSxLQUFLLFlBQVksQ0FBQyxDQUFDO0FBRy9ELFlBQU0sT0FBTyxLQUFLLE1BQU0sUUFBUSxNQUFNO0FBQ3RDLFlBQU0saUJBQWlCLFVBQVUsQ0FBQyxNQUFNLEtBQUssYUFBYSxDQUFDLENBQUM7QUFBQSxJQUM5RDtBQUFBLEVBQ0Y7QUFBQSxFQUVRLGVBQWUsR0FBd0I7QUFFN0MsU0FBSyxFQUFFLFdBQVcsRUFBRSxZQUFZLEVBQUUsUUFBUSxLQUFLO0FBQzdDLFFBQUUsZUFBZTtBQUNqQixXQUFLLE1BQU07QUFDWDtBQUFBLElBQ0Y7QUFHQSxRQUFJLEVBQUUsUUFBUSxZQUFZLFNBQVMsa0JBQWtCLEtBQUssT0FBTztBQUMvRCxXQUFLLEtBQUs7QUFBQSxJQUNaO0FBQUEsRUFDRjtBQUFBLEVBRVEsWUFBWSxHQUFnQjtBQUNsQyxVQUFNLFNBQVMsRUFBRTtBQUNqQixVQUFNLFFBQVEsT0FBTyxNQUFNLEtBQUs7QUFHaEMsYUFBUyxjQUFjLElBQUksWUFBWSxjQUFjO0FBQUEsTUFDbkQsUUFBUSxFQUFFLE1BQU07QUFBQSxNQUNoQixTQUFTO0FBQUEsSUFDWCxDQUFDLENBQUM7QUFBQSxFQUNKO0FBQUEsRUFFUSxhQUFhLEdBQWdCO0FBQ25DLE1BQUUsZUFBZTtBQUVqQixVQUFNLFFBQVEsS0FBSyxNQUFNLEtBQUs7QUFDOUIsUUFBSSxDQUFDLE1BQU87QUFHWixhQUFTLGNBQWMsSUFBSSxZQUFZLHFCQUFxQjtBQUFBLE1BQzFELFFBQVEsRUFBRSxNQUFNO0FBQUEsTUFDaEIsU0FBUztBQUFBLElBQ1gsQ0FBQyxDQUFDO0FBQUEsRUFDSjtBQUNGO0FBbkc4QjtBQUF2QixJQUFNLG1CQUFOOzs7QUNFQSxJQUFNLHdCQUFOLE1BQU0sc0JBQXFCO0FBQUEsRUFXaEMsWUFBWSxTQUE0QjtBQVJ4QztBQUFBLFNBQVEsYUFBaUM7QUFDekMsU0FBUSxXQUErQjtBQUN2QyxTQUFRLFlBQWdDO0FBQ3hDLFNBQVEsYUFBaUM7QUFDekMsU0FBUSxZQUE0QztBQUNwRCxTQUFRLGFBQWE7QUFDckIsU0FBUSxVQUFtQztBQUd6QyxTQUFLLFVBQVUsV0FBVztBQUMxQixTQUFLLGFBQWEsU0FBUyxlQUFlLFlBQVk7QUFDdEQsU0FBSyxXQUFXLFNBQVMsZUFBZSxVQUFVO0FBQ2xELFNBQUssWUFBWSxTQUFTLGVBQWUsV0FBVztBQUNwRCxTQUFLLGFBQWEsU0FBUyxlQUFlLFVBQVU7QUFDcEQsU0FBSyxZQUFZLEtBQUssVUFBVSxpQkFBOEIsZUFBZSxLQUFLO0FBRWxGLFNBQUssZUFBZTtBQUFBLEVBQ3RCO0FBQUE7QUFBQTtBQUFBO0FBQUEsRUFLQSxJQUFJLFdBQW9CO0FBQ3RCLFdBQU8sS0FBSyxZQUFZLFVBQVUsU0FBUyxRQUFRLEtBQUs7QUFBQSxFQUMxRDtBQUFBO0FBQUE7QUFBQTtBQUFBLEVBS0EsT0FBYTtBQUNYLFNBQUssU0FBUyxTQUFTO0FBRXZCLFFBQUksS0FBSyxZQUFZO0FBQ25CLFdBQUssV0FBVyxVQUFVLElBQUksUUFBUTtBQUN0QyxlQUFTLEtBQUssTUFBTSxXQUFXO0FBQUEsSUFDakM7QUFFQSxTQUFLLGFBQWE7QUFDbEIsU0FBSyxjQUFjO0FBR25CLFFBQUksS0FBSyxZQUFZO0FBQ25CLFdBQUssV0FBVyxjQUFjLGVBQVksS0FBSyxXQUFXLENBQUM7QUFBQSxJQUM3RDtBQUFBLEVBQ0Y7QUFBQTtBQUFBO0FBQUE7QUFBQSxFQUtBLE9BQWE7QUFDWCxRQUFJLEtBQUssWUFBWTtBQUNuQixXQUFLLFdBQVcsVUFBVSxPQUFPLFFBQVE7QUFDekMsZUFBUyxLQUFLLE1BQU0sV0FBVztBQUFBLElBQ2pDO0FBRUEsU0FBSyxhQUFhO0FBQ2xCLFNBQUssY0FBYztBQUFBLEVBQ3JCO0FBQUEsRUFFUSxhQUFxQjtBQUMzQixVQUFNLE1BQU0sb0JBQUksS0FBSztBQUNyQixVQUFNLFFBQVEsSUFBSSxTQUFTLEVBQUUsU0FBUyxFQUFFLFNBQVMsR0FBRyxHQUFHO0FBQ3ZELFVBQU0sVUFBVSxJQUFJLFdBQVcsRUFBRSxTQUFTLEVBQUUsU0FBUyxHQUFHLEdBQUc7QUFDM0QsV0FBTyxHQUFHLEtBQUssSUFBSSxPQUFPO0FBQUEsRUFDNUI7QUFBQSxFQUVRLGdCQUFzQjtBQUM1QixRQUFJLENBQUMsS0FBSyxVQUFXO0FBRXJCLFNBQUssVUFBVSxRQUFRLENBQUMsT0FBTyxVQUFVO0FBQ3ZDLFlBQU0sVUFBVSxPQUFPLFVBQVUsT0FBTztBQUN4QyxVQUFJLFFBQVEsS0FBSyxXQUFXLFFBQVE7QUFDbEMsY0FBTSxjQUFjO0FBQ3BCLGNBQU0sVUFBVSxJQUFJLFFBQVE7QUFBQSxNQUM5QixPQUFPO0FBQ0wsY0FBTSxjQUFjO0FBQUEsTUFDdEI7QUFBQSxJQUNGLENBQUM7QUFBQSxFQUNIO0FBQUEsRUFFUSxZQUFrQjtBQUN4QixRQUFJLENBQUMsS0FBSyxVQUFXO0FBRXJCLFNBQUssVUFBVSxRQUFRLFdBQVMsTUFBTSxVQUFVLElBQUksT0FBTyxDQUFDO0FBRzVELFNBQUssVUFBVSxVQUFVLElBQUksT0FBTztBQUVwQyxlQUFXLE1BQU07QUFDZixXQUFLLGFBQWE7QUFDbEIsV0FBSyxjQUFjO0FBQ25CLFdBQUssVUFBVSxVQUFVLE9BQU8sT0FBTztBQUFBLElBQ3pDLEdBQUcsR0FBRztBQUFBLEVBQ1I7QUFBQSxFQUVRLFNBQWU7QUFDckIsUUFBSSxLQUFLLGVBQWUsc0JBQXFCLGFBQWE7QUFDeEQsV0FBSyxLQUFLO0FBQUEsSUFDWixPQUFPO0FBQ0wsV0FBSyxVQUFVO0FBQUEsSUFDakI7QUFBQSxFQUNGO0FBQUEsRUFFUSxTQUFTLE9BQXFCO0FBQ3BDLFFBQUksS0FBSyxXQUFXLFVBQVUsRUFBRztBQUVqQyxTQUFLLGNBQWM7QUFDbkIsU0FBSyxjQUFjO0FBR25CLFFBQUksS0FBSyxXQUFXLFdBQVcsR0FBRztBQUNoQyxpQkFBVyxNQUFNLEtBQUssT0FBTyxHQUFHLEdBQUc7QUFBQSxJQUNyQztBQUFBLEVBQ0Y7QUFBQSxFQUVRLGNBQW9CO0FBQzFCLFFBQUksS0FBSyxXQUFXLFdBQVcsRUFBRztBQUNsQyxTQUFLLGFBQWEsS0FBSyxXQUFXLE1BQU0sR0FBRyxFQUFFO0FBQzdDLFNBQUssY0FBYztBQUFBLEVBQ3JCO0FBQUEsRUFFUSxXQUFpQjtBQUN2QixTQUFLLGFBQWE7QUFDbEIsU0FBSyxjQUFjO0FBQUEsRUFDckI7QUFBQSxFQUVRLGlCQUF1QjtBQUU3QixTQUFLLFdBQVcsaUJBQWlCLFNBQVMsQ0FBQyxNQUFNLEtBQUssa0JBQWtCLENBQUMsQ0FBQztBQUcxRSxhQUFTLGlCQUFpQixXQUFXLENBQUMsTUFBTSxLQUFLLGVBQWUsQ0FBQyxDQUFDO0FBR2xFLGFBQVMsY0FBMkIsMkJBQTJCLEdBQzNELGlCQUFpQixTQUFTLE1BQU0sS0FBSyxLQUFLLENBQUM7QUFBQSxFQUNqRDtBQUFBLEVBRVEsa0JBQWtCLEdBQWdCO0FBQ3hDLFVBQU0sU0FBUyxFQUFFO0FBQ2pCLFVBQU0sTUFBTSxPQUFPLFFBQXFCLGFBQWE7QUFFckQsUUFBSSxDQUFDLElBQUs7QUFFVixVQUFNLFFBQVEsSUFBSSxRQUFRO0FBQzFCLFVBQU0sU0FBUyxJQUFJLFFBQVE7QUFFM0IsUUFBSSxPQUFPO0FBQ1QsV0FBSyxTQUFTLEtBQUs7QUFBQSxJQUNyQixXQUFXLFdBQVcsYUFBYTtBQUNqQyxXQUFLLFlBQVk7QUFBQSxJQUNuQixXQUFXLFdBQVcsU0FBUztBQUM3QixXQUFLLFNBQVM7QUFBQSxJQUNoQjtBQUFBLEVBQ0Y7QUFBQSxFQUVRLGVBQWUsR0FBd0I7QUFDN0MsUUFBSSxDQUFDLEtBQUssU0FBVTtBQUdwQixNQUFFLGVBQWU7QUFFakIsUUFBSSxFQUFFLE9BQU8sT0FBTyxFQUFFLE9BQU8sS0FBSztBQUNoQyxXQUFLLFNBQVMsRUFBRSxHQUFHO0FBQUEsSUFDckIsV0FBVyxFQUFFLFFBQVEsYUFBYTtBQUNoQyxXQUFLLFlBQVk7QUFBQSxJQUNuQixXQUFXLEVBQUUsUUFBUSxVQUFVO0FBQzdCLFdBQUssU0FBUztBQUFBLElBQ2hCO0FBQUEsRUFDRjtBQUNGO0FBN0trQztBQUFyQixzQkFDYSxjQUFjO0FBRGpDLElBQU0sdUJBQU47OztBQ0RBLElBQU0sbUJBQU4sTUFBTSxpQkFBZ0I7QUFBQSxFQUszQixjQUFjO0FBSGQ7QUFBQSxTQUFpQixlQUFlO0FBQ2hDLFNBQWlCLFlBQVk7QUFHM0IsU0FBSyxVQUFVO0FBQ2YsU0FBSyxxQkFBcUI7QUFDMUIsU0FBSyx1QkFBdUI7QUFDNUIsU0FBSyxzQkFBc0I7QUFDM0IsU0FBSyxpQkFBaUI7QUFDdEIsU0FBSyxlQUFlO0FBQ3BCLFNBQUssbUJBQW1CO0FBQUEsRUFDMUI7QUFBQTtBQUFBO0FBQUE7QUFBQSxFQUtRLFlBQWtCO0FBQ3hCLFVBQU0sT0FBTyxTQUFTLGlCQUE4QixtQkFBbUI7QUFFdkUsU0FBSyxRQUFRLFNBQU87QUFDbEIsVUFBSSxpQkFBaUIsU0FBUyxNQUFNO0FBQ2xDLGNBQU0sWUFBWSxJQUFJLFFBQVE7QUFDOUIsWUFBSSxXQUFXO0FBQ2IsZUFBSyxZQUFZLFNBQVM7QUFBQSxRQUM1QjtBQUFBLE1BQ0YsQ0FBQztBQUFBLElBQ0gsQ0FBQztBQUFBLEVBQ0g7QUFBQTtBQUFBO0FBQUE7QUFBQSxFQUtRLFlBQVksV0FBeUI7QUFDM0MsVUFBTSxPQUFPLFNBQVMsaUJBQThCLG1CQUFtQjtBQUN2RSxVQUFNLFdBQVcsU0FBUyxpQkFBOEIsMkJBQTJCO0FBQ25GLFVBQU0sWUFBWSxTQUFTLGlCQUE4QiwrQkFBK0I7QUFHeEYsU0FBSyxRQUFRLE9BQUs7QUFDaEIsVUFBSSxFQUFFLFFBQVEsUUFBUSxXQUFXO0FBQy9CLFVBQUUsVUFBVSxJQUFJLFFBQVE7QUFBQSxNQUMxQixPQUFPO0FBQ0wsVUFBRSxVQUFVLE9BQU8sUUFBUTtBQUFBLE1BQzdCO0FBQUEsSUFDRixDQUFDO0FBR0QsYUFBUyxRQUFRLGFBQVc7QUFDMUIsVUFBSSxRQUFRLFFBQVEsUUFBUSxXQUFXO0FBQ3JDLGdCQUFRLFVBQVUsSUFBSSxRQUFRO0FBQUEsTUFDaEMsT0FBTztBQUNMLGdCQUFRLFVBQVUsT0FBTyxRQUFRO0FBQUEsTUFDbkM7QUFBQSxJQUNGLENBQUM7QUFHRCxjQUFVLFFBQVEsV0FBUztBQUN6QixVQUFJLE1BQU0sUUFBUSxXQUFXLFdBQVc7QUFDdEMsY0FBTSxVQUFVLElBQUksUUFBUTtBQUFBLE1BQzlCLE9BQU87QUFDTCxjQUFNLFVBQVUsT0FBTyxRQUFRO0FBQUEsTUFDakM7QUFBQSxJQUNGLENBQUM7QUFBQSxFQUNIO0FBQUE7QUFBQTtBQUFBO0FBQUEsRUFLUSx1QkFBNkI7QUFDbkMsVUFBTSxlQUFlLFNBQVMsZUFBZSxTQUFTO0FBQ3RELFVBQU0sY0FBYyxTQUFTLGVBQWUsUUFBUTtBQUNwRCxVQUFNLGtCQUFrQixTQUFTLGVBQWUsWUFBWTtBQUU1RCxRQUFJLENBQUMsZ0JBQWdCLENBQUMsZUFBZSxDQUFDLGdCQUFpQjtBQUV2RCxVQUFNLFlBQVksNkJBQU0sS0FBSyxjQUFjLGNBQWMsYUFBYSxlQUFlLEdBQW5FO0FBRWxCLGlCQUFhLGlCQUFpQixTQUFTLFNBQVM7QUFDaEQsZ0JBQVksaUJBQWlCLFNBQVMsU0FBUztBQUMvQyxvQkFBZ0IsaUJBQWlCLFNBQVMsU0FBUztBQUduRCxjQUFVO0FBQUEsRUFDWjtBQUFBO0FBQUE7QUFBQTtBQUFBLEVBS1EsY0FDTixjQUNBLGFBQ0EsaUJBQ007QUFDTixVQUFNLFVBQVUsS0FBSyxZQUFZLGFBQWEsS0FBSztBQUNuRCxVQUFNLFNBQVMsS0FBSyxZQUFZLFlBQVksS0FBSztBQUNqRCxVQUFNLFNBQVMsS0FBSyxZQUFZLGdCQUFnQixLQUFLO0FBR3JELFVBQU0sZUFBZSxLQUFLLGVBQWUsS0FBSyxZQUFZLFVBQVU7QUFFcEUsVUFBTSxrQkFBa0IsU0FBUyxlQUFlLGNBQWM7QUFDOUQsUUFBSSxpQkFBaUI7QUFDbkIsc0JBQWdCLGNBQWMsS0FBSyxhQUFhLFlBQVk7QUFBQSxJQUM5RDtBQUdBLFNBQUssaUJBQWlCLFFBQVEsY0FBYyxnQkFBZ0IsS0FBSztBQUFBLEVBQ25FO0FBQUE7QUFBQTtBQUFBO0FBQUEsRUFLUSxpQkFBaUIsUUFBZ0IsVUFBa0IsVUFBd0I7QUFDakYsVUFBTSxNQUFNLFNBQVMsZUFBZSxlQUFlO0FBQ25ELFVBQU0sUUFBUSxTQUFTLGVBQWUsaUJBQWlCO0FBQ3ZELFFBQUksQ0FBQyxPQUFPLENBQUMsTUFBTztBQUVwQixVQUFNLE9BQU8sU0FBUztBQUd0QixRQUFJLFVBQVUsT0FBTyxZQUFZLFlBQVksU0FBUztBQUV0RCxRQUFJLFdBQVcsS0FBSyxhQUFhLElBQUk7QUFFbkMsWUFBTSxjQUFjO0FBQ3BCLFVBQUksVUFBVSxJQUFJLFNBQVM7QUFBQSxJQUM3QixXQUFXLE9BQU8sR0FBRztBQUVuQixZQUFNLGNBQWMsTUFBTSxLQUFLLGFBQWEsSUFBSSxJQUFJO0FBQ3BELFVBQUksVUFBVSxJQUFJLFVBQVU7QUFBQSxJQUM5QixXQUFXLE9BQU8sR0FBRztBQUVuQixZQUFNLGNBQWMsS0FBSyxhQUFhLElBQUksSUFBSTtBQUM5QyxVQUFJLFVBQVUsSUFBSSxVQUFVO0FBQUEsSUFDOUIsT0FBTztBQUVMLFlBQU0sY0FBYztBQUNwQixVQUFJLFVBQVUsSUFBSSxTQUFTO0FBQUEsSUFDN0I7QUFBQSxFQUNGO0FBQUE7QUFBQTtBQUFBO0FBQUEsRUFLUSx5QkFBK0I7QUFDckMsVUFBTSxZQUFZLFNBQVMsZUFBZSxXQUFXO0FBQ3JELFVBQU0sZ0JBQWdCLFNBQVMsaUJBQW1DLGFBQWE7QUFDL0UsVUFBTSxZQUFZLFNBQVMsZUFBZSxXQUFXO0FBQ3JELFVBQU0saUJBQWlCLFNBQVMsZUFBZSxnQkFBZ0I7QUFFL0QsUUFBSSxDQUFDLGFBQWEsQ0FBQyxhQUFhLENBQUMsZUFBZ0I7QUFFakQsVUFBTSxrQkFBa0IsNkJBQU07QUFDNUIsWUFBTSxVQUFVLFNBQVMsaUJBQW1DLHFCQUFxQjtBQUNqRixZQUFNLFFBQVEsUUFBUTtBQUV0QixxQkFBZSxjQUFjLFVBQVUsSUFBSSxZQUFZLEdBQUcsS0FBSztBQUMvRCxnQkFBVSxXQUFXLFVBQVU7QUFHL0IsZ0JBQVUsVUFBVSxVQUFVLGNBQWMsVUFBVSxRQUFRO0FBQzlELGdCQUFVLGdCQUFnQixRQUFRLEtBQUssUUFBUSxjQUFjO0FBQUEsSUFDL0QsR0FWd0I7QUFZeEIsY0FBVSxpQkFBaUIsVUFBVSxNQUFNO0FBQ3pDLG9CQUFjLFFBQVEsUUFBTSxHQUFHLFVBQVUsVUFBVSxPQUFPO0FBQzFELHNCQUFnQjtBQUFBLElBQ2xCLENBQUM7QUFFRCxrQkFBYyxRQUFRLFFBQU07QUFDMUIsU0FBRyxpQkFBaUIsVUFBVSxlQUFlO0FBRTdDLFNBQUcsaUJBQWlCLFNBQVMsT0FBSyxFQUFFLGdCQUFnQixDQUFDO0FBQUEsSUFDdkQsQ0FBQztBQUFBLEVBQ0g7QUFBQTtBQUFBO0FBQUE7QUFBQSxFQUtRLHdCQUE4QjtBQUNwQyxVQUFNLFdBQVcsU0FBUyxlQUFlLGlCQUFpQjtBQUMxRCxVQUFNLGFBQWEsU0FBUyxlQUFlLFlBQVk7QUFFdkQsUUFBSSxDQUFDLFlBQVksQ0FBQyxXQUFZO0FBRTlCLGFBQVMsaUJBQWlCLFVBQVUsTUFBTTtBQUN4QyxpQkFBVyxXQUFXLENBQUMsU0FBUztBQUFBLElBQ2xDLENBQUM7QUFBQSxFQUNIO0FBQUE7QUFBQTtBQUFBO0FBQUEsRUFLUSxtQkFBeUI7QUFDL0IsVUFBTSxXQUFXLFNBQVMsZUFBZSxVQUFVO0FBQ25ELFVBQU0sU0FBUyxTQUFTLGVBQWUsUUFBUTtBQUUvQyxRQUFJLENBQUMsWUFBWSxDQUFDLE9BQVE7QUFFMUIsVUFBTSxRQUFRLG9CQUFJLEtBQUs7QUFDdkIsVUFBTSxnQkFBZ0IsSUFBSSxLQUFLLEtBQUs7QUFDcEMsa0JBQWMsUUFBUSxNQUFNLFFBQVEsSUFBSSxFQUFFO0FBRTFDLFdBQU8sUUFBUSxLQUFLLGNBQWMsS0FBSztBQUN2QyxhQUFTLFFBQVEsS0FBSyxjQUFjLGFBQWE7QUFBQSxFQUNuRDtBQUFBO0FBQUE7QUFBQTtBQUFBLEVBS1EsYUFBYSxLQUFxQjtBQUN4QyxXQUFPLElBQUksZUFBZSxTQUFTO0FBQUEsTUFDakMsdUJBQXVCO0FBQUEsTUFDdkIsdUJBQXVCO0FBQUEsSUFDekIsQ0FBQztBQUFBLEVBQ0g7QUFBQTtBQUFBO0FBQUE7QUFBQSxFQUtRLFlBQVksS0FBcUI7QUFDdkMsUUFBSSxDQUFDLElBQUssUUFBTztBQUNqQixXQUFPLFdBQVcsSUFBSSxRQUFRLE9BQU8sRUFBRSxFQUFFLFFBQVEsS0FBSyxHQUFHLENBQUMsS0FBSztBQUFBLEVBQ2pFO0FBQUE7QUFBQTtBQUFBO0FBQUEsRUFLUSxjQUFjLE1BQW9CO0FBQ3hDLFdBQU8sS0FBSyxZQUFZLEVBQUUsTUFBTSxHQUFHLEVBQUUsQ0FBQztBQUFBLEVBQ3hDO0FBQUE7QUFBQTtBQUFBO0FBQUEsRUFLUSxpQkFBdUI7QUFDN0IsVUFBTSxPQUFPLFNBQVMsaUJBQThCLDhDQUE4QztBQUVsRyxTQUFLLFFBQVEsU0FBTztBQUNsQixZQUFNLFFBQVEsSUFBSSxhQUFhLFNBQVM7QUFDeEMsVUFBSSxDQUFDLE1BQU87QUFFWixZQUFNLFNBQVMsU0FBUyxjQUEyQixrQ0FBa0MsS0FBSyxJQUFJO0FBQzlGLFVBQUksQ0FBQyxPQUFRO0FBRWIsVUFBSSxpQkFBaUIsU0FBUyxDQUFDLE1BQU07QUFFbkMsWUFBSyxFQUFFLE9BQXVCLFFBQVEsd0JBQXdCLEVBQUc7QUFFakUsY0FBTSxPQUFPLElBQUksY0FBYyxrQkFBa0I7QUFDakQsY0FBTSxhQUFhLElBQUksVUFBVSxTQUFTLFVBQVU7QUFHcEQsaUJBQVMsaUJBQWlCLDhCQUE4QixFQUFFLFFBQVEsT0FBSztBQUNyRSxjQUFJLE1BQU0sS0FBSztBQUNiLGtCQUFNLFVBQVUsRUFBRSxhQUFhLFNBQVM7QUFDeEMsZ0JBQUksU0FBUztBQUNYLG9CQUFNLGNBQWMsU0FBUyxjQUEyQixrQ0FBa0MsT0FBTyxJQUFJO0FBQ3JHLG9CQUFNLFlBQVksRUFBRSxjQUFjLGtCQUFrQjtBQUNwRCxrQkFBSSxlQUFlLFdBQVc7QUFDNUIscUJBQUssWUFBWSxHQUFHLGFBQWEsU0FBd0I7QUFBQSxjQUMzRDtBQUFBLFlBQ0Y7QUFBQSxVQUNGO0FBQUEsUUFDRixDQUFDO0FBR0QsWUFBSSxZQUFZO0FBQ2QsZUFBSyxZQUFZLEtBQUssUUFBUSxJQUFJO0FBQUEsUUFDcEMsT0FBTztBQUNMLGVBQUssVUFBVSxLQUFLLFFBQVEsSUFBSTtBQUFBLFFBQ2xDO0FBQUEsTUFDRixDQUFDO0FBQUEsSUFDSCxDQUFDO0FBQUEsRUFDSDtBQUFBO0FBQUE7QUFBQTtBQUFBLEVBS1EsVUFBVSxLQUFjLFFBQXFCLE1BQTRCO0FBQy9FLFFBQUksVUFBVSxJQUFJLFVBQVU7QUFDNUIsV0FBTyxVQUFVLElBQUksVUFBVTtBQUcvQixVQUFNLFFBQVE7QUFBQSxNQUNaLEVBQUUsV0FBVyxlQUFlO0FBQUEsTUFDNUIsRUFBRSxXQUFXLGdCQUFnQjtBQUFBLElBQy9CLEdBQUc7QUFBQSxNQUNELFVBQVU7QUFBQSxNQUNWLFFBQVE7QUFBQSxNQUNSLE1BQU07QUFBQSxJQUNSLENBQUM7QUFHRCxVQUFNLFVBQVUsT0FBTyxjQUFjLHdCQUF3QjtBQUM3RCxRQUFJLFNBQVM7QUFDWCxZQUFNLFNBQVMsUUFBUTtBQUN2QixhQUFPLFFBQVE7QUFBQSxRQUNiLEVBQUUsUUFBUSxPQUFPLFNBQVMsRUFBRTtBQUFBLFFBQzVCLEVBQUUsUUFBUSxHQUFHLE1BQU0sTUFBTSxTQUFTLEVBQUU7QUFBQSxNQUN0QyxHQUFHO0FBQUEsUUFDRCxVQUFVO0FBQUEsUUFDVixRQUFRO0FBQUEsUUFDUixNQUFNO0FBQUEsTUFDUixDQUFDO0FBQUEsSUFDSDtBQUFBLEVBQ0Y7QUFBQTtBQUFBO0FBQUE7QUFBQSxFQUtRLFlBQVksS0FBYyxRQUFxQixNQUE0QjtBQUVqRixVQUFNLFFBQVE7QUFBQSxNQUNaLEVBQUUsV0FBVyxnQkFBZ0I7QUFBQSxNQUM3QixFQUFFLFdBQVcsZUFBZTtBQUFBLElBQzlCLEdBQUc7QUFBQSxNQUNELFVBQVU7QUFBQSxNQUNWLFFBQVE7QUFBQSxNQUNSLE1BQU07QUFBQSxJQUNSLENBQUM7QUFHRCxVQUFNLFVBQVUsT0FBTyxjQUFjLHdCQUF3QjtBQUM3RCxRQUFJLFNBQVM7QUFDWCxZQUFNLFNBQVMsUUFBUTtBQUN2QixZQUFNLFlBQVksT0FBTyxRQUFRO0FBQUEsUUFDL0IsRUFBRSxRQUFRLEdBQUcsTUFBTSxNQUFNLFNBQVMsRUFBRTtBQUFBLFFBQ3BDLEVBQUUsUUFBUSxPQUFPLFNBQVMsRUFBRTtBQUFBLE1BQzlCLEdBQUc7QUFBQSxRQUNELFVBQVU7QUFBQSxRQUNWLFFBQVE7QUFBQSxRQUNSLE1BQU07QUFBQSxNQUNSLENBQUM7QUFFRCxnQkFBVSxXQUFXLE1BQU07QUFDekIsWUFBSSxVQUFVLE9BQU8sVUFBVTtBQUMvQixlQUFPLFVBQVUsT0FBTyxVQUFVO0FBQUEsTUFDcEM7QUFBQSxJQUNGLE9BQU87QUFDTCxVQUFJLFVBQVUsT0FBTyxVQUFVO0FBQy9CLGFBQU8sVUFBVSxPQUFPLFVBQVU7QUFBQSxJQUNwQztBQUFBLEVBQ0Y7QUFBQTtBQUFBO0FBQUE7QUFBQSxFQUtRLHFCQUEyQjtBQUNqQyxVQUFNLFdBQVcsU0FBUyxjQUEyQiwrQkFBK0I7QUFDcEYsUUFBSSxDQUFDLFNBQVU7QUFFZixhQUFTLE1BQU0sU0FBUztBQUN4QixhQUFTLGlCQUFpQixTQUFTLENBQUMsTUFBTTtBQUV4QyxVQUFLLEVBQUUsT0FBdUIsUUFBUSx3QkFBd0IsRUFBRztBQUVqRSxXQUFLLFlBQVksWUFBWTtBQUFBLElBQy9CLENBQUM7QUFBQSxFQUNIO0FBQ0Y7QUExVzZCO0FBQXRCLElBQU0sa0JBQU47OztBQ1NBLElBQU0sT0FBTixNQUFNLEtBQUk7QUFBQSxFQVFmLGNBQWM7QUFFWixTQUFLLFVBQVUsSUFBSSxrQkFBa0I7QUFDckMsU0FBSyxVQUFVLElBQUksaUJBQWlCO0FBQ3BDLFNBQUssUUFBUSxJQUFJLGdCQUFnQjtBQUNqQyxTQUFLLFNBQVMsSUFBSSxpQkFBaUI7QUFDbkMsU0FBSyxhQUFhLElBQUkscUJBQXFCLEtBQUssT0FBTztBQUN2RCxTQUFLLFFBQVEsSUFBSSxnQkFBZ0I7QUFBQSxFQUNuQztBQUNGO0FBakJpQjtBQUFWLElBQU0sTUFBTjtBQXNCUCxJQUFJO0FBS0osU0FBUyxPQUFhO0FBQ3BCLFFBQU0sSUFBSSxJQUFJO0FBR2QsTUFBSSxPQUFPLFdBQVcsYUFBYTtBQUNqQyxJQUFDLE9BQW1DLE1BQU07QUFBQSxFQUM1QztBQUNGO0FBUFM7QUFVVCxJQUFJLFNBQVMsZUFBZSxXQUFXO0FBQ3JDLFdBQVMsaUJBQWlCLG9CQUFvQixJQUFJO0FBQ3BELE9BQU87QUFDTCxPQUFLO0FBQ1A7QUFHQSxJQUFPLGNBQVE7IiwKICAibmFtZXMiOiBbXQp9Cg== diff --git a/PlanTempus.Application/wwwroot/ts/app.ts b/PlanTempus.Application/wwwroot/ts/app.ts index 16735f5..d9b665c 100644 --- a/PlanTempus.Application/wwwroot/ts/app.ts +++ b/PlanTempus.Application/wwwroot/ts/app.ts @@ -9,6 +9,7 @@ import { DrawerController } from './modules/drawers'; import { ThemeController } from './modules/theme'; import { SearchController } from './modules/search'; import { LockScreenController } from './modules/lockscreen'; +import { KasseController } from './modules/kasse'; /** * Main application class @@ -19,6 +20,7 @@ export class App { readonly theme: ThemeController; readonly search: SearchController; readonly lockScreen: LockScreenController; + readonly kasse: KasseController; constructor() { // Initialize controllers @@ -27,6 +29,7 @@ export class App { this.theme = new ThemeController(); this.search = new SearchController(); this.lockScreen = new LockScreenController(this.drawers); + this.kasse = new KasseController(); } } diff --git a/PlanTempus.Application/wwwroot/ts/modules/kasse.ts b/PlanTempus.Application/wwwroot/ts/modules/kasse.ts new file mode 100644 index 0000000..0a6213a --- /dev/null +++ b/PlanTempus.Application/wwwroot/ts/modules/kasse.ts @@ -0,0 +1,370 @@ +/** + * Kasse Controller + * + * Handles tab switching, cash calculations, and form interactions + * for the Kasse (Cash Register) page. + */ + +export class KasseController { + // Base values (from system - would come from server in real app) + private readonly startBalance = 2000; + private readonly cashSales = 3540; + + constructor() { + this.setupTabs(); + this.setupCashCalculation(); + this.setupCheckboxSelection(); + this.setupApprovalCheckbox(); + this.setupDateFilters(); + this.setupRowToggle(); + this.setupDraftRowClick(); + } + + /** + * Setup tab switching functionality + */ + private setupTabs(): void { + const tabs = document.querySelectorAll('swp-tab[data-tab]'); + + tabs.forEach(tab => { + tab.addEventListener('click', () => { + const targetTab = tab.dataset.tab; + if (targetTab) { + this.switchToTab(targetTab); + } + }); + }); + } + + /** + * Switch to a specific tab by name + */ + private switchToTab(targetTab: string): void { + const tabs = document.querySelectorAll('swp-tab[data-tab]'); + const contents = document.querySelectorAll('swp-tab-content[data-tab]'); + const statsBars = document.querySelectorAll('swp-kasse-stats[data-for-tab]'); + + // Update tab states + tabs.forEach(t => { + if (t.dataset.tab === targetTab) { + t.classList.add('active'); + } else { + t.classList.remove('active'); + } + }); + + // Update content visibility + contents.forEach(content => { + if (content.dataset.tab === targetTab) { + content.classList.add('active'); + } else { + content.classList.remove('active'); + } + }); + + // Update stats bar visibility + statsBars.forEach(stats => { + if (stats.dataset.forTab === targetTab) { + stats.classList.add('active'); + } else { + stats.classList.remove('active'); + } + }); + } + + /** + * Setup cash calculation with real-time updates + */ + private setupCashCalculation(): void { + const payoutsInput = document.getElementById('payouts') as HTMLInputElement; + const toBankInput = document.getElementById('toBank') as HTMLInputElement; + const actualCashInput = document.getElementById('actualCash') as HTMLInputElement; + + if (!payoutsInput || !toBankInput || !actualCashInput) return; + + const calculate = () => this.calculateCash(payoutsInput, toBankInput, actualCashInput); + + payoutsInput.addEventListener('input', calculate); + toBankInput.addEventListener('input', calculate); + actualCashInput.addEventListener('input', calculate); + + // Initial calculation + calculate(); + } + + /** + * Calculate expected cash and difference + */ + private calculateCash( + payoutsInput: HTMLInputElement, + toBankInput: HTMLInputElement, + actualCashInput: HTMLInputElement + ): void { + const payouts = this.parseNumber(payoutsInput.value); + const toBank = this.parseNumber(toBankInput.value); + const actual = this.parseNumber(actualCashInput.value); + + // Expected = start + sales - payouts - to bank + const expectedCash = this.startBalance + this.cashSales - payouts - toBank; + + const expectedElement = document.getElementById('expectedCash'); + if (expectedElement) { + expectedElement.textContent = this.formatNumber(expectedCash); + } + + // Calculate and display difference + this.updateDifference(actual, expectedCash, actualCashInput.value); + } + + /** + * Update difference box with color coding + */ + private updateDifference(actual: number, expected: number, rawValue: string): void { + const box = document.getElementById('differenceBox'); + const value = document.getElementById('differenceValue'); + if (!box || !value) return; + + const diff = actual - expected; + + // Remove all state classes + box.classList.remove('positive', 'negative', 'neutral'); + + if (actual === 0 && rawValue === '') { + // No input yet + value.textContent = '– kr'; + box.classList.add('neutral'); + } else if (diff > 0) { + // More cash than expected + value.textContent = '+' + this.formatNumber(diff) + ' kr'; + box.classList.add('positive'); + } else if (diff < 0) { + // Less cash than expected + value.textContent = this.formatNumber(diff) + ' kr'; + box.classList.add('negative'); + } else { + // Exact match + value.textContent = '0,00 kr'; + box.classList.add('neutral'); + } + } + + /** + * Setup checkbox selection for table rows + */ + private setupCheckboxSelection(): void { + const selectAll = document.getElementById('selectAll') as HTMLInputElement; + const rowCheckboxes = document.querySelectorAll('.row-select'); + const exportBtn = document.getElementById('exportBtn') as HTMLButtonElement; + const selectionCount = document.getElementById('selectionCount'); + + if (!selectAll || !exportBtn || !selectionCount) return; + + const updateSelection = () => { + const checked = document.querySelectorAll('.row-select:checked'); + const count = checked.length; + + selectionCount.textContent = count === 0 ? '0 valgt' : `${count} valgt`; + exportBtn.disabled = count === 0; + + // Update select all state + selectAll.checked = count === rowCheckboxes.length && count > 0; + selectAll.indeterminate = count > 0 && count < rowCheckboxes.length; + }; + + selectAll.addEventListener('change', () => { + rowCheckboxes.forEach(cb => cb.checked = selectAll.checked); + updateSelection(); + }); + + rowCheckboxes.forEach(cb => { + cb.addEventListener('change', updateSelection); + // Stop click from bubbling to row + cb.addEventListener('click', e => e.stopPropagation()); + }); + } + + /** + * Setup approval checkbox to enable/disable approve button + */ + private setupApprovalCheckbox(): void { + const checkbox = document.getElementById('confirmCheckbox') as HTMLInputElement; + const approveBtn = document.getElementById('approveBtn') as HTMLButtonElement; + + if (!checkbox || !approveBtn) return; + + checkbox.addEventListener('change', () => { + approveBtn.disabled = !checkbox.checked; + }); + } + + /** + * Setup date filter defaults (last 30 days) + */ + private setupDateFilters(): void { + const dateFrom = document.getElementById('dateFrom') as HTMLInputElement; + const dateTo = document.getElementById('dateTo') as HTMLInputElement; + + if (!dateFrom || !dateTo) return; + + const today = new Date(); + const thirtyDaysAgo = new Date(today); + thirtyDaysAgo.setDate(today.getDate() - 30); + + dateTo.value = this.formatDateISO(today); + dateFrom.value = this.formatDateISO(thirtyDaysAgo); + } + + /** + * Format number as Danish currency + */ + private formatNumber(num: number): string { + return num.toLocaleString('da-DK', { + minimumFractionDigits: 2, + maximumFractionDigits: 2 + }); + } + + /** + * Parse Danish number format + */ + private parseNumber(str: string): number { + if (!str) return 0; + return parseFloat(str.replace(/\./g, '').replace(',', '.')) || 0; + } + + /** + * Format date as ISO string (YYYY-MM-DD) + */ + private formatDateISO(date: Date): string { + return date.toISOString().split('T')[0]; + } + + /** + * Setup row toggle for expandable details + */ + private setupRowToggle(): void { + const rows = document.querySelectorAll('swp-kasse-table-row[data-id]:not(.draft-row)'); + + rows.forEach(row => { + const rowId = row.getAttribute('data-id'); + if (!rowId) return; + + const detail = document.querySelector(`swp-kasse-row-detail[data-for="${rowId}"]`); + if (!detail) return; + + row.addEventListener('click', (e) => { + // Don't toggle if clicking on checkbox + if ((e.target as HTMLElement).closest('input[type="checkbox"]')) return; + + const icon = row.querySelector('swp-row-toggle i'); + const isExpanded = row.classList.contains('expanded'); + + // Close other expanded rows + document.querySelectorAll('swp-kasse-table-row.expanded').forEach(r => { + if (r !== row) { + const otherId = r.getAttribute('data-id'); + if (otherId) { + const otherDetail = document.querySelector(`swp-kasse-row-detail[data-for="${otherId}"]`); + const otherIcon = r.querySelector('swp-row-toggle i'); + if (otherDetail && otherIcon) { + this.collapseRow(r, otherDetail, otherIcon as HTMLElement); + } + } + } + }); + + // Toggle current row + if (isExpanded) { + this.collapseRow(row, detail, icon); + } else { + this.expandRow(row, detail, icon); + } + }); + }); + } + + /** + * Expand a row with animation + */ + private expandRow(row: Element, detail: HTMLElement, icon: Element | null): void { + row.classList.add('expanded'); + detail.classList.add('expanded'); + + // Animate icon rotation + icon?.animate([ + { transform: 'rotate(0deg)' }, + { transform: 'rotate(90deg)' } + ], { + duration: 200, + easing: 'ease-out', + fill: 'forwards' + }); + + // Animate detail expansion + const content = detail.querySelector('swp-row-detail-content') as HTMLElement; + if (content) { + const height = content.offsetHeight; + detail.animate([ + { height: '0px', opacity: 0 }, + { height: `${height}px`, opacity: 1 } + ], { + duration: 250, + easing: 'ease-out', + fill: 'forwards' + }); + } + } + + /** + * Collapse a row with animation + */ + private collapseRow(row: Element, detail: HTMLElement, icon: Element | null): void { + // Animate icon rotation + icon?.animate([ + { transform: 'rotate(90deg)' }, + { transform: 'rotate(0deg)' } + ], { + duration: 200, + easing: 'ease-out', + fill: 'forwards' + }); + + // Animate detail collapse + const content = detail.querySelector('swp-row-detail-content') as HTMLElement; + if (content) { + const height = content.offsetHeight; + const animation = detail.animate([ + { height: `${height}px`, opacity: 1 }, + { height: '0px', opacity: 0 } + ], { + duration: 200, + easing: 'ease-out', + fill: 'forwards' + }); + + animation.onfinish = () => { + row.classList.remove('expanded'); + detail.classList.remove('expanded'); + }; + } else { + row.classList.remove('expanded'); + detail.classList.remove('expanded'); + } + } + + /** + * Setup draft row click to navigate to Kasseafstemning tab + */ + private setupDraftRowClick(): void { + const draftRow = document.querySelector('swp-kasse-table-row.draft-row'); + if (!draftRow) return; + + draftRow.style.cursor = 'pointer'; + draftRow.addEventListener('click', (e) => { + // Don't navigate if clicking on checkbox + if ((e.target as HTMLElement).closest('input[type="checkbox"]')) return; + + this.switchToTab('afstemning'); + }); + } +}