Site Search Update

This commit is contained in:
headlesdev 2025-05-20 17:40:29 +02:00
parent cd014f7de0
commit d1a289c4fa
6 changed files with 76 additions and 42 deletions

View File

@ -1,35 +1,22 @@
import { NextRequest, NextResponse } from "next/server";
import prisma from "@/app/prisma";
interface QueryParams {
currentPage: number;
itemPerPage: number;
search: string;
}
import Fuse from 'fuse.js';
import { Site } from "@/app/types";
export async function GET(request: NextRequest) {
const { searchParams } = request.nextUrl;
const currentPage = Number(searchParams.get("currentPage")) || 1;
const itemPerPage = Number(searchParams.get("itemPerPage")) || 1;
const itemPerPage = Number(searchParams.get("itemPerPage")) || 10;
const search = searchParams.get("search") || "";
try {
if (!search) {
const skip = (currentPage - 1) * itemPerPage;
const take = itemPerPage;
try {
const sites = await prisma.site.findMany({
skip,
take,
where: {
name: {
contains: search,
mode: "insensitive",
},
description: {
contains: search,
mode: "insensitive",
},
},
orderBy: {
name: 'asc',
},
@ -41,20 +28,56 @@ export async function GET(request: NextRequest) {
},
},
});
const total = await prisma.site.count({
where: {
name: {
contains: search,
mode: "insensitive",
const total = await prisma.site.count();
return NextResponse.json({ sites, total, currentPage, itemPerPage }, { status: 200 });
}
const allSites = await prisma.site.findMany({
include: {
networks: {
orderBy: {
name: 'asc',
},
description: {
contains: search,
mode: "insensitive",
},
},
});
return NextResponse.json({ sites, total, currentPage, itemPerPage }, { status: 200 });
const fuseOptions = {
includeScore: true,
threshold: 0.4,
keys: ['name', 'description', 'networks.name']
};
const fuse = new Fuse(allSites, fuseOptions);
const searchResults = fuse.search(search);
const filteredSites = searchResults.map(result => {
const item = result.item;
return {
...item,
id: String(item.id),
networks: item.networks?.map(network => ({
...network,
id: String(network.id),
siteId: String(network.siteId)
})) || []
} as Site;
});
const total = filteredSites.length;
const startIndex = (currentPage - 1) * itemPerPage;
const endIndex = startIndex + itemPerPage;
const paginatedSites = filteredSites.slice(startIndex, endIndex);
return NextResponse.json({
sites: paginatedSites,
total,
currentPage,
itemPerPage
}, { status: 200 });
} catch (error: any) {
console.error("Search error:", error);
return NextResponse.json({ error: "Internal Server Error" }, { status: 500 });
}
}

View File

@ -42,7 +42,7 @@ export default function SetupPage() {
}
try {
// Create user
const response = await axios.post("/api/user/create", {
await axios.post("/api/user/create", {
email,
username,
name,
@ -216,7 +216,7 @@ export default function SetupPage() {
<button className="btn btn-outline flex-1" onClick={handlePreviousStep}>
Back
</button>
<button className="btn btn-primary flex-1" onClick={handleComplete}>Next</button>
<button className="btn btn-primary flex-1" onClick={handleNextStep}>Next</button>
</div>
</div>
)}

View File

@ -38,7 +38,7 @@ export default function AddNetwork({ onNetworkAdded, siteId }: AddNetworkProps)
const successMessage = await response
if (onNetworkAdded && successMessage) {
onNetworkAdded()
setSuccess(successMessage)
setSuccess("Network added successfully")
}
} catch (apiError: any) {
setError(apiError)

View File

@ -3,14 +3,14 @@ import { Network } from "@/app/types";
const useNetworks = () => {
const addNetwork = (network: Network): Promise<string> | string => {
const addNetwork = (network: Network): Promise<Network> | string => {
if (!network.name) {
return 'Network name is required';
}
return axios.post('/api/sites/networks/add', network)
.then(() => {
return "Network added successfully";
.then((response) => {
return response.data.network;
})
.catch(err => {
throw err.response?.data?.error || 'An error occurred';

10
package-lock.json generated
View File

@ -14,6 +14,7 @@
"axios": "^1.9.0",
"bcryptjs": "^3.0.2",
"daisyui": "^5.0.35",
"fuse.js": "^7.1.0",
"js-cookie": "^3.0.5",
"jsonwebtoken": "^9.0.2",
"lucide-react": "^0.511.0",
@ -1379,6 +1380,15 @@
"url": "https://github.com/sponsors/ljharb"
}
},
"node_modules/fuse.js": {
"version": "7.1.0",
"resolved": "https://registry.npmjs.org/fuse.js/-/fuse.js-7.1.0.tgz",
"integrity": "sha512-trLf4SzuuUxfusZADLINj+dE8clK1frKdmqiJNb1Es75fmI5oY6X2mxLVUciLLjxqw/xr72Dhy+lER6dGd02FQ==",
"license": "Apache-2.0",
"engines": {
"node": ">=10"
}
},
"node_modules/get-intrinsic": {
"version": "1.3.0",
"resolved": "https://registry.npmjs.org/get-intrinsic/-/get-intrinsic-1.3.0.tgz",

View File

@ -15,6 +15,7 @@
"axios": "^1.9.0",
"bcryptjs": "^3.0.2",
"daisyui": "^5.0.35",
"fuse.js": "^7.1.0",
"js-cookie": "^3.0.5",
"jsonwebtoken": "^9.0.2",
"lucide-react": "^0.511.0",