// Copyright 2022 Signal Messenger, LLC // SPDX-License-Identifier: AGPL-3.0-only import Fuse from 'fuse.js'; import React, { useEffect, useState } from 'react'; import classNames from 'classnames'; import type { ConversationType, ShowConversationType, } from '../state/ducks/conversations'; import type { ConversationStoryType, MyStoryType, StoryViewType, } from '../types/Stories'; import type { LocalizerType } from '../types/Util'; import { MyStoriesButton } from './MyStoriesButton'; import { SearchInput } from './SearchInput'; import { StoryListItem } from './StoryListItem'; import { isNotNil } from '../util/isNotNil'; const FUSE_OPTIONS: Fuse.IFuseOptions = { getFn: (story, path) => { if (path === 'searchNames') { return [story.storyView.sender.title, story.storyView.sender.name].filter( isNotNil ); } return story.group?.title ?? ''; }, keys: [ { name: 'searchNames', weight: 1, }, { name: 'group', weight: 1, }, ], threshold: 0.1, }; function search( stories: ReadonlyArray, searchTerm: string ): Array { return new Fuse(stories, FUSE_OPTIONS) .search(searchTerm) .map(result => result.item); } function getNewestMyStory(story: MyStoryType): StoryViewType { return story.stories[story.stories.length - 1]; } export type PropsType = { hiddenStories: Array; i18n: LocalizerType; me: ConversationType; myStories: Array; onAddStory: () => unknown; onMyStoriesClicked: () => unknown; onStoryClicked: (conversationId: string) => unknown; queueStoryDownload: (storyId: string) => unknown; showConversation: ShowConversationType; stories: Array; toggleHideStories: (conversationId: string) => unknown; toggleStoriesView: () => unknown; }; export const StoriesPane = ({ hiddenStories, i18n, me, myStories, onAddStory, onMyStoriesClicked, onStoryClicked, queueStoryDownload, showConversation, stories, toggleHideStories, toggleStoriesView, }: PropsType): JSX.Element => { const [searchTerm, setSearchTerm] = useState(''); const [isShowingHiddenStories, setIsShowingHiddenStories] = useState(false); const [renderedStories, setRenderedStories] = useState>(stories); useEffect(() => { if (searchTerm) { setRenderedStories(search(stories, searchTerm)); } else { setRenderedStories(stories); } }, [searchTerm, stories]); return ( <>
{ setSearchTerm(event.target.value); }} placeholder={i18n('search')} value={searchTerm} /> 1 : false} i18n={i18n} me={me} newestStory={ myStories.length ? getNewestMyStory(myStories[0]) : undefined } onClick={onMyStoriesClicked} queueStoryDownload={queueStoryDownload} />
{renderedStories.map(story => ( { onStoryClicked(story.conversationId); }} onHideStory={toggleHideStories} onGoToConversation={conversationId => { showConversation({ conversationId }); toggleStoriesView(); }} queueStoryDownload={queueStoryDownload} story={story.storyView} /> ))} {Boolean(hiddenStories.length) && ( <> {isShowingHiddenStories && hiddenStories.map(story => ( { onStoryClicked(story.conversationId); }} onHideStory={toggleHideStories} onGoToConversation={conversationId => { showConversation({ conversationId }); toggleStoriesView(); }} queueStoryDownload={queueStoryDownload} story={story.storyView} /> ))} )} {!stories.length && i18n('Stories__list-empty')}
); };