import type { mastodon } from "masto"; import { splitProps, type Component, type JSX, Show, createRenderEffect, createSignal, createEffect, } from "solid-js"; import tootStyle from "./toot.module.css"; import { formatRelative } from "date-fns"; import Img from "../material/Img.js"; import { Body1, Body2, Caption, Subheading, Title, } from "../material/typography.js"; import { css } from "solid-styled"; import { BookmarkAddOutlined, Repeat, ReplyAll, Star, StarOutline, Bookmark, Reply, } from "@suid/icons-material"; import { useTimeSource } from "../platform/timesrc.js"; import { resolveCustomEmoji } from "../masto/toot.js"; import { Divider } from "@suid/material"; import cardStyle from "../material/cards.module.css"; import Button from "../material/Button.js"; import MediaAttachmentGrid from "./MediaAttachmentGrid.js"; import { FastAverageColor } from "fast-average-color"; import Color from "colorjs.io"; import { useDateFnLocale } from "../platform/i18n"; type TootContentViewProps = { source?: string; emojis?: mastodon.v1.CustomEmoji[]; } & JSX.HTMLAttributes; const TootContentView: Component = (props) => { const [managed, rest] = splitProps(props, ["source", "emojis"]); return (
{ createRenderEffect(() => { ref.innerHTML = managed.source ? managed.emojis ? resolveCustomEmoji(managed.source, managed.emojis) : managed.source : ""; }); }} {...rest} >
); }; const RetootIcon: Component = (props) => { const [managed, rest] = splitProps(props, ["class"]); css` .retoot-icon { padding: 0; display: inline-block; border-radius: 2px; > :global(svg) { color: green; font-size: 1rem; vertical-align: middle; } } `; return ( ); }; const ReplyIcon: Component = (props) => { const [managed, rest] = splitProps(props, ["class"]); css` .retoot-icon { padding: 0; display: inline-block; border-radius: 2px; > :global(svg) { color: var(--tutu-color-primary); font-size: 1rem; vertical-align: middle; } } `; return ( ); }; type TootActionGroupProps = { onRetoot?: (value: T) => void; onFavourite?: (value: T) => void; onBookmark?: (value: T) => void; onReply?: (value: T) => void; }; type TootCardProps = { status: mastodon.v1.Status; actionable?: boolean; evaluated?: boolean; } & TootActionGroupProps & JSX.HTMLElementTags["article"]; function isolatedCallback(e: MouseEvent) { e.stopPropagation(); } function TootActionGroup( props: TootActionGroupProps & { value: T }, ) { const toot = () => props.value; return (
); } function TootAuthorGroup(props: { status: mastodon.v1.Status; now: Date }) { const toot = () => props.status; const dateFnLocale = useDateFnLocale() return (
{ createRenderEffect(() => { e.innerHTML = resolveCustomEmoji( toot().account.displayName, toot().account.emojis, ); }); }} /> @{toot().account.username}@{new URL(toot().account.url).hostname}
); } export function TootPreviewCard(props: { src: mastodon.v1.PreviewCard; alwaysCompact?: boolean; }) { let root: HTMLAnchorElement; createEffect(() => { if (props.alwaysCompact) { root.classList.add(tootStyle.compact); return; } if (!props.src.width) return; const width = root.getBoundingClientRect().width; if (width > props.src.width) { root.classList.add(tootStyle.compact); } else { root.classList.remove(tootStyle.compact); } }); const onImgLoad = (event: Event & { currentTarget: HTMLImageElement }) => { // TODO: better extraction algorithm // I'd like to use a pattern panel and match one in the panel from the extracted color const fac = new FastAverageColor(); const result = fac.getColor(event.currentTarget); if (result.error) { console.error(result.error); fac.destroy(); return; } root.style.setProperty("--tutu-color-surface", result.hex); const focusSurface = result.isDark ? new Color(result.hex).darken(0.2) : new Color(result.hex).lighten(0.2); root.style.setProperty("--tutu-color-surface-d", focusSurface.toString()); const textColor = result.isDark ? "white" : "black"; const secondaryTextColor = new Color(textColor); secondaryTextColor.alpha = 0.75; root.style.setProperty("--tutu-color-on-surface", textColor); root.style.setProperty( "--tutu-color-secondary-text-on-surface", secondaryTextColor.toString(), ); fac.destroy(); }; return ( {props.src.title} {props.src.description} ); } const RegularToot: Component = (props) => { let rootRef: HTMLElement; const [managed, managedActionGroup, rest] = splitProps( props, ["status", "lang", "class", "actionable", "evaluated"], ["onRetoot", "onFavourite", "onBookmark", "onReply"], ); const now = useTimeSource(); const status = () => managed.status; const toot = () => status().reblog ?? status(); css` .reply-sep { margin-left: calc(var(--toot-avatar-size) + var(--card-pad) + 8px); margin-block: 8px; } `; return ( <>
{ createRenderEffect(() => { e.innerHTML = resolveCustomEmoji( status().account.displayName, toot().emojis, ); }); }} >{" "} boosted
0}>
); }; export default RegularToot;