RegularToot: refactor, add env context
This commit is contained in:
parent
cbdf5e667d
commit
ad7db8e865
5 changed files with 164 additions and 123 deletions
|
@ -7,6 +7,8 @@ import {
|
||||||
createRenderEffect,
|
createRenderEffect,
|
||||||
createSignal,
|
createSignal,
|
||||||
type Setter,
|
type Setter,
|
||||||
|
createContext,
|
||||||
|
useContext,
|
||||||
} from "solid-js";
|
} from "solid-js";
|
||||||
import tootStyle from "./toot.module.css";
|
import tootStyle from "./toot.module.css";
|
||||||
import { formatRelative, parseISO } from "date-fns";
|
import { formatRelative, parseISO } from "date-fns";
|
||||||
|
@ -38,27 +40,38 @@ import BoostIcon from "./toots/BoostIcon";
|
||||||
import PreviewCard from "./toots/PreviewCard";
|
import PreviewCard from "./toots/PreviewCard";
|
||||||
import TootPoll from "./toots/TootPoll";
|
import TootPoll from "./toots/TootPoll";
|
||||||
|
|
||||||
type TootActionGroupProps<T extends mastodon.v1.Status> = {
|
export type TootEnv = {
|
||||||
onRetoot?: (value: T) => void;
|
boost: (value: mastodon.v1.Status) => void;
|
||||||
onFavourite?: (value: T) => void;
|
favourite: (value: mastodon.v1.Status) => void;
|
||||||
onBookmark?: (value: T) => void;
|
bookmark: (value: mastodon.v1.Status) => void;
|
||||||
onReply?: (
|
reply?: (
|
||||||
value: T,
|
value: mastodon.v1.Status,
|
||||||
event: MouseEvent & { currentTarget: HTMLButtonElement },
|
event: MouseEvent & { currentTarget: HTMLButtonElement },
|
||||||
) => void;
|
) => void;
|
||||||
|
vote: (
|
||||||
|
status: mastodon.v1.Status,
|
||||||
|
votes: readonly number[],
|
||||||
|
) => void | Promise<void>;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
const TootEnvContext = /* @__PURE__ */ createContext<TootEnv>();
|
||||||
|
|
||||||
|
export const TootEnvProvider = TootEnvContext.Provider;
|
||||||
|
|
||||||
|
export function useTootEnv() {
|
||||||
|
const env = useContext(TootEnvContext);
|
||||||
|
if (!env) {
|
||||||
|
throw new TypeError("environment not found, use TootEnvProvider to provide")
|
||||||
|
}
|
||||||
|
return env
|
||||||
|
}
|
||||||
|
|
||||||
type RegularTootProps = {
|
type RegularTootProps = {
|
||||||
status: mastodon.v1.Status;
|
status: mastodon.v1.Status;
|
||||||
actionable?: boolean;
|
actionable?: boolean;
|
||||||
evaluated?: boolean;
|
evaluated?: boolean;
|
||||||
thread?: "top" | "bottom" | "middle";
|
thread?: "top" | "bottom" | "middle";
|
||||||
|
} &
|
||||||
onVote?: (value: {
|
|
||||||
status: mastodon.v1.Status;
|
|
||||||
votes: readonly number[];
|
|
||||||
}) => void | Promise<void>;
|
|
||||||
} & TootActionGroupProps<mastodon.v1.Status> &
|
|
||||||
JSX.HTMLElementTags["article"];
|
JSX.HTMLElementTags["article"];
|
||||||
|
|
||||||
function isolatedCallback(e: MouseEvent) {
|
function isolatedCallback(e: MouseEvent) {
|
||||||
|
@ -79,8 +92,9 @@ export function findRootToot(element: HTMLElement) {
|
||||||
}
|
}
|
||||||
|
|
||||||
function TootActionGroup<T extends mastodon.v1.Status>(
|
function TootActionGroup<T extends mastodon.v1.Status>(
|
||||||
props: TootActionGroupProps<T> & { value: T },
|
props: { value: T },
|
||||||
) {
|
) {
|
||||||
|
const {reply, boost, favourite, bookmark} = useTootEnv()
|
||||||
let actGrpElement: HTMLDivElement;
|
let actGrpElement: HTMLDivElement;
|
||||||
const toot = () => props.value;
|
const toot = () => props.value;
|
||||||
return (
|
return (
|
||||||
|
@ -89,10 +103,10 @@ function TootActionGroup<T extends mastodon.v1.Status>(
|
||||||
class={tootStyle.tootBottomActionGrp}
|
class={tootStyle.tootBottomActionGrp}
|
||||||
onClick={isolatedCallback}
|
onClick={isolatedCallback}
|
||||||
>
|
>
|
||||||
<Show when={props.onReply}>
|
<Show when={reply}>
|
||||||
<Button
|
<Button
|
||||||
class={tootStyle.tootActionWithCount}
|
class={tootStyle.tootActionWithCount}
|
||||||
onClick={[props.onReply!, props.value]}
|
onClick={[reply!, props.value]}
|
||||||
>
|
>
|
||||||
<ReplyAll />
|
<ReplyAll />
|
||||||
<span>{toot().repliesCount}</span>
|
<span>{toot().repliesCount}</span>
|
||||||
|
@ -104,7 +118,7 @@ function TootActionGroup<T extends mastodon.v1.Status>(
|
||||||
style={{
|
style={{
|
||||||
color: toot().reblogged ? "var(--tutu-color-primary)" : undefined,
|
color: toot().reblogged ? "var(--tutu-color-primary)" : undefined,
|
||||||
}}
|
}}
|
||||||
onClick={() => props.onRetoot?.(toot())}
|
onClick={[boost, props.value]}
|
||||||
>
|
>
|
||||||
<Repeat />
|
<Repeat />
|
||||||
<span>{toot().reblogsCount}</span>
|
<span>{toot().reblogsCount}</span>
|
||||||
|
@ -114,7 +128,7 @@ function TootActionGroup<T extends mastodon.v1.Status>(
|
||||||
style={{
|
style={{
|
||||||
color: toot().favourited ? "var(--tutu-color-primary)" : undefined,
|
color: toot().favourited ? "var(--tutu-color-primary)" : undefined,
|
||||||
}}
|
}}
|
||||||
onClick={() => props.onFavourite?.(toot())}
|
onClick={[favourite, props.value]}
|
||||||
>
|
>
|
||||||
{toot().favourited ? <Star /> : <StarOutline />}
|
{toot().favourited ? <Star /> : <StarOutline />}
|
||||||
<span>{toot().favouritesCount}</span>
|
<span>{toot().favouritesCount}</span>
|
||||||
|
@ -124,7 +138,7 @@ function TootActionGroup<T extends mastodon.v1.Status>(
|
||||||
style={{
|
style={{
|
||||||
color: toot().bookmarked ? "var(--tutu-color-primary)" : undefined,
|
color: toot().bookmarked ? "var(--tutu-color-primary)" : undefined,
|
||||||
}}
|
}}
|
||||||
onClick={() => props.onBookmark?.(toot())}
|
onClick={[bookmark, props.value]}
|
||||||
>
|
>
|
||||||
{toot().bookmarked ? <Bookmark /> : <BookmarkAddOutlined />}
|
{toot().bookmarked ? <Bookmark /> : <BookmarkAddOutlined />}
|
||||||
</Button>
|
</Button>
|
||||||
|
@ -243,11 +257,10 @@ function onToggleReveal(setValue: Setter<boolean>, event: Event) {
|
||||||
*/
|
*/
|
||||||
const RegularToot: Component<RegularTootProps> = (props) => {
|
const RegularToot: Component<RegularTootProps> = (props) => {
|
||||||
let rootRef: HTMLElement;
|
let rootRef: HTMLElement;
|
||||||
const [managed, managedActionGroup, pollProps, rest] = splitProps(
|
const {vote} = useTootEnv()
|
||||||
|
const [managed, rest] = splitProps(
|
||||||
props,
|
props,
|
||||||
["status", "lang", "class", "actionable", "evaluated", "thread"],
|
["status", "lang", "class", "actionable", "evaluated", "thread"],
|
||||||
["onRetoot", "onFavourite", "onBookmark", "onReply"],
|
|
||||||
["onVote"],
|
|
||||||
);
|
);
|
||||||
const now = useTimeSource();
|
const now = useTimeSource();
|
||||||
const status = () => managed.status;
|
const status = () => managed.status;
|
||||||
|
@ -361,18 +374,8 @@ const RegularToot: Component<RegularTootProps> = (props) => {
|
||||||
</Show>
|
</Show>
|
||||||
<Show when={toot().poll}>
|
<Show when={toot().poll}>
|
||||||
<TootPoll
|
<TootPoll
|
||||||
options={toot().poll!.options}
|
value={toot().poll!}
|
||||||
multiple={toot().poll!.multiple}
|
status={toot()}
|
||||||
votesCount={toot().poll!.votesCount}
|
|
||||||
expired={toot().poll!.expired}
|
|
||||||
expiredAt={
|
|
||||||
toot().poll!.expiresAt
|
|
||||||
? parseISO(toot().poll!.expiresAt!)
|
|
||||||
: undefined
|
|
||||||
}
|
|
||||||
voted={toot().poll!.voted}
|
|
||||||
ownVotes={toot().poll!.ownVotes || undefined}
|
|
||||||
onVote={(votes) => pollProps.onVote?.({ status: status(), votes })}
|
|
||||||
/>
|
/>
|
||||||
</Show>
|
</Show>
|
||||||
<Show when={managed.actionable}>
|
<Show when={managed.actionable}>
|
||||||
|
@ -380,7 +383,7 @@ const RegularToot: Component<RegularTootProps> = (props) => {
|
||||||
class={cardStyle.cardNoPad}
|
class={cardStyle.cardNoPad}
|
||||||
style={{ "margin-top": "8px" }}
|
style={{ "margin-top": "8px" }}
|
||||||
/>
|
/>
|
||||||
<TootActionGroup value={toot()} {...managedActionGroup} />
|
<TootActionGroup value={toot()} />
|
||||||
</Show>
|
</Show>
|
||||||
</article>
|
</article>
|
||||||
</>
|
</>
|
||||||
|
|
|
@ -13,7 +13,10 @@ import { Title } from "~material/typography";
|
||||||
import { Close as CloseIcon } from "@suid/icons-material";
|
import { Close as CloseIcon } from "@suid/icons-material";
|
||||||
import { useSessionForAcctStr } from "../masto/clients";
|
import { useSessionForAcctStr } from "../masto/clients";
|
||||||
import { resolveCustomEmoji } from "../masto/toot";
|
import { resolveCustomEmoji } from "../masto/toot";
|
||||||
import RegularToot, { findElementActionable } from "./RegularToot";
|
import RegularToot, {
|
||||||
|
findElementActionable,
|
||||||
|
TootEnvProvider,
|
||||||
|
} from "./RegularToot";
|
||||||
import type { mastodon } from "masto";
|
import type { mastodon } from "masto";
|
||||||
import cards from "~material/cards.module.css";
|
import cards from "~material/cards.module.css";
|
||||||
import { css } from "solid-styled";
|
import { css } from "solid-styled";
|
||||||
|
@ -169,6 +172,33 @@ const TootBottomSheet: Component = (props) => {
|
||||||
return Array.from(new Set(values).keys());
|
return Array.from(new Set(values).keys());
|
||||||
};
|
};
|
||||||
|
|
||||||
|
const vote = async (status: mastodon.v1.Status, votes: readonly number[]) => {
|
||||||
|
const client = session()?.client;
|
||||||
|
if (!client) return;
|
||||||
|
|
||||||
|
const toot = status.reblog ?? status;
|
||||||
|
if (!toot.poll) return;
|
||||||
|
|
||||||
|
const npoll = await client.v1.polls.$select(toot.poll.id).votes.create({
|
||||||
|
choices: votes,
|
||||||
|
});
|
||||||
|
|
||||||
|
if (status.reblog) {
|
||||||
|
setRemoteToot({
|
||||||
|
...status,
|
||||||
|
reblog: {
|
||||||
|
...status.reblog,
|
||||||
|
poll: npoll,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
setRemoteToot({
|
||||||
|
...status,
|
||||||
|
poll: npoll,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
const handleMainTootClick = (
|
const handleMainTootClick = (
|
||||||
event: MouseEvent & { currentTarget: HTMLElement },
|
event: MouseEvent & { currentTarget: HTMLElement },
|
||||||
) => {
|
) => {
|
||||||
|
@ -255,23 +285,29 @@ const TootBottomSheet: Component = (props) => {
|
||||||
|
|
||||||
<article>
|
<article>
|
||||||
<Show when={toot()}>
|
<Show when={toot()}>
|
||||||
<RegularToot
|
<TootEnvProvider
|
||||||
id={`toot-${toot()!.id}`}
|
value={{
|
||||||
class={cards.card}
|
bookmark: onBookmark,
|
||||||
style={{
|
boost: onBoost,
|
||||||
"scroll-margin-top":
|
favourite: onFav,
|
||||||
"calc(var(--scaffold-topbar-height) + 20px)",
|
vote,
|
||||||
"cursor": "auto",
|
|
||||||
"user-select": "auto",
|
|
||||||
}}
|
}}
|
||||||
status={toot()!}
|
>
|
||||||
actionable={!!actSession()}
|
<RegularToot
|
||||||
evaluated={true}
|
id={`toot-${toot()!.id}`}
|
||||||
onBookmark={onBookmark}
|
class={cards.card}
|
||||||
onRetoot={onBoost}
|
style={{
|
||||||
onFavourite={onFav}
|
"scroll-margin-top":
|
||||||
onClick={handleMainTootClick}
|
"calc(var(--scaffold-topbar-height) + 20px)",
|
||||||
></RegularToot>
|
cursor: "auto",
|
||||||
|
"user-select": "auto",
|
||||||
|
}}
|
||||||
|
status={toot()!}
|
||||||
|
actionable={!!actSession()}
|
||||||
|
evaluated={true}
|
||||||
|
onClick={handleMainTootClick}
|
||||||
|
></RegularToot>
|
||||||
|
</TootEnvProvider>
|
||||||
</Show>
|
</Show>
|
||||||
</article>
|
</article>
|
||||||
|
|
||||||
|
|
|
@ -14,6 +14,7 @@ import { setCache as setTootBottomSheetCache } from "./TootBottomSheet";
|
||||||
import RegularToot, {
|
import RegularToot, {
|
||||||
findElementActionable,
|
findElementActionable,
|
||||||
findRootToot,
|
findRootToot,
|
||||||
|
TootEnvProvider,
|
||||||
} from "./RegularToot";
|
} from "./RegularToot";
|
||||||
import cardStyle from "~material/cards.module.css";
|
import cardStyle from "~material/cards.module.css";
|
||||||
import type { ThreadNode } from "../masto/timelines";
|
import type { ThreadNode } from "../masto/timelines";
|
||||||
|
@ -232,13 +233,10 @@ const TootList: Component<{
|
||||||
openFullScreenToot(status, element, true);
|
openFullScreenToot(status, element, true);
|
||||||
};
|
};
|
||||||
|
|
||||||
const vote = async ({
|
const vote = async (
|
||||||
status,
|
status: mastodon.v1.Status,
|
||||||
votes,
|
votes: readonly number[]
|
||||||
}: {
|
) => {
|
||||||
status: mastodon.v1.Status;
|
|
||||||
votes: readonly number[];
|
|
||||||
}) => {
|
|
||||||
const client = session()?.client;
|
const client = session()?.client;
|
||||||
if (!client) return;
|
if (!client) return;
|
||||||
|
|
||||||
|
@ -263,7 +261,6 @@ const TootList: Component<{
|
||||||
poll: npoll,
|
poll: npoll,
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
};
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
|
@ -273,49 +270,50 @@ const TootList: Component<{
|
||||||
return <p>Oops: {String(err)}</p>;
|
return <p>Oops: {String(err)}</p>;
|
||||||
}}
|
}}
|
||||||
>
|
>
|
||||||
<div ref={props.ref} id={props.id} class="toot-list">
|
<TootEnvProvider value={{
|
||||||
<Index each={props.threads}>
|
boost: toggleBoost,
|
||||||
{(threadId, threadIdx) => {
|
bookmark: onBookmark,
|
||||||
const thread = createMemo(() =>
|
favourite: toggleFavourite,
|
||||||
props.onUnknownThread(threadId())?.reverse(),
|
reply: reply,
|
||||||
);
|
vote: vote
|
||||||
|
}}>
|
||||||
|
<div ref={props.ref} id={props.id} class="toot-list">
|
||||||
|
<Index each={props.threads}>
|
||||||
|
{(threadId, threadIdx) => {
|
||||||
|
const thread = createMemo(() =>
|
||||||
|
props.onUnknownThread(threadId())?.reverse(),
|
||||||
|
);
|
||||||
|
|
||||||
const threadLength = () => thread()?.length ?? 0;
|
const threadLength = () => thread()?.length ?? 0;
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Index each={thread()}>
|
<Index each={thread()}>
|
||||||
{(threadNode, index) => {
|
{(threadNode, index) => {
|
||||||
const status = () => threadNode().value;
|
const status = () => threadNode().value;
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<RegularToot
|
<RegularToot
|
||||||
data-status-id={status().id}
|
data-status-id={status().id}
|
||||||
data-thread={threadIdx}
|
data-thread-sort={index}
|
||||||
data-thread-len={threadLength()}
|
status={status()}
|
||||||
data-thread-sort={index}
|
thread={
|
||||||
status={status()}
|
threadLength() > 1
|
||||||
thread={
|
? positionTootInThread(index, threadLength())
|
||||||
threadLength() > 1
|
: undefined
|
||||||
? positionTootInThread(index, threadLength())
|
}
|
||||||
: undefined
|
class={cardStyle.card}
|
||||||
}
|
evaluated={checkIsExpended(status())}
|
||||||
class={cardStyle.card}
|
actionable={checkIsExpended(status())}
|
||||||
evaluated={checkIsExpended(status())}
|
onClick={[onItemClick, status()]}
|
||||||
actionable={checkIsExpended(status())}
|
/>
|
||||||
onBookmark={onBookmark}
|
);
|
||||||
onRetoot={toggleBoost}
|
}}
|
||||||
onFavourite={toggleFavourite}
|
</Index>
|
||||||
onReply={reply}
|
);
|
||||||
onVote={vote}
|
}}
|
||||||
onClick={[onItemClick, status()]}
|
</Index>
|
||||||
/>
|
</div>
|
||||||
);
|
</TootEnvProvider>
|
||||||
}}
|
|
||||||
</Index>
|
|
||||||
);
|
|
||||||
}}
|
|
||||||
</Index>
|
|
||||||
</div>
|
|
||||||
</ErrorBoundary>
|
</ErrorBoundary>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
|
@ -28,21 +28,17 @@ import { useTimeSource } from "~platform/timesrc";
|
||||||
import { useDateFnLocale } from "~platform/i18n";
|
import { useDateFnLocale } from "~platform/i18n";
|
||||||
import TootPollDialog from "./TootPollDialog";
|
import TootPollDialog from "./TootPollDialog";
|
||||||
import { ANIM_CURVE_STD } from "~material/theme";
|
import { ANIM_CURVE_STD } from "~material/theme";
|
||||||
|
import { useTootEnv } from "../RegularToot";
|
||||||
|
|
||||||
type TootPollProps = {
|
type TootPollProps = {
|
||||||
options: Readonly<mastodon.v1.Poll["options"]>;
|
value: mastodon.v1.Poll
|
||||||
multiple?: boolean;
|
status: mastodon.v1.Status
|
||||||
votesCount: number;
|
|
||||||
expired?: boolean;
|
|
||||||
expiredAt?: Date;
|
|
||||||
voted?: boolean;
|
|
||||||
ownVotes?: readonly number[];
|
|
||||||
|
|
||||||
onVote(votes: readonly number[]): void | Promise<void>;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
const TootPoll: Component<TootPollProps> = (props) => {
|
const TootPoll: Component<TootPollProps> = (props) => {
|
||||||
let list: HTMLUListElement;
|
let list: HTMLUListElement;
|
||||||
|
const {vote}= useTootEnv()
|
||||||
|
|
||||||
const now = useTimeSource();
|
const now = useTimeSource();
|
||||||
const dateFnLocale = useDateFnLocale();
|
const dateFnLocale = useDateFnLocale();
|
||||||
const [mustShowResult, setMustShowResult] = createSignal<boolean>();
|
const [mustShowResult, setMustShowResult] = createSignal<boolean>();
|
||||||
|
@ -50,24 +46,26 @@ const TootPoll: Component<TootPollProps> = (props) => {
|
||||||
|
|
||||||
const [initialVote, setInitialVote] = createSignal(0);
|
const [initialVote, setInitialVote] = createSignal(0);
|
||||||
|
|
||||||
|
const poll = () => props.value
|
||||||
|
|
||||||
const isShowResult = () => {
|
const isShowResult = () => {
|
||||||
const n = mustShowResult();
|
const n = mustShowResult();
|
||||||
if (typeof n !== "undefined") {
|
if (typeof n !== "undefined") {
|
||||||
return n;
|
return n;
|
||||||
}
|
}
|
||||||
|
|
||||||
return props.expired || props.voted;
|
return poll().expired || poll().voted;
|
||||||
};
|
};
|
||||||
|
|
||||||
const isOwnVote = createSelector(
|
const isOwnVote = createSelector(
|
||||||
() => props.ownVotes,
|
() => poll().ownVotes,
|
||||||
(idx: number, votes) => votes?.includes(idx) || false,
|
(idx: number, votes) => votes?.includes(idx) || false,
|
||||||
);
|
);
|
||||||
|
|
||||||
const openVote = (i: number, event: Event) => {
|
const openVote = (i: number, event: Event) => {
|
||||||
event.stopPropagation();
|
event.stopPropagation();
|
||||||
|
|
||||||
if (props.expired || props.voted) {
|
if (poll().expired || poll().voted) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -100,13 +98,13 @@ const TootPoll: Component<TootPollProps> = (props) => {
|
||||||
return (
|
return (
|
||||||
<section class="TootPoll">
|
<section class="TootPoll">
|
||||||
<div class="hints">
|
<div class="hints">
|
||||||
<span>{props.votesCount} votes in total</span>
|
<span>{poll().votesCount} votes in total</span>
|
||||||
<Show when={props.expired}>
|
<Show when={poll().expired}>
|
||||||
<span>Poll is ended</span>
|
<span>Poll is ended</span>
|
||||||
</Show>
|
</Show>
|
||||||
</div>
|
</div>
|
||||||
<List ref={list!} disablePadding class="option-list">
|
<List ref={list!} disablePadding class="option-list">
|
||||||
<Index each={props.options}>
|
<Index each={poll().options}>
|
||||||
{(option, index) => {
|
{(option, index) => {
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
|
@ -140,7 +138,7 @@ const TootPoll: Component<TootPollProps> = (props) => {
|
||||||
</Show>
|
</Show>
|
||||||
|
|
||||||
<Show
|
<Show
|
||||||
when={props.multiple}
|
when={poll().multiple}
|
||||||
fallback={
|
fallback={
|
||||||
<Radio
|
<Radio
|
||||||
checked={isOwnVote(index)}
|
checked={isOwnVote(index)}
|
||||||
|
@ -164,14 +162,14 @@ const TootPoll: Component<TootPollProps> = (props) => {
|
||||||
<Button onClick={animateAndSetMustShow}>
|
<Button onClick={animateAndSetMustShow}>
|
||||||
{isShowResult() ? "Hide result" : "Reveal result"}
|
{isShowResult() ? "Hide result" : "Reveal result"}
|
||||||
</Button>
|
</Button>
|
||||||
<Show when={props.expiredAt}>
|
<Show when={poll().expiresAt}>
|
||||||
<span>
|
<span>
|
||||||
<span style={{ "margin-inline-end": "0.5ch" }}>
|
<span style={{ "margin-inline-end": "0.5ch" }}>
|
||||||
{isBefore(now(), props.expiredAt!) ? "Expire in" : "Expired"}
|
{isBefore(now(), poll().expiresAt!) ? "Expire in" : "Expired"}
|
||||||
</span>
|
</span>
|
||||||
|
|
||||||
<time dateTime={props.expiredAt!.toISOString()}>
|
<time dateTime={poll().expiresAt!}>
|
||||||
{formatDistance(now(), props.expiredAt!, {
|
{formatDistance(now(), poll().expiresAt!, {
|
||||||
locale: dateFnLocale(),
|
locale: dateFnLocale(),
|
||||||
})}
|
})}
|
||||||
</time>
|
</time>
|
||||||
|
@ -181,8 +179,8 @@ const TootPoll: Component<TootPollProps> = (props) => {
|
||||||
|
|
||||||
<TootPollDialog
|
<TootPollDialog
|
||||||
open={showVoteDialog()}
|
open={showVoteDialog()}
|
||||||
options={props.options}
|
options={poll().options}
|
||||||
onVote={(votes) => console.debug(votes)}
|
onVote={[vote, props.status]}
|
||||||
onClose={() => setShowVoteDialog(false)}
|
onClose={() => setShowVoteDialog(false)}
|
||||||
initialVotes={[initialVote()]}
|
initialVotes={[initialVote()]}
|
||||||
/>
|
/>
|
||||||
|
|
|
@ -26,7 +26,13 @@ export type TootPollDialogPoll = {
|
||||||
initialVotes?: readonly number[];
|
initialVotes?: readonly number[];
|
||||||
multiple?: boolean;
|
multiple?: boolean;
|
||||||
|
|
||||||
onVote(votes: readonly number[]): void | Promise<void>;
|
onVote: [
|
||||||
|
(
|
||||||
|
status: mastodon.v1.Status,
|
||||||
|
votes: readonly number[],
|
||||||
|
) => void | Promise<void>,
|
||||||
|
mastodon.v1.Status,
|
||||||
|
];
|
||||||
onClose?: BottomSheetProps["onClose"] &
|
onClose?: BottomSheetProps["onClose"] &
|
||||||
((reason: "cancel" | "success") => void);
|
((reason: "cancel" | "success") => void);
|
||||||
};
|
};
|
||||||
|
@ -50,7 +56,7 @@ const TootPollDialog: Component<TootPollDialogPoll> = (props) => {
|
||||||
const sendVote = async () => {
|
const sendVote = async () => {
|
||||||
setInProgress(true);
|
setInProgress(true);
|
||||||
try {
|
try {
|
||||||
await props.onVote(votes());
|
await props.onVote[0](props.onVote[1], votes());
|
||||||
} catch (reason) {
|
} catch (reason) {
|
||||||
console.error(reason);
|
console.error(reason);
|
||||||
props.onClose?.("cancel");
|
props.onClose?.("cancel");
|
||||||
|
|
Loading…
Reference in a new issue