@sendbird/uikit-react
Version:
Sendbird UIKit for React: A feature-rich and customizable chat UI kit with messaging, channel management, and user authentication.
321 lines (317 loc) • 18.6 kB
JavaScript
import { b as __awaiter, c as __generator, _ as __assign } from '../../chunks/bundle-B7RvGM03.js';
import React__default, { forwardRef, useRef, useLayoutEffect, useCallback, useState, useEffect } from 'react';
import { useGroupChannelHandler } from '@sendbird/uikit-tools';
import { M as getHTMLTextDirection, l as isSendableMessage } from '../../chunks/bundle-D4ssOJFH.js';
import { T as TypingIndicatorType } from '../../chunks/bundle-Lj21dphY.js';
import PlaceHolder, { PlaceHolderTypes } from '../../ui/PlaceHolder.js';
import Icon, { IconTypes, IconColors } from '../../ui/Icon.js';
import { Message } from './Message.js';
import { UnreadCount } from './UnreadCount.js';
import { FrozenNotification } from './FrozenNotification.js';
import { S as SCROLL_BUFFER } from '../../chunks/bundle-DwUX_8pX.js';
import TypingIndicatorBubble from '../../ui/TypingIndicatorBubble.js';
import { n as noop, d as deleteNullish } from '../../chunks/bundle-BsH3r84n.js';
import { getMessagePartsInfo } from '../../Channel/utils/getMessagePartsInfo.js';
import { MessageProvider } from '../../Message/context.js';
import { g as getComponentKeyFromMessage, b as isContextMenuClosed } from '../../chunks/bundle-BPQwWHRF.js';
import { a as isAboutSame } from '../../chunks/bundle-DO3psIF-.js';
import { u as useGroupChannel } from '../../chunks/bundle-CH17iDvf.js';
import { u as useSendbird } from '../../chunks/bundle-BwKOzdpa.js';
import { u as useLocalization } from '../../chunks/bundle-DVsUiVO7.js';
import '@sendbird/chat/groupChannel';
import '../../utils/message/getOutgoingMessageState.js';
import '../../chunks/bundle-DHiHHi0U.js';
import '../../chunks/bundle-jVso52Zj.js';
import '../../chunks/bundle-BGdOmHwC.js';
import '../../chunks/bundle-sYB0sr_l.js';
import '../../chunks/bundle-B4JqClb8.js';
import '../../ui/Loader.js';
import '../../chunks/bundle-D0GNokoe.js';
import '../../chunks/bundle-BqABa95r.js';
import '../../chunks/bundle-3nZUyzSE.js';
import '../../chunks/bundle-c7rLu_k3.js';
import '../../Message/hooks/useDirtyGetMentions.js';
import '../../ui/DateSeparator.js';
import '../../chunks/bundle-D7eceuN1.js';
import '../../ui/MessageInput.js';
import '../../chunks/bundle-CXM8hj4p.js';
import '../../ui/IconButton.js';
import '../../ui/Button.js';
import '../../chunks/bundle-DUgLRAHL.js';
import 'dompurify';
import '../../chunks/bundle-C2fRYQA7.js';
import '../../chunks/bundle-DJDLoCXy.js';
import '../../chunks/bundle-DFELlqx3.js';
import '../../chunks/bundle-BqgOvZNO.js';
import '../../chunks/bundle-Pi8D8X0p.js';
import '@sendbird/chat';
import '@sendbird/chat/openChannel';
import '../../ui/MessageContent.js';
import '../../chunks/bundle-B--2jEB0.js';
import '../../chunks/bundle-B6WP8kqn.js';
import '../../chunks/bundle-otQQaCra.js';
import '../../chunks/bundle-cJShAJkO.js';
import '../../chunks/bundle-Hge7crLJ.js';
import 'react-dom';
import '../../ui/SortByRow.js';
import '../../chunks/bundle-BgnOPI6k.js';
import '../../ui/MessageItemReactionMenu.js';
import '../../ui/ImageRenderer.js';
import '../../ui/ReactionButton.js';
import '../../chunks/bundle-Bfy1ZXXV.js';
import '../../chunks/bundle-DVe4vLri.js';
import '../../ui/EmojiReactions.js';
import '../../ui/ReactionBadge.js';
import '../../ui/BottomSheet.js';
import '../../hooks/useModal.js';
import '../../chunks/bundle-Vv-hPvqF.js';
import '../../ui/UserListItem.js';
import '../../chunks/bundle-CZiyJYBm.js';
import '../../chunks/bundle-BIjkVTPN.js';
import '../../chunks/bundle-sT3ilq3P.js';
import '../../ui/MutedAvatarOverlay.js';
import '../../ui/Checkbox.js';
import '../../ui/UserProfile.js';
import '../../sendbirdSelectors.js';
import '../../chunks/bundle-CRkzqUcm.js';
import '../../ui/Tooltip.js';
import '../../ui/TooltipWrapper.js';
import '../../chunks/bundle-utR756AD.js';
import '../../ui/AdminMessage.js';
import '../../ui/QuoteMessage.js';
import '../../chunks/bundle-DN51BlH5.js';
import '@sendbird/chat/message';
import '../../ui/MobileMenu.js';
import '../../ui/ThreadReplies.js';
import '../../chunks/bundle-yFVGOx88.js';
import '../../ui/OGMessageItemBody.js';
import '../../chunks/bundle-B4lXe9Vx.js';
import '../../ui/MentionLabel.js';
import '../../ui/LinkLabel.js';
import '../../ui/TextMessageItemBody.js';
import '../../ui/FileMessageItemBody.js';
import '../../ui/TextButton.js';
import '../../chunks/bundle-BkDFCuVL.js';
import '../../ui/FileViewer.js';
import '../../chunks/bundle-x4EJpyZs.js';
import '../../ui/VoiceMessageItemBody.js';
import '../../ui/ProgressBar.js';
import '../../VoicePlayer/useVoicePlayer.js';
import '../../chunks/bundle-CGI7FGQ3.js';
import '../../VoiceRecorder/context.js';
import '../../ui/PlaybackTime.js';
import '../../ui/ThumbnailMessageItemBody.js';
import '../../ui/UnknownMessageItemBody.js';
import '../../ui/TemplateMessageItemBody.js';
import '../../chunks/bundle-DNaw1ZKt.js';
import '../../ui/FallbackTemplateMessageItemBody.tsx.js';
import '../../ui/LoadingTemplateMessageItemBody.tsx.js';
import '../../ui/MessageFeedbackFailedModal.js';
import '../../ui/FeedbackIconButton.js';
import '../../ui/MobileFeedbackMenu.js';
import '../../ui/MessageFeedbackModal.js';
import '../../ui/Input.js';
import '../../chunks/bundle-CetM7tZt.js';
import './SuggestedReplies.js';
import '../../chunks/bundle-B_rjSjbC.js';
import './FileViewer.js';
import '../../chunks/bundle-CgpXjjQu.js';
import '../../chunks/bundle-DO39GevF.js';
import '../../chunks/bundle-DPhArxeq.js';
import '../../chunks/bundle-COT_8pQt.js';
import './RemoveMessageModal.js';
import '../../chunks/bundle-DrY7Xn_q.js';
import '../../chunks/bundle-Bb-DXei6.js';
import '../../Channel/utils/compareMessagesForGrouping.js';
var InfiniteList = forwardRef(function (props, listRef) {
var messages = props.messages, renderMessage = props.renderMessage, scrollPositionRef = props.scrollPositionRef, scrollDistanceFromBottomRef = props.scrollDistanceFromBottomRef, onLoadPrevious = props.onLoadPrevious, onLoadNext = props.onLoadNext, _a = props.loadThreshold, loadThreshold = _a === void 0 ? 0.05 : _a, typingIndicator = props.typingIndicator, _b = props.onScrollPosition, onScrollPosition = _b === void 0 ? noop : _b, initDeps = props.initDeps;
var isFetching = React__default.useRef(false);
var direction = React__default.useRef();
var oldScrollTop = useRef(0);
// SideEffect: scroll to bottom on initialized
useLayoutEffect(function () {
if (listRef.current) {
listRef.current.scrollTop = listRef.current.scrollHeight;
}
}, initDeps);
// SideEffect: keep scroll position
useLayoutEffect(function () {
if (listRef.current) {
if (direction.current === 'top') {
listRef.current.scrollTop = listRef.current.scrollHeight - scrollPositionRef.current;
}
if (direction.current === 'bottom') {
listRef.current.scrollTop = oldScrollTop.current;
}
direction.current = undefined;
}
}, [listRef.current, messages.length]);
var handleScroll = useCallback(function () { return __awaiter(void 0, void 0, void 0, function () {
var list, threshold;
return __generator(this, function (_a) {
switch (_a.label) {
case 0:
if (!listRef.current)
return [2 /*return*/];
list = listRef.current;
onScrollPosition(getReachedStatus(list));
scrollPositionRef.current = list.scrollHeight - list.scrollTop;
scrollDistanceFromBottomRef.current = scrollPositionRef.current - list.clientHeight;
oldScrollTop.current = list.scrollTop;
if (isFetching.current)
return [2 /*return*/];
threshold = list.clientHeight * Math.min(Math.max(0, loadThreshold), 1);
if (!(list.scrollTop <= threshold)) return [3 /*break*/, 2];
isFetching.current = true;
direction.current = 'top';
return [4 /*yield*/, onLoadPrevious()];
case 1:
_a.sent();
isFetching.current = false;
return [3 /*break*/, 5];
case 2:
if (!(list.scrollHeight - list.scrollTop - list.clientHeight <= threshold)) return [3 /*break*/, 4];
isFetching.current = true;
direction.current = 'bottom';
return [4 /*yield*/, onLoadNext()];
case 3:
_a.sent();
isFetching.current = false;
return [3 /*break*/, 5];
case 4:
direction.current = undefined;
_a.label = 5;
case 5: return [2 /*return*/];
}
});
}); }, [messages.length]);
return (React__default.createElement("div", { className: "sendbird-conversation__scroll-container" },
React__default.createElement("div", { className: "sendbird-conversation__padding" }),
React__default.createElement("div", { ref: listRef, className: "sendbird-conversation__messages-padding", "data-testid": "sendbird-message-list-container", onScroll: handleScroll },
messages.map(function (message, index) { return renderMessage({ message: message, index: index }); }),
typingIndicator)));
});
function getReachedStatus(element) {
if (isAboutSame(element.scrollTop, 0, SCROLL_BUFFER)) {
return 'top';
}
if (isAboutSame(element.scrollHeight, element.clientHeight + element.scrollTop, SCROLL_BUFFER)) {
return 'bottom';
}
return 'middle';
}
var MessageList = function (props) {
var _a, _b, _c, _d, _e;
var _f = props.className, className = _f === void 0 ? '' : _f;
var _g = deleteNullish(props), _h = _g.renderMessage, renderMessage = _h === void 0 ? function (props) { return React__default.createElement(Message, __assign({}, props)); } : _h, renderMessageContent = _g.renderMessageContent, renderSuggestedReplies = _g.renderSuggestedReplies, renderCustomSeparator = _g.renderCustomSeparator, _j = _g.renderPlaceholderLoader, renderPlaceholderLoader = _j === void 0 ? function () { return React__default.createElement(PlaceHolder, { type: PlaceHolderTypes.LOADING }); } : _j, _k = _g.renderPlaceholderEmpty, renderPlaceholderEmpty = _k === void 0 ? function () { return React__default.createElement(PlaceHolder, { className: "sendbird-conversation__no-messages", type: PlaceHolderTypes.NO_MESSAGES }); } : _k, _l = _g.renderFrozenNotification, renderFrozenNotification = _l === void 0 ? function () { return React__default.createElement(FrozenNotification, { className: "sendbird-conversation__messages__notification" }); } : _l;
var _m = useGroupChannel(), _o = _m.state, channelUrl = _o.channelUrl, hasNext = _o.hasNext, loading = _o.loading, messages = _o.messages, newMessages = _o.newMessages, isScrollBottomReached = _o.isScrollBottomReached, isMessageGroupingEnabled = _o.isMessageGroupingEnabled, currentChannel = _o.currentChannel, replyType = _o.replyType, scrollPubSub = _o.scrollPubSub, loadNext = _o.loadNext, loadPrevious = _o.loadPrevious, resetNewMessages = _o.resetNewMessages, scrollRef = _o.scrollRef, scrollPositionRef = _o.scrollPositionRef, scrollDistanceFromBottomRef = _o.scrollDistanceFromBottomRef, _p = _m.actions, scrollToBottom = _p.scrollToBottom, setIsScrollBottomReached = _p.setIsScrollBottomReached;
var state = useSendbird().state;
var stringSet = useLocalization().stringSet;
var _q = useState(), unreadSinceDate = _q[0], setUnreadSinceDate = _q[1];
useEffect(function () {
if (isScrollBottomReached) {
setUnreadSinceDate(undefined);
}
else {
setUnreadSinceDate(new Date());
}
}, [isScrollBottomReached]);
/**
* 1. Move the message list scroll
* when each message's height is changed by `reactions` OR `showEdit`
* 2. Keep the scrollBottom value after fetching new message list
*/
var onMessageContentSizeChanged = function (isBottomMessageAffected) {
if (isBottomMessageAffected === void 0) { isBottomMessageAffected = false; }
var elem = scrollRef.current;
if (elem) {
var latestDistance = scrollDistanceFromBottomRef.current;
var currentDistance = elem.scrollHeight - elem.scrollTop - elem.offsetHeight;
if (latestDistance < currentDistance && (!isBottomMessageAffected || latestDistance < SCROLL_BUFFER)) {
var diff = currentDistance - latestDistance;
// Move the scroll as much as the height of the message has changed
scrollPubSub.publish('scroll', { top: elem.scrollTop + diff, lazy: false, animated: false });
}
}
};
var renderer = {
frozenNotification: function () {
if (!currentChannel || !currentChannel.isFrozen)
return null;
return renderFrozenNotification();
},
unreadMessagesNotification: function () {
if (isScrollBottomReached || !unreadSinceDate)
return null;
return (React__default.createElement(UnreadCount, { className: "sendbird-conversation__messages__notification", count: newMessages.length, lastReadAt: unreadSinceDate, onClick: function () { return scrollToBottom(); } }));
},
scrollToBottomButton: function () {
if (!hasNext() && isScrollBottomReached)
return null;
return (React__default.createElement("div", { className: "sendbird-conversation__scroll-bottom-button", onClick: function () { return scrollToBottom(); }, onKeyDown: function () { return scrollToBottom(); }, tabIndex: 0, role: "button" },
React__default.createElement(Icon, { width: "24px", height: "24px", type: IconTypes.CHEVRON_DOWN, fillColor: IconColors.PRIMARY })));
},
};
if (loading) {
return renderPlaceholderLoader();
}
if (messages.length === 0) {
return renderPlaceholderEmpty();
}
return (React__default.createElement(React__default.Fragment, null,
React__default.createElement("div", { className: "sendbird-conversation__messages ".concat(className), dir: getHTMLTextDirection(state.config.htmlTextDirection, state.config.forceLeftToRightMessageLayout) },
React__default.createElement(InfiniteList, { ref: scrollRef, initDeps: [channelUrl], scrollPositionRef: scrollPositionRef, scrollDistanceFromBottomRef: scrollDistanceFromBottomRef, onLoadNext: loadNext, onLoadPrevious: loadPrevious, onScrollPosition: function (it) {
var isScrollBottomReached = it === 'bottom';
if (newMessages.length > 0 && isScrollBottomReached) {
resetNewMessages();
}
setIsScrollBottomReached(isScrollBottomReached);
}, messages: messages, renderMessage: function (_a) {
var message = _a.message, index = _a.index;
var _b = getMessagePartsInfo({
allMessages: messages,
stringSet: stringSet,
replyType: replyType !== null && replyType !== void 0 ? replyType : 'NONE',
isMessageGroupingEnabled: isMessageGroupingEnabled !== null && isMessageGroupingEnabled !== void 0 ? isMessageGroupingEnabled : false,
currentIndex: index,
currentMessage: message,
currentChannel: currentChannel,
}), chainTop = _b.chainTop, chainBottom = _b.chainBottom, hasSeparator = _b.hasSeparator;
var isOutgoingMessage = isSendableMessage(message) && message.sender.userId === state.config.userId;
return (React__default.createElement(MessageProvider, { message: message, key: getComponentKeyFromMessage(message), isByMe: isOutgoingMessage }, renderMessage({
handleScroll: onMessageContentSizeChanged,
message: message,
hasSeparator: hasSeparator,
chainTop: chainTop,
chainBottom: chainBottom,
renderMessageContent: renderMessageContent,
renderSuggestedReplies: renderSuggestedReplies,
renderCustomSeparator: renderCustomSeparator,
})));
}, typingIndicator: !hasNext()
&& ((_b = (_a = state === null || state === void 0 ? void 0 : state.config) === null || _a === void 0 ? void 0 : _a.groupChannel) === null || _b === void 0 ? void 0 : _b.enableTypingIndicator)
&& ((_e = (_d = (_c = state === null || state === void 0 ? void 0 : state.config) === null || _c === void 0 ? void 0 : _c.groupChannel) === null || _d === void 0 ? void 0 : _d.typingIndicatorTypes) === null || _e === void 0 ? void 0 : _e.has(TypingIndicatorType.Bubble)) && (React__default.createElement(TypingIndicatorBubbleWrapper, { channelUrl: channelUrl, handleScroll: onMessageContentSizeChanged })) }),
React__default.createElement(React__default.Fragment, null, renderer.frozenNotification()),
React__default.createElement(React__default.Fragment, null, renderer.unreadMessagesNotification()),
React__default.createElement(React__default.Fragment, null, renderer.scrollToBottomButton()))));
};
var TypingIndicatorBubbleWrapper = function (props) {
var stores = useSendbird().state.stores;
var _a = useGroupChannel().state, isScrollBottomReached = _a.isScrollBottomReached, scrollPubSub = _a.scrollPubSub;
var _b = useState([]), typingMembers = _b[0], setTypingMembers = _b[1];
useGroupChannelHandler(stores.sdkStore.sdk, {
onTypingStatusUpdated: function (channel) {
if (channel.url === props.channelUrl) {
setTypingMembers(channel.getTypingUsers());
}
if (isScrollBottomReached && isContextMenuClosed()) {
setTimeout(function () {
scrollPubSub.publish('scrollToBottom', {});
}, 10);
}
},
});
return React__default.createElement(TypingIndicatorBubble, { typingMembers: typingMembers, handleScroll: props.handleScroll });
};
export { MessageList, MessageList as default };
//# sourceMappingURL=MessageList.js.map