From 55e4b807ae87b2d0e215f3d72fa3471475f3fdfc Mon Sep 17 00:00:00 2001 From: Alison Langston <46360176+alangsto@users.noreply.github.com> Date: Fri, 15 Nov 2024 09:18:52 -0500 Subject: [PATCH] feat: remove clear button from chat ui (#66) --- src/components/Sidebar/index.jsx | 35 +-------------- src/components/Sidebar/index.test.jsx | 41 ------------------ src/data/slice.js | 6 --- src/data/thunks.js | 8 ---- src/widgets/Xpert.test.jsx | 61 --------------------------- 5 files changed, 1 insertion(+), 150 deletions(-) diff --git a/src/components/Sidebar/index.jsx b/src/components/Sidebar/index.jsx index 70edc2b9..097397a3 100644 --- a/src/components/Sidebar/index.jsx +++ b/src/components/Sidebar/index.jsx @@ -1,16 +1,12 @@ import React, { useRef, useEffect } from 'react'; -import { useDispatch, useSelector } from 'react-redux'; +import { useSelector } from 'react-redux'; import PropTypes from 'prop-types'; -import { sendTrackEvent } from '@edx/frontend-platform/analytics'; import { - Button, Icon, IconButton, } from '@openedx/paragon'; import { Close } from '@openedx/paragon/icons'; -import { clearMessages } from '../../data/thunks'; -import { OPTIMIZELY_PROMPT_EXPERIMENT_KEY } from '../../data/optimizely'; import showSurvey from '../../utils/surveyMonkey'; import APIError from '../APIError'; @@ -18,7 +14,6 @@ import ChatBox from '../ChatBox'; import Disclosure from '../Disclosure'; import MessageForm from '../MessageForm'; import './Sidebar.scss'; -import { usePromptExperimentDecision } from '../../experiments'; const Sidebar = ({ courseId, @@ -32,14 +27,6 @@ const Sidebar = ({ messageList, } = useSelector(state => state.learningAssistant); const chatboxContainerRef = useRef(null); - const dispatch = useDispatch(); - - const [decision] = usePromptExperimentDecision(); - const { enabled: enabledExperiment, variationKey } = decision || {}; - const experimentPayload = enabledExperiment ? { - experiment_name: OPTIMIZELY_PROMPT_EXPERIMENT_KEY, - variation_key: variationKey, - } : {}; // this use effect is intended to scroll to the bottom of the chat window, in the case // that a message is larger than the chat window height. @@ -83,14 +70,6 @@ const Sidebar = ({ } }; - const handleClearMessages = () => { - dispatch(clearMessages()); - sendTrackEvent('edx.ui.lms.learning_assistant.clear', { - course_id: courseId, - ...experimentPayload, - }); - }; - const getMessageForm = () => ( ); @@ -120,18 +99,6 @@ const Sidebar = ({ ) } {getMessageForm()} -
- -
); diff --git a/src/components/Sidebar/index.test.jsx b/src/components/Sidebar/index.test.jsx index 9a57c70e..210daf40 100644 --- a/src/components/Sidebar/index.test.jsx +++ b/src/components/Sidebar/index.test.jsx @@ -1,11 +1,9 @@ import React from 'react'; import { screen, act } from '@testing-library/react'; -import { sendTrackEvent } from '@edx/frontend-platform/analytics'; import { usePromptExperimentDecision } from '../../experiments'; import { render as renderComponent } from '../../utils/utils.test'; import { initialState } from '../../data/slice'; -import { OPTIMIZELY_PROMPT_EXPERIMENT_KEY, OPTIMIZELY_PROMPT_EXPERIMENT_VARIATION_KEYS } from '../../data/optimizely'; import showSurvey from '../../utils/surveyMonkey'; import Sidebar from '.'; @@ -35,11 +33,6 @@ jest.mock('../../experiments', () => ({ usePromptExperimentDecision: jest.fn(), })); -const clearMessagesAction = 'clear-messages-action'; -jest.mock('../../data/thunks', () => ({ - clearMessages: () => 'clear-messages-action', -})); - const defaultProps = { courseId: 'some-course-id', isOpen: true, @@ -88,17 +81,6 @@ describe('', () => { render(undefined, { disclosureAcknowledged: true }); expect(screen.queryByTestId('sidebar-xpert')).toBeInTheDocument(); }); - - it('should dispatch clearMessages() and call sendTrackEvent() with the expected props on clear', () => { - render(undefined, { disclosureAcknowledged: true }); - - act(() => { - screen.queryByTestId('sidebar-clear-btn').click(); - }); - - expect(mockDispatch).toHaveBeenCalledWith(clearMessagesAction); - expect(sendTrackEvent).toHaveBeenCalledWith('edx.ui.lms.learning_assistant.clear', { course_id: defaultProps.courseId }); - }); }); describe('when it\'s not open', () => { @@ -133,28 +115,5 @@ describe('', () => { expect(showSurvey).toHaveBeenCalled(); }); - - it('should dispatch clearMessages() and call sendTrackEvent() with the expected props on clear', () => { - usePromptExperimentDecision.mockReturnValue([{ - enabled: true, - variationKey: OPTIMIZELY_PROMPT_EXPERIMENT_VARIATION_KEYS.UPDATED_PROMPT, - }]); - - render(undefined, { - ...defaultState, - disclosureAcknowledged: true, - }); - - act(() => { - screen.queryByTestId('sidebar-clear-btn').click(); - }); - - expect(mockDispatch).toHaveBeenCalledWith(clearMessagesAction); - expect(sendTrackEvent).toHaveBeenCalledWith('edx.ui.lms.learning_assistant.clear', { - course_id: defaultProps.courseId, - experiment_name: OPTIMIZELY_PROMPT_EXPERIMENT_KEY, - variation_key: OPTIMIZELY_PROMPT_EXPERIMENT_VARIATION_KEYS.UPDATED_PROMPT, - }); - }); }); }); diff --git a/src/data/slice.js b/src/data/slice.js index 29dbd402..ffc981af 100644 --- a/src/data/slice.js +++ b/src/data/slice.js @@ -26,11 +26,6 @@ export const learningAssistantSlice = createSlice({ setMessageList: (state, { payload }) => { state.messageList = payload.messageList; }, - resetMessages: (state) => { - state.currentMessage = ''; - state.messageList = []; - state.apiError = false; - }, setApiError: (state) => { state.apiError = true; }, @@ -56,7 +51,6 @@ export const { setCurrentMessage, clearCurrentMessage, setMessageList, - resetMessages, setApiError, setApiIsLoading, resetApiError, diff --git a/src/data/thunks.js b/src/data/thunks.js index 891c0f4d..13c64e51 100644 --- a/src/data/thunks.js +++ b/src/data/thunks.js @@ -6,7 +6,6 @@ import { fetchChatResponse, fetchLearningAssistantMessageHistory, fetchLearningA import { setCurrentMessage, clearCurrentMessage, - resetMessages, setMessageList, setApiError, setApiIsLoading, @@ -74,13 +73,6 @@ export function getChatResponse(courseId, unitId, promptExperimentVariationKey = }; } -export function clearMessages() { - return (dispatch) => { - dispatch(resetMessages()); - dispatch(resetApiError()); - }; -} - export function getLearningAssistantMessageHistory(courseId) { return async (dispatch) => { dispatch(setApiIsLoading(true)); diff --git a/src/widgets/Xpert.test.jsx b/src/widgets/Xpert.test.jsx index ec6bef04..532669f3 100644 --- a/src/widgets/Xpert.test.jsx +++ b/src/widgets/Xpert.test.jsx @@ -44,7 +44,6 @@ const assertSidebarElementsNotInDOM = () => { expect(screen.queryByRole('button', { name: 'submit' })).not.toBeInTheDocument(); expect(screen.queryByTestId('close-button')).not.toBeInTheDocument(); - expect(screen.queryByRole('button', { name: 'clear' })).not.toBeInTheDocument(); }; beforeEach(() => { @@ -115,7 +114,6 @@ test('clicking the call to action opens the sidebar', async () => { expect(screen.getByRole('textbox')).toBeVisible(); expect(screen.getByRole('button', { name: 'submit' })).toBeVisible(); expect(screen.getByTestId('close-button')).toBeVisible(); - expect(screen.getByRole('button', { name: 'clear' })).toBeVisible(); // assert that text input has focus expect(screen.getByRole('textbox')).toHaveFocus(); @@ -135,7 +133,6 @@ test('clicking the toggle button opens the sidebar', async () => { expect(screen.getByRole('textbox')).toBeVisible(); expect(screen.getByRole('button', { name: 'submit' })).toBeVisible(); expect(screen.getByTestId('close-button')).toBeVisible(); - expect(screen.getByRole('button', { name: 'clear' })).toBeVisible(); // assert that text input has focus expect(screen.getByRole('textbox')).toHaveFocus(); @@ -190,34 +187,6 @@ test('response text appears as message in the sidebar', async () => { await screen.findByText(responseMessage.content); expect(screen.getByText(responseMessage.content)).toBeInTheDocument(); }); -test('clicking the clear button clears messages in the sidebar', async () => { - const user = userEvent.setup(); - const userMessage = 'Hello, Xpert!'; - - // re-mock the fetchChatResponse API function so that we can assert that the - // responseMessage appears in the DOM and then is successfully cleared - const responseMessage = createRandomResponseForTesting(); - jest.spyOn(api, 'fetchChatResponse').mockImplementation(() => responseMessage); - - render(, { preloadedState: initialState }); - - // wait for button to appear - await screen.findByTestId('toggle-button'); - - await user.click(screen.queryByTestId('toggle-button')); - - // type the user message - const input = screen.getByRole('textbox'); - await user.type(input, userMessage); - await user.click(screen.getByRole('button', { name: 'submit' })); - - await screen.findByText(responseMessage.content); - - await user.click(screen.getByRole('button', { name: 'clear' })); - - expect(screen.queryByText(userMessage)).not.toBeInTheDocument(); - expect(screen.queryByText(responseMessage.content)).not.toBeInTheDocument(); -}); test('clicking the close button closes the sidebar', async () => { const user = userEvent.setup(); render(, { preloadedState: initialState }); @@ -307,35 +276,6 @@ test('error message should disappear when dismissed', async () => { await user.click(screen.getByText('Dismiss')); expect(screen.queryByText('Please try again by sending another question.')).not.toBeInTheDocument(); }); -test('error message should disappear when messages cleared', async () => { - const user = userEvent.setup(); - - const errorState = { - learningAssistant: { - currentMessage: '', - messageList: [], - apiIsLoading: false, - apiError: true, - // TEMPORARY: This is simply to ensure that the tests pass by hiding the disclosure. - // I will remove this and write tests in a future pull request. - disclosureAcknowledged: true, - sidebarIsOpen: false, - }, - }; - render(, { preloadedState: errorState }); - - // wait for button to appear - await screen.findByTestId('toggle-button'); - - await user.click(screen.queryByTestId('toggle-button')); - - // assert that error message exists - expect(screen.queryByText('Please try again by sending another question.')).toBeInTheDocument(); - - // clear messages, assert that error message disappears - await user.click(screen.getByText('Clear')); - expect(screen.queryByText('Please try again by sending another question.')).not.toBeInTheDocument(); -}); test('popup modal should open chat', async () => { const user = userEvent.setup(); window.localStorage.clear(); @@ -357,7 +297,6 @@ test('popup modal should open chat', async () => { expect(screen.getByRole('textbox')).toBeVisible(); expect(screen.getByRole('button', { name: 'submit' })).toBeVisible(); expect(screen.getByTestId('close-button')).toBeVisible(); - expect(screen.getByRole('button', { name: 'clear' })).toBeVisible(); }); test('popup modal should close and display CTA', async () => { const user = userEvent.setup();