-
Notifications
You must be signed in to change notification settings - Fork 311
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
chore(content-explorer): Migrate previewDialog (#3702)
Co-authored-by: mergify[bot] <37929162+mergify[bot]@users.noreply.github.com>
- Loading branch information
1 parent
ec0cd94
commit a426f57
Showing
7 changed files
with
401 additions
and
50 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
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1 +1,2 @@ | ||
README.md | ||
flow-typed/* |
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,109 @@ | ||
import * as React from 'react'; | ||
import { useIntl } from 'react-intl'; | ||
import Modal from 'react-modal'; | ||
import cloneDeep from 'lodash/cloneDeep'; | ||
|
||
import ContentPreview, { ContentPreviewProps } from '../content-preview'; | ||
import { TYPE_FILE, CLASS_MODAL_CONTENT_FULL_BLEED, CLASS_MODAL_OVERLAY, CLASS_MODAL } from '../../constants'; | ||
import type { Token, BoxItem, Collection } from '../../common/types/core'; | ||
import type APICache from '../../utils/Cache'; | ||
|
||
import messages from '../common/messages'; | ||
|
||
export interface PreviewDialogProps { | ||
apiHost: string; | ||
appElement: HTMLElement; | ||
appHost: string; | ||
cache: APICache; | ||
canDownload: boolean; | ||
contentPreviewProps: ContentPreviewProps; | ||
currentCollection: Collection; | ||
isOpen: boolean; | ||
isTouch: boolean; | ||
item: BoxItem; | ||
onCancel: () => void; | ||
onDownload: () => void; | ||
onPreview: (data: unknown) => void; | ||
parentElement: HTMLElement; | ||
previewLibraryVersion: string; | ||
requestInterceptor?: () => void; | ||
responseInterceptor?: () => void; | ||
sharedLink?: string; | ||
sharedLinkPassword?: string; | ||
staticHost: string; | ||
staticPath: string; | ||
token: Token; | ||
} | ||
|
||
const PreviewDialog = ({ | ||
apiHost, | ||
appElement, | ||
appHost, | ||
cache, | ||
canDownload, | ||
contentPreviewProps, | ||
currentCollection, | ||
isOpen, | ||
item, | ||
onCancel, | ||
onDownload, | ||
onPreview, | ||
parentElement, | ||
previewLibraryVersion, | ||
requestInterceptor, | ||
responseInterceptor, | ||
sharedLink, | ||
sharedLinkPassword, | ||
staticHost, | ||
staticPath, | ||
token, | ||
}: PreviewDialogProps) => { | ||
const { formatMessage } = useIntl(); | ||
const { items }: Collection = currentCollection; | ||
const onLoad = (data: unknown): void => { | ||
onPreview(cloneDeep(data)); | ||
}; | ||
|
||
if (!item || !items) { | ||
return null; | ||
} | ||
|
||
const files: BoxItem[] = items.filter(({ type }) => type === TYPE_FILE); | ||
return ( | ||
<Modal | ||
appElement={appElement} | ||
className={CLASS_MODAL_CONTENT_FULL_BLEED} | ||
contentLabel={formatMessage(messages.preview)} | ||
isOpen={isOpen} | ||
onRequestClose={onCancel} | ||
overlayClassName={CLASS_MODAL_OVERLAY} | ||
parentSelector={() => parentElement} | ||
portalClassName={CLASS_MODAL} | ||
> | ||
<ContentPreview | ||
{...contentPreviewProps} | ||
autoFocus | ||
apiHost={apiHost} | ||
appHost={appHost} | ||
cache={cache} | ||
canDownload={canDownload} | ||
collection={files} | ||
fileId={item.id} | ||
hasHeader | ||
onClose={onCancel} | ||
onDownload={onDownload} | ||
onLoad={onLoad} | ||
previewLibraryVersion={previewLibraryVersion} | ||
staticHost={staticHost} | ||
staticPath={staticPath} | ||
sharedLink={sharedLink} | ||
sharedLinkPassword={sharedLinkPassword} | ||
requestInterceptor={requestInterceptor} | ||
responseInterceptor={responseInterceptor} | ||
token={token} | ||
/> | ||
</Modal> | ||
); | ||
}; | ||
|
||
export default PreviewDialog; |
69 changes: 69 additions & 0 deletions
69
src/elements/content-explorer/__tests__/PreviewDialog.test.tsx
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,69 @@ | ||
import * as React from 'react'; | ||
import userEvent from '@testing-library/user-event'; | ||
import { render, screen, waitFor } from '../../../test-utils/testing-library'; | ||
import APICache from '../../../utils/Cache'; | ||
|
||
import PreviewDialog, { PreviewDialogProps } from '../PreviewDialog'; | ||
|
||
jest.mock('react-modal', () => { | ||
return jest.fn(({ children }) => <div aria-label="Preview">{children}</div>); | ||
}); | ||
|
||
describe('elements/content-explorer/PreviewDialog', () => { | ||
const defaultProps = { | ||
appElement: document.body, | ||
apiHost: 'https://api.box.com', | ||
appHost: 'https://app.box.com', | ||
cache: new APICache(), | ||
canDownload: true, | ||
contentPreviewProps: {}, | ||
currentCollection: { items: [{ id: '1', type: 'file' }] }, | ||
isOpen: true, | ||
isTouch: false, | ||
item: { id: '1', type: 'file' }, | ||
onCancel: jest.fn(), | ||
onDownload: jest.fn(), | ||
onPreview: jest.fn(), | ||
parentElement: document.createElement('div'), | ||
previewLibraryVersion: '1.0.0', | ||
staticHost: 'https://static.box.com', | ||
staticPath: '/static', | ||
token: 'token', | ||
}; | ||
|
||
const renderComponent = (props?: Partial<PreviewDialogProps>) => | ||
render(<PreviewDialog {...defaultProps} {...props} />); | ||
|
||
test('renders', async () => { | ||
renderComponent(); | ||
expect(await screen.findByLabelText('Preview')).toBeInTheDocument(); | ||
expect(await screen.findByRole('button', { name: 'Close' })); | ||
}); | ||
|
||
test('calls onCancel when modal is closed', async () => { | ||
renderComponent(); | ||
const closeButton = screen.getByRole('button', { name: 'Close' }); | ||
await userEvent.click(closeButton); | ||
expect(defaultProps.onCancel).toHaveBeenCalled(); | ||
}); | ||
|
||
test('does not render when item is null', () => { | ||
const props = { item: null }; | ||
const { container } = renderComponent(props); | ||
expect(container.firstChild).toBeNull(); | ||
}); | ||
|
||
test('does not render when items are null', () => { | ||
const props = { currentCollection: { items: null } }; | ||
const { container } = renderComponent(props); | ||
expect(container.firstChild).toBeNull(); | ||
}); | ||
|
||
test('calls onPreview with cloned data on load', () => { | ||
const data = { id: '1', type: 'file' }; | ||
renderComponent(); | ||
waitFor(() => { | ||
expect(defaultProps.onPreview).toHaveBeenCalledWith(expect.objectContaining(data)); | ||
}); | ||
}); | ||
}); |
Oops, something went wrong.