// Copyright 2025 Signal Messenger, LLC // SPDX-License-Identifier: AGPL-3.0-only import { groupBy, orderBy } from 'lodash'; import { useMemo } from 'react'; import type { Reaction } from '../components/conversation/ReactionViewer.js'; import { isEmojiVariantValue, getEmojiVariantKeyByValue, getEmojiParentKeyByVariantKey, getEmojiVariantByKey, type EmojiVariantKey, type EmojiParentKey, } from '../components/fun/data/emojis.js'; import { isNotNil } from './isNotNil.js'; import { useFunEmojiLocalizer } from '../components/fun/useFunEmojiLocalizer.js'; type ReactionWithEmojiData = Reaction & { short_name: string | undefined; short_names: Array; sheet_x: number; sheet_y: number; parentKey: EmojiParentKey; variantKey: EmojiVariantKey; }; export function useGroupedAndOrderedReactions( reactions: ReadonlyArray | undefined, groupByKey: 'variantKey' | 'parentKey' ): Array> { const emojiLocalization = useFunEmojiLocalizer(); return useMemo(() => { if (!reactions || reactions.length === 0) { return []; } const reactionsWithEmojiData: Array = reactions .map(reaction => { if (!isEmojiVariantValue(reaction.emoji)) { return undefined; } try { const variantKey = getEmojiVariantKeyByValue(reaction.emoji); const parentKey = getEmojiParentKeyByVariantKey(variantKey); const variant = getEmojiVariantByKey(variantKey); const shortName = emojiLocalization.getLocaleShortName(variantKey); return { ...reaction, short_name: shortName, short_names: [shortName].filter(isNotNil), variantKey, parentKey, sheet_x: variant.sheetX, sheet_y: variant.sheetY, }; } catch { return undefined; } }) .filter(isNotNil); const groupedReactions = Object.values( groupBy(reactionsWithEmojiData, groupByKey) ).map(groupedReaction => orderBy( groupedReaction, [reaction => reaction.from.isMe, 'timestamp'], ['desc', 'desc'] ) ); return orderBy( groupedReactions, ['length', ([{ timestamp }]) => timestamp], ['desc', 'desc'] ); }, [reactions, groupByKey, emojiLocalization]); }