@dxc-technology/halstack-react 12.0.2 → 12.2.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (87) hide show
  1. package/HalstackContext.d.ts +23 -0
  2. package/HalstackContext.js +31 -21
  3. package/breadcrumbs/Breadcrumbs.stories.tsx +1 -1
  4. package/breadcrumbs/Breadcrumbs.test.js +1 -2
  5. package/breadcrumbs/types.d.ts +24 -0
  6. package/card/Card.stories.tsx +1 -1
  7. package/chip/Chip.stories.tsx +2 -6
  8. package/common/coreTokens.d.ts +0 -1
  9. package/common/coreTokens.js +0 -1
  10. package/common/variables.d.ts +29 -0
  11. package/common/variables.js +25 -2
  12. package/container/types.d.ts +113 -11
  13. package/contextual-menu/ContextualMenu.js +56 -8
  14. package/contextual-menu/ContextualMenu.stories.tsx +50 -51
  15. package/contextual-menu/ContextualMenu.test.js +108 -66
  16. package/contextual-menu/GroupItem.js +9 -9
  17. package/contextual-menu/ItemAction.d.ts +1 -1
  18. package/contextual-menu/ItemAction.js +52 -15
  19. package/contextual-menu/SingleItem.d.ts +1 -1
  20. package/contextual-menu/SingleItem.js +14 -14
  21. package/contextual-menu/types.d.ts +13 -6
  22. package/date-input/Calendar.js +46 -30
  23. package/date-input/DateInput.accessibility.test.js +12 -13
  24. package/date-input/DateInput.js +9 -3
  25. package/date-input/DateInput.stories.tsx +5 -6
  26. package/date-input/DateInput.test.js +30 -31
  27. package/date-input/YearPicker.js +9 -4
  28. package/dialog/Dialog.stories.tsx +2 -3
  29. package/dialog/Dialog.test.js +1 -2
  30. package/divider/Divider.stories.tsx +2 -1
  31. package/dropdown/Dropdown.accessibility.test.js +1 -2
  32. package/dropdown/Dropdown.stories.tsx +1 -1
  33. package/dropdown/Dropdown.test.js +1 -2
  34. package/file-input/FileInput.js +1 -3
  35. package/file-input/FileItem.js +1 -1
  36. package/file-input/types.d.ts +0 -4
  37. package/grid/Grid.stories.tsx +3 -1
  38. package/header/Header.stories.tsx +1 -1
  39. package/icon/Icon.stories.tsx +1 -1
  40. package/number-input/NumberInput.accessibility.test.js +1 -2
  41. package/number-input/NumberInput.js +5 -2
  42. package/number-input/NumberInput.stories.tsx +11 -16
  43. package/number-input/NumberInput.test.js +1 -2
  44. package/package.json +19 -16
  45. package/paginator/Paginator.accessibility.test.js +1 -2
  46. package/paginator/Paginator.test.js +1 -2
  47. package/password-input/PasswordInput.accessibility.test.js +1 -2
  48. package/password-input/PasswordInput.js +6 -3
  49. package/password-input/PasswordInput.stories.tsx +10 -1
  50. package/password-input/PasswordInput.test.js +1 -2
  51. package/resultset-table/ResultsetTable.accessibility.test.js +1 -2
  52. package/resultset-table/ResultsetTable.js +33 -9
  53. package/resultset-table/ResultsetTable.stories.tsx +2 -2
  54. package/resultset-table/ResultsetTable.test.js +102 -32
  55. package/resultset-table/types.d.ts +3 -2
  56. package/select/Select.accessibility.test.js +1 -2
  57. package/select/Select.js +3 -1
  58. package/select/Select.stories.tsx +11 -2
  59. package/select/Select.test.js +1 -2
  60. package/sidenav/Sidenav.stories.tsx +1 -1
  61. package/slider/Slider.accessibility.test.js +1 -2
  62. package/slider/Slider.test.js +1 -2
  63. package/table/Table.accessibility.test.js +1 -2
  64. package/table/Table.stories.tsx +1 -1
  65. package/table/Table.test.js +1 -2
  66. package/tabs/Tabs.stories.tsx +2 -2
  67. package/tag/Tag.stories.tsx +1 -1
  68. package/text-input/TextInput.accessibility.test.js +1 -2
  69. package/text-input/TextInput.js +3 -1
  70. package/text-input/TextInput.stories.tsx +11 -8
  71. package/text-input/TextInput.test.js +1 -2
  72. package/toggle-group/ToggleGroup.stories.tsx +1 -1
  73. package/tooltip/Tooltip.accessibility.test.d.ts +1 -0
  74. package/tooltip/Tooltip.accessibility.test.js +144 -0
  75. package/tooltip/Tooltip.d.ts +4 -0
  76. package/tooltip/Tooltip.js +50 -0
  77. package/tooltip/Tooltip.stories.tsx +111 -0
  78. package/tooltip/Tooltip.test.d.ts +1 -0
  79. package/tooltip/Tooltip.test.js +112 -0
  80. package/tooltip/types.d.ts +16 -0
  81. package/tooltip/types.js +5 -0
  82. package/typography/Typography.stories.tsx +1 -3
  83. package/typography/Typography.test.js +23 -0
  84. package/useTheme.d.ts +23 -0
  85. package/utils/BaseTypography.js +44 -40
  86. package/utils/FocusLock.js +2 -1
  87. package/wizard/Wizard.stories.tsx +1 -1
@@ -10,11 +10,12 @@ var _taggedTemplateLiteral2 = _interopRequireDefault(require("@babel/runtime/hel
10
10
  var _slicedToArray2 = _interopRequireDefault(require("@babel/runtime/helpers/slicedToArray"));
11
11
  var _defineProperty2 = _interopRequireDefault(require("@babel/runtime/helpers/defineProperty"));
12
12
  var _react = _interopRequireWildcard(require("react"));
13
- var _styledComponents = _interopRequireDefault(require("styled-components"));
13
+ var _styledComponents = _interopRequireWildcard(require("styled-components"));
14
14
  var _coreTokens = _interopRequireDefault(require("../common/coreTokens"));
15
15
  var _Divider = _interopRequireDefault(require("../divider/Divider"));
16
16
  var _Inset = _interopRequireDefault(require("../inset/Inset"));
17
17
  var _MenuItem = _interopRequireDefault(require("./MenuItem"));
18
+ var _useTheme = _interopRequireDefault(require("../useTheme"));
18
19
  var _templateObject, _templateObject2, _templateObject3;
19
20
  function _getRequireWildcardCache(e) { if ("function" != typeof WeakMap) return null; var r = new WeakMap(), t = new WeakMap(); return (_getRequireWildcardCache = function _getRequireWildcardCache(e) { return e ? t : r; })(e); }
20
21
  function _interopRequireWildcard(e, r) { if (!r && e && e.__esModule) return e; if (null === e || "object" != _typeof(e) && "function" != typeof e) return { "default": e }; var t = _getRequireWildcardCache(r); if (t && t.has(e)) return t.get(e); var n = { __proto__: null }, a = Object.defineProperty && Object.getOwnPropertyDescriptor; for (var u in e) if ("default" !== u && {}.hasOwnProperty.call(e, u)) { var i = a ? Object.getOwnPropertyDescriptor(e, u) : null; i && (i.get || i.set) ? Object.defineProperty(n, u, i) : n[u] = e[u]; } return n["default"] = e, t && t.set(e, n), n; }
@@ -48,15 +49,20 @@ var DxcContextualMenu = function DxcContextualMenu(_ref) {
48
49
  _useState2 = (0, _slicedToArray2["default"])(_useState, 2),
49
50
  selectedItemId = _useState2[0],
50
51
  setSelectedItemId = _useState2[1];
52
+ var contextualMenuRef = (0, _react.useRef)(null);
51
53
  var itemsWithId = (0, _react.useMemo)(function () {
52
54
  return addIdToItems(items);
53
55
  }, [items]);
56
+ var colorsTheme = (0, _useTheme["default"])();
54
57
  var renderSection = function renderSection(section, currentSectionIndex, length) {
55
58
  return /*#__PURE__*/_react["default"].createElement(_react.Fragment, {
56
59
  key: "section-".concat(currentSectionIndex)
57
60
  }, /*#__PURE__*/_react["default"].createElement("li", {
58
- role: "group"
59
- }, section.title != null && /*#__PURE__*/_react["default"].createElement(Title, null, section.title), /*#__PURE__*/_react["default"].createElement(SectionList, null, section.items.map(function (item, index) {
61
+ role: "group",
62
+ "aria-labelledby": section.title
63
+ }, section.title != null && /*#__PURE__*/_react["default"].createElement(Title, {
64
+ id: section.title
65
+ }, section.title), /*#__PURE__*/_react["default"].createElement(SectionList, null, section.items.map(function (item, index) {
60
66
  return /*#__PURE__*/_react["default"].createElement(_MenuItem["default"], {
61
67
  item: item,
62
68
  key: "".concat(item.label, "-").concat(index)
@@ -68,8 +74,26 @@ var DxcContextualMenu = function DxcContextualMenu(_ref) {
68
74
  color: "lightGrey"
69
75
  })));
70
76
  };
71
- return /*#__PURE__*/_react["default"].createElement(ContextualMenu, {
72
- role: "menu"
77
+ var _useState3 = (0, _react.useState)(true),
78
+ _useState4 = (0, _slicedToArray2["default"])(_useState3, 2),
79
+ firstUpdate = _useState4[0],
80
+ setFirstUpdate = _useState4[1];
81
+ (0, _react.useLayoutEffect)(function () {
82
+ if (selectedItemId !== -1 && firstUpdate) {
83
+ var _contextualMenuEl$scr;
84
+ var contextualMenuEl = contextualMenuRef === null || contextualMenuRef === void 0 ? void 0 : contextualMenuRef.current;
85
+ var selectedItemEl = contextualMenuEl === null || contextualMenuEl === void 0 ? void 0 : contextualMenuEl.querySelector("[aria-selected='true']");
86
+ contextualMenuEl === null || contextualMenuEl === void 0 ? void 0 : (_contextualMenuEl$scr = contextualMenuEl.scrollTo) === null || _contextualMenuEl$scr === void 0 ? void 0 : _contextualMenuEl$scr.call(contextualMenuEl, {
87
+ top: (selectedItemEl === null || selectedItemEl === void 0 ? void 0 : selectedItemEl.offsetTop) - (contextualMenuEl === null || contextualMenuEl === void 0 ? void 0 : contextualMenuEl.clientHeight) / 2
88
+ });
89
+ setFirstUpdate(false);
90
+ }
91
+ }, [firstUpdate, selectedItemId]);
92
+ return /*#__PURE__*/_react["default"].createElement(_styledComponents.ThemeProvider, {
93
+ theme: colorsTheme.contextualMenu
94
+ }, /*#__PURE__*/_react["default"].createElement(ContextualMenu, {
95
+ role: "menu",
96
+ ref: contextualMenuRef
73
97
  }, /*#__PURE__*/_react["default"].createElement(ContextualMenuContext.Provider, {
74
98
  value: {
75
99
  selectedItemId: selectedItemId,
@@ -80,9 +104,33 @@ var DxcContextualMenu = function DxcContextualMenu(_ref) {
80
104
  item: item,
81
105
  key: "".concat(item.label, "-").concat(index)
82
106
  });
83
- })));
107
+ }))));
84
108
  };
85
- var ContextualMenu = _styledComponents["default"].ul(_templateObject || (_templateObject = (0, _taggedTemplateLiteral2["default"])(["\n box-sizing: border-box;\n margin: 0;\n border: 1px solid ", ";\n border-radius: 0.25rem;\n padding: ", " ", ";\n display: grid;\n gap: ", ";\n min-width: 248px;\n max-height: 100%;\n background-color: ", ";\n overflow-y: auto;\n &::-webkit-scrollbar {\n width: 8px;\n height: 8px;\n }\n &::-webkit-scrollbar-thumb {\n background-color: ", ";\n border-radius: 0.25rem;\n }\n &::-webkit-scrollbar-track {\n background-color: ", ";\n border-radius: 0.25rem;\n }\n"])), _coreTokens["default"].color_grey_200, _coreTokens["default"].spacing_16, _coreTokens["default"].spacing_8, _coreTokens["default"].spacing_4, _coreTokens["default"].color_white, _coreTokens["default"].color_grey_700, _coreTokens["default"].color_grey_300);
109
+ var ContextualMenu = _styledComponents["default"].ul(_templateObject || (_templateObject = (0, _taggedTemplateLiteral2["default"])(["\n box-sizing: border-box;\n margin: 0;\n border: 1px solid ", ";\n border-radius: 0.25rem;\n padding: ", " ", ";\n display: grid;\n gap: ", ";\n min-width: 248px;\n max-height: 100%;\n background-color: ", ";\n overflow-y: auto;\n &::-webkit-scrollbar {\n width: 8px;\n height: 8px;\n }\n &::-webkit-scrollbar-thumb {\n background-color: ", ";\n border-radius: 0.25rem;\n }\n &::-webkit-scrollbar-track {\n background-color: ", ";\n border-radius: 0.25rem;\n }\n"])), function (_ref2) {
110
+ var theme = _ref2.theme;
111
+ return theme.borderColor;
112
+ }, _coreTokens["default"].spacing_16, _coreTokens["default"].spacing_8, _coreTokens["default"].spacing_4, function (_ref3) {
113
+ var theme = _ref3.theme;
114
+ return theme.backgroundColor;
115
+ }, _coreTokens["default"].color_grey_700, _coreTokens["default"].color_grey_300);
86
116
  var SectionList = _styledComponents["default"].ul(_templateObject2 || (_templateObject2 = (0, _taggedTemplateLiteral2["default"])(["\n list-style: none;\n margin: 0;\n padding: 0;\n display: grid;\n gap: ", ";\n"])), _coreTokens["default"].spacing_4);
87
- var Title = _styledComponents["default"].span(_templateObject3 || (_templateObject3 = (0, _taggedTemplateLiteral2["default"])(["\n margin: 0 0 ", " 0;\n padding: ", ";\n color: ", ";\n font-family: ", ";\n font-size: ", ";\n font-weight: ", ";\n line-height: 24px;\n"])), _coreTokens["default"].spacing_4, _coreTokens["default"].spacing_4, _coreTokens["default"].color_grey_900, _coreTokens["default"].type_sans, _coreTokens["default"].type_scale_03, _coreTokens["default"].type_semibold);
117
+ var Title = _styledComponents["default"].h2(_templateObject3 || (_templateObject3 = (0, _taggedTemplateLiteral2["default"])(["\n margin: 0 0 ", " 0;\n padding: ", ";\n color: ", ";\n font-family: ", ";\n font-size: ", ";\n font-style: ", ";\n font-weight: ", ";\n line-height: ", ";\n"])), _coreTokens["default"].spacing_4, _coreTokens["default"].spacing_4, function (_ref4) {
118
+ var theme = _ref4.theme;
119
+ return theme.sectionTitleFontColor;
120
+ }, function (_ref5) {
121
+ var theme = _ref5.theme;
122
+ return theme.fontFamily;
123
+ }, function (_ref6) {
124
+ var theme = _ref6.theme;
125
+ return theme.sectionTitleFontSize;
126
+ }, function (_ref7) {
127
+ var theme = _ref7.theme;
128
+ return theme.sectionTitleFontStyle;
129
+ }, function (_ref8) {
130
+ var theme = _ref8.theme;
131
+ return theme.sectionTitleFontWeight;
132
+ }, function (_ref9) {
133
+ var theme = _ref9.theme;
134
+ return theme.sectionTitleLineHeight;
135
+ });
88
136
  var _default = exports["default"] = DxcContextualMenu;
@@ -4,10 +4,11 @@ import DxcContextualMenu, { ContextualMenuContext } from "./ContextualMenu";
4
4
  import DxcContainer from "../container/Container";
5
5
  import SingleItem from "./SingleItem";
6
6
  import ExampleContainer from "../../.storybook/components/ExampleContainer";
7
- import { userEvent, within } from "@storybook/testing-library";
8
7
  import DxcBadge from "../badge/Badge";
9
8
  import { disabledRules } from "../../test/accessibility/rules/specific/contextual-menu/disabledRules";
10
9
  import preview from "../../.storybook/preview";
10
+ import { ThemeProvider } from "styled-components";
11
+ import useTheme from "../useTheme";
11
12
 
12
13
  export default {
13
14
  title: "Contextual Menu",
@@ -29,7 +30,7 @@ const items = [{ label: "Item 1" }, { label: "Item 2" }, { label: "Item 3" }, {
29
30
 
30
31
  const sections = [
31
32
  {
32
- title: "Team repositories",
33
+ title: "Section title",
33
34
  items: [{ label: "Approved locations" }, { label: "Approved locations" }, { label: "Approved locations" }],
34
35
  },
35
36
  {
@@ -54,7 +55,7 @@ const groupItems = [
54
55
  icon: "bookmark",
55
56
  badge: <DxcBadge color="purple" label="Experimental" />,
56
57
  },
57
- { label: "Selected Item 3" },
58
+ { label: "Selected Item 3", selectedByDefault: true },
58
59
  ],
59
60
  },
60
61
  ],
@@ -77,7 +78,7 @@ const itemsWithIcon = [
77
78
  {
78
79
  label: "Item 1",
79
80
  icon: (
80
- <svg xmlns="http://www.w3.org/2000/svg" height="24" viewBox="0 -960 960 960" width="24">
81
+ <svg xmlns="http://www.w3.org/2000/svg" height="24" viewBox="0 -960 960 960" width="24" fill="currentColor">
81
82
  <path d="M200-120v-640q0-33 23.5-56.5T280-840h400q33 0 56.5 23.5T760-760v640L480-240 200-120Zm80-122 200-86 200 86v-518H280v518Zm0-518h400-400Z" />
82
83
  </svg>
83
84
  ),
@@ -114,7 +115,7 @@ const sectionsWithScroll = [
114
115
  { label: "Approved locations" },
115
116
  { label: "Approved locations" },
116
117
  { label: "Approved locations" },
117
- { label: "Approved locations" },
118
+ { label: "Approved locations", selectedByDefault: true },
118
119
  ],
119
120
  },
120
121
  ];
@@ -124,7 +125,7 @@ const itemsWithTruncatedText = [
124
125
  label: "Item with a very long label that should be truncated",
125
126
  badge: <DxcBadge color="green" label="New" />,
126
127
  icon: (
127
- <svg xmlns="http://www.w3.org/2000/svg" height="24" viewBox="0 -960 960 960" width="24">
128
+ <svg xmlns="http://www.w3.org/2000/svg" height="24" viewBox="0 -960 960 960" width="24" fill="currentColor">
128
129
  <path d="M200-120v-640q0-33 23.5-56.5T280-840h400q33 0 56.5 23.5T760-760v640L480-240 200-120Zm80-122 200-86 200 86v-518H280v518Zm0-518h400-400Z" />
129
130
  </svg>
130
131
  ),
@@ -135,7 +136,7 @@ const itemsWithTruncatedText = [
135
136
  },
136
137
  ];
137
138
 
138
- const ContextualMenu = () => (
139
+ export const Chromatic = () => (
139
140
  <>
140
141
  <Title title="Default" theme="light" level={3} />
141
142
  <ExampleContainer>
@@ -171,7 +172,7 @@ const ContextualMenu = () => (
171
172
  <DxcContextualMenu items={itemsWithTruncatedText} />
172
173
  </DxcContainer>
173
174
  </ExampleContainer>
174
- <Title title="With scroll" theme="light" level={3} />
175
+ <Title title="With auto-scroll" theme="light" level={3} />
175
176
  <ExampleContainer>
176
177
  <DxcContainer height="300px" width="300px">
177
178
  <DxcContextualMenu items={sectionsWithScroll} />
@@ -186,47 +187,45 @@ const ContextualMenu = () => (
186
187
  </>
187
188
  );
188
189
 
189
- export const Chromatic = ContextualMenu.bind({});
190
- Chromatic.play = async ({ canvasElement }) => {
191
- const canvas = within(canvasElement);
192
- await userEvent.click(canvas.getByText("Grouped Item 1"));
193
- await userEvent.click(canvas.getByText("Grouped Item 2"));
194
- await userEvent.click(canvas.getByText("Selected Item 3"));
195
- };
190
+ export const SingleItemStates = () => {
191
+ const colorsTheme = useTheme();
196
192
 
197
- export const SingleItemStates = () => (
198
- <DxcContainer width="300px">
199
- <ContextualMenuContext.Provider value={{ selectedItemId: -1, setSelectedItemId: () => {} }}>
200
- <Title title="Default" theme="light" level={3} />
201
- <ExampleContainer>
202
- <SingleItem {...items[0]} id={0} depthLevel={0} />
203
- </ExampleContainer>
204
- <Title title="Focus" theme="light" level={3} />
205
- <ExampleContainer pseudoState="pseudo-focus">
206
- <SingleItem {...items[0]} id={0} depthLevel={0} />
207
- </ExampleContainer>
208
- <Title title="Hover" theme="light" level={3} />
209
- <ExampleContainer pseudoState="pseudo-hover">
210
- <SingleItem {...items[0]} id={0} depthLevel={0} />
211
- </ExampleContainer>
212
- <Title title="Active" theme="light" level={3} />
213
- <ExampleContainer pseudoState="pseudo-active">
214
- <SingleItem {...items[0]} id={0} depthLevel={0} />
215
- </ExampleContainer>
216
- </ContextualMenuContext.Provider>
217
- <ContextualMenuContext.Provider value={{ selectedItemId: 0, setSelectedItemId: () => {} }}>
218
- <Title title="Selected" theme="light" level={3} />
219
- <ExampleContainer>
220
- <SingleItem {...items[0]} id={0} depthLevel={0} />
221
- </ExampleContainer>
222
- <Title title="Selected hover" theme="light" level={3} />
223
- <ExampleContainer pseudoState="pseudo-hover">
224
- <SingleItem {...items[0]} id={0} depthLevel={0} />
225
- </ExampleContainer>
226
- <Title title="Selected active" theme="light" level={3} />
227
- <ExampleContainer pseudoState="pseudo-active">
228
- <SingleItem {...items[0]} id={0} depthLevel={0} />
229
- </ExampleContainer>
230
- </ContextualMenuContext.Provider>
231
- </DxcContainer>
232
- );
193
+ return (
194
+ <ThemeProvider theme={colorsTheme.contextualMenu}>
195
+ <DxcContainer width="300px">
196
+ <ContextualMenuContext.Provider value={{ selectedItemId: -1, setSelectedItemId: () => {} }}>
197
+ <Title title="Default" theme="light" level={3} />
198
+ <ExampleContainer>
199
+ <SingleItem {...items[0]} id={0} depthLevel={0} />
200
+ </ExampleContainer>
201
+ <Title title="Focus" theme="light" level={3} />
202
+ <ExampleContainer pseudoState="pseudo-focus">
203
+ <SingleItem {...items[0]} id={0} depthLevel={0} />
204
+ </ExampleContainer>
205
+ <Title title="Hover" theme="light" level={3} />
206
+ <ExampleContainer pseudoState="pseudo-hover">
207
+ <SingleItem {...items[0]} id={0} depthLevel={0} />
208
+ </ExampleContainer>
209
+ <Title title="Active" theme="light" level={3} />
210
+ <ExampleContainer pseudoState="pseudo-active">
211
+ <SingleItem {...items[0]} id={0} depthLevel={0} />
212
+ </ExampleContainer>
213
+ </ContextualMenuContext.Provider>
214
+ <ContextualMenuContext.Provider value={{ selectedItemId: 0, setSelectedItemId: () => {} }}>
215
+ <Title title="Selected" theme="light" level={3} />
216
+ <ExampleContainer>
217
+ <SingleItem {...items[0]} id={0} depthLevel={0} />
218
+ </ExampleContainer>
219
+ <Title title="Selected hover" theme="light" level={3} />
220
+ <ExampleContainer pseudoState="pseudo-hover">
221
+ <SingleItem {...items[0]} id={0} depthLevel={0} />
222
+ </ExampleContainer>
223
+ <Title title="Selected active" theme="light" level={3} />
224
+ <ExampleContainer pseudoState="pseudo-active">
225
+ <SingleItem {...items[0]} id={0} depthLevel={0} />
226
+ </ExampleContainer>
227
+ </ContextualMenuContext.Provider>
228
+ </DxcContainer>
229
+ </ThemeProvider>
230
+ );
231
+ };
@@ -60,144 +60,186 @@ var groups = [{
60
60
  label: "Item 8"
61
61
  }];
62
62
  describe("Contextual menu component tests", function () {
63
- test("Default - Renders with correct aria attributes", function () {
64
- var _render = (0, _react2.render)( /*#__PURE__*/_react["default"].createElement(_ContextualMenu["default"], {
65
- items: items
66
- })),
67
- getAllByRole = _render.getAllByRole,
68
- getByRole = _render.getByRole;
69
- expect(getAllByRole("menuitem").length).toBe(4);
70
- var actions = getAllByRole("button");
71
- expect(actions[0].getAttribute("aria-selected")).toBeTruthy();
72
- expect(getByRole("menu")).toBeTruthy();
73
- });
74
- test("Group - Group items collapse when clicked", /*#__PURE__*/(0, _asyncToGenerator2["default"])( /*#__PURE__*/_regenerator["default"].mark(function _callee() {
75
- var _render2, queryByText, getByText, getAllByRole, group1;
63
+ test("Single - Renders with correct aria attributes", /*#__PURE__*/(0, _asyncToGenerator2["default"])( /*#__PURE__*/_regenerator["default"].mark(function _callee() {
64
+ var _render, getAllByRole, getByRole, actions;
76
65
  return _regenerator["default"].wrap(function _callee$(_context) {
77
66
  while (1) switch (_context.prev = _context.next) {
78
67
  case 0:
79
- _render2 = (0, _react2.render)( /*#__PURE__*/_react["default"].createElement(_ContextualMenu["default"], {
68
+ _render = (0, _react2.render)( /*#__PURE__*/_react["default"].createElement(_ContextualMenu["default"], {
69
+ items: items
70
+ })), getAllByRole = _render.getAllByRole, getByRole = _render.getByRole;
71
+ expect(getAllByRole("menuitem").length).toBe(4);
72
+ actions = getAllByRole("button");
73
+ _context.next = 5;
74
+ return _userEvent["default"].click(actions[0]);
75
+ case 5:
76
+ expect(actions[0].getAttribute("aria-selected")).toBeTruthy();
77
+ expect(getByRole("menu")).toBeTruthy();
78
+ case 7:
79
+ case "end":
80
+ return _context.stop();
81
+ }
82
+ }, _callee);
83
+ })));
84
+ test("Single - An item can appear as selected by default by using the attribute selectedByDefault", function () {
85
+ var test = [{
86
+ label: "Tested item",
87
+ selectedByDefault: true
88
+ }];
89
+ var _render2 = (0, _react2.render)( /*#__PURE__*/_react["default"].createElement(_ContextualMenu["default"], {
90
+ items: test
91
+ })),
92
+ getByRole = _render2.getByRole;
93
+ var item = getByRole("button");
94
+ expect(item.getAttribute("aria-selected")).toBeTruthy();
95
+ });
96
+ test("Group - Group items collapse when clicked", /*#__PURE__*/(0, _asyncToGenerator2["default"])( /*#__PURE__*/_regenerator["default"].mark(function _callee2() {
97
+ var _render3, queryByText, getByText;
98
+ return _regenerator["default"].wrap(function _callee2$(_context2) {
99
+ while (1) switch (_context2.prev = _context2.next) {
100
+ case 0:
101
+ _render3 = (0, _react2.render)( /*#__PURE__*/_react["default"].createElement(_ContextualMenu["default"], {
80
102
  items: groups
81
- })), queryByText = _render2.queryByText, getByText = _render2.getByText, getAllByRole = _render2.getAllByRole;
82
- group1 = getAllByRole("button")[0];
83
- _context.next = 4;
103
+ })), queryByText = _render3.queryByText, getByText = _render3.getByText;
104
+ _context2.next = 3;
84
105
  return _userEvent["default"].click(getByText("Grouped Item 1"));
85
- case 4:
106
+ case 3:
86
107
  expect(getByText("Item 1")).toBeTruthy();
87
108
  expect(getByText("Grouped Item 2")).toBeTruthy();
88
- _context.next = 8;
109
+ _context2.next = 7;
89
110
  return _userEvent["default"].click(getByText("Grouped Item 2"));
90
- case 8:
111
+ case 7:
91
112
  expect(getByText("Item 2")).toBeTruthy();
92
113
  expect(getByText("Item 3")).toBeTruthy();
93
- _context.next = 12;
114
+ _context2.next = 11;
94
115
  return _userEvent["default"].click(getByText("Grouped Item 1"));
95
- case 12:
116
+ case 11:
96
117
  expect(queryByText("Item 1")).toBeFalsy();
97
118
  expect(queryByText("Item 2")).toBeFalsy();
98
119
  expect(queryByText("Item 3")).toBeFalsy();
99
- case 15:
120
+ case 14:
100
121
  case "end":
101
- return _context.stop();
122
+ return _context2.stop();
102
123
  }
103
- }, _callee);
124
+ }, _callee2);
104
125
  })));
105
- test("Group - Renders with correct aria attributes", /*#__PURE__*/(0, _asyncToGenerator2["default"])( /*#__PURE__*/_regenerator["default"].mark(function _callee2() {
106
- var _render3, getByText, getAllByRole, group1, actions;
107
- return _regenerator["default"].wrap(function _callee2$(_context2) {
108
- while (1) switch (_context2.prev = _context2.next) {
126
+ test("Group - Renders with correct aria attributes", /*#__PURE__*/(0, _asyncToGenerator2["default"])( /*#__PURE__*/_regenerator["default"].mark(function _callee3() {
127
+ var _render4, getAllByRole, group1, optionToBeClicked;
128
+ return _regenerator["default"].wrap(function _callee3$(_context3) {
129
+ while (1) switch (_context3.prev = _context3.next) {
109
130
  case 0:
110
- _render3 = (0, _react2.render)( /*#__PURE__*/_react["default"].createElement(_ContextualMenu["default"], {
131
+ _render4 = (0, _react2.render)( /*#__PURE__*/_react["default"].createElement(_ContextualMenu["default"], {
111
132
  items: groups
112
- })), getByText = _render3.getByText, getAllByRole = _render3.getAllByRole;
133
+ })), getAllByRole = _render4.getAllByRole;
113
134
  group1 = getAllByRole("button")[0];
114
- _context2.next = 4;
135
+ _context3.next = 4;
115
136
  return _userEvent["default"].click(group1);
116
137
  case 4:
117
138
  expect(group1.getAttribute("aria-expanded")).toBeTruthy();
118
139
  expect(group1.getAttribute("aria-controls")).toBe(getAllByRole("list")[0].id);
119
- _context2.next = 8;
120
- return _userEvent["default"].click(getByText("Grouped Item 2"));
140
+ _context3.next = 8;
141
+ return _userEvent["default"].click(getAllByRole("button")[2]);
121
142
  case 8:
122
- _context2.next = 10;
123
- return _userEvent["default"].click(getByText("Grouped Item 3"));
143
+ _context3.next = 10;
144
+ return _userEvent["default"].click(getAllByRole("button")[6]);
124
145
  case 10:
125
146
  expect(getAllByRole("menuitem").length).toBe(10);
126
- actions = getAllByRole("button");
127
- expect(actions[4].getAttribute("aria-selected")).toBeTruthy();
128
- case 13:
147
+ optionToBeClicked = getAllByRole("button")[4];
148
+ _context3.next = 14;
149
+ return _userEvent["default"].click(optionToBeClicked);
150
+ case 14:
151
+ expect(optionToBeClicked.getAttribute("aria-selected")).toBeTruthy();
152
+ case 15:
129
153
  case "end":
130
- return _context2.stop();
154
+ return _context3.stop();
131
155
  }
132
- }, _callee2);
156
+ }, _callee3);
133
157
  })));
134
- test("Group - Collapsed groups render as selected when containing a selected item", /*#__PURE__*/(0, _asyncToGenerator2["default"])( /*#__PURE__*/_regenerator["default"].mark(function _callee3() {
135
- var _render4, getAllByRole, group1, group2, item;
136
- return _regenerator["default"].wrap(function _callee3$(_context3) {
137
- while (1) switch (_context3.prev = _context3.next) {
158
+ test("Group - A grouped item, selected by default, must be visible (expanded group) in the first render of the component", function () {
159
+ var test = [{
160
+ label: "Grouped item",
161
+ items: [{
162
+ label: "Tested item",
163
+ selectedByDefault: true
164
+ }]
165
+ }];
166
+ var _render5 = (0, _react2.render)( /*#__PURE__*/_react["default"].createElement(_ContextualMenu["default"], {
167
+ items: test
168
+ })),
169
+ getByText = _render5.getByText,
170
+ getAllByRole = _render5.getAllByRole;
171
+ expect(getByText("Tested item")).toBeTruthy();
172
+ expect(getAllByRole("button")[1].getAttribute("aria-selected")).toBeTruthy();
173
+ });
174
+ test("Group - Collapsed groups render as selected when containing a selected item", /*#__PURE__*/(0, _asyncToGenerator2["default"])( /*#__PURE__*/_regenerator["default"].mark(function _callee4() {
175
+ var _render6, getAllByRole, group1, group2, item;
176
+ return _regenerator["default"].wrap(function _callee4$(_context4) {
177
+ while (1) switch (_context4.prev = _context4.next) {
138
178
  case 0:
139
- _render4 = (0, _react2.render)( /*#__PURE__*/_react["default"].createElement(_ContextualMenu["default"], {
179
+ _render6 = (0, _react2.render)( /*#__PURE__*/_react["default"].createElement(_ContextualMenu["default"], {
140
180
  items: groups
141
- })), getAllByRole = _render4.getAllByRole;
181
+ })), getAllByRole = _render6.getAllByRole;
142
182
  group1 = getAllByRole("button")[0];
143
- _context3.next = 4;
183
+ _context4.next = 4;
144
184
  return _userEvent["default"].click(group1);
145
185
  case 4:
146
186
  group2 = getAllByRole("button")[2];
147
- _context3.next = 7;
187
+ _context4.next = 7;
148
188
  return _userEvent["default"].click(group2);
149
189
  case 7:
150
190
  item = getAllByRole("button")[3];
151
- _context3.next = 10;
191
+ _context4.next = 10;
152
192
  return _userEvent["default"].click(item);
153
193
  case 10:
154
194
  expect(item.getAttribute("aria-selected")).toBeTruthy();
155
195
  expect(group1.getAttribute("aria-selected")).toBe("false");
156
196
  expect(group2.getAttribute("aria-selected")).toBe("false");
157
- _context3.next = 15;
197
+ _context4.next = 15;
158
198
  return _userEvent["default"].click(group2);
159
199
  case 15:
160
200
  expect(group2.getAttribute("aria-selected")).toBe("true");
161
- _context3.next = 18;
201
+ _context4.next = 18;
162
202
  return _userEvent["default"].click(group1);
163
203
  case 18:
164
204
  expect(group1.getAttribute("aria-selected")).toBe("true");
165
205
  case 19:
166
206
  case "end":
167
- return _context3.stop();
207
+ return _context4.stop();
168
208
  }
169
- }, _callee3);
209
+ }, _callee4);
170
210
  })));
171
- test("Sections - Renders with correct aria attributes", /*#__PURE__*/(0, _asyncToGenerator2["default"])( /*#__PURE__*/_regenerator["default"].mark(function _callee4() {
172
- var _render5, getAllByRole, actions;
173
- return _regenerator["default"].wrap(function _callee4$(_context4) {
174
- while (1) switch (_context4.prev = _context4.next) {
211
+ test("Sections - Renders with correct aria attributes", /*#__PURE__*/(0, _asyncToGenerator2["default"])( /*#__PURE__*/_regenerator["default"].mark(function _callee5() {
212
+ var _render7, getAllByRole, actions, section;
213
+ return _regenerator["default"].wrap(function _callee5$(_context5) {
214
+ while (1) switch (_context5.prev = _context5.next) {
175
215
  case 0:
176
- _render5 = (0, _react2.render)( /*#__PURE__*/_react["default"].createElement(_ContextualMenu["default"], {
216
+ _render7 = (0, _react2.render)( /*#__PURE__*/_react["default"].createElement(_ContextualMenu["default"], {
177
217
  items: sections
178
- })), getAllByRole = _render5.getAllByRole;
218
+ })), getAllByRole = _render7.getAllByRole;
179
219
  expect(getAllByRole("menuitem").length).toBe(6);
180
220
  actions = getAllByRole("button");
181
- _context4.next = 5;
221
+ _context5.next = 5;
182
222
  return _userEvent["default"].click(actions[0]);
183
223
  case 5:
184
224
  expect(actions[0].getAttribute("aria-selected")).toBeTruthy();
185
225
  expect(getAllByRole("group").length).toBe(2);
186
- case 7:
226
+ section = getAllByRole("group")[0];
227
+ expect(section.getAttribute("aria-labelledby")).toBe("Team repositories");
228
+ case 9:
187
229
  case "end":
188
- return _context4.stop();
230
+ return _context5.stop();
189
231
  }
190
- }, _callee4);
232
+ }, _callee5);
191
233
  })));
192
234
  test("The onSelect event from each item is called correctly", function () {
193
235
  var test = [{
194
236
  label: "Tested item",
195
237
  onSelect: jest.fn()
196
238
  }];
197
- var _render6 = (0, _react2.render)( /*#__PURE__*/_react["default"].createElement(_ContextualMenu["default"], {
239
+ var _render8 = (0, _react2.render)( /*#__PURE__*/_react["default"].createElement(_ContextualMenu["default"], {
198
240
  items: test
199
241
  })),
200
- getByRole = _render6.getByRole;
242
+ getByRole = _render8.getByRole;
201
243
  var item = getByRole("button");
202
244
  _react2.fireEvent.click(item);
203
245
  expect(test[0].onSelect).toHaveBeenCalled();
@@ -23,26 +23,26 @@ function _getRequireWildcardCache(e) { if ("function" != typeof WeakMap) return
23
23
  function _interopRequireWildcard(e, r) { if (!r && e && e.__esModule) return e; if (null === e || "object" != _typeof(e) && "function" != typeof e) return { "default": e }; var t = _getRequireWildcardCache(r); if (t && t.has(e)) return t.get(e); var n = { __proto__: null }, a = Object.defineProperty && Object.getOwnPropertyDescriptor; for (var u in e) if ("default" !== u && {}.hasOwnProperty.call(e, u)) { var i = a ? Object.getOwnPropertyDescriptor(e, u) : null; i && (i.get || i.set) ? Object.defineProperty(n, u, i) : n[u] = e[u]; } return n["default"] = e, t && t.set(e, n), n; }
24
24
  var isGroupSelected = function isGroupSelected(items, selectedItemId) {
25
25
  return items.some(function (item) {
26
- return "id" in item ? item.id === selectedItemId : isGroupSelected(item.items, selectedItemId);
26
+ if ("items" in item) return isGroupSelected(item.items, selectedItemId);else if (selectedItemId !== -1) return item.id === selectedItemId;else return item.selectedByDefault;
27
27
  });
28
28
  };
29
29
  var GroupItem = function GroupItem(_ref) {
30
30
  var items = _ref.items,
31
31
  props = (0, _objectWithoutProperties2["default"])(_ref, _excluded);
32
32
  var groupMenuId = "group-menu-".concat(props.label);
33
- var _useState = (0, _react.useState)(false),
33
+ var _useContext = (0, _react.useContext)(_ContextualMenu.ContextualMenuContext),
34
+ selectedItemId = _useContext.selectedItemId;
35
+ var groupSelected = (0, _react.useMemo)(function () {
36
+ return isGroupSelected(items, selectedItemId);
37
+ }, [items, selectedItemId]);
38
+ var _useState = (0, _react.useState)(groupSelected && selectedItemId === -1 ? true : false),
34
39
  _useState2 = (0, _slicedToArray2["default"])(_useState, 2),
35
40
  isOpen = _useState2[0],
36
41
  setIsOpen = _useState2[1];
37
- var _useContext = (0, _react.useContext)(_ContextualMenu.ContextualMenuContext),
38
- selectedItemId = _useContext.selectedItemId;
39
- var selected = (0, _react.useMemo)(function () {
40
- return !isOpen && isGroupSelected(items, selectedItemId);
41
- }, [isOpen, items, selectedItemId]);
42
42
  return /*#__PURE__*/_react["default"].createElement(_react["default"].Fragment, null, /*#__PURE__*/_react["default"].createElement(_ItemAction["default"], (0, _extends2["default"])({
43
43
  "aria-controls": groupMenuId,
44
44
  "aria-expanded": isOpen ? true : undefined,
45
- "aria-selected": selected,
45
+ "aria-selected": groupSelected && !isOpen,
46
46
  collapseIcon: isOpen ? /*#__PURE__*/_react["default"].createElement(_Icon["default"], {
47
47
  icon: "filled_expand_less"
48
48
  }) : /*#__PURE__*/_react["default"].createElement(_Icon["default"], {
@@ -53,7 +53,7 @@ var GroupItem = function GroupItem(_ref) {
53
53
  return !isOpen;
54
54
  });
55
55
  },
56
- selected: selected
56
+ selected: groupSelected && !isOpen
57
57
  }, props)), isOpen && /*#__PURE__*/_react["default"].createElement(ItemsList, {
58
58
  id: groupMenuId
59
59
  }, items.map(function (item) {
@@ -1,4 +1,4 @@
1
1
  import React from "react";
2
2
  import { ItemActionProps } from "./types";
3
- declare const ItemAction: ({ badge, collapseIcon, icon, label, depthLevel, selected, ...props }: ItemActionProps) => React.JSX.Element;
3
+ declare const ItemAction: ({ badge, collapseIcon, icon, label, depthLevel, ...props }: ItemActionProps) => React.JSX.Element;
4
4
  export default ItemAction;