import { catchError, createRenderEffect, createResource, createSignal, createUniqueId, For, Switch, Match, onCleanup, Show, type Component, } from "solid-js"; import Scaffold from "../material/Scaffold"; import { AppBar, Avatar, Button, CircularProgress, Divider, IconButton, ListItemAvatar, ListItemIcon, ListItemText, MenuItem, Toolbar, } from "@suid/material"; import { Close, Edit, ExpandMore, Group, MoreVert, OpenInBrowser, PersonOff, PlaylistAdd, Send, Share, Translate, Verified, } from "@suid/icons-material"; import { Title } from "../material/typography"; import { useNavigate, 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 } 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"; const Profile: Component = () => { const navigate = useNavigate(); 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 menuButId = 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 [profileErrorUncaught] = createResource( () => [session().client, params.id] as const, async ([client, id]) => { return await client.v1.accounts.$select(id).fetch(); }, ); const profile = () => catchError(profileErrorUncaught, (err) => { console.error(err); }); const isCurrentSessionProfile = () => { return session().account?.inf?.url === profile()?.url; }; const [recentTootFilter, setRecentTootFilter] = createSignal({ pinned: true, boost: false, reply: true, original: true, }); const [recentToots, recentTootChunk, { refetch: refetchRecentToots }] = createTimeline( () => session().client.v1.accounts.$select(params.id).statuses, () => { const { boost, reply } = recentTootFilter(); return { limit: 20, excludeReblogs: !boost, excludeReplies: !reply }; }, ); const [pinnedToots, pinnedTootChunk] = createTimelineSnapshot( () => session().client.v1.accounts.$select(params.id).statuses, () => { return { limit: 20, pinned: true }; }, ); 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); return ( createRenderEffect(() => (e.innerHTML = displayName())) } > } class="Profile" > document.getElementById(menuButId)!.getBoundingClientRect() } > { const { left, right, top } = event.currentTarget.getBoundingClientRect(); openSubscribeMenu({ left, right, top, e: 1, }); }} > Subscribe... } > Edit... Subscribers Blocklist Translate Name and Bio... Mention in... Open in browser... share({ url: profile()?.url })}> Share...
obx.observe(e)} src={bannerImg()} style={{ "object-fit": "cover", width: "100%", height: "100%", }} crossOrigin="anonymous" onLoad={async (event) => { const ins = new FastAverageColor(); const colors = ins.getColor(event.currentTarget); setBannerSampledColors({ average: colors.hex, text: colors.isDark ? "white" : "black", }); ins.destroy(); }} >
createRenderEffect(() => { e.innerHTML = resolveCustomEmoji( session().account?.inf?.displayName || "", session().account?.inf?.emojis ?? [], ); }) } > 's Home
createRenderEffect(() => (e.innerHTML = displayName())) } > {fullUsername()}
{<>}
createRenderEffect(() => (e.innerHTML = description() || "")) } >
{(item, index) => { return ( ); }}
{item.name} { createRenderEffect(() => (e.innerHTML = item.value)); }} >
0}>
}>
); }; export default Profile;