From 08526e90f01badc5212c342fc4548d72dba3d37d Mon Sep 17 00:00:00 2001 From: MiroslavDionisiev Date: Tue, 3 Dec 2024 12:51:08 +0200 Subject: [PATCH 1/5] feat: [UEPR-116] add youtube video playlists and open videos in modal --- .../youtube-video-button.jsx | 45 +++++++ .../youtube-video-button.scss | 70 ++++++++++ .../youtube-video-modal.jsx | 47 +++++++ .../youtube-video-modal.scss | 58 ++++++++ src/views/ideas/ideas.jsx | 127 +++++++++++++++++- src/views/ideas/ideas.scss | 96 ++++++++++++- src/views/ideas/l10n.json | 5 + static/images/ideas/video-cc-label.svg | 4 + static/images/ideas/youtube-icon.svg | 12 ++ webpack.config.js | 1 + 10 files changed, 452 insertions(+), 13 deletions(-) create mode 100644 src/components/youtube-video-button/youtube-video-button.jsx create mode 100644 src/components/youtube-video-button/youtube-video-button.scss create mode 100644 src/components/youtube-video-modal/youtube-video-modal.jsx create mode 100644 src/components/youtube-video-modal/youtube-video-modal.scss create mode 100644 static/images/ideas/video-cc-label.svg create mode 100644 static/images/ideas/youtube-icon.svg diff --git a/src/components/youtube-video-button/youtube-video-button.jsx b/src/components/youtube-video-button/youtube-video-button.jsx new file mode 100644 index 000000000..5868b77b6 --- /dev/null +++ b/src/components/youtube-video-button/youtube-video-button.jsx @@ -0,0 +1,45 @@ +import React from 'react'; +import PropTypes from 'prop-types'; + +import './youtube-video-button.scss'; + +const parseViewCount = viewCount => + parseInt(viewCount, 10).toLocaleString('en-US', { + notation: 'compact', + compactDisplay: 'short' + }); + +export const YoutubeVideoButton = ({onSelectedVideo, ...videoData}) => ( +
onSelectedVideo(videoData.videoId)} + > +
+ +
{videoData.length}
+
+
+
{videoData.title}
+
+
{videoData.channel}
+
+ {`${parseViewCount(videoData.views)} ยท ${videoData.uploadTime}`} +
+
+ {videoData.hasCC && } +
+
+); + +YoutubeVideoButton.propTypes = { + videoId: PropTypes.string, + title: PropTypes.string, + thumbnail: PropTypes.string, + channel: PropTypes.string, + uploadTime: PropTypes.string, + length: PropTypes.string, + views: PropTypes.string, + hasCC: PropTypes.bool, + onSelectedVideo: PropTypes.func +}; diff --git a/src/components/youtube-video-button/youtube-video-button.scss b/src/components/youtube-video-button/youtube-video-button.scss new file mode 100644 index 000000000..9ff0a6502 --- /dev/null +++ b/src/components/youtube-video-button/youtube-video-button.scss @@ -0,0 +1,70 @@ +@import "../../colors"; + +.youtbe-video-button { + display: flex; + flex-direction: column; + gap: 0.75rem; + + width: 13.125rem; + + .thumbnail { + position: relative; + width: 100%; + + img { + position: relative; + width: 100%; + border-radius: 12px; + } + + .duration { + position: absolute; + z-index: 1; + bottom : 0.5rem; + right: 0; + + margin: 0.5rem; + padding: 0.25rem; + border-radius: 4px; + background-color: $overlay-gray; + color: $type-white; + + font-size: 0.75rem; + font-weight: 500; + font-family: "Helvetica Neue", Helvetica, Arial, sans-serif; + } + } + + .video-info { + margin-right: 1.5rem; + text-align: start; + + .video-title { + overflow: hidden; + text-overflow: ellipsis; + display: -webkit-box; + -webkit-box-orient: vertical; + -webkit-line-clamp: 2; + line-height: 1.2rem; + max-height: 2.4rem; + + font-family: "Helvetica Neue", Helvetica, Arial, sans-serif; + font-weight: 600; + font-size: 0.875rem; + color: black; + } + + .video-metadata { + margin: 0.5rem 4rem 0.25rem 0; + + color: $ui-dark-gray; + font-size: 0.75rem; + font-weight: 500; + line-height: 1.125rem; + } + } + + &:hover { + cursor: pointer; + } +} \ No newline at end of file diff --git a/src/components/youtube-video-modal/youtube-video-modal.jsx b/src/components/youtube-video-modal/youtube-video-modal.jsx new file mode 100644 index 000000000..60d503628 --- /dev/null +++ b/src/components/youtube-video-modal/youtube-video-modal.jsx @@ -0,0 +1,47 @@ +import React from 'react'; +import ReactModal from 'react-modal'; +import PropTypes from 'prop-types'; +import Button from '../forms/button.jsx'; +const classNames = require('classnames'); + +import './youtube-video-modal.scss'; + +export const YoutubeVideoModal = ({videoId, onClose = () => {}, className}) => { + if (!videoId) return null; + return ( + +
+
+