Files
freeCodeCamp/common/app/routes/challenges/components/classic/Editor.jsx

152 lines
3.7 KiB
JavaScript
Raw Normal View History

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';
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,
height: PropTypes.number,
content: PropTypes.string,
2016-05-13 20:36:54 -07:00
mode: PropTypes.string,
updateFile: PropTypes.func
};
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;
},
'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);