Skip to content
This repository has been archived by the owner on Jul 2, 2024. It is now read-only.

DEVPROD-3049: Remove LeafyGreen column utils #2181

Merged
merged 15 commits into from
Dec 18, 2023
Merged
Show file tree
Hide file tree
Changes from 7 commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
5 changes: 4 additions & 1 deletion cypress.config.ts
Original file line number Diff line number Diff line change
Expand Up @@ -3,7 +3,10 @@ import { execSync } from "child_process";

export default defineConfig({
e2e: {
retries: 3,
retries: {
runMode: 3,
openMode: 0,
},
Comment on lines +6 to +9
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Thanks!

baseUrl: "http://localhost:3000",
projectId: "yshv48",
reporterOptions: {
Expand Down
12 changes: 7 additions & 5 deletions cypress/integration/version/task_duration.ts
Original file line number Diff line number Diff line change
Expand Up @@ -31,7 +31,7 @@ describe("Task Duration Tab", () => {
cy.dataCy("leafygreen-table-row").should("have.length", 3);
cy.location("search").should(
"include",
"duration=DESC&page=0&statuses=running-umbrella,started,dispatched"
"duration=DESC&page=0&statuses=running-umbrella%2Cstarted%2Cdispatched"
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This indicates to me that the query params aren't being properly uriencoded or that we are double encoding them.

Copy link
Contributor Author

@sophstad sophstad Dec 13, 2023

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Well isn't the URL now being encoded, and it wasn't before? , is a reserved character in the URI specification which is why commas included our query param value should be encoded (as per these docs). query-string's .stringify() has encode = true as default, which is what ultimately applies the encoding.

);
// Clear status filter.
cy.dataCy("status-filter-popover").click();
Expand Down Expand Up @@ -61,24 +61,26 @@ describe("Task Duration Tab", () => {
});

it("updates URL appropriately when sort is changing", () => {
const durationSortControl = "button[aria-label='Sort by Task Duration']";
// The default sort (DURATION DESC) should be applied
cy.location("search").should("include", "duration=DESC");
const longestTask = "test-thirdparty";
cy.contains(longestTask).should("be.visible");
cy.dataCy("leafygreen-table-row").first().should("contain", longestTask);
cy.dataCy("duration-sort-icon").click();
cy.get(durationSortControl).click();
cy.location("search").should("not.include", "duration");
cy.dataCy("duration-sort-icon").click();
cy.get(durationSortControl).click();
cy.location("search").should("include", "duration=ASC");
const shortestTask = "test-auth";
cy.contains(shortestTask).should("be.visible");
cy.dataCy("leafygreen-table-row").first().should("contain", shortestTask);
});

it("clearing all filters resets to the default sort", () => {
cy.dataCy("duration-sort-icon").click();
const durationSortControl = "button[aria-label='Sort by Task Duration']";
cy.get(durationSortControl).click();
cy.location("search").should("not.include", "duration");
cy.dataCy("duration-sort-icon").click();
cy.get(durationSortControl).click();
cy.location("search").should("include", "duration=ASC");
cy.contains("Clear all filters").click();
cy.location("search").should("include", "duration=DESC");
Expand Down
2 changes: 1 addition & 1 deletion src/analytics/version/useVersionAnalytics.ts
Original file line number Diff line number Diff line change
Expand Up @@ -9,7 +9,7 @@ import {
import { VERSION } from "gql/queries";

type Action =
| { name: "Filter Tasks"; filterBy: string }
| { name: "Filter Tasks"; filterBy: string | string[] }
| {
name: "Sort Tasks Table";
sortBy:
Expand Down
37 changes: 20 additions & 17 deletions src/components/Table/BaseTable.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -27,14 +27,15 @@ import TableLoader from "./TableLoader";
declare module "@tanstack/table-core" {
// eslint-disable-next-line @typescript-eslint/no-unused-vars
interface ColumnMeta<TData extends RowData, TValue> {
filterComponent?: (column: any) => JSX.Element;
search?: {
"data-cy"?: string;
placeholder?: string;
};
sortComponent?: (column: any) => JSX.Element;
treeSelect?: {
"data-cy"?: string;
// Configures whether or not the tree select should be filtered to only represent values found in the table.
// Note that this may not be very performant for large tables.
filterOptions?: boolean;
options: TreeDataEntry[];
};
// Overcome react-table's column width limitations
Expand Down Expand Up @@ -67,41 +68,43 @@ export const BaseTable = <T extends LGRowData>({
{table.getHeaderGroups().map((headerGroup) => (
<HeaderRow key={headerGroup.id}>
{headerGroup.headers.map((header) => {
const { columnDef } = header.column;
const { columnDef } = header.column ?? {};
const { meta } = columnDef;
return (
<HeaderCell
key={header.id}
header={header}
style={
columnDef?.meta?.width && { width: columnDef?.meta?.width }
}
style={meta?.width && { width: columnDef?.meta?.width }}
>
{flexRender(columnDef.header, header.getContext())}
{columnDef?.meta?.sortComponent?.({
column: header.column,
})}
{columnDef?.meta?.filterComponent?.({
column: header.column,
})}
{header.column.getCanFilter() &&
(columnDef?.meta?.treeSelect ? (
(meta?.treeSelect ? (
<TableFilterPopover
data-cy={columnDef?.meta?.treeSelect?.["data-cy"]}
data-cy={meta.treeSelect?.["data-cy"]}
onConfirm={(value) =>
header.column.setFilterValue(value)
}
options={columnDef?.meta?.treeSelect?.options}
options={
meta.treeSelect?.filterOptions
? meta.treeSelect.options.filter(
({ value }) =>
!!header.column
.getFacetedUniqueValues()
.get(value)
)
: meta.treeSelect.options
}
value={
(header?.column?.getFilterValue() as string[]) ?? []
}
/>
) : (
<TableSearchPopover
data-cy={columnDef?.meta?.search?.["data-cy"]}
data-cy={meta?.search?.["data-cy"]}
onConfirm={(value) =>
header.column.setFilterValue(value)
}
placeholder={columnDef?.meta?.search?.placeholder}
placeholder={meta?.search?.placeholder}
value={
(header?.column?.getFilterValue() as string) ?? ""
}
Expand Down
119 changes: 0 additions & 119 deletions src/components/Table/LGFilters.tsx

This file was deleted.

70 changes: 31 additions & 39 deletions src/pages/hosts/HostsTable.tsx
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
import { useRef, useState } from "react";
import { useEffect, useRef, useState } from "react";
import { useLeafyGreenTable } from "@leafygreen-ui/table";
import {
ColumnFiltersState,
Expand All @@ -12,8 +12,7 @@ import { BaseTable } from "components/Table/BaseTable";
import { onChangeHandler } from "components/Table/utils";
import { hostStatuses } from "constants/hosts";
import { getHostRoute, getTaskRoute } from "constants/routes";
import { HostSortBy, HostsQuery } from "gql/generated/types";
import { useTableSort } from "hooks";
import { HostSortBy, HostsQuery, SortDirection } from "gql/generated/types";
import { useQueryParams } from "hooks/useQueryParam";
import { mapIdToFilterParam } from "types/host";
import { Unpacked } from "types/utils";
Expand All @@ -39,14 +38,8 @@ export const HostsTable: React.FC<Props> = ({
}) => {
const { sendEvent } = useHostsTableAnalytics();

const tableSortHandler = useTableSort({
sendAnalyticsEvents: () => sendEvent({ name: "Sort Hosts" }),
});

const [, setQueryParams] = useQueryParams();

const [filters, setFilters] = useState<ColumnFiltersState>(initialFilters);
const [sorting, setSorting] = useState<SortingState>(initialSorting);
const [rowSelection, setRowSelection] = useState<RowSelectionState>({});

const updateRowSelection = (rowState: RowSelectionState) => {
Expand All @@ -56,18 +49,6 @@ export const HostsTable: React.FC<Props> = ({
setSelectedHosts(selectedHosts);
};

const updateFilters = (filterState: ColumnFiltersState) => {
const updatedParams = { page: "0" };

filterState.forEach(({ id, value }) => {
const key = mapIdToFilterParam[id];
updatedParams[key] = value;
});

setQueryParams(updatedParams);
sendEvent({ name: "Filter Hosts", filterBy: Object.keys(filterState) });
};

const tableContainerRef = useRef<HTMLDivElement>(null);
const table = useLeafyGreenTable<Host>({
columns,
Expand All @@ -80,35 +61,46 @@ export const HostsTable: React.FC<Props> = ({
// https://github.com/TanStack/table/issues/4289
sortDescFirst: false,
},
initialState: {
columnFilters: initialFilters,
sorting: initialSorting,
},
state: {
columnFilters: filters,
rowSelection,
sorting,
},
hasSelectableRows: true,
manualFiltering: true,
manualSorting: true,
manualPagination: true,
onColumnFiltersChange: onChangeHandler<ColumnFiltersState>(
setFilters,
(updatedState) => {
updateFilters(updatedState);
table.resetRowSelection();
}
),
onRowSelectionChange: onChangeHandler<RowSelectionState>(
setRowSelection,
updateRowSelection
),
onSortingChange: onChangeHandler<SortingState>(
setSorting,
(updatedState) => {
tableSortHandler(updatedState);
table.resetRowSelection();
}
),
});

const { getState } = table;
const { columnFilters, sorting } = getState();

useEffect(() => {
const updatedParams = { page: "0" };

columnFilters.forEach(({ id, value }) => {
const key = mapIdToFilterParam[id];
updatedParams[key] = value;
});

sorting.forEach(({ desc, id }) => {
updatedParams[id] = desc ? SortDirection.Desc : SortDirection.Asc;
});

setQueryParams(updatedParams);

if (columnFilters.length) {
sendEvent({ name: "Filter Hosts", filterBy: Object.keys(columnFilters) });
}
if (sorting.length) {
sendEvent({ name: "Sort Hosts" });
}
}, [columnFilters, sorting]); // eslint-disable-line react-hooks/exhaustive-deps

return (
<BaseTable
data-cy="hosts-table"
Expand Down
Loading