@genspectrum/dashboard-components 0.4.2 → 0.4.4

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.
@@ -1124,7 +1124,7 @@
1124
1124
  "type": {
1125
1125
  "text": "Meta<Required<AggregateProps>>"
1126
1126
  },
1127
- "default": "{ title: 'Visualization/Aggregate', component: 'gs-aggregate', argTypes: { fields: [{ control: 'object' }], views: { options: ['table'], control: { type: 'check' }, }, width: { control: 'text' }, height: { control: 'text' }, headline: { control: 'text' }, }, parameters: withComponentDocs({ fetchMock: { mocks: [ { matcher: { name: 'aggregatedData', url: AGGREGATED_ENDPOINT, body: { fields: ['division', 'host'], country: 'USA', }, }, response: { status: 200, body: aggregatedData, }, }, ], }, componentDocs: { opensShadowDom: true, expectsChildren: false, codeExample, }, }), tags: ['autodocs'], }"
1127
+ "default": "{ title: 'Visualization/Aggregate', component: 'gs-aggregate', argTypes: { fields: [{ control: 'object' }], views: { options: ['table'], control: { type: 'check' }, }, width: { control: 'text' }, height: { control: 'text' }, headline: { control: 'text' }, initialSortField: { control: 'text' }, initialSortDirection: { options: ['ascending', 'descending'], control: { type: 'radio' }, }, }, parameters: withComponentDocs({ fetchMock: { mocks: [ { matcher: { name: 'aggregatedData', url: AGGREGATED_ENDPOINT, body: { fields: ['division', 'host'], country: 'USA', }, }, response: { status: 200, body: aggregatedData, }, }, ], }, componentDocs: { opensShadowDom: true, expectsChildren: false, codeExample, }, }), tags: ['autodocs'], }"
1128
1128
  },
1129
1129
  {
1130
1130
  "kind": "variable",
@@ -1132,7 +1132,7 @@
1132
1132
  "type": {
1133
1133
  "text": "StoryObj<Required<AggregateProps>>"
1134
1134
  },
1135
- "default": "{ render: (args) => html` <gs-app lapis=\"${LAPIS_URL}\"> <gs-aggregate .fields=${args.fields} .filter=${args.filter} .views=${args.views} .width=${args.width} .height=${args.height} .headline=${args.headline} ></gs-aggregate> </gs-app> `, args: { fields: ['division', 'host'], views: ['table'], filter: { country: 'USA', }, width: '100%', height: '700px', headline: 'Aggregate', }, }"
1135
+ "default": "{ render: (args) => html` <gs-app lapis=\"${LAPIS_URL}\"> <gs-aggregate .fields=${args.fields} .filter=${args.filter} .views=${args.views} .width=${args.width} .height=${args.height} .headline=${args.headline} .initialSortField=${args.initialSortField} .initialSortDirection=${args.initialSortDirection} ></gs-aggregate> </gs-app> `, args: { fields: ['division', 'host'], views: ['table'], filter: { country: 'USA', }, width: '100%', height: '700px', headline: 'Aggregate', initialSortField: 'count', initialSortDirection: 'descending', }, }"
1136
1136
  }
1137
1137
  ],
1138
1138
  "exports": [
@@ -1222,6 +1222,26 @@
1222
1222
  "default": "'Aggregate'",
1223
1223
  "description": "The headline of the component. Set to an empty string to hide the headline.",
1224
1224
  "attribute": "headline"
1225
+ },
1226
+ {
1227
+ "kind": "field",
1228
+ "name": "initialSortField",
1229
+ "type": {
1230
+ "text": "string"
1231
+ },
1232
+ "default": "'count'",
1233
+ "description": "The field by which the table is initially sorted.\nMust be one of the fields specified in the fields property, 'count', or 'proportion'.",
1234
+ "attribute": "initialSortField"
1235
+ },
1236
+ {
1237
+ "kind": "field",
1238
+ "name": "initialSortDirection",
1239
+ "type": {
1240
+ "text": "'ascending' | 'descending'"
1241
+ },
1242
+ "default": "'descending'",
1243
+ "description": "The initial sort direction of the table.",
1244
+ "attribute": "initialSortDirection"
1225
1245
  }
1226
1246
  ],
1227
1247
  "attributes": [
@@ -1278,6 +1298,24 @@
1278
1298
  "default": "'Aggregate'",
1279
1299
  "description": "The headline of the component. Set to an empty string to hide the headline.",
1280
1300
  "fieldName": "headline"
1301
+ },
1302
+ {
1303
+ "name": "initialSortField",
1304
+ "type": {
1305
+ "text": "string"
1306
+ },
1307
+ "default": "'count'",
1308
+ "description": "The field by which the table is initially sorted.\nMust be one of the fields specified in the fields property, 'count', or 'proportion'.",
1309
+ "fieldName": "initialSortField"
1310
+ },
1311
+ {
1312
+ "name": "initialSortDirection",
1313
+ "type": {
1314
+ "text": "'ascending' | 'descending'"
1315
+ },
1316
+ "default": "'descending'",
1317
+ "description": "The initial sort direction of the table.",
1318
+ "fieldName": "initialSortDirection"
1281
1319
  }
1282
1320
  ],
1283
1321
  "superclass": {
@@ -1836,7 +1836,7 @@ const Toolbar$3 = ({
1836
1836
  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 */';
1837
1837
  const minMaxPercentSliderCss = 'input[type=range]::-webkit-slider-thumb {\n -webkit-appearance: none;\n pointer-events: all;\n width: 24px;\n height: 24px;\n background-color: #fff;\n border-radius: 50%;\n box-shadow: 0 0 0 1px #C6C6C6;\n cursor: pointer;\n}\n\ninput[type=range]::-moz-range-thumb {\n -webkit-appearance: none;\n pointer-events: all;\n width: 24px;\n height: 24px;\n background-color: #fff;\n border-radius: 50%;\n box-shadow: 0 0 0 1px #C6C6C6;\n cursor: pointer;\n}\n\ninput[type=range]::-webkit-slider-thumb:hover {\n background: #f7f7f7;\n}\n\ninput[type=range]::-webkit-slider-thumb:active {\n box-shadow: inset 0 0 3px #387bbe, 0 0 9px #387bbe;\n -webkit-box-shadow: inset 0 0 3px #387bbe, 0 0 9px #387bbe;\n}\n\ninput[type="range"] {\n -webkit-appearance: none;\n -moz-appearance: none;\n appearance: none;\n height: 2px;\n width: 100%;\n position: absolute;\n background-color: #C6C6C6;\n pointer-events: none;\n}';
1838
1838
  const tailwindStyle = `/*
1839
- ! tailwindcss v3.4.3 | MIT License | https://tailwindcss.com
1839
+ ! tailwindcss v3.4.4 | MIT License | https://tailwindcss.com
1840
1840
  *//*
1841
1841
  1. Prevent padding and border from affecting element width. (https://github.com/mozdevs/cssremedy/issues/4)
1842
1842
  2. Allow adding a border to an element by just adding a border-width. (https://github.com/tailwindcss/tailwindcss/pull/116)
@@ -2295,6 +2295,10 @@ html {
2295
2295
  -webkit-tap-highlight-color: transparent;
2296
2296
  }
2297
2297
 
2298
+ * {
2299
+ scrollbar-color: currentColor transparent;
2300
+ }
2301
+
2298
2302
  :root {
2299
2303
  color-scheme: light;
2300
2304
  --in: 72.06% 0.191 231.6;
@@ -7281,9 +7285,24 @@ const AggregateTable = ({ data, fields }) => {
7281
7285
  ];
7282
7286
  return /* @__PURE__ */ u$1(Table, { data, columns: headers, pagination: true });
7283
7287
  };
7284
- async function queryAggregateData(variant, fields, lapis, signal) {
7288
+ const compareAscending = (a2, b3) => {
7289
+ if (typeof a2 === "number" && typeof b3 === "number") {
7290
+ return a2 - b3;
7291
+ }
7292
+ const strA = a2 != null ? String(a2) : "";
7293
+ const strB = b3 != null ? String(b3) : "";
7294
+ return strA.localeCompare(strB);
7295
+ };
7296
+ async function queryAggregateData(variant, fields, lapis, initialSort = { field: "count", direction: "descending" }, signal) {
7297
+ const validSortFields = ["count", "proportion", ...fields];
7298
+ if (!validSortFields.includes(initialSort.field)) {
7299
+ throw new Error(`InitialSort field not in fields. Valid fields are: ${validSortFields.join(", ")}`);
7300
+ }
7285
7301
  const fetchData = new FetchAggregatedOperator(variant, fields);
7286
- const data = (await fetchData.evaluate(lapis, signal)).content;
7302
+ const sortData = new SortOperator(fetchData, (a2, b3) => {
7303
+ return initialSort.direction === "ascending" ? compareAscending(a2[initialSort.field], b3[initialSort.field]) : compareAscending(b3[initialSort.field], a2[initialSort.field]);
7304
+ });
7305
+ const data = (await sortData.evaluate(lapis, signal)).content;
7287
7306
  const total = data.reduce((acc, row) => acc + row.count, 0);
7288
7307
  return data.map(
7289
7308
  (row) => ({
@@ -7298,15 +7317,32 @@ const Aggregate = ({
7298
7317
  height,
7299
7318
  headline = "Mutations",
7300
7319
  filter,
7301
- fields
7320
+ fields,
7321
+ initialSortField,
7322
+ initialSortDirection
7302
7323
  }) => {
7303
7324
  const size2 = { height, width };
7304
- 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 }) }) }) });
7325
+ 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(
7326
+ AggregateInner,
7327
+ {
7328
+ fields,
7329
+ filter,
7330
+ views,
7331
+ initialSortField,
7332
+ initialSortDirection
7333
+ }
7334
+ ) }) }) });
7305
7335
  };
7306
- const AggregateInner = ({ fields, views, filter }) => {
7336
+ const AggregateInner = ({
7337
+ fields,
7338
+ views,
7339
+ filter,
7340
+ initialSortField,
7341
+ initialSortDirection
7342
+ }) => {
7307
7343
  const lapis = P(LapisUrlContext);
7308
7344
  const { data, error, isLoading } = useQuery(async () => {
7309
- return queryAggregateData(filter, fields, lapis);
7345
+ return queryAggregateData(filter, fields, lapis, { field: initialSortField, direction: initialSortDirection });
7310
7346
  }, [filter, fields, lapis]);
7311
7347
  if (isLoading) {
7312
7348
  return /* @__PURE__ */ u$1(LoadingDisplay, {});
@@ -7358,6 +7394,8 @@ let AggregateComponent = class extends PreactLitAdapterWithGridJsStyles {
7358
7394
  this.width = "100%";
7359
7395
  this.height = "700px";
7360
7396
  this.headline = "Aggregate";
7397
+ this.initialSortField = "count";
7398
+ this.initialSortDirection = "descending";
7361
7399
  }
7362
7400
  render() {
7363
7401
  return /* @__PURE__ */ u$1(
@@ -7368,7 +7406,9 @@ let AggregateComponent = class extends PreactLitAdapterWithGridJsStyles {
7368
7406
  filter: this.filter,
7369
7407
  width: this.width,
7370
7408
  height: this.height,
7371
- headline: this.headline
7409
+ headline: this.headline,
7410
+ initialSortField: this.initialSortField,
7411
+ initialSortDirection: this.initialSortDirection
7372
7412
  }
7373
7413
  );
7374
7414
  }
@@ -7391,6 +7431,12 @@ __decorateClass$4([
7391
7431
  __decorateClass$4([
7392
7432
  n2({ type: String })
7393
7433
  ], AggregateComponent.prototype, "headline", 2);
7434
+ __decorateClass$4([
7435
+ n2({ type: String })
7436
+ ], AggregateComponent.prototype, "initialSortField", 2);
7437
+ __decorateClass$4([
7438
+ n2({ type: String })
7439
+ ], AggregateComponent.prototype, "initialSortDirection", 2);
7394
7440
  AggregateComponent = __decorateClass$4([
7395
7441
  t$2("gs-aggregate")
7396
7442
  ], AggregateComponent);
@@ -7516,7 +7562,7 @@ const DateRangeSelector = ({
7516
7562
  initialDateTo
7517
7563
  }) => {
7518
7564
  const size2 = { width, height: "3rem" };
7519
- return /* @__PURE__ */ u$1(ErrorBoundary, { size: size2, children: /* @__PURE__ */ u$1(ResizeContainer, { size: size2, children: /* @__PURE__ */ u$1(
7565
+ return /* @__PURE__ */ u$1(ErrorBoundary, { size: size2, children: /* @__PURE__ */ u$1("div", { style: { width }, children: /* @__PURE__ */ u$1(
7520
7566
  DateRangeSelectorInner,
7521
7567
  {
7522
7568
  customSelectOptions,