diff --git a/.eslintrc b/.eslintrc
index 92fad475c3..0e95beca2c 100644
--- a/.eslintrc
+++ b/.eslintrc
@@ -227,7 +227,7 @@
"react/jsx-boolean-value": [1, "always"],
"jsx-quotes": [1, "prefer-single"],
"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-vars": 1,
"react/no-did-mount-set-state": 2,
diff --git a/client/err-saga.js b/client/err-saga.js
new file mode 100644
index 0000000000..d681e4cfcc
--- /dev/null
+++ b/client/err-saga.js
@@ -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));
+}
diff --git a/client/synchronise-history.js b/client/history-saga.js
similarity index 96%
rename from client/synchronise-history.js
rename to client/history-saga.js
index 3e36cf9747..25e65340b9 100644
--- a/client/synchronise-history.js
+++ b/client/history-saga.js
@@ -20,7 +20,7 @@ const emptyLocation = {
let prevKey;
let isSyncing = false;
-export default function synchroniseHistory(
+export default function historySaga(
history,
updateLocation,
goTo,
diff --git a/client/index.js b/client/index.js
index c3796092cb..ceaeb012ed 100644
--- a/client/index.js
+++ b/client/index.js
@@ -9,7 +9,8 @@ import { hydrate } from 'thundercats';
import { render$ } from 'thundercats-react';
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 DOMContianer = document.getElementById('fcc');
@@ -38,9 +39,16 @@ app$({ history, location: appLocation })
({ nextLocation, props }, appCat) => ({ nextLocation, props, appCat })
)
.doOnNext(({ appCat }) => {
- const { updateLocation, goTo, goBack } = appCat.getActions('appActions');
const appStore$ = appCat.getStore('appStore');
+ const {
+ toast,
+ updateLocation,
+ goTo,
+ goBack
+ } = appCat.getActions('appActions');
+
+
const routerState$ = appStore$
.map(({ location }) => location)
.distinctUntilChanged(
@@ -50,22 +58,24 @@ app$({ history, location: appLocation })
// set page title
appStore$
.pluck('title')
+ .distinctUntilChanged()
.doOnNext(title => document.title = title)
.subscribe(() => {});
- appStore$
- .pluck('err')
- .filter(err => !!err)
- .distinctUntilChanged()
- .subscribe(err => console.error(err));
-
- synchroniseHistory(
+ historySaga(
history,
updateLocation,
goTo,
goBack,
routerState$
);
+
+ const err$ = appStore$
+ .pluck('err')
+ .filter(err => !!err)
+ .distinctUntilChanged();
+
+ errSaga(err$, toast);
})
// allow store subscribe to subscribe to actions
.delay(10)
diff --git a/client/less/main.less b/client/less/main.less
index 69679ccbe2..053e5ddd4c 100644
--- a/client/less/main.less
+++ b/client/less/main.less
@@ -1138,3 +1138,4 @@ code {
@import "chat.less";
@import "jobs.less";
@import "challenge.less";
+@import "toastr.less";
diff --git a/client/less/toastr.less b/client/less/toastr.less
new file mode 100644
index 0000000000..f4774004a2
--- /dev/null
+++ b/client/less/toastr.less
@@ -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;
+ }
+ }
+}
diff --git a/common/app/App.jsx b/common/app/App.jsx
index 847542cce4..69e7d8b082 100644
--- a/common/app/App.jsx
+++ b/common/app/App.jsx
@@ -1,16 +1,33 @@
import React, { PropTypes } from 'react';
-import { contain } from 'thundercats-react';
import { Row } from 'react-bootstrap';
+import { ToastMessage, ToastContainer } from 'react-toastr';
+import { contain } from 'thundercats-react';
import { Nav } from './components/Nav';
+const toastMessageFactory = React.createFactory(ToastMessage.animation);
+
export default contain(
{
+ actions: ['appActions'],
store: 'appStore',
fetchAction: 'appActions.getUser',
isPrimed({ username }) {
return !!username;
},
+ map({
+ username,
+ points,
+ picture,
+ toast
+ }) {
+ return {
+ username,
+ points,
+ picture,
+ toast
+ };
+ },
getPayload(props) {
return {
isPrimed: !!props.username
@@ -21,11 +38,31 @@ export default contain(
displayName: 'FreeCodeCamp',
propTypes: {
+ appActions: PropTypes.object,
children: PropTypes.node,
+ username: PropTypes.string,
points: PropTypes.number,
picture: PropTypes.string,
- title: PropTypes.string,
- username: PropTypes.string
+ toast: PropTypes.object
+ },
+
+ 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() {
@@ -38,6 +75,10 @@ export default contain(
{ this.props.children }
+
);
}
diff --git a/common/app/flux/Actions.js b/common/app/flux/Actions.js
index 3594be6c8e..3a0915be93 100644
--- a/common/app/flux/Actions.js
+++ b/common/app/flux/Actions.js
@@ -33,8 +33,24 @@ export default Actions({
},
// routing
+ // goTo(path: String) => path
goTo: null,
+
+ // goBack(arg?) => arg?
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) {
return {
transform(state) {
diff --git a/common/app/flux/Store.js b/common/app/flux/Store.js
index 36116ad7b0..2914bc268c 100644
--- a/common/app/flux/Store.js
+++ b/common/app/flux/Store.js
@@ -28,7 +28,8 @@ export default Store({
const {
updateLocation,
getUser,
- setTitle
+ setTitle,
+ toast
} = cat.getActions('appActions');
register(
@@ -39,7 +40,8 @@ export default Store({
setTitle
)
),
- updateLocation
+ updateLocation,
+ toast
)
);
diff --git a/common/app/routes/Hikes/flux/Actions.js b/common/app/routes/Hikes/flux/Actions.js
index 5bf63fe56d..580ff0e074 100644
--- a/common/app/routes/Hikes/flux/Actions.js
+++ b/common/app/routes/Hikes/flux/Actions.js
@@ -242,7 +242,13 @@ export default Actions({
return {
...state,
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
diff --git a/package.json b/package.json
index f4de269e23..89925a45c3 100644
--- a/package.json
+++ b/package.json
@@ -54,6 +54,7 @@
"express-session": "^1.12.1",
"express-state": "^1.2.0",
"express-validator": "^2.18.0",
+ "fbjs": "^0.6.0",
"fetchr": "~0.5.12",
"forever": "~0.15.1",
"frameguard": "~0.2.2",
@@ -109,6 +110,7 @@
"react-motion": "~0.3.1",
"react-router": "^1.0.0",
"react-router-bootstrap": "https://github.com/FreeCodeCamp/react-router-bootstrap.git#freecodecamp",
+ "react-toastr": "^2.3.0",
"react-vimeo": "~0.0.3",
"request": "^2.65.0",
"rev-del": "^1.0.5",