From 530b2d76f4a8317bb6bd69405c2cefe07f18c6f1 Mon Sep 17 00:00:00 2001 From: DD Liu Date: Tue, 25 Jul 2017 11:54:17 -0400 Subject: [PATCH 1/4] fix name --- src/index.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/index.js b/src/index.js index 986f2302..6611b719 100644 --- a/src/index.js +++ b/src/index.js @@ -1,3 +1,3 @@ import PaintEditor from './containers/paint-editor.jsx'; -export default PaintEditorComponent; +export default PaintEditor; From dc683f1d827b3e42297c1d14bf58c7644b5a2eba Mon Sep 17 00:00:00 2001 From: DD Liu Date: Wed, 26 Jul 2017 20:39:12 -0400 Subject: [PATCH 2/4] switch to jest --- .babelrc | 3 +++ package.json | 18 ++++++++++++++++-- src/.eslintrc.js | 8 +++++++- src/components/paint-editor.jsx | 3 +-- src/containers/paint-editor.jsx | 2 +- src/containers/paper-canvas.jsx | 2 +- src/log/log.js | 4 ++-- src/reducers/combine-reducers.js | 5 +++-- src/reducers/tools.js | 6 +++--- src/tools/tool-types.js | 2 +- test/__mocks__/fileMock.js | 3 +++ test/__mocks__/styleMock.js | 3 +++ test/tools-reducer-test.js | 25 ------------------------- test/unit/tools-reducer.test.js | 23 +++++++++++++++++++++++ 14 files changed, 67 insertions(+), 40 deletions(-) create mode 100644 .babelrc create mode 100644 test/__mocks__/fileMock.js create mode 100644 test/__mocks__/styleMock.js delete mode 100644 test/tools-reducer-test.js create mode 100644 test/unit/tools-reducer.test.js diff --git a/.babelrc b/.babelrc new file mode 100644 index 00000000..facd1809 --- /dev/null +++ b/.babelrc @@ -0,0 +1,3 @@ +{ + "presets": ["es2015", "react"] +} \ No newline at end of file diff --git a/package.json b/package.json index 3190fdc5..2c5130de 100644 --- a/package.json +++ b/package.json @@ -9,8 +9,8 @@ "deploy": "touch playground/.nojekyll && gh-pages -t -d playground -m \"Build for $(git log --pretty=format:%H -n1)\"", "lint": "eslint . --ext .js,.jsx", "start": "webpack-dev-server", - "tap": "./node_modules/.bin/tap ./test/*.js", - "test": "npm run lint && npm run build && npm run tap", + "test": "npm run lint && npm run build && npm run unit-test", + "unit": "jest", "watch": "webpack --progress --colors --watch" }, "author": "Massachusetts Institute of Technology", @@ -28,17 +28,22 @@ "autoprefixer": "7.1.1", "babel-core": "^6.23.1", "babel-eslint": "^7.1.1", + "babel-jest": "^20.0.3", "babel-loader": "^7.0.0", "babel-plugin-transform-object-rest-spread": "^6.22.0", "babel-preset-es2015": "^6.22.0", "babel-preset-react": "^6.22.0", "classnames": "2.2.5", "css-loader": "0.28.3", + "enzyme": "^2.8.2", "eslint": "^3.16.1", + "eslint-config-import": "^0.13.0", "eslint-config-scratch": "^3.0.0", + "eslint-plugin-import": "^2.7.0", "eslint-plugin-react": "^7.0.1", "gh-pages": "github:rschamp/gh-pages#publish-branch-to-subfolder", "html-webpack-plugin": "2.28.0", + "jest": "^20.0.4", "lodash.defaultsdeep": "4.6.0", "minilog": "3.1.0", "mkdirp": "^0.5.1", @@ -51,12 +56,21 @@ "react-dom": "15.5.4", "react-intl": "2.3.0", "react-redux": "5.0.5", + "react-test-renderer": "^15.5.4", "redux": "3.6.0", + "redux-mock-store": "^1.2.3", "redux-throttle": "0.1.1", + "regenerator-runtime": "^0.10.5", "rimraf": "^2.6.1", "style-loader": "^0.18.0", "tap": "^10.2.0", "webpack": "^2.4.1", "webpack-dev-server": "^2.4.1" + }, + "jest": { + "moduleNameMapper": { + "\\.(jpg|jpeg|png|gif|eot|otf|webp|svg|ttf|woff|woff2|mp4|webm|wav|mp3|m4a|aac|oga)$": "/test/__mocks__/fileMock.js", + "\\.(css|less)$": "/test/__mocks__/styleMock.js" + } } } diff --git a/src/.eslintrc.js b/src/.eslintrc.js index 7ff859ca..d5985f2d 100644 --- a/src/.eslintrc.js +++ b/src/.eslintrc.js @@ -1,7 +1,13 @@ module.exports = { root: true, - extends: ['scratch', 'scratch/es6', 'scratch/react'], + extends: ['scratch', 'scratch/es6', 'scratch/react', 'import'], env: { browser: true + }, + rules: { + 'import/no-mutable-exports': 'error', + 'import/no-commonjs': 'error', + 'import/no-amd': 'error', + 'import/no-nodejs-modules': 'error' } }; diff --git a/src/components/paint-editor.jsx b/src/components/paint-editor.jsx index db93953f..a22a01e6 100644 --- a/src/components/paint-editor.jsx +++ b/src/components/paint-editor.jsx @@ -14,5 +14,4 @@ PaintEditorComponent.propTypes = { }) }; - -module.exports = PaintEditorComponent; +export default PaintEditorComponent; diff --git a/src/containers/paint-editor.jsx b/src/containers/paint-editor.jsx index efe05522..4e219058 100644 --- a/src/containers/paint-editor.jsx +++ b/src/containers/paint-editor.jsx @@ -42,7 +42,7 @@ const mapDispatchToProps = dispatch => ({ } }); -module.exports = connect( +export default connect( mapStateToProps, mapDispatchToProps )(PaintEditor); diff --git a/src/containers/paper-canvas.jsx b/src/containers/paper-canvas.jsx index 5d813a0f..5eb3a9aa 100644 --- a/src/containers/paper-canvas.jsx +++ b/src/containers/paper-canvas.jsx @@ -44,4 +44,4 @@ PaperCanvas.propTypes = { }) }; -module.exports = PaperCanvas; +export default PaperCanvas; diff --git a/src/log/log.js b/src/log/log.js index 9b991a58..5ea3bb07 100644 --- a/src/log/log.js +++ b/src/log/log.js @@ -1,4 +1,4 @@ -const minilog = require('minilog'); +import minilog from 'minilog'; minilog.enable(); -module.exports = minilog('paint-editor'); +export default minilog('paint-editor'); diff --git a/src/reducers/combine-reducers.js b/src/reducers/combine-reducers.js index a533b763..985c9b5b 100644 --- a/src/reducers/combine-reducers.js +++ b/src/reducers/combine-reducers.js @@ -1,5 +1,6 @@ import {combineReducers} from 'redux'; +import toolReducer from './tools'; -module.exports = combineReducers({ - tool: require('./tools') +export default combineReducers({ + tool: toolReducer }); diff --git a/src/reducers/tools.js b/src/reducers/tools.js index 59cf1afb..28832833 100644 --- a/src/reducers/tools.js +++ b/src/reducers/tools.js @@ -1,5 +1,5 @@ -const ToolTypes = require('../tools/tool-types'); -const log = require('../log/log'); +import ToolTypes from '../tools/tool-types'; +import log from '../log/log'; const CHANGE_TOOL = 'scratch-paint/tools/CHANGE_TOOL'; const initialState = ToolTypes.BRUSH; @@ -29,4 +29,4 @@ reducer.changeTool = function (tool) { }; }; -module.exports = reducer; +export default reducer; diff --git a/src/tools/tool-types.js b/src/tools/tool-types.js index 12c8b0f4..61524cb7 100644 --- a/src/tools/tool-types.js +++ b/src/tools/tool-types.js @@ -9,4 +9,4 @@ class ToolTypes { ToolTypes.BRUSH = new ToolTypes('BRUSH'); ToolTypes.ERASER = new ToolTypes('ERASER'); -module.exports = ToolTypes; +export default ToolTypes; diff --git a/test/__mocks__/fileMock.js b/test/__mocks__/fileMock.js new file mode 100644 index 00000000..59890f6a --- /dev/null +++ b/test/__mocks__/fileMock.js @@ -0,0 +1,3 @@ +// __mocks__/fileMock.js + +module.exports = 'test-file-stub'; diff --git a/test/__mocks__/styleMock.js b/test/__mocks__/styleMock.js new file mode 100644 index 00000000..d988e23b --- /dev/null +++ b/test/__mocks__/styleMock.js @@ -0,0 +1,3 @@ +// __mocks__/styleMock.js + +module.exports = {}; diff --git a/test/tools-reducer-test.js b/test/tools-reducer-test.js deleted file mode 100644 index 3da061d1..00000000 --- a/test/tools-reducer-test.js +++ /dev/null @@ -1,25 +0,0 @@ -const test = require('tap').test; -const ToolTypes = require('../src/tools/tool-types'); -const reducer = require('../src/reducers/tools'); - -test('initialState', t => { - let defaultState; - t.assert(reducer(defaultState /* state */, {type: 'anything'} /* action */) instanceof ToolTypes); - t.end(); -}); - -test('changeTool', t => { - let defaultState; - t.assert(reducer(defaultState /* state */, reducer.changeTool(ToolTypes.ERASER) /* action */), ToolTypes.ERASER); - t.assert( - reducer(ToolTypes.ERASER /* state */, reducer.changeTool(ToolTypes.ERASER) /* action */), ToolTypes.ERASER); - t.assert(reducer(ToolTypes.BRUSH /* state */, reducer.changeTool(ToolTypes.ERASER) /* action */), ToolTypes.ERASER); - t.end(); -}); - -test('invalidChangeTool', t => { - t.assert( - reducer(ToolTypes.BRUSH /* state */, reducer.changeTool('non-existant tool') /* action */), ToolTypes.BRUSH); - t.assert(reducer(ToolTypes.BRUSH /* state */, reducer.changeTool() /* action */), ToolTypes.BRUSH); - t.end(); -}); diff --git a/test/unit/tools-reducer.test.js b/test/unit/tools-reducer.test.js new file mode 100644 index 00000000..f0c729f2 --- /dev/null +++ b/test/unit/tools-reducer.test.js @@ -0,0 +1,23 @@ +/* eslint-env jest */ +import ToolTypes from '../../src/tools/tool-types'; +import reducer from '../../src/reducers/tools'; + +test('initialState', () => { + let defaultState; + expect(reducer(defaultState /* state */, {type: 'anything'} /* action */) instanceof ToolTypes).toBeTruthy(); +}); + +test('changeTool', () => { + let defaultState; + expect(reducer(defaultState /* state */, reducer.changeTool(ToolTypes.ERASER) /* action */)).toBe(ToolTypes.ERASER); + expect(reducer(ToolTypes.ERASER /* state */, reducer.changeTool(ToolTypes.ERASER) /* action */)) + .toBe(ToolTypes.ERASER); + expect(reducer(ToolTypes.BRUSH /* state */, reducer.changeTool(ToolTypes.ERASER) /* action */)) + .toBe(ToolTypes.ERASER); +}); + +test('invalidChangeTool', () => { + expect(reducer(ToolTypes.BRUSH /* state */, reducer.changeTool('non-existant tool') /* action */)) + .toBe(ToolTypes.BRUSH); + expect(reducer(ToolTypes.BRUSH /* state */, reducer.changeTool() /* action */)).toBe(ToolTypes.BRUSH); +}); From a4891d22b070ef21ec301fb30ef3c34c92f968f8 Mon Sep 17 00:00:00 2001 From: DD Liu Date: Thu, 27 Jul 2017 00:34:33 -0400 Subject: [PATCH 3/4] code review --- .babelrc | 1 + package.json | 8 +++++--- src/components/paint-editor.jsx | 5 ++--- src/containers/paint-editor.jsx | 21 ++++++++++----------- src/containers/paper-canvas.jsx | 20 +++++++++----------- src/log/log.js | 2 +- src/playground/playground.jsx | 7 ++----- src/reducers/tools.js | 9 +++------ src/tools/tool-types.js | 16 ++++++---------- test/unit/tools-reducer.test.js | 2 +- 10 files changed, 40 insertions(+), 51 deletions(-) diff --git a/.babelrc b/.babelrc index facd1809..dc76ced0 100644 --- a/.babelrc +++ b/.babelrc @@ -1,3 +1,4 @@ { + "plugins": ["transform-object-rest-spread"], "presets": ["es2015", "react"] } \ No newline at end of file diff --git a/package.json b/package.json index 2c5130de..ba1ba054 100644 --- a/package.json +++ b/package.json @@ -9,7 +9,7 @@ "deploy": "touch playground/.nojekyll && gh-pages -t -d playground -m \"Build for $(git log --pretty=format:%H -n1)\"", "lint": "eslint . --ext .js,.jsx", "start": "webpack-dev-server", - "test": "npm run lint && npm run build && npm run unit-test", + "test": "npm run lint && npm run build && npm run unit", "unit": "jest", "watch": "webpack --progress --colors --watch" }, @@ -44,15 +44,17 @@ "gh-pages": "github:rschamp/gh-pages#publish-branch-to-subfolder", "html-webpack-plugin": "2.28.0", "jest": "^20.0.4", + "keymirror": "0.1.1", + "lodash.bindall": "4.4.0", "lodash.defaultsdeep": "4.6.0", "minilog": "3.1.0", "mkdirp": "^0.5.1", - "paper": "^0.11.4", + "paper": "0.11.4", "postcss-import": "^10.0.0", "postcss-loader": "^2.0.5", "postcss-simple-vars": "^4.0.0", "prop-types": "^15.5.10", - "react": "15.5.4", + "react": "15.6.1", "react-dom": "15.5.4", "react-intl": "2.3.0", "react-redux": "5.0.5", diff --git a/src/components/paint-editor.jsx b/src/components/paint-editor.jsx index a22a01e6..4c3d025c 100644 --- a/src/components/paint-editor.jsx +++ b/src/components/paint-editor.jsx @@ -1,6 +1,7 @@ import PropTypes from 'prop-types'; import React from 'react'; import PaperCanvas from '../containers/paper-canvas.jsx'; +import ToolTypes from '../tools/tool-types.js'; const PaintEditorComponent = props => ( ( ); PaintEditorComponent.propTypes = { - tool: PropTypes.shape({ - name: PropTypes.string.isRequired - }) + tool: PropTypes.oneOf(Object.keys(ToolTypes)).isRequired }; export default PaintEditorComponent; diff --git a/src/containers/paint-editor.jsx b/src/containers/paint-editor.jsx index 4e219058..40cc74e4 100644 --- a/src/containers/paint-editor.jsx +++ b/src/containers/paint-editor.jsx @@ -7,26 +7,25 @@ import {connect} from 'react-redux'; class PaintEditor extends React.Component { componentDidMount () { - const onKeyPress = this.props.onKeyPress; - document.onkeydown = function (e) { - e = e || window.event; - onKeyPress(e); - }; + document.addEventListener('keydown', this.props.onKeyPress); + } + componentWillUnmount () { + document.removeEventListener('keydown', this.props.onKeyPress); } render () { + const { + onKeyPress, // eslint-disable-line no-unused-vars + ...props + } = this.props; return ( - + ); } } PaintEditor.propTypes = { onKeyPress: PropTypes.func.isRequired, - tool: PropTypes.shape({ - name: PropTypes.string.isRequired - }) + tool: PropTypes.oneOf(Object.keys(ToolTypes)).isRequired }; const mapStateToProps = state => ({ diff --git a/src/containers/paper-canvas.jsx b/src/containers/paper-canvas.jsx index 5eb3a9aa..a3316dbf 100644 --- a/src/containers/paper-canvas.jsx +++ b/src/containers/paper-canvas.jsx @@ -4,13 +4,8 @@ import paper from 'paper'; import ToolTypes from '../tools/tool-types.js'; class PaperCanvas extends React.Component { - constructor (props) { - super(props); - this.state = { - }; - } componentDidMount () { - paper.setup('paper-canvas'); + paper.setup(this.canvas); // Create a Paper.js Path to draw a line into it: const path = new paper.Path(); // Give the stroke a color @@ -25,23 +20,26 @@ class PaperCanvas extends React.Component { paper.view.draw(); } componentWillReceiveProps (nextProps) { - if (nextProps.tool !== this.props.tool && nextProps.tool instanceof ToolTypes) { + if (nextProps.tool !== this.props.tool) { // TODO switch tool } } componentWillUnmount () { + paper.remove(); } render () { return ( - + { + this.canvas = canvas; + }} + /> ); } } PaperCanvas.propTypes = { - tool: PropTypes.shape({ - name: PropTypes.string.isRequired - }) + tool: PropTypes.oneOf(Object.keys(ToolTypes)).isRequired }; export default PaperCanvas; diff --git a/src/log/log.js b/src/log/log.js index 5ea3bb07..91414157 100644 --- a/src/log/log.js +++ b/src/log/log.js @@ -1,4 +1,4 @@ import minilog from 'minilog'; minilog.enable(); -export default minilog('paint-editor'); +export default minilog('scratch-paint'); diff --git a/src/playground/playground.jsx b/src/playground/playground.jsx index 31ad10cf..8a27161e 100644 --- a/src/playground/playground.jsx +++ b/src/playground/playground.jsx @@ -2,15 +2,12 @@ import React from 'react'; import ReactDOM from 'react-dom'; import PaintEditor from '..'; import {Provider} from 'react-redux'; -import {createStore, applyMiddleware} from 'redux'; -import throttle from 'redux-throttle'; +import {createStore} from 'redux'; import reducer from '../reducers/combine-reducers'; const appTarget = document.createElement('div'); document.body.appendChild(appTarget); -const store = applyMiddleware( - throttle(300, {leading: true, trailing: true}) -)(createStore)( +const store = createStore( reducer, window.__REDUX_DEVTOOLS_EXTENSION__ && window.__REDUX_DEVTOOLS_EXTENSION__() ); diff --git a/src/reducers/tools.js b/src/reducers/tools.js index 28832833..57b7d4c7 100644 --- a/src/reducers/tools.js +++ b/src/reducers/tools.js @@ -8,10 +8,10 @@ const reducer = function (state, action) { if (typeof state === 'undefined') state = initialState; switch (action.type) { case CHANGE_TOOL: - if (action.tool instanceof ToolTypes) { + if (action.tool in ToolTypes) { return action.tool; } - log.warn(`Warning: Tool type does not exist: ${action.tool}`); + log.warn(`Tool type does not exist: ${action.tool}`); /* falls through */ default: return state; @@ -22,10 +22,7 @@ const reducer = function (state, action) { reducer.changeTool = function (tool) { return { type: CHANGE_TOOL, - tool: tool, - meta: { - throttle: 30 - } + tool: tool }; }; diff --git a/src/tools/tool-types.js b/src/tools/tool-types.js index 61524cb7..9a850d80 100644 --- a/src/tools/tool-types.js +++ b/src/tools/tool-types.js @@ -1,12 +1,8 @@ -class ToolTypes { - constructor (name) { - this.name = name; - } - toString () { - return `ToolTypes.${this.name}`; - } -} -ToolTypes.BRUSH = new ToolTypes('BRUSH'); -ToolTypes.ERASER = new ToolTypes('ERASER'); +import keyMirror from 'keymirror'; + +const ToolTypes = keyMirror({ + BRUSH: null, + ERASER: null +}); export default ToolTypes; diff --git a/test/unit/tools-reducer.test.js b/test/unit/tools-reducer.test.js index f0c729f2..428b3118 100644 --- a/test/unit/tools-reducer.test.js +++ b/test/unit/tools-reducer.test.js @@ -4,7 +4,7 @@ import reducer from '../../src/reducers/tools'; test('initialState', () => { let defaultState; - expect(reducer(defaultState /* state */, {type: 'anything'} /* action */) instanceof ToolTypes).toBeTruthy(); + expect(reducer(defaultState /* state */, {type: 'anything'} /* action */) in ToolTypes).toBeTruthy(); }); test('changeTool', () => { From 21f24e339c45b47952592113211b4a588f51e2d6 Mon Sep 17 00:00:00 2001 From: DD Liu Date: Thu, 27 Jul 2017 07:57:24 -0400 Subject: [PATCH 4/4] newline --- .babelrc | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.babelrc b/.babelrc index dc76ced0..2e236249 100644 --- a/.babelrc +++ b/.babelrc @@ -1,4 +1,4 @@ { "plugins": ["transform-object-rest-spread"], "presets": ["es2015", "react"] -} \ No newline at end of file +}