@genspectrum/dashboard-components 0.3.1 → 0.4.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (37) hide show
  1. package/custom-elements.json +86 -29
  2. package/dist/dashboard-components.js +391 -207
  3. package/dist/dashboard-components.js.map +1 -1
  4. package/dist/genspectrum-components.d.ts +37 -12
  5. package/dist/style.css +58 -50
  6. package/package.json +2 -1
  7. package/src/operator/RenameFieldOperator.spec.ts +28 -0
  8. package/src/operator/RenameFieldOperator.ts +19 -0
  9. package/src/preact/aggregatedData/aggregate.tsx +1 -1
  10. package/src/preact/components/info.stories.tsx +8 -8
  11. package/src/preact/components/info.tsx +113 -20
  12. package/src/preact/dateRangeSelector/computeInitialValues.spec.ts +99 -0
  13. package/src/preact/dateRangeSelector/computeInitialValues.ts +73 -0
  14. package/src/preact/dateRangeSelector/date-range-selector.stories.tsx +93 -4
  15. package/src/preact/dateRangeSelector/date-range-selector.tsx +27 -88
  16. package/src/preact/dateRangeSelector/selectableOptions.ts +79 -0
  17. package/src/preact/locationFilter/location-filter.tsx +1 -1
  18. package/src/preact/mutationComparison/mutation-comparison.tsx +1 -1
  19. package/src/preact/mutationFilter/mutation-filter.stories.tsx +3 -6
  20. package/src/preact/mutationFilter/mutation-filter.tsx +48 -54
  21. package/src/preact/mutations/mutations.tsx +1 -1
  22. package/src/preact/prevalenceOverTime/prevalence-over-time.stories.tsx +3 -0
  23. package/src/preact/prevalenceOverTime/prevalence-over-time.tsx +7 -3
  24. package/src/preact/relativeGrowthAdvantage/relative-growth-advantage.stories.tsx +2 -0
  25. package/src/preact/relativeGrowthAdvantage/relative-growth-advantage.tsx +6 -2
  26. package/src/query/queryPrevalenceOverTime.ts +13 -7
  27. package/src/query/queryRelativeGrowthAdvantage.ts +11 -7
  28. package/src/web-components/input/gs-date-range-selector.stories.ts +11 -5
  29. package/src/web-components/input/gs-date-range-selector.tsx +22 -5
  30. package/src/web-components/input/gs-location-filter.stories.ts +6 -7
  31. package/src/web-components/input/gs-location-filter.tsx +3 -2
  32. package/src/web-components/input/gs-mutation-filter.stories.ts +1 -8
  33. package/src/web-components/input/gs-mutation-filter.tsx +1 -9
  34. package/src/web-components/visualization/gs-prevalence-over-time.stories.ts +6 -2
  35. package/src/web-components/visualization/gs-prevalence-over-time.tsx +11 -0
  36. package/src/web-components/visualization/gs-relative-growth-advantage.stories.ts +3 -0
  37. package/src/web-components/visualization/gs-relative-growth-advantage.tsx +11 -0
@@ -6,6 +6,7 @@ import { options, createContext as createContext$1, Fragment, render } from "pre
6
6
  import { Grid } from "gridjs";
7
7
  import { Chart, registerables, Scale } from "chart.js";
8
8
  import { VennDiagramController, ArcSlice, extractSets } from "chartjs-chart-venn";
9
+ import { autoUpdate, computePosition, offset, shift, size } from "@floating-ui/dom";
9
10
  import { ReactiveElement } from "@lit/reactive-element";
10
11
  import { BarWithErrorBarsController, BarWithErrorBar } from "chartjs-chart-error-bars";
11
12
  import flatpickr from "flatpickr";
@@ -1308,8 +1309,8 @@ const ErrorDisplay = ({ error }) => {
1308
1309
  ] })
1309
1310
  ] });
1310
1311
  };
1311
- const ResizeContainer = ({ children, size }) => {
1312
- return /* @__PURE__ */ u$1("div", { style: size, children });
1312
+ const ResizeContainer = ({ children, size: size2 }) => {
1313
+ return /* @__PURE__ */ u$1("div", { style: size2, children });
1313
1314
  };
1314
1315
  const Headline = ({ heading, children }) => {
1315
1316
  if (!heading) {
@@ -1331,33 +1332,107 @@ const ResizingHeadline = ({ heading, children }) => {
1331
1332
  /* @__PURE__ */ u$1("div", { style: { height: `calc(100% - ${h1Height})` }, children })
1332
1333
  ] });
1333
1334
  };
1334
- const ErrorBoundary = ({ size, headline, children }) => {
1335
+ const ErrorBoundary = ({ size: size2, headline, children }) => {
1335
1336
  const [internalError] = b2();
1336
1337
  if (internalError) {
1337
- return /* @__PURE__ */ u$1(ResizeContainer, { size, children: /* @__PURE__ */ u$1(Headline, { heading: headline, children: /* @__PURE__ */ u$1(ErrorDisplay, { error: internalError }) }) });
1338
+ return /* @__PURE__ */ u$1(ResizeContainer, { size: size2, children: /* @__PURE__ */ u$1(Headline, { heading: headline, children: /* @__PURE__ */ u$1(ErrorDisplay, { error: internalError }) }) });
1338
1339
  }
1339
1340
  return /* @__PURE__ */ u$1(Fragment, { children });
1340
1341
  };
1341
- const Info = ({ children, size }) => {
1342
+ const Info = ({ children, height }) => {
1342
1343
  const [showHelp, setShowHelp] = p(false);
1344
+ const referenceRef = F(null);
1345
+ const floatingRef = F(null);
1346
+ useFloatingUi(referenceRef, floatingRef, height, showHelp);
1343
1347
  const toggleHelp = () => {
1344
1348
  setShowHelp(!showHelp);
1345
1349
  };
1346
- return /* @__PURE__ */ u$1("div", { className: "relative", children: [
1347
- /* @__PURE__ */ u$1("button", { className: "btn btn-xs", onClick: toggleHelp, children: "?" }),
1348
- showHelp && /* @__PURE__ */ u$1(
1350
+ useCloseOnEsc(setShowHelp);
1351
+ useCloseOnClickOutside(floatingRef, referenceRef, setShowHelp);
1352
+ return /* @__PURE__ */ u$1("div", { className: "relative z-10", children: [
1353
+ /* @__PURE__ */ u$1("button", { type: "button", className: "btn btn-xs", onClick: toggleHelp, ref: referenceRef, children: "?" }),
1354
+ /* @__PURE__ */ u$1(
1349
1355
  "div",
1350
1356
  {
1351
- className: "absolute top-8 right-6 bg-white p-2 border border-black flex flex-col overflow-auto shadow-lg rounded z-50",
1352
- style: size,
1357
+ ref: floatingRef,
1358
+ className: "bg-white p-2 border border-gray-100 shadow-lg rounded overflow-y-auto opacity-90",
1359
+ style: { position: "absolute", zIndex: 10, display: showHelp ? "" : "none" },
1353
1360
  children: [
1354
1361
  /* @__PURE__ */ u$1("div", { className: "flex flex-col", children }),
1355
- /* @__PURE__ */ u$1("div", { className: "flex justify-end", children: /* @__PURE__ */ u$1("button", { className: "text-sm underline mt-2", onClick: toggleHelp, children: "Close" }) })
1362
+ /* @__PURE__ */ u$1(
1363
+ "button",
1364
+ {
1365
+ onClick: () => setShowHelp(false),
1366
+ className: "float-right underline text-sm hover:text-blue-700 mr-2",
1367
+ children: "Close"
1368
+ }
1369
+ )
1356
1370
  ]
1357
1371
  }
1358
1372
  )
1359
1373
  ] });
1360
1374
  };
1375
+ function useFloatingUi(referenceRef, floatingRef, height, showHelp) {
1376
+ const cleanupRef = F(null);
1377
+ _(() => {
1378
+ if (!referenceRef.current || !floatingRef.current) {
1379
+ return;
1380
+ }
1381
+ const { current: reference } = referenceRef;
1382
+ const { current: floating } = floatingRef;
1383
+ const update = () => {
1384
+ computePosition(reference, floating, {
1385
+ middleware: [
1386
+ offset(10),
1387
+ shift(),
1388
+ size({
1389
+ apply({}) {
1390
+ floating.style.width = "100vw";
1391
+ floating.style.height = height ? height : "50vh";
1392
+ }
1393
+ })
1394
+ ]
1395
+ }).then(({ x, y: y2 }) => {
1396
+ floating.style.left = `${x}px`;
1397
+ floating.style.top = `${y2}px`;
1398
+ });
1399
+ };
1400
+ update();
1401
+ cleanupRef.current = autoUpdate(reference, floating, update);
1402
+ return () => {
1403
+ if (cleanupRef.current) {
1404
+ cleanupRef.current();
1405
+ }
1406
+ };
1407
+ }, [showHelp, height, referenceRef, floatingRef]);
1408
+ }
1409
+ function useCloseOnClickOutside(floatingRef, referenceRef, setShowHelp) {
1410
+ _(() => {
1411
+ const handleClickOutside = (event) => {
1412
+ const path = event.composedPath();
1413
+ if (floatingRef.current && !path.includes(floatingRef.current) && referenceRef.current && !path.includes(referenceRef.current)) {
1414
+ setShowHelp(false);
1415
+ }
1416
+ };
1417
+ document.addEventListener("mousedown", handleClickOutside);
1418
+ return () => {
1419
+ document.removeEventListener("mousedown", handleClickOutside);
1420
+ };
1421
+ }, [floatingRef, referenceRef, setShowHelp]);
1422
+ }
1423
+ function useCloseOnEsc(setShowHelp) {
1424
+ _(() => {
1425
+ const handleKeyDown = (event) => {
1426
+ if (event.key === "Escape") {
1427
+ setShowHelp(false);
1428
+ }
1429
+ };
1430
+ document.addEventListener("keydown", handleKeyDown);
1431
+ return () => {
1432
+ document.removeEventListener("keydown", handleKeyDown);
1433
+ };
1434
+ }, [setShowHelp]);
1435
+ }
1361
1436
  const InfoHeadline1 = ({ children }) => {
1362
1437
  return /* @__PURE__ */ u$1("h1", { className: "text-lg font-bold", children });
1363
1438
  };
@@ -1627,8 +1702,8 @@ const MutationComparison = ({
1627
1702
  height,
1628
1703
  headline = "Mutation comparison"
1629
1704
  }) => {
1630
- const size = { height, width };
1631
- return /* @__PURE__ */ u$1(ErrorBoundary, { size, headline, children: /* @__PURE__ */ u$1(ResizeContainer, { size, children: /* @__PURE__ */ u$1(Headline, { heading: headline, children: /* @__PURE__ */ u$1(MutationComparisonInner, { variants, sequenceType, views }) }) }) });
1705
+ const size2 = { height, width };
1706
+ return /* @__PURE__ */ u$1(ErrorBoundary, { size: size2, headline, children: /* @__PURE__ */ u$1(ResizeContainer, { size: size2, children: /* @__PURE__ */ u$1(Headline, { heading: headline, children: /* @__PURE__ */ u$1(MutationComparisonInner, { variants, sequenceType, views }) }) }) });
1632
1707
  };
1633
1708
  const MutationComparisonInner = ({
1634
1709
  variants,
@@ -1741,7 +1816,7 @@ const Toolbar$3 = ({
1741
1816
  filename: "mutation_comparison.csv"
1742
1817
  }
1743
1818
  ),
1744
- /* @__PURE__ */ u$1(Info, { children: "Info for mutation comparison" })
1819
+ /* @__PURE__ */ u$1(Info, { height: "100px", children: "Info for mutation comparison" })
1745
1820
  ] });
1746
1821
  };
1747
1822
  const gridJsStyle = '.gridjs-head button, .gridjs-footer button {\n cursor: pointer;\n background-color: transparent;\n background-image: none;\n padding: 0;\n margin: 0;\n border: none;\n outline: none;\n}\n\n.gridjs-temp {\n position: relative;\n}\n\n.gridjs-head {\n width: 100%;\n margin-bottom: 5px;\n padding: 5px 1px;\n}\n.gridjs-head::after {\n content: "";\n display: block;\n clear: both;\n}\n.gridjs-head:empty {\n padding: 0;\n border: none;\n}\n\n.gridjs-container {\n overflow: hidden;\n display: inline-block;\n padding: 2px;\n color: #000;\n position: relative;\n z-index: 0;\n}\n\n.gridjs-footer {\n display: block;\n position: relative;\n width: 100%;\n z-index: 5;\n padding: 12px 24px;\n border-top: 1px solid #e5e7eb;\n background-color: #fff;\n box-shadow: 0 1px 3px 0 rgba(0, 0, 0, 0.1), 0 1px 2px 0 rgba(0, 0, 0, 0.26);\n border-radius: 0 0 8px 8px;\n border-bottom-width: 1px;\n border-color: #e5e7eb;\n}\n.gridjs-footer:empty {\n padding: 0;\n border: none;\n}\n\ninput.gridjs-input {\n outline: none;\n background-color: #fff;\n border: 1px solid #d2d6dc;\n border-radius: 5px;\n padding: 10px 13px;\n font-size: 14px;\n line-height: 1.45;\n -webkit-appearance: none;\n -moz-appearance: none;\n appearance: none;\n}\ninput.gridjs-input:focus {\n box-shadow: 0 0 0 3px rgba(149, 189, 243, 0.5);\n border-color: #9bc2f7;\n}\n\n.gridjs-pagination {\n color: #3d4044;\n}\n.gridjs-pagination::after {\n content: "";\n display: block;\n clear: both;\n}\n.gridjs-pagination .gridjs-summary {\n float: left;\n margin-top: 5px;\n}\n.gridjs-pagination .gridjs-pages {\n float: right;\n}\n.gridjs-pagination .gridjs-pages button {\n padding: 5px 14px;\n border: 1px solid #d2d6dc;\n background-color: #fff;\n border-right: none;\n outline: none;\n -webkit-user-select: none;\n -moz-user-select: none;\n user-select: none;\n}\n.gridjs-pagination .gridjs-pages button:focus {\n box-shadow: 0 0 0 2px rgba(149, 189, 243, 0.5);\n position: relative;\n margin-right: -1px;\n border-right: 1px solid #d2d6dc;\n}\n.gridjs-pagination .gridjs-pages button:hover {\n background-color: #f7f7f7;\n color: rgb(60, 66, 87);\n outline: none;\n}\n.gridjs-pagination .gridjs-pages button:disabled,\n.gridjs-pagination .gridjs-pages button[disabled],\n.gridjs-pagination .gridjs-pages button:hover:disabled {\n cursor: default;\n background-color: #fff;\n color: #6b7280;\n}\n.gridjs-pagination .gridjs-pages button.gridjs-spread {\n cursor: default;\n box-shadow: none;\n background-color: #fff;\n}\n.gridjs-pagination .gridjs-pages button.gridjs-currentPage {\n background-color: #f7f7f7;\n font-weight: bold;\n}\n.gridjs-pagination .gridjs-pages button:last-child {\n border-bottom-right-radius: 6px;\n border-top-right-radius: 6px;\n border-right: 1px solid #d2d6dc;\n}\n.gridjs-pagination .gridjs-pages button:first-child {\n border-bottom-left-radius: 6px;\n border-top-left-radius: 6px;\n}\n.gridjs-pagination .gridjs-pages button:last-child:focus {\n margin-right: 0;\n}\n\nbutton.gridjs-sort {\n float: right;\n height: 24px;\n width: 13px;\n background-color: transparent;\n background-repeat: no-repeat;\n background-position-x: center;\n cursor: pointer;\n padding: 0;\n margin: 0;\n border: none;\n outline: none;\n background-size: contain;\n}\nbutton.gridjs-sort-neutral {\n opacity: 0.3;\n background-image: url("");\n background-position-y: center;\n}\nbutton.gridjs-sort-asc {\n background-image: url("");\n background-position-y: 35%;\n background-size: 10px;\n}\nbutton.gridjs-sort-desc {\n background-image: url("");\n background-position-y: 65%;\n background-size: 10px;\n}\nbutton.gridjs-sort:focus {\n outline: none;\n}\n\ntable.gridjs-table {\n width: 100%;\n max-width: 100%;\n border-collapse: collapse;\n text-align: left;\n display: table;\n margin: 0;\n padding: 0;\n overflow: auto;\n table-layout: fixed;\n}\n\n.gridjs-tbody {\n background-color: #fff;\n}\n\ntd.gridjs-td {\n border: 1px solid #e5e7eb;\n padding: 12px 24px;\n background-color: #fff;\n box-sizing: content-box;\n}\ntd.gridjs-td:first-child {\n border-left: none;\n}\ntd.gridjs-td:last-child {\n border-right: none;\n}\ntd.gridjs-message {\n text-align: center;\n}\n\nth.gridjs-th {\n position: relative;\n color: #6b7280;\n background-color: #f9fafb;\n border: 1px solid #e5e7eb;\n border-top: none;\n padding: 14px 24px;\n -webkit-user-select: none;\n -moz-user-select: none;\n user-select: none;\n box-sizing: border-box;\n white-space: nowrap;\n outline: none;\n vertical-align: middle;\n}\nth.gridjs-th .gridjs-th-content {\n text-overflow: ellipsis;\n overflow: hidden;\n width: 100%;\n float: left;\n}\nth.gridjs-th-sort {\n cursor: pointer;\n}\nth.gridjs-th-sort .gridjs-th-content {\n width: calc(100% - 15px);\n}\nth.gridjs-th-sort:hover {\n background-color: #e5e7eb;\n}\nth.gridjs-th-sort:focus {\n background-color: #e5e7eb;\n}\nth.gridjs-th-fixed {\n position: sticky;\n box-shadow: 0 1px 0 0 #e5e7eb;\n}\n@supports (-moz-appearance: none) {\n th.gridjs-th-fixed {\n box-shadow: 0 0 0 1px #e5e7eb;\n }\n}\nth.gridjs-th:first-child {\n border-left: none;\n}\nth.gridjs-th:last-child {\n border-right: none;\n}\n\n.gridjs-tr {\n border: none;\n}\n.gridjs-tr-selected td {\n background-color: #ebf5ff;\n}\n.gridjs-tr:last-child td {\n border-bottom: 0;\n}\n\n.gridjs *,\n.gridjs :after,\n.gridjs :before {\n box-sizing: border-box;\n}\n\n.gridjs-wrapper {\n position: relative;\n z-index: 1;\n overflow: auto;\n width: 100%;\n -webkit-font-smoothing: antialiased;\n -moz-osx-font-smoothing: grayscale;\n box-shadow: 0 1px 3px 0 rgba(0, 0, 0, 0.1), 0 1px 2px 0 rgba(0, 0, 0, 0.26);\n border-radius: 8px 8px 0 0;\n display: block;\n border-top-width: 1px;\n border-color: #e5e7eb;\n}\n.gridjs-wrapper:nth-last-of-type(2) {\n border-radius: 8px;\n border-bottom-width: 1px;\n}\n\n.gridjs-search {\n float: left;\n}\n.gridjs-search-input {\n width: 250px;\n}\n\n.gridjs-loading-bar {\n z-index: 10;\n position: absolute;\n left: 0;\n right: 0;\n top: 0;\n bottom: 0;\n background-color: #fff;\n opacity: 0.5;\n}\n.gridjs-loading-bar::after {\n position: absolute;\n top: 0;\n right: 0;\n bottom: 0;\n left: 0;\n transform: translateX(-100%);\n background-image: linear-gradient(90deg, rgba(204, 204, 204, 0) 0, rgba(204, 204, 204, 0.2) 20%, rgba(204, 204, 204, 0.5) 60%, rgba(204, 204, 204, 0));\n animation: shimmer 2s infinite;\n content: "";\n}\n@keyframes shimmer {\n 100% {\n transform: translateX(100%);\n }\n}\n\n.gridjs-td .gridjs-checkbox {\n display: block;\n margin: auto;\n cursor: pointer;\n}\n\n.gridjs-resizable {\n position: absolute;\n top: 0;\n bottom: 0;\n right: 0;\n width: 5px;\n}\n.gridjs-resizable:hover {\n cursor: ew-resize;\n background-color: #9bc2f7;\n}\n/*# sourceMappingURL=mermaid.css?inline.map */';
@@ -4224,24 +4299,30 @@ input.tab:checked + .tab-content,
4224
4299
  .relative {
4225
4300
  position: relative;
4226
4301
  }
4302
+ .-right-3 {
4303
+ right: -0.75rem;
4304
+ }
4305
+ .-top-3 {
4306
+ top: -0.75rem;
4307
+ }
4227
4308
  .right-2 {
4228
4309
  right: 0.5rem;
4229
4310
  }
4230
- .right-6 {
4231
- right: 1.5rem;
4232
- }
4233
4311
  .top-2 {
4234
4312
  top: 0.5rem;
4235
4313
  }
4236
- .top-8 {
4237
- top: 2rem;
4238
- }
4239
- .z-50 {
4240
- z-index: 50;
4314
+ .z-10 {
4315
+ z-index: 10;
4241
4316
  }
4242
4317
  .z-\\[1\\] {
4243
4318
  z-index: 1;
4244
4319
  }
4320
+ .float-right {
4321
+ float: right;
4322
+ }
4323
+ .m-1 {
4324
+ margin: 0.25rem;
4325
+ }
4245
4326
  .m-2 {
4246
4327
  margin: 0.5rem;
4247
4328
  }
@@ -4257,24 +4338,21 @@ input.tab:checked + .tab-content,
4257
4338
  margin-top: 1rem;
4258
4339
  margin-bottom: 1rem;
4259
4340
  }
4260
- .mb-1 {
4261
- margin-bottom: 0.25rem;
4262
- }
4263
4341
  .mb-2 {
4264
4342
  margin-bottom: 0.5rem;
4265
4343
  }
4266
- .me-1 {
4267
- margin-inline-end: 0.25rem;
4268
- }
4269
4344
  .ml-2 {
4270
4345
  margin-left: 0.5rem;
4271
4346
  }
4272
- .mt-2 {
4273
- margin-top: 0.5rem;
4347
+ .mr-2 {
4348
+ margin-right: 0.5rem;
4274
4349
  }
4275
4350
  .mt-4 {
4276
4351
  margin-top: 1rem;
4277
4352
  }
4353
+ .inline-block {
4354
+ display: inline-block;
4355
+ }
4278
4356
  .inline {
4279
4357
  display: inline;
4280
4358
  }
@@ -4308,13 +4386,6 @@ input.tab:checked + .tab-content,
4308
4386
  .w-full {
4309
4387
  width: 100%;
4310
4388
  }
4311
- .min-w-0 {
4312
- min-width: 0px;
4313
- }
4314
- .min-w-max {
4315
- min-width: -moz-max-content;
4316
- min-width: max-content;
4317
- }
4318
4389
  .max-w-screen-lg {
4319
4390
  max-width: 1024px;
4320
4391
  }
@@ -4333,32 +4404,23 @@ input.tab:checked + .tab-content,
4333
4404
  .flex-wrap {
4334
4405
  flex-wrap: wrap;
4335
4406
  }
4336
- .flex-nowrap {
4337
- flex-wrap: nowrap;
4338
- }
4339
4407
  .items-center {
4340
4408
  align-items: center;
4341
4409
  }
4342
- .justify-end {
4343
- justify-content: flex-end;
4344
- }
4345
4410
  .justify-center {
4346
4411
  justify-content: center;
4347
4412
  }
4348
4413
  .justify-between {
4349
4414
  justify-content: space-between;
4350
4415
  }
4351
- .gap-1 {
4352
- gap: 0.25rem;
4353
- }
4354
4416
  .gap-2 {
4355
4417
  gap: 0.5rem;
4356
4418
  }
4357
4419
  .overflow-auto {
4358
4420
  overflow: auto;
4359
4421
  }
4360
- .overflow-scroll {
4361
- overflow: scroll;
4422
+ .overflow-y-auto {
4423
+ overflow-y: auto;
4362
4424
  }
4363
4425
  .whitespace-nowrap {
4364
4426
  white-space: nowrap;
@@ -4375,8 +4437,8 @@ input.tab:checked + .tab-content,
4375
4437
  .rounded-box {
4376
4438
  border-radius: var(--rounded-box, 1rem);
4377
4439
  }
4378
- .rounded-lg {
4379
- border-radius: 0.5rem;
4440
+ .rounded-full {
4441
+ border-radius: 9999px;
4380
4442
  }
4381
4443
  .rounded-md {
4382
4444
  border-radius: 0.375rem;
@@ -4403,9 +4465,11 @@ input.tab:checked + .tab-content,
4403
4465
  .border-b-2 {
4404
4466
  border-bottom-width: 2px;
4405
4467
  }
4406
- .border-black {
4407
- --tw-border-opacity: 1;
4408
- border-color: rgb(0 0 0 / var(--tw-border-opacity));
4468
+ .border-solid {
4469
+ border-style: solid;
4470
+ }
4471
+ .border-none {
4472
+ border-style: none;
4409
4473
  }
4410
4474
  .border-error {
4411
4475
  --tw-border-opacity: 1;
@@ -4423,6 +4487,10 @@ input.tab:checked + .tab-content,
4423
4487
  --tw-border-opacity: 1;
4424
4488
  border-color: rgb(156 163 175 / var(--tw-border-opacity));
4425
4489
  }
4490
+ .border-red-500 {
4491
+ --tw-border-opacity: 1;
4492
+ border-color: rgb(239 68 68 / var(--tw-border-opacity));
4493
+ }
4426
4494
  .bg-base-100 {
4427
4495
  --tw-bg-opacity: 1;
4428
4496
  background-color: var(--fallback-b1,oklch(var(--b1)/var(--tw-bg-opacity)));
@@ -4435,6 +4503,9 @@ input.tab:checked + .tab-content,
4435
4503
  --tw-bg-opacity: 1;
4436
4504
  background-color: rgb(255 255 255 / var(--tw-bg-opacity));
4437
4505
  }
4506
+ .p-1 {
4507
+ padding: 0.25rem;
4508
+ }
4438
4509
  .p-2 {
4439
4510
  padding: 0.5rem;
4440
4511
  }
@@ -4442,21 +4513,13 @@ input.tab:checked + .tab-content,
4442
4513
  padding-left: 0.5rem;
4443
4514
  padding-right: 0.5rem;
4444
4515
  }
4445
- .px-2\\.5 {
4446
- padding-left: 0.625rem;
4447
- padding-right: 0.625rem;
4448
- }
4449
4516
  .px-4 {
4450
4517
  padding-left: 1rem;
4451
4518
  padding-right: 1rem;
4452
4519
  }
4453
- .py-0 {
4454
- padding-top: 0px;
4455
- padding-bottom: 0px;
4456
- }
4457
- .py-0\\.5 {
4458
- padding-top: 0.125rem;
4459
- padding-bottom: 0.125rem;
4520
+ .py-1 {
4521
+ padding-top: 0.25rem;
4522
+ padding-bottom: 0.25rem;
4460
4523
  }
4461
4524
  .py-16 {
4462
4525
  padding-top: 4rem;
@@ -4517,6 +4580,9 @@ input.tab:checked + .tab-content,
4517
4580
  .underline {
4518
4581
  text-decoration-line: underline;
4519
4582
  }
4583
+ .opacity-90 {
4584
+ opacity: 0.9;
4585
+ }
4520
4586
  .shadow {
4521
4587
  --tw-shadow: 0 1px 3px 0 rgb(0 0 0 / 0.1), 0 1px 2px -1px rgb(0 0 0 / 0.1);
4522
4588
  --tw-shadow-colored: 0 1px 3px 0 var(--tw-shadow-color), 0 1px 2px -1px var(--tw-shadow-color);
@@ -4542,10 +4608,18 @@ input.tab:checked + .tab-content,
4542
4608
  .duration-150 {
4543
4609
  transition-duration: 150ms;
4544
4610
  }
4611
+ .focus-within\\:border-gray-400:focus-within {
4612
+ --tw-border-opacity: 1;
4613
+ border-color: rgb(156 163 175 / var(--tw-border-opacity));
4614
+ }
4545
4615
  .hover\\:bg-gray-100:hover {
4546
4616
  --tw-bg-opacity: 1;
4547
4617
  background-color: rgb(243 244 246 / var(--tw-bg-opacity));
4548
4618
  }
4619
+ .hover\\:text-blue-700:hover {
4620
+ --tw-text-opacity: 1;
4621
+ color: rgb(29 78 216 / var(--tw-text-opacity));
4622
+ }
4549
4623
  .hover\\:text-blue-800:hover {
4550
4624
  --tw-text-opacity: 1;
4551
4625
  color: rgb(30 64 175 / var(--tw-text-opacity));
@@ -4557,6 +4631,15 @@ input.tab:checked + .tab-content,
4557
4631
  .hover\\:text-gray-700:hover {
4558
4632
  --tw-text-opacity: 1;
4559
4633
  color: rgb(55 65 81 / var(--tw-text-opacity));
4634
+ }
4635
+ .focus\\:outline-none:focus {
4636
+ outline: 2px solid transparent;
4637
+ outline-offset: 2px;
4638
+ }
4639
+ .focus\\:ring-0:focus {
4640
+ --tw-ring-offset-shadow: var(--tw-ring-inset) 0 0 0 var(--tw-ring-offset-width) var(--tw-ring-offset-color);
4641
+ --tw-ring-shadow: var(--tw-ring-inset) 0 0 0 calc(0px + var(--tw-ring-offset-width)) var(--tw-ring-color);
4642
+ box-shadow: var(--tw-ring-offset-shadow), var(--tw-ring-shadow), var(--tw-shadow, 0 0 #0000);
4560
4643
  }`;
4561
4644
  var __defProp$9 = Object.defineProperty;
4562
4645
  var __getOwnPropDesc$9 = Object.getOwnPropertyDescriptor;
@@ -4921,8 +5004,8 @@ const Mutations = ({
4921
5004
  height,
4922
5005
  headline = "Mutations"
4923
5006
  }) => {
4924
- const size = { height, width };
4925
- return /* @__PURE__ */ u$1(ErrorBoundary, { size, headline, children: /* @__PURE__ */ u$1(ResizeContainer, { size, children: /* @__PURE__ */ u$1(Headline, { heading: headline, children: /* @__PURE__ */ u$1(MutationsInner, { variant, sequenceType, views }) }) }) });
5007
+ const size2 = { height, width };
5008
+ return /* @__PURE__ */ u$1(ErrorBoundary, { size: size2, headline, children: /* @__PURE__ */ u$1(ResizeContainer, { size: size2, children: /* @__PURE__ */ u$1(Headline, { heading: headline, children: /* @__PURE__ */ u$1(MutationsInner, { variant, sequenceType, views }) }) }) });
4926
5009
  };
4927
5010
  const MutationsInner = ({ variant, sequenceType, views }) => {
4928
5011
  const lapis = P(LapisUrlContext);
@@ -5041,7 +5124,7 @@ const Toolbar$2 = ({
5041
5124
  filename: "insertions.csv"
5042
5125
  }
5043
5126
  ),
5044
- /* @__PURE__ */ u$1(Info, { children: "Info for mutations" })
5127
+ /* @__PURE__ */ u$1(Info, { height: "100px", children: "Info for mutations" })
5045
5128
  ] });
5046
5129
  };
5047
5130
  var __defProp$7 = Object.defineProperty;
@@ -6467,6 +6550,17 @@ class MapOperator {
6467
6550
  };
6468
6551
  }
6469
6552
  }
6553
+ class RenameFieldOperator extends MapOperator {
6554
+ constructor(child, oldFieldName, newFieldName) {
6555
+ super(
6556
+ child,
6557
+ (value) => ({
6558
+ ...value,
6559
+ [newFieldName]: value[oldFieldName]
6560
+ })
6561
+ );
6562
+ }
6563
+ }
6470
6564
  class SlidingOperator {
6471
6565
  constructor(child, windowSize, aggregate) {
6472
6566
  this.child = child;
@@ -6486,12 +6580,12 @@ class SlidingOperator {
6486
6580
  return { content };
6487
6581
  }
6488
6582
  }
6489
- function queryPrevalenceOverTime(numeratorFilter, denominatorFilter, granularity, smoothingWindow, lapis, signal) {
6583
+ function queryPrevalenceOverTime(numeratorFilter, denominatorFilter, granularity, smoothingWindow, lapis, lapisDateField, signal) {
6490
6584
  const numeratorFilters = makeArray(numeratorFilter);
6491
- const denominatorData = fetchAndPrepare(denominatorFilter, granularity, smoothingWindow);
6585
+ const denominatorData = fetchAndPrepare(denominatorFilter, granularity, smoothingWindow, lapisDateField);
6492
6586
  const subQueries = numeratorFilters.map(async (namedLapisFilter) => {
6493
6587
  const { displayName, lapisFilter } = namedLapisFilter;
6494
- const numeratorData = fetchAndPrepare(lapisFilter, granularity, smoothingWindow);
6588
+ const numeratorData = fetchAndPrepare(lapisFilter, granularity, smoothingWindow, lapisDateField);
6495
6589
  const divide = new DivisionOperator(
6496
6590
  numeratorData,
6497
6591
  denominatorData,
@@ -6515,9 +6609,10 @@ function makeArray(arrayOrSingleItem) {
6515
6609
  }
6516
6610
  return [arrayOrSingleItem];
6517
6611
  }
6518
- function fetchAndPrepare(filter, granularity, smoothingWindow) {
6519
- const fetchData = new FetchAggregatedOperator(filter, ["date"]);
6520
- const mapData = new MapOperator(fetchData, (d2) => mapDateToGranularityRange(d2, granularity));
6612
+ function fetchAndPrepare(filter, granularity, smoothingWindow, lapisDateField) {
6613
+ const fetchData = new FetchAggregatedOperator(filter, [lapisDateField]);
6614
+ const dataWithFixedDateKey = new RenameFieldOperator(fetchData, lapisDateField, "date");
6615
+ const mapData = new MapOperator(dataWithFixedDateKey, (d2) => mapDateToGranularityRange(d2, granularity));
6521
6616
  const groupByData = new GroupByAndSumOperator(mapData, "dateRange", "count");
6522
6617
  const fillData = new FillMissingOperator(
6523
6618
  groupByData,
@@ -6637,10 +6732,11 @@ const PrevalenceOverTime = ({
6637
6732
  confidenceIntervalMethods,
6638
6733
  width,
6639
6734
  height,
6640
- headline = "Prevalence over time"
6735
+ headline = "Prevalence over time",
6736
+ lapisDateField
6641
6737
  }) => {
6642
- const size = { height, width };
6643
- return /* @__PURE__ */ u$1(ErrorBoundary, { size, headline, children: /* @__PURE__ */ u$1(ResizeContainer, { size, children: /* @__PURE__ */ u$1(Headline, { heading: headline, children: /* @__PURE__ */ u$1(
6738
+ const size2 = { height, width };
6739
+ return /* @__PURE__ */ u$1(ErrorBoundary, { size: size2, headline, children: /* @__PURE__ */ u$1(ResizeContainer, { size: size2, children: /* @__PURE__ */ u$1(Headline, { heading: headline, children: /* @__PURE__ */ u$1(
6644
6740
  PrevalenceOverTimeInner,
6645
6741
  {
6646
6742
  numerator,
@@ -6648,7 +6744,8 @@ const PrevalenceOverTime = ({
6648
6744
  granularity,
6649
6745
  smoothingWindow,
6650
6746
  views,
6651
- confidenceIntervalMethods
6747
+ confidenceIntervalMethods,
6748
+ lapisDateField
6652
6749
  }
6653
6750
  ) }) }) });
6654
6751
  };
@@ -6658,11 +6755,12 @@ const PrevalenceOverTimeInner = ({
6658
6755
  granularity,
6659
6756
  smoothingWindow,
6660
6757
  views,
6661
- confidenceIntervalMethods
6758
+ confidenceIntervalMethods,
6759
+ lapisDateField
6662
6760
  }) => {
6663
6761
  const lapis = P(LapisUrlContext);
6664
6762
  const { data, error, isLoading } = useQuery(
6665
- () => queryPrevalenceOverTime(numerator, denominator, granularity, smoothingWindow, lapis),
6763
+ () => queryPrevalenceOverTime(numerator, denominator, granularity, smoothingWindow, lapis, lapisDateField),
6666
6764
  [lapis, numerator, denominator, granularity, smoothingWindow]
6667
6765
  );
6668
6766
  if (isLoading) {
@@ -6779,8 +6877,8 @@ const Toolbar$1 = ({
6779
6877
  /* @__PURE__ */ u$1(PrevalenceOverTimeInfo, {})
6780
6878
  ] });
6781
6879
  };
6782
- const PrevalenceOverTimeInfo = () => {
6783
- return /* @__PURE__ */ u$1(Info, { size: { width: "600px", height: "30vh" }, children: [
6880
+ const PrevalenceOverTimeInfo = ({}) => {
6881
+ return /* @__PURE__ */ u$1(Info, { height: "100px", children: [
6784
6882
  /* @__PURE__ */ u$1(InfoHeadline1, { children: "Prevalence over time" }),
6785
6883
  /* @__PURE__ */ u$1(InfoParagraph, { children: "Prevalence over time info." })
6786
6884
  ] });
@@ -6808,6 +6906,7 @@ let PrevalenceOverTimeComponent = class extends PreactLitAdapterWithGridJsStyles
6808
6906
  this.headline = "Prevalence over time";
6809
6907
  this.width = "100%";
6810
6908
  this.height = "700px";
6909
+ this.lapisDateField = "date";
6811
6910
  }
6812
6911
  render() {
6813
6912
  return /* @__PURE__ */ u$1(
@@ -6821,7 +6920,8 @@ let PrevalenceOverTimeComponent = class extends PreactLitAdapterWithGridJsStyles
6821
6920
  confidenceIntervalMethods: this.confidenceIntervalMethods,
6822
6921
  width: this.width,
6823
6922
  height: this.height,
6824
- headline: this.headline
6923
+ headline: this.headline,
6924
+ lapisDateField: this.lapisDateField
6825
6925
  }
6826
6926
  );
6827
6927
  }
@@ -6853,6 +6953,9 @@ __decorateClass$6([
6853
6953
  __decorateClass$6([
6854
6954
  n2({ type: String })
6855
6955
  ], PrevalenceOverTimeComponent.prototype, "height", 2);
6956
+ __decorateClass$6([
6957
+ n2({ type: String })
6958
+ ], PrevalenceOverTimeComponent.prototype, "lapisDateField", 2);
6856
6959
  PrevalenceOverTimeComponent = __decorateClass$6([
6857
6960
  t$2("gs-prevalence-over-time")
6858
6961
  ], PrevalenceOverTimeComponent);
@@ -6963,11 +7066,13 @@ const tooltip = () => {
6963
7066
  }
6964
7067
  };
6965
7068
  };
6966
- async function queryRelativeGrowthAdvantage(numerator, denominator, generationTime, lapis, signal) {
6967
- const fetchNumerator = new FetchAggregatedOperator(numerator, ["date"]);
6968
- const fetchDenominator = new FetchAggregatedOperator(denominator, ["date"]);
6969
- const mapNumerator = new MapOperator(fetchNumerator, toYearMonthDay);
6970
- const mapDenominator = new MapOperator(fetchDenominator, toYearMonthDay);
7069
+ async function queryRelativeGrowthAdvantage(numerator, denominator, generationTime, lapis, lapisDateField, signal) {
7070
+ const fetchNumerator = new FetchAggregatedOperator(numerator, [lapisDateField]);
7071
+ const fetchDenominator = new FetchAggregatedOperator(denominator, [lapisDateField]);
7072
+ const mapToFixedDateKeyNumerator = new RenameFieldOperator(fetchNumerator, lapisDateField, "date");
7073
+ const mapToFixedDateKeyDenominator = new RenameFieldOperator(fetchDenominator, lapisDateField, "date");
7074
+ const mapNumerator = new MapOperator(mapToFixedDateKeyNumerator, toYearMonthDay);
7075
+ const mapDenominator = new MapOperator(mapToFixedDateKeyDenominator, toYearMonthDay);
6971
7076
  const [numeratorData, denominatorData] = await Promise.all([
6972
7077
  mapNumerator.evaluate(lapis, signal),
6973
7078
  mapDenominator.evaluate(lapis, signal)
@@ -7048,16 +7153,18 @@ const RelativeGrowthAdvantage = ({
7048
7153
  numerator,
7049
7154
  denominator,
7050
7155
  generationTime,
7051
- headline = "Relative growth advantage"
7156
+ headline = "Relative growth advantage",
7157
+ lapisDateField
7052
7158
  }) => {
7053
- const size = { height, width };
7054
- return /* @__PURE__ */ u$1(ErrorBoundary, { size, headline, children: /* @__PURE__ */ u$1(ResizeContainer, { size, children: /* @__PURE__ */ u$1(Headline, { heading: headline, children: /* @__PURE__ */ u$1(
7159
+ const size2 = { height, width };
7160
+ return /* @__PURE__ */ u$1(ErrorBoundary, { size: size2, headline, children: /* @__PURE__ */ u$1(ResizeContainer, { size: size2, children: /* @__PURE__ */ u$1(Headline, { heading: headline, children: /* @__PURE__ */ u$1(
7055
7161
  RelativeGrowthAdvantageInner,
7056
7162
  {
7057
7163
  views,
7058
7164
  numerator,
7059
7165
  denominator,
7060
- generationTime
7166
+ generationTime,
7167
+ lapisDateField
7061
7168
  }
7062
7169
  ) }) }) });
7063
7170
  };
@@ -7065,12 +7172,13 @@ const RelativeGrowthAdvantageInner = ({
7065
7172
  numerator,
7066
7173
  denominator,
7067
7174
  generationTime,
7068
- views
7175
+ views,
7176
+ lapisDateField
7069
7177
  }) => {
7070
7178
  const lapis = P(LapisUrlContext);
7071
7179
  const [yAxisScaleType, setYAxisScaleType] = p("linear");
7072
7180
  const { data, error, isLoading } = useQuery(
7073
- () => queryRelativeGrowthAdvantage(numerator, denominator, generationTime, lapis),
7181
+ () => queryRelativeGrowthAdvantage(numerator, denominator, generationTime, lapis, lapisDateField),
7074
7182
  [lapis, numerator, denominator, generationTime, views]
7075
7183
  );
7076
7184
  if (isLoading) {
@@ -7141,7 +7249,7 @@ const RelativeGrowthAdvantageToolbar = ({
7141
7249
  ] });
7142
7250
  };
7143
7251
  const RelativeGrowthAdvantageInfo = ({ generationTime }) => {
7144
- return /* @__PURE__ */ u$1(Info, { size: { width: "600px", height: "30vh" }, children: [
7252
+ return /* @__PURE__ */ u$1(Info, { children: [
7145
7253
  /* @__PURE__ */ u$1(InfoHeadline1, { children: "Relative growth advantage" }),
7146
7254
  /* @__PURE__ */ u$1(InfoParagraph, { children: [
7147
7255
  "If variants spread pre-dominantly by local transmission across demographic groups, this estimate reflects the relative viral intrinsic growth advantage of the focal variant in the selected country and time frame. We report the relative growth advantage per ",
@@ -7182,6 +7290,7 @@ let RelativeGrowthAdvantageComponent = class extends PreactLitAdapter {
7182
7290
  this.headline = "Relative growth advantage";
7183
7291
  this.width = "100%";
7184
7292
  this.height = "700px";
7293
+ this.lapisDateField = "date";
7185
7294
  }
7186
7295
  render() {
7187
7296
  return /* @__PURE__ */ u$1(
@@ -7193,7 +7302,8 @@ let RelativeGrowthAdvantageComponent = class extends PreactLitAdapter {
7193
7302
  views: this.views,
7194
7303
  width: this.width,
7195
7304
  height: this.height,
7196
- headline: this.headline
7305
+ headline: this.headline,
7306
+ lapisDateField: this.lapisDateField
7197
7307
  }
7198
7308
  );
7199
7309
  }
@@ -7219,6 +7329,9 @@ __decorateClass$5([
7219
7329
  __decorateClass$5([
7220
7330
  n2({ type: String })
7221
7331
  ], RelativeGrowthAdvantageComponent.prototype, "height", 2);
7332
+ __decorateClass$5([
7333
+ n2({ type: String })
7334
+ ], RelativeGrowthAdvantageComponent.prototype, "lapisDateField", 2);
7222
7335
  RelativeGrowthAdvantageComponent = __decorateClass$5([
7223
7336
  t$2("gs-relative-growth-advantage")
7224
7337
  ], RelativeGrowthAdvantageComponent);
@@ -7261,8 +7374,8 @@ const Aggregate = ({
7261
7374
  filter,
7262
7375
  fields
7263
7376
  }) => {
7264
- const size = { height, width };
7265
- return /* @__PURE__ */ u$1(ErrorBoundary, { size, headline, children: /* @__PURE__ */ u$1(ResizeContainer, { size, children: /* @__PURE__ */ u$1(Headline, { heading: headline, children: /* @__PURE__ */ u$1(AggregateInner, { fields, filter, views }) }) }) });
7377
+ const size2 = { height, width };
7378
+ return /* @__PURE__ */ u$1(ErrorBoundary, { size: size2, headline, children: /* @__PURE__ */ u$1(ResizeContainer, { size: size2, children: /* @__PURE__ */ u$1(Headline, { heading: headline, children: /* @__PURE__ */ u$1(AggregateInner, { fields, filter, views }) }) }) });
7266
7379
  };
7267
7380
  const AggregateInner = ({ fields, views, filter }) => {
7268
7381
  const lapis = P(LapisUrlContext);
@@ -7296,7 +7409,7 @@ const AggregatedDataTabs = ({ data, views, fields }) => {
7296
7409
  const Toolbar = ({ data }) => {
7297
7410
  return /* @__PURE__ */ u$1("div", { class: "flex flex-row", children: [
7298
7411
  /* @__PURE__ */ u$1(CsvDownloadButton, { className: "mx-1 btn btn-xs", getData: () => data, filename: "aggregate.csv" }),
7299
- /* @__PURE__ */ u$1(Info, { children: "Info for aggregate" })
7412
+ /* @__PURE__ */ u$1(Info, { height: "100px", children: "Info for aggregate" })
7300
7413
  ] });
7301
7414
  };
7302
7415
  var __defProp$4 = Object.defineProperty;
@@ -7355,13 +7468,6 @@ __decorateClass$4([
7355
7468
  AggregateComponent = __decorateClass$4([
7356
7469
  t$2("gs-aggregate")
7357
7470
  ], AggregateComponent);
7358
- const toYYYYMMDD = (date) => {
7359
- if (!date) {
7360
- return void 0;
7361
- }
7362
- const options2 = { year: "numeric", month: "2-digit", day: "2-digit" };
7363
- return date.toLocaleDateString("en-CA", options2);
7364
- };
7365
7471
  const PRESET_VALUE_CUSTOM = "custom";
7366
7472
  const PRESET_VALUE_ALL_TIMES = "allTimes";
7367
7473
  const PRESET_VALUE_LAST_2_WEEKS = "last2Weeks";
@@ -7378,21 +7484,121 @@ const presets = {
7378
7484
  [PRESET_VALUE_LAST_3_MONTHS]: { label: "Last 3 months" },
7379
7485
  [PRESET_VALUE_LAST_6_MONTHS]: { label: "Last 6 months" }
7380
7486
  };
7487
+ const getSelectableOptions = (customSelectOptions) => {
7488
+ const presetOptions = Object.entries(presets).map(([key, value]) => {
7489
+ return { label: value.label, value: key };
7490
+ });
7491
+ const customOptions = customSelectOptions.map((customSelectOption) => {
7492
+ return { label: customSelectOption.label, value: customSelectOption.label };
7493
+ });
7494
+ return [...presetOptions, ...customOptions];
7495
+ };
7496
+ const getDatesForSelectorValue = (selectorValue, customSelectOptions, earliestDate) => {
7497
+ const today = /* @__PURE__ */ new Date();
7498
+ const customSelectOption = customSelectOptions.find((option) => option.label === selectorValue);
7499
+ if (customSelectOption) {
7500
+ return { dateFrom: new Date(customSelectOption.dateFrom), dateTo: new Date(customSelectOption.dateTo) };
7501
+ }
7502
+ switch (selectorValue) {
7503
+ case PRESET_VALUE_LAST_2_WEEKS: {
7504
+ const twoWeeksAgo = new Date(today);
7505
+ twoWeeksAgo.setDate(today.getDate() - 14);
7506
+ return { dateFrom: twoWeeksAgo, dateTo: today };
7507
+ }
7508
+ case PRESET_VALUE_LAST_MONTH: {
7509
+ const lastMonth = new Date(today);
7510
+ lastMonth.setMonth(today.getMonth() - 1);
7511
+ return { dateFrom: lastMonth, dateTo: today };
7512
+ }
7513
+ case PRESET_VALUE_LAST_2_MONTHS: {
7514
+ const twoMonthsAgo = new Date(today);
7515
+ twoMonthsAgo.setMonth(today.getMonth() - 2);
7516
+ return { dateFrom: twoMonthsAgo, dateTo: today };
7517
+ }
7518
+ case PRESET_VALUE_LAST_3_MONTHS: {
7519
+ const threeMonthsAgo = new Date(today);
7520
+ threeMonthsAgo.setMonth(today.getMonth() - 3);
7521
+ return { dateFrom: threeMonthsAgo, dateTo: today };
7522
+ }
7523
+ case PRESET_VALUE_LAST_6_MONTHS: {
7524
+ const sixMonthsAgo = new Date(today);
7525
+ sixMonthsAgo.setMonth(today.getMonth() - 6);
7526
+ return { dateFrom: sixMonthsAgo, dateTo: today };
7527
+ }
7528
+ case PRESET_VALUE_ALL_TIMES: {
7529
+ return { dateFrom: new Date(earliestDate), dateTo: today };
7530
+ }
7531
+ default:
7532
+ return { dateFrom: today, dateTo: today };
7533
+ }
7534
+ };
7535
+ function computeInitialValues(initialValue, initialDateFrom, initialDateTo, earliestDate, customSelectOptions) {
7536
+ if (isUndefinedOrEmpty(initialDateFrom) && isUndefinedOrEmpty(initialDateTo)) {
7537
+ const selectableOptions = getSelectableOptions(customSelectOptions);
7538
+ const initialSelectedDateRange = initialValue !== void 0 && selectableOptions.some((option) => option.value === initialValue) ? initialValue : PRESET_VALUE_LAST_6_MONTHS;
7539
+ const { dateFrom, dateTo } = getDatesForSelectorValue(
7540
+ initialSelectedDateRange,
7541
+ customSelectOptions,
7542
+ earliestDate
7543
+ );
7544
+ return {
7545
+ initialSelectedDateRange,
7546
+ initialSelectedDateFrom: dateFrom,
7547
+ initialSelectedDateTo: dateTo
7548
+ };
7549
+ }
7550
+ const initialSelectedDateFrom = isUndefinedOrEmpty(initialDateFrom) ? new Date(earliestDate) : new Date(initialDateFrom);
7551
+ let initialSelectedDateTo = isUndefinedOrEmpty(initialDateTo) ? /* @__PURE__ */ new Date() : new Date(initialDateTo);
7552
+ if (isNaN(initialSelectedDateFrom.getTime())) {
7553
+ throw new UserFacingError(
7554
+ "Invalid initialDateFrom",
7555
+ `Invalid initialDateFrom "${initialDateFrom}", It must be of the format YYYY-MM-DD`
7556
+ );
7557
+ }
7558
+ if (isNaN(initialSelectedDateTo.getTime())) {
7559
+ throw new UserFacingError(
7560
+ "Invalid initialDateTo",
7561
+ `Invalid initialDateTo "${initialDateTo}", It must be of the format YYYY-MM-DD`
7562
+ );
7563
+ }
7564
+ if (initialSelectedDateFrom > initialSelectedDateTo) {
7565
+ initialSelectedDateTo = initialSelectedDateFrom;
7566
+ }
7567
+ return {
7568
+ initialSelectedDateRange: PRESET_VALUE_CUSTOM,
7569
+ initialSelectedDateFrom,
7570
+ initialSelectedDateTo
7571
+ };
7572
+ }
7573
+ function isUndefinedOrEmpty(value) {
7574
+ return value === void 0 || value === "";
7575
+ }
7576
+ const toYYYYMMDD = (date) => {
7577
+ if (!date) {
7578
+ return void 0;
7579
+ }
7580
+ const options2 = { year: "numeric", month: "2-digit", day: "2-digit" };
7581
+ return date.toLocaleDateString("en-CA", options2);
7582
+ };
7381
7583
  const DateRangeSelector = ({
7382
7584
  customSelectOptions,
7383
7585
  earliestDate = "1900-01-01",
7384
7586
  initialValue,
7385
7587
  width,
7386
- dateColumn
7588
+ dateColumn,
7589
+ initialDateFrom,
7590
+ initialDateTo
7387
7591
  }) => {
7388
- const size = { width, height: "3rem" };
7389
- return /* @__PURE__ */ u$1(ErrorBoundary, { size, children: /* @__PURE__ */ u$1(ResizeContainer, { size, children: /* @__PURE__ */ u$1(
7592
+ const size2 = { width, height: "3rem" };
7593
+ return /* @__PURE__ */ u$1(ErrorBoundary, { size: size2, children: /* @__PURE__ */ u$1(ResizeContainer, { size: size2, children: /* @__PURE__ */ u$1(
7390
7594
  DateRangeSelectorInner,
7391
7595
  {
7392
7596
  customSelectOptions,
7393
7597
  earliestDate,
7394
7598
  initialValue,
7395
- dateColumn
7599
+ dateColumn,
7600
+ initialDateFrom,
7601
+ initialDateTo
7396
7602
  }
7397
7603
  ) }) });
7398
7604
  };
@@ -7400,20 +7606,28 @@ const DateRangeSelectorInner = ({
7400
7606
  customSelectOptions,
7401
7607
  earliestDate = "1900-01-01",
7402
7608
  initialValue,
7403
- dateColumn
7609
+ dateColumn,
7610
+ initialDateFrom,
7611
+ initialDateTo
7404
7612
  }) => {
7613
+ const initialValues = computeInitialValues(
7614
+ initialValue,
7615
+ initialDateFrom,
7616
+ initialDateTo,
7617
+ earliestDate,
7618
+ customSelectOptions
7619
+ );
7405
7620
  const fromDatePickerRef = F(null);
7406
7621
  const toDatePickerRef = F(null);
7407
7622
  const divRef = F(null);
7408
7623
  const [dateFromPicker, setDateFromPicker] = p(null);
7409
7624
  const [dateToPicker, setDateToPicker] = p(null);
7410
- const selectableOptions = getSelectableOptions(customSelectOptions);
7411
7625
  const [selectedDateRange, setSelectedDateRange] = p(
7412
- initialValue !== void 0 && selectableOptions.some((option) => option.value === initialValue) ? initialValue : PRESET_VALUE_LAST_6_MONTHS
7626
+ initialValues.initialSelectedDateRange
7413
7627
  );
7414
7628
  const [selectedDates, setSelectedDates] = p({
7415
- dateFrom: getDatesForSelectorValue("last6Months", customSelectOptions, earliestDate).dateFrom,
7416
- dateTo: getDatesForSelectorValue("last6Months", customSelectOptions, earliestDate).dateTo
7629
+ dateFrom: initialValues.initialSelectedDateFrom,
7630
+ dateTo: initialValues.initialSelectedDateTo
7417
7631
  });
7418
7632
  _(() => {
7419
7633
  const commonConfig = {
@@ -7492,7 +7706,7 @@ const DateRangeSelectorInner = ({
7492
7706
  /* @__PURE__ */ u$1(
7493
7707
  Select,
7494
7708
  {
7495
- items: selectableOptions,
7709
+ items: getSelectableOptions(customSelectOptions),
7496
7710
  selected: selectedDateRange,
7497
7711
  selectStyle: "select-bordered rounded-none join-item grow",
7498
7712
  onChange: (event) => {
@@ -7527,54 +7741,6 @@ const DateRangeSelectorInner = ({
7527
7741
  )
7528
7742
  ] });
7529
7743
  };
7530
- const getSelectableOptions = (customSelectOptions) => {
7531
- const presetOptions = Object.entries(presets).map(([key, value]) => {
7532
- return { label: value.label, value: key };
7533
- });
7534
- const customOptions = customSelectOptions.map((customSelectOption) => {
7535
- return { label: customSelectOption.label, value: customSelectOption.label };
7536
- });
7537
- return [...presetOptions, ...customOptions];
7538
- };
7539
- const getDatesForSelectorValue = (selectorValue, customSelectOptions, earliestDate) => {
7540
- const today = /* @__PURE__ */ new Date();
7541
- const customSelectOption = customSelectOptions.find((option) => option.label === selectorValue);
7542
- if (customSelectOption) {
7543
- return { dateFrom: new Date(customSelectOption.dateFrom), dateTo: new Date(customSelectOption.dateTo) };
7544
- }
7545
- switch (selectorValue) {
7546
- case PRESET_VALUE_LAST_2_WEEKS: {
7547
- const twoWeeksAgo = new Date(today);
7548
- twoWeeksAgo.setDate(today.getDate() - 14);
7549
- return { dateFrom: twoWeeksAgo, dateTo: today };
7550
- }
7551
- case PRESET_VALUE_LAST_MONTH: {
7552
- const lastMonth = new Date(today);
7553
- lastMonth.setMonth(today.getMonth() - 1);
7554
- return { dateFrom: lastMonth, dateTo: today };
7555
- }
7556
- case PRESET_VALUE_LAST_2_MONTHS: {
7557
- const twoMonthsAgo = new Date(today);
7558
- twoMonthsAgo.setMonth(today.getMonth() - 2);
7559
- return { dateFrom: twoMonthsAgo, dateTo: today };
7560
- }
7561
- case PRESET_VALUE_LAST_3_MONTHS: {
7562
- const threeMonthsAgo = new Date(today);
7563
- threeMonthsAgo.setMonth(today.getMonth() - 3);
7564
- return { dateFrom: threeMonthsAgo, dateTo: today };
7565
- }
7566
- case PRESET_VALUE_LAST_6_MONTHS: {
7567
- const sixMonthsAgo = new Date(today);
7568
- sixMonthsAgo.setMonth(today.getMonth() - 6);
7569
- return { dateFrom: sixMonthsAgo, dateTo: today };
7570
- }
7571
- case PRESET_VALUE_ALL_TIMES: {
7572
- return { dateFrom: new Date(earliestDate), dateTo: today };
7573
- }
7574
- default:
7575
- return { dateFrom: today, dateTo: today };
7576
- }
7577
- };
7578
7744
  var __defProp$3 = Object.defineProperty;
7579
7745
  var __getOwnPropDesc$3 = Object.getOwnPropertyDescriptor;
7580
7746
  var __decorateClass$3 = (decorators, target, key, kind) => {
@@ -7592,6 +7758,8 @@ let DateRangeSelectorComponent = class extends PreactLitAdapter {
7592
7758
  this.customSelectOptions = [];
7593
7759
  this.earliestDate = "1900-01-01";
7594
7760
  this.initialValue = "last6Months";
7761
+ this.initialDateFrom = "";
7762
+ this.initialDateTo = "";
7595
7763
  this.width = "100%";
7596
7764
  this.dateColumn = "date";
7597
7765
  }
@@ -7602,6 +7770,8 @@ let DateRangeSelectorComponent = class extends PreactLitAdapter {
7602
7770
  customSelectOptions: this.customSelectOptions,
7603
7771
  earliestDate: this.earliestDate,
7604
7772
  initialValue: this.initialValue,
7773
+ initialDateFrom: this.initialDateFrom,
7774
+ initialDateTo: this.initialDateTo,
7605
7775
  dateColumn: this.dateColumn,
7606
7776
  width: this.width
7607
7777
  }
@@ -7617,6 +7787,12 @@ __decorateClass$3([
7617
7787
  __decorateClass$3([
7618
7788
  n2()
7619
7789
  ], DateRangeSelectorComponent.prototype, "initialValue", 2);
7790
+ __decorateClass$3([
7791
+ n2()
7792
+ ], DateRangeSelectorComponent.prototype, "initialDateFrom", 2);
7793
+ __decorateClass$3([
7794
+ n2()
7795
+ ], DateRangeSelectorComponent.prototype, "initialDateTo", 2);
7620
7796
  __decorateClass$3([
7621
7797
  n2({ type: String })
7622
7798
  ], DateRangeSelectorComponent.prototype, "width", 2);
@@ -7670,8 +7846,8 @@ function compareLocationEntries(fields) {
7670
7846
  };
7671
7847
  }
7672
7848
  const LocationFilter = ({ width, initialValue, fields }) => {
7673
- const size = { width, height: "3rem" };
7674
- return /* @__PURE__ */ u$1(ErrorBoundary, { size, children: /* @__PURE__ */ u$1(ResizeContainer, { size, children: /* @__PURE__ */ u$1(LocationFilterInner, { initialValue, fields }) }) });
7849
+ const size2 = { width, height: "3rem" };
7850
+ return /* @__PURE__ */ u$1(ErrorBoundary, { size: size2, children: /* @__PURE__ */ u$1(ResizeContainer, { size: size2, children: /* @__PURE__ */ u$1(LocationFilterInner, { initialValue, fields }) }) });
7675
7851
  };
7676
7852
  const LocationFilterInner = ({ initialValue, fields }) => {
7677
7853
  const lapis = P(LapisUrlContext);
@@ -7725,7 +7901,7 @@ const LocationFilterInner = ({ initialValue, fields }) => {
7725
7901
  };
7726
7902
  const parseLocation = (location, fields) => {
7727
7903
  const fieldValues = location.split("/").map((part) => part.trim());
7728
- return fieldValues.reduce((acc, fieldValue, i2) => ({ ...acc, [fields[i2]]: fieldValue }), {});
7904
+ return fields.reduce((acc, field, i2) => ({ ...acc, [field]: fieldValues[i2] }), {});
7729
7905
  };
7730
7906
  const hasMatchingEntry = (data, eventDetail) => {
7731
7907
  if (data === null) {
@@ -7774,8 +7950,8 @@ async function fetchAutocompleteList(lapis, field, signal) {
7774
7950
  return data.map((item) => item[field]);
7775
7951
  }
7776
7952
  const TextInput = ({ width, lapisField, placeholderText, initialValue }) => {
7777
- const size = { width, height: "3rem" };
7778
- return /* @__PURE__ */ u$1(ErrorBoundary, { size, children: /* @__PURE__ */ u$1(ResizeContainer, { size, children: /* @__PURE__ */ u$1(TextInputInner, { lapisField, placeholderText, initialValue }) }) });
7953
+ const size2 = { width, height: "3rem" };
7954
+ return /* @__PURE__ */ u$1(ErrorBoundary, { size: size2, children: /* @__PURE__ */ u$1(ResizeContainer, { size: size2, children: /* @__PURE__ */ u$1(TextInputInner, { lapisField, placeholderText, initialValue }) }) });
7779
7955
  };
7780
7956
  const TextInputInner = ({
7781
7957
  lapisField,
@@ -7951,9 +8127,8 @@ const DeleteIcon = () => {
7951
8127
  }
7952
8128
  );
7953
8129
  };
7954
- const MutationFilter = ({ initialValue, width, height }) => {
7955
- const size = { height, width };
7956
- return /* @__PURE__ */ u$1(ErrorBoundary, { size, children: /* @__PURE__ */ u$1(ResizeContainer, { size, children: /* @__PURE__ */ u$1(MutationFilterInner, { initialValue }) }) });
8130
+ const MutationFilter = ({ initialValue, width }) => {
8131
+ return /* @__PURE__ */ u$1(ErrorBoundary, { size: { height: "3.375rem", width }, children: /* @__PURE__ */ u$1("div", { style: width, children: /* @__PURE__ */ u$1(MutationFilterInner, { initialValue }) }) });
7957
8132
  };
7958
8133
  const MutationFilterInner = ({ initialValue }) => {
7959
8134
  const referenceGenome = P(ReferenceGenomeContext);
@@ -7965,6 +8140,9 @@ const MutationFilterInner = ({ initialValue }) => {
7965
8140
  const formRef = F(null);
7966
8141
  const handleSubmit = (event) => {
7967
8142
  event.preventDefault();
8143
+ if (inputValue === "") {
8144
+ return;
8145
+ }
7968
8146
  const parsedMutation = parseAndValidateMutation(inputValue, referenceGenome);
7969
8147
  if (parsedMutation === null) {
7970
8148
  setIsError(true);
@@ -8004,8 +8182,9 @@ const MutationFilterInner = ({ initialValue }) => {
8004
8182
  setInputValue(event.target.value);
8005
8183
  setIsError(false);
8006
8184
  };
8007
- return /* @__PURE__ */ u$1("div", { class: `h-full w-full rounded-lg border border-gray-300 bg-white p-2 overflow-scroll`, children: [
8008
- /* @__PURE__ */ u$1("div", { class: "flex justify-between", children: [
8185
+ return /* @__PURE__ */ u$1("form", { className: "w-full border boder-gray-300 rounded-md relative", onSubmit: handleSubmit, ref: formRef, children: [
8186
+ /* @__PURE__ */ u$1("div", { className: "absolute -top-3 -right-3", children: /* @__PURE__ */ u$1(Info, { height: "100px", children: "Info for mutation filter" }) }),
8187
+ /* @__PURE__ */ u$1("div", { className: "w-full flex p-2 flex-wrap items-center", children: [
8009
8188
  /* @__PURE__ */ u$1(
8010
8189
  SelectedMutationDisplay,
8011
8190
  {
@@ -8014,22 +8193,27 @@ const MutationFilterInner = ({ initialValue }) => {
8014
8193
  fireChangeEvent
8015
8194
  }
8016
8195
  ),
8017
- /* @__PURE__ */ u$1(Info, { children: "Info for mutation filter" })
8018
- ] }),
8019
- /* @__PURE__ */ u$1("form", { className: "mt-2 w-full", onSubmit: handleSubmit, ref: formRef, children: /* @__PURE__ */ u$1("label", { className: `input flex items-center gap-2 ${isError ? "input-error" : "input-bordered"}`, children: [
8020
8196
  /* @__PURE__ */ u$1(
8021
- "input",
8197
+ "div",
8022
8198
  {
8023
- className: "grow min-w-0",
8024
- type: "text",
8025
- value: inputValue,
8026
- onInput: handleInputChange,
8027
- placeholder: getPlaceholder(referenceGenome),
8028
- onBlur: handleOnBlur
8199
+ className: `w-full flex border ${isError ? "border-red-500" : "border-gray-300"} border-solid m-2 text-sm focus-within:border-gray-400 `,
8200
+ children: [
8201
+ /* @__PURE__ */ u$1(
8202
+ "input",
8203
+ {
8204
+ className: "grow flex-1 p-1 border-none focus:outline-none focus:ring-0",
8205
+ type: "text",
8206
+ value: inputValue,
8207
+ onInput: handleInputChange,
8208
+ placeholder: getPlaceholder(referenceGenome),
8209
+ onBlur: handleOnBlur
8210
+ }
8211
+ ),
8212
+ /* @__PURE__ */ u$1("button", { type: "submit", className: "btn btn-xs m-1", children: "+" })
8213
+ ]
8029
8214
  }
8030
- ),
8031
- /* @__PURE__ */ u$1("button", { className: "btn btn-sm", children: "+" })
8032
- ] }) })
8215
+ )
8216
+ ] })
8033
8217
  ] });
8034
8218
  };
8035
8219
  function getInitialState(initialValue, referenceGenome) {
@@ -8075,35 +8259,39 @@ const SelectedMutationDisplay = ({ selectedFilters, setSelectedFilters, fireChan
8075
8259
  setSelectedFilters(newSelectedValues);
8076
8260
  fireChangeEvent(newSelectedValues);
8077
8261
  };
8078
- return /* @__PURE__ */ u$1("ul", { class: "flex flex-wrap", children: [
8079
- selectedFilters.nucleotideMutations.map((mutation) => /* @__PURE__ */ u$1("li", { children: /* @__PURE__ */ u$1(
8262
+ return /* @__PURE__ */ u$1(Fragment, { children: [
8263
+ selectedFilters.nucleotideMutations.map((mutation) => /* @__PURE__ */ u$1(
8080
8264
  SelectedNucleotideMutation,
8081
8265
  {
8082
8266
  mutation,
8083
8267
  onDelete: (mutation2) => onSelectedRemoved(mutation2, "nucleotideMutations")
8084
- }
8085
- ) }, mutation.toString())),
8086
- selectedFilters.aminoAcidMutations.map((mutation) => /* @__PURE__ */ u$1("li", { children: /* @__PURE__ */ u$1(
8268
+ },
8269
+ mutation.toString()
8270
+ )),
8271
+ selectedFilters.aminoAcidMutations.map((mutation) => /* @__PURE__ */ u$1(
8087
8272
  SelectedAminoAcidMutation,
8088
8273
  {
8089
8274
  mutation,
8090
8275
  onDelete: (mutation2) => onSelectedRemoved(mutation2, "aminoAcidMutations")
8091
- }
8092
- ) }, mutation.toString())),
8093
- selectedFilters.nucleotideInsertions.map((insertion) => /* @__PURE__ */ u$1("li", { children: /* @__PURE__ */ u$1(
8276
+ },
8277
+ mutation.toString()
8278
+ )),
8279
+ selectedFilters.nucleotideInsertions.map((insertion) => /* @__PURE__ */ u$1(
8094
8280
  SelectedNucleotideInsertion,
8095
8281
  {
8096
8282
  insertion,
8097
8283
  onDelete: (insertion2) => onSelectedRemoved(insertion2, "nucleotideInsertions")
8098
- }
8099
- ) }, insertion.toString())),
8100
- selectedFilters.aminoAcidInsertions.map((insertion) => /* @__PURE__ */ u$1("li", { children: /* @__PURE__ */ u$1(
8284
+ },
8285
+ insertion.toString()
8286
+ )),
8287
+ selectedFilters.aminoAcidInsertions.map((insertion) => /* @__PURE__ */ u$1(
8101
8288
  SelectedAminoAcidInsertion,
8102
8289
  {
8103
8290
  insertion,
8104
8291
  onDelete: (insertion2) => onSelectedRemoved(insertion2, "aminoAcidInsertions")
8105
- }
8106
- ) }, insertion.toString()))
8292
+ },
8293
+ insertion.toString()
8294
+ ))
8107
8295
  ] });
8108
8296
  };
8109
8297
  const SelectedAminoAcidInsertion = ({ insertion, onDelete }) => {
@@ -8165,12 +8353,12 @@ const SelectedFilter = ({
8165
8353
  textColor
8166
8354
  }) => {
8167
8355
  return /* @__PURE__ */ u$1(
8168
- "div",
8356
+ "span",
8169
8357
  {
8170
- class: "flex items-center flex-nowrap gap-1 rounded me-1 px-2.5 py-0.5 font-medium text-xs mb-1 min-w-max",
8358
+ class: "inline-block mx-1 px-2 py-1 font-medium text-xs rounded-full",
8171
8359
  style: { backgroundColor, color: textColor },
8172
8360
  children: [
8173
- /* @__PURE__ */ u$1("div", { className: "whitespace-nowrap min-w-max", children: mutation.toString() }),
8361
+ mutation.toString(),
8174
8362
  /* @__PURE__ */ u$1("button", { type: "button", onClick: () => onDelete(mutation), children: /* @__PURE__ */ u$1(DeleteIcon, {}) })
8175
8363
  ]
8176
8364
  }
@@ -8200,10 +8388,9 @@ let MutationFilterComponent = class extends PreactLitAdapter {
8200
8388
  super(...arguments);
8201
8389
  this.initialValue = void 0;
8202
8390
  this.width = "100%";
8203
- this.height = "6.5rem";
8204
8391
  }
8205
8392
  render() {
8206
- return /* @__PURE__ */ u$1(ReferenceGenomesAwaiter, { children: /* @__PURE__ */ u$1(MutationFilter, { initialValue: this.initialValue, width: this.width, height: this.height }) });
8393
+ return /* @__PURE__ */ u$1(ReferenceGenomesAwaiter, { children: /* @__PURE__ */ u$1(MutationFilter, { initialValue: this.initialValue, width: this.width }) });
8207
8394
  }
8208
8395
  };
8209
8396
  __decorateClass([
@@ -8212,9 +8399,6 @@ __decorateClass([
8212
8399
  __decorateClass([
8213
8400
  n2({ type: String })
8214
8401
  ], MutationFilterComponent.prototype, "width", 2);
8215
- __decorateClass([
8216
- n2({ type: String })
8217
- ], MutationFilterComponent.prototype, "height", 2);
8218
8402
  MutationFilterComponent = __decorateClass([
8219
8403
  t$2("gs-mutation-filter")
8220
8404
  ], MutationFilterComponent);