@tscircuit/fake-snippets 0.0.93 → 0.0.95
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.
- package/api/generated-index.js +32 -0
- package/bun-tests/fake-snippets-api/routes/datasheets/get.test.ts +37 -0
- package/dist/bundle.js +18 -4
- package/dist/index.d.ts +4 -2
- package/dist/index.js +6 -0
- package/fake-snippets-api/lib/db/db-client.ts +6 -0
- package/fake-snippets-api/routes/api/datasheets/get.ts +15 -5
- package/package.json +1 -1
- package/src/App.tsx +4 -0
- package/src/hooks/use-create-datasheet.ts +30 -0
- package/src/hooks/use-datasheet.ts +18 -0
- package/src/pages/datasheet.tsx +87 -0
- package/src/pages/datasheets.tsx +80 -0
package/dist/index.d.ts
CHANGED
|
@@ -1069,7 +1069,7 @@ declare const createDatabase: ({ seed }?: {
|
|
|
1069
1069
|
}[] | null;
|
|
1070
1070
|
datasheet_pdf_urls: string[] | null;
|
|
1071
1071
|
}[];
|
|
1072
|
-
}, "addOrder" | "getOrderById" | "getOrderFilesByOrderId" | "addOrderQuote" | "getOrderQuoteById" | "getJlcpcbOrderStatesByOrderId" | "getJlcpcbOrderStepRunsByJlcpcbOrderStateId" | "updateOrder" | "addJlcpcbOrderState" | "updateJlcpcbOrderState" | "addOrderFile" | "getOrderFileById" | "addAccount" | "addAccountPackage" | "getAccountPackageById" | "updateAccountPackage" | "deleteAccountPackage" | "addSnippet" | "getLatestSnippets" | "getTrendingSnippets" | "getPackagesByAuthor" | "getSnippetByAuthorAndName" | "updateSnippet" | "getSnippetById" | "searchSnippets" | "searchPackages" | "deleteSnippet" | "addSession" | "getSessions" | "createLoginPage" | "getLoginPage" | "updateLoginPage" | "getAccount" | "updateAccount" | "createSession" | "addStar" | "removeStar" | "hasStarred" | "addPackage" | "updatePackage" | "getPackageById" | "getPackageReleaseById" | "addPackageRelease" | "updatePackageRelease" | "deletePackageFile" | "addPackageFile" | "updatePackageFile" | "getStarCount" | "getPackageFilesByReleaseId" | "updatePackageReleaseFsSha" | "addAiReview" | "updateAiReview" | "getAiReviewById" | "listAiReviews" | "addDatasheet" | "getDatasheetById" | "listDatasheets" | "updateDatasheet"> & {
|
|
1072
|
+
}, "addOrder" | "getOrderById" | "getOrderFilesByOrderId" | "addOrderQuote" | "getOrderQuoteById" | "getJlcpcbOrderStatesByOrderId" | "getJlcpcbOrderStepRunsByJlcpcbOrderStateId" | "updateOrder" | "addJlcpcbOrderState" | "updateJlcpcbOrderState" | "addOrderFile" | "getOrderFileById" | "addAccount" | "addAccountPackage" | "getAccountPackageById" | "updateAccountPackage" | "deleteAccountPackage" | "addSnippet" | "getLatestSnippets" | "getTrendingSnippets" | "getPackagesByAuthor" | "getSnippetByAuthorAndName" | "updateSnippet" | "getSnippetById" | "searchSnippets" | "searchPackages" | "deleteSnippet" | "addSession" | "getSessions" | "createLoginPage" | "getLoginPage" | "updateLoginPage" | "getAccount" | "updateAccount" | "createSession" | "addStar" | "removeStar" | "hasStarred" | "addPackage" | "updatePackage" | "getPackageById" | "getPackageReleaseById" | "addPackageRelease" | "updatePackageRelease" | "deletePackageFile" | "addPackageFile" | "updatePackageFile" | "getStarCount" | "getPackageFilesByReleaseId" | "updatePackageReleaseFsSha" | "addAiReview" | "updateAiReview" | "getAiReviewById" | "listAiReviews" | "addDatasheet" | "getDatasheetById" | "getDatasheetByChipName" | "listDatasheets" | "updateDatasheet"> & {
|
|
1073
1073
|
addOrder: (order: Omit<Order, "order_id">) => Order;
|
|
1074
1074
|
getOrderById: (orderId: string) => Order | undefined;
|
|
1075
1075
|
getOrderFilesByOrderId: (orderId: string) => OrderFile[];
|
|
@@ -1155,6 +1155,7 @@ declare const createDatabase: ({ seed }?: {
|
|
|
1155
1155
|
chip_name: string;
|
|
1156
1156
|
}) => Datasheet;
|
|
1157
1157
|
getDatasheetById: (datasheetId: string) => Datasheet | undefined;
|
|
1158
|
+
getDatasheetByChipName: (chipName: string) => Datasheet | undefined;
|
|
1158
1159
|
listDatasheets: ({ chip_name, is_popular, }?: {
|
|
1159
1160
|
chip_name?: string;
|
|
1160
1161
|
is_popular?: boolean;
|
|
@@ -1420,7 +1421,7 @@ declare const createDatabase: ({ seed }?: {
|
|
|
1420
1421
|
}[] | null;
|
|
1421
1422
|
datasheet_pdf_urls: string[] | null;
|
|
1422
1423
|
}[];
|
|
1423
|
-
}, "addOrder" | "getOrderById" | "getOrderFilesByOrderId" | "addOrderQuote" | "getOrderQuoteById" | "getJlcpcbOrderStatesByOrderId" | "getJlcpcbOrderStepRunsByJlcpcbOrderStateId" | "updateOrder" | "addJlcpcbOrderState" | "updateJlcpcbOrderState" | "addOrderFile" | "getOrderFileById" | "addAccount" | "addAccountPackage" | "getAccountPackageById" | "updateAccountPackage" | "deleteAccountPackage" | "addSnippet" | "getLatestSnippets" | "getTrendingSnippets" | "getPackagesByAuthor" | "getSnippetByAuthorAndName" | "updateSnippet" | "getSnippetById" | "searchSnippets" | "searchPackages" | "deleteSnippet" | "addSession" | "getSessions" | "createLoginPage" | "getLoginPage" | "updateLoginPage" | "getAccount" | "updateAccount" | "createSession" | "addStar" | "removeStar" | "hasStarred" | "addPackage" | "updatePackage" | "getPackageById" | "getPackageReleaseById" | "addPackageRelease" | "updatePackageRelease" | "deletePackageFile" | "addPackageFile" | "updatePackageFile" | "getStarCount" | "getPackageFilesByReleaseId" | "updatePackageReleaseFsSha" | "addAiReview" | "updateAiReview" | "getAiReviewById" | "listAiReviews" | "addDatasheet" | "getDatasheetById" | "listDatasheets" | "updateDatasheet"> & {
|
|
1424
|
+
}, "addOrder" | "getOrderById" | "getOrderFilesByOrderId" | "addOrderQuote" | "getOrderQuoteById" | "getJlcpcbOrderStatesByOrderId" | "getJlcpcbOrderStepRunsByJlcpcbOrderStateId" | "updateOrder" | "addJlcpcbOrderState" | "updateJlcpcbOrderState" | "addOrderFile" | "getOrderFileById" | "addAccount" | "addAccountPackage" | "getAccountPackageById" | "updateAccountPackage" | "deleteAccountPackage" | "addSnippet" | "getLatestSnippets" | "getTrendingSnippets" | "getPackagesByAuthor" | "getSnippetByAuthorAndName" | "updateSnippet" | "getSnippetById" | "searchSnippets" | "searchPackages" | "deleteSnippet" | "addSession" | "getSessions" | "createLoginPage" | "getLoginPage" | "updateLoginPage" | "getAccount" | "updateAccount" | "createSession" | "addStar" | "removeStar" | "hasStarred" | "addPackage" | "updatePackage" | "getPackageById" | "getPackageReleaseById" | "addPackageRelease" | "updatePackageRelease" | "deletePackageFile" | "addPackageFile" | "updatePackageFile" | "getStarCount" | "getPackageFilesByReleaseId" | "updatePackageReleaseFsSha" | "addAiReview" | "updateAiReview" | "getAiReviewById" | "listAiReviews" | "addDatasheet" | "getDatasheetById" | "getDatasheetByChipName" | "listDatasheets" | "updateDatasheet"> & {
|
|
1424
1425
|
addOrder: (order: Omit<Order, "order_id">) => Order;
|
|
1425
1426
|
getOrderById: (orderId: string) => Order | undefined;
|
|
1426
1427
|
getOrderFilesByOrderId: (orderId: string) => OrderFile[];
|
|
@@ -1506,6 +1507,7 @@ declare const createDatabase: ({ seed }?: {
|
|
|
1506
1507
|
chip_name: string;
|
|
1507
1508
|
}) => Datasheet;
|
|
1508
1509
|
getDatasheetById: (datasheetId: string) => Datasheet | undefined;
|
|
1510
|
+
getDatasheetByChipName: (chipName: string) => Datasheet | undefined;
|
|
1509
1511
|
listDatasheets: ({ chip_name, is_popular, }?: {
|
|
1510
1512
|
chip_name?: string;
|
|
1511
1513
|
is_popular?: boolean;
|
package/dist/index.js
CHANGED
|
@@ -3127,6 +3127,12 @@ var initializer = combine(databaseSchema.parse({}), (set, get) => ({
|
|
|
3127
3127
|
const state = get();
|
|
3128
3128
|
return state.datasheets.find((d) => d.datasheet_id === datasheetId);
|
|
3129
3129
|
},
|
|
3130
|
+
getDatasheetByChipName: (chipName) => {
|
|
3131
|
+
const state = get();
|
|
3132
|
+
return state.datasheets.find(
|
|
3133
|
+
(d) => d.chip_name.toLowerCase() === chipName.toLowerCase()
|
|
3134
|
+
);
|
|
3135
|
+
},
|
|
3130
3136
|
listDatasheets: ({
|
|
3131
3137
|
chip_name,
|
|
3132
3138
|
is_popular
|
|
@@ -1397,6 +1397,12 @@ const initializer = combine(databaseSchema.parse({}), (set, get) => ({
|
|
|
1397
1397
|
const state = get()
|
|
1398
1398
|
return state.datasheets.find((d) => d.datasheet_id === datasheetId)
|
|
1399
1399
|
},
|
|
1400
|
+
getDatasheetByChipName: (chipName: string): Datasheet | undefined => {
|
|
1401
|
+
const state = get()
|
|
1402
|
+
return state.datasheets.find(
|
|
1403
|
+
(d) => d.chip_name.toLowerCase() === chipName.toLowerCase(),
|
|
1404
|
+
)
|
|
1405
|
+
},
|
|
1400
1406
|
listDatasheets: ({
|
|
1401
1407
|
chip_name,
|
|
1402
1408
|
is_popular,
|
|
@@ -5,16 +5,26 @@ import { z } from "zod"
|
|
|
5
5
|
export default withRouteSpec({
|
|
6
6
|
methods: ["GET", "POST"],
|
|
7
7
|
auth: "session",
|
|
8
|
-
queryParams: z
|
|
9
|
-
|
|
10
|
-
|
|
8
|
+
queryParams: z
|
|
9
|
+
.object({
|
|
10
|
+
datasheet_id: z.string().optional(),
|
|
11
|
+
chip_name: z.string().optional(),
|
|
12
|
+
})
|
|
13
|
+
.refine((val) => val.datasheet_id || val.chip_name, {
|
|
14
|
+
message: "datasheet_id or chip_name required",
|
|
15
|
+
}),
|
|
11
16
|
jsonBody: z.any().optional(),
|
|
12
17
|
jsonResponse: z.object({
|
|
13
18
|
datasheet: datasheetSchema,
|
|
14
19
|
}),
|
|
15
20
|
})(async (req, ctx) => {
|
|
16
|
-
const { datasheet_id } = req.query
|
|
17
|
-
|
|
21
|
+
const { datasheet_id, chip_name } = req.query
|
|
22
|
+
let datasheet
|
|
23
|
+
if (datasheet_id) {
|
|
24
|
+
datasheet = ctx.db.getDatasheetById(datasheet_id)
|
|
25
|
+
} else if (chip_name) {
|
|
26
|
+
datasheet = ctx.db.getDatasheetByChipName(chip_name)
|
|
27
|
+
}
|
|
18
28
|
if (!datasheet) {
|
|
19
29
|
return ctx.error(404, {
|
|
20
30
|
error_code: "datasheet_not_found",
|
package/package.json
CHANGED
package/src/App.tsx
CHANGED
|
@@ -68,6 +68,8 @@ const DevLoginPage = lazyImport(() => import("@/pages/dev-login"))
|
|
|
68
68
|
const ViewPackagePage = lazyImport(() => import("@/pages/view-package"))
|
|
69
69
|
const PackageBuildsPage = lazyImport(() => import("@/pages/package-builds"))
|
|
70
70
|
const TrendingPage = lazyImport(() => import("@/pages/trending"))
|
|
71
|
+
const DatasheetPage = lazyImport(() => import("@/pages/datasheet"))
|
|
72
|
+
const DatasheetsPage = lazyImport(() => import("@/pages/datasheets"))
|
|
71
73
|
const PackageEditorPage = lazyImport(async () => {
|
|
72
74
|
const [editorModule] = await Promise.all([
|
|
73
75
|
import("@/pages/package-editor"),
|
|
@@ -180,6 +182,8 @@ function App() {
|
|
|
180
182
|
<Route path="/settings" component={SettingsPage} />
|
|
181
183
|
<Route path="/search" component={SearchPage} />
|
|
182
184
|
<Route path="/trending" component={TrendingPage} />
|
|
185
|
+
<Route path="/datasheets" component={DatasheetsPage} />
|
|
186
|
+
<Route path="/datasheets/:chipName" component={DatasheetPage} />
|
|
183
187
|
<Route path="/authorize" component={AuthenticatePage} />
|
|
184
188
|
<Route path="/my-orders" component={MyOrdersPage} />
|
|
185
189
|
<Route path="/dev-login" component={DevLoginPage} />
|
|
@@ -0,0 +1,30 @@
|
|
|
1
|
+
import { useMutation, useQueryClient } from "react-query"
|
|
2
|
+
import { useAxios } from "@/hooks/use-axios"
|
|
3
|
+
import type { Datasheet } from "fake-snippets-api/lib/db/schema"
|
|
4
|
+
|
|
5
|
+
export const useCreateDatasheet = ({
|
|
6
|
+
onSuccess,
|
|
7
|
+
}: { onSuccess?: (datasheet: Datasheet) => void } = {}) => {
|
|
8
|
+
const axios = useAxios()
|
|
9
|
+
const queryClient = useQueryClient()
|
|
10
|
+
|
|
11
|
+
return useMutation(
|
|
12
|
+
["createDatasheet"],
|
|
13
|
+
async ({ chip_name }: { chip_name: string }) => {
|
|
14
|
+
const { data } = await axios.post("/datasheets/create", { chip_name })
|
|
15
|
+
await axios.get("/_fake/run_async_tasks")
|
|
16
|
+
return data.datasheet as Datasheet
|
|
17
|
+
},
|
|
18
|
+
{
|
|
19
|
+
onSuccess: (datasheet, variables) => {
|
|
20
|
+
if (variables?.chip_name) {
|
|
21
|
+
queryClient.invalidateQueries(["datasheet", variables.chip_name])
|
|
22
|
+
}
|
|
23
|
+
onSuccess?.(datasheet)
|
|
24
|
+
},
|
|
25
|
+
onError: (error: any) => {
|
|
26
|
+
console.error("Error creating datasheet:", error)
|
|
27
|
+
},
|
|
28
|
+
},
|
|
29
|
+
)
|
|
30
|
+
}
|
|
@@ -0,0 +1,18 @@
|
|
|
1
|
+
import { useQuery } from "react-query"
|
|
2
|
+
import { useAxios } from "@/hooks/use-axios"
|
|
3
|
+
import type { Datasheet } from "fake-snippets-api/lib/db/schema"
|
|
4
|
+
|
|
5
|
+
export const useDatasheet = (chipName: string | null) => {
|
|
6
|
+
const axios = useAxios()
|
|
7
|
+
return useQuery<Datasheet, Error & { status: number }>(
|
|
8
|
+
["datasheet", chipName],
|
|
9
|
+
async () => {
|
|
10
|
+
if (!chipName) throw new Error("chip name required")
|
|
11
|
+
const { data } = await axios.get("/datasheets/get", {
|
|
12
|
+
params: { chip_name: chipName },
|
|
13
|
+
})
|
|
14
|
+
return data.datasheet as Datasheet
|
|
15
|
+
},
|
|
16
|
+
{ enabled: Boolean(chipName), retry: false, refetchOnWindowFocus: false },
|
|
17
|
+
)
|
|
18
|
+
}
|
|
@@ -0,0 +1,87 @@
|
|
|
1
|
+
import { useParams } from "wouter"
|
|
2
|
+
import { useDatasheet } from "@/hooks/use-datasheet"
|
|
3
|
+
import { useCreateDatasheet } from "@/hooks/use-create-datasheet"
|
|
4
|
+
import Header from "@/components/Header"
|
|
5
|
+
import Footer from "@/components/Footer"
|
|
6
|
+
import type { Datasheet } from "fake-snippets-api/lib/db/schema"
|
|
7
|
+
|
|
8
|
+
export const DatasheetPage = () => {
|
|
9
|
+
const { chipName } = useParams<{ chipName: string }>()
|
|
10
|
+
const datasheetQuery = useDatasheet(chipName)
|
|
11
|
+
const createDatasheet = useCreateDatasheet()
|
|
12
|
+
|
|
13
|
+
const handleCreate = () => {
|
|
14
|
+
if (!chipName) return
|
|
15
|
+
createDatasheet.mutate({ chip_name: chipName })
|
|
16
|
+
}
|
|
17
|
+
|
|
18
|
+
return (
|
|
19
|
+
<div className="min-h-screen flex flex-col">
|
|
20
|
+
<Header />
|
|
21
|
+
<main className="container mx-auto flex-1 px-4 py-8">
|
|
22
|
+
<h1 className="text-3xl font-bold mb-6">{chipName} Datasheet</h1>
|
|
23
|
+
{datasheetQuery.isLoading ? (
|
|
24
|
+
<p>Loading...</p>
|
|
25
|
+
) : datasheetQuery.data ? (
|
|
26
|
+
<div>
|
|
27
|
+
<h2 className="text-xl font-semibold mb-2">Pin Information</h2>
|
|
28
|
+
{datasheetQuery.data.pin_information ? (
|
|
29
|
+
<table className="table-auto border-collapse mb-6">
|
|
30
|
+
<thead>
|
|
31
|
+
<tr>
|
|
32
|
+
<th className="border px-2 py-1">Pin</th>
|
|
33
|
+
<th className="border px-2 py-1">Name</th>
|
|
34
|
+
<th className="border px-2 py-1">Description</th>
|
|
35
|
+
</tr>
|
|
36
|
+
</thead>
|
|
37
|
+
<tbody>
|
|
38
|
+
{datasheetQuery.data.pin_information.map((pin) => (
|
|
39
|
+
<tr key={pin.pin_number}>
|
|
40
|
+
<td className="border px-2 py-1">{pin.pin_number}</td>
|
|
41
|
+
<td className="border px-2 py-1">{pin.name}</td>
|
|
42
|
+
<td className="border px-2 py-1">{pin.description}</td>
|
|
43
|
+
</tr>
|
|
44
|
+
))}
|
|
45
|
+
</tbody>
|
|
46
|
+
</table>
|
|
47
|
+
) : (
|
|
48
|
+
<p>No pin information available.</p>
|
|
49
|
+
)}
|
|
50
|
+
|
|
51
|
+
<h2 className="text-xl font-semibold mb-2">PDFs</h2>
|
|
52
|
+
{datasheetQuery.data.datasheet_pdf_urls ? (
|
|
53
|
+
<ul className="list-disc pl-5">
|
|
54
|
+
{datasheetQuery.data.datasheet_pdf_urls.map((url) => (
|
|
55
|
+
<li key={url}>
|
|
56
|
+
<a href={url} className="text-blue-600 underline">
|
|
57
|
+
{url}
|
|
58
|
+
</a>
|
|
59
|
+
</li>
|
|
60
|
+
))}
|
|
61
|
+
</ul>
|
|
62
|
+
) : (
|
|
63
|
+
<p>No datasheet PDFs available.</p>
|
|
64
|
+
)}
|
|
65
|
+
</div>
|
|
66
|
+
) : datasheetQuery.error &&
|
|
67
|
+
(datasheetQuery.error as any).status === 404 ? (
|
|
68
|
+
<div>
|
|
69
|
+
<p>No datasheet found.</p>
|
|
70
|
+
<button
|
|
71
|
+
className="mt-2 px-4 py-2 bg-blue-500 text-white rounded"
|
|
72
|
+
onClick={handleCreate}
|
|
73
|
+
disabled={createDatasheet.isLoading}
|
|
74
|
+
>
|
|
75
|
+
{createDatasheet.isLoading ? "Creating..." : "Create Datasheet"}
|
|
76
|
+
</button>
|
|
77
|
+
</div>
|
|
78
|
+
) : (
|
|
79
|
+
<p>Error loading datasheet.</p>
|
|
80
|
+
)}
|
|
81
|
+
</main>
|
|
82
|
+
<Footer />
|
|
83
|
+
</div>
|
|
84
|
+
)
|
|
85
|
+
}
|
|
86
|
+
|
|
87
|
+
export default DatasheetPage
|
|
@@ -0,0 +1,80 @@
|
|
|
1
|
+
import React, { useState } from "react"
|
|
2
|
+
import { useQuery } from "react-query"
|
|
3
|
+
import { useAxios } from "@/hooks/use-axios"
|
|
4
|
+
import Header from "@/components/Header"
|
|
5
|
+
import Footer from "@/components/Footer"
|
|
6
|
+
import { Input } from "@/components/ui/input"
|
|
7
|
+
import { Search } from "lucide-react"
|
|
8
|
+
import { Link } from "wouter"
|
|
9
|
+
|
|
10
|
+
interface DatasheetSummary {
|
|
11
|
+
datasheet_id: string
|
|
12
|
+
chip_name: string
|
|
13
|
+
}
|
|
14
|
+
|
|
15
|
+
export const DatasheetsPage: React.FC = () => {
|
|
16
|
+
const axios = useAxios()
|
|
17
|
+
const [searchQuery, setSearchQuery] = useState("")
|
|
18
|
+
|
|
19
|
+
const {
|
|
20
|
+
data: datasheets,
|
|
21
|
+
isLoading,
|
|
22
|
+
error,
|
|
23
|
+
} = useQuery(
|
|
24
|
+
["datasheetList", searchQuery],
|
|
25
|
+
async () => {
|
|
26
|
+
const params = new URLSearchParams()
|
|
27
|
+
if (searchQuery) {
|
|
28
|
+
params.append("chip_name", searchQuery)
|
|
29
|
+
} else {
|
|
30
|
+
params.append("is_popular", "true")
|
|
31
|
+
}
|
|
32
|
+
const { data } = await axios.get(
|
|
33
|
+
`/api/datasheets/list?${params.toString()}`,
|
|
34
|
+
)
|
|
35
|
+
return data.datasheets as DatasheetSummary[]
|
|
36
|
+
},
|
|
37
|
+
{ keepPreviousData: true },
|
|
38
|
+
)
|
|
39
|
+
|
|
40
|
+
return (
|
|
41
|
+
<div className="min-h-screen flex flex-col">
|
|
42
|
+
<Header />
|
|
43
|
+
<main className="container mx-auto flex-1 px-4 py-8">
|
|
44
|
+
<h1 className="text-3xl font-bold mb-6">Datasheets</h1>
|
|
45
|
+
<div className="max-w-md mb-6">
|
|
46
|
+
<div className="relative">
|
|
47
|
+
<Search className="absolute left-3 top-1/2 -translate-y-1/2 text-gray-400 h-4 w-4" />
|
|
48
|
+
<Input
|
|
49
|
+
type="search"
|
|
50
|
+
placeholder="Search datasheets..."
|
|
51
|
+
className="pl-10"
|
|
52
|
+
value={searchQuery}
|
|
53
|
+
onChange={(e) => setSearchQuery(e.target.value)}
|
|
54
|
+
aria-label="Search datasheets"
|
|
55
|
+
role="searchbox"
|
|
56
|
+
/>
|
|
57
|
+
</div>
|
|
58
|
+
</div>
|
|
59
|
+
{isLoading ? (
|
|
60
|
+
<p>Loading...</p>
|
|
61
|
+
) : error ? (
|
|
62
|
+
<p>Error loading datasheets.</p>
|
|
63
|
+
) : datasheets && datasheets.length > 0 ? (
|
|
64
|
+
<ul className="list-disc pl-5 space-y-2">
|
|
65
|
+
{datasheets.map((ds) => (
|
|
66
|
+
<li key={ds.datasheet_id}>
|
|
67
|
+
<Link href={`/datasheets/${ds.chip_name}`}>{ds.chip_name}</Link>
|
|
68
|
+
</li>
|
|
69
|
+
))}
|
|
70
|
+
</ul>
|
|
71
|
+
) : (
|
|
72
|
+
<p>No datasheets found.</p>
|
|
73
|
+
)}
|
|
74
|
+
</main>
|
|
75
|
+
<Footer />
|
|
76
|
+
</div>
|
|
77
|
+
)
|
|
78
|
+
}
|
|
79
|
+
|
|
80
|
+
export default DatasheetsPage
|