UNPKG

botframework-webchat-component

Version:
201 lines (193 loc) 35.1 kB
"use strict"; Object.defineProperty(exports, "__esModule", { value: true }); exports.default = useObserveFocusVisible; var _botframeworkWebchatApi = require("botframework-webchat-api"); var _react = require("react"); var _supportPseudoClass = _interopRequireDefault(require("../../Utils/supportPseudoClass")); var _useNonce3 = _interopRequireDefault(require("./useNonce")); var _useValueRef = _interopRequireDefault(require("./useValueRef")); 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) { var _i = arr == null ? null : typeof Symbol !== "undefined" && arr[Symbol.iterator] || arr["@@iterator"]; if (_i == null) return; var _arr = []; var _n = true; var _d = false; var _s, _e; try { for (_i = _i.call(arr); !(_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 usePonyfill = _botframeworkWebchatApi.hooks.usePonyfill; var INPUT_TYPES_ALLOW_LIST = ['date', 'datetime-local', 'datetime', 'email', 'month', 'number', 'password', 'search', 'tel', 'text', 'time', 'url', 'week']; /** * Computes whether the given element should automatically trigger the * `focus-visible` class being added, i.e. whether it should always match * `:focus-visible` when focused. * @param {Element} el * @return {boolean} */ function focusTriggersKeyboardModality(el) { var isContentEditable = el.isContentEditable, readOnly = el.readOnly, tagName = el.tagName, type = el.type; return tagName === 'INPUT' && INPUT_TYPES_ALLOW_LIST.includes(type) && !readOnly || tagName === 'TEXTAREA' && !readOnly || isContentEditable; } function createEventSubscription(target, types, handler) { var subscribed; var subscribe = function subscribe() { if (!subscribed) { types.forEach(function (type) { return target.addEventListener(type, handler); }); subscribed = true; } }; var unsubscribe = function unsubscribe() { if (subscribed) { types.forEach(function (type) { return target.removeEventListener(type, handler); }); subscribed = undefined; } }; return { pause: unsubscribe, resume: subscribe }; } // TODO: Add tests // 1. Focus via keyboard vs. mouse // 2. Focus via keyboard, switch app, switch back (expect to get another focusVisible after switch back) // 3. Focus via mouse, switch app, switch back (do NOT expect to get another focusVisible after switch back) function useObserveFocusVisibleForLegacyBrowsers(targetRef, onFocusVisibleRef) { var _usePonyfill = usePonyfill(), _usePonyfill2 = _slicedToArray(_usePonyfill, 1), Date = _usePonyfill2[0].Date; // This polyfill algorithm is adopted from https://github.com/WICG/focus-visible. var blurSinceRef = (0, _react.useRef)(0); var hadKeyboardEventRef = (0, _react.useRef)(true); var hasFocusVisibleRef = (0, _react.useRef)(false); var eventSubscription = (0, _react.useMemo)(function () { return createEventSubscription(document, ['mousemove', 'mousedown', 'mouseup', 'pointermove', 'pointerdown', 'pointerup', 'touchmove', 'touchstart', 'touchend'], function (event) { var _nodeName; if (((_nodeName = event.target.nodeName) === null || _nodeName === void 0 ? void 0 : _nodeName.toLowerCase()) !== 'html') { hadKeyboardEventRef.current = false; eventSubscription.pause(); } }); }, [hadKeyboardEventRef]); var setHasFocusVisible = (0, _react.useCallback)(function (nextHasFocusVisible) { if (hasFocusVisibleRef.current !== nextHasFocusVisible) { hasFocusVisibleRef.current = nextHasFocusVisible; nextHasFocusVisible && (onFocusVisibleRef === null || onFocusVisibleRef === void 0 ? void 0 : onFocusVisibleRef.current()); } }, [hasFocusVisibleRef, onFocusVisibleRef]); var handleKeyDown = (0, _react.useCallback)(function (event) { if (event.altKey || event.ctrlKey || event.metaKey) { return; } if (event.target === targetRef.current) { setHasFocusVisible(true); } hadKeyboardEventRef.current = true; }, [hadKeyboardEventRef, setHasFocusVisible, targetRef]); var handlePointerDown = (0, _react.useCallback)(function () { hadKeyboardEventRef.current = false; }, [hadKeyboardEventRef]); var handleFocus = (0, _react.useCallback)(function (_ref) { var target = _ref.target; target === targetRef.current && (hadKeyboardEventRef.current || focusTriggersKeyboardModality(target)) && setHasFocusVisible(true); }, [hadKeyboardEventRef, setHasFocusVisible, targetRef]); var handleBlur = (0, _react.useCallback)(function (event) { if (event.target === targetRef.current && hasFocusVisibleRef.current) { blurSinceRef.current = Date.now(); setHasFocusVisible(false); } }, [blurSinceRef, Date, hasFocusVisibleRef, setHasFocusVisible, targetRef]); var handleVisibilityChange = (0, _react.useCallback)(function () { if (document.visibilityState === 'hidden') { // The element is blurred due to "visibilityState" set to "hidden". // 100ms is referenced from the WICG polyfill. // eslint-disable-next-line no-magic-numbers if (Date.now() - blurSinceRef.current < 100) { hadKeyboardEventRef.current = true; } eventSubscription.resume(); } }, [blurSinceRef, Date, eventSubscription, hadKeyboardEventRef]); (0, _react.useEffect)(function () { document.addEventListener('keydown', handleKeyDown, true); document.addEventListener('mousedown', handlePointerDown, true); document.addEventListener('pointerdown', handlePointerDown, true); document.addEventListener('touchstart', handlePointerDown, true); document.addEventListener('visibilitychange', handleVisibilityChange, true); return function () { document.removeEventListener('keydown', handleKeyDown); document.removeEventListener('mousedown', handlePointerDown); document.removeEventListener('pointerdown', handlePointerDown); document.removeEventListener('touchstart', handlePointerDown); document.removeEventListener('visibilitychange', handleVisibilityChange); }; }, [handleKeyDown, handlePointerDown, handleVisibilityChange]); (0, _react.useEffect)(function () { var target = targetRef.current; target.addEventListener('blur', handleBlur, true); target.addEventListener('focus', handleFocus, true); return function () { target.removeEventListener('blur', handleBlur); target.removeEventListener('focus', handleFocus); }; // We specifically add "targetRef.current" here. // If the target element changed, we should reattach our event listeners. }, [handleBlur, handleFocus, targetRef]); (0, _react.useEffect)(function () { eventSubscription.resume(); return function () { return eventSubscription.pause(); }; }, [eventSubscription]); } function useObserveFocusVisibleForModernBrowsers(targetRef, onFocusVisibleRef) { var handleFocus = (0, _react.useCallback)(function () { var current = targetRef.current; if ( // "msMatchesSelector" is vendor-prefixed version of "matches". // eslint-disable-next-line dot-notation (current.matches || current['msMatchesSelector']).call(current, ':focus-visible')) { onFocusVisibleRef === null || onFocusVisibleRef === void 0 ? void 0 : onFocusVisibleRef.current(); } }, [onFocusVisibleRef, targetRef]); (0, _react.useEffect)(function () { var target = targetRef.current; target.addEventListener('focus', handleFocus); return function () { return target.removeEventListener('focus', handleFocus); }; // We specifically add "targetRef.current" here. // If the target element changed, we should reattach our event listeners. // eslint-disable-next-line react-hooks/exhaustive-deps }, [handleFocus, targetRef, targetRef.current]); } function useObserveFocusVisible(targetRef, onFocusVisible) { var _useNonce = (0, _useNonce3.default)(), _useNonce2 = _slicedToArray(_useNonce, 1), nonce = _useNonce2[0]; var onFocusVisibleRef = (0, _useValueRef.default)(onFocusVisible); // The nonce is use for browser capabilities. Just in case the "nonce" had changed unexpectedly, the capabilities of the browser should never change. // Thus, we are using an initial version of "nonce". In case web devs changed the "nonce" to an invalid value, we won't break rules of hooks (as stated below). var nonceRef = (0, _react.useRef)(nonce); // ":focus-visible" selector is supported from Chrome/Edge 86+ and not supported in IE11 or Safari. // Doing a capability check on pseudo classes requires injecting a stylesheet, thus nonce is needed. var supportFocusVisible = (0, _react.useMemo)(function () { return (0, _supportPseudoClass.default)(':focus-visible', nonceRef.current); }, [nonceRef]); // Since "supportPseudoClass" is a browser capability, the result should be constant during the page lifetime. // Thus, running hooks conditionally is okay here. if (supportFocusVisible) { // eslint-disable-next-line react-hooks/rules-of-hooks useObserveFocusVisibleForModernBrowsers(targetRef, onFocusVisibleRef); } else { // eslint-disable-next-line react-hooks/rules-of-hooks useObserveFocusVisibleForLegacyBrowsers(targetRef, onFocusVisibleRef); } } //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_botframeworkWebchatApi","require","_react","_supportPseudoClass","_interopRequireDefault","_useNonce3","_useValueRef","obj","__esModule","default","_slicedToArray","arr","i","_arrayWithHoles","_iterableToArrayLimit","_unsupportedIterableToArray","_nonIterableRest","TypeError","o","minLen","_arrayLikeToArray","n","Object","prototype","toString","call","slice","constructor","name","Array","from","test","len","length","arr2","_i","Symbol","iterator","_arr","_n","_d","_s","_e","next","done","push","value","err","isArray","usePonyfill","hooks","INPUT_TYPES_ALLOW_LIST","focusTriggersKeyboardModality","el","isContentEditable","readOnly","tagName","type","includes","createEventSubscription","target","types","handler","subscribed","subscribe","forEach","addEventListener","unsubscribe","removeEventListener","undefined","pause","resume","useObserveFocusVisibleForLegacyBrowsers","targetRef","onFocusVisibleRef","_usePonyfill","_usePonyfill2","Date","blurSinceRef","useRef","hadKeyboardEventRef","hasFocusVisibleRef","eventSubscription","useMemo","document","event","_nodeName","nodeName","toLowerCase","current","setHasFocusVisible","useCallback","nextHasFocusVisible","handleKeyDown","altKey","ctrlKey","metaKey","handlePointerDown","handleFocus","_ref","handleBlur","now","handleVisibilityChange","visibilityState","useEffect","useObserveFocusVisibleForModernBrowsers","matches","useObserveFocusVisible","onFocusVisible","_useNonce","useNonce","_useNonce2","nonce","useValueRef","nonceRef","supportFocusVisible","supportPseudoClass"],"sourceRoot":"component:///","sources":["../../../src/hooks/internal/useObserveFocusVisible.ts"],"sourcesContent":["import { hooks } from 'botframework-webchat-api';\nimport { MutableRefObject, RefObject, useCallback, useEffect, useMemo, useRef } from 'react';\n\nimport supportPseudoClass from '../../Utils/supportPseudoClass';\nimport useNonce from './useNonce';\nimport useValueRef from './useValueRef';\n\nconst { usePonyfill } = hooks;\n\nconst INPUT_TYPES_ALLOW_LIST = [\n  'date',\n  'datetime-local',\n  'datetime',\n  'email',\n  'month',\n  'number',\n  'password',\n  'search',\n  'tel',\n  'text',\n  'time',\n  'url',\n  'week'\n];\n\n/**\n * Computes whether the given element should automatically trigger the\n * `focus-visible` class being added, i.e. whether it should always match\n * `:focus-visible` when focused.\n * @param {Element} el\n * @return {boolean}\n */\nfunction focusTriggersKeyboardModality(el: HTMLInputElement | HTMLTextAreaElement): boolean {\n  const { isContentEditable, readOnly, tagName, type } = el;\n\n  return (\n    (tagName === 'INPUT' && INPUT_TYPES_ALLOW_LIST.includes(type) && !readOnly) ||\n    (tagName === 'TEXTAREA' && !readOnly) ||\n    isContentEditable\n  );\n}\n\nfunction createEventSubscription(\n  target: Element | Node,\n  types: string[],\n  handler: (event: Event) => void\n): {\n  pause: () => void;\n  resume: () => void;\n} {\n  let subscribed: true;\n\n  const subscribe = () => {\n    if (!subscribed) {\n      types.forEach(type => target.addEventListener(type, handler));\n      subscribed = true;\n    }\n  };\n\n  const unsubscribe = () => {\n    if (subscribed) {\n      types.forEach(type => target.removeEventListener(type, handler));\n      subscribed = undefined;\n    }\n  };\n\n  return {\n    pause: unsubscribe,\n    resume: subscribe\n  };\n}\n\n// TODO: Add tests\n//       1. Focus via keyboard vs. mouse\n//       2. Focus via keyboard, switch app, switch back (expect to get another focusVisible after switch back)\n//       3. Focus via mouse, switch app, switch back (do NOT expect to get another focusVisible after switch back)\nfunction useObserveFocusVisibleForLegacyBrowsers(\n  targetRef: RefObject<HTMLElement>,\n  onFocusVisibleRef: MutableRefObject<() => void>\n) {\n  const [{ Date }] = usePonyfill();\n  // This polyfill algorithm is adopted from https://github.com/WICG/focus-visible.\n  const blurSinceRef = useRef(0);\n  const hadKeyboardEventRef = useRef(true);\n  const hasFocusVisibleRef = useRef(false);\n\n  const eventSubscription = useMemo(\n    () =>\n      createEventSubscription(\n        document,\n        [\n          'mousemove',\n          'mousedown',\n          'mouseup',\n          'pointermove',\n          'pointerdown',\n          'pointerup',\n          'touchmove',\n          'touchstart',\n          'touchend'\n        ],\n        event => {\n          if ((event.target as HTMLElement).nodeName?.toLowerCase() !== 'html') {\n            hadKeyboardEventRef.current = false;\n            eventSubscription.pause();\n          }\n        }\n      ),\n    [hadKeyboardEventRef]\n  );\n\n  const setHasFocusVisible = useCallback(\n    nextHasFocusVisible => {\n      if (hasFocusVisibleRef.current !== nextHasFocusVisible) {\n        hasFocusVisibleRef.current = nextHasFocusVisible;\n        nextHasFocusVisible && onFocusVisibleRef?.current();\n      }\n    },\n    [hasFocusVisibleRef, onFocusVisibleRef]\n  );\n\n  const handleKeyDown = useCallback(\n    (event: KeyboardEvent) => {\n      if (event.altKey || event.ctrlKey || event.metaKey) {\n        return;\n      }\n\n      if (event.target === targetRef.current) {\n        setHasFocusVisible(true);\n      }\n\n      hadKeyboardEventRef.current = true;\n    },\n    [hadKeyboardEventRef, setHasFocusVisible, targetRef]\n  );\n\n  const handlePointerDown = useCallback(() => {\n    hadKeyboardEventRef.current = false;\n  }, [hadKeyboardEventRef]);\n\n  const handleFocus = useCallback(\n    ({ target }: Event) => {\n      target === targetRef.current &&\n        (hadKeyboardEventRef.current || focusTriggersKeyboardModality(target as HTMLInputElement)) &&\n        setHasFocusVisible(true);\n    },\n    [hadKeyboardEventRef, setHasFocusVisible, targetRef]\n  );\n\n  const handleBlur = useCallback(\n    (event: Event) => {\n      if (event.target === targetRef.current && hasFocusVisibleRef.current) {\n        blurSinceRef.current = Date.now();\n\n        setHasFocusVisible(false);\n      }\n    },\n    [blurSinceRef, Date, hasFocusVisibleRef, setHasFocusVisible, targetRef]\n  );\n\n  const handleVisibilityChange = useCallback(() => {\n    if (document.visibilityState === 'hidden') {\n      // The element is blurred due to \"visibilityState\" set to \"hidden\".\n      // 100ms is referenced from the WICG polyfill.\n      // eslint-disable-next-line no-magic-numbers\n      if (Date.now() - blurSinceRef.current < 100) {\n        hadKeyboardEventRef.current = true;\n      }\n\n      eventSubscription.resume();\n    }\n  }, [blurSinceRef, Date, eventSubscription, hadKeyboardEventRef]);\n\n  useEffect(() => {\n    document.addEventListener('keydown', handleKeyDown, true);\n    document.addEventListener('mousedown', handlePointerDown, true);\n    document.addEventListener('pointerdown', handlePointerDown, true);\n    document.addEventListener('touchstart', handlePointerDown, true);\n    document.addEventListener('visibilitychange', handleVisibilityChange, true);\n\n    return () => {\n      document.removeEventListener('keydown', handleKeyDown);\n      document.removeEventListener('mousedown', handlePointerDown);\n      document.removeEventListener('pointerdown', handlePointerDown);\n      document.removeEventListener('touchstart', handlePointerDown);\n      document.removeEventListener('visibilitychange', handleVisibilityChange);\n    };\n  }, [handleKeyDown, handlePointerDown, handleVisibilityChange]);\n\n  useEffect(() => {\n    const { current: target } = targetRef;\n\n    target.addEventListener('blur', handleBlur, true);\n    target.addEventListener('focus', handleFocus, true);\n\n    return () => {\n      target.removeEventListener('blur', handleBlur);\n      target.removeEventListener('focus', handleFocus);\n    };\n\n    // We specifically add \"targetRef.current\" here.\n    // If the target element changed, we should reattach our event listeners.\n  }, [handleBlur, handleFocus, targetRef]);\n\n  useEffect(() => {\n    eventSubscription.resume();\n\n    return () => eventSubscription.pause();\n  }, [eventSubscription]);\n}\n\nfunction useObserveFocusVisibleForModernBrowsers(\n  targetRef: RefObject<HTMLElement>,\n  onFocusVisibleRef: MutableRefObject<() => void>\n) {\n  const handleFocus = useCallback(() => {\n    const { current } = targetRef;\n\n    if (\n      // \"msMatchesSelector\" is vendor-prefixed version of \"matches\".\n      // eslint-disable-next-line dot-notation\n      (current.matches || (current['msMatchesSelector'] as (selector: string) => boolean)).call(\n        current,\n        ':focus-visible'\n      )\n    ) {\n      onFocusVisibleRef?.current();\n    }\n  }, [onFocusVisibleRef, targetRef]);\n\n  useEffect(() => {\n    const { current: target } = targetRef;\n\n    target.addEventListener('focus', handleFocus);\n\n    return () => target.removeEventListener('focus', handleFocus);\n\n    // We specifically add \"targetRef.current\" here.\n    // If the target element changed, we should reattach our event listeners.\n    // eslint-disable-next-line react-hooks/exhaustive-deps\n  }, [handleFocus, targetRef, targetRef.current]);\n}\n\nexport default function useObserveFocusVisible(targetRef: RefObject<HTMLElement>, onFocusVisible: () => void) {\n  const [nonce] = useNonce();\n  const onFocusVisibleRef = useValueRef(onFocusVisible);\n\n  // The nonce is use for browser capabilities. Just in case the \"nonce\" had changed unexpectedly, the capabilities of the browser should never change.\n  // Thus, we are using an initial version of \"nonce\". In case web devs changed the \"nonce\" to an invalid value, we won't break rules of hooks (as stated below).\n  const nonceRef = useRef(nonce);\n\n  // \":focus-visible\" selector is supported from Chrome/Edge 86+ and not supported in IE11 or Safari.\n  // Doing a capability check on pseudo classes requires injecting a stylesheet, thus nonce is needed.\n  const supportFocusVisible = useMemo(() => supportPseudoClass(':focus-visible', nonceRef.current), [nonceRef]);\n\n  // Since \"supportPseudoClass\" is a browser capability, the result should be constant during the page lifetime.\n  // Thus, running hooks conditionally is okay here.\n  if (supportFocusVisible) {\n    // eslint-disable-next-line react-hooks/rules-of-hooks\n    useObserveFocusVisibleForModernBrowsers(targetRef, onFocusVisibleRef);\n  } else {\n    // eslint-disable-next-line react-hooks/rules-of-hooks\n    useObserveFocusVisibleForLegacyBrowsers(targetRef, onFocusVisibleRef);\n  }\n}\n"],"mappings":"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"}