Documentation

Welcome to my documentation page. Here, you'll find reusable code snippets and explanations for various projects and technologies I work with.

My Code Library

Learning to Fetch data from an API (Lesson 6) Location components(folder) -> ProductList.tsx(file)

import { TypeFetchedProduct } from '@/actions/Products'
import Image from 'next/image'
import Link from 'next/link'
import React from 'react'

export default function Product({product}:{product:TypeFetchedProduct}) {
  return (
    <Link href={"/"} className='flex items-center gap-3 hover:bg-blue-100 rounded-md px-4 py-2 transition-all hover:-translate-y-1 duration-500'>
        <Image
        src={product.imageUrl}
        width={500}
        height={500}
        alt={product.name}
        className='w-16 h-16 rounded-lg'   
        /> 
        <div>
        <h2 className='text-blue-700'>{product.imageUrl}</h2>
        <div className="flex items-center justify-between">
        <p><span>Price:</span>${product.price}</p>
        <p><span>Stock:</span>{product.stock}items</p>
        </div>

        </div>
        
    </Link>
  )
}

Learning to Fetch data from an API (Lesson 5) Location components(folder) -> ProductList.tsx(file)

import React from 'react'
import Product from './Product'
import { TypeFetchedProduct } from '@/actions/Products'

export default function ProductList({ products }: { products: TypeFetchedProduct[] }) {
  return (
    <div className='space-y-3'>
      {
        products.map((product)=>{
          return(
            <Product key={product.id} product={product}/>
          )
        })
      }   
    </div>
  )
}

Learning to Fetch data from an API (Lesson 4) Location components(folder) -> StorePage.tsx(file)

import React from 'react'
import Header from './Header'
import ProductList from './ProductList'
import ProductDetail from './ProductDetail'
import { TypeFetchedProduct } from '@/actions/Products'

export default function StorePage({receivedProducts}:{receivedProducts:TypeFetchedProduct[]}) {
  return (
    <div className='min-h-screen bg-gradient-to-r from-blue-600 to-blue-950 py-8 px-8'>

        <div className="rounded-lg bg-blue-100 min-h-screen">
            <Header/>
            <main className='grid grid-cols-12 bg-gray-50 min-h-screen'>
              <div className='col-span-4 pt-3'>
                <ProductList products={receivedProducts}/>
              </div>
              <div className='col-span-8 bg-gray-100'>
                <ProductDetail/>
              </div>
            </main>
            
        </div>
      
    </div>
  )
}

Learning to Fetch data from an API (Lesson 3) Location app(folder) -> page.tsx(file)

import { getProducts } from '@/actions/Products'
import StorePage from '@/components/StorePage'
import React from 'react'

export default async function page() {
  const fetchedProducts = (await getProducts()) || []
  return (
    <div>
      {/* Show something incase there are no products */}
      {
        fetchedProducts && fetchedProducts.length>0?(<StorePage receivedProducts={fetchedProducts}/>):(<div><h2>No Products</h2></div>)
      }  
    </div>
  )
}

Learning to Fetch data from an API (Lesson 2) actions(folder) -> products.ts(file)

"use server"

// Then lastly we have to define the types for the data we are fetching.
//Right now it has any which is dangerous
//Defining the types for that data is so good because it is going to help us all around.

export interface TypeFetchedProduct{
    id:string,
    imageUrl:string,
    stock:number,
    name:string,
    price:number,
}

const API = "https://inventory-app-ten-gilt.vercel.app/api/v1/products"
export async function getProducts(){
    try {
        //send a response
        const response = await fetch(API)
        //the result obtained, convert it into json
        const result = await response.json()
        //return the result which has been converted
        // const fetchedProducts = result.data;
        //but just above the data is too much and we don't want all that data to come just a few.
        //so here we only get that particular thing we want from the data.
        const fetchedProducts = result.data.map((fetchedProduct:any)=>{
            return{
                id:fetchedProduct.id,
                imageUrl:fetchedProduct.productThumbnail,
                stock:fetchedProduct.stockQty,
                name:fetchedProduct.name,
                price:fetchedProduct.productPrice,
            }
        })

        return fetchedProducts as TypeFetchedProduct[]
        // return result     
    } catch (error) {
        console.log(error)
        return[]
        
    }
}

Learning to Fetch data from an API (Lesson 1) actions(folder) -> products.ts(file)

"use server"

const API = "https://inventory-app-ten-gilt.vercel.app/api/v1/products"
export async function getProducts(){
    try {
        //send a response
        const response = await fetch(API)
        //the result obtained, convert it into json
        const result = await response.json()
        //return the result which has been converted
        return result     
    } catch (error) {
        console.log(error)
        return[]
        
    }
}

Product Detail (Details about a particular product)

import Image from 'next/image'
import React from 'react'

import { Bookmark } from 'lucide-react'
import { Button } from './ui/button'

export default function ProductDetail() {
  return (
    <div className='relative'>
        <Image
                src={""}
                width={500}
                height={500}
                alt="image"
                className='w-full h-[400px] object-cover'   
                /> 

    <h1 className="line-clamp-2 scroll-m-20 text-4xl font-extrabold tracking-tight lg:text-5xl absolute top-[45%] max-w-3xl bg-gradient-to-r from-blue-950 to-blue-500 left-[50%] -translate-x-1/2 text-white px-3 py-1 rounded-md">
    Lenovo Thinkpad E14, Intel Core I5-1135g7, 8gb Ram, 256gb Ssd Black
    </h1>
    <p className='px-4 py-4'>The Lenovo ThinkPad E14 is a business-oriented laptop that combines solid performance with reliability. It features an Intel Core i5 processor, offering a good balance of speed and power efficiency for everyday tasks like office work, web browsing, and multimedia.</p>
    <div className="flex py-4 px-4 w-full justify-between">
    <div className="flex items-center justify-center gap-8">
        <p><span className='text-blue-700'>Brand:</span>Hp</p>
        <p><span className='text-blue-700'>Stock:</span>32items</p>
    </div>
    <div className='px-4 '>  
        <Button>
        <Bookmark/>
        Bookmark
        </Button>
    </div>
    </div>
   
      
    </div>
  )
}

Main Section (Product Listing and Detailed Page) 2 Product Alone

import Image from 'next/image'
import Link from 'next/link'
import React from 'react'

export default function Product() {
  return (
    <Link href={"/"} className='flex items-center gap-3 hover:bg-blue-100 rounded-md px-4 py-2 transition-all hover:-translate-y-1 duration-500'>
        <Image
        src={""}
        width={500}
        height={500}
        alt="image"
        className='w-16 h-16 rounded-lg'   
        /> 
        <div>
        <h2 className='text-blue-700'>HP Spectre X360 Laptop 14-ef2017nia 9t729ea - 1TB SSD - 8GBRAM</h2>
        <div className="flex items-center justify-between">
        <p><span>Brand:</span>Hp</p>
        <p><span>Stock:</span>32items</p>
        </div>

        </div>
        
    </Link>
  )
}

Main Section (Product Listing and Detailed Page) 2 Product Listing

import React from 'react'
import Product from './Product'

export default function ProductList() {
  return (
    <div className='space-y-3'>
      <Product/>
      <Product/>
      <Product/>
      <Product/>
      <Product/>
    </div>
  )
}

Main Section (Product Listing and Detailed Page) 1

import React from 'react'
import Header from './Header'
import ProductList from './ProductList'

export default function StorePage() {
  return (
    <div className='min-h-screen bg-gradient-to-r from-blue-600 to-blue-950 py-8 px-8'>

        <div className="rounded-lg bg-blue-100 min-h-screen">
            <Header/>
            <main className='grid grid-cols-12 bg-gray-50 min-h-screen'>
              <div className='col-span-4 pt-3'>
                <ProductList/>
              </div>
              <div className='col-span-8 bg-gray-100'>
                Product detailed page
              </div>


            </main>
            
        </div>
      
    </div>
  )
}

Next Image Putting Outside links

  <Image
        src={""}
        width={500}
        height={500}
        alt="image"
        className='w-56 h-56'   
        /> 

Next Image (Outside Links) Configuration to Allow all Outside Links.

import type { NextConfig } from "next";

const nextConfig: NextConfig = {
  images: {
    remotePatterns: [
      {
        protocol: 'https',
        hostname: '**',
        
      },
    ],
  },
};

export default nextConfig;

Page Group -> This is done with pages which have the same layout.

Eg: (Group1) In brackets -> Services(Folder)-> page.tsx , Contact(Folder)-> page.tsx , Layout.tsx

Layout.tsx

import React from 'react'

export default function GroupOnelayout({children}:{children:any}) {
  return (
    <div>
        <h1>This is Group 1 layout.</h1>
        {children}     
    </div>
  )
}

Search params -> These ones want to look like the params but they are different. Location -> products(folder) -> [id] -> page.tsx

 products
page.tsx
{/* Products */}
      <a href="/product/1?page=5" className="text-blue-600">Product 1</a>



[id]
page.tsx

http://localhost:3000/product/2?page=5
import React from 'react'

export default async function page({
    params,
    searchParams,
  }: {
    params: Promise<{ slug: string, id:string }>
    searchParams: Promise<{ [key: string]: string | string[] | undefined }>
  }) { 
    const id = (await params).id
    const page = (await searchParams).page
  return (
    <div>
      <h2>This is a detailed Page.</h2>
      Page - {id}
      SearchParam - {page}



      {/* http://localhost:3000/product/5 -> dynamic param */}

      {/* http://localhost:3000/product/5?page=3 -> search param */}
    </div>
  )
}



This is a detailed Page.
Page - 2SearchParam - 5





Dynamic page. [slug] or [id] or Detailed Page Location-> Products(folder) -> id(folder) -> page.tsx

import React from 'react'

export default async function page({
    params,
  }: {
    params: Promise<{ slug: string , id:string}>
    
  }) { 
    const id = (await params).id
  return (
    <div>
      <h2>This is a detailed Page.</h2>
      Page - {id}
    </div>
  )
}

Error Page ( app(folder) -> error.tsx(file) )

import React from 'react'

export default function error() {
  return (
    <div>
      You have got an error.
    </div>
  )
}

Not Found Page (http://localhost:3000/doesnot) A page that does not exist

import React from 'react'

export default function NotFound() {
  return (
    <div>
      This page does not exist.
      
    </div>
  )
}

Create 6 (Final Ui) Location -> Components(folder) -> LatestProjects.tsx(file)

'use client'

import React from 'react'
import Image from 'next/image'
import Link from 'next/link'
import { motion } from 'framer-motion'

interface Project {
  title: string
  description: string
  image: string
  link: string
}

const projects: Project[] = [
  {
    title: "Sefbuy",
    description: "SEFBUY is an Ecommerce website designed to provide a seamless shopping experience for customers.",
    image: "/project1.png?height=300&width=500",
    link: "https://sefbuy.com/"
  },
  {
    title: "Lamudi",
    description: "Lamudi is a leading real estate platform connecting buyers and sellers across Uganda.",
    image: "/project2.png?height=300&width=500",
    link: "https://www.lamudi.co.ug/Lamudi/Index.aspx"
  },
  {
    title: "Gaba Hope For Kids",
    description: "This is a Charity Organization supporting unprivileged children with Education, Healthcare and Empowerment in Uganda.",
    image: "/project3.png?height=300&width=500",
    link: "https://www.gabahopeforkids.org/"
  }
]

const BreathingBackground = () => (
  <div className="absolute inset-0 z-0">
    <svg width="100%" height="100%" xmlns="http://www.w3.org/2000/svg">
      <defs>
        <pattern id="grid" width="50" height="50" patternUnits="userSpaceOnUse">
          <path d="M 50 0 L 0 0 0 50" fill="none" stroke="rgba(255,255,255,0.5)" strokeWidth="1"/>
        </pattern>
      </defs>
      <motion.rect
        width="100%"
        height="100%"
        fill="url(#grid)"
        animate={{
          scale: [1, 1.05, 1],
          opacity: [0.3, 0.5, 0.3],
        }}
        transition={{
          duration: 5,
          repeat: Infinity,
          ease: "easeInOut"
        }}
      />
    </svg>
  </div>
)

const AnimatedSection = motion.section

export default function LatestProjects({projectData}: {projectData: Project[]}) {
  return (
    <AnimatedSection className="relative overflow-hidden">
      <BreathingBackground />
      <div className="latest-container relative z-10">
        <h2 className="text-4xl font-semibold text-center mb-4  text-gray-800 dark:text-white name-header">Latest Projects</h2>
        <div className="latest-card-container">
          {projectData.map((project, index) => (
            <motion.div
              key={index}
              initial={{ opacity: 0, y: 50 }}
              animate={{ opacity: 1, y: 0 }}
              transition={{ duration: 0.5, delay: index * 0.2 }}
              className="bg-white dark:bg-gray-800 rounded-xl shadow-xl overflow-hidden transform hover:scale-105 transition duration-300"
            >
              <div className="relative h-64 w-full image-container">
                <Image 
                  src={project.image} 
                  alt={project.title} 
                  layout="fill" 
                  objectFit="cover"
                  className="transition-transform duration-300 hover:scale-110 inner-image"
                />
              </div>
              <div className="p-6">
                <h3 className="text-2xl font-semibold mb-3 text-gray-800 dark:text-white">{project.title}</h3>
                <p className="text-gray-600 dark:text-gray-300 mb-4">{project.description}</p>
                <Link href={project.link} className="inline-block bg-blue-500 hover:bg-blue-600 text-white font-bold py-2 px-4 rounded transition duration-300">
                  View Project
                </Link>
              </div>
            </motion.div>
          ))}
        </div>
        <div className="w-[100%] h-[10vh] mt-3 flex items-center justify-center">

          <button className='bg-blue-500  hover:bg-blue-600 text-white font-bold py-3 px-6 rounded-lg transition duration-300 text-sm'>
            <Link href="/projects">
            View All Projects  
            </Link>
          </button>
        </div>
      </div>
    </AnimatedSection>
  )
}

Create 5 (Go on the Ui) Location -> Projects(folder) -> page.tsx(file)

import { fetchProject } from '@/actions/Project'
import LatestProjects from '@/components/LatestProjects'
import Link from 'next/link'

import React from 'react'

export default async function page() {
  const dbProjects = await fetchProject() || []
  return (
    <div className='p-5 mt-[72px]'>

      <div className='w-full p-3 flex'>
        <Link href="/project/new" className='px-3 py-4 bg-blue-500 text-white rounded-[10px] mt-4 font-bold'>
        Create New Project
        </Link>
        
      </div>
      <LatestProjects projectData={dbProjects}/>    
    </div>
  )
}

Create 4 (Getting things from the database) Location -> Location -> Overall outside all others -> actions(folder) -> Projects.ts(file)

export async function fetchProject(){
  try {
   const fetchedProject = await db.project.findMany({
    orderBy:{
      createdAt:"desc"
    }
   }) 
   return fetchedProject
  } catch (error) {
   console.log(error) 
  }
}

Create 3 (Server actions) Location -> Overall outside all others -> actions(folder) -> Projects.ts(file)

"use server"

import { ProjectProps } from "@/components/Forms/CreateNewProjectForm";
import { db } from "@/prisma/db";
import { revalidatePath } from "next/cache";

export async function createNewProject(data:ProjectProps){
 try {
  const createdNewProject = await db.project.create({
    data
  })
 revalidatePath("/project")

  return createdNewProject
  
 } catch (error) {
  console.log(error)
 }
}

Create 2 (Form) Location -> Components(folder) -> Forms(folder) -> CreateNewProjectForm(file)

"use client";

import React, { useState } from "react";
import { useForm } from "react-hook-form";
import toast from "react-hot-toast";
import { useRouter } from "next/navigation";
import { Card, CardContent, CardHeader, CardTitle } from "@/components/ui/card"
import { Input } from "@/components/ui/input"
import { Textarea } from "@/components/ui/textarea"
import { Button } from "@/components/ui/button"
import { Label } from "@/components/ui/label"
import { createNewProject } from "@/actions/Project";
import Image from "next/image";
import { UploadButton } from "@/utils/uploadthing";

export type ProjectProps = {
    image :string;
    title: string;
    description: string;
    link:string;
    slug: string;
}

export default function CreateNewProjectForm() {
  
  const {
    register,
    handleSubmit,
    reset,
    formState: { errors },
  } = useForm<ProjectProps>();
  const [formError, setFormError] = useState("");
  const [loading, setLoading] = useState(false);
  const[imageUrl,setImageUrl] = useState("/emptyImage.png")
  const router = useRouter();

  async function saveData(data: ProjectProps) {
    data.slug = data.title.toLowerCase().split(" ").join("-");
    data.image = imageUrl;
    try {
      setLoading(true)
      await createNewProject(data) 
      toast.success("Project created successfully.")
      router.push("/project")
      router.refresh()
      reset()  
    } catch (error) {
      toast.error("Failed to create the project.")
      console.log(error) 
    } finally {
      setLoading(false)
    }
  }

  return (
    <section className="p-5 mt-[72px]">
      <Card className="w-full max-w-2xl mx-auto bg-gradient-to-br from-blue-500 to-blue-900 shadow-xl">
      <CardHeader className="text-white">
        <CardTitle className="text-2xl font-bold text-center mb-2">New Project</CardTitle>
      </CardHeader>
      <CardContent className="bg-white bg-opacity-90 rounded-b-lg">
        <form className="space-y-6" onSubmit={handleSubmit(saveData)}>
          <div className="space-y-2">
            <Label htmlFor="title" className="text-gray-700">Project Title</Label>
            <Input
              type="text"
              id="title"
              {...register("title", { required: "Title is required" })}
              placeholder="Enter project title"
              className="bg-white"
            />
            {errors.title && (
              <p className="text-sm text-red-500">{errors.title.message}</p>
            )}
          </div>

          

          <div className="space-y-2">
            <Label htmlFor="description" className="text-gray-700">Description</Label>
            <Textarea
              id="description"
              rows={4}
              {...register("description", {
                required: "Description is required",
              })}
              placeholder="Enter a brief description of the project"
              className="bg-white"
            />
            {errors.description && (
              <p className="text-sm text-red-500">{errors.description.message}</p>
            )}
          </div>

          {formError && <p className="text-sm text-red-500">{formError}</p>}

          <div className="space-y-2">
            <Label htmlFor="link" className="text-gray-700">Link for your Project</Label>
            <Input
              type="text"
              id="link"
              {...register("link", { required: "Link for your project is required" })}
              placeholder="Enter your Project link"
              className="bg-white"
            />
            {errors.title && (
              <p className="text-sm text-red-500">{errors.link?.message}</p>
            )}
          </div>
          <div className="space-y-2 ">
          <Label htmlFor="title" className="text-gray-700">Project Image</Label>
          <div className="w-full min-h-[40vh]  bg-slate-300 rounded-[10px]">
            <div className="w-full h-[30vh] flex items-center justify-center ">
              <Image
              src={imageUrl}
              width={512}
              height={512}
              alt="Image name"
              className="max-h-[100%] max-w-[40%]"
              
              />

            </div>
            <div className="w-full min h-[15vh] flex items-center justify-center ">
            <UploadButton
              endpoint="imageUploader"
              onClientUploadComplete={(res) => {
          // Do something with the response
             console.log("Files: ", res);
             setImageUrl(res[0].url)
            //  alert("Upload Completed");
        }}
        onUploadError={(error: Error) => {
          // Do something with the error.
          alert(`ERROR! ${error.message}`);
        }}
      />

            </div>

          </div>

          </div>
          <Button 
            type="submit" 
            disabled={loading}
            className="w-full bg-blue-600 hover:bg-blue-700 text-white"
          >
            {loading ? "Creating New Project..." : "Create New Project"}
          </Button>
        </form>
      </CardContent>
    </Card>
    </section>
  );
}

Create 1 (Form Page) Location ->Projects(folder) ->New(folder) ->page.tsx(file)

import CreateNewProjectForm from '@/components/Forms/CreateNewProjectForm'
import React from 'react'

export default function page() {
  return (
    <div>
        <CreateNewProjectForm/>     
    </div>
  )
}

Git Commands

Download Git -> Which gives us Gitbash, Git Graphical User Interface(GUI), Git Command Line Interface(CMD)

Commands
1. ls -> list all the items in the current directory.
2. clear -> wipes clean the entire gitbash terminal.
3. ls -l -> Gives you a fully detailed analysis of all the folders in the 
            current directory.
4. ls -a -> Shows any possible hidden files in the current directory.
5. mkdir -> Create a new directory eg mkdir songs, creates new folder             
            called songs.
6. rm -r -> Delete a directory or a folder eg rm -r songs
7. rm -rf -> Forceful delete of a directory or a folder eg rm -rf songs
8. cd    -> Change the directory eg cd movies/ , change and enter the 
            directory or folder.
9. code . -> Open Vscode
10. touch -> Create a new file eg touch index.html
11. cat   -> Gives a detailed analysis of the contents inside a file eg 
            cat index.html
12. rm   -> delete a file eg rm index.html
13. rm -f -> forceful delete a file eg rm -f index.html
14. mv -> Renames files or folders eg mv index.html services.html, renames 
    index.html to services.html
15. pwd -> print current directory
16. cd.. -> Go backwards 

Prisma the Ui where the fetched items are being shown from the database. (2)


"use client"

import React, { useState } from "react";
import Link from "next/link";
import { CodeSnippet } from "@prisma/client";
import CodeSnippetCard from "./CodeSnippet";


export default function CodeLibrary({codeData}: {codeData: CodeSnippet[]}) {
  const [search, setSearch] = useState("");  



  const handleCopy = (code: string) => {
    navigator.clipboard.writeText(code);
  };

  const filteredSnippets = codeData.filter((snippet) =>
    snippet.title.toLowerCase().includes(search.toLowerCase())
  );

  return (
    <div className="min-h-screen bg-blue-950 flex flex-col items-center w-full rounded-[5px]">
      <h1 className="lg:text-3xl md:text-2xl sm:text-2xl text-2xl font-bold text-white mb-6 mt-6">My Code Library</h1>
      {/* search bar */}
      <div className="w-full p-[0.5rem] flex items-center justify-center">
      <input
        type="text"
        placeholder="Search Code Snippets..."
        value={search}
        onChange={(e) => setSearch(e.target.value)}
        className=" px-4 py-2 rounded-[20px] w-full max-w-3xl bg-white text-blue-950 font-bold placeholder-gray-500 focus:outline-none focus:ring focus:ring-blue-500"
      />
      </div>

      <div className="w-full p-[0.5rem] flex items-center justify-start">
      <Link href="/new" className="bg-blue-900 mb-2 text-white border-white rounded-[10px] border-b-[1px] border-l-[1px] p-2">
      <button>
        <p>Create New Snippet</p>
      </button>
      </Link>
      </div>



      

      

      <div className="code-snippet-container">
        {filteredSnippets.map((snippet, index) => (
          <CodeSnippetCard
            key={index}
            title={snippet.title}
            code={snippet.code}
            description={snippet.description}
            onCopy={() => handleCopy(snippet.code)}
          />
        ))}
        {filteredSnippets.length === 0 && (
          <p className="text-gray-500 text-center">No snippets found.</p>
        )}
      </div>
    </div>
  );
}


Prisma the Ui where the fetched items are being shown from the database. (1)

import CodeLibrary from '@/components/CodeLibrary'
import { fetchCode } from '../../actions/CodeSnippet'

export default async function Docs() {
  const code = await fetchCode() || []
  // console.log(code)
  return (
        <CodeLibrary codeData={code}/>
  )
}

Prisma Getting or Aquiring things from the database onto the Ui

import { db } from "@/prisma/db";

export async function fetchCode(){
  try {
   const fetchedCode = await db.codeSnippet.findMany({
    orderBy:{
      createdAt:"desc"
    }
   }) 
   return fetchedCode
  } catch (error) {
   console.log(error) 
  }
}

Prisma Create Functionality in actions

import { CodeSnippetProps } from "@/components/CreateNewCodeSnippetForm";
import { db } from "@/prisma/db";
import { revalidatePath } from "next/cache";


export async function createCode(data:CodeSnippetProps){
  // console.log(data)
  //We create the data to be shown in the database.
 try {
  const createdCode = await db.codeSnippet.create({
    data
  })
  // console.log(createdCode)
  //Then we return to be able to use this data anywhere else eg to fetch the data on the ui.
 revalidatePath("/docs")
  return createdCode
 } catch (error) {
  console.log(error)
 }
}

Prisma Create Form

"use client";

import React, { useState } from "react";
import { useForm } from "react-hook-form";
import toast from "react-hot-toast";
import { useRouter } from "next/navigation";
import { createCode } from "@/actions/CodeSnippet";
import { Card, CardContent, CardHeader, CardTitle } from "@/components/ui/card"
import { Input } from "@/components/ui/input"
import { Textarea } from "@/components/ui/textarea"
import { Button } from "@/components/ui/button"
import { Label } from "@/components/ui/label"

export type CodeSnippetProps = {
  title: string;
  code: string;
  description: string;
  slug: string;
};

export default function CreateNewCodeSnippetForm() {
  const {
    register,
    handleSubmit,
    reset,
    formState: { errors },
  } = useForm<CodeSnippetProps>();
  const [formError, setFormError] = useState("");
  const [loading, setLoading] = useState(false);
  const router = useRouter();

  async function saveData(data: CodeSnippetProps) {
    data.slug = data.title.toLowerCase().split(" ").join("-");
    try {
      setLoading(true)
      await createCode(data)
      toast.success("Codesnippet created successfully.")
      router.push("/docs")
      router.refresh()
      reset()  
    } catch (error) {
      toast.error("Failed to create the code snippet.")
      console.log(error) 
    } finally {
      setLoading(false)
    }
  }

  return (
    <Card className="w-full max-w-2xl mx-auto bg-gradient-to-br from-blue-500 to-blue-900 shadow-xl">
      <CardHeader className="text-white">
        <CardTitle className="text-2xl font-bold text-center mb-2">Create New Code Snippet</CardTitle>
      </CardHeader>
      <CardContent className="bg-white bg-opacity-90 rounded-b-lg">
        <form className="space-y-6" onSubmit={handleSubmit(saveData)}>
          <div className="space-y-2">
            <Label htmlFor="title" className="text-gray-700">Snippet Title</Label>
            <Input
              type="text"
              id="title"
              {...register("title", { required: "Title is required" })}
              placeholder="Enter snippet title"
              className="bg-white"
            />
            {errors.title && (
              <p className="text-sm text-red-500">{errors.title.message}</p>
            )}
          </div>

          <div className="space-y-2">
            <Label htmlFor="code" className="text-gray-700">Code</Label>
            <Textarea
              id="code"
              rows={6}
              {...register("code", { required: "Code is required" })}
              placeholder="Enter your code here"
              className="font-mono bg-white"
            />
            {errors.code && <p className="text-sm text-red-500">{errors.code.message}</p>}
          </div>

          <div className="space-y-2">
            <Label htmlFor="description" className="text-gray-700">Description</Label>
            <Textarea
              id="description"
              rows={4}
              {...register("description", {
                required: "Description is required",
              })}
              placeholder="Enter a brief description of the code snippet"
              className="bg-white"
            />
            {errors.description && (
              <p className="text-sm text-red-500">{errors.description.message}</p>
            )}
          </div>

          {formError && <p className="text-sm text-red-500">{formError}</p>}

          <Button 
            type="submit" 
            disabled={loading}
            className="w-full bg-blue-600 hover:bg-blue-700 text-white"
          >
            {loading ? "Creating Snippet..." : "Create Code Snippet"}
          </Button>
        </form>
      </CardContent>
    </Card>
  );
}

Prisma additions to package.json

"scripts": {
    "postinstall": "prisma generate",
    "preview": "next build && next start"
  }

Prisma schema

generator client {
  provider = "prisma-client-js"
}

datasource db {
  provider = "mongodb"
  url      = env("DATABASE_URL")
}

model CodeSnippet {
  id          String   @id @default(auto()) @map("_id") @db.ObjectId
  title       String
  code        String
  description String
  slug        String @unique
  createdAt   DateTime @default(now())
  updatedAt   DateTime @updatedAt
}

Prisma Instance

import { PrismaClient } from "@prisma/client";

declare global {
  var prisma: PrismaClient | undefined;
}

export const db = globalThis.prisma || new PrismaClient();
if (process.env.NODE_ENV !== "production") globalThis.prisma = db;

.env

 DATABASE_URL="mongodb+srv://clancyro1789:VD62uLe4VVYZeTsr@cluster0.7rqx7.mongodb.net/(any new db name)"

Generate Random Numbers in JavaScript

function getRandomNumber(min, max) {
  return Math.floor(Math.random() * (max - min + 1)) + min;
}

console.log(getRandomNumber(1, 100)); // Example: Generates a random number between 1 and 100