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
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>
)
}
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>
)
}
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>
)
}
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>
)
}
"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[]
}
}
"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[]
}
}
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>
)
}
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>
)
}
import React from 'react'
import Product from './Product'
export default function ProductList() {
return (
<div className='space-y-3'>
<Product/>
<Product/>
<Product/>
<Product/>
<Product/>
</div>
)
}
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>
)
}
<Image
src={""}
width={500}
height={500}
alt="image"
className='w-56 h-56'
/>
import type { NextConfig } from "next";
const nextConfig: NextConfig = {
images: {
remotePatterns: [
{
protocol: 'https',
hostname: '**',
},
],
},
};
export default nextConfig;
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>
)
}
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
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>
)
}
import React from 'react'
export default function error() {
return (
<div>
You have got an error.
</div>
)
}
import React from 'react'
export default function NotFound() {
return (
<div>
This page does not exist.
</div>
)
}
'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>
)
}
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>
)
}
export async function fetchProject(){
try {
const fetchedProject = await db.project.findMany({
orderBy:{
createdAt:"desc"
}
})
return fetchedProject
} catch (error) {
console.log(error)
}
}
"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)
}
}
"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>
);
}
import CreateNewProjectForm from '@/components/Forms/CreateNewProjectForm'
import React from 'react'
export default function page() {
return (
<div>
<CreateNewProjectForm/>
</div>
)
}
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
"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>
);
}
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}/>
)
}
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)
}
}
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)
}
}
"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>
);
}
"scripts": {
"postinstall": "prisma generate",
"preview": "next build && next start"
}
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
}
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;
DATABASE_URL="mongodb+srv://clancyro1789:VD62uLe4VVYZeTsr@cluster0.7rqx7.mongodb.net/(any new db name)"