import "./Timeline.css"; import { FormattedMessage } from "react-intl"; import { useCallback, useMemo } from "react"; import { useInView } from "react-intersection-observer"; import { TaggedNostrEvent, EventKind, u256 } from "@snort/system"; import Icon from "Icons/Icon"; import { dedupeByPubkey, findTag } from "SnortUtils"; import ProfileImage from "Element/ProfileImage"; import useTimelineFeed, { TimelineFeed, TimelineSubject } from "Feed/TimelineFeed"; import Note from "Element/Note"; import useModeration from "Hooks/useModeration"; import { LiveStreams } from "Element/LiveStreams"; export interface TimelineProps { postsOnly: boolean; subject: TimelineSubject; method: "TIME_RANGE" | "LIMIT_UNTIL"; ignoreModeration?: boolean; window?: number; now?: number; loadMore?: boolean; noSort?: boolean; } /** * A list of notes by "subject" */ const Timeline = (props: TimelineProps) => { const feedOptions = useMemo(() => { return { method: props.method, window: props.window, now: props.now, }; }, [props]); const feed: TimelineFeed = useTimelineFeed(props.subject, feedOptions); const { muted, isMuted } = useModeration(); const { ref, inView } = useInView(); const filterPosts = useCallback( (nts: readonly TaggedNostrEvent[]) => { const a = [...nts.filter(a => a.kind !== EventKind.LiveEvent)]; props.noSort || a.sort((a, b) => b.created_at - a.created_at); return a ?.filter(a => (props.postsOnly ? !a.tags.some(b => b[0] === "e") : true)) .filter(a => props.ignoreModeration || !isMuted(a.pubkey)); }, [props.postsOnly, muted, props.ignoreModeration], ); const mainFeed = useMemo(() => { return filterPosts(feed.main ?? []); }, [feed, filterPosts]); const latestFeed = useMemo(() => { return filterPosts(feed.latest ?? []).filter(a => !mainFeed.some(b => b.id === a.id)); }, [feed, filterPosts]); const relatedFeed = useCallback( (id: u256) => { return (feed.related ?? []).filter(a => findTag(a, "e") === id); }, [feed.related], ); const liveStreams = useMemo(() => { return (feed.main ?? []).filter(a => a.kind === EventKind.LiveEvent && findTag(a, "status") === "live"); }, [feed]); const latestAuthors = useMemo(() => { return dedupeByPubkey(latestFeed).map(e => e.pubkey); }, [latestFeed]); function onShowLatest(scrollToTop = false) { feed.showLatest(); if (scrollToTop) { window.scrollTo(0, 0); } } return ( <> {latestFeed.length > 0 && ( <>
onShowLatest()} ref={ref}> {latestAuthors.slice(0, 3).map(p => { return ; })}
{!inView && (
onShowLatest(true)}> {latestAuthors.slice(0, 3).map(p => { return ; })}
)} )} {mainFeed.map(e => ( ))} {(props.loadMore === undefined || props.loadMore === true) && (
)} ); }; export default Timeline;