@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.
- package/dist/{application-entry-point-c87294b0.cjs.dev.js → application-entry-point-18d8fba0.cjs.dev.js} +7 -4
- package/dist/{application-entry-point-10a5e1a5.esm.js → application-entry-point-1b23fb6b.esm.js} +6 -4
- package/dist/{application-entry-point-8c4b8e53.cjs.prod.js → application-entry-point-74a06151.cjs.prod.js} +4 -3
- package/dist/commercetools-frontend-application-shell.cjs.dev.js +12 -11
- package/dist/commercetools-frontend-application-shell.cjs.prod.js +12 -11
- package/dist/commercetools-frontend-application-shell.esm.js +12 -11
- package/dist/{custom-view-dev-host-5039dc1d.esm.js → custom-view-dev-host-091163ea.esm.js} +20 -16
- package/dist/{custom-view-dev-host-b5e3a16a.cjs.dev.js → custom-view-dev-host-17daf42a.cjs.dev.js} +21 -16
- package/dist/{custom-view-dev-host-21561a3a.cjs.prod.js → custom-view-dev-host-a682a499.cjs.prod.js} +21 -16
- package/dist/declarations-connectors/src/index.d.ts +1 -1
- package/dist/declarations-connectors/src/utils/index.d.ts +1 -0
- package/dist/declarations-connectors/src/utils/select-user-language-from-storage/index.d.ts +1 -0
- package/dist/declarations-connectors/src/utils/select-user-language-from-storage/select-user-language-from-storage.d.ts +1 -0
- package/dist/{index-17f02024.cjs.dev.js → index-1d1cc31f.cjs.dev.js} +225 -368
- package/dist/{index-e4de734f.esm.js → index-1dadca21.esm.js} +227 -368
- package/dist/{index-614accc4.cjs.dev.js → index-25183095.cjs.dev.js} +2 -2
- package/dist/{index-11b385bb.cjs.prod.js → index-3cfc1f1e.cjs.prod.js} +213 -348
- package/dist/{index-245e2980.cjs.prod.js → index-52c724ed.cjs.prod.js} +2 -2
- package/dist/{index-86039df7.esm.js → index-5aaa33bb.esm.js} +2 -2
- package/dist/{navbar-022383bd.cjs.dev.js → navbar-586f7774.cjs.dev.js} +111 -92
- package/dist/{navbar-844d350d.esm.js → navbar-88e0fd1f.esm.js} +110 -92
- package/dist/{navbar-acc2cd1b.cjs.prod.js → navbar-93183a2d.cjs.prod.js} +111 -92
- package/dist/oidc-258fc018.cjs.prod.js +115 -0
- package/dist/oidc-35e8e62a.esm.js +100 -0
- package/dist/oidc-87d116c1.cjs.dev.js +115 -0
- package/dist/{oidc-callback-c014b1b0.esm.js → oidc-callback-019d623d.esm.js} +16 -14
- package/dist/{oidc-callback-ad64d7f6.cjs.dev.js → oidc-callback-47743232.cjs.dev.js} +16 -14
- package/dist/{oidc-callback-9beece27.cjs.prod.js → oidc-callback-6bdb3c6f.cjs.prod.js} +16 -14
- package/dist/{project-container-3e3c7013.cjs.dev.js → project-container-2245f020.cjs.dev.js} +53 -19
- package/dist/{project-container-e11b2fc6.esm.js → project-container-7fce9e66.esm.js} +52 -19
- package/dist/{project-container-f1710162.cjs.prod.js → project-container-954dbf0f.cjs.prod.js} +53 -19
- package/dist/{project-expired-39589063.esm.js → project-expired-1b0845c5.esm.js} +12 -11
- package/dist/{project-expired-f29e6d47.cjs.dev.js → project-expired-c941b592.cjs.dev.js} +12 -11
- package/dist/{project-expired-59169760.cjs.prod.js → project-expired-ee8b232c.cjs.prod.js} +12 -11
- package/dist/{project-not-found-88730a64.esm.js → project-not-found-340217f6.esm.js} +11 -10
- package/dist/{project-not-found-d968ede6.cjs.dev.js → project-not-found-9b7cfe88.cjs.dev.js} +11 -10
- package/dist/{project-not-found-625f0e91.cjs.prod.js → project-not-found-9cee9625.cjs.prod.js} +11 -10
- package/dist/{project-not-initialized-6d69541c.esm.js → project-not-initialized-55fd8df4.esm.js} +12 -11
- package/dist/{project-not-initialized-22d54dab.cjs.prod.js → project-not-initialized-7a058b68.cjs.prod.js} +12 -11
- package/dist/{project-not-initialized-f346dc17.cjs.dev.js → project-not-initialized-7b3843a3.cjs.dev.js} +12 -11
- package/dist/{project-suspended-d48e7d51.cjs.prod.js → project-suspended-12618898.cjs.prod.js} +11 -10
- package/dist/{project-suspended-be2e3265.esm.js → project-suspended-529b09d6.esm.js} +11 -10
- package/dist/{project-suspended-6a886974.cjs.dev.js → project-suspended-78e94b85.cjs.dev.js} +11 -10
- package/dist/{redirect-to-login-12f467b8.cjs.prod.js → redirect-to-login-3e4a6434.cjs.prod.js} +13 -10
- package/dist/{redirect-to-login-3bee13ba.cjs.dev.js → redirect-to-login-66ea895a.cjs.dev.js} +13 -10
- package/dist/{redirect-to-login-2944c890.esm.js → redirect-to-login-edbfacbc.esm.js} +13 -10
- package/dist/{redirect-to-logout-645e12ca.cjs.prod.js → redirect-to-logout-52a7810f.cjs.prod.js} +14 -12
- package/dist/{redirect-to-logout-0196921c.esm.js → redirect-to-logout-5d5fc361.esm.js} +14 -12
- package/dist/{redirect-to-logout-477ea146.cjs.dev.js → redirect-to-logout-b331b037.cjs.dev.js} +14 -12
- package/dist/{redirector-72ccfbc2.cjs.dev.js → redirector-0efdd994.cjs.dev.js} +4 -3
- package/dist/{redirector-d856975f.esm.js → redirector-656c6ee7.esm.js} +4 -3
- package/dist/{redirector-0c72d0a4.cjs.prod.js → redirector-c858d578.cjs.prod.js} +4 -3
- package/dist/{requests-in-flight-loader-82b93073.esm.js → requests-in-flight-loader-20021ccc.esm.js} +11 -10
- package/dist/{requests-in-flight-loader-08cfa2ce.cjs.prod.js → requests-in-flight-loader-64d2e12d.cjs.prod.js} +11 -10
- package/dist/{requests-in-flight-loader-fb6a69f6.cjs.dev.js → requests-in-flight-loader-83cab813.cjs.dev.js} +11 -10
- package/dist/{service-page-project-switcher-2d65c6f7.cjs.dev.js → service-page-project-switcher-49dabe13.cjs.dev.js} +1 -1
- package/dist/{service-page-project-switcher-1e41f587.esm.js → service-page-project-switcher-6cdd506b.esm.js} +1 -1
- package/dist/{service-page-project-switcher-2746dbcc.cjs.prod.js → service-page-project-switcher-f1b43eb7.cjs.prod.js} +1 -1
- package/dist/{use-applications-menu-823a2492.cjs.dev.js → use-applications-menu-48d924bd.cjs.prod.js} +47 -39
- package/dist/{use-applications-menu-14a5a1f4.cjs.prod.js → use-applications-menu-7f548a7a.cjs.dev.js} +47 -39
- package/dist/{use-applications-menu-1514af11.esm.js → use-applications-menu-b871849c.esm.js} +44 -37
- package/dist/{user-settings-menu-d75f4958.cjs.prod.js → user-settings-menu-6660f508.cjs.prod.js} +29 -22
- package/dist/{user-settings-menu-f98bea89.esm.js → user-settings-menu-afa82f2a.esm.js} +29 -22
- package/dist/{user-settings-menu-6113cdd3.cjs.dev.js → user-settings-menu-f5c74042.cjs.dev.js} +29 -22
- package/package.json +22 -22
- package/ssr/dist/commercetools-frontend-application-shell-ssr.cjs.dev.js +2 -1
- package/ssr/dist/commercetools-frontend-application-shell-ssr.cjs.prod.js +2 -1
- package/ssr/dist/commercetools-frontend-application-shell-ssr.esm.js +2 -1
- package/test-utils/dist/commercetools-frontend-application-shell-test-utils.cjs.dev.js +26 -22
- package/test-utils/dist/commercetools-frontend-application-shell-test-utils.cjs.prod.js +26 -22
- package/test-utils/dist/commercetools-frontend-application-shell-test-utils.esm.js +20 -17
- package/dist/oidc-8827f9fe.cjs.dev.js +0 -98
- package/dist/oidc-b2520905.esm.js +0 -84
- package/dist/oidc-d74e6aa2.cjs.prod.js +0 -98
- package/dist/quick-access-67db2a39.cjs.dev.js +0 -1893
- package/dist/quick-access-8c34e976.esm.js +0 -1865
- 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 };
|