tutu/src/timelines/toots/TootPoll.tsx

185 lines
4.7 KiB
TypeScript
Raw Normal View History

2024-11-23 20:55:37 +08:00
import {
batch,
createSelector,
createSignal,
Index,
Show,
untrack,
type Component,
} from "solid-js";
import "./TootPoll.css";
import type { mastodon } from "masto";
import { resolveCustomEmoji } from "../../masto/toot";
import {
Button,
Checkbox,
Divider,
List,
ListItemButton,
ListItemText,
Radio,
} from "@suid/material";
2024-11-24 17:16:06 +08:00
import { formatDistance, isBefore } from "date-fns";
2024-11-23 20:55:37 +08:00
import { useTimeSource } from "~platform/timesrc";
import { useDateFnLocale } from "~platform/i18n";
import TootPollDialog from "./TootPollDialog";
import { ANIM_CURVE_STD } from "~material/theme";
2024-11-23 22:21:14 +08:00
import { useTootEnv } from "../RegularToot";
2024-11-23 20:55:37 +08:00
type TootPollProps = {
2024-11-24 17:16:06 +08:00
value: mastodon.v1.Poll;
status: mastodon.v1.Status;
2024-11-23 20:55:37 +08:00
};
const TootPoll: Component<TootPollProps> = (props) => {
2025-01-04 17:10:54 +08:00
let list!: HTMLUListElement;
2024-11-24 17:16:06 +08:00
const { vote } = useTootEnv();
2024-11-23 22:21:14 +08:00
2024-11-23 20:55:37 +08:00
const now = useTimeSource();
const dateFnLocale = useDateFnLocale();
const [mustShowResult, setMustShowResult] = createSignal<boolean>();
const [showVoteDialog, setShowVoteDialog] = createSignal(false);
const [initialVote, setInitialVote] = createSignal(0);
2024-11-24 17:16:06 +08:00
const poll = () => props.value;
2024-11-23 22:21:14 +08:00
2024-11-23 20:55:37 +08:00
const isShowResult = () => {
const n = mustShowResult();
if (typeof n !== "undefined") {
return n;
}
2024-11-23 22:21:14 +08:00
return poll().expired || poll().voted;
2024-11-23 20:55:37 +08:00
};
const isOwnVote = createSelector(
2024-11-23 22:21:14 +08:00
() => poll().ownVotes,
2024-11-23 20:55:37 +08:00
(idx: number, votes) => votes?.includes(idx) || false,
);
const openVote = (i: number, event: Event) => {
event.stopPropagation();
2024-11-23 22:21:14 +08:00
if (poll().expired || poll().voted) {
2024-11-23 20:55:37 +08:00
return;
}
batch(() => {
setInitialVote(i);
setShowVoteDialog(true);
});
};
const animateAndSetMustShow = (event: Event) => {
event.stopPropagation();
list.animate(
{
opacity: [0.5, 0, 0.5],
},
{
duration: 220,
easing: ANIM_CURVE_STD,
},
);
setMustShowResult((x) => {
if (typeof x === "undefined") {
return !untrack(isShowResult);
} else {
return undefined;
}
});
};
return (
<section class="TootPoll">
<div class="hints">
2024-11-23 22:21:14 +08:00
<span>{poll().votesCount} votes in total</span>
<Show when={poll().expired}>
2024-11-23 20:55:37 +08:00
<span>Poll is ended</span>
</Show>
</div>
<List ref={list!} disablePadding class="option-list">
2024-11-23 22:21:14 +08:00
<Index each={poll().options}>
2024-11-23 20:55:37 +08:00
{(option, index) => {
return (
<>
<Show when={index === 0}>
<Divider />
</Show>
<ListItemButton
onClick={[openVote, index]}
class="poll-item"
aria-disabled={isShowResult()}
>
<ListItemText>
<span
2024-11-24 17:16:06 +08:00
innerHTML={resolveCustomEmoji(
option().title,
option().emojis,
)}
2024-11-23 20:55:37 +08:00
></span>
</ListItemText>
<Show when={isShowResult()}>
<span>
<Show when={typeof option().votesCount !== "undefined"}>
{option().votesCount} votes
</Show>
</span>
</Show>
<Show
2024-11-23 22:21:14 +08:00
when={poll().multiple}
2024-11-23 20:55:37 +08:00
fallback={
<Radio
checked={isOwnVote(index)}
disabled={isShowResult()}
/>
}
>
<Checkbox
checked={isOwnVote(index)}
disabled={isShowResult()}
/>
</Show>
</ListItemButton>
<Divider />
</>
);
}}
</Index>
</List>
<div class="trailers">
<Button onClick={animateAndSetMustShow}>
{isShowResult() ? "Hide result" : "Reveal result"}
</Button>
2024-11-23 22:21:14 +08:00
<Show when={poll().expiresAt}>
2024-11-23 20:55:37 +08:00
<span>
<span style={{ "margin-inline-end": "0.5ch" }}>
2024-11-23 22:21:14 +08:00
{isBefore(now(), poll().expiresAt!) ? "Expire in" : "Expired"}
2024-11-23 20:55:37 +08:00
</span>
2024-11-23 22:21:14 +08:00
<time dateTime={poll().expiresAt!}>
{formatDistance(now(), poll().expiresAt!, {
2024-11-23 20:55:37 +08:00
locale: dateFnLocale(),
})}
</time>
</span>
</Show>
</div>
<TootPollDialog
open={showVoteDialog()}
2024-11-23 22:21:14 +08:00
options={poll().options}
2025-01-04 20:58:42 +08:00
multiple={poll().multiple}
2024-11-23 22:21:14 +08:00
onVote={[vote, props.status]}
2024-11-23 20:55:37 +08:00
onClose={() => setShowVoteDialog(false)}
initialVotes={[initialVote()]}
/>
</section>
);
};
export default TootPoll;