Refactor and enhance job-related components and context.
All checks were successful
forgejo/romanjaros/portfolio/pipeline/head This commit looks good
All checks were successful
forgejo/romanjaros/portfolio/pipeline/head This commit looks good
This commit is contained in:
parent
ee8f8cf6e3
commit
f16f36ec92
13 changed files with 180 additions and 101 deletions
|
@ -63,7 +63,7 @@
|
||||||
"tags": [
|
"tags": [
|
||||||
"Typescript",
|
"Typescript",
|
||||||
"React",
|
"React",
|
||||||
"Graphql",
|
"GraphQL",
|
||||||
"REST",
|
"REST",
|
||||||
"Mui",
|
"Mui",
|
||||||
"Azure DevOps"
|
"Azure DevOps"
|
||||||
|
|
|
@ -1,4 +1,4 @@
|
||||||
import {Card} from "../../components/Card";
|
import {Kbd} from "../../components/Kbd";
|
||||||
|
|
||||||
export const AboutMe = () => {
|
export const AboutMe = () => {
|
||||||
const age = Math.floor(
|
const age = Math.floor(
|
||||||
|
@ -13,9 +13,10 @@ export const AboutMe = () => {
|
||||||
<>
|
<>
|
||||||
<section className="leading-snug text-slate-500 dark:text-slate-400">
|
<section className="leading-snug text-slate-500 dark:text-slate-400">
|
||||||
<p>
|
<p>
|
||||||
👋, my name is Roman Jaroš. I am {age}. {work} years working as software engineer.
|
Hello, my name is Roman Jaroš. I am {age}. <span className="underline dark:text-white decoration-pink-500">
|
||||||
|
{work} years working as software engineer. </span>
|
||||||
I have been programming since I was 15 years old and still love to learn new technologies.
|
I have been programming since I was 15 years old and still love to learn new technologies.
|
||||||
Outside of the programming world, I maintain servers with about 120 docker containers.
|
Outside of the programming world, I maintain servers with around 80 docker containers.
|
||||||
Outside of the IT world, I enjoy reading self improvement books, meditating,
|
Outside of the IT world, I enjoy reading self improvement books, meditating,
|
||||||
alternative medicine and model painting or playing video games.
|
alternative medicine and model painting or playing video games.
|
||||||
</p>
|
</p>
|
||||||
|
@ -24,9 +25,12 @@ export const AboutMe = () => {
|
||||||
<b>I am contractor and currently only for full remote jobs.</b>
|
<b>I am contractor and currently only for full remote jobs.</b>
|
||||||
</p>
|
</p>
|
||||||
<br />
|
<br />
|
||||||
<div className="flex justify-center flex-col md:flex-row gap-2">
|
<div className="flex md:flex-row gap-2">
|
||||||
<Card title="Web development" description="I offer my experience in web development." />
|
<Kbd>UI design</Kbd>
|
||||||
<Card title="UI/UX Design" description="I offer design web application in figma or penpot." />
|
<Kbd>Web development</Kbd>
|
||||||
|
<Kbd>Automation testing</Kbd>
|
||||||
|
<Kbd>DevOps</Kbd>
|
||||||
|
<Kbd>Monitoring</Kbd>
|
||||||
</div>
|
</div>
|
||||||
</section>
|
</section>
|
||||||
</>
|
</>
|
||||||
|
|
|
@ -1,18 +1,19 @@
|
||||||
|
"use client"
|
||||||
|
|
||||||
import {MapPinIcon, MoveRightIcon} from "lucide-react";
|
import {MapPinIcon, MoveRightIcon} from "lucide-react";
|
||||||
import {notNil} from "../../utils";
|
import {notNil} from "../../utils/notNil";
|
||||||
import {Job} from "../type";
|
import {useJobs} from "../context/useJobs";
|
||||||
import {FC} from "react";
|
|
||||||
|
|
||||||
type JobsProps = {
|
export const Jobs = () => {
|
||||||
jobs: Job[] | undefined
|
const {jobs, filter} = useJobs()
|
||||||
}
|
|
||||||
|
|
||||||
export const Jobs: FC<JobsProps> = ({jobs}) => {
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
<section className="px-4">
|
<section className="px-4">
|
||||||
<ol className="relative border-s border-gray-200 dark:border-gray-700">
|
<ol className="relative border-s border-gray-200 dark:border-gray-700">
|
||||||
{jobs?.filter(notNil).map(({name, started, ended, description, tags, link}, index) => (
|
{jobs
|
||||||
|
?.filter(notNil)
|
||||||
|
.filter(({tags}) => filter != undefined ? tags.includes(filter) : true)
|
||||||
|
.map(({name, started, ended, description, tags, link}, index) => (
|
||||||
<li key={index} className="mb-10 ms-6">
|
<li key={index} className="mb-10 ms-6">
|
||||||
<span
|
<span
|
||||||
className="absolute flex items-center justify-center w-6 h-6 bg-white rounded-full -start-3 ring-8 ring-white dark:ring-slate-800 dark:bg-slate-800">
|
className="absolute flex items-center justify-center w-6 h-6 bg-white rounded-full -start-3 ring-8 ring-white dark:ring-slate-800 dark:bg-slate-800">
|
||||||
|
|
|
@ -1,40 +1,53 @@
|
||||||
import {notNil} from "../../utils";
|
"use client"
|
||||||
|
|
||||||
|
import {notNil} from "../../utils/notNil";
|
||||||
import {StarIcon} from "lucide-react";
|
import {StarIcon} from "lucide-react";
|
||||||
import {FC} from "react";
|
import {FC} from "react";
|
||||||
import {Job} from "../type";
|
import {useJobs} from "../context/useJobs";
|
||||||
|
|
||||||
type SkillsProps = {
|
export const Skills: FC = () => {
|
||||||
jobs: Job[] | undefined
|
const {jobs, filterJobs, filter} = useJobs()
|
||||||
}
|
const skills = new Set(jobs.filter(notNil).flatMap(({tags}) => tags).sort());
|
||||||
|
|
||||||
export const Skills: FC<SkillsProps> = ({jobs}) => {
|
|
||||||
|
|
||||||
const skills = new Set(jobs?.filter(notNil).flatMap(({tags}) => tags).sort());
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
<section>
|
<section>
|
||||||
<div className="flex items-center my-2">
|
<div className="flex items-center my-2">
|
||||||
<p className="mr-2 w-12 ms-1 text-sm font-medium text-gray-500 dark:text-gray-400">Czech</p>
|
<p className="mr-2 w-24 ms-1 text-sm font-medium text-gray-500 dark:text-gray-400">Czech, native</p>
|
||||||
|
<StarIcon fill="#fde047" className="w-4 h-4 ms-1 text-yellow-300"/>
|
||||||
|
<StarIcon fill="#fde047" className="w-4 h-4 ms-1 text-yellow-300"/>
|
||||||
|
<StarIcon fill="#fde047" className="w-4 h-4 ms-1 text-yellow-300"/>
|
||||||
<StarIcon fill="#fde047" className="w-4 h-4 ms-1 text-yellow-300"/>
|
<StarIcon fill="#fde047" className="w-4 h-4 ms-1 text-yellow-300"/>
|
||||||
<StarIcon fill="#fde047" className="w-4 h-4 ms-1 text-yellow-300"/>
|
<StarIcon fill="#fde047" className="w-4 h-4 ms-1 text-yellow-300"/>
|
||||||
<StarIcon fill="#fde047" className="w-4 h-4 ms-1 text-yellow-300"/>
|
<StarIcon fill="#fde047" className="w-4 h-4 ms-1 text-yellow-300"/>
|
||||||
</div>
|
</div>
|
||||||
<div className="flex items-center my-2">
|
<div className="flex items-center my-2">
|
||||||
<p className="mr-2 w-12 ms-1 text-sm font-medium text-gray-500 dark:text-gray-400">English </p>
|
<p className="mr-2 w-24 ms-1 text-sm font-medium text-gray-500 dark:text-gray-400">English, B2</p>
|
||||||
|
<StarIcon fill="#fde047" className="w-4 h-4 ms-1 text-yellow-300"/>
|
||||||
|
<StarIcon fill="#fde047" className="w-4 h-4 ms-1 text-yellow-300"/>
|
||||||
|
<StarIcon fill="#fde047" className="w-4 h-4 ms-1 text-yellow-300"/>
|
||||||
<StarIcon fill="#fde047" className="w-4 h-4 ms-1 text-yellow-300"/>
|
<StarIcon fill="#fde047" className="w-4 h-4 ms-1 text-yellow-300"/>
|
||||||
<StarIcon className="w-4 h-4 ms-1 text-gray-300 dark:text-gray-500"/>
|
<StarIcon className="w-4 h-4 ms-1 text-gray-300 dark:text-gray-500"/>
|
||||||
<StarIcon className="w-4 h-4 ms-1 text-gray-300 dark:text-gray-500"/>
|
<StarIcon className="w-4 h-4 ms-1 text-gray-300 dark:text-gray-500"/>
|
||||||
</div>
|
</div>
|
||||||
</section>
|
</section>
|
||||||
<section className="flex flex-wrap gap-2">
|
<section>
|
||||||
|
<div className="flex flex-wrap gap-2">
|
||||||
{Array.from(skills)?.map((name) => (
|
{Array.from(skills)?.map((name) => (
|
||||||
<span
|
<span
|
||||||
key={name}
|
key={name}
|
||||||
className="bg-indigo-100 text-indigo-800 text-xs font-medium me-2 px-2.5 py-0.5 rounded-full dark:bg-indigo-900 dark:text-indigo-300">
|
onClick={() => filterJobs(name)}
|
||||||
|
className={[
|
||||||
|
"cursor-pointer text-xs font-medium me-2 px-2.5 py-0.5 rounded-full select-none",
|
||||||
|
filter === name
|
||||||
|
? "bg-red-100 text-red-800 dark:bg-red-900 dark:text-red-300"
|
||||||
|
: "bg-indigo-100 text-indigo-800 dark:bg-indigo-900 dark:text-indigo-300"
|
||||||
|
].join(" ")}>
|
||||||
{name}
|
{name}
|
||||||
</span>
|
</span>
|
||||||
))}
|
))}
|
||||||
|
</div>
|
||||||
|
<p className="mt-5 text-xs text-gray-400">Psss, you can click on pill to filter.</p>
|
||||||
</section>
|
</section>
|
||||||
</>
|
</>
|
||||||
)
|
)
|
||||||
|
|
|
@ -1,12 +1,14 @@
|
||||||
export const Title = () => {
|
export const Title = () => {
|
||||||
return (
|
return (
|
||||||
<div className="flex justify-center text-4xl sm:text-6xl my-8">
|
<div className="flex justify-center text-4xl sm:text-6xl my-8 text-slate-400">
|
||||||
<span className="text-gray-600 dark:text-white"><</span>
|
|
||||||
<h1>
|
<h1>
|
||||||
<span className="text-gray-600 dark:text-white">Just</span>
|
<span><</span>
|
||||||
<span className="text-orange-800 dark:text-orange-400">Roman</span>
|
<span
|
||||||
|
className="text-transparent bg-clip-text bg-gradient-to-r to-emerald-600 from-sky-400">
|
||||||
|
Roman {" "}
|
||||||
|
</span>
|
||||||
|
<span>/></span>
|
||||||
</h1>
|
</h1>
|
||||||
<span className="text-orange-800 dark:text-orange-400"> /></span>
|
|
||||||
</div>
|
</div>
|
||||||
)
|
)
|
||||||
}
|
}
|
25
src/app/context/JobsContext.tsx
Normal file
25
src/app/context/JobsContext.tsx
Normal file
|
@ -0,0 +1,25 @@
|
||||||
|
"use client"
|
||||||
|
|
||||||
|
import {createContext, ReactNode, FC, useState, Dispatch, SetStateAction} from "react";
|
||||||
|
import {Job} from "../type";
|
||||||
|
|
||||||
|
type JobsContextPayload = {
|
||||||
|
data: Job[]
|
||||||
|
filter?: string
|
||||||
|
setFilter: Dispatch<SetStateAction<string | undefined>>
|
||||||
|
} | undefined
|
||||||
|
|
||||||
|
export const JobsContext = createContext<JobsContextPayload>(undefined);
|
||||||
|
|
||||||
|
type JobsContextProviderProps = {
|
||||||
|
jobs: Job[]
|
||||||
|
children: ReactNode[]
|
||||||
|
}
|
||||||
|
|
||||||
|
export const JobsProvider: FC<JobsContextProviderProps> = ({children, jobs: data}) => {
|
||||||
|
const [filter, setFilter] = useState<string>()
|
||||||
|
|
||||||
|
return (
|
||||||
|
<JobsContext value={{data, filter, setFilter}}>{children}</JobsContext>
|
||||||
|
)
|
||||||
|
}
|
25
src/app/context/useJobs.ts
Normal file
25
src/app/context/useJobs.ts
Normal file
|
@ -0,0 +1,25 @@
|
||||||
|
"use client"
|
||||||
|
|
||||||
|
import {useContext} from "react";
|
||||||
|
import {JobsContext} from "./JobsContext";
|
||||||
|
|
||||||
|
export const useJobs = () => {
|
||||||
|
const context = useContext(JobsContext)
|
||||||
|
if (!context) {
|
||||||
|
throw new Error('useJobs must be used within a JobsProvider!');
|
||||||
|
}
|
||||||
|
|
||||||
|
const handleFilterJob = (name: string) => {
|
||||||
|
if (name === context.filter) {
|
||||||
|
context.setFilter(undefined)
|
||||||
|
} else {
|
||||||
|
context.setFilter(name)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
jobs: context.data,
|
||||||
|
filter: context?.filter,
|
||||||
|
filterJobs: handleFilterJob,
|
||||||
|
}
|
||||||
|
}
|
6
src/app/fetch.ts
Normal file
6
src/app/fetch.ts
Normal file
|
@ -0,0 +1,6 @@
|
||||||
|
import {Job} from "./type";
|
||||||
|
|
||||||
|
export const fetchJobs = async (): Promise<Job[] | undefined> => {
|
||||||
|
const res = await fetch(`${process.env.NEXT_PUBLIC_SITE_URL}/jobs.json`, {cache: "no-store"})
|
||||||
|
return await res?.json();
|
||||||
|
}
|
|
@ -10,7 +10,10 @@ export default function AppLayout({
|
||||||
<html lang="en">
|
<html lang="en">
|
||||||
<head>
|
<head>
|
||||||
<meta charSet="utf-8"/>
|
<meta charSet="utf-8"/>
|
||||||
|
<title>Roman Jaroš</title>
|
||||||
<meta name="viewport" content="width=device-width, initial-scale=1"/>
|
<meta name="viewport" content="width=device-width, initial-scale=1"/>
|
||||||
|
<script defer src="https://wa.romanjaros.dev/script.js"
|
||||||
|
data-website-id="27410757-9ce4-4dad-a0ad-a1e2a0303495"></script>
|
||||||
</head>
|
</head>
|
||||||
<body className="bg-white dark:bg-slate-800">
|
<body className="bg-white dark:bg-slate-800">
|
||||||
<main>{children}</main>
|
<main>{children}</main>
|
||||||
|
|
|
@ -4,12 +4,13 @@ import {Contact} from "./components/Contact";
|
||||||
import {Footer} from "./components/Footer";
|
import {Footer} from "./components/Footer";
|
||||||
import {Jobs} from "./components/Jobs";
|
import {Jobs} from "./components/Jobs";
|
||||||
import {Skills} from "./components/Skills";
|
import {Skills} from "./components/Skills";
|
||||||
import {Job} from "./type";
|
import {fetchJobs} from "./fetch";
|
||||||
|
import {JobsProvider} from "./context/JobsContext";
|
||||||
|
import {use} from "react";
|
||||||
|
|
||||||
export default async function Page() {
|
export default function Page() {
|
||||||
|
|
||||||
const res = await fetch(`${process.env.NEXT_PUBLIC_SITE_URL}/jobs.json`, {cache: "no-store"})
|
const jobs = use(fetchJobs())
|
||||||
const data: Job[] | undefined = await res?.json();
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
|
@ -17,8 +18,10 @@ export default async function Page() {
|
||||||
<Title />
|
<Title />
|
||||||
<AboutMe />
|
<AboutMe />
|
||||||
<Contact />
|
<Contact />
|
||||||
<Skills jobs={data} />
|
<JobsProvider jobs={jobs ?? []}>
|
||||||
<Jobs jobs={data} />
|
<Skills />
|
||||||
|
<Jobs />
|
||||||
|
</JobsProvider>
|
||||||
<Footer />
|
<Footer />
|
||||||
</div>
|
</div>
|
||||||
</>
|
</>
|
||||||
|
|
|
@ -1,17 +0,0 @@
|
||||||
import {FC} from "react";
|
|
||||||
|
|
||||||
type CardProps = {
|
|
||||||
title: string;
|
|
||||||
description: string;
|
|
||||||
}
|
|
||||||
|
|
||||||
export const Card: FC<CardProps> = ({title, description}) => {
|
|
||||||
return (
|
|
||||||
<div
|
|
||||||
className="w-full md:w-80 block p-6 bg-white border border-gray-200 rounded-lg shadow dark:bg-gray-800 dark:border-gray-700"
|
|
||||||
>
|
|
||||||
<h5 className="mb-2 text-2xl font-bold tracking-tight text-gray-900 dark:text-white">{title}</h5>
|
|
||||||
<p className="font-normal text-gray-700 dark:text-gray-400">{description}</p>
|
|
||||||
</div>
|
|
||||||
)
|
|
||||||
}
|
|
14
src/components/Kbd.tsx
Normal file
14
src/components/Kbd.tsx
Normal file
|
@ -0,0 +1,14 @@
|
||||||
|
import {FC} from "react";
|
||||||
|
|
||||||
|
type KbdProps = {
|
||||||
|
children: string
|
||||||
|
}
|
||||||
|
|
||||||
|
export const Kbd: FC<KbdProps> = ({children}) => {
|
||||||
|
return (
|
||||||
|
<kbd
|
||||||
|
className="px-2 py-1.5 text-xs font-semibold text-gray-800 bg-gray-100 border border-gray-200 rounded-lg dark:bg-gray-600 dark:text-gray-100 dark:border-gray-500">
|
||||||
|
{children}
|
||||||
|
</kbd>
|
||||||
|
)
|
||||||
|
}
|
Loading…
Add table
Reference in a new issue