From d5cc61fbda71321afcdee0d41bbeeb3568e96ec3 Mon Sep 17 00:00:00 2001 From: samreid Date: Thu, 27 Feb 2020 11:38:06 -0700 Subject: [PATCH] Add repo to migration list, see https://github.com/phetsims/chipper/issues/875 --- area-model-common-tests.html | 40 +- js/AreaModelCommonA11yStrings.js | 337 ++-- js/area-model-common-strings.js | 14 + js/area-model-common-test-config.js | 50 - js/area-model-common-tests.js | 14 +- js/areaModelCommon.js | 8 +- js/common/AreaModelCommonConstants.js | 242 ++- js/common/AreaModelCommonGlobals.js | 24 +- js/common/AreaModelCommonQueryParameters.js | 22 +- js/common/model/Area.js | 440 +++-- js/common/model/AreaCalculationChoice.js | 44 +- js/common/model/AreaDisplay.js | 222 ++- js/common/model/AreaModelCommonModel.js | 202 +- js/common/model/OrientationPair.js | 156 +- js/common/model/PartialProductsChoice.js | 36 +- js/common/model/Partition.js | 106 +- js/common/model/PartitionedArea.js | 80 +- js/common/model/Polynomial.js | 172 +- js/common/model/PolynomialTests.js | 90 +- js/common/model/Term.js | 272 ++- js/common/model/TermList.js | 188 +- js/common/model/TermListTests.js | 102 +- js/common/model/TermTests.js | 28 +- .../view/AreaCalculationRadioButtonGroup.js | 115 +- js/common/view/AreaDisplayNode.js | 532 +++--- js/common/view/AreaModelCommonAccordionBox.js | 91 +- js/common/view/AreaModelCommonColorProfile.js | 252 ++- .../view/AreaModelCommonRadioButtonGroup.js | 61 +- js/common/view/AreaScreenView.js | 588 +++--- js/common/view/CalculationNode.js | 372 ++-- js/common/view/PartialProductLabelNode.js | 313 ++-- .../view/PartialProductRadioButtonGroup.js | 236 ++- js/common/view/PoolableLayerNode.js | 141 +- js/common/view/RangeLabelNode.js | 217 ++- js/common/view/TotalAreaNode.js | 155 +- .../view/calculation/CalculationGroup.js | 106 +- js/common/view/calculation/CalculationLine.js | 560 +++--- .../view/calculation/CalculationLinesNode.js | 636 ++++--- .../view/calculation/DistributionLine.js | 103 +- js/common/view/calculation/ExpandedLine.js | 91 +- js/common/view/calculation/Minus.js | 94 +- js/common/view/calculation/MinusesLine.js | 43 +- js/common/view/calculation/MultipliedLine.js | 43 +- js/common/view/calculation/MultiplyX.js | 162 +- js/common/view/calculation/OrderedLine.js | 43 +- js/common/view/calculation/Parentheses.js | 194 +- js/common/view/calculation/PlaceholderBox.js | 96 +- js/common/view/calculation/Plus.js | 94 +- js/common/view/calculation/QuestionMark.js | 92 +- .../view/calculation/QuestionMarkLine.js | 41 +- js/common/view/calculation/SumLine.js | 41 +- js/common/view/calculation/TermText.js | 102 +- js/common/view/calculation/TotalsLine.js | 77 +- js/game/model/AreaChallenge.js | 882 +++++---- js/game/model/AreaChallengeDescription.js | 1622 ++++++++--------- js/game/model/AreaChallengeType.js | 32 +- js/game/model/AreaLevel.js | 256 ++- js/game/model/Entry.js | 109 +- js/game/model/EntryDisplayType.js | 36 +- js/game/model/EntryStatus.js | 36 +- js/game/model/EntryType.js | 86 +- js/game/model/GameAreaDisplay.js | 137 +- js/game/model/GameAreaModel.js | 318 ++-- js/game/model/GameState.js | 62 +- js/game/model/GenericGameAreaModel.js | 113 +- js/game/model/InputMethod.js | 74 +- js/game/model/VariablesGameAreaModel.js | 97 +- js/game/view/GameAreaDisplayNode.js | 351 ++-- js/game/view/GameAreaScreenView.js | 976 +++++----- js/game/view/GameAudio.js | 75 +- js/game/view/GameEditableLabelNode.js | 279 ++- js/game/view/PolynomialEditNode.js | 409 +++-- js/generic/model/GenericArea.js | 230 ++- js/generic/model/GenericAreaDisplay.js | 45 +- js/generic/model/GenericAreaModel.js | 122 +- js/generic/model/GenericLayout.js | 168 +- js/generic/model/GenericPartition.js | 49 +- js/generic/view/GenericAreaDisplayNode.js | 332 ++-- js/generic/view/GenericAreaScreenView.js | 168 +- js/generic/view/GenericFactorsNode.js | 234 ++- js/generic/view/GenericLayoutSelectionNode.js | 459 +++-- js/generic/view/GenericPartitionedAreaNode.js | 129 +- js/generic/view/PartitionSizeEditNode.js | 149 +- js/generic/view/TermAccumulator.js | 338 ++-- js/generic/view/TermEditNode.js | 257 ++- js/generic/view/TermKeypadPanel.js | 316 ++-- js/proportional/model/PartitionLineChoice.js | 36 +- js/proportional/model/ProportionalArea.js | 412 +++-- .../model/ProportionalAreaDisplay.js | 101 +- .../model/ProportionalAreaModel.js | 184 +- js/proportional/view/CalculationBox.js | 168 +- js/proportional/view/CountingAreaNode.js | 214 ++- .../view/PartitionRadioButtonGroup.js | 176 +- .../view/ProportionalAreaDisplayNode.js | 666 ++++--- .../view/ProportionalAreaGridLinesNode.js | 101 +- .../view/ProportionalAreaScreenView.js | 496 +++-- .../view/ProportionalDragHandle.js | 365 ++-- .../view/ProportionalFactorsNode.js | 286 ++- .../view/ProportionalPartitionLineNode.js | 440 +++-- js/proportional/view/SceneRadioButtonGroup.js | 95 +- js/proportional/view/TiledAreaNode.js | 468 +++-- js/screens/DecimalsScreen.js | 133 +- js/screens/ExploreScreen.js | 119 +- js/screens/GenericGameScreen.js | 79 +- js/screens/GenericScreen.js | 83 +- js/screens/MultiplyScreen.js | 133 +- js/screens/PartitionScreen.js | 135 +- js/screens/VariablesGameScreen.js | 79 +- js/screens/VariablesScreen.js | 83 +- mipmaps/explore-screen-icon_png.js | 38 + mipmaps/explore-screen-navbar_png.js | 38 + mipmaps/generic-game-screen-icon_png.js | 38 + mipmaps/generic-game-screen-navbar_png.js | 38 + mipmaps/generic-screen-icon_png.js | 38 + mipmaps/level-1-icon_png.js | 38 + mipmaps/level-2-icon_png.js | 38 + mipmaps/level-3-icon_png.js | 38 + mipmaps/level-4-icon_png.js | 38 + mipmaps/level-5-icon_png.js | 38 + mipmaps/level-6-icon_png.js | 38 + mipmaps/multiply-screen-icon_png.js | 38 + mipmaps/multiply-screen-navbar_png.js | 38 + mipmaps/partition-screen-icon_png.js | 38 + mipmaps/partition-screen-navbar_png.js | 38 + mipmaps/variables-game-screen-icon_png.js | 38 + mipmaps/variables-game-screen-navbar_png.js | 38 + mipmaps/variables-screen-icon_png.js | 38 + 127 files changed, 11770 insertions(+), 11492 deletions(-) create mode 100644 js/area-model-common-strings.js delete mode 100644 js/area-model-common-test-config.js create mode 100644 mipmaps/explore-screen-icon_png.js create mode 100644 mipmaps/explore-screen-navbar_png.js create mode 100644 mipmaps/generic-game-screen-icon_png.js create mode 100644 mipmaps/generic-game-screen-navbar_png.js create mode 100644 mipmaps/generic-screen-icon_png.js create mode 100644 mipmaps/level-1-icon_png.js create mode 100644 mipmaps/level-2-icon_png.js create mode 100644 mipmaps/level-3-icon_png.js create mode 100644 mipmaps/level-4-icon_png.js create mode 100644 mipmaps/level-5-icon_png.js create mode 100644 mipmaps/level-6-icon_png.js create mode 100644 mipmaps/multiply-screen-icon_png.js create mode 100644 mipmaps/multiply-screen-navbar_png.js create mode 100644 mipmaps/partition-screen-icon_png.js create mode 100644 mipmaps/partition-screen-navbar_png.js create mode 100644 mipmaps/variables-game-screen-icon_png.js create mode 100644 mipmaps/variables-game-screen-navbar_png.js create mode 100644 mipmaps/variables-screen-icon_png.js diff --git a/area-model-common-tests.html b/area-model-common-tests.html index fdfe4679..9bbfe6c6 100644 --- a/area-model-common-tests.html +++ b/area-model-common-tests.html @@ -18,6 +18,28 @@ \ No newline at end of file diff --git a/js/AreaModelCommonA11yStrings.js b/js/AreaModelCommonA11yStrings.js index 488ec91d..8be37ad8 100644 --- a/js/AreaModelCommonA11yStrings.js +++ b/js/AreaModelCommonA11yStrings.js @@ -7,180 +7,177 @@ * * @author Jonathan Olson */ -define( require => { - 'use strict'; - const areaModelCommon = require( 'AREA_MODEL_COMMON/areaModelCommon' ); +import areaModelCommon from './areaModelCommon.js'; - const AreaModelCommonA11yStrings = { - areaEqualsPattern: { - value: 'Area equals {{area}}' - }, - areaGrid: { - value: 'Area Grid' - }, - areaGridRectanglePattern: { - value: '{{height}} by {{width}} rectangle.' - }, - areaGridSize: { - value: 'Area Grid Size' - }, - base10AreaTiles: { - value: 'Base 10 area tiles' - }, - betweenCalculationLines: { - value: 'is equivalent to ' - }, - calculationBoxTitle: { - value: 'Calculation' - }, - calculationBoxDescription: { - value: 'Show or hide the calculation.' - }, - countingNumbersDescription: { - value: 'Count each unit of area' - }, - countingNumbersLabel: { - value: 'Counting Numbers' - }, - countingNumbersPattern: { - value: 'Area units count to {{count}}.' - }, - dragHandle: { - value: 'Drag handle' - }, - dragHandleDescriptionPattern: { - value: 'Drag the handle to build a rectangle up to {{height}} by {{width}}.' - }, - erase: { - value: 'Erase' - }, - eraseDescription: { - value: 'Clear the rectangle and build a new one.' - }, - factorsBox: { - value: 'Factors' - }, - factorsBoxDescription: { - value: 'Show or hide the factors.' - }, - factorsTimesPattern: { - value: '{{height}} times {{width}}' - }, - gridLinesLabel: { - value: 'Grid lines' - }, - hidePartialProducts: { - value: 'Hide partial products' - }, - horizontalDimensionCapitalized: { - value: 'Width' - }, - horizontalPartition: { - value: 'Horizontal partition' - }, - horizontalPartitionHandle: { - value: 'Horizontal partition handle' - }, - horizontalPartitionHandleDescription: { - value: 'Partition rectangle into two smaller areas by splitting the height.' - }, - horizontalPicker: { - value: 'Width' - }, - horizontalPickerDescription: { - value: 'Change the dimensions of the rectangle.' - }, - multiplyDescription: { - value: 'Play with multiplication by interacting with an area model.' - }, - onePartialProductFactorPattern: { - value: 'No partition set. Product factors are {{first}}.' - }, - onePartialProductPattern: { - value: 'No partition set. Product is {{first}}.' - }, - partitionDescription: { - value: 'Partition your area model and explore the distributive property.' - }, - partitionSelectionDescription: { - value: 'Switch between vertical and horizontal partition lines.' - }, - placeholder: { - value: 'empty' - }, - productBox: { - value: 'Product' - }, - productBoxDescription: { - value: 'Show or hide the product.' - }, - productTimesPattern: { - value: '{{left}} times {{right}}' - }, - quantityPattern: { - value: 'left parenthesis {{content}} right parenthesis' - }, - questionMark: { - value: 'undefined' - }, - sceneSelectionPattern: { - value: '{{height}} by {{width}}' - }, - showPartialProductFactors: { - value: 'Show partial product factors' - }, - showPartialProducts: { - value: 'Show partial products' - }, - sumMinus: { - value: 'minus' - }, - sumPlus: { - value: 'plus' - }, - threePartitionsSplitPattern: { - value: '{{partition}} of {{size}} split into {{size1}}, {{size2}} and {{size3}}.' - }, - twoPartialProductFactorsPattern: { - value: 'There are two partitions. Partial product factors are {{first}} and {{second}}.' - }, - twoPartialProductsPattern: { - value: 'There are two partitions. Partial products are {{first}} and {{second}}.' - }, - twoPartitionsSplitPattern: { - value: '{{partition}} of {{size}} split into {{size1}} and {{size2}}.' - }, - verticalDimensionCapitalized: { - value: 'Height' - }, - verticalPartition: { - value: 'Vertical partition' - }, - verticalPartitionHandle: { - value: 'Vertical partition handle' - }, - verticalPartitionHandleDescription: { - value: 'Partition rectangle into two smaller areas by splitting the width.' - }, - verticalPicker: { - value: 'Height' - }, - verticalPickerDescription: { - value: 'Change the dimensions of the rectangle.' - } - }; +const AreaModelCommonA11yStrings = { + areaEqualsPattern: { + value: 'Area equals {{area}}' + }, + areaGrid: { + value: 'Area Grid' + }, + areaGridRectanglePattern: { + value: '{{height}} by {{width}} rectangle.' + }, + areaGridSize: { + value: 'Area Grid Size' + }, + base10AreaTiles: { + value: 'Base 10 area tiles' + }, + betweenCalculationLines: { + value: 'is equivalent to ' + }, + calculationBoxTitle: { + value: 'Calculation' + }, + calculationBoxDescription: { + value: 'Show or hide the calculation.' + }, + countingNumbersDescription: { + value: 'Count each unit of area' + }, + countingNumbersLabel: { + value: 'Counting Numbers' + }, + countingNumbersPattern: { + value: 'Area units count to {{count}}.' + }, + dragHandle: { + value: 'Drag handle' + }, + dragHandleDescriptionPattern: { + value: 'Drag the handle to build a rectangle up to {{height}} by {{width}}.' + }, + erase: { + value: 'Erase' + }, + eraseDescription: { + value: 'Clear the rectangle and build a new one.' + }, + factorsBox: { + value: 'Factors' + }, + factorsBoxDescription: { + value: 'Show or hide the factors.' + }, + factorsTimesPattern: { + value: '{{height}} times {{width}}' + }, + gridLinesLabel: { + value: 'Grid lines' + }, + hidePartialProducts: { + value: 'Hide partial products' + }, + horizontalDimensionCapitalized: { + value: 'Width' + }, + horizontalPartition: { + value: 'Horizontal partition' + }, + horizontalPartitionHandle: { + value: 'Horizontal partition handle' + }, + horizontalPartitionHandleDescription: { + value: 'Partition rectangle into two smaller areas by splitting the height.' + }, + horizontalPicker: { + value: 'Width' + }, + horizontalPickerDescription: { + value: 'Change the dimensions of the rectangle.' + }, + multiplyDescription: { + value: 'Play with multiplication by interacting with an area model.' + }, + onePartialProductFactorPattern: { + value: 'No partition set. Product factors are {{first}}.' + }, + onePartialProductPattern: { + value: 'No partition set. Product is {{first}}.' + }, + partitionDescription: { + value: 'Partition your area model and explore the distributive property.' + }, + partitionSelectionDescription: { + value: 'Switch between vertical and horizontal partition lines.' + }, + placeholder: { + value: 'empty' + }, + productBox: { + value: 'Product' + }, + productBoxDescription: { + value: 'Show or hide the product.' + }, + productTimesPattern: { + value: '{{left}} times {{right}}' + }, + quantityPattern: { + value: 'left parenthesis {{content}} right parenthesis' + }, + questionMark: { + value: 'undefined' + }, + sceneSelectionPattern: { + value: '{{height}} by {{width}}' + }, + showPartialProductFactors: { + value: 'Show partial product factors' + }, + showPartialProducts: { + value: 'Show partial products' + }, + sumMinus: { + value: 'minus' + }, + sumPlus: { + value: 'plus' + }, + threePartitionsSplitPattern: { + value: '{{partition}} of {{size}} split into {{size1}}, {{size2}} and {{size3}}.' + }, + twoPartialProductFactorsPattern: { + value: 'There are two partitions. Partial product factors are {{first}} and {{second}}.' + }, + twoPartialProductsPattern: { + value: 'There are two partitions. Partial products are {{first}} and {{second}}.' + }, + twoPartitionsSplitPattern: { + value: '{{partition}} of {{size}} split into {{size1}} and {{size2}}.' + }, + verticalDimensionCapitalized: { + value: 'Height' + }, + verticalPartition: { + value: 'Vertical partition' + }, + verticalPartitionHandle: { + value: 'Vertical partition handle' + }, + verticalPartitionHandleDescription: { + value: 'Partition rectangle into two smaller areas by splitting the width.' + }, + verticalPicker: { + value: 'Height' + }, + verticalPickerDescription: { + value: 'Change the dimensions of the rectangle.' + } +}; - if ( phet.chipper.queryParameters.stringTest === 'xss' ) { - for ( const key in AreaModelCommonA11yStrings ) { - AreaModelCommonA11yStrings[ key ].value += ''; - } +if ( phet.chipper.queryParameters.stringTest === 'xss' ) { + for ( const key in AreaModelCommonA11yStrings ) { + AreaModelCommonA11yStrings[ key ].value += ''; } +} - // verify that object is immutable, without the runtime penalty in production code - if ( assert ) { Object.freeze( AreaModelCommonA11yStrings ); } +// verify that object is immutable, without the runtime penalty in production code +if ( assert ) { Object.freeze( AreaModelCommonA11yStrings ); } - areaModelCommon.register( 'AreaModelCommonA11yStrings', AreaModelCommonA11yStrings ); +areaModelCommon.register( 'AreaModelCommonA11yStrings', AreaModelCommonA11yStrings ); - return AreaModelCommonA11yStrings; -} ); +export default AreaModelCommonA11yStrings; \ No newline at end of file diff --git a/js/area-model-common-strings.js b/js/area-model-common-strings.js new file mode 100644 index 00000000..cc781257 --- /dev/null +++ b/js/area-model-common-strings.js @@ -0,0 +1,14 @@ +// Copyright 2020, University of Colorado Boulder + +/** + * Auto-generated from modulify, DO NOT manually modify. + */ + +import getStringModule from '../../chipper/js/getStringModule.js'; +import areaModelCommon from './areaModelCommon.js'; + +const areaModelCommonStrings = getStringModule( 'AREA_MODEL_COMMON' ); + +areaModelCommon.register( 'areaModelCommonStrings', areaModelCommonStrings ); + +export default areaModelCommonStrings; diff --git a/js/area-model-common-test-config.js b/js/area-model-common-test-config.js deleted file mode 100644 index 75fa3f19..00000000 --- a/js/area-model-common-test-config.js +++ /dev/null @@ -1,50 +0,0 @@ -// Copyright 2018-2019, University of Colorado Boulder - -/* - * IMPORTANT: This file was auto-generated by "grunt generate-config". Please do not modify this directly. Instead - * please modify area-model-common/package.json to control dependencies. - * - * RequireJS configuration file for the area-model-common sim. - * Paths are relative to the location of this file. - */ - -require.config( { - - deps: [ 'area-model-common-tests' ], - - paths: { - - // Third-party libs - text: '../../sherpa/lib/text-2.0.12', - - // PhET plugins - sound: '../../chipper/js/requirejs-plugins/sound', - image: '../../chipper/js/requirejs-plugins/image', - mipmap: '../../chipper/js/requirejs-plugins/mipmap', - string: '../../chipper/js/requirejs-plugins/string', - ifphetio: '../../chipper/js/requirejs-plugins/ifphetio', - - // PhET libs, uppercase names to identify them in require.js imports. - // IMPORTANT: DO NOT modify. This file is auto-generated. See documentation at the top. - AREA_MODEL_COMMON: '.', - AXON: '../../axon/js', - BRAND: '../../brand/' + phet.chipper.brand + '/js', - DOT: '../../dot/js', - JOIST: '../../joist/js', - KITE: '../../kite/js', - PHETCOMMON: '../../phetcommon/js', - PHET_CORE: '../../phet-core/js', - PHET_IO: '../../phet-io/js', - REPOSITORY: '..', - SCENERY: '../../scenery/js', - SCENERY_PHET: '../../scenery-phet/js', - SUN: '../../sun/js', - TAMBO: '../../tambo/js', - TANDEM: '../../tandem/js', - TWIXT: '../../twixt/js', - VEGAS: '../../vegas/js' - }, - - // optional cache bust to make browser refresh load all included scripts, can be disabled with ?cacheBust=false - urlArgs: phet.chipper.getCacheBustArgs() -} ); diff --git a/js/area-model-common-tests.js b/js/area-model-common-tests.js index 45e17548..9525ac4e 100644 --- a/js/area-model-common-tests.js +++ b/js/area-model-common-tests.js @@ -5,14 +5,10 @@ * * @author Jonathan Olson */ -define( require => { - 'use strict'; - // modules - require( 'AREA_MODEL_COMMON/common/model/TermTests' ); - require( 'AREA_MODEL_COMMON/common/model/TermListTests' ); - require( 'AREA_MODEL_COMMON/common/model/PolynomialTests' ); +import './common/model/PolynomialTests.js'; +import './common/model/TermListTests.js'; +import './common/model/TermTests.js'; - // Since our tests are loaded asynchronously, we must direct QUnit to begin the tests - QUnit.start(); -} ); \ No newline at end of file +// Since our tests are loaded asynchronously, we must direct QUnit to begin the tests +QUnit.start(); \ No newline at end of file diff --git a/js/areaModelCommon.js b/js/areaModelCommon.js index a65be964..2884104b 100644 --- a/js/areaModelCommon.js +++ b/js/areaModelCommon.js @@ -5,11 +5,7 @@ * * @author Jonathan Olson */ -define( require => { - 'use strict'; - // modules - const Namespace = require( 'PHET_CORE/Namespace' ); +import Namespace from '../../phet-core/js/Namespace.js'; - return new Namespace( 'areaModelCommon' ); -} ); \ No newline at end of file +export default new Namespace( 'areaModelCommon' ); \ No newline at end of file diff --git a/js/common/AreaModelCommonConstants.js b/js/common/AreaModelCommonConstants.js index ff76c8d0..77e962fd 100644 --- a/js/common/AreaModelCommonConstants.js +++ b/js/common/AreaModelCommonConstants.js @@ -5,126 +5,122 @@ * * @author Jonathan Olson */ -define( require => { - 'use strict'; - - // modules - const areaModelCommon = require( 'AREA_MODEL_COMMON/areaModelCommon' ); - const MathSymbolFont = require( 'SCENERY_PHET/MathSymbolFont' ); - const OrientationPair = require( 'AREA_MODEL_COMMON/common/model/OrientationPair' ); - const PhetFont = require( 'SCENERY_PHET/PhetFont' ); - const Vector2 = require( 'DOT/Vector2' ); - - // constants - const LARGE_PARTIAL_PRODUCT_FONT_SIZE = 19; - const NORMAL_EDIT_FONT_SIZE = 18; - - return areaModelCommon.register( 'AreaModelCommonConstants', { - - // {PhetFont} All fonts - FACTORS_TERM_FONT: new PhetFont( 36 ), // Terms/numbers in the factors box - FACTORS_PAREN_FONT: new PhetFont( 40 ), // Parentheses in the factors box - CALCULATION_X_FONT: new PhetFont( 16 ), - CALCULATION_PAREN_FONT: new PhetFont( 16 ), - CALCULATION_DOT_FONT: new PhetFont( { size: 16, weight: 'bold' } ), - CALCULATION_TERM_FONT: new PhetFont( 16 ), - TITLE_FONT: new PhetFont( 18 ), - TOTAL_AREA_LABEL_FONT: new PhetFont( 30 ), - TOTAL_AREA_VALUE_FONT: new PhetFont( { size: 30, weight: 'bold' } ), - SYMBOL_FONT: new PhetFont( 20 ), - PARTIAL_PRODUCT_FONT: new PhetFont( LARGE_PARTIAL_PRODUCT_FONT_SIZE ), - PARTIAL_FACTOR_FONT: new PhetFont( 14 ), - TERM_EDIT_READOUT_FONT: new PhetFont( NORMAL_EDIT_FONT_SIZE ), - POLYNOMIAL_EDIT_READOUT_FONT: new PhetFont( NORMAL_EDIT_FONT_SIZE ), - PROPORTIONAL_PARTITION_READOUT_FONT: new PhetFont( { size: NORMAL_EDIT_FONT_SIZE, weight: 'bold' } ), - TOTAL_SIZE_READOUT_FONT: new PhetFont( { size: 22, weight: 'bold' } ), - KEYPAD_FONT: new PhetFont( 20 ), - KEYPAD_READOUT_FONT: new PhetFont( 20 ), - LAYOUT_FONT: new PhetFont( 16 ), - BUTTON_FONT: new PhetFont( 20 ), - SCORE_INCREASE_FONT: new PhetFont( { size: 18, weight: 'bold' } ), - COUNTING_ICON_FONT: new PhetFont( 22 ), - COUNTING_FONT: new PhetFont( 20 ), - REWARD_NODE_FONT: new PhetFont( { size: 35, weight: 'bold' } ), - GAME_MAIN_LABEL_FONT: new PhetFont( { size: NORMAL_EDIT_FONT_SIZE, weight: 'bold' } ), - GAME_MAIN_EDIT_FONT: new PhetFont( NORMAL_EDIT_FONT_SIZE ), - GAME_PARTIAL_PRODUCT_LABEL_FONT: new PhetFont( { size: LARGE_PARTIAL_PRODUCT_FONT_SIZE, weight: 'bold' } ), - GAME_PARTIAL_PRODUCT_EDIT_FONT: new PhetFont( LARGE_PARTIAL_PRODUCT_FONT_SIZE ), - GAME_TOTAL_FONT: new PhetFont( { size: 30, weight: 'bold' } ), - GAME_POLYNOMIAL_EDIT_FONT: new PhetFont( { size: 22, weight: 'bold' } ), - GAME_STATUS_BAR_BOLD_FONT: new PhetFont( { size: 18, weight: 'bold' } ), - GAME_STATUS_BAR_NON_BOLD_FONT: new PhetFont( { size: 18 } ), - GAME_STATUS_BAR_PROMPT_FONT: new PhetFont( { size: 30, weight: 'bold' } ), - - // {string} - The string to be provided to RichText for a mathematical-looking x - X_VARIABLE_RICH_STRING: 'x', - - // {number} Two different area sizes (they are square), one needed for the intro sim - AREA_SIZE: 350, - LARGE_AREA_SIZE: 450, - - // {Vector2} We need to place the areas in different locations depending on the screen - MAIN_AREA_OFFSET: new Vector2( 180, 80 ), - LARGE_AREA_OFFSET: new Vector2( 80, 80 ), - GAME_AREA_OFFSET: new Vector2( 180, 200 ), - - // {number} - Panel options - LAYOUT_SPACING: 10, - PANEL_CORNER_RADIUS: 5, - PANEL_INTERIOR_MAX: 230, // Maximum width of the content inside the panels - - // {number} - Partition drag handle options - PARTITION_HANDLE_OFFSET: 15, - PARTITION_HANDLE_RADIUS: 10, - - // {number} - Relative positions (from 0 to 1) of where the generic partition lines should be - GENERIC_SINGLE_OFFSET: 0.62, // if there is one line - GENERIC_FIRST_OFFSET: 0.45, - GENERIC_SECOND_OFFSET: 0.78, - - // {number} - Like the generic view, but for the icon - GENERIC_ICON_SINGLE_OFFSET: 0.68, - GENERIC_ICON_FIRST_OFFSET: 0.55, - GENERIC_ICON_SECOND_OFFSET: 0.80, - - // {Vector2} - Offset vector from the upper-left of the area to the x,y location where the dimension line labels - // would intersect. - PROPORTIONAL_RANGE_OFFSET: new Vector2( -35, -28 ), - GENERIC_RANGE_OFFSET: new Vector2( -60, -40 ), - - // {number} - Space between the area and the keypad - KEYPAD_LEFT_PADDING: 25, - - // {number} - Number of challenges per level - NUM_CHALLENGES: 6, - - // {number} - The perfect score for a level - PERFECT_SCORE: 12, - - // {number} - Padding in-between content and surrounding parentheses in the calculation area - CALCULATION_PAREN_PADDING: 0, - - // {number} - Padding in-between an end parenthesis and start parenthese, e.g. between )( - CALCULATION_PAREN_PAREN_PADDING: 0, - - // {number} - Padding around an x (used for multiplication) - CALCULATION_X_PADDING: 3, - - // {number} - Padding around a dot (used for multiplication) - CALCULATION_DOT_PADDING: 3, - - // {number} - Padding around most (binary) operations in the calculation - CALCULATION_OP_PADDING: 5, - - // {number} - Padding between a term and an adjacent parenthesis, e.g. "x(" or ")x" - CALCULATION_TERM_PAREN_PADDING: 1, - - // {OrientationPair.} - The opposite-orientation offset to use for term edit nodes, e.g. - // node[ orientation.opposite.coordinate ] = PARTITION_OFFSET.get( orientation ) - PARTITION_OFFSET: new OrientationPair( -20, -30 ), - - // {string} - The character we use as a generic decimal character to get an approximate width for numeric - // representations. - MEASURING_CHARACTER: '9' - } ); -} ); + +import Vector2 from '../../../dot/js/Vector2.js'; +import MathSymbolFont from '../../../scenery-phet/js/MathSymbolFont.js'; +import PhetFont from '../../../scenery-phet/js/PhetFont.js'; +import areaModelCommon from '../areaModelCommon.js'; +import OrientationPair from './model/OrientationPair.js'; + +// constants +const LARGE_PARTIAL_PRODUCT_FONT_SIZE = 19; +const NORMAL_EDIT_FONT_SIZE = 18; + +export default areaModelCommon.register( 'AreaModelCommonConstants', { + + // {PhetFont} All fonts + FACTORS_TERM_FONT: new PhetFont( 36 ), // Terms/numbers in the factors box + FACTORS_PAREN_FONT: new PhetFont( 40 ), // Parentheses in the factors box + CALCULATION_X_FONT: new PhetFont( 16 ), + CALCULATION_PAREN_FONT: new PhetFont( 16 ), + CALCULATION_DOT_FONT: new PhetFont( { size: 16, weight: 'bold' } ), + CALCULATION_TERM_FONT: new PhetFont( 16 ), + TITLE_FONT: new PhetFont( 18 ), + TOTAL_AREA_LABEL_FONT: new PhetFont( 30 ), + TOTAL_AREA_VALUE_FONT: new PhetFont( { size: 30, weight: 'bold' } ), + SYMBOL_FONT: new PhetFont( 20 ), + PARTIAL_PRODUCT_FONT: new PhetFont( LARGE_PARTIAL_PRODUCT_FONT_SIZE ), + PARTIAL_FACTOR_FONT: new PhetFont( 14 ), + TERM_EDIT_READOUT_FONT: new PhetFont( NORMAL_EDIT_FONT_SIZE ), + POLYNOMIAL_EDIT_READOUT_FONT: new PhetFont( NORMAL_EDIT_FONT_SIZE ), + PROPORTIONAL_PARTITION_READOUT_FONT: new PhetFont( { size: NORMAL_EDIT_FONT_SIZE, weight: 'bold' } ), + TOTAL_SIZE_READOUT_FONT: new PhetFont( { size: 22, weight: 'bold' } ), + KEYPAD_FONT: new PhetFont( 20 ), + KEYPAD_READOUT_FONT: new PhetFont( 20 ), + LAYOUT_FONT: new PhetFont( 16 ), + BUTTON_FONT: new PhetFont( 20 ), + SCORE_INCREASE_FONT: new PhetFont( { size: 18, weight: 'bold' } ), + COUNTING_ICON_FONT: new PhetFont( 22 ), + COUNTING_FONT: new PhetFont( 20 ), + REWARD_NODE_FONT: new PhetFont( { size: 35, weight: 'bold' } ), + GAME_MAIN_LABEL_FONT: new PhetFont( { size: NORMAL_EDIT_FONT_SIZE, weight: 'bold' } ), + GAME_MAIN_EDIT_FONT: new PhetFont( NORMAL_EDIT_FONT_SIZE ), + GAME_PARTIAL_PRODUCT_LABEL_FONT: new PhetFont( { size: LARGE_PARTIAL_PRODUCT_FONT_SIZE, weight: 'bold' } ), + GAME_PARTIAL_PRODUCT_EDIT_FONT: new PhetFont( LARGE_PARTIAL_PRODUCT_FONT_SIZE ), + GAME_TOTAL_FONT: new PhetFont( { size: 30, weight: 'bold' } ), + GAME_POLYNOMIAL_EDIT_FONT: new PhetFont( { size: 22, weight: 'bold' } ), + GAME_STATUS_BAR_BOLD_FONT: new PhetFont( { size: 18, weight: 'bold' } ), + GAME_STATUS_BAR_NON_BOLD_FONT: new PhetFont( { size: 18 } ), + GAME_STATUS_BAR_PROMPT_FONT: new PhetFont( { size: 30, weight: 'bold' } ), + + // {string} - The string to be provided to RichText for a mathematical-looking x + X_VARIABLE_RICH_STRING: 'x', + + // {number} Two different area sizes (they are square), one needed for the intro sim + AREA_SIZE: 350, + LARGE_AREA_SIZE: 450, + + // {Vector2} We need to place the areas in different locations depending on the screen + MAIN_AREA_OFFSET: new Vector2( 180, 80 ), + LARGE_AREA_OFFSET: new Vector2( 80, 80 ), + GAME_AREA_OFFSET: new Vector2( 180, 200 ), + + // {number} - Panel options + LAYOUT_SPACING: 10, + PANEL_CORNER_RADIUS: 5, + PANEL_INTERIOR_MAX: 230, // Maximum width of the content inside the panels + + // {number} - Partition drag handle options + PARTITION_HANDLE_OFFSET: 15, + PARTITION_HANDLE_RADIUS: 10, + + // {number} - Relative positions (from 0 to 1) of where the generic partition lines should be + GENERIC_SINGLE_OFFSET: 0.62, // if there is one line + GENERIC_FIRST_OFFSET: 0.45, + GENERIC_SECOND_OFFSET: 0.78, + + // {number} - Like the generic view, but for the icon + GENERIC_ICON_SINGLE_OFFSET: 0.68, + GENERIC_ICON_FIRST_OFFSET: 0.55, + GENERIC_ICON_SECOND_OFFSET: 0.80, + + // {Vector2} - Offset vector from the upper-left of the area to the x,y location where the dimension line labels + // would intersect. + PROPORTIONAL_RANGE_OFFSET: new Vector2( -35, -28 ), + GENERIC_RANGE_OFFSET: new Vector2( -60, -40 ), + + // {number} - Space between the area and the keypad + KEYPAD_LEFT_PADDING: 25, + + // {number} - Number of challenges per level + NUM_CHALLENGES: 6, + + // {number} - The perfect score for a level + PERFECT_SCORE: 12, + + // {number} - Padding in-between content and surrounding parentheses in the calculation area + CALCULATION_PAREN_PADDING: 0, + + // {number} - Padding in-between an end parenthesis and start parenthese, e.g. between )( + CALCULATION_PAREN_PAREN_PADDING: 0, + + // {number} - Padding around an x (used for multiplication) + CALCULATION_X_PADDING: 3, + + // {number} - Padding around a dot (used for multiplication) + CALCULATION_DOT_PADDING: 3, + + // {number} - Padding around most (binary) operations in the calculation + CALCULATION_OP_PADDING: 5, + + // {number} - Padding between a term and an adjacent parenthesis, e.g. "x(" or ")x" + CALCULATION_TERM_PAREN_PADDING: 1, + + // {OrientationPair.} - The opposite-orientation offset to use for term edit nodes, e.g. + // node[ orientation.opposite.coordinate ] = PARTITION_OFFSET.get( orientation ) + PARTITION_OFFSET: new OrientationPair( -20, -30 ), + + // {string} - The character we use as a generic decimal character to get an approximate width for numeric + // representations. + MEASURING_CHARACTER: '9' +} ); \ No newline at end of file diff --git a/js/common/AreaModelCommonGlobals.js b/js/common/AreaModelCommonGlobals.js index 8a4c23bb..2ed76fb9 100644 --- a/js/common/AreaModelCommonGlobals.js +++ b/js/common/AreaModelCommonGlobals.js @@ -5,21 +5,17 @@ * * @author Jonathan Olson */ -define( require => { - 'use strict'; - // modules - const AlignGroup = require( 'SCENERY/nodes/AlignGroup' ); - const areaModelCommon = require( 'AREA_MODEL_COMMON/areaModelCommon' ); +import AlignGroup from '../../../scenery/js/nodes/AlignGroup.js'; +import areaModelCommon from '../areaModelCommon.js'; - return areaModelCommon.register( 'AreaModelCommonGlobals', { +export default areaModelCommon.register( 'AreaModelCommonGlobals', { - // @public {AlignGroup} - Used to properly horizontally align all of the panels/accordions/etc. across screens. - panelAlignGroup: new AlignGroup( { - matchVertical: false - } ), + // @public {AlignGroup} - Used to properly horizontally align all of the panels/accordions/etc. across screens. + panelAlignGroup: new AlignGroup( { + matchVertical: false + } ), - // @public {AlignGroup} - Used for the radio group selection icons (so they are consistent across screens) - selectionButtonAlignGroup: new AlignGroup() - } ); -} ); + // @public {AlignGroup} - Used for the radio group selection icons (so they are consistent across screens) + selectionButtonAlignGroup: new AlignGroup() +} ); \ No newline at end of file diff --git a/js/common/AreaModelCommonQueryParameters.js b/js/common/AreaModelCommonQueryParameters.js index 8ddce7b4..2004d99e 100644 --- a/js/common/AreaModelCommonQueryParameters.js +++ b/js/common/AreaModelCommonQueryParameters.js @@ -5,22 +5,18 @@ * * @author Jonathan Olson */ -define( require => { - 'use strict'; - // modules - const areaModelCommon = require( 'AREA_MODEL_COMMON/areaModelCommon' ); +import areaModelCommon from '../areaModelCommon.js'; - const AreaModelCommonQueryParameters = QueryStringMachine.getAll( { +const AreaModelCommonQueryParameters = QueryStringMachine.getAll( { - // When provided, generic screens will have terms set to make the calculation area as large as possible, for debugging - maximumCalculationSize: { type: 'flag' }, + // When provided, generic screens will have terms set to make the calculation area as large as possible, for debugging + maximumCalculationSize: { type: 'flag' }, - // If set, uses constructed English strings instead of MathML for the accessible parallel DOM. - rawMath: { type: 'flag' } - } ); + // If set, uses constructed English strings instead of MathML for the accessible parallel DOM. + rawMath: { type: 'flag' } +} ); - areaModelCommon.register( 'AreaModelCommonQueryParameters', AreaModelCommonQueryParameters ); +areaModelCommon.register( 'AreaModelCommonQueryParameters', AreaModelCommonQueryParameters ); - return AreaModelCommonQueryParameters; -} ); +export default AreaModelCommonQueryParameters; \ No newline at end of file diff --git a/js/common/model/Area.js b/js/common/model/Area.js index fb9e03c1..21e57db3 100644 --- a/js/common/model/Area.js +++ b/js/common/model/Area.js @@ -5,246 +5,242 @@ * * @author Jonathan Olson */ -define( require => { - 'use strict'; - - // modules - const areaModelCommon = require( 'AREA_MODEL_COMMON/areaModelCommon' ); - const DerivedProperty = require( 'AXON/DerivedProperty' ); - const inherit = require( 'PHET_CORE/inherit' ); - const NumberProperty = require( 'AXON/NumberProperty' ); - const Orientation = require( 'PHET_CORE/Orientation' ); - const OrientationPair = require( 'AREA_MODEL_COMMON/common/model/OrientationPair' ); - const PartitionedArea = require( 'AREA_MODEL_COMMON/common/model/PartitionedArea' ); - const Polynomial = require( 'AREA_MODEL_COMMON/common/model/Polynomial' ); - const Property = require( 'AXON/Property' ); - const TermList = require( 'AREA_MODEL_COMMON/common/model/TermList' ); - const validate = require( 'AXON/validate' ); +import DerivedProperty from '../../../../axon/js/DerivedProperty.js'; +import NumberProperty from '../../../../axon/js/NumberProperty.js'; +import Property from '../../../../axon/js/Property.js'; +import validate from '../../../../axon/js/validate.js'; +import inherit from '../../../../phet-core/js/inherit.js'; +import Orientation from '../../../../phet-core/js/Orientation.js'; +import areaModelCommon from '../../areaModelCommon.js'; +import OrientationPair from './OrientationPair.js'; +import PartitionedArea from './PartitionedArea.js'; +import Polynomial from './Polynomial.js'; +import TermList from './TermList.js'; + +/** + * @constructor + * @extends {Object} + * + * @param {OrientationPair.>} partitions - The passed-in partitions become "owned" by this Area + * object (and they should not be shared by multiple areas + * ever). Usually created in subtypes anyways. + * @param {OrientationPair.>} colorProperties + * @param {number} coordinateRangeMax - The maximum value that partition coordinate ranges may take. A (proportional) + * - partition can be held at the max, but if released at the max it will jump back + * - to 0. Only one value is needed because the area is always square. + * @param {boolean} allowExponents - Whether exponents (powers of x) are allowed for this area + */ +function Area( partitions, colorProperties, coordinateRangeMax, allowExponents ) { + const self = this; + + // @public {OrientationPair.>} - Partitions for each orientation + this.partitions = partitions; + + // @public {Array.} - All partitions, regardless of orientation + this.allPartitions = partitions.horizontal.concat( partitions.vertical ); + + // @public {OrientationPair.>} - Colors for each orientation + this.colorProperties = colorProperties; + + // @public {number} - The maximum value that partition coordinate ranges may take. + this.coordinateRangeMax = coordinateRangeMax; + + // @public {boolean} - Whether exponents (powers of x) are allowed for this area + this.allowExponents = allowExponents; + + // @public {Property.} - The index of the highlighted calculation line (if using the LINE_BY_LINE choice). + this.calculationIndexProperty = new NumberProperty( 0 ); + + // @public {Array.} - An array of 2-dimensional sections of area defined by a horizontal and + // vertical pair of partitions. + this.partitionedAreas = _.flatten( partitions.horizontal.map( function( horizontalPartition ) { + return partitions.vertical.map( function( verticalPartition ) { + return self.createPartitionedArea( new OrientationPair( horizontalPartition, verticalPartition ) ); + } ); + } ) ); + + // @public {OrientationPair.>} - Null if there is no defined total. Otherwise it's the + // sum of the sizes of all (defined) partitions of the given orientation. + this.totalProperties = OrientationPair.create( this.createMappedTermsArrayProperty.bind( this, function( terms ) { + return new Polynomial( terms ); + } ) ); + + // @public {OrientationPair.>} - Null if there is no defined partition. Otherwise it's a + // list of the sizes of all (defined) partitions of the given orientation. This does NOT combine terms with the + // same exponent, unlike this.totalProperties. + this.termListProperties = OrientationPair.create( this.createMappedTermsArrayProperty.bind( this, function( terms ) { + return new TermList( terms ); + } ) ); + + // @public {Property.} - Null if there is no defined total, otherwise the total area (width of the + // "area" times its height). + this.totalAreaProperty = new DerivedProperty( + this.totalProperties.values, + function( horizontalTotal, verticalTotal ) { + return horizontalTotal && verticalTotal && horizontalTotal.times( verticalTotal ); + }, { + useDeepEquality: true + } ); + + // @public {OrientationPair.>} - Displayed term list for the product. Null if there is no + // defined total. + this.displayProperties = allowExponents ? this.termListProperties : this.totalProperties; + + // @public {OrientationPair.>>} - For each orientation, will contain a property with an + // unsorted list of unique boundary locations (the minimum or maximum coordinates of partitions). So if there are + // two partitions for an orientation, one from 1 to 5, and the other from 5 to 7, the value of the property will be + // [ 1, 5, 7 ] + this.partitionBoundariesProperties = OrientationPair.create( this.createPartitionBoundariesProperty.bind( this ) ); +} + +areaModelCommon.register( 'Area', Area ); + +export default inherit( Object, Area, { /** - * @constructor - * @extends {Object} + * Creates a partitioned area given two partitions. + * @protected * - * @param {OrientationPair.>} partitions - The passed-in partitions become "owned" by this Area - * object (and they should not be shared by multiple areas - * ever). Usually created in subtypes anyways. - * @param {OrientationPair.>} colorProperties - * @param {number} coordinateRangeMax - The maximum value that partition coordinate ranges may take. A (proportional) - * - partition can be held at the max, but if released at the max it will jump back - * - to 0. Only one value is needed because the area is always square. - * @param {boolean} allowExponents - Whether exponents (powers of x) are allowed for this area + * @param {OrientationPair.} partitions + * @returns {PartitionedArea} */ - function Area( partitions, colorProperties, coordinateRangeMax, allowExponents ) { - const self = this; + createPartitionedArea: function( partitions ) { + const partitionedArea = new PartitionedArea( partitions ); + + // By default, have the area linked to the partitions. This won't work for the game. + // NOTE: Since we "own" the partitions memory-wise, we don't need to unlink here since they should all be GC'ed + // at the same time. + Property.multilink( + [ partitions.horizontal.sizeProperty, partitions.vertical.sizeProperty ], + function( horizontalSize, verticalSize ) { + if ( horizontalSize === null || verticalSize === null ) { + partitionedArea.areaProperty.value = null; + } + else { + partitionedArea.areaProperty.value = horizontalSize.times( verticalSize ); + } + } ); - // @public {OrientationPair.>} - Partitions for each orientation - this.partitions = partitions; + return partitionedArea; + }, - // @public {Array.} - All partitions, regardless of orientation - this.allPartitions = partitions.horizontal.concat( partitions.vertical ); + /** + * Resets the area to its initial values. + * @public + */ + reset: function() { + // NOTE: Not resetting partitions here. The subtype takes care of that action (which may be indirect) + this.calculationIndexProperty.reset(); + }, - // @public {OrientationPair.>} - Colors for each orientation - this.colorProperties = colorProperties; + /** + * Erase the area to a 1x1, see https://github.com/phetsims/area-model-common/issues/77 + * @public + */ + erase: function() { + // Overridden in subtypes + }, - // @public {number} - The maximum value that partition coordinate ranges may take. - this.coordinateRangeMax = coordinateRangeMax; + /** + * Returns all defined partitions for a given orientation. + * @public + * + * @param {Orientation} orientation + * @returns {Array.} + */ + getDefinedPartitions: function( orientation ) { + validate( orientation, { validValues: Orientation.VALUES } ); - // @public {boolean} - Whether exponents (powers of x) are allowed for this area - this.allowExponents = allowExponents; + return this.partitions.get( orientation ).filter( function( partition ) { + return partition.isDefined(); + } ); + }, - // @public {Property.} - The index of the highlighted calculation line (if using the LINE_BY_LINE choice). - this.calculationIndexProperty = new NumberProperty( 0 ); + /** + * Returns an array of Terms containing all of the defined partition sizes for the given orientation. + * @public + * + * @param {Orientation} orientation + * @returns {Array.} + */ + getTerms: function( orientation ) { + validate( orientation, { validValues: Orientation.VALUES } ); - // @public {Array.} - An array of 2-dimensional sections of area defined by a horizontal and - // vertical pair of partitions. - this.partitionedAreas = _.flatten( partitions.horizontal.map( function( horizontalPartition ) { - return partitions.vertical.map( function( verticalPartition ) { - return self.createPartitionedArea( new OrientationPair( horizontalPartition, verticalPartition ) ); - } ); - } ) ); + return this.getDefinedPartitions( orientation ).map( function( partition ) { + return partition.sizeProperty.value; + } ); + }, - // @public {OrientationPair.>} - Null if there is no defined total. Otherwise it's the - // sum of the sizes of all (defined) partitions of the given orientation. - this.totalProperties = OrientationPair.create( this.createMappedTermsArrayProperty.bind( this, function( terms ) { - return new Polynomial( terms ); - } ) ); + /** + * Returns a TermList containing all of the defined partition sizes for the given orientation. + * @public + * + * @param {Orientation} orientation + * @returns {TermList} + */ + getTermList: function( orientation ) { + validate( orientation, { validValues: Orientation.VALUES } ); - // @public {OrientationPair.>} - Null if there is no defined partition. Otherwise it's a - // list of the sizes of all (defined) partitions of the given orientation. This does NOT combine terms with the - // same exponent, unlike this.totalProperties. - this.termListProperties = OrientationPair.create( this.createMappedTermsArrayProperty.bind( this, function( terms ) { - return new TermList( terms ); + return new TermList( this.getTerms( orientation ) ); + }, + + /** + * Creates a DerivedProperty with `map( totalSizeTerms )` for a particular orientation, where totalSizeTerms + * is an array of the total terms for that orientation. + * @private + * + * @param {function} map - function( {Array.} ): * + * @param {Orientation} orientation + * @returns {Property.<*|null>} + */ + createMappedTermsArrayProperty: function( map, orientation ) { + const self = this; + + const properties = _.flatten( this.partitions.get( orientation ).map( function( partition ) { + return [ partition.sizeProperty, partition.visibleProperty ]; } ) ); - // @public {Property.} - Null if there is no defined total, otherwise the total area (width of the - // "area" times its height). - this.totalAreaProperty = new DerivedProperty( - this.totalProperties.values, - function( horizontalTotal, verticalTotal ) { - return horizontalTotal && verticalTotal && horizontalTotal.times( verticalTotal ); - }, { - useDeepEquality: true - } ); + return new DerivedProperty( properties, function() { + const terms = self.getTerms( orientation ); + if ( terms.length ) { + return map( terms ); + } + else { + return null; + } + }, { + useDeepEquality: true + } ); + }, - // @public {OrientationPair.>} - Displayed term list for the product. Null if there is no - // defined total. - this.displayProperties = allowExponents ? this.termListProperties : this.totalProperties; + /** + * Returns a property that will contain an array of all unique partition boundaries (the minimum or maximum + * coordinate locations of a partition). + * @private + * + * @param {Orientation} orientation + * @returns {Property.>} + */ + createPartitionBoundariesProperty: function( orientation ) { + const partitions = this.partitions.get( orientation ); - // @public {OrientationPair.>>} - For each orientation, will contain a property with an - // unsorted list of unique boundary locations (the minimum or maximum coordinates of partitions). So if there are - // two partitions for an orientation, one from 1 to 5, and the other from 5 to 7, the value of the property will be - // [ 1, 5, 7 ] - this.partitionBoundariesProperties = OrientationPair.create( this.createPartitionBoundariesProperty.bind( this ) ); - } + // Property dependencies + const partitionProperties = _.flatten( partitions.map( function( partition ) { + return [ partition.coordinateRangeProperty, partition.visibleProperty ]; + } ) ); - areaModelCommon.register( 'Area', Area ); - - return inherit( Object, Area, { - /** - * Creates a partitioned area given two partitions. - * @protected - * - * @param {OrientationPair.} partitions - * @returns {PartitionedArea} - */ - createPartitionedArea: function( partitions ) { - const partitionedArea = new PartitionedArea( partitions ); - - // By default, have the area linked to the partitions. This won't work for the game. - // NOTE: Since we "own" the partitions memory-wise, we don't need to unlink here since they should all be GC'ed - // at the same time. - Property.multilink( - [ partitions.horizontal.sizeProperty, partitions.vertical.sizeProperty ], - function( horizontalSize, verticalSize ) { - if ( horizontalSize === null || verticalSize === null ) { - partitionedArea.areaProperty.value = null; - } - else { - partitionedArea.areaProperty.value = horizontalSize.times( verticalSize ); - } - } ); - - return partitionedArea; - }, - - /** - * Resets the area to its initial values. - * @public - */ - reset: function() { - // NOTE: Not resetting partitions here. The subtype takes care of that action (which may be indirect) - this.calculationIndexProperty.reset(); - }, - - /** - * Erase the area to a 1x1, see https://github.com/phetsims/area-model-common/issues/77 - * @public - */ - erase: function() { - // Overridden in subtypes - }, - - /** - * Returns all defined partitions for a given orientation. - * @public - * - * @param {Orientation} orientation - * @returns {Array.} - */ - getDefinedPartitions: function( orientation ) { - validate( orientation, { validValues: Orientation.VALUES } ); - - return this.partitions.get( orientation ).filter( function( partition ) { - return partition.isDefined(); - } ); - }, - - /** - * Returns an array of Terms containing all of the defined partition sizes for the given orientation. - * @public - * - * @param {Orientation} orientation - * @returns {Array.} - */ - getTerms: function( orientation ) { - validate( orientation, { validValues: Orientation.VALUES } ); - - return this.getDefinedPartitions( orientation ).map( function( partition ) { - return partition.sizeProperty.value; - } ); - }, - - /** - * Returns a TermList containing all of the defined partition sizes for the given orientation. - * @public - * - * @param {Orientation} orientation - * @returns {TermList} - */ - getTermList: function( orientation ) { - validate( orientation, { validValues: Orientation.VALUES } ); - - return new TermList( this.getTerms( orientation ) ); - }, - - /** - * Creates a DerivedProperty with `map( totalSizeTerms )` for a particular orientation, where totalSizeTerms - * is an array of the total terms for that orientation. - * @private - * - * @param {function} map - function( {Array.} ): * - * @param {Orientation} orientation - * @returns {Property.<*|null>} - */ - createMappedTermsArrayProperty: function( map, orientation ) { - const self = this; - - const properties = _.flatten( this.partitions.get( orientation ).map( function( partition ) { - return [ partition.sizeProperty, partition.visibleProperty ]; - } ) ); - - return new DerivedProperty( properties, function() { - const terms = self.getTerms( orientation ); - if ( terms.length ) { - return map( terms ); + return new DerivedProperty( partitionProperties, function() { + return _.uniq( _.flatten( partitions.map( function( partition ) { + const range = partition.coordinateRangeProperty.value; + + // Ignore null range or invisible + if ( range === null || !partition.visibleProperty.value ) { + return []; } else { - return null; + return [ range.min, range.max ]; } - }, { - useDeepEquality: true - } ); - }, - - /** - * Returns a property that will contain an array of all unique partition boundaries (the minimum or maximum - * coordinate locations of a partition). - * @private - * - * @param {Orientation} orientation - * @returns {Property.>} - */ - createPartitionBoundariesProperty: function( orientation ) { - const partitions = this.partitions.get( orientation ); - - // Property dependencies - const partitionProperties = _.flatten( partitions.map( function( partition ) { - return [ partition.coordinateRangeProperty, partition.visibleProperty ]; - } ) ); - - return new DerivedProperty( partitionProperties, function() { - return _.uniq( _.flatten( partitions.map( function( partition ) { - const range = partition.coordinateRangeProperty.value; - - // Ignore null range or invisible - if ( range === null || !partition.visibleProperty.value ) { - return []; - } - else { - return [ range.min, range.max ]; - } - } ) ) ); - } ); - } - } ); -} ); + } ) ) ); + } ); + } +} ); \ No newline at end of file diff --git a/js/common/model/AreaCalculationChoice.js b/js/common/model/AreaCalculationChoice.js index 2d8df388..791fb083 100644 --- a/js/common/model/AreaCalculationChoice.js +++ b/js/common/model/AreaCalculationChoice.js @@ -5,35 +5,31 @@ * * @author Jonathan Olson */ -define( require => { - 'use strict'; - // modules - const areaModelCommon = require( 'AREA_MODEL_COMMON/areaModelCommon' ); +import areaModelCommon from '../../areaModelCommon.js'; - const AreaCalculationChoice = { - // Don't show the calculation panel/box at all - HIDDEN: 'HIDDEN', +const AreaCalculationChoice = { + // Don't show the calculation panel/box at all + HIDDEN: 'HIDDEN', - // Show one line (with adjacent lines somewhat faded). Only supported by the calculation panel (using - // CalculationNode), since the CalculationBox doesn't have room or the desire for the complexity. - LINE_BY_LINE: 'LINE_BY_LINE', + // Show one line (with adjacent lines somewhat faded). Only supported by the calculation panel (using + // CalculationNode), since the CalculationBox doesn't have room or the desire for the complexity. + LINE_BY_LINE: 'LINE_BY_LINE', - // Show all lines at once (supported by calculation panel/box) - SHOW_ALL_LINES: 'SHOW_ALL_LINES' - }; + // Show all lines at once (supported by calculation panel/box) + SHOW_ALL_LINES: 'SHOW_ALL_LINES' +}; - areaModelCommon.register( 'AreaCalculationChoice', AreaCalculationChoice ); +areaModelCommon.register( 'AreaCalculationChoice', AreaCalculationChoice ); - // @public {Array.} - All values the enumeration can take. - AreaCalculationChoice.VALUES = [ - AreaCalculationChoice.HIDDEN, - AreaCalculationChoice.LINE_BY_LINE, - AreaCalculationChoice.SHOW_ALL_LINES - ]; +// @public {Array.} - All values the enumeration can take. +AreaCalculationChoice.VALUES = [ + AreaCalculationChoice.HIDDEN, + AreaCalculationChoice.LINE_BY_LINE, + AreaCalculationChoice.SHOW_ALL_LINES +]; - // verify that enumeration is immutable, without the runtime penalty in production code - if ( assert ) { Object.freeze( AreaCalculationChoice ); } +// verify that enumeration is immutable, without the runtime penalty in production code +if ( assert ) { Object.freeze( AreaCalculationChoice ); } - return AreaCalculationChoice; -} ); +export default AreaCalculationChoice; \ No newline at end of file diff --git a/js/common/model/AreaDisplay.js b/js/common/model/AreaDisplay.js index 3785142f..8a08368d 100644 --- a/js/common/model/AreaDisplay.js +++ b/js/common/model/AreaDisplay.js @@ -8,132 +8,128 @@ * * @author Jonathan Olson */ -define( require => { - 'use strict'; - // modules - const areaModelCommon = require( 'AREA_MODEL_COMMON/areaModelCommon' ); - const DerivedProperty = require( 'AXON/DerivedProperty' ); - const DynamicProperty = require( 'AXON/DynamicProperty' ); - const inherit = require( 'PHET_CORE/inherit' ); - const merge = require( 'PHET_CORE/merge' ); - const OrientationPair = require( 'AREA_MODEL_COMMON/common/model/OrientationPair' ); +import DerivedProperty from '../../../../axon/js/DerivedProperty.js'; +import DynamicProperty from '../../../../axon/js/DynamicProperty.js'; +import inherit from '../../../../phet-core/js/inherit.js'; +import merge from '../../../../phet-core/js/merge.js'; +import areaModelCommon from '../../areaModelCommon.js'; +import OrientationPair from './OrientationPair.js'; - /** - * @constructor - * @extends {Object} - * - * @param {Property.} areaProperty - This changes when the scene changes (we have one area per scene) - */ - function AreaDisplay( areaProperty ) { - // @public {Property.} - this.areaProperty = areaProperty; +/** + * @constructor + * @extends {Object} + * + * @param {Property.} areaProperty - This changes when the scene changes (we have one area per scene) + */ +function AreaDisplay( areaProperty ) { + // @public {Property.} + this.areaProperty = areaProperty; + + // @public {OrientationPair.>>} + this.partitionsProperties = this.wrapOrientationPair( _.property( 'partitions' ) ); + + // @public {Property.>} + this.allPartitionsProperty = this.wrapObject( _.property( 'allPartitions' ) ); - // @public {OrientationPair.>>} - this.partitionsProperties = this.wrapOrientationPair( _.property( 'partitions' ) ); + // @public {OrientationPair.>} + this.colorProperties = this.wrapOrientationPairProperty( _.property( 'colorProperties' ) ); - // @public {Property.>} - this.allPartitionsProperty = this.wrapObject( _.property( 'allPartitions' ) ); + // @public {Property.} + this.coordinateRangeMaxProperty = this.wrapObject( _.property( 'coordinateRangeMax' ) ); - // @public {OrientationPair.>} - this.colorProperties = this.wrapOrientationPairProperty( _.property( 'colorProperties' ) ); + // @public {Property.} + this.allowExponentsProperty = this.wrapObject( _.property( 'allowExponents' ) ); - // @public {Property.} - this.coordinateRangeMaxProperty = this.wrapObject( _.property( 'coordinateRangeMax' ) ); + // @public {Property.} + this.calculationIndexProperty = this.wrapProperty( _.property( 'calculationIndexProperty' ) ); - // @public {Property.} - this.allowExponentsProperty = this.wrapObject( _.property( 'allowExponents' ) ); + // @public {Property.>} + this.partitionedAreasProperty = this.wrapObject( _.property( 'partitionedAreas' ) ); - // @public {Property.} - this.calculationIndexProperty = this.wrapProperty( _.property( 'calculationIndexProperty' ) ); + // @public {OrientationPair.>} + this.totalProperties = this.wrapOrientationPairProperty( _.property( 'totalProperties' ) ); - // @public {Property.>} - this.partitionedAreasProperty = this.wrapObject( _.property( 'partitionedAreas' ) ); + // @public {OrientationPair.>} + this.termListProperties = this.wrapOrientationPairProperty( _.property( 'termListProperties' ) ); + + // @public {Property.} + this.totalAreaProperty = this.wrapProperty( _.property( 'totalAreaProperty' ), { + useDeepEquality: true + } ); + + // @public {OrientationPair.>} + this.displayProperties = this.wrapOrientationPairProperty( _.property( 'displayProperties' ) ); + + // @public {OrientationPair.>>} + this.partitionBoundariesProperties = this.wrapOrientationPairProperty( _.property( 'partitionBoundariesProperties' ) ); +} + +areaModelCommon.register( 'AreaDisplay', AreaDisplay ); + +export default inherit( Object, AreaDisplay, { + /** + * Wraps an orientation pair into one that contains properties. + * @protected + * + * @param {function} map - function( {Area} ): {OrientationPair.<*>} + * @param {Object} [options] + * @returns {OrientationPair.>} + */ + wrapOrientationPair: function( map, options ) { + const self = this; - // @public {OrientationPair.>} - this.totalProperties = this.wrapOrientationPairProperty( _.property( 'totalProperties' ) ); + return OrientationPair.create( function( orientation ) { + return self.wrapObject( function( area ) { + return map( area ).get( orientation ); + }, options ); + } ); + }, - // @public {OrientationPair.>} - this.termListProperties = this.wrapOrientationPairProperty( _.property( 'termListProperties' ) ); + /** + * Wraps an orientation pair of properties + * @protected + * + * NOTE: This is like wrapOrientationPair, but with the critical difference of using wrapProperty internally instead + * of wrapObject. + * + * @param {function} map - function( {Area} ): {OrientationPair.>} + * @param {Object} [options] + * @returns {OrientationPair.>} + */ + wrapOrientationPairProperty: function( map, options ) { + const self = this; - // @public {Property.} - this.totalAreaProperty = this.wrapProperty( _.property( 'totalAreaProperty' ), { - useDeepEquality: true + return OrientationPair.create( function( orientation ) { + return self.wrapProperty( function( area ) { + return map( area ).get( orientation ); + }, options ); } ); + }, - // @public {OrientationPair.>} - this.displayProperties = this.wrapOrientationPairProperty( _.property( 'displayProperties' ) ); + /** + * Wraps a property. + * @protected + * + * @param {function} map - function( {Area} ): {Property.<*>} + * @param {Object} [options] + * @returns {Property.<*>} + */ + wrapProperty: function( map, options ) { + return new DynamicProperty( this.areaProperty, merge( { + derive: map + }, options ) ); + }, - // @public {OrientationPair.>>} - this.partitionBoundariesProperties = this.wrapOrientationPairProperty( _.property( 'partitionBoundariesProperties' ) ); + /** + * Wraps an object into a property. + * @protected + * + * @param {function} map - function( {Area} ): {*} + * @param {Object} [options] + * @returns {Property.<*>} + */ + wrapObject: function( map, options ) { + return new DerivedProperty( [ this.areaProperty ], map, options ); } - - areaModelCommon.register( 'AreaDisplay', AreaDisplay ); - - return inherit( Object, AreaDisplay, { - /** - * Wraps an orientation pair into one that contains properties. - * @protected - * - * @param {function} map - function( {Area} ): {OrientationPair.<*>} - * @param {Object} [options] - * @returns {OrientationPair.>} - */ - wrapOrientationPair: function( map, options ) { - const self = this; - - return OrientationPair.create( function( orientation ) { - return self.wrapObject( function( area ) { - return map( area ).get( orientation ); - }, options ); - } ); - }, - - /** - * Wraps an orientation pair of properties - * @protected - * - * NOTE: This is like wrapOrientationPair, but with the critical difference of using wrapProperty internally instead - * of wrapObject. - * - * @param {function} map - function( {Area} ): {OrientationPair.>} - * @param {Object} [options] - * @returns {OrientationPair.>} - */ - wrapOrientationPairProperty: function( map, options ) { - const self = this; - - return OrientationPair.create( function( orientation ) { - return self.wrapProperty( function( area ) { - return map( area ).get( orientation ); - }, options ); - } ); - }, - - /** - * Wraps a property. - * @protected - * - * @param {function} map - function( {Area} ): {Property.<*>} - * @param {Object} [options] - * @returns {Property.<*>} - */ - wrapProperty: function( map, options ) { - return new DynamicProperty( this.areaProperty, merge( { - derive: map - }, options ) ); - }, - - /** - * Wraps an object into a property. - * @protected - * - * @param {function} map - function( {Area} ): {*} - * @param {Object} [options] - * @returns {Property.<*>} - */ - wrapObject: function( map, options ) { - return new DerivedProperty( [ this.areaProperty ], map, options ); - } - } ); -} ); +} ); \ No newline at end of file diff --git a/js/common/model/AreaModelCommonModel.js b/js/common/model/AreaModelCommonModel.js index d70e30b2..1e716e19 100644 --- a/js/common/model/AreaModelCommonModel.js +++ b/js/common/model/AreaModelCommonModel.js @@ -5,123 +5,119 @@ * * @author Jonathan Olson */ -define( require => { - 'use strict'; - - // modules - const Area = require( 'AREA_MODEL_COMMON/common/model/Area' ); - const AreaCalculationChoice = require( 'AREA_MODEL_COMMON/common/model/AreaCalculationChoice' ); - const areaModelCommon = require( 'AREA_MODEL_COMMON/areaModelCommon' ); - const AreaModelCommonColorProfile = require( 'AREA_MODEL_COMMON/common/view/AreaModelCommonColorProfile' ); - const BooleanProperty = require( 'AXON/BooleanProperty' ); - const DerivedProperty = require( 'AXON/DerivedProperty' ); - const inherit = require( 'PHET_CORE/inherit' ); - const merge = require( 'PHET_CORE/merge' ); - const PartialProductsChoice = require( 'AREA_MODEL_COMMON/common/model/PartialProductsChoice' ); - const Property = require( 'AXON/Property' ); - /** - * @constructor - * @extends {Object} - * - * @param {Array.} areas - A list of all areas that can be switched between. - * @param {Area} defaultArea - The initial area - * @param {Object} [options] - */ - function AreaModelCommonModel( areas, defaultArea, options ) { +import BooleanProperty from '../../../../axon/js/BooleanProperty.js'; +import DerivedProperty from '../../../../axon/js/DerivedProperty.js'; +import Property from '../../../../axon/js/Property.js'; +import inherit from '../../../../phet-core/js/inherit.js'; +import merge from '../../../../phet-core/js/merge.js'; +import areaModelCommon from '../../areaModelCommon.js'; +import AreaModelCommonColorProfile from '../view/AreaModelCommonColorProfile.js'; +import Area from './Area.js'; +import AreaCalculationChoice from './AreaCalculationChoice.js'; +import PartialProductsChoice from './PartialProductsChoice.js'; - assert && assert( options === undefined || typeof options === 'object', 'If provided, options should be an object' ); +/** + * @constructor + * @extends {Object} + * + * @param {Array.} areas - A list of all areas that can be switched between. + * @param {Area} defaultArea - The initial area + * @param {Object} [options] + */ +function AreaModelCommonModel( areas, defaultArea, options ) { - options = merge( { - allowExponents: false, - isProportional: false, - initialAreaBoxExpanded: false, - initialAreaCalculationChoice: AreaCalculationChoice.HIDDEN, - initialPartialProductsChoice: PartialProductsChoice.HIDDEN - }, options ); + assert && assert( options === undefined || typeof options === 'object', 'If provided, options should be an object' ); - const self = this; + options = merge( { + allowExponents: false, + isProportional: false, + initialAreaBoxExpanded: false, + initialAreaCalculationChoice: AreaCalculationChoice.HIDDEN, + initialPartialProductsChoice: PartialProductsChoice.HIDDEN + }, options ); - // @public {Array.} - All areas that can be switched between - this.areas = areas; + const self = this; - // @public {boolean} - Whether exponents (powers of x) are allowed - this.allowExponents = options.allowExponents; + // @public {Array.} - All areas that can be switched between + this.areas = areas; - // @public {boolean} - Whether the area is proportional (or generic) - this.isProportional = options.isProportional; + // @public {boolean} - Whether exponents (powers of x) are allowed + this.allowExponents = options.allowExponents; - // @public {OrientationPair.>} - this.colorProperties = options.isProportional - ? AreaModelCommonColorProfile.proportionalColorProperties - : AreaModelCommonColorProfile.genericColorProperties; + // @public {boolean} - Whether the area is proportional (or generic) + this.isProportional = options.isProportional; - // @public {Property.} - The current area - this.currentAreaProperty = new Property( defaultArea, { - valueType: Area - } ); + // @public {OrientationPair.>} + this.colorProperties = options.isProportional + ? AreaModelCommonColorProfile.proportionalColorProperties + : AreaModelCommonColorProfile.genericColorProperties; - // @public {AreaDisplay} - this.areaDisplay = this.createAreaDisplay( this.currentAreaProperty ); + // @public {Property.} - The current area + this.currentAreaProperty = new Property( defaultArea, { + valueType: Area + } ); - // @public {Property.} - this.factorsBoxExpandedProperty = new BooleanProperty( true ); + // @public {AreaDisplay} + this.areaDisplay = this.createAreaDisplay( this.currentAreaProperty ); - // @public {Property.} - this.areaBoxExpandedProperty = new BooleanProperty( options.initialAreaBoxExpanded ); + // @public {Property.} + this.factorsBoxExpandedProperty = new BooleanProperty( true ); - // @public {Property.} + this.areaBoxExpandedProperty = new BooleanProperty( options.initialAreaBoxExpanded ); - // @public {Property.} - We need to properly update whenever the area changes OR any one of the - // individual totalAreaProperties. Since we are guaranteed to get a callback for these cases whenever one of the - // totalAreaProeprties changes, we listen to those instead. - this.totalAreaProperty = new DerivedProperty( totalAreaProperties, function() { - return self.currentAreaProperty.value.totalAreaProperty.value; - }, { - useDeepEquality: true - } ); - } + // @public {Property.} areaProperty - * @returns {AreaDisplay} - */ - createAreaDisplay: function( areaProperty ) { - throw new Error( 'abstract method' ); - }, - - /** - * Returns the model to its initial state. - * @public - */ - reset: function() { - this.currentAreaProperty.reset(); - this.factorsBoxExpandedProperty.reset(); - this.areaBoxExpandedProperty.reset(); - this.areaCalculationChoiceProperty.reset(); - this.partialProductsChoiceProperty.reset(); - - this.areas.forEach( function( area ) { - area.reset(); - } ); - } + const totalAreaProperties = [ this.currentAreaProperty ].concat( this.areas.map( function( area ) { + return area.totalAreaProperty; + } ) ); + + // @public {Property.} - We need to properly update whenever the area changes OR any one of the + // individual totalAreaProperties. Since we are guaranteed to get a callback for these cases whenever one of the + // totalAreaProeprties changes, we listen to those instead. + this.totalAreaProperty = new DerivedProperty( totalAreaProperties, function() { + return self.currentAreaProperty.value.totalAreaProperty.value; + }, { + useDeepEquality: true } ); -} ); +} + +areaModelCommon.register( 'AreaModelCommonModel', AreaModelCommonModel ); + +export default inherit( Object, AreaModelCommonModel, { + /** + * Abstract, returns an AreaDisplay concrete type. + * @protected + * + * @param {Property.} areaProperty + * @returns {AreaDisplay} + */ + createAreaDisplay: function( areaProperty ) { + throw new Error( 'abstract method' ); + }, + + /** + * Returns the model to its initial state. + * @public + */ + reset: function() { + this.currentAreaProperty.reset(); + this.factorsBoxExpandedProperty.reset(); + this.areaBoxExpandedProperty.reset(); + this.areaCalculationChoiceProperty.reset(); + this.partialProductsChoiceProperty.reset(); + + this.areas.forEach( function( area ) { + area.reset(); + } ); + } +} ); \ No newline at end of file diff --git a/js/common/model/OrientationPair.js b/js/common/model/OrientationPair.js index b8fb24ce..828103bb 100644 --- a/js/common/model/OrientationPair.js +++ b/js/common/model/OrientationPair.js @@ -5,93 +5,89 @@ * * @author Jonathan Olson */ -define( require => { - 'use strict'; - // modules - const areaModelCommon = require( 'AREA_MODEL_COMMON/areaModelCommon' ); - const inherit = require( 'PHET_CORE/inherit' ); - const Orientation = require( 'PHET_CORE/Orientation' ); - const validate = require( 'AXON/validate' ); +import validate from '../../../../axon/js/validate.js'; +import inherit from '../../../../phet-core/js/inherit.js'; +import Orientation from '../../../../phet-core/js/Orientation.js'; +import areaModelCommon from '../../areaModelCommon.js'; - /** - * @constructor - * @extends {Object} - * - * @param {*} horizontal - Value for the horizontal orientation - * @param {*} vertical - Value for the vertical orientation - */ - function OrientationPair( horizontal, vertical ) { - // @public {*} - this.horizontal = horizontal; +/** + * @constructor + * @extends {Object} + * + * @param {*} horizontal - Value for the horizontal orientation + * @param {*} vertical - Value for the vertical orientation + */ +function OrientationPair( horizontal, vertical ) { + // @public {*} + this.horizontal = horizontal; - // @public {*} - this.vertical = vertical; + // @public {*} + this.vertical = vertical; - // @public {Array.<*>} - this.values = [ horizontal, vertical ]; - } + // @public {Array.<*>} + this.values = [ horizontal, vertical ]; +} - areaModelCommon.register( 'OrientationPair', OrientationPair ); +areaModelCommon.register( 'OrientationPair', OrientationPair ); - return inherit( Object, OrientationPair, { - /** - * Returns the value associated with the particular orientation. - * @public - * - * @param {Orientation} orientation - * @returns {*} - */ - get: function( orientation ) { - validate( orientation, { validValues: Orientation.VALUES } ); +export default inherit( Object, OrientationPair, { + /** + * Returns the value associated with the particular orientation. + * @public + * + * @param {Orientation} orientation + * @returns {*} + */ + get: function( orientation ) { + validate( orientation, { validValues: Orientation.VALUES } ); - return orientation === Orientation.HORIZONTAL ? this.horizontal : this.vertical; - }, + return orientation === Orientation.HORIZONTAL ? this.horizontal : this.vertical; + }, - /** - * Returns a new OrientationPair with mapped values. - * @public - * - * @param {Function} mapFunction - function( {*}, {Orientation} ): {*} - * @returns {OrientationPair.<*>} - With the mapped values - */ - map: function( mapFunction ) { - return new OrientationPair( - mapFunction( this.horizontal, Orientation.HORIZONTAL ), - mapFunction( this.vertical, Orientation.VERTICAL ) - ); - }, + /** + * Returns a new OrientationPair with mapped values. + * @public + * + * @param {Function} mapFunction - function( {*}, {Orientation} ): {*} + * @returns {OrientationPair.<*>} - With the mapped values + */ + map: function( mapFunction ) { + return new OrientationPair( + mapFunction( this.horizontal, Orientation.HORIZONTAL ), + mapFunction( this.vertical, Orientation.VERTICAL ) + ); + }, - /** - * Calls the callback on each item of the orientation pair. - * @public - * - * @param {Function} callback - function( {*}, {Orientation} ) - */ - forEach: function( callback ) { - callback( this.horizontal, Orientation.HORIZONTAL ); - callback( this.vertical, Orientation.VERTICAL ); - }, + /** + * Calls the callback on each item of the orientation pair. + * @public + * + * @param {Function} callback - function( {*}, {Orientation} ) + */ + forEach: function( callback ) { + callback( this.horizontal, Orientation.HORIZONTAL ); + callback( this.vertical, Orientation.VERTICAL ); + }, - /** - * Calls reset() on each item in the orientation pair. - * @public - */ - reset: function() { - this.forEach( function( value ) { - value.reset(); - } ); - } - }, { - /** - * Creates an orientation pair based on a factory method. - * @public - * - * @param {function} factory - Called factory( {Orientation} ) : {*}, called once for each orientation to determine - * the value. - */ - create: function( factory ) { - return new OrientationPair( factory( Orientation.HORIZONTAL ), factory( Orientation.VERTICAL ) ); - } - } ); -} ); + /** + * Calls reset() on each item in the orientation pair. + * @public + */ + reset: function() { + this.forEach( function( value ) { + value.reset(); + } ); + } +}, { + /** + * Creates an orientation pair based on a factory method. + * @public + * + * @param {function} factory - Called factory( {Orientation} ) : {*}, called once for each orientation to determine + * the value. + */ + create: function( factory ) { + return new OrientationPair( factory( Orientation.HORIZONTAL ), factory( Orientation.VERTICAL ) ); + } +} ); \ No newline at end of file diff --git a/js/common/model/PartialProductsChoice.js b/js/common/model/PartialProductsChoice.js index e2ef7151..54cf5ca7 100644 --- a/js/common/model/PartialProductsChoice.js +++ b/js/common/model/PartialProductsChoice.js @@ -5,29 +5,25 @@ * * @author Jonathan Olson */ -define( require => { - 'use strict'; - // modules - const areaModelCommon = require( 'AREA_MODEL_COMMON/areaModelCommon' ); +import areaModelCommon from '../../areaModelCommon.js'; - const PartialProductsChoice = { - HIDDEN: 'HIDDEN', // e.g. nothing shown - PRODUCTS: 'PRODUCTS', // e.g. '52' - FACTORS: 'FACTORS' // e.g. '26 x 2' - }; +const PartialProductsChoice = { + HIDDEN: 'HIDDEN', // e.g. nothing shown + PRODUCTS: 'PRODUCTS', // e.g. '52' + FACTORS: 'FACTORS' // e.g. '26 x 2' +}; - areaModelCommon.register( 'PartialProductsChoice', PartialProductsChoice ); +areaModelCommon.register( 'PartialProductsChoice', PartialProductsChoice ); - // @public {Array.} - All values the enumeration can take. - PartialProductsChoice.VALUES = [ - PartialProductsChoice.HIDDEN, - PartialProductsChoice.PRODUCTS, - PartialProductsChoice.FACTORS - ]; +// @public {Array.} - All values the enumeration can take. +PartialProductsChoice.VALUES = [ + PartialProductsChoice.HIDDEN, + PartialProductsChoice.PRODUCTS, + PartialProductsChoice.FACTORS +]; - // verify that enum is immutable, without the runtime penalty in production code - if ( assert ) { Object.freeze( PartialProductsChoice ); } +// verify that enum is immutable, without the runtime penalty in production code +if ( assert ) { Object.freeze( PartialProductsChoice ); } - return PartialProductsChoice; -} ); +export default PartialProductsChoice; \ No newline at end of file diff --git a/js/common/model/Partition.js b/js/common/model/Partition.js index 22167c98..fe9931b9 100644 --- a/js/common/model/Partition.js +++ b/js/common/model/Partition.js @@ -5,67 +5,63 @@ * * @author Jonathan Olson */ -define( require => { - 'use strict'; - // modules - const areaModelCommon = require( 'AREA_MODEL_COMMON/areaModelCommon' ); - const BooleanProperty = require( 'AXON/BooleanProperty' ); - const inherit = require( 'PHET_CORE/inherit' ); - const Orientation = require( 'PHET_CORE/Orientation' ); - const Property = require( 'AXON/Property' ); - const Range = require( 'DOT/Range' ); - const Term = require( 'AREA_MODEL_COMMON/common/model/Term' ); - const validate = require( 'AXON/validate' ); +import BooleanProperty from '../../../../axon/js/BooleanProperty.js'; +import Property from '../../../../axon/js/Property.js'; +import validate from '../../../../axon/js/validate.js'; +import Range from '../../../../dot/js/Range.js'; +import inherit from '../../../../phet-core/js/inherit.js'; +import Orientation from '../../../../phet-core/js/Orientation.js'; +import areaModelCommon from '../../areaModelCommon.js'; +import Term from './Term.js'; - /** - * @constructor - * @extends {Object} - * - * @param {Orientation} orientation - * @param {Property.} colorProperty - */ - function Partition( orientation, colorProperty ) { - validate( orientation, { validValues: Orientation.VALUES } ); - assert && assert( colorProperty instanceof Property ); - - // @public {Property.} - Null indicates the size is not defined. - this.sizeProperty = new Property( null, { - useDeepEquality: true, - isValidValue: Term.isTermOrNull - } ); - - // @public {Orientation} - an intrinsic property of the Partition - this.orientation = orientation; +/** + * @constructor + * @extends {Object} + * + * @param {Orientation} orientation + * @param {Property.} colorProperty + */ +function Partition( orientation, colorProperty ) { + validate( orientation, { validValues: Orientation.VALUES } ); + assert && assert( colorProperty instanceof Property ); - // @public {Property.} - this.colorProperty = colorProperty; + // @public {Property.} - Null indicates the size is not defined. + this.sizeProperty = new Property( null, { + useDeepEquality: true, + isValidValue: Term.isTermOrNull + } ); - // @public {Property.} - Owned property, does not need to be disposed. - this.visibleProperty = new BooleanProperty( true ); + // @public {Orientation} - an intrinsic property of the Partition + this.orientation = orientation; - // @public {Property.} - The contained 'section' of the full available model area. Should be null when - // coordinates can't be computed. For generic partitions, it will be from 0 to 1. For proportional partitions, it - // will be from 0 to its maximum size. Owned property, does not need to be disposed. - this.coordinateRangeProperty = new Property( null, { - useDeepEquality: true, - isValidValue: function( value ) { - return value === null || value instanceof Range; - } - } ); - } + // @public {Property.} + this.colorProperty = colorProperty; - areaModelCommon.register( 'Partition', Partition ); + // @public {Property.} - Owned property, does not need to be disposed. + this.visibleProperty = new BooleanProperty( true ); - return inherit( Object, Partition, { - /** - * Returns whether this partition is defined, i.e. "is shown in the area, and has a size" - * @public - * - * @returns {boolean} - */ - isDefined: function() { - return this.visibleProperty.value && this.sizeProperty.value !== null; + // @public {Property.} - The contained 'section' of the full available model area. Should be null when + // coordinates can't be computed. For generic partitions, it will be from 0 to 1. For proportional partitions, it + // will be from 0 to its maximum size. Owned property, does not need to be disposed. + this.coordinateRangeProperty = new Property( null, { + useDeepEquality: true, + isValidValue: function( value ) { + return value === null || value instanceof Range; } } ); -} ); +} + +areaModelCommon.register( 'Partition', Partition ); + +export default inherit( Object, Partition, { + /** + * Returns whether this partition is defined, i.e. "is shown in the area, and has a size" + * @public + * + * @returns {boolean} + */ + isDefined: function() { + return this.visibleProperty.value && this.sizeProperty.value !== null; + } +} ); \ No newline at end of file diff --git a/js/common/model/PartitionedArea.js b/js/common/model/PartitionedArea.js index b1401682..a07344ad 100644 --- a/js/common/model/PartitionedArea.js +++ b/js/common/model/PartitionedArea.js @@ -5,50 +5,46 @@ * * @author Jonathan Olson */ -define( require => { - 'use strict'; - // modules - const areaModelCommon = require( 'AREA_MODEL_COMMON/areaModelCommon' ); - const DerivedProperty = require( 'AXON/DerivedProperty' ); - const inherit = require( 'PHET_CORE/inherit' ); - const Property = require( 'AXON/Property' ); - const Term = require( 'AREA_MODEL_COMMON/common/model/Term' ); +import DerivedProperty from '../../../../axon/js/DerivedProperty.js'; +import Property from '../../../../axon/js/Property.js'; +import inherit from '../../../../phet-core/js/inherit.js'; +import areaModelCommon from '../../areaModelCommon.js'; +import Term from './Term.js'; +/** + * @constructor + * @extends {Object} + * + * @param {OrientationPair.} partitions + */ +function PartitionedArea( partitions ) { + + // @public {OrientationPair.} + this.partitions = partitions; + + // @public {Property.} - Area may not be defined if the size of a partition is not defined. + this.areaProperty = new Property( null, { + useDeepEquality: true, + isValidValue: Term.isTermOrNull + } ); + + // @public {Property.} + this.visibleProperty = DerivedProperty.and( [ + partitions.horizontal.visibleProperty, + partitions.vertical.visibleProperty + ] ); +} + +areaModelCommon.register( 'PartitionedArea', PartitionedArea ); + +export default inherit( Object, PartitionedArea, { /** - * @constructor - * @extends {Object} - * - * @param {OrientationPair.} partitions + * Cleans up references. + * @public */ - function PartitionedArea( partitions ) { - - // @public {OrientationPair.} - this.partitions = partitions; - - // @public {Property.} - Area may not be defined if the size of a partition is not defined. - this.areaProperty = new Property( null, { - useDeepEquality: true, - isValidValue: Term.isTermOrNull - } ); - - // @public {Property.} - this.visibleProperty = DerivedProperty.and( [ - partitions.horizontal.visibleProperty, - partitions.vertical.visibleProperty - ] ); + dispose: function() { + this.visibleProperty.dispose(); + this.areaProperty.dispose(); } - - areaModelCommon.register( 'PartitionedArea', PartitionedArea ); - - return inherit( Object, PartitionedArea, { - /** - * Cleans up references. - * @public - */ - dispose: function() { - this.visibleProperty.dispose(); - this.areaProperty.dispose(); - } - } ); -} ); +} ); \ No newline at end of file diff --git a/js/common/model/Polynomial.js b/js/common/model/Polynomial.js index 53dec58e..1790fd0e 100644 --- a/js/common/model/Polynomial.js +++ b/js/common/model/Polynomial.js @@ -5,106 +5,102 @@ * * @author Jonathan Olson */ -define( require => { - 'use strict'; - // modules - const areaModelCommon = require( 'AREA_MODEL_COMMON/areaModelCommon' ); - const inherit = require( 'PHET_CORE/inherit' ); - const Term = require( 'AREA_MODEL_COMMON/common/model/Term' ); - const TermList = require( 'AREA_MODEL_COMMON/common/model/TermList' ); +import inherit from '../../../../phet-core/js/inherit.js'; +import areaModelCommon from '../../areaModelCommon.js'; +import Term from './Term.js'; +import TermList from './TermList.js'; - /** - * @constructor - * @extends {TermList} - * - * @param {Array.} terms - */ - function Polynomial( terms ) { - - const combinedTerms = []; - const sortedTerms = _.sortBy( terms, function( term ) { - return -term.power; - } ); +/** + * @constructor + * @extends {TermList} + * + * @param {Array.} terms + */ +function Polynomial( terms ) { - while ( sortedTerms.length ) { - let coefficient = 0; - const power = sortedTerms[ 0 ].power; + const combinedTerms = []; + const sortedTerms = _.sortBy( terms, function( term ) { + return -term.power; + } ); - while ( sortedTerms.length && sortedTerms[ 0 ].power === power ) { - coefficient += sortedTerms[ 0 ].coefficient; - sortedTerms.shift(); - } + while ( sortedTerms.length ) { + let coefficient = 0; + const power = sortedTerms[ 0 ].power; - if ( coefficient !== 0 ) { - combinedTerms.push( new Term( coefficient, power ) ); - } + while ( sortedTerms.length && sortedTerms[ 0 ].power === power ) { + coefficient += sortedTerms[ 0 ].coefficient; + sortedTerms.shift(); } - // If empty, add a zero term - if ( combinedTerms.length === 0 ) { - combinedTerms.push( new Term( 0 ) ); + if ( coefficient !== 0 ) { + combinedTerms.push( new Term( coefficient, power ) ); } + } - TermList.call( this, combinedTerms ); + // If empty, add a zero term + if ( combinedTerms.length === 0 ) { + combinedTerms.push( new Term( 0 ) ); } - areaModelCommon.register( 'Polynomial', Polynomial ); + TermList.call( this, combinedTerms ); +} - return inherit( TermList, Polynomial, { - /** - * Returns the coefficient in front of the term with the specific power. If it doesn't exist, 0 is used (since it's - * like an implicit term with a 0-coefficient) - * @public - * - * @param {number} power - * @returns {number} - */ - getCoefficient: function( power ) { - const term = _.find( this.terms, function( term ) { - return term.power === power; - } ); - if ( term ) { - return term.coefficient; - } - else { - return 0; - } - }, +areaModelCommon.register( 'Polynomial', Polynomial ); - /** - * Returns a new Term with the coefficient and power for the specified coefficient in our polynomial. - * @public - * - * @param {number} power - * @returns {Term} - */ - getTerm: function( power ) { - return new Term( this.getCoefficient( power ), power ); - }, +export default inherit( TermList, Polynomial, { + /** + * Returns the coefficient in front of the term with the specific power. If it doesn't exist, 0 is used (since it's + * like an implicit term with a 0-coefficient) + * @public + * + * @param {number} power + * @returns {number} + */ + getCoefficient: function( power ) { + const term = _.find( this.terms, function( term ) { + return term.power === power; + } ); + if ( term ) { + return term.coefficient; + } + else { + return 0; + } + }, - /** - * Addition of polynomials. - * @public - * @override - * - * @param {TermList} termList - * @returns {Polynomial} - */ - plus: function( termList ) { - return new Polynomial( this.terms.concat( termList.terms ) ); - }, + /** + * Returns a new Term with the coefficient and power for the specified coefficient in our polynomial. + * @public + * + * @param {number} power + * @returns {Term} + */ + getTerm: function( power ) { + return new Term( this.getCoefficient( power ), power ); + }, - /** - * Multiplication of polynomials. - * @public - * @override - * - * @param {TermList} termList - * @returns {Polynomial} - */ - times: function( termList ) { - return new Polynomial( TermList.prototype.times.call( this, termList ).terms ); - } - } ); -} ); + /** + * Addition of polynomials. + * @public + * @override + * + * @param {TermList} termList + * @returns {Polynomial} + */ + plus: function( termList ) { + return new Polynomial( this.terms.concat( termList.terms ) ); + }, + + /** + * Multiplication of polynomials. + * @public + * @override + * + * @param {TermList} termList + * @returns {Polynomial} + */ + times: function( termList ) { + return new Polynomial( TermList.prototype.times.call( this, termList ).terms ); + } +} ); \ No newline at end of file diff --git a/js/common/model/PolynomialTests.js b/js/common/model/PolynomialTests.js index 06fbcb0b..99e6ba85 100644 --- a/js/common/model/PolynomialTests.js +++ b/js/common/model/PolynomialTests.js @@ -5,56 +5,52 @@ * * @author Jonathan Olson */ -define( require => { - 'use strict'; - // modules - const Polynomial = require( 'AREA_MODEL_COMMON/common/model/Polynomial' ); - const Term = require( 'AREA_MODEL_COMMON/common/model/Term' ); +import Polynomial from './Polynomial.js'; +import Term from './Term.js'; - QUnit.module( 'Polynomial' ); +QUnit.module( 'Polynomial' ); - QUnit.test( 'Combining factors', function( assert ) { - assert.ok( new Polynomial( [ - new Term( 2, 1 ), - new Term( -3, 1 ) - ] ).getTerm( 1 ).equals( new Term( -1, 1 ) ), 'Should be combined into one' ); - } ); +QUnit.test( 'Combining factors', function( assert ) { + assert.ok( new Polynomial( [ + new Term( 2, 1 ), + new Term( -3, 1 ) + ] ).getTerm( 1 ).equals( new Term( -1, 1 ) ), 'Should be combined into one' ); +} ); - QUnit.test( 'Times', function( assert ) { - assert.ok( new Polynomial( [ - new Term( 4, 1 ), - new Term( -5, 0 ) - ] ).times( new Polynomial( [ - new Term( 2, 2 ), - new Term( 3, 1 ), - new Term( -6, 0 ) - ] ) ).equals( new Polynomial( [ - new Term( 8, 3 ), - new Term( 2, 2 ), - new Term( -39, 1 ), - new Term( 30, 0 ) - ] ) ), 'Example multiplication' ); +QUnit.test( 'Times', function( assert ) { + assert.ok( new Polynomial( [ + new Term( 4, 1 ), + new Term( -5, 0 ) + ] ).times( new Polynomial( [ + new Term( 2, 2 ), + new Term( 3, 1 ), + new Term( -6, 0 ) + ] ) ).equals( new Polynomial( [ + new Term( 8, 3 ), + new Term( 2, 2 ), + new Term( -39, 1 ), + new Term( 30, 0 ) + ] ) ), 'Example multiplication' ); - assert.ok( new Polynomial( [ - new Term( 3, 1 ), // 3x - new Term( -2, 0 ) // -2 - ] ).times( new Polynomial( [ - new Term( 2, 1 ), // 2x - new Term( 1, 0 ) // 1 - ] ) ).equals( new Polynomial( [ - new Term( 6, 2 ), // 6x^2 - new Term( -1, 1 ), // -x - new Term( -2, 0 ) // -2 - ] ) ), 'Example multiplication' ); + assert.ok( new Polynomial( [ + new Term( 3, 1 ), // 3x + new Term( -2, 0 ) // -2 + ] ).times( new Polynomial( [ + new Term( 2, 1 ), // 2x + new Term( 1, 0 ) // 1 + ] ) ).equals( new Polynomial( [ + new Term( 6, 2 ), // 6x^2 + new Term( -1, 1 ), // -x + new Term( -2, 0 ) // -2 + ] ) ), 'Example multiplication' ); - assert.ok( new Polynomial( [ - new Term( 1, 2 ) // x^2 - ] ).times( new Polynomial( [ - new Term( 1, 0 ), // 1 - new Term( 1, 0 ) // 1 - ] ) ).equals( new Polynomial( [ - new Term( 2, 2 ) // 2x^2 - ] ) ), 'Combination and multiplication' ); - } ); -} ); + assert.ok( new Polynomial( [ + new Term( 1, 2 ) // x^2 + ] ).times( new Polynomial( [ + new Term( 1, 0 ), // 1 + new Term( 1, 0 ) // 1 + ] ) ).equals( new Polynomial( [ + new Term( 2, 2 ) // 2x^2 + ] ) ), 'Combination and multiplication' ); +} ); \ No newline at end of file diff --git a/js/common/model/Term.js b/js/common/model/Term.js index e1c8ef14..ad8fcce3 100644 --- a/js/common/model/Term.js +++ b/js/common/model/Term.js @@ -5,160 +5,156 @@ * * @author Jonathan Olson */ -define( require => { - 'use strict'; - // modules - const areaModelCommon = require( 'AREA_MODEL_COMMON/areaModelCommon' ); - const AreaModelCommonConstants = require( 'AREA_MODEL_COMMON/common/AreaModelCommonConstants' ); - const inherit = require( 'PHET_CORE/inherit' ); - const MathSymbols = require( 'SCENERY_PHET/MathSymbols' ); - const Utils = require( 'DOT/Utils' ); +import Utils from '../../../../dot/js/Utils.js'; +import inherit from '../../../../phet-core/js/inherit.js'; +import MathSymbols from '../../../../scenery-phet/js/MathSymbols.js'; +import areaModelCommon from '../../areaModelCommon.js'; +import AreaModelCommonConstants from '../AreaModelCommonConstants.js'; +/** + * @constructor + * @extends {Object} + * + * @param {number} coefficient + * @param {number} [power] + */ +function Term( coefficient, power ) { + // Properly handle 0x. This generally removes a lot of special-cases (e.g. 0x^2 equals 0), and allows things like + // Polynomial to easily have one term of each power (where if the constant is 0, its power is 0). Also applies to + // things like sorting by power or how we want things displayed (0, not 0x). + // See https://github.com/phetsims/area-model-common/issues/6 + if ( coefficient === 0 ) { + power = 0; + } + + // The power argument is optional--if not supplied, the power defaults to 0. + power = ( power === undefined ? 0 : power ); + + assert && assert( typeof coefficient === 'number' && isFinite( coefficient ), + 'Coefficient only needs to be a finite number' ); + + assert && assert( typeof power === 'number' && isFinite( power ) ); + + // @public {number} + this.coefficient = coefficient; + + // @public {number} + this.power = power; +} + +areaModelCommon.register( 'Term', Term ); + +export default inherit( Object, Term, { /** - * @constructor - * @extends {Object} + * Term multiplication. + * @public * - * @param {number} coefficient - * @param {number} [power] + * @param {Term} term + * @returns {Term} */ - function Term( coefficient, power ) { - // Properly handle 0x. This generally removes a lot of special-cases (e.g. 0x^2 equals 0), and allows things like - // Polynomial to easily have one term of each power (where if the constant is 0, its power is 0). Also applies to - // things like sorting by power or how we want things displayed (0, not 0x). - // See https://github.com/phetsims/area-model-common/issues/6 - if ( coefficient === 0 ) { - power = 0; - } + times: function( term ) { + return new Term( this.coefficient * term.coefficient, this.power + term.power ); + }, - // The power argument is optional--if not supplied, the power defaults to 0. - power = ( power === undefined ? 0 : power ); + /** + * Equality + * @public + * + * @param {Term} term + * @returns {boolean} + */ + equals: function( term ) { + // Handle floating-point error for common cases. Epsilon guessed at what may be most relevant if this is moved + // to common code. + return Utils.equalsEpsilon( this.coefficient, term.coefficient, 1e-7 ) && this.power === term.power; + }, - assert && assert( typeof coefficient === 'number' && isFinite( coefficient ), - 'Coefficient only needs to be a finite number' ); + /** + * Returns a string representation of the term suitable for RichText, but without any signs. + * @public + * + * @returns {string} + */ + toNoSignRichString: function() { + let string = ''; - assert && assert( typeof power === 'number' && isFinite( power ) ); + if ( Math.abs( this.coefficient ) !== 1 || this.power === 0 ) { + string += Utils.toFixedNumber( Math.abs( this.coefficient ), 2 ); + } + if ( this.power > 0 ) { + string += AreaModelCommonConstants.X_VARIABLE_RICH_STRING; + } + if ( this.power > 1 ) { + string += '' + this.power + ''; + } - // @public {number} - this.coefficient = coefficient; + return string; + }, - // @public {number} - this.power = power; - } + /** + * Returns a string representation of the term suitable for RichText. + * @public + * + * @param {boolean} includeBinaryOperation - If true, assumes we are in a sum and not the first term so includes + * an initial plus or minus. If false, only a unary minus would be included. + * @returns {string} + */ + toRichString: function( includeBinaryOperation ) { + assert && assert( typeof includeBinaryOperation === 'boolean' ); - areaModelCommon.register( 'Term', Term ); - - return inherit( Object, Term, { - /** - * Term multiplication. - * @public - * - * @param {Term} term - * @returns {Term} - */ - times: function( term ) { - return new Term( this.coefficient * term.coefficient, this.power + term.power ); - }, - - /** - * Equality - * @public - * - * @param {Term} term - * @returns {boolean} - */ - equals: function( term ) { - // Handle floating-point error for common cases. Epsilon guessed at what may be most relevant if this is moved - // to common code. - return Utils.equalsEpsilon( this.coefficient, term.coefficient, 1e-7 ) && this.power === term.power; - }, - - /** - * Returns a string representation of the term suitable for RichText, but without any signs. - * @public - * - * @returns {string} - */ - toNoSignRichString: function() { - let string = ''; - - if ( Math.abs( this.coefficient ) !== 1 || this.power === 0 ) { - string += Utils.toFixedNumber( Math.abs( this.coefficient ), 2 ); - } - if ( this.power > 0 ) { - string += AreaModelCommonConstants.X_VARIABLE_RICH_STRING; - } - if ( this.power > 1 ) { - string += '' + this.power + ''; - } + let string = ''; - return string; - }, - - /** - * Returns a string representation of the term suitable for RichText. - * @public - * - * @param {boolean} includeBinaryOperation - If true, assumes we are in a sum and not the first term so includes - * an initial plus or minus. If false, only a unary minus would be included. - * @returns {string} - */ - toRichString: function( includeBinaryOperation ) { - assert && assert( typeof includeBinaryOperation === 'boolean' ); - - let string = ''; - - if ( includeBinaryOperation ) { - if ( this.coefficient < 0 ) { - string += ' ' + MathSymbols.MINUS + ' '; - } - else { - string += ' ' + MathSymbols.PLUS + ' '; - } + if ( includeBinaryOperation ) { + if ( this.coefficient < 0 ) { + string += ' ' + MathSymbols.MINUS + ' '; } else { - if ( this.coefficient < 0 ) { - string += MathSymbols.UNARY_MINUS; - } + string += ' ' + MathSymbols.PLUS + ' '; } + } + else { + if ( this.coefficient < 0 ) { + string += MathSymbols.UNARY_MINUS; + } + } - string += this.toNoSignRichString(); + string += this.toNoSignRichString(); - return string; + return string; + } +}, { + /** + * Returns the longest generic term's toRichString (for proper sizing). + * @public + * + * @param {boolean} allowExponents - Whether powers of x can be included + * @param {number} digitCount - If no powers of x allowed, how many numeric digits can be allowed. + * @returns {string} + */ + getLargestGenericString: function( allowExponents, digitCount ) { + const digits = _.range( 0, digitCount ).map( function() { + return AreaModelCommonConstants.MEASURING_CHARACTER; + } ).join( '' ); + + if ( allowExponents ) { + // The square is an example of an exponent that will increase the height of the displayed string, so we want to + // include it if exponents are allowed. + return MathSymbols.MINUS + digits + 'x2'; } - }, { - /** - * Returns the longest generic term's toRichString (for proper sizing). - * @public - * - * @param {boolean} allowExponents - Whether powers of x can be included - * @param {number} digitCount - If no powers of x allowed, how many numeric digits can be allowed. - * @returns {string} - */ - getLargestGenericString: function( allowExponents, digitCount ) { - const digits = _.range( 0, digitCount ).map( function() { - return AreaModelCommonConstants.MEASURING_CHARACTER; - } ).join( '' ); - - if ( allowExponents ) { - // The square is an example of an exponent that will increase the height of the displayed string, so we want to - // include it if exponents are allowed. - return MathSymbols.MINUS + digits + 'x2'; - } - else { - return MathSymbols.MINUS + digits; - } - }, - - /** - * Returns whether the parameter is a Term (or is null) - * @public - * - * @param {*} thing - * @returns {boolean} - */ - isTermOrNull: function( thing ) { - return thing === null || thing instanceof Term; + else { + return MathSymbols.MINUS + digits; } - } ); -} ); + }, + + /** + * Returns whether the parameter is a Term (or is null) + * @public + * + * @param {*} thing + * @returns {boolean} + */ + isTermOrNull: function( thing ) { + return thing === null || thing instanceof Term; + } +} ); \ No newline at end of file diff --git a/js/common/model/TermList.js b/js/common/model/TermList.js index 5b31ebb5..a10fe0e7 100644 --- a/js/common/model/TermList.js +++ b/js/common/model/TermList.js @@ -6,113 +6,109 @@ * * @author Jonathan Olson */ -define( require => { - 'use strict'; - // modules - const areaModelCommon = require( 'AREA_MODEL_COMMON/areaModelCommon' ); - const inherit = require( 'PHET_CORE/inherit' ); +import inherit from '../../../../phet-core/js/inherit.js'; +import areaModelCommon from '../../areaModelCommon.js'; - /** - * @constructor - * @extends {Object} - * - * @param {Array.} terms - */ - function TermList( terms ) { +/** + * @constructor + * @extends {Object} + * + * @param {Array.} terms + */ +function TermList( terms ) { - // @public {Array.} - this.terms = terms; - } + // @public {Array.} + this.terms = terms; +} - areaModelCommon.register( 'TermList', TermList ); +areaModelCommon.register( 'TermList', TermList ); - return inherit( Object, TermList, { +export default inherit( Object, TermList, { - /** - * Addition of term lists. - * @public - * - * @param {TermList} termList - * @returns {TermList} - */ - plus: function( termList ) { - return new TermList( this.terms.concat( termList.terms ) ); - }, + /** + * Addition of term lists. + * @public + * + * @param {TermList} termList + * @returns {TermList} + */ + plus: function( termList ) { + return new TermList( this.terms.concat( termList.terms ) ); + }, + + /** + * Multiplication of term lists. + * @public + * + * @param {TermList} termList + * @returns {TermList} + */ + times: function( termList ) { + return new TermList( _.flatten( this.terms.map( function( term ) { + return termList.terms.map( function( otherTerm ) { + return term.times( otherTerm ); + } ); + } ) ) ); + }, - /** - * Multiplication of term lists. - * @public - * - * @param {TermList} termList - * @returns {TermList} - */ - times: function( termList ) { - return new TermList( _.flatten( this.terms.map( function( term ) { - return termList.terms.map( function( otherTerm ) { - return term.times( otherTerm ); - } ); - } ) ) ); - }, + /** + * Returns a new TermList, (stable) sorted by the exponent. + * @public + * + * @returns {TermList} + */ + orderedByExponent: function() { + return new TermList( _.sortBy( this.terms, function( term ) { + return -term.power; + } ) ); + }, - /** - * Returns a new TermList, (stable) sorted by the exponent. - * @public - * - * @returns {TermList} - */ - orderedByExponent: function() { - return new TermList( _.sortBy( this.terms, function( term ) { - return -term.power; - } ) ); - }, + /** + * Returns whether any of the terms have a negative coefficient. + * @public + * + * @returns {boolean} + */ + hasNegativeTerm: function() { + return _.some( this.terms, function( term ) { + return term.coefficient < 0; + } ); + }, - /** - * Returns whether any of the terms have a negative coefficient. - * @public - * - * @returns {boolean} - */ - hasNegativeTerm: function() { - return _.some( this.terms, function( term ) { - return term.coefficient < 0; - } ); - }, + /** + * Returns a string suitable for RichText + * @public + * + * @returns {string} + */ + toRichString: function() { + return this.terms.map( function( term, index ) { + return term.toRichString( index > 0 ); + } ).join( '' ); + }, - /** - * Returns a string suitable for RichText - * @public - * - * @returns {string} - */ - toRichString: function() { - return this.terms.map( function( term, index ) { - return term.toRichString( index > 0 ); - } ).join( '' ); - }, + /** + * Equality for just whether the terms are the same (so a TermList can be compared to a Polynomial and be equal + * despite being different types.) Note that Polynomial orders the terms so this order-dependent check will still + * work. + * @public + * + * @param {TermList} termList + */ + equals: function( termList ) { + if ( this.terms.length !== termList.terms.length ) { + return false; + } - /** - * Equality for just whether the terms are the same (so a TermList can be compared to a Polynomial and be equal - * despite being different types.) Note that Polynomial orders the terms so this order-dependent check will still - * work. - * @public - * - * @param {TermList} termList - */ - equals: function( termList ) { - if ( this.terms.length !== termList.terms.length ) { + // This uses a reverse search instead of a forward search for optimization--probably not important for Area Model, + // but optimized in case it is moved to common code. + for ( let i = this.terms.length - 1; i >= 0; i-- ) { + if ( !this.terms[ i ].equals( termList.terms[ i ] ) ) { return false; } - - // This uses a reverse search instead of a forward search for optimization--probably not important for Area Model, - // but optimized in case it is moved to common code. - for ( let i = this.terms.length - 1; i >= 0; i-- ) { - if ( !this.terms[ i ].equals( termList.terms[ i ] ) ) { - return false; - } - } - - return true; } - } ); -} ); + + return true; + } +} ); \ No newline at end of file diff --git a/js/common/model/TermListTests.js b/js/common/model/TermListTests.js index 865227a8..6522da73 100644 --- a/js/common/model/TermListTests.js +++ b/js/common/model/TermListTests.js @@ -5,61 +5,57 @@ * * @author Jonathan Olson */ -define( require => { - 'use strict'; - // modules - const Term = require( 'AREA_MODEL_COMMON/common/model/Term' ); - const TermList = require( 'AREA_MODEL_COMMON/common/model/TermList' ); +import Term from './Term.js'; +import TermList from './TermList.js'; - QUnit.module( 'TermList' ); +QUnit.module( 'TermList' ); - QUnit.test( 'Times', function( assert ) { - assert.ok( new TermList( [ - new Term( 3, 1 ), // 3x - new Term( -2, 0 ) // -2 - ] ).times( new TermList( [ - new Term( 2, 1 ), // 2x - new Term( 1, 0 ) // 1 - ] ) ).equals( new TermList( [ - new Term( 6, 2 ), // 6x^2 - new Term( 3, 1 ), // 3x - new Term( -4, 1 ), // -4x - new Term( -2, 0 ) // -2 - ] ) ), 'Example multiplication' ); - } ); - - QUnit.test( 'Ordering', function( assert ) { - assert.ok( new TermList( [ - new Term( 3, 2 ), - new Term( 1, 1 ), - new Term( 2, 0 ) - ] ).orderedByExponent().equals( new TermList( [ - new Term( 3, 2 ), - new Term( 1, 1 ), - new Term( 2, 0 ) - ] ) ), 'Ordering (no change)' ); - assert.ok( new TermList( [ - new Term( 2, 0 ), - new Term( 1, 1 ), - new Term( 3, 2 ) - ] ).orderedByExponent().equals( new TermList( [ - new Term( 3, 2 ), - new Term( 1, 1 ), - new Term( 2, 0 ) - ] ) ), 'Ordering (reversed)' ); - } ); +QUnit.test( 'Times', function( assert ) { + assert.ok( new TermList( [ + new Term( 3, 1 ), // 3x + new Term( -2, 0 ) // -2 + ] ).times( new TermList( [ + new Term( 2, 1 ), // 2x + new Term( 1, 0 ) // 1 + ] ) ).equals( new TermList( [ + new Term( 6, 2 ), // 6x^2 + new Term( 3, 1 ), // 3x + new Term( -4, 1 ), // -4x + new Term( -2, 0 ) // -2 + ] ) ), 'Example multiplication' ); +} ); - QUnit.test( 'Negative test', function( assert ) { - assert.ok( !new TermList( [ - new Term( 3, 2 ), - new Term( 1, 1 ), - new Term( 2, 0 ) - ] ).hasNegativeTerm(), 'No negative' ); - assert.ok( new TermList( [ - new Term( 3, 2 ), - new Term( 1, 1 ), - new Term( -2, 0 ) - ] ).hasNegativeTerm(), 'Has negative' ); - } ); +QUnit.test( 'Ordering', function( assert ) { + assert.ok( new TermList( [ + new Term( 3, 2 ), + new Term( 1, 1 ), + new Term( 2, 0 ) + ] ).orderedByExponent().equals( new TermList( [ + new Term( 3, 2 ), + new Term( 1, 1 ), + new Term( 2, 0 ) + ] ) ), 'Ordering (no change)' ); + assert.ok( new TermList( [ + new Term( 2, 0 ), + new Term( 1, 1 ), + new Term( 3, 2 ) + ] ).orderedByExponent().equals( new TermList( [ + new Term( 3, 2 ), + new Term( 1, 1 ), + new Term( 2, 0 ) + ] ) ), 'Ordering (reversed)' ); } ); + +QUnit.test( 'Negative test', function( assert ) { + assert.ok( !new TermList( [ + new Term( 3, 2 ), + new Term( 1, 1 ), + new Term( 2, 0 ) + ] ).hasNegativeTerm(), 'No negative' ); + assert.ok( new TermList( [ + new Term( 3, 2 ), + new Term( 1, 1 ), + new Term( -2, 0 ) + ] ).hasNegativeTerm(), 'Has negative' ); +} ); \ No newline at end of file diff --git a/js/common/model/TermTests.js b/js/common/model/TermTests.js index a524af4d..6f8426a5 100644 --- a/js/common/model/TermTests.js +++ b/js/common/model/TermTests.js @@ -5,23 +5,19 @@ * * @author Jonathan Olson */ -define( require => { - 'use strict'; - // modules - const Term = require( 'AREA_MODEL_COMMON/common/model/Term' ); +import Term from './Term.js'; - QUnit.module( 'Term' ); +QUnit.module( 'Term' ); - QUnit.test( 'Equality', function( assert ) { - assert.ok( new Term( 1, 2 ).equals( new Term( 1, 2 ) ), 'Basic equality' ); - assert.ok( !new Term( 2, 2 ).equals( new Term( 1, 2 ) ), 'Basic inequality' ); - assert.ok( !new Term( 1, 2 ).equals( new Term( 1, 1 ) ), 'Basic inequality' ); - } ); - - QUnit.test( 'Multiplication', function( assert ) { - assert.ok( new Term( 3, 0 ).times( new Term( 5, 0 ) ).equals( new Term( 15, 0 ) ), 'Basic multiplication' ); - assert.ok( new Term( 3, 1 ).times( new Term( 5, 0 ) ).equals( new Term( 15, 1 ) ), 'Basic multiplication (one exponent)' ); - assert.ok( new Term( 3, 2 ).times( new Term( 5, 1 ) ).equals( new Term( 15, 3 ) ), 'Basic multiplication (two exponents)' ); - } ); +QUnit.test( 'Equality', function( assert ) { + assert.ok( new Term( 1, 2 ).equals( new Term( 1, 2 ) ), 'Basic equality' ); + assert.ok( !new Term( 2, 2 ).equals( new Term( 1, 2 ) ), 'Basic inequality' ); + assert.ok( !new Term( 1, 2 ).equals( new Term( 1, 1 ) ), 'Basic inequality' ); } ); + +QUnit.test( 'Multiplication', function( assert ) { + assert.ok( new Term( 3, 0 ).times( new Term( 5, 0 ) ).equals( new Term( 15, 0 ) ), 'Basic multiplication' ); + assert.ok( new Term( 3, 1 ).times( new Term( 5, 0 ) ).equals( new Term( 15, 1 ) ), 'Basic multiplication (one exponent)' ); + assert.ok( new Term( 3, 2 ).times( new Term( 5, 1 ) ).equals( new Term( 15, 3 ) ), 'Basic multiplication (two exponents)' ); +} ); \ No newline at end of file diff --git a/js/common/view/AreaCalculationRadioButtonGroup.js b/js/common/view/AreaCalculationRadioButtonGroup.js index 72c13375..dca696c2 100644 --- a/js/common/view/AreaCalculationRadioButtonGroup.js +++ b/js/common/view/AreaCalculationRadioButtonGroup.js @@ -7,69 +7,66 @@ * * @author Jonathan Olson */ -define( require => { - 'use strict'; - // modules - const AlignBox = require( 'SCENERY/nodes/AlignBox' ); - const AreaCalculationChoice = require( 'AREA_MODEL_COMMON/common/model/AreaCalculationChoice' ); - const areaModelCommon = require( 'AREA_MODEL_COMMON/areaModelCommon' ); - const AreaModelCommonColorProfile = require( 'AREA_MODEL_COMMON/common/view/AreaModelCommonColorProfile' ); - const AreaModelCommonRadioButtonGroup = require( 'AREA_MODEL_COMMON/common/view/AreaModelCommonRadioButtonGroup' ); - const FontAwesomeNode = require( 'SUN/FontAwesomeNode' ); - const inherit = require( 'PHET_CORE/inherit' ); - const Rectangle = require( 'SCENERY/nodes/Rectangle' ); - const VBox = require( 'SCENERY/nodes/VBox' ); +import inherit from '../../../../phet-core/js/inherit.js'; +import AlignBox from '../../../../scenery/js/nodes/AlignBox.js'; +import Rectangle from '../../../../scenery/js/nodes/Rectangle.js'; +import VBox from '../../../../scenery/js/nodes/VBox.js'; +import FontAwesomeNode from '../../../../sun/js/FontAwesomeNode.js'; +import areaModelCommon from '../../areaModelCommon.js'; +import AreaCalculationChoice from '../model/AreaCalculationChoice.js'; +import AreaModelCommonColorProfile from './AreaModelCommonColorProfile.js'; +import AreaModelCommonRadioButtonGroup from './AreaModelCommonRadioButtonGroup.js'; - /** - * @constructor - * @extends {AreaModelCommonRadioButtonGroup} - * - * @param {Property.} areaCalculationChoiceProperty - * @param {AlignGroup} selectionButtonAlignGroup - */ - function AreaCalculationRadioButtonGroup( areaCalculationChoiceProperty, selectionButtonAlignGroup ) { +/** + * @constructor + * @extends {AreaModelCommonRadioButtonGroup} + * + * @param {Property.} areaCalculationChoiceProperty + * @param {AlignGroup} selectionButtonAlignGroup + */ +function AreaCalculationRadioButtonGroup( areaCalculationChoiceProperty, selectionButtonAlignGroup ) { - const darkColorProperty = AreaModelCommonColorProfile.calculationIconDarkProperty; - const lightColorProperty = AreaModelCommonColorProfile.calculationIconLightProperty; + const darkColorProperty = AreaModelCommonColorProfile.calculationIconDarkProperty; + const lightColorProperty = AreaModelCommonColorProfile.calculationIconLightProperty; - AreaModelCommonRadioButtonGroup.call( this, areaCalculationChoiceProperty, [ { - value: AreaCalculationChoice.HIDDEN, - node: new AlignBox( new FontAwesomeNode( 'eye_close', { scale: 0.8 } ), { group: selectionButtonAlignGroup } ) - }, { - value: AreaCalculationChoice.LINE_BY_LINE, - node: new AlignBox( createCalculationIcon( darkColorProperty, lightColorProperty ), { group: selectionButtonAlignGroup } ) - }, { - value: AreaCalculationChoice.SHOW_ALL_LINES, - node: new AlignBox( createCalculationIcon( darkColorProperty, darkColorProperty ), { group: selectionButtonAlignGroup } ) - } ] ); - } + AreaModelCommonRadioButtonGroup.call( this, areaCalculationChoiceProperty, [ { + value: AreaCalculationChoice.HIDDEN, + node: new AlignBox( new FontAwesomeNode( 'eye_close', { scale: 0.8 } ), { group: selectionButtonAlignGroup } ) + }, { + value: AreaCalculationChoice.LINE_BY_LINE, + node: new AlignBox( createCalculationIcon( darkColorProperty, lightColorProperty ), { group: selectionButtonAlignGroup } ) + }, { + value: AreaCalculationChoice.SHOW_ALL_LINES, + node: new AlignBox( createCalculationIcon( darkColorProperty, darkColorProperty ), { group: selectionButtonAlignGroup } ) + } ] ); +} - areaModelCommon.register( 'AreaCalculationRadioButtonGroup', AreaCalculationRadioButtonGroup ); +areaModelCommon.register( 'AreaCalculationRadioButtonGroup', AreaCalculationRadioButtonGroup ); - /** - * Creates a calculation icon with two fills. - * @private - * - * @param {Property.} topColorProperty - Fill for the top line - * @param {Property.} bottomColorProperty - Fill for the bottom-most three lines - * @returns {Node} - */ - function createCalculationIcon( topColorProperty, bottomColorProperty ) { - const height = 5; - const fullWidth = 30; - const partialWidth = 20; - return new VBox( { - children: [ - new Rectangle( 0, 0, partialWidth, height, { fill: topColorProperty } ), - new Rectangle( 0, 0, fullWidth, height, { fill: bottomColorProperty } ), - new Rectangle( 0, 0, partialWidth, height, { fill: bottomColorProperty } ), - new Rectangle( 0, 0, fullWidth, height, { fill: bottomColorProperty } ) - ], - align: 'left', - spacing: 2 - } ); - } +/** + * Creates a calculation icon with two fills. + * @private + * + * @param {Property.} topColorProperty - Fill for the top line + * @param {Property.} bottomColorProperty - Fill for the bottom-most three lines + * @returns {Node} + */ +function createCalculationIcon( topColorProperty, bottomColorProperty ) { + const height = 5; + const fullWidth = 30; + const partialWidth = 20; + return new VBox( { + children: [ + new Rectangle( 0, 0, partialWidth, height, { fill: topColorProperty } ), + new Rectangle( 0, 0, fullWidth, height, { fill: bottomColorProperty } ), + new Rectangle( 0, 0, partialWidth, height, { fill: bottomColorProperty } ), + new Rectangle( 0, 0, fullWidth, height, { fill: bottomColorProperty } ) + ], + align: 'left', + spacing: 2 + } ); +} - return inherit( AreaModelCommonRadioButtonGroup, AreaCalculationRadioButtonGroup ); -} ); +inherit( AreaModelCommonRadioButtonGroup, AreaCalculationRadioButtonGroup ); +export default AreaCalculationRadioButtonGroup; \ No newline at end of file diff --git a/js/common/view/AreaDisplayNode.js b/js/common/view/AreaDisplayNode.js index 96a4ef52..1716591e 100644 --- a/js/common/view/AreaDisplayNode.js +++ b/js/common/view/AreaDisplayNode.js @@ -7,309 +7,305 @@ * * @author Jonathan Olson */ -define( require => { - 'use strict'; - - // modules - const areaModelCommon = require( 'AREA_MODEL_COMMON/areaModelCommon' ); - const AreaModelCommonA11yStrings = require( 'AREA_MODEL_COMMON/AreaModelCommonA11yStrings' ); - const AreaModelCommonColorProfile = require( 'AREA_MODEL_COMMON/common/view/AreaModelCommonColorProfile' ); - const AreaModelCommonConstants = require( 'AREA_MODEL_COMMON/common/AreaModelCommonConstants' ); - const Bounds2 = require( 'DOT/Bounds2' ); - const DerivedProperty = require( 'AXON/DerivedProperty' ); - const EraserButton = require( 'SCENERY_PHET/buttons/EraserButton' ); - const inherit = require( 'PHET_CORE/inherit' ); - const merge = require( 'PHET_CORE/merge' ); - const ModelViewTransform2 = require( 'PHETCOMMON/view/ModelViewTransform2' ); - const Node = require( 'SCENERY/nodes/Node' ); - const Orientation = require( 'PHET_CORE/Orientation' ); - const OrientationPair = require( 'AREA_MODEL_COMMON/common/model/OrientationPair' ); - const PartialProductLabelNode = require( 'AREA_MODEL_COMMON/common/view/PartialProductLabelNode' ); - const PartialProductsChoice = require( 'AREA_MODEL_COMMON/common/model/PartialProductsChoice' ); - const PoolableLayerNode = require( 'AREA_MODEL_COMMON/common/view/PoolableLayerNode' ); - const Property = require( 'AXON/Property' ); - const RangeLabelNode = require( 'AREA_MODEL_COMMON/common/view/RangeLabelNode' ); - const Rectangle = require( 'SCENERY/nodes/Rectangle' ); - const StringUtils = require( 'PHETCOMMON/util/StringUtils' ); - - // a11y strings - const eraseString = AreaModelCommonA11yStrings.erase.value; - const eraseDescriptionString = AreaModelCommonA11yStrings.eraseDescription.value; - const horizontalDimensionCapitalizedString = AreaModelCommonA11yStrings.horizontalDimensionCapitalized.value; - const onePartialProductFactorPatternString = AreaModelCommonA11yStrings.onePartialProductFactorPattern.value; - const onePartialProductPatternString = AreaModelCommonA11yStrings.onePartialProductPattern.value; - const productTimesPatternString = AreaModelCommonA11yStrings.productTimesPattern.value; - const threePartitionsSplitPatternString = AreaModelCommonA11yStrings.threePartitionsSplitPattern.value; - const twoPartialProductFactorsPatternString = AreaModelCommonA11yStrings.twoPartialProductFactorsPattern.value; - const twoPartialProductsPatternString = AreaModelCommonA11yStrings.twoPartialProductsPattern.value; - const twoPartitionsSplitPatternString = AreaModelCommonA11yStrings.twoPartitionsSplitPattern.value; - const verticalDimensionCapitalizedString = AreaModelCommonA11yStrings.verticalDimensionCapitalized.value; - /** - * @constructor - * @extends {Node} - * - * @param {AreaDisplay} areaDisplay - * @param {Property.} partialProductsChoiceProperty - * @param {Object} [options] - */ - function AreaDisplayNode( areaDisplay, partialProductsChoiceProperty, options ) { - options = merge( { +import DerivedProperty from '../../../../axon/js/DerivedProperty.js'; +import Property from '../../../../axon/js/Property.js'; +import Bounds2 from '../../../../dot/js/Bounds2.js'; +import inherit from '../../../../phet-core/js/inherit.js'; +import merge from '../../../../phet-core/js/merge.js'; +import Orientation from '../../../../phet-core/js/Orientation.js'; +import StringUtils from '../../../../phetcommon/js/util/StringUtils.js'; +import ModelViewTransform2 from '../../../../phetcommon/js/view/ModelViewTransform2.js'; +import EraserButton from '../../../../scenery-phet/js/buttons/EraserButton.js'; +import Node from '../../../../scenery/js/nodes/Node.js'; +import Rectangle from '../../../../scenery/js/nodes/Rectangle.js'; +import areaModelCommon from '../../areaModelCommon.js'; +import AreaModelCommonA11yStrings from '../../AreaModelCommonA11yStrings.js'; +import AreaModelCommonConstants from '../AreaModelCommonConstants.js'; +import OrientationPair from '../model/OrientationPair.js'; +import PartialProductsChoice from '../model/PartialProductsChoice.js'; +import AreaModelCommonColorProfile from './AreaModelCommonColorProfile.js'; +import PartialProductLabelNode from './PartialProductLabelNode.js'; +import PoolableLayerNode from './PoolableLayerNode.js'; +import RangeLabelNode from './RangeLabelNode.js'; + +// a11y strings +const eraseString = AreaModelCommonA11yStrings.erase.value; +const eraseDescriptionString = AreaModelCommonA11yStrings.eraseDescription.value; +const horizontalDimensionCapitalizedString = AreaModelCommonA11yStrings.horizontalDimensionCapitalized.value; +const onePartialProductFactorPatternString = AreaModelCommonA11yStrings.onePartialProductFactorPattern.value; +const onePartialProductPatternString = AreaModelCommonA11yStrings.onePartialProductPattern.value; +const productTimesPatternString = AreaModelCommonA11yStrings.productTimesPattern.value; +const threePartitionsSplitPatternString = AreaModelCommonA11yStrings.threePartitionsSplitPattern.value; +const twoPartialProductFactorsPatternString = AreaModelCommonA11yStrings.twoPartialProductFactorsPattern.value; +const twoPartialProductsPatternString = AreaModelCommonA11yStrings.twoPartialProductsPattern.value; +const twoPartitionsSplitPatternString = AreaModelCommonA11yStrings.twoPartitionsSplitPattern.value; +const verticalDimensionCapitalizedString = AreaModelCommonA11yStrings.verticalDimensionCapitalized.value; - // These do not change for a given AreaDisplayNode - allowExponents: false, - isProportional: false, - useLargeArea: false - }, options ); +/** + * @constructor + * @extends {Node} + * + * @param {AreaDisplay} areaDisplay + * @param {Property.} partialProductsChoiceProperty + * @param {Object} [options] + */ +function AreaDisplayNode( areaDisplay, partialProductsChoiceProperty, options ) { + options = merge( { - const self = this; + // These do not change for a given AreaDisplayNode + allowExponents: false, + isProportional: false, + useLargeArea: false + }, options ); - Node.call( this ); + const self = this; - // @public {AreaDisplay} - this.areaDisplay = areaDisplay; + Node.call( this ); - // @public {Node} - Layers (a11y) - this.accessibleParagraphNode = new Node( { - tagName: 'p' - } ); - this.areaLayer = new Node(); - this.labelLayer = new Node(); + // @public {AreaDisplay} + this.areaDisplay = areaDisplay; - this.addChild( this.accessibleParagraphNode ); - this.addChild( this.areaLayer ); - this.addChild( this.labelLayer ); + // @public {Node} - Layers (a11y) + this.accessibleParagraphNode = new Node( { + tagName: 'p' + } ); + this.areaLayer = new Node(); + this.labelLayer = new Node(); - // @public {number} - this.viewSize = options.useLargeArea ? AreaModelCommonConstants.LARGE_AREA_SIZE : AreaModelCommonConstants.AREA_SIZE; + this.addChild( this.accessibleParagraphNode ); + this.addChild( this.areaLayer ); + this.addChild( this.labelLayer ); - // A11y description for the partitions for each orientation - const accessiblePartitionNodes = OrientationPair.create( function( orientation ) { - const partitionLabel = new Node( { - tagName: 'span' - } ); - Property.multilink( [ - areaDisplay.partitionsProperties.get( orientation ), - areaDisplay.totalProperties.get( orientation ) - ], function( partitions, total ) { - partitions = partitions.filter( function( partition ) { - return partition.sizeProperty.value !== null && partition.visibleProperty.value === true; - } ); - if ( partitions.length < 2 || total === null ) { - partitionLabel.innerContent = ''; - } - else if ( partitions.length === 2 ) { - partitionLabel.innerContent = StringUtils.fillIn( twoPartitionsSplitPatternString, { - partition: orientation === Orientation.HORIZONTAL ? horizontalDimensionCapitalizedString : verticalDimensionCapitalizedString, - size: total.toRichString(), - size1: partitions[ 0 ].sizeProperty.value.toRichString( false ), - size2: partitions[ 1 ].sizeProperty.value.toRichString( false ) - } ); - } - else if ( partitions.length === 3 ) { - partitionLabel.innerContent = StringUtils.fillIn( threePartitionsSplitPatternString, { - partition: orientation === Orientation.HORIZONTAL ? horizontalDimensionCapitalizedString : verticalDimensionCapitalizedString, - size: total.toRichString(), - size1: partitions[ 0 ].sizeProperty.value.toRichString( false ), - size2: partitions[ 1 ].sizeProperty.value.toRichString( false ), - size3: partitions[ 2 ].sizeProperty.value.toRichString( false ) - } ); - } - else { - throw new Error( 'unexpected number of partitions for a11y' ); - } - } ); - return partitionLabel; - } ); - this.accessibleParagraphNode.addChild( accessiblePartitionNodes.vertical ); - this.accessibleParagraphNode.addChild( accessiblePartitionNodes.horizontal ); + // @public {number} + this.viewSize = options.useLargeArea ? AreaModelCommonConstants.LARGE_AREA_SIZE : AreaModelCommonConstants.AREA_SIZE; - // A11y description for the partial products - const accessiblePartialProductNode = new Node( { + // A11y description for the partitions for each orientation + const accessiblePartitionNodes = OrientationPair.create( function( orientation ) { + const partitionLabel = new Node( { tagName: 'span' } ); - let accessiblePartialMultilink = null; - areaDisplay.partitionedAreasProperty.link( function( partitionedAreas ) { - if ( accessiblePartialMultilink ) { - accessiblePartialMultilink.dispose(); + Property.multilink( [ + areaDisplay.partitionsProperties.get( orientation ), + areaDisplay.totalProperties.get( orientation ) + ], function( partitions, total ) { + partitions = partitions.filter( function( partition ) { + return partition.sizeProperty.value !== null && partition.visibleProperty.value === true; + } ); + if ( partitions.length < 2 || total === null ) { + partitionLabel.innerContent = ''; } - const properties = [ - partialProductsChoiceProperty - ].concat( partitionedAreas.map( function( partitionedArea ) { return partitionedArea.areaProperty; } ) ) - .concat( partitionedAreas.map( function( partitionedArea ) { return partitionedArea.visibleProperty; } ) ); - accessiblePartialMultilink = Property.multilink( properties, function() { - const activePartitionedAreas = areaDisplay.partitionedAreasProperty.value.filter( function( partitionedArea ) { - return partitionedArea.visibleProperty.value && - partitionedArea.areaProperty.value !== null && - partitionedArea.partitions.vertical.sizeProperty.value !== null && - partitionedArea.partitions.horizontal.sizeProperty.value !== null; + else if ( partitions.length === 2 ) { + partitionLabel.innerContent = StringUtils.fillIn( twoPartitionsSplitPatternString, { + partition: orientation === Orientation.HORIZONTAL ? horizontalDimensionCapitalizedString : verticalDimensionCapitalizedString, + size: total.toRichString(), + size1: partitions[ 0 ].sizeProperty.value.toRichString( false ), + size2: partitions[ 1 ].sizeProperty.value.toRichString( false ) } ); - const fillObject = {}; - let fillString; - if ( activePartitionedAreas.length > 2 || - activePartitionedAreas.length === 0 || - partialProductsChoiceProperty.value === PartialProductsChoice.HIDDEN ) { - accessiblePartialProductNode.innerContent = ''; - } - else if ( partialProductsChoiceProperty.value === PartialProductsChoice.PRODUCTS ) { - fillString = onePartialProductPatternString; - fillObject.first = activePartitionedAreas[ 0 ].areaProperty.value.toRichString( false ); + } + else if ( partitions.length === 3 ) { + partitionLabel.innerContent = StringUtils.fillIn( threePartitionsSplitPatternString, { + partition: orientation === Orientation.HORIZONTAL ? horizontalDimensionCapitalizedString : verticalDimensionCapitalizedString, + size: total.toRichString(), + size1: partitions[ 0 ].sizeProperty.value.toRichString( false ), + size2: partitions[ 1 ].sizeProperty.value.toRichString( false ), + size3: partitions[ 2 ].sizeProperty.value.toRichString( false ) + } ); + } + else { + throw new Error( 'unexpected number of partitions for a11y' ); + } + } ); + return partitionLabel; + } ); + this.accessibleParagraphNode.addChild( accessiblePartitionNodes.vertical ); + this.accessibleParagraphNode.addChild( accessiblePartitionNodes.horizontal ); - if ( activePartitionedAreas.length === 2 ) { - fillString = twoPartialProductsPatternString; - fillObject.second = activePartitionedAreas[ 1 ].areaProperty.value.toRichString( false ); - } + // A11y description for the partial products + const accessiblePartialProductNode = new Node( { + tagName: 'span' + } ); + let accessiblePartialMultilink = null; + areaDisplay.partitionedAreasProperty.link( function( partitionedAreas ) { + if ( accessiblePartialMultilink ) { + accessiblePartialMultilink.dispose(); + } + const properties = [ + partialProductsChoiceProperty + ].concat( partitionedAreas.map( function( partitionedArea ) { return partitionedArea.areaProperty; } ) ) + .concat( partitionedAreas.map( function( partitionedArea ) { return partitionedArea.visibleProperty; } ) ); + accessiblePartialMultilink = Property.multilink( properties, function() { + const activePartitionedAreas = areaDisplay.partitionedAreasProperty.value.filter( function( partitionedArea ) { + return partitionedArea.visibleProperty.value && + partitionedArea.areaProperty.value !== null && + partitionedArea.partitions.vertical.sizeProperty.value !== null && + partitionedArea.partitions.horizontal.sizeProperty.value !== null; + } ); + const fillObject = {}; + let fillString; + if ( activePartitionedAreas.length > 2 || + activePartitionedAreas.length === 0 || + partialProductsChoiceProperty.value === PartialProductsChoice.HIDDEN ) { + accessiblePartialProductNode.innerContent = ''; + } + else if ( partialProductsChoiceProperty.value === PartialProductsChoice.PRODUCTS ) { + fillString = onePartialProductPatternString; + fillObject.first = activePartitionedAreas[ 0 ].areaProperty.value.toRichString( false ); - accessiblePartialProductNode.innerContent = StringUtils.fillIn( fillString, fillObject ); + if ( activePartitionedAreas.length === 2 ) { + fillString = twoPartialProductsPatternString; + fillObject.second = activePartitionedAreas[ 1 ].areaProperty.value.toRichString( false ); } - else if ( partialProductsChoiceProperty.value === PartialProductsChoice.FACTORS ) { - fillString = onePartialProductFactorPatternString; - fillObject.first = StringUtils.fillIn( productTimesPatternString, { - left: activePartitionedAreas[ 0 ].partitions.vertical.sizeProperty.value.toRichString( false ), - right: activePartitionedAreas[ 0 ].partitions.horizontal.sizeProperty.value.toRichString( false ) - } ); - if ( activePartitionedAreas.length === 2 ) { - fillString = twoPartialProductFactorsPatternString; - fillObject.second = StringUtils.fillIn( productTimesPatternString, { - left: activePartitionedAreas[ 1 ].partitions.vertical.sizeProperty.value.toRichString( false ), - right: activePartitionedAreas[ 1 ].partitions.horizontal.sizeProperty.value.toRichString( false ) - } ); - } + accessiblePartialProductNode.innerContent = StringUtils.fillIn( fillString, fillObject ); + } + else if ( partialProductsChoiceProperty.value === PartialProductsChoice.FACTORS ) { + fillString = onePartialProductFactorPatternString; + fillObject.first = StringUtils.fillIn( productTimesPatternString, { + left: activePartitionedAreas[ 0 ].partitions.vertical.sizeProperty.value.toRichString( false ), + right: activePartitionedAreas[ 0 ].partitions.horizontal.sizeProperty.value.toRichString( false ) + } ); - accessiblePartialProductNode.innerContent = StringUtils.fillIn( fillString, fillObject ); - } - else { - throw new Error( 'unknown situation for a11y partial products' ); + if ( activePartitionedAreas.length === 2 ) { + fillString = twoPartialProductFactorsPatternString; + fillObject.second = StringUtils.fillIn( productTimesPatternString, { + left: activePartitionedAreas[ 1 ].partitions.vertical.sizeProperty.value.toRichString( false ), + right: activePartitionedAreas[ 1 ].partitions.horizontal.sizeProperty.value.toRichString( false ) + } ); } - } ); - } ); - this.accessibleParagraphNode.addChild( accessiblePartialProductNode ); - - const modelBoundsProperty = new DerivedProperty( [ areaDisplay.coordinateRangeMaxProperty ], function( coordinateRangeMax ) { - return new Bounds2( 0, 0, coordinateRangeMax, coordinateRangeMax ); - } ); - const viewBounds = new Bounds2( 0, 0, this.viewSize, this.viewSize ); - // @protected {Property.} - Maps from coordinate range values to view values. - this.modelViewTransformProperty = new DerivedProperty( [ modelBoundsProperty ], function( modelBounds ) { - return ModelViewTransform2.createRectangleMapping( modelBounds, viewBounds ); + accessiblePartialProductNode.innerContent = StringUtils.fillIn( fillString, fillObject ); + } + else { + throw new Error( 'unknown situation for a11y partial products' ); + } } ); + } ); + this.accessibleParagraphNode.addChild( accessiblePartialProductNode ); - // Dimension line views - Orientation.VALUES.forEach( function( orientation ) { - const colorProperty = self.areaDisplay.colorProperties.get( orientation ); - const termListProperty = self.areaDisplay.displayProperties.get( orientation ); - const tickLocationsProperty = new DerivedProperty( - [ areaDisplay.partitionBoundariesProperties.get( orientation ) ], - function( partitionBoundaries ) { - return partitionBoundaries.map( function( boundary ) { - return orientation.modelToView( self.modelViewTransformProperty.value, boundary ); - } ); - } ); - self.labelLayer.addChild( new RangeLabelNode( - termListProperty, - orientation, - tickLocationsProperty, - colorProperty, - options.isProportional - ) ); - } ); + const modelBoundsProperty = new DerivedProperty( [ areaDisplay.coordinateRangeMaxProperty ], function( coordinateRangeMax ) { + return new Bounds2( 0, 0, coordinateRangeMax, coordinateRangeMax ); + } ); + const viewBounds = new Bounds2( 0, 0, this.viewSize, this.viewSize ); - // @private {boolean} - Whether we need to update the labels. It's expensive, so we only do it at most once a frame. - this.productPositionLabelsDirty = true; - const invalidateProductLabels = function() { - self.productPositionLabelsDirty = true; - }; - - // @protected {Array.} - this.productLabels = []; - - // Create pooled partial product labels - this.labelLayer.addChild( new PoolableLayerNode( { - usedArray: this.productLabels, - updatedCallback: invalidateProductLabels, - arrayProperty: areaDisplay.partitionedAreasProperty, - createNode: function( partitionedArea ) { - return new PartialProductLabelNode( - partialProductsChoiceProperty, - new Property( partitionedArea ), - options.allowExponents - ); - }, - getItemProperty: function( productLabel ) { - return productLabel.partitionedAreaProperty; - } - } ) ); + // @protected {Property.} - Maps from coordinate range values to view values. + this.modelViewTransformProperty = new DerivedProperty( [ modelBoundsProperty ], function( modelBounds ) { + return ModelViewTransform2.createRectangleMapping( modelBounds, viewBounds ); + } ); - // Note this needs to be linked after the product labels are created, so the order dependency works - areaDisplay.allPartitionsProperty.link( function( newAllPartitions, oldAllPartitions ) { - oldAllPartitions && oldAllPartitions.forEach( function( partition ) { - partition.coordinateRangeProperty.unlink( invalidateProductLabels ); - } ); - newAllPartitions.forEach( function( partition ) { - partition.coordinateRangeProperty.lazyLink( invalidateProductLabels ); + // Dimension line views + Orientation.VALUES.forEach( function( orientation ) { + const colorProperty = self.areaDisplay.colorProperties.get( orientation ); + const termListProperty = self.areaDisplay.displayProperties.get( orientation ); + const tickLocationsProperty = new DerivedProperty( + [ areaDisplay.partitionBoundariesProperties.get( orientation ) ], + function( partitionBoundaries ) { + return partitionBoundaries.map( function( boundary ) { + return orientation.modelToView( self.modelViewTransformProperty.value, boundary ); + } ); } ); + self.labelLayer.addChild( new RangeLabelNode( + termListProperty, + orientation, + tickLocationsProperty, + colorProperty, + options.isProportional + ) ); + } ); - invalidateProductLabels(); + // @private {boolean} - Whether we need to update the labels. It's expensive, so we only do it at most once a frame. + this.productPositionLabelsDirty = true; + const invalidateProductLabels = function() { + self.productPositionLabelsDirty = true; + }; + + // @protected {Array.} + this.productLabels = []; + + // Create pooled partial product labels + this.labelLayer.addChild( new PoolableLayerNode( { + usedArray: this.productLabels, + updatedCallback: invalidateProductLabels, + arrayProperty: areaDisplay.partitionedAreasProperty, + createNode: function( partitionedArea ) { + return new PartialProductLabelNode( + partialProductsChoiceProperty, + new Property( partitionedArea ), + options.allowExponents + ); + }, + getItemProperty: function( productLabel ) { + return productLabel.partitionedAreaProperty; + } + } ) ); + + // Note this needs to be linked after the product labels are created, so the order dependency works + areaDisplay.allPartitionsProperty.link( function( newAllPartitions, oldAllPartitions ) { + oldAllPartitions && oldAllPartitions.forEach( function( partition ) { + partition.coordinateRangeProperty.unlink( invalidateProductLabels ); + } ); + newAllPartitions.forEach( function( partition ) { + partition.coordinateRangeProperty.lazyLink( invalidateProductLabels ); } ); - // Also invalidate our label positions when the label type changes. - // See https://github.com/phetsims/area-model-common/issues/109 - partialProductsChoiceProperty.lazyLink( invalidateProductLabels ); + invalidateProductLabels(); + } ); - // Do it once initially for proper layout at the start - this.positionProductLabels(); + // Also invalidate our label positions when the label type changes. + // See https://github.com/phetsims/area-model-common/issues/109 + partialProductsChoiceProperty.lazyLink( invalidateProductLabels ); - // @public {Node} - Exposed publicly for a11y, and used in subclasses - this.eraseButton = new EraserButton( { - listener: function() { - areaDisplay.areaProperty.value.erase(); - }, - center: options.isProportional - ? AreaModelCommonConstants.PROPORTIONAL_RANGE_OFFSET - : AreaModelCommonConstants.GENERIC_RANGE_OFFSET, - touchAreaXDilation: 8, - touchAreaYDilation: 8, - - // a11y - innerContent: eraseString, - descriptionContent: eraseDescriptionString - } ); + // Do it once initially for proper layout at the start + this.positionProductLabels(); - // @protected {Node} - this.backgroundNode = new Rectangle( 0, 0, this.viewSize, this.viewSize, { - fill: AreaModelCommonColorProfile.areaBackgroundProperty - } ); + // @public {Node} - Exposed publicly for a11y, and used in subclasses + this.eraseButton = new EraserButton( { + listener: function() { + areaDisplay.areaProperty.value.erase(); + }, + center: options.isProportional + ? AreaModelCommonConstants.PROPORTIONAL_RANGE_OFFSET + : AreaModelCommonConstants.GENERIC_RANGE_OFFSET, + touchAreaXDilation: 8, + touchAreaYDilation: 8, + + // a11y + innerContent: eraseString, + descriptionContent: eraseDescriptionString + } ); - // @protected {Node} - this.borderNode = new Rectangle( 0, 0, this.viewSize, this.viewSize, { - stroke: AreaModelCommonColorProfile.areaBorderProperty - } ); + // @protected {Node} + this.backgroundNode = new Rectangle( 0, 0, this.viewSize, this.viewSize, { + fill: AreaModelCommonColorProfile.areaBackgroundProperty + } ); - this.labelLayer.addChild( this.eraseButton ); - } + // @protected {Node} + this.borderNode = new Rectangle( 0, 0, this.viewSize, this.viewSize, { + stroke: AreaModelCommonColorProfile.areaBorderProperty + } ); - areaModelCommon.register( 'AreaDisplayNode', AreaDisplayNode ); + this.labelLayer.addChild( this.eraseButton ); +} - return inherit( Node, AreaDisplayNode, { - /** - * Updates expensive-to-update things only once a frame (for performance). - * @public - */ - update: function() { - if ( !this.productPositionLabelsDirty ) { return; } - this.productPositionLabelsDirty = false; +areaModelCommon.register( 'AreaDisplayNode', AreaDisplayNode ); - this.positionProductLabels(); - }, +export default inherit( Node, AreaDisplayNode, { + /** + * Updates expensive-to-update things only once a frame (for performance). + * @public + */ + update: function() { + if ( !this.productPositionLabelsDirty ) { return; } + this.productPositionLabelsDirty = false; - /** - * Positions all of the partial products labels. - * @protected - */ - positionProductLabels: function() { - throw new Error( 'abstract method' ); - } - } ); -} ); + this.positionProductLabels(); + }, + + /** + * Positions all of the partial products labels. + * @protected + */ + positionProductLabels: function() { + throw new Error( 'abstract method' ); + } +} ); \ No newline at end of file diff --git a/js/common/view/AreaModelCommonAccordionBox.js b/js/common/view/AreaModelCommonAccordionBox.js index 649397ea..32fc7405 100644 --- a/js/common/view/AreaModelCommonAccordionBox.js +++ b/js/common/view/AreaModelCommonAccordionBox.js @@ -7,55 +7,52 @@ * * @author Jonathan Olson */ -define( require => { - 'use strict'; - // modules - const AccordionBox = require( 'SUN/AccordionBox' ); - const areaModelCommon = require( 'AREA_MODEL_COMMON/areaModelCommon' ); - const AreaModelCommonColorProfile = require( 'AREA_MODEL_COMMON/common/view/AreaModelCommonColorProfile' ); - const AreaModelCommonConstants = require( 'AREA_MODEL_COMMON/common/AreaModelCommonConstants' ); - const inherit = require( 'PHET_CORE/inherit' ); - const merge = require( 'PHET_CORE/merge' ); - const Text = require( 'SCENERY/nodes/Text' ); +import inherit from '../../../../phet-core/js/inherit.js'; +import merge from '../../../../phet-core/js/merge.js'; +import Text from '../../../../scenery/js/nodes/Text.js'; +import AccordionBox from '../../../../sun/js/AccordionBox.js'; +import areaModelCommon from '../../areaModelCommon.js'; +import AreaModelCommonConstants from '../AreaModelCommonConstants.js'; +import AreaModelCommonColorProfile from './AreaModelCommonColorProfile.js'; - /** - * @constructor - * @extends {AccordionBox} - * - * @param {string} titleString - * @param {Property.} expandedProperty - * @param {Node} content - * @param {Object} [options] - */ - function AreaModelCommonAccordionBox( titleString, expandedProperty, content, options ) { - options = merge( { - titleNode: new Text( titleString, { - font: AreaModelCommonConstants.TITLE_FONT, - maxWidth: options.maxTitleWidth || 200 - } ), - expandedProperty: expandedProperty, - contentXMargin: 15, - contentYMargin: 12, - resize: true, - cornerRadius: AreaModelCommonConstants.PANEL_CORNER_RADIUS, - fill: AreaModelCommonColorProfile.panelBackgroundProperty, - stroke: AreaModelCommonColorProfile.panelBorderProperty, - titleAlignX: 'left', - titleXSpacing: 8, - buttonXMargin: 10, - buttonYMargin: 8, - expandCollapseButtonOptions: { - sideLength: 20, - touchAreaXDilation: 5, - touchAreaYDilation: 5 - } - }, options ); +/** + * @constructor + * @extends {AccordionBox} + * + * @param {string} titleString + * @param {Property.} expandedProperty + * @param {Node} content + * @param {Object} [options] + */ +function AreaModelCommonAccordionBox( titleString, expandedProperty, content, options ) { + options = merge( { + titleNode: new Text( titleString, { + font: AreaModelCommonConstants.TITLE_FONT, + maxWidth: options.maxTitleWidth || 200 + } ), + expandedProperty: expandedProperty, + contentXMargin: 15, + contentYMargin: 12, + resize: true, + cornerRadius: AreaModelCommonConstants.PANEL_CORNER_RADIUS, + fill: AreaModelCommonColorProfile.panelBackgroundProperty, + stroke: AreaModelCommonColorProfile.panelBorderProperty, + titleAlignX: 'left', + titleXSpacing: 8, + buttonXMargin: 10, + buttonYMargin: 8, + expandCollapseButtonOptions: { + sideLength: 20, + touchAreaXDilation: 5, + touchAreaYDilation: 5 + } + }, options ); - AccordionBox.call( this, content, options ); - } + AccordionBox.call( this, content, options ); +} - areaModelCommon.register( 'AreaModelCommonAccordionBox', AreaModelCommonAccordionBox ); +areaModelCommon.register( 'AreaModelCommonAccordionBox', AreaModelCommonAccordionBox ); - return inherit( AccordionBox, AreaModelCommonAccordionBox ); -} ); +inherit( AccordionBox, AreaModelCommonAccordionBox ); +export default AreaModelCommonAccordionBox; \ No newline at end of file diff --git a/js/common/view/AreaModelCommonColorProfile.js b/js/common/view/AreaModelCommonColorProfile.js index 30b032cb..0c4ad54d 100644 --- a/js/common/view/AreaModelCommonColorProfile.js +++ b/js/common/view/AreaModelCommonColorProfile.js @@ -5,162 +5,158 @@ * * @author Jonathan Olson */ -define( require => { - 'use strict'; - - // modules - const areaModelCommon = require( 'AREA_MODEL_COMMON/areaModelCommon' ); - const Color = require( 'SCENERY/util/Color' ); - const ColorProfile = require( 'SCENERY_PHET/ColorProfile' ); - const OrientationPair = require( 'AREA_MODEL_COMMON/common/model/OrientationPair' ); - const PhetColorScheme = require( 'SCENERY_PHET/PhetColorScheme' ); - - // Initial colors for each profile, by string key. Only profile currently is default (still helpful for making color - // tweaks with the top-level files) - const AreaModelCommonColorProfile = new ColorProfile( [ 'default' ], { - /*---------------------------------------------------------------------------* - * Common colors - *----------------------------------------------------------------------------*/ - - // Main background color for the sim - background: { default: new Color( 244, 252, 254 ) }, - - // Radio buttons for scene selection / area-model calculation / partial products - radioBorder: { default: new Color( 97, 200, 216 ) }, - radioBackground: { default: Color.WHITE }, - - // Things that look like panels (except for the keypad panel) - panelBorder: { default: new Color( 0x3, 0x3, 0x3 ) }, - panelBackground: { default: Color.WHITE }, - - // Main appearance - areaBackground: { default: Color.WHITE }, - areaBorder: { default: Color.BLACK }, - - // Partition line (stroke includes handle) - partitionLineBorder: { default: Color.BLACK }, - partitionLineStroke: { default: Color.BLACK }, - - // Calculation "base" colors - calculationActive: { default: Color.BLACK }, - calculationInactive: { default: new Color( 0xaaaaaa ) }, - - // Calculation next/previous arrows - calculationArrowUp: { default: Color.BLACK }, - calculationArrowDisabled: { default: new Color( 0xaaaaaa ) }, - // Calculation icon (in area-model-calculation panel) - calculationIconDark: { default: Color.BLACK }, - calculationIconLight: { default: new Color( 0xaaaaaa ) }, +import ColorProfile from '../../../../scenery-phet/js/ColorProfile.js'; +import PhetColorScheme from '../../../../scenery-phet/js/PhetColorScheme.js'; +import Color from '../../../../scenery/js/util/Color.js'; +import areaModelCommon from '../../areaModelCommon.js'; +import OrientationPair from '../model/OrientationPair.js'; - // Shown behind partial product labels - partialProductBackground: { default: new Color( 255, 255, 255, 0.75 ) }, - partialProductBorder: { default: new Color( 0, 0, 0, 0.2 ) }, +// Initial colors for each profile, by string key. Only profile currently is default (still helpful for making color +// tweaks with the top-level files) +const AreaModelCommonColorProfile = new ColorProfile( [ 'default' ], { + /*---------------------------------------------------------------------------* + * Common colors + *----------------------------------------------------------------------------*/ - selectionSeparator: { default: new Color( 0xaaaaaa ) }, + // Main background color for the sim + background: { default: new Color( 244, 252, 254 ) }, - /*---------------------------------------------------------------------------* - * Proportional colors - *----------------------------------------------------------------------------*/ + // Radio buttons for scene selection / area-model calculation / partial products + radioBorder: { default: new Color( 97, 200, 216 ) }, + radioBackground: { default: Color.WHITE }, - // Main "color" identity for proportional width/height - proportionalWidth: { default: new Color( 181, 45, 0 ) }, // red - proportionalHeight: { default: new Color( 0, 71, 253 ) }, // blue + // Things that look like panels (except for the keypad panel) + panelBorder: { default: new Color( 0x3, 0x3, 0x3 ) }, + panelBackground: { default: Color.WHITE }, - // Grid lines for within the area - gridLine: { default: new Color( 0xdd, 0xdd, 0xdd ) }, + // Main appearance + areaBackground: { default: Color.WHITE }, + areaBorder: { default: Color.BLACK }, - // The "active" part of the area (within the width/height selected) - proportionalActiveAreaBorder: { default: new Color( 0x66, 0x66, 0x66 ) }, - proportionalActiveAreaBackground: { default: new Color( 0, 0, 0, 0.1 ) }, + // Partition line (stroke includes handle) + partitionLineBorder: { default: Color.BLACK }, + partitionLineStroke: { default: Color.BLACK }, - // Drag handle to the lower-right of the proportional areas - proportionalDragHandleBorder: { default: new Color( 0x66, 0x66, 0x66 ) }, - proportionalDragHandleBackground: { default: new Color( 172, 201, 184 ) }, + // Calculation "base" colors + calculationActive: { default: Color.BLACK }, + calculationInactive: { default: new Color( 0xaaaaaa ) }, - // Tiles (proportional screens) - bigTile: { default: new Color( 255, 220, 120 ) }, - mediumTile: { default: new Color( 249, 244, 136 ) }, - smallTile: { default: new Color( 252, 250, 202 ) }, - semiTransparentSmallTile: { default: new Color( 243, 235, 43, 0.25 ) }, // blends onto white to look the same as smallTile - tileBorder: { default: new Color( 0xaaaaaa ) }, + // Calculation next/previous arrows + calculationArrowUp: { default: Color.BLACK }, + calculationArrowDisabled: { default: new Color( 0xaaaaaa ) }, - // Proportional icon colors - gridIcon: { default: new Color( 0x55, 0x55, 0x55 ) }, - tileIconStroke: { default: Color.BLACK }, - partitionLineIconBorder: { default: new Color( 0xaa, 0xaa, 0xaa ) }, - partitionLineIconBackground: { default: Color.WHITE }, - partitionLineIconLine: { default: Color.BLACK }, - partitionLineIconHandle: { default: new Color( 0x33, 0x33, 0x33 ) }, + // Calculation icon (in area-model-calculation panel) + calculationIconDark: { default: Color.BLACK }, + calculationIconLight: { default: new Color( 0xaaaaaa ) }, - // Color for the counting/numbering labels for each grid square - countingLabel: { default: Color.BLACK }, + // Shown behind partial product labels + partialProductBackground: { default: new Color( 255, 255, 255, 0.75 ) }, + partialProductBorder: { default: new Color( 0, 0, 0, 0.2 ) }, - /*---------------------------------------------------------------------------* - * Generic colors - *----------------------------------------------------------------------------*/ + selectionSeparator: { default: new Color( 0xaaaaaa ) }, - // Main "color" identity for generic width/height - genericWidth: { default: new Color( 0, 165, 83 ) }, // green - genericHeight: { default: new Color( 91, 42, 194 ) }, // purple + /*---------------------------------------------------------------------------* + * Proportional colors + *----------------------------------------------------------------------------*/ - // Edit button - editButtonBackground: { default: new Color( 241, 232, 0 ) }, + // Main "color" identity for proportional width/height + proportionalWidth: { default: new Color( 181, 45, 0 ) }, // red + proportionalHeight: { default: new Color( 0, 71, 253 ) }, // blue - // Edit readout - editActiveBackground: { default: new Color( 255, 255, 130 ) }, - editInactiveBackground: { default: Color.WHITE }, + // Grid lines for within the area + gridLine: { default: new Color( 0xdd, 0xdd, 0xdd ) }, - // Keypad panel - keypadPanelBorder: { default: new Color( 0x99, 0x99, 0x99 ) }, - keypadPanelBackground: { default: new Color( 230, 230, 230 ) }, + // The "active" part of the area (within the width/height selected) + proportionalActiveAreaBorder: { default: new Color( 0x66, 0x66, 0x66 ) }, + proportionalActiveAreaBackground: { default: new Color( 0, 0, 0, 0.1 ) }, - keypadReadoutBorder: { default: Color.BLACK }, - keypadReadoutBackground: { default: Color.WHITE }, + // Drag handle to the lower-right of the proportional areas + proportionalDragHandleBorder: { default: new Color( 0x66, 0x66, 0x66 ) }, + proportionalDragHandleBackground: { default: new Color( 172, 201, 184 ) }, - keypadEnterBackground: { default: new Color( 241, 232, 0 ) }, + // Tiles (proportional screens) + bigTile: { default: new Color( 255, 220, 120 ) }, + mediumTile: { default: new Color( 249, 244, 136 ) }, + smallTile: { default: new Color( 252, 250, 202 ) }, + semiTransparentSmallTile: { default: new Color( 243, 235, 43, 0.25 ) }, // blends onto white to look the same as smallTile + tileBorder: { default: new Color( 0xaaaaaa ) }, - // Area sign highlights - genericPositiveBackground: { default: new Color( 0xd4f3fe ) }, - genericNegativeBackground: { default: new Color( 0xe5a5ab ) }, + // Proportional icon colors + gridIcon: { default: new Color( 0x55, 0x55, 0x55 ) }, + tileIconStroke: { default: Color.BLACK }, + partitionLineIconBorder: { default: new Color( 0xaa, 0xaa, 0xaa ) }, + partitionLineIconBackground: { default: Color.WHITE }, + partitionLineIconLine: { default: Color.BLACK }, + partitionLineIconHandle: { default: new Color( 0x33, 0x33, 0x33 ) }, - // Layout grid icon color - layoutGrid: { default: Color.BLACK }, - layoutIconFill: { default: Color.WHITE }, - layoutHover: { default: new Color( 240, 240, 240 ) }, + // Color for the counting/numbering labels for each grid square + countingLabel: { default: Color.BLACK }, - /*---------------------------------------------------------------------------* - * Game colors - *----------------------------------------------------------------------------*/ + /*---------------------------------------------------------------------------* + * Generic colors + *----------------------------------------------------------------------------*/ - numbersIconBackground: { default: new Color( '#b26fac' ) }, - variablesIconBackground: { default: new Color( '#6f9fb2' ) }, + // Main "color" identity for generic width/height + genericWidth: { default: new Color( 0, 165, 83 ) }, // green + genericHeight: { default: new Color( 91, 42, 194 ) }, // purple - gameButtonBackground: { default: new Color( 241, 232, 0 ) }, + // Edit button + editButtonBackground: { default: new Color( 241, 232, 0 ) }, - // border/text highlights for editable values - errorStatus: { default: Color.RED }, - dirtyStatus: { default: new Color( '#3B97BA' ) }, + // Edit readout + editActiveBackground: { default: new Color( 255, 255, 130 ) }, + editInactiveBackground: { default: Color.WHITE }, - dynamicPartialProduct: { default: new Color( 128, 130, 133 ) }, - fixedPartialProduct: { default: Color.BLACK }, - totalEditable: { default: Color.BLACK }, + // Keypad panel + keypadPanelBorder: { default: new Color( 0x99, 0x99, 0x99 ) }, + keypadPanelBackground: { default: new Color( 230, 230, 230 ) }, - startOverButtonBaseColor: { default: PhetColorScheme.BUTTON_YELLOW } + keypadReadoutBorder: { default: Color.BLACK }, + keypadReadoutBackground: { default: Color.WHITE }, - } ); + keypadEnterBackground: { default: new Color( 241, 232, 0 ) }, - areaModelCommon.register( 'AreaModelCommonColorProfile', AreaModelCommonColorProfile ); + // Area sign highlights + genericPositiveBackground: { default: new Color( 0xd4f3fe ) }, + genericNegativeBackground: { default: new Color( 0xe5a5ab ) }, - // @public {OrientationPair.>} - AreaModelCommonColorProfile.proportionalColorProperties = new OrientationPair( - AreaModelCommonColorProfile.proportionalWidthProperty, - AreaModelCommonColorProfile.proportionalHeightProperty - ); - AreaModelCommonColorProfile.genericColorProperties = new OrientationPair( - AreaModelCommonColorProfile.genericWidthProperty, - AreaModelCommonColorProfile.genericHeightProperty - ); + // Layout grid icon color + layoutGrid: { default: Color.BLACK }, + layoutIconFill: { default: Color.WHITE }, + layoutHover: { default: new Color( 240, 240, 240 ) }, + + /*---------------------------------------------------------------------------* + * Game colors + *----------------------------------------------------------------------------*/ + + numbersIconBackground: { default: new Color( '#b26fac' ) }, + variablesIconBackground: { default: new Color( '#6f9fb2' ) }, + + gameButtonBackground: { default: new Color( 241, 232, 0 ) }, + + // border/text highlights for editable values + errorStatus: { default: Color.RED }, + dirtyStatus: { default: new Color( '#3B97BA' ) }, + + dynamicPartialProduct: { default: new Color( 128, 130, 133 ) }, + fixedPartialProduct: { default: Color.BLACK }, + totalEditable: { default: Color.BLACK }, + + startOverButtonBaseColor: { default: PhetColorScheme.BUTTON_YELLOW } - return AreaModelCommonColorProfile; } ); + +areaModelCommon.register( 'AreaModelCommonColorProfile', AreaModelCommonColorProfile ); + +// @public {OrientationPair.>} +AreaModelCommonColorProfile.proportionalColorProperties = new OrientationPair( + AreaModelCommonColorProfile.proportionalWidthProperty, + AreaModelCommonColorProfile.proportionalHeightProperty +); +AreaModelCommonColorProfile.genericColorProperties = new OrientationPair( + AreaModelCommonColorProfile.genericWidthProperty, + AreaModelCommonColorProfile.genericHeightProperty +); + +export default AreaModelCommonColorProfile; \ No newline at end of file diff --git a/js/common/view/AreaModelCommonRadioButtonGroup.js b/js/common/view/AreaModelCommonRadioButtonGroup.js index d1744451..46a18c1e 100644 --- a/js/common/view/AreaModelCommonRadioButtonGroup.js +++ b/js/common/view/AreaModelCommonRadioButtonGroup.js @@ -5,39 +5,36 @@ * * @author Jonathan Olson */ -define( require => { - 'use strict'; - // modules - const areaModelCommon = require( 'AREA_MODEL_COMMON/areaModelCommon' ); - const AreaModelCommonColorProfile = require( 'AREA_MODEL_COMMON/common/view/AreaModelCommonColorProfile' ); - const inherit = require( 'PHET_CORE/inherit' ); - const merge = require( 'PHET_CORE/merge' ); - const RadioButtonGroup = require( 'SUN/buttons/RadioButtonGroup' ); +import inherit from '../../../../phet-core/js/inherit.js'; +import merge from '../../../../phet-core/js/merge.js'; +import RadioButtonGroup from '../../../../sun/js/buttons/RadioButtonGroup.js'; +import areaModelCommon from '../../areaModelCommon.js'; +import AreaModelCommonColorProfile from './AreaModelCommonColorProfile.js'; - /** - * @constructor - * @extends {RadioButtonGroup} - * - * @param {Property.<*>} property - * @param {Array.} items - See RadioButtonGroup for more info - * @param {Object} [options] - */ - function AreaModelCommonRadioButtonGroup( property, items, options ) { - RadioButtonGroup.call( this, property, items, merge( { - orientation: 'horizontal', - buttonContentXMargin: 10, - buttonContentYMargin: 10, - selectedLineWidth: 2, - deselectedLineWidth: 1.5, - touchAreaXDilation: 6, - touchAreaYDilation: 6, - selectedStroke: AreaModelCommonColorProfile.radioBorderProperty, - baseColor: AreaModelCommonColorProfile.radioBackgroundProperty - }, options ) ); - } +/** + * @constructor + * @extends {RadioButtonGroup} + * + * @param {Property.<*>} property + * @param {Array.} items - See RadioButtonGroup for more info + * @param {Object} [options] + */ +function AreaModelCommonRadioButtonGroup( property, items, options ) { + RadioButtonGroup.call( this, property, items, merge( { + orientation: 'horizontal', + buttonContentXMargin: 10, + buttonContentYMargin: 10, + selectedLineWidth: 2, + deselectedLineWidth: 1.5, + touchAreaXDilation: 6, + touchAreaYDilation: 6, + selectedStroke: AreaModelCommonColorProfile.radioBorderProperty, + baseColor: AreaModelCommonColorProfile.radioBackgroundProperty + }, options ) ); +} - areaModelCommon.register( 'AreaModelCommonRadioButtonGroup', AreaModelCommonRadioButtonGroup ); +areaModelCommon.register( 'AreaModelCommonRadioButtonGroup', AreaModelCommonRadioButtonGroup ); - return inherit( RadioButtonGroup, AreaModelCommonRadioButtonGroup ); -} ); +inherit( RadioButtonGroup, AreaModelCommonRadioButtonGroup ); +export default AreaModelCommonRadioButtonGroup; \ No newline at end of file diff --git a/js/common/view/AreaScreenView.js b/js/common/view/AreaScreenView.js index 7db7405a..bbfc54eb 100644 --- a/js/common/view/AreaScreenView.js +++ b/js/common/view/AreaScreenView.js @@ -7,335 +7,331 @@ * * @author Jonathan Olson */ -define( require => { - 'use strict'; - - // modules - const AlignBox = require( 'SCENERY/nodes/AlignBox' ); - const AreaCalculationRadioButtonGroup = require( 'AREA_MODEL_COMMON/common/view/AreaCalculationRadioButtonGroup' ); - const areaModelCommon = require( 'AREA_MODEL_COMMON/areaModelCommon' ); - const AreaModelCommonA11yStrings = require( 'AREA_MODEL_COMMON/AreaModelCommonA11yStrings' ); - const AreaModelCommonAccordionBox = require( 'AREA_MODEL_COMMON/common/view/AreaModelCommonAccordionBox' ); - const AreaModelCommonColorProfile = require( 'AREA_MODEL_COMMON/common/view/AreaModelCommonColorProfile' ); - const AreaModelCommonConstants = require( 'AREA_MODEL_COMMON/common/AreaModelCommonConstants' ); - const AreaModelCommonGlobals = require( 'AREA_MODEL_COMMON/common/AreaModelCommonGlobals' ); - const AreaModelCommonModel = require( 'AREA_MODEL_COMMON/common/model/AreaModelCommonModel' ); - const Bounds2 = require( 'DOT/Bounds2' ); - const CalculationBox = require( 'AREA_MODEL_COMMON/proportional/view/CalculationBox' ); - const CalculationNode = require( 'AREA_MODEL_COMMON/common/view/CalculationNode' ); - const inherit = require( 'PHET_CORE/inherit' ); - const interleave = require( 'PHET_CORE/interleave' ); - const Line = require( 'SCENERY/nodes/Line' ); - const merge = require( 'PHET_CORE/merge' ); - const Panel = require( 'SUN/Panel' ); - const PartialProductRadioButtonGroup = require( 'AREA_MODEL_COMMON/common/view/PartialProductRadioButtonGroup' ); - const Property = require( 'AXON/Property' ); - const required = require( 'PHET_CORE/required' ); - const ResetAllButton = require( 'SCENERY_PHET/buttons/ResetAllButton' ); - const ScreenView = require( 'JOIST/ScreenView' ); - const Text = require( 'SCENERY/nodes/Text' ); - const TotalAreaNode = require( 'AREA_MODEL_COMMON/common/view/TotalAreaNode' ); - const VBox = require( 'SCENERY/nodes/VBox' ); - - // strings - const areaModelCalculationString = require( 'string!AREA_MODEL_COMMON/areaModelCalculation' ); - const dimensionsString = require( 'string!AREA_MODEL_COMMON/dimensions' ); - const factorsString = require( 'string!AREA_MODEL_COMMON/factors' ); - const partialProductsString = require( 'string!AREA_MODEL_COMMON/partialProducts' ); - const productString = require( 'string!AREA_MODEL_COMMON/product' ); - const totalAreaOfModelString = require( 'string!AREA_MODEL_COMMON/totalAreaOfModel' ); - - // a11y strings - const factorsBoxString = AreaModelCommonA11yStrings.factorsBox.value; - const factorsBoxDescriptionString = AreaModelCommonA11yStrings.factorsBoxDescription.value; - const productBoxString = AreaModelCommonA11yStrings.productBox.value; - const productBoxDescriptionString = AreaModelCommonA11yStrings.productBoxDescription.value; - /** - * @constructor - * @extends {ScreenView} - * - * @param {AreaModelCommonModel} model - * @param {Object} config - */ - function AreaScreenView( model, config ) { - config = merge( { +import Property from '../../../../axon/js/Property.js'; +import Bounds2 from '../../../../dot/js/Bounds2.js'; +import ScreenView from '../../../../joist/js/ScreenView.js'; +import inherit from '../../../../phet-core/js/inherit.js'; +import interleave from '../../../../phet-core/js/interleave.js'; +import merge from '../../../../phet-core/js/merge.js'; +import required from '../../../../phet-core/js/required.js'; +import ResetAllButton from '../../../../scenery-phet/js/buttons/ResetAllButton.js'; +import AlignBox from '../../../../scenery/js/nodes/AlignBox.js'; +import Line from '../../../../scenery/js/nodes/Line.js'; +import Text from '../../../../scenery/js/nodes/Text.js'; +import VBox from '../../../../scenery/js/nodes/VBox.js'; +import Panel from '../../../../sun/js/Panel.js'; +import areaModelCommonStrings from '../../area-model-common-strings.js'; +import areaModelCommon from '../../areaModelCommon.js'; +import AreaModelCommonA11yStrings from '../../AreaModelCommonA11yStrings.js'; +import CalculationBox from '../../proportional/view/CalculationBox.js'; +import AreaModelCommonConstants from '../AreaModelCommonConstants.js'; +import AreaModelCommonGlobals from '../AreaModelCommonGlobals.js'; +import AreaModelCommonModel from '../model/AreaModelCommonModel.js'; +import AreaCalculationRadioButtonGroup from './AreaCalculationRadioButtonGroup.js'; +import AreaModelCommonAccordionBox from './AreaModelCommonAccordionBox.js'; +import AreaModelCommonColorProfile from './AreaModelCommonColorProfile.js'; +import CalculationNode from './CalculationNode.js'; +import PartialProductRadioButtonGroup from './PartialProductRadioButtonGroup.js'; +import TotalAreaNode from './TotalAreaNode.js'; + +const areaModelCalculationString = areaModelCommonStrings.areaModelCalculation; +const dimensionsString = areaModelCommonStrings.dimensions; +const factorsString = areaModelCommonStrings.factors; +const partialProductsString = areaModelCommonStrings.partialProducts; +const productString = areaModelCommonStrings.product; +const totalAreaOfModelString = areaModelCommonStrings.totalAreaOfModel; + +// a11y strings +const factorsBoxString = AreaModelCommonA11yStrings.factorsBox.value; +const factorsBoxDescriptionString = AreaModelCommonA11yStrings.factorsBoxDescription.value; +const productBoxString = AreaModelCommonA11yStrings.productBox.value; +const productBoxDescriptionString = AreaModelCommonA11yStrings.productBoxDescription.value; - // {number} How many decimal places should be shown - decimalPlaces: required( config.decimalPlaces ), +/** + * @constructor + * @extends {ScreenView} + * + * @param {AreaModelCommonModel} model + * @param {Object} config + */ +function AreaScreenView( model, config ) { + config = merge( { - // {boolean} Whether we show options that let the user select the partial product style - showProductsSelection: true, + // {number} How many decimal places should be shown + decimalPlaces: required( config.decimalPlaces ), - // {boolean} (optional) - Whether we show options that let the user select the calculation style - showCalculationSelection: true, + // {boolean} Whether we show options that let the user select the partial product style + showProductsSelection: true, - // {boolean} (optional) - Selected area background and products box use a light-tile-colored background - useTileLikeBackground: false, + // {boolean} (optional) - Whether we show options that let the user select the calculation style + showCalculationSelection: true, - // {boolean} (optional) - Uses "product" and "factors" to be simpler and more multiplication-like - useSimplifiedNames: false, + // {boolean} (optional) - Selected area background and products box use a light-tile-colored background + useTileLikeBackground: false, - // {boolean} (optional) - If true, changes the location/size of the area to take up more space - useLargeArea: false, + // {boolean} (optional) - Uses "product" and "factors" to be simpler and more multiplication-like + useSimplifiedNames: false, - // {boolean} (optional) - If true, a simplified accordion box will be used for the calculation lines (instead of a panel). - // Notably, the accordion box does NOT support line-by-line appearance, and can be collapsed. - useCalculationBox: false - }, config ); + // {boolean} (optional) - If true, changes the location/size of the area to take up more space + useLargeArea: false, - assert && assert( model instanceof AreaModelCommonModel ); + // {boolean} (optional) - If true, a simplified accordion box will be used for the calculation lines (instead of a panel). + // Notably, the accordion box does NOT support line-by-line appearance, and can be collapsed. + useCalculationBox: false + }, config ); - ScreenView.call( this ); + assert && assert( model instanceof AreaModelCommonModel ); - // @protected {AreaModelCommonModel} - this.model = model; + ScreenView.call( this ); - // @protected {boolean} - this.useTileLikeBackground = config.useTileLikeBackground; - this.useLargeArea = config.useLargeArea; - this.showProductsSelection = config.showProductsSelection; - this.showCalculationSelection = config.showCalculationSelection; + // @protected {AreaModelCommonModel} + this.model = model; - // @protected {Node} - Exposed for a11y selection - this.productsSelectionPanel = this.createPanelContent( - partialProductsString, - AreaModelCommonGlobals.panelAlignGroup, - new PartialProductRadioButtonGroup( model, AreaModelCommonGlobals.selectionButtonAlignGroup ) - ); + // @protected {boolean} + this.useTileLikeBackground = config.useTileLikeBackground; + this.useLargeArea = config.useLargeArea; + this.showProductsSelection = config.showProductsSelection; + this.showCalculationSelection = config.showCalculationSelection; - // @public {Node} (a11y) - this.calculationSelectionPanel = this.createPanelContent( - areaModelCalculationString, - AreaModelCommonGlobals.panelAlignGroup, - new AreaCalculationRadioButtonGroup( model.areaCalculationChoiceProperty, AreaModelCommonGlobals.selectionButtonAlignGroup ) - ); - const selectionContent = new VBox( { - spacing: 15 - } ); - this.getSelectionNodesProperty().link( function( selectionNodes ) { - selectionContent.children = interleave( selectionNodes, function() { - return new Line( { - x2: AreaModelCommonConstants.PANEL_INTERIOR_MAX, - stroke: AreaModelCommonColorProfile.selectionSeparatorProperty - } ); - } ); - } ); + // @protected {Node} - Exposed for a11y selection + this.productsSelectionPanel = this.createPanelContent( + partialProductsString, + AreaModelCommonGlobals.panelAlignGroup, + new PartialProductRadioButtonGroup( model, AreaModelCommonGlobals.selectionButtonAlignGroup ) + ); - // @protected {Node} (a11y) - Shows radio button groups to select partial product / calculation / partition line options. - this.selectionPanel = new Panel( selectionContent, { - xMargin: 15, - yMargin: 10, - fill: AreaModelCommonColorProfile.panelBackgroundProperty, - stroke: AreaModelCommonColorProfile.panelBorderProperty, - cornerRadius: AreaModelCommonConstants.PANEL_CORNER_RADIUS + // @public {Node} (a11y) + this.calculationSelectionPanel = this.createPanelContent( + areaModelCalculationString, + AreaModelCommonGlobals.panelAlignGroup, + new AreaCalculationRadioButtonGroup( model.areaCalculationChoiceProperty, AreaModelCommonGlobals.selectionButtonAlignGroup ) + ); + const selectionContent = new VBox( { + spacing: 15 + } ); + this.getSelectionNodesProperty().link( function( selectionNodes ) { + selectionContent.children = interleave( selectionNodes, function() { + return new Line( { + x2: AreaModelCommonConstants.PANEL_INTERIOR_MAX, + stroke: AreaModelCommonColorProfile.selectionSeparatorProperty + } ); } ); + } ); - const factorsBoxContent = new AlignBox( this.createFactorsNode( model, config.decimalPlaces ), { - group: AreaModelCommonGlobals.panelAlignGroup, - xAlign: 'center' - } ); + // @protected {Node} (a11y) - Shows radio button groups to select partial product / calculation / partition line options. + this.selectionPanel = new Panel( selectionContent, { + xMargin: 15, + yMargin: 10, + fill: AreaModelCommonColorProfile.panelBackgroundProperty, + stroke: AreaModelCommonColorProfile.panelBorderProperty, + cornerRadius: AreaModelCommonConstants.PANEL_CORNER_RADIUS + } ); - // @protected {Node} (a11y) - Exposed for a11y order - this.factorsBox = new AreaModelCommonAccordionBox( - config.useSimplifiedNames ? factorsString : dimensionsString, - model.factorsBoxExpandedProperty, - factorsBoxContent, - { - // Cut some spacing from the exponent-enabled one, as it looks like way too much padding otherwise - contentYSpacing: model.allowExponents ? 5 : 8, - - // a11y - labelTagName: 'h3', - labelContent: factorsBoxString, - titleBarOptions: { - descriptionContent: factorsBoxDescriptionString - } - } ); + const factorsBoxContent = new AlignBox( this.createFactorsNode( model, config.decimalPlaces ), { + group: AreaModelCommonGlobals.panelAlignGroup, + xAlign: 'center' + } ); - const areaBoxContent = new AlignBox( new TotalAreaNode( - model.totalAreaProperty, - model.isProportional, - model.isProportional ? model.getMaximumAreaString() : '', - this.useTileLikeBackground - ), { - group: AreaModelCommonGlobals.panelAlignGroup, - xAlign: 'center' + // @protected {Node} (a11y) - Exposed for a11y order + this.factorsBox = new AreaModelCommonAccordionBox( + config.useSimplifiedNames ? factorsString : dimensionsString, + model.factorsBoxExpandedProperty, + factorsBoxContent, + { + // Cut some spacing from the exponent-enabled one, as it looks like way too much padding otherwise + contentYSpacing: model.allowExponents ? 5 : 8, + + // a11y + labelTagName: 'h3', + labelContent: factorsBoxString, + titleBarOptions: { + descriptionContent: factorsBoxDescriptionString + } } ); - // @protected {Node} (a11y) - this.areaBox = new AreaModelCommonAccordionBox( - config.useSimplifiedNames ? productString : totalAreaOfModelString, - model.areaBoxExpandedProperty, - areaBoxContent, { - // a11y - labelTagName: 'h3', - labelContent: productBoxString, - titleBarOptions: { - descriptionContent: productBoxDescriptionString - } + const areaBoxContent = new AlignBox( new TotalAreaNode( + model.totalAreaProperty, + model.isProportional, + model.isProportional ? model.getMaximumAreaString() : '', + this.useTileLikeBackground + ), { + group: AreaModelCommonGlobals.panelAlignGroup, + xAlign: 'center' + } ); + + // @protected {Node} (a11y) + this.areaBox = new AreaModelCommonAccordionBox( + config.useSimplifiedNames ? productString : totalAreaOfModelString, + model.areaBoxExpandedProperty, + areaBoxContent, { + // a11y + labelTagName: 'h3', + labelContent: productBoxString, + titleBarOptions: { + descriptionContent: productBoxDescriptionString } - ); + } + ); - // @protected {VBox} - Available for subtype positioning relative to this. - this.rightPanelContainer = new VBox( { - children: this.getRightAlignNodes(), - spacing: AreaModelCommonConstants.LAYOUT_SPACING - } ); - this.addChild( new AlignBox( this.rightPanelContainer, { - alignBounds: this.layoutBounds, - xAlign: 'right', - yAlign: 'top', - margin: AreaModelCommonConstants.LAYOUT_SPACING - } ) ); - - // @protected {Node|null} (a11y) - The calculation panel/box near the bottom of the screen - this.calculationNode = null; - if ( config.useCalculationBox ) { - const calculationTop = AreaModelCommonConstants.MAIN_AREA_OFFSET.y + + // @protected {VBox} - Available for subtype positioning relative to this. + this.rightPanelContainer = new VBox( { + children: this.getRightAlignNodes(), + spacing: AreaModelCommonConstants.LAYOUT_SPACING + } ); + this.addChild( new AlignBox( this.rightPanelContainer, { + alignBounds: this.layoutBounds, + xAlign: 'right', + yAlign: 'top', + margin: AreaModelCommonConstants.LAYOUT_SPACING + } ) ); + + // @protected {Node|null} (a11y) - The calculation panel/box near the bottom of the screen + this.calculationNode = null; + if ( config.useCalculationBox ) { + const calculationTop = AreaModelCommonConstants.MAIN_AREA_OFFSET.y + AreaModelCommonConstants.AREA_SIZE + AreaModelCommonConstants.LAYOUT_SPACING + 30; - const calculationBottom = this.layoutBounds.bottom - AreaModelCommonConstants.LAYOUT_SPACING; - const calculationBounds = new Bounds2( 0, 0, AreaModelCommonConstants.AREA_SIZE, calculationBottom - calculationTop ); - this.calculationNode = new CalculationBox( model, calculationBounds, { - x: AreaModelCommonConstants.MAIN_AREA_OFFSET.x, - y: calculationTop - } ); - } - else { - this.calculationNode = new CalculationNode( model ); - } - this.addChild( this.calculationNode ); - - // @protected {Node} (a11y) - Reset all button - this.resetAllButton = new ResetAllButton( { - listener: function() { - model.reset(); - }, - right: this.layoutBounds.right - AreaModelCommonConstants.LAYOUT_SPACING, - bottom: this.layoutBounds.bottom - AreaModelCommonConstants.LAYOUT_SPACING + const calculationBottom = this.layoutBounds.bottom - AreaModelCommonConstants.LAYOUT_SPACING; + const calculationBounds = new Bounds2( 0, 0, AreaModelCommonConstants.AREA_SIZE, calculationBottom - calculationTop ); + this.calculationNode = new CalculationBox( model, calculationBounds, { + x: AreaModelCommonConstants.MAIN_AREA_OFFSET.x, + y: calculationTop } ); - this.addChild( this.resetAllButton ); - - // @protected {AreaDisplayNode} - this.areaDisplayNode = this.createAreaDisplayNode( model ); - this.addChild( this.areaDisplayNode ); } + else { + this.calculationNode = new CalculationNode( model ); + } + this.addChild( this.calculationNode ); - areaModelCommon.register( 'AreaScreenView', AreaScreenView ); - - return inherit( ScreenView, AreaScreenView, { - /** - * Steps the view forward, updating things that only update once a frame. - * @public - * - * @param {number} dt - */ - step: function( dt ) { - - // No animation is happening in the view. This is for batching updates to happen only once a frame. - this.calculationNode.update(); - this.areaDisplayNode.update(); + // @protected {Node} (a11y) - Reset all button + this.resetAllButton = new ResetAllButton( { + listener: function() { + model.reset(); }, + right: this.layoutBounds.right - AreaModelCommonConstants.LAYOUT_SPACING, + bottom: this.layoutBounds.bottom - AreaModelCommonConstants.LAYOUT_SPACING + } ); + this.addChild( this.resetAllButton ); - /** - * The content on the right side varies depending on the subtype, so we provide overriding here. - * @protected - * - * @returns {Array.} - */ - getRightAlignNodes: function() { - const children = [ - this.factorsBox, - this.areaBox - ]; - if ( this.showCalculationSelection || this.showProductsSelection ) { - children.push( this.selectionPanel ); - } - return children; - }, + // @protected {AreaDisplayNode} + this.areaDisplayNode = this.createAreaDisplayNode( model ); + this.addChild( this.areaDisplayNode ); +} - /** - * The content embedded in the selection panel varies depending on the subtype, so we provide overriding here. - * @protected - * - * NOTE: We need to support the fact that this can change, so it's a Property - * - * @returns {Property.>} - */ - getSelectionNodesProperty: function() { - const selectionNodes = []; - if ( this.showProductsSelection ) { - selectionNodes.push( this.productsSelectionPanel ); - } - if ( this.showCalculationSelection ) { - selectionNodes.push( this.calculationSelectionPanel ); - } - return new Property( selectionNodes ); - }, +areaModelCommon.register( 'AreaScreenView', AreaScreenView ); - /** - * Creates a panel interior with the title left-aligned, and the content somewhat offset from the left with a - * guaranteed margin. - * @protected - * - * @param {string} titleString - * @param {AlignGroup} panelAlignGroup - * @param {Node} content - */ - createPanelContent: function( titleString, panelAlignGroup, content ) { - return new VBox( { - children: [ - new AlignBox( new Text( titleString, { - font: AreaModelCommonConstants.TITLE_FONT, - maxWidth: AreaModelCommonConstants.PANEL_INTERIOR_MAX - } ), { - group: panelAlignGroup, - xAlign: 'left' - } ), - new AlignBox( content, { - group: panelAlignGroup, - xAlign: 'center', - xMargin: 15 - } ) - ], - spacing: 10 - } ); - }, +export default inherit( ScreenView, AreaScreenView, { + /** + * Steps the view forward, updating things that only update once a frame. + * @public + * + * @param {number} dt + */ + step: function( dt ) { - /** - * Returns the ideal translation for instances of AreaDisplayNode on the main view. - * @protected - * - * @returns {Vector2} - */ - getDisplayTranslation: function() { - return this.layoutBounds.leftTop.plus( - this.useLargeArea ? AreaModelCommonConstants.LARGE_AREA_OFFSET : AreaModelCommonConstants.MAIN_AREA_OFFSET - ); - }, + // No animation is happening in the view. This is for batching updates to happen only once a frame. + this.calculationNode.update(); + this.areaDisplayNode.update(); + }, - /** - * Creates the "factors" (dimensions) content for the accordion box. - * @public - * - * @param {AreaModelCommonModel} model - * @param {number} decimalPlaces - * @returns {Node} - */ - createFactorsNode: function( model, decimalPlaces ) { - throw new Error( 'abstract method, should be implemented by subtype' ); - }, + /** + * The content on the right side varies depending on the subtype, so we provide overriding here. + * @protected + * + * @returns {Array.} + */ + getRightAlignNodes: function() { + const children = [ + this.factorsBox, + this.areaBox + ]; + if ( this.showCalculationSelection || this.showProductsSelection ) { + children.push( this.selectionPanel ); + } + return children; + }, - /** - * Creates the main area display view for the screen. - * @public - * - * @param {AreaModelCommonModel} model - * @returns {AreaDisplayNode} - */ - createAreaDisplayNode: function( model ) { - throw new Error( 'abstract method, should be implemented by subtype' ); + /** + * The content embedded in the selection panel varies depending on the subtype, so we provide overriding here. + * @protected + * + * NOTE: We need to support the fact that this can change, so it's a Property + * + * @returns {Property.>} + */ + getSelectionNodesProperty: function() { + const selectionNodes = []; + if ( this.showProductsSelection ) { + selectionNodes.push( this.productsSelectionPanel ); } - } ); -} ); + if ( this.showCalculationSelection ) { + selectionNodes.push( this.calculationSelectionPanel ); + } + return new Property( selectionNodes ); + }, + + /** + * Creates a panel interior with the title left-aligned, and the content somewhat offset from the left with a + * guaranteed margin. + * @protected + * + * @param {string} titleString + * @param {AlignGroup} panelAlignGroup + * @param {Node} content + */ + createPanelContent: function( titleString, panelAlignGroup, content ) { + return new VBox( { + children: [ + new AlignBox( new Text( titleString, { + font: AreaModelCommonConstants.TITLE_FONT, + maxWidth: AreaModelCommonConstants.PANEL_INTERIOR_MAX + } ), { + group: panelAlignGroup, + xAlign: 'left' + } ), + new AlignBox( content, { + group: panelAlignGroup, + xAlign: 'center', + xMargin: 15 + } ) + ], + spacing: 10 + } ); + }, + + /** + * Returns the ideal translation for instances of AreaDisplayNode on the main view. + * @protected + * + * @returns {Vector2} + */ + getDisplayTranslation: function() { + return this.layoutBounds.leftTop.plus( + this.useLargeArea ? AreaModelCommonConstants.LARGE_AREA_OFFSET : AreaModelCommonConstants.MAIN_AREA_OFFSET + ); + }, + + /** + * Creates the "factors" (dimensions) content for the accordion box. + * @public + * + * @param {AreaModelCommonModel} model + * @param {number} decimalPlaces + * @returns {Node} + */ + createFactorsNode: function( model, decimalPlaces ) { + throw new Error( 'abstract method, should be implemented by subtype' ); + }, + + /** + * Creates the main area display view for the screen. + * @public + * + * @param {AreaModelCommonModel} model + * @returns {AreaDisplayNode} + */ + createAreaDisplayNode: function( model ) { + throw new Error( 'abstract method, should be implemented by subtype' ); + } +} ); \ No newline at end of file diff --git a/js/common/view/CalculationNode.js b/js/common/view/CalculationNode.js index d4d7d7ee..0d430bdc 100644 --- a/js/common/view/CalculationNode.js +++ b/js/common/view/CalculationNode.js @@ -8,197 +8,193 @@ * * @author Jonathan Olson */ -define( require => { - 'use strict'; - - // modules - const AreaCalculationChoice = require( 'AREA_MODEL_COMMON/common/model/AreaCalculationChoice' ); - const areaModelCommon = require( 'AREA_MODEL_COMMON/areaModelCommon' ); - const AreaModelCommonColorProfile = require( 'AREA_MODEL_COMMON/common/view/AreaModelCommonColorProfile' ); - const AreaModelCommonConstants = require( 'AREA_MODEL_COMMON/common/AreaModelCommonConstants' ); - const CalculationLinesNode = require( 'AREA_MODEL_COMMON/common/view/calculation/CalculationLinesNode' ); - const FireListener = require( 'SCENERY/listeners/FireListener' ); - const inherit = require( 'PHET_CORE/inherit' ); - const Node = require( 'SCENERY/nodes/Node' ); - const Path = require( 'SCENERY/nodes/Path' ); - const Rectangle = require( 'SCENERY/nodes/Rectangle' ); - const Shape = require( 'KITE/Shape' ); - - // constants - const MAX_LINE_WIDTH = 680; // may need to be updated if the panel size is changed. - const LINE_BY_LINE_EXPANSION = 25; - const ARROW_SIZE = 18; - const ARROW_TOUCH_DILATION = 8; - /** - * @constructor - * @extends {Node} - * - * @param {AreaModelCommonModel} model - * @param {Object} [nodeOptions] - */ - function CalculationNode( model, nodeOptions ) { - - const self = this; - - Node.call( this ); - - // @private {CalculationLinesNode} - this.calculationLinesNode = new CalculationLinesNode( model ); - - const background = new Rectangle( { - cornerRadius: AreaModelCommonConstants.PANEL_CORNER_RADIUS, - fill: AreaModelCommonColorProfile.panelBackgroundProperty, - stroke: AreaModelCommonColorProfile.panelBorderProperty - } ); - this.addChild( background ); - - const previousShape = new Shape().moveTo( 0, 0 ) - .lineTo( ARROW_SIZE, 0 ) - .lineTo( ARROW_SIZE / 2, -ARROW_SIZE * 0.8 ) - .close(); - const previousArrow = new Path( previousShape, { - fill: AreaModelCommonColorProfile.calculationArrowUpProperty, - cursor: 'pointer' - } ); - previousArrow.mouseArea = previousArrow.localBounds; - previousArrow.touchArea = previousArrow.localBounds.dilated( ARROW_TOUCH_DILATION ); - const previousListener = new FireListener( { - fire: function() { - self.calculationLinesNode.moveToPreviousLine(); - } - } ); - previousArrow.addInputListener( previousListener ); - this.calculationLinesNode.previousEnabledProperty.link( function( enabled ) { - previousListener.interrupt(); - previousArrow.pickable = enabled; - previousArrow.fill = enabled - ? AreaModelCommonColorProfile.calculationArrowUpProperty - : AreaModelCommonColorProfile.calculationArrowDisabledProperty; - } ); - - this.addChild( previousArrow ); - const nextShape = new Shape().moveTo( 0, 0 ) - .lineTo( ARROW_SIZE, 0 ) - .lineTo( ARROW_SIZE / 2, ARROW_SIZE * 0.8 ) - .close(); - const nextArrow = new Path( nextShape, { - fill: AreaModelCommonColorProfile.calculationArrowUpProperty, - cursor: 'pointer' - } ); - nextArrow.mouseArea = nextArrow.localBounds; - nextArrow.touchArea = nextArrow.localBounds.dilated( ARROW_TOUCH_DILATION ); - this.addChild( nextArrow ); - const nextListener = new FireListener( { - fire: function() { - self.calculationLinesNode.moveToNextLine(); - } - } ); - nextArrow.addInputListener( nextListener ); - this.calculationLinesNode.nextEnabledProperty.link( function( enabled ) { - nextListener.interrupt(); - nextArrow.pickable = enabled; - nextArrow.fill = enabled - ? AreaModelCommonColorProfile.calculationArrowUpProperty - : AreaModelCommonColorProfile.calculationArrowDisabledProperty; - } ); - - model.areaCalculationChoiceProperty.link( function( choice ) { - previousArrow.visible = nextArrow.visible = choice === AreaCalculationChoice.LINE_BY_LINE; - } ); - - this.mutate( nodeOptions ); - - this.addChild( this.calculationLinesNode ); - - model.areaCalculationChoiceProperty.link( function( choice ) { - self.visible = choice !== AreaCalculationChoice.HIDDEN; - } ); - - function update() { - if ( !self.calculationLinesNode.calculationLinesProperty.value.length ) { - return; - } - - const isLineByLine = model.areaCalculationChoiceProperty.value === AreaCalculationChoice.LINE_BY_LINE; - let maxLineWidth = _.reduce( self.calculationLinesNode.calculationLinesProperty.value, function( max, line ) { - return Math.max( max, line.node.width ); - }, 0 ); - - // If we are LINE_BY_LINE, we won't have as much room because of the buttons - const availableLineWidth = MAX_LINE_WIDTH + ( isLineByLine ? 0 : LINE_BY_LINE_EXPANSION ); - - // Scale the calculation down if necessary, so we can fit within our MAX_LINE_WIDTH - self.calculationLinesNode.setScaleMagnitude( maxLineWidth > availableLineWidth ? ( availableLineWidth / maxLineWidth ) : 1 ); - - // If we scaled things down, our maxLineWidth will be the available width (we don't want our 'panel' larger) - maxLineWidth = Math.min( maxLineWidth, availableLineWidth ); - - let backgroundBounds = self.calculationLinesNode.bounds; - - // If we removed lines for the "line-by-line", make sure we take up enough room to not change size. - if ( backgroundBounds.width < maxLineWidth ) { - backgroundBounds = backgroundBounds.dilatedX( ( maxLineWidth - backgroundBounds.width ) / 2 ); - } - - // Add some space around the lines - backgroundBounds = backgroundBounds.dilated( 5 ); - - // Add some space for the next/previous buttons - if ( isLineByLine ) { - backgroundBounds.maxX += LINE_BY_LINE_EXPANSION; - } - - // Minimum width of the area size - if ( backgroundBounds.width < AreaModelCommonConstants.AREA_SIZE ) { - backgroundBounds = backgroundBounds.dilatedX( ( AreaModelCommonConstants.AREA_SIZE - backgroundBounds.width ) / 2 ); - } - - // Minimum height - if ( backgroundBounds.height < 120 ) { - backgroundBounds = backgroundBounds.dilatedY( ( 120 - backgroundBounds.height ) / 2 ); - } - - // If we support decimals, use a slightly larger minimum. - // See https://github.com/phetsims/area-model-decimals/issues/6 - const decimalsLineByLineWidth = AreaModelCommonConstants.AREA_SIZE + 40; - if ( isLineByLine && - model.currentAreaProperty.value.partitionSnapSize && - model.currentAreaProperty.value.partitionSnapSize < 1 && - backgroundBounds.width < decimalsLineByLineWidth ) { - backgroundBounds = backgroundBounds.dilatedX( ( decimalsLineByLineWidth - backgroundBounds.width ) / 2 ); - } - - background.rectBounds = backgroundBounds; - previousArrow.rightTop = backgroundBounds.eroded( 5 ).rightTop; - nextArrow.rightBottom = backgroundBounds.eroded( 5 ).rightBottom; - - // Empirically determined to be the best Y location given all of the combinations. Hard to determine with - // computation, so left here. - self.centerY = 543 - AreaModelCommonConstants.LAYOUT_SPACING; - - // First try to center - self.centerX = AreaModelCommonConstants.MAIN_AREA_OFFSET.x + AreaModelCommonConstants.AREA_SIZE / 2; - if ( self.left < AreaModelCommonConstants.LAYOUT_SPACING ) { - - // If that doesn't work, don't let it go out of bounds left-side. - self.left = AreaModelCommonConstants.LAYOUT_SPACING; - } - } +import Shape from '../../../../kite/js/Shape.js'; +import inherit from '../../../../phet-core/js/inherit.js'; +import FireListener from '../../../../scenery/js/listeners/FireListener.js'; +import Node from '../../../../scenery/js/nodes/Node.js'; +import Path from '../../../../scenery/js/nodes/Path.js'; +import Rectangle from '../../../../scenery/js/nodes/Rectangle.js'; +import areaModelCommon from '../../areaModelCommon.js'; +import AreaModelCommonConstants from '../AreaModelCommonConstants.js'; +import AreaCalculationChoice from '../model/AreaCalculationChoice.js'; +import AreaModelCommonColorProfile from './AreaModelCommonColorProfile.js'; +import CalculationLinesNode from './calculation/CalculationLinesNode.js'; + +// constants +const MAX_LINE_WIDTH = 680; // may need to be updated if the panel size is changed. +const LINE_BY_LINE_EXPANSION = 25; +const ARROW_SIZE = 18; +const ARROW_TOUCH_DILATION = 8; - this.calculationLinesNode.displayUpdatedEmitter.addListener( update ); - update(); - } +/** + * @constructor + * @extends {Node} + * + * @param {AreaModelCommonModel} model + * @param {Object} [nodeOptions] + */ +function CalculationNode( model, nodeOptions ) { - areaModelCommon.register( 'CalculationNode', CalculationNode ); + const self = this; - return inherit( Node, CalculationNode, { - /** - * Updates the calculation lines. - * @public - */ - update: function() { - this.calculationLinesNode.update(); + Node.call( this ); + + // @private {CalculationLinesNode} + this.calculationLinesNode = new CalculationLinesNode( model ); + + const background = new Rectangle( { + cornerRadius: AreaModelCommonConstants.PANEL_CORNER_RADIUS, + fill: AreaModelCommonColorProfile.panelBackgroundProperty, + stroke: AreaModelCommonColorProfile.panelBorderProperty + } ); + this.addChild( background ); + + const previousShape = new Shape().moveTo( 0, 0 ) + .lineTo( ARROW_SIZE, 0 ) + .lineTo( ARROW_SIZE / 2, -ARROW_SIZE * 0.8 ) + .close(); + const previousArrow = new Path( previousShape, { + fill: AreaModelCommonColorProfile.calculationArrowUpProperty, + cursor: 'pointer' + } ); + previousArrow.mouseArea = previousArrow.localBounds; + previousArrow.touchArea = previousArrow.localBounds.dilated( ARROW_TOUCH_DILATION ); + const previousListener = new FireListener( { + fire: function() { + self.calculationLinesNode.moveToPreviousLine(); } } ); -} ); + previousArrow.addInputListener( previousListener ); + this.calculationLinesNode.previousEnabledProperty.link( function( enabled ) { + previousListener.interrupt(); + previousArrow.pickable = enabled; + previousArrow.fill = enabled + ? AreaModelCommonColorProfile.calculationArrowUpProperty + : AreaModelCommonColorProfile.calculationArrowDisabledProperty; + } ); + + this.addChild( previousArrow ); + const nextShape = new Shape().moveTo( 0, 0 ) + .lineTo( ARROW_SIZE, 0 ) + .lineTo( ARROW_SIZE / 2, ARROW_SIZE * 0.8 ) + .close(); + const nextArrow = new Path( nextShape, { + fill: AreaModelCommonColorProfile.calculationArrowUpProperty, + cursor: 'pointer' + } ); + nextArrow.mouseArea = nextArrow.localBounds; + nextArrow.touchArea = nextArrow.localBounds.dilated( ARROW_TOUCH_DILATION ); + this.addChild( nextArrow ); + const nextListener = new FireListener( { + fire: function() { + self.calculationLinesNode.moveToNextLine(); + } + } ); + nextArrow.addInputListener( nextListener ); + this.calculationLinesNode.nextEnabledProperty.link( function( enabled ) { + nextListener.interrupt(); + nextArrow.pickable = enabled; + nextArrow.fill = enabled + ? AreaModelCommonColorProfile.calculationArrowUpProperty + : AreaModelCommonColorProfile.calculationArrowDisabledProperty; + } ); + + model.areaCalculationChoiceProperty.link( function( choice ) { + previousArrow.visible = nextArrow.visible = choice === AreaCalculationChoice.LINE_BY_LINE; + } ); + + this.mutate( nodeOptions ); + + this.addChild( this.calculationLinesNode ); + + model.areaCalculationChoiceProperty.link( function( choice ) { + self.visible = choice !== AreaCalculationChoice.HIDDEN; + } ); + + function update() { + if ( !self.calculationLinesNode.calculationLinesProperty.value.length ) { + return; + } + + const isLineByLine = model.areaCalculationChoiceProperty.value === AreaCalculationChoice.LINE_BY_LINE; + let maxLineWidth = _.reduce( self.calculationLinesNode.calculationLinesProperty.value, function( max, line ) { + return Math.max( max, line.node.width ); + }, 0 ); + + // If we are LINE_BY_LINE, we won't have as much room because of the buttons + const availableLineWidth = MAX_LINE_WIDTH + ( isLineByLine ? 0 : LINE_BY_LINE_EXPANSION ); + + // Scale the calculation down if necessary, so we can fit within our MAX_LINE_WIDTH + self.calculationLinesNode.setScaleMagnitude( maxLineWidth > availableLineWidth ? ( availableLineWidth / maxLineWidth ) : 1 ); + + // If we scaled things down, our maxLineWidth will be the available width (we don't want our 'panel' larger) + maxLineWidth = Math.min( maxLineWidth, availableLineWidth ); + + let backgroundBounds = self.calculationLinesNode.bounds; + + // If we removed lines for the "line-by-line", make sure we take up enough room to not change size. + if ( backgroundBounds.width < maxLineWidth ) { + backgroundBounds = backgroundBounds.dilatedX( ( maxLineWidth - backgroundBounds.width ) / 2 ); + } + + // Add some space around the lines + backgroundBounds = backgroundBounds.dilated( 5 ); + + // Add some space for the next/previous buttons + if ( isLineByLine ) { + backgroundBounds.maxX += LINE_BY_LINE_EXPANSION; + } + + // Minimum width of the area size + if ( backgroundBounds.width < AreaModelCommonConstants.AREA_SIZE ) { + backgroundBounds = backgroundBounds.dilatedX( ( AreaModelCommonConstants.AREA_SIZE - backgroundBounds.width ) / 2 ); + } + + // Minimum height + if ( backgroundBounds.height < 120 ) { + backgroundBounds = backgroundBounds.dilatedY( ( 120 - backgroundBounds.height ) / 2 ); + } + + // If we support decimals, use a slightly larger minimum. + // See https://github.com/phetsims/area-model-decimals/issues/6 + const decimalsLineByLineWidth = AreaModelCommonConstants.AREA_SIZE + 40; + if ( isLineByLine && + model.currentAreaProperty.value.partitionSnapSize && + model.currentAreaProperty.value.partitionSnapSize < 1 && + backgroundBounds.width < decimalsLineByLineWidth ) { + backgroundBounds = backgroundBounds.dilatedX( ( decimalsLineByLineWidth - backgroundBounds.width ) / 2 ); + } + + background.rectBounds = backgroundBounds; + previousArrow.rightTop = backgroundBounds.eroded( 5 ).rightTop; + nextArrow.rightBottom = backgroundBounds.eroded( 5 ).rightBottom; + + // Empirically determined to be the best Y location given all of the combinations. Hard to determine with + // computation, so left here. + self.centerY = 543 - AreaModelCommonConstants.LAYOUT_SPACING; + + // First try to center + self.centerX = AreaModelCommonConstants.MAIN_AREA_OFFSET.x + AreaModelCommonConstants.AREA_SIZE / 2; + if ( self.left < AreaModelCommonConstants.LAYOUT_SPACING ) { + + // If that doesn't work, don't let it go out of bounds left-side. + self.left = AreaModelCommonConstants.LAYOUT_SPACING; + } + } + + this.calculationLinesNode.displayUpdatedEmitter.addListener( update ); + update(); +} + +areaModelCommon.register( 'CalculationNode', CalculationNode ); + +export default inherit( Node, CalculationNode, { + /** + * Updates the calculation lines. + * @public + */ + update: function() { + this.calculationLinesNode.update(); + } +} ); \ No newline at end of file diff --git a/js/common/view/PartialProductLabelNode.js b/js/common/view/PartialProductLabelNode.js index b33e7b3e..bc9ff39a 100644 --- a/js/common/view/PartialProductLabelNode.js +++ b/js/common/view/PartialProductLabelNode.js @@ -7,174 +7,171 @@ * * @author Jonathan Olson */ -define( require => { - 'use strict'; - - // modules - const areaModelCommon = require( 'AREA_MODEL_COMMON/areaModelCommon' ); - const AreaModelCommonColorProfile = require( 'AREA_MODEL_COMMON/common/view/AreaModelCommonColorProfile' ); - const AreaModelCommonConstants = require( 'AREA_MODEL_COMMON/common/AreaModelCommonConstants' ); - const DerivedProperty = require( 'AXON/DerivedProperty' ); - const DynamicProperty = require( 'AXON/DynamicProperty' ); - const HBox = require( 'SCENERY/nodes/HBox' ); - const inherit = require( 'PHET_CORE/inherit' ); - const MathSymbols = require( 'SCENERY_PHET/MathSymbols' ); - const Node = require( 'SCENERY/nodes/Node' ); - const PartialProductsChoice = require( 'AREA_MODEL_COMMON/common/model/PartialProductsChoice' ); - const Property = require( 'AXON/Property' ); - const Rectangle = require( 'SCENERY/nodes/Rectangle' ); - const RichText = require( 'SCENERY/nodes/RichText' ); - const Text = require( 'SCENERY/nodes/Text' ); - const Vector2 = require( 'DOT/Vector2' ); - - /** - * @constructor - * @extends {Node} - * - * @param {Property.} partialProductsChoiceProperty - * @param {Property.} partitionedAreaProperty - * @param {boolean} allowExponents - Whether exponents (powers of x) are allowed - */ - function PartialProductLabelNode( partialProductsChoiceProperty, partitionedAreaProperty, allowExponents ) { - assert && assert( partialProductsChoiceProperty instanceof Property ); - assert && assert( typeof allowExponents === 'boolean' ); - - const self = this; - - Node.call( this ); - - // @public {Property.} - Exposed for improved positioning capability AND setting with pool - this.partitionedAreaProperty = partitionedAreaProperty; - - const areaProperty = new DynamicProperty( partitionedAreaProperty, { - derive: 'areaProperty' - } ); - const visibleProperty = new DynamicProperty( partitionedAreaProperty, { - derive: 'visibleProperty', - defaultValue: false - } ); - const horizontalSizeProperty = new DynamicProperty( partitionedAreaProperty, { - derive: 'partitions.horizontal.sizeProperty' - } ); - const verticalSizeProperty = new DynamicProperty( partitionedAreaProperty, { - derive: 'partitions.vertical.sizeProperty' - } ); - - const background = new Rectangle( { - cornerRadius: 3, - stroke: new DerivedProperty( - [ areaProperty, AreaModelCommonColorProfile.partialProductBorderProperty ], - function( area, color ) { - return ( area === null || area.coefficient === 0 ) ? 'transparent' : color; - } ), - fill: AreaModelCommonColorProfile.partialProductBackgroundProperty - } ); - this.addChild( background ); - - const box = new HBox( { - align: 'origin' - } ); - this.addChild( box ); - - // Visibility - Property.multilink( [ partialProductsChoiceProperty, visibleProperty ], function( choice, areaVisible ) { - self.visible = areaVisible && ( choice !== PartialProductsChoice.HIDDEN ); - } ); - - // RichTexts (we reuse the same instances to prevent GC and cpu cost) - const productRichText = new RichText( '', { - font: AreaModelCommonConstants.PARTIAL_PRODUCT_FONT - } ); - const factorsTextOptions = { - font: AreaModelCommonConstants.PARTIAL_FACTOR_FONT - }; - const horizontalRichText = new RichText( '', factorsTextOptions ); - const verticalRichText = new RichText( '', factorsTextOptions ); - - const rectangleSize = allowExponents ? 12 : 14; - - // Shifting the rectangles down, so we don't incur a large performance penalty for size-testing things - const rectangleExponentPadding = allowExponents ? 1.3 : 0; - const rectangleCenterY = new Text( ' ', factorsTextOptions ).centerY - rectangleSize / 2 + rectangleExponentPadding; - const horizontalRectangle = new Rectangle( 0, rectangleCenterY, rectangleSize, rectangleSize, { - stroke: 'black', - lineWidth: 0.7 - } ); - const verticalRectangle = new Rectangle( 0, rectangleCenterY, rectangleSize, rectangleSize, { - stroke: 'black', - lineWidth: 0.7 - } ); - if ( allowExponents ) { - const exponentPadding = 2; - horizontalRectangle.localBounds = horizontalRectangle.localBounds.dilatedX( exponentPadding ); - verticalRectangle.localBounds = verticalRectangle.localBounds.dilatedX( exponentPadding ); - } - - // Persistent text nodes (for performance) - const leftParenNode = new Text( '(', factorsTextOptions ); - const middleParensNode = new Text( ')(', factorsTextOptions ); - const rightParenNode = new Text( ')', factorsTextOptions ); - const timesNode = new Text( MathSymbols.TIMES, factorsTextOptions ); - - // Text/alignment - Property.multilink( - [ horizontalSizeProperty, verticalSizeProperty, partialProductsChoiceProperty ], - function( horizontalSize, verticalSize, choice ) { - let children; - - // Hidden - if ( choice === PartialProductsChoice.HIDDEN ) { - children = []; - } - // Product - else if ( choice === PartialProductsChoice.PRODUCTS ) { - productRichText.text = ( horizontalSize === null || verticalSize === null ) - ? '?' - : horizontalSize.times( verticalSize ).toRichString( false ); - children = [ productRichText ]; - } +import DerivedProperty from '../../../../axon/js/DerivedProperty.js'; +import DynamicProperty from '../../../../axon/js/DynamicProperty.js'; +import Property from '../../../../axon/js/Property.js'; +import Vector2 from '../../../../dot/js/Vector2.js'; +import inherit from '../../../../phet-core/js/inherit.js'; +import MathSymbols from '../../../../scenery-phet/js/MathSymbols.js'; +import HBox from '../../../../scenery/js/nodes/HBox.js'; +import Node from '../../../../scenery/js/nodes/Node.js'; +import Rectangle from '../../../../scenery/js/nodes/Rectangle.js'; +import RichText from '../../../../scenery/js/nodes/RichText.js'; +import Text from '../../../../scenery/js/nodes/Text.js'; +import areaModelCommon from '../../areaModelCommon.js'; +import AreaModelCommonConstants from '../AreaModelCommonConstants.js'; +import PartialProductsChoice from '../model/PartialProductsChoice.js'; +import AreaModelCommonColorProfile from './AreaModelCommonColorProfile.js'; - // Factors - else { +/** + * @constructor + * @extends {Node} + * + * @param {Property.} partialProductsChoiceProperty + * @param {Property.} partitionedAreaProperty + * @param {boolean} allowExponents - Whether exponents (powers of x) are allowed + */ +function PartialProductLabelNode( partialProductsChoiceProperty, partitionedAreaProperty, allowExponents ) { + assert && assert( partialProductsChoiceProperty instanceof Property ); + assert && assert( typeof allowExponents === 'boolean' ); + + const self = this; + + Node.call( this ); + + // @public {Property.} - Exposed for improved positioning capability AND setting with pool + this.partitionedAreaProperty = partitionedAreaProperty; + + const areaProperty = new DynamicProperty( partitionedAreaProperty, { + derive: 'areaProperty' + } ); + const visibleProperty = new DynamicProperty( partitionedAreaProperty, { + derive: 'visibleProperty', + defaultValue: false + } ); + const horizontalSizeProperty = new DynamicProperty( partitionedAreaProperty, { + derive: 'partitions.horizontal.sizeProperty' + } ); + const verticalSizeProperty = new DynamicProperty( partitionedAreaProperty, { + derive: 'partitions.vertical.sizeProperty' + } ); + + const background = new Rectangle( { + cornerRadius: 3, + stroke: new DerivedProperty( + [ areaProperty, AreaModelCommonColorProfile.partialProductBorderProperty ], + function( area, color ) { + return ( area === null || area.coefficient === 0 ) ? 'transparent' : color; + } ), + fill: AreaModelCommonColorProfile.partialProductBackgroundProperty + } ); + this.addChild( background ); + + const box = new HBox( { + align: 'origin' + } ); + this.addChild( box ); + + // Visibility + Property.multilink( [ partialProductsChoiceProperty, visibleProperty ], function( choice, areaVisible ) { + self.visible = areaVisible && ( choice !== PartialProductsChoice.HIDDEN ); + } ); + + // RichTexts (we reuse the same instances to prevent GC and cpu cost) + const productRichText = new RichText( '', { + font: AreaModelCommonConstants.PARTIAL_PRODUCT_FONT + } ); + const factorsTextOptions = { + font: AreaModelCommonConstants.PARTIAL_FACTOR_FONT + }; + const horizontalRichText = new RichText( '', factorsTextOptions ); + const verticalRichText = new RichText( '', factorsTextOptions ); + + const rectangleSize = allowExponents ? 12 : 14; + + // Shifting the rectangles down, so we don't incur a large performance penalty for size-testing things + const rectangleExponentPadding = allowExponents ? 1.3 : 0; + const rectangleCenterY = new Text( ' ', factorsTextOptions ).centerY - rectangleSize / 2 + rectangleExponentPadding; + const horizontalRectangle = new Rectangle( 0, rectangleCenterY, rectangleSize, rectangleSize, { + stroke: 'black', + lineWidth: 0.7 + } ); + const verticalRectangle = new Rectangle( 0, rectangleCenterY, rectangleSize, rectangleSize, { + stroke: 'black', + lineWidth: 0.7 + } ); + if ( allowExponents ) { + const exponentPadding = 2; + horizontalRectangle.localBounds = horizontalRectangle.localBounds.dilatedX( exponentPadding ); + verticalRectangle.localBounds = verticalRectangle.localBounds.dilatedX( exponentPadding ); + } - const horizontalNode = horizontalSize + // Persistent text nodes (for performance) + const leftParenNode = new Text( '(', factorsTextOptions ); + const middleParensNode = new Text( ')(', factorsTextOptions ); + const rightParenNode = new Text( ')', factorsTextOptions ); + const timesNode = new Text( MathSymbols.TIMES, factorsTextOptions ); + + // Text/alignment + Property.multilink( + [ horizontalSizeProperty, verticalSizeProperty, partialProductsChoiceProperty ], + function( horizontalSize, verticalSize, choice ) { + let children; + + // Hidden + if ( choice === PartialProductsChoice.HIDDEN ) { + children = []; + } + + // Product + else if ( choice === PartialProductsChoice.PRODUCTS ) { + productRichText.text = ( horizontalSize === null || verticalSize === null ) + ? '?' + : horizontalSize.times( verticalSize ).toRichString( false ); + children = [ productRichText ]; + } + + // Factors + else { + + const horizontalNode = horizontalSize ? horizontalRichText.setText( horizontalSize.toRichString( false ) ) : horizontalRectangle; - const verticalNode = verticalSize + const verticalNode = verticalSize ? verticalRichText.setText( verticalSize.toRichString( false ) ) : verticalRectangle; - if ( allowExponents ) { - box.spacing = 0; - children = [ - leftParenNode, - verticalNode, - middleParensNode, - horizontalNode, - rightParenNode - ]; - } - else { - box.spacing = 2; - children = [ - verticalNode, - timesNode, - horizontalNode - ]; - } + if ( allowExponents ) { + box.spacing = 0; + children = [ + leftParenNode, + verticalNode, + middleParensNode, + horizontalNode, + rightParenNode + ]; + } + else { + box.spacing = 2; + children = [ + verticalNode, + timesNode, + horizontalNode + ]; } + } - box.children = children; + box.children = children; - if ( isFinite( box.width ) ) { - box.center = Vector2.ZERO; - background.rectBounds = box.bounds.dilatedXY( 4, 2 ); - } - } ); - } + if ( isFinite( box.width ) ) { + box.center = Vector2.ZERO; + background.rectBounds = box.bounds.dilatedXY( 4, 2 ); + } + } ); +} - areaModelCommon.register( 'PartialProductLabelNode', PartialProductLabelNode ); +areaModelCommon.register( 'PartialProductLabelNode', PartialProductLabelNode ); - return inherit( Node, PartialProductLabelNode ); -} ); +inherit( Node, PartialProductLabelNode ); +export default PartialProductLabelNode; \ No newline at end of file diff --git a/js/common/view/PartialProductRadioButtonGroup.js b/js/common/view/PartialProductRadioButtonGroup.js index 6563af18..b5680f50 100644 --- a/js/common/view/PartialProductRadioButtonGroup.js +++ b/js/common/view/PartialProductRadioButtonGroup.js @@ -7,137 +7,133 @@ * * @author Jonathan Olson */ -define( require => { - 'use strict'; - // modules - const AlignBox = require( 'SCENERY/nodes/AlignBox' ); - const AlignGroup = require( 'SCENERY/nodes/AlignGroup' ); - const areaModelCommon = require( 'AREA_MODEL_COMMON/areaModelCommon' ); - const AreaModelCommonA11yStrings = require( 'AREA_MODEL_COMMON/AreaModelCommonA11yStrings' ); - const AreaModelCommonConstants = require( 'AREA_MODEL_COMMON/common/AreaModelCommonConstants' ); - const AreaModelCommonRadioButtonGroup = require( 'AREA_MODEL_COMMON/common/view/AreaModelCommonRadioButtonGroup' ); - const DerivedProperty = require( 'AXON/DerivedProperty' ); - const FontAwesomeNode = require( 'SUN/FontAwesomeNode' ); - const HBox = require( 'SCENERY/nodes/HBox' ); - const inherit = require( 'PHET_CORE/inherit' ); - const MathSymbols = require( 'SCENERY_PHET/MathSymbols' ); - const Node = require( 'SCENERY/nodes/Node' ); - const Orientation = require( 'PHET_CORE/Orientation' ); - const OrientationPair = require( 'AREA_MODEL_COMMON/common/model/OrientationPair' ); - const PartialProductsChoice = require( 'AREA_MODEL_COMMON/common/model/PartialProductsChoice' ); - const Text = require( 'SCENERY/nodes/Text' ); +import DerivedProperty from '../../../../axon/js/DerivedProperty.js'; +import inherit from '../../../../phet-core/js/inherit.js'; +import Orientation from '../../../../phet-core/js/Orientation.js'; +import MathSymbols from '../../../../scenery-phet/js/MathSymbols.js'; +import AlignBox from '../../../../scenery/js/nodes/AlignBox.js'; +import AlignGroup from '../../../../scenery/js/nodes/AlignGroup.js'; +import HBox from '../../../../scenery/js/nodes/HBox.js'; +import Node from '../../../../scenery/js/nodes/Node.js'; +import Text from '../../../../scenery/js/nodes/Text.js'; +import FontAwesomeNode from '../../../../sun/js/FontAwesomeNode.js'; +import areaModelCommon from '../../areaModelCommon.js'; +import AreaModelCommonA11yStrings from '../../AreaModelCommonA11yStrings.js'; +import AreaModelCommonConstants from '../AreaModelCommonConstants.js'; +import OrientationPair from '../model/OrientationPair.js'; +import PartialProductsChoice from '../model/PartialProductsChoice.js'; +import AreaModelCommonRadioButtonGroup from './AreaModelCommonRadioButtonGroup.js'; - // a11y strings - const hidePartialProductsString = AreaModelCommonA11yStrings.hidePartialProducts.value; - const showPartialProductsString = AreaModelCommonA11yStrings.showPartialProducts.value; - const showPartialProductFactorsString = AreaModelCommonA11yStrings.showPartialProductFactors.value; +// a11y strings +const hidePartialProductsString = AreaModelCommonA11yStrings.hidePartialProducts.value; +const showPartialProductsString = AreaModelCommonA11yStrings.showPartialProducts.value; +const showPartialProductFactorsString = AreaModelCommonA11yStrings.showPartialProductFactors.value; - /** - * @constructor - * @extends {AreaModelCommonRadioButtonGroup} - * - * @param {AreaModelCommonModel} model - * @param {AlignGroup} selectionButtonAlignGroup - */ - function PartialProductRadioButtonGroup( model, selectionButtonAlignGroup ) { +/** + * @constructor + * @extends {AreaModelCommonRadioButtonGroup} + * + * @param {AreaModelCommonModel} model + * @param {AlignGroup} selectionButtonAlignGroup + */ +function PartialProductRadioButtonGroup( model, selectionButtonAlignGroup ) { - // hardcoded strings since they shouldn't be translatable - const templateLabels = OrientationPair.create( function( orientation ) { - return new Text( orientation === Orientation.HORIZONTAL ? 'b' : 'a', { - font: AreaModelCommonConstants.SYMBOL_FONT, - fill: new DerivedProperty( - [ model.partialProductsChoiceProperty, model.colorProperties.get( orientation ) ], - function( value, color ) { - return value === PartialProductsChoice.FACTORS ? color : 'black'; - } ) - } ); + // hardcoded strings since they shouldn't be translatable + const templateLabels = OrientationPair.create( function( orientation ) { + return new Text( orientation === Orientation.HORIZONTAL ? 'b' : 'a', { + font: AreaModelCommonConstants.SYMBOL_FONT, + fill: new DerivedProperty( + [ model.partialProductsChoiceProperty, model.colorProperties.get( orientation ) ], + function( value, color ) { + return value === PartialProductsChoice.FACTORS ? color : 'black'; + } ) } ); + } ); - // Both are built here so it is a consistent size across screens. - const iconGroup = new AlignGroup(); - const exponentsIcon = new AlignBox( PartialProductRadioButtonGroup.createExponentIcon( templateLabels ), { - group: iconGroup - } ); - const noExponentsIcon = new AlignBox( PartialProductRadioButtonGroup.createNonExponentIcon( templateLabels ), { - group: iconGroup - } ); + // Both are built here so it is a consistent size across screens. + const iconGroup = new AlignGroup(); + const exponentsIcon = new AlignBox( PartialProductRadioButtonGroup.createExponentIcon( templateLabels ), { + group: iconGroup + } ); + const noExponentsIcon = new AlignBox( PartialProductRadioButtonGroup.createNonExponentIcon( templateLabels ), { + group: iconGroup + } ); - AreaModelCommonRadioButtonGroup.call( this, model.partialProductsChoiceProperty, [ - { - value: PartialProductsChoice.HIDDEN, - node: new AlignBox( new FontAwesomeNode( 'eye_close', { scale: 0.8 } ), { group: selectionButtonAlignGroup } ), + AreaModelCommonRadioButtonGroup.call( this, model.partialProductsChoiceProperty, [ + { + value: PartialProductsChoice.HIDDEN, + node: new AlignBox( new FontAwesomeNode( 'eye_close', { scale: 0.8 } ), { group: selectionButtonAlignGroup } ), - // a11y - labelContent: hidePartialProductsString - }, - { - value: PartialProductsChoice.PRODUCTS, + // a11y + labelContent: hidePartialProductsString + }, + { + value: PartialProductsChoice.PRODUCTS, - // Hardcoded 'A' string since we don't want it to be translatable - node: new AlignBox( new Text( 'A', { font: AreaModelCommonConstants.SYMBOL_FONT } ), { group: selectionButtonAlignGroup } ), + // Hardcoded 'A' string since we don't want it to be translatable + node: new AlignBox( new Text( 'A', { font: AreaModelCommonConstants.SYMBOL_FONT } ), { group: selectionButtonAlignGroup } ), - // a11y - labelContent: showPartialProductsString - }, - { - value: PartialProductsChoice.FACTORS, - node: new AlignBox( model.allowExponents ? exponentsIcon : noExponentsIcon, { group: selectionButtonAlignGroup } ), + // a11y + labelContent: showPartialProductsString + }, + { + value: PartialProductsChoice.FACTORS, + node: new AlignBox( model.allowExponents ? exponentsIcon : noExponentsIcon, { group: selectionButtonAlignGroup } ), - // a11y - labelContent: showPartialProductFactorsString - } - ] ); - } + // a11y + labelContent: showPartialProductFactorsString + } + ] ); +} - areaModelCommon.register( 'PartialProductRadioButtonGroup', PartialProductRadioButtonGroup ); +areaModelCommon.register( 'PartialProductRadioButtonGroup', PartialProductRadioButtonGroup ); - return inherit( AreaModelCommonRadioButtonGroup, PartialProductRadioButtonGroup, {}, { - /** - * Creates an 'exponents-allowed' icon based on a pair of nodes. - * @private - * - * @param {OrientationPair.} nodes - * @returns {Node} - */ - createExponentIcon: function( nodes ) { - return new HBox( { - children: [ - new Text( '(', { - font: AreaModelCommonConstants.SYMBOL_FONT - } ), - new Node( { children: [ nodes.vertical ] } ), - new Text( ')(', { - font: AreaModelCommonConstants.SYMBOL_FONT - } ), - new Node( { children: [ nodes.horizontal ] } ), - new Text( ')', { - font: AreaModelCommonConstants.SYMBOL_FONT - } ) - ], - spacing: 0, - scale: 0.9 - } ); - }, +export default inherit( AreaModelCommonRadioButtonGroup, PartialProductRadioButtonGroup, {}, { + /** + * Creates an 'exponents-allowed' icon based on a pair of nodes. + * @private + * + * @param {OrientationPair.} nodes + * @returns {Node} + */ + createExponentIcon: function( nodes ) { + return new HBox( { + children: [ + new Text( '(', { + font: AreaModelCommonConstants.SYMBOL_FONT + } ), + new Node( { children: [ nodes.vertical ] } ), + new Text( ')(', { + font: AreaModelCommonConstants.SYMBOL_FONT + } ), + new Node( { children: [ nodes.horizontal ] } ), + new Text( ')', { + font: AreaModelCommonConstants.SYMBOL_FONT + } ) + ], + spacing: 0, + scale: 0.9 + } ); + }, - /** - * Creates an 'no-exponents' icon based on a pair of nodes. - * @private - * - * @param {OrientationPair.} nodes - * @returns {Node} - */ - createNonExponentIcon: function( nodes ) { - return new HBox( { - children: [ - new Node( { children: [ nodes.vertical ] } ), - new Text( MathSymbols.TIMES, { - font: AreaModelCommonConstants.SYMBOL_FONT - } ), - new Node( { children: [ nodes.horizontal ] } ) - ], - spacing: 5 - } ); - } - } ); -} ); + /** + * Creates an 'no-exponents' icon based on a pair of nodes. + * @private + * + * @param {OrientationPair.} nodes + * @returns {Node} + */ + createNonExponentIcon: function( nodes ) { + return new HBox( { + children: [ + new Node( { children: [ nodes.vertical ] } ), + new Text( MathSymbols.TIMES, { + font: AreaModelCommonConstants.SYMBOL_FONT + } ), + new Node( { children: [ nodes.horizontal ] } ) + ], + spacing: 5 + } ); + } +} ); \ No newline at end of file diff --git a/js/common/view/PoolableLayerNode.js b/js/common/view/PoolableLayerNode.js index 6ed653b9..32e966d7 100644 --- a/js/common/view/PoolableLayerNode.js +++ b/js/common/view/PoolableLayerNode.js @@ -12,83 +12,80 @@ * * @author Jonathan Olson */ -define( require => { - 'use strict'; - - // modules - const areaModelCommon = require( 'AREA_MODEL_COMMON/areaModelCommon' ); - const inherit = require( 'PHET_CORE/inherit' ); - const merge = require( 'PHET_CORE/merge' ); - const Node = require( 'SCENERY/nodes/Node' ); - - /** - * @constructor - * @extends {Node} - * - * So you have a Property.>, and you want to lazily create ItemNodes for each? And say each ItemNode - * has something like itemNode.itemProperty which controls which item it displays? And if the property is null, it - * doesn't display? Do I have the incredibly-specific helper type for you! For the LOW LOW price of moving it to a - * common repo, YOU COULD HAVE IT TOO! - * - * Hopefully this doesn't become a common pattern. We have 3+ usages of it, and it cleans things up overall to have - * the not-super-simple logic in one place. Enjoy. - * - * @param {Object} config - */ - function PoolableLayerNode( config ) { - const self = this; - - config = merge( { - // required - arrayProperty: null, // {Property.>} - Property that has an array of items - createNode: null, // {function} - function( {*} item ): {Node} - Create a node from an item - getItemProperty: null, // {function} - function( {*} itemNode ): {Property.<*>} - ItemNode => Item Property - - // Allow providing references - usedArray: [], - unusedArray: [], - - // Called after we run an update. - updatedCallback: null - }, config ); - - Node.call( this ); - - const usedArray = config.usedArray; - const unusedArray = config.unusedArray; - - config.arrayProperty.link( function( items ) { - - // Unuse all of the item nodes (set their property to null, hiding them, and put them in the unused array) - while ( usedArray.length ) { - const oldItemNode = usedArray.pop(); - config.getItemProperty( oldItemNode ).value = null; - unusedArray.push( oldItemNode ); - } - items.forEach( function( item ) { - let itemNode; +import inherit from '../../../../phet-core/js/inherit.js'; +import merge from '../../../../phet-core/js/merge.js'; +import Node from '../../../../scenery/js/nodes/Node.js'; +import areaModelCommon from '../../areaModelCommon.js'; + +/** + * @constructor + * @extends {Node} + * + * So you have a Property.>, and you want to lazily create ItemNodes for each? And say each ItemNode + * has something like itemNode.itemProperty which controls which item it displays? And if the property is null, it + * doesn't display? Do I have the incredibly-specific helper type for you! For the LOW LOW price of moving it to a + * common repo, YOU COULD HAVE IT TOO! + * + * Hopefully this doesn't become a common pattern. We have 3+ usages of it, and it cleans things up overall to have + * the not-super-simple logic in one place. Enjoy. + * + * @param {Object} config + */ +function PoolableLayerNode( config ) { + const self = this; + + config = merge( { + // required + arrayProperty: null, // {Property.>} - Property that has an array of items + createNode: null, // {function} - function( {*} item ): {Node} - Create a node from an item + getItemProperty: null, // {function} - function( {*} itemNode ): {Property.<*>} - ItemNode => Item Property + + // Allow providing references + usedArray: [], + unusedArray: [], - // Grab one from the pool - if ( unusedArray.length ) { - itemNode = unusedArray.pop(); - config.getItemProperty( itemNode ).value = item; - } + // Called after we run an update. + updatedCallback: null + }, config ); - // Or create a new one - else { - itemNode = config.createNode( item ); - self.addChild( itemNode ); - } + Node.call( this ); - usedArray.push( itemNode ); - } ); + const usedArray = config.usedArray; + const unusedArray = config.unusedArray; - config.updatedCallback && config.updatedCallback(); + config.arrayProperty.link( function( items ) { + + // Unuse all of the item nodes (set their property to null, hiding them, and put them in the unused array) + while ( usedArray.length ) { + const oldItemNode = usedArray.pop(); + config.getItemProperty( oldItemNode ).value = null; + unusedArray.push( oldItemNode ); + } + + items.forEach( function( item ) { + let itemNode; + + // Grab one from the pool + if ( unusedArray.length ) { + itemNode = unusedArray.pop(); + config.getItemProperty( itemNode ).value = item; + } + + // Or create a new one + else { + itemNode = config.createNode( item ); + self.addChild( itemNode ); + } + + usedArray.push( itemNode ); } ); - } - areaModelCommon.register( 'PoolableLayerNode', PoolableLayerNode ); + config.updatedCallback && config.updatedCallback(); + } ); +} + +areaModelCommon.register( 'PoolableLayerNode', PoolableLayerNode ); - return inherit( Node, PoolableLayerNode ); -} ); +inherit( Node, PoolableLayerNode ); +export default PoolableLayerNode; \ No newline at end of file diff --git a/js/common/view/RangeLabelNode.js b/js/common/view/RangeLabelNode.js index 84d6e1f9..e6451c09 100644 --- a/js/common/view/RangeLabelNode.js +++ b/js/common/view/RangeLabelNode.js @@ -8,138 +8,135 @@ * * @author Jonathan Olson */ -define( require => { - 'use strict'; - - // modules - const areaModelCommon = require( 'AREA_MODEL_COMMON/areaModelCommon' ); - const AreaModelCommonConstants = require( 'AREA_MODEL_COMMON/common/AreaModelCommonConstants' ); - const inherit = require( 'PHET_CORE/inherit' ); - const Line = require( 'SCENERY/nodes/Line' ); - const Node = require( 'SCENERY/nodes/Node' ); - const Orientation = require( 'PHET_CORE/Orientation' ); - const RichText = require( 'SCENERY/nodes/RichText' ); - const Vector2 = require( 'DOT/Vector2' ); - - // constants - const TICK_LENGTH = 10; // How long the tick marks are for the range labels - - /** - * @constructor - * @extends {Node} - * - * @param {Property.} termListProperty - * @param {Orientation} orientation - * @param {Property.>} tickLocationsProperty - In view coordinates - * @param {Property.} colorProperty - * @param {boolean} isProportional - Whether the area is shown as proportional (instead of generic) - */ - function RangeLabelNode( termListProperty, orientation, tickLocationsProperty, colorProperty, isProportional ) { - - const self = this; - - Node.call( this ); - - const rangeOffset = ( isProportional + +import Vector2 from '../../../../dot/js/Vector2.js'; +import inherit from '../../../../phet-core/js/inherit.js'; +import Orientation from '../../../../phet-core/js/Orientation.js'; +import Line from '../../../../scenery/js/nodes/Line.js'; +import Node from '../../../../scenery/js/nodes/Node.js'; +import RichText from '../../../../scenery/js/nodes/RichText.js'; +import areaModelCommon from '../../areaModelCommon.js'; +import AreaModelCommonConstants from '../AreaModelCommonConstants.js'; + +// constants +const TICK_LENGTH = 10; // How long the tick marks are for the range labels + +/** + * @constructor + * @extends {Node} + * + * @param {Property.} termListProperty + * @param {Orientation} orientation + * @param {Property.>} tickLocationsProperty - In view coordinates + * @param {Property.} colorProperty + * @param {boolean} isProportional - Whether the area is shown as proportional (instead of generic) + */ +function RangeLabelNode( termListProperty, orientation, tickLocationsProperty, colorProperty, isProportional ) { + + const self = this; + + Node.call( this ); + + const rangeOffset = ( isProportional ? AreaModelCommonConstants.PROPORTIONAL_RANGE_OFFSET : AreaModelCommonConstants.GENERIC_RANGE_OFFSET )[ orientation.opposite.coordinate ]; - const richText = new RichText( '', { - font: AreaModelCommonConstants.TOTAL_SIZE_READOUT_FONT, - fill: colorProperty - } ); + const richText = new RichText( '', { + font: AreaModelCommonConstants.TOTAL_SIZE_READOUT_FONT, + fill: colorProperty + } ); - // Constrain width on the left side (don't let it go out of the layout bounds) - if ( orientation === Orientation.VERTICAL ) { - const verticalRangeOffset = isProportional + // Constrain width on the left side (don't let it go out of the layout bounds) + if ( orientation === Orientation.VERTICAL ) { + const verticalRangeOffset = isProportional ? AreaModelCommonConstants.PROPORTIONAL_RANGE_OFFSET : AreaModelCommonConstants.GENERIC_RANGE_OFFSET; - richText.maxWidth = AreaModelCommonConstants.MAIN_AREA_OFFSET.x + verticalRangeOffset.x - AreaModelCommonConstants.LAYOUT_SPACING; - } + richText.maxWidth = AreaModelCommonConstants.MAIN_AREA_OFFSET.x + verticalRangeOffset.x - AreaModelCommonConstants.LAYOUT_SPACING; + } - // Update the label richText - termListProperty.link( function( termList ) { + // Update the label richText + termListProperty.link( function( termList ) { - const hasTerms = termList !== null && termList.terms.length > 0; + const hasTerms = termList !== null && termList.terms.length > 0; - richText.visible = hasTerms; - if ( hasTerms ) { - richText.text = termList.toRichString(); + richText.visible = hasTerms; + if ( hasTerms ) { + richText.text = termList.toRichString(); - // Relative positioning - if ( orientation === Orientation.HORIZONTAL ) { - richText.centerBottom = Vector2.ZERO; - } - else { - richText.rightCenter = Vector2.ZERO; - } + // Relative positioning + if ( orientation === Orientation.HORIZONTAL ) { + richText.centerBottom = Vector2.ZERO; } - } ); + else { + richText.rightCenter = Vector2.ZERO; + } + } + } ); - // Wrap our text in a label, so that we can handle positioning independent of bounds checks - const textContainer = new Node( { - children: [ richText ] - } ); - this.addChild( textContainer ); + // Wrap our text in a label, so that we can handle positioning independent of bounds checks + const textContainer = new Node( { + children: [ richText ] + } ); + this.addChild( textContainer ); - // Coordinate that doesn't change. Customized offsets added - textContainer[ orientation.opposite.coordinate ] = rangeOffset + ( orientation === Orientation.HORIZONTAL ? -3 : -5 ); + // Coordinate that doesn't change. Customized offsets added + textContainer[ orientation.opposite.coordinate ] = rangeOffset + ( orientation === Orientation.HORIZONTAL ? -3 : -5 ); - // Our main line, that the tick marks will be off of - const line = new Line( { - stroke: colorProperty - } ); - this.addChild( line ); + // Our main line, that the tick marks will be off of + const line = new Line( { + stroke: colorProperty + } ); + this.addChild( line ); - const ticks = []; + const ticks = []; - // Update the layout - tickLocationsProperty.link( function( tickLocations ) { - assert && assert( tickLocations.length === 0 || tickLocations.length >= 2 ); + // Update the layout + tickLocationsProperty.link( function( tickLocations ) { + assert && assert( tickLocations.length === 0 || tickLocations.length >= 2 ); - if ( tickLocations.length === 0 ) { - ticks.forEach( function( tick ) { - tick.visible = false; + if ( tickLocations.length === 0 ) { + ticks.forEach( function( tick ) { + tick.visible = false; + } ); + } + else { + + // Add any ticks that we need + while ( ticks.length < tickLocations.length ) { + const tick = new Line( { + y1: 0, + y2: TICK_LENGTH / 2, + stroke: colorProperty, + rotation: orientation === Orientation.HORIZONTAL ? 0 : -Math.PI / 2 } ); + ticks.push( tick ); + self.addChild( tick ); } - else { - // Add any ticks that we need - while ( ticks.length < tickLocations.length ) { - const tick = new Line( { - y1: 0, - y2: TICK_LENGTH / 2, - stroke: colorProperty, - rotation: orientation === Orientation.HORIZONTAL ? 0 : -Math.PI / 2 - } ); - ticks.push( tick ); - self.addChild( tick ); - } + ticks.forEach( function( tick, index ) { + if ( index < tickLocations.length ) { + tick.visible = true; + tick.translation = orientation.toVector( tickLocations[ index ], rangeOffset ); - ticks.forEach( function( tick, index ) { - if ( index < tickLocations.length ) { - tick.visible = true; - tick.translation = orientation.toVector( tickLocations[ index ], rangeOffset ); - - // The first/last ticks should have a different length - tick.y1 = ( index === 0 || index === tickLocations.length - 1 ) ? -TICK_LENGTH / 2 : 0; - } - else { - tick.visible = false; - } - } ); + // The first/last ticks should have a different length + tick.y1 = ( index === 0 || index === tickLocations.length - 1 ) ? -TICK_LENGTH / 2 : 0; + } + else { + tick.visible = false; + } + } ); - const minLocation = tickLocations[ 0 ]; - const maxLocation = tickLocations[ tickLocations.length - 1 ]; + const minLocation = tickLocations[ 0 ]; + const maxLocation = tickLocations[ tickLocations.length - 1 ]; - line.p1 = orientation.toVector( minLocation, rangeOffset ); - line.p2 = orientation.toVector( maxLocation, rangeOffset ); - textContainer[ orientation.coordinate ] = ( maxLocation + minLocation ) / 2; // centered - } - } ); - } + line.p1 = orientation.toVector( minLocation, rangeOffset ); + line.p2 = orientation.toVector( maxLocation, rangeOffset ); + textContainer[ orientation.coordinate ] = ( maxLocation + minLocation ) / 2; // centered + } + } ); +} - areaModelCommon.register( 'RangeLabelNode', RangeLabelNode ); +areaModelCommon.register( 'RangeLabelNode', RangeLabelNode ); - return inherit( Node, RangeLabelNode ); -} ); +inherit( Node, RangeLabelNode ); +export default RangeLabelNode; \ No newline at end of file diff --git a/js/common/view/TotalAreaNode.js b/js/common/view/TotalAreaNode.js index 854d6a13..454e9202 100644 --- a/js/common/view/TotalAreaNode.js +++ b/js/common/view/TotalAreaNode.js @@ -7,96 +7,93 @@ * * @author Jonathan Olson */ -define( require => { - 'use strict'; - // modules - const AlignBox = require( 'SCENERY/nodes/AlignBox' ); - const areaModelCommon = require( 'AREA_MODEL_COMMON/areaModelCommon' ); - const AreaModelCommonA11yStrings = require( 'AREA_MODEL_COMMON/AreaModelCommonA11yStrings' ); - const AreaModelCommonColorProfile = require( 'AREA_MODEL_COMMON/common/view/AreaModelCommonColorProfile' ); - const AreaModelCommonConstants = require( 'AREA_MODEL_COMMON/common/AreaModelCommonConstants' ); - const Bounds2 = require( 'DOT/Bounds2' ); - const HBox = require( 'SCENERY/nodes/HBox' ); - const inherit = require( 'PHET_CORE/inherit' ); - const MathSymbols = require( 'SCENERY_PHET/MathSymbols' ); - const Node = require( 'SCENERY/nodes/Node' ); - const Panel = require( 'SUN/Panel' ); - const RichText = require( 'SCENERY/nodes/RichText' ); - const StringUtils = require( 'PHETCOMMON/util/StringUtils' ); - const Term = require( 'AREA_MODEL_COMMON/common/model/Term' ); - const Text = require( 'SCENERY/nodes/Text' ); +import Bounds2 from '../../../../dot/js/Bounds2.js'; +import inherit from '../../../../phet-core/js/inherit.js'; +import StringUtils from '../../../../phetcommon/js/util/StringUtils.js'; +import MathSymbols from '../../../../scenery-phet/js/MathSymbols.js'; +import AlignBox from '../../../../scenery/js/nodes/AlignBox.js'; +import HBox from '../../../../scenery/js/nodes/HBox.js'; +import Node from '../../../../scenery/js/nodes/Node.js'; +import RichText from '../../../../scenery/js/nodes/RichText.js'; +import Text from '../../../../scenery/js/nodes/Text.js'; +import Panel from '../../../../sun/js/Panel.js'; +import areaModelCommonStrings from '../../area-model-common-strings.js'; +import areaModelCommon from '../../areaModelCommon.js'; +import AreaModelCommonA11yStrings from '../../AreaModelCommonA11yStrings.js'; +import AreaModelCommonConstants from '../AreaModelCommonConstants.js'; +import Term from '../model/Term.js'; +import AreaModelCommonColorProfile from './AreaModelCommonColorProfile.js'; - // strings - const areaString = require( 'string!AREA_MODEL_COMMON/area' ); +const areaString = areaModelCommonStrings.area; - // a11y strings - const areaEqualsPatternString = AreaModelCommonA11yStrings.areaEqualsPattern.value; +// a11y strings +const areaEqualsPatternString = AreaModelCommonA11yStrings.areaEqualsPattern.value; - /** - * @constructor - * @extends {Node} - * - * @param {Property.} totalAreaProperty - * @param {boolean} isProportional - Whether the area is shown as proportional (instead of generic) - * @param {string} maximumWidthString - If proportional - * @param {boolean} useTileLikeBackground - Whether the "tile" color should be used with an area background (if any) - */ - function TotalAreaNode( totalAreaProperty, isProportional, maximumWidthString, useTileLikeBackground ) { - const self = this; +/** + * @constructor + * @extends {Node} + * + * @param {Property.} totalAreaProperty + * @param {boolean} isProportional - Whether the area is shown as proportional (instead of generic) + * @param {string} maximumWidthString - If proportional + * @param {boolean} useTileLikeBackground - Whether the "tile" color should be used with an area background (if any) + */ +function TotalAreaNode( totalAreaProperty, isProportional, maximumWidthString, useTileLikeBackground ) { + const self = this; - // If powers of x are supported, we need to have a slightly different initial height so we can align-bottom. - const areaText = new RichText( Term.getLargestGenericString( true, 3 ), { - font: AreaModelCommonConstants.TOTAL_AREA_VALUE_FONT - } ); + // If powers of x are supported, we need to have a slightly different initial height so we can align-bottom. + const areaText = new RichText( Term.getLargestGenericString( true, 3 ), { + font: AreaModelCommonConstants.TOTAL_AREA_VALUE_FONT + } ); - let areaNode; - if ( isProportional ) { + let areaNode; + if ( isProportional ) { - // Has numeric display, so it doesn't need maxWidth - areaText.text = maximumWidthString; - areaNode = new HBox( { - spacing: 8, - children: [ - new Text( areaString, { font: AreaModelCommonConstants.TOTAL_AREA_LABEL_FONT } ), - new Text( MathSymbols.EQUAL_TO, { font: AreaModelCommonConstants.TOTAL_AREA_LABEL_FONT } ), - // AlignBox it so that it is always centered and keeps the same bounds - new Panel( new AlignBox( areaText, { alignBounds: areaText.bounds.copy(), yAlign: 'bottom' } ), { - fill: useTileLikeBackground - ? AreaModelCommonColorProfile.smallTileProperty - : AreaModelCommonColorProfile.proportionalActiveAreaBackgroundProperty - } ) - ] - } ); - } - else { - areaText.maxWidth = AreaModelCommonConstants.PANEL_INTERIOR_MAX; + // Has numeric display, so it doesn't need maxWidth + areaText.text = maximumWidthString; + areaNode = new HBox( { + spacing: 8, + children: [ + new Text( areaString, { font: AreaModelCommonConstants.TOTAL_AREA_LABEL_FONT } ), + new Text( MathSymbols.EQUAL_TO, { font: AreaModelCommonConstants.TOTAL_AREA_LABEL_FONT } ), + // AlignBox it so that it is always centered and keeps the same bounds + new Panel( new AlignBox( areaText, { alignBounds: areaText.bounds.copy(), yAlign: 'bottom' } ), { + fill: useTileLikeBackground + ? AreaModelCommonColorProfile.smallTileProperty + : AreaModelCommonColorProfile.proportionalActiveAreaBackgroundProperty + } ) + ] + } ); + } + else { + areaText.maxWidth = AreaModelCommonConstants.PANEL_INTERIOR_MAX; - // AlignBox it so that it is always centered and keeps the same bounds - areaNode = new AlignBox( areaText, { alignBounds: new Bounds2( 0, 0, AreaModelCommonConstants.PANEL_INTERIOR_MAX, areaText.height ) } ); - } + // AlignBox it so that it is always centered and keeps the same bounds + areaNode = new AlignBox( areaText, { alignBounds: new Bounds2( 0, 0, AreaModelCommonConstants.PANEL_INTERIOR_MAX, areaText.height ) } ); + } - Node.call( this, { - children: [ - areaNode - ], - maxWidth: AreaModelCommonConstants.PANEL_INTERIOR_MAX, + Node.call( this, { + children: [ + areaNode + ], + maxWidth: AreaModelCommonConstants.PANEL_INTERIOR_MAX, - // a11y - tagName: 'p' - } ); + // a11y + tagName: 'p' + } ); - // Update the text. - totalAreaProperty.link( function( polynomial ) { - const labelString = polynomial === null ? '?' : polynomial.toRichString(); - areaText.text = labelString; - self.innerContent = StringUtils.fillIn( areaEqualsPatternString, { - area: labelString - } ); + // Update the text. + totalAreaProperty.link( function( polynomial ) { + const labelString = polynomial === null ? '?' : polynomial.toRichString(); + areaText.text = labelString; + self.innerContent = StringUtils.fillIn( areaEqualsPatternString, { + area: labelString } ); - } + } ); +} - areaModelCommon.register( 'TotalAreaNode', TotalAreaNode ); +areaModelCommon.register( 'TotalAreaNode', TotalAreaNode ); - return inherit( Node, TotalAreaNode ); -} ); +inherit( Node, TotalAreaNode ); +export default TotalAreaNode; \ No newline at end of file diff --git a/js/common/view/calculation/CalculationGroup.js b/js/common/view/calculation/CalculationGroup.js index c3c1d1d8..686f0c99 100644 --- a/js/common/view/calculation/CalculationGroup.js +++ b/js/common/view/calculation/CalculationGroup.js @@ -7,72 +7,68 @@ * * @author Jonathan Olson */ -define( require => { - 'use strict'; - // modules - const areaModelCommon = require( 'AREA_MODEL_COMMON/areaModelCommon' ); - const HBox = require( 'SCENERY/nodes/HBox' ); - const inherit = require( 'PHET_CORE/inherit' ); - const Poolable = require( 'PHET_CORE/Poolable' ); +import inherit from '../../../../../phet-core/js/inherit.js'; +import Poolable from '../../../../../phet-core/js/Poolable.js'; +import HBox from '../../../../../scenery/js/nodes/HBox.js'; +import areaModelCommon from '../../../areaModelCommon.js'; - /** - * @constructor - * @extends {HBox} - * - * @param {Array.} nodes - Each should have a clean() method to support pooling - * @param {number} spacing - */ - function CalculationGroup( nodes, spacing ) { - assert && assert( Array.isArray( nodes ) ); - assert && assert( typeof spacing === 'number' ); - - // @public {string} - this.accessibleText = nodes.map( function( node ) { - return node.accessibleText; - } ).join( ' ' ); +/** + * @constructor + * @extends {HBox} + * + * @param {Array.} nodes - Each should have a clean() method to support pooling + * @param {number} spacing + */ +function CalculationGroup( nodes, spacing ) { + assert && assert( Array.isArray( nodes ) ); + assert && assert( typeof spacing === 'number' ); - // @private {Array.|null} - this.nodes = nodes; + // @public {string} + this.accessibleText = nodes.map( function( node ) { + return node.accessibleText; + } ).join( ' ' ); - if ( !this.initialized ) { - this.initialized = true; + // @private {Array.|null} + this.nodes = nodes; - HBox.call( this, { - align: 'bottom', + if ( !this.initialized ) { + this.initialized = true; - // a11y - accessibleNamespace: 'http://www.w3.org/1998/Math/MathML' - } ); - } + HBox.call( this, { + align: 'bottom', - this.mutate( { - tagName: nodes.length > 1 ? 'mrow' : null, - spacing: spacing, - children: nodes + // a11y + accessibleNamespace: 'http://www.w3.org/1998/Math/MathML' } ); } - areaModelCommon.register( 'CalculationGroup', CalculationGroup ); - - inherit( HBox, CalculationGroup, { - /** - * Clears the state of this node (releasing references) so it can be freed to the pool (and potentially GC'ed). - * @public - */ - clean: function() { - // Remove our content - this.removeAllChildren(); - this.nodes.forEach( function( node ) { - node.clean(); - } ); - this.nodes = null; - - this.freeToPool(); - } + this.mutate( { + tagName: nodes.length > 1 ? 'mrow' : null, + spacing: spacing, + children: nodes } ); +} - Poolable.mixInto( CalculationGroup ); +areaModelCommon.register( 'CalculationGroup', CalculationGroup ); - return CalculationGroup; +inherit( HBox, CalculationGroup, { + /** + * Clears the state of this node (releasing references) so it can be freed to the pool (and potentially GC'ed). + * @public + */ + clean: function() { + // Remove our content + this.removeAllChildren(); + this.nodes.forEach( function( node ) { + node.clean(); + } ); + this.nodes = null; + + this.freeToPool(); + } } ); + +Poolable.mixInto( CalculationGroup ); + +export default CalculationGroup; \ No newline at end of file diff --git a/js/common/view/calculation/CalculationLine.js b/js/common/view/calculation/CalculationLine.js index 9789e20f..979f08d5 100644 --- a/js/common/view/calculation/CalculationLine.js +++ b/js/common/view/calculation/CalculationLine.js @@ -5,305 +5,301 @@ * * @author Jonathan Olson */ -define( require => { - 'use strict'; - - // modules - const areaModelCommon = require( 'AREA_MODEL_COMMON/areaModelCommon' ); - const AreaModelCommonColorProfile = require( 'AREA_MODEL_COMMON/common/view/AreaModelCommonColorProfile' ); - const AreaModelCommonConstants = require( 'AREA_MODEL_COMMON/common/AreaModelCommonConstants' ); - const CalculationGroup = require( 'AREA_MODEL_COMMON/common/view/calculation/CalculationGroup' ); - const DerivedProperty = require( 'AXON/DerivedProperty' ); - const inherit = require( 'PHET_CORE/inherit' ); - const Minus = require( 'AREA_MODEL_COMMON/common/view/calculation/Minus' ); - const MultiplyX = require( 'AREA_MODEL_COMMON/common/view/calculation/MultiplyX' ); - const OrientationPair = require( 'AREA_MODEL_COMMON/common/model/OrientationPair' ); - const Parentheses = require( 'AREA_MODEL_COMMON/common/view/calculation/Parentheses' ); - const PlaceholderBox = require( 'AREA_MODEL_COMMON/common/view/calculation/PlaceholderBox' ); - const Plus = require( 'AREA_MODEL_COMMON/common/view/calculation/Plus' ); - const QuestionMark = require( 'AREA_MODEL_COMMON/common/view/calculation/QuestionMark' ); - const TermText = require( 'AREA_MODEL_COMMON/common/view/calculation/TermText' ); - /** - * @constructor - * @extends {Object} - * - * @param {number} index - * @param {OrientationPair.>} colorProperties - * @param {Property.} activeIndexProperty - * @param {boolean} allowExponents - Whether exponents (powers of x) are allowed - * @param {boolean} isProportional - Whether the area is shown as proportional (instead of generic) - */ - function CalculationLine( index, colorProperties, activeIndexProperty, allowExponents, isProportional ) { - assert && assert( typeof index === 'number' ); +import DerivedProperty from '../../../../../axon/js/DerivedProperty.js'; +import inherit from '../../../../../phet-core/js/inherit.js'; +import areaModelCommon from '../../../areaModelCommon.js'; +import AreaModelCommonConstants from '../../AreaModelCommonConstants.js'; +import OrientationPair from '../../model/OrientationPair.js'; +import AreaModelCommonColorProfile from '../AreaModelCommonColorProfile.js'; +import CalculationGroup from './CalculationGroup.js'; +import Minus from './Minus.js'; +import MultiplyX from './MultiplyX.js'; +import Parentheses from './Parentheses.js'; +import PlaceholderBox from './PlaceholderBox.js'; +import Plus from './Plus.js'; +import QuestionMark from './QuestionMark.js'; +import TermText from './TermText.js'; - const self = this; +/** + * @constructor + * @extends {Object} + * + * @param {number} index + * @param {OrientationPair.>} colorProperties + * @param {Property.} activeIndexProperty + * @param {boolean} allowExponents - Whether exponents (powers of x) are allowed + * @param {boolean} isProportional - Whether the area is shown as proportional (instead of generic) + */ +function CalculationLine( index, colorProperties, activeIndexProperty, allowExponents, isProportional ) { + assert && assert( typeof index === 'number' ); - // @public {Node|null} - The {Node}, if provided, should have a `node.clean()` method to release references - // (usually freeing it to a pool) and a `node.accessibleText` {string} representing the description of the line. - // Filled in later, should be non-null outside CalculationLine usage. - this.node = null; + const self = this; - // @public {number} - this.index = index; + // @public {Node|null} - The {Node}, if provided, should have a `node.clean()` method to release references + // (usually freeing it to a pool) and a `node.accessibleText` {string} representing the description of the line. + // Filled in later, should be non-null outside CalculationLine usage. + this.node = null; - // @public {CalculationLine|null} - Linked-list support for easy traversal through lines - this.previousLine = null; - this.nextLine = null; + // @public {number} + this.index = index; - // @private {boolean} - this.allowExponents = allowExponents; - this.isProportional = isProportional; + // @public {CalculationLine|null} - Linked-list support for easy traversal through lines + this.previousLine = null; + this.nextLine = null; - // @private {Property.} - this.isActiveProperty = new DerivedProperty( [ activeIndexProperty ], function( activeIndex ) { - return activeIndex === null || activeIndex === index; - } ); + // @private {boolean} + this.allowExponents = allowExponents; + this.isProportional = isProportional; + + // @private {Property.} + this.isActiveProperty = new DerivedProperty( [ activeIndexProperty ], function( activeIndex ) { + return activeIndex === null || activeIndex === index; + } ); + + // @private {Property.} + this.baseColorProperty = new DerivedProperty( [ + this.isActiveProperty, + AreaModelCommonColorProfile.calculationActiveProperty, + AreaModelCommonColorProfile.calculationInactiveProperty + ], function( isActive, activeColor, inactiveColor ) { + return isActive ? activeColor : inactiveColor; + }, { + useDeepEquality: true + } ); - // @private {Property.} - this.baseColorProperty = new DerivedProperty( [ - this.isActiveProperty, - AreaModelCommonColorProfile.calculationActiveProperty, + // @private {OrientationPair.>} + this.orientedColorProperties = OrientationPair.create( function( orientation ) { + return new DerivedProperty( [ + self.isActiveProperty, + colorProperties.get( orientation ), AreaModelCommonColorProfile.calculationInactiveProperty ], function( isActive, activeColor, inactiveColor ) { return isActive ? activeColor : inactiveColor; }, { useDeepEquality: true } ); + } ); +} - // @private {OrientationPair.>} - this.orientedColorProperties = OrientationPair.create( function( orientation ) { - return new DerivedProperty( [ - self.isActiveProperty, - colorProperties.get( orientation ), - AreaModelCommonColorProfile.calculationInactiveProperty - ], function( isActive, activeColor, inactiveColor ) { - return isActive ? activeColor : inactiveColor; - }, { - useDeepEquality: true - } ); - } ); - } +areaModelCommon.register( 'CalculationLine', CalculationLine ); + +export default inherit( Object, CalculationLine, { + /** + * Creates a TermText with the baseColor. + * @public + * + * @param {TermList|Term} term + * @param {boolean} excludeSign + * @returns {TermText} + */ + baseTermText: function( term, excludeSign ) { + assert && assert( typeof excludeSign === 'boolean' ); + + return TermText.createFromPool( term, this.baseColorProperty, excludeSign ); + }, - areaModelCommon.register( 'CalculationLine', CalculationLine ); - - return inherit( Object, CalculationLine, { - /** - * Creates a TermText with the baseColor. - * @public - * - * @param {TermList|Term} term - * @param {boolean} excludeSign - * @returns {TermText} - */ - baseTermText: function( term, excludeSign ) { - assert && assert( typeof excludeSign === 'boolean' ); - - return TermText.createFromPool( term, this.baseColorProperty, excludeSign ); - }, - - /** - * Creates a TermText with the color of a specific orientation. - * @public - * - * @param {Orientation} orientation - * @param {TermList|Term} term - * @returns {TermText} - */ - orientedTermText: function( orientation, term ) { - return TermText.createFromPool( term, this.orientedColorProperties.get( orientation ), false ); - }, - - /** - * Creates a PlaceholderBox with the color of a specific orientation. - * @public - * - * @param {Orientation} orientation - * @returns {PlaceholderBox} - */ - orientedPlaceholderBox: function( orientation ) { - return PlaceholderBox.createFromPool( this.orientedColorProperties.get( orientation ), this.allowExponents ); - }, - - /** - * Creates a MultiplyX with the specified content. - * @public - * - * @param {Node} leftContent - * @param {Node} rightContent - * @returns {MultiplyX} - */ - multiplyX: function( leftContent, rightContent ) { - return MultiplyX.createFromPool( leftContent, rightContent, this.baseColorProperty ); - }, - - /** - * Creates a Parentheses with the specified content. - * @public - * - * @param {Node} content - * @returns {Parentheses} - */ - parentheses: function( content ) { - return Parentheses.createFromPool( content, this.baseColorProperty ); - }, - - /** - * Creates a QuestionMark - * @public - * - * @returns {QuestionMark} - */ - questionMark: function() { - return QuestionMark.createFromPool( this.baseColorProperty ); - }, - - /** - * Creates a Plus - * @public - * - * @returns {Plus} - */ - plus: function() { - return Plus.createFromPool( this.baseColorProperty ); - }, - - /** - * Creates a Minus - * @public - * - * @returns {Minus} - */ - minus: function() { - return Minus.createFromPool( this.baseColorProperty ); - }, - - /** - * Creates a calculation group. - * @public - * - * @param {Array.} nodes - * @param {number} spacing - * @returns {Parentheses} - */ - group: function( nodes, spacing ) { - return CalculationGroup.createFromPool( nodes, spacing ); - }, - - /** - * Returns the grouping of all nodes provided, with plusses in-between each node. - * @public - * - * @param {Array.} nodes - * @returns {Node} - */ - sumGroup: function( nodes ) { - const self = this; - - return this.group( _.flatten( nodes.map( function( node, index ) { - return index > 0 ? [ self.plus(), node ] : [ node ]; - } ) ), AreaModelCommonConstants.CALCULATION_OP_PADDING ); - }, - - /** - * Returns a grouping of all (oriented) terms provided, with plusses in-between each term. - * @public - * - * @param {Array.} terms - * @param {Orientation} orientation - * @returns {Node} - */ - sumOrientedTerms: function( terms, orientation ) { - const self = this; - - return this.sumGroup( terms.map( function( term ) { - return self.orientedTermText( orientation, term ); - } ) ); - }, - - /** - * Returns a grouping of all (non-oriented) terms, with plusses/minuses in-between each term (depending on the sign) - * @public - * - * @param {Array.} terms - * @returns {Node} - */ - sumOrDifferenceOfTerms: function( terms ) { - const self = this; - - return this.group( _.flatten( terms.map( function( term, index ) { - const result = []; - - if ( index > 0 ) { - result.push( term.coefficient >= 0 ? self.plus() : self.minus() ); - } - - result.push( self.baseTermText( term, index > 0 ) ); - - return result; - } ) ), AreaModelCommonConstants.CALCULATION_OP_PADDING ); - }, - - /** - * Returns a grouping of all (oriented) terms provided, with plusses in-between each term (negative grouped in - * parentheses). - * @public - * - * @param {Array.} terms - * @returns {Node} - */ - sumWithNegativeParens: function( terms ) { - const self = this; - - return this.sumGroup( terms.map( function( term ) { - let text = self.baseTermText( term, false ); - if ( term.coefficient < 0 ) { - text = self.parentheses( text ); - } - return text; - } ) ); - }, - - /** - * Returns an array with this lines (and any previous/next lines) in the correct order (up to 3 lines). - * @public - * - * @returns {Array.} - */ - getAdjacentLines: function() { + /** + * Creates a TermText with the color of a specific orientation. + * @public + * + * @param {Orientation} orientation + * @param {TermList|Term} term + * @returns {TermText} + */ + orientedTermText: function( orientation, term ) { + return TermText.createFromPool( term, this.orientedColorProperties.get( orientation ), false ); + }, + + /** + * Creates a PlaceholderBox with the color of a specific orientation. + * @public + * + * @param {Orientation} orientation + * @returns {PlaceholderBox} + */ + orientedPlaceholderBox: function( orientation ) { + return PlaceholderBox.createFromPool( this.orientedColorProperties.get( orientation ), this.allowExponents ); + }, + + /** + * Creates a MultiplyX with the specified content. + * @public + * + * @param {Node} leftContent + * @param {Node} rightContent + * @returns {MultiplyX} + */ + multiplyX: function( leftContent, rightContent ) { + return MultiplyX.createFromPool( leftContent, rightContent, this.baseColorProperty ); + }, + + /** + * Creates a Parentheses with the specified content. + * @public + * + * @param {Node} content + * @returns {Parentheses} + */ + parentheses: function( content ) { + return Parentheses.createFromPool( content, this.baseColorProperty ); + }, + + /** + * Creates a QuestionMark + * @public + * + * @returns {QuestionMark} + */ + questionMark: function() { + return QuestionMark.createFromPool( this.baseColorProperty ); + }, + + /** + * Creates a Plus + * @public + * + * @returns {Plus} + */ + plus: function() { + return Plus.createFromPool( this.baseColorProperty ); + }, + + /** + * Creates a Minus + * @public + * + * @returns {Minus} + */ + minus: function() { + return Minus.createFromPool( this.baseColorProperty ); + }, + + /** + * Creates a calculation group. + * @public + * + * @param {Array.} nodes + * @param {number} spacing + * @returns {Parentheses} + */ + group: function( nodes, spacing ) { + return CalculationGroup.createFromPool( nodes, spacing ); + }, + + /** + * Returns the grouping of all nodes provided, with plusses in-between each node. + * @public + * + * @param {Array.} nodes + * @returns {Node} + */ + sumGroup: function( nodes ) { + const self = this; + + return this.group( _.flatten( nodes.map( function( node, index ) { + return index > 0 ? [ self.plus(), node ] : [ node ]; + } ) ), AreaModelCommonConstants.CALCULATION_OP_PADDING ); + }, + + /** + * Returns a grouping of all (oriented) terms provided, with plusses in-between each term. + * @public + * + * @param {Array.} terms + * @param {Orientation} orientation + * @returns {Node} + */ + sumOrientedTerms: function( terms, orientation ) { + const self = this; + + return this.sumGroup( terms.map( function( term ) { + return self.orientedTermText( orientation, term ); + } ) ); + }, + + /** + * Returns a grouping of all (non-oriented) terms, with plusses/minuses in-between each term (depending on the sign) + * @public + * + * @param {Array.} terms + * @returns {Node} + */ + sumOrDifferenceOfTerms: function( terms ) { + const self = this; + + return this.group( _.flatten( terms.map( function( term, index ) { const result = []; - if ( this.previousLine ) { - result.push( this.previousLine ); - } - result.push( this ); - if ( this.nextLine ) { - result.push( this.nextLine ); + + if ( index > 0 ) { + result.push( term.coefficient >= 0 ? self.plus() : self.minus() ); } + + result.push( self.baseTermText( term, index > 0 ) ); + return result; - }, - - /** - * Removes external references. - * @public - */ - dispose: function() { - this.node.clean(); - - this.orientedColorProperties.horizontal.dispose(); - this.orientedColorProperties.vertical.dispose(); - this.baseColorProperty.dispose(); - this.isActiveProperty.dispose(); + } ) ), AreaModelCommonConstants.CALCULATION_OP_PADDING ); + }, + + /** + * Returns a grouping of all (oriented) terms provided, with plusses in-between each term (negative grouped in + * parentheses). + * @public + * + * @param {Array.} terms + * @returns {Node} + */ + sumWithNegativeParens: function( terms ) { + const self = this; + + return this.sumGroup( terms.map( function( term ) { + let text = self.baseTermText( term, false ); + if ( term.coefficient < 0 ) { + text = self.parentheses( text ); + } + return text; + } ) ); + }, + + /** + * Returns an array with this lines (and any previous/next lines) in the correct order (up to 3 lines). + * @public + * + * @returns {Array.} + */ + getAdjacentLines: function() { + const result = []; + if ( this.previousLine ) { + result.push( this.previousLine ); } - }, { - // @public {number} - Calculation line indices. Each individual type of line will have an index value in the order - // it would show up in the calculation panel. This index is used to determine what the "active" line is (for the - // line-by-line view), so that when updating the calculation it can attempt to stay on the same active line. - TOTALS_LINE_INDEX: 0, - EXPANDED_LINE_INDEX: 1, - DISTRIBUTION_LINE_INDEX: 2, - MULTIPLIED_LINE_INDEX: 3, - ORDERED_LINE_INDEX: 4, - MINUSES_LINE_INDEX: 5, - SUM_LINE_INDEX: 6 - } ); -} ); + result.push( this ); + if ( this.nextLine ) { + result.push( this.nextLine ); + } + return result; + }, + + /** + * Removes external references. + * @public + */ + dispose: function() { + this.node.clean(); + + this.orientedColorProperties.horizontal.dispose(); + this.orientedColorProperties.vertical.dispose(); + this.baseColorProperty.dispose(); + this.isActiveProperty.dispose(); + } +}, { + // @public {number} - Calculation line indices. Each individual type of line will have an index value in the order + // it would show up in the calculation panel. This index is used to determine what the "active" line is (for the + // line-by-line view), so that when updating the calculation it can attempt to stay on the same active line. + TOTALS_LINE_INDEX: 0, + EXPANDED_LINE_INDEX: 1, + DISTRIBUTION_LINE_INDEX: 2, + MULTIPLIED_LINE_INDEX: 3, + ORDERED_LINE_INDEX: 4, + MINUSES_LINE_INDEX: 5, + SUM_LINE_INDEX: 6 +} ); \ No newline at end of file diff --git a/js/common/view/calculation/CalculationLinesNode.js b/js/common/view/calculation/CalculationLinesNode.js index 83cada4f..d4a55ad4 100644 --- a/js/common/view/calculation/CalculationLinesNode.js +++ b/js/common/view/calculation/CalculationLinesNode.js @@ -5,380 +5,376 @@ * * @author Jonathan Olson */ -define( require => { - 'use strict'; - - // modules - const AreaCalculationChoice = require( 'AREA_MODEL_COMMON/common/model/AreaCalculationChoice' ); - const areaModelCommon = require( 'AREA_MODEL_COMMON/areaModelCommon' ); - const AreaModelCommonA11yStrings = require( 'AREA_MODEL_COMMON/AreaModelCommonA11yStrings' ); - const AreaModelCommonQueryParameters = require( 'AREA_MODEL_COMMON/common/AreaModelCommonQueryParameters' ); - const BooleanProperty = require( 'AXON/BooleanProperty' ); - const DerivedProperty = require( 'AXON/DerivedProperty' ); - const DistributionLine = require( 'AREA_MODEL_COMMON/common/view/calculation/DistributionLine' ); - const DynamicProperty = require( 'AXON/DynamicProperty' ); - const Emitter = require( 'AXON/Emitter' ); - const ExpandedLine = require( 'AREA_MODEL_COMMON/common/view/calculation/ExpandedLine' ); - const inherit = require( 'PHET_CORE/inherit' ); - const MinusesLine = require( 'AREA_MODEL_COMMON/common/view/calculation/MinusesLine' ); - const MultipliedLine = require( 'AREA_MODEL_COMMON/common/view/calculation/MultipliedLine' ); - const Node = require( 'SCENERY/nodes/Node' ); - const OrderedLine = require( 'AREA_MODEL_COMMON/common/view/calculation/OrderedLine' ); - const Orientation = require( 'PHET_CORE/Orientation' ); - const Property = require( 'AXON/Property' ); - const QuestionMarkLine = require( 'AREA_MODEL_COMMON/common/view/calculation/QuestionMarkLine' ); - const SumLine = require( 'AREA_MODEL_COMMON/common/view/calculation/SumLine' ); - const TermList = require( 'AREA_MODEL_COMMON/common/model/TermList' ); - const TotalsLine = require( 'AREA_MODEL_COMMON/common/view/calculation/TotalsLine' ); - const VBox = require( 'SCENERY/nodes/VBox' ); - - // a11y strings - const betweenCalculationLinesString = AreaModelCommonA11yStrings.betweenCalculationLines.value; - /** - * @constructor - * @extends {VBox} - * - * @param {AreaModelCommonModel} model - */ - function CalculationLinesNode( model ) { - const self = this; +import BooleanProperty from '../../../../../axon/js/BooleanProperty.js'; +import DerivedProperty from '../../../../../axon/js/DerivedProperty.js'; +import DynamicProperty from '../../../../../axon/js/DynamicProperty.js'; +import Emitter from '../../../../../axon/js/Emitter.js'; +import Property from '../../../../../axon/js/Property.js'; +import inherit from '../../../../../phet-core/js/inherit.js'; +import Orientation from '../../../../../phet-core/js/Orientation.js'; +import Node from '../../../../../scenery/js/nodes/Node.js'; +import VBox from '../../../../../scenery/js/nodes/VBox.js'; +import areaModelCommon from '../../../areaModelCommon.js'; +import AreaModelCommonA11yStrings from '../../../AreaModelCommonA11yStrings.js'; +import AreaModelCommonQueryParameters from '../../AreaModelCommonQueryParameters.js'; +import AreaCalculationChoice from '../../model/AreaCalculationChoice.js'; +import TermList from '../../model/TermList.js'; +import DistributionLine from './DistributionLine.js'; +import ExpandedLine from './ExpandedLine.js'; +import MinusesLine from './MinusesLine.js'; +import MultipliedLine from './MultipliedLine.js'; +import OrderedLine from './OrderedLine.js'; +import QuestionMarkLine from './QuestionMarkLine.js'; +import SumLine from './SumLine.js'; +import TotalsLine from './TotalsLine.js'; + +// a11y strings +const betweenCalculationLinesString = AreaModelCommonA11yStrings.betweenCalculationLines.value; + +/** + * @constructor + * @extends {VBox} + * + * @param {AreaModelCommonModel} model + */ +function CalculationLinesNode( model ) { + const self = this; - Node.call( this ); + Node.call( this ); - // @private {Node} - this.box = new VBox( { - spacing: 1 - } ); - this.addChild( this.box ); + // @private {Node} + this.box = new VBox( { + spacing: 1 + } ); + this.addChild( this.box ); - if ( !AreaModelCommonQueryParameters.rawMath ) { - this.accessibleNamespace = 'http://www.w3.org/1998/Math/MathML'; - this.tagName = 'math'; + if ( !AreaModelCommonQueryParameters.rawMath ) { + this.accessibleNamespace = 'http://www.w3.org/1998/Math/MathML'; + this.tagName = 'math'; - this.box.accessibleNamespace = 'http://www.w3.org/1998/Math/MathML'; - this.box.tagName = 'mtable'; - } + this.box.accessibleNamespace = 'http://www.w3.org/1998/Math/MathML'; + this.box.tagName = 'mtable'; + } - // @public {Property.} - Whether there are previous/next lines (when in line-by-line mode) - this.previousEnabledProperty = new BooleanProperty( false ); - this.nextEnabledProperty = new BooleanProperty( false ); + // @public {Property.} - Whether there are previous/next lines (when in line-by-line mode) + this.previousEnabledProperty = new BooleanProperty( false ); + this.nextEnabledProperty = new BooleanProperty( false ); - // @public {Property.>} - All of our "current" lines - this.calculationLinesProperty = new Property( [] ); + // @public {Property.>} - All of our "current" lines + this.calculationLinesProperty = new Property( [] ); - // @public {Emitter} - Fired whenever the displayed appearance has updated. - this.displayUpdatedEmitter = new Emitter(); + // @public {Emitter} - Fired whenever the displayed appearance has updated. + this.displayUpdatedEmitter = new Emitter(); - // @private {AreaModelCommonModel} - this.model = model; + // @private {AreaModelCommonModel} + this.model = model; - // @private {boolean} - Whether the actual CalculationLinesNode need updating. - this.linesDirty = true; + // @private {boolean} - Whether the actual CalculationLinesNode need updating. + this.linesDirty = true; - // @private {boolean} - Whether the display of the lines (index/visibility change) needs updating. - this.displayDirty = true; + // @private {boolean} - Whether the display of the lines (index/visibility change) needs updating. + this.displayDirty = true; - // @private {Property.} - The current index (for whatever area) - this.areaIndexProperty = new DynamicProperty( model.currentAreaProperty, { - derive: 'calculationIndexProperty', - bidirectional: true + // @private {Property.} - The current index (for whatever area) + this.areaIndexProperty = new DynamicProperty( model.currentAreaProperty, { + derive: 'calculationIndexProperty', + bidirectional: true + } ); + + // @private {Property.} - The effective current index (for whatever area) that we will use for display + this.effectiveIndexProperty = new DerivedProperty( + [ this.areaIndexProperty, model.areaCalculationChoiceProperty ], + function( index, choice ) { + return choice === AreaCalculationChoice.LINE_BY_LINE ? index : null; } ); - // @private {Property.} - The effective current index (for whatever area) that we will use for display - this.effectiveIndexProperty = new DerivedProperty( - [ this.areaIndexProperty, model.areaCalculationChoiceProperty ], - function( index, choice ) { - return choice === AreaCalculationChoice.LINE_BY_LINE ? index : null; + const setLinesDirty = function() { self.linesDirty = true; }; + const setDisplayDirty = function() { self.displayDirty = true; }; + + // Listen for changes that would make the display need an update + model.areaCalculationChoiceProperty.lazyLink( setDisplayDirty ); + this.areaIndexProperty.lazyLink( setDisplayDirty ); + + // Listen for changes that would make everything need an update + model.currentAreaProperty.link( function( newArea, oldArea ) { + if ( oldArea ) { + oldArea.allPartitions.forEach( function( partition ) { + partition.sizeProperty.unlink( setLinesDirty ); + partition.visibleProperty.unlink( setLinesDirty ); } ); + } - const setLinesDirty = function() { self.linesDirty = true; }; - const setDisplayDirty = function() { self.displayDirty = true; }; + newArea.allPartitions.forEach( function( partition ) { + partition.sizeProperty.lazyLink( setLinesDirty ); + partition.visibleProperty.lazyLink( setLinesDirty ); + } ); - // Listen for changes that would make the display need an update - model.areaCalculationChoiceProperty.lazyLink( setDisplayDirty ); - this.areaIndexProperty.lazyLink( setDisplayDirty ); + setLinesDirty(); - // Listen for changes that would make everything need an update - model.currentAreaProperty.link( function( newArea, oldArea ) { - if ( oldArea ) { - oldArea.allPartitions.forEach( function( partition ) { - partition.sizeProperty.unlink( setLinesDirty ); - partition.visibleProperty.unlink( setLinesDirty ); - } ); - } + self.update(); + } ); +} - newArea.allPartitions.forEach( function( partition ) { - partition.sizeProperty.lazyLink( setLinesDirty ); - partition.visibleProperty.lazyLink( setLinesDirty ); - } ); +areaModelCommon.register( 'CalculationLinesNode', CalculationLinesNode ); - setLinesDirty(); +export default inherit( Node, CalculationLinesNode, { + /** + * Called whenever the calculation may need an update. + * @public + */ + update: function() { + // Don't update anything if things are hidden + if ( this.model.areaCalculationChoiceProperty.value === AreaCalculationChoice.HIDDEN ) { + return; + } - self.update(); - } ); - } + this.updateLines(); + this.updateDisplay(); + }, - areaModelCommon.register( 'CalculationLinesNode', CalculationLinesNode ); - - return inherit( Node, CalculationLinesNode, { - /** - * Called whenever the calculation may need an update. - * @public - */ - update: function() { - // Don't update anything if things are hidden - if ( this.model.areaCalculationChoiceProperty.value === AreaCalculationChoice.HIDDEN ) { - return; - } + /** + * Moves the display to the previous line. + * @public + */ + moveToPreviousLine: function() { + const activeLine = this.getActiveLine(); + if ( activeLine.previousLine ) { + this.areaIndexProperty.value = activeLine.previousLine.index; + } + }, - this.updateLines(); - this.updateDisplay(); - }, + /** + * Moves the display to the next line. + * @public + */ + moveToNextLine: function() { + const activeLine = this.getActiveLine(); + if ( activeLine.nextLine ) { + this.areaIndexProperty.value = activeLine.nextLine.index; + } + }, - /** - * Moves the display to the previous line. - * @public - */ - moveToPreviousLine: function() { - const activeLine = this.getActiveLine(); - if ( activeLine.previousLine ) { - this.areaIndexProperty.value = activeLine.previousLine.index; - } - }, + /** + * Removes and disposes children. + * @private + */ + wipe: function() { + while ( this.box.children.length ) { + this.box.children[ 0 ].dispose(); + } + }, - /** - * Moves the display to the next line. - * @public - */ - moveToNextLine: function() { - const activeLine = this.getActiveLine(); - if ( activeLine.nextLine ) { - this.areaIndexProperty.value = activeLine.nextLine.index; - } - }, - - /** - * Removes and disposes children. - * @private - */ - wipe: function() { - while ( this.box.children.length ) { - this.box.children[ 0 ].dispose(); - } - }, - - /** - * Update the internally-stored calculation lines. - * @private - */ - updateLines: function() { - if ( !this.linesDirty ) { - return; - } + /** + * Update the internally-stored calculation lines. + * @private + */ + updateLines: function() { + if ( !this.linesDirty ) { + return; + } - // As a sanity check, just remove all children here (so we don't leak things) - this.wipe(); + // As a sanity check, just remove all children here (so we don't leak things) + this.wipe(); - // Release line references that we had before - this.calculationLinesProperty.value.forEach( function( calculationLine ) { - calculationLine.dispose(); - } ); + // Release line references that we had before + this.calculationLinesProperty.value.forEach( function( calculationLine ) { + calculationLine.dispose(); + } ); - // Create new lines - this.calculationLinesProperty.value = CalculationLinesNode.createLines( - this.model.currentAreaProperty.value, - this.effectiveIndexProperty, - this.model.allowExponents, - this.model.isProportional - ); - - this.linesDirty = false; - this.displayDirty = true; - }, - - /** - * Update the display of the calculation lines. - * @private - */ - updateDisplay: function() { - if ( !this.displayDirty ) { - return; - } + // Create new lines + this.calculationLinesProperty.value = CalculationLinesNode.createLines( + this.model.currentAreaProperty.value, + this.effectiveIndexProperty, + this.model.allowExponents, + this.model.isProportional + ); + + this.linesDirty = false; + this.displayDirty = true; + }, + + /** + * Update the display of the calculation lines. + * @private + */ + updateDisplay: function() { + if ( !this.displayDirty ) { + return; + } + + // As a sanity check, just remove all children here (so we don't leak things) + this.wipe(); - // As a sanity check, just remove all children here (so we don't leak things) - this.wipe(); + let displayedLines = this.calculationLinesProperty.value; - let displayedLines = this.calculationLinesProperty.value; + // If we are in line-by-line mode, display adjacent lines + if ( this.model.areaCalculationChoiceProperty.value === AreaCalculationChoice.LINE_BY_LINE ) { - // If we are in line-by-line mode, display adjacent lines - if ( this.model.areaCalculationChoiceProperty.value === AreaCalculationChoice.LINE_BY_LINE ) { + const activeLine = this.getActiveLine(); + displayedLines = activeLine.getAdjacentLines(); - const activeLine = this.getActiveLine(); - displayedLines = activeLine.getAdjacentLines(); + this.previousEnabledProperty.value = !!activeLine.previousLine; + this.nextEnabledProperty.value = !!activeLine.nextLine; + } + else { + this.previousEnabledProperty.value = false; + this.nextEnabledProperty.value = false; + } - this.previousEnabledProperty.value = !!activeLine.previousLine; - this.nextEnabledProperty.value = !!activeLine.nextLine; + this.box.children = displayedLines.map( function( line, index ) { + const lineNode = new Node( { + children: [ + line.node + ] + } ); + if ( AreaModelCommonQueryParameters.rawMath ) { + lineNode.tagName = 'span'; + lineNode.innerContent = line.node.accessibleText; + lineNode.containerTagName = 'span'; + line.node.accessibleVisible = false; } else { - this.previousEnabledProperty.value = false; - this.nextEnabledProperty.value = false; + lineNode.accessibleNamespace = 'http://www.w3.org/1998/Math/MathML'; + lineNode.tagName = 'mtr'; } - - this.box.children = displayedLines.map( function( line, index ) { - const lineNode = new Node( { - children: [ - line.node - ] - } ); + if ( index > 0 ) { if ( AreaModelCommonQueryParameters.rawMath ) { - lineNode.tagName = 'span'; - lineNode.innerContent = line.node.accessibleText; - lineNode.containerTagName = 'span'; - line.node.accessibleVisible = false; + lineNode.labelTagName = 'span'; + lineNode.labelContent = betweenCalculationLinesString; } else { - lineNode.accessibleNamespace = 'http://www.w3.org/1998/Math/MathML'; - lineNode.tagName = 'mtr'; + lineNode.insertChild( 0, new Node( { + // a11y + tagName: 'mtext', + accessibleNamespace: 'http://www.w3.org/1998/Math/MathML', + innerContent: betweenCalculationLinesString + } ) ); } - if ( index > 0 ) { - if ( AreaModelCommonQueryParameters.rawMath ) { - lineNode.labelTagName = 'span'; - lineNode.labelContent = betweenCalculationLinesString; - } - else { - lineNode.insertChild( 0, new Node( { - // a11y - tagName: 'mtext', - accessibleNamespace: 'http://www.w3.org/1998/Math/MathML', - innerContent: betweenCalculationLinesString - } ) ); - } + } + return lineNode; + } ); + + this.displayDirty = false; + this.displayUpdatedEmitter.emit(); + }, + + /** + * Returns the first active line, or null otherwise. + * @private + * + * @returns {CalculationLine|null} + */ + getActiveLine: function() { + let activeLine = _.find( this.calculationLinesProperty.value, function( line ) { + return line.isActiveProperty.value; + } ) || null; + + // If no line is currently active (maybe it was removed?), switch to the next-best line + if ( !activeLine ) { + let nextBestLine = null; + const lastIndex = this.areaIndexProperty.value; + this.calculationLinesProperty.value.forEach( function( calculationLine ) { + if ( calculationLine.index <= lastIndex ) { + nextBestLine = calculationLine; } - return lineNode; } ); - this.displayDirty = false; - this.displayUpdatedEmitter.emit(); - }, - - /** - * Returns the first active line, or null otherwise. - * @private - * - * @returns {CalculationLine|null} - */ - getActiveLine: function() { - let activeLine = _.find( this.calculationLinesProperty.value, function( line ) { - return line.isActiveProperty.value; - } ) || null; - - // If no line is currently active (maybe it was removed?), switch to the next-best line - if ( !activeLine ) { - let nextBestLine = null; - const lastIndex = this.areaIndexProperty.value; - this.calculationLinesProperty.value.forEach( function( calculationLine ) { - if ( calculationLine.index <= lastIndex ) { - nextBestLine = calculationLine; - } - } ); - - // Update the index property to point to the correct line - this.areaIndexProperty.value = nextBestLine.index; - activeLine = nextBestLine; - } + // Update the index property to point to the correct line + this.areaIndexProperty.value = nextBestLine.index; + activeLine = nextBestLine; + } - return activeLine; + return activeLine; + } +}, { + /** + * Creates an array of calculation lines. + * @private + * + * @param {Area} area + * @param {Property.} activeIndexProperty - null when all lines should be active + * @param {boolean} allowExponents - Whether exponents (powers of x) are allowed + * @param {boolean} isProportional - Whether the area is shown as proportional (instead of generic) + * @returns {Array.} + */ + createLines: function( area, activeIndexProperty, allowExponents, isProportional ) { + // Whether there are ANY shown partitions for a given orientation + const horizontalEmpty = area.getDefinedPartitions( Orientation.HORIZONTAL ).length === 0; + const verticalEmpty = area.getDefinedPartitions( Orientation.VERTICAL ).length === 0; + + // If both are empty, show a question mark + if ( horizontalEmpty && verticalEmpty ) { + return [ new QuestionMarkLine( area, activeIndexProperty, allowExponents, isProportional ) ]; + } + // If only one is empty, show boxes + else if ( horizontalEmpty || verticalEmpty ) { + return [ new TotalsLine( area, activeIndexProperty, allowExponents, isProportional ) ]; } - }, { - /** - * Creates an array of calculation lines. - * @private - * - * @param {Area} area - * @param {Property.} activeIndexProperty - null when all lines should be active - * @param {boolean} allowExponents - Whether exponents (powers of x) are allowed - * @param {boolean} isProportional - Whether the area is shown as proportional (instead of generic) - * @returns {Array.} - */ - createLines: function( area, activeIndexProperty, allowExponents, isProportional ) { - // Whether there are ANY shown partitions for a given orientation - const horizontalEmpty = area.getDefinedPartitions( Orientation.HORIZONTAL ).length === 0; - const verticalEmpty = area.getDefinedPartitions( Orientation.VERTICAL ).length === 0; - - // If both are empty, show a question mark - if ( horizontalEmpty && verticalEmpty ) { - return [ new QuestionMarkLine( area, activeIndexProperty, allowExponents, isProportional ) ]; - } - // If only one is empty, show boxes - else if ( horizontalEmpty || verticalEmpty ) { - return [ new TotalsLine( area, activeIndexProperty, allowExponents, isProportional ) ]; - } - const horizontalTermList = area.getTermList( Orientation.HORIZONTAL ); - const verticalTermList = area.getTermList( Orientation.VERTICAL ); + const horizontalTermList = area.getTermList( Orientation.HORIZONTAL ); + const verticalTermList = area.getTermList( Orientation.VERTICAL ); - const horizontalTerms = horizontalTermList.terms; - const verticalTerms = verticalTermList.terms; + const horizontalTerms = horizontalTermList.terms; + const verticalTerms = verticalTermList.terms; - // The total/sum for each orientation - const horizontalPolynomial = area.totalProperties.horizontal.value; - const verticalPolynomial = area.totalProperties.vertical.value; + // The total/sum for each orientation + const horizontalPolynomial = area.totalProperties.horizontal.value; + const verticalPolynomial = area.totalProperties.vertical.value; - // E.g. for ( 2 ) * ( 3 + x ), the result will be the terms 6 and 2x. - const multipliedTermList = new TermList( _.flatten( verticalTerms.map( function( verticalTerm ) { - return horizontalTerms.map( function( horizontalTerm ) { - return horizontalTerm.times( verticalTerm ); - } ); - } ) ) ); - const orderedTermList = multipliedTermList.orderedByExponent(); - const totalPolynomial = area.totalAreaProperty.value; + // E.g. for ( 2 ) * ( 3 + x ), the result will be the terms 6 and 2x. + const multipliedTermList = new TermList( _.flatten( verticalTerms.map( function( verticalTerm ) { + return horizontalTerms.map( function( horizontalTerm ) { + return horizontalTerm.times( verticalTerm ); + } ); + } ) ) ); + const orderedTermList = multipliedTermList.orderedByExponent(); + const totalPolynomial = area.totalAreaProperty.value; - // Logic for what calculation lines are needed - const needsExpansion = !allowExponents && ( !horizontalTermList.equals( horizontalPolynomial ) || + // Logic for what calculation lines are needed + const needsExpansion = !allowExponents && ( !horizontalTermList.equals( horizontalPolynomial ) || !verticalTermList.equals( verticalPolynomial ) ); - const needsDistribution = horizontalTermList.terms.length !== 1 || verticalTermList.terms.length !== 1; - const needsMultiplied = needsDistribution && !multipliedTermList.equals( totalPolynomial ); - const needsOrdered = needsMultiplied && !orderedTermList.equals( multipliedTermList ) && + const needsDistribution = horizontalTermList.terms.length !== 1 || verticalTermList.terms.length !== 1; + const needsMultiplied = needsDistribution && !multipliedTermList.equals( totalPolynomial ); + const needsOrdered = needsMultiplied && !orderedTermList.equals( multipliedTermList ) && !( orderedTermList.equals( totalPolynomial ) && ( !allowExponents || !orderedTermList.hasNegativeTerm() ) ); - const needsMinuses = needsMultiplied && allowExponents && + const needsMinuses = needsMultiplied && allowExponents && orderedTermList.hasNegativeTerm() && !orderedTermList.equals( totalPolynomial ); - // Add the actual lines - const lines = []; - // e.g. ( -x + x^2 )( x^2 - x ) <--- example used for everything except the ExpansionLine - lines.push( new TotalsLine( area, activeIndexProperty, allowExponents, isProportional ) ); - if ( needsExpansion ) { - // e.g. ( -5 + 2 )( 7 + 3 ) <---- if we have a proportional one where Totals Line is e.g. -3 * 10 - lines.push( new ExpandedLine( horizontalTerms, verticalTerms, area, activeIndexProperty, allowExponents, isProportional ) ); - } - if ( needsDistribution ) { - // e.g. (-x)(x^2) + (-x)(-x) + (x^2)(x^2) + (x^2)(-x) - lines.push( new DistributionLine( horizontalTerms, verticalTerms, area, activeIndexProperty, allowExponents, isProportional ) ); - } - if ( needsMultiplied ) { - // e.g. (-x^3) + x^2 + x^4 + (-x^3) - lines.push( new MultipliedLine( multipliedTermList, area, activeIndexProperty, allowExponents, isProportional ) ); - } - if ( needsOrdered ) { - // e.g. x^4 + (-x^3) + (-x^3) + x^2 - lines.push( new OrderedLine( orderedTermList, area, activeIndexProperty, allowExponents, isProportional ) ); - } - if ( needsMinuses ) { - // e.g. x^4 - x^3 - x^3 + x^2 - lines.push( new MinusesLine( orderedTermList, area, activeIndexProperty, allowExponents, isProportional ) ); - } - // e.g. x^4 - 2x^3 + x^2 - lines.push( new SumLine( area, activeIndexProperty, allowExponents, isProportional ) ); - - // Link the lines together, so it is easy to traverse - for ( let i = 1; i < lines.length; i++ ) { - lines[ i - 1 ].nextLine = lines[ i ]; - lines[ i ].previousLine = lines[ i - 1 ]; - } + // Add the actual lines + const lines = []; + // e.g. ( -x + x^2 )( x^2 - x ) <--- example used for everything except the ExpansionLine + lines.push( new TotalsLine( area, activeIndexProperty, allowExponents, isProportional ) ); + if ( needsExpansion ) { + // e.g. ( -5 + 2 )( 7 + 3 ) <---- if we have a proportional one where Totals Line is e.g. -3 * 10 + lines.push( new ExpandedLine( horizontalTerms, verticalTerms, area, activeIndexProperty, allowExponents, isProportional ) ); + } + if ( needsDistribution ) { + // e.g. (-x)(x^2) + (-x)(-x) + (x^2)(x^2) + (x^2)(-x) + lines.push( new DistributionLine( horizontalTerms, verticalTerms, area, activeIndexProperty, allowExponents, isProportional ) ); + } + if ( needsMultiplied ) { + // e.g. (-x^3) + x^2 + x^4 + (-x^3) + lines.push( new MultipliedLine( multipliedTermList, area, activeIndexProperty, allowExponents, isProportional ) ); + } + if ( needsOrdered ) { + // e.g. x^4 + (-x^3) + (-x^3) + x^2 + lines.push( new OrderedLine( orderedTermList, area, activeIndexProperty, allowExponents, isProportional ) ); + } + if ( needsMinuses ) { + // e.g. x^4 - x^3 - x^3 + x^2 + lines.push( new MinusesLine( orderedTermList, area, activeIndexProperty, allowExponents, isProportional ) ); + } + // e.g. x^4 - 2x^3 + x^2 + lines.push( new SumLine( area, activeIndexProperty, allowExponents, isProportional ) ); - return lines; + // Link the lines together, so it is easy to traverse + for ( let i = 1; i < lines.length; i++ ) { + lines[ i - 1 ].nextLine = lines[ i ]; + lines[ i ].previousLine = lines[ i - 1 ]; } - } ); -} ); + + return lines; + } +} ); \ No newline at end of file diff --git a/js/common/view/calculation/DistributionLine.js b/js/common/view/calculation/DistributionLine.js index b8742bf6..5d9a506f 100644 --- a/js/common/view/calculation/DistributionLine.js +++ b/js/common/view/calculation/DistributionLine.js @@ -5,56 +5,53 @@ * * @author Jonathan Olson */ -define( require => { - 'use strict'; - - // modules - const areaModelCommon = require( 'AREA_MODEL_COMMON/areaModelCommon' ); - const AreaModelCommonConstants = require( 'AREA_MODEL_COMMON/common/AreaModelCommonConstants' ); - const CalculationLine = require( 'AREA_MODEL_COMMON/common/view/calculation/CalculationLine' ); - const inherit = require( 'PHET_CORE/inherit' ); - const Orientation = require( 'PHET_CORE/Orientation' ); - - /** - * @constructor - * @extends {CalculationLine} - * - * @param {Array.} horizontalTerms - * @param {Array.} verticalTerms - * @param {Area} area - * @param {Property.} activeIndexProperty - * @param {boolean} allowExponents - Whether exponents (powers of x) are allowed - * @param {boolean} isProportional - Whether the area is shown as proportional (instead of generic) - */ - function DistributionLine( horizontalTerms, verticalTerms, area, activeIndexProperty, allowExponents, isProportional ) { - const self = this; - - CalculationLine.call( this, CalculationLine.DISTRIBUTION_LINE_INDEX, area.colorProperties, activeIndexProperty, allowExponents, isProportional ); - - this.node = this.sumGroup( _.flatten( verticalTerms.map( function( verticalTerm ) { - return horizontalTerms.map( function( horizontalTerm ) { - const horizontalText = self.orientedTermText( Orientation.HORIZONTAL, horizontalTerm ); - const verticalText = self.orientedTermText( Orientation.VERTICAL, verticalTerm ); - - // Proportional uses X-multiplication, see https://github.com/phetsims/area-model-common/issues/71 - if ( isProportional ) { - return self.parentheses( self.multiplyX( verticalText, horizontalText ) ); - } - else if ( allowExponents ) { - return self.group( [ - self.parentheses( verticalText ), - self.parentheses( horizontalText ) - ], AreaModelCommonConstants.CALCULATION_PAREN_PAREN_PADDING ); - } - // Generic Screen (non-proportional, no exponents) uses dot, see https://github.com/phetsims/area-model-common/issues/72 - else { - return self.parentheses( self.multiplyX( verticalText, horizontalText ) ); - } - } ); - } ) ) ); - } - - areaModelCommon.register( 'DistributionLine', DistributionLine ); - - return inherit( CalculationLine, DistributionLine ); -} ); + +import inherit from '../../../../../phet-core/js/inherit.js'; +import Orientation from '../../../../../phet-core/js/Orientation.js'; +import areaModelCommon from '../../../areaModelCommon.js'; +import AreaModelCommonConstants from '../../AreaModelCommonConstants.js'; +import CalculationLine from './CalculationLine.js'; + +/** + * @constructor + * @extends {CalculationLine} + * + * @param {Array.} horizontalTerms + * @param {Array.} verticalTerms + * @param {Area} area + * @param {Property.} activeIndexProperty + * @param {boolean} allowExponents - Whether exponents (powers of x) are allowed + * @param {boolean} isProportional - Whether the area is shown as proportional (instead of generic) + */ +function DistributionLine( horizontalTerms, verticalTerms, area, activeIndexProperty, allowExponents, isProportional ) { + const self = this; + + CalculationLine.call( this, CalculationLine.DISTRIBUTION_LINE_INDEX, area.colorProperties, activeIndexProperty, allowExponents, isProportional ); + + this.node = this.sumGroup( _.flatten( verticalTerms.map( function( verticalTerm ) { + return horizontalTerms.map( function( horizontalTerm ) { + const horizontalText = self.orientedTermText( Orientation.HORIZONTAL, horizontalTerm ); + const verticalText = self.orientedTermText( Orientation.VERTICAL, verticalTerm ); + + // Proportional uses X-multiplication, see https://github.com/phetsims/area-model-common/issues/71 + if ( isProportional ) { + return self.parentheses( self.multiplyX( verticalText, horizontalText ) ); + } + else if ( allowExponents ) { + return self.group( [ + self.parentheses( verticalText ), + self.parentheses( horizontalText ) + ], AreaModelCommonConstants.CALCULATION_PAREN_PAREN_PADDING ); + } + // Generic Screen (non-proportional, no exponents) uses dot, see https://github.com/phetsims/area-model-common/issues/72 + else { + return self.parentheses( self.multiplyX( verticalText, horizontalText ) ); + } + } ); + } ) ) ); +} + +areaModelCommon.register( 'DistributionLine', DistributionLine ); + +inherit( CalculationLine, DistributionLine ); +export default DistributionLine; \ No newline at end of file diff --git a/js/common/view/calculation/ExpandedLine.js b/js/common/view/calculation/ExpandedLine.js index 7a0fd2a5..2c305ba5 100644 --- a/js/common/view/calculation/ExpandedLine.js +++ b/js/common/view/calculation/ExpandedLine.js @@ -6,55 +6,52 @@ * * @author Jonathan Olson */ -define( require => { - 'use strict'; - - // modules - const areaModelCommon = require( 'AREA_MODEL_COMMON/areaModelCommon' ); - const AreaModelCommonConstants = require( 'AREA_MODEL_COMMON/common/AreaModelCommonConstants' ); - const CalculationLine = require( 'AREA_MODEL_COMMON/common/view/calculation/CalculationLine' ); - const inherit = require( 'PHET_CORE/inherit' ); - const Orientation = require( 'PHET_CORE/Orientation' ); - - /** - * @constructor - * @extends {CalculationLine} - * - * @param {Array.} horizontalTerms - * @param {Array.} verticalTerms - * @param {Area} area - * @param {Property.} activeIndexProperty - * @param {boolean} allowExponents - Whether exponents (powers of x) are allowed - * @param {boolean} isProportional - Whether the area is shown as proportional (instead of generic) - */ - function ExpandedLine( horizontalTerms, verticalTerms, area, activeIndexProperty, allowExponents, isProportional ) { - CalculationLine.call( this, CalculationLine.EXPANDED_LINE_INDEX, area.colorProperties, activeIndexProperty, allowExponents, isProportional ); - - const isHorizontalSingle = horizontalTerms.length === 1; - const isVerticalSingle = verticalTerms.length === 1; - - let horizontalNode = this.sumOrientedTerms( horizontalTerms, Orientation.HORIZONTAL ); - let verticalNode = this.sumOrientedTerms( verticalTerms, Orientation.VERTICAL ); - - if ( !isHorizontalSingle || allowExponents ) { - horizontalNode = this.parentheses( horizontalNode ); - } - if ( !isVerticalSingle || allowExponents ) { - verticalNode = this.parentheses( verticalNode ); - } - - if ( isProportional ) { - this.node = this.multiplyX( verticalNode, horizontalNode ); - } - else { - const spacing = ( isHorizontalSingle || isVerticalSingle ) + +import inherit from '../../../../../phet-core/js/inherit.js'; +import Orientation from '../../../../../phet-core/js/Orientation.js'; +import areaModelCommon from '../../../areaModelCommon.js'; +import AreaModelCommonConstants from '../../AreaModelCommonConstants.js'; +import CalculationLine from './CalculationLine.js'; + +/** + * @constructor + * @extends {CalculationLine} + * + * @param {Array.} horizontalTerms + * @param {Array.} verticalTerms + * @param {Area} area + * @param {Property.} activeIndexProperty + * @param {boolean} allowExponents - Whether exponents (powers of x) are allowed + * @param {boolean} isProportional - Whether the area is shown as proportional (instead of generic) + */ +function ExpandedLine( horizontalTerms, verticalTerms, area, activeIndexProperty, allowExponents, isProportional ) { + CalculationLine.call( this, CalculationLine.EXPANDED_LINE_INDEX, area.colorProperties, activeIndexProperty, allowExponents, isProportional ); + + const isHorizontalSingle = horizontalTerms.length === 1; + const isVerticalSingle = verticalTerms.length === 1; + + let horizontalNode = this.sumOrientedTerms( horizontalTerms, Orientation.HORIZONTAL ); + let verticalNode = this.sumOrientedTerms( verticalTerms, Orientation.VERTICAL ); + + if ( !isHorizontalSingle || allowExponents ) { + horizontalNode = this.parentheses( horizontalNode ); + } + if ( !isVerticalSingle || allowExponents ) { + verticalNode = this.parentheses( verticalNode ); + } + + if ( isProportional ) { + this.node = this.multiplyX( verticalNode, horizontalNode ); + } + else { + const spacing = ( isHorizontalSingle || isVerticalSingle ) ? AreaModelCommonConstants.CALCULATION_TERM_PAREN_PADDING : AreaModelCommonConstants.CALCULATION_PAREN_PAREN_PADDING; - this.node = this.group( [ verticalNode, horizontalNode ], spacing ); - } + this.node = this.group( [ verticalNode, horizontalNode ], spacing ); } +} - areaModelCommon.register( 'ExpandedLine', ExpandedLine ); +areaModelCommon.register( 'ExpandedLine', ExpandedLine ); - return inherit( CalculationLine, ExpandedLine ); -} ); +inherit( CalculationLine, ExpandedLine ); +export default ExpandedLine; \ No newline at end of file diff --git a/js/common/view/calculation/Minus.js b/js/common/view/calculation/Minus.js index c3f1be2c..7723e1cc 100644 --- a/js/common/view/calculation/Minus.js +++ b/js/common/view/calculation/Minus.js @@ -7,65 +7,61 @@ * * @author Jonathan Olson */ -define( require => { - 'use strict'; - // modules - const areaModelCommon = require( 'AREA_MODEL_COMMON/areaModelCommon' ); - const AreaModelCommonA11yStrings = require( 'AREA_MODEL_COMMON/AreaModelCommonA11yStrings' ); - const AreaModelCommonConstants = require( 'AREA_MODEL_COMMON/common/AreaModelCommonConstants' ); - const inherit = require( 'PHET_CORE/inherit' ); - const MathSymbols = require( 'SCENERY_PHET/MathSymbols' ); - const Poolable = require( 'PHET_CORE/Poolable' ); - const Property = require( 'AXON/Property' ); - const Text = require( 'SCENERY/nodes/Text' ); +import Property from '../../../../../axon/js/Property.js'; +import inherit from '../../../../../phet-core/js/inherit.js'; +import Poolable from '../../../../../phet-core/js/Poolable.js'; +import MathSymbols from '../../../../../scenery-phet/js/MathSymbols.js'; +import Text from '../../../../../scenery/js/nodes/Text.js'; +import areaModelCommon from '../../../areaModelCommon.js'; +import AreaModelCommonA11yStrings from '../../../AreaModelCommonA11yStrings.js'; +import AreaModelCommonConstants from '../../AreaModelCommonConstants.js'; - // a11y strings - const sumMinusString = AreaModelCommonA11yStrings.sumMinus.value; +// a11y strings +const sumMinusString = AreaModelCommonA11yStrings.sumMinus.value; - /** - * @constructor - * @extends {Text} - * - * @param {Property.} baseColorProperty - */ - function Minus( baseColorProperty ) { - assert && assert( baseColorProperty instanceof Property ); - - // @public {string} - this.accessibleText = sumMinusString; +/** + * @constructor + * @extends {Text} + * + * @param {Property.} baseColorProperty + */ +function Minus( baseColorProperty ) { + assert && assert( baseColorProperty instanceof Property ); - if ( !this.initialized ) { - this.initialized = true; + // @public {string} + this.accessibleText = sumMinusString; - Text.call( this, MathSymbols.MINUS, { - font: AreaModelCommonConstants.CALCULATION_PAREN_FONT, + if ( !this.initialized ) { + this.initialized = true; - // a11y - tagName: 'mo', - accessibleNamespace: 'http://www.w3.org/1998/Math/MathML', - innerContent: '−' - } ); - } + Text.call( this, MathSymbols.MINUS, { + font: AreaModelCommonConstants.CALCULATION_PAREN_FONT, - this.fill = baseColorProperty; + // a11y + tagName: 'mo', + accessibleNamespace: 'http://www.w3.org/1998/Math/MathML', + innerContent: '−' + } ); } - areaModelCommon.register( 'Minus', Minus ); + this.fill = baseColorProperty; +} - inherit( Text, Minus, { - /** - * Clears the state of this node (releasing references) so it can be freed to the pool (and potentially GC'ed). - * @public - */ - clean: function() { - this.fill = null; +areaModelCommon.register( 'Minus', Minus ); - this.freeToPool(); - } - } ); - - Poolable.mixInto( Minus ); +inherit( Text, Minus, { + /** + * Clears the state of this node (releasing references) so it can be freed to the pool (and potentially GC'ed). + * @public + */ + clean: function() { + this.fill = null; - return Minus; + this.freeToPool(); + } } ); + +Poolable.mixInto( Minus ); + +export default Minus; \ No newline at end of file diff --git a/js/common/view/calculation/MinusesLine.js b/js/common/view/calculation/MinusesLine.js index 7106beff..5f323901 100644 --- a/js/common/view/calculation/MinusesLine.js +++ b/js/common/view/calculation/MinusesLine.js @@ -6,31 +6,28 @@ * * @author Jonathan Olson */ -define( require => { - 'use strict'; - // modules - const areaModelCommon = require( 'AREA_MODEL_COMMON/areaModelCommon' ); - const CalculationLine = require( 'AREA_MODEL_COMMON/common/view/calculation/CalculationLine' ); - const inherit = require( 'PHET_CORE/inherit' ); +import inherit from '../../../../../phet-core/js/inherit.js'; +import areaModelCommon from '../../../areaModelCommon.js'; +import CalculationLine from './CalculationLine.js'; - /** - * @constructor - * @extends {CalculationLine} - * - * @param {TermList} orderedTermList - * @param {Area} area - * @param {Property.} activeIndexProperty - * @param {boolean} allowExponents - Whether exponents (powers of x) are allowed - * @param {boolean} isProportional - Whether the area is shown as proportional (instead of generic) - */ - function MinusesLine( orderedTermList, area, activeIndexProperty, allowExponents, isProportional ) { - CalculationLine.call( this, CalculationLine.MINUSES_LINE_INDEX, area.colorProperties, activeIndexProperty, allowExponents, isProportional ); +/** + * @constructor + * @extends {CalculationLine} + * + * @param {TermList} orderedTermList + * @param {Area} area + * @param {Property.} activeIndexProperty + * @param {boolean} allowExponents - Whether exponents (powers of x) are allowed + * @param {boolean} isProportional - Whether the area is shown as proportional (instead of generic) + */ +function MinusesLine( orderedTermList, area, activeIndexProperty, allowExponents, isProportional ) { + CalculationLine.call( this, CalculationLine.MINUSES_LINE_INDEX, area.colorProperties, activeIndexProperty, allowExponents, isProportional ); - this.node = this.sumOrDifferenceOfTerms( orderedTermList.terms ); - } + this.node = this.sumOrDifferenceOfTerms( orderedTermList.terms ); +} - areaModelCommon.register( 'MinusesLine', MinusesLine ); +areaModelCommon.register( 'MinusesLine', MinusesLine ); - return inherit( CalculationLine, MinusesLine ); -} ); +inherit( CalculationLine, MinusesLine ); +export default MinusesLine; \ No newline at end of file diff --git a/js/common/view/calculation/MultipliedLine.js b/js/common/view/calculation/MultipliedLine.js index 92700e23..07e89f15 100644 --- a/js/common/view/calculation/MultipliedLine.js +++ b/js/common/view/calculation/MultipliedLine.js @@ -6,31 +6,28 @@ * * @author Jonathan Olson */ -define( require => { - 'use strict'; - // modules - const areaModelCommon = require( 'AREA_MODEL_COMMON/areaModelCommon' ); - const CalculationLine = require( 'AREA_MODEL_COMMON/common/view/calculation/CalculationLine' ); - const inherit = require( 'PHET_CORE/inherit' ); +import inherit from '../../../../../phet-core/js/inherit.js'; +import areaModelCommon from '../../../areaModelCommon.js'; +import CalculationLine from './CalculationLine.js'; - /** - * @constructor - * @extends {CalculationLine} - * - * @param {TermList} multipliedTermList - * @param {Area} area - * @param {Property.} activeIndexProperty - * @param {boolean} allowExponents - Whether exponents (powers of x) are allowed - * @param {boolean} isProportional - Whether the area is shown as proportional (instead of generic) - */ - function MultipliedLine( multipliedTermList, area, activeIndexProperty, allowExponents, isProportional ) { - CalculationLine.call( this, CalculationLine.MULTIPLIED_LINE_INDEX, area.colorProperties, activeIndexProperty, allowExponents, isProportional ); +/** + * @constructor + * @extends {CalculationLine} + * + * @param {TermList} multipliedTermList + * @param {Area} area + * @param {Property.} activeIndexProperty + * @param {boolean} allowExponents - Whether exponents (powers of x) are allowed + * @param {boolean} isProportional - Whether the area is shown as proportional (instead of generic) + */ +function MultipliedLine( multipliedTermList, area, activeIndexProperty, allowExponents, isProportional ) { + CalculationLine.call( this, CalculationLine.MULTIPLIED_LINE_INDEX, area.colorProperties, activeIndexProperty, allowExponents, isProportional ); - this.node = this.sumWithNegativeParens( multipliedTermList.terms ); - } + this.node = this.sumWithNegativeParens( multipliedTermList.terms ); +} - areaModelCommon.register( 'MultipliedLine', MultipliedLine ); +areaModelCommon.register( 'MultipliedLine', MultipliedLine ); - return inherit( CalculationLine, MultipliedLine ); -} ); +inherit( CalculationLine, MultipliedLine ); +export default MultipliedLine; \ No newline at end of file diff --git a/js/common/view/calculation/MultiplyX.js b/js/common/view/calculation/MultiplyX.js index 198ea79a..7d5f52a1 100644 --- a/js/common/view/calculation/MultiplyX.js +++ b/js/common/view/calculation/MultiplyX.js @@ -7,102 +7,98 @@ * * @author Jonathan Olson */ -define( require => { - 'use strict'; - - // modules - const areaModelCommon = require( 'AREA_MODEL_COMMON/areaModelCommon' ); - const AreaModelCommonA11yStrings = require( 'AREA_MODEL_COMMON/AreaModelCommonA11yStrings' ); - const AreaModelCommonConstants = require( 'AREA_MODEL_COMMON/common/AreaModelCommonConstants' ); - const HBox = require( 'SCENERY/nodes/HBox' ); - const inherit = require( 'PHET_CORE/inherit' ); - const MathSymbols = require( 'SCENERY_PHET/MathSymbols' ); - const Node = require( 'SCENERY/nodes/Node' ); - const Poolable = require( 'PHET_CORE/Poolable' ); - const Property = require( 'AXON/Property' ); - const StringUtils = require( 'PHETCOMMON/util/StringUtils' ); - const Text = require( 'SCENERY/nodes/Text' ); - - // a11y strings - const productTimesPatternString = AreaModelCommonA11yStrings.productTimesPattern.value; - /** - * @constructor - * @extends {HBox} - * - * @param {Node} leftContent - Should have a clean() method to support pooling - * @param {Node} rightContent - Should have a clean() method to support pooling - * @param {Property.} baseColorProperty - */ - function MultiplyX( leftContent, rightContent, baseColorProperty ) { - assert && assert( leftContent instanceof Node ); - assert && assert( rightContent instanceof Node ); - assert && assert( baseColorProperty instanceof Property ); - - // @public {string} - this.accessibleText = StringUtils.fillIn( productTimesPatternString, { - left: leftContent.accessibleText, - right: rightContent.accessibleText - } ); - - // @private {Node|null} - this.leftContent = leftContent; - this.rightContent = rightContent; - - if ( !this.initialized ) { - this.initialized = true; +import Property from '../../../../../axon/js/Property.js'; +import inherit from '../../../../../phet-core/js/inherit.js'; +import Poolable from '../../../../../phet-core/js/Poolable.js'; +import StringUtils from '../../../../../phetcommon/js/util/StringUtils.js'; +import MathSymbols from '../../../../../scenery-phet/js/MathSymbols.js'; +import HBox from '../../../../../scenery/js/nodes/HBox.js'; +import Node from '../../../../../scenery/js/nodes/Node.js'; +import Text from '../../../../../scenery/js/nodes/Text.js'; +import areaModelCommon from '../../../areaModelCommon.js'; +import AreaModelCommonA11yStrings from '../../../AreaModelCommonA11yStrings.js'; +import AreaModelCommonConstants from '../../AreaModelCommonConstants.js'; + +// a11y strings +const productTimesPatternString = AreaModelCommonA11yStrings.productTimesPattern.value; - // @private {Text} - Persistent (since it's declared in the constructor instead of the initialize function, this - // will persist for the life of this node). - this.timesNode = new Text( MathSymbols.TIMES, { - font: AreaModelCommonConstants.CALCULATION_X_FONT, +/** + * @constructor + * @extends {HBox} + * + * @param {Node} leftContent - Should have a clean() method to support pooling + * @param {Node} rightContent - Should have a clean() method to support pooling + * @param {Property.} baseColorProperty + */ +function MultiplyX( leftContent, rightContent, baseColorProperty ) { + assert && assert( leftContent instanceof Node ); + assert && assert( rightContent instanceof Node ); + assert && assert( baseColorProperty instanceof Property ); + + // @public {string} + this.accessibleText = StringUtils.fillIn( productTimesPatternString, { + left: leftContent.accessibleText, + right: rightContent.accessibleText + } ); - // a11y - tagName: 'mo', - accessibleNamespace: 'http://www.w3.org/1998/Math/MathML', - innerContent: '×' - } ); + // @private {Node|null} + this.leftContent = leftContent; + this.rightContent = rightContent; - HBox.call( this, { - children: [ this.timesNode ], - spacing: AreaModelCommonConstants.CALCULATION_X_PADDING, - align: 'bottom' - } ); - } + if ( !this.initialized ) { + this.initialized = true; - assert && assert( this.children.length === 1, 'Should only have the timesNode' ); + // @private {Text} - Persistent (since it's declared in the constructor instead of the initialize function, this + // will persist for the life of this node). + this.timesNode = new Text( MathSymbols.TIMES, { + font: AreaModelCommonConstants.CALCULATION_X_FONT, - this.insertChild( 0, leftContent ); - this.addChild( rightContent ); + // a11y + tagName: 'mo', + accessibleNamespace: 'http://www.w3.org/1998/Math/MathML', + innerContent: '×' + } ); - this.timesNode.fill = baseColorProperty; + HBox.call( this, { + children: [ this.timesNode ], + spacing: AreaModelCommonConstants.CALCULATION_X_PADDING, + align: 'bottom' + } ); } - areaModelCommon.register( 'MultiplyX', MultiplyX ); + assert && assert( this.children.length === 1, 'Should only have the timesNode' ); - inherit( HBox, MultiplyX, { - /** - * Clears the state of this node (releasing references) so it can be freed to the pool (and potentially GC'ed). - * @public - */ - clean: function() { - assert && assert( this.children.length === 3, 'Should have two content nodes and our timesNode' ); + this.insertChild( 0, leftContent ); + this.addChild( rightContent ); - // Remove our content - this.removeChild( this.leftContent ); - this.removeChild( this.rightContent ); - this.leftContent.clean(); - this.rightContent.clean(); - this.leftContent = null; - this.rightContent = null; + this.timesNode.fill = baseColorProperty; +} - this.timesNode.fill = null; +areaModelCommon.register( 'MultiplyX', MultiplyX ); - this.freeToPool(); - } - } ); +inherit( HBox, MultiplyX, { + /** + * Clears the state of this node (releasing references) so it can be freed to the pool (and potentially GC'ed). + * @public + */ + clean: function() { + assert && assert( this.children.length === 3, 'Should have two content nodes and our timesNode' ); - Poolable.mixInto( MultiplyX ); + // Remove our content + this.removeChild( this.leftContent ); + this.removeChild( this.rightContent ); + this.leftContent.clean(); + this.rightContent.clean(); + this.leftContent = null; + this.rightContent = null; - return MultiplyX; + this.timesNode.fill = null; + + this.freeToPool(); + } } ); + +Poolable.mixInto( MultiplyX ); + +export default MultiplyX; \ No newline at end of file diff --git a/js/common/view/calculation/OrderedLine.js b/js/common/view/calculation/OrderedLine.js index 0bfb5cd6..720988af 100644 --- a/js/common/view/calculation/OrderedLine.js +++ b/js/common/view/calculation/OrderedLine.js @@ -5,31 +5,28 @@ * * @author Jonathan Olson */ -define( require => { - 'use strict'; - // modules - const areaModelCommon = require( 'AREA_MODEL_COMMON/areaModelCommon' ); - const CalculationLine = require( 'AREA_MODEL_COMMON/common/view/calculation/CalculationLine' ); - const inherit = require( 'PHET_CORE/inherit' ); +import inherit from '../../../../../phet-core/js/inherit.js'; +import areaModelCommon from '../../../areaModelCommon.js'; +import CalculationLine from './CalculationLine.js'; - /** - * @constructor - * @extends {CalculationLine} - * - * @param {TermList} orderedTermList - * @param {Area} area - * @param {Property.} activeIndexProperty - * @param {boolean} allowExponents - Whether exponents (powers of x) are allowed - * @param {boolean} isProportional - Whether the area is shown as proportional (instead of generic) - */ - function OrderedLine( orderedTermList, area, activeIndexProperty, allowExponents, isProportional ) { - CalculationLine.call( this, CalculationLine.ORDERED_LINE_INDEX, area.colorProperties, activeIndexProperty, allowExponents, isProportional ); +/** + * @constructor + * @extends {CalculationLine} + * + * @param {TermList} orderedTermList + * @param {Area} area + * @param {Property.} activeIndexProperty + * @param {boolean} allowExponents - Whether exponents (powers of x) are allowed + * @param {boolean} isProportional - Whether the area is shown as proportional (instead of generic) + */ +function OrderedLine( orderedTermList, area, activeIndexProperty, allowExponents, isProportional ) { + CalculationLine.call( this, CalculationLine.ORDERED_LINE_INDEX, area.colorProperties, activeIndexProperty, allowExponents, isProportional ); - this.node = this.sumWithNegativeParens( orderedTermList.terms ); - } + this.node = this.sumWithNegativeParens( orderedTermList.terms ); +} - areaModelCommon.register( 'OrderedLine', OrderedLine ); +areaModelCommon.register( 'OrderedLine', OrderedLine ); - return inherit( CalculationLine, OrderedLine ); -} ); +inherit( CalculationLine, OrderedLine ); +export default OrderedLine; \ No newline at end of file diff --git a/js/common/view/calculation/Parentheses.js b/js/common/view/calculation/Parentheses.js index 55470b48..5e11bb33 100644 --- a/js/common/view/calculation/Parentheses.js +++ b/js/common/view/calculation/Parentheses.js @@ -7,115 +7,111 @@ * * @author Jonathan Olson */ -define( require => { - 'use strict'; - - // modules - const areaModelCommon = require( 'AREA_MODEL_COMMON/areaModelCommon' ); - const AreaModelCommonA11yStrings = require( 'AREA_MODEL_COMMON/AreaModelCommonA11yStrings' ); - const AreaModelCommonConstants = require( 'AREA_MODEL_COMMON/common/AreaModelCommonConstants' ); - const HBox = require( 'SCENERY/nodes/HBox' ); - const inherit = require( 'PHET_CORE/inherit' ); - const Node = require( 'SCENERY/nodes/Node' ); - const Poolable = require( 'PHET_CORE/Poolable' ); - const Property = require( 'AXON/Property' ); - const StringUtils = require( 'PHETCOMMON/util/StringUtils' ); - const Text = require( 'SCENERY/nodes/Text' ); - - // a11y strings - const quantityPatternString = AreaModelCommonA11yStrings.quantityPattern.value; - /** - * @constructor - * @extends {HBox} - * - * @param {Node} content - Should have a clean() method to support pooling - * @param {Property.} baseColorProperty - */ - function Parentheses( content, baseColorProperty ) { - assert && assert( content instanceof Node ); - assert && assert( baseColorProperty instanceof Property ); +import Property from '../../../../../axon/js/Property.js'; +import inherit from '../../../../../phet-core/js/inherit.js'; +import Poolable from '../../../../../phet-core/js/Poolable.js'; +import StringUtils from '../../../../../phetcommon/js/util/StringUtils.js'; +import HBox from '../../../../../scenery/js/nodes/HBox.js'; +import Node from '../../../../../scenery/js/nodes/Node.js'; +import Text from '../../../../../scenery/js/nodes/Text.js'; +import areaModelCommon from '../../../areaModelCommon.js'; +import AreaModelCommonA11yStrings from '../../../AreaModelCommonA11yStrings.js'; +import AreaModelCommonConstants from '../../AreaModelCommonConstants.js'; + +// a11y strings +const quantityPatternString = AreaModelCommonA11yStrings.quantityPattern.value; + +/** + * @constructor + * @extends {HBox} + * + * @param {Node} content - Should have a clean() method to support pooling + * @param {Property.} baseColorProperty + */ +function Parentheses( content, baseColorProperty ) { + assert && assert( content instanceof Node ); + assert && assert( baseColorProperty instanceof Property ); + + // @public {string} + this.accessibleText = StringUtils.fillIn( quantityPatternString, { + content: content.accessibleText + } ); + + // @private {Node|null} + this.content = content; + + if ( !this.initialized ) { + this.initialized = true; + + // @private {Text} - Persistent (since these are declared in the constructor instead of the initialize function, + // they will persist for the life of this node). + this.leftParen = new Text( '(', { + font: AreaModelCommonConstants.CALCULATION_PAREN_FONT, + + // a11y + tagName: 'mo', + accessibleNamespace: 'http://www.w3.org/1998/Math/MathML', + innerContent: '(' + } ); + this.leftParen.setAccessibleAttribute( 'form', 'prefix', { + namespace: 'http://www.w3.org/1998/Math/MathML' + } ); - // @public {string} - this.accessibleText = StringUtils.fillIn( quantityPatternString, { - content: content.accessibleText + // @private {Text} - See notes above + this.rightParen = new Text( ')', { + font: AreaModelCommonConstants.CALCULATION_PAREN_FONT, + + // a11y + tagName: 'mo', + accessibleNamespace: 'http://www.w3.org/1998/Math/MathML', + innerContent: ')' + } ); + this.rightParen.setAccessibleAttribute( 'form', 'postfix', { + namespace: 'http://www.w3.org/1998/Math/MathML' } ); - // @private {Node|null} - this.content = content; - - if ( !this.initialized ) { - this.initialized = true; - - // @private {Text} - Persistent (since these are declared in the constructor instead of the initialize function, - // they will persist for the life of this node). - this.leftParen = new Text( '(', { - font: AreaModelCommonConstants.CALCULATION_PAREN_FONT, - - // a11y - tagName: 'mo', - accessibleNamespace: 'http://www.w3.org/1998/Math/MathML', - innerContent: '(' - } ); - this.leftParen.setAccessibleAttribute( 'form', 'prefix', { - namespace: 'http://www.w3.org/1998/Math/MathML' - } ); - - // @private {Text} - See notes above - this.rightParen = new Text( ')', { - font: AreaModelCommonConstants.CALCULATION_PAREN_FONT, - - // a11y - tagName: 'mo', - accessibleNamespace: 'http://www.w3.org/1998/Math/MathML', - innerContent: ')' - } ); - this.rightParen.setAccessibleAttribute( 'form', 'postfix', { - namespace: 'http://www.w3.org/1998/Math/MathML' - } ); - - HBox.call( this, { - children: [ this.leftParen, this.rightParen ], - spacing: AreaModelCommonConstants.CALCULATION_PAREN_PADDING, - - // a11y - align: 'bottom', - tagName: 'mrow', - accessibleNamespace: 'http://www.w3.org/1998/Math/MathML' - } ); - } - - assert && assert( this.children.length === 2, 'Should only have a left and right paren at this moment' ); - - this.insertChild( 1, content ); - - this.leftParen.fill = baseColorProperty; - this.rightParen.fill = baseColorProperty; + HBox.call( this, { + children: [ this.leftParen, this.rightParen ], + spacing: AreaModelCommonConstants.CALCULATION_PAREN_PADDING, + + // a11y + align: 'bottom', + tagName: 'mrow', + accessibleNamespace: 'http://www.w3.org/1998/Math/MathML' + } ); } - areaModelCommon.register( 'Parentheses', Parentheses ); + assert && assert( this.children.length === 2, 'Should only have a left and right paren at this moment' ); - inherit( HBox, Parentheses, { - /** - * Clears the state of this node (releasing references) so it can be freed to the pool (and potentially GC'ed). - * @public - */ - clean: function() { - assert && assert( this.children.length === 3, 'Should only have a left and right paren AND content' ); + this.insertChild( 1, content ); - // Remove our content - this.removeChild( this.content ); - this.content.clean(); - this.content = null; + this.leftParen.fill = baseColorProperty; + this.rightParen.fill = baseColorProperty; +} - this.leftParen.fill = null; - this.rightParen.fill = null; +areaModelCommon.register( 'Parentheses', Parentheses ); - this.freeToPool(); - } - } ); +inherit( HBox, Parentheses, { + /** + * Clears the state of this node (releasing references) so it can be freed to the pool (and potentially GC'ed). + * @public + */ + clean: function() { + assert && assert( this.children.length === 3, 'Should only have a left and right paren AND content' ); + + // Remove our content + this.removeChild( this.content ); + this.content.clean(); + this.content = null; - Poolable.mixInto( Parentheses ); + this.leftParen.fill = null; + this.rightParen.fill = null; - return Parentheses; + this.freeToPool(); + } } ); + +Poolable.mixInto( Parentheses ); + +export default Parentheses; \ No newline at end of file diff --git a/js/common/view/calculation/PlaceholderBox.js b/js/common/view/calculation/PlaceholderBox.js index 11478659..30cbc57e 100644 --- a/js/common/view/calculation/PlaceholderBox.js +++ b/js/common/view/calculation/PlaceholderBox.js @@ -7,65 +7,61 @@ * * @author Jonathan Olson */ -define( require => { - 'use strict'; - // modules - const areaModelCommon = require( 'AREA_MODEL_COMMON/areaModelCommon' ); - const AreaModelCommonA11yStrings = require( 'AREA_MODEL_COMMON/AreaModelCommonA11yStrings' ); - const inherit = require( 'PHET_CORE/inherit' ); - const Poolable = require( 'PHET_CORE/Poolable' ); - const Property = require( 'AXON/Property' ); - const Rectangle = require( 'SCENERY/nodes/Rectangle' ); +import Property from '../../../../../axon/js/Property.js'; +import inherit from '../../../../../phet-core/js/inherit.js'; +import Poolable from '../../../../../phet-core/js/Poolable.js'; +import Rectangle from '../../../../../scenery/js/nodes/Rectangle.js'; +import areaModelCommon from '../../../areaModelCommon.js'; +import AreaModelCommonA11yStrings from '../../../AreaModelCommonA11yStrings.js'; - // a11y strings - const placeholderString = AreaModelCommonA11yStrings.placeholder.value; +// a11y strings +const placeholderString = AreaModelCommonA11yStrings.placeholder.value; - /** - * @constructor - * @extends {Rectangle} - * - * @param {Property.} colorProperty - * @param {boolean} allowExponents - Whether exponents (powers of x) are allowed - */ - function PlaceholderBox( colorProperty, allowExponents ) { - assert && assert( colorProperty instanceof Property ); - assert && assert( typeof allowExponents === 'boolean' ); - - if ( !this.initialized ) { - this.initialized = true; +/** + * @constructor + * @extends {Rectangle} + * + * @param {Property.} colorProperty + * @param {boolean} allowExponents - Whether exponents (powers of x) are allowed + */ +function PlaceholderBox( colorProperty, allowExponents ) { + assert && assert( colorProperty instanceof Property ); + assert && assert( typeof allowExponents === 'boolean' ); - // @public {string} - this.accessibleText = placeholderString; + if ( !this.initialized ) { + this.initialized = true; - Rectangle.call( this, 0, 0, 16, 16, { - lineWidth: 0.7, + // @public {string} + this.accessibleText = placeholderString; - // a11y - tagName: 'mi', - accessibleNamespace: 'http://www.w3.org/1998/Math/MathML', - innerContent: placeholderString - } ); - } + Rectangle.call( this, 0, 0, 16, 16, { + lineWidth: 0.7, - this.stroke = colorProperty; - this.localBounds = this.selfBounds.dilatedX( allowExponents ? 2 : 0 ); + // a11y + tagName: 'mi', + accessibleNamespace: 'http://www.w3.org/1998/Math/MathML', + innerContent: placeholderString + } ); } - areaModelCommon.register( 'PlaceholderBox', PlaceholderBox ); + this.stroke = colorProperty; + this.localBounds = this.selfBounds.dilatedX( allowExponents ? 2 : 0 ); +} - inherit( Rectangle, PlaceholderBox, { - /** - * Clears the state of this node (releasing references) so it can be freed to the pool (and potentially GC'ed). - * @public - */ - clean: function() { - this.stroke = null; - this.freeToPool(); - } - } ); +areaModelCommon.register( 'PlaceholderBox', PlaceholderBox ); - Poolable.mixInto( PlaceholderBox ); - - return PlaceholderBox; +inherit( Rectangle, PlaceholderBox, { + /** + * Clears the state of this node (releasing references) so it can be freed to the pool (and potentially GC'ed). + * @public + */ + clean: function() { + this.stroke = null; + this.freeToPool(); + } } ); + +Poolable.mixInto( PlaceholderBox ); + +export default PlaceholderBox; \ No newline at end of file diff --git a/js/common/view/calculation/Plus.js b/js/common/view/calculation/Plus.js index 4dc10832..78d53f5d 100644 --- a/js/common/view/calculation/Plus.js +++ b/js/common/view/calculation/Plus.js @@ -7,65 +7,61 @@ * * @author Jonathan Olson */ -define( require => { - 'use strict'; - // modules - const areaModelCommon = require( 'AREA_MODEL_COMMON/areaModelCommon' ); - const AreaModelCommonA11yStrings = require( 'AREA_MODEL_COMMON/AreaModelCommonA11yStrings' ); - const AreaModelCommonConstants = require( 'AREA_MODEL_COMMON/common/AreaModelCommonConstants' ); - const inherit = require( 'PHET_CORE/inherit' ); - const MathSymbols = require( 'SCENERY_PHET/MathSymbols' ); - const Poolable = require( 'PHET_CORE/Poolable' ); - const Property = require( 'AXON/Property' ); - const Text = require( 'SCENERY/nodes/Text' ); +import Property from '../../../../../axon/js/Property.js'; +import inherit from '../../../../../phet-core/js/inherit.js'; +import Poolable from '../../../../../phet-core/js/Poolable.js'; +import MathSymbols from '../../../../../scenery-phet/js/MathSymbols.js'; +import Text from '../../../../../scenery/js/nodes/Text.js'; +import areaModelCommon from '../../../areaModelCommon.js'; +import AreaModelCommonA11yStrings from '../../../AreaModelCommonA11yStrings.js'; +import AreaModelCommonConstants from '../../AreaModelCommonConstants.js'; - // a11y strings - const sumPlusString = AreaModelCommonA11yStrings.sumPlus.value; +// a11y strings +const sumPlusString = AreaModelCommonA11yStrings.sumPlus.value; - /** - * @constructor - * @extends {Text} - * - * @param {Property.} baseColorProperty - */ - function Plus( baseColorProperty ) { - assert && assert( baseColorProperty instanceof Property ); - - if ( !this.initialized ) { - this.initialized = true; +/** + * @constructor + * @extends {Text} + * + * @param {Property.} baseColorProperty + */ +function Plus( baseColorProperty ) { + assert && assert( baseColorProperty instanceof Property ); - // @public {string} - this.accessibleText = sumPlusString; + if ( !this.initialized ) { + this.initialized = true; - Text.call( this, MathSymbols.PLUS, { - font: AreaModelCommonConstants.CALCULATION_PAREN_FONT, + // @public {string} + this.accessibleText = sumPlusString; - // a11y - tagName: 'mo', - accessibleNamespace: 'http://www.w3.org/1998/Math/MathML', - innerContent: '+' - } ); - } + Text.call( this, MathSymbols.PLUS, { + font: AreaModelCommonConstants.CALCULATION_PAREN_FONT, - this.fill = baseColorProperty; + // a11y + tagName: 'mo', + accessibleNamespace: 'http://www.w3.org/1998/Math/MathML', + innerContent: '+' + } ); } - areaModelCommon.register( 'Plus', Plus ); + this.fill = baseColorProperty; +} - inherit( Text, Plus, { - /** - * Clears the state of this node (releasing references) so it can be freed to the pool (and potentially GC'ed). - * @public - */ - clean: function() { - this.fill = null; +areaModelCommon.register( 'Plus', Plus ); - this.freeToPool(); - } - } ); - - Poolable.mixInto( Plus ); +inherit( Text, Plus, { + /** + * Clears the state of this node (releasing references) so it can be freed to the pool (and potentially GC'ed). + * @public + */ + clean: function() { + this.fill = null; - return Plus; + this.freeToPool(); + } } ); + +Poolable.mixInto( Plus ); + +export default Plus; \ No newline at end of file diff --git a/js/common/view/calculation/QuestionMark.js b/js/common/view/calculation/QuestionMark.js index 0cc64afa..9f93bfa4 100644 --- a/js/common/view/calculation/QuestionMark.js +++ b/js/common/view/calculation/QuestionMark.js @@ -7,64 +7,60 @@ * * @author Jonathan Olson */ -define( require => { - 'use strict'; - // modules - const areaModelCommon = require( 'AREA_MODEL_COMMON/areaModelCommon' ); - const AreaModelCommonA11yStrings = require( 'AREA_MODEL_COMMON/AreaModelCommonA11yStrings' ); - const AreaModelCommonConstants = require( 'AREA_MODEL_COMMON/common/AreaModelCommonConstants' ); - const inherit = require( 'PHET_CORE/inherit' ); - const Poolable = require( 'PHET_CORE/Poolable' ); - const Property = require( 'AXON/Property' ); - const Text = require( 'SCENERY/nodes/Text' ); +import Property from '../../../../../axon/js/Property.js'; +import inherit from '../../../../../phet-core/js/inherit.js'; +import Poolable from '../../../../../phet-core/js/Poolable.js'; +import Text from '../../../../../scenery/js/nodes/Text.js'; +import areaModelCommon from '../../../areaModelCommon.js'; +import AreaModelCommonA11yStrings from '../../../AreaModelCommonA11yStrings.js'; +import AreaModelCommonConstants from '../../AreaModelCommonConstants.js'; - // a11y strings - const questionMarkString = AreaModelCommonA11yStrings.questionMark.value; +// a11y strings +const questionMarkString = AreaModelCommonA11yStrings.questionMark.value; - /** - * @constructor - * @extends {Text} - * - * @param {Property.} baseColorProperty - */ - function QuestionMark( baseColorProperty ) { - assert && assert( baseColorProperty instanceof Property ); - - if ( !this.initialized ) { - this.initialized = true; +/** + * @constructor + * @extends {Text} + * + * @param {Property.} baseColorProperty + */ +function QuestionMark( baseColorProperty ) { + assert && assert( baseColorProperty instanceof Property ); - // @public {string} - this.accessibleText = questionMarkString; + if ( !this.initialized ) { + this.initialized = true; - Text.call( this, '?', { - font: AreaModelCommonConstants.CALCULATION_TERM_FONT, + // @public {string} + this.accessibleText = questionMarkString; - // a11y - tagName: 'mi', - accessibleNamespace: 'http://www.w3.org/1998/Math/MathML', - innerContent: questionMarkString - } ); - } + Text.call( this, '?', { + font: AreaModelCommonConstants.CALCULATION_TERM_FONT, - this.fill = baseColorProperty; + // a11y + tagName: 'mi', + accessibleNamespace: 'http://www.w3.org/1998/Math/MathML', + innerContent: questionMarkString + } ); } - areaModelCommon.register( 'QuestionMark', QuestionMark ); + this.fill = baseColorProperty; +} - inherit( Text, QuestionMark, { - /** - * Clears the state of this node (releasing references) so it can be freed to the pool (and potentially GC'ed). - * @public - */ - clean: function() { - this.fill = null; +areaModelCommon.register( 'QuestionMark', QuestionMark ); - this.freeToPool(); - } - } ); - - Poolable.mixInto( QuestionMark ); +inherit( Text, QuestionMark, { + /** + * Clears the state of this node (releasing references) so it can be freed to the pool (and potentially GC'ed). + * @public + */ + clean: function() { + this.fill = null; - return QuestionMark; + this.freeToPool(); + } } ); + +Poolable.mixInto( QuestionMark ); + +export default QuestionMark; \ No newline at end of file diff --git a/js/common/view/calculation/QuestionMarkLine.js b/js/common/view/calculation/QuestionMarkLine.js index 9c134687..0a582d55 100644 --- a/js/common/view/calculation/QuestionMarkLine.js +++ b/js/common/view/calculation/QuestionMarkLine.js @@ -5,30 +5,27 @@ * * @author Jonathan Olson */ -define( require => { - 'use strict'; - // modules - const areaModelCommon = require( 'AREA_MODEL_COMMON/areaModelCommon' ); - const CalculationLine = require( 'AREA_MODEL_COMMON/common/view/calculation/CalculationLine' ); - const inherit = require( 'PHET_CORE/inherit' ); +import inherit from '../../../../../phet-core/js/inherit.js'; +import areaModelCommon from '../../../areaModelCommon.js'; +import CalculationLine from './CalculationLine.js'; - /** - * @constructor - * @extends {CalculationLine} - * - * @param {Area} area - * @param {Property.} activeIndexProperty - * @param {boolean} allowExponents - Whether exponents (powers of x) are allowed - * @param {boolean} isProportional - Whether the area is shown as proportional (instead of generic) - */ - function QuestionMarkLine( area, activeIndexProperty, allowExponents, isProportional ) { - CalculationLine.call( this, CalculationLine.TOTALS_LINE_INDEX, area.colorProperties, activeIndexProperty, allowExponents, isProportional ); +/** + * @constructor + * @extends {CalculationLine} + * + * @param {Area} area + * @param {Property.} activeIndexProperty + * @param {boolean} allowExponents - Whether exponents (powers of x) are allowed + * @param {boolean} isProportional - Whether the area is shown as proportional (instead of generic) + */ +function QuestionMarkLine( area, activeIndexProperty, allowExponents, isProportional ) { + CalculationLine.call( this, CalculationLine.TOTALS_LINE_INDEX, area.colorProperties, activeIndexProperty, allowExponents, isProportional ); - this.node = this.questionMark(); - } + this.node = this.questionMark(); +} - areaModelCommon.register( 'QuestionMarkLine', QuestionMarkLine ); +areaModelCommon.register( 'QuestionMarkLine', QuestionMarkLine ); - return inherit( CalculationLine, QuestionMarkLine ); -} ); +inherit( CalculationLine, QuestionMarkLine ); +export default QuestionMarkLine; \ No newline at end of file diff --git a/js/common/view/calculation/SumLine.js b/js/common/view/calculation/SumLine.js index 12f657f1..776fdbba 100644 --- a/js/common/view/calculation/SumLine.js +++ b/js/common/view/calculation/SumLine.js @@ -5,30 +5,27 @@ * * @author Jonathan Olson */ -define( require => { - 'use strict'; - // modules - const areaModelCommon = require( 'AREA_MODEL_COMMON/areaModelCommon' ); - const CalculationLine = require( 'AREA_MODEL_COMMON/common/view/calculation/CalculationLine' ); - const inherit = require( 'PHET_CORE/inherit' ); +import inherit from '../../../../../phet-core/js/inherit.js'; +import areaModelCommon from '../../../areaModelCommon.js'; +import CalculationLine from './CalculationLine.js'; - /** - * @constructor - * @extends {CalculationLine} - * - * @param {Area} area - * @param {Property.} activeIndexProperty - * @param {boolean} allowExponents - Whether exponents (powers of x) are allowed - * @param {boolean} isProportional - Whether the area is shown as proportional (instead of generic) - */ - function SumLine( area, activeIndexProperty, allowExponents, isProportional ) { - CalculationLine.call( this, CalculationLine.SUM_LINE_INDEX, area.colorProperties, activeIndexProperty, allowExponents, isProportional ); +/** + * @constructor + * @extends {CalculationLine} + * + * @param {Area} area + * @param {Property.} activeIndexProperty + * @param {boolean} allowExponents - Whether exponents (powers of x) are allowed + * @param {boolean} isProportional - Whether the area is shown as proportional (instead of generic) + */ +function SumLine( area, activeIndexProperty, allowExponents, isProportional ) { + CalculationLine.call( this, CalculationLine.SUM_LINE_INDEX, area.colorProperties, activeIndexProperty, allowExponents, isProportional ); - this.node = this.baseTermText( area.totalAreaProperty.value, false ); - } + this.node = this.baseTermText( area.totalAreaProperty.value, false ); +} - areaModelCommon.register( 'SumLine', SumLine ); +areaModelCommon.register( 'SumLine', SumLine ); - return inherit( CalculationLine, SumLine ); -} ); +inherit( CalculationLine, SumLine ); +export default SumLine; \ No newline at end of file diff --git a/js/common/view/calculation/TermText.js b/js/common/view/calculation/TermText.js index f0245c83..743a0647 100644 --- a/js/common/view/calculation/TermText.js +++ b/js/common/view/calculation/TermText.js @@ -7,70 +7,66 @@ * * @author Jonathan Olson */ -define( require => { - 'use strict'; - // modules - const areaModelCommon = require( 'AREA_MODEL_COMMON/areaModelCommon' ); - const AreaModelCommonConstants = require( 'AREA_MODEL_COMMON/common/AreaModelCommonConstants' ); - const inherit = require( 'PHET_CORE/inherit' ); - const Poolable = require( 'PHET_CORE/Poolable' ); - const Property = require( 'AXON/Property' ); - const RichText = require( 'SCENERY/nodes/RichText' ); - const Term = require( 'AREA_MODEL_COMMON/common/model/Term' ); - const TermList = require( 'AREA_MODEL_COMMON/common/model/TermList' ); +import Property from '../../../../../axon/js/Property.js'; +import inherit from '../../../../../phet-core/js/inherit.js'; +import Poolable from '../../../../../phet-core/js/Poolable.js'; +import RichText from '../../../../../scenery/js/nodes/RichText.js'; +import areaModelCommon from '../../../areaModelCommon.js'; +import AreaModelCommonConstants from '../../AreaModelCommonConstants.js'; +import Term from '../../model/Term.js'; +import TermList from '../../model/TermList.js'; - /** - * @constructor - * @extends {RichText} - * - * @param {TermList|Term} term - * @param {Property.} colorProperty - * @param {boolean} excludeSign - */ - function TermText( term, colorProperty, excludeSign ) { - assert && assert( term instanceof Term || term instanceof TermList ); - assert && assert( colorProperty instanceof Property ); - assert && assert( typeof excludeSign === 'boolean' || excludeSign === undefined ); - - const text = excludeSign ? term.toNoSignRichString() : term.toRichString( false ); +/** + * @constructor + * @extends {RichText} + * + * @param {TermList|Term} term + * @param {Property.} colorProperty + * @param {boolean} excludeSign + */ +function TermText( term, colorProperty, excludeSign ) { + assert && assert( term instanceof Term || term instanceof TermList ); + assert && assert( colorProperty instanceof Property ); + assert && assert( typeof excludeSign === 'boolean' || excludeSign === undefined ); - // @public {string} - this.accessibleText = text; + const text = excludeSign ? term.toNoSignRichString() : term.toRichString( false ); - if ( !this.initialized ) { - this.initialized = true; + // @public {string} + this.accessibleText = text; - RichText.call( this, ' ', { - font: AreaModelCommonConstants.CALCULATION_TERM_FONT, + if ( !this.initialized ) { + this.initialized = true; - // a11y - tagName: 'mn', - accessibleNamespace: 'http://www.w3.org/1998/Math/MathML' - } ); - } + RichText.call( this, ' ', { + font: AreaModelCommonConstants.CALCULATION_TERM_FONT, - this.mutate( { - text: text, - fill: colorProperty, - innerContent: text + // a11y + tagName: 'mn', + accessibleNamespace: 'http://www.w3.org/1998/Math/MathML' } ); } - areaModelCommon.register( 'TermText', TermText ); - - inherit( RichText, TermText, { - /** - * Clears the state of this node (releasing references) so it can be freed to the pool (and potentially GC'ed). - * @public - */ - clean: function() { - this.fill = null; - this.freeToPool(); - } + this.mutate( { + text: text, + fill: colorProperty, + innerContent: text } ); +} - Poolable.mixInto( TermText ); +areaModelCommon.register( 'TermText', TermText ); - return TermText; +inherit( RichText, TermText, { + /** + * Clears the state of this node (releasing references) so it can be freed to the pool (and potentially GC'ed). + * @public + */ + clean: function() { + this.fill = null; + this.freeToPool(); + } } ); + +Poolable.mixInto( TermText ); + +export default TermText; \ No newline at end of file diff --git a/js/common/view/calculation/TotalsLine.js b/js/common/view/calculation/TotalsLine.js index 37e567c9..24a09b4f 100644 --- a/js/common/view/calculation/TotalsLine.js +++ b/js/common/view/calculation/TotalsLine.js @@ -5,46 +5,43 @@ * * @author Jonathan Olson */ -define( require => { - 'use strict'; - - // modules - const areaModelCommon = require( 'AREA_MODEL_COMMON/areaModelCommon' ); - const AreaModelCommonConstants = require( 'AREA_MODEL_COMMON/common/AreaModelCommonConstants' ); - const CalculationLine = require( 'AREA_MODEL_COMMON/common/view/calculation/CalculationLine' ); - const inherit = require( 'PHET_CORE/inherit' ); - - /** - * @constructor - * @extends {CalculationLine} - * - * @param {Area} area - * @param {Property.} activeIndexProperty - * @param {boolean} allowExponents - Whether exponents (powers of x) are allowed - * @param {boolean} isProportional - Whether the area is shown as proportional (instead of generic) - */ - function TotalsLine( area, activeIndexProperty, allowExponents, isProportional ) { - const self = this; - - CalculationLine.call( this, CalculationLine.TOTALS_LINE_INDEX, area.colorProperties, activeIndexProperty, allowExponents, isProportional ); - - const totalTexts = area.displayProperties.map( function( orientationTotal, orientation ) { - return orientationTotal.value ? self.orientedTermText( orientation, orientationTotal.value ) - : self.orientedPlaceholderBox( orientation ); - } ); - - if ( allowExponents ) { - this.node = this.group( [ - this.parentheses( totalTexts.vertical ), - this.parentheses( totalTexts.horizontal ) - ], AreaModelCommonConstants.CALCULATION_PAREN_PAREN_PADDING ); - } - else { - this.node = this.multiplyX( totalTexts.vertical, totalTexts.horizontal ); - } + +import inherit from '../../../../../phet-core/js/inherit.js'; +import areaModelCommon from '../../../areaModelCommon.js'; +import AreaModelCommonConstants from '../../AreaModelCommonConstants.js'; +import CalculationLine from './CalculationLine.js'; + +/** + * @constructor + * @extends {CalculationLine} + * + * @param {Area} area + * @param {Property.} activeIndexProperty + * @param {boolean} allowExponents - Whether exponents (powers of x) are allowed + * @param {boolean} isProportional - Whether the area is shown as proportional (instead of generic) + */ +function TotalsLine( area, activeIndexProperty, allowExponents, isProportional ) { + const self = this; + + CalculationLine.call( this, CalculationLine.TOTALS_LINE_INDEX, area.colorProperties, activeIndexProperty, allowExponents, isProportional ); + + const totalTexts = area.displayProperties.map( function( orientationTotal, orientation ) { + return orientationTotal.value ? self.orientedTermText( orientation, orientationTotal.value ) + : self.orientedPlaceholderBox( orientation ); + } ); + + if ( allowExponents ) { + this.node = this.group( [ + this.parentheses( totalTexts.vertical ), + this.parentheses( totalTexts.horizontal ) + ], AreaModelCommonConstants.CALCULATION_PAREN_PAREN_PADDING ); + } + else { + this.node = this.multiplyX( totalTexts.vertical, totalTexts.horizontal ); } +} - areaModelCommon.register( 'TotalsLine', TotalsLine ); +areaModelCommon.register( 'TotalsLine', TotalsLine ); - return inherit( CalculationLine, TotalsLine ); -} ); +inherit( CalculationLine, TotalsLine ); +export default TotalsLine; \ No newline at end of file diff --git a/js/game/model/AreaChallenge.js b/js/game/model/AreaChallenge.js index 9d065fe2..8d6732de 100644 --- a/js/game/model/AreaChallenge.js +++ b/js/game/model/AreaChallenge.js @@ -5,515 +5,511 @@ * * @author Jonathan Olson */ -define( require => { - 'use strict'; - - // modules - const areaModelCommon = require( 'AREA_MODEL_COMMON/areaModelCommon' ); - const DerivedProperty = require( 'AXON/DerivedProperty' ); - const dimensionForEach = require( 'PHET_CORE/dimensionForEach' ); - const dimensionMap = require( 'PHET_CORE/dimensionMap' ); - const Entry = require( 'AREA_MODEL_COMMON/game/model/Entry' ); - const EntryDisplayType = require( 'AREA_MODEL_COMMON/game/model/EntryDisplayType' ); - const EntryStatus = require( 'AREA_MODEL_COMMON/game/model/EntryStatus' ); - const EntryType = require( 'AREA_MODEL_COMMON/game/model/EntryType' ); - const GameState = require( 'AREA_MODEL_COMMON/game/model/GameState' ); - const GenericArea = require( 'AREA_MODEL_COMMON/generic/model/GenericArea' ); - const inherit = require( 'PHET_CORE/inherit' ); - const InputMethod = require( 'AREA_MODEL_COMMON/game/model/InputMethod' ); - const merge = require( 'PHET_CORE/merge' ); - const Orientation = require( 'PHET_CORE/Orientation' ); - const OrientationPair = require( 'AREA_MODEL_COMMON/common/model/OrientationPair' ); - const Polynomial = require( 'AREA_MODEL_COMMON/common/model/Polynomial' ); - const Property = require( 'AXON/Property' ); - const Term = require( 'AREA_MODEL_COMMON/common/model/Term' ); - /** - * @constructor - * @extends {Object} - * - * @param {AreaChallengeDescription} description - */ - function AreaChallenge( description ) { - const self = this; +import DerivedProperty from '../../../../axon/js/DerivedProperty.js'; +import Property from '../../../../axon/js/Property.js'; +import dimensionForEach from '../../../../phet-core/js/dimensionForEach.js'; +import dimensionMap from '../../../../phet-core/js/dimensionMap.js'; +import inherit from '../../../../phet-core/js/inherit.js'; +import merge from '../../../../phet-core/js/merge.js'; +import Orientation from '../../../../phet-core/js/Orientation.js'; +import areaModelCommon from '../../areaModelCommon.js'; +import OrientationPair from '../../common/model/OrientationPair.js'; +import Polynomial from '../../common/model/Polynomial.js'; +import Term from '../../common/model/Term.js'; +import GenericArea from '../../generic/model/GenericArea.js'; +import Entry from './Entry.js'; +import EntryDisplayType from './EntryDisplayType.js'; +import EntryStatus from './EntryStatus.js'; +import EntryType from './EntryType.js'; +import GameState from './GameState.js'; +import InputMethod from './InputMethod.js'; - // Reassign a permuted version so we don't have a chance to screw up referencing the wrong thing - description = description.getPermutedDescription(); +/** + * @constructor + * @extends {Object} + * + * @param {AreaChallengeDescription} description + */ +function AreaChallenge( description ) { + const self = this; - // @public {AreaChallengeDescription} - this.description = description; + // Reassign a permuted version so we don't have a chance to screw up referencing the wrong thing + description = description.getPermutedDescription(); - // @public {Property.} - this.stateProperty = new Property( GameState.FIRST_ATTEMPT ); + // @public {AreaChallengeDescription} + this.description = description; - // @public {GenericArea} - used in _.property( 'area' ) - this.area = new GenericArea( description.layout, description.allowExponents ); + // @public {Property.} + this.stateProperty = new Property( GameState.FIRST_ATTEMPT ); - // @public {OrientationPair.>} - The actual partition sizes - this.partitionSizes = OrientationPair.create( function( orientation ) { - return AreaChallenge.generatePartitionTerms( - description.partitionTypes.get( orientation ).length, - description.allowExponents - ); - } ); + // @public {GenericArea} - used in _.property( 'area' ) + this.area = new GenericArea( description.layout, description.allowExponents ); - // @public {OrientationPair.>} Entries for the size of each partition. - this.partitionSizeEntries = OrientationPair.create( function( orientation ) { - return self.partitionSizes.get( orientation ).map( function( size, index ) { - return new Entry( size, { - type: description.partitionTypes.get( orientation )[ index ], - displayType: EntryType.toDisplayType( description.partitionTypes.get( orientation )[ index ] ), - inputMethod: description.numberOrVariable( InputMethod.CONSTANT, InputMethod.TERM ), - numberOfDigits: description.numberOrVariable( description.partitionTypes.get( orientation ).length - index, 1 ) - } ); + // @public {OrientationPair.>} - The actual partition sizes + this.partitionSizes = OrientationPair.create( function( orientation ) { + return AreaChallenge.generatePartitionTerms( + description.partitionTypes.get( orientation ).length, + description.allowExponents + ); + } ); + + // @public {OrientationPair.>} Entries for the size of each partition. + this.partitionSizeEntries = OrientationPair.create( function( orientation ) { + return self.partitionSizes.get( orientation ).map( function( size, index ) { + return new Entry( size, { + type: description.partitionTypes.get( orientation )[ index ], + displayType: EntryType.toDisplayType( description.partitionTypes.get( orientation )[ index ] ), + inputMethod: description.numberOrVariable( InputMethod.CONSTANT, InputMethod.TERM ), + numberOfDigits: description.numberOrVariable( description.partitionTypes.get( orientation ).length - index, 1 ) } ); } ); + } ); - // @public {OrientationPair.|null} - If we're non-unique, it will hold the 0th-place coefficients (e.g. for - // x+3 times x-7, it would hold the terms 3 and -7). It will always be two 1st-order polynomials times each other. - this.swappableSizes = this.description.unique ? null : this.partitionSizes.map( _.property( 1 ) ); + // @public {OrientationPair.|null} - If we're non-unique, it will hold the 0th-place coefficients (e.g. for + // x+3 times x-7, it would hold the terms 3 and -7). It will always be two 1st-order polynomials times each other. + this.swappableSizes = this.description.unique ? null : this.partitionSizes.map( _.property( 1 ) ); - // @public {OrientationPair.|null} - If we're non-unique, it will hold the 0th-place entries (e.g. for - // x+3 times x-7, it would hold the entries for 3 and -7). It will always be two 1st-order polynomials times each - // other. - this.swappableEntries = this.description.unique ? null : this.partitionSizeEntries.map( _.property( 1 ) ); + // @public {OrientationPair.|null} - If we're non-unique, it will hold the 0th-place entries (e.g. for + // x+3 times x-7, it would hold the entries for 3 and -7). It will always be two 1st-order polynomials times each + // other. + this.swappableEntries = this.description.unique ? null : this.partitionSizeEntries.map( _.property( 1 ) ); + + // @public {OrientationPair.>>} - Basically the values of the partitionSizeEntries, but + // null if the entry's status is 'error'. + this.nonErrorPartitionSizeProperties = OrientationPair.create( function( orientation ) { + return self.partitionSizeEntries.get( orientation ).map( _.property( 'nonErrorValueProperty' ) ); + } ); - // @public {OrientationPair.>>} - Basically the values of the partitionSizeEntries, but - // null if the entry's status is 'error'. - this.nonErrorPartitionSizeProperties = OrientationPair.create( function( orientation ) { - return self.partitionSizeEntries.get( orientation ).map( _.property( 'nonErrorValueProperty' ) ); + // @public {Array.>} + this.partialProductSizes = this.partitionSizes.vertical.map( function( verticalSize ) { + return self.partitionSizes.horizontal.map( function( horizontalSize ) { + return horizontalSize.times( verticalSize ); } ); + } ); - // @public {Array.>} - this.partialProductSizes = this.partitionSizes.vertical.map( function( verticalSize ) { - return self.partitionSizes.horizontal.map( function( horizontalSize ) { - return horizontalSize.times( verticalSize ); - } ); + // @public {Array.>} + this.partialProductSizeEntries = dimensionMap( 2, this.partialProductSizes, function( size, verticalIndex, horizontalIndex ) { + + // The number of allowed digits in entry. Basically it's the sum of vertical and horizontal (multiplication sums + // the number of digits). The far-right/bototm partition gets 1 digit, and successively higher numbers of digits + // are used for consecutive partitions. + const numbersDigits = description.partitionTypes.vertical.length + description.partitionTypes.horizontal.length - verticalIndex - horizontalIndex; + const type = description.productTypes[ verticalIndex ][ horizontalIndex ]; + const entry = new Entry( size, { + type: type, + displayType: EntryType.toDisplayType( type ), + inputMethod: description.numberOrVariable( InputMethod.CONSTANT, InputMethod.TERM ), + + // Always let them put in 1 more digit than the actual answer, see https://github.com/phetsims/area-model-common/issues/63 + numberOfDigits: description.numberOrVariable( numbersDigits, 2 ) + 1 } ); + // Link up if dynamic + if ( type === EntryType.DYNAMIC ) { + + // No unlink needed, since this is just for setup. We have a fixed number of these. + Property.multilink( [ + self.nonErrorPartitionSizeProperties.horizontal[ horizontalIndex ], + self.nonErrorPartitionSizeProperties.vertical[ verticalIndex ] + ], function( horizontal, vertical ) { + // horizontal or vertical could be null (resulting in null) + entry.valueProperty.value = horizontal && vertical && horizontal.times( vertical ); + } ); + } + return entry; + } ); - // @public {Array.>} - this.partialProductSizeEntries = dimensionMap( 2, this.partialProductSizes, function( size, verticalIndex, horizontalIndex ) { - - // The number of allowed digits in entry. Basically it's the sum of vertical and horizontal (multiplication sums - // the number of digits). The far-right/bototm partition gets 1 digit, and successively higher numbers of digits - // are used for consecutive partitions. - const numbersDigits = description.partitionTypes.vertical.length + description.partitionTypes.horizontal.length - verticalIndex - horizontalIndex; - const type = description.productTypes[ verticalIndex ][ horizontalIndex ]; - const entry = new Entry( size, { - type: type, - displayType: EntryType.toDisplayType( type ), - inputMethod: description.numberOrVariable( InputMethod.CONSTANT, InputMethod.TERM ), + // We need at least a certain number of partitions to reach x^2 in the total (either at least an x^2 on one side, + // or two x-powers on each side). + const hasXSquaredTotal = ( this.partitionSizes.horizontal.length + this.partitionSizes.vertical.length ) >= 4; - // Always let them put in 1 more digit than the actual answer, see https://github.com/phetsims/area-model-common/issues/63 - numberOfDigits: description.numberOrVariable( numbersDigits, 2 ) + 1 - } ); - // Link up if dynamic - if ( type === EntryType.DYNAMIC ) { - - // No unlink needed, since this is just for setup. We have a fixed number of these. - Property.multilink( [ - self.nonErrorPartitionSizeProperties.horizontal[ horizontalIndex ], - self.nonErrorPartitionSizeProperties.vertical[ verticalIndex ] - ], function( horizontal, vertical ) { - // horizontal or vertical could be null (resulting in null) - entry.valueProperty.value = horizontal && vertical && horizontal.times( vertical ); - } ); - } - return entry; - } ); + // @public {OrientationPair.} + this.totals = OrientationPair.create( function( orientation ) { + return new Polynomial( self.partitionSizes.get( orientation ) ); + } ); - // We need at least a certain number of partitions to reach x^2 in the total (either at least an x^2 on one side, - // or two x-powers on each side). - const hasXSquaredTotal = ( this.partitionSizes.horizontal.length + this.partitionSizes.vertical.length ) >= 4; + // @public {OrientationPair.>} + this.totalProperties = OrientationPair.create( function( orientation ) { + return new Property( self.totals.get( orientation ) ); + } ); - // @public {OrientationPair.} - this.totals = OrientationPair.create( function( orientation ) { - return new Polynomial( self.partitionSizes.get( orientation ) ); + // @public {Polynomial} + this.total = this.totals.horizontal.times( this.totals.vertical ); + + const totalOptions = { + inputMethod: description.numberOrVariable( InputMethod.CONSTANT, hasXSquaredTotal ? InputMethod.POLYNOMIAL_2 : InputMethod.POLYNOMIAL_1 ), + numberOfDigits: ( description.allowExponents ? 2 : ( this.partitionSizes.horizontal.length + this.partitionSizes.vertical.length ) ) + }; + + // @private {InputMethod} + this.totalInputMethod = totalOptions.inputMethod; + + // @public {Entry} + this.totalConstantEntry = new Entry( this.total.getTerm( 0 ), merge( { + correctValue: this.total.getTerm( 0 ), + type: description.totalType, + displayType: EntryType.toDisplayType( description.totalType ) + }, totalOptions ) ); + this.totalXEntry = new Entry( this.total.getTerm( 1 ), merge( { + correctValue: this.total.getTerm( 1 ), + type: description.numberOrVariable( EntryType.GIVEN, description.totalType ), + displayType: description.numberOrVariable( EntryDisplayType.READOUT, EntryType.toDisplayType( description.totalType ) ) + }, totalOptions ) ); + this.totalXSquaredEntry = new Entry( this.total.getTerm( 2 ), merge( { + correctValue: this.total.getTerm( 2 ), + type: description.numberOrVariable( EntryType.GIVEN, description.totalType ), + displayType: description.numberOrVariable( EntryDisplayType.READOUT, EntryType.toDisplayType( description.totalType ) ) + }, totalOptions ) ); + + // @public {Array.} - All of the coefficient entries that are used by this challenge. + this.totalCoefficientEntries = [ this.totalConstantEntry ]; + if ( totalOptions.inputMethod !== InputMethod.CONSTANT ) { + this.totalCoefficientEntries.push( this.totalXEntry ); + } + if ( totalOptions.inputMethod === InputMethod.POLYNOMIAL_2 ) { + this.totalCoefficientEntries.push( this.totalXSquaredEntry ); + } + + // @public {Property.} + this.totalProperty = new DerivedProperty( + [ this.totalConstantEntry.valueProperty, this.totalXEntry.valueProperty, this.totalXSquaredEntry.valueProperty ], + function( constant, x, xSquared ) { + const terms = [ constant, x, xSquared ].filter( function( term ) { + return term !== null; + } ); + return terms.length ? new Polynomial( terms ) : null; } ); - // @public {OrientationPair.>} - this.totalProperties = OrientationPair.create( function( orientation ) { - return new Property( self.totals.get( orientation ) ); + // All of the entries for the challenge - Not including the polynomial "total" coefficient entries + const mainEntries = this.partitionSizeEntries.horizontal + .concat( this.partitionSizeEntries.vertical ) + .concat( _.flatten( this.partialProductSizeEntries ) ); + const checkingNotificationProperties = mainEntries.map( _.property( 'valueProperty' ) ) + .concat( this.totalCoefficientEntries.map( _.property( 'statusProperty' ) ) ); + + // @public {Property.} - Whether the check button should be enabled + this.allowCheckingProperty = new DerivedProperty( checkingNotificationProperties, function() { + const allDirtyCoefficients = _.every( self.totalCoefficientEntries, function( entry ) { + return entry.type === EntryType.EDITABLE && entry.statusProperty.value === EntryStatus.DIRTY; + } ); + const hasNullMain = _.some( mainEntries, function( entry ) { + return entry.valueProperty.value === null && entry.type === EntryType.EDITABLE; } ); + return !hasNullMain && !allDirtyCoefficients; + } ); - // @public {Polynomial} - this.total = this.totals.horizontal.times( this.totals.vertical ); - - const totalOptions = { - inputMethod: description.numberOrVariable( InputMethod.CONSTANT, hasXSquaredTotal ? InputMethod.POLYNOMIAL_2 : InputMethod.POLYNOMIAL_1 ), - numberOfDigits: ( description.allowExponents ? 2 : ( this.partitionSizes.horizontal.length + this.partitionSizes.vertical.length ) ) - }; - - // @private {InputMethod} - this.totalInputMethod = totalOptions.inputMethod; - - // @public {Entry} - this.totalConstantEntry = new Entry( this.total.getTerm( 0 ), merge( { - correctValue: this.total.getTerm( 0 ), - type: description.totalType, - displayType: EntryType.toDisplayType( description.totalType ) - }, totalOptions ) ); - this.totalXEntry = new Entry( this.total.getTerm( 1 ), merge( { - correctValue: this.total.getTerm( 1 ), - type: description.numberOrVariable( EntryType.GIVEN, description.totalType ), - displayType: description.numberOrVariable( EntryDisplayType.READOUT, EntryType.toDisplayType( description.totalType ) ) - }, totalOptions ) ); - this.totalXSquaredEntry = new Entry( this.total.getTerm( 2 ), merge( { - correctValue: this.total.getTerm( 2 ), - type: description.numberOrVariable( EntryType.GIVEN, description.totalType ), - displayType: description.numberOrVariable( EntryDisplayType.READOUT, EntryType.toDisplayType( description.totalType ) ) - }, totalOptions ) ); - - // @public {Array.} - All of the coefficient entries that are used by this challenge. - this.totalCoefficientEntries = [ this.totalConstantEntry ]; - if ( totalOptions.inputMethod !== InputMethod.CONSTANT ) { - this.totalCoefficientEntries.push( this.totalXEntry ); - } - if ( totalOptions.inputMethod === InputMethod.POLYNOMIAL_2 ) { - this.totalCoefficientEntries.push( this.totalXSquaredEntry ); - } + /*---------------------------------------------------------------------------* + * Dynamic hooks + *----------------------------------------------------------------------------*/ - // @public {Property.} - this.totalProperty = new DerivedProperty( - [ this.totalConstantEntry.valueProperty, this.totalXEntry.valueProperty, this.totalXSquaredEntry.valueProperty ], - function( constant, x, xSquared ) { - const terms = [ constant, x, xSquared ].filter( function( term ) { + // Now hook up dynamic parts, setting their values to null + Orientation.VALUES.forEach( function( orientation ) { + if ( description.dimensionTypes.get( orientation ) === EntryType.DYNAMIC ) { + const nonErrorProperties = self.nonErrorPartitionSizeProperties.get( orientation ); + Property.multilink( nonErrorProperties, function() { + const terms = _.map( nonErrorProperties, 'value' ).filter( function( term ) { return term !== null; } ); - return terms.length ? new Polynomial( terms ) : null; - } ); - - // All of the entries for the challenge - Not including the polynomial "total" coefficient entries - const mainEntries = this.partitionSizeEntries.horizontal - .concat( this.partitionSizeEntries.vertical ) - .concat( _.flatten( this.partialProductSizeEntries ) ); - const checkingNotificationProperties = mainEntries.map( _.property( 'valueProperty' ) ) - .concat( this.totalCoefficientEntries.map( _.property( 'statusProperty' ) ) ); - - // @public {Property.} - Whether the check button should be enabled - this.allowCheckingProperty = new DerivedProperty( checkingNotificationProperties, function() { - const allDirtyCoefficients = _.every( self.totalCoefficientEntries, function( entry ) { - return entry.type === EntryType.EDITABLE && entry.statusProperty.value === EntryStatus.DIRTY; - } ); - const hasNullMain = _.some( mainEntries, function( entry ) { - return entry.valueProperty.value === null && entry.type === EntryType.EDITABLE; + const lostATerm = terms.length !== nonErrorProperties.length; + self.totalProperties.get( orientation ).value = ( terms.length && !lostATerm ) ? new Polynomial( terms ) : null; } ); - return !hasNullMain && !allDirtyCoefficients; - } ); - - /*---------------------------------------------------------------------------* - * Dynamic hooks - *----------------------------------------------------------------------------*/ - - // Now hook up dynamic parts, setting their values to null - Orientation.VALUES.forEach( function( orientation ) { - if ( description.dimensionTypes.get( orientation ) === EntryType.DYNAMIC ) { - const nonErrorProperties = self.nonErrorPartitionSizeProperties.get( orientation ); - Property.multilink( nonErrorProperties, function() { - const terms = _.map( nonErrorProperties, 'value' ).filter( function( term ) { - return term !== null; - } ); - const lostATerm = terms.length !== nonErrorProperties.length; - self.totalProperties.get( orientation ).value = ( terms.length && !lostATerm ) ? new Polynomial( terms ) : null; - } ); - } - } ); + } + } ); - // @private {boolean} - Pick an arbitrary side to be wrong in particular variables 6-1 cases, see - // https://github.com/phetsims/area-model-common/issues/42 - this.arbitraryNonUniqueWrongOrientation = phet.joist.random.nextBoolean() ? Orientation.HORIZONTAL : Orientation.VERTICAL; - } + // @private {boolean} - Pick an arbitrary side to be wrong in particular variables 6-1 cases, see + // https://github.com/phetsims/area-model-common/issues/42 + this.arbitraryNonUniqueWrongOrientation = phet.joist.random.nextBoolean() ? Orientation.HORIZONTAL : Orientation.VERTICAL; +} - areaModelCommon.register( 'AreaChallenge', AreaChallenge ); +areaModelCommon.register( 'AreaChallenge', AreaChallenge ); - return inherit( Object, AreaChallenge, { - /** - * Returns a list of all of the editable properties that are incorrect. - * @public - * - * @returns {Array.} - */ - getIncorrectEntries: function() { - const self = this; +export default inherit( Object, AreaChallenge, { + /** + * Returns a list of all of the editable properties that are incorrect. + * @public + * + * @returns {Array.} + */ + getIncorrectEntries: function() { + const self = this; - const incorrectEntries = []; + const incorrectEntries = []; - function compareEntry( entry, expectedValue ) { - if ( entry.valueProperty.value === null || !entry.valueProperty.value.equals( expectedValue ) ) { - incorrectEntries.push( entry ); - } + function compareEntry( entry, expectedValue ) { + if ( entry.valueProperty.value === null || !entry.valueProperty.value.equals( expectedValue ) ) { + incorrectEntries.push( entry ); } + } - // NOTE: Since the only non-unique case is variables 6-1, we just check our secondary properties. - if ( !this.description.unique ) { - // Logic described by https://github.com/phetsims/area-model-common/issues/39 - // Addendum to logic in https://github.com/phetsims/area-model-common/issues/42 - if ( this.hasNonUniqueBadMatch() ) { - incorrectEntries.push( this.swappableEntries.get( this.arbitraryNonUniqueWrongOrientation ) ); - } - else { - if ( !this.nonUniqueHorizontalMatches() ) { - incorrectEntries.push( this.swappableEntries.horizontal ); - } - if ( !this.nonUniqueVerticalMatches() ) { - incorrectEntries.push( this.swappableEntries.vertical ); - } - } + // NOTE: Since the only non-unique case is variables 6-1, we just check our secondary properties. + if ( !this.description.unique ) { + // Logic described by https://github.com/phetsims/area-model-common/issues/39 + // Addendum to logic in https://github.com/phetsims/area-model-common/issues/42 + if ( this.hasNonUniqueBadMatch() ) { + incorrectEntries.push( this.swappableEntries.get( this.arbitraryNonUniqueWrongOrientation ) ); } else { - this.partitionSizeEntries.horizontal.forEach( function( entry, index ) { - compareEntry( entry, self.partitionSizes.horizontal[ index ] ); - } ); - this.partitionSizeEntries.vertical.forEach( function( entry, index ) { - compareEntry( entry, self.partitionSizes.vertical[ index ] ); - } ); - dimensionForEach( 2, this.partialProductSizeEntries, function( entry, verticalIndex, horizontalIndex ) { - compareEntry( entry, self.partialProductSizes[ verticalIndex ][ horizontalIndex ] ); - } ); - - compareEntry( this.totalConstantEntry, this.total.getTerm( 0 ) ); - if ( this.totalInputMethod !== InputMethod.CONSTANT ) { - compareEntry( this.totalXEntry, this.total.getTerm( 1 ) ); + if ( !this.nonUniqueHorizontalMatches() ) { + incorrectEntries.push( this.swappableEntries.horizontal ); } - if ( this.totalInputMethod === InputMethod.POLYNOMIAL_2 ) { - compareEntry( this.totalXSquaredEntry, this.total.getTerm( 2 ) ); + if ( !this.nonUniqueVerticalMatches() ) { + incorrectEntries.push( this.swappableEntries.vertical ); } } - - return _.uniq( incorrectEntries ).filter( function( entry ) { - return entry.displayType === EntryDisplayType.EDITABLE; + } + else { + this.partitionSizeEntries.horizontal.forEach( function( entry, index ) { + compareEntry( entry, self.partitionSizes.horizontal[ index ] ); + } ); + this.partitionSizeEntries.vertical.forEach( function( entry, index ) { + compareEntry( entry, self.partitionSizes.vertical[ index ] ); + } ); + dimensionForEach( 2, this.partialProductSizeEntries, function( entry, verticalIndex, horizontalIndex ) { + compareEntry( entry, self.partialProductSizes[ verticalIndex ][ horizontalIndex ] ); } ); - }, - - /** - * Returns whether our horizontal (secondary) partition size equals one of the expected (secondary) partition sizes. - * @private - * - * @returns {boolean} - */ - nonUniqueHorizontalMatches: function() { - const actual = this.swappableEntries.horizontal.valueProperty.value; - return actual !== null && ( actual.equals( this.swappableSizes.horizontal ) || actual.equals( this.swappableSizes.vertical ) ); - }, - - /** - * Returns whether our vertical (secondary) partition size equals one of the expected (secondary) partition sizes. - * @private - * - * @returns {boolean} - */ - nonUniqueVerticalMatches: function() { - const actual = this.swappableEntries.vertical.valueProperty.value; - - return actual !== null && ( actual.equals( this.swappableSizes.horizontal ) || actual.equals( this.swappableSizes.vertical ) ); - }, - - /** - * Returns whether a permutation of our secondary partition sizes matches the expected sizes. Helpful for the case - * where values can be swapped between locations. - * @private - * - * @returns {boolean} - */ - hasNonUniqueMatch: function() { - const expected1 = this.swappableSizes.horizontal; - const expected2 = this.swappableSizes.vertical; - const actual1 = this.swappableEntries.horizontal.valueProperty.value; - const actual2 = this.swappableEntries.vertical.valueProperty.value; - - return actual1 !== null && actual2 !== null && - ( ( actual1.equals( expected1 ) && actual2.equals( expected2 ) ) || - ( actual1.equals( expected2 ) && actual2.equals( expected1 ) ) ); - }, - - /** - * Returns whether both properties match one answer but not the other. - * @private - * - * @returns {boolean} - */ - hasNonUniqueBadMatch: function() { - // Check for a case where both properties match one answer but not the other - return this.nonUniqueHorizontalMatches() && this.nonUniqueVerticalMatches() && !this.hasNonUniqueMatch(); - }, - - /** - * Remove highlights for non-unique changes, see https://github.com/phetsims/area-model-common/issues/42 - * @private - */ - checkNonUniqueChanges: function() { - if ( !this.description.unique ) { - if ( this.hasNonUniqueBadMatch() ) { - this.swappableEntries.horizontal.statusProperty.value = EntryStatus.NORMAL; - this.swappableEntries.vertical.statusProperty.value = EntryStatus.NORMAL; - } + compareEntry( this.totalConstantEntry, this.total.getTerm( 0 ) ); + if ( this.totalInputMethod !== InputMethod.CONSTANT ) { + compareEntry( this.totalXEntry, this.total.getTerm( 1 ) ); + } + if ( this.totalInputMethod === InputMethod.POLYNOMIAL_2 ) { + compareEntry( this.totalXSquaredEntry, this.total.getTerm( 2 ) ); } - }, + } - /** - * Shows the answers to the challenge. - * @public - */ - showAnswers: function() { - const self = this; + return _.uniq( incorrectEntries ).filter( function( entry ) { + return entry.displayType === EntryDisplayType.EDITABLE; + } ); + }, - // Match solutions for 6-1 variables, see https://github.com/phetsims/area-model-common/issues/42 - if ( !this.description.unique ) { - let reversed = false; + /** + * Returns whether our horizontal (secondary) partition size equals one of the expected (secondary) partition sizes. + * @private + * + * @returns {boolean} + */ + nonUniqueHorizontalMatches: function() { + const actual = this.swappableEntries.horizontal.valueProperty.value; + return actual !== null && ( actual.equals( this.swappableSizes.horizontal ) || actual.equals( this.swappableSizes.vertical ) ); + }, - const expected1 = this.swappableSizes.horizontal; - const expected2 = this.swappableSizes.vertical; + /** + * Returns whether our vertical (secondary) partition size equals one of the expected (secondary) partition sizes. + * @private + * + * @returns {boolean} + */ + nonUniqueVerticalMatches: function() { + const actual = this.swappableEntries.vertical.valueProperty.value; - const actual1Entry = this.swappableEntries.horizontal; - const actual2Entry = this.swappableEntries.vertical; + return actual !== null && ( actual.equals( this.swappableSizes.horizontal ) || actual.equals( this.swappableSizes.vertical ) ); + }, - const actual1 = actual1Entry.valueProperty.value; - const actual2 = actual2Entry.valueProperty.value; + /** + * Returns whether a permutation of our secondary partition sizes matches the expected sizes. Helpful for the case + * where values can be swapped between locations. + * @private + * + * @returns {boolean} + */ + hasNonUniqueMatch: function() { + const expected1 = this.swappableSizes.horizontal; + const expected2 = this.swappableSizes.vertical; - if ( actual1 && actual2 ) { - const matches1 = actual1.equals( expected1 ) || actual1.equals( expected2 ); - const matches2 = actual2.equals( expected1 ) || actual2.equals( expected2 ); + const actual1 = this.swappableEntries.horizontal.valueProperty.value; + const actual2 = this.swappableEntries.vertical.valueProperty.value; - if ( matches1 !== matches2 && ( actual1.equals( expected2 ) || actual2.equals( expected1 ) ) ) { - reversed = true; - } - } + return actual1 !== null && actual2 !== null && + ( ( actual1.equals( expected1 ) && actual2.equals( expected2 ) ) || + ( actual1.equals( expected2 ) && actual2.equals( expected1 ) ) ); + }, - if ( reversed ) { - actual1Entry.valueProperty.value = expected2; - actual2Entry.valueProperty.value = expected1; - this.totalProperties.horizontal.value = this.totals.vertical; - this.totalProperties.vertical.value = this.totals.horizontal; - } - else { - actual1Entry.valueProperty.value = expected1; - actual2Entry.valueProperty.value = expected2; - this.totalProperties.horizontal.value = this.totals.horizontal; - this.totalProperties.vertical.value = this.totals.vertical; - } - actual1Entry.statusProperty.value = EntryStatus.NORMAL; - actual2Entry.statusProperty.value = EntryStatus.NORMAL; - } - else { - this.partitionSizeEntries.horizontal.forEach( function( entry, index ) { - entry.valueProperty.value = self.partitionSizes.horizontal[ index ]; - } ); - this.partitionSizeEntries.vertical.forEach( function( entry, index ) { - entry.valueProperty.value = self.partitionSizes.vertical[ index ]; - } ); + /** + * Returns whether both properties match one answer but not the other. + * @private + * + * @returns {boolean} + */ + hasNonUniqueBadMatch: function() { + // Check for a case where both properties match one answer but not the other + return this.nonUniqueHorizontalMatches() && this.nonUniqueVerticalMatches() && !this.hasNonUniqueMatch(); + }, - this.totalProperties.horizontal.value = this.totals.horizontal; - this.totalProperties.vertical.value = this.totals.vertical; + /** + * Remove highlights for non-unique changes, see https://github.com/phetsims/area-model-common/issues/42 + * @private + */ + checkNonUniqueChanges: function() { + if ( !this.description.unique ) { + if ( this.hasNonUniqueBadMatch() ) { + this.swappableEntries.horizontal.statusProperty.value = EntryStatus.NORMAL; + this.swappableEntries.vertical.statusProperty.value = EntryStatus.NORMAL; } + } + }, - dimensionForEach( 2, this.partialProductSizeEntries, function( entry, verticalIndex, horizontalIndex ) { - entry.valueProperty.value = self.partialProductSizes[ verticalIndex ][ horizontalIndex ]; - entry.statusProperty.value = EntryStatus.NORMAL; - } ); + /** + * Shows the answers to the challenge. + * @public + */ + showAnswers: function() { + const self = this; - this.totalConstantEntry.valueProperty.value = this.total.getTerm( 0 ); - this.totalXEntry.valueProperty.value = this.total.getTerm( 1 ); - this.totalXSquaredEntry.valueProperty.value = this.total.getTerm( 2 ); - this.totalConstantEntry.statusProperty.value = EntryStatus.NORMAL; - this.totalXEntry.statusProperty.value = EntryStatus.NORMAL; - this.totalXSquaredEntry.statusProperty.value = EntryStatus.NORMAL; - }, - - /** - * Checks the user's input against the known answer. - * @public - * - * @returns {number} - The amount of score gained - */ - check: function() { - let scoreIncrease = 0; - - const badEntries = this.getIncorrectEntries(); - const isCorrect = badEntries.length === 0; - - const currentState = this.stateProperty.value; - - if ( !isCorrect ) { - badEntries.forEach( function( badEntry ) { - badEntry.statusProperty.value = EntryStatus.INCORRECT; - } ); - } + // Match solutions for 6-1 variables, see https://github.com/phetsims/area-model-common/issues/42 + if ( !this.description.unique ) { + let reversed = false; + + const expected1 = this.swappableSizes.horizontal; + const expected2 = this.swappableSizes.vertical; + + const actual1Entry = this.swappableEntries.horizontal; + const actual2Entry = this.swappableEntries.vertical; + + const actual1 = actual1Entry.valueProperty.value; + const actual2 = actual2Entry.valueProperty.value; - if ( currentState === GameState.FIRST_ATTEMPT ) { - if ( isCorrect ) { - scoreIncrease = 2; + if ( actual1 && actual2 ) { + const matches1 = actual1.equals( expected1 ) || actual1.equals( expected2 ); + const matches2 = actual2.equals( expected1 ) || actual2.equals( expected2 ); + + if ( matches1 !== matches2 && ( actual1.equals( expected2 ) || actual2.equals( expected1 ) ) ) { + reversed = true; } - this.stateProperty.value = isCorrect ? GameState.CORRECT_ANSWER : GameState.WRONG_FIRST_ANSWER; } - else if ( currentState === GameState.SECOND_ATTEMPT ) { - if ( isCorrect ) { - scoreIncrease = 1; - } - this.stateProperty.value = isCorrect ? GameState.CORRECT_ANSWER : GameState.WRONG_SECOND_ANSWER; + + if ( reversed ) { + actual1Entry.valueProperty.value = expected2; + actual2Entry.valueProperty.value = expected1; + this.totalProperties.horizontal.value = this.totals.vertical; + this.totalProperties.vertical.value = this.totals.horizontal; } else { - throw new Error( 'How is check possible here?' ); + actual1Entry.valueProperty.value = expected1; + actual2Entry.valueProperty.value = expected2; + this.totalProperties.horizontal.value = this.totals.horizontal; + this.totalProperties.vertical.value = this.totals.vertical; } + actual1Entry.statusProperty.value = EntryStatus.NORMAL; + actual2Entry.statusProperty.value = EntryStatus.NORMAL; + } + else { + this.partitionSizeEntries.horizontal.forEach( function( entry, index ) { + entry.valueProperty.value = self.partitionSizes.horizontal[ index ]; + } ); + this.partitionSizeEntries.vertical.forEach( function( entry, index ) { + entry.valueProperty.value = self.partitionSizes.vertical[ index ]; + } ); - return scoreIncrease; - }, - - /** - * Move to try another time. - * @public - */ - tryAgain: function() { - this.stateProperty.value = GameState.SECOND_ATTEMPT; + this.totalProperties.horizontal.value = this.totals.horizontal; + this.totalProperties.vertical.value = this.totals.vertical; } - }, { - - /** - * Generates a series of (semi) random terms for partition sizes for a particular orientation. - * @private - * - * @param {number} quantity - * @param {boolean} allowExponents - Whether exponents (powers of x) are allowed for this area - * @returns {Array.} - */ - generatePartitionTerms: function( quantity, allowExponents ) { - const maxPower = quantity - 1; - return _.range( maxPower, -1 ).map( function( power ) { - return AreaChallenge.generateTerm( power, maxPower, quantity, allowExponents ); + + dimensionForEach( 2, this.partialProductSizeEntries, function( entry, verticalIndex, horizontalIndex ) { + entry.valueProperty.value = self.partialProductSizes[ verticalIndex ][ horizontalIndex ]; + entry.statusProperty.value = EntryStatus.NORMAL; + } ); + + this.totalConstantEntry.valueProperty.value = this.total.getTerm( 0 ); + this.totalXEntry.valueProperty.value = this.total.getTerm( 1 ); + this.totalXSquaredEntry.valueProperty.value = this.total.getTerm( 2 ); + this.totalConstantEntry.statusProperty.value = EntryStatus.NORMAL; + this.totalXEntry.statusProperty.value = EntryStatus.NORMAL; + this.totalXSquaredEntry.statusProperty.value = EntryStatus.NORMAL; + }, + + /** + * Checks the user's input against the known answer. + * @public + * + * @returns {number} - The amount of score gained + */ + check: function() { + let scoreIncrease = 0; + + const badEntries = this.getIncorrectEntries(); + const isCorrect = badEntries.length === 0; + + const currentState = this.stateProperty.value; + + if ( !isCorrect ) { + badEntries.forEach( function( badEntry ) { + badEntry.statusProperty.value = EntryStatus.INCORRECT; } ); - }, - - /** - * Generates a (semi) random term for a partition size. - * @private - * - * @param {number} power - Power of 'x' or '10' that the single digit is multiplied times - * @param {number} maxPower - Maximum power for all terms of this orientation. - * @param {number} quantity - Quantity of terms generated total - * @param {boolean} allowExponents - Whether exponents (powers of x) are allowed - * @returns {Term} - */ - generateTerm: function( power, maxPower, quantity, allowExponents ) { - if ( allowExponents ) { - - // Don't let leading x or x^2 have a coefficient. - if ( power === maxPower && power > 0 ) { - return new Term( 1, power ); - } - else { - const sign = phet.joist.random.nextBoolean() ? 1 : -1; + } - // Exclude a 1 if our length is 1 (so that we don't just have a single 1 as a dimensinon, so there is the - // ability to have a partition line) - const digit = phet.joist.random.nextIntBetween( ( sign > 0 && quantity === 1 ) ? 2 : 1, 9 ); - return new Term( sign * digit, power ); - } + if ( currentState === GameState.FIRST_ATTEMPT ) { + if ( isCorrect ) { + scoreIncrease = 2; + } + this.stateProperty.value = isCorrect ? GameState.CORRECT_ANSWER : GameState.WRONG_FIRST_ANSWER; + } + else if ( currentState === GameState.SECOND_ATTEMPT ) { + if ( isCorrect ) { + scoreIncrease = 1; + } + this.stateProperty.value = isCorrect ? GameState.CORRECT_ANSWER : GameState.WRONG_SECOND_ANSWER; + } + else { + throw new Error( 'How is check possible here?' ); + } + + return scoreIncrease; + }, + + /** + * Move to try another time. + * @public + */ + tryAgain: function() { + this.stateProperty.value = GameState.SECOND_ATTEMPT; + } +}, { + + /** + * Generates a series of (semi) random terms for partition sizes for a particular orientation. + * @private + * + * @param {number} quantity + * @param {boolean} allowExponents - Whether exponents (powers of x) are allowed for this area + * @returns {Array.} + */ + generatePartitionTerms: function( quantity, allowExponents ) { + const maxPower = quantity - 1; + return _.range( maxPower, -1 ).map( function( power ) { + return AreaChallenge.generateTerm( power, maxPower, quantity, allowExponents ); + } ); + }, + + /** + * Generates a (semi) random term for a partition size. + * @private + * + * @param {number} power - Power of 'x' or '10' that the single digit is multiplied times + * @param {number} maxPower - Maximum power for all terms of this orientation. + * @param {number} quantity - Quantity of terms generated total + * @param {boolean} allowExponents - Whether exponents (powers of x) are allowed + * @returns {Term} + */ + generateTerm: function( power, maxPower, quantity, allowExponents ) { + if ( allowExponents ) { + + // Don't let leading x or x^2 have a coefficient. + if ( power === maxPower && power > 0 ) { + return new Term( 1, power ); } else { + const sign = phet.joist.random.nextBoolean() ? 1 : -1; - // Exclude a 1 if our length is 1 - return new Term( phet.joist.random.nextIntBetween( quantity === 1 ? 2 : 1, 9 ) * Math.pow( 10, power ) ); + // Exclude a 1 if our length is 1 (so that we don't just have a single 1 as a dimensinon, so there is the + // ability to have a partition line) + const digit = phet.joist.random.nextIntBetween( ( sign > 0 && quantity === 1 ) ? 2 : 1, 9 ); + return new Term( sign * digit, power ); } } - } ); -} ); + else { + + // Exclude a 1 if our length is 1 + return new Term( phet.joist.random.nextIntBetween( quantity === 1 ? 2 : 1, 9 ) * Math.pow( 10, power ) ); + } + } +} ); \ No newline at end of file diff --git a/js/game/model/AreaChallengeDescription.js b/js/game/model/AreaChallengeDescription.js index 5f7197f9..9dc28a76 100644 --- a/js/game/model/AreaChallengeDescription.js +++ b/js/game/model/AreaChallengeDescription.js @@ -5,831 +5,827 @@ * * @author Jonathan Olson */ -define( require => { - 'use strict'; - - // modules - const AreaChallengeType = require( 'AREA_MODEL_COMMON/game/model/AreaChallengeType' ); - const areaModelCommon = require( 'AREA_MODEL_COMMON/areaModelCommon' ); - const EntryType = require( 'AREA_MODEL_COMMON/game/model/EntryType' ); - const GenericLayout = require( 'AREA_MODEL_COMMON/generic/model/GenericLayout' ); - const inherit = require( 'PHET_CORE/inherit' ); - const merge = require( 'PHET_CORE/merge' ); - const OrientationPair = require( 'AREA_MODEL_COMMON/common/model/OrientationPair' ); - const Permutation = require( 'DOT/Permutation' ); - - // strings - const levelPromptOneProductOneLengthString = require( 'string!AREA_MODEL_COMMON/levelPrompt.oneProduct.oneLength' ); - const levelPromptOneProductTextString = require( 'string!AREA_MODEL_COMMON/levelPrompt.oneProduct.text' ); - const levelPromptOneProductTotalAreaString = require( 'string!AREA_MODEL_COMMON/levelPrompt.oneProduct.totalArea' ); - const levelPromptThreeLengthsString = require( 'string!AREA_MODEL_COMMON/levelPrompt.threeLengths' ); - const levelPromptTotalAreaString = require( 'string!AREA_MODEL_COMMON/levelPrompt.totalArea' ); - const levelPromptTwoLengthsString = require( 'string!AREA_MODEL_COMMON/levelPrompt.twoLengths' ); - const levelPromptTwoProductsString = require( 'string!AREA_MODEL_COMMON/levelPrompt.twoProducts' ); - - // shortcuts - const EDITABLE = EntryType.EDITABLE; - const DYNAMIC = EntryType.DYNAMIC; - const GIVEN = EntryType.GIVEN; - - // We need the ability to generate random permutations for different numbers of elements. It's simplest if we - // enumerate the possibilities here. - const permutations = { - 1: Permutation.permutations( 1 ), - 2: Permutation.permutations( 2 ), - 3: Permutation.permutations( 3 ) - }; +import Permutation from '../../../../dot/js/Permutation.js'; +import inherit from '../../../../phet-core/js/inherit.js'; +import merge from '../../../../phet-core/js/merge.js'; +import areaModelCommonStrings from '../../area-model-common-strings.js'; +import areaModelCommon from '../../areaModelCommon.js'; +import OrientationPair from '../../common/model/OrientationPair.js'; +import GenericLayout from '../../generic/model/GenericLayout.js'; +import AreaChallengeType from './AreaChallengeType.js'; +import EntryType from './EntryType.js'; + +const levelPromptOneProductOneLengthString = areaModelCommonStrings.levelPrompt.oneProduct.oneLength; +const levelPromptOneProductTextString = areaModelCommonStrings.levelPrompt.oneProduct.text; +const levelPromptOneProductTotalAreaString = areaModelCommonStrings.levelPrompt.oneProduct.totalArea; +const levelPromptThreeLengthsString = areaModelCommonStrings.levelPrompt.threeLengths; +const levelPromptTotalAreaString = areaModelCommonStrings.levelPrompt.totalArea; +const levelPromptTwoLengthsString = areaModelCommonStrings.levelPrompt.twoLengths; +const levelPromptTwoProductsString = areaModelCommonStrings.levelPrompt.twoProducts; + +// shortcuts +const EDITABLE = EntryType.EDITABLE; +const DYNAMIC = EntryType.DYNAMIC; +const GIVEN = EntryType.GIVEN; + +// We need the ability to generate random permutations for different numbers of elements. It's simplest if we +// enumerate the possibilities here. +const permutations = { + 1: Permutation.permutations( 1 ), + 2: Permutation.permutations( 2 ), + 3: Permutation.permutations( 3 ) +}; + +/** + * @constructor + * @extends {Object} + * + * @param {Object} config + */ +function AreaChallengeDescription( config ) { + config = merge( { + // required + horizontal: null, // {Array.} + vertical: null, // {Array.} + products: null, // {Array.>} + total: null, // {EntryType} + horizontalTotal: null, // {EntryType} + verticalTotal: null, // {EntryType} + type: null, // {AreaChallengeType} + + // optional + shufflable: true, + unique: true + }, config ); + + assert && assert( Array.isArray( config.horizontal ) ); + assert && assert( Array.isArray( config.vertical ) ); + assert && assert( Array.isArray( config.products ) ); + assert && assert( _.includes( EntryType.VALUES, config.total ) ); + assert && assert( _.includes( EntryType.VALUES, config.horizontalTotal ) ); + assert && assert( _.includes( EntryType.VALUES, config.verticalTotal ) ); + assert && assert( _.includes( AreaChallengeType.VALUES, config.type ) ); + + // @public {OrientationPair.>} - Entry types for partition sizes + this.partitionTypes = new OrientationPair( config.horizontal, config.vertical ); + + // @public {Array.>} - Entry types for partitioned areas + this.productTypes = config.products; + + // @public {OrientationPair.} - Entry types for horizontal and vertical dimension totals + this.dimensionTypes = new OrientationPair( config.horizontalTotal, config.verticalTotal ); + + // @public {EntryType} - Entry type for the total area + this.totalType = config.total; + + // @public {AreaChallengeType} - The type of challenge + this.type = config.type; + + // @public {boolean} + this.allowExponents = this.type === AreaChallengeType.VARIABLES; + + // @public {boolean} - Whether transposing is supported + this.transposable = this.type === AreaChallengeType.NUMBERS; + + // @public {boolean} + this.shufflable = config.shufflable; + + // @public {boolean} + this.unique = config.unique; + + // @public {GenericLayout} + this.layout = GenericLayout.fromValues( config.horizontal.length, config.vertical.length ); +} + +areaModelCommon.register( 'AreaChallengeDescription', AreaChallengeDescription ); + +/** + * Returns a string key used for looking up the proper prompt in promptMap below. + * @private + * + * @param {boolean} hasAreaEntry + * @param {number} numProductEntries + * @param {number} numPartitionEntries + * @returns {string} + */ +function getPromptKey( hasAreaEntry, numProductEntries, numPartitionEntries ) { + return hasAreaEntry + ',' + numProductEntries + ',' + numPartitionEntries; +} + +const promptMap = {}; +promptMap[ getPromptKey( true, 0, 0 ) ] = levelPromptTotalAreaString; +promptMap[ getPromptKey( false, 1, 0 ) ] = levelPromptOneProductTextString; +promptMap[ getPromptKey( false, 2, 0 ) ] = levelPromptTwoProductsString; +promptMap[ getPromptKey( true, 1, 0 ) ] = levelPromptOneProductTotalAreaString; +promptMap[ getPromptKey( false, 1, 1 ) ] = levelPromptOneProductOneLengthString; +promptMap[ getPromptKey( false, 0, 2 ) ] = levelPromptTwoLengthsString; +promptMap[ getPromptKey( false, 0, 3 ) ] = levelPromptThreeLengthsString; + +function isEditable( type ) { + return type === EntryType.EDITABLE; +} + +inherit( Object, AreaChallengeDescription, { /** - * @constructor - * @extends {Object} + * Returns the string representing the prompt for this challenge (what should be done to solve it). + * @public * - * @param {Object} config + * @returns {string} */ - function AreaChallengeDescription( config ) { - config = merge( { - // required - horizontal: null, // {Array.} - vertical: null, // {Array.} - products: null, // {Array.>} - total: null, // {EntryType} - horizontalTotal: null, // {EntryType} - verticalTotal: null, // {EntryType} - type: null, // {AreaChallengeType} - - // optional - shufflable: true, - unique: true - }, config ); - - assert && assert( Array.isArray( config.horizontal ) ); - assert && assert( Array.isArray( config.vertical ) ); - assert && assert( Array.isArray( config.products ) ); - assert && assert( _.includes( EntryType.VALUES, config.total ) ); - assert && assert( _.includes( EntryType.VALUES, config.horizontalTotal ) ); - assert && assert( _.includes( EntryType.VALUES, config.verticalTotal ) ); - assert && assert( _.includes( AreaChallengeType.VALUES, config.type ) ); - - // @public {OrientationPair.>} - Entry types for partition sizes - this.partitionTypes = new OrientationPair( config.horizontal, config.vertical ); - - // @public {Array.>} - Entry types for partitioned areas - this.productTypes = config.products; - - // @public {OrientationPair.} - Entry types for horizontal and vertical dimension totals - this.dimensionTypes = new OrientationPair( config.horizontalTotal, config.verticalTotal ); - - // @public {EntryType} - Entry type for the total area - this.totalType = config.total; - - // @public {AreaChallengeType} - The type of challenge - this.type = config.type; - - // @public {boolean} - this.allowExponents = this.type === AreaChallengeType.VARIABLES; - - // @public {boolean} - Whether transposing is supported - this.transposable = this.type === AreaChallengeType.NUMBERS; - - // @public {boolean} - this.shufflable = config.shufflable; - - // @public {boolean} - this.unique = config.unique; - - // @public {GenericLayout} - this.layout = GenericLayout.fromValues( config.horizontal.length, config.vertical.length ); - } + getPromptString: function() { + const hasAreaEntry = isEditable( this.totalType ); + const numProductEntries = _.flatten( this.productTypes ).filter( isEditable ).length; + const numPartitionEntries = this.partitionTypes.horizontal.concat( this.partitionTypes.vertical ).filter( isEditable ).length; - areaModelCommon.register( 'AreaChallengeDescription', AreaChallengeDescription ); + const text = promptMap[ getPromptKey( hasAreaEntry, numProductEntries, numPartitionEntries ) ]; + assert && assert( text ); + + return text; + }, /** - * Returns a string key used for looking up the proper prompt in promptMap below. - * @private + * Creates a permuted/transposed version of this description, where allowed. + * @public * - * @param {boolean} hasAreaEntry - * @param {number} numProductEntries - * @param {number} numPartitionEntries - * @returns {string} + * @returns {AreaChallengeDescription} */ - function getPromptKey( hasAreaEntry, numProductEntries, numPartitionEntries ) { - return hasAreaEntry + ',' + numProductEntries + ',' + numPartitionEntries; - } + getPermutedDescription: function() { + const options = { + horizontal: this.partitionTypes.horizontal, + vertical: this.partitionTypes.vertical, + products: this.productTypes, + total: this.totalType, + horizontalTotal: this.dimensionTypes.horizontal, + verticalTotal: this.dimensionTypes.vertical, + type: this.type, + transposable: this.transposable, + unique: this.unique + }; + + if ( this.shufflable ) { + // Horizontal shuffle + const horizontalPermutation = phet.joist.random.sample( permutations[ options.horizontal.length ] ); + options.horizontal = horizontalPermutation.apply( options.horizontal ); + options.products = options.products.map( function( row ) { + return horizontalPermutation.apply( row ); + } ); + + // Vertical shuffle + const verticalPermutation = phet.joist.random.sample( permutations[ options.vertical.length ] ); + options.vertical = verticalPermutation.apply( options.vertical ); + options.products = verticalPermutation.apply( options.products ); + } - const promptMap = {}; - promptMap[ getPromptKey( true, 0, 0 ) ] = levelPromptTotalAreaString; - promptMap[ getPromptKey( false, 1, 0 ) ] = levelPromptOneProductTextString; - promptMap[ getPromptKey( false, 2, 0 ) ] = levelPromptTwoProductsString; - promptMap[ getPromptKey( true, 1, 0 ) ] = levelPromptOneProductTotalAreaString; - promptMap[ getPromptKey( false, 1, 1 ) ] = levelPromptOneProductOneLengthString; - promptMap[ getPromptKey( false, 0, 2 ) ] = levelPromptTwoLengthsString; - promptMap[ getPromptKey( false, 0, 3 ) ] = levelPromptThreeLengthsString; - - function isEditable( type ) { - return type === EntryType.EDITABLE; - } + if ( this.transposable && phet.joist.random.nextBoolean() ) { + const tmpPartition = options.horizontal; + options.horizontal = options.vertical; + options.vertical = tmpPartition; - inherit( Object, AreaChallengeDescription, { - /** - * Returns the string representing the prompt for this challenge (what should be done to solve it). - * @public - * - * @returns {string} - */ - getPromptString: function() { - const hasAreaEntry = isEditable( this.totalType ); - const numProductEntries = _.flatten( this.productTypes ).filter( isEditable ).length; - const numPartitionEntries = this.partitionTypes.horizontal.concat( this.partitionTypes.vertical ).filter( isEditable ).length; - - const text = promptMap[ getPromptKey( hasAreaEntry, numProductEntries, numPartitionEntries ) ]; - assert && assert( text ); - - return text; - }, - - /** - * Creates a permuted/transposed version of this description, where allowed. - * @public - * - * @returns {AreaChallengeDescription} - */ - getPermutedDescription: function() { - const options = { - horizontal: this.partitionTypes.horizontal, - vertical: this.partitionTypes.vertical, - products: this.productTypes, - total: this.totalType, - horizontalTotal: this.dimensionTypes.horizontal, - verticalTotal: this.dimensionTypes.vertical, - type: this.type, - transposable: this.transposable, - unique: this.unique - }; - - if ( this.shufflable ) { - // Horizontal shuffle - const horizontalPermutation = phet.joist.random.sample( permutations[ options.horizontal.length ] ); - options.horizontal = horizontalPermutation.apply( options.horizontal ); - options.products = options.products.map( function( row ) { - return horizontalPermutation.apply( row ); - } ); + const tmpTotal = options.horizontalTotal; + options.horizontalTotal = options.verticalTotal; + options.verticalTotal = tmpTotal; - // Vertical shuffle - const verticalPermutation = phet.joist.random.sample( permutations[ options.vertical.length ] ); - options.vertical = verticalPermutation.apply( options.vertical ); - options.products = verticalPermutation.apply( options.products ); - } - - if ( this.transposable && phet.joist.random.nextBoolean() ) { - const tmpPartition = options.horizontal; - options.horizontal = options.vertical; - options.vertical = tmpPartition; - - const tmpTotal = options.horizontalTotal; - options.horizontalTotal = options.verticalTotal; - options.verticalTotal = tmpTotal; - - options.products = _.range( options.vertical.length ).map( function( verticalIndex ) { - return _.range( options.horizontal.length ).map( function( horizontalIndex ) { - return options.products[ horizontalIndex ][ verticalIndex ]; - } ); + options.products = _.range( options.vertical.length ).map( function( verticalIndex ) { + return _.range( options.horizontal.length ).map( function( horizontalIndex ) { + return options.products[ horizontalIndex ][ verticalIndex ]; } ); - } - - return new AreaChallengeDescription( options ); - }, - - /** - * Returns a conditional value (like a ternary) based on whether this is a number or variable challenge. - * @public - * - * @param {*} numberTypeValue - * @param {*} variableTypeValue - * @returns {*} - */ - numberOrVariable: function( numberTypeValue, variableTypeValue ) { - return this.type === AreaChallengeType.VARIABLES ? variableTypeValue : numberTypeValue; + } ); } - } ); - - /*---------------------------------------------------------------------------* - * Numbers 1 - *----------------------------------------------------------------------------*/ - - // L1-1 - AreaChallengeDescription.LEVEL_1_NUMBERS_1 = new AreaChallengeDescription( { - horizontal: [ GIVEN, GIVEN ], - vertical: [ GIVEN ], - products: [ - [ GIVEN, GIVEN ] - ], - total: EDITABLE, - horizontalTotal: GIVEN, - verticalTotal: GIVEN, - type: AreaChallengeType.NUMBERS - } ); - - // L1-2 - AreaChallengeDescription.LEVEL_1_NUMBERS_2 = new AreaChallengeDescription( { - horizontal: [ GIVEN, GIVEN ], - vertical: [ GIVEN ], - products: [ - [ EDITABLE, GIVEN ] - ], - total: GIVEN, - horizontalTotal: GIVEN, - verticalTotal: GIVEN, - type: AreaChallengeType.NUMBERS - } ); - - // L1-3 - AreaChallengeDescription.LEVEL_1_NUMBERS_3 = new AreaChallengeDescription( { - horizontal: [ GIVEN, GIVEN, GIVEN ], - vertical: [ GIVEN ], - products: [ - [ GIVEN, GIVEN, GIVEN ] - ], - total: EDITABLE, - horizontalTotal: GIVEN, - verticalTotal: GIVEN, - type: AreaChallengeType.NUMBERS - } ); - - // L1-4 - AreaChallengeDescription.LEVEL_1_NUMBERS_4 = new AreaChallengeDescription( { - horizontal: [ GIVEN, GIVEN, GIVEN ], - vertical: [ GIVEN ], - products: [ - [ EDITABLE, GIVEN, GIVEN ] - ], - total: GIVEN, - horizontalTotal: GIVEN, - verticalTotal: GIVEN, - type: AreaChallengeType.NUMBERS - } ); - - // L1-5 - AreaChallengeDescription.LEVEL_1_NUMBERS_5 = new AreaChallengeDescription( { - horizontal: [ GIVEN, GIVEN ], - vertical: [ GIVEN, GIVEN ], - products: [ - [ GIVEN, GIVEN ], - [ GIVEN, GIVEN ] - ], - total: EDITABLE, - horizontalTotal: GIVEN, - verticalTotal: GIVEN, - type: AreaChallengeType.NUMBERS - } ); - - // L1-6 - AreaChallengeDescription.LEVEL_1_NUMBERS_6 = new AreaChallengeDescription( { - horizontal: [ GIVEN, GIVEN ], - vertical: [ GIVEN, GIVEN ], - products: [ - [ EDITABLE, GIVEN ], - [ GIVEN, GIVEN ] - ], - total: GIVEN, - horizontalTotal: GIVEN, - verticalTotal: GIVEN, - type: AreaChallengeType.NUMBERS - } ); - - /*---------------------------------------------------------------------------* - * Numbers 2 - *----------------------------------------------------------------------------*/ - - // L2-1 - AreaChallengeDescription.LEVEL_2_NUMBERS_1 = new AreaChallengeDescription( { - horizontal: [ GIVEN, GIVEN ], - vertical: [ GIVEN, GIVEN ], - products: [ - [ EDITABLE, GIVEN ], - [ GIVEN, EDITABLE ] - ], - total: GIVEN, - horizontalTotal: GIVEN, - verticalTotal: GIVEN, - type: AreaChallengeType.NUMBERS - } ); - - // L2-2 - AreaChallengeDescription.LEVEL_2_NUMBERS_2 = new AreaChallengeDescription( { - horizontal: [ GIVEN, GIVEN ], - vertical: [ GIVEN, GIVEN ], - products: [ - [ EDITABLE, GIVEN ], - [ GIVEN, GIVEN ] - ], - total: EDITABLE, - horizontalTotal: GIVEN, - verticalTotal: GIVEN, - type: AreaChallengeType.NUMBERS - } ); - - // L2-3 - AreaChallengeDescription.LEVEL_2_NUMBERS_3 = new AreaChallengeDescription( { - horizontal: [ GIVEN, GIVEN, GIVEN ], - vertical: [ GIVEN, GIVEN ], - products: [ - [ EDITABLE, GIVEN, GIVEN ], - [ GIVEN, EDITABLE, GIVEN ] - ], - total: GIVEN, - horizontalTotal: GIVEN, - verticalTotal: GIVEN, - type: AreaChallengeType.NUMBERS - } ); - - // L2-4 - AreaChallengeDescription.LEVEL_2_NUMBERS_4 = new AreaChallengeDescription( { - horizontal: [ GIVEN, GIVEN, GIVEN ], - vertical: [ GIVEN, GIVEN ], - products: [ - [ EDITABLE, GIVEN, GIVEN ], - [ GIVEN, GIVEN, GIVEN ] - ], - total: EDITABLE, - horizontalTotal: GIVEN, - verticalTotal: GIVEN, - type: AreaChallengeType.NUMBERS - } ); - - // L2-5 - AreaChallengeDescription.LEVEL_2_NUMBERS_5 = new AreaChallengeDescription( { - horizontal: [ GIVEN, GIVEN, GIVEN ], - vertical: [ GIVEN, GIVEN, GIVEN ], - products: [ - [ EDITABLE, GIVEN, GIVEN ], - [ GIVEN, EDITABLE, GIVEN ], - [ GIVEN, GIVEN, GIVEN ] - ], - total: GIVEN, - horizontalTotal: GIVEN, - verticalTotal: GIVEN, - type: AreaChallengeType.NUMBERS - } ); - - /*---------------------------------------------------------------------------* - * Numbers 3 - *----------------------------------------------------------------------------*/ - - // L3-1 - AreaChallengeDescription.LEVEL_3_NUMBERS_1 = new AreaChallengeDescription( { - horizontal: [ EDITABLE, EDITABLE ], - vertical: [ GIVEN ], - products: [ - [ DYNAMIC, GIVEN ] - ], - total: GIVEN, - horizontalTotal: GIVEN, - verticalTotal: GIVEN, - type: AreaChallengeType.NUMBERS - } ); - - // L3-2 - AreaChallengeDescription.LEVEL_3_NUMBERS_2 = new AreaChallengeDescription( { - horizontal: [ EDITABLE, GIVEN ], - vertical: [ EDITABLE ], - products: [ - [ DYNAMIC, GIVEN ] - ], - total: GIVEN, - horizontalTotal: GIVEN, - verticalTotal: DYNAMIC, - type: AreaChallengeType.NUMBERS - } ); - - // L3-3 - AreaChallengeDescription.LEVEL_3_NUMBERS_3 = new AreaChallengeDescription( { - horizontal: [ EDITABLE, EDITABLE, GIVEN ], - vertical: [ GIVEN ], - products: [ - [ DYNAMIC, GIVEN, GIVEN ] - ], - total: GIVEN, - horizontalTotal: GIVEN, - verticalTotal: GIVEN, - type: AreaChallengeType.NUMBERS - } ); - - // L3-4 - AreaChallengeDescription.LEVEL_3_NUMBERS_4 = new AreaChallengeDescription( { - horizontal: [ EDITABLE, GIVEN, GIVEN ], - vertical: [ EDITABLE ], - products: [ - [ GIVEN, GIVEN, GIVEN ] - ], - total: GIVEN, - horizontalTotal: GIVEN, - verticalTotal: DYNAMIC, - type: AreaChallengeType.NUMBERS - } ); - - // L3-5 - AreaChallengeDescription.LEVEL_3_NUMBERS_5 = new AreaChallengeDescription( { - horizontal: [ EDITABLE, GIVEN ], - vertical: [ EDITABLE, GIVEN ], - products: [ - [ GIVEN, DYNAMIC ], - [ DYNAMIC, GIVEN ] - ], - total: GIVEN, - horizontalTotal: GIVEN, - verticalTotal: GIVEN, - type: AreaChallengeType.NUMBERS - } ); - - // L3-6 - AreaChallengeDescription.LEVEL_3_NUMBERS_6 = new AreaChallengeDescription( { - horizontal: [ EDITABLE, GIVEN ], - vertical: [ EDITABLE, GIVEN ], - products: [ - [ DYNAMIC, GIVEN ], - [ GIVEN, GIVEN ] - ], - total: GIVEN, - horizontalTotal: GIVEN, - verticalTotal: GIVEN, - type: AreaChallengeType.NUMBERS - } ); - - /*---------------------------------------------------------------------------* - * Numbers 4 - *----------------------------------------------------------------------------*/ - - // L4-1 - AreaChallengeDescription.LEVEL_4_NUMBERS_1 = new AreaChallengeDescription( { - horizontal: [ EDITABLE, EDITABLE, EDITABLE ], - vertical: [ GIVEN, GIVEN ], - products: [ - [ DYNAMIC, GIVEN, GIVEN ], - [ GIVEN, DYNAMIC, GIVEN ] - ], - total: GIVEN, - horizontalTotal: GIVEN, - verticalTotal: GIVEN, - type: AreaChallengeType.NUMBERS - } ); - - // L4-2 - AreaChallengeDescription.LEVEL_4_NUMBERS_2 = new AreaChallengeDescription( { - horizontal: [ EDITABLE, EDITABLE, GIVEN ], - vertical: [ EDITABLE, GIVEN ], - products: [ - [ DYNAMIC, GIVEN, GIVEN ], - [ GIVEN, GIVEN, GIVEN ] - ], - total: GIVEN, - horizontalTotal: GIVEN, - verticalTotal: GIVEN, - type: AreaChallengeType.NUMBERS - } ); - - // L4-3 - AreaChallengeDescription.LEVEL_4_NUMBERS_3 = new AreaChallengeDescription( { - horizontal: [ EDITABLE, EDITABLE, GIVEN ], - vertical: [ EDITABLE, GIVEN ], - products: [ - [ DYNAMIC, GIVEN, GIVEN ], - [ GIVEN, DYNAMIC, GIVEN ] - ], - total: GIVEN, - horizontalTotal: GIVEN, - verticalTotal: GIVEN, - type: AreaChallengeType.NUMBERS - } ); - - // L4-4 - AreaChallengeDescription.LEVEL_4_NUMBERS_4 = new AreaChallengeDescription( { - horizontal: [ EDITABLE, EDITABLE, EDITABLE ], - vertical: [ GIVEN, GIVEN, GIVEN ], - products: [ - [ DYNAMIC, GIVEN, GIVEN ], - [ GIVEN, DYNAMIC, GIVEN ], - [ GIVEN, GIVEN, DYNAMIC ] - ], - total: GIVEN, - horizontalTotal: GIVEN, - verticalTotal: GIVEN, - type: AreaChallengeType.NUMBERS - } ); - - // L4-5 - AreaChallengeDescription.LEVEL_4_NUMBERS_5 = new AreaChallengeDescription( { - horizontal: [ EDITABLE, EDITABLE, GIVEN ], - vertical: [ EDITABLE, GIVEN, GIVEN ], - products: [ - [ DYNAMIC, GIVEN, GIVEN ], - [ GIVEN, DYNAMIC, GIVEN ], - [ GIVEN, GIVEN, GIVEN ] - ], - total: GIVEN, - horizontalTotal: GIVEN, - verticalTotal: GIVEN, - type: AreaChallengeType.NUMBERS - } ); - - /*---------------------------------------------------------------------------* - * Numbers 5 - *----------------------------------------------------------------------------*/ - - // L5-1 - AreaChallengeDescription.LEVEL_5_NUMBERS_1 = new AreaChallengeDescription( { - horizontal: [ EDITABLE, GIVEN ], - vertical: [ EDITABLE ], - products: [ - [ GIVEN, GIVEN ] - ], - total: GIVEN, - horizontalTotal: DYNAMIC, - verticalTotal: DYNAMIC, - type: AreaChallengeType.NUMBERS - } ); - - // L5-3 - AreaChallengeDescription.LEVEL_5_NUMBERS_3 = new AreaChallengeDescription( { - horizontal: [ EDITABLE, GIVEN, GIVEN ], - vertical: [ EDITABLE ], - products: [ - [ GIVEN, GIVEN, GIVEN ] - ], - total: GIVEN, - horizontalTotal: DYNAMIC, - verticalTotal: DYNAMIC, - type: AreaChallengeType.NUMBERS - } ); - - /*---------------------------------------------------------------------------* - * Numbers 6 - *----------------------------------------------------------------------------*/ - - // L6-1 - AreaChallengeDescription.LEVEL_6_NUMBERS_1 = new AreaChallengeDescription( { - horizontal: [ EDITABLE, EDITABLE ], - vertical: [ EDITABLE, GIVEN ], - products: [ - [ GIVEN, GIVEN ], - [ GIVEN, GIVEN ] - ], - total: GIVEN, - horizontalTotal: DYNAMIC, - verticalTotal: DYNAMIC, - type: AreaChallengeType.NUMBERS - } ); - - /*---------------------------------------------------------------------------* - * Variables 1 - *----------------------------------------------------------------------------*/ - - // L1-1 - AreaChallengeDescription.LEVEL_1_VARIABLES_1 = new AreaChallengeDescription( { - horizontal: [ GIVEN, GIVEN ], - vertical: [ GIVEN ], - products: [ - [ EDITABLE, GIVEN ] - ], - total: GIVEN, - horizontalTotal: GIVEN, - verticalTotal: GIVEN, - type: AreaChallengeType.VARIABLES - } ); - - // L1-2 - AreaChallengeDescription.LEVEL_1_VARIABLES_2 = new AreaChallengeDescription( { - horizontal: [ GIVEN, GIVEN ], - vertical: [ GIVEN ], - products: [ - [ GIVEN, GIVEN ] - ], - total: EDITABLE, - horizontalTotal: GIVEN, - verticalTotal: GIVEN, - type: AreaChallengeType.VARIABLES - } ); - - // L1-3 - AreaChallengeDescription.LEVEL_1_VARIABLES_3 = new AreaChallengeDescription( { - horizontal: [ GIVEN, GIVEN, GIVEN ], - vertical: [ GIVEN ], - products: [ - [ EDITABLE, GIVEN, GIVEN ] - ], - total: GIVEN, - horizontalTotal: GIVEN, - verticalTotal: GIVEN, - type: AreaChallengeType.VARIABLES - } ); - - // L1-4 - AreaChallengeDescription.LEVEL_1_VARIABLES_4 = new AreaChallengeDescription( { - horizontal: [ GIVEN, GIVEN, GIVEN ], - vertical: [ GIVEN ], - products: [ - [ GIVEN, GIVEN, GIVEN ] - ], - total: EDITABLE, - horizontalTotal: GIVEN, - verticalTotal: GIVEN, - type: AreaChallengeType.VARIABLES - } ); - - /*---------------------------------------------------------------------------* - * Variables 2 - *----------------------------------------------------------------------------*/ - - // L2-1 - AreaChallengeDescription.LEVEL_2_VARIABLES_1 = new AreaChallengeDescription( { - horizontal: [ GIVEN, GIVEN ], - vertical: [ GIVEN, GIVEN ], - products: [ - [ EDITABLE, GIVEN ], - [ GIVEN, EDITABLE ] - ], - total: GIVEN, - horizontalTotal: GIVEN, - verticalTotal: GIVEN, - type: AreaChallengeType.VARIABLES - } ); - - // L2-2 - AreaChallengeDescription.LEVEL_2_VARIABLES_2 = new AreaChallengeDescription( { - horizontal: [ GIVEN, GIVEN ], - vertical: [ GIVEN, GIVEN ], - products: [ - [ EDITABLE, GIVEN ], - [ GIVEN, GIVEN ] - ], - total: EDITABLE, - horizontalTotal: GIVEN, - verticalTotal: GIVEN, - type: AreaChallengeType.VARIABLES - } ); - - /*---------------------------------------------------------------------------* - * Variables 3 - *----------------------------------------------------------------------------*/ - - // L3-1 - AreaChallengeDescription.LEVEL_3_VARIABLES_1 = new AreaChallengeDescription( { - horizontal: [ EDITABLE, GIVEN ], - vertical: [ EDITABLE ], - products: [ - [ GIVEN, GIVEN ] - ], - total: GIVEN, - horizontalTotal: GIVEN, - verticalTotal: DYNAMIC, - type: AreaChallengeType.VARIABLES - } ); - - // L3-2 - AreaChallengeDescription.LEVEL_3_VARIABLES_2 = new AreaChallengeDescription( { - horizontal: [ EDITABLE, EDITABLE ], - vertical: [ GIVEN ], - products: [ - [ DYNAMIC, GIVEN ] - ], - total: GIVEN, - horizontalTotal: GIVEN, - verticalTotal: GIVEN, - type: AreaChallengeType.VARIABLES - } ); - - // L3-3 - AreaChallengeDescription.LEVEL_3_VARIABLES_3 = new AreaChallengeDescription( { - horizontal: [ EDITABLE, GIVEN ], - vertical: [ GIVEN ], - products: [ - [ GIVEN, EDITABLE ] - ], - total: GIVEN, - horizontalTotal: GIVEN, - verticalTotal: GIVEN, - type: AreaChallengeType.VARIABLES - } ); - - // L3-4 - AreaChallengeDescription.LEVEL_3_VARIABLES_4 = new AreaChallengeDescription( { - horizontal: [ EDITABLE, GIVEN, GIVEN ], - vertical: [ EDITABLE ], - products: [ - [ GIVEN, DYNAMIC, GIVEN ] - ], - total: GIVEN, - horizontalTotal: GIVEN, - verticalTotal: DYNAMIC, - type: AreaChallengeType.VARIABLES - } ); - - // L3-5 - AreaChallengeDescription.LEVEL_3_VARIABLES_5 = new AreaChallengeDescription( { - horizontal: [ EDITABLE, EDITABLE, GIVEN ], - vertical: [ GIVEN ], - products: [ - [ DYNAMIC, GIVEN, GIVEN ] - ], - total: GIVEN, - horizontalTotal: GIVEN, - verticalTotal: GIVEN, - type: AreaChallengeType.VARIABLES - } ); - - // L3-6 - AreaChallengeDescription.LEVEL_3_VARIABLES_6 = new AreaChallengeDescription( { - horizontal: [ EDITABLE, GIVEN, GIVEN ], - vertical: [ GIVEN ], - products: [ - [ GIVEN, EDITABLE, GIVEN ] - ], - total: GIVEN, - horizontalTotal: GIVEN, - verticalTotal: GIVEN, - type: AreaChallengeType.VARIABLES - } ); - - /*---------------------------------------------------------------------------* - * Variables 4 - *----------------------------------------------------------------------------*/ - - // L4-1 - AreaChallengeDescription.LEVEL_4_VARIABLES_1 = new AreaChallengeDescription( { - horizontal: [ EDITABLE, GIVEN ], - vertical: [ EDITABLE, GIVEN ], - products: [ - [ GIVEN, DYNAMIC ], - [ DYNAMIC, GIVEN ] - ], - total: GIVEN, - horizontalTotal: GIVEN, - verticalTotal: GIVEN, - type: AreaChallengeType.VARIABLES - } ); - - // L4-2 - AreaChallengeDescription.LEVEL_4_VARIABLES_2 = new AreaChallengeDescription( { - horizontal: [ EDITABLE, GIVEN ], - vertical: [ GIVEN, GIVEN ], - products: [ - [ GIVEN, EDITABLE ], - [ GIVEN, GIVEN ] - ], - total: GIVEN, - horizontalTotal: GIVEN, - verticalTotal: GIVEN, - type: AreaChallengeType.VARIABLES - } ); - - /*---------------------------------------------------------------------------* - * Variables 5 - *----------------------------------------------------------------------------*/ - - // L5-1 - AreaChallengeDescription.LEVEL_5_VARIABLES_1 = new AreaChallengeDescription( { - horizontal: [ EDITABLE, GIVEN ], - vertical: [ EDITABLE ], - products: [ - [ GIVEN, GIVEN ] - ], - total: GIVEN, - horizontalTotal: DYNAMIC, - verticalTotal: DYNAMIC, - type: AreaChallengeType.VARIABLES - } ); - - // L5-2 - AreaChallengeDescription.LEVEL_5_VARIABLES_2 = new AreaChallengeDescription( { - horizontal: [ EDITABLE, GIVEN, GIVEN ], - vertical: [ EDITABLE ], - products: [ - [ GIVEN, GIVEN, GIVEN ] - ], - total: GIVEN, - horizontalTotal: DYNAMIC, - verticalTotal: DYNAMIC, - type: AreaChallengeType.VARIABLES - } ); - - /*---------------------------------------------------------------------------* - * Variables 6 - *----------------------------------------------------------------------------*/ - - // L6-1 - AreaChallengeDescription.LEVEL_6_VARIABLES_1 = new AreaChallengeDescription( { - horizontal: [ GIVEN, EDITABLE ], - vertical: [ GIVEN, EDITABLE ], - products: [ - [ GIVEN, DYNAMIC ], - [ DYNAMIC, GIVEN ] - ], - total: GIVEN, - horizontalTotal: DYNAMIC, - verticalTotal: DYNAMIC, - type: AreaChallengeType.VARIABLES, - shufflable: false, - unique: false - } ); - - return AreaChallengeDescription; + + return new AreaChallengeDescription( options ); + }, + + /** + * Returns a conditional value (like a ternary) based on whether this is a number or variable challenge. + * @public + * + * @param {*} numberTypeValue + * @param {*} variableTypeValue + * @returns {*} + */ + numberOrVariable: function( numberTypeValue, variableTypeValue ) { + return this.type === AreaChallengeType.VARIABLES ? variableTypeValue : numberTypeValue; + } +} ); + +/*---------------------------------------------------------------------------* +* Numbers 1 +*----------------------------------------------------------------------------*/ + +// L1-1 +AreaChallengeDescription.LEVEL_1_NUMBERS_1 = new AreaChallengeDescription( { + horizontal: [ GIVEN, GIVEN ], + vertical: [ GIVEN ], + products: [ + [ GIVEN, GIVEN ] + ], + total: EDITABLE, + horizontalTotal: GIVEN, + verticalTotal: GIVEN, + type: AreaChallengeType.NUMBERS +} ); + +// L1-2 +AreaChallengeDescription.LEVEL_1_NUMBERS_2 = new AreaChallengeDescription( { + horizontal: [ GIVEN, GIVEN ], + vertical: [ GIVEN ], + products: [ + [ EDITABLE, GIVEN ] + ], + total: GIVEN, + horizontalTotal: GIVEN, + verticalTotal: GIVEN, + type: AreaChallengeType.NUMBERS +} ); + +// L1-3 +AreaChallengeDescription.LEVEL_1_NUMBERS_3 = new AreaChallengeDescription( { + horizontal: [ GIVEN, GIVEN, GIVEN ], + vertical: [ GIVEN ], + products: [ + [ GIVEN, GIVEN, GIVEN ] + ], + total: EDITABLE, + horizontalTotal: GIVEN, + verticalTotal: GIVEN, + type: AreaChallengeType.NUMBERS +} ); + +// L1-4 +AreaChallengeDescription.LEVEL_1_NUMBERS_4 = new AreaChallengeDescription( { + horizontal: [ GIVEN, GIVEN, GIVEN ], + vertical: [ GIVEN ], + products: [ + [ EDITABLE, GIVEN, GIVEN ] + ], + total: GIVEN, + horizontalTotal: GIVEN, + verticalTotal: GIVEN, + type: AreaChallengeType.NUMBERS +} ); + +// L1-5 +AreaChallengeDescription.LEVEL_1_NUMBERS_5 = new AreaChallengeDescription( { + horizontal: [ GIVEN, GIVEN ], + vertical: [ GIVEN, GIVEN ], + products: [ + [ GIVEN, GIVEN ], + [ GIVEN, GIVEN ] + ], + total: EDITABLE, + horizontalTotal: GIVEN, + verticalTotal: GIVEN, + type: AreaChallengeType.NUMBERS +} ); + +// L1-6 +AreaChallengeDescription.LEVEL_1_NUMBERS_6 = new AreaChallengeDescription( { + horizontal: [ GIVEN, GIVEN ], + vertical: [ GIVEN, GIVEN ], + products: [ + [ EDITABLE, GIVEN ], + [ GIVEN, GIVEN ] + ], + total: GIVEN, + horizontalTotal: GIVEN, + verticalTotal: GIVEN, + type: AreaChallengeType.NUMBERS +} ); + +/*---------------------------------------------------------------------------* +* Numbers 2 +*----------------------------------------------------------------------------*/ + +// L2-1 +AreaChallengeDescription.LEVEL_2_NUMBERS_1 = new AreaChallengeDescription( { + horizontal: [ GIVEN, GIVEN ], + vertical: [ GIVEN, GIVEN ], + products: [ + [ EDITABLE, GIVEN ], + [ GIVEN, EDITABLE ] + ], + total: GIVEN, + horizontalTotal: GIVEN, + verticalTotal: GIVEN, + type: AreaChallengeType.NUMBERS +} ); + +// L2-2 +AreaChallengeDescription.LEVEL_2_NUMBERS_2 = new AreaChallengeDescription( { + horizontal: [ GIVEN, GIVEN ], + vertical: [ GIVEN, GIVEN ], + products: [ + [ EDITABLE, GIVEN ], + [ GIVEN, GIVEN ] + ], + total: EDITABLE, + horizontalTotal: GIVEN, + verticalTotal: GIVEN, + type: AreaChallengeType.NUMBERS +} ); + +// L2-3 +AreaChallengeDescription.LEVEL_2_NUMBERS_3 = new AreaChallengeDescription( { + horizontal: [ GIVEN, GIVEN, GIVEN ], + vertical: [ GIVEN, GIVEN ], + products: [ + [ EDITABLE, GIVEN, GIVEN ], + [ GIVEN, EDITABLE, GIVEN ] + ], + total: GIVEN, + horizontalTotal: GIVEN, + verticalTotal: GIVEN, + type: AreaChallengeType.NUMBERS +} ); + +// L2-4 +AreaChallengeDescription.LEVEL_2_NUMBERS_4 = new AreaChallengeDescription( { + horizontal: [ GIVEN, GIVEN, GIVEN ], + vertical: [ GIVEN, GIVEN ], + products: [ + [ EDITABLE, GIVEN, GIVEN ], + [ GIVEN, GIVEN, GIVEN ] + ], + total: EDITABLE, + horizontalTotal: GIVEN, + verticalTotal: GIVEN, + type: AreaChallengeType.NUMBERS +} ); + +// L2-5 +AreaChallengeDescription.LEVEL_2_NUMBERS_5 = new AreaChallengeDescription( { + horizontal: [ GIVEN, GIVEN, GIVEN ], + vertical: [ GIVEN, GIVEN, GIVEN ], + products: [ + [ EDITABLE, GIVEN, GIVEN ], + [ GIVEN, EDITABLE, GIVEN ], + [ GIVEN, GIVEN, GIVEN ] + ], + total: GIVEN, + horizontalTotal: GIVEN, + verticalTotal: GIVEN, + type: AreaChallengeType.NUMBERS +} ); + +/*---------------------------------------------------------------------------* +* Numbers 3 +*----------------------------------------------------------------------------*/ + +// L3-1 +AreaChallengeDescription.LEVEL_3_NUMBERS_1 = new AreaChallengeDescription( { + horizontal: [ EDITABLE, EDITABLE ], + vertical: [ GIVEN ], + products: [ + [ DYNAMIC, GIVEN ] + ], + total: GIVEN, + horizontalTotal: GIVEN, + verticalTotal: GIVEN, + type: AreaChallengeType.NUMBERS +} ); + +// L3-2 +AreaChallengeDescription.LEVEL_3_NUMBERS_2 = new AreaChallengeDescription( { + horizontal: [ EDITABLE, GIVEN ], + vertical: [ EDITABLE ], + products: [ + [ DYNAMIC, GIVEN ] + ], + total: GIVEN, + horizontalTotal: GIVEN, + verticalTotal: DYNAMIC, + type: AreaChallengeType.NUMBERS +} ); + +// L3-3 +AreaChallengeDescription.LEVEL_3_NUMBERS_3 = new AreaChallengeDescription( { + horizontal: [ EDITABLE, EDITABLE, GIVEN ], + vertical: [ GIVEN ], + products: [ + [ DYNAMIC, GIVEN, GIVEN ] + ], + total: GIVEN, + horizontalTotal: GIVEN, + verticalTotal: GIVEN, + type: AreaChallengeType.NUMBERS +} ); + +// L3-4 +AreaChallengeDescription.LEVEL_3_NUMBERS_4 = new AreaChallengeDescription( { + horizontal: [ EDITABLE, GIVEN, GIVEN ], + vertical: [ EDITABLE ], + products: [ + [ GIVEN, GIVEN, GIVEN ] + ], + total: GIVEN, + horizontalTotal: GIVEN, + verticalTotal: DYNAMIC, + type: AreaChallengeType.NUMBERS +} ); + +// L3-5 +AreaChallengeDescription.LEVEL_3_NUMBERS_5 = new AreaChallengeDescription( { + horizontal: [ EDITABLE, GIVEN ], + vertical: [ EDITABLE, GIVEN ], + products: [ + [ GIVEN, DYNAMIC ], + [ DYNAMIC, GIVEN ] + ], + total: GIVEN, + horizontalTotal: GIVEN, + verticalTotal: GIVEN, + type: AreaChallengeType.NUMBERS +} ); + +// L3-6 +AreaChallengeDescription.LEVEL_3_NUMBERS_6 = new AreaChallengeDescription( { + horizontal: [ EDITABLE, GIVEN ], + vertical: [ EDITABLE, GIVEN ], + products: [ + [ DYNAMIC, GIVEN ], + [ GIVEN, GIVEN ] + ], + total: GIVEN, + horizontalTotal: GIVEN, + verticalTotal: GIVEN, + type: AreaChallengeType.NUMBERS +} ); + +/*---------------------------------------------------------------------------* +* Numbers 4 +*----------------------------------------------------------------------------*/ + +// L4-1 +AreaChallengeDescription.LEVEL_4_NUMBERS_1 = new AreaChallengeDescription( { + horizontal: [ EDITABLE, EDITABLE, EDITABLE ], + vertical: [ GIVEN, GIVEN ], + products: [ + [ DYNAMIC, GIVEN, GIVEN ], + [ GIVEN, DYNAMIC, GIVEN ] + ], + total: GIVEN, + horizontalTotal: GIVEN, + verticalTotal: GIVEN, + type: AreaChallengeType.NUMBERS +} ); + +// L4-2 +AreaChallengeDescription.LEVEL_4_NUMBERS_2 = new AreaChallengeDescription( { + horizontal: [ EDITABLE, EDITABLE, GIVEN ], + vertical: [ EDITABLE, GIVEN ], + products: [ + [ DYNAMIC, GIVEN, GIVEN ], + [ GIVEN, GIVEN, GIVEN ] + ], + total: GIVEN, + horizontalTotal: GIVEN, + verticalTotal: GIVEN, + type: AreaChallengeType.NUMBERS +} ); + +// L4-3 +AreaChallengeDescription.LEVEL_4_NUMBERS_3 = new AreaChallengeDescription( { + horizontal: [ EDITABLE, EDITABLE, GIVEN ], + vertical: [ EDITABLE, GIVEN ], + products: [ + [ DYNAMIC, GIVEN, GIVEN ], + [ GIVEN, DYNAMIC, GIVEN ] + ], + total: GIVEN, + horizontalTotal: GIVEN, + verticalTotal: GIVEN, + type: AreaChallengeType.NUMBERS +} ); + +// L4-4 +AreaChallengeDescription.LEVEL_4_NUMBERS_4 = new AreaChallengeDescription( { + horizontal: [ EDITABLE, EDITABLE, EDITABLE ], + vertical: [ GIVEN, GIVEN, GIVEN ], + products: [ + [ DYNAMIC, GIVEN, GIVEN ], + [ GIVEN, DYNAMIC, GIVEN ], + [ GIVEN, GIVEN, DYNAMIC ] + ], + total: GIVEN, + horizontalTotal: GIVEN, + verticalTotal: GIVEN, + type: AreaChallengeType.NUMBERS +} ); + +// L4-5 +AreaChallengeDescription.LEVEL_4_NUMBERS_5 = new AreaChallengeDescription( { + horizontal: [ EDITABLE, EDITABLE, GIVEN ], + vertical: [ EDITABLE, GIVEN, GIVEN ], + products: [ + [ DYNAMIC, GIVEN, GIVEN ], + [ GIVEN, DYNAMIC, GIVEN ], + [ GIVEN, GIVEN, GIVEN ] + ], + total: GIVEN, + horizontalTotal: GIVEN, + verticalTotal: GIVEN, + type: AreaChallengeType.NUMBERS +} ); + +/*---------------------------------------------------------------------------* +* Numbers 5 +*----------------------------------------------------------------------------*/ + +// L5-1 +AreaChallengeDescription.LEVEL_5_NUMBERS_1 = new AreaChallengeDescription( { + horizontal: [ EDITABLE, GIVEN ], + vertical: [ EDITABLE ], + products: [ + [ GIVEN, GIVEN ] + ], + total: GIVEN, + horizontalTotal: DYNAMIC, + verticalTotal: DYNAMIC, + type: AreaChallengeType.NUMBERS +} ); + +// L5-3 +AreaChallengeDescription.LEVEL_5_NUMBERS_3 = new AreaChallengeDescription( { + horizontal: [ EDITABLE, GIVEN, GIVEN ], + vertical: [ EDITABLE ], + products: [ + [ GIVEN, GIVEN, GIVEN ] + ], + total: GIVEN, + horizontalTotal: DYNAMIC, + verticalTotal: DYNAMIC, + type: AreaChallengeType.NUMBERS +} ); + +/*---------------------------------------------------------------------------* +* Numbers 6 +*----------------------------------------------------------------------------*/ + +// L6-1 +AreaChallengeDescription.LEVEL_6_NUMBERS_1 = new AreaChallengeDescription( { + horizontal: [ EDITABLE, EDITABLE ], + vertical: [ EDITABLE, GIVEN ], + products: [ + [ GIVEN, GIVEN ], + [ GIVEN, GIVEN ] + ], + total: GIVEN, + horizontalTotal: DYNAMIC, + verticalTotal: DYNAMIC, + type: AreaChallengeType.NUMBERS +} ); + +/*---------------------------------------------------------------------------* +* Variables 1 +*----------------------------------------------------------------------------*/ + +// L1-1 +AreaChallengeDescription.LEVEL_1_VARIABLES_1 = new AreaChallengeDescription( { + horizontal: [ GIVEN, GIVEN ], + vertical: [ GIVEN ], + products: [ + [ EDITABLE, GIVEN ] + ], + total: GIVEN, + horizontalTotal: GIVEN, + verticalTotal: GIVEN, + type: AreaChallengeType.VARIABLES +} ); + +// L1-2 +AreaChallengeDescription.LEVEL_1_VARIABLES_2 = new AreaChallengeDescription( { + horizontal: [ GIVEN, GIVEN ], + vertical: [ GIVEN ], + products: [ + [ GIVEN, GIVEN ] + ], + total: EDITABLE, + horizontalTotal: GIVEN, + verticalTotal: GIVEN, + type: AreaChallengeType.VARIABLES +} ); + +// L1-3 +AreaChallengeDescription.LEVEL_1_VARIABLES_3 = new AreaChallengeDescription( { + horizontal: [ GIVEN, GIVEN, GIVEN ], + vertical: [ GIVEN ], + products: [ + [ EDITABLE, GIVEN, GIVEN ] + ], + total: GIVEN, + horizontalTotal: GIVEN, + verticalTotal: GIVEN, + type: AreaChallengeType.VARIABLES +} ); + +// L1-4 +AreaChallengeDescription.LEVEL_1_VARIABLES_4 = new AreaChallengeDescription( { + horizontal: [ GIVEN, GIVEN, GIVEN ], + vertical: [ GIVEN ], + products: [ + [ GIVEN, GIVEN, GIVEN ] + ], + total: EDITABLE, + horizontalTotal: GIVEN, + verticalTotal: GIVEN, + type: AreaChallengeType.VARIABLES +} ); + +/*---------------------------------------------------------------------------* +* Variables 2 +*----------------------------------------------------------------------------*/ + +// L2-1 +AreaChallengeDescription.LEVEL_2_VARIABLES_1 = new AreaChallengeDescription( { + horizontal: [ GIVEN, GIVEN ], + vertical: [ GIVEN, GIVEN ], + products: [ + [ EDITABLE, GIVEN ], + [ GIVEN, EDITABLE ] + ], + total: GIVEN, + horizontalTotal: GIVEN, + verticalTotal: GIVEN, + type: AreaChallengeType.VARIABLES +} ); + +// L2-2 +AreaChallengeDescription.LEVEL_2_VARIABLES_2 = new AreaChallengeDescription( { + horizontal: [ GIVEN, GIVEN ], + vertical: [ GIVEN, GIVEN ], + products: [ + [ EDITABLE, GIVEN ], + [ GIVEN, GIVEN ] + ], + total: EDITABLE, + horizontalTotal: GIVEN, + verticalTotal: GIVEN, + type: AreaChallengeType.VARIABLES +} ); + +/*---------------------------------------------------------------------------* +* Variables 3 +*----------------------------------------------------------------------------*/ + +// L3-1 +AreaChallengeDescription.LEVEL_3_VARIABLES_1 = new AreaChallengeDescription( { + horizontal: [ EDITABLE, GIVEN ], + vertical: [ EDITABLE ], + products: [ + [ GIVEN, GIVEN ] + ], + total: GIVEN, + horizontalTotal: GIVEN, + verticalTotal: DYNAMIC, + type: AreaChallengeType.VARIABLES +} ); + +// L3-2 +AreaChallengeDescription.LEVEL_3_VARIABLES_2 = new AreaChallengeDescription( { + horizontal: [ EDITABLE, EDITABLE ], + vertical: [ GIVEN ], + products: [ + [ DYNAMIC, GIVEN ] + ], + total: GIVEN, + horizontalTotal: GIVEN, + verticalTotal: GIVEN, + type: AreaChallengeType.VARIABLES +} ); + +// L3-3 +AreaChallengeDescription.LEVEL_3_VARIABLES_3 = new AreaChallengeDescription( { + horizontal: [ EDITABLE, GIVEN ], + vertical: [ GIVEN ], + products: [ + [ GIVEN, EDITABLE ] + ], + total: GIVEN, + horizontalTotal: GIVEN, + verticalTotal: GIVEN, + type: AreaChallengeType.VARIABLES +} ); + +// L3-4 +AreaChallengeDescription.LEVEL_3_VARIABLES_4 = new AreaChallengeDescription( { + horizontal: [ EDITABLE, GIVEN, GIVEN ], + vertical: [ EDITABLE ], + products: [ + [ GIVEN, DYNAMIC, GIVEN ] + ], + total: GIVEN, + horizontalTotal: GIVEN, + verticalTotal: DYNAMIC, + type: AreaChallengeType.VARIABLES } ); + +// L3-5 +AreaChallengeDescription.LEVEL_3_VARIABLES_5 = new AreaChallengeDescription( { + horizontal: [ EDITABLE, EDITABLE, GIVEN ], + vertical: [ GIVEN ], + products: [ + [ DYNAMIC, GIVEN, GIVEN ] + ], + total: GIVEN, + horizontalTotal: GIVEN, + verticalTotal: GIVEN, + type: AreaChallengeType.VARIABLES +} ); + +// L3-6 +AreaChallengeDescription.LEVEL_3_VARIABLES_6 = new AreaChallengeDescription( { + horizontal: [ EDITABLE, GIVEN, GIVEN ], + vertical: [ GIVEN ], + products: [ + [ GIVEN, EDITABLE, GIVEN ] + ], + total: GIVEN, + horizontalTotal: GIVEN, + verticalTotal: GIVEN, + type: AreaChallengeType.VARIABLES +} ); + +/*---------------------------------------------------------------------------* +* Variables 4 +*----------------------------------------------------------------------------*/ + +// L4-1 +AreaChallengeDescription.LEVEL_4_VARIABLES_1 = new AreaChallengeDescription( { + horizontal: [ EDITABLE, GIVEN ], + vertical: [ EDITABLE, GIVEN ], + products: [ + [ GIVEN, DYNAMIC ], + [ DYNAMIC, GIVEN ] + ], + total: GIVEN, + horizontalTotal: GIVEN, + verticalTotal: GIVEN, + type: AreaChallengeType.VARIABLES +} ); + +// L4-2 +AreaChallengeDescription.LEVEL_4_VARIABLES_2 = new AreaChallengeDescription( { + horizontal: [ EDITABLE, GIVEN ], + vertical: [ GIVEN, GIVEN ], + products: [ + [ GIVEN, EDITABLE ], + [ GIVEN, GIVEN ] + ], + total: GIVEN, + horizontalTotal: GIVEN, + verticalTotal: GIVEN, + type: AreaChallengeType.VARIABLES +} ); + +/*---------------------------------------------------------------------------* +* Variables 5 +*----------------------------------------------------------------------------*/ + +// L5-1 +AreaChallengeDescription.LEVEL_5_VARIABLES_1 = new AreaChallengeDescription( { + horizontal: [ EDITABLE, GIVEN ], + vertical: [ EDITABLE ], + products: [ + [ GIVEN, GIVEN ] + ], + total: GIVEN, + horizontalTotal: DYNAMIC, + verticalTotal: DYNAMIC, + type: AreaChallengeType.VARIABLES +} ); + +// L5-2 +AreaChallengeDescription.LEVEL_5_VARIABLES_2 = new AreaChallengeDescription( { + horizontal: [ EDITABLE, GIVEN, GIVEN ], + vertical: [ EDITABLE ], + products: [ + [ GIVEN, GIVEN, GIVEN ] + ], + total: GIVEN, + horizontalTotal: DYNAMIC, + verticalTotal: DYNAMIC, + type: AreaChallengeType.VARIABLES +} ); + +/*---------------------------------------------------------------------------* +* Variables 6 +*----------------------------------------------------------------------------*/ + +// L6-1 +AreaChallengeDescription.LEVEL_6_VARIABLES_1 = new AreaChallengeDescription( { + horizontal: [ GIVEN, EDITABLE ], + vertical: [ GIVEN, EDITABLE ], + products: [ + [ GIVEN, DYNAMIC ], + [ DYNAMIC, GIVEN ] + ], + total: GIVEN, + horizontalTotal: DYNAMIC, + verticalTotal: DYNAMIC, + type: AreaChallengeType.VARIABLES, + shufflable: false, + unique: false +} ); + +export default AreaChallengeDescription; \ No newline at end of file diff --git a/js/game/model/AreaChallengeType.js b/js/game/model/AreaChallengeType.js index 5fa7c5ae..85d55778 100644 --- a/js/game/model/AreaChallengeType.js +++ b/js/game/model/AreaChallengeType.js @@ -5,27 +5,23 @@ * * @author Jonathan Olson */ -define( require => { - 'use strict'; - // modules - const areaModelCommon = require( 'AREA_MODEL_COMMON/areaModelCommon' ); +import areaModelCommon from '../../areaModelCommon.js'; - const AreaChallengeType = { - NUMBERS: 'NUMBERS', - VARIABLES: 'VARIABLES' - }; +const AreaChallengeType = { + NUMBERS: 'NUMBERS', + VARIABLES: 'VARIABLES' +}; - areaModelCommon.register( 'AreaChallengeType', AreaChallengeType ); +areaModelCommon.register( 'AreaChallengeType', AreaChallengeType ); - // @public {Array.} - All values the enumeration can take. - AreaChallengeType.VALUES = [ - AreaChallengeType.NUMBERS, - AreaChallengeType.VARIABLES - ]; +// @public {Array.} - All values the enumeration can take. +AreaChallengeType.VALUES = [ + AreaChallengeType.NUMBERS, + AreaChallengeType.VARIABLES +]; - // verify that enumeration is immutable, without the runtime penalty in production code - if ( assert ) { Object.freeze( AreaChallengeType ); } +// verify that enumeration is immutable, without the runtime penalty in production code +if ( assert ) { Object.freeze( AreaChallengeType ); } - return AreaChallengeType; -} ); +export default AreaChallengeType; \ No newline at end of file diff --git a/js/game/model/AreaLevel.js b/js/game/model/AreaLevel.js index e950a9e1..e700846b 100644 --- a/js/game/model/AreaLevel.js +++ b/js/game/model/AreaLevel.js @@ -5,149 +5,145 @@ * * @author Jonathan Olson */ -define( require => { - 'use strict'; - - // modules - const AreaChallenge = require( 'AREA_MODEL_COMMON/game/model/AreaChallenge' ); - const areaModelCommon = require( 'AREA_MODEL_COMMON/areaModelCommon' ); - const AreaModelCommonConstants = require( 'AREA_MODEL_COMMON/common/AreaModelCommonConstants' ); - const DerivedProperty = require( 'AXON/DerivedProperty' ); - const GameState = require( 'AREA_MODEL_COMMON/game/model/GameState' ); - const inherit = require( 'PHET_CORE/inherit' ); - const NumberProperty = require( 'AXON/NumberProperty' ); - /** - * @constructor - * @extends {Object} - * - * @param {number} number - * @param {AreaChallengeType} type - * @param {Property.} colorProperty - * @param {Array.} challengeDescriptions - */ - function AreaLevel( number, type, colorProperty, challengeDescriptions ) { - const self = this; +import DerivedProperty from '../../../../axon/js/DerivedProperty.js'; +import NumberProperty from '../../../../axon/js/NumberProperty.js'; +import inherit from '../../../../phet-core/js/inherit.js'; +import areaModelCommon from '../../areaModelCommon.js'; +import AreaModelCommonConstants from '../../common/AreaModelCommonConstants.js'; +import AreaChallenge from './AreaChallenge.js'; +import GameState from './GameState.js'; - // @public {number} - Will be the value 1 for "Level 1". Not using the 0-based values used in VEGAS, so sometimes - // this value will need to be decremented when passed to VEGAS components. - this.number = number; +/** + * @constructor + * @extends {Object} + * + * @param {number} number + * @param {AreaChallengeType} type + * @param {Property.} colorProperty + * @param {Array.} challengeDescriptions + */ +function AreaLevel( number, type, colorProperty, challengeDescriptions ) { + const self = this; - // @public {AreaChallengeType} - this.type = type; + // @public {number} - Will be the value 1 for "Level 1". Not using the 0-based values used in VEGAS, so sometimes + // this value will need to be decremented when passed to VEGAS components. + this.number = number; - // @public {Property.} - this.colorProperty = colorProperty; + // @public {AreaChallengeType} + this.type = type; - // @public {Array.} - Descriptions for each type of level - this.challengeDescriptions = challengeDescriptions; + // @public {Property.} + this.colorProperty = colorProperty; - // @public {Property.} - Ranges from 0 to AreaModelCommonConstants.PERFECT_SCORE - // (since 2 points are rewarded for first attempt correct) - this.scoreProperty = new NumberProperty( 0 ); + // @public {Array.} - Descriptions for each type of level + this.challengeDescriptions = challengeDescriptions; - // @public {Array.} - this.challenges = this.generateChallenges(); + // @public {Property.} - Ranges from 0 to AreaModelCommonConstants.PERFECT_SCORE + // (since 2 points are rewarded for first attempt correct) + this.scoreProperty = new NumberProperty( 0 ); + + // @public {Array.} + this.challenges = this.generateChallenges(); + + // @public {Property.} - The index of the current challenge. + this.challengeIndexProperty = new NumberProperty( 0 ); + + // @public {Property.} + this.currentChallengeProperty = new DerivedProperty( [ this.challengeIndexProperty ], function( index ) { + return self.challenges[ index ]; + } ); - // @public {Property.} - The index of the current challenge. - this.challengeIndexProperty = new NumberProperty( 0 ); + // @public {boolean} - Whether the level is finished + this.finished = false; +} - // @public {Property.} - this.currentChallengeProperty = new DerivedProperty( [ this.challengeIndexProperty ], function( index ) { - return self.challenges[ index ]; +areaModelCommon.register( 'AreaLevel', AreaLevel ); + +export default inherit( Object, AreaLevel, { + /** + * Generates six challenges. + * @private + * + * @returns {Array.} + */ + generateChallenges: function() { + + // Always include the first description as the first challenge + let descriptions = [ this.challengeDescriptions[ 0 ] ]; + + // Shuffle the rest of them in a random order + descriptions = descriptions.concat( phet.joist.random.shuffle( descriptions.slice( 1 ) ) ); + + // Then fill with random challenges if there are any more spaces + while ( descriptions.length < AreaModelCommonConstants.NUM_CHALLENGES ) { + descriptions.push( phet.joist.random.sample( this.challengeDescriptions ) ); + } + + // Generate based on the descriptions + return descriptions.map( function( description ) { + return new AreaChallenge( description ); } ); + }, - // @public {boolean} - Whether the level is finished - this.finished = false; - } + /** + * Selects the level (resetting progress and generates a new challenge). It is not the same as starting the level, + * It's more of a "switch to" operation, since there will already be challenges. We want to delay the resetting of + * the level until it's selected again (unless it was already finished). + * @public + */ + select: function() { + if ( this.finished ) { + this.finished = false; + this.reset(); + } + }, + + /** + * Marks the level as finished. This means challenges will be regenerated if the level is selected again. + * @public + */ + finish: function() { + this.finished = true; + }, - areaModelCommon.register( 'AreaLevel', AreaLevel ); - - return inherit( Object, AreaLevel, { - /** - * Generates six challenges. - * @private - * - * @returns {Array.} - */ - generateChallenges: function() { - - // Always include the first description as the first challenge - let descriptions = [ this.challengeDescriptions[ 0 ] ]; - - // Shuffle the rest of them in a random order - descriptions = descriptions.concat( phet.joist.random.shuffle( descriptions.slice( 1 ) ) ); - - // Then fill with random challenges if there are any more spaces - while ( descriptions.length < AreaModelCommonConstants.NUM_CHALLENGES ) { - descriptions.push( phet.joist.random.sample( this.challengeDescriptions ) ); - } - - // Generate based on the descriptions - return descriptions.map( function( description ) { - return new AreaChallenge( description ); - } ); - }, - - /** - * Selects the level (resetting progress and generates a new challenge). It is not the same as starting the level, - * It's more of a "switch to" operation, since there will already be challenges. We want to delay the resetting of - * the level until it's selected again (unless it was already finished). - * @public - */ - select: function() { - if ( this.finished ) { - this.finished = false; - this.reset(); - } - }, - - /** - * Marks the level as finished. This means challenges will be regenerated if the level is selected again. - * @public - */ - finish: function() { - this.finished = true; - }, - - /** - * Move to the next challenge. - * @public - */ - next: function() { - if ( this.challengeIndexProperty.value === AreaModelCommonConstants.NUM_CHALLENGES - 1 ) { - this.finish(); - this.currentChallengeProperty.value.stateProperty.value = GameState.LEVEL_COMPLETE; - } - else { - this.challengeIndexProperty.value += 1; - } - }, - - /** - * When we start over, we want to reset the score, but not immediately change the challenges yet (we'll wait until - * we re-select this level). - * @public - * - * See https://github.com/phetsims/area-model-common/issues/87 and - * https://github.com/phetsims/area-model-common/issues/96. - */ - startOver: function() { - this.scoreProperty.reset(); + /** + * Move to the next challenge. + * @public + */ + next: function() { + if ( this.challengeIndexProperty.value === AreaModelCommonConstants.NUM_CHALLENGES - 1 ) { this.finish(); - }, + this.currentChallengeProperty.value.stateProperty.value = GameState.LEVEL_COMPLETE; + } + else { + this.challengeIndexProperty.value += 1; + } + }, - /** - * Returns the model to its initial state. - * @public - */ - reset: function() { - this.challenges = this.generateChallenges(); + /** + * When we start over, we want to reset the score, but not immediately change the challenges yet (we'll wait until + * we re-select this level). + * @public + * + * See https://github.com/phetsims/area-model-common/issues/87 and + * https://github.com/phetsims/area-model-common/issues/96. + */ + startOver: function() { + this.scoreProperty.reset(); + this.finish(); + }, + + /** + * Returns the model to its initial state. + * @public + */ + reset: function() { + this.challenges = this.generateChallenges(); - this.scoreProperty.reset(); - this.challengeIndexProperty.reset(); + this.scoreProperty.reset(); + this.challengeIndexProperty.reset(); - this.challengeIndexProperty.notifyListenersStatic(); - } - } ); -} ); + this.challengeIndexProperty.notifyListenersStatic(); + } +} ); \ No newline at end of file diff --git a/js/game/model/Entry.js b/js/game/model/Entry.js index 084aa661..8b8969d9 100644 --- a/js/game/model/Entry.js +++ b/js/game/model/Entry.js @@ -5,71 +5,68 @@ * * @author Jonathan Olson */ -define( require => { - 'use strict'; - // modules - const areaModelCommon = require( 'AREA_MODEL_COMMON/areaModelCommon' ); - const DerivedProperty = require( 'AXON/DerivedProperty' ); - const EntryDisplayType = require( 'AREA_MODEL_COMMON/game/model/EntryDisplayType' ); - const EntryStatus = require( 'AREA_MODEL_COMMON/game/model/EntryStatus' ); - const EntryType = require( 'AREA_MODEL_COMMON/game/model/EntryType' ); - const inherit = require( 'PHET_CORE/inherit' ); - const InputMethod = require( 'AREA_MODEL_COMMON/game/model/InputMethod' ); - const merge = require( 'PHET_CORE/merge' ); - const Property = require( 'AXON/Property' ); - const Term = require( 'AREA_MODEL_COMMON/common/model/Term' ); +import DerivedProperty from '../../../../axon/js/DerivedProperty.js'; +import Property from '../../../../axon/js/Property.js'; +import inherit from '../../../../phet-core/js/inherit.js'; +import merge from '../../../../phet-core/js/merge.js'; +import areaModelCommon from '../../areaModelCommon.js'; +import Term from '../../common/model/Term.js'; +import EntryDisplayType from './EntryDisplayType.js'; +import EntryStatus from './EntryStatus.js'; +import EntryType from './EntryType.js'; +import InputMethod from './InputMethod.js'; - /** - * @constructor - * @extends {Object} - * - * @param {Term|null} value - The initial value - * @param {Object} [options] - */ - function Entry( value, options ) { - options = merge( { - type: EntryType.GIVEN, - displayType: EntryDisplayType.HIDDEN, - inputMethod: InputMethod.CONSTANT, - numberOfDigits: 0, - correctValue: null // Only used for the total coefficients - }, options ); +/** + * @constructor + * @extends {Object} + * + * @param {Term|null} value - The initial value + * @param {Object} [options] + */ +function Entry( value, options ) { + options = merge( { + type: EntryType.GIVEN, + displayType: EntryDisplayType.HIDDEN, + inputMethod: InputMethod.CONSTANT, + numberOfDigits: 0, + correctValue: null // Only used for the total coefficients + }, options ); - // Always start off by editing null, and it should be the default value. - if ( options.displayType === EntryDisplayType.EDITABLE ) { - value = null; - } + // Always start off by editing null, and it should be the default value. + if ( options.displayType === EntryDisplayType.EDITABLE ) { + value = null; + } - // @public {Property.} - The current value of the entry - this.valueProperty = new Property( value, { - useDeepEquality: true, - isValidValue: Term.isTermOrNull - } ); + // @public {Property.} - The current value of the entry + this.valueProperty = new Property( value, { + useDeepEquality: true, + isValidValue: Term.isTermOrNull + } ); - // @public {EntryType} - Whether we are dynamic/editable/given. - this.type = options.type; + // @public {EntryType} - Whether we are dynamic/editable/given. + this.type = options.type; - // @public {EntryDisplayType} - Whether we are a readout or editable/hidden - this.displayType = options.displayType; + // @public {EntryDisplayType} - Whether we are a readout or editable/hidden + this.displayType = options.displayType; - // @public {InputMethod} - What format should be used if we are edited? (Need different keypads or a polynomial - // input) - this.inputMethod = options.inputMethod; + // @public {InputMethod} - What format should be used if we are edited? (Need different keypads or a polynomial + // input) + this.inputMethod = options.inputMethod; - // @public {number} - this.digits = options.numberOfDigits; + // @public {number} + this.digits = options.numberOfDigits; - // @public {Property.} - this.statusProperty = new Property( EntryStatus.DIRTY ); + // @public {Property.} + this.statusProperty = new Property( EntryStatus.DIRTY ); - // @public {Property.} - Our value, except for null if there is an error highlight - this.nonErrorValueProperty = new DerivedProperty( [ this.valueProperty, this.statusProperty ], function( value, highlight ) { - return ( highlight === EntryStatus.INCORRECT ) ? null : value; - } ); - } + // @public {Property.} - Our value, except for null if there is an error highlight + this.nonErrorValueProperty = new DerivedProperty( [ this.valueProperty, this.statusProperty ], function( value, highlight ) { + return ( highlight === EntryStatus.INCORRECT ) ? null : value; + } ); +} - areaModelCommon.register( 'Entry', Entry ); +areaModelCommon.register( 'Entry', Entry ); - return inherit( Object, Entry ); -} ); +inherit( Object, Entry ); +export default Entry; \ No newline at end of file diff --git a/js/game/model/EntryDisplayType.js b/js/game/model/EntryDisplayType.js index bf5ddd3b..a8d22132 100644 --- a/js/game/model/EntryDisplayType.js +++ b/js/game/model/EntryDisplayType.js @@ -5,29 +5,25 @@ * * @author Jonathan Olson */ -define( require => { - 'use strict'; - // modules - const areaModelCommon = require( 'AREA_MODEL_COMMON/areaModelCommon' ); +import areaModelCommon from '../../areaModelCommon.js'; - const EntryDisplayType = { - EDITABLE: 'EDITABLE', - READOUT: 'READOUT', - HIDDEN: 'HIDDEN' - }; +const EntryDisplayType = { + EDITABLE: 'EDITABLE', + READOUT: 'READOUT', + HIDDEN: 'HIDDEN' +}; - areaModelCommon.register( 'EntryDisplayType', EntryDisplayType ); +areaModelCommon.register( 'EntryDisplayType', EntryDisplayType ); - // @public {Array.} - All values the enumeration can take. - EntryDisplayType.VALUES = [ - EntryDisplayType.EDITABLE, // editable, and shows the edited value - EntryDisplayType.READOUT, // just the value shown, does not look editable - EntryDisplayType.HIDDEN // nothing shown - ]; +// @public {Array.} - All values the enumeration can take. +EntryDisplayType.VALUES = [ + EntryDisplayType.EDITABLE, // editable, and shows the edited value + EntryDisplayType.READOUT, // just the value shown, does not look editable + EntryDisplayType.HIDDEN // nothing shown +]; - // verify that enumeration is immutable, without the runtime penalty in production code - if ( assert ) { Object.freeze( EntryDisplayType ); } +// verify that enumeration is immutable, without the runtime penalty in production code +if ( assert ) { Object.freeze( EntryDisplayType ); } - return EntryDisplayType; -} ); +export default EntryDisplayType; \ No newline at end of file diff --git a/js/game/model/EntryStatus.js b/js/game/model/EntryStatus.js index 7c3a5850..aa6ac384 100644 --- a/js/game/model/EntryStatus.js +++ b/js/game/model/EntryStatus.js @@ -5,29 +5,25 @@ * * @author Jonathan Olson */ -define( require => { - 'use strict'; - // modules - const areaModelCommon = require( 'AREA_MODEL_COMMON/areaModelCommon' ); +import areaModelCommon from '../../areaModelCommon.js'; - const EntryStatus = { - NORMAL: 'NORMAL', - DIRTY: 'DIRTY', // needs to be interacted with before submitting - INCORRECT: 'INCORRECT' // was wrong after submission - }; +const EntryStatus = { + NORMAL: 'NORMAL', + DIRTY: 'DIRTY', // needs to be interacted with before submitting + INCORRECT: 'INCORRECT' // was wrong after submission +}; - areaModelCommon.register( 'EntryStatus', EntryStatus ); +areaModelCommon.register( 'EntryStatus', EntryStatus ); - // @public {Array.} - All values the enumeration can take. - EntryStatus.VALUES = [ - EntryStatus.NORMAL, - EntryStatus.DIRTY, - EntryStatus.INCORRECT - ]; +// @public {Array.} - All values the enumeration can take. +EntryStatus.VALUES = [ + EntryStatus.NORMAL, + EntryStatus.DIRTY, + EntryStatus.INCORRECT +]; - // verify that enumeration is immutable, without the runtime penalty in production code - if ( assert ) { Object.freeze( EntryStatus ); } +// verify that enumeration is immutable, without the runtime penalty in production code +if ( assert ) { Object.freeze( EntryStatus ); } - return EntryStatus; -} ); +export default EntryStatus; \ No newline at end of file diff --git a/js/game/model/EntryType.js b/js/game/model/EntryType.js index 76f92e52..332983dd 100644 --- a/js/game/model/EntryType.js +++ b/js/game/model/EntryType.js @@ -5,48 +5,44 @@ * * @author Jonathan Olson */ -define( require => { - 'use strict'; - - // modules - const areaModelCommon = require( 'AREA_MODEL_COMMON/areaModelCommon' ); - const EntryDisplayType = require( 'AREA_MODEL_COMMON/game/model/EntryDisplayType' ); - - const EntryType = { - EDITABLE: 'EDITABLE', - DYNAMIC: 'DYNAMIC', - GIVEN: 'GIVEN' - }; - - areaModelCommon.register( 'EntryType', EntryType ); - - // @public {Array.} - All values the enumeration can take. - EntryType.VALUES = [ - EntryType.EDITABLE, // the user inputs this value - EntryType.DYNAMIC, // this value can change (be computed) based on the user's input - EntryType.GIVEN // this value is fixed for a given challenge - ]; - - const gameToDisplayMap = {}; - gameToDisplayMap[ EntryType.EDITABLE ] = EntryDisplayType.EDITABLE; - gameToDisplayMap[ EntryType.DYNAMIC ] = EntryDisplayType.READOUT; - gameToDisplayMap[ EntryType.GIVEN ] = EntryDisplayType.READOUT; - - /** - * Returns the preferred display type for a given game value. - * @public - * - * @param {EntryType} type - * @returns {boolean} - */ - EntryType.toDisplayType = function( type ) { - assert && assert( _.includes( EntryType.VALUES, type ) ); - - return gameToDisplayMap[ type ]; - }; - - // verify that enumeration is immutable, without the runtime penalty in production code - if ( assert ) { Object.freeze( EntryType ); } - - return EntryType; -} ); + +import areaModelCommon from '../../areaModelCommon.js'; +import EntryDisplayType from './EntryDisplayType.js'; + +const EntryType = { + EDITABLE: 'EDITABLE', + DYNAMIC: 'DYNAMIC', + GIVEN: 'GIVEN' +}; + +areaModelCommon.register( 'EntryType', EntryType ); + +// @public {Array.} - All values the enumeration can take. +EntryType.VALUES = [ + EntryType.EDITABLE, // the user inputs this value + EntryType.DYNAMIC, // this value can change (be computed) based on the user's input + EntryType.GIVEN // this value is fixed for a given challenge +]; + +const gameToDisplayMap = {}; +gameToDisplayMap[ EntryType.EDITABLE ] = EntryDisplayType.EDITABLE; +gameToDisplayMap[ EntryType.DYNAMIC ] = EntryDisplayType.READOUT; +gameToDisplayMap[ EntryType.GIVEN ] = EntryDisplayType.READOUT; + +/** + * Returns the preferred display type for a given game value. + * @public + * + * @param {EntryType} type + * @returns {boolean} + */ +EntryType.toDisplayType = function( type ) { + assert && assert( _.includes( EntryType.VALUES, type ) ); + + return gameToDisplayMap[ type ]; +}; + +// verify that enumeration is immutable, without the runtime penalty in production code +if ( assert ) { Object.freeze( EntryType ); } + +export default EntryType; \ No newline at end of file diff --git a/js/game/model/GameAreaDisplay.js b/js/game/model/GameAreaDisplay.js index d122f0fd..ed62ad47 100644 --- a/js/game/model/GameAreaDisplay.js +++ b/js/game/model/GameAreaDisplay.js @@ -5,88 +5,85 @@ * * @author Jonathan Olson */ -define( require => { - 'use strict'; - // modules - const AreaChallenge = require( 'AREA_MODEL_COMMON/game/model/AreaChallenge' ); - const AreaChallengeDescription = require( 'AREA_MODEL_COMMON/game/model/AreaChallengeDescription' ); - const areaModelCommon = require( 'AREA_MODEL_COMMON/areaModelCommon' ); - const DerivedProperty = require( 'AXON/DerivedProperty' ); - const DynamicProperty = require( 'AXON/DynamicProperty' ); - const Entry = require( 'AREA_MODEL_COMMON/game/model/Entry' ); - const EntryType = require( 'AREA_MODEL_COMMON/game/model/EntryType' ); - const GenericAreaDisplay = require( 'AREA_MODEL_COMMON/generic/model/GenericAreaDisplay' ); - const inherit = require( 'PHET_CORE/inherit' ); - const OrientationPair = require( 'AREA_MODEL_COMMON/common/model/OrientationPair' ); - const Property = require( 'AXON/Property' ); +import DerivedProperty from '../../../../axon/js/DerivedProperty.js'; +import DynamicProperty from '../../../../axon/js/DynamicProperty.js'; +import Property from '../../../../axon/js/Property.js'; +import inherit from '../../../../phet-core/js/inherit.js'; +import areaModelCommon from '../../areaModelCommon.js'; +import OrientationPair from '../../common/model/OrientationPair.js'; +import GenericAreaDisplay from '../../generic/model/GenericAreaDisplay.js'; +import AreaChallenge from './AreaChallenge.js'; +import AreaChallengeDescription from './AreaChallengeDescription.js'; +import Entry from './Entry.js'; +import EntryType from './EntryType.js'; - /** - * @constructor - * @extends {GenericAreaDisplay} - * - * @param {Property.} areaChallengeProperty - */ - function GameAreaDisplay( areaChallengeProperty ) { - const self = this; +/** + * @constructor + * @extends {GenericAreaDisplay} + * + * @param {Property.} areaChallengeProperty + */ +function GameAreaDisplay( areaChallengeProperty ) { + const self = this; - // This placeholder will never be seen in the user interface, it is to help make sure areaChallengeProperty is never - // null, see below. - const placeholderChallenge = new AreaChallenge( AreaChallengeDescription.LEVEL_1_NUMBERS_1 ); + // This placeholder will never be seen in the user interface, it is to help make sure areaChallengeProperty is never + // null, see below. + const placeholderChallenge = new AreaChallenge( AreaChallengeDescription.LEVEL_1_NUMBERS_1 ); - // @public {Property.} - Always has an AreaChallenge, unlike the passed-in nullable variety. This is - // because we want to show the view of the last challenge as we animate back to the level-selection screen. - this.areaChallengeProperty = new Property( placeholderChallenge ); - areaChallengeProperty.link( function( areaChallenge ) { - if ( areaChallenge ) { - self.areaChallengeProperty.value = areaChallenge; - } - } ); + // @public {Property.} - Always has an AreaChallenge, unlike the passed-in nullable variety. This is + // because we want to show the view of the last challenge as we animate back to the level-selection screen. + this.areaChallengeProperty = new Property( placeholderChallenge ); + areaChallengeProperty.link( function( areaChallenge ) { + if ( areaChallenge ) { + self.areaChallengeProperty.value = areaChallenge; + } + } ); - GenericAreaDisplay.call( this, new DerivedProperty( [ this.areaChallengeProperty ], _.property( 'area' ) ) ); + GenericAreaDisplay.call( this, new DerivedProperty( [ this.areaChallengeProperty ], _.property( 'area' ) ) ); - // @public {OrientationPair.>} - Values for dimension line label and product box, null is - // hidden. - // NOTE: Overridden from the AreaDisplay version. - this.totalProperties = OrientationPair.create( function( orientation ) { - return new DynamicProperty( self.areaChallengeProperty, { - derive: function( areaChallenge ) { - return areaChallenge.totalProperties.get( orientation ); - } - } ); + // @public {OrientationPair.>} - Values for dimension line label and product box, null is + // hidden. + // NOTE: Overridden from the AreaDisplay version. + this.totalProperties = OrientationPair.create( function( orientation ) { + return new DynamicProperty( self.areaChallengeProperty, { + derive: function( areaChallenge ) { + return areaChallenge.totalProperties.get( orientation ); + } } ); + } ); - // @public {OrientationPair.>>} - // Partition sizes. Inner values may be changed by the view client. + // @public {OrientationPair.>>} + // Partition sizes. Inner values may be changed by the view client. - this.partitionSizeEntriesProperties = OrientationPair.create( function( orientation ) { - return new DerivedProperty( [ self.areaChallengeProperty ], function( areaChallenge ) { - // If there's only one value on a side (and it's a given), there is no use showing a size here. - if ( areaChallenge.partitionSizeEntries.get( orientation ).length === 1 && - areaChallenge.description.partitionTypes.get( orientation )[ 0 ] === EntryType.GIVEN ) { - return [ new Entry( null ) ]; - } - else { - return areaChallenge.partitionSizeEntries.get( orientation ); - } - } ); + this.partitionSizeEntriesProperties = OrientationPair.create( function( orientation ) { + return new DerivedProperty( [ self.areaChallengeProperty ], function( areaChallenge ) { + // If there's only one value on a side (and it's a given), there is no use showing a size here. + if ( areaChallenge.partitionSizeEntries.get( orientation ).length === 1 && + areaChallenge.description.partitionTypes.get( orientation )[ 0 ] === EntryType.GIVEN ) { + return [ new Entry( null ) ]; + } + else { + return areaChallenge.partitionSizeEntries.get( orientation ); + } } ); + } ); - // @public {Property.>} - Reference to a 2D array for the grid of partial products. - // First index is vertical (for the row), second is horizontal (for the column) - this.partialProductEntriesProperty = new DerivedProperty( [ this.areaChallengeProperty ], _.property( 'partialProductSizeEntries' ) ); + // @public {Property.>} - Reference to a 2D array for the grid of partial products. + // First index is vertical (for the row), second is horizontal (for the column) + this.partialProductEntriesProperty = new DerivedProperty( [ this.areaChallengeProperty ], _.property( 'partialProductSizeEntries' ) ); - // @public {Property.>} - Reference to an array of editable properties for the total area. - // Uses just one for an editable "constant" value, and multiple properties for polynomial entry (one per term). - this.totalEntriesProperty = new DerivedProperty( [ this.areaChallengeProperty ], _.property( 'totalCoefficientEntries' ) ); + // @public {Property.>} - Reference to an array of editable properties for the total area. + // Uses just one for an editable "constant" value, and multiple properties for polynomial entry (one per term). + this.totalEntriesProperty = new DerivedProperty( [ this.areaChallengeProperty ], _.property( 'totalCoefficientEntries' ) ); - // @public {Property.} - The "total area" property reference - this.totalProperty = new DynamicProperty( this.areaChallengeProperty, { - derive: 'totalProperty' - } ); - } + // @public {Property.} - The "total area" property reference + this.totalProperty = new DynamicProperty( this.areaChallengeProperty, { + derive: 'totalProperty' + } ); +} - areaModelCommon.register( 'GameAreaDisplay', GameAreaDisplay ); +areaModelCommon.register( 'GameAreaDisplay', GameAreaDisplay ); - return inherit( GenericAreaDisplay, GameAreaDisplay ); -} ); +inherit( GenericAreaDisplay, GameAreaDisplay ); +export default GameAreaDisplay; \ No newline at end of file diff --git a/js/game/model/GameAreaModel.js b/js/game/model/GameAreaModel.js index 01d9f287..79adee8a 100644 --- a/js/game/model/GameAreaModel.js +++ b/js/game/model/GameAreaModel.js @@ -5,177 +5,173 @@ * * @author Jonathan Olson */ -define( require => { - 'use strict'; - - // modules - const AreaLevel = require( 'AREA_MODEL_COMMON/game/model/AreaLevel' ); - const areaModelCommon = require( 'AREA_MODEL_COMMON/areaModelCommon' ); - const DynamicProperty = require( 'AXON/DynamicProperty' ); - const Entry = require( 'AREA_MODEL_COMMON/game/model/Entry' ); - const EntryStatus = require( 'AREA_MODEL_COMMON/game/model/EntryStatus' ); - const GameState = require( 'AREA_MODEL_COMMON/game/model/GameState' ); - const inherit = require( 'PHET_CORE/inherit' ); - const Property = require( 'AXON/Property' ); +import DynamicProperty from '../../../../axon/js/DynamicProperty.js'; +import Property from '../../../../axon/js/Property.js'; +import inherit from '../../../../phet-core/js/inherit.js'; +import areaModelCommon from '../../areaModelCommon.js'; +import AreaLevel from './AreaLevel.js'; +import Entry from './Entry.js'; +import EntryStatus from './EntryStatus.js'; +import GameState from './GameState.js'; + +/** + * @constructor + * @extends {Object} + * + * @param {Array.} levels + * @param {boolean} hasExponents + */ +function GameAreaModel( levels, hasExponents ) { + + // @public {Array.} + this.levels = levels; + + // @public {boolean} + this.hasExponents = hasExponents; + + // @public {Property.} - The current level + this.currentLevelProperty = new Property( null, { + isValidValue: function( value ) { + return value === null || value instanceof AreaLevel; + } + } ); + + // @public {Property.} + this.currentChallengeProperty = new DynamicProperty( this.currentLevelProperty, { + derive: 'currentChallengeProperty' + } ); + this.currentChallengeProperty.lazyLink( this.activeEntryProperty.reset.bind( this.activeEntryProperty ) ); + + // @public {Property.} - This is null when there is no current challenge (e.g. level selection) + this.stateProperty = new DynamicProperty( this.currentChallengeProperty, { + derive: 'stateProperty', + bidirectional: true + }, { + validValues: GameState.VALUES.concat( [ null ] ) + } ); + + // @public {Property.} - Whether the check button should be enabled + this.allowCheckingProperty = new DynamicProperty( this.currentChallengeProperty, { + derive: 'allowCheckingProperty', + defaultValue: false + } ); +} + +areaModelCommon.register( 'GameAreaModel', GameAreaModel ); + +export default inherit( Object, GameAreaModel, { /** - * @constructor - * @extends {Object} + * Selects a given level, making it the current level. + * @public * - * @param {Array.} levels - * @param {boolean} hasExponents + * @param {AreaLevel} level */ - function GameAreaModel( levels, hasExponents ) { + selectLevel: function( level ) { + level.select(); + this.currentLevelProperty.value = level; + }, - // @public {Array.} - this.levels = levels; + /** + * Sets the value of the current editable property to that of the provided term. + * @public + * + * @param {Term} term + */ + setActiveTerm: function( term ) { + // Appearance change for https://github.com/phetsims/area-model-common/issues/42, handles only showing the + // "wrong" values for the variables 6-1 case (non-unique) + this.currentChallengeProperty.value.checkNonUniqueChanges(); - // @public {boolean} - this.hasExponents = hasExponents; + this.activeEntryProperty.value.valueProperty.value = term; + this.activeEntryProperty.value.statusProperty.value = EntryStatus.NORMAL; + this.activeEntryProperty.value = null; + }, - // @public {Property.} - The current level - this.currentLevelProperty = new Property( null, { - isValidValue: function( value ) { - return value === null || value instanceof AreaLevel; - } - } ); + /** + * Checks the user's input against the known answer. + * @public + */ + check: function() { + // Close any keypads, see https://github.com/phetsims/area-model-common/issues/66 + this.activeEntryProperty.value = null; - // @public {Property.} - this.currentChallengeProperty = new DynamicProperty( this.currentLevelProperty, { - derive: 'currentChallengeProperty' - } ); - this.currentChallengeProperty.lazyLink( this.activeEntryProperty.reset.bind( this.activeEntryProperty ) ); - - // @public {Property.} - This is null when there is no current challenge (e.g. level selection) - this.stateProperty = new DynamicProperty( this.currentChallengeProperty, { - derive: 'stateProperty', - bidirectional: true - }, { - validValues: GameState.VALUES.concat( [ null ] ) - } ); + this.currentLevelProperty.value.scoreProperty.value += challenge.check(); + }, - // @public {Property.} - Whether the check button should be enabled - this.allowCheckingProperty = new DynamicProperty( this.currentChallengeProperty, { - derive: 'allowCheckingProperty', - defaultValue: false - } ); - } + /** + * Move to try another time. + * @public + */ + tryAgain: function() { + if ( this.currentChallengeProperty.value ) { + this.currentChallengeProperty.value.tryAgain(); + } + }, + + /** + * Move to the next challenge. + * @public + */ + next: function() { + if ( this.currentLevelProperty.value ) { + this.currentLevelProperty.value.next(); + } + }, - areaModelCommon.register( 'GameAreaModel', GameAreaModel ); - - return inherit( Object, GameAreaModel, { - /** - * Selects a given level, making it the current level. - * @public - * - * @param {AreaLevel} level - */ - selectLevel: function( level ) { - level.select(); - this.currentLevelProperty.value = level; - }, - - /** - * Sets the value of the current editable property to that of the provided term. - * @public - * - * @param {Term} term - */ - setActiveTerm: function( term ) { - // Appearance change for https://github.com/phetsims/area-model-common/issues/42, handles only showing the - // "wrong" values for the variables 6-1 case (non-unique) - this.currentChallengeProperty.value.checkNonUniqueChanges(); - - this.activeEntryProperty.value.valueProperty.value = term; - this.activeEntryProperty.value.statusProperty.value = EntryStatus.NORMAL; - this.activeEntryProperty.value = null; - }, - - /** - * Checks the user's input against the known answer. - * @public - */ - check: function() { - // Close any keypads, see https://github.com/phetsims/area-model-common/issues/66 - this.activeEntryProperty.value = null; - - const challenge = this.currentChallengeProperty.value; - - this.currentLevelProperty.value.scoreProperty.value += challenge.check(); - }, - - /** - * Move to try another time. - * @public - */ - tryAgain: function() { - if ( this.currentChallengeProperty.value ) { - this.currentChallengeProperty.value.tryAgain(); - } - }, - - /** - * Move to the next challenge. - * @public - */ - next: function() { - if ( this.currentLevelProperty.value ) { - this.currentLevelProperty.value.next(); - } - }, - - /** - * Goes to the level selection. - * @public - */ - moveToLevelSelection: function() { - this.currentLevelProperty.value = null; // move to no level - }, - - /** - * Shows the answer. - * @public - */ - showAnswer: function() { - if ( this.currentChallengeProperty.value ) { - this.currentChallengeProperty.value.showAnswers(); - this.currentChallengeProperty.value.stateProperty.value = GameState.SHOW_SOLUTION; - this.activeEntryProperty.reset(); - } - }, - - /** - * Fills in the answers without moving to the "SHOW_SOLUTION" state. - * @public - */ - cheat: function() { - if ( this.currentChallengeProperty.value ) { - - // filling in the correct answers to the entries - this.currentChallengeProperty.value.showAnswers(); - - // unselecting the edited one and closing the keypad if necessary - this.activeEntryProperty.reset(); - } - }, - - /** - * Returns the model to its initial state. - * @public - */ - reset: function() { + /** + * Goes to the level selection. + * @public + */ + moveToLevelSelection: function() { + this.currentLevelProperty.value = null; // move to no level + }, + + /** + * Shows the answer. + * @public + */ + showAnswer: function() { + if ( this.currentChallengeProperty.value ) { + this.currentChallengeProperty.value.showAnswers(); + this.currentChallengeProperty.value.stateProperty.value = GameState.SHOW_SOLUTION; this.activeEntryProperty.reset(); - this.currentLevelProperty.reset(); - this.levels.forEach( function( level ) { - level.reset(); - } ); } - } ); -} ); + }, + + /** + * Fills in the answers without moving to the "SHOW_SOLUTION" state. + * @public + */ + cheat: function() { + if ( this.currentChallengeProperty.value ) { + + // filling in the correct answers to the entries + this.currentChallengeProperty.value.showAnswers(); + + // unselecting the edited one and closing the keypad if necessary + this.activeEntryProperty.reset(); + } + }, + + /** + * Returns the model to its initial state. + * @public + */ + reset: function() { + this.activeEntryProperty.reset(); + this.currentLevelProperty.reset(); + this.levels.forEach( function( level ) { + level.reset(); + } ); + } +} ); \ No newline at end of file diff --git a/js/game/model/GameState.js b/js/game/model/GameState.js index c6630cce..78fe6a98 100644 --- a/js/game/model/GameState.js +++ b/js/game/model/GameState.js @@ -6,47 +6,43 @@ * * @author Jonathan Olson */ -define( require => { - 'use strict'; - // modules - const areaModelCommon = require( 'AREA_MODEL_COMMON/areaModelCommon' ); +import areaModelCommon from '../../areaModelCommon.js'; - const GameState = { - // "check" button, editable - FIRST_ATTEMPT: 'FIRST_ATTEMPT', - SECOND_ATTEMPT: 'SECOND_ATTEMPT', +const GameState = { + // "check" button, editable + FIRST_ATTEMPT: 'FIRST_ATTEMPT', + SECOND_ATTEMPT: 'SECOND_ATTEMPT', - // "next" button, happy face with +1 or +2 depending on score. NOT editable - CORRECT_ANSWER: 'CORRECT_ANSWER', + // "next" button, happy face with +1 or +2 depending on score. NOT editable + CORRECT_ANSWER: 'CORRECT_ANSWER', - // "try again" button, sad face (editable?) - triggers next state on edit? - WRONG_FIRST_ANSWER: 'WRONG_FIRST_ANSWER', + // "try again" button, sad face (editable?) - triggers next state on edit? + WRONG_FIRST_ANSWER: 'WRONG_FIRST_ANSWER', - // "show solution" button, sad face (editable?) - no trigger on edit? - WRONG_SECOND_ANSWER: 'WRONG_SECOND_ANSWER', + // "show solution" button, sad face (editable?) - no trigger on edit? + WRONG_SECOND_ANSWER: 'WRONG_SECOND_ANSWER', - // "next" button, NOT editable, replaced with a solution - SHOW_SOLUTION: 'SHOW_SOLUTION', + // "next" button, NOT editable, replaced with a solution + SHOW_SOLUTION: 'SHOW_SOLUTION', - LEVEL_COMPLETE: 'LEVEL_COMPLETE' - }; + LEVEL_COMPLETE: 'LEVEL_COMPLETE' +}; - areaModelCommon.register( 'GameState', GameState ); +areaModelCommon.register( 'GameState', GameState ); - // @public {Array.} - All values the enumeration can take. - GameState.VALUES = [ - GameState.FIRST_ATTEMPT, - GameState.SECOND_ATTEMPT, - GameState.CORRECT_ANSWER, - GameState.WRONG_FIRST_ANSWER, - GameState.WRONG_SECOND_ANSWER, - GameState.SHOW_SOLUTION, - GameState.LEVEL_COMPLETE - ]; +// @public {Array.} - All values the enumeration can take. +GameState.VALUES = [ + GameState.FIRST_ATTEMPT, + GameState.SECOND_ATTEMPT, + GameState.CORRECT_ANSWER, + GameState.WRONG_FIRST_ANSWER, + GameState.WRONG_SECOND_ANSWER, + GameState.SHOW_SOLUTION, + GameState.LEVEL_COMPLETE +]; - // verify that enum is immutable, without the runtime penalty in production code - if ( assert ) { Object.freeze( GameState ); } +// verify that enum is immutable, without the runtime penalty in production code +if ( assert ) { Object.freeze( GameState ); } - return GameState; -} ); +export default GameState; \ No newline at end of file diff --git a/js/game/model/GenericGameAreaModel.js b/js/game/model/GenericGameAreaModel.js index d30112fb..4abbb0a6 100644 --- a/js/game/model/GenericGameAreaModel.js +++ b/js/game/model/GenericGameAreaModel.js @@ -5,65 +5,62 @@ * * @author Jonathan Olson */ -define( require => { - 'use strict'; - // modules - const AreaChallengeDescription = require( 'AREA_MODEL_COMMON/game/model/AreaChallengeDescription' ); - const AreaChallengeType = require( 'AREA_MODEL_COMMON/game/model/AreaChallengeType' ); - const AreaLevel = require( 'AREA_MODEL_COMMON/game/model/AreaLevel' ); - const areaModelCommon = require( 'AREA_MODEL_COMMON/areaModelCommon' ); - const AreaModelCommonColorProfile = require( 'AREA_MODEL_COMMON/common/view/AreaModelCommonColorProfile' ); - const GameAreaModel = require( 'AREA_MODEL_COMMON/game/model/GameAreaModel' ); - const inherit = require( 'PHET_CORE/inherit' ); +import inherit from '../../../../phet-core/js/inherit.js'; +import areaModelCommon from '../../areaModelCommon.js'; +import AreaModelCommonColorProfile from '../../common/view/AreaModelCommonColorProfile.js'; +import AreaChallengeDescription from './AreaChallengeDescription.js'; +import AreaChallengeType from './AreaChallengeType.js'; +import AreaLevel from './AreaLevel.js'; +import GameAreaModel from './GameAreaModel.js'; - /** - * @constructor - * @extends {Object} - */ - function GenericGameAreaModel() { - GameAreaModel.call( this, [ - new AreaLevel( 1, AreaChallengeType.NUMBERS, AreaModelCommonColorProfile.numbersIconBackgroundProperty, [ - AreaChallengeDescription.LEVEL_1_NUMBERS_1, - AreaChallengeDescription.LEVEL_1_NUMBERS_2, - AreaChallengeDescription.LEVEL_1_NUMBERS_3, - AreaChallengeDescription.LEVEL_1_NUMBERS_4, - AreaChallengeDescription.LEVEL_1_NUMBERS_5, - AreaChallengeDescription.LEVEL_1_NUMBERS_6 - ] ), - new AreaLevel( 2, AreaChallengeType.NUMBERS, AreaModelCommonColorProfile.numbersIconBackgroundProperty, [ - AreaChallengeDescription.LEVEL_2_NUMBERS_1, - AreaChallengeDescription.LEVEL_2_NUMBERS_2, - AreaChallengeDescription.LEVEL_2_NUMBERS_3, - AreaChallengeDescription.LEVEL_2_NUMBERS_4, - AreaChallengeDescription.LEVEL_2_NUMBERS_5 - ] ), - new AreaLevel( 3, AreaChallengeType.NUMBERS, AreaModelCommonColorProfile.numbersIconBackgroundProperty, [ - AreaChallengeDescription.LEVEL_3_NUMBERS_1, - AreaChallengeDescription.LEVEL_3_NUMBERS_2, - AreaChallengeDescription.LEVEL_3_NUMBERS_3, - AreaChallengeDescription.LEVEL_3_NUMBERS_4, - AreaChallengeDescription.LEVEL_3_NUMBERS_5, - AreaChallengeDescription.LEVEL_3_NUMBERS_6 - ] ), - new AreaLevel( 4, AreaChallengeType.NUMBERS, AreaModelCommonColorProfile.numbersIconBackgroundProperty, [ - AreaChallengeDescription.LEVEL_4_NUMBERS_1, - AreaChallengeDescription.LEVEL_4_NUMBERS_2, - AreaChallengeDescription.LEVEL_4_NUMBERS_3, - AreaChallengeDescription.LEVEL_4_NUMBERS_4, - AreaChallengeDescription.LEVEL_4_NUMBERS_5 - ] ), - new AreaLevel( 5, AreaChallengeType.NUMBERS, AreaModelCommonColorProfile.numbersIconBackgroundProperty, [ - AreaChallengeDescription.LEVEL_5_NUMBERS_1, - AreaChallengeDescription.LEVEL_5_NUMBERS_3 - ] ), - new AreaLevel( 6, AreaChallengeType.NUMBERS, AreaModelCommonColorProfile.numbersIconBackgroundProperty, [ - AreaChallengeDescription.LEVEL_6_NUMBERS_1 - ] ) - ], false ); - } +/** + * @constructor + * @extends {Object} + */ +function GenericGameAreaModel() { + GameAreaModel.call( this, [ + new AreaLevel( 1, AreaChallengeType.NUMBERS, AreaModelCommonColorProfile.numbersIconBackgroundProperty, [ + AreaChallengeDescription.LEVEL_1_NUMBERS_1, + AreaChallengeDescription.LEVEL_1_NUMBERS_2, + AreaChallengeDescription.LEVEL_1_NUMBERS_3, + AreaChallengeDescription.LEVEL_1_NUMBERS_4, + AreaChallengeDescription.LEVEL_1_NUMBERS_5, + AreaChallengeDescription.LEVEL_1_NUMBERS_6 + ] ), + new AreaLevel( 2, AreaChallengeType.NUMBERS, AreaModelCommonColorProfile.numbersIconBackgroundProperty, [ + AreaChallengeDescription.LEVEL_2_NUMBERS_1, + AreaChallengeDescription.LEVEL_2_NUMBERS_2, + AreaChallengeDescription.LEVEL_2_NUMBERS_3, + AreaChallengeDescription.LEVEL_2_NUMBERS_4, + AreaChallengeDescription.LEVEL_2_NUMBERS_5 + ] ), + new AreaLevel( 3, AreaChallengeType.NUMBERS, AreaModelCommonColorProfile.numbersIconBackgroundProperty, [ + AreaChallengeDescription.LEVEL_3_NUMBERS_1, + AreaChallengeDescription.LEVEL_3_NUMBERS_2, + AreaChallengeDescription.LEVEL_3_NUMBERS_3, + AreaChallengeDescription.LEVEL_3_NUMBERS_4, + AreaChallengeDescription.LEVEL_3_NUMBERS_5, + AreaChallengeDescription.LEVEL_3_NUMBERS_6 + ] ), + new AreaLevel( 4, AreaChallengeType.NUMBERS, AreaModelCommonColorProfile.numbersIconBackgroundProperty, [ + AreaChallengeDescription.LEVEL_4_NUMBERS_1, + AreaChallengeDescription.LEVEL_4_NUMBERS_2, + AreaChallengeDescription.LEVEL_4_NUMBERS_3, + AreaChallengeDescription.LEVEL_4_NUMBERS_4, + AreaChallengeDescription.LEVEL_4_NUMBERS_5 + ] ), + new AreaLevel( 5, AreaChallengeType.NUMBERS, AreaModelCommonColorProfile.numbersIconBackgroundProperty, [ + AreaChallengeDescription.LEVEL_5_NUMBERS_1, + AreaChallengeDescription.LEVEL_5_NUMBERS_3 + ] ), + new AreaLevel( 6, AreaChallengeType.NUMBERS, AreaModelCommonColorProfile.numbersIconBackgroundProperty, [ + AreaChallengeDescription.LEVEL_6_NUMBERS_1 + ] ) + ], false ); +} - areaModelCommon.register( 'GenericGameAreaModel', GenericGameAreaModel ); +areaModelCommon.register( 'GenericGameAreaModel', GenericGameAreaModel ); - return inherit( GameAreaModel, GenericGameAreaModel ); -} ); +inherit( GameAreaModel, GenericGameAreaModel ); +export default GenericGameAreaModel; \ No newline at end of file diff --git a/js/game/model/InputMethod.js b/js/game/model/InputMethod.js index 4cda9966..2983eae2 100644 --- a/js/game/model/InputMethod.js +++ b/js/game/model/InputMethod.js @@ -5,42 +5,38 @@ * * @author Jonathan Olson */ -define( require => { - 'use strict'; - - // modules - const areaModelCommon = require( 'AREA_MODEL_COMMON/areaModelCommon' ); - - const InputMethod = { - CONSTANT: 'CONSTANT', - TERM: 'TERM', - POLYNOMIAL_2: 'POLYNOMIAL_2', // with x^2 - POLYNOMIAL_1: 'POLYNOMIAL_1' // without x^2 - }; - - areaModelCommon.register( 'InputMethod', InputMethod ); - - /** - * Whether an entry needs polynomial or term input. - * @public - * - * @param {InputMethod} inputMethod - * @returns {boolean} - */ - InputMethod.isPolynomial = function( inputMethod ) { - return inputMethod === InputMethod.POLYNOMIAL_1 || inputMethod === InputMethod.POLYNOMIAL_2; - }; - - // @public {Array.} - All values the enumeration can take. - InputMethod.VALUES = [ - InputMethod.CONSTANT, - InputMethod.TERM, - InputMethod.POLYNOMIAL_2, - InputMethod.POLYNOMIAL_1 - ]; - - // verify that enumeration is immutable, without the runtime penalty in production code - if ( assert ) { Object.freeze( InputMethod ); } - - return InputMethod; -} ); + +import areaModelCommon from '../../areaModelCommon.js'; + +const InputMethod = { + CONSTANT: 'CONSTANT', + TERM: 'TERM', + POLYNOMIAL_2: 'POLYNOMIAL_2', // with x^2 + POLYNOMIAL_1: 'POLYNOMIAL_1' // without x^2 +}; + +areaModelCommon.register( 'InputMethod', InputMethod ); + +/** + * Whether an entry needs polynomial or term input. + * @public + * + * @param {InputMethod} inputMethod + * @returns {boolean} + */ +InputMethod.isPolynomial = function( inputMethod ) { + return inputMethod === InputMethod.POLYNOMIAL_1 || inputMethod === InputMethod.POLYNOMIAL_2; +}; + +// @public {Array.} - All values the enumeration can take. +InputMethod.VALUES = [ + InputMethod.CONSTANT, + InputMethod.TERM, + InputMethod.POLYNOMIAL_2, + InputMethod.POLYNOMIAL_1 +]; + +// verify that enumeration is immutable, without the runtime penalty in production code +if ( assert ) { Object.freeze( InputMethod ); } + +export default InputMethod; \ No newline at end of file diff --git a/js/game/model/VariablesGameAreaModel.js b/js/game/model/VariablesGameAreaModel.js index 88adf3fd..5df5aa22 100644 --- a/js/game/model/VariablesGameAreaModel.js +++ b/js/game/model/VariablesGameAreaModel.js @@ -5,57 +5,54 @@ * * @author Jonathan Olson */ -define( require => { - 'use strict'; - // modules - const AreaChallengeDescription = require( 'AREA_MODEL_COMMON/game/model/AreaChallengeDescription' ); - const AreaChallengeType = require( 'AREA_MODEL_COMMON/game/model/AreaChallengeType' ); - const AreaLevel = require( 'AREA_MODEL_COMMON/game/model/AreaLevel' ); - const areaModelCommon = require( 'AREA_MODEL_COMMON/areaModelCommon' ); - const AreaModelCommonColorProfile = require( 'AREA_MODEL_COMMON/common/view/AreaModelCommonColorProfile' ); - const GameAreaModel = require( 'AREA_MODEL_COMMON/game/model/GameAreaModel' ); - const inherit = require( 'PHET_CORE/inherit' ); +import inherit from '../../../../phet-core/js/inherit.js'; +import areaModelCommon from '../../areaModelCommon.js'; +import AreaModelCommonColorProfile from '../../common/view/AreaModelCommonColorProfile.js'; +import AreaChallengeDescription from './AreaChallengeDescription.js'; +import AreaChallengeType from './AreaChallengeType.js'; +import AreaLevel from './AreaLevel.js'; +import GameAreaModel from './GameAreaModel.js'; - /** - * @constructor - * @extends {Object} - */ - function VariablesGameAreaModel() { - GameAreaModel.call( this, [ - new AreaLevel( 1, AreaChallengeType.VARIABLES, AreaModelCommonColorProfile.variablesIconBackgroundProperty, [ - AreaChallengeDescription.LEVEL_1_VARIABLES_1, - AreaChallengeDescription.LEVEL_1_VARIABLES_2, - AreaChallengeDescription.LEVEL_1_VARIABLES_3, - AreaChallengeDescription.LEVEL_1_VARIABLES_4 - ] ), - new AreaLevel( 2, AreaChallengeType.VARIABLES, AreaModelCommonColorProfile.variablesIconBackgroundProperty, [ - AreaChallengeDescription.LEVEL_2_VARIABLES_1, - AreaChallengeDescription.LEVEL_2_VARIABLES_2 - ] ), - new AreaLevel( 3, AreaChallengeType.VARIABLES, AreaModelCommonColorProfile.variablesIconBackgroundProperty, [ - AreaChallengeDescription.LEVEL_3_VARIABLES_1, - AreaChallengeDescription.LEVEL_3_VARIABLES_2, - AreaChallengeDescription.LEVEL_3_VARIABLES_3, - AreaChallengeDescription.LEVEL_3_VARIABLES_4, - AreaChallengeDescription.LEVEL_3_VARIABLES_5, - AreaChallengeDescription.LEVEL_3_VARIABLES_6 - ] ), - new AreaLevel( 4, AreaChallengeType.VARIABLES, AreaModelCommonColorProfile.variablesIconBackgroundProperty, [ - AreaChallengeDescription.LEVEL_4_VARIABLES_1, - AreaChallengeDescription.LEVEL_4_VARIABLES_2 - ] ), - new AreaLevel( 5, AreaChallengeType.VARIABLES, AreaModelCommonColorProfile.variablesIconBackgroundProperty, [ - AreaChallengeDescription.LEVEL_5_VARIABLES_1, - AreaChallengeDescription.LEVEL_5_VARIABLES_2 - ] ), - new AreaLevel( 6, AreaChallengeType.VARIABLES, AreaModelCommonColorProfile.variablesIconBackgroundProperty, [ - AreaChallengeDescription.LEVEL_6_VARIABLES_1 - ] ) - ], true ); - } +/** + * @constructor + * @extends {Object} + */ +function VariablesGameAreaModel() { + GameAreaModel.call( this, [ + new AreaLevel( 1, AreaChallengeType.VARIABLES, AreaModelCommonColorProfile.variablesIconBackgroundProperty, [ + AreaChallengeDescription.LEVEL_1_VARIABLES_1, + AreaChallengeDescription.LEVEL_1_VARIABLES_2, + AreaChallengeDescription.LEVEL_1_VARIABLES_3, + AreaChallengeDescription.LEVEL_1_VARIABLES_4 + ] ), + new AreaLevel( 2, AreaChallengeType.VARIABLES, AreaModelCommonColorProfile.variablesIconBackgroundProperty, [ + AreaChallengeDescription.LEVEL_2_VARIABLES_1, + AreaChallengeDescription.LEVEL_2_VARIABLES_2 + ] ), + new AreaLevel( 3, AreaChallengeType.VARIABLES, AreaModelCommonColorProfile.variablesIconBackgroundProperty, [ + AreaChallengeDescription.LEVEL_3_VARIABLES_1, + AreaChallengeDescription.LEVEL_3_VARIABLES_2, + AreaChallengeDescription.LEVEL_3_VARIABLES_3, + AreaChallengeDescription.LEVEL_3_VARIABLES_4, + AreaChallengeDescription.LEVEL_3_VARIABLES_5, + AreaChallengeDescription.LEVEL_3_VARIABLES_6 + ] ), + new AreaLevel( 4, AreaChallengeType.VARIABLES, AreaModelCommonColorProfile.variablesIconBackgroundProperty, [ + AreaChallengeDescription.LEVEL_4_VARIABLES_1, + AreaChallengeDescription.LEVEL_4_VARIABLES_2 + ] ), + new AreaLevel( 5, AreaChallengeType.VARIABLES, AreaModelCommonColorProfile.variablesIconBackgroundProperty, [ + AreaChallengeDescription.LEVEL_5_VARIABLES_1, + AreaChallengeDescription.LEVEL_5_VARIABLES_2 + ] ), + new AreaLevel( 6, AreaChallengeType.VARIABLES, AreaModelCommonColorProfile.variablesIconBackgroundProperty, [ + AreaChallengeDescription.LEVEL_6_VARIABLES_1 + ] ) + ], true ); +} - areaModelCommon.register( 'VariablesGameAreaModel', VariablesGameAreaModel ); +areaModelCommon.register( 'VariablesGameAreaModel', VariablesGameAreaModel ); - return inherit( GameAreaModel, VariablesGameAreaModel ); -} ); +inherit( GameAreaModel, VariablesGameAreaModel ); +export default VariablesGameAreaModel; \ No newline at end of file diff --git a/js/game/view/GameAreaDisplayNode.js b/js/game/view/GameAreaDisplayNode.js index decab225..36c74b6c 100644 --- a/js/game/view/GameAreaDisplayNode.js +++ b/js/game/view/GameAreaDisplayNode.js @@ -7,198 +7,195 @@ * * @author Jonathan Olson */ -define( require => { - 'use strict'; - - // modules - const areaModelCommon = require( 'AREA_MODEL_COMMON/areaModelCommon' ); - const AreaModelCommonColorProfile = require( 'AREA_MODEL_COMMON/common/view/AreaModelCommonColorProfile' ); - const AreaModelCommonConstants = require( 'AREA_MODEL_COMMON/common/AreaModelCommonConstants' ); - const DerivedProperty = require( 'AXON/DerivedProperty' ); - const Entry = require( 'AREA_MODEL_COMMON/game/model/Entry' ); - const EntryType = require( 'AREA_MODEL_COMMON/game/model/EntryType' ); - const GameEditableLabelNode = require( 'AREA_MODEL_COMMON/game/view/GameEditableLabelNode' ); - const GenericAreaDisplayNode = require( 'AREA_MODEL_COMMON/generic/view/GenericAreaDisplayNode' ); - const inherit = require( 'PHET_CORE/inherit' ); - const InputMethod = require( 'AREA_MODEL_COMMON/game/model/InputMethod' ); - const Node = require( 'SCENERY/nodes/Node' ); - const Orientation = require( 'PHET_CORE/Orientation' ); - const OrientationPair = require( 'AREA_MODEL_COMMON/common/model/OrientationPair' ); - const Property = require( 'AXON/Property' ); - const RangeLabelNode = require( 'AREA_MODEL_COMMON/common/view/RangeLabelNode' ); - const Rectangle = require( 'SCENERY/nodes/Rectangle' ); - const TermKeypadPanel = require( 'AREA_MODEL_COMMON/generic/view/TermKeypadPanel' ); - - // constants - const MAX_PARTITIONS = 3; // The maximum number of partitions for a specific dimension - - /** - * @constructor - * @extends {Node} - * - * @param {GameAreaDisplay} areaDisplay - * @param {Property.} activeEntryProperty - * @param {Property.} gameStateProperty - * @param {function} setActiveTerm - function( {Term|null} ) - Called when the value of the edited term should be set. - */ - function GameAreaDisplayNode( areaDisplay, activeEntryProperty, gameStateProperty, setActiveTerm ) { - const self = this; - - Node.call( this ); - - const singleOffset = AreaModelCommonConstants.AREA_SIZE * AreaModelCommonConstants.GENERIC_SINGLE_OFFSET; - const firstOffset = AreaModelCommonConstants.AREA_SIZE * AreaModelCommonConstants.GENERIC_FIRST_OFFSET; - const secondOffset = AreaModelCommonConstants.AREA_SIZE * AreaModelCommonConstants.GENERIC_SECOND_OFFSET; - const fullOffset = AreaModelCommonConstants.AREA_SIZE; - - // Background fill and stroke - this.addChild( new Rectangle( 0, 0, AreaModelCommonConstants.AREA_SIZE, AreaModelCommonConstants.AREA_SIZE, { - fill: AreaModelCommonColorProfile.areaBackgroundProperty, - stroke: AreaModelCommonColorProfile.areaBorderProperty - } ) ); - - this.addChild( GenericAreaDisplayNode.createPartitionLines( areaDisplay.layoutProperty, AreaModelCommonConstants.AREA_SIZE ) ); - - // Range views - const tickVariations = { - 1: [ 0, fullOffset ], - 2: [ 0, singleOffset, fullOffset ], - 3: [ 0, firstOffset, secondOffset, fullOffset ] - }; - Orientation.VALUES.forEach( function( orientation ) { - const colorProperty = AreaModelCommonColorProfile.genericColorProperties.get( orientation ); - const termListProperty = areaDisplay.totalProperties.get( orientation ); - const tickLocationsProperty = new DerivedProperty( [ areaDisplay.layoutProperty ], function( layout ) { - return tickVariations[ layout.getPartitionQuantity( orientation ) ]; - } ); - self.addChild( new RangeLabelNode( termListProperty, orientation, tickLocationsProperty, colorProperty, false ) ); - } ); - // {OrientationPair.>>} - The visual centers of all of the partitions. - // This duplicates some logic from GenericArea's coordinateRangeProperty handling, but here we need the full-length - // array every time. - const centerProperties = OrientationPair.create( function( orientation ) { - return [ - new DerivedProperty( [ areaDisplay.layoutProperty ], function( layout ) { - const partitionCount = layout.getPartitionQuantity( orientation ); - if ( partitionCount === 1 ) { - return fullOffset / 2; - } - else if ( partitionCount === 2 ) { - return singleOffset / 2; - } - else if ( partitionCount === 3 ) { - return firstOffset / 2; - } - } ), - new DerivedProperty( [ areaDisplay.layoutProperty ], function( layout ) { - const partitionCount = layout.getPartitionQuantity( orientation ); - if ( partitionCount === 2 ) { - return ( fullOffset + singleOffset ) / 2; - } - else if ( partitionCount === 3 ) { - return ( secondOffset + firstOffset ) / 2; - } - else { - return 0; // no need to position here, since this will never be used with a partitionCount of 1 - } - } ), - new Property( ( fullOffset + secondOffset ) / 2 ) - ]; - } ); +import DerivedProperty from '../../../../axon/js/DerivedProperty.js'; +import Property from '../../../../axon/js/Property.js'; +import inherit from '../../../../phet-core/js/inherit.js'; +import Orientation from '../../../../phet-core/js/Orientation.js'; +import Node from '../../../../scenery/js/nodes/Node.js'; +import Rectangle from '../../../../scenery/js/nodes/Rectangle.js'; +import areaModelCommon from '../../areaModelCommon.js'; +import AreaModelCommonConstants from '../../common/AreaModelCommonConstants.js'; +import OrientationPair from '../../common/model/OrientationPair.js'; +import AreaModelCommonColorProfile from '../../common/view/AreaModelCommonColorProfile.js'; +import RangeLabelNode from '../../common/view/RangeLabelNode.js'; +import GenericAreaDisplayNode from '../../generic/view/GenericAreaDisplayNode.js'; +import TermKeypadPanel from '../../generic/view/TermKeypadPanel.js'; +import Entry from '../model/Entry.js'; +import EntryType from '../model/EntryType.js'; +import InputMethod from '../model/InputMethod.js'; +import GameEditableLabelNode from './GameEditableLabelNode.js'; + +// constants +const MAX_PARTITIONS = 3; // The maximum number of partitions for a specific dimension - // Partition size labels - Orientation.VALUES.forEach( function( orientation ) { - _.range( 0, MAX_PARTITIONS ).forEach( function( partitionIndex ) { - const entryProperty = new DerivedProperty( - [ areaDisplay.partitionSizeEntriesProperties.get( orientation ) ], - function( entries ) { - return entries[ partitionIndex ] ? entries[ partitionIndex ] : new Entry( null ); - } ); - const colorProperty = AreaModelCommonColorProfile.genericColorProperties.get( orientation ); - - const label = new GameEditableLabelNode( { - entryProperty: entryProperty, - gameStateProperty: gameStateProperty, - activeEntryProperty: activeEntryProperty, - colorProperty: colorProperty, - allowExponentsProperty: areaDisplay.allowExponentsProperty, - orientation: orientation +/** + * @constructor + * @extends {Node} + * + * @param {GameAreaDisplay} areaDisplay + * @param {Property.} activeEntryProperty + * @param {Property.} gameStateProperty + * @param {function} setActiveTerm - function( {Term|null} ) - Called when the value of the edited term should be set. + */ +function GameAreaDisplayNode( areaDisplay, activeEntryProperty, gameStateProperty, setActiveTerm ) { + const self = this; + + Node.call( this ); + + const singleOffset = AreaModelCommonConstants.AREA_SIZE * AreaModelCommonConstants.GENERIC_SINGLE_OFFSET; + const firstOffset = AreaModelCommonConstants.AREA_SIZE * AreaModelCommonConstants.GENERIC_FIRST_OFFSET; + const secondOffset = AreaModelCommonConstants.AREA_SIZE * AreaModelCommonConstants.GENERIC_SECOND_OFFSET; + const fullOffset = AreaModelCommonConstants.AREA_SIZE; + + // Background fill and stroke + this.addChild( new Rectangle( 0, 0, AreaModelCommonConstants.AREA_SIZE, AreaModelCommonConstants.AREA_SIZE, { + fill: AreaModelCommonColorProfile.areaBackgroundProperty, + stroke: AreaModelCommonColorProfile.areaBorderProperty + } ) ); + + this.addChild( GenericAreaDisplayNode.createPartitionLines( areaDisplay.layoutProperty, AreaModelCommonConstants.AREA_SIZE ) ); + + // Range views + const tickVariations = { + 1: [ 0, fullOffset ], + 2: [ 0, singleOffset, fullOffset ], + 3: [ 0, firstOffset, secondOffset, fullOffset ] + }; + Orientation.VALUES.forEach( function( orientation ) { + const colorProperty = AreaModelCommonColorProfile.genericColorProperties.get( orientation ); + const termListProperty = areaDisplay.totalProperties.get( orientation ); + const tickLocationsProperty = new DerivedProperty( [ areaDisplay.layoutProperty ], function( layout ) { + return tickVariations[ layout.getPartitionQuantity( orientation ) ]; + } ); + self.addChild( new RangeLabelNode( termListProperty, orientation, tickLocationsProperty, colorProperty, false ) ); + } ); + + // {OrientationPair.>>} - The visual centers of all of the partitions. + // This duplicates some logic from GenericArea's coordinateRangeProperty handling, but here we need the full-length + // array every time. + const centerProperties = OrientationPair.create( function( orientation ) { + return [ + new DerivedProperty( [ areaDisplay.layoutProperty ], function( layout ) { + const partitionCount = layout.getPartitionQuantity( orientation ); + if ( partitionCount === 1 ) { + return fullOffset / 2; + } + else if ( partitionCount === 2 ) { + return singleOffset / 2; + } + else if ( partitionCount === 3 ) { + return firstOffset / 2; + } + } ), + new DerivedProperty( [ areaDisplay.layoutProperty ], function( layout ) { + const partitionCount = layout.getPartitionQuantity( orientation ); + if ( partitionCount === 2 ) { + return ( fullOffset + singleOffset ) / 2; + } + else if ( partitionCount === 3 ) { + return ( secondOffset + firstOffset ) / 2; + } + else { + return 0; // no need to position here, since this will never be used with a partitionCount of 1 + } + } ), + new Property( ( fullOffset + secondOffset ) / 2 ) + ]; + } ); + + // Partition size labels + Orientation.VALUES.forEach( function( orientation ) { + _.range( 0, MAX_PARTITIONS ).forEach( function( partitionIndex ) { + const entryProperty = new DerivedProperty( + [ areaDisplay.partitionSizeEntriesProperties.get( orientation ) ], + function( entries ) { + return entries[ partitionIndex ] ? entries[ partitionIndex ] : new Entry( null ); } ); + const colorProperty = AreaModelCommonColorProfile.genericColorProperties.get( orientation ); - label[ orientation.opposite.coordinate ] = AreaModelCommonConstants.PARTITION_OFFSET.get( orientation ); - self.addChild( label ); - - centerProperties.get( orientation )[ partitionIndex ].link( function( location ) { - label[ orientation.coordinate ] = location; - } ); + const label = new GameEditableLabelNode( { + entryProperty: entryProperty, + gameStateProperty: gameStateProperty, + activeEntryProperty: activeEntryProperty, + colorProperty: colorProperty, + allowExponentsProperty: areaDisplay.allowExponentsProperty, + orientation: orientation } ); - } ); - // Labels for each partitioned area - _.range( 0, MAX_PARTITIONS ).forEach( function( horizontalIndex ) { - _.range( 0, MAX_PARTITIONS ).forEach( function( verticalIndex ) { - const entryProperty = new DerivedProperty( [ areaDisplay.partialProductEntriesProperty ], function( values ) { - return ( values[ verticalIndex ] && values[ verticalIndex ][ horizontalIndex ] ) - ? values[ verticalIndex ][ horizontalIndex ] - : new Entry( null ); - } ); + label[ orientation.opposite.coordinate ] = AreaModelCommonConstants.PARTITION_OFFSET.get( orientation ); + self.addChild( label ); - const colorProperty = new DerivedProperty( [ - entryProperty, - AreaModelCommonColorProfile.dynamicPartialProductProperty, - AreaModelCommonColorProfile.fixedPartialProductProperty - ], function( entry, dynamicColor, fixedColor ) { - if ( entry && entry.type === EntryType.DYNAMIC ) { - return dynamicColor; - } - else { - return fixedColor; - } - } ); + centerProperties.get( orientation )[ partitionIndex ].link( function( location ) { + label[ orientation.coordinate ] = location; + } ); + } ); + } ); + + // Labels for each partitioned area + _.range( 0, MAX_PARTITIONS ).forEach( function( horizontalIndex ) { + _.range( 0, MAX_PARTITIONS ).forEach( function( verticalIndex ) { + const entryProperty = new DerivedProperty( [ areaDisplay.partialProductEntriesProperty ], function( values ) { + return ( values[ verticalIndex ] && values[ verticalIndex ][ horizontalIndex ] ) + ? values[ verticalIndex ][ horizontalIndex ] + : new Entry( null ); + } ); - const label = new GameEditableLabelNode( { - entryProperty: entryProperty, - gameStateProperty: gameStateProperty, - activeEntryProperty: activeEntryProperty, - colorProperty: colorProperty, - allowExponentsProperty: areaDisplay.allowExponentsProperty, - orientation: Orientation.VERTICAL, - labelFont: AreaModelCommonConstants.GAME_PARTIAL_PRODUCT_LABEL_FONT, - editFont: AreaModelCommonConstants.GAME_PARTIAL_PRODUCT_EDIT_FONT - } ); - self.addChild( label ); + const colorProperty = new DerivedProperty( [ + entryProperty, + AreaModelCommonColorProfile.dynamicPartialProductProperty, + AreaModelCommonColorProfile.fixedPartialProductProperty + ], function( entry, dynamicColor, fixedColor ) { + if ( entry && entry.type === EntryType.DYNAMIC ) { + return dynamicColor; + } + else { + return fixedColor; + } + } ); - centerProperties.horizontal[ horizontalIndex ].linkAttribute( label, 'x' ); - centerProperties.vertical[ verticalIndex ].linkAttribute( label, 'y' ); + const label = new GameEditableLabelNode( { + entryProperty: entryProperty, + gameStateProperty: gameStateProperty, + activeEntryProperty: activeEntryProperty, + colorProperty: colorProperty, + allowExponentsProperty: areaDisplay.allowExponentsProperty, + orientation: Orientation.VERTICAL, + labelFont: AreaModelCommonConstants.GAME_PARTIAL_PRODUCT_LABEL_FONT, + editFont: AreaModelCommonConstants.GAME_PARTIAL_PRODUCT_EDIT_FONT } ); - } ); + self.addChild( label ); - const digitsProperty = new DerivedProperty( [ activeEntryProperty ], function( entry ) { - return entry ? entry.digits : 1; + centerProperties.horizontal[ horizontalIndex ].linkAttribute( label, 'x' ); + centerProperties.vertical[ verticalIndex ].linkAttribute( label, 'y' ); } ); + } ); - const keypadOptions = { - // padding constant allows it to fit between the area and the other panels - x: AreaModelCommonConstants.AREA_SIZE + AreaModelCommonConstants.KEYPAD_LEFT_PADDING, - top: 0 - }; - const noExponentKeypadPanel = new TermKeypadPanel( digitsProperty, false, false, setActiveTerm, keypadOptions ); - const exponentKeypadPanel = new TermKeypadPanel( digitsProperty, true, true, setActiveTerm, keypadOptions ); + const digitsProperty = new DerivedProperty( [ activeEntryProperty ], function( entry ) { + return entry ? entry.digits : 1; + } ); - this.addChild( noExponentKeypadPanel ); - this.addChild( exponentKeypadPanel ); + const keypadOptions = { + // padding constant allows it to fit between the area and the other panels + x: AreaModelCommonConstants.AREA_SIZE + AreaModelCommonConstants.KEYPAD_LEFT_PADDING, + top: 0 + }; + const noExponentKeypadPanel = new TermKeypadPanel( digitsProperty, false, false, setActiveTerm, keypadOptions ); + const exponentKeypadPanel = new TermKeypadPanel( digitsProperty, true, true, setActiveTerm, keypadOptions ); - activeEntryProperty.link( function( newEntry ) { - noExponentKeypadPanel.clear(); - exponentKeypadPanel.clear(); + this.addChild( noExponentKeypadPanel ); + this.addChild( exponentKeypadPanel ); - noExponentKeypadPanel.visible = newEntry !== null && newEntry.inputMethod === InputMethod.CONSTANT; - exponentKeypadPanel.visible = newEntry !== null && newEntry.inputMethod === InputMethod.TERM; - } ); - } + activeEntryProperty.link( function( newEntry ) { + noExponentKeypadPanel.clear(); + exponentKeypadPanel.clear(); + + noExponentKeypadPanel.visible = newEntry !== null && newEntry.inputMethod === InputMethod.CONSTANT; + exponentKeypadPanel.visible = newEntry !== null && newEntry.inputMethod === InputMethod.TERM; + } ); +} - areaModelCommon.register( 'GameAreaDisplayNode', GameAreaDisplayNode ); +areaModelCommon.register( 'GameAreaDisplayNode', GameAreaDisplayNode ); - return inherit( Node, GameAreaDisplayNode ); -} ); +inherit( Node, GameAreaDisplayNode ); +export default GameAreaDisplayNode; \ No newline at end of file diff --git a/js/game/view/GameAreaScreenView.js b/js/game/view/GameAreaScreenView.js index 3be36f68..ba16fd3c 100644 --- a/js/game/view/GameAreaScreenView.js +++ b/js/game/view/GameAreaScreenView.js @@ -7,543 +7,539 @@ * * @author Jonathan Olson */ -define( require => { - 'use strict'; - - // modules - const PhetFont = require( 'SCENERY_PHET/PhetFont' ); - const AlignBox = require( 'SCENERY/nodes/AlignBox' ); - const areaModelCommon = require( 'AREA_MODEL_COMMON/areaModelCommon' ); - const AreaModelCommonColorProfile = require( 'AREA_MODEL_COMMON/common/view/AreaModelCommonColorProfile' ); - const AreaModelCommonConstants = require( 'AREA_MODEL_COMMON/common/AreaModelCommonConstants' ); - const AreaModelCommonGlobals = require( 'AREA_MODEL_COMMON/common/AreaModelCommonGlobals' ); - const DerivedProperty = require( 'AXON/DerivedProperty' ); - const DynamicProperty = require( 'AXON/DynamicProperty' ); - const Easing = require( 'TWIXT/Easing' ); - const Entry = require( 'AREA_MODEL_COMMON/game/model/Entry' ); - const EntryDisplayType = require( 'AREA_MODEL_COMMON/game/model/EntryDisplayType' ); - const FaceNode = require( 'SCENERY_PHET/FaceNode' ); - const FaceWithPointsNode = require( 'SCENERY_PHET/FaceWithPointsNode' ); - const FiniteStatusBar = require( 'VEGAS/FiniteStatusBar' ); - const GameAreaDisplay = require( 'AREA_MODEL_COMMON/game/model/GameAreaDisplay' ); - const GameAreaDisplayNode = require( 'AREA_MODEL_COMMON/game/view/GameAreaDisplayNode' ); - const GameAreaModel = require( 'AREA_MODEL_COMMON/game/model/GameAreaModel' ); - const GameAudio = require( 'AREA_MODEL_COMMON/game/view/GameAudio' ); - const GameEditableLabelNode = require( 'AREA_MODEL_COMMON/game/view/GameEditableLabelNode' ); - const GameState = require( 'AREA_MODEL_COMMON/game/model/GameState' ); - const GenericFactorsNode = require( 'AREA_MODEL_COMMON/generic/view/GenericFactorsNode' ); - const HBox = require( 'SCENERY/nodes/HBox' ); - const Image = require( 'SCENERY/nodes/Image' ); - const inherit = require( 'PHET_CORE/inherit' ); - const LevelCompletedNode = require( 'VEGAS/LevelCompletedNode' ); - const LevelSelectionButton = require( 'VEGAS/LevelSelectionButton' ); - const Node = require( 'SCENERY/nodes/Node' ); - const NumberProperty = require( 'AXON/NumberProperty' ); - const Orientation = require( 'PHET_CORE/Orientation' ); - const Panel = require( 'SUN/Panel' ); - const PolynomialEditNode = require( 'AREA_MODEL_COMMON/game/view/PolynomialEditNode' ); - const Property = require( 'AXON/Property' ); - const RectangularPushButton = require( 'SUN/buttons/RectangularPushButton' ); - const ResetAllButton = require( 'SCENERY_PHET/buttons/ResetAllButton' ); - const RewardNode = require( 'VEGAS/RewardNode' ); - const RichText = require( 'SCENERY/nodes/RichText' ); - const ScoreDisplayLabeledStars = require( 'VEGAS/ScoreDisplayLabeledStars' ); - const ScoreDisplayStars = require( 'VEGAS/ScoreDisplayStars' ); - const ScreenView = require( 'JOIST/ScreenView' ); - const StarNode = require( 'SCENERY_PHET/StarNode' ); - const Term = require( 'AREA_MODEL_COMMON/common/model/Term' ); - const Text = require( 'SCENERY/nodes/Text' ); - const TransitionNode = require( 'TWIXT/TransitionNode' ); - const VBox = require( 'SCENERY/nodes/VBox' ); - - // strings - const checkString = require( 'string!VEGAS/check' ); - const chooseYourLevelString = require( 'string!VEGAS/chooseYourLevel' ); - const dimensionsString = require( 'string!AREA_MODEL_COMMON/dimensions' ); - const nextString = require( 'string!VEGAS/next' ); - const showAnswerString = require( 'string!VEGAS/showAnswer' ); - const totalAreaOfModelString = require( 'string!AREA_MODEL_COMMON/totalAreaOfModel' ); - const tryAgainString = require( 'string!VEGAS/tryAgain' ); - - // images - const level1IconImage = require( 'mipmap!AREA_MODEL_COMMON/level-1-icon.png' ); - const level2IconImage = require( 'mipmap!AREA_MODEL_COMMON/level-2-icon.png' ); - const level3IconImage = require( 'mipmap!AREA_MODEL_COMMON/level-3-icon.png' ); - const level4IconImage = require( 'mipmap!AREA_MODEL_COMMON/level-4-icon.png' ); - const level5IconImage = require( 'mipmap!AREA_MODEL_COMMON/level-5-icon.png' ); - const level6IconImage = require( 'mipmap!AREA_MODEL_COMMON/level-6-icon.png' ); - - // constants - const LEVEL_ICON_IMAGES = [ - level1IconImage, - level2IconImage, - level3IconImage, - level4IconImage, - level5IconImage, - level6IconImage - ]; - /** - * @constructor - * @extends {ScreenView} - * - * @param {GameAreaModel} model - */ - function GameAreaScreenView( model ) { - assert && assert( model instanceof GameAreaModel ); +import DerivedProperty from '../../../../axon/js/DerivedProperty.js'; +import DynamicProperty from '../../../../axon/js/DynamicProperty.js'; +import NumberProperty from '../../../../axon/js/NumberProperty.js'; +import Property from '../../../../axon/js/Property.js'; +import ScreenView from '../../../../joist/js/ScreenView.js'; +import inherit from '../../../../phet-core/js/inherit.js'; +import Orientation from '../../../../phet-core/js/Orientation.js'; +import ResetAllButton from '../../../../scenery-phet/js/buttons/ResetAllButton.js'; +import FaceNode from '../../../../scenery-phet/js/FaceNode.js'; +import FaceWithPointsNode from '../../../../scenery-phet/js/FaceWithPointsNode.js'; +import PhetFont from '../../../../scenery-phet/js/PhetFont.js'; +import StarNode from '../../../../scenery-phet/js/StarNode.js'; +import AlignBox from '../../../../scenery/js/nodes/AlignBox.js'; +import HBox from '../../../../scenery/js/nodes/HBox.js'; +import Image from '../../../../scenery/js/nodes/Image.js'; +import Node from '../../../../scenery/js/nodes/Node.js'; +import RichText from '../../../../scenery/js/nodes/RichText.js'; +import Text from '../../../../scenery/js/nodes/Text.js'; +import VBox from '../../../../scenery/js/nodes/VBox.js'; +import RectangularPushButton from '../../../../sun/js/buttons/RectangularPushButton.js'; +import Panel from '../../../../sun/js/Panel.js'; +import Easing from '../../../../twixt/js/Easing.js'; +import TransitionNode from '../../../../twixt/js/TransitionNode.js'; +import FiniteStatusBar from '../../../../vegas/js/FiniteStatusBar.js'; +import LevelCompletedNode from '../../../../vegas/js/LevelCompletedNode.js'; +import LevelSelectionButton from '../../../../vegas/js/LevelSelectionButton.js'; +import RewardNode from '../../../../vegas/js/RewardNode.js'; +import ScoreDisplayLabeledStars from '../../../../vegas/js/ScoreDisplayLabeledStars.js'; +import ScoreDisplayStars from '../../../../vegas/js/ScoreDisplayStars.js'; +import vegasStrings from '../../../../vegas/js/vegas-strings.js'; +import level1IconImage from '../../../mipmaps/level-1-icon_png.js'; +import level2IconImage from '../../../mipmaps/level-2-icon_png.js'; +import level3IconImage from '../../../mipmaps/level-3-icon_png.js'; +import level4IconImage from '../../../mipmaps/level-4-icon_png.js'; +import level5IconImage from '../../../mipmaps/level-5-icon_png.js'; +import level6IconImage from '../../../mipmaps/level-6-icon_png.js'; +import areaModelCommonStrings from '../../area-model-common-strings.js'; +import areaModelCommon from '../../areaModelCommon.js'; +import AreaModelCommonConstants from '../../common/AreaModelCommonConstants.js'; +import AreaModelCommonGlobals from '../../common/AreaModelCommonGlobals.js'; +import Term from '../../common/model/Term.js'; +import AreaModelCommonColorProfile from '../../common/view/AreaModelCommonColorProfile.js'; +import GenericFactorsNode from '../../generic/view/GenericFactorsNode.js'; +import Entry from '../model/Entry.js'; +import EntryDisplayType from '../model/EntryDisplayType.js'; +import GameAreaDisplay from '../model/GameAreaDisplay.js'; +import GameAreaModel from '../model/GameAreaModel.js'; +import GameState from '../model/GameState.js'; +import GameAreaDisplayNode from './GameAreaDisplayNode.js'; +import GameAudio from './GameAudio.js'; +import GameEditableLabelNode from './GameEditableLabelNode.js'; +import PolynomialEditNode from './PolynomialEditNode.js'; + +const checkString = vegasStrings.check; +const chooseYourLevelString = vegasStrings.chooseYourLevel; +const dimensionsString = areaModelCommonStrings.dimensions; +const nextString = vegasStrings.next; +const showAnswerString = vegasStrings.showAnswer; +const totalAreaOfModelString = areaModelCommonStrings.totalAreaOfModel; +const tryAgainString = vegasStrings.tryAgain; + + +// constants +const LEVEL_ICON_IMAGES = [ + level1IconImage, + level2IconImage, + level3IconImage, + level4IconImage, + level5IconImage, + level6IconImage +]; - const self = this; - - ScreenView.call( this ); +/** + * @constructor + * @extends {ScreenView} + * + * @param {GameAreaModel} model + */ +function GameAreaScreenView( model ) { + assert && assert( model instanceof GameAreaModel ); - // @private {Node} - The "left" half of the sliding layer, displayed first - this.levelSelectionLayer = new Node(); + const self = this; - // @private {Node} - The "right" half of the sliding layer, will slide into view when the user selects a level - this.challengeLayer = new Node(); + ScreenView.call( this ); - // @private {GameAudio} - Responsible for all audio - this.audio = new GameAudio( model ); + // @private {Node} - The "left" half of the sliding layer, displayed first + this.levelSelectionLayer = new Node(); - // @private {TransitionNode} - this.transitionNode = new TransitionNode( this.visibleBoundsProperty, { - content: this.levelSelectionLayer, - useBoundsClip: false, // better performance without the clipping - cachedNodes: [ this.levelSelectionLayer, this.challengeLayer ] - } ); - this.addChild( this.transitionNode ); - model.currentLevelProperty.lazyLink( function( level ) { - if ( level ) { - self.transitionNode.slideLeftTo( self.challengeLayer, { - duration: 0.4, - targetOptions: { - easing: Easing.QUADRATIC_IN_OUT - } - } ); - } - else { - self.transitionNode.dissolveTo( self.levelSelectionLayer, { - duration: 0.4, - gamma: 2.2, - targetOptions: { - easing: Easing.LINEAR - } - } ); - } - } ); + // @private {Node} - The "right" half of the sliding layer, will slide into view when the user selects a level + this.challengeLayer = new Node(); - const levelIcons = LEVEL_ICON_IMAGES.map( function( iconImage ) { - return new Image( iconImage ); - } ); + // @private {GameAudio} - Responsible for all audio + this.audio = new GameAudio( model ); - const buttonSpacing = 30; - const levelButtons = model.levels.map( function( level, index ) { - return new LevelSelectionButton( levelIcons[ index ], level.scoreProperty, { - scoreDisplayConstructor: ScoreDisplayStars, - scoreDisplayOptions: { - numberOfStars: AreaModelCommonConstants.NUM_CHALLENGES, - perfectScore: AreaModelCommonConstants.PERFECT_SCORE - }, - listener: function() { - model.selectLevel( level ); - }, - baseColor: level.colorProperty + // @private {TransitionNode} + this.transitionNode = new TransitionNode( this.visibleBoundsProperty, { + content: this.levelSelectionLayer, + useBoundsClip: false, // better performance without the clipping + cachedNodes: [ this.levelSelectionLayer, this.challengeLayer ] + } ); + this.addChild( this.transitionNode ); + model.currentLevelProperty.lazyLink( function( level ) { + if ( level ) { + self.transitionNode.slideLeftTo( self.challengeLayer, { + duration: 0.4, + targetOptions: { + easing: Easing.QUADRATIC_IN_OUT + } } ); - } ); + } + else { + self.transitionNode.dissolveTo( self.levelSelectionLayer, { + duration: 0.4, + gamma: 2.2, + targetOptions: { + easing: Easing.LINEAR + } + } ); + } + } ); - this.levelSelectionLayer.addChild( new VBox( { - children: _.chunk( levelButtons, 3 ).map( function( children ) { - return new HBox( { - children: children, - spacing: buttonSpacing - } ); - } ), - spacing: buttonSpacing, - center: this.layoutBounds.center - } ) ); - - this.levelSelectionLayer.addChild( new Text( chooseYourLevelString, { - centerX: this.layoutBounds.centerX, - centerY: ( this.layoutBounds.top + this.levelSelectionLayer.top ) / 2, - font: new PhetFont( 30 ) - } ) ); - - // Status bar - let lastKnownLevel = null; - // Create a property that holds the "last known" level, so that we don't change the view when we are switching - // away from the current level back to the level selection. - const lastLevelProperty = new DerivedProperty( [ model.currentLevelProperty ], function( level ) { - level = level || lastKnownLevel; - lastKnownLevel = level; - return level; - } ); - const scoreProperty = new DynamicProperty( lastLevelProperty, { - derive: 'scoreProperty' - } ); - const statusBar = new FiniteStatusBar( this.layoutBounds, this.visibleBoundsProperty, scoreProperty, { - challengeIndexProperty: new DynamicProperty( lastLevelProperty, { - derive: 'challengeIndexProperty', - defaultValue: 1 - } ), - numberOfChallengesProperty: new NumberProperty( AreaModelCommonConstants.NUM_CHALLENGES ), - levelProperty: new DerivedProperty( [ lastLevelProperty ], function( level ) { - return level ? level.number : 1; - } ), - scoreDisplayConstructor: ScoreDisplayLabeledStars, + const levelIcons = LEVEL_ICON_IMAGES.map( function( iconImage ) { + return new Image( iconImage ); + } ); + + const buttonSpacing = 30; + const levelButtons = model.levels.map( function( level, index ) { + return new LevelSelectionButton( levelIcons[ index ], level.scoreProperty, { + scoreDisplayConstructor: ScoreDisplayStars, scoreDisplayOptions: { numberOfStars: AreaModelCommonConstants.NUM_CHALLENGES, perfectScore: AreaModelCommonConstants.PERFECT_SCORE }, - startOverButtonOptions: { - listener: function() { - // Reset the level on "Start Over", see https://github.com/phetsims/area-model-common/issues/87 - model.currentLevelProperty.value.startOver(); - model.currentLevelProperty.value = null; - } - }, - font: AreaModelCommonConstants.GAME_STATUS_BAR_NON_BOLD_FONT, - levelTextOptions: { - font: AreaModelCommonConstants.GAME_STATUS_BAR_BOLD_FONT + listener: function() { + model.selectLevel( level ); }, - floatToTop: true, - barFill: new DynamicProperty( lastLevelProperty, { - derive: 'colorProperty', - defaultValue: 'black' - } ) - } ); - this.challengeLayer.addChild( statusBar ); - - // Prompt - const promptText = new Text( ' ', { - font: AreaModelCommonConstants.GAME_STATUS_BAR_PROMPT_FONT, - pickable: false, - maxWidth: 600, - top: this.layoutBounds.top + statusBar.height + 20 - } ); - this.challengeLayer.addChild( promptText ); - new DynamicProperty( model.currentLevelProperty, { - derive: 'currentChallengeProperty' - } ).link( function( challenge ) { - // Could be null - if ( challenge ) { - promptText.text = challenge.description.getPromptString(); - // Center around the area's center. - promptText.centerX = self.layoutBounds.left + AreaModelCommonConstants.GAME_AREA_OFFSET.x + AreaModelCommonConstants.AREA_SIZE / 2; - // Don't let it go off the left side of the screen - promptText.left = Math.max( promptText.left, self.layoutBounds.left + 20 ); - } + baseColor: level.colorProperty } ); + } ); - // Reset All button - const resetAllButton = new ResetAllButton( { + this.levelSelectionLayer.addChild( new VBox( { + children: _.chunk( levelButtons, 3 ).map( function( children ) { + return new HBox( { + children: children, + spacing: buttonSpacing + } ); + } ), + spacing: buttonSpacing, + center: this.layoutBounds.center + } ) ); + + this.levelSelectionLayer.addChild( new Text( chooseYourLevelString, { + centerX: this.layoutBounds.centerX, + centerY: ( this.layoutBounds.top + this.levelSelectionLayer.top ) / 2, + font: new PhetFont( 30 ) + } ) ); + + // Status bar + let lastKnownLevel = null; + // Create a property that holds the "last known" level, so that we don't change the view when we are switching + // away from the current level back to the level selection. + const lastLevelProperty = new DerivedProperty( [ model.currentLevelProperty ], function( level ) { + level = level || lastKnownLevel; + lastKnownLevel = level; + return level; + } ); + const scoreProperty = new DynamicProperty( lastLevelProperty, { + derive: 'scoreProperty' + } ); + const statusBar = new FiniteStatusBar( this.layoutBounds, this.visibleBoundsProperty, scoreProperty, { + challengeIndexProperty: new DynamicProperty( lastLevelProperty, { + derive: 'challengeIndexProperty', + defaultValue: 1 + } ), + numberOfChallengesProperty: new NumberProperty( AreaModelCommonConstants.NUM_CHALLENGES ), + levelProperty: new DerivedProperty( [ lastLevelProperty ], function( level ) { + return level ? level.number : 1; + } ), + scoreDisplayConstructor: ScoreDisplayLabeledStars, + scoreDisplayOptions: { + numberOfStars: AreaModelCommonConstants.NUM_CHALLENGES, + perfectScore: AreaModelCommonConstants.PERFECT_SCORE + }, + startOverButtonOptions: { listener: function() { - model.reset(); - }, - right: this.layoutBounds.right - AreaModelCommonConstants.LAYOUT_SPACING, - bottom: this.layoutBounds.bottom - AreaModelCommonConstants.LAYOUT_SPACING - } ); - this.levelSelectionLayer.addChild( resetAllButton ); + // Reset the level on "Start Over", see https://github.com/phetsims/area-model-common/issues/87 + model.currentLevelProperty.value.startOver(); + model.currentLevelProperty.value = null; + } + }, + font: AreaModelCommonConstants.GAME_STATUS_BAR_NON_BOLD_FONT, + levelTextOptions: { + font: AreaModelCommonConstants.GAME_STATUS_BAR_BOLD_FONT + }, + floatToTop: true, + barFill: new DynamicProperty( lastLevelProperty, { + derive: 'colorProperty', + defaultValue: 'black' + } ) + } ); + this.challengeLayer.addChild( statusBar ); + + // Prompt + const promptText = new Text( ' ', { + font: AreaModelCommonConstants.GAME_STATUS_BAR_PROMPT_FONT, + pickable: false, + maxWidth: 600, + top: this.layoutBounds.top + statusBar.height + 20 + } ); + this.challengeLayer.addChild( promptText ); + new DynamicProperty( model.currentLevelProperty, { + derive: 'currentChallengeProperty' + } ).link( function( challenge ) { + // Could be null + if ( challenge ) { + promptText.text = challenge.description.getPromptString(); + // Center around the area's center. + promptText.centerX = self.layoutBounds.left + AreaModelCommonConstants.GAME_AREA_OFFSET.x + AreaModelCommonConstants.AREA_SIZE / 2; + // Don't let it go off the left side of the screen + promptText.left = Math.max( promptText.left, self.layoutBounds.left + 20 ); + } + } ); - /*---------------------------------------------------------------------------* - * Area display - *----------------------------------------------------------------------------*/ + // Reset All button + const resetAllButton = new ResetAllButton( { + listener: function() { + model.reset(); + }, + right: this.layoutBounds.right - AreaModelCommonConstants.LAYOUT_SPACING, + bottom: this.layoutBounds.bottom - AreaModelCommonConstants.LAYOUT_SPACING + } ); + this.levelSelectionLayer.addChild( resetAllButton ); - // @private {GameAreaDisplay} - this.areaDisplay = new GameAreaDisplay( model.currentChallengeProperty ); + /*---------------------------------------------------------------------------* + * Area display + *----------------------------------------------------------------------------*/ - const gameAreaNode = new GameAreaDisplayNode( this.areaDisplay, model.activeEntryProperty, model.stateProperty, function( term ) { - model.setActiveTerm( term ); - } ); - this.challengeLayer.addChild( gameAreaNode ); - gameAreaNode.translation = this.layoutBounds.leftTop.plus( AreaModelCommonConstants.GAME_AREA_OFFSET ); + // @private {GameAreaDisplay} + this.areaDisplay = new GameAreaDisplay( model.currentChallengeProperty ); - /*---------------------------------------------------------------------------* - * Panels - *----------------------------------------------------------------------------*/ + const gameAreaNode = new GameAreaDisplayNode( this.areaDisplay, model.activeEntryProperty, model.stateProperty, function( term ) { + model.setActiveTerm( term ); + } ); + this.challengeLayer.addChild( gameAreaNode ); + gameAreaNode.translation = this.layoutBounds.leftTop.plus( AreaModelCommonConstants.GAME_AREA_OFFSET ); - const panelAlignGroup = AreaModelCommonGlobals.panelAlignGroup; + /*---------------------------------------------------------------------------* + * Panels + *----------------------------------------------------------------------------*/ - const factorsNode = new GenericFactorsNode( this.areaDisplay.totalProperties, this.areaDisplay.allowExponentsProperty ); - const factorsContent = this.createPanel( dimensionsString, panelAlignGroup, factorsNode ); + const panelAlignGroup = AreaModelCommonGlobals.panelAlignGroup; - // If we have a polynomial, don't use this editable property (use the polynomial editor component instead) - const totalTermEntryProperty = new DerivedProperty( [ this.areaDisplay.totalEntriesProperty ], function( totalEntries ) { - return totalEntries.length === 1 ? totalEntries[ 0 ] : new Entry( null ); - } ); + const factorsNode = new GenericFactorsNode( this.areaDisplay.totalProperties, this.areaDisplay.allowExponentsProperty ); + const factorsContent = this.createPanel( dimensionsString, panelAlignGroup, factorsNode ); - const totalNode = new GameEditableLabelNode( { - entryProperty: totalTermEntryProperty, - gameStateProperty: model.stateProperty, - activeEntryProperty: model.activeEntryProperty, - colorProperty: AreaModelCommonColorProfile.totalEditableProperty, - allowExponentsProperty: this.areaDisplay.allowExponentsProperty, - orientation: Orientation.HORIZONTAL, - labelFont: AreaModelCommonConstants.GAME_TOTAL_FONT, - editFont: AreaModelCommonConstants.GAME_TOTAL_FONT - } ); - const polynomialEditNode = new PolynomialEditNode( this.areaDisplay.totalProperty, this.areaDisplay.totalEntriesProperty, function() { - if ( model.stateProperty.value === GameState.WRONG_FIRST_ANSWER ) { - model.stateProperty.value = GameState.SECOND_ATTEMPT; - } - } ); - const polynomialReadoutText = new RichText( '?', { - font: AreaModelCommonConstants.TOTAL_AREA_LABEL_FONT, - maxWidth: AreaModelCommonConstants.PANEL_INTERIOR_MAX - } ); - this.areaDisplay.totalProperty.link( function( total ) { - if ( total ) { - polynomialReadoutText.text = total.toRichString( false ); - } - } ); + // If we have a polynomial, don't use this editable property (use the polynomial editor component instead) + const totalTermEntryProperty = new DerivedProperty( [ this.areaDisplay.totalEntriesProperty ], function( totalEntries ) { + return totalEntries.length === 1 ? totalEntries[ 0 ] : new Entry( null ); + } ); - const totalContainer = new Node(); - Property.multilink( - [ this.areaDisplay.totalEntriesProperty, model.stateProperty ], - function( totalEntries, gameState ) { - if ( totalEntries.length > 1 ) { - if ( totalEntries[ 0 ].displayType === EntryDisplayType.EDITABLE && - gameState !== GameState.CORRECT_ANSWER && - gameState !== GameState.SHOW_SOLUTION ) { - totalContainer.children = [ polynomialEditNode ]; - } - else { - totalContainer.children = [ polynomialReadoutText ]; - } + const totalNode = new GameEditableLabelNode( { + entryProperty: totalTermEntryProperty, + gameStateProperty: model.stateProperty, + activeEntryProperty: model.activeEntryProperty, + colorProperty: AreaModelCommonColorProfile.totalEditableProperty, + allowExponentsProperty: this.areaDisplay.allowExponentsProperty, + orientation: Orientation.HORIZONTAL, + labelFont: AreaModelCommonConstants.GAME_TOTAL_FONT, + editFont: AreaModelCommonConstants.GAME_TOTAL_FONT + } ); + const polynomialEditNode = new PolynomialEditNode( this.areaDisplay.totalProperty, this.areaDisplay.totalEntriesProperty, function() { + if ( model.stateProperty.value === GameState.WRONG_FIRST_ANSWER ) { + model.stateProperty.value = GameState.SECOND_ATTEMPT; + } + } ); + const polynomialReadoutText = new RichText( '?', { + font: AreaModelCommonConstants.TOTAL_AREA_LABEL_FONT, + maxWidth: AreaModelCommonConstants.PANEL_INTERIOR_MAX + } ); + this.areaDisplay.totalProperty.link( function( total ) { + if ( total ) { + polynomialReadoutText.text = total.toRichString( false ); + } + } ); + + const totalContainer = new Node(); + Property.multilink( + [ this.areaDisplay.totalEntriesProperty, model.stateProperty ], + function( totalEntries, gameState ) { + if ( totalEntries.length > 1 ) { + if ( totalEntries[ 0 ].displayType === EntryDisplayType.EDITABLE && + gameState !== GameState.CORRECT_ANSWER && + gameState !== GameState.SHOW_SOLUTION ) { + totalContainer.children = [ polynomialEditNode ]; } else { - totalContainer.children = [ totalNode ]; + totalContainer.children = [ polynomialReadoutText ]; } - } ); - - const productContent = this.createPanel( totalAreaOfModelString, panelAlignGroup, totalContainer ); - - const panelBox = new VBox( { - children: [ - factorsContent, - productContent - ], - spacing: AreaModelCommonConstants.LAYOUT_SPACING + } + else { + totalContainer.children = [ totalNode ]; + } } ); - this.challengeLayer.addChild( new AlignBox( panelBox, { - alignBounds: this.layoutBounds, - xAlign: 'right', - yAlign: 'top', - topMargin: gameAreaNode.y, - rightMargin: AreaModelCommonConstants.LAYOUT_SPACING - } ) ); - - /** - * Creates a game-style button that may be enabled via a property - * - * @param {string} label - * @param {function} listener - The callback for when the button is pressed - * @param {Property.} [enabledProperty] - */ - function createGameButton( label, listener, enabledProperty ) { - const button = new RectangularPushButton( { - content: new Text( label, { - font: AreaModelCommonConstants.BUTTON_FONT, - maxWidth: 200 - } ), - touchAreaXDilation: 10, - touchAreaYDilation: 10, - listener: listener, - baseColor: AreaModelCommonColorProfile.gameButtonBackgroundProperty, - centerX: panelBox.centerX, - top: panelBox.bottom + 80 - } ); - enabledProperty && enabledProperty.link( function( enabled ) { - button.enabled = enabled; - } ); - self.challengeLayer.addChild( button ); - return button; - } - const checkButton = createGameButton( checkString, function() { - model.check(); - }, model.allowCheckingProperty ); + const productContent = this.createPanel( totalAreaOfModelString, panelAlignGroup, totalContainer ); - const tryAgainButton = createGameButton( tryAgainString, function() { - model.tryAgain(); - } ); + const panelBox = new VBox( { + children: [ + factorsContent, + productContent + ], + spacing: AreaModelCommonConstants.LAYOUT_SPACING + } ); + this.challengeLayer.addChild( new AlignBox( panelBox, { + alignBounds: this.layoutBounds, + xAlign: 'right', + yAlign: 'top', + topMargin: gameAreaNode.y, + rightMargin: AreaModelCommonConstants.LAYOUT_SPACING + } ) ); - const nextButton = createGameButton( nextString, function() { - model.next(); + /** + * Creates a game-style button that may be enabled via a property + * + * @param {string} label + * @param {function} listener - The callback for when the button is pressed + * @param {Property.} [enabledProperty] + */ + function createGameButton( label, listener, enabledProperty ) { + const button = new RectangularPushButton( { + content: new Text( label, { + font: AreaModelCommonConstants.BUTTON_FONT, + maxWidth: 200 + } ), + touchAreaXDilation: 10, + touchAreaYDilation: 10, + listener: listener, + baseColor: AreaModelCommonColorProfile.gameButtonBackgroundProperty, + centerX: panelBox.centerX, + top: panelBox.bottom + 80 } ); - - const showAnswerButton = createGameButton( showAnswerString, function() { - model.showAnswer(); + enabledProperty && enabledProperty.link( function( enabled ) { + button.enabled = enabled; } ); + self.challengeLayer.addChild( button ); + return button; + } - // Cheat button, see https://github.com/phetsims/area-model-common/issues/116 and - // https://github.com/phetsims/area-model-common/issues/163 - if ( phet.chipper.queryParameters.showAnswers ) { - var cheatButton = new RectangularPushButton( { - content: new FaceNode( 40 ), - top: showAnswerButton.bottom + 10, - centerX: showAnswerButton.centerX, - listener: function() { - model.cheat(); - } - } ); - this.challengeLayer.addChild( cheatButton ); - } + const checkButton = createGameButton( checkString, function() { + model.check(); + }, model.allowCheckingProperty ); - const faceScoreNode = new FaceWithPointsNode( { - faceDiameter: 90, - pointsAlignment: 'rightBottom', - pointsFont: AreaModelCommonConstants.SCORE_INCREASE_FONT, - spacing: 10, - centerX: showAnswerButton.centerX, // a bit unclean, since the text hasn't been positioned yet. - top: showAnswerButton.bottom + 10 + const tryAgainButton = createGameButton( tryAgainString, function() { + model.tryAgain(); + } ); + + const nextButton = createGameButton( nextString, function() { + model.next(); + } ); + + const showAnswerButton = createGameButton( showAnswerString, function() { + model.showAnswer(); + } ); + + // Cheat button, see https://github.com/phetsims/area-model-common/issues/116 and + // https://github.com/phetsims/area-model-common/issues/163 + if ( phet.chipper.queryParameters.showAnswers ) { + var cheatButton = new RectangularPushButton( { + content: new FaceNode( 40 ), + top: showAnswerButton.bottom + 10, + centerX: showAnswerButton.centerX, + listener: function() { + model.cheat(); + } } ); - this.challengeLayer.addChild( faceScoreNode ); - - const levelCompleteContainer = new Node(); - this.challengeLayer.addChild( levelCompleteContainer ); - - // @private {RewardNode|null} - We need to step it when there is one - this.rewardNode = null; - - const rewardNodes = RewardNode.createRandomNodes( [ - new FaceNode( 40, { headStroke: 'black', headLineWidth: 1.5 } ), - new StarNode() - ], 100 ); - Orientation.VALUES.forEach( function( orientation ) { - const colorProperty = AreaModelCommonColorProfile.genericColorProperties.get( orientation ); - - _.range( 1, 10 ).forEach( function( digit ) { - [ -1, 1 ].forEach( function( sign ) { - const powers = model.hasExponents ? [ 0, 1, 2 ] : [ 0, 0, 0 ]; - powers.forEach( function( power ) { - rewardNodes.push( new RichText( new Term( sign * digit, power ).toRichString( false ), { - font: AreaModelCommonConstants.REWARD_NODE_FONT, - fill: colorProperty - } ) ); - } ); + this.challengeLayer.addChild( cheatButton ); + } + + const faceScoreNode = new FaceWithPointsNode( { + faceDiameter: 90, + pointsAlignment: 'rightBottom', + pointsFont: AreaModelCommonConstants.SCORE_INCREASE_FONT, + spacing: 10, + centerX: showAnswerButton.centerX, // a bit unclean, since the text hasn't been positioned yet. + top: showAnswerButton.bottom + 10 + } ); + this.challengeLayer.addChild( faceScoreNode ); + + const levelCompleteContainer = new Node(); + this.challengeLayer.addChild( levelCompleteContainer ); + + // @private {RewardNode|null} - We need to step it when there is one + this.rewardNode = null; + + const rewardNodes = RewardNode.createRandomNodes( [ + new FaceNode( 40, { headStroke: 'black', headLineWidth: 1.5 } ), + new StarNode() + ], 100 ); + Orientation.VALUES.forEach( function( orientation ) { + const colorProperty = AreaModelCommonColorProfile.genericColorProperties.get( orientation ); + + _.range( 1, 10 ).forEach( function( digit ) { + [ -1, 1 ].forEach( function( sign ) { + const powers = model.hasExponents ? [ 0, 1, 2 ] : [ 0, 0, 0 ]; + powers.forEach( function( power ) { + rewardNodes.push( new RichText( new Term( sign * digit, power ).toRichString( false ), { + font: AreaModelCommonConstants.REWARD_NODE_FONT, + fill: colorProperty + } ) ); } ); } ); } ); + } ); - let levelCompletedNode = null; - - model.stateProperty.link( function( state, oldState ) { - // When we switch back to level selection, try to leave things as they were. - if ( state !== null ) { - gameAreaNode.visible = state !== GameState.LEVEL_COMPLETE; - panelBox.visible = state !== GameState.LEVEL_COMPLETE; - statusBar.visible = state !== GameState.LEVEL_COMPLETE; - promptText.visible = state !== GameState.LEVEL_COMPLETE; - levelCompleteContainer.visible = state === GameState.LEVEL_COMPLETE; - checkButton.visible = state === GameState.FIRST_ATTEMPT || + let levelCompletedNode = null; + + model.stateProperty.link( function( state, oldState ) { + // When we switch back to level selection, try to leave things as they were. + if ( state !== null ) { + gameAreaNode.visible = state !== GameState.LEVEL_COMPLETE; + panelBox.visible = state !== GameState.LEVEL_COMPLETE; + statusBar.visible = state !== GameState.LEVEL_COMPLETE; + promptText.visible = state !== GameState.LEVEL_COMPLETE; + levelCompleteContainer.visible = state === GameState.LEVEL_COMPLETE; + checkButton.visible = state === GameState.FIRST_ATTEMPT || + state === GameState.SECOND_ATTEMPT; + tryAgainButton.visible = state === GameState.WRONG_FIRST_ANSWER; + nextButton.visible = state === GameState.CORRECT_ANSWER || + state === GameState.SHOW_SOLUTION; + showAnswerButton.visible = state === GameState.WRONG_SECOND_ANSWER; + faceScoreNode.visible = state === GameState.CORRECT_ANSWER || + state === GameState.WRONG_FIRST_ANSWER || + state === GameState.WRONG_SECOND_ANSWER; + if ( cheatButton ) { + cheatButton.visible = state === GameState.FIRST_ATTEMPT || state === GameState.SECOND_ATTEMPT; - tryAgainButton.visible = state === GameState.WRONG_FIRST_ANSWER; - nextButton.visible = state === GameState.CORRECT_ANSWER || - state === GameState.SHOW_SOLUTION; - showAnswerButton.visible = state === GameState.WRONG_SECOND_ANSWER; - faceScoreNode.visible = state === GameState.CORRECT_ANSWER || - state === GameState.WRONG_FIRST_ANSWER || - state === GameState.WRONG_SECOND_ANSWER; - if ( cheatButton ) { - cheatButton.visible = state === GameState.FIRST_ATTEMPT || - state === GameState.SECOND_ATTEMPT; - } } - if ( state === GameState.CORRECT_ANSWER ) { - faceScoreNode.smile(); - faceScoreNode.setPoints( oldState === GameState.FIRST_ATTEMPT ? 2 : 1 ); - } - else if ( state === GameState.WRONG_FIRST_ANSWER || state === GameState.WRONG_SECOND_ANSWER ) { - faceScoreNode.frown(); - } - if ( state === GameState.LEVEL_COMPLETE ) { - const level = model.currentLevelProperty.value; - - levelCompletedNode && levelCompletedNode.dispose(); - levelCompletedNode = new LevelCompletedNode( - level.number, - level.scoreProperty.value, - AreaModelCommonConstants.PERFECT_SCORE, - AreaModelCommonConstants.NUM_CHALLENGES, - false, 0, 0, 0, - function() { - model.moveToLevelSelection(); - }, { - cornerRadius: 8, - center: self.layoutBounds.center, - fill: level.colorProperty, - contentMaxWidth: 400 - } ); - - levelCompleteContainer.children = [ - levelCompletedNode - ]; - - if ( level.scoreProperty.value === AreaModelCommonConstants.PERFECT_SCORE ) { - self.rewardNode = new RewardNode( { - nodes: rewardNodes - } ); - levelCompleteContainer.insertChild( 0, self.rewardNode ); - } + } + if ( state === GameState.CORRECT_ANSWER ) { + faceScoreNode.smile(); + faceScoreNode.setPoints( oldState === GameState.FIRST_ATTEMPT ? 2 : 1 ); + } + else if ( state === GameState.WRONG_FIRST_ANSWER || state === GameState.WRONG_SECOND_ANSWER ) { + faceScoreNode.frown(); + } + if ( state === GameState.LEVEL_COMPLETE ) { + const level = model.currentLevelProperty.value; + + levelCompletedNode && levelCompletedNode.dispose(); + levelCompletedNode = new LevelCompletedNode( + level.number, + level.scoreProperty.value, + AreaModelCommonConstants.PERFECT_SCORE, + AreaModelCommonConstants.NUM_CHALLENGES, + false, 0, 0, 0, + function() { + model.moveToLevelSelection(); + }, { + cornerRadius: 8, + center: self.layoutBounds.center, + fill: level.colorProperty, + contentMaxWidth: 400 + } ); + + levelCompleteContainer.children = [ + levelCompletedNode + ]; + + if ( level.scoreProperty.value === AreaModelCommonConstants.PERFECT_SCORE ) { + self.rewardNode = new RewardNode( { + nodes: rewardNodes + } ); + levelCompleteContainer.insertChild( 0, self.rewardNode ); } - else { - if ( self.rewardNode ) { - self.rewardNode.detach(); - self.rewardNode.dispose(); - self.rewardNode = null; - } + } + else { + if ( self.rewardNode ) { + self.rewardNode.detach(); + self.rewardNode.dispose(); + self.rewardNode = null; } - } ); - } + } + } ); +} - areaModelCommon.register( 'GameAreaScreenView', GameAreaScreenView ); - - return inherit( ScreenView, GameAreaScreenView, { - /** - * Creates a panel interior with the title left-aligned, and the content somewhat offset from the left with a - * guaranteed margin. - * @private - * - * @param {string} titleString - * @param {AlignGroup} panelAlignGroup - * @param {Node} content - */ - createPanel: function( titleString, panelAlignGroup, content ) { - const panelContent = new VBox( { - children: [ - new AlignBox( new Text( titleString, { - font: AreaModelCommonConstants.TITLE_FONT, - maxWidth: AreaModelCommonConstants.PANEL_INTERIOR_MAX - } ), { - group: panelAlignGroup, - xAlign: 'left' - } ), - new AlignBox( content, { - group: panelAlignGroup, - xAlign: 'center' - } ) - ], - spacing: 10 - } ); - return new Panel( panelContent, { - xMargin: 15, - yMargin: 10, - fill: AreaModelCommonColorProfile.panelBackgroundProperty, - stroke: AreaModelCommonColorProfile.panelBorderProperty, - cornerRadius: AreaModelCommonConstants.PANEL_CORNER_RADIUS - } ); - }, +areaModelCommon.register( 'GameAreaScreenView', GameAreaScreenView ); - /** - * Steps forward in time. - * @public - * - * @param {number} dt - */ - step: function( dt ) { - this.transitionNode.step( dt ); +export default inherit( ScreenView, GameAreaScreenView, { + /** + * Creates a panel interior with the title left-aligned, and the content somewhat offset from the left with a + * guaranteed margin. + * @private + * + * @param {string} titleString + * @param {AlignGroup} panelAlignGroup + * @param {Node} content + */ + createPanel: function( titleString, panelAlignGroup, content ) { + const panelContent = new VBox( { + children: [ + new AlignBox( new Text( titleString, { + font: AreaModelCommonConstants.TITLE_FONT, + maxWidth: AreaModelCommonConstants.PANEL_INTERIOR_MAX + } ), { + group: panelAlignGroup, + xAlign: 'left' + } ), + new AlignBox( content, { + group: panelAlignGroup, + xAlign: 'center' + } ) + ], + spacing: 10 + } ); + return new Panel( panelContent, { + xMargin: 15, + yMargin: 10, + fill: AreaModelCommonColorProfile.panelBackgroundProperty, + stroke: AreaModelCommonColorProfile.panelBorderProperty, + cornerRadius: AreaModelCommonConstants.PANEL_CORNER_RADIUS + } ); + }, - this.rewardNode && this.rewardNode.step( dt ); - } - } ); -} ); + /** + * Steps forward in time. + * @public + * + * @param {number} dt + */ + step: function( dt ) { + this.transitionNode.step( dt ); + + this.rewardNode && this.rewardNode.step( dt ); + } +} ); \ No newline at end of file diff --git a/js/game/view/GameAudio.js b/js/game/view/GameAudio.js index 386e208c..aceb20b0 100644 --- a/js/game/view/GameAudio.js +++ b/js/game/view/GameAudio.js @@ -5,51 +5,48 @@ * * @author Jonathan Olson */ -define( require => { - 'use strict'; - // modules - const areaModelCommon = require( 'AREA_MODEL_COMMON/areaModelCommon' ); - const AreaModelCommonConstants = require( 'AREA_MODEL_COMMON/common/AreaModelCommonConstants' ); - const GameAudioPlayer = require( 'VEGAS/GameAudioPlayer' ); - const GameState = require( 'AREA_MODEL_COMMON/game/model/GameState' ); - const inherit = require( 'PHET_CORE/inherit' ); +import inherit from '../../../../phet-core/js/inherit.js'; +import GameAudioPlayer from '../../../../vegas/js/GameAudioPlayer.js'; +import areaModelCommon from '../../areaModelCommon.js'; +import AreaModelCommonConstants from '../../common/AreaModelCommonConstants.js'; +import GameState from '../model/GameState.js'; - /** - * @constructor - * @extends {Object} - * - * @param {GameAreaModel} model - */ - function GameAudio( model ) { - const audioPlayer = new GameAudioPlayer(); +/** + * @constructor + * @extends {Object} + * + * @param {GameAreaModel} model + */ +function GameAudio( model ) { + const audioPlayer = new GameAudioPlayer(); - model.stateProperty.link( function( state, oldState ) { - // If we just moved to/from level section (outside of a level), don't fire sounds. - if ( state === null || oldState === null ) { return; } + model.stateProperty.link( function( state, oldState ) { + // If we just moved to/from level section (outside of a level), don't fire sounds. + if ( state === null || oldState === null ) { return; } - if ( state === GameState.CORRECT_ANSWER ) { - audioPlayer.correctAnswer(); + if ( state === GameState.CORRECT_ANSWER ) { + audioPlayer.correctAnswer(); + } + if ( state === GameState.WRONG_FIRST_ANSWER || state === GameState.WRONG_SECOND_ANSWER ) { + audioPlayer.wrongAnswer(); + } + if ( state === GameState.LEVEL_COMPLETE ) { + const score = model.currentLevelProperty.value.scoreProperty.value; + if ( score === AreaModelCommonConstants.PERFECT_SCORE ) { + audioPlayer.gameOverPerfectScore(); } - if ( state === GameState.WRONG_FIRST_ANSWER || state === GameState.WRONG_SECOND_ANSWER ) { - audioPlayer.wrongAnswer(); + else if ( score === 0 ) { + audioPlayer.gameOverZeroScore(); } - if ( state === GameState.LEVEL_COMPLETE ) { - const score = model.currentLevelProperty.value.scoreProperty.value; - if ( score === AreaModelCommonConstants.PERFECT_SCORE ) { - audioPlayer.gameOverPerfectScore(); - } - else if ( score === 0 ) { - audioPlayer.gameOverZeroScore(); - } - else { - audioPlayer.gameOverImperfectScore(); - } + else { + audioPlayer.gameOverImperfectScore(); } - } ); - } + } + } ); +} - areaModelCommon.register( 'GameAudio', GameAudio ); +areaModelCommon.register( 'GameAudio', GameAudio ); - return inherit( Object, GameAudio ); -} ); +inherit( Object, GameAudio ); +export default GameAudio; \ No newline at end of file diff --git a/js/game/view/GameEditableLabelNode.js b/js/game/view/GameEditableLabelNode.js index 188c94af..7386020e 100644 --- a/js/game/view/GameEditableLabelNode.js +++ b/js/game/view/GameEditableLabelNode.js @@ -7,157 +7,154 @@ * * @author Jonathan Olson */ -define( require => { - 'use strict'; - - // modules - const areaModelCommon = require( 'AREA_MODEL_COMMON/areaModelCommon' ); - const AreaModelCommonColorProfile = require( 'AREA_MODEL_COMMON/common/view/AreaModelCommonColorProfile' ); - const AreaModelCommonConstants = require( 'AREA_MODEL_COMMON/common/AreaModelCommonConstants' ); - const DerivedProperty = require( 'AXON/DerivedProperty' ); - const DynamicProperty = require( 'AXON/DynamicProperty' ); - const EntryDisplayType = require( 'AREA_MODEL_COMMON/game/model/EntryDisplayType' ); - const EntryStatus = require( 'AREA_MODEL_COMMON/game/model/EntryStatus' ); - const GameState = require( 'AREA_MODEL_COMMON/game/model/GameState' ); - const inherit = require( 'PHET_CORE/inherit' ); - const merge = require( 'PHET_CORE/merge' ); - const Node = require( 'SCENERY/nodes/Node' ); - const Orientation = require( 'PHET_CORE/Orientation' ); - const Property = require( 'AXON/Property' ); - const RichText = require( 'SCENERY/nodes/RichText' ); - const TermEditNode = require( 'AREA_MODEL_COMMON/generic/view/TermEditNode' ); - const validate = require( 'AXON/validate' ); - const Vector2 = require( 'DOT/Vector2' ); - - /** - * @constructor - * @extends {Node} - * - * @param {Object} config - See constructor - */ - function GameEditableLabelNode( config ) { - - config = merge( { - // required - entryProperty: null, // {Property.} - gameStateProperty: null, // {Property.} - activeEntryProperty: null, // {Property.} - colorProperty: null, // {Property.} - allowExponentsProperty: null, // {Property.} - orientation: null, // {Orientation} - - // optional - labelFont: AreaModelCommonConstants.GAME_MAIN_LABEL_FONT, - editFont: AreaModelCommonConstants.GAME_MAIN_EDIT_FONT - }, config ); - - assert && assert( config.entryProperty instanceof Property ); - assert && assert( config.gameStateProperty instanceof Property ); - assert && assert( config.activeEntryProperty instanceof Property ); - assert && assert( config.colorProperty instanceof Property ); - assert && assert( config.allowExponentsProperty instanceof Property ); - validate( config.orientation, { validValues: Orientation.VALUES } ); - - Node.call( this ); - - // Helpful to break out some values - const entryProperty = config.entryProperty; - const gameStateProperty = config.gameStateProperty; - const activeEntryProperty = config.activeEntryProperty; - const colorProperty = config.colorProperty; - const allowExponentsProperty = config.allowExponentsProperty; - const orientation = config.orientation; - - const valueProperty = new DynamicProperty( entryProperty, { - derive: 'valueProperty', - bidirectional: true - } ); - const digitsProperty = new DerivedProperty( [ entryProperty ], _.property( 'digits' ) ); - const statusProperty = new DynamicProperty( entryProperty, { - derive: 'statusProperty' - } ); - const isActiveProperty = new DerivedProperty( - [ entryProperty, activeEntryProperty ], - function( entry, activeEntry ) { - return entry === activeEntry; - } ); - - const readoutText = new RichText( '?', { - fill: colorProperty, - font: config.labelFont - } ); - this.addChild( readoutText ); - valueProperty.link( function( termOrList ) { - readoutText.text = termOrList === null ? '?' : termOrList.toRichString( false ); - readoutText.center = Vector2.ZERO; +import DerivedProperty from '../../../../axon/js/DerivedProperty.js'; +import DynamicProperty from '../../../../axon/js/DynamicProperty.js'; +import Property from '../../../../axon/js/Property.js'; +import validate from '../../../../axon/js/validate.js'; +import Vector2 from '../../../../dot/js/Vector2.js'; +import inherit from '../../../../phet-core/js/inherit.js'; +import merge from '../../../../phet-core/js/merge.js'; +import Orientation from '../../../../phet-core/js/Orientation.js'; +import Node from '../../../../scenery/js/nodes/Node.js'; +import RichText from '../../../../scenery/js/nodes/RichText.js'; +import areaModelCommon from '../../areaModelCommon.js'; +import AreaModelCommonConstants from '../../common/AreaModelCommonConstants.js'; +import AreaModelCommonColorProfile from '../../common/view/AreaModelCommonColorProfile.js'; +import TermEditNode from '../../generic/view/TermEditNode.js'; +import EntryDisplayType from '../model/EntryDisplayType.js'; +import EntryStatus from '../model/EntryStatus.js'; +import GameState from '../model/GameState.js'; + +/** + * @constructor + * @extends {Node} + * + * @param {Object} config - See constructor + */ +function GameEditableLabelNode( config ) { + + config = merge( { + // required + entryProperty: null, // {Property.} + gameStateProperty: null, // {Property.} + activeEntryProperty: null, // {Property.} + colorProperty: null, // {Property.} + allowExponentsProperty: null, // {Property.} + orientation: null, // {Orientation} + + // optional + labelFont: AreaModelCommonConstants.GAME_MAIN_LABEL_FONT, + editFont: AreaModelCommonConstants.GAME_MAIN_EDIT_FONT + }, config ); + + assert && assert( config.entryProperty instanceof Property ); + assert && assert( config.gameStateProperty instanceof Property ); + assert && assert( config.activeEntryProperty instanceof Property ); + assert && assert( config.colorProperty instanceof Property ); + assert && assert( config.allowExponentsProperty instanceof Property ); + validate( config.orientation, { validValues: Orientation.VALUES } ); + + Node.call( this ); + + // Helpful to break out some values + const entryProperty = config.entryProperty; + const gameStateProperty = config.gameStateProperty; + const activeEntryProperty = config.activeEntryProperty; + const colorProperty = config.colorProperty; + const allowExponentsProperty = config.allowExponentsProperty; + const orientation = config.orientation; + + const valueProperty = new DynamicProperty( entryProperty, { + derive: 'valueProperty', + bidirectional: true + } ); + const digitsProperty = new DerivedProperty( [ entryProperty ], _.property( 'digits' ) ); + const statusProperty = new DynamicProperty( entryProperty, { + derive: 'statusProperty' + } ); + const isActiveProperty = new DerivedProperty( + [ entryProperty, activeEntryProperty ], + function( entry, activeEntry ) { + return entry === activeEntry; } ); - const textColorProperty = new DerivedProperty( - [ statusProperty, colorProperty, AreaModelCommonColorProfile.errorStatusProperty ], - function( highlight, color, errorColor ) { - if ( highlight === EntryStatus.INCORRECT ) { - return errorColor; - } - else { - return color; - } - } ); - const borderColorProperty = new DerivedProperty( [ - statusProperty, - colorProperty, - AreaModelCommonColorProfile.errorStatusProperty, - AreaModelCommonColorProfile.dirtyStatusProperty - ], function( highlight, color, errorColor, dirtyColor ) { - if ( highlight === EntryStatus.NORMAL ) { + const readoutText = new RichText( '?', { + fill: colorProperty, + font: config.labelFont + } ); + this.addChild( readoutText ); + + valueProperty.link( function( termOrList ) { + readoutText.text = termOrList === null ? '?' : termOrList.toRichString( false ); + readoutText.center = Vector2.ZERO; + } ); + + const textColorProperty = new DerivedProperty( + [ statusProperty, colorProperty, AreaModelCommonColorProfile.errorStatusProperty ], + function( highlight, color, errorColor ) { + if ( highlight === EntryStatus.INCORRECT ) { + return errorColor; + } + else { return color; } - else if ( highlight === EntryStatus.DIRTY ) { - return dirtyColor; + } ); + const borderColorProperty = new DerivedProperty( [ + statusProperty, + colorProperty, + AreaModelCommonColorProfile.errorStatusProperty, + AreaModelCommonColorProfile.dirtyStatusProperty + ], function( highlight, color, errorColor, dirtyColor ) { + if ( highlight === EntryStatus.NORMAL ) { + return color; + } + else if ( highlight === EntryStatus.DIRTY ) { + return dirtyColor; + } + else { + return errorColor; + } + } ); + const termEditNode = new TermEditNode( new Property( orientation ), valueProperty, { + textColorProperty: textColorProperty, + borderColorProperty: borderColorProperty, + isActiveProperty: isActiveProperty, + digitCountProperty: digitsProperty, + allowExponentsProperty: allowExponentsProperty, + editCallback: function() { + if ( gameStateProperty.value === GameState.WRONG_FIRST_ANSWER ) { + gameStateProperty.value = GameState.SECOND_ATTEMPT; + } + if ( activeEntryProperty.value !== entryProperty.value ) { + activeEntryProperty.value = entryProperty.value; } else { - return errorColor; + // Pressing on the edit button when that keypad is already open will instead close the keypad. + // See https://github.com/phetsims/area-model-common/issues/127 + activeEntryProperty.value = null; } - } ); - const termEditNode = new TermEditNode( new Property( orientation ), valueProperty, { - textColorProperty: textColorProperty, - borderColorProperty: borderColorProperty, - isActiveProperty: isActiveProperty, - digitCountProperty: digitsProperty, - allowExponentsProperty: allowExponentsProperty, - editCallback: function() { - if ( gameStateProperty.value === GameState.WRONG_FIRST_ANSWER ) { - gameStateProperty.value = GameState.SECOND_ATTEMPT; - } - if ( activeEntryProperty.value !== entryProperty.value ) { - activeEntryProperty.value = entryProperty.value; - } - else { - // Pressing on the edit button when that keypad is already open will instead close the keypad. - // See https://github.com/phetsims/area-model-common/issues/127 - activeEntryProperty.value = null; - } - }, - font: config.editFont - } ); - this.addChild( termEditNode ); + }, + font: config.editFont + } ); + this.addChild( termEditNode ); - function centerTermEditNode() { - termEditNode.center = Vector2.ZERO; - } + function centerTermEditNode() { + termEditNode.center = Vector2.ZERO; + } - digitsProperty.link( centerTermEditNode ); - allowExponentsProperty.link( centerTermEditNode ); + digitsProperty.link( centerTermEditNode ); + allowExponentsProperty.link( centerTermEditNode ); - Property.multilink( [ entryProperty, gameStateProperty ], function( entry, gameState ) { - const isReadoutOverride = gameState === GameState.CORRECT_ANSWER || gameState === GameState.SHOW_SOLUTION; - readoutText.visible = entry.displayType === EntryDisplayType.READOUT || - ( isReadoutOverride && entry.displayType === EntryDisplayType.EDITABLE ); - termEditNode.visible = entry.displayType === EntryDisplayType.EDITABLE && !isReadoutOverride; - } ); - } + Property.multilink( [ entryProperty, gameStateProperty ], function( entry, gameState ) { + const isReadoutOverride = gameState === GameState.CORRECT_ANSWER || gameState === GameState.SHOW_SOLUTION; + readoutText.visible = entry.displayType === EntryDisplayType.READOUT || + ( isReadoutOverride && entry.displayType === EntryDisplayType.EDITABLE ); + termEditNode.visible = entry.displayType === EntryDisplayType.EDITABLE && !isReadoutOverride; + } ); +} - areaModelCommon.register( 'GameEditableLabelNode', GameEditableLabelNode ); +areaModelCommon.register( 'GameEditableLabelNode', GameEditableLabelNode ); - return inherit( Node, GameEditableLabelNode ); -} ); +inherit( Node, GameEditableLabelNode ); +export default GameEditableLabelNode; \ No newline at end of file diff --git a/js/game/view/PolynomialEditNode.js b/js/game/view/PolynomialEditNode.js index 2f59f074..1542155d 100644 --- a/js/game/view/PolynomialEditNode.js +++ b/js/game/view/PolynomialEditNode.js @@ -7,219 +7,216 @@ * * @author Jonathan Olson */ -define( require => { - 'use strict'; - - // modules - const areaModelCommon = require( 'AREA_MODEL_COMMON/areaModelCommon' ); - const AreaModelCommonColorProfile = require( 'AREA_MODEL_COMMON/common/view/AreaModelCommonColorProfile' ); - const AreaModelCommonConstants = require( 'AREA_MODEL_COMMON/common/AreaModelCommonConstants' ); - const DerivedProperty = require( 'AXON/DerivedProperty' ); - const DynamicProperty = require( 'AXON/DynamicProperty' ); - const Entry = require( 'AREA_MODEL_COMMON/game/model/Entry' ); - const EntryStatus = require( 'AREA_MODEL_COMMON/game/model/EntryStatus' ); - const inherit = require( 'PHET_CORE/inherit' ); - const InputMethod = require( 'AREA_MODEL_COMMON/game/model/InputMethod' ); - const MathSymbols = require( 'SCENERY_PHET/MathSymbols' ); - const Node = require( 'SCENERY/nodes/Node' ); - const NumberPicker = require( 'SCENERY_PHET/NumberPicker' ); - const Polynomial = require( 'AREA_MODEL_COMMON/common/model/Polynomial' ); - const Property = require( 'AXON/Property' ); - const Range = require( 'DOT/Range' ); - const Rectangle = require( 'SCENERY/nodes/Rectangle' ); - const RichText = require( 'SCENERY/nodes/RichText' ); - const Term = require( 'AREA_MODEL_COMMON/common/model/Term' ); - const Text = require( 'SCENERY/nodes/Text' ); - const VBox = require( 'SCENERY/nodes/VBox' ); - - /** - * @constructor - * @extends {Node} - * - * @param {Property.} polynomialProperty - * @param {Property.>} totalEntriesProperty - * @param {function} editedCallback - Called with no arguments when something is edited - */ - function PolynomialEditNode( polynomialProperty, totalEntriesProperty, editedCallback ) { - const longestString = new Polynomial( [ - new Term( -9, 2 ), - new Term( -9, 1 ), - new Term( -9, 0 ) - ] ).toRichString(); - - const readoutText = new RichText( longestString, { - font: AreaModelCommonConstants.POLYNOMIAL_EDIT_READOUT_FONT - } ); - - const readoutBackgroundRectangle = Rectangle.bounds( readoutText.bounds.dilatedXY( 30, 5 ), { - cornerRadius: 3, - stroke: 'black', - fill: 'white' - } ); - readoutText.centerY = readoutBackgroundRectangle.centerY; // Don't reposition vertically with exponents - polynomialProperty.link( function( polynomial ) { - readoutText.text = polynomial === null ? '0' : polynomial.toRichString(); - readoutText.centerX = readoutBackgroundRectangle.centerX; - } ); - const readout = new Node( { - children: [ - readoutBackgroundRectangle, - readoutText - ] - } ); - - const editFont = AreaModelCommonConstants.GAME_POLYNOMIAL_EDIT_FONT; - - // {Property.} - const constantEntryProperty = new DerivedProperty( [ totalEntriesProperty ], function( totalEntries ) { - return totalEntries.length > 1 ? totalEntries[ 0 ] : new Entry( null ); - } ); - const xEntryProperty = new DerivedProperty( [ totalEntriesProperty ], function( totalEntries ) { - return totalEntries.length > 1 ? totalEntries[ 1 ] : new Entry( null ); - } ); - const xSquaredEntryProperty = new DerivedProperty( [ totalEntriesProperty ], function( totalEntries ) { - return totalEntries.length > 2 ? totalEntries[ 2 ] : new Entry( null ); - } ); - const constantProperty = new DynamicProperty( constantEntryProperty, { - derive: 'valueProperty', - map: function( term ) { - return term === null ? 0 : term.coefficient; - }, - inverseMap: function( number ) { - return new Term( number, 0 ); - }, - bidirectional: true - } ); +import DerivedProperty from '../../../../axon/js/DerivedProperty.js'; +import DynamicProperty from '../../../../axon/js/DynamicProperty.js'; +import Property from '../../../../axon/js/Property.js'; +import Range from '../../../../dot/js/Range.js'; +import inherit from '../../../../phet-core/js/inherit.js'; +import MathSymbols from '../../../../scenery-phet/js/MathSymbols.js'; +import NumberPicker from '../../../../scenery-phet/js/NumberPicker.js'; +import Node from '../../../../scenery/js/nodes/Node.js'; +import Rectangle from '../../../../scenery/js/nodes/Rectangle.js'; +import RichText from '../../../../scenery/js/nodes/RichText.js'; +import Text from '../../../../scenery/js/nodes/Text.js'; +import VBox from '../../../../scenery/js/nodes/VBox.js'; +import areaModelCommon from '../../areaModelCommon.js'; +import AreaModelCommonConstants from '../../common/AreaModelCommonConstants.js'; +import Polynomial from '../../common/model/Polynomial.js'; +import Term from '../../common/model/Term.js'; +import AreaModelCommonColorProfile from '../../common/view/AreaModelCommonColorProfile.js'; +import Entry from '../model/Entry.js'; +import EntryStatus from '../model/EntryStatus.js'; +import InputMethod from '../model/InputMethod.js'; - const xProperty = new DynamicProperty( xEntryProperty, { - derive: 'valueProperty', - map: function( term ) { - return term === null ? 0 : term.coefficient; - }, - inverseMap: function( number ) { - return new Term( number, 1 ); - }, - bidirectional: true - } ); - - const xSquaredProperty = new DynamicProperty( xSquaredEntryProperty, { - derive: 'valueProperty', - map: function( term ) { - return term === null ? 0 : term.coefficient; - }, - inverseMap: function( number ) { - return new Term( number, 2 ); - }, - bidirectional: true - } ); - - // When one is changed, we want to make sure that all entries are not marked as dirty internally (so that the user - // can submit after just changing one value). This is done by providing an actual value to the property. - function provideEntryValues() { - [ constantEntryProperty, xEntryProperty, xSquaredEntryProperty ].forEach( function( entryProperty, index ) { - const valueProperty = entryProperty.value.valueProperty; - if ( valueProperty.value === null ) { - valueProperty.value = new Term( 0, index ); - } - } ); - } - - function linkProperty( property, entryProperty ) { - property.link( function() { - // Only flag the values as edited when the user makes a change (not when we set it as part of a challenge) - if ( property.isExternallyChanging ) { - editedCallback(); - provideEntryValues(); - entryProperty.value.statusProperty.value = EntryStatus.NORMAL; - } - } ); - } - - linkProperty( constantProperty, constantEntryProperty ); - linkProperty( xProperty, xEntryProperty ); - linkProperty( xSquaredProperty, xSquaredEntryProperty ); - - // [-81,81] is the actual range we need for editable values, - // see https://github.com/phetsims/area-model-common/issues/94 - const rangeProperty = new Property( new Range( -81, 81 ) ); - - function getPickerColorProperty( entryProperty ) { - return new DerivedProperty( [ - new DynamicProperty( entryProperty, { derive: 'statusProperty' } ), - AreaModelCommonColorProfile.errorStatusProperty, - AreaModelCommonColorProfile.dirtyStatusProperty - ], function( highlight, errorColor, dirtyColor ) { - if ( highlight === EntryStatus.NORMAL ) { - return 'black'; - } - else if ( highlight === EntryStatus.DIRTY ) { - return dirtyColor; - } - else { - return errorColor; - } - } ); - } - - const constantPicker = new NumberPicker( constantProperty, rangeProperty, { - color: getPickerColorProperty( constantEntryProperty ) - } ); - const xPicker = new NumberPicker( xProperty, rangeProperty, { - color: getPickerColorProperty( xEntryProperty ) - } ); - const xSquaredPicker = new NumberPicker( xSquaredProperty, rangeProperty, { - color: getPickerColorProperty( xSquaredEntryProperty ) +/** + * @constructor + * @extends {Node} + * + * @param {Property.} polynomialProperty + * @param {Property.>} totalEntriesProperty + * @param {function} editedCallback - Called with no arguments when something is edited + */ +function PolynomialEditNode( polynomialProperty, totalEntriesProperty, editedCallback ) { + const longestString = new Polynomial( [ + new Term( -9, 2 ), + new Term( -9, 1 ), + new Term( -9, 0 ) + ] ).toRichString(); + + const readoutText = new RichText( longestString, { + font: AreaModelCommonConstants.POLYNOMIAL_EDIT_READOUT_FONT + } ); + + const readoutBackgroundRectangle = Rectangle.bounds( readoutText.bounds.dilatedXY( 30, 5 ), { + cornerRadius: 3, + stroke: 'black', + fill: 'white' + } ); + readoutText.centerY = readoutBackgroundRectangle.centerY; // Don't reposition vertically with exponents + polynomialProperty.link( function( polynomial ) { + readoutText.text = polynomial === null ? '0' : polynomial.toRichString(); + readoutText.centerX = readoutBackgroundRectangle.centerX; + } ); + const readout = new Node( { + children: [ + readoutBackgroundRectangle, + readoutText + ] + } ); + + const editFont = AreaModelCommonConstants.GAME_POLYNOMIAL_EDIT_FONT; + + // {Property.} + const constantEntryProperty = new DerivedProperty( [ totalEntriesProperty ], function( totalEntries ) { + return totalEntries.length > 1 ? totalEntries[ 0 ] : new Entry( null ); + } ); + const xEntryProperty = new DerivedProperty( [ totalEntriesProperty ], function( totalEntries ) { + return totalEntries.length > 1 ? totalEntries[ 1 ] : new Entry( null ); + } ); + const xSquaredEntryProperty = new DerivedProperty( [ totalEntriesProperty ], function( totalEntries ) { + return totalEntries.length > 2 ? totalEntries[ 2 ] : new Entry( null ); + } ); + + const constantProperty = new DynamicProperty( constantEntryProperty, { + derive: 'valueProperty', + map: function( term ) { + return term === null ? 0 : term.coefficient; + }, + inverseMap: function( number ) { + return new Term( number, 0 ); + }, + bidirectional: true + } ); + + const xProperty = new DynamicProperty( xEntryProperty, { + derive: 'valueProperty', + map: function( term ) { + return term === null ? 0 : term.coefficient; + }, + inverseMap: function( number ) { + return new Term( number, 1 ); + }, + bidirectional: true + } ); + + const xSquaredProperty = new DynamicProperty( xSquaredEntryProperty, { + derive: 'valueProperty', + map: function( term ) { + return term === null ? 0 : term.coefficient; + }, + inverseMap: function( number ) { + return new Term( number, 2 ); + }, + bidirectional: true + } ); + + // When one is changed, we want to make sure that all entries are not marked as dirty internally (so that the user + // can submit after just changing one value). This is done by providing an actual value to the property. + function provideEntryValues() { + [ constantEntryProperty, xEntryProperty, xSquaredEntryProperty ].forEach( function( entryProperty, index ) { + const valueProperty = entryProperty.value.valueProperty; + if ( valueProperty.value === null ) { + valueProperty.value = new Term( 0, index ); + } } ); + } - const xText = new RichText( AreaModelCommonConstants.X_VARIABLE_RICH_STRING, { font: editFont } ); - const xSquaredText = new RichText( AreaModelCommonConstants.X_VARIABLE_RICH_STRING + '2', { font: editFont } ); - const plus1 = new Text( MathSymbols.PLUS, { font: editFont } ); - const plus2 = new Text( MathSymbols.PLUS, { font: editFont } ); - - const xSquaredChildren = [ - xSquaredPicker, - xSquaredText, - plus1, - xPicker, - xText, - plus2, - constantPicker - ]; - const xChildren = [ - xPicker, - xText, - plus2, - constantPicker - ]; - - const pickerContainer = new Node(); - // Hide the x^2 term if we won't use it - constantEntryProperty.link( function( constantEntry ) { - pickerContainer.children = constantEntry.inputMethod === InputMethod.POLYNOMIAL_2 - ? xSquaredChildren - : xChildren; + function linkProperty( property, entryProperty ) { + property.link( function() { + // Only flag the values as edited when the user makes a change (not when we set it as part of a challenge) + if ( property.isExternallyChanging ) { + editedCallback(); + provideEntryValues(); + entryProperty.value.statusProperty.value = EntryStatus.NORMAL; + } } ); + } - xSquaredChildren.forEach( function( node, index ) { - if ( index > 0 ) { - node.left = xSquaredChildren[ index - 1 ].right + 5; + linkProperty( constantProperty, constantEntryProperty ); + linkProperty( xProperty, xEntryProperty ); + linkProperty( xSquaredProperty, xSquaredEntryProperty ); + + // [-81,81] is the actual range we need for editable values, + // see https://github.com/phetsims/area-model-common/issues/94 + const rangeProperty = new Property( new Range( -81, 81 ) ); + + function getPickerColorProperty( entryProperty ) { + return new DerivedProperty( [ + new DynamicProperty( entryProperty, { derive: 'statusProperty' } ), + AreaModelCommonColorProfile.errorStatusProperty, + AreaModelCommonColorProfile.dirtyStatusProperty + ], function( highlight, errorColor, dirtyColor ) { + if ( highlight === EntryStatus.NORMAL ) { + return 'black'; + } + else if ( highlight === EntryStatus.DIRTY ) { + return dirtyColor; + } + else { + return errorColor; } - } ); - constantPicker.centerY = xText.centerY; - xPicker.centerY = xText.centerY; - xSquaredPicker.centerY = xText.centerY; - - VBox.call( this, { - children: [ - readout, - pickerContainer - ], - spacing: 10 } ); } - areaModelCommon.register( 'PolynomialEditNode', PolynomialEditNode ); - - return inherit( VBox, PolynomialEditNode ); -} ); + const constantPicker = new NumberPicker( constantProperty, rangeProperty, { + color: getPickerColorProperty( constantEntryProperty ) + } ); + const xPicker = new NumberPicker( xProperty, rangeProperty, { + color: getPickerColorProperty( xEntryProperty ) + } ); + const xSquaredPicker = new NumberPicker( xSquaredProperty, rangeProperty, { + color: getPickerColorProperty( xSquaredEntryProperty ) + } ); + + const xText = new RichText( AreaModelCommonConstants.X_VARIABLE_RICH_STRING, { font: editFont } ); + const xSquaredText = new RichText( AreaModelCommonConstants.X_VARIABLE_RICH_STRING + '2', { font: editFont } ); + const plus1 = new Text( MathSymbols.PLUS, { font: editFont } ); + const plus2 = new Text( MathSymbols.PLUS, { font: editFont } ); + + const xSquaredChildren = [ + xSquaredPicker, + xSquaredText, + plus1, + xPicker, + xText, + plus2, + constantPicker + ]; + const xChildren = [ + xPicker, + xText, + plus2, + constantPicker + ]; + + const pickerContainer = new Node(); + // Hide the x^2 term if we won't use it + constantEntryProperty.link( function( constantEntry ) { + pickerContainer.children = constantEntry.inputMethod === InputMethod.POLYNOMIAL_2 + ? xSquaredChildren + : xChildren; + } ); + + xSquaredChildren.forEach( function( node, index ) { + if ( index > 0 ) { + node.left = xSquaredChildren[ index - 1 ].right + 5; + } + } ); + constantPicker.centerY = xText.centerY; + xPicker.centerY = xText.centerY; + xSquaredPicker.centerY = xText.centerY; + + VBox.call( this, { + children: [ + readout, + pickerContainer + ], + spacing: 10 + } ); +} + +areaModelCommon.register( 'PolynomialEditNode', PolynomialEditNode ); + +inherit( VBox, PolynomialEditNode ); +export default PolynomialEditNode; \ No newline at end of file diff --git a/js/generic/model/GenericArea.js b/js/generic/model/GenericArea.js index b274108f..cc3ab30a 100644 --- a/js/generic/model/GenericArea.js +++ b/js/generic/model/GenericArea.js @@ -7,127 +7,123 @@ * * @author Jonathan Olson */ -define( require => { - 'use strict'; - - // modules - const Area = require( 'AREA_MODEL_COMMON/common/model/Area' ); - const areaModelCommon = require( 'AREA_MODEL_COMMON/areaModelCommon' ); - const AreaModelCommonColorProfile = require( 'AREA_MODEL_COMMON/common/view/AreaModelCommonColorProfile' ); - const AreaModelCommonConstants = require( 'AREA_MODEL_COMMON/common/AreaModelCommonConstants' ); - const AreaModelCommonQueryParameters = require( 'AREA_MODEL_COMMON/common/AreaModelCommonQueryParameters' ); - const GenericPartition = require( 'AREA_MODEL_COMMON/generic/model/GenericPartition' ); - const inherit = require( 'PHET_CORE/inherit' ); - const Orientation = require( 'PHET_CORE/Orientation' ); - const OrientationPair = require( 'AREA_MODEL_COMMON/common/model/OrientationPair' ); - const Property = require( 'AXON/Property' ); - const Range = require( 'DOT/Range' ); - const Term = require( 'AREA_MODEL_COMMON/common/model/Term' ); - /** - * @constructor - * @extends {Area} - * - * @param {GenericLayout} layout - * @param {boolean} allowExponents - Whether the user is able to add powers of x. - */ - function GenericArea( layout, allowExponents ) { - assert && assert( typeof allowExponents === 'boolean' ); - - const self = this; - - // If we allow powers of X, we'll only allow 1 digit in front. - const firstDigitCount = allowExponents ? 1 : 3; - const secondDigitCount = allowExponents ? 1 : 2; - const thirdDigitCount = 1; - - const horizontalPartitions = [ - new GenericPartition( Orientation.HORIZONTAL, firstDigitCount ), - new GenericPartition( Orientation.HORIZONTAL, secondDigitCount ), - new GenericPartition( Orientation.HORIZONTAL, thirdDigitCount ) - ].slice( 0, layout.size.width ); - - const verticalPartitions = [ - new GenericPartition( Orientation.VERTICAL, firstDigitCount ), - new GenericPartition( Orientation.VERTICAL, secondDigitCount ), - new GenericPartition( Orientation.VERTICAL, thirdDigitCount ) - ].slice( 0, layout.size.height ); - - Area.call( - this, - new OrientationPair( horizontalPartitions, verticalPartitions ), - AreaModelCommonColorProfile.genericColorProperties, - 1, - allowExponents - ); - - if ( AreaModelCommonQueryParameters.maximumCalculationSize ) { - horizontalPartitions.forEach( function( partition, index ) { - partition.sizeProperty.value = new Term( -Math.pow( 10, partition.digitCount ) + 1, allowExponents ? 2 - index : 0 ); - } ); - verticalPartitions.forEach( function( partition, index ) { - partition.sizeProperty.value = new Term( -Math.pow( 10, partition.digitCount ) + 1, allowExponents ? 2 - index : 0 ); - } ); - } +import Property from '../../../../axon/js/Property.js'; +import Range from '../../../../dot/js/Range.js'; +import inherit from '../../../../phet-core/js/inherit.js'; +import Orientation from '../../../../phet-core/js/Orientation.js'; +import areaModelCommon from '../../areaModelCommon.js'; +import AreaModelCommonConstants from '../../common/AreaModelCommonConstants.js'; +import AreaModelCommonQueryParameters from '../../common/AreaModelCommonQueryParameters.js'; +import Area from '../../common/model/Area.js'; +import OrientationPair from '../../common/model/OrientationPair.js'; +import Term from '../../common/model/Term.js'; +import AreaModelCommonColorProfile from '../../common/view/AreaModelCommonColorProfile.js'; +import GenericPartition from './GenericPartition.js'; - // @public {GenericLayout} - this.layout = layout; - - // Set up partition coordinate/size - Orientation.VALUES.forEach( function( orientation ) { - const partitionCount = layout.getPartitionQuantity( orientation ); - const partitions = self.partitions.get( orientation ); - - if ( partitionCount === 1 ) { - partitions[ 0 ].coordinateRangeProperty.value = new Range( 0, 1 ); - } - else if ( partitionCount === 2 ) { - partitions[ 0 ].coordinateRangeProperty.value = new Range( 0, AreaModelCommonConstants.GENERIC_SINGLE_OFFSET ); - partitions[ 1 ].coordinateRangeProperty.value = new Range( AreaModelCommonConstants.GENERIC_SINGLE_OFFSET, 1 ); - } - else if ( partitionCount === 3 ) { - partitions[ 0 ].coordinateRangeProperty.value = new Range( 0, AreaModelCommonConstants.GENERIC_FIRST_OFFSET ); - partitions[ 1 ].coordinateRangeProperty.value = new Range( AreaModelCommonConstants.GENERIC_FIRST_OFFSET, AreaModelCommonConstants.GENERIC_SECOND_OFFSET ); - partitions[ 2 ].coordinateRangeProperty.value = new Range( AreaModelCommonConstants.GENERIC_SECOND_OFFSET, 1 ); - } +/** + * @constructor + * @extends {Area} + * + * @param {GenericLayout} layout + * @param {boolean} allowExponents - Whether the user is able to add powers of x. + */ +function GenericArea( layout, allowExponents ) { + assert && assert( typeof allowExponents === 'boolean' ); + + const self = this; + + // If we allow powers of X, we'll only allow 1 digit in front. + const firstDigitCount = allowExponents ? 1 : 3; + const secondDigitCount = allowExponents ? 1 : 2; + const thirdDigitCount = 1; + + const horizontalPartitions = [ + new GenericPartition( Orientation.HORIZONTAL, firstDigitCount ), + new GenericPartition( Orientation.HORIZONTAL, secondDigitCount ), + new GenericPartition( Orientation.HORIZONTAL, thirdDigitCount ) + ].slice( 0, layout.size.width ); + + const verticalPartitions = [ + new GenericPartition( Orientation.VERTICAL, firstDigitCount ), + new GenericPartition( Orientation.VERTICAL, secondDigitCount ), + new GenericPartition( Orientation.VERTICAL, thirdDigitCount ) + ].slice( 0, layout.size.height ); + + Area.call( + this, + new OrientationPair( horizontalPartitions, verticalPartitions ), + AreaModelCommonColorProfile.genericColorProperties, + 1, + allowExponents + ); + + if ( AreaModelCommonQueryParameters.maximumCalculationSize ) { + horizontalPartitions.forEach( function( partition, index ) { + partition.sizeProperty.value = new Term( -Math.pow( 10, partition.digitCount ) + 1, allowExponents ? 2 - index : 0 ); + } ); + verticalPartitions.forEach( function( partition, index ) { + partition.sizeProperty.value = new Term( -Math.pow( 10, partition.digitCount ) + 1, allowExponents ? 2 - index : 0 ); } ); - - // @public {Property.} - If it exists, the partition being actively edited. - this.activePartitionProperty = new Property( null ); } - areaModelCommon.register( 'GenericArea', GenericArea ); - - return inherit( Area, GenericArea, { - /** - * Resets the area to its initial values. - * @public - * @override - */ - reset: function() { - Area.prototype.reset.call( this ); - - this.allPartitions.forEach( function( partition ) { - partition.sizeProperty.reset(); - } ); - - this.activePartitionProperty.reset(); - }, - - /** - * Erase the area to a 1x1, see https://github.com/phetsims/area-model-common/issues/77 - * @public - * @override - */ - erase: function() { - Area.prototype.erase.call( this ); - - // Clear all partition values - this.allPartitions.forEach( function( partition ) { - partition.sizeProperty.value = null; - } ); - - this.activePartitionProperty.reset(); + // @public {GenericLayout} + this.layout = layout; + + // Set up partition coordinate/size + Orientation.VALUES.forEach( function( orientation ) { + const partitionCount = layout.getPartitionQuantity( orientation ); + const partitions = self.partitions.get( orientation ); + + if ( partitionCount === 1 ) { + partitions[ 0 ].coordinateRangeProperty.value = new Range( 0, 1 ); + } + else if ( partitionCount === 2 ) { + partitions[ 0 ].coordinateRangeProperty.value = new Range( 0, AreaModelCommonConstants.GENERIC_SINGLE_OFFSET ); + partitions[ 1 ].coordinateRangeProperty.value = new Range( AreaModelCommonConstants.GENERIC_SINGLE_OFFSET, 1 ); + } + else if ( partitionCount === 3 ) { + partitions[ 0 ].coordinateRangeProperty.value = new Range( 0, AreaModelCommonConstants.GENERIC_FIRST_OFFSET ); + partitions[ 1 ].coordinateRangeProperty.value = new Range( AreaModelCommonConstants.GENERIC_FIRST_OFFSET, AreaModelCommonConstants.GENERIC_SECOND_OFFSET ); + partitions[ 2 ].coordinateRangeProperty.value = new Range( AreaModelCommonConstants.GENERIC_SECOND_OFFSET, 1 ); } } ); -} ); + + // @public {Property.} - If it exists, the partition being actively edited. + this.activePartitionProperty = new Property( null ); +} + +areaModelCommon.register( 'GenericArea', GenericArea ); + +export default inherit( Area, GenericArea, { + /** + * Resets the area to its initial values. + * @public + * @override + */ + reset: function() { + Area.prototype.reset.call( this ); + + this.allPartitions.forEach( function( partition ) { + partition.sizeProperty.reset(); + } ); + + this.activePartitionProperty.reset(); + }, + + /** + * Erase the area to a 1x1, see https://github.com/phetsims/area-model-common/issues/77 + * @public + * @override + */ + erase: function() { + Area.prototype.erase.call( this ); + + // Clear all partition values + this.allPartitions.forEach( function( partition ) { + partition.sizeProperty.value = null; + } ); + + this.activePartitionProperty.reset(); + } +} ); \ No newline at end of file diff --git a/js/generic/model/GenericAreaDisplay.js b/js/generic/model/GenericAreaDisplay.js index bcb750cc..9fdcb2f9 100644 --- a/js/generic/model/GenericAreaDisplay.js +++ b/js/generic/model/GenericAreaDisplay.js @@ -5,33 +5,30 @@ * * @author Jonathan Olson */ -define( require => { - 'use strict'; - // modules - const AreaDisplay = require( 'AREA_MODEL_COMMON/common/model/AreaDisplay' ); - const areaModelCommon = require( 'AREA_MODEL_COMMON/areaModelCommon' ); - const inherit = require( 'PHET_CORE/inherit' ); +import inherit from '../../../../phet-core/js/inherit.js'; +import areaModelCommon from '../../areaModelCommon.js'; +import AreaDisplay from '../../common/model/AreaDisplay.js'; - /** - * @constructor - * @extends {AreaDisplay} - * - * @param {Property.} areaProperty - */ - function GenericAreaDisplay( areaProperty ) { - AreaDisplay.call( this, areaProperty ); +/** + * @constructor + * @extends {AreaDisplay} + * + * @param {Property.} areaProperty + */ +function GenericAreaDisplay( areaProperty ) { + AreaDisplay.call( this, areaProperty ); - // @public {Property.} - this.layoutProperty = this.wrapObject( _.property( 'layout' ) ); + // @public {Property.} + this.layoutProperty = this.wrapObject( _.property( 'layout' ) ); - // @public {Property.} - this.activePartitionProperty = this.wrapProperty( _.property( 'activePartitionProperty' ), { - bidirectional: true - } ); - } + // @public {Property.} + this.activePartitionProperty = this.wrapProperty( _.property( 'activePartitionProperty' ), { + bidirectional: true + } ); +} - areaModelCommon.register( 'GenericAreaDisplay', GenericAreaDisplay ); +areaModelCommon.register( 'GenericAreaDisplay', GenericAreaDisplay ); - return inherit( AreaDisplay, GenericAreaDisplay ); -} ); +inherit( AreaDisplay, GenericAreaDisplay ); +export default GenericAreaDisplay; \ No newline at end of file diff --git a/js/generic/model/GenericAreaModel.js b/js/generic/model/GenericAreaModel.js index 952e1ba6..64163179 100644 --- a/js/generic/model/GenericAreaModel.js +++ b/js/generic/model/GenericAreaModel.js @@ -5,81 +5,77 @@ * * @author Jonathan Olson */ -define( require => { - 'use strict'; - // modules - const areaModelCommon = require( 'AREA_MODEL_COMMON/areaModelCommon' ); - const AreaModelCommonModel = require( 'AREA_MODEL_COMMON/common/model/AreaModelCommonModel' ); - const GenericArea = require( 'AREA_MODEL_COMMON/generic/model/GenericArea' ); - const GenericAreaDisplay = require( 'AREA_MODEL_COMMON/generic/model/GenericAreaDisplay' ); - const GenericLayout = require( 'AREA_MODEL_COMMON/generic/model/GenericLayout' ); - const inherit = require( 'PHET_CORE/inherit' ); - const merge = require( 'PHET_CORE/merge' ); - const Property = require( 'AXON/Property' ); +import Property from '../../../../axon/js/Property.js'; +import inherit from '../../../../phet-core/js/inherit.js'; +import merge from '../../../../phet-core/js/merge.js'; +import areaModelCommon from '../../areaModelCommon.js'; +import AreaModelCommonModel from '../../common/model/AreaModelCommonModel.js'; +import GenericArea from './GenericArea.js'; +import GenericAreaDisplay from './GenericAreaDisplay.js'; +import GenericLayout from './GenericLayout.js'; - // constants - const DEFAULT_LAYOUT = GenericLayout.TWO_BY_TWO; +// constants +const DEFAULT_LAYOUT = GenericLayout.TWO_BY_TWO; - /** - * @constructor - * @extends {AreaModelCommonModel} - * - * @param {Object} [options] - */ - function GenericAreaModel( options ) { - const self = this; +/** + * @constructor + * @extends {AreaModelCommonModel} + * + * @param {Object} [options] + */ +function GenericAreaModel( options ) { + const self = this; - assert && assert( options === undefined || typeof options === 'object', 'If provided, options should be an object' ); + assert && assert( options === undefined || typeof options === 'object', 'If provided, options should be an object' ); - options = merge( { - allowExponents: false - }, options ); + options = merge( { + allowExponents: false + }, options ); - // @public {Property.} - The current layout that is visible/selected. - this.genericLayoutProperty = new Property( DEFAULT_LAYOUT ); + // @public {Property.} - The current layout that is visible/selected. + this.genericLayoutProperty = new Property( DEFAULT_LAYOUT ); - const areas = GenericLayout.VALUES.map( function( layout ) { - return new GenericArea( layout, options.allowExponents ); - } ); + const areas = GenericLayout.VALUES.map( function( layout ) { + return new GenericArea( layout, options.allowExponents ); + } ); - const defaultArea = _.find( areas, function( area ) { - return area.layout === DEFAULT_LAYOUT; - } ); + const defaultArea = _.find( areas, function( area ) { + return area.layout === DEFAULT_LAYOUT; + } ); - AreaModelCommonModel.call( this, areas, defaultArea, options ); + AreaModelCommonModel.call( this, areas, defaultArea, options ); - // Adjust the current area based on the layout. - this.genericLayoutProperty.link( function( layout ) { - self.currentAreaProperty.value = _.find( self.areas, function( area ) { - return area.layout === layout; - } ); + // Adjust the current area based on the layout. + this.genericLayoutProperty.link( function( layout ) { + self.currentAreaProperty.value = _.find( self.areas, function( area ) { + return area.layout === layout; } ); - } + } ); +} - areaModelCommon.register( 'GenericAreaModel', GenericAreaModel ); +areaModelCommon.register( 'GenericAreaModel', GenericAreaModel ); - return inherit( AreaModelCommonModel, GenericAreaModel, { - /** - * Returns a concrete AreaDisplay subtype - * @protected - * - * @param {Property.} areaProperty - * @returns {GenericAreaDisplay} - */ - createAreaDisplay: function( areaProperty ) { - return new GenericAreaDisplay( areaProperty ); - }, +export default inherit( AreaModelCommonModel, GenericAreaModel, { + /** + * Returns a concrete AreaDisplay subtype + * @protected + * + * @param {Property.} areaProperty + * @returns {GenericAreaDisplay} + */ + createAreaDisplay: function( areaProperty ) { + return new GenericAreaDisplay( areaProperty ); + }, - /** - * Returns the model to its initial state. - * @public - * @override - */ - reset: function() { - AreaModelCommonModel.prototype.reset.call( this ); + /** + * Returns the model to its initial state. + * @public + * @override + */ + reset: function() { + AreaModelCommonModel.prototype.reset.call( this ); - this.genericLayoutProperty.reset(); - } - } ); -} ); + this.genericLayoutProperty.reset(); + } +} ); \ No newline at end of file diff --git a/js/generic/model/GenericLayout.js b/js/generic/model/GenericLayout.js index 33700269..6f12a35f 100644 --- a/js/generic/model/GenericLayout.js +++ b/js/generic/model/GenericLayout.js @@ -5,101 +5,97 @@ * * @author Jonathan Olson */ -define( require => { - 'use strict'; - // modules - const areaModelCommon = require( 'AREA_MODEL_COMMON/areaModelCommon' ); - const Dimension2 = require( 'DOT/Dimension2' ); - const inherit = require( 'PHET_CORE/inherit' ); - const Orientation = require( 'PHET_CORE/Orientation' ); - const validate = require( 'AXON/validate' ); +import validate from '../../../../axon/js/validate.js'; +import Dimension2 from '../../../../dot/js/Dimension2.js'; +import inherit from '../../../../phet-core/js/inherit.js'; +import Orientation from '../../../../phet-core/js/Orientation.js'; +import areaModelCommon from '../../areaModelCommon.js'; +/** + * @constructor + * @extends {Object} + * @private (use enumeration of objects) + * + * @param {Dimension2} size + */ +function GenericLayout( size ) { + // @public {Dimension2} - Dimension describes how many partitions are available in each orientation. + this.size = size; +} + +areaModelCommon.register( 'GenericLayout', GenericLayout ); + +inherit( Object, GenericLayout, { /** - * @constructor - * @extends {Object} - * @private (use enumeration of objects) + * Returns the number of partitions for the specific orientation. + * @public * - * @param {Dimension2} size + * @param {Orientation} orientation + * @returns {number} */ - function GenericLayout( size ) { - // @public {Dimension2} - Dimension describes how many partitions are available in each orientation. - this.size = size; - } - - areaModelCommon.register( 'GenericLayout', GenericLayout ); - - inherit( Object, GenericLayout, { - /** - * Returns the number of partitions for the specific orientation. - * @public - * - * @param {Orientation} orientation - * @returns {number} - */ - getPartitionQuantity: function( orientation ) { - validate( orientation, { validValues: Orientation.VALUES } ); + getPartitionQuantity: function( orientation ) { + validate( orientation, { validValues: Orientation.VALUES } ); - return orientation === Orientation.HORIZONTAL ? this.size.width : this.size.height; - } - } ); + return orientation === Orientation.HORIZONTAL ? this.size.width : this.size.height; + } +} ); - // @public {GenericLayout} - GenericLayout.ONE_BY_ONE = new GenericLayout( new Dimension2( 1, 1 ) ); - GenericLayout.ONE_BY_TWO = new GenericLayout( new Dimension2( 1, 2 ) ); - GenericLayout.ONE_BY_THREE = new GenericLayout( new Dimension2( 1, 3 ) ); - GenericLayout.TWO_BY_ONE = new GenericLayout( new Dimension2( 2, 1 ) ); - GenericLayout.TWO_BY_TWO = new GenericLayout( new Dimension2( 2, 2 ) ); - GenericLayout.TWO_BY_THREE = new GenericLayout( new Dimension2( 2, 3 ) ); - GenericLayout.THREE_BY_ONE = new GenericLayout( new Dimension2( 3, 1 ) ); - GenericLayout.THREE_BY_TWO = new GenericLayout( new Dimension2( 3, 2 ) ); - GenericLayout.THREE_BY_THREE = new GenericLayout( new Dimension2( 3, 3 ) ); +// @public {GenericLayout} +GenericLayout.ONE_BY_ONE = new GenericLayout( new Dimension2( 1, 1 ) ); +GenericLayout.ONE_BY_TWO = new GenericLayout( new Dimension2( 1, 2 ) ); +GenericLayout.ONE_BY_THREE = new GenericLayout( new Dimension2( 1, 3 ) ); +GenericLayout.TWO_BY_ONE = new GenericLayout( new Dimension2( 2, 1 ) ); +GenericLayout.TWO_BY_TWO = new GenericLayout( new Dimension2( 2, 2 ) ); +GenericLayout.TWO_BY_THREE = new GenericLayout( new Dimension2( 2, 3 ) ); +GenericLayout.THREE_BY_ONE = new GenericLayout( new Dimension2( 3, 1 ) ); +GenericLayout.THREE_BY_TWO = new GenericLayout( new Dimension2( 3, 2 ) ); +GenericLayout.THREE_BY_THREE = new GenericLayout( new Dimension2( 3, 3 ) ); - // @public {Array.} - All values the enumeration can take. - GenericLayout.VALUES = [ - GenericLayout.ONE_BY_ONE, - GenericLayout.ONE_BY_TWO, - GenericLayout.ONE_BY_THREE, - GenericLayout.TWO_BY_ONE, - GenericLayout.TWO_BY_TWO, - GenericLayout.TWO_BY_THREE, - GenericLayout.THREE_BY_ONE, - GenericLayout.THREE_BY_TWO, - GenericLayout.THREE_BY_THREE - ]; +// @public {Array.} - All values the enumeration can take. +GenericLayout.VALUES = [ + GenericLayout.ONE_BY_ONE, + GenericLayout.ONE_BY_TWO, + GenericLayout.ONE_BY_THREE, + GenericLayout.TWO_BY_ONE, + GenericLayout.TWO_BY_TWO, + GenericLayout.TWO_BY_THREE, + GenericLayout.THREE_BY_ONE, + GenericLayout.THREE_BY_TWO, + GenericLayout.THREE_BY_THREE +]; - /** - * Returns the layout value given a specific width and height. - * @public - * - * @param {number} width - * @param {number} height - */ - GenericLayout.fromValues = function( width, height ) { - assert && assert( typeof width === 'number' && isFinite( width ) && width % 1 === 0 && width >= 1 && width <= 3 ); - assert && assert( typeof height === 'number' && isFinite( height ) && height % 1 === 0 && height >= 1 && height <= 3 ); +/** + * Returns the layout value given a specific width and height. + * @public + * + * @param {number} width + * @param {number} height + */ +GenericLayout.fromValues = function( width, height ) { + assert && assert( typeof width === 'number' && isFinite( width ) && width % 1 === 0 && width >= 1 && width <= 3 ); + assert && assert( typeof height === 'number' && isFinite( height ) && height % 1 === 0 && height >= 1 && height <= 3 ); - return { - 1: { - 1: GenericLayout.ONE_BY_ONE, - 2: GenericLayout.ONE_BY_TWO, - 3: GenericLayout.ONE_BY_THREE - }, - 2: { - 1: GenericLayout.TWO_BY_ONE, - 2: GenericLayout.TWO_BY_TWO, - 3: GenericLayout.TWO_BY_THREE - }, - 3: { - 1: GenericLayout.THREE_BY_ONE, - 2: GenericLayout.THREE_BY_TWO, - 3: GenericLayout.THREE_BY_THREE - } - }[ width ][ height ]; - }; + return { + 1: { + 1: GenericLayout.ONE_BY_ONE, + 2: GenericLayout.ONE_BY_TWO, + 3: GenericLayout.ONE_BY_THREE + }, + 2: { + 1: GenericLayout.TWO_BY_ONE, + 2: GenericLayout.TWO_BY_TWO, + 3: GenericLayout.TWO_BY_THREE + }, + 3: { + 1: GenericLayout.THREE_BY_ONE, + 2: GenericLayout.THREE_BY_TWO, + 3: GenericLayout.THREE_BY_THREE + } + }[ width ][ height ]; +}; - // verify that enumeration is immutable, without the runtime penalty in production code - if ( assert ) { Object.freeze( GenericLayout ); } +// verify that enumeration is immutable, without the runtime penalty in production code +if ( assert ) { Object.freeze( GenericLayout ); } - return GenericLayout; -} ); +export default GenericLayout; \ No newline at end of file diff --git a/js/generic/model/GenericPartition.js b/js/generic/model/GenericPartition.js index 4ac13e97..caeafce3 100644 --- a/js/generic/model/GenericPartition.js +++ b/js/generic/model/GenericPartition.js @@ -5,35 +5,32 @@ * * @author Jonathan Olson */ -define( require => { - 'use strict'; - // modules - const areaModelCommon = require( 'AREA_MODEL_COMMON/areaModelCommon' ); - const AreaModelCommonColorProfile = require( 'AREA_MODEL_COMMON/common/view/AreaModelCommonColorProfile' ); - const inherit = require( 'PHET_CORE/inherit' ); - const Orientation = require( 'PHET_CORE/Orientation' ); - const Partition = require( 'AREA_MODEL_COMMON/common/model/Partition' ); - const validate = require( 'AXON/validate' ); +import validate from '../../../../axon/js/validate.js'; +import inherit from '../../../../phet-core/js/inherit.js'; +import Orientation from '../../../../phet-core/js/Orientation.js'; +import areaModelCommon from '../../areaModelCommon.js'; +import Partition from '../../common/model/Partition.js'; +import AreaModelCommonColorProfile from '../../common/view/AreaModelCommonColorProfile.js'; - /** - * @constructor - * @extends {Partition} - * - * @param {Orientation} orientation - * @param {number} digitCount - */ - function GenericPartition( orientation, digitCount ) { - validate( orientation, { validValues: Orientation.VALUES } ); - assert && assert( typeof digitCount === 'number' ); +/** + * @constructor + * @extends {Partition} + * + * @param {Orientation} orientation + * @param {number} digitCount + */ +function GenericPartition( orientation, digitCount ) { + validate( orientation, { validValues: Orientation.VALUES } ); + assert && assert( typeof digitCount === 'number' ); - Partition.call( this, orientation, AreaModelCommonColorProfile.genericColorProperties.get( orientation ) ); + Partition.call( this, orientation, AreaModelCommonColorProfile.genericColorProperties.get( orientation ) ); - // @public {number} - How many digits to allow in the editor - this.digitCount = digitCount; - } + // @public {number} - How many digits to allow in the editor + this.digitCount = digitCount; +} - areaModelCommon.register( 'GenericPartition', GenericPartition ); +areaModelCommon.register( 'GenericPartition', GenericPartition ); - return inherit( Partition, GenericPartition ); -} ); +inherit( Partition, GenericPartition ); +export default GenericPartition; \ No newline at end of file diff --git a/js/generic/view/GenericAreaDisplayNode.js b/js/generic/view/GenericAreaDisplayNode.js index b526ed41..25bfd3f4 100644 --- a/js/generic/view/GenericAreaDisplayNode.js +++ b/js/generic/view/GenericAreaDisplayNode.js @@ -7,185 +7,181 @@ * * @author Jonathan Olson */ -define( require => { - 'use strict'; - - // modules - const AreaDisplayNode = require( 'AREA_MODEL_COMMON/common/view/AreaDisplayNode' ); - const areaModelCommon = require( 'AREA_MODEL_COMMON/areaModelCommon' ); - const AreaModelCommonColorProfile = require( 'AREA_MODEL_COMMON/common/view/AreaModelCommonColorProfile' ); - const AreaModelCommonConstants = require( 'AREA_MODEL_COMMON/common/AreaModelCommonConstants' ); - const DerivedProperty = require( 'AXON/DerivedProperty' ); - const GenericPartitionedAreaNode = require( 'AREA_MODEL_COMMON/generic/view/GenericPartitionedAreaNode' ); - const inherit = require( 'PHET_CORE/inherit' ); - const Line = require( 'SCENERY/nodes/Line' ); - const Node = require( 'SCENERY/nodes/Node' ); - const Orientation = require( 'PHET_CORE/Orientation' ); - const PartitionSizeEditNode = require( 'AREA_MODEL_COMMON/generic/view/PartitionSizeEditNode' ); - const PoolableLayerNode = require( 'AREA_MODEL_COMMON/common/view/PoolableLayerNode' ); - const Property = require( 'AXON/Property' ); - const TermKeypadPanel = require( 'AREA_MODEL_COMMON/generic/view/TermKeypadPanel' ); - const Vector2 = require( 'DOT/Vector2' ); +import DerivedProperty from '../../../../axon/js/DerivedProperty.js'; +import Property from '../../../../axon/js/Property.js'; +import Vector2 from '../../../../dot/js/Vector2.js'; +import inherit from '../../../../phet-core/js/inherit.js'; +import Orientation from '../../../../phet-core/js/Orientation.js'; +import Line from '../../../../scenery/js/nodes/Line.js'; +import Node from '../../../../scenery/js/nodes/Node.js'; +import areaModelCommon from '../../areaModelCommon.js'; +import AreaModelCommonConstants from '../../common/AreaModelCommonConstants.js'; +import AreaDisplayNode from '../../common/view/AreaDisplayNode.js'; +import AreaModelCommonColorProfile from '../../common/view/AreaModelCommonColorProfile.js'; +import PoolableLayerNode from '../../common/view/PoolableLayerNode.js'; +import GenericPartitionedAreaNode from './GenericPartitionedAreaNode.js'; +import PartitionSizeEditNode from './PartitionSizeEditNode.js'; +import TermKeypadPanel from './TermKeypadPanel.js'; + +/** + * @constructor + * @extends {AreaDisplayNode} + * + * @param {GenericAreaDisplay} areaDisplay + * @param {boolean} allowExponents - Whether the user is able to add powers of x. + * @param {Property.} partialProductsChoiceProperty + * @param {Object} [nodeOptions] + */ +function GenericAreaDisplayNode( areaDisplay, allowExponents, partialProductsChoiceProperty, nodeOptions ) { + assert && assert( typeof allowExponents === 'boolean' ); + assert && assert( partialProductsChoiceProperty instanceof Property ); + + const self = this; + + AreaDisplayNode.call( this, areaDisplay, partialProductsChoiceProperty, { + allowExponents: allowExponents, + isProportional: false + } ); + + this.areaLayer.addChild( this.backgroundNode ); + + // Sign-colored partition area backgrounds (effectively pooled) + this.areaLayer.addChild( new PoolableLayerNode( { + arrayProperty: areaDisplay.partitionedAreasProperty, + createNode: function( partitionedArea ) { + return new GenericPartitionedAreaNode( new Property( partitionedArea ), self.modelViewTransformProperty ); + }, + getItemProperty: function( partitionedAreaNode ) { + return partitionedAreaNode.partitionedAreaProperty; + } + } ) ); + + this.areaLayer.addChild( this.borderNode ); + + // Partition lines + this.areaLayer.addChild( GenericAreaDisplayNode.createPartitionLines( areaDisplay.layoutProperty, this.viewSize ) ); + + // Edit readouts/buttons + this.labelLayer.addChild( new PoolableLayerNode( { + arrayProperty: areaDisplay.allPartitionsProperty, + createNode: function( partition ) { + return new PartitionSizeEditNode( + areaDisplay.activePartitionProperty, + new Property( partition ), + self.modelViewTransformProperty, + allowExponents + ); + }, + getItemProperty: function( editNode ) { + return editNode.partitionProperty; + } + } ) ); + + // Keypad + const digitCountProperty = new DerivedProperty( [ areaDisplay.activePartitionProperty ], function( activePartition ) { + return activePartition === null ? 1 : activePartition.digitCount; + } ); + const termKeypadPanel = new TermKeypadPanel( digitCountProperty, allowExponents, true, function( term ) { + // Update the size of the partition. + areaDisplay.activePartitionProperty.value.sizeProperty.value = term; + + // Hide the keypad. + areaDisplay.activePartitionProperty.value = null; + }, { + x: this.viewSize + AreaModelCommonConstants.KEYPAD_LEFT_PADDING, + centerY: this.viewSize / 2 + } ); + this.labelLayer.addChild( termKeypadPanel ); + + // If this changes, we clear and switch to it + areaDisplay.activePartitionProperty.link( function( newArea ) { + termKeypadPanel.visible = newArea !== null; + termKeypadPanel.clear(); + } ); + + this.mutate( nodeOptions ); +} + +areaModelCommon.register( 'GenericAreaDisplayNode', GenericAreaDisplayNode ); + +export default inherit( AreaDisplayNode, GenericAreaDisplayNode, { /** - * @constructor - * @extends {AreaDisplayNode} - * - * @param {GenericAreaDisplay} areaDisplay - * @param {boolean} allowExponents - Whether the user is able to add powers of x. - * @param {Property.} partialProductsChoiceProperty - * @param {Object} [nodeOptions] + * Positions all of the partial products labels. + * @protected + * @override */ - function GenericAreaDisplayNode( areaDisplay, allowExponents, partialProductsChoiceProperty, nodeOptions ) { - assert && assert( typeof allowExponents === 'boolean' ); - assert && assert( partialProductsChoiceProperty instanceof Property ); - + positionProductLabels: function() { const self = this; - AreaDisplayNode.call( this, areaDisplay, partialProductsChoiceProperty, { - allowExponents: allowExponents, - isProportional: false - } ); - - this.areaLayer.addChild( this.backgroundNode ); - - // Sign-colored partition area backgrounds (effectively pooled) - this.areaLayer.addChild( new PoolableLayerNode( { - arrayProperty: areaDisplay.partitionedAreasProperty, - createNode: function( partitionedArea ) { - return new GenericPartitionedAreaNode( new Property( partitionedArea ), self.modelViewTransformProperty ); - }, - getItemProperty: function( partitionedAreaNode ) { - return partitionedAreaNode.partitionedAreaProperty; - } - } ) ); - - this.areaLayer.addChild( this.borderNode ); - - // Partition lines - this.areaLayer.addChild( GenericAreaDisplayNode.createPartitionLines( areaDisplay.layoutProperty, this.viewSize ) ); - - // Edit readouts/buttons - this.labelLayer.addChild( new PoolableLayerNode( { - arrayProperty: areaDisplay.allPartitionsProperty, - createNode: function( partition ) { - return new PartitionSizeEditNode( - areaDisplay.activePartitionProperty, - new Property( partition ), - self.modelViewTransformProperty, - allowExponents - ); - }, - getItemProperty: function( editNode ) { - return editNode.partitionProperty; - } - } ) ); - - // Keypad - const digitCountProperty = new DerivedProperty( [ areaDisplay.activePartitionProperty ], function( activePartition ) { - return activePartition === null ? 1 : activePartition.digitCount; + this.productLabels.forEach( function( productLabel ) { + Orientation.VALUES.forEach( function( orientation ) { + const range = productLabel.partitionedAreaProperty.value.partitions.get( orientation ).coordinateRangeProperty.value; + if ( range !== null ) { + productLabel[ orientation.coordinate ] = orientation.modelToView( self.modelViewTransformProperty.value, range.getCenter() ); + } + } ); } ); - const termKeypadPanel = new TermKeypadPanel( digitCountProperty, allowExponents, true, function( term ) { - // Update the size of the partition. - areaDisplay.activePartitionProperty.value.sizeProperty.value = term; - - // Hide the keypad. - areaDisplay.activePartitionProperty.value = null; - }, { - x: this.viewSize + AreaModelCommonConstants.KEYPAD_LEFT_PADDING, - centerY: this.viewSize / 2 + } +}, { + /** + * Creates a partition line (view only) + * @private + * + * @param {Orientation} orientation + * @param {number} offset + * @param {number} viewSize - In view units, the size of the main area + * @param {Property.} visibilityProperty + */ + createPartitionLine: function( orientation, offset, viewSize, visibilityProperty ) { + const firstPoint = new Vector2( 0, 0 ); + const secondPoint = new Vector2( 0, 0 ); + + firstPoint[ orientation.coordinate ] = offset; + secondPoint[ orientation.coordinate ] = offset; + firstPoint[ orientation.opposite.coordinate ] = viewSize; + secondPoint[ orientation.opposite.coordinate ] = 0; + + const line = new Line( { + p1: firstPoint, + p2: secondPoint, + stroke: AreaModelCommonColorProfile.partitionLineStrokeProperty } ); - this.labelLayer.addChild( termKeypadPanel ); + visibilityProperty.linkAttribute( line, 'visible' ); + return line; + }, - // If this changes, we clear and switch to it - areaDisplay.activePartitionProperty.link( function( newArea ) { - termKeypadPanel.visible = newArea !== null; - termKeypadPanel.clear(); - } ); + /** + * Creates a set of generic partition lines. + * @public + * + * @param {Property.} layoutProperty + * @param {number} viewSize + * @returns {Node} + */ + createPartitionLines: function( layoutProperty, viewSize ) { + const singleOffset = viewSize * AreaModelCommonConstants.GENERIC_SINGLE_OFFSET; + const firstOffset = viewSize * AreaModelCommonConstants.GENERIC_FIRST_OFFSET; + const secondOffset = viewSize * AreaModelCommonConstants.GENERIC_SECOND_OFFSET; - this.mutate( nodeOptions ); - } + const resultNode = new Node(); - areaModelCommon.register( 'GenericAreaDisplayNode', GenericAreaDisplayNode ); - - return inherit( AreaDisplayNode, GenericAreaDisplayNode, { - /** - * Positions all of the partial products labels. - * @protected - * @override - */ - positionProductLabels: function() { - const self = this; - - this.productLabels.forEach( function( productLabel ) { - Orientation.VALUES.forEach( function( orientation ) { - const range = productLabel.partitionedAreaProperty.value.partitions.get( orientation ).coordinateRangeProperty.value; - if ( range !== null ) { - productLabel[ orientation.coordinate ] = orientation.modelToView( self.modelViewTransformProperty.value, range.getCenter() ); - } - } ); + Orientation.VALUES.forEach( function( orientation ) { + const hasTwoProperty = new DerivedProperty( [ layoutProperty ], function( layout ) { + return layout.getPartitionQuantity( orientation ) === 2; } ); - } - }, { - /** - * Creates a partition line (view only) - * @private - * - * @param {Orientation} orientation - * @param {number} offset - * @param {number} viewSize - In view units, the size of the main area - * @param {Property.} visibilityProperty - */ - createPartitionLine: function( orientation, offset, viewSize, visibilityProperty ) { - const firstPoint = new Vector2( 0, 0 ); - const secondPoint = new Vector2( 0, 0 ); - - firstPoint[ orientation.coordinate ] = offset; - secondPoint[ orientation.coordinate ] = offset; - firstPoint[ orientation.opposite.coordinate ] = viewSize; - secondPoint[ orientation.opposite.coordinate ] = 0; - - const line = new Line( { - p1: firstPoint, - p2: secondPoint, - stroke: AreaModelCommonColorProfile.partitionLineStrokeProperty + const hasThreeProperty = new DerivedProperty( [ layoutProperty ], function( layout ) { + return layout.getPartitionQuantity( orientation ) === 3; } ); - visibilityProperty.linkAttribute( line, 'visible' ); - return line; - }, - - /** - * Creates a set of generic partition lines. - * @public - * - * @param {Property.} layoutProperty - * @param {number} viewSize - * @returns {Node} - */ - createPartitionLines: function( layoutProperty, viewSize ) { - const singleOffset = viewSize * AreaModelCommonConstants.GENERIC_SINGLE_OFFSET; - const firstOffset = viewSize * AreaModelCommonConstants.GENERIC_FIRST_OFFSET; - const secondOffset = viewSize * AreaModelCommonConstants.GENERIC_SECOND_OFFSET; - - const resultNode = new Node(); - Orientation.VALUES.forEach( function( orientation ) { - const hasTwoProperty = new DerivedProperty( [ layoutProperty ], function( layout ) { - return layout.getPartitionQuantity( orientation ) === 2; - } ); - const hasThreeProperty = new DerivedProperty( [ layoutProperty ], function( layout ) { - return layout.getPartitionQuantity( orientation ) === 3; - } ); - - const singleLine = GenericAreaDisplayNode.createPartitionLine( orientation, singleOffset, viewSize, hasTwoProperty ); - const firstLine = GenericAreaDisplayNode.createPartitionLine( orientation, firstOffset, viewSize, hasThreeProperty ); - const secondLine = GenericAreaDisplayNode.createPartitionLine( orientation, secondOffset, viewSize, hasThreeProperty ); - resultNode.addChild( singleLine ); - resultNode.addChild( firstLine ); - resultNode.addChild( secondLine ); - } ); + const singleLine = GenericAreaDisplayNode.createPartitionLine( orientation, singleOffset, viewSize, hasTwoProperty ); + const firstLine = GenericAreaDisplayNode.createPartitionLine( orientation, firstOffset, viewSize, hasThreeProperty ); + const secondLine = GenericAreaDisplayNode.createPartitionLine( orientation, secondOffset, viewSize, hasThreeProperty ); + resultNode.addChild( singleLine ); + resultNode.addChild( firstLine ); + resultNode.addChild( secondLine ); + } ); - return resultNode; - } - } ); -} ); + return resultNode; + } +} ); \ No newline at end of file diff --git a/js/generic/view/GenericAreaScreenView.js b/js/generic/view/GenericAreaScreenView.js index 35ce6d88..fbb536a1 100644 --- a/js/generic/view/GenericAreaScreenView.js +++ b/js/generic/view/GenericAreaScreenView.js @@ -7,100 +7,96 @@ * * @author Jonathan Olson */ -define( require => { - 'use strict'; - // modules - const areaModelCommon = require( 'AREA_MODEL_COMMON/areaModelCommon' ); - const AreaScreenView = require( 'AREA_MODEL_COMMON/common/view/AreaScreenView' ); - const DerivedProperty = require( 'AXON/DerivedProperty' ); - const DynamicProperty = require( 'AXON/DynamicProperty' ); - const GenericAreaDisplayNode = require( 'AREA_MODEL_COMMON/generic/view/GenericAreaDisplayNode' ); - const GenericAreaModel = require( 'AREA_MODEL_COMMON/generic/model/GenericAreaModel' ); - const GenericFactorsNode = require( 'AREA_MODEL_COMMON/generic/view/GenericFactorsNode' ); - const GenericLayoutSelectionNode = require( 'AREA_MODEL_COMMON/generic/view/GenericLayoutSelectionNode' ); - const inherit = require( 'PHET_CORE/inherit' ); - const Node = require( 'SCENERY/nodes/Node' ); - const OrientationPair = require( 'AREA_MODEL_COMMON/common/model/OrientationPair' ); - const Property = require( 'AXON/Property' ); +import DerivedProperty from '../../../../axon/js/DerivedProperty.js'; +import DynamicProperty from '../../../../axon/js/DynamicProperty.js'; +import Property from '../../../../axon/js/Property.js'; +import inherit from '../../../../phet-core/js/inherit.js'; +import Node from '../../../../scenery/js/nodes/Node.js'; +import areaModelCommon from '../../areaModelCommon.js'; +import OrientationPair from '../../common/model/OrientationPair.js'; +import AreaScreenView from '../../common/view/AreaScreenView.js'; +import GenericAreaModel from '../model/GenericAreaModel.js'; +import GenericAreaDisplayNode from './GenericAreaDisplayNode.js'; +import GenericFactorsNode from './GenericFactorsNode.js'; +import GenericLayoutSelectionNode from './GenericLayoutSelectionNode.js'; - /** - * @constructor - * @extends {AreaScreenView} - * - * @param {GenericAreaModel} model - * @param {number} decimalPlaces - */ - function GenericAreaScreenView( model, decimalPlaces ) { - assert && assert( model instanceof GenericAreaModel ); - assert && assert( typeof decimalPlaces === 'number' ); +/** + * @constructor + * @extends {AreaScreenView} + * + * @param {GenericAreaModel} model + * @param {number} decimalPlaces + */ +function GenericAreaScreenView( model, decimalPlaces ) { + assert && assert( model instanceof GenericAreaModel ); + assert && assert( typeof decimalPlaces === 'number' ); - // @private {Node} - this.popupLayer = new Node(); + // @private {Node} + this.popupLayer = new Node(); - // @private {Node|null} - Will be filled in with getRightAlignNodes (we need lazy creation here unfortunately). - // We need to construct it later when factorsBox.width is defined (so we can properly size it), but we can't wait - // until the supertype is fully constructed since then our right align setup will have been constructed fully. - this.layoutSelectionNode = null; + // @private {Node|null} - Will be filled in with getRightAlignNodes (we need lazy creation here unfortunately). + // We need to construct it later when factorsBox.width is defined (so we can properly size it), but we can't wait + // until the supertype is fully constructed since then our right align setup will have been constructed fully. + this.layoutSelectionNode = null; - AreaScreenView.call( this, model, { - isProportional: false, - decimalPlaces: decimalPlaces - } ); + AreaScreenView.call( this, model, { + isProportional: false, + decimalPlaces: decimalPlaces + } ); - this.addChild( this.popupLayer ); - } + this.addChild( this.popupLayer ); +} - areaModelCommon.register( 'GenericAreaScreenView', GenericAreaScreenView ); +areaModelCommon.register( 'GenericAreaScreenView', GenericAreaScreenView ); - return inherit( AreaScreenView, GenericAreaScreenView, { - /** - * @protected - * @override - * - * @returns {Array.} - */ - getRightAlignNodes: function() { - assert && assert( !this.layoutSelectionNode, 'Should not be called multiple times or it will leak memory' ); +export default inherit( AreaScreenView, GenericAreaScreenView, { + /** + * @protected + * @override + * + * @returns {Array.} + */ + getRightAlignNodes: function() { + assert && assert( !this.layoutSelectionNode, 'Should not be called multiple times or it will leak memory' ); - this.layoutSelectionNode = new GenericLayoutSelectionNode( - this.model.genericLayoutProperty, - this.popupLayer, - this.factorsBox.width - ); - return [ this.layoutSelectionNode ].concat( AreaScreenView.prototype.getRightAlignNodes.call( this ) ); - }, + this.layoutSelectionNode = new GenericLayoutSelectionNode( + this.model.genericLayoutProperty, + this.popupLayer, + this.factorsBox.width + ); + return [ this.layoutSelectionNode ].concat( AreaScreenView.prototype.getRightAlignNodes.call( this ) ); + }, - /** - * Creates the main area display view for the screen. - * @public - * @override - * - * @param {GenericAreaModel} model - * @returns {GenericAreaDisplayNode} - */ - createAreaDisplayNode: function( model ) { - return new GenericAreaDisplayNode( model.areaDisplay, model.allowExponents, model.partialProductsChoiceProperty, { - translation: this.getDisplayTranslation() - } ); - }, + /** + * Creates the main area display view for the screen. + * @public + * @override + * + * @param {GenericAreaModel} model + * @returns {GenericAreaDisplayNode} + */ + createAreaDisplayNode: function( model ) { + return new GenericAreaDisplayNode( model.areaDisplay, model.allowExponents, model.partialProductsChoiceProperty, { + translation: this.getDisplayTranslation() + } ); + }, - /** - * Creates the "factors" (dimensions) content for the accordion box. - * @public - * @override - * - * @param {GenericAreaModel} model - * @param {number} decimalPlaces - * @returns {Node} - */ - createFactorsNode: function( model, decimalPlaces ) { - const dynamicProperties = OrientationPair.create( function( orientation ) { - return new DynamicProperty( new DerivedProperty( [ model.currentAreaProperty ], function( area ) { - return area.displayProperties.get( orientation ); - } ) ); - } ); - return new GenericFactorsNode( dynamicProperties, new Property( model.allowExponents ) ); - } - } ); -} ); + /** + * Creates the "factors" (dimensions) content for the accordion box. + * @public + * @override + * + * @param {GenericAreaModel} model + * @param {number} decimalPlaces + * @returns {Node} + */ + createFactorsNode: function( model, decimalPlaces ) { + const dynamicProperties = OrientationPair.create( function( orientation ) { + return new DynamicProperty( new DerivedProperty( [ model.currentAreaProperty ], function( area ) { + return area.displayProperties.get( orientation ); + } ) ); + } ); + return new GenericFactorsNode( dynamicProperties, new Property( model.allowExponents ) ); + } +} ); \ No newline at end of file diff --git a/js/generic/view/GenericFactorsNode.js b/js/generic/view/GenericFactorsNode.js index 2cd5bfa2..0b07c339 100644 --- a/js/generic/view/GenericFactorsNode.js +++ b/js/generic/view/GenericFactorsNode.js @@ -7,134 +7,130 @@ * * @author Jonathan Olson */ -define( require => { - 'use strict'; - - // modules - const AlignBox = require( 'SCENERY/nodes/AlignBox' ); - const areaModelCommon = require( 'AREA_MODEL_COMMON/areaModelCommon' ); - const AreaModelCommonColorProfile = require( 'AREA_MODEL_COMMON/common/view/AreaModelCommonColorProfile' ); - const AreaModelCommonConstants = require( 'AREA_MODEL_COMMON/common/AreaModelCommonConstants' ); - const Bounds2 = require( 'DOT/Bounds2' ); - const HBox = require( 'SCENERY/nodes/HBox' ); - const inherit = require( 'PHET_CORE/inherit' ); - const MathSymbols = require( 'SCENERY_PHET/MathSymbols' ); - const Node = require( 'SCENERY/nodes/Node' ); - const Orientation = require( 'PHET_CORE/Orientation' ); - const Rectangle = require( 'SCENERY/nodes/Rectangle' ); - const RichText = require( 'SCENERY/nodes/RichText' ); - const Text = require( 'SCENERY/nodes/Text' ); - const validate = require( 'AXON/validate' ); - - // constants - const BOX_SIZE = 30; - const PAREN_BOUNDS = new Text( ')(', { - font: AreaModelCommonConstants.FACTORS_PAREN_FONT, - boundsMethod: 'accurate' - } ).bounds; - /** - * @constructor - * @extends {HBox} - * - * @param {OrientationPair.>} displayProperties - The term lists to be displayed - * @param {Property.} allowExponentsProperty - Whether exponents (powers of x) are allowed - */ - function GenericFactorsNode( displayProperties, allowExponentsProperty ) { - const self = this; +import validate from '../../../../axon/js/validate.js'; +import Bounds2 from '../../../../dot/js/Bounds2.js'; +import inherit from '../../../../phet-core/js/inherit.js'; +import Orientation from '../../../../phet-core/js/Orientation.js'; +import MathSymbols from '../../../../scenery-phet/js/MathSymbols.js'; +import AlignBox from '../../../../scenery/js/nodes/AlignBox.js'; +import HBox from '../../../../scenery/js/nodes/HBox.js'; +import Node from '../../../../scenery/js/nodes/Node.js'; +import Rectangle from '../../../../scenery/js/nodes/Rectangle.js'; +import RichText from '../../../../scenery/js/nodes/RichText.js'; +import Text from '../../../../scenery/js/nodes/Text.js'; +import areaModelCommon from '../../areaModelCommon.js'; +import AreaModelCommonConstants from '../../common/AreaModelCommonConstants.js'; +import AreaModelCommonColorProfile from '../../common/view/AreaModelCommonColorProfile.js'; + +// constants +const BOX_SIZE = 30; +const PAREN_BOUNDS = new Text( ')(', { + font: AreaModelCommonConstants.FACTORS_PAREN_FONT, + boundsMethod: 'accurate' +} ).bounds; - const readouts = displayProperties.map( function( displayProperty, orientation ) { - return self.createOrientationReadout( orientation, displayProperty ); - } ); +/** + * @constructor + * @extends {HBox} + * + * @param {OrientationPair.>} displayProperties - The term lists to be displayed + * @param {Property.} allowExponentsProperty - Whether exponents (powers of x) are allowed + */ +function GenericFactorsNode( displayProperties, allowExponentsProperty ) { + const self = this; - const leftParenText = new Text( '(', { font: AreaModelCommonConstants.FACTORS_PAREN_FONT } ); - const middleParenText = new Text( ')(', { font: AreaModelCommonConstants.FACTORS_PAREN_FONT } ); - const rightParenText = new Text( ')', { font: AreaModelCommonConstants.FACTORS_PAREN_FONT } ); - const xText = new Text( MathSymbols.TIMES, { font: AreaModelCommonConstants.FACTORS_TERM_FONT } ); + const readouts = displayProperties.map( function( displayProperty, orientation ) { + return self.createOrientationReadout( orientation, displayProperty ); + } ); - // Have the X take up at least the same vertical bounds as the parentheses - xText.localBounds = xText.localBounds.union( - new Bounds2( 0, middleParenText.localBounds.minY, 0, middleParenText.localBounds.maxY ) - ); + const leftParenText = new Text( '(', { font: AreaModelCommonConstants.FACTORS_PAREN_FONT } ); + const middleParenText = new Text( ')(', { font: AreaModelCommonConstants.FACTORS_PAREN_FONT } ); + const rightParenText = new Text( ')', { font: AreaModelCommonConstants.FACTORS_PAREN_FONT } ); + const xText = new Text( MathSymbols.TIMES, { font: AreaModelCommonConstants.FACTORS_TERM_FONT } ); - // Center the box vertically, so that when maxWidth kicks in, we stay vertically centered in our area of the box - const box = new HBox( { - spacing: 10, - align: 'origin' - } ); + // Have the X take up at least the same vertical bounds as the parentheses + xText.localBounds = xText.localBounds.union( + new Bounds2( 0, middleParenText.localBounds.minY, 0, middleParenText.localBounds.maxY ) + ); - allowExponentsProperty.link( function( allowExponents ) { - box.children = allowExponents ? [ - leftParenText, - readouts.vertical, - middleParenText, - readouts.horizontal, - rightParenText - ] : [ - readouts.vertical, - xText, - readouts.horizontal - ]; - } ); + // Center the box vertically, so that when maxWidth kicks in, we stay vertically centered in our area of the box + const box = new HBox( { + spacing: 10, + align: 'origin' + } ); - const spacer = new Node(); + allowExponentsProperty.link( function( allowExponents ) { + box.children = allowExponents ? [ + leftParenText, + readouts.vertical, + middleParenText, + readouts.horizontal, + rightParenText + ] : [ + readouts.vertical, + xText, + readouts.horizontal + ]; + } ); - AlignBox.call( this, new Node( { - children: [ box, spacer ], - maxWidth: AreaModelCommonConstants.PANEL_INTERIOR_MAX - } ) ); + const spacer = new Node(); - // Set our alignBounds to the maximum size we can be, so that we remain centered nicely in the accordion box. - allowExponentsProperty.link( function( allowExponents ) { - const maxBounds = Bounds2.NOTHING.copy(); - maxBounds.includeBounds( new RichText( allowExponents ? 'x2' : 'x', { - font: AreaModelCommonConstants.FACTORS_TERM_FONT - } ).bounds ); - maxBounds.includeBounds( PAREN_BOUNDS ); + AlignBox.call( this, new Node( { + children: [ box, spacer ], + maxWidth: AreaModelCommonConstants.PANEL_INTERIOR_MAX + } ) ); - self.alignBounds = new Bounds2( 0, 0, AreaModelCommonConstants.PANEL_INTERIOR_MAX, maxBounds.height ); - spacer.localBounds = new Bounds2( 0, maxBounds.minY, 0, maxBounds.maxY ); - } ); - } + // Set our alignBounds to the maximum size we can be, so that we remain centered nicely in the accordion box. + allowExponentsProperty.link( function( allowExponents ) { + const maxBounds = Bounds2.NOTHING.copy(); + maxBounds.includeBounds( new RichText( allowExponents ? 'x2' : 'x', { + font: AreaModelCommonConstants.FACTORS_TERM_FONT + } ).bounds ); + maxBounds.includeBounds( PAREN_BOUNDS ); - areaModelCommon.register( 'GenericFactorsNode', GenericFactorsNode ); - - return inherit( AlignBox, GenericFactorsNode, { - /** - * Creates a readout for the total sum for a particular orientation. - * @private - * - * @param {Orientation} orientation - * @param {Property.} displayProperty - * @returns {Node} - */ - createOrientationReadout: function( orientation, displayProperty ) { - validate( orientation, { validValues: Orientation.VALUES } ); - - const colorProperty = AreaModelCommonColorProfile.genericColorProperties.get( orientation ); - - const richText = new RichText( '', { - font: AreaModelCommonConstants.FACTORS_TERM_FONT, - fill: colorProperty - } ); - - const box = new Rectangle( 0, 0, BOX_SIZE, BOX_SIZE, { - stroke: colorProperty, - centerY: PAREN_BOUNDS.centerY // So that it is perfectly vertically aligned - } ); - - const node = new Node(); - displayProperty.link( function( termList ) { - if ( termList === null ) { - node.children = [ box ]; - } - else { - richText.text = termList.toRichString(); - node.children = [ richText ]; - } - } ); - - return node; - } + self.alignBounds = new Bounds2( 0, 0, AreaModelCommonConstants.PANEL_INTERIOR_MAX, maxBounds.height ); + spacer.localBounds = new Bounds2( 0, maxBounds.minY, 0, maxBounds.maxY ); } ); -} ); +} + +areaModelCommon.register( 'GenericFactorsNode', GenericFactorsNode ); + +export default inherit( AlignBox, GenericFactorsNode, { + /** + * Creates a readout for the total sum for a particular orientation. + * @private + * + * @param {Orientation} orientation + * @param {Property.} displayProperty + * @returns {Node} + */ + createOrientationReadout: function( orientation, displayProperty ) { + validate( orientation, { validValues: Orientation.VALUES } ); + + const colorProperty = AreaModelCommonColorProfile.genericColorProperties.get( orientation ); + + const richText = new RichText( '', { + font: AreaModelCommonConstants.FACTORS_TERM_FONT, + fill: colorProperty + } ); + + const box = new Rectangle( 0, 0, BOX_SIZE, BOX_SIZE, { + stroke: colorProperty, + centerY: PAREN_BOUNDS.centerY // So that it is perfectly vertically aligned + } ); + + const node = new Node(); + displayProperty.link( function( termList ) { + if ( termList === null ) { + node.children = [ box ]; + } + else { + richText.text = termList.toRichString(); + node.children = [ richText ]; + } + } ); + + return node; + } +} ); \ No newline at end of file diff --git a/js/generic/view/GenericLayoutSelectionNode.js b/js/generic/view/GenericLayoutSelectionNode.js index f663f344..54ec203a 100644 --- a/js/generic/view/GenericLayoutSelectionNode.js +++ b/js/generic/view/GenericLayoutSelectionNode.js @@ -9,248 +9,245 @@ * * @author Jonathan Olson */ -define( require => { - 'use strict'; - // modules - const areaModelCommon = require( 'AREA_MODEL_COMMON/areaModelCommon' ); - const AreaModelCommonColorProfile = require( 'AREA_MODEL_COMMON/common/view/AreaModelCommonColorProfile' ); - const AreaModelCommonConstants = require( 'AREA_MODEL_COMMON/common/AreaModelCommonConstants' ); - const BooleanProperty = require( 'AXON/BooleanProperty' ); - const DerivedProperty = require( 'AXON/DerivedProperty' ); - const FireListener = require( 'SCENERY/listeners/FireListener' ); - const GenericLayout = require( 'AREA_MODEL_COMMON/generic/model/GenericLayout' ); - const HBox = require( 'SCENERY/nodes/HBox' ); - const inherit = require( 'PHET_CORE/inherit' ); - const Line = require( 'SCENERY/nodes/Line' ); - const Node = require( 'SCENERY/nodes/Node' ); - const Path = require( 'SCENERY/nodes/Path' ); - const Rectangle = require( 'SCENERY/nodes/Rectangle' ); - const Shape = require( 'KITE/Shape' ); - const Text = require( 'SCENERY/nodes/Text' ); - const VBox = require( 'SCENERY/nodes/VBox' ); +import BooleanProperty from '../../../../axon/js/BooleanProperty.js'; +import DerivedProperty from '../../../../axon/js/DerivedProperty.js'; +import Shape from '../../../../kite/js/Shape.js'; +import inherit from '../../../../phet-core/js/inherit.js'; +import FireListener from '../../../../scenery/js/listeners/FireListener.js'; +import HBox from '../../../../scenery/js/nodes/HBox.js'; +import Line from '../../../../scenery/js/nodes/Line.js'; +import Node from '../../../../scenery/js/nodes/Node.js'; +import Path from '../../../../scenery/js/nodes/Path.js'; +import Rectangle from '../../../../scenery/js/nodes/Rectangle.js'; +import Text from '../../../../scenery/js/nodes/Text.js'; +import VBox from '../../../../scenery/js/nodes/VBox.js'; +import areaModelCommon from '../../areaModelCommon.js'; +import AreaModelCommonConstants from '../../common/AreaModelCommonConstants.js'; +import AreaModelCommonColorProfile from '../../common/view/AreaModelCommonColorProfile.js'; +import GenericLayout from '../model/GenericLayout.js'; - /** - * @constructor - * @extends {Node} - * - * @param {Property.} genericLayoutProperty - * @param {Node} listParent - * @param {number} width - */ - function GenericLayoutSelectionNode( genericLayoutProperty, listParent, width ) { - Node.call( this ); - - const self = this; - - // Our rectangles will be stroked, so we need to subtract 1 due to the lineWidth - width -= 1; - - const comboBoxItems = GenericLayout.VALUES.map( function( layout ) { - return { - node: new HBox( { - children: [ - createLayoutIcon( layout.size, 0.7 ), - new Text( layout.size.height + 'x' + layout.size.width, { - font: AreaModelCommonConstants.LAYOUT_FONT - } ) - ], - spacing: 14 - } ), - value: layout - }; - } ); - - const maxItemHeight = Math.max.apply( Math, _.map( _.map( comboBoxItems, 'node' ), 'height' ) ); - const itemMargin = 6; - const arrowMargin = 8; - - const rectHeight = maxItemHeight + 2 * itemMargin; - const rectangle = new Rectangle( { - rectWidth: width, - rectHeight: maxItemHeight + 2 * itemMargin, - fill: 'white', - stroke: 'black', - cornerRadius: AreaModelCommonConstants.PANEL_CORNER_RADIUS, - cursor: 'pointer' - } ); - this.addChild( rectangle ); - - const arrowSize = 15; - const arrow = new Path( new Shape().moveTo( 0, 0 ).lineTo( arrowSize, 0 ).lineTo( arrowSize * 0.5, arrowSize * 0.9 ).close(), { - fill: 'black', - right: rectangle.right - arrowMargin, - centerY: rectangle.centerY, - pickable: false - } ); - this.addChild( arrow ); - - const separatorX = arrow.left - arrowMargin; - this.addChild( new Line( { - x1: separatorX, - y1: 0, - x2: separatorX, - y2: rectHeight, - lineWidth: 0.5, - stroke: 'black', - pickable: false - } ) ); - - const currentLabel = new Node( { - pickable: false - } ); - genericLayoutProperty.link( function( layout ) { - currentLabel.children = [ - _.find( comboBoxItems, function( item ) { - return item.value === layout; - } ).node - ]; - currentLabel.left = itemMargin; - currentLabel.centerY = rectangle.centerY; - } ); - this.addChild( currentLabel ); - - const popup = new Rectangle( { - rectWidth: separatorX, - rectHeight: separatorX, - fill: 'white', - stroke: 'black', - cornerRadius: AreaModelCommonConstants.PANEL_CORNER_RADIUS, - pickable: true - } ); - - const buttonSpacing = 12; - const buttonsNode = new VBox( { - children: [ 1, 2, 3 ].map( function( numVertical ) { - return new HBox( { - children: [ 1, 2, 3 ].map( function( numHorizontal ) { - const layout = GenericLayout.fromValues( numHorizontal, numVertical ); - // NOTE: Yes, it's weird this constant is here. We used to scale most things down by this amount. Now we - // want the same appearance (but without the scaling, because it was bad practice), so to get the icon to - // have the same appearance, a scale factor is needed. - const oldScale = 0.7; - const icon = createLayoutIcon( layout.size, oldScale * oldScale ); - icon.scale( 1 / oldScale ); - icon.pickable = false; - const cornerRadius = 3; - const background = Rectangle.roundedBounds( icon.bounds.dilated( cornerRadius ), cornerRadius, cornerRadius, { - cursor: 'pointer' +/** + * @constructor + * @extends {Node} + * + * @param {Property.} genericLayoutProperty + * @param {Node} listParent + * @param {number} width + */ +function GenericLayoutSelectionNode( genericLayoutProperty, listParent, width ) { + Node.call( this ); + + const self = this; + + // Our rectangles will be stroked, so we need to subtract 1 due to the lineWidth + width -= 1; + + const comboBoxItems = GenericLayout.VALUES.map( function( layout ) { + return { + node: new HBox( { + children: [ + createLayoutIcon( layout.size, 0.7 ), + new Text( layout.size.height + 'x' + layout.size.width, { + font: AreaModelCommonConstants.LAYOUT_FONT + } ) + ], + spacing: 14 + } ), + value: layout + }; + } ); + + const maxItemHeight = Math.max.apply( Math, _.map( _.map( comboBoxItems, 'node' ), 'height' ) ); + const itemMargin = 6; + const arrowMargin = 8; + + const rectHeight = maxItemHeight + 2 * itemMargin; + const rectangle = new Rectangle( { + rectWidth: width, + rectHeight: maxItemHeight + 2 * itemMargin, + fill: 'white', + stroke: 'black', + cornerRadius: AreaModelCommonConstants.PANEL_CORNER_RADIUS, + cursor: 'pointer' + } ); + this.addChild( rectangle ); + + const arrowSize = 15; + const arrow = new Path( new Shape().moveTo( 0, 0 ).lineTo( arrowSize, 0 ).lineTo( arrowSize * 0.5, arrowSize * 0.9 ).close(), { + fill: 'black', + right: rectangle.right - arrowMargin, + centerY: rectangle.centerY, + pickable: false + } ); + this.addChild( arrow ); + + const separatorX = arrow.left - arrowMargin; + this.addChild( new Line( { + x1: separatorX, + y1: 0, + x2: separatorX, + y2: rectHeight, + lineWidth: 0.5, + stroke: 'black', + pickable: false + } ) ); + + const currentLabel = new Node( { + pickable: false + } ); + genericLayoutProperty.link( function( layout ) { + currentLabel.children = [ + _.find( comboBoxItems, function( item ) { + return item.value === layout; + } ).node + ]; + currentLabel.left = itemMargin; + currentLabel.centerY = rectangle.centerY; + } ); + this.addChild( currentLabel ); + + const popup = new Rectangle( { + rectWidth: separatorX, + rectHeight: separatorX, + fill: 'white', + stroke: 'black', + cornerRadius: AreaModelCommonConstants.PANEL_CORNER_RADIUS, + pickable: true + } ); + + const buttonSpacing = 12; + const buttonsNode = new VBox( { + children: [ 1, 2, 3 ].map( function( numVertical ) { + return new HBox( { + children: [ 1, 2, 3 ].map( function( numHorizontal ) { + const layout = GenericLayout.fromValues( numHorizontal, numVertical ); + // NOTE: Yes, it's weird this constant is here. We used to scale most things down by this amount. Now we + // want the same appearance (but without the scaling, because it was bad practice), so to get the icon to + // have the same appearance, a scale factor is needed. + const oldScale = 0.7; + const icon = createLayoutIcon( layout.size, oldScale * oldScale ); + icon.scale( 1 / oldScale ); + icon.pickable = false; + const cornerRadius = 3; + const background = Rectangle.roundedBounds( icon.bounds.dilated( cornerRadius ), cornerRadius, cornerRadius, { + cursor: 'pointer' + } ); + background.touchArea = background.localBounds.dilated( buttonSpacing / 2 ); + const listener = new FireListener( { + fire: function() { + genericLayoutProperty.value = layout; + visibleProperty.value = false; // hide + } + } ); + background.stroke = new DerivedProperty( + [ genericLayoutProperty, AreaModelCommonColorProfile.radioBorderProperty ], + function( currentLayout, highlightColor ) { + if ( currentLayout === layout ) { + return highlightColor; + } + else { + return 'transparent'; + } } ); - background.touchArea = background.localBounds.dilated( buttonSpacing / 2 ); - const listener = new FireListener( { - fire: function() { - genericLayoutProperty.value = layout; - visibleProperty.value = false; // hide + background.fill = new DerivedProperty( + [ listener.isHoveringProperty, AreaModelCommonColorProfile.layoutHoverProperty ], + function( isHovering, hoverColor ) { + if ( isHovering ) { + return hoverColor; + } + else { + return 'transparent'; } } ); - background.stroke = new DerivedProperty( - [ genericLayoutProperty, AreaModelCommonColorProfile.radioBorderProperty ], - function( currentLayout, highlightColor ) { - if ( currentLayout === layout ) { - return highlightColor; - } - else { - return 'transparent'; - } - } ); - background.fill = new DerivedProperty( - [ listener.isHoveringProperty, AreaModelCommonColorProfile.layoutHoverProperty ], - function( isHovering, hoverColor ) { - if ( isHovering ) { - return hoverColor; - } - else { - return 'transparent'; - } - } ); - return new Node( { - children: [ background, icon ], - inputListeners: [ listener ] - } ); - } ), - spacing: buttonSpacing - } ); - } ), - spacing: buttonSpacing - } ); - const panelMargin = 20; - buttonsNode.scale( ( popup.width - 2 * panelMargin ) / buttonsNode.width ); - buttonsNode.center = popup.center; - popup.addChild( buttonsNode ); + return new Node( { + children: [ background, icon ], + inputListeners: [ listener ] + } ); + } ), + spacing: buttonSpacing + } ); + } ), + spacing: buttonSpacing + } ); + const panelMargin = 20; + buttonsNode.scale( ( popup.width - 2 * panelMargin ) / buttonsNode.width ); + buttonsNode.center = popup.center; + popup.addChild( buttonsNode ); + + var visibleProperty = new BooleanProperty( false ); + popup.addInputListener( { + down: function( event ) { + event.handle(); + } + } ); - var visibleProperty = new BooleanProperty( false ); - popup.addInputListener( { - down: function( event ) { - event.handle(); + // Handle dismissing the selection if the user clicks outside + const dismissListener = { + down: function( event ) { + if ( !event.trail.isExtensionOf( self.getUniqueTrail() ) ) { + visibleProperty.value = false; } - } ); + } + }; + visibleProperty.lazyLink( function( visible ) { + if ( visible ) { + const matrix = self.getUniqueTrail().getMatrixTo( listParent.getUniqueTrail() ); + popup.setScaleMagnitude( matrix.getScaleVector().x ); + // We subtract 1 off so that the strokes line up, and we don't get a "double-stroked" effect. + popup.leftTop = matrix.timesVector2( rectangle.leftBottom.plusXY( 0, -1 ) ); + listParent.addChild( popup ); + + phet.joist.display.addInputListener( dismissListener ); + } + else { + listParent.removeChild( popup ); - // Handle dismissing the selection if the user clicks outside - const dismissListener = { - down: function( event ) { - if ( !event.trail.isExtensionOf( self.getUniqueTrail() ) ) { - visibleProperty.value = false; - } - } - }; - visibleProperty.lazyLink( function( visible ) { - if ( visible ) { - const matrix = self.getUniqueTrail().getMatrixTo( listParent.getUniqueTrail() ); - popup.setScaleMagnitude( matrix.getScaleVector().x ); - // We subtract 1 off so that the strokes line up, and we don't get a "double-stroked" effect. - popup.leftTop = matrix.timesVector2( rectangle.leftBottom.plusXY( 0, -1 ) ); - listParent.addChild( popup ); + phet.joist.display.removeInputListener( dismissListener ); + } + } ); - phet.joist.display.addInputListener( dismissListener ); - } - else { - listParent.removeChild( popup ); + rectangle.addInputListener( { + down: function( event ) { + visibleProperty.toggle(); + } + } ); +} - phet.joist.display.removeInputListener( dismissListener ); - } - } ); +areaModelCommon.register( 'GenericLayoutSelectionNode', GenericLayoutSelectionNode ); - rectangle.addInputListener( { - down: function( event ) { - visibleProperty.toggle(); - } - } ); +/** + * Creates a layout icon based on the given size. + * @private + * + * @param {Dimension2} size + * @param {number} lineWidth + * @returns {Node} + */ +function createLayoutIcon( size, lineWidth ) { + const length = 21; + const shape = new Shape().rect( 0, 0, length, length ); + if ( size.width === 2 ) { + shape.moveTo( length * AreaModelCommonConstants.GENERIC_ICON_SINGLE_OFFSET, 0 ).verticalLineTo( length ); } - - areaModelCommon.register( 'GenericLayoutSelectionNode', GenericLayoutSelectionNode ); - - /** - * Creates a layout icon based on the given size. - * @private - * - * @param {Dimension2} size - * @param {number} lineWidth - * @returns {Node} - */ - function createLayoutIcon( size, lineWidth ) { - const length = 21; - const shape = new Shape().rect( 0, 0, length, length ); - if ( size.width === 2 ) { - shape.moveTo( length * AreaModelCommonConstants.GENERIC_ICON_SINGLE_OFFSET, 0 ).verticalLineTo( length ); - } - else if ( size.width === 3 ) { - shape.moveTo( length * AreaModelCommonConstants.GENERIC_ICON_FIRST_OFFSET, 0 ).verticalLineTo( length ); - shape.moveTo( length * AreaModelCommonConstants.GENERIC_ICON_SECOND_OFFSET, 0 ).verticalLineTo( length ); - } - if ( size.height === 2 ) { - shape.moveTo( 0, length * AreaModelCommonConstants.GENERIC_ICON_SINGLE_OFFSET ).horizontalLineTo( length ); - } - else if ( size.height === 3 ) { - shape.moveTo( 0, length * AreaModelCommonConstants.GENERIC_ICON_FIRST_OFFSET ).horizontalLineTo( length ); - shape.moveTo( 0, length * AreaModelCommonConstants.GENERIC_ICON_SECOND_OFFSET ).horizontalLineTo( length ); - } - return new Path( shape, { - lineWidth: lineWidth, - stroke: AreaModelCommonColorProfile.layoutGridProperty, - fill: AreaModelCommonColorProfile.layoutIconFillProperty - } ); + else if ( size.width === 3 ) { + shape.moveTo( length * AreaModelCommonConstants.GENERIC_ICON_FIRST_OFFSET, 0 ).verticalLineTo( length ); + shape.moveTo( length * AreaModelCommonConstants.GENERIC_ICON_SECOND_OFFSET, 0 ).verticalLineTo( length ); } - - return inherit( Node, GenericLayoutSelectionNode ); -} ); + if ( size.height === 2 ) { + shape.moveTo( 0, length * AreaModelCommonConstants.GENERIC_ICON_SINGLE_OFFSET ).horizontalLineTo( length ); + } + else if ( size.height === 3 ) { + shape.moveTo( 0, length * AreaModelCommonConstants.GENERIC_ICON_FIRST_OFFSET ).horizontalLineTo( length ); + shape.moveTo( 0, length * AreaModelCommonConstants.GENERIC_ICON_SECOND_OFFSET ).horizontalLineTo( length ); + } + return new Path( shape, { + lineWidth: lineWidth, + stroke: AreaModelCommonColorProfile.layoutGridProperty, + fill: AreaModelCommonColorProfile.layoutIconFillProperty + } ); +} + +inherit( Node, GenericLayoutSelectionNode ); +export default GenericLayoutSelectionNode; \ No newline at end of file diff --git a/js/generic/view/GenericPartitionedAreaNode.js b/js/generic/view/GenericPartitionedAreaNode.js index a3789b0b..53b0ae3e 100644 --- a/js/generic/view/GenericPartitionedAreaNode.js +++ b/js/generic/view/GenericPartitionedAreaNode.js @@ -7,79 +7,76 @@ * * @author Jonathan Olson */ -define( require => { - 'use strict'; - // modules - const areaModelCommon = require( 'AREA_MODEL_COMMON/areaModelCommon' ); - const AreaModelCommonColorProfile = require( 'AREA_MODEL_COMMON/common/view/AreaModelCommonColorProfile' ); - const DynamicProperty = require( 'AXON/DynamicProperty' ); - const inherit = require( 'PHET_CORE/inherit' ); - const Property = require( 'AXON/Property' ); - const Rectangle = require( 'SCENERY/nodes/Rectangle' ); +import DynamicProperty from '../../../../axon/js/DynamicProperty.js'; +import Property from '../../../../axon/js/Property.js'; +import inherit from '../../../../phet-core/js/inherit.js'; +import Rectangle from '../../../../scenery/js/nodes/Rectangle.js'; +import areaModelCommon from '../../areaModelCommon.js'; +import AreaModelCommonColorProfile from '../../common/view/AreaModelCommonColorProfile.js'; - /** - * @constructor - * @extends {Node} - * - * @param {Property.} partitionedAreaProperty - * @param {Property.} modelViewTransformProperty - */ - function GenericPartitionedAreaNode( partitionedAreaProperty, modelViewTransformProperty ) { - const self = this; +/** + * @constructor + * @extends {Node} + * + * @param {Property.} partitionedAreaProperty + * @param {Property.} modelViewTransformProperty + */ +function GenericPartitionedAreaNode( partitionedAreaProperty, modelViewTransformProperty ) { + const self = this; - // We'll set the fill/size/etc. below. - Rectangle.call( this, {} ); + // We'll set the fill/size/etc. below. + Rectangle.call( this, {} ); - // @public {Property.} - Exposed so it can be set later - this.partitionedAreaProperty = partitionedAreaProperty; + // @public {Property.} - Exposed so it can be set later + this.partitionedAreaProperty = partitionedAreaProperty; - // Fill - new DynamicProperty( partitionedAreaProperty, { - derive: 'areaProperty' - } ).link( function( area ) { - if ( area === null || area.coefficient === 0 ) { - self.fill = null; - } - else if ( area.coefficient > 0 ) { - self.fill = AreaModelCommonColorProfile.genericPositiveBackgroundProperty; - } - else { - self.fill = AreaModelCommonColorProfile.genericNegativeBackgroundProperty; - } - } ); + // Fill + new DynamicProperty( partitionedAreaProperty, { + derive: 'areaProperty' + } ).link( function( area ) { + if ( area === null || area.coefficient === 0 ) { + self.fill = null; + } + else if ( area.coefficient > 0 ) { + self.fill = AreaModelCommonColorProfile.genericPositiveBackgroundProperty; + } + else { + self.fill = AreaModelCommonColorProfile.genericNegativeBackgroundProperty; + } + } ); - // Visibility - new DynamicProperty( partitionedAreaProperty, { - derive: 'visibleProperty', - defaultValue: false - } ).linkAttribute( this, 'visible' ); + // Visibility + new DynamicProperty( partitionedAreaProperty, { + derive: 'visibleProperty', + defaultValue: false + } ).linkAttribute( this, 'visible' ); - // Adjust our rectangle dimension/location so that we take up the bounds defined by the partitioned area. Our area - // can change, so we need to swap out or multilink when the area changes (kept so we can dispose it) - let rangeMultilinks = null; // {OrientationPair.|null} - partitionedAreaProperty.link( function( partitionedArea ) { - // Release any previous references - rangeMultilinks && rangeMultilinks.forEach( function( rangeMultilink ) { - rangeMultilink.dispose(); - } ); - rangeMultilinks = null; - if ( partitionedArea ) { - rangeMultilinks = partitionedArea.partitions.map( function( partition, orientation ) { - return Property.multilink( - [ partition.coordinateRangeProperty, modelViewTransformProperty ], - function( range, modelViewTransform ) { - if ( range !== null ) { - self[ orientation.rectCoordinate ] = modelViewTransform.modelToViewX( range.min ); - self[ orientation.rectSize ] = modelViewTransform.modelToViewX( range.getLength() ); - } - } ); - } ); - } + // Adjust our rectangle dimension/location so that we take up the bounds defined by the partitioned area. Our area + // can change, so we need to swap out or multilink when the area changes (kept so we can dispose it) + let rangeMultilinks = null; // {OrientationPair.|null} + partitionedAreaProperty.link( function( partitionedArea ) { + // Release any previous references + rangeMultilinks && rangeMultilinks.forEach( function( rangeMultilink ) { + rangeMultilink.dispose(); } ); - } + rangeMultilinks = null; + if ( partitionedArea ) { + rangeMultilinks = partitionedArea.partitions.map( function( partition, orientation ) { + return Property.multilink( + [ partition.coordinateRangeProperty, modelViewTransformProperty ], + function( range, modelViewTransform ) { + if ( range !== null ) { + self[ orientation.rectCoordinate ] = modelViewTransform.modelToViewX( range.min ); + self[ orientation.rectSize ] = modelViewTransform.modelToViewX( range.getLength() ); + } + } ); + } ); + } + } ); +} - areaModelCommon.register( 'GenericPartitionedAreaNode', GenericPartitionedAreaNode ); +areaModelCommon.register( 'GenericPartitionedAreaNode', GenericPartitionedAreaNode ); - return inherit( Rectangle, GenericPartitionedAreaNode ); -} ); +inherit( Rectangle, GenericPartitionedAreaNode ); +export default GenericPartitionedAreaNode; \ No newline at end of file diff --git a/js/generic/view/PartitionSizeEditNode.js b/js/generic/view/PartitionSizeEditNode.js index 72ead35b..67686071 100644 --- a/js/generic/view/PartitionSizeEditNode.js +++ b/js/generic/view/PartitionSizeEditNode.js @@ -7,92 +7,89 @@ * * @author Jonathan Olson */ -define( require => { - 'use strict'; - // modules - const areaModelCommon = require( 'AREA_MODEL_COMMON/areaModelCommon' ); - const AreaModelCommonConstants = require( 'AREA_MODEL_COMMON/common/AreaModelCommonConstants' ); - const Color = require( 'SCENERY/util/Color' ); - const DerivedProperty = require( 'AXON/DerivedProperty' ); - const DynamicProperty = require( 'AXON/DynamicProperty' ); - const inherit = require( 'PHET_CORE/inherit' ); - const Orientation = require( 'PHET_CORE/Orientation' ); - const Property = require( 'AXON/Property' ); - const TermEditNode = require( 'AREA_MODEL_COMMON/generic/view/TermEditNode' ); +import DerivedProperty from '../../../../axon/js/DerivedProperty.js'; +import DynamicProperty from '../../../../axon/js/DynamicProperty.js'; +import Property from '../../../../axon/js/Property.js'; +import inherit from '../../../../phet-core/js/inherit.js'; +import Orientation from '../../../../phet-core/js/Orientation.js'; +import Color from '../../../../scenery/js/util/Color.js'; +import areaModelCommon from '../../areaModelCommon.js'; +import AreaModelCommonConstants from '../../common/AreaModelCommonConstants.js'; +import TermEditNode from './TermEditNode.js'; - /** - * @constructor - * @extends {Node} - * - * @param {Property.} activePartitionProperty - * @param {Property.} partitionProperty - * @param {Property.} modelViewTransformProperty - * @param {boolean} allowExponents - Whether exponents (powers of x) are allowed - */ - function PartitionSizeEditNode( activePartitionProperty, partitionProperty, modelViewTransformProperty, allowExponents ) { - const self = this; +/** + * @constructor + * @extends {Node} + * + * @param {Property.} activePartitionProperty + * @param {Property.} partitionProperty + * @param {Property.} modelViewTransformProperty + * @param {boolean} allowExponents - Whether exponents (powers of x) are allowed + */ +function PartitionSizeEditNode( activePartitionProperty, partitionProperty, modelViewTransformProperty, allowExponents ) { + const self = this; - // @public {Property.} - Exposed so it can be changed after creation in pooling. - this.partitionProperty = partitionProperty; + // @public {Property.} - Exposed so it can be changed after creation in pooling. + this.partitionProperty = partitionProperty; - const orientationProperty = new DerivedProperty( [ partitionProperty ], function( partition ) { - return partition ? partition.orientation : Orientation.HORIZONTAL; // Default if we have none - } ); - const sizeProperty = new DynamicProperty( partitionProperty, { derive: 'sizeProperty' } ); - const colorProperty = new DynamicProperty( partitionProperty, { - derive: 'colorProperty', - defaultValue: Color.MAGENTA // Should not see this, but need a valid color - } ); + const orientationProperty = new DerivedProperty( [ partitionProperty ], function( partition ) { + return partition ? partition.orientation : Orientation.HORIZONTAL; // Default if we have none + } ); + const sizeProperty = new DynamicProperty( partitionProperty, { derive: 'sizeProperty' } ); + const colorProperty = new DynamicProperty( partitionProperty, { + derive: 'colorProperty', + defaultValue: Color.MAGENTA // Should not see this, but need a valid color + } ); - TermEditNode.call( this, orientationProperty, sizeProperty, { - textColorProperty: colorProperty, - borderColorProperty: colorProperty, - isActiveProperty: new DerivedProperty( - [ activePartitionProperty, partitionProperty ], - function( activePartition, partition ) { - return activePartition === partition; - } ), - digitCountProperty: new DerivedProperty( [ partitionProperty ], function( partition ) { - return partition ? partition.digitCount : 1; // Default if we have none + TermEditNode.call( this, orientationProperty, sizeProperty, { + textColorProperty: colorProperty, + borderColorProperty: colorProperty, + isActiveProperty: new DerivedProperty( + [ activePartitionProperty, partitionProperty ], + function( activePartition, partition ) { + return activePartition === partition; } ), - allowExponentsProperty: new Property( allowExponents ), - editCallback: function() { - if ( activePartitionProperty.value !== partitionProperty.value ) { - activePartitionProperty.value = partitionProperty.value; - } - else { - // Pressing on the edit button when that keypad is already open will instead close the keypad. - // See https://github.com/phetsims/area-model-common/issues/127 - activePartitionProperty.value = null; - } + digitCountProperty: new DerivedProperty( [ partitionProperty ], function( partition ) { + return partition ? partition.digitCount : 1; // Default if we have none + } ), + allowExponentsProperty: new Property( allowExponents ), + editCallback: function() { + if ( activePartitionProperty.value !== partitionProperty.value ) { + activePartitionProperty.value = partitionProperty.value; } - } ); - - // Primary orientation (location of range center) - const coordinateRangeProperty = new DynamicProperty( partitionProperty, { derive: 'coordinateRangeProperty' } ); - Property.multilink( - [ partitionProperty, coordinateRangeProperty, modelViewTransformProperty ], - function( partition, range, modelViewTransform ) { - if ( range && partition ) { - self[ partition.orientation.centerCoordinate ] = partition.orientation.modelToView( modelViewTransform, range.getCenter() ); - } - } ); + else { + // Pressing on the edit button when that keypad is already open will instead close the keypad. + // See https://github.com/phetsims/area-model-common/issues/127 + activePartitionProperty.value = null; + } + } + } ); - // Secondary (offsets) - partitionProperty.link( function( partition ) { - if ( partition ) { - self[ partition.orientation.opposite.centerCoordinate ] = AreaModelCommonConstants.PARTITION_OFFSET.get( partition.orientation ); + // Primary orientation (location of range center) + const coordinateRangeProperty = new DynamicProperty( partitionProperty, { derive: 'coordinateRangeProperty' } ); + Property.multilink( + [ partitionProperty, coordinateRangeProperty, modelViewTransformProperty ], + function( partition, range, modelViewTransform ) { + if ( range && partition ) { + self[ partition.orientation.centerCoordinate ] = partition.orientation.modelToView( modelViewTransform, range.getCenter() ); } } ); - new DynamicProperty( partitionProperty, { - derive: 'visibleProperty', - defaultValue: false - } ).linkAttribute( this, 'visible' ); - } + // Secondary (offsets) + partitionProperty.link( function( partition ) { + if ( partition ) { + self[ partition.orientation.opposite.centerCoordinate ] = AreaModelCommonConstants.PARTITION_OFFSET.get( partition.orientation ); + } + } ); + + new DynamicProperty( partitionProperty, { + derive: 'visibleProperty', + defaultValue: false + } ).linkAttribute( this, 'visible' ); +} - areaModelCommon.register( 'PartitionSizeEditNode', PartitionSizeEditNode ); +areaModelCommon.register( 'PartitionSizeEditNode', PartitionSizeEditNode ); - return inherit( TermEditNode, PartitionSizeEditNode ); -} ); +inherit( TermEditNode, PartitionSizeEditNode ); +export default PartitionSizeEditNode; \ No newline at end of file diff --git a/js/generic/view/TermAccumulator.js b/js/generic/view/TermAccumulator.js index 55a8aca9..c76854c5 100644 --- a/js/generic/view/TermAccumulator.js +++ b/js/generic/view/TermAccumulator.js @@ -7,195 +7,191 @@ * * @author Jonathan Olson { - 'use strict'; - - // modules - const AbstractKeyAccumulator = require( 'SCENERY_PHET/keypad/AbstractKeyAccumulator' ); - const areaModelCommon = require( 'AREA_MODEL_COMMON/areaModelCommon' ); - const AreaModelCommonConstants = require( 'AREA_MODEL_COMMON/common/AreaModelCommonConstants' ); - const DerivedProperty = require( 'AXON/DerivedProperty' ); - const inherit = require( 'PHET_CORE/inherit' ); - const KeyID = require( 'SCENERY_PHET/keypad/KeyID' ); - const MathSymbols = require( 'SCENERY_PHET/MathSymbols' ); - const Term = require( 'AREA_MODEL_COMMON/common/model/Term' ); - - // constants - const NONZERO_DIGIT_STRINGS = _.range( 1, 10 ).map( function( n ) { return '' + n; } ); - const DIGIT_STRINGS = _.range( 0, 10 ).map( function( n ) { return '' + n; } ); + +import DerivedProperty from '../../../../axon/js/DerivedProperty.js'; +import inherit from '../../../../phet-core/js/inherit.js'; +import AbstractKeyAccumulator from '../../../../scenery-phet/js/keypad/AbstractKeyAccumulator.js'; +import KeyID from '../../../../scenery-phet/js/keypad/KeyID.js'; +import MathSymbols from '../../../../scenery-phet/js/MathSymbols.js'; +import areaModelCommon from '../../areaModelCommon.js'; +import AreaModelCommonConstants from '../../common/AreaModelCommonConstants.js'; +import Term from '../../common/model/Term.js'; + +// constants +const NONZERO_DIGIT_STRINGS = _.range( 1, 10 ).map( function( n ) { return '' + n; } ); +const DIGIT_STRINGS = _.range( 0, 10 ).map( function( n ) { return '' + n; } ); + +/** + * @constructor + * @extends {AbstractKeyAccumulator} + * + * @param {Property.} digitCountProperty + */ +function TermAccumulator( digitCountProperty ) { /** - * @constructor - * @extends {AbstractKeyAccumulator} + * Whether a set of proposed keys is allowed, see https://github.com/phetsims/area-model-common/issues/138 + * @public + * @override * - * @param {Property.} digitCountProperty + * @param {Array.} proposedKeys + * @returns {boolean} */ - function TermAccumulator( digitCountProperty ) { - - /** - * Whether a set of proposed keys is allowed, see https://github.com/phetsims/area-model-common/issues/138 - * @public - * @override - * - * @param {Array.} proposedKeys - * @returns {boolean} - */ - this.defaultValidator = function( proposedKeys ) { - let xCount = 0; - let digitCount = 0; - - - proposedKeys.forEach( function( key ) { - if ( key === KeyID.X || key === KeyID.X_SQUARED ) { - xCount++; - } - - if ( _.includes( DIGIT_STRINGS, key ) ) { - digitCount++; - } - } ); - - return xCount <= 1 && digitCount <= digitCountProperty.value; - }; - - // Validators to be passed to AbstractKeyAccumulator - const validators = [ this.defaultValidator ]; - - AbstractKeyAccumulator.call( this, validators ); - - // @public {Property.} - For display - this.richStringProperty = new DerivedProperty( [ this.accumulatedKeysProperty ], function( accumulatedKeys ) { - return accumulatedKeys.map( function( key ) { - if ( key === KeyID.PLUS_MINUS ) { - return MathSymbols.UNARY_MINUS; - } - else if ( key === KeyID.X ) { - return AreaModelCommonConstants.X_VARIABLE_RICH_STRING; - } - else if ( key === KeyID.X_SQUARED ) { - return AreaModelCommonConstants.X_VARIABLE_RICH_STRING + '2'; - } - else { - return key; - } - } ).join( '' ); - } ); + this.defaultValidator = function( proposedKeys ) { + let xCount = 0; + let digitCount = 0; - // @public {Property.} - The term used if 'enter' is pressed - this.termProperty = new DerivedProperty( [ this.accumulatedKeysProperty ], function( accumulatedKeys ) { - const lastKey = accumulatedKeys[ accumulatedKeys.length - 1 ]; - let coefficient = 1; - let power = 0; - if ( lastKey === KeyID.X ) { - power = 1; - accumulatedKeys = accumulatedKeys.slice( 0, accumulatedKeys.length - 1 ); + proposedKeys.forEach( function( key ) { + if ( key === KeyID.X || key === KeyID.X_SQUARED ) { + xCount++; } - else if ( lastKey === KeyID.X_SQUARED ) { - power = 2; - accumulatedKeys = accumulatedKeys.slice( 0, accumulatedKeys.length - 1 ); - } - if ( accumulatedKeys[ 0 ] === KeyID.PLUS_MINUS ) { - accumulatedKeys = accumulatedKeys.slice( 1 ); - // handle -x - if ( accumulatedKeys.length === 0 ) { - coefficient = -1; - } - else { - accumulatedKeys = [ '-' ].concat( accumulatedKeys ); - } + if ( _.includes( DIGIT_STRINGS, key ) ) { + digitCount++; } + } ); + + return xCount <= 1 && digitCount <= digitCountProperty.value; + }; + + // Validators to be passed to AbstractKeyAccumulator + const validators = [ this.defaultValidator ]; - const digitString = accumulatedKeys.join( '' ); - if ( digitString === '' || digitString === '-' ) { - if ( power === 0 ) { - return null; - } + AbstractKeyAccumulator.call( this, validators ); + + // @public {Property.} - For display + this.richStringProperty = new DerivedProperty( [ this.accumulatedKeysProperty ], function( accumulatedKeys ) { + return accumulatedKeys.map( function( key ) { + if ( key === KeyID.PLUS_MINUS ) { + return MathSymbols.UNARY_MINUS; + } + else if ( key === KeyID.X ) { + return AreaModelCommonConstants.X_VARIABLE_RICH_STRING; + } + else if ( key === KeyID.X_SQUARED ) { + return AreaModelCommonConstants.X_VARIABLE_RICH_STRING + '2'; } else { - coefficient = parseInt( digitString, 10 ); + return key; } + } ).join( '' ); + } ); - return new Term( coefficient, power ); - } ); - } + // @public {Property.} - The term used if 'enter' is pressed + this.termProperty = new DerivedProperty( [ this.accumulatedKeysProperty ], function( accumulatedKeys ) { + const lastKey = accumulatedKeys[ accumulatedKeys.length - 1 ]; - areaModelCommon.register( 'TermAccumulator', TermAccumulator ); - - return inherit( AbstractKeyAccumulator, TermAccumulator, { - /** - * Handles what happens when a key is pressed and create proposed set of keys to be passed to Validator - * @public - * @override - * - * @param {KeyID} keyIdentifier - identifier for the key pressed - */ - handleKeyPressed: function( keyIdentifier ) { - - const currentKeys = this.accumulatedKeysProperty.get(); - - // Whether we have a negative sign in our current input - let negative = _.includes( currentKeys, KeyID.PLUS_MINUS ); - - // The power of x (X or X_SQUARED) in our input (otherwise undefined). This keypad only allows one "power" of X, - // e.g. 0, 1 or 2 (corresponding to multiplying times 1, x, x^2). This is the corresponding key for that power. - let power = _.find( currentKeys, function( key ) { - return key === KeyID.X || key === KeyID.X_SQUARED; - } ); - - // All of the digits in our current input. (just numerical parts, not powers of x or negative signs) - let digits = currentKeys.filter( function( key ) { - return _.includes( DIGIT_STRINGS, key ); - } ); - - // Helpful booleans for what our pressed key is. - const isDigit = _.includes( NONZERO_DIGIT_STRINGS, keyIdentifier ); - const isZero = keyIdentifier === KeyID.ZERO; - const isBackspace = keyIdentifier === KeyID.BACKSPACE; - const isPlusMinus = keyIdentifier === KeyID.PLUS_MINUS; - const isX = keyIdentifier === KeyID.X; - const isXSquared = keyIdentifier === KeyID.X_SQUARED; - - if ( isBackspace ) { - if ( power ) { - power = null; - } - else if ( digits.length ) { - digits.pop(); - } - else { - negative = false; - } - } - else if ( isX || isXSquared ) { - if ( !power ) { - power = keyIdentifier; - } - } - else if ( isPlusMinus ) { - negative = !negative; - } - else if ( isZero ) { - if ( digits[ 0 ] !== KeyID.ZERO ) { - digits.push( keyIdentifier ); - } - } - else if ( isDigit ) { - if ( digits[ 0 ] === KeyID.ZERO ) { - digits = [ keyIdentifier ]; - } - else { - digits.push( keyIdentifier ); - } + let coefficient = 1; + let power = 0; + if ( lastKey === KeyID.X ) { + power = 1; + accumulatedKeys = accumulatedKeys.slice( 0, accumulatedKeys.length - 1 ); + } + else if ( lastKey === KeyID.X_SQUARED ) { + power = 2; + accumulatedKeys = accumulatedKeys.slice( 0, accumulatedKeys.length - 1 ); + } + if ( accumulatedKeys[ 0 ] === KeyID.PLUS_MINUS ) { + accumulatedKeys = accumulatedKeys.slice( 1 ); + + // handle -x + if ( accumulatedKeys.length === 0 ) { + coefficient = -1; } else { - throw new Error( 'unknown digit: ' + keyIdentifier ); + accumulatedKeys = [ '-' ].concat( accumulatedKeys ); } + } - // Validate and update the keys - const proposedKeys = ( negative ? [ KeyID.PLUS_MINUS ] : [] ).concat( digits ).concat( power ? [ power ] : [] ); - this.validateKeys( proposedKeys ) && this.updateKeys( proposedKeys ); + const digitString = accumulatedKeys.join( '' ); + if ( digitString === '' || digitString === '-' ) { + if ( power === 0 ) { + return null; + } + } + else { + coefficient = parseInt( digitString, 10 ); } + + return new Term( coefficient, power ); } ); -} ); +} + +areaModelCommon.register( 'TermAccumulator', TermAccumulator ); + +export default inherit( AbstractKeyAccumulator, TermAccumulator, { + /** + * Handles what happens when a key is pressed and create proposed set of keys to be passed to Validator + * @public + * @override + * + * @param {KeyID} keyIdentifier - identifier for the key pressed + */ + handleKeyPressed: function( keyIdentifier ) { + + const currentKeys = this.accumulatedKeysProperty.get(); + + // Whether we have a negative sign in our current input + let negative = _.includes( currentKeys, KeyID.PLUS_MINUS ); + + // The power of x (X or X_SQUARED) in our input (otherwise undefined). This keypad only allows one "power" of X, + // e.g. 0, 1 or 2 (corresponding to multiplying times 1, x, x^2). This is the corresponding key for that power. + let power = _.find( currentKeys, function( key ) { + return key === KeyID.X || key === KeyID.X_SQUARED; + } ); + + // All of the digits in our current input. (just numerical parts, not powers of x or negative signs) + let digits = currentKeys.filter( function( key ) { + return _.includes( DIGIT_STRINGS, key ); + } ); + + // Helpful booleans for what our pressed key is. + const isDigit = _.includes( NONZERO_DIGIT_STRINGS, keyIdentifier ); + const isZero = keyIdentifier === KeyID.ZERO; + const isBackspace = keyIdentifier === KeyID.BACKSPACE; + const isPlusMinus = keyIdentifier === KeyID.PLUS_MINUS; + const isX = keyIdentifier === KeyID.X; + const isXSquared = keyIdentifier === KeyID.X_SQUARED; + + if ( isBackspace ) { + if ( power ) { + power = null; + } + else if ( digits.length ) { + digits.pop(); + } + else { + negative = false; + } + } + else if ( isX || isXSquared ) { + if ( !power ) { + power = keyIdentifier; + } + } + else if ( isPlusMinus ) { + negative = !negative; + } + else if ( isZero ) { + if ( digits[ 0 ] !== KeyID.ZERO ) { + digits.push( keyIdentifier ); + } + } + else if ( isDigit ) { + if ( digits[ 0 ] === KeyID.ZERO ) { + digits = [ keyIdentifier ]; + } + else { + digits.push( keyIdentifier ); + } + } + else { + throw new Error( 'unknown digit: ' + keyIdentifier ); + } + + // Validate and update the keys + const proposedKeys = ( negative ? [ KeyID.PLUS_MINUS ] : [] ).concat( digits ).concat( power ? [ power ] : [] ); + this.validateKeys( proposedKeys ) && this.updateKeys( proposedKeys ); + } +} ); \ No newline at end of file diff --git a/js/generic/view/TermEditNode.js b/js/generic/view/TermEditNode.js index 5b3868c4..b8a8aad7 100644 --- a/js/generic/view/TermEditNode.js +++ b/js/generic/view/TermEditNode.js @@ -7,140 +7,137 @@ * * @author Jonathan Olson */ -define( require => { - 'use strict'; - - // modules - const areaModelCommon = require( 'AREA_MODEL_COMMON/areaModelCommon' ); - const AreaModelCommonColorProfile = require( 'AREA_MODEL_COMMON/common/view/AreaModelCommonColorProfile' ); - const AreaModelCommonConstants = require( 'AREA_MODEL_COMMON/common/AreaModelCommonConstants' ); - const BooleanProperty = require( 'AXON/BooleanProperty' ); - const Color = require( 'SCENERY/util/Color' ); - const FireListener = require( 'SCENERY/listeners/FireListener' ); - const FontAwesomeNode = require( 'SUN/FontAwesomeNode' ); - const inherit = require( 'PHET_CORE/inherit' ); - const LayoutBox = require( 'SCENERY/nodes/LayoutBox' ); - const merge = require( 'PHET_CORE/merge' ); - const NumberProperty = require( 'AXON/NumberProperty' ); - const Property = require( 'AXON/Property' ); - const Rectangle = require( 'SCENERY/nodes/Rectangle' ); - const RectangularPushButton = require( 'SUN/buttons/RectangularPushButton' ); - const RichText = require( 'SCENERY/nodes/RichText' ); - const Term = require( 'AREA_MODEL_COMMON/common/model/Term' ); - - /** - * @constructor - * @extends {Node} - * - * @param {Property.} orientationProperty - * @param {Property.} termProperty - * @param {Object} [options] - */ - function TermEditNode( orientationProperty, termProperty, options ) { - assert && assert( orientationProperty instanceof Property ); - assert && assert( termProperty instanceof Property ); - - options = merge( { - // {Property.} - The color of the readout text - textColorProperty: new Property( Color.BLACK ), - - // {Property.} - The color of the border around the readout - borderColorProperty: new Property( Color.BLACK ), - - // {Property.} - Whether this term is the one being edited right now - isActiveProperty: new BooleanProperty( false ), - - // {Property.} - How many digits are allowed to be used for this term - digitCountProperty: new NumberProperty( 1 ), - - // {Property.} - Whether exponents are allowed for this term - allowExponentsProperty: new BooleanProperty( false ), - - // {function} - Called with no arguments when this term should start being edited - editCallback: _.noop, - - // {Font} - font: AreaModelCommonConstants.TERM_EDIT_READOUT_FONT - }, options ); - - const self = this; - - const readoutText = new RichText( '', { - fill: options.textColorProperty, - font: options.font - } ); - - const readoutBackground = new Rectangle( { - stroke: options.borderColorProperty, - cornerRadius: 4, - children: [ - readoutText - ], - // Allow clicking the readout to edit, see https://github.com/phetsims/area-model-common/issues/23 - cursor: 'pointer', - inputListeners: [ - new FireListener( { - fire: options.editCallback - } ) - ] - } ); - - LayoutBox.call( this, { - orientation: orientationProperty.value.layoutBoxOrientation, - spacing: 4, - children: [ - readoutBackground, - new RectangularPushButton( { - content: new FontAwesomeNode( 'pencil_square_o', { - scale: 0.4, - xMargin: 6, - yMargin: 4 - } ), - listener: function() { - options.editCallback(); - }, - baseColor: AreaModelCommonColorProfile.editButtonBackgroundProperty - } ) - ] - } ); - orientationProperty.link( function( orientation ) { - self.orientation = orientation.layoutBoxOrientation; - } ); - - options.isActiveProperty.link( function( isActive ) { - readoutBackground.fill = isActive - ? AreaModelCommonColorProfile.editActiveBackgroundProperty - : AreaModelCommonColorProfile.editInactiveBackgroundProperty; - } ); - - function updateText() { - if ( termProperty.value === null ) { - readoutText.text = ''; - } - else { - readoutText.text = termProperty.value.toRichString( false ); - } - - readoutText.center = readoutBackground.selfBounds.center; - readoutBackground.touchArea = readoutBackground.parentToLocalBounds( self.localBounds ).dilated( 6 ); - } - function updateDigits() { - readoutText.text = Term.getLargestGenericString( options.allowExponentsProperty.value, options.digitCountProperty.value ); - readoutBackground.rectWidth = readoutText.width + 5; - readoutBackground.rectHeight = readoutText.height + 5; - updateText(); +import BooleanProperty from '../../../../axon/js/BooleanProperty.js'; +import NumberProperty from '../../../../axon/js/NumberProperty.js'; +import Property from '../../../../axon/js/Property.js'; +import inherit from '../../../../phet-core/js/inherit.js'; +import merge from '../../../../phet-core/js/merge.js'; +import FireListener from '../../../../scenery/js/listeners/FireListener.js'; +import LayoutBox from '../../../../scenery/js/nodes/LayoutBox.js'; +import Rectangle from '../../../../scenery/js/nodes/Rectangle.js'; +import RichText from '../../../../scenery/js/nodes/RichText.js'; +import Color from '../../../../scenery/js/util/Color.js'; +import RectangularPushButton from '../../../../sun/js/buttons/RectangularPushButton.js'; +import FontAwesomeNode from '../../../../sun/js/FontAwesomeNode.js'; +import areaModelCommon from '../../areaModelCommon.js'; +import AreaModelCommonConstants from '../../common/AreaModelCommonConstants.js'; +import Term from '../../common/model/Term.js'; +import AreaModelCommonColorProfile from '../../common/view/AreaModelCommonColorProfile.js'; + +/** + * @constructor + * @extends {Node} + * + * @param {Property.} orientationProperty + * @param {Property.} termProperty + * @param {Object} [options] + */ +function TermEditNode( orientationProperty, termProperty, options ) { + assert && assert( orientationProperty instanceof Property ); + assert && assert( termProperty instanceof Property ); + + options = merge( { + // {Property.} - The color of the readout text + textColorProperty: new Property( Color.BLACK ), + + // {Property.} - The color of the border around the readout + borderColorProperty: new Property( Color.BLACK ), + + // {Property.} - Whether this term is the one being edited right now + isActiveProperty: new BooleanProperty( false ), + + // {Property.} - How many digits are allowed to be used for this term + digitCountProperty: new NumberProperty( 1 ), + + // {Property.} - Whether exponents are allowed for this term + allowExponentsProperty: new BooleanProperty( false ), + + // {function} - Called with no arguments when this term should start being edited + editCallback: _.noop, + + // {Font} + font: AreaModelCommonConstants.TERM_EDIT_READOUT_FONT + }, options ); + + const self = this; + + const readoutText = new RichText( '', { + fill: options.textColorProperty, + font: options.font + } ); + + const readoutBackground = new Rectangle( { + stroke: options.borderColorProperty, + cornerRadius: 4, + children: [ + readoutText + ], + // Allow clicking the readout to edit, see https://github.com/phetsims/area-model-common/issues/23 + cursor: 'pointer', + inputListeners: [ + new FireListener( { + fire: options.editCallback + } ) + ] + } ); + + LayoutBox.call( this, { + orientation: orientationProperty.value.layoutBoxOrientation, + spacing: 4, + children: [ + readoutBackground, + new RectangularPushButton( { + content: new FontAwesomeNode( 'pencil_square_o', { + scale: 0.4, + xMargin: 6, + yMargin: 4 + } ), + listener: function() { + options.editCallback(); + }, + baseColor: AreaModelCommonColorProfile.editButtonBackgroundProperty + } ) + ] + } ); + orientationProperty.link( function( orientation ) { + self.orientation = orientation.layoutBoxOrientation; + } ); + + options.isActiveProperty.link( function( isActive ) { + readoutBackground.fill = isActive + ? AreaModelCommonColorProfile.editActiveBackgroundProperty + : AreaModelCommonColorProfile.editInactiveBackgroundProperty; + } ); + + function updateText() { + if ( termProperty.value === null ) { + readoutText.text = ''; + } + else { + readoutText.text = termProperty.value.toRichString( false ); } - termProperty.lazyLink( updateText ); - options.digitCountProperty.lazyLink( updateDigits ); - options.allowExponentsProperty.lazyLink( updateDigits ); - orientationProperty.lazyLink( updateDigits ); + readoutText.center = readoutBackground.selfBounds.center; + readoutBackground.touchArea = readoutBackground.parentToLocalBounds( self.localBounds ).dilated( 6 ); + } - updateDigits(); + function updateDigits() { + readoutText.text = Term.getLargestGenericString( options.allowExponentsProperty.value, options.digitCountProperty.value ); + readoutBackground.rectWidth = readoutText.width + 5; + readoutBackground.rectHeight = readoutText.height + 5; + updateText(); } - areaModelCommon.register( 'TermEditNode', TermEditNode ); + termProperty.lazyLink( updateText ); + options.digitCountProperty.lazyLink( updateDigits ); + options.allowExponentsProperty.lazyLink( updateDigits ); + orientationProperty.lazyLink( updateDigits ); + + updateDigits(); +} + +areaModelCommon.register( 'TermEditNode', TermEditNode ); - return inherit( LayoutBox, TermEditNode ); -} ); +inherit( LayoutBox, TermEditNode ); +export default TermEditNode; \ No newline at end of file diff --git a/js/generic/view/TermKeypadPanel.js b/js/generic/view/TermKeypadPanel.js index 647606b4..ec4bbba1 100644 --- a/js/generic/view/TermKeypadPanel.js +++ b/js/generic/view/TermKeypadPanel.js @@ -7,174 +7,170 @@ * * @author Jonathan Olson */ -define( require => { - 'use strict'; - - // modules - const areaModelCommon = require( 'AREA_MODEL_COMMON/areaModelCommon' ); - const AreaModelCommonColorProfile = require( 'AREA_MODEL_COMMON/common/view/AreaModelCommonColorProfile' ); - const AreaModelCommonConstants = require( 'AREA_MODEL_COMMON/common/AreaModelCommonConstants' ); - const BackspaceIcon = require( 'SCENERY_PHET/BackspaceIcon' ); - const inherit = require( 'PHET_CORE/inherit' ); - const Key = require( 'SCENERY_PHET/keypad/Key' ); - const KeyID = require( 'SCENERY_PHET/keypad/KeyID' ); - const Keypad = require( 'SCENERY_PHET/keypad/Keypad' ); - const MathSymbols = require( 'SCENERY_PHET/MathSymbols' ); - const Node = require( 'SCENERY/nodes/Node' ); - const Panel = require( 'SUN/Panel' ); - const Rectangle = require( 'SCENERY/nodes/Rectangle' ); - const RectangularPushButton = require( 'SUN/buttons/RectangularPushButton' ); - const RichText = require( 'SCENERY/nodes/RichText' ); - const Term = require( 'AREA_MODEL_COMMON/common/model/Term' ); - const TermAccumulator = require( 'AREA_MODEL_COMMON/generic/view/TermAccumulator' ); - const Text = require( 'SCENERY/nodes/Text' ); - const VBox = require( 'SCENERY/nodes/VBox' ); - - // strings - const enterString = require( 'string!AREA_MODEL_COMMON/enter' ); - - // layout constants - const positiveKeys = [ - [ - new Key( '7', KeyID.SEVEN ), - new Key( '8', KeyID.EIGHT ), - new Key( '9', KeyID.NINE ) - ], - [ - new Key( '4', KeyID.FOUR ), - new Key( '5', KeyID.FIVE ), - new Key( '6', KeyID.SIX ) - ], - [ - new Key( '1', KeyID.ONE ), - new Key( '2', KeyID.TWO ), - new Key( '3', KeyID.THREE ) - ] - ]; - const zeroAndBackspace = [ - new Key( '0', KeyID.ZERO ), - new Key( ( new BackspaceIcon( { scale: 1.5 } ) ), KeyID.BACKSPACE ) - ]; - const noExponentLayout = positiveKeys.concat( [ - [ new Key( MathSymbols.PLUS + '/' + MathSymbols.MINUS, KeyID.PLUS_MINUS ) ].concat( zeroAndBackspace ) - ] ); - const noNegativeLayout = positiveKeys.concat( [ - [ null ].concat( zeroAndBackspace ) - ] ); - const exponentLayout = noExponentLayout.concat( [ - [ - null, - new Key( new RichText( AreaModelCommonConstants.X_VARIABLE_RICH_STRING + '2', { font: AreaModelCommonConstants.KEYPAD_FONT } ), KeyID.X_SQUARED ), - new Key( new RichText( AreaModelCommonConstants.X_VARIABLE_RICH_STRING, { font: AreaModelCommonConstants.KEYPAD_FONT } ), KeyID.X ) - ] - ] ); - /** - * @constructor - * @extends {Panel} - * - * @param {Property.} digitCountProperty - * @param {boolean} allowExponents - Whether exponents (powers of x) are allowed - * @param {boolean} allowNegative - * @param {function} enterCallback - function( {Term|null} ) - The entered term, or null if there is no valid term entered. - * @param {Object} [nodeOptions] - */ - function TermKeypadPanel( digitCountProperty, allowExponents, allowNegative, enterCallback, nodeOptions ) { - assert && assert( allowNegative || !allowExponents, 'We have no non-negative exponent keyboard layout' ); - - // Handles logic for key-presses and conversion to strings/Terms. - const termAccumulator = new TermAccumulator( digitCountProperty ); - - // @private {Keypad} - this.keypad = new Keypad( allowExponents ? exponentLayout : ( allowNegative ? noExponentLayout : noNegativeLayout ), { - accumulator: termAccumulator - } ); - - const readoutBackground = new Rectangle( { - fill: AreaModelCommonColorProfile.keypadReadoutBackgroundProperty, - stroke: AreaModelCommonColorProfile.keypadReadoutBorderProperty, - cornerRadius: AreaModelCommonConstants.PANEL_CORNER_RADIUS - } ); - - const readoutTextOptions = { - font: AreaModelCommonConstants.KEYPAD_READOUT_FONT - }; - - const readoutText = new RichText( '', readoutTextOptions ); - - function updateText( string ) { - // Trick to be able to position an empty string - readoutText.visible = string.length > 0; - if ( readoutText.visible ) { - readoutText.text = string; - readoutText.centerX = readoutBackground.centerX; - } - } +import inherit from '../../../../phet-core/js/inherit.js'; +import BackspaceIcon from '../../../../scenery-phet/js/BackspaceIcon.js'; +import Key from '../../../../scenery-phet/js/keypad/Key.js'; +import KeyID from '../../../../scenery-phet/js/keypad/KeyID.js'; +import Keypad from '../../../../scenery-phet/js/keypad/Keypad.js'; +import MathSymbols from '../../../../scenery-phet/js/MathSymbols.js'; +import Node from '../../../../scenery/js/nodes/Node.js'; +import Rectangle from '../../../../scenery/js/nodes/Rectangle.js'; +import RichText from '../../../../scenery/js/nodes/RichText.js'; +import Text from '../../../../scenery/js/nodes/Text.js'; +import VBox from '../../../../scenery/js/nodes/VBox.js'; +import RectangularPushButton from '../../../../sun/js/buttons/RectangularPushButton.js'; +import Panel from '../../../../sun/js/Panel.js'; +import areaModelCommonStrings from '../../area-model-common-strings.js'; +import areaModelCommon from '../../areaModelCommon.js'; +import AreaModelCommonConstants from '../../common/AreaModelCommonConstants.js'; +import Term from '../../common/model/Term.js'; +import AreaModelCommonColorProfile from '../../common/view/AreaModelCommonColorProfile.js'; +import TermAccumulator from './TermAccumulator.js'; + +const enterString = areaModelCommonStrings.enter; + +// layout constants +const positiveKeys = [ + [ + new Key( '7', KeyID.SEVEN ), + new Key( '8', KeyID.EIGHT ), + new Key( '9', KeyID.NINE ) + ], + [ + new Key( '4', KeyID.FOUR ), + new Key( '5', KeyID.FIVE ), + new Key( '6', KeyID.SIX ) + ], + [ + new Key( '1', KeyID.ONE ), + new Key( '2', KeyID.TWO ), + new Key( '3', KeyID.THREE ) + ] +]; +const zeroAndBackspace = [ + new Key( '0', KeyID.ZERO ), + new Key( ( new BackspaceIcon( { scale: 1.5 } ) ), KeyID.BACKSPACE ) +]; +const noExponentLayout = positiveKeys.concat( [ + [ new Key( MathSymbols.PLUS + '/' + MathSymbols.MINUS, KeyID.PLUS_MINUS ) ].concat( zeroAndBackspace ) +] ); +const noNegativeLayout = positiveKeys.concat( [ + [ null ].concat( zeroAndBackspace ) +] ); +const exponentLayout = noExponentLayout.concat( [ + [ + null, + new Key( new RichText( AreaModelCommonConstants.X_VARIABLE_RICH_STRING + '2', { font: AreaModelCommonConstants.KEYPAD_FONT } ), KeyID.X_SQUARED ), + new Key( new RichText( AreaModelCommonConstants.X_VARIABLE_RICH_STRING, { font: AreaModelCommonConstants.KEYPAD_FONT } ), KeyID.X ) + ] +] ); + +/** + * @constructor + * @extends {Panel} + * + * @param {Property.} digitCountProperty + * @param {boolean} allowExponents - Whether exponents (powers of x) are allowed + * @param {boolean} allowNegative + * @param {function} enterCallback - function( {Term|null} ) - The entered term, or null if there is no valid term entered. + * @param {Object} [nodeOptions] + */ +function TermKeypadPanel( digitCountProperty, allowExponents, allowNegative, enterCallback, nodeOptions ) { + assert && assert( allowNegative || !allowExponents, 'We have no non-negative exponent keyboard layout' ); - // Update the text when the accumulator's string output changes - termAccumulator.richStringProperty.link( updateText ); + // Handles logic for key-presses and conversion to strings/Terms. + const termAccumulator = new TermAccumulator( digitCountProperty ); - // When the active partition changes, resize the background to fit to the largest size. - digitCountProperty.link( function( digitCount ) { - // Temporarily use a different string - readoutText.text = Term.getLargestGenericString( allowExponents, digitCount ); + // @private {Keypad} + this.keypad = new Keypad( allowExponents ? exponentLayout : ( allowNegative ? noExponentLayout : noNegativeLayout ), { + accumulator: termAccumulator + } ); - // Update the background - readoutBackground.setRectBounds( readoutText.bounds.dilatedXY( 10, 1 ) ); + const readoutBackground = new Rectangle( { + fill: AreaModelCommonColorProfile.keypadReadoutBackgroundProperty, + stroke: AreaModelCommonColorProfile.keypadReadoutBorderProperty, + cornerRadius: AreaModelCommonConstants.PANEL_CORNER_RADIUS + } ); - // Reposition our text - readoutText.center = readoutBackground.center; + const readoutTextOptions = { + font: AreaModelCommonConstants.KEYPAD_READOUT_FONT + }; - // Reset the text value back to what it should be. - updateText( termAccumulator.richStringProperty.value ); - } ); + const readoutText = new RichText( '', readoutTextOptions ); - Panel.call( this, new VBox( { - children: [ - new Node( { - // We position the text over the background manually - children: [ - readoutBackground, - readoutText - ] - } ), - this.keypad, - new RectangularPushButton( { - content: new Text( enterString, { - font: AreaModelCommonConstants.KEYPAD_FONT, - maxWidth: 100 - } ), - touchAreaXDilation: 5, - touchAreaYDilation: 5, - xMargin: 15, - yMargin: 5, - listener: function() { - enterCallback( termAccumulator.termProperty.value ); - }, - baseColor: AreaModelCommonColorProfile.keypadEnterBackgroundProperty - } ) - ], - spacing: 10 - } ), { - cornerRadius: AreaModelCommonConstants.PANEL_CORNER_RADIUS, - xMargin: 15, - yMargin: 15, - fill: AreaModelCommonColorProfile.keypadPanelBackgroundProperty, - stroke: AreaModelCommonColorProfile.keypadPanelBorderProperty - } ); - - this.mutate( nodeOptions ); + function updateText( string ) { + // Trick to be able to position an empty string + readoutText.visible = string.length > 0; + if ( readoutText.visible ) { + readoutText.text = string; + readoutText.centerX = readoutBackground.centerX; + } } - areaModelCommon.register( 'TermKeypadPanel', TermKeypadPanel ); + // Update the text when the accumulator's string output changes + termAccumulator.richStringProperty.link( updateText ); - return inherit( Panel, TermKeypadPanel, { - /** - * Clears the keypad's content. - * @public - */ - clear: function() { - this.keypad.clear(); - } + // When the active partition changes, resize the background to fit to the largest size. + digitCountProperty.link( function( digitCount ) { + // Temporarily use a different string + readoutText.text = Term.getLargestGenericString( allowExponents, digitCount ); + + // Update the background + readoutBackground.setRectBounds( readoutText.bounds.dilatedXY( 10, 1 ) ); + + // Reposition our text + readoutText.center = readoutBackground.center; + + // Reset the text value back to what it should be. + updateText( termAccumulator.richStringProperty.value ); + } ); + + Panel.call( this, new VBox( { + children: [ + new Node( { + // We position the text over the background manually + children: [ + readoutBackground, + readoutText + ] + } ), + this.keypad, + new RectangularPushButton( { + content: new Text( enterString, { + font: AreaModelCommonConstants.KEYPAD_FONT, + maxWidth: 100 + } ), + touchAreaXDilation: 5, + touchAreaYDilation: 5, + xMargin: 15, + yMargin: 5, + listener: function() { + enterCallback( termAccumulator.termProperty.value ); + }, + baseColor: AreaModelCommonColorProfile.keypadEnterBackgroundProperty + } ) + ], + spacing: 10 + } ), { + cornerRadius: AreaModelCommonConstants.PANEL_CORNER_RADIUS, + xMargin: 15, + yMargin: 15, + fill: AreaModelCommonColorProfile.keypadPanelBackgroundProperty, + stroke: AreaModelCommonColorProfile.keypadPanelBorderProperty } ); -} ); + + this.mutate( nodeOptions ); +} + +areaModelCommon.register( 'TermKeypadPanel', TermKeypadPanel ); + +export default inherit( Panel, TermKeypadPanel, { + /** + * Clears the keypad's content. + * @public + */ + clear: function() { + this.keypad.clear(); + } +} ); \ No newline at end of file diff --git a/js/proportional/model/PartitionLineChoice.js b/js/proportional/model/PartitionLineChoice.js index 52bc1666..0e2114bf 100644 --- a/js/proportional/model/PartitionLineChoice.js +++ b/js/proportional/model/PartitionLineChoice.js @@ -5,29 +5,25 @@ * * @author Jonathan Olson */ -define( require => { - 'use strict'; - // modules - const areaModelCommon = require( 'AREA_MODEL_COMMON/areaModelCommon' ); +import areaModelCommon from '../../areaModelCommon.js'; - const PartitionLineChoice = { - NONE: 'NONE', // No partition lines - ONE: 'ONE', // One at a time (toggles between the two) - BOTH: 'BOTH' // Both partition lines available at all times - }; +const PartitionLineChoice = { + NONE: 'NONE', // No partition lines + ONE: 'ONE', // One at a time (toggles between the two) + BOTH: 'BOTH' // Both partition lines available at all times +}; - areaModelCommon.register( 'PartitionLineChoice', PartitionLineChoice ); +areaModelCommon.register( 'PartitionLineChoice', PartitionLineChoice ); - // @public {Array.} - All values the enumeration can take. - PartitionLineChoice.VALUES = [ - PartitionLineChoice.NONE, - PartitionLineChoice.ONE, - PartitionLineChoice.BOTH - ]; +// @public {Array.} - All values the enumeration can take. +PartitionLineChoice.VALUES = [ + PartitionLineChoice.NONE, + PartitionLineChoice.ONE, + PartitionLineChoice.BOTH +]; - // verify that enumeration is immutable, without the runtime penalty in production code - if ( assert ) { Object.freeze( PartitionLineChoice ); } +// verify that enumeration is immutable, without the runtime penalty in production code +if ( assert ) { Object.freeze( PartitionLineChoice ); } - return PartitionLineChoice; -} ); +export default PartitionLineChoice; \ No newline at end of file diff --git a/js/proportional/model/ProportionalArea.js b/js/proportional/model/ProportionalArea.js index a0dd1669..438a027c 100644 --- a/js/proportional/model/ProportionalArea.js +++ b/js/proportional/model/ProportionalArea.js @@ -5,218 +5,214 @@ * * @author Jonathan Olson */ -define( require => { - 'use strict'; - - // modules - const Area = require( 'AREA_MODEL_COMMON/common/model/Area' ); - const areaModelCommon = require( 'AREA_MODEL_COMMON/areaModelCommon' ); - const AreaModelCommonColorProfile = require( 'AREA_MODEL_COMMON/common/view/AreaModelCommonColorProfile' ); - const BooleanProperty = require( 'AXON/BooleanProperty' ); - const DerivedProperty = require( 'AXON/DerivedProperty' ); - const inherit = require( 'PHET_CORE/inherit' ); - const merge = require( 'PHET_CORE/merge' ); - const NumberProperty = require( 'AXON/NumberProperty' ); - const Orientation = require( 'PHET_CORE/Orientation' ); - const OrientationPair = require( 'AREA_MODEL_COMMON/common/model/OrientationPair' ); - const Partition = require( 'AREA_MODEL_COMMON/common/model/Partition' ); - const PartitionLineChoice = require( 'AREA_MODEL_COMMON/proportional/model/PartitionLineChoice' ); - const Property = require( 'AXON/Property' ); - const Range = require( 'DOT/Range' ); - const Term = require( 'AREA_MODEL_COMMON/common/model/Term' ); - /** - * @constructor - * @extends {Area} - * - * @param {Object} [options] - */ - function ProportionalArea( options ) { - const self = this; - - options = merge( { - maximumSize: 20, // {number} - Maximum size our area can take up - minimumSize: 1, // {number} - Minimum size our area can take up - initialWidth: 1, // {number} - Initial width - initialHeight: 1, // {number} - Initial height - eraseWidth: 1, // {number} - The width that will be set with the erase button - eraseHeight: 1, // {number} - The height that will be set with the erase button - initialHorizontalSplit: 0, // {number} - Initial location (if any) of a horizontal partition split - initialVerticalSplit: 0, // {number} - Initial location (if any) of a vertical partition split - snapSize: 1, // {number} - Smallest unit size (that is snapped to) - partitionSnapSize: 10, // {number} - Smallest left/top partition size - gridSpacing: 1, // {number} - Space between grid lines - smallTileSize: 1, // {number} - Size of the smallest tile available (or for the thin tiles, the shorter length) - largeTileSize: 10, // {number} - Size of the largest tile available (or for the thin tiles, the longer length) - tilesAvailable: true, // {boolean} - Whether tiles can be shown on this area - countingAvailable: false, // {boolean} - Whether numbers can be shown on each grid section - partitionLineChoice: PartitionLineChoice.BOTH // {PartitionLineChoice} - What partition lines are shown - }, options ); - - // @public {OrientationPair.>} - Width/height of the contained area. - this.activeTotalProperties = new OrientationPair( - new NumberProperty( options.initialWidth ), - new NumberProperty( options.initialHeight ) - ); - - // @public {Property.} - If PartitionLineChoice.ONE is active, which partition line is active - this.visiblePartitionOrientationProperty = new Property( Orientation.HORIZONTAL ); - - // @public {OrientationPair.>} - Location of the partition lines - this.partitionSplitProperties = new OrientationPair( - new NumberProperty( options.initialHorizontalSplit ), - new NumberProperty( options.initialVerticalSplit ) - ); - - // @public {OrientationPair.>} - this.partitionSplitUserControlledProperties = new OrientationPair( - new BooleanProperty( false ), - new BooleanProperty( false ) - ); - - // @public {number} - this.maximumSize = options.maximumSize; - this.minimumSize = options.minimumSize; - this.eraseWidth = options.eraseWidth; - this.eraseHeight = options.eraseHeight; - this.snapSize = options.snapSize; - this.partitionSnapSize = options.partitionSnapSize; - this.gridSpacing = options.gridSpacing; - this.smallTileSize = options.smallTileSize; - this.largeTileSize = options.largeTileSize; - - // @public {boolean} - this.tilesAvailable = options.tilesAvailable; - this.countingAvailable = options.countingAvailable; - - // @public {PartitionLineChoice} - this.partitionLineChoice = options.partitionLineChoice; - - // @public {OrientationPair.} - Whether to display arrows next to each partition line that - // indicates it is draggable. - this.hasHintArrows = new OrientationPair( new BooleanProperty( true ), new BooleanProperty( true ) ); - - // @public {OrientationPair.>} - Whether the partition line for each orientation is visible - this.partitionSplitVisibleProperties = OrientationPair.create( function( orientation ) { - return new DerivedProperty( - [ self.activeTotalProperties.get( orientation ), self.visiblePartitionOrientationProperty ], - function( totalSize, visibleOrientation ) { - if ( options.partitionLineChoice === PartitionLineChoice.NONE ) { return false; } - if ( options.partitionLineChoice === PartitionLineChoice.ONE && orientation !== visibleOrientation ) { return false; } - - // Given the number of digits in the decimals sim (with potential future changes), 1e-7 should be sufficiently - // small (but not too small). - return totalSize >= ( self.partitionSnapSize + self.snapSize ) - 1e-7; - } ); - } ); +import BooleanProperty from '../../../../axon/js/BooleanProperty.js'; +import DerivedProperty from '../../../../axon/js/DerivedProperty.js'; +import NumberProperty from '../../../../axon/js/NumberProperty.js'; +import Property from '../../../../axon/js/Property.js'; +import Range from '../../../../dot/js/Range.js'; +import inherit from '../../../../phet-core/js/inherit.js'; +import merge from '../../../../phet-core/js/merge.js'; +import Orientation from '../../../../phet-core/js/Orientation.js'; +import areaModelCommon from '../../areaModelCommon.js'; +import Area from '../../common/model/Area.js'; +import OrientationPair from '../../common/model/OrientationPair.js'; +import Partition from '../../common/model/Partition.js'; +import Term from '../../common/model/Term.js'; +import AreaModelCommonColorProfile from '../../common/view/AreaModelCommonColorProfile.js'; +import PartitionLineChoice from './PartitionLineChoice.js'; - // @public {OrientationPair.>} - Like partitionSplitProperties, but null if the partition line is not visible - this.visiblePartitionLineSplitProperties = OrientationPair.create( function( orientation ) { - return new DerivedProperty( - [ self.partitionSplitProperties.get( orientation ), self.partitionSplitVisibleProperties.get( orientation ) ], - function( partitionSplit, partitionVisible ) { - return partitionVisible ? partitionSplit : null; - } ); - } ); +/** + * @constructor + * @extends {Area} + * + * @param {Object} [options] + */ +function ProportionalArea( options ) { + const self = this; + + options = merge( { + maximumSize: 20, // {number} - Maximum size our area can take up + minimumSize: 1, // {number} - Minimum size our area can take up + initialWidth: 1, // {number} - Initial width + initialHeight: 1, // {number} - Initial height + eraseWidth: 1, // {number} - The width that will be set with the erase button + eraseHeight: 1, // {number} - The height that will be set with the erase button + initialHorizontalSplit: 0, // {number} - Initial location (if any) of a horizontal partition split + initialVerticalSplit: 0, // {number} - Initial location (if any) of a vertical partition split + snapSize: 1, // {number} - Smallest unit size (that is snapped to) + partitionSnapSize: 10, // {number} - Smallest left/top partition size + gridSpacing: 1, // {number} - Space between grid lines + smallTileSize: 1, // {number} - Size of the smallest tile available (or for the thin tiles, the shorter length) + largeTileSize: 10, // {number} - Size of the largest tile available (or for the thin tiles, the longer length) + tilesAvailable: true, // {boolean} - Whether tiles can be shown on this area + countingAvailable: false, // {boolean} - Whether numbers can be shown on each grid section + partitionLineChoice: PartitionLineChoice.BOTH // {PartitionLineChoice} - What partition lines are shown + }, options ); + + // @public {OrientationPair.>} - Width/height of the contained area. + this.activeTotalProperties = new OrientationPair( + new NumberProperty( options.initialWidth ), + new NumberProperty( options.initialHeight ) + ); + + // @public {Property.} - If PartitionLineChoice.ONE is active, which partition line is active + this.visiblePartitionOrientationProperty = new Property( Orientation.HORIZONTAL ); + + // @public {OrientationPair.>} - Location of the partition lines + this.partitionSplitProperties = new OrientationPair( + new NumberProperty( options.initialHorizontalSplit ), + new NumberProperty( options.initialVerticalSplit ) + ); + + // @public {OrientationPair.>} + this.partitionSplitUserControlledProperties = new OrientationPair( + new BooleanProperty( false ), + new BooleanProperty( false ) + ); + + // @public {number} + this.maximumSize = options.maximumSize; + this.minimumSize = options.minimumSize; + this.eraseWidth = options.eraseWidth; + this.eraseHeight = options.eraseHeight; + this.snapSize = options.snapSize; + this.partitionSnapSize = options.partitionSnapSize; + this.gridSpacing = options.gridSpacing; + this.smallTileSize = options.smallTileSize; + this.largeTileSize = options.largeTileSize; + + // @public {boolean} + this.tilesAvailable = options.tilesAvailable; + this.countingAvailable = options.countingAvailable; + + // @public {PartitionLineChoice} + this.partitionLineChoice = options.partitionLineChoice; + + // @public {OrientationPair.} - Whether to display arrows next to each partition line that + // indicates it is draggable. + this.hasHintArrows = new OrientationPair( new BooleanProperty( true ), new BooleanProperty( true ) ); + + // @public {OrientationPair.>} - Whether the partition line for each orientation is visible + this.partitionSplitVisibleProperties = OrientationPair.create( function( orientation ) { + return new DerivedProperty( + [ self.activeTotalProperties.get( orientation ), self.visiblePartitionOrientationProperty ], + function( totalSize, visibleOrientation ) { + if ( options.partitionLineChoice === PartitionLineChoice.NONE ) { return false; } + if ( options.partitionLineChoice === PartitionLineChoice.ONE && orientation !== visibleOrientation ) { return false; } + + // Given the number of digits in the decimals sim (with potential future changes), 1e-7 should be sufficiently + // small (but not too small). + return totalSize >= ( self.partitionSnapSize + self.snapSize ) - 1e-7; + } ); + } ); - const horizontalPartitions = [ - new Partition( Orientation.HORIZONTAL, AreaModelCommonColorProfile.proportionalWidthProperty ), - new Partition( Orientation.HORIZONTAL, AreaModelCommonColorProfile.proportionalWidthProperty ) - ]; - - const verticalPartitions = [ - new Partition( Orientation.VERTICAL, AreaModelCommonColorProfile.proportionalHeightProperty ), - new Partition( Orientation.VERTICAL, AreaModelCommonColorProfile.proportionalHeightProperty ) - ]; - - // @public {OrientationPair.} - The primary (upper/left) and secondary (lower/right) - // partitions, separated out for easy access. - this.primaryPartitions = new OrientationPair( horizontalPartitions[ 0 ], verticalPartitions[ 0 ] ); - this.secondaryPartitions = new OrientationPair( horizontalPartitions[ 1 ], verticalPartitions[ 1 ] ); - - Area.call( - this, - new OrientationPair( horizontalPartitions, verticalPartitions ), - AreaModelCommonColorProfile.proportionalColorProperties, - this.maximumSize, - false - ); - - // Keep partition sizes up-to-date - Orientation.VALUES.forEach( function( orientation ) { - Property.multilink( - [ self.activeTotalProperties.get( orientation ), self.visiblePartitionLineSplitProperties.get( orientation ) ], - function( size, split ) { - // Ignore splits at the boundary or outside our active area. - if ( split <= 0 || split >= size ) { - split = null; - } - - const primaryPartition = self.primaryPartitions.get( orientation ); - const secondaryPartition = self.secondaryPartitions.get( orientation ); - - secondaryPartition.visibleProperty.value = split !== null; - - if ( split ) { - primaryPartition.sizeProperty.value = new Term( split ); - secondaryPartition.sizeProperty.value = new Term( size - split ); - primaryPartition.coordinateRangeProperty.value = new Range( 0, split ); - secondaryPartition.coordinateRangeProperty.value = new Range( split, size ); - } - else { - primaryPartition.sizeProperty.value = new Term( size ); - secondaryPartition.sizeProperty.value = null; - primaryPartition.coordinateRangeProperty.value = new Range( 0, size ); - secondaryPartition.coordinateRangeProperty.value = null; - } - } ); - - // Remove splits that are at or past the current boundary. - self.activeTotalProperties.get( orientation ).link( function( total ) { - if ( self.partitionSplitProperties.get( orientation ).value >= total ) { - self.partitionSplitProperties.get( orientation ).value = self.partitionSplitUserControlledProperties.get( orientation ).value ? total : 0; + // @public {OrientationPair.>} - Like partitionSplitProperties, but null if the partition line is not visible + this.visiblePartitionLineSplitProperties = OrientationPair.create( function( orientation ) { + return new DerivedProperty( + [ self.partitionSplitProperties.get( orientation ), self.partitionSplitVisibleProperties.get( orientation ) ], + function( partitionSplit, partitionVisible ) { + return partitionVisible ? partitionSplit : null; + } ); + } ); + + const horizontalPartitions = [ + new Partition( Orientation.HORIZONTAL, AreaModelCommonColorProfile.proportionalWidthProperty ), + new Partition( Orientation.HORIZONTAL, AreaModelCommonColorProfile.proportionalWidthProperty ) + ]; + + const verticalPartitions = [ + new Partition( Orientation.VERTICAL, AreaModelCommonColorProfile.proportionalHeightProperty ), + new Partition( Orientation.VERTICAL, AreaModelCommonColorProfile.proportionalHeightProperty ) + ]; + + // @public {OrientationPair.} - The primary (upper/left) and secondary (lower/right) + // partitions, separated out for easy access. + this.primaryPartitions = new OrientationPair( horizontalPartitions[ 0 ], verticalPartitions[ 0 ] ); + this.secondaryPartitions = new OrientationPair( horizontalPartitions[ 1 ], verticalPartitions[ 1 ] ); + + Area.call( + this, + new OrientationPair( horizontalPartitions, verticalPartitions ), + AreaModelCommonColorProfile.proportionalColorProperties, + this.maximumSize, + false + ); + + // Keep partition sizes up-to-date + Orientation.VALUES.forEach( function( orientation ) { + Property.multilink( + [ self.activeTotalProperties.get( orientation ), self.visiblePartitionLineSplitProperties.get( orientation ) ], + function( size, split ) { + // Ignore splits at the boundary or outside our active area. + if ( split <= 0 || split >= size ) { + split = null; + } + + const primaryPartition = self.primaryPartitions.get( orientation ); + const secondaryPartition = self.secondaryPartitions.get( orientation ); + + secondaryPartition.visibleProperty.value = split !== null; + + if ( split ) { + primaryPartition.sizeProperty.value = new Term( split ); + secondaryPartition.sizeProperty.value = new Term( size - split ); + primaryPartition.coordinateRangeProperty.value = new Range( 0, split ); + secondaryPartition.coordinateRangeProperty.value = new Range( split, size ); + } + else { + primaryPartition.sizeProperty.value = new Term( size ); + secondaryPartition.sizeProperty.value = null; + primaryPartition.coordinateRangeProperty.value = new Range( 0, size ); + secondaryPartition.coordinateRangeProperty.value = null; } } ); - } ); - } - areaModelCommon.register( 'ProportionalArea', ProportionalArea ); - - return inherit( Area, ProportionalArea, { - /** - * Returns a string like 10x10 that can be used for the size. - * @public - * - * @returns {string} - */ - getDimensionString: function() { - return this.maximumSize + 'x' + this.maximumSize; - }, - - /** - * Resets the area to its initial values. - * @public - * @override - */ - reset: function() { - Area.prototype.reset.call( this ); - - this.hasHintArrows.reset(); - this.partitionSplitProperties.reset(); - this.visiblePartitionOrientationProperty.reset(); - this.activeTotalProperties.reset(); - }, - - /** - * Erase the area to a 1x1, see https://github.com/phetsims/area-model-common/issues/77 - * @public - * @override - */ - erase: function() { - Area.prototype.erase.call( this ); - - this.activeTotalProperties.horizontal.value = this.eraseWidth; - this.activeTotalProperties.vertical.value = this.eraseHeight; - } + // Remove splits that are at or past the current boundary. + self.activeTotalProperties.get( orientation ).link( function( total ) { + if ( self.partitionSplitProperties.get( orientation ).value >= total ) { + self.partitionSplitProperties.get( orientation ).value = self.partitionSplitUserControlledProperties.get( orientation ).value ? total : 0; + } + } ); } ); -} ); +} + +areaModelCommon.register( 'ProportionalArea', ProportionalArea ); + +export default inherit( Area, ProportionalArea, { + /** + * Returns a string like 10x10 that can be used for the size. + * @public + * + * @returns {string} + */ + getDimensionString: function() { + return this.maximumSize + 'x' + this.maximumSize; + }, + + /** + * Resets the area to its initial values. + * @public + * @override + */ + reset: function() { + Area.prototype.reset.call( this ); + + this.hasHintArrows.reset(); + this.partitionSplitProperties.reset(); + this.visiblePartitionOrientationProperty.reset(); + this.activeTotalProperties.reset(); + }, + + /** + * Erase the area to a 1x1, see https://github.com/phetsims/area-model-common/issues/77 + * @public + * @override + */ + erase: function() { + Area.prototype.erase.call( this ); + + this.activeTotalProperties.horizontal.value = this.eraseWidth; + this.activeTotalProperties.vertical.value = this.eraseHeight; + } +} ); \ No newline at end of file diff --git a/js/proportional/model/ProportionalAreaDisplay.js b/js/proportional/model/ProportionalAreaDisplay.js index 4c32db1d..50897011 100644 --- a/js/proportional/model/ProportionalAreaDisplay.js +++ b/js/proportional/model/ProportionalAreaDisplay.js @@ -5,69 +5,66 @@ * * @author Jonathan Olson */ -define( require => { - 'use strict'; - // modules - const AreaDisplay = require( 'AREA_MODEL_COMMON/common/model/AreaDisplay' ); - const areaModelCommon = require( 'AREA_MODEL_COMMON/areaModelCommon' ); - const inherit = require( 'PHET_CORE/inherit' ); +import inherit from '../../../../phet-core/js/inherit.js'; +import areaModelCommon from '../../areaModelCommon.js'; +import AreaDisplay from '../../common/model/AreaDisplay.js'; - /** - * @constructor - * @extends {AreaDisplay} - * - * @param {Property.} areaProperty - */ - function ProportionalAreaDisplay( areaProperty ) { - AreaDisplay.call( this, areaProperty ); +/** + * @constructor + * @extends {AreaDisplay} + * + * @param {Property.} areaProperty + */ +function ProportionalAreaDisplay( areaProperty ) { + AreaDisplay.call( this, areaProperty ); - // @public {OrientationPair.>} - this.activeTotalProperties = this.wrapOrientationPairProperty( _.property( 'activeTotalProperties' ), { - bidirectional: true - } ); + // @public {OrientationPair.>} + this.activeTotalProperties = this.wrapOrientationPairProperty( _.property( 'activeTotalProperties' ), { + bidirectional: true + } ); - // @public {Property.} - this.visiblePartitionOrientationProperty = this.wrapProperty( _.property( 'visiblePartitionOrientationProperty' ) ); + // @public {Property.} + this.visiblePartitionOrientationProperty = this.wrapProperty( _.property( 'visiblePartitionOrientationProperty' ) ); - // @public {OrientationPair.>} - this.partitionSplitProperties = this.wrapOrientationPairProperty( _.property( 'partitionSplitProperties' ), { - bidirectional: true - } ); + // @public {OrientationPair.>} + this.partitionSplitProperties = this.wrapOrientationPairProperty( _.property( 'partitionSplitProperties' ), { + bidirectional: true + } ); - // @public {OrientationPair.>} - this.partitionSplitUserControlledProperties = this.wrapOrientationPairProperty( _.property( 'partitionSplitUserControlledProperties' ), { - bidirectional: true - } ); + // @public {OrientationPair.>} + this.partitionSplitUserControlledProperties = this.wrapOrientationPairProperty( _.property( 'partitionSplitUserControlledProperties' ), { + bidirectional: true + } ); - // @public {Property.} - this.maximumSizeProperty = this.wrapObject( _.property( 'maximumSize' ) ); - this.snapSizeProperty = this.wrapObject( _.property( 'snapSize' ) ); - this.partitionSnapSizeProperty = this.wrapObject( _.property( 'partitionSnapSize' ) ); - this.smallTileSizeProperty = this.wrapObject( _.property( 'smallTileSize' ) ); - this.largeTileSizeProperty = this.wrapObject( _.property( 'largeTileSize' ) ); + // @public {Property.} + this.maximumSizeProperty = this.wrapObject( _.property( 'maximumSize' ) ); + this.snapSizeProperty = this.wrapObject( _.property( 'snapSize' ) ); + this.partitionSnapSizeProperty = this.wrapObject( _.property( 'partitionSnapSize' ) ); + this.smallTileSizeProperty = this.wrapObject( _.property( 'smallTileSize' ) ); + this.largeTileSizeProperty = this.wrapObject( _.property( 'largeTileSize' ) ); - // @public {Property.} - this.tilesAvailableProperty = this.wrapObject( _.property( 'tilesAvailable' ) ); - this.countingAvailableProperty = this.wrapObject( _.property( 'countingAvailable' ) ); + // @public {Property.} + this.tilesAvailableProperty = this.wrapObject( _.property( 'tilesAvailable' ) ); + this.countingAvailableProperty = this.wrapObject( _.property( 'countingAvailable' ) ); - // @public {OrientationPair.>} - this.hasHintArrows = this.wrapOrientationPairProperty( _.property( 'hasHintArrows' ), { - bidirectional: true - } ); + // @public {OrientationPair.>} + this.hasHintArrows = this.wrapOrientationPairProperty( _.property( 'hasHintArrows' ), { + bidirectional: true + } ); - // @public {OrientationPair.>} - this.partitionSplitVisibleProperties = this.wrapOrientationPairProperty( _.property( 'partitionSplitVisibleProperties' ) ); + // @public {OrientationPair.>} + this.partitionSplitVisibleProperties = this.wrapOrientationPairProperty( _.property( 'partitionSplitVisibleProperties' ) ); - // @public {OrientationPair.>} - this.visiblePartitionLineSplitProperties = this.wrapOrientationPairProperty( _.property( 'visiblePartitionLineSplitProperties' ) ); + // @public {OrientationPair.>} + this.visiblePartitionLineSplitProperties = this.wrapOrientationPairProperty( _.property( 'visiblePartitionLineSplitProperties' ) ); - // @public {OrientationPair.>} - this.primaryPartitionsProperty = this.wrapOrientationPair( _.property( 'primaryPartitions' ) ); - this.secondaryPartitionsProperty = this.wrapOrientationPair( _.property( 'secondaryPartitions' ) ); - } + // @public {OrientationPair.>} + this.primaryPartitionsProperty = this.wrapOrientationPair( _.property( 'primaryPartitions' ) ); + this.secondaryPartitionsProperty = this.wrapOrientationPair( _.property( 'secondaryPartitions' ) ); +} - areaModelCommon.register( 'ProportionalAreaDisplay', ProportionalAreaDisplay ); +areaModelCommon.register( 'ProportionalAreaDisplay', ProportionalAreaDisplay ); - return inherit( AreaDisplay, ProportionalAreaDisplay ); -} ); +inherit( AreaDisplay, ProportionalAreaDisplay ); +export default ProportionalAreaDisplay; \ No newline at end of file diff --git a/js/proportional/model/ProportionalAreaModel.js b/js/proportional/model/ProportionalAreaModel.js index 5dbffa6b..5796293d 100644 --- a/js/proportional/model/ProportionalAreaModel.js +++ b/js/proportional/model/ProportionalAreaModel.js @@ -5,106 +5,102 @@ * * @author Jonathan Olson */ -define( require => { - 'use strict'; - - // modules - const areaModelCommon = require( 'AREA_MODEL_COMMON/areaModelCommon' ); - const AreaModelCommonModel = require( 'AREA_MODEL_COMMON/common/model/AreaModelCommonModel' ); - const BooleanProperty = require( 'AXON/BooleanProperty' ); - const inherit = require( 'PHET_CORE/inherit' ); - const merge = require( 'PHET_CORE/merge' ); - const PhetFont = require( 'SCENERY_PHET/PhetFont' ); - const ProportionalArea = require( 'AREA_MODEL_COMMON/proportional/model/ProportionalArea' ); - const ProportionalAreaDisplay = require( 'AREA_MODEL_COMMON/proportional/model/ProportionalAreaDisplay' ); - const TextBounds = require( 'SCENERY/util/TextBounds' ); - const Utils = require( 'DOT/Utils' ); - - // constants - const TEST_FONT = new PhetFont( 12 ); - /** - * @constructor - * @extends {AreaModelCommonModel} - * - * @param {Array.} areaOptionObjects - An array of options objects to be passed to the ProportionalArea - * constructors. - * @param {Object} [options] - */ - function ProportionalAreaModel( areaOptionObjects, options ) { +import BooleanProperty from '../../../../axon/js/BooleanProperty.js'; +import Utils from '../../../../dot/js/Utils.js'; +import inherit from '../../../../phet-core/js/inherit.js'; +import merge from '../../../../phet-core/js/merge.js'; +import PhetFont from '../../../../scenery-phet/js/PhetFont.js'; +import TextBounds from '../../../../scenery/js/util/TextBounds.js'; +import areaModelCommon from '../../areaModelCommon.js'; +import AreaModelCommonModel from '../../common/model/AreaModelCommonModel.js'; +import ProportionalArea from './ProportionalArea.js'; +import ProportionalAreaDisplay from './ProportionalAreaDisplay.js'; - options = merge( { - isProportional: true, - initialAreaBoxExpanded: true - }, options ); +// constants +const TEST_FONT = new PhetFont( 12 ); - const areas = areaOptionObjects.map( function( options ) { - return new ProportionalArea( options ); - } ); +/** + * @constructor + * @extends {AreaModelCommonModel} + * + * @param {Array.} areaOptionObjects - An array of options objects to be passed to the ProportionalArea + * constructors. + * @param {Object} [options] + */ +function ProportionalAreaModel( areaOptionObjects, options ) { - AreaModelCommonModel.call( this, areas, areas[ 0 ], options ); + options = merge( { + isProportional: true, + initialAreaBoxExpanded: true + }, options ); - // @public {BooleanProperty} - this.gridLinesVisibleProperty = new BooleanProperty( true ); + const areas = areaOptionObjects.map( function( options ) { + return new ProportionalArea( options ); + } ); - // @public {BooleanProperty} - this.tilesVisibleProperty = new BooleanProperty( false ); + AreaModelCommonModel.call( this, areas, areas[ 0 ], options ); - // @public {BooleanProperty} - this.countingVisibleProperty = new BooleanProperty( true ); + // @public {BooleanProperty} + this.gridLinesVisibleProperty = new BooleanProperty( true ); - // @public {BooleanProperty} - this.calculationBoxVisibleProperty = new BooleanProperty( false ); - } + // @public {BooleanProperty} + this.tilesVisibleProperty = new BooleanProperty( false ); - areaModelCommon.register( 'ProportionalAreaModel', ProportionalAreaModel ); - - return inherit( AreaModelCommonModel, ProportionalAreaModel, { - /** - * Returns a concrete AreaDisplay subtype - * @protected - * - * @param {Property.} areaProperty - * @returns {ProportionalAreaDisplay} - */ - createAreaDisplay: function( areaProperty ) { - return new ProportionalAreaDisplay( areaProperty ); - }, - - /** - * Returns a string that should be the longest string possible for our given areas. - * @public - * - * @returns {string} - */ - getMaximumAreaString: function() { - let maxString = '9'; - let maxLength = 0; - this.areas.forEach( function( area ) { - const representativeSize = area.snapSize + area.maximumSize; - // Round because of floating point precision - const string = '' + Utils.toFixedNumber( representativeSize * representativeSize, 8 ); // Square for area - const length = TextBounds.approximateCanvasWidth( TEST_FONT, string ); - if ( length > maxLength ) { - maxLength = length; - maxString = string; - } - } ); - return maxString; - }, - - /** - * Returns the model to its initial state. - * @public - * @override - */ - reset: function() { - AreaModelCommonModel.prototype.reset.call( this ); - - this.gridLinesVisibleProperty.reset(); - this.tilesVisibleProperty.reset(); - this.countingVisibleProperty.reset(); - this.calculationBoxVisibleProperty.reset(); - } - } ); -} ); + // @public {BooleanProperty} + this.countingVisibleProperty = new BooleanProperty( true ); + + // @public {BooleanProperty} + this.calculationBoxVisibleProperty = new BooleanProperty( false ); +} + +areaModelCommon.register( 'ProportionalAreaModel', ProportionalAreaModel ); + +export default inherit( AreaModelCommonModel, ProportionalAreaModel, { + /** + * Returns a concrete AreaDisplay subtype + * @protected + * + * @param {Property.} areaProperty + * @returns {ProportionalAreaDisplay} + */ + createAreaDisplay: function( areaProperty ) { + return new ProportionalAreaDisplay( areaProperty ); + }, + + /** + * Returns a string that should be the longest string possible for our given areas. + * @public + * + * @returns {string} + */ + getMaximumAreaString: function() { + let maxString = '9'; + let maxLength = 0; + this.areas.forEach( function( area ) { + const representativeSize = area.snapSize + area.maximumSize; + // Round because of floating point precision + const string = '' + Utils.toFixedNumber( representativeSize * representativeSize, 8 ); // Square for area + const length = TextBounds.approximateCanvasWidth( TEST_FONT, string ); + if ( length > maxLength ) { + maxLength = length; + maxString = string; + } + } ); + return maxString; + }, + + /** + * Returns the model to its initial state. + * @public + * @override + */ + reset: function() { + AreaModelCommonModel.prototype.reset.call( this ); + + this.gridLinesVisibleProperty.reset(); + this.tilesVisibleProperty.reset(); + this.countingVisibleProperty.reset(); + this.calculationBoxVisibleProperty.reset(); + } +} ); \ No newline at end of file diff --git a/js/proportional/view/CalculationBox.js b/js/proportional/view/CalculationBox.js index 90931f31..89923d83 100644 --- a/js/proportional/view/CalculationBox.js +++ b/js/proportional/view/CalculationBox.js @@ -7,98 +7,94 @@ * * @author Jonathan Olson */ -define( require => { - 'use strict'; - // modules - const AlignBox = require( 'SCENERY/nodes/AlignBox' ); - const AreaCalculationChoice = require( 'AREA_MODEL_COMMON/common/model/AreaCalculationChoice' ); - const areaModelCommon = require( 'AREA_MODEL_COMMON/areaModelCommon' ); - const AreaModelCommonA11yStrings = require( 'AREA_MODEL_COMMON/AreaModelCommonA11yStrings' ); - const AreaModelCommonAccordionBox = require( 'AREA_MODEL_COMMON/common/view/AreaModelCommonAccordionBox' ); - const Bounds2 = require( 'DOT/Bounds2' ); - const CalculationLinesNode = require( 'AREA_MODEL_COMMON/common/view/calculation/CalculationLinesNode' ); - const inherit = require( 'PHET_CORE/inherit' ); +import Bounds2 from '../../../../dot/js/Bounds2.js'; +import inherit from '../../../../phet-core/js/inherit.js'; +import AlignBox from '../../../../scenery/js/nodes/AlignBox.js'; +import areaModelCommonStrings from '../../area-model-common-strings.js'; +import areaModelCommon from '../../areaModelCommon.js'; +import AreaModelCommonA11yStrings from '../../AreaModelCommonA11yStrings.js'; +import AreaCalculationChoice from '../../common/model/AreaCalculationChoice.js'; +import AreaModelCommonAccordionBox from '../../common/view/AreaModelCommonAccordionBox.js'; +import CalculationLinesNode from '../../common/view/calculation/CalculationLinesNode.js'; - // strings - const calculationString = require( 'string!AREA_MODEL_COMMON/calculation' ); +const calculationString = areaModelCommonStrings.calculation; - // a11y strings - const calculationBoxTitleString = AreaModelCommonA11yStrings.calculationBoxTitle.value; - const calculationBoxDescriptionString = AreaModelCommonA11yStrings.calculationBoxDescription.value; +// a11y strings +const calculationBoxTitleString = AreaModelCommonA11yStrings.calculationBoxTitle.value; +const calculationBoxDescriptionString = AreaModelCommonA11yStrings.calculationBoxDescription.value; - // constants - const MARGIN = 8; +// constants +const MARGIN = 8; - /** - * @constructor - * @extends {AreaModelCommonAccordionBox} - * - * @param {ProportionalAreaModel} model - * @param {Bounds2} bounds - Where to lay out the box in view space - * @param {Object} [nodeOptions] - */ - function CalculationBox( model, bounds, nodeOptions ) { - - const self = this; - - // @private {ProportionalAreaModel} - this.model = model; - - // @private {CalculationLinesNode} - this.calculationLinesNode = new CalculationLinesNode( model ); - - const alignBox = new AlignBox( this.calculationLinesNode, { - // Since our AccordionBox expands by our MARGIN, we need to set content bounds without that - alignBounds: bounds.eroded( MARGIN ), - pickable: false - } ); - - AreaModelCommonAccordionBox.call( this, calculationString, model.calculationBoxVisibleProperty, alignBox, { - // Different margins than our other accordion boxes - contentXMargin: MARGIN, - contentYMargin: MARGIN, - - // Different spacing - maxTitleWidth: 290, - - // We don't have room for the normal spacing, so we need to make things closer together than they normally are. - contentXSpacing: -10, - - // a11y - labelTagName: 'h3', - labelContent: calculationBoxTitleString, - titleBarOptions: { - descriptionContent: calculationBoxDescriptionString - } - } ); - - model.areaCalculationChoiceProperty.link( function( choice ) { - assert && assert( choice !== AreaCalculationChoice.LINE_BY_LINE, 'Should be HIDDEN or SHOW_ALL_LINES' ); - self.visible = choice !== AreaCalculationChoice.HIDDEN; - } ); - - // Resize things so our AccordionBox is the correct size (we can't get bounds correct initially, because of the - // expand button shifting content) - alignBox.alignBounds = new Bounds2( - 0, - 0, - alignBox.alignBounds.width - ( this.width - bounds.width ), - alignBox.alignBounds.height - ( this.height - bounds.height ) - ); - - this.mutate( nodeOptions ); - } +/** + * @constructor + * @extends {AreaModelCommonAccordionBox} + * + * @param {ProportionalAreaModel} model + * @param {Bounds2} bounds - Where to lay out the box in view space + * @param {Object} [nodeOptions] + */ +function CalculationBox( model, bounds, nodeOptions ) { + + const self = this; + + // @private {ProportionalAreaModel} + this.model = model; + + // @private {CalculationLinesNode} + this.calculationLinesNode = new CalculationLinesNode( model ); + + const alignBox = new AlignBox( this.calculationLinesNode, { + // Since our AccordionBox expands by our MARGIN, we need to set content bounds without that + alignBounds: bounds.eroded( MARGIN ), + pickable: false + } ); - areaModelCommon.register( 'CalculationBox', CalculationBox ); + AreaModelCommonAccordionBox.call( this, calculationString, model.calculationBoxVisibleProperty, alignBox, { + // Different margins than our other accordion boxes + contentXMargin: MARGIN, + contentYMargin: MARGIN, - return inherit( AreaModelCommonAccordionBox, CalculationBox, { - /** - * Updates the content of the calculation box (if needed). - * @public - */ - update: function() { - this.calculationLinesNode.update(); + // Different spacing + maxTitleWidth: 290, + + // We don't have room for the normal spacing, so we need to make things closer together than they normally are. + contentXSpacing: -10, + + // a11y + labelTagName: 'h3', + labelContent: calculationBoxTitleString, + titleBarOptions: { + descriptionContent: calculationBoxDescriptionString } } ); -} ); + + model.areaCalculationChoiceProperty.link( function( choice ) { + assert && assert( choice !== AreaCalculationChoice.LINE_BY_LINE, 'Should be HIDDEN or SHOW_ALL_LINES' ); + self.visible = choice !== AreaCalculationChoice.HIDDEN; + } ); + + // Resize things so our AccordionBox is the correct size (we can't get bounds correct initially, because of the + // expand button shifting content) + alignBox.alignBounds = new Bounds2( + 0, + 0, + alignBox.alignBounds.width - ( this.width - bounds.width ), + alignBox.alignBounds.height - ( this.height - bounds.height ) + ); + + this.mutate( nodeOptions ); +} + +areaModelCommon.register( 'CalculationBox', CalculationBox ); + +export default inherit( AreaModelCommonAccordionBox, CalculationBox, { + /** + * Updates the content of the calculation box (if needed). + * @public + */ + update: function() { + this.calculationLinesNode.update(); + } +} ); \ No newline at end of file diff --git a/js/proportional/view/CountingAreaNode.js b/js/proportional/view/CountingAreaNode.js index 8793a127..b5d6c455 100644 --- a/js/proportional/view/CountingAreaNode.js +++ b/js/proportional/view/CountingAreaNode.js @@ -7,138 +7,134 @@ * * @author Jonathan Olson */ -define( require => { - 'use strict'; - // modules - const areaModelCommon = require( 'AREA_MODEL_COMMON/areaModelCommon' ); - const AreaModelCommonColorProfile = require( 'AREA_MODEL_COMMON/common/view/AreaModelCommonColorProfile' ); - const AreaModelCommonConstants = require( 'AREA_MODEL_COMMON/common/AreaModelCommonConstants' ); - const inherit = require( 'PHET_CORE/inherit' ); - const Node = require( 'SCENERY/nodes/Node' ); - const Text = require( 'SCENERY/nodes/Text' ); - const Vector2 = require( 'DOT/Vector2' ); +import Vector2 from '../../../../dot/js/Vector2.js'; +import inherit from '../../../../phet-core/js/inherit.js'; +import Node from '../../../../scenery/js/nodes/Node.js'; +import Text from '../../../../scenery/js/nodes/Text.js'; +import areaModelCommon from '../../areaModelCommon.js'; +import AreaModelCommonConstants from '../../common/AreaModelCommonConstants.js'; +import AreaModelCommonColorProfile from '../../common/view/AreaModelCommonColorProfile.js'; - // constants - const scratchVector = new Vector2( 0, 0 ); // Created so we can minimize object creation and garbage collection +// constants +const scratchVector = new Vector2( 0, 0 ); // Created so we can minimize object creation and garbage collection - /** - * @constructor - * @extends {Node} - * - * @param {OrientationPair.>} activeTotalProperties - * @param {Property.} modelViewTransformProperty - * @param {Property.} countingVisibleProperty - */ - function CountingAreaNode( activeTotalProperties, modelViewTransformProperty, countingVisibleProperty ) { +/** + * @constructor + * @extends {Node} + * + * @param {OrientationPair.>} activeTotalProperties + * @param {Property.} modelViewTransformProperty + * @param {Property.} countingVisibleProperty + */ +function CountingAreaNode( activeTotalProperties, modelViewTransformProperty, countingVisibleProperty ) { - Node.call( this ); + Node.call( this ); - const self = this; + const self = this; - // @private {OrientationPair.>} - this.activeTotalProperties = activeTotalProperties; + // @private {OrientationPair.>} + this.activeTotalProperties = activeTotalProperties; - // @private {Property.} - this.modelViewTransformProperty = modelViewTransformProperty; + // @private {Property.} + this.modelViewTransformProperty = modelViewTransformProperty; - // @private {Property.} - this.countingVisibleProperty = countingVisibleProperty; + // @private {Property.} + this.countingVisibleProperty = countingVisibleProperty; - // @private {Array.} - We reuse these to avoid GC/performance issues - this.textNodes = []; + // @private {Array.} - We reuse these to avoid GC/performance issues + this.textNodes = []; - // @private {boolean} - Whether we should be redrawn - this.dirty = true; + // @private {boolean} - Whether we should be redrawn + this.dirty = true; - // Things we depend on - function invalidate() { - self.dirty = true; - } + // Things we depend on + function invalidate() { + self.dirty = true; + } - countingVisibleProperty.link( invalidate ); - activeTotalProperties.horizontal.link( invalidate ); - activeTotalProperties.vertical.link( invalidate ); - modelViewTransformProperty.link( invalidate ); + countingVisibleProperty.link( invalidate ); + activeTotalProperties.horizontal.link( invalidate ); + activeTotalProperties.vertical.link( invalidate ); + modelViewTransformProperty.link( invalidate ); - countingVisibleProperty.linkAttribute( this, 'visible' ); - } + countingVisibleProperty.linkAttribute( this, 'visible' ); +} - areaModelCommon.register( 'CountingAreaNode', CountingAreaNode ); - - return inherit( Node, CountingAreaNode, { - /** - * Creates a reusable text node with a given number. - * @private - * - * @param {number} number - * @returns {Text} - */ - createTextNode: function( number ) { - const text = new Text( number, { - font: AreaModelCommonConstants.COUNTING_FONT, - fill: AreaModelCommonColorProfile.countingLabelProperty - } ); - this.textNodes.push( text ); - this.addChild( text ); - return text; - }, - - /** - * Returns the reusable text node with a given number. - * @private - * - * @param {number} number - * @returns {Text} - */ - getTextNode: function( number ) { - let text = this.textNodes[ number - 1 ]; - if ( !text ) { - text = this.createTextNode( number ); - } - return text; - }, +areaModelCommon.register( 'CountingAreaNode', CountingAreaNode ); + +export default inherit( Node, CountingAreaNode, { + /** + * Creates a reusable text node with a given number. + * @private + * + * @param {number} number + * @returns {Text} + */ + createTextNode: function( number ) { + const text = new Text( number, { + font: AreaModelCommonConstants.COUNTING_FONT, + fill: AreaModelCommonColorProfile.countingLabelProperty + } ); + this.textNodes.push( text ); + this.addChild( text ); + return text; + }, + + /** + * Returns the reusable text node with a given number. + * @private + * + * @param {number} number + * @returns {Text} + */ + getTextNode: function( number ) { + let text = this.textNodes[ number - 1 ]; + if ( !text ) { + text = this.createTextNode( number ); + } + return text; + }, - /** - * Updates the view for tiled areas (since it is somewhat expensive to re-draw, and we don't want it being done - * multiple times per frame. - * @private - */ - update: function() { - const modelViewTransform = this.modelViewTransformProperty.value; + /** + * Updates the view for tiled areas (since it is somewhat expensive to re-draw, and we don't want it being done + * multiple times per frame. + * @private + */ + update: function() { + const modelViewTransform = this.modelViewTransformProperty.value; - // Ignore updates if we are not dirty - if ( !this.dirty ) { return; } - this.dirty = false; + // Ignore updates if we are not dirty + if ( !this.dirty ) { return; } + this.dirty = false; - if ( !this.countingVisibleProperty.value ) { return; } + if ( !this.countingVisibleProperty.value ) { return; } - // Coordinate mapping into the view - const modelToViewX = modelViewTransform.modelToViewX.bind( modelViewTransform ); - const modelToViewY = modelViewTransform.modelToViewY.bind( modelViewTransform ); + // Coordinate mapping into the view + const modelToViewX = modelViewTransform.modelToViewX.bind( modelViewTransform ); + const modelToViewY = modelViewTransform.modelToViewY.bind( modelViewTransform ); - const width = this.activeTotalProperties.horizontal.value; - const height = this.activeTotalProperties.vertical.value; + const width = this.activeTotalProperties.horizontal.value; + const height = this.activeTotalProperties.vertical.value; - let cellNumber = 1; - for ( let row = 0; row < height; row++ ) { - const rowCenter = modelToViewY( row + 0.5 ); + let cellNumber = 1; + for ( let row = 0; row < height; row++ ) { + const rowCenter = modelToViewY( row + 0.5 ); - for ( let col = 0; col < width; col++ ) { - const colCenter = modelToViewX( col + 0.5 ); + for ( let col = 0; col < width; col++ ) { + const colCenter = modelToViewX( col + 0.5 ); - const text = this.getTextNode( cellNumber ); - text.center = scratchVector.setXY( colCenter, rowCenter ); - text.visible = true; + const text = this.getTextNode( cellNumber ); + text.center = scratchVector.setXY( colCenter, rowCenter ); + text.visible = true; - cellNumber++; - } + cellNumber++; } + } - // Hide the rest of the text nodes (that should NOT show up) - for ( ; cellNumber - 1 < this.textNodes.length; cellNumber++ ) { - this.textNodes[ cellNumber - 1 ].visible = false; - } + // Hide the rest of the text nodes (that should NOT show up) + for ( ; cellNumber - 1 < this.textNodes.length; cellNumber++ ) { + this.textNodes[ cellNumber - 1 ].visible = false; } - } ); -} ); + } +} ); \ No newline at end of file diff --git a/js/proportional/view/PartitionRadioButtonGroup.js b/js/proportional/view/PartitionRadioButtonGroup.js index 490816a6..fd97320a 100644 --- a/js/proportional/view/PartitionRadioButtonGroup.js +++ b/js/proportional/view/PartitionRadioButtonGroup.js @@ -7,105 +7,101 @@ * * @author Jonathan Olson */ -define( require => { - 'use strict'; - // modules - const AlignBox = require( 'SCENERY/nodes/AlignBox' ); - const areaModelCommon = require( 'AREA_MODEL_COMMON/areaModelCommon' ); - const AreaModelCommonA11yStrings = require( 'AREA_MODEL_COMMON/AreaModelCommonA11yStrings' ); - const AreaModelCommonColorProfile = require( 'AREA_MODEL_COMMON/common/view/AreaModelCommonColorProfile' ); - const AreaModelCommonRadioButtonGroup = require( 'AREA_MODEL_COMMON/common/view/AreaModelCommonRadioButtonGroup' ); - const DerivedProperty = require( 'AXON/DerivedProperty' ); - const inherit = require( 'PHET_CORE/inherit' ); - const Line = require( 'SCENERY/nodes/Line' ); - const Orientation = require( 'PHET_CORE/Orientation' ); - const OrientationPair = require( 'AREA_MODEL_COMMON/common/model/OrientationPair' ); - const Path = require( 'SCENERY/nodes/Path' ); - const ProportionalPartitionLineNode = require( 'AREA_MODEL_COMMON/proportional/view/ProportionalPartitionLineNode' ); - const Rectangle = require( 'SCENERY/nodes/Rectangle' ); - const Vector2 = require( 'DOT/Vector2' ); +import DerivedProperty from '../../../../axon/js/DerivedProperty.js'; +import Vector2 from '../../../../dot/js/Vector2.js'; +import inherit from '../../../../phet-core/js/inherit.js'; +import Orientation from '../../../../phet-core/js/Orientation.js'; +import AlignBox from '../../../../scenery/js/nodes/AlignBox.js'; +import Line from '../../../../scenery/js/nodes/Line.js'; +import Path from '../../../../scenery/js/nodes/Path.js'; +import Rectangle from '../../../../scenery/js/nodes/Rectangle.js'; +import areaModelCommon from '../../areaModelCommon.js'; +import AreaModelCommonA11yStrings from '../../AreaModelCommonA11yStrings.js'; +import OrientationPair from '../../common/model/OrientationPair.js'; +import AreaModelCommonColorProfile from '../../common/view/AreaModelCommonColorProfile.js'; +import AreaModelCommonRadioButtonGroup from '../../common/view/AreaModelCommonRadioButtonGroup.js'; +import ProportionalPartitionLineNode from './ProportionalPartitionLineNode.js'; - // a11y strings - const horizontalPartitionString = AreaModelCommonA11yStrings.horizontalPartition.value; - const partitionSelectionDescriptionString = AreaModelCommonA11yStrings.partitionSelectionDescription.value; - const verticalPartitionString = AreaModelCommonA11yStrings.verticalPartition.value; +// a11y strings +const horizontalPartitionString = AreaModelCommonA11yStrings.horizontalPartition.value; +const partitionSelectionDescriptionString = AreaModelCommonA11yStrings.partitionSelectionDescription.value; +const verticalPartitionString = AreaModelCommonA11yStrings.verticalPartition.value; +/** + * @constructor + * @extends {AreaModelCommonRadioButtonGroup} + * + * @param {Property.} currentAreaOrientationProperty + * @param {AlignGroup} selectionButtonAlignGroup + */ +function PartitionRadioButtonGroup( currentAreaOrientationProperty, selectionButtonAlignGroup ) { + AreaModelCommonRadioButtonGroup.call( this, currentAreaOrientationProperty, Orientation.VALUES.map( function( orientation ) { + const icon = PartitionRadioButtonGroup.createPartitionOrientationIcon( orientation, currentAreaOrientationProperty ); + return { + value: orientation, + node: new AlignBox( icon, { group: selectionButtonAlignGroup } ), + + // a11y + labelContent: orientation === Orientation.HORIZONTAL ? verticalPartitionString : horizontalPartitionString + }; + } ), { + // Less margin than others desired here + buttonContentXMargin: 7, + buttonContentYMargin: 7, + + // a11y + descriptionContent: partitionSelectionDescriptionString + } ); +} + +areaModelCommon.register( 'PartitionRadioButtonGroup', PartitionRadioButtonGroup ); + +export default inherit( AreaModelCommonRadioButtonGroup, PartitionRadioButtonGroup, {}, { /** - * @constructor - * @extends {AreaModelCommonRadioButtonGroup} + * Creates an icon showing a switch to partition lines of a given orientation. + * @private * + * @param {Orientation} orientation * @param {Property.} currentAreaOrientationProperty - * @param {AlignGroup} selectionButtonAlignGroup + * @returns {Node} */ - function PartitionRadioButtonGroup( currentAreaOrientationProperty, selectionButtonAlignGroup ) { - AreaModelCommonRadioButtonGroup.call( this, currentAreaOrientationProperty, Orientation.VALUES.map( function( orientation ) { - const icon = PartitionRadioButtonGroup.createPartitionOrientationIcon( orientation, currentAreaOrientationProperty ); - return { - value: orientation, - node: new AlignBox( icon, { group: selectionButtonAlignGroup } ), - - // a11y - labelContent: orientation === Orientation.HORIZONTAL ? verticalPartitionString : horizontalPartitionString - }; - } ), { - // Less margin than others desired here - buttonContentXMargin: 7, - buttonContentYMargin: 7, - - // a11y - descriptionContent: partitionSelectionDescriptionString + createPartitionOrientationIcon: function( orientation, currentAreaOrientationProperty ) { + // The size of our rectangle + const sizes = new OrientationPair( 36, 24 ); + const background = new Rectangle( 0, 0, sizes.horizontal, sizes.vertical, { + stroke: AreaModelCommonColorProfile.partitionLineIconBorderProperty, + fill: AreaModelCommonColorProfile.partitionLineIconBackgroundProperty } ); - } - areaModelCommon.register( 'PartitionRadioButtonGroup', PartitionRadioButtonGroup ); + // Expand bounds a bit, to allow room for the line-handle icon part (so we have even padding) + background.localBounds = background.localBounds.dilated( 7 ); - return inherit( AreaModelCommonRadioButtonGroup, PartitionRadioButtonGroup, {}, { - /** - * Creates an icon showing a switch to partition lines of a given orientation. - * @private - * - * @param {Orientation} orientation - * @param {Property.} currentAreaOrientationProperty - * @returns {Node} - */ - createPartitionOrientationIcon: function( orientation, currentAreaOrientationProperty ) { - // The size of our rectangle - const sizes = new OrientationPair( 36, 24 ); - const background = new Rectangle( 0, 0, sizes.horizontal, sizes.vertical, { - stroke: AreaModelCommonColorProfile.partitionLineIconBorderProperty, - fill: AreaModelCommonColorProfile.partitionLineIconBackgroundProperty - } ); + const p1 = new Vector2( 0, 0 ); + const p2 = new Vector2( 0, 0 ); + p1[ orientation.coordinate ] = sizes.get( orientation ) * 2 / 3; + p2[ orientation.coordinate ] = sizes.get( orientation ) * 2 / 3; + p2[ orientation.opposite.coordinate ] = sizes.get( orientation.opposite ) * 1.1; - // Expand bounds a bit, to allow room for the line-handle icon part (so we have even padding) - background.localBounds = background.localBounds.dilated( 7 ); + background.children = [ + new Line( p1, p2, { + stroke: AreaModelCommonColorProfile.partitionLineIconLineProperty + } ), + new Path( ProportionalPartitionLineNode.HANDLE_ARROW_SHAPES.get( orientation ), { + fill: new DerivedProperty( + [ + currentAreaOrientationProperty, + AreaModelCommonColorProfile.proportionalColorProperties.get( orientation ), + AreaModelCommonColorProfile.partitionLineIconHandleProperty + ], + function( currentOrientation, widthColor, handleColor ) { + return currentOrientation === orientation ? widthColor : handleColor; + } ), + scale: 0.5, + translation: p2 + } ) + ]; - const p1 = new Vector2( 0, 0 ); - const p2 = new Vector2( 0, 0 ); - p1[ orientation.coordinate ] = sizes.get( orientation ) * 2 / 3; - p2[ orientation.coordinate ] = sizes.get( orientation ) * 2 / 3; - p2[ orientation.opposite.coordinate ] = sizes.get( orientation.opposite ) * 1.1; - - background.children = [ - new Line( p1, p2, { - stroke: AreaModelCommonColorProfile.partitionLineIconLineProperty - } ), - new Path( ProportionalPartitionLineNode.HANDLE_ARROW_SHAPES.get( orientation ), { - fill: new DerivedProperty( - [ - currentAreaOrientationProperty, - AreaModelCommonColorProfile.proportionalColorProperties.get( orientation ), - AreaModelCommonColorProfile.partitionLineIconHandleProperty - ], - function( currentOrientation, widthColor, handleColor ) { - return currentOrientation === orientation ? widthColor : handleColor; - } ), - scale: 0.5, - translation: p2 - } ) - ]; - - return background; - } - } ); -} ); + return background; + } +} ); \ No newline at end of file diff --git a/js/proportional/view/ProportionalAreaDisplayNode.js b/js/proportional/view/ProportionalAreaDisplayNode.js index 5ac693d6..97c75cf4 100644 --- a/js/proportional/view/ProportionalAreaDisplayNode.js +++ b/js/proportional/view/ProportionalAreaDisplayNode.js @@ -7,389 +7,385 @@ * * @author Jonathan Olson */ -define( require => { - 'use strict'; - - // modules - const AreaDisplayNode = require( 'AREA_MODEL_COMMON/common/view/AreaDisplayNode' ); - const areaModelCommon = require( 'AREA_MODEL_COMMON/areaModelCommon' ); - const AreaModelCommonA11yStrings = require( 'AREA_MODEL_COMMON/AreaModelCommonA11yStrings' ); - const AreaModelCommonColorProfile = require( 'AREA_MODEL_COMMON/common/view/AreaModelCommonColorProfile' ); - const AreaModelCommonConstants = require( 'AREA_MODEL_COMMON/common/AreaModelCommonConstants' ); - const BooleanProperty = require( 'AXON/BooleanProperty' ); - const CountingAreaNode = require( 'AREA_MODEL_COMMON/proportional/view/CountingAreaNode' ); - const DerivedProperty = require( 'AXON/DerivedProperty' ); - const DynamicProperty = require( 'AXON/DynamicProperty' ); - const inherit = require( 'PHET_CORE/inherit' ); - const merge = require( 'PHET_CORE/merge' ); - const Node = require( 'SCENERY/nodes/Node' ); - const Orientation = require( 'PHET_CORE/Orientation' ); - const Property = require( 'AXON/Property' ); - const ProportionalAreaGridLinesNode = require( 'AREA_MODEL_COMMON/proportional/view/ProportionalAreaGridLinesNode' ); - const ProportionalDragHandle = require( 'AREA_MODEL_COMMON/proportional/view/ProportionalDragHandle' ); - const ProportionalPartitionLineNode = require( 'AREA_MODEL_COMMON/proportional/view/ProportionalPartitionLineNode' ); - const Range = require( 'DOT/Range' ); - const Rectangle = require( 'SCENERY/nodes/Rectangle' ); - const StringUtils = require( 'PHETCOMMON/util/StringUtils' ); - const Text = require( 'SCENERY/nodes/Text' ); - const TiledAreaNode = require( 'AREA_MODEL_COMMON/proportional/view/TiledAreaNode' ); - const Utils = require( 'DOT/Utils' ); - - // a11y strings - const areaGridString = AreaModelCommonA11yStrings.areaGrid.value; - const areaGridRectanglePatternString = AreaModelCommonA11yStrings.areaGridRectanglePattern.value; - const countingNumbersPatternString = AreaModelCommonA11yStrings.countingNumbersPattern.value; - /** - * @constructor - * @extends {AreaDisplayNode} - * - * @param {ProportionalAreaDisplay} areaDisplay - * @param {Property.} partialProductsChoiceProperty - * @param {Object} [options] - * @param {Object} [nodeOptions] - */ - function ProportionalAreaDisplayNode( areaDisplay, partialProductsChoiceProperty, options, nodeOptions ) { +import BooleanProperty from '../../../../axon/js/BooleanProperty.js'; +import DerivedProperty from '../../../../axon/js/DerivedProperty.js'; +import DynamicProperty from '../../../../axon/js/DynamicProperty.js'; +import Property from '../../../../axon/js/Property.js'; +import Range from '../../../../dot/js/Range.js'; +import Utils from '../../../../dot/js/Utils.js'; +import inherit from '../../../../phet-core/js/inherit.js'; +import merge from '../../../../phet-core/js/merge.js'; +import Orientation from '../../../../phet-core/js/Orientation.js'; +import StringUtils from '../../../../phetcommon/js/util/StringUtils.js'; +import Node from '../../../../scenery/js/nodes/Node.js'; +import Rectangle from '../../../../scenery/js/nodes/Rectangle.js'; +import Text from '../../../../scenery/js/nodes/Text.js'; +import areaModelCommon from '../../areaModelCommon.js'; +import AreaModelCommonA11yStrings from '../../AreaModelCommonA11yStrings.js'; +import AreaModelCommonConstants from '../../common/AreaModelCommonConstants.js'; +import AreaDisplayNode from '../../common/view/AreaDisplayNode.js'; +import AreaModelCommonColorProfile from '../../common/view/AreaModelCommonColorProfile.js'; +import CountingAreaNode from './CountingAreaNode.js'; +import ProportionalAreaGridLinesNode from './ProportionalAreaGridLinesNode.js'; +import ProportionalDragHandle from './ProportionalDragHandle.js'; +import ProportionalPartitionLineNode from './ProportionalPartitionLineNode.js'; +import TiledAreaNode from './TiledAreaNode.js'; + +// a11y strings +const areaGridString = AreaModelCommonA11yStrings.areaGrid.value; +const areaGridRectanglePatternString = AreaModelCommonA11yStrings.areaGridRectanglePattern.value; +const countingNumbersPatternString = AreaModelCommonA11yStrings.countingNumbersPattern.value; - const self = this; +/** + * @constructor + * @extends {AreaDisplayNode} + * + * @param {ProportionalAreaDisplay} areaDisplay + * @param {Property.} partialProductsChoiceProperty + * @param {Object} [options] + * @param {Object} [nodeOptions] + */ +function ProportionalAreaDisplayNode( areaDisplay, partialProductsChoiceProperty, options, nodeOptions ) { - options = merge( { + const self = this; - // Meant to be overridden - gridLinesVisibleProperty: new BooleanProperty( false ), - tilesVisibleProperty: new BooleanProperty( false ), - countingVisibleProperty: new BooleanProperty( false ), - useTileLikeBackground: false, - useLargeArea: false, + options = merge( { - // Specified for supertype - isProportional: true - }, options ); + // Meant to be overridden + gridLinesVisibleProperty: new BooleanProperty( false ), + tilesVisibleProperty: new BooleanProperty( false ), + countingVisibleProperty: new BooleanProperty( false ), + useTileLikeBackground: false, + useLargeArea: false, - nodeOptions = merge( { - // a11y - tagName: 'div', - labelTagName: 'h3', - labelContent: areaGridString - }, nodeOptions ); + // Specified for supertype + isProportional: true + }, options ); - AreaDisplayNode.call( this, areaDisplay, partialProductsChoiceProperty, options ); + nodeOptions = merge( { + // a11y + tagName: 'div', + labelTagName: 'h3', + labelContent: areaGridString + }, nodeOptions ); - const countingLabel = new Node( { - tagName: 'span' - } ); - this.accessibleParagraphNode.insertChild( 0, countingLabel ); - options.countingVisibleProperty.linkAttribute( countingLabel, 'visible' ); + AreaDisplayNode.call( this, areaDisplay, partialProductsChoiceProperty, options ); - const areaAccessibleLabel = new Node( { - tagName: 'span' + const countingLabel = new Node( { + tagName: 'span' + } ); + this.accessibleParagraphNode.insertChild( 0, countingLabel ); + options.countingVisibleProperty.linkAttribute( countingLabel, 'visible' ); + + const areaAccessibleLabel = new Node( { + tagName: 'span' + } ); + this.accessibleParagraphNode.insertChild( 0, areaAccessibleLabel ); + Property.multilink( areaDisplay.activeTotalProperties.values, function( width, height ) { + areaAccessibleLabel.innerContent = StringUtils.fillIn( areaGridRectanglePatternString, { + width: width, + height: height } ); - this.accessibleParagraphNode.insertChild( 0, areaAccessibleLabel ); - Property.multilink( areaDisplay.activeTotalProperties.values, function( width, height ) { - areaAccessibleLabel.innerContent = StringUtils.fillIn( areaGridRectanglePatternString, { - width: width, - height: height - } ); - countingLabel.innerContent = StringUtils.fillIn( countingNumbersPatternString, { - count: Utils.toFixedNumber( width * height, Utils.numberOfDecimalPlaces( width ) + Utils.numberOfDecimalPlaces( height ) ) - } ); + countingLabel.innerContent = StringUtils.fillIn( countingNumbersPatternString, { + count: Utils.toFixedNumber( width * height, Utils.numberOfDecimalPlaces( width ) + Utils.numberOfDecimalPlaces( height ) ) } ); + } ); - // Background fill - this.areaLayer.addChild( this.backgroundNode ); + // Background fill + this.areaLayer.addChild( this.backgroundNode ); - // Grid lines - const gridLinesNode = new ProportionalAreaGridLinesNode( areaDisplay.areaProperty, this.modelViewTransformProperty ); - this.areaLayer.addChild( gridLinesNode ); - options.gridLinesVisibleProperty.linkAttribute( gridLinesNode, 'visible' ); + // Grid lines + const gridLinesNode = new ProportionalAreaGridLinesNode( areaDisplay.areaProperty, this.modelViewTransformProperty ); + this.areaLayer.addChild( gridLinesNode ); + options.gridLinesVisibleProperty.linkAttribute( gridLinesNode, 'visible' ); - // Active area background - const activeAreaBackground = new Rectangle( { - fill: options.useTileLikeBackground - ? AreaModelCommonColorProfile.semiTransparentSmallTileProperty - : AreaModelCommonColorProfile.proportionalActiveAreaBackgroundProperty, - stroke: AreaModelCommonColorProfile.proportionalActiveAreaBorderProperty + // Active area background + const activeAreaBackground = new Rectangle( { + fill: options.useTileLikeBackground + ? AreaModelCommonColorProfile.semiTransparentSmallTileProperty + : AreaModelCommonColorProfile.proportionalActiveAreaBackgroundProperty, + stroke: AreaModelCommonColorProfile.proportionalActiveAreaBorderProperty + } ); + Property.multilink( + [ areaDisplay.activeTotalProperties.horizontal, this.modelViewTransformProperty ], + function( totalWidth, modelViewTransform ) { + activeAreaBackground.rectWidth = modelViewTransform.modelToViewX( totalWidth ); } ); - Property.multilink( - [ areaDisplay.activeTotalProperties.horizontal, this.modelViewTransformProperty ], - function( totalWidth, modelViewTransform ) { - activeAreaBackground.rectWidth = modelViewTransform.modelToViewX( totalWidth ); - } ); - Property.multilink( - [ areaDisplay.activeTotalProperties.vertical, this.modelViewTransformProperty ], - function( totalHeight, modelViewTransform ) { - activeAreaBackground.rectHeight = modelViewTransform.modelToViewY( totalHeight ); - } ); - this.areaLayer.addChild( activeAreaBackground ); + Property.multilink( + [ areaDisplay.activeTotalProperties.vertical, this.modelViewTransformProperty ], + function( totalHeight, modelViewTransform ) { + activeAreaBackground.rectHeight = modelViewTransform.modelToViewY( totalHeight ); + } ); + this.areaLayer.addChild( activeAreaBackground ); - const tilesVisibleProperty = new DerivedProperty( - [ areaDisplay.tilesAvailableProperty, options.tilesVisibleProperty ], - function( tilesAvailable, tilesVisible ) { - return tilesAvailable && tilesVisible; - } ); + const tilesVisibleProperty = new DerivedProperty( + [ areaDisplay.tilesAvailableProperty, options.tilesVisibleProperty ], + function( tilesAvailable, tilesVisible ) { + return tilesAvailable && tilesVisible; + } ); - // @private {TiledAreaNode|null} - Tiles (optionally enabled) - this.tiledAreaNode = new TiledAreaNode( areaDisplay, this.modelViewTransformProperty, tilesVisibleProperty ); - this.areaLayer.addChild( this.tiledAreaNode ); + // @private {TiledAreaNode|null} - Tiles (optionally enabled) + this.tiledAreaNode = new TiledAreaNode( areaDisplay, this.modelViewTransformProperty, tilesVisibleProperty ); + this.areaLayer.addChild( this.tiledAreaNode ); - // Background stroke - this.areaLayer.addChild( this.borderNode ); + // Background stroke + this.areaLayer.addChild( this.borderNode ); - // Active area drag handle - this.areaLayer.addChild( new ProportionalDragHandle( - areaDisplay.areaProperty, - areaDisplay.activeTotalProperties, - this.modelViewTransformProperty - ) ); + // Active area drag handle + this.areaLayer.addChild( new ProportionalDragHandle( + areaDisplay.areaProperty, + areaDisplay.activeTotalProperties, + this.modelViewTransformProperty + ) ); - const countingVisibleProperty = new DerivedProperty( - [ areaDisplay.countingAvailableProperty, options.countingVisibleProperty ], - function( countingAvailable, countingVisible ) { - return countingAvailable && countingVisible; - } ); + const countingVisibleProperty = new DerivedProperty( + [ areaDisplay.countingAvailableProperty, options.countingVisibleProperty ], + function( countingAvailable, countingVisible ) { + return countingAvailable && countingVisible; + } ); - // @private {CountingAreaNode|null} - Counts of numbers for squares (optionally enabled) - this.countingAreaNode = new CountingAreaNode( - areaDisplay.activeTotalProperties, - this.modelViewTransformProperty, - countingVisibleProperty + // @private {CountingAreaNode|null} - Counts of numbers for squares (optionally enabled) + this.countingAreaNode = new CountingAreaNode( + areaDisplay.activeTotalProperties, + this.modelViewTransformProperty, + countingVisibleProperty + ); + this.areaLayer.addChild( this.countingAreaNode ); + + // Partition lines + Orientation.VALUES.forEach( function( orientation ) { + self.areaLayer.addChild( new ProportionalPartitionLineNode( + areaDisplay, + self.modelViewTransformProperty, + orientation ) ); - this.areaLayer.addChild( this.countingAreaNode ); - - // Partition lines - Orientation.VALUES.forEach( function( orientation ) { - self.areaLayer.addChild( new ProportionalPartitionLineNode( - areaDisplay, - self.modelViewTransformProperty, - orientation ) + } ); + + // Partition labels + Orientation.VALUES.forEach( function( orientation ) { + const partitionsProperties = areaDisplay.partitionsProperties.get( orientation ); + + // because we will have at most 2 + const labels = [ 0, 1 ].map( function( index ) { + const partitionProperty = new DerivedProperty( [ partitionsProperties ], function( partitions ) { + return partitions[ index ]; + } ); + const label = self.createPartitionLabel( + partitionProperty, + areaDisplay.secondaryPartitionsProperty.get( orientation ), + index, + orientation ); + self.labelLayer.addChild( label ); + return label; } ); - // Partition labels - Orientation.VALUES.forEach( function( orientation ) { - const partitionsProperties = areaDisplay.partitionsProperties.get( orientation ); - - // because we will have at most 2 - const labels = [ 0, 1 ].map( function( index ) { - const partitionProperty = new DerivedProperty( [ partitionsProperties ], function( partitions ) { - return partitions[ index ]; - } ); - const label = self.createPartitionLabel( - partitionProperty, - areaDisplay.secondaryPartitionsProperty.get( orientation ), - index, - orientation - ); - self.labelLayer.addChild( label ); - return label; + const labelListener = self.positionPartitionLabels.bind( self, orientation, labels ); + partitionsProperties.link( function( partitions, oldPartitions ) { + oldPartitions && oldPartitions.forEach( function( partition ) { + partition.coordinateRangeProperty.unlink( labelListener ); } ); - - const labelListener = self.positionPartitionLabels.bind( self, orientation, labels ); - partitionsProperties.link( function( partitions, oldPartitions ) { - oldPartitions && oldPartitions.forEach( function( partition ) { - partition.coordinateRangeProperty.unlink( labelListener ); - } ); - partitions.forEach( function( partition ) { - partition.coordinateRangeProperty.link( labelListener ); - } ); - labelListener(); + partitions.forEach( function( partition ) { + partition.coordinateRangeProperty.link( labelListener ); } ); - areaDisplay.primaryPartitionsProperty.get( orientation ).link( labelListener ); - areaDisplay.secondaryPartitionsProperty.get( orientation ).link( labelListener ); + labelListener(); } ); + areaDisplay.primaryPartitionsProperty.get( orientation ).link( labelListener ); + areaDisplay.secondaryPartitionsProperty.get( orientation ).link( labelListener ); + } ); - this.mutate( nodeOptions ); - } + this.mutate( nodeOptions ); +} - areaModelCommon.register( 'ProportionalAreaDisplayNode', ProportionalAreaDisplayNode ); - - return inherit( AreaDisplayNode, ProportionalAreaDisplayNode, { - /** - * Updates expensive-to-update things. - * @public - */ - update: function() { - AreaDisplayNode.prototype.update.call( this ); - - this.tiledAreaNode.update(); - this.countingAreaNode.update(); - }, - - /** - * Returns the partial product node at the given horizontal/vertical indices. - * @private - * - * @param {number} horizontalIndex - * @param {number} verticalIndex - * @returns {PartialProductLabelNode} - */ - getProductLabel: function( horizontalIndex, verticalIndex ) { - const horizontalPartitions = this.areaDisplay.partitionsProperties.horizontal.value; - const verticalPartitions = this.areaDisplay.partitionsProperties.vertical.value; - - return _.find( this.productLabels, function( productLabel ) { - const partitions = productLabel.partitionedAreaProperty.value.partitions; - return partitions.get( Orientation.HORIZONTAL ) === horizontalPartitions[ horizontalIndex ] && - partitions.get( Orientation.VERTICAL ) === verticalPartitions[ verticalIndex ]; - } ); - }, - - /** - * Positions all of the partial products labels. - * @protected - * @override - */ - positionProductLabels: function() { - const self = this; - - // {OrientationPair.>} - Current view ranges (if non-null) for each orientation - const rangesPair = this.areaDisplay.partitionsProperties.map( function( partitionsProperties, orientation ) { - return partitionsProperties.value.map( function( partition ) { - const range = partition.coordinateRangeProperty.value; - if ( range === null ) { - return null; - } - return new Range( - orientation.modelToView( self.modelViewTransformProperty.value, range.min ), - orientation.modelToView( self.modelViewTransformProperty.value, range.max ) - ); - } ); - } ); +areaModelCommon.register( 'ProportionalAreaDisplayNode', ProportionalAreaDisplayNode ); - // First, center the labels (if they have defined ranges) - this.productLabels.forEach( function( productLabel ) { - rangesPair.forEach( function( ranges, orientation ) { - const partition = productLabel.partitionedAreaProperty.value.partitions.get( orientation ); - const range = ranges[ _.indexOf( self.areaDisplay.partitionsProperties.get( orientation ).value, partition ) ]; - if ( range ) { - productLabel[ orientation.coordinate ] = range.getCenter(); - } - } ); - } ); +export default inherit( AreaDisplayNode, ProportionalAreaDisplayNode, { + /** + * Updates expensive-to-update things. + * @public + */ + update: function() { + AreaDisplayNode.prototype.update.call( this ); + + this.tiledAreaNode.update(); + this.countingAreaNode.update(); + }, + + /** + * Returns the partial product node at the given horizontal/vertical indices. + * @private + * + * @param {number} horizontalIndex + * @param {number} verticalIndex + * @returns {PartialProductLabelNode} + */ + getProductLabel: function( horizontalIndex, verticalIndex ) { + const horizontalPartitions = this.areaDisplay.partitionsProperties.horizontal.value; + const verticalPartitions = this.areaDisplay.partitionsProperties.vertical.value; + + return _.find( this.productLabels, function( productLabel ) { + const partitions = productLabel.partitionedAreaProperty.value.partitions; + return partitions.get( Orientation.HORIZONTAL ) === horizontalPartitions[ horizontalIndex ] && + partitions.get( Orientation.VERTICAL ) === verticalPartitions[ verticalIndex ]; + } ); + }, - // Handle each row separately - [ 0, 1 ].forEach( function( verticalIndex ) { - const verticalRange = rangesPair.vertical[ verticalIndex ]; - - // Bail if this row isn't shown at all. - if ( verticalRange === null ) { return; } - - const leftLabel = self.getProductLabel( 0, verticalIndex ); - const rightLabel = self.getProductLabel( 1, verticalIndex ); - - // We may not be able to access labels if we are in a partial state (some properties have changed, but others - // have not). - if ( leftLabel && rightLabel ) { - const isRightPartitionVisible = rightLabel.partitionedAreaProperty.value.visibleProperty.value; - const leftOverlapBump = 22; - const labelOverlapBump = 10; - - const hasLeftOverlap = rangesPair.vertical[ 1 ] !== null && leftLabel.left < -5; - const canAvoidLeftOverlap = leftLabel.top - leftOverlapBump >= verticalRange.min - 5; - const hasLabelOverlap = isRightPartitionVisible && leftLabel.right > rightLabel.left; - const canAvoidLabelOverlap = leftLabel.top - labelOverlapBump >= verticalRange.min - 3; - - let leftOffset = 0; - let rightOffset = 0; - if ( hasLeftOverlap && canAvoidLeftOverlap ) { - leftOffset = leftOverlapBump; - } - if ( hasLabelOverlap && canAvoidLabelOverlap ) { - const labelOverlapOffset = Math.max( labelOverlapBump, verticalRange.getLength() / 6 ); - leftOffset = Math.max( leftOffset, labelOverlapOffset ); - rightOffset = labelOverlapOffset; - } - - // Ignore Intellij inspections, we know what we are doing. - if ( leftOffset ) { - leftLabel.y -= leftOffset; - } - if ( rightOffset && isRightPartitionVisible ) { - rightLabel.y += rightOffset; - } + /** + * Positions all of the partial products labels. + * @protected + * @override + */ + positionProductLabels: function() { + const self = this; + + // {OrientationPair.>} - Current view ranges (if non-null) for each orientation + const rangesPair = this.areaDisplay.partitionsProperties.map( function( partitionsProperties, orientation ) { + return partitionsProperties.value.map( function( partition ) { + const range = partition.coordinateRangeProperty.value; + if ( range === null ) { + return null; } + return new Range( + orientation.modelToView( self.modelViewTransformProperty.value, range.min ), + orientation.modelToView( self.modelViewTransformProperty.value, range.max ) + ); } ); - }, + } ); - /** - * Position the partition labels (along the top/side). - * @private - * - * @param {Orientation} orientation - * @param {Node} labels - */ - positionPartitionLabels: function( orientation, labels ) { - const primaryRange = this.areaDisplay.primaryPartitionsProperty.get( orientation ).value.coordinateRangeProperty.value; - const secondaryRange = this.areaDisplay.secondaryPartitionsProperty.get( orientation ).value.coordinateRangeProperty.value; + // First, center the labels (if they have defined ranges) + this.productLabels.forEach( function( productLabel ) { + rangesPair.forEach( function( ranges, orientation ) { + const partition = productLabel.partitionedAreaProperty.value.partitions.get( orientation ); + const range = ranges[ _.indexOf( self.areaDisplay.partitionsProperties.get( orientation ).value, partition ) ]; + if ( range ) { + productLabel[ orientation.coordinate ] = range.getCenter(); + } + } ); + } ); - const min = orientation.modelToView( this.modelViewTransformProperty.value, primaryRange.min ); - const middle = orientation.modelToView( this.modelViewTransformProperty.value, primaryRange.max ); - const max = secondaryRange ? orientation.modelToView( this.modelViewTransformProperty.value, secondaryRange.max ) : 0; + // Handle each row separately + [ 0, 1 ].forEach( function( verticalIndex ) { + const verticalRange = rangesPair.vertical[ verticalIndex ]; - labels[ 0 ][ orientation.coordinate ] = ( min + middle ) / 2; - labels[ 1 ][ orientation.coordinate ] = ( middle + max ) / 2; + // Bail if this row isn't shown at all. + if ( verticalRange === null ) { return; } - const pad = orientation === Orientation.HORIZONTAL ? 2 : 0; + const leftLabel = self.getProductLabel( 0, verticalIndex ); + const rightLabel = self.getProductLabel( 1, verticalIndex ); - if ( secondaryRange && labels[ 0 ][ orientation.maxSide ] > labels[ 1 ][ orientation.minSide ] - pad * 2 ) { - const center = ( labels[ 0 ][ orientation.maxSide ] + labels[ 1 ][ orientation.minSide ] ) / 2; + // We may not be able to access labels if we are in a partial state (some properties have changed, but others + // have not). + if ( leftLabel && rightLabel ) { + const isRightPartitionVisible = rightLabel.partitionedAreaProperty.value.visibleProperty.value; + const leftOverlapBump = 22; + const labelOverlapBump = 10; - labels[ 0 ][ orientation.maxSide ] = center - pad; - labels[ 1 ][ orientation.minSide ] = center + pad; - } - }, - - /** - * Creates a partition label for the given orientation. - * @private - * - * @param {Property.} partitionProperty - * @param {Property.} secondaryPartitionProperty - The partition that is empty if there is only one - * @param {number} index - The index of the partition - * @param {Orientation} orientation - * @returns {Node} - */ - createPartitionLabel: function( partitionProperty, secondaryPartitionProperty, index, orientation ) { - const text = new Text( '', { - font: AreaModelCommonConstants.PROPORTIONAL_PARTITION_READOUT_FONT, - fill: new DynamicProperty( partitionProperty, { derive: 'colorProperty' } ) - } ); + const hasLeftOverlap = rangesPair.vertical[ 1 ] !== null && leftLabel.left < -5; + const canAvoidLeftOverlap = leftLabel.top - leftOverlapBump >= verticalRange.min - 5; + const hasLabelOverlap = isRightPartitionVisible && leftLabel.right > rightLabel.left; + const canAvoidLabelOverlap = leftLabel.top - labelOverlapBump >= verticalRange.min - 3; - const labelContainer = new Node( { - children: [ text ] - } ); + let leftOffset = 0; + let rightOffset = 0; + if ( hasLeftOverlap && canAvoidLeftOverlap ) { + leftOffset = leftOverlapBump; + } + if ( hasLabelOverlap && canAvoidLabelOverlap ) { + const labelOverlapOffset = Math.max( labelOverlapBump, verticalRange.getLength() / 6 ); + leftOffset = Math.max( leftOffset, labelOverlapOffset ); + rightOffset = labelOverlapOffset; + } - // Text label - new DynamicProperty( partitionProperty, { - derive: 'sizeProperty' - } ).link( function( size ) { - if ( size === null ) { - text.text = ''; + // Ignore Intellij inspections, we know what we are doing. + if ( leftOffset ) { + leftLabel.y -= leftOffset; } - else { - text.text = size.toRichString( false ); - text[ orientation.centerCoordinate ] = 0; + if ( rightOffset && isRightPartitionVisible ) { + rightLabel.y += rightOffset; } - } ); + } + } ); + }, + + /** + * Position the partition labels (along the top/side). + * @private + * + * @param {Orientation} orientation + * @param {Node} labels + */ + positionPartitionLabels: function( orientation, labels ) { + const primaryRange = this.areaDisplay.primaryPartitionsProperty.get( orientation ).value.coordinateRangeProperty.value; + const secondaryRange = this.areaDisplay.secondaryPartitionsProperty.get( orientation ).value.coordinateRangeProperty.value; + + const min = orientation.modelToView( this.modelViewTransformProperty.value, primaryRange.min ); + const middle = orientation.modelToView( this.modelViewTransformProperty.value, primaryRange.max ); + const max = secondaryRange ? orientation.modelToView( this.modelViewTransformProperty.value, secondaryRange.max ) : 0; - // Secondary coordinate - if ( orientation === Orientation.HORIZONTAL ) { - labelContainer.top = AreaModelCommonConstants.PROPORTIONAL_RANGE_OFFSET.y + 4; + labels[ 0 ][ orientation.coordinate ] = ( min + middle ) / 2; + labels[ 1 ][ orientation.coordinate ] = ( middle + max ) / 2; + + const pad = orientation === Orientation.HORIZONTAL ? 2 : 0; + + if ( secondaryRange && labels[ 0 ][ orientation.maxSide ] > labels[ 1 ][ orientation.minSide ] - pad * 2 ) { + const center = ( labels[ 0 ][ orientation.maxSide ] + labels[ 1 ][ orientation.minSide ] ) / 2; + + labels[ 0 ][ orientation.maxSide ] = center - pad; + labels[ 1 ][ orientation.minSide ] = center + pad; + } + }, + + /** + * Creates a partition label for the given orientation. + * @private + * + * @param {Property.} partitionProperty + * @param {Property.} secondaryPartitionProperty - The partition that is empty if there is only one + * @param {number} index - The index of the partition + * @param {Orientation} orientation + * @returns {Node} + */ + createPartitionLabel: function( partitionProperty, secondaryPartitionProperty, index, orientation ) { + const text = new Text( '', { + font: AreaModelCommonConstants.PROPORTIONAL_PARTITION_READOUT_FONT, + fill: new DynamicProperty( partitionProperty, { derive: 'colorProperty' } ) + } ); + + const labelContainer = new Node( { + children: [ text ] + } ); + + // Text label + new DynamicProperty( partitionProperty, { + derive: 'sizeProperty' + } ).link( function( size ) { + if ( size === null ) { + text.text = ''; } else { - labelContainer.left = AreaModelCommonConstants.PROPORTIONAL_RANGE_OFFSET.x + 6; + text.text = size.toRichString( false ); + text[ orientation.centerCoordinate ] = 0; } + } ); - const partitionVisibleProperty = new DynamicProperty( partitionProperty, { derive: 'visibleProperty' } ); - const secondaryPartitionSizeProperty = new DynamicProperty( secondaryPartitionProperty, { derive: 'sizeProperty' } ); + // Secondary coordinate + if ( orientation === Orientation.HORIZONTAL ) { + labelContainer.top = AreaModelCommonConstants.PROPORTIONAL_RANGE_OFFSET.y + 4; + } + else { + labelContainer.left = AreaModelCommonConstants.PROPORTIONAL_RANGE_OFFSET.x + 6; + } - Property.multilink( - [ partitionVisibleProperty, secondaryPartitionSizeProperty ], - function( visible, secondarySize ) { - labelContainer.visible = visible && secondarySize !== null; - } ); + const partitionVisibleProperty = new DynamicProperty( partitionProperty, { derive: 'visibleProperty' } ); + const secondaryPartitionSizeProperty = new DynamicProperty( secondaryPartitionProperty, { derive: 'sizeProperty' } ); - return labelContainer; - } - } ); -} ); + Property.multilink( + [ partitionVisibleProperty, secondaryPartitionSizeProperty ], + function( visible, secondarySize ) { + labelContainer.visible = visible && secondarySize !== null; + } ); + + return labelContainer; + } +} ); \ No newline at end of file diff --git a/js/proportional/view/ProportionalAreaGridLinesNode.js b/js/proportional/view/ProportionalAreaGridLinesNode.js index fa0d3cf3..a47acedb 100644 --- a/js/proportional/view/ProportionalAreaGridLinesNode.js +++ b/js/proportional/view/ProportionalAreaGridLinesNode.js @@ -7,55 +7,52 @@ * * @author Jonathan Olson */ -define( require => { - 'use strict'; - - // modules - const areaModelCommon = require( 'AREA_MODEL_COMMON/areaModelCommon' ); - const AreaModelCommonColorProfile = require( 'AREA_MODEL_COMMON/common/view/AreaModelCommonColorProfile' ); - const inherit = require( 'PHET_CORE/inherit' ); - const Path = require( 'SCENERY/nodes/Path' ); - const Property = require( 'AXON/Property' ); - const Shape = require( 'KITE/Shape' ); - - // constants - const GRID_LINE_WIDTH = 0.5; - const HALF_GRID_LINE_WIDTH = GRID_LINE_WIDTH / 2; - - /** - * @constructor - * @extends {Node} - * - * @param {Property.} areaProperty - * @param {Property.} modelViewTransformProperty - */ - function ProportionalAreaGridLinesNode( areaProperty, modelViewTransformProperty ) { - const self = this; - - Path.call( this, null, { - stroke: AreaModelCommonColorProfile.gridLineProperty - } ); - - Property.multilink( [ areaProperty, modelViewTransformProperty ], function( area, modelViewTransform ) { - const maxX = modelViewTransform.modelToViewX( area.maximumSize ); - const maxY = modelViewTransform.modelToViewY( area.maximumSize ); - - const shape = new Shape(); - for ( let i = area.gridSpacing; i < area.maximumSize; i += area.gridSpacing ) { - const x = modelViewTransform.modelToViewX( i ); - const y = modelViewTransform.modelToViewY( i ); - - shape.moveTo( HALF_GRID_LINE_WIDTH, y ); - shape.lineTo( maxX - HALF_GRID_LINE_WIDTH, y ); - - shape.moveTo( x, HALF_GRID_LINE_WIDTH ); - shape.lineTo( x, maxY - HALF_GRID_LINE_WIDTH ); - } - self.shape = shape; - } ); - } - - areaModelCommon.register( 'ProportionalAreaGridLinesNode', ProportionalAreaGridLinesNode ); - - return inherit( Path, ProportionalAreaGridLinesNode ); -} ); + +import Property from '../../../../axon/js/Property.js'; +import Shape from '../../../../kite/js/Shape.js'; +import inherit from '../../../../phet-core/js/inherit.js'; +import Path from '../../../../scenery/js/nodes/Path.js'; +import areaModelCommon from '../../areaModelCommon.js'; +import AreaModelCommonColorProfile from '../../common/view/AreaModelCommonColorProfile.js'; + +// constants +const GRID_LINE_WIDTH = 0.5; +const HALF_GRID_LINE_WIDTH = GRID_LINE_WIDTH / 2; + +/** + * @constructor + * @extends {Node} + * + * @param {Property.} areaProperty + * @param {Property.} modelViewTransformProperty + */ +function ProportionalAreaGridLinesNode( areaProperty, modelViewTransformProperty ) { + const self = this; + + Path.call( this, null, { + stroke: AreaModelCommonColorProfile.gridLineProperty + } ); + + Property.multilink( [ areaProperty, modelViewTransformProperty ], function( area, modelViewTransform ) { + const maxX = modelViewTransform.modelToViewX( area.maximumSize ); + const maxY = modelViewTransform.modelToViewY( area.maximumSize ); + + const shape = new Shape(); + for ( let i = area.gridSpacing; i < area.maximumSize; i += area.gridSpacing ) { + const x = modelViewTransform.modelToViewX( i ); + const y = modelViewTransform.modelToViewY( i ); + + shape.moveTo( HALF_GRID_LINE_WIDTH, y ); + shape.lineTo( maxX - HALF_GRID_LINE_WIDTH, y ); + + shape.moveTo( x, HALF_GRID_LINE_WIDTH ); + shape.lineTo( x, maxY - HALF_GRID_LINE_WIDTH ); + } + self.shape = shape; + } ); +} + +areaModelCommon.register( 'ProportionalAreaGridLinesNode', ProportionalAreaGridLinesNode ); + +inherit( Path, ProportionalAreaGridLinesNode ); +export default ProportionalAreaGridLinesNode; \ No newline at end of file diff --git a/js/proportional/view/ProportionalAreaScreenView.js b/js/proportional/view/ProportionalAreaScreenView.js index fc7c02a3..7a01237c 100644 --- a/js/proportional/view/ProportionalAreaScreenView.js +++ b/js/proportional/view/ProportionalAreaScreenView.js @@ -7,281 +7,277 @@ * * @author Jonathan Olson */ -define( require => { - 'use strict'; - // modules - const AlignGroup = require( 'SCENERY/nodes/AlignGroup' ); - const areaModelCommon = require( 'AREA_MODEL_COMMON/areaModelCommon' ); - const AreaModelCommonA11yStrings = require( 'AREA_MODEL_COMMON/AreaModelCommonA11yStrings' ); - const AreaModelCommonColorProfile = require( 'AREA_MODEL_COMMON/common/view/AreaModelCommonColorProfile' ); - const AreaModelCommonConstants = require( 'AREA_MODEL_COMMON/common/AreaModelCommonConstants' ); - const AreaModelCommonGlobals = require( 'AREA_MODEL_COMMON/common/AreaModelCommonGlobals' ); - const AreaScreenView = require( 'AREA_MODEL_COMMON/common/view/AreaScreenView' ); - const Checkbox = require( 'SUN/Checkbox' ); - const DerivedProperty = require( 'AXON/DerivedProperty' ); - const DynamicProperty = require( 'AXON/DynamicProperty' ); - const HBox = require( 'SCENERY/nodes/HBox' ); - const inherit = require( 'PHET_CORE/inherit' ); - const merge = require( 'PHET_CORE/merge' ); - const PartitionLineChoice = require( 'AREA_MODEL_COMMON/proportional/model/PartitionLineChoice' ); - const PartitionRadioButtonGroup = require( 'AREA_MODEL_COMMON/proportional/view/PartitionRadioButtonGroup' ); - const Path = require( 'SCENERY/nodes/Path' ); - const ProportionalAreaDisplayNode = require( 'AREA_MODEL_COMMON/proportional/view/ProportionalAreaDisplayNode' ); - const ProportionalAreaModel = require( 'AREA_MODEL_COMMON/proportional/model/ProportionalAreaModel' ); - const ProportionalFactorsNode = require( 'AREA_MODEL_COMMON/proportional/view/ProportionalFactorsNode' ); - const Rectangle = require( 'SCENERY/nodes/Rectangle' ); - const SceneRadioButtonGroup = require( 'AREA_MODEL_COMMON/proportional/view/SceneRadioButtonGroup' ); - const Shape = require( 'KITE/Shape' ); - const Text = require( 'SCENERY/nodes/Text' ); - const VBox = require( 'SCENERY/nodes/VBox' ); +import DerivedProperty from '../../../../axon/js/DerivedProperty.js'; +import DynamicProperty from '../../../../axon/js/DynamicProperty.js'; +import Shape from '../../../../kite/js/Shape.js'; +import inherit from '../../../../phet-core/js/inherit.js'; +import merge from '../../../../phet-core/js/merge.js'; +import AlignGroup from '../../../../scenery/js/nodes/AlignGroup.js'; +import HBox from '../../../../scenery/js/nodes/HBox.js'; +import Path from '../../../../scenery/js/nodes/Path.js'; +import Rectangle from '../../../../scenery/js/nodes/Rectangle.js'; +import Text from '../../../../scenery/js/nodes/Text.js'; +import VBox from '../../../../scenery/js/nodes/VBox.js'; +import Checkbox from '../../../../sun/js/Checkbox.js'; +import areaModelCommonStrings from '../../area-model-common-strings.js'; +import areaModelCommon from '../../areaModelCommon.js'; +import AreaModelCommonA11yStrings from '../../AreaModelCommonA11yStrings.js'; +import AreaModelCommonConstants from '../../common/AreaModelCommonConstants.js'; +import AreaModelCommonGlobals from '../../common/AreaModelCommonGlobals.js'; +import AreaModelCommonColorProfile from '../../common/view/AreaModelCommonColorProfile.js'; +import AreaScreenView from '../../common/view/AreaScreenView.js'; +import PartitionLineChoice from '../model/PartitionLineChoice.js'; +import ProportionalAreaModel from '../model/ProportionalAreaModel.js'; +import PartitionRadioButtonGroup from './PartitionRadioButtonGroup.js'; +import ProportionalAreaDisplayNode from './ProportionalAreaDisplayNode.js'; +import ProportionalFactorsNode from './ProportionalFactorsNode.js'; +import SceneRadioButtonGroup from './SceneRadioButtonGroup.js'; - // strings - const partitionString = require( 'string!AREA_MODEL_COMMON/partition' ); +const partitionString = areaModelCommonStrings.partition; - // a11y strings - const base10AreaTilesString = AreaModelCommonA11yStrings.base10AreaTiles.value; - const countingNumbersDescriptionString = AreaModelCommonA11yStrings.countingNumbersDescription.value; - const countingNumbersLabelString = AreaModelCommonA11yStrings.countingNumbersLabel.value; - const gridLinesLabelString = AreaModelCommonA11yStrings.gridLinesLabel.value; +// a11y strings +const base10AreaTilesString = AreaModelCommonA11yStrings.base10AreaTiles.value; +const countingNumbersDescriptionString = AreaModelCommonA11yStrings.countingNumbersDescription.value; +const countingNumbersLabelString = AreaModelCommonA11yStrings.countingNumbersLabel.value; +const gridLinesLabelString = AreaModelCommonA11yStrings.gridLinesLabel.value; - // constants - const RADIO_ICON_SIZE = 30; +// constants +const RADIO_ICON_SIZE = 30; - /** - * @constructor - * @extends {AreaScreenView} - * - * @param {ProportionalAreaModel} model - * @param {Object} [options] - */ - function ProportionalAreaScreenView( model, options ) { - assert && assert( model instanceof ProportionalAreaModel ); +/** + * @constructor + * @extends {AreaScreenView} + * + * @param {ProportionalAreaModel} model + * @param {Object} [options] + */ +function ProportionalAreaScreenView( model, options ) { + assert && assert( model instanceof ProportionalAreaModel ); - options = merge( { - decimalPlaces: 0, - isProportional: true - }, options ); + options = merge( { + decimalPlaces: 0, + isProportional: true + }, options ); - // @private {Node} - Scene selection, created before super call since it will be added in it. - this.sceneSelectionNode = new SceneRadioButtonGroup( model ); + // @private {Node} - Scene selection, created before super call since it will be added in it. + this.sceneSelectionNode = new SceneRadioButtonGroup( model ); - const currentAreaOrientationProperty = new DynamicProperty( model.currentAreaProperty, { - derive: 'visiblePartitionOrientationProperty', - bidirectional: true - } ); + const currentAreaOrientationProperty = new DynamicProperty( model.currentAreaProperty, { + derive: 'visiblePartitionOrientationProperty', + bidirectional: true + } ); - // Should have its own align group, so we don't modify other sizes - const partitionSelectionAlignGroup = new AlignGroup(); + // Should have its own align group, so we don't modify other sizes + const partitionSelectionAlignGroup = new AlignGroup(); - // @private {Node} - Allows controlling which partition is currently visible (if we only show one) - this.partitionSelectionPanel = this.createPanelContent( - partitionString, - AreaModelCommonGlobals.panelAlignGroup, - new PartitionRadioButtonGroup( currentAreaOrientationProperty, partitionSelectionAlignGroup ) - ); + // @private {Node} - Allows controlling which partition is currently visible (if we only show one) + this.partitionSelectionPanel = this.createPanelContent( + partitionString, + AreaModelCommonGlobals.panelAlignGroup, + new PartitionRadioButtonGroup( currentAreaOrientationProperty, partitionSelectionAlignGroup ) + ); - AreaScreenView.call( this, model, options ); + AreaScreenView.call( this, model, options ); - // Checkboxes - const gridCheckbox = new Checkbox( this.createGridIconNode(), model.gridLinesVisibleProperty, { - // a11y - labelTagName: 'label', - labelContent: gridLinesLabelString - } ); - const tileCheckbox = new Checkbox( this.createTileIconNode(), model.tilesVisibleProperty, { - // a11y - labelTagName: 'label', - labelContent: base10AreaTilesString - } ); - const countingCheckbox = new Checkbox( this.createCountingIconNode(), model.countingVisibleProperty, { - // a11y - labelTagName: 'label', - labelContent: countingNumbersLabelString, - descriptionContent: countingNumbersDescriptionString - } ); + // Checkboxes + const gridCheckbox = new Checkbox( this.createGridIconNode(), model.gridLinesVisibleProperty, { + // a11y + labelTagName: 'label', + labelContent: gridLinesLabelString + } ); + const tileCheckbox = new Checkbox( this.createTileIconNode(), model.tilesVisibleProperty, { + // a11y + labelTagName: 'label', + labelContent: base10AreaTilesString + } ); + const countingCheckbox = new Checkbox( this.createCountingIconNode(), model.countingVisibleProperty, { + // a11y + labelTagName: 'label', + labelContent: countingNumbersLabelString, + descriptionContent: countingNumbersDescriptionString + } ); - const checkboxContainer = new VBox( { - children: [gridCheckbox, countingCheckbox, tileCheckbox], - align: 'left', - spacing: 20, - // Manual positioning works best here - top: 50, - left: 600 - } ); - this.addChild( checkboxContainer ); + const checkboxContainer = new VBox( { + children: [ gridCheckbox, countingCheckbox, tileCheckbox ], + align: 'left', + spacing: 20, + // Manual positioning works best here + top: 50, + left: 600 + } ); + this.addChild( checkboxContainer ); - model.currentAreaProperty.link( function( area ) { - checkboxContainer.removeAllChildren(); + model.currentAreaProperty.link( function( area ) { + checkboxContainer.removeAllChildren(); - // Don't show the grid/tiles checkboxes if counting is enabled - if ( !area.countingAvailable ) { - checkboxContainer.addChild( gridCheckbox ); - if ( area.tilesAvailable ) { - checkboxContainer.addChild( tileCheckbox ); - } + // Don't show the grid/tiles checkboxes if counting is enabled + if ( !area.countingAvailable ) { + checkboxContainer.addChild( gridCheckbox ); + if ( area.tilesAvailable ) { + checkboxContainer.addChild( tileCheckbox ); } - else { - checkboxContainer.addChild( countingCheckbox ); - } - } ); + } + else { + checkboxContainer.addChild( countingCheckbox ); + } + } ); - // "Play Area" (a11y) - this.pdomPlayAreaNode.accessibleOrder = [ - this.areaDisplayNode, - this.factorsBox, - this.areaBox, - this.productsSelectionPanel, - this.calculationSelectionPanel, - this.partitionSelectionPanel, - this.calculationNode - ].filter( function( node ) { return node !== undefined; } ); // this.partitionSelectionPanel may not exist + // "Play Area" (a11y) + this.pdomPlayAreaNode.accessibleOrder = [ + this.areaDisplayNode, + this.factorsBox, + this.areaBox, + this.productsSelectionPanel, + this.calculationSelectionPanel, + this.partitionSelectionPanel, + this.calculationNode + ].filter( function( node ) { return node !== undefined; } ); // this.partitionSelectionPanel may not exist - // "Control Panel" (a11y) - this.pdomControlAreaNode.accessibleOrder = [ - gridCheckbox, - tileCheckbox, - countingCheckbox, - this.sceneSelectionNode, - this.resetAllButton - ]; - } + // "Control Panel" (a11y) + this.pdomControlAreaNode.accessibleOrder = [ + gridCheckbox, + tileCheckbox, + countingCheckbox, + this.sceneSelectionNode, + this.resetAllButton + ]; +} - areaModelCommon.register( 'ProportionalAreaScreenView', ProportionalAreaScreenView ); +areaModelCommon.register( 'ProportionalAreaScreenView', ProportionalAreaScreenView ); - return inherit( AreaScreenView, ProportionalAreaScreenView, { - /** - * @protected - * @override - * - * @returns {Array.} - */ - getRightAlignNodes: function() { - return AreaScreenView.prototype.getRightAlignNodes.call( this ).concat( [this.sceneSelectionNode] ); - }, - - /** - * @protected - * @override - * - * @returns {Property.>} - */ - getSelectionNodesProperty: function() { - const self = this; +export default inherit( AreaScreenView, ProportionalAreaScreenView, { + /** + * @protected + * @override + * + * @returns {Array.} + */ + getRightAlignNodes: function() { + return AreaScreenView.prototype.getRightAlignNodes.call( this ).concat( [ this.sceneSelectionNode ] ); + }, - // Use a Property here so we don't recreate when we don't have to (just on area changes) - const hasPartitionSelectionProperty = new DerivedProperty( [this.model.currentAreaProperty], function( area ) { - return area.partitionLineChoice === PartitionLineChoice.ONE; - } ); + /** + * @protected + * @override + * + * @returns {Property.>} + */ + getSelectionNodesProperty: function() { + const self = this; - // Conditionally include our partition selection on top of what else is included - return new DerivedProperty( - [AreaScreenView.prototype.getSelectionNodesProperty.call( this ), hasPartitionSelectionProperty], - function( selectionNodes, hasPartitionSelection ) { - return hasPartitionSelection ? selectionNodes.concat( [self.partitionSelectionPanel] ) : selectionNodes; - } ); - }, + // Use a Property here so we don't recreate when we don't have to (just on area changes) + const hasPartitionSelectionProperty = new DerivedProperty( [ this.model.currentAreaProperty ], function( area ) { + return area.partitionLineChoice === PartitionLineChoice.ONE; + } ); - /** - * Creates the main area display view for the screen. - * @public - * @override - * - * @param {ProportionalAreaModel} model - * @returns {ProportionalAreaDisplayNode} - */ - createAreaDisplayNode: function( model ) { - return new ProportionalAreaDisplayNode( model.areaDisplay, model.partialProductsChoiceProperty, { - gridLinesVisibleProperty: model.gridLinesVisibleProperty, - tilesVisibleProperty: model.tilesVisibleProperty, - countingVisibleProperty: model.countingVisibleProperty, - useTileLikeBackground: this.useTileLikeBackground, - useLargeArea: this.useLargeArea - }, { - translation: this.getDisplayTranslation() + // Conditionally include our partition selection on top of what else is included + return new DerivedProperty( + [ AreaScreenView.prototype.getSelectionNodesProperty.call( this ), hasPartitionSelectionProperty ], + function( selectionNodes, hasPartitionSelection ) { + return hasPartitionSelection ? selectionNodes.concat( [ self.partitionSelectionPanel ] ) : selectionNodes; } ); - }, + }, - /** - * Creates the "factors" (dimensions) content for the accordion box. - * @public - * @override - * - * @param {ProportionalAreaModel} model - * @param {number} decimalPlaces - * @returns {Node} - */ - createFactorsNode: function( model, decimalPlaces ) { - return new ProportionalFactorsNode( model.currentAreaProperty, model.areaDisplay.activeTotalProperties, decimalPlaces ); - }, + /** + * Creates the main area display view for the screen. + * @public + * @override + * + * @param {ProportionalAreaModel} model + * @returns {ProportionalAreaDisplayNode} + */ + createAreaDisplayNode: function( model ) { + return new ProportionalAreaDisplayNode( model.areaDisplay, model.partialProductsChoiceProperty, { + gridLinesVisibleProperty: model.gridLinesVisibleProperty, + tilesVisibleProperty: model.tilesVisibleProperty, + countingVisibleProperty: model.countingVisibleProperty, + useTileLikeBackground: this.useTileLikeBackground, + useLargeArea: this.useLargeArea + }, { + translation: this.getDisplayTranslation() + } ); + }, - /** - * Creates a grid icon. - * @private - * - * @returns {Node} - */ - createGridIconNode: function() { - const gridIconShape = new Shape() - .moveTo( RADIO_ICON_SIZE / 4, 0 ) - .lineTo( RADIO_ICON_SIZE / 4, RADIO_ICON_SIZE ) - .moveTo( RADIO_ICON_SIZE / 2, 0 ) - .lineTo( RADIO_ICON_SIZE / 2, RADIO_ICON_SIZE ) - .moveTo( RADIO_ICON_SIZE * 3 / 4, 0 ) - .lineTo( RADIO_ICON_SIZE * 3 / 4, RADIO_ICON_SIZE ) - .moveTo( 0, RADIO_ICON_SIZE / 4 ) - .lineTo( RADIO_ICON_SIZE, RADIO_ICON_SIZE / 4 ) - .moveTo( 0, RADIO_ICON_SIZE / 2 ) - .lineTo( RADIO_ICON_SIZE, RADIO_ICON_SIZE / 2 ) - .moveTo( 0, RADIO_ICON_SIZE * 3 / 4 ) - .lineTo( RADIO_ICON_SIZE, RADIO_ICON_SIZE * 3 / 4 ); - return new Path( gridIconShape, { - stroke: AreaModelCommonColorProfile.gridIconProperty - } ); - }, + /** + * Creates the "factors" (dimensions) content for the accordion box. + * @public + * @override + * + * @param {ProportionalAreaModel} model + * @param {number} decimalPlaces + * @returns {Node} + */ + createFactorsNode: function( model, decimalPlaces ) { + return new ProportionalFactorsNode( model.currentAreaProperty, model.areaDisplay.activeTotalProperties, decimalPlaces ); + }, - /** - * Creates a tile icon. - * @private - * - * @returns {Node} - */ - createTileIconNode: function() { - const tileIconOptions = { - fill: AreaModelCommonColorProfile.smallTileProperty, - stroke: AreaModelCommonColorProfile.tileIconStrokeProperty, - lineWidth: 0.5 - }; - const SMALL_TILE_ICON_SIZE = RADIO_ICON_SIZE / 10; - return new HBox( { - children: [ - new Rectangle( 0, 0, RADIO_ICON_SIZE, RADIO_ICON_SIZE, tileIconOptions ), - new Rectangle( 0, 0, SMALL_TILE_ICON_SIZE, RADIO_ICON_SIZE, tileIconOptions ), - new VBox( { - children: [ - new Rectangle( 0, 0, SMALL_TILE_ICON_SIZE, SMALL_TILE_ICON_SIZE, tileIconOptions ), - new Rectangle( 0, 0, SMALL_TILE_ICON_SIZE, SMALL_TILE_ICON_SIZE, tileIconOptions ), - new Rectangle( 0, 0, SMALL_TILE_ICON_SIZE, SMALL_TILE_ICON_SIZE, tileIconOptions ) - ], - spacing: 0 - } ) - ], - align: 'top', - spacing: RADIO_ICON_SIZE / 8 - } ); - }, + /** + * Creates a grid icon. + * @private + * + * @returns {Node} + */ + createGridIconNode: function() { + const gridIconShape = new Shape() + .moveTo( RADIO_ICON_SIZE / 4, 0 ) + .lineTo( RADIO_ICON_SIZE / 4, RADIO_ICON_SIZE ) + .moveTo( RADIO_ICON_SIZE / 2, 0 ) + .lineTo( RADIO_ICON_SIZE / 2, RADIO_ICON_SIZE ) + .moveTo( RADIO_ICON_SIZE * 3 / 4, 0 ) + .lineTo( RADIO_ICON_SIZE * 3 / 4, RADIO_ICON_SIZE ) + .moveTo( 0, RADIO_ICON_SIZE / 4 ) + .lineTo( RADIO_ICON_SIZE, RADIO_ICON_SIZE / 4 ) + .moveTo( 0, RADIO_ICON_SIZE / 2 ) + .lineTo( RADIO_ICON_SIZE, RADIO_ICON_SIZE / 2 ) + .moveTo( 0, RADIO_ICON_SIZE * 3 / 4 ) + .lineTo( RADIO_ICON_SIZE, RADIO_ICON_SIZE * 3 / 4 ); + return new Path( gridIconShape, { + stroke: AreaModelCommonColorProfile.gridIconProperty + } ); + }, - /** - * Creates a counting icon. - * @private - * - * @returns {Node} - */ - createCountingIconNode: function() { - // Hardcoded string, see https://github.com/phetsims/area-model-common/issues/104 - return new Text( '123', { - font: AreaModelCommonConstants.COUNTING_ICON_FONT - } ); - } - } ); -} ); + /** + * Creates a tile icon. + * @private + * + * @returns {Node} + */ + createTileIconNode: function() { + const tileIconOptions = { + fill: AreaModelCommonColorProfile.smallTileProperty, + stroke: AreaModelCommonColorProfile.tileIconStrokeProperty, + lineWidth: 0.5 + }; + const SMALL_TILE_ICON_SIZE = RADIO_ICON_SIZE / 10; + return new HBox( { + children: [ + new Rectangle( 0, 0, RADIO_ICON_SIZE, RADIO_ICON_SIZE, tileIconOptions ), + new Rectangle( 0, 0, SMALL_TILE_ICON_SIZE, RADIO_ICON_SIZE, tileIconOptions ), + new VBox( { + children: [ + new Rectangle( 0, 0, SMALL_TILE_ICON_SIZE, SMALL_TILE_ICON_SIZE, tileIconOptions ), + new Rectangle( 0, 0, SMALL_TILE_ICON_SIZE, SMALL_TILE_ICON_SIZE, tileIconOptions ), + new Rectangle( 0, 0, SMALL_TILE_ICON_SIZE, SMALL_TILE_ICON_SIZE, tileIconOptions ) + ], + spacing: 0 + } ) + ], + align: 'top', + spacing: RADIO_ICON_SIZE / 8 + } ); + }, + + /** + * Creates a counting icon. + * @private + * + * @returns {Node} + */ + createCountingIconNode: function() { + // Hardcoded string, see https://github.com/phetsims/area-model-common/issues/104 + return new Text( '123', { + font: AreaModelCommonConstants.COUNTING_ICON_FONT + } ); + } +} ); \ No newline at end of file diff --git a/js/proportional/view/ProportionalDragHandle.js b/js/proportional/view/ProportionalDragHandle.js index 812f9c74..21e8552c 100644 --- a/js/proportional/view/ProportionalDragHandle.js +++ b/js/proportional/view/ProportionalDragHandle.js @@ -8,208 +8,205 @@ * * @author Jonathan Olson */ -define( require => { - 'use strict'; - - // modules - const areaModelCommon = require( 'AREA_MODEL_COMMON/areaModelCommon' ); - const AreaModelCommonA11yStrings = require( 'AREA_MODEL_COMMON/AreaModelCommonA11yStrings' ); - const AreaModelCommonColorProfile = require( 'AREA_MODEL_COMMON/common/view/AreaModelCommonColorProfile' ); - const BooleanProperty = require( 'AXON/BooleanProperty' ); - const Circle = require( 'SCENERY/nodes/Circle' ); - const DragListener = require( 'SCENERY/listeners/DragListener' ); - const inherit = require( 'PHET_CORE/inherit' ); - const KeyboardDragListener = require( 'SCENERY/listeners/KeyboardDragListener' ); - const Line = require( 'SCENERY/nodes/Line' ); - const Node = require( 'SCENERY/nodes/Node' ); - const Orientation = require( 'PHET_CORE/Orientation' ); - const Property = require( 'AXON/Property' ); - const Shape = require( 'KITE/Shape' ); - const StringUtils = require( 'PHETCOMMON/util/StringUtils' ); - const Utils = require( 'DOT/Utils' ); - const Vector2 = require( 'DOT/Vector2' ); - const Vector2Property = require( 'DOT/Vector2Property' ); - - // a11y strings - const dragHandleString = AreaModelCommonA11yStrings.dragHandle.value; - const dragHandleDescriptionPatternString = AreaModelCommonA11yStrings.dragHandleDescriptionPattern.value; - - // constants - const DRAG_OFFSET = 8; - const DRAG_RADIUS = 10.5; - const CIRCLE_DRAG_OFFSET = DRAG_OFFSET + Math.sqrt( 2 ) / 2 * DRAG_RADIUS; - - /** - * @constructor - * @extends {Node} - * - * @param {Property.} areaProperty - * @param {OrientationPair.>} activeTotalProperties - * @param {Property.} modelViewTransformProperty - */ - function ProportionalDragHandle( areaProperty, activeTotalProperties, modelViewTransformProperty ) { - - const self = this; - - // {Property.} - Whether this is being dragged (we only apply offsets when dragged) - const draggedProperty = new BooleanProperty( false ); - - // The current view "offset" from where the pointer is compared to the point it is controlling - const offsetProperty = new Vector2Property( new Vector2( 0, 0 ) ); - - const line = new Line( { - stroke: AreaModelCommonColorProfile.proportionalDragHandleBorderProperty - } ); - const circle = new Circle( DRAG_RADIUS, { - touchArea: Shape.circle( 0, 0, DRAG_RADIUS * 2 ), - focusHighlight: Shape.circle( 0, 0, DRAG_RADIUS * 1.5 ), - fill: AreaModelCommonColorProfile.proportionalDragHandleBackgroundProperty, - stroke: AreaModelCommonColorProfile.proportionalDragHandleBorderProperty, - cursor: 'pointer', - - // a11y - tagName: 'div', - innerContent: dragHandleString, - focusable: true +import BooleanProperty from '../../../../axon/js/BooleanProperty.js'; +import Property from '../../../../axon/js/Property.js'; +import Utils from '../../../../dot/js/Utils.js'; +import Vector2 from '../../../../dot/js/Vector2.js'; +import Vector2Property from '../../../../dot/js/Vector2Property.js'; +import Shape from '../../../../kite/js/Shape.js'; +import inherit from '../../../../phet-core/js/inherit.js'; +import Orientation from '../../../../phet-core/js/Orientation.js'; +import StringUtils from '../../../../phetcommon/js/util/StringUtils.js'; +import DragListener from '../../../../scenery/js/listeners/DragListener.js'; +import KeyboardDragListener from '../../../../scenery/js/listeners/KeyboardDragListener.js'; +import Circle from '../../../../scenery/js/nodes/Circle.js'; +import Line from '../../../../scenery/js/nodes/Line.js'; +import Node from '../../../../scenery/js/nodes/Node.js'; +import areaModelCommon from '../../areaModelCommon.js'; +import AreaModelCommonA11yStrings from '../../AreaModelCommonA11yStrings.js'; +import AreaModelCommonColorProfile from '../../common/view/AreaModelCommonColorProfile.js'; + +// a11y strings +const dragHandleString = AreaModelCommonA11yStrings.dragHandle.value; +const dragHandleDescriptionPatternString = AreaModelCommonA11yStrings.dragHandleDescriptionPattern.value; + +// constants +const DRAG_OFFSET = 8; +const DRAG_RADIUS = 10.5; +const CIRCLE_DRAG_OFFSET = DRAG_OFFSET + Math.sqrt( 2 ) / 2 * DRAG_RADIUS; + +/** + * @constructor + * @extends {Node} + * + * @param {Property.} areaProperty + * @param {OrientationPair.>} activeTotalProperties + * @param {Property.} modelViewTransformProperty + */ +function ProportionalDragHandle( areaProperty, activeTotalProperties, modelViewTransformProperty ) { + + const self = this; + + // {Property.} - Whether this is being dragged (we only apply offsets when dragged) + const draggedProperty = new BooleanProperty( false ); + + // The current view "offset" from where the pointer is compared to the point it is controlling + const offsetProperty = new Vector2Property( new Vector2( 0, 0 ) ); + + const line = new Line( { + stroke: AreaModelCommonColorProfile.proportionalDragHandleBorderProperty + } ); + + const circle = new Circle( DRAG_RADIUS, { + touchArea: Shape.circle( 0, 0, DRAG_RADIUS * 2 ), + focusHighlight: Shape.circle( 0, 0, DRAG_RADIUS * 1.5 ), + fill: AreaModelCommonColorProfile.proportionalDragHandleBackgroundProperty, + stroke: AreaModelCommonColorProfile.proportionalDragHandleBorderProperty, + cursor: 'pointer', + + // a11y + tagName: 'div', + innerContent: dragHandleString, + focusable: true + } ); + + // Potential workaround for https://github.com/phetsims/area-model-common/issues/173 (Safari SVG dirty region issue) + circle.addChild( new Circle( DRAG_RADIUS + 10, { + pickable: false, + fill: 'transparent' + } ) ); + + areaProperty.link( function( area ) { + circle.descriptionContent = StringUtils.fillIn( dragHandleDescriptionPatternString, { + width: area.maximumSize, + height: area.maximumSize } ); + } ); - // Potential workaround for https://github.com/phetsims/area-model-common/issues/173 (Safari SVG dirty region issue) - circle.addChild( new Circle( DRAG_RADIUS + 10, { - pickable: false, - fill: 'transparent' - } ) ); + let initialOffset; - areaProperty.link( function( area ) { - circle.descriptionContent = StringUtils.fillIn( dragHandleDescriptionPatternString, { - width: area.maximumSize, - height: area.maximumSize - } ); - } ); + function updateOffsetProperty( event, listener ) { + const area = areaProperty.value; + const modelViewTransform = modelViewTransformProperty.value; - let initialOffset; + // We use somewhat complicated drag code, since we both snap AND have an offset from where the pointer + // actually is (and we want it to be efficient). + const pointerViewPoint = listener.parentPoint; + const viewPoint = pointerViewPoint.minusScalar( CIRCLE_DRAG_OFFSET ).minus( initialOffset ); + const modelPoint = modelViewTransform.viewToModelPosition( viewPoint ); - function updateOffsetProperty( event, listener ) { - const area = areaProperty.value; - const modelViewTransform = modelViewTransformProperty.value; + const snapSizeInverse = 1 / area.snapSize; - // We use somewhat complicated drag code, since we both snap AND have an offset from where the pointer - // actually is (and we want it to be efficient). - const pointerViewPoint = listener.parentPoint; - const viewPoint = pointerViewPoint.minusScalar( CIRCLE_DRAG_OFFSET ).minus( initialOffset ); - const modelPoint = modelViewTransform.viewToModelPosition( viewPoint ); + let width = Utils.roundSymmetric( modelPoint.x * snapSizeInverse ) / snapSizeInverse; + let height = Utils.roundSymmetric( modelPoint.y * snapSizeInverse ) / snapSizeInverse; - const snapSizeInverse = 1 / area.snapSize; + width = Utils.clamp( width, area.minimumSize, area.maximumSize ); + height = Utils.clamp( height, area.minimumSize, area.maximumSize ); - let width = Utils.roundSymmetric( modelPoint.x * snapSizeInverse ) / snapSizeInverse; - let height = Utils.roundSymmetric( modelPoint.y * snapSizeInverse ) / snapSizeInverse; + activeTotalProperties.horizontal.value = width; + activeTotalProperties.vertical.value = height; - width = Utils.clamp( width, area.minimumSize, area.maximumSize ); - height = Utils.clamp( height, area.minimumSize, area.maximumSize ); + offsetProperty.value = new Vector2( + viewPoint.x - modelViewTransform.modelToViewX( width ), + viewPoint.y - modelViewTransform.modelToViewY( height ) + ); + } - activeTotalProperties.horizontal.value = width; - activeTotalProperties.vertical.value = height; + const dragListener = new DragListener( { + targetNode: this, + applyOffset: false, + start: function( event, listener ) { + initialOffset = listener.localPoint.minusScalar( CIRCLE_DRAG_OFFSET ); + updateOffsetProperty( event, listener ); + }, + drag: updateOffsetProperty + } ); + dragListener.isPressedProperty.link( draggedProperty.set.bind( draggedProperty ) ); + + // Interrupt the drag when one of our parameters changes + areaProperty.lazyLink( dragListener.interrupt.bind( dragListener ) ); + modelViewTransformProperty.lazyLink( dragListener.interrupt.bind( dragListener ) ); + circle.addInputListener( dragListener ); + + Node.call( this, { + children: [ + line, + circle + ] + } ); + + const locationProperty = new Vector2Property( new Vector2( 0, 0 ) ); + + function updateLocationProperty() { + locationProperty.value = new Vector2( + activeTotalProperties.horizontal.value, + activeTotalProperties.vertical.value + ); + } - offsetProperty.value = new Vector2( - viewPoint.x - modelViewTransform.modelToViewX( width ), - viewPoint.y - modelViewTransform.modelToViewY( height ) - ); + updateLocationProperty(); + locationProperty.lazyLink( function( location ) { + activeTotalProperties.horizontal.value = location.x; + activeTotalProperties.vertical.value = location.y; + } ); + activeTotalProperties.horizontal.lazyLink( updateLocationProperty ); + activeTotalProperties.vertical.lazyLink( updateLocationProperty ); + + let keyboardListener; + Property.multilink( [ areaProperty, modelViewTransformProperty ], function( area, modelViewTransform ) { + if ( keyboardListener ) { + circle.interruptInput(); + circle.removeInputListener( keyboardListener ); + keyboardListener.dispose(); } - - const dragListener = new DragListener( { - targetNode: this, - applyOffset: false, - start: function( event, listener ) { - initialOffset = listener.localPoint.minusScalar( CIRCLE_DRAG_OFFSET ); - updateOffsetProperty( event, listener ); + keyboardListener = new KeyboardDragListener( { + downDelta: modelViewTransform.modelToViewDeltaX( area.snapSize ), + shiftDownDelta: modelViewTransform.modelToViewDeltaX( area.snapSize ), + transform: modelViewTransform, + drag: function( delta ) { + let width = activeTotalProperties.horizontal.value; + let height = activeTotalProperties.vertical.value; + + width += delta.x; + height += delta.y; + + width = Utils.roundToInterval( Utils.clamp( width, area.minimumSize, area.maximumSize ), area.snapSize ); + height = Utils.roundToInterval( Utils.clamp( height, area.minimumSize, area.maximumSize ), area.snapSize ); + + activeTotalProperties.horizontal.value = width; + activeTotalProperties.vertical.value = height; }, - drag: updateOffsetProperty - } ); - dragListener.isPressedProperty.link( draggedProperty.set.bind( draggedProperty ) ); - - // Interrupt the drag when one of our parameters changes - areaProperty.lazyLink( dragListener.interrupt.bind( dragListener ) ); - modelViewTransformProperty.lazyLink( dragListener.interrupt.bind( dragListener ) ); - circle.addInputListener( dragListener ); - - Node.call( this, { - children: [ - line, - circle - ] + moveOnHoldDelay: 750, + moveOnHoldInterval: 70 } ); - const locationProperty = new Vector2Property( new Vector2( 0, 0 ) ); + circle.addInputListener( keyboardListener ); + } ); - function updateLocationProperty() { - locationProperty.value = new Vector2( - activeTotalProperties.horizontal.value, - activeTotalProperties.vertical.value - ); + // Apply offsets while dragging for a smoother experience. + // See https://github.com/phetsims/area-model-common/issues/3 + Property.multilink( [ draggedProperty, offsetProperty ], function( dragged, offset ) { + let combinedOffset = 0; + if ( dragged ) { + // Project to the line y=x, and limit for when the user goes to 1x1 or the max. + combinedOffset = Utils.clamp( ( offset.x + offset.y ) / 2, -10, 10 ); } - - updateLocationProperty(); - locationProperty.lazyLink( function( location ) { - activeTotalProperties.horizontal.value = location.x; - activeTotalProperties.vertical.value = location.y; - } ); - activeTotalProperties.horizontal.lazyLink( updateLocationProperty ); - activeTotalProperties.vertical.lazyLink( updateLocationProperty ); - - let keyboardListener; - Property.multilink( [ areaProperty, modelViewTransformProperty ], function( area, modelViewTransform ) { - if ( keyboardListener ) { - circle.interruptInput(); - circle.removeInputListener( keyboardListener ); - keyboardListener.dispose(); - } - keyboardListener = new KeyboardDragListener( { - downDelta: modelViewTransform.modelToViewDeltaX( area.snapSize ), - shiftDownDelta: modelViewTransform.modelToViewDeltaX( area.snapSize ), - transform: modelViewTransform, - drag: function( delta ) { - let width = activeTotalProperties.horizontal.value; - let height = activeTotalProperties.vertical.value; - - width += delta.x; - height += delta.y; - - width = Utils.roundToInterval( Utils.clamp( width, area.minimumSize, area.maximumSize ), area.snapSize ); - height = Utils.roundToInterval( Utils.clamp( height, area.minimumSize, area.maximumSize ), area.snapSize ); - - activeTotalProperties.horizontal.value = width; - activeTotalProperties.vertical.value = height; - }, - moveOnHoldDelay: 750, - moveOnHoldInterval: 70 + line.x2 = line.y2 = combinedOffset + DRAG_OFFSET; + circle.x = circle.y = combinedOffset + CIRCLE_DRAG_OFFSET; + } ); + + // Update the offset of the drag handle + Orientation.VALUES.forEach( function( orientation ) { + Property.multilink( + [ activeTotalProperties.get( orientation ), modelViewTransformProperty ], + function( value, modelViewTransform ) { + self[ orientation.coordinate ] = orientation.modelToView( modelViewTransform, value ); } ); + } ); +} - circle.addInputListener( keyboardListener ); - } ); - - // Apply offsets while dragging for a smoother experience. - // See https://github.com/phetsims/area-model-common/issues/3 - Property.multilink( [ draggedProperty, offsetProperty ], function( dragged, offset ) { - let combinedOffset = 0; - if ( dragged ) { - // Project to the line y=x, and limit for when the user goes to 1x1 or the max. - combinedOffset = Utils.clamp( ( offset.x + offset.y ) / 2, -10, 10 ); - } - line.x2 = line.y2 = combinedOffset + DRAG_OFFSET; - circle.x = circle.y = combinedOffset + CIRCLE_DRAG_OFFSET; - } ); - - // Update the offset of the drag handle - Orientation.VALUES.forEach( function( orientation ) { - Property.multilink( - [ activeTotalProperties.get( orientation ), modelViewTransformProperty ], - function( value, modelViewTransform ) { - self[ orientation.coordinate ] = orientation.modelToView( modelViewTransform, value ); - } ); - } ); - } - - areaModelCommon.register( 'ProportionalDragHandle', ProportionalDragHandle ); +areaModelCommon.register( 'ProportionalDragHandle', ProportionalDragHandle ); - return inherit( Node, ProportionalDragHandle ); -} ); +inherit( Node, ProportionalDragHandle ); +export default ProportionalDragHandle; \ No newline at end of file diff --git a/js/proportional/view/ProportionalFactorsNode.js b/js/proportional/view/ProportionalFactorsNode.js index f3920414..41d8bca5 100644 --- a/js/proportional/view/ProportionalFactorsNode.js +++ b/js/proportional/view/ProportionalFactorsNode.js @@ -7,164 +7,160 @@ * * @author Jonathan Olson */ -define( require => { - 'use strict'; - // modules - const areaModelCommon = require( 'AREA_MODEL_COMMON/areaModelCommon' ); - const AreaModelCommonA11yStrings = require( 'AREA_MODEL_COMMON/AreaModelCommonA11yStrings' ); - const AreaModelCommonColorProfile = require( 'AREA_MODEL_COMMON/common/view/AreaModelCommonColorProfile' ); - const AreaModelCommonConstants = require( 'AREA_MODEL_COMMON/common/AreaModelCommonConstants' ); - const AreaModelCommonQueryParameters = require( 'AREA_MODEL_COMMON/common/AreaModelCommonQueryParameters' ); - const DerivedProperty = require( 'AXON/DerivedProperty' ); - const DynamicProperty = require( 'AXON/DynamicProperty' ); - const HBox = require( 'SCENERY/nodes/HBox' ); - const inherit = require( 'PHET_CORE/inherit' ); - const MathSymbols = require( 'SCENERY_PHET/MathSymbols' ); - const Node = require( 'SCENERY/nodes/Node' ); - const NumberPicker = require( 'SCENERY_PHET/NumberPicker' ); - const Orientation = require( 'PHET_CORE/Orientation' ); - const Property = require( 'AXON/Property' ); - const Range = require( 'DOT/Range' ); - const StringUtils = require( 'PHETCOMMON/util/StringUtils' ); - const Text = require( 'SCENERY/nodes/Text' ); - const Utils = require( 'DOT/Utils' ); - const validate = require( 'AXON/validate' ); +import DerivedProperty from '../../../../axon/js/DerivedProperty.js'; +import DynamicProperty from '../../../../axon/js/DynamicProperty.js'; +import Property from '../../../../axon/js/Property.js'; +import validate from '../../../../axon/js/validate.js'; +import Range from '../../../../dot/js/Range.js'; +import Utils from '../../../../dot/js/Utils.js'; +import inherit from '../../../../phet-core/js/inherit.js'; +import Orientation from '../../../../phet-core/js/Orientation.js'; +import StringUtils from '../../../../phetcommon/js/util/StringUtils.js'; +import MathSymbols from '../../../../scenery-phet/js/MathSymbols.js'; +import NumberPicker from '../../../../scenery-phet/js/NumberPicker.js'; +import HBox from '../../../../scenery/js/nodes/HBox.js'; +import Node from '../../../../scenery/js/nodes/Node.js'; +import Text from '../../../../scenery/js/nodes/Text.js'; +import areaModelCommon from '../../areaModelCommon.js'; +import AreaModelCommonA11yStrings from '../../AreaModelCommonA11yStrings.js'; +import AreaModelCommonConstants from '../../common/AreaModelCommonConstants.js'; +import AreaModelCommonQueryParameters from '../../common/AreaModelCommonQueryParameters.js'; +import AreaModelCommonColorProfile from '../../common/view/AreaModelCommonColorProfile.js'; - // a11y strings - const factorsTimesPatternString = AreaModelCommonA11yStrings.factorsTimesPattern.value; - const horizontalPickerString = AreaModelCommonA11yStrings.horizontalPicker.value; - const horizontalPickerDescriptionString = AreaModelCommonA11yStrings.horizontalPickerDescription.value; - const verticalPickerString = AreaModelCommonA11yStrings.verticalPicker.value; - const verticalPickerDescriptionString = AreaModelCommonA11yStrings.verticalPickerDescription.value; +// a11y strings +const factorsTimesPatternString = AreaModelCommonA11yStrings.factorsTimesPattern.value; +const horizontalPickerString = AreaModelCommonA11yStrings.horizontalPicker.value; +const horizontalPickerDescriptionString = AreaModelCommonA11yStrings.horizontalPickerDescription.value; +const verticalPickerString = AreaModelCommonA11yStrings.verticalPicker.value; +const verticalPickerDescriptionString = AreaModelCommonA11yStrings.verticalPickerDescription.value; - /** - * @constructor - * @extends {HBox} - * - * @param {Property.} currentAreaProperty - * @param {OrientationPair.>} - activeTotalProperties - * @param {number} decimalPlaces - The number of decimal places to show in the picker (when needed) - */ - function ProportionalFactorsNode( currentAreaProperty, activeTotalProperties, decimalPlaces ) { - Node.call( this ); - - const self = this; +/** + * @constructor + * @extends {HBox} + * + * @param {Property.} currentAreaProperty + * @param {OrientationPair.>} - activeTotalProperties + * @param {number} decimalPlaces - The number of decimal places to show in the picker (when needed) + */ +function ProportionalFactorsNode( currentAreaProperty, activeTotalProperties, decimalPlaces ) { + Node.call( this ); - if ( AreaModelCommonQueryParameters.rawMath ) { - self.tagName = 'div'; - Property.multilink( activeTotalProperties.values, function( horizontalTotal, verticalTotal ) { - self.innerContent = StringUtils.fillIn( factorsTimesPatternString, { - width: horizontalTotal, - height: verticalTotal - } ); - } ); - } - else { - const ns = 'http://www.w3.org/1998/Math/MathML'; - const verticalNode = new Node( { - // a11y - tagName: 'mn', - accessibleNamespace: ns - } ); - activeTotalProperties.vertical.link( function( verticalTotal ) { - verticalNode.innerContent = '' + verticalTotal; - } ); - const horizontalNode = new Node( { - // a11y - tagName: 'mn', - accessibleNamespace: ns - } ); - activeTotalProperties.horizontal.link( function( horizontalTotal ) { - horizontalNode.innerContent = '' + horizontalTotal; - } ); + const self = this; - const mathNode = new Node( { - tagName: 'math', - accessibleNamespace: ns, - children: [ - new Node( { - tagName: 'mrow', - accessibleNamespace: ns, - children: [ - verticalNode, - new Node( { - tagName: 'mo', - accessibleNamespace: ns, - innerContent: '×' - } ), - horizontalNode - ] - } ) - ] + if ( AreaModelCommonQueryParameters.rawMath ) { + self.tagName = 'div'; + Property.multilink( activeTotalProperties.values, function( horizontalTotal, verticalTotal ) { + self.innerContent = StringUtils.fillIn( factorsTimesPatternString, { + width: horizontalTotal, + height: verticalTotal } ); - this.addChild( mathNode ); - } + } ); + } + else { + const ns = 'http://www.w3.org/1998/Math/MathML'; + const verticalNode = new Node( { + // a11y + tagName: 'mn', + accessibleNamespace: ns + } ); + activeTotalProperties.vertical.link( function( verticalTotal ) { + verticalNode.innerContent = '' + verticalTotal; + } ); + const horizontalNode = new Node( { + // a11y + tagName: 'mn', + accessibleNamespace: ns + } ); + activeTotalProperties.horizontal.link( function( horizontalTotal ) { + horizontalNode.innerContent = '' + horizontalTotal; + } ); - this.addChild( new HBox( { + const mathNode = new Node( { + tagName: 'math', + accessibleNamespace: ns, children: [ - this.createPicker( Orientation.VERTICAL, currentAreaProperty, decimalPlaces ), - new Text( MathSymbols.TIMES, { font: AreaModelCommonConstants.FACTORS_TERM_FONT } ), - this.createPicker( Orientation.HORIZONTAL, currentAreaProperty, decimalPlaces ) - ], - spacing: 10 - } ) ); + new Node( { + tagName: 'mrow', + accessibleNamespace: ns, + children: [ + verticalNode, + new Node( { + tagName: 'mo', + accessibleNamespace: ns, + innerContent: '×' + } ), + horizontalNode + ] + } ) + ] + } ); + this.addChild( mathNode ); } - areaModelCommon.register( 'ProportionalFactorsNode', ProportionalFactorsNode ); + this.addChild( new HBox( { + children: [ + this.createPicker( Orientation.VERTICAL, currentAreaProperty, decimalPlaces ), + new Text( MathSymbols.TIMES, { font: AreaModelCommonConstants.FACTORS_TERM_FONT } ), + this.createPicker( Orientation.HORIZONTAL, currentAreaProperty, decimalPlaces ) + ], + spacing: 10 + } ) ); +} - return inherit( Node, ProportionalFactorsNode, { - /** - * Creates a picker that adjusts the specified orientation's total size. - * @private - * - * @param {Orientation} orientation - * @param {Property.} currentAreaProperty - * @param {number} decimalPlaces - */ - createPicker: function( orientation, currentAreaProperty, decimalPlaces ) { - validate( orientation, { validValues: Orientation.VALUES } ); +areaModelCommon.register( 'ProportionalFactorsNode', ProportionalFactorsNode ); - // {Property.>} - const currentTotalProperty = new DerivedProperty( [ currentAreaProperty ], function( area ) { - return area.activeTotalProperties.get( orientation ); - } ); +export default inherit( Node, ProportionalFactorsNode, { + /** + * Creates a picker that adjusts the specified orientation's total size. + * @private + * + * @param {Orientation} orientation + * @param {Property.} currentAreaProperty + * @param {number} decimalPlaces + */ + createPicker: function( orientation, currentAreaProperty, decimalPlaces ) { + validate( orientation, { validValues: Orientation.VALUES } ); - // {Property.} - const bidirectionalProperty = new DynamicProperty( currentTotalProperty, { - bidirectional: true - } ); + // {Property.>} + const currentTotalProperty = new DerivedProperty( [ currentAreaProperty ], function( area ) { + return area.activeTotalProperties.get( orientation ); + } ); - // {Property.} - const rangeProperty = new DerivedProperty( [ currentAreaProperty ], function( area ) { - return new Range( area.minimumSize, area.maximumSize ); - } ); + // {Property.} + const bidirectionalProperty = new DynamicProperty( currentTotalProperty, { + bidirectional: true + } ); - return new NumberPicker( bidirectionalProperty, rangeProperty, { - upFunction: function( value ) { - return Utils.toFixedNumber( value + currentAreaProperty.value.snapSize, decimalPlaces ); - }, - downFunction: function( value ) { - return Utils.toFixedNumber( value - currentAreaProperty.value.snapSize, decimalPlaces ); - }, - decimalPlaces: decimalPlaces, - scale: 1.5, - formatValue: function( value ) { - // Epsilon chosen to avoid round-off errors while not "rounding" any values in the decimals sims improperly. - if ( Utils.equalsEpsilon( value, Utils.roundSymmetric( value ), 1e-6 ) ) { - return Utils.toFixed( value, 0 ); - } - else { - return Utils.toFixed( value, 1 ); - } - }, - color: AreaModelCommonColorProfile.proportionalColorProperties.get( orientation ), + // {Property.} + const rangeProperty = new DerivedProperty( [ currentAreaProperty ], function( area ) { + return new Range( area.minimumSize, area.maximumSize ); + } ); - // a11y - labelContent: orientation === Orientation.HORIZONTAL ? horizontalPickerString : verticalPickerString, - descriptionContent: orientation === Orientation.HORIZONTAL ? horizontalPickerDescriptionString : verticalPickerDescriptionString, - a11yMapValue: value => Utils.toFixedNumber( value, decimalPlaces ) - } ); - } - } ); -} ); + return new NumberPicker( bidirectionalProperty, rangeProperty, { + upFunction: function( value ) { + return Utils.toFixedNumber( value + currentAreaProperty.value.snapSize, decimalPlaces ); + }, + downFunction: function( value ) { + return Utils.toFixedNumber( value - currentAreaProperty.value.snapSize, decimalPlaces ); + }, + decimalPlaces: decimalPlaces, + scale: 1.5, + formatValue: function( value ) { + // Epsilon chosen to avoid round-off errors while not "rounding" any values in the decimals sims improperly. + if ( Utils.equalsEpsilon( value, Utils.roundSymmetric( value ), 1e-6 ) ) { + return Utils.toFixed( value, 0 ); + } + else { + return Utils.toFixed( value, 1 ); + } + }, + color: AreaModelCommonColorProfile.proportionalColorProperties.get( orientation ), + + // a11y + labelContent: orientation === Orientation.HORIZONTAL ? horizontalPickerString : verticalPickerString, + descriptionContent: orientation === Orientation.HORIZONTAL ? horizontalPickerDescriptionString : verticalPickerDescriptionString, + a11yMapValue: value => Utils.toFixedNumber( value, decimalPlaces ) + } ); + } +} ); \ No newline at end of file diff --git a/js/proportional/view/ProportionalPartitionLineNode.js b/js/proportional/view/ProportionalPartitionLineNode.js index bd771f31..01627a71 100644 --- a/js/proportional/view/ProportionalPartitionLineNode.js +++ b/js/proportional/view/ProportionalPartitionLineNode.js @@ -7,241 +7,237 @@ * * @author Jonathan Olson */ -define( require => { - 'use strict'; - - // modules - const AccessibleSlider = require( 'SUN/accessibility/AccessibleSlider' ); - const areaModelCommon = require( 'AREA_MODEL_COMMON/areaModelCommon' ); - const AreaModelCommonA11yStrings = require( 'AREA_MODEL_COMMON/AreaModelCommonA11yStrings' ); - const AreaModelCommonColorProfile = require( 'AREA_MODEL_COMMON/common/view/AreaModelCommonColorProfile' ); - const AreaModelCommonConstants = require( 'AREA_MODEL_COMMON/common/AreaModelCommonConstants' ); - const ArrowNode = require( 'SCENERY_PHET/ArrowNode' ); - const BooleanProperty = require( 'AXON/BooleanProperty' ); - const DerivedProperty = require( 'AXON/DerivedProperty' ); - const DragListener = require( 'SCENERY/listeners/DragListener' ); - const DynamicProperty = require( 'AXON/DynamicProperty' ); - const FocusHighlightPath = require( 'SCENERY/accessibility/FocusHighlightPath' ); - const inherit = require( 'PHET_CORE/inherit' ); - const Line = require( 'SCENERY/nodes/Line' ); - const Matrix3 = require( 'DOT/Matrix3' ); - const Node = require( 'SCENERY/nodes/Node' ); - const Orientation = require( 'PHET_CORE/Orientation' ); - const OrientationPair = require( 'AREA_MODEL_COMMON/common/model/OrientationPair' ); - const Path = require( 'SCENERY/nodes/Path' ); - const Property = require( 'AXON/Property' ); - const Range = require( 'DOT/Range' ); - const Shape = require( 'KITE/Shape' ); - const Utils = require( 'DOT/Utils' ); - const validate = require( 'AXON/validate' ); - - // a11y strings - const horizontalPartitionHandleString = AreaModelCommonA11yStrings.horizontalPartitionHandle.value; - const horizontalPartitionHandleDescriptionString = AreaModelCommonA11yStrings.horizontalPartitionHandleDescription.value; - const verticalPartitionHandleString = AreaModelCommonA11yStrings.verticalPartitionHandle.value; - const verticalPartitionHandleDescriptionString = AreaModelCommonA11yStrings.verticalPartitionHandleDescription.value; - - /** - * @mixes AccessibleSlider - * @constructor - * @extends {Node} - * - * @param {ProportionalAreaDisplay} areaDisplay - * @param {Property.} modelViewTransformProperty - * @param {Orientation} orientation - */ - function ProportionalPartitionLineNode( areaDisplay, modelViewTransformProperty, orientation ) { - validate( orientation, { validValues: Orientation.VALUES } ); - - const self = this; - - Node.call( this ); - - // @private {ProportionalAreaDisplay} - this.areaDisplay = areaDisplay; - - const showHintArrowsProperty = areaDisplay.hasHintArrows.get( orientation ); - - let minHintArrow; - let maxHintArrow; - const hintOffset = 15; - const hintLength = 20; - const arrowOptions = { - fill: 'yellow', - pickable: false - }; - if ( orientation === Orientation.HORIZONTAL ) { - minHintArrow = new ArrowNode( -hintOffset, 0, -( hintLength + hintOffset ), 0, arrowOptions ); - maxHintArrow = new ArrowNode( hintOffset, 0, hintLength + hintOffset, 0, arrowOptions ); - } - else { - minHintArrow = new ArrowNode( 0, -hintOffset, 0, -( hintLength + hintOffset ), arrowOptions ); - maxHintArrow = new ArrowNode( 0, hintOffset, 0, hintLength + hintOffset, arrowOptions ); - } - showHintArrowsProperty.linkAttribute( minHintArrow, 'visible' ); - showHintArrowsProperty.linkAttribute( maxHintArrow, 'visible' ); - - const handleShape = ProportionalPartitionLineNode.HANDLE_ARROW_SHAPES.get( orientation ); - const handleMouseBounds = handleShape.bounds; - const handleTouchBounds = handleMouseBounds.dilated( 5 ); - - // We need to cut off the corners that would overlap between the two partition line handles, so we create a clipping - // area and intersect with that. See https://github.com/phetsims/area-model-common/issues/80. - let handleClipShape = new Shape().moveToPoint( handleTouchBounds.leftTop ) - .lineToPoint( handleTouchBounds.leftBottom ) - .lineToPoint( handleTouchBounds.rightBottom ) - .lineToPoint( handleTouchBounds.rightTop.blend( handleTouchBounds.rightBottom, 0.4 ) ) - .lineToPoint( handleTouchBounds.rightTop.blend( handleTouchBounds.leftTop, 0.4 ) ) - .close(); - if ( orientation === Orientation.VERTICAL ) { - handleClipShape = handleClipShape.transformed( Matrix3.rotation2( Math.PI ) ); - } - const handle = new Path( handleShape, { - mouseArea: Shape.bounds( handleMouseBounds ).shapeIntersection( handleClipShape ), - touchArea: Shape.bounds( handleTouchBounds ).shapeIntersection( handleClipShape ), - fill: areaDisplay.colorProperties.get( orientation ), - stroke: AreaModelCommonColorProfile.partitionLineBorderProperty, - cursor: 'pointer', - children: [ - minHintArrow, - maxHintArrow - ] - } ); +import BooleanProperty from '../../../../axon/js/BooleanProperty.js'; +import DerivedProperty from '../../../../axon/js/DerivedProperty.js'; +import DynamicProperty from '../../../../axon/js/DynamicProperty.js'; +import Property from '../../../../axon/js/Property.js'; +import validate from '../../../../axon/js/validate.js'; +import Matrix3 from '../../../../dot/js/Matrix3.js'; +import Range from '../../../../dot/js/Range.js'; +import Utils from '../../../../dot/js/Utils.js'; +import Shape from '../../../../kite/js/Shape.js'; +import inherit from '../../../../phet-core/js/inherit.js'; +import Orientation from '../../../../phet-core/js/Orientation.js'; +import ArrowNode from '../../../../scenery-phet/js/ArrowNode.js'; +import FocusHighlightPath from '../../../../scenery/js/accessibility/FocusHighlightPath.js'; +import DragListener from '../../../../scenery/js/listeners/DragListener.js'; +import Line from '../../../../scenery/js/nodes/Line.js'; +import Node from '../../../../scenery/js/nodes/Node.js'; +import Path from '../../../../scenery/js/nodes/Path.js'; +import AccessibleSlider from '../../../../sun/js/accessibility/AccessibleSlider.js'; +import areaModelCommon from '../../areaModelCommon.js'; +import AreaModelCommonA11yStrings from '../../AreaModelCommonA11yStrings.js'; +import AreaModelCommonConstants from '../../common/AreaModelCommonConstants.js'; +import OrientationPair from '../../common/model/OrientationPair.js'; +import AreaModelCommonColorProfile from '../../common/view/AreaModelCommonColorProfile.js'; + +// a11y strings +const horizontalPartitionHandleString = AreaModelCommonA11yStrings.horizontalPartitionHandle.value; +const horizontalPartitionHandleDescriptionString = AreaModelCommonA11yStrings.horizontalPartitionHandleDescription.value; +const verticalPartitionHandleString = AreaModelCommonA11yStrings.verticalPartitionHandle.value; +const verticalPartitionHandleDescriptionString = AreaModelCommonA11yStrings.verticalPartitionHandleDescription.value; - const line = new Line( { - stroke: AreaModelCommonColorProfile.partitionLineStrokeProperty, - lineWidth: 2, - cursor: 'pointer' - } ); +/** + * @mixes AccessibleSlider + * @constructor + * @extends {Node} + * + * @param {ProportionalAreaDisplay} areaDisplay + * @param {Property.} modelViewTransformProperty + * @param {Orientation} orientation + */ +function ProportionalPartitionLineNode( areaDisplay, modelViewTransformProperty, orientation ) { + validate( orientation, { validValues: Orientation.VALUES } ); - Node.call( this, { - children: [ - line, - handle - ] - } ); + const self = this; - // Relevant properties - const partitionSplitProperty = areaDisplay.partitionSplitProperties.get( orientation ); - const oppositeActiveTotalProperty = areaDisplay.activeTotalProperties.get( orientation.opposite ); - const activeTotalProperty = areaDisplay.activeTotalProperties.get( orientation ); - - // We need to reverse the accessible property for the vertical case. - // See https://github.com/phetsims/area-model-introduction/issues/2 - const accessibleProperty = orientation === Orientation.HORIZONTAL - ? partitionSplitProperty - : new DynamicProperty( new Property( partitionSplitProperty ), { - bidirectional: true, - map: function( v ) { return -v; }, - inverseMap: function( v ) { return -v; } - }, { - valueType: 'number' // AccessibleSlider doesn't want anything besides a number - } ); - const accessibleRangeProperty = new DerivedProperty( - [ activeTotalProperty, areaDisplay.snapSizeProperty ], - function( total, snapSize ) { - const size = total - snapSize; - return orientation === Orientation.HORIZONTAL ? new Range( 0, size ) : new Range( -size, 0 ); - } ); - - // a11y - this.initializeAccessibleSlider( accessibleProperty, accessibleRangeProperty, new BooleanProperty( true ), { - constrainValue: Utils.roundSymmetric, - keyboardStep: 1, - shiftKeyboardStep: 1, - pageKeyboardStep: 5, - ariaOrientation: orientation.layoutBoxOrientation, - a11yMapValue: function( v ) { - // Reverse the negation above for readouts - return ( orientation === Orientation.HORIZONTAL ? 1 : -1 ) * v; - }, - roundToStepSize: true - } ); + Node.call( this ); - this.labelTagName = 'label'; - this.labelContent = orientation === Orientation.HORIZONTAL ? verticalPartitionHandleString : horizontalPartitionHandleString; - this.descriptionContent = orientation === Orientation.HORIZONTAL ? verticalPartitionHandleDescriptionString : horizontalPartitionHandleDescriptionString; + // @private {ProportionalAreaDisplay} + this.areaDisplay = areaDisplay; - this.focusHighlight = new FocusHighlightPath( handleShape.getOffsetShape( 5 ) ); - handle.addChild( this.focusHighlight ); - this.focusHighlightLayerable = true; + const showHintArrowsProperty = areaDisplay.hasHintArrows.get( orientation ); - // Main coordinate (when dragging) - Property.multilink( [ partitionSplitProperty, modelViewTransformProperty ], function( split, modelViewTransform ) { - self[ orientation.coordinate ] = orientation.modelToView( modelViewTransform, split ); - } ); + let minHintArrow; + let maxHintArrow; + const hintOffset = 15; + const hintLength = 20; + const arrowOptions = { + fill: 'yellow', + pickable: false + }; + if ( orientation === Orientation.HORIZONTAL ) { + minHintArrow = new ArrowNode( -hintOffset, 0, -( hintLength + hintOffset ), 0, arrowOptions ); + maxHintArrow = new ArrowNode( hintOffset, 0, hintLength + hintOffset, 0, arrowOptions ); + } + else { + minHintArrow = new ArrowNode( 0, -hintOffset, 0, -( hintLength + hintOffset ), arrowOptions ); + maxHintArrow = new ArrowNode( 0, hintOffset, 0, hintLength + hintOffset, arrowOptions ); + } + showHintArrowsProperty.linkAttribute( minHintArrow, 'visible' ); + showHintArrowsProperty.linkAttribute( maxHintArrow, 'visible' ); + + const handleShape = ProportionalPartitionLineNode.HANDLE_ARROW_SHAPES.get( orientation ); + const handleMouseBounds = handleShape.bounds; + const handleTouchBounds = handleMouseBounds.dilated( 5 ); + + // We need to cut off the corners that would overlap between the two partition line handles, so we create a clipping + // area and intersect with that. See https://github.com/phetsims/area-model-common/issues/80. + let handleClipShape = new Shape().moveToPoint( handleTouchBounds.leftTop ) + .lineToPoint( handleTouchBounds.leftBottom ) + .lineToPoint( handleTouchBounds.rightBottom ) + .lineToPoint( handleTouchBounds.rightTop.blend( handleTouchBounds.rightBottom, 0.4 ) ) + .lineToPoint( handleTouchBounds.rightTop.blend( handleTouchBounds.leftTop, 0.4 ) ) + .close(); + if ( orientation === Orientation.VERTICAL ) { + handleClipShape = handleClipShape.transformed( Matrix3.rotation2( Math.PI ) ); + } - // Opposite coordinate (how wide the area is in the other direction) - Property.multilink( - [ oppositeActiveTotalProperty, modelViewTransformProperty ], - function( oppositeTotal, modelViewTransform ) { - const offsetValue = orientation.opposite.modelToView( modelViewTransform, oppositeTotal ) + - AreaModelCommonConstants.PARTITION_HANDLE_OFFSET; - handle[ orientation.opposite.coordinate ] = offsetValue; - line[ orientation.opposite.coordinate + '2' ] = offsetValue; - line.mouseArea = line.localBounds.dilated( 4 ); - line.touchArea = line.localBounds.dilated( 8 ); - } ); - - // Visibility - areaDisplay.partitionSplitVisibleProperties.get( orientation ).linkAttribute( self, 'visible' ); - - let dragHandler; - modelViewTransformProperty.link( function( modelViewTransform ) { - if ( dragHandler ) { - self.removeInputListener( dragHandler ); - dragHandler.dispose(); - } - dragHandler = new DragListener( { - transform: modelViewTransform, - drag: function( event, listener ) { - let value = listener.modelPoint[ orientation.coordinate ]; - - value = Utils.roundSymmetric( value / areaDisplay.partitionSnapSizeProperty.value ) * - areaDisplay.partitionSnapSizeProperty.value; - value = Utils.clamp( value, 0, activeTotalProperty.value ); - - // Hint arrows disappear when the actual split changes during a drag, see - // https://github.com/phetsims/area-model-common/issues/68 - const currentSplitValue = partitionSplitProperty.value; - if ( value !== currentSplitValue && value !== 0 ) { - showHintArrowsProperty.value = false; - } - - partitionSplitProperty.value = value; - }, - - end: function() { - if ( partitionSplitProperty.value === activeTotalProperty.value ) { - partitionSplitProperty.value = 0; - } - } - } ); - dragHandler.isUserControlledProperty.link( function( controlled ) { - areaDisplay.partitionSplitUserControlledProperties.get( orientation ).value = controlled; - } ); - self.addInputListener( dragHandler ); + const handle = new Path( handleShape, { + mouseArea: Shape.bounds( handleMouseBounds ).shapeIntersection( handleClipShape ), + touchArea: Shape.bounds( handleTouchBounds ).shapeIntersection( handleClipShape ), + fill: areaDisplay.colorProperties.get( orientation ), + stroke: AreaModelCommonColorProfile.partitionLineBorderProperty, + cursor: 'pointer', + children: [ + minHintArrow, + maxHintArrow + ] + } ); + + const line = new Line( { + stroke: AreaModelCommonColorProfile.partitionLineStrokeProperty, + lineWidth: 2, + cursor: 'pointer' + } ); + + Node.call( this, { + children: [ + line, + handle + ] + } ); + + // Relevant properties + const partitionSplitProperty = areaDisplay.partitionSplitProperties.get( orientation ); + const oppositeActiveTotalProperty = areaDisplay.activeTotalProperties.get( orientation.opposite ); + const activeTotalProperty = areaDisplay.activeTotalProperties.get( orientation ); + + // We need to reverse the accessible property for the vertical case. + // See https://github.com/phetsims/area-model-introduction/issues/2 + const accessibleProperty = orientation === Orientation.HORIZONTAL + ? partitionSplitProperty + : new DynamicProperty( new Property( partitionSplitProperty ), { + bidirectional: true, + map: function( v ) { return -v; }, + inverseMap: function( v ) { return -v; } + }, { + valueType: 'number' // AccessibleSlider doesn't want anything besides a number + } ); + const accessibleRangeProperty = new DerivedProperty( + [ activeTotalProperty, areaDisplay.snapSizeProperty ], + function( total, snapSize ) { + const size = total - snapSize; + return orientation === Orientation.HORIZONTAL ? new Range( 0, size ) : new Range( -size, 0 ); } ); - } - areaModelCommon.register( 'ProportionalPartitionLineNode', ProportionalPartitionLineNode ); + // a11y + this.initializeAccessibleSlider( accessibleProperty, accessibleRangeProperty, new BooleanProperty( true ), { + constrainValue: Utils.roundSymmetric, + keyboardStep: 1, + shiftKeyboardStep: 1, + pageKeyboardStep: 5, + ariaOrientation: orientation.layoutBoxOrientation, + a11yMapValue: function( v ) { + // Reverse the negation above for readouts + return ( orientation === Orientation.HORIZONTAL ? 1 : -1 ) * v; + }, + roundToStepSize: true + } ); - // Handle arrows - const arrowHalfLength = 10; - const arrowHalfWidth = 10; - const verticalArrowShape = new Shape() - .moveTo( -arrowHalfLength, 0 ) - .lineTo( arrowHalfLength, arrowHalfWidth ) - .lineTo( arrowHalfLength, -arrowHalfWidth ) - .close(); - const horizontalArrowShape = verticalArrowShape.transformed( Matrix3.rotation2( Math.PI / 2 ) ); + this.labelTagName = 'label'; + this.labelContent = orientation === Orientation.HORIZONTAL ? verticalPartitionHandleString : horizontalPartitionHandleString; + this.descriptionContent = orientation === Orientation.HORIZONTAL ? verticalPartitionHandleDescriptionString : horizontalPartitionHandleDescriptionString; - inherit( Node, ProportionalPartitionLineNode, {}, { - HANDLE_ARROW_SHAPES: new OrientationPair( horizontalArrowShape, verticalArrowShape ) + this.focusHighlight = new FocusHighlightPath( handleShape.getOffsetShape( 5 ) ); + handle.addChild( this.focusHighlight ); + this.focusHighlightLayerable = true; + + // Main coordinate (when dragging) + Property.multilink( [ partitionSplitProperty, modelViewTransformProperty ], function( split, modelViewTransform ) { + self[ orientation.coordinate ] = orientation.modelToView( modelViewTransform, split ); } ); - AccessibleSlider.mixInto( ProportionalPartitionLineNode ); + // Opposite coordinate (how wide the area is in the other direction) + Property.multilink( + [ oppositeActiveTotalProperty, modelViewTransformProperty ], + function( oppositeTotal, modelViewTransform ) { + const offsetValue = orientation.opposite.modelToView( modelViewTransform, oppositeTotal ) + + AreaModelCommonConstants.PARTITION_HANDLE_OFFSET; + handle[ orientation.opposite.coordinate ] = offsetValue; + line[ orientation.opposite.coordinate + '2' ] = offsetValue; + line.mouseArea = line.localBounds.dilated( 4 ); + line.touchArea = line.localBounds.dilated( 8 ); + } ); + + // Visibility + areaDisplay.partitionSplitVisibleProperties.get( orientation ).linkAttribute( self, 'visible' ); - return ProportionalPartitionLineNode; + let dragHandler; + modelViewTransformProperty.link( function( modelViewTransform ) { + if ( dragHandler ) { + self.removeInputListener( dragHandler ); + dragHandler.dispose(); + } + dragHandler = new DragListener( { + transform: modelViewTransform, + drag: function( event, listener ) { + let value = listener.modelPoint[ orientation.coordinate ]; + + value = Utils.roundSymmetric( value / areaDisplay.partitionSnapSizeProperty.value ) * + areaDisplay.partitionSnapSizeProperty.value; + value = Utils.clamp( value, 0, activeTotalProperty.value ); + + // Hint arrows disappear when the actual split changes during a drag, see + // https://github.com/phetsims/area-model-common/issues/68 + const currentSplitValue = partitionSplitProperty.value; + if ( value !== currentSplitValue && value !== 0 ) { + showHintArrowsProperty.value = false; + } + + partitionSplitProperty.value = value; + }, + + end: function() { + if ( partitionSplitProperty.value === activeTotalProperty.value ) { + partitionSplitProperty.value = 0; + } + } + } ); + dragHandler.isUserControlledProperty.link( function( controlled ) { + areaDisplay.partitionSplitUserControlledProperties.get( orientation ).value = controlled; + } ); + self.addInputListener( dragHandler ); + } ); +} + +areaModelCommon.register( 'ProportionalPartitionLineNode', ProportionalPartitionLineNode ); + +// Handle arrows +const arrowHalfLength = 10; +const arrowHalfWidth = 10; +const verticalArrowShape = new Shape() + .moveTo( -arrowHalfLength, 0 ) + .lineTo( arrowHalfLength, arrowHalfWidth ) + .lineTo( arrowHalfLength, -arrowHalfWidth ) + .close(); +const horizontalArrowShape = verticalArrowShape.transformed( Matrix3.rotation2( Math.PI / 2 ) ); + +inherit( Node, ProportionalPartitionLineNode, {}, { + HANDLE_ARROW_SHAPES: new OrientationPair( horizontalArrowShape, verticalArrowShape ) } ); + +AccessibleSlider.mixInto( ProportionalPartitionLineNode ); + +export default ProportionalPartitionLineNode; \ No newline at end of file diff --git a/js/proportional/view/SceneRadioButtonGroup.js b/js/proportional/view/SceneRadioButtonGroup.js index d4598a39..49a7fbea 100644 --- a/js/proportional/view/SceneRadioButtonGroup.js +++ b/js/proportional/view/SceneRadioButtonGroup.js @@ -7,59 +7,56 @@ * * @author Jonathan Olson */ -define( require => { - 'use strict'; - // modules - const AlignBox = require( 'SCENERY/nodes/AlignBox' ); - const AlignGroup = require( 'SCENERY/nodes/AlignGroup' ); - const areaModelCommon = require( 'AREA_MODEL_COMMON/areaModelCommon' ); - const AreaModelCommonA11yStrings = require( 'AREA_MODEL_COMMON/AreaModelCommonA11yStrings' ); - const AreaModelCommonConstants = require( 'AREA_MODEL_COMMON/common/AreaModelCommonConstants' ); - const AreaModelCommonRadioButtonGroup = require( 'AREA_MODEL_COMMON/common/view/AreaModelCommonRadioButtonGroup' ); - const inherit = require( 'PHET_CORE/inherit' ); - const StringUtils = require( 'PHETCOMMON/util/StringUtils' ); - const Text = require( 'SCENERY/nodes/Text' ); +import inherit from '../../../../phet-core/js/inherit.js'; +import StringUtils from '../../../../phetcommon/js/util/StringUtils.js'; +import AlignBox from '../../../../scenery/js/nodes/AlignBox.js'; +import AlignGroup from '../../../../scenery/js/nodes/AlignGroup.js'; +import Text from '../../../../scenery/js/nodes/Text.js'; +import areaModelCommon from '../../areaModelCommon.js'; +import AreaModelCommonA11yStrings from '../../AreaModelCommonA11yStrings.js'; +import AreaModelCommonConstants from '../../common/AreaModelCommonConstants.js'; +import AreaModelCommonRadioButtonGroup from '../../common/view/AreaModelCommonRadioButtonGroup.js'; - // a11y strings - const areaGridSizeString = AreaModelCommonA11yStrings.areaGridSize.value; - const sceneSelectionPatternString = AreaModelCommonA11yStrings.sceneSelectionPattern.value; +// a11y strings +const areaGridSizeString = AreaModelCommonA11yStrings.areaGridSize.value; +const sceneSelectionPatternString = AreaModelCommonA11yStrings.sceneSelectionPattern.value; - /** - * @constructor - * @extends {AreaModelCommonRadioButtonGroup} - * - * @param {ProportionalAreaModel} model - * @param {Object} [nodeOptions] - */ - function SceneRadioButtonGroup( model, nodeOptions ) { - const group = new AlignGroup(); // have all the buttons the same size +/** + * @constructor + * @extends {AreaModelCommonRadioButtonGroup} + * + * @param {ProportionalAreaModel} model + * @param {Object} [nodeOptions] + */ +function SceneRadioButtonGroup( model, nodeOptions ) { + const group = new AlignGroup(); // have all the buttons the same size - assert && assert( model.areas.length === 2 || model.areas.length === 3, - 'We only have strings for the 2 or 3 case (right now)' ); + assert && assert( model.areas.length === 2 || model.areas.length === 3, + 'We only have strings for the 2 or 3 case (right now)' ); - AreaModelCommonRadioButtonGroup.call( this, model.currentAreaProperty, model.areas.map( function( area ) { - return { - value: area, - node: new AlignBox( new Text( area.getDimensionString(), { - font: AreaModelCommonConstants.SYMBOL_FONT - } ), { group: group } ), + AreaModelCommonRadioButtonGroup.call( this, model.currentAreaProperty, model.areas.map( function( area ) { + return { + value: area, + node: new AlignBox( new Text( area.getDimensionString(), { + font: AreaModelCommonConstants.SYMBOL_FONT + } ), { group: group } ), - // a11y - labelContent: StringUtils.fillIn( sceneSelectionPatternString, { - width: area.maximumSize, - height: area.maximumSize - } ) - }; - } ), { // a11y - labelContent: areaGridSizeString - } ); - - this.mutate( nodeOptions ); - } - - areaModelCommon.register( 'SceneRadioButtonGroup', SceneRadioButtonGroup ); - - return inherit( AreaModelCommonRadioButtonGroup, SceneRadioButtonGroup ); -} ); + labelContent: StringUtils.fillIn( sceneSelectionPatternString, { + width: area.maximumSize, + height: area.maximumSize + } ) + }; + } ), { + // a11y + labelContent: areaGridSizeString + } ); + + this.mutate( nodeOptions ); +} + +areaModelCommon.register( 'SceneRadioButtonGroup', SceneRadioButtonGroup ); + +inherit( AreaModelCommonRadioButtonGroup, SceneRadioButtonGroup ); +export default SceneRadioButtonGroup; \ No newline at end of file diff --git a/js/proportional/view/TiledAreaNode.js b/js/proportional/view/TiledAreaNode.js index 4f70cbfd..35a41896 100644 --- a/js/proportional/view/TiledAreaNode.js +++ b/js/proportional/view/TiledAreaNode.js @@ -7,264 +7,260 @@ * * @author Jonathan Olson */ -define( require => { - 'use strict'; - - // modules - const areaModelCommon = require( 'AREA_MODEL_COMMON/areaModelCommon' ); - const AreaModelCommonColorProfile = require( 'AREA_MODEL_COMMON/common/view/AreaModelCommonColorProfile' ); - const inherit = require( 'PHET_CORE/inherit' ); - const Node = require( 'SCENERY/nodes/Node' ); - const Orientation = require( 'PHET_CORE/Orientation' ); - const Path = require( 'SCENERY/nodes/Path' ); - const Property = require( 'AXON/Property' ); - const Shape = require( 'KITE/Shape' ); - const Utils = require( 'DOT/Utils' ); - /** - * @constructor - * @extends {Node} - * - * @param {ProportionalAreaDisplay} areaDisplay - * @param {Property.} modelViewTransformProperty - * @param {Property.} tilesVisibleProperty - */ - function TiledAreaNode( areaDisplay, modelViewTransformProperty, tilesVisibleProperty ) { +import Property from '../../../../axon/js/Property.js'; +import Utils from '../../../../dot/js/Utils.js'; +import Shape from '../../../../kite/js/Shape.js'; +import inherit from '../../../../phet-core/js/inherit.js'; +import Orientation from '../../../../phet-core/js/Orientation.js'; +import Node from '../../../../scenery/js/nodes/Node.js'; +import Path from '../../../../scenery/js/nodes/Path.js'; +import areaModelCommon from '../../areaModelCommon.js'; +import AreaModelCommonColorProfile from '../../common/view/AreaModelCommonColorProfile.js'; - const self = this; +/** + * @constructor + * @extends {Node} + * + * @param {ProportionalAreaDisplay} areaDisplay + * @param {Property.} modelViewTransformProperty + * @param {Property.} tilesVisibleProperty + */ +function TiledAreaNode( areaDisplay, modelViewTransformProperty, tilesVisibleProperty ) { - // @private {Property.} - this.areaDisplay = areaDisplay; + const self = this; - // @private {Property.} - this.modelViewTransformProperty = modelViewTransformProperty; + // @private {Property.} + this.areaDisplay = areaDisplay; - // @private {Property.} - this.tilesVisibleProperty = tilesVisibleProperty; + // @private {Property.} + this.modelViewTransformProperty = modelViewTransformProperty; - // @private {Property.} - this.smallTileSizeProperty = areaDisplay.smallTileSizeProperty; - this.largeTileSizeProperty = areaDisplay.largeTileSizeProperty; - this.maximumSizeProperty = areaDisplay.maximumSizeProperty; + // @private {Property.} + this.tilesVisibleProperty = tilesVisibleProperty; - // @private {boolean} - Whether we should be redrawn - this.dirty = true; + // @private {Property.} + this.smallTileSizeProperty = areaDisplay.smallTileSizeProperty; + this.largeTileSizeProperty = areaDisplay.largeTileSizeProperty; + this.maximumSizeProperty = areaDisplay.maximumSizeProperty; - // Things we depend on - function invalidate() { - self.dirty = true; - } + // @private {boolean} - Whether we should be redrawn + this.dirty = true; - tilesVisibleProperty.lazyLink( invalidate ); - modelViewTransformProperty.lazyLink( invalidate ); - this.smallTileSizeProperty.lazyLink( invalidate ); - this.largeTileSizeProperty.lazyLink( invalidate ); - areaDisplay.allPartitionsProperty.link( function( partitions, oldPartitions ) { - oldPartitions && oldPartitions.forEach( function( partition ) { - partition.visibleProperty.unlink( invalidate ); - partition.coordinateRangeProperty.unlink( invalidate ); - } ); - partitions.forEach( function( partition ) { - partition.visibleProperty.link( invalidate ); - partition.coordinateRangeProperty.link( invalidate ); - } ); - invalidate(); - } ); - invalidate(); + // Things we depend on + function invalidate() { + self.dirty = true; + } - // @private {Path} - Background color paths for each section - this.bigPath = new Path( null, { - fill: AreaModelCommonColorProfile.bigTileProperty + tilesVisibleProperty.lazyLink( invalidate ); + modelViewTransformProperty.lazyLink( invalidate ); + this.smallTileSizeProperty.lazyLink( invalidate ); + this.largeTileSizeProperty.lazyLink( invalidate ); + areaDisplay.allPartitionsProperty.link( function( partitions, oldPartitions ) { + oldPartitions && oldPartitions.forEach( function( partition ) { + partition.visibleProperty.unlink( invalidate ); + partition.coordinateRangeProperty.unlink( invalidate ); } ); - this.horizontalPath = new Path( null, { - fill: AreaModelCommonColorProfile.mediumTileProperty - } ); - this.verticalPath = new Path( null, { - fill: AreaModelCommonColorProfile.mediumTileProperty - } ); - this.smallPath = new Path( null, { - fill: AreaModelCommonColorProfile.smallTileProperty + partitions.forEach( function( partition ) { + partition.visibleProperty.link( invalidate ); + partition.coordinateRangeProperty.link( invalidate ); } ); + invalidate(); + } ); + invalidate(); - // @private {Path} - Grid line paths. We'll use clipping to control where they are visible - this.smallGridPath = new Path( null, { - stroke: AreaModelCommonColorProfile.tileBorderProperty - } ); - this.horizontalGridPath = new Path( null, { - stroke: AreaModelCommonColorProfile.tileBorderProperty - } ); - this.verticalGridPath = new Path( null, { - stroke: AreaModelCommonColorProfile.tileBorderProperty + // @private {Path} - Background color paths for each section + this.bigPath = new Path( null, { + fill: AreaModelCommonColorProfile.bigTileProperty + } ); + this.horizontalPath = new Path( null, { + fill: AreaModelCommonColorProfile.mediumTileProperty + } ); + this.verticalPath = new Path( null, { + fill: AreaModelCommonColorProfile.mediumTileProperty + } ); + this.smallPath = new Path( null, { + fill: AreaModelCommonColorProfile.smallTileProperty + } ); + + // @private {Path} - Grid line paths. We'll use clipping to control where they are visible + this.smallGridPath = new Path( null, { + stroke: AreaModelCommonColorProfile.tileBorderProperty + } ); + this.horizontalGridPath = new Path( null, { + stroke: AreaModelCommonColorProfile.tileBorderProperty + } ); + this.verticalGridPath = new Path( null, { + stroke: AreaModelCommonColorProfile.tileBorderProperty + } ); + + Property.multilink( + [ modelViewTransformProperty, this.maximumSizeProperty, this.smallTileSizeProperty ], + function( modelViewTransform, maximumSize, smallTileSize ) { + // Grid line shapes + const smallGridShape = new Shape(); + const horizontalGridShape = new Shape(); + const verticalGridShape = new Shape(); + const maxX = modelViewTransform.modelToViewX( maximumSize ); + const maxY = modelViewTransform.modelToViewY( maximumSize ); + + // We need the grid lines to extend out past each side a bit for correct appearance + for ( let i = -1; i < maximumSize / smallTileSize + 1; i++ ) { + const x = modelViewTransform.modelToViewX( i * smallTileSize ); + const y = modelViewTransform.modelToViewY( i * smallTileSize ); + + smallGridShape.moveTo( x, 0 ).lineTo( x, maxY ); + smallGridShape.moveTo( 0, y ).lineTo( maxX, y ); + + verticalGridShape.moveTo( x, 0 ).lineTo( x, maxY ); + horizontalGridShape.moveTo( 0, y ).lineTo( maxX, y ); + } + + // Made immutable for potential performance gains + self.smallGridPath.shape = smallGridShape.makeImmutable(); + self.horizontalGridPath.shape = horizontalGridShape.makeImmutable(); + self.verticalGridPath.shape = verticalGridShape.makeImmutable(); } ); - Property.multilink( - [ modelViewTransformProperty, this.maximumSizeProperty, this.smallTileSizeProperty ], - function( modelViewTransform, maximumSize, smallTileSize ) { - // Grid line shapes - const smallGridShape = new Shape(); - const horizontalGridShape = new Shape(); - const verticalGridShape = new Shape(); - const maxX = modelViewTransform.modelToViewX( maximumSize ); - const maxY = modelViewTransform.modelToViewY( maximumSize ); - - // We need the grid lines to extend out past each side a bit for correct appearance - for ( let i = -1; i < maximumSize / smallTileSize + 1; i++ ) { - const x = modelViewTransform.modelToViewX( i * smallTileSize ); - const y = modelViewTransform.modelToViewY( i * smallTileSize ); - - smallGridShape.moveTo( x, 0 ).lineTo( x, maxY ); - smallGridShape.moveTo( 0, y ).lineTo( maxX, y ); - - verticalGridShape.moveTo( x, 0 ).lineTo( x, maxY ); - horizontalGridShape.moveTo( 0, y ).lineTo( maxX, y ); - } + // @private {Path} - Contains extra overlay lines to fill in the 'stroked' appearance. + this.extraLinesPath = new Path( null, { + stroke: AreaModelCommonColorProfile.tileBorderProperty + } ); - // Made immutable for potential performance gains - self.smallGridPath.shape = smallGridShape.makeImmutable(); - self.horizontalGridPath.shape = horizontalGridShape.makeImmutable(); - self.verticalGridPath.shape = verticalGridShape.makeImmutable(); - } ); + Node.call( this, { + children: [ + this.bigPath, + this.horizontalPath, + this.verticalPath, + this.smallPath, + this.smallGridPath, + this.horizontalGridPath, + this.verticalGridPath, + this.extraLinesPath + ] + } ); +} - // @private {Path} - Contains extra overlay lines to fill in the 'stroked' appearance. - this.extraLinesPath = new Path( null, { - stroke: AreaModelCommonColorProfile.tileBorderProperty - } ); +areaModelCommon.register( 'TiledAreaNode', TiledAreaNode ); - Node.call( this, { - children: [ - this.bigPath, - this.horizontalPath, - this.verticalPath, - this.smallPath, - this.smallGridPath, - this.horizontalGridPath, - this.verticalGridPath, - this.extraLinesPath - ] +export default inherit( Node, TiledAreaNode, { + /** + * For each partition of a particular orientation, fires the callback with range information already in view + * coordinates. + * @private + * + * @param {Orientation} orientation + * @param {function} callback - callback( largeCount, smallCount, min, border, max ) + */ + forEachPartition: function( orientation, callback ) { + const self = this; + + this.areaDisplay.partitionsProperties.get( orientation ).value.forEach( function( partition ) { + const range = partition.coordinateRangeProperty.value; + + // Ignore partitions without a visible well-defined range. + if ( !partition.visibleProperty.value || range === null ) { + return; + } + + const size = range.getLength(); + const largeCount = Math.floor( Utils.toFixedNumber( size / self.largeTileSizeProperty.value, 3 ) ); + + const smallCount = Utils.roundSymmetric( + ( size - self.largeTileSizeProperty.value * largeCount ) / self.smallTileSizeProperty.value + ); + const min = orientation.modelToView( self.modelViewTransformProperty.value, range.min ); + const border = orientation.modelToView( + self.modelViewTransformProperty.value, range.min + largeCount * self.largeTileSizeProperty.value + ); + const max = orientation.modelToView( self.modelViewTransformProperty.value, range.max ); + + callback( largeCount, smallCount, min, border, max ); } ); - } + }, - areaModelCommon.register( 'TiledAreaNode', TiledAreaNode ); - - return inherit( Node, TiledAreaNode, { - /** - * For each partition of a particular orientation, fires the callback with range information already in view - * coordinates. - * @private - * - * @param {Orientation} orientation - * @param {function} callback - callback( largeCount, smallCount, min, border, max ) - */ - forEachPartition: function( orientation, callback ) { - const self = this; - - this.areaDisplay.partitionsProperties.get( orientation ).value.forEach( function( partition ) { - const range = partition.coordinateRangeProperty.value; - - // Ignore partitions without a visible well-defined range. - if ( !partition.visibleProperty.value || range === null ) { - return; - } + /** + * Updates the view for tiled areas (since it is somewhat expensive to re-draw, and we don't want it being done + * multiple times per frame. + * @private + */ + update: function() { + const self = this; - const size = range.getLength(); - const largeCount = Math.floor( Utils.toFixedNumber( size / self.largeTileSizeProperty.value, 3 ) ); + // Ignore updates if we are not dirty + if ( !this.dirty ) { return; } + this.dirty = false; - const smallCount = Utils.roundSymmetric( - ( size - self.largeTileSizeProperty.value * largeCount ) / self.smallTileSizeProperty.value - ); - const min = orientation.modelToView( self.modelViewTransformProperty.value, range.min ); - const border = orientation.modelToView( - self.modelViewTransformProperty.value, range.min + largeCount * self.largeTileSizeProperty.value - ); - const max = orientation.modelToView( self.modelViewTransformProperty.value, range.max ); + // Coordinate mapping into the view + const modelToViewX = this.modelViewTransformProperty.value.modelToViewX.bind( this.modelViewTransformProperty.value ); + const modelToViewY = this.modelViewTransformProperty.value.modelToViewY.bind( this.modelViewTransformProperty.value ); - callback( largeCount, smallCount, min, border, max ); - } ); - }, - - /** - * Updates the view for tiled areas (since it is somewhat expensive to re-draw, and we don't want it being done - * multiple times per frame. - * @private - */ - update: function() { - const self = this; - - // Ignore updates if we are not dirty - if ( !this.dirty ) { return; } - this.dirty = false; - - // Coordinate mapping into the view - const modelToViewX = this.modelViewTransformProperty.value.modelToViewX.bind( this.modelViewTransformProperty.value ); - const modelToViewY = this.modelViewTransformProperty.value.modelToViewY.bind( this.modelViewTransformProperty.value ); - - const largeTileSize = this.largeTileSizeProperty.value; - const maximumSize = this.maximumSizeProperty.value; - - this.visible = this.tilesVisibleProperty.value; - - const bigShape = new Shape(); - const horizontalShape = new Shape(); - const verticalShape = new Shape(); - const smallShape = new Shape(); - const extraLinesShape = new Shape(); - - this.forEachPartition( Orientation.HORIZONTAL, function( horizontalLargeCount, horizontalSmallCount, xMin, xBorder, xMax ) { - self.forEachPartition( Orientation.VERTICAL, function( verticalLargeCount, verticalSmallCount, yMin, yBorder, yMax ) { - - // Add in extra lines on the far sides of large sections. - let i; - for ( i = 0; i < horizontalLargeCount; i++ ) { - const x = xMin + modelToViewX( ( i + 1 ) * largeTileSize ); - extraLinesShape.moveTo( x, 0 ).lineTo( x, modelToViewY( maximumSize ) ); - } - for ( i = 0; i < verticalLargeCount; i++ ) { - const y = yMin + modelToViewY( ( i + 1 ) * largeTileSize ); - extraLinesShape.moveTo( 0, y ).lineTo( modelToViewX( maximumSize ), y ); - } - - // Add sections to the relevant shapes. - if ( horizontalLargeCount && verticalLargeCount ) { - bigShape.rect( xMin, yMin, xBorder - xMin, yBorder - yMin ); - } - if ( horizontalLargeCount && verticalSmallCount ) { - horizontalShape.rect( xMin, yBorder, xBorder - xMin, yMax - yBorder ); - } - if ( horizontalSmallCount && verticalLargeCount ) { - verticalShape.rect( xBorder, yMin, xMax - xBorder, yBorder - yMin ); - } - if ( horizontalSmallCount && verticalSmallCount ) { - smallShape.rect( xBorder, yBorder, xMax - xBorder, yMax - yBorder ); - } - } ); + const largeTileSize = this.largeTileSizeProperty.value; + const maximumSize = this.maximumSizeProperty.value; + + this.visible = this.tilesVisibleProperty.value; + + const bigShape = new Shape(); + const horizontalShape = new Shape(); + const verticalShape = new Shape(); + const smallShape = new Shape(); + const extraLinesShape = new Shape(); + + this.forEachPartition( Orientation.HORIZONTAL, function( horizontalLargeCount, horizontalSmallCount, xMin, xBorder, xMax ) { + self.forEachPartition( Orientation.VERTICAL, function( verticalLargeCount, verticalSmallCount, yMin, yBorder, yMax ) { + + // Add in extra lines on the far sides of large sections. + let i; + for ( i = 0; i < horizontalLargeCount; i++ ) { + const x = xMin + modelToViewX( ( i + 1 ) * largeTileSize ); + extraLinesShape.moveTo( x, 0 ).lineTo( x, modelToViewY( maximumSize ) ); + } + for ( i = 0; i < verticalLargeCount; i++ ) { + const y = yMin + modelToViewY( ( i + 1 ) * largeTileSize ); + extraLinesShape.moveTo( 0, y ).lineTo( modelToViewX( maximumSize ), y ); + } + + // Add sections to the relevant shapes. + if ( horizontalLargeCount && verticalLargeCount ) { + bigShape.rect( xMin, yMin, xBorder - xMin, yBorder - yMin ); + } + if ( horizontalLargeCount && verticalSmallCount ) { + horizontalShape.rect( xMin, yBorder, xBorder - xMin, yMax - yBorder ); + } + if ( horizontalSmallCount && verticalLargeCount ) { + verticalShape.rect( xBorder, yMin, xMax - xBorder, yBorder - yMin ); + } + if ( horizontalSmallCount && verticalSmallCount ) { + smallShape.rect( xBorder, yBorder, xMax - xBorder, yMax - yBorder ); + } } ); + } ); - // Make the shapes immutable so that listeners don't have to be added - bigShape.makeImmutable(); - horizontalShape.makeImmutable(); - verticalShape.makeImmutable(); - smallShape.makeImmutable(); - extraLinesShape.makeImmutable(); - - // Adjust the backgrounds to fit their respective areas - this.bigPath.shape = bigShape; - this.horizontalPath.shape = horizontalShape; - this.verticalPath.shape = verticalShape; - this.smallPath.shape = smallShape; - - // Selectively display grid lines as a "stroke" over the background - this.smallGridPath.clipArea = smallShape; - this.horizontalGridPath.clipArea = horizontalShape; - this.verticalGridPath.clipArea = verticalShape; - - // Display extra lines, and clip it to fit the active area. - this.extraLinesPath.shape = extraLinesShape; - this.extraLinesPath.clipArea = Shape.rect( - 0, - 0, - modelToViewX( this.areaDisplay.activeTotalProperties.horizontal.value ), - modelToViewY( this.areaDisplay.activeTotalProperties.vertical.value ) - ); - } - } ); -} ); + // Make the shapes immutable so that listeners don't have to be added + bigShape.makeImmutable(); + horizontalShape.makeImmutable(); + verticalShape.makeImmutable(); + smallShape.makeImmutable(); + extraLinesShape.makeImmutable(); + + // Adjust the backgrounds to fit their respective areas + this.bigPath.shape = bigShape; + this.horizontalPath.shape = horizontalShape; + this.verticalPath.shape = verticalShape; + this.smallPath.shape = smallShape; + + // Selectively display grid lines as a "stroke" over the background + this.smallGridPath.clipArea = smallShape; + this.horizontalGridPath.clipArea = horizontalShape; + this.verticalGridPath.clipArea = verticalShape; + + // Display extra lines, and clip it to fit the active area. + this.extraLinesPath.shape = extraLinesShape; + this.extraLinesPath.clipArea = Shape.rect( + 0, + 0, + modelToViewX( this.areaDisplay.activeTotalProperties.horizontal.value ), + modelToViewY( this.areaDisplay.activeTotalProperties.vertical.value ) + ); + } +} ); \ No newline at end of file diff --git a/js/screens/DecimalsScreen.js b/js/screens/DecimalsScreen.js index 2fcae7ee..281e0812 100644 --- a/js/screens/DecimalsScreen.js +++ b/js/screens/DecimalsScreen.js @@ -5,79 +5,76 @@ * * @author Jonathan Olson */ -define( require => { - 'use strict'; - // modules - const areaModelCommon = require( 'AREA_MODEL_COMMON/areaModelCommon' ); - const AreaModelCommonColorProfile = require( 'AREA_MODEL_COMMON/common/view/AreaModelCommonColorProfile' ); - const inherit = require( 'PHET_CORE/inherit' ); - const merge = require( 'PHET_CORE/merge' ); - const ProportionalAreaModel = require( 'AREA_MODEL_COMMON/proportional/model/ProportionalAreaModel' ); - const ProportionalAreaScreenView = require( 'AREA_MODEL_COMMON/proportional/view/ProportionalAreaScreenView' ); - const Screen = require( 'JOIST/Screen' ); +import Screen from '../../../joist/js/Screen.js'; +import inherit from '../../../phet-core/js/inherit.js'; +import merge from '../../../phet-core/js/merge.js'; +import areaModelCommonStrings from '../area-model-common-strings.js'; +import areaModelCommon from '../areaModelCommon.js'; +import AreaModelCommonColorProfile from '../common/view/AreaModelCommonColorProfile.js'; +import ProportionalAreaModel from '../proportional/model/ProportionalAreaModel.js'; +import ProportionalAreaScreenView from '../proportional/view/ProportionalAreaScreenView.js'; - // strings - const screenDecimalsString = require( 'string!AREA_MODEL_COMMON/screen.decimals' ); +const screenDecimalsString = areaModelCommonStrings.screen.decimals; - /** - * @constructor - */ - function DecimalsScreen() { +/** + * @constructor + */ +function DecimalsScreen() { - const options = { - name: screenDecimalsString, - backgroundColorProperty: AreaModelCommonColorProfile.backgroundProperty - }; + const options = { + name: screenDecimalsString, + backgroundColorProperty: AreaModelCommonColorProfile.backgroundProperty + }; - const commonAreaOptions = { - eraseWidth: 0.1, - eraseHeight: 0.1, - snapSize: 0.1, - gridSpacing: 0.1, - smallTileSize: 0.1, - largeTileSize: 1 - }; + const commonAreaOptions = { + eraseWidth: 0.1, + eraseHeight: 0.1, + snapSize: 0.1, + gridSpacing: 0.1, + smallTileSize: 0.1, + largeTileSize: 1 + }; - Screen.call( this, - function() { - return new ProportionalAreaModel( [ - merge( { - maximumSize: 1, - minimumSize: 0.1, - initialWidth: 0.5, - initialHeight: 0.5, - initialVerticalSplit: 0.2, - partitionSnapSize: 0.1 - }, commonAreaOptions ), - merge( { - maximumSize: 2, - minimumSize: 0.1, - initialWidth: 1, - initialHeight: 1, - initialVerticalSplit: 0.5, - partitionSnapSize: 0.1 - }, commonAreaOptions ), - merge( { - maximumSize: 3, - minimumSize: 0.1, - initialWidth: 1, - initialHeight: 1, - initialVerticalSplit: 0.5, - partitionSnapSize: 0.1 - }, commonAreaOptions ) - ] ); - }, - function( model ) { - return new ProportionalAreaScreenView( model, { - decimalPlaces: 1 - } ); - }, - options - ); - } + Screen.call( this, + function() { + return new ProportionalAreaModel( [ + merge( { + maximumSize: 1, + minimumSize: 0.1, + initialWidth: 0.5, + initialHeight: 0.5, + initialVerticalSplit: 0.2, + partitionSnapSize: 0.1 + }, commonAreaOptions ), + merge( { + maximumSize: 2, + minimumSize: 0.1, + initialWidth: 1, + initialHeight: 1, + initialVerticalSplit: 0.5, + partitionSnapSize: 0.1 + }, commonAreaOptions ), + merge( { + maximumSize: 3, + minimumSize: 0.1, + initialWidth: 1, + initialHeight: 1, + initialVerticalSplit: 0.5, + partitionSnapSize: 0.1 + }, commonAreaOptions ) + ] ); + }, + function( model ) { + return new ProportionalAreaScreenView( model, { + decimalPlaces: 1 + } ); + }, + options + ); +} - areaModelCommon.register( 'DecimalsScreen', DecimalsScreen ); +areaModelCommon.register( 'DecimalsScreen', DecimalsScreen ); - return inherit( Screen, DecimalsScreen ); -} ); +inherit( Screen, DecimalsScreen ); +export default DecimalsScreen; \ No newline at end of file diff --git a/js/screens/ExploreScreen.js b/js/screens/ExploreScreen.js index d0413e16..8028415d 100644 --- a/js/screens/ExploreScreen.js +++ b/js/screens/ExploreScreen.js @@ -5,72 +5,67 @@ * * @author Jonathan Olson */ -define( require => { - 'use strict'; - // modules - const areaModelCommon = require( 'AREA_MODEL_COMMON/areaModelCommon' ); - const AreaModelCommonColorProfile = require( 'AREA_MODEL_COMMON/common/view/AreaModelCommonColorProfile' ); - const Image = require( 'SCENERY/nodes/Image' ); - const inherit = require( 'PHET_CORE/inherit' ); - const ProportionalAreaModel = require( 'AREA_MODEL_COMMON/proportional/model/ProportionalAreaModel' ); - const ProportionalAreaScreenView = require( 'AREA_MODEL_COMMON/proportional/view/ProportionalAreaScreenView' ); - const Screen = require( 'JOIST/Screen' ); +import Screen from '../../../joist/js/Screen.js'; +import inherit from '../../../phet-core/js/inherit.js'; +import Image from '../../../scenery/js/nodes/Image.js'; +import exploreScreenIconImage from '../../mipmaps/explore-screen-icon_png.js'; +import exploreScreenNavbarImage from '../../mipmaps/explore-screen-navbar_png.js'; +import areaModelCommonStrings from '../area-model-common-strings.js'; +import areaModelCommon from '../areaModelCommon.js'; +import AreaModelCommonColorProfile from '../common/view/AreaModelCommonColorProfile.js'; +import ProportionalAreaModel from '../proportional/model/ProportionalAreaModel.js'; +import ProportionalAreaScreenView from '../proportional/view/ProportionalAreaScreenView.js'; - // images - const exploreScreenIconImage = require( 'mipmap!AREA_MODEL_COMMON/explore-screen-icon.png' ); - const exploreScreenNavbarImage = require( 'mipmap!AREA_MODEL_COMMON/explore-screen-navbar.png' ); +const screenExploreString = areaModelCommonStrings.screen.explore; - // strings - const screenExploreString = require( 'string!AREA_MODEL_COMMON/screen.explore' ); - - /** - * @constructor - */ - function ExploreScreen() { +/** + * @constructor + */ +function ExploreScreen() { - const options = { - name: screenExploreString, - backgroundColorProperty: AreaModelCommonColorProfile.backgroundProperty, - homeScreenIcon: new Image( exploreScreenIconImage ), - navigationBarIcon: new Image( exploreScreenNavbarImage ) - }; + const options = { + name: screenExploreString, + backgroundColorProperty: AreaModelCommonColorProfile.backgroundProperty, + homeScreenIcon: new Image( exploreScreenIconImage ), + navigationBarIcon: new Image( exploreScreenNavbarImage ) + }; - Screen.call( this, - function() { - return new ProportionalAreaModel( [ - { - maximumSize: 20, - minimumSize: 1, - initialWidth: 10, - initialHeight: 10, - initialVerticalSplit: 5, - snapSize: 1, - partitionSnapSize: 1, - gridSpacing: 1, - smallTileSize: 1, - largeTileSize: 10 - }, - { - maximumSize: 100, - minimumSize: 1, - initialWidth: 50, - initialHeight: 50, - eraseWidth: 10, - eraseHeight: 10, - initialVerticalSplit: 30, - snapSize: 1, - gridSpacing: 10, - tilesAvailable: false - } - ] ); - }, - function( model ) { return new ProportionalAreaScreenView( model ); }, - options - ); - } + Screen.call( this, + function() { + return new ProportionalAreaModel( [ + { + maximumSize: 20, + minimumSize: 1, + initialWidth: 10, + initialHeight: 10, + initialVerticalSplit: 5, + snapSize: 1, + partitionSnapSize: 1, + gridSpacing: 1, + smallTileSize: 1, + largeTileSize: 10 + }, + { + maximumSize: 100, + minimumSize: 1, + initialWidth: 50, + initialHeight: 50, + eraseWidth: 10, + eraseHeight: 10, + initialVerticalSplit: 30, + snapSize: 1, + gridSpacing: 10, + tilesAvailable: false + } + ] ); + }, + function( model ) { return new ProportionalAreaScreenView( model ); }, + options + ); +} - areaModelCommon.register( 'ExploreScreen', ExploreScreen ); +areaModelCommon.register( 'ExploreScreen', ExploreScreen ); - return inherit( Screen, ExploreScreen ); -} ); +inherit( Screen, ExploreScreen ); +export default ExploreScreen; \ No newline at end of file diff --git a/js/screens/GenericGameScreen.js b/js/screens/GenericGameScreen.js index 2507e65e..2df4bfa5 100644 --- a/js/screens/GenericGameScreen.js +++ b/js/screens/GenericGameScreen.js @@ -5,45 +5,40 @@ * * @author Jonathan Olson */ -define( require => { - 'use strict'; - - // modules - const areaModelCommon = require( 'AREA_MODEL_COMMON/areaModelCommon' ); - const AreaModelCommonColorProfile = require( 'AREA_MODEL_COMMON/common/view/AreaModelCommonColorProfile' ); - const GameAreaScreenView = require( 'AREA_MODEL_COMMON/game/view/GameAreaScreenView' ); - const GenericGameAreaModel = require( 'AREA_MODEL_COMMON/game/model/GenericGameAreaModel' ); - const Image = require( 'SCENERY/nodes/Image' ); - const inherit = require( 'PHET_CORE/inherit' ); - const Screen = require( 'JOIST/Screen' ); - - // images - const genericGameScreenIconImage = require( 'mipmap!AREA_MODEL_COMMON/generic-game-screen-icon.png' ); - const genericGameScreenNavbarImage = require( 'mipmap!AREA_MODEL_COMMON/generic-game-screen-navbar.png' ); - - // strings - const screenGameString = require( 'string!AREA_MODEL_COMMON/screen.game' ); - - /** - * @constructor - */ - function GenericGameScreen() { - - const options = { - name: screenGameString, - backgroundColorProperty: AreaModelCommonColorProfile.backgroundProperty, - homeScreenIcon: new Image( genericGameScreenIconImage ), - navigationBarIcon: new Image( genericGameScreenNavbarImage ) - }; - - Screen.call( this, - function() { return new GenericGameAreaModel(); }, - function( model ) { return new GameAreaScreenView( model ); }, - options - ); - } - - areaModelCommon.register( 'GenericGameScreen', GenericGameScreen ); - - return inherit( Screen, GenericGameScreen ); -} ); + +import Screen from '../../../joist/js/Screen.js'; +import inherit from '../../../phet-core/js/inherit.js'; +import Image from '../../../scenery/js/nodes/Image.js'; +import genericGameScreenIconImage from '../../mipmaps/generic-game-screen-icon_png.js'; +import genericGameScreenNavbarImage from '../../mipmaps/generic-game-screen-navbar_png.js'; +import areaModelCommonStrings from '../area-model-common-strings.js'; +import areaModelCommon from '../areaModelCommon.js'; +import AreaModelCommonColorProfile from '../common/view/AreaModelCommonColorProfile.js'; +import GenericGameAreaModel from '../game/model/GenericGameAreaModel.js'; +import GameAreaScreenView from '../game/view/GameAreaScreenView.js'; + +const screenGameString = areaModelCommonStrings.screen.game; + +/** + * @constructor + */ +function GenericGameScreen() { + + const options = { + name: screenGameString, + backgroundColorProperty: AreaModelCommonColorProfile.backgroundProperty, + homeScreenIcon: new Image( genericGameScreenIconImage ), + navigationBarIcon: new Image( genericGameScreenNavbarImage ) + }; + + Screen.call( this, + function() { return new GenericGameAreaModel(); }, + function( model ) { return new GameAreaScreenView( model ); }, + options + ); +} + +areaModelCommon.register( 'GenericGameScreen', GenericGameScreen ); + +inherit( Screen, GenericGameScreen ); +export default GenericGameScreen; \ No newline at end of file diff --git a/js/screens/GenericScreen.js b/js/screens/GenericScreen.js index afda4480..b173b802 100644 --- a/js/screens/GenericScreen.js +++ b/js/screens/GenericScreen.js @@ -5,47 +5,42 @@ * * @author Jonathan Olson */ -define( require => { - 'use strict'; - - // modules - const areaModelCommon = require( 'AREA_MODEL_COMMON/areaModelCommon' ); - const AreaModelCommonColorProfile = require( 'AREA_MODEL_COMMON/common/view/AreaModelCommonColorProfile' ); - const GenericAreaModel = require( 'AREA_MODEL_COMMON/generic/model/GenericAreaModel' ); - const GenericAreaScreenView = require( 'AREA_MODEL_COMMON/generic/view/GenericAreaScreenView' ); - const Image = require( 'SCENERY/nodes/Image' ); - const inherit = require( 'PHET_CORE/inherit' ); - const Screen = require( 'JOIST/Screen' ); - - // images - const genericScreenIconImage = require( 'mipmap!AREA_MODEL_COMMON/generic-screen-icon.png' ); - - // strings - const screenGenericString = require( 'string!AREA_MODEL_COMMON/screen.generic' ); - - /** - * @constructor - */ - function GenericScreen() { - - const options = { - name: screenGenericString, - backgroundColorProperty: AreaModelCommonColorProfile.backgroundProperty, - homeScreenIcon: new Image( genericScreenIconImage ) - }; - - Screen.call( this, - function() { - return new GenericAreaModel( { - allowExponents: false - } ); - }, - function( model ) { return new GenericAreaScreenView( model, 0 ); }, - options - ); - } - - areaModelCommon.register( 'GenericScreen', GenericScreen ); - - return inherit( Screen, GenericScreen ); -} ); + +import Screen from '../../../joist/js/Screen.js'; +import inherit from '../../../phet-core/js/inherit.js'; +import Image from '../../../scenery/js/nodes/Image.js'; +import genericScreenIconImage from '../../mipmaps/generic-screen-icon_png.js'; +import areaModelCommonStrings from '../area-model-common-strings.js'; +import areaModelCommon from '../areaModelCommon.js'; +import AreaModelCommonColorProfile from '../common/view/AreaModelCommonColorProfile.js'; +import GenericAreaModel from '../generic/model/GenericAreaModel.js'; +import GenericAreaScreenView from '../generic/view/GenericAreaScreenView.js'; + +const screenGenericString = areaModelCommonStrings.screen.generic; + +/** + * @constructor + */ +function GenericScreen() { + + const options = { + name: screenGenericString, + backgroundColorProperty: AreaModelCommonColorProfile.backgroundProperty, + homeScreenIcon: new Image( genericScreenIconImage ) + }; + + Screen.call( this, + function() { + return new GenericAreaModel( { + allowExponents: false + } ); + }, + function( model ) { return new GenericAreaScreenView( model, 0 ); }, + options + ); +} + +areaModelCommon.register( 'GenericScreen', GenericScreen ); + +inherit( Screen, GenericScreen ); +export default GenericScreen; \ No newline at end of file diff --git a/js/screens/MultiplyScreen.js b/js/screens/MultiplyScreen.js index 9651f2b4..97f1c5f0 100644 --- a/js/screens/MultiplyScreen.js +++ b/js/screens/MultiplyScreen.js @@ -5,82 +5,77 @@ * * @author Jonathan Olson */ -define( require => { - 'use strict'; - // modules - const areaModelCommon = require( 'AREA_MODEL_COMMON/areaModelCommon' ); - const AreaModelCommonA11yStrings = require( 'AREA_MODEL_COMMON/AreaModelCommonA11yStrings' ); - const AreaModelCommonColorProfile = require( 'AREA_MODEL_COMMON/common/view/AreaModelCommonColorProfile' ); - const Image = require( 'SCENERY/nodes/Image' ); - const inherit = require( 'PHET_CORE/inherit' ); - const merge = require( 'PHET_CORE/merge' ); - const PartialProductsChoice = require( 'AREA_MODEL_COMMON/common/model/PartialProductsChoice' ); - const PartitionLineChoice = require( 'AREA_MODEL_COMMON/proportional/model/PartitionLineChoice' ); - const ProportionalAreaModel = require( 'AREA_MODEL_COMMON/proportional/model/ProportionalAreaModel' ); - const ProportionalAreaScreenView = require( 'AREA_MODEL_COMMON/proportional/view/ProportionalAreaScreenView' ); - const Screen = require( 'JOIST/Screen' ); +import Screen from '../../../joist/js/Screen.js'; +import inherit from '../../../phet-core/js/inherit.js'; +import merge from '../../../phet-core/js/merge.js'; +import Image from '../../../scenery/js/nodes/Image.js'; +import multiplyScreenIconImage from '../../mipmaps/multiply-screen-icon_png.js'; +import multiplyScreenNavbarImage from '../../mipmaps/multiply-screen-navbar_png.js'; +import areaModelCommonStrings from '../area-model-common-strings.js'; +import areaModelCommon from '../areaModelCommon.js'; +import AreaModelCommonA11yStrings from '../AreaModelCommonA11yStrings.js'; +import PartialProductsChoice from '../common/model/PartialProductsChoice.js'; +import AreaModelCommonColorProfile from '../common/view/AreaModelCommonColorProfile.js'; +import PartitionLineChoice from '../proportional/model/PartitionLineChoice.js'; +import ProportionalAreaModel from '../proportional/model/ProportionalAreaModel.js'; +import ProportionalAreaScreenView from '../proportional/view/ProportionalAreaScreenView.js'; - // images - const multiplyScreenIconImage = require( 'mipmap!AREA_MODEL_COMMON/multiply-screen-icon.png' ); - const multiplyScreenNavbarImage = require( 'mipmap!AREA_MODEL_COMMON/multiply-screen-navbar.png' ); +const screenMultiplyString = areaModelCommonStrings.screen.multiply; - // strings - const screenMultiplyString = require( 'string!AREA_MODEL_COMMON/screen.multiply' ); +// a11y strings +const multiplyDescriptionString = AreaModelCommonA11yStrings.multiplyDescription.value; - // a11y strings - const multiplyDescriptionString = AreaModelCommonA11yStrings.multiplyDescription.value; - - /** - * @constructor - */ - function MultiplyScreen() { +/** + * @constructor + */ +function MultiplyScreen() { - const options = { - name: screenMultiplyString, - backgroundColorProperty: AreaModelCommonColorProfile.backgroundProperty, - homeScreenIcon: new Image( multiplyScreenIconImage ), - navigationBarIcon: new Image( multiplyScreenNavbarImage ), + const options = { + name: screenMultiplyString, + backgroundColorProperty: AreaModelCommonColorProfile.backgroundProperty, + homeScreenIcon: new Image( multiplyScreenIconImage ), + navigationBarIcon: new Image( multiplyScreenNavbarImage ), - // a11y - descriptionContent: multiplyDescriptionString - }; + // a11y + descriptionContent: multiplyDescriptionString + }; - const commonAreaOptions = { - minimumSize: 1, - initialWidth: 1, - initialHeight: 1, - snapSize: 1, - gridSpacing: 1, - partitionLineChoice: PartitionLineChoice.NONE, - tilesAvailable: false, - productsAvailable: false, - countingAvailable: true - }; + const commonAreaOptions = { + minimumSize: 1, + initialWidth: 1, + initialHeight: 1, + snapSize: 1, + gridSpacing: 1, + partitionLineChoice: PartitionLineChoice.NONE, + tilesAvailable: false, + productsAvailable: false, + countingAvailable: true + }; - Screen.call( this, - function() { - return new ProportionalAreaModel( [ - merge( { maximumSize: 10 }, commonAreaOptions ), - merge( { maximumSize: 12 }, commonAreaOptions ) - ], { - initialPartialProductsChoice: PartialProductsChoice.HIDDEN - } ); - }, - function( model ) { - return new ProportionalAreaScreenView( model, { - showProductsSelection: false, - showCalculationSelection: false, - useTileLikeBackground: true, - useSimplifiedNames: true, - useLargeArea: true - } ); - }, - options - ); - } + Screen.call( this, + function() { + return new ProportionalAreaModel( [ + merge( { maximumSize: 10 }, commonAreaOptions ), + merge( { maximumSize: 12 }, commonAreaOptions ) + ], { + initialPartialProductsChoice: PartialProductsChoice.HIDDEN + } ); + }, + function( model ) { + return new ProportionalAreaScreenView( model, { + showProductsSelection: false, + showCalculationSelection: false, + useTileLikeBackground: true, + useSimplifiedNames: true, + useLargeArea: true + } ); + }, + options + ); +} - areaModelCommon.register( 'MultiplyScreen', MultiplyScreen ); +areaModelCommon.register( 'MultiplyScreen', MultiplyScreen ); - return inherit( Screen, MultiplyScreen ); -} ); +inherit( Screen, MultiplyScreen ); +export default MultiplyScreen; \ No newline at end of file diff --git a/js/screens/PartitionScreen.js b/js/screens/PartitionScreen.js index d0b34f93..11dea72d 100644 --- a/js/screens/PartitionScreen.js +++ b/js/screens/PartitionScreen.js @@ -5,83 +5,78 @@ * * @author Jonathan Olson */ -define( require => { - 'use strict'; - // modules - const AreaCalculationChoice = require( 'AREA_MODEL_COMMON/common/model/AreaCalculationChoice' ); - const areaModelCommon = require( 'AREA_MODEL_COMMON/areaModelCommon' ); - const AreaModelCommonA11yStrings = require( 'AREA_MODEL_COMMON/AreaModelCommonA11yStrings' ); - const AreaModelCommonColorProfile = require( 'AREA_MODEL_COMMON/common/view/AreaModelCommonColorProfile' ); - const Image = require( 'SCENERY/nodes/Image' ); - const inherit = require( 'PHET_CORE/inherit' ); - const merge = require( 'PHET_CORE/merge' ); - const PartitionLineChoice = require( 'AREA_MODEL_COMMON/proportional/model/PartitionLineChoice' ); - const ProportionalAreaModel = require( 'AREA_MODEL_COMMON/proportional/model/ProportionalAreaModel' ); - const ProportionalAreaScreenView = require( 'AREA_MODEL_COMMON/proportional/view/ProportionalAreaScreenView' ); - const Screen = require( 'JOIST/Screen' ); +import Screen from '../../../joist/js/Screen.js'; +import inherit from '../../../phet-core/js/inherit.js'; +import merge from '../../../phet-core/js/merge.js'; +import Image from '../../../scenery/js/nodes/Image.js'; +import partitionScreenIconImage from '../../mipmaps/partition-screen-icon_png.js'; +import partitionScreenNavbarImage from '../../mipmaps/partition-screen-navbar_png.js'; +import areaModelCommonStrings from '../area-model-common-strings.js'; +import areaModelCommon from '../areaModelCommon.js'; +import AreaModelCommonA11yStrings from '../AreaModelCommonA11yStrings.js'; +import AreaCalculationChoice from '../common/model/AreaCalculationChoice.js'; +import AreaModelCommonColorProfile from '../common/view/AreaModelCommonColorProfile.js'; +import PartitionLineChoice from '../proportional/model/PartitionLineChoice.js'; +import ProportionalAreaModel from '../proportional/model/ProportionalAreaModel.js'; +import ProportionalAreaScreenView from '../proportional/view/ProportionalAreaScreenView.js'; - // images - const partitionScreenIconImage = require( 'mipmap!AREA_MODEL_COMMON/partition-screen-icon.png' ); - const partitionScreenNavbarImage = require( 'mipmap!AREA_MODEL_COMMON/partition-screen-navbar.png' ); +const screenPartitionString = areaModelCommonStrings.screen.partition; - // strings - const screenPartitionString = require( 'string!AREA_MODEL_COMMON/screen.partition' ); +// a11y strings +const partitionDescriptionString = AreaModelCommonA11yStrings.partitionDescription.value; - // a11y strings - const partitionDescriptionString = AreaModelCommonA11yStrings.partitionDescription.value; - - /** - * @constructor - */ - function PartitionScreen() { +/** + * @constructor + */ +function PartitionScreen() { - const options = { - name: screenPartitionString, - backgroundColorProperty: AreaModelCommonColorProfile.backgroundProperty, - homeScreenIcon: new Image( partitionScreenIconImage ), - navigationBarIcon: new Image( partitionScreenNavbarImage ), + const options = { + name: screenPartitionString, + backgroundColorProperty: AreaModelCommonColorProfile.backgroundProperty, + homeScreenIcon: new Image( partitionScreenIconImage ), + navigationBarIcon: new Image( partitionScreenNavbarImage ), - // a11y - descriptionContent: partitionDescriptionString - }; + // a11y + descriptionContent: partitionDescriptionString + }; - const commonAreaOptions = { - minimumSize: 1, - initialWidth: 5, - initialHeight: 5, - initialVerticalSplit: 2, - initialHorizontalSplit: 2, - partitionLineChoice: PartitionLineChoice.ONE, - snapSize: 1, - gridSpacing: 1, - partitionSnapSize: 1, - tilesAvailable: false, - productsAvailable: false - }; + const commonAreaOptions = { + minimumSize: 1, + initialWidth: 5, + initialHeight: 5, + initialVerticalSplit: 2, + initialHorizontalSplit: 2, + partitionLineChoice: PartitionLineChoice.ONE, + snapSize: 1, + gridSpacing: 1, + partitionSnapSize: 1, + tilesAvailable: false, + productsAvailable: false + }; - Screen.call( this, - function() { - return new ProportionalAreaModel( [ - merge( { maximumSize: 10 }, commonAreaOptions ), - merge( { maximumSize: 12 }, commonAreaOptions ) - ], { - initialAreaCalculationChoice: AreaCalculationChoice.SHOW_ALL_LINES - } ); - }, - function( model ) { - return new ProportionalAreaScreenView( model, { - showCalculationSelection: false, - useTileLikeBackground: true, - useSimplifiedNames: true, - useCalculationBox: true - } ); - }, - options - ); - } + Screen.call( this, + function() { + return new ProportionalAreaModel( [ + merge( { maximumSize: 10 }, commonAreaOptions ), + merge( { maximumSize: 12 }, commonAreaOptions ) + ], { + initialAreaCalculationChoice: AreaCalculationChoice.SHOW_ALL_LINES + } ); + }, + function( model ) { + return new ProportionalAreaScreenView( model, { + showCalculationSelection: false, + useTileLikeBackground: true, + useSimplifiedNames: true, + useCalculationBox: true + } ); + }, + options + ); +} - areaModelCommon.register( 'PartitionScreen', PartitionScreen ); +areaModelCommon.register( 'PartitionScreen', PartitionScreen ); - return inherit( Screen, PartitionScreen ); -} ); +inherit( Screen, PartitionScreen ); +export default PartitionScreen; \ No newline at end of file diff --git a/js/screens/VariablesGameScreen.js b/js/screens/VariablesGameScreen.js index cb1b07e9..6b2f9055 100644 --- a/js/screens/VariablesGameScreen.js +++ b/js/screens/VariablesGameScreen.js @@ -5,45 +5,40 @@ * * @author Jonathan Olson */ -define( require => { - 'use strict'; - - // modules - const areaModelCommon = require( 'AREA_MODEL_COMMON/areaModelCommon' ); - const AreaModelCommonColorProfile = require( 'AREA_MODEL_COMMON/common/view/AreaModelCommonColorProfile' ); - const GameAreaScreenView = require( 'AREA_MODEL_COMMON/game/view/GameAreaScreenView' ); - const Image = require( 'SCENERY/nodes/Image' ); - const inherit = require( 'PHET_CORE/inherit' ); - const Screen = require( 'JOIST/Screen' ); - const VariablesGameAreaModel = require( 'AREA_MODEL_COMMON/game/model/VariablesGameAreaModel' ); - - // images - const variablesGameScreenIconImage = require( 'mipmap!AREA_MODEL_COMMON/variables-game-screen-icon.png' ); - const variablesGameScreenNavbarImage = require( 'mipmap!AREA_MODEL_COMMON/variables-game-screen-navbar.png' ); - - // strings - const screenGameString = require( 'string!AREA_MODEL_COMMON/screen.game' ); - - /** - * @constructor - */ - function VariablesGameScreen() { - - const options = { - name: screenGameString, - backgroundColorProperty: AreaModelCommonColorProfile.backgroundProperty, - homeScreenIcon: new Image( variablesGameScreenIconImage ), - navigationBarIcon: new Image( variablesGameScreenNavbarImage ) - }; - - Screen.call( this, - function() { return new VariablesGameAreaModel(); }, - function( model ) { return new GameAreaScreenView( model ); }, - options - ); - } - - areaModelCommon.register( 'VariablesGameScreen', VariablesGameScreen ); - - return inherit( Screen, VariablesGameScreen ); -} ); + +import Screen from '../../../joist/js/Screen.js'; +import inherit from '../../../phet-core/js/inherit.js'; +import Image from '../../../scenery/js/nodes/Image.js'; +import variablesGameScreenIconImage from '../../mipmaps/variables-game-screen-icon_png.js'; +import variablesGameScreenNavbarImage from '../../mipmaps/variables-game-screen-navbar_png.js'; +import areaModelCommonStrings from '../area-model-common-strings.js'; +import areaModelCommon from '../areaModelCommon.js'; +import AreaModelCommonColorProfile from '../common/view/AreaModelCommonColorProfile.js'; +import VariablesGameAreaModel from '../game/model/VariablesGameAreaModel.js'; +import GameAreaScreenView from '../game/view/GameAreaScreenView.js'; + +const screenGameString = areaModelCommonStrings.screen.game; + +/** + * @constructor + */ +function VariablesGameScreen() { + + const options = { + name: screenGameString, + backgroundColorProperty: AreaModelCommonColorProfile.backgroundProperty, + homeScreenIcon: new Image( variablesGameScreenIconImage ), + navigationBarIcon: new Image( variablesGameScreenNavbarImage ) + }; + + Screen.call( this, + function() { return new VariablesGameAreaModel(); }, + function( model ) { return new GameAreaScreenView( model ); }, + options + ); +} + +areaModelCommon.register( 'VariablesGameScreen', VariablesGameScreen ); + +inherit( Screen, VariablesGameScreen ); +export default VariablesGameScreen; \ No newline at end of file diff --git a/js/screens/VariablesScreen.js b/js/screens/VariablesScreen.js index 36ccb024..f4ec5c23 100644 --- a/js/screens/VariablesScreen.js +++ b/js/screens/VariablesScreen.js @@ -5,47 +5,42 @@ * * @author Jonathan Olson */ -define( require => { - 'use strict'; - - // modules - const areaModelCommon = require( 'AREA_MODEL_COMMON/areaModelCommon' ); - const AreaModelCommonColorProfile = require( 'AREA_MODEL_COMMON/common/view/AreaModelCommonColorProfile' ); - const GenericAreaModel = require( 'AREA_MODEL_COMMON/generic/model/GenericAreaModel' ); - const GenericAreaScreenView = require( 'AREA_MODEL_COMMON/generic/view/GenericAreaScreenView' ); - const Image = require( 'SCENERY/nodes/Image' ); - const inherit = require( 'PHET_CORE/inherit' ); - const Screen = require( 'JOIST/Screen' ); - - // images - const variablesScreenIconImage = require( 'mipmap!AREA_MODEL_COMMON/variables-screen-icon.png' ); - - // strings - const screenVariablesString = require( 'string!AREA_MODEL_COMMON/screen.variables' ); - - /** - * @constructor - */ - function VariablesScreen() { - - const options = { - name: screenVariablesString, - backgroundColorProperty: AreaModelCommonColorProfile.backgroundProperty, - homeScreenIcon: new Image( variablesScreenIconImage ) - }; - - Screen.call( this, - function() { - return new GenericAreaModel( { - allowExponents: true - } ); - }, - function( model ) { return new GenericAreaScreenView( model, 0 ); }, - options - ); - } - - areaModelCommon.register( 'VariablesScreen', VariablesScreen ); - - return inherit( Screen, VariablesScreen ); -} ); + +import Screen from '../../../joist/js/Screen.js'; +import inherit from '../../../phet-core/js/inherit.js'; +import Image from '../../../scenery/js/nodes/Image.js'; +import variablesScreenIconImage from '../../mipmaps/variables-screen-icon_png.js'; +import areaModelCommonStrings from '../area-model-common-strings.js'; +import areaModelCommon from '../areaModelCommon.js'; +import AreaModelCommonColorProfile from '../common/view/AreaModelCommonColorProfile.js'; +import GenericAreaModel from '../generic/model/GenericAreaModel.js'; +import GenericAreaScreenView from '../generic/view/GenericAreaScreenView.js'; + +const screenVariablesString = areaModelCommonStrings.screen.variables; + +/** + * @constructor + */ +function VariablesScreen() { + + const options = { + name: screenVariablesString, + backgroundColorProperty: AreaModelCommonColorProfile.backgroundProperty, + homeScreenIcon: new Image( variablesScreenIconImage ) + }; + + Screen.call( this, + function() { + return new GenericAreaModel( { + allowExponents: true + } ); + }, + function( model ) { return new GenericAreaScreenView( model, 0 ); }, + options + ); +} + +areaModelCommon.register( 'VariablesScreen', VariablesScreen ); + +inherit( Screen, VariablesScreen ); +export default VariablesScreen; \ No newline at end of file diff --git a/mipmaps/explore-screen-icon_png.js b/mipmaps/explore-screen-icon_png.js new file mode 100644 index 00000000..15b54596 --- /dev/null +++ b/mipmaps/explore-screen-icon_png.js @@ -0,0 +1,38 @@ +/* eslint-disable */ +var mipmaps = [ { + "width": 916, + "height": 623, + "url": "" +}, { + "width": 458, + "height": 312, + "url": "" +}, { + "width": 229, + "height": 156, + "url": "" +}, { + "width": 115, + "height": 78, + "url": "" +}, { + "width": 58, + "height": 39, + "url": "" +} ]; +mipmaps.forEach( function( mipmap ) { + mipmap.img = new Image(); + window.phetImages.push( mipmap.img ); // make sure it's loaded before the sim launches + mipmap.img.src = mipmap.url; // trigger the loading of the image for its level + mipmap.canvas = document.createElement( 'canvas' ); + mipmap.canvas.width = mipmap.width; + mipmap.canvas.height = mipmap.height; + var context = mipmap.canvas.getContext( '2d' ); + mipmap.updateCanvas = function() { + if ( mipmap.img.complete && ( typeof mipmap.img.naturalWidth === 'undefined' || mipmap.img.naturalWidth > 0 ) ) { + context.drawImage( mipmap.img, 0, 0 ); + delete mipmap.updateCanvas; + } + }; +} ); +export default mipmaps; \ No newline at end of file diff --git a/mipmaps/explore-screen-navbar_png.js b/mipmaps/explore-screen-navbar_png.js new file mode 100644 index 00000000..4e4c832c --- /dev/null +++ b/mipmaps/explore-screen-navbar_png.js @@ -0,0 +1,38 @@ +/* eslint-disable */ +var mipmaps = [ { + "width": 306, + "height": 208, + "url": "" +}, { + "width": 153, + "height": 104, + "url": "" +}, { + "width": 77, + "height": 52, + "url": "" +}, { + "width": 39, + "height": 26, + "url": "" +}, { + "width": 20, + "height": 13, + "url": "" +} ]; +mipmaps.forEach( function( mipmap ) { + mipmap.img = new Image(); + window.phetImages.push( mipmap.img ); // make sure it's loaded before the sim launches + mipmap.img.src = mipmap.url; // trigger the loading of the image for its level + mipmap.canvas = document.createElement( 'canvas' ); + mipmap.canvas.width = mipmap.width; + mipmap.canvas.height = mipmap.height; + var context = mipmap.canvas.getContext( '2d' ); + mipmap.updateCanvas = function() { + if ( mipmap.img.complete && ( typeof mipmap.img.naturalWidth === 'undefined' || mipmap.img.naturalWidth > 0 ) ) { + context.drawImage( mipmap.img, 0, 0 ); + delete mipmap.updateCanvas; + } + }; +} ); +export default mipmaps; \ No newline at end of file diff --git a/mipmaps/generic-game-screen-icon_png.js b/mipmaps/generic-game-screen-icon_png.js new file mode 100644 index 00000000..080a13ac --- /dev/null +++ b/mipmaps/generic-game-screen-icon_png.js @@ -0,0 +1,38 @@ +/* eslint-disable */ +var mipmaps = [ { + "width": 916, + "height": 623, + "url": "" +}, { + "width": 458, + "height": 312, + "url": "" +}, { + "width": 229, + "height": 156, + "url": "" +}, { + "width": 115, + "height": 78, + "url": "" +}, { + "width": 58, + "height": 39, + "url": "" +} ]; +mipmaps.forEach( function( mipmap ) { + mipmap.img = new Image(); + window.phetImages.push( mipmap.img ); // make sure it's loaded before the sim launches + mipmap.img.src = mipmap.url; // trigger the loading of the image for its level + mipmap.canvas = document.createElement( 'canvas' ); + mipmap.canvas.width = mipmap.width; + mipmap.canvas.height = mipmap.height; + var context = mipmap.canvas.getContext( '2d' ); + mipmap.updateCanvas = function() { + if ( mipmap.img.complete && ( typeof mipmap.img.naturalWidth === 'undefined' || mipmap.img.naturalWidth > 0 ) ) { + context.drawImage( mipmap.img, 0, 0 ); + delete mipmap.updateCanvas; + } + }; +} ); +export default mipmaps; \ No newline at end of file diff --git a/mipmaps/generic-game-screen-navbar_png.js b/mipmaps/generic-game-screen-navbar_png.js new file mode 100644 index 00000000..9b58d3f4 --- /dev/null +++ b/mipmaps/generic-game-screen-navbar_png.js @@ -0,0 +1,38 @@ +/* eslint-disable */ +var mipmaps = [ { + "width": 306, + "height": 208, + "url": "" +}, { + "width": 153, + "height": 104, + "url": "" +}, { + "width": 77, + "height": 52, + "url": "" +}, { + "width": 39, + "height": 26, + "url": "" +}, { + "width": 20, + "height": 13, + "url": "" +} ]; +mipmaps.forEach( function( mipmap ) { + mipmap.img = new Image(); + window.phetImages.push( mipmap.img ); // make sure it's loaded before the sim launches + mipmap.img.src = mipmap.url; // trigger the loading of the image for its level + mipmap.canvas = document.createElement( 'canvas' ); + mipmap.canvas.width = mipmap.width; + mipmap.canvas.height = mipmap.height; + var context = mipmap.canvas.getContext( '2d' ); + mipmap.updateCanvas = function() { + if ( mipmap.img.complete && ( typeof mipmap.img.naturalWidth === 'undefined' || mipmap.img.naturalWidth > 0 ) ) { + context.drawImage( mipmap.img, 0, 0 ); + delete mipmap.updateCanvas; + } + }; +} ); +export default mipmaps; \ No newline at end of file diff --git a/mipmaps/generic-screen-icon_png.js b/mipmaps/generic-screen-icon_png.js new file mode 100644 index 00000000..78229308 --- /dev/null +++ b/mipmaps/generic-screen-icon_png.js @@ -0,0 +1,38 @@ +/* eslint-disable */ +var mipmaps = [ { + "width": 916, + "height": 623, + "url": "" +}, { + "width": 458, + "height": 312, + "url": "" +}, { + "width": 229, + "height": 156, + "url": "" +}, { + "width": 115, + "height": 78, + "url": "" +}, { + "width": 58, + "height": 39, + "url": "" +} ]; +mipmaps.forEach( function( mipmap ) { + mipmap.img = new Image(); + window.phetImages.push( mipmap.img ); // make sure it's loaded before the sim launches + mipmap.img.src = mipmap.url; // trigger the loading of the image for its level + mipmap.canvas = document.createElement( 'canvas' ); + mipmap.canvas.width = mipmap.width; + mipmap.canvas.height = mipmap.height; + var context = mipmap.canvas.getContext( '2d' ); + mipmap.updateCanvas = function() { + if ( mipmap.img.complete && ( typeof mipmap.img.naturalWidth === 'undefined' || mipmap.img.naturalWidth > 0 ) ) { + context.drawImage( mipmap.img, 0, 0 ); + delete mipmap.updateCanvas; + } + }; +} ); +export default mipmaps; \ No newline at end of file diff --git a/mipmaps/level-1-icon_png.js b/mipmaps/level-1-icon_png.js new file mode 100644 index 00000000..ff854e25 --- /dev/null +++ b/mipmaps/level-1-icon_png.js @@ -0,0 +1,38 @@ +/* eslint-disable */ +var mipmaps = [ { + "width": 150, + "height": 196, + "url": "" +}, { + "width": 75, + "height": 98, + "url": "" +}, { + "width": 38, + "height": 49, + "url": "" +}, { + "width": 19, + "height": 25, + "url": "" +}, { + "width": 10, + "height": 13, + "url": "" +} ]; +mipmaps.forEach( function( mipmap ) { + mipmap.img = new Image(); + window.phetImages.push( mipmap.img ); // make sure it's loaded before the sim launches + mipmap.img.src = mipmap.url; // trigger the loading of the image for its level + mipmap.canvas = document.createElement( 'canvas' ); + mipmap.canvas.width = mipmap.width; + mipmap.canvas.height = mipmap.height; + var context = mipmap.canvas.getContext( '2d' ); + mipmap.updateCanvas = function() { + if ( mipmap.img.complete && ( typeof mipmap.img.naturalWidth === 'undefined' || mipmap.img.naturalWidth > 0 ) ) { + context.drawImage( mipmap.img, 0, 0 ); + delete mipmap.updateCanvas; + } + }; +} ); +export default mipmaps; \ No newline at end of file diff --git a/mipmaps/level-2-icon_png.js b/mipmaps/level-2-icon_png.js new file mode 100644 index 00000000..bd2f3588 --- /dev/null +++ b/mipmaps/level-2-icon_png.js @@ -0,0 +1,38 @@ +/* eslint-disable */ +var mipmaps = [ { + "width": 150, + "height": 196, + "url": "" +}, { + "width": 75, + "height": 98, + "url": "" +}, { + "width": 38, + "height": 49, + "url": "" +}, { + "width": 19, + "height": 25, + "url": "" +}, { + "width": 10, + "height": 13, + "url": "" +} ]; +mipmaps.forEach( function( mipmap ) { + mipmap.img = new Image(); + window.phetImages.push( mipmap.img ); // make sure it's loaded before the sim launches + mipmap.img.src = mipmap.url; // trigger the loading of the image for its level + mipmap.canvas = document.createElement( 'canvas' ); + mipmap.canvas.width = mipmap.width; + mipmap.canvas.height = mipmap.height; + var context = mipmap.canvas.getContext( '2d' ); + mipmap.updateCanvas = function() { + if ( mipmap.img.complete && ( typeof mipmap.img.naturalWidth === 'undefined' || mipmap.img.naturalWidth > 0 ) ) { + context.drawImage( mipmap.img, 0, 0 ); + delete mipmap.updateCanvas; + } + }; +} ); +export default mipmaps; \ No newline at end of file diff --git a/mipmaps/level-3-icon_png.js b/mipmaps/level-3-icon_png.js new file mode 100644 index 00000000..a8628faf --- /dev/null +++ b/mipmaps/level-3-icon_png.js @@ -0,0 +1,38 @@ +/* eslint-disable */ +var mipmaps = [ { + "width": 150, + "height": 196, + "url": "" +}, { + "width": 75, + "height": 98, + "url": "" +}, { + "width": 38, + "height": 49, + "url": "" +}, { + "width": 19, + "height": 25, + "url": "" +}, { + "width": 10, + "height": 13, + "url": "" +} ]; +mipmaps.forEach( function( mipmap ) { + mipmap.img = new Image(); + window.phetImages.push( mipmap.img ); // make sure it's loaded before the sim launches + mipmap.img.src = mipmap.url; // trigger the loading of the image for its level + mipmap.canvas = document.createElement( 'canvas' ); + mipmap.canvas.width = mipmap.width; + mipmap.canvas.height = mipmap.height; + var context = mipmap.canvas.getContext( '2d' ); + mipmap.updateCanvas = function() { + if ( mipmap.img.complete && ( typeof mipmap.img.naturalWidth === 'undefined' || mipmap.img.naturalWidth > 0 ) ) { + context.drawImage( mipmap.img, 0, 0 ); + delete mipmap.updateCanvas; + } + }; +} ); +export default mipmaps; \ No newline at end of file diff --git a/mipmaps/level-4-icon_png.js b/mipmaps/level-4-icon_png.js new file mode 100644 index 00000000..005b5ea5 --- /dev/null +++ b/mipmaps/level-4-icon_png.js @@ -0,0 +1,38 @@ +/* eslint-disable */ +var mipmaps = [ { + "width": 150, + "height": 196, + "url": "" +}, { + "width": 75, + "height": 98, + "url": "" +}, { + "width": 38, + "height": 49, + "url": "" +}, { + "width": 19, + "height": 25, + "url": "" +}, { + "width": 10, + "height": 13, + "url": "" +} ]; +mipmaps.forEach( function( mipmap ) { + mipmap.img = new Image(); + window.phetImages.push( mipmap.img ); // make sure it's loaded before the sim launches + mipmap.img.src = mipmap.url; // trigger the loading of the image for its level + mipmap.canvas = document.createElement( 'canvas' ); + mipmap.canvas.width = mipmap.width; + mipmap.canvas.height = mipmap.height; + var context = mipmap.canvas.getContext( '2d' ); + mipmap.updateCanvas = function() { + if ( mipmap.img.complete && ( typeof mipmap.img.naturalWidth === 'undefined' || mipmap.img.naturalWidth > 0 ) ) { + context.drawImage( mipmap.img, 0, 0 ); + delete mipmap.updateCanvas; + } + }; +} ); +export default mipmaps; \ No newline at end of file diff --git a/mipmaps/level-5-icon_png.js b/mipmaps/level-5-icon_png.js new file mode 100644 index 00000000..7e3862a7 --- /dev/null +++ b/mipmaps/level-5-icon_png.js @@ -0,0 +1,38 @@ +/* eslint-disable */ +var mipmaps = [ { + "width": 150, + "height": 196, + "url": "" +}, { + "width": 75, + "height": 98, + "url": "" +}, { + "width": 38, + "height": 49, + "url": "" +}, { + "width": 19, + "height": 25, + "url": "" +}, { + "width": 10, + "height": 13, + "url": "" +} ]; +mipmaps.forEach( function( mipmap ) { + mipmap.img = new Image(); + window.phetImages.push( mipmap.img ); // make sure it's loaded before the sim launches + mipmap.img.src = mipmap.url; // trigger the loading of the image for its level + mipmap.canvas = document.createElement( 'canvas' ); + mipmap.canvas.width = mipmap.width; + mipmap.canvas.height = mipmap.height; + var context = mipmap.canvas.getContext( '2d' ); + mipmap.updateCanvas = function() { + if ( mipmap.img.complete && ( typeof mipmap.img.naturalWidth === 'undefined' || mipmap.img.naturalWidth > 0 ) ) { + context.drawImage( mipmap.img, 0, 0 ); + delete mipmap.updateCanvas; + } + }; +} ); +export default mipmaps; \ No newline at end of file diff --git a/mipmaps/level-6-icon_png.js b/mipmaps/level-6-icon_png.js new file mode 100644 index 00000000..89c85f6b --- /dev/null +++ b/mipmaps/level-6-icon_png.js @@ -0,0 +1,38 @@ +/* eslint-disable */ +var mipmaps = [ { + "width": 150, + "height": 196, + "url": "" +}, { + "width": 75, + "height": 98, + "url": "" +}, { + "width": 38, + "height": 49, + "url": "" +}, { + "width": 19, + "height": 25, + "url": "" +}, { + "width": 10, + "height": 13, + "url": "" +} ]; +mipmaps.forEach( function( mipmap ) { + mipmap.img = new Image(); + window.phetImages.push( mipmap.img ); // make sure it's loaded before the sim launches + mipmap.img.src = mipmap.url; // trigger the loading of the image for its level + mipmap.canvas = document.createElement( 'canvas' ); + mipmap.canvas.width = mipmap.width; + mipmap.canvas.height = mipmap.height; + var context = mipmap.canvas.getContext( '2d' ); + mipmap.updateCanvas = function() { + if ( mipmap.img.complete && ( typeof mipmap.img.naturalWidth === 'undefined' || mipmap.img.naturalWidth > 0 ) ) { + context.drawImage( mipmap.img, 0, 0 ); + delete mipmap.updateCanvas; + } + }; +} ); +export default mipmaps; \ No newline at end of file diff --git a/mipmaps/multiply-screen-icon_png.js b/mipmaps/multiply-screen-icon_png.js new file mode 100644 index 00000000..0a557467 --- /dev/null +++ b/mipmaps/multiply-screen-icon_png.js @@ -0,0 +1,38 @@ +/* eslint-disable */ +var mipmaps = [ { + "width": 1142, + "height": 777, + "url": "" +}, { + "width": 571, + "height": 389, + "url": "" +}, { + "width": 286, + "height": 195, + "url": "" +}, { + "width": 143, + "height": 98, + "url": "" +}, { + "width": 72, + "height": 49, + "url": "" +} ]; +mipmaps.forEach( function( mipmap ) { + mipmap.img = new Image(); + window.phetImages.push( mipmap.img ); // make sure it's loaded before the sim launches + mipmap.img.src = mipmap.url; // trigger the loading of the image for its level + mipmap.canvas = document.createElement( 'canvas' ); + mipmap.canvas.width = mipmap.width; + mipmap.canvas.height = mipmap.height; + var context = mipmap.canvas.getContext( '2d' ); + mipmap.updateCanvas = function() { + if ( mipmap.img.complete && ( typeof mipmap.img.naturalWidth === 'undefined' || mipmap.img.naturalWidth > 0 ) ) { + context.drawImage( mipmap.img, 0, 0 ); + delete mipmap.updateCanvas; + } + }; +} ); +export default mipmaps; \ No newline at end of file diff --git a/mipmaps/multiply-screen-navbar_png.js b/mipmaps/multiply-screen-navbar_png.js new file mode 100644 index 00000000..1cdd3217 --- /dev/null +++ b/mipmaps/multiply-screen-navbar_png.js @@ -0,0 +1,38 @@ +/* eslint-disable */ +var mipmaps = [ { + "width": 306, + "height": 208, + "url": "" +}, { + "width": 153, + "height": 104, + "url": "" +}, { + "width": 77, + "height": 52, + "url": "" +}, { + "width": 39, + "height": 26, + "url": "" +}, { + "width": 20, + "height": 13, + "url": "" +} ]; +mipmaps.forEach( function( mipmap ) { + mipmap.img = new Image(); + window.phetImages.push( mipmap.img ); // make sure it's loaded before the sim launches + mipmap.img.src = mipmap.url; // trigger the loading of the image for its level + mipmap.canvas = document.createElement( 'canvas' ); + mipmap.canvas.width = mipmap.width; + mipmap.canvas.height = mipmap.height; + var context = mipmap.canvas.getContext( '2d' ); + mipmap.updateCanvas = function() { + if ( mipmap.img.complete && ( typeof mipmap.img.naturalWidth === 'undefined' || mipmap.img.naturalWidth > 0 ) ) { + context.drawImage( mipmap.img, 0, 0 ); + delete mipmap.updateCanvas; + } + }; +} ); +export default mipmaps; \ No newline at end of file diff --git a/mipmaps/partition-screen-icon_png.js b/mipmaps/partition-screen-icon_png.js new file mode 100644 index 00000000..1db08487 --- /dev/null +++ b/mipmaps/partition-screen-icon_png.js @@ -0,0 +1,38 @@ +/* eslint-disable */ +var mipmaps = [ { + "width": 1142, + "height": 777, + "url": "" +}, { + "width": 571, + "height": 389, + "url": "" +}, { + "width": 286, + "height": 195, + "url": "" +}, { + "width": 143, + "height": 98, + "url": "" +}, { + "width": 72, + "height": 49, + "url": "" +} ]; +mipmaps.forEach( function( mipmap ) { + mipmap.img = new Image(); + window.phetImages.push( mipmap.img ); // make sure it's loaded before the sim launches + mipmap.img.src = mipmap.url; // trigger the loading of the image for its level + mipmap.canvas = document.createElement( 'canvas' ); + mipmap.canvas.width = mipmap.width; + mipmap.canvas.height = mipmap.height; + var context = mipmap.canvas.getContext( '2d' ); + mipmap.updateCanvas = function() { + if ( mipmap.img.complete && ( typeof mipmap.img.naturalWidth === 'undefined' || mipmap.img.naturalWidth > 0 ) ) { + context.drawImage( mipmap.img, 0, 0 ); + delete mipmap.updateCanvas; + } + }; +} ); +export default mipmaps; \ No newline at end of file diff --git a/mipmaps/partition-screen-navbar_png.js b/mipmaps/partition-screen-navbar_png.js new file mode 100644 index 00000000..ab255976 --- /dev/null +++ b/mipmaps/partition-screen-navbar_png.js @@ -0,0 +1,38 @@ +/* eslint-disable */ +var mipmaps = [ { + "width": 306, + "height": 208, + "url": "" +}, { + "width": 153, + "height": 104, + "url": "" +}, { + "width": 77, + "height": 52, + "url": "" +}, { + "width": 39, + "height": 26, + "url": "" +}, { + "width": 20, + "height": 13, + "url": "" +} ]; +mipmaps.forEach( function( mipmap ) { + mipmap.img = new Image(); + window.phetImages.push( mipmap.img ); // make sure it's loaded before the sim launches + mipmap.img.src = mipmap.url; // trigger the loading of the image for its level + mipmap.canvas = document.createElement( 'canvas' ); + mipmap.canvas.width = mipmap.width; + mipmap.canvas.height = mipmap.height; + var context = mipmap.canvas.getContext( '2d' ); + mipmap.updateCanvas = function() { + if ( mipmap.img.complete && ( typeof mipmap.img.naturalWidth === 'undefined' || mipmap.img.naturalWidth > 0 ) ) { + context.drawImage( mipmap.img, 0, 0 ); + delete mipmap.updateCanvas; + } + }; +} ); +export default mipmaps; \ No newline at end of file diff --git a/mipmaps/variables-game-screen-icon_png.js b/mipmaps/variables-game-screen-icon_png.js new file mode 100644 index 00000000..13fcad1e --- /dev/null +++ b/mipmaps/variables-game-screen-icon_png.js @@ -0,0 +1,38 @@ +/* eslint-disable */ +var mipmaps = [ { + "width": 916, + "height": 623, + "url": "" +}, { + "width": 458, + "height": 312, + "url": "" +}, { + "width": 229, + "height": 156, + "url": "" +}, { + "width": 115, + "height": 78, + "url": "" +}, { + "width": 58, + "height": 39, + "url": "" +} ]; +mipmaps.forEach( function( mipmap ) { + mipmap.img = new Image(); + window.phetImages.push( mipmap.img ); // make sure it's loaded before the sim launches + mipmap.img.src = mipmap.url; // trigger the loading of the image for its level + mipmap.canvas = document.createElement( 'canvas' ); + mipmap.canvas.width = mipmap.width; + mipmap.canvas.height = mipmap.height; + var context = mipmap.canvas.getContext( '2d' ); + mipmap.updateCanvas = function() { + if ( mipmap.img.complete && ( typeof mipmap.img.naturalWidth === 'undefined' || mipmap.img.naturalWidth > 0 ) ) { + context.drawImage( mipmap.img, 0, 0 ); + delete mipmap.updateCanvas; + } + }; +} ); +export default mipmaps; \ No newline at end of file diff --git a/mipmaps/variables-game-screen-navbar_png.js b/mipmaps/variables-game-screen-navbar_png.js new file mode 100644 index 00000000..9b58d3f4 --- /dev/null +++ b/mipmaps/variables-game-screen-navbar_png.js @@ -0,0 +1,38 @@ +/* eslint-disable */ +var mipmaps = [ { + "width": 306, + "height": 208, + "url": "" +}, { + "width": 153, + "height": 104, + "url": "" +}, { + "width": 77, + "height": 52, + "url": "" +}, { + "width": 39, + "height": 26, + "url": "" +}, { + "width": 20, + "height": 13, + "url": "" +} ]; +mipmaps.forEach( function( mipmap ) { + mipmap.img = new Image(); + window.phetImages.push( mipmap.img ); // make sure it's loaded before the sim launches + mipmap.img.src = mipmap.url; // trigger the loading of the image for its level + mipmap.canvas = document.createElement( 'canvas' ); + mipmap.canvas.width = mipmap.width; + mipmap.canvas.height = mipmap.height; + var context = mipmap.canvas.getContext( '2d' ); + mipmap.updateCanvas = function() { + if ( mipmap.img.complete && ( typeof mipmap.img.naturalWidth === 'undefined' || mipmap.img.naturalWidth > 0 ) ) { + context.drawImage( mipmap.img, 0, 0 ); + delete mipmap.updateCanvas; + } + }; +} ); +export default mipmaps; \ No newline at end of file diff --git a/mipmaps/variables-screen-icon_png.js b/mipmaps/variables-screen-icon_png.js new file mode 100644 index 00000000..2d516dde --- /dev/null +++ b/mipmaps/variables-screen-icon_png.js @@ -0,0 +1,38 @@ +/* eslint-disable */ +var mipmaps = [ { + "width": 916, + "height": 623, + "url": "" +}, { + "width": 458, + "height": 312, + "url": "" +}, { + "width": 229, + "height": 156, + "url": "" +}, { + "width": 115, + "height": 78, + "url": "" +}, { + "width": 58, + "height": 39, + "url": "" +} ]; +mipmaps.forEach( function( mipmap ) { + mipmap.img = new Image(); + window.phetImages.push( mipmap.img ); // make sure it's loaded before the sim launches + mipmap.img.src = mipmap.url; // trigger the loading of the image for its level + mipmap.canvas = document.createElement( 'canvas' ); + mipmap.canvas.width = mipmap.width; + mipmap.canvas.height = mipmap.height; + var context = mipmap.canvas.getContext( '2d' ); + mipmap.updateCanvas = function() { + if ( mipmap.img.complete && ( typeof mipmap.img.naturalWidth === 'undefined' || mipmap.img.naturalWidth > 0 ) ) { + context.drawImage( mipmap.img, 0, 0 ); + delete mipmap.updateCanvas; + } + }; +} ); +export default mipmaps; \ No newline at end of file