| 
									
										
										
										
											2016-05-14 16:25:27 -07:00
										 |  |  | import { Subject } from 'rx'; | 
					
						
							| 
									
										
										
										
											2016-03-05 21:06:04 -08:00
										 |  |  | import React, { PropTypes } from 'react'; | 
					
						
							|  |  |  | import { createSelector } from 'reselect'; | 
					
						
							|  |  |  | import { connect } from 'react-redux'; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import Codemirror from 'react-codemirror'; | 
					
						
							|  |  |  | import NoSSR from 'react-no-ssr'; | 
					
						
							|  |  |  | import PureComponent from 'react-pure-render/component'; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-27 16:42:05 +01:00
										 |  |  | import MouseTrap from 'mousetrap'; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-26 21:07:22 -08:00
										 |  |  | import CodeMirrorSkeleton from '../CodeMirrorSkeleton.jsx'; | 
					
						
							| 
									
										
										
										
											2016-12-31 15:45:14 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-03-05 21:06:04 -08:00
										 |  |  | const mapStateToProps = createSelector( | 
					
						
							|  |  |  |   state => state.app.windowHeight, | 
					
						
							|  |  |  |   state => state.app.navHeight, | 
					
						
							|  |  |  |   (windowHeight, navHeight) => ({ height: windowHeight - navHeight - 50 }) | 
					
						
							|  |  |  | ); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-05-14 16:25:27 -07:00
										 |  |  | const editorDebounceTimeout = 750; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-03-05 21:06:04 -08:00
										 |  |  | const options = { | 
					
						
							| 
									
										
										
										
											2017-01-25 15:29:52 +01:00
										 |  |  |   lint: {esversion: 6}, | 
					
						
							| 
									
										
										
										
											2016-03-05 21:06:04 -08:00
										 |  |  |   lineNumbers: true, | 
					
						
							|  |  |  |   mode: 'javascript', | 
					
						
							|  |  |  |   theme: 'monokai', | 
					
						
							|  |  |  |   runnable: true, | 
					
						
							|  |  |  |   matchBrackets: true, | 
					
						
							|  |  |  |   autoCloseBrackets: true, | 
					
						
							|  |  |  |   scrollbarStyle: 'null', | 
					
						
							|  |  |  |   lineWrapping: true, | 
					
						
							|  |  |  |   gutters: ['CodeMirror-lint-markers'] | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | export class Editor extends PureComponent { | 
					
						
							| 
									
										
										
										
											2016-05-14 16:25:27 -07:00
										 |  |  |   constructor(...args) { | 
					
						
							|  |  |  |     super(...args); | 
					
						
							|  |  |  |     this._editorContent$ = new Subject(); | 
					
						
							|  |  |  |     this.handleChange = this.handleChange.bind(this); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2016-03-05 21:06:04 -08:00
										 |  |  |   static displayName = 'Editor'; | 
					
						
							|  |  |  |   static propTypes = { | 
					
						
							| 
									
										
										
										
											2016-05-20 12:42:26 -07:00
										 |  |  |     executeChallenge: PropTypes.func, | 
					
						
							| 
									
										
										
										
											2016-05-09 13:42:39 -07:00
										 |  |  |     height: PropTypes.number, | 
					
						
							|  |  |  |     content: PropTypes.string, | 
					
						
							| 
									
										
										
										
											2016-05-13 20:36:54 -07:00
										 |  |  |     mode: PropTypes.string, | 
					
						
							|  |  |  |     updateFile: PropTypes.func | 
					
						
							| 
									
										
										
										
											2016-05-09 13:42:39 -07:00
										 |  |  |   }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   static defaultProps = { | 
					
						
							|  |  |  |     content: '// Happy Coding!', | 
					
						
							|  |  |  |     mode: 'javascript' | 
					
						
							| 
									
										
										
										
											2016-03-05 21:06:04 -08:00
										 |  |  |   }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-05-20 12:42:26 -07:00
										 |  |  |   createOptions = createSelector( | 
					
						
							|  |  |  |     state => state.options, | 
					
						
							|  |  |  |     state => state.executeChallenge, | 
					
						
							|  |  |  |     state => state.mode, | 
					
						
							|  |  |  |     (options, executeChallenge, mode) => ({ | 
					
						
							|  |  |  |       ...options, | 
					
						
							|  |  |  |       mode, | 
					
						
							|  |  |  |       extraKeys: { | 
					
						
							| 
									
										
										
										
											2017-01-27 16:42:05 +01:00
										 |  |  |         Esc() { | 
					
						
							|  |  |  |           document.activeElement.blur(); | 
					
						
							|  |  |  |         }, | 
					
						
							| 
									
										
										
										
											2016-05-20 12:42:26 -07:00
										 |  |  |         Tab(cm) { | 
					
						
							|  |  |  |           if (cm.somethingSelected()) { | 
					
						
							|  |  |  |             return cm.indentSelection('add'); | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  |           const spaces = Array(cm.getOption('indentUnit') + 1).join(' '); | 
					
						
							|  |  |  |           return cm.replaceSelection(spaces); | 
					
						
							|  |  |  |         }, | 
					
						
							|  |  |  |         'Shift-Tab': function(cm) { | 
					
						
							|  |  |  |           if (cm.somethingSelected()) { | 
					
						
							|  |  |  |             return cm.indentSelection('subtract'); | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  |           const spaces = Array(cm.getOption('indentUnit') + 1).join(' '); | 
					
						
							|  |  |  |           return cm.replaceSelection(spaces); | 
					
						
							|  |  |  |         }, | 
					
						
							|  |  |  |         'Ctrl-Enter': function() { | 
					
						
							|  |  |  |           executeChallenge(); | 
					
						
							|  |  |  |           return false; | 
					
						
							|  |  |  |         }, | 
					
						
							|  |  |  |         'Cmd-Enter': function() { | 
					
						
							|  |  |  |           executeChallenge(); | 
					
						
							|  |  |  |           return false; | 
					
						
							| 
									
										
										
										
											2016-10-24 10:56:48 +02:00
										 |  |  |         }, | 
					
						
							|  |  |  |         'Ctrl-/': function(cm) { | 
					
						
							|  |  |  |           cm.toggleComment(); | 
					
						
							|  |  |  |         }, | 
					
						
							|  |  |  |         'Cmd-/': function(cm) { | 
					
						
							|  |  |  |           cm.toggleComment(); | 
					
						
							| 
									
										
										
										
											2016-05-20 12:42:26 -07:00
										 |  |  |         } | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     }) | 
					
						
							|  |  |  |   ); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-05-14 16:25:27 -07:00
										 |  |  |   componentDidMount() { | 
					
						
							|  |  |  |     const { updateFile = (() => {}) } = this.props; | 
					
						
							|  |  |  |     this._subscription = this._editorContent$ | 
					
						
							|  |  |  |       .debounce(editorDebounceTimeout) | 
					
						
							|  |  |  |       .distinctUntilChanged() | 
					
						
							|  |  |  |       .subscribe( | 
					
						
							|  |  |  |         updateFile, | 
					
						
							|  |  |  |         err => { throw err; } | 
					
						
							|  |  |  |       ); | 
					
						
							| 
									
										
										
										
											2017-01-27 16:42:05 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     MouseTrap.bind('e', () => { | 
					
						
							|  |  |  |       this.refs.editor.focus(); | 
					
						
							|  |  |  |     }, 'keyup'); | 
					
						
							| 
									
										
										
										
											2016-05-14 16:25:27 -07:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   componentWillUnmount() { | 
					
						
							|  |  |  |     if (this._subscription) { | 
					
						
							|  |  |  |       this._subscription.dispose(); | 
					
						
							|  |  |  |       this._subscription = null; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2017-01-27 16:42:05 +01:00
										 |  |  |     MouseTrap.unbind('e', 'keyup'); | 
					
						
							| 
									
										
										
										
											2016-05-14 16:25:27 -07:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   handleChange(value) { | 
					
						
							|  |  |  |     if (this._subscription) { | 
					
						
							|  |  |  |       this._editorContent$.onNext(value); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-03-05 21:06:04 -08:00
										 |  |  |   render() { | 
					
						
							| 
									
										
										
										
											2016-05-20 12:42:26 -07:00
										 |  |  |     const { executeChallenge, content, height, mode } = this.props; | 
					
						
							| 
									
										
										
										
											2016-03-05 21:06:04 -08:00
										 |  |  |     const style = {}; | 
					
						
							|  |  |  |     if (height) { | 
					
						
							|  |  |  |       style.height = height + 'px'; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return ( | 
					
						
							|  |  |  |       <div | 
					
						
							|  |  |  |         className='challenges-editor' | 
					
						
							| 
									
										
										
										
											2016-06-23 16:57:26 -07:00
										 |  |  |         style={ style } | 
					
						
							|  |  |  |         > | 
					
						
							| 
									
										
										
										
											2016-12-31 15:45:14 +00:00
										 |  |  |         <NoSSR onSSR={ <CodeMirrorSkeleton content={ content } /> }> | 
					
						
							| 
									
										
										
										
											2016-03-05 21:06:04 -08:00
										 |  |  |           <Codemirror | 
					
						
							| 
									
										
										
										
											2016-05-14 16:25:27 -07:00
										 |  |  |             onChange={ this.handleChange } | 
					
						
							| 
									
										
										
										
											2016-05-20 12:42:26 -07:00
										 |  |  |             options={ this.createOptions({ executeChallenge, mode, options }) } | 
					
						
							| 
									
										
										
										
											2017-01-27 16:42:05 +01:00
										 |  |  |             ref='editor' | 
					
						
							| 
									
										
										
										
											2016-06-23 16:57:26 -07:00
										 |  |  |             value={ content } | 
					
						
							|  |  |  |           /> | 
					
						
							| 
									
										
										
										
											2016-03-05 21:06:04 -08:00
										 |  |  |         </NoSSR> | 
					
						
							|  |  |  |       </div> | 
					
						
							|  |  |  |     ); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | export default connect(mapStateToProps)(Editor); |