import { catchError, createResource, createSignal, createUniqueId, For, Switch, Match, onCleanup, Show, type Component, createMemo, } from "solid-js"; import Scaffold from "~material/Scaffold"; import { Avatar, Button, Checkbox, CircularProgress, Divider, IconButton, ListItemAvatar, ListItemIcon, ListItemSecondaryAction, ListItemText, MenuItem, } from "@suid/material"; import { Close, Edit, ExpandMore, Group, Lock, MoreVert, OpenInBrowser, PersonOff, PlaylistAdd, Send, Share, SmartToySharp, Subject, Verified, } from "@suid/icons-material"; import { Body2, Title } from "~material/typography"; import { useParams } from "@solidjs/router"; import { useSessionForAcctStr } from "../masto/clients"; import { resolveCustomEmoji } from "../masto/toot"; import { FastAverageColor } from "fast-average-color"; import { useWindowSize } from "@solid-primitives/resize-observer"; import { createTimeline, createTimelineSnapshot, emptyTimeline, } from "../masto/timelines"; import TootList from "../timelines/TootList"; import { createTimeSource, TimeSourceProvider } from "~platform/timesrc"; import TootFilterButton from "./TootFilterButton"; import Menu, { createManagedMenuState } from "~material/Menu"; import { share } from "~platform/share"; import "./Profile.css"; import { useNavigator } from "~platform/StackedRouter"; import { createSingluarItemSelection, default as ItemSelectionProvider, } from "../timelines/toots/ItemSelectionProvider"; import AppTopBar from "~material/AppTopBar"; import type { Account } from "../accounts/stores"; import DocumentTitle from "~platform/DocumentTitle"; const Profile: Component = () => { const { pop } = useNavigator(); const params = useParams<{ acct: string; id: string }>(); const acctText = () => decodeURIComponent(params.acct); const session = useSessionForAcctStr(acctText); const [bannerSampledColors, setBannerSampledColors] = createSignal<{ average: string; text: string; }>(); const windowSize = useWindowSize(); const time = createTimeSource(); const [, selectionState] = createSingluarItemSelection(); const menuButId = createUniqueId(); const recentTootListId = createUniqueId(); const optMenuId = createUniqueId(); const [menuOpen, setMenuOpen] = createSignal(false); const [openSubscribeMenu, subscribeMenuState] = createManagedMenuState(); const [scrolledPastBanner, setScrolledPastBanner] = createSignal(false); const obx = new IntersectionObserver( (entries) => { const ent = entries[0]; if (ent.intersectionRatio < 0.1) { setScrolledPastBanner(true); } else { setScrolledPastBanner(false); } }, { threshold: 0.1, }, ); onCleanup(() => obx.disconnect()); const [profileUncaught] = createResource( () => [session().client, params.id] as const, async ([client, id]) => { if (id.startsWith("@")) { return await client.v1.accounts.lookup({ acct: id.slice(1) }); } return await client.v1.accounts.$select(id).fetch(); }, ); const profile = () => { try { return profileUncaught(); } catch (reason) { console.error(reason); } }; const profileAcctId = () => { if (params.id.startsWith("@")) { // Webfinger return profile()?.id; } else { return params.id; } }; const isCurrentSessionProfile = () => { return (session().account as Account).inf?.url === profile()?.url; }; const [recentTootFilter, setRecentTootFilter] = createSignal({ pinned: true, boost: false, reply: true, original: true, }); const recentTimeline = () => { const id = profileAcctId(); if (id) { return session().client.v1.accounts.$select(id).statuses; } else { return emptyTimeline; } }; const [recentToots, recentTootChunk, { refetch: refetchRecentToots }] = createTimeline(recentTimeline, () => { const { boost, reply } = recentTootFilter(); return { limit: 20, excludeReblogs: !boost, excludeReplies: !reply }; }); const [pinnedToots, pinnedTootChunk] = createTimelineSnapshot( recentTimeline, () => { return { limit: 20, pinned: true }; }, ); const [relationshipUncaught, { mutate: mutateRelationship }] = createResource( () => [session(), profileAcctId()] as const, async ([sess, id]) => { if (!sess.account || !id) return; // No account, no relation const relations = await session().client.v1.accounts.relationships.fetch({ id: [id], }); return relations.length > 0 ? relations[0] : undefined; }, ); const relationship = () => catchError(relationshipUncaught, (reason) => { console.error(reason); }); const bannerImg = () => profile()?.header; const avatarImg = () => profile()?.avatar; const displayName = () => resolveCustomEmoji(profile()?.displayName || "", profile()?.emojis ?? []); const fullUsername = () => (profile()?.acct ? `@${profile()!.acct!}` : ""); // TODO: full user name const description = () => profile()?.note; const isTootListLoading = () => recentTootChunk.loading || (recentTootFilter().pinned && pinnedTootChunk.loading); const sessionDisplayName = createMemo(() => resolveCustomEmoji( (session().account as Account).inf?.displayName || "", (session().account as Account).inf?.emojis ?? [], ), ); const toggleSubscribeHome = async (event: Event) => { const client = session().client; const acctId = profileAcctId(); if (!session().account || !acctId) return; const isSubscribed = relationship()?.following ?? false; mutateRelationship((x) => Object.assign({ following: !isSubscribed }, x)); subscribeMenuState.onClose(event); if (isSubscribed) { const nrel = await client.v1.accounts.$select(acctId).unfollow(); mutateRelationship(nrel); } else { const nrel = await client.v1.accounts.$select(acctId).follow(); mutateRelationship(nrel); } }; return ( <> {profile()?.displayName ?? "Someone"} } class="Profile" > ); }; export default Profile;