UNPKG

botframework-webchat-component

Version:
290 lines (256 loc) 37.1 kB
"use strict"; Object.defineProperty(exports, "__esModule", { value: true }); exports.connectStackedLayout = exports.default = void 0; var _botframeworkWebchatApi = require("botframework-webchat-api"); var _classnames = _interopRequireDefault(require("classnames")); var _propTypes = _interopRequireDefault(require("prop-types")); var _react = _interopRequireDefault(require("react")); var _Bubble = _interopRequireDefault(require("./Bubble")); var _connectToWebChat = _interopRequireDefault(require("../connectToWebChat")); var _isZeroOrPositive = _interopRequireDefault(require("../Utils/isZeroOrPositive")); var _ScreenReaderText = _interopRequireDefault(require("../ScreenReaderText")); var _textFormatToContentType = _interopRequireDefault(require("../Utils/textFormatToContentType")); var _useStyleSet3 = _interopRequireDefault(require("../hooks/useStyleSet")); var _useStyleToEmotionObject = _interopRequireDefault(require("../hooks/internal/useStyleToEmotionObject")); var _useUniqueId = _interopRequireDefault(require("../hooks/internal/useUniqueId")); function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; } function _slicedToArray(arr, i) { return _arrayWithHoles(arr) || _iterableToArrayLimit(arr, i) || _unsupportedIterableToArray(arr, i) || _nonIterableRest(); } function _nonIterableRest() { throw new TypeError("Invalid attempt to destructure non-iterable instance.\nIn order to be iterable, non-array objects must have a [Symbol.iterator]() method."); } function _unsupportedIterableToArray(o, minLen) { if (!o) return; if (typeof o === "string") return _arrayLikeToArray(o, minLen); var n = Object.prototype.toString.call(o).slice(8, -1); if (n === "Object" && o.constructor) n = o.constructor.name; if (n === "Map" || n === "Set") return Array.from(o); if (n === "Arguments" || /^(?:Ui|I)nt(?:8|16|32)(?:Clamped)?Array$/.test(n)) return _arrayLikeToArray(o, minLen); } function _arrayLikeToArray(arr, len) { if (len == null || len > arr.length) len = arr.length; for (var i = 0, arr2 = new Array(len); i < len; i++) { arr2[i] = arr[i]; } return arr2; } function _iterableToArrayLimit(arr, i) { if (typeof Symbol === "undefined" || !(Symbol.iterator in Object(arr))) return; var _arr = []; var _n = true; var _d = false; var _e = undefined; try { for (var _i = arr[Symbol.iterator](), _s; !(_n = (_s = _i.next()).done); _n = true) { _arr.push(_s.value); if (i && _arr.length === i) break; } } catch (err) { _d = true; _e = err; } finally { try { if (!_n && _i["return"] != null) _i["return"](); } finally { if (_d) throw _e; } } return _arr; } function _arrayWithHoles(arr) { if (Array.isArray(arr)) return arr; } var useAvatarForBot = _botframeworkWebchatApi.hooks.useAvatarForBot, useAvatarForUser = _botframeworkWebchatApi.hooks.useAvatarForUser, useLocalizer = _botframeworkWebchatApi.hooks.useLocalizer, useStyleOptions = _botframeworkWebchatApi.hooks.useStyleOptions; var ROOT_STYLE = { '&.webchat__stacked-layout': { position: 'relative', // This is to keep screen reader text in the destinated area. '& .webchat__stacked-layout__attachment-row, & .webchat__stacked-layout__main, & .webchat__stacked-layout__message-row, & .webchat__stacked-layout__status': { display: 'flex' }, '& .webchat__stacked-layout__alignment-pad': { flexShrink: 0 }, '& .webchat__stacked-layout__attachment': { width: '100%' }, '& .webchat__stacked-layout__avatar-gutter': { display: 'flex', flexDirection: 'column', flexShrink: 0 }, '&.webchat__stacked-layout--from-user': { '& .webchat__stacked-layout__attachment-row, & .webchat__stacked-layout__main, & .webchat__stacked-layout__message-row, & .webchat__stacked-layout__status': { flexDirection: 'row-reverse' } }, '& .webchat__stacked-layout__content': { flex: 1, // This is for bottom aligning an avatar with a message bubble shorter than the avatar. // Related to the test at activityGrouping.avatarMiddleware.atBottom.js. display: 'flex', flexDirection: 'column', // This "overflow: hidden" is to make sure text overflow will get clipped correctly. // Related to the test at basic.js "long URLs with keep-all". overflow: 'hidden' }, '& .webchat__stacked-layout__nub-pad': { flexShrink: 0 } } }; var connectStackedLayout = function connectStackedLayout() { for (var _len = arguments.length, selectors = new Array(_len), _key = 0; _key < _len; _key++) { selectors[_key] = arguments[_key]; } return _connectToWebChat.default.apply(void 0, [function (_ref, _ref2) { var language = _ref.language, _ref$styleSet$options = _ref.styleSet.options, botAvatarInitials = _ref$styleSet$options.botAvatarInitials, userAvatarInitials = _ref$styleSet$options.userAvatarInitials; var _ref2$activity = _ref2.activity; _ref2$activity = _ref2$activity === void 0 ? {} : _ref2$activity; var _ref2$activity$from = _ref2$activity.from; _ref2$activity$from = _ref2$activity$from === void 0 ? {} : _ref2$activity$from; var role = _ref2$activity$from.role; return { avatarInitials: role === 'user' ? userAvatarInitials : botAvatarInitials, language: language, // TODO: [P4] We want to deprecate botAvatarInitials/userAvatarInitials because they are not as helpful as avatarInitials botAvatarInitials: botAvatarInitials, userAvatarInitials: userAvatarInitials }; }].concat(selectors)); }; exports.connectStackedLayout = connectStackedLayout; var StackedLayout = function StackedLayout(_ref3) { var activity = _ref3.activity, hideTimestamp = _ref3.hideTimestamp, renderActivityStatus = _ref3.renderActivityStatus, renderAttachment = _ref3.renderAttachment, renderAvatar = _ref3.renderAvatar, showCallout = _ref3.showCallout; var _useStyleOptions = useStyleOptions(), _useStyleOptions2 = _slicedToArray(_useStyleOptions, 1), _useStyleOptions2$ = _useStyleOptions2[0], bubbleNubOffset = _useStyleOptions2$.bubbleNubOffset, bubbleNubSize = _useStyleOptions2$.bubbleNubSize, bubbleFromUserNubOffset = _useStyleOptions2$.bubbleFromUserNubOffset, bubbleFromUserNubSize = _useStyleOptions2$.bubbleFromUserNubSize; var _useAvatarForBot = useAvatarForBot(), _useAvatarForBot2 = _slicedToArray(_useAvatarForBot, 1), botInitials = _useAvatarForBot2[0].initials; var _useAvatarForUser = useAvatarForUser(), _useAvatarForUser2 = _slicedToArray(_useAvatarForUser, 1), userInitials = _useAvatarForUser2[0].initials; var _useStyleSet = (0, _useStyleSet3.default)(), _useStyleSet2 = _slicedToArray(_useStyleSet, 1), stackedLayoutStyleSet = _useStyleSet2[0].stackedLayout; var ariaLabelId = (0, _useUniqueId.default)('webchat__stacked-layout__id'); var localize = useLocalizer(); var rootClassName = (0, _useStyleToEmotionObject.default)()(ROOT_STYLE) + ''; var showActivityStatus = typeof renderActivityStatus === 'function'; var _activity$attachments = activity.attachments, attachments = _activity$attachments === void 0 ? [] : _activity$attachments, _activity$channelData = activity.channelData; _activity$channelData = _activity$channelData === void 0 ? {} : _activity$channelData; var _activity$channelData2 = _activity$channelData.messageBack; _activity$channelData2 = _activity$channelData2 === void 0 ? {} : _activity$channelData2; var messageBackDisplayText = _activity$channelData2.displayText, _activity$from = activity.from; _activity$from = _activity$from === void 0 ? {} : _activity$from; var role = _activity$from.role, text = activity.text, textFormat = activity.textFormat; var activityDisplayText = messageBackDisplayText || text; var fromUser = role === 'user'; var attachedAlt = localize(fromUser ? 'ACTIVITY_YOU_ATTACHED_ALT' : 'ACTIVITY_BOT_ATTACHED_ALT'); var greetingAlt = (fromUser ? localize('ACTIVITY_YOU_SAID_ALT') : localize('ACTIVITY_BOT_SAID_ALT', botInitials || '')).replace(/[\t-\r \xA0\u1680\u2000-\u200A\u2028\u2029\u202F\u205F\u3000\uFEFF]{2,}/g, ' '); var initials = fromUser ? userInitials : botInitials; var nubOffset = fromUser ? bubbleFromUserNubOffset : bubbleNubOffset; var nubSize = fromUser ? bubbleFromUserNubSize : bubbleNubSize; var otherInitials = fromUser ? botInitials : userInitials; var otherNubSize = fromUser ? bubbleNubSize : bubbleFromUserNubSize; var hasAvatar = initials || typeof initials === 'string'; var hasOtherAvatar = otherInitials || typeof otherInitials === 'string'; var hasNub = typeof nubSize === 'number'; var hasOtherNub = typeof otherNubSize === 'number'; var topAlignedCallout = (0, _isZeroOrPositive.default)(nubOffset); var extraTrailing = !hasOtherAvatar && hasOtherNub; // This is for bot message with user nub and no user avatar. And vice versa. var showAvatar = showCallout && hasAvatar && !!renderAvatar; var showNub = showCallout && hasNub && (topAlignedCallout || !attachments.length); return /*#__PURE__*/_react.default.createElement("div", { "aria-labelledby": activityDisplayText ? ariaLabelId : undefined, "aria-roledescription": "activity", className: (0, _classnames.default)('webchat__stacked-layout', rootClassName, stackedLayoutStyleSet + '', { 'webchat__stacked-layout--extra-trailing': extraTrailing, 'webchat__stacked-layout--from-user': fromUser, 'webchat__stacked-layout--hide-avatar': hasAvatar && !showAvatar, 'webchat__stacked-layout--hide-nub': hasNub && !showNub, 'webchat__stacked-layout--no-message': !activityDisplayText, 'webchat__stacked-layout--show-avatar': showAvatar, 'webchat__stacked-layout--show-nub': showNub, 'webchat__stacked-layout--top-callout': topAlignedCallout }), role: "group" }, /*#__PURE__*/_react.default.createElement("div", { className: "webchat__stacked-layout__main" }, /*#__PURE__*/_react.default.createElement("div", { className: "webchat__stacked-layout__avatar-gutter" }, showAvatar && renderAvatar({ activity: activity })), /*#__PURE__*/_react.default.createElement("div", { className: "webchat__stacked-layout__content" }, !!activityDisplayText && /*#__PURE__*/_react.default.createElement("div", { "aria-roledescription": "message", className: "webchat__stacked-layout__message-row" // Disable "Prop `id` is forbidden on DOM Nodes" rule because we are using the ID prop for accessibility. /* eslint-disable-next-line react/forbid-dom-props */ , id: ariaLabelId, role: "group" }, /*#__PURE__*/_react.default.createElement(_ScreenReaderText.default, { text: greetingAlt }), /*#__PURE__*/_react.default.createElement(_Bubble.default, { className: "webchat__stacked-layout__message", fromUser: fromUser, nub: showNub || (hasAvatar || hasNub) && 'hidden' }, renderAttachment({ activity: activity, attachment: { content: activityDisplayText, contentType: (0, _textFormatToContentType.default)(textFormat) } }))), attachments.map(function (attachment, index) { return /*#__PURE__*/_react.default.createElement("div", { "aria-roledescription": "attachment", className: (0, _classnames.default)('webchat__stacked-layout__attachment-row', { 'webchat__stacked-layout__attachment-row--first': !index }) /* attachments do not have an ID, it is always indexed by number */ /* eslint-disable-next-line react/no-array-index-key */ , key: index, role: "group" }, /*#__PURE__*/_react.default.createElement(_ScreenReaderText.default, { text: attachedAlt }), /*#__PURE__*/_react.default.createElement(_Bubble.default, { className: "webchat__stacked-layout__attachment", fromUser: fromUser /* eslint-disable-next-line react/no-array-index-key */ , key: index, nub: (hasAvatar || hasNub) && 'hidden' }, renderAttachment({ activity: activity, attachment: attachment }))); })), /*#__PURE__*/_react.default.createElement("div", { className: "webchat__stacked-layout__alignment-pad" })), showActivityStatus && /*#__PURE__*/_react.default.createElement("div", { className: "webchat__stacked-layout__status" }, /*#__PURE__*/_react.default.createElement("div", { className: "webchat__stacked-layout__avatar-gutter" }), /*#__PURE__*/_react.default.createElement("div", { className: "webchat__stacked-layout__nub-pad" }), renderActivityStatus({ hideTimestamp: hideTimestamp }), /*#__PURE__*/_react.default.createElement("div", { className: "webchat__stacked-layout__alignment-pad" }))); }; StackedLayout.defaultProps = { hideTimestamp: false, renderActivityStatus: false, renderAvatar: false, showCallout: true }; StackedLayout.propTypes = { activity: _propTypes.default.shape({ attachments: _propTypes.default.array, channelData: _propTypes.default.shape({ messageBack: _propTypes.default.shape({ displayText: _propTypes.default.string }) }), from: _propTypes.default.shape({ role: _propTypes.default.string.isRequired }).isRequired, text: _propTypes.default.string, textFormat: _propTypes.default.string, timestamp: _propTypes.default.string, type: _propTypes.default.string.isRequired }).isRequired, hideTimestamp: _propTypes.default.bool, renderActivityStatus: _propTypes.default.oneOfType([_propTypes.default.oneOf([false]), _propTypes.default.func]), renderAttachment: _propTypes.default.func.isRequired, renderAvatar: _propTypes.default.oneOfType([_propTypes.default.oneOf([false]), _propTypes.default.func]), showCallout: _propTypes.default.bool }; var _default = StackedLayout; exports.default = _default; //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/Activity/StackedLayout.js"],"names":["useAvatarForBot","hooks","useAvatarForUser","useLocalizer","useStyleOptions","ROOT_STYLE","position","display","flexShrink","width","flexDirection","flex","overflow","connectStackedLayout","selectors","connectToWebChat","language","styleSet","options","botAvatarInitials","userAvatarInitials","activity","from","role","avatarInitials","StackedLayout","hideTimestamp","renderActivityStatus","renderAttachment","renderAvatar","showCallout","bubbleNubOffset","bubbleNubSize","bubbleFromUserNubOffset","bubbleFromUserNubSize","botInitials","initials","userInitials","stackedLayoutStyleSet","stackedLayout","ariaLabelId","localize","rootClassName","showActivityStatus","attachments","channelData","messageBack","messageBackDisplayText","displayText","text","textFormat","activityDisplayText","fromUser","attachedAlt","greetingAlt","replace","nubOffset","nubSize","otherInitials","otherNubSize","hasAvatar","hasOtherAvatar","hasNub","hasOtherNub","topAlignedCallout","extraTrailing","showAvatar","showNub","length","undefined","attachment","content","contentType","map","index","defaultProps","propTypes","PropTypes","shape","array","string","isRequired","timestamp","type","bool","oneOfType","oneOf","func"],"mappings":";;;;;;;AAEA;;AACA;;AACA;;AACA;;AAEA;;AACA;;AACA;;AACA;;AACA;;AACA;;AACA;;AAEA;;;;;;;;;;;;;;;;IAEQA,e,GAAqEC,6B,CAArED,e;IAAiBE,gB,GAAoDD,6B,CAApDC,gB;IAAkBC,Y,GAAkCF,6B,CAAlCE,Y;IAAcC,e,GAAoBH,6B,CAApBG,e;AAEzD,IAAMC,UAAU,GAAG;AACjB,+BAA6B;AAC3BC,IAAAA,QAAQ,EAAE,UADiB;AACL;AAEtB,iKAA6J;AAC3JC,MAAAA,OAAO,EAAE;AADkJ,KAHlI;AAO3B,iDAA6C;AAC3CC,MAAAA,UAAU,EAAE;AAD+B,KAPlB;AAW3B,8CAA0C;AACxCC,MAAAA,KAAK,EAAE;AADiC,KAXf;AAe3B,iDAA6C;AAC3CF,MAAAA,OAAO,EAAE,MADkC;AAE3CG,MAAAA,aAAa,EAAE,QAF4B;AAG3CF,MAAAA,UAAU,EAAE;AAH+B,KAflB;AAqB3B,4CAAwC;AACtC,mKAA6J;AAC3JE,QAAAA,aAAa,EAAE;AAD4I;AADvH,KArBb;AA2B3B,2CAAuC;AACrCC,MAAAA,IAAI,EAAE,CAD+B;AAGrC;AACA;AACAJ,MAAAA,OAAO,EAAE,MAL4B;AAMrCG,MAAAA,aAAa,EAAE,QANsB;AAQrC;AACA;AACAE,MAAAA,QAAQ,EAAE;AAV2B,KA3BZ;AAwC3B,2CAAuC;AACrCJ,MAAAA,UAAU,EAAE;AADyB;AAxCZ;AADZ,CAAnB;;AA+CA,IAAMK,oBAAoB,GAAG,SAAvBA,oBAAuB;AAAA,oCAAIC,SAAJ;AAAIA,IAAAA,SAAJ;AAAA;;AAAA,SAC3BC,yCACE;AAAA,QAEIC,QAFJ,QAEIA,QAFJ;AAAA,qCAGIC,QAHJ,CAIMC,OAJN;AAAA,QAIiBC,iBAJjB,yBAIiBA,iBAJjB;AAAA,QAIoCC,kBAJpC,yBAIoCA,kBAJpC;AAAA,+BAOIC,QAPJ;AAAA,iDAOwC,EAPxC;AAAA,6CAOgBC,IAPhB;AAAA,2DAOiC,EAPjC;AAAA,QAOwBC,IAPxB,uBAOwBA,IAPxB;AAAA,WAQM;AACJC,MAAAA,cAAc,EAAED,IAAI,KAAK,MAAT,GAAkBH,kBAAlB,GAAuCD,iBADnD;AAEJH,MAAAA,QAAQ,EAARA,QAFI;AAIJ;AACAG,MAAAA,iBAAiB,EAAjBA,iBALI;AAMJC,MAAAA,kBAAkB,EAAlBA;AANI,KARN;AAAA,GADF,SAiBKN,SAjBL,EAD2B;AAAA,CAA7B;;;;AAqBA,IAAMW,aAAa,GAAG,SAAhBA,aAAgB,QAOhB;AAAA,MANJJ,QAMI,SANJA,QAMI;AAAA,MALJK,aAKI,SALJA,aAKI;AAAA,MAJJC,oBAII,SAJJA,oBAII;AAAA,MAHJC,gBAGI,SAHJA,gBAGI;AAAA,MAFJC,YAEI,SAFJA,YAEI;AAAA,MADJC,WACI,SADJA,WACI;;AAAA,yBACyF1B,eAAe,EADxG;AAAA;AAAA;AAAA,MACK2B,eADL,sBACKA,eADL;AAAA,MACsBC,aADtB,sBACsBA,aADtB;AAAA,MACqCC,uBADrC,sBACqCA,uBADrC;AAAA,MAC8DC,qBAD9D,sBAC8DA,qBAD9D;;AAAA,yBAEgClC,eAAe,EAF/C;AAAA;AAAA,MAEemC,WAFf,wBAEKC,QAFL;;AAAA,0BAGiClC,gBAAgB,EAHjD;AAAA;AAAA,MAGemC,YAHf,yBAGKD,QAHL;;AAAA,qBAI+C,4BAJ/C;AAAA;AAAA,MAIoBE,qBAJpB,oBAIKC,aAJL;;AAKJ,MAAMC,WAAW,GAAG,0BAAY,6BAAZ,CAApB;AACA,MAAMC,QAAQ,GAAGtC,YAAY,EAA7B;AACA,MAAMuC,aAAa,GAAG,wCAA0BrC,UAA1B,IAAwC,EAA9D;AACA,MAAMsC,kBAAkB,GAAG,OAAOhB,oBAAP,KAAgC,UAA3D;AARI,8BAgBAN,QAhBA,CAWFuB,WAXE;AAAA,MAWFA,WAXE,sCAWY,EAXZ;AAAA,8BAgBAvB,QAhBA,CAYFwB,WAZE;AAAA,6DAY2E,EAZ3E;AAAA,qDAYaC,WAZb;AAAA,+DAYoE,EAZpE;AAAA,MAYyCC,sBAZzC,0BAY4BC,WAZ5B;AAAA,uBAgBA3B,QAhBA,CAaFC,IAbE;AAAA,+CAae,EAbf;AAAA,MAaMC,IAbN,kBAaMA,IAbN;AAAA,MAcF0B,IAdE,GAgBA5B,QAhBA,CAcF4B,IAdE;AAAA,MAeFC,UAfE,GAgBA7B,QAhBA,CAeF6B,UAfE;AAkBJ,MAAMC,mBAAmB,GAAGJ,sBAAsB,IAAIE,IAAtD;AACA,MAAMG,QAAQ,GAAG7B,IAAI,KAAK,MAA1B;AAEA,MAAM8B,WAAW,GAAGZ,QAAQ,CAACW,QAAQ,GAAG,2BAAH,GAAiC,2BAA1C,CAA5B;AACA,MAAME,WAAW,GAAG,CAACF,QAAQ,GACzBX,QAAQ,CAAC,uBAAD,CADiB,GAEzBA,QAAQ,CAAC,uBAAD,EAA0BN,WAAW,IAAI,EAAzC,CAFQ,EAGlBoB,OAHkB,CAGV,0EAHU,EAGE,GAHF,CAApB;AAKA,MAAMnB,QAAQ,GAAGgB,QAAQ,GAAGf,YAAH,GAAkBF,WAA3C;AACA,MAAMqB,SAAS,GAAGJ,QAAQ,GAAGnB,uBAAH,GAA6BF,eAAvD;AACA,MAAM0B,OAAO,GAAGL,QAAQ,GAAGlB,qBAAH,GAA2BF,aAAnD;AACA,MAAM0B,aAAa,GAAGN,QAAQ,GAAGjB,WAAH,GAAiBE,YAA/C;AACA,MAAMsB,YAAY,GAAGP,QAAQ,GAAGpB,aAAH,GAAmBE,qBAAhD;AAEA,MAAM0B,SAAS,GAAGxB,QAAQ,IAAI,OAAOA,QAAP,KAAoB,QAAlD;AACA,MAAMyB,cAAc,GAAGH,aAAa,IAAI,OAAOA,aAAP,KAAyB,QAAjE;AACA,MAAMI,MAAM,GAAG,OAAOL,OAAP,KAAmB,QAAlC;AACA,MAAMM,WAAW,GAAG,OAAOJ,YAAP,KAAwB,QAA5C;AACA,MAAMK,iBAAiB,GAAG,+BAAiBR,SAAjB,CAA1B;AAEA,MAAMS,aAAa,GAAG,CAACJ,cAAD,IAAmBE,WAAzC,CAvCI,CAuCkD;;AAEtD,MAAMG,UAAU,GAAGpC,WAAW,IAAI8B,SAAf,IAA4B,CAAC,CAAC/B,YAAjD;AACA,MAAMsC,OAAO,GAAGrC,WAAW,IAAIgC,MAAf,KAA0BE,iBAAiB,IAAI,CAACpB,WAAW,CAACwB,MAA5D,CAAhB;AAEA,sBACE;AACE,uBAAiBjB,mBAAmB,GAAGX,WAAH,GAAiB6B,SADvD;AAEE,4BAAqB,UAFvB;AAGE,IAAA,SAAS,EAAE,yBAAW,yBAAX,EAAsC3B,aAAtC,EAAqDJ,qBAAqB,GAAG,EAA7E,EAAiF;AAC1F,iDAA2C2B,aAD+C;AAE1F,4CAAsCb,QAFoD;AAG1F,8CAAwCQ,SAAS,IAAI,CAACM,UAHoC;AAI1F,2CAAqCJ,MAAM,IAAI,CAACK,OAJ0C;AAK1F,6CAAuC,CAAChB,mBALkD;AAM1F,8CAAwCe,UANkD;AAO1F,2CAAqCC,OAPqD;AAQ1F,8CAAwCH;AARkD,KAAjF,CAHb;AAaE,IAAA,IAAI,EAAC;AAbP,kBAeE;AAAK,IAAA,SAAS,EAAC;AAAf,kBACE;AAAK,IAAA,SAAS,EAAC;AAAf,KAAyDE,UAAU,IAAIrC,YAAY,CAAC;AAAER,IAAAA,QAAQ,EAARA;AAAF,GAAD,CAAnF,CADF,eAEE;AAAK,IAAA,SAAS,EAAC;AAAf,KACG,CAAC,CAAC8B,mBAAF,iBACC;AACE,4BAAqB,SADvB;AAEE,IAAA,SAAS,EAAC,sCAFZ,CAGE;;AACA;AAJF;AAKE,IAAA,EAAE,EAAEX,WALN;AAME,IAAA,IAAI,EAAC;AANP,kBAQE,6BAAC,yBAAD;AAAkB,IAAA,IAAI,EAAEc;AAAxB,IARF,eASE,6BAAC,eAAD;AACE,IAAA,SAAS,EAAC,kCADZ;AAEE,IAAA,QAAQ,EAAEF,QAFZ;AAGE,IAAA,GAAG,EAAEe,OAAO,IAAK,CAACP,SAAS,IAAIE,MAAd,KAAyB;AAH5C,KAKGlC,gBAAgB,CAAC;AAChBP,IAAAA,QAAQ,EAARA,QADgB;AAEhBiD,IAAAA,UAAU,EAAE;AACVC,MAAAA,OAAO,EAAEpB,mBADC;AAEVqB,MAAAA,WAAW,EAAE,sCAAwBtB,UAAxB;AAFH;AAFI,GAAD,CALnB,CATF,CAFJ,EA0BGN,WAAW,CAAC6B,GAAZ,CAAgB,UAACH,UAAD,EAAaI,KAAb;AAAA,wBACf;AACE,8BAAqB,YADvB;AAEE,MAAA,SAAS,EAAE,yBAAW,yCAAX,EAAsD;AAC/D,0DAAkD,CAACA;AADY,OAAtD;AAGX;;AACA;AANF;AAOE,MAAA,GAAG,EAAEA,KAPP;AAQE,MAAA,IAAI,EAAC;AARP,oBAUE,6BAAC,yBAAD;AAAkB,MAAA,IAAI,EAAErB;AAAxB,MAVF,eAWE,6BAAC,eAAD;AACE,MAAA,SAAS,EAAC,qCADZ;AAEE,MAAA,QAAQ,EAAED;AACV;AAHF;AAIE,MAAA,GAAG,EAAEsB,KAJP;AAKE,MAAA,GAAG,EAAE,CAACd,SAAS,IAAIE,MAAd,KAAyB;AALhC,OAOGlC,gBAAgB,CAAC;AAAEP,MAAAA,QAAQ,EAARA,QAAF;AAAYiD,MAAAA,UAAU,EAAVA;AAAZ,KAAD,CAPnB,CAXF,CADe;AAAA,GAAhB,CA1BH,CAFF,eAoDE;AAAK,IAAA,SAAS,EAAC;AAAf,IApDF,CAfF,EAqEG3B,kBAAkB,iBACjB;AAAK,IAAA,SAAS,EAAC;AAAf,kBACE;AAAK,IAAA,SAAS,EAAC;AAAf,IADF,eAEE;AAAK,IAAA,SAAS,EAAC;AAAf,IAFF,EAGGhB,oBAAoB,CAAC;AAAED,IAAAA,aAAa,EAAbA;AAAF,GAAD,CAHvB,eAIE;AAAK,IAAA,SAAS,EAAC;AAAf,IAJF,CAtEJ,CADF;AAgFD,CAnID;;AAqIAD,aAAa,CAACkD,YAAd,GAA6B;AAC3BjD,EAAAA,aAAa,EAAE,KADY;AAE3BC,EAAAA,oBAAoB,EAAE,KAFK;AAG3BE,EAAAA,YAAY,EAAE,KAHa;AAI3BC,EAAAA,WAAW,EAAE;AAJc,CAA7B;AAOAL,aAAa,CAACmD,SAAd,GAA0B;AACxBvD,EAAAA,QAAQ,EAAEwD,mBAAUC,KAAV,CAAgB;AACxBlC,IAAAA,WAAW,EAAEiC,mBAAUE,KADC;AAExBlC,IAAAA,WAAW,EAAEgC,mBAAUC,KAAV,CAAgB;AAC3BhC,MAAAA,WAAW,EAAE+B,mBAAUC,KAAV,CAAgB;AAC3B9B,QAAAA,WAAW,EAAE6B,mBAAUG;AADI,OAAhB;AADc,KAAhB,CAFW;AAOxB1D,IAAAA,IAAI,EAAEuD,mBAAUC,KAAV,CAAgB;AACpBvD,MAAAA,IAAI,EAAEsD,mBAAUG,MAAV,CAAiBC;AADH,KAAhB,EAEHA,UATqB;AAUxBhC,IAAAA,IAAI,EAAE4B,mBAAUG,MAVQ;AAWxB9B,IAAAA,UAAU,EAAE2B,mBAAUG,MAXE;AAYxBE,IAAAA,SAAS,EAAEL,mBAAUG,MAZG;AAaxBG,IAAAA,IAAI,EAAEN,mBAAUG,MAAV,CAAiBC;AAbC,GAAhB,EAcPA,UAfqB;AAgBxBvD,EAAAA,aAAa,EAAEmD,mBAAUO,IAhBD;AAiBxBzD,EAAAA,oBAAoB,EAAEkD,mBAAUQ,SAAV,CAAoB,CAACR,mBAAUS,KAAV,CAAgB,CAAC,KAAD,CAAhB,CAAD,EAA2BT,mBAAUU,IAArC,CAApB,CAjBE;AAkBxB3D,EAAAA,gBAAgB,EAAEiD,mBAAUU,IAAV,CAAeN,UAlBT;AAmBxBpD,EAAAA,YAAY,EAAEgD,mBAAUQ,SAAV,CAAoB,CAACR,mBAAUS,KAAV,CAAgB,CAAC,KAAD,CAAhB,CAAD,EAA2BT,mBAAUU,IAArC,CAApB,CAnBU;AAoBxBzD,EAAAA,WAAW,EAAE+C,mBAAUO;AApBC,CAA1B;eAuBe3D,a","sourceRoot":"component:///","sourcesContent":["/* eslint complexity: [\"error\", 30] */\n\nimport { hooks } from 'botframework-webchat-api';\nimport classNames from 'classnames';\nimport PropTypes from 'prop-types';\nimport React from 'react';\n\nimport Bubble from './Bubble';\nimport connectToWebChat from '../connectToWebChat';\nimport isZeroOrPositive from '../Utils/isZeroOrPositive';\nimport ScreenReaderText from '../ScreenReaderText';\nimport textFormatToContentType from '../Utils/textFormatToContentType';\nimport useStyleSet from '../hooks/useStyleSet';\nimport useStyleToEmotionObject from '../hooks/internal/useStyleToEmotionObject';\n\nimport useUniqueId from '../hooks/internal/useUniqueId';\n\nconst { useAvatarForBot, useAvatarForUser, useLocalizer, useStyleOptions } = hooks;\n\nconst ROOT_STYLE = {\n  '&.webchat__stacked-layout': {\n    position: 'relative', // This is to keep screen reader text in the destinated area.\n\n    '& .webchat__stacked-layout__attachment-row, & .webchat__stacked-layout__main, & .webchat__stacked-layout__message-row, & .webchat__stacked-layout__status': {\n      display: 'flex'\n    },\n\n    '& .webchat__stacked-layout__alignment-pad': {\n      flexShrink: 0\n    },\n\n    '& .webchat__stacked-layout__attachment': {\n      width: '100%'\n    },\n\n    '& .webchat__stacked-layout__avatar-gutter': {\n      display: 'flex',\n      flexDirection: 'column',\n      flexShrink: 0\n    },\n\n    '&.webchat__stacked-layout--from-user': {\n      '& .webchat__stacked-layout__attachment-row, & .webchat__stacked-layout__main, & .webchat__stacked-layout__message-row, & .webchat__stacked-layout__status': {\n        flexDirection: 'row-reverse'\n      }\n    },\n\n    '& .webchat__stacked-layout__content': {\n      flex: 1,\n\n      // This is for bottom aligning an avatar with a message bubble shorter than the avatar.\n      // Related to the test at activityGrouping.avatarMiddleware.atBottom.js.\n      display: 'flex',\n      flexDirection: 'column',\n\n      // This \"overflow: hidden\" is to make sure text overflow will get clipped correctly.\n      // Related to the test at basic.js \"long URLs with keep-all\".\n      overflow: 'hidden'\n    },\n\n    '& .webchat__stacked-layout__nub-pad': {\n      flexShrink: 0\n    }\n  }\n};\n\nconst connectStackedLayout = (...selectors) =>\n  connectToWebChat(\n    (\n      {\n        language,\n        styleSet: {\n          options: { botAvatarInitials, userAvatarInitials }\n        }\n      },\n      { activity: { from: { role } = {} } = {} }\n    ) => ({\n      avatarInitials: role === 'user' ? userAvatarInitials : botAvatarInitials,\n      language,\n\n      // TODO: [P4] We want to deprecate botAvatarInitials/userAvatarInitials because they are not as helpful as avatarInitials\n      botAvatarInitials,\n      userAvatarInitials\n    }),\n    ...selectors\n  );\n\nconst StackedLayout = ({\n  activity,\n  hideTimestamp,\n  renderActivityStatus,\n  renderAttachment,\n  renderAvatar,\n  showCallout\n}) => {\n  const [{ bubbleNubOffset, bubbleNubSize, bubbleFromUserNubOffset, bubbleFromUserNubSize }] = useStyleOptions();\n  const [{ initials: botInitials }] = useAvatarForBot();\n  const [{ initials: userInitials }] = useAvatarForUser();\n  const [{ stackedLayout: stackedLayoutStyleSet }] = useStyleSet();\n  const ariaLabelId = useUniqueId('webchat__stacked-layout__id');\n  const localize = useLocalizer();\n  const rootClassName = useStyleToEmotionObject()(ROOT_STYLE) + '';\n  const showActivityStatus = typeof renderActivityStatus === 'function';\n\n  const {\n    attachments = [],\n    channelData: { messageBack: { displayText: messageBackDisplayText } = {} } = {},\n    from: { role } = {},\n    text,\n    textFormat\n  } = activity;\n\n  const activityDisplayText = messageBackDisplayText || text;\n  const fromUser = role === 'user';\n\n  const attachedAlt = localize(fromUser ? 'ACTIVITY_YOU_ATTACHED_ALT' : 'ACTIVITY_BOT_ATTACHED_ALT');\n  const greetingAlt = (fromUser\n    ? localize('ACTIVITY_YOU_SAID_ALT')\n    : localize('ACTIVITY_BOT_SAID_ALT', botInitials || '')\n  ).replace(/\\s{2,}/gu, ' ');\n\n  const initials = fromUser ? userInitials : botInitials;\n  const nubOffset = fromUser ? bubbleFromUserNubOffset : bubbleNubOffset;\n  const nubSize = fromUser ? bubbleFromUserNubSize : bubbleNubSize;\n  const otherInitials = fromUser ? botInitials : userInitials;\n  const otherNubSize = fromUser ? bubbleNubSize : bubbleFromUserNubSize;\n\n  const hasAvatar = initials || typeof initials === 'string';\n  const hasOtherAvatar = otherInitials || typeof otherInitials === 'string';\n  const hasNub = typeof nubSize === 'number';\n  const hasOtherNub = typeof otherNubSize === 'number';\n  const topAlignedCallout = isZeroOrPositive(nubOffset);\n\n  const extraTrailing = !hasOtherAvatar && hasOtherNub; // This is for bot message with user nub and no user avatar. And vice versa.\n\n  const showAvatar = showCallout && hasAvatar && !!renderAvatar;\n  const showNub = showCallout && hasNub && (topAlignedCallout || !attachments.length);\n\n  return (\n    <div\n      aria-labelledby={activityDisplayText ? ariaLabelId : undefined}\n      aria-roledescription=\"activity\"\n      className={classNames('webchat__stacked-layout', rootClassName, stackedLayoutStyleSet + '', {\n        'webchat__stacked-layout--extra-trailing': extraTrailing,\n        'webchat__stacked-layout--from-user': fromUser,\n        'webchat__stacked-layout--hide-avatar': hasAvatar && !showAvatar,\n        'webchat__stacked-layout--hide-nub': hasNub && !showNub,\n        'webchat__stacked-layout--no-message': !activityDisplayText,\n        'webchat__stacked-layout--show-avatar': showAvatar,\n        'webchat__stacked-layout--show-nub': showNub,\n        'webchat__stacked-layout--top-callout': topAlignedCallout\n      })}\n      role=\"group\"\n    >\n      <div className=\"webchat__stacked-layout__main\">\n        <div className=\"webchat__stacked-layout__avatar-gutter\">{showAvatar && renderAvatar({ activity })}</div>\n        <div className=\"webchat__stacked-layout__content\">\n          {!!activityDisplayText && (\n            <div\n              aria-roledescription=\"message\"\n              className=\"webchat__stacked-layout__message-row\"\n              // Disable \"Prop `id` is forbidden on DOM Nodes\" rule because we are using the ID prop for accessibility.\n              /* eslint-disable-next-line react/forbid-dom-props */\n              id={ariaLabelId}\n              role=\"group\"\n            >\n              <ScreenReaderText text={greetingAlt} />\n              <Bubble\n                className=\"webchat__stacked-layout__message\"\n                fromUser={fromUser}\n                nub={showNub || ((hasAvatar || hasNub) && 'hidden')}\n              >\n                {renderAttachment({\n                  activity,\n                  attachment: {\n                    content: activityDisplayText,\n                    contentType: textFormatToContentType(textFormat)\n                  }\n                })}\n              </Bubble>\n            </div>\n          )}\n          {attachments.map((attachment, index) => (\n            <div\n              aria-roledescription=\"attachment\"\n              className={classNames('webchat__stacked-layout__attachment-row', {\n                'webchat__stacked-layout__attachment-row--first': !index\n              })}\n              /* attachments do not have an ID, it is always indexed by number */\n              /* eslint-disable-next-line react/no-array-index-key */\n              key={index}\n              role=\"group\"\n            >\n              <ScreenReaderText text={attachedAlt} />\n              <Bubble\n                className=\"webchat__stacked-layout__attachment\"\n                fromUser={fromUser}\n                /* eslint-disable-next-line react/no-array-index-key */\n                key={index}\n                nub={(hasAvatar || hasNub) && 'hidden'}\n              >\n                {renderAttachment({ activity, attachment })}\n              </Bubble>\n            </div>\n          ))}\n        </div>\n        <div className=\"webchat__stacked-layout__alignment-pad\" />\n      </div>\n      {showActivityStatus && (\n        <div className=\"webchat__stacked-layout__status\">\n          <div className=\"webchat__stacked-layout__avatar-gutter\" />\n          <div className=\"webchat__stacked-layout__nub-pad\" />\n          {renderActivityStatus({ hideTimestamp })}\n          <div className=\"webchat__stacked-layout__alignment-pad\" />\n        </div>\n      )}\n    </div>\n  );\n};\n\nStackedLayout.defaultProps = {\n  hideTimestamp: false,\n  renderActivityStatus: false,\n  renderAvatar: false,\n  showCallout: true\n};\n\nStackedLayout.propTypes = {\n  activity: PropTypes.shape({\n    attachments: PropTypes.array,\n    channelData: PropTypes.shape({\n      messageBack: PropTypes.shape({\n        displayText: PropTypes.string\n      })\n    }),\n    from: PropTypes.shape({\n      role: PropTypes.string.isRequired\n    }).isRequired,\n    text: PropTypes.string,\n    textFormat: PropTypes.string,\n    timestamp: PropTypes.string,\n    type: PropTypes.string.isRequired\n  }).isRequired,\n  hideTimestamp: PropTypes.bool,\n  renderActivityStatus: PropTypes.oneOfType([PropTypes.oneOf([false]), PropTypes.func]),\n  renderAttachment: PropTypes.func.isRequired,\n  renderAvatar: PropTypes.oneOfType([PropTypes.oneOf([false]), PropTypes.func]),\n  showCallout: PropTypes.bool\n};\n\nexport default StackedLayout;\n\nexport { connectStackedLayout };\n"]}