Add toasts to react app

This commit is contained in:
Berkeley Martinez
2016-01-07 14:51:41 -08:00
parent d6cce6e7ca
commit c6bd4695af
11 changed files with 373 additions and 17 deletions

View File

@ -227,7 +227,7 @@
"react/jsx-boolean-value": [1, "always"], "react/jsx-boolean-value": [1, "always"],
"jsx-quotes": [1, "prefer-single"], "jsx-quotes": [1, "prefer-single"],
"react/jsx-no-undef": 1, "react/jsx-no-undef": 1,
"react/jsx-sort-props": 1, "react/jsx-sort-props": [1, { "ignoreCase": true }],
"react/jsx-uses-react": 1, "react/jsx-uses-react": 1,
"react/jsx-uses-vars": 1, "react/jsx-uses-vars": 1,
"react/no-did-mount-set-state": 2, "react/no-did-mount-set-state": 2,

9
client/err-saga.js Normal file
View File

@ -0,0 +1,9 @@
export default function toastSaga(err$, toast) {
err$
.doOnNext(() => toast({
type: 'error',
title: 'Oops, something went wrong',
message: `Something went wrong, please try again later`
}))
.subscribe(err => console.error(err));
}

View File

@ -20,7 +20,7 @@ const emptyLocation = {
let prevKey; let prevKey;
let isSyncing = false; let isSyncing = false;
export default function synchroniseHistory( export default function historySaga(
history, history,
updateLocation, updateLocation,
goTo, goTo,

View File

@ -9,7 +9,8 @@ import { hydrate } from 'thundercats';
import { render$ } from 'thundercats-react'; import { render$ } from 'thundercats-react';
import { app$ } from '../common/app'; import { app$ } from '../common/app';
import synchroniseHistory from './synchronise-history'; import historySaga from './history-saga';
import errSaga from './err-saga';
const debug = debugFactory('fcc:client'); const debug = debugFactory('fcc:client');
const DOMContianer = document.getElementById('fcc'); const DOMContianer = document.getElementById('fcc');
@ -38,9 +39,16 @@ app$({ history, location: appLocation })
({ nextLocation, props }, appCat) => ({ nextLocation, props, appCat }) ({ nextLocation, props }, appCat) => ({ nextLocation, props, appCat })
) )
.doOnNext(({ appCat }) => { .doOnNext(({ appCat }) => {
const { updateLocation, goTo, goBack } = appCat.getActions('appActions');
const appStore$ = appCat.getStore('appStore'); const appStore$ = appCat.getStore('appStore');
const {
toast,
updateLocation,
goTo,
goBack
} = appCat.getActions('appActions');
const routerState$ = appStore$ const routerState$ = appStore$
.map(({ location }) => location) .map(({ location }) => location)
.distinctUntilChanged( .distinctUntilChanged(
@ -50,22 +58,24 @@ app$({ history, location: appLocation })
// set page title // set page title
appStore$ appStore$
.pluck('title') .pluck('title')
.distinctUntilChanged()
.doOnNext(title => document.title = title) .doOnNext(title => document.title = title)
.subscribe(() => {}); .subscribe(() => {});
appStore$ historySaga(
.pluck('err')
.filter(err => !!err)
.distinctUntilChanged()
.subscribe(err => console.error(err));
synchroniseHistory(
history, history,
updateLocation, updateLocation,
goTo, goTo,
goBack, goBack,
routerState$ routerState$
); );
const err$ = appStore$
.pluck('err')
.filter(err => !!err)
.distinctUntilChanged();
errSaga(err$, toast);
}) })
// allow store subscribe to subscribe to actions // allow store subscribe to subscribe to actions
.delay(10) .delay(10)

View File

@ -1138,3 +1138,4 @@ code {
@import "chat.less"; @import "chat.less";
@import "jobs.less"; @import "jobs.less";
@import "challenge.less"; @import "challenge.less";
@import "toastr.less";

269
client/less/toastr.less Normal file
View File

@ -0,0 +1,269 @@
// sourced from https://github.com/CodeSeven/toastr
// MIT license
// Mix-ins
.borderRadius(@radius) {
-moz-border-radius: @radius;
-webkit-border-radius: @radius;
border-radius: @radius;
}
.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;
}
}
}

View File

@ -1,16 +1,33 @@
import React, { PropTypes } from 'react'; import React, { PropTypes } from 'react';
import { contain } from 'thundercats-react';
import { Row } from 'react-bootstrap'; import { Row } from 'react-bootstrap';
import { ToastMessage, ToastContainer } from 'react-toastr';
import { contain } from 'thundercats-react';
import { Nav } from './components/Nav'; import { Nav } from './components/Nav';
const toastMessageFactory = React.createFactory(ToastMessage.animation);
export default contain( export default contain(
{ {
actions: ['appActions'],
store: 'appStore', store: 'appStore',
fetchAction: 'appActions.getUser', fetchAction: 'appActions.getUser',
isPrimed({ username }) { isPrimed({ username }) {
return !!username; return !!username;
}, },
map({
username,
points,
picture,
toast
}) {
return {
username,
points,
picture,
toast
};
},
getPayload(props) { getPayload(props) {
return { return {
isPrimed: !!props.username isPrimed: !!props.username
@ -21,11 +38,31 @@ export default contain(
displayName: 'FreeCodeCamp', displayName: 'FreeCodeCamp',
propTypes: { propTypes: {
appActions: PropTypes.object,
children: PropTypes.node, children: PropTypes.node,
username: PropTypes.string,
points: PropTypes.number, points: PropTypes.number,
picture: PropTypes.string, picture: PropTypes.string,
title: PropTypes.string, toast: PropTypes.object
username: PropTypes.string },
componentWillReceiveProps({ toast: nextToast }) {
const { toast = {} } = this.props;
if (
toast &&
nextToast &&
toast.id !== nextToast.id
) {
this.refs.toaster[nextToast.type || 'success'](
nextToast.message,
nextToast.title,
{
closeButton: true,
timeOut: 10000
}
);
}
}, },
render() { render() {
@ -38,6 +75,10 @@ export default contain(
<Row> <Row>
{ this.props.children } { this.props.children }
</Row> </Row>
<ToastContainer
className='toast-bottom-right'
ref='toaster'
toastMessageFactory={ toastMessageFactory } />
</div> </div>
); );
} }

View File

@ -33,8 +33,24 @@ export default Actions({
}, },
// routing // routing
// goTo(path: String) => path
goTo: null, goTo: null,
// goBack(arg?) => arg?
goBack: null, goBack: null,
// toast(args: { type?: String, message: String, title: String }) => args
toast(args) {
return {
transform(state) {
const id = state.toast && state.toast.id ? state.toast.id : 0;
const toast = { ...args, id: id + 1 };
return { ...state, toast };
}
};
},
// updateLocation(location: { pathname: String }) => location
updateLocation(location) { updateLocation(location) {
return { return {
transform(state) { transform(state) {

View File

@ -28,7 +28,8 @@ export default Store({
const { const {
updateLocation, updateLocation,
getUser, getUser,
setTitle setTitle,
toast
} = cat.getActions('appActions'); } = cat.getActions('appActions');
register( register(
@ -39,7 +40,8 @@ export default Store({
setTitle setTitle
) )
), ),
updateLocation updateLocation,
toast
) )
); );

View File

@ -242,7 +242,13 @@ export default Actions({
return { return {
...state, ...state,
points: username ? state.points + 1 : state.points, points: username ? state.points + 1 : state.points,
hikesApp hikesApp,
toast: {
title: 'Congratulations!',
message: 'Hike completed',
id: state.toast && state.toast.id ? state.toast.id + 1 : 0,
type: 'success'
}
}; };
}, },
optimistic: optimisticSave optimistic: optimisticSave

View File

@ -54,6 +54,7 @@
"express-session": "^1.12.1", "express-session": "^1.12.1",
"express-state": "^1.2.0", "express-state": "^1.2.0",
"express-validator": "^2.18.0", "express-validator": "^2.18.0",
"fbjs": "^0.6.0",
"fetchr": "~0.5.12", "fetchr": "~0.5.12",
"forever": "~0.15.1", "forever": "~0.15.1",
"frameguard": "~0.2.2", "frameguard": "~0.2.2",
@ -109,6 +110,7 @@
"react-motion": "~0.3.1", "react-motion": "~0.3.1",
"react-router": "^1.0.0", "react-router": "^1.0.0",
"react-router-bootstrap": "https://github.com/FreeCodeCamp/react-router-bootstrap.git#freecodecamp", "react-router-bootstrap": "https://github.com/FreeCodeCamp/react-router-bootstrap.git#freecodecamp",
"react-toastr": "^2.3.0",
"react-vimeo": "~0.0.3", "react-vimeo": "~0.0.3",
"request": "^2.65.0", "request": "^2.65.0",
"rev-del": "^1.0.5", "rev-del": "^1.0.5",