Add public MentorProfile page
cgen-ebcf6a2faf1d4c8880c4f27d797b33ed
This commit is contained in:
parent
52d3990d53
commit
fb983b69cd
1 changed files with 107 additions and 0 deletions
107
client/pages/community/MentorProfile.tsx
Normal file
107
client/pages/community/MentorProfile.tsx
Normal file
|
|
@ -0,0 +1,107 @@
|
||||||
|
import Layout from "@/components/Layout";
|
||||||
|
import { Badge } from "@/components/ui/badge";
|
||||||
|
import { Button } from "@/components/ui/button";
|
||||||
|
import { Card, CardContent, CardDescription, CardHeader, CardTitle } from "@/components/ui/card";
|
||||||
|
import { useEffect, useMemo, useState } from "react";
|
||||||
|
import { useNavigate, useParams } from "react-router-dom";
|
||||||
|
import { aethexSocialService } from "@/lib/aethex-social-service";
|
||||||
|
|
||||||
|
interface MentorRow {
|
||||||
|
user_id: string;
|
||||||
|
bio: string | null;
|
||||||
|
expertise: string[] | null;
|
||||||
|
available: boolean;
|
||||||
|
hourly_rate: number | null;
|
||||||
|
user_profiles?: {
|
||||||
|
id: string;
|
||||||
|
full_name: string | null;
|
||||||
|
username: string | null;
|
||||||
|
avatar_url: string | null;
|
||||||
|
bio: string | null;
|
||||||
|
} | null;
|
||||||
|
}
|
||||||
|
|
||||||
|
export default function MentorProfile() {
|
||||||
|
const { username } = useParams<{ username: string }>();
|
||||||
|
const navigate = useNavigate();
|
||||||
|
const [mentor, setMentor] = useState<MentorRow | null>(null);
|
||||||
|
const [loading, setLoading] = useState(true);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
const load = async () => {
|
||||||
|
setLoading(true);
|
||||||
|
try {
|
||||||
|
const rows = (await aethexSocialService.listMentors({ q: username, limit: 50 })) as MentorRow[];
|
||||||
|
const found = rows.find((r) => (r.user_profiles?.username || "").toLowerCase() === (username || "").toLowerCase());
|
||||||
|
setMentor(found || null);
|
||||||
|
} catch {
|
||||||
|
setMentor(null);
|
||||||
|
} finally {
|
||||||
|
setLoading(false);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
load();
|
||||||
|
}, [username]);
|
||||||
|
|
||||||
|
const displayName = useMemo(() => mentor?.user_profiles?.full_name || mentor?.user_profiles?.username || "Mentor", [mentor]);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Layout>
|
||||||
|
<div className="container mx-auto max-w-7xl px-4 py-12">
|
||||||
|
<div className="mb-6">
|
||||||
|
<Badge variant="outline" className="mb-2">Mentorship</Badge>
|
||||||
|
<h1 className="text-3xl font-bold">{loading ? "Loading…" : displayName}</h1>
|
||||||
|
{!loading && (
|
||||||
|
<p className="text-muted-foreground mt-1">{mentor?.user_profiles?.bio || mentor?.bio || "Mentor profile"}</p>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{loading && (
|
||||||
|
<Card><CardContent className="p-6 text-sm text-muted-foreground">Loading profile…</CardContent></Card>
|
||||||
|
)}
|
||||||
|
|
||||||
|
{!loading && !mentor && (
|
||||||
|
<Card><CardContent className="p-6 text-sm text-muted-foreground">Mentor not found.</CardContent></Card>
|
||||||
|
)}
|
||||||
|
|
||||||
|
{!loading && mentor && (
|
||||||
|
<div className="grid gap-6 md:grid-cols-3">
|
||||||
|
<Card className="md:col-span-2">
|
||||||
|
<CardHeader>
|
||||||
|
<CardTitle>About</CardTitle>
|
||||||
|
<CardDescription>Background and focus areas</CardDescription>
|
||||||
|
</CardHeader>
|
||||||
|
<CardContent className="space-y-3">
|
||||||
|
{mentor.bio && <p className="text-sm text-muted-foreground">{mentor.bio}</p>}
|
||||||
|
<div className="flex flex-wrap gap-2">
|
||||||
|
{(mentor.expertise || []).map((tag) => (
|
||||||
|
<Badge key={tag} variant="outline">{tag}</Badge>
|
||||||
|
))}
|
||||||
|
</div>
|
||||||
|
</CardContent>
|
||||||
|
</Card>
|
||||||
|
|
||||||
|
<Card>
|
||||||
|
<CardHeader>
|
||||||
|
<CardTitle>Booking</CardTitle>
|
||||||
|
<CardDescription>Availability and rate</CardDescription>
|
||||||
|
</CardHeader>
|
||||||
|
<CardContent className="space-y-3">
|
||||||
|
<div className="text-sm">
|
||||||
|
<div>Availability: {mentor.available ? "Accepting requests" : "Unavailable"}</div>
|
||||||
|
{typeof mentor.hourly_rate === "number" && (
|
||||||
|
<div>Rate: ${mentor.hourly_rate}/hr</div>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
<div className="flex gap-2">
|
||||||
|
<Button onClick={() => navigate(`/community/mentorship?m=${mentor.user_profiles?.username || mentor.user_id}`)}>Request mentorship</Button>
|
||||||
|
<Button variant="outline" onClick={() => navigate("/community/mentorship")}>Back to directory</Button>
|
||||||
|
</div>
|
||||||
|
</CardContent>
|
||||||
|
</Card>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
</Layout>
|
||||||
|
);
|
||||||
|
}
|
||||||
Loading…
Reference in a new issue