Project created from basic template
This commit is contained in:
14
src/client/app.tsx
Normal file
14
src/client/app.tsx
Normal file
@@ -0,0 +1,14 @@
|
||||
function App() {
|
||||
return (
|
||||
<div className="min-h-screen w-full max-w-4xl mx-auto p-4">
|
||||
{/* Replace this placeholder content with your app components */}
|
||||
<div className="text-center mt-72">
|
||||
<h1 className="text-2xl mb-4 opacity-50">
|
||||
Building your new project...
|
||||
</h1>
|
||||
</div>
|
||||
</div>
|
||||
);
|
||||
}
|
||||
|
||||
export default App;
|
117
src/client/components/demo/ai.tsx
Normal file
117
src/client/components/demo/ai.tsx
Normal file
@@ -0,0 +1,117 @@
|
||||
import { useState } from "react";
|
||||
import { useMutation } from "@tanstack/react-query";
|
||||
import { queryClient } from "@/client/rpc-client";
|
||||
|
||||
export function AIDemo() {
|
||||
const [message, setMessage] = useState("");
|
||||
const [systemPrompt, setSystemPrompt] = useState("");
|
||||
const [personPrompt, setPersonPrompt] = useState("");
|
||||
|
||||
const {
|
||||
data: completionData,
|
||||
mutate: complete,
|
||||
isPending: isCompleting,
|
||||
} = useMutation(queryClient.demo.ai.complete.mutationOptions());
|
||||
|
||||
const {
|
||||
data: personData,
|
||||
mutate: generatePerson,
|
||||
isPending: isGenerating,
|
||||
} = useMutation(queryClient.demo.ai.generate.mutationOptions());
|
||||
|
||||
// When using this demo, remove any UI below that is not relevant for the user
|
||||
return (
|
||||
<div className="max-w-4xl mx-auto p-6 space-y-8">
|
||||
<h2 className="text-2xl font-bold">AI Demo</h2>
|
||||
|
||||
<div className="space-y-6">
|
||||
<h3 className="text-xl font-semibold">Chat Completion</h3>
|
||||
|
||||
<div className="space-y-4">
|
||||
<input
|
||||
className="w-full p-3 border border-gray-300 rounded-md"
|
||||
placeholder="System prompt (optional)"
|
||||
value={systemPrompt}
|
||||
onChange={(e) => setSystemPrompt(e.target.value)}
|
||||
/>
|
||||
<input
|
||||
className="w-full p-3 border border-gray-300 rounded-md"
|
||||
placeholder="Your message"
|
||||
value={message}
|
||||
onChange={(e) => setMessage(e.target.value)}
|
||||
/>
|
||||
<button
|
||||
className="px-6 py-3 border border-gray-300 rounded-md disabled:opacity-50 disabled:cursor-not-allowed hover:bg-gray-50"
|
||||
onClick={() => {
|
||||
if (message) {
|
||||
complete({
|
||||
message,
|
||||
systemPrompt: systemPrompt || undefined,
|
||||
});
|
||||
}
|
||||
}}
|
||||
disabled={isCompleting || !message}
|
||||
>
|
||||
{isCompleting ? "Generating..." : "Send Message"}
|
||||
</button>
|
||||
</div>
|
||||
|
||||
{completionData && (
|
||||
<div className="p-4 border border-gray-200 rounded-md">
|
||||
<h4 className="font-medium mb-2">Response:</h4>
|
||||
<p className="leading-relaxed">{completionData.response}</p>
|
||||
</div>
|
||||
)}
|
||||
</div>
|
||||
|
||||
<div className="space-y-6">
|
||||
<h3 className="text-xl font-semibold">Generate Person</h3>
|
||||
|
||||
<div className="space-y-4">
|
||||
<input
|
||||
className="w-full p-3 border border-gray-300 rounded-md"
|
||||
placeholder="Describe the person you want to generate"
|
||||
value={personPrompt}
|
||||
onChange={(e) => setPersonPrompt(e.target.value)}
|
||||
/>
|
||||
<button
|
||||
className="px-6 py-3 border border-gray-300 rounded-md disabled:opacity-50 disabled:cursor-not-allowed hover:bg-gray-50"
|
||||
onClick={() => {
|
||||
if (personPrompt) {
|
||||
generatePerson({ prompt: personPrompt });
|
||||
}
|
||||
}}
|
||||
disabled={isGenerating || !personPrompt}
|
||||
>
|
||||
{isGenerating ? "Generating..." : "Generate Person"}
|
||||
</button>
|
||||
</div>
|
||||
|
||||
{personData && (
|
||||
<div className="p-4 border border-gray-200 rounded-md">
|
||||
<h4 className="font-medium mb-3">Generated Person:</h4>
|
||||
<div className="space-y-2">
|
||||
<p>
|
||||
<strong>Name:</strong> {personData.person.name}
|
||||
</p>
|
||||
<p>
|
||||
<strong>Age:</strong> {personData.person.age}
|
||||
</p>
|
||||
<p>
|
||||
<strong>Occupation:</strong> {personData.person.occupation}
|
||||
</p>
|
||||
<p>
|
||||
<strong>Bio:</strong> {personData.person.bio}
|
||||
</p>
|
||||
{personData.person.nickname && (
|
||||
<p>
|
||||
<strong>Nickname:</strong> {personData.person.nickname}
|
||||
</p>
|
||||
)}
|
||||
</div>
|
||||
</div>
|
||||
)}
|
||||
</div>
|
||||
</div>
|
||||
);
|
||||
}
|
63
src/client/components/demo/rpc.tsx
Normal file
63
src/client/components/demo/rpc.tsx
Normal file
@@ -0,0 +1,63 @@
|
||||
import { useState } from "react";
|
||||
import { useMutation, useQuery } from "@tanstack/react-query";
|
||||
import { queryClient, rpcClient } from "@/client/rpc-client";
|
||||
|
||||
export function RPCDemo() {
|
||||
const [value, setValue] = useState("");
|
||||
|
||||
// Live data query - automatically updates when server data changes
|
||||
// Use .experimental_liveOptions() for real-time subscriptions
|
||||
// Use .queryOptions() for static data that doesn't need live updates
|
||||
const { data: items } = useQuery(
|
||||
queryClient.demo.storage.live.list.experimental_liveOptions()
|
||||
);
|
||||
|
||||
// Mutations handle data changes with loading states
|
||||
// Automatically invalidates related queries on success
|
||||
const { mutate: createItem, isPending: isCreatingItem } = useMutation(
|
||||
queryClient.demo.storage.create.mutationOptions()
|
||||
);
|
||||
|
||||
return (
|
||||
<div>
|
||||
<h2>Storage Demo</h2>
|
||||
|
||||
<div>
|
||||
<input
|
||||
placeholder="Value"
|
||||
value={value}
|
||||
onChange={(e) => setValue(e.target.value)}
|
||||
/>
|
||||
<button
|
||||
onClick={() => {
|
||||
if (value) {
|
||||
createItem({ value });
|
||||
setValue("");
|
||||
}
|
||||
}}
|
||||
disabled={isCreatingItem}
|
||||
>
|
||||
{isCreatingItem ? "Adding..." : "Add Item"}
|
||||
</button>
|
||||
</div>
|
||||
|
||||
<div>
|
||||
<h3>Items:</h3>
|
||||
{items?.map((item) => (
|
||||
<div key={item.id}>
|
||||
{item.id}: {item.value}
|
||||
<button
|
||||
onClick={() => {
|
||||
// Direct RPC calls bypass React Query caching/mutations
|
||||
// Use for one-off operations or when you need immediate execution
|
||||
return rpcClient.demo.storage.remove(item.id);
|
||||
}}
|
||||
>
|
||||
Remove
|
||||
</button>
|
||||
</div>
|
||||
))}
|
||||
</div>
|
||||
</div>
|
||||
);
|
||||
}
|
15
src/client/main.tsx
Normal file
15
src/client/main.tsx
Normal file
@@ -0,0 +1,15 @@
|
||||
import { StrictMode } from "react";
|
||||
import { createRoot } from "react-dom/client";
|
||||
import "./styles/globals.css";
|
||||
import App from "./app.tsx";
|
||||
import { QueryClient, QueryClientProvider } from "@tanstack/react-query";
|
||||
|
||||
const queryClient = new QueryClient();
|
||||
|
||||
createRoot(document.getElementById("root")!).render(
|
||||
<StrictMode>
|
||||
<QueryClientProvider client={queryClient}>
|
||||
<App />
|
||||
</QueryClientProvider>
|
||||
</StrictMode>,
|
||||
);
|
12
src/client/rpc-client.ts
Normal file
12
src/client/rpc-client.ts
Normal file
@@ -0,0 +1,12 @@
|
||||
import type { RouterClient } from "@orpc/server";
|
||||
import { createORPCClient } from "@orpc/client";
|
||||
import { RPCLink } from "@orpc/client/fetch";
|
||||
import { createTanstackQueryUtils } from "@orpc/tanstack-query";
|
||||
|
||||
import type { router } from "@/server/rpc";
|
||||
|
||||
const link = new RPCLink({ url: `${window.location.origin}/rpc` });
|
||||
|
||||
export const rpcClient: RouterClient<typeof router> = createORPCClient(link);
|
||||
|
||||
export const queryClient = createTanstackQueryUtils(rpcClient);
|
1
src/client/styles/globals.css
Normal file
1
src/client/styles/globals.css
Normal file
@@ -0,0 +1 @@
|
||||
@import "tailwindcss";
|
1
src/client/vite-env.d.ts
vendored
Normal file
1
src/client/vite-env.d.ts
vendored
Normal file
@@ -0,0 +1 @@
|
||||
/// <reference types="vite/client" />
|
Reference in New Issue
Block a user