Merge branch 'add-translations' of https://github.com/sensebox/React-Ardublockly into add-translations
This commit is contained in:
		
						commit
						5a7e62adcf
					
				| @ -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 => { | ||||
|       var data = type === 'share' ? 'content' : type; | ||||
|       var project = res.data[data]; | ||||
|       if(project){ | ||||
|       if (project) { | ||||
|         dispatch({ | ||||
|           type: GET_PROJECT, | ||||
|           payload: project | ||||
| @ -34,24 +33,24 @@ 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 => { | ||||
|       if(err.response){ | ||||
|       if (err.response) { | ||||
|         dispatch(returnErrors(err.response.data.message, err.response.status, 'GET_PROJECT_FAIL')); | ||||
|       } | ||||
|       dispatch({type: PROJECT_PROGRESS}); | ||||
|       dispatch({ type: PROJECT_PROGRESS }); | ||||
|     }); | ||||
| }; | ||||
| 
 | ||||
| export const getProjects = (type) => (dispatch) => { | ||||
|   dispatch({type: PROJECT_PROGRESS}); | ||||
|   dispatch({ type: PROJECT_PROGRESS }); | ||||
|   axios.get(`${process.env.REACT_APP_BLOCKLY_API}/${type}`) | ||||
|     .then(res => { | ||||
|       var data = type === 'project' ? 'projects' : 'galleries'; | ||||
| @ -60,14 +59,14 @@ export const getProjects = (type) => (dispatch) => { | ||||
|         type: GET_PROJECTS, | ||||
|         payload: projects | ||||
|       }); | ||||
|       dispatch({type: PROJECT_PROGRESS}); | ||||
|       dispatch({ type: PROJECT_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_PROJECTS_FAIL')); | ||||
|       } | ||||
|       dispatch({type: PROJECT_PROGRESS}); | ||||
|       dispatch({ type: PROJECT_PROGRESS }); | ||||
|     }); | ||||
| }; | ||||
| 
 | ||||
| @ -78,7 +77,7 @@ 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) | ||||
| @ -91,15 +90,15 @@ 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 => { | ||||
|       if(err.response){ | ||||
|         if(type === 'project'){ | ||||
|       if (err.response) { | ||||
|         if (type === 'project') { | ||||
|           dispatch(returnErrors(err.response.data.message, err.response.status, 'PROJECT_UPDATE_FAIL')); | ||||
|         } else { | ||||
|           dispatch(returnErrors(err.response.data.message, err.response.status, 'GALLERY_UPDATE_FAIL')); | ||||
| @ -119,14 +118,14 @@ 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')); | ||||
|       } | ||||
|     }) | ||||
|     .catch(err => { | ||||
|       if(err.response){ | ||||
|       if (err.response) { | ||||
|         dispatch(returnErrors(err.response.data.message, err.response.status, 'PROJECT_DELETE_FAIL')); | ||||
|       } | ||||
|     }); | ||||
| @ -137,7 +136,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 +144,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 +156,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')); | ||||
|       } | ||||
|     }); | ||||
|  | ||||
| @ -1,10 +1,10 @@ | ||||
| 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) => { | ||||
|   dispatch({type: TUTORIAL_PROGRESS}); | ||||
|   dispatch({ type: TUTORIAL_PROGRESS }); | ||||
|   axios.get(`${process.env.REACT_APP_BLOCKLY_API}/tutorial/${id}`) | ||||
|     .then(res => { | ||||
|       var tutorial = res.data.tutorial; | ||||
| @ -13,7 +13,7 @@ export const getTutorial = (id) => (dispatch, getState) => { | ||||
|           type: TUTORIAL_SUCCESS, | ||||
|           payload: status | ||||
|         }); | ||||
|         dispatch({type: TUTORIAL_PROGRESS}); | ||||
|         dispatch({ type: TUTORIAL_PROGRESS }); | ||||
|         dispatch({ | ||||
|           type: GET_TUTORIAL, | ||||
|           payload: tutorial | ||||
| @ -22,15 +22,15 @@ export const getTutorial = (id) => (dispatch, getState) => { | ||||
|       }); | ||||
|     }) | ||||
|     .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}); | ||||
|   dispatch({ type: TUTORIAL_PROGRESS }); | ||||
|   axios.get(`${process.env.REACT_APP_BLOCKLY_API}/tutorial`) | ||||
|     .then(res => { | ||||
|       var tutorials = res.data.tutorials; | ||||
| @ -44,15 +44,15 @@ export const getTutorials = () => (dispatch, getState) => { | ||||
|           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 }); | ||||
|     }); | ||||
| }; | ||||
| 
 | ||||
| @ -69,7 +69,7 @@ export const assigneBadge = (id) => (dispatch, getState) => { | ||||
|       dispatch(returnSuccess(badge, res.status, 'ASSIGNE_BADGE_SUCCESS')); | ||||
|     }) | ||||
|     .catch(err => { | ||||
|       if(err.response){ | ||||
|       if (err.response) { | ||||
|         dispatch(returnErrors(err.response.data.message, err.response.status, 'ASSIGNE_BADGE_FAIL')); | ||||
|       } | ||||
|     }); | ||||
| @ -90,7 +90,7 @@ export const deleteTutorial = (id) => (dispatch, getState) => { | ||||
|       dispatch(returnSuccess(res.data.message, res.status, 'TUTORIAL_DELETE_SUCCESS')); | ||||
|     }) | ||||
|     .catch(err => { | ||||
|       if(err.response){ | ||||
|       if (err.response) { | ||||
|         dispatch(returnErrors(err.response.data.message, err.response.status, 'TUTORIAL_DELETE_FAIL')); | ||||
|       } | ||||
|     }); | ||||
| @ -128,7 +128,7 @@ export const tutorialCheck = (status, step) => (dispatch, getState) => { | ||||
|     payload: tutorialsStatus | ||||
|   }); | ||||
|   dispatch(tutorialChange()); | ||||
|   dispatch(returnSuccess('','','TUTORIAL_CHECK_SUCCESS')); | ||||
|   dispatch(returnSuccess('', '', 'TUTORIAL_CHECK_SUCCESS')); | ||||
| }; | ||||
| 
 | ||||
| export const storeTutorialXml = (code) => (dispatch, getState) => { | ||||
| @ -161,9 +161,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; | ||||
| @ -180,7 +180,7 @@ const existingTutorials = (tutorials, status) => new Promise(function(resolve, r | ||||
|   }); | ||||
| }); | ||||
| 
 | ||||
| const existingTutorial = (tutorial, status) => new Promise(function(resolve, reject){ | ||||
| const existingTutorial = (tutorial, status) => new Promise(function (resolve, reject) { | ||||
|   var tutorialsId = tutorial._id; | ||||
|   var statusIndex = status.findIndex(status => status._id === tutorialsId); | ||||
|   if (statusIndex > -1) { | ||||
|  | ||||
| @ -859,7 +859,7 @@ 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 | ||||
| @ -879,6 +879,10 @@ Blockly.Msg.messages_SHARE_FAIL = "Fehler beim Erstellen eines Links zum Teilen | ||||
| 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 | ||||
|  */ | ||||
| @ -912,6 +916,8 @@ 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" | ||||
| 
 | ||||
| 
 | ||||
| /** | ||||
| @ -944,6 +950,21 @@ Blockly.Msg.notfound_head = "Die von Ihnen angeforderte Seite kann nicht gefunde | ||||
| 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' | ||||
| 
 | ||||
| /** | ||||
|  * 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 | ||||
|  */ | ||||
| @ -951,6 +972,27 @@ Blockly.Msg.notfound_text = "Die gesuchte Seite wurde möglicherweise entfernt, | ||||
| 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_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" | ||||
| export const De = Blockly.Msg; | ||||
|  | ||||
| @ -21,7 +21,7 @@ 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"; | ||||
| 
 | ||||
| const styles = (theme) => ({ | ||||
| @ -101,46 +101,46 @@ class Navbar extends Component { | ||||
|           </div> | ||||
|           <List> | ||||
|             {[{ 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( | ||||
|                     <Link to={item.link} key={index} style={{ textDecoration: 'none', color: 'inherit' }}> | ||||
|                       <ListItem button onClick={this.toggleDrawer}> | ||||
|                         <ListItemIcon><FontAwesomeIcon icon={item.icon} /></ListItemIcon> | ||||
|                         <ListItemText primary={item.text} /> | ||||
|                       </ListItem> | ||||
|                     </Link> | ||||
|                   ); | ||||
|                 } | ||||
|             { 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 ( | ||||
|                   <Link to={item.link} key={index} style={{ textDecoration: 'none', color: 'inherit' }}> | ||||
|                     <ListItem button onClick={this.toggleDrawer}> | ||||
|                       <ListItemIcon><FontAwesomeIcon icon={item.icon} /></ListItemIcon> | ||||
|                       <ListItemText primary={item.text} /> | ||||
|                     </ListItem> | ||||
|                   </Link> | ||||
|                 ); | ||||
|               } | ||||
|             } | ||||
|             )} | ||||
|           </List> | ||||
|           <Divider classes={{ root: this.props.classes.appBarColor }} style={{ marginTop: 'auto' }} /> | ||||
|           <List> | ||||
|             {[{ 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( | ||||
|                     <Link to={item.link} key={index} style={{ textDecoration: 'none', color: 'inherit' }}> | ||||
|                       <ListItem button onClick={item.function ? () => {item.function(); this.toggleDrawer();} : this.toggleDrawer}> | ||||
|                         <ListItemIcon><FontAwesomeIcon icon={item.icon} /></ListItemIcon> | ||||
|                         <ListItemText primary={item.text} /> | ||||
|                       </ListItem> | ||||
|                     </Link> | ||||
|                   ); | ||||
|                 } | ||||
|             { 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 ( | ||||
|                   <Link to={item.link} key={index} style={{ textDecoration: 'none', color: 'inherit' }}> | ||||
|                     <ListItem button onClick={item.function ? () => { item.function(); this.toggleDrawer(); } : this.toggleDrawer}> | ||||
|                       <ListItemIcon><FontAwesomeIcon icon={item.icon} /></ListItemIcon> | ||||
|                       <ListItemText primary={item.text} /> | ||||
|                     </ListItem> | ||||
|                   </Link> | ||||
|                 ); | ||||
|               } | ||||
|             } | ||||
|             )} | ||||
|           </List> | ||||
|         </Drawer> | ||||
|         {this.props.tutorialIsLoading || this.props.projectIsLoading ? | ||||
|           <LinearProgress style={{marginBottom: '30px', boxShadow: '0px 2px 4px -1px rgba(0,0,0,0.2),0px 4px 5px 0px rgba(0,0,0,0.14),0px 1px 10px 0px rgba(0,0,0,0.12)'}}/> | ||||
|         : null} | ||||
|           <LinearProgress style={{ marginBottom: '30px', boxShadow: '0px 2px 4px -1px rgba(0,0,0,0.2),0px 4px 5px 0px rgba(0,0,0,0.14),0px 1px 10px 0px rgba(0,0,0,0.12)' }} /> | ||||
|           : null} | ||||
|       </div> | ||||
|     ); | ||||
|   } | ||||
|  | ||||
| @ -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 { | ||||
|         <Backdrop open invisible> | ||||
|           <CircularProgress color="primary" /> | ||||
|         </Backdrop> | ||||
|       : this.props.project ? | ||||
|         <div> | ||||
|           {this.props.type !== 'share' ? | ||||
|             <Breadcrumbs content={[{ link: `/${this.props.type}`, title: data },{ link: this.props.location.pathname, title: this.props.project.title }]} /> | ||||
|           : null} | ||||
|           <Home project={this.props.project} projectType={this.props.type}/> | ||||
|         </div> : null | ||||
|         : this.props.project ? | ||||
|           <div> | ||||
|             {this.props.type !== 'share' ? | ||||
|               <Breadcrumbs content={[{ link: `/${this.props.type}`, title: data }, { link: this.props.location.pathname, title: this.props.project.title }]} /> | ||||
|               : null} | ||||
|             <Home project={this.props.project} projectType={this.props.type} /> | ||||
|           </div> : null | ||||
|     ); | ||||
|   }; | ||||
| } | ||||
|  | ||||
| @ -4,7 +4,7 @@ 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 +42,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 +88,7 @@ class ProjectHome extends Component { | ||||
|           <Backdrop open invisible> | ||||
|             <CircularProgress color="primary" /> | ||||
|           </Backdrop> | ||||
|         : | ||||
|           : | ||||
|           <div> | ||||
|             {this.props.projects.length > 0 ? | ||||
|               <Grid container spacing={2}> | ||||
| @ -97,37 +97,37 @@ class ProjectHome extends Component { | ||||
|                     <Grid item xs={12} sm={6} md={4} xl={3} key={i}> | ||||
|                       <Paper style={{ padding: '1rem', position: 'relative', overflow: 'hidden' }}> | ||||
|                         <Link to={`/${data === 'Projekte' ? 'project' : 'gallery'}/${project._id}`} style={{ textDecoration: 'none', color: 'inherit' }}> | ||||
|                           <h3 style={{marginTop: 0}}>{project.title}</h3> | ||||
|                           <Divider style={{marginTop: '1rem', marginBottom: '10px'}}/> | ||||
|                           <h3 style={{ marginTop: 0 }}>{project.title}</h3> | ||||
|                           <Divider style={{ marginTop: '1rem', marginBottom: '10px' }} /> | ||||
|                           <BlocklyWindow | ||||
|                             svg | ||||
|                             blockDisabled | ||||
|                             initialXml={project.xml} | ||||
|                           /> | ||||
|                           <Typography variant='body2' style={{fontStyle: 'italic', margin: 0, marginTop: '-10px'}}>{project.description}</Typography> | ||||
|                           <Typography variant='body2' style={{ fontStyle: 'italic', margin: 0, marginTop: '-10px' }}>{project.description}</Typography> | ||||
|                         </Link> | ||||
|                         {this.props.user && this.props.user.email === project.creator ? | ||||
|                           <div> | ||||
|                             <Divider style={{marginTop: '10px', marginBottom: '10px'}}/> | ||||
|                             <div style={{float: 'right'}}> | ||||
|                             <Divider style={{ marginTop: '10px', marginBottom: '10px' }} /> | ||||
|                             <div style={{ float: 'right' }}> | ||||
|                               <WorkspaceFunc | ||||
|                                 multiple | ||||
|                                 project={project} | ||||
|                                 projectType={this.props.location.pathname.replace('/','')} | ||||
|                                 projectType={this.props.location.pathname.replace('/', '')} | ||||
|                               /> | ||||
|                             </div> | ||||
|                           </div> | ||||
|                       : null} | ||||
|                           : null} | ||||
|                       </Paper> | ||||
|                     </Grid> | ||||
|                   ) | ||||
|                 })} | ||||
|               </Grid> | ||||
|             : <div> | ||||
|                 <Typography style={{marginBottom: '10px'}}>Es sind aktuell keine Projekte vorhanden.</Typography> | ||||
|                 {this.props.location.pathname.replace('/','') === 'project' ? | ||||
|               : <div> | ||||
|                 <Typography style={{ marginBottom: '10px' }}>Es sind aktuell keine Projekte vorhanden.</Typography> | ||||
|                 {this.props.location.pathname.replace('/', '') === 'project' ? | ||||
|                   <Typography>Erstelle jetzt dein <Link to={'/'} className={this.props.classes.link}>eigenes Projekt</Link> oder lasse dich von Projektbeispielen in der <Link to={'/gallery'} className={this.props.classes.link}>Galerie</Link> inspirieren.</Typography> | ||||
|                 : null} | ||||
|                   : null} | ||||
|               </div> | ||||
|             } | ||||
|           </div> | ||||
|  | ||||
| @ -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} | ||||
|       > | ||||
|         <div style={{ marginTop: '10px' }}> | ||||
|           <Paper style={{textAlign: 'center'}}> | ||||
|           <Paper style={{ textAlign: 'center' }}> | ||||
|             {this.props.message.msg.image && this.props.message.msg.image.path ? | ||||
|               <Avatar src={`${process.env.REACT_APP_MYBADGES}/media/${this.props.message.msg.image.path}`} style={{width: '200px', height: '200px', marginLeft: 'auto', marginRight: 'auto'}}/> | ||||
|             : <Avatar style={{width: '200px', height: '200px', marginLeft: 'auto', marginRight: 'auto'}}></Avatar>} | ||||
|             <Typography variant='h6' style={{display: 'flex', cursor: 'default', paddingBottom: '6px'}}> | ||||
|               <div style={{flexGrow:1, marginLeft: '10px', marginRight: '10px'}}>{this.props.message.msg.name}</div> | ||||
|               <Avatar src={`${process.env.REACT_APP_MYBADGES}/media/${this.props.message.msg.image.path}`} style={{ width: '200px', height: '200px', marginLeft: 'auto', marginRight: 'auto' }} /> | ||||
|               : <Avatar style={{ width: '200px', height: '200px', marginLeft: 'auto', marginRight: 'auto' }}></Avatar>} | ||||
|             <Typography variant='h6' style={{ display: 'flex', cursor: 'default', paddingBottom: '6px' }}> | ||||
|               <div style={{ flexGrow: 1, marginLeft: '10px', marginRight: '10px' }}>{this.props.message.msg.name}</div> | ||||
|             </Typography> | ||||
|           </Paper> | ||||
|           <Typography style={{marginTop: '10px'}}> | ||||
|             Eine Übersicht über alle erhaltenen Badges im Kontext Blockly for senseBox findest du <Link to={'/user/badge'} className={this.props.classes.link}>hier</Link>. | ||||
|           <Typography style={{ marginTop: '10px' }}> | ||||
|             {Blockly.Msg.badges_explaination}<Link to={'/user/badge'} className={this.props.classes.link}>{Blockly.Msg.labels_here}</Link>. | ||||
|           </Typography> | ||||
|         </div> | ||||
|       </Dialog> | ||||
|  | ||||
| @ -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 ( | ||||
|       <div style={{marginBottom: '10px', padding: '18.5px 14px', borderRadius: '25px', border: '1px solid lightgrey', width: 'calc(100% - 28px)'}}> | ||||
|       <div style={{ marginBottom: '10px', padding: '18.5px 14px', borderRadius: '25px', border: '1px solid lightgrey', width: 'calc(100% - 28px)' }}> | ||||
|         {!this.props.task ? | ||||
|           <FormControlLabel | ||||
|             labelPlacement="end" | ||||
| @ -126,41 +126,42 @@ class BlocklyExample extends Component { | ||||
|               /> | ||||
|             } | ||||
|           /> | ||||
|         : <FormLabel style={{color: 'black'}}>Musterlösung</FormLabel>} | ||||
|           : <FormLabel style={{ color: 'black' }}>{Blockly.Msg.builder_solution}</FormLabel>} | ||||
|         {this.state.checked ? !this.props.value || this.props.error ? | ||||
|           <FormHelperText style={{lineHeight: 'initial'}} className={this.props.classes.errorColor}>{`Reiche deine Blöcke ein, indem du auf den '${this.props.task ? 'Musterlösung einreichen' : 'Beispiel einreichen'}'-Button klickst.`}</FormHelperText> | ||||
|         : this.state.input ? <FormHelperText style={{lineHeight: 'initial'}}>Die letzte Einreichung erfolgte um {this.state.input} Uhr.</FormHelperText> : null | ||||
|         : null} | ||||
|           <FormHelperText style={{ lineHeight: 'initial' }} className={this.props.classes.errorColor}>{`Reiche deine Blöcke ein, indem du auf den '${this.props.task ? Blockly.Msg.builder_solution_submit : Blockly.Msg.builder_example_submit}'-Button klickst.`}</FormHelperText> | ||||
|           : this.state.input ? <FormHelperText style={{ lineHeight: 'initial' }}>Die letzte Einreichung erfolgte um {this.state.input} Uhr.</FormHelperText> : null | ||||
|           : null} | ||||
|         {this.state.checked && !this.props.task ? | ||||
|           <FormHelperText style={{lineHeight: 'initial'}}>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.</FormHelperText> | ||||
|         : null} | ||||
|           <FormHelperText style={{ lineHeight: 'initial' }}>{Blockly.Msg.builder_comment}</FormHelperText> | ||||
|           : null} | ||||
|         {/* ensure that the correct xml-file is displayed in the workspace */} | ||||
|         {this.state.checked && this.state.xml? (() => { | ||||
|           return( | ||||
|             <div style={{marginTop: '10px'}}> | ||||
|         {this.state.checked && this.state.xml ? (() => { | ||||
|           return ( | ||||
|             <div style={{ marginTop: '10px' }}> | ||||
|               <Grid container className={!this.props.value || this.props.error ? this.props.classes.errorBorder : null}> | ||||
|                 <Grid item xs={12}> | ||||
|                   <BlocklyWindow | ||||
|                     blockDisabled={this.props.task} | ||||
|                     trashcan={false} | ||||
|                     initialXml={this.state.xml} | ||||
|                     blocklyCSS={{height: '500px'}} | ||||
|                     blocklyCSS={{ height: '500px' }} | ||||
|                   /> | ||||
|                 </Grid> | ||||
|               </Grid> | ||||
|               <Button | ||||
|                 className={!this.props.value || this.props.error ? this.props.classes.errorButton : null } | ||||
|                 style={{marginTop: '5px', height: '40px'}} | ||||
|                 className={!this.props.value || this.props.error ? this.props.classes.errorButton : null} | ||||
|                 style={{ marginTop: '5px', height: '40px' }} | ||||
|                 variant='contained' | ||||
|                 color='primary' | ||||
|                 disabled={this.state.disabled} | ||||
|                 onClick={() => this.setXml()} | ||||
|               > | ||||
|                 {this.props.task ? 'Musterlösung einreichen' : 'Beispiel einreichen'} | ||||
|                 {this.props.task ? Blockly.Msg.builder_solution_submit : Blockly.Msg.builder_example_submit} | ||||
|               </Button> | ||||
|             </div> | ||||
|           )})() | ||||
|         : null} | ||||
|           ) | ||||
|         })() | ||||
|           : null} | ||||
|       </div> | ||||
|     ); | ||||
|   }; | ||||
| @ -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)); | ||||
|  | ||||
| @ -1,15 +1,13 @@ | ||||
| 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 { checkError, readJSON, jsonString, progress, tutorialId, resetTutorial as resetTutorialBuilder } from '../../../actions/tutorialBuilderActions'; | ||||
| import { getTutorials, resetTutorial, deleteTutorial } 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 Textfield from './Textfield'; | ||||
| @ -43,7 +41,7 @@ const styles = (theme) => ({ | ||||
|     marginTop: '5px', | ||||
|     height: '40px', | ||||
|     backgroundColor: theme.palette.error.dark, | ||||
|     '&:hover':{ | ||||
|     '&:hover': { | ||||
|       backgroundColor: theme.palette.error.dark | ||||
|     } | ||||
|   } | ||||
| @ -72,16 +70,16 @@ class Builder extends Component { | ||||
|   } | ||||
| 
 | ||||
|   componentDidUpdate(props, state) { | ||||
|     if(props.message !== this.props.message){ | ||||
|       if(this.props.message.id === 'GET_TUTORIALS_FAIL'){ | ||||
|     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 +88,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 +141,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); | ||||
| @ -186,8 +184,8 @@ class Builder extends Component { | ||||
|         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 +194,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); | ||||
|           } | ||||
|         } | ||||
| @ -214,7 +212,7 @@ class Builder extends Component { | ||||
| 
 | ||||
|   submitNew = () => { | ||||
|     var newTutorial = this.submit(); | ||||
|     if(newTutorial){ | ||||
|     if (newTutorial) { | ||||
|       axios.post(`${process.env.REACT_APP_BLOCKLY_API}/tutorial/`, newTutorial) | ||||
|         .then(res => { | ||||
|           var tutorial = res.data.tutorial; | ||||
| @ -229,7 +227,7 @@ class Builder extends Component { | ||||
| 
 | ||||
|   submitUpdate = () => { | ||||
|     var updatedTutorial = this.submit(); | ||||
|     if(updatedTutorial){ | ||||
|     if (updatedTutorial) { | ||||
|       axios.put(`${process.env.REACT_APP_BLOCKLY_API}/tutorial/${this.props.id}`, updatedTutorial) | ||||
|         .then(res => { | ||||
|           var tutorial = res.data.tutorial; | ||||
| @ -251,30 +249,30 @@ class Builder extends Component { | ||||
|         <h1>Tutorial-Builder</h1> | ||||
| 
 | ||||
|         <RadioGroup row value={this.state.tutorial} onChange={(e) => this.onChange(e.target.value)}> | ||||
|           <FormControlLabel style={{color: 'black'}} | ||||
|           <FormControlLabel style={{ color: 'black' }} | ||||
|             value="new" | ||||
|             control={<Radio color="primary" />} | ||||
|             label="neues Tutorial erstellen" | ||||
|             labelPlacement="end" | ||||
|           /> | ||||
|           {filteredTutorials.length > 0 ? | ||||
|           <div> | ||||
|             <FormControlLabel style={{color: 'black'}} | ||||
|               disabled={this.props.index === 0} | ||||
|               value="change" | ||||
|               control={<Radio color="primary" />} | ||||
|               label="bestehendes Tutorial ändern" | ||||
|               labelPlacement="end" | ||||
|             /> | ||||
|             <FormControlLabel style={{color: 'black'}} | ||||
|               disabled={this.props.index === 0} | ||||
|               value="delete" | ||||
|               control={<Radio color="primary" />} | ||||
|               label="bestehendes Tutorial löschen" | ||||
|               labelPlacement="end" | ||||
|             /> | ||||
|           </div> | ||||
|         : null} | ||||
|             <div> | ||||
|               <FormControlLabel style={{ color: 'black' }} | ||||
|                 disabled={this.props.index === 0} | ||||
|                 value="change" | ||||
|                 control={<Radio color="primary" />} | ||||
|                 label="bestehendes Tutorial ändern" | ||||
|                 labelPlacement="end" | ||||
|               /> | ||||
|               <FormControlLabel style={{ color: 'black' }} | ||||
|                 disabled={this.props.index === 0} | ||||
|                 value="delete" | ||||
|                 control={<Radio color="primary" />} | ||||
|                 label="bestehendes Tutorial löschen" | ||||
|                 labelPlacement="end" | ||||
|               /> | ||||
|             </div> | ||||
|             : null} | ||||
|         </RadioGroup> | ||||
| 
 | ||||
|         <Divider variant='fullWidth' style={{ margin: '10px 0 15px 0' }} /> | ||||
| @ -294,7 +292,7 @@ class Builder extends Component { | ||||
|             </label> | ||||
|             <Button style={{ marginRight: '10px', marginBottom: '10px' }} variant='contained' color='primary' onClick={() => this.uploadJsonString()}>String laden</Button> | ||||
|           </div> | ||||
|         : <FormControl variant="outlined" style={{width: '100%'}}> | ||||
|           : <FormControl variant="outlined" style={{ width: '100%' }}> | ||||
|             <InputLabel id="select-outlined-label">Tutorial</InputLabel> | ||||
|             <Select | ||||
|               color='primary' | ||||
| @ -313,37 +311,37 @@ class Builder extends Component { | ||||
|         <Divider variant='fullWidth' style={{ margin: '10px 0 15px 0' }} /> | ||||
| 
 | ||||
|         {this.state.tutorial === 'new' || (this.state.tutorial === 'change' && this.props.id !== '') ? | ||||
|         /*Tutorial-Builder-Form*/ | ||||
|         <div> | ||||
|           {this.props.error.type ? | ||||
|             <FormHelperText style={{ lineHeight: 'initial' }} className={this.props.classes.errorColor}>{`Ein Tutorial muss mindestens jeweils eine Instruktion und eine Aufgabe enthalten.`}</FormHelperText> | ||||
|             : null} | ||||
|           {/* <Id error={this.props.error.id} value={this.props.id} /> */} | ||||
|           <Textfield value={this.props.title} property={'title'} label={'Titel'} error={this.props.error.title} /> | ||||
|           <Textfield value={this.props.badge} property={'badge'} label={'Badge'} /> | ||||
|           /*Tutorial-Builder-Form*/ | ||||
|           <div> | ||||
|             {this.props.error.type ? | ||||
|               <FormHelperText style={{ lineHeight: 'initial' }} className={this.props.classes.errorColor}>{`Ein Tutorial muss mindestens jeweils eine Instruktion und eine Aufgabe enthalten.`}</FormHelperText> | ||||
|               : null} | ||||
|             {/* <Id error={this.props.error.id} value={this.props.id} /> */} | ||||
|             <Textfield value={this.props.title} property={'title'} label={'Titel'} error={this.props.error.title} /> | ||||
|             <Textfield value={this.props.badge} property={'badge'} label={'Badge'} /> | ||||
| 
 | ||||
|           {this.props.steps.map((step, i) => | ||||
|             <Step step={step} index={i} key={i} /> | ||||
|           )} | ||||
|             {this.props.steps.map((step, i) => | ||||
|               <Step step={step} index={i} key={i} /> | ||||
|             )} | ||||
| 
 | ||||
|           {/*submit or reset*/} | ||||
|           <Divider variant='fullWidth' style={{ margin: '30px 0 10px 0' }} /> | ||||
|           {this.state.tutorial === 'new' ? | ||||
|             <div> | ||||
|               <Button style={{ marginRight: '10px', marginTop: '10px' }} variant='contained' color='primary' onClick={() => this.submitNew()}>Tutorial erstellen</Button> | ||||
|               <Button style={{ marginTop: '10px' }} variant='contained' onClick={() => this.resetFull()}>Zurücksetzen</Button> | ||||
|             </div> | ||||
|           : <div> | ||||
|             <Button style={{ marginRight: '10px', marginTop: '10px' }} variant='contained' color='primary' onClick={() => this.submitUpdate()}>Tutorial ändern</Button> | ||||
|             <Button style={{ marginTop: '10px' }} variant='contained' onClick={() => this.resetTutorial()}>Zurücksetzen</Button> | ||||
|             {/*submit or reset*/} | ||||
|             <Divider variant='fullWidth' style={{ margin: '30px 0 10px 0' }} /> | ||||
|             {this.state.tutorial === 'new' ? | ||||
|               <div> | ||||
|                 <Button style={{ marginRight: '10px', marginTop: '10px' }} variant='contained' color='primary' onClick={() => this.submitNew()}>Tutorial erstellen</Button> | ||||
|                 <Button style={{ marginTop: '10px' }} variant='contained' onClick={() => this.resetFull()}>Zurücksetzen</Button> | ||||
|               </div> | ||||
|               : <div> | ||||
|                 <Button style={{ marginRight: '10px', marginTop: '10px' }} variant='contained' color='primary' onClick={() => this.submitUpdate()}>Tutorial ändern</Button> | ||||
|                 <Button style={{ marginTop: '10px' }} variant='contained' onClick={() => this.resetTutorial()}>Zurücksetzen</Button> | ||||
|               </div> | ||||
|             } | ||||
| 
 | ||||
|             <Backdrop className={this.props.classes.backdrop} open={this.props.isProgress}> | ||||
|               <CircularProgress color="inherit" /> | ||||
|             </Backdrop> | ||||
|           </div> | ||||
|           } | ||||
| 
 | ||||
|           <Backdrop className={this.props.classes.backdrop} open={this.props.isProgress}> | ||||
|             <CircularProgress color="inherit" /> | ||||
|           </Backdrop> | ||||
|         </div> | ||||
|         : null} | ||||
|           : null} | ||||
| 
 | ||||
|         {this.state.tutorial === 'delete' && this.props.id !== '' ? | ||||
|           <Button | ||||
| @ -351,7 +349,7 @@ class Builder extends Component { | ||||
|             variant='contained' | ||||
|             color='primary' | ||||
|             onClick={() => this.props.deleteTutorial()}>Tutorial löschen</Button> | ||||
|         : null} | ||||
|           : null} | ||||
| 
 | ||||
|         <Dialog | ||||
|           open={this.state.open} | ||||
| @ -406,7 +404,6 @@ Builder.propTypes = { | ||||
|   change: PropTypes.number.isRequired, | ||||
|   error: PropTypes.object.isRequired, | ||||
|   json: PropTypes.string.isRequired, | ||||
|   badge: PropTypes.string.isRequired, | ||||
|   isProgress: PropTypes.bool.isRequired, | ||||
|   tutorials: PropTypes.array.isRequired, | ||||
|   message: PropTypes.object.isRequired, | ||||
|  | ||||
| @ -12,6 +12,8 @@ import GridListTile from '@material-ui/core/GridListTile'; | ||||
| import GridListTileBar from '@material-ui/core/GridListTileBar'; | ||||
| import FormHelperText from '@material-ui/core/FormHelperText'; | ||||
| import FormLabel from '@material-ui/core/FormLabel'; | ||||
| import * as Blockly from 'blockly' | ||||
| 
 | ||||
| 
 | ||||
| const styles = theme => ({ | ||||
|   multiGridListTile: { | ||||
| @ -67,8 +69,8 @@ class Requirements extends Component { | ||||
|     return ( | ||||
|       <div style={{ marginBottom: '10px', padding: '18.5px 14px', borderRadius: '25px', border: '1px solid lightgrey', width: 'calc(100% - 28px)' }}> | ||||
|         <FormLabel style={{ color: 'black' }}>Hardware</FormLabel> | ||||
|         <FormHelperText style={this.props.error ? { lineHeight: 'initial', marginTop: '5px' } : { marginTop: '5px', lineHeight: 'initial', marginBottom: '10px' }}>Beachte, dass die Reihenfolge des Auswählens maßgebend ist.</FormHelperText> | ||||
|         {this.props.error ? <FormHelperText className={this.props.classes.errorColor}>Wähle mindestens eine Hardware-Komponente aus.</FormHelperText> : null} | ||||
|         <FormHelperText style={this.props.error ? { lineHeight: 'initial', marginTop: '5px' } : { marginTop: '5px', lineHeight: 'initial', marginBottom: '10px' }}>{Blockly.Msg.builder_hardware_order}</FormHelperText> | ||||
|         {this.props.error ? <FormHelperText className={this.props.classes.errorColor}>{Blockly.Msg.builder_hardware_helper}</FormHelperText> : null} | ||||
|         <GridList cellHeight={100} cols={cols} spacing={10}> | ||||
|           {hardware.map((picture, i) => ( | ||||
|             <GridListTile key={i} onClick={() => this.onChange(picture.id)} classes={{ tile: this.props.value.filter(value => value === picture.id).length > 0 ? this.props.classes.active : this.props.classes.border }}> | ||||
|  | ||||
| @ -2,8 +2,6 @@ import React, { Component } from 'react'; | ||||
| import PropTypes from 'prop-types'; | ||||
| import { connect } from 'react-redux'; | ||||
| import { changeContent } from '../../../actions/tutorialBuilderActions'; | ||||
| import { getTutorials, resetTutorial } from '../../../actions/tutorialActions'; | ||||
| import { clearMessages } from '../../../actions/messageActions'; | ||||
| 
 | ||||
| import FormGroup from '@material-ui/core/FormGroup'; | ||||
| import Checkbox from '@material-ui/core/Checkbox'; | ||||
| @ -11,7 +9,7 @@ import FormControlLabel from '@material-ui/core/FormControlLabel'; | ||||
| import FormLabel from '@material-ui/core/FormLabel'; | ||||
| import FormHelperText from '@material-ui/core/FormHelperText'; | ||||
| import FormControl from '@material-ui/core/FormControl'; | ||||
| 
 | ||||
| import * as Blockly from 'blockly' | ||||
| class Requirements extends Component { | ||||
| 
 | ||||
|   onChange = (e) => { | ||||
| @ -29,8 +27,8 @@ class Requirements extends Component { | ||||
|   render() { | ||||
|     return ( | ||||
|       <FormControl style={{ marginBottom: '10px', padding: '18.5px 14px', borderRadius: '25px', border: '1px solid lightgrey', width: 'calc(100% - 28px)' }}> | ||||
|         <FormLabel style={{ color: 'black' }}>Voraussetzungen</FormLabel> | ||||
|         <FormHelperText style={{ marginTop: '5px' }}>Beachte, dass die Reihenfolge des Anhakens maßgebend ist.</FormHelperText> | ||||
|         <FormLabel style={{ color: 'black' }}>{Blockly.Msg.builder_requirements_head}</FormLabel> | ||||
|         <FormHelperText style={{ marginTop: '5px' }}>{Blockly.Msg.builder_requirements_order}</FormHelperText> | ||||
|         <FormGroup> | ||||
|           {this.props.tutorials.filter(tutorial => tutorial._id !== this.props.id).map((tutorial, i) => | ||||
|             <FormControlLabel | ||||
|  | ||||
| @ -91,7 +91,7 @@ class Hardware extends Component { | ||||
|           content={this.state.content} | ||||
|           onClose={this.handleClose} | ||||
|           onClick={this.handleClose} | ||||
|           button={'Schließen'} | ||||
|           button={Blockly.Msg.button_close} | ||||
|         > | ||||
|           <div> | ||||
|             <img src={`/media/hardware/${this.state.hardwareInfo.src}`} width="100%" alt={this.state.hardwareInfo.name} /> | ||||
|  | ||||
| @ -46,7 +46,7 @@ class HintTutorialExists extends Component { | ||||
|       window.localStorage.setItem('news', e.target.checked); | ||||
|     } | ||||
|     else { | ||||
|       window.localStorage.deleteItem('news'); | ||||
|       window.localStorage.removeItem('news'); | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
| @ -57,30 +57,30 @@ class HintTutorialExists extends Component { | ||||
|         fullWidth | ||||
|         maxWidth={'sm'} | ||||
|         open={this.state.open} | ||||
|         title={'Neuigkeiten'} | ||||
|         title={Blockly.Msg.messages_newblockly_head} | ||||
|         content={''} | ||||
|         onClose={this.toggleDialog} | ||||
|         onClick={this.toggleDialog} | ||||
|         button={Blockly.Msg.button_close} | ||||
|       > | ||||
|         <div> | ||||
|           Es gibt ab jetzt Tutorials zu verschiedenen Themen. Schau mal <Link to="/tutorial" className={this.props.classes.link}>hier</Link> vorbei. | ||||
|           <FormControlLabel | ||||
|             style={{ marginTop: '20px' }} | ||||
|             classes={{ label: this.props.classes.label }} | ||||
|             control={ | ||||
|               <Checkbox | ||||
|                 size={'small'} | ||||
|                 value={true} | ||||
|                 checked={this.state.checked} | ||||
|                 onChange={(e) => this.onChange(e)} | ||||
|                 name="dialog" | ||||
|                 color="primary" | ||||
|               /> | ||||
|             } | ||||
|             label={'Dialog nicht mehr anzeigen'} | ||||
|           /> | ||||
|           {Blockly.Msg.messages_newblockly_text}<Link to="/tutorial" className={this.props.classes.link}>test</Link> | ||||
|         </div> | ||||
|         <FormControlLabel | ||||
|           style={{ marginTop: '20px' }} | ||||
|           classes={{ label: this.props.classes.label }} | ||||
|           control={ | ||||
|             <Checkbox | ||||
|               size={'small'} | ||||
|               value={true} | ||||
|               checked={this.state.checked} | ||||
|               onChange={(e) => this.onChange(e)} | ||||
|               name="dialog" | ||||
|               color="primary" | ||||
|             /> | ||||
|           } | ||||
|           label={Blockly.Msg.labels_donotshowagain} | ||||
|         /> | ||||
|       </Dialog> | ||||
|     ); | ||||
|   }; | ||||
|  | ||||
| @ -13,6 +13,7 @@ import Tooltip from '@material-ui/core/Tooltip'; | ||||
| 
 | ||||
| import { FontAwesomeIcon } from "@fortawesome/react-fontawesome"; | ||||
| import { faCheck, faTimes } from "@fortawesome/free-solid-svg-icons"; | ||||
| import * as Blockly from 'blockly' | ||||
| 
 | ||||
| const styles = theme => ({ | ||||
|   outerDiv: { | ||||
| @ -63,7 +64,7 @@ class Requirement extends Component { | ||||
|     var tutorialIds = requirements.map(requirement => requirement._id); | ||||
|     return ( | ||||
|       <div style={{ marginTop: '20px', marginBottom: '5px' }}> | ||||
|         <Typography>Bevor du mit diesem Tutorial fortfährst solltest du folgende Tutorials erfolgreich abgeschlossen haben:</Typography> | ||||
|         <Typography>{Blockly.Msg.tutorials_requirements}</Typography> | ||||
|         <List component="div"> | ||||
|           {tutorialIds.map((tutorialId, i) => { | ||||
|             var title = requirements[i].title | ||||
|  | ||||
| @ -18,6 +18,8 @@ import Button from '@material-ui/core/Button'; | ||||
| import { faClipboardCheck } from "@fortawesome/free-solid-svg-icons"; | ||||
| import { FontAwesomeIcon } from "@fortawesome/react-fontawesome"; | ||||
| 
 | ||||
| import * as Blockly from 'blockly' | ||||
| 
 | ||||
| const styles = (theme) => ({ | ||||
|   compile: { | ||||
|     backgroundColor: theme.palette.button.compile, | ||||
| @ -57,7 +59,7 @@ class SolutionCheck extends Component { | ||||
|     const steps = this.props.tutorial.steps; | ||||
|     return ( | ||||
|       <div> | ||||
|         <Tooltip title='Lösung kontrollieren' arrow> | ||||
|         <Tooltip title={Blockly.Msg.tooltip_check_solution} arrow> | ||||
|           <IconButton | ||||
|             className={this.props.classes.compile} | ||||
|             style={{ width: '40px', height: '40px', marginRight: '5px' }} | ||||
| @ -76,7 +78,7 @@ class SolutionCheck extends Component { | ||||
|           content={this.state.msg.text} | ||||
|           onClose={this.toggleDialog} | ||||
|           onClick={this.toggleDialog} | ||||
|           button={'Schließen'} | ||||
|           button={Blockly.Msg.button_close} | ||||
|         > | ||||
|           {this.state.msg.type === 'success' ? | ||||
|             <div style={{ marginTop: '20px', display: 'flex' }}> | ||||
| @ -88,7 +90,7 @@ class SolutionCheck extends Component { | ||||
|                   color="primary" | ||||
|                   onClick={() => { this.toggleDialog(); this.props.history.push(`/tutorial/`) }} | ||||
|                 > | ||||
|                   Tutorials-Übersicht | ||||
|                   {Blockly.Msg.button_tutorial_overview} | ||||
|                 </Button> | ||||
|                 : | ||||
|                 <Button | ||||
| @ -97,7 +99,7 @@ class SolutionCheck extends Component { | ||||
|                   color="primary" | ||||
|                   onClick={() => { this.toggleDialog(); this.props.tutorialStep(this.props.activeStep + 1) }} | ||||
|                 > | ||||
|                   nächster Schritt | ||||
|                   {Blockly.Msg.button_next} | ||||
|                 </Button> | ||||
|               } | ||||
|             </div> | ||||
|  | ||||
| @ -51,7 +51,6 @@ class StepperHorizontal extends Component { | ||||
| 
 | ||||
|   render() { | ||||
|     var tutorialId = this.props.tutorial._id; | ||||
|     var tutorialIndex = this.props.currentTutorialIndex; | ||||
|     var status = this.props.status.filter(status => status._id === tutorialId)[0]; | ||||
|     var tasks = status.tasks; | ||||
|     var error = tasks.filter(task => task.type === 'error').length > 0; | ||||
| @ -71,7 +70,7 @@ class StepperHorizontal extends Component { | ||||
|         <div className={this.props.classes.stepper}> | ||||
|           <Button | ||||
|             disabled//={tutorialIndex === 0}
 | ||||
|             //onClick={() => { this.props.history.push(`/tutorial/${tutorials[tutorialIndex - 1].id}`) }}
 | ||||
|           //onClick={() => { this.props.history.push(`/tutorial/${tutorials[tutorialIndex - 1].id}`) }}
 | ||||
|           > | ||||
|             {'<'} | ||||
|           </Button> | ||||
| @ -83,7 +82,7 @@ class StepperHorizontal extends Component { | ||||
|           </Tooltip> | ||||
|           <Button | ||||
|             disabled//={tutorialIndex + 1 === tutorials.length}
 | ||||
|             //onClick={() => { this.props.history.push(`/tutorial/${tutorials[tutorialIndex + 1].id}`) }}
 | ||||
|           //onClick={() => { this.props.history.push(`/tutorial/${tutorials[tutorialIndex + 1].id}`) }}
 | ||||
|           > | ||||
|             {'>'} | ||||
|           </Button> | ||||
|  | ||||
| @ -12,7 +12,7 @@ import Instruction from './Instruction'; | ||||
| import Assessment from './Assessment'; | ||||
| import Badge from './Badge'; | ||||
| import NotFound from '../NotFound'; | ||||
| 
 | ||||
| import * as Blockly from 'blockly' | ||||
| import { detectWhitespacesAndReturnReadableResult } from '../../helpers/whitespace'; | ||||
| 
 | ||||
| 
 | ||||
| @ -26,10 +26,10 @@ class Tutorial extends Component { | ||||
|   } | ||||
| 
 | ||||
|   componentDidUpdate(props, state) { | ||||
|     if(this.props.tutorial && !this.props.isLoading && this.props.tutorial._id != this.props.match.params.tutorialId) { | ||||
|     if (this.props.tutorial && !this.props.isLoading && this.props.tutorial._id != this.props.match.params.tutorialId) { | ||||
|       this.props.getTutorial(this.props.match.params.tutorialId); | ||||
|     } | ||||
|     if(this.props.message.id === 'GET_TUTORIAL_FAIL'){ | ||||
|     if (this.props.message.id === 'GET_TUTORIAL_FAIL') { | ||||
|       alert(this.props.message.msg); | ||||
|     } | ||||
|   } | ||||
| @ -37,7 +37,7 @@ class Tutorial extends Component { | ||||
|   componentWillUnmount() { | ||||
|     this.props.resetTutorial(); | ||||
|     this.props.workspaceName(null); | ||||
|     if(this.props.message.msg){ | ||||
|     if (this.props.message.msg) { | ||||
|       this.props.clearMessages(); | ||||
|     } | ||||
|   } | ||||
| @ -47,37 +47,38 @@ class Tutorial extends Component { | ||||
|       <div> | ||||
|         {this.props.isLoading ? null : | ||||
|           !this.props.tutorial ? | ||||
|             this.props.message.id === 'GET_TUTORIAL_FAIL' ? <NotFound button={{ title: 'Zurück zur Tutorials-Übersicht', link: '/tutorial' }} /> : null | ||||
|             this.props.message.id === 'GET_TUTORIAL_FAIL' ? <NotFound button={{ title: Blockly.Msg.messages_GET_TUTORIAL_FAIL, link: '/tutorial' }} /> : null | ||||
|             : (() => { | ||||
|                 var tutorial = this.props.tutorial; | ||||
|                 var steps = this.props.tutorial.steps; | ||||
|                 var step = steps[this.props.activeStep]; | ||||
|                 var name = `${detectWhitespacesAndReturnReadableResult(tutorial.title)}_${detectWhitespacesAndReturnReadableResult(step.headline)}`; | ||||
|                 return( | ||||
|                   <div> | ||||
|                     <Breadcrumbs content={[{ link: '/tutorial', title: 'Tutorial' }, { link: `/tutorial/${this.props.tutorial._id}`, title: tutorial.title }]} /> | ||||
|               var tutorial = this.props.tutorial; | ||||
|               var steps = this.props.tutorial.steps; | ||||
|               var step = steps[this.props.activeStep]; | ||||
|               var name = `${detectWhitespacesAndReturnReadableResult(tutorial.title)}_${detectWhitespacesAndReturnReadableResult(step.headline)}`; | ||||
|               return ( | ||||
|                 <div> | ||||
|                   <Breadcrumbs content={[{ link: '/tutorial', title: 'Tutorial' }, { link: `/tutorial/${this.props.tutorial._id}`, title: tutorial.title }]} /> | ||||
| 
 | ||||
|                     <StepperHorizontal /> | ||||
|                     <Badge /> | ||||
|                   <StepperHorizontal /> | ||||
|                   <Badge /> | ||||
| 
 | ||||
|                     <div style={{ display: 'flex' }}> | ||||
|                       <StepperVertical steps={steps} /> | ||||
|                       {/* calc(Card-padding: 10px + Button-height: 35px + Button-marginTop: 15px)*/} | ||||
|                       <Card style={{ padding: '10px 10px 60px 10px', display: 'block', position: 'relative', height: 'max-content', width: '100%' }}> | ||||
|                         {step ? | ||||
|                           step.type === 'instruction' ? | ||||
|                             <Instruction step={step} /> | ||||
|                             : <Assessment step={step} name={name} /> // if step.type === 'assessment'
 | ||||
|                           : null} | ||||
|                   <div style={{ display: 'flex' }}> | ||||
|                     <StepperVertical steps={steps} /> | ||||
|                     {/* calc(Card-padding: 10px + Button-height: 35px + Button-marginTop: 15px)*/} | ||||
|                     <Card style={{ padding: '10px 10px 60px 10px', display: 'block', position: 'relative', height: 'max-content', width: '100%' }}> | ||||
|                       {step ? | ||||
|                         step.type === 'instruction' ? | ||||
|                           <Instruction step={step} /> | ||||
|                           : <Assessment step={step} name={name} /> // if step.type === 'assessment'
 | ||||
|                         : null} | ||||
| 
 | ||||
|                         <div style={{ marginTop: '20px', position: 'absolute', bottom: '10px' }}> | ||||
|                           <Button style={{ marginRight: '10px', height: '35px' }} variant='contained' disabled={this.props.activeStep === 0} onClick={() => this.props.tutorialStep(this.props.activeStep - 1)}>Zurück</Button> | ||||
|                           <Button style={{ height: '35px' }} variant='contained' color='primary' disabled={this.props.activeStep === tutorial.steps.length - 1} onClick={() => this.props.tutorialStep(this.props.activeStep + 1)}>Weiter</Button> | ||||
|                         </div> | ||||
|                       </Card> | ||||
|                     </div> | ||||
|                       <div style={{ marginTop: '20px', position: 'absolute', bottom: '10px' }}> | ||||
|                         <Button style={{ marginRight: '10px', height: '35px' }} variant='contained' disabled={this.props.activeStep === 0} onClick={() => this.props.tutorialStep(this.props.activeStep - 1)}>Zurück</Button> | ||||
|                         <Button style={{ height: '35px' }} variant='contained' color='primary' disabled={this.props.activeStep === tutorial.steps.length - 1} onClick={() => this.props.tutorialStep(this.props.activeStep + 1)}>Weiter</Button> | ||||
|                       </div> | ||||
|                     </Card> | ||||
|                   </div> | ||||
|               )})() | ||||
|                 </div> | ||||
|               ) | ||||
|             })() | ||||
|         } | ||||
|       </div> | ||||
|     ); | ||||
|  | ||||
| @ -19,7 +19,7 @@ import Divider from '@material-ui/core/Divider'; | ||||
| import InputAdornment from '@material-ui/core/InputAdornment'; | ||||
| import CircularProgress from '@material-ui/core/CircularProgress'; | ||||
| import Link from '@material-ui/core/Link'; | ||||
| 
 | ||||
| import * as Blockly from 'blockly' | ||||
| 
 | ||||
| export class Login extends Component { | ||||
| 
 | ||||
| @ -37,21 +37,21 @@ export class Login extends Component { | ||||
|     }; | ||||
|   } | ||||
| 
 | ||||
|   componentDidUpdate(props){ | ||||
|   componentDidUpdate(props) { | ||||
|     console.log(this.state.redirect); | ||||
|     const { message } = this.props; | ||||
|     if (message !== props.message) { | ||||
|       if(message.id === 'LOGIN_SUCCESS'){ | ||||
|         if(this.state.redirect){ | ||||
|       if (message.id === 'LOGIN_SUCCESS') { | ||||
|         if (this.state.redirect) { | ||||
|           this.props.history.push(this.state.redirect); | ||||
|         } | ||||
|         else{ | ||||
|         else { | ||||
|           this.props.history.goBack(); | ||||
|         } | ||||
|       } | ||||
|       // Check for login error
 | ||||
|       else if(message.id === 'LOGIN_FAIL'){ | ||||
|         this.setState({ email: '', password: '', snackbar: true, key: Date.now(), message: 'Der Benutzername oder das Passwort ist nicht korrekt.', type: 'error' }); | ||||
|       else if (message.id === 'LOGIN_FAIL') { | ||||
|         this.setState({ email: '', password: '', snackbar: true, key: Date.now(), message: Blockly.Msg.messages_LOGIN_FAIL, type: 'error' }); | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| @ -62,8 +62,8 @@ export class Login extends Component { | ||||
| 
 | ||||
|   onSubmit = e => { | ||||
|     e.preventDefault(); | ||||
|     const {email, password} = this.state; | ||||
|     if(email !== '' && password !== ''){ | ||||
|     const { email, password } = this.state; | ||||
|     if (email !== '' && password !== '') { | ||||
|       // create user object
 | ||||
|       const user = { | ||||
|         email, | ||||
| @ -83,13 +83,13 @@ export class Login extends Component { | ||||
|     e.preventDefault(); | ||||
|   }; | ||||
| 
 | ||||
|   render(){ | ||||
|     return( | ||||
|   render() { | ||||
|     return ( | ||||
|       <div> | ||||
|         <Breadcrumbs content={[{ link: '/user/login', title: 'Anmelden' }]} /> | ||||
| 
 | ||||
|         <div style={{maxWidth: '500px', marginLeft: 'auto', marginRight: 'auto'}}> | ||||
|           <h1>Anmelden</h1> | ||||
|         <div style={{ maxWidth: '500px', marginLeft: 'auto', marginRight: 'auto' }}> | ||||
|           <h1>{Blockly.Msg.login_head}</h1> | ||||
|           <Alert> | ||||
|             Zur Anmeldung ist ein Konto auf <Link color='primary' rel="noreferrer" target="_blank" href={'https://opensensemap.org/'}>openSenseMap</Link> Voraussetzung. | ||||
|           </Alert> | ||||
| @ -100,10 +100,10 @@ export class Login extends Component { | ||||
|             key={this.state.key} | ||||
|           /> | ||||
|           <TextField | ||||
|             style={{marginBottom: '10px'}} | ||||
|             style={{ marginBottom: '10px' }} | ||||
|             // variant='outlined'
 | ||||
|             type='text' | ||||
|             label='E-Mail oder Nutzername' | ||||
|             label={Blockly.Msg.labels_username} | ||||
|             name='email' | ||||
|             value={this.state.email} | ||||
|             onChange={this.onChange} | ||||
| @ -133,17 +133,17 @@ export class Login extends Component { | ||||
|             fullWidth={true} | ||||
|           /> | ||||
|           <p> | ||||
|             <Button color="primary" variant='contained' onClick={this.onSubmit} style={{width: '100%'}}> | ||||
|             <Button color="primary" variant='contained' onClick={this.onSubmit} style={{ width: '100%' }}> | ||||
|               {this.props.progress ? | ||||
|                 <div style={{height: '24.5px'}}><CircularProgress color="inherit" size={20}/></div> | ||||
|               : 'Anmelden'} | ||||
|                 <div style={{ height: '24.5px' }}><CircularProgress color="inherit" size={20} /></div> | ||||
|                 : 'Anmelden'} | ||||
|             </Button> | ||||
|           </p> | ||||
|           <p style={{textAlign: 'center', fontSize: '0.8rem'}}> | ||||
|           <p style={{ textAlign: 'center', fontSize: '0.8rem' }}> | ||||
|             <Link rel="noreferrer" target="_blank" href={'https://opensensemap.org/'} color="primary">Passwort vergessen?</Link> | ||||
|           </p> | ||||
|           <Divider variant='fullWidth'/> | ||||
|           <p style={{textAlign: 'center', paddingRight: "34px", paddingLeft: "34px"}}> | ||||
|           <Divider variant='fullWidth' /> | ||||
|           <p style={{ textAlign: 'center', paddingRight: "34px", paddingLeft: "34px" }}> | ||||
|             Du hast noch kein Konto? Registriere dich auf <Link rel="noreferrer" target="_blank" href={'https://opensensemap.org/'}>openSenseMap</Link>. | ||||
|           </p> | ||||
|         </div> | ||||
| @ -156,7 +156,6 @@ Login.propTypes = { | ||||
|   message: PropTypes.object.isRequired, | ||||
|   login: PropTypes.func.isRequired, | ||||
|   clearMessages: PropTypes.func.isRequired, | ||||
|   message: PropTypes.object.isRequired, | ||||
|   progress: PropTypes.bool.isRequired | ||||
| }; | ||||
| 
 | ||||
|  | ||||
| @ -9,10 +9,7 @@ import Snackbar from '../Snackbar'; | ||||
| import Dialog from '../Dialog'; | ||||
| 
 | ||||
| import { withStyles } from '@material-ui/core/styles'; | ||||
| import Button from '@material-ui/core/Button'; | ||||
| import IconButton from '@material-ui/core/IconButton'; | ||||
| import Tooltip from '@material-ui/core/Tooltip'; | ||||
| import Typography from '@material-ui/core/Typography'; | ||||
| 
 | ||||
| import { faUpload } from "@fortawesome/free-solid-svg-icons"; | ||||
| import { FontAwesomeIcon } from "@fortawesome/react-fontawesome"; | ||||
|  | ||||
| @ -1,4 +1,4 @@ | ||||
| import { TUTORIAL_PROGRESS, GET_TUTORIAL, GET_TUTORIALS, TUTORIAL_SUCCESS, TUTORIAL_ERROR, TUTORIAL_CHANGE, TUTORIAL_XML, TUTORIAL_ID, TUTORIAL_STEP } from '../actions/types'; | ||||
| import { TUTORIAL_PROGRESS, GET_TUTORIAL, GET_TUTORIALS, TUTORIAL_SUCCESS, TUTORIAL_ERROR, TUTORIAL_CHANGE, TUTORIAL_XML, TUTORIAL_STEP } from '../actions/types'; | ||||
| 
 | ||||
| 
 | ||||
| const initialStatus = () => { | ||||
|  | ||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user