import bindAll from 'lodash.bindall'; import defaultsDeep from 'lodash.defaultsdeep'; import PropTypes from 'prop-types'; import React from 'react'; import CustomProceduresComponent from '../components/custom-procedures/custom-procedures.jsx'; import LazyScratchBlocks from '../lib/tw-lazy-scratch-blocks'; import {connect} from 'react-redux'; import Color from './custom-procedures-util/color.js'; function createHeavyColorFromHex(hex, percentage) { const rgb = Color.hexToRgb(hex); const hsv = Color.rgbToHsv(rgb); if (hsv.v > 0.6) { // so that pure white can still get color change hsv.v -= percentage / 2; } // only white-black have this property // so we can avoid adding red to them if (!(hsv.h === 0 && hsv.s === 0)) { hsv.s += percentage * hsv.v; } // make sure values arent invalid if (hsv.v > 1) hsv.v = 1; if (hsv.v < 0) hsv.v = 0; if (hsv.s > 1) hsv.s = 1; if (hsv.s < 0) hsv.s = 0; const newRgb = Color.hsvToRgb(hsv); return Color.rgbToHex(newRgb); } class CustomProcedures extends React.Component { constructor (props) { super(props); bindAll(this, [ 'handleAddLabel', 'handleAddBoolean', 'handleAddCommand', 'handleAddTextNumber', 'handleToggleWarp', 'handleToggleReturns', 'handleCancel', 'handleOk', 'handleChangeType', 'handleBlockColorChange', 'setHexBlockColor', 'setBlocks', 'handleTestFunction' ]); this.state = { rtlOffset: 0, warp: false, returns: false, editing: false, blockColor: '#000000', type: 'statement' }; } componentWillUnmount () { if (this.workspace) { this.workspace.dispose(); } } setBlocks (blocksRef) { if (!blocksRef) return; this.blocks = blocksRef; const workspaceConfig = defaultsDeep({}, CustomProcedures.defaultOptions, this.props.options, {rtl: this.props.isRtl} ); // @todo This is a hack to make there be no toolbox. const ScratchBlocks = LazyScratchBlocks.get(); const oldDefaultToolbox = ScratchBlocks.Blocks.defaultToolbox; ScratchBlocks.Blocks.defaultToolbox = null; this.workspace = ScratchBlocks.inject(this.blocks, workspaceConfig); ScratchBlocks.Blocks.defaultToolbox = oldDefaultToolbox; // Create the procedure declaration block for editing the mutation. this.mutationRoot = this.workspace.newBlock('procedures_declaration'); // Make the declaration immovable, undeletable and have no context menu this.mutationRoot.setMovable(false); this.mutationRoot.setDeletable(false); this.mutationRoot.contextMenu = false; this.workspace.addChangeListener(() => { this.mutationRoot.onChangeFn(); // Keep the block centered on the workspace const metrics = this.workspace.getMetrics(); const {x, y} = this.mutationRoot.getRelativeToSurfaceXY(); const dy = (metrics.viewHeight / 2) - (this.mutationRoot.height / 2) - y; let dx; if (this.props.isRtl) { // // TODO: https://github.com/LLK/scratch-gui/issues/2838 // This is temporary until we can figure out what's going on width // block positioning on the workspace for RTL. // Workspace is always origin top-left, with x increasing to the right // Calculate initial starting offset and save it, every other move // has to take the original offset into account. // Calculate a new left postion based on new width // Convert current x position into LTR (mirror) x position (uses original offset) // Use the difference between ltrX and mirrorX as the amount to move const ltrX = ((metrics.viewWidth / 2) - (this.mutationRoot.width / 2) + 25); const mirrorX = x - ((x - this.state.rtlOffset) * 2); if (mirrorX === ltrX) { return; } dx = mirrorX - ltrX; const midPoint = metrics.viewWidth / 2; if (x === 0) { // if it's the first time positioning, it should always move right if (this.mutationRoot.width < midPoint) { dx = ltrX; } else if (this.mutationRoot.width < metrics.viewWidth) { dx = midPoint - ((metrics.viewWidth - this.mutationRoot.width) / 2); } else { dx = midPoint + (this.mutationRoot.width - metrics.viewWidth); } this.mutationRoot.moveBy(dx, dy); this.setState({rtlOffset: this.mutationRoot.getRelativeToSurfaceXY().x}); return; } if (this.mutationRoot.width > metrics.viewWidth) { dx = dx + this.mutationRoot.width - metrics.viewWidth; } } else { dx = (metrics.viewWidth / 2) - (this.mutationRoot.width / 2) - x; // If the procedure declaration is wider than the view width, // keep the right-hand side of the procedure in view. if (this.mutationRoot.width > metrics.viewWidth) { dx = metrics.viewWidth - this.mutationRoot.width - x; } } this.mutationRoot.moveBy(dx, dy); }); this.mutationRoot.domToMutation(this.props.mutator); this.mutationRoot.initSvg(); this.mutationRoot.render(); this.setState({ warp: this.mutationRoot.getWarp(), returns: this.mutationRoot.getReturns(), editing: this.mutationRoot.getEdited(), // sometimes color[0] exists but sometimes it doesnt // i can blame gsa for this or just do nothing about it :troll: blockColor: this.mutationRoot.color ? this.mutationRoot.color[0] : this.mutationRoot.colour_ }); // Allow the initial events to run to position this block, then focus. setTimeout(() => { this.mutationRoot.focusLastEditor_(); // if editing, apply block color if (this.state.editing && this.mutationRoot.color) { this.handleBlockColorChange({ target: { value: this.mutationRoot.color[0] } }); } }); } handleCancel () { this.props.onRequestClose(); } handleOk () { this.mutationRoot.setEdited(true) const newMutation = this.mutationRoot ? this.mutationRoot.mutationToDom(true) : null; this.props.onRequestClose(newMutation); } handleAddLabel () { if (this.mutationRoot) { this.mutationRoot.addLabelExternal(); } } handleAddBoolean () { if (this.mutationRoot) { this.mutationRoot.addBooleanExternal(); } } handleAddCommand () { if (this.mutationRoot) { this.mutationRoot.addCommandExternal(); } } handleAddTextNumber () { if (this.mutationRoot) { this.mutationRoot.addStringNumberExternal(); } } handleToggleWarp () { if (this.mutationRoot) { const newWarp = !this.mutationRoot.getWarp(); this.mutationRoot.setWarp(newWarp); this.setState({warp: newWarp}); } } handleToggleReturns () { if (this.mutationRoot) { const newReturns = !this.mutationRoot.getReturns(); this.mutationRoot.setReturns(newReturns); this.handleChangeType(newReturns ? 'string' : 'statement'); this.setState({returns: newReturns}); } } handleChangeType (value) { if (this.mutationRoot) { const newType = value; this.mutationRoot.setType(newType); this.setState({type: newType}); } } handleBlockColorChange (element) { if (this.mutationRoot) { const newColor = element.target.value; this.mutationRoot.setColor( newColor, createHeavyColorFromHex(newColor, 0.15), createHeavyColorFromHex(newColor, 0.25) ); this.setState({blockColor: newColor}); } } setHexBlockColor (hex) { this.handleBlockColorChange({ target: { value: hex } }); } handleTestFunction (type) { if (this.mutationRoot) { switch (type) { case 'icon': { const iconUri = ''; this.mutationRoot.setImage(iconUri); break; } case 'removeicon': { this.mutationRoot.unsetImage(); break; } } } } render () { return ( ); } } CustomProcedures.propTypes = { isRtl: PropTypes.bool, mutator: PropTypes.instanceOf(Element), onRequestClose: PropTypes.func.isRequired, options: PropTypes.shape({ media: PropTypes.string, zoom: PropTypes.shape({ controls: PropTypes.bool, wheel: PropTypes.bool, startScale: PropTypes.number }), comments: PropTypes.bool, collapse: PropTypes.bool }) }; CustomProcedures.defaultOptions = { zoom: { controls: false, wheel: false, startScale: 0.9 }, comments: false, collapse: false, scrollbars: true }; CustomProcedures.defaultProps = { options: CustomProcedures.defaultOptions }; const mapStateToProps = state => ({ isRtl: state.locales.isRtl, mutator: state.scratchGui.customProcedures.mutator }); export default connect( mapStateToProps )(CustomProcedures);