@sendbird/uikit-react
Version:
Sendbird UIKit for React: A feature-rich and customizable chat UI kit with messaging, channel management, and user authentication.
149 lines (142 loc) • 11 kB
JavaScript
import { _ as __assign } from '../chunks/bundle-DlZj_j5B.js';
import React__default, { useRef, useContext } from 'react';
import { createPortal } from 'react-dom';
import { L as LocalizationContext } from '../chunks/bundle-DxLnjDoJ.js';
import '../hooks/useModal.js';
import { e as isMultipleFilesMessage, d as isFileMessage, j as isSupportedFileView, k as isVideo, l as isImage } from '../chunks/bundle-C5Cl3Igw.js';
import { n as noop } from '../chunks/bundle-CaW4IP0_.js';
import { A as Avatar } from '../chunks/bundle-Dgt7lb2O.js';
import { L as Label, a as LabelTypography, b as LabelColors } from '../chunks/bundle-FZ_FdxZT.js';
import Icon, { IconTypes, IconColors } from './Icon.js';
import { o as SLIDER_BUTTON_ICON_SIDE_LENGTH } from '../chunks/bundle-C8kxBudB.js';
import { u as useKeyDown } from '../chunks/bundle-86aNeIDh.js';
import { a as MODAL_ROOT, M as Modal } from '../chunks/bundle-BDItDVZl.js';
import '../chunks/bundle-Dhutxl0X.js';
import '../chunks/bundle-D89Qj0P4.js';
import '../chunks/bundle-DZaN4z9l.js';
import '../chunks/bundle-DWJPyrDa.js';
import './IconButton.js';
import './Button.js';
import '../chunks/bundle-CSjCAZib.js';
import '../chunks/bundle-BnaoVUUl.js';
import '../chunks/bundle-C1jTnXNH.js';
import '@sendbird/chat';
import '@sendbird/chat/groupChannel';
import '@sendbird/chat/openChannel';
import '../utils/message/getOutgoingMessageState.js';
import '../chunks/bundle-OJHU7Q3U.js';
import '../chunks/bundle-0bJlK18Z.js';
import './ImageRenderer.js';
import '../chunks/bundle-h43SVTxb.js';
import '@sendbird/uikit-tools';
// https://developer.mozilla.org/en-US/docs/Web/HTTP/Basics_of_HTTP/MIME_types/Complete_list_of_MIME_types
var ViewerTypes = {
SINGLE: 'SINGLE',
MULTI: 'MULTI',
};
function mapFileViewerComponentProps(_a) {
var props = _a.props;
if (props.viewerType === ViewerTypes.MULTI) {
var _b = props, fileInfoList = _b.fileInfoList, currentIndex = _b.currentIndex;
return fileInfoList[currentIndex];
}
var fileInfo = props;
return {
name: fileInfo.name,
type: fileInfo.type,
url: fileInfo.url,
};
}
function DeleteButton(props) {
if (props.viewerType !== ViewerTypes.MULTI) {
var onDelete_1 = props.onDelete, isByMe = props.isByMe, disableDelete_1 = props.disableDelete, className = props.className;
return (isByMe)
? (React__default.createElement("div", { className: "sendbird-fileviewer__header__right__actions__delete ".concat(className) },
React__default.createElement(Icon, { className: disableDelete_1 ? 'disabled' : '', type: IconTypes.DELETE, fillColor: disableDelete_1 ? IconColors.GRAY : IconColors.ON_BACKGROUND_1, height: "24px", width: "24px", onClick: function (e) { if (!disableDelete_1) {
onDelete_1 === null || onDelete_1 === void 0 ? void 0 : onDelete_1(e);
} } })))
: React__default.createElement(React__default.Fragment, null);
}
return React__default.createElement(React__default.Fragment, null);
}
// this is a slider component that is used to navigate between images
function Slider(props) {
if (props.viewerType === ViewerTypes.MULTI) {
var onClickLeft_1 = props.onClickLeft, onClickRight_1 = props.onClickRight;
return (React__default.createElement("div", { className: "sendbird-file-viewer-slider" },
React__default.createElement("div", { className: "sendbird-file-viewer-arrow--left" },
React__default.createElement(Icon, { type: IconTypes.SLIDE_LEFT, fillColor: IconColors.ON_BACKGROUND_1, height: SLIDER_BUTTON_ICON_SIDE_LENGTH, width: SLIDER_BUTTON_ICON_SIDE_LENGTH, onClick: function (e) {
onClickLeft_1 === null || onClickLeft_1 === void 0 ? void 0 : onClickLeft_1();
e.stopPropagation();
} })),
React__default.createElement("div", { className: "sendbird-file-viewer-arrow--right" },
React__default.createElement(Icon, { type: IconTypes.SLIDE_LEFT, fillColor: IconColors.ON_BACKGROUND_1, height: SLIDER_BUTTON_ICON_SIDE_LENGTH, width: SLIDER_BUTTON_ICON_SIDE_LENGTH, onClick: function (e) {
e.stopPropagation();
onClickRight_1 === null || onClickRight_1 === void 0 ? void 0 : onClickRight_1();
} }))));
}
// return empty fragment if viewerType is not ViewerTypes.MULTI
return React__default.createElement(React__default.Fragment, null);
}
var FileViewerComponent = function (props) {
var profileUrl = props.profileUrl, nickname = props.nickname, onClose = props.onClose, onDownloadClick = props.onDownloadClick;
var _a = props, onClickLeft = _a.onClickLeft, onClickRight = _a.onClickRight;
var ref = useRef(null);
var onKeyDown = useKeyDown(ref, {
Escape: function (e) { return onClose === null || onClose === void 0 ? void 0 : onClose(e); },
ArrowLeft: function () { return onClickLeft === null || onClickLeft === void 0 ? void 0 : onClickLeft(); },
ArrowRight: function () { return onClickRight === null || onClickRight === void 0 ? void 0 : onClickRight(); },
});
var _b = mapFileViewerComponentProps({ props: props }), name = _b.name, type = _b.type, url = _b.url;
var stringSet = useContext(LocalizationContext).stringSet;
return (React__default.createElement(Modal, { onClose: onClose },
React__default.createElement("div", { className: "sendbird-fileviewer", "data-testid": "sendbird-fileviewer", onKeyDown: onKeyDown,
// to focus
tabIndex: 1, ref: ref },
React__default.createElement("div", { className: "sendbird-fileviewer__header" },
React__default.createElement("div", { className: "sendbird-fileviewer__header__left" },
React__default.createElement("div", { className: "sendbird-fileviewer__header__left__avatar" },
React__default.createElement(Avatar, { height: "32px", width: "32px", src: profileUrl })),
React__default.createElement("div", { className: "sendbird-fileviewer__header__left__text-container" },
React__default.createElement(Label, { className: "sendbird-fileviewer__header__left__filename", type: LabelTypography.H_2, color: LabelColors.ONBACKGROUND_1 }, name),
React__default.createElement(Label, { className: "sendbird-fileviewer__header__left__sender-name", type: LabelTypography.BODY_1, color: LabelColors.ONBACKGROUND_2 }, nickname))),
React__default.createElement("div", { className: "sendbird-fileviewer__header__right" },
isSupportedFileView(type) && (React__default.createElement("div", { className: "sendbird-fileviewer__header__right__actions" },
React__default.createElement("a", { className: "sendbird-fileviewer__header__right__actions__download", rel: "noopener noreferrer", href: url, target: "_blank", onClick: onDownloadClick },
React__default.createElement(Icon, { type: IconTypes.DOWNLOAD, fillColor: IconColors.ON_BACKGROUND_1, height: "24px", width: "24px" })),
React__default.createElement(DeleteButton, __assign({ className: 'sendbird-fileviewer__header__right__actions__delete' }, props)))),
React__default.createElement("div", { className: "sendbird-fileviewer__header__right__actions__close" },
React__default.createElement(Icon, { type: IconTypes.CLOSE, fillColor: IconColors.ON_BACKGROUND_1, height: "24px", width: "24px", onClick: function (e) { return onClose === null || onClose === void 0 ? void 0 : onClose(e); } })))),
React__default.createElement("div", { className: "sendbird-fileviewer__content" },
isVideo(type) && (React__default.createElement("video", { controls: true, className: "sendbird-fileviewer__content__video" },
React__default.createElement("source", { src: url, type: type }))),
isImage(type) && (React__default.createElement("img", { src: url, alt: name, className: props.viewerType === ViewerTypes.MULTI
? 'sendbird-fileviewer__content__img__multi'
: 'sendbird-fileviewer__content__img' })),
!isSupportedFileView(type) && (React__default.createElement("div", { className: "sendbird-fileviewer__content__unsupported" },
React__default.createElement(Label, { type: LabelTypography.H_1, color: LabelColors.ONBACKGROUND_1 }, (stringSet === null || stringSet === void 0 ? void 0 : stringSet.UI__FILE_VIEWER__UNSUPPORT) || 'Unsupported message'))),
React__default.createElement(Slider, __assign({}, props))))));
};
function FileViewer(_a) {
var _b, _c, _d;
var message = _a.message, _e = _a.statefulFileInfoList, statefulFileInfoList = _e === void 0 ? [] : _e, onClose = _a.onClose, _f = _a.isByMe, isByMe = _f === void 0 ? false : _f, onDelete = _a.onDelete, currentIndex = _a.currentIndex, onClickLeft = _a.onClickLeft, onClickRight = _a.onClickRight, onDownloadClick = _a.onDownloadClick;
if (isMultipleFilesMessage(message)) {
var castedMessage = message;
return (React__default.createElement(FileViewerComponent, { profileUrl: castedMessage.sender.profileUrl, nickname: castedMessage.sender.nickname, viewerType: ViewerTypes.MULTI, fileInfoList: statefulFileInfoList.filter(function (fileInfo) {
return fileInfo.url; // Caution: This assumes that defined url means file upload has completed.
}).map(function (fileInfo) {
return {
name: fileInfo.fileName || '',
type: fileInfo.mimeType || '',
url: fileInfo.url || '',
};
}), currentIndex: currentIndex || 0, onClickLeft: onClickLeft || noop, onClickRight: onClickRight || noop, onClose: onClose, onDownloadClick: onDownloadClick }));
}
else if (isFileMessage(message)) {
var castedMessage = message;
return createPortal((React__default.createElement(FileViewerComponent, { profileUrl: (_b = castedMessage.sender) === null || _b === void 0 ? void 0 : _b.profileUrl, nickname: (_c = castedMessage.sender) === null || _c === void 0 ? void 0 : _c.nickname, name: castedMessage.name, type: castedMessage.type, url: castedMessage === null || castedMessage === void 0 ? void 0 : castedMessage.url, isByMe: isByMe, disableDelete: (((_d = castedMessage.threadInfo) === null || _d === void 0 ? void 0 : _d.replyCount) || 0) > 0, onClose: onClose, onDelete: onDelete || noop, onDownloadClick: onDownloadClick })), document.getElementById(MODAL_ROOT));
}
return React__default.createElement(React__default.Fragment, null);
}
export { FileViewerComponent, FileViewer as default };
//# sourceMappingURL=FileViewer.js.map