@commercetools-frontend/application-shell 24.10.0 → 24.12.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (77) hide show
  1. package/dist/{application-entry-point-c87294b0.cjs.dev.js → application-entry-point-18d8fba0.cjs.dev.js} +7 -4
  2. package/dist/{application-entry-point-10a5e1a5.esm.js → application-entry-point-1b23fb6b.esm.js} +6 -4
  3. package/dist/{application-entry-point-8c4b8e53.cjs.prod.js → application-entry-point-74a06151.cjs.prod.js} +4 -3
  4. package/dist/commercetools-frontend-application-shell.cjs.dev.js +12 -11
  5. package/dist/commercetools-frontend-application-shell.cjs.prod.js +12 -11
  6. package/dist/commercetools-frontend-application-shell.esm.js +12 -11
  7. package/dist/{custom-view-dev-host-5039dc1d.esm.js → custom-view-dev-host-091163ea.esm.js} +20 -16
  8. package/dist/{custom-view-dev-host-b5e3a16a.cjs.dev.js → custom-view-dev-host-17daf42a.cjs.dev.js} +21 -16
  9. package/dist/{custom-view-dev-host-21561a3a.cjs.prod.js → custom-view-dev-host-a682a499.cjs.prod.js} +21 -16
  10. package/dist/declarations-connectors/src/index.d.ts +1 -1
  11. package/dist/declarations-connectors/src/utils/index.d.ts +1 -0
  12. package/dist/declarations-connectors/src/utils/select-user-language-from-storage/index.d.ts +1 -0
  13. package/dist/declarations-connectors/src/utils/select-user-language-from-storage/select-user-language-from-storage.d.ts +1 -0
  14. package/dist/{index-17f02024.cjs.dev.js → index-1d1cc31f.cjs.dev.js} +225 -368
  15. package/dist/{index-e4de734f.esm.js → index-1dadca21.esm.js} +227 -368
  16. package/dist/{index-614accc4.cjs.dev.js → index-25183095.cjs.dev.js} +2 -2
  17. package/dist/{index-11b385bb.cjs.prod.js → index-3cfc1f1e.cjs.prod.js} +213 -348
  18. package/dist/{index-245e2980.cjs.prod.js → index-52c724ed.cjs.prod.js} +2 -2
  19. package/dist/{index-86039df7.esm.js → index-5aaa33bb.esm.js} +2 -2
  20. package/dist/{navbar-022383bd.cjs.dev.js → navbar-586f7774.cjs.dev.js} +111 -92
  21. package/dist/{navbar-844d350d.esm.js → navbar-88e0fd1f.esm.js} +110 -92
  22. package/dist/{navbar-acc2cd1b.cjs.prod.js → navbar-93183a2d.cjs.prod.js} +111 -92
  23. package/dist/oidc-258fc018.cjs.prod.js +115 -0
  24. package/dist/oidc-35e8e62a.esm.js +100 -0
  25. package/dist/oidc-87d116c1.cjs.dev.js +115 -0
  26. package/dist/{oidc-callback-c014b1b0.esm.js → oidc-callback-019d623d.esm.js} +16 -14
  27. package/dist/{oidc-callback-ad64d7f6.cjs.dev.js → oidc-callback-47743232.cjs.dev.js} +16 -14
  28. package/dist/{oidc-callback-9beece27.cjs.prod.js → oidc-callback-6bdb3c6f.cjs.prod.js} +16 -14
  29. package/dist/{project-container-3e3c7013.cjs.dev.js → project-container-2245f020.cjs.dev.js} +53 -19
  30. package/dist/{project-container-e11b2fc6.esm.js → project-container-7fce9e66.esm.js} +52 -19
  31. package/dist/{project-container-f1710162.cjs.prod.js → project-container-954dbf0f.cjs.prod.js} +53 -19
  32. package/dist/{project-expired-39589063.esm.js → project-expired-1b0845c5.esm.js} +12 -11
  33. package/dist/{project-expired-f29e6d47.cjs.dev.js → project-expired-c941b592.cjs.dev.js} +12 -11
  34. package/dist/{project-expired-59169760.cjs.prod.js → project-expired-ee8b232c.cjs.prod.js} +12 -11
  35. package/dist/{project-not-found-88730a64.esm.js → project-not-found-340217f6.esm.js} +11 -10
  36. package/dist/{project-not-found-d968ede6.cjs.dev.js → project-not-found-9b7cfe88.cjs.dev.js} +11 -10
  37. package/dist/{project-not-found-625f0e91.cjs.prod.js → project-not-found-9cee9625.cjs.prod.js} +11 -10
  38. package/dist/{project-not-initialized-6d69541c.esm.js → project-not-initialized-55fd8df4.esm.js} +12 -11
  39. package/dist/{project-not-initialized-22d54dab.cjs.prod.js → project-not-initialized-7a058b68.cjs.prod.js} +12 -11
  40. package/dist/{project-not-initialized-f346dc17.cjs.dev.js → project-not-initialized-7b3843a3.cjs.dev.js} +12 -11
  41. package/dist/{project-suspended-d48e7d51.cjs.prod.js → project-suspended-12618898.cjs.prod.js} +11 -10
  42. package/dist/{project-suspended-be2e3265.esm.js → project-suspended-529b09d6.esm.js} +11 -10
  43. package/dist/{project-suspended-6a886974.cjs.dev.js → project-suspended-78e94b85.cjs.dev.js} +11 -10
  44. package/dist/{redirect-to-login-12f467b8.cjs.prod.js → redirect-to-login-3e4a6434.cjs.prod.js} +13 -10
  45. package/dist/{redirect-to-login-3bee13ba.cjs.dev.js → redirect-to-login-66ea895a.cjs.dev.js} +13 -10
  46. package/dist/{redirect-to-login-2944c890.esm.js → redirect-to-login-edbfacbc.esm.js} +13 -10
  47. package/dist/{redirect-to-logout-645e12ca.cjs.prod.js → redirect-to-logout-52a7810f.cjs.prod.js} +14 -12
  48. package/dist/{redirect-to-logout-0196921c.esm.js → redirect-to-logout-5d5fc361.esm.js} +14 -12
  49. package/dist/{redirect-to-logout-477ea146.cjs.dev.js → redirect-to-logout-b331b037.cjs.dev.js} +14 -12
  50. package/dist/{redirector-72ccfbc2.cjs.dev.js → redirector-0efdd994.cjs.dev.js} +4 -3
  51. package/dist/{redirector-d856975f.esm.js → redirector-656c6ee7.esm.js} +4 -3
  52. package/dist/{redirector-0c72d0a4.cjs.prod.js → redirector-c858d578.cjs.prod.js} +4 -3
  53. package/dist/{requests-in-flight-loader-82b93073.esm.js → requests-in-flight-loader-20021ccc.esm.js} +11 -10
  54. package/dist/{requests-in-flight-loader-08cfa2ce.cjs.prod.js → requests-in-flight-loader-64d2e12d.cjs.prod.js} +11 -10
  55. package/dist/{requests-in-flight-loader-fb6a69f6.cjs.dev.js → requests-in-flight-loader-83cab813.cjs.dev.js} +11 -10
  56. package/dist/{service-page-project-switcher-2d65c6f7.cjs.dev.js → service-page-project-switcher-49dabe13.cjs.dev.js} +1 -1
  57. package/dist/{service-page-project-switcher-1e41f587.esm.js → service-page-project-switcher-6cdd506b.esm.js} +1 -1
  58. package/dist/{service-page-project-switcher-2746dbcc.cjs.prod.js → service-page-project-switcher-f1b43eb7.cjs.prod.js} +1 -1
  59. package/dist/{use-applications-menu-823a2492.cjs.dev.js → use-applications-menu-48d924bd.cjs.prod.js} +47 -39
  60. package/dist/{use-applications-menu-14a5a1f4.cjs.prod.js → use-applications-menu-7f548a7a.cjs.dev.js} +47 -39
  61. package/dist/{use-applications-menu-1514af11.esm.js → use-applications-menu-b871849c.esm.js} +44 -37
  62. package/dist/{user-settings-menu-d75f4958.cjs.prod.js → user-settings-menu-6660f508.cjs.prod.js} +29 -22
  63. package/dist/{user-settings-menu-f98bea89.esm.js → user-settings-menu-afa82f2a.esm.js} +29 -22
  64. package/dist/{user-settings-menu-6113cdd3.cjs.dev.js → user-settings-menu-f5c74042.cjs.dev.js} +29 -22
  65. package/package.json +22 -22
  66. package/ssr/dist/commercetools-frontend-application-shell-ssr.cjs.dev.js +2 -1
  67. package/ssr/dist/commercetools-frontend-application-shell-ssr.cjs.prod.js +2 -1
  68. package/ssr/dist/commercetools-frontend-application-shell-ssr.esm.js +2 -1
  69. package/test-utils/dist/commercetools-frontend-application-shell-test-utils.cjs.dev.js +26 -22
  70. package/test-utils/dist/commercetools-frontend-application-shell-test-utils.cjs.prod.js +26 -22
  71. package/test-utils/dist/commercetools-frontend-application-shell-test-utils.esm.js +20 -17
  72. package/dist/oidc-8827f9fe.cjs.dev.js +0 -98
  73. package/dist/oidc-b2520905.esm.js +0 -84
  74. package/dist/oidc-d74e6aa2.cjs.prod.js +0 -98
  75. package/dist/quick-access-67db2a39.cjs.dev.js +0 -1893
  76. package/dist/quick-access-8c34e976.esm.js +0 -1865
  77. package/dist/quick-access-9001b324.cjs.prod.js +0 -1875
@@ -1,1865 +0,0 @@
1
- import _slicedToArray from '@babel/runtime-corejs3/helpers/esm/slicedToArray';
2
- import _mapInstanceProperty from '@babel/runtime-corejs3/core-js-stable/instance/map';
3
- import _Array$isArray from '@babel/runtime-corejs3/core-js-stable/array/is-array';
4
- import _JSON$stringify from '@babel/runtime-corejs3/core-js-stable/json/stringify';
5
- import _forEachInstanceProperty from '@babel/runtime-corejs3/core-js-stable/instance/for-each';
6
- import _trimInstanceProperty from '@babel/runtime-corejs3/core-js-stable/instance/trim';
7
- import _startsWithInstanceProperty from '@babel/runtime-corejs3/core-js-stable/instance/starts-with';
8
- import { useReducer, useRef, useCallback, useState, useEffect } from 'react';
9
- import { useApolloClient } from '@apollo/client/react';
10
- import { useFeatureToggles } from '@flopflip/react-broadcast';
11
- import { oneLineTrim } from 'common-tags';
12
- import debounce from 'debounce-async';
13
- import { defineMessages, useIntl, FormattedMessage } from 'react-intl';
14
- import { useHistory } from 'react-router-dom';
15
- import { useApplicationContext } from '@commercetools-frontend/application-shell-connectors';
16
- import { SUPPORT_PORTAL_URL, LOGOUT_REASONS, GRAPHQL_TARGETS, MC_API_PROXY_TARGETS } from '@commercetools-frontend/constants';
17
- import { hasSomePermissions } from '@commercetools-frontend/permissions';
18
- import { useAsyncDispatch, actions } from '@commercetools-frontend/sdk';
19
- import { l as location } from './location-f21dbc25.esm.js';
20
- import _Object$keys from '@babel/runtime-corejs3/core-js-stable/object/keys';
21
- import _Object$getOwnPropertySymbols from '@babel/runtime-corejs3/core-js-stable/object/get-own-property-symbols';
22
- import _Object$getOwnPropertyDescriptor from '@babel/runtime-corejs3/core-js-stable/object/get-own-property-descriptor';
23
- import _Object$getOwnPropertyDescriptors from '@babel/runtime-corejs3/core-js-stable/object/get-own-property-descriptors';
24
- import _Object$defineProperties from '@babel/runtime-corejs3/core-js-stable/object/define-properties';
25
- import _Object$defineProperty from '@babel/runtime-corejs3/core-js-stable/object/define-property';
26
- import _defineProperty from '@babel/runtime-corejs3/helpers/esm/defineProperty';
27
- import _includesInstanceProperty from '@babel/runtime-corejs3/core-js-stable/instance/includes';
28
- import _sliceInstanceProperty from '@babel/runtime-corejs3/core-js-stable/instance/slice';
29
- import _filterInstanceProperty from '@babel/runtime-corejs3/core-js-stable/instance/filter';
30
- import _findIndexInstanceProperty from '@babel/runtime-corejs3/core-js-stable/instance/find-index';
31
- import { css, keyframes, ClassNames } from '@emotion/react';
32
- import Fuse from 'fuse.js';
33
- import last from 'lodash/last';
34
- import { customProperties, designTokens } from '@commercetools-uikit/design-system';
35
- import { AngleThinRightIcon, SearchIcon } from '@commercetools-uikit/icons';
36
- import LoadingSpinner from '@commercetools-uikit/loading-spinner';
37
- import { jsxs, jsx } from '@emotion/react/jsx-runtime';
38
- import { B as ButlerContainer, p as pimIndexerStates } from './index-e4de734f.esm.js';
39
- import _Promise from '@babel/runtime-corejs3/core-js-stable/promise';
40
- import _reduceInstanceProperty from '@babel/runtime-corejs3/core-js-stable/instance/reduce';
41
- import _findInstanceProperty from '@babel/runtime-corejs3/core-js-stable/instance/find';
42
- import './index-86039df7.esm.js';
43
- import '@babel/runtime-corejs3/core-js-stable/object/entries';
44
- import '@babel/runtime-corejs3/core-js-stable/instance/concat';
45
- import '@babel/runtime-corejs3/core-js-stable/reflect/has';
46
- import '@reduxjs/toolkit';
47
- import 'lodash/mapValues';
48
- import 'omit-empty-es';
49
- import 'redux';
50
- import 'redux-thunk';
51
- import '@commercetools-frontend/notifications';
52
- import '@babel/runtime-corejs3/core-js-stable/instance/index-of';
53
- import '@commercetools-frontend/sentry';
54
- import '@commercetools-frontend/react-notifications';
55
- import 'redux-logger';
56
- import '@emotion/styled/base';
57
- import '@commercetools-frontend/application-components';
58
- import '@commercetools-frontend/i18n';
59
- import './oidc-b2520905.esm.js';
60
- import '@babel/runtime-corejs3/core-js-stable/url';
61
- import '@commercetools-uikit/spacings';
62
- import '@commercetools-uikit/flat-button';
63
- import '@babel/runtime-corejs3/helpers/objectWithoutProperties';
64
- import 'memoize-one';
65
- import 'react-select';
66
- import '@commercetools-uikit/accessible-hidden';
67
- import '@commercetools-uikit/select-input';
68
- import '@commercetools-uikit/text';
69
- import '@commercetools-frontend/assets/images/ct-logo.svg';
70
- import '@commercetools-frontend/browser-history';
71
- import '@commercetools-frontend/l10n';
72
- import '@babel/runtime-corejs3/core-js-stable/reflect/construct';
73
- import '@babel/runtime-corejs3/helpers/classCallCheck';
74
- import '@babel/runtime-corejs3/helpers/createClass';
75
- import '@babel/runtime-corejs3/helpers/possibleConstructorReturn';
76
- import '@babel/runtime-corejs3/helpers/getPrototypeOf';
77
- import '@babel/runtime-corejs3/helpers/inherits';
78
- import '@babel/runtime-corejs3/core-js-stable/object/from-entries';
79
- import '@babel/runtime-corejs3/core-js-stable/instance/flags';
80
- import '@flopflip/combine-adapters';
81
- import '@flopflip/http-adapter';
82
- import '@flopflip/launchdarkly-adapter';
83
- import '@flopflip/types';
84
- import 'react-redux';
85
- import '@commercetools-uikit/design-system/materials/resets.css';
86
- import '@commercetools-frontend/application-config/ssr';
87
- import '@flopflip/memory-adapter';
88
- import '@babel/runtime-corejs3/core-js-stable/instance/some';
89
- import '@commercetools-frontend/actions-global';
90
-
91
- function _EMOTION_STRINGIFIED_CSS_ERROR__$1() { return "You have tried to stringify object returned from `css` function. It isn't supposed to be used directly (e.g. as value of the `className` prop), but rather handed to emotion so it can handle it (e.g. as value of `css` prop)."; }
92
- var _ref$1 = process.env.NODE_ENV === "production" ? {
93
- name: "13n9jnb",
94
- styles: "align-self:center;>*{display:block;}"
95
- } : {
96
- name: "1qkkrpz-ButlerCommand",
97
- styles: "align-self:center;>*{display:block;};label:ButlerCommand;/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbImJ1dGxlci1jb21tYW5kLnRzeCJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFrRGdCIiwiZmlsZSI6ImJ1dGxlci1jb21tYW5kLnRzeCIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IE1vdXNlRXZlbnRIYW5kbGVyIH0gZnJvbSAncmVhY3QnO1xuaW1wb3J0IHsgY3NzIH0gZnJvbSAnQGVtb3Rpb24vcmVhY3QnO1xuaW1wb3J0IHsgY3VzdG9tUHJvcGVydGllcyB9IGZyb20gJ0Bjb21tZXJjZXRvb2xzLXVpa2l0L2Rlc2lnbi1zeXN0ZW0nO1xuaW1wb3J0IHsgQW5nbGVUaGluUmlnaHRJY29uIH0gZnJvbSAnQGNvbW1lcmNldG9vbHMtdWlraXQvaWNvbnMnO1xuaW1wb3J0IHR5cGUgeyBDb21tYW5kIH0gZnJvbSAnLi4vdHlwZXMnO1xuXG50eXBlIFByb3BzID0ge1xuICBjb21tYW5kOiBDb21tYW5kO1xuICBpc1NlbGVjdGVkPzogYm9vbGVhbjtcbiAgb25DbGljazogTW91c2VFdmVudEhhbmRsZXI8SFRNTERpdkVsZW1lbnQ+O1xuICBvbk1vdXNlRW50ZXI6IE1vdXNlRXZlbnRIYW5kbGVyPEhUTUxEaXZFbGVtZW50Pjtcbn07XG5cbmNvbnN0IEJ1dGxlckNvbW1hbmQgPSAocHJvcHM6IFByb3BzKSA9PiAoXG4gIDxkaXZcbiAgICBrZXk9e3Byb3BzLmNvbW1hbmQuaWR9XG4gICAgZGF0YS10ZXN0aWQ9e2BxdWljay1hY2Nlc3MtcmVzdWx0KCR7cHJvcHMuY29tbWFuZC5pZH0pYH1cbiAgICBhcmlhLWN1cnJlbnQ9e3Byb3BzLmlzU2VsZWN0ZWQgPT09IHRydWUgPyAndHJ1ZScgOiAnZmFsc2UnfVxuICAgIGNzcz17Y3NzYFxuICAgICAgZGlzcGxheTogZmxleDtcbiAgICAgIHBhZGRpbmc6IDAgJHtjdXN0b21Qcm9wZXJ0aWVzLnNwYWNpbmdNfTtcbiAgICAgIGhlaWdodDogMzZweDtcbiAgICAgIGZvbnQtc2l6ZTogMTZweDtcbiAgICAgIGZvbnQtd2VpZ2h0OiAyMDA7XG4gICAgICBsaW5lLWhlaWdodDogMzZweDtcbiAgICAgIGN1cnNvcjogZGVmYXVsdDtcbiAgICAgICR7cHJvcHMuaXNTZWxlY3RlZCA9PT0gdHJ1ZVxuICAgICAgICA/IGBcbiAgICAgICAgICAgIGJhY2tncm91bmQ6ICR7Y3VzdG9tUHJvcGVydGllcy5jb2xvckFjY2VudH07XG4gICAgICAgICAgICBjb2xvcjogJHtjdXN0b21Qcm9wZXJ0aWVzLmNvbG9yU3VyZmFjZX07XG4gICAgICAgICAgYFxuICAgICAgICA6ICcnfVxuICAgIGB9XG4gICAgb25Nb3VzZUVudGVyPXtwcm9wcy5vbk1vdXNlRW50ZXJ9XG4gICAgb25DbGljaz17cHJvcHMub25DbGlja31cbiAgPlxuICAgIDxkaXZcbiAgICAgIGNzcz17Y3NzYFxuICAgICAgICBmbGV4OiAxIGF1dG87XG4gICAgICAgIHdoaXRlLXNwYWNlOiBub3dyYXA7XG4gICAgICAgIG92ZXJmbG93OiBoaWRkZW47XG4gICAgICAgIHRleHQtb3ZlcmZsb3c6IGVsbGlwc2lzO1xuICAgICAgYH1cbiAgICA+XG4gICAgICB7cHJvcHMuY29tbWFuZC50ZXh0fVxuICAgIDwvZGl2PlxuICAgIHsoKEFycmF5LmlzQXJyYXkocHJvcHMuY29tbWFuZC5zdWJDb21tYW5kcykgJiZcbiAgICAgIHByb3BzLmNvbW1hbmQuc3ViQ29tbWFuZHMubGVuZ3RoID4gMCkgfHxcbiAgICAgIHR5cGVvZiBwcm9wcy5jb21tYW5kLnN1YkNvbW1hbmRzID09PSAnZnVuY3Rpb24nKSAmJiAoXG4gICAgICA8ZGl2XG4gICAgICAgIGNzcz17Y3NzYFxuICAgICAgICAgIGFsaWduLXNlbGY6IGNlbnRlcjtcbiAgICAgICAgICA+ICoge1xuICAgICAgICAgICAgZGlzcGxheTogYmxvY2s7XG4gICAgICAgICAgfVxuICAgICAgICBgfVxuICAgICAgPlxuICAgICAgICA8QW5nbGVUaGluUmlnaHRJY29uXG4gICAgICAgICAgc2l6ZT1cIm1lZGl1bVwiXG4gICAgICAgICAgY29sb3I9e3Byb3BzLmlzU2VsZWN0ZWQgPyAnc3VyZmFjZScgOiAnbmV1dHJhbDYwJ31cbiAgICAgICAgLz5cbiAgICAgIDwvZGl2PlxuICAgICl9XG4gIDwvZGl2PlxuKTtcblxuQnV0bGVyQ29tbWFuZC5kaXNwbGF5TmFtZSA9ICdCdXRsZXJDb21tYW5kJztcblxuZXhwb3J0IGRlZmF1bHQgQnV0bGVyQ29tbWFuZDtcbiJdfQ== */",
98
- toString: _EMOTION_STRINGIFIED_CSS_ERROR__$1
99
- };
100
- var _ref2 = process.env.NODE_ENV === "production" ? {
101
- name: "lhzw0z",
102
- styles: "flex:1 auto;white-space:nowrap;overflow:hidden;text-overflow:ellipsis"
103
- } : {
104
- name: "zq3ooq-ButlerCommand",
105
- styles: "flex:1 auto;white-space:nowrap;overflow:hidden;text-overflow:ellipsis;label:ButlerCommand;/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbImJ1dGxlci1jb21tYW5kLnRzeCJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFxQ2MiLCJmaWxlIjoiYnV0bGVyLWNvbW1hbmQudHN4Iiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgTW91c2VFdmVudEhhbmRsZXIgfSBmcm9tICdyZWFjdCc7XG5pbXBvcnQgeyBjc3MgfSBmcm9tICdAZW1vdGlvbi9yZWFjdCc7XG5pbXBvcnQgeyBjdXN0b21Qcm9wZXJ0aWVzIH0gZnJvbSAnQGNvbW1lcmNldG9vbHMtdWlraXQvZGVzaWduLXN5c3RlbSc7XG5pbXBvcnQgeyBBbmdsZVRoaW5SaWdodEljb24gfSBmcm9tICdAY29tbWVyY2V0b29scy11aWtpdC9pY29ucyc7XG5pbXBvcnQgdHlwZSB7IENvbW1hbmQgfSBmcm9tICcuLi90eXBlcyc7XG5cbnR5cGUgUHJvcHMgPSB7XG4gIGNvbW1hbmQ6IENvbW1hbmQ7XG4gIGlzU2VsZWN0ZWQ/OiBib29sZWFuO1xuICBvbkNsaWNrOiBNb3VzZUV2ZW50SGFuZGxlcjxIVE1MRGl2RWxlbWVudD47XG4gIG9uTW91c2VFbnRlcjogTW91c2VFdmVudEhhbmRsZXI8SFRNTERpdkVsZW1lbnQ+O1xufTtcblxuY29uc3QgQnV0bGVyQ29tbWFuZCA9IChwcm9wczogUHJvcHMpID0+IChcbiAgPGRpdlxuICAgIGtleT17cHJvcHMuY29tbWFuZC5pZH1cbiAgICBkYXRhLXRlc3RpZD17YHF1aWNrLWFjY2Vzcy1yZXN1bHQoJHtwcm9wcy5jb21tYW5kLmlkfSlgfVxuICAgIGFyaWEtY3VycmVudD17cHJvcHMuaXNTZWxlY3RlZCA9PT0gdHJ1ZSA/ICd0cnVlJyA6ICdmYWxzZSd9XG4gICAgY3NzPXtjc3NgXG4gICAgICBkaXNwbGF5OiBmbGV4O1xuICAgICAgcGFkZGluZzogMCAke2N1c3RvbVByb3BlcnRpZXMuc3BhY2luZ019O1xuICAgICAgaGVpZ2h0OiAzNnB4O1xuICAgICAgZm9udC1zaXplOiAxNnB4O1xuICAgICAgZm9udC13ZWlnaHQ6IDIwMDtcbiAgICAgIGxpbmUtaGVpZ2h0OiAzNnB4O1xuICAgICAgY3Vyc29yOiBkZWZhdWx0O1xuICAgICAgJHtwcm9wcy5pc1NlbGVjdGVkID09PSB0cnVlXG4gICAgICAgID8gYFxuICAgICAgICAgICAgYmFja2dyb3VuZDogJHtjdXN0b21Qcm9wZXJ0aWVzLmNvbG9yQWNjZW50fTtcbiAgICAgICAgICAgIGNvbG9yOiAke2N1c3RvbVByb3BlcnRpZXMuY29sb3JTdXJmYWNlfTtcbiAgICAgICAgICBgXG4gICAgICAgIDogJyd9XG4gICAgYH1cbiAgICBvbk1vdXNlRW50ZXI9e3Byb3BzLm9uTW91c2VFbnRlcn1cbiAgICBvbkNsaWNrPXtwcm9wcy5vbkNsaWNrfVxuICA+XG4gICAgPGRpdlxuICAgICAgY3NzPXtjc3NgXG4gICAgICAgIGZsZXg6IDEgYXV0bztcbiAgICAgICAgd2hpdGUtc3BhY2U6IG5vd3JhcDtcbiAgICAgICAgb3ZlcmZsb3c6IGhpZGRlbjtcbiAgICAgICAgdGV4dC1vdmVyZmxvdzogZWxsaXBzaXM7XG4gICAgICBgfVxuICAgID5cbiAgICAgIHtwcm9wcy5jb21tYW5kLnRleHR9XG4gICAgPC9kaXY+XG4gICAgeygoQXJyYXkuaXNBcnJheShwcm9wcy5jb21tYW5kLnN1YkNvbW1hbmRzKSAmJlxuICAgICAgcHJvcHMuY29tbWFuZC5zdWJDb21tYW5kcy5sZW5ndGggPiAwKSB8fFxuICAgICAgdHlwZW9mIHByb3BzLmNvbW1hbmQuc3ViQ29tbWFuZHMgPT09ICdmdW5jdGlvbicpICYmIChcbiAgICAgIDxkaXZcbiAgICAgICAgY3NzPXtjc3NgXG4gICAgICAgICAgYWxpZ24tc2VsZjogY2VudGVyO1xuICAgICAgICAgID4gKiB7XG4gICAgICAgICAgICBkaXNwbGF5OiBibG9jaztcbiAgICAgICAgICB9XG4gICAgICAgIGB9XG4gICAgICA+XG4gICAgICAgIDxBbmdsZVRoaW5SaWdodEljb25cbiAgICAgICAgICBzaXplPVwibWVkaXVtXCJcbiAgICAgICAgICBjb2xvcj17cHJvcHMuaXNTZWxlY3RlZCA/ICdzdXJmYWNlJyA6ICduZXV0cmFsNjAnfVxuICAgICAgICAvPlxuICAgICAgPC9kaXY+XG4gICAgKX1cbiAgPC9kaXY+XG4pO1xuXG5CdXRsZXJDb21tYW5kLmRpc3BsYXlOYW1lID0gJ0J1dGxlckNvbW1hbmQnO1xuXG5leHBvcnQgZGVmYXVsdCBCdXRsZXJDb21tYW5kO1xuIl19 */",
106
- toString: _EMOTION_STRINGIFIED_CSS_ERROR__$1
107
- };
108
- const ButlerCommand = props => jsxs("div", {
109
- "data-testid": `quick-access-result(${props.command.id})`,
110
- "aria-current": props.isSelected === true ? 'true' : 'false',
111
- css: /*#__PURE__*/css("display:flex;padding:0 ", customProperties.spacingM, ";height:36px;font-size:16px;font-weight:200;line-height:36px;cursor:default;", props.isSelected === true ? `
112
- background: ${customProperties.colorAccent};
113
- color: ${customProperties.colorSurface};
114
- ` : '', ";" + (process.env.NODE_ENV === "production" ? "" : ";label:ButlerCommand;"), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbImJ1dGxlci1jb21tYW5kLnRzeCJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFrQlkiLCJmaWxlIjoiYnV0bGVyLWNvbW1hbmQudHN4Iiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgTW91c2VFdmVudEhhbmRsZXIgfSBmcm9tICdyZWFjdCc7XG5pbXBvcnQgeyBjc3MgfSBmcm9tICdAZW1vdGlvbi9yZWFjdCc7XG5pbXBvcnQgeyBjdXN0b21Qcm9wZXJ0aWVzIH0gZnJvbSAnQGNvbW1lcmNldG9vbHMtdWlraXQvZGVzaWduLXN5c3RlbSc7XG5pbXBvcnQgeyBBbmdsZVRoaW5SaWdodEljb24gfSBmcm9tICdAY29tbWVyY2V0b29scy11aWtpdC9pY29ucyc7XG5pbXBvcnQgdHlwZSB7IENvbW1hbmQgfSBmcm9tICcuLi90eXBlcyc7XG5cbnR5cGUgUHJvcHMgPSB7XG4gIGNvbW1hbmQ6IENvbW1hbmQ7XG4gIGlzU2VsZWN0ZWQ/OiBib29sZWFuO1xuICBvbkNsaWNrOiBNb3VzZUV2ZW50SGFuZGxlcjxIVE1MRGl2RWxlbWVudD47XG4gIG9uTW91c2VFbnRlcjogTW91c2VFdmVudEhhbmRsZXI8SFRNTERpdkVsZW1lbnQ+O1xufTtcblxuY29uc3QgQnV0bGVyQ29tbWFuZCA9IChwcm9wczogUHJvcHMpID0+IChcbiAgPGRpdlxuICAgIGtleT17cHJvcHMuY29tbWFuZC5pZH1cbiAgICBkYXRhLXRlc3RpZD17YHF1aWNrLWFjY2Vzcy1yZXN1bHQoJHtwcm9wcy5jb21tYW5kLmlkfSlgfVxuICAgIGFyaWEtY3VycmVudD17cHJvcHMuaXNTZWxlY3RlZCA9PT0gdHJ1ZSA/ICd0cnVlJyA6ICdmYWxzZSd9XG4gICAgY3NzPXtjc3NgXG4gICAgICBkaXNwbGF5OiBmbGV4O1xuICAgICAgcGFkZGluZzogMCAke2N1c3RvbVByb3BlcnRpZXMuc3BhY2luZ019O1xuICAgICAgaGVpZ2h0OiAzNnB4O1xuICAgICAgZm9udC1zaXplOiAxNnB4O1xuICAgICAgZm9udC13ZWlnaHQ6IDIwMDtcbiAgICAgIGxpbmUtaGVpZ2h0OiAzNnB4O1xuICAgICAgY3Vyc29yOiBkZWZhdWx0O1xuICAgICAgJHtwcm9wcy5pc1NlbGVjdGVkID09PSB0cnVlXG4gICAgICAgID8gYFxuICAgICAgICAgICAgYmFja2dyb3VuZDogJHtjdXN0b21Qcm9wZXJ0aWVzLmNvbG9yQWNjZW50fTtcbiAgICAgICAgICAgIGNvbG9yOiAke2N1c3RvbVByb3BlcnRpZXMuY29sb3JTdXJmYWNlfTtcbiAgICAgICAgICBgXG4gICAgICAgIDogJyd9XG4gICAgYH1cbiAgICBvbk1vdXNlRW50ZXI9e3Byb3BzLm9uTW91c2VFbnRlcn1cbiAgICBvbkNsaWNrPXtwcm9wcy5vbkNsaWNrfVxuICA+XG4gICAgPGRpdlxuICAgICAgY3NzPXtjc3NgXG4gICAgICAgIGZsZXg6IDEgYXV0bztcbiAgICAgICAgd2hpdGUtc3BhY2U6IG5vd3JhcDtcbiAgICAgICAgb3ZlcmZsb3c6IGhpZGRlbjtcbiAgICAgICAgdGV4dC1vdmVyZmxvdzogZWxsaXBzaXM7XG4gICAgICBgfVxuICAgID5cbiAgICAgIHtwcm9wcy5jb21tYW5kLnRleHR9XG4gICAgPC9kaXY+XG4gICAgeygoQXJyYXkuaXNBcnJheShwcm9wcy5jb21tYW5kLnN1YkNvbW1hbmRzKSAmJlxuICAgICAgcHJvcHMuY29tbWFuZC5zdWJDb21tYW5kcy5sZW5ndGggPiAwKSB8fFxuICAgICAgdHlwZW9mIHByb3BzLmNvbW1hbmQuc3ViQ29tbWFuZHMgPT09ICdmdW5jdGlvbicpICYmIChcbiAgICAgIDxkaXZcbiAgICAgICAgY3NzPXtjc3NgXG4gICAgICAgICAgYWxpZ24tc2VsZjogY2VudGVyO1xuICAgICAgICAgID4gKiB7XG4gICAgICAgICAgICBkaXNwbGF5OiBibG9jaztcbiAgICAgICAgICB9XG4gICAgICAgIGB9XG4gICAgICA+XG4gICAgICAgIDxBbmdsZVRoaW5SaWdodEljb25cbiAgICAgICAgICBzaXplPVwibWVkaXVtXCJcbiAgICAgICAgICBjb2xvcj17cHJvcHMuaXNTZWxlY3RlZCA/ICdzdXJmYWNlJyA6ICduZXV0cmFsNjAnfVxuICAgICAgICAvPlxuICAgICAgPC9kaXY+XG4gICAgKX1cbiAgPC9kaXY+XG4pO1xuXG5CdXRsZXJDb21tYW5kLmRpc3BsYXlOYW1lID0gJ0J1dGxlckNvbW1hbmQnO1xuXG5leHBvcnQgZGVmYXVsdCBCdXRsZXJDb21tYW5kO1xuIl19 */"),
115
- onMouseEnter: props.onMouseEnter,
116
- onClick: props.onClick,
117
- children: [jsx("div", {
118
- css: _ref2,
119
- children: props.command.text
120
- }), (_Array$isArray(props.command.subCommands) && props.command.subCommands.length > 0 || typeof props.command.subCommands === 'function') && jsx("div", {
121
- css: _ref$1,
122
- children: jsx(AngleThinRightIcon, {
123
- size: "medium",
124
- color: props.isSelected ? 'surface' : 'neutral60'
125
- })
126
- })]
127
- }, props.command.id);
128
- ButlerCommand.displayName = 'ButlerCommand';
129
-
130
- var messages = defineMessages({
131
- inputPlacehoder: {
132
- id: 'QuickAccess.inputPlaceholder',
133
- defaultMessage: 'Go to...'
134
- },
135
- offline: {
136
- id: 'QuickAccess.offline',
137
- defaultMessage: 'Offline'
138
- },
139
- noResults: {
140
- id: 'QuickAccess.noResults',
141
- defaultMessage: 'No results found'
142
- },
143
- // create-commands
144
- setResourceLanguage: {
145
- id: 'QuickAccess.setResourceLanguage',
146
- defaultMessage: 'Set Resource Language'
147
- },
148
- openDashboard: {
149
- id: 'QuickAccess.openDashboard',
150
- defaultMessage: 'Open Dashboard'
151
- },
152
- openProducts: {
153
- id: 'QuickAccess.openProducts',
154
- defaultMessage: 'Open Products'
155
- },
156
- openProductList: {
157
- id: 'QuickAccess.openProductList',
158
- defaultMessage: 'Open Product List'
159
- },
160
- openProductVariantGeneral: {
161
- id: 'QuickAccess.openProductVariantGeneral',
162
- defaultMessage: 'General'
163
- },
164
- openProductVariantList: {
165
- id: 'QuickAccess.openProductVariantList',
166
- defaultMessage: 'Variants'
167
- },
168
- openProductVariantSearch: {
169
- id: 'QuickAccess.openProductVariantSearch',
170
- defaultMessage: 'Int. / Ext. Search'
171
- },
172
- openModifiedProducts: {
173
- id: 'QuickAccess.openModifiedProducts',
174
- defaultMessage: 'Open Review Modified Products'
175
- },
176
- openPimSearch: {
177
- id: 'QuickAccess.openPimSearch',
178
- defaultMessage: 'Open PIM Search'
179
- },
180
- openAddProducts: {
181
- id: 'QuickAccess.openAddProducts',
182
- defaultMessage: 'Open Add Products'
183
- },
184
- openCategories: {
185
- id: 'QuickAccess.openCategories',
186
- defaultMessage: 'Open Categories'
187
- },
188
- openCategoriesList: {
189
- id: 'QuickAccess.openCategoriesList',
190
- defaultMessage: 'Open Categories List'
191
- },
192
- openCategoriesSearch: {
193
- id: 'QuickAccess.openCategoriesSearch',
194
- defaultMessage: 'Open Categories Search'
195
- },
196
- openAddCategory: {
197
- id: 'QuickAccess.openAddCategory',
198
- defaultMessage: 'Open Add Category'
199
- },
200
- openCustomers: {
201
- id: 'QuickAccess.openCustomers',
202
- defaultMessage: 'Open Customers'
203
- },
204
- openCustomersList: {
205
- id: 'QuickAccess.openCustomersList',
206
- defaultMessage: 'Open Customers List'
207
- },
208
- openAddCustomer: {
209
- id: 'QuickAccess.openAddCustomer',
210
- defaultMessage: 'Open Add Customer'
211
- },
212
- openCustomerGroupsList: {
213
- id: 'QuickAccess.openCustomerGroupsList',
214
- defaultMessage: 'Open Customer Groups List'
215
- },
216
- openAddCustomerGroup: {
217
- id: 'QuickAccess.openAddCustomerGroup',
218
- defaultMessage: 'Open Add Customer Group'
219
- },
220
- openOrders: {
221
- id: 'QuickAccess.openOrders',
222
- defaultMessage: 'Open Orders'
223
- },
224
- openOrdersList: {
225
- id: 'QuickAccess.openOrdersList',
226
- defaultMessage: 'Open Orders List'
227
- },
228
- openAddOrder: {
229
- id: 'QuickAccess.openAddOrder',
230
- defaultMessage: 'Open Add Order'
231
- },
232
- openDiscounts: {
233
- id: 'QuickAccess.openDiscounts',
234
- defaultMessage: 'Open Discounts'
235
- },
236
- openProductDiscountsList: {
237
- id: 'QuickAccess.openProductDiscountsList',
238
- defaultMessage: 'Open Product Discounts List'
239
- },
240
- openCartDiscountsList: {
241
- id: 'QuickAccess.openCartDiscountsList',
242
- defaultMessage: 'Open Cart Discounts List'
243
- },
244
- openDiscountCodesList: {
245
- id: 'QuickAccess.openDiscountCodesList',
246
- defaultMessage: 'Open Discount Codes List'
247
- },
248
- openAddDiscount: {
249
- id: 'QuickAccess.openAddDiscount',
250
- defaultMessage: 'Open Add Discount'
251
- },
252
- openAddProductDiscount: {
253
- id: 'QuickAccess.openAddProductDiscount',
254
- defaultMessage: 'Open Add Product Discount'
255
- },
256
- openAddCartDiscount: {
257
- id: 'QuickAccess.openAddCartDiscount',
258
- defaultMessage: 'Open Add Cart Discount'
259
- },
260
- openAddDiscountCode: {
261
- id: 'QuickAccess.openAddDiscountCode',
262
- defaultMessage: 'Open Add Discount Code'
263
- },
264
- openSettings: {
265
- id: 'QuickAccess.openSettings',
266
- defaultMessage: 'Open Settings'
267
- },
268
- openProjectSettings: {
269
- id: 'QuickAccess.openProjectSettings',
270
- defaultMessage: 'Open Project Settings'
271
- },
272
- openProjectSettingsInternationalTab: {
273
- id: 'QuickAccess.openProjectSettingsInternationalTab',
274
- defaultMessage: 'Open Project Settings • International'
275
- },
276
- openProjectSettingsTaxesTab: {
277
- id: 'QuickAccess.openProjectSettingsTaxesTab',
278
- defaultMessage: 'Open Project Settings • Taxes'
279
- },
280
- openProjectSettingsShippingMethodsTab: {
281
- id: 'QuickAccess.openProjectSettingsShippingMethodsTab',
282
- defaultMessage: 'Open Project Settings • Shipping Methods'
283
- },
284
- openProjectSettingsChannelsTab: {
285
- id: 'QuickAccess.openProjectSettingsChannelsTab',
286
- defaultMessage: 'Open Project Settings • Channels'
287
- },
288
- openProjectSettingsStoresTab: {
289
- id: 'QuickAccess.openProjectSettingsStoresTab',
290
- defaultMessage: 'Open Project Settings • Stores'
291
- },
292
- openProductTypesSettings: {
293
- id: 'QuickAccess.openProductTypesSettings',
294
- defaultMessage: 'Open Product Types Settings'
295
- },
296
- openDeveloperSettings: {
297
- id: 'QuickAccess.openDeveloperSettings',
298
- defaultMessage: 'Open Developer Settings'
299
- },
300
- openCustomApplicationsSettings: {
301
- id: 'QuickAccess.openCustomApplicationsSettings',
302
- defaultMessage: 'Open Custom Applications Settings'
303
- },
304
- openApiClientsList: {
305
- id: 'QuickAccess.openApiClientsList',
306
- defaultMessage: 'Open API Clients'
307
- },
308
- openAddApiClient: {
309
- id: 'QuickAccess.openAddApiClient',
310
- defaultMessage: 'Open Add API Client'
311
- },
312
- openSupport: {
313
- id: 'QuickAccess.openSupport',
314
- defaultMessage: 'Open Support'
315
- },
316
- openMyProfile: {
317
- id: 'QuickAccess.openMyProfile',
318
- defaultMessage: 'Open My Profile'
319
- },
320
- showPrivacyPolicy: {
321
- id: 'QuickAccess.showPrivacyPolicy',
322
- defaultMessage: 'Show Privacy Policy'
323
- },
324
- logout: {
325
- id: 'QuickAccess.logout',
326
- defaultMessage: 'Logout'
327
- },
328
- useProject: {
329
- id: 'QuickAccess.useProject',
330
- defaultMessage: 'Switch to project "{projectName}"'
331
- },
332
- openManageProjects: {
333
- id: 'QuickAccess.openManageProject',
334
- defaultMessage: 'Open Manage Projects'
335
- },
336
- openManageOrganizations: {
337
- id: 'QuickAccess.openManageOrganizations',
338
- defaultMessage: 'Open Manage Organizations'
339
- },
340
- // subcommands
341
- openVariantById: {
342
- id: 'QuickAccess.openVariantById',
343
- defaultMessage: 'Open Variant "{id}" (id)'
344
- },
345
- openVariantByKey: {
346
- id: 'QuickAccess.openVariantByKey',
347
- defaultMessage: 'Open Variant "{key}" (key)'
348
- },
349
- openVariantBySku: {
350
- id: 'QuickAccess.openVariantBySku',
351
- defaultMessage: 'Open Variant "{sku}" (sku)'
352
- },
353
- showProduct: {
354
- id: 'QuickAccess.showProduct',
355
- defaultMessage: 'Show Product "{productName}"'
356
- },
357
- showProductVariant: {
358
- id: 'QuickAccess.showProductVariant',
359
- defaultMessage: 'Show Product Variant "{variantName}"'
360
- },
361
- showProductVariantAttributes: {
362
- id: 'QuickAccess.showProductVariantAttributes',
363
- defaultMessage: 'Show Attributes'
364
- },
365
- showProductVariantImages: {
366
- id: 'QuickAccess.showProductVariantImages',
367
- defaultMessage: 'Show Images'
368
- },
369
- showProductVariantPrices: {
370
- id: 'QuickAccess.showProductVariantPrices',
371
- defaultMessage: 'Show Prices'
372
- },
373
- showProductVariantInventory: {
374
- id: 'QuickAccess.showProductVariantInventory',
375
- defaultMessage: 'Show Inventory'
376
- }
377
- });
378
-
379
- function ownKeys(e, r) { var t = _Object$keys(e); if (_Object$getOwnPropertySymbols) { var o = _Object$getOwnPropertySymbols(e); r && (o = _filterInstanceProperty(o).call(o, function (r) { return _Object$getOwnPropertyDescriptor(e, r).enumerable; })), t.push.apply(t, o); } return t; }
380
- function _objectSpread(e) { for (var r = 1; r < arguments.length; r++) { var _context10, _context11; var t = null != arguments[r] ? arguments[r] : {}; r % 2 ? _forEachInstanceProperty(_context10 = ownKeys(Object(t), !0)).call(_context10, function (r) { _defineProperty(e, r, t[r]); }) : _Object$getOwnPropertyDescriptors ? _Object$defineProperties(e, _Object$getOwnPropertyDescriptors(t)) : _forEachInstanceProperty(_context11 = ownKeys(Object(t))).call(_context11, function (r) { _Object$defineProperty(e, r, _Object$getOwnPropertyDescriptor(t, r)); }); } return e; }
381
- function _EMOTION_STRINGIFIED_CSS_ERROR__() { return "You have tried to stringify object returned from `css` function. It isn't supposed to be used directly (e.g. as value of the `className` prop), but rather handed to emotion so it can handle it (e.g. as value of `css` prop)."; }
382
- const isSelectAllCombo = event => event.key === 'a' && event.metaKey && !event.ctrlKey && !event.altKey && !event.shiftKey;
383
- const isCloseCombo = event => event.key === 'Escape' && !event.metaKey && !event.ctrlKey && !event.altKey && !event.shiftKey;
384
- const getPlatform = () => {
385
- var _context, _context2, _context3, _context4;
386
- if (_includesInstanceProperty(_context = navigator.appVersion).call(_context, 'Win')) return 'windows';
387
- if (_includesInstanceProperty(_context2 = navigator.appVersion).call(_context2, 'Mac')) return 'macos';
388
- if (_includesInstanceProperty(_context3 = navigator.appVersion).call(_context3, 'X11')) return 'unix';
389
- if (_includesInstanceProperty(_context4 = navigator.appVersion).call(_context4, 'Linux')) return 'linux';
390
- return null;
391
- };
392
- const hasNewWindowModifier = event => {
393
- const platform = getPlatform();
394
- switch (platform) {
395
- case 'macos':
396
- return event.metaKey;
397
- default:
398
- return event.ctrlKey;
399
- }
400
- };
401
- const shakeAnimation = keyframes`
402
- from,
403
- to {
404
- transform: translate3d(0, 0, 0);
405
- }
406
-
407
- 14%,
408
- 42%,
409
- 70% {
410
- transform: translate3d(-3px, 0, 0);
411
- }
412
-
413
- 28%,
414
- 56%,
415
- 84% {
416
- transform: translate3d(3px, 0, 0);
417
- }
418
- `;
419
- const initialState = {
420
- hasNetworkError: false,
421
- isLoading: false,
422
- searchText: '',
423
- selectedResult: -1,
424
- // Used for UX when browsing through history
425
- enableHistory: true,
426
- results: [],
427
- stack: []
428
- };
429
- const reducer = function () {
430
- var _context5;
431
- let state = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : initialState;
432
- let action = arguments.length > 1 ? arguments[1] : undefined;
433
- switch (action.type) {
434
- case 'networkError':
435
- return _objectSpread(_objectSpread({}, state), {}, {
436
- hasNetworkError: action.payload
437
- });
438
- case 'loading':
439
- return _objectSpread(_objectSpread({}, state), {}, {
440
- isLoading: action.payload
441
- });
442
- case 'selectedResult':
443
- return _objectSpread(_objectSpread({}, state), {}, {
444
- selectedResult: action.payload
445
- });
446
- case 'incrementSelectedResult':
447
- return _objectSpread(_objectSpread({}, state), {}, {
448
- selectedResult: state.selectedResult === state.results.length - 1 ? 0 : state.selectedResult + 1,
449
- enableHistory: false
450
- });
451
- case 'decrementSelectedResult':
452
- return _objectSpread(_objectSpread({}, state), {}, {
453
- selectedResult: state.selectedResult < 1 ? state.results.length - 1 : state.selectedResult - 1,
454
- enableHistory: false
455
- });
456
- case 'pickCommandFromHistory':
457
- return _objectSpread(_objectSpread({}, state), {}, {
458
- selectedResult: 0,
459
- searchText: action.payload.searchText,
460
- results: action.payload.results,
461
- stack: []
462
- // The history does not get changed here, it will be changed along
463
- // with the regular flow.
464
- });
465
- case 'setNextCommands':
466
- return _objectSpread(_objectSpread({}, state), {}, {
467
- stack: [...state.stack, {
468
- searchText: state.searchText,
469
- results: state.results,
470
- selectedResult: state.selectedResult
471
- }],
472
- selectedResult: 0,
473
- enableHistory: false,
474
- results: action.payload.results
475
- });
476
- case 'setPrevCommands':
477
- return _objectSpread(_objectSpread({}, state), {}, {
478
- searchText: action.payload.searchText,
479
- results: action.payload.results,
480
- selectedResult: 0,
481
- enableHistory: false,
482
- // omit last item
483
- stack: _sliceInstanceProperty(_context5 = state.stack).call(_context5, 0, -1)
484
- });
485
- case 'searchText':
486
- return _objectSpread(_objectSpread({}, state), {}, {
487
- searchText: action.payload,
488
- // clear network error when search text is cleared, so that users
489
- // are tempted to retry
490
- hasNetworkError: action.payload.length > 0 && state.hasNetworkError
491
- });
492
- case 'setSearchTextResults':
493
- return _objectSpread(_objectSpread({}, state), {}, {
494
- results: action.payload,
495
- selectedResult: action.payload.length > 0 ? 0 : -1,
496
- enableHistory: true,
497
- stack: []
498
- });
499
- case 'resetSearchText':
500
- return _objectSpread(_objectSpread({}, state), {}, {
501
- searchText: '',
502
- results: [],
503
- selectedResult: -1
504
- });
505
- case 'resetResultsWhenClosing':
506
- return _objectSpread(_objectSpread({}, state), {}, {
507
- selectedResult: -1,
508
- enableHistory: true
509
- });
510
- case 'reset':
511
- return initialState;
512
- default:
513
- return state;
514
- }
515
- };
516
- var _ref = process.env.NODE_ENV === "production" ? {
517
- name: "zjik7",
518
- styles: "display:flex"
519
- } : {
520
- name: "vnvgom-Butler",
521
- styles: "display:flex;label:Butler;/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["butler.tsx"],"names":[],"mappings":"AA8mBkB","file":"butler.tsx","sourcesContent":["import {\n  KeyboardEventHandler,\n  ChangeEventHandler,\n  KeyboardEvent,\n  MouseEventHandler,\n  MouseEvent,\n  useReducer,\n  useRef,\n  useCallback,\n} from 'react';\nimport { css, keyframes, ClassNames } from '@emotion/react';\nimport Fuse from 'fuse.js';\nimport last from 'lodash/last';\nimport { FormattedMessage, useIntl } from 'react-intl';\nimport { designTokens as uiKitDesignTokens } from '@commercetools-uikit/design-system';\nimport { SearchIcon } from '@commercetools-uikit/icons';\nimport LoadingSpinner from '@commercetools-uikit/loading-spinner';\nimport ButlerCommand from '../butler-command';\nimport ButlerContainer from '../butler-container';\nimport messages from '../messages';\nimport type {\n  Command,\n  SearchText,\n  SelectedResult,\n  Stack,\n  HistoryEntry,\n} from '../types';\n\nconst isSelectAllCombo = (event: KeyboardEvent<HTMLInputElement>) =>\n  event.key === 'a' &&\n  event.metaKey &&\n  !event.ctrlKey &&\n  !event.altKey &&\n  !event.shiftKey;\n\nconst isCloseCombo = (event: KeyboardEvent<HTMLInputElement>) =>\n  event.key === 'Escape' &&\n  !event.metaKey &&\n  !event.ctrlKey &&\n  !event.altKey &&\n  !event.shiftKey;\n\nconst getPlatform = () => {\n  if (navigator.appVersion.includes('Win')) return 'windows';\n  if (navigator.appVersion.includes('Mac')) return 'macos';\n  if (navigator.appVersion.includes('X11')) return 'unix';\n  if (navigator.appVersion.includes('Linux')) return 'linux';\n\n  return null;\n};\n\nconst hasNewWindowModifier = (\n  event: KeyboardEvent<HTMLElement> | MouseEvent<HTMLElement>\n) => {\n  const platform = getPlatform();\n  switch (platform) {\n    case 'macos':\n      return event.metaKey;\n    default:\n      return event.ctrlKey;\n  }\n};\n\nconst shakeAnimation = keyframes`\n  from,\n  to {\n    transform: translate3d(0, 0, 0);\n  }\n\n  14%,\n  42%,\n  70% {\n    transform: translate3d(-3px, 0, 0);\n  }\n\n  28%,\n  56%,\n  84% {\n    transform: translate3d(3px, 0, 0);\n  }\n`;\n\ntype State = {\n  hasNetworkError: boolean;\n  isLoading: boolean;\n  searchText: SearchText;\n  selectedResult: SelectedResult;\n  // Used for UX when browsing through history\n  enableHistory: boolean;\n  results: Command[];\n  stack: Stack[];\n};\ntype Action =\n  | { type: 'networkError'; payload: boolean }\n  | { type: 'loading'; payload: boolean }\n  | { type: 'selectedResult'; payload: number }\n  | { type: 'incrementSelectedResult' }\n  | { type: 'decrementSelectedResult' }\n  | {\n      type: 'pickCommandFromHistory';\n      payload: { searchText: SearchText; results: Command[] };\n    }\n  | { type: 'setNextCommands'; payload: { results: Command[] } }\n  | {\n      type: 'setPrevCommands';\n      payload: { searchText: SearchText; results: Command[] };\n    }\n  | { type: 'searchText'; payload: string }\n  | { type: 'setSearchTextResults'; payload: Command[] }\n  | { type: 'resetSearchText' }\n  | { type: 'resetResultsWhenClosing' }\n  | { type: 'reset' };\nconst initialState = {\n  hasNetworkError: false,\n  isLoading: false,\n  searchText: '',\n  selectedResult: -1,\n  // Used for UX when browsing through history\n  enableHistory: true,\n  results: [],\n  stack: [],\n};\nconst reducer = (state: State = initialState, action: Action): State => {\n  switch (action.type) {\n    case 'networkError':\n      return { ...state, hasNetworkError: action.payload };\n    case 'loading':\n      return { ...state, isLoading: action.payload };\n    case 'selectedResult':\n      return { ...state, selectedResult: action.payload };\n    case 'incrementSelectedResult':\n      return {\n        ...state,\n        selectedResult:\n          state.selectedResult === state.results.length - 1\n            ? 0\n            : state.selectedResult + 1,\n        enableHistory: false,\n      };\n    case 'decrementSelectedResult':\n      return {\n        ...state,\n        selectedResult:\n          state.selectedResult < 1\n            ? state.results.length - 1\n            : state.selectedResult - 1,\n        enableHistory: false,\n      };\n    case 'pickCommandFromHistory':\n      return {\n        ...state,\n        selectedResult: 0,\n        searchText: action.payload.searchText,\n        results: action.payload.results,\n        stack: [],\n        // The history does not get changed here, it will be changed along\n        // with the regular flow.\n      };\n    case 'setNextCommands':\n      return {\n        ...state,\n        stack: [\n          ...state.stack,\n          {\n            searchText: state.searchText,\n            results: state.results,\n            selectedResult: state.selectedResult,\n          },\n        ],\n        selectedResult: 0,\n        enableHistory: false,\n        results: action.payload.results,\n      };\n    case 'setPrevCommands':\n      return {\n        ...state,\n        searchText: action.payload.searchText,\n        results: action.payload.results,\n        selectedResult: 0,\n        enableHistory: false,\n        // omit last item\n        stack: state.stack.slice(0, -1),\n      };\n    case 'searchText':\n      return {\n        ...state,\n        searchText: action.payload,\n        // clear network error when search text is cleared, so that users\n        // are tempted to retry\n        hasNetworkError: action.payload.length > 0 && state.hasNetworkError,\n      };\n    case 'setSearchTextResults':\n      return {\n        ...state,\n        results: action.payload,\n        selectedResult: action.payload.length > 0 ? 0 : -1,\n        enableHistory: true,\n        stack: [],\n      };\n    case 'resetSearchText':\n      return { ...state, searchText: '', results: [], selectedResult: -1 };\n    case 'resetResultsWhenClosing':\n      return { ...state, selectedResult: -1, enableHistory: true };\n    case 'reset':\n      return initialState;\n    default:\n      return state;\n  }\n};\n\ntype Props = {\n  historyEntries: HistoryEntry[];\n  onHistoryEntriesChange: (historyEntries: HistoryEntry[]) => void;\n  search: (searchText: SearchText) => Promise<Command[]>;\n  getNextCommands: (command: Command) => Promise<Command[]>;\n  executeCommand: (command: Command, meta: { openInNewTab: boolean }) => void;\n  onClose: () => void;\n  classNameShakeAnimation: string;\n};\nconst Butler = (props: Props) => {\n  const intl = useIntl();\n  const [state, dispatch] = useReducer(reducer, initialState);\n\n  const shouldSelectFieldText = useRef(false);\n  const isNewWindowCombo = useRef(false);\n  const skipNextSelection = useRef(false);\n  const searchContainerRef = useRef<HTMLDivElement>(null);\n  const searchInputRef = useRef<HTMLInputElement>(null);\n\n  const setHasNetworkError = useCallback(() => {\n    dispatch({ type: 'networkError', payload: true });\n  }, []);\n  const unsetHasNetworkError = useCallback(() => {\n    dispatch({ type: 'networkError', payload: false });\n  }, []);\n  const setIsLoading = useCallback(() => {\n    dispatch({ type: 'loading', payload: true });\n  }, []);\n  const unsetIsLoading = useCallback(() => {\n    dispatch({ type: 'loading', payload: false });\n  }, []);\n\n  // Destructure functions from props to reference them in the hook dependency list\n  const {\n    search: searchFromParent,\n    onClose: onCloseFromParent,\n    executeCommand: executeCommandFromParent,\n    onHistoryEntriesChange: onHistoryEntriesChangeFromParent,\n    getNextCommands: getNextCommandsFromParent,\n  } = props;\n\n  const shake = useCallback(() => {\n    if (searchContainerRef.current) {\n      searchContainerRef.current.classList.remove(\n        props.classNameShakeAnimation\n      );\n      // -> triggering reflow\n      // eslint-disable-next-line no-void\n      void searchContainerRef.current.offsetWidth;\n      searchContainerRef.current.classList.add(props.classNameShakeAnimation);\n    }\n  }, [props.classNameShakeAnimation]);\n\n  const execute = useCallback(\n    (command: Command, meta: { openInNewTab: boolean }) => {\n      // Only main entries get added to history, so when a subcommand is executed,\n      // we add the main command of it to the history (the top-level command).\n      //\n      // The key to identify history entries by is always the searchText\n      // There will never be two history entries with the same searchText\n      const entry =\n        state.stack.length === 0\n          ? // The stack is empty, so we are executing a top-level command\n            { searchText: state.searchText, results: state.results }\n          : // We are executing a subcommand, so we get the top-level command for it,\n            // which is at the bottom of the stack.\n            {\n              searchText: state.stack[0].searchText,\n              results: state.stack[0].results,\n            };\n\n      // Add the entry to the history, while excluding any earlier history entry\n      // with the same search text. This effectively \"moves\" that entry to the\n      // top of the history (with the most recent results), or appends a new entry\n      // when it didn't exist before.\n      onHistoryEntriesChangeFromParent([\n        ...props.historyEntries.filter(\n          (command) => command.searchText !== entry.searchText\n        ),\n        entry,\n      ]);\n\n      dispatch({ type: 'resetSearchText' });\n\n      onCloseFromParent();\n\n      executeCommandFromParent(command, meta);\n    },\n    [\n      executeCommandFromParent,\n      onCloseFromParent,\n      onHistoryEntriesChangeFromParent,\n      props.historyEntries,\n      state.results,\n      state.searchText,\n      state.stack,\n    ]\n  );\n  const handleKeyDown = useCallback<KeyboardEventHandler<HTMLInputElement>>(\n    (event) => {\n      // Preventing cursor jumps can only happen in onKeyDown, but not in onKeyUp\n      event.persist();\n\n      // We want to know when the user presses cmd+enter (cmd being a meta key).\n      // We are only told about this in keyDown, but not in keyUp, so we need\n      // to handle it here\n      if (event.key === 'Enter' && hasNewWindowModifier(event)) {\n        isNewWindowCombo.current = true;\n        return;\n      }\n\n      // Avoid selecting the whole page when user selectes everything with\n      // a keyboard shortcut. There is probably a better way to do this though.\n      // This prevents the whole page from being selected in case the user\n      // 1) opens the search box\n      // 2) types into it\n      // 3) selects all text using cmd+a\n      // 4) closes the search box with esc\n      // Without this handling, the whole page would now be selected\n      if (isSelectAllCombo(event)) {\n        // This stops the browser from selecting anything\n        event.preventDefault();\n        if (searchInputRef.current) {\n          // This selects the text in the search input\n          searchInputRef.current.setSelectionRange(0, state.searchText.length);\n        }\n        return;\n      }\n\n      // avoid interfering with other key combinations using modifier keys\n      if (event.ctrlKey || event.altKey || event.shiftKey || event.metaKey)\n        return;\n      if (isCloseCombo(event)) return;\n\n      // skip next mouseEnter to avoid setting selectedResult when cursor just\n      // happens to be where the results will pop up\n      skipNextSelection.current = true;\n\n      if (event.key === 'ArrowDown') {\n        // prevent cursor from jumping to end of text input\n        event.preventDefault();\n        dispatch({ type: 'incrementSelectedResult' });\n        return;\n      }\n      if (event.key === 'ArrowUp') {\n        // browse through history\n        if (\n          state.searchText.length === 0 ||\n          (state.selectedResult < 1 && state.enableHistory)\n        ) {\n          shouldSelectFieldText.current = true;\n          const selectedIndex =\n            state.searchText.length === 0\n              ? // When going back the first step\n                -1\n              : // When going back more than one step\n                props.historyEntries.findIndex(\n                  (command) => command.searchText === state.searchText\n                );\n          // Pick the previous command from the history\n          const prevCommand =\n            selectedIndex === -1\n              ? // previous command on top of the history when going back on\n                // first step\n                last(props.historyEntries)\n              : // previous command is deeper down\n                // When the history does not exist (negative index), then\n                // this implicitly returns undefined\n                props.historyEntries[selectedIndex - 1];\n          // Skip when no previous entry exists in the history\n          if (!prevCommand) return;\n          dispatch({\n            type: 'pickCommandFromHistory',\n            payload: {\n              searchText: prevCommand.searchText,\n              results: prevCommand.results,\n            },\n          });\n          return;\n        }\n        // prevent cursor from jumping to beginning of text input\n        event.preventDefault();\n        dispatch({ type: 'decrementSelectedResult' });\n        return;\n      }\n      if (state.selectedResult > -1) {\n        if (event.key === 'ArrowRight') {\n          const command = state.results[state.selectedResult];\n          const searchText = state.searchText;\n          const isCursorAtEnd =\n            searchInputRef.current &&\n            state.searchText.length === searchInputRef.current.selectionStart;\n\n          const isEverythingSelected =\n            searchInputRef.current &&\n            searchInputRef.current.selectionStart === 0 &&\n            state.searchText.length === searchInputRef.current.selectionEnd;\n\n          // only allow diving in when cursor is at end of input or when\n          // the complete text is selected (when browsing through history)\n          if (!isCursorAtEnd && !isEverythingSelected) return;\n\n          unsetHasNetworkError();\n\n          // NOTE: since we need to fetch the \"next command\", which is an async operation,\n          // we use a IIFE to process that and eventually update the state.\n          (async () => {\n            if (command) {\n              const nextCommands = await getNextCommandsFromParent(command);\n              // avoid moving cursor when there are sub-options\n              if (nextCommands.length > 0) {\n                // Ensure the search text has not changed while we were loading\n                // the next results, otherwise we'd interrupt the user.\n                // Throw away the results in case the search text has changed.\n                if (state.searchText === searchText) {\n                  dispatch({\n                    type: 'setNextCommands',\n                    payload: { results: nextCommands },\n                  });\n                }\n                return;\n              }\n            }\n            shake();\n          })();\n          return;\n        }\n        if (event.key === 'ArrowLeft') {\n          // go left in stack\n          const prevCommand = last(state.stack);\n\n          // do nothing when we can't go left anymore\n          if (!prevCommand) return;\n\n          // prevent cursor from jumping a char to the left in text input\n          event.preventDefault();\n\n          dispatch({\n            type: 'setPrevCommands',\n            payload: {\n              searchText: prevCommand.searchText,\n              results: prevCommand.results,\n            },\n          });\n          return;\n        }\n      }\n    },\n    [\n      getNextCommandsFromParent,\n      props.historyEntries,\n      shake,\n      state,\n      unsetHasNetworkError,\n    ]\n  );\n  const handleKeyUp = useCallback<KeyboardEventHandler<HTMLInputElement>>(\n    (event) => {\n      // setting the selection can only happen in onKeyUp\n      if (shouldSelectFieldText.current) {\n        const input = event.target as HTMLInputElement;\n        input.focus();\n        input.select();\n        shouldSelectFieldText.current = false;\n      }\n\n      if (event.key !== 'Enter' && !isNewWindowCombo.current) return true;\n\n      // User just triggered the search\n      if (state.selectedResult === -1) return true;\n\n      // User had something selected and wants to go there\n      execute(state.results[state.selectedResult], {\n        openInNewTab: isNewWindowCombo.current,\n      });\n\n      isNewWindowCombo.current = false;\n      return true;\n    },\n    [execute, state.results, state.selectedResult]\n  );\n  const handleChange = useCallback<ChangeEventHandler<HTMLInputElement>>(\n    (event) => {\n      const searchText = event.target.value;\n      if (searchText.trim().length === 0) {\n        dispatch({ type: 'reset' });\n        return;\n      }\n\n      dispatch({ type: 'searchText', payload: searchText });\n\n      // A search via network is only triggered when there\n      // are more than three characters. So no false loading\n      // indication is given.\n      if (searchText.trim().length > 3) {\n        setIsLoading();\n      }\n\n      searchFromParent(searchText).then(\n        (asyncResults: Command[]) => {\n          unsetHasNetworkError();\n          unsetIsLoading();\n\n          const fuse = new Fuse(asyncResults, {\n            keys: [\n              { name: 'text', weight: 0.6 },\n              { name: 'keywords', weight: 0.4 },\n            ],\n            minMatchCharLength: 2,\n            includeScore: true,\n          });\n\n          const searchResults = fuse\n            .search(searchText)\n            // Filter out results with a matching score over 0.75\n            .filter((result) => (result.score ? result.score < 0.75 : false))\n            // Keep a maximal of 9 results\n            .slice(0, 9);\n\n          dispatch({\n            type: 'setSearchTextResults',\n            payload: searchResults.map((result) => result.item),\n          });\n        },\n        (error: Error) => {\n          // eslint-disable-next-line no-console\n          if (process.env.NODE_ENV !== 'production') console.error(error);\n          unsetIsLoading();\n          setHasNetworkError();\n        }\n      );\n    },\n    [\n      searchFromParent,\n      setHasNetworkError,\n      setIsLoading,\n      unsetHasNetworkError,\n      unsetIsLoading,\n    ]\n  );\n  const handleContainerClick = useCallback(() => {\n    dispatch({ type: 'resetResultsWhenClosing' });\n    onCloseFromParent();\n  }, [onCloseFromParent]);\n\n  const createCommandMouseEnterHandler = useCallback<\n    (index: number) => MouseEventHandler<HTMLDivElement>\n  >(\n    (index) => () => {\n      // In case the cursor happened to be in a location where a\n      // result would appear, it would trigger onMouseEnter and the\n      // result would be selected immediately. This is not something\n      // a user would expect, hence we prevent it from happening.\n      // The user has to move the cursor to an option explicitly for\n      // it to become active. However, the user can always click and\n      // that action will be triggered.\n      if (skipNextSelection.current) {\n        skipNextSelection.current = false;\n        return;\n      }\n\n      // sets the selected result, mainly for the hover effect\n      dispatch({ type: 'selectedResult', payload: index });\n    },\n    []\n  );\n  const createCommandClickHandler = useCallback<\n    (command: Command) => MouseEventHandler<HTMLDivElement>\n  >(\n    (command) => (event) => {\n      execute(command, {\n        openInNewTab: hasNewWindowModifier(event),\n      });\n    },\n    [execute]\n  );\n\n  return (\n    <ButlerContainer\n      onClick={handleContainerClick}\n      data-testid=\"quick-access\"\n      tabIndex={-1}\n    >\n      <div\n        ref={searchContainerRef}\n        css={css`\n          background-color: ${uiKitDesignTokens.colorSurface};\n          border: 0;\n          border-radius: ${uiKitDesignTokens.borderRadius4};\n          min-height: 40px;\n\n          /* one more than app-bar (20000) and one more than the overlay (20001) */\n          z-index: 20002;\n          width: 400px;\n          margin: 40px auto;\n          overflow: hidden;\n          -webkit-box-shadow: 0 10px 30px -8px rgba(0, 0, 0, 0.75);\n          -moz-box-shadow: 0 10px 30px -8px rgba(0, 0, 0, 0.75);\n          box-shadow: 0 10px 30px -8px rgba(0, 0, 0, 0.75);\n          padding-bottom: ${state.hasNetworkError\n            ? '0'\n            : uiKitDesignTokens.spacingS};\n        `}\n        onClick={(event) => {\n          // Avoid closing when the searchContainer itself is clicked\n          // If we don't do this, then the overlay will close when e.g.\n          // the search input is clicked.\n          event.stopPropagation();\n          event.preventDefault();\n        }}\n      >\n        <div\n          css={css`\n            display: flex;\n          `}\n        >\n          <label\n            htmlFor=\"quick-access-search-input\"\n            css={css`\n              align-self: center;\n              padding-left: ${uiKitDesignTokens.spacingM};\n              margin-top: ${uiKitDesignTokens.spacingS};\n            `}\n          >\n            <SearchIcon color=\"neutral60\" />\n          </label>\n          <input\n            id=\"quick-access-search-input\"\n            ref={searchInputRef}\n            placeholder={intl.formatMessage(messages.inputPlacehoder)}\n            type=\"text\"\n            css={css`\n              width: 100%;\n              border: 0;\n              outline: 0;\n              font-size: 22px;\n              font-weight: 300;\n              padding: ${uiKitDesignTokens.spacingM}\n                ${uiKitDesignTokens.spacingM} ${uiKitDesignTokens.spacingS}\n                ${uiKitDesignTokens.spacingS};\n              &::placeholder {\n                color: ${uiKitDesignTokens.colorNeutral60};\n              }\n            `}\n            value={state.searchText}\n            onChange={handleChange}\n            onKeyDown={handleKeyDown}\n            onKeyUp={handleKeyUp}\n            autoFocus={true}\n            autoComplete=\"off\"\n            data-testid=\"quick-access-search-input\"\n          />\n          {state.isLoading && (\n            <div\n              css={css`\n                align-self: center;\n                margin-top: ${uiKitDesignTokens.spacingS};\n                margin-right: ${uiKitDesignTokens.spacingS};\n              `}\n            >\n              <LoadingSpinner />\n            </div>\n          )}\n        </div>\n        {(() => {\n          if (state.hasNetworkError)\n            return (\n              <div\n                css={css`\n                  overflow: hidden;\n                  white-space: nowrap;\n                  cursor: default;\n                  background: ${uiKitDesignTokens.colorError};\n                  text-align: center;\n                  text-transform: uppercase;\n                  color: ${uiKitDesignTokens.colorSurface};\n                  font-size: ${uiKitDesignTokens.fontSize20};\n                  padding: ${uiKitDesignTokens.spacingXs};\n                `}\n              >\n                <FormattedMessage {...messages.offline} />\n              </div>\n            );\n\n          if (state.results.length === 0 && state.searchText.trim().length > 0)\n            return (\n              <div\n                css={css`\n                  overflow: hidden;\n                  white-space: nowrap;\n                  cursor: default;\n                  background: ${uiKitDesignTokens.colorNeutral};\n                  color: ${uiKitDesignTokens.colorSolid};\n                  text-align: center;\n                  text-transform: uppercase;\n                  font-size: ${uiKitDesignTokens.fontSize20};\n                  padding: ${uiKitDesignTokens.spacingXs};\n                `}\n              >\n                <FormattedMessage {...messages.noResults} />\n              </div>\n            );\n\n          return state.results.map((command, index) => (\n            <ButlerCommand\n              key={command.id}\n              command={command}\n              isSelected={state.selectedResult === index}\n              onMouseEnter={createCommandMouseEnterHandler(index)}\n              onClick={createCommandClickHandler(command)}\n            />\n          ));\n        })()}\n      </div>\n    </ButlerContainer>\n  );\n};\nButler.displayName = 'Butler';\n\nconst ButlerWithAnimation = (props: Omit<Props, 'classNameShakeAnimation'>) => (\n  <ClassNames>\n    {({ css }) => (\n      <Butler\n        {...props}\n        classNameShakeAnimation={css`\n          animation-duration: 0.45s;\n          animation-fill-mode: both;\n          animation-name: ${shakeAnimation};\n        `}\n      />\n    )}\n  </ClassNames>\n);\n\nexport default ButlerWithAnimation;\n"]} */",
522
- toString: _EMOTION_STRINGIFIED_CSS_ERROR__
523
- };
524
- const Butler = props => {
525
- const intl = useIntl();
526
- const _useReducer = useReducer(reducer, initialState),
527
- _useReducer2 = _slicedToArray(_useReducer, 2),
528
- state = _useReducer2[0],
529
- dispatch = _useReducer2[1];
530
- const shouldSelectFieldText = useRef(false);
531
- const isNewWindowCombo = useRef(false);
532
- const skipNextSelection = useRef(false);
533
- const searchContainerRef = useRef(null);
534
- const searchInputRef = useRef(null);
535
- const setHasNetworkError = useCallback(() => {
536
- dispatch({
537
- type: 'networkError',
538
- payload: true
539
- });
540
- }, []);
541
- const unsetHasNetworkError = useCallback(() => {
542
- dispatch({
543
- type: 'networkError',
544
- payload: false
545
- });
546
- }, []);
547
- const setIsLoading = useCallback(() => {
548
- dispatch({
549
- type: 'loading',
550
- payload: true
551
- });
552
- }, []);
553
- const unsetIsLoading = useCallback(() => {
554
- dispatch({
555
- type: 'loading',
556
- payload: false
557
- });
558
- }, []);
559
-
560
- // Destructure functions from props to reference them in the hook dependency list
561
- const searchFromParent = props.search,
562
- onCloseFromParent = props.onClose,
563
- executeCommandFromParent = props.executeCommand,
564
- onHistoryEntriesChangeFromParent = props.onHistoryEntriesChange,
565
- getNextCommandsFromParent = props.getNextCommands;
566
- const shake = useCallback(() => {
567
- if (searchContainerRef.current) {
568
- searchContainerRef.current.classList.remove(props.classNameShakeAnimation);
569
- // -> triggering reflow
570
- // eslint-disable-next-line no-void
571
- void searchContainerRef.current.offsetWidth;
572
- searchContainerRef.current.classList.add(props.classNameShakeAnimation);
573
- }
574
- }, [props.classNameShakeAnimation]);
575
- const execute = useCallback((command, meta) => {
576
- var _context6;
577
- // Only main entries get added to history, so when a subcommand is executed,
578
- // we add the main command of it to the history (the top-level command).
579
- //
580
- // The key to identify history entries by is always the searchText
581
- // There will never be two history entries with the same searchText
582
- const entry = state.stack.length === 0 ?
583
- // The stack is empty, so we are executing a top-level command
584
- {
585
- searchText: state.searchText,
586
- results: state.results
587
- } :
588
- // We are executing a subcommand, so we get the top-level command for it,
589
- // which is at the bottom of the stack.
590
- {
591
- searchText: state.stack[0].searchText,
592
- results: state.stack[0].results
593
- };
594
-
595
- // Add the entry to the history, while excluding any earlier history entry
596
- // with the same search text. This effectively "moves" that entry to the
597
- // top of the history (with the most recent results), or appends a new entry
598
- // when it didn't exist before.
599
- onHistoryEntriesChangeFromParent([..._filterInstanceProperty(_context6 = props.historyEntries).call(_context6, command => command.searchText !== entry.searchText), entry]);
600
- dispatch({
601
- type: 'resetSearchText'
602
- });
603
- onCloseFromParent();
604
- executeCommandFromParent(command, meta);
605
- }, [executeCommandFromParent, onCloseFromParent, onHistoryEntriesChangeFromParent, props.historyEntries, state.results, state.searchText, state.stack]);
606
- const handleKeyDown = useCallback(event => {
607
- // Preventing cursor jumps can only happen in onKeyDown, but not in onKeyUp
608
- event.persist();
609
-
610
- // We want to know when the user presses cmd+enter (cmd being a meta key).
611
- // We are only told about this in keyDown, but not in keyUp, so we need
612
- // to handle it here
613
- if (event.key === 'Enter' && hasNewWindowModifier(event)) {
614
- isNewWindowCombo.current = true;
615
- return;
616
- }
617
-
618
- // Avoid selecting the whole page when user selectes everything with
619
- // a keyboard shortcut. There is probably a better way to do this though.
620
- // This prevents the whole page from being selected in case the user
621
- // 1) opens the search box
622
- // 2) types into it
623
- // 3) selects all text using cmd+a
624
- // 4) closes the search box with esc
625
- // Without this handling, the whole page would now be selected
626
- if (isSelectAllCombo(event)) {
627
- // This stops the browser from selecting anything
628
- event.preventDefault();
629
- if (searchInputRef.current) {
630
- // This selects the text in the search input
631
- searchInputRef.current.setSelectionRange(0, state.searchText.length);
632
- }
633
- return;
634
- }
635
-
636
- // avoid interfering with other key combinations using modifier keys
637
- if (event.ctrlKey || event.altKey || event.shiftKey || event.metaKey) return;
638
- if (isCloseCombo(event)) return;
639
-
640
- // skip next mouseEnter to avoid setting selectedResult when cursor just
641
- // happens to be where the results will pop up
642
- skipNextSelection.current = true;
643
- if (event.key === 'ArrowDown') {
644
- // prevent cursor from jumping to end of text input
645
- event.preventDefault();
646
- dispatch({
647
- type: 'incrementSelectedResult'
648
- });
649
- return;
650
- }
651
- if (event.key === 'ArrowUp') {
652
- // browse through history
653
- if (state.searchText.length === 0 || state.selectedResult < 1 && state.enableHistory) {
654
- var _context7;
655
- shouldSelectFieldText.current = true;
656
- const selectedIndex = state.searchText.length === 0 ?
657
- // When going back the first step
658
- -1 :
659
- // When going back more than one step
660
- _findIndexInstanceProperty(_context7 = props.historyEntries).call(_context7, command => command.searchText === state.searchText);
661
- // Pick the previous command from the history
662
- const prevCommand = selectedIndex === -1 ?
663
- // previous command on top of the history when going back on
664
- // first step
665
- last(props.historyEntries) :
666
- // previous command is deeper down
667
- // When the history does not exist (negative index), then
668
- // this implicitly returns undefined
669
- props.historyEntries[selectedIndex - 1];
670
- // Skip when no previous entry exists in the history
671
- if (!prevCommand) return;
672
- dispatch({
673
- type: 'pickCommandFromHistory',
674
- payload: {
675
- searchText: prevCommand.searchText,
676
- results: prevCommand.results
677
- }
678
- });
679
- return;
680
- }
681
- // prevent cursor from jumping to beginning of text input
682
- event.preventDefault();
683
- dispatch({
684
- type: 'decrementSelectedResult'
685
- });
686
- return;
687
- }
688
- if (state.selectedResult > -1) {
689
- if (event.key === 'ArrowRight') {
690
- const command = state.results[state.selectedResult];
691
- const searchText = state.searchText;
692
- const isCursorAtEnd = searchInputRef.current && state.searchText.length === searchInputRef.current.selectionStart;
693
- const isEverythingSelected = searchInputRef.current && searchInputRef.current.selectionStart === 0 && state.searchText.length === searchInputRef.current.selectionEnd;
694
-
695
- // only allow diving in when cursor is at end of input or when
696
- // the complete text is selected (when browsing through history)
697
- if (!isCursorAtEnd && !isEverythingSelected) return;
698
- unsetHasNetworkError();
699
-
700
- // NOTE: since we need to fetch the "next command", which is an async operation,
701
- // we use a IIFE to process that and eventually update the state.
702
- (async () => {
703
- if (command) {
704
- const nextCommands = await getNextCommandsFromParent(command);
705
- // avoid moving cursor when there are sub-options
706
- if (nextCommands.length > 0) {
707
- // Ensure the search text has not changed while we were loading
708
- // the next results, otherwise we'd interrupt the user.
709
- // Throw away the results in case the search text has changed.
710
- if (state.searchText === searchText) {
711
- dispatch({
712
- type: 'setNextCommands',
713
- payload: {
714
- results: nextCommands
715
- }
716
- });
717
- }
718
- return;
719
- }
720
- }
721
- shake();
722
- })();
723
- return;
724
- }
725
- if (event.key === 'ArrowLeft') {
726
- // go left in stack
727
- const prevCommand = last(state.stack);
728
-
729
- // do nothing when we can't go left anymore
730
- if (!prevCommand) return;
731
-
732
- // prevent cursor from jumping a char to the left in text input
733
- event.preventDefault();
734
- dispatch({
735
- type: 'setPrevCommands',
736
- payload: {
737
- searchText: prevCommand.searchText,
738
- results: prevCommand.results
739
- }
740
- });
741
- return;
742
- }
743
- }
744
- }, [getNextCommandsFromParent, props.historyEntries, shake, state, unsetHasNetworkError]);
745
- const handleKeyUp = useCallback(event => {
746
- // setting the selection can only happen in onKeyUp
747
- if (shouldSelectFieldText.current) {
748
- const input = event.target;
749
- input.focus();
750
- input.select();
751
- shouldSelectFieldText.current = false;
752
- }
753
- if (event.key !== 'Enter' && !isNewWindowCombo.current) return true;
754
-
755
- // User just triggered the search
756
- if (state.selectedResult === -1) return true;
757
-
758
- // User had something selected and wants to go there
759
- execute(state.results[state.selectedResult], {
760
- openInNewTab: isNewWindowCombo.current
761
- });
762
- isNewWindowCombo.current = false;
763
- return true;
764
- }, [execute, state.results, state.selectedResult]);
765
- const handleChange = useCallback(event => {
766
- const searchText = event.target.value;
767
- if (_trimInstanceProperty(searchText).call(searchText).length === 0) {
768
- dispatch({
769
- type: 'reset'
770
- });
771
- return;
772
- }
773
- dispatch({
774
- type: 'searchText',
775
- payload: searchText
776
- });
777
-
778
- // A search via network is only triggered when there
779
- // are more than three characters. So no false loading
780
- // indication is given.
781
- if (_trimInstanceProperty(searchText).call(searchText).length > 3) {
782
- setIsLoading();
783
- }
784
- searchFromParent(searchText).then(asyncResults => {
785
- var _context8, _context9;
786
- unsetHasNetworkError();
787
- unsetIsLoading();
788
- const fuse = new Fuse(asyncResults, {
789
- keys: [{
790
- name: 'text',
791
- weight: 0.6
792
- }, {
793
- name: 'keywords',
794
- weight: 0.4
795
- }],
796
- minMatchCharLength: 2,
797
- includeScore: true
798
- });
799
- const searchResults = _sliceInstanceProperty(_context8 = _filterInstanceProperty(_context9 = fuse.search(searchText)
800
- // Filter out results with a matching score over 0.75
801
- ).call(_context9, result => result.score ? result.score < 0.75 : false)
802
- // Keep a maximal of 9 results
803
- ).call(_context8, 0, 9);
804
- dispatch({
805
- type: 'setSearchTextResults',
806
- payload: _mapInstanceProperty(searchResults).call(searchResults, result => result.item)
807
- });
808
- }, error => {
809
- // eslint-disable-next-line no-console
810
- if (process.env.NODE_ENV !== 'production') console.error(error);
811
- unsetIsLoading();
812
- setHasNetworkError();
813
- });
814
- }, [searchFromParent, setHasNetworkError, setIsLoading, unsetHasNetworkError, unsetIsLoading]);
815
- const handleContainerClick = useCallback(() => {
816
- dispatch({
817
- type: 'resetResultsWhenClosing'
818
- });
819
- onCloseFromParent();
820
- }, [onCloseFromParent]);
821
- const createCommandMouseEnterHandler = useCallback(index => () => {
822
- // In case the cursor happened to be in a location where a
823
- // result would appear, it would trigger onMouseEnter and the
824
- // result would be selected immediately. This is not something
825
- // a user would expect, hence we prevent it from happening.
826
- // The user has to move the cursor to an option explicitly for
827
- // it to become active. However, the user can always click and
828
- // that action will be triggered.
829
- if (skipNextSelection.current) {
830
- skipNextSelection.current = false;
831
- return;
832
- }
833
-
834
- // sets the selected result, mainly for the hover effect
835
- dispatch({
836
- type: 'selectedResult',
837
- payload: index
838
- });
839
- }, []);
840
- const createCommandClickHandler = useCallback(command => event => {
841
- execute(command, {
842
- openInNewTab: hasNewWindowModifier(event)
843
- });
844
- }, [execute]);
845
- return jsx(ButlerContainer, {
846
- onClick: handleContainerClick,
847
- "data-testid": "quick-access",
848
- tabIndex: -1,
849
- children: jsxs("div", {
850
- ref: searchContainerRef,
851
- css: /*#__PURE__*/css("background-color:", designTokens.colorSurface, ";border:0;border-radius:", designTokens.borderRadius4, ";min-height:40px;z-index:20002;width:400px;margin:40px auto;overflow:hidden;-webkit-box-shadow:0 10px 30px -8px rgba(0, 0, 0, 0.75);-moz-box-shadow:0 10px 30px -8px rgba(0, 0, 0, 0.75);box-shadow:0 10px 30px -8px rgba(0, 0, 0, 0.75);padding-bottom:", state.hasNetworkError ? '0' : designTokens.spacingS, ";" + (process.env.NODE_ENV === "production" ? "" : ";label:Butler;"), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["butler.tsx"],"names":[],"mappings":"AAmlBgB","file":"butler.tsx","sourcesContent":["import {\n  KeyboardEventHandler,\n  ChangeEventHandler,\n  KeyboardEvent,\n  MouseEventHandler,\n  MouseEvent,\n  useReducer,\n  useRef,\n  useCallback,\n} from 'react';\nimport { css, keyframes, ClassNames } from '@emotion/react';\nimport Fuse from 'fuse.js';\nimport last from 'lodash/last';\nimport { FormattedMessage, useIntl } from 'react-intl';\nimport { designTokens as uiKitDesignTokens } from '@commercetools-uikit/design-system';\nimport { SearchIcon } from '@commercetools-uikit/icons';\nimport LoadingSpinner from '@commercetools-uikit/loading-spinner';\nimport ButlerCommand from '../butler-command';\nimport ButlerContainer from '../butler-container';\nimport messages from '../messages';\nimport type {\n  Command,\n  SearchText,\n  SelectedResult,\n  Stack,\n  HistoryEntry,\n} from '../types';\n\nconst isSelectAllCombo = (event: KeyboardEvent<HTMLInputElement>) =>\n  event.key === 'a' &&\n  event.metaKey &&\n  !event.ctrlKey &&\n  !event.altKey &&\n  !event.shiftKey;\n\nconst isCloseCombo = (event: KeyboardEvent<HTMLInputElement>) =>\n  event.key === 'Escape' &&\n  !event.metaKey &&\n  !event.ctrlKey &&\n  !event.altKey &&\n  !event.shiftKey;\n\nconst getPlatform = () => {\n  if (navigator.appVersion.includes('Win')) return 'windows';\n  if (navigator.appVersion.includes('Mac')) return 'macos';\n  if (navigator.appVersion.includes('X11')) return 'unix';\n  if (navigator.appVersion.includes('Linux')) return 'linux';\n\n  return null;\n};\n\nconst hasNewWindowModifier = (\n  event: KeyboardEvent<HTMLElement> | MouseEvent<HTMLElement>\n) => {\n  const platform = getPlatform();\n  switch (platform) {\n    case 'macos':\n      return event.metaKey;\n    default:\n      return event.ctrlKey;\n  }\n};\n\nconst shakeAnimation = keyframes`\n  from,\n  to {\n    transform: translate3d(0, 0, 0);\n  }\n\n  14%,\n  42%,\n  70% {\n    transform: translate3d(-3px, 0, 0);\n  }\n\n  28%,\n  56%,\n  84% {\n    transform: translate3d(3px, 0, 0);\n  }\n`;\n\ntype State = {\n  hasNetworkError: boolean;\n  isLoading: boolean;\n  searchText: SearchText;\n  selectedResult: SelectedResult;\n  // Used for UX when browsing through history\n  enableHistory: boolean;\n  results: Command[];\n  stack: Stack[];\n};\ntype Action =\n  | { type: 'networkError'; payload: boolean }\n  | { type: 'loading'; payload: boolean }\n  | { type: 'selectedResult'; payload: number }\n  | { type: 'incrementSelectedResult' }\n  | { type: 'decrementSelectedResult' }\n  | {\n      type: 'pickCommandFromHistory';\n      payload: { searchText: SearchText; results: Command[] };\n    }\n  | { type: 'setNextCommands'; payload: { results: Command[] } }\n  | {\n      type: 'setPrevCommands';\n      payload: { searchText: SearchText; results: Command[] };\n    }\n  | { type: 'searchText'; payload: string }\n  | { type: 'setSearchTextResults'; payload: Command[] }\n  | { type: 'resetSearchText' }\n  | { type: 'resetResultsWhenClosing' }\n  | { type: 'reset' };\nconst initialState = {\n  hasNetworkError: false,\n  isLoading: false,\n  searchText: '',\n  selectedResult: -1,\n  // Used for UX when browsing through history\n  enableHistory: true,\n  results: [],\n  stack: [],\n};\nconst reducer = (state: State = initialState, action: Action): State => {\n  switch (action.type) {\n    case 'networkError':\n      return { ...state, hasNetworkError: action.payload };\n    case 'loading':\n      return { ...state, isLoading: action.payload };\n    case 'selectedResult':\n      return { ...state, selectedResult: action.payload };\n    case 'incrementSelectedResult':\n      return {\n        ...state,\n        selectedResult:\n          state.selectedResult === state.results.length - 1\n            ? 0\n            : state.selectedResult + 1,\n        enableHistory: false,\n      };\n    case 'decrementSelectedResult':\n      return {\n        ...state,\n        selectedResult:\n          state.selectedResult < 1\n            ? state.results.length - 1\n            : state.selectedResult - 1,\n        enableHistory: false,\n      };\n    case 'pickCommandFromHistory':\n      return {\n        ...state,\n        selectedResult: 0,\n        searchText: action.payload.searchText,\n        results: action.payload.results,\n        stack: [],\n        // The history does not get changed here, it will be changed along\n        // with the regular flow.\n      };\n    case 'setNextCommands':\n      return {\n        ...state,\n        stack: [\n          ...state.stack,\n          {\n            searchText: state.searchText,\n            results: state.results,\n            selectedResult: state.selectedResult,\n          },\n        ],\n        selectedResult: 0,\n        enableHistory: false,\n        results: action.payload.results,\n      };\n    case 'setPrevCommands':\n      return {\n        ...state,\n        searchText: action.payload.searchText,\n        results: action.payload.results,\n        selectedResult: 0,\n        enableHistory: false,\n        // omit last item\n        stack: state.stack.slice(0, -1),\n      };\n    case 'searchText':\n      return {\n        ...state,\n        searchText: action.payload,\n        // clear network error when search text is cleared, so that users\n        // are tempted to retry\n        hasNetworkError: action.payload.length > 0 && state.hasNetworkError,\n      };\n    case 'setSearchTextResults':\n      return {\n        ...state,\n        results: action.payload,\n        selectedResult: action.payload.length > 0 ? 0 : -1,\n        enableHistory: true,\n        stack: [],\n      };\n    case 'resetSearchText':\n      return { ...state, searchText: '', results: [], selectedResult: -1 };\n    case 'resetResultsWhenClosing':\n      return { ...state, selectedResult: -1, enableHistory: true };\n    case 'reset':\n      return initialState;\n    default:\n      return state;\n  }\n};\n\ntype Props = {\n  historyEntries: HistoryEntry[];\n  onHistoryEntriesChange: (historyEntries: HistoryEntry[]) => void;\n  search: (searchText: SearchText) => Promise<Command[]>;\n  getNextCommands: (command: Command) => Promise<Command[]>;\n  executeCommand: (command: Command, meta: { openInNewTab: boolean }) => void;\n  onClose: () => void;\n  classNameShakeAnimation: string;\n};\nconst Butler = (props: Props) => {\n  const intl = useIntl();\n  const [state, dispatch] = useReducer(reducer, initialState);\n\n  const shouldSelectFieldText = useRef(false);\n  const isNewWindowCombo = useRef(false);\n  const skipNextSelection = useRef(false);\n  const searchContainerRef = useRef<HTMLDivElement>(null);\n  const searchInputRef = useRef<HTMLInputElement>(null);\n\n  const setHasNetworkError = useCallback(() => {\n    dispatch({ type: 'networkError', payload: true });\n  }, []);\n  const unsetHasNetworkError = useCallback(() => {\n    dispatch({ type: 'networkError', payload: false });\n  }, []);\n  const setIsLoading = useCallback(() => {\n    dispatch({ type: 'loading', payload: true });\n  }, []);\n  const unsetIsLoading = useCallback(() => {\n    dispatch({ type: 'loading', payload: false });\n  }, []);\n\n  // Destructure functions from props to reference them in the hook dependency list\n  const {\n    search: searchFromParent,\n    onClose: onCloseFromParent,\n    executeCommand: executeCommandFromParent,\n    onHistoryEntriesChange: onHistoryEntriesChangeFromParent,\n    getNextCommands: getNextCommandsFromParent,\n  } = props;\n\n  const shake = useCallback(() => {\n    if (searchContainerRef.current) {\n      searchContainerRef.current.classList.remove(\n        props.classNameShakeAnimation\n      );\n      // -> triggering reflow\n      // eslint-disable-next-line no-void\n      void searchContainerRef.current.offsetWidth;\n      searchContainerRef.current.classList.add(props.classNameShakeAnimation);\n    }\n  }, [props.classNameShakeAnimation]);\n\n  const execute = useCallback(\n    (command: Command, meta: { openInNewTab: boolean }) => {\n      // Only main entries get added to history, so when a subcommand is executed,\n      // we add the main command of it to the history (the top-level command).\n      //\n      // The key to identify history entries by is always the searchText\n      // There will never be two history entries with the same searchText\n      const entry =\n        state.stack.length === 0\n          ? // The stack is empty, so we are executing a top-level command\n            { searchText: state.searchText, results: state.results }\n          : // We are executing a subcommand, so we get the top-level command for it,\n            // which is at the bottom of the stack.\n            {\n              searchText: state.stack[0].searchText,\n              results: state.stack[0].results,\n            };\n\n      // Add the entry to the history, while excluding any earlier history entry\n      // with the same search text. This effectively \"moves\" that entry to the\n      // top of the history (with the most recent results), or appends a new entry\n      // when it didn't exist before.\n      onHistoryEntriesChangeFromParent([\n        ...props.historyEntries.filter(\n          (command) => command.searchText !== entry.searchText\n        ),\n        entry,\n      ]);\n\n      dispatch({ type: 'resetSearchText' });\n\n      onCloseFromParent();\n\n      executeCommandFromParent(command, meta);\n    },\n    [\n      executeCommandFromParent,\n      onCloseFromParent,\n      onHistoryEntriesChangeFromParent,\n      props.historyEntries,\n      state.results,\n      state.searchText,\n      state.stack,\n    ]\n  );\n  const handleKeyDown = useCallback<KeyboardEventHandler<HTMLInputElement>>(\n    (event) => {\n      // Preventing cursor jumps can only happen in onKeyDown, but not in onKeyUp\n      event.persist();\n\n      // We want to know when the user presses cmd+enter (cmd being a meta key).\n      // We are only told about this in keyDown, but not in keyUp, so we need\n      // to handle it here\n      if (event.key === 'Enter' && hasNewWindowModifier(event)) {\n        isNewWindowCombo.current = true;\n        return;\n      }\n\n      // Avoid selecting the whole page when user selectes everything with\n      // a keyboard shortcut. There is probably a better way to do this though.\n      // This prevents the whole page from being selected in case the user\n      // 1) opens the search box\n      // 2) types into it\n      // 3) selects all text using cmd+a\n      // 4) closes the search box with esc\n      // Without this handling, the whole page would now be selected\n      if (isSelectAllCombo(event)) {\n        // This stops the browser from selecting anything\n        event.preventDefault();\n        if (searchInputRef.current) {\n          // This selects the text in the search input\n          searchInputRef.current.setSelectionRange(0, state.searchText.length);\n        }\n        return;\n      }\n\n      // avoid interfering with other key combinations using modifier keys\n      if (event.ctrlKey || event.altKey || event.shiftKey || event.metaKey)\n        return;\n      if (isCloseCombo(event)) return;\n\n      // skip next mouseEnter to avoid setting selectedResult when cursor just\n      // happens to be where the results will pop up\n      skipNextSelection.current = true;\n\n      if (event.key === 'ArrowDown') {\n        // prevent cursor from jumping to end of text input\n        event.preventDefault();\n        dispatch({ type: 'incrementSelectedResult' });\n        return;\n      }\n      if (event.key === 'ArrowUp') {\n        // browse through history\n        if (\n          state.searchText.length === 0 ||\n          (state.selectedResult < 1 && state.enableHistory)\n        ) {\n          shouldSelectFieldText.current = true;\n          const selectedIndex =\n            state.searchText.length === 0\n              ? // When going back the first step\n                -1\n              : // When going back more than one step\n                props.historyEntries.findIndex(\n                  (command) => command.searchText === state.searchText\n                );\n          // Pick the previous command from the history\n          const prevCommand =\n            selectedIndex === -1\n              ? // previous command on top of the history when going back on\n                // first step\n                last(props.historyEntries)\n              : // previous command is deeper down\n                // When the history does not exist (negative index), then\n                // this implicitly returns undefined\n                props.historyEntries[selectedIndex - 1];\n          // Skip when no previous entry exists in the history\n          if (!prevCommand) return;\n          dispatch({\n            type: 'pickCommandFromHistory',\n            payload: {\n              searchText: prevCommand.searchText,\n              results: prevCommand.results,\n            },\n          });\n          return;\n        }\n        // prevent cursor from jumping to beginning of text input\n        event.preventDefault();\n        dispatch({ type: 'decrementSelectedResult' });\n        return;\n      }\n      if (state.selectedResult > -1) {\n        if (event.key === 'ArrowRight') {\n          const command = state.results[state.selectedResult];\n          const searchText = state.searchText;\n          const isCursorAtEnd =\n            searchInputRef.current &&\n            state.searchText.length === searchInputRef.current.selectionStart;\n\n          const isEverythingSelected =\n            searchInputRef.current &&\n            searchInputRef.current.selectionStart === 0 &&\n            state.searchText.length === searchInputRef.current.selectionEnd;\n\n          // only allow diving in when cursor is at end of input or when\n          // the complete text is selected (when browsing through history)\n          if (!isCursorAtEnd && !isEverythingSelected) return;\n\n          unsetHasNetworkError();\n\n          // NOTE: since we need to fetch the \"next command\", which is an async operation,\n          // we use a IIFE to process that and eventually update the state.\n          (async () => {\n            if (command) {\n              const nextCommands = await getNextCommandsFromParent(command);\n              // avoid moving cursor when there are sub-options\n              if (nextCommands.length > 0) {\n                // Ensure the search text has not changed while we were loading\n                // the next results, otherwise we'd interrupt the user.\n                // Throw away the results in case the search text has changed.\n                if (state.searchText === searchText) {\n                  dispatch({\n                    type: 'setNextCommands',\n                    payload: { results: nextCommands },\n                  });\n                }\n                return;\n              }\n            }\n            shake();\n          })();\n          return;\n        }\n        if (event.key === 'ArrowLeft') {\n          // go left in stack\n          const prevCommand = last(state.stack);\n\n          // do nothing when we can't go left anymore\n          if (!prevCommand) return;\n\n          // prevent cursor from jumping a char to the left in text input\n          event.preventDefault();\n\n          dispatch({\n            type: 'setPrevCommands',\n            payload: {\n              searchText: prevCommand.searchText,\n              results: prevCommand.results,\n            },\n          });\n          return;\n        }\n      }\n    },\n    [\n      getNextCommandsFromParent,\n      props.historyEntries,\n      shake,\n      state,\n      unsetHasNetworkError,\n    ]\n  );\n  const handleKeyUp = useCallback<KeyboardEventHandler<HTMLInputElement>>(\n    (event) => {\n      // setting the selection can only happen in onKeyUp\n      if (shouldSelectFieldText.current) {\n        const input = event.target as HTMLInputElement;\n        input.focus();\n        input.select();\n        shouldSelectFieldText.current = false;\n      }\n\n      if (event.key !== 'Enter' && !isNewWindowCombo.current) return true;\n\n      // User just triggered the search\n      if (state.selectedResult === -1) return true;\n\n      // User had something selected and wants to go there\n      execute(state.results[state.selectedResult], {\n        openInNewTab: isNewWindowCombo.current,\n      });\n\n      isNewWindowCombo.current = false;\n      return true;\n    },\n    [execute, state.results, state.selectedResult]\n  );\n  const handleChange = useCallback<ChangeEventHandler<HTMLInputElement>>(\n    (event) => {\n      const searchText = event.target.value;\n      if (searchText.trim().length === 0) {\n        dispatch({ type: 'reset' });\n        return;\n      }\n\n      dispatch({ type: 'searchText', payload: searchText });\n\n      // A search via network is only triggered when there\n      // are more than three characters. So no false loading\n      // indication is given.\n      if (searchText.trim().length > 3) {\n        setIsLoading();\n      }\n\n      searchFromParent(searchText).then(\n        (asyncResults: Command[]) => {\n          unsetHasNetworkError();\n          unsetIsLoading();\n\n          const fuse = new Fuse(asyncResults, {\n            keys: [\n              { name: 'text', weight: 0.6 },\n              { name: 'keywords', weight: 0.4 },\n            ],\n            minMatchCharLength: 2,\n            includeScore: true,\n          });\n\n          const searchResults = fuse\n            .search(searchText)\n            // Filter out results with a matching score over 0.75\n            .filter((result) => (result.score ? result.score < 0.75 : false))\n            // Keep a maximal of 9 results\n            .slice(0, 9);\n\n          dispatch({\n            type: 'setSearchTextResults',\n            payload: searchResults.map((result) => result.item),\n          });\n        },\n        (error: Error) => {\n          // eslint-disable-next-line no-console\n          if (process.env.NODE_ENV !== 'production') console.error(error);\n          unsetIsLoading();\n          setHasNetworkError();\n        }\n      );\n    },\n    [\n      searchFromParent,\n      setHasNetworkError,\n      setIsLoading,\n      unsetHasNetworkError,\n      unsetIsLoading,\n    ]\n  );\n  const handleContainerClick = useCallback(() => {\n    dispatch({ type: 'resetResultsWhenClosing' });\n    onCloseFromParent();\n  }, [onCloseFromParent]);\n\n  const createCommandMouseEnterHandler = useCallback<\n    (index: number) => MouseEventHandler<HTMLDivElement>\n  >(\n    (index) => () => {\n      // In case the cursor happened to be in a location where a\n      // result would appear, it would trigger onMouseEnter and the\n      // result would be selected immediately. This is not something\n      // a user would expect, hence we prevent it from happening.\n      // The user has to move the cursor to an option explicitly for\n      // it to become active. However, the user can always click and\n      // that action will be triggered.\n      if (skipNextSelection.current) {\n        skipNextSelection.current = false;\n        return;\n      }\n\n      // sets the selected result, mainly for the hover effect\n      dispatch({ type: 'selectedResult', payload: index });\n    },\n    []\n  );\n  const createCommandClickHandler = useCallback<\n    (command: Command) => MouseEventHandler<HTMLDivElement>\n  >(\n    (command) => (event) => {\n      execute(command, {\n        openInNewTab: hasNewWindowModifier(event),\n      });\n    },\n    [execute]\n  );\n\n  return (\n    <ButlerContainer\n      onClick={handleContainerClick}\n      data-testid=\"quick-access\"\n      tabIndex={-1}\n    >\n      <div\n        ref={searchContainerRef}\n        css={css`\n          background-color: ${uiKitDesignTokens.colorSurface};\n          border: 0;\n          border-radius: ${uiKitDesignTokens.borderRadius4};\n          min-height: 40px;\n\n          /* one more than app-bar (20000) and one more than the overlay (20001) */\n          z-index: 20002;\n          width: 400px;\n          margin: 40px auto;\n          overflow: hidden;\n          -webkit-box-shadow: 0 10px 30px -8px rgba(0, 0, 0, 0.75);\n          -moz-box-shadow: 0 10px 30px -8px rgba(0, 0, 0, 0.75);\n          box-shadow: 0 10px 30px -8px rgba(0, 0, 0, 0.75);\n          padding-bottom: ${state.hasNetworkError\n            ? '0'\n            : uiKitDesignTokens.spacingS};\n        `}\n        onClick={(event) => {\n          // Avoid closing when the searchContainer itself is clicked\n          // If we don't do this, then the overlay will close when e.g.\n          // the search input is clicked.\n          event.stopPropagation();\n          event.preventDefault();\n        }}\n      >\n        <div\n          css={css`\n            display: flex;\n          `}\n        >\n          <label\n            htmlFor=\"quick-access-search-input\"\n            css={css`\n              align-self: center;\n              padding-left: ${uiKitDesignTokens.spacingM};\n              margin-top: ${uiKitDesignTokens.spacingS};\n            `}\n          >\n            <SearchIcon color=\"neutral60\" />\n          </label>\n          <input\n            id=\"quick-access-search-input\"\n            ref={searchInputRef}\n            placeholder={intl.formatMessage(messages.inputPlacehoder)}\n            type=\"text\"\n            css={css`\n              width: 100%;\n              border: 0;\n              outline: 0;\n              font-size: 22px;\n              font-weight: 300;\n              padding: ${uiKitDesignTokens.spacingM}\n                ${uiKitDesignTokens.spacingM} ${uiKitDesignTokens.spacingS}\n                ${uiKitDesignTokens.spacingS};\n              &::placeholder {\n                color: ${uiKitDesignTokens.colorNeutral60};\n              }\n            `}\n            value={state.searchText}\n            onChange={handleChange}\n            onKeyDown={handleKeyDown}\n            onKeyUp={handleKeyUp}\n            autoFocus={true}\n            autoComplete=\"off\"\n            data-testid=\"quick-access-search-input\"\n          />\n          {state.isLoading && (\n            <div\n              css={css`\n                align-self: center;\n                margin-top: ${uiKitDesignTokens.spacingS};\n                margin-right: ${uiKitDesignTokens.spacingS};\n              `}\n            >\n              <LoadingSpinner />\n            </div>\n          )}\n        </div>\n        {(() => {\n          if (state.hasNetworkError)\n            return (\n              <div\n                css={css`\n                  overflow: hidden;\n                  white-space: nowrap;\n                  cursor: default;\n                  background: ${uiKitDesignTokens.colorError};\n                  text-align: center;\n                  text-transform: uppercase;\n                  color: ${uiKitDesignTokens.colorSurface};\n                  font-size: ${uiKitDesignTokens.fontSize20};\n                  padding: ${uiKitDesignTokens.spacingXs};\n                `}\n              >\n                <FormattedMessage {...messages.offline} />\n              </div>\n            );\n\n          if (state.results.length === 0 && state.searchText.trim().length > 0)\n            return (\n              <div\n                css={css`\n                  overflow: hidden;\n                  white-space: nowrap;\n                  cursor: default;\n                  background: ${uiKitDesignTokens.colorNeutral};\n                  color: ${uiKitDesignTokens.colorSolid};\n                  text-align: center;\n                  text-transform: uppercase;\n                  font-size: ${uiKitDesignTokens.fontSize20};\n                  padding: ${uiKitDesignTokens.spacingXs};\n                `}\n              >\n                <FormattedMessage {...messages.noResults} />\n              </div>\n            );\n\n          return state.results.map((command, index) => (\n            <ButlerCommand\n              key={command.id}\n              command={command}\n              isSelected={state.selectedResult === index}\n              onMouseEnter={createCommandMouseEnterHandler(index)}\n              onClick={createCommandClickHandler(command)}\n            />\n          ));\n        })()}\n      </div>\n    </ButlerContainer>\n  );\n};\nButler.displayName = 'Butler';\n\nconst ButlerWithAnimation = (props: Omit<Props, 'classNameShakeAnimation'>) => (\n  <ClassNames>\n    {({ css }) => (\n      <Butler\n        {...props}\n        classNameShakeAnimation={css`\n          animation-duration: 0.45s;\n          animation-fill-mode: both;\n          animation-name: ${shakeAnimation};\n        `}\n      />\n    )}\n  </ClassNames>\n);\n\nexport default ButlerWithAnimation;\n"]} */"),
852
- onClick: event => {
853
- // Avoid closing when the searchContainer itself is clicked
854
- // If we don't do this, then the overlay will close when e.g.
855
- // the search input is clicked.
856
- event.stopPropagation();
857
- event.preventDefault();
858
- },
859
- children: [jsxs("div", {
860
- css: _ref,
861
- children: [jsx("label", {
862
- htmlFor: "quick-access-search-input",
863
- css: /*#__PURE__*/css("align-self:center;padding-left:", designTokens.spacingM, ";margin-top:", designTokens.spacingS, ";" + (process.env.NODE_ENV === "production" ? "" : ";label:Butler;"), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["butler.tsx"],"names":[],"mappings":"AAonBoB","file":"butler.tsx","sourcesContent":["import {\n  KeyboardEventHandler,\n  ChangeEventHandler,\n  KeyboardEvent,\n  MouseEventHandler,\n  MouseEvent,\n  useReducer,\n  useRef,\n  useCallback,\n} from 'react';\nimport { css, keyframes, ClassNames } from '@emotion/react';\nimport Fuse from 'fuse.js';\nimport last from 'lodash/last';\nimport { FormattedMessage, useIntl } from 'react-intl';\nimport { designTokens as uiKitDesignTokens } from '@commercetools-uikit/design-system';\nimport { SearchIcon } from '@commercetools-uikit/icons';\nimport LoadingSpinner from '@commercetools-uikit/loading-spinner';\nimport ButlerCommand from '../butler-command';\nimport ButlerContainer from '../butler-container';\nimport messages from '../messages';\nimport type {\n  Command,\n  SearchText,\n  SelectedResult,\n  Stack,\n  HistoryEntry,\n} from '../types';\n\nconst isSelectAllCombo = (event: KeyboardEvent<HTMLInputElement>) =>\n  event.key === 'a' &&\n  event.metaKey &&\n  !event.ctrlKey &&\n  !event.altKey &&\n  !event.shiftKey;\n\nconst isCloseCombo = (event: KeyboardEvent<HTMLInputElement>) =>\n  event.key === 'Escape' &&\n  !event.metaKey &&\n  !event.ctrlKey &&\n  !event.altKey &&\n  !event.shiftKey;\n\nconst getPlatform = () => {\n  if (navigator.appVersion.includes('Win')) return 'windows';\n  if (navigator.appVersion.includes('Mac')) return 'macos';\n  if (navigator.appVersion.includes('X11')) return 'unix';\n  if (navigator.appVersion.includes('Linux')) return 'linux';\n\n  return null;\n};\n\nconst hasNewWindowModifier = (\n  event: KeyboardEvent<HTMLElement> | MouseEvent<HTMLElement>\n) => {\n  const platform = getPlatform();\n  switch (platform) {\n    case 'macos':\n      return event.metaKey;\n    default:\n      return event.ctrlKey;\n  }\n};\n\nconst shakeAnimation = keyframes`\n  from,\n  to {\n    transform: translate3d(0, 0, 0);\n  }\n\n  14%,\n  42%,\n  70% {\n    transform: translate3d(-3px, 0, 0);\n  }\n\n  28%,\n  56%,\n  84% {\n    transform: translate3d(3px, 0, 0);\n  }\n`;\n\ntype State = {\n  hasNetworkError: boolean;\n  isLoading: boolean;\n  searchText: SearchText;\n  selectedResult: SelectedResult;\n  // Used for UX when browsing through history\n  enableHistory: boolean;\n  results: Command[];\n  stack: Stack[];\n};\ntype Action =\n  | { type: 'networkError'; payload: boolean }\n  | { type: 'loading'; payload: boolean }\n  | { type: 'selectedResult'; payload: number }\n  | { type: 'incrementSelectedResult' }\n  | { type: 'decrementSelectedResult' }\n  | {\n      type: 'pickCommandFromHistory';\n      payload: { searchText: SearchText; results: Command[] };\n    }\n  | { type: 'setNextCommands'; payload: { results: Command[] } }\n  | {\n      type: 'setPrevCommands';\n      payload: { searchText: SearchText; results: Command[] };\n    }\n  | { type: 'searchText'; payload: string }\n  | { type: 'setSearchTextResults'; payload: Command[] }\n  | { type: 'resetSearchText' }\n  | { type: 'resetResultsWhenClosing' }\n  | { type: 'reset' };\nconst initialState = {\n  hasNetworkError: false,\n  isLoading: false,\n  searchText: '',\n  selectedResult: -1,\n  // Used for UX when browsing through history\n  enableHistory: true,\n  results: [],\n  stack: [],\n};\nconst reducer = (state: State = initialState, action: Action): State => {\n  switch (action.type) {\n    case 'networkError':\n      return { ...state, hasNetworkError: action.payload };\n    case 'loading':\n      return { ...state, isLoading: action.payload };\n    case 'selectedResult':\n      return { ...state, selectedResult: action.payload };\n    case 'incrementSelectedResult':\n      return {\n        ...state,\n        selectedResult:\n          state.selectedResult === state.results.length - 1\n            ? 0\n            : state.selectedResult + 1,\n        enableHistory: false,\n      };\n    case 'decrementSelectedResult':\n      return {\n        ...state,\n        selectedResult:\n          state.selectedResult < 1\n            ? state.results.length - 1\n            : state.selectedResult - 1,\n        enableHistory: false,\n      };\n    case 'pickCommandFromHistory':\n      return {\n        ...state,\n        selectedResult: 0,\n        searchText: action.payload.searchText,\n        results: action.payload.results,\n        stack: [],\n        // The history does not get changed here, it will be changed along\n        // with the regular flow.\n      };\n    case 'setNextCommands':\n      return {\n        ...state,\n        stack: [\n          ...state.stack,\n          {\n            searchText: state.searchText,\n            results: state.results,\n            selectedResult: state.selectedResult,\n          },\n        ],\n        selectedResult: 0,\n        enableHistory: false,\n        results: action.payload.results,\n      };\n    case 'setPrevCommands':\n      return {\n        ...state,\n        searchText: action.payload.searchText,\n        results: action.payload.results,\n        selectedResult: 0,\n        enableHistory: false,\n        // omit last item\n        stack: state.stack.slice(0, -1),\n      };\n    case 'searchText':\n      return {\n        ...state,\n        searchText: action.payload,\n        // clear network error when search text is cleared, so that users\n        // are tempted to retry\n        hasNetworkError: action.payload.length > 0 && state.hasNetworkError,\n      };\n    case 'setSearchTextResults':\n      return {\n        ...state,\n        results: action.payload,\n        selectedResult: action.payload.length > 0 ? 0 : -1,\n        enableHistory: true,\n        stack: [],\n      };\n    case 'resetSearchText':\n      return { ...state, searchText: '', results: [], selectedResult: -1 };\n    case 'resetResultsWhenClosing':\n      return { ...state, selectedResult: -1, enableHistory: true };\n    case 'reset':\n      return initialState;\n    default:\n      return state;\n  }\n};\n\ntype Props = {\n  historyEntries: HistoryEntry[];\n  onHistoryEntriesChange: (historyEntries: HistoryEntry[]) => void;\n  search: (searchText: SearchText) => Promise<Command[]>;\n  getNextCommands: (command: Command) => Promise<Command[]>;\n  executeCommand: (command: Command, meta: { openInNewTab: boolean }) => void;\n  onClose: () => void;\n  classNameShakeAnimation: string;\n};\nconst Butler = (props: Props) => {\n  const intl = useIntl();\n  const [state, dispatch] = useReducer(reducer, initialState);\n\n  const shouldSelectFieldText = useRef(false);\n  const isNewWindowCombo = useRef(false);\n  const skipNextSelection = useRef(false);\n  const searchContainerRef = useRef<HTMLDivElement>(null);\n  const searchInputRef = useRef<HTMLInputElement>(null);\n\n  const setHasNetworkError = useCallback(() => {\n    dispatch({ type: 'networkError', payload: true });\n  }, []);\n  const unsetHasNetworkError = useCallback(() => {\n    dispatch({ type: 'networkError', payload: false });\n  }, []);\n  const setIsLoading = useCallback(() => {\n    dispatch({ type: 'loading', payload: true });\n  }, []);\n  const unsetIsLoading = useCallback(() => {\n    dispatch({ type: 'loading', payload: false });\n  }, []);\n\n  // Destructure functions from props to reference them in the hook dependency list\n  const {\n    search: searchFromParent,\n    onClose: onCloseFromParent,\n    executeCommand: executeCommandFromParent,\n    onHistoryEntriesChange: onHistoryEntriesChangeFromParent,\n    getNextCommands: getNextCommandsFromParent,\n  } = props;\n\n  const shake = useCallback(() => {\n    if (searchContainerRef.current) {\n      searchContainerRef.current.classList.remove(\n        props.classNameShakeAnimation\n      );\n      // -> triggering reflow\n      // eslint-disable-next-line no-void\n      void searchContainerRef.current.offsetWidth;\n      searchContainerRef.current.classList.add(props.classNameShakeAnimation);\n    }\n  }, [props.classNameShakeAnimation]);\n\n  const execute = useCallback(\n    (command: Command, meta: { openInNewTab: boolean }) => {\n      // Only main entries get added to history, so when a subcommand is executed,\n      // we add the main command of it to the history (the top-level command).\n      //\n      // The key to identify history entries by is always the searchText\n      // There will never be two history entries with the same searchText\n      const entry =\n        state.stack.length === 0\n          ? // The stack is empty, so we are executing a top-level command\n            { searchText: state.searchText, results: state.results }\n          : // We are executing a subcommand, so we get the top-level command for it,\n            // which is at the bottom of the stack.\n            {\n              searchText: state.stack[0].searchText,\n              results: state.stack[0].results,\n            };\n\n      // Add the entry to the history, while excluding any earlier history entry\n      // with the same search text. This effectively \"moves\" that entry to the\n      // top of the history (with the most recent results), or appends a new entry\n      // when it didn't exist before.\n      onHistoryEntriesChangeFromParent([\n        ...props.historyEntries.filter(\n          (command) => command.searchText !== entry.searchText\n        ),\n        entry,\n      ]);\n\n      dispatch({ type: 'resetSearchText' });\n\n      onCloseFromParent();\n\n      executeCommandFromParent(command, meta);\n    },\n    [\n      executeCommandFromParent,\n      onCloseFromParent,\n      onHistoryEntriesChangeFromParent,\n      props.historyEntries,\n      state.results,\n      state.searchText,\n      state.stack,\n    ]\n  );\n  const handleKeyDown = useCallback<KeyboardEventHandler<HTMLInputElement>>(\n    (event) => {\n      // Preventing cursor jumps can only happen in onKeyDown, but not in onKeyUp\n      event.persist();\n\n      // We want to know when the user presses cmd+enter (cmd being a meta key).\n      // We are only told about this in keyDown, but not in keyUp, so we need\n      // to handle it here\n      if (event.key === 'Enter' && hasNewWindowModifier(event)) {\n        isNewWindowCombo.current = true;\n        return;\n      }\n\n      // Avoid selecting the whole page when user selectes everything with\n      // a keyboard shortcut. There is probably a better way to do this though.\n      // This prevents the whole page from being selected in case the user\n      // 1) opens the search box\n      // 2) types into it\n      // 3) selects all text using cmd+a\n      // 4) closes the search box with esc\n      // Without this handling, the whole page would now be selected\n      if (isSelectAllCombo(event)) {\n        // This stops the browser from selecting anything\n        event.preventDefault();\n        if (searchInputRef.current) {\n          // This selects the text in the search input\n          searchInputRef.current.setSelectionRange(0, state.searchText.length);\n        }\n        return;\n      }\n\n      // avoid interfering with other key combinations using modifier keys\n      if (event.ctrlKey || event.altKey || event.shiftKey || event.metaKey)\n        return;\n      if (isCloseCombo(event)) return;\n\n      // skip next mouseEnter to avoid setting selectedResult when cursor just\n      // happens to be where the results will pop up\n      skipNextSelection.current = true;\n\n      if (event.key === 'ArrowDown') {\n        // prevent cursor from jumping to end of text input\n        event.preventDefault();\n        dispatch({ type: 'incrementSelectedResult' });\n        return;\n      }\n      if (event.key === 'ArrowUp') {\n        // browse through history\n        if (\n          state.searchText.length === 0 ||\n          (state.selectedResult < 1 && state.enableHistory)\n        ) {\n          shouldSelectFieldText.current = true;\n          const selectedIndex =\n            state.searchText.length === 0\n              ? // When going back the first step\n                -1\n              : // When going back more than one step\n                props.historyEntries.findIndex(\n                  (command) => command.searchText === state.searchText\n                );\n          // Pick the previous command from the history\n          const prevCommand =\n            selectedIndex === -1\n              ? // previous command on top of the history when going back on\n                // first step\n                last(props.historyEntries)\n              : // previous command is deeper down\n                // When the history does not exist (negative index), then\n                // this implicitly returns undefined\n                props.historyEntries[selectedIndex - 1];\n          // Skip when no previous entry exists in the history\n          if (!prevCommand) return;\n          dispatch({\n            type: 'pickCommandFromHistory',\n            payload: {\n              searchText: prevCommand.searchText,\n              results: prevCommand.results,\n            },\n          });\n          return;\n        }\n        // prevent cursor from jumping to beginning of text input\n        event.preventDefault();\n        dispatch({ type: 'decrementSelectedResult' });\n        return;\n      }\n      if (state.selectedResult > -1) {\n        if (event.key === 'ArrowRight') {\n          const command = state.results[state.selectedResult];\n          const searchText = state.searchText;\n          const isCursorAtEnd =\n            searchInputRef.current &&\n            state.searchText.length === searchInputRef.current.selectionStart;\n\n          const isEverythingSelected =\n            searchInputRef.current &&\n            searchInputRef.current.selectionStart === 0 &&\n            state.searchText.length === searchInputRef.current.selectionEnd;\n\n          // only allow diving in when cursor is at end of input or when\n          // the complete text is selected (when browsing through history)\n          if (!isCursorAtEnd && !isEverythingSelected) return;\n\n          unsetHasNetworkError();\n\n          // NOTE: since we need to fetch the \"next command\", which is an async operation,\n          // we use a IIFE to process that and eventually update the state.\n          (async () => {\n            if (command) {\n              const nextCommands = await getNextCommandsFromParent(command);\n              // avoid moving cursor when there are sub-options\n              if (nextCommands.length > 0) {\n                // Ensure the search text has not changed while we were loading\n                // the next results, otherwise we'd interrupt the user.\n                // Throw away the results in case the search text has changed.\n                if (state.searchText === searchText) {\n                  dispatch({\n                    type: 'setNextCommands',\n                    payload: { results: nextCommands },\n                  });\n                }\n                return;\n              }\n            }\n            shake();\n          })();\n          return;\n        }\n        if (event.key === 'ArrowLeft') {\n          // go left in stack\n          const prevCommand = last(state.stack);\n\n          // do nothing when we can't go left anymore\n          if (!prevCommand) return;\n\n          // prevent cursor from jumping a char to the left in text input\n          event.preventDefault();\n\n          dispatch({\n            type: 'setPrevCommands',\n            payload: {\n              searchText: prevCommand.searchText,\n              results: prevCommand.results,\n            },\n          });\n          return;\n        }\n      }\n    },\n    [\n      getNextCommandsFromParent,\n      props.historyEntries,\n      shake,\n      state,\n      unsetHasNetworkError,\n    ]\n  );\n  const handleKeyUp = useCallback<KeyboardEventHandler<HTMLInputElement>>(\n    (event) => {\n      // setting the selection can only happen in onKeyUp\n      if (shouldSelectFieldText.current) {\n        const input = event.target as HTMLInputElement;\n        input.focus();\n        input.select();\n        shouldSelectFieldText.current = false;\n      }\n\n      if (event.key !== 'Enter' && !isNewWindowCombo.current) return true;\n\n      // User just triggered the search\n      if (state.selectedResult === -1) return true;\n\n      // User had something selected and wants to go there\n      execute(state.results[state.selectedResult], {\n        openInNewTab: isNewWindowCombo.current,\n      });\n\n      isNewWindowCombo.current = false;\n      return true;\n    },\n    [execute, state.results, state.selectedResult]\n  );\n  const handleChange = useCallback<ChangeEventHandler<HTMLInputElement>>(\n    (event) => {\n      const searchText = event.target.value;\n      if (searchText.trim().length === 0) {\n        dispatch({ type: 'reset' });\n        return;\n      }\n\n      dispatch({ type: 'searchText', payload: searchText });\n\n      // A search via network is only triggered when there\n      // are more than three characters. So no false loading\n      // indication is given.\n      if (searchText.trim().length > 3) {\n        setIsLoading();\n      }\n\n      searchFromParent(searchText).then(\n        (asyncResults: Command[]) => {\n          unsetHasNetworkError();\n          unsetIsLoading();\n\n          const fuse = new Fuse(asyncResults, {\n            keys: [\n              { name: 'text', weight: 0.6 },\n              { name: 'keywords', weight: 0.4 },\n            ],\n            minMatchCharLength: 2,\n            includeScore: true,\n          });\n\n          const searchResults = fuse\n            .search(searchText)\n            // Filter out results with a matching score over 0.75\n            .filter((result) => (result.score ? result.score < 0.75 : false))\n            // Keep a maximal of 9 results\n            .slice(0, 9);\n\n          dispatch({\n            type: 'setSearchTextResults',\n            payload: searchResults.map((result) => result.item),\n          });\n        },\n        (error: Error) => {\n          // eslint-disable-next-line no-console\n          if (process.env.NODE_ENV !== 'production') console.error(error);\n          unsetIsLoading();\n          setHasNetworkError();\n        }\n      );\n    },\n    [\n      searchFromParent,\n      setHasNetworkError,\n      setIsLoading,\n      unsetHasNetworkError,\n      unsetIsLoading,\n    ]\n  );\n  const handleContainerClick = useCallback(() => {\n    dispatch({ type: 'resetResultsWhenClosing' });\n    onCloseFromParent();\n  }, [onCloseFromParent]);\n\n  const createCommandMouseEnterHandler = useCallback<\n    (index: number) => MouseEventHandler<HTMLDivElement>\n  >(\n    (index) => () => {\n      // In case the cursor happened to be in a location where a\n      // result would appear, it would trigger onMouseEnter and the\n      // result would be selected immediately. This is not something\n      // a user would expect, hence we prevent it from happening.\n      // The user has to move the cursor to an option explicitly for\n      // it to become active. However, the user can always click and\n      // that action will be triggered.\n      if (skipNextSelection.current) {\n        skipNextSelection.current = false;\n        return;\n      }\n\n      // sets the selected result, mainly for the hover effect\n      dispatch({ type: 'selectedResult', payload: index });\n    },\n    []\n  );\n  const createCommandClickHandler = useCallback<\n    (command: Command) => MouseEventHandler<HTMLDivElement>\n  >(\n    (command) => (event) => {\n      execute(command, {\n        openInNewTab: hasNewWindowModifier(event),\n      });\n    },\n    [execute]\n  );\n\n  return (\n    <ButlerContainer\n      onClick={handleContainerClick}\n      data-testid=\"quick-access\"\n      tabIndex={-1}\n    >\n      <div\n        ref={searchContainerRef}\n        css={css`\n          background-color: ${uiKitDesignTokens.colorSurface};\n          border: 0;\n          border-radius: ${uiKitDesignTokens.borderRadius4};\n          min-height: 40px;\n\n          /* one more than app-bar (20000) and one more than the overlay (20001) */\n          z-index: 20002;\n          width: 400px;\n          margin: 40px auto;\n          overflow: hidden;\n          -webkit-box-shadow: 0 10px 30px -8px rgba(0, 0, 0, 0.75);\n          -moz-box-shadow: 0 10px 30px -8px rgba(0, 0, 0, 0.75);\n          box-shadow: 0 10px 30px -8px rgba(0, 0, 0, 0.75);\n          padding-bottom: ${state.hasNetworkError\n            ? '0'\n            : uiKitDesignTokens.spacingS};\n        `}\n        onClick={(event) => {\n          // Avoid closing when the searchContainer itself is clicked\n          // If we don't do this, then the overlay will close when e.g.\n          // the search input is clicked.\n          event.stopPropagation();\n          event.preventDefault();\n        }}\n      >\n        <div\n          css={css`\n            display: flex;\n          `}\n        >\n          <label\n            htmlFor=\"quick-access-search-input\"\n            css={css`\n              align-self: center;\n              padding-left: ${uiKitDesignTokens.spacingM};\n              margin-top: ${uiKitDesignTokens.spacingS};\n            `}\n          >\n            <SearchIcon color=\"neutral60\" />\n          </label>\n          <input\n            id=\"quick-access-search-input\"\n            ref={searchInputRef}\n            placeholder={intl.formatMessage(messages.inputPlacehoder)}\n            type=\"text\"\n            css={css`\n              width: 100%;\n              border: 0;\n              outline: 0;\n              font-size: 22px;\n              font-weight: 300;\n              padding: ${uiKitDesignTokens.spacingM}\n                ${uiKitDesignTokens.spacingM} ${uiKitDesignTokens.spacingS}\n                ${uiKitDesignTokens.spacingS};\n              &::placeholder {\n                color: ${uiKitDesignTokens.colorNeutral60};\n              }\n            `}\n            value={state.searchText}\n            onChange={handleChange}\n            onKeyDown={handleKeyDown}\n            onKeyUp={handleKeyUp}\n            autoFocus={true}\n            autoComplete=\"off\"\n            data-testid=\"quick-access-search-input\"\n          />\n          {state.isLoading && (\n            <div\n              css={css`\n                align-self: center;\n                margin-top: ${uiKitDesignTokens.spacingS};\n                margin-right: ${uiKitDesignTokens.spacingS};\n              `}\n            >\n              <LoadingSpinner />\n            </div>\n          )}\n        </div>\n        {(() => {\n          if (state.hasNetworkError)\n            return (\n              <div\n                css={css`\n                  overflow: hidden;\n                  white-space: nowrap;\n                  cursor: default;\n                  background: ${uiKitDesignTokens.colorError};\n                  text-align: center;\n                  text-transform: uppercase;\n                  color: ${uiKitDesignTokens.colorSurface};\n                  font-size: ${uiKitDesignTokens.fontSize20};\n                  padding: ${uiKitDesignTokens.spacingXs};\n                `}\n              >\n                <FormattedMessage {...messages.offline} />\n              </div>\n            );\n\n          if (state.results.length === 0 && state.searchText.trim().length > 0)\n            return (\n              <div\n                css={css`\n                  overflow: hidden;\n                  white-space: nowrap;\n                  cursor: default;\n                  background: ${uiKitDesignTokens.colorNeutral};\n                  color: ${uiKitDesignTokens.colorSolid};\n                  text-align: center;\n                  text-transform: uppercase;\n                  font-size: ${uiKitDesignTokens.fontSize20};\n                  padding: ${uiKitDesignTokens.spacingXs};\n                `}\n              >\n                <FormattedMessage {...messages.noResults} />\n              </div>\n            );\n\n          return state.results.map((command, index) => (\n            <ButlerCommand\n              key={command.id}\n              command={command}\n              isSelected={state.selectedResult === index}\n              onMouseEnter={createCommandMouseEnterHandler(index)}\n              onClick={createCommandClickHandler(command)}\n            />\n          ));\n        })()}\n      </div>\n    </ButlerContainer>\n  );\n};\nButler.displayName = 'Butler';\n\nconst ButlerWithAnimation = (props: Omit<Props, 'classNameShakeAnimation'>) => (\n  <ClassNames>\n    {({ css }) => (\n      <Butler\n        {...props}\n        classNameShakeAnimation={css`\n          animation-duration: 0.45s;\n          animation-fill-mode: both;\n          animation-name: ${shakeAnimation};\n        `}\n      />\n    )}\n  </ClassNames>\n);\n\nexport default ButlerWithAnimation;\n"]} */"),
864
- children: jsx(SearchIcon, {
865
- color: "neutral60"
866
- })
867
- }), jsx("input", {
868
- id: "quick-access-search-input",
869
- ref: searchInputRef,
870
- placeholder: intl.formatMessage(messages.inputPlacehoder),
871
- type: "text",
872
- css: /*#__PURE__*/css("width:100%;border:0;outline:0;font-size:22px;font-weight:300;padding:", designTokens.spacingM, " ", designTokens.spacingM, " ", designTokens.spacingS, " ", designTokens.spacingS, ";&::placeholder{color:", designTokens.colorNeutral60, ";}" + (process.env.NODE_ENV === "production" ? "" : ";label:Butler;"), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["butler.tsx"],"names":[],"mappings":"AAioBoB","file":"butler.tsx","sourcesContent":["import {\n  KeyboardEventHandler,\n  ChangeEventHandler,\n  KeyboardEvent,\n  MouseEventHandler,\n  MouseEvent,\n  useReducer,\n  useRef,\n  useCallback,\n} from 'react';\nimport { css, keyframes, ClassNames } from '@emotion/react';\nimport Fuse from 'fuse.js';\nimport last from 'lodash/last';\nimport { FormattedMessage, useIntl } from 'react-intl';\nimport { designTokens as uiKitDesignTokens } from '@commercetools-uikit/design-system';\nimport { SearchIcon } from '@commercetools-uikit/icons';\nimport LoadingSpinner from '@commercetools-uikit/loading-spinner';\nimport ButlerCommand from '../butler-command';\nimport ButlerContainer from '../butler-container';\nimport messages from '../messages';\nimport type {\n  Command,\n  SearchText,\n  SelectedResult,\n  Stack,\n  HistoryEntry,\n} from '../types';\n\nconst isSelectAllCombo = (event: KeyboardEvent<HTMLInputElement>) =>\n  event.key === 'a' &&\n  event.metaKey &&\n  !event.ctrlKey &&\n  !event.altKey &&\n  !event.shiftKey;\n\nconst isCloseCombo = (event: KeyboardEvent<HTMLInputElement>) =>\n  event.key === 'Escape' &&\n  !event.metaKey &&\n  !event.ctrlKey &&\n  !event.altKey &&\n  !event.shiftKey;\n\nconst getPlatform = () => {\n  if (navigator.appVersion.includes('Win')) return 'windows';\n  if (navigator.appVersion.includes('Mac')) return 'macos';\n  if (navigator.appVersion.includes('X11')) return 'unix';\n  if (navigator.appVersion.includes('Linux')) return 'linux';\n\n  return null;\n};\n\nconst hasNewWindowModifier = (\n  event: KeyboardEvent<HTMLElement> | MouseEvent<HTMLElement>\n) => {\n  const platform = getPlatform();\n  switch (platform) {\n    case 'macos':\n      return event.metaKey;\n    default:\n      return event.ctrlKey;\n  }\n};\n\nconst shakeAnimation = keyframes`\n  from,\n  to {\n    transform: translate3d(0, 0, 0);\n  }\n\n  14%,\n  42%,\n  70% {\n    transform: translate3d(-3px, 0, 0);\n  }\n\n  28%,\n  56%,\n  84% {\n    transform: translate3d(3px, 0, 0);\n  }\n`;\n\ntype State = {\n  hasNetworkError: boolean;\n  isLoading: boolean;\n  searchText: SearchText;\n  selectedResult: SelectedResult;\n  // Used for UX when browsing through history\n  enableHistory: boolean;\n  results: Command[];\n  stack: Stack[];\n};\ntype Action =\n  | { type: 'networkError'; payload: boolean }\n  | { type: 'loading'; payload: boolean }\n  | { type: 'selectedResult'; payload: number }\n  | { type: 'incrementSelectedResult' }\n  | { type: 'decrementSelectedResult' }\n  | {\n      type: 'pickCommandFromHistory';\n      payload: { searchText: SearchText; results: Command[] };\n    }\n  | { type: 'setNextCommands'; payload: { results: Command[] } }\n  | {\n      type: 'setPrevCommands';\n      payload: { searchText: SearchText; results: Command[] };\n    }\n  | { type: 'searchText'; payload: string }\n  | { type: 'setSearchTextResults'; payload: Command[] }\n  | { type: 'resetSearchText' }\n  | { type: 'resetResultsWhenClosing' }\n  | { type: 'reset' };\nconst initialState = {\n  hasNetworkError: false,\n  isLoading: false,\n  searchText: '',\n  selectedResult: -1,\n  // Used for UX when browsing through history\n  enableHistory: true,\n  results: [],\n  stack: [],\n};\nconst reducer = (state: State = initialState, action: Action): State => {\n  switch (action.type) {\n    case 'networkError':\n      return { ...state, hasNetworkError: action.payload };\n    case 'loading':\n      return { ...state, isLoading: action.payload };\n    case 'selectedResult':\n      return { ...state, selectedResult: action.payload };\n    case 'incrementSelectedResult':\n      return {\n        ...state,\n        selectedResult:\n          state.selectedResult === state.results.length - 1\n            ? 0\n            : state.selectedResult + 1,\n        enableHistory: false,\n      };\n    case 'decrementSelectedResult':\n      return {\n        ...state,\n        selectedResult:\n          state.selectedResult < 1\n            ? state.results.length - 1\n            : state.selectedResult - 1,\n        enableHistory: false,\n      };\n    case 'pickCommandFromHistory':\n      return {\n        ...state,\n        selectedResult: 0,\n        searchText: action.payload.searchText,\n        results: action.payload.results,\n        stack: [],\n        // The history does not get changed here, it will be changed along\n        // with the regular flow.\n      };\n    case 'setNextCommands':\n      return {\n        ...state,\n        stack: [\n          ...state.stack,\n          {\n            searchText: state.searchText,\n            results: state.results,\n            selectedResult: state.selectedResult,\n          },\n        ],\n        selectedResult: 0,\n        enableHistory: false,\n        results: action.payload.results,\n      };\n    case 'setPrevCommands':\n      return {\n        ...state,\n        searchText: action.payload.searchText,\n        results: action.payload.results,\n        selectedResult: 0,\n        enableHistory: false,\n        // omit last item\n        stack: state.stack.slice(0, -1),\n      };\n    case 'searchText':\n      return {\n        ...state,\n        searchText: action.payload,\n        // clear network error when search text is cleared, so that users\n        // are tempted to retry\n        hasNetworkError: action.payload.length > 0 && state.hasNetworkError,\n      };\n    case 'setSearchTextResults':\n      return {\n        ...state,\n        results: action.payload,\n        selectedResult: action.payload.length > 0 ? 0 : -1,\n        enableHistory: true,\n        stack: [],\n      };\n    case 'resetSearchText':\n      return { ...state, searchText: '', results: [], selectedResult: -1 };\n    case 'resetResultsWhenClosing':\n      return { ...state, selectedResult: -1, enableHistory: true };\n    case 'reset':\n      return initialState;\n    default:\n      return state;\n  }\n};\n\ntype Props = {\n  historyEntries: HistoryEntry[];\n  onHistoryEntriesChange: (historyEntries: HistoryEntry[]) => void;\n  search: (searchText: SearchText) => Promise<Command[]>;\n  getNextCommands: (command: Command) => Promise<Command[]>;\n  executeCommand: (command: Command, meta: { openInNewTab: boolean }) => void;\n  onClose: () => void;\n  classNameShakeAnimation: string;\n};\nconst Butler = (props: Props) => {\n  const intl = useIntl();\n  const [state, dispatch] = useReducer(reducer, initialState);\n\n  const shouldSelectFieldText = useRef(false);\n  const isNewWindowCombo = useRef(false);\n  const skipNextSelection = useRef(false);\n  const searchContainerRef = useRef<HTMLDivElement>(null);\n  const searchInputRef = useRef<HTMLInputElement>(null);\n\n  const setHasNetworkError = useCallback(() => {\n    dispatch({ type: 'networkError', payload: true });\n  }, []);\n  const unsetHasNetworkError = useCallback(() => {\n    dispatch({ type: 'networkError', payload: false });\n  }, []);\n  const setIsLoading = useCallback(() => {\n    dispatch({ type: 'loading', payload: true });\n  }, []);\n  const unsetIsLoading = useCallback(() => {\n    dispatch({ type: 'loading', payload: false });\n  }, []);\n\n  // Destructure functions from props to reference them in the hook dependency list\n  const {\n    search: searchFromParent,\n    onClose: onCloseFromParent,\n    executeCommand: executeCommandFromParent,\n    onHistoryEntriesChange: onHistoryEntriesChangeFromParent,\n    getNextCommands: getNextCommandsFromParent,\n  } = props;\n\n  const shake = useCallback(() => {\n    if (searchContainerRef.current) {\n      searchContainerRef.current.classList.remove(\n        props.classNameShakeAnimation\n      );\n      // -> triggering reflow\n      // eslint-disable-next-line no-void\n      void searchContainerRef.current.offsetWidth;\n      searchContainerRef.current.classList.add(props.classNameShakeAnimation);\n    }\n  }, [props.classNameShakeAnimation]);\n\n  const execute = useCallback(\n    (command: Command, meta: { openInNewTab: boolean }) => {\n      // Only main entries get added to history, so when a subcommand is executed,\n      // we add the main command of it to the history (the top-level command).\n      //\n      // The key to identify history entries by is always the searchText\n      // There will never be two history entries with the same searchText\n      const entry =\n        state.stack.length === 0\n          ? // The stack is empty, so we are executing a top-level command\n            { searchText: state.searchText, results: state.results }\n          : // We are executing a subcommand, so we get the top-level command for it,\n            // which is at the bottom of the stack.\n            {\n              searchText: state.stack[0].searchText,\n              results: state.stack[0].results,\n            };\n\n      // Add the entry to the history, while excluding any earlier history entry\n      // with the same search text. This effectively \"moves\" that entry to the\n      // top of the history (with the most recent results), or appends a new entry\n      // when it didn't exist before.\n      onHistoryEntriesChangeFromParent([\n        ...props.historyEntries.filter(\n          (command) => command.searchText !== entry.searchText\n        ),\n        entry,\n      ]);\n\n      dispatch({ type: 'resetSearchText' });\n\n      onCloseFromParent();\n\n      executeCommandFromParent(command, meta);\n    },\n    [\n      executeCommandFromParent,\n      onCloseFromParent,\n      onHistoryEntriesChangeFromParent,\n      props.historyEntries,\n      state.results,\n      state.searchText,\n      state.stack,\n    ]\n  );\n  const handleKeyDown = useCallback<KeyboardEventHandler<HTMLInputElement>>(\n    (event) => {\n      // Preventing cursor jumps can only happen in onKeyDown, but not in onKeyUp\n      event.persist();\n\n      // We want to know when the user presses cmd+enter (cmd being a meta key).\n      // We are only told about this in keyDown, but not in keyUp, so we need\n      // to handle it here\n      if (event.key === 'Enter' && hasNewWindowModifier(event)) {\n        isNewWindowCombo.current = true;\n        return;\n      }\n\n      // Avoid selecting the whole page when user selectes everything with\n      // a keyboard shortcut. There is probably a better way to do this though.\n      // This prevents the whole page from being selected in case the user\n      // 1) opens the search box\n      // 2) types into it\n      // 3) selects all text using cmd+a\n      // 4) closes the search box with esc\n      // Without this handling, the whole page would now be selected\n      if (isSelectAllCombo(event)) {\n        // This stops the browser from selecting anything\n        event.preventDefault();\n        if (searchInputRef.current) {\n          // This selects the text in the search input\n          searchInputRef.current.setSelectionRange(0, state.searchText.length);\n        }\n        return;\n      }\n\n      // avoid interfering with other key combinations using modifier keys\n      if (event.ctrlKey || event.altKey || event.shiftKey || event.metaKey)\n        return;\n      if (isCloseCombo(event)) return;\n\n      // skip next mouseEnter to avoid setting selectedResult when cursor just\n      // happens to be where the results will pop up\n      skipNextSelection.current = true;\n\n      if (event.key === 'ArrowDown') {\n        // prevent cursor from jumping to end of text input\n        event.preventDefault();\n        dispatch({ type: 'incrementSelectedResult' });\n        return;\n      }\n      if (event.key === 'ArrowUp') {\n        // browse through history\n        if (\n          state.searchText.length === 0 ||\n          (state.selectedResult < 1 && state.enableHistory)\n        ) {\n          shouldSelectFieldText.current = true;\n          const selectedIndex =\n            state.searchText.length === 0\n              ? // When going back the first step\n                -1\n              : // When going back more than one step\n                props.historyEntries.findIndex(\n                  (command) => command.searchText === state.searchText\n                );\n          // Pick the previous command from the history\n          const prevCommand =\n            selectedIndex === -1\n              ? // previous command on top of the history when going back on\n                // first step\n                last(props.historyEntries)\n              : // previous command is deeper down\n                // When the history does not exist (negative index), then\n                // this implicitly returns undefined\n                props.historyEntries[selectedIndex - 1];\n          // Skip when no previous entry exists in the history\n          if (!prevCommand) return;\n          dispatch({\n            type: 'pickCommandFromHistory',\n            payload: {\n              searchText: prevCommand.searchText,\n              results: prevCommand.results,\n            },\n          });\n          return;\n        }\n        // prevent cursor from jumping to beginning of text input\n        event.preventDefault();\n        dispatch({ type: 'decrementSelectedResult' });\n        return;\n      }\n      if (state.selectedResult > -1) {\n        if (event.key === 'ArrowRight') {\n          const command = state.results[state.selectedResult];\n          const searchText = state.searchText;\n          const isCursorAtEnd =\n            searchInputRef.current &&\n            state.searchText.length === searchInputRef.current.selectionStart;\n\n          const isEverythingSelected =\n            searchInputRef.current &&\n            searchInputRef.current.selectionStart === 0 &&\n            state.searchText.length === searchInputRef.current.selectionEnd;\n\n          // only allow diving in when cursor is at end of input or when\n          // the complete text is selected (when browsing through history)\n          if (!isCursorAtEnd && !isEverythingSelected) return;\n\n          unsetHasNetworkError();\n\n          // NOTE: since we need to fetch the \"next command\", which is an async operation,\n          // we use a IIFE to process that and eventually update the state.\n          (async () => {\n            if (command) {\n              const nextCommands = await getNextCommandsFromParent(command);\n              // avoid moving cursor when there are sub-options\n              if (nextCommands.length > 0) {\n                // Ensure the search text has not changed while we were loading\n                // the next results, otherwise we'd interrupt the user.\n                // Throw away the results in case the search text has changed.\n                if (state.searchText === searchText) {\n                  dispatch({\n                    type: 'setNextCommands',\n                    payload: { results: nextCommands },\n                  });\n                }\n                return;\n              }\n            }\n            shake();\n          })();\n          return;\n        }\n        if (event.key === 'ArrowLeft') {\n          // go left in stack\n          const prevCommand = last(state.stack);\n\n          // do nothing when we can't go left anymore\n          if (!prevCommand) return;\n\n          // prevent cursor from jumping a char to the left in text input\n          event.preventDefault();\n\n          dispatch({\n            type: 'setPrevCommands',\n            payload: {\n              searchText: prevCommand.searchText,\n              results: prevCommand.results,\n            },\n          });\n          return;\n        }\n      }\n    },\n    [\n      getNextCommandsFromParent,\n      props.historyEntries,\n      shake,\n      state,\n      unsetHasNetworkError,\n    ]\n  );\n  const handleKeyUp = useCallback<KeyboardEventHandler<HTMLInputElement>>(\n    (event) => {\n      // setting the selection can only happen in onKeyUp\n      if (shouldSelectFieldText.current) {\n        const input = event.target as HTMLInputElement;\n        input.focus();\n        input.select();\n        shouldSelectFieldText.current = false;\n      }\n\n      if (event.key !== 'Enter' && !isNewWindowCombo.current) return true;\n\n      // User just triggered the search\n      if (state.selectedResult === -1) return true;\n\n      // User had something selected and wants to go there\n      execute(state.results[state.selectedResult], {\n        openInNewTab: isNewWindowCombo.current,\n      });\n\n      isNewWindowCombo.current = false;\n      return true;\n    },\n    [execute, state.results, state.selectedResult]\n  );\n  const handleChange = useCallback<ChangeEventHandler<HTMLInputElement>>(\n    (event) => {\n      const searchText = event.target.value;\n      if (searchText.trim().length === 0) {\n        dispatch({ type: 'reset' });\n        return;\n      }\n\n      dispatch({ type: 'searchText', payload: searchText });\n\n      // A search via network is only triggered when there\n      // are more than three characters. So no false loading\n      // indication is given.\n      if (searchText.trim().length > 3) {\n        setIsLoading();\n      }\n\n      searchFromParent(searchText).then(\n        (asyncResults: Command[]) => {\n          unsetHasNetworkError();\n          unsetIsLoading();\n\n          const fuse = new Fuse(asyncResults, {\n            keys: [\n              { name: 'text', weight: 0.6 },\n              { name: 'keywords', weight: 0.4 },\n            ],\n            minMatchCharLength: 2,\n            includeScore: true,\n          });\n\n          const searchResults = fuse\n            .search(searchText)\n            // Filter out results with a matching score over 0.75\n            .filter((result) => (result.score ? result.score < 0.75 : false))\n            // Keep a maximal of 9 results\n            .slice(0, 9);\n\n          dispatch({\n            type: 'setSearchTextResults',\n            payload: searchResults.map((result) => result.item),\n          });\n        },\n        (error: Error) => {\n          // eslint-disable-next-line no-console\n          if (process.env.NODE_ENV !== 'production') console.error(error);\n          unsetIsLoading();\n          setHasNetworkError();\n        }\n      );\n    },\n    [\n      searchFromParent,\n      setHasNetworkError,\n      setIsLoading,\n      unsetHasNetworkError,\n      unsetIsLoading,\n    ]\n  );\n  const handleContainerClick = useCallback(() => {\n    dispatch({ type: 'resetResultsWhenClosing' });\n    onCloseFromParent();\n  }, [onCloseFromParent]);\n\n  const createCommandMouseEnterHandler = useCallback<\n    (index: number) => MouseEventHandler<HTMLDivElement>\n  >(\n    (index) => () => {\n      // In case the cursor happened to be in a location where a\n      // result would appear, it would trigger onMouseEnter and the\n      // result would be selected immediately. This is not something\n      // a user would expect, hence we prevent it from happening.\n      // The user has to move the cursor to an option explicitly for\n      // it to become active. However, the user can always click and\n      // that action will be triggered.\n      if (skipNextSelection.current) {\n        skipNextSelection.current = false;\n        return;\n      }\n\n      // sets the selected result, mainly for the hover effect\n      dispatch({ type: 'selectedResult', payload: index });\n    },\n    []\n  );\n  const createCommandClickHandler = useCallback<\n    (command: Command) => MouseEventHandler<HTMLDivElement>\n  >(\n    (command) => (event) => {\n      execute(command, {\n        openInNewTab: hasNewWindowModifier(event),\n      });\n    },\n    [execute]\n  );\n\n  return (\n    <ButlerContainer\n      onClick={handleContainerClick}\n      data-testid=\"quick-access\"\n      tabIndex={-1}\n    >\n      <div\n        ref={searchContainerRef}\n        css={css`\n          background-color: ${uiKitDesignTokens.colorSurface};\n          border: 0;\n          border-radius: ${uiKitDesignTokens.borderRadius4};\n          min-height: 40px;\n\n          /* one more than app-bar (20000) and one more than the overlay (20001) */\n          z-index: 20002;\n          width: 400px;\n          margin: 40px auto;\n          overflow: hidden;\n          -webkit-box-shadow: 0 10px 30px -8px rgba(0, 0, 0, 0.75);\n          -moz-box-shadow: 0 10px 30px -8px rgba(0, 0, 0, 0.75);\n          box-shadow: 0 10px 30px -8px rgba(0, 0, 0, 0.75);\n          padding-bottom: ${state.hasNetworkError\n            ? '0'\n            : uiKitDesignTokens.spacingS};\n        `}\n        onClick={(event) => {\n          // Avoid closing when the searchContainer itself is clicked\n          // If we don't do this, then the overlay will close when e.g.\n          // the search input is clicked.\n          event.stopPropagation();\n          event.preventDefault();\n        }}\n      >\n        <div\n          css={css`\n            display: flex;\n          `}\n        >\n          <label\n            htmlFor=\"quick-access-search-input\"\n            css={css`\n              align-self: center;\n              padding-left: ${uiKitDesignTokens.spacingM};\n              margin-top: ${uiKitDesignTokens.spacingS};\n            `}\n          >\n            <SearchIcon color=\"neutral60\" />\n          </label>\n          <input\n            id=\"quick-access-search-input\"\n            ref={searchInputRef}\n            placeholder={intl.formatMessage(messages.inputPlacehoder)}\n            type=\"text\"\n            css={css`\n              width: 100%;\n              border: 0;\n              outline: 0;\n              font-size: 22px;\n              font-weight: 300;\n              padding: ${uiKitDesignTokens.spacingM}\n                ${uiKitDesignTokens.spacingM} ${uiKitDesignTokens.spacingS}\n                ${uiKitDesignTokens.spacingS};\n              &::placeholder {\n                color: ${uiKitDesignTokens.colorNeutral60};\n              }\n            `}\n            value={state.searchText}\n            onChange={handleChange}\n            onKeyDown={handleKeyDown}\n            onKeyUp={handleKeyUp}\n            autoFocus={true}\n            autoComplete=\"off\"\n            data-testid=\"quick-access-search-input\"\n          />\n          {state.isLoading && (\n            <div\n              css={css`\n                align-self: center;\n                margin-top: ${uiKitDesignTokens.spacingS};\n                margin-right: ${uiKitDesignTokens.spacingS};\n              `}\n            >\n              <LoadingSpinner />\n            </div>\n          )}\n        </div>\n        {(() => {\n          if (state.hasNetworkError)\n            return (\n              <div\n                css={css`\n                  overflow: hidden;\n                  white-space: nowrap;\n                  cursor: default;\n                  background: ${uiKitDesignTokens.colorError};\n                  text-align: center;\n                  text-transform: uppercase;\n                  color: ${uiKitDesignTokens.colorSurface};\n                  font-size: ${uiKitDesignTokens.fontSize20};\n                  padding: ${uiKitDesignTokens.spacingXs};\n                `}\n              >\n                <FormattedMessage {...messages.offline} />\n              </div>\n            );\n\n          if (state.results.length === 0 && state.searchText.trim().length > 0)\n            return (\n              <div\n                css={css`\n                  overflow: hidden;\n                  white-space: nowrap;\n                  cursor: default;\n                  background: ${uiKitDesignTokens.colorNeutral};\n                  color: ${uiKitDesignTokens.colorSolid};\n                  text-align: center;\n                  text-transform: uppercase;\n                  font-size: ${uiKitDesignTokens.fontSize20};\n                  padding: ${uiKitDesignTokens.spacingXs};\n                `}\n              >\n                <FormattedMessage {...messages.noResults} />\n              </div>\n            );\n\n          return state.results.map((command, index) => (\n            <ButlerCommand\n              key={command.id}\n              command={command}\n              isSelected={state.selectedResult === index}\n              onMouseEnter={createCommandMouseEnterHandler(index)}\n              onClick={createCommandClickHandler(command)}\n            />\n          ));\n        })()}\n      </div>\n    </ButlerContainer>\n  );\n};\nButler.displayName = 'Butler';\n\nconst ButlerWithAnimation = (props: Omit<Props, 'classNameShakeAnimation'>) => (\n  <ClassNames>\n    {({ css }) => (\n      <Butler\n        {...props}\n        classNameShakeAnimation={css`\n          animation-duration: 0.45s;\n          animation-fill-mode: both;\n          animation-name: ${shakeAnimation};\n        `}\n      />\n    )}\n  </ClassNames>\n);\n\nexport default ButlerWithAnimation;\n"]} */"),
873
- value: state.searchText,
874
- onChange: handleChange,
875
- onKeyDown: handleKeyDown,
876
- onKeyUp: handleKeyUp,
877
- autoFocus: true,
878
- autoComplete: "off",
879
- "data-testid": "quick-access-search-input"
880
- }), state.isLoading && jsx("div", {
881
- css: /*#__PURE__*/css("align-self:center;margin-top:", designTokens.spacingS, ";margin-right:", designTokens.spacingS, ";" + (process.env.NODE_ENV === "production" ? "" : ";label:Butler;"), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["butler.tsx"],"names":[],"mappings":"AAwpBsB","file":"butler.tsx","sourcesContent":["import {\n  KeyboardEventHandler,\n  ChangeEventHandler,\n  KeyboardEvent,\n  MouseEventHandler,\n  MouseEvent,\n  useReducer,\n  useRef,\n  useCallback,\n} from 'react';\nimport { css, keyframes, ClassNames } from '@emotion/react';\nimport Fuse from 'fuse.js';\nimport last from 'lodash/last';\nimport { FormattedMessage, useIntl } from 'react-intl';\nimport { designTokens as uiKitDesignTokens } from '@commercetools-uikit/design-system';\nimport { SearchIcon } from '@commercetools-uikit/icons';\nimport LoadingSpinner from '@commercetools-uikit/loading-spinner';\nimport ButlerCommand from '../butler-command';\nimport ButlerContainer from '../butler-container';\nimport messages from '../messages';\nimport type {\n  Command,\n  SearchText,\n  SelectedResult,\n  Stack,\n  HistoryEntry,\n} from '../types';\n\nconst isSelectAllCombo = (event: KeyboardEvent<HTMLInputElement>) =>\n  event.key === 'a' &&\n  event.metaKey &&\n  !event.ctrlKey &&\n  !event.altKey &&\n  !event.shiftKey;\n\nconst isCloseCombo = (event: KeyboardEvent<HTMLInputElement>) =>\n  event.key === 'Escape' &&\n  !event.metaKey &&\n  !event.ctrlKey &&\n  !event.altKey &&\n  !event.shiftKey;\n\nconst getPlatform = () => {\n  if (navigator.appVersion.includes('Win')) return 'windows';\n  if (navigator.appVersion.includes('Mac')) return 'macos';\n  if (navigator.appVersion.includes('X11')) return 'unix';\n  if (navigator.appVersion.includes('Linux')) return 'linux';\n\n  return null;\n};\n\nconst hasNewWindowModifier = (\n  event: KeyboardEvent<HTMLElement> | MouseEvent<HTMLElement>\n) => {\n  const platform = getPlatform();\n  switch (platform) {\n    case 'macos':\n      return event.metaKey;\n    default:\n      return event.ctrlKey;\n  }\n};\n\nconst shakeAnimation = keyframes`\n  from,\n  to {\n    transform: translate3d(0, 0, 0);\n  }\n\n  14%,\n  42%,\n  70% {\n    transform: translate3d(-3px, 0, 0);\n  }\n\n  28%,\n  56%,\n  84% {\n    transform: translate3d(3px, 0, 0);\n  }\n`;\n\ntype State = {\n  hasNetworkError: boolean;\n  isLoading: boolean;\n  searchText: SearchText;\n  selectedResult: SelectedResult;\n  // Used for UX when browsing through history\n  enableHistory: boolean;\n  results: Command[];\n  stack: Stack[];\n};\ntype Action =\n  | { type: 'networkError'; payload: boolean }\n  | { type: 'loading'; payload: boolean }\n  | { type: 'selectedResult'; payload: number }\n  | { type: 'incrementSelectedResult' }\n  | { type: 'decrementSelectedResult' }\n  | {\n      type: 'pickCommandFromHistory';\n      payload: { searchText: SearchText; results: Command[] };\n    }\n  | { type: 'setNextCommands'; payload: { results: Command[] } }\n  | {\n      type: 'setPrevCommands';\n      payload: { searchText: SearchText; results: Command[] };\n    }\n  | { type: 'searchText'; payload: string }\n  | { type: 'setSearchTextResults'; payload: Command[] }\n  | { type: 'resetSearchText' }\n  | { type: 'resetResultsWhenClosing' }\n  | { type: 'reset' };\nconst initialState = {\n  hasNetworkError: false,\n  isLoading: false,\n  searchText: '',\n  selectedResult: -1,\n  // Used for UX when browsing through history\n  enableHistory: true,\n  results: [],\n  stack: [],\n};\nconst reducer = (state: State = initialState, action: Action): State => {\n  switch (action.type) {\n    case 'networkError':\n      return { ...state, hasNetworkError: action.payload };\n    case 'loading':\n      return { ...state, isLoading: action.payload };\n    case 'selectedResult':\n      return { ...state, selectedResult: action.payload };\n    case 'incrementSelectedResult':\n      return {\n        ...state,\n        selectedResult:\n          state.selectedResult === state.results.length - 1\n            ? 0\n            : state.selectedResult + 1,\n        enableHistory: false,\n      };\n    case 'decrementSelectedResult':\n      return {\n        ...state,\n        selectedResult:\n          state.selectedResult < 1\n            ? state.results.length - 1\n            : state.selectedResult - 1,\n        enableHistory: false,\n      };\n    case 'pickCommandFromHistory':\n      return {\n        ...state,\n        selectedResult: 0,\n        searchText: action.payload.searchText,\n        results: action.payload.results,\n        stack: [],\n        // The history does not get changed here, it will be changed along\n        // with the regular flow.\n      };\n    case 'setNextCommands':\n      return {\n        ...state,\n        stack: [\n          ...state.stack,\n          {\n            searchText: state.searchText,\n            results: state.results,\n            selectedResult: state.selectedResult,\n          },\n        ],\n        selectedResult: 0,\n        enableHistory: false,\n        results: action.payload.results,\n      };\n    case 'setPrevCommands':\n      return {\n        ...state,\n        searchText: action.payload.searchText,\n        results: action.payload.results,\n        selectedResult: 0,\n        enableHistory: false,\n        // omit last item\n        stack: state.stack.slice(0, -1),\n      };\n    case 'searchText':\n      return {\n        ...state,\n        searchText: action.payload,\n        // clear network error when search text is cleared, so that users\n        // are tempted to retry\n        hasNetworkError: action.payload.length > 0 && state.hasNetworkError,\n      };\n    case 'setSearchTextResults':\n      return {\n        ...state,\n        results: action.payload,\n        selectedResult: action.payload.length > 0 ? 0 : -1,\n        enableHistory: true,\n        stack: [],\n      };\n    case 'resetSearchText':\n      return { ...state, searchText: '', results: [], selectedResult: -1 };\n    case 'resetResultsWhenClosing':\n      return { ...state, selectedResult: -1, enableHistory: true };\n    case 'reset':\n      return initialState;\n    default:\n      return state;\n  }\n};\n\ntype Props = {\n  historyEntries: HistoryEntry[];\n  onHistoryEntriesChange: (historyEntries: HistoryEntry[]) => void;\n  search: (searchText: SearchText) => Promise<Command[]>;\n  getNextCommands: (command: Command) => Promise<Command[]>;\n  executeCommand: (command: Command, meta: { openInNewTab: boolean }) => void;\n  onClose: () => void;\n  classNameShakeAnimation: string;\n};\nconst Butler = (props: Props) => {\n  const intl = useIntl();\n  const [state, dispatch] = useReducer(reducer, initialState);\n\n  const shouldSelectFieldText = useRef(false);\n  const isNewWindowCombo = useRef(false);\n  const skipNextSelection = useRef(false);\n  const searchContainerRef = useRef<HTMLDivElement>(null);\n  const searchInputRef = useRef<HTMLInputElement>(null);\n\n  const setHasNetworkError = useCallback(() => {\n    dispatch({ type: 'networkError', payload: true });\n  }, []);\n  const unsetHasNetworkError = useCallback(() => {\n    dispatch({ type: 'networkError', payload: false });\n  }, []);\n  const setIsLoading = useCallback(() => {\n    dispatch({ type: 'loading', payload: true });\n  }, []);\n  const unsetIsLoading = useCallback(() => {\n    dispatch({ type: 'loading', payload: false });\n  }, []);\n\n  // Destructure functions from props to reference them in the hook dependency list\n  const {\n    search: searchFromParent,\n    onClose: onCloseFromParent,\n    executeCommand: executeCommandFromParent,\n    onHistoryEntriesChange: onHistoryEntriesChangeFromParent,\n    getNextCommands: getNextCommandsFromParent,\n  } = props;\n\n  const shake = useCallback(() => {\n    if (searchContainerRef.current) {\n      searchContainerRef.current.classList.remove(\n        props.classNameShakeAnimation\n      );\n      // -> triggering reflow\n      // eslint-disable-next-line no-void\n      void searchContainerRef.current.offsetWidth;\n      searchContainerRef.current.classList.add(props.classNameShakeAnimation);\n    }\n  }, [props.classNameShakeAnimation]);\n\n  const execute = useCallback(\n    (command: Command, meta: { openInNewTab: boolean }) => {\n      // Only main entries get added to history, so when a subcommand is executed,\n      // we add the main command of it to the history (the top-level command).\n      //\n      // The key to identify history entries by is always the searchText\n      // There will never be two history entries with the same searchText\n      const entry =\n        state.stack.length === 0\n          ? // The stack is empty, so we are executing a top-level command\n            { searchText: state.searchText, results: state.results }\n          : // We are executing a subcommand, so we get the top-level command for it,\n            // which is at the bottom of the stack.\n            {\n              searchText: state.stack[0].searchText,\n              results: state.stack[0].results,\n            };\n\n      // Add the entry to the history, while excluding any earlier history entry\n      // with the same search text. This effectively \"moves\" that entry to the\n      // top of the history (with the most recent results), or appends a new entry\n      // when it didn't exist before.\n      onHistoryEntriesChangeFromParent([\n        ...props.historyEntries.filter(\n          (command) => command.searchText !== entry.searchText\n        ),\n        entry,\n      ]);\n\n      dispatch({ type: 'resetSearchText' });\n\n      onCloseFromParent();\n\n      executeCommandFromParent(command, meta);\n    },\n    [\n      executeCommandFromParent,\n      onCloseFromParent,\n      onHistoryEntriesChangeFromParent,\n      props.historyEntries,\n      state.results,\n      state.searchText,\n      state.stack,\n    ]\n  );\n  const handleKeyDown = useCallback<KeyboardEventHandler<HTMLInputElement>>(\n    (event) => {\n      // Preventing cursor jumps can only happen in onKeyDown, but not in onKeyUp\n      event.persist();\n\n      // We want to know when the user presses cmd+enter (cmd being a meta key).\n      // We are only told about this in keyDown, but not in keyUp, so we need\n      // to handle it here\n      if (event.key === 'Enter' && hasNewWindowModifier(event)) {\n        isNewWindowCombo.current = true;\n        return;\n      }\n\n      // Avoid selecting the whole page when user selectes everything with\n      // a keyboard shortcut. There is probably a better way to do this though.\n      // This prevents the whole page from being selected in case the user\n      // 1) opens the search box\n      // 2) types into it\n      // 3) selects all text using cmd+a\n      // 4) closes the search box with esc\n      // Without this handling, the whole page would now be selected\n      if (isSelectAllCombo(event)) {\n        // This stops the browser from selecting anything\n        event.preventDefault();\n        if (searchInputRef.current) {\n          // This selects the text in the search input\n          searchInputRef.current.setSelectionRange(0, state.searchText.length);\n        }\n        return;\n      }\n\n      // avoid interfering with other key combinations using modifier keys\n      if (event.ctrlKey || event.altKey || event.shiftKey || event.metaKey)\n        return;\n      if (isCloseCombo(event)) return;\n\n      // skip next mouseEnter to avoid setting selectedResult when cursor just\n      // happens to be where the results will pop up\n      skipNextSelection.current = true;\n\n      if (event.key === 'ArrowDown') {\n        // prevent cursor from jumping to end of text input\n        event.preventDefault();\n        dispatch({ type: 'incrementSelectedResult' });\n        return;\n      }\n      if (event.key === 'ArrowUp') {\n        // browse through history\n        if (\n          state.searchText.length === 0 ||\n          (state.selectedResult < 1 && state.enableHistory)\n        ) {\n          shouldSelectFieldText.current = true;\n          const selectedIndex =\n            state.searchText.length === 0\n              ? // When going back the first step\n                -1\n              : // When going back more than one step\n                props.historyEntries.findIndex(\n                  (command) => command.searchText === state.searchText\n                );\n          // Pick the previous command from the history\n          const prevCommand =\n            selectedIndex === -1\n              ? // previous command on top of the history when going back on\n                // first step\n                last(props.historyEntries)\n              : // previous command is deeper down\n                // When the history does not exist (negative index), then\n                // this implicitly returns undefined\n                props.historyEntries[selectedIndex - 1];\n          // Skip when no previous entry exists in the history\n          if (!prevCommand) return;\n          dispatch({\n            type: 'pickCommandFromHistory',\n            payload: {\n              searchText: prevCommand.searchText,\n              results: prevCommand.results,\n            },\n          });\n          return;\n        }\n        // prevent cursor from jumping to beginning of text input\n        event.preventDefault();\n        dispatch({ type: 'decrementSelectedResult' });\n        return;\n      }\n      if (state.selectedResult > -1) {\n        if (event.key === 'ArrowRight') {\n          const command = state.results[state.selectedResult];\n          const searchText = state.searchText;\n          const isCursorAtEnd =\n            searchInputRef.current &&\n            state.searchText.length === searchInputRef.current.selectionStart;\n\n          const isEverythingSelected =\n            searchInputRef.current &&\n            searchInputRef.current.selectionStart === 0 &&\n            state.searchText.length === searchInputRef.current.selectionEnd;\n\n          // only allow diving in when cursor is at end of input or when\n          // the complete text is selected (when browsing through history)\n          if (!isCursorAtEnd && !isEverythingSelected) return;\n\n          unsetHasNetworkError();\n\n          // NOTE: since we need to fetch the \"next command\", which is an async operation,\n          // we use a IIFE to process that and eventually update the state.\n          (async () => {\n            if (command) {\n              const nextCommands = await getNextCommandsFromParent(command);\n              // avoid moving cursor when there are sub-options\n              if (nextCommands.length > 0) {\n                // Ensure the search text has not changed while we were loading\n                // the next results, otherwise we'd interrupt the user.\n                // Throw away the results in case the search text has changed.\n                if (state.searchText === searchText) {\n                  dispatch({\n                    type: 'setNextCommands',\n                    payload: { results: nextCommands },\n                  });\n                }\n                return;\n              }\n            }\n            shake();\n          })();\n          return;\n        }\n        if (event.key === 'ArrowLeft') {\n          // go left in stack\n          const prevCommand = last(state.stack);\n\n          // do nothing when we can't go left anymore\n          if (!prevCommand) return;\n\n          // prevent cursor from jumping a char to the left in text input\n          event.preventDefault();\n\n          dispatch({\n            type: 'setPrevCommands',\n            payload: {\n              searchText: prevCommand.searchText,\n              results: prevCommand.results,\n            },\n          });\n          return;\n        }\n      }\n    },\n    [\n      getNextCommandsFromParent,\n      props.historyEntries,\n      shake,\n      state,\n      unsetHasNetworkError,\n    ]\n  );\n  const handleKeyUp = useCallback<KeyboardEventHandler<HTMLInputElement>>(\n    (event) => {\n      // setting the selection can only happen in onKeyUp\n      if (shouldSelectFieldText.current) {\n        const input = event.target as HTMLInputElement;\n        input.focus();\n        input.select();\n        shouldSelectFieldText.current = false;\n      }\n\n      if (event.key !== 'Enter' && !isNewWindowCombo.current) return true;\n\n      // User just triggered the search\n      if (state.selectedResult === -1) return true;\n\n      // User had something selected and wants to go there\n      execute(state.results[state.selectedResult], {\n        openInNewTab: isNewWindowCombo.current,\n      });\n\n      isNewWindowCombo.current = false;\n      return true;\n    },\n    [execute, state.results, state.selectedResult]\n  );\n  const handleChange = useCallback<ChangeEventHandler<HTMLInputElement>>(\n    (event) => {\n      const searchText = event.target.value;\n      if (searchText.trim().length === 0) {\n        dispatch({ type: 'reset' });\n        return;\n      }\n\n      dispatch({ type: 'searchText', payload: searchText });\n\n      // A search via network is only triggered when there\n      // are more than three characters. So no false loading\n      // indication is given.\n      if (searchText.trim().length > 3) {\n        setIsLoading();\n      }\n\n      searchFromParent(searchText).then(\n        (asyncResults: Command[]) => {\n          unsetHasNetworkError();\n          unsetIsLoading();\n\n          const fuse = new Fuse(asyncResults, {\n            keys: [\n              { name: 'text', weight: 0.6 },\n              { name: 'keywords', weight: 0.4 },\n            ],\n            minMatchCharLength: 2,\n            includeScore: true,\n          });\n\n          const searchResults = fuse\n            .search(searchText)\n            // Filter out results with a matching score over 0.75\n            .filter((result) => (result.score ? result.score < 0.75 : false))\n            // Keep a maximal of 9 results\n            .slice(0, 9);\n\n          dispatch({\n            type: 'setSearchTextResults',\n            payload: searchResults.map((result) => result.item),\n          });\n        },\n        (error: Error) => {\n          // eslint-disable-next-line no-console\n          if (process.env.NODE_ENV !== 'production') console.error(error);\n          unsetIsLoading();\n          setHasNetworkError();\n        }\n      );\n    },\n    [\n      searchFromParent,\n      setHasNetworkError,\n      setIsLoading,\n      unsetHasNetworkError,\n      unsetIsLoading,\n    ]\n  );\n  const handleContainerClick = useCallback(() => {\n    dispatch({ type: 'resetResultsWhenClosing' });\n    onCloseFromParent();\n  }, [onCloseFromParent]);\n\n  const createCommandMouseEnterHandler = useCallback<\n    (index: number) => MouseEventHandler<HTMLDivElement>\n  >(\n    (index) => () => {\n      // In case the cursor happened to be in a location where a\n      // result would appear, it would trigger onMouseEnter and the\n      // result would be selected immediately. This is not something\n      // a user would expect, hence we prevent it from happening.\n      // The user has to move the cursor to an option explicitly for\n      // it to become active. However, the user can always click and\n      // that action will be triggered.\n      if (skipNextSelection.current) {\n        skipNextSelection.current = false;\n        return;\n      }\n\n      // sets the selected result, mainly for the hover effect\n      dispatch({ type: 'selectedResult', payload: index });\n    },\n    []\n  );\n  const createCommandClickHandler = useCallback<\n    (command: Command) => MouseEventHandler<HTMLDivElement>\n  >(\n    (command) => (event) => {\n      execute(command, {\n        openInNewTab: hasNewWindowModifier(event),\n      });\n    },\n    [execute]\n  );\n\n  return (\n    <ButlerContainer\n      onClick={handleContainerClick}\n      data-testid=\"quick-access\"\n      tabIndex={-1}\n    >\n      <div\n        ref={searchContainerRef}\n        css={css`\n          background-color: ${uiKitDesignTokens.colorSurface};\n          border: 0;\n          border-radius: ${uiKitDesignTokens.borderRadius4};\n          min-height: 40px;\n\n          /* one more than app-bar (20000) and one more than the overlay (20001) */\n          z-index: 20002;\n          width: 400px;\n          margin: 40px auto;\n          overflow: hidden;\n          -webkit-box-shadow: 0 10px 30px -8px rgba(0, 0, 0, 0.75);\n          -moz-box-shadow: 0 10px 30px -8px rgba(0, 0, 0, 0.75);\n          box-shadow: 0 10px 30px -8px rgba(0, 0, 0, 0.75);\n          padding-bottom: ${state.hasNetworkError\n            ? '0'\n            : uiKitDesignTokens.spacingS};\n        `}\n        onClick={(event) => {\n          // Avoid closing when the searchContainer itself is clicked\n          // If we don't do this, then the overlay will close when e.g.\n          // the search input is clicked.\n          event.stopPropagation();\n          event.preventDefault();\n        }}\n      >\n        <div\n          css={css`\n            display: flex;\n          `}\n        >\n          <label\n            htmlFor=\"quick-access-search-input\"\n            css={css`\n              align-self: center;\n              padding-left: ${uiKitDesignTokens.spacingM};\n              margin-top: ${uiKitDesignTokens.spacingS};\n            `}\n          >\n            <SearchIcon color=\"neutral60\" />\n          </label>\n          <input\n            id=\"quick-access-search-input\"\n            ref={searchInputRef}\n            placeholder={intl.formatMessage(messages.inputPlacehoder)}\n            type=\"text\"\n            css={css`\n              width: 100%;\n              border: 0;\n              outline: 0;\n              font-size: 22px;\n              font-weight: 300;\n              padding: ${uiKitDesignTokens.spacingM}\n                ${uiKitDesignTokens.spacingM} ${uiKitDesignTokens.spacingS}\n                ${uiKitDesignTokens.spacingS};\n              &::placeholder {\n                color: ${uiKitDesignTokens.colorNeutral60};\n              }\n            `}\n            value={state.searchText}\n            onChange={handleChange}\n            onKeyDown={handleKeyDown}\n            onKeyUp={handleKeyUp}\n            autoFocus={true}\n            autoComplete=\"off\"\n            data-testid=\"quick-access-search-input\"\n          />\n          {state.isLoading && (\n            <div\n              css={css`\n                align-self: center;\n                margin-top: ${uiKitDesignTokens.spacingS};\n                margin-right: ${uiKitDesignTokens.spacingS};\n              `}\n            >\n              <LoadingSpinner />\n            </div>\n          )}\n        </div>\n        {(() => {\n          if (state.hasNetworkError)\n            return (\n              <div\n                css={css`\n                  overflow: hidden;\n                  white-space: nowrap;\n                  cursor: default;\n                  background: ${uiKitDesignTokens.colorError};\n                  text-align: center;\n                  text-transform: uppercase;\n                  color: ${uiKitDesignTokens.colorSurface};\n                  font-size: ${uiKitDesignTokens.fontSize20};\n                  padding: ${uiKitDesignTokens.spacingXs};\n                `}\n              >\n                <FormattedMessage {...messages.offline} />\n              </div>\n            );\n\n          if (state.results.length === 0 && state.searchText.trim().length > 0)\n            return (\n              <div\n                css={css`\n                  overflow: hidden;\n                  white-space: nowrap;\n                  cursor: default;\n                  background: ${uiKitDesignTokens.colorNeutral};\n                  color: ${uiKitDesignTokens.colorSolid};\n                  text-align: center;\n                  text-transform: uppercase;\n                  font-size: ${uiKitDesignTokens.fontSize20};\n                  padding: ${uiKitDesignTokens.spacingXs};\n                `}\n              >\n                <FormattedMessage {...messages.noResults} />\n              </div>\n            );\n\n          return state.results.map((command, index) => (\n            <ButlerCommand\n              key={command.id}\n              command={command}\n              isSelected={state.selectedResult === index}\n              onMouseEnter={createCommandMouseEnterHandler(index)}\n              onClick={createCommandClickHandler(command)}\n            />\n          ));\n        })()}\n      </div>\n    </ButlerContainer>\n  );\n};\nButler.displayName = 'Butler';\n\nconst ButlerWithAnimation = (props: Omit<Props, 'classNameShakeAnimation'>) => (\n  <ClassNames>\n    {({ css }) => (\n      <Butler\n        {...props}\n        classNameShakeAnimation={css`\n          animation-duration: 0.45s;\n          animation-fill-mode: both;\n          animation-name: ${shakeAnimation};\n        `}\n      />\n    )}\n  </ClassNames>\n);\n\nexport default ButlerWithAnimation;\n"]} */"),
882
- children: jsx(LoadingSpinner, {})
883
- })]
884
- }), ((_context0, _context1) => {
885
- if (state.hasNetworkError) return jsx("div", {
886
- css: /*#__PURE__*/css("overflow:hidden;white-space:nowrap;cursor:default;background:", designTokens.colorError, ";text-align:center;text-transform:uppercase;color:", designTokens.colorSurface, ";font-size:", designTokens.fontSize20, ";padding:", designTokens.spacingXs, ";" + (process.env.NODE_ENV === "production" ? "" : ";label:Butler;"), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["butler.tsx"],"names":[],"mappings":"AAsqBwB","file":"butler.tsx","sourcesContent":["import {\n  KeyboardEventHandler,\n  ChangeEventHandler,\n  KeyboardEvent,\n  MouseEventHandler,\n  MouseEvent,\n  useReducer,\n  useRef,\n  useCallback,\n} from 'react';\nimport { css, keyframes, ClassNames } from '@emotion/react';\nimport Fuse from 'fuse.js';\nimport last from 'lodash/last';\nimport { FormattedMessage, useIntl } from 'react-intl';\nimport { designTokens as uiKitDesignTokens } from '@commercetools-uikit/design-system';\nimport { SearchIcon } from '@commercetools-uikit/icons';\nimport LoadingSpinner from '@commercetools-uikit/loading-spinner';\nimport ButlerCommand from '../butler-command';\nimport ButlerContainer from '../butler-container';\nimport messages from '../messages';\nimport type {\n  Command,\n  SearchText,\n  SelectedResult,\n  Stack,\n  HistoryEntry,\n} from '../types';\n\nconst isSelectAllCombo = (event: KeyboardEvent<HTMLInputElement>) =>\n  event.key === 'a' &&\n  event.metaKey &&\n  !event.ctrlKey &&\n  !event.altKey &&\n  !event.shiftKey;\n\nconst isCloseCombo = (event: KeyboardEvent<HTMLInputElement>) =>\n  event.key === 'Escape' &&\n  !event.metaKey &&\n  !event.ctrlKey &&\n  !event.altKey &&\n  !event.shiftKey;\n\nconst getPlatform = () => {\n  if (navigator.appVersion.includes('Win')) return 'windows';\n  if (navigator.appVersion.includes('Mac')) return 'macos';\n  if (navigator.appVersion.includes('X11')) return 'unix';\n  if (navigator.appVersion.includes('Linux')) return 'linux';\n\n  return null;\n};\n\nconst hasNewWindowModifier = (\n  event: KeyboardEvent<HTMLElement> | MouseEvent<HTMLElement>\n) => {\n  const platform = getPlatform();\n  switch (platform) {\n    case 'macos':\n      return event.metaKey;\n    default:\n      return event.ctrlKey;\n  }\n};\n\nconst shakeAnimation = keyframes`\n  from,\n  to {\n    transform: translate3d(0, 0, 0);\n  }\n\n  14%,\n  42%,\n  70% {\n    transform: translate3d(-3px, 0, 0);\n  }\n\n  28%,\n  56%,\n  84% {\n    transform: translate3d(3px, 0, 0);\n  }\n`;\n\ntype State = {\n  hasNetworkError: boolean;\n  isLoading: boolean;\n  searchText: SearchText;\n  selectedResult: SelectedResult;\n  // Used for UX when browsing through history\n  enableHistory: boolean;\n  results: Command[];\n  stack: Stack[];\n};\ntype Action =\n  | { type: 'networkError'; payload: boolean }\n  | { type: 'loading'; payload: boolean }\n  | { type: 'selectedResult'; payload: number }\n  | { type: 'incrementSelectedResult' }\n  | { type: 'decrementSelectedResult' }\n  | {\n      type: 'pickCommandFromHistory';\n      payload: { searchText: SearchText; results: Command[] };\n    }\n  | { type: 'setNextCommands'; payload: { results: Command[] } }\n  | {\n      type: 'setPrevCommands';\n      payload: { searchText: SearchText; results: Command[] };\n    }\n  | { type: 'searchText'; payload: string }\n  | { type: 'setSearchTextResults'; payload: Command[] }\n  | { type: 'resetSearchText' }\n  | { type: 'resetResultsWhenClosing' }\n  | { type: 'reset' };\nconst initialState = {\n  hasNetworkError: false,\n  isLoading: false,\n  searchText: '',\n  selectedResult: -1,\n  // Used for UX when browsing through history\n  enableHistory: true,\n  results: [],\n  stack: [],\n};\nconst reducer = (state: State = initialState, action: Action): State => {\n  switch (action.type) {\n    case 'networkError':\n      return { ...state, hasNetworkError: action.payload };\n    case 'loading':\n      return { ...state, isLoading: action.payload };\n    case 'selectedResult':\n      return { ...state, selectedResult: action.payload };\n    case 'incrementSelectedResult':\n      return {\n        ...state,\n        selectedResult:\n          state.selectedResult === state.results.length - 1\n            ? 0\n            : state.selectedResult + 1,\n        enableHistory: false,\n      };\n    case 'decrementSelectedResult':\n      return {\n        ...state,\n        selectedResult:\n          state.selectedResult < 1\n            ? state.results.length - 1\n            : state.selectedResult - 1,\n        enableHistory: false,\n      };\n    case 'pickCommandFromHistory':\n      return {\n        ...state,\n        selectedResult: 0,\n        searchText: action.payload.searchText,\n        results: action.payload.results,\n        stack: [],\n        // The history does not get changed here, it will be changed along\n        // with the regular flow.\n      };\n    case 'setNextCommands':\n      return {\n        ...state,\n        stack: [\n          ...state.stack,\n          {\n            searchText: state.searchText,\n            results: state.results,\n            selectedResult: state.selectedResult,\n          },\n        ],\n        selectedResult: 0,\n        enableHistory: false,\n        results: action.payload.results,\n      };\n    case 'setPrevCommands':\n      return {\n        ...state,\n        searchText: action.payload.searchText,\n        results: action.payload.results,\n        selectedResult: 0,\n        enableHistory: false,\n        // omit last item\n        stack: state.stack.slice(0, -1),\n      };\n    case 'searchText':\n      return {\n        ...state,\n        searchText: action.payload,\n        // clear network error when search text is cleared, so that users\n        // are tempted to retry\n        hasNetworkError: action.payload.length > 0 && state.hasNetworkError,\n      };\n    case 'setSearchTextResults':\n      return {\n        ...state,\n        results: action.payload,\n        selectedResult: action.payload.length > 0 ? 0 : -1,\n        enableHistory: true,\n        stack: [],\n      };\n    case 'resetSearchText':\n      return { ...state, searchText: '', results: [], selectedResult: -1 };\n    case 'resetResultsWhenClosing':\n      return { ...state, selectedResult: -1, enableHistory: true };\n    case 'reset':\n      return initialState;\n    default:\n      return state;\n  }\n};\n\ntype Props = {\n  historyEntries: HistoryEntry[];\n  onHistoryEntriesChange: (historyEntries: HistoryEntry[]) => void;\n  search: (searchText: SearchText) => Promise<Command[]>;\n  getNextCommands: (command: Command) => Promise<Command[]>;\n  executeCommand: (command: Command, meta: { openInNewTab: boolean }) => void;\n  onClose: () => void;\n  classNameShakeAnimation: string;\n};\nconst Butler = (props: Props) => {\n  const intl = useIntl();\n  const [state, dispatch] = useReducer(reducer, initialState);\n\n  const shouldSelectFieldText = useRef(false);\n  const isNewWindowCombo = useRef(false);\n  const skipNextSelection = useRef(false);\n  const searchContainerRef = useRef<HTMLDivElement>(null);\n  const searchInputRef = useRef<HTMLInputElement>(null);\n\n  const setHasNetworkError = useCallback(() => {\n    dispatch({ type: 'networkError', payload: true });\n  }, []);\n  const unsetHasNetworkError = useCallback(() => {\n    dispatch({ type: 'networkError', payload: false });\n  }, []);\n  const setIsLoading = useCallback(() => {\n    dispatch({ type: 'loading', payload: true });\n  }, []);\n  const unsetIsLoading = useCallback(() => {\n    dispatch({ type: 'loading', payload: false });\n  }, []);\n\n  // Destructure functions from props to reference them in the hook dependency list\n  const {\n    search: searchFromParent,\n    onClose: onCloseFromParent,\n    executeCommand: executeCommandFromParent,\n    onHistoryEntriesChange: onHistoryEntriesChangeFromParent,\n    getNextCommands: getNextCommandsFromParent,\n  } = props;\n\n  const shake = useCallback(() => {\n    if (searchContainerRef.current) {\n      searchContainerRef.current.classList.remove(\n        props.classNameShakeAnimation\n      );\n      // -> triggering reflow\n      // eslint-disable-next-line no-void\n      void searchContainerRef.current.offsetWidth;\n      searchContainerRef.current.classList.add(props.classNameShakeAnimation);\n    }\n  }, [props.classNameShakeAnimation]);\n\n  const execute = useCallback(\n    (command: Command, meta: { openInNewTab: boolean }) => {\n      // Only main entries get added to history, so when a subcommand is executed,\n      // we add the main command of it to the history (the top-level command).\n      //\n      // The key to identify history entries by is always the searchText\n      // There will never be two history entries with the same searchText\n      const entry =\n        state.stack.length === 0\n          ? // The stack is empty, so we are executing a top-level command\n            { searchText: state.searchText, results: state.results }\n          : // We are executing a subcommand, so we get the top-level command for it,\n            // which is at the bottom of the stack.\n            {\n              searchText: state.stack[0].searchText,\n              results: state.stack[0].results,\n            };\n\n      // Add the entry to the history, while excluding any earlier history entry\n      // with the same search text. This effectively \"moves\" that entry to the\n      // top of the history (with the most recent results), or appends a new entry\n      // when it didn't exist before.\n      onHistoryEntriesChangeFromParent([\n        ...props.historyEntries.filter(\n          (command) => command.searchText !== entry.searchText\n        ),\n        entry,\n      ]);\n\n      dispatch({ type: 'resetSearchText' });\n\n      onCloseFromParent();\n\n      executeCommandFromParent(command, meta);\n    },\n    [\n      executeCommandFromParent,\n      onCloseFromParent,\n      onHistoryEntriesChangeFromParent,\n      props.historyEntries,\n      state.results,\n      state.searchText,\n      state.stack,\n    ]\n  );\n  const handleKeyDown = useCallback<KeyboardEventHandler<HTMLInputElement>>(\n    (event) => {\n      // Preventing cursor jumps can only happen in onKeyDown, but not in onKeyUp\n      event.persist();\n\n      // We want to know when the user presses cmd+enter (cmd being a meta key).\n      // We are only told about this in keyDown, but not in keyUp, so we need\n      // to handle it here\n      if (event.key === 'Enter' && hasNewWindowModifier(event)) {\n        isNewWindowCombo.current = true;\n        return;\n      }\n\n      // Avoid selecting the whole page when user selectes everything with\n      // a keyboard shortcut. There is probably a better way to do this though.\n      // This prevents the whole page from being selected in case the user\n      // 1) opens the search box\n      // 2) types into it\n      // 3) selects all text using cmd+a\n      // 4) closes the search box with esc\n      // Without this handling, the whole page would now be selected\n      if (isSelectAllCombo(event)) {\n        // This stops the browser from selecting anything\n        event.preventDefault();\n        if (searchInputRef.current) {\n          // This selects the text in the search input\n          searchInputRef.current.setSelectionRange(0, state.searchText.length);\n        }\n        return;\n      }\n\n      // avoid interfering with other key combinations using modifier keys\n      if (event.ctrlKey || event.altKey || event.shiftKey || event.metaKey)\n        return;\n      if (isCloseCombo(event)) return;\n\n      // skip next mouseEnter to avoid setting selectedResult when cursor just\n      // happens to be where the results will pop up\n      skipNextSelection.current = true;\n\n      if (event.key === 'ArrowDown') {\n        // prevent cursor from jumping to end of text input\n        event.preventDefault();\n        dispatch({ type: 'incrementSelectedResult' });\n        return;\n      }\n      if (event.key === 'ArrowUp') {\n        // browse through history\n        if (\n          state.searchText.length === 0 ||\n          (state.selectedResult < 1 && state.enableHistory)\n        ) {\n          shouldSelectFieldText.current = true;\n          const selectedIndex =\n            state.searchText.length === 0\n              ? // When going back the first step\n                -1\n              : // When going back more than one step\n                props.historyEntries.findIndex(\n                  (command) => command.searchText === state.searchText\n                );\n          // Pick the previous command from the history\n          const prevCommand =\n            selectedIndex === -1\n              ? // previous command on top of the history when going back on\n                // first step\n                last(props.historyEntries)\n              : // previous command is deeper down\n                // When the history does not exist (negative index), then\n                // this implicitly returns undefined\n                props.historyEntries[selectedIndex - 1];\n          // Skip when no previous entry exists in the history\n          if (!prevCommand) return;\n          dispatch({\n            type: 'pickCommandFromHistory',\n            payload: {\n              searchText: prevCommand.searchText,\n              results: prevCommand.results,\n            },\n          });\n          return;\n        }\n        // prevent cursor from jumping to beginning of text input\n        event.preventDefault();\n        dispatch({ type: 'decrementSelectedResult' });\n        return;\n      }\n      if (state.selectedResult > -1) {\n        if (event.key === 'ArrowRight') {\n          const command = state.results[state.selectedResult];\n          const searchText = state.searchText;\n          const isCursorAtEnd =\n            searchInputRef.current &&\n            state.searchText.length === searchInputRef.current.selectionStart;\n\n          const isEverythingSelected =\n            searchInputRef.current &&\n            searchInputRef.current.selectionStart === 0 &&\n            state.searchText.length === searchInputRef.current.selectionEnd;\n\n          // only allow diving in when cursor is at end of input or when\n          // the complete text is selected (when browsing through history)\n          if (!isCursorAtEnd && !isEverythingSelected) return;\n\n          unsetHasNetworkError();\n\n          // NOTE: since we need to fetch the \"next command\", which is an async operation,\n          // we use a IIFE to process that and eventually update the state.\n          (async () => {\n            if (command) {\n              const nextCommands = await getNextCommandsFromParent(command);\n              // avoid moving cursor when there are sub-options\n              if (nextCommands.length > 0) {\n                // Ensure the search text has not changed while we were loading\n                // the next results, otherwise we'd interrupt the user.\n                // Throw away the results in case the search text has changed.\n                if (state.searchText === searchText) {\n                  dispatch({\n                    type: 'setNextCommands',\n                    payload: { results: nextCommands },\n                  });\n                }\n                return;\n              }\n            }\n            shake();\n          })();\n          return;\n        }\n        if (event.key === 'ArrowLeft') {\n          // go left in stack\n          const prevCommand = last(state.stack);\n\n          // do nothing when we can't go left anymore\n          if (!prevCommand) return;\n\n          // prevent cursor from jumping a char to the left in text input\n          event.preventDefault();\n\n          dispatch({\n            type: 'setPrevCommands',\n            payload: {\n              searchText: prevCommand.searchText,\n              results: prevCommand.results,\n            },\n          });\n          return;\n        }\n      }\n    },\n    [\n      getNextCommandsFromParent,\n      props.historyEntries,\n      shake,\n      state,\n      unsetHasNetworkError,\n    ]\n  );\n  const handleKeyUp = useCallback<KeyboardEventHandler<HTMLInputElement>>(\n    (event) => {\n      // setting the selection can only happen in onKeyUp\n      if (shouldSelectFieldText.current) {\n        const input = event.target as HTMLInputElement;\n        input.focus();\n        input.select();\n        shouldSelectFieldText.current = false;\n      }\n\n      if (event.key !== 'Enter' && !isNewWindowCombo.current) return true;\n\n      // User just triggered the search\n      if (state.selectedResult === -1) return true;\n\n      // User had something selected and wants to go there\n      execute(state.results[state.selectedResult], {\n        openInNewTab: isNewWindowCombo.current,\n      });\n\n      isNewWindowCombo.current = false;\n      return true;\n    },\n    [execute, state.results, state.selectedResult]\n  );\n  const handleChange = useCallback<ChangeEventHandler<HTMLInputElement>>(\n    (event) => {\n      const searchText = event.target.value;\n      if (searchText.trim().length === 0) {\n        dispatch({ type: 'reset' });\n        return;\n      }\n\n      dispatch({ type: 'searchText', payload: searchText });\n\n      // A search via network is only triggered when there\n      // are more than three characters. So no false loading\n      // indication is given.\n      if (searchText.trim().length > 3) {\n        setIsLoading();\n      }\n\n      searchFromParent(searchText).then(\n        (asyncResults: Command[]) => {\n          unsetHasNetworkError();\n          unsetIsLoading();\n\n          const fuse = new Fuse(asyncResults, {\n            keys: [\n              { name: 'text', weight: 0.6 },\n              { name: 'keywords', weight: 0.4 },\n            ],\n            minMatchCharLength: 2,\n            includeScore: true,\n          });\n\n          const searchResults = fuse\n            .search(searchText)\n            // Filter out results with a matching score over 0.75\n            .filter((result) => (result.score ? result.score < 0.75 : false))\n            // Keep a maximal of 9 results\n            .slice(0, 9);\n\n          dispatch({\n            type: 'setSearchTextResults',\n            payload: searchResults.map((result) => result.item),\n          });\n        },\n        (error: Error) => {\n          // eslint-disable-next-line no-console\n          if (process.env.NODE_ENV !== 'production') console.error(error);\n          unsetIsLoading();\n          setHasNetworkError();\n        }\n      );\n    },\n    [\n      searchFromParent,\n      setHasNetworkError,\n      setIsLoading,\n      unsetHasNetworkError,\n      unsetIsLoading,\n    ]\n  );\n  const handleContainerClick = useCallback(() => {\n    dispatch({ type: 'resetResultsWhenClosing' });\n    onCloseFromParent();\n  }, [onCloseFromParent]);\n\n  const createCommandMouseEnterHandler = useCallback<\n    (index: number) => MouseEventHandler<HTMLDivElement>\n  >(\n    (index) => () => {\n      // In case the cursor happened to be in a location where a\n      // result would appear, it would trigger onMouseEnter and the\n      // result would be selected immediately. This is not something\n      // a user would expect, hence we prevent it from happening.\n      // The user has to move the cursor to an option explicitly for\n      // it to become active. However, the user can always click and\n      // that action will be triggered.\n      if (skipNextSelection.current) {\n        skipNextSelection.current = false;\n        return;\n      }\n\n      // sets the selected result, mainly for the hover effect\n      dispatch({ type: 'selectedResult', payload: index });\n    },\n    []\n  );\n  const createCommandClickHandler = useCallback<\n    (command: Command) => MouseEventHandler<HTMLDivElement>\n  >(\n    (command) => (event) => {\n      execute(command, {\n        openInNewTab: hasNewWindowModifier(event),\n      });\n    },\n    [execute]\n  );\n\n  return (\n    <ButlerContainer\n      onClick={handleContainerClick}\n      data-testid=\"quick-access\"\n      tabIndex={-1}\n    >\n      <div\n        ref={searchContainerRef}\n        css={css`\n          background-color: ${uiKitDesignTokens.colorSurface};\n          border: 0;\n          border-radius: ${uiKitDesignTokens.borderRadius4};\n          min-height: 40px;\n\n          /* one more than app-bar (20000) and one more than the overlay (20001) */\n          z-index: 20002;\n          width: 400px;\n          margin: 40px auto;\n          overflow: hidden;\n          -webkit-box-shadow: 0 10px 30px -8px rgba(0, 0, 0, 0.75);\n          -moz-box-shadow: 0 10px 30px -8px rgba(0, 0, 0, 0.75);\n          box-shadow: 0 10px 30px -8px rgba(0, 0, 0, 0.75);\n          padding-bottom: ${state.hasNetworkError\n            ? '0'\n            : uiKitDesignTokens.spacingS};\n        `}\n        onClick={(event) => {\n          // Avoid closing when the searchContainer itself is clicked\n          // If we don't do this, then the overlay will close when e.g.\n          // the search input is clicked.\n          event.stopPropagation();\n          event.preventDefault();\n        }}\n      >\n        <div\n          css={css`\n            display: flex;\n          `}\n        >\n          <label\n            htmlFor=\"quick-access-search-input\"\n            css={css`\n              align-self: center;\n              padding-left: ${uiKitDesignTokens.spacingM};\n              margin-top: ${uiKitDesignTokens.spacingS};\n            `}\n          >\n            <SearchIcon color=\"neutral60\" />\n          </label>\n          <input\n            id=\"quick-access-search-input\"\n            ref={searchInputRef}\n            placeholder={intl.formatMessage(messages.inputPlacehoder)}\n            type=\"text\"\n            css={css`\n              width: 100%;\n              border: 0;\n              outline: 0;\n              font-size: 22px;\n              font-weight: 300;\n              padding: ${uiKitDesignTokens.spacingM}\n                ${uiKitDesignTokens.spacingM} ${uiKitDesignTokens.spacingS}\n                ${uiKitDesignTokens.spacingS};\n              &::placeholder {\n                color: ${uiKitDesignTokens.colorNeutral60};\n              }\n            `}\n            value={state.searchText}\n            onChange={handleChange}\n            onKeyDown={handleKeyDown}\n            onKeyUp={handleKeyUp}\n            autoFocus={true}\n            autoComplete=\"off\"\n            data-testid=\"quick-access-search-input\"\n          />\n          {state.isLoading && (\n            <div\n              css={css`\n                align-self: center;\n                margin-top: ${uiKitDesignTokens.spacingS};\n                margin-right: ${uiKitDesignTokens.spacingS};\n              `}\n            >\n              <LoadingSpinner />\n            </div>\n          )}\n        </div>\n        {(() => {\n          if (state.hasNetworkError)\n            return (\n              <div\n                css={css`\n                  overflow: hidden;\n                  white-space: nowrap;\n                  cursor: default;\n                  background: ${uiKitDesignTokens.colorError};\n                  text-align: center;\n                  text-transform: uppercase;\n                  color: ${uiKitDesignTokens.colorSurface};\n                  font-size: ${uiKitDesignTokens.fontSize20};\n                  padding: ${uiKitDesignTokens.spacingXs};\n                `}\n              >\n                <FormattedMessage {...messages.offline} />\n              </div>\n            );\n\n          if (state.results.length === 0 && state.searchText.trim().length > 0)\n            return (\n              <div\n                css={css`\n                  overflow: hidden;\n                  white-space: nowrap;\n                  cursor: default;\n                  background: ${uiKitDesignTokens.colorNeutral};\n                  color: ${uiKitDesignTokens.colorSolid};\n                  text-align: center;\n                  text-transform: uppercase;\n                  font-size: ${uiKitDesignTokens.fontSize20};\n                  padding: ${uiKitDesignTokens.spacingXs};\n                `}\n              >\n                <FormattedMessage {...messages.noResults} />\n              </div>\n            );\n\n          return state.results.map((command, index) => (\n            <ButlerCommand\n              key={command.id}\n              command={command}\n              isSelected={state.selectedResult === index}\n              onMouseEnter={createCommandMouseEnterHandler(index)}\n              onClick={createCommandClickHandler(command)}\n            />\n          ));\n        })()}\n      </div>\n    </ButlerContainer>\n  );\n};\nButler.displayName = 'Butler';\n\nconst ButlerWithAnimation = (props: Omit<Props, 'classNameShakeAnimation'>) => (\n  <ClassNames>\n    {({ css }) => (\n      <Butler\n        {...props}\n        classNameShakeAnimation={css`\n          animation-duration: 0.45s;\n          animation-fill-mode: both;\n          animation-name: ${shakeAnimation};\n        `}\n      />\n    )}\n  </ClassNames>\n);\n\nexport default ButlerWithAnimation;\n"]} */"),
887
- children: jsx(FormattedMessage, _objectSpread({}, messages.offline))
888
- });
889
- if (state.results.length === 0 && _trimInstanceProperty(_context0 = state.searchText).call(_context0).length > 0) return jsx("div", {
890
- css: /*#__PURE__*/css("overflow:hidden;white-space:nowrap;cursor:default;background:", designTokens.colorNeutral, ";color:", designTokens.colorSolid, ";text-align:center;text-transform:uppercase;font-size:", designTokens.fontSize20, ";padding:", designTokens.spacingXs, ";" + (process.env.NODE_ENV === "production" ? "" : ";label:Butler;"), process.env.NODE_ENV === "production" ? "" : "/*# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["butler.tsx"],"names":[],"mappings":"AAyrBwB","file":"butler.tsx","sourcesContent":["import {\n  KeyboardEventHandler,\n  ChangeEventHandler,\n  KeyboardEvent,\n  MouseEventHandler,\n  MouseEvent,\n  useReducer,\n  useRef,\n  useCallback,\n} from 'react';\nimport { css, keyframes, ClassNames } from '@emotion/react';\nimport Fuse from 'fuse.js';\nimport last from 'lodash/last';\nimport { FormattedMessage, useIntl } from 'react-intl';\nimport { designTokens as uiKitDesignTokens } from '@commercetools-uikit/design-system';\nimport { SearchIcon } from '@commercetools-uikit/icons';\nimport LoadingSpinner from '@commercetools-uikit/loading-spinner';\nimport ButlerCommand from '../butler-command';\nimport ButlerContainer from '../butler-container';\nimport messages from '../messages';\nimport type {\n  Command,\n  SearchText,\n  SelectedResult,\n  Stack,\n  HistoryEntry,\n} from '../types';\n\nconst isSelectAllCombo = (event: KeyboardEvent<HTMLInputElement>) =>\n  event.key === 'a' &&\n  event.metaKey &&\n  !event.ctrlKey &&\n  !event.altKey &&\n  !event.shiftKey;\n\nconst isCloseCombo = (event: KeyboardEvent<HTMLInputElement>) =>\n  event.key === 'Escape' &&\n  !event.metaKey &&\n  !event.ctrlKey &&\n  !event.altKey &&\n  !event.shiftKey;\n\nconst getPlatform = () => {\n  if (navigator.appVersion.includes('Win')) return 'windows';\n  if (navigator.appVersion.includes('Mac')) return 'macos';\n  if (navigator.appVersion.includes('X11')) return 'unix';\n  if (navigator.appVersion.includes('Linux')) return 'linux';\n\n  return null;\n};\n\nconst hasNewWindowModifier = (\n  event: KeyboardEvent<HTMLElement> | MouseEvent<HTMLElement>\n) => {\n  const platform = getPlatform();\n  switch (platform) {\n    case 'macos':\n      return event.metaKey;\n    default:\n      return event.ctrlKey;\n  }\n};\n\nconst shakeAnimation = keyframes`\n  from,\n  to {\n    transform: translate3d(0, 0, 0);\n  }\n\n  14%,\n  42%,\n  70% {\n    transform: translate3d(-3px, 0, 0);\n  }\n\n  28%,\n  56%,\n  84% {\n    transform: translate3d(3px, 0, 0);\n  }\n`;\n\ntype State = {\n  hasNetworkError: boolean;\n  isLoading: boolean;\n  searchText: SearchText;\n  selectedResult: SelectedResult;\n  // Used for UX when browsing through history\n  enableHistory: boolean;\n  results: Command[];\n  stack: Stack[];\n};\ntype Action =\n  | { type: 'networkError'; payload: boolean }\n  | { type: 'loading'; payload: boolean }\n  | { type: 'selectedResult'; payload: number }\n  | { type: 'incrementSelectedResult' }\n  | { type: 'decrementSelectedResult' }\n  | {\n      type: 'pickCommandFromHistory';\n      payload: { searchText: SearchText; results: Command[] };\n    }\n  | { type: 'setNextCommands'; payload: { results: Command[] } }\n  | {\n      type: 'setPrevCommands';\n      payload: { searchText: SearchText; results: Command[] };\n    }\n  | { type: 'searchText'; payload: string }\n  | { type: 'setSearchTextResults'; payload: Command[] }\n  | { type: 'resetSearchText' }\n  | { type: 'resetResultsWhenClosing' }\n  | { type: 'reset' };\nconst initialState = {\n  hasNetworkError: false,\n  isLoading: false,\n  searchText: '',\n  selectedResult: -1,\n  // Used for UX when browsing through history\n  enableHistory: true,\n  results: [],\n  stack: [],\n};\nconst reducer = (state: State = initialState, action: Action): State => {\n  switch (action.type) {\n    case 'networkError':\n      return { ...state, hasNetworkError: action.payload };\n    case 'loading':\n      return { ...state, isLoading: action.payload };\n    case 'selectedResult':\n      return { ...state, selectedResult: action.payload };\n    case 'incrementSelectedResult':\n      return {\n        ...state,\n        selectedResult:\n          state.selectedResult === state.results.length - 1\n            ? 0\n            : state.selectedResult + 1,\n        enableHistory: false,\n      };\n    case 'decrementSelectedResult':\n      return {\n        ...state,\n        selectedResult:\n          state.selectedResult < 1\n            ? state.results.length - 1\n            : state.selectedResult - 1,\n        enableHistory: false,\n      };\n    case 'pickCommandFromHistory':\n      return {\n        ...state,\n        selectedResult: 0,\n        searchText: action.payload.searchText,\n        results: action.payload.results,\n        stack: [],\n        // The history does not get changed here, it will be changed along\n        // with the regular flow.\n      };\n    case 'setNextCommands':\n      return {\n        ...state,\n        stack: [\n          ...state.stack,\n          {\n            searchText: state.searchText,\n            results: state.results,\n            selectedResult: state.selectedResult,\n          },\n        ],\n        selectedResult: 0,\n        enableHistory: false,\n        results: action.payload.results,\n      };\n    case 'setPrevCommands':\n      return {\n        ...state,\n        searchText: action.payload.searchText,\n        results: action.payload.results,\n        selectedResult: 0,\n        enableHistory: false,\n        // omit last item\n        stack: state.stack.slice(0, -1),\n      };\n    case 'searchText':\n      return {\n        ...state,\n        searchText: action.payload,\n        // clear network error when search text is cleared, so that users\n        // are tempted to retry\n        hasNetworkError: action.payload.length > 0 && state.hasNetworkError,\n      };\n    case 'setSearchTextResults':\n      return {\n        ...state,\n        results: action.payload,\n        selectedResult: action.payload.length > 0 ? 0 : -1,\n        enableHistory: true,\n        stack: [],\n      };\n    case 'resetSearchText':\n      return { ...state, searchText: '', results: [], selectedResult: -1 };\n    case 'resetResultsWhenClosing':\n      return { ...state, selectedResult: -1, enableHistory: true };\n    case 'reset':\n      return initialState;\n    default:\n      return state;\n  }\n};\n\ntype Props = {\n  historyEntries: HistoryEntry[];\n  onHistoryEntriesChange: (historyEntries: HistoryEntry[]) => void;\n  search: (searchText: SearchText) => Promise<Command[]>;\n  getNextCommands: (command: Command) => Promise<Command[]>;\n  executeCommand: (command: Command, meta: { openInNewTab: boolean }) => void;\n  onClose: () => void;\n  classNameShakeAnimation: string;\n};\nconst Butler = (props: Props) => {\n  const intl = useIntl();\n  const [state, dispatch] = useReducer(reducer, initialState);\n\n  const shouldSelectFieldText = useRef(false);\n  const isNewWindowCombo = useRef(false);\n  const skipNextSelection = useRef(false);\n  const searchContainerRef = useRef<HTMLDivElement>(null);\n  const searchInputRef = useRef<HTMLInputElement>(null);\n\n  const setHasNetworkError = useCallback(() => {\n    dispatch({ type: 'networkError', payload: true });\n  }, []);\n  const unsetHasNetworkError = useCallback(() => {\n    dispatch({ type: 'networkError', payload: false });\n  }, []);\n  const setIsLoading = useCallback(() => {\n    dispatch({ type: 'loading', payload: true });\n  }, []);\n  const unsetIsLoading = useCallback(() => {\n    dispatch({ type: 'loading', payload: false });\n  }, []);\n\n  // Destructure functions from props to reference them in the hook dependency list\n  const {\n    search: searchFromParent,\n    onClose: onCloseFromParent,\n    executeCommand: executeCommandFromParent,\n    onHistoryEntriesChange: onHistoryEntriesChangeFromParent,\n    getNextCommands: getNextCommandsFromParent,\n  } = props;\n\n  const shake = useCallback(() => {\n    if (searchContainerRef.current) {\n      searchContainerRef.current.classList.remove(\n        props.classNameShakeAnimation\n      );\n      // -> triggering reflow\n      // eslint-disable-next-line no-void\n      void searchContainerRef.current.offsetWidth;\n      searchContainerRef.current.classList.add(props.classNameShakeAnimation);\n    }\n  }, [props.classNameShakeAnimation]);\n\n  const execute = useCallback(\n    (command: Command, meta: { openInNewTab: boolean }) => {\n      // Only main entries get added to history, so when a subcommand is executed,\n      // we add the main command of it to the history (the top-level command).\n      //\n      // The key to identify history entries by is always the searchText\n      // There will never be two history entries with the same searchText\n      const entry =\n        state.stack.length === 0\n          ? // The stack is empty, so we are executing a top-level command\n            { searchText: state.searchText, results: state.results }\n          : // We are executing a subcommand, so we get the top-level command for it,\n            // which is at the bottom of the stack.\n            {\n              searchText: state.stack[0].searchText,\n              results: state.stack[0].results,\n            };\n\n      // Add the entry to the history, while excluding any earlier history entry\n      // with the same search text. This effectively \"moves\" that entry to the\n      // top of the history (with the most recent results), or appends a new entry\n      // when it didn't exist before.\n      onHistoryEntriesChangeFromParent([\n        ...props.historyEntries.filter(\n          (command) => command.searchText !== entry.searchText\n        ),\n        entry,\n      ]);\n\n      dispatch({ type: 'resetSearchText' });\n\n      onCloseFromParent();\n\n      executeCommandFromParent(command, meta);\n    },\n    [\n      executeCommandFromParent,\n      onCloseFromParent,\n      onHistoryEntriesChangeFromParent,\n      props.historyEntries,\n      state.results,\n      state.searchText,\n      state.stack,\n    ]\n  );\n  const handleKeyDown = useCallback<KeyboardEventHandler<HTMLInputElement>>(\n    (event) => {\n      // Preventing cursor jumps can only happen in onKeyDown, but not in onKeyUp\n      event.persist();\n\n      // We want to know when the user presses cmd+enter (cmd being a meta key).\n      // We are only told about this in keyDown, but not in keyUp, so we need\n      // to handle it here\n      if (event.key === 'Enter' && hasNewWindowModifier(event)) {\n        isNewWindowCombo.current = true;\n        return;\n      }\n\n      // Avoid selecting the whole page when user selectes everything with\n      // a keyboard shortcut. There is probably a better way to do this though.\n      // This prevents the whole page from being selected in case the user\n      // 1) opens the search box\n      // 2) types into it\n      // 3) selects all text using cmd+a\n      // 4) closes the search box with esc\n      // Without this handling, the whole page would now be selected\n      if (isSelectAllCombo(event)) {\n        // This stops the browser from selecting anything\n        event.preventDefault();\n        if (searchInputRef.current) {\n          // This selects the text in the search input\n          searchInputRef.current.setSelectionRange(0, state.searchText.length);\n        }\n        return;\n      }\n\n      // avoid interfering with other key combinations using modifier keys\n      if (event.ctrlKey || event.altKey || event.shiftKey || event.metaKey)\n        return;\n      if (isCloseCombo(event)) return;\n\n      // skip next mouseEnter to avoid setting selectedResult when cursor just\n      // happens to be where the results will pop up\n      skipNextSelection.current = true;\n\n      if (event.key === 'ArrowDown') {\n        // prevent cursor from jumping to end of text input\n        event.preventDefault();\n        dispatch({ type: 'incrementSelectedResult' });\n        return;\n      }\n      if (event.key === 'ArrowUp') {\n        // browse through history\n        if (\n          state.searchText.length === 0 ||\n          (state.selectedResult < 1 && state.enableHistory)\n        ) {\n          shouldSelectFieldText.current = true;\n          const selectedIndex =\n            state.searchText.length === 0\n              ? // When going back the first step\n                -1\n              : // When going back more than one step\n                props.historyEntries.findIndex(\n                  (command) => command.searchText === state.searchText\n                );\n          // Pick the previous command from the history\n          const prevCommand =\n            selectedIndex === -1\n              ? // previous command on top of the history when going back on\n                // first step\n                last(props.historyEntries)\n              : // previous command is deeper down\n                // When the history does not exist (negative index), then\n                // this implicitly returns undefined\n                props.historyEntries[selectedIndex - 1];\n          // Skip when no previous entry exists in the history\n          if (!prevCommand) return;\n          dispatch({\n            type: 'pickCommandFromHistory',\n            payload: {\n              searchText: prevCommand.searchText,\n              results: prevCommand.results,\n            },\n          });\n          return;\n        }\n        // prevent cursor from jumping to beginning of text input\n        event.preventDefault();\n        dispatch({ type: 'decrementSelectedResult' });\n        return;\n      }\n      if (state.selectedResult > -1) {\n        if (event.key === 'ArrowRight') {\n          const command = state.results[state.selectedResult];\n          const searchText = state.searchText;\n          const isCursorAtEnd =\n            searchInputRef.current &&\n            state.searchText.length === searchInputRef.current.selectionStart;\n\n          const isEverythingSelected =\n            searchInputRef.current &&\n            searchInputRef.current.selectionStart === 0 &&\n            state.searchText.length === searchInputRef.current.selectionEnd;\n\n          // only allow diving in when cursor is at end of input or when\n          // the complete text is selected (when browsing through history)\n          if (!isCursorAtEnd && !isEverythingSelected) return;\n\n          unsetHasNetworkError();\n\n          // NOTE: since we need to fetch the \"next command\", which is an async operation,\n          // we use a IIFE to process that and eventually update the state.\n          (async () => {\n            if (command) {\n              const nextCommands = await getNextCommandsFromParent(command);\n              // avoid moving cursor when there are sub-options\n              if (nextCommands.length > 0) {\n                // Ensure the search text has not changed while we were loading\n                // the next results, otherwise we'd interrupt the user.\n                // Throw away the results in case the search text has changed.\n                if (state.searchText === searchText) {\n                  dispatch({\n                    type: 'setNextCommands',\n                    payload: { results: nextCommands },\n                  });\n                }\n                return;\n              }\n            }\n            shake();\n          })();\n          return;\n        }\n        if (event.key === 'ArrowLeft') {\n          // go left in stack\n          const prevCommand = last(state.stack);\n\n          // do nothing when we can't go left anymore\n          if (!prevCommand) return;\n\n          // prevent cursor from jumping a char to the left in text input\n          event.preventDefault();\n\n          dispatch({\n            type: 'setPrevCommands',\n            payload: {\n              searchText: prevCommand.searchText,\n              results: prevCommand.results,\n            },\n          });\n          return;\n        }\n      }\n    },\n    [\n      getNextCommandsFromParent,\n      props.historyEntries,\n      shake,\n      state,\n      unsetHasNetworkError,\n    ]\n  );\n  const handleKeyUp = useCallback<KeyboardEventHandler<HTMLInputElement>>(\n    (event) => {\n      // setting the selection can only happen in onKeyUp\n      if (shouldSelectFieldText.current) {\n        const input = event.target as HTMLInputElement;\n        input.focus();\n        input.select();\n        shouldSelectFieldText.current = false;\n      }\n\n      if (event.key !== 'Enter' && !isNewWindowCombo.current) return true;\n\n      // User just triggered the search\n      if (state.selectedResult === -1) return true;\n\n      // User had something selected and wants to go there\n      execute(state.results[state.selectedResult], {\n        openInNewTab: isNewWindowCombo.current,\n      });\n\n      isNewWindowCombo.current = false;\n      return true;\n    },\n    [execute, state.results, state.selectedResult]\n  );\n  const handleChange = useCallback<ChangeEventHandler<HTMLInputElement>>(\n    (event) => {\n      const searchText = event.target.value;\n      if (searchText.trim().length === 0) {\n        dispatch({ type: 'reset' });\n        return;\n      }\n\n      dispatch({ type: 'searchText', payload: searchText });\n\n      // A search via network is only triggered when there\n      // are more than three characters. So no false loading\n      // indication is given.\n      if (searchText.trim().length > 3) {\n        setIsLoading();\n      }\n\n      searchFromParent(searchText).then(\n        (asyncResults: Command[]) => {\n          unsetHasNetworkError();\n          unsetIsLoading();\n\n          const fuse = new Fuse(asyncResults, {\n            keys: [\n              { name: 'text', weight: 0.6 },\n              { name: 'keywords', weight: 0.4 },\n            ],\n            minMatchCharLength: 2,\n            includeScore: true,\n          });\n\n          const searchResults = fuse\n            .search(searchText)\n            // Filter out results with a matching score over 0.75\n            .filter((result) => (result.score ? result.score < 0.75 : false))\n            // Keep a maximal of 9 results\n            .slice(0, 9);\n\n          dispatch({\n            type: 'setSearchTextResults',\n            payload: searchResults.map((result) => result.item),\n          });\n        },\n        (error: Error) => {\n          // eslint-disable-next-line no-console\n          if (process.env.NODE_ENV !== 'production') console.error(error);\n          unsetIsLoading();\n          setHasNetworkError();\n        }\n      );\n    },\n    [\n      searchFromParent,\n      setHasNetworkError,\n      setIsLoading,\n      unsetHasNetworkError,\n      unsetIsLoading,\n    ]\n  );\n  const handleContainerClick = useCallback(() => {\n    dispatch({ type: 'resetResultsWhenClosing' });\n    onCloseFromParent();\n  }, [onCloseFromParent]);\n\n  const createCommandMouseEnterHandler = useCallback<\n    (index: number) => MouseEventHandler<HTMLDivElement>\n  >(\n    (index) => () => {\n      // In case the cursor happened to be in a location where a\n      // result would appear, it would trigger onMouseEnter and the\n      // result would be selected immediately. This is not something\n      // a user would expect, hence we prevent it from happening.\n      // The user has to move the cursor to an option explicitly for\n      // it to become active. However, the user can always click and\n      // that action will be triggered.\n      if (skipNextSelection.current) {\n        skipNextSelection.current = false;\n        return;\n      }\n\n      // sets the selected result, mainly for the hover effect\n      dispatch({ type: 'selectedResult', payload: index });\n    },\n    []\n  );\n  const createCommandClickHandler = useCallback<\n    (command: Command) => MouseEventHandler<HTMLDivElement>\n  >(\n    (command) => (event) => {\n      execute(command, {\n        openInNewTab: hasNewWindowModifier(event),\n      });\n    },\n    [execute]\n  );\n\n  return (\n    <ButlerContainer\n      onClick={handleContainerClick}\n      data-testid=\"quick-access\"\n      tabIndex={-1}\n    >\n      <div\n        ref={searchContainerRef}\n        css={css`\n          background-color: ${uiKitDesignTokens.colorSurface};\n          border: 0;\n          border-radius: ${uiKitDesignTokens.borderRadius4};\n          min-height: 40px;\n\n          /* one more than app-bar (20000) and one more than the overlay (20001) */\n          z-index: 20002;\n          width: 400px;\n          margin: 40px auto;\n          overflow: hidden;\n          -webkit-box-shadow: 0 10px 30px -8px rgba(0, 0, 0, 0.75);\n          -moz-box-shadow: 0 10px 30px -8px rgba(0, 0, 0, 0.75);\n          box-shadow: 0 10px 30px -8px rgba(0, 0, 0, 0.75);\n          padding-bottom: ${state.hasNetworkError\n            ? '0'\n            : uiKitDesignTokens.spacingS};\n        `}\n        onClick={(event) => {\n          // Avoid closing when the searchContainer itself is clicked\n          // If we don't do this, then the overlay will close when e.g.\n          // the search input is clicked.\n          event.stopPropagation();\n          event.preventDefault();\n        }}\n      >\n        <div\n          css={css`\n            display: flex;\n          `}\n        >\n          <label\n            htmlFor=\"quick-access-search-input\"\n            css={css`\n              align-self: center;\n              padding-left: ${uiKitDesignTokens.spacingM};\n              margin-top: ${uiKitDesignTokens.spacingS};\n            `}\n          >\n            <SearchIcon color=\"neutral60\" />\n          </label>\n          <input\n            id=\"quick-access-search-input\"\n            ref={searchInputRef}\n            placeholder={intl.formatMessage(messages.inputPlacehoder)}\n            type=\"text\"\n            css={css`\n              width: 100%;\n              border: 0;\n              outline: 0;\n              font-size: 22px;\n              font-weight: 300;\n              padding: ${uiKitDesignTokens.spacingM}\n                ${uiKitDesignTokens.spacingM} ${uiKitDesignTokens.spacingS}\n                ${uiKitDesignTokens.spacingS};\n              &::placeholder {\n                color: ${uiKitDesignTokens.colorNeutral60};\n              }\n            `}\n            value={state.searchText}\n            onChange={handleChange}\n            onKeyDown={handleKeyDown}\n            onKeyUp={handleKeyUp}\n            autoFocus={true}\n            autoComplete=\"off\"\n            data-testid=\"quick-access-search-input\"\n          />\n          {state.isLoading && (\n            <div\n              css={css`\n                align-self: center;\n                margin-top: ${uiKitDesignTokens.spacingS};\n                margin-right: ${uiKitDesignTokens.spacingS};\n              `}\n            >\n              <LoadingSpinner />\n            </div>\n          )}\n        </div>\n        {(() => {\n          if (state.hasNetworkError)\n            return (\n              <div\n                css={css`\n                  overflow: hidden;\n                  white-space: nowrap;\n                  cursor: default;\n                  background: ${uiKitDesignTokens.colorError};\n                  text-align: center;\n                  text-transform: uppercase;\n                  color: ${uiKitDesignTokens.colorSurface};\n                  font-size: ${uiKitDesignTokens.fontSize20};\n                  padding: ${uiKitDesignTokens.spacingXs};\n                `}\n              >\n                <FormattedMessage {...messages.offline} />\n              </div>\n            );\n\n          if (state.results.length === 0 && state.searchText.trim().length > 0)\n            return (\n              <div\n                css={css`\n                  overflow: hidden;\n                  white-space: nowrap;\n                  cursor: default;\n                  background: ${uiKitDesignTokens.colorNeutral};\n                  color: ${uiKitDesignTokens.colorSolid};\n                  text-align: center;\n                  text-transform: uppercase;\n                  font-size: ${uiKitDesignTokens.fontSize20};\n                  padding: ${uiKitDesignTokens.spacingXs};\n                `}\n              >\n                <FormattedMessage {...messages.noResults} />\n              </div>\n            );\n\n          return state.results.map((command, index) => (\n            <ButlerCommand\n              key={command.id}\n              command={command}\n              isSelected={state.selectedResult === index}\n              onMouseEnter={createCommandMouseEnterHandler(index)}\n              onClick={createCommandClickHandler(command)}\n            />\n          ));\n        })()}\n      </div>\n    </ButlerContainer>\n  );\n};\nButler.displayName = 'Butler';\n\nconst ButlerWithAnimation = (props: Omit<Props, 'classNameShakeAnimation'>) => (\n  <ClassNames>\n    {({ css }) => (\n      <Butler\n        {...props}\n        classNameShakeAnimation={css`\n          animation-duration: 0.45s;\n          animation-fill-mode: both;\n          animation-name: ${shakeAnimation};\n        `}\n      />\n    )}\n  </ClassNames>\n);\n\nexport default ButlerWithAnimation;\n"]} */"),
891
- children: jsx(FormattedMessage, _objectSpread({}, messages.noResults))
892
- });
893
- return _mapInstanceProperty(_context1 = state.results).call(_context1, (command, index) => jsx(ButlerCommand, {
894
- command: command,
895
- isSelected: state.selectedResult === index,
896
- onMouseEnter: createCommandMouseEnterHandler(index),
897
- onClick: createCommandClickHandler(command)
898
- }, command.id));
899
- })()]
900
- })
901
- });
902
- };
903
- Butler.displayName = 'Butler';
904
- const ButlerWithAnimation = props => jsx(ClassNames, {
905
- children: _ref2 => {
906
- let css = _ref2.css;
907
- return jsx(Butler, _objectSpread(_objectSpread({}, props), {}, {
908
- classNameShakeAnimation: css`
909
- animation-duration: 0.45s;
910
- animation-fill-mode: both;
911
- animation-name: ${shakeAnimation};
912
- `
913
- }));
914
- }
915
- });
916
-
917
- // eslint-disable-next-line import/prefer-default-export
918
-
919
- const permissions = {
920
- ViewOrders: 'ViewOrders',
921
- ManageOrders: 'ManageOrders',
922
- ViewProducts: 'ViewProducts',
923
- ManageProducts: 'ManageProducts',
924
- ViewCategories: 'ViewCategories',
925
- ManageCategories: 'ManageCategories',
926
- ViewCustomers: 'ViewCustomers',
927
- ManageCustomers: 'ManageCustomers',
928
- ViewCustomerGroups: 'ViewCustomerGroups',
929
- ManageCustomerGroups: 'ManageCustomerGroups',
930
- ViewProductDiscounts: 'ViewProductDiscounts',
931
- ManageProductDiscounts: 'ManageProductDiscounts',
932
- ViewDiscountCodes: 'ViewDiscountCodes',
933
- ManageDiscountCodes: 'ManageDiscountCodes',
934
- ViewCartDiscounts: 'ViewCartDiscounts',
935
- ManageCartDiscounts: 'ManageCartDiscounts',
936
- ViewProjectSettings: 'ViewProjectSettings',
937
- ManageProjectSettings: 'ManageProjectSettings',
938
- ViewDeveloperSettings: 'ViewDeveloperSettings',
939
- ManageDeveloperSettings: 'ManageDeveloperSettings',
940
- ViewProductTypes: 'ViewProductTypes',
941
- ManageProductTypes: 'ManageProductTypes'
942
- };
943
-
944
- const actionTypes = {
945
- go: 'go'
946
- };
947
-
948
- function nonNullable(value) {
949
- return value !== null && value !== undefined && typeof value !== 'boolean';
950
- }
951
- const createCommands = _ref => {
952
- var _context, _context2, _context3, _context4, _context5, _context6, _context7, _context8, _context9, _context0, _context11;
953
- let intl = _ref.intl,
954
- applicationContext = _ref.applicationContext,
955
- featureToggles = _ref.featureToggles,
956
- changeProjectDataLocale = _ref.changeProjectDataLocale;
957
- return _filterInstanceProperty(_context = [applicationContext.project && applicationContext.permissions && featureToggles.canViewDashboard && hasSomePermissions([permissions.ViewOrders], applicationContext.permissions) && {
958
- id: 'go/dashboard',
959
- text: intl.formatMessage(messages.openDashboard),
960
- keywords: ['Go to Dashboard'],
961
- action: {
962
- type: actionTypes.go,
963
- to: `/${applicationContext.project.key}/dashboard`
964
- }
965
- }, applicationContext.project && applicationContext.permissions && hasSomePermissions([permissions.ViewProducts], applicationContext.permissions) && {
966
- id: 'go/products',
967
- text: intl.formatMessage(messages.openProducts),
968
- keywords: ['Go to Products'],
969
- action: {
970
- type: actionTypes.go,
971
- to: `/${applicationContext.project.key}/products`
972
- },
973
- subCommands: _filterInstanceProperty(_context2 = [hasSomePermissions([permissions.ViewProducts], applicationContext.permissions) && {
974
- id: 'go/products/list',
975
- text: intl.formatMessage(messages.openProductList),
976
- action: {
977
- type: actionTypes.go,
978
- to: `/${applicationContext.project.key}/products`
979
- }
980
- }, hasSomePermissions([permissions.ViewProducts], applicationContext.permissions) && {
981
- id: 'go/products/modified',
982
- text: intl.formatMessage(messages.openModifiedProducts),
983
- action: {
984
- type: actionTypes.go,
985
- to: `/${applicationContext.project.key}/products/modified`
986
- }
987
- }, hasSomePermissions([permissions.ViewProducts], applicationContext.permissions) && featureToggles.pimSearch && {
988
- id: 'go/products/pim-search',
989
- text: intl.formatMessage(messages.openPimSearch),
990
- action: {
991
- type: actionTypes.go,
992
- to: `/${applicationContext.project.key}/products`
993
- }
994
- }, hasSomePermissions([permissions.ManageProducts], applicationContext.permissions) && {
995
- id: 'go/products/add',
996
- text: intl.formatMessage(messages.openAddProducts),
997
- action: {
998
- type: actionTypes.go,
999
- to: `/${applicationContext.project.key}/products/new`
1000
- }
1001
- }]).call(_context2, nonNullable)
1002
- }, applicationContext.project && applicationContext.permissions && hasSomePermissions([permissions.ViewCategories], applicationContext.permissions) && {
1003
- id: 'go/categories',
1004
- text: intl.formatMessage(messages.openCategories),
1005
- keywords: ['Go to Categories'],
1006
- action: {
1007
- type: actionTypes.go,
1008
- to: `/${applicationContext.project.key}/categories`
1009
- },
1010
- subCommands: _filterInstanceProperty(_context3 = [hasSomePermissions([permissions.ViewCategories], applicationContext.permissions) && {
1011
- id: 'go/categories/list',
1012
- text: intl.formatMessage(messages.openCategoriesList),
1013
- action: {
1014
- type: actionTypes.go,
1015
- to: `/${applicationContext.project.key}/categories?mode=list`
1016
- }
1017
- }, hasSomePermissions([permissions.ViewCategories], applicationContext.permissions) && {
1018
- id: 'go/categories/search',
1019
- text: intl.formatMessage(messages.openCategoriesSearch),
1020
- action: {
1021
- type: actionTypes.go,
1022
- to: `/${applicationContext.project.key}/categories?mode=search`
1023
- }
1024
- }, hasSomePermissions([permissions.ManageCategories], applicationContext.permissions) && {
1025
- id: 'go/categories/add',
1026
- text: intl.formatMessage(messages.openAddCategory),
1027
- action: {
1028
- type: actionTypes.go,
1029
- to: `/${applicationContext.project.key}/categories/new`
1030
- }
1031
- }]).call(_context3, nonNullable)
1032
- }, applicationContext.project && applicationContext.permissions && hasSomePermissions([permissions.ViewCustomers, permissions.ViewCustomerGroups], applicationContext.permissions) && {
1033
- id: 'go/customers',
1034
- text: intl.formatMessage(messages.openCustomers),
1035
- keywords: ['Go to Customers'],
1036
- action: {
1037
- type: actionTypes.go,
1038
- to: `/${applicationContext.project.key}/customers`
1039
- },
1040
- subCommands: _filterInstanceProperty(_context4 = [hasSomePermissions([permissions.ViewCustomers], applicationContext.permissions) && {
1041
- id: 'go/customers/list',
1042
- text: intl.formatMessage(messages.openCustomersList),
1043
- action: {
1044
- type: actionTypes.go,
1045
- to: `/${applicationContext.project.key}/customers`
1046
- }
1047
- }, hasSomePermissions([permissions.ManageCustomers], applicationContext.permissions) && {
1048
- id: 'go/customers/new',
1049
- text: intl.formatMessage(messages.openAddCustomer),
1050
- action: {
1051
- type: actionTypes.go,
1052
- to: `/${applicationContext.project.key}/customers/new`
1053
- }
1054
- }, hasSomePermissions([permissions.ViewCustomerGroups], applicationContext.permissions) && {
1055
- id: 'go/customer/customer-groups',
1056
- text: intl.formatMessage(messages.openCustomerGroupsList),
1057
- action: {
1058
- type: actionTypes.go,
1059
- to: `/${applicationContext.project.key}/customers/customer-groups`
1060
- }
1061
- }, hasSomePermissions([permissions.ManageCustomerGroups], applicationContext.permissions) && {
1062
- id: 'go/customers/customer-groups/add',
1063
- text: intl.formatMessage(messages.openAddCustomerGroup),
1064
- action: {
1065
- type: actionTypes.go,
1066
- to: `/${applicationContext.project.key}/customers/customer-groups/new`
1067
- }
1068
- }]).call(_context4, nonNullable)
1069
- }, applicationContext.project && applicationContext.permissions && hasSomePermissions([permissions.ViewOrders], applicationContext.permissions) && {
1070
- id: 'go/orders',
1071
- text: intl.formatMessage(messages.openOrders),
1072
- keywords: ['Go to Orders'],
1073
- action: {
1074
- type: actionTypes.go,
1075
- to: `/${applicationContext.project.key}/orders`
1076
- },
1077
- subCommands: _filterInstanceProperty(_context5 = [hasSomePermissions([permissions.ViewOrders], applicationContext.permissions) && {
1078
- id: 'go/orders/list',
1079
- text: intl.formatMessage(messages.openOrdersList),
1080
- action: {
1081
- type: actionTypes.go,
1082
- to: `/${applicationContext.project.key}/orders`
1083
- }
1084
- }, hasSomePermissions([permissions.ManageOrders], applicationContext.permissions) && {
1085
- id: 'go/orders/add',
1086
- text: intl.formatMessage(messages.openAddOrder),
1087
- action: {
1088
- type: actionTypes.go,
1089
- to: `/${applicationContext.project.key}/orders/new`
1090
- }
1091
- }]).call(_context5, nonNullable)
1092
- }, applicationContext.project && applicationContext.permissions && hasSomePermissions([permissions.ViewDiscountCodes, permissions.ViewProductDiscounts, permissions.ViewCartDiscounts], applicationContext.permissions) && {
1093
- id: 'go/discounts',
1094
- text: intl.formatMessage(messages.openDiscounts),
1095
- keywords: ['Go to Discounts'],
1096
- action: {
1097
- type: actionTypes.go,
1098
- to: `/${applicationContext.project.key}/discounts`
1099
- },
1100
- subCommands: _filterInstanceProperty(_context6 = [hasSomePermissions([permissions.ViewProductDiscounts], applicationContext.permissions) && {
1101
- id: 'go/discounts/products/list',
1102
- text: intl.formatMessage(messages.openProductDiscountsList),
1103
- action: {
1104
- type: actionTypes.go,
1105
- to: `/${applicationContext.project.key}/discounts/products`
1106
- }
1107
- }, hasSomePermissions([permissions.ViewCartDiscounts], applicationContext.permissions) && {
1108
- id: 'go/discounts/carts/list',
1109
- text: intl.formatMessage(messages.openCartDiscountsList),
1110
- action: {
1111
- type: actionTypes.go,
1112
- to: `/${applicationContext.project.key}/discounts/carts`
1113
- }
1114
- }, hasSomePermissions([permissions.ViewDiscountCodes], applicationContext.permissions) && {
1115
- id: 'go/discounts/codes/list',
1116
- text: intl.formatMessage(messages.openDiscountCodesList),
1117
- action: {
1118
- type: actionTypes.go,
1119
- to: `/${applicationContext.project.key}/discounts/codes`
1120
- }
1121
- }, hasSomePermissions([permissions.ManageProductDiscounts, permissions.ManageDiscountCodes, permissions.ManageCartDiscounts], applicationContext.permissions) && {
1122
- id: 'go/discounts/add',
1123
- text: intl.formatMessage(messages.openAddDiscount),
1124
- action: {
1125
- type: actionTypes.go,
1126
- to: `/${applicationContext.project.key}/discounts/new`
1127
- },
1128
- subCommands: _filterInstanceProperty(_context7 = [hasSomePermissions([permissions.ManageProductDiscounts], applicationContext.permissions) && {
1129
- id: 'go/discounts/product/add',
1130
- text: intl.formatMessage(messages.openAddProductDiscount),
1131
- action: {
1132
- type: actionTypes.go,
1133
- to: `/${applicationContext.project.key}/discounts/products/new`
1134
- }
1135
- }, hasSomePermissions([permissions.ManageCartDiscounts], applicationContext.permissions) && {
1136
- id: 'go/discounts/cart/add',
1137
- text: intl.formatMessage(messages.openAddCartDiscount),
1138
- action: {
1139
- type: actionTypes.go,
1140
- to: `/${applicationContext.project.key}/discounts/carts/new`
1141
- }
1142
- }, hasSomePermissions([permissions.ManageDiscountCodes], applicationContext.permissions) && {
1143
- id: 'go/discounts/code/add',
1144
- text: intl.formatMessage(messages.openAddCartDiscount),
1145
- action: {
1146
- type: actionTypes.go,
1147
- to: `/${applicationContext.project.key}/discounts/codes/new`
1148
- }
1149
- }]).call(_context7, nonNullable)
1150
- }]).call(_context6, nonNullable)
1151
- }, applicationContext.project && applicationContext.permissions && hasSomePermissions([permissions.ViewProjectSettings, permissions.ViewDeveloperSettings, permissions.ViewProductTypes], applicationContext.permissions) && {
1152
- id: 'go/settings',
1153
- text: intl.formatMessage(messages.openSettings),
1154
- keywords: ['Go to Settings'],
1155
- action: {
1156
- type: actionTypes.go,
1157
- to: `/${applicationContext.project.key}/settings/project`
1158
- },
1159
- subCommands: _filterInstanceProperty(_context8 = [hasSomePermissions([permissions.ViewProjectSettings, permissions.ManageProjectSettings], applicationContext.permissions) && {
1160
- id: 'go/settings/project',
1161
- text: intl.formatMessage(messages.openProjectSettings),
1162
- action: {
1163
- type: actionTypes.go,
1164
- to: `/${applicationContext.project.key}/settings/project`
1165
- },
1166
- subCommands: _filterInstanceProperty(_context9 = [{
1167
- id: 'go/settings/project/international',
1168
- text: intl.formatMessage(messages.openProjectSettingsInternationalTab),
1169
- action: {
1170
- type: actionTypes.go,
1171
- to: `/${applicationContext.project.key}/settings/project/international`
1172
- }
1173
- }, {
1174
- id: 'go/settings/project/taxes',
1175
- text: intl.formatMessage(messages.openProjectSettingsTaxesTab),
1176
- action: {
1177
- type: actionTypes.go,
1178
- to: `/${applicationContext.project.key}/settings/project/taxes`
1179
- }
1180
- }, {
1181
- id: 'go/settings/project/shipping-methods',
1182
- text: intl.formatMessage(messages.openProjectSettingsShippingMethodsTab),
1183
- action: {
1184
- type: actionTypes.go,
1185
- to: `/${applicationContext.project.key}/settings/project/shipping-methods`
1186
- }
1187
- }, {
1188
- id: 'go/settings/project/channels',
1189
- text: intl.formatMessage(messages.openProjectSettingsChannelsTab),
1190
- action: {
1191
- type: actionTypes.go,
1192
- to: `/${applicationContext.project.key}/settings/project/channels`
1193
- }
1194
- }, {
1195
- id: 'go/settings/project/stores',
1196
- text: intl.formatMessage(messages.openProjectSettingsStoresTab),
1197
- action: {
1198
- type: actionTypes.go,
1199
- to: `/${applicationContext.project.key}/settings/project/stores`
1200
- }
1201
- }]).call(_context9, nonNullable)
1202
- }, hasSomePermissions([permissions.ViewProductTypes], applicationContext.permissions) && {
1203
- id: 'go/settings/product-types',
1204
- text: intl.formatMessage(messages.openProductTypesSettings),
1205
- action: {
1206
- type: actionTypes.go,
1207
- to: `/${applicationContext.project.key}/settings/product-types`
1208
- }
1209
- }, hasSomePermissions([permissions.ViewDeveloperSettings], applicationContext.permissions) && {
1210
- id: 'go/settings/developer',
1211
- text: intl.formatMessage(messages.openDeveloperSettings),
1212
- action: {
1213
- type: actionTypes.go,
1214
- to: `/${applicationContext.project.key}/settings/developer`
1215
- },
1216
- subCommands: _filterInstanceProperty(_context0 = [hasSomePermissions([permissions.ViewDeveloperSettings], applicationContext.permissions) && {
1217
- id: 'go/settings/developer/api-clients/list',
1218
- text: intl.formatMessage(messages.openApiClientsList),
1219
- action: {
1220
- type: actionTypes.go,
1221
- to: `/${applicationContext.project.key}/settings/developer/api-clients`
1222
- }
1223
- }, hasSomePermissions([permissions.ManageDeveloperSettings], applicationContext.permissions) && {
1224
- id: 'go/settings/developer/api-clients/add',
1225
- text: intl.formatMessage(messages.openAddApiClient),
1226
- action: {
1227
- type: actionTypes.go,
1228
- to: `/${applicationContext.project.key}/settings/developer/api-clients/new`
1229
- }
1230
- }]).call(_context0, nonNullable)
1231
- }, featureToggles.customApplications && hasSomePermissions([permissions.ManageProjectSettings], applicationContext.permissions) && {
1232
- id: 'go/settings/custom-applications',
1233
- text: intl.formatMessage(messages.openCustomApplicationsSettings),
1234
- action: {
1235
- type: actionTypes.go,
1236
- to: `/${applicationContext.project.key}/settings/custom-applications`
1237
- }
1238
- }]).call(_context8, nonNullable)
1239
- }, applicationContext.project && applicationContext.project.languages && applicationContext.project.languages.length > 1 && {
1240
- id: 'action/set-resource-language',
1241
- text: intl.formatMessage(messages.setResourceLanguage),
1242
- keywords: ['set resource locale', 'set project data language', 'set project data locale'],
1243
- action: () => void 0,
1244
- // We would know these statically, but we define them here as we don't
1245
- // want to include them in the top-level search results
1246
- subCommands: () => {
1247
- var _context1, _context10;
1248
- return _Promise.resolve(_filterInstanceProperty(_context1 = _mapInstanceProperty(_context10 = applicationContext.project ? applicationContext.project.languages : []).call(_context10, language => changeProjectDataLocale && {
1249
- id: `action/set-resource-language/${language}`,
1250
- text: oneLineTrim`
1251
- ${language}
1252
- ${language === applicationContext.dataLocale ? ' (active)' : ''}
1253
- `,
1254
- action: () => {
1255
- changeProjectDataLocale(language);
1256
-
1257
- // We reload, since ProjectDataLocale is written in a way where
1258
- // only the tree under the parent container reloads, but
1259
- // not all of them reload.
1260
- // So this action would seem like it had not effect, unless we
1261
- // reload
1262
- location.reload();
1263
- }
1264
- })).call(_context1, nonNullable));
1265
- }
1266
- }, {
1267
- id: 'go/support',
1268
- text: intl.formatMessage(messages.openSupport),
1269
- keywords: ['Go to support'],
1270
- action: {
1271
- type: actionTypes.go,
1272
- to: SUPPORT_PORTAL_URL
1273
- }
1274
- }, {
1275
- id: 'go/account-profile',
1276
- text: intl.formatMessage(messages.openMyProfile),
1277
- keywords: ['Go to user account', 'Go to profile', 'Open profile'],
1278
- action: {
1279
- type: actionTypes.go,
1280
- to: `/account/profile`
1281
- }
1282
- }, {
1283
- id: 'go/privacy-policy',
1284
- text: intl.formatMessage(messages.showPrivacyPolicy),
1285
- keywords: ['Open Privacy Policy'],
1286
- action: {
1287
- type: actionTypes.go,
1288
- to: 'https://commercetools.com/privacy#suppliers'
1289
- }
1290
- }, {
1291
- id: 'go/logout',
1292
- text: intl.formatMessage(messages.logout),
1293
- keywords: ['Sign out'],
1294
- action: {
1295
- type: actionTypes.go,
1296
- to: `/logout?reason=${LOGOUT_REASONS.USER}`
1297
- }
1298
- }, {
1299
- id: 'go/manage-projects',
1300
- text: intl.formatMessage(messages.openManageProjects),
1301
- keywords: ['Go to manage projects', 'Go to projects', 'Open projects list'],
1302
- action: {
1303
- type: actionTypes.go,
1304
- to: `/account/projects`
1305
- }
1306
- }, {
1307
- id: 'go/manage-organizations',
1308
- text: intl.formatMessage(messages.openManageOrganizations),
1309
- keywords: ['Go to manage organizations', 'Go to organizations', 'Open organizations list'],
1310
- action: {
1311
- type: actionTypes.go,
1312
- to: `/account/organizations`
1313
- }
1314
- }, ...(applicationContext.user ? _mapInstanceProperty(_context11 = applicationContext.user.projects.results).call(_context11, userProject => ({
1315
- id: `go/project(${userProject.key})`,
1316
- text: intl.formatMessage(messages.useProject, {
1317
- projectName: userProject.name
1318
- }),
1319
- keywords: [userProject.key],
1320
- action: () => {
1321
- // Switching projects needs a full redirect so that
1322
- // the feature flags are reloaded (and things caches get destroyed)
1323
- window.location.href = `/${userProject.key}`;
1324
- }
1325
- })) : [])]).call(_context, nonNullable);
1326
- };
1327
-
1328
- const STORAGE_KEY = 'quickAccessHistoryEntries';
1329
- const saveHistoryEntries = historyEntries => {
1330
- try {
1331
- window.sessionStorage.setItem(STORAGE_KEY, _JSON$stringify(historyEntries));
1332
- return true;
1333
- } catch (error) {
1334
- return false;
1335
- }
1336
- };
1337
- const loadHistoryEntries = () => {
1338
- try {
1339
- const value = sessionStorage.getItem(STORAGE_KEY);
1340
- return value ? JSON.parse(value) : [];
1341
- } catch (error) {
1342
- return [];
1343
- }
1344
- };
1345
-
1346
- var QuickAccessProductQuery = { kind: "Document", definitions: [{ kind: "OperationDefinition", operation: "query", name: { kind: "Name", value: "QuickAccessProduct" }, variableDefinitions: [{ kind: "VariableDefinition", variable: { kind: "Variable", name: { kind: "Name", value: "productId" } }, type: { kind: "NonNullType", type: { kind: "NamedType", name: { kind: "Name", value: "String" } } }, directives: [] }], directives: [], selectionSet: { kind: "SelectionSet", selections: [{ kind: "Field", name: { kind: "Name", value: "product" }, arguments: [{ kind: "Argument", name: { kind: "Name", value: "id" }, value: { kind: "Variable", name: { kind: "Name", value: "productId" } } }], directives: [], selectionSet: { kind: "SelectionSet", selections: [{ kind: "Field", name: { kind: "Name", value: "id" }, arguments: [], directives: [] }, { kind: "Field", name: { kind: "Name", value: "masterData" }, arguments: [], directives: [], selectionSet: { kind: "SelectionSet", selections: [{ kind: "Field", name: { kind: "Name", value: "staged" }, arguments: [], directives: [], selectionSet: { kind: "SelectionSet", selections: [{ kind: "Field", name: { kind: "Name", value: "allVariants" }, arguments: [], directives: [], selectionSet: { kind: "SelectionSet", selections: [{ kind: "Field", name: { kind: "Name", value: "id" }, arguments: [], directives: [] }, { kind: "Field", name: { kind: "Name", value: "key" }, arguments: [], directives: [] }, { kind: "Field", name: { kind: "Name", value: "sku" }, arguments: [], directives: [] }] } }] } }] } }] } }] } }], loc: { start: 0, end: 208, source: { body: "query QuickAccessProduct($productId: String!) {\n product(id: $productId) {\n id\n masterData {\n staged {\n allVariants {\n id\n key\n sku\n }\n }\n }\n }\n}\n", name: "GraphQL request", locationOffset: { line: 1, column: 1 } } } };
1347
- const createProductVariantSubCommands = _ref => {
1348
- let intl = _ref.intl,
1349
- applicationContext = _ref.applicationContext,
1350
- productId = _ref.productId,
1351
- variantId = _ref.variantId;
1352
- const canViewProducts = hasSomePermissions([permissions.ViewProducts], applicationContext.permissions);
1353
- if (!canViewProducts || !applicationContext.project) return [];
1354
- return [{
1355
- id: `go/product(${productId})/variant(${variantId})/attributes`,
1356
- text: intl.formatMessage(messages.showProductVariantAttributes),
1357
- action: {
1358
- type: actionTypes.go,
1359
- to: oneLineTrim`
1360
- /${applicationContext.project.key}
1361
- /products
1362
- /${productId}
1363
- /variants
1364
- /${variantId}
1365
- /attributes
1366
- `
1367
- }
1368
- }, {
1369
- id: `go/product(${productId})/variant${variantId}/images`,
1370
- text: intl.formatMessage(messages.showProductVariantImages),
1371
- action: {
1372
- type: actionTypes.go,
1373
- to: oneLineTrim`
1374
- /${applicationContext.project.key}
1375
- /products
1376
- /${productId}
1377
- /variants
1378
- /${variantId}
1379
- /images
1380
- `
1381
- }
1382
- }, {
1383
- id: `go/product(${productId})/variant(${variantId})/prices`,
1384
- text: intl.formatMessage(messages.showProductVariantPrices),
1385
- action: {
1386
- type: actionTypes.go,
1387
- to: oneLineTrim`
1388
- /${applicationContext.project.key}
1389
- /products
1390
- /${productId}
1391
- /variants
1392
- /${variantId}
1393
- /prices
1394
- `
1395
- }
1396
- }, {
1397
- id: `go/product(${productId})/variant(${variantId})/inventory`,
1398
- text: intl.formatMessage(messages.showProductVariantInventory),
1399
- action: {
1400
- type: actionTypes.go,
1401
- to: oneLineTrim`
1402
- /${applicationContext.project.key}
1403
- /products
1404
- /${productId}
1405
- /variants
1406
- /${variantId}
1407
- /inventory
1408
- `
1409
- }
1410
- }];
1411
- };
1412
- const formatVariantMessage = (variant, intl) => {
1413
- if (variant.sku) return intl.formatMessage(messages.openVariantBySku, {
1414
- sku: variant.sku
1415
- });
1416
- if (variant.key) return intl.formatMessage(messages.openVariantByKey, {
1417
- key: variant.key
1418
- });
1419
- return intl.formatMessage(messages.openVariantById, {
1420
- id: variant.id
1421
- });
1422
- };
1423
- const createProductVariantListSubCommands = async _ref2 => {
1424
- let intl = _ref2.intl,
1425
- applicationContext = _ref2.applicationContext,
1426
- productId = _ref2.productId,
1427
- execQuery = _ref2.execQuery;
1428
- const canViewProducts = hasSomePermissions([permissions.ViewProducts], applicationContext.permissions);
1429
- if (!canViewProducts) return [];
1430
- const data = await execQuery(QuickAccessProductQuery, {
1431
- productId
1432
- }, {
1433
- target: GRAPHQL_TARGETS.COMMERCETOOLS_PLATFORM
1434
- });
1435
- if (data && data.product && data.product.masterData && data.product.masterData.staged && applicationContext.project) {
1436
- var _context;
1437
- const projectKey = applicationContext.project.key;
1438
- return _mapInstanceProperty(_context = data.product.masterData.staged.allVariants).call(_context, variant => ({
1439
- id: `go/product(${productId})/variant(${variant.id})`,
1440
- text: formatVariantMessage(variant, intl),
1441
- subCommands: createProductVariantSubCommands({
1442
- intl,
1443
- applicationContext,
1444
- productId,
1445
- variantId: variant.id
1446
- }),
1447
- action: {
1448
- type: actionTypes.go,
1449
- to: oneLineTrim`
1450
- /${projectKey}
1451
- /products
1452
- /${productId}
1453
- /variants
1454
- /${variant.id}
1455
- `
1456
- }
1457
- }));
1458
- }
1459
- return [];
1460
- };
1461
- const createProductTabsSubCommands = _ref3 => {
1462
- let intl = _ref3.intl,
1463
- applicationContext = _ref3.applicationContext,
1464
- productId = _ref3.productId;
1465
- const canViewProducts = hasSomePermissions([permissions.ViewProducts], applicationContext.permissions);
1466
- if (!canViewProducts || !applicationContext.project) return [];
1467
- return [{
1468
- id: `go/product(${productId})/general`,
1469
- text: intl.formatMessage(messages.openProductVariantGeneral),
1470
- action: {
1471
- type: actionTypes.go,
1472
- to: oneLineTrim`
1473
- /${applicationContext.project.key}
1474
- /products
1475
- /${productId}
1476
- /general
1477
- `
1478
- }
1479
- }, {
1480
- id: `go/product(${productId})/variants`,
1481
- text: intl.formatMessage(messages.openProductVariantList),
1482
- action: {
1483
- type: actionTypes.go,
1484
- to: oneLineTrim`
1485
- /${applicationContext.project.key}
1486
- /products
1487
- /${productId}
1488
- /variants
1489
- `
1490
- },
1491
- subCommands: execQuery => createProductVariantListSubCommands({
1492
- intl,
1493
- applicationContext,
1494
- productId,
1495
- execQuery
1496
- })
1497
- }, {
1498
- id: `go/product(${productId})/search`,
1499
- text: intl.formatMessage(messages.openProductVariantSearch),
1500
- action: {
1501
- type: actionTypes.go,
1502
- to: oneLineTrim`
1503
- /${applicationContext.project.key}
1504
- /products
1505
- /${productId}
1506
- /search
1507
- `
1508
- }
1509
- }];
1510
- };
1511
-
1512
- const sanitize = param => param
1513
- // Replace all \ with \\ (to prevent generate escape characters)
1514
- .replace(/\\/g, '\\\\')
1515
- // Replace all " with \"
1516
- .replace(/"/g, '\\"');
1517
- const flattenCommands = async (results, execQuery) => {
1518
- async function flatten(commands) {
1519
- return _reduceInstanceProperty(commands).call(commands, async (prevPromise, command) => {
1520
- const prevResults = await prevPromise;
1521
- if (command.subCommands) {
1522
- if (typeof command.subCommands === 'function') {
1523
- const subCommands = await command.subCommands(execQuery);
1524
- const flattenSubCommands = await flatten(subCommands);
1525
- return [...prevResults, command, ...flattenSubCommands];
1526
- }
1527
- const flattenSubCommands = await flatten(command.subCommands);
1528
- return [...prevResults, command, ...flattenSubCommands];
1529
- }
1530
- return [...prevResults, command];
1531
- }, _Promise.resolve([]));
1532
- }
1533
- return await flatten(results);
1534
- };
1535
-
1536
- // Once ui-kit exposes its fallback mechanism, we can use the same one here
1537
- const translate = (nameAllLocales, projectDataLocale) => {
1538
- const matchedTranslation = _findInstanceProperty(nameAllLocales).call(nameAllLocales, translation => translation.locale === projectDataLocale && translation.value);
1539
- if (matchedTranslation) return matchedTranslation.value;
1540
-
1541
- // Fall back to the first available locale
1542
- if (nameAllLocales.length > 0) return nameAllLocales[0].value;
1543
- return '';
1544
- };
1545
-
1546
- var QuickAccessQuery = { kind: "Document", definitions: [{ kind: "OperationDefinition", operation: "query", name: { kind: "Name", value: "QuickAccess" }, variableDefinitions: [{ kind: "VariableDefinition", variable: { kind: "Variable", name: { kind: "Name", value: "searchText" } }, type: { kind: "NonNullType", type: { kind: "NamedType", name: { kind: "Name", value: "String" } } }, directives: [] }, { kind: "VariableDefinition", variable: { kind: "Variable", name: { kind: "Name", value: "canViewProducts" } }, type: { kind: "NonNullType", type: { kind: "NamedType", name: { kind: "Name", value: "Boolean" } } }, directives: [] }, { kind: "VariableDefinition", variable: { kind: "Variable", name: { kind: "Name", value: "productsWhereClause" } }, type: { kind: "NamedType", name: { kind: "Name", value: "String" } }, directives: [] }, { kind: "VariableDefinition", variable: { kind: "Variable", name: { kind: "Name", value: "includeProductsByIds" } }, type: { kind: "NonNullType", type: { kind: "NamedType", name: { kind: "Name", value: "Boolean" } } }, directives: [] }], directives: [], selectionSet: { kind: "SelectionSet", selections: [{ kind: "Field", alias: { kind: "Name", value: "productsByIds" }, name: { kind: "Name", value: "products" }, arguments: [{ kind: "Argument", name: { kind: "Name", value: "where" }, value: { kind: "Variable", name: { kind: "Name", value: "productsWhereClause" } } }], directives: [{ kind: "Directive", name: { kind: "Name", value: "include" }, arguments: [{ kind: "Argument", name: { kind: "Name", value: "if" }, value: { kind: "Variable", name: { kind: "Name", value: "includeProductsByIds" } } }] }], selectionSet: { kind: "SelectionSet", selections: [{ kind: "Field", name: { kind: "Name", value: "results" }, arguments: [], directives: [], selectionSet: { kind: "SelectionSet", selections: [{ kind: "Field", name: { kind: "Name", value: "id" }, arguments: [], directives: [] }, { kind: "Field", name: { kind: "Name", value: "masterData" }, arguments: [], directives: [], selectionSet: { kind: "SelectionSet", selections: [{ kind: "Field", name: { kind: "Name", value: "staged" }, arguments: [], directives: [], selectionSet: { kind: "SelectionSet", selections: [{ kind: "Field", name: { kind: "Name", value: "nameAllLocales" }, arguments: [], directives: [], selectionSet: { kind: "SelectionSet", selections: [{ kind: "Field", name: { kind: "Name", value: "locale" }, arguments: [], directives: [] }, { kind: "Field", name: { kind: "Name", value: "value" }, arguments: [], directives: [] }] } }] } }] } }] } }] } }, { kind: "Field", alias: { kind: "Name", value: "productById" }, name: { kind: "Name", value: "product" }, arguments: [{ kind: "Argument", name: { kind: "Name", value: "id" }, value: { kind: "Variable", name: { kind: "Name", value: "searchText" } } }], directives: [{ kind: "Directive", name: { kind: "Name", value: "include" }, arguments: [{ kind: "Argument", name: { kind: "Name", value: "if" }, value: { kind: "Variable", name: { kind: "Name", value: "canViewProducts" } } }] }], selectionSet: { kind: "SelectionSet", selections: [{ kind: "Field", name: { kind: "Name", value: "id" }, arguments: [], directives: [] }, { kind: "Field", name: { kind: "Name", value: "masterData" }, arguments: [], directives: [], selectionSet: { kind: "SelectionSet", selections: [{ kind: "Field", name: { kind: "Name", value: "staged" }, arguments: [], directives: [], selectionSet: { kind: "SelectionSet", selections: [{ kind: "Field", name: { kind: "Name", value: "nameAllLocales" }, arguments: [], directives: [], selectionSet: { kind: "SelectionSet", selections: [{ kind: "Field", name: { kind: "Name", value: "locale" }, arguments: [], directives: [] }, { kind: "Field", name: { kind: "Name", value: "value" }, arguments: [], directives: [] }] } }] } }] } }] } }, { kind: "Field", alias: { kind: "Name", value: "productByKey" }, name: { kind: "Name", value: "product" }, arguments: [{ kind: "Argument", name: { kind: "Name", value: "key" }, value: { kind: "Variable", name: { kind: "Name", value: "searchText" } } }], directives: [{ kind: "Directive", name: { kind: "Name", value: "include" }, arguments: [{ kind: "Argument", name: { kind: "Name", value: "if" }, value: { kind: "Variable", name: { kind: "Name", value: "canViewProducts" } } }] }], selectionSet: { kind: "SelectionSet", selections: [{ kind: "Field", name: { kind: "Name", value: "id" }, arguments: [], directives: [] }, { kind: "Field", name: { kind: "Name", value: "masterData" }, arguments: [], directives: [], selectionSet: { kind: "SelectionSet", selections: [{ kind: "Field", name: { kind: "Name", value: "staged" }, arguments: [], directives: [], selectionSet: { kind: "SelectionSet", selections: [{ kind: "Field", name: { kind: "Name", value: "nameAllLocales" }, arguments: [], directives: [], selectionSet: { kind: "SelectionSet", selections: [{ kind: "Field", name: { kind: "Name", value: "locale" }, arguments: [], directives: [] }, { kind: "Field", name: { kind: "Name", value: "value" }, arguments: [], directives: [] }] } }] } }] } }] } }, { kind: "Field", alias: { kind: "Name", value: "productByVariantSku" }, name: { kind: "Name", value: "product" }, arguments: [{ kind: "Argument", name: { kind: "Name", value: "sku" }, value: { kind: "Variable", name: { kind: "Name", value: "searchText" } } }], directives: [{ kind: "Directive", name: { kind: "Name", value: "include" }, arguments: [{ kind: "Argument", name: { kind: "Name", value: "if" }, value: { kind: "Variable", name: { kind: "Name", value: "canViewProducts" } } }] }], selectionSet: { kind: "SelectionSet", selections: [{ kind: "Field", name: { kind: "Name", value: "id" }, arguments: [], directives: [] }, { kind: "Field", name: { kind: "Name", value: "masterData" }, arguments: [], directives: [], selectionSet: { kind: "SelectionSet", selections: [{ kind: "Field", name: { kind: "Name", value: "staged" }, arguments: [], directives: [], selectionSet: { kind: "SelectionSet", selections: [{ kind: "Field", name: { kind: "Name", value: "nameAllLocales" }, arguments: [], directives: [], selectionSet: { kind: "SelectionSet", selections: [{ kind: "Field", name: { kind: "Name", value: "locale" }, arguments: [], directives: [] }, { kind: "Field", name: { kind: "Name", value: "value" }, arguments: [], directives: [] }] } }, { kind: "Field", name: { kind: "Name", value: "variant" }, arguments: [{ kind: "Argument", name: { kind: "Name", value: "sku" }, value: { kind: "Variable", name: { kind: "Name", value: "searchText" } } }], directives: [], selectionSet: { kind: "SelectionSet", selections: [{ kind: "Field", name: { kind: "Name", value: "sku" }, arguments: [], directives: [] }, { kind: "Field", name: { kind: "Name", value: "key" }, arguments: [], directives: [] }, { kind: "Field", name: { kind: "Name", value: "id" }, arguments: [], directives: [] }] } }] } }] } }] } }, { kind: "Field", alias: { kind: "Name", value: "productByVariantKey" }, name: { kind: "Name", value: "product" }, arguments: [{ kind: "Argument", name: { kind: "Name", value: "variantKey" }, value: { kind: "Variable", name: { kind: "Name", value: "searchText" } } }], directives: [{ kind: "Directive", name: { kind: "Name", value: "include" }, arguments: [{ kind: "Argument", name: { kind: "Name", value: "if" }, value: { kind: "Variable", name: { kind: "Name", value: "canViewProducts" } } }] }], selectionSet: { kind: "SelectionSet", selections: [{ kind: "Field", name: { kind: "Name", value: "id" }, arguments: [], directives: [] }, { kind: "Field", name: { kind: "Name", value: "masterData" }, arguments: [], directives: [], selectionSet: { kind: "SelectionSet", selections: [{ kind: "Field", name: { kind: "Name", value: "staged" }, arguments: [], directives: [], selectionSet: { kind: "SelectionSet", selections: [{ kind: "Field", name: { kind: "Name", value: "nameAllLocales" }, arguments: [], directives: [], selectionSet: { kind: "SelectionSet", selections: [{ kind: "Field", name: { kind: "Name", value: "locale" }, arguments: [], directives: [] }, { kind: "Field", name: { kind: "Name", value: "value" }, arguments: [], directives: [] }] } }, { kind: "Field", name: { kind: "Name", value: "variant" }, arguments: [{ kind: "Argument", name: { kind: "Name", value: "key" }, value: { kind: "Variable", name: { kind: "Name", value: "searchText" } } }], directives: [], selectionSet: { kind: "SelectionSet", selections: [{ kind: "Field", name: { kind: "Name", value: "sku" }, arguments: [], directives: [] }, { kind: "Field", name: { kind: "Name", value: "key" }, arguments: [], directives: [] }, { kind: "Field", name: { kind: "Name", value: "id" }, arguments: [], directives: [] }] } }] } }] } }] } }] } }], loc: { start: 0, end: 1407, source: { body: "query QuickAccess(\n $searchText: String!\n $canViewProducts: Boolean!\n $productsWhereClause: String\n $includeProductsByIds: Boolean!\n) {\n productsByIds: products(where: $productsWhereClause)\n @include(if: $includeProductsByIds) {\n results {\n id\n masterData {\n staged {\n nameAllLocales {\n locale\n value\n }\n }\n }\n }\n }\n\n productById: product(id: $searchText) @include(if: $canViewProducts) {\n id\n masterData {\n staged {\n nameAllLocales {\n locale\n value\n }\n }\n }\n }\n\n productByKey: product(key: $searchText) @include(if: $canViewProducts) {\n id\n masterData {\n staged {\n nameAllLocales {\n locale\n value\n }\n }\n }\n }\n\n productByVariantSku: product(sku: $searchText)\n @include(if: $canViewProducts) {\n id\n masterData {\n staged {\n nameAllLocales {\n locale\n value\n }\n variant(sku: $searchText) {\n sku\n key\n id\n }\n }\n }\n }\n\n productByVariantKey: product(variantKey: $searchText)\n @include(if: $canViewProducts) {\n id\n masterData {\n staged {\n nameAllLocales {\n locale\n value\n }\n variant(key: $searchText) {\n sku\n key\n id\n }\n }\n }\n }\n}\n", name: "GraphQL request", locationOffset: { line: 1, column: 1 } } } };
1547
- const searchProductIdsAction = (searchText, projectKey, dataLocale) => actions.post({
1548
- uri: `/${projectKey}/search/products`,
1549
- mcApiProxyTarget: MC_API_PROXY_TARGETS.PIM_SEARCH,
1550
- payload: {
1551
- query: {
1552
- fullText: {
1553
- field: 'name',
1554
- language: dataLocale,
1555
- value: searchText
1556
- }
1557
- },
1558
- sort: [{
1559
- field: 'name',
1560
- language: dataLocale,
1561
- order: 'desc'
1562
- }],
1563
- limit: 9,
1564
- offset: 0
1565
- }
1566
- });
1567
- const pimIndexerStatusAction = (projectKey, dataLocale) =>
1568
- // TODO this should be sdkActions.head()
1569
- // and then we should check whether the response code is
1570
- // - 200 meaning the project is indexed
1571
- // - 404 meaning the project is not indexed
1572
- //
1573
- // But there is a problem in tne node-sdk client as it tries to
1574
- // .json()-parse the response to HEAD requests which results in an
1575
- // error, so we send a regular request for now and limit to no results
1576
- // instead to keep the payload minimal
1577
- actions.post({
1578
- uri: `/${projectKey}/search/products`,
1579
- mcApiProxyTarget: MC_API_PROXY_TARGETS.PIM_SEARCH,
1580
- payload: {
1581
- query: {
1582
- fullText: {
1583
- field: 'name',
1584
- language: dataLocale,
1585
- value: 'availability-check'
1586
- }
1587
- },
1588
- limit: 0,
1589
- offset: 0
1590
- }
1591
- });
1592
- const QuickAccess = props => {
1593
- const _useState = useState(loadHistoryEntries()),
1594
- _useState2 = _slicedToArray(_useState, 2),
1595
- historyEntries = _useState2[0],
1596
- setHistoryEntries = _useState2[1];
1597
- const handleHistoryEntriesChange = useCallback(entries => {
1598
- // Keep the history in sync with the session storage
1599
- saveHistoryEntries(entries);
1600
- setHistoryEntries(entries);
1601
- }, []);
1602
- const history = useHistory();
1603
- const apolloClient = useApolloClient();
1604
- const intl = useIntl();
1605
- const _useFeatureToggles = useFeatureToggles({
1606
- pimSearch: true,
1607
- customApplications: true,
1608
- canViewDashboard: true
1609
- }),
1610
- _useFeatureToggles2 = _slicedToArray(_useFeatureToggles, 3),
1611
- isPimSearchEnabled = _useFeatureToggles2[0],
1612
- isCustomApplicationsEnabled = _useFeatureToggles2[1],
1613
- isCanViewDashboardEnabled = _useFeatureToggles2[2];
1614
- const applicationContext = useApplicationContext();
1615
-
1616
- // Destructure functions from props to reference them in the hook dependency list
1617
- const onPimIndexerStateChangeFromParent = props.onPimIndexerStateChange;
1618
- const dispatchFetchProductIds = useAsyncDispatch();
1619
- const fetchPimSearchProductIds = useCallback(async searchText => {
1620
- if (applicationContext.project && applicationContext.dataLocale) {
1621
- var _context;
1622
- const result = await dispatchFetchProductIds(searchProductIdsAction(searchText, applicationContext.project.key, applicationContext.dataLocale));
1623
- return result && result.hits ? _mapInstanceProperty(_context = result.hits).call(_context, hit => hit.id) : [];
1624
- }
1625
- return [];
1626
- }, [applicationContext.dataLocale, applicationContext.project, dispatchFetchProductIds]);
1627
- const dispatchFetchPimIndexerStatus = useAsyncDispatch();
1628
- const fetchPimIndexerStatus = useCallback(async () => {
1629
- if (applicationContext.project && applicationContext.dataLocale) {
1630
- try {
1631
- dispatchFetchPimIndexerStatus(pimIndexerStatusAction(applicationContext.project.key, applicationContext.dataLocale));
1632
- return pimIndexerStates.INDEXED;
1633
- } catch (error) {
1634
- // eslint-disable-next-line no-console
1635
- if (process.env.NODE_ENV !== 'production') console.error(error);
1636
- // project is not using pim-indexer when response error code is 404,
1637
- // but we treat all errors as non-indexed as a safe guard, so we're
1638
- // not checking the response error code at all
1639
- return pimIndexerStates.NOT_INDEXED;
1640
- }
1641
- }
1642
- return pimIndexerStates.NOT_INDEXED;
1643
- }, [applicationContext.dataLocale, applicationContext.project, dispatchFetchPimIndexerStatus]);
1644
- const getProjectIndexStatus = useCallback(async () => {
1645
- // skip when there is no project
1646
- if (!applicationContext.project) return pimIndexerStates.NOT_INDEXED;
1647
- const canViewProducts = hasSomePermissions([permissions.ViewProducts], applicationContext.permissions);
1648
-
1649
- // skip checking when user can't view products anyways
1650
- if (!canViewProducts) return pimIndexerStates.NOT_INDEXED;
1651
- return await fetchPimIndexerStatus();
1652
- }, [applicationContext.permissions, applicationContext.project, fetchPimIndexerStatus]);
1653
- useEffect(() => {
1654
- if (props.pimIndexerState === pimIndexerStates.UNCHECKED) {
1655
- getProjectIndexStatus().then(status => {
1656
- onPimIndexerStateChangeFromParent(status);
1657
- });
1658
- }
1659
- // eslint-disable-next-line react-hooks/exhaustive-deps
1660
- }, []); // <-- run only once, when component mounts
1661
-
1662
- const execQuery = useCallback((Query, variables, context) => apolloClient.query({
1663
- query: Query,
1664
- errorPolicy: 'ignore',
1665
- variables,
1666
- context
1667
- }).then(response => response.data), [apolloClient]);
1668
- const getNextCommands = useCallback(async command => {
1669
- if (!command.subCommands) return [];
1670
- if (_Array$isArray(command.subCommands)) return command.subCommands;
1671
- return await command.subCommands(execQuery);
1672
- }, [execQuery]);
1673
- const getProjectCommands = useCallback(async searchText => {
1674
- const idsOfProductsMatchingSearchText = props.pimIndexerState === pimIndexerStates.INDEXED ? await fetchPimSearchProductIds(searchText) : [];
1675
- const canViewProducts = hasSomePermissions([permissions.ViewProducts], applicationContext.permissions);
1676
- const data = await execQuery(QuickAccessQuery, {
1677
- searchText: sanitize(searchText),
1678
- // Pass conditional arguments to disable some of the queries
1679
- canViewProducts,
1680
- productsWhereClause: `id in (${_mapInstanceProperty(idsOfProductsMatchingSearchText).call(idsOfProductsMatchingSearchText, id => _JSON$stringify(id)).join(', ')})`,
1681
- includeProductsByIds: Boolean(canViewProducts && idsOfProductsMatchingSearchText.length > 0)
1682
- }, {
1683
- target: GRAPHQL_TARGETS.COMMERCETOOLS_PLATFORM
1684
- });
1685
- const commands = [];
1686
- if (data && data.productByVariantKey && data.productByVariantKey.masterData && data.productByVariantKey.masterData.staged && data.productByVariantKey.masterData.staged.variant && applicationContext.project && applicationContext.dataLocale) {
1687
- const productId = data.productByVariantKey.id;
1688
- const variantId = data.productByVariantKey.masterData.staged.variant.id;
1689
- const variantKey = data.productByVariantKey.masterData.staged.variant.key;
1690
- commands.push({
1691
- id: `go/product-variant-by-key/product(${productId}/variant(${variantId})`,
1692
- text: intl.formatMessage(messages.showProductVariant, {
1693
- variantName: translate(data.productByVariantKey.masterData.staged.nameAllLocales, applicationContext.dataLocale)
1694
- }),
1695
- keywords: variantKey ? [variantKey] : undefined,
1696
- action: {
1697
- type: actionTypes.go,
1698
- to: oneLineTrim`
1699
- /${applicationContext.project.key}
1700
- /products
1701
- /${productId}
1702
- /variants
1703
- /${variantId}
1704
- `
1705
- },
1706
- subCommands: createProductVariantSubCommands({
1707
- intl,
1708
- applicationContext,
1709
- productId,
1710
- variantId
1711
- })
1712
- });
1713
- }
1714
- if (data && data.productByVariantSku && data.productByVariantSku.masterData && data.productByVariantSku.masterData.staged && data.productByVariantSku.masterData.staged.variant && applicationContext.project && applicationContext.dataLocale) {
1715
- const productId = data.productByVariantSku.id;
1716
- const variantId = data.productByVariantSku.masterData.staged.variant.id;
1717
- commands.push({
1718
- id: `go/product-variant-by-sku/product(${productId})/variant(${variantId})`,
1719
- text: intl.formatMessage(messages.showProductVariant, {
1720
- variantName: data.productByVariantSku.masterData.staged.variant.sku
1721
- }),
1722
- action: {
1723
- type: actionTypes.go,
1724
- to: oneLineTrim`
1725
- /${applicationContext.project.key}
1726
- /products
1727
- /${productId}
1728
- /variants
1729
- /${variantId}
1730
- `
1731
- },
1732
- subCommands: createProductVariantSubCommands({
1733
- intl,
1734
- applicationContext,
1735
- productId,
1736
- variantId
1737
- })
1738
- });
1739
- }
1740
- if (data && data.productById && data.productById.masterData && data.productById.masterData.staged && data.productById.masterData.staged.nameAllLocales && applicationContext.project && applicationContext.dataLocale) {
1741
- const productId = data.productById.id;
1742
- commands.push({
1743
- id: `go/product-by-id/product(${productId})`,
1744
- text: intl.formatMessage(messages.showProduct, {
1745
- productName: translate(data.productById.masterData.staged.nameAllLocales, applicationContext.dataLocale)
1746
- }),
1747
- keywords: [productId],
1748
- action: {
1749
- type: actionTypes.go,
1750
- to: `/${applicationContext.project.key}/products/${productId}`
1751
- },
1752
- subCommands: createProductTabsSubCommands({
1753
- intl,
1754
- applicationContext,
1755
- productId
1756
- })
1757
- });
1758
- }
1759
- if (data && data.productsByIds && data.productsByIds.results) {
1760
- var _context2;
1761
- _forEachInstanceProperty(_context2 = data.productsByIds.results).call(_context2, product => {
1762
- if (product.masterData.staged && applicationContext.project && applicationContext.dataLocale) {
1763
- commands.push({
1764
- id: `go/product-by-search-text/product(${product.id})`,
1765
- text: intl.formatMessage(messages.showProduct, {
1766
- productName: translate(product.masterData.staged.nameAllLocales, applicationContext.dataLocale)
1767
- }),
1768
- keywords: [product.id],
1769
- action: {
1770
- type: actionTypes.go,
1771
- to: `/${applicationContext.project.key}/products/${product.id}`
1772
- },
1773
- subCommands: createProductTabsSubCommands({
1774
- intl,
1775
- applicationContext,
1776
- productId: product.id
1777
- })
1778
- });
1779
- }
1780
- });
1781
- }
1782
- if (data && data.productByKey && applicationContext.project && applicationContext.dataLocale) {
1783
- const productId = data.productByKey.id;
1784
- commands.push({
1785
- id: `go/product-by-key/product(${productId})`,
1786
- text: intl.formatMessage(messages.showProduct, {
1787
- productName: searchText
1788
- }),
1789
- action: {
1790
- type: actionTypes.go,
1791
- to: `/${applicationContext.project.key}/products/${productId}`
1792
- },
1793
- subCommands: createProductTabsSubCommands({
1794
- intl,
1795
- applicationContext,
1796
- productId
1797
- })
1798
- });
1799
- }
1800
- return commands;
1801
- }, [applicationContext, execQuery, fetchPimSearchProductIds, intl, props.pimIndexerState]);
1802
- const debouncedGetProjectCommands = debounce(getProjectCommands, 200, {
1803
- cancelObj: 'canceled'
1804
- });
1805
- const search = useCallback(async searchText => {
1806
- const generalCommands = createCommands({
1807
- applicationContext,
1808
- changeProjectDataLocale: props.onChangeProjectDataLocale,
1809
- intl,
1810
- featureToggles: {
1811
- pimSearch: isPimSearchEnabled,
1812
- customApplications: isCustomApplicationsEnabled,
1813
- canViewDashboard: isCanViewDashboardEnabled
1814
- }
1815
- });
1816
- if (!applicationContext.project) return generalCommands;
1817
-
1818
- // Avoid searching for short texts, as we won't get any good results
1819
- // anyways. This results in commands popping up immediately when the user
1820
- // starts typing, which gives the whole search a much more repsonsive
1821
- // feeling.
1822
- if (_trimInstanceProperty(searchText).call(searchText).length < 3) return generalCommands;
1823
- try {
1824
- const projectCommands = await debouncedGetProjectCommands(searchText);
1825
- const allCommands = [...generalCommands, ...projectCommands];
1826
- return await flattenCommands(allCommands, execQuery);
1827
- } catch (error) {
1828
- // When the debounced search is canceled, it throws with "canceled"
1829
- // In that case we know that another search is going to happen
1830
- // and we just resolve with the general commands.
1831
- if (error === 'canceled') return generalCommands;
1832
- throw error;
1833
- }
1834
- }, [applicationContext, debouncedGetProjectCommands, execQuery, intl, isCanViewDashboardEnabled, isCustomApplicationsEnabled, isPimSearchEnabled, props.onChangeProjectDataLocale]);
1835
- const executeCommand = useCallback((command, meta) => {
1836
- var _context3;
1837
- if (typeof command.action === 'function') {
1838
- // Idea: We could handle these errors and set them on status bar of Butler
1839
- // We can also handle sync/async commands by checking command.action.then
1840
- command.action();
1841
- return;
1842
- }
1843
- // open in new window
1844
- // and always open other pages in a new window
1845
- if (meta.openInNewTab || !_startsWithInstanceProperty(_context3 = command.action.to).call(_context3, '/')) {
1846
- // eslint-disable-next-line no-restricted-globals
1847
- open(command.action.to, '_blank');
1848
- } else if (applicationContext.environment.useFullRedirectsForLinks) {
1849
- location.replace(command.action.to);
1850
- } else {
1851
- history.push(command.action.to);
1852
- }
1853
- }, [applicationContext.environment.useFullRedirectsForLinks, history]);
1854
- return jsx(ButlerWithAnimation, {
1855
- historyEntries: historyEntries,
1856
- onHistoryEntriesChange: handleHistoryEntriesChange,
1857
- search: search,
1858
- executeCommand: executeCommand,
1859
- onClose: props.onClose,
1860
- getNextCommands: getNextCommands
1861
- });
1862
- };
1863
- QuickAccess.displayName = 'QuickAccess';
1864
-
1865
- export { QuickAccess as default };