@patternfly/chatbot 2.2.0-prerelease.11 → 2.2.0-prerelease.13

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 (93) hide show
  1. package/dist/cjs/ChatbotConversationHistoryNav/ChatbotConversationHistoryDropdown.js +3 -1
  2. package/dist/cjs/ChatbotHeader/ChatbotHeaderCloseButton.js +3 -1
  3. package/dist/cjs/ChatbotHeader/ChatbotHeaderMenu.js +3 -1
  4. package/dist/cjs/ChatbotHeader/ChatbotHeaderOptionsDropdown.js +3 -1
  5. package/dist/cjs/ChatbotHeader/ChatbotHeaderSelectorDropdown.js +3 -1
  6. package/dist/cjs/ChatbotToggle/ChatbotToggle.js +3 -1
  7. package/dist/cjs/Message/Message.d.ts +12 -1
  8. package/dist/cjs/Message/Message.js +11 -6
  9. package/dist/cjs/Message/QuickResponse/QuickResponse.d.ts +3 -1
  10. package/dist/cjs/Message/QuickResponse/QuickResponse.js +2 -1
  11. package/dist/cjs/Message/UserFeedback/CloseButton.d.ts +10 -0
  12. package/dist/cjs/Message/UserFeedback/CloseButton.js +14 -0
  13. package/dist/cjs/Message/UserFeedback/UserFeedback.d.ts +39 -0
  14. package/dist/cjs/Message/UserFeedback/UserFeedback.js +55 -0
  15. package/dist/cjs/Message/UserFeedback/UserFeedback.test.d.ts +1 -0
  16. package/dist/cjs/Message/UserFeedback/UserFeedback.test.js +146 -0
  17. package/dist/cjs/Message/UserFeedback/UserFeedbackComplete.d.ts +42 -0
  18. package/dist/cjs/Message/UserFeedback/UserFeedbackComplete.js +117 -0
  19. package/dist/cjs/Message/UserFeedback/UserFeedbackComplete.test.d.ts +1 -0
  20. package/dist/cjs/Message/UserFeedback/UserFeedbackComplete.test.js +249 -0
  21. package/dist/cjs/MessageBar/AttachButton.js +3 -1
  22. package/dist/cjs/MessageBar/SendButton.js +3 -1
  23. package/dist/cjs/MessageBar/StopButton.js +3 -1
  24. package/dist/cjs/ResponseActions/ResponseActionButton.d.ts +4 -1
  25. package/dist/cjs/ResponseActions/ResponseActionButton.js +21 -6
  26. package/dist/cjs/ResponseActions/ResponseActions.d.ts +8 -2
  27. package/dist/cjs/ResponseActions/ResponseActions.js +7 -7
  28. package/dist/css/main.css +69 -11
  29. package/dist/css/main.css.map +1 -1
  30. package/dist/esm/ChatbotConversationHistoryNav/ChatbotConversationHistoryDropdown.js +3 -1
  31. package/dist/esm/ChatbotHeader/ChatbotHeaderCloseButton.js +3 -1
  32. package/dist/esm/ChatbotHeader/ChatbotHeaderMenu.js +3 -1
  33. package/dist/esm/ChatbotHeader/ChatbotHeaderOptionsDropdown.js +3 -1
  34. package/dist/esm/ChatbotHeader/ChatbotHeaderSelectorDropdown.js +3 -1
  35. package/dist/esm/ChatbotToggle/ChatbotToggle.js +3 -1
  36. package/dist/esm/Message/Message.d.ts +12 -1
  37. package/dist/esm/Message/Message.js +8 -3
  38. package/dist/esm/Message/QuickResponse/QuickResponse.d.ts +3 -1
  39. package/dist/esm/Message/QuickResponse/QuickResponse.js +2 -1
  40. package/dist/esm/Message/UserFeedback/CloseButton.d.ts +10 -0
  41. package/dist/esm/Message/UserFeedback/CloseButton.js +9 -0
  42. package/dist/esm/Message/UserFeedback/UserFeedback.d.ts +39 -0
  43. package/dist/esm/Message/UserFeedback/UserFeedback.js +50 -0
  44. package/dist/esm/Message/UserFeedback/UserFeedback.test.d.ts +1 -0
  45. package/dist/esm/Message/UserFeedback/UserFeedback.test.js +141 -0
  46. package/dist/esm/Message/UserFeedback/UserFeedbackComplete.d.ts +42 -0
  47. package/dist/esm/Message/UserFeedback/UserFeedbackComplete.js +112 -0
  48. package/dist/esm/Message/UserFeedback/UserFeedbackComplete.test.d.ts +1 -0
  49. package/dist/esm/Message/UserFeedback/UserFeedbackComplete.test.js +244 -0
  50. package/dist/esm/MessageBar/AttachButton.js +3 -1
  51. package/dist/esm/MessageBar/SendButton.js +3 -1
  52. package/dist/esm/MessageBar/StopButton.js +3 -1
  53. package/dist/esm/ResponseActions/ResponseActionButton.d.ts +4 -1
  54. package/dist/esm/ResponseActions/ResponseActionButton.js +18 -3
  55. package/dist/esm/ResponseActions/ResponseActions.d.ts +8 -2
  56. package/dist/esm/ResponseActions/ResponseActions.js +7 -7
  57. package/dist/tsconfig.tsbuildinfo +1 -1
  58. package/package.json +1 -1
  59. package/patternfly-docs/content/extensions/chatbot/examples/Messages/MessageWithFeedback.tsx +71 -0
  60. package/patternfly-docs/content/extensions/chatbot/examples/Messages/MessageWithFeedbackTimeout.tsx +27 -0
  61. package/patternfly-docs/content/extensions/chatbot/examples/Messages/Messages.md +37 -7
  62. package/patternfly-docs/content/extensions/chatbot/examples/UI/UI.md +3 -6
  63. package/patternfly-docs/content/extensions/chatbot/examples/demos/AttachmentDemos.md +14 -0
  64. package/patternfly-docs/content/extensions/chatbot/examples/demos/Feedback.tsx +104 -0
  65. package/src/AttachMenu/AttachMenu.scss +1 -1
  66. package/src/ChatbotConversationHistoryNav/ChatbotConversationHistoryDropdown.tsx +7 -1
  67. package/src/ChatbotConversationHistoryNav/ChatbotConversationHistoryNav.scss +8 -1
  68. package/src/ChatbotHeader/ChatbotHeaderCloseButton.tsx +7 -1
  69. package/src/ChatbotHeader/ChatbotHeaderMenu.tsx +7 -1
  70. package/src/ChatbotHeader/ChatbotHeaderOptionsDropdown.tsx +8 -1
  71. package/src/ChatbotHeader/ChatbotHeaderSelectorDropdown.tsx +8 -1
  72. package/src/ChatbotModal/ChatbotModal.scss +1 -1
  73. package/src/ChatbotToggle/ChatbotToggle.tsx +6 -1
  74. package/src/CodeModal/CodeModal.scss +1 -1
  75. package/src/FileDetails/FileDetails.scss +1 -1
  76. package/src/Message/CodeBlockMessage/CodeBlockMessage.scss +1 -1
  77. package/src/Message/Message.scss +1 -1
  78. package/src/Message/Message.tsx +24 -2
  79. package/src/Message/QuickResponse/QuickResponse.tsx +6 -2
  80. package/src/Message/UserFeedback/CloseButton.tsx +21 -0
  81. package/src/Message/UserFeedback/UserFeedback.scss +53 -0
  82. package/src/Message/UserFeedback/UserFeedback.test.tsx +257 -0
  83. package/src/Message/UserFeedback/UserFeedback.tsx +132 -0
  84. package/src/Message/UserFeedback/UserFeedbackComplete.test.tsx +255 -0
  85. package/src/Message/UserFeedback/UserFeedbackComplete.tsx +211 -0
  86. package/src/MessageBar/AttachButton.tsx +2 -0
  87. package/src/MessageBar/SendButton.tsx +2 -0
  88. package/src/MessageBar/StopButton.tsx +2 -0
  89. package/src/ResponseActions/ResponseActionButton.tsx +14 -2
  90. package/src/ResponseActions/ResponseActions.tsx +26 -2
  91. package/src/Settings/Settings.scss +2 -2
  92. package/src/SourceDetailsMenuItem/SourceDetailsMenuItem.scss +1 -1
  93. package/src/main.scss +1 -0
@@ -0,0 +1,141 @@
1
+ var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) {
2
+ function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }
3
+ return new (P || (P = Promise))(function (resolve, reject) {
4
+ function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }
5
+ function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } }
6
+ function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }
7
+ step((generator = generator.apply(thisArg, _arguments || [])).next());
8
+ });
9
+ };
10
+ import React from 'react';
11
+ import { render, screen } from '@testing-library/react';
12
+ import '@testing-library/jest-dom';
13
+ import userEvent from '@testing-library/user-event';
14
+ import UserFeedback from './UserFeedback';
15
+ const MOCK_RESPONSES = [
16
+ { id: '1', content: 'Helpful information', onClick: () => alert('Clicked helpful information') },
17
+ { id: '2', content: 'Easy to understand', onClick: () => alert('Clicked easy to understand') },
18
+ { id: '3', content: 'Resolved my issue', onClick: () => alert('Clicked resolved my issue') }
19
+ ];
20
+ describe('UserFeedback', () => {
21
+ it('should render correctly', () => {
22
+ render(React.createElement(UserFeedback, { onClose: jest.fn, onSubmit: jest.fn, quickResponses: MOCK_RESPONSES, timestamp: "12/12/12" }));
23
+ expect(screen.getByRole('heading', { name: /Why did you choose this rating?/i })).toBeTruthy();
24
+ expect(screen.getByRole('list', { name: 'Quick feedback for message received at 12/12/12' })).toBeTruthy();
25
+ expect(screen.getByRole('button', { name: /Helpful information/i })).toBeTruthy();
26
+ expect(screen.getByRole('button', { name: /Easy to understand/i })).toBeTruthy();
27
+ expect(screen.getByRole('button', { name: /Resolved my issue/i })).toBeTruthy();
28
+ expect(screen.getByRole('button', { name: /Submit/i })).toBeTruthy();
29
+ expect(screen.getByRole('button', { name: 'Close feedback for message received at 12/12/12' })).toBeTruthy();
30
+ expect(screen.getByRole('button', { name: /Cancel/i })).toBeTruthy();
31
+ expect(screen.queryByRole('textbox', { name: /Provide optional additional feedback/i })).toBeFalsy();
32
+ });
33
+ it('should render different title correctly', () => {
34
+ render(React.createElement(UserFeedback, { timestamp: "12/12/12", onClose: jest.fn, onSubmit: jest.fn, quickResponses: MOCK_RESPONSES, title: "Thanks! Why?" }));
35
+ expect(screen.getByText('Thanks! Why?')).toBeTruthy();
36
+ });
37
+ it('should render different submit button text correctly', () => {
38
+ render(React.createElement(UserFeedback, { timestamp: "12/12/12", onClose: jest.fn, onSubmit: jest.fn, quickResponses: MOCK_RESPONSES, submitWord: "Give feedback" }));
39
+ expect(screen.getByRole('button', { name: /Give feedback/i })).toBeTruthy();
40
+ });
41
+ it('should render text area correctly', () => {
42
+ render(React.createElement(UserFeedback, { timestamp: "12/12/12", onClose: jest.fn, onSubmit: jest.fn, quickResponses: MOCK_RESPONSES, hasTextArea: true }));
43
+ expect(screen.getByRole('textbox', { name: /Provide optional additional feedback/i })).toBeTruthy();
44
+ });
45
+ it('should call onTextAreaChange correctly', () => __awaiter(void 0, void 0, void 0, function* () {
46
+ const spy = jest.fn();
47
+ render(React.createElement(UserFeedback, { timestamp: "12/12/12", onClose: jest.fn, onSubmit: jest.fn, quickResponses: MOCK_RESPONSES, hasTextArea: true, onTextAreaChange: spy }));
48
+ const textbox = screen.getByRole('textbox', { name: /Provide optional additional feedback/i });
49
+ yield userEvent.type(textbox, 'test');
50
+ expect(spy).toHaveBeenCalledTimes(4);
51
+ }));
52
+ it('should render different placeholder correctly', () => {
53
+ render(React.createElement(UserFeedback, { timestamp: "12/12/12", onClose: jest.fn, onSubmit: jest.fn, quickResponses: MOCK_RESPONSES, hasTextArea: true, textAreaPlaceholder: "Provide any other information" }));
54
+ expect(screen.getByRole('textbox', { name: /Provide optional additional feedback/i })).toHaveAttribute('placeholder', 'Provide any other information');
55
+ });
56
+ it('should render different text area label correctly', () => {
57
+ render(React.createElement(UserFeedback, { timestamp: "12/12/12", onClose: jest.fn, onSubmit: jest.fn, quickResponses: MOCK_RESPONSES, hasTextArea: true, textAreaAriaLabel: "Provide more details" }));
58
+ expect(screen.getByRole('textbox', { name: /Provide more details/i })).toBeTruthy();
59
+ });
60
+ it('should handle onClose correctly when close button is clicked', () => __awaiter(void 0, void 0, void 0, function* () {
61
+ const spy = jest.fn();
62
+ render(React.createElement(UserFeedback, { onSubmit: jest.fn, quickResponses: MOCK_RESPONSES, onClose: spy, timestamp: "12/12/12" }));
63
+ const closeButton = screen.getByRole('button', { name: 'Close feedback for message received at 12/12/12' });
64
+ expect(closeButton).toBeTruthy();
65
+ yield userEvent.click(closeButton);
66
+ expect(spy).toHaveBeenCalledTimes(1);
67
+ }));
68
+ it('should be able to change close button aria label', () => {
69
+ const spy = jest.fn();
70
+ render(React.createElement(UserFeedback, { timestamp: "12/12/12", onSubmit: jest.fn, quickResponses: MOCK_RESPONSES, onClose: spy, closeButtonAriaLabel: "Ima button" }));
71
+ expect(screen.getByRole('button', { name: /Ima button/i })).toBeTruthy();
72
+ });
73
+ it('should handle onClose correctly when cancel button is clicked', () => __awaiter(void 0, void 0, void 0, function* () {
74
+ const spy = jest.fn();
75
+ render(React.createElement(UserFeedback, { onSubmit: jest.fn, quickResponses: MOCK_RESPONSES, onClose: spy, timestamp: "12/12/12" }));
76
+ const cancelButton = screen.getByRole('button', { name: 'Cancel' });
77
+ expect(cancelButton).toBeTruthy();
78
+ yield userEvent.click(cancelButton);
79
+ expect(spy).toHaveBeenCalledTimes(1);
80
+ }));
81
+ it('should change cancel word correctly', () => {
82
+ render(React.createElement(UserFeedback, { onSubmit: jest.fn, quickResponses: MOCK_RESPONSES, onClose: jest.fn, cancelWord: "Exit", timestamp: "12/12/12" }));
83
+ expect(screen.getByRole('button', { name: 'Exit' })).toBeTruthy();
84
+ });
85
+ it('should handle className', () => __awaiter(void 0, void 0, void 0, function* () {
86
+ render(React.createElement(UserFeedback, { timestamp: "12/12/12", onClose: jest.fn, onSubmit: jest.fn, quickResponses: MOCK_RESPONSES, className: "test", "data-testid": "card" }));
87
+ expect(screen.getByTestId('card')).toHaveClass('test');
88
+ }));
89
+ it('should apply id', () => __awaiter(void 0, void 0, void 0, function* () {
90
+ render(React.createElement(UserFeedback, { timestamp: "12/12/12", onClose: jest.fn, onSubmit: jest.fn, quickResponses: MOCK_RESPONSES, id: "test", "data-testid": "card" }));
91
+ expect(screen.getByTestId('card').parentElement).toHaveAttribute('id', 'test');
92
+ }));
93
+ it('should handle submit correctly when nothing is selected', () => __awaiter(void 0, void 0, void 0, function* () {
94
+ const spy = jest.fn();
95
+ render(React.createElement(UserFeedback, { timestamp: "12/12/12", onClose: jest.fn, onSubmit: spy, quickResponses: MOCK_RESPONSES }));
96
+ yield userEvent.click(screen.getByRole('button', { name: /Submit/i }));
97
+ expect(spy).toHaveBeenCalledTimes(1);
98
+ expect(spy).toHaveBeenCalledWith(undefined, '');
99
+ }));
100
+ it('should handle submit correctly when item is selected', () => __awaiter(void 0, void 0, void 0, function* () {
101
+ const spy = jest.fn();
102
+ render(React.createElement(UserFeedback, { timestamp: "12/12/12", onClose: jest.fn, onSubmit: spy, quickResponses: MOCK_RESPONSES }));
103
+ yield userEvent.click(screen.getByRole('button', { name: /Easy to understand/i }));
104
+ yield userEvent.click(screen.getByRole('button', { name: /Submit/i }));
105
+ expect(spy).toHaveBeenCalledTimes(1);
106
+ expect(spy).toHaveBeenCalledWith('2', '');
107
+ }));
108
+ it('should handle submit correctly when there is just text input', () => __awaiter(void 0, void 0, void 0, function* () {
109
+ const spy = jest.fn();
110
+ render(React.createElement(UserFeedback, { timestamp: "12/12/12", onClose: jest.fn, onSubmit: spy, quickResponses: MOCK_RESPONSES, hasTextArea: true }));
111
+ yield userEvent.type(screen.getByRole('textbox', { name: /Provide optional additional feedback/i }), 'What a great experience!');
112
+ yield userEvent.click(screen.getByRole('button', { name: /Submit/i }));
113
+ expect(spy).toHaveBeenCalledTimes(1);
114
+ expect(spy).toHaveBeenCalledWith(undefined, 'What a great experience!');
115
+ }));
116
+ it('should handle submit correctly when item is selected and there is text input', () => __awaiter(void 0, void 0, void 0, function* () {
117
+ const spy = jest.fn();
118
+ render(React.createElement(UserFeedback, { timestamp: "12/12/12", onClose: jest.fn, onSubmit: spy, quickResponses: MOCK_RESPONSES, hasTextArea: true }));
119
+ yield userEvent.click(screen.getByRole('button', { name: /Easy to understand/i }));
120
+ yield userEvent.type(screen.getByRole('textbox', { name: /Provide optional additional feedback/i }), 'What a great experience!');
121
+ yield userEvent.click(screen.getByRole('button', { name: /Submit/i }));
122
+ expect(spy).toHaveBeenCalledTimes(1);
123
+ expect(spy).toHaveBeenCalledWith('2', 'What a great experience!');
124
+ }));
125
+ it('should default title heading level to h1', () => {
126
+ render(React.createElement(UserFeedback, { timestamp: "12/12/12", onClose: jest.fn, onSubmit: jest.fn, quickResponses: MOCK_RESPONSES }));
127
+ expect(screen.getByRole('heading', { level: 1, name: /Why did you choose this rating?/i })).toBeTruthy();
128
+ });
129
+ it('should be able to change title heading level', () => {
130
+ render(React.createElement(UserFeedback, { timestamp: "12/12/12", onClose: jest.fn, onSubmit: jest.fn, quickResponses: MOCK_RESPONSES, headingLevel: "h6" }));
131
+ expect(screen.getByRole('heading', { level: 6, name: /Why did you choose this rating?/i })).toBeTruthy();
132
+ });
133
+ it('should focus on load by default', () => {
134
+ render(React.createElement(UserFeedback, { timestamp: "12/12/12", onClose: jest.fn, onSubmit: jest.fn, quickResponses: MOCK_RESPONSES, "data-testid": "card" }));
135
+ expect(screen.getByTestId('card').parentElement).toHaveFocus();
136
+ });
137
+ it('should not focus on load if focusOnLoad = false', () => {
138
+ render(React.createElement(UserFeedback, { timestamp: "12/12/12", onClose: jest.fn, onSubmit: jest.fn, quickResponses: MOCK_RESPONSES, "data-testid": "card", focusOnLoad: false }));
139
+ expect(screen.getByTestId('card').parentElement).not.toHaveFocus();
140
+ });
141
+ });
@@ -0,0 +1,42 @@
1
+ import React from 'react';
2
+ import { CardProps, OUIAProps } from '@patternfly/react-core';
3
+ export interface UserFeedbackCompleteProps extends Omit<CardProps, 'ref'>, OUIAProps {
4
+ /** Additional classes for the pagination navigation container. */
5
+ className?: string;
6
+ /** Substitute for the English phrase "Thank you". */
7
+ title?: string;
8
+ /** Substitute for the English phrase "You have successfully sent your feedback! Thank you for responding." */
9
+ body?: string | React.ReactNode;
10
+ /** Callback function for when close button is clicked */
11
+ onClose?: () => void;
12
+ /** Aria label for close button */
13
+ closeButtonAriaLabel?: string;
14
+ /** Function to be executed on timeout. Relevant when the timeout prop is set. */
15
+ onTimeout?: () => void;
16
+ /** If set to true, the timeout is 8000 milliseconds. If a number is provided, card will
17
+ * be dismissed after that amount of time in milliseconds.
18
+ */
19
+ timeout?: number | boolean;
20
+ /** If the user hovers over the card and `timeout` expires, this is how long to wait
21
+ * before finally dismissing the alert.
22
+ */
23
+ timeoutAnimation?: number;
24
+ /** Callback for when mouse hovers over card */
25
+ onMouseEnter?: (e: React.MouseEvent<HTMLDivElement>) => void;
26
+ /** Callback for when mouse stops hovering over card */
27
+ onMouseLeave?: (e: React.MouseEvent<HTMLDivElement>) => void;
28
+ /** Value to overwrite the randomly generated data-ouia-component-id.*/
29
+ ouiaId?: number | string;
30
+ /** Set the value of data-ouia-safe. Only set to true when the component is in a static state, i.e. no animations are occurring. At all other times, this value must be false. */
31
+ ouiaSafe?: boolean;
32
+ /** Flag to indicate if the card is in a live region. */
33
+ isLiveRegion?: boolean;
34
+ /** Uniquely identifies the completion card. */
35
+ id?: string;
36
+ /** Whether to focus card on load */
37
+ focusOnLoad?: boolean;
38
+ /** Timestamp passed in by Message for more context in aria announcements */
39
+ timestamp?: string;
40
+ }
41
+ declare const UserFeedbackComplete: React.FunctionComponent<UserFeedbackCompleteProps>;
42
+ export default UserFeedbackComplete;
@@ -0,0 +1,112 @@
1
+ var __rest = (this && this.__rest) || function (s, e) {
2
+ var t = {};
3
+ for (var p in s) if (Object.prototype.hasOwnProperty.call(s, p) && e.indexOf(p) < 0)
4
+ t[p] = s[p];
5
+ if (s != null && typeof Object.getOwnPropertySymbols === "function")
6
+ for (var i = 0, p = Object.getOwnPropertySymbols(s); i < p.length; i++) {
7
+ if (e.indexOf(p[i]) < 0 && Object.prototype.propertyIsEnumerable.call(s, p[i]))
8
+ t[p[i]] = s[p[i]];
9
+ }
10
+ return t;
11
+ };
12
+ // ============================================================================
13
+ // Chatbot Main - Messages - Feedback Complete Card
14
+ // ============================================================================
15
+ import React from 'react';
16
+ // Import PatternFly components
17
+ import { Card, CardBody, CardHeader, CardTitle, useOUIAProps } from '@patternfly/react-core';
18
+ import CloseButton from './CloseButton';
19
+ const UserFeedbackComplete = (_a) => {
20
+ var { className, title = 'Feedback submitted', body = "We've received your response. Thank you for sharing your feedback!", timestamp, timeout = false, timeoutAnimation = 3000, onTimeout, onClose, closeButtonAriaLabel = `Close feedback for message received at ${timestamp}`, onMouseEnter, onMouseLeave, ouiaId, ouiaSafe, isLiveRegion, id, focusOnLoad = true } = _a, props = __rest(_a, ["className", "title", "body", "timestamp", "timeout", "timeoutAnimation", "onTimeout", "onClose", "closeButtonAriaLabel", "onMouseEnter", "onMouseLeave", "ouiaId", "ouiaSafe", "isLiveRegion", "id", "focusOnLoad"]);
21
+ const [timedOut, setTimedOut] = React.useState(false);
22
+ const [timedOutAnimation, setTimedOutAnimation] = React.useState(true);
23
+ const [isMouseOver, setIsMouseOver] = React.useState();
24
+ const [containsFocus, setContainsFocus] = React.useState();
25
+ const dismissed = timedOut && timedOutAnimation && !isMouseOver && !containsFocus;
26
+ const divRef = React.useRef(null);
27
+ const ouiaProps = useOUIAProps('User Feedback Complete', ouiaId, ouiaSafe);
28
+ React.useEffect(() => {
29
+ var _a;
30
+ if (focusOnLoad) {
31
+ (_a = divRef.current) === null || _a === void 0 ? void 0 : _a.focus();
32
+ }
33
+ }, []);
34
+ React.useEffect(() => {
35
+ const calculatedTimeout = timeout === true ? 8000 : Number(timeout);
36
+ if (calculatedTimeout > 0) {
37
+ const timer = setTimeout(() => setTimedOut(true), calculatedTimeout);
38
+ return () => clearTimeout(timer);
39
+ }
40
+ }, [timeout]);
41
+ React.useEffect(() => {
42
+ const onDocumentFocus = () => {
43
+ if (divRef.current) {
44
+ if (divRef.current.contains(document.activeElement)) {
45
+ setContainsFocus(true);
46
+ setTimedOutAnimation(false);
47
+ }
48
+ else if (containsFocus) {
49
+ setContainsFocus(false);
50
+ }
51
+ }
52
+ };
53
+ document.addEventListener('focus', onDocumentFocus, true);
54
+ return () => document.removeEventListener('focus', onDocumentFocus, true);
55
+ }, [containsFocus]);
56
+ React.useEffect(() => {
57
+ if (containsFocus === false || isMouseOver === false) {
58
+ const timer = setTimeout(() => setTimedOutAnimation(true), timeoutAnimation);
59
+ return () => clearTimeout(timer);
60
+ }
61
+ }, [containsFocus, isMouseOver, timeoutAnimation]);
62
+ React.useEffect(() => {
63
+ dismissed && onTimeout && onTimeout();
64
+ }, [dismissed, onTimeout]);
65
+ if (dismissed) {
66
+ return null;
67
+ }
68
+ const myOnMouseEnter = (ev) => {
69
+ setIsMouseOver(true);
70
+ setTimedOutAnimation(false);
71
+ onMouseEnter && onMouseEnter(ev);
72
+ };
73
+ const myOnMouseLeave = (ev) => {
74
+ setIsMouseOver(false);
75
+ onMouseLeave && onMouseLeave(ev);
76
+ };
77
+ return (
78
+ /* card does not have ref forwarding; hence wrapper div */
79
+ React.createElement("div", Object.assign({ ref: divRef, onMouseEnter: myOnMouseEnter, onMouseLeave: myOnMouseLeave }, (isLiveRegion && {
80
+ 'aria-live': 'polite',
81
+ 'aria-atomic': 'false'
82
+ }), { id: id, tabIndex: 0, "aria-label": title }, ouiaProps),
83
+ React.createElement(Card, Object.assign({ className: `pf-chatbot__feedback-card ${className ? className : ''}` }, props),
84
+ React.createElement(CardHeader, { actions:
85
+ /* eslint-disable indent */
86
+ onClose
87
+ ? {
88
+ actions: React.createElement(CloseButton, { onClose: onClose, ariaLabel: closeButtonAriaLabel })
89
+ }
90
+ : undefined }),
91
+ React.createElement("div", { className: "pf-chatbot__feedback-complete-body" },
92
+ React.createElement("div", { className: "pf-chatbot__feedback-complete-image" },
93
+ React.createElement("svg", { width: "60", height: "64", viewBox: "0 0 60 64", fill: "none", xmlns: "http://www.w3.org/2000/svg" },
94
+ React.createElement("path", { d: "M53.0555 21.5975H6.94323C3.57013 21.5975 0.835693 24.3319 0.835693 27.705V57.8925C0.835693 61.2656 3.57013 64 6.94323 64H53.0555C56.4286 64 59.1631 61.2656 59.1631 57.8925V27.705C59.1631 24.3319 56.4286 21.5975 53.0555 21.5975Z", fill: "#F8AE54" }),
95
+ React.createElement("path", { d: "M55.8973 19.8247C52.5894 15.7926 29.9992 0 29.9992 0C29.9992 0 7.40996 15.7926 4.10102 19.8247C0.79312 23.8568 0.835476 25.7184 0.835476 27.8899H59.1629C59.1629 25.7184 59.2052 23.8578 55.8973 19.8257V19.8247Z", fill: "#FFCC17" }),
96
+ React.createElement("g", null,
97
+ React.createElement("path", { d: "M53.0567 8.48981H6.94336V61.8388H53.0567V8.48981Z", fill: "#F2F2F2" })),
98
+ React.createElement("path", { d: "M51.6589 7.49908H8.34204V60.8481H51.6589V7.49908Z", fill: "white" }),
99
+ React.createElement("path", { d: "M0.835693 29.1296V57.8925C0.835693 59.2375 1.27165 60.4803 2.00823 61.4896L23.0303 43.5462L0.835693 29.1296Z", fill: "#FFCC17" }),
100
+ React.createElement("path", { d: "M36.9695 43.5472L57.9905 61.4907C58.7271 60.4813 59.1631 59.2386 59.1631 57.8935V29.1306L36.9685 43.5472H36.9695Z", fill: "#FFF4CC" }),
101
+ React.createElement("path", { d: "M0.835693 57.8925V57.8067L22.4146 42.7992L29.9994 37.5244L37.5842 42.7992L59.1641 57.8067V57.8925C59.1641 61.2665 56.4296 64 53.0566 64H6.94323C3.57024 64 0.835693 61.2665 0.835693 57.8925Z", fill: "#FFE072" }),
102
+ React.createElement("g", null,
103
+ React.createElement("path", { d: "M22.1563 42.978L0.835693 57.8067V56.6993L22.1563 42.978Z", fill: "#FEF07C" })),
104
+ React.createElement("g", null,
105
+ React.createElement("path", { d: "M37.8425 42.978L59.1631 57.8067V56.6993L37.8425 42.978Z", fill: "#FEF07C" })),
106
+ React.createElement("path", { d: "M37.8037 32.2373C42.1136 27.9273 42.1136 20.9395 37.8037 16.6295C33.4937 12.3196 26.5059 12.3196 22.196 16.6295C17.886 20.9395 17.886 27.9273 22.196 32.2373C26.5059 36.5472 33.4937 36.5472 37.8037 32.2373Z", fill: "#0066CC" }),
107
+ React.createElement("path", { d: "M27.7803 30.1276C27.6098 30.1276 27.4497 30.0614 27.3298 29.9406L22.9465 25.5562C22.8267 25.4364 22.7595 25.2762 22.7595 25.1068C22.7595 24.9374 22.8256 24.7762 22.9465 24.6554L24.2379 23.364C24.3577 23.2442 24.5178 23.177 24.6883 23.177C24.8587 23.177 25.0168 23.2431 25.1377 23.363L27.7803 26.0056L34.861 18.9259C34.9808 18.8061 35.1409 18.7389 35.3103 18.7389C35.4798 18.7389 35.6389 18.8051 35.7597 18.9239L37.0531 20.2173C37.173 20.3361 37.2401 20.4962 37.2401 20.6677C37.2401 20.8392 37.174 20.9983 37.0531 21.1181L28.2317 29.9406C28.1119 30.0604 27.9518 30.1265 27.7823 30.1265L27.7803 30.1276Z", fill: "white" }))),
108
+ React.createElement("div", { className: "pf-chatbot__feedback-complete-text" },
109
+ React.createElement(CardTitle, { className: "pf-chatbot__feedback-complete-title" }, title),
110
+ React.createElement(CardBody, { className: `pf-chatbot__feedback-complete-body` }, body))))));
111
+ };
112
+ export default UserFeedbackComplete;
@@ -0,0 +1 @@
1
+ import '@testing-library/jest-dom';
@@ -0,0 +1,244 @@
1
+ var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) {
2
+ function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }
3
+ return new (P || (P = Promise))(function (resolve, reject) {
4
+ function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }
5
+ function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } }
6
+ function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }
7
+ step((generator = generator.apply(thisArg, _arguments || [])).next());
8
+ });
9
+ };
10
+ import React from 'react';
11
+ import { act, render, screen } from '@testing-library/react';
12
+ import '@testing-library/jest-dom';
13
+ import userEvent from '@testing-library/user-event';
14
+ import UserFeedbackComplete from './UserFeedbackComplete';
15
+ describe('UserFeedbackComplete', () => {
16
+ it('should render correctly', () => {
17
+ render(React.createElement(UserFeedbackComplete, { timestamp: "12/12/12" }));
18
+ expect(screen.getByText('Feedback submitted')).toBeTruthy();
19
+ screen.getByText(/We've received your response. Thank you for sharing your feedback!/i);
20
+ expect(screen.queryByRole('button', { name: /Close/i })).toBeFalsy();
21
+ });
22
+ it('should render different title correctly', () => {
23
+ render(React.createElement(UserFeedbackComplete, { timestamp: "12/12/12", title: "Thanks!" }));
24
+ expect(screen.getByText('Thanks!')).toBeTruthy();
25
+ screen.getByText(/We've received your response. Thank you for sharing your feedback!/i);
26
+ });
27
+ it('should render different string body correctly', () => {
28
+ render(React.createElement(UserFeedbackComplete, { timestamp: "12/12/12", body: "Thanks!" }));
29
+ expect(screen.getByText('Feedback submitted')).toBeTruthy();
30
+ expect(screen.getByText('Thanks!')).toBeTruthy();
31
+ });
32
+ it('should render different node body correctly', () => {
33
+ render(React.createElement(UserFeedbackComplete, { timestamp: "12/12/12", body: React.createElement("div", null, "Thanks!") }));
34
+ expect(screen.getByText('Feedback submitted')).toBeTruthy();
35
+ expect(screen.getByText('Thanks!')).toBeTruthy();
36
+ });
37
+ it('should handle onClose correctly', () => __awaiter(void 0, void 0, void 0, function* () {
38
+ const spy = jest.fn();
39
+ render(React.createElement(UserFeedbackComplete, { timestamp: "12/12/12", onClose: spy }));
40
+ const closeButton = screen.getByRole('button', { name: 'Close feedback for message received at 12/12/12' });
41
+ expect(closeButton).toBeTruthy();
42
+ yield userEvent.click(closeButton);
43
+ expect(spy).toHaveBeenCalledTimes(1);
44
+ }));
45
+ it('should be able to change close button aria label', () => {
46
+ const spy = jest.fn();
47
+ render(React.createElement(UserFeedbackComplete, { timestamp: "12/12/12", onClose: spy, closeButtonAriaLabel: "Ima button" }));
48
+ expect(screen.getByRole('button', { name: /Ima button/i })).toBeTruthy();
49
+ });
50
+ it('should handle className', () => __awaiter(void 0, void 0, void 0, function* () {
51
+ render(React.createElement(UserFeedbackComplete, { timestamp: "12/12/12", className: "test", "data-testid": "card" }));
52
+ expect(screen.getByTestId('card')).toHaveClass('test');
53
+ }));
54
+ it('should apply id', () => __awaiter(void 0, void 0, void 0, function* () {
55
+ render(React.createElement(UserFeedbackComplete, { timestamp: "12/12/12", id: "test", "data-testid": "card" }));
56
+ expect(screen.getByTestId('card').parentElement).toHaveAttribute('id', 'test');
57
+ }));
58
+ it('renders with no timeout by default', () => {
59
+ jest.useFakeTimers();
60
+ render(React.createElement(UserFeedbackComplete, { timestamp: "12/12/12" }));
61
+ act(() => {
62
+ jest.advanceTimersByTime(8000);
63
+ });
64
+ expect(screen.getByText('Feedback submitted')).toBeVisible();
65
+ jest.useRealTimers();
66
+ });
67
+ it('should handle timeout correctly after 8000ms when timeout = true', () => __awaiter(void 0, void 0, void 0, function* () {
68
+ jest.useFakeTimers();
69
+ render(React.createElement(UserFeedbackComplete, { timestamp: "12/12/12", timeout: true }));
70
+ act(() => {
71
+ jest.advanceTimersByTime(7999);
72
+ });
73
+ expect(screen.getByText('Feedback submitted')).toBeVisible();
74
+ act(() => {
75
+ jest.advanceTimersByTime(1);
76
+ });
77
+ expect(screen.queryByText('Feedback submitted')).not.toBeInTheDocument();
78
+ jest.useRealTimers();
79
+ }));
80
+ it('should handle timeout correctly when timeout = numeric value', () => __awaiter(void 0, void 0, void 0, function* () {
81
+ jest.useFakeTimers();
82
+ render(React.createElement(UserFeedbackComplete, { timestamp: "12/12/12", timeout: 300 }));
83
+ act(() => {
84
+ jest.advanceTimersByTime(299);
85
+ });
86
+ expect(screen.getByText('Feedback submitted')).toBeVisible();
87
+ act(() => {
88
+ jest.advanceTimersByTime(1);
89
+ });
90
+ expect(screen.queryByText('Feedback submitted')).not.toBeInTheDocument();
91
+ jest.useRealTimers();
92
+ }));
93
+ it('does not get removed on timeout if the user is focused on the card', () => __awaiter(void 0, void 0, void 0, function* () {
94
+ const user = userEvent.setup({
95
+ advanceTimers: (delay) => jest.advanceTimersByTime(delay)
96
+ });
97
+ jest.useFakeTimers();
98
+ render(React.createElement(UserFeedbackComplete, { timestamp: "12/12/12", timeout: true, "data-testid": "card" }));
99
+ expect(screen.getByText('Feedback submitted')).toBeTruthy();
100
+ yield user.click(screen.getByTestId('card'));
101
+ act(() => {
102
+ jest.advanceTimersByTime(8000);
103
+ });
104
+ expect(screen.getByText('Feedback submitted')).toBeTruthy();
105
+ jest.useRealTimers();
106
+ }));
107
+ it('does not remove the card on timeout if the user is hovered over it', () => __awaiter(void 0, void 0, void 0, function* () {
108
+ const user = userEvent.setup({
109
+ advanceTimers: (delay) => jest.advanceTimersByTime(delay)
110
+ });
111
+ jest.useFakeTimers();
112
+ render(React.createElement(UserFeedbackComplete, { timestamp: "12/12/12", timeout: true, "data-testid": "card" }));
113
+ const card = screen.getByTestId('card');
114
+ yield user.hover(card);
115
+ act(() => {
116
+ jest.advanceTimersByTime(8000);
117
+ });
118
+ expect(card).toBeVisible();
119
+ jest.useRealTimers();
120
+ }));
121
+ it('removes the card after the user removes focus from the card and 3000ms have passed', () => __awaiter(void 0, void 0, void 0, function* () {
122
+ const user = userEvent.setup({
123
+ advanceTimers: (delay) => jest.advanceTimersByTime(delay)
124
+ });
125
+ jest.useFakeTimers();
126
+ render(React.createElement("div", null,
127
+ React.createElement("input", null),
128
+ React.createElement(UserFeedbackComplete, { timestamp: "12/12/12", timeout: true, "data-testid": "card" })));
129
+ const card = screen.getByTestId('card');
130
+ yield user.click(card);
131
+ act(() => {
132
+ jest.advanceTimersByTime(8000);
133
+ });
134
+ yield user.click(screen.getByRole('textbox'));
135
+ act(() => {
136
+ jest.advanceTimersByTime(3000);
137
+ });
138
+ expect(screen.queryByText('Feedback submitted')).not.toBeInTheDocument();
139
+ jest.useRealTimers();
140
+ }));
141
+ it('removes the card after the user removes hover from the card and 3000ms have passed', () => __awaiter(void 0, void 0, void 0, function* () {
142
+ const user = userEvent.setup({
143
+ advanceTimers: (delay) => jest.advanceTimersByTime(delay)
144
+ });
145
+ jest.useFakeTimers();
146
+ render(React.createElement("div", null,
147
+ React.createElement("input", null),
148
+ React.createElement(UserFeedbackComplete, { timestamp: "12/12/12", timeout: true, "data-testid": "card" })));
149
+ const card = screen.getByTestId('card');
150
+ yield user.hover(card);
151
+ act(() => {
152
+ jest.advanceTimersByTime(8000);
153
+ });
154
+ yield user.hover(screen.getByRole('textbox'));
155
+ act(() => {
156
+ jest.advanceTimersByTime(3000);
157
+ });
158
+ expect(screen.queryByText('Feedback submitted')).not.toBeInTheDocument();
159
+ jest.useRealTimers();
160
+ }));
161
+ it('removes the card after the user removes hover from the card and timeoutAnimation time has passed', () => __awaiter(void 0, void 0, void 0, function* () {
162
+ const user = userEvent.setup({
163
+ advanceTimers: (delay) => jest.advanceTimersByTime(delay)
164
+ });
165
+ jest.useFakeTimers();
166
+ render(React.createElement("div", null,
167
+ React.createElement("input", null),
168
+ React.createElement(UserFeedbackComplete, { timestamp: "12/12/12", timeout: true, "data-testid": "card", timeoutAnimation: 1000 })));
169
+ const card = screen.getByTestId('card');
170
+ yield user.hover(card);
171
+ act(() => {
172
+ jest.advanceTimersByTime(8000);
173
+ });
174
+ yield user.hover(screen.getByRole('textbox'));
175
+ act(() => {
176
+ jest.advanceTimersByTime(1000);
177
+ });
178
+ expect(screen.queryByText('Feedback submitted')).not.toBeInTheDocument();
179
+ jest.useRealTimers();
180
+ }));
181
+ it('does not call the onTimeout callback before the timeout period has expired', () => {
182
+ const spy = jest.fn();
183
+ jest.useFakeTimers();
184
+ render(React.createElement(UserFeedbackComplete, { timestamp: "12/12/12", timeout: true, "data-testid": "card", onTimeout: spy }));
185
+ act(() => {
186
+ jest.advanceTimersByTime(7999);
187
+ });
188
+ expect(spy).not.toHaveBeenCalled();
189
+ jest.useRealTimers();
190
+ });
191
+ it('calls the onTimeout callback after the timeout period has expired', () => {
192
+ jest.useFakeTimers();
193
+ const spy = jest.fn();
194
+ render(React.createElement(UserFeedbackComplete, { timestamp: "12/12/12", timeout: true, "data-testid": "card", onTimeout: spy }));
195
+ act(() => {
196
+ jest.advanceTimersByTime(8000);
197
+ });
198
+ expect(spy).toHaveBeenCalledTimes(1);
199
+ jest.useRealTimers();
200
+ });
201
+ it('renders without aria-live and aria-atomic attributes by default', () => {
202
+ render(React.createElement(UserFeedbackComplete, { timestamp: "12/12/12", timeout: true, "data-testid": "card" }));
203
+ const card = screen.getByTestId('card').parentElement;
204
+ expect(card).not.toHaveAttribute('aria-live');
205
+ expect(card).not.toHaveAttribute('aria-atomic');
206
+ });
207
+ it('has an aria-live value of polite and aria-atomic value of false when isLiveRegion = true', () => {
208
+ render(React.createElement(UserFeedbackComplete, { timestamp: "12/12/12", timeout: true, "data-testid": "card", isLiveRegion: true }));
209
+ const card = screen.getByTestId('card').parentElement;
210
+ expect(card).toHaveAttribute('aria-live', 'polite');
211
+ expect(card).toHaveAttribute('aria-atomic', 'false');
212
+ });
213
+ it('calls onMouseEnter appropriately', () => __awaiter(void 0, void 0, void 0, function* () {
214
+ const spy = jest.fn();
215
+ const user = userEvent.setup();
216
+ render(React.createElement("div", null,
217
+ React.createElement("input", null),
218
+ React.createElement(UserFeedbackComplete, { timestamp: "12/12/12", "data-testid": "card", onMouseEnter: spy })));
219
+ const card = screen.getByTestId('card');
220
+ expect(spy).toHaveBeenCalledTimes(0);
221
+ yield user.hover(card);
222
+ expect(spy).toHaveBeenCalledTimes(1);
223
+ }));
224
+ it('calls onMouseLeave appropriately', () => __awaiter(void 0, void 0, void 0, function* () {
225
+ const spy = jest.fn();
226
+ const user = userEvent.setup();
227
+ render(React.createElement("div", null,
228
+ React.createElement("input", null),
229
+ React.createElement(UserFeedbackComplete, { timestamp: "12/12/12", "data-testid": "card", onMouseLeave: spy })));
230
+ const card = screen.getByTestId('card');
231
+ expect(spy).toHaveBeenCalledTimes(0);
232
+ yield user.hover(card);
233
+ yield user.hover(screen.getByRole('textbox'));
234
+ expect(spy).toHaveBeenCalledTimes(1);
235
+ }));
236
+ it('should focus on load by default', () => {
237
+ render(React.createElement(UserFeedbackComplete, { timestamp: "12/12/12", "data-testid": "card" }));
238
+ expect(screen.getByTestId('card').parentElement).toHaveFocus();
239
+ });
240
+ it('should not focus on load if focusOnLoad = false', () => {
241
+ render(React.createElement(UserFeedbackComplete, { timestamp: "12/12/12", "data-testid": "card", focusOnLoad: false }));
242
+ expect(screen.getByTestId('card').parentElement).not.toHaveFocus();
243
+ });
244
+ });
@@ -25,7 +25,9 @@ const AttachButtonBase = (_a) => {
25
25
  });
26
26
  return (React.createElement(React.Fragment, null,
27
27
  React.createElement("input", Object.assign({ "data-testid": inputTestId }, getInputProps())),
28
- React.createElement(Tooltip, Object.assign({ id: "pf-chatbot__tooltip--attach", content: tooltipContent, position: "top", entryDelay: (tooltipProps === null || tooltipProps === void 0 ? void 0 : tooltipProps.entryDelay) || 0, exitDelay: (tooltipProps === null || tooltipProps === void 0 ? void 0 : tooltipProps.exitDelay) || 0, distance: (tooltipProps === null || tooltipProps === void 0 ? void 0 : tooltipProps.distance) || 8, animationDuration: (tooltipProps === null || tooltipProps === void 0 ? void 0 : tooltipProps.animationDuration) || 0 }, tooltipProps),
28
+ React.createElement(Tooltip, Object.assign({ id: "pf-chatbot__tooltip--attach", content: tooltipContent, position: "top", entryDelay: (tooltipProps === null || tooltipProps === void 0 ? void 0 : tooltipProps.entryDelay) || 0, exitDelay: (tooltipProps === null || tooltipProps === void 0 ? void 0 : tooltipProps.exitDelay) || 0, distance: (tooltipProps === null || tooltipProps === void 0 ? void 0 : tooltipProps.distance) || 8, animationDuration: (tooltipProps === null || tooltipProps === void 0 ? void 0 : tooltipProps.animationDuration) || 0,
29
+ // prevents VO announcements of both aria label and tooltip
30
+ aria: "none" }, tooltipProps),
29
31
  React.createElement(Button, Object.assign({ variant: "plain", ref: innerRef, className: `pf-chatbot__button--attach ${className !== null && className !== void 0 ? className : ''}`, "aria-label": props['aria-label'] || 'Attach button', isDisabled: isDisabled, onClick: onClick !== null && onClick !== void 0 ? onClick : open, icon: React.createElement(Icon, { iconSize: "xl", isInline: true },
30
32
  React.createElement(PaperclipIcon, null)) }, props)))));
31
33
  };
@@ -18,7 +18,9 @@ import { Button, Tooltip, Icon } from '@patternfly/react-core';
18
18
  import { PaperPlaneIcon } from '@patternfly/react-icons/dist/esm/icons/paper-plane-icon';
19
19
  export const SendButton = (_a) => {
20
20
  var { className, onClick, tooltipProps, tooltipContent = 'Send' } = _a, props = __rest(_a, ["className", "onClick", "tooltipProps", "tooltipContent"]);
21
- return (React.createElement(Tooltip, Object.assign({ id: "pf-chatbot__tooltip--send", content: tooltipContent, position: (tooltipProps === null || tooltipProps === void 0 ? void 0 : tooltipProps.position) || 'top', entryDelay: (tooltipProps === null || tooltipProps === void 0 ? void 0 : tooltipProps.entryDelay) || 0, exitDelay: (tooltipProps === null || tooltipProps === void 0 ? void 0 : tooltipProps.exitDelay) || 0, distance: (tooltipProps === null || tooltipProps === void 0 ? void 0 : tooltipProps.distance) || 8, animationDuration: (tooltipProps === null || tooltipProps === void 0 ? void 0 : tooltipProps.animationDuration) || 0 }, tooltipProps),
21
+ return (React.createElement(Tooltip, Object.assign({ id: "pf-chatbot__tooltip--send", content: tooltipContent, position: (tooltipProps === null || tooltipProps === void 0 ? void 0 : tooltipProps.position) || 'top', entryDelay: (tooltipProps === null || tooltipProps === void 0 ? void 0 : tooltipProps.entryDelay) || 0, exitDelay: (tooltipProps === null || tooltipProps === void 0 ? void 0 : tooltipProps.exitDelay) || 0, distance: (tooltipProps === null || tooltipProps === void 0 ? void 0 : tooltipProps.distance) || 8, animationDuration: (tooltipProps === null || tooltipProps === void 0 ? void 0 : tooltipProps.animationDuration) || 0,
22
+ // prevents VO announcements of both aria label and tooltip
23
+ aria: "none" }, tooltipProps),
22
24
  React.createElement(Button, Object.assign({ variant: "plain", className: `pf-chatbot__button--send ${className !== null && className !== void 0 ? className : ''}`, "aria-label": props['aria-label'] || 'Send button', onClick: onClick, icon: React.createElement(Icon, { iconSize: "xl", isInline: true },
23
25
  React.createElement(PaperPlaneIcon, null)) }, props))));
24
26
  };
@@ -17,7 +17,9 @@ import React from 'react';
17
17
  import { Button, Tooltip, Icon } from '@patternfly/react-core';
18
18
  export const StopButton = (_a) => {
19
19
  var { className, onClick, tooltipProps, tooltipContent = 'Stop' } = _a, props = __rest(_a, ["className", "onClick", "tooltipProps", "tooltipContent"]);
20
- return (React.createElement(Tooltip, Object.assign({ id: "pf-chatbot__tooltip--stop", content: tooltipContent, position: (tooltipProps === null || tooltipProps === void 0 ? void 0 : tooltipProps.position) || 'top', entryDelay: (tooltipProps === null || tooltipProps === void 0 ? void 0 : tooltipProps.entryDelay) || 0, exitDelay: (tooltipProps === null || tooltipProps === void 0 ? void 0 : tooltipProps.exitDelay) || 0, distance: (tooltipProps === null || tooltipProps === void 0 ? void 0 : tooltipProps.distance) || 8, animationDuration: (tooltipProps === null || tooltipProps === void 0 ? void 0 : tooltipProps.animationDuration) || 0 }, tooltipProps),
20
+ return (React.createElement(Tooltip, Object.assign({ id: "pf-chatbot__tooltip--stop", content: tooltipContent, position: (tooltipProps === null || tooltipProps === void 0 ? void 0 : tooltipProps.position) || 'top', entryDelay: (tooltipProps === null || tooltipProps === void 0 ? void 0 : tooltipProps.entryDelay) || 0, exitDelay: (tooltipProps === null || tooltipProps === void 0 ? void 0 : tooltipProps.exitDelay) || 0, distance: (tooltipProps === null || tooltipProps === void 0 ? void 0 : tooltipProps.distance) || 8, animationDuration: (tooltipProps === null || tooltipProps === void 0 ? void 0 : tooltipProps.animationDuration) || 0,
21
+ // prevents VO announcements of both aria label and tooltip
22
+ aria: "none" }, tooltipProps),
21
23
  React.createElement(Button, Object.assign({ className: `pf-chatbot__button--stop ${className !== null && className !== void 0 ? className : ''}`, variant: "link", "aria-label": props['aria-label'] || 'Stop button', onClick: onClick, icon: React.createElement(Icon, { iconSize: "xl", isInline: true },
22
24
  React.createElement("svg", { width: "16", height: "16", viewBox: "0 0 16 16", fill: "none", xmlns: "http://www.w3.org/2000/svg" },
23
25
  React.createElement("path", { d: "M0.5 3C0.5 1.62109 1.62109 0.5 3 0.5H13C14.3789 0.5 15.5 1.62109 15.5 3V13C15.5 14.3789 14.3789 15.5 13 15.5H3C1.62109 15.5 0.5 14.3789 0.5 13V3Z", fill: "currentColor" }))) }, props))));
@@ -21,6 +21,9 @@ export interface ResponseActionButtonProps {
21
21
  tooltipProps?: TooltipProps;
22
22
  /** Whether button is in clicked state */
23
23
  isClicked?: boolean;
24
+ /** Ref applied to button */
25
+ innerRef?: React.Ref<HTMLButtonElement>;
24
26
  }
25
- export declare const ResponseActionButton: React.FunctionComponent<ResponseActionButtonProps>;
27
+ export declare const ResponseActionButtonBase: React.FunctionComponent<ResponseActionButtonProps>;
28
+ declare const ResponseActionButton: React.ForwardRefExoticComponent<ResponseActionButtonProps & React.RefAttributes<HTMLButtonElement>>;
26
29
  export default ResponseActionButton;