p5.js-web-editor/client/modules/IDE/components/Editor.js

282 lines
9.1 KiB
JavaScript
Raw Normal View History

2016-06-27 20:03:22 +00:00
import React, { PropTypes } from 'react';
import EditorAccessibility from '../components/EditorAccessibility';
2016-06-23 22:29:55 +00:00
import CodeMirror from 'codemirror';
2016-09-07 19:05:25 +00:00
import beautifyJS from 'js-beautify';
const beautifyCSS = beautifyJS.css;
const beautifyHTML = beautifyJS.html;
2016-06-23 22:29:55 +00:00
import 'codemirror/mode/javascript/javascript';
import 'codemirror/mode/css/css';
import 'codemirror/mode/htmlmixed/htmlmixed';
2016-06-23 22:29:55 +00:00
import 'codemirror/addon/selection/active-line';
2016-07-12 21:38:24 +00:00
import 'codemirror/addon/lint/lint';
import 'codemirror/addon/lint/javascript-lint';
import 'codemirror/addon/lint/css-lint';
import 'codemirror/addon/lint/html-lint';
import 'codemirror/addon/comment/comment';
import 'codemirror/keymap/sublime';
2016-08-05 20:58:59 +00:00
import 'codemirror/addon/search/jump-to-line';
2016-07-12 21:38:24 +00:00
import { JSHINT } from 'jshint';
window.JSHINT = JSHINT;
import { CSSLint } from 'csslint';
window.CSSLint = CSSLint;
import { HTMLHint } from 'htmlhint';
window.HTMLHint = HTMLHint;
const beepUrl = require('../../../sounds/audioAlert.mp3');
2016-09-07 20:33:01 +00:00
import InlineSVG from 'react-inlinesvg';
const downArrowUrl = require('../../../images/down-arrow.svg');
import classNames from 'classnames';
2016-06-23 22:29:55 +00:00
2016-07-14 01:50:59 +00:00
import { debounce } from 'throttle-debounce';
import loopProtect from 'loop-protect';
2016-06-23 22:29:55 +00:00
class Editor extends React.Component {
2016-09-07 20:41:56 +00:00
constructor(props) {
super(props);
this.tidyCode = this.tidyCode.bind(this);
}
2016-06-23 22:29:55 +00:00
componentDidMount() {
this.beep = new Audio(beepUrl);
this.widgets = [];
2016-06-23 22:29:55 +00:00
this._cm = CodeMirror(this.refs.container, { // eslint-disable-line
2016-09-20 15:51:09 +00:00
theme: `p5-${this.props.theme}`,
value: this.props.file.content,
2016-06-23 22:29:55 +00:00
lineNumbers: true,
styleActiveLine: true,
2016-07-13 15:59:47 +00:00
inputStyle: 'contenteditable',
2016-07-12 19:58:11 +00:00
mode: 'javascript',
2016-07-12 21:38:24 +00:00
lineWrapping: true,
gutters: ['CodeMirror-lint-markers'],
2016-08-05 20:58:59 +00:00
keyMap: 'sublime',
lint: {
onUpdateLinting: () => {
debounce(2000, (annotations) => {
this.props.clearLintMessage();
annotations.forEach((x) => {
if (x.from.line > -1) {
this.props.updateLintMessage(x.severity, (x.from.line + 1), x.message);
}
});
if (this.props.lintMessages.length > 0 && this.props.lintWarning) {
this.beep.play();
2016-08-06 03:08:44 +00:00
}
});
}
2016-08-05 20:58:59 +00:00
}
2016-06-23 22:29:55 +00:00
});
this._cm.on('change', debounce(1000, () => {
this.props.setUnsavedChanges(true);
this.props.updateFileContent(this.props.file.name, this._cm.getValue());
this.checkForInfiniteLoop((infiniteLoop, prevs) => {
if (!infiniteLoop && prevs) {
this.props.startSketch();
}
});
2016-07-14 01:50:59 +00:00
}));
2016-08-05 20:58:59 +00:00
this._cm.on('keyup', () => {
2016-08-25 16:32:06 +00:00
const temp = `line ${parseInt((this._cm.getCursor().line) + 1, 10)}`;
document.getElementById('current-line').innerHTML = temp;
2016-08-05 20:58:59 +00:00
});
this._cm.on('keydown', (_cm, e) => {
if (e.key === 'Tab' && e.shiftKey) {
2016-09-07 20:41:56 +00:00
this.tidyCode();
}
});
this._cm.getWrapperElement().style['font-size'] = `${this.props.fontSize}px`;
this._cm.setOption('indentWithTabs', this.props.isTabIndent);
this._cm.setOption('tabSize', this.props.indentationAmount);
2016-06-23 22:29:55 +00:00
}
componentDidUpdate(prevProps) {
if (this.props.file.content !== prevProps.file.content &&
this.props.file.content !== this._cm.getValue()) {
this._cm.setValue(this.props.file.content); // eslint-disable-line no-underscore-dangle
setTimeout(() => this.props.setUnsavedChanges(false), 500);
2016-06-23 22:29:55 +00:00
}
if (this.props.fontSize !== prevProps.fontSize) {
this._cm.getWrapperElement().style['font-size'] = `${this.props.fontSize}px`;
}
2016-07-06 15:27:39 +00:00
if (this.props.indentationAmount !== prevProps.indentationAmount) {
this._cm.setOption('tabSize', this.props.indentationAmount);
}
2016-07-11 03:11:06 +00:00
if (this.props.isTabIndent !== prevProps.isTabIndent) {
this._cm.setOption('indentWithTabs', this.props.isTabIndent);
}
if (this.props.file.name !== prevProps.name) {
if (this.props.file.name.match(/.+\.js$/)) {
this._cm.setOption('mode', 'javascript');
} else if (this.props.file.name.match(/.+\.css$/)) {
this._cm.setOption('mode', 'css');
} else if (this.props.file.name.match(/.+\.html$/)) {
this._cm.setOption('mode', 'htmlmixed');
}
}
2016-09-20 15:51:09 +00:00
if (this.props.theme !== prevProps.theme) {
this._cm.setOption('theme', `p5-${this.props.theme}`);
}
2016-06-23 22:29:55 +00:00
}
componentWillUnmount() {
this._cm = null;
}
2016-09-07 20:41:56 +00:00
tidyCode() {
const beautifyOptions = {
indent_size: this.props.indentationAmount,
indent_with_tabs: this.props.isTabIndent
};
const mode = this._cm.getOption('mode');
if (mode === 'javascript') {
this._cm.doc.setValue(beautifyJS(this._cm.doc.getValue(), beautifyOptions));
} else if (mode === 'css') {
this._cm.doc.setValue(beautifyCSS(this._cm.doc.getValue(), beautifyOptions));
} else if (mode === 'htmlmixed') {
this._cm.doc.setValue(beautifyHTML(this._cm.doc.getValue(), beautifyOptions));
}
}
checkForInfiniteLoop(callback) {
const prevIsplaying = this.props.isPlaying;
let infiniteLoop = false;
let prevLine;
this.props.stopSketch();
this.props.resetInfiniteLoops();
let iframe;
for (let i = 0; i < this.widgets.length; ++i) {
this._cm.removeLineWidget(this.widgets[i]);
}
this.widgets.length = 0;
loopProtect.alias = 'protect';
loopProtect.hit = (line) => {
if (line !== prevLine) {
this.props.detectInfiniteLoops();
infiniteLoop = true;
callback(infiniteLoop, prevIsplaying);
const msg = document.createElement('div');
const loopError = `line ${line}: This loop is taking too long to run. This might be an infinite loop.`;
msg.appendChild(document.createTextNode(loopError));
msg.className = 'lint-error';
this.widgets.push(this._cm.addLineWidget(line - 1, msg, { coverGutter: false, noHScroll: true }));
prevLine = line;
}
};
const processed = loopProtect(this.props.file.content);
const iframeForLoop = document.getElementById('iframeForLoop');
if (iframeForLoop === null) {
iframe = document.createElement('iframe');
iframe.id = 'iframeForLoop';
iframe.style.display = 'none';
document.body.appendChild(iframe);
} else {
iframeForLoop.srcdoc = '';
const win = iframeForLoop.contentWindow;
const doc = win.document;
doc.open();
win.protect = loopProtect;
doc.write(`<!DOCTYPE html>
<html>
<head>
<script src="https://cdnjs.cloudflare.com/ajax/libs/p5.js/0.5.2/p5.min.js"></script>
<script src="https://cdnjs.cloudflare.com/ajax/libs/p5.js/0.5.2/addons/p5.dom.min.js"></script>
</head>
<body>
<script>
${processed}
</script>
</body>
</html>`);
win.onerror = () => true;
doc.close();
}
callback(infiniteLoop, prevIsplaying, prevLine);
}
2016-06-23 22:29:55 +00:00
_cm: CodeMirror.Editor
render() {
2016-09-07 20:33:01 +00:00
const editorSectionClass = classNames({
editor: true,
'editor--options': this.props.editorOptionsVisible
});
2016-08-05 20:58:59 +00:00
return (
<section
title="code editor"
role="main"
2016-09-07 20:33:01 +00:00
className={editorSectionClass}
>
2016-09-07 20:33:01 +00:00
<button
className="editor__options-button"
2016-09-28 16:09:42 +00:00
aria-label="editor options"
2016-09-08 02:49:29 +00:00
tabIndex="0"
onClick={(e) => {
e.target.focus();
this.props.showEditorOptions();
}}
2016-09-07 20:41:56 +00:00
onBlur={() => setTimeout(this.props.closeEditorOptions, 200)}
2016-09-07 20:33:01 +00:00
>
<InlineSVG src={downArrowUrl} />
</button>
2016-09-28 16:09:42 +00:00
<ul className="editor__options" title="editor options">
2016-09-07 20:33:01 +00:00
<li>
2016-09-07 20:41:56 +00:00
<a onClick={this.tidyCode}>Tidy</a>
2016-09-07 20:33:01 +00:00
</li>
<li>
2016-09-07 21:47:22 +00:00
<a onClick={this.props.showKeyboardShortcutModal}>Keyboard Shortcuts</a>
2016-09-07 20:33:01 +00:00
</li>
</ul>
2016-08-12 18:19:23 +00:00
<div ref="container" className="editor-holder" tabIndex="0">
</div>
<EditorAccessibility
lintMessages={this.props.lintMessages}
2016-08-12 18:23:34 +00:00
lineNumber={this.props.lineNumber}
/>
2016-08-12 18:19:23 +00:00
</section>
2016-08-05 20:58:59 +00:00
);
2016-06-23 22:29:55 +00:00
}
}
2016-06-27 20:03:22 +00:00
Editor.propTypes = {
2016-08-11 18:09:59 +00:00
lintWarning: PropTypes.bool.isRequired,
2016-08-12 18:23:34 +00:00
lineNumber: PropTypes.string.isRequired,
2016-08-11 17:24:02 +00:00
lintMessages: PropTypes.array.isRequired,
updateLintMessage: PropTypes.func.isRequired,
clearLintMessage: PropTypes.func.isRequired,
updateLineNumber: PropTypes.func.isRequired,
2016-07-11 02:52:48 +00:00
indentationAmount: PropTypes.number.isRequired,
2016-07-12 03:40:30 +00:00
isTabIndent: PropTypes.bool.isRequired,
updateFileContent: PropTypes.func.isRequired,
fontSize: PropTypes.number.isRequired,
file: PropTypes.shape({
name: PropTypes.string.isRequired,
content: PropTypes.string.isRequired
2016-09-07 20:33:01 +00:00
}),
editorOptionsVisible: PropTypes.bool.isRequired,
showEditorOptions: PropTypes.func.isRequired,
2016-09-07 21:47:22 +00:00
closeEditorOptions: PropTypes.func.isRequired,
showKeyboardShortcutModal: PropTypes.func.isRequired,
2016-09-20 15:51:09 +00:00
setUnsavedChanges: PropTypes.func.isRequired,
infiniteLoop: PropTypes.bool.isRequired,
detectInfiniteLoops: PropTypes.func.isRequired,
resetInfiniteLoops: PropTypes.func.isRequired,
stopSketch: PropTypes.func.isRequired,
startSketch: PropTypes.func.isRequired,
isPlaying: PropTypes.bool.isRequired,
theme: PropTypes.string.isRequired
2016-06-27 20:03:22 +00:00
};
2016-06-23 22:29:55 +00:00
export default Editor;