-
Notifications
You must be signed in to change notification settings - Fork 81
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
feat: preview library block changes in course unit [FC-0062] (#1506)
Creates a new preview library block modal. Intercepts the message when the block is iframed to open the new modal.
- Loading branch information
1 parent
31f59d6
commit 7aa5acc
Showing
17 changed files
with
458 additions
and
69 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,19 @@ | ||
import { useMutation } from '@tanstack/react-query'; | ||
|
||
import { acceptLibraryBlockChanges, ignoreLibraryBlockChanges } from './api'; | ||
|
||
/** | ||
* Hook that provides a "mutation" that can be used to accept library block changes. | ||
*/ | ||
// eslint-disable-next-line import/prefer-default-export | ||
export const useAcceptLibraryBlockChanges = () => useMutation({ | ||
mutationFn: acceptLibraryBlockChanges, | ||
}); | ||
|
||
/** | ||
* Hook that provides a "mutation" that can be used to ignore library block changes. | ||
*/ | ||
// eslint-disable-next-line import/prefer-default-export | ||
export const useIgnoreLibraryBlockChanges = () => useMutation({ | ||
mutationFn: ignoreLibraryBlockChanges, | ||
}); |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,4 @@ | ||
.lib-preview-xblock-changes-modal { | ||
border-bottom-right-radius: 0; | ||
border-bottom-left-radius: 0; | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,138 @@ | ||
import userEvent from '@testing-library/user-event'; | ||
import MockAdapter from 'axios-mock-adapter/types'; | ||
import { | ||
act, | ||
render as baseRender, | ||
screen, | ||
initializeMocks, | ||
waitFor, | ||
} from '../../testUtils'; | ||
|
||
import PreviewLibraryXBlockChanges, { LibraryChangesMessageData } from '.'; | ||
import { messageTypes } from '../constants'; | ||
import { IframeProvider } from '../context/iFrameContext'; | ||
import { libraryBlockChangesUrl } from '../data/api'; | ||
import { ToastActionData } from '../../generic/toast-context'; | ||
import { getLibraryBlockMetadataUrl } from '../../library-authoring/data/api'; | ||
|
||
const usageKey = 'some-id'; | ||
const defaultEventData: LibraryChangesMessageData = { | ||
displayName: 'Test block', | ||
downstreamBlockId: usageKey, | ||
upstreamBlockId: 'some-lib-id', | ||
upstreamBlockVersionSynced: 1, | ||
isVertical: false, | ||
}; | ||
|
||
const mockSendMessageToIframe = jest.fn(); | ||
jest.mock('../context/hooks', () => ({ | ||
useIframe: () => ({ | ||
sendMessageToIframe: mockSendMessageToIframe, | ||
}), | ||
})); | ||
const render = (eventData?: LibraryChangesMessageData) => { | ||
baseRender(<PreviewLibraryXBlockChanges />, { | ||
extraWrapper: ({ children }) => <IframeProvider>{ children }</IframeProvider>, | ||
}); | ||
const message = { | ||
data: { | ||
type: messageTypes.showXBlockLibraryChangesPreview, | ||
payload: eventData || defaultEventData, | ||
}, | ||
}; | ||
// Dispatch showXBlockLibraryChangesPreview message event to open the preivew modal. | ||
act(() => { | ||
window.dispatchEvent(new MessageEvent('message', message)); | ||
}); | ||
}; | ||
|
||
let axiosMock: MockAdapter; | ||
let mockShowToast: (message: string, action?: ToastActionData | undefined) => void; | ||
|
||
describe('<PreviewLibraryXBlockChanges />', () => { | ||
beforeEach(() => { | ||
const mocks = initializeMocks(); | ||
axiosMock = mocks.axiosMock; | ||
mockShowToast = mocks.mockShowToast; | ||
}); | ||
|
||
it('renders modal', async () => { | ||
render(); | ||
|
||
expect(await screen.findByText('Preview changes: Test block')).toBeInTheDocument(); | ||
expect(await screen.findByRole('button', { name: 'Accept changes' })).toBeInTheDocument(); | ||
expect(await screen.findByRole('button', { name: 'Ignore changes' })).toBeInTheDocument(); | ||
expect(await screen.findByRole('button', { name: 'Cancel' })).toBeInTheDocument(); | ||
expect(await screen.findByRole('tab', { name: 'New version' })).toBeInTheDocument(); | ||
expect(await screen.findByRole('tab', { name: 'Old version' })).toBeInTheDocument(); | ||
}); | ||
|
||
it('renders displayName for units', async () => { | ||
render({ ...defaultEventData, isVertical: true, displayName: '' }); | ||
|
||
expect(await screen.findByText('Preview changes: Unit')).toBeInTheDocument(); | ||
}); | ||
|
||
it('renders default displayName for components with no displayName', async () => { | ||
render({ ...defaultEventData, displayName: '' }); | ||
|
||
expect(await screen.findByText('Preview changes: Component')).toBeInTheDocument(); | ||
}); | ||
|
||
it('renders both new and old title if they are different', async () => { | ||
axiosMock.onGet(getLibraryBlockMetadataUrl(defaultEventData.upstreamBlockId)).reply(200, { | ||
displayName: 'New test block', | ||
}); | ||
render(); | ||
|
||
expect(await screen.findByText('Preview changes: Test block -> New test block')).toBeInTheDocument(); | ||
}); | ||
|
||
it('accept changes works', async () => { | ||
axiosMock.onPost(libraryBlockChangesUrl(usageKey)).reply(200, {}); | ||
render(); | ||
|
||
expect(await screen.findByText('Preview changes: Test block')).toBeInTheDocument(); | ||
const acceptBtn = await screen.findByRole('button', { name: 'Accept changes' }); | ||
userEvent.click(acceptBtn); | ||
await waitFor(() => { | ||
expect(mockSendMessageToIframe).toHaveBeenCalledWith(messageTypes.refreshXBlock, null); | ||
expect(axiosMock.history.post.length).toEqual(1); | ||
expect(axiosMock.history.post[0].url).toEqual(libraryBlockChangesUrl(usageKey)); | ||
}); | ||
expect(screen.queryByText('Preview changes: Test block')).not.toBeInTheDocument(); | ||
}); | ||
|
||
it('shows toast if accept changes fails', async () => { | ||
axiosMock.onPost(libraryBlockChangesUrl(usageKey)).reply(500, {}); | ||
render(); | ||
|
||
expect(await screen.findByText('Preview changes: Test block')).toBeInTheDocument(); | ||
const acceptBtn = await screen.findByRole('button', { name: 'Accept changes' }); | ||
userEvent.click(acceptBtn); | ||
await waitFor(() => { | ||
expect(mockSendMessageToIframe).not.toHaveBeenCalledWith(messageTypes.refreshXBlock, null); | ||
expect(axiosMock.history.post.length).toEqual(1); | ||
expect(axiosMock.history.post[0].url).toEqual(libraryBlockChangesUrl(usageKey)); | ||
}); | ||
expect(screen.queryByText('Preview changes: Test block')).not.toBeInTheDocument(); | ||
expect(mockShowToast).toHaveBeenCalledWith('Failed to update component'); | ||
}); | ||
|
||
it('ignore changes works', async () => { | ||
axiosMock.onDelete(libraryBlockChangesUrl(usageKey)).reply(200, {}); | ||
render(); | ||
|
||
expect(await screen.findByText('Preview changes: Test block')).toBeInTheDocument(); | ||
const ignoreBtn = await screen.findByRole('button', { name: 'Ignore changes' }); | ||
userEvent.click(ignoreBtn); | ||
const ignoreConfirmBtn = await screen.findByRole('button', { name: 'Ignore' }); | ||
userEvent.click(ignoreConfirmBtn); | ||
await waitFor(() => { | ||
expect(mockSendMessageToIframe).toHaveBeenCalledWith(messageTypes.refreshXBlock, null); | ||
expect(axiosMock.history.delete.length).toEqual(1); | ||
expect(axiosMock.history.delete[0].url).toEqual(libraryBlockChangesUrl(usageKey)); | ||
}); | ||
expect(screen.queryByText('Preview changes: Test block')).not.toBeInTheDocument(); | ||
}); | ||
}); |
Oops, something went wrong.