init
This commit is contained in:
		
						commit
						4a1397651f
					
				
							
								
								
									
										23
									
								
								.env.example
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										23
									
								
								.env.example
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,23 @@ | |||||||
|  | # Since the ".env" file is gitignored, you can use the ".env.example" file to | ||||||
|  | # build a new ".env" file when you clone the repo. Keep this file up-to-date | ||||||
|  | # when you add new variables to `.env`. | ||||||
|  | 
 | ||||||
|  | # This file will be committed to version control, so make sure not to have any | ||||||
|  | # secrets in it. If you are cloning this repo, create a copy of this file named | ||||||
|  | # ".env" and populate it with your secrets. | ||||||
|  | 
 | ||||||
|  | # When adding additional environment variables, the schema in "/src/env.js" | ||||||
|  | # should be updated accordingly. | ||||||
|  | 
 | ||||||
|  | # Next Auth | ||||||
|  | # You can generate a new secret on the command line with: | ||||||
|  | # npx auth secret | ||||||
|  | # https://next-auth.js.org/configuration/options#secret | ||||||
|  | AUTH_SECRET="" | ||||||
|  | 
 | ||||||
|  | # Next Auth Discord Provider | ||||||
|  | AUTH_DISCORD_ID="" | ||||||
|  | AUTH_DISCORD_SECRET="" | ||||||
|  | 
 | ||||||
|  | # Drizzle | ||||||
|  | DATABASE_URL="postgresql://postgres:password@localhost:5432/wiki-antifa" | ||||||
							
								
								
									
										61
									
								
								.eslintrc.cjs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										61
									
								
								.eslintrc.cjs
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,61 @@ | |||||||
|  | /** @type {import("eslint").Linter.Config} */ | ||||||
|  | const config = { | ||||||
|  |   "parser": "@typescript-eslint/parser", | ||||||
|  |   "parserOptions": { | ||||||
|  |     "project": true | ||||||
|  |   }, | ||||||
|  |   "plugins": [ | ||||||
|  |     "@typescript-eslint", | ||||||
|  |     "drizzle" | ||||||
|  |   ], | ||||||
|  |   "extends": [ | ||||||
|  |     "next/core-web-vitals", | ||||||
|  |     "plugin:@typescript-eslint/recommended-type-checked", | ||||||
|  |     "plugin:@typescript-eslint/stylistic-type-checked" | ||||||
|  |   ], | ||||||
|  |   "rules": { | ||||||
|  |     "@typescript-eslint/array-type": "off", | ||||||
|  |     "@typescript-eslint/consistent-type-definitions": "off", | ||||||
|  |     "@typescript-eslint/consistent-type-imports": [ | ||||||
|  |       "warn", | ||||||
|  |       { | ||||||
|  |         "prefer": "type-imports", | ||||||
|  |         "fixStyle": "inline-type-imports" | ||||||
|  |       } | ||||||
|  |     ], | ||||||
|  |     "@typescript-eslint/no-unused-vars": [ | ||||||
|  |       "warn", | ||||||
|  |       { | ||||||
|  |         "argsIgnorePattern": "^_" | ||||||
|  |       } | ||||||
|  |     ], | ||||||
|  |     "@typescript-eslint/require-await": "off", | ||||||
|  |     "@typescript-eslint/no-misused-promises": [ | ||||||
|  |       "error", | ||||||
|  |       { | ||||||
|  |         "checksVoidReturn": { | ||||||
|  |           "attributes": false | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |     ], | ||||||
|  |     "drizzle/enforce-delete-with-where": [ | ||||||
|  |       "error", | ||||||
|  |       { | ||||||
|  |         "drizzleObjectName": [ | ||||||
|  |           "db", | ||||||
|  |           "ctx.db" | ||||||
|  |         ] | ||||||
|  |       } | ||||||
|  |     ], | ||||||
|  |     "drizzle/enforce-update-with-where": [ | ||||||
|  |       "error", | ||||||
|  |       { | ||||||
|  |         "drizzleObjectName": [ | ||||||
|  |           "db", | ||||||
|  |           "ctx.db" | ||||||
|  |         ] | ||||||
|  |       } | ||||||
|  |     ] | ||||||
|  |   } | ||||||
|  | } | ||||||
|  | module.exports = config; | ||||||
							
								
								
									
										46
									
								
								.gitignore
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										46
									
								
								.gitignore
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @ -0,0 +1,46 @@ | |||||||
|  | # See https://help.github.com/articles/ignoring-files/ for more about ignoring files. | ||||||
|  | 
 | ||||||
|  | # dependencies | ||||||
|  | /node_modules | ||||||
|  | /.pnp | ||||||
|  | .pnp.js | ||||||
|  | 
 | ||||||
|  | # testing | ||||||
|  | /coverage | ||||||
|  | 
 | ||||||
|  | # database | ||||||
|  | /prisma/db.sqlite | ||||||
|  | /prisma/db.sqlite-journal | ||||||
|  | db.sqlite | ||||||
|  | 
 | ||||||
|  | # next.js | ||||||
|  | /.next/ | ||||||
|  | /out/ | ||||||
|  | next-env.d.ts | ||||||
|  | 
 | ||||||
|  | # production | ||||||
|  | /build | ||||||
|  | 
 | ||||||
|  | # misc | ||||||
|  | .DS_Store | ||||||
|  | *.pem | ||||||
|  | 
 | ||||||
|  | # debug | ||||||
|  | npm-debug.log* | ||||||
|  | yarn-debug.log* | ||||||
|  | yarn-error.log* | ||||||
|  | .pnpm-debug.log* | ||||||
|  | 
 | ||||||
|  | # local env files | ||||||
|  | # do not commit any .env files to git, except for the .env.example file. https://create.t3.gg/en/usage/env-variables#using-environment-variables | ||||||
|  | .env | ||||||
|  | .env*.local | ||||||
|  | 
 | ||||||
|  | # vercel | ||||||
|  | .vercel | ||||||
|  | 
 | ||||||
|  | # typescript | ||||||
|  | *.tsbuildinfo | ||||||
|  | 
 | ||||||
|  | # idea files | ||||||
|  | .idea | ||||||
							
								
								
									
										29
									
								
								README.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										29
									
								
								README.md
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,29 @@ | |||||||
|  | # Create T3 App | ||||||
|  | 
 | ||||||
|  | This is a [T3 Stack](https://create.t3.gg/) project bootstrapped with `create-t3-app`. | ||||||
|  | 
 | ||||||
|  | ## What's next? How do I make an app with this? | ||||||
|  | 
 | ||||||
|  | We try to keep this project as simple as possible, so you can start with just the scaffolding we set up for you, and add additional things later when they become necessary. | ||||||
|  | 
 | ||||||
|  | If you are not familiar with the different technologies used in this project, please refer to the respective docs. If you still are in the wind, please join our [Discord](https://t3.gg/discord) and ask for help. | ||||||
|  | 
 | ||||||
|  | - [Next.js](https://nextjs.org) | ||||||
|  | - [NextAuth.js](https://next-auth.js.org) | ||||||
|  | - [Prisma](https://prisma.io) | ||||||
|  | - [Drizzle](https://orm.drizzle.team) | ||||||
|  | - [Tailwind CSS](https://tailwindcss.com) | ||||||
|  | - [tRPC](https://trpc.io) | ||||||
|  | 
 | ||||||
|  | ## Learn More | ||||||
|  | 
 | ||||||
|  | To learn more about the [T3 Stack](https://create.t3.gg/), take a look at the following resources: | ||||||
|  | 
 | ||||||
|  | - [Documentation](https://create.t3.gg/) | ||||||
|  | - [Learn the T3 Stack](https://create.t3.gg/en/faq#what-learning-resources-are-currently-available) — Check out these awesome tutorials | ||||||
|  | 
 | ||||||
|  | You can check out the [create-t3-app GitHub repository](https://github.com/t3-oss/create-t3-app) — your feedback and contributions are welcome! | ||||||
|  | 
 | ||||||
|  | ## How do I deploy this? | ||||||
|  | 
 | ||||||
|  | Follow our deployment guides for [Vercel](https://create.t3.gg/en/deployment/vercel), [Netlify](https://create.t3.gg/en/deployment/netlify) and [Docker](https://create.t3.gg/en/deployment/docker) for more information. | ||||||
							
								
								
									
										21
									
								
								components.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										21
									
								
								components.json
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,21 @@ | |||||||
|  | { | ||||||
|  |   "$schema": "https://ui.shadcn.com/schema.json", | ||||||
|  |   "style": "new-york", | ||||||
|  |   "rsc": true, | ||||||
|  |   "tsx": true, | ||||||
|  |   "tailwind": { | ||||||
|  |     "config": "tailwind.config.ts", | ||||||
|  |     "css": "src/styles/globals.css", | ||||||
|  |     "baseColor": "neutral", | ||||||
|  |     "cssVariables": true, | ||||||
|  |     "prefix": "" | ||||||
|  |   }, | ||||||
|  |   "aliases": { | ||||||
|  |     "components": "@/components", | ||||||
|  |     "utils": "@/lib/utils", | ||||||
|  |     "ui": "@/components/ui", | ||||||
|  |     "lib": "@/lib", | ||||||
|  |     "hooks": "@/hooks" | ||||||
|  |   }, | ||||||
|  |   "iconLibrary": "lucide" | ||||||
|  | } | ||||||
							
								
								
									
										12
									
								
								drizzle.config.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										12
									
								
								drizzle.config.ts
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,12 @@ | |||||||
|  | import { type Config } from "drizzle-kit"; | ||||||
|  | 
 | ||||||
|  | import { env } from "@/env"; | ||||||
|  | 
 | ||||||
|  | export default { | ||||||
|  |   schema: "./src/server/db/schema.ts", | ||||||
|  |   dialect: "postgresql", | ||||||
|  |   dbCredentials: { | ||||||
|  |     url: env.DATABASE_URL, | ||||||
|  |   }, | ||||||
|  |   tablesFilter: ["wiki-antifa_*"], | ||||||
|  | } satisfies Config; | ||||||
							
								
								
									
										10
									
								
								next.config.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										10
									
								
								next.config.js
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,10 @@ | |||||||
|  | /** | ||||||
|  |  * Run `build` or `dev` with `SKIP_ENV_VALIDATION` to skip env validation. This is especially useful | ||||||
|  |  * for Docker builds. | ||||||
|  |  */ | ||||||
|  | import "./src/env.js"; | ||||||
|  | 
 | ||||||
|  | /** @type {import("next").NextConfig} */ | ||||||
|  | const config = {}; | ||||||
|  | 
 | ||||||
|  | export default config; | ||||||
							
								
								
									
										84
									
								
								package.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										84
									
								
								package.json
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,84 @@ | |||||||
|  | { | ||||||
|  |   "name": "wiki-antifa", | ||||||
|  |   "version": "0.1.0", | ||||||
|  |   "private": true, | ||||||
|  |   "type": "module", | ||||||
|  |   "scripts": { | ||||||
|  |     "build": "next build", | ||||||
|  |     "check": "next lint && tsc --noEmit", | ||||||
|  |     "db:generate": "drizzle-kit generate", | ||||||
|  |     "db:migrate": "drizzle-kit migrate", | ||||||
|  |     "db:push": "drizzle-kit push", | ||||||
|  |     "db:studio": "drizzle-kit studio", | ||||||
|  |     "dev": "next dev --turbo", | ||||||
|  |     "format:check": "prettier --check \"**/*.{ts,tsx,js,jsx,mdx}\" --cache", | ||||||
|  |     "format:write": "prettier --write \"**/*.{ts,tsx,js,jsx,mdx}\" --cache", | ||||||
|  |     "lint": "next lint", | ||||||
|  |     "lint:fix": "next lint --fix", | ||||||
|  |     "preview": "next build && next start", | ||||||
|  |     "start": "next start", | ||||||
|  |     "typecheck": "tsc --noEmit" | ||||||
|  |   }, | ||||||
|  |   "dependencies": { | ||||||
|  |     "@auth/drizzle-adapter": "^1.7.2", | ||||||
|  |     "@hookform/resolvers": "^4.1.3", | ||||||
|  |     "@paralleldrive/cuid2": "^2.2.2", | ||||||
|  |     "@radix-ui/react-avatar": "^1.1.3", | ||||||
|  |     "@radix-ui/react-dialog": "^1.1.6", | ||||||
|  |     "@radix-ui/react-label": "^2.1.2", | ||||||
|  |     "@radix-ui/react-popover": "^1.1.6", | ||||||
|  |     "@radix-ui/react-separator": "^1.1.2", | ||||||
|  |     "@radix-ui/react-slot": "^1.1.2", | ||||||
|  |     "@radix-ui/react-tooltip": "^1.1.8", | ||||||
|  |     "@t3-oss/env-nextjs": "^0.10.1", | ||||||
|  |     "@tanstack/react-query": "^5.50.0", | ||||||
|  |     "@tiptap/extension-color": "^2.11.5", | ||||||
|  |     "@tiptap/extension-image": "^2.11.5", | ||||||
|  |     "@tiptap/extension-list-item": "^2.11.5", | ||||||
|  |     "@tiptap/extension-text-style": "^2.11.5", | ||||||
|  |     "@tiptap/pm": "^2.11.5", | ||||||
|  |     "@tiptap/react": "^2.11.5", | ||||||
|  |     "@tiptap/starter-kit": "^2.11.5", | ||||||
|  |     "@trpc/client": "^11.0.0-rc.446", | ||||||
|  |     "@trpc/react-query": "^11.0.0-rc.446", | ||||||
|  |     "@trpc/server": "^11.0.0-rc.446", | ||||||
|  |     "class-variance-authority": "^0.7.1", | ||||||
|  |     "clsx": "^2.1.1", | ||||||
|  |     "drizzle-orm": "^0.33.0", | ||||||
|  |     "geist": "^1.3.0", | ||||||
|  |     "lucide-react": "^0.477.0", | ||||||
|  |     "next": "^15.0.1", | ||||||
|  |     "next-auth": "5.0.0-beta.25", | ||||||
|  |     "postgres": "^3.4.4", | ||||||
|  |     "react": "^18.3.1", | ||||||
|  |     "react-dom": "^18.3.1", | ||||||
|  |     "react-hook-form": "^7.54.2", | ||||||
|  |     "react-textarea-autosize": "^8.5.7", | ||||||
|  |     "server-only": "^0.0.1", | ||||||
|  |     "superjson": "^2.2.1", | ||||||
|  |     "tailwind-merge": "^3.0.2", | ||||||
|  |     "tailwindcss-animate": "^1.0.7", | ||||||
|  |     "zod": "^3.24.2" | ||||||
|  |   }, | ||||||
|  |   "devDependencies": { | ||||||
|  |     "@types/eslint": "^8.56.10", | ||||||
|  |     "@types/node": "^20.14.10", | ||||||
|  |     "@types/react": "^18.3.3", | ||||||
|  |     "@types/react-dom": "^18.3.0", | ||||||
|  |     "@typescript-eslint/eslint-plugin": "^8.1.0", | ||||||
|  |     "@typescript-eslint/parser": "^8.1.0", | ||||||
|  |     "drizzle-kit": "^0.24.0", | ||||||
|  |     "eslint": "^8.57.0", | ||||||
|  |     "eslint-config-next": "^15.0.1", | ||||||
|  |     "eslint-plugin-drizzle": "^0.2.3", | ||||||
|  |     "postcss": "^8.4.39", | ||||||
|  |     "prettier": "^3.3.2", | ||||||
|  |     "prettier-plugin-tailwindcss": "^0.6.5", | ||||||
|  |     "tailwindcss": "^3.4.3", | ||||||
|  |     "typescript": "^5.5.3" | ||||||
|  |   }, | ||||||
|  |   "ct3aMetadata": { | ||||||
|  |     "initVersion": "7.38.1" | ||||||
|  |   }, | ||||||
|  |   "packageManager": "pnpm@10.3.0" | ||||||
|  | } | ||||||
							
								
								
									
										6191
									
								
								pnpm-lock.yaml
									
									
									
										generated
									
									
									
										Normal file
									
								
							
							
						
						
									
										6191
									
								
								pnpm-lock.yaml
									
									
									
										generated
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										5
									
								
								postcss.config.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										5
									
								
								postcss.config.js
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,5 @@ | |||||||
|  | export default { | ||||||
|  |   plugins: { | ||||||
|  |     tailwindcss: {}, | ||||||
|  |   }, | ||||||
|  | }; | ||||||
							
								
								
									
										4
									
								
								prettier.config.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										4
									
								
								prettier.config.js
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,4 @@ | |||||||
|  | /** @type {import('prettier').Config & import('prettier-plugin-tailwindcss').PluginOptions} */ | ||||||
|  | export default { | ||||||
|  |   plugins: ["prettier-plugin-tailwindcss"], | ||||||
|  | }; | ||||||
							
								
								
									
										
											BIN
										
									
								
								public/favicon.ico
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								public/favicon.ico
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| After Width: | Height: | Size: 23 KiB | 
| @ -0,0 +1,81 @@ | |||||||
|  | "use client"; | ||||||
|  | import React from "react"; | ||||||
|  | import { | ||||||
|  |   Dialog, | ||||||
|  |   DialogContent, | ||||||
|  |   DialogDescription, | ||||||
|  |   DialogHeader, | ||||||
|  |   DialogTitle, | ||||||
|  |   DialogTrigger, | ||||||
|  | } from "@/components/ui/dialog"; | ||||||
|  | 
 | ||||||
|  | import { zodResolver } from "@hookform/resolvers/zod"; | ||||||
|  | import { useForm } from "react-hook-form"; | ||||||
|  | import { z } from "zod"; | ||||||
|  | 
 | ||||||
|  | import { Button } from "@/components/ui/button"; | ||||||
|  | import { | ||||||
|  |   Form, | ||||||
|  |   FormControl, | ||||||
|  |   FormField, | ||||||
|  |   FormItem, | ||||||
|  |   FormLabel, | ||||||
|  |   FormMessage, | ||||||
|  | } from "@/components/ui/form"; | ||||||
|  | import { Input } from "@/components/ui/input"; | ||||||
|  | import { articleSchema } from "@/lib/validation/zod/article"; | ||||||
|  | import { createArticle } from "@/server/actions/article"; | ||||||
|  | 
 | ||||||
|  | const formSchema = articleSchema.pick({ | ||||||
|  |   title: true, | ||||||
|  | }); | ||||||
|  | 
 | ||||||
|  | function CreateArticleDialog() { | ||||||
|  |   const [open, setOpen] = React.useState<boolean>(false); | ||||||
|  |   const form = useForm<z.infer<typeof formSchema>>({ | ||||||
|  |     resolver: zodResolver(formSchema), | ||||||
|  |     defaultValues: { | ||||||
|  |       title: "", | ||||||
|  |     }, | ||||||
|  |   }); | ||||||
|  | 
 | ||||||
|  |   // 2. Define a submit handler.
 | ||||||
|  |   async function onSubmit(values: z.infer<typeof formSchema>) { | ||||||
|  |     setOpen(false); | ||||||
|  |     await createArticle(values); | ||||||
|  |   } | ||||||
|  |   return ( | ||||||
|  |     <Dialog open={open} onOpenChange={setOpen}> | ||||||
|  |       <Button asChild> | ||||||
|  |         <DialogTrigger>Artikel Erstellen</DialogTrigger> | ||||||
|  |       </Button> | ||||||
|  |       <DialogContent> | ||||||
|  |         <DialogHeader> | ||||||
|  |           <DialogTitle>Neuen Artikel Erstellen</DialogTitle> | ||||||
|  |           <DialogDescription></DialogDescription> | ||||||
|  |         </DialogHeader> | ||||||
|  |         <Form {...form}> | ||||||
|  |           <form onSubmit={form.handleSubmit(onSubmit)} className="space-y-8"> | ||||||
|  |             <FormField | ||||||
|  |               control={form.control} | ||||||
|  |               name="title" | ||||||
|  |               render={({ field }) => ( | ||||||
|  |                 <FormItem> | ||||||
|  |                   <FormLabel>Titel</FormLabel> | ||||||
|  |                   <FormControl> | ||||||
|  |                     <Input placeholder="Artikel Titel" {...field} /> | ||||||
|  |                   </FormControl> | ||||||
|  | 
 | ||||||
|  |                   <FormMessage /> | ||||||
|  |                 </FormItem> | ||||||
|  |               )} | ||||||
|  |             /> | ||||||
|  |             <Button type="submit">Artikel Erstellen</Button> | ||||||
|  |           </form> | ||||||
|  |         </Form> | ||||||
|  |       </DialogContent> | ||||||
|  |     </Dialog> | ||||||
|  |   ); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export default CreateArticleDialog; | ||||||
							
								
								
									
										19
									
								
								src/app/(PAGES)/artikel/[slug]/edit/page.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										19
									
								
								src/app/(PAGES)/artikel/[slug]/edit/page.tsx
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,19 @@ | |||||||
|  | import { hasPermission, Role } from "@/lib/validation/has-permission"; | ||||||
|  | import { auth } from "@/server/auth"; | ||||||
|  | import { api } from "@/trpc/server"; | ||||||
|  | import { notFound } from "next/navigation"; | ||||||
|  | import React from "react"; | ||||||
|  | import Editor from "@/components/article/editor/article-form"; | ||||||
|  | 
 | ||||||
|  | async function Page({ params }: { params: Promise<{ slug: string }> }) { | ||||||
|  |   const { slug } = await params; | ||||||
|  |   const session = await auth(); | ||||||
|  |   const isEditor = session?.user | ||||||
|  |     ? hasPermission(session.user.role, Role.EDITOR) | ||||||
|  |     : false; | ||||||
|  |   const article = await api.article.get({ slug: slug }); | ||||||
|  |   if (!article || !isEditor) return notFound(); | ||||||
|  |   return <Editor server_article={article} />; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export default Page; | ||||||
							
								
								
									
										40
									
								
								src/app/(PAGES)/artikel/[slug]/page.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										40
									
								
								src/app/(PAGES)/artikel/[slug]/page.tsx
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,40 @@ | |||||||
|  | import { Button } from "@/components/ui/button"; | ||||||
|  | import { hasPermission, Role } from "@/lib/validation/has-permission"; | ||||||
|  | import { auth } from "@/server/auth"; | ||||||
|  | import { api } from "@/trpc/server"; | ||||||
|  | import { Edit, Edit2Icon, Edit3, MoreVertical, Trash } from "lucide-react"; | ||||||
|  | import Link from "next/link"; | ||||||
|  | import { notFound } from "next/navigation"; | ||||||
|  | import React from "react"; | ||||||
|  | 
 | ||||||
|  | async function Page({ params }: { params: Promise<{ slug: string }> }) { | ||||||
|  |   const { slug } = await params; | ||||||
|  |   const article = await api.article.get({ slug: slug }); | ||||||
|  |   if (!article) return notFound(); | ||||||
|  |   const session = await auth(); | ||||||
|  |   const isEditor = session?.user | ||||||
|  |     ? hasPermission(session.user.role, Role.EDITOR) | ||||||
|  |     : false; | ||||||
|  |   return ( | ||||||
|  |     <div> | ||||||
|  |       <div className="flex w-full items-center justify-between"> | ||||||
|  |         <h1 className="text-4xl font-bold">{article.title}</h1> | ||||||
|  |         {isEditor && ( | ||||||
|  |           <div className="space-x-2"> | ||||||
|  |             <Button asChild variant={"outline"}> | ||||||
|  |               <Link href={`/artikel/${article.slug}/edit`}> | ||||||
|  |                 <Edit className="size-4" /> | ||||||
|  |                 <span>Bearbeiten</span> | ||||||
|  |               </Link> | ||||||
|  |             </Button> | ||||||
|  |             <Button size={"icon"} variant={"outline"}> | ||||||
|  |               <MoreVertical className="size-4" /> | ||||||
|  |             </Button> | ||||||
|  |           </div> | ||||||
|  |         )} | ||||||
|  |       </div> | ||||||
|  |     </div> | ||||||
|  |   ); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export default Page; | ||||||
							
								
								
									
										9
									
								
								src/app/(PAGES)/artikel/page.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										9
									
								
								src/app/(PAGES)/artikel/page.tsx
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,9 @@ | |||||||
|  | import React from 'react' | ||||||
|  | 
 | ||||||
|  | function Page() { | ||||||
|  |   return ( | ||||||
|  |     <div>Alle Artikel Page</div> | ||||||
|  |   ) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export default Page | ||||||
							
								
								
									
										8
									
								
								src/app/(PAGES)/kategorie/[name]/page.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										8
									
								
								src/app/(PAGES)/kategorie/[name]/page.tsx
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,8 @@ | |||||||
|  | import React from "react"; | ||||||
|  | 
 | ||||||
|  | async function Page({ params }: { params: Promise<{ name: string }> }) { | ||||||
|  |   const { name } = await params; | ||||||
|  |   return <div>Kategorie {name}</div>; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export default Page; | ||||||
							
								
								
									
										7
									
								
								src/app/(PAGES)/kategorie/page.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										7
									
								
								src/app/(PAGES)/kategorie/page.tsx
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,7 @@ | |||||||
|  | import React from "react"; | ||||||
|  | 
 | ||||||
|  | function Page() { | ||||||
|  |   return <div>Alle Kategorien</div>; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export default Page; | ||||||
							
								
								
									
										18
									
								
								src/app/(PAGES)/layout.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										18
									
								
								src/app/(PAGES)/layout.tsx
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,18 @@ | |||||||
|  | import { AppSidebar } from "@/components/layout/app-sidebar"; | ||||||
|  | import Navbar from "@/components/layout/navbar"; | ||||||
|  | import { SidebarProvider } from "@/components/ui/sidebar"; | ||||||
|  | import React from "react"; | ||||||
|  | 
 | ||||||
|  | function Layout({ children }: { children: React.ReactNode }) { | ||||||
|  |   return ( | ||||||
|  |     <SidebarProvider> | ||||||
|  |       <AppSidebar /> | ||||||
|  |       <div className="w-full"> | ||||||
|  |         <Navbar /> | ||||||
|  |         <main className="p-4">{children}</main> | ||||||
|  |       </div> | ||||||
|  |     </SidebarProvider> | ||||||
|  |   ); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export default Layout; | ||||||
							
								
								
									
										21
									
								
								src/app/(PAGES)/page.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										21
									
								
								src/app/(PAGES)/page.tsx
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,21 @@ | |||||||
|  | import { auth } from "@/server/auth"; | ||||||
|  | import { api } from "@/trpc/server"; | ||||||
|  | import Link from "next/link"; | ||||||
|  | 
 | ||||||
|  | export default async function Home() { | ||||||
|  |   const session = await auth(); | ||||||
|  |   const articles = await api.article.getAllPreviews(); | ||||||
|  |   return ( | ||||||
|  |     <div className=""> | ||||||
|  |       <h1 className="text-4xl font-bold">Anti Rechts Wiki</h1> | ||||||
|  | 
 | ||||||
|  |       <menu> | ||||||
|  |         {articles.map((article) => ( | ||||||
|  |           <li key={article.slug}> | ||||||
|  |             <Link href={`/artikel/${article.slug}`}>{article.title}</Link> | ||||||
|  |           </li> | ||||||
|  |         ))} | ||||||
|  |       </menu> | ||||||
|  |     </div> | ||||||
|  |   ); | ||||||
|  | } | ||||||
							
								
								
									
										3
									
								
								src/app/api/auth/[...nextauth]/route.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								src/app/api/auth/[...nextauth]/route.ts
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,3 @@ | |||||||
|  | import { handlers } from "@/server/auth"; | ||||||
|  | 
 | ||||||
|  | export const { GET, POST } = handlers; | ||||||
							
								
								
									
										34
									
								
								src/app/api/trpc/[trpc]/route.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										34
									
								
								src/app/api/trpc/[trpc]/route.ts
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,34 @@ | |||||||
|  | import { fetchRequestHandler } from "@trpc/server/adapters/fetch"; | ||||||
|  | import { type NextRequest } from "next/server"; | ||||||
|  | 
 | ||||||
|  | import { env } from "@/env"; | ||||||
|  | import { appRouter } from "@/server/api/root"; | ||||||
|  | import { createTRPCContext } from "@/server/api/trpc"; | ||||||
|  | 
 | ||||||
|  | /** | ||||||
|  |  * This wraps the `createTRPCContext` helper and provides the required context for the tRPC API when | ||||||
|  |  * handling a HTTP request (e.g. when you make requests from Client Components). | ||||||
|  |  */ | ||||||
|  | const createContext = async (req: NextRequest) => { | ||||||
|  |   return createTRPCContext({ | ||||||
|  |     headers: req.headers, | ||||||
|  |   }); | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | const handler = (req: NextRequest) => | ||||||
|  |   fetchRequestHandler({ | ||||||
|  |     endpoint: "/api/trpc", | ||||||
|  |     req, | ||||||
|  |     router: appRouter, | ||||||
|  |     createContext: () => createContext(req), | ||||||
|  |     onError: | ||||||
|  |       env.NODE_ENV === "development" | ||||||
|  |         ? ({ path, error }) => { | ||||||
|  |             console.error( | ||||||
|  |               `❌ tRPC failed on ${path ?? "<no-path>"}: ${error.message}` | ||||||
|  |             ); | ||||||
|  |           } | ||||||
|  |         : undefined, | ||||||
|  |   }); | ||||||
|  | 
 | ||||||
|  | export { handler as GET, handler as POST }; | ||||||
							
								
								
									
										24
									
								
								src/app/layout.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										24
									
								
								src/app/layout.tsx
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,24 @@ | |||||||
|  | import "@/styles/globals.css"; | ||||||
|  | 
 | ||||||
|  | import { GeistSans } from "geist/font/sans"; | ||||||
|  | import { type Metadata } from "next"; | ||||||
|  | 
 | ||||||
|  | import { TRPCReactProvider } from "@/trpc/react"; | ||||||
|  | 
 | ||||||
|  | export const metadata: Metadata = { | ||||||
|  |   title: "Create T3 App", | ||||||
|  |   description: "Generated by create-t3-app", | ||||||
|  |   icons: [{ rel: "icon", url: "/favicon.ico" }], | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | export default function RootLayout({ | ||||||
|  |   children, | ||||||
|  | }: Readonly<{ children: React.ReactNode }>) { | ||||||
|  |   return ( | ||||||
|  |     <html lang="en" className={`${GeistSans.variable}`}> | ||||||
|  |       <body> | ||||||
|  |         <TRPCReactProvider>{children}</TRPCReactProvider> | ||||||
|  |       </body> | ||||||
|  |     </html> | ||||||
|  |   ); | ||||||
|  | } | ||||||
							
								
								
									
										95
									
								
								src/components/article/editor/article-form.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										95
									
								
								src/components/article/editor/article-form.tsx
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,95 @@ | |||||||
|  | "use client"; | ||||||
|  | 
 | ||||||
|  | import "./styles.css"; | ||||||
|  | import { Article } from "@/server/db/schema"; | ||||||
|  | 
 | ||||||
|  | import React from "react"; | ||||||
|  | 
 | ||||||
|  | import { z } from "zod"; | ||||||
|  | import { zodResolver } from "@hookform/resolvers/zod"; | ||||||
|  | import { useForm } from "react-hook-form"; | ||||||
|  | import TextareaAutosize from "react-textarea-autosize"; | ||||||
|  | 
 | ||||||
|  | import { | ||||||
|  |   Form, | ||||||
|  |   FormControl, | ||||||
|  |   FormField, | ||||||
|  |   FormItem, | ||||||
|  |   FormMessage, | ||||||
|  | } from "@/components/ui/form"; | ||||||
|  | import { articleSchema } from "@/lib/validation/zod/article"; | ||||||
|  | import { debounce } from "@/lib/utils"; | ||||||
|  | import { updateArticle } from "@/server/actions/article"; | ||||||
|  | import Editor from "./editor"; | ||||||
|  | 
 | ||||||
|  | export default ({ server_article }: { server_article: Article }) => { | ||||||
|  |   const form = useForm<z.infer<typeof articleSchema>>({ | ||||||
|  |     resolver: zodResolver(articleSchema), | ||||||
|  |     defaultValues: { | ||||||
|  |       title: server_article?.title ?? "", | ||||||
|  |       content: | ||||||
|  |         server_article?.content ?? | ||||||
|  |         `<h2>
 | ||||||
|  |             Hey bearbeite mich! | ||||||
|  |          </h2>`,
 | ||||||
|  |     }, | ||||||
|  |   }); | ||||||
|  | 
 | ||||||
|  |   // 2. Define a submit handler.
 | ||||||
|  |   async function onSubmit(values: z.infer<typeof articleSchema>) { | ||||||
|  |     await updateArticle(values, server_article.id); | ||||||
|  |   } | ||||||
|  |   const debouncedSubmit = React.useCallback( | ||||||
|  |     debounce(() => { | ||||||
|  |       form.handleSubmit(onSubmit)(); | ||||||
|  |     }, 1000), | ||||||
|  |     [form], | ||||||
|  |   ); | ||||||
|  | 
 | ||||||
|  |   return ( | ||||||
|  |     <Form {...form}> | ||||||
|  |       <form onSubmit={form.handleSubmit(onSubmit)} className="space-y-8"> | ||||||
|  |         <FormField | ||||||
|  |           control={form.control} | ||||||
|  |           name="title" | ||||||
|  |           render={({ field }) => ( | ||||||
|  |             <FormItem> | ||||||
|  |               <FormControl> | ||||||
|  |                 <TextareaAutosize | ||||||
|  |                   value={field.value} | ||||||
|  |                   onChange={(e) => { | ||||||
|  |                     field.onChange(e); | ||||||
|  |                     debouncedSubmit(); | ||||||
|  |                   }} | ||||||
|  |                 /> | ||||||
|  |               </FormControl> | ||||||
|  |               <FormMessage /> | ||||||
|  |             </FormItem> | ||||||
|  |           )} | ||||||
|  |         /> | ||||||
|  | 
 | ||||||
|  |         <FormField | ||||||
|  |           control={form.control} | ||||||
|  |           name="content" | ||||||
|  |           render={({ field }) => ( | ||||||
|  |             <FormItem> | ||||||
|  |               <FormControl> | ||||||
|  |                 <Editor | ||||||
|  |                   editorProviderProps={{ | ||||||
|  |                     content: field.value, | ||||||
|  |                     onUpdate: (value) => { | ||||||
|  |                       field.onChange(value.editor.getHTML()); | ||||||
|  |                       debouncedSubmit(); | ||||||
|  |                     }, | ||||||
|  |                   }} | ||||||
|  |                 /> | ||||||
|  |               </FormControl> | ||||||
|  | 
 | ||||||
|  |               <FormMessage /> | ||||||
|  |             </FormItem> | ||||||
|  |           )} | ||||||
|  |         /> | ||||||
|  |       </form> | ||||||
|  |     </Form> | ||||||
|  |   ); | ||||||
|  | }; | ||||||
							
								
								
									
										24
									
								
								src/components/article/editor/editor.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										24
									
								
								src/components/article/editor/editor.tsx
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,24 @@ | |||||||
|  | "use client"; | ||||||
|  | import React from "react"; | ||||||
|  | import { EditorProvider, EditorProviderProps } from "@tiptap/react"; | ||||||
|  | import { MenuBar } from "./menu-bar"; | ||||||
|  | import { extensions } from "./extentions"; | ||||||
|  | 
 | ||||||
|  | function Editor({ | ||||||
|  |   editorProviderProps, | ||||||
|  |   readOnly, | ||||||
|  | }: { | ||||||
|  |   editorProviderProps: EditorProviderProps; | ||||||
|  |   readOnly?: boolean; | ||||||
|  | }) { | ||||||
|  |   return ( | ||||||
|  |     <EditorProvider | ||||||
|  |       immediatelyRender={false} | ||||||
|  |       extensions={extensions} | ||||||
|  |       slotBefore={!readOnly && <MenuBar />} | ||||||
|  |       {...editorProviderProps} | ||||||
|  |     /> | ||||||
|  |   ); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export default Editor; | ||||||
							
								
								
									
										22
									
								
								src/components/article/editor/extentions.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										22
									
								
								src/components/article/editor/extentions.ts
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,22 @@ | |||||||
|  | import { Color } from "@tiptap/extension-color"; | ||||||
|  | import ListItem from "@tiptap/extension-list-item"; | ||||||
|  | import TextStyle from "@tiptap/extension-text-style"; | ||||||
|  | import StarterKit from "@tiptap/starter-kit"; | ||||||
|  | import Image from "@tiptap/extension-image"; | ||||||
|  | 
 | ||||||
|  | export const extensions = [ | ||||||
|  |   Color.configure({ types: [TextStyle.name, ListItem.name] }), | ||||||
|  |   /* @ts-ignore */ | ||||||
|  |   TextStyle.configure({ types: [ListItem.name] }), | ||||||
|  |   StarterKit.configure({ | ||||||
|  |     bulletList: { | ||||||
|  |       keepMarks: true, | ||||||
|  |       keepAttributes: false, // TODO : Making this as `false` becase marks are not preserved when I try to preserve attrs, awaiting a bit of help
 | ||||||
|  |     }, | ||||||
|  |     orderedList: { | ||||||
|  |       keepMarks: true, | ||||||
|  |       keepAttributes: false, // TODO : Making this as `false` becase marks are not preserved when I try to preserve attrs, awaiting a bit of help
 | ||||||
|  |     }, | ||||||
|  |   }), | ||||||
|  |   Image, | ||||||
|  | ]; | ||||||
							
								
								
									
										226
									
								
								src/components/article/editor/menu-bar.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										226
									
								
								src/components/article/editor/menu-bar.tsx
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,226 @@ | |||||||
|  | import { Button } from "@/components/ui/button"; | ||||||
|  | import { useCurrentEditor } from "@tiptap/react"; | ||||||
|  | import { | ||||||
|  |   BoldIcon, | ||||||
|  |   CodeIcon, | ||||||
|  |   CodeSquareIcon, | ||||||
|  |   Heading2Icon, | ||||||
|  |   Heading3Icon, | ||||||
|  |   Heading4Icon, | ||||||
|  |   Heading5Icon, | ||||||
|  |   Heading6Icon, | ||||||
|  |   ItalicIcon, | ||||||
|  |   ListIcon, | ||||||
|  |   ListOrderedIcon, | ||||||
|  |   QuoteIcon, | ||||||
|  |   RedoIcon, | ||||||
|  |   SeparatorHorizontalIcon, | ||||||
|  |   StrikethroughIcon, | ||||||
|  |   TextIcon, | ||||||
|  |   UndoIcon, | ||||||
|  | } from "lucide-react"; | ||||||
|  | 
 | ||||||
|  | export const MenuBar = () => { | ||||||
|  |   const { editor } = useCurrentEditor(); | ||||||
|  | 
 | ||||||
|  |   if (!editor) { | ||||||
|  |     return null; | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   return ( | ||||||
|  |     <div className="control-group my-4"> | ||||||
|  |       <div className="Button-group flex items-center gap-1"> | ||||||
|  |         <Button | ||||||
|  |           size={"icon"} | ||||||
|  |           variant={"outline"} | ||||||
|  |           onClick={() => editor.chain().focus().toggleBold().run()} | ||||||
|  |           disabled={!editor.can().chain().focus().toggleBold().run()} | ||||||
|  |           className={editor.isActive("bold") ? "is-active" : ""} | ||||||
|  |         > | ||||||
|  |           <BoldIcon className="size-4" /> | ||||||
|  |         </Button> | ||||||
|  |         <Button | ||||||
|  |           size={"icon"} | ||||||
|  |           variant={"outline"} | ||||||
|  |           onClick={() => editor.chain().focus().toggleItalic().run()} | ||||||
|  |           disabled={!editor.can().chain().focus().toggleItalic().run()} | ||||||
|  |           className={editor.isActive("italic") ? "is-active" : ""} | ||||||
|  |         > | ||||||
|  |           <ItalicIcon className="size-4" /> | ||||||
|  |         </Button> | ||||||
|  |         <Button | ||||||
|  |           size={"icon"} | ||||||
|  |           variant={"outline"} | ||||||
|  |           onClick={() => editor.chain().focus().toggleStrike().run()} | ||||||
|  |           disabled={!editor.can().chain().focus().toggleStrike().run()} | ||||||
|  |           className={editor.isActive("strike") ? "is-active" : ""} | ||||||
|  |         > | ||||||
|  |           <StrikethroughIcon className="size-4" /> | ||||||
|  |         </Button> | ||||||
|  |         <Button | ||||||
|  |           size={"icon"} | ||||||
|  |           variant={"outline"} | ||||||
|  |           onClick={() => editor.chain().focus().toggleCode().run()} | ||||||
|  |           disabled={!editor.can().chain().focus().toggleCode().run()} | ||||||
|  |           className={editor.isActive("code") ? "is-active" : ""} | ||||||
|  |         > | ||||||
|  |           <CodeIcon className="size-4" /> | ||||||
|  |         </Button> | ||||||
|  | 
 | ||||||
|  |         <Button | ||||||
|  |           variant={"outline"} | ||||||
|  |           size={"icon"} | ||||||
|  |           onClick={() => editor.chain().focus().setParagraph().run()} | ||||||
|  |           className={editor.isActive("paragraph") ? "is-active" : ""} | ||||||
|  |         > | ||||||
|  |           <TextIcon className="size-4" /> | ||||||
|  |         </Button> | ||||||
|  | 
 | ||||||
|  |         <Button | ||||||
|  |           variant={"outline"} | ||||||
|  |           size={"icon"} | ||||||
|  |           onClick={() => | ||||||
|  |             editor.chain().focus().toggleHeading({ level: 2 }).run() | ||||||
|  |           } | ||||||
|  |           className={ | ||||||
|  |             editor.isActive("heading", { level: 2 }) ? "is-active" : "" | ||||||
|  |           } | ||||||
|  |         > | ||||||
|  |           <Heading2Icon className="size-4" /> | ||||||
|  |         </Button> | ||||||
|  |         <Button | ||||||
|  |           variant={"outline"} | ||||||
|  |           size={"icon"} | ||||||
|  |           onClick={() => | ||||||
|  |             editor.chain().focus().toggleHeading({ level: 3 }).run() | ||||||
|  |           } | ||||||
|  |           className={ | ||||||
|  |             editor.isActive("heading", { level: 3 }) ? "is-active" : "" | ||||||
|  |           } | ||||||
|  |         > | ||||||
|  |           <Heading3Icon className="size-4" /> | ||||||
|  |         </Button> | ||||||
|  |         <Button | ||||||
|  |           variant={"outline"} | ||||||
|  |           size={"icon"} | ||||||
|  |           onClick={() => | ||||||
|  |             editor.chain().focus().toggleHeading({ level: 4 }).run() | ||||||
|  |           } | ||||||
|  |           className={ | ||||||
|  |             editor.isActive("heading", { level: 4 }) ? "is-active" : "" | ||||||
|  |           } | ||||||
|  |         > | ||||||
|  |           <Heading4Icon className="size-4" /> | ||||||
|  |         </Button> | ||||||
|  |         <Button | ||||||
|  |           variant={"outline"} | ||||||
|  |           size={"icon"} | ||||||
|  |           onClick={() => | ||||||
|  |             editor.chain().focus().toggleHeading({ level: 5 }).run() | ||||||
|  |           } | ||||||
|  |           className={ | ||||||
|  |             editor.isActive("heading", { level: 5 }) ? "is-active" : "" | ||||||
|  |           } | ||||||
|  |         > | ||||||
|  |           <Heading5Icon className="size-4" /> | ||||||
|  |         </Button> | ||||||
|  |         <Button | ||||||
|  |           variant={"outline"} | ||||||
|  |           size={"icon"} | ||||||
|  |           onClick={() => | ||||||
|  |             editor.chain().focus().toggleHeading({ level: 6 }).run() | ||||||
|  |           } | ||||||
|  |           className={ | ||||||
|  |             editor.isActive("heading", { level: 6 }) ? "is-active" : "" | ||||||
|  |           } | ||||||
|  |         > | ||||||
|  |           <Heading6Icon className="size-4" /> | ||||||
|  |         </Button> | ||||||
|  |         <Button | ||||||
|  |           variant={"outline"} | ||||||
|  |           size={"icon"} | ||||||
|  |           onClick={() => editor.chain().focus().toggleBulletList().run()} | ||||||
|  |           className={editor.isActive("bulletList") ? "is-active" : ""} | ||||||
|  |         > | ||||||
|  |           <ListIcon className="size-4" /> | ||||||
|  |         </Button> | ||||||
|  |         <Button | ||||||
|  |           variant={"outline"} | ||||||
|  |           size={"icon"} | ||||||
|  |           onClick={() => editor.chain().focus().toggleOrderedList().run()} | ||||||
|  |           className={editor.isActive("orderedList") ? "is-active" : ""} | ||||||
|  |         > | ||||||
|  |           <ListOrderedIcon className="size-4" /> | ||||||
|  |         </Button> | ||||||
|  |         <Button | ||||||
|  |           variant={"outline"} | ||||||
|  |           size={"icon"} | ||||||
|  |           onClick={() => editor.chain().focus().toggleCodeBlock().run()} | ||||||
|  |           className={editor.isActive("codeBlock") ? "is-active" : ""} | ||||||
|  |         > | ||||||
|  |           <CodeSquareIcon className="size-4" /> | ||||||
|  |         </Button> | ||||||
|  |         <Button | ||||||
|  |           variant={"outline"} | ||||||
|  |           size={"icon"} | ||||||
|  |           onClick={() => editor.chain().focus().toggleBlockquote().run()} | ||||||
|  |           className={editor.isActive("blockquote") ? "is-active" : ""} | ||||||
|  |         > | ||||||
|  |           <QuoteIcon className="size-4" /> | ||||||
|  |         </Button> | ||||||
|  |         <Button | ||||||
|  |           variant={"outline"} | ||||||
|  |           size={"icon"} | ||||||
|  |           onClick={() => editor.chain().focus().setHorizontalRule().run()} | ||||||
|  |         > | ||||||
|  |           <SeparatorHorizontalIcon className="size-4" /> | ||||||
|  |         </Button> | ||||||
|  |         <Button | ||||||
|  |           variant={"outline"} | ||||||
|  |           onClick={() => editor.chain().focus().setHardBreak().run()} | ||||||
|  |         > | ||||||
|  |           Hard Break | ||||||
|  |         </Button> | ||||||
|  | 
 | ||||||
|  |         <Button | ||||||
|  |           onClick={() => editor.chain().focus().setColor("#958DF1").run()} | ||||||
|  |           className={ | ||||||
|  |             editor.isActive("textStyle", { color: "#958DF1" }) | ||||||
|  |               ? "is-active" | ||||||
|  |               : "" | ||||||
|  |           } | ||||||
|  |         > | ||||||
|  |           Purple | ||||||
|  |         </Button> | ||||||
|  |         {/* <Button | ||||||
|  |           variant={"outline"} | ||||||
|  |           onClick={() => editor.chain().focus().unsetAllMarks().run()} | ||||||
|  |         > | ||||||
|  |           Formatierung aufheben | ||||||
|  |         </Button> | ||||||
|  |         <Button | ||||||
|  |           variant={"outline"} | ||||||
|  |           onClick={() => editor.chain().focus().clearNodes().run()} | ||||||
|  |         > | ||||||
|  |           Clear nodes | ||||||
|  |         </Button> */} | ||||||
|  |         <Button | ||||||
|  |           variant={"outline"} | ||||||
|  |           size={"icon"} | ||||||
|  |           onClick={() => editor.chain().focus().undo().run()} | ||||||
|  |           disabled={!editor.can().chain().focus().undo().run()} | ||||||
|  |         > | ||||||
|  |           <UndoIcon className="size-4" /> | ||||||
|  |         </Button> | ||||||
|  |         <Button | ||||||
|  |           variant={"outline"} | ||||||
|  |           size={"icon"} | ||||||
|  |           onClick={() => editor.chain().focus().redo().run()} | ||||||
|  |           disabled={!editor.can().chain().focus().redo().run()} | ||||||
|  |         > | ||||||
|  |           <RedoIcon className="size-4" /> | ||||||
|  |         </Button> | ||||||
|  |       </div> | ||||||
|  |     </div> | ||||||
|  |   ); | ||||||
|  | }; | ||||||
							
								
								
									
										95
									
								
								src/components/article/editor/styles.css
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										95
									
								
								src/components/article/editor/styles.css
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,95 @@ | |||||||
|  | /* Basic editor styles */ | ||||||
|  | .tiptap > *:first-child { | ||||||
|  |   margin-top: 0; | ||||||
|  | } | ||||||
|  | .ProseMirror:focus { | ||||||
|  |   outline: none; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /* List styles */ | ||||||
|  | .tiptap ul, | ||||||
|  | .tiptap ol { | ||||||
|  |   padding: 0 1rem; | ||||||
|  |   margin: 1.25rem 1rem 1.25rem 0.4rem; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | .tiptap ul li p, | ||||||
|  | .tiptap ol li p { | ||||||
|  |   margin-top: 0.25em; | ||||||
|  |   margin-bottom: 0.25em; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /* Heading styles */ | ||||||
|  | .tiptap h1, | ||||||
|  | .tiptap h2, | ||||||
|  | .tiptap h3, | ||||||
|  | .tiptap h4, | ||||||
|  | .tiptap h5, | ||||||
|  | .tiptap h6 { | ||||||
|  |   line-height: 1.1; | ||||||
|  |   margin-top: 2.5rem; | ||||||
|  |   text-wrap: pretty; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | .tiptap h1, | ||||||
|  | .tiptap h2 { | ||||||
|  |   margin-top: 3.5rem; | ||||||
|  |   margin-bottom: 1.5rem; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | .tiptap h1 { | ||||||
|  |   font-size: 1.4rem; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | .tiptap h2 { | ||||||
|  |   font-size: 1.2rem; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | .tiptap h3 { | ||||||
|  |   font-size: 1.1rem; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | .tiptap h4, | ||||||
|  | .tiptap h5, | ||||||
|  | .tiptap h6 { | ||||||
|  |   font-size: 1rem; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /* Code and preformatted text styles */ | ||||||
|  | .tiptap code { | ||||||
|  |   background-color: var(--purple-light); | ||||||
|  |   border-radius: 0.4rem; | ||||||
|  |   color: var(--black); | ||||||
|  |   font-size: 0.85rem; | ||||||
|  |   padding: 0.25em 0.3em; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | .tiptap pre { | ||||||
|  |   background: var(--black); | ||||||
|  |   border-radius: 0.5rem; | ||||||
|  |   color: var(--white); | ||||||
|  |   font-family: "JetBrainsMono", monospace; | ||||||
|  |   margin: 1.5rem 0; | ||||||
|  |   padding: 0.75rem 1rem; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | .tiptap pre code { | ||||||
|  |   background: none; | ||||||
|  |   color: inherit; | ||||||
|  |   font-size: 0.8rem; | ||||||
|  |   padding: 0; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /* Blockquote styles */ | ||||||
|  | .tiptap blockquote { | ||||||
|  |   border-left: 3px solid var(--gray-3); | ||||||
|  |   margin: 1.5rem 0; | ||||||
|  |   padding-left: 1rem; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /* Horizontal rule styles */ | ||||||
|  | .tiptap hr { | ||||||
|  |   border: none; | ||||||
|  |   border-top: 1px solid var(--gray-2); | ||||||
|  |   margin: 2rem 0; | ||||||
|  | } | ||||||
							
								
								
									
										25
									
								
								src/components/avatar.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										25
									
								
								src/components/avatar.tsx
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,25 @@ | |||||||
|  | import React from "react"; | ||||||
|  | import { | ||||||
|  |   Avatar as AvatarComponent, | ||||||
|  |   AvatarFallback, | ||||||
|  |   AvatarImage, | ||||||
|  | } from "@/components/ui/avatar"; | ||||||
|  | 
 | ||||||
|  | function Avatar({ | ||||||
|  |   src, | ||||||
|  |   fb, | ||||||
|  |   className, | ||||||
|  | }: { | ||||||
|  |   src?: string | null; | ||||||
|  |   fb?: string | null; | ||||||
|  |   className?: string; | ||||||
|  | }) { | ||||||
|  |   return ( | ||||||
|  |     <AvatarComponent className={className}> | ||||||
|  |       <AvatarImage src={src!} /> | ||||||
|  |       <AvatarFallback>{fb?.slice(0, 2)?.toUpperCase()}</AvatarFallback> | ||||||
|  |     </AvatarComponent> | ||||||
|  |   ); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export default Avatar; | ||||||
							
								
								
									
										87
									
								
								src/components/layout/app-sidebar.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										87
									
								
								src/components/layout/app-sidebar.tsx
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,87 @@ | |||||||
|  | import * as React from "react"; | ||||||
|  | 
 | ||||||
|  | import { | ||||||
|  |   Sidebar, | ||||||
|  |   SidebarContent, | ||||||
|  |   SidebarFooter, | ||||||
|  |   SidebarGroup, | ||||||
|  |   SidebarGroupContent, | ||||||
|  |   SidebarGroupLabel, | ||||||
|  |   SidebarHeader, | ||||||
|  |   SidebarMenu, | ||||||
|  | } from "@/components/ui/sidebar"; | ||||||
|  | import Link from "next/link"; | ||||||
|  | import { Separator } from "../ui/separator"; | ||||||
|  | import SidebarLink from "./sidebar-link"; | ||||||
|  | import { Button } from "../ui/button"; | ||||||
|  | import { HomeIcon } from "lucide-react"; | ||||||
|  | import { api } from "@/trpc/server"; | ||||||
|  | 
 | ||||||
|  | export async function AppSidebar({ | ||||||
|  |   ...props | ||||||
|  | }: React.ComponentProps<typeof Sidebar>) { | ||||||
|  |   const articles = await api.article.getAllPreviews(); | ||||||
|  |   return ( | ||||||
|  |     <Sidebar {...props}> | ||||||
|  |       <SidebarHeader className="flex h-14 items-center justify-center border-b"> | ||||||
|  |         <Link href={"/"}> | ||||||
|  |           <span className="text-2xl font-bold">Wiki</span> | ||||||
|  |         </Link> | ||||||
|  |       </SidebarHeader> | ||||||
|  |       <SidebarContent> | ||||||
|  |         <SidebarGroup> | ||||||
|  |           <Button asChild> | ||||||
|  |             <Link href={"/"}> | ||||||
|  |               <HomeIcon className="size-4" /> | ||||||
|  |               <span>Start</span> | ||||||
|  |             </Link> | ||||||
|  |           </Button> | ||||||
|  |         </SidebarGroup> | ||||||
|  |         {/* We create a SidebarGroup for each parent. */} | ||||||
|  |         {/* {data.navMain.map((item) => ( | ||||||
|  |           <SidebarGroup key={item.title}> | ||||||
|  |             <SidebarGroupLabel>{item.title}</SidebarGroupLabel> | ||||||
|  |             <SidebarGroupContent> | ||||||
|  |               <SidebarMenu> | ||||||
|  |                 {item.items.map((item) => ( | ||||||
|  |                   <SidebarLink key={item.title} {...item} /> | ||||||
|  |                 ))} | ||||||
|  |               </SidebarMenu> | ||||||
|  |             </SidebarGroupContent> | ||||||
|  |           </SidebarGroup> | ||||||
|  |         ))} */} | ||||||
|  |         <SidebarGroup> | ||||||
|  |           <SidebarGroupLabel>Artikel</SidebarGroupLabel> | ||||||
|  |           <SidebarGroupContent> | ||||||
|  |             <SidebarMenu> | ||||||
|  |               {articles?.map((article) => ( | ||||||
|  |                 <SidebarLink | ||||||
|  |                   key={article.slug} | ||||||
|  |                   title={article.title!} | ||||||
|  |                   url={`/artikel/${article.slug}`} | ||||||
|  |                 /> | ||||||
|  |               ))} | ||||||
|  |             </SidebarMenu> | ||||||
|  |           </SidebarGroupContent> | ||||||
|  |         </SidebarGroup> | ||||||
|  |       </SidebarContent> | ||||||
|  |       <SidebarFooter> | ||||||
|  |         <div className="flex items-center justify-center gap-4"> | ||||||
|  |           <Link | ||||||
|  |             className="text-xs text-muted-foreground underline" | ||||||
|  |             href={"/impressum"} | ||||||
|  |           > | ||||||
|  |             Impressum | ||||||
|  |           </Link> | ||||||
|  |           <Separator orientation="vertical" /> | ||||||
|  |           <Link | ||||||
|  |             href={"/datenschutz"} | ||||||
|  |             className="text-xs text-muted-foreground underline" | ||||||
|  |           > | ||||||
|  |             Datenschutz | ||||||
|  |           </Link> | ||||||
|  |         </div> | ||||||
|  |       </SidebarFooter> | ||||||
|  |     </Sidebar> | ||||||
|  |   ); | ||||||
|  | } | ||||||
							
								
								
									
										59
									
								
								src/components/layout/navbar.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										59
									
								
								src/components/layout/navbar.tsx
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,59 @@ | |||||||
|  | import React from "react"; | ||||||
|  | import { Input } from "../ui/input"; | ||||||
|  | import { auth } from "@/server/auth"; | ||||||
|  | import { Button } from "../ui/button"; | ||||||
|  | import Link from "next/link"; | ||||||
|  | import Avatar from "../avatar"; | ||||||
|  | import { hasPermission, Role } from "@/lib/validation/has-permission"; | ||||||
|  | import { | ||||||
|  |   Popover, | ||||||
|  |   PopoverContent, | ||||||
|  |   PopoverTrigger, | ||||||
|  | } from "@/components/ui/popover"; | ||||||
|  | import CreateArticleDialog from "@/app/(PAGES)/_components/article/create-article-dialog"; | ||||||
|  | 
 | ||||||
|  | async function Navbar() { | ||||||
|  |   const session = await auth(); | ||||||
|  |   const isEditor = session?.user | ||||||
|  |     ? hasPermission(session.user.role, Role.EDITOR) | ||||||
|  |     : false; | ||||||
|  |   return ( | ||||||
|  |     <div className="flex h-14 items-center justify-between border-b bg-sidebar px-4"> | ||||||
|  |       <Input className="w-full max-w-xs" placeholder="Suche..." /> | ||||||
|  | 
 | ||||||
|  |       <div className="flex items-center gap-4"> | ||||||
|  |         {isEditor && ( | ||||||
|  |           <Popover> | ||||||
|  |             <PopoverTrigger asChild> | ||||||
|  |               <Button>Erstellen</Button> | ||||||
|  |             </PopoverTrigger> | ||||||
|  | 
 | ||||||
|  |             <PopoverContent | ||||||
|  |               className="w-full max-w-48 space-y-2 bg-sidebar" | ||||||
|  |               align="end" | ||||||
|  |               // side="left"
 | ||||||
|  |             > | ||||||
|  |               <CreateArticleDialog /> | ||||||
|  |               <Button className="w-full" asChild> | ||||||
|  |                 <Link href={"/kategorie/neu"}>Kategorie erstellen</Link> | ||||||
|  |               </Button> | ||||||
|  |             </PopoverContent> | ||||||
|  |           </Popover> | ||||||
|  |         )} | ||||||
|  |         {session ? ( | ||||||
|  |           <Avatar | ||||||
|  |             className="size-8" | ||||||
|  |             src={session.user.image} | ||||||
|  |             fb={session.user.name} | ||||||
|  |           /> | ||||||
|  |         ) : ( | ||||||
|  |           <Button asChild> | ||||||
|  |             <Link href={"/api/auth/signin"}>Anmelden</Link> | ||||||
|  |           </Button> | ||||||
|  |         )} | ||||||
|  |       </div> | ||||||
|  |     </div> | ||||||
|  |   ); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export default Navbar; | ||||||
							
								
								
									
										19
									
								
								src/components/layout/sidebar-link.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										19
									
								
								src/components/layout/sidebar-link.tsx
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,19 @@ | |||||||
|  | "use client"; | ||||||
|  | 
 | ||||||
|  | import React from "react"; | ||||||
|  | import { usePathname } from "next/navigation"; | ||||||
|  | import { SidebarMenuButton, SidebarMenuItem } from "@/components/ui/sidebar"; | ||||||
|  | import Link from "next/link"; | ||||||
|  | 
 | ||||||
|  | function SidebarLink({ url, title }: { url: string; title: string }) { | ||||||
|  |   const isActive = usePathname() === url; | ||||||
|  |   return ( | ||||||
|  |     <SidebarMenuItem> | ||||||
|  |       <SidebarMenuButton asChild isActive={isActive}> | ||||||
|  |         <Link href={url}>{title}</Link> | ||||||
|  |       </SidebarMenuButton> | ||||||
|  |     </SidebarMenuItem> | ||||||
|  |   ); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export default SidebarLink; | ||||||
							
								
								
									
										50
									
								
								src/components/ui/avatar.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										50
									
								
								src/components/ui/avatar.tsx
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,50 @@ | |||||||
|  | "use client" | ||||||
|  | 
 | ||||||
|  | import * as React from "react" | ||||||
|  | import * as AvatarPrimitive from "@radix-ui/react-avatar" | ||||||
|  | 
 | ||||||
|  | import { cn } from "@/lib/utils" | ||||||
|  | 
 | ||||||
|  | const Avatar = React.forwardRef< | ||||||
|  |   React.ElementRef<typeof AvatarPrimitive.Root>, | ||||||
|  |   React.ComponentPropsWithoutRef<typeof AvatarPrimitive.Root> | ||||||
|  | >(({ className, ...props }, ref) => ( | ||||||
|  |   <AvatarPrimitive.Root | ||||||
|  |     ref={ref} | ||||||
|  |     className={cn( | ||||||
|  |       "relative flex h-10 w-10 shrink-0 overflow-hidden rounded-full", | ||||||
|  |       className | ||||||
|  |     )} | ||||||
|  |     {...props} | ||||||
|  |   /> | ||||||
|  | )) | ||||||
|  | Avatar.displayName = AvatarPrimitive.Root.displayName | ||||||
|  | 
 | ||||||
|  | const AvatarImage = React.forwardRef< | ||||||
|  |   React.ElementRef<typeof AvatarPrimitive.Image>, | ||||||
|  |   React.ComponentPropsWithoutRef<typeof AvatarPrimitive.Image> | ||||||
|  | >(({ className, ...props }, ref) => ( | ||||||
|  |   <AvatarPrimitive.Image | ||||||
|  |     ref={ref} | ||||||
|  |     className={cn("aspect-square h-full w-full", className)} | ||||||
|  |     {...props} | ||||||
|  |   /> | ||||||
|  | )) | ||||||
|  | AvatarImage.displayName = AvatarPrimitive.Image.displayName | ||||||
|  | 
 | ||||||
|  | const AvatarFallback = React.forwardRef< | ||||||
|  |   React.ElementRef<typeof AvatarPrimitive.Fallback>, | ||||||
|  |   React.ComponentPropsWithoutRef<typeof AvatarPrimitive.Fallback> | ||||||
|  | >(({ className, ...props }, ref) => ( | ||||||
|  |   <AvatarPrimitive.Fallback | ||||||
|  |     ref={ref} | ||||||
|  |     className={cn( | ||||||
|  |       "flex h-full w-full items-center justify-center rounded-full bg-muted", | ||||||
|  |       className | ||||||
|  |     )} | ||||||
|  |     {...props} | ||||||
|  |   /> | ||||||
|  | )) | ||||||
|  | AvatarFallback.displayName = AvatarPrimitive.Fallback.displayName | ||||||
|  | 
 | ||||||
|  | export { Avatar, AvatarImage, AvatarFallback } | ||||||
							
								
								
									
										57
									
								
								src/components/ui/button.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										57
									
								
								src/components/ui/button.tsx
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,57 @@ | |||||||
|  | import * as React from "react" | ||||||
|  | import { Slot } from "@radix-ui/react-slot" | ||||||
|  | import { cva, type VariantProps } from "class-variance-authority" | ||||||
|  | 
 | ||||||
|  | import { cn } from "@/lib/utils" | ||||||
|  | 
 | ||||||
|  | const buttonVariants = cva( | ||||||
|  |   "inline-flex items-center justify-center gap-2 whitespace-nowrap rounded-md text-sm font-medium transition-colors focus-visible:outline-none focus-visible:ring-1 focus-visible:ring-ring disabled:pointer-events-none disabled:opacity-50 [&_svg]:pointer-events-none [&_svg]:size-4 [&_svg]:shrink-0", | ||||||
|  |   { | ||||||
|  |     variants: { | ||||||
|  |       variant: { | ||||||
|  |         default: | ||||||
|  |           "bg-primary text-primary-foreground shadow hover:bg-primary/90", | ||||||
|  |         destructive: | ||||||
|  |           "bg-destructive text-destructive-foreground shadow-sm hover:bg-destructive/90", | ||||||
|  |         outline: | ||||||
|  |           "border border-input bg-background shadow-sm hover:bg-accent hover:text-accent-foreground", | ||||||
|  |         secondary: | ||||||
|  |           "bg-secondary text-secondary-foreground shadow-sm hover:bg-secondary/80", | ||||||
|  |         ghost: "hover:bg-accent hover:text-accent-foreground", | ||||||
|  |         link: "text-primary underline-offset-4 hover:underline", | ||||||
|  |       }, | ||||||
|  |       size: { | ||||||
|  |         default: "h-9 px-4 py-2", | ||||||
|  |         sm: "h-8 rounded-md px-3 text-xs", | ||||||
|  |         lg: "h-10 rounded-md px-8", | ||||||
|  |         icon: "h-9 w-9", | ||||||
|  |       }, | ||||||
|  |     }, | ||||||
|  |     defaultVariants: { | ||||||
|  |       variant: "default", | ||||||
|  |       size: "default", | ||||||
|  |     }, | ||||||
|  |   } | ||||||
|  | ) | ||||||
|  | 
 | ||||||
|  | export interface ButtonProps | ||||||
|  |   extends React.ButtonHTMLAttributes<HTMLButtonElement>, | ||||||
|  |     VariantProps<typeof buttonVariants> { | ||||||
|  |   asChild?: boolean | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | const Button = React.forwardRef<HTMLButtonElement, ButtonProps>( | ||||||
|  |   ({ className, variant, size, asChild = false, ...props }, ref) => { | ||||||
|  |     const Comp = asChild ? Slot : "button" | ||||||
|  |     return ( | ||||||
|  |       <Comp | ||||||
|  |         className={cn(buttonVariants({ variant, size, className }))} | ||||||
|  |         ref={ref} | ||||||
|  |         {...props} | ||||||
|  |       /> | ||||||
|  |     ) | ||||||
|  |   } | ||||||
|  | ) | ||||||
|  | Button.displayName = "Button" | ||||||
|  | 
 | ||||||
|  | export { Button, buttonVariants } | ||||||
							
								
								
									
										122
									
								
								src/components/ui/dialog.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										122
									
								
								src/components/ui/dialog.tsx
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,122 @@ | |||||||
|  | "use client" | ||||||
|  | 
 | ||||||
|  | import * as React from "react" | ||||||
|  | import * as DialogPrimitive from "@radix-ui/react-dialog" | ||||||
|  | import { X } from "lucide-react" | ||||||
|  | 
 | ||||||
|  | import { cn } from "@/lib/utils" | ||||||
|  | 
 | ||||||
|  | const Dialog = DialogPrimitive.Root | ||||||
|  | 
 | ||||||
|  | const DialogTrigger = DialogPrimitive.Trigger | ||||||
|  | 
 | ||||||
|  | const DialogPortal = DialogPrimitive.Portal | ||||||
|  | 
 | ||||||
|  | const DialogClose = DialogPrimitive.Close | ||||||
|  | 
 | ||||||
|  | const DialogOverlay = React.forwardRef< | ||||||
|  |   React.ElementRef<typeof DialogPrimitive.Overlay>, | ||||||
|  |   React.ComponentPropsWithoutRef<typeof DialogPrimitive.Overlay> | ||||||
|  | >(({ className, ...props }, ref) => ( | ||||||
|  |   <DialogPrimitive.Overlay | ||||||
|  |     ref={ref} | ||||||
|  |     className={cn( | ||||||
|  |       "fixed inset-0 z-50 bg-black/80  data-[state=open]:animate-in data-[state=closed]:animate-out data-[state=closed]:fade-out-0 data-[state=open]:fade-in-0", | ||||||
|  |       className | ||||||
|  |     )} | ||||||
|  |     {...props} | ||||||
|  |   /> | ||||||
|  | )) | ||||||
|  | DialogOverlay.displayName = DialogPrimitive.Overlay.displayName | ||||||
|  | 
 | ||||||
|  | const DialogContent = React.forwardRef< | ||||||
|  |   React.ElementRef<typeof DialogPrimitive.Content>, | ||||||
|  |   React.ComponentPropsWithoutRef<typeof DialogPrimitive.Content> | ||||||
|  | >(({ className, children, ...props }, ref) => ( | ||||||
|  |   <DialogPortal> | ||||||
|  |     <DialogOverlay /> | ||||||
|  |     <DialogPrimitive.Content | ||||||
|  |       ref={ref} | ||||||
|  |       className={cn( | ||||||
|  |         "fixed left-[50%] top-[50%] z-50 grid w-full max-w-lg translate-x-[-50%] translate-y-[-50%] gap-4 border bg-background p-6 shadow-lg duration-200 data-[state=open]:animate-in data-[state=closed]:animate-out data-[state=closed]:fade-out-0 data-[state=open]:fade-in-0 data-[state=closed]:zoom-out-95 data-[state=open]:zoom-in-95 data-[state=closed]:slide-out-to-left-1/2 data-[state=closed]:slide-out-to-top-[48%] data-[state=open]:slide-in-from-left-1/2 data-[state=open]:slide-in-from-top-[48%] sm:rounded-lg", | ||||||
|  |         className | ||||||
|  |       )} | ||||||
|  |       {...props} | ||||||
|  |     > | ||||||
|  |       {children} | ||||||
|  |       <DialogPrimitive.Close className="absolute right-4 top-4 rounded-sm opacity-70 ring-offset-background transition-opacity hover:opacity-100 focus:outline-none focus:ring-2 focus:ring-ring focus:ring-offset-2 disabled:pointer-events-none data-[state=open]:bg-accent data-[state=open]:text-muted-foreground"> | ||||||
|  |         <X className="h-4 w-4" /> | ||||||
|  |         <span className="sr-only">Close</span> | ||||||
|  |       </DialogPrimitive.Close> | ||||||
|  |     </DialogPrimitive.Content> | ||||||
|  |   </DialogPortal> | ||||||
|  | )) | ||||||
|  | DialogContent.displayName = DialogPrimitive.Content.displayName | ||||||
|  | 
 | ||||||
|  | const DialogHeader = ({ | ||||||
|  |   className, | ||||||
|  |   ...props | ||||||
|  | }: React.HTMLAttributes<HTMLDivElement>) => ( | ||||||
|  |   <div | ||||||
|  |     className={cn( | ||||||
|  |       "flex flex-col space-y-1.5 text-center sm:text-left", | ||||||
|  |       className | ||||||
|  |     )} | ||||||
|  |     {...props} | ||||||
|  |   /> | ||||||
|  | ) | ||||||
|  | DialogHeader.displayName = "DialogHeader" | ||||||
|  | 
 | ||||||
|  | const DialogFooter = ({ | ||||||
|  |   className, | ||||||
|  |   ...props | ||||||
|  | }: React.HTMLAttributes<HTMLDivElement>) => ( | ||||||
|  |   <div | ||||||
|  |     className={cn( | ||||||
|  |       "flex flex-col-reverse sm:flex-row sm:justify-end sm:space-x-2", | ||||||
|  |       className | ||||||
|  |     )} | ||||||
|  |     {...props} | ||||||
|  |   /> | ||||||
|  | ) | ||||||
|  | DialogFooter.displayName = "DialogFooter" | ||||||
|  | 
 | ||||||
|  | const DialogTitle = React.forwardRef< | ||||||
|  |   React.ElementRef<typeof DialogPrimitive.Title>, | ||||||
|  |   React.ComponentPropsWithoutRef<typeof DialogPrimitive.Title> | ||||||
|  | >(({ className, ...props }, ref) => ( | ||||||
|  |   <DialogPrimitive.Title | ||||||
|  |     ref={ref} | ||||||
|  |     className={cn( | ||||||
|  |       "text-lg font-semibold leading-none tracking-tight", | ||||||
|  |       className | ||||||
|  |     )} | ||||||
|  |     {...props} | ||||||
|  |   /> | ||||||
|  | )) | ||||||
|  | DialogTitle.displayName = DialogPrimitive.Title.displayName | ||||||
|  | 
 | ||||||
|  | const DialogDescription = React.forwardRef< | ||||||
|  |   React.ElementRef<typeof DialogPrimitive.Description>, | ||||||
|  |   React.ComponentPropsWithoutRef<typeof DialogPrimitive.Description> | ||||||
|  | >(({ className, ...props }, ref) => ( | ||||||
|  |   <DialogPrimitive.Description | ||||||
|  |     ref={ref} | ||||||
|  |     className={cn("text-sm text-muted-foreground", className)} | ||||||
|  |     {...props} | ||||||
|  |   /> | ||||||
|  | )) | ||||||
|  | DialogDescription.displayName = DialogPrimitive.Description.displayName | ||||||
|  | 
 | ||||||
|  | export { | ||||||
|  |   Dialog, | ||||||
|  |   DialogPortal, | ||||||
|  |   DialogOverlay, | ||||||
|  |   DialogTrigger, | ||||||
|  |   DialogClose, | ||||||
|  |   DialogContent, | ||||||
|  |   DialogHeader, | ||||||
|  |   DialogFooter, | ||||||
|  |   DialogTitle, | ||||||
|  |   DialogDescription, | ||||||
|  | } | ||||||
							
								
								
									
										178
									
								
								src/components/ui/form.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										178
									
								
								src/components/ui/form.tsx
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,178 @@ | |||||||
|  | "use client" | ||||||
|  | 
 | ||||||
|  | import * as React from "react" | ||||||
|  | import * as LabelPrimitive from "@radix-ui/react-label" | ||||||
|  | import { Slot } from "@radix-ui/react-slot" | ||||||
|  | import { | ||||||
|  |   Controller, | ||||||
|  |   FormProvider, | ||||||
|  |   useFormContext, | ||||||
|  |   type ControllerProps, | ||||||
|  |   type FieldPath, | ||||||
|  |   type FieldValues, | ||||||
|  | } from "react-hook-form" | ||||||
|  | 
 | ||||||
|  | import { cn } from "@/lib/utils" | ||||||
|  | import { Label } from "@/components/ui/label" | ||||||
|  | 
 | ||||||
|  | const Form = FormProvider | ||||||
|  | 
 | ||||||
|  | type FormFieldContextValue< | ||||||
|  |   TFieldValues extends FieldValues = FieldValues, | ||||||
|  |   TName extends FieldPath<TFieldValues> = FieldPath<TFieldValues> | ||||||
|  | > = { | ||||||
|  |   name: TName | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | const FormFieldContext = React.createContext<FormFieldContextValue>( | ||||||
|  |   {} as FormFieldContextValue | ||||||
|  | ) | ||||||
|  | 
 | ||||||
|  | const FormField = < | ||||||
|  |   TFieldValues extends FieldValues = FieldValues, | ||||||
|  |   TName extends FieldPath<TFieldValues> = FieldPath<TFieldValues> | ||||||
|  | >({ | ||||||
|  |   ...props | ||||||
|  | }: ControllerProps<TFieldValues, TName>) => { | ||||||
|  |   return ( | ||||||
|  |     <FormFieldContext.Provider value={{ name: props.name }}> | ||||||
|  |       <Controller {...props} /> | ||||||
|  |     </FormFieldContext.Provider> | ||||||
|  |   ) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | const useFormField = () => { | ||||||
|  |   const fieldContext = React.useContext(FormFieldContext) | ||||||
|  |   const itemContext = React.useContext(FormItemContext) | ||||||
|  |   const { getFieldState, formState } = useFormContext() | ||||||
|  | 
 | ||||||
|  |   const fieldState = getFieldState(fieldContext.name, formState) | ||||||
|  | 
 | ||||||
|  |   if (!fieldContext) { | ||||||
|  |     throw new Error("useFormField should be used within <FormField>") | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   const { id } = itemContext | ||||||
|  | 
 | ||||||
|  |   return { | ||||||
|  |     id, | ||||||
|  |     name: fieldContext.name, | ||||||
|  |     formItemId: `${id}-form-item`, | ||||||
|  |     formDescriptionId: `${id}-form-item-description`, | ||||||
|  |     formMessageId: `${id}-form-item-message`, | ||||||
|  |     ...fieldState, | ||||||
|  |   } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | type FormItemContextValue = { | ||||||
|  |   id: string | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | const FormItemContext = React.createContext<FormItemContextValue>( | ||||||
|  |   {} as FormItemContextValue | ||||||
|  | ) | ||||||
|  | 
 | ||||||
|  | const FormItem = React.forwardRef< | ||||||
|  |   HTMLDivElement, | ||||||
|  |   React.HTMLAttributes<HTMLDivElement> | ||||||
|  | >(({ className, ...props }, ref) => { | ||||||
|  |   const id = React.useId() | ||||||
|  | 
 | ||||||
|  |   return ( | ||||||
|  |     <FormItemContext.Provider value={{ id }}> | ||||||
|  |       <div ref={ref} className={cn("space-y-2", className)} {...props} /> | ||||||
|  |     </FormItemContext.Provider> | ||||||
|  |   ) | ||||||
|  | }) | ||||||
|  | FormItem.displayName = "FormItem" | ||||||
|  | 
 | ||||||
|  | const FormLabel = React.forwardRef< | ||||||
|  |   React.ElementRef<typeof LabelPrimitive.Root>, | ||||||
|  |   React.ComponentPropsWithoutRef<typeof LabelPrimitive.Root> | ||||||
|  | >(({ className, ...props }, ref) => { | ||||||
|  |   const { error, formItemId } = useFormField() | ||||||
|  | 
 | ||||||
|  |   return ( | ||||||
|  |     <Label | ||||||
|  |       ref={ref} | ||||||
|  |       className={cn(error && "text-destructive", className)} | ||||||
|  |       htmlFor={formItemId} | ||||||
|  |       {...props} | ||||||
|  |     /> | ||||||
|  |   ) | ||||||
|  | }) | ||||||
|  | FormLabel.displayName = "FormLabel" | ||||||
|  | 
 | ||||||
|  | const FormControl = React.forwardRef< | ||||||
|  |   React.ElementRef<typeof Slot>, | ||||||
|  |   React.ComponentPropsWithoutRef<typeof Slot> | ||||||
|  | >(({ ...props }, ref) => { | ||||||
|  |   const { error, formItemId, formDescriptionId, formMessageId } = useFormField() | ||||||
|  | 
 | ||||||
|  |   return ( | ||||||
|  |     <Slot | ||||||
|  |       ref={ref} | ||||||
|  |       id={formItemId} | ||||||
|  |       aria-describedby={ | ||||||
|  |         !error | ||||||
|  |           ? `${formDescriptionId}` | ||||||
|  |           : `${formDescriptionId} ${formMessageId}` | ||||||
|  |       } | ||||||
|  |       aria-invalid={!!error} | ||||||
|  |       {...props} | ||||||
|  |     /> | ||||||
|  |   ) | ||||||
|  | }) | ||||||
|  | FormControl.displayName = "FormControl" | ||||||
|  | 
 | ||||||
|  | const FormDescription = React.forwardRef< | ||||||
|  |   HTMLParagraphElement, | ||||||
|  |   React.HTMLAttributes<HTMLParagraphElement> | ||||||
|  | >(({ className, ...props }, ref) => { | ||||||
|  |   const { formDescriptionId } = useFormField() | ||||||
|  | 
 | ||||||
|  |   return ( | ||||||
|  |     <p | ||||||
|  |       ref={ref} | ||||||
|  |       id={formDescriptionId} | ||||||
|  |       className={cn("text-[0.8rem] text-muted-foreground", className)} | ||||||
|  |       {...props} | ||||||
|  |     /> | ||||||
|  |   ) | ||||||
|  | }) | ||||||
|  | FormDescription.displayName = "FormDescription" | ||||||
|  | 
 | ||||||
|  | const FormMessage = React.forwardRef< | ||||||
|  |   HTMLParagraphElement, | ||||||
|  |   React.HTMLAttributes<HTMLParagraphElement> | ||||||
|  | >(({ className, children, ...props }, ref) => { | ||||||
|  |   const { error, formMessageId } = useFormField() | ||||||
|  |   const body = error ? String(error?.message ?? "") : children | ||||||
|  | 
 | ||||||
|  |   if (!body) { | ||||||
|  |     return null | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   return ( | ||||||
|  |     <p | ||||||
|  |       ref={ref} | ||||||
|  |       id={formMessageId} | ||||||
|  |       className={cn("text-[0.8rem] font-medium text-destructive", className)} | ||||||
|  |       {...props} | ||||||
|  |     > | ||||||
|  |       {body} | ||||||
|  |     </p> | ||||||
|  |   ) | ||||||
|  | }) | ||||||
|  | FormMessage.displayName = "FormMessage" | ||||||
|  | 
 | ||||||
|  | export { | ||||||
|  |   useFormField, | ||||||
|  |   Form, | ||||||
|  |   FormItem, | ||||||
|  |   FormLabel, | ||||||
|  |   FormControl, | ||||||
|  |   FormDescription, | ||||||
|  |   FormMessage, | ||||||
|  |   FormField, | ||||||
|  | } | ||||||
							
								
								
									
										22
									
								
								src/components/ui/input.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										22
									
								
								src/components/ui/input.tsx
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,22 @@ | |||||||
|  | import * as React from "react" | ||||||
|  | 
 | ||||||
|  | import { cn } from "@/lib/utils" | ||||||
|  | 
 | ||||||
|  | const Input = React.forwardRef<HTMLInputElement, React.ComponentProps<"input">>( | ||||||
|  |   ({ className, type, ...props }, ref) => { | ||||||
|  |     return ( | ||||||
|  |       <input | ||||||
|  |         type={type} | ||||||
|  |         className={cn( | ||||||
|  |           "flex h-9 w-full rounded-md border border-input bg-transparent px-3 py-1 text-base shadow-sm transition-colors file:border-0 file:bg-transparent file:text-sm file:font-medium file:text-foreground placeholder:text-muted-foreground focus-visible:outline-none focus-visible:ring-1 focus-visible:ring-ring disabled:cursor-not-allowed disabled:opacity-50 md:text-sm", | ||||||
|  |           className | ||||||
|  |         )} | ||||||
|  |         ref={ref} | ||||||
|  |         {...props} | ||||||
|  |       /> | ||||||
|  |     ) | ||||||
|  |   } | ||||||
|  | ) | ||||||
|  | Input.displayName = "Input" | ||||||
|  | 
 | ||||||
|  | export { Input } | ||||||
							
								
								
									
										26
									
								
								src/components/ui/label.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										26
									
								
								src/components/ui/label.tsx
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,26 @@ | |||||||
|  | "use client" | ||||||
|  | 
 | ||||||
|  | import * as React from "react" | ||||||
|  | import * as LabelPrimitive from "@radix-ui/react-label" | ||||||
|  | import { cva, type VariantProps } from "class-variance-authority" | ||||||
|  | 
 | ||||||
|  | import { cn } from "@/lib/utils" | ||||||
|  | 
 | ||||||
|  | const labelVariants = cva( | ||||||
|  |   "text-sm font-medium leading-none peer-disabled:cursor-not-allowed peer-disabled:opacity-70" | ||||||
|  | ) | ||||||
|  | 
 | ||||||
|  | const Label = React.forwardRef< | ||||||
|  |   React.ElementRef<typeof LabelPrimitive.Root>, | ||||||
|  |   React.ComponentPropsWithoutRef<typeof LabelPrimitive.Root> & | ||||||
|  |     VariantProps<typeof labelVariants> | ||||||
|  | >(({ className, ...props }, ref) => ( | ||||||
|  |   <LabelPrimitive.Root | ||||||
|  |     ref={ref} | ||||||
|  |     className={cn(labelVariants(), className)} | ||||||
|  |     {...props} | ||||||
|  |   /> | ||||||
|  | )) | ||||||
|  | Label.displayName = LabelPrimitive.Root.displayName | ||||||
|  | 
 | ||||||
|  | export { Label } | ||||||
							
								
								
									
										33
									
								
								src/components/ui/popover.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										33
									
								
								src/components/ui/popover.tsx
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,33 @@ | |||||||
|  | "use client" | ||||||
|  | 
 | ||||||
|  | import * as React from "react" | ||||||
|  | import * as PopoverPrimitive from "@radix-ui/react-popover" | ||||||
|  | 
 | ||||||
|  | import { cn } from "@/lib/utils" | ||||||
|  | 
 | ||||||
|  | const Popover = PopoverPrimitive.Root | ||||||
|  | 
 | ||||||
|  | const PopoverTrigger = PopoverPrimitive.Trigger | ||||||
|  | 
 | ||||||
|  | const PopoverAnchor = PopoverPrimitive.Anchor | ||||||
|  | 
 | ||||||
|  | const PopoverContent = React.forwardRef< | ||||||
|  |   React.ElementRef<typeof PopoverPrimitive.Content>, | ||||||
|  |   React.ComponentPropsWithoutRef<typeof PopoverPrimitive.Content> | ||||||
|  | >(({ className, align = "center", sideOffset = 4, ...props }, ref) => ( | ||||||
|  |   <PopoverPrimitive.Portal> | ||||||
|  |     <PopoverPrimitive.Content | ||||||
|  |       ref={ref} | ||||||
|  |       align={align} | ||||||
|  |       sideOffset={sideOffset} | ||||||
|  |       className={cn( | ||||||
|  |         "z-50 w-72 rounded-md border bg-popover p-4 text-popover-foreground shadow-md outline-none data-[state=open]:animate-in data-[state=closed]:animate-out data-[state=closed]:fade-out-0 data-[state=open]:fade-in-0 data-[state=closed]:zoom-out-95 data-[state=open]:zoom-in-95 data-[side=bottom]:slide-in-from-top-2 data-[side=left]:slide-in-from-right-2 data-[side=right]:slide-in-from-left-2 data-[side=top]:slide-in-from-bottom-2", | ||||||
|  |         className | ||||||
|  |       )} | ||||||
|  |       {...props} | ||||||
|  |     /> | ||||||
|  |   </PopoverPrimitive.Portal> | ||||||
|  | )) | ||||||
|  | PopoverContent.displayName = PopoverPrimitive.Content.displayName | ||||||
|  | 
 | ||||||
|  | export { Popover, PopoverTrigger, PopoverContent, PopoverAnchor } | ||||||
							
								
								
									
										31
									
								
								src/components/ui/separator.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										31
									
								
								src/components/ui/separator.tsx
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,31 @@ | |||||||
|  | "use client" | ||||||
|  | 
 | ||||||
|  | import * as React from "react" | ||||||
|  | import * as SeparatorPrimitive from "@radix-ui/react-separator" | ||||||
|  | 
 | ||||||
|  | import { cn } from "@/lib/utils" | ||||||
|  | 
 | ||||||
|  | const Separator = React.forwardRef< | ||||||
|  |   React.ElementRef<typeof SeparatorPrimitive.Root>, | ||||||
|  |   React.ComponentPropsWithoutRef<typeof SeparatorPrimitive.Root> | ||||||
|  | >( | ||||||
|  |   ( | ||||||
|  |     { className, orientation = "horizontal", decorative = true, ...props }, | ||||||
|  |     ref | ||||||
|  |   ) => ( | ||||||
|  |     <SeparatorPrimitive.Root | ||||||
|  |       ref={ref} | ||||||
|  |       decorative={decorative} | ||||||
|  |       orientation={orientation} | ||||||
|  |       className={cn( | ||||||
|  |         "shrink-0 bg-border", | ||||||
|  |         orientation === "horizontal" ? "h-[1px] w-full" : "h-full w-[1px]", | ||||||
|  |         className | ||||||
|  |       )} | ||||||
|  |       {...props} | ||||||
|  |     /> | ||||||
|  |   ) | ||||||
|  | ) | ||||||
|  | Separator.displayName = SeparatorPrimitive.Root.displayName | ||||||
|  | 
 | ||||||
|  | export { Separator } | ||||||
							
								
								
									
										140
									
								
								src/components/ui/sheet.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										140
									
								
								src/components/ui/sheet.tsx
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,140 @@ | |||||||
|  | "use client" | ||||||
|  | 
 | ||||||
|  | import * as React from "react" | ||||||
|  | import * as SheetPrimitive from "@radix-ui/react-dialog" | ||||||
|  | import { cva, type VariantProps } from "class-variance-authority" | ||||||
|  | import { X } from "lucide-react" | ||||||
|  | 
 | ||||||
|  | import { cn } from "@/lib/utils" | ||||||
|  | 
 | ||||||
|  | const Sheet = SheetPrimitive.Root | ||||||
|  | 
 | ||||||
|  | const SheetTrigger = SheetPrimitive.Trigger | ||||||
|  | 
 | ||||||
|  | const SheetClose = SheetPrimitive.Close | ||||||
|  | 
 | ||||||
|  | const SheetPortal = SheetPrimitive.Portal | ||||||
|  | 
 | ||||||
|  | const SheetOverlay = React.forwardRef< | ||||||
|  |   React.ElementRef<typeof SheetPrimitive.Overlay>, | ||||||
|  |   React.ComponentPropsWithoutRef<typeof SheetPrimitive.Overlay> | ||||||
|  | >(({ className, ...props }, ref) => ( | ||||||
|  |   <SheetPrimitive.Overlay | ||||||
|  |     className={cn( | ||||||
|  |       "fixed inset-0 z-50 bg-black/80  data-[state=open]:animate-in data-[state=closed]:animate-out data-[state=closed]:fade-out-0 data-[state=open]:fade-in-0", | ||||||
|  |       className | ||||||
|  |     )} | ||||||
|  |     {...props} | ||||||
|  |     ref={ref} | ||||||
|  |   /> | ||||||
|  | )) | ||||||
|  | SheetOverlay.displayName = SheetPrimitive.Overlay.displayName | ||||||
|  | 
 | ||||||
|  | const sheetVariants = cva( | ||||||
|  |   "fixed z-50 gap-4 bg-background p-6 shadow-lg transition ease-in-out data-[state=closed]:duration-300 data-[state=open]:duration-500 data-[state=open]:animate-in data-[state=closed]:animate-out", | ||||||
|  |   { | ||||||
|  |     variants: { | ||||||
|  |       side: { | ||||||
|  |         top: "inset-x-0 top-0 border-b data-[state=closed]:slide-out-to-top data-[state=open]:slide-in-from-top", | ||||||
|  |         bottom: | ||||||
|  |           "inset-x-0 bottom-0 border-t data-[state=closed]:slide-out-to-bottom data-[state=open]:slide-in-from-bottom", | ||||||
|  |         left: "inset-y-0 left-0 h-full w-3/4 border-r data-[state=closed]:slide-out-to-left data-[state=open]:slide-in-from-left sm:max-w-sm", | ||||||
|  |         right: | ||||||
|  |           "inset-y-0 right-0 h-full w-3/4 border-l data-[state=closed]:slide-out-to-right data-[state=open]:slide-in-from-right sm:max-w-sm", | ||||||
|  |       }, | ||||||
|  |     }, | ||||||
|  |     defaultVariants: { | ||||||
|  |       side: "right", | ||||||
|  |     }, | ||||||
|  |   } | ||||||
|  | ) | ||||||
|  | 
 | ||||||
|  | interface SheetContentProps | ||||||
|  |   extends React.ComponentPropsWithoutRef<typeof SheetPrimitive.Content>, | ||||||
|  |     VariantProps<typeof sheetVariants> {} | ||||||
|  | 
 | ||||||
|  | const SheetContent = React.forwardRef< | ||||||
|  |   React.ElementRef<typeof SheetPrimitive.Content>, | ||||||
|  |   SheetContentProps | ||||||
|  | >(({ side = "right", className, children, ...props }, ref) => ( | ||||||
|  |   <SheetPortal> | ||||||
|  |     <SheetOverlay /> | ||||||
|  |     <SheetPrimitive.Content | ||||||
|  |       ref={ref} | ||||||
|  |       className={cn(sheetVariants({ side }), className)} | ||||||
|  |       {...props} | ||||||
|  |     > | ||||||
|  |       <SheetPrimitive.Close className="absolute right-4 top-4 rounded-sm opacity-70 ring-offset-background transition-opacity hover:opacity-100 focus:outline-none focus:ring-2 focus:ring-ring focus:ring-offset-2 disabled:pointer-events-none data-[state=open]:bg-secondary"> | ||||||
|  |         <X className="h-4 w-4" /> | ||||||
|  |         <span className="sr-only">Close</span> | ||||||
|  |       </SheetPrimitive.Close> | ||||||
|  |       {children} | ||||||
|  |     </SheetPrimitive.Content> | ||||||
|  |   </SheetPortal> | ||||||
|  | )) | ||||||
|  | SheetContent.displayName = SheetPrimitive.Content.displayName | ||||||
|  | 
 | ||||||
|  | const SheetHeader = ({ | ||||||
|  |   className, | ||||||
|  |   ...props | ||||||
|  | }: React.HTMLAttributes<HTMLDivElement>) => ( | ||||||
|  |   <div | ||||||
|  |     className={cn( | ||||||
|  |       "flex flex-col space-y-2 text-center sm:text-left", | ||||||
|  |       className | ||||||
|  |     )} | ||||||
|  |     {...props} | ||||||
|  |   /> | ||||||
|  | ) | ||||||
|  | SheetHeader.displayName = "SheetHeader" | ||||||
|  | 
 | ||||||
|  | const SheetFooter = ({ | ||||||
|  |   className, | ||||||
|  |   ...props | ||||||
|  | }: React.HTMLAttributes<HTMLDivElement>) => ( | ||||||
|  |   <div | ||||||
|  |     className={cn( | ||||||
|  |       "flex flex-col-reverse sm:flex-row sm:justify-end sm:space-x-2", | ||||||
|  |       className | ||||||
|  |     )} | ||||||
|  |     {...props} | ||||||
|  |   /> | ||||||
|  | ) | ||||||
|  | SheetFooter.displayName = "SheetFooter" | ||||||
|  | 
 | ||||||
|  | const SheetTitle = React.forwardRef< | ||||||
|  |   React.ElementRef<typeof SheetPrimitive.Title>, | ||||||
|  |   React.ComponentPropsWithoutRef<typeof SheetPrimitive.Title> | ||||||
|  | >(({ className, ...props }, ref) => ( | ||||||
|  |   <SheetPrimitive.Title | ||||||
|  |     ref={ref} | ||||||
|  |     className={cn("text-lg font-semibold text-foreground", className)} | ||||||
|  |     {...props} | ||||||
|  |   /> | ||||||
|  | )) | ||||||
|  | SheetTitle.displayName = SheetPrimitive.Title.displayName | ||||||
|  | 
 | ||||||
|  | const SheetDescription = React.forwardRef< | ||||||
|  |   React.ElementRef<typeof SheetPrimitive.Description>, | ||||||
|  |   React.ComponentPropsWithoutRef<typeof SheetPrimitive.Description> | ||||||
|  | >(({ className, ...props }, ref) => ( | ||||||
|  |   <SheetPrimitive.Description | ||||||
|  |     ref={ref} | ||||||
|  |     className={cn("text-sm text-muted-foreground", className)} | ||||||
|  |     {...props} | ||||||
|  |   /> | ||||||
|  | )) | ||||||
|  | SheetDescription.displayName = SheetPrimitive.Description.displayName | ||||||
|  | 
 | ||||||
|  | export { | ||||||
|  |   Sheet, | ||||||
|  |   SheetPortal, | ||||||
|  |   SheetOverlay, | ||||||
|  |   SheetTrigger, | ||||||
|  |   SheetClose, | ||||||
|  |   SheetContent, | ||||||
|  |   SheetHeader, | ||||||
|  |   SheetFooter, | ||||||
|  |   SheetTitle, | ||||||
|  |   SheetDescription, | ||||||
|  | } | ||||||
							
								
								
									
										781
									
								
								src/components/ui/sidebar.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										781
									
								
								src/components/ui/sidebar.tsx
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,781 @@ | |||||||
|  | "use client"; | ||||||
|  | 
 | ||||||
|  | import * as React from "react"; | ||||||
|  | import { Slot } from "@radix-ui/react-slot"; | ||||||
|  | import { VariantProps, cva } from "class-variance-authority"; | ||||||
|  | import { PanelLeft } from "lucide-react"; | ||||||
|  | 
 | ||||||
|  | import { useIsMobile } from "@/hooks/use-mobile"; | ||||||
|  | import { cn } from "@/lib/utils"; | ||||||
|  | import { Button } from "@/components/ui/button"; | ||||||
|  | import { Input } from "@/components/ui/input"; | ||||||
|  | import { Separator } from "@/components/ui/separator"; | ||||||
|  | import { | ||||||
|  |   Sheet, | ||||||
|  |   SheetContent, | ||||||
|  |   SheetDescription, | ||||||
|  |   SheetHeader, | ||||||
|  |   SheetTitle, | ||||||
|  | } from "@/components/ui/sheet"; | ||||||
|  | import { Skeleton } from "@/components/ui/skeleton"; | ||||||
|  | import { | ||||||
|  |   Tooltip, | ||||||
|  |   TooltipContent, | ||||||
|  |   TooltipProvider, | ||||||
|  |   TooltipTrigger, | ||||||
|  | } from "@/components/ui/tooltip"; | ||||||
|  | 
 | ||||||
|  | const SIDEBAR_COOKIE_NAME = "sidebar_state"; | ||||||
|  | const SIDEBAR_COOKIE_MAX_AGE = 60 * 60 * 24 * 7; | ||||||
|  | const SIDEBAR_WIDTH = "16rem"; | ||||||
|  | const SIDEBAR_WIDTH_MOBILE = "18rem"; | ||||||
|  | const SIDEBAR_WIDTH_ICON = "3rem"; | ||||||
|  | const SIDEBAR_KEYBOARD_SHORTCUT = "y"; | ||||||
|  | 
 | ||||||
|  | type SidebarContext = { | ||||||
|  |   state: "expanded" | "collapsed"; | ||||||
|  |   open: boolean; | ||||||
|  |   setOpen: (open: boolean) => void; | ||||||
|  |   openMobile: boolean; | ||||||
|  |   setOpenMobile: (open: boolean) => void; | ||||||
|  |   isMobile: boolean; | ||||||
|  |   toggleSidebar: () => void; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | const SidebarContext = React.createContext<SidebarContext | null>(null); | ||||||
|  | 
 | ||||||
|  | function useSidebar() { | ||||||
|  |   const context = React.useContext(SidebarContext); | ||||||
|  |   if (!context) { | ||||||
|  |     throw new Error("useSidebar must be used within a SidebarProvider."); | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   return context; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | const SidebarProvider = React.forwardRef< | ||||||
|  |   HTMLDivElement, | ||||||
|  |   React.ComponentProps<"div"> & { | ||||||
|  |     defaultOpen?: boolean; | ||||||
|  |     open?: boolean; | ||||||
|  |     onOpenChange?: (open: boolean) => void; | ||||||
|  |   } | ||||||
|  | >( | ||||||
|  |   ( | ||||||
|  |     { | ||||||
|  |       defaultOpen = true, | ||||||
|  |       open: openProp, | ||||||
|  |       onOpenChange: setOpenProp, | ||||||
|  |       className, | ||||||
|  |       style, | ||||||
|  |       children, | ||||||
|  |       ...props | ||||||
|  |     }, | ||||||
|  |     ref, | ||||||
|  |   ) => { | ||||||
|  |     const isMobile = useIsMobile(); | ||||||
|  |     const [openMobile, setOpenMobile] = React.useState(false); | ||||||
|  | 
 | ||||||
|  |     // This is the internal state of the sidebar.
 | ||||||
|  |     // We use openProp and setOpenProp for control from outside the component.
 | ||||||
|  |     const [_open, _setOpen] = React.useState(defaultOpen); | ||||||
|  |     const open = openProp ?? _open; | ||||||
|  |     const setOpen = React.useCallback( | ||||||
|  |       (value: boolean | ((value: boolean) => boolean)) => { | ||||||
|  |         const openState = typeof value === "function" ? value(open) : value; | ||||||
|  |         if (setOpenProp) { | ||||||
|  |           setOpenProp(openState); | ||||||
|  |         } else { | ||||||
|  |           _setOpen(openState); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         // This sets the cookie to keep the sidebar state.
 | ||||||
|  |         document.cookie = `${SIDEBAR_COOKIE_NAME}=${openState}; path=/; max-age=${SIDEBAR_COOKIE_MAX_AGE}`; | ||||||
|  |       }, | ||||||
|  |       [setOpenProp, open], | ||||||
|  |     ); | ||||||
|  | 
 | ||||||
|  |     // Helper to toggle the sidebar.
 | ||||||
|  |     const toggleSidebar = React.useCallback(() => { | ||||||
|  |       return isMobile | ||||||
|  |         ? setOpenMobile((open) => !open) | ||||||
|  |         : setOpen((open) => !open); | ||||||
|  |     }, [isMobile, setOpen, setOpenMobile]); | ||||||
|  | 
 | ||||||
|  |     // Adds a keyboard shortcut to toggle the sidebar.
 | ||||||
|  |     React.useEffect(() => { | ||||||
|  |       const handleKeyDown = (event: KeyboardEvent) => { | ||||||
|  |         if ( | ||||||
|  |           event.key === SIDEBAR_KEYBOARD_SHORTCUT && | ||||||
|  |           (event.metaKey || event.ctrlKey) | ||||||
|  |         ) { | ||||||
|  |           event.preventDefault(); | ||||||
|  |           toggleSidebar(); | ||||||
|  |         } | ||||||
|  |       }; | ||||||
|  | 
 | ||||||
|  |       window.addEventListener("keydown", handleKeyDown); | ||||||
|  |       return () => window.removeEventListener("keydown", handleKeyDown); | ||||||
|  |     }, [toggleSidebar]); | ||||||
|  | 
 | ||||||
|  |     // We add a state so that we can do data-state="expanded" or "collapsed".
 | ||||||
|  |     // This makes it easier to style the sidebar with Tailwind classes.
 | ||||||
|  |     const state = open ? "expanded" : "collapsed"; | ||||||
|  | 
 | ||||||
|  |     const contextValue = React.useMemo<SidebarContext>( | ||||||
|  |       () => ({ | ||||||
|  |         state, | ||||||
|  |         open, | ||||||
|  |         setOpen, | ||||||
|  |         isMobile, | ||||||
|  |         openMobile, | ||||||
|  |         setOpenMobile, | ||||||
|  |         toggleSidebar, | ||||||
|  |       }), | ||||||
|  |       [ | ||||||
|  |         state, | ||||||
|  |         open, | ||||||
|  |         setOpen, | ||||||
|  |         isMobile, | ||||||
|  |         openMobile, | ||||||
|  |         setOpenMobile, | ||||||
|  |         toggleSidebar, | ||||||
|  |       ], | ||||||
|  |     ); | ||||||
|  | 
 | ||||||
|  |     return ( | ||||||
|  |       <SidebarContext.Provider value={contextValue}> | ||||||
|  |         <TooltipProvider delayDuration={0}> | ||||||
|  |           <div | ||||||
|  |             style={ | ||||||
|  |               { | ||||||
|  |                 "--sidebar-width": SIDEBAR_WIDTH, | ||||||
|  |                 "--sidebar-width-icon": SIDEBAR_WIDTH_ICON, | ||||||
|  |                 ...style, | ||||||
|  |               } as React.CSSProperties | ||||||
|  |             } | ||||||
|  |             className={cn( | ||||||
|  |               "group/sidebar-wrapper flex min-h-svh w-full has-[[data-variant=inset]]:bg-sidebar", | ||||||
|  |               className, | ||||||
|  |             )} | ||||||
|  |             ref={ref} | ||||||
|  |             {...props} | ||||||
|  |           > | ||||||
|  |             {children} | ||||||
|  |           </div> | ||||||
|  |         </TooltipProvider> | ||||||
|  |       </SidebarContext.Provider> | ||||||
|  |     ); | ||||||
|  |   }, | ||||||
|  | ); | ||||||
|  | SidebarProvider.displayName = "SidebarProvider"; | ||||||
|  | 
 | ||||||
|  | const Sidebar = React.forwardRef< | ||||||
|  |   HTMLDivElement, | ||||||
|  |   React.ComponentProps<"div"> & { | ||||||
|  |     side?: "left" | "right"; | ||||||
|  |     variant?: "sidebar" | "floating" | "inset"; | ||||||
|  |     collapsible?: "offcanvas" | "icon" | "none"; | ||||||
|  |   } | ||||||
|  | >( | ||||||
|  |   ( | ||||||
|  |     { | ||||||
|  |       side = "left", | ||||||
|  |       variant = "sidebar", | ||||||
|  |       collapsible = "offcanvas", | ||||||
|  |       className, | ||||||
|  |       children, | ||||||
|  |       ...props | ||||||
|  |     }, | ||||||
|  |     ref, | ||||||
|  |   ) => { | ||||||
|  |     const { isMobile, state, openMobile, setOpenMobile } = useSidebar(); | ||||||
|  | 
 | ||||||
|  |     if (collapsible === "none") { | ||||||
|  |       return ( | ||||||
|  |         <div | ||||||
|  |           className={cn( | ||||||
|  |             "flex h-full w-[--sidebar-width] flex-col bg-sidebar text-sidebar-foreground", | ||||||
|  |             className, | ||||||
|  |           )} | ||||||
|  |           ref={ref} | ||||||
|  |           {...props} | ||||||
|  |         > | ||||||
|  |           {children} | ||||||
|  |         </div> | ||||||
|  |       ); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     if (isMobile) { | ||||||
|  |       return ( | ||||||
|  |         <Sheet open={openMobile} onOpenChange={setOpenMobile} {...props}> | ||||||
|  |           <SheetContent | ||||||
|  |             data-sidebar="sidebar" | ||||||
|  |             data-mobile="true" | ||||||
|  |             className="w-[--sidebar-width] bg-sidebar p-0 text-sidebar-foreground [&>button]:hidden" | ||||||
|  |             style={ | ||||||
|  |               { | ||||||
|  |                 "--sidebar-width": SIDEBAR_WIDTH_MOBILE, | ||||||
|  |               } as React.CSSProperties | ||||||
|  |             } | ||||||
|  |             side={side} | ||||||
|  |           > | ||||||
|  |             <SheetHeader className="sr-only"> | ||||||
|  |               <SheetTitle>Sidebar</SheetTitle> | ||||||
|  |               <SheetDescription>Displays the mobile sidebar.</SheetDescription> | ||||||
|  |             </SheetHeader> | ||||||
|  |             <div className="flex h-full w-full flex-col">{children}</div> | ||||||
|  |           </SheetContent> | ||||||
|  |         </Sheet> | ||||||
|  |       ); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     return ( | ||||||
|  |       <div | ||||||
|  |         ref={ref} | ||||||
|  |         className="group peer hidden text-sidebar-foreground md:block" | ||||||
|  |         data-state={state} | ||||||
|  |         data-collapsible={state === "collapsed" ? collapsible : ""} | ||||||
|  |         data-variant={variant} | ||||||
|  |         data-side={side} | ||||||
|  |       > | ||||||
|  |         {/* This is what handles the sidebar gap on desktop */} | ||||||
|  |         <div | ||||||
|  |           className={cn( | ||||||
|  |             "relative w-[--sidebar-width] bg-transparent transition-[width] duration-200 ease-linear", | ||||||
|  |             "group-data-[collapsible=offcanvas]:w-0", | ||||||
|  |             "group-data-[side=right]:rotate-180", | ||||||
|  |             variant === "floating" || variant === "inset" | ||||||
|  |               ? "group-data-[collapsible=icon]:w-[calc(var(--sidebar-width-icon)_+_theme(spacing.4))]" | ||||||
|  |               : "group-data-[collapsible=icon]:w-[--sidebar-width-icon]", | ||||||
|  |           )} | ||||||
|  |         /> | ||||||
|  |         <div | ||||||
|  |           className={cn( | ||||||
|  |             "fixed inset-y-0 z-10 hidden h-svh w-[--sidebar-width] transition-[left,right,width] duration-200 ease-linear md:flex", | ||||||
|  |             side === "left" | ||||||
|  |               ? "left-0 group-data-[collapsible=offcanvas]:left-[calc(var(--sidebar-width)*-1)]" | ||||||
|  |               : "right-0 group-data-[collapsible=offcanvas]:right-[calc(var(--sidebar-width)*-1)]", | ||||||
|  |             // Adjust the padding for floating and inset variants.
 | ||||||
|  |             variant === "floating" || variant === "inset" | ||||||
|  |               ? "p-2 group-data-[collapsible=icon]:w-[calc(var(--sidebar-width-icon)_+_theme(spacing.4)_+2px)]" | ||||||
|  |               : "group-data-[collapsible=icon]:w-[--sidebar-width-icon] group-data-[side=left]:border-r group-data-[side=right]:border-l", | ||||||
|  |             className, | ||||||
|  |           )} | ||||||
|  |           {...props} | ||||||
|  |         > | ||||||
|  |           <div | ||||||
|  |             data-sidebar="sidebar" | ||||||
|  |             className="flex h-full w-full flex-col bg-sidebar group-data-[variant=floating]:rounded-lg group-data-[variant=floating]:border group-data-[variant=floating]:border-sidebar-border group-data-[variant=floating]:shadow" | ||||||
|  |           > | ||||||
|  |             {children} | ||||||
|  |           </div> | ||||||
|  |         </div> | ||||||
|  |       </div> | ||||||
|  |     ); | ||||||
|  |   }, | ||||||
|  | ); | ||||||
|  | Sidebar.displayName = "Sidebar"; | ||||||
|  | 
 | ||||||
|  | const SidebarTrigger = React.forwardRef< | ||||||
|  |   React.ElementRef<typeof Button>, | ||||||
|  |   React.ComponentProps<typeof Button> | ||||||
|  | >(({ className, onClick, ...props }, ref) => { | ||||||
|  |   const { toggleSidebar } = useSidebar(); | ||||||
|  | 
 | ||||||
|  |   return ( | ||||||
|  |     <Button | ||||||
|  |       ref={ref} | ||||||
|  |       data-sidebar="trigger" | ||||||
|  |       variant="ghost" | ||||||
|  |       size="icon" | ||||||
|  |       className={cn("h-7 w-7", className)} | ||||||
|  |       onClick={(event) => { | ||||||
|  |         onClick?.(event); | ||||||
|  |         toggleSidebar(); | ||||||
|  |       }} | ||||||
|  |       {...props} | ||||||
|  |     > | ||||||
|  |       <PanelLeft /> | ||||||
|  |       <span className="sr-only">Toggle Sidebar</span> | ||||||
|  |     </Button> | ||||||
|  |   ); | ||||||
|  | }); | ||||||
|  | SidebarTrigger.displayName = "SidebarTrigger"; | ||||||
|  | 
 | ||||||
|  | const SidebarRail = React.forwardRef< | ||||||
|  |   HTMLButtonElement, | ||||||
|  |   React.ComponentProps<"button"> | ||||||
|  | >(({ className, ...props }, ref) => { | ||||||
|  |   const { toggleSidebar } = useSidebar(); | ||||||
|  | 
 | ||||||
|  |   return ( | ||||||
|  |     <button | ||||||
|  |       ref={ref} | ||||||
|  |       data-sidebar="rail" | ||||||
|  |       aria-label="Toggle Sidebar" | ||||||
|  |       tabIndex={-1} | ||||||
|  |       onClick={toggleSidebar} | ||||||
|  |       title="Toggle Sidebar" | ||||||
|  |       className={cn( | ||||||
|  |         "absolute inset-y-0 z-20 hidden w-4 -translate-x-1/2 transition-all ease-linear after:absolute after:inset-y-0 after:left-1/2 after:w-[2px] hover:after:bg-sidebar-border group-data-[side=left]:-right-4 group-data-[side=right]:left-0 sm:flex", | ||||||
|  |         "[[data-side=left]_&]:cursor-w-resize [[data-side=right]_&]:cursor-e-resize", | ||||||
|  |         "[[data-side=left][data-state=collapsed]_&]:cursor-e-resize [[data-side=right][data-state=collapsed]_&]:cursor-w-resize", | ||||||
|  |         "group-data-[collapsible=offcanvas]:translate-x-0 group-data-[collapsible=offcanvas]:after:left-full group-data-[collapsible=offcanvas]:hover:bg-sidebar", | ||||||
|  |         "[[data-side=left][data-collapsible=offcanvas]_&]:-right-2", | ||||||
|  |         "[[data-side=right][data-collapsible=offcanvas]_&]:-left-2", | ||||||
|  |         className, | ||||||
|  |       )} | ||||||
|  |       {...props} | ||||||
|  |     /> | ||||||
|  |   ); | ||||||
|  | }); | ||||||
|  | SidebarRail.displayName = "SidebarRail"; | ||||||
|  | 
 | ||||||
|  | const SidebarInset = React.forwardRef< | ||||||
|  |   HTMLDivElement, | ||||||
|  |   React.ComponentProps<"main"> | ||||||
|  | >(({ className, ...props }, ref) => { | ||||||
|  |   return ( | ||||||
|  |     <main | ||||||
|  |       ref={ref} | ||||||
|  |       className={cn( | ||||||
|  |         "relative flex w-full flex-1 flex-col bg-background", | ||||||
|  |         "md:peer-data-[variant=inset]:m-2 md:peer-data-[state=collapsed]:peer-data-[variant=inset]:ml-2 md:peer-data-[variant=inset]:ml-0 md:peer-data-[variant=inset]:rounded-xl md:peer-data-[variant=inset]:shadow", | ||||||
|  |         className, | ||||||
|  |       )} | ||||||
|  |       {...props} | ||||||
|  |     /> | ||||||
|  |   ); | ||||||
|  | }); | ||||||
|  | SidebarInset.displayName = "SidebarInset"; | ||||||
|  | 
 | ||||||
|  | const SidebarInput = React.forwardRef< | ||||||
|  |   React.ElementRef<typeof Input>, | ||||||
|  |   React.ComponentProps<typeof Input> | ||||||
|  | >(({ className, ...props }, ref) => { | ||||||
|  |   return ( | ||||||
|  |     <Input | ||||||
|  |       ref={ref} | ||||||
|  |       data-sidebar="input" | ||||||
|  |       className={cn( | ||||||
|  |         "h-8 w-full bg-background shadow-none focus-visible:ring-2 focus-visible:ring-sidebar-ring", | ||||||
|  |         className, | ||||||
|  |       )} | ||||||
|  |       {...props} | ||||||
|  |     /> | ||||||
|  |   ); | ||||||
|  | }); | ||||||
|  | SidebarInput.displayName = "SidebarInput"; | ||||||
|  | 
 | ||||||
|  | const SidebarHeader = React.forwardRef< | ||||||
|  |   HTMLDivElement, | ||||||
|  |   React.ComponentProps<"div"> | ||||||
|  | >(({ className, ...props }, ref) => { | ||||||
|  |   return ( | ||||||
|  |     <div | ||||||
|  |       ref={ref} | ||||||
|  |       data-sidebar="header" | ||||||
|  |       className={cn("flex flex-col gap-2 p-2", className)} | ||||||
|  |       {...props} | ||||||
|  |     /> | ||||||
|  |   ); | ||||||
|  | }); | ||||||
|  | SidebarHeader.displayName = "SidebarHeader"; | ||||||
|  | 
 | ||||||
|  | const SidebarFooter = React.forwardRef< | ||||||
|  |   HTMLDivElement, | ||||||
|  |   React.ComponentProps<"div"> | ||||||
|  | >(({ className, ...props }, ref) => { | ||||||
|  |   return ( | ||||||
|  |     <div | ||||||
|  |       ref={ref} | ||||||
|  |       data-sidebar="footer" | ||||||
|  |       className={cn("flex flex-col gap-2 p-2", className)} | ||||||
|  |       {...props} | ||||||
|  |     /> | ||||||
|  |   ); | ||||||
|  | }); | ||||||
|  | SidebarFooter.displayName = "SidebarFooter"; | ||||||
|  | 
 | ||||||
|  | const SidebarSeparator = React.forwardRef< | ||||||
|  |   React.ElementRef<typeof Separator>, | ||||||
|  |   React.ComponentProps<typeof Separator> | ||||||
|  | >(({ className, ...props }, ref) => { | ||||||
|  |   return ( | ||||||
|  |     <Separator | ||||||
|  |       ref={ref} | ||||||
|  |       data-sidebar="separator" | ||||||
|  |       className={cn("mx-2 w-auto bg-sidebar-border", className)} | ||||||
|  |       {...props} | ||||||
|  |     /> | ||||||
|  |   ); | ||||||
|  | }); | ||||||
|  | SidebarSeparator.displayName = "SidebarSeparator"; | ||||||
|  | 
 | ||||||
|  | const SidebarContent = React.forwardRef< | ||||||
|  |   HTMLDivElement, | ||||||
|  |   React.ComponentProps<"div"> | ||||||
|  | >(({ className, ...props }, ref) => { | ||||||
|  |   return ( | ||||||
|  |     <div | ||||||
|  |       ref={ref} | ||||||
|  |       data-sidebar="content" | ||||||
|  |       className={cn( | ||||||
|  |         "flex min-h-0 flex-1 flex-col gap-2 overflow-auto group-data-[collapsible=icon]:overflow-hidden", | ||||||
|  |         className, | ||||||
|  |       )} | ||||||
|  |       {...props} | ||||||
|  |     /> | ||||||
|  |   ); | ||||||
|  | }); | ||||||
|  | SidebarContent.displayName = "SidebarContent"; | ||||||
|  | 
 | ||||||
|  | const SidebarGroup = React.forwardRef< | ||||||
|  |   HTMLDivElement, | ||||||
|  |   React.ComponentProps<"div"> | ||||||
|  | >(({ className, ...props }, ref) => { | ||||||
|  |   return ( | ||||||
|  |     <div | ||||||
|  |       ref={ref} | ||||||
|  |       data-sidebar="group" | ||||||
|  |       className={cn("relative flex w-full min-w-0 flex-col p-2", className)} | ||||||
|  |       {...props} | ||||||
|  |     /> | ||||||
|  |   ); | ||||||
|  | }); | ||||||
|  | SidebarGroup.displayName = "SidebarGroup"; | ||||||
|  | 
 | ||||||
|  | const SidebarGroupLabel = React.forwardRef< | ||||||
|  |   HTMLDivElement, | ||||||
|  |   React.ComponentProps<"div"> & { asChild?: boolean } | ||||||
|  | >(({ className, asChild = false, ...props }, ref) => { | ||||||
|  |   const Comp = asChild ? Slot : "div"; | ||||||
|  | 
 | ||||||
|  |   return ( | ||||||
|  |     <Comp | ||||||
|  |       ref={ref} | ||||||
|  |       data-sidebar="group-label" | ||||||
|  |       className={cn( | ||||||
|  |         "flex h-8 shrink-0 items-center rounded-md px-2 text-xs font-medium text-sidebar-foreground/70 outline-none ring-sidebar-ring transition-[margin,opacity] duration-200 ease-linear focus-visible:ring-2 [&>svg]:size-4 [&>svg]:shrink-0", | ||||||
|  |         "group-data-[collapsible=icon]:-mt-8 group-data-[collapsible=icon]:opacity-0", | ||||||
|  |         className, | ||||||
|  |       )} | ||||||
|  |       {...props} | ||||||
|  |     /> | ||||||
|  |   ); | ||||||
|  | }); | ||||||
|  | SidebarGroupLabel.displayName = "SidebarGroupLabel"; | ||||||
|  | 
 | ||||||
|  | const SidebarGroupAction = React.forwardRef< | ||||||
|  |   HTMLButtonElement, | ||||||
|  |   React.ComponentProps<"button"> & { asChild?: boolean } | ||||||
|  | >(({ className, asChild = false, ...props }, ref) => { | ||||||
|  |   const Comp = asChild ? Slot : "button"; | ||||||
|  | 
 | ||||||
|  |   return ( | ||||||
|  |     <Comp | ||||||
|  |       ref={ref} | ||||||
|  |       data-sidebar="group-action" | ||||||
|  |       className={cn( | ||||||
|  |         "absolute right-3 top-3.5 flex aspect-square w-5 items-center justify-center rounded-md p-0 text-sidebar-foreground outline-none ring-sidebar-ring transition-transform hover:bg-sidebar-accent hover:text-sidebar-accent-foreground focus-visible:ring-2 [&>svg]:size-4 [&>svg]:shrink-0", | ||||||
|  |         // Increases the hit area of the button on mobile.
 | ||||||
|  |         "after:absolute after:-inset-2 after:md:hidden", | ||||||
|  |         "group-data-[collapsible=icon]:hidden", | ||||||
|  |         className, | ||||||
|  |       )} | ||||||
|  |       {...props} | ||||||
|  |     /> | ||||||
|  |   ); | ||||||
|  | }); | ||||||
|  | SidebarGroupAction.displayName = "SidebarGroupAction"; | ||||||
|  | 
 | ||||||
|  | const SidebarGroupContent = React.forwardRef< | ||||||
|  |   HTMLDivElement, | ||||||
|  |   React.ComponentProps<"div"> | ||||||
|  | >(({ className, ...props }, ref) => ( | ||||||
|  |   <div | ||||||
|  |     ref={ref} | ||||||
|  |     data-sidebar="group-content" | ||||||
|  |     className={cn("w-full text-sm", className)} | ||||||
|  |     {...props} | ||||||
|  |   /> | ||||||
|  | )); | ||||||
|  | SidebarGroupContent.displayName = "SidebarGroupContent"; | ||||||
|  | 
 | ||||||
|  | const SidebarMenu = React.forwardRef< | ||||||
|  |   HTMLUListElement, | ||||||
|  |   React.ComponentProps<"ul"> | ||||||
|  | >(({ className, ...props }, ref) => ( | ||||||
|  |   <ul | ||||||
|  |     ref={ref} | ||||||
|  |     data-sidebar="menu" | ||||||
|  |     className={cn("flex w-full min-w-0 flex-col gap-1", className)} | ||||||
|  |     {...props} | ||||||
|  |   /> | ||||||
|  | )); | ||||||
|  | SidebarMenu.displayName = "SidebarMenu"; | ||||||
|  | 
 | ||||||
|  | const SidebarMenuItem = React.forwardRef< | ||||||
|  |   HTMLLIElement, | ||||||
|  |   React.ComponentProps<"li"> | ||||||
|  | >(({ className, ...props }, ref) => ( | ||||||
|  |   <li | ||||||
|  |     ref={ref} | ||||||
|  |     data-sidebar="menu-item" | ||||||
|  |     className={cn("group/menu-item relative", className)} | ||||||
|  |     {...props} | ||||||
|  |   /> | ||||||
|  | )); | ||||||
|  | SidebarMenuItem.displayName = "SidebarMenuItem"; | ||||||
|  | 
 | ||||||
|  | const sidebarMenuButtonVariants = cva( | ||||||
|  |   "peer/menu-button flex w-full items-center gap-2 overflow-hidden rounded-md p-2 text-left text-sm outline-none ring-sidebar-ring transition-[width,height,padding] hover:bg-sidebar-accent hover:text-sidebar-accent-foreground focus-visible:ring-2 active:bg-sidebar-accent active:text-sidebar-accent-foreground disabled:pointer-events-none disabled:opacity-50 group-has-[[data-sidebar=menu-action]]/menu-item:pr-8 aria-disabled:pointer-events-none aria-disabled:opacity-50 data-[active=true]:bg-sidebar-accent data-[active=true]:font-medium data-[active=true]:text-sidebar-accent-foreground data-[state=open]:hover:bg-sidebar-accent data-[state=open]:hover:text-sidebar-accent-foreground group-data-[collapsible=icon]:!size-8 group-data-[collapsible=icon]:!p-2 [&>span:last-child]:truncate [&>svg]:size-4 [&>svg]:shrink-0", | ||||||
|  |   { | ||||||
|  |     variants: { | ||||||
|  |       variant: { | ||||||
|  |         default: "hover:bg-sidebar-accent hover:text-sidebar-accent-foreground", | ||||||
|  |         outline: | ||||||
|  |           "bg-background shadow-[0_0_0_1px_hsl(var(--sidebar-border))] hover:bg-sidebar-accent hover:text-sidebar-accent-foreground hover:shadow-[0_0_0_1px_hsl(var(--sidebar-accent))]", | ||||||
|  |       }, | ||||||
|  |       size: { | ||||||
|  |         default: "h-8 text-sm", | ||||||
|  |         sm: "h-7 text-xs", | ||||||
|  |         lg: "h-12 text-sm group-data-[collapsible=icon]:!p-0", | ||||||
|  |       }, | ||||||
|  |     }, | ||||||
|  |     defaultVariants: { | ||||||
|  |       variant: "default", | ||||||
|  |       size: "default", | ||||||
|  |     }, | ||||||
|  |   }, | ||||||
|  | ); | ||||||
|  | 
 | ||||||
|  | const SidebarMenuButton = React.forwardRef< | ||||||
|  |   HTMLButtonElement, | ||||||
|  |   React.ComponentProps<"button"> & { | ||||||
|  |     asChild?: boolean; | ||||||
|  |     isActive?: boolean; | ||||||
|  |     tooltip?: string | React.ComponentProps<typeof TooltipContent>; | ||||||
|  |   } & VariantProps<typeof sidebarMenuButtonVariants> | ||||||
|  | >( | ||||||
|  |   ( | ||||||
|  |     { | ||||||
|  |       asChild = false, | ||||||
|  |       isActive = false, | ||||||
|  |       variant = "default", | ||||||
|  |       size = "default", | ||||||
|  |       tooltip, | ||||||
|  |       className, | ||||||
|  |       ...props | ||||||
|  |     }, | ||||||
|  |     ref, | ||||||
|  |   ) => { | ||||||
|  |     const Comp = asChild ? Slot : "button"; | ||||||
|  |     const { isMobile, state } = useSidebar(); | ||||||
|  | 
 | ||||||
|  |     const button = ( | ||||||
|  |       <Comp | ||||||
|  |         ref={ref} | ||||||
|  |         data-sidebar="menu-button" | ||||||
|  |         data-size={size} | ||||||
|  |         data-active={isActive} | ||||||
|  |         className={cn(sidebarMenuButtonVariants({ variant, size }), className)} | ||||||
|  |         {...props} | ||||||
|  |       /> | ||||||
|  |     ); | ||||||
|  | 
 | ||||||
|  |     if (!tooltip) { | ||||||
|  |       return button; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     if (typeof tooltip === "string") { | ||||||
|  |       tooltip = { | ||||||
|  |         children: tooltip, | ||||||
|  |       }; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     return ( | ||||||
|  |       <Tooltip> | ||||||
|  |         <TooltipTrigger asChild>{button}</TooltipTrigger> | ||||||
|  |         <TooltipContent | ||||||
|  |           side="right" | ||||||
|  |           align="center" | ||||||
|  |           hidden={state !== "collapsed" || isMobile} | ||||||
|  |           {...tooltip} | ||||||
|  |         /> | ||||||
|  |       </Tooltip> | ||||||
|  |     ); | ||||||
|  |   }, | ||||||
|  | ); | ||||||
|  | SidebarMenuButton.displayName = "SidebarMenuButton"; | ||||||
|  | 
 | ||||||
|  | const SidebarMenuAction = React.forwardRef< | ||||||
|  |   HTMLButtonElement, | ||||||
|  |   React.ComponentProps<"button"> & { | ||||||
|  |     asChild?: boolean; | ||||||
|  |     showOnHover?: boolean; | ||||||
|  |   } | ||||||
|  | >(({ className, asChild = false, showOnHover = false, ...props }, ref) => { | ||||||
|  |   const Comp = asChild ? Slot : "button"; | ||||||
|  | 
 | ||||||
|  |   return ( | ||||||
|  |     <Comp | ||||||
|  |       ref={ref} | ||||||
|  |       data-sidebar="menu-action" | ||||||
|  |       className={cn( | ||||||
|  |         "absolute right-1 top-1.5 flex aspect-square w-5 items-center justify-center rounded-md p-0 text-sidebar-foreground outline-none ring-sidebar-ring transition-transform hover:bg-sidebar-accent hover:text-sidebar-accent-foreground focus-visible:ring-2 peer-hover/menu-button:text-sidebar-accent-foreground [&>svg]:size-4 [&>svg]:shrink-0", | ||||||
|  |         // Increases the hit area of the button on mobile.
 | ||||||
|  |         "after:absolute after:-inset-2 after:md:hidden", | ||||||
|  |         "peer-data-[size=sm]/menu-button:top-1", | ||||||
|  |         "peer-data-[size=default]/menu-button:top-1.5", | ||||||
|  |         "peer-data-[size=lg]/menu-button:top-2.5", | ||||||
|  |         "group-data-[collapsible=icon]:hidden", | ||||||
|  |         showOnHover && | ||||||
|  |           "group-focus-within/menu-item:opacity-100 group-hover/menu-item:opacity-100 data-[state=open]:opacity-100 peer-data-[active=true]/menu-button:text-sidebar-accent-foreground md:opacity-0", | ||||||
|  |         className, | ||||||
|  |       )} | ||||||
|  |       {...props} | ||||||
|  |     /> | ||||||
|  |   ); | ||||||
|  | }); | ||||||
|  | SidebarMenuAction.displayName = "SidebarMenuAction"; | ||||||
|  | 
 | ||||||
|  | const SidebarMenuBadge = React.forwardRef< | ||||||
|  |   HTMLDivElement, | ||||||
|  |   React.ComponentProps<"div"> | ||||||
|  | >(({ className, ...props }, ref) => ( | ||||||
|  |   <div | ||||||
|  |     ref={ref} | ||||||
|  |     data-sidebar="menu-badge" | ||||||
|  |     className={cn( | ||||||
|  |       "pointer-events-none absolute right-1 flex h-5 min-w-5 select-none items-center justify-center rounded-md px-1 text-xs font-medium tabular-nums text-sidebar-foreground", | ||||||
|  |       "peer-hover/menu-button:text-sidebar-accent-foreground peer-data-[active=true]/menu-button:text-sidebar-accent-foreground", | ||||||
|  |       "peer-data-[size=sm]/menu-button:top-1", | ||||||
|  |       "peer-data-[size=default]/menu-button:top-1.5", | ||||||
|  |       "peer-data-[size=lg]/menu-button:top-2.5", | ||||||
|  |       "group-data-[collapsible=icon]:hidden", | ||||||
|  |       className, | ||||||
|  |     )} | ||||||
|  |     {...props} | ||||||
|  |   /> | ||||||
|  | )); | ||||||
|  | SidebarMenuBadge.displayName = "SidebarMenuBadge"; | ||||||
|  | 
 | ||||||
|  | const SidebarMenuSkeleton = React.forwardRef< | ||||||
|  |   HTMLDivElement, | ||||||
|  |   React.ComponentProps<"div"> & { | ||||||
|  |     showIcon?: boolean; | ||||||
|  |   } | ||||||
|  | >(({ className, showIcon = false, ...props }, ref) => { | ||||||
|  |   // Random width between 50 to 90%.
 | ||||||
|  |   const width = React.useMemo(() => { | ||||||
|  |     return `${Math.floor(Math.random() * 40) + 50}%`; | ||||||
|  |   }, []); | ||||||
|  | 
 | ||||||
|  |   return ( | ||||||
|  |     <div | ||||||
|  |       ref={ref} | ||||||
|  |       data-sidebar="menu-skeleton" | ||||||
|  |       className={cn("flex h-8 items-center gap-2 rounded-md px-2", className)} | ||||||
|  |       {...props} | ||||||
|  |     > | ||||||
|  |       {showIcon && ( | ||||||
|  |         <Skeleton | ||||||
|  |           className="size-4 rounded-md" | ||||||
|  |           data-sidebar="menu-skeleton-icon" | ||||||
|  |         /> | ||||||
|  |       )} | ||||||
|  |       <Skeleton | ||||||
|  |         className="h-4 max-w-[--skeleton-width] flex-1" | ||||||
|  |         data-sidebar="menu-skeleton-text" | ||||||
|  |         style={ | ||||||
|  |           { | ||||||
|  |             "--skeleton-width": width, | ||||||
|  |           } as React.CSSProperties | ||||||
|  |         } | ||||||
|  |       /> | ||||||
|  |     </div> | ||||||
|  |   ); | ||||||
|  | }); | ||||||
|  | SidebarMenuSkeleton.displayName = "SidebarMenuSkeleton"; | ||||||
|  | 
 | ||||||
|  | const SidebarMenuSub = React.forwardRef< | ||||||
|  |   HTMLUListElement, | ||||||
|  |   React.ComponentProps<"ul"> | ||||||
|  | >(({ className, ...props }, ref) => ( | ||||||
|  |   <ul | ||||||
|  |     ref={ref} | ||||||
|  |     data-sidebar="menu-sub" | ||||||
|  |     className={cn( | ||||||
|  |       "mx-3.5 flex min-w-0 translate-x-px flex-col gap-1 border-l border-sidebar-border px-2.5 py-0.5", | ||||||
|  |       "group-data-[collapsible=icon]:hidden", | ||||||
|  |       className, | ||||||
|  |     )} | ||||||
|  |     {...props} | ||||||
|  |   /> | ||||||
|  | )); | ||||||
|  | SidebarMenuSub.displayName = "SidebarMenuSub"; | ||||||
|  | 
 | ||||||
|  | const SidebarMenuSubItem = React.forwardRef< | ||||||
|  |   HTMLLIElement, | ||||||
|  |   React.ComponentProps<"li"> | ||||||
|  | >(({ ...props }, ref) => <li ref={ref} {...props} />); | ||||||
|  | SidebarMenuSubItem.displayName = "SidebarMenuSubItem"; | ||||||
|  | 
 | ||||||
|  | const SidebarMenuSubButton = React.forwardRef< | ||||||
|  |   HTMLAnchorElement, | ||||||
|  |   React.ComponentProps<"a"> & { | ||||||
|  |     asChild?: boolean; | ||||||
|  |     size?: "sm" | "md"; | ||||||
|  |     isActive?: boolean; | ||||||
|  |   } | ||||||
|  | >(({ asChild = false, size = "md", isActive, className, ...props }, ref) => { | ||||||
|  |   const Comp = asChild ? Slot : "a"; | ||||||
|  | 
 | ||||||
|  |   return ( | ||||||
|  |     <Comp | ||||||
|  |       ref={ref} | ||||||
|  |       data-sidebar="menu-sub-button" | ||||||
|  |       data-size={size} | ||||||
|  |       data-active={isActive} | ||||||
|  |       className={cn( | ||||||
|  |         "flex h-7 min-w-0 -translate-x-px items-center gap-2 overflow-hidden rounded-md px-2 text-sidebar-foreground outline-none ring-sidebar-ring hover:bg-sidebar-accent hover:text-sidebar-accent-foreground focus-visible:ring-2 active:bg-sidebar-accent active:text-sidebar-accent-foreground disabled:pointer-events-none disabled:opacity-50 aria-disabled:pointer-events-none aria-disabled:opacity-50 [&>span:last-child]:truncate [&>svg]:size-4 [&>svg]:shrink-0 [&>svg]:text-sidebar-accent-foreground", | ||||||
|  |         "data-[active=true]:bg-sidebar-accent data-[active=true]:text-sidebar-accent-foreground", | ||||||
|  |         size === "sm" && "text-xs", | ||||||
|  |         size === "md" && "text-sm", | ||||||
|  |         "group-data-[collapsible=icon]:hidden", | ||||||
|  |         className, | ||||||
|  |       )} | ||||||
|  |       {...props} | ||||||
|  |     /> | ||||||
|  |   ); | ||||||
|  | }); | ||||||
|  | SidebarMenuSubButton.displayName = "SidebarMenuSubButton"; | ||||||
|  | 
 | ||||||
|  | export { | ||||||
|  |   Sidebar, | ||||||
|  |   SidebarContent, | ||||||
|  |   SidebarFooter, | ||||||
|  |   SidebarGroup, | ||||||
|  |   SidebarGroupAction, | ||||||
|  |   SidebarGroupContent, | ||||||
|  |   SidebarGroupLabel, | ||||||
|  |   SidebarHeader, | ||||||
|  |   SidebarInput, | ||||||
|  |   SidebarInset, | ||||||
|  |   SidebarMenu, | ||||||
|  |   SidebarMenuAction, | ||||||
|  |   SidebarMenuBadge, | ||||||
|  |   SidebarMenuButton, | ||||||
|  |   SidebarMenuItem, | ||||||
|  |   SidebarMenuSkeleton, | ||||||
|  |   SidebarMenuSub, | ||||||
|  |   SidebarMenuSubButton, | ||||||
|  |   SidebarMenuSubItem, | ||||||
|  |   SidebarProvider, | ||||||
|  |   SidebarRail, | ||||||
|  |   SidebarSeparator, | ||||||
|  |   SidebarTrigger, | ||||||
|  |   useSidebar, | ||||||
|  | }; | ||||||
							
								
								
									
										15
									
								
								src/components/ui/skeleton.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										15
									
								
								src/components/ui/skeleton.tsx
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,15 @@ | |||||||
|  | import { cn } from "@/lib/utils" | ||||||
|  | 
 | ||||||
|  | function Skeleton({ | ||||||
|  |   className, | ||||||
|  |   ...props | ||||||
|  | }: React.HTMLAttributes<HTMLDivElement>) { | ||||||
|  |   return ( | ||||||
|  |     <div | ||||||
|  |       className={cn("animate-pulse rounded-md bg-primary/10", className)} | ||||||
|  |       {...props} | ||||||
|  |     /> | ||||||
|  |   ) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export { Skeleton } | ||||||
							
								
								
									
										32
									
								
								src/components/ui/tooltip.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										32
									
								
								src/components/ui/tooltip.tsx
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,32 @@ | |||||||
|  | "use client" | ||||||
|  | 
 | ||||||
|  | import * as React from "react" | ||||||
|  | import * as TooltipPrimitive from "@radix-ui/react-tooltip" | ||||||
|  | 
 | ||||||
|  | import { cn } from "@/lib/utils" | ||||||
|  | 
 | ||||||
|  | const TooltipProvider = TooltipPrimitive.Provider | ||||||
|  | 
 | ||||||
|  | const Tooltip = TooltipPrimitive.Root | ||||||
|  | 
 | ||||||
|  | const TooltipTrigger = TooltipPrimitive.Trigger | ||||||
|  | 
 | ||||||
|  | const TooltipContent = React.forwardRef< | ||||||
|  |   React.ElementRef<typeof TooltipPrimitive.Content>, | ||||||
|  |   React.ComponentPropsWithoutRef<typeof TooltipPrimitive.Content> | ||||||
|  | >(({ className, sideOffset = 4, ...props }, ref) => ( | ||||||
|  |   <TooltipPrimitive.Portal> | ||||||
|  |     <TooltipPrimitive.Content | ||||||
|  |       ref={ref} | ||||||
|  |       sideOffset={sideOffset} | ||||||
|  |       className={cn( | ||||||
|  |         "z-50 overflow-hidden rounded-md bg-primary px-3 py-1.5 text-xs text-primary-foreground animate-in fade-in-0 zoom-in-95 data-[state=closed]:animate-out data-[state=closed]:fade-out-0 data-[state=closed]:zoom-out-95 data-[side=bottom]:slide-in-from-top-2 data-[side=left]:slide-in-from-right-2 data-[side=right]:slide-in-from-left-2 data-[side=top]:slide-in-from-bottom-2", | ||||||
|  |         className | ||||||
|  |       )} | ||||||
|  |       {...props} | ||||||
|  |     /> | ||||||
|  |   </TooltipPrimitive.Portal> | ||||||
|  | )) | ||||||
|  | TooltipContent.displayName = TooltipPrimitive.Content.displayName | ||||||
|  | 
 | ||||||
|  | export { Tooltip, TooltipTrigger, TooltipContent, TooltipProvider } | ||||||
							
								
								
									
										52
									
								
								src/env.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										52
									
								
								src/env.js
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,52 @@ | |||||||
|  | import { createEnv } from "@t3-oss/env-nextjs"; | ||||||
|  | import { z } from "zod"; | ||||||
|  | 
 | ||||||
|  | export const env = createEnv({ | ||||||
|  |   /** | ||||||
|  |    * Specify your server-side environment variables schema here. This way you can ensure the app | ||||||
|  |    * isn't built with invalid env vars. | ||||||
|  |    */ | ||||||
|  |   server: { | ||||||
|  |     AUTH_SECRET: | ||||||
|  |       process.env.NODE_ENV === "production" | ||||||
|  |         ? z.string() | ||||||
|  |         : z.string().optional(), | ||||||
|  |     AUTH_DISCORD_ID: z.string(), | ||||||
|  |     AUTH_DISCORD_SECRET: z.string(), | ||||||
|  |     DATABASE_URL: z.string().url(), | ||||||
|  |     NODE_ENV: z | ||||||
|  |       .enum(["development", "test", "production"]) | ||||||
|  |       .default("development"), | ||||||
|  |   }, | ||||||
|  | 
 | ||||||
|  |   /** | ||||||
|  |    * Specify your client-side environment variables schema here. This way you can ensure the app | ||||||
|  |    * isn't built with invalid env vars. To expose them to the client, prefix them with | ||||||
|  |    * `NEXT_PUBLIC_`. | ||||||
|  |    */ | ||||||
|  |   client: { | ||||||
|  |     // NEXT_PUBLIC_CLIENTVAR: z.string(),
 | ||||||
|  |   }, | ||||||
|  | 
 | ||||||
|  |   /** | ||||||
|  |    * You can't destruct `process.env` as a regular object in the Next.js edge runtimes (e.g. | ||||||
|  |    * middlewares) or client-side so we need to destruct manually. | ||||||
|  |    */ | ||||||
|  |   runtimeEnv: { | ||||||
|  |     AUTH_SECRET: process.env.AUTH_SECRET, | ||||||
|  |     AUTH_DISCORD_ID: process.env.AUTH_DISCORD_ID, | ||||||
|  |     AUTH_DISCORD_SECRET: process.env.AUTH_DISCORD_SECRET, | ||||||
|  |     DATABASE_URL: process.env.DATABASE_URL, | ||||||
|  |     NODE_ENV: process.env.NODE_ENV, | ||||||
|  |   }, | ||||||
|  |   /** | ||||||
|  |    * Run `build` or `dev` with `SKIP_ENV_VALIDATION` to skip env validation. This is especially | ||||||
|  |    * useful for Docker builds. | ||||||
|  |    */ | ||||||
|  |   skipValidation: !!process.env.SKIP_ENV_VALIDATION, | ||||||
|  |   /** | ||||||
|  |    * Makes it so that empty strings are treated as undefined. `SOME_VAR: z.string()` and | ||||||
|  |    * `SOME_VAR=''` will throw an error. | ||||||
|  |    */ | ||||||
|  |   emptyStringAsUndefined: true, | ||||||
|  | }); | ||||||
							
								
								
									
										19
									
								
								src/hooks/use-mobile.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										19
									
								
								src/hooks/use-mobile.tsx
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,19 @@ | |||||||
|  | import * as React from "react" | ||||||
|  | 
 | ||||||
|  | const MOBILE_BREAKPOINT = 768 | ||||||
|  | 
 | ||||||
|  | export function useIsMobile() { | ||||||
|  |   const [isMobile, setIsMobile] = React.useState<boolean | undefined>(undefined) | ||||||
|  | 
 | ||||||
|  |   React.useEffect(() => { | ||||||
|  |     const mql = window.matchMedia(`(max-width: ${MOBILE_BREAKPOINT - 1}px)`) | ||||||
|  |     const onChange = () => { | ||||||
|  |       setIsMobile(window.innerWidth < MOBILE_BREAKPOINT) | ||||||
|  |     } | ||||||
|  |     mql.addEventListener("change", onChange) | ||||||
|  |     setIsMobile(window.innerWidth < MOBILE_BREAKPOINT) | ||||||
|  |     return () => mql.removeEventListener("change", onChange) | ||||||
|  |   }, []) | ||||||
|  | 
 | ||||||
|  |   return !!isMobile | ||||||
|  | } | ||||||
							
								
								
									
										22
									
								
								src/lib/utils.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										22
									
								
								src/lib/utils.ts
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,22 @@ | |||||||
|  | import { clsx, type ClassValue } from "clsx"; | ||||||
|  | import { twMerge } from "tailwind-merge"; | ||||||
|  | 
 | ||||||
|  | export function cn(...inputs: ClassValue[]) { | ||||||
|  |   return twMerge(clsx(inputs)); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export function generateSlug(title: string) { | ||||||
|  |   return title.toLowerCase().replace(/\s+/g, "-"); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export function debounce<T extends (...args: any[]) => void>( | ||||||
|  |   func: T, | ||||||
|  |   delay: number, | ||||||
|  | ): (...args: Parameters<T>) => void { | ||||||
|  |   let timeoutId: ReturnType<typeof setTimeout>; | ||||||
|  | 
 | ||||||
|  |   return (...args: Parameters<T>) => { | ||||||
|  |     clearTimeout(timeoutId); | ||||||
|  |     timeoutId = setTimeout(() => func(...args), delay); | ||||||
|  |   }; | ||||||
|  | } | ||||||
							
								
								
									
										16
									
								
								src/lib/validation/has-permission.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										16
									
								
								src/lib/validation/has-permission.tsx
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,16 @@ | |||||||
|  | /* Bitmap Based Roles | ||||||
|  |  *  Admin and editor role === 6 | ||||||
|  |  *  User role === 1 | ||||||
|  |  *  Editor role === 2 | ||||||
|  |  *  Admin role === 4 | ||||||
|  |  *  Highest role is 7 | ||||||
|  |  */ | ||||||
|  | export enum Role { | ||||||
|  |   USER = 1, // 001
 | ||||||
|  |   EDITOR = 2, // 010
 | ||||||
|  |   ADMIN = 4, // 100
 | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | export function hasPermission(userRole: number, requiredRole: Role): boolean { | ||||||
|  |   return (userRole & requiredRole) !== 0; | ||||||
|  | } | ||||||
							
								
								
									
										7
									
								
								src/lib/validation/zod/article.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										7
									
								
								src/lib/validation/zod/article.ts
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,7 @@ | |||||||
|  | import { z } from "zod"; | ||||||
|  | 
 | ||||||
|  | export const articleSchema = z.object({ | ||||||
|  |   title: z.string().min(1), | ||||||
|  |   content: z.string().optional(), | ||||||
|  |   authorId: z.string().optional(), | ||||||
|  | }); | ||||||
							
								
								
									
										5
									
								
								src/lib/validation/zod/category.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										5
									
								
								src/lib/validation/zod/category.ts
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,5 @@ | |||||||
|  | import { z } from "zod"; | ||||||
|  | 
 | ||||||
|  | export const categorySchema = z.object({ | ||||||
|  |   name: z.string().min(1), | ||||||
|  | }); | ||||||
							
								
								
									
										32
									
								
								src/server/actions/article.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										32
									
								
								src/server/actions/article.ts
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,32 @@ | |||||||
|  | "use server"; | ||||||
|  | 
 | ||||||
|  | import { articleSchema } from "@/lib/validation/zod/article"; | ||||||
|  | import { api } from "@/trpc/server"; | ||||||
|  | import { revalidatePath } from "next/cache"; | ||||||
|  | import { redirect } from "next/navigation"; | ||||||
|  | import { z } from "zod"; | ||||||
|  | 
 | ||||||
|  | export async function createArticle(article: z.infer<typeof articleSchema>) { | ||||||
|  |   const result = await api.article.create({ | ||||||
|  |     article, | ||||||
|  |   }); | ||||||
|  |   if (!result[0]?.slug?.length) return false; | ||||||
|  |   return redirect(`/artikel/${result[0].slug}/edit`); | ||||||
|  | } | ||||||
|  | export async function updateArticle( | ||||||
|  |   article: z.infer<typeof articleSchema>, | ||||||
|  |   articleId: string, | ||||||
|  | ) { | ||||||
|  |   const result = await api.article.update({ | ||||||
|  |     article, | ||||||
|  |     articleId, | ||||||
|  |   }); | ||||||
|  |   //   if (!result[0]?.id?.length) return false;
 | ||||||
|  |   //   return revalidatePath(`/artikel/${result[0].id}/edit`);
 | ||||||
|  | } | ||||||
|  | export async function deleteArticle(articleId: string) { | ||||||
|  |   const result = await api.article.delete({ | ||||||
|  |     articleId, | ||||||
|  |   }); | ||||||
|  |   // if (!result[0]?.id?.length) return false;
 | ||||||
|  | } | ||||||
							
								
								
									
										31
									
								
								src/server/actions/category.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										31
									
								
								src/server/actions/category.ts
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,31 @@ | |||||||
|  | "use server"; | ||||||
|  | 
 | ||||||
|  | import { categorySchema } from "@/lib/validation/zod/category"; | ||||||
|  | import { api } from "@/trpc/server"; | ||||||
|  | import { revalidatePath } from "next/cache"; | ||||||
|  | import { redirect } from "next/navigation"; | ||||||
|  | import { z } from "zod"; | ||||||
|  | 
 | ||||||
|  | export async function createCategory(category: z.infer<typeof categorySchema>) { | ||||||
|  |   const result = await api.category.create({ | ||||||
|  |     category, | ||||||
|  |   }); | ||||||
|  |   if (!result[0]?.slug?.length) return false; | ||||||
|  |   return redirect(`/kategorie/${result[0].slug}/edit`); | ||||||
|  | } | ||||||
|  | export async function updateCategory( | ||||||
|  |   category: z.infer<typeof categorySchema>, | ||||||
|  |   categoryId: string, | ||||||
|  | ) { | ||||||
|  |   const result = await api.category.update({ | ||||||
|  |     category, | ||||||
|  |     categoryId, | ||||||
|  |   }); | ||||||
|  |   //   if (!result[0]?.id?.length) return false;
 | ||||||
|  |   //   return revalidatePath(`/artikel/${result[0].id}/edit`);
 | ||||||
|  | } | ||||||
|  | export async function deleteCategory(categoryId: string) { | ||||||
|  |   const result = await api.category.delete({ | ||||||
|  |     categoryId, | ||||||
|  |   }); | ||||||
|  | } | ||||||
							
								
								
									
										25
									
								
								src/server/api/root.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										25
									
								
								src/server/api/root.ts
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,25 @@ | |||||||
|  | import { articleRouter } from "./routers/article"; | ||||||
|  | import { categoryRouter } from "@/server/api/routers/category"; | ||||||
|  | import { createCallerFactory, createTRPCRouter } from "@/server/api/trpc"; | ||||||
|  | 
 | ||||||
|  | /** | ||||||
|  |  * This is the primary router for your server. | ||||||
|  |  * | ||||||
|  |  * All routers added in /api/routers should be manually added here. | ||||||
|  |  */ | ||||||
|  | export const appRouter = createTRPCRouter({ | ||||||
|  |   article: articleRouter, | ||||||
|  |   category: categoryRouter, | ||||||
|  | }); | ||||||
|  | 
 | ||||||
|  | // export type definition of API
 | ||||||
|  | export type AppRouter = typeof appRouter; | ||||||
|  | 
 | ||||||
|  | /** | ||||||
|  |  * Create a server-side caller for the tRPC API. | ||||||
|  |  * @example | ||||||
|  |  * const trpc = createCaller(createContext); | ||||||
|  |  * const res = await trpc.post.all(); | ||||||
|  |  *       ^? Post[] | ||||||
|  |  */ | ||||||
|  | export const createCaller = createCallerFactory(appRouter); | ||||||
							
								
								
									
										90
									
								
								src/server/api/routers/article.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										90
									
								
								src/server/api/routers/article.ts
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,90 @@ | |||||||
|  | import { z } from "zod"; | ||||||
|  | 
 | ||||||
|  | import { | ||||||
|  |   createTRPCRouter, | ||||||
|  |   protectedProcedure, | ||||||
|  |   publicProcedure, | ||||||
|  | } from "@/server/api/trpc"; | ||||||
|  | import { articles } from "@/server/db/schema"; | ||||||
|  | import { articleSchema } from "@/lib/validation/zod/article"; | ||||||
|  | import { eq } from "drizzle-orm"; | ||||||
|  | import { hasPermission, Role } from "@/lib/validation/has-permission"; | ||||||
|  | import { generateSlug } from "@/lib/utils"; | ||||||
|  | 
 | ||||||
|  | export const articleRouter = createTRPCRouter({ | ||||||
|  |   get: publicProcedure | ||||||
|  |     .input(z.object({ slug: z.string() })) | ||||||
|  |     .query(async ({ ctx, input }) => { | ||||||
|  |       return await ctx.db.query.articles.findFirst({ | ||||||
|  |         where: eq(articles.slug, input.slug), | ||||||
|  |       }); | ||||||
|  |     }), | ||||||
|  | 
 | ||||||
|  |   getAll: publicProcedure | ||||||
|  |     .input(z.object({ categoryId: z.string() }).optional()) | ||||||
|  |     .query(async ({ ctx, input }) => { | ||||||
|  |       return await ctx.db.query.articles.findMany({ | ||||||
|  |         where: input?.categoryId | ||||||
|  |           ? eq(articles.categoryId, input.categoryId) | ||||||
|  |           : undefined, | ||||||
|  |       }); | ||||||
|  |     }), | ||||||
|  |   getAllPreviews: publicProcedure | ||||||
|  |     .input(z.object({ categoryId: z.string() }).optional()) | ||||||
|  |     .query(async ({ ctx, input }) => { | ||||||
|  |       return await ctx.db.query.articles.findMany({ | ||||||
|  |         where: input?.categoryId | ||||||
|  |           ? eq(articles.categoryId, input.categoryId) | ||||||
|  |           : undefined, | ||||||
|  |         columns: { | ||||||
|  |           title: true, | ||||||
|  |           slug: true, | ||||||
|  |           createdAt: true, | ||||||
|  |         }, | ||||||
|  |       }); | ||||||
|  |     }), | ||||||
|  |   create: protectedProcedure | ||||||
|  |     .input(z.object({ article: articleSchema })) | ||||||
|  |     .mutation(async ({ ctx, input }) => { | ||||||
|  |       const isEditor = hasPermission(ctx.session.user.role, Role.EDITOR); | ||||||
|  |       if (!isEditor) { | ||||||
|  |         throw new Error("You are not allowed to create articles"); | ||||||
|  |       } | ||||||
|  |       const slug = generateSlug(input.article.title); | ||||||
|  |       return await ctx.db | ||||||
|  |         .insert(articles) | ||||||
|  |         .values({ ...input.article, slug }) | ||||||
|  |         .returning({ | ||||||
|  |           slug: articles.slug, | ||||||
|  |         }); | ||||||
|  |     }), | ||||||
|  |   update: protectedProcedure | ||||||
|  |     .input(z.object({ article: articleSchema, articleId: z.string() })) | ||||||
|  |     .mutation(async ({ ctx, input }) => { | ||||||
|  |       const isEditor = hasPermission(ctx.session.user.role, Role.EDITOR); | ||||||
|  |       if (!isEditor) { | ||||||
|  |         throw new Error("You are not allowed to update articles"); | ||||||
|  |       } | ||||||
|  |       return await ctx.db | ||||||
|  |         .update(articles) | ||||||
|  |         .set(input.article) | ||||||
|  |         .where(eq(articles.id, input.articleId)) | ||||||
|  |         .returning({ | ||||||
|  |           id: articles.id, | ||||||
|  |         }); | ||||||
|  |     }), | ||||||
|  |   delete: protectedProcedure | ||||||
|  |     .input(z.object({ articleId: z.string() })) | ||||||
|  |     .mutation(async ({ ctx, input }) => { | ||||||
|  |       const isEditor = hasPermission(ctx.session.user.role, Role.EDITOR); | ||||||
|  |       if (!isEditor) { | ||||||
|  |         throw new Error("You are not allowed to delete articles"); | ||||||
|  |       } | ||||||
|  |       return await ctx.db | ||||||
|  |         .delete(articles) | ||||||
|  |         .where(eq(articles.id, input.articleId)) | ||||||
|  |         .returning({ | ||||||
|  |           id: articles.id, | ||||||
|  |         }); | ||||||
|  |     }), | ||||||
|  | }); | ||||||
							
								
								
									
										71
									
								
								src/server/api/routers/category.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										71
									
								
								src/server/api/routers/category.ts
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,71 @@ | |||||||
|  | import { z } from "zod"; | ||||||
|  | import { | ||||||
|  |   createTRPCRouter, | ||||||
|  |   protectedProcedure, | ||||||
|  |   publicProcedure, | ||||||
|  | } from "@/server/api/trpc"; | ||||||
|  | import { categories } from "@/server/db/schema"; | ||||||
|  | 
 | ||||||
|  | import { eq } from "drizzle-orm"; | ||||||
|  | import { hasPermission, Role } from "@/lib/validation/has-permission"; | ||||||
|  | import { categorySchema } from "@/lib/validation/zod/category"; | ||||||
|  | import { generateSlug } from "@/lib/utils"; | ||||||
|  | 
 | ||||||
|  | export const categoryRouter = createTRPCRouter({ | ||||||
|  |   get: publicProcedure | ||||||
|  |     .input(z.object({ slug: z.string() })) | ||||||
|  |     .query(async ({ ctx, input }) => { | ||||||
|  |       return await ctx.db.query.categories.findFirst({ | ||||||
|  |         where: eq(categories.slug, input.slug), | ||||||
|  |       }); | ||||||
|  |     }), | ||||||
|  | 
 | ||||||
|  |   getAll: publicProcedure.query(async ({ ctx }) => { | ||||||
|  |     return await ctx.db.query.categories.findMany(); | ||||||
|  |   }), | ||||||
|  | 
 | ||||||
|  |   create: protectedProcedure | ||||||
|  |     .input(z.object({ category: categorySchema })) | ||||||
|  |     .mutation(async ({ ctx, input }) => { | ||||||
|  |       const isEditor = hasPermission(ctx.session.user.role, Role.EDITOR); | ||||||
|  |       if (!isEditor) { | ||||||
|  |         throw new Error("You are not allowed to create categories"); | ||||||
|  |       } | ||||||
|  |       const slug = generateSlug(input.category.name); | ||||||
|  |       return await ctx.db | ||||||
|  |         .insert(categories) | ||||||
|  |         .values({ ...input.category, slug }) | ||||||
|  |         .returning({ | ||||||
|  |           slug: categories.slug, | ||||||
|  |         }); | ||||||
|  |     }), | ||||||
|  |   update: protectedProcedure | ||||||
|  |     .input(z.object({ category: categorySchema, categoryId: z.string() })) | ||||||
|  |     .mutation(async ({ ctx, input }) => { | ||||||
|  |       const isEditor = hasPermission(ctx.session.user.role, Role.EDITOR); | ||||||
|  |       if (!isEditor) { | ||||||
|  |         throw new Error("You are not allowed to update categories"); | ||||||
|  |       } | ||||||
|  |       return await ctx.db | ||||||
|  |         .update(categories) | ||||||
|  |         .set(input.category) | ||||||
|  |         .where(eq(categories.id, input.categoryId)) | ||||||
|  |         .returning({ | ||||||
|  |           id: categories.id, | ||||||
|  |         }); | ||||||
|  |     }), | ||||||
|  |   delete: protectedProcedure | ||||||
|  |     .input(z.object({ categoryId: z.string() })) | ||||||
|  |     .mutation(async ({ ctx, input }) => { | ||||||
|  |       const isEditor = hasPermission(ctx.session.user.role, Role.EDITOR); | ||||||
|  |       if (!isEditor) { | ||||||
|  |         throw new Error("You are not allowed to delete articles"); | ||||||
|  |       } | ||||||
|  |       return await ctx.db | ||||||
|  |         .delete(categories) | ||||||
|  |         .where(eq(categories.id, input.categoryId)) | ||||||
|  |         .returning({ | ||||||
|  |           id: categories.id, | ||||||
|  |         }); | ||||||
|  |     }), | ||||||
|  | }); | ||||||
							
								
								
									
										133
									
								
								src/server/api/trpc.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										133
									
								
								src/server/api/trpc.ts
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,133 @@ | |||||||
|  | /** | ||||||
|  |  * YOU PROBABLY DON'T NEED TO EDIT THIS FILE, UNLESS: | ||||||
|  |  * 1. You want to modify request context (see Part 1). | ||||||
|  |  * 2. You want to create a new middleware or type of procedure (see Part 3). | ||||||
|  |  * | ||||||
|  |  * TL;DR - This is where all the tRPC server stuff is created and plugged in. The pieces you will | ||||||
|  |  * need to use are documented accordingly near the end. | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | import { initTRPC, TRPCError } from "@trpc/server"; | ||||||
|  | import superjson from "superjson"; | ||||||
|  | import { ZodError } from "zod"; | ||||||
|  | 
 | ||||||
|  | import { auth } from "@/server/auth"; | ||||||
|  | import { db } from "@/server/db"; | ||||||
|  | 
 | ||||||
|  | /** | ||||||
|  |  * 1. CONTEXT | ||||||
|  |  * | ||||||
|  |  * This section defines the "contexts" that are available in the backend API. | ||||||
|  |  * | ||||||
|  |  * These allow you to access things when processing a request, like the database, the session, etc. | ||||||
|  |  * | ||||||
|  |  * This helper generates the "internals" for a tRPC context. The API handler and RSC clients each | ||||||
|  |  * wrap this and provides the required context. | ||||||
|  |  * | ||||||
|  |  * @see https://trpc.io/docs/server/context
 | ||||||
|  |  */ | ||||||
|  | export const createTRPCContext = async (opts: { headers: Headers }) => { | ||||||
|  |   const session = await auth(); | ||||||
|  | 
 | ||||||
|  |   return { | ||||||
|  |     db, | ||||||
|  |     session, | ||||||
|  |     ...opts, | ||||||
|  |   }; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | /** | ||||||
|  |  * 2. INITIALIZATION | ||||||
|  |  * | ||||||
|  |  * This is where the tRPC API is initialized, connecting the context and transformer. We also parse | ||||||
|  |  * ZodErrors so that you get typesafety on the frontend if your procedure fails due to validation | ||||||
|  |  * errors on the backend. | ||||||
|  |  */ | ||||||
|  | const t = initTRPC.context<typeof createTRPCContext>().create({ | ||||||
|  |   transformer: superjson, | ||||||
|  |   errorFormatter({ shape, error }) { | ||||||
|  |     return { | ||||||
|  |       ...shape, | ||||||
|  |       data: { | ||||||
|  |         ...shape.data, | ||||||
|  |         zodError: | ||||||
|  |           error.cause instanceof ZodError ? error.cause.flatten() : null, | ||||||
|  |       }, | ||||||
|  |     }; | ||||||
|  |   }, | ||||||
|  | }); | ||||||
|  | 
 | ||||||
|  | /** | ||||||
|  |  * Create a server-side caller. | ||||||
|  |  * | ||||||
|  |  * @see https://trpc.io/docs/server/server-side-calls
 | ||||||
|  |  */ | ||||||
|  | export const createCallerFactory = t.createCallerFactory; | ||||||
|  | 
 | ||||||
|  | /** | ||||||
|  |  * 3. ROUTER & PROCEDURE (THE IMPORTANT BIT) | ||||||
|  |  * | ||||||
|  |  * These are the pieces you use to build your tRPC API. You should import these a lot in the | ||||||
|  |  * "/src/server/api/routers" directory. | ||||||
|  |  */ | ||||||
|  | 
 | ||||||
|  | /** | ||||||
|  |  * This is how you create new routers and sub-routers in your tRPC API. | ||||||
|  |  * | ||||||
|  |  * @see https://trpc.io/docs/router
 | ||||||
|  |  */ | ||||||
|  | export const createTRPCRouter = t.router; | ||||||
|  | 
 | ||||||
|  | /** | ||||||
|  |  * Middleware for timing procedure execution and adding an artificial delay in development. | ||||||
|  |  * | ||||||
|  |  * You can remove this if you don't like it, but it can help catch unwanted waterfalls by simulating | ||||||
|  |  * network latency that would occur in production but not in local development. | ||||||
|  |  */ | ||||||
|  | const timingMiddleware = t.middleware(async ({ next, path }) => { | ||||||
|  |   const start = Date.now(); | ||||||
|  | 
 | ||||||
|  |   if (t._config.isDev) { | ||||||
|  |     // artificial delay in dev
 | ||||||
|  |     const waitMs = Math.floor(Math.random() * 400) + 100; | ||||||
|  |     await new Promise((resolve) => setTimeout(resolve, waitMs)); | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   const result = await next(); | ||||||
|  | 
 | ||||||
|  |   const end = Date.now(); | ||||||
|  |   console.log(`[TRPC] ${path} took ${end - start}ms to execute`); | ||||||
|  | 
 | ||||||
|  |   return result; | ||||||
|  | }); | ||||||
|  | 
 | ||||||
|  | /** | ||||||
|  |  * Public (unauthenticated) procedure | ||||||
|  |  * | ||||||
|  |  * This is the base piece you use to build new queries and mutations on your tRPC API. It does not | ||||||
|  |  * guarantee that a user querying is authorized, but you can still access user session data if they | ||||||
|  |  * are logged in. | ||||||
|  |  */ | ||||||
|  | export const publicProcedure = t.procedure.use(timingMiddleware); | ||||||
|  | 
 | ||||||
|  | /** | ||||||
|  |  * Protected (authenticated) procedure | ||||||
|  |  * | ||||||
|  |  * If you want a query or mutation to ONLY be accessible to logged in users, use this. It verifies | ||||||
|  |  * the session is valid and guarantees `ctx.session.user` is not null. | ||||||
|  |  * | ||||||
|  |  * @see https://trpc.io/docs/procedures
 | ||||||
|  |  */ | ||||||
|  | export const protectedProcedure = t.procedure | ||||||
|  |   .use(timingMiddleware) | ||||||
|  |   .use(({ ctx, next }) => { | ||||||
|  |     if (!ctx.session || !ctx.session.user) { | ||||||
|  |       throw new TRPCError({ code: "UNAUTHORIZED" }); | ||||||
|  |     } | ||||||
|  |     return next({ | ||||||
|  |       ctx: { | ||||||
|  |         // infers the `session` as non-nullable
 | ||||||
|  |         session: { ...ctx.session, user: ctx.session.user }, | ||||||
|  |       }, | ||||||
|  |     }); | ||||||
|  |   }); | ||||||
							
								
								
									
										68
									
								
								src/server/auth/config.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										68
									
								
								src/server/auth/config.ts
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,68 @@ | |||||||
|  | import { DrizzleAdapter } from "@auth/drizzle-adapter"; | ||||||
|  | import { type DefaultSession, type NextAuthConfig } from "next-auth"; | ||||||
|  | import DiscordProvider from "next-auth/providers/discord"; | ||||||
|  | 
 | ||||||
|  | import { db } from "@/server/db"; | ||||||
|  | import { | ||||||
|  |   accounts, | ||||||
|  |   sessions, | ||||||
|  |   users, | ||||||
|  |   verificationTokens, | ||||||
|  | } from "@/server/db/schema"; | ||||||
|  | import { Adapter } from "next-auth/adapters"; | ||||||
|  | 
 | ||||||
|  | /** | ||||||
|  |  * Module augmentation for `next-auth` types. Allows us to add custom properties to the `session` | ||||||
|  |  * object and keep type safety. | ||||||
|  |  * | ||||||
|  |  * @see https://next-auth.js.org/getting-started/typescript#module-augmentation
 | ||||||
|  |  */ | ||||||
|  | declare module "next-auth" { | ||||||
|  |   interface Session extends DefaultSession { | ||||||
|  |     user: { | ||||||
|  |       id: string; | ||||||
|  |       // ...other properties
 | ||||||
|  |       role: UserRole; | ||||||
|  |     } & DefaultSession["user"]; | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   interface User { | ||||||
|  |     // ...other properties
 | ||||||
|  |     role: UserRole; | ||||||
|  |   } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /** | ||||||
|  |  * Options for NextAuth.js used to configure adapters, providers, callbacks, etc. | ||||||
|  |  * | ||||||
|  |  * @see https://next-auth.js.org/configuration/options
 | ||||||
|  |  */ | ||||||
|  | export const authConfig = { | ||||||
|  |   providers: [ | ||||||
|  |     DiscordProvider, | ||||||
|  |     /** | ||||||
|  |      * ...add more providers here. | ||||||
|  |      * | ||||||
|  |      * Most other providers require a bit more work than the Discord provider. For example, the | ||||||
|  |      * GitHub provider requires you to add the `refresh_token_expires_in` field to the Account | ||||||
|  |      * model. Refer to the NextAuth.js docs for the provider you want to use. Example: | ||||||
|  |      * | ||||||
|  |      * @see https://next-auth.js.org/providers/github
 | ||||||
|  |      */ | ||||||
|  |   ], | ||||||
|  |   adapter: DrizzleAdapter(db, { | ||||||
|  |     usersTable: users, | ||||||
|  |     accountsTable: accounts, | ||||||
|  |     sessionsTable: sessions, | ||||||
|  |     verificationTokensTable: verificationTokens, | ||||||
|  |   }) as Adapter, | ||||||
|  |   callbacks: { | ||||||
|  |     session: ({ session, user }) => ({ | ||||||
|  |       ...session, | ||||||
|  |       user: { | ||||||
|  |         ...session.user, | ||||||
|  |         id: user.id, | ||||||
|  |       }, | ||||||
|  |     }), | ||||||
|  |   }, | ||||||
|  | } satisfies NextAuthConfig; | ||||||
							
								
								
									
										10
									
								
								src/server/auth/index.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										10
									
								
								src/server/auth/index.ts
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,10 @@ | |||||||
|  | import NextAuth from "next-auth"; | ||||||
|  | import { cache } from "react"; | ||||||
|  | 
 | ||||||
|  | import { authConfig } from "./config"; | ||||||
|  | 
 | ||||||
|  | const { auth: uncachedAuth, handlers, signIn, signOut } = NextAuth(authConfig); | ||||||
|  | 
 | ||||||
|  | const auth = cache(uncachedAuth); | ||||||
|  | 
 | ||||||
|  | export { auth, handlers, signIn, signOut }; | ||||||
							
								
								
									
										18
									
								
								src/server/db/index.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										18
									
								
								src/server/db/index.ts
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,18 @@ | |||||||
|  | import { drizzle } from "drizzle-orm/postgres-js"; | ||||||
|  | import postgres from "postgres"; | ||||||
|  | 
 | ||||||
|  | import { env } from "@/env"; | ||||||
|  | import * as schema from "./schema"; | ||||||
|  | 
 | ||||||
|  | /** | ||||||
|  |  * Cache the database connection in development. This avoids creating a new connection on every HMR | ||||||
|  |  * update. | ||||||
|  |  */ | ||||||
|  | const globalForDb = globalThis as unknown as { | ||||||
|  |   conn: postgres.Sql | undefined; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | const conn = globalForDb.conn ?? postgres(env.DATABASE_URL); | ||||||
|  | if (env.NODE_ENV !== "production") globalForDb.conn = conn; | ||||||
|  | 
 | ||||||
|  | export const db = drizzle(conn, { schema }); | ||||||
							
								
								
									
										168
									
								
								src/server/db/schema.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										168
									
								
								src/server/db/schema.ts
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,168 @@ | |||||||
|  | import { createId } from "@paralleldrive/cuid2"; | ||||||
|  | import { relations, sql } from "drizzle-orm"; | ||||||
|  | import { | ||||||
|  |   index, | ||||||
|  |   integer, | ||||||
|  |   pgTableCreator, | ||||||
|  |   primaryKey, | ||||||
|  |   text, | ||||||
|  |   timestamp, | ||||||
|  |   varchar, | ||||||
|  | } from "drizzle-orm/pg-core"; | ||||||
|  | import { type AdapterAccount } from "next-auth/adapters"; | ||||||
|  | /** | ||||||
|  |  * This is an example of how to use the multi-project schema feature of Drizzle ORM. Use the same | ||||||
|  |  * database instance for multiple projects. | ||||||
|  |  * | ||||||
|  |  * @see https://orm.drizzle.team/docs/goodies#multi-project-schema
 | ||||||
|  |  */ | ||||||
|  | export const createTable = pgTableCreator((name) => `wiki-antifa_${name}`); | ||||||
|  | 
 | ||||||
|  | export const articles = createTable( | ||||||
|  |   "article", | ||||||
|  |   { | ||||||
|  |     id: varchar("id", { length: 255 }) | ||||||
|  |       .primaryKey() | ||||||
|  |       .$defaultFn(() => createId()) | ||||||
|  |       .notNull(), | ||||||
|  |     title: varchar("title", { length: 256 }), | ||||||
|  |     slug: varchar("slug", { length: 256 }).unique(), | ||||||
|  |     authorId: varchar("author_id", { length: 255 }), | ||||||
|  |     content: text("content"), | ||||||
|  |     categoryId: varchar("category_id", { length: 255 }), | ||||||
|  |     createdAt: timestamp("created_at", { withTimezone: true }) | ||||||
|  |       .default(sql`CURRENT_TIMESTAMP`) | ||||||
|  |       .notNull(), | ||||||
|  |     updatedAt: timestamp("updated_at", { withTimezone: true }).$onUpdate( | ||||||
|  |       () => new Date(), | ||||||
|  |     ), | ||||||
|  |   }, | ||||||
|  |   (example) => ({ | ||||||
|  |     articleTitleIndex: index("article_title_idx").on(example.title), | ||||||
|  |   }), | ||||||
|  | ); | ||||||
|  | export type Article = typeof articles.$inferSelect; | ||||||
|  | 
 | ||||||
|  | export const articleRelations = relations(articles, ({ one }) => ({ | ||||||
|  |   categories: one(categories, { | ||||||
|  |     fields: [articles.categoryId], | ||||||
|  |     references: [categories.id], | ||||||
|  |   }), | ||||||
|  | })); | ||||||
|  | 
 | ||||||
|  | export const categories = createTable( | ||||||
|  |   "category", | ||||||
|  |   { | ||||||
|  |     id: varchar("id", { length: 255 }) | ||||||
|  |       .primaryKey() | ||||||
|  |       .$defaultFn(() => createId()) | ||||||
|  |       .notNull(), | ||||||
|  |     name: varchar("name", { length: 256 }), | ||||||
|  |     slug: varchar("slug", { length: 256 }).unique(), | ||||||
|  | 
 | ||||||
|  |     createdAt: timestamp("created_at", { withTimezone: true }) | ||||||
|  |       .default(sql`CURRENT_TIMESTAMP`) | ||||||
|  |       .notNull(), | ||||||
|  |     updatedAt: timestamp("updated_at", { withTimezone: true }).$onUpdate( | ||||||
|  |       () => new Date(), | ||||||
|  |     ), | ||||||
|  |   }, | ||||||
|  |   (example) => ({ | ||||||
|  |     categoryNameIndex: index("category_name_idx").on(example.name), | ||||||
|  |   }), | ||||||
|  | ); | ||||||
|  | export type Category = typeof categories.$inferSelect; | ||||||
|  | 
 | ||||||
|  | export const categoryRelations = relations(categories, ({ many }) => ({ | ||||||
|  |   articles: many(articles), | ||||||
|  | })); | ||||||
|  | 
 | ||||||
|  | export const users = createTable("user", { | ||||||
|  |   id: varchar("id", { length: 255 }) | ||||||
|  |     .notNull() | ||||||
|  |     .primaryKey() | ||||||
|  |     .$defaultFn(() => crypto.randomUUID()), | ||||||
|  |   name: varchar("name", { length: 255 }), | ||||||
|  |   email: varchar("email", { length: 255 }).notNull(), | ||||||
|  |   role: integer("role").default(1).notNull(), | ||||||
|  |   emailVerified: timestamp("email_verified", { | ||||||
|  |     mode: "date", | ||||||
|  |     withTimezone: true, | ||||||
|  |   }).default(sql`CURRENT_TIMESTAMP`), | ||||||
|  |   image: varchar("image", { length: 255 }), | ||||||
|  | }); | ||||||
|  | 
 | ||||||
|  | export const usersRelations = relations(users, ({ many }) => ({ | ||||||
|  |   accounts: many(accounts), | ||||||
|  | })); | ||||||
|  | 
 | ||||||
|  | export const accounts = createTable( | ||||||
|  |   "account", | ||||||
|  |   { | ||||||
|  |     userId: varchar("user_id", { length: 255 }) | ||||||
|  |       .notNull() | ||||||
|  |       .references(() => users.id), | ||||||
|  |     type: varchar("type", { length: 255 }) | ||||||
|  |       .$type<AdapterAccount["type"]>() | ||||||
|  |       .notNull(), | ||||||
|  |     provider: varchar("provider", { length: 255 }).notNull(), | ||||||
|  |     providerAccountId: varchar("provider_account_id", { | ||||||
|  |       length: 255, | ||||||
|  |     }).notNull(), | ||||||
|  |     refresh_token: text("refresh_token"), | ||||||
|  |     access_token: text("access_token"), | ||||||
|  |     expires_at: integer("expires_at"), | ||||||
|  |     token_type: varchar("token_type", { length: 255 }), | ||||||
|  |     scope: varchar("scope", { length: 255 }), | ||||||
|  |     id_token: text("id_token"), | ||||||
|  |     session_state: varchar("session_state", { length: 255 }), | ||||||
|  |   }, | ||||||
|  |   (account) => ({ | ||||||
|  |     compoundKey: primaryKey({ | ||||||
|  |       columns: [account.provider, account.providerAccountId], | ||||||
|  |     }), | ||||||
|  |     userIdIdx: index("account_user_id_idx").on(account.userId), | ||||||
|  |   }), | ||||||
|  | ); | ||||||
|  | 
 | ||||||
|  | export const accountsRelations = relations(accounts, ({ one }) => ({ | ||||||
|  |   user: one(users, { fields: [accounts.userId], references: [users.id] }), | ||||||
|  | })); | ||||||
|  | 
 | ||||||
|  | export const sessions = createTable( | ||||||
|  |   "session", | ||||||
|  |   { | ||||||
|  |     sessionToken: varchar("session_token", { length: 255 }) | ||||||
|  |       .notNull() | ||||||
|  |       .primaryKey(), | ||||||
|  |     userId: varchar("user_id", { length: 255 }) | ||||||
|  |       .notNull() | ||||||
|  |       .references(() => users.id), | ||||||
|  |     expires: timestamp("expires", { | ||||||
|  |       mode: "date", | ||||||
|  |       withTimezone: true, | ||||||
|  |     }).notNull(), | ||||||
|  |   }, | ||||||
|  |   (session) => ({ | ||||||
|  |     userIdIdx: index("session_user_id_idx").on(session.userId), | ||||||
|  |   }), | ||||||
|  | ); | ||||||
|  | 
 | ||||||
|  | export const sessionsRelations = relations(sessions, ({ one }) => ({ | ||||||
|  |   user: one(users, { fields: [sessions.userId], references: [users.id] }), | ||||||
|  | })); | ||||||
|  | 
 | ||||||
|  | export const verificationTokens = createTable( | ||||||
|  |   "verification_token", | ||||||
|  |   { | ||||||
|  |     identifier: varchar("identifier", { length: 255 }).notNull(), | ||||||
|  |     token: varchar("token", { length: 255 }).notNull(), | ||||||
|  |     expires: timestamp("expires", { | ||||||
|  |       mode: "date", | ||||||
|  |       withTimezone: true, | ||||||
|  |     }).notNull(), | ||||||
|  |   }, | ||||||
|  |   (vt) => ({ | ||||||
|  |     compoundKey: primaryKey({ columns: [vt.identifier, vt.token] }), | ||||||
|  |   }), | ||||||
|  | ); | ||||||
							
								
								
									
										84
									
								
								src/styles/globals.css
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										84
									
								
								src/styles/globals.css
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,84 @@ | |||||||
|  | @tailwind base; | ||||||
|  | @tailwind components; | ||||||
|  | @tailwind utilities; | ||||||
|  | 
 | ||||||
|  | @layer base { | ||||||
|  |   :root { | ||||||
|  |     --background: 0 0% 100%; | ||||||
|  |     --foreground: 0 0% 3.9%; | ||||||
|  |     --card: 0 0% 100%; | ||||||
|  |     --card-foreground: 0 0% 3.9%; | ||||||
|  |     --popover: 0 0% 100%; | ||||||
|  |     --popover-foreground: 0 0% 3.9%; | ||||||
|  |     --primary: 0 0% 9%; | ||||||
|  |     --primary-foreground: 0 0% 98%; | ||||||
|  |     --secondary: 0 0% 96.1%; | ||||||
|  |     --secondary-foreground: 0 0% 9%; | ||||||
|  |     --muted: 0 0% 96.1%; | ||||||
|  |     --muted-foreground: 0 0% 45.1%; | ||||||
|  |     --accent: 0 0% 96.1%; | ||||||
|  |     --accent-foreground: 0 0% 9%; | ||||||
|  |     --destructive: 0 84.2% 60.2%; | ||||||
|  |     --destructive-foreground: 0 0% 98%; | ||||||
|  |     --border: 0 0% 89.8%; | ||||||
|  |     --input: 0 0% 89.8%; | ||||||
|  |     --ring: 0 0% 3.9%; | ||||||
|  |     --chart-1: 12 76% 61%; | ||||||
|  |     --chart-2: 173 58% 39%; | ||||||
|  |     --chart-3: 197 37% 24%; | ||||||
|  |     --chart-4: 43 74% 66%; | ||||||
|  |     --chart-5: 27 87% 67%; | ||||||
|  |     --radius: 0.5rem | ||||||
|  |   ; | ||||||
|  |     --sidebar-background: 0 0% 98%; | ||||||
|  |     --sidebar-foreground: 240 5.3% 26.1%; | ||||||
|  |     --sidebar-primary: 240 5.9% 10%; | ||||||
|  |     --sidebar-primary-foreground: 0 0% 98%; | ||||||
|  |     --sidebar-accent: 240 4.8% 95.9%; | ||||||
|  |     --sidebar-accent-foreground: 240 5.9% 10%; | ||||||
|  |     --sidebar-border: 220 13% 91%; | ||||||
|  |     --sidebar-ring: 217.2 91.2% 59.8%} | ||||||
|  |   .dark { | ||||||
|  |     --background: 0 0% 3.9%; | ||||||
|  |     --foreground: 0 0% 98%; | ||||||
|  |     --card: 0 0% 3.9%; | ||||||
|  |     --card-foreground: 0 0% 98%; | ||||||
|  |     --popover: 0 0% 3.9%; | ||||||
|  |     --popover-foreground: 0 0% 98%; | ||||||
|  |     --primary: 0 0% 98%; | ||||||
|  |     --primary-foreground: 0 0% 9%; | ||||||
|  |     --secondary: 0 0% 14.9%; | ||||||
|  |     --secondary-foreground: 0 0% 98%; | ||||||
|  |     --muted: 0 0% 14.9%; | ||||||
|  |     --muted-foreground: 0 0% 63.9%; | ||||||
|  |     --accent: 0 0% 14.9%; | ||||||
|  |     --accent-foreground: 0 0% 98%; | ||||||
|  |     --destructive: 0 62.8% 30.6%; | ||||||
|  |     --destructive-foreground: 0 0% 98%; | ||||||
|  |     --border: 0 0% 14.9%; | ||||||
|  |     --input: 0 0% 14.9%; | ||||||
|  |     --ring: 0 0% 83.1%; | ||||||
|  |     --chart-1: 220 70% 50%; | ||||||
|  |     --chart-2: 160 60% 45%; | ||||||
|  |     --chart-3: 30 80% 55%; | ||||||
|  |     --chart-4: 280 65% 60%; | ||||||
|  |     --chart-5: 340 75% 55% | ||||||
|  |   ; | ||||||
|  |     --sidebar-background: 240 5.9% 10%; | ||||||
|  |     --sidebar-foreground: 240 4.8% 95.9%; | ||||||
|  |     --sidebar-primary: 224.3 76.3% 48%; | ||||||
|  |     --sidebar-primary-foreground: 0 0% 100%; | ||||||
|  |     --sidebar-accent: 240 3.7% 15.9%; | ||||||
|  |     --sidebar-accent-foreground: 240 4.8% 95.9%; | ||||||
|  |     --sidebar-border: 240 3.7% 15.9%; | ||||||
|  |     --sidebar-ring: 217.2 91.2% 59.8%} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | @layer base { | ||||||
|  |   * { | ||||||
|  |     @apply border-border; | ||||||
|  |   } | ||||||
|  |   body { | ||||||
|  |     @apply bg-background text-foreground; | ||||||
|  |   } | ||||||
|  | } | ||||||
							
								
								
									
										25
									
								
								src/trpc/query-client.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										25
									
								
								src/trpc/query-client.ts
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,25 @@ | |||||||
|  | import { | ||||||
|  |   defaultShouldDehydrateQuery, | ||||||
|  |   QueryClient, | ||||||
|  | } from "@tanstack/react-query"; | ||||||
|  | import SuperJSON from "superjson"; | ||||||
|  | 
 | ||||||
|  | export const createQueryClient = () => | ||||||
|  |   new QueryClient({ | ||||||
|  |     defaultOptions: { | ||||||
|  |       queries: { | ||||||
|  |         // With SSR, we usually want to set some default staleTime
 | ||||||
|  |         // above 0 to avoid refetching immediately on the client
 | ||||||
|  |         staleTime: 30 * 1000, | ||||||
|  |       }, | ||||||
|  |       dehydrate: { | ||||||
|  |         serializeData: SuperJSON.serialize, | ||||||
|  |         shouldDehydrateQuery: (query) => | ||||||
|  |           defaultShouldDehydrateQuery(query) || | ||||||
|  |           query.state.status === "pending", | ||||||
|  |       }, | ||||||
|  |       hydrate: { | ||||||
|  |         deserializeData: SuperJSON.deserialize, | ||||||
|  |       }, | ||||||
|  |     }, | ||||||
|  |   }); | ||||||
							
								
								
									
										76
									
								
								src/trpc/react.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										76
									
								
								src/trpc/react.tsx
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,76 @@ | |||||||
|  | "use client"; | ||||||
|  | 
 | ||||||
|  | import { QueryClientProvider, type QueryClient } from "@tanstack/react-query"; | ||||||
|  | import { loggerLink, unstable_httpBatchStreamLink } from "@trpc/client"; | ||||||
|  | import { createTRPCReact } from "@trpc/react-query"; | ||||||
|  | import { type inferRouterInputs, type inferRouterOutputs } from "@trpc/server"; | ||||||
|  | import { useState } from "react"; | ||||||
|  | import SuperJSON from "superjson"; | ||||||
|  | 
 | ||||||
|  | import { type AppRouter } from "@/server/api/root"; | ||||||
|  | import { createQueryClient } from "./query-client"; | ||||||
|  | 
 | ||||||
|  | let clientQueryClientSingleton: QueryClient | undefined = undefined; | ||||||
|  | const getQueryClient = () => { | ||||||
|  |   if (typeof window === "undefined") { | ||||||
|  |     // Server: always make a new query client
 | ||||||
|  |     return createQueryClient(); | ||||||
|  |   } | ||||||
|  |   // Browser: use singleton pattern to keep the same query client
 | ||||||
|  |   return (clientQueryClientSingleton ??= createQueryClient()); | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | export const api = createTRPCReact<AppRouter>(); | ||||||
|  | 
 | ||||||
|  | /** | ||||||
|  |  * Inference helper for inputs. | ||||||
|  |  * | ||||||
|  |  * @example type HelloInput = RouterInputs['example']['hello'] | ||||||
|  |  */ | ||||||
|  | export type RouterInputs = inferRouterInputs<AppRouter>; | ||||||
|  | 
 | ||||||
|  | /** | ||||||
|  |  * Inference helper for outputs. | ||||||
|  |  * | ||||||
|  |  * @example type HelloOutput = RouterOutputs['example']['hello'] | ||||||
|  |  */ | ||||||
|  | export type RouterOutputs = inferRouterOutputs<AppRouter>; | ||||||
|  | 
 | ||||||
|  | export function TRPCReactProvider(props: { children: React.ReactNode }) { | ||||||
|  |   const queryClient = getQueryClient(); | ||||||
|  | 
 | ||||||
|  |   const [trpcClient] = useState(() => | ||||||
|  |     api.createClient({ | ||||||
|  |       links: [ | ||||||
|  |         loggerLink({ | ||||||
|  |           enabled: (op) => | ||||||
|  |             process.env.NODE_ENV === "development" || | ||||||
|  |             (op.direction === "down" && op.result instanceof Error), | ||||||
|  |         }), | ||||||
|  |         unstable_httpBatchStreamLink({ | ||||||
|  |           transformer: SuperJSON, | ||||||
|  |           url: getBaseUrl() + "/api/trpc", | ||||||
|  |           headers: () => { | ||||||
|  |             const headers = new Headers(); | ||||||
|  |             headers.set("x-trpc-source", "nextjs-react"); | ||||||
|  |             return headers; | ||||||
|  |           }, | ||||||
|  |         }), | ||||||
|  |       ], | ||||||
|  |     }) | ||||||
|  |   ); | ||||||
|  | 
 | ||||||
|  |   return ( | ||||||
|  |     <QueryClientProvider client={queryClient}> | ||||||
|  |       <api.Provider client={trpcClient} queryClient={queryClient}> | ||||||
|  |         {props.children} | ||||||
|  |       </api.Provider> | ||||||
|  |     </QueryClientProvider> | ||||||
|  |   ); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | function getBaseUrl() { | ||||||
|  |   if (typeof window !== "undefined") return window.location.origin; | ||||||
|  |   if (process.env.VERCEL_URL) return `https://${process.env.VERCEL_URL}`; | ||||||
|  |   return `http://localhost:${process.env.PORT ?? 3000}`; | ||||||
|  | } | ||||||
							
								
								
									
										30
									
								
								src/trpc/server.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										30
									
								
								src/trpc/server.ts
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,30 @@ | |||||||
|  | import "server-only"; | ||||||
|  | 
 | ||||||
|  | import { createHydrationHelpers } from "@trpc/react-query/rsc"; | ||||||
|  | import { headers } from "next/headers"; | ||||||
|  | import { cache } from "react"; | ||||||
|  | 
 | ||||||
|  | import { createCaller, type AppRouter } from "@/server/api/root"; | ||||||
|  | import { createTRPCContext } from "@/server/api/trpc"; | ||||||
|  | import { createQueryClient } from "./query-client"; | ||||||
|  | 
 | ||||||
|  | /** | ||||||
|  |  * This wraps the `createTRPCContext` helper and provides the required context for the tRPC API when | ||||||
|  |  * handling a tRPC call from a React Server Component. | ||||||
|  |  */ | ||||||
|  | const createContext = cache(async () => { | ||||||
|  |   const heads = new Headers(await headers()); | ||||||
|  |   heads.set("x-trpc-source", "rsc"); | ||||||
|  | 
 | ||||||
|  |   return createTRPCContext({ | ||||||
|  |     headers: heads, | ||||||
|  |   }); | ||||||
|  | }); | ||||||
|  | 
 | ||||||
|  | const getQueryClient = cache(createQueryClient); | ||||||
|  | const caller = createCaller(createContext); | ||||||
|  | 
 | ||||||
|  | export const { trpc: api, HydrateClient } = createHydrationHelpers<AppRouter>( | ||||||
|  |   caller, | ||||||
|  |   getQueryClient | ||||||
|  | ); | ||||||
							
								
								
									
										1
									
								
								src/types.d.ts
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								src/types.d.ts
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @ -0,0 +1 @@ | |||||||
|  | type UserRole = number; | ||||||
							
								
								
									
										60
									
								
								start-database.sh
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										60
									
								
								start-database.sh
									
									
									
									
									
										Executable file
									
								
							| @ -0,0 +1,60 @@ | |||||||
|  | #!/usr/bin/env bash | ||||||
|  | # Use this script to start a docker container for a local development database | ||||||
|  | 
 | ||||||
|  | # TO RUN ON WINDOWS: | ||||||
|  | # 1. Install WSL (Windows Subsystem for Linux) - https://learn.microsoft.com/en-us/windows/wsl/install | ||||||
|  | # 2. Install Docker Desktop for Windows - https://docs.docker.com/docker-for-windows/install/ | ||||||
|  | # 3. Open WSL - `wsl` | ||||||
|  | # 4. Run this script - `./start-database.sh` | ||||||
|  | 
 | ||||||
|  | # On Linux and macOS you can run this script directly - `./start-database.sh` | ||||||
|  | 
 | ||||||
|  | DB_CONTAINER_NAME="wiki-antifa-postgres" | ||||||
|  | 
 | ||||||
|  | if ! [ -x "$(command -v docker)" ]; then | ||||||
|  |   echo -e "Docker is not installed. Please install docker and try again.\nDocker install guide: https://docs.docker.com/engine/install/" | ||||||
|  |   exit 1 | ||||||
|  | fi | ||||||
|  | 
 | ||||||
|  | if ! docker info > /dev/null 2>&1; then | ||||||
|  |   echo "Docker daemon is not running. Please start Docker and try again." | ||||||
|  |   exit 1 | ||||||
|  | fi | ||||||
|  | 
 | ||||||
|  | if [ "$(docker ps -q -f name=$DB_CONTAINER_NAME)" ]; then | ||||||
|  |   echo "Database container '$DB_CONTAINER_NAME' already running" | ||||||
|  |   exit 0 | ||||||
|  | fi | ||||||
|  | 
 | ||||||
|  | if [ "$(docker ps -q -a -f name=$DB_CONTAINER_NAME)" ]; then | ||||||
|  |   docker start "$DB_CONTAINER_NAME" | ||||||
|  |   echo "Existing database container '$DB_CONTAINER_NAME' started" | ||||||
|  |   exit 0 | ||||||
|  | fi | ||||||
|  | 
 | ||||||
|  | # import env variables from .env | ||||||
|  | set -a | ||||||
|  | source .env | ||||||
|  | 
 | ||||||
|  | DB_PASSWORD=$(echo "$DATABASE_URL" | awk -F':' '{print $3}' | awk -F'@' '{print $1}') | ||||||
|  | DB_PORT=$(echo "$DATABASE_URL" | awk -F':' '{print $4}' | awk -F'\/' '{print $1}') | ||||||
|  | 
 | ||||||
|  | if [ "$DB_PASSWORD" = "password" ]; then | ||||||
|  |   echo "You are using the default database password" | ||||||
|  |   read -p "Should we generate a random password for you? [y/N]: " -r REPLY | ||||||
|  |   if ! [[ $REPLY =~ ^[Yy]$ ]]; then | ||||||
|  |     echo "Please change the default password in the .env file and try again" | ||||||
|  |     exit 1 | ||||||
|  |   fi | ||||||
|  |   # Generate a random URL-safe password | ||||||
|  |   DB_PASSWORD=$(openssl rand -base64 12 | tr '+/' '-_') | ||||||
|  |   sed -i -e "s#:password@#:$DB_PASSWORD@#" .env | ||||||
|  | fi | ||||||
|  | 
 | ||||||
|  | docker run -d \ | ||||||
|  |   --name $DB_CONTAINER_NAME \ | ||||||
|  |   -e POSTGRES_USER="postgres" \ | ||||||
|  |   -e POSTGRES_PASSWORD="$DB_PASSWORD" \ | ||||||
|  |   -e POSTGRES_DB=wiki-antifa \ | ||||||
|  |   -p "$DB_PORT":5432 \ | ||||||
|  |   docker.io/postgres && echo "Database container '$DB_CONTAINER_NAME' was successfully created" | ||||||
							
								
								
									
										75
									
								
								tailwind.config.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										75
									
								
								tailwind.config.ts
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,75 @@ | |||||||
|  | import { type Config } from "tailwindcss"; | ||||||
|  | import { fontFamily } from "tailwindcss/defaultTheme"; | ||||||
|  | 
 | ||||||
|  | export default { | ||||||
|  |     darkMode: ["class"], | ||||||
|  |     content: ["./src/**/*.tsx"], | ||||||
|  |   theme: { | ||||||
|  |   	extend: { | ||||||
|  |   		fontFamily: { | ||||||
|  |   			sans: [ | ||||||
|  |   				'var(--font-geist-sans)', | ||||||
|  |                     ...fontFamily.sans | ||||||
|  |                 ] | ||||||
|  |   		}, | ||||||
|  |   		borderRadius: { | ||||||
|  |   			lg: 'var(--radius)', | ||||||
|  |   			md: 'calc(var(--radius) - 2px)', | ||||||
|  |   			sm: 'calc(var(--radius) - 4px)' | ||||||
|  |   		}, | ||||||
|  |   		colors: { | ||||||
|  |   			background: 'hsl(var(--background))', | ||||||
|  |   			foreground: 'hsl(var(--foreground))', | ||||||
|  |   			card: { | ||||||
|  |   				DEFAULT: 'hsl(var(--card))', | ||||||
|  |   				foreground: 'hsl(var(--card-foreground))' | ||||||
|  |   			}, | ||||||
|  |   			popover: { | ||||||
|  |   				DEFAULT: 'hsl(var(--popover))', | ||||||
|  |   				foreground: 'hsl(var(--popover-foreground))' | ||||||
|  |   			}, | ||||||
|  |   			primary: { | ||||||
|  |   				DEFAULT: 'hsl(var(--primary))', | ||||||
|  |   				foreground: 'hsl(var(--primary-foreground))' | ||||||
|  |   			}, | ||||||
|  |   			secondary: { | ||||||
|  |   				DEFAULT: 'hsl(var(--secondary))', | ||||||
|  |   				foreground: 'hsl(var(--secondary-foreground))' | ||||||
|  |   			}, | ||||||
|  |   			muted: { | ||||||
|  |   				DEFAULT: 'hsl(var(--muted))', | ||||||
|  |   				foreground: 'hsl(var(--muted-foreground))' | ||||||
|  |   			}, | ||||||
|  |   			accent: { | ||||||
|  |   				DEFAULT: 'hsl(var(--accent))', | ||||||
|  |   				foreground: 'hsl(var(--accent-foreground))' | ||||||
|  |   			}, | ||||||
|  |   			destructive: { | ||||||
|  |   				DEFAULT: 'hsl(var(--destructive))', | ||||||
|  |   				foreground: 'hsl(var(--destructive-foreground))' | ||||||
|  |   			}, | ||||||
|  |   			border: 'hsl(var(--border))', | ||||||
|  |   			input: 'hsl(var(--input))', | ||||||
|  |   			ring: 'hsl(var(--ring))', | ||||||
|  |   			chart: { | ||||||
|  |   				'1': 'hsl(var(--chart-1))', | ||||||
|  |   				'2': 'hsl(var(--chart-2))', | ||||||
|  |   				'3': 'hsl(var(--chart-3))', | ||||||
|  |   				'4': 'hsl(var(--chart-4))', | ||||||
|  |   				'5': 'hsl(var(--chart-5))' | ||||||
|  |   			}, | ||||||
|  |   			sidebar: { | ||||||
|  |   				DEFAULT: 'hsl(var(--sidebar-background))', | ||||||
|  |   				foreground: 'hsl(var(--sidebar-foreground))', | ||||||
|  |   				primary: 'hsl(var(--sidebar-primary))', | ||||||
|  |   				'primary-foreground': 'hsl(var(--sidebar-primary-foreground))', | ||||||
|  |   				accent: 'hsl(var(--sidebar-accent))', | ||||||
|  |   				'accent-foreground': 'hsl(var(--sidebar-accent-foreground))', | ||||||
|  |   				border: 'hsl(var(--sidebar-border))', | ||||||
|  |   				ring: 'hsl(var(--sidebar-ring))' | ||||||
|  |   			} | ||||||
|  |   		} | ||||||
|  |   	} | ||||||
|  |   }, | ||||||
|  |   plugins: [require("tailwindcss-animate")], | ||||||
|  | } satisfies Config; | ||||||
							
								
								
									
										42
									
								
								tsconfig.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										42
									
								
								tsconfig.json
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,42 @@ | |||||||
|  | { | ||||||
|  |   "compilerOptions": { | ||||||
|  |     /* Base Options: */ | ||||||
|  |     "esModuleInterop": true, | ||||||
|  |     "skipLibCheck": true, | ||||||
|  |     "target": "es2022", | ||||||
|  |     "allowJs": true, | ||||||
|  |     "resolveJsonModule": true, | ||||||
|  |     "moduleDetection": "force", | ||||||
|  |     "isolatedModules": true, | ||||||
|  | 
 | ||||||
|  |     /* Strictness */ | ||||||
|  |     "strict": true, | ||||||
|  |     "noUncheckedIndexedAccess": true, | ||||||
|  |     "checkJs": true, | ||||||
|  | 
 | ||||||
|  |     /* Bundled projects */ | ||||||
|  |     "lib": ["dom", "dom.iterable", "ES2022"], | ||||||
|  |     "noEmit": true, | ||||||
|  |     "module": "ESNext", | ||||||
|  |     "moduleResolution": "Bundler", | ||||||
|  |     "jsx": "preserve", | ||||||
|  |     "plugins": [{ "name": "next" }], | ||||||
|  |     "incremental": true, | ||||||
|  | 
 | ||||||
|  |     /* Path Aliases */ | ||||||
|  |     "baseUrl": ".", | ||||||
|  |     "paths": { | ||||||
|  |       "@/*": ["./src/*"] | ||||||
|  |     } | ||||||
|  |   }, | ||||||
|  |   "include": [ | ||||||
|  |     ".eslintrc.cjs", | ||||||
|  |     "next-env.d.ts", | ||||||
|  |     "**/*.ts", | ||||||
|  |     "**/*.tsx", | ||||||
|  |     "**/*.cjs", | ||||||
|  |     "**/*.js", | ||||||
|  |     ".next/types/**/*.ts" | ||||||
|  |   ], | ||||||
|  |   "exclude": ["node_modules"] | ||||||
|  | } | ||||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user