botframework-webchat-component
Version:
React component of botframework-webchat
290 lines (256 loc) • 37.1 kB
JavaScript
"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":"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","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"]}