chore: Update translations
This commit is contained in:
@ -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>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
@ -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",
|
||||||
|
@ -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",
|
||||||
|
Reference in New Issue
Block a user