UNPKG

xom-stream-chat-react

Version:

React components to create chat conversations or livestream style chat

1,428 lines (1,212 loc) 473 kB
'use strict'; Object.defineProperty(exports, '__esModule', { value: true }); function _interopDefault (ex) { return (ex && (typeof ex === 'object') && 'default' in ex) ? ex['default'] : ex; } var _extends = _interopDefault(require('@babel/runtime/helpers/extends')); var _classCallCheck = _interopDefault(require('@babel/runtime/helpers/classCallCheck')); var _createClass = _interopDefault(require('@babel/runtime/helpers/createClass')); var _possibleConstructorReturn = _interopDefault(require('@babel/runtime/helpers/possibleConstructorReturn')); var _getPrototypeOf = _interopDefault(require('@babel/runtime/helpers/getPrototypeOf')); var _assertThisInitialized = _interopDefault(require('@babel/runtime/helpers/assertThisInitialized')); var _inherits = _interopDefault(require('@babel/runtime/helpers/inherits')); var _defineProperty = _interopDefault(require('@babel/runtime/helpers/defineProperty')); var React = require('react'); var React__default = _interopDefault(React); var ReactPlayer = _interopDefault(require('react-player')); var PropTypes = _interopDefault(require('prop-types')); var sanitizeUrl = require('@braintree/sanitize-url'); var Carousel = require('react-images'); var Carousel__default = _interopDefault(Carousel); var reactFileUtils = require('react-file-utils'); var prettybytes = _interopDefault(require('pretty-bytes')); var _toConsumableArray = _interopDefault(require('@babel/runtime/helpers/toConsumableArray')); var _regeneratorRuntime = _interopDefault(require('@babel/runtime/regenerator')); var _asyncToGenerator = _interopDefault(require('@babel/runtime/helpers/asyncToGenerator')); var uuidv4 = _interopDefault(require('uuid/v4')); var Immutable = _interopDefault(require('seamless-immutable')); var Visibility = _interopDefault(require('visibilityjs')); var streamChat = require('stream-chat'); var anchorme = _interopDefault(require('anchorme')); var emojiRegex = _interopDefault(require('emoji-regex')); var ReactMarkdown = _interopDefault(require('react-markdown/with-html')); var truncate = _interopDefault(require('lodash/truncate')); var data = _interopDefault(require('emoji-mart/data/all.json')); var emojiMart = require('emoji-mart'); var _typeof = _interopDefault(require('@babel/runtime/helpers/typeof')); var _slicedToArray = _interopDefault(require('@babel/runtime/helpers/slicedToArray')); var getCaretCoordinates = _interopDefault(require('textarea-caret')); var CustomEvent = _interopDefault(require('custom-event')); var Textarea = _interopDefault(require('react-textarea-autosize')); var uniq = _interopDefault(require('lodash/uniq')); var moment = _interopDefault(require('moment')); var debounce = _interopDefault(require('lodash/debounce')); var throttle = _interopDefault(require('lodash/throttle')); var uniqBy = _interopDefault(require('lodash.uniqby')); var deepequal = _interopDefault(require('deep-equal')); var _objectWithoutProperties = _interopDefault(require('@babel/runtime/helpers/objectWithoutProperties')); /** * SafeAnchor - In all ways similar to a regular anchor tag. * The difference is that it sanitizes the href value and prevents XSS * @extends PureComponent */ var SafeAnchor = /*#__PURE__*/ function (_React$PureComponent) { _inherits(SafeAnchor, _React$PureComponent); function SafeAnchor() { _classCallCheck(this, SafeAnchor); return _possibleConstructorReturn(this, _getPrototypeOf(SafeAnchor).apply(this, arguments)); } _createClass(SafeAnchor, [{ key: "render", value: function render() { var href = sanitizeUrl.sanitizeUrl(this.props.href); return React__default.createElement("a", _extends({}, this.props, { href: href }), this.props.children); } }]); return SafeAnchor; }(React__default.PureComponent); var giphyLogo = ""; /** * Card - Simple Card Layout * * @example ./docs/Card.md * @extends PureComponent */ var Card = /*#__PURE__*/ function (_React$PureComponent) { _inherits(Card, _React$PureComponent); function Card() { var _getPrototypeOf2; var _this; _classCallCheck(this, Card); for (var _len = arguments.length, args = new Array(_len), _key = 0; _key < _len; _key++) { args[_key] = arguments[_key]; } _this = _possibleConstructorReturn(this, (_getPrototypeOf2 = _getPrototypeOf(Card)).call.apply(_getPrototypeOf2, [this].concat(args))); _defineProperty(_assertThisInitialized(_this), "trimUrl", function (url) { var trimmedUrl; if (url !== undefined && url !== null) { trimmedUrl = url.replace(/^(?:https?:\/\/)?(?:www\.)?/i, '').split('/')[0]; } return trimmedUrl; }); return _this; } _createClass(Card, [{ key: "render", value: function render() { var _this$props = this.props, title = _this$props.title, title_link = _this$props.title_link, text = _this$props.text, type = _this$props.type, image_url = _this$props.image_url, thumb_url = _this$props.thumb_url, og_scrape_url = _this$props.og_scrape_url; var image = thumb_url || image_url; if (!title && !title_link && !image) { return React__default.createElement("div", { className: "str-chat__message-attachment-card str-chat__message-attachment-card--".concat(type) }, React__default.createElement("div", { className: "str-chat__message-attachment-card--content" }, React__default.createElement("div", { className: "str-chat__message-attachment-card--text" }, "this content could not be displayed"))); } if (!title_link && !og_scrape_url) { return null; } return React__default.createElement("div", { className: "str-chat__message-attachment-card str-chat__message-attachment-card--".concat(type) }, image && React__default.createElement("div", { className: "str-chat__message-attachment-card--header" }, React__default.createElement("img", { src: image, alt: image })), React__default.createElement("div", { className: "str-chat__message-attachment-card--content" }, React__default.createElement("div", { className: "str-chat__message-attachment-card--flex" }, title && React__default.createElement("div", { className: "str-chat__message-attachment-card--title" }, title), text && React__default.createElement("div", { className: "str-chat__message-attachment-card--text" }, text), (title_link || og_scrape_url) && React__default.createElement(SafeAnchor, { href: title_link || og_scrape_url, target: "_blank", rel: "noopener noreferrer", className: "str-chat__message-attachment-card--url" }, this.trimUrl(title_link || og_scrape_url))), type === 'giphy' && React__default.createElement("img", { className: "str-chat__message-attachment-card__giphy-logo", src: giphyLogo, alt: "giphy logo" }))); } }]); return Card; }(React__default.PureComponent); _defineProperty(Card, "propTypes", { /** Title returned by the OG scraper */ title: PropTypes.string, /** Link returned by the OG scraper */ title_link: PropTypes.string, /** The scraped url, used as a fallback if the OG-data doesn't include a link */ og_scrape_url: PropTypes.string, /** The url of the full sized image */ image_url: PropTypes.string, /** The url for thumbnail sized image*/ thumb_url: PropTypes.string, /** Description returned by the OG scraper */ text: PropTypes.string }); /** * Image - Small wrapper around an image tag, supports thumbnails * * @example ./docs/Image.md * @extends PureComponent */ var Image = /*#__PURE__*/ function (_React$PureComponent) { _inherits(Image, _React$PureComponent); function Image() { var _getPrototypeOf2; var _this; _classCallCheck(this, Image); for (var _len = arguments.length, args = new Array(_len), _key = 0; _key < _len; _key++) { args[_key] = arguments[_key]; } _this = _possibleConstructorReturn(this, (_getPrototypeOf2 = _getPrototypeOf(Image)).call.apply(_getPrototypeOf2, [this].concat(args))); _defineProperty(_assertThisInitialized(_this), "state", { modalIsOpen: false, currentIndex: 0 }); _defineProperty(_assertThisInitialized(_this), "toggleModal", function () { _this.setState(function (state) { return { modalIsOpen: !state.modalIsOpen }; }); }); return _this; } _createClass(Image, [{ key: "render", value: function render() { var _this$props = this.props, image_url = _this$props.image_url, thumb_url = _this$props.thumb_url, fallback = _this$props.fallback; var formattedArray = [{ src: image_url || thumb_url }]; return React__default.createElement(React__default.Fragment, null, React__default.createElement("img", { className: "str-chat__message-attachment--img", onClick: this.toggleModal, src: thumb_url || image_url, alt: fallback }), React__default.createElement(Carousel.ModalGateway, null, this.state.modalIsOpen ? React__default.createElement(Carousel.Modal, { onClose: this.toggleModal }, React__default.createElement(Carousel__default, { views: formattedArray })) : null)); } }]); return Image; }(React__default.PureComponent); _defineProperty(Image, "propTypes", { /** The full size image url */ image_url: PropTypes.string, /** The thumb url */ thumb_url: PropTypes.string, /** The text fallback for the image */ fallback: PropTypes.string }); /** * AttachmentActions - The actions you can take on an attachment * * @example ./docs/AttachmentActions.md * @extends PureComponent */ var AttachmentActions = /*#__PURE__*/ function (_React$PureComponent) { _inherits(AttachmentActions, _React$PureComponent); function AttachmentActions() { _classCallCheck(this, AttachmentActions); return _possibleConstructorReturn(this, _getPrototypeOf(AttachmentActions).apply(this, arguments)); } _createClass(AttachmentActions, [{ key: "render", value: function render() { var _this = this; var _this$props = this.props, text = _this$props.text, id = _this$props.id, actions = _this$props.actions, actionHandler = _this$props.actionHandler; return React__default.createElement("div", { className: "str-chat__message-attachment-actions" }, React__default.createElement("form", { className: "str-chat__message-attachment-actions-form" }, React__default.createElement("span", { key: 0 }, text), actions.map(function (action) { return React__default.createElement("button", { className: "str-chat__message-attachment-actions-button str-chat__message-attachment-actions-button--".concat(action.style), key: "".concat(id, "-").concat(action.value), "data-value": action.value, onClick: actionHandler.bind(_this, action.name, action.value) }, action.text); }))); } }]); return AttachmentActions; }(React__default.PureComponent); _defineProperty(AttachmentActions, "propTypes", { /** Unique id for action button key. Key is generated by concatenating this id with action value - {`${id}-${action.value}`} */ id: PropTypes.string.isRequired, /** The text for the form input */ text: PropTypes.string, /** A list of actions */ actions: PropTypes.array.isRequired, /** * * Handler for actions. Actions in combination with attachments can be used to build [commands](https://getstream.io/chat/docs/#channel_commands). * * @param name {string} Name of action * @param value {string} Value of action * @param event Dom event that triggered this handler */ actionHandler: PropTypes.func.isRequired }); var Audio = /*#__PURE__*/ function (_React$Component) { _inherits(Audio, _React$Component); function Audio(props) { var _this; _classCallCheck(this, Audio); _this = _possibleConstructorReturn(this, _getPrototypeOf(Audio).call(this, props)); _defineProperty(_assertThisInitialized(_this), "playAudio", function () { if (_this.audioRef.current !== null) { _this.audioRef.current.pause(); _this.updateProgress(); _this.setState({ playing: true, updateProgress: setInterval(_this.updateProgress, 500) }); //$FlowFixMe _this.audioRef.current.play(); } }); _defineProperty(_assertThisInitialized(_this), "pauseAudio", function () { if (_this.audioRef.current !== null) { _this.audioRef.current.pause(); } _this.setState({ playing: false }); window.clearInterval(_this.state.updateProgress); }); _defineProperty(_assertThisInitialized(_this), "updateProgress", function () { if (_this.audioRef.current !== null) { var position = _this.audioRef.current.currentTime; var duration = _this.audioRef.current.duration; var progress = 100 / duration * position; _this.setState({ progress: progress }); if (position === duration) { _this.pauseAudio(); } } }); _this.state = { open: false, playing: false, progress: 0, updateProgress: null }; _this.audioRef = React.createRef(); return _this; } _createClass(Audio, [{ key: "componentWillUnmount", value: function componentWillUnmount() { window.clearInterval(this.state.updateProgress); } }, { key: "render", value: function render() { var _this2 = this; var og = this.props.og; var url = og.asset_url; var image = og.image_url; return React.createElement("div", { className: "str-chat__audio" }, React.createElement("div", { className: "str-chat__audio__wrapper" }, React.createElement("audio", { ref: this.audioRef }, React.createElement("source", { src: url, type: "audio/mp3" })), React.createElement("div", { className: "str-chat__audio__image" }, React.createElement("div", { className: "str-chat__audio__image--overlay" }, !this.state.playing ? React.createElement("div", { onClick: function onClick() { return _this2.playAudio(); }, className: "str-chat__audio__image--button" }, React.createElement("svg", { width: "40", height: "40", viewBox: "0 0 64 64", xmlns: "http://www.w3.org/2000/svg" }, React.createElement("path", { d: "M32 58c14.36 0 26-11.64 26-26S46.36 6 32 6 6 17.64 6 32s11.64 26 26 26zm0 6C14.327 64 0 49.673 0 32 0 14.327 14.327 0 32 0c17.673 0 32 14.327 32 32 0 17.673-14.327 32-32 32zm13.237-28.412L26.135 45.625a3.27 3.27 0 0 1-4.426-1.4 3.319 3.319 0 0 1-.372-1.47L21 23.36c-.032-1.823 1.41-3.327 3.222-3.358a3.263 3.263 0 0 1 1.473.322l19.438 9.36a3.311 3.311 0 0 1 .103 5.905z", fillRule: "nonzero" }))) : React.createElement("div", { onClick: function onClick() { return _this2.pauseAudio(); }, className: "str-chat__audio__image--button" }, React.createElement("svg", { width: "40", height: "40", viewBox: "0 0 64 64", xmlns: "http://www.w3.org/2000/svg" }, React.createElement("path", { d: "M32 58.215c14.478 0 26.215-11.737 26.215-26.215S46.478 5.785 32 5.785 5.785 17.522 5.785 32 17.522 58.215 32 58.215zM32 64C14.327 64 0 49.673 0 32 0 14.327 14.327 0 32 0c17.673 0 32 14.327 32 32 0 17.673-14.327 32-32 32zm-7.412-45.56h2.892a2.17 2.17 0 0 1 2.17 2.17v23.865a2.17 2.17 0 0 1-2.17 2.17h-2.892a2.17 2.17 0 0 1-2.17-2.17V20.61a2.17 2.17 0 0 1 2.17-2.17zm12.293 0h2.893a2.17 2.17 0 0 1 2.17 2.17v23.865a2.17 2.17 0 0 1-2.17 2.17h-2.893a2.17 2.17 0 0 1-2.17-2.17V20.61a2.17 2.17 0 0 1 2.17-2.17z", fillRule: "nonzero" })))), React.createElement("img", { src: image, alt: "".concat(og.description) })), React.createElement("div", { className: "str-chat__audio__content" }, React.createElement("span", { className: "str-chat__audio__content--title" }, React.createElement("strong", null, og.title)), React.createElement("span", { className: "str-chat__audio__content--subtitle" }, og.text), React.createElement("div", { className: "str-chat__audio__content--progress" }, React.createElement("div", { style: { width: "".concat(this.state.progress, "%") } }))))); } }]); return Audio; }(React.Component); _defineProperty(Audio, "propTypes", { /** Attachment object of audio type */ og: PropTypes.object }); /** * Attachment - The message attachment * * @example ./docs/Attachment.md * @extends PureComponent */ var Attachment = /*#__PURE__*/ function (_PureComponent) { _inherits(Attachment, _PureComponent); function Attachment() { var _getPrototypeOf2; var _this; _classCallCheck(this, Attachment); for (var _len = arguments.length, args = new Array(_len), _key = 0; _key < _len; _key++) { args[_key] = arguments[_key]; } _this = _possibleConstructorReturn(this, (_getPrototypeOf2 = _getPrototypeOf(Attachment)).call.apply(_getPrototypeOf2, [this].concat(args))); _defineProperty(_assertThisInitialized(_this), "attachmentRef", React__default.createRef()); _defineProperty(_assertThisInitialized(_this), "renderAttachmentActions", function (a) { return React__default.createElement(AttachmentActions, _extends({ key: 'key-actions-' + a.id }, a, { actionHandler: _this.props.actionHandler })); }); _defineProperty(_assertThisInitialized(_this), "renderAttachment", function (a) { return React__default.createElement("div", { className: "str-chat__attachment", key: "key-image-".concat(a.id) }, React__default.createElement(Card, _extends({}, a, { key: "key-card-".concat(a.id) })), _this.renderAttachmentActions(a)); }); return _this; } _createClass(Attachment, [{ key: "attachmentType", value: function attachmentType(a) { var type, extra; if (a.actions && a.actions.length > 0) { extra = 'actions'; } if (a.type === 'giphy' || a.type === 'imgur') { type = 'card'; } else if (a.type === 'image' && (a.title_link || a.og_scrape_url)) { type = 'card'; } else if (a.type === 'image') { type = 'image'; } else if (a.type === 'file') { type = 'file'; } else if (a.type === 'audio') { type = 'audio'; } else if (a.type === 'video') { type = 'media'; } else { type = 'card'; extra = 'no-image'; } return { type: type, extra: extra }; } }, { key: "render", value: function render() { var a = this.props.attachment; if (!a) { return null; } var _this$attachmentType = this.attachmentType(a), type = _this$attachmentType.type, extra = _this$attachmentType.extra; if (type === 'card' && !a.title_link && !a.og_scrape_url) { return null; } var results = []; if (type === 'image') { if (a.actions && a.actions.length) { results.push(React__default.createElement("div", { className: "str-chat__attachment", key: "key-image-".concat(a.id) }, React__default.createElement(Image, a), this.renderAttachmentActions(a))); } else { results.push(React__default.createElement(Image, _extends({}, a, { key: "key-image-".concat(a.id) }))); } } else if (type === 'file') { a.asset_url && results.push(React__default.createElement("div", { className: "str-chat__message-attachment-file--item", key: "key-file-".concat(a.id) }, React__default.createElement(reactFileUtils.FileIcon, { mimeType: a.mime_type, filename: a.title, big: true, size: 30 }), React__default.createElement("div", { className: "str-chat__message-attachment-file--item-text" }, React__default.createElement(SafeAnchor, { href: a.asset_url, download: true }, a.title), a.file_size && React__default.createElement("span", null, prettybytes(a.file_size))))); } else if (type === 'audio') { results.push(React__default.createElement("div", { className: "str-chat__attachment", key: "key-video-".concat(a.id) }, React__default.createElement(Audio, { og: a }))); } else if (type === 'media') { if (a.actions && a.actions.length) { results.push(React__default.createElement("div", { className: "str-chat__attachment", key: "key-video-".concat(a.id) }, React__default.createElement("div", { className: "str-chat__player-wrapper" }, React__default.createElement(ReactPlayer, { className: "react-player", url: a.asset_url, width: "100%", height: "100%", controls: true })), this.renderAttachmentActions(a))); } else { results.push(React__default.createElement("div", { className: "str-chat__player-wrapper", key: "key-video-".concat(a.id) }, React__default.createElement(ReactPlayer, { className: "react-player", url: a.asset_url, width: "100%", height: "100%", controls: true }))); } } else { if (a.actions && a.actions.length) { results.push(this.renderAttachment(a)); } else { results.push(React__default.createElement(Card, _extends({}, a, { key: "key-card-".concat(a.id) }))); } } if (results.length === 0) return null; return React__default.createElement("div", { className: "str-chat__message-attachment str-chat__message-attachment--".concat(type, " str-chat__message-attachment--").concat(a.type, " str-chat__message-attachment--").concat(type, "--").concat(extra), ref: this.attachmentRef }, results); } }]); return Attachment; }(React.PureComponent); _defineProperty(Attachment, "propTypes", { /** * The attachment to render * @see See [Attachment structure](https://getstream.io/chat/docs/#message_format) * * */ attachment: PropTypes.object.isRequired, /** * * Handler for actions. Actions in combination with attachments can be used to build [commands](https://getstream.io/chat/docs/#channel_commands). * * @param name {string} Name of action * @param value {string} Value of action * @param event Dom event that triggered this handler */ actionHandler: PropTypes.func.isRequired }); Attachment.propTypes = {}; /** * Avatar - A round avatar image with fallback to username's first letter * * @example ./docs/Avatar.md * @extends PureComponent */ var Avatar = /*#__PURE__*/ function (_React$PureComponent) { _inherits(Avatar, _React$PureComponent); function Avatar() { var _getPrototypeOf2; var _this; _classCallCheck(this, Avatar); for (var _len = arguments.length, args = new Array(_len), _key = 0; _key < _len; _key++) { args[_key] = arguments[_key]; } _this = _possibleConstructorReturn(this, (_getPrototypeOf2 = _getPrototypeOf(Avatar)).call.apply(_getPrototypeOf2, [this].concat(args))); _defineProperty(_assertThisInitialized(_this), "state", { errored: false, loaded: false }); _defineProperty(_assertThisInitialized(_this), "getInitials", function (name) { return name ? name.split(' ').slice(0, 1).map(function (name) { return name.charAt(0); }) : null; }); _defineProperty(_assertThisInitialized(_this), "onLoad", function () { _this.setState({ loaded: true }); }); _defineProperty(_assertThisInitialized(_this), "onError", function () { _this.setState({ errored: true }); }); return _this; } _createClass(Avatar, [{ key: "componentDidUpdate", value: function componentDidUpdate(prevProps) { if (prevProps.image !== this.props.image) { this.setState({ loaded: false, errored: false }); } } }, { key: "render", value: function render() { var _this$props = this.props, size = _this$props.size, name = _this$props.name, shape = _this$props.shape, image = _this$props.image; var initials = this.getInitials(name); return React__default.createElement("div", { className: "str-chat__avatar str-chat__avatar--".concat(shape), title: name, style: { width: size, height: size, flexBasis: size, lineHeight: size + 'px', fontSize: size / 2 } }, image && !this.state.errored ? React__default.createElement("img", { src: image, alt: initials, className: 'str-chat__avatar-image' + (this.state.loaded ? ' str-chat__avatar-image--loaded' : ''), style: { width: size, height: size, flexBasis: size, objectFit: 'cover' }, onLoad: this.onLoad, onError: this.onError }) : React__default.createElement("div", { className: "str-chat__avatar-fallback" }, initials)); } }]); return Avatar; }(React__default.PureComponent); _defineProperty(Avatar, "propTypes", { /** image url */ image: PropTypes.string, /** name of the picture, used for title tag fallback */ name: PropTypes.string, /** shape of the avatar, circle, rounded or square */ shape: PropTypes.oneOf(['circle', 'rounded', 'square']), /** size in pixels */ size: PropTypes.number }); _defineProperty(Avatar, "defaultProps", { size: 32, shape: 'circle' }); function ownKeys(object, enumerableOnly) { var keys = Object.keys(object); if (Object.getOwnPropertySymbols) { var symbols = Object.getOwnPropertySymbols(object); if (enumerableOnly) symbols = symbols.filter(function (sym) { return Object.getOwnPropertyDescriptor(object, sym).enumerable; }); keys.push.apply(keys, symbols); } return keys; } function _objectSpread(target) { for (var i = 1; i < arguments.length; i++) { var source = arguments[i] != null ? arguments[i] : {}; if (i % 2) { ownKeys(Object(source), true).forEach(function (key) { _defineProperty(target, key, source[key]); }); } else if (Object.getOwnPropertyDescriptors) { Object.defineProperties(target, Object.getOwnPropertyDescriptors(source)); } else { ownKeys(Object(source)).forEach(function (key) { Object.defineProperty(target, key, Object.getOwnPropertyDescriptor(source, key)); }); } } return target; } var ChatContext = React__default.createContext({ client: null }); function withChatContext(OriginalComponent) { var ContextAwareComponent = function ContextComponent(props) { return React__default.createElement(ChatContext.Consumer, null, function (context) { var mergedProps = _objectSpread({}, context, {}, props); return React__default.createElement(OriginalComponent, mergedProps); }); }; ContextAwareComponent.displayName = OriginalComponent.displayName || OriginalComponent.name || 'Component'; ContextAwareComponent.displayName = ContextAwareComponent.displayName.replace('Base', ''); return ContextAwareComponent; } var ChannelContext = React__default.createContext({}); function withChannelContext(OriginalComponent) { var ContextAwareComponent = function ContextComponent(props) { return React__default.createElement(ChannelContext.Consumer, null, function (channelContext) { return React__default.createElement(OriginalComponent, _extends({}, channelContext, props)); }); }; ContextAwareComponent.displayName = OriginalComponent.displayName || OriginalComponent.name || 'Component'; ContextAwareComponent.displayName = ContextAwareComponent.displayName.replace('Base', ''); return ContextAwareComponent; } /** * LoadingIndicator - Just a simple loading spinner.. * * @example ./docs/LoadingIndicator.md * @extends PureComponent */ var LoadingIndicator = /*#__PURE__*/ function (_React$PureComponent) { _inherits(LoadingIndicator, _React$PureComponent); function LoadingIndicator() { var _getPrototypeOf2; var _this; _classCallCheck(this, LoadingIndicator); for (var _len = arguments.length, args = new Array(_len), _key = 0; _key < _len; _key++) { args[_key] = arguments[_key]; } _this = _possibleConstructorReturn(this, (_getPrototypeOf2 = _getPrototypeOf(LoadingIndicator)).call.apply(_getPrototypeOf2, [this].concat(args))); _defineProperty(_assertThisInitialized(_this), "stopRef", React__default.createRef()); return _this; } _createClass(LoadingIndicator, [{ key: "render", value: function render() { var _this$props = this.props, size = _this$props.size, color = _this$props.color; return React__default.createElement("div", { className: 'str-chat__loading-indicator ' + color, style: { width: size, height: size } }, React__default.createElement("svg", { width: size, height: size, viewBox: "0 0 30 30", xmlns: "http://www.w3.org/2000/svg" }, React__default.createElement("defs", null, React__default.createElement("linearGradient", { x1: "50%", y1: "0%", x2: "50%", y2: "100%", id: "a" }, React__default.createElement("stop", { stopColor: "#FFF", stopOpacity: "0", offset: "0%" }), React__default.createElement("stop", { ref: this.stopRef, offset: "100%", stopColor: color, stopOpacity: "1", style: { stopColor: color } }))), React__default.createElement("path", { d: "M2.518 23.321l1.664-1.11A12.988 12.988 0 0 0 15 28c7.18 0 13-5.82 13-13S22.18 2 15 2V0c8.284 0 15 6.716 15 15 0 8.284-6.716 15-15 15-5.206 0-9.792-2.652-12.482-6.679z", fill: "url(#a)", fillRule: "evenodd" }))); } }]); return LoadingIndicator; }(React__default.PureComponent); _defineProperty(LoadingIndicator, "propTypes", { /** The size of the loading icon */ size: PropTypes.number, /** Set the color of the LoadingIndicator */ color: PropTypes.string }); _defineProperty(LoadingIndicator, "defaultProps", { size: 15, color: '#006CFF' }); /** * LoadingErrorIndicator - UI component for error indicator in Channel. * * @example ./docs/LoadingErrorIndicator.md * @extends PureComponent */ var LoadingErrorIndicator = /*#__PURE__*/ function (_React$PureComponent) { _inherits(LoadingErrorIndicator, _React$PureComponent); function LoadingErrorIndicator() { _classCallCheck(this, LoadingErrorIndicator); return _possibleConstructorReturn(this, _getPrototypeOf(LoadingErrorIndicator).apply(this, arguments)); } _createClass(LoadingErrorIndicator, [{ key: "render", value: function render() { if (!this.props.error) return null; return React__default.createElement("div", null, "Error: ", this.props.error.message); } }]); return LoadingErrorIndicator; }(React__default.PureComponent); _defineProperty(LoadingErrorIndicator, "propTypes", { /** Error object */ error: PropTypes.oneOfType([PropTypes.shape({ message: PropTypes.string }), PropTypes.bool]) }); _defineProperty(LoadingErrorIndicator, "defaultProps", { error: false }); function ownKeys$1(object, enumerableOnly) { var keys = Object.keys(object); if (Object.getOwnPropertySymbols) { var symbols = Object.getOwnPropertySymbols(object); if (enumerableOnly) symbols = symbols.filter(function (sym) { return Object.getOwnPropertyDescriptor(object, sym).enumerable; }); keys.push.apply(keys, symbols); } return keys; } function _objectSpread$1(target) { for (var i = 1; i < arguments.length; i++) { var source = arguments[i] != null ? arguments[i] : {}; if (i % 2) { ownKeys$1(Object(source), true).forEach(function (key) { _defineProperty(target, key, source[key]); }); } else if (Object.getOwnPropertyDescriptors) { Object.defineProperties(target, Object.getOwnPropertyDescriptors(source)); } else { ownKeys$1(Object(source)).forEach(function (key) { Object.defineProperty(target, key, Object.getOwnPropertyDescriptor(source, key)); }); } } return target; } var emojiSetDef = { spriteUrl: 'https://getstream.imgix.net/images/emoji-sprite.png', size: 20, sheetColumns: 2, sheetRows: 3, sheetSize: 64 }; var commonEmoji = { emoticons: [], short_names: [], custom: true }; var defaultMinimalEmojis = [_objectSpread$1({ id: 'like', name: 'like', colons: ':+1:', sheet_x: 0, sheet_y: 0 }, commonEmoji, {}, emojiSetDef), _objectSpread$1({ id: 'love', name: 'love', colons: ':heart:', sheet_x: 1, sheet_y: 2 }, commonEmoji, {}, emojiSetDef), _objectSpread$1({ id: 'haha', name: 'haha', colons: ':joy:', sheet_x: 1, sheet_y: 0 }, commonEmoji, {}, emojiSetDef), _objectSpread$1({ id: 'wow', name: 'wow', colons: ':astonished:', sheet_x: 0, sheet_y: 2 }, commonEmoji, {}, emojiSetDef), _objectSpread$1({ id: 'sad', name: 'sad', colons: ':pensive:', sheet_x: 0, sheet_y: 1 }, commonEmoji, {}, emojiSetDef), _objectSpread$1({ id: 'angry', name: 'angry', colons: ':angry:', sheet_x: 1, sheet_y: 1 }, commonEmoji, {}, emojiSetDef)]; var d = Object.assign({}, data); d.emojis = {}; // use this only for small lists like in ReactionSelector var emojiData = d; var isOnlyEmojis = function isOnlyEmojis(text) { if (!text) return false; var noEmojis = text.replace(emojiRegex(), ''); var noSpace = noEmojis.replace(/[\s\n]/gm, ''); return !noSpace; }; var isPromise = function isPromise(thing) { return thing && typeof thing.then === 'function'; }; var byDate = function byDate(a, b) { return a.created_at - b.created_at; }; // https://stackoverflow.com/a/29234240/7625485 var formatArray = function formatArray(dict) { var arr2 = Object.keys(dict); var arr3 = []; arr2.forEach(function (item, i) { return arr3.push(dict[arr2[i]].user.name || dict[arr2[i]].user.id); }); var outStr = ''; if (arr3.length === 1) { outStr = arr3[0] + ' is typing...'; } else if (arr3.length === 2) { //joins all with "and" but =no commas //example: "bob and sam" outStr = arr3.join(' and ') + ' are typing...'; } else if (arr3.length > 2) { //joins all with commas, but last one gets ", and" (oxford comma!) //example: "bob, joe, and sam" outStr = arr3.slice(0, -1).join(', ') + ', and ' + arr3.slice(-1) + ' are typing...'; } return outStr; }; var renderText = function renderText(message) { // take the @ mentions and turn them into markdown? // translate links var text = message.text; var mentioned_users = message.mentioned_users; if (!text) { return; } var allowed = ['html', 'root', 'text', 'break', 'paragraph', 'emphasis', 'strong', 'link', 'list', 'listItem', 'code', 'inlineCode', 'blockquote']; var urls = anchorme(text, { list: true }); var _iteratorNormalCompletion = true; var _didIteratorError = false; var _iteratorError = undefined; try { for (var _iterator = urls[Symbol.iterator](), _step; !(_iteratorNormalCompletion = (_step = _iterator.next()).done); _iteratorNormalCompletion = true) { var urlInfo = _step.value; var displayLink = truncate(urlInfo.encoded.replace(/^(www\.)/, ''), { length: 20, omission: '...' }); var _mkdown = "[".concat(displayLink, "](").concat(urlInfo.protocol).concat(urlInfo.encoded, ")"); text = text.replace(urlInfo.raw, _mkdown); } } catch (err) { _didIteratorError = true; _iteratorError = err; } finally { try { if (!_iteratorNormalCompletion && _iterator.return != null) { _iterator.return(); } } finally { if (_didIteratorError) { throw _iteratorError; } } } var newText = text; if (mentioned_users && mentioned_users.length) { for (var i = 0; i < mentioned_users.length; i++) { var username = mentioned_users[i].name || mentioned_users[i].id; var mkdown = "**@".concat(username, "**"); var re = new RegExp("@".concat(username), 'g'); newText = newText.replace(re, mkdown); } } return React__default.createElement(ReactMarkdown, { allowedTypes: allowed, source: newText, linkTarget: "_blank", plugins: [], escapeHtml: true, skipHtml: false }); }; // https://stackoverflow.com/a/6860916/2570866 function generateRandomId() { // prettier-ignore return S4() + S4() + "-" + S4() + "-" + S4() + "-" + S4() + "-" + S4() + S4() + S4(); } function S4() { return ((1 + Math.random()) * 0x10000 | 0).toString(16).substring(1); } var smartRender = function smartRender(ElementOrComponentOrLiteral, props, fallback) { if (ElementOrComponentOrLiteral === undefined) { ElementOrComponentOrLiteral = fallback; } if (React__default.isValidElement(ElementOrComponentOrLiteral)) { // Flow cast through any, to make flow believe it's a React.Element var element = ElementOrComponentOrLiteral; // eslint-disable-line return element; } // Flow cast through any to remove React.Element after previous check var ComponentOrLiteral = ElementOrComponentOrLiteral; if (typeof ComponentOrLiteral === 'string' || typeof ComponentOrLiteral === 'number' || typeof ComponentOrLiteral === 'boolean' || ComponentOrLiteral == null) { return ComponentOrLiteral; } return React__default.createElement(ComponentOrLiteral, props); }; var MESSAGE_ACTIONS = { edit: 'edit', delete: 'delete', flag: 'flag', mute: 'mute' }; /** * MessageActionsBox - A component for taking action on a message * * @example ./docs/MessageActionsBox.md * @extends PureComponent */ var MessageActionsBox = /*#__PURE__*/ function (_React$Component) { _inherits(MessageActionsBox, _React$Component); function MessageActionsBox() { var _getPrototypeOf2; var _this; _classCallCheck(this, MessageActionsBox); for (var _len = arguments.length, args = new Array(_len), _key = 0; _key < _len; _key++) { args[_key] = arguments[_key]; } _this = _possibleConstructorReturn(this, (_getPrototypeOf2 = _getPrototypeOf(MessageActionsBox)).call.apply(_getPrototypeOf2, [this].concat(args))); _defineProperty(_assertThisInitialized(_this), "actionsBoxRef", React__default.createRef()); _defineProperty(_assertThisInitialized(_this), "state", { reverse: false, rect: null }); return _this; } _createClass(MessageActionsBox, [{ key: "componentDidMount", value: function componentDidMount() {} }, { key: "componentDidUpdate", value: function () { var _componentDidUpdate = _asyncToGenerator( /*#__PURE__*/ _regeneratorRuntime.mark(function _callee(prevProps) { var ml; return _regeneratorRuntime.wrap(function _callee$(_context) { while (1) { switch (_context.prev = _context.next) { case 0: if (!(!prevProps.open && this.props.open)) { _context.next = 6; break; } if (!(this.state.rect === null)) { _context.next = 4; break; } _context.next = 4; return this.setState({ rect: this.actionsBoxRef.current.getBoundingClientRect() }); case 4: ml = this.props.messageListRect; if (this.props.mine) { this.setState({ reverse: this.state.rect.left < ml.left ? true : false }); } else if (!this.props.mine) { this.setState({ reverse: this.state.rect.right + 5 > ml.right ? true : false }); } case 6: case "end": return _context.stop(); } } }, _callee, this); })); function componentDidUpdate(_x) { return _componentDidUpdate.apply(this, arguments); } return componentDidUpdate; }() }, { key: "render", value: function render() { var _this$props = this.props, handleFlag = _this$props.handleFlag, handleMute = _this$props.handleMute, handleEdit = _this$props.handleEdit, handleDelete = _this$props.handleDelete, getMessageActions = _this$props.getMessageActions; var messageActions = getMessageActions(); return React__default.createElement("div", { className: "str-chat__message-actions-box\n ".concat(this.props.open ? 'str-chat__message-actions-box--open' : '', "\n ").concat(this.props.mine ? 'str-chat__message-actions-box--mine' : '', "\n ").concat(this.state.reverse ? 'str-chat__message-actions-box--reverse' : '', "\n "), ref: this.actionsBoxRef }, React__default.createElement("ul", { className: "str-chat__message-actions-list" }, messageActions.indexOf(MESSAGE_ACTIONS.flag) > -1 && React__default.createElement("button", { onClick: handleFlag }, React__default.createElement("li", { className: "str-chat__message-actions-list-item" }, "Flag")), messageActions.indexOf(MESSAGE_ACTIONS.mute) > -1 && React__default.createElement("button", { onClick: handleMute }, React__default.createElement("li", { className: "str-chat__message-actions-list-item" }, "Mute")), messageActions.indexOf(MESSAGE_ACTIONS.edit) > -1 && React__default.createElement("button", { onClick: handleEdit }, React__default.createElement("li", { className: "str-chat__message-actions-list-item" }, "Edit Message")), messageActions.indexOf(MESSAGE_ACTIONS.delete) > -1 && React__default.createElement("button", { onClick: handleDelete }, React__default.createElement("li", { className: "str-chat__message-actions-list-item" }, "Delete")))); } }]); return MessageActionsBox; }(React__default.Component); _defineProperty(MessageActionsBox, "propTypes", { /** If the message actions box should be open or not */ open: PropTypes.bool.isRequired, /** * @deprecated * * The message component, most logic is delegated to this component and MessageActionsBox uses the following functions explicitly: * `handleFlag`, `handleMute`, `handleEdit`, `handleDelete`, `canDeleteMessage`, `canEditMessage`, `isMyMessage`, `isAdmin` */ Message: PropTypes.oneOfType([PropTypes.node, PropTypes.func, PropTypes.object]).isRequired, /** If message belongs to current user. */ mine: PropTypes.bool, /** DOMRect object for parent MessageList component */ messageListRect: PropTypes.object, /** * Handler for flaging a current message * * @param event React's MouseEventHandler event * @returns void * */ handleFlag: PropTypes.func, /** * Handler for muting a current message * * @param event React's MouseEventHandler event * @returns void * */ handleMute: PropTypes.func, /** * Handler for editing a current message * * @param event React's MouseEventHandler event * @returns void * */ handleEdit: PropTypes.func, /** * Handler for deleting a current message * * @param event React's MouseEventHandler event * @returns void * */ handleDelete: PropTypes.func, /** * Returns array of avalable message actions for current message. * Please check [Message](https://github.com/GetStream/stream-chat-react/blob/master/src/components/Message.js) component for default implementation. */ getMessageActions: PropTypes.func }); _defineProperty(MessageActionsBox, "defaultProps", { open: false }); function ownKeys$2(object, enumerableOnly) { var keys = Object.keys(object); if (Object.getOwnPropertySymbols) { var symbols = Object.getOwnPropertySymbols(object); if (enumerableOnly) symbols = symbols.filter(function (sym) { return Object.getOwnPropertyDescriptor(object, sym).enumerable; }); keys.push.apply(keys, symbols); } return keys; } function _objectSpread$2(target) { for (var i = 1; i < arguments.length; i++) { var source = arguments[i] != null ? arguments[i] : {}; if (i % 2) { ownKeys$2(Object(source), true).forEach(function (key) { _defineProperty(target, key, source[key]); }); } else if (Object.getOwnPropertyDescriptors) { Object.defineProperties(target, Object.getOwnPropertyDescriptors(source)); } else { ownKeys$2(Object(source)).forEach(function (key) { Object.defineProperty(target, key, Object.getOwnPropertyDescriptor(source, key)); }); } } return target; } var ReactionsList = /*#__PURE__*/ function (_React$Component) { _inherits(ReactionsList, _React$Component); function ReactionsList(props) { var _this; _classCallCheck(this, ReactionsList); _this = _possibleConstructorReturn(this, _getPrototypeOf(ReactionsList).call(this, props)); _defineProperty(_assertThisInitialized(_this), "_renderReactions", function (reactions) { var reactionsByType = {}; reactions.map(function (item) { if (reactions[item.type] === undefined) { return reactionsByType[item.type] = [item]; } else { return reactionsByType[item.type] = [].concat(_toConsumableArray(reactionsByType[item.type]), [item]); } }); var reactionsEmojis = _this.props.reactionOptions.reduce(function (acc, cur) { return _objectSpread$2({}, acc, _defineProperty({}, cur.id, cur)); }, {}); return Object.keys(reactionsByType).map(function (type) { return reactionsEmojis[type] ? React__default.createElement("li", { key: reactionsEmojis[type].id }, React__default.createElement(emojiMart.NimbleEmoji, _extends({ emoji: reactionsEmojis[type] }, emojiSetDef, { size: 16, data: emojiData })), ' ', "\xA0") : null; }); }); _defineProperty(_assertThisInitialized(_this), "_getReactionCount", function () { var reaction_counts = _this.props.reaction_counts; var count = null; if (reaction_counts !== null && reaction_counts !== undefined && Object.keys(reaction_counts).length > 0) { count = 0; Object.keys(reaction_counts).map(function (key) { return count += reaction_counts[key]; }); } return count; }); return _this; } _createClass(ReactionsList, [{ key: "render", value: function render() { return React__default.createElement("div", { className: "str-chat__reaction-list ".concat(this.props.reverse ? 'str-chat__reaction-list--reverse' : ''), onClick: this.props.onClick, ref: this.reactionList }, React__default.createElement("ul", null, this._renderReactions(this.props.reactions), React__default.createElement("li", null, React__default.createElement("span", { className: "str-chat__reaction-list--counter" }, this._getReactionCount())))); } }]); return ReactionsList; }(React__default.Component); _defineProperty(ReactionsList, "propTypes", { /*