Skip to content

Commit

Permalink
feat: add upload progress modal (openedx#1113)
Browse files Browse the repository at this point in the history
* feat: add upload progress modal

* fix: increase code coverage

* fix: fix code to be more readable

* fix: delete empty file
  • Loading branch information
KristinAoki authored Jun 24, 2024
1 parent 6414196 commit 8ef804b
Show file tree
Hide file tree
Showing 17 changed files with 599 additions and 198 deletions.
9 changes: 6 additions & 3 deletions src/files-and-videos/files-page/FilesPage.test.jsx
Original file line number Diff line number Diff line change
Expand Up @@ -563,7 +563,10 @@ describe('FilesAndUploads', () => {
const addStatus = store.getState().assets.addingStatus;
expect(addStatus).toEqual(RequestStatus.FAILED);
});
expect(screen.getByText('Error')).toBeVisible();
const addStatus = store.getState().assets.addingStatus;
expect(addStatus).toEqual(RequestStatus.FAILED);

expect(screen.getByText('Upload error')).toBeVisible();
});

it('404 validation should show error', async () => {
Expand All @@ -575,7 +578,7 @@ describe('FilesAndUploads', () => {
const addStatus = store.getState().assets.addingStatus;
expect(addStatus).toEqual(RequestStatus.FAILED);

expect(screen.getByText('Error')).toBeVisible();
expect(screen.getByText('Upload error')).toBeVisible();
});

it('404 upload should show error', async () => {
Expand All @@ -588,7 +591,7 @@ describe('FilesAndUploads', () => {
const addStatus = store.getState().assets.addingStatus;
expect(addStatus).toEqual(RequestStatus.FAILED);

expect(screen.getByText('Error')).toBeVisible();
expect(screen.getByText('Upload error')).toBeVisible();
});

it('404 delete should show error', async () => {
Expand Down
9 changes: 9 additions & 0 deletions src/files-and-videos/files-page/messages.js
Original file line number Diff line number Diff line change
Expand Up @@ -111,6 +111,15 @@ const messages = defineMessages({
defaultMessage: 'Cancel',
description: 'The message displayed in the button to confirm cancelling the upload',
},
lockFileTooltipContent: {
id: 'course-authoring.files-and-uploads.file-info.lockFile.tooltip.content',
defaultMessage: `By default, anyone can access a file you upload if
they know the web URL, even if they are not enrolled in your course.
You can prevent outside access to a file by locking the file. When
you lock a file, the web URL only allows learners who are enrolled
in your course and signed in to access the file.`,
description: 'Tooltip message for the lock icon in the table view of files',
},
});

export default messages;
6 changes: 5 additions & 1 deletion src/files-and-videos/generic/EditFileErrors.jsx
Original file line number Diff line number Diff line change
@@ -1,6 +1,7 @@
import React from 'react';
import PropTypes from 'prop-types';
import { injectIntl, intlShape } from '@edx/frontend-platform/i18n';
import { Alert } from '@openedx/paragon';
import { ErrorAlert } from '@edx/frontend-lib-content-components';
import { RequestStatus } from '../../data/constants';
import messages from './messages';
Expand All @@ -24,10 +25,13 @@ const EditFileErrors = ({
{intl.formatMessage(messages.errorAlertMessage, { message: errorMessages.loading })}
</ErrorAlert>
<ErrorAlert
hideHeading={false}
hideHeading
dismissError={() => resetErrors({ errorType: 'add' })}
isError={addFileStatus === RequestStatus.FAILED}
>
<Alert.Heading>
{intl.formatMessage(messages.uploadErrorAlertTitle)}
</Alert.Heading>
<ul className="p-0">
{errorMessages.add.map(message => (
<li key={`add-error-${message}`} style={{ listStyle: 'none' }}>
Expand Down
6 changes: 6 additions & 0 deletions src/files-and-videos/generic/messages.js
Original file line number Diff line number Diff line change
Expand Up @@ -202,6 +202,12 @@ const messages = defineMessages({
failedLabel: {
id: 'course-authoring.files-and-uploads.filter.failed.label',
defaultMessage: 'Failed',
description: 'Label for failed sort button in sort and filter modal',
},
uploadErrorAlertTitle: {
id: 'course-authoring.files-and-uploads.error.upload.title',
defaultMessage: 'Upload error',
description: 'Title for upload error alert',
},
});

Expand Down
47 changes: 30 additions & 17 deletions src/files-and-videos/videos-page/VideosPage.jsx
Original file line number Diff line number Diff line change
@@ -1,19 +1,18 @@
import React, { useEffect, useRef } from 'react';
import PropTypes from 'prop-types';
import { Helmet } from 'react-helmet';
import { useDispatch, useSelector } from 'react-redux';
import {
injectIntl,
FormattedMessage,
intlShape,
} from '@edx/frontend-platform/i18n';
import {
useToggle,
ActionRow,
Button,
CheckboxFilter,
Container,
Alert,
Spinner,
useToggle,
} from '@openedx/paragon';
import Placeholder from '@edx/frontend-lib-content-components';

Expand All @@ -29,6 +28,7 @@ import {
markVideoUploadsInProgressAsFailed,
resetErrors,
updateVideoOrder,
cancelAllUploads,
} from './data/thunks';
import messages from './messages';
import VideosPageProvider from './VideosPageProvider';
Expand All @@ -41,11 +41,12 @@ import {
ThumbnailColumn,
TranscriptColumn,
} from '../generic';
import TranscriptSettings from './transcript-settings';
import VideoThumbnail from './VideoThumbnail';
import { getFormattedDuration, resampleFile } from './data/utils';
import FILES_AND_UPLOAD_TYPE_FILTERS from '../generic/constants';
import TranscriptSettings from './transcript-settings';
import VideoInfoModalSidebar from './info-sidebar';
import VideoThumbnail from './VideoThumbnail';
import UploadModal from './upload-modal';

const VideosPage = ({
courseId,
Expand All @@ -58,11 +59,12 @@ const VideosPage = ({
openTranscriptSettings,
closeTranscriptSettings,
] = useToggle(false);
const [
isUploadTrackerOpen,
openUploadTracker,
closeUploadTracker,
] = useToggle(false);
const courseDetails = useModel('courseDetails', courseId);
document.title = getPageHeadTitle(
courseDetails?.name,
intl.formatMessage(messages.heading),
);

useEffect(() => {
dispatch(fetchVideos(courseId));
Expand All @@ -80,7 +82,7 @@ const VideosPage = ({
pageSettings,
} = useSelector((state) => state.videos);

const uploadingIdsRef = useRef([]);
const uploadingIdsRef = useRef({ uploadData: {}, uploadCount: 0 });

useEffect(() => {
window.onbeforeunload = () => {
Expand All @@ -90,6 +92,11 @@ const VideosPage = ({
}
return undefined;
};
if (addVideoStatus === RequestStatus.IN_PROGRESS) {
openUploadTracker();
} else {
closeUploadTracker();
}
}, [addVideoStatus]);

const {
Expand All @@ -103,11 +110,12 @@ const VideosPage = ({
const supportedFileFormats = {
'video/*': videoSupportedFileFormats || FILES_AND_UPLOAD_TYPE_FILTERS.video,
};

const handleUploadCancel = () => dispatch(cancelAllUploads(courseId, uploadingIdsRef.current.uploadData));
const handleErrorReset = (error) => dispatch(resetErrors(error));
const handleAddFile = (files) => {
handleErrorReset({ errorType: 'add' });
files.forEach((file) => dispatch(addVideoFile(courseId, file, videoIds, uploadingIdsRef)));
uploadingIdsRef.current.uploadCount = files.length;
dispatch(addVideoFile(courseId, files, videoIds, uploadingIdsRef));
};
const handleDeleteFile = (id) => dispatch(deleteVideoFile(courseId, id));
const handleDownloadFile = (selectedRows) => dispatch(fetchVideoDownload({ selectedRows, courseId }));
Expand Down Expand Up @@ -222,6 +230,9 @@ const VideosPage = ({

return (
<VideosPageProvider courseId={courseId}>
<Helmet>
<title>{getPageHeadTitle(courseDetails?.name, intl.formatMessage(messages.heading))}</title>
</Helmet>
<Container size="xl" className="p-4 pt-4.5">
<EditFileErrors
resetErrors={handleErrorReset}
Expand All @@ -231,11 +242,6 @@ const VideosPage = ({
updateFileStatus={updateVideoStatus}
loadingStatus={loadingStatus}
/>
<Alert variant="warning" show={addVideoStatus === RequestStatus.IN_PROGRESS}>
<div className="video-upload-warning-text"><Spinner animation="border" variant="warning" className="video-upload-spinner mr-3" screenReaderText="loading" />
<p className="d-inline"><FormattedMessage {...messages.videoUploadAlertLabel} /></p>
</div>
</Alert>
<ActionRow>
<div className="h2">
<FormattedMessage {...messages.heading} />
Expand Down Expand Up @@ -287,6 +293,13 @@ const VideosPage = ({
/>
</>
)}
<UploadModal
{...{
isUploadTrackerOpen,
currentUploadingIdsRef: uploadingIdsRef.current,
handleUploadCancel,
}}
/>
</Container>
</VideosPageProvider>
);
Expand Down
Loading

0 comments on commit 8ef804b

Please sign in to comment.