diff --git a/.env b/.env index 3ba890f..81b9303 100644 --- a/.env +++ b/.env @@ -3,6 +3,7 @@ REACT_APP_BOARD=sensebox-mcu REACT_APP_BLOCKLY_API=https://api.blockly.sensebox.de REACT_APP_MYBADGES=https://mybadges.org +REACT_APP_MYBADGES_API=https://mybadges.org/api/v1 # in days REACT_APP_SHARE_LINK_EXPIRES=30 diff --git a/package.json b/package.json index 60a336a..4a234e7 100644 --- a/package.json +++ b/package.json @@ -32,7 +32,7 @@ }, "scripts": { "start": "react-scripts start", - "dev": "set \"REACT_APP_BLOCKLY_API=http://localhost:8080\" && npm start", + "dev": "set \"REACT_APP_BLOCKLY_API=http://localhost:8080\" && set \"REACT_APP_MYBADGES_API=http://localhost:3001/api/v1\"&& npm start", "build": "react-scripts build", "test": "react-scripts test", "eject": "react-scripts eject" diff --git a/src/App.js b/src/App.js index 1eccac4..ef9ca04 100644 --- a/src/App.js +++ b/src/App.js @@ -11,10 +11,7 @@ import './App.css'; import { ThemeProvider, createMuiTheme } from '@material-ui/core/styles'; -import Navbar from './components/Navbar'; -import Footer from './components/Footer'; -import Routes from './components/Route/Routes'; -import Cookies from './components/Cookies'; +import Content from './components/Content'; const theme = createMuiTheme({ palette: { @@ -43,12 +40,7 @@ class App extends Component { -
- - - -
-
+
diff --git a/src/actions/authActions.js b/src/actions/authActions.js index 7cdd1c0..77e3f4b 100644 --- a/src/actions/authActions.js +++ b/src/actions/authActions.js @@ -1,8 +1,8 @@ -import { MYBADGES_CONNECT, MYBADGES_DISCONNECT, USER_LOADED, USER_LOADING, AUTH_ERROR, LOGIN_SUCCESS, LOGIN_FAIL, LOGOUT_SUCCESS, LOGOUT_FAIL, REFRESH_TOKEN_SUCCESS } from '../actions/types'; +import { MYBADGES_CONNECT, MYBADGES_DISCONNECT, GET_STATUS, USER_LOADED, USER_LOADING, AUTH_ERROR, LOGIN_SUCCESS, LOGIN_FAIL, LOGOUT_SUCCESS, LOGOUT_FAIL, REFRESH_TOKEN_SUCCESS } from '../actions/types'; import axios from 'axios'; -import { returnErrors, returnSuccess } from './messageActions' - +import { returnErrors, returnSuccess } from './messageActions'; +import { setLanguage } from './generalActions'; // Check token & load user export const loadUser = () => (dispatch) => { @@ -12,6 +12,11 @@ export const loadUser = () => (dispatch) => { }); const config = { success: res => { + dispatch({ + type: GET_STATUS, + payload: res.data.user.status + }); + dispatch(setLanguage(res.data.user.language)); dispatch({ type: USER_LOADED, payload: res.data.user @@ -21,6 +26,14 @@ export const loadUser = () => (dispatch) => { if(err.response){ dispatch(returnErrors(err.response.data.message, err.response.status)); } + var status = []; + if (window.localStorage.getItem('status')) { + status = JSON.parse(window.localStorage.getItem('status')); + } + dispatch({ + type: GET_STATUS, + payload: status + }); dispatch({ type: AUTH_ERROR }); @@ -41,6 +54,9 @@ const timeToLogout = 14.9*60*1000; // nearly 15 minutes corresponding to the API // Login user export const login = ({ email, password }) => (dispatch) => { + dispatch({ + type: USER_LOADING + }); // Headers const config = { headers: { @@ -57,10 +73,15 @@ export const login = ({ email, password }) => (dispatch) => { timeToLogout ); logoutTimerId = logoutTimer(); + dispatch(setLanguage(res.data.user.language)); dispatch({ type: LOGIN_SUCCESS, payload: res.data }); + dispatch({ + type: GET_STATUS, + payload: res.data.user.status + }); dispatch(returnSuccess(res.data.message, res.status, 'LOGIN_SUCCESS')); }) .catch(err => { @@ -68,57 +89,73 @@ export const login = ({ email, password }) => (dispatch) => { dispatch({ type: LOGIN_FAIL }); + var status = []; + if (window.localStorage.getItem('status')) { + status = JSON.parse(window.localStorage.getItem('status')); + } + dispatch({ + type: GET_STATUS, + payload: status + }); }); }; // Connect to MyBadges-Account export const connectMyBadges = ({ username, password }) => (dispatch, getState) => { - // Headers const config = { - headers: { - 'Content-Type': 'application/json' + success: res => { + var user = getState().auth.user; + user.badge = res.data.account; + user.badges = res.data.badges; + dispatch({ + type: MYBADGES_CONNECT, + payload: user + }); + dispatch(returnSuccess(res.data.message, res.status, 'MYBADGES_CONNECT_SUCCESS')); + }, + error: err => { + dispatch(returnErrors(err.response.data.message, err.response.status, 'MYBADGES_CONNECT_FAIL')); } }; // Request Body const body = JSON.stringify({ username, password }); axios.post(`${process.env.REACT_APP_BLOCKLY_API}/user/badge`, body, config) .then(res => { - var user = getState().auth.user; - user.badge = res.data.account; - user.badges = res.data.badges; - dispatch({ - type: MYBADGES_CONNECT, - payload: user - }); - dispatch(returnSuccess(res.data.message, res.status, 'MYBADGES_CONNECT_SUCCESS')); + res.config.success(res); }) .catch(err => { - dispatch(returnErrors(err.response.data.message, err.response.status, 'MYBADGES_CONNECT_FAIL')); + if(err.response && err.response.status !== 401){ + err.config.error(err); + } }); }; // Disconnect MyBadges-Account export const disconnectMyBadges = () => (dispatch, getState) => { - // Headers const config = { - headers: { - 'Content-Type': 'application/json' + success: res => { + var user = getState().auth.user; + user.badge = null; + user.badges = null; + dispatch({ + type: MYBADGES_DISCONNECT, + payload: user + }); + dispatch(returnSuccess(res.data.message, res.status, 'MYBADGES_DISCONNECT_SUCCESS')); + }, + error: err => { + dispatch(returnErrors(err.response.data.message, err.response.status, 'MYBADGES_DISCONNECT_FAIL')); } }; - axios.put(`${process.env.REACT_APP_BLOCKLY_API}/user/badge`, config) + axios.put(`${process.env.REACT_APP_BLOCKLY_API}/user/badge`, {}, config) .then(res => { - var user = getState().auth.user; - user.badge = null; - user.badges = null; - dispatch({ - type: MYBADGES_DISCONNECT, - payload: user - }); - dispatch(returnSuccess(res.data.message, res.status, 'MYBADGES_DISCONNECT_SUCCESS')); + res.config.success(res); }) .catch(err => { - dispatch(returnErrors(err.response.data.message, err.response.status, 'MYBADGES_DISCONNECT_FAIL')); + if(err.response && err.response.status !== 401){ + err.config.error(err); + } }); }; @@ -130,6 +167,22 @@ export const logout = () => (dispatch) => { dispatch({ type: LOGOUT_SUCCESS }); + var status = []; + if (window.localStorage.getItem('status')) { + status = JSON.parse(window.localStorage.getItem('status')); + } + dispatch({ + type: GET_STATUS, + payload: status + }); + var locale = 'en_US'; + if (window.localStorage.getItem('locale')) { + locale = window.localStorage.getItem('locale'); + } + else if (navigator.language === 'de-DE'){ + locale = 'de_DE'; + } + dispatch(setLanguage(locale)); dispatch(returnSuccess(res.data.message, res.status, 'LOGOUT_SUCCESS')); clearTimeout(logoutTimerId); }, @@ -138,6 +191,14 @@ export const logout = () => (dispatch) => { dispatch({ type: LOGOUT_FAIL }); + var status = []; + if (window.localStorage.getItem('status')) { + status = JSON.parse(window.localStorage.getItem('status')); + } + dispatch({ + type: GET_STATUS, + payload: status + }); clearTimeout(logoutTimerId); } }; @@ -146,7 +207,7 @@ export const logout = () => (dispatch) => { res.config.success(res); }) .catch(err => { - if(err.response.status !== 401){ + if(err.response && err.response.status !== 401){ err.config.error(err); } }); diff --git a/src/actions/generalActions.js b/src/actions/generalActions.js index 6585a9a..754f438 100644 --- a/src/actions/generalActions.js +++ b/src/actions/generalActions.js @@ -1,4 +1,4 @@ -import { VISIT } from './types'; +import { VISIT, LANGUAGE, RENDERER, STATISTICS } from './types'; export const visitPage = () => (dispatch) => { @@ -6,3 +6,27 @@ export const visitPage = () => (dispatch) => { type: VISIT }); }; + +export const setLanguage = (language) => (dispatch, getState) => { + if(!getState().auth.progress && !getState().auth.isAuthenticated){ + window.localStorage.setItem('locale', language); + } + dispatch({ + type: LANGUAGE, + payload: language + }); +}; + +export const setRenderer = (renderer) => (dispatch) => { + dispatch({ + type: RENDERER, + payload: renderer + }); +}; + +export const setStatistics = (showStatistics) => (dispatch) => { + dispatch({ + type: STATISTICS, + payload: showStatistics + }); +}; diff --git a/src/actions/projectActions.js b/src/actions/projectActions.js index 8ab01cf..6caef34 100644 --- a/src/actions/projectActions.js +++ b/src/actions/projectActions.js @@ -1,7 +1,6 @@ import { PROJECT_PROGRESS, GET_PROJECT, GET_PROJECTS, PROJECT_TYPE, PROJECT_DESCRIPTION } from './types'; import axios from 'axios'; -import { workspaceName } from './workspaceActions'; import { returnErrors, returnSuccess } from './messageActions'; export const setType = (type) => (dispatch) => { @@ -19,13 +18,13 @@ export const setDescription = (description) => (dispatch) => { }; export const getProject = (type, id) => (dispatch) => { - dispatch({type: PROJECT_PROGRESS}); + dispatch({ type: PROJECT_PROGRESS }); dispatch(setType(type)); - axios.get(`${process.env.REACT_APP_BLOCKLY_API}/${type}/${id}`) - .then(res => { + const config = { + success: res => { var data = type === 'share' ? 'content' : type; var project = res.data[data]; - if(project){ + if (project) { dispatch({ type: GET_PROJECT, payload: project @@ -34,40 +33,56 @@ export const getProject = (type, id) => (dispatch) => { type: PROJECT_DESCRIPTION, payload: project.description }); - dispatch({type: PROJECT_PROGRESS}); + dispatch({ type: PROJECT_PROGRESS }); dispatch(returnSuccess(res.data.message, res.status, 'GET_PROJECT_SUCCESS')); } - else{ - dispatch({type: PROJECT_PROGRESS}); + else { + dispatch({ type: PROJECT_PROGRESS }); dispatch(returnErrors(res.data.message, res.status, 'PROJECT_EMPTY')); } - }) - .catch(err => { + }, + error: err => { if(err.response){ dispatch(returnErrors(err.response.data.message, err.response.status, 'GET_PROJECT_FAIL')); } dispatch({type: PROJECT_PROGRESS}); + } + }; + axios.get(`${process.env.REACT_APP_BLOCKLY_API}/${type}/${id}`, config) + .then(res => { + res.config.success(res); + }) + .catch(err => { + err.config.error(err); }); }; export const getProjects = (type) => (dispatch) => { dispatch({type: PROJECT_PROGRESS}); - axios.get(`${process.env.REACT_APP_BLOCKLY_API}/${type}`) - .then(res => { + const config = { + success: res => { var data = type === 'project' ? 'projects' : 'galleries'; var projects = res.data[data]; dispatch({ type: GET_PROJECTS, payload: projects }); - dispatch({type: PROJECT_PROGRESS}); + dispatch({ type: PROJECT_PROGRESS }); dispatch(returnSuccess(res.data.message, res.status)); - }) - .catch(err => { + }, + error: err => { if(err.response){ dispatch(returnErrors(err.response.data.message, err.response.status, 'GET_PROJECTS_FAIL')); } dispatch({type: PROJECT_PROGRESS}); + } + }; + axios.get(`${process.env.REACT_APP_BLOCKLY_API}/${type}`, config) + .then(res => { + res.config.success(res); + }) + .catch(err => { + err.config.error(err); }); }; @@ -78,11 +93,11 @@ export const updateProject = (type, id) => (dispatch, getState) => { title: workspace.name }; var project = getState().project; - if(type==='gallery'){ + if (type === 'gallery') { body.description = project.description; } - axios.put(`${process.env.REACT_APP_BLOCKLY_API}/${type}/${id}`, body) - .then(res => { + const config = { + success: res => { var project = res.data[type]; var projects = getState().project.projects; var index = projects.findIndex(res => res._id === project._id); @@ -91,13 +106,13 @@ export const updateProject = (type, id) => (dispatch, getState) => { type: GET_PROJECTS, payload: projects }); - if(type === 'project'){ + if (type === 'project') { dispatch(returnSuccess(res.data.message, res.status, 'PROJECT_UPDATE_SUCCESS')); } else { dispatch(returnSuccess(res.data.message, res.status, 'GALLERY_UPDATE_SUCCESS')); } - }) - .catch(err => { + }, + error: err => { if(err.response){ if(type === 'project'){ dispatch(returnErrors(err.response.data.message, err.response.status, 'PROJECT_UPDATE_FAIL')); @@ -105,13 +120,21 @@ export const updateProject = (type, id) => (dispatch, getState) => { dispatch(returnErrors(err.response.data.message, err.response.status, 'GALLERY_UPDATE_FAIL')); } } + } + }; + axios.put(`${process.env.REACT_APP_BLOCKLY_API}/${type}/${id}`, body, config) + .then(res => { + res.config.success(res); + }) + .catch(err => { + err.config.error(err); }); }; export const deleteProject = (type, id) => (dispatch, getState) => { var project = getState().project; - axios.delete(`${process.env.REACT_APP_BLOCKLY_API}/${type}/${id}`) - .then(res => { + const config = { + success: res => { var projects = getState().project.projects; var index = projects.findIndex(res => res._id === id); projects.splice(index, 1) @@ -119,15 +142,23 @@ export const deleteProject = (type, id) => (dispatch, getState) => { type: GET_PROJECTS, payload: projects }); - if(type === 'project'){ + if (type === 'project') { dispatch(returnSuccess(res.data.message, res.status, 'PROJECT_DELETE_SUCCESS')); } else { dispatch(returnSuccess(res.data.message, res.status, 'GALLERY_DELETE_SUCCESS')); } + }, + error: err => { + dispatch(returnErrors(err.response.data.message, err.response.status, 'PROJECT_DELETE_FAIL')); + } + }; + axios.delete(`${process.env.REACT_APP_BLOCKLY_API}/${type}/${id}`, config) + .then(res => { + res.config.success(res); }) .catch(err => { - if(err.response){ - dispatch(returnErrors(err.response.data.message, err.response.status, 'PROJECT_DELETE_FAIL')); + if(err.response && err.response.status !== 401){ + err.config.error(err); } }); }; @@ -137,7 +168,7 @@ export const shareProject = (title, type, id) => (dispatch, getState) => { var body = { title: title }; - if(type === 'project'){ + if (type === 'project') { body.projectId = id; } else { body.xml = getState().workspace.code.xml; @@ -145,7 +176,7 @@ export const shareProject = (title, type, id) => (dispatch, getState) => { axios.post(`${process.env.REACT_APP_BLOCKLY_API}/share`, body) .then(res => { var shareContent = res.data.content; - if(body.projectId){ + if (body.projectId) { var projects = getState().project.projects; var index = projects.findIndex(res => res._id === id); projects[index].shared = shareContent.expiresAt; @@ -157,7 +188,7 @@ export const shareProject = (title, type, id) => (dispatch, getState) => { dispatch(returnSuccess(res.data.message, shareContent._id, 'SHARE_SUCCESS')); }) .catch(err => { - if(err.response){ + if (err.response) { dispatch(returnErrors(err.response.data.message, err.response.status, 'SHARE_FAIL')); } }); diff --git a/src/actions/tutorialActions.js b/src/actions/tutorialActions.js index 8983da9..7a9801a 100644 --- a/src/actions/tutorialActions.js +++ b/src/actions/tutorialActions.js @@ -1,10 +1,14 @@ -import { MYBADGES_DISCONNECT, TUTORIAL_PROGRESS, GET_TUTORIAL, GET_TUTORIALS, TUTORIAL_SUCCESS, TUTORIAL_ERROR, TUTORIAL_CHANGE, TUTORIAL_XML, TUTORIAL_ID, TUTORIAL_STEP } from './types'; +import { MYBADGES_DISCONNECT, TUTORIAL_PROGRESS, GET_TUTORIAL, GET_TUTORIALS, TUTORIAL_SUCCESS, TUTORIAL_ERROR, TUTORIAL_CHANGE, TUTORIAL_XML, TUTORIAL_STEP } from './types'; import axios from 'axios'; import { returnErrors, returnSuccess } from './messageActions'; -export const getTutorial = (id) => (dispatch, getState) => { +export const tutorialProgress = () => (dispatch) => { dispatch({type: TUTORIAL_PROGRESS}); +}; + + +export const getTutorial = (id) => (dispatch, getState) => { axios.get(`${process.env.REACT_APP_BLOCKLY_API}/tutorial/${id}`) .then(res => { var tutorial = res.data.tutorial; @@ -13,24 +17,24 @@ export const getTutorial = (id) => (dispatch, getState) => { type: TUTORIAL_SUCCESS, payload: status }); - dispatch({type: TUTORIAL_PROGRESS}); + dispatch(updateStatus(status)); dispatch({ type: GET_TUTORIAL, payload: tutorial }); + dispatch({type: TUTORIAL_PROGRESS}); dispatch(returnSuccess(res.data.message, res.status)); }); }) .catch(err => { - if(err.response){ + if (err.response) { dispatch(returnErrors(err.response.data.message, err.response.status, 'GET_TUTORIAL_FAIL')); } - dispatch({type: TUTORIAL_PROGRESS}); + dispatch({ type: TUTORIAL_PROGRESS }); }); }; export const getTutorials = () => (dispatch, getState) => { - dispatch({type: TUTORIAL_PROGRESS}); axios.get(`${process.env.REACT_APP_BLOCKLY_API}/tutorial`) .then(res => { var tutorials = res.data.tutorials; @@ -40,25 +44,27 @@ export const getTutorials = () => (dispatch, getState) => { type: TUTORIAL_SUCCESS, payload: status }); + console.log('zwei'); + dispatch(updateStatus(status)); dispatch({ type: GET_TUTORIALS, payload: tutorials }); - dispatch({type: TUTORIAL_PROGRESS}); + dispatch({ type: TUTORIAL_PROGRESS }); dispatch(returnSuccess(res.data.message, res.status)); }); }) .catch(err => { - if(err.response){ + if (err.response) { dispatch(returnErrors(err.response.data.message, err.response.status, 'GET_TUTORIALS_FAIL')); } - dispatch({type: TUTORIAL_PROGRESS}); + dispatch({ type: TUTORIAL_PROGRESS }); }); }; export const assigneBadge = (id) => (dispatch, getState) => { - axios.put(`${process.env.REACT_APP_BLOCKLY_API}/user/badge/${id}`) - .then(res => { + const config = { + success: res => { var badge = res.data.badge; var user = getState().auth.user; user.badges.push(badge._id); @@ -67,19 +73,45 @@ export const assigneBadge = (id) => (dispatch, getState) => { payload: user }); dispatch(returnSuccess(badge, res.status, 'ASSIGNE_BADGE_SUCCESS')); + }, + error: err => { + dispatch(returnErrors(err.response.data.message, err.response.status, 'ASSIGNE_BADGE_FAIL')); + } + }; + axios.put(`${process.env.REACT_APP_BLOCKLY_API}/user/badge/${id}`, {}, config) + .then(res => { + res.config.success(res); }) .catch(err => { - if(err.response){ - dispatch(returnErrors(err.response.data.message, err.response.status, 'ASSIGNE_BADGE_FAIL')); + if(err.response && err.response.status !== 401){ + err.config.error(err); } }); }; +export const updateStatus = (status) => (dispatch, getState) => { + if(getState().auth.isAuthenticated){ + // update user account in database - sync with redux store + axios.put(`${process.env.REACT_APP_BLOCKLY_API}/user/status`, {status: status}) + .then(res => { + // dispatch(returnSuccess(badge, res.status, 'UPDATE_STATUS_SUCCESS')); + }) + .catch(err => { + if(err.response){ + // dispatch(returnErrors(err.response.data.message, err.response.status, 'UPDATE_STATUS_FAIL')); + } + }); + } else { + // update locale storage - sync with redux store + window.localStorage.setItem('status', JSON.stringify(status)); + } +}; + export const deleteTutorial = (id) => (dispatch, getState) => { var tutorial = getState().tutorial; var id = getState().builder.id; - axios.delete(`${process.env.REACT_APP_BLOCKLY_API}/tutorial/${id}`) - .then(res => { + const config = { + success: res => { var tutorials = tutorial.tutorials; var index = tutorials.findIndex(res => res._id === id); tutorials.splice(index, 1) @@ -88,10 +120,18 @@ export const deleteTutorial = (id) => (dispatch, getState) => { payload: tutorials }); dispatch(returnSuccess(res.data.message, res.status, 'TUTORIAL_DELETE_SUCCESS')); + }, + error: err => { + dispatch(returnErrors(err.response.data.message, err.response.status, 'TUTORIAL_DELETE_FAIL')); + } + }; + axios.delete(`${process.env.REACT_APP_BLOCKLY_API}/tutorial/${id}`, config) + .then(res => { + res.config.success(res); }) .catch(err => { - if(err.response){ - dispatch(returnErrors(err.response.data.message, err.response.status, 'TUTORIAL_DELETE_FAIL')); + if(err.response && err.response.status !== 401){ + err.config.error(err); } }); }; @@ -127,8 +167,10 @@ export const tutorialCheck = (status, step) => (dispatch, getState) => { type: status === 'success' ? TUTORIAL_SUCCESS : TUTORIAL_ERROR, payload: tutorialsStatus }); + console.log('drei'); + dispatch(updateStatus(tutorialsStatus)); dispatch(tutorialChange()); - dispatch(returnSuccess('','','TUTORIAL_CHECK_SUCCESS')); + dispatch(returnSuccess('', '', 'TUTORIAL_CHECK_SUCCESS')); }; export const storeTutorialXml = (code) => (dispatch, getState) => { @@ -149,6 +191,7 @@ export const storeTutorialXml = (code) => (dispatch, getState) => { type: TUTORIAL_XML, payload: tutorialsStatus }); + dispatch(updateStatus(tutorialsStatus)); } } }; @@ -161,9 +204,9 @@ export const tutorialStep = (step) => (dispatch) => { }; -const existingTutorials = (tutorials, status) => new Promise(function(resolve, reject){ +const existingTutorials = (tutorials, status) => new Promise(function (resolve, reject) { var newstatus; - new Promise(function(resolve, reject){ + new Promise(function (resolve, reject) { var existingTutorialIds = tutorials.map((tutorial, i) => { existingTutorial(tutorial, status).then(status => { newstatus = status; diff --git a/src/actions/tutorialBuilderActions.js b/src/actions/tutorialBuilderActions.js index 356e8fa..25f27b0 100644 --- a/src/actions/tutorialBuilderActions.js +++ b/src/actions/tutorialBuilderActions.js @@ -191,6 +191,9 @@ export const setSubmitError = () => (dispatch, getState) => { if (builder.title === '') { dispatch(setError(undefined, 'title')); } + if (builder.title === null) { + dispatch(setError(undefined, 'badge')); + } var type = builder.steps.map((step, i) => { // media and xml are directly checked for errors in their components and // therefore do not have to be checked again @@ -230,7 +233,7 @@ export const setSubmitError = () => (dispatch, getState) => { export const checkError = () => (dispatch, getState) => { dispatch(setSubmitError()); var error = getState().builder.error; - if (error.id || error.title || error.type) { + if (error.id || error.title || error.badge ||error.type) { return true; } for (var i = 0; i < error.steps.length; i++) { @@ -251,10 +254,9 @@ export const progress = (inProgress) => (dispatch) => { export const resetTutorial = () => (dispatch, getState) => { dispatch(jsonString('')); dispatch(tutorialTitle('')); - dispatch(tutorialBadge('')); + dispatch(tutorialBadge(undefined)); var steps = [ { - id: 1, type: 'instruction', headline: '', text: '', @@ -282,7 +284,7 @@ export const readJSON = (json) => (dispatch, getState) => { // accept only valid attributes var steps = json.steps.map((step, i) => { var object = { - // id: step.id, + _id: step._id, type: step.type, headline: step.headline, text: step.text diff --git a/src/actions/types.js b/src/actions/types.js index b3bff75..a5ab516 100644 --- a/src/actions/types.js +++ b/src/actions/types.js @@ -23,6 +23,7 @@ export const NAME = 'NAME'; export const TUTORIAL_PROGRESS = 'TUTORIAL_PROGRESS'; export const GET_TUTORIAL = 'GET_TUTORIAL'; export const GET_TUTORIALS = 'GET_TUTORIALS'; +export const GET_STATUS = 'GET_STATUS'; export const TUTORIAL_SUCCESS = 'TUTORIAL_SUCCESS'; export const TUTORIAL_ERROR = 'TUTORIAL_ERROR'; export const TUTORIAL_CHANGE = 'TUTORIAL_CHANGE'; @@ -46,6 +47,9 @@ export const PROGRESS = 'PROGRESS'; export const VISIT = 'VISIT'; +export const LANGUAGE = 'LANGUAGE'; +export const RENDERER = 'RENDERER'; +export const STATISTICS = 'STATISTICS'; // messages export const GET_ERRORS = 'GET_ERRORS'; diff --git a/src/components/Blockly/BlocklyWindow.js b/src/components/Blockly/BlocklyWindow.js index f3c680d..d77d620 100644 --- a/src/components/Blockly/BlocklyWindow.js +++ b/src/components/Blockly/BlocklyWindow.js @@ -2,15 +2,15 @@ import React, { Component } from 'react'; import PropTypes from 'prop-types'; import { connect } from 'react-redux'; import { onChangeWorkspace, clearStats } from '../../actions/workspaceActions'; -import { De } from './msg/de'; -import { En } from './msg/en'; + import BlocklyComponent from './BlocklyComponent'; import BlocklySvg from './BlocklySvg'; + import * as Blockly from 'blockly/core'; import './blocks/index'; import './generator/index'; -import { initialXml } from './initialXml.js'; +import { initialXml } from './initialXml.js'; class BlocklyWindow extends Component { @@ -18,26 +18,9 @@ class BlocklyWindow extends Component { constructor(props) { super(props); this.simpleWorkspace = React.createRef(); - var locale = window.localStorage.getItem('locale'); - this.state = { - renderer: window.localStorage.getItem('renderer'), - }; - if (locale === null) { - if (navigator.language === 'de-DE') { - locale = 'de'; - } else { - locale = 'en'; - } - } - if (locale === 'de') { - Blockly.setLocale(De); - } else if (locale === 'en') { - Blockly.setLocale(En); - } } componentDidMount() { - const workspace = Blockly.getMainWorkspace(); this.props.onChangeWorkspace({}); this.props.clearStats(); @@ -62,6 +45,15 @@ class BlocklyWindow extends Component { if (!xml) xml = initialXml; Blockly.Xml.domToWorkspace(Blockly.Xml.textToDom(xml), workspace); } + if(props.language !== this.props.language){ + // change language + if (!xml) xml = initialXml; + var xmlDom = Blockly.Xml.textToDom(xml); + Blockly.Xml.clearWorkspaceAndLoadFromXml(xmlDom, workspace); + // var toolbox = workspace.getToolbox(); + // console.log(toolbox); + // workspace.updateToolbox(toolbox.toolboxDef_); + } Blockly.svgResize(workspace); } @@ -72,7 +64,7 @@ class BlocklyWindow extends Component { style={this.props.svg ? { height: 0 } : this.props.blocklyCSS} readOnly={this.props.readOnly !== undefined ? this.props.readOnly : false} trashcan={this.props.trashcan !== undefined ? this.props.trashcan : true} - renderer={this.state.renderer} + renderer={this.props.renderer} zoom={{ // https://developers.google.com/blockly/guides/configure/web/zoom controls: this.props.zoomControls !== undefined ? this.props.zoomControls : true, wheel: false, @@ -106,8 +98,14 @@ class BlocklyWindow extends Component { BlocklyWindow.propTypes = { onChangeWorkspace: PropTypes.func.isRequired, - clearStats: PropTypes.func.isRequired + clearStats: PropTypes.func.isRequired, + renderer: PropTypes.string.isRequired, + language: PropTypes.string.isRequired }; +const mapStateToProps = state => ({ + renderer: state.general.renderer, + language: state.general.language +}); -export default connect(null, { onChangeWorkspace, clearStats })(BlocklyWindow); +export default connect(mapStateToProps, { onChangeWorkspace, clearStats })(BlocklyWindow); diff --git a/src/components/Blockly/msg/de.js b/src/components/Blockly/msg/de.js index 2dec197..7974c65 100644 --- a/src/components/Blockly/msg/de.js +++ b/src/components/Blockly/msg/de.js @@ -800,4 +800,228 @@ Blockly.Msg.senseBox_mqtt_password = "Passwort"; Blockly.Msg.sensebox_mqtt_subscribe = "Subscribe to Feed" Blockly.Msg.senseBox_mqtt_publish = "Sende an Feed/Topic"; + + +/** + * Typed Variable Modal + * + */ + + +Blockly.Msg.TYPED_VAR_MODAL_CONFIRM_BUTTON = "Ok"; +Blockly.Msg.TYPED_VAR_MODAL_VARIABLE_NAME_LABEL = "Variablen Name: "; +Blockly.Msg.TYPED_VAR_MODAL_TYPES_LABEL = "Variable Typen"; +Blockly.Msg.TYPED_VAR_MODAL_CANCEL_BUTTON = "Abbrechen"; +Blockly.Msg.TYPED_VAR_MODAL_TITLE = "Erstelle Variable"; +Blockly.Msg.TYPED_VAR_MODAL_INVALID_NAME = "Der Name ist ungültig, bitte versuche einen anderen." + +/** + * Add Translation for Blocks above + * --------------------------------------------------------------- + * Add Translation for the UI below + */ + +/** + * Toolbox + */ +Blockly.Msg.toolbox_sensors = "Sensoren"; +Blockly.Msg.toolbox_logic = "Logik"; +Blockly.Msg.toolbox_loops = "Schleifen"; +Blockly.Msg.toolbox_math = "Mathematik"; +Blockly.Msg.toolbox_io = "Eingang/Ausgang"; +Blockly.Msg.toolbox_time = "Zeit"; +Blockly.Msg.toolbox_functions = "Funktionen"; +Blockly.Msg.toolbox_variables = "Variablen"; + +/** + * Tooltips + * + */ + +Blockly.Msg.tooltip_compile_code = "Code kompilieren" +Blockly.Msg.tooltip_save_blocks = "Blöcke speichern"; +Blockly.Msg.tooltip_open_blocks = "Blöcke öffnen"; +Blockly.Msg.tooltip_screenshot = "Screenshot erstellen"; +Blockly.Msg.tooltip_clear_workspace = "Workspace zurücksetzen"; +Blockly.Msg.tooltip_share_blocks = "Blöcke teilen"; +Blockly.Msg.tooltip_show_code = "Code anzeigen"; +Blockly.Msg.tooltip_hide_code = "Code ausblenden" +Blockly.Msg.tooltip_delete_project = "Projekt löschen" +Blockly.Msg.tooltip_project_name = "Name des Projektes" +Blockly.Msg.tooltip_download_project = "Projekt herunterladen" +Blockly.Msg.tooltip_open_project = "Projekt öffnen" +Blockly.Msg.tooltip_update_project = "Projekt aktualisieren" +Blockly.Msg.tooltip_save_project = "Projekt speichern" +Blockly.Msg.tooltip_create_project = "Projekt erstellen" +Blockly.Msg.tooltip_share_project = "Projekt teilen" +Blockly.Msg.tooltip_reset_workspace = "Workspace zurücksetzen" +Blockly.Msg.tooltip_copy_link = "Link kopieren" +Blockly.Msg.tooltip_trashcan_hide = 'gelöschte Blöcke ausblenden' +Blockly.Msg.tooltip_trashcan_delete = 'Blöcke endgültig löschen' +Blockly.Msg.tooltip_project_title = "Titel des Projektes" +Blockly.Msg.tooltip_check_solution = "Lösung kontrollieren" + +/** + * Messages + * + */ + +Blockly.Msg.messages_delete_project_failed = "Fehler beim Löschen des Projektes. Versuche es noch einmal." +Blockly.Msg.messages_reset_workspace_success = "Das Projekt wurde erfolgreich zurückgesetzt" +Blockly.Msg.messages_PROJECT_UPDATE_SUCCESS = "Das Projekt wurde erfolgreich aktualisiert." +Blockly.Msg.messages_GALLERY_UPDATE_SUCCESS = "Das Galerie-Projekt wurde erfolgreich aktualisiert." +Blockly.Msg.messages_PROJECT_UPDATE_FAIL = "Fehler beim Aktualisieren des Projektes. Versuche es noch einmal." +Blockly.Msg.messages_GALLERY_UPDATE_FAIL = "Fehler beim Aktualisieren des Galerie-Projektes. Versuche es noch einmal." +Blockly.Msg.messages_gallery_save_fail_1 = "Fehler beim Speichern des " +Blockly.Msg.messages_gallery_save_fail_2 = "Projektes. Versuche es noch einmal." +Blockly.Msg.messages_SHARE_SUCCESS = 'Programm teilen' +Blockly.Msg.messages_SHARE_FAIL = "Fehler beim Erstellen eines Links zum Teilen deines Programmes. Versuche es noch einmal." +Blockly.Msg.messages_copylink_success = 'Link erfolgreich in Zwischenablage gespeichert.' +Blockly.Msg.messages_rename_success_01 = 'Das Projekt wurde erfolgreich in ' +Blockly.Msg.messages_rename_success_02 = 'umbenannt.' +Blockly.Msg.messages_newblockly_head = "Willkommen zur neuen Version Blockly für die senseBox" +Blockly.Msg.messages_newblockly_text = "Die neue Blockly Version befindet sich zurzeit in der Testphase. Alle Neuigkeiten findet ihr hier:" +Blockly.Msg.messages_GET_TUTORIAL_FAIL = 'Zurück zur Tutorials-Übersicht' +Blockly.Msg.messages_LOGIN_FAIL = 'Der Benutzername oder das Passwort ist nicht korrekt.' +/** + * Share Dialog + */ + +Blockly.Msg.sharedialog_headline = "Dein Link wurde erstellt."; +Blockly.Msg.sharedialog_text = "Über den folgenden Link kannst du dein Programm teilen."; + +/** + * Project rename Dialog + */ + +Blockly.Msg.renamedialog_headline = "Projekt benennen"; +Blockly.Msg.renamedialog_text = "Bitte gib einen Namen für das Projekt ein und bestätige diesen mit einem Klick auf 'Bestätigen'." + +/** + * Compile Dialog + * + */ + +Blockly.Msg.compiledialog_headline = "Fehler" +Blockly.Msg.compiledialog_text = "Beim kompilieren ist ein Fehler aufgetreten. Überprüfe deine Blöcke und versuche es erneut" + +/** + * Buttons + * + */ + +Blockly.Msg.button_cancel = "Abbrechen"; +Blockly.Msg.button_close = "Schließen"; +Blockly.Msg.button_accept = "Bestätigen"; +Blockly.Msg.button_compile = "Kompilieren"; +Blockly.Msg.button_create_variableCreate = "Erstelle Variable"; +Blockly.Msg.button_back = "Zurück" +Blockly.Msg.button_next = "nächster Schritt" +Blockly.Msg.button_tutorial_overview = "Tutorial Übersicht" +Blockly.Msg.button_login = "Anmelden" + +/** + * + */ + +Blockly.Msg.filename = "Dateiname"; +Blockly.Msg.projectname = "Projektname"; + +/** + * Settings + */ +Blockly.Msg.settings_head = "Einstellungen" +Blockly.Msg.settings_language = "Sprache" +Blockly.Msg.settings_language_text = "Auswahl der Sprache gilt für die gesamte Anwendung. Es kann zwischen Deutsch und Englisch unterschieden werden." +Blockly.Msg.settings_language_de = "Deutsch" +Blockly.Msg.settings_language_en = "Englisch" +Blockly.Msg.settings_renderer = "Renderer" +Blockly.Msg.settings_renderer_text = "Der eingestellte Renderer bestimmt das Aussehen der Blöcke. Es kann zwischen 'Geras' und 'Zelos' unterschieden werden, wobei 'Zelos' insbesondere für eine Touch-Anwendung geeignet ist." +Blockly.Msg.settings_statistics = "Statistiken" +Blockly.Msg.settings_statistics_text = "Die Anzeige von Statistiken zur Nutzung der Blöcke oberhalb der Arbeitsfläche kann ein- oder ausgeblendet werden." +Blockly.Msg.settings_statistics_on = "An" +Blockly.Msg.settings_statistics_off = "Aus" + +/** + * 404 + */ + +Blockly.Msg.notfound_head = "Die von Ihnen angeforderte Seite kann nicht gefunden werden." +Blockly.Msg.notfound_text = "Die gesuchte Seite wurde möglicherweise entfernt, ihr Name wurde geändert oder sie ist vorübergehend nicht verfügbar." + + +/** + * Labels + */ + +Blockly.Msg.labels_donotshowagain = 'Dialog nicht mehr anzeigen' +Blockly.Msg.labels_here = "hier" +Blockly.Msg.labels_username = 'E-Mail oder Nutzername' +Blockly.Msg.labels_password = "Passwort" + +/** + * Badges + */ + +Blockly.Msg.badges_explaination = "Eine Übersicht über alle erhaltenen Badges im Kontext Blockly for senseBox findest du " +Blockly.Msg.badges_ASSIGNE_BADGE_SUCCESS_01 = "Herzlichen Glückwunsch! Du hast den Badge " +Blockly.Msg.badges_ASSIGNE_BADGE_SUCCESS_02 = " erhalten." +/** + * Tutorials + */ + +Blockly.Msg.tutorials_assessment_task = "Aufgabe" +Blockly.Msg.tutorials_hardware_head = "Für die Umsetzung benötigst du folgende Hardware:" +Blockly.Msg.tutorials_hardware_moreInformation = "Weitere Informationen zur Hardware-Komponente findest du" +Blockly.Msg.tutorials_hardware_here = "hier"; +Blockly.Msg.tutorials_requirements = "Bevor du mit diesem Tutorial fortfährst solltest du folgende Tutorials erfolgreich abgeschlossen haben:" + + +/** + * Tutorial Builder + */ + +Blockly.Msg.builder_solution = "Lösung" +Blockly.Msg.builder_solution_submit = "Lösung einreichen" +Blockly.Msg.builder_example_submit = "Beispiel einreichen" +Blockly.Msg.builder_comment = "Anmerkung: Man kann den initialen Setup()- bzw. Endlosschleifen()-Block löschen. Zusätzlich ist es möglich u.a. nur einen beliebigen Block auszuwählen, ohne dass dieser als deaktiviert dargestellt wird." +Blockly.Msg.builder_hardware_order = "Beachte, dass die Reihenfolge des Auswählens maßgebend ist." +Blockly.Msg.builder_hardware_helper = "Wähle mindestens eine Hardware-Komponente aus." +Blockly.Msg.builder_requirements_head = "Voraussetzungen" +Blockly.Msg.builder_requirements_order = "Beachte, dass die Reihenfolge des Anhakens maßgebend ist." + +/** + * Login + */ + + +Blockly.Msg.login_head = "Anmelden" +Blockly.Msg.login_osem_account_01 = "Du benötigst einen " +Blockly.Msg.login_osem_account_02 = "Account um dich einzuloggen" +Blockly.Msg.login_lostpassword = "Du hast dein Passwort vergessen?" +Blockly.Msg.login_createaccount = "Falls du noch keinen Account hast erstellen einen auf " +/** + * Navbar + */ + +Blockly.Msg.navbar_tutorials = "Tutorials" +Blockly.Msg.navbar_tutorialbuilder = "Tutorial erstellen" +Blockly.Msg.navbar_gallery = "Gallerie" +Blockly.Msg.navbar_projects = "Projekte" + +Blockly.Msg.navbar_menu = "Menü" +Blockly.Msg.navbar_login = "Einloggen" +Blockly.Msg.navbar_mybadges = "myBadges" +Blockly.Msg.navbar_account = "Konto" +Blockly.Msg.navbar_logout = "Abmelden" +Blockly.Msg.navbar_settings = "Einstellungen" + +/** + * Codeviewer + */ + +Blockly.Msg.codeviewer_arduino = "Arduino Quellcode" +Blockly.Msg.codeviewer_xml = "XML Blöcke" + + export const De = Blockly.Msg; diff --git a/src/components/Blockly/msg/en.js b/src/components/Blockly/msg/en.js index 696d68e..e7ca221 100644 --- a/src/components/Blockly/msg/en.js +++ b/src/components/Blockly/msg/en.js @@ -782,4 +782,209 @@ Blockly.Msg.senseBox_mqtt_password = "Password"; Blockly.Msg.sensebox_mqtt_subscribe = "Subscribe to Feed" Blockly.Msg.senseBox_mqtt_publish = "Publish to Feed/Topic"; +/** + * Add Translation for Blocks above + * --------------------------------------------------------------- + * Add Translation for the UI below + */ + +/** + * Toolbox + */ +Blockly.Msg.toolbox_sensors = "Sensors"; +Blockly.Msg.toolbox_logic = "Logic"; +Blockly.Msg.toolbox_loops = "Loops"; +Blockly.Msg.toolbox_math = "Math"; +Blockly.Msg.toolbox_io = "Input/Output"; +Blockly.Msg.toolbox_time = "Time"; +Blockly.Msg.toolbox_functions = "Functions"; +Blockly.Msg.toolbox_variables = "Variables"; + +/** + * Tooltips + * + */ + +Blockly.Msg.tooltip_compile_code = "Compile code" +Blockly.Msg.tooltip_save_blocks = "Save blocks"; +Blockly.Msg.tooltip_open_blocks = "Open blocks"; +Blockly.Msg.tooltip_screenshot = "Download screenshot"; +Blockly.Msg.tooltip_clear_workspace = "Reset workspace"; +Blockly.Msg.tooltip_share_blocks = "Share blocks"; +Blockly.Msg.tooltip_show_code = "Show code"; +Blockly.Msg.tooltip_hide_code = "Hide code " +Blockly.Msg.tooltip_delete_project = "Delete project" +Blockly.Msg.tooltip_project_name = "Project name" +Blockly.Msg.tooltip_download_project = "Download project" +Blockly.Msg.tooltip_open_project = "Open project" +Blockly.Msg.tooltip_update_project = "Update project" +Blockly.Msg.tooltip_save_project = "Save project" +Blockly.Msg.tooltip_create_project = "Create project" +Blockly.Msg.tooltip_share_project = "Share project" +Blockly.Msg.tooltip_reset_workspace = "Reset workspace" +Blockly.Msg.tooltip_copy_link = "Cooy link" +Blockly.Msg.tooltip_trashcan_hide = "hide deleted blocks" +Blockly.Msg.tooltip_trashcan_delete = "empty trashcan" +Blockly.Msg.tooltip_project_title = "Project title" +Blockly.Msg.tooltip_check_solution = "Check solution" + +/** + * Messages + * + */ + +Blockly.Msg.messages_delete_project_failed = "Error deleting the project. Try again." +Blockly.Msg.messages_reset_workspace_success = "The project has been successfully reset." +Blockly.Msg.messages_PROJECT_UPDATE_SUCCESS = "The project was successfully updated." +Blockly.Msg.messages_GALLERY_UPDATE_SUCCESS = "The gallery project was successfully updated." +Blockly.Msg.messages_PROJECT_UPDATE_FAIL = "Error updating the project. Try again." +Blockly.Msg.messages_GALLERY_UPDATE_FAIL = "Error updating the gallery project. Try again." +Blockly.Msg.messages_gallery_save_fail_1 = "Error saving the " +Blockly.Msg.messages_gallery_save_fail_2 = "Project. Try again." +Blockly.Msg.messages_SHARE_SUCCESS = 'Share program' +Blockly.Msg.messages_SHARE_FAIL = "Error creating a link to share your program. Try again." +Blockly.Msg.messages_copylink_success = 'Link successfully saved to clipboard.' +Blockly.Msg.messages_rename_success_01 = 'The project was successfully saved to ' +Blockly.Msg.messages_rename_success_02 = 'renamed.' +Blockly.Msg.messages_newblockly_head = 'Welcome to the new version Blockly for the senseBox' +Blockly.Msg.messages_newblockly_text = "The new Blockly version is currently in testing. You can find all the news here:" +Blockly.Msg.messages_GET_TUTORIAL_FAIL = 'Back to tutorials overview' +Blockly.Msg.messages_LOGIN_FAIL = 'The username or password is incorrect.' +Blockly.Msg.messages_login_error = "Enter both a username and a password." +/** + * Share Dialog + */ + +Blockly.Msg.sharedialog_headline = "Your link has been created."; +Blockly.Msg.sharedialog_text = "You can share your program using the following link."; + +/** + * Project rename Dialog + */ + +Blockly.Msg.renamedialog_headline = "Rename project"; +Blockly.Msg.renamedialog_text = "Please enter a name for the project and confirm it by clicking 'Confirm'." +/** + * Compile Dialog + * + */ + +Blockly.Msg.compiledialog_headline = "Error" +Blockly.Msg.compiledialog_text = "While compiling an error occured. Please check your blocks and try again" + +/** + * Buttons + * + */ + +Blockly.Msg.button_cancel = "Cancel"; +Blockly.Msg.button_close = "Close"; +Blockly.Msg.button_accept = "Ok"; +Blockly.Msg.button_compile = "Compile"; +Blockly.Msg.button_create_variableCreate = "Create Variable"; +Blockly.Msg.button_back = "Back" +Blockly.Msg.button_next = "Next step" +Blockly.Msg.button_tutorial_overview = "Tutorial overview" +Blockly.Msg.button_login = "Login" + +/** + * + */ + +Blockly.Msg.filename = "Filename"; +Blockly.Msg.projectname = "Projectname"; + +/** + * 404 + */ + +Blockly.Msg.notfound_head = "The page you requested cannot be found." +Blockly.Msg.notfound_text = "The page you are looking for may have been removed, its name changed, or it may be temporarily unavailable." + +/** + * Labels + */ +Blockly.Msg.labels_donotshowagain = 'Do not show dialog again' +Blockly.Msg.labels_here = 'here' +Blockly.Msg.labels_username = 'Email or username' +Blockly.Msg.labels_password = "Password" +/** + * Badges + */ + +Blockly.Msg.badges_explaination = "An overview of all badges received in the Blockly for senseBox context can be found " +Blockly.Msg.badges_ASSIGNE_BADGE_SUCCESS_01 = "Congratulations! You have received the badge " +Blockly.Msg.badges_ASSIGNE_BADGE_SUCCESS_02 = "." +/** + * Tutorials + */ + +Blockly.Msg.tutorials_assessment_task = "Task" +Blockly.Msg.tutorials_hardware_head = "For the implementation you need the following hardware:" +Blockly.Msg.tutorials_hardware_moreInformation = "You can find more information about the hardware component." +Blockly.Msg.tutorials_hardware_here = "here"; +Blockly.Msg.tutorials_requirements = "Before continuing with this tutorial, you should have successfully completed the following tutorials:" + + +/** + * Tutorial Builder + */ + +Blockly.Msg.builder_solution = "Solution" +Blockly.Msg.builder_solution_submit = "Submit Solution" +Blockly.Msg.builder_example_submit = "Submit example" +Blockly.Msg.builder_comment = "Note: You can delete the initial setup() or infinite loop() block. Additionally, it is possible to select only any block, among others, without displaying it as disabled." +Blockly.Msg.builder_hardware_order = "Note that the order of selection is authoritative." +Blockly.Msg.builder_hardware_helper = "Select at least one hardware component." +Blockly.Msg.builder_requirements_head = "Requirements." +Blockly.Msg.builder_requirements_order = "Note that the order of ticking is authoritative." + +/** + * Login + */ + + +Blockly.Msg.login_head = "Login" +Blockly.Msg.login_osem_account_01 = "You need to have an " +Blockly.Msg.login_osem_account_02 = "Account to login" +Blockly.Msg.login_lostpassword = "Lost your password?" +Blockly.Msg.login_createaccount = "If you don't have an openSenseMap account please register on " + + + +/** + * Settings + */ +Blockly.Msg.settings_head = "Settings" +Blockly.Msg.settings_language = "Language" +Blockly.Msg.settings_language_text = "Selection of the language applies to the entire application. A distinction can be made between German and English." +Blockly.Msg.settings_language_de = "German" +Blockly.Msg.settings_language_en = "English" +Blockly.Msg.settings_renderer = "Renderer" +Blockly.Msg.settings_renderer_text = "The selected renderer determines the appearance of the blocks. A distinction can be made between 'Geras' and 'Zelos', whereby 'Zelos' is particularly suitable for a touch application." +Blockly.Msg.settings_statistics = "Statistics" +Blockly.Msg.settings_statistics_text = "The display of statistics on the usage of the blocks above the workspace can be shown or hidden." +Blockly.Msg.settings_statistics_on = "On" +Blockly.Msg.settings_statistics_off = "Off" + +/** + * Navbar + */ + +Blockly.Msg.navbar_tutorials = "Tutorials" +Blockly.Msg.navbar_tutorialbuilder = "Create tutorial" +Blockly.Msg.navbar_gallery = "Gallery" +Blockly.Msg.navbar_projects = "Projects" + +Blockly.Msg.navbar_menu = "Menu" +Blockly.Msg.navbar_login = "Login" +Blockly.Msg.navbar_mybadges = "myBadges" +Blockly.Msg.navbar_account = "Account" +Blockly.Msg.navbar_logout = "Logout" +Blockly.Msg.navbar_settings = "Settings" + + +Blockly.Msg.codeviewer_arduino = "Arduino Source Code" +Blockly.Msg.codeviewer_xml = "XML Blocks" + export const En = Blockly.Msg; diff --git a/src/components/Blockly/toolbox/Toolbox.js b/src/components/Blockly/toolbox/Toolbox.js index 47fede7..cce1097 100644 --- a/src/components/Blockly/toolbox/Toolbox.js +++ b/src/components/Blockly/toolbox/Toolbox.js @@ -38,7 +38,7 @@ class Toolbox extends React.Component { render() { return ( - + @@ -301,7 +301,7 @@ class Toolbox extends React.Component { {/* */} - + @@ -310,7 +310,7 @@ class Toolbox extends React.Component { - + @@ -349,7 +349,7 @@ class Toolbox extends React.Component { - + @@ -369,7 +369,7 @@ class Toolbox extends React.Component { - + @@ -422,15 +422,15 @@ class Toolbox extends React.Component { - + - + - + diff --git a/src/components/CodeViewer.js b/src/components/CodeViewer.js index 35a972a..dd41fc4 100644 --- a/src/components/CodeViewer.js +++ b/src/components/CodeViewer.js @@ -13,6 +13,7 @@ import MuiAccordion from '@material-ui/core/Accordion'; import MuiAccordionSummary from '@material-ui/core/AccordionSummary'; import MuiAccordionDetails from '@material-ui/core/AccordionDetails'; import { Card } from '@material-ui/core'; +import * as Blockly from 'blockly' const Accordion = withStyles((theme) => ({ @@ -94,7 +95,7 @@ class CodeViewer extends Component { > {curlyBrackets} -
Arduino Quellcode
+
{Blockly.Msg.codeviewer_arduino}
@@ -112,7 +113,7 @@ class CodeViewer extends Component {
         >
           
             {unequal}
-            
XML Blöcke
+
{Blockly.Msg.codeviewer_xml}
diff --git a/src/components/Content.js b/src/components/Content.js
new file mode 100644
index 0000000..9260421
--- /dev/null
+++ b/src/components/Content.js
@@ -0,0 +1,54 @@
+import React, { Component } from 'react';
+import PropTypes from 'prop-types';
+import { connect } from 'react-redux';
+
+import * as Blockly from 'blockly/core';
+import { De } from './Blockly/msg/de';
+import { En } from './Blockly/msg/en';
+
+import Navbar from './Navbar';
+import Footer from './Footer';
+import Routes from './Route/Routes';
+import Cookies from './Cookies';
+
+class Content extends Component {
+
+  componentDidMount() {
+    if (this.props.language === 'de_DE') {
+      Blockly.setLocale(De);
+    } else if (this.props.language === 'en_US') {
+      Blockly.setLocale(En);
+    }
+  }
+
+  componentDidUpdate(props){
+    if(props.language !== this.props.language){
+      if (this.props.language === 'de_DE') {
+        Blockly.setLocale(De);
+      } else if (this.props.language === 'en_US') {
+        Blockly.setLocale(En);
+      }
+    }
+  }
+
+  render() {
+    return (
+      
+ + + +
+
+ ); + } +} + +Content.propTypes = { + language: PropTypes.string.isRequired +}; + +const mapStateToProps = state => ({ + language: state.general.language +}); + +export default connect(mapStateToProps, null)(Content); diff --git a/src/components/Home.js b/src/components/Home.js index 7c385ac..f5ec9cb 100644 --- a/src/components/Home.js +++ b/src/components/Home.js @@ -48,7 +48,6 @@ class Home extends Component { state = { codeOn: false, - stats: window.localStorage.getItem('stats'), snackbar: false, type: '', key: '', @@ -91,7 +90,7 @@ class Home extends Component { render() { return (
- {this.state.stats ? + {this.props.statistics ?
: null } @@ -136,11 +135,13 @@ class Home extends Component { Home.propTypes = { clearStats: PropTypes.func.isRequired, workspaceName: PropTypes.func.isRequired, - message: PropTypes.object.isRequired + message: PropTypes.object.isRequired, + statistics: PropTypes.bool.isRequired }; const mapStateToProps = state => ({ - message: state.message + message: state.message, + statistics: state.general.statistics }); diff --git a/src/components/Navbar.js b/src/components/Navbar.js index 2bc7b62..dbdd561 100644 --- a/src/components/Navbar.js +++ b/src/components/Navbar.js @@ -21,8 +21,10 @@ import ListItemIcon from '@material-ui/core/ListItemIcon'; import ListItemText from '@material-ui/core/ListItemText'; import LinearProgress from '@material-ui/core/LinearProgress'; -import { faBars, faChevronLeft, faLayerGroup, faSignInAlt, faSignOutAlt, faCertificate, faUserCircle, faIdCard, faEnvelope, faCog, faChalkboardTeacher, faFolderPlus, faTools, faLightbulb } from "@fortawesome/free-solid-svg-icons"; +import { faBars, faChevronLeft, faLayerGroup, faSignInAlt, faSignOutAlt, faCertificate, faUserCircle, faCog, faChalkboardTeacher, faTools, faLightbulb } from "@fortawesome/free-solid-svg-icons"; import { FontAwesomeIcon } from "@fortawesome/react-fontawesome"; +import * as Blockly from 'blockly' + const styles = (theme) => ({ drawerWidth: { @@ -92,7 +94,7 @@ class Navbar extends Component {
- Menü + {Blockly.Msg.navbar_menu}
@@ -100,47 +102,47 @@ class Navbar extends Component {
- {[{ text: 'Tutorials', icon: faChalkboardTeacher, link: "/tutorial" }, - { text: 'Tutorial-Builder', icon: faTools, link: "/tutorial/builder", restriction: this.props.user && this.props.user.blocklyRole !== 'user' && this.props.isAuthenticated}, - { text: 'Galerie', icon: faLightbulb, link: "/gallery" }, - { text: 'Projekte', icon: faLayerGroup, link: "/project", restriction: this.props.isAuthenticated }].map((item, index) => { - if(item.restriction || Object.keys(item).filter(attribute => attribute === 'restriction').length === 0){ - return( - - - - - - - ); - } + {[{ text: Blockly.Msg.navbar_tutorials, icon: faChalkboardTeacher, link: "/tutorial" }, + { text: Blockly.Msg.navbar_tutorialbuilder, icon: faTools, link: "/tutorial/builder", restriction: this.props.user && this.props.user.blocklyRole !== 'user' && this.props.isAuthenticated }, + { text: Blockly.Msg.navbar_gallery, icon: faLightbulb, link: "/gallery" }, + { text: Blockly.Msg.navbar_projects, icon: faLayerGroup, link: "/project", restriction: this.props.isAuthenticated }].map((item, index) => { + if (item.restriction || Object.keys(item).filter(attribute => attribute === 'restriction').length === 0) { + return ( + + + + + + + ); } + } )} - {[{ text: 'Anmelden', icon: faSignInAlt, link: '/user/login', restriction: !this.props.isAuthenticated }, - { text: 'Konto', icon: faUserCircle, link: '/user', restriction: this.props.isAuthenticated }, - { text: 'MyBadges', icon: faCertificate, link: '/user/badge', restriction: this.props.isAuthenticated }, - { text: 'Abmelden', icon: faSignOutAlt, function: this.props.logout, restriction: this.props.isAuthenticated }, - { text: 'Einstellungen', icon: faCog, link: "/settings" }].map((item, index) => { - if(item.restriction || Object.keys(item).filter(attribute => attribute === 'restriction').length === 0){ - return( - - {item.function(); this.toggleDrawer();} : this.toggleDrawer}> - - - - - ); - } + {[{ text: Blockly.Msg.navbar_login, icon: faSignInAlt, link: '/user/login', restriction: !this.props.isAuthenticated }, + { text: Blockly.Msg.navbar_account, icon: faUserCircle, link: '/user', restriction: this.props.isAuthenticated }, + { text: Blockly.Msg.navbar_mybadges, icon: faCertificate, link: '/user/badge', restriction: this.props.isAuthenticated }, + { text: Blockly.Msg.navbar_logout, icon: faSignOutAlt, function: this.props.logout, restriction: this.props.isAuthenticated }, + { text: Blockly.Msg.navbar_settings, icon: faCog, link: "/settings" }].map((item, index) => { + if (item.restriction || Object.keys(item).filter(attribute => attribute === 'restriction').length === 0) { + return ( + + { item.function(); this.toggleDrawer(); } : this.toggleDrawer}> + + + + + ); } + } )} {this.props.tutorialIsLoading || this.props.projectIsLoading ? - - : null} + + : null}
); } diff --git a/src/components/NotFound.js b/src/components/NotFound.js index 85a9c6f..01b03f7 100644 --- a/src/components/NotFound.js +++ b/src/components/NotFound.js @@ -6,31 +6,39 @@ import { withRouter } from 'react-router-dom'; import Button from '@material-ui/core/Button'; import Typography from '@material-ui/core/Typography'; +import * as Blockly from 'blockly' class NotFound extends Component { + + componentDidMount(){ + // Ensure that Blockly.setLocale is adopted in the component. + // Otherwise, the text will not be displayed until the next update of the component. + this.forceUpdate(); + } + render() { return (
- - Die von Ihnen angeforderte Seite kann nicht gefunden werden. - Die gesuchte Seite wurde möglicherweise entfernt, ihr Name wurde geändert oder sie ist vorübergehend nicht verfügbar. + + {Blockly.Msg.notfound_head} + {Blockly.Msg.notfound_text} {this.props.button ? - : + : }
diff --git a/src/components/Project/Project.js b/src/components/Project/Project.js index a3a2403..fa2353c 100644 --- a/src/components/Project/Project.js +++ b/src/components/Project/Project.js @@ -5,9 +5,7 @@ import { workspaceName } from '../../actions/workspaceActions'; import { getProject, resetProject } from '../../actions/projectActions'; import { clearMessages, returnErrors } from '../../actions/messageActions'; -import axios from 'axios'; import { withRouter } from 'react-router-dom'; -import { createNameId } from 'mnemonic-id'; import Home from '../Home'; import Breadcrumbs from '../Breadcrumbs'; @@ -24,16 +22,16 @@ class Project extends Component { } componentDidUpdate(props) { - if(props.location.pathname !== this.props.location.pathname || - props.match.params[`${this.props.type}Id`] !== this.props.match.params[`${this.props.type}Id`]){ - if(this.props.message.msg){ + if (props.location.pathname !== this.props.location.pathname || + props.match.params[`${this.props.type}Id`] !== this.props.match.params[`${this.props.type}Id`]) { + if (this.props.message.msg) { this.props.clearMessages(); } this.getProject(); } - if(this.props.message !== props.message){ - if(this.props.message.id === 'PROJECT_EMPTY' || this.props.message.id === 'GET_PROJECT_FAIL'){ - if(this.props.type!=='share'){ + if (this.props.message !== props.message) { + if (this.props.message.id === 'PROJECT_EMPTY' || this.props.message.id === 'GET_PROJECT_FAIL') { + if (this.props.type !== 'share') { this.props.returnErrors('', 404, 'GET_PROJECT_FAIL'); this.props.history.push(`/${this.props.type}`); } else { @@ -41,10 +39,10 @@ class Project extends Component { this.props.returnErrors('', 404, 'GET_SHARE_FAIL'); } } - else if(this.props.message.id === 'GET_PROJECT_SUCCESS'){ + else if (this.props.message.id === 'GET_PROJECT_SUCCESS') { this.props.workspaceName(this.props.project.title); } - else if(this.props.message.id === 'PROJECT_DELETE_SUCCESS' || this.props.message.id === 'GALLERY_DELETE_SUCCESS'){ + else if (this.props.message.id === 'PROJECT_DELETE_SUCCESS' || this.props.message.id === 'GALLERY_DELETE_SUCCESS') { this.props.history.push(`/${this.props.type}`); } } @@ -56,8 +54,8 @@ class Project extends Component { } getProject = () => { - var id = this.props.location.pathname.replace(/\/[a-z]{1,}\//,''); - var param = this.props.location.pathname.replace(`/${id}`,'').replace('/',''); + var id = this.props.location.pathname.replace(/\/[a-z]{1,}\//, ''); + var param = this.props.location.pathname.replace(`/${id}`, '').replace('/', ''); console.log('param', param); console.log(id); this.props.getProject(param, id); @@ -70,13 +68,13 @@ class Project extends Component { - : this.props.project ? -
- {this.props.type !== 'share' ? - - : null} - -
: null + : this.props.project ? +
+ {this.props.type !== 'share' ? + + : null} + +
: null ); }; } diff --git a/src/components/Project/ProjectHome.js b/src/components/Project/ProjectHome.js index 07e559b..c7a8d8c 100644 --- a/src/components/Project/ProjectHome.js +++ b/src/components/Project/ProjectHome.js @@ -4,7 +4,6 @@ import { connect } from 'react-redux'; import { getProjects, resetProject } from '../../actions/projectActions'; import { clearMessages } from '../../actions/messageActions'; -import axios from 'axios'; import { Link, withRouter } from 'react-router-dom'; import Breadcrumbs from '../Breadcrumbs'; @@ -42,31 +41,31 @@ class ProjectHome extends Component { } componentDidMount() { - var type = this.props.location.pathname.replace('/',''); + var type = this.props.location.pathname.replace('/', ''); this.props.getProjects(type); - if(this.props.message){ - if(this.props.message.id === 'PROJECT_DELETE_SUCCESS'){ + if (this.props.message) { + if (this.props.message.id === 'PROJECT_DELETE_SUCCESS') { this.setState({ snackbar: true, key: Date.now(), message: `Dein Projekt wurde erfolgreich gelöscht.`, type: 'success' }); } - else if(this.props.message.id === 'GALLERY_DELETE_SUCCESS'){ + else if (this.props.message.id === 'GALLERY_DELETE_SUCCESS') { this.setState({ snackbar: true, key: Date.now(), message: `Dein Galerie-Projekt wurde erfolgreich gelöscht.`, type: 'success' }); } - else if(this.props.message.id === 'GET_PROJECT_FAIL'){ - this.setState({ snackbar: true, key: Date.now(), message: `Dein angefragtes ${type === 'gallery' ? 'Galerie-':''}Projekt konnte nicht gefunden werden.`, type: 'error' }); + else if (this.props.message.id === 'GET_PROJECT_FAIL') { + this.setState({ snackbar: true, key: Date.now(), message: `Dein angefragtes ${type === 'gallery' ? 'Galerie-' : ''}Projekt konnte nicht gefunden werden.`, type: 'error' }); } } } componentDidUpdate(props) { - if(props.location.pathname !== this.props.location.pathname){ - this.setState({snackbar: false}); - this.props.getProjects(this.props.location.pathname.replace('/','')); + if (props.location.pathname !== this.props.location.pathname) { + this.setState({ snackbar: false }); + this.props.getProjects(this.props.location.pathname.replace('/', '')); } - if(props.message !== this.props.message){ - if(this.props.message.id === 'PROJECT_DELETE_SUCCESS'){ + if (props.message !== this.props.message) { + if (this.props.message.id === 'PROJECT_DELETE_SUCCESS') { this.setState({ snackbar: true, key: Date.now(), message: `Dein Projekt wurde erfolgreich gelöscht.`, type: 'success' }); } - else if(this.props.message.id === 'GALLERY_DELETE_SUCCESS'){ + else if (this.props.message.id === 'GALLERY_DELETE_SUCCESS') { this.setState({ snackbar: true, key: Date.now(), message: `Dein Galerie-Projekt wurde erfolgreich gelöscht.`, type: 'success' }); } } @@ -88,7 +87,7 @@ class ProjectHome extends Component { - : + :
{this.props.projects.length > 0 ? @@ -97,37 +96,37 @@ class ProjectHome extends Component { -

{project.title}

- +

{project.title}

+ - {project.description} + {project.description} {this.props.user && this.props.user.email === project.creator ?
- -
+ +
- : null} + : null} ) })} - :
- Es sind aktuell keine Projekte vorhanden. - {this.props.location.pathname.replace('/','') === 'project' ? + :
+ Es sind aktuell keine Projekte vorhanden. + {this.props.location.pathname.replace('/', '') === 'project' ? Erstelle jetzt dein eigenes Projekt oder lasse dich von Projektbeispielen in der Galerie inspirieren. - : null} + : null}
}
diff --git a/src/components/Route/IsLoggedRoute.js b/src/components/Route/IsLoggedRoute.js index f071b2b..2fb3617 100644 --- a/src/components/Route/IsLoggedRoute.js +++ b/src/components/Route/IsLoggedRoute.js @@ -9,6 +9,7 @@ class IsLoggedRoute extends Component { render() { return ( + !this.props.progress ? @@ -23,17 +24,19 @@ class IsLoggedRoute extends Component { /> ) } - /> + /> : null ); } } IsLoggedRoute.propTypes = { - isAuthenticated: PropTypes.bool.isRequired + isAuthenticated: PropTypes.bool.isRequired, + progress: PropTypes.bool.isRequired }; const mapStateToProps = state => ({ isAuthenticated: state.auth.isAuthenticated, + progress: state.auth.progress }); export default connect(mapStateToProps, null)(IsLoggedRoute); diff --git a/src/components/Route/PrivateRoute.js b/src/components/Route/PrivateRoute.js index ffd321a..0d41355 100644 --- a/src/components/Route/PrivateRoute.js +++ b/src/components/Route/PrivateRoute.js @@ -9,6 +9,7 @@ class PrivateRoute extends Component { render() { return ( + !this.props.progress ? @@ -25,17 +26,19 @@ class PrivateRoute extends Component { ) })() } - /> + /> : null ); } } PrivateRoute.propTypes = { - isAuthenticated: PropTypes.bool.isRequired + isAuthenticated: PropTypes.bool.isRequired, + progress: PropTypes.bool.isRequired }; const mapStateToProps = state => ({ - isAuthenticated: state.auth.isAuthenticated + isAuthenticated: state.auth.isAuthenticated, + progress: state.auth.progress }); export default connect(mapStateToProps, null)(withRouter(PrivateRoute)); diff --git a/src/components/Route/PrivateRouteCreator.js b/src/components/Route/PrivateRouteCreator.js index 0efd48c..e279969 100644 --- a/src/components/Route/PrivateRouteCreator.js +++ b/src/components/Route/PrivateRouteCreator.js @@ -9,6 +9,7 @@ class PrivateRoute extends Component { render() { return ( + !this.props.progress ? @@ -27,19 +28,21 @@ class PrivateRoute extends Component { ) })() } - /> + /> : null ); } } PrivateRoute.propTypes = { isAuthenticated: PropTypes.bool.isRequired, - user: PropTypes.object + user: PropTypes.object, + progress: PropTypes.bool.isRequired }; const mapStateToProps = state => ({ isAuthenticated: state.auth.isAuthenticated, - user: state.auth.user + user: state.auth.user, + progress: state.auth.progress }); export default connect(mapStateToProps, null)(withRouter(PrivateRoute)); diff --git a/src/components/Route/PublicRoute.js b/src/components/Route/PublicRoute.js new file mode 100644 index 0000000..99d4e5a --- /dev/null +++ b/src/components/Route/PublicRoute.js @@ -0,0 +1,32 @@ +import React, { Component } from 'react'; +import PropTypes from 'prop-types'; +import { connect } from 'react-redux'; + +import { Route } from 'react-router-dom'; + + +class PublicRoute extends Component { + + render() { + return ( + !this.props.progress ? + + this.props.children + } + /> + : null + ); + } +} + +PublicRoute.propTypes = { + progress: PropTypes.bool.isRequired +}; + +const mapStateToProps = state => ({ + progress: state.auth.progress +}); + +export default connect(mapStateToProps, null)(PublicRoute); diff --git a/src/components/Route/Routes.js b/src/components/Route/Routes.js index 3121edf..5207d45 100644 --- a/src/components/Route/Routes.js +++ b/src/components/Route/Routes.js @@ -5,6 +5,7 @@ import { visitPage } from '../../actions/generalActions'; import { Route, Switch, withRouter } from 'react-router-dom'; +import PublicRoute from './PublicRoute'; import PrivateRoute from './PrivateRoute'; import PrivateRouteCreator from './PrivateRouteCreator'; import IsLoggedRoute from './IsLoggedRoute'; @@ -34,18 +35,30 @@ class Routes extends Component { return (
- + + + {/* Tutorials */} - + + + - + - + + + {/* Sharing */} - + + + {/* Gallery-Projects */} - - + + + + + + {/* User-Projects */} @@ -64,12 +77,20 @@ class Routes extends Component { {/* settings */} - + + + {/* privacy */} - - + + + + + + {/* Not Found */} - + + +
); diff --git a/src/components/Settings/LanguageSelector.js b/src/components/Settings/LanguageSelector.js index dea7465..f069b1b 100644 --- a/src/components/Settings/LanguageSelector.js +++ b/src/components/Settings/LanguageSelector.js @@ -1,43 +1,58 @@ -import React from 'react'; -import { makeStyles } from '@material-ui/core/styles'; +import React, { Component } from 'react'; +import PropTypes from 'prop-types'; +import { connect } from 'react-redux'; +import { setLanguage } from '../../actions/generalActions'; + +import * as Blockly from 'blockly/core'; + import InputLabel from '@material-ui/core/InputLabel'; import MenuItem from '@material-ui/core/MenuItem'; import FormControl from '@material-ui/core/FormControl'; import Select from '@material-ui/core/Select'; +import Typography from '@material-ui/core/Typography'; +import FormHelperText from '@material-ui/core/FormHelperText'; -const useStyles = makeStyles((theme) => ({ - formControl: { - margin: theme.spacing(1), - minWidth: 120, - }, - selectEmpty: { - marginTop: theme.spacing(2), - }, -})); +class LanguageSelector extends Component { -export default function LanguageSelector() { - const classes = useStyles(); - const [lang, setLang] = React.useState(window.localStorage.getItem('locale')); + componentDidMount(){ + // Ensure that Blockly.setLocale is adopted in the component. + // Otherwise, the text will not be displayed until the next update of the component. + this.forceUpdate(); + } - const handleChange = (event) => { - setLang(event.target.value); - window.localStorage.setItem('locale', event.target.value); - }; + handleChange = (event) => { + this.props.setLanguage(event.target.value); + } - return ( -
- - Sprache - - -
+ render(){ + return( +
+ {Blockly.Msg.settings_language} + {Blockly.Msg.settings_language_text} + + {Blockly.Msg.settings_language} + + +
); + } } + +LanguageSelector.propTypes = { + setLanguage: PropTypes.func.isRequired, + language: PropTypes.string.isRequired +}; + +const mapStateToProps = state => ({ + language: state.general.language +}); + +export default connect(mapStateToProps, { setLanguage })(LanguageSelector); diff --git a/src/components/Settings/RenderSelector.js b/src/components/Settings/RenderSelector.js index 5cf7a9f..7359283 100644 --- a/src/components/Settings/RenderSelector.js +++ b/src/components/Settings/RenderSelector.js @@ -1,45 +1,57 @@ -import React from 'react'; -import { makeStyles } from '@material-ui/core/styles'; +import React, { Component } from 'react'; +import PropTypes from 'prop-types'; +import { connect } from 'react-redux'; +import { setRenderer } from '../../actions/generalActions'; + +import * as Blockly from 'blockly/core' + import InputLabel from '@material-ui/core/InputLabel'; import MenuItem from '@material-ui/core/MenuItem'; import FormControl from '@material-ui/core/FormControl'; import Select from '@material-ui/core/Select'; +import Typography from '@material-ui/core/Typography'; +import FormHelperText from '@material-ui/core/FormHelperText'; -const useStyles = makeStyles((theme) => ({ - formControl: { - margin: theme.spacing(1), - minWidth: 400, - }, - selectEmpty: { - marginTop: theme.spacing(2), - }, -})); -export default function RenderSelector() { - const classes = useStyles(); - const [renderer, setRenderer] = React.useState(window.localStorage.getItem('renderer')); +class RenderSelector extends Component { - const handleChange = (event) => { - setRenderer(event.target.value); - window.localStorage.setItem('renderer', event.target.value); - }; + componentDidMount(){ + // Ensure that Blockly.setLocale is adopted in the component. + // Otherwise, the text will not be displayed until the next update of the component. + this.forceUpdate(); + } + render(){ return ( -
- - Renderer - - -

Der Renderer bestimmt das aussehen der Blöcke

-
+
+ {Blockly.Msg.settings_renderer} + {Blockly.Msg.settings_renderer_text} + + {Blockly.Msg.settings_renderer} + + +
); + } } + +RenderSelector.propTypes = { + setRenderer: PropTypes.func.isRequired, + language: PropTypes.string.isRequired, + renderer: PropTypes.string.isRequired +}; + +const mapStateToProps = state => ({ + renderer: state.general.renderer, + language: state.general.language +}); + +export default connect(mapStateToProps, { setRenderer })(RenderSelector); diff --git a/src/components/Settings/Settings.js b/src/components/Settings/Settings.js index cbbedc9..24b3bfa 100644 --- a/src/components/Settings/Settings.js +++ b/src/components/Settings/Settings.js @@ -1,32 +1,65 @@ import React, { Component } from 'react'; +import PropTypes from 'prop-types'; +import { connect } from 'react-redux'; + import { withRouter } from 'react-router-dom'; -import Button from '@material-ui/core/Button'; -import Typography from '@material-ui/core/Typography'; +import * as Blockly from 'blockly/core'; + +import Breadcrumbs from '../Breadcrumbs'; import LanguageSelector from './LanguageSelector'; import RenderSelector from './RenderSelector'; import StatsSelector from './StatsSelector'; +import Button from '@material-ui/core/Button'; +import Paper from '@material-ui/core/Paper'; class Settings extends Component { - render() { - return ( -
- Einstellungen - - - - -
- ); - }; + + componentDidMount(){ + // Ensure that Blockly.setLocale is adopted in the component. + // Otherwise, the text will not be displayed until the next update of the component. + this.forceUpdate(); + } + + render() { + return ( +
+ + +

{Blockly.Msg.settings_head}

+ + + + + + + + + + + + +
+ ); + }; } -export default withRouter(Settings); +Settings.propTypes = { + language: PropTypes.string.isRequired, + pageVisits: PropTypes.number.isRequired +}; + +const mapStateToProps = state => ({ + language: state.general.language, + pageVisits: state.general.pageVisits +}); + +export default connect(mapStateToProps, null)(withRouter(Settings)); diff --git a/src/components/Settings/StatsSelector.js b/src/components/Settings/StatsSelector.js index 8c1322d..b0b3cfe 100644 --- a/src/components/Settings/StatsSelector.js +++ b/src/components/Settings/StatsSelector.js @@ -1,44 +1,56 @@ -import React from 'react'; -import { makeStyles } from '@material-ui/core/styles'; +import React, { Component } from 'react'; +import PropTypes from 'prop-types'; +import { connect } from 'react-redux'; +import { setStatistics } from '../../actions/generalActions'; + +import * as Blockly from 'blockly/core' + import InputLabel from '@material-ui/core/InputLabel'; import MenuItem from '@material-ui/core/MenuItem'; import FormControl from '@material-ui/core/FormControl'; import Select from '@material-ui/core/Select'; +import Typography from '@material-ui/core/Typography'; +import FormHelperText from '@material-ui/core/FormHelperText'; -const useStyles = makeStyles((theme) => ({ - formControl: { - margin: theme.spacing(1), - minWidth: 400, - }, - selectEmpty: { - marginTop: theme.spacing(2), - }, -})); +class StatsSelector extends Component { -export default function StatsSelector() { - const classes = useStyles(); - const [stats, setStats] = React.useState(window.localStorage.getItem('stats')); - - const handleChange = (event) => { - setStats(event.target.value); - window.localStorage.setItem('stats', event.target.value); - }; + componentDidMount(){ + // Ensure that Blockly.setLocale is adopted in the component. + // Otherwise, the text will not be displayed until the next update of the component. + this.forceUpdate(); + } + render(){ return ( -
- - Statistiken - - -

Schaltet die Statistiken Oberhalb der Arbeitsfläche ein bzw. aus

-
+
+ {Blockly.Msg.settings_statistics} + {Blockly.Msg.settings_statistics_text} + + {Blockly.Msg.settings_statistics} + + +
); + } } + +StatsSelector.propTypes = { + setStatistics: PropTypes.func.isRequired, + language: PropTypes.string.isRequired, + statistics: PropTypes.bool.isRequired +}; + +const mapStateToProps = state => ({ + statistics: state.general.statistics, + language: state.general.language +}); + +export default connect(mapStateToProps, { setStatistics })(StatsSelector); diff --git a/src/components/Tutorial/Assessment.js b/src/components/Tutorial/Assessment.js index 75e180d..c869cc5 100644 --- a/src/components/Tutorial/Assessment.js +++ b/src/components/Tutorial/Assessment.js @@ -11,6 +11,7 @@ import withWidth, { isWidthDown } from '@material-ui/core/withWidth'; import Grid from '@material-ui/core/Grid'; import Card from '@material-ui/core/Card'; import Typography from '@material-ui/core/Typography'; +import * as Blockly from 'blockly' class Assessment extends Component { @@ -45,7 +46,7 @@ class Assessment extends Component { - Arbeitsauftrag + {Blockly.Msg.tutorials_assessment_task} {currentTask.text}
diff --git a/src/components/Tutorial/Badge.js b/src/components/Tutorial/Badge.js index 97103fa..2727fa5 100644 --- a/src/components/Tutorial/Badge.js +++ b/src/components/Tutorial/Badge.js @@ -11,6 +11,7 @@ import { withStyles } from '@material-ui/core/styles'; import Paper from '@material-ui/core/Paper'; import Typography from '@material-ui/core/Typography'; import Avatar from '@material-ui/core/Avatar'; +import * as Blockly from 'blockly'; const styles = (theme) => ({ link: { @@ -32,22 +33,22 @@ class Badge extends Component { content: '' }; - componentDidUpdate(props){ - if(this.props.message.id === 'TUTORIAL_CHECK_SUCCESS'){ - if(this.props.tutorial.badge){ + componentDidUpdate(props) { + if (this.props.message.id === 'TUTORIAL_CHECK_SUCCESS') { + if (this.props.tutorial.badge) { // is connected to MyBadges? - if(this.props.isAuthenticated && this.props.user && this.props.user.badge){ - if(this.props.user.badges && !this.props.user.badges.includes(this.props.tutorial.badge)){ - if(this.isSuccess()){ + if (this.props.isAuthenticated && this.props.user && this.props.user.badge) { + if (this.props.user.badges && !this.props.user.badges.includes(this.props.tutorial.badge)) { + if (this.isSuccess()) { this.props.assigneBadge(this.props.tutorial.badge); } } } } } - if(props.message !== this.props.message){ - if(this.props.message.id === 'ASSIGNE_BADGE_SUCCESS'){ - this.setState({title: `Badge: ${this.props.message.msg.name}`, content: `Herzlichen Glückwunsch! Du hast den Badge ${this.props.message.msg.name} erhalten.`, open: true}); + if (props.message !== this.props.message) { + if (this.props.message.id === 'ASSIGNE_BADGE_SUCCESS') { + this.setState({ title: `Badge: ${this.props.message.msg.name}`, content: `${Blockly.Msg.badges_ASSIGNE_BADGE_SUCCESS_01} ${this.props.message.msg.name} ${Blockly.Msg.badges_ASSIGNE_BADGE_SUCCESS_02}`, open: true }); } } } @@ -57,7 +58,7 @@ class Badge extends Component { var status = this.props.status.filter(status => status._id === tutorialId)[0]; var tasks = status.tasks; var success = tasks.filter(task => task.type === 'success').length / tasks.length; - if(success===1){ + if (success === 1) { return true; } return false; @@ -74,21 +75,21 @@ class Badge extends Component { open={this.state.open} title={this.state.title} content={this.state.content} - onClose={() => {this.toggleDialog();}} - onClick={() => {this.toggleDialog();}} - button={'Schließen'} + onClose={() => { this.toggleDialog(); }} + onClick={() => { this.toggleDialog(); }} + button={Blockly.Msg.button_close} >
- + {this.props.message.msg.image && this.props.message.msg.image.path ? - - : } - -
{this.props.message.msg.name}
+ + : } + +
{this.props.message.msg.name}
- - Eine Übersicht über alle erhaltenen Badges im Kontext Blockly for senseBox findest du hier. + + {Blockly.Msg.badges_explaination}{Blockly.Msg.labels_here}.
diff --git a/src/components/Tutorial/Builder/Badge.js b/src/components/Tutorial/Builder/Badge.js new file mode 100644 index 0000000..88386a3 --- /dev/null +++ b/src/components/Tutorial/Builder/Badge.js @@ -0,0 +1,190 @@ +import React, { Component } from 'react'; +import PropTypes from 'prop-types'; +import { connect } from 'react-redux'; +import { tutorialBadge, deleteProperty, setError, deleteError } from '../../../actions/tutorialBuilderActions'; + +import axios from 'axios'; + +import { withStyles } from '@material-ui/core/styles'; +import Switch from '@material-ui/core/Switch'; +import FormControlLabel from '@material-ui/core/FormControlLabel'; +import FormHelperText from '@material-ui/core/FormHelperText'; +import List from '@material-ui/core/List'; +import ListItem from '@material-ui/core/ListItem'; +import ListItemText from '@material-ui/core/ListItemText'; +import IconButton from '@material-ui/core/IconButton'; +import OutlinedInput from '@material-ui/core/OutlinedInput'; +import InputLabel from '@material-ui/core/InputLabel'; +import FormControl from '@material-ui/core/FormControl'; + +import { FontAwesomeIcon } from "@fortawesome/react-fontawesome"; +import { faTimes } from "@fortawesome/free-solid-svg-icons"; + +const styles = (theme) => ({ + errorColor: { + color: `${theme.palette.error.dark} !important` + }, + errorColorShrink: { + color: `rgba(0, 0, 0, 0.54) !important` + }, + errorBorder: { + borderColor: `${theme.palette.error.dark} !important` + } +}); + +class Badge extends Component { + + constructor(props){ + super(props); + this.state={ + checked: props.badge ? true : false, + badgeName: '', + filteredBadges: [], + badges: [] + }; + } + + componentDidMount(){ + this.getBadges(); + } + + componentDidUpdate(props){ + if(props.badge !== this.props.badge){ + this.setState({ checked: this.props.badge !== undefined ? true : false, badgeName: this.props.badge ? this.state.badges.filter(badge => badge._id === this.props.badge)[0].name : '' }); + } + } + + getBadges = () => { + axios.get(`${process.env.REACT_APP_MYBADGES_API}/badge`) + .then(res => { + this.setState({badges: res.data.badges, badgeName: this.props.badge ? res.data.badges.filter(badge => badge._id === this.props.badge)[0].name : '' }); + }) + .catch(err => { + console.log(err); + }); + }; + + deleteBadge = () => { + this.setState({ filteredBadges: [], badgeName: '' }); + this.props.tutorialBadge(null); + this.props.setError(this.props.index, 'badge'); + }; + + setBadge = (badge) => { + this.setState({ filteredBadges: [] }); + this.props.tutorialBadge(badge._id); + this.props.deleteError(this.props.index, 'badge'); + }; + + onChange = e => { + this.setState({ badgeName: e.target.value }); + }; + + onChangeBadge = e => { + if(e.target.value && this.props.badge === null){ + var filteredBadges = this.state.badges.filter(badge => new RegExp(e.target.value, 'i').test(badge.name)); + if(filteredBadges.length < 1){ + filteredBadges = ['Keine Übereinstimmung gefunden.']; + } + this.setState({filteredBadges: filteredBadges}); + } + else { + this.setState({filteredBadges: []}); + } + }; + + onChangeSwitch = (value) => { + var oldValue = this.state.checked; + this.setState({checked: value}); + if(oldValue !== value){ + if(value){ + this.props.setError(this.props.index, 'badge'); + this.props.tutorialBadge(null); + } else { + this.props.deleteError(this.props.index, 'badge'); + this.props.tutorialBadge(undefined); + } + } + } + + render() { + return ( +
+ this.onChangeSwitch(e.target.checked)} + color="primary" + /> + } + /> + {this.state.checked ? +
+ + + {'Badge'} + + this.onChange(e)} + onInput={(e) => this.onChangeBadge(e)} + fullWidth={true} + endAdornment={ + + + + } + /> + {this.props.error && this.state.filteredBadges.length === 0 ? + Wähle ein Badge aus. + : null} + + + {this.state.filteredBadges.map((badge, i) => ( + badge === 'Keine Übereinstimmung gefunden.' ? + + {badge} + + : + {this.setBadge(badge)}} style={{border: '1px solid rgba(0, 0, 0, 0.23)', borderRadius: '25px'}}> + {`${badge.name}`} + + ))} + +
+ : null} +
+ ); + }; +} + +Badge.propTypes = { + tutorialBadge: PropTypes.func.isRequired, + deleteProperty: PropTypes.func.isRequired, + setError: PropTypes.func.isRequired, + deleteError: PropTypes.func.isRequired, + badge: PropTypes.string.isRequired +}; + +const mapStateToProps = state => ({ + badge: state.builder.badge, + change: state.builder.change +}); + + +export default connect(mapStateToProps, { tutorialBadge, deleteProperty, setError, deleteError })(withStyles(styles, {withTheme: true})(Badge)); diff --git a/src/components/Tutorial/Builder/BlocklyExample.js b/src/components/Tutorial/Builder/BlocklyExample.js index 01f52f4..75dc222 100644 --- a/src/components/Tutorial/Builder/BlocklyExample.js +++ b/src/components/Tutorial/Builder/BlocklyExample.js @@ -29,7 +29,7 @@ const styles = (theme) => ({ marginTop: '5px', height: '40px', backgroundColor: theme.palette.error.dark, - '&:hover':{ + '&:hover': { backgroundColor: theme.palette.error.dark } } @@ -37,16 +37,16 @@ const styles = (theme) => ({ class BlocklyExample extends Component { - constructor(props){ + constructor(props) { super(props); - this.state={ + this.state = { checked: props.task ? props.task : props.value ? true : false, input: null, disabled: false }; } - componentDidMount(){ + componentDidMount() { moment.updateLocale('de', localization); this.isError(); // if(this.props.task){ @@ -54,42 +54,42 @@ class BlocklyExample extends Component { // } } - componentDidUpdate(props, state){ - if(props.task !== this.props.task || props.value !== this.props.value){ - this.setState({checked: this.props.task ? this.props.task : this.props.value ? true : false}, + componentDidUpdate(props, state) { + if (props.task !== this.props.task || props.value !== this.props.value) { + this.setState({ checked: this.props.task ? this.props.task : this.props.value ? true : false }, () => this.isError() ); } - if(state.checked !== this.state.checked && this.state.checked){ + if (state.checked !== this.state.checked && this.state.checked) { this.isError(); } - if(props.xml !== this.props.xml){ + if (props.xml !== this.props.xml) { // check if there is at least one block, otherwise the workspace cannot be submitted var workspace = Blockly.getMainWorkspace(); var areBlocks = workspace.getAllBlocks().length > 0; - this.setState({disabled: !areBlocks}); + this.setState({ disabled: !areBlocks }); } } isError = () => { - if(this.state.checked){ + if (this.state.checked) { var xml = this.props.value; // check if value is valid xml; - try{ + try { Blockly.Xml.textToDom(xml); this.props.deleteError(this.props.index, 'xml'); } - catch(err){ + catch (err) { xml = initialXml; // not valid xml, throw error in redux store this.props.setError(this.props.index, 'xml'); } - if(!this.props.task){ + if (!this.props.task) { // instruction can also display only one block, which does not necessarily // have to be the initial block xml = xml.replace('deletable="false"', 'deletable="true"'); } - this.setState({xml: xml}); + this.setState({ xml: xml }); } else { this.props.deleteError(this.props.index, 'xml'); @@ -98,8 +98,8 @@ class BlocklyExample extends Component { onChange = (value) => { var oldValue = this.state.checked; - this.setState({checked: value}); - if(oldValue !== value && !value){ + this.setState({ checked: value }); + if (oldValue !== value && !value) { this.props.deleteError(this.props.index, 'xml'); this.props.deleteProperty(this.props.index, 'xml'); } @@ -108,12 +108,12 @@ class BlocklyExample extends Component { setXml = () => { var xml = this.props.xml; this.props.changeContent(xml, this.props.index, 'xml'); - this.setState({input: moment(Date.now()).format('LTS')}); + this.setState({ input: moment(Date.now()).format('LTS') }); } render() { return ( -
+
{!this.props.task ? } /> - : Musterlösung} + : {Blockly.Msg.builder_solution}} {this.state.checked ? !this.props.value || this.props.error ? - {`Reiche deine Blöcke ein, indem du auf den '${this.props.task ? 'Musterlösung einreichen' : 'Beispiel einreichen'}'-Button klickst.`} - : this.state.input ? Die letzte Einreichung erfolgte um {this.state.input} Uhr. : null - : null} + {`Reiche deine Blöcke ein, indem du auf den '${this.props.task ? Blockly.Msg.builder_solution_submit : Blockly.Msg.builder_example_submit}'-Button klickst.`} + : this.state.input ? Die letzte Einreichung erfolgte um {this.state.input} Uhr. : null + : null} {this.state.checked && !this.props.task ? - Anmerkung: Man kann den initialen Setup()- bzw. Endlosschleifen()-Block löschen. Zusätzlich ist es möglich u.a. nur einen beliebigen Block auszuwählen, ohne dass dieser als deaktiviert dargestellt wird. - : null} + {Blockly.Msg.builder_comment} + : null} {/* ensure that the correct xml-file is displayed in the workspace */} - {this.state.checked && this.state.xml? (() => { - return( -
+ {this.state.checked && this.state.xml ? (() => { + return ( +
- )})() - : null} + ) + })() + : null}
); }; @@ -179,4 +180,4 @@ const mapStateToProps = state => ({ }); -export default connect(mapStateToProps, { changeContent, deleteProperty, setError, deleteError })(withStyles(styles, {withTheme: true})(BlocklyExample)); +export default connect(mapStateToProps, { changeContent, deleteProperty, setError, deleteError })(withStyles(styles, { withTheme: true })(BlocklyExample)); diff --git a/src/components/Tutorial/Builder/Builder.js b/src/components/Tutorial/Builder/Builder.js index 78c7657..74fbea4 100644 --- a/src/components/Tutorial/Builder/Builder.js +++ b/src/components/Tutorial/Builder/Builder.js @@ -2,16 +2,15 @@ import React, { Component } from 'react'; import PropTypes from 'prop-types'; import { connect } from 'react-redux'; import { checkError, readJSON, jsonString, progress, tutorialId, resetTutorial as resetTutorialBuilder} from '../../../actions/tutorialBuilderActions'; -import { getTutorials, resetTutorial, deleteTutorial } from '../../../actions/tutorialActions'; +import { getTutorials, resetTutorial, deleteTutorial, tutorialProgress } from '../../../actions/tutorialActions'; import { clearMessages } from '../../../actions/messageActions'; import axios from 'axios'; import { withRouter } from 'react-router-dom'; -import { saveAs } from 'file-saver'; -import { detectWhitespacesAndReturnReadableResult } from '../../../helpers/whitespace'; import Breadcrumbs from '../../Breadcrumbs'; +import Badge from './Badge'; import Textfield from './Textfield'; import Step from './Step'; import Dialog from '../../Dialog'; @@ -43,7 +42,7 @@ const styles = (theme) => ({ marginTop: '5px', height: '40px', backgroundColor: theme.palette.error.dark, - '&:hover':{ + '&:hover': { backgroundColor: theme.palette.error.dark } } @@ -68,20 +67,29 @@ class Builder extends Component { } componentDidMount() { - this.props.getTutorials(); + this.props.tutorialProgress(); + // retrieve tutorials only if a potential user is loaded - authentication + // is finished (success or failed) + if(!this.props.authProgress){ + this.props.getTutorials(); + } } componentDidUpdate(props, state) { + if(props.authProgress !== this.props.authProgress && !this.props.authProgress){ + // authentication is completed + this.props.getTutorials(); + } if(props.message !== this.props.message){ if(this.props.message.id === 'GET_TUTORIALS_FAIL'){ // alert(this.props.message.msg); this.props.clearMessages(); } - else if(this.props.message.id === 'TUTORIAL_DELETE_SUCCESS'){ + else if (this.props.message.id === 'TUTORIAL_DELETE_SUCCESS') { this.onChange('new'); this.setState({ snackbar: true, key: Date.now(), message: `Das Tutorial wurde erfolgreich gelöscht.`, type: 'success' }); } - else if(this.props.message.id === 'TUTORIAL_DELETE_FAIL'){ + else if (this.props.message.id === 'TUTORIAL_DELETE_FAIL') { this.setState({ snackbar: true, key: Date.now(), message: `Fehler beim Löschen des Tutorials. Versuche es noch einmal.`, type: 'error' }); } } @@ -90,7 +98,7 @@ class Builder extends Component { componentWillUnmount() { this.resetFull(); this.props.resetTutorial(); - if(this.props.message.msg){ + if (this.props.message.msg) { this.props.clearMessages(); } } @@ -143,12 +151,12 @@ class Builder extends Component { onChange = (value) => { this.props.resetTutorialBuilder(); this.props.tutorialId(''); - this.setState({tutorial: value}); + this.setState({ tutorial: value }); } onChangeId = (value) => { this.props.tutorialId(value); - if(this.state.tutorial === 'change'){ + if (this.state.tutorial === 'change') { this.props.progress(true); var tutorial = this.props.tutorials.filter(tutorial => tutorial._id === value)[0]; this.props.readJSON(tutorial); @@ -181,13 +189,18 @@ class Builder extends Component { var steps = this.props.steps; var newTutorial = new FormData(); newTutorial.append('title', this.props.title); - newTutorial.append('badge', this.props.badge); + if(this.props.badge){ + newTutorial.append('badge', this.props.badge); + } steps.forEach((step, i) => { + if(step._id){ + newTutorial.append(`steps[${i}][_id]`, step._id); + } newTutorial.append(`steps[${i}][type]`, step.type); newTutorial.append(`steps[${i}][headline]`, step.headline); newTutorial.append(`steps[${i}][text]`, step.text); - if(i === 0 && step.type === 'instruction'){ - if(step.requirements){ // optional + if (i === 0 && step.type === 'instruction') { + if (step.requirements) { // optional step.requirements.forEach((requirement, j) => { newTutorial.append(`steps[${i}][requirements][${j}]`, requirement); }); @@ -196,14 +209,14 @@ class Builder extends Component { newTutorial.append(`steps[${i}][hardware][${j}]`, hardware); }); } - if(step.xml){ // optional + if (step.xml) { // optional newTutorial.append(`steps[${i}][xml]`, step.xml); } - if(step.media){ // optional - if(step.media.youtube){ + if (step.media) { // optional + if (step.media.youtube) { newTutorial.append(`steps[${i}][media][youtube]`, step.media.youtube); } - if(step.media.picture){ + if (step.media.picture) { newTutorial.append(`steps[${i}][media][picture]`, step.media.picture); } } @@ -215,14 +228,22 @@ class Builder extends Component { submitNew = () => { var newTutorial = this.submit(); if(newTutorial){ - axios.post(`${process.env.REACT_APP_BLOCKLY_API}/tutorial/`, newTutorial) - .then(res => { + const config = { + success: res => { var tutorial = res.data.tutorial; this.props.history.push(`/tutorial/${tutorial._id}`); - }) - .catch(err => { + }, + error: err => { this.setState({ snackbar: true, key: Date.now(), message: `Fehler beim Erstellen des Tutorials. Versuche es noch einmal.`, type: 'error' }); window.scrollTo(0, 0); + } + }; + axios.post(`${process.env.REACT_APP_BLOCKLY_API}/tutorial/`, newTutorial, config) + .then(res => { + res.config.success(res); + }) + .catch(err => { + err.config.error(err); }); } } @@ -230,14 +251,22 @@ class Builder extends Component { submitUpdate = () => { var updatedTutorial = this.submit(); if(updatedTutorial){ - axios.put(`${process.env.REACT_APP_BLOCKLY_API}/tutorial/${this.props.id}`, updatedTutorial) - .then(res => { + const config = { + success: res => { var tutorial = res.data.tutorial; this.props.history.push(`/tutorial/${tutorial._id}`); - }) - .catch(err => { + }, + error: err => { this.setState({ snackbar: true, key: Date.now(), message: `Fehler beim Ändern des Tutorials. Versuche es noch einmal.`, type: 'error' }); window.scrollTo(0, 0); + } + }; + axios.put(`${process.env.REACT_APP_BLOCKLY_API}/tutorial/${this.props.id}`, updatedTutorial, config) + .then(res => { + res.config.success(res); + }) + .catch(err => { + err.config.error(err); }); } } @@ -251,30 +280,30 @@ class Builder extends Component {

Tutorial-Builder

this.onChange(e.target.value)}> - } label="neues Tutorial erstellen" labelPlacement="end" /> {filteredTutorials.length > 0 ? -
- } - label="bestehendes Tutorial ändern" - labelPlacement="end" - /> - } - label="bestehendes Tutorial löschen" - labelPlacement="end" - /> -
- : null} +
+ } + label="bestehendes Tutorial ändern" + labelPlacement="end" + /> + } + label="bestehendes Tutorial löschen" + labelPlacement="end" + /> +
+ : null}
@@ -294,7 +323,7 @@ class Builder extends Component {
- : + : Tutorial