@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/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.object({
9
- datasheet_id: z.string(),
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
- const datasheet = ctx.db.getDatasheetById(datasheet_id)
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
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@tscircuit/fake-snippets",
3
- "version": "0.0.93",
3
+ "version": "0.0.95",
4
4
  "type": "module",
5
5
  "repository": {
6
6
  "type": "git",
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