From 57955cc5ebfb6df11f22f12911c13f2d9fbb328c Mon Sep 17 00:00:00 2001 From: ivan Date: Sun, 5 Mar 2017 19:09:00 -0300 Subject: [PATCH 1/5] Ivan - Fix gulp deploy [skip ci] --- client/gulp/config.js | 9 ++------- client/gulp/tasks/serverphp.js | 9 --------- client/src/config.js | 2 ++ client/src/index.html | 1 + client/src/lib-app/api-call.js | 11 ++++------- 5 files changed, 9 insertions(+), 23 deletions(-) delete mode 100644 client/gulp/tasks/serverphp.js create mode 100644 client/src/config.js diff --git a/client/gulp/config.js b/client/gulp/config.js index 481c1b5f..a142e407 100644 --- a/client/gulp/config.js +++ b/client/gulp/config.js @@ -5,22 +5,17 @@ module.exports = { 'serverport': 3000, 'scripts': { - 'src': './src/**/*.js', + 'src': './src/*.js', 'dest': './build/js/' }, - 'phpserver': { - 'base': './src/server/', - 'port': 8000 - }, - 'images': { 'src': './src/assets/images/**/*.{jpeg,jpg,png}', 'dest': './build/images/' }, 'styles': { - 'src': './src/**/*.scss', + 'src': './src/*.scss', 'dest': './build/css/' }, diff --git a/client/gulp/tasks/serverphp.js b/client/gulp/tasks/serverphp.js deleted file mode 100644 index ea36f95f..00000000 --- a/client/gulp/tasks/serverphp.js +++ /dev/null @@ -1,9 +0,0 @@ -'use strict'; - -var config = require('../config'); -var gulp = require('gulp'); -var connect = require('gulp-connect-php'); - -gulp.task('serverphp', function() { - //connect.server(config.phpserver); -}); diff --git a/client/src/config.js b/client/src/config.js new file mode 100644 index 00000000..e206592e --- /dev/null +++ b/client/src/config.js @@ -0,0 +1,2 @@ +root = 'http://localhost:3000'; +apiRoot = 'http://localhost:3000/api'; diff --git a/client/src/index.html b/client/src/index.html index d6e6adbb..a398c80c 100644 --- a/client/src/index.html +++ b/client/src/index.html @@ -14,6 +14,7 @@
+ diff --git a/client/src/lib-app/api-call.js b/client/src/lib-app/api-call.js index b20cf618..1e61dc52 100644 --- a/client/src/lib-app/api-call.js +++ b/client/src/lib-app/api-call.js @@ -2,9 +2,6 @@ const _ = require('lodash'); const APIUtils = require('lib-core/APIUtils'); const SessionStore = require('lib-app/session-store'); -const url = 'http://localhost:3000'; -const apiUrl = 'http://localhost:3000/api'; - function processData (data, dataAsForm = false) { let newData; @@ -31,7 +28,7 @@ module.exports = { call: function ({path, data, plain, dataAsForm}) { console.log('request ' + path, data); return new Promise(function (resolve, reject) { - APIUtils.post(apiUrl + path, processData(data, dataAsForm), dataAsForm) + APIUtils.post(apiRoot + path, processData(data, dataAsForm), dataAsForm) .then(function (result) { console.log(result); @@ -53,14 +50,14 @@ module.exports = { }, getFileLink(filePath) { - return apiUrl + '/system/download?file=' + filePath; + return apiRoot + '/system/download?file=' + filePath; }, getAPIUrl() { - return apiUrl; + return apiRoot; }, getURL() { - return url; + return root; } }; \ No newline at end of file From 2b478befa09fb6d4d5a892adfd87cde0092ec8b9 Mon Sep 17 00:00:00 2001 From: ivan Date: Tue, 7 Mar 2017 01:33:30 -0300 Subject: [PATCH 2/5] Ivan - Create installation base layout [skip ci] --- client/gulp/config.js | 1 + client/gulp/tasks/browserify.js | 6 ++ client/gulp/tasks/development.js | 2 +- client/gulp/tasks/watch.js | 2 +- client/src/app/Routes.js | 18 +++++ client/src/app/install/install-layout.js | 80 +++++++++++++++++++ client/src/app/install/install-layout.scss | 75 +++++++++++++++++ .../app/install/install-step-1-language.js | 14 ++++ .../install/install-step-2-requirements.js | 14 ++++ .../app/install/install-step-3-settings.js | 14 ++++ .../install/install-step-4-administrator.js | 14 ++++ .../src/app/install/install-step-5-install.js | 14 ++++ .../app/install/install-step-6-completed.js | 14 ++++ client/src/data/languages/en.js | 8 ++ 14 files changed, 274 insertions(+), 2 deletions(-) create mode 100644 client/src/app/install/install-layout.js create mode 100644 client/src/app/install/install-layout.scss create mode 100644 client/src/app/install/install-step-1-language.js create mode 100644 client/src/app/install/install-step-2-requirements.js create mode 100644 client/src/app/install/install-step-3-settings.js create mode 100644 client/src/app/install/install-step-4-administrator.js create mode 100644 client/src/app/install/install-step-5-install.js create mode 100644 client/src/app/install/install-step-6-completed.js diff --git a/client/gulp/config.js b/client/gulp/config.js index a142e407..23403c8d 100644 --- a/client/gulp/config.js +++ b/client/gulp/config.js @@ -16,6 +16,7 @@ module.exports = { 'styles': { 'src': './src/*.scss', + 'watch': './src/**/*.scss', 'dest': './build/css/' }, diff --git a/client/gulp/tasks/browserify.js b/client/gulp/tasks/browserify.js index 352a2ebe..51e38361 100644 --- a/client/gulp/tasks/browserify.js +++ b/client/gulp/tasks/browserify.js @@ -66,4 +66,10 @@ gulp.task('browserify', function() { // Only run watchify if NOT production return buildScript('index.js', !global.isProd); +}); + +gulp.task('config', function() { + + return gulp.src(config.sourceDir + 'config.js') + .pipe(gulp.dest(config.scripts.dest)) }); \ No newline at end of file diff --git a/client/gulp/tasks/development.js b/client/gulp/tasks/development.js index 4d745d1e..fada97fe 100644 --- a/client/gulp/tasks/development.js +++ b/client/gulp/tasks/development.js @@ -10,6 +10,6 @@ gulp.task('dev', ['clean'], function(callback) { global.isProd = false; // Run all tasks once - return runSequence(['sass', 'imagemin', 'browserify', 'copyFonts', 'copyIndex', 'copyIcons', 'serverphp', 'fonts'], 'watch', callback); + return runSequence(['sass', 'imagemin', 'browserify', 'copyFonts', 'copyIndex', 'copyIcons', 'config', 'fonts'], 'watch', callback); }); \ No newline at end of file diff --git a/client/gulp/tasks/watch.js b/client/gulp/tasks/watch.js index 8c7207c2..975bc7f7 100644 --- a/client/gulp/tasks/watch.js +++ b/client/gulp/tasks/watch.js @@ -6,7 +6,7 @@ var config = require('../config'); gulp.task('watch', ['browserSync', 'server'], function() { // Scripts are automatically watched by Watchify inside Browserify task - gulp.watch(config.styles.src, ['sass']); + gulp.watch(config.styles.watch, ['sass']); gulp.watch(config.images.src, ['imagemin']); gulp.watch(config.sourceDir + 'index.html', ['copyIndex']); diff --git a/client/src/app/Routes.js b/client/src/app/Routes.js index ee18c790..1c4e880f 100644 --- a/client/src/app/Routes.js +++ b/client/src/app/Routes.js @@ -53,6 +53,15 @@ import AdminPanelSystemPreferences from 'app/admin/panel/settings/admin-panel-sy import AdminPanelAdvancedSettings from 'app/admin/panel/settings/admin-panel-advanced-settings'; import AdminPanelEmailTemplates from 'app/admin/panel/settings/admin-panel-email-templates'; +// INSTALLATION +import InstallLayout from 'app/install/install-layout'; +import InstallStep1Language from 'app/install/install-step-1-language'; +import InstallStep2Requirements from 'app/install/install-step-2-requirements'; +import InstallStep3Settings from 'app/install/install-step-3-settings'; +import InstallStep4Administrator from 'app/install/install-step-4-administrator'; +import InstallStep5Install from 'app/install/install-step-5-install'; +import InstallStep6Completed from 'app/install/install-step-6-completed'; + const history = syncHistoryWithStore(browserHistory, store); export default ( @@ -82,6 +91,15 @@ export default ( + + + + + + + + + diff --git a/client/src/app/install/install-layout.js b/client/src/app/install/install-layout.js new file mode 100644 index 00000000..442e4984 --- /dev/null +++ b/client/src/app/install/install-layout.js @@ -0,0 +1,80 @@ +import React from 'react'; +import classNames from 'classnames'; + +import i18n from 'lib-app/i18n'; + +import Widget from 'core-components/widget'; +import Icon from 'core-components/icon'; + +const steps = [ + 'LANGUAGE', + 'SERVER_REQUIREMENTS', + 'SETTINGS_SETUP', + 'ADMIN_SETUP', + 'INSTALL', + 'COMPLETED' +]; + +class InstallLayout extends React.Component { + + render() { + return ( + +
+
+ OpenSupports Installation +
+
+
+ {i18n('INSTALL_HEADER_TITLE')} +
+
+ {i18n('INSTALL_HEADER_DESCRIPTION')} +
+
+
+ +
+
+
    + {steps.map(this.renderStep.bind(this))} +
+
+
+ {this.props.children} +
+
+
+ ); + } + + renderStep(key, index) { + let classes = { + 'install-layout__step': true, + 'install-layout__step_current': index === this.getCurrentStep(), + 'install-layout__step_previous': index < this.getCurrentStep() + }; + let icon = 'circle-thin'; + + if(index === this.getCurrentStep()) { + icon = 'arrow-circle-right'; + } else if(index < this.getCurrentStep()) { + icon = 'check-circle'; + } + + return ( +
  • + + + {index+1}. {i18n(key)} + +
  • + ) + } + + getCurrentStep() { + return 2; + } +} + +export default InstallLayout; \ No newline at end of file diff --git a/client/src/app/install/install-layout.scss b/client/src/app/install/install-layout.scss new file mode 100644 index 00000000..f6e7cf15 --- /dev/null +++ b/client/src/app/install/install-layout.scss @@ -0,0 +1,75 @@ +@import "../../scss/vars"; + +.install-layout { + margin: 0 auto; + width: 900px; + min-height: 0; + + @media screen and (max-width: 900px) { + width: 100%; + } + + &__header { + text-align: left; + + &-logo { + display: inline-block; + max-width: 300px; + } + + &-text { + display: inline-block; + margin-left: 20px; + } + + &-title { + text-align: left; + font-weight: bold; + font-size: $font-size--md; + color: $primary-black; + } + + &-description { + text-align: left; + color: $dark-grey; + } + } + + &__body { + margin-top: 20px; + text-align: left; + } + + &__steps { + padding: 0; + list-style-type: none; + border-right: 1px solid $primary-blue; + } + + &__step { + color: $primary-black; + + &-icon { + color: $primary-blue; + margin-right: 10px; + } + + &_current { + + .install-layout__step-icon { + color: $secondary-blue; + } + } + + &_previous { + + .install-layout__step-icon { + color: $primary-green; + } + } + } + + &__content { + min-height: 130px; + } +} diff --git a/client/src/app/install/install-step-1-language.js b/client/src/app/install/install-step-1-language.js new file mode 100644 index 00000000..eaa6c379 --- /dev/null +++ b/client/src/app/install/install-step-1-language.js @@ -0,0 +1,14 @@ +import React from 'react'; + +class InstallStep1Language extends React.Component { + + render() { + return ( +
    + INSTALLATION +
    + ); + } +} + +export default InstallStep1Language; \ No newline at end of file diff --git a/client/src/app/install/install-step-2-requirements.js b/client/src/app/install/install-step-2-requirements.js new file mode 100644 index 00000000..8066728a --- /dev/null +++ b/client/src/app/install/install-step-2-requirements.js @@ -0,0 +1,14 @@ +import React from 'react'; + +class InstallStep2Requirements extends React.Component { + + render() { + return ( +
    + INSTALLATION +
    + ); + } +} + +export default InstallStep2Requirements; \ No newline at end of file diff --git a/client/src/app/install/install-step-3-settings.js b/client/src/app/install/install-step-3-settings.js new file mode 100644 index 00000000..31048b3c --- /dev/null +++ b/client/src/app/install/install-step-3-settings.js @@ -0,0 +1,14 @@ +import React from 'react'; + +class InstallStep3Settings extends React.Component { + + render() { + return ( +
    + INSTALLATION +
    + ); + } +} + +export default InstallStep3Settings; \ No newline at end of file diff --git a/client/src/app/install/install-step-4-administrator.js b/client/src/app/install/install-step-4-administrator.js new file mode 100644 index 00000000..f3a63395 --- /dev/null +++ b/client/src/app/install/install-step-4-administrator.js @@ -0,0 +1,14 @@ +import React from 'react'; + +class InstallStep4Administrator extends React.Component { + + render() { + return ( +
    + INSTALLATION +
    + ); + } +} + +export default InstallStep4Administrator; \ No newline at end of file diff --git a/client/src/app/install/install-step-5-install.js b/client/src/app/install/install-step-5-install.js new file mode 100644 index 00000000..70145bbd --- /dev/null +++ b/client/src/app/install/install-step-5-install.js @@ -0,0 +1,14 @@ +import React from 'react'; + +class InstallStep4Install extends React.Component { + + render() { + return ( +
    + INSTALLATION +
    + ); + } +} + +export default InstallStep4Install; \ No newline at end of file diff --git a/client/src/app/install/install-step-6-completed.js b/client/src/app/install/install-step-6-completed.js new file mode 100644 index 00000000..dcc56d3b --- /dev/null +++ b/client/src/app/install/install-step-6-completed.js @@ -0,0 +1,14 @@ +import React from 'react'; + +class InstallStep4Completed extends React.Component { + + render() { + return ( +
    + INSTALLATION +
    + ); + } +} + +export default InstallStep4Completed; \ No newline at end of file diff --git a/client/src/data/languages/en.js b/client/src/data/languages/en.js index b56acad0..bb4c2670 100644 --- a/client/src/data/languages/en.js +++ b/client/src/data/languages/en.js @@ -208,6 +208,14 @@ export default { 'ACTIVITY_DELETE_USER': 'deleted user', 'ACTIVITY_UN_BAN_USER': 'banned user', + 'SERVER_REQUIREMENTS': 'Server requirements', + 'SETTINGS_SETUP': 'Settings setup', + 'ADMIN_SETUP': 'Admin setup', + 'INSTALL': 'Install', + 'COMPLETED': 'Completed', + 'INSTALL_HEADER_TITLE': 'OpenSupports Installation Wizard', + 'INSTALL_HEADER_DESCRIPTION': 'This wizard will help you to configure and install OpenSupports on your website', + //VIEW DESCRIPTIONS 'CREATE_TICKET_DESCRIPTION': 'This is a form for creating tickets. Fill the form and send us your issues/doubts/suggestions. Our support system will answer it as soon as possible.', 'TICKET_LIST_DESCRIPTION': 'Here you can find a list of all tickets you have sent to our support team.', From d6f33a7b329c8846f4ddfec1f9e1c18c3a4c1c73 Mon Sep 17 00:00:00 2001 From: ivan Date: Thu, 9 Mar 2017 00:32:35 -0300 Subject: [PATCH 3/5] Ivan - Add first and second step to installation [skip ci] --- client/src/app/Routes.js | 4 +- client/src/app/install/install-layout.js | 17 +++- .../app/install/install-step-1-language.js | 39 ++++++++- .../app/install/install-step-1-language.scss | 13 +++ .../install/install-step-2-requirements.js | 85 ++++++++++++++++++- .../install/install-step-2-requirements.scss | 51 +++++++++++ .../app/install/install-step-3-settings.scss | 1 + .../install/install-step-4-administrator.scss | 1 + .../src/app/install/install-step-5-install.js | 4 +- .../app/install/install-step-5-install.scss | 1 + .../app/install/install-step-6-completed.js | 4 +- .../app/install/install-step-6-completed.scss | 1 + client/src/core-components/table.js | 2 + client/src/data/fixtures/system-fixtures.js | 33 ++++++- client/src/data/languages/en.js | 15 +++- 15 files changed, 257 insertions(+), 14 deletions(-) create mode 100644 client/src/app/install/install-step-1-language.scss create mode 100644 client/src/app/install/install-step-2-requirements.scss create mode 100644 client/src/app/install/install-step-3-settings.scss create mode 100644 client/src/app/install/install-step-4-administrator.scss create mode 100644 client/src/app/install/install-step-5-install.scss create mode 100644 client/src/app/install/install-step-6-completed.scss diff --git a/client/src/app/Routes.js b/client/src/app/Routes.js index 1c4e880f..db90bd6b 100644 --- a/client/src/app/Routes.js +++ b/client/src/app/Routes.js @@ -97,8 +97,8 @@ export default ( - - + + diff --git a/client/src/app/install/install-layout.js b/client/src/app/install/install-layout.js index 442e4984..61a5c1d7 100644 --- a/client/src/app/install/install-layout.js +++ b/client/src/app/install/install-layout.js @@ -1,5 +1,6 @@ import React from 'react'; import classNames from 'classnames'; +import _ from 'lodash'; import i18n from 'lib-app/i18n'; @@ -73,7 +74,21 @@ class InstallLayout extends React.Component { } getCurrentStep() { - return 2; + const pathname = this.props.location.pathname; + + if(_.includes(pathname, 'step-1')) { + return 0; + } else if(_.includes(pathname, 'step-2')) { + return 1; + } else if(_.includes(pathname, 'step-3')) { + return 2; + } else if(_.includes(pathname, 'step-4')) { + return 3; + } else if(_.includes(pathname, 'step-5')) { + return 4; + } else if(_.includes(pathname, 'step-6')) { + return 5; + } } } diff --git a/client/src/app/install/install-step-1-language.js b/client/src/app/install/install-step-1-language.js index eaa6c379..890ada50 100644 --- a/client/src/app/install/install-step-1-language.js +++ b/client/src/app/install/install-step-1-language.js @@ -1,14 +1,47 @@ import React from 'react'; +import {connect} from 'react-redux'; +import {browserHistory} from 'react-router'; + +import i18n from 'lib-app/i18n'; +import ConfigActions from 'actions/config-actions'; + +import LanguageSelector from 'app-components/language-selector'; +import Button from 'core-components/button'; +import Header from 'core-components/header'; class InstallStep1Language extends React.Component { render() { return ( -
    - INSTALLATION +
    +
    + +
    + +
    ); } + + getLanguageSelectorProps() { + return { + className: 'install-step-1__languages', + value: this.props.config.language, + type: 'allowed', + onChange: this.changeLanguage.bind(this) + }; + } + + changeLanguage(event) { + this.props.dispatch(ConfigActions.changeLanguage(event.target.value)); + } } -export default InstallStep1Language; \ No newline at end of file + +export default connect((store) => { + return { + config: store.config + }; +})(InstallStep1Language); \ No newline at end of file diff --git a/client/src/app/install/install-step-1-language.scss b/client/src/app/install/install-step-1-language.scss new file mode 100644 index 00000000..c21546e8 --- /dev/null +++ b/client/src/app/install/install-step-1-language.scss @@ -0,0 +1,13 @@ +@import "../../scss/vars"; + +.install-step-1 { + + &__label { + font-size: $font-size--md; + margin-bottom: 20px; + } + + &__button { + margin-top: 20px; + } +} \ No newline at end of file diff --git a/client/src/app/install/install-step-2-requirements.js b/client/src/app/install/install-step-2-requirements.js index 8066728a..8b532704 100644 --- a/client/src/app/install/install-step-2-requirements.js +++ b/client/src/app/install/install-step-2-requirements.js @@ -1,14 +1,95 @@ import React from 'react'; +import _ from 'lodash'; +import classNames from 'classnames'; +import {browserHistory} from 'react-router'; + +import i18n from 'lib-app/i18n'; +import API from 'lib-app/api-call'; + +import Icon from 'core-components/icon'; +import Button from 'core-components/button'; +import Header from 'core-components/header'; +import Table from 'core-components/table'; class InstallStep2Requirements extends React.Component { + + state = { + loading: true, + requirements: { + phpVersion: {}, + PDO: {}, + files: {}, + configFile: {} + } + }; + + componentDidMount() { + this.retrieveRequirements(); + } render() { return ( -
    - INSTALLATION +
    +
    +
    + +
    + +
    + +
    ); } + + getTableProps() { + return { + className: 'install-step-2__requirement-list', + pageSize: 30, + loading: this.state.loading, + headers: [ + {key: 'name', value: i18n('REQUIREMENT'), className: 'col-md-3'}, + {key: 'value', value: i18n('VALUE'), className: 'col-md-9'} + ], + rows: Object.keys(this.state.requirements).map(this.getRequirement.bind(this)) + }; + } + + getRequirement(key) { + const requirement = this.state.requirements[key]; + + return { + className: this.getRequirementClass(requirement), + name: requirement.name, + value: ( +
    + {requirement.value} + +
    + ) + }; + } + + getRequirementClass(requirement) { + let classes = { + 'install-step-2__requirement': true, + 'install-step-2__requirement_error': !requirement.ok + }; + + return classNames(classes); + } + + isAllOk() { + return _.every(this.state.requirements, {ok: true}); + } + + retrieveRequirements() { + this.setState({loading: true}, () => API.call({path: '/system/check-requirements'}).then(({data}) => this.setState({requirements: data, loading: false}))); + } } export default InstallStep2Requirements; \ No newline at end of file diff --git a/client/src/app/install/install-step-2-requirements.scss b/client/src/app/install/install-step-2-requirements.scss new file mode 100644 index 00000000..1c5e2bd5 --- /dev/null +++ b/client/src/app/install/install-step-2-requirements.scss @@ -0,0 +1,51 @@ +@import "../../scss/vars"; + +.install-step-2 { + + &__refresh { + margin: 15px 0; + + &-button { + width: 120px; + position: relative; + text-align: right; + } + + &-icon { + position: absolute; + left: 8px; + top: 8px; + } + } + + &__requirement-list { + margin-bottom: 20px; + } + + &__requirement { + color: $secondary-blue; + + &-value { + width: 200px; + } + + &-assert { + color: $primary-green; + float: right; + margin-top: 3px; + } + + &_error { + color: $secondary-red; + background-color: #ffdfe0 !important; + + .install-step-2__requirement-assert { + color: $secondary-red; + } + } + } + + &__next { + + } +} \ No newline at end of file diff --git a/client/src/app/install/install-step-3-settings.scss b/client/src/app/install/install-step-3-settings.scss new file mode 100644 index 00000000..c990d9a8 --- /dev/null +++ b/client/src/app/install/install-step-3-settings.scss @@ -0,0 +1 @@ +@import "../../scss/vars"; diff --git a/client/src/app/install/install-step-4-administrator.scss b/client/src/app/install/install-step-4-administrator.scss new file mode 100644 index 00000000..c990d9a8 --- /dev/null +++ b/client/src/app/install/install-step-4-administrator.scss @@ -0,0 +1 @@ +@import "../../scss/vars"; diff --git a/client/src/app/install/install-step-5-install.js b/client/src/app/install/install-step-5-install.js index 70145bbd..61d60200 100644 --- a/client/src/app/install/install-step-5-install.js +++ b/client/src/app/install/install-step-5-install.js @@ -1,6 +1,6 @@ import React from 'react'; -class InstallStep4Install extends React.Component { +class InstallStep5Install extends React.Component { render() { return ( @@ -11,4 +11,4 @@ class InstallStep4Install extends React.Component { } } -export default InstallStep4Install; \ No newline at end of file +export default InstallStep5Install; \ No newline at end of file diff --git a/client/src/app/install/install-step-5-install.scss b/client/src/app/install/install-step-5-install.scss new file mode 100644 index 00000000..c990d9a8 --- /dev/null +++ b/client/src/app/install/install-step-5-install.scss @@ -0,0 +1 @@ +@import "../../scss/vars"; diff --git a/client/src/app/install/install-step-6-completed.js b/client/src/app/install/install-step-6-completed.js index dcc56d3b..d2733f8f 100644 --- a/client/src/app/install/install-step-6-completed.js +++ b/client/src/app/install/install-step-6-completed.js @@ -1,6 +1,6 @@ import React from 'react'; -class InstallStep4Completed extends React.Component { +class InstallStep6Completed extends React.Component { render() { return ( @@ -11,4 +11,4 @@ class InstallStep4Completed extends React.Component { } } -export default InstallStep4Completed; \ No newline at end of file +export default InstallStep6Completed; \ No newline at end of file diff --git a/client/src/app/install/install-step-6-completed.scss b/client/src/app/install/install-step-6-completed.scss new file mode 100644 index 00000000..c990d9a8 --- /dev/null +++ b/client/src/app/install/install-step-6-completed.scss @@ -0,0 +1 @@ +@import "../../scss/vars"; diff --git a/client/src/core-components/table.js b/client/src/core-components/table.js index 099557d4..fe882f72 100644 --- a/client/src/core-components/table.js +++ b/client/src/core-components/table.js @@ -146,6 +146,8 @@ class Table extends React.Component { 'table__row-highlighted': row.highlighted }; + classes[row.className] = (row.className); + return classNames(classes); } diff --git a/client/src/data/fixtures/system-fixtures.js b/client/src/data/fixtures/system-fixtures.js index b71b7219..9a7ebe26 100644 --- a/client/src/data/fixtures/system-fixtures.js +++ b/client/src/data/fixtures/system-fixtures.js @@ -63,7 +63,38 @@ module.exports = [ return { status: 'success', data: {} - } + }; + } + }, + { + path: '/system/check-requirements', + time: 50, + response: function () { + return { + status: 'success', + data: { + phpVersion: { + name: 'PHP Version', + value: 5.6, + ok: true + }, + PDO: { + name: 'PDO Module', + value: 'Available', + ok: true + }, + files: { + name: 'Folder: /api/files', + value: 'Writable', + ok: true + }, + configFile: { + name: 'File: /api/config.php', + value: 'No writable', + ok: false + } + } + }; } }, { diff --git a/client/src/data/languages/en.js b/client/src/data/languages/en.js index bb4c2670..80d15eda 100644 --- a/client/src/data/languages/en.js +++ b/client/src/data/languages/en.js @@ -173,7 +173,8 @@ export default { 'ACTIVITY': 'Activity', 'HOME': 'Home', 'TICKET_NUMBER': 'Ticket number', - + 'NEXT': 'Next', + 'CHART_CREATE_TICKET': 'Tickets created', 'CHART_CLOSE': 'Tickets closed', 'CHART_SIGNUP': 'Signups', @@ -215,6 +216,18 @@ export default { 'COMPLETED': 'Completed', 'INSTALL_HEADER_TITLE': 'OpenSupports Installation Wizard', 'INSTALL_HEADER_DESCRIPTION': 'This wizard will help you to configure and install OpenSupports on your website', + 'SELECT_LANGUAGE': 'Select language', + 'REQUIREMENT': 'Requirement', + 'VALUE': 'Value', + 'REFRESH': 'Refresh', + + 'STEP_TITLE': 'Step {current} of {total} - {title}', + 'STEP_1_DESCRIPTION': 'Select your preferred language for the installation wizard.', + 'STEP_2_DESCRIPTION': 'Here are listed the requirements for running OpenSupports. Please make sure that all requirements are satisfied.', + 'STEP_3_DESCRIPTION': 'Select your preferred language for the installation wizard.', + 'STEP_4_DESCRIPTION': 'Select your preferred language for the installation wizard.', + 'STEP_5_DESCRIPTION': 'Select your preferred language for the installation wizard.', + 'STEP_6_DESCRIPTION': 'Select your preferred language for the installation wizard.', //VIEW DESCRIPTIONS 'CREATE_TICKET_DESCRIPTION': 'This is a form for creating tickets. Fill the form and send us your issues/doubts/suggestions. Our support system will answer it as soon as possible.', From 01192aaeb706370d29c2d04d725d4f6c79cda493 Mon Sep 17 00:00:00 2001 From: ivan Date: Sat, 11 Mar 2017 02:19:02 -0300 Subject: [PATCH 4/5] Ivan - Add all install steps [skip ci] --- client/gulp/config.js | 3 +- client/gulp/tasks/watch.js | 2 +- client/src/app-components/toggle-button.js | 4 +- client/src/app-components/toggle-button.scss | 7 ++ client/src/app/Routes.js | 12 +-- client/src/app/install/install-layout.js | 4 +- .../app/install/install-step-3-database.js | 78 +++++++++++++++++++ .../app/install/install-step-3-database.scss | 18 +++++ .../app/install/install-step-3-settings.js | 14 ---- .../app/install/install-step-3-settings.scss | 1 - .../install/install-step-4-administrator.js | 14 ---- .../install/install-step-4-administrator.scss | 1 - .../app/install/install-step-4-user-system.js | 68 ++++++++++++++++ .../install/install-step-4-user-system.scss | 14 ++++ .../src/app/install/install-step-5-admin.js | 77 ++++++++++++++++++ .../src/app/install/install-step-5-admin.scss | 18 +++++ .../src/app/install/install-step-5-install.js | 14 ---- .../app/install/install-step-5-install.scss | 1 - .../app/install/install-step-6-completed.js | 18 ++++- client/src/core-components/form-field.js | 11 ++- client/src/data/fixtures/system-fixtures.js | 35 ++++++++- client/src/data/languages/en.js | 23 ++++-- client/src/index.html | 1 + 23 files changed, 368 insertions(+), 70 deletions(-) create mode 100644 client/src/app/install/install-step-3-database.js create mode 100644 client/src/app/install/install-step-3-database.scss delete mode 100644 client/src/app/install/install-step-3-settings.js delete mode 100644 client/src/app/install/install-step-3-settings.scss delete mode 100644 client/src/app/install/install-step-4-administrator.js delete mode 100644 client/src/app/install/install-step-4-administrator.scss create mode 100644 client/src/app/install/install-step-4-user-system.js create mode 100644 client/src/app/install/install-step-4-user-system.scss create mode 100644 client/src/app/install/install-step-5-admin.js create mode 100644 client/src/app/install/install-step-5-admin.scss delete mode 100644 client/src/app/install/install-step-5-install.js delete mode 100644 client/src/app/install/install-step-5-install.scss diff --git a/client/gulp/config.js b/client/gulp/config.js index 23403c8d..86e35dc7 100644 --- a/client/gulp/config.js +++ b/client/gulp/config.js @@ -15,8 +15,7 @@ module.exports = { }, 'styles': { - 'src': './src/*.scss', - 'watch': './src/**/*.scss', + 'src': './src/**/*.scss', 'dest': './build/css/' }, diff --git a/client/gulp/tasks/watch.js b/client/gulp/tasks/watch.js index 975bc7f7..8c7207c2 100644 --- a/client/gulp/tasks/watch.js +++ b/client/gulp/tasks/watch.js @@ -6,7 +6,7 @@ var config = require('../config'); gulp.task('watch', ['browserSync', 'server'], function() { // Scripts are automatically watched by Watchify inside Browserify task - gulp.watch(config.styles.watch, ['sass']); + gulp.watch(config.styles.src, ['sass']); gulp.watch(config.images.src, ['imagemin']); gulp.watch(config.sourceDir + 'index.html', ['copyIndex']); diff --git a/client/src/app-components/toggle-button.js b/client/src/app-components/toggle-button.js index ee32845b..ebd3098e 100644 --- a/client/src/app-components/toggle-button.js +++ b/client/src/app-components/toggle-button.js @@ -4,7 +4,6 @@ import classNames from 'classnames'; import i18n from 'lib-app/i18n'; class ToggleButton extends React.Component { - static propTypes = { value: React.PropTypes.bool, @@ -22,6 +21,7 @@ class ToggleButton extends React.Component { getClass() { let classes = { 'toggle-button': true, + 'toggle-button_disabled': (this.props.disabled), [this.props.className]: (this.props.className) }; @@ -30,7 +30,7 @@ class ToggleButton extends React.Component { onClick() { - if (this.props.onChange) { + if (this.props.onChange && !this.props.disabled) { this.props.onChange({ target: { value: !this.props.value diff --git a/client/src/app-components/toggle-button.scss b/client/src/app-components/toggle-button.scss index 6006ae56..f0c3ce7f 100644 --- a/client/src/app-components/toggle-button.scss +++ b/client/src/app-components/toggle-button.scss @@ -7,4 +7,11 @@ border-radius: 4px; text-align: center; user-select: none; + cursor: pointer; + + &_disabled { + cursor: default; + background-color: transparent; + color: $dark-grey; + } } \ No newline at end of file diff --git a/client/src/app/Routes.js b/client/src/app/Routes.js index db90bd6b..f6d3c668 100644 --- a/client/src/app/Routes.js +++ b/client/src/app/Routes.js @@ -57,9 +57,9 @@ import AdminPanelEmailTemplates from 'app/admin/panel/settings/admin-panel-email import InstallLayout from 'app/install/install-layout'; import InstallStep1Language from 'app/install/install-step-1-language'; import InstallStep2Requirements from 'app/install/install-step-2-requirements'; -import InstallStep3Settings from 'app/install/install-step-3-settings'; -import InstallStep4Administrator from 'app/install/install-step-4-administrator'; -import InstallStep5Install from 'app/install/install-step-5-install'; +import InstallStep3Database from 'app/install/install-step-3-database'; +import InstallStep4UserSystem from 'app/install/install-step-4-user-system'; +import InstallStep5Admin from 'app/install/install-step-5-admin'; import InstallStep6Completed from 'app/install/install-step-6-completed'; const history = syncHistoryWithStore(browserHistory, store); @@ -95,9 +95,9 @@ export default ( - - - + + + diff --git a/client/src/app/install/install-layout.js b/client/src/app/install/install-layout.js index 61a5c1d7..b74b1cdd 100644 --- a/client/src/app/install/install-layout.js +++ b/client/src/app/install/install-layout.js @@ -10,9 +10,9 @@ import Icon from 'core-components/icon'; const steps = [ 'LANGUAGE', 'SERVER_REQUIREMENTS', - 'SETTINGS_SETUP', + 'DATABASE_CONFIGURATION', + 'USER_SYSTEM', 'ADMIN_SETUP', - 'INSTALL', 'COMPLETED' ]; diff --git a/client/src/app/install/install-step-3-database.js b/client/src/app/install/install-step-3-database.js new file mode 100644 index 00000000..08ed8ebb --- /dev/null +++ b/client/src/app/install/install-step-3-database.js @@ -0,0 +1,78 @@ +import React from 'react'; +import {browserHistory} from 'react-router'; + +import i18n from 'lib-app/i18n'; +import API from 'lib-app/api-call'; + +import Button from 'core-components/button'; +import Header from 'core-components/header'; +import Form from 'core-components/form'; +import FormField from 'core-components/form-field'; +import SubmitButton from 'core-components/submit-button'; +import Message from 'core-components/message'; + +class InstallStep3Database extends React.Component { + + state = { + loading: false, + error: false, + errorMessage: '' + }; + + render() { + return ( +
    +
    + {this.renderMessage()} +
    + + + + +
    + {i18n('NEXT')} + +
    + +
    + ); + } + + renderMessage() { + let message = null; + + if(this.state.error) { + message = ( + + {i18n('ERROR_UPDATING_SETTINGS')}: {this.state.errorMessage} + + ); + } + + return message; + } + + onPreviousClick(event) { + event.preventDefault(); + browserHistory.push('/install/step-2'); + } + + onSubmit(form) { + this.setState({ + loading: true + }, () => { + API.call({ + path: '/system/init-database', + data: form + }) + .then(() => browserHistory.push('/install/step-4')) + .catch(({message}) => this.setState({ + loading: false, + error: true, + errorMessage: message + })); + }); + } +} + +export default InstallStep3Database; \ No newline at end of file diff --git a/client/src/app/install/install-step-3-database.scss b/client/src/app/install/install-step-3-database.scss new file mode 100644 index 00000000..6a4bac27 --- /dev/null +++ b/client/src/app/install/install-step-3-database.scss @@ -0,0 +1,18 @@ +@import "../../scss/vars"; + +.install-step-3 { + + &__message { + margin-bottom: 20px; + } + + &__previous { + float: left; + } + + &__next { + float: left; + position: absolute; + margin-left: 286px; + } +} \ No newline at end of file diff --git a/client/src/app/install/install-step-3-settings.js b/client/src/app/install/install-step-3-settings.js deleted file mode 100644 index 31048b3c..00000000 --- a/client/src/app/install/install-step-3-settings.js +++ /dev/null @@ -1,14 +0,0 @@ -import React from 'react'; - -class InstallStep3Settings extends React.Component { - - render() { - return ( -
    - INSTALLATION -
    - ); - } -} - -export default InstallStep3Settings; \ No newline at end of file diff --git a/client/src/app/install/install-step-3-settings.scss b/client/src/app/install/install-step-3-settings.scss deleted file mode 100644 index c990d9a8..00000000 --- a/client/src/app/install/install-step-3-settings.scss +++ /dev/null @@ -1 +0,0 @@ -@import "../../scss/vars"; diff --git a/client/src/app/install/install-step-4-administrator.js b/client/src/app/install/install-step-4-administrator.js deleted file mode 100644 index f3a63395..00000000 --- a/client/src/app/install/install-step-4-administrator.js +++ /dev/null @@ -1,14 +0,0 @@ -import React from 'react'; - -class InstallStep4Administrator extends React.Component { - - render() { - return ( -
    - INSTALLATION -
    - ); - } -} - -export default InstallStep4Administrator; \ No newline at end of file diff --git a/client/src/app/install/install-step-4-administrator.scss b/client/src/app/install/install-step-4-administrator.scss deleted file mode 100644 index c990d9a8..00000000 --- a/client/src/app/install/install-step-4-administrator.scss +++ /dev/null @@ -1 +0,0 @@ -@import "../../scss/vars"; diff --git a/client/src/app/install/install-step-4-user-system.js b/client/src/app/install/install-step-4-user-system.js new file mode 100644 index 00000000..b5f5af88 --- /dev/null +++ b/client/src/app/install/install-step-4-user-system.js @@ -0,0 +1,68 @@ +import React from 'react'; +import {browserHistory} from 'react-router'; + +import i18n from 'lib-app/i18n'; +import API from 'lib-app/api-call'; + +import ToggleButton from 'app-components/toggle-button'; +import Button from 'core-components/button'; +import Header from 'core-components/header'; +import Form from 'core-components/form'; +import FormField from 'core-components/form-field'; +import SubmitButton from 'core-components/submit-button'; + +class InstallStep4UserSystem extends React.Component { + + state = { + form: { + 'user-system-enabled': true, + 'registration': true + } + }; + + render() { + return ( +
    +
    +
    + + +
    + {i18n('NEXT')} + +
    + +
    + ); + } + + onChange(form) { + this.setState({ + form: { + 'user-system-enabled': form['user-system-enabled'], + 'registration': form['user-system-enabled'] && form['registration'] + } + }); + } + + onPreviousClick(event) { + event.preventDefault(); + browserHistory.push('/install/step-3'); + } + + onSubmit(form) { + API.call({ + path: '/system/init-settings', + data: { + 'user-system-enabled': form['user-system-enabled'], + 'registration': form['registration'] + } + }).then(() => browserHistory.push('/install/step-5')); + } + + isDisabled() { + return !this.state.form['user-system-enabled']; + } +} + +export default InstallStep4UserSystem; \ No newline at end of file diff --git a/client/src/app/install/install-step-4-user-system.scss b/client/src/app/install/install-step-4-user-system.scss new file mode 100644 index 00000000..92e326bb --- /dev/null +++ b/client/src/app/install/install-step-4-user-system.scss @@ -0,0 +1,14 @@ +@import "../../scss/vars"; + +.install-step-4 { + + &__previous { + margin-right: 20px; + } + + &__next { + float: left; + position: absolute; + margin-left: 286px; + } +} \ No newline at end of file diff --git a/client/src/app/install/install-step-5-admin.js b/client/src/app/install/install-step-5-admin.js new file mode 100644 index 00000000..0c5c1903 --- /dev/null +++ b/client/src/app/install/install-step-5-admin.js @@ -0,0 +1,77 @@ +import React from 'react'; +import {browserHistory} from 'react-router'; + +import i18n from 'lib-app/i18n'; +import API from 'lib-app/api-call'; + +import Button from 'core-components/button'; +import Header from 'core-components/header'; +import Form from 'core-components/form'; +import FormField from 'core-components/form-field'; +import SubmitButton from 'core-components/submit-button'; +import Message from 'core-components/message'; + +class InstallStep5Admin extends React.Component { + + state = { + loading: false, + error: false, + errorMessage: '' + }; + + render() { + return ( +
    +
    + {this.renderMessage()} +
    + + + +
    + {i18n('NEXT')} + +
    + +
    + ); + } + + renderMessage() { + let message = null; + + if(this.state.error) { + message = ( + + {i18n('ERROR_UPDATING_SETTINGS')}: {this.state.errorMessage} + + ); + } + + return message; + } + + onPreviousClick(event) { + event.preventDefault(); + browserHistory.push('/install/step-4'); + } + + onSubmit(form) { + this.setState({ + loading: true + }, () => { + API.call({ + path: '/system/init-admin', + data: form + }) + .then(() => browserHistory.push('/install/step-6')) + .catch(({message}) => this.setState({ + loading: false, + error: true, + errorMessage: message + })); + }); + } +} + +export default InstallStep5Admin; \ No newline at end of file diff --git a/client/src/app/install/install-step-5-admin.scss b/client/src/app/install/install-step-5-admin.scss new file mode 100644 index 00000000..2b8eb92d --- /dev/null +++ b/client/src/app/install/install-step-5-admin.scss @@ -0,0 +1,18 @@ +@import "../../scss/vars"; + +.install-step-5 { + + &__message { + margin-bottom: 20px; + } + + &__previous { + margin-right: 20px; + } + + &__next { + float: left; + position: absolute; + margin-left: 286px; + } +} \ No newline at end of file diff --git a/client/src/app/install/install-step-5-install.js b/client/src/app/install/install-step-5-install.js deleted file mode 100644 index 61d60200..00000000 --- a/client/src/app/install/install-step-5-install.js +++ /dev/null @@ -1,14 +0,0 @@ -import React from 'react'; - -class InstallStep5Install extends React.Component { - - render() { - return ( -
    - INSTALLATION -
    - ); - } -} - -export default InstallStep5Install; \ No newline at end of file diff --git a/client/src/app/install/install-step-5-install.scss b/client/src/app/install/install-step-5-install.scss deleted file mode 100644 index c990d9a8..00000000 --- a/client/src/app/install/install-step-5-install.scss +++ /dev/null @@ -1 +0,0 @@ -@import "../../scss/vars"; diff --git a/client/src/app/install/install-step-6-completed.js b/client/src/app/install/install-step-6-completed.js index d2733f8f..e610a83d 100644 --- a/client/src/app/install/install-step-6-completed.js +++ b/client/src/app/install/install-step-6-completed.js @@ -1,11 +1,25 @@ import React from 'react'; +import {browserHistory} from 'react-router'; + +import i18n from 'lib-app/i18n'; +import API from 'lib-app/api-call'; + +import Header from 'core-components/header'; +import Message from 'core-components/message'; class InstallStep6Completed extends React.Component { + componentDidMount() { + setTimeout(() => browserHistory.push('/admin'), 5000); + } + render() { return ( -
    - INSTALLATION +
    +
    + + {i18n('INSTALLATION_COMPLETED_DESCRIPTION')} +
    ); } diff --git a/client/src/core-components/form-field.js b/client/src/core-components/form-field.js index 376c11db..ee5a1441 100644 --- a/client/src/core-components/form-field.js +++ b/client/src/core-components/form-field.js @@ -29,7 +29,8 @@ class FormField extends React.Component { }; static defaultProps = { - field: 'input' + field: 'input', + fieldProps: {} }; static getDefaultValue(field) { @@ -120,7 +121,7 @@ class FormField extends React.Component { getFieldProps() { let props = _.extend({}, this.props.fieldProps, { - disabled: this.context.loading, + disabled: this.isDisabled(), errored: !!this.props.error, name: this.props.name, placeholder: this.props.placeholder, @@ -171,6 +172,12 @@ class FormField extends React.Component { this.props.onChange(event) } } + + isDisabled() { + const fieldProps = this.props.fieldProps; + + return (fieldProps.disabled === undefined) ? this.context.loading : fieldProps.disabled; + } focus() { if (this.refs.nativeField) { diff --git a/client/src/data/fixtures/system-fixtures.js b/client/src/data/fixtures/system-fixtures.js index 9a7ebe26..d160a64a 100644 --- a/client/src/data/fixtures/system-fixtures.js +++ b/client/src/data/fixtures/system-fixtures.js @@ -56,6 +56,37 @@ module.exports = [ } } }, + { + path: '/system/init-settings', + time: 50, + response: function() { + return { + status: 'success', + data: {} + }; + } + }, + { + path: '/system/init-database', + time: 50, + response: function() { + return { + status: 'success', + message: 'ERROR_SERVER', + data: {} + }; + } + }, + { + path: '/system/init-admin', + time: 50, + response: function() { + return { + status: 'success', + data: {} + }; + } + }, { path: '/system/edit-settings', time: 50, @@ -90,8 +121,8 @@ module.exports = [ }, configFile: { name: 'File: /api/config.php', - value: 'No writable', - ok: false + value: 'Writable', + ok: true } } }; diff --git a/client/src/data/languages/en.js b/client/src/data/languages/en.js index 80d15eda..b6b8ee57 100644 --- a/client/src/data/languages/en.js +++ b/client/src/data/languages/en.js @@ -210,9 +210,8 @@ export default { 'ACTIVITY_UN_BAN_USER': 'banned user', 'SERVER_REQUIREMENTS': 'Server requirements', - 'SETTINGS_SETUP': 'Settings setup', + 'DATABASE_CONFIGURATION': 'Database configuration', 'ADMIN_SETUP': 'Admin setup', - 'INSTALL': 'Install', 'COMPLETED': 'Completed', 'INSTALL_HEADER_TITLE': 'OpenSupports Installation Wizard', 'INSTALL_HEADER_DESCRIPTION': 'This wizard will help you to configure and install OpenSupports on your website', @@ -220,14 +219,26 @@ export default { 'REQUIREMENT': 'Requirement', 'VALUE': 'Value', 'REFRESH': 'Refresh', + 'USER_SYSTEM': 'User System', + 'PREVIOUS': 'Previous', + 'DATABASE_HOST': 'MySQL server', + 'DATABASE_NAME': 'MySQL database name', + 'DATABASE_USER': 'MySQL user', + 'DATABASE_PASSWORD': 'MySQL password', + 'ADMIN_NAME': 'Admin account name', + 'ADMIN_EMAIL': 'Admin account email', + 'ADMIN_PASSWORD': 'Admin account password', + 'ADMIN_PASSWORD_DESCRIPTION': 'Please remember this password. It is needed for accessing the admin panel. You can change it later.', + 'INSTALLATION_COMPLETED': 'Installation completed.', + 'INSTALLATION_COMPLETED_DESCRIPTION': 'The installation of OpenSupports is completed. Redirecting to admin panel...', 'STEP_TITLE': 'Step {current} of {total} - {title}', 'STEP_1_DESCRIPTION': 'Select your preferred language for the installation wizard.', 'STEP_2_DESCRIPTION': 'Here are listed the requirements for running OpenSupports. Please make sure that all requirements are satisfied.', - 'STEP_3_DESCRIPTION': 'Select your preferred language for the installation wizard.', - 'STEP_4_DESCRIPTION': 'Select your preferred language for the installation wizard.', - 'STEP_5_DESCRIPTION': 'Select your preferred language for the installation wizard.', - 'STEP_6_DESCRIPTION': 'Select your preferred language for the installation wizard.', + 'STEP_3_DESCRIPTION': 'Please fill the MySQL database configuration.', + 'STEP_4_DESCRIPTION': 'Please select your user system preferences.', + 'STEP_5_DESCRIPTION': 'Please configure the administrator account.', + 'STEP_6_DESCRIPTION': 'Installation is completed.', //VIEW DESCRIPTIONS 'CREATE_TICKET_DESCRIPTION': 'This is a form for creating tickets. Fill the form and send us your issues/doubts/suggestions. Our support system will answer it as soon as possible.', diff --git a/client/src/index.html b/client/src/index.html index a398c80c..04d3e7f6 100644 --- a/client/src/index.html +++ b/client/src/index.html @@ -9,6 +9,7 @@ OS4 + From a183d1ed79726573965893e56d0f317d6015d736 Mon Sep 17 00:00:00 2001 From: ivan Date: Sat, 11 Mar 2017 03:52:43 -0300 Subject: [PATCH 5/5] Ivan - Add redirection to completed if install is already done [skip ci] --- client/src/app/install/install-layout.js | 15 ++++++++++++++- 1 file changed, 14 insertions(+), 1 deletion(-) diff --git a/client/src/app/install/install-layout.js b/client/src/app/install/install-layout.js index b74b1cdd..c7fb2601 100644 --- a/client/src/app/install/install-layout.js +++ b/client/src/app/install/install-layout.js @@ -1,8 +1,10 @@ import React from 'react'; import classNames from 'classnames'; import _ from 'lodash'; +import {browserHistory} from 'react-router'; import i18n from 'lib-app/i18n'; +import API from 'lib-app/api-call'; import Widget from 'core-components/widget'; import Icon from 'core-components/icon'; @@ -18,6 +20,17 @@ const steps = [ class InstallLayout extends React.Component { + componentDidMount() { + API.call({ + path: '/system/installation-step', + data: {} + }).then((result) => { + if(result.data != this.getCurrentStep()) { + browserHistory.push('/install/step-' + (result.data + 1)); + } + }); + } + render() { return ( @@ -64,7 +77,7 @@ class InstallLayout extends React.Component { } return ( -
  • +
  • {index+1}. {i18n(key)}