Compare commits

...

5 commits

Author SHA1 Message Date
thislight
4c717a0cb7
TootComposer: use useAppLocale
All checks were successful
/ depoly (push) Successful in 1m25s
2024-11-23 23:48:22 +08:00
thislight
6895367fad
RegularToot: remove css module 2024-11-23 23:46:18 +08:00
thislight
9fe86d12b0
i18n: optimize performance 2024-11-23 23:39:52 +08:00
thislight
296de7d23b
TootActionGroup: remove css module 2024-11-23 23:04:35 +08:00
thislight
ad7db8e865
RegularToot: refactor, add env context 2024-11-23 22:22:55 +08:00
13 changed files with 454 additions and 522 deletions

View file

@ -1,4 +1,4 @@
import { Route, Router } from "@solidjs/router"; import { Route } from "@solidjs/router";
import { ThemeProvider } from "@suid/material"; import { ThemeProvider } from "@suid/material";
import { import {
Component, Component,
@ -17,7 +17,12 @@ import {
} from "./masto/clients.js"; } from "./masto/clients.js";
import { $accounts, updateAcctInf } from "./accounts/stores.js"; import { $accounts, updateAcctInf } from "./accounts/stores.js";
import { useStore } from "@nanostores/solid"; import { useStore } from "@nanostores/solid";
import { DateFnScope, useLanguage } from "./platform/i18n.jsx"; import {
AppLocaleProvider,
createCurrentLanguage,
createCurrentRegion,
createDateFnLocaleResource,
} from "./platform/i18n.jsx";
import { useRegisterSW } from "virtual:pwa-register/solid"; import { useRegisterSW } from "virtual:pwa-register/solid";
import { import {
isJSONRPCResult, isJSONRPCResult,
@ -67,7 +72,9 @@ const Routing: Component = () => {
const App: Component = () => { const App: Component = () => {
const theme = useRootTheme(); const theme = useRootTheme();
const accts = useStore($accounts); const accts = useStore($accounts);
const lang = useLanguage(); const lang = createCurrentLanguage();
const region = createCurrentRegion();
const dateFnLocale = createDateFnLocaleResource(region);
const [serviceWorker, setServiceWorker] = createSignal< const [serviceWorker, setServiceWorker] = createSignal<
ServiceWorker | undefined ServiceWorker | undefined
>(undefined, { name: "serviceWorker" }); >(undefined, { name: "serviceWorker" });
@ -150,7 +157,13 @@ const App: Component = () => {
}} }}
> >
<ThemeProvider theme={theme}> <ThemeProvider theme={theme}>
<DateFnScope> <AppLocaleProvider
value={{
language: lang,
region: region,
dateFn: dateFnLocale,
}}
>
<ClientProvider value={clients}> <ClientProvider value={clients}>
<ServiceWorkerProvider <ServiceWorkerProvider
value={{ value={{
@ -162,7 +175,7 @@ const App: Component = () => {
<Routing /> <Routing />
</ServiceWorkerProvider> </ServiceWorkerProvider>
</ClientProvider> </ClientProvider>
</DateFnScope> </AppLocaleProvider>
</ThemeProvider> </ThemeProvider>
</ErrorBoundary> </ErrorBoundary>
); );

View file

@ -1,12 +1,12 @@
import { import {
ParentComponent, catchError,
createContext, createContext,
createMemo, createMemo,
createResource, createResource,
useContext, useContext,
} from "solid-js"; } from "solid-js";
import { match } from "@formatjs/intl-localematcher"; import { match } from "@formatjs/intl-localematcher";
import { Accessor, createEffect, createSignal } from "solid-js"; import { Accessor } from "solid-js";
import { $settings } from "../settings/stores"; import { $settings } from "../settings/stores";
import { enGB } from "date-fns/locale/en-GB"; import { enGB } from "date-fns/locale/en-GB";
import { useStore } from "@nanostores/solid"; import { useStore } from "@nanostores/solid";
@ -17,13 +17,6 @@ import {
type Template, type Template,
} from "@solid-primitives/i18n"; } from "@solid-primitives/i18n";
async function synchronised(
name: string,
callback: () => Promise<void> | void,
): Promise<void> {
await navigator.locks.request(name, callback);
}
export const SUPPORTED_LANGS = ["en", "zh-Hans"] as const; export const SUPPORTED_LANGS = ["en", "zh-Hans"] as const;
export const SUPPORTED_REGIONS = ["en_US", "en_GB", "zh_CN"] as const; export const SUPPORTED_REGIONS = ["en_US", "en_GB", "zh_CN"] as const;
@ -38,14 +31,6 @@ export function autoMatchLangTag() {
return match(Array.from(navigator.languages), SUPPORTED_LANGS, DEFAULT_LANG); return match(Array.from(navigator.languages), SUPPORTED_LANGS, DEFAULT_LANG);
} }
const DateFnLocaleCx = /* __@PURE__ */ createContext<Accessor<Locale>>(
() => enGB,
);
const cachedDateFnLocale: Record<string, Locale> = {
enGB,
};
export function autoMatchRegion() { export function autoMatchRegion() {
const specifiers = navigator.languages.map((x) => x.split("-")); const specifiers = navigator.languages.map((x) => x.split("-"));
@ -70,7 +55,7 @@ export function autoMatchRegion() {
return "en_GB"; return "en_GB";
} }
export function useRegion() { export function createCurrentRegion() {
const appSettings = useStore($settings); const appSettings = useStore($settings);
return createMemo( return createMemo(
@ -100,53 +85,6 @@ async function importDateFnLocale(tag: string): Promise<Locale> {
} }
} }
/**
* Provides runtime values and fetch dependencies for date-fns locale
*/
export const DateFnScope: ParentComponent = (props) => {
const [dateFnLocale, setDateFnLocale] = createSignal(enGB, {
name: "dateFnLocale",
});
const region = useRegion();
createEffect(() => {
const dateFnLocaleName = region();
if (cachedDateFnLocale[dateFnLocaleName]) {
setDateFnLocale(cachedDateFnLocale[dateFnLocaleName]);
} else {
synchronised("i18n-wrapper-load-date-fns-locale", async () => {
if (cachedDateFnLocale[dateFnLocaleName]) {
setDateFnLocale(cachedDateFnLocale[dateFnLocaleName]);
return;
}
const target = `date-fns/locale/${dateFnLocaleName}`;
try {
const mod = await importDateFnLocale(dateFnLocaleName);
cachedDateFnLocale[dateFnLocaleName] = mod;
setDateFnLocale(mod);
} catch (reason) {
console.error(
{
act: "load-date-fns-locale",
stat: "failed",
reason,
target,
},
"failed to load date-fns locale",
);
}
});
}
});
return (
<DateFnLocaleCx.Provider value={dateFnLocale}>
{props.children}
</DateFnLocaleCx.Provider>
);
};
/** /**
* Get the {@link Locale} object for date-fns. * Get the {@link Locale} object for date-fns.
* *
@ -155,11 +93,11 @@ export const DateFnScope: ParentComponent = (props) => {
* @returns Accessor for Locale * @returns Accessor for Locale
*/ */
export function useDateFnLocale(): Accessor<Locale> { export function useDateFnLocale(): Accessor<Locale> {
const cx = useContext(DateFnLocaleCx); const { dateFn } = useAppLocale();
return cx; return dateFn;
} }
export function useLanguage() { export function createCurrentLanguage() {
const settings = useStore($settings); const settings = useStore($settings);
return () => settings().language || autoMatchLangTag(); return () => settings().language || autoMatchLangTag();
} }
@ -179,7 +117,7 @@ type MergedImportedModule<T> = T extends []
export function createStringResource< export function createStringResource<
T extends ImportFn<Record<string, string | Template<any> | undefined>>[], T extends ImportFn<Record<string, string | Template<any> | undefined>>[],
>(...importFns: T) { >(...importFns: T) {
const language = useLanguage(); // TODO: this function costs to much, provide a global cache const language = createCurrentLanguage();
const cache: Record<string, MergedImportedModule<T>> = {}; const cache: Record<string, MergedImportedModule<T>> = {};
return createResource( return createResource(
@ -209,3 +147,38 @@ export function createTranslator<
return [translator(res[0], resolveTemplate), res] as const; return [translator(res[0], resolveTemplate), res] as const;
} }
export type AppLocale = {
dateFn: () => Locale;
language: () => string;
region: () => string;
};
const AppLocaleContext = /* @__PURE__ */ createContext<AppLocale>();
export const AppLocaleProvider = AppLocaleContext.Provider;
export function useAppLocale() {
const l = useContext(AppLocaleContext);
if (!l) {
throw new TypeError("app locale not found");
}
return l;
}
export function createDateFnLocaleResource(region: () => string) {
const [localeUncaught] = createResource(
region,
async (region) => {
return await importDateFnLocale(region);
},
{ initialValue: enGB },
);
return createMemo(
() =>
catchError(localeUncaught, (reason) => {
console.error("fetch date-fns locale", reason);
}) ?? enGB,
);
}

View file

@ -1,57 +0,0 @@
import type { mastodon } from "masto";
import { Show, type Component } from "solid-js";
import tootStyle from "./toot.module.css";
import { formatRelative } from "date-fns";
import Img from "~material/Img";
import { Body2 } from "~material/typography";
import { appliedCustomEmoji } from "../masto/toot";
import { TootPreviewCard } from "./RegularToot";
type CompactTootProps = {
status: mastodon.v1.Status;
now: Date;
class?: string;
};
const CompactToot: Component<CompactTootProps> = (props) => {
const toot = () => props.status;
return (
<section
class={[tootStyle.toot, tootStyle.compact, props.class || ""].join(" ")}
lang={toot().language || undefined}
>
<Img
src={toot().account.avatar}
class={[tootStyle.tootAvatar].join(" ")}
/>
<div class={[tootStyle.compactAuthorGroup].join(" ")}>
<Body2
ref={(e: { innerHTML: string }) => {
appliedCustomEmoji(
e,
toot().account.displayName,
toot().account.emojis,
);
}}
></Body2>
<span class={tootStyle.compactAuthorUsername}>
@{toot().account.username}@{new URL(toot().account.url).hostname}
</span>
<time datetime={toot().createdAt}>
{formatRelative(props.now, toot().createdAt)}
</time>
</div>
<div
ref={(e: { innerHTML: string }) => {
appliedCustomEmoji(e, toot().content, toot().emojis);
}}
class={[tootStyle.compactTootContent].join(" ")}
></div>
<Show when={toot().card}>
<TootPreviewCard src={toot().card!} alwaysCompact />
</Show>
</section>
);
};
export default CompactToot;

View file

@ -0,0 +1,78 @@
.RegularToot {
--card-pad: 16px;
--card-gut: 16px;
--toot-avatar-size: 40px;
margin-block: 0;
position: relative;
contain: layout style;
cursor: pointer;
transition:
margin-top 60ms var(--tutu-anim-curve-sharp),
margin-bottom 60ms var(--tutu-anim-curve-sharp),
height 60ms var(--tutu-anim-curve-sharp),
var(--tutu-transition-shadow);
border-radius: 0;
time {
color: var(--tutu-color-secondary-text-on-surface);
}
>.retoot-grp {
display: flex;
gap: 0.25em;
margin-bottom: 8px;
align-items: center;
> :first-child {
margin-right: 0.25em;
}
}
& .custom-emoji {
height: 1em;
object-fit: contain;
}
&.expanded {
margin-block: 20px;
box-shadow: var(--tutu-shadow-e9);
}
&.thread-top,
&.thread-mid,
&.thread-btm {
position: relative;
&::before {
content: "";
position: absolute;
left: 36px;
background-color: var(--tutu-color-secondary);
width: 2px;
display: block;
}
}
&.thread-mid {
&::before {
top: 0;
bottom: 0;
}
}
&.thread-top {
&::before {
top: 16px;
bottom: 0;
}
}
&.thread-btm {
&::before {
top: 0;
height: 16px;
}
}
}

View file

@ -7,63 +7,62 @@ 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 } from "date-fns";
import Img from "~material/Img.js"; import Img from "~material/Img.js";
import { Body2 } from "~material/typography.js"; import { Body2 } from "~material/typography.js";
import { css } from "solid-styled"; import { SmartToySharp, Lock } from "@suid/icons-material";
import {
BookmarkAddOutlined,
Repeat,
ReplyAll,
Star,
StarOutline,
Bookmark,
Share,
SmartToySharp,
Lock,
} from "@suid/icons-material";
import { useTimeSource } from "~platform/timesrc.js"; import { useTimeSource } from "~platform/timesrc.js";
import { resolveCustomEmoji } from "../masto/toot.js"; import { resolveCustomEmoji } from "../masto/toot.js";
import { Divider } from "@suid/material"; import { Divider } from "@suid/material";
import cardStyle from "~material/cards.module.css"; import cardStyle from "~material/cards.module.css";
import Button from "~material/Button.js";
import MediaAttachmentGrid from "./toots/MediaAttachmentGrid.jsx"; import MediaAttachmentGrid from "./toots/MediaAttachmentGrid.jsx";
import { useDateFnLocale } from "~platform/i18n"; import { useDateFnLocale } from "~platform/i18n";
import { canShare, share } from "~platform/share";
import { makeAcctText, useDefaultSession } from "../masto/clients"; import { makeAcctText, useDefaultSession } from "../masto/clients";
import TootContent from "./toots/TootContent"; import TootContent from "./toots/TootContent";
import BoostIcon from "./toots/BoostIcon"; 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";
import TootActionGroup from "./toots/TootActionGroup.js";
import "./RegularToot.css";
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";
} & JSX.HTMLElementTags["article"];
onVote?: (value: {
status: mastodon.v1.Status;
votes: readonly number[];
}) => void | Promise<void>;
} & TootActionGroupProps<mastodon.v1.Status> &
JSX.HTMLElementTags["article"];
function isolatedCallback(e: MouseEvent) {
e.stopPropagation();
}
export function findRootToot(element: HTMLElement) { export function findRootToot(element: HTMLElement) {
let current: HTMLElement | null = element; let current: HTMLElement | null = element;
@ -78,73 +77,6 @@ export function findRootToot(element: HTMLElement) {
return current; return current;
} }
function TootActionGroup<T extends mastodon.v1.Status>(
props: TootActionGroupProps<T> & { value: T },
) {
let actGrpElement: HTMLDivElement;
const toot = () => props.value;
return (
<div
ref={actGrpElement!}
class={tootStyle.tootBottomActionGrp}
onClick={isolatedCallback}
>
<Show when={props.onReply}>
<Button
class={tootStyle.tootActionWithCount}
onClick={[props.onReply!, props.value]}
>
<ReplyAll />
<span>{toot().repliesCount}</span>
</Button>
</Show>
<Button
class={tootStyle.tootActionWithCount}
style={{
color: toot().reblogged ? "var(--tutu-color-primary)" : undefined,
}}
onClick={() => props.onRetoot?.(toot())}
>
<Repeat />
<span>{toot().reblogsCount}</span>
</Button>
<Button
class={tootStyle.tootActionWithCount}
style={{
color: toot().favourited ? "var(--tutu-color-primary)" : undefined,
}}
onClick={() => props.onFavourite?.(toot())}
>
{toot().favourited ? <Star /> : <StarOutline />}
<span>{toot().favouritesCount}</span>
</Button>
<Button
class={tootStyle.tootAction}
style={{
color: toot().bookmarked ? "var(--tutu-color-primary)" : undefined,
}}
onClick={() => props.onBookmark?.(toot())}
>
{toot().bookmarked ? <Bookmark /> : <BookmarkAddOutlined />}
</Button>
<Show when={canShare({ url: toot().url ?? undefined })}>
<Button
class={tootStyle.tootAction}
aria-label="Share"
onClick={async () => {
await share({
url: toot().url ?? undefined,
});
}}
>
<Share />
</Button>
</Show>
</div>
);
}
function TootAuthorGroup( function TootAuthorGroup(
props: { props: {
status: mastodon.v1.Status; status: mastodon.v1.Status;
@ -220,6 +152,8 @@ function onToggleReveal(setValue: Setter<boolean>, event: Event) {
* this component under a `<DefaultSessionProvier />` with correct * this component under a `<DefaultSessionProvier />` with correct
* session. * session.
* *
* This component requires be under `<TootEnvProvider />`.
*
* **Handling Clicks** * **Handling Clicks**
* There are multiple actions supported in the component. Some handlers * There are multiple actions supported in the component. Some handlers
* are passed in, some should be handled as the click event. * are passed in, some should be handled as the click event.
@ -241,80 +175,39 @@ function onToggleReveal(setValue: Setter<boolean>, event: Event) {
* You can extract the intent from the attributes of the "actionable" element. * You can extract the intent from the attributes of the "actionable" element.
* The action type is the dataset's `action`. * The action type is the dataset's `action`.
*/ */
const RegularToot: Component<RegularTootProps> = (props) => { const RegularToot: Component<RegularTootProps> = (oprops) => {
let rootRef: HTMLElement; let rootRef: HTMLElement;
const [managed, managedActionGroup, pollProps, rest] = splitProps( const [props, rest] = splitProps(oprops, [
props, "status",
["status", "lang", "class", "actionable", "evaluated", "thread"], "lang",
["onRetoot", "onFavourite", "onBookmark", "onReply"], "class",
["onVote"], "actionable",
); "evaluated",
"thread",
]);
const now = useTimeSource(); const now = useTimeSource();
const status = () => managed.status; const status = () => props.status;
const toot = () => status().reblog ?? status(); const toot = () => status().reblog ?? status();
const session = useDefaultSession(); const session = useDefaultSession();
const [reveal, setReveal] = createSignal(false); const [reveal, setReveal] = createSignal(false);
css`
.reply-sep {
margin-left: calc(var(--toot-avatar-size) + var(--card-pad) + 8px);
margin-block: 8px;
}
.thread-top,
.thread-mid,
.thread-btm {
position: relative;
&::before {
content: "";
position: absolute;
left: 36px;
background-color: var(--tutu-color-secondary);
width: 2px;
display: block;
}
}
.thread-mid {
&::before {
top: 0;
bottom: 0;
}
}
.thread-top {
&::before {
top: 16px;
bottom: 0;
}
}
.thread-btm {
&::before {
top: 0;
height: 16px;
}
}
`;
return ( return (
<> <>
<article <article
classList={{ classList={{
[tootStyle.toot]: true, "RegularToot": true,
[tootStyle.expanded]: managed.evaluated, "expanded": props.evaluated,
"thread-top": managed.thread === "top", "thread-top": props.thread === "top",
"thread-mid": managed.thread === "middle", "thread-mid": props.thread === "middle",
"thread-btm": managed.thread === "bottom", "thread-btm": props.thread === "bottom",
[managed.class || ""]: true, [props.class || ""]: true,
}} }}
ref={rootRef!} ref={rootRef!}
lang={toot().language || managed.lang} lang={toot().language || props.lang}
{...rest} {...rest}
> >
<Show when={!!status().reblog}> <Show when={!!status().reblog}>
<div class={tootStyle.tootRetootGrp}> <div class="retoot-grp">
<BoostIcon /> <BoostIcon />
<Body2 <Body2
ref={(e: { innerHTML: string }) => { ref={(e: { innerHTML: string }) => {
@ -360,27 +253,14 @@ const RegularToot: Component<RegularTootProps> = (props) => {
/> />
</Show> </Show>
<Show when={toot().poll}> <Show when={toot().poll}>
<TootPoll <TootPoll value={toot().poll!} status={toot()} />
options={toot().poll!.options}
multiple={toot().poll!.multiple}
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={props.actionable}>
<Divider <Divider
class={cardStyle.cardNoPad} class={cardStyle.cardNoPad}
style={{ "margin-top": "8px" }} style={{ "margin-top": "8px" }}
/> />
<TootActionGroup value={toot()} {...managedActionGroup} /> <TootActionGroup value={toot()} class={cardStyle.cardGutSkip} />
</Show> </Show>
</article> </article>
</> </>

View file

@ -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>

View file

@ -42,7 +42,7 @@ import {
import type { Account } from "../accounts/stores"; import type { Account } from "../accounts/stores";
import "./TootComposer.css"; import "./TootComposer.css";
import BottomSheet from "~material/BottomSheet"; import BottomSheet from "~material/BottomSheet";
import { useLanguage } from "~platform/i18n"; import { useAppLocale } from "~platform/i18n";
import iso639_1 from "iso-639-1"; import iso639_1 from "iso-639-1";
import ChooseTootLang from "./ChooseTootLang"; import ChooseTootLang from "./ChooseTootLang";
import type { mastodon } from "masto"; import type { mastodon } from "masto";
@ -98,7 +98,8 @@ const TootVisibilityPickerDialog: Component<{
style={{ style={{
"border-top": "1px solid #ddd", "border-top": "1px solid #ddd",
background: "var(--tutu-color-surface)", background: "var(--tutu-color-surface)",
padding: "8px 16px calc(8px + var(--safe-area-inset-bottom, 0px))", padding:
"8px 16px calc(8px + var(--safe-area-inset-bottom, 0px))",
width: "100%", width: "100%",
"text-align": "end", "text-align": "end",
}} }}
@ -232,7 +233,7 @@ const TootComposer: Component<{
const [permPicker, setPermPicker] = createSignal(false); const [permPicker, setPermPicker] = createSignal(false);
const [language, setLanguage] = createSignal("en"); const [language, setLanguage] = createSignal("en");
const [langPickerOpen, setLangPickerOpen] = createSignal(false); const [langPickerOpen, setLangPickerOpen] = createSignal(false);
const appLanguage = useLanguage(); const { language: appLanguage } = useAppLocale();
const [openMenu, menuState] = createManagedMenuState(); const [openMenu, menuState] = createManagedMenuState();
const randomPlaceholder = useRandomChoice(() => [ const randomPlaceholder = useRandomChoice(() => [

View file

@ -6,6 +6,7 @@ import {
createSelector, createSelector,
Index, Index,
createMemo, createMemo,
For,
} from "solid-js"; } from "solid-js";
import { type mastodon } from "masto"; import { type mastodon } from "masto";
import { vibrate } from "~platform/hardware"; import { vibrate } from "~platform/hardware";
@ -14,6 +15,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 +234,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 +262,6 @@ const TootList: Component<{
poll: npoll, poll: npoll,
}); });
} }
}; };
return ( return (
@ -273,49 +271,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">
<For 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()]} </For>
/> </div>
); </TootEnvProvider>
}}
</Index>
);
}}
</Index>
</div>
</ErrorBoundary> </ErrorBoundary>
); );
}; };

View file

@ -1,41 +1,3 @@
.toot {
--card-pad: 16px;
--card-gut: 16px;
--toot-avatar-size: 40px;
margin-block: 0;
position: relative;
contain: content;
cursor: pointer;
&.toot {
/* fix composition ordering: I think the css module processor should aware the overriding and behaves, but no */
transition:
margin-top 60ms var(--tutu-anim-curve-sharp),
margin-bottom 60ms var(--tutu-anim-curve-sharp),
height 60ms var(--tutu-anim-curve-sharp),
var(--tutu-transition-shadow);
border-radius: 0;
}
&>.toot {
box-shadow: none;
}
time {
color: var(--tutu-color-secondary-text-on-surface);
}
& :global(.custom-emoji) {
height: 1em;
object-fit: contain;
}
&.expanded {
margin-block: 20px;
box-shadow: var(--tutu-shadow-e9);
}
}
.tootAuthorGrp { .tootAuthorGrp {
display: flex; display: flex;
align-items: flex-start; align-items: flex-start;
@ -88,90 +50,3 @@
border: 1px solid var(--tutu-color-surface); border: 1px solid var(--tutu-color-surface);
background-color: var(--tutu-color-surface-d); background-color: var(--tutu-color-surface-d);
} }
.toot.compact {
display: grid;
grid-template-columns: auto 1fr;
gap: 8px;
row-gap: 0;
padding-block: var(--card-gut, 16px);
padding-inline: var(--card-pad, 16px);
> :first-child {
grid-row: 1/3;
}
> :last-child {
grid-column: 2 /3;
}
}
.compactAuthorGroup {
display: flex;
gap: 8px;
align-items: center;
margin-bottom: 8px;
flex-flow: row wrap;
justify-content: flex-end;
>.compactAuthorUsername {
color: var(--tutu-color-secondary-text-on-surface);
flex-grow: 1;
}
>time {
color: var(--tutu-color-secondary-text-on-surface);
}
}
.tootRetootGrp {
display: flex;
gap: 0.25em;
margin-bottom: 8px;
align-items: center;
> :first-child {
margin-right: 0.25em;
}
}
.tootBottomActionGrp {
composes: cardGutSkip from "~material/cards.module.css";
padding-block: calc((var(--card-gut) - 10px) / 2);
animation: 225ms var(--tutu-anim-curve-std) tootBottomExpanding;
display: flex;
flex-flow: row wrap;
justify-content: space-evenly;
>button {
color: var(--tutu-color-on-surface);
padding: 10px 8px;
>svg {
font-size: 20px;
}
}
}
.tootActionWithCount {
display: flex;
align-items: center;
gap: 8px;
}
.tootAction {
display: flex;
align-items: center;
justify-content: center;
}
@keyframes tootBottomExpanding {
0% {
opacity: 0;
}
100% {
opacity: 1;
}
}

View file

@ -0,0 +1,41 @@
.TootActionGroup {
padding-block: calc((var(--card-gut) - 10px) / 2);
contain: layout style;
animation: 225ms var(--tutu-anim-curve-std) TootActionGroup_fade-in;
display: flex;
flex-flow: row wrap;
justify-content: space-evenly;
>button {
color: var(--tutu-color-on-surface);
padding: 10px 8px;
>svg {
font-size: 20px;
}
}
>* {
display: flex;
align-items: center;
}
>.with-count {
gap: 8px;
}
>.plain {
justify-content: center;
}
}
@keyframes TootActionGroup_fade-in {
0% {
opacity: 0;
}
100% {
opacity: 1;
}
}

View file

@ -0,0 +1,89 @@
import type { mastodon } from "masto";
import { useTootEnv } from "../RegularToot";
import { Button } from "@suid/material";
import { Show } from "solid-js";
import {
Bookmark,
BookmarkAddOutlined,
Repeat,
ReplyAll,
Share,
Star,
StarOutline,
} from "@suid/icons-material";
import { canShare, share } from "~platform/share";
import "./TootActionGroup.css";
async function shareContent(toot: mastodon.v1.Status) {
return await share({
url: toot.url ?? undefined,
});
}
function isolatedCallback(e: MouseEvent) {
e.stopPropagation();
}
function TootActionGroup<T extends mastodon.v1.Status>(props: {
value: T;
class?: string;
}) {
const { reply, boost, favourite, bookmark } = useTootEnv();
let actGrpElement: HTMLDivElement;
const toot = () => props.value;
return (
<div
ref={actGrpElement!}
class={`TootActionGroup ${props.class || ""}`}
onClick={isolatedCallback}
>
<Show when={reply}>
<Button class="with-count" onClick={[reply!, props.value]}>
<ReplyAll />
<span>{toot().repliesCount}</span>
</Button>
</Show>
<Button
class="with-count"
style={{
color: toot().reblogged ? "var(--tutu-color-primary)" : undefined,
}}
onClick={[boost, props.value]}
>
<Repeat />
<span>{toot().reblogsCount}</span>
</Button>
<Button
class="with-count"
style={{
color: toot().favourited ? "var(--tutu-color-primary)" : undefined,
}}
onClick={[favourite, props.value]}
>
{toot().favourited ? <Star /> : <StarOutline />}
<span>{toot().favouritesCount}</span>
</Button>
<Button
class="plain"
style={{
color: toot().bookmarked ? "var(--tutu-color-primary)" : undefined,
}}
onClick={[bookmark, props.value]}
>
{toot().bookmarked ? <Bookmark /> : <BookmarkAddOutlined />}
</Button>
<Show when={canShare({ url: toot().url ?? undefined })}>
<Button
class="plain"
aria-label="Share"
onClick={[shareContent, toot()]}
>
<Share />
</Button>
</Show>
</div>
);
}
export default TootActionGroup;

View file

@ -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()]}
/> />

View file

@ -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");