import paper from '@scratch/paper'; import PropTypes from 'prop-types'; import React from 'react'; import {connect} from 'react-redux'; import bindAll from 'lodash.bindall'; import Modes from '../modes/modes'; import {MIXED} from '../helper/style-path'; import {changeFillColor, DEFAULT_COLOR} from '../reducers/fill-color'; import {changeStrokeColor} from '../reducers/stroke-color'; import {changeMode} from '../reducers/modes'; import {clearSelectedItems, setSelectedItems} from '../reducers/selected-items'; import {clearSelection, getSelectedLeafItems} from '../helper/selection'; import RectTool from '../helper/tools/rect-tool'; import RectModeComponent from '../components/rect-mode/rect-mode.jsx'; class RectMode extends React.Component { constructor (props) { super(props); bindAll(this, [ 'activateTool', 'deactivateTool' ]); } componentDidMount () { if (this.props.isRectModeActive) { this.activateTool(this.props); } } componentWillReceiveProps (nextProps) { if (this.tool && nextProps.colorState !== this.props.colorState) { this.tool.setColorState(nextProps.colorState); } if (this.tool && nextProps.selectedItems !== this.props.selectedItems) { this.tool.onSelectionChanged(nextProps.selectedItems); } if (nextProps.isRectModeActive && !this.props.isRectModeActive) { this.activateTool(); } else if (!nextProps.isRectModeActive && this.props.isRectModeActive) { this.deactivateTool(); } } shouldComponentUpdate (nextProps) { return nextProps.isRectModeActive !== this.props.isRectModeActive; } activateTool () { clearSelection(this.props.clearSelectedItems); // If fill and stroke color are both mixed/transparent/absent, set fill to default and stroke to transparent. // If exactly one of fill or stroke color is set, set the other one to transparent. // This way the tool won't draw an invisible state, or be unclear about what will be drawn. const {fillColor, strokeColor, strokeWidth} = this.props.colorState; const fillColorPresent = fillColor !== MIXED && fillColor !== null; const strokeColorPresent = strokeColor !== MIXED && strokeColor !== null && strokeWidth !== null && strokeWidth !== 0; if (!fillColorPresent && !strokeColorPresent) { this.props.onChangeFillColor(DEFAULT_COLOR); this.props.onChangeStrokeColor(null); } else if (!fillColorPresent && strokeColorPresent) { this.props.onChangeFillColor(null); } else if (fillColorPresent && !strokeColorPresent) { this.props.onChangeStrokeColor(null); } this.tool = new RectTool( this.props.setSelectedItems, this.props.clearSelectedItems, this.props.onUpdateSvg ); this.tool.setColorState(this.props.colorState); this.tool.activate(); } deactivateTool () { this.tool.deactivateTool(); this.tool.remove(); this.tool = null; } render () { return ( ); } } RectMode.propTypes = { clearSelectedItems: PropTypes.func.isRequired, colorState: PropTypes.shape({ fillColor: PropTypes.string, strokeColor: PropTypes.string, strokeWidth: PropTypes.number }).isRequired, handleMouseDown: PropTypes.func.isRequired, isRectModeActive: PropTypes.bool.isRequired, onChangeFillColor: PropTypes.func.isRequired, onChangeStrokeColor: PropTypes.func.isRequired, onUpdateSvg: PropTypes.func.isRequired, selectedItems: PropTypes.arrayOf(PropTypes.instanceOf(paper.Item)), setSelectedItems: PropTypes.func.isRequired }; const mapStateToProps = state => ({ colorState: state.scratchPaint.color, isRectModeActive: state.scratchPaint.mode === Modes.RECT, selectedItems: state.scratchPaint.selectedItems }); const mapDispatchToProps = dispatch => ({ clearSelectedItems: () => { dispatch(clearSelectedItems()); }, setSelectedItems: () => { dispatch(setSelectedItems(getSelectedLeafItems())); }, handleMouseDown: () => { dispatch(changeMode(Modes.RECT)); }, onChangeFillColor: fillColor => { dispatch(changeFillColor(fillColor)); }, onChangeStrokeColor: strokeColor => { dispatch(changeStrokeColor(strokeColor)); } }); export default connect( mapStateToProps, mapDispatchToProps )(RectMode);