Add toasts to react app
This commit is contained in:
@ -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
9
client/err-saga.js
Normal 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));
|
||||||
|
}
|
@ -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,
|
@ -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)
|
||||||
|
@ -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
269
client/less/toastr.less
Normal 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;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@ -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>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
@ -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) {
|
||||||
|
@ -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
|
||||||
)
|
)
|
||||||
);
|
);
|
||||||
|
|
||||||
|
@ -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
|
||||||
|
@ -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",
|
||||||
|
Reference in New Issue
Block a user