🚧 search on table

This commit is contained in:
Fred KISSIE
2026-01-29 05:48:41 +01:00
parent d374ea6ea6
commit b04385a340
5 changed files with 64 additions and 30 deletions

14
package-lock.json generated
View File

@@ -104,6 +104,7 @@
"tailwind-merge": "3.4.0", "tailwind-merge": "3.4.0",
"topojson-client": "3.1.0", "topojson-client": "3.1.0",
"tw-animate-css": "1.4.0", "tw-animate-css": "1.4.0",
"use-debounce": "^10.1.0",
"uuid": "13.0.0", "uuid": "13.0.0",
"vaul": "1.1.2", "vaul": "1.1.2",
"visionscarto-world-atlas": "1.0.0", "visionscarto-world-atlas": "1.0.0",
@@ -13944,7 +13945,6 @@
"resolved": "https://registry.npmjs.org/eslint-plugin-import/-/eslint-plugin-import-2.32.0.tgz", "resolved": "https://registry.npmjs.org/eslint-plugin-import/-/eslint-plugin-import-2.32.0.tgz",
"integrity": "sha512-whOE1HFo/qJDyX4SnXzP4N6zOWn79WhnCUY/iDR0mPfQZO8wcYE4JClzI2oZrhBnnMUCBCHZhO6VQyoBU95mZA==", "integrity": "sha512-whOE1HFo/qJDyX4SnXzP4N6zOWn79WhnCUY/iDR0mPfQZO8wcYE4JClzI2oZrhBnnMUCBCHZhO6VQyoBU95mZA==",
"license": "MIT", "license": "MIT",
"peer": true,
"dependencies": { "dependencies": {
"@rtsao/scc": "^1.1.0", "@rtsao/scc": "^1.1.0",
"array-includes": "^3.1.9", "array-includes": "^3.1.9",
@@ -23240,6 +23240,18 @@
} }
} }
}, },
"node_modules/use-debounce": {
"version": "10.1.0",
"resolved": "https://registry.npmjs.org/use-debounce/-/use-debounce-10.1.0.tgz",
"integrity": "sha512-lu87Za35V3n/MyMoEpD5zJv0k7hCn0p+V/fK2kWD+3k2u3kOCwO593UArbczg1fhfs2rqPEnHpULJ3KmGdDzvg==",
"license": "MIT",
"engines": {
"node": ">= 16.0.0"
},
"peerDependencies": {
"react": "*"
}
},
"node_modules/use-intl": { "node_modules/use-intl": {
"version": "4.7.0", "version": "4.7.0",
"resolved": "https://registry.npmjs.org/use-intl/-/use-intl-4.7.0.tgz", "resolved": "https://registry.npmjs.org/use-intl/-/use-intl-4.7.0.tgz",

View File

@@ -128,6 +128,7 @@
"tailwind-merge": "3.4.0", "tailwind-merge": "3.4.0",
"topojson-client": "3.1.0", "topojson-client": "3.1.0",
"tw-animate-css": "1.4.0", "tw-animate-css": "1.4.0",
"use-debounce": "^10.1.0",
"uuid": "13.0.0", "uuid": "13.0.0",
"vaul": "1.1.2", "vaul": "1.1.2",
"visionscarto-world-atlas": "1.0.0", "visionscarto-world-atlas": "1.0.0",
@@ -152,6 +153,7 @@
"@types/express": "5.0.6", "@types/express": "5.0.6",
"@types/express-session": "1.18.2", "@types/express-session": "1.18.2",
"@types/jmespath": "0.15.2", "@types/jmespath": "0.15.2",
"@types/js-yaml": "4.0.9",
"@types/jsonwebtoken": "9.0.10", "@types/jsonwebtoken": "9.0.10",
"@types/node": "24.10.2", "@types/node": "24.10.2",
"@types/nodemailer": "7.0.4", "@types/nodemailer": "7.0.4",
@@ -164,7 +166,6 @@
"@types/topojson-client": "3.1.5", "@types/topojson-client": "3.1.5",
"@types/ws": "8.18.1", "@types/ws": "8.18.1",
"@types/yargs": "17.0.35", "@types/yargs": "17.0.35",
"@types/js-yaml": "4.0.9",
"babel-plugin-react-compiler": "1.0.0", "babel-plugin-react-compiler": "1.0.0",
"drizzle-kit": "0.31.8", "drizzle-kit": "0.31.8",
"esbuild": "0.27.2", "esbuild": "0.27.2",

View File

@@ -4,7 +4,7 @@ import { remoteExitNodes } from "@server/db";
import logger from "@server/logger"; import logger from "@server/logger";
import HttpCode from "@server/types/HttpCode"; import HttpCode from "@server/types/HttpCode";
import response from "@server/lib/response"; import response from "@server/lib/response";
import { and, count, eq, inArray, or, sql } from "drizzle-orm"; import { and, count, eq, ilike, inArray, or, sql } from "drizzle-orm";
import { NextFunction, Request, Response } from "express"; import { NextFunction, Request, Response } from "express";
import createHttpError from "http-errors"; import createHttpError from "http-errors";
import { z } from "zod"; import { z } from "zod";
@@ -87,10 +87,29 @@ const listSitesSchema = z.object({
.min(0) .min(0)
.optional() .optional()
.catch(1) .catch(1)
.default(1) .default(1),
query: z.string().optional()
}); });
function querySites(orgId: string, accessibleSiteIds: number[]) { function querySites(
orgId: string,
accessibleSiteIds: number[],
query: string = ""
) {
let conditions = and(
inArray(sites.siteId, accessibleSiteIds),
eq(sites.orgId, orgId)
);
if (query) {
conditions = and(
conditions,
or(
ilike(sites.name, "%" + query + "%"),
ilike(sites.niceId, "%" + query + "%")
)
);
}
return db return db
.select({ .select({
siteId: sites.siteId, siteId: sites.siteId,
@@ -118,12 +137,7 @@ function querySites(orgId: string, accessibleSiteIds: number[]) {
remoteExitNodes, remoteExitNodes,
eq(remoteExitNodes.exitNodeId, sites.exitNodeId) eq(remoteExitNodes.exitNodeId, sites.exitNodeId)
) )
.where( .where(conditions);
and(
inArray(sites.siteId, accessibleSiteIds),
eq(sites.orgId, orgId)
)
);
} }
type SiteWithUpdateAvailable = Awaited<ReturnType<typeof querySites>>[0] & { type SiteWithUpdateAvailable = Awaited<ReturnType<typeof querySites>>[0] & {
@@ -162,7 +176,7 @@ export async function listSites(
) )
); );
} }
const { pageSize, page } = parsedQuery.data; const { pageSize, page, query } = parsedQuery.data;
const parsedParams = listSitesParamsSchema.safeParse(req.params); const parsedParams = listSitesParamsSchema.safeParse(req.params);
if (!parsedParams.success) { if (!parsedParams.success) {
@@ -206,7 +220,7 @@ export async function listSites(
} }
const accessibleSiteIds = accessibleSites.map((site) => site.siteId); const accessibleSiteIds = accessibleSites.map((site) => site.siteId);
const baseQuery = querySites(orgId, accessibleSiteIds); const baseQuery = querySites(orgId, accessibleSiteIds, query);
const countQuery = db const countQuery = db
.select({ count: count() }) .select({ count: count() })

View File

@@ -21,7 +21,7 @@ import { toast } from "@app/hooks/useToast";
import { createApiClient, formatAxiosError } from "@app/lib/api"; import { createApiClient, formatAxiosError } from "@app/lib/api";
import { parseDataSize } from "@app/lib/dataSize"; import { parseDataSize } from "@app/lib/dataSize";
import { build } from "@server/build"; import { build } from "@server/build";
import { Column } from "@tanstack/react-table"; import { Column, type PaginationState } from "@tanstack/react-table";
import { import {
ArrowRight, ArrowRight,
ArrowUpDown, ArrowUpDown,
@@ -31,7 +31,8 @@ import {
import { useTranslations } from "next-intl"; import { useTranslations } from "next-intl";
import Link from "next/link"; import Link from "next/link";
import { usePathname, useRouter, useSearchParams } from "next/navigation"; import { usePathname, useRouter, useSearchParams } from "next/navigation";
import { useEffect, useState, useTransition } from "react"; import { useState, useTransition } from "react";
import { useDebouncedCallback } from "use-debounce";
export type SiteRow = { export type SiteRow = {
id: number; id: number;
@@ -419,10 +420,20 @@ export default function SitesTable({
} }
]; ];
console.log({ const handlePaginationChange = (newPage: PaginationState) => {
sites, const sp = new URLSearchParams(searchParams);
pagination sp.set("page", (newPage.pageIndex + 1).toString());
}); sp.set("pageSize", newPage.pageSize.toString());
startTransition(() => router.push(`${pathname}?${sp.toString()}`));
};
// const = useDebouncedCallback()
const handleSearchChange = useDebouncedCallback((query: string) => {
const sp = new URLSearchParams(searchParams);
sp.set("query", query);
startTransition(() => router.push(`${pathname}?${sp.toString()}`));
}, 300);
return ( return (
<> <>
@@ -456,15 +467,10 @@ export default function SitesTable({
searchPlaceholder={t("searchSitesProgress")} searchPlaceholder={t("searchSitesProgress")}
manualFiltering manualFiltering
pagination={pagination} pagination={pagination}
onPaginationChange={(newPage) => { onPaginationChange={handlePaginationChange}
const sp = new URLSearchParams(searchParams);
sp.set("page", (newPage.pageIndex + 1).toString());
sp.set("pageSize", newPage.pageSize.toString());
startTransition(() =>
router.push(`${pathname}?${sp.toString()}`)
);
}}
onAdd={() => router.push(`/${orgId}/settings/sites/create`)} onAdd={() => router.push(`/${orgId}/settings/sites/create`)}
searchQuery={searchParams.get("query")?.toString()}
onSearch={handleSearchChange}
addButtonText={t("siteAdd")} addButtonText={t("siteAdd")}
onRefresh={() => startTransition(refreshData)} onRefresh={() => startTransition(refreshData)}
isRefreshing={isRefreshing} isRefreshing={isRefreshing}

View File

@@ -189,7 +189,7 @@ type DataTableProps<TData, TValue> = {
enableColumnVisibility?: boolean; enableColumnVisibility?: boolean;
manualFiltering?: boolean; manualFiltering?: boolean;
onSearch?: (input: string) => void; onSearch?: (input: string) => void;
searchValue?: string; searchQuery?: string;
pagination?: DataTablePaginationState; pagination?: DataTablePaginationState;
onPaginationChange?: DataTablePaginationUpdateFn; onPaginationChange?: DataTablePaginationUpdateFn;
persistColumnVisibility?: boolean | string; persistColumnVisibility?: boolean | string;
@@ -221,7 +221,7 @@ export function DataTable<TData, TValue>({
pagination: paginationState, pagination: paginationState,
stickyLeftColumn, stickyLeftColumn,
onSearch, onSearch,
searchValue, searchQuery,
onPaginationChange, onPaginationChange,
stickyRightColumn stickyRightColumn
}: DataTableProps<TData, TValue>) { }: DataTableProps<TData, TValue>) {
@@ -508,7 +508,8 @@ export function DataTable<TData, TValue>({
<div className="relative w-full sm:max-w-sm"> <div className="relative w-full sm:max-w-sm">
<Input <Input
placeholder={searchPlaceholder} placeholder={searchPlaceholder}
value={searchValue ?? globalFilter ?? ""} defaultValue={searchQuery}
value={onSearch ? undefined : globalFilter}
onChange={(e) => { onChange={(e) => {
onSearch onSearch
? onSearch(e.currentTarget.value) ? onSearch(e.currentTarget.value)