diff --git a/packages/core/upload/admin/src/pages/App/components/DocAssetCard.js b/packages/core/upload/admin/src/pages/App/components/DocAssetCard.js
new file mode 100644
index 0000000000..81a2d702f0
--- /dev/null
+++ b/packages/core/upload/admin/src/pages/App/components/DocAssetCard.js
@@ -0,0 +1,69 @@
+import React from 'react';
+import PropTypes from 'prop-types';
+import styled from 'styled-components';
+import {
+ Card,
+ CardAction,
+ CardAsset,
+ CardBadge,
+ CardBody,
+ CardCheckbox,
+ CardContent,
+ CardHeader,
+ CardTitle,
+ CardSubtitle,
+} from '@strapi/parts/Card';
+import { IconButton } from '@strapi/parts/IconButton';
+import EditIcon from '@strapi/icons/EditIcon';
+import IconDocumentation from '@strapi/icons/IconDocumentation';
+import { useIntl } from 'react-intl';
+import { getTrad } from '../../../utils';
+
+const Extension = styled.span`
+ text-transform: uppercase;
+`;
+
+const IconWrapper = styled.span`
+ svg {
+ font-size: 3rem;
+ }
+`;
+
+export const DocAssetCard = ({ name, extension }) => {
+ const { formatMessage } = useIntl();
+
+ return (
+
+
+
+
+ }
+ />
+
+
+
+
+
+
+
+
+
+ {name}
+
+ {extension.replace('.', '')}
+
+
+
+ {formatMessage({ id: getTrad('settings.section.doc.label'), defaultMessage: 'Doc' })}
+
+
+
+ );
+};
+
+DocAssetCard.propTypes = {
+ extension: PropTypes.string.isRequired,
+ name: PropTypes.string.isRequired,
+};
diff --git a/packages/core/upload/admin/src/pages/App/components/ImageAssetCard.js b/packages/core/upload/admin/src/pages/App/components/ImageAssetCard.js
new file mode 100644
index 0000000000..c61c592a0b
--- /dev/null
+++ b/packages/core/upload/admin/src/pages/App/components/ImageAssetCard.js
@@ -0,0 +1,61 @@
+import React from 'react';
+import PropTypes from 'prop-types';
+import styled from 'styled-components';
+import {
+ Card,
+ CardAction,
+ CardAsset,
+ CardBadge,
+ CardBody,
+ CardCheckbox,
+ CardContent,
+ CardHeader,
+ CardTitle,
+ CardSubtitle,
+} from '@strapi/parts/Card';
+import { IconButton } from '@strapi/parts/IconButton';
+import EditIcon from '@strapi/icons/EditIcon';
+import { useIntl } from 'react-intl';
+import { getTrad } from '../../../utils';
+
+const Extension = styled.span`
+ text-transform: uppercase;
+`;
+
+export const ImageAssetCard = ({ name, extension, height, width, thumbnail }) => {
+ const { formatMessage } = useIntl();
+
+ return (
+
+
+
+
+ }
+ />
+
+
+
+
+
+ {name}
+
+ {extension.replace('.', '')} - {height}✕{width}
+
+
+
+ {formatMessage({ id: getTrad('settings.section.image.label'), defaultMessage: 'Image' })}
+
+
+
+ );
+};
+
+ImageAssetCard.propTypes = {
+ extension: PropTypes.string.isRequired,
+ height: PropTypes.number.isRequired,
+ name: PropTypes.string.isRequired,
+ width: PropTypes.number.isRequired,
+ thumbnail: PropTypes.string.isRequired,
+};
diff --git a/packages/core/upload/admin/src/pages/App/components/ListView.js b/packages/core/upload/admin/src/pages/App/components/ListView.js
index f48a6468b2..1550c21f06 100644
--- a/packages/core/upload/admin/src/pages/App/components/ListView.js
+++ b/packages/core/upload/admin/src/pages/App/components/ListView.js
@@ -1,9 +1,50 @@
import React from 'react';
import PropTypes from 'prop-types';
+import { GridLayout } from '@strapi/parts/Layout';
+import { KeyboardNavigable } from '@strapi/parts/KeyboardNavigable';
+import { ImageAssetCard } from './ImageAssetCard';
+import { VideoAssetCard } from './VideoAssetCard';
+import { DocAssetCard } from './DocAssetCard';
-// TODO: implement the view
export const ListView = ({ assets }) => {
- return
Number of assets: {assets.length}
;
+ return (
+
+
+ {assets.map(asset => {
+ if (asset.mime.includes('video')) {
+ return (
+
+ );
+ }
+
+ if (asset.mime.includes('image')) {
+ return (
+
+ );
+ }
+
+ return (
+
+ );
+ })}
+
+
+ );
};
ListView.propTypes = {
diff --git a/packages/core/upload/admin/src/pages/App/components/VideoAssetCard.js b/packages/core/upload/admin/src/pages/App/components/VideoAssetCard.js
new file mode 100644
index 0000000000..fb41e24b67
--- /dev/null
+++ b/packages/core/upload/admin/src/pages/App/components/VideoAssetCard.js
@@ -0,0 +1,67 @@
+import React, { useState } from 'react';
+import PropTypes from 'prop-types';
+import styled from 'styled-components';
+import {
+ Card,
+ CardAction,
+ CardAsset,
+ CardBadge,
+ CardBody,
+ CardCheckbox,
+ CardContent,
+ CardHeader,
+ CardTitle,
+ CardSubtitle,
+ CardTimer,
+} from '@strapi/parts/Card';
+import { IconButton } from '@strapi/parts/IconButton';
+import EditIcon from '@strapi/icons/EditIcon';
+import { useIntl } from 'react-intl';
+import { VideoPreview } from './VideoPreview';
+import { getTrad, formatDuration } from '../../../utils';
+
+const Extension = styled.span`
+ text-transform: uppercase;
+`;
+
+export const VideoAssetCard = ({ name, extension, url, mime }) => {
+ const { formatMessage } = useIntl();
+ const [duration, setDuration] = useState();
+ const formattedDuration = duration ? formatDuration(duration) : undefined;
+
+ return (
+
+
+
+
+ }
+ />
+
+
+
+
+ {formattedDuration || '...'}
+
+
+
+ {name}
+
+ {extension.replace('.', '')}
+
+
+
+ {formatMessage({ id: getTrad('settings.section.video.label'), defaultMessage: 'Doc' })}
+
+
+
+ );
+};
+
+VideoAssetCard.propTypes = {
+ extension: PropTypes.string.isRequired,
+ mime: PropTypes.string.isRequired,
+ name: PropTypes.string.isRequired,
+ url: PropTypes.string.isRequired,
+};
diff --git a/packages/core/upload/admin/src/pages/App/components/VideoPreview.js b/packages/core/upload/admin/src/pages/App/components/VideoPreview.js
new file mode 100644
index 0000000000..f5474c8bda
--- /dev/null
+++ b/packages/core/upload/admin/src/pages/App/components/VideoPreview.js
@@ -0,0 +1,52 @@
+/* eslint-disable jsx-a11y/media-has-caption */
+import React, { useRef, useState } from 'react';
+import PropTypes from 'prop-types';
+import styled from 'styled-components';
+
+const VideoPreviewWrapper = styled.div`
+ canvas {
+ display: block;
+ max-width: 100%;
+ max-height: 100%;
+ }
+`;
+
+export const VideoPreview = ({ url, mime, onLoadDuration }) => {
+ const [loaded, setLoaded] = useState(false);
+ const videoRef = useRef(null);
+ const canvasRef = useRef(null);
+
+ const handleThumbnailVisibility = () => {
+ const video = videoRef.current;
+ const canvas = canvasRef.current;
+
+ canvas.height = video.videoHeight;
+ canvas.width = video.videoWidth;
+ canvas.getContext('2d').drawImage(video, 0, 0);
+
+ onLoadDuration(video.duration);
+ setLoaded(true);
+ };
+
+ return (
+
+ {!loaded && (
+
+ )}
+
+
+
+ );
+};
+
+VideoPreview.propTypes = {
+ url: PropTypes.string.isRequired,
+ mime: PropTypes.string.isRequired,
+ onLoadDuration: PropTypes.func.isRequired,
+};
diff --git a/packages/core/upload/admin/src/pages/App/components/tests/ListView.test.js b/packages/core/upload/admin/src/pages/App/components/tests/ListView.test.js
new file mode 100644
index 0000000000..96ea02eabc
--- /dev/null
+++ b/packages/core/upload/admin/src/pages/App/components/tests/ListView.test.js
@@ -0,0 +1,792 @@
+import React from 'react';
+import { ThemeProvider, lightTheme } from '@strapi/parts';
+import { render as renderTL } from '@testing-library/react';
+import { ListView } from '../ListView';
+import en from '../../../../translations/en.json';
+
+jest.mock('../../../../utils', () => ({
+ ...jest.requireActual('../../../../utils'),
+ getTrad: x => x,
+}));
+
+jest.mock('react-intl', () => ({
+ FormattedMessage: ({ id }) => id,
+ useIntl: () => ({ formatMessage: jest.fn(({ id }) => en[id]) }),
+}));
+
+const data = [
+ {
+ id: 1,
+ name: 'strapi-cover_1fabc982ce.png',
+ alternativeText: '',
+ caption: '',
+ width: 1066,
+ height: 551,
+ formats: {
+ thumbnail: {
+ name: 'thumbnail_strapi-cover_1fabc982ce.png',
+ hash: 'thumbnail_strapi_cover_1fabc982ce_5b43615ed5',
+ ext: '.png',
+ mime: 'image/png',
+ width: 245,
+ height: 127,
+ size: 3.37,
+ path: null,
+ url: '/uploads/thumbnail_strapi_cover_1fabc982ce_5b43615ed5.png',
+ },
+ large: {
+ name: 'large_strapi-cover_1fabc982ce.png',
+ hash: 'large_strapi_cover_1fabc982ce_5b43615ed5',
+ ext: '.png',
+ mime: 'image/png',
+ width: 1000,
+ height: 517,
+ size: 22.43,
+ path: null,
+ url: '/uploads/large_strapi_cover_1fabc982ce_5b43615ed5.png',
+ },
+ medium: {
+ name: 'medium_strapi-cover_1fabc982ce.png',
+ hash: 'medium_strapi_cover_1fabc982ce_5b43615ed5',
+ ext: '.png',
+ mime: 'image/png',
+ width: 750,
+ height: 388,
+ size: 14.62,
+ path: null,
+ url: '/uploads/medium_strapi_cover_1fabc982ce_5b43615ed5.png',
+ },
+ small: {
+ name: 'small_strapi-cover_1fabc982ce.png',
+ hash: 'small_strapi_cover_1fabc982ce_5b43615ed5',
+ ext: '.png',
+ mime: 'image/png',
+ width: 500,
+ height: 258,
+ size: 8.38,
+ path: null,
+ url: '/uploads/small_strapi_cover_1fabc982ce_5b43615ed5.png',
+ },
+ },
+ hash: 'strapi_cover_1fabc982ce_5b43615ed5',
+ ext: '.png',
+ mime: 'image/png',
+ size: 6.85,
+ url: '/uploads/strapi_cover_1fabc982ce_5b43615ed5.png',
+ previewUrl: null,
+ provider: 'local',
+ provider_metadata: null,
+ created_at: '2021-09-14T07:32:50.816Z',
+ updated_at: '2021-09-14T07:32:50.816Z',
+ },
+ {
+ id: 5,
+ name: 'mov_bbb.mp4',
+ alternativeText: '',
+ caption: '',
+ width: null,
+ height: null,
+ formats: null,
+ hash: 'mov_bbb_2f3907f7aa',
+ ext: '.mp4',
+ mime: 'video/mp4',
+ size: 788.49,
+ url: '/uploads/mov_bbb_2f3907f7aa.mp4',
+ previewUrl: null,
+ provider: 'local',
+ provider_metadata: null,
+ created_at: '2021-09-14T07:48:30.882Z',
+ updated_at: '2021-09-14T07:48:30.882Z',
+ },
+ {
+ id: 6,
+ name: 'CARTE MARIAGE AVS - Printemps.pdf',
+ alternativeText: '',
+ caption: '',
+ width: null,
+ height: null,
+ formats: null,
+ hash: 'CARTE_MARIAGE_AVS_Printemps_1f87b19e18',
+ ext: '.pdf',
+ mime: 'application/pdf',
+ size: 422.37,
+ url: '/uploads/CARTE_MARIAGE_AVS_Printemps_1f87b19e18.pdf',
+ previewUrl: null,
+ provider: 'local',
+ provider_metadata: null,
+ created_at: '2021-09-14T07:51:59.845Z',
+ updated_at: '2021-09-14T07:51:59.845Z',
+ },
+];
+
+describe('MediaLibrary / ListView', () => {
+ it('snapshots the listview', () => {
+ const { container } = renderTL(
+
+
+
+ );
+
+ expect(container).toMatchInlineSnapshot(`
+ .c29 {
+ border: 0;
+ -webkit-clip: rect(0 0 0 0);
+ clip: rect(0 0 0 0);
+ height: 1px;
+ margin: -1px;
+ overflow: hidden;
+ padding: 0;
+ position: absolute;
+ width: 1px;
+ }
+
+ .c0 {
+ display: grid;
+ grid-template-columns: repeat(auto-fit,minmax(250px,1fr));
+ grid-gap: 16px;
+ }
+
+ .c1 {
+ background: #ffffff;
+ border-radius: 4px;
+ box-shadow: 0px 1px 4px rgba(33,33,52,0.1);
+ }
+
+ .c11 {
+ padding-top: 8px;
+ padding-right: 12px;
+ padding-bottom: 8px;
+ padding-left: 12px;
+ }
+
+ .c16 {
+ background: #f6f6f9;
+ color: #666687;
+ padding: 4px;
+ border-radius: 4px;
+ }
+
+ .c23 {
+ background: #32324d;
+ color: #ffffff;
+ padding: 4px;
+ border-radius: 4px;
+ }
+
+ .c4 {
+ position: absolute;
+ top: 12px;
+ left: 12px;
+ }
+
+ .c6 {
+ position: absolute;
+ top: 12px;
+ right: 12px;
+ }
+
+ .c2 {
+ display: -webkit-box;
+ display: -webkit-flex;
+ display: -ms-flexbox;
+ display: flex;
+ -webkit-flex-direction: row;
+ -ms-flex-direction: row;
+ flex-direction: row;
+ -webkit-box-pack: center;
+ -webkit-justify-content: center;
+ -ms-flex-pack: center;
+ justify-content: center;
+ -webkit-align-items: center;
+ -webkit-box-align: center;
+ -ms-flex-align: center;
+ align-items: center;
+ }
+
+ .c12 {
+ display: -webkit-box;
+ display: -webkit-flex;
+ display: -ms-flexbox;
+ display: flex;
+ -webkit-flex-direction: row;
+ -ms-flex-direction: row;
+ flex-direction: row;
+ -webkit-align-items: center;
+ -webkit-box-align: center;
+ -ms-flex-align: center;
+ align-items: center;
+ }
+
+ .c10 {
+ margin: 0;
+ padding: 0;
+ max-height: 100%;
+ max-width: 100%;
+ }
+
+ .c9 {
+ display: -webkit-box;
+ display: -webkit-flex;
+ display: -ms-flexbox;
+ display: flex;
+ -webkit-box-pack: center;
+ -webkit-justify-content: center;
+ -ms-flex-pack: center;
+ justify-content: center;
+ height: 10.25rem;
+ width: 100%;
+ background: repeating-conic-gradient(#f6f6f9 0% 25%,transparent 0% 50%) 50% / 20px 20px;
+ }
+
+ .c13 {
+ font-weight: 500;
+ font-size: 0.75rem;
+ line-height: 1.33;
+ color: #32324d;
+ }
+
+ .c14 {
+ font-weight: 400;
+ font-size: 0.75rem;
+ line-height: 1.33;
+ color: #666687;
+ }
+
+ .c19 {
+ font-weight: 400;
+ font-size: 0.875rem;
+ line-height: 1.43;
+ color: #32324d;
+ }
+
+ .c25 {
+ font-weight: 400;
+ font-size: 0.75rem;
+ line-height: 1.33;
+ color: #ffffff;
+ }
+
+ .c20 {
+ font-weight: 600;
+ line-height: 1.14;
+ }
+
+ .c21 {
+ font-weight: 600;
+ font-size: 0.6875rem;
+ line-height: 1.45;
+ text-transform: uppercase;
+ }
+
+ .c17 {
+ display: inline-block;
+ }
+
+ .c18 {
+ margin-left: auto;
+ }
+
+ .c5 {
+ margin: 0;
+ height: 18px;
+ min-width: 18px;
+ border-radius: 4px;
+ border: 1px solid #c0c0cf;
+ -webkit-appearance: none;
+ background-color: #ffffff;
+ }
+
+ .c5:checked {
+ background-color: #4945ff;
+ border: 1px solid #4945ff;
+ }
+
+ .c5:checked:after {
+ content: '';
+ display: block;
+ position: relative;
+ background: url() no-repeat no-repeat center center;
+ width: 10px;
+ height: 10px;
+ left: 50%;
+ top: 50%;
+ -webkit-transform: translateX(-50%) translateY(-50%);
+ -ms-transform: translateX(-50%) translateY(-50%);
+ transform: translateX(-50%) translateY(-50%);
+ }
+
+ .c5:checked:disabled:after {
+ background: url() no-repeat no-repeat center center;
+ }
+
+ .c5:disabled {
+ background-color: #dcdce4;
+ border: 1px solid #c0c0cf;
+ }
+
+ .c5:indeterminate {
+ background-color: #4945ff;
+ border: 1px solid #4945ff;
+ }
+
+ .c5:indeterminate:after {
+ content: '';
+ display: block;
+ position: relative;
+ color: white;
+ height: 2px;
+ width: 10px;
+ background-color: #ffffff;
+ left: 50%;
+ top: 50%;
+ -webkit-transform: translateX(-50%) translateY(-50%);
+ -ms-transform: translateX(-50%) translateY(-50%);
+ transform: translateX(-50%) translateY(-50%);
+ }
+
+ .c5:indeterminate:disabled {
+ background-color: #dcdce4;
+ border: 1px solid #c0c0cf;
+ }
+
+ .c5:indeterminate:disabled:after {
+ background-color: #8e8ea9;
+ }
+
+ .c3 {
+ position: relative;
+ border-bottom: 1px solid #eaeaef;
+ }
+
+ .c24 {
+ position: absolute;
+ bottom: 4px;
+ right: 4px;
+ }
+
+ .c7 {
+ display: -webkit-box;
+ display: -webkit-flex;
+ display: -ms-flexbox;
+ display: flex;
+ cursor: pointer;
+ padding: 8px;
+ border-radius: 4px;
+ background: #ffffff;
+ border: 1px solid #dcdce4;
+ }
+
+ .c7 svg {
+ height: 12px;
+ width: 12px;
+ }
+
+ .c7 svg > g,
+ .c7 svg path {
+ fill: #ffffff;
+ }
+
+ .c7[aria-disabled='true'] {
+ pointer-events: none;
+ }
+
+ .c8 svg > g,
+ .c8 svg path {
+ fill: #8e8ea9;
+ }
+
+ .c8:hover svg > g,
+ .c8:hover svg path {
+ fill: #666687;
+ }
+
+ .c8:active svg > g,
+ .c8:active svg path {
+ fill: #a5a5ba;
+ }
+
+ .c8[aria-disabled='true'] {
+ background-color: #eaeaef;
+ }
+
+ .c8[aria-disabled='true'] svg path {
+ fill: #666687;
+ }
+
+ .c15 {
+ text-transform: uppercase;
+ }
+
+ .c22 canvas {
+ display: block;
+ max-width: 100%;
+ max-height: 100%;
+ }
+
+ .c26 {
+ text-transform: uppercase;
+ }
+
+ .c28 {
+ text-transform: uppercase;
+ }
+
+ .c27 svg {
+ font-size: 3rem;
+ }
+
+
+
+
+
+
+
+
+
+
+
+

+
+
+
+
+
+
+
+ strapi-cover_1fabc982ce.png
+
+
+
+ png
+
+ -
+ 551
+ ✕
+ 1066
+
+
+
+
+ Image
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ mov_bbb.mp4
+
+
+
+ mp4
+
+
+
+
+
+ Video
+
+
+
+
+
+
+
+
+
+
+
+ CARTE MARIAGE AVS - Printemps.pdf
+
+
+
+ pdf
+
+
+
+
+
+ Doc
+
+
+
+
+
+
+
+
+
+ `);
+ });
+});
diff --git a/packages/core/upload/admin/src/translations/en.json b/packages/core/upload/admin/src/translations/en.json
index 6235a99665..63352e9ef0 100644
--- a/packages/core/upload/admin/src/translations/en.json
+++ b/packages/core/upload/admin/src/translations/en.json
@@ -75,8 +75,9 @@
"settings.form.videoPreview.description": "It will generate a six-second preview of the video (GIF)",
"settings.form.videoPreview.label": "Preview",
"settings.header.label": "Media Library - Settings",
- "settings.section.image.label": "IMAGE",
- "settings.section.video.label": "VIDEO",
+ "settings.section.image.label": "Image",
+ "settings.section.video.label": "Video",
+ "settings.section.doc.label": "Doc",
"settings.sub-header.label": "Configure the settings for the media library",
"sort.created_at_asc": "Oldest uploads",
"sort.created_at_desc": "Most recent uploads",
diff --git a/packages/core/upload/admin/src/utils/formatDuration.js b/packages/core/upload/admin/src/utils/formatDuration.js
new file mode 100644
index 0000000000..98a84b2bb7
--- /dev/null
+++ b/packages/core/upload/admin/src/utils/formatDuration.js
@@ -0,0 +1,12 @@
+export const formatDuration = durationInSecond => {
+ const formatter = new Intl.DateTimeFormat('default', {
+ hour: 'numeric',
+ minute: 'numeric',
+ second: 'numeric',
+ });
+
+ const date = new Date(1970, 0, 1);
+ date.setSeconds(durationInSecond);
+
+ return formatter.format(date);
+};
diff --git a/packages/core/upload/admin/src/utils/index.js b/packages/core/upload/admin/src/utils/index.js
index f1f63f28bd..381fdc35da 100644
--- a/packages/core/upload/admin/src/utils/index.js
+++ b/packages/core/upload/admin/src/utils/index.js
@@ -18,3 +18,4 @@ export { default as getYupError } from './getYupError';
export { default as ItemTypes } from './ItemTypes';
export { default as unformatBytes } from './unformatBytes';
export { default as urlSchema } from './urlYupSchema';
+export * from './formatDuration';