From 37803aa4d38bb055c4ac7b93f88be37b7cc5c983 Mon Sep 17 00:00:00 2001 From: Paul Kaplan <pkaplan@media.mit.edu> Date: Tue, 21 Mar 2017 09:29:15 -0400 Subject: [PATCH] Add sound and costume tabs --- package.json | 1 + src/components/asset-panel/asset-panel.css | 18 +++ src/components/asset-panel/asset-panel.jsx | 23 ++++ .../icon--sound.svg} | Bin src/components/asset-panel/selector.css | 38 +++++++ src/components/asset-panel/selector.jsx | 61 ++++++++++ src/components/blocks/blocks.css | 12 +- src/components/gui/gui.css | 48 ++++++-- src/components/gui/gui.jsx | 51 +++++++-- src/containers/costume-tab.jsx | 98 ++++++++++++++++ src/containers/sound-library.jsx | 3 +- src/containers/sound-tab.jsx | 107 ++++++++++++++++++ src/css/colors.css | 5 +- 13 files changed, 433 insertions(+), 32 deletions(-) create mode 100644 src/components/asset-panel/asset-panel.css create mode 100644 src/components/asset-panel/asset-panel.jsx rename src/components/{target-pane/icon--sound-dark.svg => asset-panel/icon--sound.svg} (100%) create mode 100644 src/components/asset-panel/selector.css create mode 100644 src/components/asset-panel/selector.jsx create mode 100644 src/containers/costume-tab.jsx create mode 100644 src/containers/sound-tab.jsx diff --git a/package.json b/package.json index 41888d82b..9250d26e1 100644 --- a/package.json +++ b/package.json @@ -52,6 +52,7 @@ "react-modal": "1.7.2", "react-redux": "5.0.3", "react-style-proptype": "2.0.1", + "react-tabs": "^0.8.2", "redux": "3.6.0", "redux-throttle": "0.1.1", "scratch-audio": "latest", diff --git a/src/components/asset-panel/asset-panel.css b/src/components/asset-panel/asset-panel.css new file mode 100644 index 000000000..8ff5ca4cb --- /dev/null +++ b/src/components/asset-panel/asset-panel.css @@ -0,0 +1,18 @@ +@import "../../css/units.css"; +@import "../../css/colors.css"; + +.wrapper { + display: flex; + flex-grow: 1; + border: 1px solid $ui-pane-border; + border-top-right-radius: $space; + background: $ui-pane-gray; + font-family: "Helvetica Neue", Helvetica, Arial, sans-serif; + font-size: 0.85rem; +} + +.detail-area { + flex-grow: 1; + flex-shrink: 0; + border-left: 1px solid $ui-pane-border; +} diff --git a/src/components/asset-panel/asset-panel.jsx b/src/components/asset-panel/asset-panel.jsx new file mode 100644 index 000000000..2d6ea710b --- /dev/null +++ b/src/components/asset-panel/asset-panel.jsx @@ -0,0 +1,23 @@ +const React = require('react'); + +const Box = require('../box/box.jsx'); +const Selector = require('./selector.jsx'); +const styles = require('./asset-panel.css'); + +const AssetPanel = props => ( + <Box className={styles.wrapper}> + <Selector + className={styles.selector} + {...props} + /> + <Box className={styles.detailArea}> + {/* @todo editor area */} + </Box> + </Box> +); + +AssetPanel.propTypes = { + ...Selector.propTypes +}; + +module.exports = AssetPanel; diff --git a/src/components/target-pane/icon--sound-dark.svg b/src/components/asset-panel/icon--sound.svg similarity index 100% rename from src/components/target-pane/icon--sound-dark.svg rename to src/components/asset-panel/icon--sound.svg diff --git a/src/components/asset-panel/selector.css b/src/components/asset-panel/selector.css new file mode 100644 index 000000000..8964c9b13 --- /dev/null +++ b/src/components/asset-panel/selector.css @@ -0,0 +1,38 @@ +/* Need to use a fixed height for the new container to make list scrollable */ +$new-height: 60px; + +.wrapper { + width: 200px; + position: relative; +} + +.new-item { + background: white; + border-bottom: 1px solid #ddd; + height: $new-height; + display: flex; + align-items: center; + padding-left: 1.75rem; + cursor: pointer; + font-size: 0.85rem; +} + +.list-area { + position: absolute; + width: 100%; + top: 60px; + height: calc(100% - $new-height); + overflow-y: scroll; +} + +.list-item { + width: 5rem; + min-height: 5rem; + margin: 1rem auto; +} + +.delete-button { + position: absolute; + top: 2px; + right: 2px; +} diff --git a/src/components/asset-panel/selector.jsx b/src/components/asset-panel/selector.jsx new file mode 100644 index 000000000..5250c3e7d --- /dev/null +++ b/src/components/asset-panel/selector.jsx @@ -0,0 +1,61 @@ +const React = require('react'); + +const SpriteSelectorItem = require('../sprite-selector-item/sprite-selector-item.jsx'); +const Box = require('../box/box.jsx'); +const styles = require('./selector.css'); + +const Selector = props => { + const { + items, + newText, + selectedItemIndex, + onDeleteClick, + onItemClick, + onNewClick + } = props; + + return ( + <Box className={styles.wrapper}> + <Box + className={styles.newItem} + onClick={onNewClick} + > + {newText} + </Box> + <Box className={styles.listArea}> + {items.map((item, index) => { + const _onItemClick = () => onItemClick(item); + const _onDeleteClick = e => { + e.stopPropagation(); + onDeleteClick(item); + }; + return ( + <SpriteSelectorItem + className={styles.listItem} + costumeURL={item.image} + key={`asset-${index}`} + name={item.name} + selected={index === selectedItemIndex} + onClick={_onItemClick} + onDeleteButtonClick={_onDeleteClick} + /> + ); + })} + </Box> + </Box> + ); +}; + +Selector.propTypes = { + items: React.PropTypes.arrayOf(React.PropTypes.shape({ + image: React.PropTypes.string, + name: React.PropTypes.string + })), + newText: React.PropTypes.string, + onDeleteClick: React.PropTypes.func, + onItemClick: React.PropTypes.func, + onNewClick: React.PropTypes.func, + selectedItemIndex: React.PropTypes.number +}; + +module.exports = Selector; diff --git a/src/components/blocks/blocks.css b/src/components/blocks/blocks.css index 137a8f886..80689e48c 100644 --- a/src/components/blocks/blocks.css +++ b/src/components/blocks/blocks.css @@ -1,4 +1,7 @@ -$border-style: 1px solid #dbdbdb; +@import "../../css/units.css"; +@import "../../css/colors.css"; + +$border-style: 1px solid $ui-pane-border; .blocks :global(.injectionDiv){ position: absolute; @@ -7,12 +10,7 @@ $border-style: 1px solid #dbdbdb; bottom: 0; left: 0; border: $border-style; - - /* - @todo: using _space doesn't compute to the right amount? - Related to `scratch-blocks` - */ - border-top-right-radius: 0.75rem; + border-top-right-radius: $space; } .blocks :global(.blocklyMainBackground) { diff --git a/src/components/gui/gui.css b/src/components/gui/gui.css index e587122a7..43a0191b7 100644 --- a/src/components/gui/gui.css +++ b/src/components/gui/gui.css @@ -21,24 +21,48 @@ height: 100%; } -.blocks-wrapper { - /* - scratch-blocks is based on absolute positioning, which injects - inside this element and becomes the child - */ +.editor-wrapper { + flex-basis: 600px; + flex-grow: 1; + flex-shrink: 0; position: relative; - flex-basis: 600px; + display: flex; + flex-direction: column; +} + +.tab-list { + height: $stage-menu-height; + display: flex; + align-items: flex-end; + flex-shrink: 0; + + font-family: "Helvetica Neue", Helvetica, Arial, sans-serif; + font-size: 0.85rem; + + /* Overrides for react-tabs styling */ + margin: 0 0 0 1rem !important; + border-bottom: 0 !important; +} + +.tabs { + position: relative; flex-grow: 1; flex-shrink: 0; + display: flex; + flex-direction: column; +} - /* - Normally we'd use padding, but the absolute positioning ignores it, - so use margin instead. Temporary, until tabs are inserted. - */ - margin-top: $stage-menu-height; +.tab-panel { + position: relative; + flex-grow: 1; + flex-shrink: 0; + display: flex; +} - background: #e8edf1; +.blocks-wrapper { + flex-grow: 1; + position: relative; } .stage-and-target-wrapper { diff --git a/src/components/gui/gui.jsx b/src/components/gui/gui.jsx index 36dbec0e2..a131ad40a 100644 --- a/src/components/gui/gui.jsx +++ b/src/components/gui/gui.jsx @@ -1,12 +1,14 @@ const React = require('react'); const VM = require('scratch-vm'); - const Blocks = require('../../containers/blocks.jsx'); +const CostumeTab = require('../../containers/costume-tab.jsx'); const GreenFlag = require('../../containers/green-flag.jsx'); const TargetPane = require('../../containers/target-pane.jsx'); +const SoundTab = require('../../containers/sound-tab.jsx'); const Stage = require('../../containers/stage.jsx'); const StopAll = require('../../containers/stop-all.jsx'); const MenuBar = require('../menu-bar/menu-bar.jsx'); +const {Tab, Tabs, TabList, TabPanel} = require('react-tabs'); const Box = require('../box/box.jsx'); const styles = require('./gui.css'); @@ -25,6 +27,14 @@ const GUIComponent = props => { </Box> ); } + + // @todo hack to resize blockly manually in case resize happened while hidden + const handleTabSelect = tabIndex => { + if (tabIndex === 0) { + setTimeout(() => window.dispatchEvent(new Event('resize'))); + } + }; + return ( <Box className={styles.pageWrapper} @@ -33,14 +43,35 @@ const GUIComponent = props => { <MenuBar /> <Box className={styles.bodyWrapper}> <Box className={styles.flexWrapper}> - <Box className={styles.blocksWrapper}> - <Blocks - grow={1} - options={{ - media: `${basePath}static/blocks-media/` - }} - vm={vm} - /> + <Box className={styles.editorWrapper}> + <Tabs + className={styles.tabs} + forceRenderTabPanel={true} // eslint-disable-line react/jsx-boolean-value + onSelect={handleTabSelect} + > + <TabList className={styles.tabList}> + <Tab>Scripts</Tab> + <Tab>Costumes</Tab> + <Tab>Sounds</Tab> + </TabList> + <TabPanel className={styles.tabPanel}> + <Box className={styles.blocksWrapper}> + <Blocks + grow={1} + options={{ + media: `${basePath}static/blocks-media/` + }} + vm={vm} + /> + </Box> + </TabPanel> + <TabPanel className={styles.tabPanel}> + <CostumeTab vm={vm} /> + </TabPanel> + <TabPanel className={styles.tabPanel}> + <SoundTab vm={vm} /> + </TabPanel> + </Tabs> </Box> <Box className={styles.stageAndTargetWrapper} > @@ -48,7 +79,7 @@ const GUIComponent = props => { <GreenFlag vm={vm} /> <StopAll vm={vm} /> </Box> - + <Box className={styles.stageWrapper} > <Stage shrink={0} diff --git a/src/containers/costume-tab.jsx b/src/containers/costume-tab.jsx new file mode 100644 index 000000000..0f569fcd4 --- /dev/null +++ b/src/containers/costume-tab.jsx @@ -0,0 +1,98 @@ +const React = require('react'); +const bindAll = require('lodash.bindall'); + +const VM = require('scratch-vm'); + +const AssetPanel = require('../components/asset-panel/asset-panel.jsx'); + +const {connect} = require('react-redux'); + +const { + openCostumeLibrary +} = require('../reducers/modals'); + +class CostumeTab extends React.Component { + constructor (props) { + super(props); + bindAll(this, [ + 'handleSelectCostume', + 'handleDeleteCostume' + ]); + this.state = {selectedCostumeIndex: 0}; + } + + handleSelectCostume (item) { + this.setState({selectedCostumeIndex: this.props.vm.editingTarget.getCostumeIndexByName(item.name)}); + } + + handleDeleteCostume (item) { + // @todo the VM should handle all of this logic + const {editingTarget} = this.props.vm; + const i = editingTarget.getCostumeIndexByName(item.name); + + if (i === editingTarget.currentCostume) { + editingTarget.setCostume(i - 1); + } + + editingTarget.sprite.costumes = editingTarget.sprite.costumes + .slice(0, i) + .concat(editingTarget.sprite.costumes.slice(i + 1)); + this.props.vm.emitTargetsUpdate(); + // @todo not sure if this is getting redrawn correctly + this.props.vm.runtime.requestRedraw(); + + this.setState({ + selectedCostumeIndex: this.state.selectedCostumeIndex % editingTarget.sprite.costumes.length + }); + } + + render () { + const { + vm, + onNewCostumeClick + } = this.props; + + const costumes = vm.editingTarget ? vm.editingTarget.sprite.costumes.map(costume => ( + { + image: costume.skin, + name: costume.name + } + )) : []; + + const addText = vm.editingTarget && vm.editingTarget.isStage ? 'Add Backdrop' : 'Add Costume'; + + return ( + <AssetPanel + items={costumes} + newText={addText} + selectedItemIndex={this.state.selectedCostumeIndex} + onDeleteClick={this.handleDeleteCostume} + onItemClick={this.handleSelectCostume} + onNewClick={onNewCostumeClick} + /> + ); + } +} + +CostumeTab.propTypes = { + ...AssetPanel.propTypes, + vm: React.PropTypes.instanceOf(VM) +}; + +const mapStateToProps = state => ({ + editingTarget: state.targets.editingTarget, + sprites: state.targets.sprites, + costumeLibraryVisible: state.modals.costumeLibrary +}); + +const mapDispatchToProps = dispatch => ({ + onNewCostumeClick: e => { + e.preventDefault(); + dispatch(openCostumeLibrary()); + } +}); + +module.exports = connect( + mapStateToProps, + mapDispatchToProps +)(CostumeTab); diff --git a/src/containers/sound-library.jsx b/src/containers/sound-library.jsx index 010f8e23e..4db5ee3ea 100644 --- a/src/containers/sound-library.jsx +++ b/src/containers/sound-library.jsx @@ -4,7 +4,8 @@ const VM = require('scratch-vm'); const AudioEngine = require('scratch-audio'); const LibaryComponent = require('../components/library/library.jsx'); -const soundIcon = require('../components/target-pane/icon--sound-dark.svg'); + +const soundIcon = require('../components/asset-panel/icon--sound.svg'); const soundLibraryContent = require('../lib/libraries/sounds.json'); diff --git a/src/containers/sound-tab.jsx b/src/containers/sound-tab.jsx new file mode 100644 index 000000000..74578b590 --- /dev/null +++ b/src/containers/sound-tab.jsx @@ -0,0 +1,107 @@ +const React = require('react'); +const bindAll = require('lodash.bindall'); + +const VM = require('scratch-vm'); + +const AssetPanel = require('../components/asset-panel/asset-panel.jsx'); +const soundIcon = require('../components/asset-panel/icon--sound.svg'); + +const {connect} = require('react-redux'); + +const { + openSoundLibrary +} = require('../reducers/modals'); + +class SoundTab extends React.Component { + constructor (props) { + super(props); + bindAll(this, [ + 'getSoundIndexByName', + 'handleSelectSound', + 'handleDeleteSound' + ]); + this.state = {selectedSoundIndex: 0}; + } + + getSoundIndexByName (name) { + // @todo should be in VM + let i = -1; + this.props.vm.editingTarget.sprite.sounds.forEach((sound, soundIndex) => { + if (sound.name === name) { + i = soundIndex; + } + }); + return i; + } + + handleSelectSound (item) { + const selectedSoundIndex = this.getSoundIndexByName(item.name); + const sound = this.props.vm.editingTarget.sprite.sounds[selectedSoundIndex]; + this.props.vm.editingTarget.audioPlayer.playSound(sound.md5); + this.setState({selectedSoundIndex}); + } + + handleDeleteSound (item) { + // @todo the VM should handle all of this logic + const {editingTarget} = this.props.vm; + const i = this.getSoundIndexByName(item.name); + editingTarget.sprite.sounds = editingTarget.sprite.sounds + .slice(0, i) + .concat(editingTarget.sprite.sounds.slice(i + 1)); + this.props.vm.emitTargetsUpdate(); + this.props.vm.runtime.requestRedraw(); + + this.setState({ + selectedSoundIndex: this.state.selectedSoundIndex % editingTarget.sprite.sounds.length + }); + } + + render () { + const { + vm, + onNewSoundClick + } = this.props; + + const sounds = vm.editingTarget ? vm.editingTarget.sprite.sounds.map(sound => ( + { + image: soundIcon, + name: sound.name + } + )) : []; + + + return ( + <AssetPanel + items={sounds} + newText={'Add Sound'} + selectedItemIndex={this.state.selectedSoundIndex} + onDeleteClick={this.handleDeleteSound} + onItemClick={this.handleSelectSound} + onNewClick={onNewSoundClick} + /> + ); + } +} + +SoundTab.propTypes = { + ...AssetPanel.propTypes, + vm: React.PropTypes.instanceOf(VM) +}; + +const mapStateToProps = state => ({ + editingTarget: state.targets.editingTarget, + sprites: state.targets.sprites, + soundLibraryVisible: state.modals.soundLibrary +}); + +const mapDispatchToProps = dispatch => ({ + onNewSoundClick: e => { + e.preventDefault(); + dispatch(openSoundLibrary()); + } +}); + +module.exports = connect( + mapStateToProps, + mapDispatchToProps +)(SoundTab); diff --git a/src/css/colors.css b/src/css/colors.css index 03bf83a7a..c223ea39e 100644 --- a/src/css/colors.css +++ b/src/css/colors.css @@ -1,2 +1,3 @@ -$ui-pane-gray: #f9f9f9; -$blue: #4c97ff; +$ui-pane-border: #D9D9D9; +$ui-pane-gray: #F9F9F9; +$blue: #4C97FF; -- GitLab