From 4fa5d8662e0247455c2010ed101667c432c73e83 Mon Sep 17 00:00:00 2001 From: samreid Date: Thu, 27 Feb 2020 11:38:07 -0700 Subject: [PATCH] Add repo to migration list, see https://github.com/phetsims/chipper/issues/875 --- arithmetic_en.html | 48 +- images/large-pointing-hand_png.js | 5 + images/small-pointing-hand_png.js | 5 + js/arithmetic-config.js | 51 -- js/arithmetic-main.js | 57 +- js/arithmetic-strings.js | 14 + js/arithmetic.js | 8 +- js/common/ArithmeticConstants.js | 35 +- js/common/ArithmeticGlobals.js | 19 +- js/common/ArithmeticQueryParameters.js | 26 +- js/common/model/ArithmeticModel.js | 636 +++++++++--------- js/common/model/FaceModel.js | 80 ++- js/common/model/GameState.js | 33 +- js/common/model/LevelModel.js | 192 +++--- js/common/model/ProblemModel.js | 48 +- .../view/ArithmeticFaceWithPointsNode.js | 167 +++-- js/common/view/ArithmeticView.js | 391 ++++++----- js/common/view/EquationInputNode.js | 230 ++++--- js/common/view/EquationNode.js | 120 ++-- js/common/view/LevelCompletedNodeWrapper.js | 93 ++- js/common/view/LevelSelectionNode.js | 277 ++++---- js/common/view/ScoreboardNode.js | 238 ++++--- js/common/view/WorkspaceNode.js | 409 ++++++----- js/common/view/table/AbstractCell.js | 282 ++++---- .../view/table/MultiplicationTableBodyCell.js | 81 ++- .../table/MultiplicationTableHeaderCell.js | 83 ++- .../view/table/MultiplicationTableNode.js | 548 ++++++++------- js/divide/DivideScreen.js | 79 ++- js/divide/model/DivideModel.js | 196 +++--- js/divide/view/DivideEquationNode.js | 115 ++-- js/divide/view/DivideScreenIconNode.js | 89 ++- js/divide/view/DivideScreenTableNode.js | 111 ++- js/divide/view/DivideView.js | 71 +- js/factor/FactorScreen.js | 79 ++- js/factor/model/FactorModel.js | 197 +++--- js/factor/view/CellInteractionListener.js | 102 ++- js/factor/view/FactorEquationNode.js | 85 ++- js/factor/view/FactorScreenIconNode.js | 111 ++- js/factor/view/FactorScreenTableNode.js | 454 ++++++------- js/factor/view/FactorView.js | 69 +- js/multiply/MultiplyScreen.js | 79 ++- js/multiply/model/MultiplyModel.js | 157 +++-- js/multiply/view/MultiplyEquationNode.js | 85 ++- js/multiply/view/MultiplyScreenIconNode.js | 79 ++- js/multiply/view/MultiplyScreenTableNode.js | 115 ++-- js/multiply/view/MultiplyView.js | 77 +-- .../arithmetic-phet-io-elements-baseline.js | 1 + .../arithmetic-phet-io-elements-overrides.js | 1 + js/phet-io/arithmetic-phet-io-types.js | 1 + mipmaps/divide_level_1_icon_png.js | 38 ++ mipmaps/divide_level_2_icon_png.js | 38 ++ mipmaps/divide_level_3_icon_png.js | 38 ++ mipmaps/factor_level_1_icon_png.js | 38 ++ mipmaps/factor_level_2_icon_png.js | 38 ++ mipmaps/factor_level_3_icon_png.js | 38 ++ mipmaps/multiply_level_1_icon_png.js | 38 ++ mipmaps/multiply_level_2_icon_png.js | 38 ++ mipmaps/multiply_level_3_icon_png.js | 38 ++ 58 files changed, 3532 insertions(+), 3339 deletions(-) create mode 100644 images/large-pointing-hand_png.js create mode 100644 images/small-pointing-hand_png.js delete mode 100644 js/arithmetic-config.js create mode 100644 js/arithmetic-strings.js create mode 100644 mipmaps/divide_level_1_icon_png.js create mode 100644 mipmaps/divide_level_2_icon_png.js create mode 100644 mipmaps/divide_level_3_icon_png.js create mode 100644 mipmaps/factor_level_1_icon_png.js create mode 100644 mipmaps/factor_level_2_icon_png.js create mode 100644 mipmaps/factor_level_3_icon_png.js create mode 100644 mipmaps/multiply_level_1_icon_png.js create mode 100644 mipmaps/multiply_level_2_icon_png.js create mode 100644 mipmaps/multiply_level_3_icon_png.js diff --git a/arithmetic_en.html b/arithmetic_en.html index ea03e2a9..b75ae5b6 100644 --- a/arithmetic_en.html +++ b/arithmetic_en.html @@ -17,6 +17,36 @@ \ No newline at end of file diff --git a/images/large-pointing-hand_png.js b/images/large-pointing-hand_png.js new file mode 100644 index 00000000..0d345d2b --- /dev/null +++ b/images/large-pointing-hand_png.js @@ -0,0 +1,5 @@ +/* eslint-disable */ +var img = new Image(); +window.phetImages.push( img ); +img.src = ''; +export default img; diff --git a/images/small-pointing-hand_png.js b/images/small-pointing-hand_png.js new file mode 100644 index 00000000..a91cad0f --- /dev/null +++ b/images/small-pointing-hand_png.js @@ -0,0 +1,5 @@ +/* eslint-disable */ +var img = new Image(); +window.phetImages.push( img ); +img.src = ''; +export default img; diff --git a/js/arithmetic-config.js b/js/arithmetic-config.js deleted file mode 100644 index 83f7001a..00000000 --- a/js/arithmetic-config.js +++ /dev/null @@ -1,51 +0,0 @@ -// Copyright 2014-2019, University of Colorado Boulder - -/* - * IMPORTANT: This file was auto-generated by "grunt generate-config". Please do not modify this directly. Instead - * please modify arithmetic/package.json to control dependencies. - * - * RequireJS configuration file for the arithmetic sim. - * Paths are relative to the location of this file. - */ - -require.config( { - - deps: [ 'arithmetic-main' ], - - 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. - ARITHMETIC: '.', - 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', - UTTERANCE_QUEUE: '../../utterance-queue/js', - VEGAS: '../../vegas/js' - }, - - // Cache busting is applied by default, but can be disabled via ?cacheBust=false, see initialize-globals.js - urlArgs: phet.chipper.getCacheBustArgs() -} ); diff --git a/js/arithmetic-main.js b/js/arithmetic-main.js index 2ba1763c..4dcc0120 100644 --- a/js/arithmetic-main.js +++ b/js/arithmetic-main.js @@ -5,39 +5,36 @@ * * @author John Blanco, Andrey Zelenkov (MLearner) */ -define( require => { - 'use strict'; - // modules - const DivideScreen = require( 'ARITHMETIC/divide/DivideScreen' ); - const FactorScreen = require( 'ARITHMETIC/factor/FactorScreen' ); - const MultiplyScreen = require( 'ARITHMETIC/multiply/MultiplyScreen' ); - const Sim = require( 'JOIST/Sim' ); - const SimLauncher = require( 'JOIST/SimLauncher' ); - const Tandem = require( 'TANDEM/Tandem' ); +import Sim from '../../joist/js/Sim.js'; +import SimLauncher from '../../joist/js/SimLauncher.js'; +import Tandem from '../../tandem/js/Tandem.js'; +import arithmeticStrings from './arithmetic-strings.js'; +import DivideScreen from './divide/DivideScreen.js'; +import FactorScreen from './factor/FactorScreen.js'; +import MultiplyScreen from './multiply/MultiplyScreen.js'; - // strings and images - const arithmeticTitleString = require( 'string!ARITHMETIC/arithmetic.title' ); +// strings and images +const arithmeticTitleString = arithmeticStrings.arithmetic.title; - // constants - const tandem = Tandem.ROOT; +// constants +const tandem = Tandem.ROOT; - const simOptions = { - credits: { - leadDesign: 'Michael Dubson, Amanda McGarry', - softwareDevelopment: 'John Blanco, Michael Dubson', - team: 'Bryce Gruneich, Karina K. R. Hensberry, Trish Loeblein, Ariel Paul, Kathy Perkins, Beth Stade', - qualityAssurance: 'Steele Dalton, Bryce Griebenow, Elise Morgan, Oliver Orejola, Ben Roberts, Bryan Yoelin', - thanks: 'Thanks to Mobile Learner Labs for working with the PhET development team to convert this simulation to HTML5.' - } - }; +const simOptions = { + credits: { + leadDesign: 'Michael Dubson, Amanda McGarry', + softwareDevelopment: 'John Blanco, Michael Dubson', + team: 'Bryce Gruneich, Karina K. R. Hensberry, Trish Loeblein, Ariel Paul, Kathy Perkins, Beth Stade', + qualityAssurance: 'Steele Dalton, Bryce Griebenow, Elise Morgan, Oliver Orejola, Ben Roberts, Bryan Yoelin', + thanks: 'Thanks to Mobile Learner Labs for working with the PhET development team to convert this simulation to HTML5.' + } +}; - SimLauncher.launch( function() { - // Create and start the sim - new Sim( arithmeticTitleString, [ - new MultiplyScreen( { tandem: tandem.createTandem( 'multiplyScreen' ) } ), - new FactorScreen( { tandem: tandem.createTandem( 'factorScreen' ) } ), - new DivideScreen( { tandem: tandem.createTandem( 'divideScreen' ) } ) - ], simOptions ).start(); - } ); +SimLauncher.launch( function() { + // Create and start the sim + new Sim( arithmeticTitleString, [ + new MultiplyScreen( { tandem: tandem.createTandem( 'multiplyScreen' ) } ), + new FactorScreen( { tandem: tandem.createTandem( 'factorScreen' ) } ), + new DivideScreen( { tandem: tandem.createTandem( 'divideScreen' ) } ) + ], simOptions ).start(); } ); \ No newline at end of file diff --git a/js/arithmetic-strings.js b/js/arithmetic-strings.js new file mode 100644 index 00000000..009b8d71 --- /dev/null +++ b/js/arithmetic-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 arithmetic from './arithmetic.js'; + +const arithmeticStrings = getStringModule( 'ARITHMETIC' ); + +arithmetic.register( 'arithmeticStrings', arithmeticStrings ); + +export default arithmeticStrings; diff --git a/js/arithmetic.js b/js/arithmetic.js index 07a17042..fd807b3a 100644 --- a/js/arithmetic.js +++ b/js/arithmetic.js @@ -5,11 +5,7 @@ * * @author John Blanco */ -define( require => { - 'use strict'; - // modules - const Namespace = require( 'PHET_CORE/Namespace' ); +import Namespace from '../../phet-core/js/Namespace.js'; - return new Namespace( 'arithmetic' ); -} ); \ No newline at end of file +export default new Namespace( 'arithmetic' ); \ No newline at end of file diff --git a/js/common/ArithmeticConstants.js b/js/common/ArithmeticConstants.js index 441e9ec0..5ae18f72 100644 --- a/js/common/ArithmeticConstants.js +++ b/js/common/ArithmeticConstants.js @@ -6,26 +6,21 @@ * @author Andrey Zelenkov (MLearner) */ -define( require => { - 'use strict'; +import Screen from '../../../joist/js/Screen.js'; +import PhetFont from '../../../scenery-phet/js/PhetFont.js'; +import arithmetic from '../arithmetic.js'; - // modules - const arithmetic = require( 'ARITHMETIC/arithmetic' ); - const PhetFont = require( 'SCENERY_PHET/PhetFont' ); - const Screen = require( 'JOIST/Screen' ); +const ArithmeticConstants = { + BACKGROUND_COLOR: 'rgb( 173, 202, 255 )', + CURSOR_BLINK_INTERVAL: 500, // duration of animation in milliseconds + EQUATION_FONT_TEXT: new PhetFont( { size: 32 } ), + ICON_BACKGROUND_COLOR: 'rgb( 173, 202, 255 )', + INPUT_LENGTH_MAX: 3, // max input length + SCREEN_ICON_SIZE: Screen.MINIMUM_HOME_SCREEN_ICON_SIZE, // size of screen icons + NUM_STARS: 5, // number of stars in select level buttons + WORKSPACE_BACKGROUND_COLOR: 'rgb(130,181,252)' +}; - const ArithmeticConstants = { - BACKGROUND_COLOR: 'rgb( 173, 202, 255 )', - CURSOR_BLINK_INTERVAL: 500, // duration of animation in milliseconds - EQUATION_FONT_TEXT: new PhetFont( { size: 32 } ), - ICON_BACKGROUND_COLOR: 'rgb( 173, 202, 255 )', - INPUT_LENGTH_MAX: 3, // max input length - SCREEN_ICON_SIZE: Screen.MINIMUM_HOME_SCREEN_ICON_SIZE, // size of screen icons - NUM_STARS: 5, // number of stars in select level buttons - WORKSPACE_BACKGROUND_COLOR: 'rgb(130,181,252)' - }; +arithmetic.register( 'ArithmeticConstants', ArithmeticConstants ); - arithmetic.register( 'ArithmeticConstants', ArithmeticConstants ); - - return ArithmeticConstants; -} ); \ No newline at end of file +export default ArithmeticConstants; \ No newline at end of file diff --git a/js/common/ArithmeticGlobals.js b/js/common/ArithmeticGlobals.js index 2383c07b..a5e83aca 100644 --- a/js/common/ArithmeticGlobals.js +++ b/js/common/ArithmeticGlobals.js @@ -6,18 +6,13 @@ * @author John Blanco */ -define( require => { - 'use strict'; +import Property from '../../../axon/js/Property.js'; +import arithmetic from '../arithmetic.js'; - // modules - const arithmetic = require( 'ARITHMETIC/arithmetic' ); - const Property = require( 'AXON/Property' ); +const ArithmeticGlobals = { + timerEnabledProperty: new Property( false ) +}; - const ArithmeticGlobals = { - timerEnabledProperty: new Property( false ) - }; +arithmetic.register( 'ArithmeticGlobals', ArithmeticGlobals ); - arithmetic.register( 'ArithmeticGlobals', ArithmeticGlobals ); - - return ArithmeticGlobals; -} ); \ No newline at end of file +export default ArithmeticGlobals; \ No newline at end of file diff --git a/js/common/ArithmeticQueryParameters.js b/js/common/ArithmeticQueryParameters.js index 7f21559f..87dd292b 100644 --- a/js/common/ArithmeticQueryParameters.js +++ b/js/common/ArithmeticQueryParameters.js @@ -5,24 +5,20 @@ * * @author John Blanco */ -define( require => { - 'use strict'; - // modules - const arithmetic = require( 'ARITHMETIC/arithmetic' ); +import arithmetic from '../arithmetic.js'; - const ArithmeticQueryParameters = QueryStringMachine.getAll( { +const ArithmeticQueryParameters = QueryStringMachine.getAll( { - // automatically answer most problems to enable faster testing of level completion - autoAnswer: { type: 'flag' } - } ); + // automatically answer most problems to enable faster testing of level completion + autoAnswer: { type: 'flag' } +} ); - // prevent auto answer in versions that are intended for publication - if ( phet.chipper.isProduction && !phet.chipper.isDebugBuild ) { - ArithmeticQueryParameters.autoAnswer = false; - } +// prevent auto answer in versions that are intended for publication +if ( phet.chipper.isProduction && !phet.chipper.isDebugBuild ) { + ArithmeticQueryParameters.autoAnswer = false; +} - arithmetic.register( 'ArithmeticQueryParameters', ArithmeticQueryParameters ); +arithmetic.register( 'ArithmeticQueryParameters', ArithmeticQueryParameters ); - return ArithmeticQueryParameters; -} ); +export default ArithmeticQueryParameters; \ No newline at end of file diff --git a/js/common/model/ArithmeticModel.js b/js/common/model/ArithmeticModel.js index 04a1bbae..195713a6 100644 --- a/js/common/model/ArithmeticModel.js +++ b/js/common/model/ArithmeticModel.js @@ -7,357 +7,353 @@ * @author Andrey Zelenkov (MLearner) * @author John Blanco */ -define( require => { - 'use strict'; - - // modules - const arithmetic = require( 'ARITHMETIC/arithmetic' ); - const ArithmeticGlobals = require( 'ARITHMETIC/common/ArithmeticGlobals' ); - const ArithmeticQueryParameters = require( 'ARITHMETIC/common/ArithmeticQueryParameters' ); - const BooleanIO = require( 'TANDEM/types/BooleanIO' ); - const Emitter = require( 'AXON/Emitter' ); - const FaceModel = require( 'ARITHMETIC/common/model/FaceModel' ); - const GameState = require( 'ARITHMETIC/common/model/GameState' ); - const inherit = require( 'PHET_CORE/inherit' ); - const LevelModel = require( 'ARITHMETIC/common/model/LevelModel' ); - const merge = require( 'PHET_CORE/merge' ); - const NumberIO = require( 'TANDEM/types/NumberIO' ); - const ProblemModel = require( 'ARITHMETIC/common/model/ProblemModel' ); - const Property = require( 'AXON/Property' ); - const StringIO = require( 'TANDEM/types/StringIO' ); - const timer = require( 'AXON/timer' ); - - // constants - const FEEDBACK_TIME = 1200; // in milliseconds, time that the feedback is presented before moving to next problem - /** - * Constructor for ArithmeticModel - * @constructor - */ - function ArithmeticModel( tandem, options ) { - const self = this; +import Emitter from '../../../../axon/js/Emitter.js'; +import Property from '../../../../axon/js/Property.js'; +import timer from '../../../../axon/js/timer.js'; +import inherit from '../../../../phet-core/js/inherit.js'; +import merge from '../../../../phet-core/js/merge.js'; +import BooleanIO from '../../../../tandem/js/types/BooleanIO.js'; +import NumberIO from '../../../../tandem/js/types/NumberIO.js'; +import StringIO from '../../../../tandem/js/types/StringIO.js'; +import arithmetic from '../../arithmetic.js'; +import ArithmeticGlobals from '../ArithmeticGlobals.js'; +import ArithmeticQueryParameters from '../ArithmeticQueryParameters.js'; +import FaceModel from './FaceModel.js'; +import GameState from './GameState.js'; +import LevelModel from './LevelModel.js'; +import ProblemModel from './ProblemModel.js'; + +// constants +const FEEDBACK_TIME = 1200; // in milliseconds, time that the feedback is presented before moving to next problem - // @private - for PhET-iO - this.checkAnswerEmitter = new Emitter( { - tandem: tandem.createTandem( 'checkAnswerEmitter' ), - parameters: [ - { name: 'multiplicand', phetioType: NumberIO }, - { name: 'product', phetioType: NumberIO }, - { name: 'multiplier', phetioType: NumberIO }, - { name: 'isCorrect', phetioType: BooleanIO }, - { name: 'asString', phetioType: StringIO }, - { name: 'input', phetioType: StringIO } - ] - } ); +/** + * Constructor for ArithmeticModel + * @constructor + */ +function ArithmeticModel( tandem, options ) { + const self = this; + + // @private - for PhET-iO + this.checkAnswerEmitter = new Emitter( { + tandem: tandem.createTandem( 'checkAnswerEmitter' ), + parameters: [ + { name: 'multiplicand', phetioType: NumberIO }, + { name: 'product', phetioType: NumberIO }, + { name: 'multiplier', phetioType: NumberIO }, + { name: 'isCorrect', phetioType: BooleanIO }, + { name: 'asString', phetioType: StringIO }, + { name: 'input', phetioType: StringIO } + ] + } ); - // set up the 'fillEquation' function, which is used to fill in the missing portion(s) based on the user's inputs - options = merge( { fillEquation: null }, options ); - this.fillEquation = options.fillEquation; // @public + // set up the 'fillEquation' function, which is used to fill in the missing portion(s) based on the user's inputs + options = merge( { fillEquation: null }, options ); + this.fillEquation = options.fillEquation; // @public - // @public - active game level, null represents none - this.levelNumberProperty = new Property( null ); + // @public - active game level, null represents none + this.levelNumberProperty = new Property( null ); - // @public - user's input value - this.inputProperty = new Property( '' ); + // @public - user's input value + this.inputProperty = new Property( '' ); - // @public - reference to the portion of the equation that is awaiting input from the user - this.activeInputProperty = new Property( null ); + // @public - reference to the portion of the equation that is awaiting input from the user + this.activeInputProperty = new Property( null ); - // @public - current game state - this.stateProperty = new Property( GameState.SELECTING_LEVEL ); + // @public - current game state + this.stateProperty = new Property( GameState.SELECTING_LEVEL ); - // @public - emitter that emits an even when a refresh occurs - this.refreshEmitter = new Emitter(); + // @public - emitter that emits an even when a refresh occurs + this.refreshEmitter = new Emitter(); - // @public - array of models that correspond to a given difficulty level - this.levelModels = [ - // level 1 - new LevelModel( 6 ), - // level 2 - new LevelModel( 9 ), - // level 3 - new LevelModel( 12 ) - ]; + // @public - array of models that correspond to a given difficulty level + this.levelModels = [ + // level 1 + new LevelModel( 6 ), + // level 2 + new LevelModel( 9 ), + // level 3 + new LevelModel( 12 ) + ]; - // @public - portion of the model that represents a single problem - this.problemModel = new ProblemModel(); + // @public - portion of the model that represents a single problem + this.problemModel = new ProblemModel(); - // @public - model for smile face - this.faceModel = new FaceModel(); + // @public - model for smile face + this.faceModel = new FaceModel(); - // handles game state transitions that pertain to the model (does not require handling GameState.SELECTING_LEVEL) - this.stateProperty.lazyLink( function( newState, oldState ) { - if ( oldState === GameState.SELECTING_LEVEL && newState === GameState.AWAITING_USER_INPUT ) { + // handles game state transitions that pertain to the model (does not require handling GameState.SELECTING_LEVEL) + this.stateProperty.lazyLink( function( newState, oldState ) { + if ( oldState === GameState.SELECTING_LEVEL && newState === GameState.AWAITING_USER_INPUT ) { - // start (or restart) the game timer - self.activeLevelModel.gameTimer.start(); + // start (or restart) the game timer + self.activeLevelModel.gameTimer.start(); - // update display score - self.activeLevelModel.displayScoreProperty.set( self.activeLevelModel.currentScoreProperty.get() ); - } - } ); - } + // update display score + self.activeLevelModel.displayScoreProperty.set( self.activeLevelModel.currentScoreProperty.get() ); + } + } ); +} - arithmetic.register( 'ArithmeticModel', ArithmeticModel ); +arithmetic.register( 'ArithmeticModel', ArithmeticModel ); - return inherit( Object, ArithmeticModel, { +export default inherit( Object, ArithmeticModel, { - // @protected - get the current level model, use this to make the code more readable - get activeLevelModel() { - return this.levelModels[ this.levelNumberProperty.get() ]; - }, + // @protected - get the current level model, use this to make the code more readable + get activeLevelModel() { + return this.levelModels[ this.levelNumberProperty.get() ]; + }, - /** - * Check whether the answer submitted by the user is correct. The user's answer must have been stored in the - * appropriate portion of the problem model before this method is invoked. Doing it this way allows this general - * method to be used to verify the answer. - * @public - */ - submitAnswer: function() { - const self = this; + /** + * Check whether the answer submitted by the user is correct. The user's answer must have been stored in the + * appropriate portion of the problem model before this method is invoked. Doing it this way allows this general + * method to be used to verify the answer. + * @public + */ + submitAnswer: function() { + const self = this; - const isCorrect = this.problemModel.multiplicandProperty.get() * this.problemModel.multiplierProperty.get() === this.problemModel.productProperty.get(); - const string = this.problemModel.multiplicandProperty.get() + ' x ' + this.problemModel.multiplierProperty.get() + ' = ' + this.problemModel.productProperty.get(); - this.checkAnswerEmitter.emit( + const isCorrect = this.problemModel.multiplicandProperty.get() * this.problemModel.multiplierProperty.get() === this.problemModel.productProperty.get(); + const string = this.problemModel.multiplicandProperty.get() + ' x ' + this.problemModel.multiplierProperty.get() + ' = ' + this.problemModel.productProperty.get(); + this.checkAnswerEmitter.emit( + this.problemModel.multiplicandProperty.get(), + this.problemModel.productProperty.get(), + this.problemModel.multiplierProperty.get(), + isCorrect, + string, + this.inputProperty.get() ); + if ( isCorrect ) { + + // add the problem value to the total score + this.activeLevelModel.currentScoreProperty.value += this.problemModel.possiblePointsProperty.get(); + + // update the displayed score + this.activeLevelModel.displayScoreProperty.set( this.activeLevelModel.currentScoreProperty.get() ); + + // set the face to smile + this.faceModel.pointsToDisplayProperty.set( this.problemModel.possiblePointsProperty.get() ); + this.faceModel.isSmileProperty.set( true ); + this.faceModel.showFace(); + + // mark this table entry as solved + this.activeLevelModel.markCellAsUsed( this.problemModel.multiplicandProperty.get(), - this.problemModel.productProperty.get(), - this.problemModel.multiplierProperty.get(), - isCorrect, - string, - this.inputProperty.get() ); - if ( isCorrect ) { - - // add the problem value to the total score - this.activeLevelModel.currentScoreProperty.value += this.problemModel.possiblePointsProperty.get(); - - // update the displayed score - this.activeLevelModel.displayScoreProperty.set( this.activeLevelModel.currentScoreProperty.get() ); - - // set the face to smile - this.faceModel.pointsToDisplayProperty.set( this.problemModel.possiblePointsProperty.get() ); - this.faceModel.isSmileProperty.set( true ); - this.faceModel.showFace(); - - // mark this table entry as solved - this.activeLevelModel.markCellAsUsed( - this.problemModel.multiplicandProperty.get(), - this.problemModel.multiplierProperty.get() - ); - - // show the feedback that indicates a correct answer - this.stateProperty.set( GameState.DISPLAYING_CORRECT_ANSWER_FEEDBACK ); - - // start a timer that will set up the next problem - this.feedbackTimer = timer.setTimeout( - function() { - self.feedbackTimer = null; - self.nextProblem(); - }, - FEEDBACK_TIME - ); - } - // incorrect answer - else { - // player will not get points for this task - this.problemModel.possiblePointsProperty.set( 0 ); - - // set face model state - this.faceModel.pointsToDisplayProperty.set( this.problemModel.possiblePointsProperty.get() ); - this.faceModel.isSmileProperty.set( false ); - this.faceModel.showFace(); - - // set the appropriate state - this.stateProperty.set( GameState.DISPLAYING_INCORRECT_ANSWER_FEEDBACK ); - } - }, - - /** - * Move to the next problem or, if all problems have been answered, move to the state where results are shown. - * @private - */ - nextProblem: function() { - if ( this.setUpUnansweredProblem() ) { - this.inputProperty.reset(); - this.stateProperty.set( GameState.AWAITING_USER_INPUT ); - } - else { - // all problems have been answered, the level is now complete - this.stateProperty.set( GameState.SHOWING_LEVEL_COMPLETED_DIALOG ); - this.activeLevelModel.gameTimer.stop(); - } - }, + this.problemModel.multiplierProperty.get() + ); + + // show the feedback that indicates a correct answer + this.stateProperty.set( GameState.DISPLAYING_CORRECT_ANSWER_FEEDBACK ); + + // start a timer that will set up the next problem + this.feedbackTimer = timer.setTimeout( + function() { + self.feedbackTimer = null; + self.nextProblem(); + }, + FEEDBACK_TIME + ); + } + // incorrect answer + else { + // player will not get points for this task + this.problemModel.possiblePointsProperty.set( 0 ); + + // set face model state + this.faceModel.pointsToDisplayProperty.set( this.problemModel.possiblePointsProperty.get() ); + this.faceModel.isSmileProperty.set( false ); + this.faceModel.showFace(); + + // set the appropriate state + this.stateProperty.set( GameState.DISPLAYING_INCORRECT_ANSWER_FEEDBACK ); + } + }, - /** - * Retry the currently presented problem. - * @public - */ - retryProblem: function() { + /** + * Move to the next problem or, if all problems have been answered, move to the state where results are shown. + * @private + */ + nextProblem: function() { + if ( this.setUpUnansweredProblem() ) { + this.inputProperty.reset(); this.stateProperty.set( GameState.AWAITING_USER_INPUT ); - }, - - /** - * Pick an unanswered problem and set it up in the model. Must be overridden in sub-types, since the way problems - * are set up varies. - * - * Returns true if able to set up a problem, false if not. A return value of false generally indicates that all - * problems have been answered. - * - * @protected - */ - setUpUnansweredProblem: function() { - throw new Error( 'this function must be overridden in sub-classes' ); - }, - - /** - * Automatically answer most of the problems for this level. This is useful for testing, since it can save time - * when testing how the sim behaves when a user finishing answering all questions for a level. - * - * IMPORTANT: We need to be VERY CAREFUL that this is never available in the published sim. - * - * @protected - */ - autoAnswer: function() { - // does nothing in the base class, override in descendant classes if desired - }, - - // @public - returnToLevelSelectScreen: function() { - - if ( this.stateProperty.get() === GameState.AWAITING_USER_INPUT ) { - // reset any partial input that the user may have entered - this.inputProperty.reset(); - } + } + else { + // all problems have been answered, the level is now complete + this.stateProperty.set( GameState.SHOWING_LEVEL_COMPLETED_DIALOG ); + this.activeLevelModel.gameTimer.stop(); + } + }, - // save state of current level - this.saveGameEnvironment(); + /** + * Retry the currently presented problem. + * @public + */ + retryProblem: function() { + this.stateProperty.set( GameState.AWAITING_USER_INPUT ); + }, - // if there is a timer running for displaying feedback, cancel it - if ( this.feedbackTimer ) { - timer.clearTimeout( this.feedbackTimer ); - } + /** + * Pick an unanswered problem and set it up in the model. Must be overridden in sub-types, since the way problems + * are set up varies. + * + * Returns true if able to set up a problem, false if not. A return value of false generally indicates that all + * problems have been answered. + * + * @protected + */ + setUpUnansweredProblem: function() { + throw new Error( 'this function must be overridden in sub-classes' ); + }, - // go back to the level selection screen - this.stateProperty.set( GameState.SELECTING_LEVEL ); - }, + /** + * Automatically answer most of the problems for this level. This is useful for testing, since it can save time + * when testing how the sim behaves when a user finishing answering all questions for a level. + * + * IMPORTANT: We need to be VERY CAREFUL that this is never available in the published sim. + * + * @protected + */ + autoAnswer: function() { + // does nothing in the base class, override in descendant classes if desired + }, - // @public - refreshLevel: function() { - if ( this.feedbackTimer ) { - timer.clearTimeout( this.feedbackTimer ); - } - this.resetLevel(); - this.activeLevelModel.displayScoreProperty.reset(); - this.nextProblem(); - this.activeLevelModel.gameTimer.start(); // may already be running, if so this is a no-op - this.refreshEmitter.emit(); + // @public + returnToLevelSelectScreen: function() { - // automatically answer most of the problems if enabled - this is for testing - ArithmeticQueryParameters.autoAnswer && this.autoAnswer(); - }, - - // @private - resetLevelModels: function() { - this.levelModels.forEach( function( levelModel ) { - levelModel.reset(); - } ); - }, - - // @public - set the level to be played, initializing or restoring the level as appropriate - setLevel: function( level ) { - this.levelNumberProperty.set( level ); - - // restore or init new environment for game - if ( this.levelModels[ level ].environment ) { - this.restoreGameEnvironment( this.levelModels[ level ].environment ); - if ( this.stateProperty.get() === GameState.DISPLAYING_CORRECT_ANSWER_FEEDBACK ) { - - // The user hit the back button before the feedback timer expired, so the next problem wasn't set up. We need - // to set it up now. See https://github.com/phetsims/arithmetic/issues/145 - this.nextProblem(); - } - } - else { - this.nextProblem(); + if ( this.stateProperty.get() === GameState.AWAITING_USER_INPUT ) { + // reset any partial input that the user may have entered + this.inputProperty.reset(); + } - // automatically answer most of the problems if enabled - this is for testing - ArithmeticQueryParameters.autoAnswer && this.autoAnswer(); - } - }, + // save state of current level + this.saveGameEnvironment(); - // @private - resetLevel: function() { - this.activeLevelModel.reset(); - this.inputProperty.reset(); - this.problemModel.reset(); - this.faceModel.reset(); - this.faceModel.hideFace(); - }, + // if there is a timer running for displaying feedback, cancel it + if ( this.feedbackTimer ) { + timer.clearTimeout( this.feedbackTimer ); + } - // @public - select an unused multiplican-multiplier pair - selectUnusedMultiplierPair: function() { - return this.activeLevelModel.selectUnusedMultiplierPair(); - }, + // go back to the level selection screen + this.stateProperty.set( GameState.SELECTING_LEVEL ); + }, - // @public - reset the scores, clear the boards - reset: function() { + // @public + refreshLevel: function() { + if ( this.feedbackTimer ) { + timer.clearTimeout( this.feedbackTimer ); + } + this.resetLevel(); + this.activeLevelModel.displayScoreProperty.reset(); + this.nextProblem(); + this.activeLevelModel.gameTimer.start(); // may already be running, if so this is a no-op + this.refreshEmitter.emit(); + + // automatically answer most of the problems if enabled - this is for testing + ArithmeticQueryParameters.autoAnswer && this.autoAnswer(); + }, + + // @private + resetLevelModels: function() { + this.levelModels.forEach( function( levelModel ) { + levelModel.reset(); + } ); + }, - this.levelNumberProperty.reset(); - this.inputProperty.reset(); - this.activeInputProperty.reset(); - this.stateProperty.reset(); - - // reset levels model - this.resetLevelModels(); - - // clear game level states - this.clearGameEnvironments(); - - // reset sound and timer on/off settings - ArithmeticGlobals.timerEnabledProperty.reset(); - }, - - // clear environments of all levels - clearGameEnvironments: function() { - this.levelModels.forEach( function( levelModel ) { - levelModel.environment = null; - } ); - }, - - // @private - set the 'game environment', generally used when switching to a different level - restoreGameEnvironment: function( environment ) { - this.activeLevelModel.currentScoreProperty.set( environment.currentScore ); - this.activeInputProperty.set( environment.activeInput ); - this.problemModel.multiplicandProperty.set( environment.multiplicand ); - this.problemModel.multiplierProperty.set( environment.multiplier ); - this.problemModel.productProperty.set( environment.product ); - this.problemModel.possiblePointsProperty.set( environment.possiblePoints ); - this.stateProperty.set( environment.state ); - this.inputProperty.set( environment.input ); - this.activeLevelModel.gameTimer.elapsedTimeProperty.value = environment.elapsedTime; - - // Elapsed time must account for any time that has gone by since the environment was saved. - if ( this.stateProperty.get() !== GameState.LEVEL_COMPLETED && - this.stateProperty.get() !== GameState.SHOWING_LEVEL_COMPLETED_DIALOG ) { - this.activeLevelModel.gameTimer.elapsedTimeProperty.value = - this.activeLevelModel.gameTimer.elapsedTimeProperty.value + - Math.floor( ( new Date().getTime() - environment.systemTimeWhenSaveOccurred ) / 1000 ); + // @public - set the level to be played, initializing or restoring the level as appropriate + setLevel: function( level ) { + this.levelNumberProperty.set( level ); + + // restore or init new environment for game + if ( this.levelModels[ level ].environment ) { + this.restoreGameEnvironment( this.levelModels[ level ].environment ); + if ( this.stateProperty.get() === GameState.DISPLAYING_CORRECT_ANSWER_FEEDBACK ) { + + // The user hit the back button before the feedback timer expired, so the next problem wasn't set up. We need + // to set it up now. See https://github.com/phetsims/arithmetic/issues/145 + this.nextProblem(); } - }, - - // save game environment of current level - saveGameEnvironment: function() { - this.activeLevelModel.environment = { - input: this.inputProperty.get(), - multiplicand: this.problemModel.multiplicandProperty.get(), - multiplier: this.problemModel.multiplierProperty.get(), - product: this.problemModel.productProperty.get(), - state: this.stateProperty.get(), - currentScore: this.activeLevelModel.currentScoreProperty.get(), - elapsedTime: this.activeLevelModel.gameTimer.elapsedTimeProperty.value, - systemTimeWhenSaveOccurred: new Date().getTime(), - possiblePoints: this.problemModel.possiblePointsProperty.get(), - activeInput: this.activeInputProperty.get() - }; } - } ); -} ); + else { + this.nextProblem(); + + // automatically answer most of the problems if enabled - this is for testing + ArithmeticQueryParameters.autoAnswer && this.autoAnswer(); + } + }, + + // @private + resetLevel: function() { + this.activeLevelModel.reset(); + this.inputProperty.reset(); + this.problemModel.reset(); + this.faceModel.reset(); + this.faceModel.hideFace(); + }, + + // @public - select an unused multiplican-multiplier pair + selectUnusedMultiplierPair: function() { + return this.activeLevelModel.selectUnusedMultiplierPair(); + }, + + // @public - reset the scores, clear the boards + reset: function() { + + this.levelNumberProperty.reset(); + this.inputProperty.reset(); + this.activeInputProperty.reset(); + this.stateProperty.reset(); + + // reset levels model + this.resetLevelModels(); + + // clear game level states + this.clearGameEnvironments(); + + // reset sound and timer on/off settings + ArithmeticGlobals.timerEnabledProperty.reset(); + }, + + // clear environments of all levels + clearGameEnvironments: function() { + this.levelModels.forEach( function( levelModel ) { + levelModel.environment = null; + } ); + }, + + // @private - set the 'game environment', generally used when switching to a different level + restoreGameEnvironment: function( environment ) { + this.activeLevelModel.currentScoreProperty.set( environment.currentScore ); + this.activeInputProperty.set( environment.activeInput ); + this.problemModel.multiplicandProperty.set( environment.multiplicand ); + this.problemModel.multiplierProperty.set( environment.multiplier ); + this.problemModel.productProperty.set( environment.product ); + this.problemModel.possiblePointsProperty.set( environment.possiblePoints ); + this.stateProperty.set( environment.state ); + this.inputProperty.set( environment.input ); + this.activeLevelModel.gameTimer.elapsedTimeProperty.value = environment.elapsedTime; + + // Elapsed time must account for any time that has gone by since the environment was saved. + if ( this.stateProperty.get() !== GameState.LEVEL_COMPLETED && + this.stateProperty.get() !== GameState.SHOWING_LEVEL_COMPLETED_DIALOG ) { + this.activeLevelModel.gameTimer.elapsedTimeProperty.value = + this.activeLevelModel.gameTimer.elapsedTimeProperty.value + + Math.floor( ( new Date().getTime() - environment.systemTimeWhenSaveOccurred ) / 1000 ); + } + }, + + // save game environment of current level + saveGameEnvironment: function() { + this.activeLevelModel.environment = { + input: this.inputProperty.get(), + multiplicand: this.problemModel.multiplicandProperty.get(), + multiplier: this.problemModel.multiplierProperty.get(), + product: this.problemModel.productProperty.get(), + state: this.stateProperty.get(), + currentScore: this.activeLevelModel.currentScoreProperty.get(), + elapsedTime: this.activeLevelModel.gameTimer.elapsedTimeProperty.value, + systemTimeWhenSaveOccurred: new Date().getTime(), + possiblePoints: this.problemModel.possiblePointsProperty.get(), + activeInput: this.activeInputProperty.get() + }; + } +} ); \ No newline at end of file diff --git a/js/common/model/FaceModel.js b/js/common/model/FaceModel.js index e62e9536..f8040ada 100644 --- a/js/common/model/FaceModel.js +++ b/js/common/model/FaceModel.js @@ -5,51 +5,47 @@ * * @author Andrey Zelenkov (MLearner) */ -define( require => { - 'use strict'; - // modules - const arithmetic = require( 'ARITHMETIC/arithmetic' ); - const Emitter = require( 'AXON/Emitter' ); - const inherit = require( 'PHET_CORE/inherit' ); - const Property = require( 'AXON/Property' ); +import Emitter from '../../../../axon/js/Emitter.js'; +import Property from '../../../../axon/js/Property.js'; +import inherit from '../../../../phet-core/js/inherit.js'; +import arithmetic from '../../arithmetic.js'; - function FaceModel() { +function FaceModel() { - // @public - Points to be displayed near the face. In this simulation, the user gets 1 point when they get a - // challenge correct on the first try, zero otherwise. - this.pointsToDisplayProperty = new Property( 1 ); + // @public - Points to be displayed near the face. In this simulation, the user gets 1 point when they get a + // challenge correct on the first try, zero otherwise. + this.pointsToDisplayProperty = new Property( 1 ); - // @public - flag that controls the expression that the face should depict - this.isSmileProperty = new Property( true ); + // @public - flag that controls the expression that the face should depict + this.isSmileProperty = new Property( true ); - // @public - emitters for showing and hiding the face - this.showFaceEmitter = new Emitter(); - this.hideFaceEmitter = new Emitter(); - } + // @public - emitters for showing and hiding the face + this.showFaceEmitter = new Emitter(); + this.hideFaceEmitter = new Emitter(); +} + +arithmetic.register( 'FaceModel', FaceModel ); + +export default inherit( Object, FaceModel, { - arithmetic.register( 'FaceModel', FaceModel ); - - return inherit( Object, FaceModel, { - - // @public - showFace: function() { - // Use an emitter to indicate that the face should be shown rather than a property, since by design it is shown - // and then fades. - this.showFaceEmitter.emit(); - }, - - // @public - hideFace: function() { - // Emit an event that indicates that the face should be hidden, should be ignored if the face is not currently - // shown. - this.hideFaceEmitter.emit(); - }, - - // public - reset: function() { - this.pointsToDisplayProperty.reset(); - this.isSmileProperty.reset(); - } - } ); -} ); + // @public + showFace: function() { + // Use an emitter to indicate that the face should be shown rather than a property, since by design it is shown + // and then fades. + this.showFaceEmitter.emit(); + }, + + // @public + hideFace: function() { + // Emit an event that indicates that the face should be hidden, should be ignored if the face is not currently + // shown. + this.hideFaceEmitter.emit(); + }, + + // public + reset: function() { + this.pointsToDisplayProperty.reset(); + this.isSmileProperty.reset(); + } +} ); \ No newline at end of file diff --git a/js/common/model/GameState.js b/js/common/model/GameState.js index 6fe07574..ea59c768 100644 --- a/js/common/model/GameState.js +++ b/js/common/model/GameState.js @@ -6,27 +6,22 @@ * @author Andrey Zelenkov (Mlearner) * @author John Blanco */ -define( require => { - 'use strict'; - // modules - const arithmetic = require( 'ARITHMETIC/arithmetic' ); +import arithmetic from '../../arithmetic.js'; - // @public - const GameState = { - SELECTING_LEVEL: 'SELECTING_LEVEL', - AWAITING_USER_INPUT: 'AWAITING_USER_INPUT', - DISPLAYING_CORRECT_ANSWER_FEEDBACK: 'DISPLAYING_CORRECT_ANSWER_FEEDBACK', - DISPLAYING_INCORRECT_ANSWER_FEEDBACK: 'DISPLAYING_INCORRECT_ANSWER_FEEDBACK', - SHOWING_LEVEL_COMPLETED_DIALOG: 'SHOWING_LEVEL_COMPLETED_DIALOG', - LEVEL_COMPLETED: 'LEVEL_COMPLETED' - }; +// @public +const GameState = { + SELECTING_LEVEL: 'SELECTING_LEVEL', + AWAITING_USER_INPUT: 'AWAITING_USER_INPUT', + DISPLAYING_CORRECT_ANSWER_FEEDBACK: 'DISPLAYING_CORRECT_ANSWER_FEEDBACK', + DISPLAYING_INCORRECT_ANSWER_FEEDBACK: 'DISPLAYING_INCORRECT_ANSWER_FEEDBACK', + SHOWING_LEVEL_COMPLETED_DIALOG: 'SHOWING_LEVEL_COMPLETED_DIALOG', + LEVEL_COMPLETED: 'LEVEL_COMPLETED' +}; - // 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 ); } - arithmetic.register( 'GameState', GameState ); +arithmetic.register( 'GameState', GameState ); - return GameState; - -} ); \ No newline at end of file +export default GameState; \ No newline at end of file diff --git a/js/common/model/LevelModel.js b/js/common/model/LevelModel.js index 09c59c11..656f9ef8 100644 --- a/js/common/model/LevelModel.js +++ b/js/common/model/LevelModel.js @@ -6,116 +6,112 @@ * @author Andrey Zelenkov (MLearner) * @author John Blanco */ -define( require => { - 'use strict'; - - // modules - const arithmetic = require( 'ARITHMETIC/arithmetic' ); - const GameTimer = require( 'VEGAS/GameTimer' ); - const inherit = require( 'PHET_CORE/inherit' ); - const Property = require( 'AXON/Property' ); - - /** - * @param {number} tableSize - width and height of the multiplication table, which is assumed to be square - * @constructor - */ - function LevelModel( tableSize ) { - - // observable model properties - this.bestTimeProperty = new Property( null ); // @public - best time for level - this.currentScoreProperty = new Property( 0 ); // @public - current score for level - this.displayScoreProperty = new Property( 0 ); // @public - score for displaying in level select buttons - - // non-Property model values - this.tableSize = tableSize; // @public, read only - this.perfectScore = tableSize * tableSize; // @public, read only - this.gameTimer = new GameTimer(); // @public - timer for this level - - // @private - 2d array that tracks the 'used' state of each of the cells in the multiplication table for this level, - // accessed through methods defined in the inherit block - this.cellUsedStates = new Array( tableSize ); - for ( let i = 0; i < tableSize; i++ ) { - this.cellUsedStates[ i ] = new Array( tableSize ); - } - this.clearCellUsedStates(); - this.environment = null; // @public - storage area used for saving/restoring this level's state in the parent model +import Property from '../../../../axon/js/Property.js'; +import inherit from '../../../../phet-core/js/inherit.js'; +import GameTimer from '../../../../vegas/js/GameTimer.js'; +import arithmetic from '../../arithmetic.js'; + +/** + * @param {number} tableSize - width and height of the multiplication table, which is assumed to be square + * @constructor + */ +function LevelModel( tableSize ) { + + // observable model properties + this.bestTimeProperty = new Property( null ); // @public - best time for level + this.currentScoreProperty = new Property( 0 ); // @public - current score for level + this.displayScoreProperty = new Property( 0 ); // @public - score for displaying in level select buttons + + // non-Property model values + this.tableSize = tableSize; // @public, read only + this.perfectScore = tableSize * tableSize; // @public, read only + this.gameTimer = new GameTimer(); // @public - timer for this level + + // @private - 2d array that tracks the 'used' state of each of the cells in the multiplication table for this level, + // accessed through methods defined in the inherit block + this.cellUsedStates = new Array( tableSize ); + for ( let i = 0; i < tableSize; i++ ) { + this.cellUsedStates[ i ] = new Array( tableSize ); } + this.clearCellUsedStates(); - arithmetic.register( 'LevelModel', LevelModel ); + this.environment = null; // @public - storage area used for saving/restoring this level's state in the parent model +} - return inherit( Object, LevelModel, { +arithmetic.register( 'LevelModel', LevelModel ); - // @public - reset this level - reset: function() { +export default inherit( Object, LevelModel, { - // reset this level model's explicitly defined properties - this.bestTimeProperty.reset(); - this.currentScoreProperty.reset(); - this.displayScoreProperty.reset(); + // @public - reset this level + reset: function() { - // reset the states of the cells - this.clearCellUsedStates(); + // reset this level model's explicitly defined properties + this.bestTimeProperty.reset(); + this.currentScoreProperty.reset(); + this.displayScoreProperty.reset(); - // reset game timer - this.gameTimer.stop(); - this.gameTimer.elapsedTimeProperty.value = 0; - }, + // reset the states of the cells + this.clearCellUsedStates(); - // @private - clear the usage state for all cells - clearCellUsedStates: function() { - // done as c-style loops for optimum performance - for ( let i = 0; i < this.tableSize; i++ ) { - for ( let j = 0; j < this.tableSize; j++ ) { - this.cellUsedStates[ i ][ j ] = false; - } + // reset game timer + this.gameTimer.stop(); + this.gameTimer.elapsedTimeProperty.value = 0; + }, + + // @private - clear the usage state for all cells + clearCellUsedStates: function() { + // done as c-style loops for optimum performance + for ( let i = 0; i < this.tableSize; i++ ) { + for ( let j = 0; j < this.tableSize; j++ ) { + this.cellUsedStates[ i ][ j ] = false; } - }, - - // @public - mark the cell associated with the provided multiplicand and multiplier as used - markCellAsUsed: function( multiplicand, multiplier ) { - this.cellUsedStates[ multiplicand - 1 ][ multiplier - 1 ] = true; - }, - - // @public - get the usage state for the requested cell - isCellUsed: function( multiplicand, multiplier ) { - return this.cellUsedStates[ multiplicand - 1 ][ multiplier - 1 ]; - }, - - // @public - chose a multiplicand-multiplier pair randomly from those that are available - selectUnusedMultiplierPair: function() { - const availableMultiplicands = []; - const availableMultipliers = []; - - // find available multiplicand rows with at least one unused cell - this.cellUsedStates.forEach( function( multipliers, index ) { - if ( multipliers.indexOf( false ) !== -1 ) { - availableMultiplicands.push( index + 1 ); - } - } ); - - // no more available multipliers - if ( !availableMultiplicands.length ) { - return null; + } + }, + + // @public - mark the cell associated with the provided multiplicand and multiplier as used + markCellAsUsed: function( multiplicand, multiplier ) { + this.cellUsedStates[ multiplicand - 1 ][ multiplier - 1 ] = true; + }, + + // @public - get the usage state for the requested cell + isCellUsed: function( multiplicand, multiplier ) { + return this.cellUsedStates[ multiplicand - 1 ][ multiplier - 1 ]; + }, + + // @public - chose a multiplicand-multiplier pair randomly from those that are available + selectUnusedMultiplierPair: function() { + const availableMultiplicands = []; + const availableMultipliers = []; + + // find available multiplicand rows with at least one unused cell + this.cellUsedStates.forEach( function( multipliers, index ) { + if ( multipliers.indexOf( false ) !== -1 ) { + availableMultiplicands.push( index + 1 ); } + } ); + + // no more available multipliers + if ( !availableMultiplicands.length ) { + return null; + } - // set multiplicand - const multiplicand = phet.joist.random.shuffle( availableMultiplicands )[ 0 ]; + // set multiplicand + const multiplicand = phet.joist.random.shuffle( availableMultiplicands )[ 0 ]; - // find available multipliers - this.cellUsedStates[ multiplicand - 1 ].forEach( function( isProblemAnswered, index ) { - if ( !isProblemAnswered ) { - availableMultipliers.push( index + 1 ); - } - } ); + // find available multipliers + this.cellUsedStates[ multiplicand - 1 ].forEach( function( isProblemAnswered, index ) { + if ( !isProblemAnswered ) { + availableMultipliers.push( index + 1 ); + } + } ); - // set multiplier - const multiplier = phet.joist.random.sample( availableMultipliers ); + // set multiplier + const multiplier = phet.joist.random.sample( availableMultipliers ); - return { - multiplicand: multiplicand, - multiplier: multiplier - }; - } - } ); -} ); + return { + multiplicand: multiplicand, + multiplier: multiplier + }; + } +} ); \ No newline at end of file diff --git a/js/common/model/ProblemModel.js b/js/common/model/ProblemModel.js index 67353623..a7cb4934 100644 --- a/js/common/model/ProblemModel.js +++ b/js/common/model/ProblemModel.js @@ -6,36 +6,32 @@ * * @author Andrey Zelenkov (MLearner) */ -define( require => { - 'use strict'; - // modules - const arithmetic = require( 'ARITHMETIC/arithmetic' ); - const inherit = require( 'PHET_CORE/inherit' ); - const Property = require( 'AXON/Property' ); +import Property from '../../../../axon/js/Property.js'; +import inherit from '../../../../phet-core/js/inherit.js'; +import arithmetic from '../../arithmetic.js'; - /** - * @constructor - */ - function ProblemModel() { +/** + * @constructor + */ +function ProblemModel() { - // @public - model properties, initialized to undefined, but only ever set to {number} - this.multiplicandProperty = new Property( undefined ); - this.multiplierProperty = new Property( undefined ); - this.productProperty = new Property( undefined ); // product of multiplication - this.possiblePointsProperty = new Property( 1 ); // points for correct completion of current task, can go down on incorrect answers - } + // @public - model properties, initialized to undefined, but only ever set to {number} + this.multiplicandProperty = new Property( undefined ); + this.multiplierProperty = new Property( undefined ); + this.productProperty = new Property( undefined ); // product of multiplication + this.possiblePointsProperty = new Property( 1 ); // points for correct completion of current task, can go down on incorrect answers +} - arithmetic.register( 'ProblemModel', ProblemModel ); +arithmetic.register( 'ProblemModel', ProblemModel ); - return inherit( Object, ProblemModel, { +export default inherit( Object, ProblemModel, { - // @public - reset: function() { - this.multiplicandProperty.reset(); - this.multiplierProperty.reset(); - this.productProperty.reset(); - this.possiblePointsProperty.reset(); - } - } ); + // @public + reset: function() { + this.multiplicandProperty.reset(); + this.multiplierProperty.reset(); + this.productProperty.reset(); + this.possiblePointsProperty.reset(); + } } ); \ No newline at end of file diff --git a/js/common/view/ArithmeticFaceWithPointsNode.js b/js/common/view/ArithmeticFaceWithPointsNode.js index 62e53a54..d092c6ef 100644 --- a/js/common/view/ArithmeticFaceWithPointsNode.js +++ b/js/common/view/ArithmeticFaceWithPointsNode.js @@ -7,97 +7,94 @@ * @author Andrey Zelenkov (MLearner) * @author John Blanco */ -define( require => { - 'use strict'; - - // modules - const arithmetic = require( 'ARITHMETIC/arithmetic' ); - const FaceWithPointsNode = require( 'SCENERY_PHET/FaceWithPointsNode' ); - const inherit = require( 'PHET_CORE/inherit' ); - const merge = require( 'PHET_CORE/merge' ); - const PhetFont = require( 'SCENERY_PHET/PhetFont' ); - const timer = require( 'AXON/timer' ); - const Utils = require( 'DOT/Utils' ); - - // constants - const UPDATE_PERIOD = 1 / 60 * 1000; // milliseconds, intended to match the expected frame rate - const OPAQUE_TIME = 1000; // milliseconds - const FADE_TIME = 1000; // milliseconds - - /** - * @param {Object} faceModel model for smile face. - * @param {Object} [options] for face node. - * - * @constructor - */ - function ArithmeticFaceWithPointsNode( faceModel, options ) { - const self = this; - - FaceWithPointsNode.call( this, merge( { - pointsFont: new PhetFont( { size: 26, weight: 'bold' } ), - visible: false // Initially invisible, must receive a showFace event to become visible. - }, options ) ); - - // set score of smile face - faceModel.pointsToDisplayProperty.link( function( points ) { - self.setPoints( points ); - } ); - - // set the facial expression - faceModel.isSmileProperty.link( function( isFaceSmile ) { - if ( isFaceSmile ) { - self.smile(); - } - else { - self.frown(); - } - } ); - - // Timer for fading the face. - let timerID = null; - // Handle the event that indicates that the face should be shown. - faceModel.showFaceEmitter.addListener( function() { +import timer from '../../../../axon/js/timer.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 FaceWithPointsNode from '../../../../scenery-phet/js/FaceWithPointsNode.js'; +import PhetFont from '../../../../scenery-phet/js/PhetFont.js'; +import arithmetic from '../../arithmetic.js'; - // make face fully visible - self.visible = true; - self.opacity = 1; +// constants +const UPDATE_PERIOD = 1 / 60 * 1000; // milliseconds, intended to match the expected frame rate +const OPAQUE_TIME = 1000; // milliseconds +const FADE_TIME = 1000; // milliseconds - // Set the countdown to the total for the opaque time and the fade time. - let countdown = OPAQUE_TIME + FADE_TIME; - - // cancel previous timer if it exists - if ( timerID !== null ) { +/** + * @param {Object} faceModel model for smile face. + * @param {Object} [options] for face node. + * + * @constructor + */ +function ArithmeticFaceWithPointsNode( faceModel, options ) { + const self = this; + + FaceWithPointsNode.call( this, merge( { + pointsFont: new PhetFont( { size: 26, weight: 'bold' } ), + visible: false // Initially invisible, must receive a showFace event to become visible. + }, options ) ); + + // set score of smile face + faceModel.pointsToDisplayProperty.link( function( points ) { + self.setPoints( points ); + } ); + + // set the facial expression + faceModel.isSmileProperty.link( function( isFaceSmile ) { + if ( isFaceSmile ) { + self.smile(); + } + else { + self.frown(); + } + } ); + + // Timer for fading the face. + let timerID = null; + + // Handle the event that indicates that the face should be shown. + faceModel.showFaceEmitter.addListener( function() { + + // make face fully visible + self.visible = true; + self.opacity = 1; + + // Set the countdown to the total for the opaque time and the fade time. + let countdown = OPAQUE_TIME + FADE_TIME; + + // cancel previous timer if it exists + if ( timerID !== null ) { + timer.clearInterval( timerID ); + } + + // start up the new timer + timerID = timer.setInterval( function() { + countdown -= UPDATE_PERIOD; + self.opacity = Utils.clamp( countdown / FADE_TIME, 0, 1 ); + if ( self.opacity === 0 ) { timer.clearInterval( timerID ); - } - - // start up the new timer - timerID = timer.setInterval( function() { - countdown -= UPDATE_PERIOD; - self.opacity = Utils.clamp( countdown / FADE_TIME, 0, 1 ); - if ( self.opacity === 0 ) { - timer.clearInterval( timerID ); - timerID = null; - self.visible = false; - } - }, UPDATE_PERIOD ); - } ); - - // Handle the event that indicates that the face should be hidden. - faceModel.hideFaceEmitter.addListener( function() { - - // Cancel the timer (if running) - if ( timerID !== null ) { - timer.clearTimeout( timerID ); timerID = null; + self.visible = false; } + }, UPDATE_PERIOD ); + } ); + + // Handle the event that indicates that the face should be hidden. + faceModel.hideFaceEmitter.addListener( function() { + + // Cancel the timer (if running) + if ( timerID !== null ) { + timer.clearTimeout( timerID ); + timerID = null; + } - // Go completely invisible. - self.visible = false; - } ); - } + // Go completely invisible. + self.visible = false; + } ); +} - arithmetic.register( 'ArithmeticFaceWithPointsNode', ArithmeticFaceWithPointsNode ); +arithmetic.register( 'ArithmeticFaceWithPointsNode', ArithmeticFaceWithPointsNode ); - return inherit( FaceWithPointsNode, ArithmeticFaceWithPointsNode ); -} ); \ No newline at end of file +inherit( FaceWithPointsNode, ArithmeticFaceWithPointsNode ); +export default ArithmeticFaceWithPointsNode; \ No newline at end of file diff --git a/js/common/view/ArithmeticView.js b/js/common/view/ArithmeticView.js index d53a0e75..e3a34dc2 100644 --- a/js/common/view/ArithmeticView.js +++ b/js/common/view/ArithmeticView.js @@ -6,207 +6,204 @@ * @author Andrey Zelenkov (MLearner) * @author John Blanco */ -define( require => { - 'use strict'; - - // modules - const Animation = require( 'TWIXT/Animation' ); - const arithmetic = require( 'ARITHMETIC/arithmetic' ); - const Bounds2 = require( 'DOT/Bounds2' ); - const Easing = require( 'TWIXT/Easing' ); - const GameAudioPlayer = require( 'VEGAS/GameAudioPlayer' ); - const GameState = require( 'ARITHMETIC/common/model/GameState' ); - const inherit = require( 'PHET_CORE/inherit' ); - const LevelSelectionNode = require( 'ARITHMETIC/common/view/LevelSelectionNode' ); - const merge = require( 'PHET_CORE/merge' ); - const ScreenView = require( 'JOIST/ScreenView' ); - const WorkspaceNode = require( 'ARITHMETIC/common/view/WorkspaceNode' ); - - // constants - const SCREEN_CHANGE_TIME = 0.75; // seconds - - /** - * @param {ArithmeticModel} model - Main model for screen. - * @param {Node} multiplicationTableNode - Multiplication table node for given screen. - * @param {Node} equationNode - Equation node for given screen. - * @param {Object} [options] - Configuration and position options, see usage in code for details. - * @constructor - */ - function ArithmeticView( model, multiplicationTableNode, equationNode, options ) { - const self = this; - ScreenView.call( this, { layoutBounds: new Bounds2( 0, 0, 768, 504 ) } ); - - // defaults - options = merge( { - titleString: '', - showKeypad: true, - levelSelectButtonColor: 'white', - levelSelectIconSet: 'multiply' - }, options ); - - // create and add the node that allows the user to select the game level - const levelSelectionNode = new LevelSelectionNode( - model, - options.titleString, - function( level ) { model.setLevel( level ); }, - this.layoutBounds, - { - centerX: this.layoutBounds.centerX, - centerY: this.layoutBounds.centerY, - buttonBaseColor: options.levelSelectButtonColor, - iconSet: options.levelSelectIconSet - } - ); - this.addChild( levelSelectionNode ); - - // add the game components - const workspaceNode = new WorkspaceNode( - model, - multiplicationTableNode, - equationNode, - this.layoutBounds, - { showKeypad: options.showKeypad, scoreboardTitle: options.titleString } - ); - workspaceNode.left = this.layoutBounds.maxX; - workspaceNode.visible = false; - this.addChild( workspaceNode ); - - // sounds player that is used to produce the feedback sounds for the game - const gameAudioPlayer = new GameAudioPlayer(); - - // set the origin of the answer animation in the multiplication table, which depends upon the newly set position of - // the equation node. - multiplicationTableNode.animationOrigin = equationNode.productInput.center; - - // create the animations that will slide the level selection screen and the workspaces in and out - const levelSelectionScreenInAnimator = new Animation( { - duration: SCREEN_CHANGE_TIME, - easing: Easing.CUBIC_IN_OUT, - getValue: function() { - return levelSelectionNode.x; - }, - setValue: function( newXPosition ) { - levelSelectionNode.x = newXPosition; - }, - to: self.layoutBounds.minX - } ); - levelSelectionScreenInAnimator.beginEmitter.addListener( function() { - levelSelectionNode.visible = true; - levelSelectionNode.pickable = false; // prevent interaction during animation - } ); - levelSelectionScreenInAnimator.finishEmitter.addListener( function() { - levelSelectionNode.pickable = true; - } ); - - const levelSelectionScreenOutAnimator = new Animation( { - duration: SCREEN_CHANGE_TIME, - easing: Easing.CUBIC_IN_OUT, - getValue: function() { - return levelSelectionNode.x; - }, - setValue: function( newXPosition ) { - levelSelectionNode.x = newXPosition; - }, - to: self.layoutBounds.minX - levelSelectionNode.width - } ); - levelSelectionScreenOutAnimator.beginEmitter.addListener( function() { - levelSelectionNode.pickable = false; // prevent interaction during animation - } ); - levelSelectionScreenOutAnimator.finishEmitter.addListener( function() { - levelSelectionNode.visible = false; - } ); - - const workspaceNodeInAnimator = new Animation( { - duration: SCREEN_CHANGE_TIME, - easing: Easing.CUBIC_IN_OUT, - getValue: function() { - return workspaceNode.x; - }, - setValue: function( newXPosition ) { - workspaceNode.x = newXPosition; - }, - to: self.layoutBounds.minX - } ); - workspaceNodeInAnimator.beginEmitter.addListener( function() { - workspaceNode.visible = true; - workspaceNode.pickable = false; // prevent interaction during animation - } ); - workspaceNodeInAnimator.finishEmitter.addListener( function() { - workspaceNode.pickable = true; - } ); - - const workspaceNodeOutAnimator = new Animation( { - duration: SCREEN_CHANGE_TIME, - easing: Easing.CUBIC_IN_OUT, - getValue: function() { - return workspaceNode.x; - }, - setValue: function( newXPosition ) { - workspaceNode.x = newXPosition; - }, - to: self.layoutBounds.maxX - } ); - workspaceNodeOutAnimator.beginEmitter.addListener( function() { - workspaceNode.pickable = false; // prevent interaction during animation - } ); - workspaceNodeOutAnimator.finishEmitter.addListener( function() { - workspaceNode.visible = false; - } ); - - // monitor the game state and update the view and changes occur - model.stateProperty.link( function( newState, oldState ) { - - // animate the transition between the level select screen and the selected level - if ( newState === GameState.SELECTING_LEVEL && oldState ) { - - // Slide out the workspace node - workspaceNodeInAnimator.stop(); - workspaceNodeOutAnimator.start(); - - // Slide in the level selection screen - levelSelectionScreenOutAnimator.stop(); - levelSelectionScreenInAnimator.start(); - } - else if ( newState !== GameState.SELECTING_LEVEL && oldState === GameState.SELECTING_LEVEL ) { - - // Slide in the workspace node - workspaceNodeOutAnimator.stop(); - workspaceNodeInAnimator.start(); - // Slide out the level selection screen - levelSelectionScreenInAnimator.stop(); - levelSelectionScreenOutAnimator.start(); - // levelSelectionScreenAnimatorOld.stop().to( { x: self.layoutBounds.minX - levelSelectionNode.width }, ANIMATION_TIME ).start( phet.joist.elapsedTime ); - } +import Bounds2 from '../../../../dot/js/Bounds2.js'; +import ScreenView from '../../../../joist/js/ScreenView.js'; +import inherit from '../../../../phet-core/js/inherit.js'; +import merge from '../../../../phet-core/js/merge.js'; +import Animation from '../../../../twixt/js/Animation.js'; +import Easing from '../../../../twixt/js/Easing.js'; +import GameAudioPlayer from '../../../../vegas/js/GameAudioPlayer.js'; +import arithmetic from '../../arithmetic.js'; +import GameState from '../model/GameState.js'; +import LevelSelectionNode from './LevelSelectionNode.js'; +import WorkspaceNode from './WorkspaceNode.js'; + +// constants +const SCREEN_CHANGE_TIME = 0.75; // seconds - // play the appropriate audio, if any, for this state transition - if ( ( oldState === GameState.AWAITING_USER_INPUT || oldState === GameState.DISPLAYING_INCORRECT_ANSWER_FEEDBACK ) - && newState === GameState.DISPLAYING_CORRECT_ANSWER_FEEDBACK ) { - // play the correct answer sound - gameAudioPlayer.correctAnswer(); +/** + * @param {ArithmeticModel} model - Main model for screen. + * @param {Node} multiplicationTableNode - Multiplication table node for given screen. + * @param {Node} equationNode - Equation node for given screen. + * @param {Object} [options] - Configuration and position options, see usage in code for details. + * @constructor + */ +function ArithmeticView( model, multiplicationTableNode, equationNode, options ) { + const self = this; + ScreenView.call( this, { layoutBounds: new Bounds2( 0, 0, 768, 504 ) } ); + + // defaults + options = merge( { + titleString: '', + showKeypad: true, + levelSelectButtonColor: 'white', + levelSelectIconSet: 'multiply' + }, options ); + + // create and add the node that allows the user to select the game level + const levelSelectionNode = new LevelSelectionNode( + model, + options.titleString, + function( level ) { model.setLevel( level ); }, + this.layoutBounds, + { + centerX: this.layoutBounds.centerX, + centerY: this.layoutBounds.centerY, + buttonBaseColor: options.levelSelectButtonColor, + iconSet: options.levelSelectIconSet + } + ); + this.addChild( levelSelectionNode ); + + // add the game components + const workspaceNode = new WorkspaceNode( + model, + multiplicationTableNode, + equationNode, + this.layoutBounds, + { showKeypad: options.showKeypad, scoreboardTitle: options.titleString } + ); + workspaceNode.left = this.layoutBounds.maxX; + workspaceNode.visible = false; + this.addChild( workspaceNode ); + + // sounds player that is used to produce the feedback sounds for the game + const gameAudioPlayer = new GameAudioPlayer(); + + // set the origin of the answer animation in the multiplication table, which depends upon the newly set position of + // the equation node. + multiplicationTableNode.animationOrigin = equationNode.productInput.center; + + // create the animations that will slide the level selection screen and the workspaces in and out + const levelSelectionScreenInAnimator = new Animation( { + duration: SCREEN_CHANGE_TIME, + easing: Easing.CUBIC_IN_OUT, + getValue: function() { + return levelSelectionNode.x; + }, + setValue: function( newXPosition ) { + levelSelectionNode.x = newXPosition; + }, + to: self.layoutBounds.minX + } ); + levelSelectionScreenInAnimator.beginEmitter.addListener( function() { + levelSelectionNode.visible = true; + levelSelectionNode.pickable = false; // prevent interaction during animation + } ); + levelSelectionScreenInAnimator.finishEmitter.addListener( function() { + levelSelectionNode.pickable = true; + } ); + + const levelSelectionScreenOutAnimator = new Animation( { + duration: SCREEN_CHANGE_TIME, + easing: Easing.CUBIC_IN_OUT, + getValue: function() { + return levelSelectionNode.x; + }, + setValue: function( newXPosition ) { + levelSelectionNode.x = newXPosition; + }, + to: self.layoutBounds.minX - levelSelectionNode.width + } ); + levelSelectionScreenOutAnimator.beginEmitter.addListener( function() { + levelSelectionNode.pickable = false; // prevent interaction during animation + } ); + levelSelectionScreenOutAnimator.finishEmitter.addListener( function() { + levelSelectionNode.visible = false; + } ); + + const workspaceNodeInAnimator = new Animation( { + duration: SCREEN_CHANGE_TIME, + easing: Easing.CUBIC_IN_OUT, + getValue: function() { + return workspaceNode.x; + }, + setValue: function( newXPosition ) { + workspaceNode.x = newXPosition; + }, + to: self.layoutBounds.minX + } ); + workspaceNodeInAnimator.beginEmitter.addListener( function() { + workspaceNode.visible = true; + workspaceNode.pickable = false; // prevent interaction during animation + } ); + workspaceNodeInAnimator.finishEmitter.addListener( function() { + workspaceNode.pickable = true; + } ); + + const workspaceNodeOutAnimator = new Animation( { + duration: SCREEN_CHANGE_TIME, + easing: Easing.CUBIC_IN_OUT, + getValue: function() { + return workspaceNode.x; + }, + setValue: function( newXPosition ) { + workspaceNode.x = newXPosition; + }, + to: self.layoutBounds.maxX + } ); + workspaceNodeOutAnimator.beginEmitter.addListener( function() { + workspaceNode.pickable = false; // prevent interaction during animation + } ); + workspaceNodeOutAnimator.finishEmitter.addListener( function() { + workspaceNode.visible = false; + } ); + + // monitor the game state and update the view and changes occur + model.stateProperty.link( function( newState, oldState ) { + + // animate the transition between the level select screen and the selected level + if ( newState === GameState.SELECTING_LEVEL && oldState ) { + + // Slide out the workspace node + workspaceNodeInAnimator.stop(); + workspaceNodeOutAnimator.start(); + + // Slide in the level selection screen + levelSelectionScreenOutAnimator.stop(); + levelSelectionScreenInAnimator.start(); + } + else if ( newState !== GameState.SELECTING_LEVEL && oldState === GameState.SELECTING_LEVEL ) { + + // Slide in the workspace node + workspaceNodeOutAnimator.stop(); + workspaceNodeInAnimator.start(); + + // Slide out the level selection screen + levelSelectionScreenInAnimator.stop(); + levelSelectionScreenOutAnimator.start(); + // levelSelectionScreenAnimatorOld.stop().to( { x: self.layoutBounds.minX - levelSelectionNode.width }, ANIMATION_TIME ).start( phet.joist.elapsedTime ); + } + + // play the appropriate audio, if any, for this state transition + if ( ( oldState === GameState.AWAITING_USER_INPUT || oldState === GameState.DISPLAYING_INCORRECT_ANSWER_FEEDBACK ) + && newState === GameState.DISPLAYING_CORRECT_ANSWER_FEEDBACK ) { + // play the correct answer sound + gameAudioPlayer.correctAnswer(); + } + else if ( oldState === GameState.AWAITING_USER_INPUT && newState === GameState.DISPLAYING_INCORRECT_ANSWER_FEEDBACK ) { + // play the incorrect answer sound + gameAudioPlayer.wrongAnswer(); + } + else if ( oldState === GameState.DISPLAYING_CORRECT_ANSWER_FEEDBACK && newState === GameState.SHOWING_LEVEL_COMPLETED_DIALOG ) { + const resultScore = model.activeLevelModel.currentScoreProperty.get(); + const perfectScore = model.activeLevelModel.perfectScore; + + if ( resultScore === perfectScore ) { + gameAudioPlayer.gameOverPerfectScore(); } - else if ( oldState === GameState.AWAITING_USER_INPUT && newState === GameState.DISPLAYING_INCORRECT_ANSWER_FEEDBACK ) { - // play the incorrect answer sound - gameAudioPlayer.wrongAnswer(); + else if ( resultScore === 0 ) { + gameAudioPlayer.gameOverZeroScore(); } - else if ( oldState === GameState.DISPLAYING_CORRECT_ANSWER_FEEDBACK && newState === GameState.SHOWING_LEVEL_COMPLETED_DIALOG ) { - const resultScore = model.activeLevelModel.currentScoreProperty.get(); - const perfectScore = model.activeLevelModel.perfectScore; - - if ( resultScore === perfectScore ) { - gameAudioPlayer.gameOverPerfectScore(); - } - else if ( resultScore === 0 ) { - gameAudioPlayer.gameOverZeroScore(); - } - else { - gameAudioPlayer.gameOverImperfectScore(); - } + else { + gameAudioPlayer.gameOverImperfectScore(); } - } ); - } + } + } ); +} - arithmetic.register( 'ArithmeticView', ArithmeticView ); +arithmetic.register( 'ArithmeticView', ArithmeticView ); - return inherit( ScreenView, ArithmeticView ); -} ); +inherit( ScreenView, ArithmeticView ); +export default ArithmeticView; \ No newline at end of file diff --git a/js/common/view/EquationInputNode.js b/js/common/view/EquationInputNode.js index 103625a9..abacaacc 100644 --- a/js/common/view/EquationInputNode.js +++ b/js/common/view/EquationInputNode.js @@ -7,124 +7,120 @@ * * @author Andrey Zelenkov (MLearner) */ -define( require => { - 'use strict'; - - // modules - const arithmetic = require( 'ARITHMETIC/arithmetic' ); - const ArithmeticConstants = require( 'ARITHMETIC/common/ArithmeticConstants' ); - const HBox = require( 'SCENERY/nodes/HBox' ); - const inherit = require( 'PHET_CORE/inherit' ); - const Node = require( 'SCENERY/nodes/Node' ); - const Rectangle = require( 'SCENERY/nodes/Rectangle' ); - const Text = require( 'SCENERY/nodes/Text' ); - const timer = require( 'AXON/timer' ); - - // strings - const unknownValueIndicatorString = require( 'string!ARITHMETIC/unknownValueIndicator' ); - - // constants - const INTERACTIVE_FILL = 'white'; - const NON_INTERACTIVE_FILL = '#dddddd'; - const MIN_X_MARGIN = 5; - const CURSOR_HEIGHT = new Text( '8', { font: ArithmeticConstants.EQUATION_FONT_TEXT } ).height * 0.7; - - // convenience function to avoid duplicated code - const updateBoxPosition = function( box, inputSize ) { - box.centerX = inputSize.width / 2; - box.centerY = inputSize.height / 2; - }; + +import timer from '../../../../axon/js/timer.js'; +import inherit from '../../../../phet-core/js/inherit.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 Text from '../../../../scenery/js/nodes/Text.js'; +import arithmeticStrings from '../../arithmetic-strings.js'; +import arithmetic from '../../arithmetic.js'; +import ArithmeticConstants from '../ArithmeticConstants.js'; + +const unknownValueIndicatorString = arithmeticStrings.unknownValueIndicator; + +// constants +const INTERACTIVE_FILL = 'white'; +const NON_INTERACTIVE_FILL = '#dddddd'; +const MIN_X_MARGIN = 5; +const CURSOR_HEIGHT = new Text( '8', { font: ArithmeticConstants.EQUATION_FONT_TEXT } ).height * 0.7; + +// convenience function to avoid duplicated code +const updateBoxPosition = function( box, inputSize ) { + box.centerX = inputSize.width / 2; + box.centerY = inputSize.height / 2; +}; + +/** + * @param {Property} valueProperty for observing and changing by input + * @param {Dimension2} size - Dimensions of this input component. + * + * @constructor + */ +function EquationInputNode( valueProperty, size ) { + const self = this; + Node.call( this ); + + // @private - create text and save reference for use in public methods + this.inputText = new Text( unknownValueIndicatorString, { + font: ArithmeticConstants.EQUATION_FONT_TEXT, + maxWidth: size.width - 2 * MIN_X_MARGIN + } ); + + // @private - create cursor and save reference for use in public methods + this.textCursor = new Rectangle( 0, 2, 1, CURSOR_HEIGHT, { fill: 'black' } ); + this.cursorContainer = new Node( { children: [ this.textCursor ] } ); + + // @private - save reference to input size value for use in public methods + this.inputSize = size; + + // update text when the value changes + valueProperty.lazyLink( function( value ) { + self.inputText.setText( isNaN( value ) ? '' : value ); + updateBoxPosition( self._box, size ); + } ); + + // set up blinking of cursor + timer.setInterval( function() { + self.textCursor.visible = !self.textCursor.visible; + }, ArithmeticConstants.CURSOR_BLINK_INTERVAL ); + + // @private - background of this input box + this.background = new Rectangle( 0, 0, size.width, size.height, 5, 5, { fill: NON_INTERACTIVE_FILL } ); + this.addChild( this.background ); + + // @private - horizontal box containing the input text and the cursor + this._box = new HBox( { + children: [ this.inputText, this.cursorContainer ], + centerX: size.width / 2, + centerY: size.height / 2 + } ); + this.addChild( this._box ); + + // unfocused state by default + this.setFocus( false ); +} + +arithmetic.register( 'EquationInputNode', EquationInputNode ); + +export default inherit( Node, EquationInputNode, { /** - * @param {Property} valueProperty for observing and changing by input - * @param {Dimension2} size - Dimensions of this input component. - * - * @constructor + * Clear the textual value shown in this node. This is done regardless of the value of the value property that is + * being monitored by this node. + * @public */ - function EquationInputNode( valueProperty, size ) { - const self = this; - Node.call( this ); - - // @private - create text and save reference for use in public methods - this.inputText = new Text( unknownValueIndicatorString, { - font: ArithmeticConstants.EQUATION_FONT_TEXT, - maxWidth: size.width - 2 * MIN_X_MARGIN - } ); - - // @private - create cursor and save reference for use in public methods - this.textCursor = new Rectangle( 0, 2, 1, CURSOR_HEIGHT, { fill: 'black' } ); - this.cursorContainer = new Node( { children: [ this.textCursor ] } ); - - // @private - save reference to input size value for use in public methods - this.inputSize = size; - - // update text when the value changes - valueProperty.lazyLink( function( value ) { - self.inputText.setText( isNaN( value ) ? '' : value ); - updateBoxPosition( self._box, size ); - } ); - - // set up blinking of cursor - timer.setInterval( function() { - self.textCursor.visible = !self.textCursor.visible; - }, ArithmeticConstants.CURSOR_BLINK_INTERVAL ); - - // @private - background of this input box - this.background = new Rectangle( 0, 0, size.width, size.height, 5, 5, { fill: NON_INTERACTIVE_FILL } ); - this.addChild( this.background ); - - // @private - horizontal box containing the input text and the cursor - this._box = new HBox( { - children: [ this.inputText, this.cursorContainer ], - centerX: size.width / 2, - centerY: size.height / 2 - } ); - this.addChild( this._box ); - - // unfocused state by default - this.setFocus( false ); - } + clear: function() { + this.inputText.setText( '' ); + updateBoxPosition( this._box, this.inputSize ); + }, - arithmetic.register( 'EquationInputNode', EquationInputNode ); - - return inherit( Node, EquationInputNode, { - - /** - * Clear the textual value shown in this node. This is done regardless of the value of the value property that is - * being monitored by this node. - * @public - */ - clear: function() { - this.inputText.setText( '' ); - updateBoxPosition( this._box, this.inputSize ); - }, - - /** - * Set or remove focus, which for this component simply turns the blinking cursor on or off. - * @param {boolean} focus - * @public - */ - setFocus: function( focus ) { - this.cursorContainer.visible = focus; - }, - - /** - * Set the appearance of this node to indicate to the user that it is interactive, meaning that their actions are - * going to change its value. - * @param {boolean} interactive - * @public - */ - setInteractiveAppearance: function( interactive ) { - this.background.fill = interactive ? INTERACTIVE_FILL : NON_INTERACTIVE_FILL; - }, - - /** - * Set the textual value of this node to a 'placeholder' value (a question mark at the time of this writing). - * @public - */ - setPlaceholder: function() { - this.inputText.setText( unknownValueIndicatorString ); - updateBoxPosition( this._box, this.inputSize ); - } - } ); -} ); + /** + * Set or remove focus, which for this component simply turns the blinking cursor on or off. + * @param {boolean} focus + * @public + */ + setFocus: function( focus ) { + this.cursorContainer.visible = focus; + }, + + /** + * Set the appearance of this node to indicate to the user that it is interactive, meaning that their actions are + * going to change its value. + * @param {boolean} interactive + * @public + */ + setInteractiveAppearance: function( interactive ) { + this.background.fill = interactive ? INTERACTIVE_FILL : NON_INTERACTIVE_FILL; + }, + + /** + * Set the textual value of this node to a 'placeholder' value (a question mark at the time of this writing). + * @public + */ + setPlaceholder: function() { + this.inputText.setText( unknownValueIndicatorString ); + updateBoxPosition( this._box, this.inputSize ); + } +} ); \ No newline at end of file diff --git a/js/common/view/EquationNode.js b/js/common/view/EquationNode.js index 553d0163..7a9663e7 100644 --- a/js/common/view/EquationNode.js +++ b/js/common/view/EquationNode.js @@ -6,76 +6,72 @@ * * @author Andrey Zelenkov (MLearner) */ -define( require => { - 'use strict'; - // modules - const arithmetic = require( 'ARITHMETIC/arithmetic' ); - const Dimension2 = require( 'DOT/Dimension2' ); - const EquationInputNode = require( 'ARITHMETIC/common/view/EquationInputNode' ); - const HBox = require( 'SCENERY/nodes/HBox' ); - const inherit = require( 'PHET_CORE/inherit' ); - const MathSymbols = require( 'SCENERY_PHET/MathSymbols' ); - const merge = require( 'PHET_CORE/merge' ); - const PhetFont = require( 'SCENERY_PHET/PhetFont' ); - const Text = require( 'SCENERY/nodes/Text' ); +import Dimension2 from '../../../../dot/js/Dimension2.js'; +import inherit from '../../../../phet-core/js/inherit.js'; +import merge from '../../../../phet-core/js/merge.js'; +import MathSymbols from '../../../../scenery-phet/js/MathSymbols.js'; +import PhetFont from '../../../../scenery-phet/js/PhetFont.js'; +import HBox from '../../../../scenery/js/nodes/HBox.js'; +import Text from '../../../../scenery/js/nodes/Text.js'; +import arithmetic from '../../arithmetic.js'; +import EquationInputNode from './EquationInputNode.js'; - // constants - const FONT_EQUALS = new PhetFont( 45 ); - const FONT_X = new PhetFont( 50 ); - const INPUT_SIZE_MULTIPLIER = new Dimension2( 70, 45 ); // size of input boxes for multiplier - const INPUT_SIZE_PRODUCT = new Dimension2( 80, 45 ); // size of input box for product - const SPACING = 20; // spacing between equation elements - const SYMBOL_COLOR = '#FFFF00'; +// constants +const FONT_EQUALS = new PhetFont( 45 ); +const FONT_X = new PhetFont( 50 ); +const INPUT_SIZE_MULTIPLIER = new Dimension2( 70, 45 ); // size of input boxes for multiplier +const INPUT_SIZE_PRODUCT = new Dimension2( 80, 45 ); // size of input box for product +const SPACING = 20; // spacing between equation elements +const SYMBOL_COLOR = '#FFFF00'; - /** - * @param {Property.} multiplicandProperty - Property necessary for creating multiplicand input. - * @param {Property.} multiplierProperty - Property necessary for creating multiplier input. - * @param {Property.} productProperty - Property necessary for creating product input. - * @param {Property.} productProperty - Property necessary for creating product input. - * @param {Object} [options] - * @constructor - */ - function EquationNode( multiplicandProperty, multiplierProperty, productProperty, options ) { +/** + * @param {Property.} multiplicandProperty - Property necessary for creating multiplicand input. + * @param {Property.} multiplierProperty - Property necessary for creating multiplier input. + * @param {Property.} productProperty - Property necessary for creating product input. + * @param {Property.} productProperty - Property necessary for creating product input. + * @param {Object} [options] + * @constructor + */ +function EquationNode( multiplicandProperty, multiplierProperty, productProperty, options ) { - options = merge( { - spacing: SPACING, - resize: false - }, options ); + options = merge( { + spacing: SPACING, + resize: false + }, options ); - // @public - Set up the three nodes that depict the numbers in the equation. - this.multiplicandInput = new EquationInputNode( multiplicandProperty, INPUT_SIZE_MULTIPLIER ); - this.multiplierInput = new EquationInputNode( multiplierProperty, INPUT_SIZE_MULTIPLIER ); - this.productInput = new EquationInputNode( productProperty, INPUT_SIZE_PRODUCT ); + // @public - Set up the three nodes that depict the numbers in the equation. + this.multiplicandInput = new EquationInputNode( multiplicandProperty, INPUT_SIZE_MULTIPLIER ); + this.multiplierInput = new EquationInputNode( multiplierProperty, INPUT_SIZE_MULTIPLIER ); + this.productInput = new EquationInputNode( productProperty, INPUT_SIZE_PRODUCT ); - // @private - Set up the equals sign, which can potentially be changed to a not equals sign. - this.equalsSign = new Text( '', { font: FONT_EQUALS, fill: SYMBOL_COLOR } ); - this.setShowEqual( true ); // Default to equals equation until set otherwise. + // @private - Set up the equals sign, which can potentially be changed to a not equals sign. + this.equalsSign = new Text( '', { font: FONT_EQUALS, fill: SYMBOL_COLOR } ); + this.setShowEqual( true ); // Default to equals equation until set otherwise. - options.children = [ - this.multiplicandInput, - new Text( MathSymbols.TIMES, { font: FONT_X, fill: SYMBOL_COLOR } ), - this.multiplierInput, - this.equalsSign, - this.productInput - ]; + options.children = [ + this.multiplicandInput, + new Text( MathSymbols.TIMES, { font: FONT_X, fill: SYMBOL_COLOR } ), + this.multiplierInput, + this.equalsSign, + this.productInput + ]; - // Perform the layout by placing everything in an HBox. - HBox.call( this, options ); - } + // Perform the layout by placing everything in an HBox. + HBox.call( this, options ); +} - arithmetic.register( 'EquationNode', EquationNode ); +arithmetic.register( 'EquationNode', EquationNode ); - return inherit( HBox, EquationNode, { +export default inherit( HBox, EquationNode, { - /** - * Set the equation to depict equals or not equals. - * - * @param {boolean} showEqual - * @protected - */ - setShowEqual: function( showEqual ) { - this.equalsSign.text = showEqual ? MathSymbols.EQUAL_TO : MathSymbols.NOT_EQUAL_TO; - } - } ); -} ); + /** + * Set the equation to depict equals or not equals. + * + * @param {boolean} showEqual + * @protected + */ + setShowEqual: function( showEqual ) { + this.equalsSign.text = showEqual ? MathSymbols.EQUAL_TO : MathSymbols.NOT_EQUAL_TO; + } +} ); \ No newline at end of file diff --git a/js/common/view/LevelCompletedNodeWrapper.js b/js/common/view/LevelCompletedNodeWrapper.js index 096fc844..f4e4a0df 100644 --- a/js/common/view/LevelCompletedNodeWrapper.js +++ b/js/common/view/LevelCompletedNodeWrapper.js @@ -5,56 +5,53 @@ * * @author Andrey Zelenkov (MLearner) */ -define( require => { - 'use strict'; - // modules - const arithmetic = require( 'ARITHMETIC/arithmetic' ); - const ArithmeticConstants = require( 'ARITHMETIC/common/ArithmeticConstants' ); - const Color = require( 'SCENERY/util/Color' ); - const GameState = require( 'ARITHMETIC/common/model/GameState' ); - const inherit = require( 'PHET_CORE/inherit' ); - const LevelCompletedNode = require( 'VEGAS/LevelCompletedNode' ); - const Node = require( 'SCENERY/nodes/Node' ); +import inherit from '../../../../phet-core/js/inherit.js'; +import Node from '../../../../scenery/js/nodes/Node.js'; +import Color from '../../../../scenery/js/util/Color.js'; +import LevelCompletedNode from '../../../../vegas/js/LevelCompletedNode.js'; +import arithmetic from '../../arithmetic.js'; +import ArithmeticConstants from '../ArithmeticConstants.js'; +import GameState from '../model/GameState.js'; - /** - * @param {Array} levelModels - Array of descriptions for each level. Necessary to get perfect and current score for - * completed level. - * @param {Property} levelNumberProperty - Level property. - * @param {Property} stateProperty - Game state property. - * @param {Property} timerEnabledProperty - Timer enabled flag. - * @param {Function} continueCallback - Callback listener for continue button. - * @param {Bounds2} layoutBounds - Bounds of main screen. Necessary for placing components. - * @constructor - */ - function LevelCompletedNodeWrapper( levelModels, levelNumberProperty, stateProperty, timerEnabledProperty, continueCallback, layoutBounds ) { - const self = this; - Node.call( this ); +/** + * @param {Array} levelModels - Array of descriptions for each level. Necessary to get perfect and current score for + * completed level. + * @param {Property} levelNumberProperty - Level property. + * @param {Property} stateProperty - Game state property. + * @param {Property} timerEnabledProperty - Timer enabled flag. + * @param {Function} continueCallback - Callback listener for continue button. + * @param {Bounds2} layoutBounds - Bounds of main screen. Necessary for placing components. + * @constructor + */ +function LevelCompletedNodeWrapper( levelModels, levelNumberProperty, stateProperty, timerEnabledProperty, continueCallback, layoutBounds ) { + const self = this; + Node.call( this ); - // Show this node only when the level has been completed. - stateProperty.lazyLink( function( state ) { - if ( state === GameState.SHOWING_LEVEL_COMPLETED_DIALOG ) { - const levelModel = levelModels[ levelNumberProperty.value ]; - self.addChild( new LevelCompletedNode( - levelNumberProperty.value + 1, - levelModel.currentScoreProperty.get(), - levelModel.perfectScore, - ArithmeticConstants.NUM_STARS, - timerEnabledProperty.value, - levelModel.gameTimer.elapsedTimeProperty.value, - levelModel.bestTimeProperty.get(), - ( levelModel.gameTimer.elapsedTimeProperty.get() < levelModel.bestTimeProperty.get() ), - continueCallback, - { fill: new Color( 255, 235, 205 ), centerX: layoutBounds.maxX / 2, centerY: layoutBounds.maxY / 2 } - ) ); - } - else if ( state === GameState.AWAITING_USER_INPUT || state === GameState.LEVEL_COMPLETED ) { - self.removeAllChildren(); - } - } ); - } + // Show this node only when the level has been completed. + stateProperty.lazyLink( function( state ) { + if ( state === GameState.SHOWING_LEVEL_COMPLETED_DIALOG ) { + const levelModel = levelModels[ levelNumberProperty.value ]; + self.addChild( new LevelCompletedNode( + levelNumberProperty.value + 1, + levelModel.currentScoreProperty.get(), + levelModel.perfectScore, + ArithmeticConstants.NUM_STARS, + timerEnabledProperty.value, + levelModel.gameTimer.elapsedTimeProperty.value, + levelModel.bestTimeProperty.get(), + ( levelModel.gameTimer.elapsedTimeProperty.get() < levelModel.bestTimeProperty.get() ), + continueCallback, + { fill: new Color( 255, 235, 205 ), centerX: layoutBounds.maxX / 2, centerY: layoutBounds.maxY / 2 } + ) ); + } + else if ( state === GameState.AWAITING_USER_INPUT || state === GameState.LEVEL_COMPLETED ) { + self.removeAllChildren(); + } + } ); +} - arithmetic.register( 'LevelCompletedNodeWrapper', LevelCompletedNodeWrapper ); +arithmetic.register( 'LevelCompletedNodeWrapper', LevelCompletedNodeWrapper ); - return inherit( Node, LevelCompletedNodeWrapper ); -} ); \ No newline at end of file +inherit( Node, LevelCompletedNodeWrapper ); +export default LevelCompletedNodeWrapper; \ No newline at end of file diff --git a/js/common/view/LevelSelectionNode.js b/js/common/view/LevelSelectionNode.js index 7e186598..9a0b1403 100644 --- a/js/common/view/LevelSelectionNode.js +++ b/js/common/view/LevelSelectionNode.js @@ -6,147 +6,142 @@ * @author Andrey Zelenkov (MLearner) * @author John Blanco */ -define( require => { - 'use strict'; - // modules - const arithmetic = require( 'ARITHMETIC/arithmetic' ); - const ArithmeticConstants = require( 'ARITHMETIC/common/ArithmeticConstants' ); - const ArithmeticGlobals = require( 'ARITHMETIC/common/ArithmeticGlobals' ); - const HBox = require( 'SCENERY/nodes/HBox' ); - const Image = require( 'SCENERY/nodes/Image' ); - const inherit = require( 'PHET_CORE/inherit' ); - const LevelSelectionButton = require( 'VEGAS/LevelSelectionButton' ); - const merge = require( 'PHET_CORE/merge' ); - const Node = require( 'SCENERY/nodes/Node' ); - const PhetFont = require( 'SCENERY_PHET/PhetFont' ); - const ResetAllButton = require( 'SCENERY_PHET/buttons/ResetAllButton' ); - const Text = require( 'SCENERY/nodes/Text' ); - const TimerToggleButton = require( 'SCENERY_PHET/buttons/TimerToggleButton' ); - const VBox = require( 'SCENERY/nodes/VBox' ); +import inherit from '../../../../phet-core/js/inherit.js'; +import merge from '../../../../phet-core/js/merge.js'; +import ResetAllButton from '../../../../scenery-phet/js/buttons/ResetAllButton.js'; +import TimerToggleButton from '../../../../scenery-phet/js/buttons/TimerToggleButton.js'; +import PhetFont from '../../../../scenery-phet/js/PhetFont.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 Text from '../../../../scenery/js/nodes/Text.js'; +import VBox from '../../../../scenery/js/nodes/VBox.js'; +import LevelSelectionButton from '../../../../vegas/js/LevelSelectionButton.js'; +import divideLevel1Icon from '../../../mipmaps/divide_level_1_icon_png.js'; +import divideLevel2Icon from '../../../mipmaps/divide_level_2_icon_png.js'; +import divideLevel3Icon from '../../../mipmaps/divide_level_3_icon_png.js'; +import factorLevel1Icon from '../../../mipmaps/factor_level_1_icon_png.js'; +import factorLevel2Icon from '../../../mipmaps/factor_level_2_icon_png.js'; +import factorLevel3Icon from '../../../mipmaps/factor_level_3_icon_png.js'; +import multiplyLevel1Icon from '../../../mipmaps/multiply_level_1_icon_png.js'; +import multiplyLevel2Icon from '../../../mipmaps/multiply_level_2_icon_png.js'; +import multiplyLevel3Icon from '../../../mipmaps/multiply_level_3_icon_png.js'; +import arithmeticStrings from '../../arithmetic-strings.js'; +import arithmetic from '../../arithmetic.js'; +import ArithmeticConstants from '../ArithmeticConstants.js'; +import ArithmeticGlobals from '../ArithmeticGlobals.js'; + +// constants +const CHOOSE_LEVEL_TITLE_FONT = new PhetFont( { size: 24 } ); +const TAB_TITLE_FONT = new PhetFont( { size: 54 } ); +const BUTTON_LENGTH = 150; + +// icon sets, used to place on the buttons +const ICON_SETS = { + multiply: [ + multiplyLevel1Icon, + multiplyLevel2Icon, + multiplyLevel3Icon + ], + factor: [ + factorLevel1Icon, + factorLevel2Icon, + factorLevel3Icon + ], + divide: [ + divideLevel1Icon, + divideLevel2Icon, + divideLevel3Icon + ] +}; + +const chooseYourLevelString = arithmeticStrings.chooseYourLevel; - // images - const multiplyLevel1Icon = require( 'mipmap!ARITHMETIC/multiply_level_1_icon.png' ); - const multiplyLevel2Icon = require( 'mipmap!ARITHMETIC/multiply_level_2_icon.png' ); - const multiplyLevel3Icon = require( 'mipmap!ARITHMETIC/multiply_level_3_icon.png' ); - const factorLevel1Icon = require( 'mipmap!ARITHMETIC/factor_level_1_icon.png' ); - const factorLevel2Icon = require( 'mipmap!ARITHMETIC/factor_level_2_icon.png' ); - const factorLevel3Icon = require( 'mipmap!ARITHMETIC/factor_level_3_icon.png' ); - const divideLevel1Icon = require( 'mipmap!ARITHMETIC/divide_level_1_icon.png' ); - const divideLevel2Icon = require( 'mipmap!ARITHMETIC/divide_level_2_icon.png' ); - const divideLevel3Icon = require( 'mipmap!ARITHMETIC/divide_level_3_icon.png' ); - - // constants - const CHOOSE_LEVEL_TITLE_FONT = new PhetFont( { size: 24 } ); - const TAB_TITLE_FONT = new PhetFont( { size: 54 } ); - const BUTTON_LENGTH = 150; - - // icon sets, used to place on the buttons - const ICON_SETS = { - multiply: [ - multiplyLevel1Icon, - multiplyLevel2Icon, - multiplyLevel3Icon - ], - factor: [ - factorLevel1Icon, - factorLevel2Icon, - factorLevel3Icon - ], - divide: [ - divideLevel1Icon, - divideLevel2Icon, - divideLevel3Icon - ] - }; - - // strings - const chooseYourLevelString = require( 'string!ARITHMETIC/chooseYourLevel' ); - - /** - * @param {ArithmeticModel} model - Main model for screen. - * @param {string} titleString - Title string for given screen. - * @param {Function} callback - Callback function call after pressing button. - * @param {Bounds2} layoutBounds - Bounds of screen on which this will appear, used for layout - * @param {Object} [options] - * @constructor - */ - function LevelSelectionNode( model, titleString, callback, layoutBounds, options ) { - Node.call( this ); - - // Default options - options = merge( { - buttonBaseColor: 'white', - iconSet: 'multiply' // valid values are 'multiply', 'factor', and 'divide' - }, options ); - - // add title - const tabTitle = new Text( titleString, { - font: TAB_TITLE_FONT, - centerX: layoutBounds.centerX, - top: layoutBounds.height * 0.1 - } ); - this.addChild( tabTitle ); - - // add choose level title - const chooseLevelTitle = new Text( chooseYourLevelString, { - font: CHOOSE_LEVEL_TITLE_FONT, - centerX: layoutBounds.centerX, - top: tabTitle.bottom + 15 - } ); - this.addChild( chooseLevelTitle ); - - // add select level buttons - assert && assert( model.levelModels.length === ICON_SETS[ options.iconSet ].length, 'Number of icons doesn\'t match number of levels' ); - const levelSelectButtons = model.levelModels.map( function( level, levelIndex ) { - return new LevelSelectionButton( - new Image( ICON_SETS[ options.iconSet ][ levelIndex ] ), - model.levelModels[ levelIndex ].displayScoreProperty, - { - buttonWidth: BUTTON_LENGTH, - buttonHeight: BUTTON_LENGTH, - baseColor: options.buttonBaseColor, - bestTimeProperty: model.levelModels[ levelIndex ].bestTimeProperty, - bestTimeVisibleProperty: ArithmeticGlobals.timerEnabledProperty, - listener: function() { - callback( levelIndex ); - }, - scoreDisplayOptions: { - numberOfStars: ArithmeticConstants.NUM_STARS, - perfectScore: level.perfectScore - } +/** + * @param {ArithmeticModel} model - Main model for screen. + * @param {string} titleString - Title string for given screen. + * @param {Function} callback - Callback function call after pressing button. + * @param {Bounds2} layoutBounds - Bounds of screen on which this will appear, used for layout + * @param {Object} [options] + * @constructor + */ +function LevelSelectionNode( model, titleString, callback, layoutBounds, options ) { + Node.call( this ); + + // Default options + options = merge( { + buttonBaseColor: 'white', + iconSet: 'multiply' // valid values are 'multiply', 'factor', and 'divide' + }, options ); + + // add title + const tabTitle = new Text( titleString, { + font: TAB_TITLE_FONT, + centerX: layoutBounds.centerX, + top: layoutBounds.height * 0.1 + } ); + this.addChild( tabTitle ); + + // add choose level title + const chooseLevelTitle = new Text( chooseYourLevelString, { + font: CHOOSE_LEVEL_TITLE_FONT, + centerX: layoutBounds.centerX, + top: tabTitle.bottom + 15 + } ); + this.addChild( chooseLevelTitle ); + + // add select level buttons + assert && assert( model.levelModels.length === ICON_SETS[ options.iconSet ].length, 'Number of icons doesn\'t match number of levels' ); + const levelSelectButtons = model.levelModels.map( function( level, levelIndex ) { + return new LevelSelectionButton( + new Image( ICON_SETS[ options.iconSet ][ levelIndex ] ), + model.levelModels[ levelIndex ].displayScoreProperty, + { + buttonWidth: BUTTON_LENGTH, + buttonHeight: BUTTON_LENGTH, + baseColor: options.buttonBaseColor, + bestTimeProperty: model.levelModels[ levelIndex ].bestTimeProperty, + bestTimeVisibleProperty: ArithmeticGlobals.timerEnabledProperty, + listener: function() { + callback( levelIndex ); + }, + scoreDisplayOptions: { + numberOfStars: ArithmeticConstants.NUM_STARS, + perfectScore: level.perfectScore } - ); - } ); - const selectLevelButtonsHBox = new HBox( { spacing: 50, children: levelSelectButtons } ); - selectLevelButtonsHBox.top = chooseLevelTitle.bottom + 15; - selectLevelButtonsHBox.centerX = chooseLevelTitle.centerX; - this.addChild( selectLevelButtonsHBox ); - - // add timer and sound buttons - const soundAndTimerButtons = new VBox( { - spacing: 5, - children: [ - new TimerToggleButton( ArithmeticGlobals.timerEnabledProperty ) - ], - right: layoutBounds.maxX * 0.08, - bottom: layoutBounds.maxY * 0.95 - } ); - this.addChild( soundAndTimerButtons ); - - // add reset all button - const resetAllButton = new ResetAllButton( { - listener: function() {model.reset(); }, - right: layoutBounds.maxX * 0.98, - bottom: layoutBounds.maxY * 0.95 - } ); - this.addChild( resetAllButton ); - - // pass options through to superclass - this.mutate( options ); - } - - arithmetic.register( 'LevelSelectionNode', LevelSelectionNode ); - - return inherit( Node, LevelSelectionNode ); -} ); + } + ); + } ); + const selectLevelButtonsHBox = new HBox( { spacing: 50, children: levelSelectButtons } ); + selectLevelButtonsHBox.top = chooseLevelTitle.bottom + 15; + selectLevelButtonsHBox.centerX = chooseLevelTitle.centerX; + this.addChild( selectLevelButtonsHBox ); + + // add timer and sound buttons + const soundAndTimerButtons = new VBox( { + spacing: 5, + children: [ + new TimerToggleButton( ArithmeticGlobals.timerEnabledProperty ) + ], + right: layoutBounds.maxX * 0.08, + bottom: layoutBounds.maxY * 0.95 + } ); + this.addChild( soundAndTimerButtons ); + + // add reset all button + const resetAllButton = new ResetAllButton( { + listener: function() {model.reset(); }, + right: layoutBounds.maxX * 0.98, + bottom: layoutBounds.maxY * 0.95 + } ); + this.addChild( resetAllButton ); + + // pass options through to superclass + this.mutate( options ); +} + +arithmetic.register( 'LevelSelectionNode', LevelSelectionNode ); + +inherit( Node, LevelSelectionNode ); +export default LevelSelectionNode; \ No newline at end of file diff --git a/js/common/view/ScoreboardNode.js b/js/common/view/ScoreboardNode.js index 687ef1f0..0fb7efc5 100644 --- a/js/common/view/ScoreboardNode.js +++ b/js/common/view/ScoreboardNode.js @@ -6,124 +6,122 @@ * @author Andrey Zelenkov (MLearner) * @author John Blanco */ -define( require => { - 'use strict'; - - // modules - const arithmetic = require( 'ARITHMETIC/arithmetic' ); - const GameState = require( 'ARITHMETIC/common/model/GameState' ); - const GameTimer = require( 'VEGAS/GameTimer' ); - const inherit = require( 'PHET_CORE/inherit' ); - const merge = require( 'PHET_CORE/merge' ); - const Node = require( 'SCENERY/nodes/Node' ); - const Panel = require( 'SUN/Panel' ); - const PhetFont = require( 'SCENERY_PHET/PhetFont' ); - const RefreshButton = require( 'SCENERY_PHET/buttons/RefreshButton' ); - const StringUtils = require( 'PHETCOMMON/util/StringUtils' ); - const Text = require( 'SCENERY/nodes/Text' ); - const VBox = require( 'SCENERY/nodes/VBox' ); - - // strings - const labelScorePatternString = require( 'string!VEGAS/label.scorePattern' ); - const labelTimeString = require( 'string!VEGAS/label.time' ); - const patternLevel0LevelNumberString = require( 'string!ARITHMETIC/pattern.level.0levelNumber' ); - - // constants - const FONT = new PhetFont( { size: 18 } ); - const FONT_BOLD = new PhetFont( { size: 18, weight: 'bold' } ); - const PANEL_OPTIONS = { - fill: '#dddddd', - lineWidth: 0.5, - xMargin: 23, - yMargin: 23, - cornerRadius: 5, - align: 'center' + +import inherit from '../../../../phet-core/js/inherit.js'; +import merge from '../../../../phet-core/js/merge.js'; +import StringUtils from '../../../../phetcommon/js/util/StringUtils.js'; +import RefreshButton from '../../../../scenery-phet/js/buttons/RefreshButton.js'; +import PhetFont from '../../../../scenery-phet/js/PhetFont.js'; +import Node from '../../../../scenery/js/nodes/Node.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 GameTimer from '../../../../vegas/js/GameTimer.js'; +import vegasStrings from '../../../../vegas/js/vegas-strings.js'; +import arithmeticStrings from '../../arithmetic-strings.js'; +import arithmetic from '../../arithmetic.js'; +import GameState from '../model/GameState.js'; + +const labelScorePatternString = vegasStrings.label.scorePattern; +const labelTimeString = vegasStrings.label.time; +const patternLevel0LevelNumberString = arithmeticStrings.pattern.level[ '0levelNumber' ]; + +// constants +const FONT = new PhetFont( { size: 18 } ); +const FONT_BOLD = new PhetFont( { size: 18, weight: 'bold' } ); +const PANEL_OPTIONS = { + fill: '#dddddd', + lineWidth: 0.5, + xMargin: 23, + yMargin: 23, + cornerRadius: 5, + align: 'center' +}; +const SPACING = 16; + +/** + * @param {Property} levelNumberProperty - property for level displaying label. + * @param {Property} stateProperty - State of game property. + * @param {Array.} levelModels - Array of properties for score counter component. + * @param {Property} timerEnabledProperty - Time enabling flag. + * @param {Function} refreshLevelCallback - Callback listener for refresh level button. + * @param {Object} [options] - optional parameters that control the appearance and behavior of the panel + * @constructor + */ +function ScoreboardNode( levelNumberProperty, stateProperty, levelModels, timerEnabledProperty, refreshLevelCallback, options ) { + + options = merge( { + title: '' + }, options ); + const levelText = new Text( StringUtils.format( patternLevel0LevelNumberString, '' ), { font: FONT_BOLD } ); + const scoreText = new Text( StringUtils.format( labelScorePatternString, '0' ), { font: FONT } ); + const timeText = new Text( StringUtils.format( labelTimeString, GameTimer.formatTime( 0 ) ), { font: FONT } ); + + const panelOptions = merge( {}, PANEL_OPTIONS, options ); + + Node.call( this ); + + // add control panel components + const vBox = new VBox( { + spacing: SPACING, + children: [ + new Text( options.title, { font: FONT_BOLD } ), + levelText, + scoreText, + timeText, + // add refresh button + new RefreshButton( { + iconScale: 0.6, + xMargin: 14, + yMargin: 7, + listener: refreshLevelCallback + } ) + ] + } ); + this.addChild( new Panel( vBox, panelOptions ) ); + + // add observers + const updateScore = function( score ) { + scoreText.text = StringUtils.format( labelScorePatternString, score.toString() ); }; - const SPACING = 16; - - /** - * @param {Property} levelNumberProperty - property for level displaying label. - * @param {Property} stateProperty - State of game property. - * @param {Array.} levelModels - Array of properties for score counter component. - * @param {Property} timerEnabledProperty - Time enabling flag. - * @param {Function} refreshLevelCallback - Callback listener for refresh level button. - * @param {Object} [options] - optional parameters that control the appearance and behavior of the panel - * @constructor - */ - function ScoreboardNode( levelNumberProperty, stateProperty, levelModels, timerEnabledProperty, refreshLevelCallback, options ) { - - options = merge( { - title: '' - }, options ); - const levelText = new Text( StringUtils.format( patternLevel0LevelNumberString, '' ), { font: FONT_BOLD } ); - const scoreText = new Text( StringUtils.format( labelScorePatternString, '0' ), { font: FONT } ); - const timeText = new Text( StringUtils.format( labelTimeString, GameTimer.formatTime( 0 ) ), { font: FONT } ); - - const panelOptions = merge( {}, PANEL_OPTIONS, options ); - - Node.call( this ); - - // add control panel components - const vBox = new VBox( { - spacing: SPACING, - children: [ - new Text( options.title, { font: FONT_BOLD } ), - levelText, - scoreText, - timeText, - // add refresh button - new RefreshButton( { - iconScale: 0.6, - xMargin: 14, - yMargin: 7, - listener: refreshLevelCallback - } ) - ] - } ); - this.addChild( new Panel( vBox, panelOptions ) ); - - // add observers - const updateScore = function( score ) { - scoreText.text = StringUtils.format( labelScorePatternString, score.toString() ); - }; - - const updateTime = function( time ) { - timeText.text = StringUtils.format( labelTimeString, GameTimer.formatTime( time ) ); - }; - - levelNumberProperty.lazyLink( function( levelNew, levelPrevious ) { - if ( levelNew !== null ) { - levelText.text = StringUtils.format( patternLevel0LevelNumberString, ( levelNew + 1 ).toString() ); - } - else { - levelText.text = ''; - } - - // unlink observers for previous level - if ( levelModels[ levelPrevious ] ) { - levelModels[ levelPrevious ].currentScoreProperty.unlink( updateScore ); - levelModels[ levelPrevious ].gameTimer.elapsedTimeProperty.unlink( updateTime ); - } - - // link observers for new level - if ( stateProperty.value === GameState.SELECTING_LEVEL && levelModels[ levelNew ] ) { - levelModels[ levelNew ].currentScoreProperty.link( updateScore ); - levelModels[ levelNew ].gameTimer.elapsedTimeProperty.link( updateTime ); - } - } ); - - // add/remove time readout - timerEnabledProperty.link( function( isTimerEnabled ) { - if ( isTimerEnabled ) { - vBox.insertChild( 2, timeText ); // 2 - index of initial place for time - } - else { - vBox.removeChild( timeText ); - } - } ); - } - - arithmetic.register( 'ScoreboardNode', ScoreboardNode ); - - return inherit( Node, ScoreboardNode ); -} ); + + const updateTime = function( time ) { + timeText.text = StringUtils.format( labelTimeString, GameTimer.formatTime( time ) ); + }; + + levelNumberProperty.lazyLink( function( levelNew, levelPrevious ) { + if ( levelNew !== null ) { + levelText.text = StringUtils.format( patternLevel0LevelNumberString, ( levelNew + 1 ).toString() ); + } + else { + levelText.text = ''; + } + + // unlink observers for previous level + if ( levelModels[ levelPrevious ] ) { + levelModels[ levelPrevious ].currentScoreProperty.unlink( updateScore ); + levelModels[ levelPrevious ].gameTimer.elapsedTimeProperty.unlink( updateTime ); + } + + // link observers for new level + if ( stateProperty.value === GameState.SELECTING_LEVEL && levelModels[ levelNew ] ) { + levelModels[ levelNew ].currentScoreProperty.link( updateScore ); + levelModels[ levelNew ].gameTimer.elapsedTimeProperty.link( updateTime ); + } + } ); + + // add/remove time readout + timerEnabledProperty.link( function( isTimerEnabled ) { + if ( isTimerEnabled ) { + vBox.insertChild( 2, timeText ); // 2 - index of initial place for time + } + else { + vBox.removeChild( timeText ); + } + } ); +} + +arithmetic.register( 'ScoreboardNode', ScoreboardNode ); + +inherit( Node, ScoreboardNode ); +export default ScoreboardNode; \ No newline at end of file diff --git a/js/common/view/WorkspaceNode.js b/js/common/view/WorkspaceNode.js index 0db10bb8..6fdabcfc 100644 --- a/js/common/view/WorkspaceNode.js +++ b/js/common/view/WorkspaceNode.js @@ -7,228 +7,225 @@ * @author Andrey Zelenkov (MLearner) * @author John Blanco */ -define( require => { - 'use strict'; - - // modules - const arithmetic = require( 'ARITHMETIC/arithmetic' ); - const ArithmeticFaceWithPointsNode = require( 'ARITHMETIC/common/view/ArithmeticFaceWithPointsNode' ); - const ArithmeticGlobals = require( 'ARITHMETIC/common/ArithmeticGlobals' ); - const BackButton = require( 'SCENERY_PHET/buttons/BackButton' ); - const Dimension2 = require( 'DOT/Dimension2' ); - const GameState = require( 'ARITHMETIC/common/model/GameState' ); - const inherit = require( 'PHET_CORE/inherit' ); - const LevelCompletedNodeWrapper = require( 'ARITHMETIC/common/view/LevelCompletedNodeWrapper' ); - const merge = require( 'PHET_CORE/merge' ); - const Node = require( 'SCENERY/nodes/Node' ); - const NumberKeypad = require( 'SCENERY_PHET/NumberKeypad' ); - const PhetColorScheme = require( 'SCENERY_PHET/PhetColorScheme' ); - const PhetFont = require( 'SCENERY_PHET/PhetFont' ); - const ScoreboardNode = require( 'ARITHMETIC/common/view/ScoreboardNode' ); - const TextPushButton = require( 'SUN/buttons/TextPushButton' ); - - // constants - const BACK_BUTTON_BASE_COLOR = PhetColorScheme.BUTTON_YELLOW; // base color of back button - const BACK_BUTTON_MARGIN = new Dimension2( 20, 10 ); // margin of background of back button - const BUTTON_BASE_COLOR = PhetColorScheme.BUTTON_YELLOW; - const BUTTON_FONT = new PhetFont( { size: 20 } ); - - // strings - const checkString = require( 'string!ARITHMETIC/check' ); - const tryAgainString = require( 'string!ARITHMETIC/tryAgain' ); - - /** - * @param {ArithmeticModel} model - main model for screen. - * @param {Node} multiplicationTableNode - Multiplication table node for given screen. This can be (and generally is) - * different depending on the flavor of the game, i.e. multiplication, division, or factoring. This is why it is - * passed in rather than locally created. - * @param {Node} equationNode - Equation node for given screen. This can be (and generally is) different depending - * on the flavor of the game, i.e. multiplication, division, or factoring. This is why it is passed in rather than - * locally created. - * @param {Bounds2} layoutBounds - Bounds of main screen. Necessary for placing components. - * @param {Object} [options] - * - * @constructor - */ - function WorkspaceNode( model, multiplicationTableNode, equationNode, layoutBounds, options ) { - Node.call( this ); - - options = merge( { - scoreboardTitle: '', - showKeypad: true - }, options ); - - // add button for returning to the level select screen - const backButton = new BackButton( { - baseColor: BACK_BUTTON_BASE_COLOR, - xMargin: BACK_BUTTON_MARGIN.width, - yMargin: BACK_BUTTON_MARGIN.height, - scale: 0.75, // empirically determined - left: layoutBounds.maxX * 0.02, - top: layoutBounds.maxY * 0.02, - listener: function() { - model.returnToLevelSelectScreen(); - } - } ); - this.addChild( backButton ); +import Dimension2 from '../../../../dot/js/Dimension2.js'; +import inherit from '../../../../phet-core/js/inherit.js'; +import merge from '../../../../phet-core/js/merge.js'; +import BackButton from '../../../../scenery-phet/js/buttons/BackButton.js'; +import NumberKeypad from '../../../../scenery-phet/js/NumberKeypad.js'; +import PhetColorScheme from '../../../../scenery-phet/js/PhetColorScheme.js'; +import PhetFont from '../../../../scenery-phet/js/PhetFont.js'; +import Node from '../../../../scenery/js/nodes/Node.js'; +import TextPushButton from '../../../../sun/js/buttons/TextPushButton.js'; +import arithmeticStrings from '../../arithmetic-strings.js'; +import arithmetic from '../../arithmetic.js'; +import ArithmeticGlobals from '../ArithmeticGlobals.js'; +import GameState from '../model/GameState.js'; +import ArithmeticFaceWithPointsNode from './ArithmeticFaceWithPointsNode.js'; +import LevelCompletedNodeWrapper from './LevelCompletedNodeWrapper.js'; +import ScoreboardNode from './ScoreboardNode.js'; + +// constants +const BACK_BUTTON_BASE_COLOR = PhetColorScheme.BUTTON_YELLOW; // base color of back button +const BACK_BUTTON_MARGIN = new Dimension2( 20, 10 ); // margin of background of back button +const BUTTON_BASE_COLOR = PhetColorScheme.BUTTON_YELLOW; +const BUTTON_FONT = new PhetFont( { size: 20 } ); + +const checkString = arithmeticStrings.check; +const tryAgainString = arithmeticStrings.tryAgain; - // add multiplication table - multiplicationTableNode.mutate( { top: layoutBounds.maxY * 0.02, centerX: layoutBounds.width * 0.43 } ); - this.addChild( multiplicationTableNode ); - - // clear the multiplication table node on a refresh event. - model.refreshEmitter.addListener( function() { - multiplicationTableNode.refreshLevel( model.levelNumberProperty.get() ); +/** + * @param {ArithmeticModel} model - main model for screen. + * @param {Node} multiplicationTableNode - Multiplication table node for given screen. This can be (and generally is) + * different depending on the flavor of the game, i.e. multiplication, division, or factoring. This is why it is + * passed in rather than locally created. + * @param {Node} equationNode - Equation node for given screen. This can be (and generally is) different depending + * on the flavor of the game, i.e. multiplication, division, or factoring. This is why it is passed in rather than + * locally created. + * @param {Bounds2} layoutBounds - Bounds of main screen. Necessary for placing components. + * @param {Object} [options] + * + * @constructor + */ +function WorkspaceNode( model, multiplicationTableNode, equationNode, layoutBounds, options ) { + Node.call( this ); + + options = merge( { + scoreboardTitle: '', + showKeypad: true + }, options ); + + // add button for returning to the level select screen + const backButton = new BackButton( { + baseColor: BACK_BUTTON_BASE_COLOR, + xMargin: BACK_BUTTON_MARGIN.width, + yMargin: BACK_BUTTON_MARGIN.height, + scale: 0.75, // empirically determined + left: layoutBounds.maxX * 0.02, + top: layoutBounds.maxY * 0.02, + listener: function() { + model.returnToLevelSelectScreen(); + } + } ); + + this.addChild( backButton ); + + // add multiplication table + multiplicationTableNode.mutate( { top: layoutBounds.maxY * 0.02, centerX: layoutBounds.width * 0.43 } ); + this.addChild( multiplicationTableNode ); + + // clear the multiplication table node on a refresh event. + model.refreshEmitter.addListener( function() { + multiplicationTableNode.refreshLevel( model.levelNumberProperty.get() ); + } ); + + // add equation + equationNode.bottom = layoutBounds.maxY * 0.87; + equationNode.centerX = layoutBounds.width * 0.45; + this.addChild( equationNode ); + + // hide the equation node when the level has been completed + model.stateProperty.link( function( newGameState, oldGameState ) { + + // Hide the equation node when the level has been completed and when returning to the level selection screen + // after the level is complete. + equationNode.visible = !( newGameState === GameState.LEVEL_COMPLETED || + ( oldGameState === GameState.LEVEL_COMPLETED && + newGameState === GameState.SELECTING_LEVEL ) ); + } ); + + // define the width of the control panel so that it fits between the table and the bounds with some margin + const controlPanelWidth = layoutBounds.maxX - multiplicationTableNode.right - 60; + + // add control panel + const controlPanelNode = new ScoreboardNode( + model.levelNumberProperty, + model.stateProperty, + model.levelModels, + ArithmeticGlobals.timerEnabledProperty, + function() { + model.refreshLevel(); + }, + { + title: options.scoreboardTitle, + minWidth: controlPanelWidth, + maxWidth: controlPanelWidth, + centerX: ( multiplicationTableNode.right + layoutBounds.maxX ) / 2, + top: backButton.top + } + ); + controlPanelNode.top = multiplicationTableNode.top; + this.addChild( controlPanelNode ); + + // set up some variables needed for positioning the buttons + const buttonYCenter = ( equationNode.bottom + layoutBounds.maxY ) / 2 - 5; // tweaked a bit empirically + const maxButtonWidth = layoutBounds.maxX - multiplicationTableNode.bounds.maxX; + + // add keypad if necessary + if ( options.showKeypad ) { + // create and add the keypad + const keypad = new NumberKeypad( { + valueStringProperty: model.inputProperty, + validateKey: NumberKeypad.validateMaxDigits( { maxDigits: 3 } ), + centerX: controlPanelNode.centerX, + bottom: layoutBounds.maxY * 0.85 } ); + this.addChild( keypad ); - // add equation - equationNode.bottom = layoutBounds.maxY * 0.87; - equationNode.centerX = layoutBounds.width * 0.45; - this.addChild( equationNode ); - - // hide the equation node when the level has been completed + // Update the keypad state based on the game state. model.stateProperty.link( function( newGameState, oldGameState ) { - // Hide the equation node when the level has been completed and when returning to the level selection screen - // after the level is complete. - equationNode.visible = !( newGameState === GameState.LEVEL_COMPLETED || - ( oldGameState === GameState.LEVEL_COMPLETED && - newGameState === GameState.SELECTING_LEVEL ) ); - } ); - - // define the width of the control panel so that it fits between the table and the bounds with some margin - const controlPanelWidth = layoutBounds.maxX - multiplicationTableNode.right - 60; - - // add control panel - const controlPanelNode = new ScoreboardNode( - model.levelNumberProperty, - model.stateProperty, - model.levelModels, - ArithmeticGlobals.timerEnabledProperty, - function() { - model.refreshLevel(); - }, - { - title: options.scoreboardTitle, - minWidth: controlPanelWidth, - maxWidth: controlPanelWidth, - centerX: ( multiplicationTableNode.right + layoutBounds.maxX ) / 2, - top: backButton.top + if ( newGameState === GameState.DISPLAYING_INCORRECT_ANSWER_FEEDBACK ) { + // Arm the keypad for auto-clear when showing incorrect feedback. This is part of the feature where the user + // can simply start entering values again if they got the wrong answer initially. + keypad.clearOnNextKeyPress = true; } - ); - controlPanelNode.top = multiplicationTableNode.top; - this.addChild( controlPanelNode ); - - // set up some variables needed for positioning the buttons - const buttonYCenter = ( equationNode.bottom + layoutBounds.maxY ) / 2 - 5; // tweaked a bit empirically - const maxButtonWidth = layoutBounds.maxX - multiplicationTableNode.bounds.maxX; - - // add keypad if necessary - if ( options.showKeypad ) { - // create and add the keypad - const keypad = new NumberKeypad( { - valueStringProperty: model.inputProperty, - validateKey: NumberKeypad.validateMaxDigits( { maxDigits: 3 } ), - centerX: controlPanelNode.centerX, - bottom: layoutBounds.maxY * 0.85 - } ); - this.addChild( keypad ); - - // Update the keypad state based on the game state. - model.stateProperty.link( function( newGameState, oldGameState ) { - - if ( newGameState === GameState.DISPLAYING_INCORRECT_ANSWER_FEEDBACK ) { - // Arm the keypad for auto-clear when showing incorrect feedback. This is part of the feature where the user - // can simply start entering values again if they got the wrong answer initially. - keypad.clearOnNextKeyPress = true; - } - else if ( newGameState === GameState.AWAITING_USER_INPUT && - oldGameState !== GameState.DISPLAYING_INCORRECT_ANSWER_FEEDBACK ) { - keypad.clear(); - } - - // Only allow the user to input digits when expecting them. We use 'pickable' here instead of 'enabled' so that - // we don't gray out the keypad, which might visually draw attention to it. - keypad.pickable = newGameState === GameState.AWAITING_USER_INPUT || - newGameState === GameState.DISPLAYING_INCORRECT_ANSWER_FEEDBACK; - - // The keypad should be invisible once the level is completed, and should stay invisible on transition to the - // SELECTING_LEVEL state. - keypad.visible = !( newGameState === GameState.LEVEL_COMPLETED || - ( oldGameState === GameState.LEVEL_COMPLETED && - newGameState === GameState.SELECTING_LEVEL ) ); - } ); - - // add the 'Check' button, which is only used in conjunction with the keypad - const checkButton = new TextPushButton( checkString, { - font: BUTTON_FONT, - centerY: buttonYCenter, - centerX: controlPanelNode.centerX, - baseColor: BUTTON_BASE_COLOR, - maxWidth: maxButtonWidth, - listener: function() { model.fillEquation(); } - } ); - this.addChild( checkButton ); - - const updateCheckButtonState = function() { - checkButton.visible = ( model.stateProperty.get() === GameState.AWAITING_USER_INPUT ); - checkButton.enabled = model.inputProperty.get().length > 0; - }; - - // control the visibility of the 'Check' button - model.stateProperty.link( updateCheckButtonState ); - - // Monitor the string entered from the keypad and, if the user starts entering something immediately after - // receiving the feedback indicating an incorrect answer, allow them to retry the problem. - model.inputProperty.link( function( input ) { - if ( model.stateProperty.get() === GameState.DISPLAYING_INCORRECT_ANSWER_FEEDBACK ) { - model.retryProblem(); - } - updateCheckButtonState(); - } ); - } + else if ( newGameState === GameState.AWAITING_USER_INPUT && + oldGameState !== GameState.DISPLAYING_INCORRECT_ANSWER_FEEDBACK ) { + keypad.clear(); + } + + // Only allow the user to input digits when expecting them. We use 'pickable' here instead of 'enabled' so that + // we don't gray out the keypad, which might visually draw attention to it. + keypad.pickable = newGameState === GameState.AWAITING_USER_INPUT || + newGameState === GameState.DISPLAYING_INCORRECT_ANSWER_FEEDBACK; - // add smile face - this.addChild( new ArithmeticFaceWithPointsNode( model.faceModel, { - bottom: layoutBounds.maxY * 0.92, - left: layoutBounds.maxX * 0.04 - } ) ); + // The keypad should be invisible once the level is completed, and should stay invisible on transition to the + // SELECTING_LEVEL state. + keypad.visible = !( newGameState === GameState.LEVEL_COMPLETED || + ( oldGameState === GameState.LEVEL_COMPLETED && + newGameState === GameState.SELECTING_LEVEL ) ); + } ); - // add the 'try again' button - const tryAgainButton = new TextPushButton( tryAgainString, { + // add the 'Check' button, which is only used in conjunction with the keypad + const checkButton = new TextPushButton( checkString, { font: BUTTON_FONT, centerY: buttonYCenter, centerX: controlPanelNode.centerX, baseColor: BUTTON_BASE_COLOR, maxWidth: maxButtonWidth, - listener: function() { - model.inputProperty.reset(); - model.retryProblem(); - } + listener: function() { model.fillEquation(); } } ); - this.addChild( tryAgainButton ); + this.addChild( checkButton ); - // control the visibility of the 'Try Again' button - model.stateProperty.link( function( state ) { - tryAgainButton.visible = ( state === GameState.DISPLAYING_INCORRECT_ANSWER_FEEDBACK ); - } ); + const updateCheckButtonState = function() { + checkButton.visible = ( model.stateProperty.get() === GameState.AWAITING_USER_INPUT ); + checkButton.enabled = model.inputProperty.get().length > 0; + }; - // add the dialog that indicates that the level has been completed - this.addChild( new LevelCompletedNodeWrapper( - model.levelModels, - model.levelNumberProperty, - model.stateProperty, - ArithmeticGlobals.timerEnabledProperty, - function() { - model.stateProperty.set( GameState.LEVEL_COMPLETED ); - model.returnToLevelSelectScreen(); - }, - layoutBounds ) - ); - } + // control the visibility of the 'Check' button + model.stateProperty.link( updateCheckButtonState ); - arithmetic.register( 'WorkspaceNode', WorkspaceNode ); + // Monitor the string entered from the keypad and, if the user starts entering something immediately after + // receiving the feedback indicating an incorrect answer, allow them to retry the problem. + model.inputProperty.link( function( input ) { + if ( model.stateProperty.get() === GameState.DISPLAYING_INCORRECT_ANSWER_FEEDBACK ) { + model.retryProblem(); + } + updateCheckButtonState(); + } ); + } - return inherit( Node, WorkspaceNode ); -} ); + // add smile face + this.addChild( new ArithmeticFaceWithPointsNode( model.faceModel, { + bottom: layoutBounds.maxY * 0.92, + left: layoutBounds.maxX * 0.04 + } ) ); + + // add the 'try again' button + const tryAgainButton = new TextPushButton( tryAgainString, { + font: BUTTON_FONT, + centerY: buttonYCenter, + centerX: controlPanelNode.centerX, + baseColor: BUTTON_BASE_COLOR, + maxWidth: maxButtonWidth, + listener: function() { + model.inputProperty.reset(); + model.retryProblem(); + } + } ); + this.addChild( tryAgainButton ); + + // control the visibility of the 'Try Again' button + model.stateProperty.link( function( state ) { + tryAgainButton.visible = ( state === GameState.DISPLAYING_INCORRECT_ANSWER_FEEDBACK ); + } ); + + // add the dialog that indicates that the level has been completed + this.addChild( new LevelCompletedNodeWrapper( + model.levelModels, + model.levelNumberProperty, + model.stateProperty, + ArithmeticGlobals.timerEnabledProperty, + function() { + model.stateProperty.set( GameState.LEVEL_COMPLETED ); + model.returnToLevelSelectScreen(); + }, + layoutBounds ) + ); +} + +arithmetic.register( 'WorkspaceNode', WorkspaceNode ); + +inherit( Node, WorkspaceNode ); +export default WorkspaceNode; \ No newline at end of file diff --git a/js/common/view/table/AbstractCell.js b/js/common/view/table/AbstractCell.js index bab54412..fe6ce9e1 100644 --- a/js/common/view/table/AbstractCell.js +++ b/js/common/view/table/AbstractCell.js @@ -12,155 +12,151 @@ * @author Andrey Zelenkov (MLearner) * @author John Blanco */ -define( require => { - 'use strict'; - - // modules - const arithmetic = require( 'ARITHMETIC/arithmetic' ); - const inherit = require( 'PHET_CORE/inherit' ); - const merge = require( 'PHET_CORE/merge' ); - const PhetFont = require( 'SCENERY_PHET/PhetFont' ); - const Rectangle = require( 'SCENERY/nodes/Rectangle' ); - const Text = require( 'SCENERY/nodes/Text' ); - - // constants - const SMALL_FONT = new PhetFont( 17 ); - const MEDIUM_FONT = new PhetFont( 22 ); - const LARGE_FONT = new PhetFont( 30 ); - const EXAMPLE_HEIGHT_STRING = '123456789'; - const SMALL_FONT_HEIGHT = new Text( EXAMPLE_HEIGHT_STRING, { font: SMALL_FONT } ).bounds.height; - const MEDIUM_FONT_HEIGHT = new Text( EXAMPLE_HEIGHT_STRING, { font: MEDIUM_FONT } ).bounds.height; - const LARGE_FONT_HEIGHT = new Text( EXAMPLE_HEIGHT_STRING, { font: LARGE_FONT } ).bounds.height; - /** - * @param {Object} backgroundOptions - Background options for button. - * @param {Object} textOptions - Text options for button. - * @constructor - */ - function AbstractCell( backgroundOptions, textOptions ) { - Rectangle.call( this, 0, 0, backgroundOptions.width, backgroundOptions.height, merge( { - fill: 'white', - stroke: 'white', - lineWidth: 2.5 - }, backgroundOptions ) ); - - // @private - save/define text options for when the text node is created - this._textOptions = merge( { - font: chooseFont( this.bounds.height ), - fill: 'white', - centerY: this.bounds.height / 2, - initiallyVisible: true - }, textOptions ); - - // @private - define the _text variable, but create it only when needed in order to save time during startup - this._textNode = null; - - // @private - string to be displayed, used to support lazy creation of the node - this._text = ''; - - // if the text is initially visible, create the text node now, otherwise wait until it is shown - if ( this._textOptions.initiallyVisible ) { - createTextNodeIfNeeded( this ); - } +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 Rectangle from '../../../../../scenery/js/nodes/Rectangle.js'; +import Text from '../../../../../scenery/js/nodes/Text.js'; +import arithmetic from '../../../arithmetic.js'; + +// constants +const SMALL_FONT = new PhetFont( 17 ); +const MEDIUM_FONT = new PhetFont( 22 ); +const LARGE_FONT = new PhetFont( 30 ); +const EXAMPLE_HEIGHT_STRING = '123456789'; +const SMALL_FONT_HEIGHT = new Text( EXAMPLE_HEIGHT_STRING, { font: SMALL_FONT } ).bounds.height; +const MEDIUM_FONT_HEIGHT = new Text( EXAMPLE_HEIGHT_STRING, { font: MEDIUM_FONT } ).bounds.height; +const LARGE_FONT_HEIGHT = new Text( EXAMPLE_HEIGHT_STRING, { font: LARGE_FONT } ).bounds.height; + +/** + * @param {Object} backgroundOptions - Background options for button. + * @param {Object} textOptions - Text options for button. + * @constructor + */ +function AbstractCell( backgroundOptions, textOptions ) { + Rectangle.call( this, 0, 0, backgroundOptions.width, backgroundOptions.height, merge( { + fill: 'white', + stroke: 'white', + lineWidth: 2.5 + }, backgroundOptions ) ); + + // @private - save/define text options for when the text node is created + this._textOptions = merge( { + font: chooseFont( this.bounds.height ), + fill: 'white', + centerY: this.bounds.height / 2, + initiallyVisible: true + }, textOptions ); + + // @private - define the _text variable, but create it only when needed in order to save time during startup + this._textNode = null; + + // @private - string to be displayed, used to support lazy creation of the node + this._text = ''; + + // if the text is initially visible, create the text node now, otherwise wait until it is shown + if ( this._textOptions.initiallyVisible ) { + createTextNodeIfNeeded( this ); } +} + +// Convenience function for selecting appropriate font size for this cell. This is done as an optimization so that +// a new font doesn't need to be created for each cell. The sizes were empirically determined. +function chooseFont( cellHeight ) { + let font; + if ( cellHeight < 27 ) { + font = SMALL_FONT; + } + else if ( cellHeight < 35 ) { + font = MEDIUM_FONT; + } + else { + font = LARGE_FONT; + } + return font; +} - // Convenience function for selecting appropriate font size for this cell. This is done as an optimization so that - // a new font doesn't need to be created for each cell. The sizes were empirically determined. - function chooseFont( cellHeight ) { - let font; - if ( cellHeight < 27 ) { - font = SMALL_FONT; - } - else if ( cellHeight < 35 ) { - font = MEDIUM_FONT; - } - else { - font = LARGE_FONT; - } - return font; +// Convenience function that supports 'lazy' creation of the text node, which is done to minimize startup time. +function createTextNodeIfNeeded( cell ) { + + if ( cell._textNode ) { + // the text node already exists + return; } - // Convenience function that supports 'lazy' creation of the text node, which is done to minimize startup time. - function createTextNodeIfNeeded( cell ) { + cell._textNode = new Text( cell._text, cell._textOptions ); + cell._textNode.centerX = cell.bounds.width / 2; + cell.addChild( cell._textNode ); +} - if ( cell._textNode ) { - // the text node already exists - return; - } +arithmetic.register( 'AbstractCell', AbstractCell ); - cell._textNode = new Text( cell._text, cell._textOptions ); - cell._textNode.centerX = cell.bounds.width / 2; - cell.addChild( cell._textNode ); - } +export default inherit( Rectangle, AbstractCell, { + + // @public + setBackgroundFill: function( fill ) { + this.fill = fill; + }, - arithmetic.register( 'AbstractCell', AbstractCell ); - - return inherit( Rectangle, AbstractCell, { - - // @public - setBackgroundFill: function( fill ) { - this.fill = fill; - }, - - // @protected - setText: function( text ) { - this._text = text; - if ( this._textNode ) { - this._textNode.text = text; - this._textNode.centerX = this.width / 2; - } - }, - - // @public - setTextFill: function( fill ) { - createTextNodeIfNeeded( this ); - this._textNode.setFill( fill ); - }, - - // @public - showText: function() { - createTextNodeIfNeeded( this ); - this._textNode.visible = true; - }, - - // @public - hideText: function() { - if ( this._textNode ) { - this._textNode.visible = false; - } - }, - - /** - * Get the text string contained in this cell (not the text node). - * @public - * @returns {string} - */ - getTextString: function() { - return this._text; - }, - - // @public - isTextVisible: function() { - return this._textNode ? this._textNode.visible : false; - }, - - // @public - getTextHeight: function() { - let height; - switch( this._textOptions.font ) { - case SMALL_FONT: - height = SMALL_FONT_HEIGHT; - break; - case MEDIUM_FONT: - height = MEDIUM_FONT_HEIGHT; - break; - case LARGE_FONT: - height = LARGE_FONT_HEIGHT; - break; - default: - assert && assert( false, 'unrecognized font' ); - } - return height; + // @protected + setText: function( text ) { + this._text = text; + if ( this._textNode ) { + this._textNode.text = text; + this._textNode.centerX = this.width / 2; + } + }, + + // @public + setTextFill: function( fill ) { + createTextNodeIfNeeded( this ); + this._textNode.setFill( fill ); + }, + + // @public + showText: function() { + createTextNodeIfNeeded( this ); + this._textNode.visible = true; + }, + + // @public + hideText: function() { + if ( this._textNode ) { + this._textNode.visible = false; } - } ); -} ); + }, + + /** + * Get the text string contained in this cell (not the text node). + * @public + * @returns {string} + */ + getTextString: function() { + return this._text; + }, + + // @public + isTextVisible: function() { + return this._textNode ? this._textNode.visible : false; + }, + + // @public + getTextHeight: function() { + let height; + switch( this._textOptions.font ) { + case SMALL_FONT: + height = SMALL_FONT_HEIGHT; + break; + case MEDIUM_FONT: + height = MEDIUM_FONT_HEIGHT; + break; + case LARGE_FONT: + height = LARGE_FONT_HEIGHT; + break; + default: + assert && assert( false, 'unrecognized font' ); + } + return height; + } +} ); \ No newline at end of file diff --git a/js/common/view/table/MultiplicationTableBodyCell.js b/js/common/view/table/MultiplicationTableBodyCell.js index 58867822..19682320 100644 --- a/js/common/view/table/MultiplicationTableBodyCell.js +++ b/js/common/view/table/MultiplicationTableBodyCell.js @@ -7,52 +7,47 @@ * @author John Blanco */ -define( require => { - 'use strict'; - - // modules - const AbstractCell = require( 'ARITHMETIC/common/view/table/AbstractCell' ); - const arithmetic = require( 'ARITHMETIC/arithmetic' ); - const inherit = require( 'PHET_CORE/inherit' ); - const merge = require( 'PHET_CORE/merge' ); - - // constants - const HOVER_COLOR = 'rgb(238,253,77)'; - const NORMAL_COLOR = 'rgb(50,70,255)'; - const SELECT_COLOR = 'rgb(77,0,153)'; - - /** - * @param {Text} contentText - Text label for button. - * @param {Object} backgroundOptions - Background options for button. - * @constructor - */ - function MultiplicationTableBodyCell( contentText, backgroundOptions ) { - backgroundOptions = merge( { - fill: NORMAL_COLOR - }, backgroundOptions ); - AbstractCell.call( this, backgroundOptions, { initiallyVisible: false } ); - - this.setText( contentText ); - } +import inherit from '../../../../../phet-core/js/inherit.js'; +import merge from '../../../../../phet-core/js/merge.js'; +import arithmetic from '../../../arithmetic.js'; +import AbstractCell from './AbstractCell.js'; + +// constants +const HOVER_COLOR = 'rgb(238,253,77)'; +const NORMAL_COLOR = 'rgb(50,70,255)'; +const SELECT_COLOR = 'rgb(77,0,153)'; - arithmetic.register( 'MultiplicationTableBodyCell', MultiplicationTableBodyCell ); +/** + * @param {Text} contentText - Text label for button. + * @param {Object} backgroundOptions - Background options for button. + * @constructor + */ +function MultiplicationTableBodyCell( contentText, backgroundOptions ) { + backgroundOptions = merge( { + fill: NORMAL_COLOR + }, backgroundOptions ); + AbstractCell.call( this, backgroundOptions, { initiallyVisible: false } ); - return inherit( AbstractCell, MultiplicationTableBodyCell, { + this.setText( contentText ); +} - // @public - set cell into the state that indicates that the user is hovering over it - setHover: function() { - this.setBackgroundFill( HOVER_COLOR ); - }, +arithmetic.register( 'MultiplicationTableBodyCell', MultiplicationTableBodyCell ); - // @public - set cell to normal, default appearance state - setNormal: function() { - this.setBackgroundFill( NORMAL_COLOR ); - }, +export default inherit( AbstractCell, MultiplicationTableBodyCell, { - // @public - set cell to the selected state - setSelected: function() { - this.setBackgroundFill( SELECT_COLOR ); - } + // @public - set cell into the state that indicates that the user is hovering over it + setHover: function() { + this.setBackgroundFill( HOVER_COLOR ); + }, + + // @public - set cell to normal, default appearance state + setNormal: function() { + this.setBackgroundFill( NORMAL_COLOR ); + }, + + // @public - set cell to the selected state + setSelected: function() { + this.setBackgroundFill( SELECT_COLOR ); + } - } ); -} ); +} ); \ No newline at end of file diff --git a/js/common/view/table/MultiplicationTableHeaderCell.js b/js/common/view/table/MultiplicationTableHeaderCell.js index 0cdb7364..892a0919 100644 --- a/js/common/view/table/MultiplicationTableHeaderCell.js +++ b/js/common/view/table/MultiplicationTableHeaderCell.js @@ -6,51 +6,46 @@ * @author Andrey Zelenkov (MLearner) */ -define( require => { - 'use strict'; - - // modules - const AbstractCell = require( 'ARITHMETIC/common/view/table/AbstractCell' ); - const arithmetic = require( 'ARITHMETIC/arithmetic' ); - const inherit = require( 'PHET_CORE/inherit' ); - const merge = require( 'PHET_CORE/merge' ); - - // constants - const NORMAL_COLOR_BACKGROUND = 'rgb(220,60,33)'; // background normal color - const SELECT_COLOR_BACKGROUND = 'rgb(0,0,128)'; // background select color - const NORMAL_COLOR_TEXT = 'white'; // text normal color - const SELECT_COLOR_TEXT = 'rgb(255,253,56)'; // text select color - - /** - * @param {Text} contentText - Text label for button. - * @param {Object} backgroundOptions - Background options for button. - * @param {Object} textOptions - Text options for button. - * - * @constructor - */ - function MultiplicationTableHeaderCell( contentText, backgroundOptions, textOptions ) { - backgroundOptions = merge( { - fill: NORMAL_COLOR_BACKGROUND - }, backgroundOptions ); - AbstractCell.call( this, backgroundOptions, textOptions ); - - this.setText( contentText ); - } +import inherit from '../../../../../phet-core/js/inherit.js'; +import merge from '../../../../../phet-core/js/merge.js'; +import arithmetic from '../../../arithmetic.js'; +import AbstractCell from './AbstractCell.js'; + +// constants +const NORMAL_COLOR_BACKGROUND = 'rgb(220,60,33)'; // background normal color +const SELECT_COLOR_BACKGROUND = 'rgb(0,0,128)'; // background select color +const NORMAL_COLOR_TEXT = 'white'; // text normal color +const SELECT_COLOR_TEXT = 'rgb(255,253,56)'; // text select color + +/** + * @param {Text} contentText - Text label for button. + * @param {Object} backgroundOptions - Background options for button. + * @param {Object} textOptions - Text options for button. + * + * @constructor + */ +function MultiplicationTableHeaderCell( contentText, backgroundOptions, textOptions ) { + backgroundOptions = merge( { + fill: NORMAL_COLOR_BACKGROUND + }, backgroundOptions ); + AbstractCell.call( this, backgroundOptions, textOptions ); - arithmetic.register( 'MultiplicationTableHeaderCell', MultiplicationTableHeaderCell ); + this.setText( contentText ); +} - return inherit( AbstractCell, MultiplicationTableHeaderCell, { +arithmetic.register( 'MultiplicationTableHeaderCell', MultiplicationTableHeaderCell ); - // @public - setSelected: function() { - this.setBackgroundFill( SELECT_COLOR_BACKGROUND ); - this.setTextFill( SELECT_COLOR_TEXT ); - }, +export default inherit( AbstractCell, MultiplicationTableHeaderCell, { - // @public - setNormal: function() { - this.setBackgroundFill( NORMAL_COLOR_BACKGROUND ); - this.setTextFill( NORMAL_COLOR_TEXT ); - } - } ); -} ); + // @public + setSelected: function() { + this.setBackgroundFill( SELECT_COLOR_BACKGROUND ); + this.setTextFill( SELECT_COLOR_TEXT ); + }, + + // @public + setNormal: function() { + this.setBackgroundFill( NORMAL_COLOR_BACKGROUND ); + this.setTextFill( NORMAL_COLOR_TEXT ); + } +} ); \ No newline at end of file diff --git a/js/common/view/table/MultiplicationTableNode.js b/js/common/view/table/MultiplicationTableNode.js index 113cd65e..532f3601 100644 --- a/js/common/view/table/MultiplicationTableNode.js +++ b/js/common/view/table/MultiplicationTableNode.js @@ -10,314 +10,310 @@ * @author Andrey Zelenkov (MLearner) * @author John Blanco */ -define( require => { - 'use strict'; - - // modules - const Animation = require( 'TWIXT/Animation' ); - const arithmetic = require( 'ARITHMETIC/arithmetic' ); - const ArithmeticConstants = require( 'ARITHMETIC/common/ArithmeticConstants' ); - const Dimension2 = require( 'DOT/Dimension2' ); - const Easing = require( 'TWIXT/Easing' ); - const GameState = require( 'ARITHMETIC/common/model/GameState' ); - const inherit = require( 'PHET_CORE/inherit' ); - const MathSymbols = require( 'SCENERY_PHET/MathSymbols' ); - const MultiplicationTableBodyCell = require( 'ARITHMETIC/common/view/table/MultiplicationTableBodyCell' ); - const MultiplicationTableHeaderCell = require( 'ARITHMETIC/common/view/table/MultiplicationTableHeaderCell' ); - const Node = require( 'SCENERY/nodes/Node' ); - const PhetFont = require( 'SCENERY_PHET/PhetFont' ); - const Rectangle = require( 'SCENERY/nodes/Rectangle' ); - const ScreenView = require( 'JOIST/ScreenView' ); - const Text = require( 'SCENERY/nodes/Text' ); - const Utils = require( 'DOT/Utils' ); - const Vector2 = require( 'DOT/Vector2' ); - - // constants - const TABLE_SIZE = new Dimension2( 434, 320 ); // table size in screen coordinates, empirically determined - const ANSWER_ANIMATION_TIME = 0.8; // in seconds - - // Starting point for the animation of the answer. It is not ideal that this is a constant, because it means that if - // the layout changes, this will need to be manually updated, but it's tricky to get it coordinated with the layout - // in some other way. - const ANSWER_ANIMATION_ORIGIN = new Vector2( 370, 380 ); - /** - * @param {Property.} levelNumberProperty - level property. - * @param {Property.} stateProperty - current state property - * @param {Array.} levelModels - array of models for each level - * @param {boolean} animateAnswer - flag that controls whether answer appears to fly into the cell or just appears - * @constructor - */ - function MultiplicationTableNode( levelNumberProperty, stateProperty, levelModels, animateAnswer ) { - const self = this; - Node.call( this ); +import Dimension2 from '../../../../../dot/js/Dimension2.js'; +import Utils from '../../../../../dot/js/Utils.js'; +import Vector2 from '../../../../../dot/js/Vector2.js'; +import ScreenView from '../../../../../joist/js/ScreenView.js'; +import inherit from '../../../../../phet-core/js/inherit.js'; +import MathSymbols from '../../../../../scenery-phet/js/MathSymbols.js'; +import PhetFont from '../../../../../scenery-phet/js/PhetFont.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 Animation from '../../../../../twixt/js/Animation.js'; +import Easing from '../../../../../twixt/js/Easing.js'; +import arithmetic from '../../../arithmetic.js'; +import ArithmeticConstants from '../../ArithmeticConstants.js'; +import GameState from '../../model/GameState.js'; +import MultiplicationTableBodyCell from './MultiplicationTableBodyCell.js'; +import MultiplicationTableHeaderCell from './MultiplicationTableHeaderCell.js'; + +// constants +const TABLE_SIZE = new Dimension2( 434, 320 ); // table size in screen coordinates, empirically determined +const ANSWER_ANIMATION_TIME = 0.8; // in seconds + +// Starting point for the animation of the answer. It is not ideal that this is a constant, because it means that if +// the layout changes, this will need to be manually updated, but it's tricky to get it coordinated with the layout +// in some other way. +const ANSWER_ANIMATION_ORIGIN = new Vector2( 370, 380 ); - this.levelNumberProperty = levelNumberProperty; // @protected - needs to be available to sub-classes - this.viewForLevel = new Array( levelModels.length ); // @private - array with views for each level +/** + * @param {Property.} levelNumberProperty - level property. + * @param {Property.} stateProperty - current state property + * @param {Array.} levelModels - array of models for each level + * @param {boolean} animateAnswer - flag that controls whether answer appears to fly into the cell or just appears + * @constructor + */ +function MultiplicationTableNode( levelNumberProperty, stateProperty, levelModels, animateAnswer ) { + const self = this; + Node.call( this ); - // @private - three-dimensional array of the cells, indexed by [levelNumber][multiplicand][multiplier] - this.cells = new Array( levelModels.length ); + this.levelNumberProperty = levelNumberProperty; // @protected - needs to be available to sub-classes + this.viewForLevel = new Array( levelModels.length ); // @private - array with views for each level - // add stroke for all multiplication table views - const backgroundRect = new Rectangle( 0, 0, 0, 0, { - fill: 'white', - cursor: 'pointer' // this is done so that the cursor doesn't change when moving between cells - } ); - this.addChild( backgroundRect ); - - // create view of multiplication table for each of the levels - levelModels.forEach( function( level, levelIndex ) { - const tableSize = level.tableSize; - const cellOptions = { - lineWidth: Math.max( Math.ceil( ( TABLE_SIZE.width / ( tableSize + 1 ) ) / 40 ), 2 ), - width: TABLE_SIZE.width / ( tableSize + 1 ), - height: TABLE_SIZE.height / ( tableSize + 1 ) - }; - const levelRootNode = new Node( { visible: false } ); // root node for a single level - let row; - let column; - - // init store for cells - self.cells[ levelIndex ] = new Array( tableSize + 1 ); - - let cell; - let cellTop = 0; - let cellLeft = 0; - - // create the table row by row - for ( row = 0; row <= tableSize; row++ ) { - self.cells[ levelIndex ][ row ] = new Array( tableSize + 1 ); - - // first row - if ( row === 0 ) { - for ( column = 0; column <= tableSize; column++ ) { - - // first cell is the multiplier operator, others are multipliers - if ( column === 0 ) { - cell = new MultiplicationTableHeaderCell( MathSymbols.TIMES, cellOptions, { - - // specify font and size, equation empirically determined, makes font smaller for larger tables - font: new PhetFont( { size: Utils.roundSymmetric( cellOptions.height * 0.85 ) } ) - } ); - } - else { - cell = new MultiplicationTableHeaderCell( column.toString(), cellOptions ); - } - cell.top = cellTop; - cell.left = cellLeft; - cellLeft += cellOptions.width; - levelRootNode.addChild( cell ); - self.cells[ levelIndex ][ row ][ column ] = cell; + // @private - three-dimensional array of the cells, indexed by [levelNumber][multiplicand][multiplier] + this.cells = new Array( levelModels.length ); + + // add stroke for all multiplication table views + const backgroundRect = new Rectangle( 0, 0, 0, 0, { + fill: 'white', + cursor: 'pointer' // this is done so that the cursor doesn't change when moving between cells + } ); + this.addChild( backgroundRect ); + + // create view of multiplication table for each of the levels + levelModels.forEach( function( level, levelIndex ) { + const tableSize = level.tableSize; + const cellOptions = { + lineWidth: Math.max( Math.ceil( ( TABLE_SIZE.width / ( tableSize + 1 ) ) / 40 ), 2 ), + width: TABLE_SIZE.width / ( tableSize + 1 ), + height: TABLE_SIZE.height / ( tableSize + 1 ) + }; + const levelRootNode = new Node( { visible: false } ); // root node for a single level + let row; + let column; + + // init store for cells + self.cells[ levelIndex ] = new Array( tableSize + 1 ); + + let cell; + let cellTop = 0; + let cellLeft = 0; + + // create the table row by row + for ( row = 0; row <= tableSize; row++ ) { + self.cells[ levelIndex ][ row ] = new Array( tableSize + 1 ); + + // first row + if ( row === 0 ) { + for ( column = 0; column <= tableSize; column++ ) { + + // first cell is the multiplier operator, others are multipliers + if ( column === 0 ) { + cell = new MultiplicationTableHeaderCell( MathSymbols.TIMES, cellOptions, { + + // specify font and size, equation empirically determined, makes font smaller for larger tables + font: new PhetFont( { size: Utils.roundSymmetric( cellOptions.height * 0.85 ) } ) + } ); + } + else { + cell = new MultiplicationTableHeaderCell( column.toString(), cellOptions ); } + cell.top = cellTop; + cell.left = cellLeft; + cellLeft += cellOptions.width; + levelRootNode.addChild( cell ); + self.cells[ levelIndex ][ row ][ column ] = cell; } + } - // other rows - else { - for ( column = 0; column <= tableSize; column++ ) { + // other rows + else { + for ( column = 0; column <= tableSize; column++ ) { - // first cell in each row is a multiplier, others are products - if ( column === 0 ) { - cell = new MultiplicationTableHeaderCell( row.toString(), cellOptions ); - } - else { - cell = new MultiplicationTableBodyCell( - ( row * column ).toString(), - cellOptions - ); - } - cell.top = cellTop; - cell.left = cellLeft; - cellLeft += cellOptions.width; - levelRootNode.addChild( cell ); - self.cells[ levelIndex ][ row ][ column ] = cell; + // first cell in each row is a multiplier, others are products + if ( column === 0 ) { + cell = new MultiplicationTableHeaderCell( row.toString(), cellOptions ); + } + else { + cell = new MultiplicationTableBodyCell( + ( row * column ).toString(), + cellOptions + ); } + cell.top = cellTop; + cell.left = cellLeft; + cellLeft += cellOptions.width; + levelRootNode.addChild( cell ); + self.cells[ levelIndex ][ row ][ column ] = cell; } - cellTop += cellOptions.height; - cellLeft = 0; } + cellTop += cellOptions.height; + cellLeft = 0; + } - // add view to node - self.addChild( levelRootNode ); + // add view to node + self.addChild( levelRootNode ); - // save view - self.viewForLevel[ levelIndex ] = levelRootNode; - } ); + // save view + self.viewForLevel[ levelIndex ] = levelRootNode; + } ); - // set background size - backgroundRect.setRectWidth( this.bounds.width ); - backgroundRect.setRectHeight( this.bounds.height ); + // set background size + backgroundRect.setRectWidth( this.bounds.width ); + backgroundRect.setRectHeight( this.bounds.height ); - levelNumberProperty.link( function( levelNumberCurrent, levelNumberPrev ) { + levelNumberProperty.link( function( levelNumberCurrent, levelNumberPrev ) { - // show multiplication table view for the current level - if ( self.viewForLevel[ levelNumberCurrent ] ) { - self.viewForLevel[ levelNumberCurrent ].visible = true; - } + // show multiplication table view for the current level + if ( self.viewForLevel[ levelNumberCurrent ] ) { + self.viewForLevel[ levelNumberCurrent ].visible = true; + } - // hide previous multiplication table view - if ( self.viewForLevel[ levelNumberPrev ] ) { - self.viewForLevel[ levelNumberPrev ].visible = false; - } - } ); + // hide previous multiplication table view + if ( self.viewForLevel[ levelNumberPrev ] ) { + self.viewForLevel[ levelNumberPrev ].visible = false; + } + } ); - // @private - node that will be used to animate the answer moving from the equation to the location of the cell. - this.flyingProduct = new Text( 'X', { - font: ArithmeticConstants.EQUATION_FONT_TEXT, - fill: 'white', - visible: false - } ); - this.addChild( this.flyingProduct ); - - // @private - define the animation that will move the flying product - this.flyingProductAnimation = null; - - // update the visible answers each time the user gets a correct answer - stateProperty.link( function( newState, oldState ) { - if ( newState === GameState.DISPLAYING_CORRECT_ANSWER_FEEDBACK || oldState === GameState.SELECTING_LEVEL ) { - - const level = levelNumberProperty.value; // convenience var - const levelModel = levelModels[ level ]; // convenience var - - // make sure the appropriate cells are displaying their numerical values - for ( let multiplicand = 1; multiplicand <= levelModel.tableSize; multiplicand++ ) { - for ( let multiplier = 1; multiplier <= levelModel.tableSize; multiplier++ ) { - var cell = self.cells[ levelNumberProperty.value ][ multiplicand ][ multiplier ]; - if ( levelModel.isCellUsed( multiplicand, multiplier ) ) { - - // If the cell is marked as used but the text is not yet visible, animate the product to the cell. - if ( animateAnswer && !cell.isTextVisible() ) { - - // Animate the product moving from the equation to the appropriate cell within the table. - ( function() { - const destinationCell = cell; - self.flyingProduct.text = destinationCell.getTextString(); - self.flyingProduct.setScaleMagnitude( 1 ); - const flyingProductDestination = self.globalToLocalPoint( destinationCell.parentToGlobalPoint( destinationCell.center ) ); - - // create the animation - self.flyingProductAnimation = new Animation( { - duration: ANSWER_ANIMATION_TIME, - targets: [ - - // position - { - object: self.flyingProduct, - attribute: 'center', - from: ANSWER_ANIMATION_ORIGIN, - to: flyingProductDestination, - easing: Easing.CUBIC_IN_OUT + // @private - node that will be used to animate the answer moving from the equation to the location of the cell. + this.flyingProduct = new Text( 'X', { + font: ArithmeticConstants.EQUATION_FONT_TEXT, + fill: 'white', + visible: false + } ); + this.addChild( this.flyingProduct ); + + // @private - define the animation that will move the flying product + this.flyingProductAnimation = null; + + // update the visible answers each time the user gets a correct answer + stateProperty.link( function( newState, oldState ) { + if ( newState === GameState.DISPLAYING_CORRECT_ANSWER_FEEDBACK || oldState === GameState.SELECTING_LEVEL ) { + + const level = levelNumberProperty.value; // convenience var + const levelModel = levelModels[ level ]; // convenience var + + // make sure the appropriate cells are displaying their numerical values + for ( let multiplicand = 1; multiplicand <= levelModel.tableSize; multiplicand++ ) { + for ( let multiplier = 1; multiplier <= levelModel.tableSize; multiplier++ ) { + var cell = self.cells[ levelNumberProperty.value ][ multiplicand ][ multiplier ]; + if ( levelModel.isCellUsed( multiplicand, multiplier ) ) { + + // If the cell is marked as used but the text is not yet visible, animate the product to the cell. + if ( animateAnswer && !cell.isTextVisible() ) { + + // Animate the product moving from the equation to the appropriate cell within the table. + ( function() { + const destinationCell = cell; + self.flyingProduct.text = destinationCell.getTextString(); + self.flyingProduct.setScaleMagnitude( 1 ); + const flyingProductDestination = self.globalToLocalPoint( destinationCell.parentToGlobalPoint( destinationCell.center ) ); + + // create the animation + self.flyingProductAnimation = new Animation( { + duration: ANSWER_ANIMATION_TIME, + targets: [ + + // position + { + object: self.flyingProduct, + attribute: 'center', + from: ANSWER_ANIMATION_ORIGIN, + to: flyingProductDestination, + easing: Easing.CUBIC_IN_OUT + }, + + // scale + { + from: 1, + to: destinationCell.getTextHeight() / self.flyingProduct.height, + setValue: function( newScale ) { + self.flyingProduct.setScaleMagnitude( newScale ); }, - - // scale - { - from: 1, - to: destinationCell.getTextHeight() / self.flyingProduct.height, - setValue: function( newScale ) { - self.flyingProduct.setScaleMagnitude( newScale ); - }, - easing: Easing.CUBIC_IN_OUT - } - ] - } ); - self.flyingProductAnimation.beginEmitter.addListener( function() { - self.flyingProduct.visible = true; - } ); - self.flyingProductAnimation.finishEmitter.addListener( function() { - destinationCell.showText(); - self.flyingProduct.visible = false; - self.flyingProductAnimation = null; - } ); - - // start the animation - self.flyingProductAnimation.start(); - } )(); - } - else { - // No animation, so just show the text. - cell.showText(); - } + easing: Easing.CUBIC_IN_OUT + } + ] + } ); + self.flyingProductAnimation.beginEmitter.addListener( function() { + self.flyingProduct.visible = true; + } ); + self.flyingProductAnimation.finishEmitter.addListener( function() { + destinationCell.showText(); + self.flyingProduct.visible = false; + self.flyingProductAnimation = null; + } ); + + // start the animation + self.flyingProductAnimation.start(); + } )(); } else { - cell.hideText(); + // No animation, so just show the text. + cell.showText(); } } + else { + cell.hideText(); + } } } - } ); - } + } + } ); +} - arithmetic.register( 'MultiplicationTableNode', MultiplicationTableNode ); +arithmetic.register( 'MultiplicationTableNode', MultiplicationTableNode ); - return inherit( Node, MultiplicationTableNode, { +export default inherit( Node, MultiplicationTableNode, { - /** - * Set all cells for given level to the default background color - * @param {number} level - * @public - */ - setCellsToDefaultColor: function( level ) { - this.cells[ level ].forEach( function( multiplicands ) { - multiplicands.forEach( function( cell ) { - cell.setNormal(); - } ); - } ); - }, - - /** - * Clear all cells for the given level, meaning that the text is hidden and the background color is set to default. - * @param {number} level - * @public - */ - clearCells: function( level ) { - this.setCellsToDefaultColor( level ); - this.cells[ level ].forEach( function( cellRow, cellRowIndex ) { - if ( cellRowIndex > 0 ) { - cellRow.forEach( function( cell, index ) { - if ( index > 0 ) { - cell.hideText(); - } - } ); - } + /** + * Set all cells for given level to the default background color + * @param {number} level + * @public + */ + setCellsToDefaultColor: function( level ) { + this.cells[ level ].forEach( function( multiplicands ) { + multiplicands.forEach( function( cell ) { + cell.setNormal(); } ); - }, - - // @public - refresh the level, may need additional behavior added by subclasses - refreshLevel: function( level ) { - if ( this.flyingProductAnimation ) { + } ); + }, - // A refresh was initiated while the animation was in progress. This is a race condition, and details about - // it can be seen in https://github.com/phetsims/arithmetic/issues/148. The animation should be cancelled. - this.flyingProductAnimation.stop(); - self.flyingProductAnimation = null; - this.flyingProduct.visible = false; + /** + * Clear all cells for the given level, meaning that the text is hidden and the background color is set to default. + * @param {number} level + * @public + */ + clearCells: function( level ) { + this.setCellsToDefaultColor( level ); + this.cells[ level ].forEach( function( cellRow, cellRowIndex ) { + if ( cellRowIndex > 0 ) { + cellRow.forEach( function( cell, index ) { + if ( index > 0 ) { + cell.hideText(); + } + } ); } - this.clearCells( level ); - }, - - /** - * Get the position, in global coordinates, of the specified cell. - * - * @param level - * @param column - * @param row - * @public - */ - whereIsCellCenter: function( level, column, row ) { - - // Find the parent screen by moving up the scene graph. - const cell = this.cells[ level ][ row ][ column ]; - let testNode = cell; - let parentScreen = null; - while ( testNode !== null ) { - if ( testNode instanceof ScreenView ) { - parentScreen = testNode; - break; - } - testNode = testNode.parents[ 0 ]; // Move up the scene graph by one level + } ); + }, + + // @public - refresh the level, may need additional behavior added by subclasses + refreshLevel: function( level ) { + if ( this.flyingProductAnimation ) { + + // A refresh was initiated while the animation was in progress. This is a race condition, and details about + // it can be seen in https://github.com/phetsims/arithmetic/issues/148. The animation should be cancelled. + this.flyingProductAnimation.stop(); + self.flyingProductAnimation = null; + this.flyingProduct.visible = false; + } + this.clearCells( level ); + }, + + /** + * Get the position, in global coordinates, of the specified cell. + * + * @param level + * @param column + * @param row + * @public + */ + whereIsCellCenter: function( level, column, row ) { + + // Find the parent screen by moving up the scene graph. + const cell = this.cells[ level ][ row ][ column ]; + let testNode = cell; + let parentScreen = null; + while ( testNode !== null ) { + if ( testNode instanceof ScreenView ) { + parentScreen = testNode; + break; } - return parentScreen.globalToLocalPoint( cell.parentToGlobalPoint( cell.center ) ); + testNode = testNode.parents[ 0 ]; // Move up the scene graph by one level } + return parentScreen.globalToLocalPoint( cell.parentToGlobalPoint( cell.center ) ); + } - } ); -} ); +} ); \ No newline at end of file diff --git a/js/divide/DivideScreen.js b/js/divide/DivideScreen.js index 5b684056..578797de 100644 --- a/js/divide/DivideScreen.js +++ b/js/divide/DivideScreen.js @@ -6,44 +6,41 @@ * Andrey Zelenkov (MLearner) * @author John Blanco */ -define( require => { - 'use strict'; - - // modules - const arithmetic = require( 'ARITHMETIC/arithmetic' ); - const ArithmeticConstants = require( 'ARITHMETIC/common/ArithmeticConstants' ); - const DivideModel = require( 'ARITHMETIC/divide/model/DivideModel' ); - const DivideScreenIconNode = require( 'ARITHMETIC/divide/view/DivideScreenIconNode' ); - const DivideView = require( 'ARITHMETIC/divide/view/DivideView' ); - const inherit = require( 'PHET_CORE/inherit' ); - const merge = require( 'PHET_CORE/merge' ); - const Property = require( 'AXON/Property' ); - const Screen = require( 'JOIST/Screen' ); - const Tandem = require( 'TANDEM/Tandem' ); - - // strings - const divideString = require( 'string!ARITHMETIC/divide' ); - - /** - * @param {Object} [options] - * @constructor - */ - function DivideScreen( options ) { - - options = merge( { - name: divideString, - backgroundColorProperty: new Property( ArithmeticConstants.BACKGROUND_COLOR ), - homeScreenIcon: new DivideScreenIconNode(), - tandem: Tandem.REQUIRED - }, options ); - - Screen.call( this, - function() { return new DivideModel( options.tandem.createTandem( 'model' ) ); }, - function( model ) { return new DivideView( model ); }, - options ); - } - - arithmetic.register( 'DivideScreen', DivideScreen ); - - return inherit( Screen, DivideScreen ); -} ); \ No newline at end of file + +import Property from '../../../axon/js/Property.js'; +import Screen from '../../../joist/js/Screen.js'; +import inherit from '../../../phet-core/js/inherit.js'; +import merge from '../../../phet-core/js/merge.js'; +import Tandem from '../../../tandem/js/Tandem.js'; +import arithmeticStrings from '../arithmetic-strings.js'; +import arithmetic from '../arithmetic.js'; +import ArithmeticConstants from '../common/ArithmeticConstants.js'; +import DivideModel from './model/DivideModel.js'; +import DivideScreenIconNode from './view/DivideScreenIconNode.js'; +import DivideView from './view/DivideView.js'; + +const divideString = arithmeticStrings.divide; + +/** + * @param {Object} [options] + * @constructor + */ +function DivideScreen( options ) { + + options = merge( { + name: divideString, + backgroundColorProperty: new Property( ArithmeticConstants.BACKGROUND_COLOR ), + homeScreenIcon: new DivideScreenIconNode(), + tandem: Tandem.REQUIRED + }, options ); + + Screen.call( this, + function() { return new DivideModel( options.tandem.createTandem( 'model' ) ); }, + function( model ) { return new DivideView( model ); }, + options ); +} + +arithmetic.register( 'DivideScreen', DivideScreen ); + +inherit( Screen, DivideScreen ); +export default DivideScreen; \ No newline at end of file diff --git a/js/divide/model/DivideModel.js b/js/divide/model/DivideModel.js index f7a72854..6b74e63b 100644 --- a/js/divide/model/DivideModel.js +++ b/js/divide/model/DivideModel.js @@ -7,120 +7,114 @@ * @author John Blanco */ -define( require => { - 'use strict'; +import Random from '../../../../dot/js/Random.js'; +import inherit from '../../../../phet-core/js/inherit.js'; +import soundManager from '../../../../tambo/js/soundManager.js'; +import arithmetic from '../../arithmetic.js'; +import ArithmeticModel from '../../common/model/ArithmeticModel.js'; +import GameState from '../../common/model/GameState.js'; - // modules - const arithmetic = require( 'ARITHMETIC/arithmetic' ); - const ArithmeticModel = require( 'ARITHMETIC/common/model/ArithmeticModel' ); - const GameState = require( 'ARITHMETIC/common/model/GameState' ); - const inherit = require( 'PHET_CORE/inherit' ); - const Random = require( 'DOT/Random' ); - const soundManager = require( 'TAMBO/soundManager' ); +/** + * @param {Tandem} tandem + * @constructor + */ +function DivideModel( tandem ) { + const self = this; + ArithmeticModel.call( this, tandem, { + fillEquation: function() { - /** - * @param {Tandem} tandem - * @constructor - */ - function DivideModel( tandem ) { - const self = this; - ArithmeticModel.call( this, tandem, { - fillEquation: function() { + // Convert any strings entered by the user into numerical values. + self.problemModel.multiplierProperty.set( parseInt( self.problemModel.multiplierProperty.get(), 10 ) ); + self.problemModel.multiplicandProperty.set( parseInt( self.problemModel.multiplicandProperty.get(), 10 ) ); - // Convert any strings entered by the user into numerical values. - self.problemModel.multiplierProperty.set( parseInt( self.problemModel.multiplierProperty.get(), 10 ) ); - self.problemModel.multiplicandProperty.set( parseInt( self.problemModel.multiplicandProperty.get(), 10 ) ); + // Submit this answer so that it can be checked. + self.submitAnswer(); + } + } ); + this.random = new Random( { staticSeed: true } ); +} - // Submit this answer so that it can be checked. - self.submitAnswer(); - } - } ); - this.random = new Random( { staticSeed: true } ); - } +arithmetic.register( 'DivideModel', DivideModel ); - arithmetic.register( 'DivideModel', DivideModel ); +export default inherit( ArithmeticModel, DivideModel, { - return inherit( ArithmeticModel, DivideModel, { + // @public + setUpUnansweredProblem: function() { - // @public - setUpUnansweredProblem: function() { + // get available multiplier pair + const multipliers = this.selectUnusedMultiplierPair(); - // get available multiplier pair - const multipliers = this.selectUnusedMultiplierPair(); + if ( multipliers ) { - if ( multipliers ) { + // reset multiplierPair and score properties + this.problemModel.multiplicandProperty.reset(); + this.problemModel.multiplierProperty.reset(); + this.problemModel.productProperty.reset(); + this.problemModel.possiblePointsProperty.reset(); - // reset multiplierPair and score properties - this.problemModel.multiplicandProperty.reset(); - this.problemModel.multiplierProperty.reset(); - this.problemModel.productProperty.reset(); - this.problemModel.possiblePointsProperty.reset(); + // set product + this.problemModel.productProperty.set( multipliers.multiplicand * multipliers.multiplier ); - // set product - this.problemModel.productProperty.set( multipliers.multiplicand * multipliers.multiplier ); + // set multiplicand or multiplier + if ( this.random.nextBoolean() ) { + this.problemModel.multiplicandProperty.set( multipliers.multiplicand ); + this.activeInputProperty.set( 'multiplier' ); + } + else { + this.problemModel.multiplierProperty.set( multipliers.multiplier ); + this.activeInputProperty.set( 'multiplicand' ); + } - // set multiplicand or multiplier - if ( this.random.nextBoolean() ) { - this.problemModel.multiplicandProperty.set( multipliers.multiplicand ); - this.activeInputProperty.set( 'multiplier' ); - } - else { - this.problemModel.multiplierProperty.set( multipliers.multiplier ); - this.activeInputProperty.set( 'multiplicand' ); - } + return true; + } - return true; - } + // All multiplier pairs have been used, so false is returned. + return false; + }, - // All multiplier pairs have been used, so false is returned. - return false; - }, - - /** - * Automatically answer most of the questions. This is useful for testing, since it can save time when testing - * how the sim behaves when a user finishing answering all questions for a level. We need to be very careful that - * this is never available in the published sim. - * @override - * @protected - */ - autoAnswer: function() { - const self = this; - - // make sure that sound is off, since otherwise it dings for every solved problem - const soundState = soundManager.enabled; - soundManager.enabled = false; - - // answer the questions - const numQuestions = this.activeLevelModel.tableSize * this.activeLevelModel.tableSize; - const numQuestionsToAnswer = numQuestions - 1; - console.log( 'Automatically answering', numQuestionsToAnswer, 'of', numQuestions, 'questions.' ); - _.times( numQuestionsToAnswer, function( index ) { - if ( !self.problemModel.multiplicandProperty.get() ) { - self.problemModel.multiplicandProperty.set( - self.problemModel.productProperty.get() / self.problemModel.multiplierProperty.get() - ); - } - else if ( !self.problemModel.multiplierProperty.get() ) { - self.problemModel.multiplierProperty.set( - self.problemModel.productProperty.get() / self.problemModel.multiplicandProperty.get() - ); - } - else { - throw new Error( 'unexpected problem structure for problem', index ); - } - self.activeLevelModel.currentScoreProperty.value += self.problemModel.possiblePointsProperty.get(); - self.activeLevelModel.displayScoreProperty.set( self.activeLevelModel.currentScoreProperty.get() ); - self.activeLevelModel.markCellAsUsed( - self.problemModel.multiplicandProperty.get(), - self.problemModel.multiplierProperty.get() - ); - self.stateProperty.set( GameState.DISPLAYING_CORRECT_ANSWER_FEEDBACK ); - self.nextProblem(); - } ); + /** + * Automatically answer most of the questions. This is useful for testing, since it can save time when testing + * how the sim behaves when a user finishing answering all questions for a level. We need to be very careful that + * this is never available in the published sim. + * @override + * @protected + */ + autoAnswer: function() { + const self = this; - // restore the original sound state - soundManager.enabled = soundState; - } - } ); + // make sure that sound is off, since otherwise it dings for every solved problem + const soundState = soundManager.enabled; + soundManager.enabled = false; + + // answer the questions + const numQuestions = this.activeLevelModel.tableSize * this.activeLevelModel.tableSize; + const numQuestionsToAnswer = numQuestions - 1; + console.log( 'Automatically answering', numQuestionsToAnswer, 'of', numQuestions, 'questions.' ); + _.times( numQuestionsToAnswer, function( index ) { + if ( !self.problemModel.multiplicandProperty.get() ) { + self.problemModel.multiplicandProperty.set( + self.problemModel.productProperty.get() / self.problemModel.multiplierProperty.get() + ); + } + else if ( !self.problemModel.multiplierProperty.get() ) { + self.problemModel.multiplierProperty.set( + self.problemModel.productProperty.get() / self.problemModel.multiplicandProperty.get() + ); + } + else { + throw new Error( 'unexpected problem structure for problem', index ); + } + self.activeLevelModel.currentScoreProperty.value += self.problemModel.possiblePointsProperty.get(); + self.activeLevelModel.displayScoreProperty.set( self.activeLevelModel.currentScoreProperty.get() ); + self.activeLevelModel.markCellAsUsed( + self.problemModel.multiplicandProperty.get(), + self.problemModel.multiplierProperty.get() + ); + self.stateProperty.set( GameState.DISPLAYING_CORRECT_ANSWER_FEEDBACK ); + self.nextProblem(); + } ); -} ); + // restore the original sound state + soundManager.enabled = soundState; + } +} ); \ No newline at end of file diff --git a/js/divide/view/DivideEquationNode.js b/js/divide/view/DivideEquationNode.js index 20d44bd1..39a3ef26 100644 --- a/js/divide/view/DivideEquationNode.js +++ b/js/divide/view/DivideEquationNode.js @@ -5,73 +5,70 @@ * * @author Andrey Zelenkov (MLearner) */ -define( require => { - 'use strict'; - // modules - const arithmetic = require( 'ARITHMETIC/arithmetic' ); - const EquationNode = require( 'ARITHMETIC/common/view/EquationNode' ); - const GameState = require( 'ARITHMETIC/common/model/GameState' ); - const inherit = require( 'PHET_CORE/inherit' ); +import inherit from '../../../../phet-core/js/inherit.js'; +import arithmetic from '../../arithmetic.js'; +import GameState from '../../common/model/GameState.js'; +import EquationNode from '../../common/view/EquationNode.js'; - /** - * @param {Property} stateProperty - State of game property. - * @param {Property} multiplicandProperty - Property necessary for creating multiplicand input. - * @param {Property} multiplierProperty - Property necessary for creating multiplier input. - * @param {Property} productProperty - Property necessary for creating product input. - * @param {Property} inputProperty - Input property. - * @param {Property} activeInputProperty - Link to active input. - * - * @constructor - */ - function DivideEquationNode( stateProperty, multiplicandProperty, multiplierProperty, productProperty, inputProperty, activeInputProperty ) { - const self = this; - EquationNode.call( this, multiplicandProperty, multiplierProperty, productProperty ); +/** + * @param {Property} stateProperty - State of game property. + * @param {Property} multiplicandProperty - Property necessary for creating multiplicand input. + * @param {Property} multiplierProperty - Property necessary for creating multiplier input. + * @param {Property} productProperty - Property necessary for creating product input. + * @param {Property} inputProperty - Input property. + * @param {Property} activeInputProperty - Link to active input. + * + * @constructor + */ +function DivideEquationNode( stateProperty, multiplicandProperty, multiplierProperty, productProperty, inputProperty, activeInputProperty ) { + const self = this; + EquationNode.call( this, multiplicandProperty, multiplierProperty, productProperty ); - // If the input value changes, it means that the user entered something, so put it in the appropriate equation node. - inputProperty.lazyLink( function( inputString ) { - if ( activeInputProperty.get() === 'multiplicand' ) { - multiplicandProperty.set( parseInt( inputString, 10 ) ); - } - else if ( activeInputProperty.get() === 'multiplier' ) { - multiplierProperty.set( parseInt( inputString, 10 ) ); - } - } ); + // If the input value changes, it means that the user entered something, so put it in the appropriate equation node. + inputProperty.lazyLink( function( inputString ) { + if ( activeInputProperty.get() === 'multiplicand' ) { + multiplicandProperty.set( parseInt( inputString, 10 ) ); + } + else if ( activeInputProperty.get() === 'multiplier' ) { + multiplierProperty.set( parseInt( inputString, 10 ) ); + } + } ); - function updateFocus() { - if ( stateProperty.value === GameState.AWAITING_USER_INPUT ) { - self.multiplierInput.setFocus( activeInputProperty.value === 'multiplier' ); - self.multiplicandInput.setFocus( activeInputProperty.value === 'multiplicand' ); - } - else { - // Not awaiting user input, so neither input gets focus. - self.multiplierInput.setFocus( false ); - self.multiplicandInput.setFocus( false ); - } + function updateFocus() { + if ( stateProperty.value === GameState.AWAITING_USER_INPUT ) { + self.multiplierInput.setFocus( activeInputProperty.value === 'multiplier' ); + self.multiplicandInput.setFocus( activeInputProperty.value === 'multiplicand' ); } + else { + // Not awaiting user input, so neither input gets focus. + self.multiplierInput.setFocus( false ); + self.multiplicandInput.setFocus( false ); + } + } - activeInputProperty.link( function( activeInput ) { - if ( activeInput === 'multiplier' ) { - self.multiplierInput.clear(); - } - else if ( activeInput === 'multiplicand' ) { - self.multiplicandInput.clear(); - } - self.multiplicandInput.setInteractiveAppearance( activeInput === 'multiplicand' ); - self.multiplierInput.setInteractiveAppearance( activeInput === 'multiplier' ); - updateFocus(); - } ); + activeInputProperty.link( function( activeInput ) { + if ( activeInput === 'multiplier' ) { + self.multiplierInput.clear(); + } + else if ( activeInput === 'multiplicand' ) { + self.multiplicandInput.clear(); + } + self.multiplicandInput.setInteractiveAppearance( activeInput === 'multiplicand' ); + self.multiplierInput.setInteractiveAppearance( activeInput === 'multiplier' ); + updateFocus(); + } ); - stateProperty.link( function( state ) { + stateProperty.link( function( state ) { - // Display a not equal sign if the user input and incorrect answer. - self.setShowEqual( state !== GameState.DISPLAYING_INCORRECT_ANSWER_FEEDBACK ); + // Display a not equal sign if the user input and incorrect answer. + self.setShowEqual( state !== GameState.DISPLAYING_INCORRECT_ANSWER_FEEDBACK ); - updateFocus(); - } ); - } + updateFocus(); + } ); +} - arithmetic.register( 'DivideEquationNode', DivideEquationNode ); +arithmetic.register( 'DivideEquationNode', DivideEquationNode ); - return inherit( EquationNode, DivideEquationNode ); -} ); +inherit( EquationNode, DivideEquationNode ); +export default DivideEquationNode; \ No newline at end of file diff --git a/js/divide/view/DivideScreenIconNode.js b/js/divide/view/DivideScreenIconNode.js index a00136c2..f8901a2d 100644 --- a/js/divide/view/DivideScreenIconNode.js +++ b/js/divide/view/DivideScreenIconNode.js @@ -6,49 +6,46 @@ * @author Andrey Zelenkov (MLearner) * @author John Blanco (MLearner) */ -define( require => { - 'use strict'; - - // modules - const arithmetic = require( 'ARITHMETIC/arithmetic' ); - const ArithmeticConstants = require( 'ARITHMETIC/common/ArithmeticConstants' ); - const Circle = require( 'SCENERY/nodes/Circle' ); - const inherit = require( 'PHET_CORE/inherit' ); - const Line = require( 'SCENERY/nodes/Line' ); - const Rectangle = require( 'SCENERY/nodes/Rectangle' ); - - // constants - const ICON_SIZE = ArithmeticConstants.SCREEN_ICON_SIZE; - const SYMBOL_COLOR = '#FFF31E'; - const CIRCLE_RADIUS = 30; // Empirically determined - - /** - * @constructor - */ - function DivideScreenIconNode() { - - // create the background - Rectangle.call( this, 0, 0, ICON_SIZE.width, ICON_SIZE.height, { fill: ArithmeticConstants.ICON_BACKGROUND_COLOR } ); - - // Create and add the divide symbol. We are doing our own, since the Unicode one doesn't look very good. - this.addChild( new Line( 0, 0, ICON_SIZE.width * 0.4, 0, { - stroke: SYMBOL_COLOR, - lineWidth: 33, // empirically determined - center: this.center - } ) ); - this.addChild( new Circle( CIRCLE_RADIUS, { - fill: SYMBOL_COLOR, - centerX: ICON_SIZE.width / 2, - centerY: ICON_SIZE.height * 0.3 - } ) ); - this.addChild( new Circle( CIRCLE_RADIUS, { - fill: SYMBOL_COLOR, - centerX: ICON_SIZE.width / 2, - centerY: ICON_SIZE.height * 0.7 - } ) ); - } - - arithmetic.register( 'DivideScreenIconNode', DivideScreenIconNode ); - - return inherit( Rectangle, DivideScreenIconNode ); -} ); + +import inherit from '../../../../phet-core/js/inherit.js'; +import Circle from '../../../../scenery/js/nodes/Circle.js'; +import Line from '../../../../scenery/js/nodes/Line.js'; +import Rectangle from '../../../../scenery/js/nodes/Rectangle.js'; +import arithmetic from '../../arithmetic.js'; +import ArithmeticConstants from '../../common/ArithmeticConstants.js'; + +// constants +const ICON_SIZE = ArithmeticConstants.SCREEN_ICON_SIZE; +const SYMBOL_COLOR = '#FFF31E'; +const CIRCLE_RADIUS = 30; // Empirically determined + +/** + * @constructor + */ +function DivideScreenIconNode() { + + // create the background + Rectangle.call( this, 0, 0, ICON_SIZE.width, ICON_SIZE.height, { fill: ArithmeticConstants.ICON_BACKGROUND_COLOR } ); + + // Create and add the divide symbol. We are doing our own, since the Unicode one doesn't look very good. + this.addChild( new Line( 0, 0, ICON_SIZE.width * 0.4, 0, { + stroke: SYMBOL_COLOR, + lineWidth: 33, // empirically determined + center: this.center + } ) ); + this.addChild( new Circle( CIRCLE_RADIUS, { + fill: SYMBOL_COLOR, + centerX: ICON_SIZE.width / 2, + centerY: ICON_SIZE.height * 0.3 + } ) ); + this.addChild( new Circle( CIRCLE_RADIUS, { + fill: SYMBOL_COLOR, + centerX: ICON_SIZE.width / 2, + centerY: ICON_SIZE.height * 0.7 + } ) ); +} + +arithmetic.register( 'DivideScreenIconNode', DivideScreenIconNode ); + +inherit( Rectangle, DivideScreenIconNode ); +export default DivideScreenIconNode; \ No newline at end of file diff --git a/js/divide/view/DivideScreenTableNode.js b/js/divide/view/DivideScreenTableNode.js index 13b6ea16..6bc54810 100644 --- a/js/divide/view/DivideScreenTableNode.js +++ b/js/divide/view/DivideScreenTableNode.js @@ -6,76 +6,71 @@ * @author Andrey Zelenkov (MLearner) */ -define( require => { - 'use strict'; +import inherit from '../../../../phet-core/js/inherit.js'; +import arithmetic from '../../arithmetic.js'; +import GameState from '../../common/model/GameState.js'; +import MultiplicationTableNode from '../../common/view/table/MultiplicationTableNode.js'; - // modules - const arithmetic = require( 'ARITHMETIC/arithmetic' ); - const GameState = require( 'ARITHMETIC/common/model/GameState' ); - const inherit = require( 'PHET_CORE/inherit' ); - const MultiplicationTableNode = require( 'ARITHMETIC/common/view/table/MultiplicationTableNode' ); - - /** - * @param {ArithmeticModel} model - * @constructor - */ - function DivideScreenTableNode( model ) { - const self = this; - MultiplicationTableNode.call( this, model.levelNumberProperty, model.stateProperty, model.levelModels, true ); - this.problemModel = model.problemModel; - this.levelNumberProperty = model.levelNumberProperty; - - model.stateProperty.lazyLink( function( state ) { +/** + * @param {ArithmeticModel} model + * @constructor + */ +function DivideScreenTableNode( model ) { + const self = this; + MultiplicationTableNode.call( this, model.levelNumberProperty, model.stateProperty, model.levelModels, true ); + this.problemModel = model.problemModel; + this.levelNumberProperty = model.levelNumberProperty; - // set view for multiplication table after choosing multiplicand and multiplier - if ( state === GameState.AWAITING_USER_INPUT ) { + model.stateProperty.lazyLink( function( state ) { - // clear cell colors prior to showing the problem - self.setCellsToDefaultColor( self.levelNumberProperty.value ); + // set view for multiplication table after choosing multiplicand and multiplier + if ( state === GameState.AWAITING_USER_INPUT ) { - // highlight the active multiplicand or multiplier - if ( model.activeInputProperty.get() === 'multiplier' ) { - self.cells[ self.levelNumberProperty.value ][ self.problemModel.multiplicandProperty.get() ][ 0 ].setSelected(); - } - else { - assert && assert( model.activeInputProperty.get() === 'multiplicand' ); - self.cells[ self.levelNumberProperty.value ][ 0 ][ self.problemModel.multiplierProperty.get() ].setSelected(); - } - } - else if ( state === GameState.DISPLAYING_CORRECT_ANSWER_FEEDBACK ) { + // clear cell colors prior to showing the problem + self.setCellsToDefaultColor( self.levelNumberProperty.value ); - // Make the cells that correspond to the answer change color. + // highlight the active multiplicand or multiplier + if ( model.activeInputProperty.get() === 'multiplier' ) { self.cells[ self.levelNumberProperty.value ][ self.problemModel.multiplicandProperty.get() ][ 0 ].setSelected(); + } + else { + assert && assert( model.activeInputProperty.get() === 'multiplicand' ); self.cells[ self.levelNumberProperty.value ][ 0 ][ self.problemModel.multiplierProperty.get() ].setSelected(); - for ( let multiplicand = 1; multiplicand <= self.problemModel.multiplicandProperty.get(); multiplicand++ ) { - for ( let multiplier = 1; multiplier <= self.problemModel.multiplierProperty.get(); multiplier++ ) { - self.cells[ self.levelNumberProperty.value ][ multiplicand ][ multiplier ].setSelected(); - } - } } - else if ( state === GameState.LEVEL_COMPLETED ) { + } + else if ( state === GameState.DISPLAYING_CORRECT_ANSWER_FEEDBACK ) { - // set all cells to default conditions when the table has been filled - self.setCellsToDefaultColor( self.levelNumberProperty.value ); + // Make the cells that correspond to the answer change color. + self.cells[ self.levelNumberProperty.value ][ self.problemModel.multiplicandProperty.get() ][ 0 ].setSelected(); + self.cells[ self.levelNumberProperty.value ][ 0 ][ self.problemModel.multiplierProperty.get() ].setSelected(); + for ( let multiplicand = 1; multiplicand <= self.problemModel.multiplicandProperty.get(); multiplicand++ ) { + for ( let multiplier = 1; multiplier <= self.problemModel.multiplierProperty.get(); multiplier++ ) { + self.cells[ self.levelNumberProperty.value ][ multiplicand ][ multiplier ].setSelected(); + } } - } ); - } + } + else if ( state === GameState.LEVEL_COMPLETED ) { - arithmetic.register( 'DivideScreenTableNode', DivideScreenTableNode ); + // set all cells to default conditions when the table has been filled + self.setCellsToDefaultColor( self.levelNumberProperty.value ); + } + } ); +} - return inherit( MultiplicationTableNode, DivideScreenTableNode, { +arithmetic.register( 'DivideScreenTableNode', DivideScreenTableNode ); - // @public, @override - refreshLevel: function() { - MultiplicationTableNode.prototype.refreshLevel.call( this, this.levelNumberProperty.value ); +export default inherit( MultiplicationTableNode, DivideScreenTableNode, { - // highlight the appropriate header cell - if ( this.problemModel.multiplicandProperty.get() ) { - this.cells[ this.levelNumberProperty.value ][ this.problemModel.multiplicandProperty.get() ][ 0 ].setSelected(); - } - else { - this.cells[ this.levelNumberProperty.value ][ 0 ][ this.problemModel.multiplierProperty.get() ].setSelected(); - } + // @public, @override + refreshLevel: function() { + MultiplicationTableNode.prototype.refreshLevel.call( this, this.levelNumberProperty.value ); + + // highlight the appropriate header cell + if ( this.problemModel.multiplicandProperty.get() ) { + this.cells[ this.levelNumberProperty.value ][ this.problemModel.multiplicandProperty.get() ][ 0 ].setSelected(); } - } ); -} ); + else { + this.cells[ this.levelNumberProperty.value ][ 0 ][ this.problemModel.multiplierProperty.get() ].setSelected(); + } + } +} ); \ No newline at end of file diff --git a/js/divide/view/DivideView.js b/js/divide/view/DivideView.js index c3415a3f..56cd9da2 100644 --- a/js/divide/view/DivideView.js +++ b/js/divide/view/DivideView.js @@ -5,46 +5,43 @@ * * @author Andrey Zelenkov (MLearner) */ -define( require => { - 'use strict'; - // modules - const arithmetic = require( 'ARITHMETIC/arithmetic' ); - const ArithmeticView = require( 'ARITHMETIC/common/view/ArithmeticView' ); - const DivideEquationNode = require( 'ARITHMETIC/divide/view/DivideEquationNode' ); - const DivideScreenTableNode = require( 'ARITHMETIC/divide/view/DivideScreenTableNode' ); - const inherit = require( 'PHET_CORE/inherit' ); +import inherit from '../../../../phet-core/js/inherit.js'; +import arithmeticStrings from '../../arithmetic-strings.js'; +import arithmetic from '../../arithmetic.js'; +import ArithmeticView from '../../common/view/ArithmeticView.js'; +import DivideEquationNode from './DivideEquationNode.js'; +import DivideScreenTableNode from './DivideScreenTableNode.js'; - // strings - const divideString = require( 'string!ARITHMETIC/divide' ); +const divideString = arithmeticStrings.divide; - /** - * @param {DivideModel} model - Main model for screen. - * @constructor - */ - function DivideView( model ) { +/** + * @param {DivideModel} model - Main model for screen. + * @constructor + */ +function DivideView( model ) { - ArithmeticView.call( - this, - model, - new DivideScreenTableNode( model ), - new DivideEquationNode( - model.stateProperty, - model.problemModel.multiplicandProperty, - model.problemModel.multiplierProperty, - model.problemModel.productProperty, - model.inputProperty, - model.activeInputProperty - ), - { - titleString: divideString, - levelSelectButtonColor: '#BC76A5', - levelSelectIconSet: 'divide' - } - ); - } + ArithmeticView.call( + this, + model, + new DivideScreenTableNode( model ), + new DivideEquationNode( + model.stateProperty, + model.problemModel.multiplicandProperty, + model.problemModel.multiplierProperty, + model.problemModel.productProperty, + model.inputProperty, + model.activeInputProperty + ), + { + titleString: divideString, + levelSelectButtonColor: '#BC76A5', + levelSelectIconSet: 'divide' + } + ); +} - arithmetic.register( 'DivideView', DivideView ); +arithmetic.register( 'DivideView', DivideView ); - return inherit( ArithmeticView, DivideView ); -} ); +inherit( ArithmeticView, DivideView ); +export default DivideView; \ No newline at end of file diff --git a/js/factor/FactorScreen.js b/js/factor/FactorScreen.js index 2050631a..ac7844f8 100644 --- a/js/factor/FactorScreen.js +++ b/js/factor/FactorScreen.js @@ -5,44 +5,41 @@ * * @author John Blanco, Andrey Zelenkov (MLearner) */ -define( require => { - 'use strict'; - - // modules - const arithmetic = require( 'ARITHMETIC/arithmetic' ); - const ArithmeticConstants = require( 'ARITHMETIC/common/ArithmeticConstants' ); - const FactorModel = require( 'ARITHMETIC/factor/model/FactorModel' ); - const FactorScreenIconNode = require( 'ARITHMETIC/factor/view/FactorScreenIconNode' ); - const FactorView = require( 'ARITHMETIC/factor/view/FactorView' ); - const inherit = require( 'PHET_CORE/inherit' ); - const merge = require( 'PHET_CORE/merge' ); - const Property = require( 'AXON/Property' ); - const Screen = require( 'JOIST/Screen' ); - const Tandem = require( 'TANDEM/Tandem' ); - - // strings - const factorString = require( 'string!ARITHMETIC/factor' ); - - /** - * @param {Object} [options] - * @constructor - */ - function FactorScreen( options ) { - - options = merge( { - name: factorString, - homeScreenIcon: new FactorScreenIconNode(), - backgroundColorProperty: new Property( ArithmeticConstants.BACKGROUND_COLOR ), - tandem: Tandem.REQUIRED - }, options ); - - Screen.call( this, - function() { return new FactorModel( options.tandem.createTandem( 'model' ) ); }, - function( model ) { return new FactorView( model ); }, - options ); - } - - arithmetic.register( 'FactorScreen', FactorScreen ); - - return inherit( Screen, FactorScreen ); -} ); \ No newline at end of file + +import Property from '../../../axon/js/Property.js'; +import Screen from '../../../joist/js/Screen.js'; +import inherit from '../../../phet-core/js/inherit.js'; +import merge from '../../../phet-core/js/merge.js'; +import Tandem from '../../../tandem/js/Tandem.js'; +import arithmeticStrings from '../arithmetic-strings.js'; +import arithmetic from '../arithmetic.js'; +import ArithmeticConstants from '../common/ArithmeticConstants.js'; +import FactorModel from './model/FactorModel.js'; +import FactorScreenIconNode from './view/FactorScreenIconNode.js'; +import FactorView from './view/FactorView.js'; + +const factorString = arithmeticStrings.factor; + +/** + * @param {Object} [options] + * @constructor + */ +function FactorScreen( options ) { + + options = merge( { + name: factorString, + homeScreenIcon: new FactorScreenIconNode(), + backgroundColorProperty: new Property( ArithmeticConstants.BACKGROUND_COLOR ), + tandem: Tandem.REQUIRED + }, options ); + + Screen.call( this, + function() { return new FactorModel( options.tandem.createTandem( 'model' ) ); }, + function( model ) { return new FactorView( model ); }, + options ); +} + +arithmetic.register( 'FactorScreen', FactorScreen ); + +inherit( Screen, FactorScreen ); +export default FactorScreen; \ No newline at end of file diff --git a/js/factor/model/FactorModel.js b/js/factor/model/FactorModel.js index 0616e7c9..f7949f24 100644 --- a/js/factor/model/FactorModel.js +++ b/js/factor/model/FactorModel.js @@ -7,117 +7,112 @@ * @author John Blanco */ -define( require => { - 'use strict'; +import inherit from '../../../../phet-core/js/inherit.js'; +import soundManager from '../../../../tambo/js/soundManager.js'; +import arithmetic from '../../arithmetic.js'; +import ArithmeticModel from '../../common/model/ArithmeticModel.js'; +import GameState from '../../common/model/GameState.js'; - // modules - const arithmetic = require( 'ARITHMETIC/arithmetic' ); - const ArithmeticModel = require( 'ARITHMETIC/common/model/ArithmeticModel' ); - const GameState = require( 'ARITHMETIC/common/model/GameState' ); - const inherit = require( 'PHET_CORE/inherit' ); - const soundManager = require( 'TAMBO/soundManager' ); +/** + * @param {Tandem} tandem + * @constructor + */ +function FactorModel( tandem ) { + ArithmeticModel.call( this, tandem ); +} - /** - * @param {Tandem} tandem - * @constructor - */ - function FactorModel( tandem ) { - ArithmeticModel.call( this, tandem ); - } +arithmetic.register( 'FactorModel', FactorModel ); - arithmetic.register( 'FactorModel', FactorModel ); +export default inherit( ArithmeticModel, FactorModel, { - return inherit( ArithmeticModel, FactorModel, { + // @public + setUpUnansweredProblem: function() { - // @public - setUpUnansweredProblem: function() { + // get available multiplier pair + const multiplierPair = this.selectUnusedMultiplierPair(); - // get available multiplier pair - const multiplierPair = this.selectUnusedMultiplierPair(); + if ( multiplierPair ) { - if ( multiplierPair ) { + // reset multiplierPair and score properties + this.problemModel.possiblePointsProperty.reset(); + this.problemModel.multiplicandProperty.reset(); + this.problemModel.multiplierProperty.reset(); - // reset multiplierPair and score properties - this.problemModel.possiblePointsProperty.reset(); - this.problemModel.multiplicandProperty.reset(); - this.problemModel.multiplierProperty.reset(); + // set product + this.problemModel.productProperty.set( + multiplierPair.multiplicand * multiplierPair.multiplier + ); - // set product - this.problemModel.productProperty.set( - multiplierPair.multiplicand * multiplierPair.multiplier - ); + return true; + } - return true; - } + // All multiplier pairs have been used, so false is returned. + return false; + }, - // All multiplier pairs have been used, so false is returned. - return false; - }, - - /** - * Submit an answer for the currently active problem. This override exists to handle one very special case on the - * Factor screen: the situation where the user submits two or more incorrect answers in a row without pressing the - * "Try Again" button in between. In this case, there is no natural state transition, so the feedback sound is - * never played. This override forces the state transition. - * - * See https://github.com/phetsims/arithmetic/issues/160#issuecomment-164507798 for more. - * - * @override - * @public - */ - submitAnswer: function() { - if ( this.stateProperty.get() === GameState.DISPLAYING_INCORRECT_ANSWER_FEEDBACK ) { - - // force a change to the AWAITING_USER_INPUT state before checking the answer - const multiplicand = this.problemModel.multiplicandProperty.get(); - const multiplier = this.problemModel.multiplierProperty.get(); - this.retryProblem(); - this.problemModel.multiplicandProperty.set( multiplicand ); - this.problemModel.multiplierProperty.set( multiplier ); - } - ArithmeticModel.prototype.submitAnswer.call( this ); - }, - - /** - * Automatically answer most of the questions. This is useful for testing, since it can save time when testing - * how the sim behaves when a user finishing answering all questions for a level. We need to be very careful that - * this is never available in the published sim. - * @override - * @protected - */ - autoAnswer: function() { - - // make sure that sound is off, since otherwise it dings for every solved problem - const soundState = soundManager.enabled; - soundManager.enabled = false; - - // answer the questions - const self = this; - const tableSize = this.activeLevelModel.tableSize; - const numQuestions = tableSize * tableSize; - const numQuestionsToAnswer = numQuestions - 1; - const levelModel = this.activeLevelModel; // convenience var - console.log( 'Automatically answering', numQuestionsToAnswer, 'of', numQuestions, 'questions.' ); - _.times( numQuestionsToAnswer, function( index ) { - // do a brute-force factoring method, since performance isn't really an issue here - let answerFound = false; - for ( let multiplicand = 1; multiplicand <= tableSize && !answerFound; multiplicand++ ) { - for ( let multiplier = 1; multiplier <= tableSize && !answerFound; multiplier++ ) { - if ( multiplicand * multiplier === self.problemModel.productProperty.get() && !levelModel.isCellUsed( multiplicand, multiplier ) ) { - - answerFound = true; - levelModel.markCellAsUsed( multiplicand, multiplier ); - } + /** + * Submit an answer for the currently active problem. This override exists to handle one very special case on the + * Factor screen: the situation where the user submits two or more incorrect answers in a row without pressing the + * "Try Again" button in between. In this case, there is no natural state transition, so the feedback sound is + * never played. This override forces the state transition. + * + * See https://github.com/phetsims/arithmetic/issues/160#issuecomment-164507798 for more. + * + * @override + * @public + */ + submitAnswer: function() { + if ( this.stateProperty.get() === GameState.DISPLAYING_INCORRECT_ANSWER_FEEDBACK ) { + + // force a change to the AWAITING_USER_INPUT state before checking the answer + const multiplicand = this.problemModel.multiplicandProperty.get(); + const multiplier = this.problemModel.multiplierProperty.get(); + this.retryProblem(); + this.problemModel.multiplicandProperty.set( multiplicand ); + this.problemModel.multiplierProperty.set( multiplier ); + } + ArithmeticModel.prototype.submitAnswer.call( this ); + }, + + /** + * Automatically answer most of the questions. This is useful for testing, since it can save time when testing + * how the sim behaves when a user finishing answering all questions for a level. We need to be very careful that + * this is never available in the published sim. + * @override + * @protected + */ + autoAnswer: function() { + + // make sure that sound is off, since otherwise it dings for every solved problem + const soundState = soundManager.enabled; + soundManager.enabled = false; + + // answer the questions + const self = this; + const tableSize = this.activeLevelModel.tableSize; + const numQuestions = tableSize * tableSize; + const numQuestionsToAnswer = numQuestions - 1; + const levelModel = this.activeLevelModel; // convenience var + console.log( 'Automatically answering', numQuestionsToAnswer, 'of', numQuestions, 'questions.' ); + _.times( numQuestionsToAnswer, function( index ) { + // do a brute-force factoring method, since performance isn't really an issue here + let answerFound = false; + for ( let multiplicand = 1; multiplicand <= tableSize && !answerFound; multiplicand++ ) { + for ( let multiplier = 1; multiplier <= tableSize && !answerFound; multiplier++ ) { + if ( multiplicand * multiplier === self.problemModel.productProperty.get() && !levelModel.isCellUsed( multiplicand, multiplier ) ) { + + answerFound = true; + levelModel.markCellAsUsed( multiplicand, multiplier ); } } - levelModel.currentScoreProperty.value += self.problemModel.possiblePointsProperty.get(); - levelModel.displayScoreProperty.set( self.activeLevelModel.currentScoreProperty.get() ); - self.stateProperty.set( GameState.DISPLAYING_CORRECT_ANSWER_FEEDBACK ); - self.nextProblem(); - } ); - - // restore the original sound state - soundManager.enabled = soundState; - } - } ); -} ); + } + levelModel.currentScoreProperty.value += self.problemModel.possiblePointsProperty.get(); + levelModel.displayScoreProperty.set( self.activeLevelModel.currentScoreProperty.get() ); + self.stateProperty.set( GameState.DISPLAYING_CORRECT_ANSWER_FEEDBACK ); + self.nextProblem(); + } ); + + // restore the original sound state + soundManager.enabled = soundState; + } +} ); \ No newline at end of file diff --git a/js/factor/view/CellInteractionListener.js b/js/factor/view/CellInteractionListener.js index 36f9cbca..89bd6560 100644 --- a/js/factor/view/CellInteractionListener.js +++ b/js/factor/view/CellInteractionListener.js @@ -9,67 +9,63 @@ * * @author John Blanco */ -define( require => { - 'use strict'; - // modules - const arithmetic = require( 'ARITHMETIC/arithmetic' ); - const Emitter = require( 'AXON/Emitter' ); - const inherit = require( 'PHET_CORE/inherit' ); - const Property = require( 'AXON/Property' ); +import Emitter from '../../../../axon/js/Emitter.js'; +import Property from '../../../../axon/js/Property.js'; +import inherit from '../../../../phet-core/js/inherit.js'; +import arithmetic from '../../arithmetic.js'; - /** - * @constructor - */ - function CellInteractionListener() { - this.mouseOverProperty = new Property( false ); - this.touchedProperty = new Property( false ); - this.enabledProperty = new Property( true ); +/** + * @constructor + */ +function CellInteractionListener() { + this.mouseOverProperty = new Property( false ); + this.touchedProperty = new Property( false ); + this.enabledProperty = new Property( true ); - this.mouseDownEmitter = new Emitter(); - this.mouseUpEmitter = new Emitter(); - this.touchUpEmitter = new Emitter(); - } + this.mouseDownEmitter = new Emitter(); + this.mouseUpEmitter = new Emitter(); + this.touchUpEmitter = new Emitter(); +} - arithmetic.register( 'CellInteractionListener', CellInteractionListener ); +arithmetic.register( 'CellInteractionListener', CellInteractionListener ); - return inherit( Object, CellInteractionListener, { +export default inherit( Object, CellInteractionListener, { - // @public - enter: function( event, trail ) { - if ( event.pointer.type === 'mouse' ) { - this.mouseOverProperty.set( true ); - } - else if ( event.pointer.type === 'touch' ) { - this.touchedProperty.set( true ); - } - }, + // @public + enter: function( event, trail ) { + if ( event.pointer.type === 'mouse' ) { + this.mouseOverProperty.set( true ); + } + else if ( event.pointer.type === 'touch' ) { + this.touchedProperty.set( true ); + } + }, - // @public - exit: function( event, trail ) { - if ( event.pointer.type === 'mouse' ) { - this.mouseOverProperty.set( false ); - } - else if ( event.pointer.type === 'touch' ) { - this.touchedProperty.set( false ); - } - }, + // @public + exit: function( event, trail ) { + if ( event.pointer.type === 'mouse' ) { + this.mouseOverProperty.set( false ); + } + else if ( event.pointer.type === 'touch' ) { + this.touchedProperty.set( false ); + } + }, - // @public - down: function( event, trail ) { - if ( event.pointer.type === 'mouse' ) { - this.mouseDownEmitter.emit(); - } - }, + // @public + down: function( event, trail ) { + if ( event.pointer.type === 'mouse' ) { + this.mouseDownEmitter.emit(); + } + }, - // @public - up: function( event, trail ) { - if ( event.pointer.type === 'mouse' ) { - this.mouseUpEmitter.emit(); - } - else if ( event.pointer.type === 'touch' ) { - this.touchUpEmitter.emit(); - } + // @public + up: function( event, trail ) { + if ( event.pointer.type === 'mouse' ) { + this.mouseUpEmitter.emit(); } - } ); + else if ( event.pointer.type === 'touch' ) { + this.touchUpEmitter.emit(); + } + } } ); \ No newline at end of file diff --git a/js/factor/view/FactorEquationNode.js b/js/factor/view/FactorEquationNode.js index ce9cc882..eb63771a 100644 --- a/js/factor/view/FactorEquationNode.js +++ b/js/factor/view/FactorEquationNode.js @@ -5,47 +5,44 @@ * * @author Andrey Zelenkov (MLearner) */ -define( require => { - 'use strict'; - - // modules - const arithmetic = require( 'ARITHMETIC/arithmetic' ); - const EquationNode = require( 'ARITHMETIC/common/view/EquationNode' ); - const GameState = require( 'ARITHMETIC/common/model/GameState' ); - const inherit = require( 'PHET_CORE/inherit' ); - - /** - * @param {Property} stateProperty - State of game property. - * @param {Property} multiplicandProperty - Property necessary for creating multiplicand input. - * @param {Property} multiplierProperty - Property necessary for creating multiplier input. - * @param {Property} productProperty - Property necessary for creating product input. - * @constructor - */ - function FactorEquationNode( stateProperty, multiplicandProperty, multiplierProperty, productProperty ) { - const self = this; - EquationNode.call( this, multiplicandProperty, multiplierProperty, productProperty ); - - // The two multipliers are always interactive in the factor equation, so set this up now. - this.multiplicandInput.setInteractiveAppearance( true ); - this.multiplierInput.setInteractiveAppearance( true ); - - // Update contents and focus at the state changes. - stateProperty.link( function( state ) { - self.setShowEqual( state !== GameState.DISPLAYING_INCORRECT_ANSWER_FEEDBACK ); - if ( state === GameState.AWAITING_USER_INPUT ) { - - // Reset any previous answers from the user. - multiplicandProperty.reset(); - multiplierProperty.reset(); - - // Show the placeholders - self.multiplicandInput.setPlaceholder(); - self.multiplierInput.setPlaceholder(); - } - } ); - } - - arithmetic.register( 'FactorEquationNode', FactorEquationNode ); - - return inherit( EquationNode, FactorEquationNode ); -} ); + +import inherit from '../../../../phet-core/js/inherit.js'; +import arithmetic from '../../arithmetic.js'; +import GameState from '../../common/model/GameState.js'; +import EquationNode from '../../common/view/EquationNode.js'; + +/** + * @param {Property} stateProperty - State of game property. + * @param {Property} multiplicandProperty - Property necessary for creating multiplicand input. + * @param {Property} multiplierProperty - Property necessary for creating multiplier input. + * @param {Property} productProperty - Property necessary for creating product input. + * @constructor + */ +function FactorEquationNode( stateProperty, multiplicandProperty, multiplierProperty, productProperty ) { + const self = this; + EquationNode.call( this, multiplicandProperty, multiplierProperty, productProperty ); + + // The two multipliers are always interactive in the factor equation, so set this up now. + this.multiplicandInput.setInteractiveAppearance( true ); + this.multiplierInput.setInteractiveAppearance( true ); + + // Update contents and focus at the state changes. + stateProperty.link( function( state ) { + self.setShowEqual( state !== GameState.DISPLAYING_INCORRECT_ANSWER_FEEDBACK ); + if ( state === GameState.AWAITING_USER_INPUT ) { + + // Reset any previous answers from the user. + multiplicandProperty.reset(); + multiplierProperty.reset(); + + // Show the placeholders + self.multiplicandInput.setPlaceholder(); + self.multiplierInput.setPlaceholder(); + } + } ); +} + +arithmetic.register( 'FactorEquationNode', FactorEquationNode ); + +inherit( EquationNode, FactorEquationNode ); +export default FactorEquationNode; \ No newline at end of file diff --git a/js/factor/view/FactorScreenIconNode.js b/js/factor/view/FactorScreenIconNode.js index 8df3b358..43bf9a20 100644 --- a/js/factor/view/FactorScreenIconNode.js +++ b/js/factor/view/FactorScreenIconNode.js @@ -6,70 +6,67 @@ * @author Andrey Zelenkov (MLearner) * @author John Blanco (MLearner) */ -define( require => { - 'use strict'; - // modules - const arithmetic = require( 'ARITHMETIC/arithmetic' ); - const ArithmeticConstants = require( 'ARITHMETIC/common/ArithmeticConstants' ); - const inherit = require( 'PHET_CORE/inherit' ); - const Line = require( 'SCENERY/nodes/Line' ); - const PhetFont = require( 'SCENERY_PHET/PhetFont' ); - const Rectangle = require( 'SCENERY/nodes/Rectangle' ); - const Text = require( 'SCENERY/nodes/Text' ); +import inherit from '../../../../phet-core/js/inherit.js'; +import PhetFont from '../../../../scenery-phet/js/PhetFont.js'; +import Line from '../../../../scenery/js/nodes/Line.js'; +import Rectangle from '../../../../scenery/js/nodes/Rectangle.js'; +import Text from '../../../../scenery/js/nodes/Text.js'; +import arithmetic from '../../arithmetic.js'; +import ArithmeticConstants from '../../common/ArithmeticConstants.js'; - // constants - const ICON_SIZE = ArithmeticConstants.SCREEN_ICON_SIZE; - const CONNECTING_LINES_COLOR = '#FFF31E'; - const CONNECTING_LINE_WIDTH = 15; - const NUMBER_FONT = new PhetFont( 90 ); // Font size empirically determined - const QUESTION_MARK_FONT = new PhetFont( { size: 120, weight: 'bold' } ); // Font size empirically determined - const BOX_VERTICAL_INSET = 25; // Empirically determined - const CONNECTING_LINES_OPTIONS = { - stroke: CONNECTING_LINES_COLOR, - lineWidth: CONNECTING_LINE_WIDTH, - lineCap: 'round' - }; +// constants +const ICON_SIZE = ArithmeticConstants.SCREEN_ICON_SIZE; +const CONNECTING_LINES_COLOR = '#FFF31E'; +const CONNECTING_LINE_WIDTH = 15; +const NUMBER_FONT = new PhetFont( 90 ); // Font size empirically determined +const QUESTION_MARK_FONT = new PhetFont( { size: 120, weight: 'bold' } ); // Font size empirically determined +const BOX_VERTICAL_INSET = 25; // Empirically determined +const CONNECTING_LINES_OPTIONS = { + stroke: CONNECTING_LINES_COLOR, + lineWidth: CONNECTING_LINE_WIDTH, + lineCap: 'round' +}; - // utility function for creating a rectangle with text in it. - function createRectangleWithEnclosedText( text, font, xMargin, yMargin ) { - const textNode = new Text( text, { font: font } ); - const box = new Rectangle( 0, 0, textNode.width + 2 * xMargin, textNode.height + 2 * yMargin, 20, 20, { fill: 'white' } ); - textNode.center = box.center; - box.addChild( textNode ); - return box; - } +// utility function for creating a rectangle with text in it. +function createRectangleWithEnclosedText( text, font, xMargin, yMargin ) { + const textNode = new Text( text, { font: font } ); + const box = new Rectangle( 0, 0, textNode.width + 2 * xMargin, textNode.height + 2 * yMargin, 20, 20, { fill: 'white' } ); + textNode.center = box.center; + box.addChild( textNode ); + return box; +} - /** - * @constructor - */ - function FactorScreenIconNode() { +/** + * @constructor + */ +function FactorScreenIconNode() { - // create the background - Rectangle.call( this, 0, 0, ICON_SIZE.width, ICON_SIZE.height, { fill: ArithmeticConstants.ICON_BACKGROUND_COLOR } ); + // create the background + Rectangle.call( this, 0, 0, ICON_SIZE.width, ICON_SIZE.height, { fill: ArithmeticConstants.ICON_BACKGROUND_COLOR } ); - // Create and position the boxes, but don't add them yet so that we can get the layering right. - const topBox = createRectangleWithEnclosedText( '12', NUMBER_FONT, 15, 5 ); - topBox.centerX = this.width / 2; - topBox.top = BOX_VERTICAL_INSET; - const multiplicandBox = createRectangleWithEnclosedText( '?', QUESTION_MARK_FONT, 20, 5 ); - multiplicandBox.centerX = ICON_SIZE.width * 0.3; - multiplicandBox.bottom = ICON_SIZE.height - BOX_VERTICAL_INSET; - const multiplierBox = createRectangleWithEnclosedText( '?', QUESTION_MARK_FONT, 20, 5 ); - multiplierBox.centerX = ICON_SIZE.width * 0.7; - multiplierBox.bottom = multiplicandBox.bottom; + // Create and position the boxes, but don't add them yet so that we can get the layering right. + const topBox = createRectangleWithEnclosedText( '12', NUMBER_FONT, 15, 5 ); + topBox.centerX = this.width / 2; + topBox.top = BOX_VERTICAL_INSET; + const multiplicandBox = createRectangleWithEnclosedText( '?', QUESTION_MARK_FONT, 20, 5 ); + multiplicandBox.centerX = ICON_SIZE.width * 0.3; + multiplicandBox.bottom = ICON_SIZE.height - BOX_VERTICAL_INSET; + const multiplierBox = createRectangleWithEnclosedText( '?', QUESTION_MARK_FONT, 20, 5 ); + multiplierBox.centerX = ICON_SIZE.width * 0.7; + multiplierBox.bottom = multiplicandBox.bottom; - // Add the connecting lines - this.addChild( new Line( topBox.centerX, topBox.bottom, multiplicandBox.centerX, multiplicandBox.top, CONNECTING_LINES_OPTIONS ) ); - this.addChild( new Line( topBox.centerX, topBox.bottom, multiplierBox.centerX, multiplierBox.top, CONNECTING_LINES_OPTIONS ) ); + // Add the connecting lines + this.addChild( new Line( topBox.centerX, topBox.bottom, multiplicandBox.centerX, multiplicandBox.top, CONNECTING_LINES_OPTIONS ) ); + this.addChild( new Line( topBox.centerX, topBox.bottom, multiplierBox.centerX, multiplierBox.top, CONNECTING_LINES_OPTIONS ) ); - // Add the text boxes - this.addChild( topBox ); - this.addChild( multiplicandBox ); - this.addChild( multiplierBox ); - } + // Add the text boxes + this.addChild( topBox ); + this.addChild( multiplicandBox ); + this.addChild( multiplierBox ); +} - arithmetic.register( 'FactorScreenIconNode', FactorScreenIconNode ); +arithmetic.register( 'FactorScreenIconNode', FactorScreenIconNode ); - return inherit( Rectangle, FactorScreenIconNode ); -} ); +inherit( Rectangle, FactorScreenIconNode ); +export default FactorScreenIconNode; \ No newline at end of file diff --git a/js/factor/view/FactorScreenTableNode.js b/js/factor/view/FactorScreenTableNode.js index 88793b9b..89aa57f8 100644 --- a/js/factor/view/FactorScreenTableNode.js +++ b/js/factor/view/FactorScreenTableNode.js @@ -6,258 +6,252 @@ * @author Andrey Zelenkov (MLearner) * @author John Blanco */ -define( require => { - 'use strict'; - - // modules - const arithmetic = require( 'ARITHMETIC/arithmetic' ); - const CellInteractionListener = require( 'ARITHMETIC/factor/view/CellInteractionListener' ); - const GameState = require( 'ARITHMETIC/common/model/GameState' ); - const Image = require( 'SCENERY/nodes/Image' ); - const inherit = require( 'PHET_CORE/inherit' ); - const MultiplicationTableNode = require( 'ARITHMETIC/common/view/table/MultiplicationTableNode' ); - - // images - const cellPointerHandImage = require( 'image!ARITHMETIC/small-pointing-hand.png' ); - const overlayPointingHandImage = require( 'image!ARITHMETIC/large-pointing-hand.png' ); - - /** - * @param {FactorModel} model - main model class for the factor screen - * @constructor - */ - function FactorScreenTableNode( model ) { - const self = this; - MultiplicationTableNode.call( this, model.levelNumberProperty, model.stateProperty, model.levelModels, false ); - - // convenience var - const gameState = model.stateProperty; - - // Create an image of a transparent hand that will cue the user that they need to interact with the table. - const handImage = new Image( overlayPointingHandImage, { pickable: false } ); // @private - handImage.scale( ( this.width / overlayPointingHandImage.width ) * 0.25 ); - handImage.centerX = this.width * 0.55; // position empirically determined - handImage.centerY = this.height / 2; - - // Create another hand that will appear over each cell to indicate that the user can click on it. This was - // originally handled in the individual cells, but caused startup to be long due to the large number of images - // created, so was moved here. - this.cellPointer = new Image( cellPointerHandImage, { pickable: false } ); // @private - this.addChild( this.cellPointer ); - - // variables used to track cell interaction - this.cellListeners = []; // @private - this.activeCell = null; // @private - this.mouseDownCell = null; // @private - - // add 'hover' and 'down' listeners for each cell in table - this.cells.forEach( function( tableForLevel, levelIndex ) { - - self.cellListeners[ levelIndex ] = []; - - tableForLevel.forEach( function( multiplicandRow, multiplicandRowIndex ) { - - // skip zero-index because it's the header column - if ( multiplicandRowIndex ) { - - multiplicandRow.forEach( function( cell, multiplierIndex ) { - let cellListener; - - // skip zero-index because it's the header row - if ( multiplierIndex ) { - cellListener = new CellInteractionListener(); - cell.addInputListener( cellListener ); - cell.cursor = 'pointer'; - - // store cell listeners for each level - self.cellListeners[ levelIndex ].push( cellListener ); - - const updateHover = function() { - if ( model.stateProperty.get() === GameState.AWAITING_USER_INPUT ) { - self.setCellsToDefaultColor( model.levelNumberProperty.get() ); - if ( cellListener.enabledProperty.get() ) { - self.setSelectedRect( model.levelNumberProperty.get(), multiplicandRowIndex, multiplierIndex ); - cell.setHover(); - self.cellPointer.visible = true; - - if ( Math.abs( self.cellPointer.height - cell.height * 0.7 ) > 0.01 ) { - self.cellPointer.setScaleMagnitude( 1 ); - self.cellPointer.setScaleMagnitude( cell.height * 0.7 / self.cellPointer.height ); - } - self.cellPointer.centerX = cell.centerX; - self.cellPointer.centerY = cell.centerY; - self.activeCell = cell; - } - else { - self.activeCell = null; - } - } - }; - - // add 'hover' listeners - cellListener.mouseOverProperty.link( updateHover ); - cellListener.touchedProperty.link( function( touched ) { - if ( touched ) { - if ( model.stateProperty.get() === GameState.DISPLAYING_INCORRECT_ANSWER_FEEDBACK ) { - // The user has re-touched the grid after submitting an incorrect answer, so assume they want to retry. - model.retryProblem(); - } - handImage.visible = false; // stop showing hand after first interaction - updateHover(); - } - } ); - // When the user presses the mouse button, record it. - cellListener.mouseDownEmitter.addListener( function() { - self.mouseDownCell = cell; - self.activeCell = cell; - handImage.visible = false; // stop showing hand after first interaction - updateHover(); - } ); +import inherit from '../../../../phet-core/js/inherit.js'; +import Image from '../../../../scenery/js/nodes/Image.js'; +import overlayPointingHandImage from '../../../images/large-pointing-hand_png.js'; +import cellPointerHandImage from '../../../images/small-pointing-hand_png.js'; +import arithmetic from '../../arithmetic.js'; +import GameState from '../../common/model/GameState.js'; +import MultiplicationTableNode from '../../common/view/table/MultiplicationTableNode.js'; +import CellInteractionListener from './CellInteractionListener.js'; - // Define a function for submitting an answer that can be used by both mouse and the touch handlers. - const submitAnswer = function() { +/** + * @param {FactorModel} model - main model class for the factor screen + * @constructor + */ +function FactorScreenTableNode( model ) { + const self = this; + MultiplicationTableNode.call( this, model.levelNumberProperty, model.stateProperty, model.levelModels, false ); - // Record the user's answer. - model.problemModel.multiplicandProperty.set( multiplicandRowIndex ); - model.problemModel.multiplierProperty.set( multiplierIndex ); + // convenience var + const gameState = model.stateProperty; - // Disable this cell if the user's answer is correct. - if ( multiplicandRowIndex * multiplierIndex === model.problemModel.productProperty.get() ) { - cellListener.enabledProperty.set( false ); - } + // Create an image of a transparent hand that will cue the user that they need to interact with the table. + const handImage = new Image( overlayPointingHandImage, { pickable: false } ); // @private + handImage.scale( ( this.width / overlayPointingHandImage.width ) * 0.25 ); + handImage.centerX = this.width * 0.55; // position empirically determined + handImage.centerY = this.height / 2; - // Submit the user's answer for checking. - model.submitAnswer(); + // Create another hand that will appear over each cell to indicate that the user can click on it. This was + // originally handled in the individual cells, but caused startup to be long due to the large number of images + // created, so was moved here. + this.cellPointer = new Image( cellPointerHandImage, { pickable: false } ); // @private + this.addChild( this.cellPointer ); - // Update the cell highlighting to match the latest submission, which may be necessary if the user - // submitted a new answer after first submitting one or more incorrect ones. - self.setSelectedRect( model.levelNumberProperty.get(), multiplicandRowIndex, multiplierIndex ); - }; + // variables used to track cell interaction + this.cellListeners = []; // @private + this.activeCell = null; // @private + this.mouseDownCell = null; // @private - // When the user releases the mouse button, check that it's the same cell where the mouse down occurred, - // and fire if so. - cellListener.mouseUpEmitter.addListener( function() { - if ( cellListener.enabledProperty.get() && self.mouseDownCell === cell && - ( gameState.value === GameState.AWAITING_USER_INPUT || - gameState.value === GameState.DISPLAYING_INCORRECT_ANSWER_FEEDBACK ) ) { - submitAnswer(); - } - } ); + // add 'hover' and 'down' listeners for each cell in table + this.cells.forEach( function( tableForLevel, levelIndex ) { + + self.cellListeners[ levelIndex ] = []; + + tableForLevel.forEach( function( multiplicandRow, multiplicandRowIndex ) { + + // skip zero-index because it's the header column + if ( multiplicandRowIndex ) { + + multiplicandRow.forEach( function( cell, multiplierIndex ) { + let cellListener; + + // skip zero-index because it's the header row + if ( multiplierIndex ) { + cellListener = new CellInteractionListener(); + cell.addInputListener( cellListener ); + cell.cursor = 'pointer'; - // Add listener for handling the event where the user was touching and lifts their finger. - cellListener.touchUpEmitter.addListener( function() { - // It takes two touchUp events in a row from the same cell to submit an answer. + // store cell listeners for each level + self.cellListeners[ levelIndex ].push( cellListener ); + + const updateHover = function() { + if ( model.stateProperty.get() === GameState.AWAITING_USER_INPUT ) { + self.setCellsToDefaultColor( model.levelNumberProperty.get() ); if ( cellListener.enabledProperty.get() ) { - if ( self.touchUpCell === cell && gameState.value === GameState.AWAITING_USER_INPUT ) { - submitAnswer(); - } - else { - self.touchUpCell = cell; + self.setSelectedRect( model.levelNumberProperty.get(), multiplicandRowIndex, multiplierIndex ); + cell.setHover(); + self.cellPointer.visible = true; + + if ( Math.abs( self.cellPointer.height - cell.height * 0.7 ) > 0.01 ) { + self.cellPointer.setScaleMagnitude( 1 ); + self.cellPointer.setScaleMagnitude( cell.height * 0.7 / self.cellPointer.height ); } + self.cellPointer.centerX = cell.centerX; + self.cellPointer.centerY = cell.centerY; + self.activeCell = cell; } - } ); - - // cancel hover for disabled cell before next task - model.stateProperty.lazyLink( function( state ) { - if ( state === GameState.AWAITING_USER_INPUT && !cellListener.enabledProperty.get() ) { - self.setCellsToDefaultColor( model.levelNumberProperty.get() ); + else { + self.activeCell = null; + } + } + }; + + // add 'hover' listeners + cellListener.mouseOverProperty.link( updateHover ); + cellListener.touchedProperty.link( function( touched ) { + if ( touched ) { + if ( model.stateProperty.get() === GameState.DISPLAYING_INCORRECT_ANSWER_FEEDBACK ) { + // The user has re-touched the grid after submitting an incorrect answer, so assume they want to retry. + model.retryProblem(); + } + handImage.visible = false; // stop showing hand after first interaction + updateHover(); + } + } ); + + // When the user presses the mouse button, record it. + cellListener.mouseDownEmitter.addListener( function() { + self.mouseDownCell = cell; + self.activeCell = cell; + handImage.visible = false; // stop showing hand after first interaction + updateHover(); + } ); + + // Define a function for submitting an answer that can be used by both mouse and the touch handlers. + const submitAnswer = function() { + + // Record the user's answer. + model.problemModel.multiplicandProperty.set( multiplicandRowIndex ); + model.problemModel.multiplierProperty.set( multiplierIndex ); + + // Disable this cell if the user's answer is correct. + if ( multiplicandRowIndex * multiplierIndex === model.problemModel.productProperty.get() ) { + cellListener.enabledProperty.set( false ); + } + + // Submit the user's answer for checking. + model.submitAnswer(); + + // Update the cell highlighting to match the latest submission, which may be necessary if the user + // submitted a new answer after first submitting one or more incorrect ones. + self.setSelectedRect( model.levelNumberProperty.get(), multiplicandRowIndex, multiplierIndex ); + }; + + // When the user releases the mouse button, check that it's the same cell where the mouse down occurred, + // and fire if so. + cellListener.mouseUpEmitter.addListener( function() { + if ( cellListener.enabledProperty.get() && self.mouseDownCell === cell && + ( gameState.value === GameState.AWAITING_USER_INPUT || + gameState.value === GameState.DISPLAYING_INCORRECT_ANSWER_FEEDBACK ) ) { + submitAnswer(); + } + } ); + + // Add listener for handling the event where the user was touching and lifts their finger. + cellListener.touchUpEmitter.addListener( function() { + // It takes two touchUp events in a row from the same cell to submit an answer. + if ( cellListener.enabledProperty.get() ) { + if ( self.touchUpCell === cell && gameState.value === GameState.AWAITING_USER_INPUT ) { + submitAnswer(); } - } ); - } - } ); - } - } ); + else { + self.touchUpCell = cell; + } + } + } ); + + // cancel hover for disabled cell before next task + model.stateProperty.lazyLink( function( state ) { + if ( state === GameState.AWAITING_USER_INPUT && !cellListener.enabledProperty.get() ) { + self.setCellsToDefaultColor( model.levelNumberProperty.get() ); + } + } ); + } + } ); + } } ); + } ); - // Add the hand image here for proper layering. - this.addChild( handImage ); + // Add the hand image here for proper layering. + this.addChild( handImage ); - // Update the cell's appearance and state as the game state changes. - model.stateProperty.link( function( newState, oldState ) { + // Update the cell's appearance and state as the game state changes. + model.stateProperty.link( function( newState, oldState ) { - if ( oldState === GameState.SELECTING_LEVEL && newState === GameState.AWAITING_USER_INPUT ) { - self.setCellsToDefaultColor( model.levelNumberProperty.get() ); - self.updateCellListenerEnabledStates( model.levelNumberProperty.get(), model.activeLevelModel ); - } - else if ( ( newState === GameState.DISPLAYING_CORRECT_ANSWER_FEEDBACK || - newState === GameState.DISPLAYING_INCORRECT_ANSWER_FEEDBACK ) && - self.activeCell !== null ) { + if ( oldState === GameState.SELECTING_LEVEL && newState === GameState.AWAITING_USER_INPUT ) { + self.setCellsToDefaultColor( model.levelNumberProperty.get() ); + self.updateCellListenerEnabledStates( model.levelNumberProperty.get(), model.activeLevelModel ); + } + else if ( ( newState === GameState.DISPLAYING_CORRECT_ANSWER_FEEDBACK || + newState === GameState.DISPLAYING_INCORRECT_ANSWER_FEEDBACK ) && + self.activeCell !== null ) { - // Cancel hover when showing feedback - self.activeCell.setSelected(); - } + // Cancel hover when showing feedback + self.activeCell.setSelected(); + } - if ( newState === GameState.LEVEL_COMPLETED || - ( oldState === GameState.DISPLAYING_INCORRECT_ANSWER_FEEDBACK && - newState === GameState.AWAITING_USER_INPUT ) ) { + if ( newState === GameState.LEVEL_COMPLETED || + ( oldState === GameState.DISPLAYING_INCORRECT_ANSWER_FEEDBACK && + newState === GameState.AWAITING_USER_INPUT ) ) { - // clear previously selected region - self.setCellsToDefaultColor( model.levelNumberProperty.get() ); - } + // clear previously selected region + self.setCellsToDefaultColor( model.levelNumberProperty.get() ); + } - // hide the pointer when showing correct or incorrect answer feedback - if ( newState === GameState.DISPLAYING_CORRECT_ANSWER_FEEDBACK || - newState === GameState.DISPLAYING_INCORRECT_ANSWER_FEEDBACK ) { - self.cellPointer.visible = false; - } + // hide the pointer when showing correct or incorrect answer feedback + if ( newState === GameState.DISPLAYING_CORRECT_ANSWER_FEEDBACK || + newState === GameState.DISPLAYING_INCORRECT_ANSWER_FEEDBACK ) { + self.cellPointer.visible = false; + } + + // don't allow interaction when displaying a completed board + self.pickable = newState !== GameState.LEVEL_COMPLETED; + } ); +} + +arithmetic.register( 'FactorScreenTableNode', FactorScreenTableNode ); + +export default inherit( MultiplicationTableNode, FactorScreenTableNode, { - // don't allow interaction when displaying a completed board - self.pickable = newState !== GameState.LEVEL_COMPLETED; + // @private, enable all cells for given level + enableAllCells: function( levelNumber ) { + this.cellListeners[ levelNumber ].forEach( function( cellListener ) { + cellListener.enabledProperty.set( true ); } ); - } + }, - arithmetic.register( 'FactorScreenTableNode', FactorScreenTableNode ); - - return inherit( MultiplicationTableNode, FactorScreenTableNode, { - - // @private, enable all cells for given level - enableAllCells: function( levelNumber ) { - this.cellListeners[ levelNumber ].forEach( function( cellListener ) { - cellListener.enabledProperty.set( true ); - } ); - }, - - // @private, enabled or disable the cell listeners based on whether or not the cell has been used - updateCellListenerEnabledStates: function( levelNumber, levelModel ) { - const self = this; - const tableSize = levelModel.tableSize; - for ( let multiplicand = 1; multiplicand <= tableSize; multiplicand++ ) { - for ( let multiplier = 1; multiplier <= tableSize; multiplier++ ) { - self.cellListeners[ levelNumber ][ ( multiplicand - 1 ) * tableSize + ( multiplier - 1 ) ].enabledProperty.set( - !levelModel.isCellUsed( multiplicand, multiplier ) - ); - } + // @private, enabled or disable the cell listeners based on whether or not the cell has been used + updateCellListenerEnabledStates: function( levelNumber, levelModel ) { + const self = this; + const tableSize = levelModel.tableSize; + for ( let multiplicand = 1; multiplicand <= tableSize; multiplicand++ ) { + for ( let multiplier = 1; multiplier <= tableSize; multiplier++ ) { + self.cellListeners[ levelNumber ][ ( multiplicand - 1 ) * tableSize + ( multiplier - 1 ) ].enabledProperty.set( + !levelModel.isCellUsed( multiplicand, multiplier ) + ); } - }, - - // @private, set 'selected' state for all cells in given bounds and highlight the multipliers (i.e. header cells) - setSelectedRect: function( levelNumber, leftBound, rightBound ) { - // highlight multipliers - this.cells[ levelNumber ][ 0 ][ rightBound ].setSelected(); - this.cells[ levelNumber ][ leftBound ][ 0 ].setSelected(); - - // set 'selected' state for all cell in given bounds - this.cells[ levelNumber ].forEach( function( multiplicands, multiplicandIndex ) { - if ( multiplicandIndex && multiplicandIndex <= leftBound ) { - multiplicands.forEach( function( cell, multiplierIndex ) { - if ( multiplierIndex && multiplierIndex <= rightBound ) { - cell.setSelected(); - } - } ); - } - } ); - }, - - // @public, @override - setCellsToDefaultColor: function( level ) { - this.cellPointer.visible = false; - MultiplicationTableNode.prototype.setCellsToDefaultColor.call( this, level ); - }, - - // @public, @override - refreshLevel: function( level ) { - MultiplicationTableNode.prototype.refreshLevel.call( this, level ); - this.enableAllCells( level ); } - } ); -} ); + }, + + // @private, set 'selected' state for all cells in given bounds and highlight the multipliers (i.e. header cells) + setSelectedRect: function( levelNumber, leftBound, rightBound ) { + // highlight multipliers + this.cells[ levelNumber ][ 0 ][ rightBound ].setSelected(); + this.cells[ levelNumber ][ leftBound ][ 0 ].setSelected(); + + // set 'selected' state for all cell in given bounds + this.cells[ levelNumber ].forEach( function( multiplicands, multiplicandIndex ) { + if ( multiplicandIndex && multiplicandIndex <= leftBound ) { + multiplicands.forEach( function( cell, multiplierIndex ) { + if ( multiplierIndex && multiplierIndex <= rightBound ) { + cell.setSelected(); + } + } ); + } + } ); + }, + + // @public, @override + setCellsToDefaultColor: function( level ) { + this.cellPointer.visible = false; + MultiplicationTableNode.prototype.setCellsToDefaultColor.call( this, level ); + }, + + // @public, @override + refreshLevel: function( level ) { + MultiplicationTableNode.prototype.refreshLevel.call( this, level ); + this.enableAllCells( level ); + } +} ); \ No newline at end of file diff --git a/js/factor/view/FactorView.js b/js/factor/view/FactorView.js index 0bdeb04b..a28883d6 100644 --- a/js/factor/view/FactorView.js +++ b/js/factor/view/FactorView.js @@ -5,44 +5,41 @@ * * @author Andrey Zelenkov (MLearner) */ -define( require => { - 'use strict'; - // modules - const arithmetic = require( 'ARITHMETIC/arithmetic' ); - const ArithmeticView = require( 'ARITHMETIC/common/view/ArithmeticView' ); - const FactorEquationNode = require( 'ARITHMETIC/factor/view/FactorEquationNode' ); - const FactorScreenTableNode = require( 'ARITHMETIC/factor/view/FactorScreenTableNode' ); - const inherit = require( 'PHET_CORE/inherit' ); +import inherit from '../../../../phet-core/js/inherit.js'; +import arithmeticStrings from '../../arithmetic-strings.js'; +import arithmetic from '../../arithmetic.js'; +import ArithmeticView from '../../common/view/ArithmeticView.js'; +import FactorEquationNode from './FactorEquationNode.js'; +import FactorScreenTableNode from './FactorScreenTableNode.js'; - // strings - const factorString = require( 'string!ARITHMETIC/factor' ); +const factorString = arithmeticStrings.factor; - /** - * @param {FactorModel} model - Main model for screen. - * @constructor - */ - function FactorView( model ) { - ArithmeticView.call( - this, - model, - new FactorScreenTableNode( model ), - new FactorEquationNode( - model.stateProperty, - model.problemModel.multiplicandProperty, - model.problemModel.multiplierProperty, - model.problemModel.productProperty - ), - { - showKeypad: false, - titleString: factorString, - levelSelectButtonColor: '#FFC266', - levelSelectIconSet: 'factor' - } - ); - } +/** + * @param {FactorModel} model - Main model for screen. + * @constructor + */ +function FactorView( model ) { + ArithmeticView.call( + this, + model, + new FactorScreenTableNode( model ), + new FactorEquationNode( + model.stateProperty, + model.problemModel.multiplicandProperty, + model.problemModel.multiplierProperty, + model.problemModel.productProperty + ), + { + showKeypad: false, + titleString: factorString, + levelSelectButtonColor: '#FFC266', + levelSelectIconSet: 'factor' + } + ); +} - arithmetic.register( 'FactorView', FactorView ); +arithmetic.register( 'FactorView', FactorView ); - return inherit( ArithmeticView, FactorView ); -} ); +inherit( ArithmeticView, FactorView ); +export default FactorView; \ No newline at end of file diff --git a/js/multiply/MultiplyScreen.js b/js/multiply/MultiplyScreen.js index 8cd39aac..800711b3 100644 --- a/js/multiply/MultiplyScreen.js +++ b/js/multiply/MultiplyScreen.js @@ -5,44 +5,41 @@ * * @author John Blanco, Andrey Zelenkov (MLearner) */ -define( require => { - 'use strict'; - - // modules - const arithmetic = require( 'ARITHMETIC/arithmetic' ); - const ArithmeticConstants = require( 'ARITHMETIC/common/ArithmeticConstants' ); - const inherit = require( 'PHET_CORE/inherit' ); - const merge = require( 'PHET_CORE/merge' ); - const MultiplyModel = require( 'ARITHMETIC/multiply/model/MultiplyModel' ); - const MultiplyScreenIconNode = require( 'ARITHMETIC/multiply/view/MultiplyScreenIconNode' ); - const MultiplyView = require( 'ARITHMETIC/multiply/view/MultiplyView' ); - const Property = require( 'AXON/Property' ); - const Screen = require( 'JOIST/Screen' ); - const Tandem = require( 'TANDEM/Tandem' ); - - // strings - const multiplyString = require( 'string!ARITHMETIC/multiply' ); - - /** - * @param {Object} [options] - * @constructor - */ - function MultiplyScreen( options ) { - - options = merge( { - name: multiplyString, - homeScreenIcon: new MultiplyScreenIconNode(), - backgroundColorProperty: new Property( ArithmeticConstants.BACKGROUND_COLOR ), - tandem: Tandem.REQUIRED - }, options ); - - Screen.call( this, - function() { return new MultiplyModel( options.tandem.createTandem( 'model' ) ); }, - function( model ) { return new MultiplyView( model ); }, - options ); - } - - arithmetic.register( 'MultiplyScreen', MultiplyScreen ); - - return inherit( Screen, MultiplyScreen ); -} ); \ No newline at end of file + +import Property from '../../../axon/js/Property.js'; +import Screen from '../../../joist/js/Screen.js'; +import inherit from '../../../phet-core/js/inherit.js'; +import merge from '../../../phet-core/js/merge.js'; +import Tandem from '../../../tandem/js/Tandem.js'; +import arithmeticStrings from '../arithmetic-strings.js'; +import arithmetic from '../arithmetic.js'; +import ArithmeticConstants from '../common/ArithmeticConstants.js'; +import MultiplyModel from './model/MultiplyModel.js'; +import MultiplyScreenIconNode from './view/MultiplyScreenIconNode.js'; +import MultiplyView from './view/MultiplyView.js'; + +const multiplyString = arithmeticStrings.multiply; + +/** + * @param {Object} [options] + * @constructor + */ +function MultiplyScreen( options ) { + + options = merge( { + name: multiplyString, + homeScreenIcon: new MultiplyScreenIconNode(), + backgroundColorProperty: new Property( ArithmeticConstants.BACKGROUND_COLOR ), + tandem: Tandem.REQUIRED + }, options ); + + Screen.call( this, + function() { return new MultiplyModel( options.tandem.createTandem( 'model' ) ); }, + function( model ) { return new MultiplyView( model ); }, + options ); +} + +arithmetic.register( 'MultiplyScreen', MultiplyScreen ); + +inherit( Screen, MultiplyScreen ); +export default MultiplyScreen; \ No newline at end of file diff --git a/js/multiply/model/MultiplyModel.js b/js/multiply/model/MultiplyModel.js index db0ade4b..148f52f4 100644 --- a/js/multiply/model/MultiplyModel.js +++ b/js/multiply/model/MultiplyModel.js @@ -7,95 +7,90 @@ * @author John Blanco */ -define( require => { - 'use strict'; +import inherit from '../../../../phet-core/js/inherit.js'; +import soundManager from '../../../../tambo/js/soundManager.js'; +import arithmetic from '../../arithmetic.js'; +import ArithmeticModel from '../../common/model/ArithmeticModel.js'; +import GameState from '../../common/model/GameState.js'; - // modules - const arithmetic = require( 'ARITHMETIC/arithmetic' ); - const ArithmeticModel = require( 'ARITHMETIC/common/model/ArithmeticModel' ); - const GameState = require( 'ARITHMETIC/common/model/GameState' ); - const inherit = require( 'PHET_CORE/inherit' ); - const soundManager = require( 'TAMBO/soundManager' ); - - /** - * @constructor - */ - function MultiplyModel( tandem ) { - const self = this; - ArithmeticModel.call( - this, - tandem, { - fillEquation: function() { - self.problemModel.productProperty.set( parseInt( self.inputProperty.get(), 10 ) ); - self.submitAnswer(); - } +/** + * @constructor + */ +function MultiplyModel( tandem ) { + const self = this; + ArithmeticModel.call( + this, + tandem, { + fillEquation: function() { + self.problemModel.productProperty.set( parseInt( self.inputProperty.get(), 10 ) ); + self.submitAnswer(); } - ); - } - - arithmetic.register( 'MultiplyModel', MultiplyModel ); + } + ); +} - return inherit( ArithmeticModel, MultiplyModel, { +arithmetic.register( 'MultiplyModel', MultiplyModel ); - // @public - setUpUnansweredProblem: function() { +export default inherit( ArithmeticModel, MultiplyModel, { - // get available multiplier pair - const multiplierPair = this.selectUnusedMultiplierPair(); + // @public + setUpUnansweredProblem: function() { - if ( multiplierPair ) { + // get available multiplier pair + const multiplierPair = this.selectUnusedMultiplierPair(); - // reset the problem and score properties - this.problemModel.multiplicandProperty.reset(); - this.problemModel.multiplierProperty.reset(); - this.problemModel.productProperty.reset(); - this.problemModel.possiblePointsProperty.reset(); + if ( multiplierPair ) { - // set up the problem - this.problemModel.multiplicandProperty.set( multiplierPair.multiplicand ); - this.problemModel.multiplierProperty.set( multiplierPair.multiplier ); + // reset the problem and score properties + this.problemModel.multiplicandProperty.reset(); + this.problemModel.multiplierProperty.reset(); + this.problemModel.productProperty.reset(); + this.problemModel.possiblePointsProperty.reset(); - return true; - } + // set up the problem + this.problemModel.multiplicandProperty.set( multiplierPair.multiplicand ); + this.problemModel.multiplierProperty.set( multiplierPair.multiplier ); - // All multiplier pairs have been used, so false is returned. - return false; - }, - - /** - * Automatically answer most of the problems. This is useful for testing, since it can save time when evaluating - * how the sim behaves when a user finishes answering all questions for a level. We need to be very careful that - * this is never available in the published sim. - * @override - * @protected - */ - autoAnswer: function() { - - // make sure that sound is off, since otherwise it dings for every solved problem - const soundState = soundManager.enabled; - soundManager.enabled = false; - - // answer the questions - const self = this; - const numQuestions = this.activeLevelModel.tableSize * this.activeLevelModel.tableSize; - const numQuestionsToAnswer = numQuestions - 1; - console.log( 'Automatically answering', numQuestionsToAnswer, 'of', numQuestions, 'questions.' ); - _.times( numQuestionsToAnswer, function() { - self.problemModel.productProperty.set( - self.problemModel.multiplicandProperty.get() * self.problemModel.multiplierProperty.get() - ); - self.activeLevelModel.currentScoreProperty.value += self.problemModel.possiblePointsProperty.get(); - self.activeLevelModel.displayScoreProperty.set( self.activeLevelModel.currentScoreProperty.get() ); - self.activeLevelModel.markCellAsUsed( - self.problemModel.multiplicandProperty.get(), - self.problemModel.multiplierProperty.get() - ); - self.stateProperty.set( GameState.DISPLAYING_CORRECT_ANSWER_FEEDBACK ); - self.nextProblem(); - } ); - - // restore the original sound state - soundManager.enabled = soundState; + return true; } - } ); -} ); + + // All multiplier pairs have been used, so false is returned. + return false; + }, + + /** + * Automatically answer most of the problems. This is useful for testing, since it can save time when evaluating + * how the sim behaves when a user finishes answering all questions for a level. We need to be very careful that + * this is never available in the published sim. + * @override + * @protected + */ + autoAnswer: function() { + + // make sure that sound is off, since otherwise it dings for every solved problem + const soundState = soundManager.enabled; + soundManager.enabled = false; + + // answer the questions + const self = this; + const numQuestions = this.activeLevelModel.tableSize * this.activeLevelModel.tableSize; + const numQuestionsToAnswer = numQuestions - 1; + console.log( 'Automatically answering', numQuestionsToAnswer, 'of', numQuestions, 'questions.' ); + _.times( numQuestionsToAnswer, function() { + self.problemModel.productProperty.set( + self.problemModel.multiplicandProperty.get() * self.problemModel.multiplierProperty.get() + ); + self.activeLevelModel.currentScoreProperty.value += self.problemModel.possiblePointsProperty.get(); + self.activeLevelModel.displayScoreProperty.set( self.activeLevelModel.currentScoreProperty.get() ); + self.activeLevelModel.markCellAsUsed( + self.problemModel.multiplicandProperty.get(), + self.problemModel.multiplierProperty.get() + ); + self.stateProperty.set( GameState.DISPLAYING_CORRECT_ANSWER_FEEDBACK ); + self.nextProblem(); + } ); + + // restore the original sound state + soundManager.enabled = soundState; + } +} ); \ No newline at end of file diff --git a/js/multiply/view/MultiplyEquationNode.js b/js/multiply/view/MultiplyEquationNode.js index 61a3d468..ff691ad9 100644 --- a/js/multiply/view/MultiplyEquationNode.js +++ b/js/multiply/view/MultiplyEquationNode.js @@ -5,47 +5,44 @@ * * @author Andrey Zelenkov (MLearner) */ -define( require => { - 'use strict'; - - // modules - const arithmetic = require( 'ARITHMETIC/arithmetic' ); - const EquationNode = require( 'ARITHMETIC/common/view/EquationNode' ); - const GameState = require( 'ARITHMETIC/common/model/GameState' ); - const inherit = require( 'PHET_CORE/inherit' ); - - /** - * @param {Property} stateProperty - State of game property. - * @param {Property} multiplicandProperty - Property necessary for creating multiplicand input. - * @param {Property} multiplierProperty - Property necessary for creating multiplier input. - * @param {Property} inputProperty - Input property, which is the product, and is input by the user. - * - * @constructor - */ - function MultiplyEquationNode( stateProperty, multiplicandProperty, multiplierProperty, inputProperty ) { - const self = this; - EquationNode.call( this, multiplicandProperty, multiplierProperty, inputProperty ); - - // The product is always the interactive part for this equation, so set up the appearance now. - this.productInput.setInteractiveAppearance( true ); - - // Update contents and focus at the state changes. - stateProperty.link( function( newState, oldState ) { - - // Set the state of the product portion of the equation. - if ( newState === GameState.AWAITING_USER_INPUT && oldState !== GameState.DISPLAYING_INCORRECT_ANSWER_FEEDBACK ) { - self.productInput.clear(); - } - - // The input should only have focus (i.e. blinking cursor) when awaiting input from the user. - self.productInput.setFocus( newState === GameState.AWAITING_USER_INPUT ); - - // If the user got it wrong, the equation should depict a not equals sign. - self.setShowEqual( newState !== GameState.DISPLAYING_INCORRECT_ANSWER_FEEDBACK ); - } ); - } - - arithmetic.register( 'MultiplyEquationNode', MultiplyEquationNode ); - - return inherit( EquationNode, MultiplyEquationNode ); -} ); + +import inherit from '../../../../phet-core/js/inherit.js'; +import arithmetic from '../../arithmetic.js'; +import GameState from '../../common/model/GameState.js'; +import EquationNode from '../../common/view/EquationNode.js'; + +/** + * @param {Property} stateProperty - State of game property. + * @param {Property} multiplicandProperty - Property necessary for creating multiplicand input. + * @param {Property} multiplierProperty - Property necessary for creating multiplier input. + * @param {Property} inputProperty - Input property, which is the product, and is input by the user. + * + * @constructor + */ +function MultiplyEquationNode( stateProperty, multiplicandProperty, multiplierProperty, inputProperty ) { + const self = this; + EquationNode.call( this, multiplicandProperty, multiplierProperty, inputProperty ); + + // The product is always the interactive part for this equation, so set up the appearance now. + this.productInput.setInteractiveAppearance( true ); + + // Update contents and focus at the state changes. + stateProperty.link( function( newState, oldState ) { + + // Set the state of the product portion of the equation. + if ( newState === GameState.AWAITING_USER_INPUT && oldState !== GameState.DISPLAYING_INCORRECT_ANSWER_FEEDBACK ) { + self.productInput.clear(); + } + + // The input should only have focus (i.e. blinking cursor) when awaiting input from the user. + self.productInput.setFocus( newState === GameState.AWAITING_USER_INPUT ); + + // If the user got it wrong, the equation should depict a not equals sign. + self.setShowEqual( newState !== GameState.DISPLAYING_INCORRECT_ANSWER_FEEDBACK ); + } ); +} + +arithmetic.register( 'MultiplyEquationNode', MultiplyEquationNode ); + +inherit( EquationNode, MultiplyEquationNode ); +export default MultiplyEquationNode; \ No newline at end of file diff --git a/js/multiply/view/MultiplyScreenIconNode.js b/js/multiply/view/MultiplyScreenIconNode.js index 73335129..c07b1eca 100644 --- a/js/multiply/view/MultiplyScreenIconNode.js +++ b/js/multiply/view/MultiplyScreenIconNode.js @@ -6,44 +6,41 @@ * @author Andrey Zelenkov (MLearner) * @author John Blanco (MLearner) */ -define( require => { - 'use strict'; - - // modules - const arithmetic = require( 'ARITHMETIC/arithmetic' ); - const ArithmeticConstants = require( 'ARITHMETIC/common/ArithmeticConstants' ); - const inherit = require( 'PHET_CORE/inherit' ); - const Line = require( 'SCENERY/nodes/Line' ); - const Rectangle = require( 'SCENERY/nodes/Rectangle' ); - - // constants - const ICON_SIZE = ArithmeticConstants.SCREEN_ICON_SIZE; - const SYMBOL_COLOR = '#FFF31E'; - const LINE_WIDTH = 33; // empirically determined - - /** - * @constructor - */ - function MultiplyScreenIconNode() { - - // create the background - Rectangle.call( this, 0, 0, ICON_SIZE.width, ICON_SIZE.height, { fill: ArithmeticConstants.ICON_BACKGROUND_COLOR } ); - - // Add the multiply symbol. Create our own rather than use the Unicode char, since this gives us more control. - const symbolWidth = ICON_SIZE.width * 0.3; - this.addChild( new Line( 0, 0, symbolWidth, symbolWidth, { - stroke: SYMBOL_COLOR, - lineWidth: LINE_WIDTH, - center: this.center - } ) ); - this.addChild( new Line( symbolWidth, 0, 0, symbolWidth, { - stroke: SYMBOL_COLOR, - lineWidth: LINE_WIDTH, // empirically determined - center: this.center - } ) ); - } - - arithmetic.register( 'MultiplyScreenIconNode', MultiplyScreenIconNode ); - - return inherit( Rectangle, MultiplyScreenIconNode ); -} ); + +import inherit from '../../../../phet-core/js/inherit.js'; +import Line from '../../../../scenery/js/nodes/Line.js'; +import Rectangle from '../../../../scenery/js/nodes/Rectangle.js'; +import arithmetic from '../../arithmetic.js'; +import ArithmeticConstants from '../../common/ArithmeticConstants.js'; + +// constants +const ICON_SIZE = ArithmeticConstants.SCREEN_ICON_SIZE; +const SYMBOL_COLOR = '#FFF31E'; +const LINE_WIDTH = 33; // empirically determined + +/** + * @constructor + */ +function MultiplyScreenIconNode() { + + // create the background + Rectangle.call( this, 0, 0, ICON_SIZE.width, ICON_SIZE.height, { fill: ArithmeticConstants.ICON_BACKGROUND_COLOR } ); + + // Add the multiply symbol. Create our own rather than use the Unicode char, since this gives us more control. + const symbolWidth = ICON_SIZE.width * 0.3; + this.addChild( new Line( 0, 0, symbolWidth, symbolWidth, { + stroke: SYMBOL_COLOR, + lineWidth: LINE_WIDTH, + center: this.center + } ) ); + this.addChild( new Line( symbolWidth, 0, 0, symbolWidth, { + stroke: SYMBOL_COLOR, + lineWidth: LINE_WIDTH, // empirically determined + center: this.center + } ) ); +} + +arithmetic.register( 'MultiplyScreenIconNode', MultiplyScreenIconNode ); + +inherit( Rectangle, MultiplyScreenIconNode ); +export default MultiplyScreenIconNode; \ No newline at end of file diff --git a/js/multiply/view/MultiplyScreenTableNode.js b/js/multiply/view/MultiplyScreenTableNode.js index 7cbfe340..51620775 100644 --- a/js/multiply/view/MultiplyScreenTableNode.js +++ b/js/multiply/view/MultiplyScreenTableNode.js @@ -6,74 +6,69 @@ * @author Andrey Zelenkov (MLearner) */ -define( require => { - 'use strict'; +import inherit from '../../../../phet-core/js/inherit.js'; +import arithmetic from '../../arithmetic.js'; +import GameState from '../../common/model/GameState.js'; +import MultiplicationTableNode from '../../common/view/table/MultiplicationTableNode.js'; - // modules - const arithmetic = require( 'ARITHMETIC/arithmetic' ); - const GameState = require( 'ARITHMETIC/common/model/GameState' ); - const inherit = require( 'PHET_CORE/inherit' ); - const MultiplicationTableNode = require( 'ARITHMETIC/common/view/table/MultiplicationTableNode' ); - - /** - * @param {ProblemModel} problemModel - Model for single multiplication problem. - * @param {Property} levelNumberProperty - Level difficulty property. - * @param {Property} stateProperty - Current state property. - * @param {Array} levelModels - Array of descriptions for each level. - * - * @constructor - */ - function MultiplyScreenTableNode( problemModel, stateProperty, levelNumberProperty, levelModels ) { - const self = this; - MultiplicationTableNode.call( this, levelNumberProperty, stateProperty, levelModels, true ); - this.problemModel = problemModel; // @private +/** + * @param {ProblemModel} problemModel - Model for single multiplication problem. + * @param {Property} levelNumberProperty - Level difficulty property. + * @param {Property} stateProperty - Current state property. + * @param {Array} levelModels - Array of descriptions for each level. + * + * @constructor + */ +function MultiplyScreenTableNode( problemModel, stateProperty, levelNumberProperty, levelModels ) { + const self = this; + MultiplicationTableNode.call( this, levelNumberProperty, stateProperty, levelModels, true ); + this.problemModel = problemModel; // @private - stateProperty.lazyLink( function( state ) { + stateProperty.lazyLink( function( state ) { - // set view for multiplication table after choosing multiplicand and multiplier - if ( state === GameState.AWAITING_USER_INPUT ) { + // set view for multiplication table after choosing multiplicand and multiplier + if ( state === GameState.AWAITING_USER_INPUT ) { - // select the cells that correspond to the current problem - self.setCellAppearanceForProblem( levelNumberProperty.value ); - } - else if ( state === GameState.LEVEL_COMPLETED ) { + // select the cells that correspond to the current problem + self.setCellAppearanceForProblem( levelNumberProperty.value ); + } + else if ( state === GameState.LEVEL_COMPLETED ) { - // set all cells to default conditions when the table has been filled - self.setCellsToDefaultColor( levelNumberProperty.value ); - } - } ); - } + // set all cells to default conditions when the table has been filled + self.setCellsToDefaultColor( levelNumberProperty.value ); + } + } ); +} - arithmetic.register( 'MultiplyScreenTableNode', MultiplyScreenTableNode ); +arithmetic.register( 'MultiplyScreenTableNode', MultiplyScreenTableNode ); - return inherit( MultiplicationTableNode, MultiplyScreenTableNode, { +export default inherit( MultiplicationTableNode, MultiplyScreenTableNode, { - // @public, @override - refreshLevel: function( level ) { - MultiplicationTableNode.prototype.refreshLevel.call( this, level ); - this.setCellAppearanceForProblem( level ); - }, + // @public, @override + refreshLevel: function( level ) { + MultiplicationTableNode.prototype.refreshLevel.call( this, level ); + this.setCellAppearanceForProblem( level ); + }, - // @private, set the appearance of the cells based on the currently presented problem - setCellAppearanceForProblem: function( level ) { - const self = this; - this.setCellsToDefaultColor( level ); + // @private, set the appearance of the cells based on the currently presented problem + setCellAppearanceForProblem: function( level ) { + const self = this; + this.setCellsToDefaultColor( level ); - // set the header cells for this problem to the selected state - this.cells[ level ][ 0 ][ this.problemModel.multiplierProperty.get() ].setSelected(); - this.cells[ level ][ this.problemModel.multiplicandProperty.get() ][ 0 ].setSelected(); + // set the header cells for this problem to the selected state + this.cells[ level ][ 0 ][ this.problemModel.multiplierProperty.get() ].setSelected(); + this.cells[ level ][ this.problemModel.multiplicandProperty.get() ][ 0 ].setSelected(); - // create a rectangle of selected body cells with a width defined by the multiplier and a height defined by the - // multplicand - this.cells[ level ].forEach( function( multiplicand, index ) { - if ( index && index <= self.problemModel.multiplicandProperty.get() ) { - multiplicand.forEach( function( cell, index ) { - if ( index && index <= self.problemModel.multiplierProperty.get() ) { - cell.setSelected(); - } - } ); - } - } ); - } - } ); -} ); + // create a rectangle of selected body cells with a width defined by the multiplier and a height defined by the + // multplicand + this.cells[ level ].forEach( function( multiplicand, index ) { + if ( index && index <= self.problemModel.multiplicandProperty.get() ) { + multiplicand.forEach( function( cell, index ) { + if ( index && index <= self.problemModel.multiplierProperty.get() ) { + cell.setSelected(); + } + } ); + } + } ); + } +} ); \ No newline at end of file diff --git a/js/multiply/view/MultiplyView.js b/js/multiply/view/MultiplyView.js index bb80a513..b60838d1 100644 --- a/js/multiply/view/MultiplyView.js +++ b/js/multiply/view/MultiplyView.js @@ -5,48 +5,45 @@ * * @author Andrey Zelenkov (MLearner) */ -define( require => { - 'use strict'; - // modules - const arithmetic = require( 'ARITHMETIC/arithmetic' ); - const ArithmeticView = require( 'ARITHMETIC/common/view/ArithmeticView' ); - const inherit = require( 'PHET_CORE/inherit' ); - const MultiplyEquationNode = require( 'ARITHMETIC/multiply/view/MultiplyEquationNode' ); - const MultiplyScreenTableNode = require( 'ARITHMETIC/multiply/view/MultiplyScreenTableNode' ); +import inherit from '../../../../phet-core/js/inherit.js'; +import arithmeticStrings from '../../arithmetic-strings.js'; +import arithmetic from '../../arithmetic.js'; +import ArithmeticView from '../../common/view/ArithmeticView.js'; +import MultiplyEquationNode from './MultiplyEquationNode.js'; +import MultiplyScreenTableNode from './MultiplyScreenTableNode.js'; - // strings - const multiplyString = require( 'string!ARITHMETIC/multiply' ); +const multiplyString = arithmeticStrings.multiply; - /** - * @param {MultiplyModel} model - Main model for screen. - * @constructor - */ - function MultiplyView( model ) { - ArithmeticView.call( - this, - model, - new MultiplyScreenTableNode( - model.problemModel, - model.stateProperty, - model.levelNumberProperty, - model.levelModels - ), - new MultiplyEquationNode( - model.stateProperty, - model.problemModel.multiplicandProperty, - model.problemModel.multiplierProperty, - model.inputProperty - ), - { - titleString: multiplyString, - levelSelectButtonColor: '#D8F58A', - levelSelectIconSet: 'multiply' - } - ); - } +/** + * @param {MultiplyModel} model - Main model for screen. + * @constructor + */ +function MultiplyView( model ) { + ArithmeticView.call( + this, + model, + new MultiplyScreenTableNode( + model.problemModel, + model.stateProperty, + model.levelNumberProperty, + model.levelModels + ), + new MultiplyEquationNode( + model.stateProperty, + model.problemModel.multiplicandProperty, + model.problemModel.multiplierProperty, + model.inputProperty + ), + { + titleString: multiplyString, + levelSelectButtonColor: '#D8F58A', + levelSelectIconSet: 'multiply' + } + ); +} - arithmetic.register( 'MultiplyView', MultiplyView ); +arithmetic.register( 'MultiplyView', MultiplyView ); - return inherit( ArithmeticView, MultiplyView ); -} ); +inherit( ArithmeticView, MultiplyView ); +export default MultiplyView; \ No newline at end of file diff --git a/js/phet-io/arithmetic-phet-io-elements-baseline.js b/js/phet-io/arithmetic-phet-io-elements-baseline.js index 09779497..0fb46dab 100644 --- a/js/phet-io/arithmetic-phet-io-elements-baseline.js +++ b/js/phet-io/arithmetic-phet-io-elements-baseline.js @@ -1,3 +1,4 @@ + /* eslint-disable */ window.phet.phetio.phetioElementsBaseline = assert && { diff --git a/js/phet-io/arithmetic-phet-io-elements-overrides.js b/js/phet-io/arithmetic-phet-io-elements-overrides.js index cedf0757..1985c30f 100644 --- a/js/phet-io/arithmetic-phet-io-elements-overrides.js +++ b/js/phet-io/arithmetic-phet-io-elements-overrides.js @@ -1,2 +1,3 @@ + /* eslint-disable */ window.phet.phetio.phetioElementsOverrides = {}; \ No newline at end of file diff --git a/js/phet-io/arithmetic-phet-io-types.js b/js/phet-io/arithmetic-phet-io-types.js index 03cd1e2a..105613b0 100644 --- a/js/phet-io/arithmetic-phet-io-types.js +++ b/js/phet-io/arithmetic-phet-io-types.js @@ -1,3 +1,4 @@ + /* eslint-disable */ window.phet.phetio.phetioTypes = assert && { diff --git a/mipmaps/divide_level_1_icon_png.js b/mipmaps/divide_level_1_icon_png.js new file mode 100644 index 00000000..36054083 --- /dev/null +++ b/mipmaps/divide_level_1_icon_png.js @@ -0,0 +1,38 @@ +/* eslint-disable */ +var mipmaps = [ { + "width": 109, + "height": 304, + "url": "" +}, { + "width": 55, + "height": 152, + "url": "" +}, { + "width": 28, + "height": 76, + "url": "" +}, { + "width": 14, + "height": 38, + "url": "" +}, { + "width": 7, + "height": 19, + "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/divide_level_2_icon_png.js b/mipmaps/divide_level_2_icon_png.js new file mode 100644 index 00000000..43ebc5be --- /dev/null +++ b/mipmaps/divide_level_2_icon_png.js @@ -0,0 +1,38 @@ +/* eslint-disable */ +var mipmaps = [ { + "width": 119, + "height": 306, + "url": "" +}, { + "width": 60, + "height": 153, + "url": "" +}, { + "width": 30, + "height": 77, + "url": "" +}, { + "width": 15, + "height": 39, + "url": "" +}, { + "width": 8, + "height": 20, + "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/divide_level_3_icon_png.js b/mipmaps/divide_level_3_icon_png.js new file mode 100644 index 00000000..865e5366 --- /dev/null +++ b/mipmaps/divide_level_3_icon_png.js @@ -0,0 +1,38 @@ +/* eslint-disable */ +var mipmaps = [ { + "width": 142, + "height": 310, + "url": "" +}, { + "width": 71, + "height": 155, + "url": "" +}, { + "width": 36, + "height": 78, + "url": "" +}, { + "width": 18, + "height": 39, + "url": "" +}, { + "width": 9, + "height": 20, + "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/factor_level_1_icon_png.js b/mipmaps/factor_level_1_icon_png.js new file mode 100644 index 00000000..48e16658 --- /dev/null +++ b/mipmaps/factor_level_1_icon_png.js @@ -0,0 +1,38 @@ +/* eslint-disable */ +var mipmaps = [ { + "width": 124, + "height": 196, + "url": "" +}, { + "width": 62, + "height": 98, + "url": "" +}, { + "width": 31, + "height": 49, + "url": "" +}, { + "width": 16, + "height": 25, + "url": "" +}, { + "width": 8, + "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/factor_level_2_icon_png.js b/mipmaps/factor_level_2_icon_png.js new file mode 100644 index 00000000..3e89b258 --- /dev/null +++ b/mipmaps/factor_level_2_icon_png.js @@ -0,0 +1,38 @@ +/* eslint-disable */ +var mipmaps = [ { + "width": 116, + "height": 275, + "url": "" +}, { + "width": 58, + "height": 138, + "url": "" +}, { + "width": 29, + "height": 69, + "url": "" +}, { + "width": 15, + "height": 35, + "url": "" +}, { + "width": 8, + "height": 18, + "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/factor_level_3_icon_png.js b/mipmaps/factor_level_3_icon_png.js new file mode 100644 index 00000000..f0a0f5c3 --- /dev/null +++ b/mipmaps/factor_level_3_icon_png.js @@ -0,0 +1,38 @@ +/* eslint-disable */ +var mipmaps = [ { + "width": 98, + "height": 308, + "url": "" +}, { + "width": 49, + "height": 154, + "url": "" +}, { + "width": 25, + "height": 77, + "url": "" +}, { + "width": 13, + "height": 39, + "url": "" +}, { + "width": 7, + "height": 20, + "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_level_1_icon_png.js b/mipmaps/multiply_level_1_icon_png.js new file mode 100644 index 00000000..c4fa67de --- /dev/null +++ b/mipmaps/multiply_level_1_icon_png.js @@ -0,0 +1,38 @@ +/* eslint-disable */ +var mipmaps = [ { + "width": 259, + "height": 243, + "url": "" +}, { + "width": 130, + "height": 122, + "url": "" +}, { + "width": 65, + "height": 61, + "url": "" +}, { + "width": 33, + "height": 31, + "url": "" +}, { + "width": 17, + "height": 16, + "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_level_2_icon_png.js b/mipmaps/multiply_level_2_icon_png.js new file mode 100644 index 00000000..0b46be8b --- /dev/null +++ b/mipmaps/multiply_level_2_icon_png.js @@ -0,0 +1,38 @@ +/* eslint-disable */ +var mipmaps = [ { + "width": 255, + "height": 271, + "url": "" +}, { + "width": 128, + "height": 136, + "url": "" +}, { + "width": 64, + "height": 68, + "url": "" +}, { + "width": 32, + "height": 34, + "url": "" +}, { + "width": 16, + "height": 17, + "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_level_3_icon_png.js b/mipmaps/multiply_level_3_icon_png.js new file mode 100644 index 00000000..64bbc518 --- /dev/null +++ b/mipmaps/multiply_level_3_icon_png.js @@ -0,0 +1,38 @@ +/* eslint-disable */ +var mipmaps = [ { + "width": 261, + "height": 270, + "url": "" +}, { + "width": 131, + "height": 135, + "url": "" +}, { + "width": 66, + "height": 68, + "url": "" +}, { + "width": 33, + "height": 34, + "url": "" +}, { + "width": 17, + "height": 17, + "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