1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
|
// This is a Server Component
import React, { Suspense } from "react";
import db from "@/lib/db";
import {
Card,
CardHeader,
CardDescription,
CardContent,
CardFooter,
CardTitle,
} from "@/components/ui/card";
import { NLP } from "sortug-ai";
import { Volume2, Link as LinkIcon } from "lucide-react";
import { isTonal } from "@/lib/lang/utils";
import { CardResponse, SyllableToken } from "@/lib/types/cards";
import { deconstructSyllable } from "@/lib/calls/nlp";
export default async function (props: { data: CardResponse }) {
const { expression } = props.data;
const { result } = await deconstructSyllable(expression.spelling);
return (
<div className="absolute w-full h-full bg-white dark:bg-slate-800 rounded-xl backface-hidden flex flex-col justify-center gap-8 items-center p-6">
<p className="text-5xl cursor-pointer hover:text-blue-700 font-semibold text-slate-800 dark:text-slate-100 text-center">
{expression.spelling}
</p>
<Suspense fallback={<IpaDisplay ipaEntries={expression.ipa} />}>
<Deconstructed syl={result} />
</Suspense>
</div>
);
}
function Deconstructed({ syl }: { syl: SyllableToken[] }) {
return (
<div>
{syl.map((tok) => (
<span></span>
))}
</div>
);
}
// Helper component for IPA display
const IpaDisplay = ({
ipaEntries,
}: {
ipaEntries: Array<{ ipa: string; tags?: string[] }>;
}) => {
if (!ipaEntries || ipaEntries.length === 0) return null;
return (
<div className="flex items-center space-x-2 flex-wrap">
{ipaEntries.map((entry, index) => {
const tags = entry.tags ? entry.tags : [];
return (
<span key={index} className="text-lg text-blue-600 font-serif">
{entry.ipa}{" "}
{tags.length > 0 && (
<span className="text-xs text-gray-500">({tags.join(", ")})</span>
)}
</span>
);
})}
<button
className="p-1 text-blue-500 hover:text-blue-700 transition-colors"
title="Pronounce"
// onClick={() => {
// /* Pronunciation logic would be client-side or a server roundtrip for audio file. */ alert(
// "Pronunciation feature not implemented for server component.",
// );
// }}
>
<Volume2 size={20} />
</button>
</div>
);
};
function Tones({ text, lang }: WordProps) {
return <div></div>;
}
function NotTones({ text, lang }: WordProps) {
return <div></div>;
}
|