@@ -85,15 +120,20 @@ class OnboardingVideo extends React.Component {
- {!video.duration && (
+
+ {!this.props.video.duration && (
{},
getVideoCurrentTime: () => {},
+ didStopVideo: () => {},
+ didPlayVideo: () => {},
};
OnboardingVideo.propTypes = {
@@ -121,6 +163,8 @@ OnboardingVideo.propTypes = {
videos: PropTypes.object,
setVideoDuration: PropTypes.func,
getVideoCurrentTime: PropTypes.func,
+ didStopVideo: PropTypes.func,
+ didPlayVideo: PropTypes.func,
};
export default OnboardingVideo;
diff --git a/packages/strapi-admin/admin/src/components/OnboardingVideo/styles.scss b/packages/strapi-admin/admin/src/components/OnboardingVideo/styles.scss
index de4583f82e..fbb43aba9c 100644
--- a/packages/strapi-admin/admin/src/components/OnboardingVideo/styles.scss
+++ b/packages/strapi-admin/admin/src/components/OnboardingVideo/styles.scss
@@ -2,20 +2,7 @@ li.listItem {
display: block;
padding: 10px 15px;
cursor: pointer;
- &.finished {
- .title {
- color: #919bae;
- }
- .thumbWrapper {
- .play {
- background-color: #5a9e06;
- &::before {
- content: '\f00c';
- font-size: 11px;
- }
- }
- }
- }
+
&:hover {
background-color: #f7f8f8;
.title {
@@ -34,9 +21,20 @@ li.listItem {
border-radius: 2px;
overflow: hidden;
position: relative;
+ .overlay {
+ width: 100%;
+ height: 100%;
+ background-color: rgba(#0E7DE7, .8);
+ position: absolute;
+ top: 0;
+ left: 0;
+ z-index: 1;
+ }
img {
width: 100%;
height: 100%;
+ position: relative;
+ z-index: 0;
}
.play {
width: 20px;
@@ -44,22 +42,47 @@ li.listItem {
line-height: 20px;
border-radius: 50%;
background-color: #0e7de7;
+ border: 1px solid white;
position: absolute;
top: 50%;
margin-top: -10px;
left: 50%;
margin-left: -10px;
text-align: center;
+ z-index: 2;
&::before {
content: '\f04b';
font-family: 'FontAwesome';
font-size: 9px;
color: white;
display: inline-block;
- vertical-align: middle;
+ vertical-align: top;
+ height: 100%;
}
}
}
+ &.finished {
+ .title {
+ color: #919bae;
+ }
+ .thumbWrapper {
+ .overlay {
+ background-color: transparent;
+ }
+ img {
+ opacity: 0.6;
+ }
+ .play {
+ background-color: #5a9e06;
+ border-color: #5a9e06;
+ &::before {
+ content: '\f00c';
+ font-size: 11px;
+ }
+ }
+ }
+ }
+
.txtWrapper {
padding: 0 15px;
p {
@@ -73,9 +96,6 @@ li.listItem {
}
}
}
-.modal .modal-dialog {
- background: red;
-}
.hiddenPlayerWrapper {
display: none;
@@ -97,4 +117,12 @@ li.listItem {
margin-right: 0;
}
}
-}
\ No newline at end of file
+ .videoPlayer {
+ > button {
+ top: 50%;
+ margin-top: -0.75em;
+ left: 50%;
+ margin-left: -1.5em;
+ }
+ }
+}
diff --git a/packages/strapi-admin/admin/src/components/OnboardingVideos/index.js b/packages/strapi-admin/admin/src/components/OnboardingVideos/index.js
deleted file mode 100644
index 59967e5e55..0000000000
--- a/packages/strapi-admin/admin/src/components/OnboardingVideos/index.js
+++ /dev/null
@@ -1,145 +0,0 @@
-/*
- *
- * Row
- *
- */
-
-import React from 'react';
-import PropTypes from 'prop-types';
-import cn from 'classnames';
-import PopUpVideo from 'components/PopUpVideo';
-
-import styles from './styles.scss';
-const videos = [
- {
- title: 'Create your first content-type',
- id: 0,
- url:
- 'https://s3.amazonaws.com/codecademy-content/courses/React/react_video-cute.mp4',
- thumb:
- 'https://www.santevet.com/upload/admin/images/article/Chat%202/chaton_seul/santevet_chats_en_location.jpg',
- time: '1:32min',
- end: true,
- },
- {
- title: 'Fill your content with data',
- id: 1,
- url:
- 'https://s3.amazonaws.com/codecademy-content/courses/React/react_video-eek.mp4',
- thumb:
- '',
- time: '1:32min',
- end: false,
- },
- {
- title: 'Manage the API access',
- id: 2,
- url:
- 'https://s3.amazonaws.com/codecademy-content/courses/React/react_video-slow.mp4',
- thumb:
- 'https://www.santevet.com/upload/admin/images/article/Chat%202/chaton_seul/santevet_chats_en_location.jpg',
- time: '1:32min',
- end: false,
- },
- {
- title: 'Fetch data through the API',
- id: 3,
- url:
- 'https://s3.amazonaws.com/codecademy-content/courses/React/react_video-fast.mp4',
- thumb:
- 'https://www.santevet.com/upload/admin/images/article/Chat%202/chaton_seul/santevet_chats_en_location.jpg',
- time: '1:15min',
- end: false,
- },
-];
-
-class OnboardingVideos extends React.Component {
- state = { showVideos: false, showModal: false, video: '' };
-
- // componentWillReceiveProps(nextProps) {
- // //if (nextProps.pluginActionSucceeded !== this.props.pluginActionSucceeded) {
- // this.setState({ showModal: false });
- // //}
- // }
- componentDidMount() {
- this.setState({ showVideos: true });
- }
- toggleVideos = e => {
- this.setState({ showVideos: !this.state.showVideos });
- };
-
- displayModal = e => {
- e.preventDefault();
- let video = videos[e.currentTarget.parentNode.key];
- console.log(e.currentTarget.parentNode);
- console.log(video);
- this.setState({ showModal: !this.state.showModal });
- };
-
- render() {
- return (
-
-
-
-
Get started video
-
25% completed
-
-
-
-
-
-
-
-
-
- this.setState({ showModal: !this.state.showModal })
- }
- popUpWarningType="danger"
- onConfirm={this.props.onDeleteConfirm}
- src="https://s3.amazonaws.com/codecademy-content/courses/React/react_video-cute.mp4"
- />
-
- );
- }
-}
-
-OnboardingVideos.contextTypes = {};
-
-OnboardingVideos.propTypes = {};
-
-export default OnboardingVideos;
diff --git a/packages/strapi-admin/admin/src/components/OnboardingVideos/styles.scss b/packages/strapi-admin/admin/src/components/OnboardingVideos/styles.scss
deleted file mode 100644
index 069e7baaf5..0000000000
--- a/packages/strapi-admin/admin/src/components/OnboardingVideos/styles.scss
+++ /dev/null
@@ -1,152 +0,0 @@
-.videosWrapper {
- position: fixed;
- right: 15px;
- bottom: 15px;
- button,
- button:focus,
- a {
- cursor: pointer;
- outline: 0;
- }
- p {
- margin-bottom: 0;
- }
- .videosHeader {
- padding: 15px 15px 0 15px;
- p {
- font-family: Lato-Bold;
- font-size: 11px;
- text-transform: uppercase;
- display: inline-block;
- width: 50%;
- vertical-align: top;
- color: #5c5f66;
- &:last-of-type {
- text-align: right;
- color: #5a9e06;
- }
- }
- }
- .videosContent {
- background-color: white;
- margin-bottom: 10px;
- margin-right: 15px;
- box-shadow: 0 2px 4px 0 #e3e9f3;
- border-radius: 3px;
- transition: all 0.3s linear;
- &.shown {
- visibility: visible;
- opacity: 1;
- }
- &.hide {
- opacity: 0;
- visibility: hidden;
- }
- ul {
- list-style: none;
- padding: 10px 0;
- li {
- &.finished {
- .title {
- color: #919bae;
- }
- .thumbWrapper {
- .play {
- background-color: #5a9e06;
- &::before {
- content: '\f00c';
- font-size: 11px;
- }
- }
- }
- }
- a {
- display: block;
- padding: 10px 15px;
- &:hover {
- background-color: #f7f8f8;
- .title {
- color: #0e7de7;
- }
- }
- }
- .txtWrapper,
- .thumbWrapper {
- display: inline-block;
- vertical-align: middle;
- }
- .thumbWrapper {
- width: 55px;
- height: 38px;
- background-color: #d8d8d8;
- border-radius: 2px;
- overflow: hidden;
- position: relative;
- img {
- width: 100%;
- height: 100%;
- }
- .play {
- width: 20px;
- height: 20px;
- line-height: 20px;
- border-radius: 50%;
- background-color: #0e7de7;
- position: absolute;
- top: 50%;
- margin-top: -10px;
- left: 50%;
- margin-left: -10px;
- text-align: center;
- &::before {
- content: '\f04b';
- font-family: 'FontAwesome';
- font-size: 9px;
- color: white;
- display: inline-block;
- vertical-align: middle;
- }
- }
- }
- .txtWrapper {
- padding: 0 15px;
- p {
- font-size: 14px;
- font-family: Lato-SemiBold;
- }
- .time {
- font-size: 11px;
- color: #919bae;
- font-family: Lato-Bold;
- }
- }
- }
- }
- }
- .openBtn {
- width: 38px;
- height: 38px;
- float: right;
- button {
- width: 100%;
- height: 100%;
- border-radius: 50%;
- color: white;
- background: #0e7de7;
- -webkit-box-shadow: 0px 2px 4px 0px rgba(227, 233, 243, 1);
- -moz-box-shadow: 0px 2px 4px 0px rgba(227, 233, 243, 1);
- box-shadow: 0px 2px 4px 0px rgba(227, 233, 243, 1);
- i:last-of-type {
- display: none;
- }
- &.active {
- i:first-of-type {
- display: none;
- }
- i:last-of-type {
- display: block;
- }
- }
- }
- }
-}
diff --git a/packages/strapi-admin/admin/src/containers/AdminPage/actions.js b/packages/strapi-admin/admin/src/containers/AdminPage/actions.js
index 33cc36539b..5a6746276b 100644
--- a/packages/strapi-admin/admin/src/containers/AdminPage/actions.js
+++ b/packages/strapi-admin/admin/src/containers/AdminPage/actions.js
@@ -26,6 +26,6 @@ export function emitEvent(event, properties) {
return {
type: EMIT_EVENT,
event,
- properties,
+ properties
};
}
\ No newline at end of file
diff --git a/packages/strapi-admin/admin/src/containers/AdminPage/index.js b/packages/strapi-admin/admin/src/containers/AdminPage/index.js
index 0fe90cb91e..bfd5fa760c 100644
--- a/packages/strapi-admin/admin/src/containers/AdminPage/index.js
+++ b/packages/strapi-admin/admin/src/containers/AdminPage/index.js
@@ -284,7 +284,7 @@ export class AdminPage extends React.Component {
isOpen={this.props.blockApp && this.props.showGlobalAppBlocker}
{...this.props.overlayBlockerData}
/>
-
+ {this.shouldDisplayLogout() && }
);
}
diff --git a/packages/strapi-admin/admin/src/containers/Onboarding/actions.js b/packages/strapi-admin/admin/src/containers/Onboarding/actions.js
index 869700caff..643779f43d 100644
--- a/packages/strapi-admin/admin/src/containers/Onboarding/actions.js
+++ b/packages/strapi-admin/admin/src/containers/Onboarding/actions.js
@@ -4,7 +4,7 @@
*
*/
-import { GET_VIDEOS, GET_VIDEOS_SUCCEEDED, ON_CLICK, SET_VIDEOS_DURATION, UPDATE_VIDEO_START_TIME } from './constants';
+import { GET_VIDEOS, GET_VIDEOS_SUCCEEDED, ON_CLICK, SET_VIDEOS_DURATION, UPDATE_VIDEO_START_TIME, SET_VIDEO_END, REMOVE_VIDEOS } from './constants';
export function getVideos() {
return {
@@ -41,3 +41,17 @@ export function updateVideoStartTime(index, startTime) {
startTime: parseFloat(startTime, 10),
};
}
+
+export function setVideoEnd(index, end) {
+ return {
+ type: SET_VIDEO_END,
+ index: parseInt(index, 10),
+ end,
+ };
+}
+
+export function removeVideos() {
+ return {
+ type: REMOVE_VIDEOS,
+ };
+}
diff --git a/packages/strapi-admin/admin/src/containers/Onboarding/constants.js b/packages/strapi-admin/admin/src/containers/Onboarding/constants.js
index 7e3a0fe6c5..fdab1e69fb 100644
--- a/packages/strapi-admin/admin/src/containers/Onboarding/constants.js
+++ b/packages/strapi-admin/admin/src/containers/Onboarding/constants.js
@@ -10,3 +10,5 @@ export const GET_VIDEOS_SUCCEEDED =
export const ON_CLICK = 'StrapiAdmin/Onboarding/ON_CLICK';
export const SET_VIDEOS_DURATION = 'StrapiAdmin/Onboarding/SET_VIDEOS_DURATION';
export const UPDATE_VIDEO_START_TIME = 'StrapiAdmin/Onboarding/UPDATE_VIDEO_START_TIME';
+ export const SET_VIDEO_END = 'StrapiAdmin/Onboarding/SET_VIDEO_END';
+ export const REMOVE_VIDEOS = 'StrapiAdmin/Onboarding/REMOVE_VIDEOS';
diff --git a/packages/strapi-admin/admin/src/containers/Onboarding/index.js b/packages/strapi-admin/admin/src/containers/Onboarding/index.js
index 438edffcdb..4897455843 100644
--- a/packages/strapi-admin/admin/src/containers/Onboarding/index.js
+++ b/packages/strapi-admin/admin/src/containers/Onboarding/index.js
@@ -9,7 +9,7 @@ import PropTypes from 'prop-types';
import cn from 'classnames';
import { connect } from 'react-redux';
import { bindActionCreators, compose } from 'redux';
-import { getVideos, onClick, setVideoDuration, updateVideoStartTime } from './actions';
+import { getVideos, onClick, setVideoDuration, updateVideoStartTime, setVideoEnd, removeVideos } from './actions';
import injectSaga from 'utils/injectSaga';
import injectReducer from 'utils/injectReducer';
@@ -24,11 +24,16 @@ import styles from './styles.scss';
export class Onboarding extends React.Component {
state = { showVideos: false };
- componentWillMount() {
+ componentDidMount() {
this.setState({ showVideos: true });
this.props.getVideos();
}
+ componentWillUnmount() {
+ this.props.removeVideos();
+ localStorage.removeItem('videos');
+ }
+
toggleVideos = () => {
// Display videos card
this.setState(prevState => ({ showVideos: !prevState.showVideos }));
@@ -39,14 +44,38 @@ export class Onboarding extends React.Component {
this.context.emitEvent(eventName);
};
- updateLocalStorage = (index, current) => {
+ setVideoEnd = () => {
+ this.setVideoEnd();
+ }
+
+ didPlayVideo = (index, currTime) => {
+
+ const eventName = `didPlay${index}GetStartedVideo`;
+ this.context.emitEvent(eventName, {timestamp: currTime});
+ }
+
+ didStopVideo = (index, currTime) => {
+
+ const eventName = `didStop${index}Video`;
+ this.context.emitEvent(eventName, {timestamp: currTime});
+ }
+
+ updateLocalStorage = (index, current, duration) => {
// Update store
this.props.updateVideoStartTime(index, current);
// Update localStorage
let videosTime = JSON.parse(localStorage.getItem('videos'));
- videosTime.fill(0);
- videosTime[index] = current;
+ videosTime[index].startTime = current;
+
+ let percent = current * 100 / duration;
+ if (percent >= 80) {
+ if (videosTime[index].end === false) {
+ videosTime[index].end = true;
+ this.props.setVideoEnd(index, true);
+ }
+ }
+
localStorage.setItem('videos', JSON.stringify(videosTime));
};
@@ -77,6 +106,8 @@ export class Onboarding extends React.Component {
onClick={onClick}
setVideoDuration={setVideoDuration}
getVideoCurrentTime={this.updateLocalStorage}
+ didPlayVideo={this.didPlayVideo}
+ didStopVideo={this.didStopVideo}
/>
);
})}
@@ -108,7 +139,7 @@ Onboarding.propTypes = {
const mapStateToProps = makeSelectOnboarding();
function mapDispatchToProps(dispatch) {
- return bindActionCreators({ getVideos, onClick, setVideoDuration, updateVideoStartTime }, dispatch);
+ return bindActionCreators({ getVideos, onClick, setVideoDuration, updateVideoStartTime, setVideoEnd, removeVideos }, dispatch);
}
const withConnect = connect(
diff --git a/packages/strapi-admin/admin/src/containers/Onboarding/reducer.js b/packages/strapi-admin/admin/src/containers/Onboarding/reducer.js
index 92b9509aab..8c8a235fb0 100644
--- a/packages/strapi-admin/admin/src/containers/Onboarding/reducer.js
+++ b/packages/strapi-admin/admin/src/containers/Onboarding/reducer.js
@@ -5,7 +5,7 @@
*/
import { fromJS } from 'immutable';
-import { GET_VIDEOS_SUCCEEDED, ON_CLICK, SET_VIDEOS_DURATION, UPDATE_VIDEO_START_TIME } from './constants';
+import { GET_VIDEOS_SUCCEEDED, ON_CLICK, SET_VIDEOS_DURATION, UPDATE_VIDEO_START_TIME, SET_VIDEO_END, REMOVE_VIDEOS } from './constants';
const initialState = fromJS({
videos: fromJS([]),
@@ -39,6 +39,10 @@ function onboardingReducer(state = initialState, action) {
return acc.updateIn([index, 'startTime'], () => 0);
}, list);
});
+ case SET_VIDEO_END:
+ return state.updateIn(['videos', action.index, 'end'], () => action.end);
+ case REMOVE_VIDEOS:
+ return initialState;
default:
return state;
}
diff --git a/packages/strapi-admin/admin/src/containers/Onboarding/saga.js b/packages/strapi-admin/admin/src/containers/Onboarding/saga.js
index e1eb49a052..9323addd24 100644
--- a/packages/strapi-admin/admin/src/containers/Onboarding/saga.js
+++ b/packages/strapi-admin/admin/src/containers/Onboarding/saga.js
@@ -17,8 +17,15 @@ function* getVideos() {
{ noAuth: true },
);
- let currTimes = Array.apply(null, Array(videos.length)).map(Number.prototype.valueOf,0);
-
+ let currTimes = Array.apply(null, Array(videos.length)).map((e, i) => {
+ return {
+ startTime: 0,
+ end: false,
+ key: i,
+ id: videos[i].id,
+ }
+ });
+
// Retrieve start time if enable in localStorage
if (localStorage.getItem('videos')) {
currTimes.splice(0, currTimes.length, ...JSON.parse(localStorage.getItem('videos')));
@@ -31,7 +38,8 @@ function* getVideos() {
videos.map((video, index) => {
video.isOpen = false;
video.duration = null;
- video.startTime = currTimes[index];
+ video.startTime = currTimes[index].startTime;
+ video.end = currTimes[index].end;
return video;
}),