32 Commits

Author SHA1 Message Date
85af4aec77 make meeting creation user level 2026-06-18 04:55:25 +02:00
05740e122e Add AI assistant tools for site search, project details, experience matching, and calendar availability 2026-06-18 03:37:22 +02:00
95666e20e9 Merge branch 'chat-scroll-behaiviour' 2026-06-18 02:53:20 +02:00
993137068e chat scroll to bottom by default 2026-06-18 02:53:06 +02:00
5755bd3184 fix chat mobile cutoff 2026-06-18 02:49:05 +02:00
ca29bd5003 Merge branch 'project-music-srr' 2026-06-18 02:47:15 +02:00
62f808b0cf animated components hidden default 2026-06-18 02:47:03 +02:00
cb3ece4f99 ssr 2026-06-18 02:39:45 +02:00
c5faf8fa57 same mde editor in all admin forms 2026-06-18 02:12:37 +02:00
a7354ad774 Merge branch 'fix-cv-animations' 2026-06-18 01:59:00 +02:00
c742b8e457 fix height problems 2026-06-18 01:58:53 +02:00
4ce93a0466 Merge branch 'fix-cv-animations' 2026-06-18 01:43:35 +02:00
0d79adb104 fix cv animations 2026-06-18 01:41:25 +02:00
3e5be46503 Merge branch 'mdx-refactor' 2026-06-18 01:33:00 +02:00
73ba2b573d cleanup markdown 2026-06-18 01:32:05 +02:00
c1fe73dbd0 remove stupid spacer 2026-06-16 19:55:38 +02:00
af4ff18917 Merge branch 'fix-cv-animations' 2026-06-16 19:16:37 +02:00
b59fb2b3af fetch on render instead of fetch as you render 2026-06-16 19:16:31 +02:00
58dc4ce53f Merge branch 'background-music-playback' 2026-06-16 19:13:44 +02:00
53bc70ab05 use audio session api 2026-06-16 19:13:32 +02:00
1b3f30cc90 Merge branch 'fix-cv-animations' 2026-06-16 19:10:33 +02:00
fd5063d1c4 cv entry uses animated components 2026-06-16 19:10:28 +02:00
303ac83fe2 Merge branch 'backround-music-playback' 2026-06-16 19:01:21 +02:00
91315730ac background playback 2026-06-16 19:00:59 +02:00
13649cd6dc Merge branch 'fix-cv-animations' 2026-06-16 18:56:05 +02:00
1e0f033d07 refactor to use existing animated components 2026-06-16 18:56:02 +02:00
39563b6740 Merge branch 'fix-cv-animations' 2026-06-16 18:42:19 +02:00
dcc4f47ccf fix cv animations 2026-06-16 18:42:06 +02:00
c8a9ab5984 Merge branch 'ai-model-select' 2026-06-16 18:40:14 +02:00
54f108ac8d model selector 2026-06-16 18:40:08 +02:00
865ef0b316 refactor animations once again 2026-06-16 15:41:03 +02:00
63b0405a7a display waveform 2026-06-16 13:07:41 +02:00
53 changed files with 3110 additions and 1560 deletions

1032
bun.lock

File diff suppressed because it is too large Load Diff

View File

@@ -31,7 +31,11 @@
"@fortawesome/react-fontawesome": "^3.3.1",
"@gsap/react": "^2.1.2",
"@hookform/resolvers": "^5.4.0",
"@mdx-js/mdx": "^3.1.1",
"@mdx-js/react": "^3.1.1",
"@mdx-js/loader": "^3.1.1",
"@neondatabase/serverless": "^1.1.0",
"@next/mdx": "^16.2.9",
"@radix-ui/react-accordion": "^1.2.12",
"@radix-ui/react-alert-dialog": "^1.1.15",
"@radix-ui/react-aspect-ratio": "^1.1.8",
@@ -67,6 +71,7 @@
"@trpc/next": "^11.17.0",
"@trpc/react-query": "^11.17.0",
"@trpc/server": "^11.17.0",
"@types/mdx": "^2.0.14",
"@uiw/react-md-editor": "^4.1.1",
"@uploadthing/react": "^7.3.3",
"@vercel/speed-insights": "^2.0.0",
@@ -94,7 +99,6 @@
"react-day-picker": "^10.0.1",
"react-dom": "^19.2.6",
"react-hook-form": "^7.77.0",
"react-markdown": "^10.1.0",
"react-resizable-panels": "^4.11.2",
"recharts": "3.8.1",
"rehype-highlight": "^7.0.2",
@@ -103,11 +107,13 @@
"server-only": "^0.0.1",
"shadcn": "^4.10.0",
"sonner": "^2.0.7",
"superjson": "^2.2.6",
"tailwind-merge": "^3.6.0",
"tailwindcss-motion": "^1.1.1",
"type-fest": "^5.7.0",
"uploadthing": "^7.7.4",
"vaul": "^1.1.2",
"wavesurfer.js": "^7.12.8",
"zod": "^4.4.3"
},
"devDependencies": {

View File

@@ -1,62 +1,46 @@
import { useGSAP } from "@gsap/react";
"use client"
import { useRef, type HTMLAttributes, type ReactNode } from "react";
import { useGsapContext } from "~/app/_providers/GsapProvicer";
import { SplitText } from "gsap/SplitText";
import gsap from 'gsap'
import { cn } from "~/lib/utils";
import { useReveal } from "./useReveal";
const AnimateTextIn = ({
children,
animation = "type",
position = 0,
tlId = undefined,
speed = 1,
scrollOnly = false,
once = false,
debugId,
className
}: {
children: ReactNode,
animation?: "type" | "slide",
position?: gsap.Position,
tlId?: string,
scrollOnly?: boolean,
once?: boolean,
debugId?: string,
speed?: number,
className?: HTMLAttributes<HTMLDivElement>['className']
}) => {
const el = useRef<HTMLDivElement>(null)
const gsapContext = useGsapContext();
useGSAP(() => {
const rect = el.current?.getBoundingClientRect()
const scroller = gsapContext?.getScroller()
console.log(scroller)
let viewportTop = 0
let viewportBottom = window.innerHeight
if (scroller && scroller instanceof Element) {
const scrollerRect = scroller.getBoundingClientRect()
viewportTop = scrollerRect.top
viewportBottom = scrollerRect.top + scrollerRect.height
}
const isInView = rect && rect.bottom > viewportTop && rect.top < viewportBottom
console.log(isInView)
const chars = new SplitText(el.current, { type: 'chars' })
gsapContext?.addAnimation(gsap.to(el.current, { opacity: 100, duration: 0 }), 0, tlId)
const fromVars = animation === "slide"
? { opacity: 0, x: -10, duration: 0.2 * speed, stagger: { each: 0.08 * speed }, ease: 'bounce.inOut', onComplete: () => chars.revert() }
: { opacity: 0, duration: 0.01 * speed, stagger: { each: 0.04 * speed }, ease: 'bounce.inOut', onComplete: () => chars.revert() }
if (isInView && !scrollOnly) {
gsapContext?.addAnimation(gsap.from(chars.chars, fromVars), position, tlId)
} else {
gsap.from(chars.chars,
{
...fromVars,
scrollTrigger: {
trigger: el.current,
start: 'top bottom',
end: 'bottom top',
toggleActions: "play reverse play reverse",
scroller
}
})
}
}, { dependencies: [] })
useReveal(el, {
position,
scrollOnly,
once,
debugId: debugId ?? `text-${position}`,
makeReveal: (node) => {
// The wrapper starts at opacity 0 (so there's no flash of unsplit text);
// reveal the wrapper and let the per-character tween do the animation.
gsap.set(node, { opacity: 1 })
const split = new SplitText(node, { type: 'chars' })
const fromVars = animation === "slide"
? { opacity: 0, x: -10, duration: 0.2 * speed, stagger: { each: 0.08 * speed }, ease: 'bounce.inOut' }
: { opacity: 0, duration: 0.01 * speed, stagger: { each: 0.04 * speed }, ease: 'bounce.inOut' }
return gsap.from(split.chars, { ...fromVars, paused: true })
},
})
return (
<div ref={el} className={cn(className, "opacity-0")}>
{children}

View File

@@ -6,16 +6,22 @@ const AnimatePopUp = ({
position,
className,
duration=1,
ease='elastic'
ease='elastic',
scrollOnly=false,
once=false,
debugId,
}:{
children:ReactNode
position:gsap.Position,
className?:HTMLAttributes<HTMLDivElement>['className']
duration?:number,
ease?:gsap.EaseString|gsap.EaseFunction
ease?:gsap.EaseString|gsap.EaseFunction,
scrollOnly?:boolean,
once?:boolean,
debugId?:string,
}) => {
return (
<AnimatedDiv children={children} position={position} className={cn(className,'h-0 translate-y-[50] overflow-hidden')} height='auto' y={0} overflow='' ease={ease} duration={duration} />
<AnimatedDiv children={children} position={position} scrollOnly={scrollOnly} once={once} debugId={debugId} className={cn(className,'h-0 translate-y-[50] overflow-hidden')} height='auto' y={0} overflow='' ease={ease} duration={duration} />
)
}

View File

@@ -1,36 +1,40 @@
"use client"
import gsap from "gsap";
import { type HTMLAttributes,
type ReactNode, useLayoutEffect, useRef } from "react";
import { useGsapContext } from "~/app/_providers/GsapProvicer";
import { type HTMLAttributes, type ReactNode, useRef } from "react";
import { useReveal } from "./useReveal";
const AnimatedDiv = (
{
children,
position,
className,
animationMode='to',
animationMode = 'to',
scrollOnly = false,
once = false,
debugId,
...tweenVars
}:
gsap.TweenVars & {
children:ReactNode,
position:gsap.Position,
animationMode?: 'from'|'to',
className?:HTMLAttributes<HTMLDivElement>['className']
children: ReactNode,
position: gsap.Position,
animationMode?: 'from' | 'to',
scrollOnly?: boolean,
once?: boolean,
debugId?: string,
className?: HTMLAttributes<HTMLDivElement>['className']
}
) => {
const div = useRef<HTMLDivElement>(null);
const gsapContext = useGsapContext()
useLayoutEffect(() => {
let tween:gsap.core.Tween;
switch(animationMode) {
case 'from':
tween = gsap.from(div.current,tweenVars);
break;
case 'to':
tween = gsap.to(div.current,tweenVars);
break;
}
gsapContext?.addAnimation(tween,position)
},[])
const div = useRef<HTMLDivElement>(null)
useReveal(div, {
position,
scrollOnly,
once,
debugId,
makeReveal: (el) =>
animationMode === 'from'
? gsap.from(el, { ...tweenVars, paused: true })
: gsap.to(el, { ...tweenVars, paused: true }),
})
return (
<div ref={div} className={className}>
{children}

View File

@@ -0,0 +1,134 @@
'use client'
import { useGSAP } from "@gsap/react"
import { ScrollTrigger } from "gsap/all"
import type { RefObject } from "react"
import { GSAP_DEBUG, nearestScroller, useGsapContext } from "~/app/_providers/GsapProvicer"
export type UseRevealOptions = {
position: gsap.Position
/** Skip the orchestrated entrance and let ScrollTrigger drive from the start. */
scrollOnly?: boolean
/**
* Reveal once and keep it: after the element animates in (entrance or first
* scroll-in) it never reverses on leave. Default false = animate out at the
* top and back in on scroll-up.
*/
once?: boolean
debugId?: string
/**
* Build the hidden -> shown animation for `el`. It must be a single,
* *independent* animation (not added to any timeline): `play()` reveals,
* `reverse()` hides. The hook pauses it, schedules its entrance through the
* shared timeline, and lets a ScrollTrigger drive the very same animation on
* scroll — so the two modes never fight over the element.
*/
makeReveal: (el: HTMLElement) => gsap.core.Tween | gsap.core.Timeline
}
/**
* Shared reveal behavior for cards, text and pop-ups: an element in view at
* load plays an orchestrated timeline entrance, then hands the *same* tween to
* a ScrollTrigger that animates it out at the top and back in on scroll-up. An
* element off-screen at load is ScrollTrigger-driven from the start.
*/
export function useReveal(
ref: RefObject<HTMLElement | null>,
{ position, scrollOnly = false, once = false, debugId, makeReveal }: UseRevealOptions,
) {
const ctx = useGsapContext()
useGSAP(() => {
const el = ref.current
if (!el || !ctx) {
if (GSAP_DEBUG) console.log("[cv-debug][useReveal:skip]", { debugId, hasEl: !!el, hasCtx: !!ctx })
return
}
const scroller = nearestScroller(el)
const scrollerEl = scroller instanceof Element ? scroller : undefined
const rect = el.getBoundingClientRect()
let top = 0
let bottom = window.innerHeight
if (scrollerEl) {
const r = scrollerEl.getBoundingClientRect()
top = r.top
bottom = r.top + r.height
}
const isInView = rect.bottom > top && rect.top < bottom
if (GSAP_DEBUG) {
const scrollerRect = scrollerEl?.getBoundingClientRect()
console.log("[cv-debug][useReveal:register]", {
debugId,
position,
scrollOnly,
once,
isInView,
rect: { top: rect.top, bottom: rect.bottom, height: rect.height },
viewport: { top, bottom },
scroller:
scroller === window
? "window"
: {
slot: scrollerEl?.getAttribute("data-slot"),
className: scrollerEl?.className,
clientHeight: scrollerEl?.clientHeight,
scrollHeight: scrollerEl?.scrollHeight,
rect: scrollerRect ? { top: scrollerRect.top, bottom: scrollerRect.bottom, height: scrollerRect.height } : undefined,
},
})
}
const reveal = makeReveal(el)
// A reveal that animates height (pop-ups) shifts every trigger below it.
// Re-measure as it animates so positions track the real layout instead of
// only correcting at the very end. requestRefresh is throttled + deferred to
// the next frame, so this won't re-enter a ScrollTrigger callback.
reveal.eventCallback("onUpdate", () => ctx.requestRefresh())
reveal.pause()
const baseTrigger = {
trigger: el,
start: "top bottom",
end: "bottom top",
scroller: scrollerEl,
markers: GSAP_DEBUG,
id: GSAP_DEBUG ? debugId : undefined,
}
// Full behavior: in at the bottom, out at the top, and back on scroll-up.
const addReplayTrigger = () =>
ScrollTrigger.create({
...baseTrigger,
onEnter: () => reveal.play(),
onEnterBack: () => reveal.play(),
onLeave: () => reveal.reverse(),
onLeaveBack: () => reveal.reverse(),
})
if (isInView && !scrollOnly) {
// The shared timeline only decides *when* the entrance starts; the reveal
// plays independently so the ScrollTrigger can take it over afterwards.
if (GSAP_DEBUG) console.log("[cv-debug][useReveal:schedule]", { debugId, position })
ctx.schedule(() => reveal.play(), position)
// `once` elements keep their revealed state — no scroll trigger at all.
if (!once) {
if (GSAP_DEBUG) console.log("[cv-debug][useReveal:onReady]", { debugId })
ctx.onReady(addReplayTrigger)
}
} else if (isInView) {
// scrollOnly + already on screen: no enter crossing will fire, so reveal
// now. Keep a trigger for scroll-out unless this is a `once` element.
if (GSAP_DEBUG) console.log("[cv-debug][useReveal:play-now]", { debugId, position })
reveal.play()
if (!once) addReplayTrigger()
} else if (once) {
// Off-screen: reveal when first reached, then self-destruct so it never
// reverses.
if (GSAP_DEBUG) console.log("[cv-debug][useReveal:scroll-once]", { debugId, position })
ScrollTrigger.create({ ...baseTrigger, once: true, onEnter: () => reveal.play() })
} else {
if (GSAP_DEBUG) console.log("[cv-debug][useReveal:scroll-trigger-only]", { debugId, position })
addReplayTrigger()
}
}, { dependencies: [] })
}

View File

@@ -15,7 +15,7 @@ export default function FormScaffold<T extends FieldValues,>(params: {
}) {
const { form, onSubmit, title, id, className, children } = params
return (
<Card.Card className={className ? className : "w-5/6 lg:w-1/2"}>
<Card.Card className={className ? className : "w-full"}>
<Card.CardHeader>
<Card.CardTitle>
<DependentText bool={id ? true : false} true={`Update ${title}`} false={`Create ${title}`} />

View File

@@ -8,6 +8,7 @@ import type { Control, FieldValues, Path } from "react-hook-form";
import { FormControl, FormField, FormItem, FormLabel } from "~/components/ui/form";
import { Button } from "~/components/ui/button";
import { cn } from "~/lib/utils";
import { ClientMdx } from "~/components/ClientMdx";
import {
InternalLinkTextarea,
type AutocompleteTriggerConfig,
@@ -81,7 +82,7 @@ export default function MdeFormField<T extends FieldValues>(params: {
),
preview: params.renderPreview
? (source) => params.renderPreview?.(source) ?? <></>
: undefined,
: (source) => <ClientMdx source={source} fallback={source} />,
}}
/>
</FormControl>

View File

@@ -9,11 +9,51 @@ gsap.registerPlugin(useGSAP)
gsap.registerPlugin(ScrollTrigger)
gsap.registerPlugin(SplitText)
gsap.registerPlugin(GSDevTools)
// iOS Safari shows/hides its address bar at the scroll extremes, which resizes
// the viewport and makes ScrollTrigger refresh + fire spurious onLeave/onEnter
// toggles (text animating out at the bottom and not coming back). Ignoring those
// mobile-toolbar resizes keeps the real enter/leave reverse behavior intact.
ScrollTrigger.config({ ignoreMobileResize: true })
// Flip to true to draw ScrollTrigger start/end markers on every animated
// element and mount the GSDevTools timeline scrubber. Handy for seeing exactly
// where each card's enter/exit lines sit relative to the viewport.
export const GSAP_DEBUG = false
export function nearestScroller(el: Element): Element | Window {
let node: Element | null = el.parentElement
while (node) {
if (node.getAttribute('data-slot') === 'scroll-area-viewport') {
const viewport = node as HTMLElement
const rect = viewport.getBoundingClientRect()
const hasUsableBox = rect.width > 0 && rect.height > 0
const canScroll =
viewport.scrollHeight > viewport.clientHeight ||
viewport.scrollWidth > viewport.clientWidth
if (hasUsableBox && canScroll) return viewport
}
node = node.parentElement
}
return window
}
const GsapContext = createContext<{
// Add a real animation (with its own duration) to the entrance timeline.
addAnimation: (
animation: gsap.core.TimelineChild,
position: gsap.Position
) => void,
// Schedule a zero-duration callback at `position` — used to *start* an
// independent reveal tween so the timeline only orchestrates timing.
schedule: (fn: () => void, position: gsap.Position) => void,
// Run `cb` once the entrance is done (timeline complete or first user scroll).
onReady: (cb: () => void) => void,
// Re-measure all ScrollTriggers (throttled to once per frame). Call it
// whenever an animation changes content height so trigger positions stay
// aligned with the real layout.
requestRefresh: () => void,
resetTimeline: () => void,
resumeTimeline: () => void,
getScroller: () => Element | Window | null
@@ -26,14 +66,24 @@ export function useGsapContext() {
export const useTimeLine = (dep:any,all?:boolean) => {
const gsapContext = useGsapContext()
useEffect(() => {
if (GSAP_DEBUG) {
console.log("[cv-debug][useTimeLine:effect]", {
hasDep: !!dep,
isArray: dep instanceof Array,
length: dep instanceof Array ? dep.length : undefined,
all,
})
}
if (dep instanceof Array && all) {
let acc = true;
let allDepsSatisfied = dep.reduce((p,c) => c !== undefined && p ,acc )
if (allDepsSatisfied) {
if (GSAP_DEBUG) console.log("[cv-debug][useTimeLine:resume-all]")
gsapContext?.resumeTimeline()
}
} else {
if (dep) {
if (GSAP_DEBUG) console.log("[cv-debug][useTimeLine:resume]")
gsapContext?.resumeTimeline()
}
}
@@ -70,29 +120,174 @@ export default function GsapProvider({ children }: { children: ReactNode }) {
// }
return scrollerRef.current
}, [])
const devToolsCreated = useRef(false)
useGSAP(() => {
if (!tl.current) {
tl.current = gsap.timeline({ paused: true })
}
return () => { console.log("gsap cleanup") }
if (GSAP_DEBUG && tl.current && !devToolsCreated.current) {
devToolsCreated.current = true
GSDevTools.create({ animation: tl.current })
}
return () => { if (GSAP_DEBUG) console.log("gsap cleanup") }
})
const addAnimation = useCallback((animation: gsap.core.TimelineChild, position: gsap.Position) => {
console.log("add animation to:", position, tl.current !== undefined)
tl.current?.add(animation, position);
// Handoff: fire registered callbacks once, when the entrance finishes.
const readyFired = useRef(false)
const readyCbs = useRef<Array<() => void>>([])
const fireReady = useCallback(() => {
if (readyFired.current) return
if (GSAP_DEBUG) {
console.log("[cv-debug][gsap:ready]", {
callbacks: readyCbs.current.length,
duration: tl.current?.duration(),
progress: tl.current?.progress(),
})
}
readyFired.current = true
readyCbs.current.forEach((cb) => cb())
readyCbs.current = []
},[])
const onReady = useCallback((cb: () => void) => {
if (GSAP_DEBUG) console.log("[cv-debug][gsap:onReady]", { readyFired: readyFired.current })
if (readyFired.current) cb()
else readyCbs.current.push(cb)
},[])
const addAnimation = useCallback((animation: gsap.core.TimelineChild, position: gsap.Position) => {
// Content can mount in waves (e.g. nested queries resolving after the
// entrance already played). Parking a tween in a finished, paused timeline
// would freeze it at its from-state, so once the entrance is done let the
// (live) tween play on its own instead.
if (GSAP_DEBUG) {
console.log("[cv-debug][gsap:addAnimation]", {
position,
readyFired: readyFired.current,
durationBefore: tl.current?.duration(),
})
}
if (readyFired.current) return
tl.current?.add(animation, position);
if (GSAP_DEBUG) {
console.log("[cv-debug][gsap:addAnimation:done]", {
position,
durationAfter: tl.current?.duration(),
children: tl.current?.getChildren(false, true, true).length,
})
}
},[])
const schedule = useCallback((fn: () => void, position: gsap.Position) => {
// Same late-arrival case: a callback added past the playhead never fires, so
// run the reveal immediately once the entrance has finished.
if (GSAP_DEBUG) {
console.log("[cv-debug][gsap:schedule]", {
position,
readyFired: readyFired.current,
durationBefore: tl.current?.duration(),
childrenBefore: tl.current?.getChildren(false, true, true).length,
})
}
if (readyFired.current) { fn(); return }
tl.current?.add(fn, position)
if (GSAP_DEBUG) {
console.log("[cv-debug][gsap:schedule:done]", {
position,
durationAfter: tl.current?.duration(),
childrenAfter: tl.current?.getChildren(false, true, true).length,
})
}
},[])
// Throttle ScrollTrigger.refresh() to once per frame so the ResizeObserver
// can fire freely while content height animates.
const refreshQueued = useRef(false)
const scheduleRefresh = useCallback(() => {
if (refreshQueued.current) return
refreshQueued.current = true
requestAnimationFrame(() => {
refreshQueued.current = false
ScrollTrigger.refresh()
})
},[])
const scrollCleanup = useRef<(() => void) | null>(null)
const resizeObserver = useRef<ResizeObserver | null>(null)
const resetTimeline = useCallback(() => {
if (GSAP_DEBUG) {
console.log("[cv-debug][gsap:reset]", {
duration: tl.current?.duration(),
progress: tl.current?.progress(),
})
}
tl.current?.kill()
tl.current?.revert()
ScrollTrigger.getAll().forEach(st => st.kill())
resizeObserver.current?.disconnect()
scrollCleanup.current?.()
scrollCleanup.current = null
readyFired.current = false
readyCbs.current = []
tl.current = gsap.timeline({paused:true})
},[])
const resumeTimeline = useCallback(() => {
console.log("resuming timeline:",tl.current)
tl.current?.resume()
},[])
const t = tl.current
if (!t) {
if (GSAP_DEBUG) console.log("[cv-debug][gsap:resume:skip-no-timeline]")
return
}
if (GSAP_DEBUG) {
console.log("[cv-debug][gsap:resume:start]", {
duration: t.duration(),
progress: t.progress(),
paused: t.paused(),
readyFired: readyFired.current,
children: t.getChildren(false, true, true).length,
})
}
// When the orchestrated entrance finishes, hand off to scroll control and
// realign triggers against the now-settled layout.
t.eventCallback("onComplete", () => { fireReady(); ScrollTrigger.refresh() })
const scroller = getScroller()
// If the user scrolls before the entrance finishes, snap it to the end and
// switch to scroll control so the timeline and ScrollTriggers never fight
// over the same elements.
scrollCleanup.current?.()
const onFirstScroll = () => { t.progress(1); fireReady() }
scroller?.addEventListener("scroll", onFirstScroll, { once: true, passive: true })
scrollCleanup.current = () => scroller?.removeEventListener("scroll", onFirstScroll)
// Continuously realign triggers while content height changes — entrance
// growth, scroll-driven collapses, late-loading media.
if (scroller instanceof Element) {
const target = scroller.firstElementChild ?? scroller
resizeObserver.current?.disconnect()
resizeObserver.current = new ResizeObserver(scheduleRefresh)
resizeObserver.current.observe(target)
}
t.resume()
if (GSAP_DEBUG) {
console.log("[cv-debug][gsap:resume:after]", {
duration: t.duration(),
progress: t.progress(),
paused: t.paused(),
})
}
},[getScroller, fireReady, scheduleRefresh])
// Fonts/markdown/images loading also change content height after the triggers
// were created; refresh so start/end stay aligned with the real card sizes.
useEffect(() => {
const refresh = () => ScrollTrigger.refresh()
window.addEventListener("load", refresh)
document.fonts?.ready.then(refresh).catch(() => {})
return () => window.removeEventListener("load", refresh)
}, [])
return (
<GsapContext.Provider value={{ addAnimation, resetTimeline, resumeTimeline, getScroller }}>
<GsapContext.Provider value={{ addAnimation, schedule, onReady, requestRefresh: scheduleRefresh, resetTimeline, resumeTimeline, getScroller }}>
{children}
</GsapContext.Provider>
)

View File

@@ -1,22 +1,3 @@
import { httpBatchLink } from "@trpc/client";
import { trpcRouter } from "~/server/routers/_app";
function getBaseUrl() {
if (typeof window !== 'undefined')
// browser should use relative path
return '';
if (process.env.VERCEL_URL)
// reference for vercel.com
return `https://${process.env.VERCEL_URL}`;
if (process.env.RENDER_INTERNAL_HOSTNAME)
// reference for render.com
return `http://${process.env.RENDER_INTERNAL_HOSTNAME}:${process.env.PORT}`;
// assume localhost
return `http://localhost:${process.env.PORT ?? 3000}`;
};
export const servTrpc = trpcRouter.createCaller({
links: [
httpBatchLink({url: `${getBaseUrl()}/api/trpc`}),
],
});
export const servTrpc = trpcRouter.createCaller({});

View File

@@ -3,6 +3,7 @@ import { QueryClient, QueryClientProvider } from "@tanstack/react-query";
import { ReactQueryStreamedHydration } from '@tanstack/react-query-next-experimental'
import { httpBatchLink } from "@trpc/client";
import React, { useState } from "react"
import superjson from "superjson";
import { trpc } from "./Client";
import getBaseUrl from "~/app/_trpc/GetBaseUrl";
let clientQueryClient: QueryClient | undefined = undefined;
@@ -33,6 +34,7 @@ export default function TrpcProvider({ children }: { children: React.ReactNode }
links: [
httpBatchLink({
url: `${baseUrl}/api/trpc`,
transformer: superjson,
}),
],
})

View File

@@ -0,0 +1,25 @@
'use server'
import { getGoogleCalendarClient, getGoogleCalendarId } from '~/server/googleCalendar'
export async function cancelMeeting({ eventId }: { eventId: string }) {
try {
const calendar = getGoogleCalendarClient()
await calendar.events.delete({
calendarId: getGoogleCalendarId(),
eventId,
})
return {
success: true,
eventId,
message: 'Meeting removed from Gregor availability calendar.',
}
} catch (error) {
console.error('Failed to cancel meeting:', error)
return {
success: false,
error: 'Failed to remove the meeting from Gregor availability calendar.',
}
}
}

View File

@@ -1,7 +1,34 @@
'use server'
import { clerkClient, auth } from '@clerk/nextjs/server'
import { google } from 'googleapis'
import { env } from '~/env'
import { getGoogleCalendarClient, getGoogleCalendarId } from '~/server/googleCalendar'
function googleCalendarDate(date: Date) {
return date.toISOString().replace(/[-:]|\.\d{3}/g, '')
}
function createGoogleCalendarTemplateLink({
title,
description,
startTime,
endTime,
gregorEmail,
}: {
title: string
description: string
startTime: Date
endTime: Date
gregorEmail: string
}) {
const params = new URLSearchParams({
action: 'TEMPLATE',
text: title,
dates: `${googleCalendarDate(startTime)}/${googleCalendarDate(endTime)}`,
details: description,
add: gregorEmail,
})
return `https://calendar.google.com/calendar/render?${params.toString()}`
}
export async function scheduleMeeting({
title,
@@ -19,56 +46,39 @@ export async function scheduleMeeting({
attendeeName?: string
}) {
try {
const clerk = await clerkClient()
const userAuth = await auth()
const user = await clerk.users.getUser(userAuth.userId?userAuth.userId:"")
// Get admin's Google OAuth token to create the event on Gregor's calendar
const adminTokenResponse = await clerk.users.getUserOauthAccessToken(
env.ADMIN_USER_CLERK_ID,
'oauth_google',
)
const adminToken = adminTokenResponse.data[0]
if (!adminToken?.token) {
return { success: false, error: 'Admin Google Calendar not connected. Ensure the admin account is linked with Google and has calendar scope enabled.' }
}
// Try to resolve visitor's Google email for the invite
let visitorEmail: string | undefined = attendeeEmail
if (!visitorEmail) {
visitorEmail = user?.emailAddresses.at(0)?.emailAddress ?? undefined
}
const oAuth2Client = new google.auth.OAuth2()
oAuth2Client.setCredentials({ access_token: adminToken.token })
const calendar = google.calendar({ version: 'v3', auth: oAuth2Client })
const calendar = getGoogleCalendarClient()
const startTime = new Date(dateTime)
const endTime = new Date(startTime.getTime() + durationMinutes * 60 * 1000)
const attendeeNote = attendeeEmail
? `\n\nVisitor: ${attendeeName ?? 'Unknown'} <${attendeeEmail}>`
: ''
const eventDescription = `${description}${attendeeNote}`
const attendees: { email: string; displayName?: string }[] = []
if (visitorEmail) {
attendees.push({ email: visitorEmail, displayName: attendeeName })
const eventRequest = {
summary: title,
description: eventDescription,
start: { dateTime: startTime.toISOString(), timeZone: 'UTC' },
end: { dateTime: endTime.toISOString(), timeZone: 'UTC' },
}
const event = await calendar.events.insert({
calendarId: 'primary',
sendUpdates: 'all',
requestBody: {
summary: title,
description,
start: { dateTime: startTime.toISOString(), timeZone: 'UTC' },
end: { dateTime: endTime.toISOString(), timeZone: 'UTC' },
attendees,
},
sendNotifications: true
calendarId: getGoogleCalendarId(),
requestBody: eventRequest,
})
const addToCalendarLink = createGoogleCalendarTemplateLink({
title,
description,
startTime,
endTime,
gregorEmail: env.GREGOR_MEETING_EMAIL,
})
return {
success: true,
eventId: event.data.id,
htmlLink: event.data.htmlLink,
message: `Meeting "${title}" scheduled for ${startTime.toLocaleString()}${visitorEmail ? `. Invite sent to ${visitorEmail}.` : '.'}`,
addToCalendarLink,
message: `Meeting "${title}" scheduled for ${startTime.toLocaleString()}.${attendeeEmail ? ` Visitor email noted: ${attendeeEmail}.` : ''} The add-to-calendar link invites ${env.GREGOR_MEETING_EMAIL}.`,
}
} catch (error) {
console.error('Failed to schedule meeting:', error)

View File

@@ -0,0 +1,98 @@
import { ChevronsUpDown } from "lucide-react";
import { Accordion, AccordionContent, AccordionItem, AccordionTrigger } from "~/components/ui/accordion";
import { Collapsible, CollapsibleContent, CollapsibleTrigger } from "~/components/ui/collapsible";
const examples = [
{
name: "Lead",
description: "Intro paragraph with larger muted text.",
code: `<Lead>
Short opening summary.
</Lead>`,
},
{
name: "Callout",
description: "Highlighted note, tip, or warning block.",
code: `<Callout title="Heads up" variant="note">
Important context for readers.
</Callout>
<Callout title="Tip" variant="tip">
A practical recommendation.
</Callout>
<Callout title="Careful" variant="warning">
A caveat or tradeoff.
</Callout>`,
},
{
name: "ButtonLink",
description: "Button-styled internal or external link.",
code: `<ButtonLink href="/projects">
View projects
</ButtonLink>
<ButtonLink href="https://example.com" variant="outline">
External resource
</ButtonLink>`,
},
{
name: "Figure",
description: "Image with optional caption.",
code: `<Figure
src="https://example.com/image.jpg"
alt="Describe the image"
caption="Optional caption"
/>`,
},
{
name: "PullQuote",
description: "Large emphasized quote or takeaway.",
code: `<PullQuote>
A highlighted quote or strong takeaway.
</PullQuote>`,
},
{
name: "TagList",
description: "Inline list of tag badges.",
code: `<TagList tags={["nextjs", "mdx", "uploadthing"]} />`,
},
{
name: "Badge",
description: "Small inline label.",
code: `<Badge variant="outline">Next.js</Badge>`,
},
];
export default function MdxComponentReference() {
return (
<Collapsible className="rounded-lg border">
<CollapsibleTrigger className="flex w-full items-center justify-between gap-2 p-4 text-left">
<h2 className="text-base font-semibold">MDX Components</h2>
<ChevronsUpDown className="size-4 shrink-0 text-muted-foreground" />
</CollapsibleTrigger>
<CollapsibleContent className="px-4 pb-4">
<p className="text-muted-foreground text-sm">
Components available inside MDX content. Type <code className="rounded bg-muted px-1">[[</code> for internal links or <code className="rounded bg-muted px-1">&lt;</code> for component snippets.
</p>
<Accordion type="single" collapsible className="mt-3">
{examples.map((example) => (
<AccordionItem key={example.name} value={example.name}>
<AccordionTrigger>
<span>
<span className="block">{example.name}</span>
<span className="text-muted-foreground block text-xs font-normal">{example.description}</span>
</span>
</AccordionTrigger>
<AccordionContent>
<pre className="bg-muted overflow-x-auto rounded-md p-3 text-xs">
<code>{example.code}</code>
</pre>
</AccordionContent>
</AccordionItem>
))}
</Accordion>
</CollapsibleContent>
</Collapsible>
);
}

View File

@@ -4,9 +4,9 @@ import { useEffect, useState } from 'react'
import { MDXRemote } from 'next-mdx-remote'
import { serialize } from 'next-mdx-remote/serialize'
import type { MDXRemoteSerializeResult } from 'next-mdx-remote'
import { mdxComponents } from '~/app/blog/_components/mdx-components'
import { mdxComponents } from '~/components/mdx-components'
export default function BlogMdxEditorPreview(params: { source: string }) {
export default function MdxEditorPreview(params: { source: string }) {
const [compiled, setCompiled] = useState<MDXRemoteSerializeResult | null>(null)
const [error, setError] = useState<string | null>(null)

View File

@@ -0,0 +1,143 @@
'use client'
import { trpc } from '~/app/_trpc/Client'
import type { RouterOutputs } from '~/server/routers/_app'
import {
AUTOCOMPLETE_CURSOR_MARKER,
linkSuggestionsToAutocomplete,
type AutocompleteTriggerConfig,
type InternalLinkSuggestion,
type MdeAutocompleteSuggestion,
} from '~/app/_components/Form/Fields/InternalLinkTextarea'
import MdxEditorPreview from './MdxEditorPreview'
function internalLinkSuggestions(params: {
posts?: RouterOutputs['blog']['list'],
projects?: RouterOutputs['projectv2']['listWithStack'],
}): InternalLinkSuggestion[] {
const postLinks = params.posts?.map((post) => ({
label: post.title,
href: `/blog/${post.slug}`,
group: 'Blog',
})) ?? []
const projectLinks = params.projects?.map((project) => ({
label: project.title,
href: `/projects#${project.id}`,
group: 'Project',
})) ?? []
return [...postLinks, ...projectLinks]
}
const mdxAutocompleteSuggestions: MdeAutocompleteSuggestion[] = [
{
label: 'Lead',
value: `<Lead>\n${AUTOCOMPLETE_CURSOR_MARKER}\n</Lead>`,
detail: 'Intro paragraph with larger muted text.',
group: 'Component',
trigger: '<',
},
{
label: 'Callout note',
value: `<Callout title="Heads up" variant="note">\n${AUTOCOMPLETE_CURSOR_MARKER}\n</Callout>`,
detail: 'Highlighted note block.',
group: 'Component',
trigger: '<',
},
{
label: 'Callout tip',
value: `<Callout title="Tip" variant="tip">\n${AUTOCOMPLETE_CURSOR_MARKER}\n</Callout>`,
detail: 'Highlighted tip block.',
group: 'Component',
trigger: '<',
},
{
label: 'Callout warning',
value: `<Callout title="Careful" variant="warning">\n${AUTOCOMPLETE_CURSOR_MARKER}\n</Callout>`,
detail: 'Highlighted warning block.',
group: 'Component',
trigger: '<',
},
{
label: 'ButtonLink',
value: `<ButtonLink href="${AUTOCOMPLETE_CURSOR_MARKER}">\nView projects\n</ButtonLink>`,
detail: 'Button-styled internal or external link.',
group: 'Component',
trigger: '<',
},
{
label: 'Figure',
value: `<Figure\n src="${AUTOCOMPLETE_CURSOR_MARKER}"\n alt="Describe the image"\n caption="Optional caption"\n/>`,
detail: 'Image with optional caption.',
group: 'Component',
trigger: '<',
},
{
label: 'PullQuote',
value: `<PullQuote>\n${AUTOCOMPLETE_CURSOR_MARKER}\n</PullQuote>`,
detail: 'Large emphasized quote.',
group: 'Component',
trigger: '<',
},
{
label: 'TagList',
value: `<TagList tags={[${AUTOCOMPLETE_CURSOR_MARKER}]} />`,
detail: 'Inline list of tag badges.',
group: 'Component',
trigger: '<',
},
{
label: 'Badge',
value: `<Badge variant="outline">${AUTOCOMPLETE_CURSOR_MARKER}</Badge>`,
detail: 'Small inline label.',
group: 'Component',
trigger: '<',
},
{
label: 'Image',
value: `![Image](${AUTOCOMPLETE_CURSOR_MARKER})`,
detail: 'Markdown image',
group: 'Markdown',
trigger: '!',
},
]
const mdxTriggerConfigs: AutocompleteTriggerConfig[] = [
{
trigger: '[[',
label: 'Internal links',
isQueryValid: (query) => !query.includes(']'),
},
{
trigger: '<',
label: 'MDX components',
isQueryValid: (query) => !/[\s>]/.test(query),
},
{
trigger: '!',
label: 'Markdown',
isQueryValid: (query) => !/[\s\)]/.test(query),
},
]
/**
* Shared props for an MDX-aware `MdeFormField`: internal-link + component
* autocomplete, trigger configs, and a live MDX preview. Used by every admin
* form that edits MDX content (blog, project, cv entry).
*/
export function useMdxEditorFieldProps() {
const posts = trpc.blog.list.useQuery(undefined, { refetchInterval: 5000 })
const projects = trpc.projectv2.listWithStack.useQuery()
const autocompleteSuggestions = [
...linkSuggestionsToAutocomplete(internalLinkSuggestions({ posts: posts.data, projects: projects.data })),
...mdxAutocompleteSuggestions,
]
return {
autocompleteSuggestions,
triggerConfigs: mdxTriggerConfigs,
renderPreview: (source: string) => <MdxEditorPreview source={source} />,
}
}

View File

@@ -10,15 +10,8 @@ import { TextInputFormField, MdeFormField } from '~/app/_components/Form/Fields'
import { usePathname, useRouter } from 'next/navigation'
import { useTheme } from 'next-themes'
import type { RouterOutputs } from '~/server/routers/_app'
import MdxComponentReference from './MdxComponentReference'
import BlogMdxEditorPreview from './BlogMdxEditorPreview'
import {
AUTOCOMPLETE_CURSOR_MARKER,
linkSuggestionsToAutocomplete,
type AutocompleteTriggerConfig,
type InternalLinkSuggestion,
type MdeAutocompleteSuggestion,
} from '~/app/_components/Form/Fields/InternalLinkTextarea'
import MdxComponentReference from '~/app/admin/_components/MdxComponentReference'
import { useMdxEditorFieldProps } from '~/app/admin/_components/useMdxEditorFieldProps'
type BlogPost = RouterOutputs['blog']['bySlug']
@@ -35,116 +28,6 @@ function parseTags(value: string | undefined): string[] {
return value?.split(',').map((tag) => tag.trim()).filter(Boolean) ?? []
}
function internalLinkSuggestions(params: {
posts?: RouterOutputs['blog']['list'],
projects?: RouterOutputs['projectv2']['listWithStack'],
}): InternalLinkSuggestion[] {
const postLinks = params.posts?.map((post) => ({
label: post.title,
href: `/blog/${post.slug}`,
group: 'Blog',
})) ?? []
const projectLinks = params.projects?.map((project) => ({
label: project.title,
href: `/projects#${project.id}`,
group: 'Project',
})) ?? []
return [...postLinks, ...projectLinks]
}
const blogAutocompleteSuggestions: MdeAutocompleteSuggestion[] = [
{
label: 'Lead',
value: `<Lead>\n${AUTOCOMPLETE_CURSOR_MARKER}\n</Lead>`,
detail: 'Intro paragraph with larger muted text.',
group: 'Component',
trigger: '<',
},
{
label: 'Callout note',
value: `<Callout title="Heads up" variant="note">\n${AUTOCOMPLETE_CURSOR_MARKER}\n</Callout>`,
detail: 'Highlighted note block.',
group: 'Component',
trigger: '<',
},
{
label: 'Callout tip',
value: `<Callout title="Tip" variant="tip">\n${AUTOCOMPLETE_CURSOR_MARKER}\n</Callout>`,
detail: 'Highlighted tip block.',
group: 'Component',
trigger: '<',
},
{
label: 'Callout warning',
value: `<Callout title="Careful" variant="warning">\n${AUTOCOMPLETE_CURSOR_MARKER}\n</Callout>`,
detail: 'Highlighted warning block.',
group: 'Component',
trigger: '<',
},
{
label: 'ButtonLink',
value: `<ButtonLink href="${AUTOCOMPLETE_CURSOR_MARKER}">\nView projects\n</ButtonLink>`,
detail: 'Button-styled internal or external link.',
group: 'Component',
trigger: '<',
},
{
label: 'Figure',
value: `<Figure\n src="${AUTOCOMPLETE_CURSOR_MARKER}"\n alt="Describe the image"\n caption="Optional caption"\n/>`,
detail: 'Image with optional caption.',
group: 'Component',
trigger: '<',
},
{
label: 'PullQuote',
value: `<PullQuote>\n${AUTOCOMPLETE_CURSOR_MARKER}\n</PullQuote>`,
detail: 'Large emphasized quote.',
group: 'Component',
trigger: '<',
},
{
label: 'TagList',
value: `<TagList tags={[${AUTOCOMPLETE_CURSOR_MARKER}]} />`,
detail: 'Inline list of tag badges.',
group: 'Component',
trigger: '<',
},
{
label: 'Badge',
value: `<Badge variant="outline">${AUTOCOMPLETE_CURSOR_MARKER}</Badge>`,
detail: 'Small inline label.',
group: 'Component',
trigger: '<',
},
{
label: 'Image',
value: `![Image](${AUTOCOMPLETE_CURSOR_MARKER})`,
detail: 'Markdown image',
group: 'Markdown',
trigger: '!',
},
]
const blogTriggerConfigs: AutocompleteTriggerConfig[] = [
{
trigger: '[[',
label: 'Internal links',
isQueryValid: (query) => !query.includes(']'),
},
{
trigger: '<',
label: 'MDX components',
isQueryValid: (query) => !/[\s>]/.test(query),
},
{
trigger: '!',
label: 'Markdown',
isQueryValid: (query) => !/[\s\)]/.test(query),
},
]
export default function CreateUpdateBlogForm(params: { className?: string, entity?: BlogPost }) {
const [slug, setSlug] = useState<string | undefined>(params.entity?.slug)
const [originalSlug, setOriginalSlug] = useState<string | undefined>(params.entity?.slug)
@@ -162,12 +45,7 @@ export default function CreateUpdateBlogForm(params: { className?: string, entit
})
const path = usePathname()
const router = useRouter()
const posts = trpc.blog.list.useQuery(undefined, { refetchInterval: 5000 })
const projects = trpc.projectv2.listWithStack.useQuery()
const autocompleteSuggestions = [
...linkSuggestionsToAutocomplete(internalLinkSuggestions({ posts: posts.data, projects: projects.data })),
...blogAutocompleteSuggestions,
]
const mdxEditorProps = useMdxEditorFieldProps()
const createMutation = trpc.blog.insert.useMutation({
onSuccess: (data) => {
@@ -226,9 +104,7 @@ export default function CreateUpdateBlogForm(params: { className?: string, entit
name='content'
label='Content'
dataColorMode={(theme as 'dark' | 'light') ?? 'dark'}
autocompleteSuggestions={autocompleteSuggestions}
triggerConfigs={blogTriggerConfigs}
renderPreview={(source) => <BlogMdxEditorPreview source={source} />}
{...mdxEditorProps}
/>
</FormScaffold>
</div>

View File

@@ -1,91 +0,0 @@
import { Accordion, AccordionContent, AccordionItem, AccordionTrigger } from "~/components/ui/accordion";
const examples = [
{
name: "Lead",
description: "Intro paragraph with larger muted text.",
code: `<Lead>
Short opening summary for the post.
</Lead>`,
},
{
name: "Callout",
description: "Highlighted note, tip, or warning block.",
code: `<Callout title="Heads up" variant="note">
Important context for readers.
</Callout>
<Callout title="Tip" variant="tip">
A practical recommendation.
</Callout>
<Callout title="Careful" variant="warning">
A caveat or tradeoff.
</Callout>`,
},
{
name: "ButtonLink",
description: "Button-styled internal or external link.",
code: `<ButtonLink href="/projects">
View projects
</ButtonLink>
<ButtonLink href="https://example.com" variant="outline">
External resource
</ButtonLink>`,
},
{
name: "Figure",
description: "Image with optional caption.",
code: `<Figure
src="https://example.com/image.jpg"
alt="Describe the image"
caption="Optional caption"
/>`,
},
{
name: "PullQuote",
description: "Large emphasized quote or takeaway.",
code: `<PullQuote>
A highlighted quote or strong takeaway.
</PullQuote>`,
},
{
name: "TagList",
description: "Inline list of tag badges inside the post body.",
code: `<TagList tags={["nextjs", "mdx", "uploadthing"]} />`,
},
{
name: "Badge",
description: "Small inline label.",
code: `<Badge variant="outline">Next.js</Badge>`,
},
];
export default function MdxComponentReference() {
return (
<section className="rounded-lg border p-4">
<h2 className="text-base font-semibold">MDX Components</h2>
<p className="text-muted-foreground mt-1 text-sm">
Components available inside blog post content. Type <code className="rounded bg-muted px-1">[[</code> for internal links or <code className="rounded bg-muted px-1">&lt;</code> for component snippets.
</p>
<Accordion type="single" collapsible className="mt-3">
{examples.map((example) => (
<AccordionItem key={example.name} value={example.name}>
<AccordionTrigger>
<span>
<span className="block">{example.name}</span>
<span className="text-muted-foreground block text-xs font-normal">{example.description}</span>
</span>
</AccordionTrigger>
<AccordionContent>
<pre className="bg-muted overflow-x-auto rounded-md p-3 text-xs">
<code>{example.code}</code>
</pre>
</AccordionContent>
</AccordionItem>
))}
</Accordion>
</section>
);
}

View File

@@ -0,0 +1,48 @@
'use client'
import { trpc } from '~/app/_trpc/Client'
import {
Select,
SelectContent,
SelectItem,
SelectTrigger,
SelectValue,
} from '~/components/ui/select'
export default function ModelSelector({ initialValue }: { initialValue: string }) {
const utils = trpc.useUtils()
const { data: models, isLoading, error } = trpc.chat.listModels.useQuery()
const { data: model = initialValue } = trpc.chat.getModel.useQuery(undefined, {
initialData: initialValue,
})
const mutation = trpc.chat.updateModel.useMutation({
onSuccess: () => utils.chat.getModel.invalidate(),
})
// Ensure the currently-saved model is always selectable, even if the
// OpenAI list doesn't include it (e.g. a deprecated model).
const options = Array.from(new Set([model, ...(models ?? [])])).filter(Boolean)
return (
<div className="flex flex-col gap-2">
<Select value={model} onValueChange={(v) => mutation.mutate({ model: v })}>
<SelectTrigger className="w-72">
<SelectValue placeholder={isLoading ? 'Loading models…' : 'Select a model'} />
</SelectTrigger>
<SelectContent>
{options.map((id) => (
<SelectItem key={id} value={id}>
{id}
</SelectItem>
))}
</SelectContent>
</Select>
<div className="flex items-center gap-3 text-sm text-muted-foreground">
{mutation.isPending && <span>Saving</span>}
{mutation.isSuccess && !mutation.isPending && <span>Saved</span>}
{error && <span className="text-destructive">Failed to load models: {error.message}</span>}
{mutation.error && <span className="text-destructive">{mutation.error.message}</span>}
</div>
</div>
)
}

View File

@@ -1,18 +1,31 @@
import { servTrpc } from '~/app/_trpc/ServerClient'
import SystemPromptForm from './_components/SystemPromptForm'
import ModelSelector from './_components/ModelSelector'
export default async function SystemPromptPage() {
const prompt = await servTrpc.chat.getSystemPrompt()
const model = await servTrpc.chat.getModel()
return (
<div className="w-full max-w-2xl p-6 flex flex-col gap-4">
<div>
<h1 className="text-lg font-semibold">AI System Prompt</h1>
<p className="text-sm text-muted-foreground">
This prompt is sent to the model on every chat request.
</p>
<div className="w-full max-w-2xl p-6 flex flex-col gap-8">
<div className="flex flex-col gap-4">
<div>
<h1 className="text-lg font-semibold">AI Model</h1>
<p className="text-sm text-muted-foreground">
The OpenAI model used to respond to chat requests.
</p>
</div>
<ModelSelector initialValue={model} />
</div>
<div className="flex flex-col gap-4">
<div>
<h1 className="text-lg font-semibold">AI System Prompt</h1>
<p className="text-sm text-muted-foreground">
This prompt is sent to the model on every chat request.
</p>
</div>
<SystemPromptForm initialValue={prompt} />
</div>
<SystemPromptForm initialValue={prompt} />
</div>
)
}

View File

@@ -14,6 +14,8 @@ import { useState } from 'react';
import { SelectFormField, TextInputFormField, MdeFormField, CalenderFormField, BooleanFormField } from '~/app/_components/Form/Fields'
import { usePathname, useRouter } from 'next/navigation';
import {FormMutationContextProvider, type FormCreateMutationInterface} from '~/app/_components/Form/Components/MutationProvider';
import MdxComponentReference from '~/app/admin/_components/MdxComponentReference';
import { useMdxEditorFieldProps } from '~/app/admin/_components/useMdxEditorFieldProps';
export default function CreateUpdateCvEntryForm(params: { className?: string, entity?: IterableElement<RouterOutputs['entry']['select']>, isUpdate?: boolean }) {
const [id, setId] = useState<string | undefined>(params.entity ? params.entity.id : undefined)
const { theme } = useTheme()
@@ -34,6 +36,7 @@ export default function CreateUpdateCvEntryForm(params: { className?: string, en
})
let path = usePathname()
let router = useRouter()
const mdxEditorProps = useMdxEditorFieldProps()
const createMutation = trpc.entry.insert.useMutation({ onSuccess: makeOnSuccess('create', form, setId) })
const updateMutation = trpc.entry.update.useMutation({ onSuccess: makeOnSuccess('update', form) })
const deleteMutation = trpc.entry.delete.useMutation({ onSuccess: makeOnSuccess('delete', form, undefined, path, router) })
@@ -51,6 +54,8 @@ export default function CreateUpdateCvEntryForm(params: { className?: string, en
updateMutation:updateMutation,
deleteMutation:deleteMutation
}}>
<div className='flex flex-col gap-4'>
<MdxComponentReference />
<FormScaffold
form={form}
onSubmit={onSubmit}
@@ -70,11 +75,12 @@ export default function CreateUpdateCvEntryForm(params: { className?: string, en
}
</SelectFormField>
<TextInputFormField control={form.control} name='title' label='Title' />
<MdeFormField control={form.control} name='description' label='Description' dataColorMode={(theme as "dark" | "light") ? (theme as "dark" | "light") : "dark"} />
<MdeFormField control={form.control} name='description' label='Description' dataColorMode={(theme as "dark" | "light") ? (theme as "dark" | "light") : "dark"} {...mdxEditorProps} />
<CalenderFormField control={form.control} name='fromTime' label='From Date' />
<CalenderFormField control={form.control} name='toTime' label='To Date' />
<BooleanFormField control={form.control} name='hideDates' label='Hide Dates' />
</FormScaffold>
</div>
</FormMutationContextProvider>
)
}

View File

@@ -14,6 +14,8 @@ import { usePathname, useRouter } from 'next/navigation';
import { useTheme } from 'next-themes';
import { makeUseRelationShipWithNameIndex } from '~/lib/hooks';
import { FormMutationContextProvider } from '~/app/_components/Form/Components/MutationProvider';
import MdxComponentReference from '~/app/admin/_components/MdxComponentReference';
import { useMdxEditorFieldProps } from '~/app/admin/_components/useMdxEditorFieldProps';
export default function CreateUpdateProjectForm(params: { className?: string, entity?: IterableElement<RouterOutputs['project']['select']> }) {
const [id, setId] = useState<string | undefined>(params.entity ? params.entity.id : undefined)
const { theme } = useTheme()
@@ -35,6 +37,7 @@ export default function CreateUpdateProjectForm(params: { className?: string, en
})
let path = usePathname()
let router = useRouter()
const mdxEditorProps = useMdxEditorFieldProps()
const createMutation = trpc.project.insert.useMutation({ onSuccess: makeOnSuccess('create', form, setId) })
const updateMutation = trpc.project.update.useMutation({ onSuccess: makeOnSuccess('update', form) })
const deleteMutation = trpc.project.delete.useMutation({ onSuccess: makeOnSuccess('delete', form, undefined, path, router) })
@@ -50,6 +53,8 @@ export default function CreateUpdateProjectForm(params: { className?: string, en
updateMutation: updateMutation,
deleteMutation: deleteMutation
}}>
<div className='flex flex-col gap-4'>
<MdxComponentReference />
<FormScaffold
form={form}
onSubmit={onSubmit}
@@ -69,7 +74,7 @@ export default function CreateUpdateProjectForm(params: { className?: string, en
}
</SelectFormField>
<TextInputFormField control={form.control} name='title' label='Title' />
<MdeFormField control={form.control} name='description' label='Description' dataColorMode={(theme as "dark" | "light") ?? "dark"} />
<MdeFormField control={form.control} name='description' label='Description' dataColorMode={(theme as "dark" | "light") ?? "dark"} {...mdxEditorProps} />
<SelectFormField control={form.control} name='sourceType' label='Source Type' defaultValue={'open'} placeholder='open' >
<SelectItem value="open"> open </SelectItem>
<SelectItem value="closed"> closed </SelectItem>
@@ -82,6 +87,7 @@ export default function CreateUpdateProjectForm(params: { className?: string, en
<TextInputFormField control={form.control} label='Release Link' name='releaseLink' />
<IntInputFormField control={form.control} label='Order Position' name='orderPos'/>
</FormScaffold>
</div>
</FormMutationContextProvider>
)
}

View File

@@ -1,14 +1,12 @@
import { auth } from '@clerk/nextjs/server'
import { createOpenAI } from '@ai-sdk/openai'
import { streamText, tool, convertToModelMessages, stepCountIs, type UIMessage } from 'ai'
import { success, z } from 'zod'
import { streamText, convertToModelMessages, stepCountIs, type UIMessage } from 'ai'
import { eq, and } from 'drizzle-orm'
import { env } from '~/env'
import { db } from '~/server/db'
import { chatSession, chatMessage } from '~/server/dbschema/schema'
import { servTrpc } from '~/app/_trpc/ServerClient'
import { scheduleMeeting } from '~/app/actions/scheduleMeeting'
import currentTime from '~/app/actions/currentTime';
import { createChatTools } from '~/server/ai/tools'
const openai = createOpenAI({ apiKey: env.OPENAI_API_KEY })
@@ -31,7 +29,18 @@ export async function POST(req: Request) {
if (!session) return new Response('Session not found', { status: 404 })
const systemPrompt = await servTrpc.chat.getSystemPrompt() || 'You are an AI recruiter assistant.'
const configuredSystemPrompt = await servTrpc.chat.getSystemPrompt() || 'You are an AI recruiter assistant.'
const systemPrompt = `${configuredSystemPrompt}
Runtime context:
- Current server time: ${new Date().toISOString()}.
- Default meeting timezone: Europe/Berlin.
- For availability questions like "next open spot", call getAvailability once. It defaults to checking from now. Use nextAvailableSlot for the next opening, or the first item in availableSlots if needed. Do not call getAvailability again just to get more slots.
- After scheduleMeeting succeeds, include only the returned addToCalendarLink for the visitor. Format it as a Markdown link like [Add this meeting to your Google Calendar](URL); do not paste the raw URL. Explain briefly that this link lets them add the meeting to their own calendar and invite Gregor. Do not mention internal Google Calendar event links.
- You can remove meetings from Gregor's availability calendar with cancelMeeting only when you have the exact eventId from a previous scheduleMeeting result. If a visitor asks to reschedule and you have both the old eventId and a confirmed new slot, call cancelMeeting once for the old event and scheduleMeeting once for the new event. If you do not have the old eventId, ask for clarification instead of guessing.
- When rescheduling, make clear that cancelMeeting only removes the old slot from Gregor's availability calendar. If the visitor already added the old link to their own calendar, they may need to remove that copy themselves.
- Do not calculate or invent calendar availability yourself.`
const model = await servTrpc.chat.getModel()
// Save the latest user message
const lastMessage = messages[messages.length - 1]
@@ -46,45 +55,17 @@ export async function POST(req: Request) {
}
const result = streamText({
model: openai('gpt-5-mini'),
model: openai(model),
system: systemPrompt,
messages: await convertToModelMessages(messages),
tools: {
scheduleMeeting: tool({
description: 'Schedule a meeting with Gregor Lohaus and add it to his Google Calendar',
inputSchema: z.object({
title: z.string().describe('Meeting title, make something up if not provided'),
description: z.string().describe('Meeting description / agenda, make something up if not provided'),
dateTime: z
.string()
.describe(
'ISO 8601 datetime for the meeting start, e.g. 2025-04-01T10:00:00',
),
durationMinutes: z
.number()
.int()
.min(15)
.max(120)
.describe('Duration of the meeting in minutes, if none provided ask if 20 minutes is ok'),
attendeeEmail: z
.string()
.email()
.optional()
.describe('Optional Email of the visitor to invite (if provided)'),
attendeeName: z.string().optional().describe('Name of the visitor'),
}),
execute: async (input) => scheduleMeeting({ ...input }),
}),
getCurrentUnixTime: tool({
description: 'Get the current unix time to reference for meeting dates',
inputSchema: z.object({
none: z.string().optional().describe("no inputs are needed")
}),
execute: async () => currentTime()
})
},
stopWhen: stepCountIs(5),
tools: createChatTools(),
stopWhen: stepCountIs(3),
onFinish: async ({ text, finishReason }) => {
console.log('[ai:chat:onFinish]', {
finishReason,
hasText: Boolean(text),
textLength: text.length,
})
if (text && finishReason === 'stop') {
await db.insert(chatMessage).values({
sessionId,

View File

@@ -1,9 +1,10 @@
import { notFound } from "next/navigation";
import { MDXRemote } from "next-mdx-remote/rsc";
import { TRPCError } from "@trpc/server";
import matter from "gray-matter";
import { servTrpc } from "~/app/_trpc/ServerClient";
import { Badge } from "~/components/ui/badge";
import { mdxComponents } from "../_components/mdx-components";
import { mdxComponents } from "~/components/mdx-components";
type Props = {
params: Promise<{ slug: string }>;
@@ -12,37 +13,47 @@ type Props = {
export default async function BlogPostPage({ params }: Props) {
const { slug } = await params;
let post: Awaited<ReturnType<typeof servTrpc.blog.bySlug>>;
let post: Awaited<ReturnType<typeof servTrpc.blog.metadataBySlug>>;
try {
post = await servTrpc.blog.bySlug(slug);
post = await servTrpc.blog.metadataBySlug(slug);
} catch (e) {
if (e instanceof TRPCError && e.code === "NOT_FOUND") notFound();
throw e;
}
const response = await fetch(post.fileUrl, { next: { revalidate: 3600 } });
if (!response.ok) notFound();
const parsed = matter(await response.text());
const tags = Array.isArray(parsed.data.tags)
? parsed.data.tags.map((tag) => String(tag).trim()).filter(Boolean)
: post.tags;
const title = typeof parsed.data.title === "string" ? parsed.data.title : post.title;
const date = typeof parsed.data.date === "string" ? parsed.data.date : post.date;
return (
<main className="mx-auto max-w-2xl px-4 py-12">
<header className="mb-8">
<h1 className="text-3xl font-bold">{post.title}</h1>
{post.date && (
<h1 className="text-3xl font-bold">{title}</h1>
{date && (
<time className="text-muted-foreground text-sm">
{new Date(post.date).toLocaleDateString("en-US", {
{new Date(date).toLocaleDateString("en-US", {
year: "numeric",
month: "long",
day: "numeric",
})}
</time>
)}
{post.tags.length > 0 && (
{tags.length > 0 && (
<div className="mt-3 flex flex-wrap gap-1.5">
{post.tags.map((tag) => (
{tags.map((tag) => (
<Badge key={tag} variant="outline">{tag}</Badge>
))}
</div>
)}
</header>
<article className="prose dark:prose-invert max-w-none">
<MDXRemote source={post.content} components={mdxComponents} />
<MDXRemote source={parsed.content} components={mdxComponents} />
</article>
</main>
);

View File

@@ -1,6 +1,24 @@
import type { UIMessage } from "ai";
import Markdown from "react-markdown";
import { cn } from "~/lib/utils";
import { ClientMdx } from "~/components/ClientMdx";
function toolLabel(type: string) {
switch (type) {
case "tool-searchSiteContent":
return "Searching site content";
case "tool-getRelevantExperience":
return "Finding relevant experience";
case "tool-getProjectDetails":
return "Loading project details";
case "tool-getAvailability":
return "Checking availability";
case "tool-cancelMeeting":
return "Removing meeting";
case "tool-getCurrentUnixTime":
return "Checking current time";
default:
return "Using tool";
}
}
export const AssistantMessage = (props: { message: UIMessage }) => {
let message = props.message;
@@ -11,57 +29,90 @@ export const AssistantMessage = (props: { message: UIMessage }) => {
>
<div
className=
'max-w-[80%] px-4 py-2 text-sm space-y-2 bg-muted'
'max-w-[80%] min-w-0 px-4 py-2 text-sm space-y-2 bg-muted break-words [overflow-wrap:anywhere] [&_a]:break-all [&_pre]:max-w-full [&_pre]:overflow-x-auto'
>
{message.parts.map((part, i) => {
if (part.type === 'text') {
return (
<Markdown key={crypto.randomUUID()}>
{part.text}
</Markdown>
<ClientMdx key={i} source={part.text} fallback={part.text} />
)
}
if (part.type === 'tool-scheduleMeeting') {
const toolPart = part as unknown as {
type: 'tool-scheduleMeeting'
state: string
input: unknown
output?: { success: boolean; message?: string; htmlLink?: string; error?: string }
}
if (part.type === 'tool-scheduleMeeting') {
const toolPart = part as unknown as {
type: 'tool-scheduleMeeting'
state: string
input: unknown
output?: { success: boolean; error?: string }
}
if (toolPart.state === 'input-available' || toolPart.state === 'input-streaming') {
return (
<p key={i} className="text-xs opacity-70 italic">
Scheduling meeting
</p>
)
}
if (toolPart.state === 'output-available' && toolPart.output) {
const result = toolPart.output
return (
<div key={i} className="text-xs mt-1 p-2 bg-background/20 rounded">
{result.success ? (
<span>
{result.message}{' '}
{result.htmlLink && (
<a
href={result.htmlLink}
target="_blank"
rel="noopener noreferrer"
className="underline"
>
View event
</a>
)}
</span>
) : (
<span> {result.error}</span>
)}
}
if (toolPart.state === 'output-available' && toolPart.output) {
const result = toolPart.output
if (result.success) return null
return (
<div key={i} className="text-xs mt-1 p-2 bg-background/20 rounded">
<span> {result.error}</span>
</div>
)
}
}
return null
})}
)
}
}
if (part.type === 'tool-cancelMeeting') {
const toolPart = part as unknown as {
type: 'tool-cancelMeeting'
state: string
output?: { success: boolean; error?: string }
}
if (toolPart.state === 'input-available' || toolPart.state === 'input-streaming') {
return (
<p key={i} className="text-xs opacity-70 italic">
Removing meeting
</p>
)
}
if (toolPart.state === 'output-available' && toolPart.output) {
if (toolPart.output.success) return null
return (
<div key={i} className="text-xs mt-1 p-2 bg-background/20 rounded">
<span> {toolPart.output.error}</span>
</div>
)
}
}
if (part.type.startsWith('tool-')) {
const toolPart = part as unknown as {
type: string
state: string
output?: { success?: boolean; results?: unknown[]; matches?: unknown[]; availableSlots?: unknown[]; project?: unknown; error?: string }
}
if (toolPart.state === 'input-available' || toolPart.state === 'input-streaming') {
return (
<p key={i} className="text-xs opacity-70 italic">
{toolLabel(toolPart.type)}...
</p>
)
}
if (toolPart.state === 'output-available') {
const count = toolPart.output?.results?.length
?? toolPart.output?.matches?.length
?? toolPart.output?.availableSlots?.length
return (
<p key={i} className="text-xs opacity-70 italic">
{toolPart.output?.success === false
? (toolPart.output.error ?? `${toolLabel(toolPart.type)} failed`)
: count != null
? `${toolLabel(toolPart.type)} complete (${count})`
: `${toolLabel(toolPart.type)} complete`}
</p>
)
}
}
return null
})}
</div>
</div>
)

View File

@@ -1,5 +1,5 @@
'use client'
import { useState, useEffect } from 'react'
import { useState, useEffect, useRef } from 'react'
import { useChat } from '@ai-sdk/react'
import { DefaultChatTransport, type UIMessage } from 'ai'
import { Button } from '~/components/ui/button'
@@ -54,7 +54,7 @@ function addInitMessage(messageArray: UIMessage[]) {
role: 'assistant',
parts: [{
type: 'text',
text: "Hi im gregors ai assistant,you can ask me to provide general information or to schedule a meeting."
text: "Hi, I'm Gregor's AI assistant. Ask me about his experience, projects, blog posts, or availability for a meeting."
}],
})
}
@@ -83,14 +83,21 @@ function AuthenticatedChatInterface({ dbMessages, sessionId }: ChatInterfaceProp
sendMessage({ text })
}
const gsapContext = useGsapContext()
const didInitialScroll = useRef(false)
useEffect(() => {
let scroller = gsapContext?.getScroller()
if (scroller instanceof Window) {
return;
const scroller = gsapContext?.getScroller()
if (!scroller || scroller instanceof Window) {
return
}
console.log(scroller?.scrollHeight)
scroller?.scrollTo({ behavior: 'smooth', top: scroller.scrollHeight })
}, [messages])
// Jump instantly on first open so the chat starts pinned to the bottom;
// animate subsequent updates. Defer a frame so the messages have laid out
// (and any streaming content has grown) before we measure scrollHeight.
const behavior: ScrollBehavior = didInitialScroll.current ? 'smooth' : 'auto'
didInitialScroll.current = true
requestAnimationFrame(() => {
scroller.scrollTo({ behavior, top: scroller.scrollHeight })
})
}, [messages, status])
return (
<div className="flex flex-col h-full">
{messages &&
@@ -114,7 +121,7 @@ function AuthenticatedChatInterface({ dbMessages, sessionId }: ChatInterfaceProp
</div>
)}
<div className="p-4 border-t flex flex-row gap-2">
<div className="p-4 border-t flex flex-row gap-2 shrink-0">
<Textarea
name='message'
value={input}

View File

@@ -6,7 +6,7 @@ import { ScrollArea } from '~/components/ui/scroll-area';
import { memo } from 'react';
const Messages = memo(({messages,status}: { messages: UIMessage[],status:ChatStatus}) => {
return (
<ScrollArea data-scroller-priority='1' className="w-full h-[90%] max-w-4xl mx-auto">
<ScrollArea data-scroller-priority='1' className="w-full flex-1 min-h-0 max-w-4xl mx-auto">
{messages.map((message, i) => (
<Card.AnimatedCard scrollOnly={true} key={i}>
<Card.CardContent>

View File

@@ -14,7 +14,7 @@ export const UserMessage = (props:{message: UIMessage}) => {
>
<div
className=
'max-w-[80%] px-4 py-2 text-sm space-y-2 bg-primary'
'max-w-[80%] min-w-0 px-4 py-2 text-sm space-y-2 bg-primary break-words whitespace-pre-wrap [overflow-wrap:anywhere]'
>
{message}
</div>

View File

@@ -9,11 +9,11 @@ export default function ChatPage() {
const {messages,session,isLoading,error} = useMessages()
useTimeLine(messages)
return (
<div className="flex flex-col px-10 lg:px-0 w-full h-full max-w-4xl mx-auto pt-10">
<div className="flex flex-col px-10 lg:px-0 w-full h-full max-w-4xl mx-auto pt-10 pb-4">
<AnimatedPageTitle position={0}>
<span>Talk To My </span> <span> AI-Assistant</span>
</AnimatedPageTitle>
<div className='flex items-center h-[80%] w-full my-auto w-full'>
<div className='flex flex-1 min-h-0 w-full'>
{!isLoading &&
<ChatInterface sessionId={session?.id} dbMessages={messages ?? []}/>
}

View File

@@ -1,42 +1,43 @@
'use client'
import { trpc } from "~/app/_trpc/Client"
import type { ReactNode } from "react"
import CvEntry from "./CvEntry"
import type { servTrpc } from "~/app/_trpc/ServerClient"
import type { inferProcedureOutput } from "@trpc/server"
import type { RouterOutputs } from "~/server/routers/_app"
import { cn } from "~/lib/utils"
import { Card, CardContent, CardHeader, CardTitle } from "~/components/ui/card"
import { AnimatedCard, CardContent, CardHeader, CardTitle } from "~/components/ui/card"
import type { ArrayElement } from "type-fest"
import AnimatePopUp from "~/app/_components/Animated/AnimatePopUp";
import AnimateTextIn from "~/app/_components/Animated/AnimateIn"
import AnimatePopUp from "~/app/_components/Animated/AnimatePopUp"
import { ScrollArea } from "~/components/ui/scroll-area";
export type CvCategoryData = ArrayElement<RouterOutputs['categoryv2']['listAllWithEntries']>
type CvCategoryProps = {
initialData: ArrayElement<RouterOutputs['categoryv2']['listByLayoutPosition']>,
layout: "row"|"col",
children?: React.ReactElement<Parameters<typeof CvEntry>>
category: CvCategoryData,
layout: "row" | "col",
position?: number,
descriptions: Record<string, ReactNode>,
}
export default function CvCategory(props:CvCategoryProps) {
const category = trpc.categoryv2.getById.useQuery(props.initialData? props.initialData.id : "");
const entries = trpc.entryv2.byCategoryAndToDateDescending.useQuery(category.data?.id || "")
return (
<Card className={cn(props.layout == "row" ? "w-full" : "","gsapan", "h-screen")}>
<CardHeader>
<CardTitle>
{category.data?.name}
</CardTitle>
</CardHeader>
{(entries.data?.length ? entries.data?.length : 0 ) > 0 ?
<CardContent className={cn(props.layout == "row" ? "flex flex-row flex-wrap justify-center lg:justify-between" : "flex flex-col","gap-4","overflow-scroll")}>
<ScrollArea>
{entries.data?.map((entry,i) => (
<AnimatePopUp position={i}key={entry.id}>
<CvEntry className={props.layout == "row" ? "w-full lg:w-fit" : undefined} initialData={entry}/>
</AnimatePopUp>
))}
</ScrollArea>
</CardContent>
:
<></>
}
</Card>
)
export default function CvCategory({ category, layout, position = 0, descriptions }: CvCategoryProps) {
const entries = category.cvEntry
return (
<AnimatedCard position={position} className={cn(layout == "row" ? "w-full" : "", "h-screen")}>
<CardHeader>
<AnimateTextIn once position={position + 0.2} animation="slide" debugId={`cv-category-title:${category.name}:${position + 0.2}`}>
<CardTitle>{category.name}</CardTitle>
</AnimateTextIn>
</CardHeader>
{entries.length > 0 ?
<CardContent className={cn(layout == "row" ? "flex flex-row flex-wrap justify-center lg:justify-between" : "flex flex-col", "gap-4", "overflow-scroll")}>
<ScrollArea>
{entries.map((entry, i) => (
<AnimatePopUp position={position + 0.4 + i * 0.2} debugId={`cv-entry-wrapper:${category.name}:${entry.title}:${position + 0.4 + i * 0.2}`} key={entry.id}>
<CvEntry position={position + 0.4 + i * 0.2} entry={entry} description={descriptions[entry.id]} className={layout == "row" ? "w-full lg:w-fit" : undefined} />
</AnimatePopUp>
))}
</ScrollArea>
</CardContent>
:
<></>
}
</AnimatedCard>
)
}

View File

@@ -1,68 +1,52 @@
import { trpc } from "~/app/_trpc/Client"
import type { ReactNode } from "react"
import { Card, CardContent, CardFooter, CardHeader, CardTitle } from "~/components/ui/card"
import { Skeleton } from "~/components/ui/skeleton"
import { cn, type Defined } from "~/lib/utils"
import Markdown from 'react-markdown'
import { cn } from "~/lib/utils"
import { format } from 'date-fns'
import rehypeHighlight from 'rehype-highlight'
import rehypeRaw from 'rehype-raw'
import type { RouterOutputs } from "~/server/routers/_app"
import type { ArrayElement } from "type-fest"
export default function CvEntry(params: {
initialData: ArrayElement<Defined<RouterOutputs['categoryv2']['getById']>['cvEntry']>,
className?: string
import AnimateTextIn from "~/app/_components/Animated/AnimateIn"
import AnimatedDiv from "~/app/_components/Animated/AnimatedDiv"
import type { CvCategoryData } from "./CvCategory"
export type CvEntryData = ArrayElement<CvCategoryData['cvEntry']>
export default function CvEntry({ entry, description, className, position = 0 }: {
entry: CvEntryData,
description?: ReactNode,
className?: string,
position?: number
}) {
const query = trpc.entryv2.getById.useQuery(params.initialData.id);
const { data, isError, error } = query
return (
<>
{
data ?
<>
<Card className={params.className ? cn("w-fit", params.className) : "w-fit"}>
{
data.title ?
<CardHeader>
<CardTitle> {data.title} </CardTitle>
</CardHeader> :
<></>
}
{
data.description ?
<CardContent className="text-sm lg:text-base">
<article className="prose prose-zinc dark:prose-invert max-w-none">
<Markdown rehypePlugins={[rehypeHighlight, rehypeRaw]}>{data.description}</Markdown>
</article>
</CardContent> :
<></>
}
{
!data.hideDates ?
<CardFooter className="text-sm">
{`von ${format((new Date()).setTime(Date.parse(data.fromTime)), 'M. yyyy')} bis zum ${format((new Date()).setTime(Date.parse(data.toTime)), 'M. yyyy')}`}
</CardFooter> :
<></>
}
</Card>
</> :
<>
<Card>
<CardHeader>
<div className="flex flex-row">
<CardTitle> <Skeleton className="h-2rem w-5rem" /> </CardTitle>
<span className="ml-auto text-sm"> <Skeleton className="h-1rem w-3rem" /> - <Skeleton className="h-1rem w-3rem" /> </span>
</div>
</CardHeader>
<CardContent>
<div>
<Skeleton className="h-4 w-60" />
<Skeleton className="h-4 w-50" />
<Skeleton className="h-4 w-50" />
</div>
</CardContent>
</Card>
</>
<Card className={className ? cn("w-fit", className) : "w-fit"}>
{entry.title ?
<CardHeader>
<AnimateTextIn position={position} animation="slide" debugId={`cv-entry-title:${entry.title}:${position}`}>
<CardTitle> {entry.title} </CardTitle>
</AnimateTextIn>
</CardHeader> :
<></>
}
</>
{entry.description ?
<CardContent className="text-sm lg:text-base">
{/* Fade the description in place instead of collapsing its height:
the outer entry pop-up (CvCategory) measures height:auto when it
plays, so the description must stay laid out at full height or the
entry reveals too short. */}
<AnimatedDiv once position={position + 0.2} className="opacity-0" opacity={1} duration={0.5} debugId={`cv-entry-description:${entry.title}:${position + 0.2}`}>
<article className="prose prose-zinc dark:prose-invert max-w-none">
{description ?? entry.description}
</article>
</AnimatedDiv>
</CardContent> :
<></>
}
{!entry.hideDates ?
<CardFooter className="text-sm">
<AnimateTextIn position={position + 0.4} debugId={`cv-entry-dates:${entry.title}:${position + 0.4}`}>
{`von ${format(new Date(entry.fromTime), 'M. yyyy')} bis zum ${format(new Date(entry.toTime), 'M. yyyy')}`}
</AnimateTextIn>
</CardFooter> :
<></>
}
</Card>
)
}

View File

@@ -0,0 +1,59 @@
'use client'
import type { ReactNode } from "react";
import { Sidebar, SidebarContent, SidebarProvider } from "~/components/ui/sidebar";
import type { RouterOutputs } from "~/server/routers/_app"
import SidebarTriggerDisappearsOnMobile from "./SidebarTriggerDisappearsOnMobile";
import CvCategory from "./CvCategory";
import { useTimeLine } from "~/app/_providers/GsapProvicer";
export default function CvPage(props: {
cv: RouterOutputs['categoryv2']['listAllWithEntries'],
descriptions: Record<string, ReactNode>,
}) {
useTimeLine(props.cv)
const { descriptions } = props
const byPosition = (pos: "sidebar" | "header" | "col1" | "col2") =>
props.cv?.filter((c) => c.layoutPosition === pos) ?? []
const sidebarCategories = byPosition("sidebar")
const headerCategories = byPosition("header")
const col1Categories = byPosition("col1")
const col2Categories = byPosition("col2")
return (
<>
<SidebarProvider>
{sidebarCategories.length > 0 &&
<>
<SidebarTriggerDisappearsOnMobile />
<Sidebar>
<SidebarContent className="p-2 lg:pt-[3.2rem]">
{sidebarCategories.map((cat, i) => (
<CvCategory layout="col" position={i} category={cat} descriptions={descriptions} key={cat.id} />
))}
</SidebarContent>
</Sidebar>
</>
}
<div className="h-full w-full flex flex-wrap flex-row p-4 pt-8 ">
<div id="mainwrap" className="flex w-full flex-col gap-4 lg:px-[15vw]">
<div id="header" className="flex w-full h-fit flex-row gap-4 flex-wrap">
{headerCategories.map((cat, i) => (
<CvCategory layout="row" position={i} category={cat} descriptions={descriptions} key={cat.id} />
))}
</div>
<div id="colwrapper" className="flex flex-col lg:flex-row w-full h-3/4 gap-4">
<div id="col1" className={`flex flex-col w-full ${col1Categories.length > 0 ? "lg:w-1/2" : ""} h-full gap-4`}>
{col1Categories.map((cat, i) => (
<CvCategory layout="col" position={i} category={cat} descriptions={descriptions} key={cat.id} />
))}
</div>
<div id="col2" className={`flex flex-col w-full ${col2Categories.length > 0 ? "lg:w-1/2" : ""} h-full gap-4`}>
{col2Categories.map((cat, i) => (
<CvCategory layout="col" position={i} category={cat} descriptions={descriptions} key={cat.id} />
))}
</div>
</div>
</div>
</div>
</SidebarProvider>
</>
)
}

View File

@@ -1,97 +1,42 @@
'use client'
import { useGSAP } from "@gsap/react";
import { useGsapContext,useTimeLine } from "../_providers/GsapProvicer";
import { trpc } from "../_trpc/Client";
import { useRef } from "react";
import { SidebarContent, SidebarProvider, Sidebar } from "~/components/ui/sidebar";
import SidebarTriggerDisappearsOnMobile from "./_components/SidebarTriggerDisappearsOnMobile";
import CvCategory from "./_components/CvCategory";
import gsap from 'gsap'
export default function CvPage() {
const sidebarCategories = trpc.categoryv2.listByLayoutPosition.useQuery("sidebar");
const col1Categories = trpc.categoryv2.listByLayoutPosition.useQuery("col1");
const headerCategories = trpc.categoryv2.listByLayoutPosition.useQuery("header");
const col2Categories = trpc.categoryv2.listByLayoutPosition.useQuery("col2");
const gsapContext = useGsapContext()
const container = useRef<HTMLDivElement>(null)
enum Direction {
Left = 1,
Up,
Right,
Down
}
const nextGsapConf = (direction: Direction) => {
switch (direction) {
case Direction.Left:
return { x: -100, opacity: 0, duration: 0.5 }
case Direction.Up:
return { y: -100, opacity: 0, duration: 0.5 }
case Direction.Right:
return { x: 100, opacity: 0, duration: 0.5 }
case Direction.Down:
return { y: 100, opacity: 0, duration: 0.5 }
import { Suspense, type ReactNode } from "react";
import { MDXRemote } from "next-mdx-remote/rsc";
import rehypeHighlight from "rehype-highlight";
import remarkGfm from "remark-gfm";
import { servTrpc as trpc } from "../_trpc/ServerClient";
import { mdxComponents } from "~/components/mdx-components";
import Page from "./_components/Page";
export default async function CvPage() {
const cv = await trpc.categoryv2.listAllWithEntries();
// Render the MDX descriptions on the server so they exist at first paint.
// The client tree (which runs the GSAP entrance via useTimeLine) only places
// these already-rendered nodes — it never invokes the MDX renderer itself, so
// the 'use client' boundary stays intact and the animations no longer play
// against an un-rendered fallback.
const descriptions: Record<string, ReactNode> = {};
for (const category of cv ?? []) {
for (const entry of category.cvEntry) {
if (!entry.description?.trim()) continue;
descriptions[entry.id] = (
<MDXRemote
source={entry.description}
components={mdxComponents}
options={{
mdxOptions: {
format: "md",
remarkPlugins: [remarkGfm],
rehypePlugins: [rehypeHighlight],
},
}}
/>
);
}
}
useTimeLine(col2Categories)
useGSAP(() => {
const items = gsap?.utils.toArray<GSAPTweenTarget>('.gsapan');
let dir = Direction.Left;
items?.forEach(item => {
gsapContext?.addAnimation(gsap.from(item, nextGsapConf(dir)),0)
if (dir == Direction.Down) {
dir = Direction.Left
} else {
dir = dir + 1
}
})
}, { scope: container, dependencies: [headerCategories.data, sidebarCategories.data], revertOnUpdate: true })
return (
<>
<SidebarProvider ref={container}>
{sidebarCategories.data &&
<>
<SidebarTriggerDisappearsOnMobile />
<Sidebar className="gsapan ">
<SidebarContent className="p-2 lg:pt-[3.2rem]">
{sidebarCategories.data?.map((cat) => {
if (cat !== undefined) {
return (
<CvCategory layout="col" initialData={cat} key={cat.id} />
)
}
})}
</SidebarContent>
</Sidebar>
</>
}
<div className="h-full w-full flex flex-wrap flex-row p-4 pt-8 ">
<div id="mainwrap" className="flex w-full flex-col gap-4 lg:px-[15vw]">
<div id="header" className="flex w-full h-fit flex-row gap-4 flex-wrap">
{headerCategories.data?.map((cat) => {
return (
<CvCategory layout="row" initialData={cat} key={cat.id} />
)
})}
</div>
<div id="colwrapper" className="flex flex-col lg:flex-row w-full h-3/4 gap-4">
<div id="col1" className={`flex flex-col w-full ${col1Categories.data?.length ? col1Categories.data?.length : 0 > 0 ? "lg:w-1/2" : ""} h-full gap-4`}>
{col1Categories.data?.map((cat) => {
return (
<CvCategory layout="col" initialData={cat} key={cat.id} />
)
})}
</div>
<div id="col2" className={`flex flex-col w-full ${col2Categories.data?.length ? col2Categories.data?.length : 0 > 0 ? "lg:w-1/2" : ""} h-full gap-4`}>
{col2Categories.data?.map((cat) => {
return (
<CvCategory layout="col" initialData={cat} key={cat.id} />
)
})}
</div>
</div>
</div>
</div>
</SidebarProvider>
</>
<Suspense>
<Page cv={cv} descriptions={descriptions} />
</Suspense>
)
}

View File

@@ -12,6 +12,7 @@ import TrpcProvider from "./_trpc/TrpcProvider";
import ThemeProvider from './_providers/ThemeProvider'
import GsapProvider from "./_providers/GsapProvicer";
import {MessagesProvider} from "./_providers/MessagesProvider";
import { MusicPlayerProvider } from "./music/_components/MusicPlayerProvider";
import { CodeHighlightStyle } from "./_components/CodeHighlightSyle";
import { cn } from "~/lib/utils";
import AnimatedBackGroundContainer from "./_components/Animated/AnimatedBackGroundContainer";
@@ -49,14 +50,16 @@ export default async function RootLayout({
<body className="flex flex-col bg-background text-foreground">
<ThemeProvider>
<MessagesProvider>
<MusicPlayerProvider>
<AnimatedBackGroundContainer followSpeed={0.003} particleCount={100} orbitRadius={2000}>
<TopNav />
<main className="absolute lg:top-10 h-screen lg:h-[calc(100vh-var(--spacing)*10)] w-screen">
<main className="absolute lg:top-10 h-[100dvh] lg:h-[calc(100vh-var(--spacing)*10)] w-screen">
{children}
</main>
{modal}
</AnimatedBackGroundContainer>
<ChatFAB />
</MusicPlayerProvider>
</MessagesProvider>
</ThemeProvider>
</body>

View File

@@ -1,10 +1,12 @@
'use client'
import { useRef, useState } from "react";
import { useEffect, useRef, useState } from "react";
import { useTheme } from "next-themes";
import type WaveSurfer from "wavesurfer.js";
import { Download, Loader2, Pause, Play } from "lucide-react";
import { Slider } from "~/components/ui/slider";
import { Button } from "~/components/ui/button";
import { toast } from "sonner";
import { useMusicPlayer } from "./MusicPlayerProvider";
function formatTime(seconds: number) {
if (!Number.isFinite(seconds)) return "0:00";
@@ -13,29 +15,109 @@ function formatTime(seconds: number) {
return `${m}:${s.toString().padStart(2, "0")}`;
}
function cssVar(name: string, fallback: string) {
if (typeof window === "undefined") return fallback;
const v = getComputedStyle(document.documentElement).getPropertyValue(name).trim();
return v || fallback;
}
function waveColors() {
return {
waveColor: cssVar("--muted-foreground", "#9ca3af"),
progressColor: cssVar("--primary", "#e2761b"),
cursorColor: cssVar("--foreground", "#111827"),
};
}
/**
* Per-track waveform. Playback itself is owned by the shared MusicPlayer engine
* (so it keeps running across navigation); this wavesurfer instance is only a
* visual + seek surface that mirrors the engine when its track is active.
*/
export default function AudioPlayer(props: {
/** Streaming-friendly source the player actually plays. */
id: string;
src: string;
/** Original high-quality file offered via the download button. */
downloadUrl: string;
downloadName: string;
}) {
const audioRef = useRef<HTMLAudioElement>(null);
const [playing, setPlaying] = useState(false);
const [current, setCurrent] = useState(0);
const containerRef = useRef<HTMLDivElement>(null);
const wsRef = useRef<WaveSurfer | null>(null);
const [ready, setReady] = useState(false);
const [duration, setDuration] = useState(0);
const [seeking, setSeeking] = useState(false);
const [downloading, setDownloading] = useState(false);
const { resolvedTheme } = useTheme();
function togglePlay() {
const audio = audioRef.current;
if (!audio) return;
if (audio.paused) {
audio.play();
} else {
audio.pause();
const { currentId, isPlaying, currentTime, toggle, seek, subscribeTime } = useMusicPlayer();
const isActive = currentId === props.id;
// Reach live values from the once-created wavesurfer callbacks.
const isActiveRef = useRef(isActive);
isActiveRef.current = isActive;
const currentTimeRef = useRef(currentTime);
currentTimeRef.current = currentTime;
const toggleRef = useRef(toggle);
toggleRef.current = toggle;
const seekRef = useRef(seek);
seekRef.current = seek;
useEffect(() => {
let ws: WaveSurfer | null = null;
let cancelled = false;
setReady(false);
(async () => {
const WaveSurferClass = (await import("wavesurfer.js")).default;
if (cancelled || !containerRef.current) return;
const instance = WaveSurferClass.create({
container: containerRef.current,
url: props.src,
height: 44,
barWidth: 2,
barGap: 1,
barRadius: 2,
normalize: true,
cursorWidth: 1,
...waveColors(),
});
ws = instance;
wsRef.current = instance;
instance.on("ready", () => {
// This media is for drawing only — never let it make sound.
instance.setMuted(true);
setReady(true);
setDuration(instance.getDuration());
if (isActiveRef.current) instance.setTime(currentTimeRef.current);
});
// Clicking the waveform: start this track if it isn't playing, then seek.
instance.on("interaction", (time: number) => {
if (!isActiveRef.current) toggleRef.current(props.id);
seekRef.current(time);
});
})();
return () => {
cancelled = true;
ws?.destroy();
wsRef.current = null;
};
}, [props.src]);
// Mirror the engine's playback position onto the cursor while active.
useEffect(() => {
const ws = wsRef.current;
if (!ws || !ready) return;
if (!isActive) {
ws.setTime(0);
return;
}
}
ws.setTime(currentTimeRef.current);
return subscribeTime((t) => wsRef.current?.setTime(t));
}, [isActive, ready, subscribeTime]);
// Re-tint the waveform when the user toggles light/dark.
useEffect(() => {
wsRef.current?.setOptions(waveColors());
}, [resolvedTheme]);
async function handleDownload() {
setDownloading(true);
@@ -60,50 +142,33 @@ export default function AudioPlayer(props: {
}
}
const playing = isActive && isPlaying;
return (
<div className="flex items-center gap-3 rounded-lg border bg-transparent px-3 py-2">
<audio
ref={audioRef}
src={props.src}
preload="metadata"
onPlay={() => setPlaying(true)}
onPause={() => setPlaying(false)}
onLoadedMetadata={(e) => setDuration(e.currentTarget.duration)}
onTimeUpdate={(e) => {
if (!seeking) setCurrent(e.currentTarget.currentTime);
}}
onEnded={() => setPlaying(false)}
/>
<Button
type="button"
size="icon"
variant="ghost"
aria-label={playing ? "Pause" : "Play"}
onClick={togglePlay}
disabled={!ready}
onClick={() => toggle(props.id)}
>
{playing ? <Pause /> : <Play />}
</Button>
<span className="w-10 shrink-0 text-right font-mono text-xs text-muted-foreground tabular-nums">
{formatTime(current)}
{formatTime(isActive ? currentTime : 0)}
</span>
<Slider
className="flex-1"
min={0}
max={duration || 1}
step={0.1}
value={[current]}
onValueChange={([v]) => {
setSeeking(true);
setCurrent(v ?? 0);
}}
onValueCommit={([v]) => {
if (audioRef.current) audioRef.current.currentTime = v ?? 0;
setSeeking(false);
}}
/>
<div className="relative flex-1">
<div ref={containerRef} className="w-full" />
{!ready && (
<div className="absolute inset-0 flex items-center">
<div className="h-7 w-full animate-pulse rounded bg-muted-foreground/15" />
</div>
)}
</div>
<span className="w-10 shrink-0 font-mono text-xs text-muted-foreground tabular-nums">
{formatTime(duration)}

View File

@@ -0,0 +1,73 @@
'use client'
import { useState } from "react";
import { Music } from "lucide-react";
import {
Drawer,
DrawerContent,
DrawerHeader,
DrawerTitle,
DrawerTrigger,
} from "~/components/ui/drawer";
import { Slider } from "~/components/ui/slider";
import { useMusicPlayer } from "./MusicPlayerProvider";
import PlayerControls from "./PlayerControls";
function formatTime(seconds: number) {
if (!Number.isFinite(seconds)) return "0:00";
const m = Math.floor(seconds / 60);
const s = Math.floor(seconds % 60);
return `${m}:${s.toString().padStart(2, "0")}`;
}
/**
* Global, persistent mini-player: a floating button (shown once something is
* loaded) that opens a drawer with the transport controls and a scrubber, so
* playback can be controlled from any page while it keeps running in the
* background.
*/
export default function MusicMiniPlayer() {
const { currentTrack, isPlaying, currentTime, duration, seek } = useMusicPlayer();
const [open, setOpen] = useState(false);
if (!currentTrack) return null;
return (
<Drawer open={open} onOpenChange={setOpen}>
<DrawerTrigger asChild>
<button
type="button"
aria-label="Open player"
className="fixed bottom-4 left-4 z-40 flex max-w-[60vw] items-center gap-2 rounded-full border bg-background/80 px-4 py-2 shadow-lg backdrop-blur-md transition-colors hover:bg-background"
>
<Music className={isPlaying ? "animate-pulse" : ""} />
<span className="truncate text-sm">{currentTrack.title}</span>
</button>
</DrawerTrigger>
<DrawerContent>
<DrawerHeader>
<DrawerTitle>{currentTrack.title}</DrawerTitle>
</DrawerHeader>
<div className="mx-auto flex w-full max-w-md flex-col gap-4 px-4 pb-8">
<div className="flex items-center gap-2">
<span className="w-10 text-right font-mono text-xs text-muted-foreground tabular-nums">
{formatTime(currentTime)}
</span>
<Slider
className="flex-1"
min={0}
max={duration || 1}
step={0.1}
value={[currentTime]}
onValueChange={([v]) => seek(v ?? 0)}
/>
<span className="w-10 font-mono text-xs text-muted-foreground tabular-nums">
{formatTime(duration)}
</span>
</div>
<PlayerControls />
</div>
</DrawerContent>
</Drawer>
);
}

View File

@@ -0,0 +1,314 @@
'use client'
import {
createContext,
useCallback,
useContext,
useEffect,
useMemo,
useRef,
useState,
type ReactNode,
} from "react";
import { trpc } from "~/app/_trpc/Client";
import MusicMiniPlayer from "./MusicMiniPlayer";
export type PlayerTrack = {
id: string;
title: string;
/** Streaming-friendly source actually played. */
src: string;
/** Original high-quality file for the download button. */
downloadUrl: string;
downloadName: string;
};
type MusicPlayerValue = {
tracks: PlayerTrack[];
currentId: string | null;
currentTrack: PlayerTrack | null;
isPlaying: boolean;
shuffle: boolean;
currentTime: number;
duration: number;
/** Play button on a track: start it, or toggle play/pause if already current. */
toggle: (id: string) => void;
/** Global play/pause — acts on the current track (or starts the first one). */
togglePlayCurrent: () => void;
next: () => void;
previous: () => void;
toggleShuffle: () => void;
seek: (seconds: number) => void;
/** Subscribe to playback time updates (for waveform cursors). */
subscribeTime: (cb: (time: number) => void) => () => void;
};
const MusicPlayerContext = createContext<MusicPlayerValue | null>(null);
export function useMusicPlayer() {
const ctx = useContext(MusicPlayerContext);
if (!ctx) throw new Error("useMusicPlayer must be used within a MusicPlayerProvider");
return ctx;
}
export function MusicPlayerProvider({ children }: { children: ReactNode }) {
// The provider owns the playlist so playback survives navigating away from
// the music page. The query is cached, so the music page shares this request.
const { data } = trpc.music.list.useQuery();
const tracks = useMemo<PlayerTrack[]>(
() =>
(data ?? []).map((t) => ({
id: t.id,
title: t.title,
src: t.streamUrl ?? t.fileUrl,
downloadUrl: t.fileUrl,
downloadName: t.fileName,
})),
[data],
);
const [currentId, setCurrentId] = useState<string | null>(null);
const [isPlaying, setIsPlaying] = useState(false);
const [shuffle, setShuffle] = useState(false);
const [currentTime, setCurrentTime] = useState(0);
const [duration, setDuration] = useState(0);
const audioRef = useRef<HTMLAudioElement | null>(null);
const tracksRef = useRef(tracks);
tracksRef.current = tracks;
// Lightweight pub/sub so each waveform can follow playback time without every
// player re-rendering on the audio element's frequent timeupdate.
const timeSubs = useRef<Set<(t: number) => void>>(new Set());
const subscribeTime = useCallback((cb: (t: number) => void) => {
timeSubs.current.add(cb);
return () => {
timeSubs.current.delete(cb);
};
}, []);
const emitTime = useCallback((t: number) => {
timeSubs.current.forEach((cb) => cb(t));
}, []);
const currentTrack = useMemo(
() => tracks.find((t) => t.id === currentId) ?? null,
[tracks, currentId],
);
const toggle = useCallback((id: string) => {
setCurrentId((prev) => {
if (prev === id) {
setIsPlaying((p) => !p);
return prev;
}
setIsPlaying(true);
return id;
});
}, []);
const togglePlayCurrent = useCallback(() => {
setCurrentId((prev) => {
if (prev) {
setIsPlaying((p) => !p);
return prev;
}
const first = tracksRef.current[0]?.id ?? null;
if (first) setIsPlaying(true);
return first;
});
}, []);
const step = useCallback((dir: 1 | -1) => {
const ids = tracksRef.current.map((t) => t.id);
if (ids.length === 0) return;
setCurrentId((prev) => {
let nextId: string;
if (shuffle && dir === 1 && ids.length > 1) {
do {
nextId = ids[Math.floor(Math.random() * ids.length)]!;
} while (nextId === prev);
} else {
const idx = prev ? ids.indexOf(prev) : -1;
nextId = ids[(idx + dir + ids.length) % ids.length]!;
}
setIsPlaying(true);
return nextId;
});
}, [shuffle]);
const next = useCallback(() => step(1), [step]);
const previous = useCallback(() => step(-1), [step]);
const toggleShuffle = useCallback(() => setShuffle((s) => !s), []);
const stepRef = useRef(step);
stepRef.current = step;
const seek = useCallback(
(s: number) => {
if (audioRef.current) audioRef.current.currentTime = s;
setCurrentTime(s);
emitTime(s);
},
[emitTime],
);
// Persistent audio element + listeners, created once on the client.
useEffect(() => {
const audio = new Audio();
audio.preload = "metadata";
audioRef.current = audio;
const onTime = () => {
setCurrentTime(audio.currentTime);
emitTime(audio.currentTime);
};
const onMeta = () => setDuration(audio.duration || 0);
const onEnded = () => stepRef.current(1);
audio.addEventListener("timeupdate", onTime);
audio.addEventListener("loadedmetadata", onMeta);
audio.addEventListener("ended", onEnded);
return () => {
audio.pause();
audio.removeEventListener("timeupdate", onTime);
audio.removeEventListener("loadedmetadata", onMeta);
audio.removeEventListener("ended", onEnded);
audioRef.current = null;
};
}, [emitTime]);
// Swap the source when the current track changes.
useEffect(() => {
const audio = audioRef.current;
if (!audio) return;
if (!currentTrack) {
audio.removeAttribute("src");
audio.load();
return;
}
if (audio.src !== currentTrack.src) {
audio.src = currentTrack.src;
audio.load();
setCurrentTime(0);
setDuration(0);
}
}, [currentTrack]);
// Reflect the desired play state onto the element.
useEffect(() => {
const audio = audioRef.current;
if (!audio || !currentTrack) return;
if (isPlaying) audio.play().catch(() => {});
else audio.pause();
}, [isPlaying, currentTrack]);
// OS-level media controls (lock screen, notification shade, media keys, etc.)
// via the Media Session API. Wire the transport actions to our state.
useEffect(() => {
if (typeof navigator === "undefined" || !("mediaSession" in navigator)) return;
const ms = navigator.mediaSession;
const handlers: [MediaSessionAction, MediaSessionActionHandler][] = [
["play", () => setIsPlaying(true)],
["pause", () => setIsPlaying(false)],
["previoustrack", () => stepRef.current(-1)],
["nexttrack", () => stepRef.current(1)],
[
"seekto",
(details) => {
if (typeof details.seekTime === "number") seek(details.seekTime);
},
],
[
"seekbackward",
(details) => seek((audioRef.current?.currentTime ?? 0) - (details.seekOffset ?? 10)),
],
[
"seekforward",
(details) => seek((audioRef.current?.currentTime ?? 0) + (details.seekOffset ?? 10)),
],
];
for (const [action, handler] of handlers) {
try {
ms.setActionHandler(action, handler);
} catch {
// Action unsupported by this browser — ignore.
}
}
return () => {
for (const [action] of handlers) {
try {
ms.setActionHandler(action, null);
} catch {
// ignore
}
}
};
}, [seek]);
// Keep the OS-visible metadata in sync with the current track.
useEffect(() => {
if (typeof navigator === "undefined" || !("mediaSession" in navigator)) return;
navigator.mediaSession.metadata = currentTrack
? new MediaMetadata({
title: currentTrack.title,
artist: "Gregor Lohaus",
})
: null;
}, [currentTrack]);
// Reflect play/pause state to the OS so the right button is shown.
useEffect(() => {
if (typeof navigator === "undefined" || !("mediaSession" in navigator)) return;
navigator.mediaSession.playbackState = currentTrack
? isPlaying
? "playing"
: "paused"
: "none";
}, [isPlaying, currentTrack]);
// Keep the scrubber position on the OS controls in sync.
useEffect(() => {
if (typeof navigator === "undefined" || !("mediaSession" in navigator)) return;
if (!("setPositionState" in navigator.mediaSession)) return;
try {
if (duration > 0 && Number.isFinite(duration)) {
navigator.mediaSession.setPositionState({
duration,
position: Math.min(currentTime, duration),
playbackRate: audioRef.current?.playbackRate ?? 1,
});
} else {
navigator.mediaSession.setPositionState();
}
} catch {
// Some browsers throw on invalid state — ignore.
}
}, [currentTime, duration]);
const value = useMemo<MusicPlayerValue>(
() => ({
tracks,
currentId,
currentTrack,
isPlaying,
shuffle,
currentTime,
duration,
toggle,
togglePlayCurrent,
next,
previous,
toggleShuffle,
seek,
subscribeTime,
}),
[
tracks, currentId, currentTrack, isPlaying, shuffle, currentTime, duration,
toggle, togglePlayCurrent, next, previous, toggleShuffle, seek, subscribeTime,
],
);
return (
<MusicPlayerContext.Provider value={value}>
{children}
<MusicMiniPlayer />
</MusicPlayerContext.Provider>
);
}

View File

@@ -0,0 +1,68 @@
'use client'
import * as Card from "~/components/ui/card";
import { useTimeLine } from "../../_providers/GsapProvicer";
import AnimatedPageTitle from "../../_components/Animated/AnimatedPageTitle";
import AnimateTextIn from "../../_components/Animated/AnimateIn";
import { ScrollArea } from "~/components/ui/scroll-area";
import AnimatePopUp from "../../_components/Animated/AnimatePopUp";
import AudioPlayer from "./AudioPlayer";
import type { RouterOutputs } from "~/server/routers/_app";
export default function MusicPage(props: {
tracks: RouterOutputs['music']['list'],
}) {
const { tracks } = props;
useTimeLine(tracks)
return (
<ScrollArea className="px-10 lg:px-0 w-full h-full max-w-4xl mx-auto pt-10">
<AnimatedPageTitle position={0}><span>Just Some </span> <span>Music I Made</span> </AnimatedPageTitle>
<div className="flex flex-wrap h-fit content-center">
<AnimateTextIn once className="flex flex-wrap mr-[1em]" position={0.5}>
<div><p className="break-after-avoid mr-[1em]">All works on this page are licensed under:</p></div>
<div><a href="https://creativecommons.org/licenses/by-nc-sa/4.0/">CC BY-NC-SA 4.0</a></div>
</AnimateTextIn>
<AnimatePopUp duration={1} ease='elastic.inOut' position={2} once className="items-center content-center">
<div className="flex flex-row">
<img className="max-w-[1em]" src="https://mirrors.creativecommons.org/presskit/icons/cc.svg" alt="" />
<img className="max-w-[1em] ml-[1em]" src="https://mirrors.creativecommons.org/presskit/icons/by.svg" alt="" />
<img className="max-w-[1em] ml-[1em]" src="https://mirrors.creativecommons.org/presskit/icons/nc.svg" alt="" />
<img className="max-w-[1em] ml-[1em]" src="https://mirrors.creativecommons.org/presskit/icons/sa.svg" alt="" />
</div>
</AnimatePopUp>
</div>
<div className="pt-10" />
{tracks && tracks.map((track, i) => (
<div key={track.id}>
<Card.AnimatedCard position={i + 1}>
<Card.CardHeader>
<AnimateTextIn once position={i + 1.2} animation="slide">
<Card.CardTitle>{track.title}</Card.CardTitle>
</AnimateTextIn>
</Card.CardHeader>
<Card.CardContent className="flex flex-col gap-3">
{track.description && (
<AnimatePopUp once position={i + 1.25} duration={2}>
<p className="text-sm text-muted-foreground">{track.description}</p>
</AnimatePopUp>
)}
<AnimatePopUp duration={2} ease='elastic.inOut' position={i + 1.3} once>
<AudioPlayer
id={track.id}
src={track.streamUrl ?? track.fileUrl}
downloadUrl={track.fileUrl}
downloadName={track.fileName}
/>
</AnimatePopUp>
</Card.CardContent>
</Card.AnimatedCard>
<div className="pt-5" />
</div>
))}
{!tracks?.length &&
<div className="flex justify-center items-center text-muted-foreground">
No music yet.
</div>}
</ScrollArea>
);
}

View File

@@ -0,0 +1,40 @@
'use client'
import { Pause, Play, Shuffle, SkipBack, SkipForward } from "lucide-react";
import { Button } from "~/components/ui/button";
import { useMusicPlayer } from "./MusicPlayerProvider";
export default function PlayerControls() {
const { isPlaying, shuffle, togglePlayCurrent, next, previous, toggleShuffle } =
useMusicPlayer();
return (
<div className="flex items-center justify-center gap-1">
<Button type="button" size="icon" variant="ghost" aria-label="Previous track" onClick={previous}>
<SkipBack />
</Button>
<Button
type="button"
size="icon-lg"
variant="secondary"
aria-label={isPlaying ? "Pause" : "Play"}
onClick={togglePlayCurrent}
>
{isPlaying ? <Pause /> : <Play />}
</Button>
<Button type="button" size="icon" variant="ghost" aria-label="Skip to next track" onClick={next}>
<SkipForward />
</Button>
<Button
type="button"
size="icon"
variant={shuffle ? "secondary" : "ghost"}
aria-label={shuffle ? "Shuffle on" : "Shuffle off"}
aria-pressed={shuffle}
onClick={toggleShuffle}
>
<Shuffle />
</Button>
</div>
);
}

View File

@@ -1,66 +1,13 @@
'use client'
import { trpc } from "~/app/_trpc/Client";
import * as Card from "~/components/ui/card";
import { useTimeLine } from "../_providers/GsapProvicer";
import AnimatedPageTitle from "../_components/Animated/AnimatedPageTitle";
import { Spinner } from "~/components/ui/spinner";
import AnimateTextIn from "../_components/Animated/AnimateIn";
import { ScrollArea } from "~/components/ui/scroll-area";
import AnimatePopUp from "../_components/Animated/AnimatePopUp";
import AudioPlayer from "./_components/AudioPlayer";
export default function MusicPage() {
const { data: tracks, isLoading } = trpc.music.list.useQuery();
useTimeLine(tracks)
return (
<ScrollArea className="px-10 lg:px-0 w-full h-full max-w-4xl mx-auto pt-10">
<AnimatedPageTitle position={0}><span>Just Some </span> <span>Music I Made</span> </AnimatedPageTitle>
<div className="flex flex-wrap h-fit content-center">
<AnimateTextIn className="flex flex-wrap mr-[1em]" position={0.5}>
<div><p className="break-after-avoid mr-[1em]">All works on this page are licensed under:</p></div>
<div><a href="https://creativecommons.org/licenses/by-nc-sa/4.0/">CC BY-NC-SA 4.0</a></div>
</AnimateTextIn>
<AnimatePopUp position={2} className="items-center content-center">
<div className="flex flex-row">
<img className="max-w-[1em]" src="https://mirrors.creativecommons.org/presskit/icons/cc.svg" alt="" />
<img className="max-w-[1em] ml-[1em]" src="https://mirrors.creativecommons.org/presskit/icons/by.svg" alt="" />
<img className="max-w-[1em] ml-[1em]" src="https://mirrors.creativecommons.org/presskit/icons/nc.svg" alt="" />
<img className="max-w-[1em] ml-[1em]" src="https://mirrors.creativecommons.org/presskit/icons/sa.svg" alt="" />
</div>
</AnimatePopUp>
</div>
<div className="pt-10" />
{tracks && tracks.map((track, i) => (
<div key={track.id}>
<Card.AnimatedCard position={i + 1}>
<Card.CardHeader>
<AnimateTextIn position={i + 1.2} animation="slide">
<Card.CardTitle>{track.title}</Card.CardTitle>
</AnimateTextIn>
</Card.CardHeader>
<Card.CardContent className="flex flex-col gap-3">
{track.description && (
<p className="text-sm text-muted-foreground gsapant">{track.description}</p>
)}
<AnimatePopUp position={i + 1.3}>
<AudioPlayer
src={track.streamUrl ?? track.fileUrl}
downloadUrl={track.fileUrl}
downloadName={track.fileName}
/>
</AnimatePopUp>
</Card.CardContent>
</Card.AnimatedCard>
<div className="pt-5" />
</div>
))}
{!isLoading && !tracks?.length &&
<div className="flex justify-center items-center text-muted-foreground">
No music yet.
</div>
}
{isLoading && <div className="w-full h-full items-center flex flex-row content-center gap-4 justify-center">
<Spinner /> Loading Tracks
</div>}
</ScrollArea>
);
import { Suspense } from "react";
import { servTrpc as trpc } from "../_trpc/ServerClient";
import Page from "./_components/Page";
export default async function MusicPage() {
const tracks = await trpc.music.list();
return (
<Suspense>
<Page tracks={tracks} />
</Suspense>
);
}

View File

@@ -0,0 +1,111 @@
'use client'
import type { ReactNode } from "react";
import * as Card from "~/components/ui/card";
import { Badge } from "~/components/ui/badge";
import { StackBadge } from "~/components/StackBadge";
import { ScrollArea } from "~/components/ui/scroll-area";
import AnimatedPageTitle from "../../_components/Animated/AnimatedPageTitle";
import AnimateTextIn from "../../_components/Animated/AnimateIn";
import { useTimeLine } from "../../_providers/GsapProvicer";
import AnimatePopUp from "../../_components/Animated/AnimatePopUp";
import { Button } from "~/components/ui/button";
import type { RouterOutputs } from "~/server/routers/_app";
export default function ProjectsPage(props: {
projects: RouterOutputs['projectv2']['listWithStack'],
descriptions: Record<string, ReactNode>,
}) {
const { projects, descriptions } = props;
useTimeLine(projects)
if (!projects?.length) {
return (
<div className="flex justify-center items-center min-h-[200px] text-muted-foreground">
No projects yet.
</div>
);
}
return (
<ScrollArea className="px-10 lg:px-0 w-full h-full max-w-4xl mx-auto pt-10">
<AnimatedPageTitle position={0}><span>Projects I've Been</span><span> Working on</span> </AnimatedPageTitle>
<div className="pt-10" />
{projects.map((project, i) => (
<div id={project.id} key={i} className="scroll-mt-10">
<Card.AnimatedCard position={i + 1.2} key={project.id}>
<Card.CardHeader>
<div className="flex items-start justify-between gap-2 flex-wrap">
<AnimateTextIn once position={i + 1.4} animation="slide"><Card.CardTitle>{project.title}</Card.CardTitle></AnimateTextIn>
<div className="flex gap-2 flex-wrap">
{project.sourceType && (
<AnimatePopUp position={i + 2} duration={2} once>
<Badge variant={project.sourceType === "open" ? "secondary" : "outline"}>
{project.sourceType === "open" ? "Open Source" : "Closed Source"}
</Badge>
</AnimatePopUp>
)}
{project.releaseStatus && (
<Badge variant={project.releaseStatus === "released" ? "default" : "outline"}>
{project.releaseStatus === "released" ? "Released" : "Unreleased"}
</Badge>
)}
</div>
</div>
</Card.CardHeader>
{(project.description || project.sourceLink || project.releaseLink || project.techStack?.stackItems?.length) && (
<Card.CardContent className="flex flex-col gap-3">
{project.description && (
<div className="prose prose-sm dark:prose-invert max-w-none text-muted-foreground">
<AnimatePopUp once position={i + 1.4} duration={10}>
{descriptions[project.id] ?? project.description}
</AnimatePopUp>
</div>
)}
<div className="flex flex-row">
{project.techStack?.stackItems && project.techStack.stackItems.length > 0 && (
<div className="flex flex-wrap gap-1.5">
{project.techStack.stackItems.map((item, k) => (
<AnimatePopUp key={k} position={(i + 2) + k * 0.5} once> <StackBadge key={item} item={item} /> </AnimatePopUp>
))}
</div>
)}
{(project.sourceLink || project.releaseLink) && (
<div className="ml-auto flex-col lg:flex-row justify-center gap-5">
{project.sourceLink &&
<Button variant='outline' className="cursor-pointer mb-3 lg:mb-0 lg:mr-3 min-w-18">
<a
href={project.sourceLink}
target="_blank"
rel="noopener noreferrer"
className='items-center'
>
Source
</a>
</Button>
}
{project.releaseLink &&
<Button variant='default' className="cursor-pointer min-w-18 items-center">
<a
href={project.releaseLink}
target="_blank"
rel="noopener noreferrer"
className='items-center'
>
Live
</a>
</Button>
}
</div>
)}
</div>
</Card.CardContent>
)}
</Card.AnimatedCard>
<div className="pt-5" />
</div>
))}
</ScrollArea>
);
}

View File

@@ -1,116 +1,40 @@
'use client'
import { Suspense, type ReactNode } from "react";
import { MDXRemote } from "next-mdx-remote/rsc";
import rehypeHighlight from "rehype-highlight";
import remarkGfm from "remark-gfm";
import { servTrpc as trpc } from "../_trpc/ServerClient";
import { mdxComponents } from "~/components/mdx-components";
import Page from "./_components/Page";
import { trpc } from "~/app/_trpc/Client";
import * as Card from "~/components/ui/card";
import { Badge } from "~/components/ui/badge";
import { StackBadge } from "~/components/StackBadge";
import Markdown from "react-markdown";
import { ScrollArea } from "~/components/ui/scroll-area";
import AnimatedPageTitle from "../_components/Animated/AnimatedPageTitle";
import AnimateTextIn from "../_components/Animated/AnimateIn";
import { useTimeLine } from "../_providers/GsapProvicer";
import AnimatePopUp from "../_components/Animated/AnimatePopUp";
import { Button } from "~/components/ui/button";
import remarkGfm from "remark-gfm"
export default async function ProjectsPage() {
const projects = await trpc.projectv2.listWithStack();
export default function ProjectsPage() {
const { data: projects, isLoading } = trpc.projectv2.listWithStack.useQuery();
useTimeLine(projects)
if (isLoading) {
return (
<div className="flex justify-center items-center min-h-[200px] text-muted-foreground">
Loading...
</div>
);
}
if (!projects?.length) {
return (
<div className="flex justify-center items-center min-h-[200px] text-muted-foreground">
No projects yet.
</div>
// Render the MDX descriptions on the server so they exist at first paint.
// The client tree (which runs the GSAP entrance via useTimeLine) only places
// these already-rendered nodes — it never invokes the MDX renderer itself, so
// the 'use client' boundary stays intact and the animations no longer play
// against an un-rendered fallback.
const descriptions: Record<string, ReactNode> = {};
for (const project of projects ?? []) {
if (!project.description?.trim()) continue;
descriptions[project.id] = (
<MDXRemote
source={project.description}
components={mdxComponents}
options={{
mdxOptions: {
format: "md",
remarkPlugins: [remarkGfm],
rehypePlugins: [rehypeHighlight],
},
}}
/>
);
}
return (
<ScrollArea className="px-10 lg:px-0 w-full h-full max-w-4xl mx-auto pt-10">
<AnimatedPageTitle position={0}><span>Projects I've Been</span><span> Working on</span> </AnimatedPageTitle>
<div className="pt-10" />
{projects.map((project, i) => (
<div id={project.id} key={i} className="scroll-mt-10">
<Card.AnimatedCard position={i + 1.2} key={project.id}>
<Card.CardHeader>
<div className="flex items-start justify-between gap-2 flex-wrap">
<AnimateTextIn position={i + 1.4} animation="slide"><Card.CardTitle>{project.title}</Card.CardTitle></AnimateTextIn>
<div className="flex gap-2 flex-wrap">
{project.sourceType && (
<AnimatePopUp position={i + 2} duration={2}>
<Badge variant={project.sourceType === "open" ? "secondary" : "outline"}>
{project.sourceType === "open" ? "Open Source" : "Closed Source"}
</Badge>
</AnimatePopUp>
)}
{project.releaseStatus && (
<Badge variant={project.releaseStatus === "released" ? "default" : "outline"}>
{project.releaseStatus === "released" ? "Released" : "Unreleased"}
</Badge>
)}
</div>
</div>
</Card.CardHeader>
{(project.description || project.sourceLink || project.releaseLink || project.techStack?.stackItems?.length) && (
<Card.CardContent className="flex flex-col gap-3">
{project.description && (
<div className="prose prose-sm dark:prose-invert max-w-none text-muted-foreground">
<AnimatePopUp position={i + 1.4} duration={10}>
<Markdown remarkPlugins={[remarkGfm]}>{project.description}</Markdown>
</AnimatePopUp>
</div>
)}
<div className="flex flex-row">
{project.techStack?.stackItems && project.techStack.stackItems.length > 0 && (
<div className="flex flex-wrap gap-1.5">
{project.techStack.stackItems.map((item, k) => (
<AnimatePopUp key={k} position={(i + 2) + k * 0.5}> <StackBadge key={item} item={item} /> </AnimatePopUp>
))}
</div>
)}
{(project.sourceLink || project.releaseLink) && (
<div className="ml-auto flex-col lg:flex-row justify-center gap-5">
{project.sourceLink &&
<Button variant='outline' className="cursor-pointer mb-3 lg:mb-0 lg:mr-3 min-w-18">
<a
href={project.sourceLink}
target="_blank"
rel="noopener noreferrer"
className='items-center'
>
Source
</a>
</Button>
}
{project.releaseLink &&
<Button variant='default' className="cursor-pointer min-w-18 items-center">
<a
href={project.releaseLink}
target="_blank"
rel="noopener noreferrer"
className='items-center'
>
Live
</a>
</Button>
}
</div>
)}
</div>
</Card.CardContent>
)}
</Card.AnimatedCard>
<div className="pt-5" />
</div>
))}
</ScrollArea>
<Suspense>
<Page projects={projects} descriptions={descriptions} />
</Suspense>
);
}

View File

@@ -0,0 +1,79 @@
"use client";
import { evaluate } from "@mdx-js/mdx";
import { MDXProvider, useMDXComponents } from "@mdx-js/react";
import type { MDXComponents } from "mdx/types";
import { useEffect, useState, type ComponentType, type ReactNode } from "react";
import * as runtime from "react/jsx-runtime";
import rehypeHighlight from "rehype-highlight";
import remarkGfm from "remark-gfm";
import { mdxComponents } from "~/components/mdx-components";
type MdxModule = {
default: ComponentType<{ components?: MDXComponents }>;
};
type ClientMdxProps = {
source: string;
components?: MDXComponents;
format?: "md" | "mdx";
fallback?: ReactNode;
errorFallback?: (error: Error) => ReactNode;
};
export function ClientMdx({
source,
components = mdxComponents,
format = "md",
fallback = null,
errorFallback,
}: ClientMdxProps) {
const [Content, setContent] = useState<MdxModule["default"] | null>(null);
const [error, setError] = useState<Error | null>(null);
useEffect(() => {
let cancelled = false;
const trimmed = source.trim();
if (!trimmed) {
setContent(null);
setError(null);
return;
}
void evaluate(trimmed, {
...runtime,
baseUrl: import.meta.url,
format,
useMDXComponents,
rehypePlugins: [rehypeHighlight],
remarkPlugins: [remarkGfm],
})
.then((mod) => {
if (cancelled) return;
setContent(() => (mod as MdxModule).default);
setError(null);
})
.catch((nextError: unknown) => {
if (cancelled) return;
setContent(null);
setError(nextError instanceof Error ? nextError : new Error("Failed to render MDX"));
});
return () => {
cancelled = true;
};
}, [format, source]);
if (error) {
return errorFallback ? errorFallback(error) : <p>{source}</p>;
}
if (!Content) return <>{fallback}</>;
return (
<MDXProvider components={components}>
<Content components={components} />
</MDXProvider>
);
}

View File

@@ -1,5 +1,10 @@
import Link from "next/link";
import { Children, isValidElement, type ComponentPropsWithoutRef, type ReactNode } from "react";
import {
Children,
isValidElement,
type ComponentPropsWithoutRef,
type ReactNode,
} from "react";
import { Badge } from "~/components/ui/badge";
import { Button } from "~/components/ui/button";
import { cn } from "~/lib/utils";
@@ -95,12 +100,19 @@ function PullQuote({ children }: { children: ReactNode }) {
}
function ExternalLink(props: ComponentPropsWithoutRef<"a">) {
const href = props.href ?? "";
const isExternal = /^https?:\/\//.test(href);
const { className, ...rest } = props;
if (!isExternal) return <a {...props} />;
return <a {...props} target="_blank" rel="noreferrer" />;
return (
<a
{...rest}
target="_blank"
rel="noreferrer"
className={cn(
"text-sky-600 underline underline-offset-4 hover:text-sky-700 dark:text-sky-400 dark:hover:text-sky-300",
className,
)}
/>
);
}
const blockComponents = new Set<unknown>([Callout, Figure, PullQuote, TagList]);

View File

@@ -1,7 +1,8 @@
import { useGSAP } from "@gsap/react"; import * as React from "react"
"use client"
import * as React from "react"
import { useRef } from "react";
import { useGsapContext } from "~/app/_providers/GsapProvicer";
import gsap from 'gsap'
import { useReveal } from "~/app/_components/Animated/useReveal";
import { cn } from "~/lib/utils"
function Card({
@@ -27,47 +28,26 @@ function AnimatedCard({
position = 0,
size = "default",
scrollOnly = false,
once = false,
...props
}: React.ComponentProps<"div"> & { size?: "default" | "sm", position?: gsap.Position, scrollOnly?: boolean }) {
const gsapContext = useGsapContext()
}: React.ComponentProps<"div"> & { size?: "default" | "sm", position?: gsap.Position, scrollOnly?: boolean, once?: boolean }) {
const ref = useRef<HTMLDivElement | null>(null)
useGSAP(() => {
const rect = ref.current?.getBoundingClientRect()
const scroller = gsapContext?.getScroller()
console.log(scroller)
let viewportTop = 0
let viewportBottom = window.innerHeight
if (scroller && scroller instanceof Element) {
const scrollerRect = scroller.getBoundingClientRect()
viewportTop = scrollerRect.top
viewportBottom = scrollerRect.top + scrollerRect.height
}
const isInView = rect && rect.bottom > viewportTop && rect.top < viewportBottom
console.log(isInView)
const fromVars = { x: -100, opacity: 0, duration: 0.5 }
if (isInView && !scrollOnly) {
gsapContext?.addAnimation(gsap.from(ref.current, fromVars), position)
} else {
gsap.from(ref.current,
{
...fromVars,
scrollTrigger: {
trigger: ref.current,
start: 'top bottom',
end: 'bottom top',
toggleActions: "play reverse play reverse",
scroller
}
})
}
}, { dependencies: [] })
useReveal(ref, {
position,
scrollOnly,
once,
debugId: `card-${position}`,
// Start hidden via CSS (see className) so the server-rendered card never
// flashes visible before GSAP runs; reveal animates *to* the shown state.
makeReveal: (el) => gsap.to(el, { x: 0, opacity: 1, duration: 0.5, paused: true }),
})
return (
<div
ref={ref}
data-slot="card"
data-size={size}
className={cn(
"group/card flex flex-col gap-4 overflow-hidden rounded-xl bg-card py-4 text-sm text-card-foreground ring-1 ring-foreground/10 has-data-[slot=card-footer]:pb-0 has-[>img:first-child]:pt-0 data-[size=sm]:gap-3 data-[size=sm]:py-3 data-[size=sm]:has-data-[slot=card-footer]:pb-0 *:[img:first-child]:rounded-t-xl *:[img:last-child]:rounded-b-xl bg-opacity-60 backdrop-blur-sm",
"opacity-0 -translate-x-[100px] group/card flex flex-col gap-4 overflow-hidden rounded-xl bg-card py-4 text-sm text-card-foreground ring-1 ring-foreground/10 has-data-[slot=card-footer]:pb-0 has-[>img:first-child]:pt-0 data-[size=sm]:gap-3 data-[size=sm]:py-3 data-[size=sm]:has-data-[slot=card-footer]:pb-0 *:[img:first-child]:rounded-t-xl *:[img:last-child]:rounded-b-xl bg-opacity-60 backdrop-blur-sm",
className
)}
{...props}

View File

@@ -31,6 +31,10 @@ export const env = createEnv({
CLERK_SECRET_KEY: z.string(),
ADMIN_USER_CLERK_ID: z.string(),
OPENAI_API_KEY: z.string(),
GOOGLE_SERVICE_ACCOUNT_EMAIL: z.string().email(),
GOOGLE_SERVICE_ACCOUNT_PRIVATE_KEY: z.string(),
GOOGLE_CALENDAR_ID: z.string(),
GREGOR_MEETING_EMAIL: z.string().email(),
NODE_ENV: z
.enum(["development", "test", "production"])
.default("development"),
@@ -72,6 +76,10 @@ export const env = createEnv({
POSTGRES_PRISMA_URL: process.env.POSTGRES_PRISMA_URL,
ADMIN_USER_CLERK_ID: process.env.ADMIN_USER_CLERK_ID,
OPENAI_API_KEY: process.env.OPENAI_API_KEY,
GOOGLE_SERVICE_ACCOUNT_EMAIL: process.env.GOOGLE_SERVICE_ACCOUNT_EMAIL,
GOOGLE_SERVICE_ACCOUNT_PRIVATE_KEY: process.env.GOOGLE_SERVICE_ACCOUNT_PRIVATE_KEY,
GOOGLE_CALENDAR_ID: process.env.GOOGLE_CALENDAR_ID,
GREGOR_MEETING_EMAIL: process.env.GREGOR_MEETING_EMAIL,
NEXT_PUBLIC_ADMIN_USER_CLERK_ID: process.env.NEXT_PUBLIC_ADMIN_USER_CLERK_ID,
CLERK_SECRET_KEY: process.env.CLERK_SECRET_KEY,
NEXT_PUBLIC_CLERK_PUBLISHABLE_KEY: process.env.NEXT_PUBLIC_CLERK_PUBLISHABLE_KEY,

586
src/server/ai/tools.ts Normal file
View File

@@ -0,0 +1,586 @@
import "server-only";
import { tool } from "ai";
import { desc } from "drizzle-orm";
import { z } from "zod";
import { cancelMeeting } from "~/app/actions/cancelMeeting";
import { scheduleMeeting } from "~/app/actions/scheduleMeeting";
import { db } from "~/server/db";
import { blogPost, music } from "~/server/dbschema/schema";
import { getGoogleCalendarClient, getGoogleCalendarId } from "~/server/googleCalendar";
const contentTypeSchema = z.enum(["cv", "project", "blog", "music"]);
type ContentType = z.infer<typeof contentTypeSchema>;
type SearchResult = {
type: ContentType;
title: string;
snippet: string;
url: string;
score: number;
metadata?: Record<string, unknown>;
};
type ProjectWithStack = Awaited<ReturnType<typeof loadProjects>>[number];
function stripMarkup(value: string | null | undefined) {
return (value ?? "")
.replace(/```[\s\S]*?```/g, " ")
.replace(/`([^`]+)`/g, "$1")
.replace(/<[^>]+>/g, " ")
.replace(/[#*_~[\]()>-]/g, " ")
.replace(/\s+/g, " ")
.trim();
}
function tokenize(value: string) {
return Array.from(new Set(value.toLowerCase().match(/[a-z0-9+#.-]+/g) ?? []));
}
function scoreText(query: string, title: string, body: string, extraTerms: string[] = []) {
const normalizedQuery = query.trim().toLowerCase();
const titleLower = title.toLowerCase();
const bodyLower = body.toLowerCase();
const tokens = tokenize(query);
let score = 0;
if (normalizedQuery && titleLower.includes(normalizedQuery)) score += 40;
if (normalizedQuery && bodyLower.includes(normalizedQuery)) score += 20;
for (const token of tokens) {
if (titleLower.includes(token)) score += 12;
if (bodyLower.includes(token)) score += 6;
if (extraTerms.some((term) => term.toLowerCase() === token)) score += 10;
}
return score;
}
const projectCatalogTerms = new Set(["project", "projects", "portfolio", "work"]);
const genericQuestionTerms = new Set([
"a",
"about",
"all",
"any",
"are",
"can",
"current",
"do",
"give",
"have",
"list",
"me",
"of",
"on",
"show",
"site",
"tell",
"the",
"there",
"these",
"this",
"what",
"which",
"you",
]);
function isProjectCatalogQuery(query: string) {
const tokens = tokenize(query);
if (!tokens.some((token) => projectCatalogTerms.has(token))) return false;
return tokens.every((token) => projectCatalogTerms.has(token) || genericQuestionTerms.has(token));
}
function snippet(value: string, query: string, maxLength = 240) {
const text = stripMarkup(value);
if (text.length <= maxLength) return text;
const tokens = tokenize(query);
const lower = text.toLowerCase();
const firstMatch = tokens
.map((token) => lower.indexOf(token))
.filter((index) => index >= 0)
.sort((a, b) => a - b)[0] ?? 0;
const start = Math.max(0, firstMatch - 60);
const excerpt = text.slice(start, start + maxLength).trim();
return `${start > 0 ? "..." : ""}${excerpt}${start + maxLength < text.length ? "..." : ""}`;
}
function uniqueByUrl(results: SearchResult[]) {
const seen = new Set<string>();
return results.filter((result) => {
const key = `${result.type}:${result.url}:${result.title}`;
if (seen.has(key)) return false;
seen.add(key);
return true;
});
}
async function loadCvEntries() {
const categories = await db.query.cvCategory.findMany({
orderBy: (fields, { asc }) => [asc(fields.layoutPosition), asc(fields.name)],
with: {
cvEntry: {
orderBy: (fields, { desc }) => [desc(fields.toTime), desc(fields.fromTime)],
},
},
});
return categories.flatMap((category) =>
category.cvEntry.map((entry) => ({
...entry,
categoryName: category.name ?? "CV",
})),
);
}
async function loadProjects() {
return db.query.project.findMany({
orderBy: (fields, { asc }) => [asc(fields.orderPos), asc(fields.title), asc(fields.id)],
with: {
techStack: true,
},
});
}
async function buildSearchResults(query: string, types: ContentType[]) {
const selected = new Set(types);
const results: SearchResult[] = [];
if (selected.has("cv")) {
const entries = await loadCvEntries();
for (const entry of entries) {
const body = stripMarkup(`${entry.categoryName} ${entry.description ?? ""}`);
const score = scoreText(query, entry.title, body);
if (score > 0 || !query.trim()) {
results.push({
type: "cv",
title: entry.title,
snippet: snippet(body, query),
url: "/cv",
score,
metadata: {
category: entry.categoryName,
fromTime: entry.fromTime,
toTime: entry.toTime,
},
});
}
}
}
if (selected.has("project")) {
const projects = await loadProjects();
const catalogQuery = isProjectCatalogQuery(query);
for (const [index, item] of projects.entries()) {
const stackItems = item.techStack?.stackItems ?? [];
const body = stripMarkup(`${item.description ?? ""} ${stackItems.join(" ")}`);
const score = catalogQuery ? 1000 - index : scoreText(query, item.title, body, stackItems);
if (score > 0 || !query.trim() || catalogQuery) {
results.push({
type: "project",
title: item.title,
snippet: snippet(body || item.title, query),
url: `/projects#${item.id}`,
score,
metadata: {
id: item.id,
stackItems,
sourceType: item.sourceType,
releaseStatus: item.releaseStatus,
sourceLink: item.sourceLink,
releaseLink: item.releaseLink,
},
});
}
}
}
if (selected.has("blog")) {
const posts = await db
.select({
slug: blogPost.slug,
title: blogPost.title,
date: blogPost.date,
description: blogPost.description,
tags: blogPost.tags,
})
.from(blogPost)
.orderBy(desc(blogPost.date), desc(blogPost.createdAt));
for (const post of posts) {
const tags = post.tags ?? [];
const body = stripMarkup(`${post.description ?? ""} ${tags.join(" ")}`);
const score = scoreText(query, post.title, body, tags);
if (score > 0 || !query.trim()) {
results.push({
type: "blog",
title: post.title,
snippet: snippet(body || post.title, query),
url: `/blog/${post.slug}`,
score,
metadata: {
slug: post.slug,
date: post.date,
tags,
},
});
}
}
}
if (selected.has("music")) {
const tracks = await db.select().from(music).orderBy(desc(music.createdAt));
for (const track of tracks) {
const body = stripMarkup(track.description);
const score = scoreText(query, track.title, body);
if (score > 0 || !query.trim()) {
results.push({
type: "music",
title: track.title,
snippet: snippet(body || track.fileName, query),
url: "/music",
score,
metadata: {
id: track.id,
fileName: track.fileName,
hasStream: Boolean(track.streamUrl),
},
});
}
}
}
return uniqueByUrl(results).sort((a, b) => b.score - a.score || a.title.localeCompare(b.title));
}
function projectMatches(projectItem: ProjectWithStack, idOrTitle: string) {
const normalized = idOrTitle.trim().toLowerCase();
const title = projectItem.title.toLowerCase();
return projectItem.id === idOrTitle || title === normalized || title.includes(normalized);
}
function matchedTerms(text: string, terms: string[]) {
const lower = text.toLowerCase();
return terms.filter((term) => lower.includes(term.toLowerCase()));
}
function parseDate(value: string | undefined, fallback: Date) {
if (!value?.trim()) return fallback;
const date = new Date(value);
return Number.isNaN(date.getTime()) ? fallback : date;
}
function safeTimeZone(value: string | undefined) {
const timeZone = value?.trim() || "Europe/Berlin";
try {
new Intl.DateTimeFormat("en-US", { timeZone }).format(new Date());
return timeZone;
} catch {
return "Europe/Berlin";
}
}
function isLikelyEmail(value: string) {
return /^[^\s@]+@[^\s@]+\.[^\s@]+$/.test(value);
}
function overlaps(
start: Date,
end: Date,
busy: Array<{ start: Date; end: Date }>,
) {
return busy.some((item) => start < item.end && end > item.start);
}
function getZonedParts(date: Date, timeZone: string) {
const formatter = new Intl.DateTimeFormat("en-US", {
timeZone,
weekday: "short",
hourCycle: "h23",
hour: "2-digit",
minute: "2-digit",
});
const parts = Object.fromEntries(formatter.formatToParts(date).map((part) => [part.type, part.value]));
return {
weekday: parts.weekday ?? "",
hour: Number(parts.hour ?? 0),
minute: Number(parts.minute ?? 0),
};
}
function isInsideWorkingHours(start: Date, end: Date, timeZone: string, workdayStartHour: number, workdayEndHour: number) {
const startParts = getZonedParts(start, timeZone);
const endParts = getZonedParts(end, timeZone);
const weekend = startParts.weekday === "Sat" || startParts.weekday === "Sun";
const startMinutes = startParts.hour * 60 + startParts.minute;
const endMinutes = endParts.hour * 60 + endParts.minute;
return !weekend && startMinutes >= workdayStartHour * 60 && endMinutes <= workdayEndHour * 60;
}
function availabilitySlots({
rangeStart,
rangeEnd,
busy,
durationMinutes,
timeZone,
workdayStartHour,
workdayEndHour,
}: {
rangeStart: Date;
rangeEnd: Date;
busy: Array<{ start: Date; end: Date }>;
durationMinutes: number;
timeZone: string;
workdayStartHour: number;
workdayEndHour: number;
}) {
const slots: Array<{ start: string; end: string }> = [];
const stepMinutes = 30;
const durationMs = durationMinutes * 60 * 1000;
const cursor = new Date(Math.ceil(rangeStart.getTime() / (stepMinutes * 60 * 1000)) * stepMinutes * 60 * 1000);
while (cursor.getTime() + durationMs <= rangeEnd.getTime() && slots.length < 10) {
const end = new Date(cursor.getTime() + durationMs);
if (
isInsideWorkingHours(cursor, end, timeZone, workdayStartHour, workdayEndHour)
&& !overlaps(cursor, end, busy)
) {
slots.push({ start: cursor.toISOString(), end: end.toISOString() });
}
cursor.setMinutes(cursor.getMinutes() + stepMinutes);
}
return slots;
}
function logAvailability(requestId: string, message: string, data?: Record<string, unknown>) {
console.log(`[ai:getAvailability:${requestId}] ${message}`, data ?? "");
}
export function createChatTools() {
return {
scheduleMeeting: tool({
description: "Schedule a meeting with Gregor Lohaus and add it to his Google Calendar. Use getAvailability first when the user asks for a meeting at a flexible or uncertain time.",
inputSchema: z.object({
title: z.string().describe("Meeting title, make something up if not provided"),
description: z.string().describe("Meeting description / agenda, make something up if not provided"),
dateTime: z
.string()
.describe("ISO 8601 datetime for the meeting start, e.g. 2026-06-18T10:00:00+02:00"),
durationMinutes: z
.number()
.int()
.min(15)
.max(120)
.describe("Duration of the meeting in minutes, if none provided ask if 20 minutes is ok"),
attendeeEmail: z
.string()
.optional()
.describe("Optional email of the visitor to invite, if provided"),
attendeeName: z.string().optional().describe("Name of the visitor"),
}),
execute: async (input) => {
if (input.attendeeEmail && !isLikelyEmail(input.attendeeEmail)) {
return {
success: false,
error: "The attendee email does not look valid. Ask the visitor to provide a valid email address before scheduling.",
};
}
return scheduleMeeting({ ...input });
},
}),
cancelMeeting: tool({
description: "Remove a previously scheduled meeting from Gregor's dedicated availability calendar. Use only when you have the exact eventId returned by scheduleMeeting.",
inputSchema: z.object({
eventId: z.string().min(1).describe("Google Calendar event id returned by a previous scheduleMeeting tool call."),
}),
execute: async ({ eventId }) => cancelMeeting({ eventId }),
}),
searchSiteContent: tool({
description: "Search Gregor Lohaus's own website content across CV entries, projects, blog posts, and music. Use this for questions about Gregor's work, skills, writing, projects, or site content. For broad questions about Gregor's projects, use types ['project'] so the tool returns the project catalog.",
inputSchema: z.object({
query: z.string().describe("Search query, skill, technology, topic, or phrase."),
types: z.array(contentTypeSchema).optional().describe("Optional content types to search. Omit to search all site content."),
limit: z.number().int().min(1).max(12).optional().describe("Maximum number of results to return."),
}),
execute: async ({ query, types, limit }) => {
const results = await buildSearchResults(query, types?.length ? types : ["cv", "project", "blog", "music"]);
return {
success: true,
query,
results: results.slice(0, limit ?? 8).map(({ score, ...result }) => result),
};
},
}),
getRelevantExperience: tool({
description: "Find Gregor's most relevant CV entries and projects for a role, skill set, seniority, or domain. Use this for recruiter-style qualification questions.",
inputSchema: z.object({
role: z.string().optional().describe("Role or job title, such as full-stack engineer or React Native developer."),
skills: z.array(z.string()).optional().describe("Technologies, tools, or skills to match."),
domain: z.string().optional().describe("Product or business domain to match, if any."),
seniority: z.string().optional().describe("Seniority or responsibility level to match, if any."),
limit: z.number().int().min(1).max(10).optional().describe("Maximum matching entries to return."),
}),
execute: async ({ role, skills, domain, seniority, limit }) => {
const terms = [role, domain, seniority, ...(skills ?? [])].filter((value): value is string => Boolean(value?.trim()));
const query = terms.join(" ");
const results = await buildSearchResults(query, ["cv", "project"]);
const selected = results.slice(0, limit ?? 6);
return {
success: true,
query,
matches: selected.map(({ score, ...result }) => ({
...result,
matchedTerms: matchedTerms(`${result.title} ${result.snippet} ${(result.metadata?.stackItems as string[] | undefined)?.join(" ") ?? ""}`, terms),
whyRelevant: result.type === "project"
? "Project match based on title, description, and technology stack."
: "CV match based on experience title, category, and description.",
})),
};
},
}),
getProjectDetails: tool({
description: "Get detailed information for one of Gregor's projects, including description, stack, source link, release link, and project page URL.",
inputSchema: z.object({
idOrTitle: z.string().min(1).describe("Project id, exact title, or partial project title."),
}),
execute: async ({ idOrTitle }) => {
const projects = await loadProjects();
const found = projects.find((item) => projectMatches(item, idOrTitle));
if (!found) {
return {
success: false,
error: `No project matched "${idOrTitle}".`,
suggestions: projects.slice(0, 5).map((item) => ({
id: item.id,
title: item.title,
url: `/projects#${item.id}`,
})),
};
}
return {
success: true,
project: {
id: found.id,
title: found.title,
description: stripMarkup(found.description),
sourceType: found.sourceType,
sourceLink: found.sourceLink,
releaseStatus: found.releaseStatus,
releaseLink: found.releaseLink,
stackItems: found.techStack?.stackItems ?? [],
url: `/projects#${found.id}`,
},
};
},
}),
getAvailability: tool({
description: "Check Gregor's Google Calendar availability and suggest open meeting slots. Use this directly for questions like 'when is the next open spot?' or 'what times are available?'. If no date range is provided, it checks from now. Use before scheduling when the requested time is flexible.",
inputSchema: z.object({
fromDateTime: z.string().optional().describe("ISO 8601 range start. Defaults to now."),
toDateTime: z.string().optional().describe("ISO 8601 range end. Defaults to 14 days after the range start."),
durationMinutes: z.number().int().min(15).max(120).optional().describe("Desired meeting duration. Defaults to 30 minutes."),
timeZone: z.string().optional().describe("IANA time zone for working-hours filtering. Defaults to Europe/Berlin."),
workdayStartHour: z.number().int().min(0).max(23).optional().describe("Earliest local start hour. Defaults to 9."),
workdayEndHour: z.number().int().min(1).max(24).optional().describe("Latest local end hour. Defaults to 17."),
}),
execute: async (input) => {
const requestId = crypto.randomUUID();
logAvailability(requestId, "start", {
input,
});
const durationMinutes = input.durationMinutes ?? 30;
const timeZone = safeTimeZone(input.timeZone);
const workdayStartHour = input.workdayStartHour ?? 9;
const workdayEndHour = Math.max(input.workdayEndHour ?? 17, workdayStartHour + 1);
const rangeStart = parseDate(input.fromDateTime, new Date());
const defaultEnd = new Date(rangeStart.getTime() + 14 * 24 * 60 * 60 * 1000);
const requestedEnd = parseDate(input.toDateTime, defaultEnd);
const maxEnd = new Date(rangeStart.getTime() + 31 * 24 * 60 * 60 * 1000);
const rangeEnd = requestedEnd <= rangeStart ? defaultEnd : requestedEnd > maxEnd ? maxEnd : requestedEnd;
logAvailability(requestId, "resolved range", {
durationMinutes,
timeZone,
workdayStartHour,
workdayEndHour,
rangeStart: rangeStart.toISOString(),
rangeEnd: rangeEnd.toISOString(),
});
const calendar = getGoogleCalendarClient();
const calendarId = getGoogleCalendarId();
logAvailability(requestId, "service account calendar ready", {
calendarId,
});
let busy: Array<{ start: Date; end: Date }>;
try {
logAvailability(requestId, "freebusy request");
const response = await calendar.freebusy.query({
requestBody: {
timeMin: rangeStart.toISOString(),
timeMax: rangeEnd.toISOString(),
timeZone,
items: [{ id: calendarId }],
},
});
busy = (response.data.calendars?.[calendarId]?.busy ?? [])
.map((item) => ({
start: parseDate(item.start ?? undefined, rangeStart),
end: parseDate(item.end ?? undefined, rangeStart),
}))
.filter((item) => item.end > item.start);
logAvailability(requestId, "freebusy response", {
busyCount: busy.length,
});
} catch (error) {
console.error(`[ai:getAvailability:${requestId}] freebusy failed`, error);
return {
success: false,
error: "Failed to read Gregor's Google Calendar availability.",
};
}
const availableSlots = availabilitySlots({
rangeStart,
rangeEnd,
busy,
durationMinutes,
timeZone,
workdayStartHour,
workdayEndHour,
});
logAvailability(requestId, "complete", {
busyCount: busy.length,
availableSlotCount: availableSlots.length,
firstAvailableSlot: availableSlots[0],
});
const nextAvailableSlot = availableSlots[0] ?? null;
return {
success: true,
range: {
start: rangeStart.toISOString(),
end: rangeEnd.toISOString(),
timeZone,
},
durationMinutes,
busy: busy.map((item) => ({
start: item.start.toISOString(),
end: item.end.toISOString(),
})),
nextAvailableSlot,
availableSlots,
};
},
}),
};
}

View File

@@ -175,6 +175,7 @@ export const chatMessageRelations = relations(chatMessage, ({ one }) => ({
export const systemSettings = createTable(
"systemSetting",
(d) => ({
systemPropmt: d.text()
systemPropmt: d.text(),
model: d.text()
})
)

View File

@@ -0,0 +1,20 @@
import "server-only";
import { google } from "googleapis";
import { env } from "~/env";
const calendarScope = "https://www.googleapis.com/auth/calendar";
export function getGoogleCalendarClient() {
const auth = new google.auth.JWT({
email: env.GOOGLE_SERVICE_ACCOUNT_EMAIL,
key: env.GOOGLE_SERVICE_ACCOUNT_PRIVATE_KEY.replace(/\\n/g, "\n"),
scopes: [calendarScope],
});
return google.calendar({ version: "v3", auth });
}
export function getGoogleCalendarId() {
return env.GOOGLE_CALENDAR_ID;
}

View File

@@ -305,6 +305,25 @@ export const blogRouter = router({
};
}),
metadataBySlug: publicProcedure.input(z.string()).query(async ({ input: slug }) => {
const post = await db.query.blogPost.findFirst({
where(fields, operators) {
return operators.eq(fields.slug, slug);
},
});
if (!post) throw new TRPCError({ code: "NOT_FOUND", message: `Post "${slug}" not found` });
return {
slug: post.slug,
title: post.title,
date: post.date,
description: post.description,
tags: post.tags ?? [],
fileUrl: post.fileUrl,
};
}),
syncFromUploadThing: publicProcedure.mutation(async () => {
await assertAdmin();

View File

@@ -7,6 +7,26 @@ import { isAdmin } from '~/app/actions';
import { z } from 'zod';
import { eq } from 'drizzle-orm';
import { clerkClient, auth } from '@clerk/nextjs/server'
import { env } from '~/env'
export const DEFAULT_MODEL = 'gpt-5-mini'
// Models returned by the OpenAI API that aren't usable for chat completions.
const NON_CHAT_MODEL = /embedding|image|audio|realtime|transcribe|tts|whisper|moderation|dall-e|search|codex|instruct/
async function readSettings() {
return db.select().from(systemSettings).limit(1).then((r) => r[0])
}
async function writeSettings(values: { systemPropmt?: string | null; model?: string | null }) {
const current = await readSettings()
await db.delete(systemSettings)
await db.insert(systemSettings).values({
systemPropmt: values.systemPropmt ?? current?.systemPropmt ?? null,
model: values.model ?? current?.model ?? null,
})
}
export const chatRouter = router({
getSession: publicProcedure.query(async () => {
const { userId } = await auth();
@@ -66,13 +86,34 @@ export const chatRouter = router({
}),
getSystemPrompt: publicProcedure.query(async () => {
const row = await db.select().from(systemSettings).limit(1).then((r) => r[0])
const row = await readSettings()
return row?.systemPropmt ?? ''
}),
updateSystemPrompt: publicProcedure.input(z.object({ prompt: z.string() })).mutation(async ({ input }) => {
if (!(await isAdmin())) throw new TRPCError({ code: 'FORBIDDEN' })
await db.delete(systemSettings)
await db.insert(systemSettings).values({ systemPropmt: input.prompt })
await writeSettings({ systemPropmt: input.prompt })
}),
getModel: publicProcedure.query(async () => {
const row = await readSettings()
return row?.model ?? DEFAULT_MODEL
}),
listModels: publicProcedure.query(async () => {
if (!(await isAdmin())) throw new TRPCError({ code: 'FORBIDDEN' })
const res = await fetch('https://api.openai.com/v1/models', {
headers: { Authorization: `Bearer ${env.OPENAI_API_KEY}` },
})
if (!res.ok) {
throw new TRPCError({ code: 'INTERNAL_SERVER_ERROR', message: `failed to fetch models (${res.status})` })
}
const json = (await res.json()) as { data: { id: string }[] }
return json.data
.map((m) => m.id)
.filter((id) => (id.startsWith('gpt') || /^o\d/.test(id) || id.startsWith('chatgpt')) && !NON_CHAT_MODEL.test(id))
.sort()
}),
updateModel: publicProcedure.input(z.object({ model: z.string() })).mutation(async ({ input }) => {
if (!(await isAdmin())) throw new TRPCError({ code: 'FORBIDDEN' })
await writeSettings({ model: input.model })
}),
})

View File

@@ -13,6 +13,20 @@ export const cvCategoryRouter = router({
console.log(res);
return res;
}),
// Single round-trip for the whole CV page: every category (across all layout
// positions) with its entries already populated. Lets the page fetch-then-render
// instead of waterfalling per-category/per-entry queries, so all content is
// present before the entrance animation runs.
listAllWithEntries: publicProcedure.query(async () => {
const res = await db.query.cvCategory.findMany({
with: {
cvEntry: {
orderBy: (t, { desc }) => desc(t.toTime),
},
},
})
return res;
}),
getById: publicProcedure.input(z.string()).query(async ({input}) => {
const res = await db.query.cvCategory.findFirst({
where(fields, operators) {

View File

@@ -1,5 +1,8 @@
import { initTRPC } from "@trpc/server"
import superjson from "superjson"
const t = initTRPC.create();
const t = initTRPC.create({
transformer: superjson,
});
export const router = t.router;
export const publicProcedure = t.procedure;