diff --git a/client/less/toastr.less b/client/less/toastr.less index f4774004a2..de17335ed3 100644 --- a/client/less/toastr.less +++ b/client/less/toastr.less @@ -1,269 +1,10 @@ -// sourced from https://github.com/CodeSeven/toastr -// MIT license -// Mix-ins -.borderRadius(@radius) { - -moz-border-radius: @radius; - -webkit-border-radius: @radius; - border-radius: @radius; +.notification-bar { + z-index: 999999; + overflow: hidden; + // margin: 0 0 6px; + padding: 2rem; } -.boxShadow(@boxShadow) { - -moz-box-shadow: @boxShadow; - -webkit-box-shadow: @boxShadow; - box-shadow: @boxShadow; -} - -.opacity(@opacity) { - @opacityPercent: @opacity * 100; - opacity: @opacity; - -ms-filter: ~"progid:DXImageTransform.Microsoft.Alpha(Opacity=@{opacityPercent})"; - filter: ~"alpha(opacity=@{opacityPercent})"; -} - -.wordWrap(@wordWrap: break-word) { - -ms-word-wrap: @wordWrap; - word-wrap: @wordWrap; -} - -// Variables -@black: #000000; -@grey: #999999; -@light-grey: #CCCCCC; -@white: #FFFFFF; -@near-black: #030303; -@green: #51A351; -@red: #BD362F; -@blue: #2F96B4; -@orange: #F89406; -@default-container-opacity: .8; - -// Styles -.toast-title { - font-weight: bold; -} - -.toast-message { - .wordWrap(); - - a, - label { - color: @white; - } - - a:hover { - color: @light-grey; - text-decoration: none; - } -} - -.toast-close-button { - position: relative; - right: -0.3em; - top: -0.3em; - float: right; - font-size: 20px; - font-weight: bold; - color: @white; - -webkit-text-shadow: 0 1px 0 rgba(255,255,255,1); - text-shadow: 0 1px 0 rgba(255,255,255,1); - .opacity(0.8); - - &:hover, - &:focus { - color: @black; - text-decoration: none; - cursor: pointer; - .opacity(0.4); - } -} - -/*Additional properties for button version - iOS requires the button element instead of an anchor tag. - If you want the anchor version, it requires `href="#"`.*/ -button.toast-close-button { - padding: 0; - cursor: pointer; - background: transparent; - border: 0; - -webkit-appearance: none; -} - -//#endregion - -.toast-top-center { - top: 0; - right: 0; - width: 100%; -} - -.toast-bottom-center { - bottom: 0; - right: 0; - width: 100%; -} - -.toast-top-full-width { - top: 0; - right: 0; - width: 100%; -} - -.toast-bottom-full-width { - bottom: 0; - right: 0; - width: 100%; -} - -.toast-top-left { - top: 12px; - left: 12px; -} - -.toast-top-right { - top: 12px; - right: 12px; -} - -.toast-bottom-right { - right: 12px; - bottom: 12px; -} - -.toast-bottom-left { - bottom: 12px; - left: 12px; -} - -#toast-container { - position: fixed; - z-index: 999999; - // The container should not be clickable. - pointer-events: none; - * { - -moz-box-sizing: border-box; - -webkit-box-sizing: border-box; - box-sizing: border-box; - } - - > div { - position: relative; - // The toast itself should be clickable. - pointer-events: auto; - overflow: hidden; - margin: 0 0 6px; - padding: 15px 15px 15px 50px; - width: 300px; - .borderRadius(3px 3px 3px 3px); - background-position: 15px center; - background-repeat: no-repeat; - .boxShadow(0 0 12px @grey); - color: @white; - .opacity(@default-container-opacity); - } - - > :hover { - .boxShadow(0 0 12px @black); - .opacity(1); - cursor: pointer; - } - - > .toast-info { - background-image: url("") !important; - } - - > .toast-error { - background-image: url("") !important; - } - - > .toast-success { - background-image: url("") !important; - } - - > .toast-warning { - background-image: url("") !important; - } - - /*overrides*/ - &.toast-top-center > div, - &.toast-bottom-center > div { - width: 300px; - margin-left: auto; - margin-right: auto; - } - - &.toast-top-full-width > div, - &.toast-bottom-full-width > div { - width: 96%; - margin-left: auto; - margin-right: auto; - } -} - -.toast { - background-color: @near-black; -} - -.toast-success { - background-color: @green; -} - -.toast-error { - background-color: @red; -} - -.toast-info { - background-color: @blue; -} - -.toast-warning { - background-color: @orange; -} - -.toast-progress { - position: absolute; - left: 0; - bottom: 0; - height: 4px; - background-color: @black; - .opacity(0.4); -} - -/*Responsive Design*/ - -@media all and (max-width: 240px) { - #toast-container { - - > div { - padding: 8px 8px 8px 50px; - width: 11em; - } - - & .toast-close-button { - right: -0.2em; - top: -0.2em; - } - } -} - -@media all and (min-width: 241px) and (max-width: 480px) { - #toast-container { - > div { - padding: 8px 8px 8px 50px; - width: 18em; - } - - & .toast-close-button { - right: -0.2em; - top: -0.2em; - } - } -} - -@media all and (min-width: 481px) and (max-width: 768px) { - #toast-container { - > div { - padding: 15px 15px 15px 50px; - width: 25em; - } - } +.notification-bar-message { + padding-right: 2rem; } diff --git a/common/app/App.jsx b/common/app/App.jsx index 9e336e2dd8..f8939b5654 100644 --- a/common/app/App.jsx +++ b/common/app/App.jsx @@ -1,6 +1,5 @@ import React, { PropTypes } from 'react'; import { Button, Row } from 'react-bootstrap'; -import { ToastMessage, ToastContainer } from 'react-toastr'; import { connect } from 'react-redux'; import { createSelector } from 'reselect'; @@ -16,11 +15,9 @@ import { import { submitChallenge } from './routes/challenges/redux/actions'; import Nav from './components/Nav'; -import { randomCompliment } from './utils/get-words'; +import Toasts from './toasts/Toasts.jsx'; import { userSelector } from './redux/selectors'; -const toastMessageFactory = React.createFactory(ToastMessage.animation); - const mapStateToProps = createSelector( userSelector, state => state.app.shouldShowSignIn, @@ -34,7 +31,6 @@ const mapStateToProps = createSelector( toast, isMapDrawerOpen, isMapAlreadyLoaded, - showChallengeComplete ) => ({ username, points, @@ -43,7 +39,6 @@ const mapStateToProps = createSelector( shouldShowSignIn, isMapDrawerOpen, isMapAlreadyLoaded, - showChallengeComplete, isSignedIn: !!username }) ); @@ -72,7 +67,6 @@ export class FreeCodeCamp extends React.Component { toast: PropTypes.object, updateNavHeight: PropTypes.func, initWindowHeight: PropTypes.func, - showChallengeComplete: PropTypes.number, submitChallenge: PropTypes.func, isMapDrawerOpen: PropTypes.bool, isMapAlreadyLoaded: PropTypes.bool, @@ -83,38 +77,6 @@ export class FreeCodeCamp extends React.Component { params: PropTypes.object }; - componentWillReceiveProps({ - toast: nextToast = {}, - showChallengeComplete: nextCC = 0 - }) { - const { - toast = {}, - showChallengeComplete - } = this.props; - if (toast.id !== nextToast.id) { - this.refs.toaster[nextToast.type || 'success']( - nextToast.message, - nextToast.title, - { - closeButton: true, - timeOut: 10000 - } - ); - } - - if (nextCC !== showChallengeComplete) { - this.refs.toaster.success( - this.renderChallengeComplete(), - randomCompliment(), - { - closeButton: true, - timeOut: 0, - extendedTimeOut: 0 - } - ); - } - } - componentDidMount() { this.props.initWindowHeight(); if (!this.props.isSignedIn) { @@ -173,11 +135,7 @@ export class FreeCodeCamp extends React.Component { isOpen={ isMapDrawerOpen } toggleMapDrawer={ toggleMapDrawer } /> - + ); } diff --git a/common/app/create-reducer.js b/common/app/create-reducer.js index f7edc27f93..80964bc069 100644 --- a/common/app/create-reducer.js +++ b/common/app/create-reducer.js @@ -2,6 +2,7 @@ import { combineReducers } from 'redux'; import { reducer as formReducer } from 'redux-form'; import { reducer as app } from './redux'; +import { reducer as toasts } from './toasts/redux'; import entitiesReducer from './redux/entities-reducer'; import { reducer as challengesApp, @@ -13,6 +14,7 @@ export default function createReducer(sideReducers = {}) { ...sideReducers, entities: entitiesReducer, app, + toasts, challengesApp, form: formReducer.normalize({ ...projectNormalizer }) }); diff --git a/common/app/redux/reducer.js b/common/app/redux/reducer.js index c26f11cfa5..48b1a95226 100644 --- a/common/app/redux/reducer.js +++ b/common/app/redux/reducer.js @@ -19,11 +19,6 @@ export default handleActions( title: payload + ' | Free Code Camp' }), - [types.makeToast]: (state, { payload: toast }) => ({ - ...state, - toast - }), - [types.updateThisUser]: (state, { payload: user }) => ({ ...state, user, diff --git a/common/app/redux/types.js b/common/app/redux/types.js index 46a8307270..ab0864c0c7 100644 --- a/common/app/redux/types.js +++ b/common/app/redux/types.js @@ -10,7 +10,6 @@ export default createTypes([ 'updateCompletedChallenges', 'showSignIn', - 'makeToast', 'handleError', 'toggleNightMode', // used to hit the server diff --git a/common/app/routes/challenges/components/classic/Side-Panel.jsx b/common/app/routes/challenges/components/classic/Side-Panel.jsx index a81497437b..a6b0f4f5fc 100644 --- a/common/app/routes/challenges/components/classic/Side-Panel.jsx +++ b/common/app/routes/challenges/components/classic/Side-Panel.jsx @@ -11,7 +11,7 @@ import Output from './Output.jsx'; import ToolPanel from './Tool-Panel.jsx'; import { challengeSelector } from '../../redux/selectors'; import { updateHint, executeChallenge } from '../../redux/actions'; -import { makeToast } from '../../../../redux/actions'; +import { makeToast } from '../../../../toasts/redux/actions'; const bindableActions = { makeToast, executeChallenge, updateHint }; const mapStateToProps = createSelector( diff --git a/common/app/routes/challenges/redux/actions.js b/common/app/routes/challenges/redux/actions.js index 8890e2251f..d4f6a158ef 100644 --- a/common/app/routes/challenges/redux/actions.js +++ b/common/app/routes/challenges/redux/actions.js @@ -93,14 +93,6 @@ export const updateOutput = createAction(types.updateOutput, loggerToStr); export const checkChallenge = createAction(types.checkChallenge); export const showProjectSubmit = createAction(types.showProjectSubmit); -let id = 0; -export const showChallengeComplete = createAction( - types.showChallengeComplete, - () => { - id += 1; - return id; - } -); export const submitChallenge = createAction(types.submitChallenge); export const moveToNextChallenge = createAction(types.moveToNextChallenge); diff --git a/common/app/routes/challenges/redux/answer-saga.js b/common/app/routes/challenges/redux/answer-saga.js index b9ccedf028..def66ecb90 100644 --- a/common/app/routes/challenges/redux/answer-saga.js +++ b/common/app/routes/challenges/redux/answer-saga.js @@ -3,7 +3,8 @@ import types from './types'; import { getMouse } from '../utils'; import { submitChallenge, videoCompleted } from './actions'; -import { createErrorObservable, makeToast } from '../../../redux/actions'; +import { createErrorObservable } from '../../../redux/actions'; +import { makeToast } from '../../../toasts/redux/actions'; import { challengeSelector } from './selectors'; export default function answerSaga(action$, getState) { @@ -54,11 +55,7 @@ export default function answerSaga(action$, getState) { if (answer !== finalAnswer) { let infoAction; if (info) { - infoAction = makeToast({ - title: 'Have a hint', - message: info, - type: 'info' - }); + infoAction = makeToast({ message: info }); } return Observable diff --git a/common/app/routes/challenges/redux/completion-saga.js b/common/app/routes/challenges/redux/completion-saga.js index f347d0ea1a..e90368fdcc 100644 --- a/common/app/routes/challenges/redux/completion-saga.js +++ b/common/app/routes/challenges/redux/completion-saga.js @@ -1,11 +1,12 @@ import { Observable } from 'rx'; + import types from './types'; -import { showChallengeComplete, moveToNextChallenge } from './actions'; +import { moveToNextChallenge } from './actions'; import { createErrorObservable, - makeToast, updateUserPoints } from '../../../redux/actions'; +import { makeToast } from '../../../toasts/redux/actions'; import { challengeSelector } from './selectors'; import { backEndProject } from '../../../utils/challengeTypes'; @@ -18,9 +19,8 @@ function postChallenge(url, body, username) { .flatMap(({ alreadyCompleted, points }) => { return Observable.of( makeToast({ - message: randomCompliment() + - (alreadyCompleted ? '!' : '! First time Completed!'), - type: 'info' + message: randomCompliment() + + (alreadyCompleted ? '!' : '! First time Completed!') }), updateUserPoints(username, points) ); @@ -29,7 +29,7 @@ function postChallenge(url, body, username) { const challengeCompleted$ = Observable.of( moveToNextChallenge(), - username ? makeToast({ message: ' Saving...', type: 'info' }) : null + username ? makeToast({ message: ' Saving...' }) : null ); return Observable.merge(saveChallenge$, challengeCompleted$); } @@ -39,7 +39,11 @@ function submitModern(type, state) { if (tests.length > 0 && tests.every(test => test.pass && !test.err)) { if (type === types.checkChallenge) { return Observable.of( - showChallengeComplete() + makeToast({ + message: 'Go to my next challenge.', + action: 'Submit', + actionCreator: 'submitChallenge' + }) ); } @@ -58,9 +62,7 @@ function submitModern(type, state) { } } return Observable.just(makeToast({ - message: 'Not all tests are passing, yet.', - title: 'Almost There!', - type: 'info' + message: 'Not all tests are passing, yet.' })); } diff --git a/common/app/routes/challenges/redux/next-challenge-saga.js b/common/app/routes/challenges/redux/next-challenge-saga.js index 55f04e4b57..5352149ae3 100644 --- a/common/app/routes/challenges/redux/next-challenge-saga.js +++ b/common/app/routes/challenges/redux/next-challenge-saga.js @@ -2,13 +2,13 @@ import { Observable } from 'rx'; import { push } from 'react-router-redux'; import types from './types'; import { resetUi, updateCurrentChallenge } from './actions'; -import { createErrorObservable, makeToast } from '../../../redux/actions'; +import { createErrorObservable } from '../../../redux/actions'; +import { makeToast } from '../../../toasts/redux/actions'; import { getNextChallenge, getFirstChallengeOfNextBlock, getFirstChallengeOfNextSuperBlock } from '../utils'; -import { randomVerb } from '../../../utils/get-words'; import debug from 'debug'; const isDev = debug.enabled('fcc:*'); @@ -64,10 +64,7 @@ export default function nextChallengeSaga(actions$, getState) { return Observable.of( updateCurrentChallenge(nextChallenge), resetUi(), - makeToast({ - title: randomVerb(), - message: 'Your next challenge has arrived.' - }), + makeToast({ message: 'Your next challenge arrived.' }), push(`/challenges/${nextChallenge.block}/${nextChallenge.dashedName}`) ); } catch (err) { diff --git a/common/app/toasts/Toasts.jsx b/common/app/toasts/Toasts.jsx new file mode 100644 index 0000000000..de3bb90a32 --- /dev/null +++ b/common/app/toasts/Toasts.jsx @@ -0,0 +1,69 @@ +import React, { PropTypes } from 'react'; +import { connect } from 'react-redux'; +import { createSelector } from 'reselect'; +import { NotificationStack } from 'react-notification'; + +import { removeToast } from './redux/actions'; +import { submitChallenge } from '../routes/challenges/redux/actions'; + +const registeredActions = { submitChallenge }; +const mapStateToProps = state => ({ toasts: state.toasts }); +const barStyle = { + fontSize: '2rem', + // null values let our css set the style prop + padding: null +}; +const actionStyle = { + fontSize: '2rem' +}; +const addDispatchableActionsToToast = createSelector( + state => state.toasts, + state => state.dispatch, + (toasts, dispatch) => toasts.map(({ position, actionCreator, ...toast }) => { + const activeBarStyle = {}; + if (position !== 'left') { + activeBarStyle.left = null; + activeBarStyle.right = '1rem'; + } + const onClick = registeredActions[actionCreator] ? + () => dispatch(registeredActions[actionCreator]()) : + null; + return { + ...toast, + barStyle, + activeBarStyle, + actionStyle, + onClick + }; + }) +); + +export class Toasts extends React.Component { + constructor(...props) { + super(...props); + this.handleDismiss = this.handleDismiss.bind(this); + } + static displayName = 'Toasts'; + static propTypes = { + toasts: PropTypes.arrayOf(PropTypes.object), + dispatch: PropTypes.func + }; + + handleDismiss(notification) { + this.props.dispatch(removeToast(notification)); + } + + render() { + const { toasts = [], dispatch } = this.props; + return ( + + ); + } +} + +export default connect(mapStateToProps)(Toasts); diff --git a/common/app/toasts/redux/actions.js b/common/app/toasts/redux/actions.js new file mode 100644 index 0000000000..0241ba1fd1 --- /dev/null +++ b/common/app/toasts/redux/actions.js @@ -0,0 +1,20 @@ +import { createAction } from 'redux-actions'; +import types from './types'; + +let key = 0; +export const makeToast = createAction( + types.makeToast, + ({ timeout, ...rest }) => ({ + ...rest, + // assign current value of key to new toast + // and then increment key value + key: key++, + dismissAfter: timeout || 40000, + position: rest.position === 'left' ? 'left' : 'right' + }) +); + +export const removeToast = createAction( + types.removeToast, + ({ key }) => key +); diff --git a/common/app/toasts/redux/index.js b/common/app/toasts/redux/index.js new file mode 100644 index 0000000000..3a828d6dcd --- /dev/null +++ b/common/app/toasts/redux/index.js @@ -0,0 +1,3 @@ +export { default as types } from './types'; +export { default as reducer } from './reducer'; +export * as actions from './actions'; diff --git a/common/app/toasts/redux/reducer.js b/common/app/toasts/redux/reducer.js new file mode 100644 index 0000000000..ac1312041c --- /dev/null +++ b/common/app/toasts/redux/reducer.js @@ -0,0 +1,13 @@ +import { handleActions } from 'redux-actions'; +import types from './types'; + +const initialState = []; +export default handleActions({ + [types.makeToast]: (state, { payload: toast }) => [ + ...state, + toast + ], + [types.removeToast]: (state, { payload: key }) => state.filter( + toast => toast.key !== key + ) +}, initialState); diff --git a/common/app/toasts/redux/types.js b/common/app/toasts/redux/types.js new file mode 100644 index 0000000000..5472d76070 --- /dev/null +++ b/common/app/toasts/redux/types.js @@ -0,0 +1,6 @@ +import createTypes from '../../utils/create-types'; + +export default createTypes([ + 'makeToast', + 'removeToast' +], 'toast'); diff --git a/package.json b/package.json index f8a5394de4..2a161ef7b5 100644 --- a/package.json +++ b/package.json @@ -94,6 +94,7 @@ "react-images": "^0.4.6", "react-motion": "~0.4.2", "react-no-ssr": "^1.0.1", + "react-notification": "^5.0.7", "react-pure-render": "^1.0.2", "react-redux": "^4.0.6", "react-router": "^2.0.0",