botframework-webchat-component
Version:
React component of botframework-webchat
191 lines (145 loc) • 20.8 kB
JavaScript
function _typeof(obj) { "@babel/helpers - typeof"; if (typeof Symbol === "function" && typeof Symbol.iterator === "symbol") { _typeof = function _typeof(obj) { return typeof obj; }; } else { _typeof = function _typeof(obj) { return obj && typeof Symbol === "function" && obj.constructor === Symbol && obj !== Symbol.prototype ? "symbol" : typeof obj; }; } return _typeof(obj); }
Object.defineProperty(exports, "__esModule", {
value: true
});
exports.default = void 0;
var _botframeworkWebchatApi = require("botframework-webchat-api");
var _propTypes = _interopRequireDefault(require("prop-types"));
var _react = _interopRequireWildcard(require("react"));
var _simpleUpdateIn = _interopRequireDefault(require("simple-update-in"));
var _createCustomEvent = _interopRequireDefault(require("./createCustomEvent"));
var _randomId = _interopRequireDefault(require("./randomId"));
var _useInternalMarkdownIt = _interopRequireDefault(require("../hooks/internal/useInternalMarkdownIt"));
var _useStyleToEmotionObject = _interopRequireDefault(require("../hooks/internal/useStyleToEmotionObject"));
var _walkMarkdownTokens = _interopRequireDefault(require("./walkMarkdownTokens"));
function _getRequireWildcardCache() { if (typeof WeakMap !== "function") return null; var cache = new WeakMap(); _getRequireWildcardCache = function _getRequireWildcardCache() { return cache; }; return cache; }
function _interopRequireWildcard(obj) { if (obj && obj.__esModule) { return obj; } if (obj === null || _typeof(obj) !== "object" && typeof obj !== "function") { return { default: obj }; } var cache = _getRequireWildcardCache(); if (cache && cache.has(obj)) { return cache.get(obj); } var newObj = {}; var hasPropertyDescriptor = Object.defineProperty && Object.getOwnPropertyDescriptor; for (var key in obj) { if (Object.prototype.hasOwnProperty.call(obj, key)) { var desc = hasPropertyDescriptor ? Object.getOwnPropertyDescriptor(obj, key) : null; if (desc && (desc.get || desc.set)) { Object.defineProperty(newObj, key, desc); } else { newObj[key] = obj[key]; } } } newObj.default = obj; if (cache) { cache.set(obj, newObj); } return newObj; }
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 _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; }
function _toConsumableArray(arr) { return _arrayWithoutHoles(arr) || _iterableToArray(arr) || _unsupportedIterableToArray(arr) || _nonIterableSpread(); }
function _nonIterableSpread() { throw new TypeError("Invalid attempt to spread 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 _iterableToArray(iter) { if (typeof Symbol !== "undefined" && Symbol.iterator in Object(iter)) return Array.from(iter); }
function _arrayWithoutHoles(arr) { if (Array.isArray(arr)) return _arrayLikeToArray(arr); }
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 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; }
function _defineProperty(obj, key, value) { if (key in obj) { Object.defineProperty(obj, key, { value: value, enumerable: true, configurable: true, writable: true }); } else { obj[key] = value; } return obj; }
var useStyleOptions = _botframeworkWebchatApi.hooks.useStyleOptions;
function replaceAnchorWithButton(markdownTokens) {
return (0, _walkMarkdownTokens.default)(markdownTokens, function (markdownToken) {
markdownToken = _objectSpread({}, markdownToken);
switch (markdownToken.type) {
case 'link_open':
markdownToken.tag = 'button';
markdownToken.attrs = [].concat(_toConsumableArray((0, _simpleUpdateIn.default)(markdownToken.attrs, [function (_ref) {
var _ref2 = _slicedToArray(_ref, 2),
name = _ref2[0],
value = _ref2[1];
return name === 'href' && value.startsWith('#');
}], function (_ref3) {
var _ref4 = _slicedToArray(_ref3, 2),
value = _ref4[1];
return ['data-markdown-href', value.substr(1)];
})), [['type', 'button']]);
break;
case 'link_close':
markdownToken.tag = 'button';
break;
default:
break;
}
return markdownToken;
});
}
var InlineMarkdown = function InlineMarkdown(_ref5) {
var children = _ref5.children,
onReference = _ref5.onReference,
references = _ref5.references;
if (typeof children !== 'string') {
console.warn('botframework-webchat: "children" prop passed to <InlineMarkdown> must be of type string.');
children = '';
}
var _useInternalMarkdownI = (0, _useInternalMarkdownIt.default)(),
_useInternalMarkdownI2 = _slicedToArray(_useInternalMarkdownI, 1),
markdownIt = _useInternalMarkdownI2[0];
var _useStyleOptions = useStyleOptions(),
_useStyleOptions2 = _slicedToArray(_useStyleOptions, 1),
accent = _useStyleOptions2[0].accent;
var styleToClassName = (0, _useStyleToEmotionObject.default)(); // We inlined the style here because this style is:
// 1. Internal to Web Chat
// 2. Not customizable from developers (other than setting `styleOptions.accent`)
var className = (0, _react.useMemo)(function () {
return styleToClassName({
'& button[data-markdown-href]': {
appearance: 'none',
backgroundColor: 'transparent',
border: 0,
color: accent,
cursor: 'pointer',
fontFamily: 'inherit',
fontSize: 'inherit',
padding: 0
}
}) + '';
}, [accent, styleToClassName]); // Markdown-It only support references in uppercase.
references = references.map(function (reference) {
return reference.toUpperCase();
});
var _references$reduce = references.reduce(function (_ref6, ref) {
var hrefToRef = _ref6.hrefToRef,
refToHref = _ref6.refToHref;
var href = (0, _randomId.default)();
return {
hrefToRef: _objectSpread(_objectSpread({}, hrefToRef), {}, _defineProperty({}, href, ref)),
refToHref: _objectSpread(_objectSpread({}, refToHref), {}, _defineProperty({}, ref, href))
};
}, {
hrefToRef: {},
refToHref: {}
}),
hrefToRef = _references$reduce.hrefToRef,
refToHref = _references$reduce.refToHref;
var html = (0, _react.useMemo)(function () {
var tree = markdownIt.parseInline(children, {
references: references.reduce(function (references, key) {
return _objectSpread(_objectSpread({}, references), {}, _defineProperty({}, key, {
href: "#".concat(refToHref[key])
}));
}, {})
}); // Turn "<a href="#retry">Retry</a>" into "<button data-ref="retry" type="button">Retry</button>"
var updatedTree = replaceAnchorWithButton(tree);
return {
__html: markdownIt.renderer.render(updatedTree)
};
}, [children, refToHref, markdownIt, references]);
var handleClick = (0, _react.useCallback)(function (event) {
event.stopPropagation();
var href = event.target.getAttribute('data-markdown-href');
href && onReference && onReference((0, _createCustomEvent.default)('reference', {
data: hrefToRef[href]
}));
}, [hrefToRef, onReference]);
return /*#__PURE__*/_react.default.createElement("span", {
className: className,
dangerouslySetInnerHTML: html,
onClick: handleClick
});
};
InlineMarkdown.defaultProps = {
children: '',
onReference: undefined,
references: []
};
InlineMarkdown.propTypes = {
children: _propTypes.default.string,
onReference: _propTypes.default.func,
references: _propTypes.default.arrayOf(_propTypes.default.string)
};
var _default = InlineMarkdown;
exports.default = _default;
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/Utils/InlineMarkdown.js"],"names":["useStyleOptions","hooks","replaceAnchorWithButton","markdownTokens","markdownToken","type","tag","attrs","name","value","startsWith","substr","InlineMarkdown","children","onReference","references","console","warn","markdownIt","accent","styleToClassName","className","appearance","backgroundColor","border","color","cursor","fontFamily","fontSize","padding","map","reference","toUpperCase","reduce","ref","hrefToRef","refToHref","href","html","tree","parseInline","key","updatedTree","__html","renderer","render","handleClick","event","stopPropagation","target","getAttribute","data","defaultProps","undefined","propTypes","PropTypes","string","func","arrayOf"],"mappings":"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","sourceRoot":"component:///","sourcesContent":["/* eslint react/no-danger: \"off\" */\n\nimport { hooks } from 'botframework-webchat-api';\nimport PropTypes from 'prop-types';\nimport React, { useCallback, useMemo } from 'react';\nimport updateIn from 'simple-update-in';\n\nimport createCustomEvent from './createCustomEvent';\nimport randomId from './randomId';\nimport useInternalMarkdownIt from '../hooks/internal/useInternalMarkdownIt';\nimport useStyleToEmotionObject from '../hooks/internal/useStyleToEmotionObject';\nimport walkMarkdownTokens from './walkMarkdownTokens';\n\nconst { useStyleOptions } = hooks;\n\nfunction replaceAnchorWithButton(markdownTokens) {\n  return walkMarkdownTokens(markdownTokens, markdownToken => {\n    markdownToken = { ...markdownToken };\n\n    switch (markdownToken.type) {\n      case 'link_open':\n        markdownToken.tag = 'button';\n        markdownToken.attrs = [\n          ...updateIn(\n            markdownToken.attrs,\n            [([name, value]) => name === 'href' && value.startsWith('#')],\n            ([, value]) => ['data-markdown-href', value.substr(1)]\n          ),\n          ['type', 'button']\n        ];\n        break;\n\n      case 'link_close':\n        markdownToken.tag = 'button';\n        break;\n\n      default:\n        break;\n    }\n\n    return markdownToken;\n  });\n}\n\nconst InlineMarkdown = ({ children, onReference, references }) => {\n  if (typeof children !== 'string') {\n    console.warn('botframework-webchat: \"children\" prop passed to <InlineMarkdown> must be of type string.');\n    children = '';\n  }\n\n  const [markdownIt] = useInternalMarkdownIt();\n  const [{ accent }] = useStyleOptions();\n  const styleToClassName = useStyleToEmotionObject();\n\n  // We inlined the style here because this style is:\n  // 1. Internal to Web Chat\n  // 2. Not customizable from developers (other than setting `styleOptions.accent`)\n  const className = useMemo(\n    () =>\n      styleToClassName({\n        '& button[data-markdown-href]': {\n          appearance: 'none',\n          backgroundColor: 'transparent',\n          border: 0,\n          color: accent,\n          cursor: 'pointer',\n          fontFamily: 'inherit',\n          fontSize: 'inherit',\n          padding: 0\n        }\n      }) + '',\n    [accent, styleToClassName]\n  );\n\n  // Markdown-It only support references in uppercase.\n  references = references.map(reference => reference.toUpperCase());\n\n  const { hrefToRef, refToHref } = references.reduce(\n    ({ hrefToRef, refToHref }, ref) => {\n      const href = randomId();\n\n      return {\n        hrefToRef: { ...hrefToRef, [href]: ref },\n        refToHref: { ...refToHref, [ref]: href }\n      };\n    },\n    { hrefToRef: {}, refToHref: {} }\n  );\n\n  const html = useMemo(() => {\n    const tree = markdownIt.parseInline(children, {\n      references: references.reduce((references, key) => ({ ...references, [key]: { href: `#${refToHref[key]}` } }), {})\n    });\n\n    // Turn \"<a href=\"#retry\">Retry</a>\" into \"<button data-ref=\"retry\" type=\"button\">Retry</button>\"\n    const updatedTree = replaceAnchorWithButton(tree);\n\n    return { __html: markdownIt.renderer.render(updatedTree) };\n  }, [children, refToHref, markdownIt, references]);\n\n  const handleClick = useCallback(\n    event => {\n      event.stopPropagation();\n\n      const href = event.target.getAttribute('data-markdown-href');\n\n      href && onReference && onReference(createCustomEvent('reference', { data: hrefToRef[href] }));\n    },\n    [hrefToRef, onReference]\n  );\n\n  return <span className={className} dangerouslySetInnerHTML={html} onClick={handleClick} />;\n};\n\nInlineMarkdown.defaultProps = {\n  children: '',\n  onReference: undefined,\n  references: []\n};\n\nInlineMarkdown.propTypes = {\n  children: PropTypes.string,\n  onReference: PropTypes.func,\n  references: PropTypes.arrayOf(PropTypes.string)\n};\n\nexport default InlineMarkdown;\n"]}
;