From 82fc11ebab32a06975ec9932cbf940a4adffde39 Mon Sep 17 00:00:00 2001 From: DD Date: Tue, 29 Aug 2017 17:20:08 -0400 Subject: [PATCH 1/6] add structure --- src/components/paint-editor.jsx | 108 ++++++++++++++++++++++++++++---- 1 file changed, 96 insertions(+), 12 deletions(-) diff --git a/src/components/paint-editor.jsx b/src/components/paint-editor.jsx index 1459fe35..46dd614a 100644 --- a/src/components/paint-editor.jsx +++ b/src/components/paint-editor.jsx @@ -4,6 +4,8 @@ import PaperCanvas from '../containers/paper-canvas.jsx'; import BrushMode from '../containers/brush-mode.jsx'; import EraserMode from '../containers/eraser-mode.jsx'; +import styles from './paint-editor.css'; + class PaintEditorComponent extends React.Component { constructor (props) { super(props); @@ -16,19 +18,101 @@ class PaintEditorComponent extends React.Component { this.setState({canvas: canvas}); } render () { - // Modes can't work without a canvas, so we don't render them until we have it - if (this.state.canvas) { - return ( -
- - - -
- ); - } return ( -
- +
+ {/* First row */} +
+ {/* Name field */} +
+ {/* Todo use Label and BufferedInput from Gui */} + +
+ + {/* Undo/Redo */} +
+
+ + +
+
+ + {/* To be Front/back */} +
+ + +
+ + {/* To be Group/Ungroup */} +
+ + +
+
+ + {/* Second Row */} +
+ {/* To be fill */} +
+ +
+ {/* To be stroke */} +
+ +
+ +
+ Mode tools +
+
+ +
+ {/* Modes */} + {this.state.canvas ? ( +
+ + +
+ ) : null} + + {/* Canvas */} +
+ +
+
); } From bb71f56817be23e9883e83c38550e48a1ea003e5 Mon Sep 17 00:00:00 2001 From: DD Date: Wed, 30 Aug 2017 10:50:24 -0400 Subject: [PATCH 2/6] css --- src/components/paint-editor.css | 145 ++++++++++++++++++++++++++++++++ 1 file changed, 145 insertions(+) create mode 100644 src/components/paint-editor.css diff --git a/src/components/paint-editor.css b/src/components/paint-editor.css new file mode 100644 index 00000000..37991495 --- /dev/null +++ b/src/components/paint-editor.css @@ -0,0 +1,145 @@ +$ui-pane-border: #D9D9D9; +$ui-pane-gray: #F9F9F9; +$ui-background-blue: #e8edf1; + +$text-primary: #575e75; + +$motion-primary: #4C97FF; +$motion-tertiary: #3373CC; +$motion-transparent: hsla(215, 100%, 65%, 0.20); + +$red-primary: #FF661A; +$red-tertiary: #E64D00; + +$sound-primary: #CF63CF; +$sound-tertiary: #A63FA6; + +$control-primary: #FFAB19; + +$form-border: #E9EEF2; + +$space: 0.5rem; + +$sprites-per-row: 5; + +$menu-bar-height: 3rem; +$sprite-info-height: 6rem; +$stage-menu-height: 2.75rem; + +$library-header-height: 4.375rem; + +$form-radius: calc($space / 2); + + +.editor-container { + display: flex; + flex-direction: column; + padding: calc(2 * $space); +} + +.row { + display: flex; + flex-direction: row; + align-items: center; +} + +.row + .row { + margin-top: calc(2 * $space); +} + +.input-group + .input-group { + margin-left: calc(2 * $space); +} + +.waveform-container { + display: flex; + justify-content: space-around; + align-items: center; + width: 100%; + + position: relative; + + background: $ui-pane-gray; + border: 1px solid $ui-pane-border; + border-radius: 5px; + padding: 3px; +} + +$border-radius: 0.25rem; + +.button { + height: 2rem; + padding: 0.25rem; + outline: none; + background: white; + border-radius: $border-radius; + border: 1px solid #ddd; + cursor: pointer; + font-size: 0.85rem; + transition: 0.2s; +} + +.button > img { + flex-grow: 1; + max-width: 100%; + max-height: 100%; + min-width: 1.5rem; +} + +.trim-button { + display: flex; + align-items: center; + padding-right: 10px; /* To equalize with empty whitespace from image on left */ +} + +.trim-button > img { + width: 1.5rem; +} + +.trim-button-active { + filter: hue-rotate(155deg); /* @todo replace blue -> red with real submit icon */ +} + +.input-group-right { + flex-grow: 1; + display: flex; + flex-direction: row-reverse; +} + +.effect-button { + flex-basis: 150px; + color: $text-primary; +} + +.effect-button + .effect-button { + margin: 0; +} + +.effect-button img { + width: 60px; + height: 60px; +} + +.button-group { + margin: 0 1rem; +} + +.button-group .button { + border-radius: 0; + border-left: none; +} + +.button-group .button:last-of-type { + border-top-right-radius: $border-radius; + border-bottom-right-radius: $border-radius; +} + +.button-group .button:first-of-type { + border-left: 1px solid #ddd; + border-top-left-radius: $border-radius; + border-bottom-left-radius: $border-radius; +} + +.button:disabled > img { + opacity: 0.25; +} From 4c6898657b744ee3386670fe33ae2a464b29836d Mon Sep 17 00:00:00 2001 From: DD Date: Wed, 6 Sep 2017 18:01:49 -0400 Subject: [PATCH 3/6] Bring in buffered input from GUI --- src/components/forms/buffered-input-hoc.jsx | 60 +++++++++++++++++++ src/components/forms/input.css | 42 ++++++++++++++ src/components/forms/input.jsx | 27 +++++++++ src/components/forms/label.css | 19 ++++++ src/components/forms/label.jsx | 25 ++++++++ src/components/paint-editor.css | 55 ++++++++---------- src/components/paint-editor.jsx | 64 +++++++++++++++------ src/containers/paper-canvas.css | 7 +++ src/containers/paper-canvas.jsx | 5 ++ src/css/colors.css | 19 ++++++ src/css/units.css | 11 ++++ 11 files changed, 286 insertions(+), 48 deletions(-) create mode 100644 src/components/forms/buffered-input-hoc.jsx create mode 100644 src/components/forms/input.css create mode 100644 src/components/forms/input.jsx create mode 100644 src/components/forms/label.css create mode 100644 src/components/forms/label.jsx create mode 100644 src/containers/paper-canvas.css create mode 100644 src/css/colors.css create mode 100644 src/css/units.css diff --git a/src/components/forms/buffered-input-hoc.jsx b/src/components/forms/buffered-input-hoc.jsx new file mode 100644 index 00000000..246dfc03 --- /dev/null +++ b/src/components/forms/buffered-input-hoc.jsx @@ -0,0 +1,60 @@ +import bindAll from 'lodash.bindall'; +import PropTypes from 'prop-types'; +import React from 'react'; + +/** + * Higher Order Component to manage inputs that submit on blur and + * @param {React.Component} Input text input that consumes onChange, onBlur, onKeyPress + * @returns {React.Component} Buffered input that calls onSubmit on blur and + */ +export default function (Input) { + class BufferedInput extends React.Component { + constructor (props) { + super(props); + bindAll(this, [ + 'handleChange', + 'handleKeyPress', + 'handleFlush' + ]); + this.state = { + value: null + }; + } + handleKeyPress (e) { + if (e.key === 'Enter') { + this.handleFlush(); + e.target.blur(); + } + } + handleFlush () { + const isNumeric = typeof this.props.value === 'number'; + const validatesNumeric = isNumeric ? !isNaN(this.state.value) : true; + if (this.state.value !== null && validatesNumeric) { + this.props.onSubmit(isNumeric ? Number(this.state.value) : this.state.value); + } + this.setState({value: null}); + } + handleChange (e) { + this.setState({value: e.target.value}); + } + render () { + const bufferedValue = this.state.value === null ? this.props.value : this.state.value; + return ( + + ); + } + } + + BufferedInput.propTypes = { + onSubmit: PropTypes.func.isRequired, + value: PropTypes.oneOfType([PropTypes.string, PropTypes.number]) + }; + + return BufferedInput; +} diff --git a/src/components/forms/input.css b/src/components/forms/input.css new file mode 100644 index 00000000..6d5eefa5 --- /dev/null +++ b/src/components/forms/input.css @@ -0,0 +1,42 @@ +@import "../../css/units.css"; +@import "../../css/colors.css"; + +.input-form { + height: 2rem; + padding: 0 0.75rem; + + font-family: "Helvetica Neue", Helvetica, Arial, sans-serif; + font-size: 0.625rem; + font-weight: bold; + color: $text-primary; + + border-width: 1px; + border-style: solid; + border-color: $form-border; + border-radius: 2rem; + + outline: none; + cursor: text; + transition: 0.25s ease-out; /* @todo: standardize with var */ + box-shadow: none; + + /* + For truncating overflowing text gracefully + Min-width is for a bug: https://css-tricks.com/flexbox-truncated-text + @todo: move this out into a mixin or a helper component + */ + overflow: hidden; + text-overflow: ellipsis; + white-space: nowrap; + min-width: 0; +} + +.input-form:focus { + border-color: #4c97ff; + box-shadow: inset 0 0 0 -2px rgba(0, 0, 0, 0.1); +} + +.input-small { + width: 3rem; + text-align: center; +} diff --git a/src/components/forms/input.jsx b/src/components/forms/input.jsx new file mode 100644 index 00000000..6f9cbffa --- /dev/null +++ b/src/components/forms/input.jsx @@ -0,0 +1,27 @@ +import PropTypes from 'prop-types'; +import React from 'react'; +import classNames from 'classnames'; + +import styles from './input.css'; + +const Input = props => { + const {small, ...componentProps} = props; + return ( + + ); +}; + +Input.propTypes = { + small: PropTypes.bool +}; + +Input.defaultProps = { + small: false +}; + +export default Input; diff --git a/src/components/forms/label.css b/src/components/forms/label.css new file mode 100644 index 00000000..5eb280b8 --- /dev/null +++ b/src/components/forms/label.css @@ -0,0 +1,19 @@ +@import "../../css/units.css"; +@import "../../css/colors.css"; + +.input-group { + display: inline-flex; + flex-direction: row; + align-items: center; +} + +.input-label, .input-label-secondary { + font-size: 0.625rem; + margin-right: calc($space / 2); + user-select: none; + cursor: default; +} + +.input-label { + font-weight: bold; +} diff --git a/src/components/forms/label.jsx b/src/components/forms/label.jsx new file mode 100644 index 00000000..62f077a7 --- /dev/null +++ b/src/components/forms/label.jsx @@ -0,0 +1,25 @@ +import PropTypes from 'prop-types'; +import React from 'react'; + +import styles from './label.css'; + +const Label = props => ( + +); + +Label.propTypes = { + children: PropTypes.node, + secondary: PropTypes.bool, + text: PropTypes.string.isRequired +}; + +Label.defaultProps = { + secondary: false +}; + +export default Label; diff --git a/src/components/paint-editor.css b/src/components/paint-editor.css index 37991495..2048c9cc 100644 --- a/src/components/paint-editor.css +++ b/src/components/paint-editor.css @@ -1,35 +1,5 @@ -$ui-pane-border: #D9D9D9; -$ui-pane-gray: #F9F9F9; -$ui-background-blue: #e8edf1; - -$text-primary: #575e75; - -$motion-primary: #4C97FF; -$motion-tertiary: #3373CC; -$motion-transparent: hsla(215, 100%, 65%, 0.20); - -$red-primary: #FF661A; -$red-tertiary: #E64D00; - -$sound-primary: #CF63CF; -$sound-tertiary: #A63FA6; - -$control-primary: #FFAB19; - -$form-border: #E9EEF2; - -$space: 0.5rem; - -$sprites-per-row: 5; - -$menu-bar-height: 3rem; -$sprite-info-height: 6rem; -$stage-menu-height: 2.75rem; - -$library-header-height: 4.375rem; - -$form-radius: calc($space / 2); - +@import "../css/colors.css"; +@import "../css/units.css"; .editor-container { display: flex; @@ -43,6 +13,12 @@ $form-radius: calc($space / 2); align-items: center; } +.top-align-row { + padding-top:20px; + display: flex; + flex-direction: row; +} + .row + .row { margin-top: calc(2 * $space); } @@ -143,3 +119,18 @@ $border-radius: 0.25rem; .button:disabled > img { opacity: 0.25; } + +.canvas-container { + width: 503px; + height: 403px; + background-color: #e8edf1; + border: 1px solid #e8edf1; + border-radius: 2px; + position: relative; + overflow: visible; +} + +.mode-selector { + display: flex; + flex-direction: column; +} \ No newline at end of file diff --git a/src/components/paint-editor.jsx b/src/components/paint-editor.jsx index d95db51a..4b482f5d 100644 --- a/src/components/paint-editor.jsx +++ b/src/components/paint-editor.jsx @@ -6,8 +6,32 @@ import EraserMode from '../containers/eraser-mode.jsx'; import PropTypes from 'prop-types'; import LineMode from '../containers/line-mode.jsx'; +import {defineMessages, injectIntl, intlShape} from 'react-intl'; +import BufferedInputHOC from './forms/buffered-input-hoc.jsx'; +import Label from './forms/label.jsx'; +import Input from './forms/input.jsx'; + import styles from './paint-editor.css'; +const BufferedInput = BufferedInputHOC(Input); +const messages = defineMessages({ + costume: { + id: 'paint.paintEditor.costume', + description: 'Label for the name of a sound', + defaultMessage: 'Costume' + }, + fill: { + id: 'paint.paintEditor.fill', + description: 'Label for the color picker for the fill color', + defaultMessage: 'Fill' + }, + outline: { + id: 'paint.paintEditor.outline', + description: 'Label for the color picker for the outline color', + defaultMessage: 'Outline' + } +}); + class PaintEditorComponent extends React.Component { constructor (props) { super(props); @@ -26,10 +50,13 @@ class PaintEditorComponent extends React.Component {
{/* Name field */}
- {/* Todo use Label and BufferedInput from Gui */} - +
{/* Undo/Redo */} @@ -81,19 +108,23 @@ class PaintEditorComponent extends React.Component {
{/* To be fill */}
- +
{/* To be stroke */}
- +
@@ -101,7 +132,7 @@ class PaintEditorComponent extends React.Component {
-
+
{/* Modes */} {this.state.canvas ? (
@@ -136,10 +167,11 @@ class PaintEditorComponent extends React.Component { } PaintEditorComponent.propTypes = { + intl: intlShape, onUpdateSvg: PropTypes.func.isRequired, rotationCenterX: PropTypes.number, rotationCenterY: PropTypes.number, svg: PropTypes.string }; -export default PaintEditorComponent; +export default injectIntl(PaintEditorComponent); diff --git a/src/containers/paper-canvas.css b/src/containers/paper-canvas.css new file mode 100644 index 00000000..525853fd --- /dev/null +++ b/src/containers/paper-canvas.css @@ -0,0 +1,7 @@ +.paper-canvas { + width: 500px; + height: 400px; + margin: auto; + position: relative; + background-color: #fff; +} \ No newline at end of file diff --git a/src/containers/paper-canvas.jsx b/src/containers/paper-canvas.jsx index fdd6e44d..50e95b51 100644 --- a/src/containers/paper-canvas.jsx +++ b/src/containers/paper-canvas.jsx @@ -3,6 +3,8 @@ import PropTypes from 'prop-types'; import React from 'react'; import paper from 'paper'; +import styles from './paper-canvas.css'; + class PaperCanvas extends React.Component { constructor (props) { super(props); @@ -60,7 +62,10 @@ class PaperCanvas extends React.Component { render () { return ( ); } diff --git a/src/css/colors.css b/src/css/colors.css new file mode 100644 index 00000000..d524795c --- /dev/null +++ b/src/css/colors.css @@ -0,0 +1,19 @@ +$ui-pane-border: #D9D9D9; +$ui-pane-gray: #F9F9F9; +$ui-background-blue: #e8edf1; + +$text-primary: #575e75; + +$motion-primary: #4C97FF; +$motion-tertiary: #3373CC; +$motion-transparent: hsla(215, 100%, 65%, 0.20); + +$red-primary: #FF661A; +$red-tertiary: #E64D00; + +$sound-primary: #CF63CF; +$sound-tertiary: #A63FA6; + +$control-primary: #FFAB19; + +$form-border: #E9EEF2; diff --git a/src/css/units.css b/src/css/units.css new file mode 100644 index 00000000..3de6174a --- /dev/null +++ b/src/css/units.css @@ -0,0 +1,11 @@ +$space: 0.5rem; + +$sprites-per-row: 5; + +$menu-bar-height: 3rem; +$sprite-info-height: 6rem; +$stage-menu-height: 2.75rem; + +$library-header-height: 4.375rem; + +$form-radius: calc($space / 2); From f96c26ddbeda354b865ab805dfe1c0548f9a0685 Mon Sep 17 00:00:00 2001 From: DD Date: Wed, 6 Sep 2017 18:10:19 -0400 Subject: [PATCH 4/6] newlines --- src/components/paint-editor.css | 2 +- src/containers/paper-canvas.css | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/src/components/paint-editor.css b/src/components/paint-editor.css index 2048c9cc..c72d45ad 100644 --- a/src/components/paint-editor.css +++ b/src/components/paint-editor.css @@ -133,4 +133,4 @@ $border-radius: 0.25rem; .mode-selector { display: flex; flex-direction: column; -} \ No newline at end of file +} diff --git a/src/containers/paper-canvas.css b/src/containers/paper-canvas.css index 525853fd..d2f149fa 100644 --- a/src/containers/paper-canvas.css +++ b/src/containers/paper-canvas.css @@ -4,4 +4,4 @@ margin: auto; position: relative; background-color: #fff; -} \ No newline at end of file +} From a05fb5463bc5f957ba84970a3becf42498e2aa3a Mon Sep 17 00:00:00 2001 From: DD Date: Mon, 11 Sep 2017 11:05:02 -0400 Subject: [PATCH 5/6] remove sound editor specific CSS --- src/components/paint-editor.css | 48 --------------------------------- 1 file changed, 48 deletions(-) diff --git a/src/components/paint-editor.css b/src/components/paint-editor.css index c72d45ad..71651430 100644 --- a/src/components/paint-editor.css +++ b/src/components/paint-editor.css @@ -27,20 +27,6 @@ margin-left: calc(2 * $space); } -.waveform-container { - display: flex; - justify-content: space-around; - align-items: center; - width: 100%; - - position: relative; - - background: $ui-pane-gray; - border: 1px solid $ui-pane-border; - border-radius: 5px; - padding: 3px; -} - $border-radius: 0.25rem; .button { @@ -62,40 +48,6 @@ $border-radius: 0.25rem; min-width: 1.5rem; } -.trim-button { - display: flex; - align-items: center; - padding-right: 10px; /* To equalize with empty whitespace from image on left */ -} - -.trim-button > img { - width: 1.5rem; -} - -.trim-button-active { - filter: hue-rotate(155deg); /* @todo replace blue -> red with real submit icon */ -} - -.input-group-right { - flex-grow: 1; - display: flex; - flex-direction: row-reverse; -} - -.effect-button { - flex-basis: 150px; - color: $text-primary; -} - -.effect-button + .effect-button { - margin: 0; -} - -.effect-button img { - width: 60px; - height: 60px; -} - .button-group { margin: 0 1rem; } From 2121a4f438fcc550526eb98c707f4b12fbf05bd7 Mon Sep 17 00:00:00 2001 From: DD Date: Mon, 11 Sep 2017 11:13:43 -0400 Subject: [PATCH 6/6] Add comments on copied files --- src/components/forms/buffered-input-hoc.jsx | 4 ++++ src/components/forms/input.css | 4 ++++ src/components/forms/input.jsx | 4 ++++ src/components/forms/label.css | 4 ++++ src/components/forms/label.jsx | 4 ++++ src/css/colors.css | 4 ++++ src/css/units.css | 4 ++++ 7 files changed, 28 insertions(+) diff --git a/src/components/forms/buffered-input-hoc.jsx b/src/components/forms/buffered-input-hoc.jsx index 246dfc03..b3561136 100644 --- a/src/components/forms/buffered-input-hoc.jsx +++ b/src/components/forms/buffered-input-hoc.jsx @@ -1,3 +1,7 @@ +/* DO NOT EDIT +@todo This file is copied from GUI and should be pulled out into a shared library. +See https://github.com/LLK/scratch-paint/issues/13 */ + import bindAll from 'lodash.bindall'; import PropTypes from 'prop-types'; import React from 'react'; diff --git a/src/components/forms/input.css b/src/components/forms/input.css index 6d5eefa5..36855e44 100644 --- a/src/components/forms/input.css +++ b/src/components/forms/input.css @@ -1,3 +1,7 @@ +/* DO NOT EDIT +@todo This file is copied from GUI and should be pulled out into a shared library. +See https://github.com/LLK/scratch-paint/issues/13 */ + @import "../../css/units.css"; @import "../../css/colors.css"; diff --git a/src/components/forms/input.jsx b/src/components/forms/input.jsx index 6f9cbffa..1c6d087a 100644 --- a/src/components/forms/input.jsx +++ b/src/components/forms/input.jsx @@ -1,3 +1,7 @@ +/* DO NOT EDIT +@todo This file is copied from GUI and should be pulled out into a shared library. +See https://github.com/LLK/scratch-paint/issues/13 */ + import PropTypes from 'prop-types'; import React from 'react'; import classNames from 'classnames'; diff --git a/src/components/forms/label.css b/src/components/forms/label.css index 5eb280b8..7871cdd3 100644 --- a/src/components/forms/label.css +++ b/src/components/forms/label.css @@ -1,3 +1,7 @@ +/* DO NOT EDIT +@todo This file is copied from GUI and should be pulled out into a shared library. +See https://github.com/LLK/scratch-paint/issues/13 */ + @import "../../css/units.css"; @import "../../css/colors.css"; diff --git a/src/components/forms/label.jsx b/src/components/forms/label.jsx index 62f077a7..475ffb00 100644 --- a/src/components/forms/label.jsx +++ b/src/components/forms/label.jsx @@ -1,3 +1,7 @@ +/* DO NOT EDIT +@todo This file is copied from GUI and should be pulled out into a shared library. +See https://github.com/LLK/scratch-paint/issues/13 */ + import PropTypes from 'prop-types'; import React from 'react'; diff --git a/src/css/colors.css b/src/css/colors.css index d524795c..0fe15fca 100644 --- a/src/css/colors.css +++ b/src/css/colors.css @@ -1,3 +1,7 @@ +/* DO NOT EDIT +@todo This file is copied from GUI and should be pulled out into a shared library. +See https://github.com/LLK/scratch-paint/issues/13 */ + $ui-pane-border: #D9D9D9; $ui-pane-gray: #F9F9F9; $ui-background-blue: #e8edf1; diff --git a/src/css/units.css b/src/css/units.css index 3de6174a..acb466b1 100644 --- a/src/css/units.css +++ b/src/css/units.css @@ -1,3 +1,7 @@ +/* DO NOT EDIT +@todo This file is copied from GUI and should be pulled out into a shared library. +See https://github.com/LLK/scratch-paint/issues/13 */ + $space: 0.5rem; $sprites-per-row: 5;