chore: Update translations

This commit is contained in:
2023-11-21 13:54:06 +00:00
parent 2030cb2061
commit 0f5324ccad
3 changed files with 150 additions and 153 deletions

View File

@ -12,166 +12,163 @@ import Tabs, { Tab } from "./Tabs";
import { Bar, BarChart, ResponsiveContainer, Tooltip, XAxis, YAxis } from "recharts"; import { Bar, BarChart, ResponsiveContainer, Tooltip, XAxis, YAxis } from "recharts";
interface StatSlot { interface StatSlot {
time: string; time: string;
reactions: number; reactions: number;
reposts: number; reposts: number;
quotes: number; quotes: number;
mentions: number; mentions: number;
zaps: number; zaps: number;
} }
const enum NotificationSummaryPeriod { const enum NotificationSummaryPeriod {
Daily, Daily,
Weekly, Weekly,
} }
const enum NotificationSummaryFilter { const enum NotificationSummaryFilter {
Reactions = 1, Reactions = 1,
Reposts = 2, Reposts = 2,
Mentions = 4, Mentions = 4,
Zaps = 8, Zaps = 8,
All = 255, All = 255,
} }
export default function NotificationSummary({ evs }: { evs: Array<TaggedNostrEvent> }) { export default function NotificationSummary({ evs }: { evs: Array<TaggedNostrEvent> }) {
const [period, setPeriod] = useState(NotificationSummaryPeriod.Daily); const [period, setPeriod] = useState(NotificationSummaryPeriod.Daily);
const [filter, setFilter] = useState(NotificationSummaryFilter.All); const [filter, setFilter] = useState(NotificationSummaryFilter.All);
const periodTabs = [ const periodTabs = [
{ {
value: NotificationSummaryPeriod.Daily, value: NotificationSummaryPeriod.Daily,
text: <FormattedMessage defaultMessage="Daily" id="zxvhnE" />, text: <FormattedMessage defaultMessage="Daily" id="zxvhnE" />,
},
{
value: NotificationSummaryPeriod.Weekly,
text: <FormattedMessage defaultMessage="Weekly" id="/clOBU" />,
},
] as Array<Tab>;
const hasFlag = (v: number, f: NotificationSummaryFilter) => {
return (v & f) > 0;
};
const getWeek = (d: Date) => {
const onejan = new Date(d.getFullYear(), 0, 1);
const today = new Date(d.getFullYear(), d.getMonth(), d.getDate());
const dayOfYear = (today.getTime() - onejan.getTime() + 86400000) / 86400000;
return Math.ceil(dayOfYear / 7);
};
const stats = useMemo(() => {
return orderAscending(evs)
.filter(a => (period === NotificationSummaryPeriod.Daily ? a.created_at > unixNow() - 14 * Day : true))
.reduce(
(acc, v) => {
const date = new Date(v.created_at * 1000);
const key =
period === NotificationSummaryPeriod.Daily
? `${date.getMonth() + 1}/${date.getDate()}`
: getWeek(date).toString();
acc[key] ??= {
time: key,
reactions: 0,
reposts: 0,
quotes: 0,
mentions: 0,
zaps: 0,
};
if (v.kind === EventKind.Reaction) {
acc[key].reactions++;
} else if (v.kind === EventKind.Repost) {
acc[key].reposts++;
} else if (v.kind === EventKind.ZapReceipt) {
acc[key].zaps++;
}
if (v.kind === EventKind.TextNote) {
acc[key].mentions++;
}
return acc;
}, },
{ {} as Record<string, StatSlot>,
value: NotificationSummaryPeriod.Weekly, );
text: <FormattedMessage defaultMessage="Weekly" id="/clOBU" />, }, [evs, period]);
},
] as Array<Tab>;
const hasFlag = (v: number, f: NotificationSummaryFilter) => { if (evs.length === 0) return;
return (v & f) > 0;
};
const getWeek = (d: Date) => {
const onejan = new Date(d.getFullYear(), 0, 1);
const today = new Date(d.getFullYear(), d.getMonth(), d.getDate());
const dayOfYear = (today.getTime() - onejan.getTime() + 86400000) / 86400000;
return Math.ceil(dayOfYear / 7);
};
const stats = useMemo(() => {
return orderAscending(evs)
.filter(a => (period === NotificationSummaryPeriod.Daily ? a.created_at > unixNow() - 14 * Day : true))
.reduce(
(acc, v) => {
const date = new Date(v.created_at * 1000);
const key =
period === NotificationSummaryPeriod.Daily
? `${date.getMonth() + 1}/${date.getDate()}`
: getWeek(date).toString();
acc[key] ??= {
time: key,
reactions: 0,
reposts: 0,
quotes: 0,
mentions: 0,
zaps: 0,
};
if (v.kind === EventKind.Reaction) {
acc[key].reactions++;
} else if (v.kind === EventKind.Repost) {
acc[key].reposts++;
} else if (v.kind === EventKind.ZapReceipt) {
acc[key].zaps++;
}
if (v.kind === EventKind.TextNote) {
acc[key].mentions++;
}
return acc;
},
{} as Record<string, StatSlot>,
);
}, [evs, period]);
if (evs.length === 0) return;
const filterIcon = (f: NotificationSummaryFilter, icon: string, iconActiveClass: string) => {
const active = hasFlag(filter, f);
return (
<AsyncIcon
className={classNames("button-icon-sm transparent", { active, [iconActiveClass]: active })}
onClick={() => setFilter(v => v ^ f)}
name={""}
iconName={icon}
/>
);
};
const filterIcon = (f: NotificationSummaryFilter, icon: string, iconActiveClass: string) => {
const active = hasFlag(filter, f);
return ( return (
<div className="flex flex-col g12 p bb"> <AsyncIcon
<div className="flex justify-between"> className={classNames("button-icon-sm transparent", { active, [iconActiveClass]: active })}
<h2> onClick={() => setFilter(v => v ^ f)}
<FormattedMessage defaultMessage="Summary" id="PJeJFc" description="Notifications summary" /> name={""}
</h2> iconName={icon}
<div className="flex items-center g8"> />
{filterIcon(NotificationSummaryFilter.Reactions, "heart-solid", "text-heart")}
{filterIcon(NotificationSummaryFilter.Zaps, "zap-solid", "text-zap")}
{filterIcon(NotificationSummaryFilter.Reposts, "reverse-left", "text-repost")}
{filterIcon(NotificationSummaryFilter.Mentions, "at-sign", "text-mention")}
</div>
</div>
<Tabs tabs={periodTabs} tab={unwrap(periodTabs.find(a => a.value === period))} setTab={t => setPeriod(t.value)} />
<div>
<ResponsiveContainer height={200}>
<BarChart
data={Object.values(stats)}
margin={{ left: 0, right: 0 }}
style={{ userSelect: "none" }}>
<XAxis dataKey="time" />
<YAxis />
{hasFlag(filter, NotificationSummaryFilter.Reactions) && (
<Bar dataKey="reactions" fill="var(--heart)" stackId="" />
)}
{hasFlag(filter, NotificationSummaryFilter.Reposts) && (
<Bar dataKey="reposts" fill="var(--repost)" stackId="" />
)}
{hasFlag(filter, NotificationSummaryFilter.Mentions) && (
<Bar dataKey="mentions" fill="var(--mention)" stackId="" />
)}
{hasFlag(filter, NotificationSummaryFilter.Zaps) && <Bar dataKey="zaps" fill="var(--zap)" stackId="" />}
<Tooltip
cursor={{ fill: "rgba(255,255,255,0.2)" }}
content={({ active, payload }) => {
if (active && payload && payload.length) {
return (
<div className="summary-tooltip">
<div className="flex flex-col g12">
<Icon name="heart-solid" className="text-heart" />
{formatShort(payload.find(a => a.name === "reactions")?.value as number)}
</div>
<div className="flex flex-col g12">
<Icon name="zap-solid" className="text-zap" />
{formatShort(payload.find(a => a.name === "zaps")?.value as number)}
</div>
<div className="flex flex-col g12">
<Icon name="reverse-left" className="text-repost" />
{formatShort(payload.find(a => a.name === "reposts")?.value as number)}
</div>
<div className="flex flex-col g12">
<Icon name="at-sign" className="text-mention" />
{formatShort(payload.find(a => a.name === "mentions")?.value as number)}
</div>
</div>
);
}
return null;
}}
/>
</BarChart>
</ResponsiveContainer>
</div>
</div>
); );
} };
return (
<div className="flex flex-col g12 p bb">
<div className="flex justify-between">
<h2>
<FormattedMessage defaultMessage="Summary" id="PJeJFc" description="Notifications summary" />
</h2>
<div className="flex items-center g8">
{filterIcon(NotificationSummaryFilter.Reactions, "heart-solid", "text-heart")}
{filterIcon(NotificationSummaryFilter.Zaps, "zap-solid", "text-zap")}
{filterIcon(NotificationSummaryFilter.Reposts, "reverse-left", "text-repost")}
{filterIcon(NotificationSummaryFilter.Mentions, "at-sign", "text-mention")}
</div>
</div>
<Tabs tabs={periodTabs} tab={unwrap(periodTabs.find(a => a.value === period))} setTab={t => setPeriod(t.value)} />
<div>
<ResponsiveContainer height={200}>
<BarChart data={Object.values(stats)} margin={{ left: 0, right: 0 }} style={{ userSelect: "none" }}>
<XAxis dataKey="time" />
<YAxis />
{hasFlag(filter, NotificationSummaryFilter.Reactions) && (
<Bar dataKey="reactions" fill="var(--heart)" stackId="" />
)}
{hasFlag(filter, NotificationSummaryFilter.Reposts) && (
<Bar dataKey="reposts" fill="var(--repost)" stackId="" />
)}
{hasFlag(filter, NotificationSummaryFilter.Mentions) && (
<Bar dataKey="mentions" fill="var(--mention)" stackId="" />
)}
{hasFlag(filter, NotificationSummaryFilter.Zaps) && <Bar dataKey="zaps" fill="var(--zap)" stackId="" />}
<Tooltip
cursor={{ fill: "rgba(255,255,255,0.2)" }}
content={({ active, payload }) => {
if (active && payload && payload.length) {
return (
<div className="summary-tooltip">
<div className="flex flex-col g12">
<Icon name="heart-solid" className="text-heart" />
{formatShort(payload.find(a => a.name === "reactions")?.value as number)}
</div>
<div className="flex flex-col g12">
<Icon name="zap-solid" className="text-zap" />
{formatShort(payload.find(a => a.name === "zaps")?.value as number)}
</div>
<div className="flex flex-col g12">
<Icon name="reverse-left" className="text-repost" />
{formatShort(payload.find(a => a.name === "reposts")?.value as number)}
</div>
<div className="flex flex-col g12">
<Icon name="at-sign" className="text-mention" />
{formatShort(payload.find(a => a.name === "mentions")?.value as number)}
</div>
</div>
);
}
return null;
}}
/>
</BarChart>
</ResponsiveContainer>
</div>
</div>
);
}

View File

@ -344,7 +344,7 @@
"eJj8HD": "Verifiziert werden", "eJj8HD": "Verifiziert werden",
"eSzf2G": "Ein einzelner Zap von {nIn} sats wird {nOut} sats dem Zap Pool zuweisen.", "eSzf2G": "Ein einzelner Zap von {nIn} sats wird {nOut} sats dem Zap Pool zuweisen.",
"eXT2QQ": "Gruppenchat", "eXT2QQ": "Gruppenchat",
"egib+2": "{n,plural,=1{& {n} other} other{& {n} others}}", "egib+2": "{n,plural,one {}=1{& {n} anderer} other{& {n} andere}}",
"fBI91o": "Zap", "fBI91o": "Zap",
"fBlba3": "Danke für die Verwendung von {site}. Wir würden uns über eine Spende freuen.", "fBlba3": "Danke für die Verwendung von {site}. Wir würden uns über eine Spende freuen.",
"fOksnD": "Abstimmung nicht möglich, da der LNURL-Dienst keine Zaps unterstützt", "fOksnD": "Abstimmung nicht möglich, da der LNURL-Dienst keine Zaps unterstützt",

View File

@ -21,13 +21,13 @@
"08zn6O": "Kulcsok exportálása", "08zn6O": "Kulcsok exportálása",
"0Azlrb": "Menedzselés", "0Azlrb": "Menedzselés",
"0BUTMv": "Keresés...", "0BUTMv": "Keresés...",
"0HFX0T": "Use Exact Location", "0HFX0T": "Pontos hely használata",
"0jOEtS": "Érvénytelen LNURL", "0jOEtS": "Érvénytelen LNURL",
"0mch2Y": "név nem engedélyezett karaktereket tartalmaz", "0mch2Y": "név nem engedélyezett karaktereket tartalmaz",
"0siT4z": "Politika", "0siT4z": "Politika",
"0uoY11": "Státusz megjelenítése", "0uoY11": "Státusz megjelenítése",
"0yO7wF": "{n} másodperc", "0yO7wF": "{n} másodperc",
"1H4Keq": "{n} users", "1H4Keq": "{n} felhasználó",
"1Mo59U": "Biztos hogy a kedvencekből ezt a bejegyzést el akarod távolítani?", "1Mo59U": "Biztos hogy a kedvencekből ezt a bejegyzést el akarod távolítani?",
"1R43+L": "Írd be a Nostr Wallet Connect konfigurációt", "1R43+L": "Írd be a Nostr Wallet Connect konfigurációt",
"1c4YST": "Kapcsolódás a: {node} 🎉", "1c4YST": "Kapcsolódás a: {node} 🎉",
@ -276,7 +276,7 @@
"Ub+AGc": "Bejelentkezés", "Ub+AGc": "Bejelentkezés",
"Up5U7K": "Tiltás", "Up5U7K": "Tiltás",
"UrKTqQ": "Van aktív iris.to fiókod", "UrKTqQ": "Van aktív iris.to fiókod",
"VL900k": "Recommended Relays", "VL900k": "Ajánlott csomópontok",
"VN0+Fz": "Egyenleg: {amount} sats", "VN0+Fz": "Egyenleg: {amount} sats",
"VOjC1i": "Válaszd ki mely szolgáltatóhoz legyenek a fájlok feltöltve", "VOjC1i": "Válaszd ki mely szolgáltatóhoz legyenek a fájlok feltöltve",
"VR5eHw": "Publikus kulcs (npub/nprofile)", "VR5eHw": "Publikus kulcs (npub/nprofile)",
@ -344,7 +344,7 @@
"eJj8HD": "Légy Azonosítva", "eJj8HD": "Légy Azonosítva",
"eSzf2G": "Egyetlen {nIn} sats zap a zap-medencéhez {nOut} sats-ot foglal le.", "eSzf2G": "Egyetlen {nIn} sats zap a zap-medencéhez {nOut} sats-ot foglal le.",
"eXT2QQ": "Csoportbeszélgetés", "eXT2QQ": "Csoportbeszélgetés",
"egib+2": "{n,plural,=1{& {n} other} other{& {n} others}}", "egib+2": "{n,plural,=1{és még {n} valaki} other{és {n} másik felhasználó}}",
"fBI91o": "Zap", "fBI91o": "Zap",
"fBlba3": "Köszönjük, hogy a(z) {site} használod, ha teheted fontold meg az adományozást.", "fBlba3": "Köszönjük, hogy a(z) {site} használod, ha teheted fontold meg az adományozást.",
"fOksnD": "Nem szavazhatsz, mert az LNURL szolgáltatód a zap-eket nem támogatja", "fOksnD": "Nem szavazhatsz, mert az LNURL szolgáltatód a zap-eket nem támogatja",