Compare commits

..

2 Commits
main ... dev

Author SHA1 Message Date
a4ce2ce413 added form submission 2025-07-08 18:44:22 +02:00
38cf54c2e8 removed tech-label highlights; clickable techhnologie links 2025-05-27 19:54:58 +02:00
7 changed files with 72 additions and 57 deletions

View File

@ -1,15 +0,0 @@
#!/bin/bash
set -e
if [ -z "$1" ]; then
echo "Usage: $0 <project_path>"
exit 1
fi
PROJECT_PATH=$1
echo "Deploying project..."
cd "$PROJECT_PATH" || { echo "Directory not found: $PROJECT_PATH"; exit 1; }
git pull origin main
docker-compose up -d --build
echo "Deployment finished!"

View File

@ -1,24 +0,0 @@
version: "3.8"
services:
nextjs:
build: .
deploy:
update_config:
parallelism: 1
delay: 5s
order: start-first
container_name: pablo.shortman.me
restart: always
labels:
- "traefik.enable=true"
- "traefik.http.routers.pablo-shortman-me.rule=Host(`pablo.shortman.me`)"
- "traefik.http.routers.pablo-shortman-me.entrypoints=websecure"
- "traefik.http.services.pablo-shortman-me.loadbalancer.server.port=3000"
- "traefik.http.routers.pablo-shortman-me.tls.certresolver=myresolver"
expose:
- "3000"
networks:
- webproxy
networks:
webproxy:
external: true

View File

@ -2,7 +2,11 @@ import React from "react";
import Contact from "@/components/setions/contact"; import Contact from "@/components/setions/contact";
function ContactPage() { function ContactPage() {
return <Contact />; return (
<div className="pt-12">
<Contact />
</div>
);
} }
export default ContactPage; export default ContactPage;

View File

@ -24,6 +24,7 @@ import {
SelectValue, SelectValue,
} from "@/components/ui/select"; } from "@/components/ui/select";
import { cn } from "@/lib/utils"; import { cn } from "@/lib/utils";
import { sendDiscordNotification } from "@/lib/actions";
const formSchema = z.object({ const formSchema = z.object({
name: z.string().min(2).max(50), name: z.string().min(2).max(50),
@ -39,10 +40,16 @@ function ContactForm() {
}); });
// 2. Define a submit handler. // 2. Define a submit handler.
function onSubmit(values: z.infer<typeof formSchema>) { async function onSubmit({
// Do something with the form values. email,
// ✅ This will be type-safe and validated. message,
console.log(values); name,
budget,
}: z.infer<typeof formSchema>) {
await sendDiscordNotification(
`📬 **New Lead Submission**\n*Source*: pablo.shortman.me \n**Name**: ${name}\n**Email**: ${email}\n**Message**: ${message} \n**Budget**: ${budget} $`
);
form.reset();
} }
return ( return (
@ -98,8 +105,8 @@ function ContactForm() {
<SelectValue placeholder="Select your Budget" /> <SelectValue placeholder="Select your Budget" />
</SelectTrigger> </SelectTrigger>
<SelectContent> <SelectContent>
<SelectItem value="< €2k">{"less then $2k"}</SelectItem> <SelectItem value="< €2k">{"less then $1k"}</SelectItem>
<SelectItem value="> €4k">{"more then $4k"}</SelectItem> <SelectItem value="> €4k">{"more then $3k"}</SelectItem>
<SelectItem value="> €6k">{"more then $6k"}</SelectItem> <SelectItem value="> €6k">{"more then $6k"}</SelectItem>
</SelectContent> </SelectContent>
</Select> </Select>

View File

@ -1,10 +1,16 @@
import Link from "next/link";
import React from "react"; import React from "react";
export type TechIcon = "ts" | "next" | "react" | "tailwind" | "PostgreSQL"; export type TechIcon = "ts" | "next" | "react" | "tailwind" | "PostgreSQL";
const tech: { label: string; Logo: (props: any) => JSX.Element }[] = [ const tech: {
label: string;
link: string;
Logo: (props: any) => JSX.Element;
}[] = [
{ {
label: "Tailwind CSS", label: "Tailwind CSS",
link: "https://tailwindcss.com/",
Logo: (props: any) => ( Logo: (props: any) => (
<svg <svg
{...props} {...props}
@ -45,6 +51,7 @@ const tech: { label: string; Logo: (props: any) => JSX.Element }[] = [
}, },
{ {
label: "Typescript", label: "Typescript",
link: "https://www.typescriptlang.org/",
Logo: (props: any) => ( Logo: (props: any) => (
<svg <svg
{...props} {...props}
@ -72,6 +79,7 @@ const tech: { label: string; Logo: (props: any) => JSX.Element }[] = [
}, },
{ {
label: "Next JS", label: "Next JS",
link: "https://nextjs.org/",
Logo: (props: any) => ( Logo: (props: any) => (
<svg <svg
{...props} {...props}
@ -99,6 +107,8 @@ const tech: { label: string; Logo: (props: any) => JSX.Element }[] = [
}, },
{ {
label: "Drizzle ORM", label: "Drizzle ORM",
link: "https://orm.drizzle.team/",
Logo: (props: any) => ( Logo: (props: any) => (
<svg <svg
{...props} {...props}
@ -128,6 +138,7 @@ const tech: { label: string; Logo: (props: any) => JSX.Element }[] = [
{ {
label: "PostgreSQL", label: "PostgreSQL",
link: "https://www.postgresql.org/",
Logo: (props: any) => ( Logo: (props: any) => (
<svg <svg
{...props} {...props}
@ -157,6 +168,7 @@ const tech: { label: string; Logo: (props: any) => JSX.Element }[] = [
{ {
label: "Figma", label: "Figma",
link: "https://www.figma.com",
Logo: (props: any) => ( Logo: (props: any) => (
<svg <svg
{...props} {...props}
@ -184,6 +196,7 @@ const tech: { label: string; Logo: (props: any) => JSX.Element }[] = [
}, },
{ {
label: "Github", label: "Github",
link: "https://github.com/",
Logo: (props: any) => ( Logo: (props: any) => (
<svg <svg
{...props} {...props}
@ -215,18 +228,20 @@ function Tech() {
return ( return (
<div className="py-12 lg:py-20 container relative space-y-8"> <div className="py-12 lg:py-20 container relative space-y-8">
<h3 className="text-sm font-mono text-muted-foreground text-center"> <h3 className="text-sm font-mono text-muted-foreground text-center">
most liked{" "} most liked <span className="text-foreground">technologies</span> and{" "}
<span className="underline text-foreground">technologies</span> and{" "} <span className="text-foreground">tools</span>
<span className="underline text-foreground">tools</span>
</h3> </h3>
<ul className="flex w-full items-center justify-center gap-2 md:gap-4 lg:gap-8 flex-wrap"> <ul className="flex w-full items-center justify-center gap-2 md:gap-4 lg:gap-8 flex-wrap">
{tech.map(({ Logo, label }, idx) => ( {tech.map(({ Logo, label, link }, idx) => (
<li <li key={idx}>
<Link
href={link}
target="_blank"
className="flex items-center gap-2 py-2 px-6 rounded-md bg-muted" className="flex items-center gap-2 py-2 px-6 rounded-md bg-muted"
key={idx}
> >
<Logo className="size-6" /> <Logo className="size-6" />
<span>{label}</span> <span>{label}</span>
</Link>
</li> </li>
))} ))}
</ul> </ul>

13
src/lib/actions.ts Normal file
View File

@ -0,0 +1,13 @@
"use server";
export async function sendDiscordNotification(message: string) {
const webhookUrl = process.env.DISCORD_WEBHOOK_URL;
await fetch(webhookUrl!, {
method: "POST",
headers: { "Content-Type": "application/json" },
body: JSON.stringify({
content: message,
}),
});
}

15
test.yml Normal file
View File

@ -0,0 +1,15 @@
labels:
- "traefik.enable=true"
- "traefik.http.routers.logipedia.rule=Host(`logipedia.shortman.me`)"
- "traefik.http.routers.logipedia.entrypoints=websecure"
- "traefik.http.routers.logipedia.tls.certresolver=myresolver"
networks:
- logipedia_network
- webproxy
labels:
- "traefik.enable=true"
- "traefik.http.routers.pablo-shortman-me.rule=Host(`pablo.shortman.me`)"
- "traefik.http.routers.pablo-shortman-me.entrypoints=websecure"
- "traefik.http.routers.pablo-shortman-me.tls.certresolver=myresolver"
networks:
- webproxy