scratch-paint/src/containers/mode-tools.jsx

218 lines
8.5 KiB
React
Raw Normal View History

2017-12-20 14:19:13 -05:00
import paper from '@scratch/paper';
import React from 'react';
import PropTypes from 'prop-types';
import {connect} from 'react-redux';
import bindAll from 'lodash.bindall';
import ModeToolsComponent from '../components/mode-tools/mode-tools.jsx';
import {clearSelectedItems, setSelectedItems} from '../reducers/selected-items';
import {incrementPasteOffset, setClipboardItems} from '../reducers/clipboard';
import {clearSelection, getSelectedLeafItems, getSelectedRootItems} from '../helper/selection';
import {HANDLE_RATIO} from '../helper/math';
2017-12-20 14:19:13 -05:00
class ModeTools extends React.Component {
constructor (props) {
super(props);
bindAll(this, [
2017-12-21 14:49:05 -05:00
'_getSelectedUncurvedPoints',
'_getSelectedUnpointedPoints',
'hasSelectedUncurvedPoints',
'hasSelectedUnpointedPoints',
2017-12-20 14:19:13 -05:00
'handleCopyToClipboard',
2017-12-21 14:49:05 -05:00
'handleCurvePoints',
'handlePasteFromClipboard',
'handlePointPoints'
2017-12-20 14:19:13 -05:00
]);
}
2017-12-21 14:49:05 -05:00
_getSelectedUncurvedPoints () {
const items = [];
const selectedItems = getSelectedLeafItems();
for (const item of selectedItems) {
2017-12-21 14:49:05 -05:00
if (!item.segments) continue;
for (const seg of item.segments) {
if (seg.selected) {
2017-12-21 14:49:05 -05:00
const prev = seg.getPrevious();
const next = seg.getNext();
const isCurved =
(!prev || seg.handleIn.length > 0) &&
(!next || seg.handleOut.length > 0) &&
(prev && next ? seg.handleOut.isColinear(seg.handleIn) : true);
if (!isCurved) items.push(seg);
}
}
}
2017-12-21 14:49:05 -05:00
return items;
}
_getSelectedUnpointedPoints () {
const points = [];
const selectedItems = getSelectedLeafItems();
for (const item of selectedItems) {
if (!item.segments) continue;
for (const seg of item.segments) {
if (seg.selected) {
if (seg.handleIn.length > 0 || seg.handleOut.length > 0) {
points.push(seg);
}
}
}
}
return points;
}
hasSelectedUncurvedPoints () {
const points = this._getSelectedUncurvedPoints();
return points.length > 0;
}
hasSelectedUnpointedPoints () {
const points = this._getSelectedUnpointedPoints();
return points.length > 0;
}
handleCurvePoints () {
let changed;
const points = this._getSelectedUncurvedPoints();
for (const point of points) {
const prev = point.getPrevious();
const next = point.getNext();
const noHandles = point.handleIn.length === 0 && point.handleOut.length === 0;
if (!prev && !next) {
continue;
} else if (prev && next && noHandles) {
// Handles are parallel to the line from prev to next
point.handleIn = prev.point.subtract(next.point)
.normalize()
.multiply(prev.getCurve().length * HANDLE_RATIO);
} else if (prev && !next && point.handleIn.length === 0) {
// Point is end point
// Direction is average of normal at the point and direction to prev point, using the
// normal that points out from the convex side
// Lenth is curve length * HANDLE_RATIO
const convexity = prev.getCurve().getCurvatureAtTime(.1) < 0 ? -1 : 1;
2017-12-21 14:49:05 -05:00
point.handleIn = (prev.getCurve().getNormalAtTime(1)
.multiply(convexity)
2017-12-21 14:49:05 -05:00
.add(prev.point.subtract(point.point).normalize()))
.normalize()
.multiply(prev.getCurve().length * HANDLE_RATIO);
} else if (next && !prev && point.handleOut.length === 0) {
2017-12-21 14:49:05 -05:00
// Point is start point
// Direction is average of normal at the point and direction to prev point, using the
// normal that points out from the convex side
// Lenth is curve length * HANDLE_RATIO
const convexity = point.getCurve().getCurvatureAtTime(.1) < 0 ? -1 : 1;
2017-12-21 14:49:05 -05:00
point.handleOut = (point.getCurve().getNormalAtTime(0)
.multiply(convexity)
2017-12-21 14:49:05 -05:00
.add(next.point.subtract(point.point).normalize()))
.normalize()
.multiply(point.getCurve().length * HANDLE_RATIO);
2017-12-21 14:49:05 -05:00
}
// Point guaranteed to have a handle now. Make the second handle match the length and direction of first.
// This defines a curved point.
if (point.handleIn.length > 0 && next) {
point.handleOut = point.handleIn.multiply(-1);
} else if (point.handleOut.length > 0 && prev) {
point.handleIn = point.handleOut.multiply(-1);
}
changed = true;
}
if (changed) {
this.props.setSelectedItems();
this.props.onUpdateSvg();
}
}
handlePointPoints () {
let changed;
const points = this._getSelectedUnpointedPoints();
for (const point of points) {
const noHandles = point.handleIn.length === 0 && point.handleOut.length === 0;
if (!noHandles) {
point.handleIn = null;
point.handleOut = null;
changed = true;
}
}
if (changed) {
this.props.setSelectedItems();
this.props.onUpdateSvg();
}
}
2017-12-20 14:19:13 -05:00
handleCopyToClipboard () {
const selectedItems = getSelectedRootItems();
if (selectedItems.length > 0) {
const clipboardItems = [];
for (let i = 0; i < selectedItems.length; i++) {
const jsonItem = selectedItems[i].exportJSON({asString: false});
clipboardItems.push(jsonItem);
}
this.props.setClipboardItems(clipboardItems);
}
}
handlePasteFromClipboard () {
clearSelection(this.props.clearSelectedItems);
if (this.props.clipboardItems.length > 0) {
for (let i = 0; i < this.props.clipboardItems.length; i++) {
const item = paper.Base.importJSON(this.props.clipboardItems[i]);
if (item) {
item.selected = true;
}
const placedItem = paper.project.getActiveLayer().addChild(item);
placedItem.position.x += 10 * this.props.pasteOffset;
placedItem.position.y += 10 * this.props.pasteOffset;
}
this.props.incrementPasteOffset();
this.props.setSelectedItems();
paper.project.view.update();
this.props.onUpdateSvg();
}
}
render () {
return (
<ModeToolsComponent
2017-12-21 14:49:05 -05:00
hasSelectedUncurvedPoints={this.hasSelectedUncurvedPoints()}
hasSelectedUnpointedPoints={this.hasSelectedUnpointedPoints()}
2017-12-20 14:19:13 -05:00
onCopyToClipboard={this.handleCopyToClipboard}
2017-12-21 14:49:05 -05:00
onCurvePoints={this.handleCurvePoints}
2017-12-20 14:19:13 -05:00
onPasteFromClipboard={this.handlePasteFromClipboard}
2017-12-21 14:49:05 -05:00
onPointPoints={this.handlePointPoints}
2017-12-20 14:19:13 -05:00
/>
);
}
}
ModeTools.propTypes = {
2017-12-20 14:44:39 -05:00
clearSelectedItems: PropTypes.func.isRequired,
2017-12-20 14:19:13 -05:00
clipboardItems: PropTypes.arrayOf(PropTypes.array),
incrementPasteOffset: PropTypes.func.isRequired,
2017-12-20 14:44:39 -05:00
onUpdateSvg: PropTypes.func.isRequired,
2017-12-20 14:19:13 -05:00
pasteOffset: PropTypes.number,
// Listen on selected items to update hasSelectedPoints
2017-12-21 17:03:28 -05:00
selectedItems:
PropTypes.arrayOf(PropTypes.instanceOf(paper.Item)), // eslint-disable-line react/no-unused-prop-types
2017-12-20 14:44:39 -05:00
setClipboardItems: PropTypes.func.isRequired,
setSelectedItems: PropTypes.func.isRequired
2017-12-20 14:19:13 -05:00
};
const mapStateToProps = state => ({
clipboardItems: state.scratchPaint.clipboard.items,
pasteOffset: state.scratchPaint.clipboard.pasteOffset,
selectedItems: state.scratchPaint.selectedItems
2017-12-20 14:19:13 -05:00
});
const mapDispatchToProps = dispatch => ({
setClipboardItems: items => {
dispatch(setClipboardItems(items));
},
incrementPasteOffset: () => {
dispatch(incrementPasteOffset());
},
clearSelectedItems: () => {
dispatch(clearSelectedItems());
},
setSelectedItems: () => {
dispatch(setSelectedItems(getSelectedLeafItems()));
}
});
export default connect(
mapStateToProps,
mapDispatchToProps
)(ModeTools);