UNPKG

botframework-webchat-component

Version:
142 lines (122 loc) 18.8 kB
"use strict"; Object.defineProperty(exports, "__esModule", { value: true }); 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 _textFormatToContentType = _interopRequireDefault(require("./Utils/textFormatToContentType")); var _useStripMarkdown = _interopRequireDefault(require("./hooks/internal/useStripMarkdown")); var _useStyleToEmotionObject = _interopRequireDefault(require("./hooks/internal/useStyleToEmotionObject")); 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, useCreateAttachmentForScreenReaderRenderer = _botframeworkWebchatApi.hooks.useCreateAttachmentForScreenReaderRenderer, useDateFormatter = _botframeworkWebchatApi.hooks.useDateFormatter, useLocalizer = _botframeworkWebchatApi.hooks.useLocalizer; var ROOT_STYLE = { '&.webchat__screen-reader-activity': { color: 'transparent', height: 1, opacity: 0, overflow: 'hidden', position: 'absolute', top: 0, whiteSpace: 'nowrap', width: 1 } }; var ACTIVITY_NUM_ATTACHMENTS_ALT_IDS = { few: 'ACTIVITY_NUM_ATTACHMENTS_FEW_ALT', many: 'ACTIVITY_NUM_ATTACHMENTS_MANY_ALT', one: 'ACTIVITY_NUM_ATTACHMENTS_ONE_ALT', other: 'ACTIVITY_NUM_ATTACHMENTS_OTHER_ALT', two: 'ACTIVITY_NUM_ATTACHMENTS_TWO_ALT' }; // When "renderAttachments" is false, we will not render the content of attachments. // That means, it will only render "2 attachments", instead of "image attachment". // This is used in the visual transcript, where we render "Press ENTER to interact." var ScreenReaderActivity = function ScreenReaderActivity(_ref) { var activity = _ref.activity, children = _ref.children, id = _ref.id, renderAttachments = _ref.renderAttachments; var _useAvatarForBot = useAvatarForBot(), _useAvatarForBot2 = _slicedToArray(_useAvatarForBot, 1), botInitials = _useAvatarForBot2[0].initials; var createAttachmentForScreenReaderRenderer = useCreateAttachmentForScreenReaderRenderer(); var formatDate = useDateFormatter(); var localize = useLocalizer(); var localizeWithPlural = useLocalizer({ plural: true }); var rootClassName = (0, _useStyleToEmotionObject.default)()(ROOT_STYLE) + ''; 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, timestamp = activity.timestamp; var fromUser = role === 'user'; var contentTypeMarkdown = (0, _textFormatToContentType.default)(textFormat) === 'text/markdown'; var displayText = messageBackDisplayText || text; var attachmentForScreenReaderRenderers = renderAttachments ? attachments.map(function (attachment) { return createAttachmentForScreenReaderRenderer({ activity: activity, attachment: attachment }); }).filter(function (render) { return render; }) : []; 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 numGenericAttachments = attachments.length - attachmentForScreenReaderRenderers.length; var numAttachmentsAlt = !!numGenericAttachments && localizeWithPlural(ACTIVITY_NUM_ATTACHMENTS_ALT_IDS, numGenericAttachments); var textAlt = (0, _useStripMarkdown.default)(contentTypeMarkdown && displayText) || displayText; var timestampAlt = localize('ACTIVITY_STATUS_SEND_STATUS_ALT_SENT_AT', formatDate(timestamp)); return /*#__PURE__*/_react.default.createElement("article", { "aria-atomic": true, "aria-roledescription": "message", className: (0, _classnames.default)('webchat__screen-reader-activity', rootClassName) // "id" attribute is used by `aria-labelledby`. // eslint-disable-next-line react/forbid-dom-props , id: id, role: "region" }, /*#__PURE__*/_react.default.createElement("p", null, /*#__PURE__*/_react.default.createElement("span", null, greetingAlt), /*#__PURE__*/_react.default.createElement("span", null, textAlt)), !!attachmentForScreenReaderRenderers.length && /*#__PURE__*/_react.default.createElement("ul", null, attachmentForScreenReaderRenderers.map(function (render, index) { return ( /*#__PURE__*/ // eslint-disable-next-line react/no-array-index-key _react.default.createElement("li", { key: index }, render()) ); })), numAttachmentsAlt && /*#__PURE__*/_react.default.createElement("p", null, numAttachmentsAlt), /*#__PURE__*/_react.default.createElement("p", { className: "webchat__screen-reader-activity__timestamp" }, timestampAlt), children); }; ScreenReaderActivity.defaultProps = { children: undefined, id: undefined, renderAttachments: true }; ScreenReaderActivity.propTypes = { activity: _propTypes.default.any.isRequired, children: _propTypes.default.any, id: _propTypes.default.string, renderAttachments: _propTypes.default.bool }; var _default = ScreenReaderActivity; exports.default = _default; //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../src/ScreenReaderActivity.js"],"names":["useAvatarForBot","hooks","useCreateAttachmentForScreenReaderRenderer","useDateFormatter","useLocalizer","ROOT_STYLE","color","height","opacity","overflow","position","top","whiteSpace","width","ACTIVITY_NUM_ATTACHMENTS_ALT_IDS","few","many","one","other","two","ScreenReaderActivity","activity","children","id","renderAttachments","botInitials","initials","createAttachmentForScreenReaderRenderer","formatDate","localize","localizeWithPlural","plural","rootClassName","attachments","channelData","messageBack","messageBackDisplayText","displayText","from","role","text","textFormat","timestamp","fromUser","contentTypeMarkdown","attachmentForScreenReaderRenderers","map","attachment","filter","render","greetingAlt","replace","numGenericAttachments","length","numAttachmentsAlt","textAlt","timestampAlt","index","defaultProps","undefined","propTypes","PropTypes","any","isRequired","string","bool"],"mappings":"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","sourceRoot":"component:///","sourcesContent":["/* eslint no-magic-numbers: [\"error\", { \"ignore\": [2] }] */\n\nimport { hooks } from 'botframework-webchat-api';\nimport classNames from 'classnames';\nimport PropTypes from 'prop-types';\nimport React from 'react';\n\nimport textFormatToContentType from './Utils/textFormatToContentType';\nimport useStripMarkdown from './hooks/internal/useStripMarkdown';\nimport useStyleToEmotionObject from './hooks/internal/useStyleToEmotionObject';\n\nconst { useAvatarForBot, useCreateAttachmentForScreenReaderRenderer, useDateFormatter, useLocalizer } = hooks;\n\nconst ROOT_STYLE = {\n  '&.webchat__screen-reader-activity': {\n    color: 'transparent',\n    height: 1,\n    opacity: 0,\n    overflow: 'hidden',\n    position: 'absolute',\n    top: 0,\n    whiteSpace: 'nowrap',\n    width: 1\n  }\n};\n\nconst ACTIVITY_NUM_ATTACHMENTS_ALT_IDS = {\n  few: 'ACTIVITY_NUM_ATTACHMENTS_FEW_ALT',\n  many: 'ACTIVITY_NUM_ATTACHMENTS_MANY_ALT',\n  one: 'ACTIVITY_NUM_ATTACHMENTS_ONE_ALT',\n  other: 'ACTIVITY_NUM_ATTACHMENTS_OTHER_ALT',\n  two: 'ACTIVITY_NUM_ATTACHMENTS_TWO_ALT'\n};\n\n// When \"renderAttachments\" is false, we will not render the content of attachments.\n// That means, it will only render \"2 attachments\", instead of \"image attachment\".\n// This is used in the visual transcript, where we render \"Press ENTER to interact.\"\nconst ScreenReaderActivity = ({ activity, children, id, renderAttachments }) => {\n  const [{ initials: botInitials }] = useAvatarForBot();\n  const createAttachmentForScreenReaderRenderer = useCreateAttachmentForScreenReaderRenderer();\n  const formatDate = useDateFormatter();\n  const localize = useLocalizer();\n  const localizeWithPlural = useLocalizer({ plural: true });\n  const rootClassName = useStyleToEmotionObject()(ROOT_STYLE) + '';\n\n  const {\n    attachments = [],\n    channelData: { messageBack: { displayText: messageBackDisplayText } = {} } = {},\n    from: { role } = {},\n    text,\n    textFormat,\n    timestamp\n  } = activity;\n\n  const fromUser = role === 'user';\n  const contentTypeMarkdown = textFormatToContentType(textFormat) === 'text/markdown';\n  const displayText = messageBackDisplayText || text;\n\n  const attachmentForScreenReaderRenderers = renderAttachments\n    ? attachments\n        .map(attachment => createAttachmentForScreenReaderRenderer({ activity, attachment }))\n        .filter(render => render)\n    : [];\n\n  const greetingAlt = (fromUser\n    ? localize('ACTIVITY_YOU_SAID_ALT')\n    : localize('ACTIVITY_BOT_SAID_ALT', botInitials || '')\n  ).replace(/\\s{2,}/gu, ' ');\n  const numGenericAttachments = attachments.length - attachmentForScreenReaderRenderers.length;\n\n  const numAttachmentsAlt =\n    !!numGenericAttachments && localizeWithPlural(ACTIVITY_NUM_ATTACHMENTS_ALT_IDS, numGenericAttachments);\n  const textAlt = useStripMarkdown(contentTypeMarkdown && displayText) || displayText;\n  const timestampAlt = localize('ACTIVITY_STATUS_SEND_STATUS_ALT_SENT_AT', formatDate(timestamp));\n\n  return (\n    <article\n      aria-atomic={true}\n      aria-roledescription=\"message\"\n      className={classNames('webchat__screen-reader-activity', rootClassName)}\n      // \"id\" attribute is used by `aria-labelledby`.\n      // eslint-disable-next-line react/forbid-dom-props\n      id={id}\n      role=\"region\"\n    >\n      <p>\n        <span>{greetingAlt}</span>\n        <span>{textAlt}</span>\n      </p>\n      {!!attachmentForScreenReaderRenderers.length && (\n        <ul>\n          {attachmentForScreenReaderRenderers.map((render, index) => (\n            // eslint-disable-next-line react/no-array-index-key\n            <li key={index}>{render()}</li>\n          ))}\n        </ul>\n      )}\n      {numAttachmentsAlt && <p>{numAttachmentsAlt}</p>}\n      <p className=\"webchat__screen-reader-activity__timestamp\">{timestampAlt}</p>\n      {children}\n    </article>\n  );\n};\n\nScreenReaderActivity.defaultProps = {\n  children: undefined,\n  id: undefined,\n  renderAttachments: true\n};\n\nScreenReaderActivity.propTypes = {\n  activity: PropTypes.any.isRequired,\n  children: PropTypes.any,\n  id: PropTypes.string,\n  renderAttachments: PropTypes.bool\n};\n\nexport default ScreenReaderActivity;\n"]}