1
1
mirror of https://github.com/theoludwig/theoludwig.git synced 2025-05-29 22:37:44 +02:00

Compare commits

...

39 Commits

Author SHA1 Message Date
339e42acfa chore(release): 3.1.2 [skip ci] 2023-12-28 05:24:14 +00:00
c123815a86 fix(portfolio): remove Thream project as it is now deprecated 2023-12-28 06:21:32 +01:00
dd26a277a2 fix: update dependencies to latest 2023-12-28 06:21:24 +01:00
62222dbb0c chore(release): 3.1.1 [skip ci] 2023-11-07 20:34:11 +00:00
ee0a02bc8b chore: downgrade commitlint temporarily to release new versions 2023-11-07 21:33:05 +01:00
2e04053ec3 fix: update CV with latest education courses 2023-11-07 21:14:43 +01:00
45a9a69122 fix: update dependencies to latest 2023-11-07 20:57:52 +01:00
e566ef6c38 chore: better Prettier config for easier reviews 2023-10-23 23:11:59 +02:00
c7ad15a465 chore(release): 3.1.0 [skip ci] 2023-09-18 20:09:29 +00:00
f4a842efb5 build: downgrade semantic-release to v21.1.2
Ref: https://github.com/semantic-release/semantic-release/issues/2968
2023-09-18 22:08:04 +02:00
424c97019b fix: update dependencies to latest 2023-09-18 22:00:11 +02:00
c0508dc0b9 build: ignore ESLint errors for Production build
It improves performance when extra checking is not necessary.
2023-09-14 12:26:19 +02:00
f04d8a0c11 Revert "build: ignore ESLint and TypeScript errors for Production build"
This reverts commit fdab2a7ea8.
2023-09-14 12:19:26 +02:00
d29064745c feat: add IRCAD as work experience 2023-09-14 12:11:18 +02:00
95febe2a99 feat: add 3rd year of BUT Informatique as education 2023-09-14 12:01:20 +02:00
fdab2a7ea8 build: ignore ESLint and TypeScript errors for Production build
It improves performance when extra checking is not necessary.
2023-09-14 11:49:52 +02:00
35211fa279 fix: try/catch inside middleware when checking locale 2023-08-24 23:09:33 +02:00
137cceffa1 build(deps): update latest 2023-08-24 22:38:09 +02:00
f6bfc466de chore(release): 3.0.0 [skip ci] 2023-08-01 17:41:12 +00:00
e4cf714d95 test: fix styles import for unit tests 2023-08-01 19:39:09 +02:00
d3c86b2a26 chore: update Dockerfile 2023-08-01 19:34:58 +02:00
d2578abeec fix: loader improvements 2023-08-01 18:59:45 +02:00
e51e3bdc19 test: fix e2e tests + 500 error page 2023-08-01 18:18:16 +02:00
56520830e9 refactor: blog directory 2023-08-01 17:44:08 +02:00
2e0138194c refactor: avoid usage of React.FC to use JSX.Element (to stay consistent) 2023-08-01 17:22:09 +02:00
4b2e7bae90 feat: rewrite blog to Next.js v13 app directory
Improvement: Support light theme in code block
2023-08-01 17:07:19 +02:00
caa6a90418 refactor: implement light/dark themes using cookies 2023-08-01 14:11:46 +02:00
e82db952db docs: update interests 2023-08-01 13:15:03 +02:00
6b29ce9b15 feat: rewrite to Next.js v13 app directory
Improvements:
- Hide switch theme input (ugly little white square)
- i18n without subpath (e.g: /fr or /en), same url whatever the locale used
2023-07-31 19:06:46 +02:00
5640f1b434 build(deps): bump Node.js to 20.0.0 and npm to 9.0.0
BREAKING CHANGE: minimum supported Node.js >= 20.0.0 and npm >= 9.0.0
2023-07-30 19:03:36 +02:00
6d0dcb50a7 refactor: 'use client' when appropriate 2023-07-30 18:50:14 +02:00
70603f1444 chore: remove build error with Docker copy wrong node_modules 2023-07-30 18:27:15 +02:00
f42fdbfd0c chore: rename jsonresume-theme-custom to curriculum-vitae 2023-07-28 11:53:04 +02:00
6a3f335f9f fix(posts): update git blog post 2023-07-28 11:40:19 +02:00
f1509d0af1 chore: rename docker-compose.yml to compose.yaml
Ref: https://docs.docker.com/compose/compose-file/03-compose-file/
2023-07-28 11:38:34 +02:00
49599d25ed chore(release): 2.13.0 [skip ci] 2023-07-22 17:47:02 +00:00
65e0f4f8b6 fix: avoid scrolling when changing language 2023-07-22 19:40:28 +02:00
8d60c2d53a feat: add Carolo project in Portfolio 2023-07-22 19:39:57 +02:00
0bbebeab99 build(deps): update latest
Some checks failed
Analyze / analyze (javascript) (push) Failing after 1m24s
Build / build (push) Successful in 3m0s
Lint / lint (push) Successful in 2m6s
Test / test-unit (push) Successful in 1m56s
Test / test-e2e (push) Successful in 3m17s
2023-07-19 00:09:28 +02:00
160 changed files with 6716 additions and 5853 deletions

View File

@ -1 +1 @@
FROM mcr.microsoft.com/devcontainers/javascript-node:18 FROM mcr.microsoft.com/devcontainers/javascript-node:20

View File

@ -0,0 +1,9 @@
services:
workspace:
build:
context: "./"
dockerfile: "./Dockerfile"
volumes:
- "..:/workspace:cached"
command: "sleep infinity"
network_mode: "host"

View File

@ -1,25 +1,24 @@
{ {
"name": "theoludwig", "name": "theoludwig",
"dockerComposeFile": "./docker-compose.yml", "dockerComposeFile": "./compose.yaml",
"service": "workspace", "service": "workspace",
"workspaceFolder": "/workspace", "workspaceFolder": "/workspace",
"customizations": { "customizations": {
"vscode": { "vscode": {
"settings": { "settings": {
"remote.autoForwardPorts": false "remote.autoForwardPorts": false,
}, "remote.localPortHost": "allInterfaces"
"extensions": [ }
"editorconfig.editorconfig", },
"esbenp.prettier-vscode", "extensions": [
"dbaeumer.vscode-eslint", "editorconfig.editorconfig",
"bradlc.vscode-tailwindcss", "esbenp.prettier-vscode",
"mikestead.dotenv", "dbaeumer.vscode-eslint",
"davidanson.vscode-markdownlint", "davidanson.vscode-markdownlint",
"ms-azuretools.vscode-docker" "bradlc.vscode-tailwindcss",
] "mikestead.dotenv",
} "ms-azuretools.vscode-docker"
]
}, },
"forwardPorts": [3000],
"postAttachCommand": ["npm", "install"],
"remoteUser": "node" "remoteUser": "node"
} }

View File

@ -1,8 +0,0 @@
services:
workspace:
build:
context: './'
dockerfile: './Dockerfile'
volumes:
- '..:/workspace:cached'
command: 'sleep infinity'

View File

@ -1,5 +1,21 @@
.* **/.turbo
!.npmrc **/.next
build **/out
coverage **/build
node_modules **/coverage
**/node_modules
# envs
.env
.env.production
.env.development
secrets
# misc
.DS_Store
*.pem
Dockerfile
# typescript
*.tsbuildinfo
next-env.d.ts

View File

@ -1,4 +1,4 @@
# For more information see: https://editorconfig.org/ # https://editorconfig.org/
root = true root = true

View File

@ -1,2 +1,4 @@
COMPOSE_PROJECT_NAME=theoludwig COMPOSE_PROJECT_NAME=theoludwig
HOSTNAME=0.0.0.0
PORT=3000 PORT=3000
NEXT_TELEMETRY_DISABLED=1

View File

@ -1,10 +1,16 @@
{ {
"extends": ["conventions", "next/core-web-vitals", "prettier"], "extends": ["conventions", "next/core-web-vitals", "prettier"],
"plugins": ["prettier", "unicorn"], "plugins": ["prettier"],
"parserOptions": { "parserOptions": {
"project": "./tsconfig.json" "project": "./tsconfig.json"
}, },
"rules": { "rules": {
"prettier/prettier": "error" "prettier/prettier": "error"
} },
"overrides": [
{
"files": ["*.ts", "*.tsx"],
"parser": "@typescript-eslint/parser"
}
]
} }

View File

@ -1,8 +1,8 @@
--- ---
name: '🐛 Bug Report' name: "🐛 Bug Report"
about: 'Report an unexpected problem or unintended behavior.' about: "Report an unexpected problem or unintended behavior."
title: '[Bug]' title: "[Bug]"
labels: 'bug' labels: "bug"
--- ---
<!-- <!--

View File

@ -1,8 +1,8 @@
--- ---
name: '📜 Documentation' name: "📜 Documentation"
about: 'Correct spelling errors, improvements or additions to documentation files (README, CONTRIBUTING...).' about: "Correct spelling errors, improvements or additions to documentation files (README, CONTRIBUTING...)."
title: '[Documentation]' title: "[Documentation]"
labels: 'documentation' labels: "documentation"
--- ---
<!-- Please make sure your issue has not already been fixed. --> <!-- Please make sure your issue has not already been fixed. -->

View File

@ -1,8 +1,8 @@
--- ---
name: '✨ Feature Request' name: "✨ Feature Request"
about: 'Suggest a new feature idea.' about: "Suggest a new feature idea."
title: '[Feature]' title: "[Feature]"
labels: 'feature request' labels: "feature request"
--- ---
<!-- Please make sure your issue has not already been fixed. --> <!-- Please make sure your issue has not already been fixed. -->

View File

@ -1,8 +1,8 @@
--- ---
name: '🔧 Improvement' name: "🔧 Improvement"
about: 'Improve structure/format/performance/refactor/tests of the code.' about: "Improve structure/format/performance/refactor/tests of the code."
title: '[Improvement]' title: "[Improvement]"
labels: 'improvement' labels: "improvement"
--- ---
<!-- Please make sure your issue has not already been fixed. --> <!-- Please make sure your issue has not already been fixed. -->

View File

@ -1,8 +1,8 @@
--- ---
name: '🙋 Question' name: "🙋 Question"
about: 'Further information is requested.' about: "Further information is requested."
title: '[Question]' title: "[Question]"
labels: 'question' labels: "question"
--- ---
### Question ### Question

View File

@ -1,27 +0,0 @@
name: 'Analyze'
on:
push:
branches: [develop]
pull_request:
branches: [master, develop]
jobs:
analyze:
runs-on: 'ubuntu-latest'
strategy:
fail-fast: false
matrix:
language: ['javascript']
steps:
- uses: 'actions/checkout@v3.5.3'
- name: 'Initialize CodeQL'
uses: 'github/codeql-action/init@v2'
with:
languages: ${{ matrix.language }}
- name: 'Perform CodeQL Analysis'
uses: 'github/codeql-action/analyze@v2'

View File

@ -1,4 +1,4 @@
name: 'Build' name: "Build"
on: on:
push: push:
@ -8,18 +8,18 @@ on:
jobs: jobs:
build: build:
runs-on: 'ubuntu-latest' runs-on: "ubuntu-latest"
steps: steps:
- uses: 'actions/checkout@v3.5.3' - uses: "actions/checkout@v4.1.1"
- name: 'Setup Node.js' - name: "Setup Node.js"
uses: 'actions/setup-node@v3.6.0' uses: "actions/setup-node@v4.0.1"
with: with:
node-version: '18.x' node-version: "20.x"
cache: 'npm' cache: "npm"
- name: 'Install dependencies' - name: "Install dependencies"
run: 'npm clean-install' run: "npm clean-install"
- name: 'Build' - name: "Build"
run: 'npm run build' run: "npm run build"

View File

@ -1,4 +1,4 @@
name: 'Lint' name: "Lint"
on: on:
push: push:
@ -8,35 +8,35 @@ on:
jobs: jobs:
lint: lint:
runs-on: 'ubuntu-latest' runs-on: "ubuntu-latest"
steps: steps:
- uses: 'actions/checkout@v3.5.3' - uses: "actions/checkout@v4.1.1"
- name: 'Setup Node.js' - name: "Setup Node.js"
uses: 'actions/setup-node@v3.6.0' uses: "actions/setup-node@v4.0.1"
with: with:
node-version: '18.x' node-version: "20.x"
cache: 'npm' cache: "npm"
- name: 'Install dependencies' - name: "Install dependencies"
run: 'npm clean-install' run: "npm clean-install"
- name: 'lint:commit' - name: "lint:commit"
run: 'npm run lint:commit -- --to "${{ github.sha }}"' run: 'npm run lint:commit -- --to "${{ github.sha }}"'
- name: 'lint:editorconfig' - name: "lint:editorconfig"
run: 'npm run lint:editorconfig' run: "npm run lint:editorconfig"
- name: 'lint:markdown' - name: "lint:markdown"
run: 'npm run lint:markdown' run: "npm run lint:markdown"
- name: 'lint:eslint' - name: "lint:eslint"
run: 'npm run lint:eslint' run: "npm run lint:eslint"
- name: 'lint:prettier' - name: "lint:prettier"
run: 'npm run lint:prettier' run: "npm run lint:prettier"
- name: 'lint:dotenv' - name: "lint:dotenv"
uses: 'dotenv-linter/action-dotenv-linter@v2' uses: "dotenv-linter/action-dotenv-linter@v2.18.0"
with: with:
github_token: ${{ secrets.github_token }} github_token: ${{ secrets.github_token }}

View File

@ -1,4 +1,4 @@
name: 'Release' name: "Release"
on: on:
push: push:
@ -6,31 +6,31 @@ on:
jobs: jobs:
release: release:
runs-on: 'ubuntu-latest' runs-on: "ubuntu-latest"
steps: steps:
- uses: 'actions/checkout@v3.5.3' - uses: "actions/checkout@v4.1.1"
with: with:
fetch-depth: 0 fetch-depth: 0
persist-credentials: false persist-credentials: false
- name: 'Import GPG key' - name: "Import GPG key"
uses: 'crazy-max/ghaction-import-gpg@v5.3.0' uses: "crazy-max/ghaction-import-gpg@v6.0.0"
with: with:
gpg_private_key: ${{ secrets.GPG_PRIVATE_KEY }} gpg_private_key: ${{ secrets.GPG_PRIVATE_KEY }}
git_user_signingkey: true git_user_signingkey: true
git_commit_gpgsign: true git_commit_gpgsign: true
- name: 'Setup Node.js' - name: "Setup Node.js"
uses: 'actions/setup-node@v3.6.0' uses: "actions/setup-node@v4.0.1"
with: with:
node-version: '18.x' node-version: "20.x"
cache: 'npm' cache: "npm"
- name: 'Install dependencies' - name: "Install dependencies"
run: 'npm clean-install' run: "npm clean-install"
- name: 'Release' - name: "Release"
run: 'npm run release' run: "npm run release"
env: env:
GH_TOKEN: ${{ secrets.GH_TOKEN }} GH_TOKEN: ${{ secrets.GH_TOKEN }}
GIT_COMMITTER_NAME: ${{ secrets.GIT_NAME }} GIT_COMMITTER_NAME: ${{ secrets.GIT_NAME }}

View File

@ -1,4 +1,4 @@
name: 'Test' name: "Test"
on: on:
push: push:
@ -8,41 +8,41 @@ on:
jobs: jobs:
test-unit: test-unit:
runs-on: 'ubuntu-latest' runs-on: "ubuntu-latest"
steps: steps:
- uses: 'actions/checkout@v3.5.3' - uses: "actions/checkout@v4.1.1"
- name: 'Setup Node.js' - name: "Setup Node.js"
uses: 'actions/setup-node@v3.6.0' uses: "actions/setup-node@v4.0.1"
with: with:
node-version: '18.x' node-version: "20.x"
cache: 'npm' cache: "npm"
- name: 'Install dependencies' - name: "Install dependencies"
run: 'npm clean-install' run: "npm clean-install"
- name: 'Unit Test' - name: "Unit Test"
run: 'npm run test:unit' run: "npm run test:unit"
test-e2e: test-e2e:
runs-on: 'ubuntu-latest' runs-on: "ubuntu-latest"
steps: steps:
- uses: 'actions/checkout@v3.5.3' - uses: "actions/checkout@v4.1.1"
- name: 'Setup Node.js' - name: "Setup Node.js"
uses: 'actions/setup-node@v3.6.0' uses: "actions/setup-node@v4.0.1"
with: with:
node-version: '18.x' node-version: "20.x"
cache: 'npm' cache: "npm"
- name: 'Install dependencies' - name: "Install dependencies"
run: 'npm clean-install' run: "npm clean-install"
- name: 'Build' - name: "Build"
run: 'npm run build' run: "npm run build"
- name: 'html-w3c-validator' - name: "html-w3c-validator"
run: 'npm run test:html-w3c-validator' run: "npm run test:html-w3c-validator"
- name: 'End To End (e2e) Test' - name: "End To End (e2e) Test"
run: 'npm run test:e2e' run: "npm run test:e2e"

3
.gitignore vendored
View File

@ -12,9 +12,6 @@ out
build build
dist dist
public/curriculum-vitae public/curriculum-vitae
# PWA
public/workbox-*.js
public/sw.js
# testing # testing
coverage coverage

View File

@ -1,13 +1,13 @@
image: 'gitpod/workspace-full' image: "gitpod/workspace-full"
tasks: tasks:
- before: 'cp .env.example .env' - before: "cp .env.example .env"
init: 'npm install' init: "npm clean-install"
command: 'npm run dev' command: "npm run dev"
ports: ports:
- port: 3000 - port: 3000
onOpen: 'open-preview' onOpen: "open-preview"
github: github:
prebuilds: prebuilds:

View File

@ -1,8 +1,4 @@
{ {
"urls": [ "urls": ["http://127.0.0.1:3000/", "http://127.0.0.1:3000/blog"],
"http://127.0.0.1:3000/",
"http://127.0.0.1:3000/blog",
"http://127.0.0.1:3000/blog/hello-world"
],
"files": ["./public/curriculum-vitae/index.html"] "files": ["./public/curriculum-vitae/index.html"]
} }

View File

@ -1,6 +1,4 @@
{ {
"*": ["editorconfig-checker"], "**/*": ["prettier --write --ignore-unknown", "editorconfig-checker"],
"*.{js,jsx,ts,tsx}": ["prettier --write", "eslint --fix"], "*.{md,mdx}": ["markdownlint-cli2 --fix"]
"*.{css,scss,sass,json,jsonc,yml,yaml}": ["prettier --write"],
"*.{md,mdx}": ["prettier --write", "markdownlint-cli2 --fix"]
} }

View File

@ -1,6 +1,3 @@
{ {
"singleQuote": true, "semi": false
"jsxSingleQuote": true,
"semi": false,
"trailingComma": "none"
} }

View File

@ -5,7 +5,7 @@
"editor.defaultFormatter": "esbenp.prettier-vscode", "editor.defaultFormatter": "esbenp.prettier-vscode",
"editor.formatOnSave": true, "editor.formatOnSave": true,
"editor.codeActionsOnSave": { "editor.codeActionsOnSave": {
"source.fixAll": true "source.fixAll": "explicit"
}, },
"eslint.options": { "eslint.options": {
"ignorePath": ".gitignore" "ignorePath": ".gitignore"

View File

@ -33,8 +33,8 @@ The commit message guidelines adheres to [Conventional Commits](https://www.conv
### Prerequisites ### Prerequisites
- [Node.js](https://nodejs.org/) >= 16.0.0 - [Node.js](https://nodejs.org/) >= 20.0.0
- [npm](https://www.npmjs.com/) >= 8.0.0 - [npm](https://www.npmjs.com/) >= 10.0.0
### Installation ### Installation
@ -49,7 +49,7 @@ cd theoludwig
cp .env.example .env cp .env.example .env
# Install # Install
npm install npm clean-install
``` ```
### Local Development environment ### Local Development environment

View File

@ -1,21 +1,28 @@
FROM node:18.16.1 AS builder-dependencies FROM node:20.10.0 AS builder-dependencies
WORKDIR /usr/src/application WORKDIR /usr/src/application
COPY ./package*.json ./ COPY ./package*.json ./
RUN npm clean-install RUN npm clean-install
FROM node:18.16.1 AS builder FROM node:20.10.0 AS builder
ENV NEXT_TELEMETRY_DISABLED=1
ENV IS_STANDALONE=true
WORKDIR /usr/src/application WORKDIR /usr/src/application
COPY --from=builder-dependencies /usr/src/application/node_modules ./node_modules COPY --from=builder-dependencies /usr/src/application/node_modules ./node_modules
COPY ./ ./ COPY ./ ./
RUN npm run build RUN npm run build
FROM gcr.io/distroless/nodejs18-debian11:latest AS runner FROM node:20.10.0-slim AS runner
WORKDIR /usr/src/application
ENV NODE_ENV=production ENV NODE_ENV=production
ENV HOSTNAME=0.0.0.0
ENV NEXT_TELEMETRY_DISABLED=1 ENV NEXT_TELEMETRY_DISABLED=1
COPY --from=builder /usr/src/application/.next/standalone ./ ENV IS_STANDALONE=true
COPY --from=builder /usr/src/application/.next/static ./.next/static WORKDIR /usr/src/application
COPY --from=builder /usr/src/application/public ./public RUN addgroup --system --gid 1001 nodejs && adduser --system --uid 1001 applicationrunner
COPY --from=builder /usr/src/application/locales ./locales USER applicationrunner
COPY --from=builder /usr/src/application/next.config.js ./next.config.js COPY --from=builder-dependencies --chown=applicationrunner:nodejs /usr/src/application/node_modules ./node_modules
COPY --from=builder --chown=applicationrunner:nodejs /usr/src/application/.next/standalone ./
COPY --from=builder --chown=applicationrunner:nodejs /usr/src/application/.next/static ./.next/static
COPY --from=builder --chown=applicationrunner:nodejs /usr/src/application/public ./public
COPY --from=builder --chown=applicationrunner:nodejs /usr/src/application/i18n/translations ./i18n/translations
COPY --from=builder --chown=applicationrunner:nodejs /usr/src/application/next.config.js ./next.config.js
CMD ["./server.js"] CMD ["./server.js"]

View File

@ -1,4 +1,4 @@
MIT License # MIT License
Copyright (c) Théo LUDWIG Copyright (c) Théo LUDWIG

View File

@ -25,7 +25,7 @@
"pronouns": "He/Him", "pronouns": "He/Him",
"birthDate": "31/03/2003", "birthDate": "31/03/2003",
"nationality": "Alsace, France", "nationality": "Alsace, France",
"interests": ["Open-Source enthusiast", "Passionate about High-Tech"], "interests": ["Developer Full Stack", "Open-Source enthusiast"],
"skills": { "skills": {
"programmingLanguages": ["JavaScript/TypeScript", "Python", "C/C++", "PHP"], "programmingLanguages": ["JavaScript/TypeScript", "Python", "C/C++", "PHP"],
"frontend": ["HTML", "CSS", "Tailwind CSS", "React.js/Next.js"], "frontend": ["HTML", "CSS", "Tailwind CSS", "React.js/Next.js"],
@ -40,6 +40,6 @@
## 📈 Statistics ## 📈 Statistics
<p align=center> <p align=center>
<img height=175 align="center" src="https://github-readme-stats.vercel.app/api?username=theoludwig&show_icons=true&theme=dark" /> <img height=175 align="center" src="https://github-readme-stats.vercel.app/api?username=theoludwig&show_icons=true&theme=dark" alt="Théo LUDWIG's GitHub Stats" />
<img height=175 align="center" src="https://github-readme-stats.vercel.app/api/top-langs/?username=theoludwig&hide=html,css,javascript&langs_count=8&layout=compact&theme=dark" /> <img height=175 align="center" src="https://github-readme-stats.vercel.app/api/top-langs/?username=theoludwig&hide=html,css,javascript&langs_count=8&layout=compact&theme=dark" alt="Théo LUDWIG's Programming Languages" />
</p> </p>

View File

@ -0,0 +1,11 @@
import { Loader } from "@/components/design/Loader"
const Loading = (): JSX.Element => {
return (
<main className="flex flex-col flex-1 items-center justify-center">
<Loader />
</main>
)
}
export default Loading

44
app/blog/[slug]/page.tsx Normal file
View File

@ -0,0 +1,44 @@
import type { Metadata } from "next"
import { notFound } from "next/navigation"
import "katex/dist/katex.min.css"
import { getBlogPostBySlug } from "@/blog/blog"
import { BlogPost } from "@/blog/BlogPost"
interface BlogPostPageProps {
params: {
slug: string
}
}
export const generateMetadata = async (
props: BlogPostPageProps,
): Promise<Metadata> => {
const blogPost = await getBlogPostBySlug(props.params.slug)
if (blogPost == null) {
return notFound()
}
const title = `${blogPost.frontmatter.title} | Théo LUDWIG`
const description = blogPost.frontmatter.description
return {
title,
description,
openGraph: {
title,
description,
},
twitter: {
title,
description,
},
}
}
const BlogPostPage = async (props: BlogPostPageProps): Promise<JSX.Element> => {
const { params } = props
return <BlogPost slug={params.slug} />
}
export default BlogPostPage

11
app/blog/loading.tsx Normal file
View File

@ -0,0 +1,11 @@
import { Loader } from "@/components/design/Loader"
const Loading = (): JSX.Element => {
return (
<main className="flex flex-col flex-1 items-center justify-center">
<Loader />
</main>
)
}
export default Loading

40
app/blog/page.tsx Normal file
View File

@ -0,0 +1,40 @@
import { Suspense } from "react"
import type { Metadata } from "next"
import { BlogPosts } from "@/blog/BlogPosts"
import { Loader } from "@/components/design/Loader"
const title = "Blog | Théo LUDWIG"
const description =
"The latest news about my journey of learning computer science."
export const metadata: Metadata = {
title,
description,
openGraph: {
title,
description,
},
twitter: {
title,
description,
},
}
const BlogPage = async (): Promise<JSX.Element> => {
return (
<main className="flex flex-1 flex-col flex-wrap items-center">
<div className="mt-10 flex flex-col items-center">
<h1 className="text-4xl font-semibold">Blog</h1>
<p className="mt-6 text-center" data-cy="blog-post-date">
{description}
</p>
</div>
<Suspense fallback={<Loader className="mt-8" />}>
<BlogPosts />
</Suspense>
</main>
)
}
export default BlogPage

32
app/error.tsx Normal file
View File

@ -0,0 +1,32 @@
"use client"
import { useEffect } from "react"
export interface ErrorHandlingProps {
error: Error
}
const ErrorHandling = (props: ErrorHandlingProps): JSX.Element => {
const { error } = props
useEffect(() => {
console.error(error)
}, [error])
return (
<main className="flex flex-col flex-1 items-center justify-center">
<h1 className="my-6 text-4xl font-semibold">
Error{" "}
<span
className="text-yellow dark:text-yellow-dark"
data-cy="status-code"
>
500
</span>
</h1>
<p className="text-center text-lg">Server error</p>
</main>
)
}
export default ErrorHandling

View File

@ -16,7 +16,7 @@
} }
.prose [id]::before { .prose [id]::before {
content: ''; content: "";
display: block; display: block;
height: 90px; height: 90px;
margin-top: -90px; margin-top: -90px;
@ -39,9 +39,9 @@
.prose code { .prose code {
color: #ce9178; color: #ce9178;
} }
.prose :where(code):not(:where([class~='not-prose'] *))::before, .prose :where(code):not(:where([class~="not-prose"] *))::before,
.prose :where(code):not(:where([class~='not-prose'] *))::after { .prose :where(code):not(:where([class~="not-prose"] *))::after {
content: ''; content: "";
} }
.shiki { .shiki {
white-space: pre-wrap !important; white-space: pre-wrap !important;

80
app/layout.tsx Normal file
View File

@ -0,0 +1,80 @@
import type { Metadata } from "next"
import classNames from "clsx"
import "@fontsource/montserrat/400.css"
import "@fontsource/montserrat/600.css"
import "./globals.css"
import { Header } from "@/components/Header"
import { Footer } from "@/components/Footer"
import { getI18n } from "@/i18n/i18n.server"
import { getTheme } from "@/theme/theme.server"
const title = "Théo LUDWIG"
const description =
"Théo LUDWIG - Developer Full Stack • Open-Source enthusiast"
const image = "/images/icon-96x96.png"
const url = new URL("https://theoludwig.fr")
const locale = "fr-FR, en-US"
export const metadata: Metadata = {
title,
description,
metadataBase: url,
openGraph: {
title,
description,
url,
siteName: title,
images: [
{
url: image,
width: 96,
height: 96,
},
],
locale,
type: "website",
},
icons: {
icon: "/images/icon-96x96.png",
},
twitter: {
card: "summary",
title,
description,
images: [image],
},
}
interface RootLayoutProps {
children: React.ReactNode
}
const RootLayout = (props: RootLayoutProps): JSX.Element => {
const { children } = props
const i18n = getI18n()
const theme = getTheme()
return (
<html
lang={i18n.locale}
className={classNames({
dark: theme === "dark",
light: theme === "light",
})}
style={{
colorScheme: theme,
}}
>
<body className="bg-white font-headline text-black dark:bg-black dark:text-white flex flex-col min-h-screen">
<Header />
{children}
<Footer />
</body>
</html>
)
}
export default RootLayout

11
app/loading.tsx Normal file
View File

@ -0,0 +1,11 @@
import { Loader } from "@/components/design/Loader"
const Loading = (): JSX.Element => {
return (
<main className="flex flex-col flex-1 items-center justify-center">
<Loader />
</main>
)
}
export default Loading

32
app/not-found.tsx Normal file
View File

@ -0,0 +1,32 @@
import Link from "next/link"
import { getI18n } from "@/i18n/i18n.server"
const NotFound = (): JSX.Element => {
const i18n = getI18n()
return (
<main className="flex flex-col flex-1 items-center justify-center">
<h1 className="my-6 text-4xl font-semibold">
{i18n.translate("errors.error")}{" "}
<span
className="text-yellow dark:text-yellow-dark"
data-cy="status-code"
>
404
</span>
</h1>
<p className="text-center text-lg">
{i18n.translate("errors.not-found")}{" "}
<Link
href="/"
className="text-yellow hover:underline dark:text-yellow-dark"
>
{i18n.translate("errors.return-to-home-page")}
</Link>
</p>
</main>
)
}
export default NotFound

59
app/page.tsx Normal file
View File

@ -0,0 +1,59 @@
import { RevealFade } from "@/components/design/RevealFade"
import { Section } from "@/components/design/Section"
import { Interests } from "@/components/Interests"
import { Portfolio } from "@/components/Portfolio"
import { Profile } from "@/components/Profile"
import { SocialMediaList } from "@/components/Profile/SocialMediaList"
import { Skills } from "@/components/Skills"
import { OpenSource } from "@/components/OpenSource"
import { getI18n } from "@/i18n/i18n.server"
const HomePage = (): JSX.Element => {
const i18n = getI18n()
return (
<main className="flex flex-col md:mx-auto md:max-w-4xl lg:max-w-7xl">
<Section isMain id="about">
<Profile />
<SocialMediaList />
</Section>
<RevealFade>
<Section
id="interests"
heading={i18n.translate("home.interests.title")}
>
<Interests />
</Section>
</RevealFade>
<RevealFade>
<Section
id="skills"
heading={i18n.translate("home.skills.title")}
withoutShadowContainer
>
<Skills />
</Section>
</RevealFade>
<RevealFade>
<Section
id="portfolio"
heading={i18n.translate("home.portfolio.title")}
withoutShadowContainer
>
<Portfolio />
</Section>
</RevealFade>
<RevealFade>
<Section id="open-source" heading="Open source" withoutShadowContainer>
<OpenSource />
</Section>
</RevealFade>
</main>
)
}
export default HomePage

35
blog/BlogPost.tsx Normal file
View File

@ -0,0 +1,35 @@
import { notFound } from "next/navigation"
import date from "date-and-time"
import "katex/dist/katex.min.css"
import { getBlogPostBySlug } from "@/blog/blog"
import { BlogPostContent } from "@/blog/BlogPostContent"
export interface BlogPostProps {
slug: string
}
export const BlogPost = async (props: BlogPostProps): Promise<JSX.Element> => {
const { slug } = props
const blogPost = await getBlogPostBySlug(slug)
if (blogPost == null) {
return notFound()
}
return (
<main className="break-wrap-words flex flex-1 flex-col flex-wrap items-center justify-center">
<div className="my-10 flex flex-col items-center text-center">
<h1 className="text-3xl font-semibold">{blogPost.frontmatter.title}</h1>
<p className="mt-2" data-cy="blog-post-date">
{date.format(
new Date(blogPost.frontmatter.publishedOn),
"DD/MM/YYYY",
)}
</p>
</div>
<BlogPostContent content={blogPost.content} />
</main>
)
}

33
blog/BlogPostComments.tsx Normal file
View File

@ -0,0 +1,33 @@
"use client"
import Giscus from "@giscus/react"
import { useTheme } from "@/theme/theme.client"
import type { CookiesStore } from "@/utils/constants"
interface BlogPostCommentsProps {
cookiesStore: CookiesStore
}
export const BlogPostComments = (props: BlogPostCommentsProps): JSX.Element => {
const { cookiesStore } = props
const theme = useTheme(cookiesStore)
return (
<Giscus
id="comments"
repo="theoludwig/theoludwig"
repoId="MDEwOlJlcG9zaXRvcnkzNTg5NDg1NDQ="
category="General"
categoryId="DIC_kwDOFWUewM4CQ_WK"
mapping="pathname"
reactionsEnabled="1"
emitMetadata="0"
inputPosition="top"
theme={theme}
lang="en"
loading="lazy"
/>
)
}

111
blog/BlogPostContent.tsx Normal file
View File

@ -0,0 +1,111 @@
import Image from "next/image"
import Link from "next/link"
import { cookies } from "next/headers"
import { FontAwesomeIcon } from "@fortawesome/react-fontawesome"
import { faLink } from "@fortawesome/free-solid-svg-icons"
import { MDXRemote } from "next-mdx-remote/rsc"
import { nodeTypes } from "@mdx-js/mdx"
import rehypeRaw from "rehype-raw"
import remarkGfm from "remark-gfm"
import rehypeSlug from "rehype-slug"
import remarkMath from "remark-math"
import rehypeKatex from "rehype-katex"
import { getHighlighter } from "shiki"
import "katex/dist/katex.min.css"
import { getTheme } from "@/theme/theme.server"
import { remarkSyntaxHighlightingPlugin } from "@/blog/remarkSyntaxHighlightingPlugin"
import { BlogPostComments } from "@/blog/BlogPostComments"
const Heading = (
props: React.DetailedHTMLProps<
React.HTMLAttributes<HTMLHeadingElement>,
HTMLHeadingElement
>,
): JSX.Element => {
const { children, id = "" } = props
return (
<h2 {...props} className="group">
<Link
href={`#${id}`}
className="invisible !text-black group-hover:visible dark:!text-white"
>
<FontAwesomeIcon className="mr-2 inline h-4 w-4" icon={faLink} />
</Link>
{children}
</h2>
)
}
export interface BlogPostContentProps {
content: string
}
export const BlogPostContent = async (
props: BlogPostContentProps,
): Promise<JSX.Element> => {
const { content } = props
const cookiesStore = cookies()
const theme = getTheme()
const highlighter = await getHighlighter({
theme: `${theme}-plus`,
})
return (
<div className="prose mb-10">
<div className="px-8">
<MDXRemote
source={content}
options={{
mdxOptions: {
remarkPlugins: [
remarkGfm,
[remarkSyntaxHighlightingPlugin, { highlighter }],
remarkMath,
],
rehypePlugins: [
rehypeSlug,
[rehypeRaw, { passThrough: nodeTypes }],
rehypeKatex,
],
},
}}
components={{
h1: Heading,
h2: Heading,
h3: Heading,
h4: Heading,
h5: Heading,
h6: Heading,
img: (properties) => {
const { src = "", alt = "Blog Image" } = properties
const source = src.replace("../../public/", "/")
return (
<span className="flex flex-col items-center justify-center">
<Image
src={source}
alt={alt}
width={1000}
height={1000}
className="h-auto w-auto"
/>
</span>
)
},
a: (props) => {
const { href = "" } = props
if (href.startsWith("#")) {
return <a {...props} />
}
return <a target="_blank" rel="noopener noreferrer" {...props} />
},
}}
/>
<BlogPostComments cookiesStore={cookiesStore.toString()} />
</div>
</div>
)
}

42
blog/BlogPosts.tsx Normal file
View File

@ -0,0 +1,42 @@
import Link from "next/link"
import date from "date-and-time"
import { ShadowContainer } from "@/components/design/ShadowContainer"
import { getBlogPosts } from "@/blog/blog"
export const BlogPosts = async (): Promise<JSX.Element> => {
const posts = await getBlogPosts()
return (
<div className="flex w-full items-center justify-center p-8">
<div className="w-[1600px]" data-cy="blog-posts">
{posts.map((post, index) => {
const postPublishedOn = date.format(
new Date(post.frontmatter.publishedOn),
"DD/MM/YYYY",
)
return (
<Link
href={`/blog/${post.slug}`}
key={index}
locale="en"
data-cy={post.slug}
>
<ShadowContainer className="cursor-pointer p-6 transition duration-200 ease-in-out hover:-translate-y-2">
<h2 data-cy="blog-post-title" className="text-xl font-semibold">
{post.frontmatter.title}
</h2>
<p data-cy="blog-post-date" className="mt-2">
{postPublishedOn}
</p>
<p data-cy="blog-post-description" className="mt-3">
{post.frontmatter.description}
</p>
</ShadowContainer>
</Link>
)
})}
</div>
</div>
)
}

65
blog/blog.ts Normal file
View File

@ -0,0 +1,65 @@
import fs from "node:fs"
import path from "node:path"
import { cache } from "react"
import matter from "gray-matter"
export const BLOG_POSTS_PATH = path.join(process.cwd(), "blog", "posts")
export interface FrontMatter {
title: string
description: string
isPublished: boolean
publishedOn: string
}
export interface BlogPost {
frontmatter: FrontMatter
slug: string
content: string
}
export const getBlogPosts = cache(async (): Promise<BlogPost[]> => {
const blogPosts = await fs.promises.readdir(BLOG_POSTS_PATH)
const blogPostsWithTime = await Promise.all(
blogPosts.map(async (blogPostFilename) => {
const [slug, extension] = blogPostFilename.split(".")
if (slug == null || extension == null) {
throw new Error("Invalid blog post filename.")
}
const blogPostPath = path.join(BLOG_POSTS_PATH, `${slug}.${extension}`)
const blogPostContent = await fs.promises.readFile(blogPostPath, {
encoding: "utf8",
})
const { data, content } = matter(blogPostContent) as unknown as {
data: FrontMatter
content: string
}
const date = new Date(data.publishedOn)
return {
slug,
content,
frontmatter: data,
time: date.getTime(),
}
}),
)
const blogPostsSortedByPublicationDate = blogPostsWithTime
.filter((post) => {
return post.frontmatter.isPublished
})
.sort((a, b) => {
return b.time - a.time
})
return blogPostsSortedByPublicationDate
})
export const getBlogPostBySlug = cache(
async (slug: string): Promise<BlogPost | undefined> => {
const blogPosts = await getBlogPosts()
const blogPost = blogPosts.find((blogPost) => {
return blogPost.slug === slug && blogPost.frontmatter.isPublished
})
return blogPost
},
)

View File

@ -1,8 +1,8 @@
--- ---
title: '🧼 Clean Code' title: "🧼 Clean Code"
description: 'What is "Clean Code", what are "Design Patterns", and why is it so important today? Tips and tricks to make your code more readable and maintainable in the long term.' description: 'What is "Clean Code", what are "Design Patterns", and why is it so important today? Tips and tricks to make your code more readable and maintainable in the long term.'
isPublished: true isPublished: true
publishedOn: '2022-02-23T08:00:18.758Z' publishedOn: "2022-02-23T08:00:18.758Z"
--- ---
Hello! 👋 Hello! 👋
@ -110,7 +110,7 @@ const transaction = charge(user, subscription)
```typescript ```typescript
interface Car { interface Car {
carModel: string carModel: string
carColor: 'red' | 'blue' | 'yellow' carColor: "red" | "blue" | "yellow"
} }
const printCar = (car: Car): void => { const printCar = (car: Car): void => {
console.log(`${car.carModel} (${car.carColor})`) console.log(`${car.carModel} (${car.carColor})`)
@ -122,7 +122,7 @@ const printCar = (car: Car): void => {
```typescript ```typescript
interface Car { interface Car {
model: string model: string
color: 'red' | 'blue' | 'yellow' color: "red" | "blue" | "yellow"
} }
const printCar = (car: Car): void => { const printCar = (car: Car): void => {
console.log(`${car.model} (${car.color})`) console.log(`${car.model} (${car.color})`)
@ -170,17 +170,17 @@ We have to keep it as simple as possible, not to implement features that are not
### Example (bad way) ### Example (bad way)
```typescript ```typescript
import fs from 'node:fs' import fs from "node:fs"
import path from 'node:path' import path from "node:path"
const createFile = async ( const createFile = async (
name: string, name: string,
isTemporary: boolean = false isTemporary: boolean = false,
): Promise<void> => { ): Promise<void> => {
if (isTemporary) { if (isTemporary) {
return await fs.promises.writeFile(path.join('temporary', name), '') return await fs.promises.writeFile(path.join("temporary", name), "")
} }
return await fs.promises.writeFile(name, '') return await fs.promises.writeFile(name, "")
} }
``` ```
@ -189,15 +189,15 @@ const createFile = async (
### Example (good way) ### Example (good way)
```typescript ```typescript
import fs from 'node:fs' import fs from "node:fs"
import path from 'node:path' import path from "node:path"
const createFile = async (name: string): Promise<void> => { const createFile = async (name: string): Promise<void> => {
await fs.promises.writeFile(name, '') await fs.promises.writeFile(name, "")
} }
const createTemporaryFile = async (name: string): Promise<void> => { const createTemporaryFile = async (name: string): Promise<void> => {
await createFile(path.join('temporary', name)) await createFile(path.join("temporary", name))
} }
``` ```

View File

@ -1,8 +1,8 @@
--- ---
title: '🗓️ Git version control: Ultimate Guide' title: "🗓️ Git version control: Ultimate Guide"
description: 'What is `git`, what are the most used commands, best practices, and tips and tricks. The Ultimate guide to master `git` in your daily workflow.' description: "What is `git`, what are the most used commands, best practices, and tips and tricks. The Ultimate guide to master `git` in your daily workflow."
isPublished: true isPublished: true
publishedOn: '2022-10-27T14:33:07.465Z' publishedOn: "2022-10-27T14:33:07.465Z"
--- ---
Hello! 👋 Hello! 👋
@ -122,6 +122,11 @@ git checkout <branch>
# Merge a branch into the current branch # Merge a branch into the current branch
git merge <branch> git merge <branch>
# Note: Merge creates a "Merge commit" when the base branch and the branch to merge have diverged (they have different commits).
# To avoid creating a "Merge commit", we can use rebase instead of merge.
git rebase --interactive <branch-to-rebase-on>
# Combine multiple commits of a branch into one for a merge # Combine multiple commits of a branch into one for a merge
git merge --squash <branch> git merge --squash <branch>
@ -145,6 +150,13 @@ git reset --soft <branch>
# Apply the changes introduced by some existing commits # Apply the changes introduced by some existing commits
# (by first being on the branch where you want to apply the commit) # (by first being on the branch where you want to apply the commit)
git cherry-pick <commit> git cherry-pick <commit>
# To list all commits that differ between two branches
git log <branch1>..<branch2> # commits in branch2 that are not in branch1 (branch2 ahead of branch1, branch2 behind branch1)
git log <branch2>..<branch1> # commits in branch1 that are not in branch2 (branch1 ahead of branch2, branch1 behind branch2)
# Summary of commit authors across all branches, excluding merge commits.
git shortlog --summary --numbered --all --no-merges
``` ```
## `.gitignore` file ## `.gitignore` file
@ -194,7 +206,7 @@ As we have seen in the [Get started with `git` and `.gitconfig` config file](#ge
That means that **anyone can create a commit with any name and email address and claim to be whoever they want** when they create a commit. That means that **anyone can create a commit with any name and email address and claim to be whoever they want** when they create a commit.
To avoid this, you can sign your commits with a <abbr title="GNU Privacy Guard">[GPG](https://gnupg.org/)</abbr> key. To avoid this, you can sign your commits with a [GNU Privacy Guard](https://gnupg.org/) (<abbr>gpg</abbr>) key.
You can find more information about signing commits in the [official documentation](https://git-scm.com/book/en/v2/Git-Tools-Signing-Your-Work). You can find more information about signing commits in the [official documentation](https://git-scm.com/book/en/v2/Git-Tools-Signing-Your-Work).

View File

@ -1,8 +1,8 @@
--- ---
title: '👋 Hello, world!' title: "👋 Hello, world!"
description: 'First post of the blog, introduction and explanation of how this blog is made.' description: "First post of the blog, introduction and explanation of how this blog is made."
isPublished: true isPublished: true
publishedOn: '2022-02-20T08:00:18.758Z' publishedOn: "2022-02-20T08:00:18.758Z"
--- ---
Hello, world! 👋 Hello, world! 👋

View File

@ -1,8 +1,8 @@
--- ---
title: '❌ Mistakes I made as a junior developer' title: "❌ Mistakes I made as a junior developer"
description: 'Here are mistakes I made when I started, to prevent you from making the same mistakes.' description: "Here are mistakes I made when I started, to prevent you from making the same mistakes."
isPublished: true isPublished: true
publishedOn: '2022-03-14T07:42:52.989Z' publishedOn: "2022-03-14T07:42:52.989Z"
--- ---
Hello! 👋 Hello! 👋
@ -41,13 +41,13 @@ Find the right balance, between abstraction and simple implementation, start sim
When you start a new project, you should focus on the core of the project, not on the details, to release as soon as possible, a working usable version of your project also called a [**Minimum Viable Product** (MVP)](https://en.wikipedia.org/wiki/Minimum_viable_product), it is better than a half-functioning, over-engineered project. When you start a new project, you should focus on the core of the project, not on the details, to release as soon as possible, a working usable version of your project also called a [**Minimum Viable Product** (MVP)](https://en.wikipedia.org/wiki/Minimum_viable_product), it is better than a half-functioning, over-engineered project.
I made this mistake while developing [Thream](https://thream.divlo.fr), your **open source** platform to stay close with your friends and communities, **talk**, chat, **collaborate**, share and **have fun**. I made this mistake while developing [Thream](https://thream.theoludwig.fr), your **open source** platform to stay close with your friends and communities, **talk**, chat, **collaborate**, share and **have fun**.
Basically, I thought it was cool, to do a "big" v1.0.0 release with a lot of features, but in fact, it was not, because I could not even show what I was developing (to the end-users, not technical people) as I was making multiple features at the same time and also mainly focused on the **REST API** side and not at all the **website (frontend)**. Basically, I thought it was cool, to do a "big" v1.0.0 release with a lot of features, but in fact, it was not, because I could not even show what I was developing (to the end-users, not technical people) as I was making multiple features at the same time and also mainly focused on the **REST API** side and not at all the **website (frontend)**.
What I recommend you to do is to start with a **v1.0.0** release as soon as possible with the minimum required features needed for your project idea, and then gradually add new features and release new versions. What I recommend you to do is to start with a **v1.0.0** release as soon as possible with the minimum required features needed for your project idea, and then gradually add new features and release new versions.
In my example for [Thream](https://thream.divlo.fr), I could release a v1.0.0 without these features: In my example for [Thream](https://thream.theoludwig.fr), I could release a v1.0.0 without these features:
- English/French translation (could be only English) - English/French translation (could be only English)
- Light/Dark theme (could be only Dark) - Light/Dark theme (could be only Dark)
@ -55,7 +55,7 @@ In my example for [Thream](https://thream.divlo.fr), I could release a v1.0.0 wi
- User public profile - User public profile
- Channels (maybe could be only one channel per guild to start with) - Channels (maybe could be only one channel per guild to start with)
And probably more, what was really required with [Thream](https://thream.divlo.fr), is that users could authenticate, create a community of friends, and then they could communicate with each other with messages in real-time, really that was enough. And probably more, what was really required with [Thream](https://thream.theoludwig.fr), is that users could authenticate, create a community of friends, and then they could communicate with each other with messages in real-time, really that was enough.
And then with this basis, I could release, v1.1.0, v1.2.0 etc. with more features, and release new versions more often to show the progress of the project, it is also more motivating to have users testing our project and to **get feedback sooner**. And then with this basis, I could release, v1.1.0, v1.2.0 etc. with more features, and release new versions more often to show the progress of the project, it is also more motivating to have users testing our project and to **get feedback sooner**.

View File

@ -1,8 +1,8 @@
--- ---
title: '🧠 Programming Challenges' title: "🧠 Programming Challenges"
description: 'What are Programming Challenges and Competitive Programming and an introduction to Time/Space Complexity with Big O Notation.' description: "What are Programming Challenges and Competitive Programming and an introduction to Time/Space Complexity with Big O Notation."
isPublished: true isPublished: true
publishedOn: '2023-05-21T10:20:18.837Z' publishedOn: "2023-05-21T10:20:18.837Z"
--- ---
Hello! 👋 Hello! 👋
@ -216,7 +216,7 @@ $$
#### Complexity Classes (from fastest to slowest) #### Complexity Classes (from fastest to slowest)
![Big O Notation](../public/images/posts/programming-challenges/big-o-chart-notations.webp) ![Big O Notation](../../public/images/posts/programming-challenges/big-o-chart-notations.webp)
Here is a list of classes of functions that are commonly encountered when analyzing the running time of an algorithm. Here is a list of classes of functions that are commonly encountered when analyzing the running time of an algorithm.

View File

@ -1,19 +1,19 @@
--- ---
title: '🟢 Thream v1.0.0' title: "🟢 Thream v1.0.0"
description: 'Your open source platform to stay close with your friends and communities, talk, chat, collaborate, share and have fun.' description: "Your open source platform to stay close with your friends and communities, talk, chat, collaborate, share and have fun."
isPublished: true isPublished: true
publishedOn: '2022-04-11T10:24:55.206Z' publishedOn: "2022-04-11T10:24:55.206Z"
--- ---
Hello! 👋 Hello! 👋
After months of hard work, [Thream v1.0.0](https://thream.divlo.fr/) has been released! 🎉 After months of hard work, [Thream v1.0.0](https://thream.theoludwig.fr/) has been released! 🎉
[**Thream**](https://thream.divlo.fr/) is your open-source platform to stay close with your friends and communities, talk, chat, collaborate, share and have fun. [**Thream**](https://thream.theoludwig.fr/) is your open-source platform to stay close with your friends and communities, talk, chat, collaborate, share and have fun.
## Presentation ## Presentation
[**Thream**](https://thream.divlo.fr/) is a social network to stay close with your friends and communities to talk, chat, collaborate and share. [**Thream**](https://thream.theoludwig.fr/) is a social network to stay close with your friends and communities to talk, chat, collaborate and share.
The project is largely inspired by [Discord](https://discord.com), a proprietary instant messaging service, but differentiates itself by its **non-profit open source philosophy** and will integrate special features. The project is largely inspired by [Discord](https://discord.com), a proprietary instant messaging service, but differentiates itself by its **non-profit open source philosophy** and will integrate special features.
@ -21,9 +21,9 @@ The source code is available on [GitHub](https://github.com/Thream).
The idea is that a user can create an account to authenticate with an email address, and a password, or directly use an account from another platform (currently supported: Google, GitHub, Discord). Once the user is authenticated, he/she can create and join "guilds", in other words communities, in order to discuss with other people in several channels to group discussions talking about the same subject. The idea is that a user can create an account to authenticate with an email address, and a password, or directly use an account from another platform (currently supported: Google, GitHub, Discord). Once the user is authenticated, he/she can create and join "guilds", in other words communities, in order to discuss with other people in several channels to group discussions talking about the same subject.
![The Thream app on a community page](../public/images/posts/thream-v1-0-0/thream-ui.png) ![The Thream app on a community page](../../public/images/posts/thream-v1-0-0/thream-ui.png)
[**Thream**](https://thream.divlo.fr/) is a website that works on any recent browser, accessible on [thream.divlo.fr](https://thream.divlo.fr/). [**Thream**](https://thream.theoludwig.fr/) is a website that works on any recent browser, accessible on [thream.theoludwig.fr](https://thream.theoludwig.fr/).
## History ## History
@ -31,9 +31,9 @@ The idea for the project has existed since May 13, 2020, symbolized by a [public
The main goal is to put into **practice knowledge in web development** and computer science in general on a concrete project that can **easily evolve over time** where you can add many features. The main goal is to put into **practice knowledge in web development** and computer science in general on a concrete project that can **easily evolve over time** where you can add many features.
The development of the project begins under the name of **SocialProject**, on August 20, 2020, with colors close to the image of Divlo. The development of the project begins under the name of **SocialProject**, on August 20, 2020.
![SocialProject](../public/images/posts/thream-v1-0-0/social-project.jpg) ![SocialProject](../../public/images/posts/thream-v1-0-0/social-project.jpg)
When I started the project, I had little knowledge of database design, real-time management or the architecture of such a large <abbr title="Information Technology">IT</abbr> project, so this will be accompanied by many technical problems, to which we will need to find appropriate solutions. When I started the project, I had little knowledge of database design, real-time management or the architecture of such a large <abbr title="Information Technology">IT</abbr> project, so this will be accompanied by many technical problems, to which we will need to find appropriate solutions.
@ -53,7 +53,7 @@ Since the project is mainly developed during free time (mainly on weekends), the
- The **client** part, called **frontend**, what **the user sees on the screen**, such as forms, buttons and all the **graphic elements** with which the user can interact from a browser. - The **client** part, called **frontend**, what **the user sees on the screen**, such as forms, buttons and all the **graphic elements** with which the user can interact from a browser.
![HTTP Communication Schema](../public/images/posts/thream-v1-0-0/http-communication.png) ![HTTP Communication Schema](../../public/images/posts/thream-v1-0-0/http-communication.png)
This design allows the separation between the client and the server, as long as they both structure their communication according to the <abbr title="Representational state transfer">REST</abbr> architectural guidelines, using the <abbr title="Hypertext Transfer Protocol">HTTP</abbr> protocol, they will be able to communicate with each other, which makes it possible to work independently on the backend and on the frontend using different technologies and skills, really useful in teamwork. This design allows the separation between the client and the server, as long as they both structure their communication according to the <abbr title="Representational state transfer">REST</abbr> architectural guidelines, using the <abbr title="Hypertext Transfer Protocol">HTTP</abbr> protocol, they will be able to communicate with each other, which makes it possible to work independently on the backend and on the frontend using different technologies and skills, really useful in teamwork.
@ -116,4 +116,4 @@ The other interest of the project is that it is completely **open-source**, and
Feel free to give feebacks and suggestions to improve the project, and to report any bug you find. Feel free to give feebacks and suggestions to improve the project, and to report any bug you find.
**Thream** is available: [**thream.divlo.fr**](https://thream.divlo.fr/). **Thream** is available: [**thream.theoludwig.fr**](https://thream.theoludwig.fr/).

View File

@ -1,7 +1,7 @@
import type { Plugin, Transformer } from 'unified' import type { Plugin, Transformer } from "unified"
import type { Literal, Node } from 'unist' import type { Literal, Node } from "unist"
import { visit } from 'unist-util-visit' import { visit } from "unist-util-visit"
import type { Highlighter } from 'shiki' import type { Highlighter } from "shiki"
export interface RemarkSyntaxHighlightingPluginOptions { export interface RemarkSyntaxHighlightingPluginOptions {
highlighter: Highlighter highlighter: Highlighter
@ -20,11 +20,11 @@ export const remarkSyntaxHighlightingPlugin: Plugin<
Literal Literal
> = (options) => { > = (options) => {
const transformer: Transformer<RemarkSyntaxHighlightingNode> = (tree) => { const transformer: Transformer<RemarkSyntaxHighlightingNode> = (tree) => {
visit<RemarkSyntaxHighlightingNode, string>(tree, 'code', (node) => { visit<RemarkSyntaxHighlightingNode, string>(tree, "code", (node) => {
node.type = 'html' node.type = "html"
node.children = undefined node.children = undefined
node.value = options.highlighter.codeToHtml(node.value, { node.value = options.highlighter.codeToHtml(node.value, {
lang: node.lang lang: node.lang,
}) })
}) })
} }

View File

@ -1,45 +0,0 @@
import useTranslation from 'next-translate/useTranslation'
import Link from 'next/link'
import type { FooterProps } from './Footer'
import { Footer } from './Footer'
import { Header } from './Header'
export interface ErrorPageProps extends FooterProps {
statusCode: number
message: string
}
export const ErrorPage: React.FC<ErrorPageProps> = (props) => {
const { message, statusCode, version } = props
const { t } = useTranslation()
return (
<>
<div className='flex h-screen flex-col pt-0'>
<Header showLanguage />
<main className='flex min-w-full flex-1 flex-col items-center justify-center'>
<h1 className='my-6 text-4xl font-semibold'>
{t('errors:error')}{' '}
<span
className='text-yellow dark:text-yellow-dark'
data-cy='status-code'
>
{statusCode}
</span>
</h1>
<p className='text-center text-lg'>
{message}{' '}
<Link
href='/'
className='text-yellow hover:underline dark:text-yellow-dark'
>
{t('errors:return-to-home-page')}
</Link>
</p>
</main>
<Footer version={version} />
</div>
</>
)
}

View File

@ -1,42 +0,0 @@
import { useMemo } from 'react'
import Link from 'next/link'
import useTranslation from 'next-translate/useTranslation'
export interface FooterProps {
version: string
}
export const Footer: React.FC<FooterProps> = (props) => {
const { t } = useTranslation()
const { version } = props
const versionLink = useMemo(() => {
return `https://github.com/theoludwig/theoludwig/releases/tag/v${version}`
}, [version])
return (
<footer className='flex flex-col items-center justify-center border-t-2 border-gray-600 bg-white py-6 text-lg dark:border-gray-400 dark:bg-black'>
<p>
<Link
href='/'
className='text-yellow hover:underline dark:text-yellow-dark'
>
Théo LUDWIG
</Link>{' '}
| {t('common:all-rights-reserved')}
</p>
<p className='mt-1'>
Version{' '}
<a
data-cy='version-link'
className='text-yellow hover:underline dark:text-yellow-dark'
href={versionLink}
target='_blank'
rel='noopener noreferrer'
>
{version}
</a>
</p>
</footer>
)
}

View File

@ -0,0 +1,19 @@
import Link from "next/link"
import { getI18n } from "@/i18n/i18n.server"
export const FooterText = (): JSX.Element => {
const i18n = getI18n()
return (
<p>
<Link
href="/"
className="text-yellow hover:underline dark:text-yellow-dark"
>
Théo LUDWIG
</Link>{" "}
| {i18n.translate("common.all-rights-reserved")}
</p>
)
}

View File

@ -0,0 +1,28 @@
import { useMemo } from "react"
interface FooterVersionProps {
version: string
}
export const FooterVersion = (props: FooterVersionProps): JSX.Element => {
const { version } = props
const versionLink = useMemo(() => {
return `https://github.com/theoludwig/theoludwig/releases/tag/v${version}`
}, [version])
return (
<p className="mt-1">
Version{" "}
<a
data-cy="version-link"
className="text-yellow hover:underline dark:text-yellow-dark"
href={versionLink}
target="_blank"
rel="noopener noreferrer"
>
{version}
</a>
</p>
)
}

View File

@ -0,0 +1,14 @@
import { FooterText } from "./FooterText"
import { FooterVersion } from "./FooterVersion"
export const Footer = async (): Promise<JSX.Element> => {
const { readPackage } = await import("read-pkg")
const { version } = await readPackage()
return (
<footer className="flex flex-col items-center justify-center border-t-2 border-gray-600 bg-white py-6 text-lg dark:border-gray-400 dark:bg-black">
<FooterText />
<FooterVersion version={version} />
</footer>
)
}

View File

@ -1,45 +0,0 @@
import NextHead from 'next/head'
interface HeadProps {
title?: string
image?: string
description?: string
url?: string
}
export const Head: React.FC<HeadProps> = (props) => {
const {
title = 'Théo LUDWIG',
image = 'https://theoludwig.fr/images/icon-96x96.png',
description = 'Théo LUDWIG - Developer Full Stack • Passionate about High-Tech',
url = 'https://theoludwig.fr/'
} = props
return (
<NextHead>
<title>{title}</title>
<link rel='icon' type='image/png' href={image} />
{/* Meta Tag */}
<meta name='viewport' content='width=device-width, initial-scale=1.0' />
<meta name='description' content={description} />
<meta name='Language' content='fr-FR, en-US' />
<meta name='theme-color' content='#ffd800' />
{/* Open Graph Metadata */}
<meta property='og:title' content={title} />
<meta property='og:type' content='website' />
<meta property='og:url' content={url} />
<meta property='og:image' content={image} />
<meta property='og:description' content={description} />
<meta property='og:locale' content='fr-FR, en-US' />
<meta property='og:site_name' content={title} />
{/* Twitter card Metadata */}
<meta name='twitter:card' content='summary' />
<meta name='twitter:description' content={description} />
<meta name='twitter:title' content={title} />
<meta name='twitter:image' content={image} />
</NextHead>
)
}

View File

@ -1,16 +0,0 @@
export const Arrow: React.FC = () => {
return (
<svg
width='12'
height='8'
viewBox='0 0 12 8'
fill='none'
xmlns='http://www.w3.org/2000/svg'
>
<path
className='fill-current text-black dark:text-white'
d='M9.8024 0.292969L5.61855 4.58597L1.43469 0.292969L0.0566406 1.70697L5.61855 7.41397L11.1805 1.70697L9.8024 0.292969Z'
/>
</svg>
)
}

View File

@ -1,24 +0,0 @@
import Image from 'next/image'
export interface LanguageFlagProps {
language: string
}
export const LanguageFlag: React.FC<LanguageFlagProps> = (props) => {
const { language } = props
return (
<>
<Image
quality={100}
width={35}
height={35}
src={`/images/languages/${language}.svg`}
alt={language}
/>
<p data-cy='language-flag-text' className='mx-2 text-base'>
{language.toUpperCase()}
</p>
</>
)
}

View File

@ -1,81 +0,0 @@
import { useCallback, useEffect, useState, useRef } from 'react'
import useTranslation from 'next-translate/useTranslation'
import setLanguage from 'next-translate/setLanguage'
import classNames from 'clsx'
import i18n from 'i18n.json'
import { Arrow } from './Arrow'
import { LanguageFlag } from './LanguageFlag'
export const Language: React.FC = () => {
const { lang: currentLanguage } = useTranslation()
const [hiddenMenu, setHiddenMenu] = useState(true)
const languageClickRef = useRef<HTMLDivElement | null>(null)
const handleHiddenMenu = useCallback(() => {
setHiddenMenu((oldHiddenMenu) => {
return !oldHiddenMenu
})
}, [])
useEffect(() => {
const handleClickEvent = (event: MouseEvent): void => {
if (languageClickRef.current == null || event.target == null) {
return
}
if (!languageClickRef.current.contains(event.target as Node)) {
setHiddenMenu(true)
}
}
window.document.addEventListener('click', handleClickEvent)
return () => {
return window.removeEventListener('click', handleClickEvent)
}
}, [])
const handleLanguage = async (language: string): Promise<void> => {
await setLanguage(language)
}
return (
<div className='flex cursor-pointer flex-col items-center justify-center'>
<div
ref={languageClickRef}
data-cy='language-click'
className='mr-5 flex items-center'
onClick={handleHiddenMenu}
>
<LanguageFlag language={currentLanguage} />
<Arrow />
</div>
<ul
data-cy='languages-list'
className={classNames(
'absolute top-14 z-10 mr-4 mt-3 flex w-24 list-none flex-col items-center justify-center rounded-lg bg-white p-0 shadow-lightFlag dark:bg-black dark:shadow-darkFlag',
{ hidden: hiddenMenu }
)}
>
{i18n.locales.map((language, index) => {
if (language === currentLanguage) {
return null
}
return (
<li
key={index}
className='flex h-12 w-full items-center justify-center pl-2 hover:bg-[#4f545c] hover:bg-opacity-20'
onClick={async () => {
return await handleLanguage(language)
}}
>
<LanguageFlag language={language} />
</li>
)
})}
</ul>
</div>
)
}

View File

@ -0,0 +1,16 @@
export const Arrow = (): JSX.Element => {
return (
<svg
width="12"
height="8"
viewBox="0 0 12 8"
fill="none"
xmlns="http://www.w3.org/2000/svg"
>
<path
className="fill-current text-black dark:text-white"
d="M9.8024 0.292969L5.61855 4.58597L1.43469 0.292969L0.0566406 1.70697L5.61855 7.41397L11.1805 1.70697L9.8024 0.292969Z"
/>
</svg>
)
}

View File

@ -0,0 +1,30 @@
import Image from "next/image"
import type { CookiesStore } from "@/utils/constants"
import { useI18n } from "@/i18n/i18n.client"
export interface LocaleFlagProps {
locale: string
cookiesStore: CookiesStore
}
export const LocaleFlag = (props: LocaleFlagProps): JSX.Element => {
const { locale, cookiesStore } = props
const i18n = useI18n(cookiesStore)
return (
<>
<Image
quality={100}
width={35}
height={35}
src={`/images/locales/${locale}.svg`}
alt={locale}
/>
<p data-cy="locale-flag-text" className="mx-2 text-base">
{i18n.translate(`common.${locale}`)}
</p>
</>
)
}

View File

@ -0,0 +1,100 @@
"use client"
import { usePathname } from "next/navigation"
import { useCallback, useEffect, useState, useRef } from "react"
import classNames from "clsx"
import type { Locale as LocaleType, CookiesStore } from "@/utils/constants"
import { LOCALES } from "@/utils/constants"
import { Arrow } from "./Arrow"
import { LocaleFlag } from "./LocaleFlag"
export interface LocalesProps {
currentLocale: string
cookiesStore: CookiesStore
}
export const Locales = (props: LocalesProps): JSX.Element => {
const { currentLocale, cookiesStore } = props
const pathname = usePathname()
const [hiddenMenu, setHiddenMenu] = useState(true)
const languageClickRef = useRef<HTMLDivElement | null>(null)
const handleHiddenMenu = useCallback(() => {
setHiddenMenu((oldHiddenMenu) => {
return !oldHiddenMenu
})
}, [])
useEffect(() => {
const handleClickEvent = (event: MouseEvent): void => {
if (languageClickRef.current == null || event.target == null) {
return
}
if (!languageClickRef.current.contains(event.target as Node)) {
setHiddenMenu(true)
}
}
window.document.addEventListener("click", handleClickEvent)
return () => {
return window.removeEventListener("click", handleClickEvent)
}
}, [])
const handleLocale = async (locale: LocaleType): Promise<void> => {
const { setLocale } = await import("@/i18n/i18n.server")
setLocale(locale)
}
if (pathname.startsWith("/blog")) {
return <></>
}
return (
<div className="flex cursor-pointer flex-col items-center justify-center">
<div
ref={languageClickRef}
data-cy="locale-click"
className="mr-5 flex items-center"
onClick={handleHiddenMenu}
>
<LocaleFlag
locale={currentLocale}
cookiesStore={cookiesStore?.toString()}
/>
<Arrow />
</div>
<ul
data-cy="locales-list"
className={classNames(
"absolute top-14 z-10 mr-4 mt-3 flex w-32 list-none flex-col items-center justify-center rounded-lg bg-white p-0 shadow-lightFlag dark:bg-black dark:shadow-darkFlag",
{ hidden: hiddenMenu },
)}
>
{LOCALES.filter((locale) => {
return locale !== currentLocale
}).map((locale) => {
return (
<li
key={locale}
className="flex h-12 w-full items-center justify-center hover:bg-[#4f545c] hover:bg-opacity-20"
onClick={async () => {
return await handleLocale(locale)
}}
>
<LocaleFlag
locale={locale}
cookiesStore={cookiesStore?.toString()}
/>
</li>
)
})}
</ul>
</div>
)
}

View File

@ -1,75 +1,76 @@
import { useEffect, useState } from 'react' "use client"
import classNames from 'clsx'
import { useTheme } from 'next-themes'
export const SwitchTheme: React.FC = () => { import classNames from "clsx"
const [mounted, setMounted] = useState(false)
const { theme, setTheme } = useTheme()
useEffect(() => { import { useTheme } from "@/theme/theme.client"
setMounted(true) import type { CookiesStore } from "@/utils/constants"
}, [])
if (!mounted) { export interface SwitchThemeProps {
return null cookiesStore: CookiesStore
} }
const handleClick = (): void => { export const SwitchTheme = (props: SwitchThemeProps): JSX.Element => {
setTheme(theme === 'dark' ? 'light' : 'dark') const { cookiesStore } = props
const theme = useTheme(cookiesStore)
const handleClick = async (): Promise<void> => {
const { setTheme } = await import("@/theme/theme.server")
const newTheme = theme === "dark" ? "light" : "dark"
setTheme(newTheme)
} }
return ( return (
<div <div
className='flex items-center' className="flex items-center"
data-cy='switch-theme-click' data-cy="switch-theme-click"
onClick={handleClick} onClick={handleClick}
> >
<div className='relative inline-block cursor-pointer touch-pan-x select-none border-0 bg-transparent p-0'> <div className="relative inline-block cursor-pointer touch-pan-x select-none border-0 bg-transparent p-0">
<div className='h-[24px] w-[50px] rounded-[30px] bg-[#4d4d4d] p-0 text-white transition-all duration-200 ease-in-out'> <div className="h-[24px] w-[50px] rounded-[30px] bg-[#4d4d4d] p-0 text-white transition-all duration-200 ease-in-out">
<div <div
data-cy='switch-theme-dark' data-cy="switch-theme-dark"
className={classNames( className={classNames(
'absolute bottom-0 left-[8px] top-0 mb-auto mt-auto h-[10px] w-[14px] leading-[0] transition-opacity duration-[250ms] ease-in-out', "absolute bottom-0 left-[8px] top-0 mb-auto mt-auto h-[10px] w-[14px] leading-[0] transition-opacity duration-[250ms] ease-in-out",
{ {
'opacity-100': theme === 'dark', "opacity-100": theme === "dark",
'opacity-0': theme === 'light' "opacity-0": theme === "light",
} },
)} )}
> >
<span className='relative flex h-[10px] w-[10px] items-center justify-center'> <span className="relative flex h-[10px] w-[10px] items-center justify-center">
🌜 🌜
</span> </span>
</div> </div>
<div <div
data-cy='switch-theme-light' data-cy="switch-theme-light"
className={classNames( className={classNames(
'absolute bottom-0 right-[10px] top-0 mb-auto mt-auto h-[10px] w-[10px] leading-[0]', "absolute bottom-0 right-[10px] top-0 mb-auto mt-auto h-[10px] w-[10px] leading-[0]",
{ {
'opacity-100': theme === 'light', "opacity-100": theme === "light",
'opacity-0': theme === 'dark' "opacity-0": theme === "dark",
} },
)} )}
> >
<span className='relative flex h-[10px] w-[10px] items-center justify-center'> <span className="relative flex h-[10px] w-[10px] items-center justify-center">
🌞 🌞
</span> </span>
</div> </div>
</div> </div>
<div <div
className={classNames( className={classNames(
'absolute top-[1px] box-border h-[22px] w-[22px] rounded-[50%] bg-[#fafafa] text-white transition-all duration-[250ms] ease-in-out', "absolute top-[1px] box-border h-[22px] w-[22px] rounded-[50%] bg-[#fafafa] text-white transition-all duration-[250ms] ease-in-out",
{ {
'left-[27px]': theme === 'dark', "left-[27px]": theme === "dark",
'left-0': theme === 'light' "left-0": theme === "light",
} },
)} )}
style={{ border: '1px solid #4d4d4d' }} style={{ border: "1px solid #4d4d4d" }}
/> />
<input <input
data-cy='switch-theme-input' data-cy="switch-theme-input"
type='checkbox' type="checkbox"
aria-label='Dark mode toggle' aria-label="Dark mode toggle"
className='absolute m-[-1px] h-[1px] w-[1px] overflow-hidden border-0 p-0' className="absolute m-[-1px] h-[1px] w-[1px] overflow-hidden border-0 p-0 hidden"
defaultChecked defaultChecked
/> />
</div> </div>

View File

@ -1,45 +1,48 @@
import Link from 'next/link' import { cookies } from "next/headers"
import Image from 'next/image' import Link from "next/link"
import Image from "next/image"
import { Language } from './Language' import { getI18n } from "@/i18n/i18n.server"
import { SwitchTheme } from './SwitchTheme'
export interface HeaderProps { import { Locales } from "./Locales"
showLanguage?: boolean import { SwitchTheme } from "./SwitchTheme"
}
export const Header: React.FC<HeaderProps> = (props) => { export const Header = (): JSX.Element => {
const { showLanguage = false } = props const cookiesStore = cookies()
const i18n = getI18n()
return ( return (
<header className='sticky top-0 z-50 flex w-full justify-between border-b-2 border-gray-600 bg-white px-6 py-2 dark:border-gray-400 dark:bg-black'> <header className="sticky top-0 z-50 flex w-full justify-between border-b-2 border-gray-600 bg-white px-6 py-2 dark:border-gray-400 dark:bg-black">
<Link href='/'> <Link href="/">
<div className='flex items-center justify-center'> <div className="flex items-center justify-center">
<Image <Image
quality={100} quality={100}
width={60} width={60}
height={60} height={60}
src='/images/icon_small.png' src="/images/icon_small.png"
alt='Théo LUDWIG' alt="Théo LUDWIG"
priority priority
/> />
<strong className='ml-1 hidden font-headline font-semibold text-yellow dark:text-yellow-dark xs:block'> <strong className="ml-1 hidden font-headline font-semibold text-yellow dark:text-yellow-dark xs:block">
Théo LUDWIG Théo LUDWIG
</strong> </strong>
</div> </div>
</Link> </Link>
<div className='flex justify-between'> <div className="flex justify-between">
<div className='flex flex-col items-center justify-center px-6'> <div className="flex flex-col items-center justify-center px-6">
<Link <Link
href='/blog' href="/blog"
data-cy='header-blog-link' data-cy="header-blog-link"
className='text-yellow hover:underline dark:text-yellow-dark' className="text-yellow hover:underline dark:text-yellow-dark"
> >
Blog Blog
</Link> </Link>
</div> </div>
{showLanguage ? <Language /> : null} <Locales
<SwitchTheme /> currentLocale={i18n.locale}
cookiesStore={cookiesStore.toString()}
/>
<SwitchTheme cookiesStore={cookiesStore.toString()} />
</div> </div>
</header> </header>
) )

View File

@ -1,17 +1,19 @@
import htmlParser from 'html-react-parser' import htmlParser from "html-react-parser"
export interface InterestParagraphProps { export interface InterestParagraphProps {
title: string title: string
description: string description: string
} }
export const InterestParagraph: React.FC<InterestParagraphProps> = (props) => { export const InterestParagraph = (
props: InterestParagraphProps,
): JSX.Element => {
const { title, description } = props const { title, description } = props
return ( return (
<> <>
<p className='my-6 text-center text-gray dark:text-gray-dark'> <p className="my-6 text-center text-gray dark:text-gray-dark">
<strong className='text-lg font-semibold text-yellow dark:text-yellow-dark'> <strong className="text-lg font-semibold text-yellow dark:text-yellow-dark">
{title} {title}
</strong> </strong>
<br /> <br />

View File

@ -1,18 +1,18 @@
import { FontAwesomeIcon } from '@fortawesome/react-fontawesome' import { FontAwesomeIcon } from "@fortawesome/react-fontawesome"
import type { IconDefinition } from '@fortawesome/free-solid-svg-icons' import type { IconDefinition } from "@fortawesome/free-solid-svg-icons"
interface InterestItemProps { interface InterestItemProps {
title: string title: string
fontAwesomeIcon: IconDefinition fontAwesomeIcon: IconDefinition
} }
export const InterestItem: React.FC<InterestItemProps> = (props) => { export const InterestItem = (props: InterestItemProps): JSX.Element => {
const { fontAwesomeIcon, title } = props const { fontAwesomeIcon, title } = props
return ( return (
<li className='interest-item mx-2 my-2 h-8 w-8' title={title}> <li className="interest-item mx-2 my-2 h-8 w-8" title={title}>
<FontAwesomeIcon <FontAwesomeIcon
className='block h-full w-full text-yellow dark:text-yellow-dark' className="block h-full w-full text-yellow dark:text-yellow-dark"
icon={fontAwesomeIcon} icon={fontAwesomeIcon}
/> />
</li> </li>

View File

@ -1,18 +1,18 @@
import { faCode, faMicrochip } from '@fortawesome/free-solid-svg-icons' import { faCode, faMicrochip } from "@fortawesome/free-solid-svg-icons"
import { faGit } from '@fortawesome/free-brands-svg-icons' import { faGit } from "@fortawesome/free-brands-svg-icons"
import { InterestItem } from './InterestItem' import { InterestItem } from "./InterestItem"
export const InterestsList: React.FC = () => { export const InterestsList = (): JSX.Element => {
return ( return (
<div className='my-4 flex justify-center'> <div className="my-4 flex justify-center">
<ul className='m-0 flex w-96 list-none justify-around p-0'> <ul className="m-0 flex w-96 list-none justify-around p-0">
<InterestItem title='Developer Full Stack' fontAwesomeIcon={faCode} /> <InterestItem title="Developer Full Stack" fontAwesomeIcon={faCode} />
<InterestItem <InterestItem
title='Passionate about High-Tech' title="Passionate about High-Tech"
fontAwesomeIcon={faMicrochip} fontAwesomeIcon={faMicrochip}
/> />
<InterestItem title='Open-Source enthusiast' fontAwesomeIcon={faGit} /> <InterestItem title="Open-Source enthusiast" fontAwesomeIcon={faGit} />
</ul> </ul>
</div> </div>
) )

View File

@ -1,22 +1,21 @@
import useTranslation from 'next-translate/useTranslation' import { getI18n } from "@/i18n/i18n.server"
import type { InterestParagraphProps } from './InterestParagraph' import type { InterestParagraphProps } from "./InterestParagraph"
import { InterestParagraph } from './InterestParagraph' import { InterestParagraph } from "./InterestParagraph"
import { InterestsList } from './InterestsList' import { InterestsList } from "./InterestsList"
export const Interests: React.FC = () => { export const Interests = (): JSX.Element => {
const { t } = useTranslation() const i18n = getI18n()
const paragraphs: InterestParagraphProps[] = t( let paragraphs = i18n.translate<InterestParagraphProps[]>(
'home:interests.paragraphs', "home.interests.paragraphs",
{},
{
returnObjects: true
}
) )
if (!Array.isArray(paragraphs)) {
paragraphs = []
}
return ( return (
<div className='max-w-full'> <div className="max-w-full">
{paragraphs.map((paragraph, index) => { {paragraphs.map((paragraph, index) => {
return <InterestParagraph key={index} {...paragraph} /> return <InterestParagraph key={index} {...paragraph} />
})} })}

View File

@ -1,5 +1,5 @@
import { ShadowContainer } from 'components/design/ShadowContainer' import { ShadowContainer } from "@/components/design/ShadowContainer"
import { GitHubIcon } from 'components/Profile/SocialMediaList/SocialMediaIcons/GitHubIcon' import { GitHubIcon } from "@/components/Profile/SocialMediaList/SocialMediaIcons/GitHubIcon"
export interface RepositoryProps { export interface RepositoryProps {
name: string name: string
@ -7,17 +7,17 @@ export interface RepositoryProps {
href: string href: string
} }
export const Repository: React.FC<RepositoryProps> = (props) => { export const Repository = (props: RepositoryProps): JSX.Element => {
const { name, description, href } = props const { name, description, href } = props
return ( return (
<ShadowContainer className='relative !mb-4 max-h-32 cursor-pointer p-6 transition-transform duration-200 ease-in-out hover:-translate-y-2'> <ShadowContainer className="relative !mb-4 max-h-32 cursor-pointer p-6 transition-transform duration-200 ease-in-out hover:-translate-y-2">
<a href={href} target='_blank' rel='noopener noreferrer'> <a href={href} target="_blank" rel="noopener noreferrer">
<div className='flex'> <div className="flex">
<GitHubIcon className='mr-2 h-6' /> <GitHubIcon className="mr-2 h-6" />
<span className='text-yellow dark:text-yellow-dark'>{name}</span> <span className="text-yellow dark:text-yellow-dark">{name}</span>
</div> </div>
<p className='my-4'>{description}</p> <p className="my-4">{description}</p>
</a> </a>
</ShadowContainer> </ShadowContainer>
) )

View File

@ -1,33 +1,35 @@
import useTranslation from 'next-translate/useTranslation' import { getI18n } from "@/i18n/i18n.server"
import { Repository } from './Repository' import { Repository } from "./Repository"
export const OpenSource: React.FC = () => { export const OpenSource = (): JSX.Element => {
const { t } = useTranslation() const i18n = getI18n()
return ( return (
<div className='mt-0 flex max-w-full flex-col items-center'> <div className="mt-0 flex max-w-full flex-col items-center">
<p className='text-center'>{t('home:open-source.description')}</p> <p className="text-center">
<div className='my-6 grid grid-cols-1 gap-6 md:w-10/12 md:grid-cols-2'> {i18n.translate("home.open-source.description")}
</p>
<div className="my-6 grid grid-cols-1 gap-6 md:w-10/12 md:grid-cols-2">
<Repository <Repository
name='nodejs/node' name="nodejs/node"
description='Node.js JavaScript runtime 🐢🚀' description="Node.js JavaScript runtime 🐢🚀✨"
href='https://github.com/nodejs/node/commits?author=theoludwig' href="https://github.com/nodejs/node/commits?author=theoludwig"
/> />
<Repository <Repository
name='standard/standard' name="standard/standard"
description='🌟 JavaScript Style Guide, with linter & automatic code fixer' description="🌟 JavaScript Style Guide, with linter & automatic code fixer"
href='https://github.com/standard/standard/commits?author=theoludwig' href="https://github.com/standard/standard/commits?author=theoludwig"
/> />
<Repository <Repository
name='nrwl/nx' name="nrwl/nx"
description='Smart, Extensible Build Framework' description="Smart, Fast and Extensible Build System"
href='https://github.com/nrwl/nx/commits?author=theoludwig' href="https://github.com/nrwl/nx/commits?author=theoludwig"
/> />
<Repository <Repository
name='vercel/next.js' name="vercel/next.js"
description='The React Framework for Production' description="The React Framework"
href='https://github.com/vercel/next.js/commits?author=theoludwig' href="https://github.com/vercel/next.js/commits?author=theoludwig"
/> />
</div> </div>
</div> </div>

View File

@ -1,6 +1,6 @@
import Image from 'next/image' import Image from "next/image"
import { ShadowContainer } from 'components/design/ShadowContainer' import { ShadowContainer } from "@/components/design/ShadowContainer"
export interface PortfolioItemProps { export interface PortfolioItemProps {
title: string title: string
@ -9,33 +9,33 @@ export interface PortfolioItemProps {
image: string image: string
} }
export const PortfolioItem: React.FC<PortfolioItemProps> = (props) => { export const PortfolioItem = (props: PortfolioItemProps): JSX.Element => {
const { title, description, link, image } = props const { title, description, link, image } = props
return ( return (
<ShadowContainer className='relative cursor-pointer items-center sm:ml-10'> <ShadowContainer className="relative cursor-pointer items-center sm:ml-10">
<a <a
className='group inline-flex justify-center' className="group inline-flex justify-center"
target='_blank' target="_blank"
rel='noopener noreferrer' rel="noopener noreferrer"
href={link} href={link}
aria-label={title} aria-label={title}
> >
<div className='flex justify-center'> <div className="flex justify-center">
<Image <Image
quality={100} quality={100}
className='h-auto w-auto transition-opacity duration-500 group-hover:opacity-20 dark:group-hover:opacity-5' className="h-auto w-auto transition-opacity duration-500 group-hover:opacity-20 dark:group-hover:opacity-5"
width={300} width={300}
height={300} height={300}
src={image} src={image}
alt={title} alt={title}
/> />
</div> </div>
<div className='absolute bottom-0 h-auto overflow-hidden text-center opacity-0 transition-opacity duration-500 group-hover:opacity-100'> <div className="absolute bottom-0 h-auto overflow-hidden text-center opacity-0 transition-opacity duration-500 group-hover:opacity-100">
<h3 className='my-6 text-xl font-semibold text-yellow dark:text-yellow-dark'> <h3 className="my-6 text-xl font-semibold text-yellow dark:text-yellow-dark">
{title} {title}
</h3> </h3>
<p className='my-6'>{description}</p> <p className="my-6">{description}</p>
</div> </div>
</a> </a>
</ShadowContainer> </ShadowContainer>

View File

@ -1,21 +1,18 @@
import useTranslation from 'next-translate/useTranslation' import { getI18n } from "@/i18n/i18n.server"
import type { PortfolioItemProps } from './PortfolioItem' import type { PortfolioItemProps } from "./PortfolioItem"
import { PortfolioItem } from './PortfolioItem' import { PortfolioItem } from "./PortfolioItem"
export const Portfolio: React.FC = () => { export const Portfolio = (): JSX.Element => {
const { t } = useTranslation('home') const i18n = getI18n()
const items: PortfolioItemProps[] = t( let items = i18n.translate<PortfolioItemProps[]>("home.portfolio.items")
'home:portfolio.items', if (!Array.isArray(items)) {
{}, items = []
{ }
returnObjects: true
}
)
return ( return (
<div className='flex w-full flex-wrap justify-center px-3'> <div className="flex w-full flex-wrap justify-center px-3">
{items.map((item, index) => { {items.map((item, index) => {
return <PortfolioItem key={index} {...item} /> return <PortfolioItem key={index} {...item} />
})} })}

View File

@ -1,18 +1,18 @@
import useTranslation from 'next-translate/useTranslation' import { getI18n } from "@/i18n/i18n.server"
export const ProfileDescriptionBottom: React.FC = () => { export const ProfileDescriptionBottom = (): JSX.Element => {
const { t, lang } = useTranslation() const i18n = getI18n()
return ( return (
<p className='mb-8 mt-8 text-base font-normal text-gray dark:text-gray-dark'> <p className="mb-8 mt-8 text-base font-normal text-gray dark:text-gray-dark">
{t('home:about.description-bottom')} {i18n.translate("home.about.description-bottom")}
{lang === 'fr' ? ( {i18n.locale === "fr-FR" ? (
<> <>
<br /> <br />
<br /> <br />
<a <a
href='/curriculum-vitae/index.html' href="/curriculum-vitae/index.html"
className='text-yellow hover:underline dark:text-yellow-dark' className="text-yellow hover:underline dark:text-yellow-dark"
> >
Curriculum vitæ Curriculum vitæ
</a> </a>

View File

@ -1,14 +1,16 @@
import useTranslation from 'next-translate/useTranslation' import { getI18n } from "@/i18n/i18n.server"
export const ProfileInformation: React.FC = () => { export const ProfileInformation = (): JSX.Element => {
const { t } = useTranslation() const i18n = getI18n()
return ( return (
<div className='mb-6 border-b-2 border-gray-600 pb-2 font-headline dark:border-gray-400'> <div className="mb-6 border-b-2 border-gray-600 pb-2 font-headline dark:border-gray-400">
<h1 className='mb-2 text-4xl font-semibold text-yellow dark:text-yellow-dark'> <h1 className="mb-2 text-4xl font-semibold text-yellow dark:text-yellow-dark">
Théo LUDWIG Théo LUDWIG
</h1> </h1>
<h2 className='mb-3 text-base'>{t('home:about.description')}</h2> <h2 className="mb-3 text-base">
{i18n.translate("home.about.description")}
</h2>
</div> </div>
) )
} }

View File

@ -4,18 +4,18 @@ interface ProfileItemProps {
link?: string link?: string
} }
export const ProfileItem: React.FC<ProfileItemProps> = (props) => { export const ProfileItem = (props: ProfileItemProps): JSX.Element => {
const { title, value, link } = props const { title, value, link } = props
return ( return (
<li className='mb-3 before:table after:clear-both after:table'> <li className="mb-3 before:table after:clear-both after:table">
<strong className='float-left block w-28 text-sm font-bold text-black dark:text-white'> <strong className="float-left block w-28 text-sm font-bold text-black dark:text-white">
{title} {title}
</strong> </strong>
<span className='mb-4 ml-0 block text-sm font-normal text-gray dark:text-gray-dark sm:mb-0 sm:ml-32'> <span className="mb-4 ml-0 block text-sm font-normal text-gray dark:text-gray-dark sm:mb-0 sm:ml-32">
{link != null ? ( {link != null ? (
<a <a
className='text-gray hover:underline dark:text-gray-dark' className="text-gray hover:underline dark:text-gray-dark"
href={link} href={link}
> >
{value} {value}

View File

@ -1,32 +1,46 @@
import useTranslation from 'next-translate/useTranslation' "use client"
import { useMemo } from 'react'
import { BIRTH_DATE, BIRTH_DATE_STRING, getAge } from 'utils/getAge' import { useMemo } from "react"
import { ProfileItem } from './ProfileItem' import { useI18n } from "@/i18n/i18n.client"
import { BIRTH_DATE, BIRTH_DATE_STRING, getAge } from "@/utils/getAge"
import type { CookiesStore } from "@/utils/constants"
export const ProfileList: React.FC = () => { import { ProfileItem } from "./ProfileItem"
const { t } = useTranslation('home')
export interface ProfileListProps {
cookiesStore: CookiesStore
}
export const ProfileList = (props: ProfileListProps): JSX.Element => {
const { cookiesStore } = props
const i18n = useI18n(cookiesStore)
const age = useMemo(() => { const age = useMemo(() => {
return getAge(BIRTH_DATE) return getAge(BIRTH_DATE)
}, []) }, [])
return ( return (
<ul className='m-0 list-none p-0'> <ul className="m-0 list-none p-0">
<ProfileItem <ProfileItem
title={t('home:about.pronouns')} title={i18n.translate("home.about.pronouns")}
value={t('home:about.pronouns-value')} value={i18n.translate("home.about.pronouns-value")}
/> />
<ProfileItem <ProfileItem
title={t('home:about.birth-date')} title={i18n.translate("home.about.birth-date")}
value={`${BIRTH_DATE_STRING} (${age} ${t('home:about.years-old')})`} value={`${BIRTH_DATE_STRING} (${age} ${i18n.translate(
"home.about.years-old",
)})`}
/> />
<ProfileItem title={t('home:about.nationality')} value='Alsace, France' />
<ProfileItem <ProfileItem
title='Email' title={i18n.translate("home.about.nationality")}
value='contact@theoludwig.fr' value="Alsace, France"
link='mailto:contact@theoludwig.fr' />
<ProfileItem
title="Email"
value="contact@theoludwig.fr"
link="mailto:contact@theoludwig.fr"
/> />
</ul> </ul>
) )

View File

@ -1,11 +1,11 @@
import Image from 'next/image' import Image from "next/image"
import Logo from 'public/images/logo.png' import Logo from "public/images/logo.png"
export const ProfileLogo: React.FC = () => { export const ProfileLogo = (): JSX.Element => {
return ( return (
<div className='max-h-[370px] max-w-[370px] px-2 py-6'> <div className="max-h-[370px] max-w-[370px] px-2 py-6">
<Image quality={100} src={Logo} alt='Théo LUDWIG' priority /> <Image quality={100} src={Logo} alt="Théo LUDWIG" priority />
</div> </div>
) )
} }

View File

@ -1,10 +1,12 @@
import { Icon } from './Icon' import { Icon } from "./Icon"
export const EmailIcon: React.FC<React.SVGProps<SVGSVGElement>> = (props) => { export const EmailIcon = (
props: React.SVGProps<SVGSVGElement>,
): JSX.Element => {
return ( return (
<Icon {...props}> <Icon {...props}>
<title>Email</title> <title>Email</title>
<path d='M15.61 12c0 1.99-1.62 3.61-3.61 3.61-1.99 0-3.61-1.62-3.61-3.61 0-1.99 1.62-3.61 3.61-3.61 1.99 0 3.61 1.62 3.61 3.61M12 0C5.383 0 0 5.383 0 12s5.383 12 12 12c2.424 0 4.761-.722 6.76-2.087l.034-.024-1.617-1.879-.027.017A9.494 9.494 0 0112 21.54c-5.26 0-9.54-4.28-9.54-9.54 0-5.26 4.28-9.54 9.54-9.54 5.26 0 9.54 4.28 9.54 9.54a9.63 9.63 0 01-.225 2.05c-.301 1.239-1.169 1.618-1.82 1.568-.654-.053-1.42-.52-1.426-1.661V12A6.076 6.076 0 0012 5.93 6.076 6.076 0 005.93 12 6.076 6.076 0 0012 18.07a6.02 6.02 0 004.3-1.792 3.9 3.9 0 003.32 1.805c.874 0 1.74-.292 2.437-.821.719-.547 1.256-1.336 1.553-2.285.047-.154.135-.504.135-.507l.002-.013c.175-.76.253-1.52.253-2.457 0-6.617-5.383-12-12-12' /> <path d="M15.61 12c0 1.99-1.62 3.61-3.61 3.61-1.99 0-3.61-1.62-3.61-3.61 0-1.99 1.62-3.61 3.61-3.61 1.99 0 3.61 1.62 3.61 3.61M12 0C5.383 0 0 5.383 0 12s5.383 12 12 12c2.424 0 4.761-.722 6.76-2.087l.034-.024-1.617-1.879-.027.017A9.494 9.494 0 0112 21.54c-5.26 0-9.54-4.28-9.54-9.54 0-5.26 4.28-9.54 9.54-9.54 5.26 0 9.54 4.28 9.54 9.54a9.63 9.63 0 01-.225 2.05c-.301 1.239-1.169 1.618-1.82 1.568-.654-.053-1.42-.52-1.426-1.661V12A6.076 6.076 0 0012 5.93 6.076 6.076 0 005.93 12 6.076 6.076 0 0012 18.07a6.02 6.02 0 004.3-1.792 3.9 3.9 0 003.32 1.805c.874 0 1.74-.292 2.437-.821.719-.547 1.256-1.336 1.553-2.285.047-.154.135-.504.135-.507l.002-.013c.175-.76.253-1.52.253-2.457 0-6.617-5.383-12-12-12" />
</Icon> </Icon>
) )
} }

View File

@ -1,10 +1,12 @@
import { Icon } from './Icon' import { Icon } from "./Icon"
export const GitHubIcon: React.FC<React.SVGProps<SVGSVGElement>> = (props) => { export const GitHubIcon = (
props: React.SVGProps<SVGSVGElement>,
): JSX.Element => {
return ( return (
<Icon {...props}> <Icon {...props}>
<title>GitHub</title> <title>GitHub</title>
<path d='M12 .297c-6.63 0-12 5.373-12 12 0 5.303 3.438 9.8 8.205 11.385.6.113.82-.258.82-.577 0-.285-.01-1.04-.015-2.04-3.338.724-4.042-1.61-4.042-1.61C4.422 18.07 3.633 17.7 3.633 17.7c-1.087-.744.084-.729.084-.729 1.205.084 1.838 1.236 1.838 1.236 1.07 1.835 2.809 1.305 3.495.998.108-.776.417-1.305.76-1.605-2.665-.3-5.466-1.332-5.466-5.93 0-1.31.465-2.38 1.235-3.22-.135-.303-.54-1.523.105-3.176 0 0 1.005-.322 3.3 1.23.96-.267 1.98-.399 3-.405 1.02.006 2.04.138 3 .405 2.28-1.552 3.285-1.23 3.285-1.23.645 1.653.24 2.873.12 3.176.765.84 1.23 1.91 1.23 3.22 0 4.61-2.805 5.625-5.475 5.92.42.36.81 1.096.81 2.22 0 1.606-.015 2.896-.015 3.286 0 .315.21.69.825.57C20.565 22.092 24 17.592 24 12.297c0-6.627-5.373-12-12-12' /> <path d="M12 .297c-6.63 0-12 5.373-12 12 0 5.303 3.438 9.8 8.205 11.385.6.113.82-.258.82-.577 0-.285-.01-1.04-.015-2.04-3.338.724-4.042-1.61-4.042-1.61C4.422 18.07 3.633 17.7 3.633 17.7c-1.087-.744.084-.729.084-.729 1.205.084 1.838 1.236 1.838 1.236 1.07 1.835 2.809 1.305 3.495.998.108-.776.417-1.305.76-1.605-2.665-.3-5.466-1.332-5.466-5.93 0-1.31.465-2.38 1.235-3.22-.135-.303-.54-1.523.105-3.176 0 0 1.005-.322 3.3 1.23.96-.267 1.98-.399 3-.405 1.02.006 2.04.138 3 .405 2.28-1.552 3.285-1.23 3.285-1.23.645 1.653.24 2.873.12 3.176.765.84 1.23 1.91 1.23 3.22 0 4.61-2.805 5.625-5.475 5.92.42.36.81 1.096.81 2.22 0 1.606-.015 2.896-.015 3.286 0 .315.21.69.825.57C20.565 22.092 24 17.592 24 12.297c0-6.627-5.373-12-12-12" />
</Icon> </Icon>
) )
} }

View File

@ -1,10 +1,12 @@
import { Icon } from './Icon' import { Icon } from "./Icon"
export const GitLabIcon: React.FC<React.SVGProps<SVGSVGElement>> = (props) => { export const GitLabIcon = (
props: React.SVGProps<SVGSVGElement>,
): JSX.Element => {
return ( return (
<Icon {...props}> <Icon {...props}>
<title>GitLab</title> <title>GitLab</title>
<path d='M4.845.904c-.435 0-.82.28-.955.692C2.639 5.449 1.246 9.728.07 13.335a1.437 1.437 0 00.522 1.607l11.071 8.045c.2.145.472.144.67-.004l11.073-8.04a1.436 1.436 0 00.522-1.61c-1.285-3.942-2.683-8.256-3.817-11.746a1.004 1.004 0 00-.957-.684.987.987 0 00-.949.69l-2.405 7.408H8.203l-2.41-7.408a.987.987 0 00-.942-.69h-.006zm-.006 1.42l2.173 6.678H2.675zm14.326 0l2.168 6.678h-4.341zm-10.593 7.81h6.862c-1.142 3.52-2.288 7.04-3.434 10.559L8.572 10.135zm-5.514.005h4.321l3.086 9.5zm13.567 0h4.325c-2.467 3.17-4.95 6.328-7.411 9.502 1.028-3.167 2.059-6.334 3.086-9.502zM2.1 10.762l6.977 8.947-7.817-5.682a.305.305 0 01-.112-.341zm19.798 0l.952 2.922a.305.305 0 01-.11.341v.002l-7.82 5.68.026-.035z' /> <path d="M4.845.904c-.435 0-.82.28-.955.692C2.639 5.449 1.246 9.728.07 13.335a1.437 1.437 0 00.522 1.607l11.071 8.045c.2.145.472.144.67-.004l11.073-8.04a1.436 1.436 0 00.522-1.61c-1.285-3.942-2.683-8.256-3.817-11.746a1.004 1.004 0 00-.957-.684.987.987 0 00-.949.69l-2.405 7.408H8.203l-2.41-7.408a.987.987 0 00-.942-.69h-.006zm-.006 1.42l2.173 6.678H2.675zm14.326 0l2.168 6.678h-4.341zm-10.593 7.81h6.862c-1.142 3.52-2.288 7.04-3.434 10.559L8.572 10.135zm-5.514.005h4.321l3.086 9.5zm13.567 0h4.325c-2.467 3.17-4.95 6.328-7.411 9.502 1.028-3.167 2.059-6.334 3.086-9.502zM2.1 10.762l6.977 8.947-7.817-5.682a.305.305 0 01-.112-.341zm19.798 0l.952 2.922a.305.305 0 01-.11.341v.002l-7.82 5.68.026-.035z" />
</Icon> </Icon>
) )
} }

View File

@ -1,15 +1,15 @@
import classNames from 'clsx' import classNames from "clsx"
export const Icon: React.FC<React.SVGProps<SVGSVGElement>> = (props) => { export const Icon = (props: React.SVGProps<SVGSVGElement>): JSX.Element => {
const { children, className, ...rest } = props const { children, className, ...rest } = props
return ( return (
<svg <svg
xmlns='http://www.w3.org/2000/svg' xmlns="http://www.w3.org/2000/svg"
viewBox='0 0 24 24' viewBox="0 0 24 24"
className={classNames( className={classNames(
'h-8 w-8 fill-current text-black dark:text-white', "h-8 w-8 fill-current text-black dark:text-white",
className className,
)} )}
{...rest} {...rest}
> >

View File

@ -1,10 +1,10 @@
import { Icon } from './Icon' import { Icon } from "./Icon"
export const NPMIcon: React.FC<React.SVGProps<SVGSVGElement>> = (props) => { export const NPMIcon = (props: React.SVGProps<SVGSVGElement>): JSX.Element => {
return ( return (
<Icon {...props}> <Icon {...props}>
<title>npm</title> <title>npm</title>
<path d='M1.763 0C.786 0 0 .786 0 1.763v20.474C0 23.214.786 24 1.763 24h20.474c.977 0 1.763-.786 1.763-1.763V1.763C24 .786 23.214 0 22.237 0zM5.13 5.323l13.837.019-.009 13.836h-3.464l.01-10.382h-3.456L12.04 19.17H5.113z' /> <path d="M1.763 0C.786 0 0 .786 0 1.763v20.474C0 23.214.786 24 1.763 24h20.474c.977 0 1.763-.786 1.763-1.763V1.763C24 .786 23.214 0 22.237 0zM5.13 5.323l13.837.019-.009 13.836h-3.464l.01-10.382h-3.456L12.04 19.17H5.113z" />
</Icon> </Icon>
) )
} }

View File

@ -1,10 +1,12 @@
import { Icon } from './Icon' import { Icon } from "./Icon"
export const TwitchIcon: React.FC<React.SVGProps<SVGSVGElement>> = (props) => { export const TwitchIcon = (
props: React.SVGProps<SVGSVGElement>,
): JSX.Element => {
return ( return (
<Icon {...props}> <Icon {...props}>
<title>Twitch</title> <title>Twitch</title>
<path d='M11.571 4.714h1.715v5.143H11.57zm4.715 0H18v5.143h-1.714zM6 0L1.714 4.286v15.428h5.143V24l4.286-4.286h3.428L22.286 12V0zm14.571 11.143l-3.428 3.428h-3.429l-3 3v-3H6.857V1.714h13.714z' /> <path d="M11.571 4.714h1.715v5.143H11.57zm4.715 0H18v5.143h-1.714zM6 0L1.714 4.286v15.428h5.143V24l4.286-4.286h3.428L22.286 12V0zm14.571 11.143l-3.428 3.428h-3.429l-3 3v-3H6.857V1.714h13.714z" />
</Icon> </Icon>
) )
} }

View File

@ -1,10 +1,12 @@
import { Icon } from './Icon' import { Icon } from "./Icon"
export const TwitterIcon: React.FC<React.SVGProps<SVGSVGElement>> = (props) => { export const TwitterIcon = (
props: React.SVGProps<SVGSVGElement>,
): JSX.Element => {
return ( return (
<Icon {...props}> <Icon {...props}>
<title>Twitter</title> <title>Twitter</title>
<path d='M23.953 4.57a10 10 0 01-2.825.775 4.958 4.958 0 002.163-2.723c-.951.555-2.005.959-3.127 1.184a4.92 4.92 0 00-8.384 4.482C7.69 8.095 4.067 6.13 1.64 3.162a4.822 4.822 0 00-.666 2.475c0 1.71.87 3.213 2.188 4.096a4.904 4.904 0 01-2.228-.616v.06a4.923 4.923 0 003.946 4.827 4.996 4.996 0 01-2.212.085 4.936 4.936 0 004.604 3.417 9.867 9.867 0 01-6.102 2.105c-.39 0-.779-.023-1.17-.067a13.995 13.995 0 007.557 2.209c9.053 0 13.998-7.496 13.998-13.985 0-.21 0-.42-.015-.63A9.935 9.935 0 0024 4.59z' /> <path d="M23.953 4.57a10 10 0 01-2.825.775 4.958 4.958 0 002.163-2.723c-.951.555-2.005.959-3.127 1.184a4.92 4.92 0 00-8.384 4.482C7.69 8.095 4.067 6.13 1.64 3.162a4.822 4.822 0 00-.666 2.475c0 1.71.87 3.213 2.188 4.096a4.904 4.904 0 01-2.228-.616v.06a4.923 4.923 0 003.946 4.827 4.996 4.996 0 01-2.212.085 4.936 4.936 0 004.604 3.417 9.867 9.867 0 01-6.102 2.105c-.39 0-.779-.023-1.17-.067a13.995 13.995 0 007.557 2.209c9.053 0 13.998-7.496 13.998-13.985 0-.21 0-.42-.015-.63A9.935 9.935 0 0024 4.59z" />
</Icon> </Icon>
) )
} }

View File

@ -1,10 +1,12 @@
import { Icon } from './Icon' import { Icon } from "./Icon"
export const YouTubeIcon: React.FC<React.SVGProps<SVGSVGElement>> = (props) => { export const YouTubeIcon = (
props: React.SVGProps<SVGSVGElement>,
): JSX.Element => {
return ( return (
<Icon {...props}> <Icon {...props}>
<title>YouTube</title> <title>YouTube</title>
<path d='M23.498 6.186a3.016 3.016 0 00-2.122-2.136C19.505 3.545 12 3.545 12 3.545s-7.505 0-9.377.505A3.017 3.017 0 00.502 6.186C0 8.07 0 12 0 12s0 3.93.502 5.814a3.016 3.016 0 002.122 2.136c1.871.505 9.376.505 9.376.505s7.505 0 9.377-.505a3.015 3.015 0 002.122-2.136C24 15.93 24 12 24 12s0-3.93-.502-5.814zM9.545 15.568V8.432L15.818 12l-6.273 3.568z' /> <path d="M23.498 6.186a3.016 3.016 0 00-2.122-2.136C19.505 3.545 12 3.545 12 3.545s-7.505 0-9.377.505A3.017 3.017 0 00.502 6.186C0 8.07 0 12 0 12s0 3.93.502 5.814a3.016 3.016 0 002.122 2.136c1.871.505 9.376.505 9.376.505s7.505 0 9.377-.505a3.015 3.015 0 002.122-2.136C24 15.93 24 12 24 12s0-3.93-.502-5.814zM9.545 15.568V8.432L15.818 12l-6.273 3.568z" />
</Icon> </Icon>
) )
} }

View File

@ -1,21 +1,19 @@
interface SocialMediaItemProps { interface SocialMediaItemProps extends React.PropsWithChildren {
link: string link: string
ariaLabel: string ariaLabel: string
} }
export const SocialMediaItem: React.FC< export const SocialMediaItem = (props: SocialMediaItemProps): JSX.Element => {
React.PropsWithChildren<SocialMediaItemProps>
> = (props) => {
const { link, ariaLabel, children } = props const { link, ariaLabel, children } = props
return ( return (
<li className='mx-4 my-1 inline-block'> <li className="mx-4 my-1 inline-block">
<a <a
href={link} href={link}
aria-label={ariaLabel} aria-label={ariaLabel}
target='_blank' target="_blank"
rel='noopener noreferrer' rel="noopener noreferrer"
className='relative inline-block bg-transparent' className="relative inline-block bg-transparent"
> >
{children} {children}
</a> </a>

View File

@ -1,43 +1,43 @@
import { SocialMediaItem } from './SocialMediaItem' import { SocialMediaItem } from "./SocialMediaItem"
import { TwitterIcon } from './SocialMediaIcons/TwitterIcon' import { TwitterIcon } from "./SocialMediaIcons/TwitterIcon"
import { GitHubIcon } from './SocialMediaIcons/GitHubIcon' import { GitHubIcon } from "./SocialMediaIcons/GitHubIcon"
import { GitLabIcon } from './SocialMediaIcons/GitLabIcon' import { GitLabIcon } from "./SocialMediaIcons/GitLabIcon"
import { YouTubeIcon } from './SocialMediaIcons/YouTubeIcon' import { YouTubeIcon } from "./SocialMediaIcons/YouTubeIcon"
import { TwitchIcon } from './SocialMediaIcons/TwitchIcon' import { TwitchIcon } from "./SocialMediaIcons/TwitchIcon"
import { EmailIcon } from './SocialMediaIcons/EmailIcon' import { EmailIcon } from "./SocialMediaIcons/EmailIcon"
import { NPMIcon } from './SocialMediaIcons/NPMIcon' import { NPMIcon } from "./SocialMediaIcons/NPMIcon"
export const SocialMediaList: React.FC = () => { export const SocialMediaList = (): JSX.Element => {
return ( return (
<ul className='social-media-list m-0 mt-2 list-none py-4 text-center'> <ul className="social-media-list m-0 mt-2 list-none py-4 text-center">
<SocialMediaItem link='https://github.com/theoludwig' ariaLabel='GitHub'> <SocialMediaItem link="https://github.com/theoludwig" ariaLabel="GitHub">
<GitHubIcon /> <GitHubIcon />
</SocialMediaItem> </SocialMediaItem>
<SocialMediaItem link='https://gitlab.com/theoludwig' ariaLabel='GitLab'> <SocialMediaItem link="https://gitlab.com/theoludwig" ariaLabel="GitLab">
<GitLabIcon /> <GitLabIcon />
</SocialMediaItem> </SocialMediaItem>
<SocialMediaItem link='https://www.npmjs.com/~theoludwig' ariaLabel='NPM'> <SocialMediaItem link="https://www.npmjs.com/~theoludwig" ariaLabel="npm">
<NPMIcon /> <NPMIcon />
</SocialMediaItem> </SocialMediaItem>
<SocialMediaItem <SocialMediaItem
link='https://twitter.com/theoludwig_' link="https://twitter.com/theoludwig_"
ariaLabel='Twitter' ariaLabel="Twitter"
> >
<TwitterIcon /> <TwitterIcon />
</SocialMediaItem> </SocialMediaItem>
<SocialMediaItem <SocialMediaItem
link='https://www.youtube.com/@theo_ludwig' link="https://www.youtube.com/@theo_ludwig"
ariaLabel='YouTube' ariaLabel="YouTube"
> >
<YouTubeIcon /> <YouTubeIcon />
</SocialMediaItem> </SocialMediaItem>
<SocialMediaItem <SocialMediaItem
link='https://www.twitch.tv/theoludwig' link="https://www.twitch.tv/theoludwig"
ariaLabel='Twitch' ariaLabel="Twitch"
> >
<TwitchIcon /> <TwitchIcon />
</SocialMediaItem> </SocialMediaItem>
<SocialMediaItem link='mailto:contact@theoludwig.fr' ariaLabel='Email'> <SocialMediaItem link="mailto:contact@theoludwig.fr" ariaLabel="Email">
<EmailIcon /> <EmailIcon />
</SocialMediaItem> </SocialMediaItem>
</ul> </ul>

View File

@ -1,15 +1,19 @@
import { ProfileDescriptionBottom } from './ProfileDescriptionBottom' import { cookies } from "next/headers"
import { ProfileInformation } from './ProfileInfo'
import { ProfileList } from './ProfileList' import { ProfileDescriptionBottom } from "./ProfileDescriptionBottom"
import { ProfileLogo } from './ProfileLogo' import { ProfileInformation } from "./ProfileInfo"
import { ProfileList } from "./ProfileList"
import { ProfileLogo } from "./ProfileLogo"
export const Profile = (): JSX.Element => {
const cookiesStore = cookies()
export const Profile: React.FC = () => {
return ( return (
<div className='flex flex-col items-center justify-center px-10 pt-2 md:flex-row md:pt-10'> <div className="flex flex-col items-center justify-center px-10 pt-2 md:flex-row md:pt-10">
<ProfileLogo /> <ProfileLogo />
<div> <div>
<ProfileInformation /> <ProfileInformation />
<ProfileList /> <ProfileList cookiesStore={cookiesStore.toString()} />
<ProfileDescriptionBottom /> <ProfileDescriptionBottom />
</div> </div>
</div> </div>

View File

@ -1,46 +1,48 @@
import { useTheme } from 'next-themes' import Image from "next/image"
import Image from 'next/image'
import { useMemo } from 'react'
import type { SkillName } from './skills' import { getTheme } from "@/theme/theme.server"
import { skills } from './skills'
import type { SkillName } from "./skills"
import { skills } from "./skills"
export interface SkillComponentProps { export interface SkillComponentProps {
skill: SkillName skill: SkillName
} }
export const SkillComponent: React.FC<SkillComponentProps> = (props) => { export const SkillComponent = (props: SkillComponentProps): JSX.Element => {
const { skill } = props const { skill } = props
const skillProperties = skills[skill]
const { theme } = useTheme()
const image = useMemo(() => { const skillProperties = skills[skill]
if (typeof skillProperties.image === 'string') {
const theme = getTheme()
const getImage = (): string => {
if (typeof skillProperties.image === "string") {
return skillProperties.image return skillProperties.image
} }
if (theme === 'light') { if (theme === "light") {
return skillProperties.image.light return skillProperties.image.light
} }
return skillProperties.image.dark return skillProperties.image.dark
}, [skillProperties, theme]) }
return ( return (
<a <a
href={skillProperties.link} href={skillProperties.link}
className='mx-2 max-w-xl text-yellow hover:underline dark:text-yellow-dark' className="mx-2 max-w-xl text-yellow hover:underline dark:text-yellow-dark"
target='_blank' target="_blank"
rel='noopener noreferrer' rel="noopener noreferrer"
> >
<div className='text-center'> <div className="text-center">
<Image <Image
className='inline h-16 w-16' className="inline h-16 w-16"
quality={100} quality={100}
width={64} width={64}
height={64} height={64}
alt={skill} alt={skill}
src={image} src={getImage()}
/> />
<p className='mt-1'>{skill}</p> <p className="mt-1">{skill}</p>
</div> </div>
</a> </a>
) )

View File

@ -1,24 +1,24 @@
import { ShadowContainer } from 'components/design/ShadowContainer' import { ShadowContainer } from "@/components/design/ShadowContainer"
export interface SkillsSectionProps { export interface SkillsSectionProps {
title: string title: string
children: React.ReactNode children: React.ReactNode
} }
export const SkillsSection: React.FC<SkillsSectionProps> = (props) => { export const SkillsSection = (props: SkillsSectionProps): JSX.Element => {
const { title, children } = props const { title, children } = props
return ( return (
<ShadowContainer> <ShadowContainer>
<div className='mx-auto w-full px-4'> <div className="mx-auto w-full px-4">
<div className='flex flex-wrap px-4 py-6'> <div className="flex flex-wrap px-4 py-6">
<div className='flex-1'> <div className="flex-1">
<div className='mb-8 border-b border-gray-600 dark:border-white dark:border-opacity-10'> <div className="mb-8 border-b border-gray-600 dark:border-white dark:border-opacity-10">
<h3 className='my-3 text-xl font-semibold text-yellow dark:text-yellow-dark'> <h3 className="my-3 text-xl font-semibold text-yellow dark:text-yellow-dark">
{title} {title}
</h3> </h3>
</div> </div>
<div className='flex flex-wrap justify-around'>{children}</div> <div className="flex flex-wrap justify-around">{children}</div>
</div> </div>
</div> </div>
</div> </div>

View File

@ -1,40 +1,40 @@
import useTranslation from 'next-translate/useTranslation' import { getI18n } from "@/i18n/i18n.server"
import { SkillComponent } from './Skill' import { SkillComponent } from "./Skill"
import { SkillsSection } from './SkillsSection' import { SkillsSection } from "./SkillsSection"
export const Skills: React.FC = () => { export const Skills = (): JSX.Element => {
const { t } = useTranslation() const i18n = getI18n()
return ( return (
<> <>
<SkillsSection title={t('home:skills.languages')}> <SkillsSection title={i18n.translate("home.skills.languages")}>
<SkillComponent skill='TypeScript' /> <SkillComponent skill="TypeScript" />
<SkillComponent skill='Python' /> <SkillComponent skill="Python" />
<SkillComponent skill='C/C++' /> <SkillComponent skill="C/C++" />
<SkillComponent skill='PHP' /> <SkillComponent skill="PHP" />
</SkillsSection> </SkillsSection>
<SkillsSection title='Frontend'> <SkillsSection title="Frontend">
<SkillComponent skill='HTML' /> <SkillComponent skill="HTML" />
<SkillComponent skill='CSS' /> <SkillComponent skill="CSS" />
<SkillComponent skill='Tailwind CSS' /> <SkillComponent skill="Tailwind CSS" />
<SkillComponent skill='React.js (+ Next.js)' /> <SkillComponent skill="React.js (+ Next.js)" />
</SkillsSection> </SkillsSection>
<SkillsSection title='Backend'> <SkillsSection title="Backend">
<SkillComponent skill='Laravel' /> <SkillComponent skill="Laravel" />
<SkillComponent skill='Node.js' /> <SkillComponent skill="Node.js" />
<SkillComponent skill='Fastify' /> <SkillComponent skill="Fastify" />
<SkillComponent skill='PostgreSQL' /> <SkillComponent skill="PostgreSQL" />
</SkillsSection> </SkillsSection>
<SkillsSection title={t('home:skills.software-tools')}> <SkillsSection title={i18n.translate("home.skills.software-tools")}>
<SkillComponent skill='GNU/Linux' /> <SkillComponent skill="GNU/Linux" />
<SkillComponent skill='Arch Linux' /> <SkillComponent skill="Arch Linux" />
<SkillComponent skill='Visual Studio Code' /> <SkillComponent skill="Visual Studio Code" />
<SkillComponent skill='Git' /> <SkillComponent skill="Git" />
<SkillComponent skill='Docker' /> <SkillComponent skill="Docker" />
</SkillsSection> </SkillsSection>
</> </>
) )

View File

@ -5,111 +5,111 @@ export interface Skill {
export const skills = { export const skills = {
JavaScript: { JavaScript: {
link: 'https://developer.mozilla.org/docs/Web/JavaScript', link: "https://developer.mozilla.org/docs/Web/JavaScript",
image: '/images/skills/JavaScript.png' image: "/images/skills/JavaScript.png",
}, },
TypeScript: { TypeScript: {
link: 'https://www.typescriptlang.org/', link: "https://www.typescriptlang.org/",
image: '/images/skills/TypeScript.png' image: "/images/skills/TypeScript.png",
}, },
Python: { Python: {
link: 'https://www.python.org/', link: "https://www.python.org/",
image: '/images/skills/Python.png' image: "/images/skills/Python.png",
}, },
'C/C++': { "C/C++": {
link: 'https://isocpp.org/', link: "https://isocpp.org/",
image: '/images/skills/C-Cpp.png' image: "/images/skills/C-Cpp.png",
}, },
PHP: { PHP: {
link: 'https://www.php.net/', link: "https://www.php.net/",
image: '/images/skills/PHP.png' image: "/images/skills/PHP.png",
}, },
Laravel: { Laravel: {
link: 'https://laravel.com/', link: "https://laravel.com/",
image: '/images/skills/Laravel.png' image: "/images/skills/Laravel.png",
}, },
Dart: { Dart: {
link: 'https://dart.dev/', link: "https://dart.dev/",
image: '/images/skills/Dart.png' image: "/images/skills/Dart.png",
}, },
Flutter: { Flutter: {
link: 'https://flutter.dev/', link: "https://flutter.dev/",
image: '/images/skills/Flutter.webp' image: "/images/skills/Flutter.webp",
}, },
HTML: { HTML: {
link: 'https://developer.mozilla.org/docs/Web/HTML', link: "https://developer.mozilla.org/docs/Web/HTML",
image: '/images/skills/HTML.png' image: "/images/skills/HTML.png",
}, },
CSS: { CSS: {
link: 'https://developer.mozilla.org/docs/Web/CSS', link: "https://developer.mozilla.org/docs/Web/CSS",
image: '/images/skills/CSS.png' image: "/images/skills/CSS.png",
}, },
'Tailwind CSS': { "Tailwind CSS": {
link: 'https://tailwindcss.com/', link: "https://tailwindcss.com/",
image: '/images/skills/TailwindCSS.png' image: "/images/skills/TailwindCSS.png",
}, },
SASS: { SASS: {
link: 'https://sass-lang.com/', link: "https://sass-lang.com/",
image: '/images/skills/SASS.svg' image: "/images/skills/SASS.svg",
}, },
'React.js (+ Next.js)': { "React.js (+ Next.js)": {
link: 'https://reactjs.org/', link: "https://reactjs.org/",
image: '/images/skills/ReactJS.png' image: "/images/skills/ReactJS.png",
}, },
'Node.js': { "Node.js": {
link: 'https://nodejs.org/', link: "https://nodejs.org/",
image: '/images/skills/NodeJS.png' image: "/images/skills/NodeJS.png",
}, },
Fastify: { Fastify: {
link: 'https://www.fastify.io/', link: "https://www.fastify.io/",
image: { image: {
light: '/images/skills/Fastify-light.png', light: "/images/skills/Fastify-light.png",
dark: '/images/skills/Fastify-dark.png' dark: "/images/skills/Fastify-dark.png",
} },
}, },
Prisma: { Prisma: {
link: 'https://www.prisma.io/', link: "https://www.prisma.io/",
image: { image: {
light: '/images/skills/Prisma-light.png', light: "/images/skills/Prisma-light.png",
dark: '/images/skills/Prisma-dark.png' dark: "/images/skills/Prisma-dark.png",
} },
}, },
PostgreSQL: { PostgreSQL: {
link: 'https://www.postgresql.org/', link: "https://www.postgresql.org/",
image: '/images/skills/PostgreSQL.png' image: "/images/skills/PostgreSQL.png",
}, },
MySQL: { MySQL: {
link: 'https://www.mysql.com/', link: "https://www.mysql.com/",
image: '/images/skills/MySQL.png' image: "/images/skills/MySQL.png",
}, },
Strapi: { Strapi: {
link: 'https://strapi.io/', link: "https://strapi.io/",
image: '/images/skills/Strapi.png' image: "/images/skills/Strapi.png",
}, },
'Visual Studio Code': { "Visual Studio Code": {
link: 'https://code.visualstudio.com/', link: "https://code.visualstudio.com/",
image: '/images/skills/VisualStudioCode.png' image: "/images/skills/VisualStudioCode.png",
}, },
Git: { Git: {
link: 'https://git-scm.com/', link: "https://git-scm.com/",
image: '/images/skills/Git.png' image: "/images/skills/Git.png",
}, },
Ubuntu: { Ubuntu: {
link: 'https://ubuntu.com/', link: "https://ubuntu.com/",
image: '/images/skills/Ubuntu.png' image: "/images/skills/Ubuntu.png",
}, },
'Arch Linux': { "Arch Linux": {
link: 'https://archlinux.org/', link: "https://archlinux.org/",
image: '/images/skills/ArchLinux.png' image: "/images/skills/ArchLinux.png",
}, },
'GNU/Linux': { "GNU/Linux": {
link: 'https://www.gnu.org/', link: "https://www.gnu.org/",
image: '/images/skills/GNU-Linux.png' image: "/images/skills/GNU-Linux.png",
}, },
Docker: { Docker: {
link: 'https://www.docker.com/', link: "https://www.docker.com/",
image: '/images/skills/Docker.png' image: "/images/skills/Docker.png",
} },
} as const } as const
export type SkillName = keyof typeof skills export type SkillName = keyof typeof skills

View File

@ -0,0 +1,28 @@
import classNames from "clsx"
export interface LoaderProps {
width?: number
height?: number
className?: string
}
export const Loader = (props: LoaderProps): JSX.Element => {
const { width = 50, height = 50, className } = props
return (
<div
style={{
width,
height,
}}
className={classNames(
"animate-spin inline-block border-[3px] border-current border-t-transparent text-yellow dark:text-yellow-dark rounded-full",
className,
)}
role="status"
aria-label="loading"
>
<span className="sr-only">Loading...</span>
</div>
)
}

View File

@ -1,6 +1,10 @@
import { useEffect, useRef } from 'react' "use client"
export const RevealFade: React.FC<React.PropsWithChildren> = (props) => { import { useEffect, useRef } from "react"
export type RevealFadeProps = React.PropsWithChildren
export const RevealFade = (props: RevealFadeProps): JSX.Element => {
const { children } = props const { children } = props
const htmlElement = useRef<HTMLDivElement>(null) const htmlElement = useRef<HTMLDivElement>(null)
@ -8,25 +12,25 @@ export const RevealFade: React.FC<React.PropsWithChildren> = (props) => {
useEffect(() => { useEffect(() => {
const observer = new window.IntersectionObserver( const observer = new window.IntersectionObserver(
(entries, observer) => { (entries, observer) => {
entries.forEach((entry) => { for (const entry of entries) {
if (entry.isIntersecting) { if (entry.isIntersecting) {
entry.target.className = entry.target.className =
'opacity-100 visible translate-y-0 transition-all duration-700 ease-in-out' "opacity-100 visible translate-y-0 transition-all duration-700 ease-in-out"
observer.unobserve(entry.target) observer.unobserve(entry.target)
} }
}) }
}, },
{ {
root: null, root: null,
rootMargin: '0px', rootMargin: "0px",
threshold: 0.28 threshold: 0.28,
} },
) )
observer.observe(htmlElement.current as HTMLDivElement) observer.observe(htmlElement.current as HTMLDivElement)
}, []) }, [])
return ( return (
<div ref={htmlElement} className='invisible -translate-y-7 opacity-0'> <div ref={htmlElement} className="invisible -translate-y-7 opacity-0">
{children} {children}
</div> </div>
) )

View File

@ -1,10 +1,10 @@
type SectionHeadingProps = React.ComponentPropsWithRef<'h2'> type SectionHeadingProps = React.ComponentPropsWithRef<"h2">
export const SectionHeading: React.FC<SectionHeadingProps> = (props) => { export const SectionHeading = (props: SectionHeadingProps): JSX.Element => {
const { children, ...rest } = props const { children, ...rest } = props
return ( return (
<h2 {...rest} className='mb-3 mt-1 text-center text-4xl font-semibold'> <h2 {...rest} className="mb-3 mt-1 text-center text-4xl font-semibold">
{children} {children}
</h2> </h2>
) )

View File

@ -1,14 +1,14 @@
import { ShadowContainer } from '../ShadowContainer' import { ShadowContainer } from "@/components/design/ShadowContainer"
import { SectionHeading } from './SectionHeading' import { SectionHeading } from "@/components/design/Section/SectionHeading"
type SectionProps = React.ComponentPropsWithRef<'section'> & { type SectionProps = React.ComponentPropsWithRef<"section"> & {
heading?: string heading?: string
description?: string description?: string
isMain?: boolean isMain?: boolean
withoutShadowContainer?: boolean withoutShadowContainer?: boolean
} }
export const Section: React.FC<SectionProps> = (props) => { export const Section = (props: SectionProps): JSX.Element => {
const { const {
children, children,
heading, heading,
@ -20,13 +20,13 @@ export const Section: React.FC<SectionProps> = (props) => {
if (isMain) { if (isMain) {
return ( return (
<div className='w-full px-3'> <div className="w-full px-3">
<ShadowContainer style={{ marginTop: 50 }}> <ShadowContainer style={{ marginTop: 50 }}>
<section {...rest}> <section {...rest}>
{heading != null ? ( {heading != null ? (
<SectionHeading>{heading}</SectionHeading> <SectionHeading>{heading}</SectionHeading>
) : null} ) : null}
<div className='w-full px-3'>{children}</div> <div className="w-full px-3">{children}</div>
</section> </section>
</ShadowContainer> </ShadowContainer>
</div> </div>
@ -37,7 +37,7 @@ export const Section: React.FC<SectionProps> = (props) => {
return ( return (
<section {...rest}> <section {...rest}>
{heading != null ? <SectionHeading>{heading}</SectionHeading> : null} {heading != null ? <SectionHeading>{heading}</SectionHeading> : null}
<div className='w-full px-3'>{children}</div> <div className="w-full px-3">{children}</div>
</section> </section>
) )
} }
@ -52,13 +52,13 @@ export const Section: React.FC<SectionProps> = (props) => {
</SectionHeading> </SectionHeading>
) : null} ) : null}
{description != null ? ( {description != null ? (
<p style={{ marginTop: 7 }} className='text-center'> <p style={{ marginTop: 7 }} className="text-center">
{description} {description}
</p> </p>
) : null} ) : null}
<div className='w-full px-3'> <div className="w-full px-3">
<ShadowContainer> <ShadowContainer>
<div className='w-full px-16 py-4 leading-8'>{children}</div> <div className="w-full px-16 py-4 leading-8">{children}</div>
</ShadowContainer> </ShadowContainer>
</div> </div>
</section> </section>

View File

@ -1,15 +1,15 @@
import classNames from 'clsx' import classNames from "clsx"
type ShadowContainerProps = React.ComponentPropsWithRef<'div'> type ShadowContainerProps = React.ComponentPropsWithRef<"div">
export const ShadowContainer: React.FC<ShadowContainerProps> = (props) => { export const ShadowContainer = (props: ShadowContainerProps): JSX.Element => {
const { children, className, ...rest } = props const { children, className, ...rest } = props
return ( return (
<div <div
className={classNames( className={classNames(
'mb-12 h-full max-w-full break-words rounded-2xl border border-solid border-[#000] shadow-light dark:shadow-dark ', "mb-12 h-full max-w-full break-words rounded-2xl border border-solid border-[#000] shadow-light dark:shadow-dark ",
className className,
)} )}
{...rest} {...rest}
> >

View File

@ -1,11 +1,12 @@
services: services:
theoludwig: theoludwig:
container_name: ${COMPOSE_PROJECT_NAME} container_name: ${COMPOSE_PROJECT_NAME}
image: 'theoludwig' image: "theoludwig"
restart: "unless-stopped"
build: build:
context: './' context: "./"
ports: ports:
- '${PORT-3000}:${PORT-3000}' - "${PORT-3000}:${PORT-3000}"
environment: environment:
PORT: ${PORT-3000} PORT: ${PORT-3000}
env_file: '.env' env_file: ".env"

20
curriculum-vitae/build.js Normal file
View File

@ -0,0 +1,20 @@
import { fileURLToPath } from "node:url"
import fs from "node:fs"
import { build } from "vite"
const curriculumVitae = new URL("./", import.meta.url)
const curriculumVitaeDist = new URL("./dist", curriculumVitae)
const publicCurriculumVitaeOutputURL = new URL(
"../public/curriculum-vitae",
import.meta.url,
)
await build({
root: fileURLToPath(curriculumVitae),
base: "/curriculum-vitae/",
})
await fs.promises.cp(curriculumVitaeDist, publicCurriculumVitaeOutputURL, {
recursive: true,
})

Some files were not shown because too many files have changed in this diff Show More