diff --git a/geometric-optics-strings_en.json b/geometric-optics-strings_en.json index 555e0535..c9d00fe7 100644 --- a/geometric-optics-strings_en.json +++ b/geometric-optics-strings_en.json @@ -77,6 +77,9 @@ "valueCentimetersPattern": { "value": "{{value}} cm" }, + "arrow": { + "value": "Arrow" + }, "pencil": { "value": "Pencil" }, diff --git a/images/arrowIcon.png b/images/arrowIcon.png new file mode 100644 index 00000000..0c7d1bed Binary files /dev/null and b/images/arrowIcon.png differ diff --git a/images/arrowIcon_png.js b/images/arrowIcon_png.js new file mode 100644 index 00000000..109e70d2 --- /dev/null +++ b/images/arrowIcon_png.js @@ -0,0 +1,8 @@ +/* eslint-disable */ +import asyncLoader from '../../phet-core/js/asyncLoader.js'; + +const image = new Image(); +const unlock = asyncLoader.createLock( image ); +image.onload = unlock; +image.src = ''; +export default image; \ No newline at end of file diff --git a/images/license.json b/images/license.json index 8787c21f..8da8c90a 100644 --- a/images/license.json +++ b/images/license.json @@ -1,4 +1,11 @@ { + "arrowIcon.png": { + "text": [ + "Copyright 2022 University of Colorado Boulder" + ], + "projectURL": "https://phet.colorado.edu", + "license": "contact phethelp@colorado.edu" + }, "pencilIcon.png": { "text": [ "Copyright 2021 University of Colorado Boulder" diff --git a/js/common/GOColors.ts b/js/common/GOColors.ts index 185657c8..01b2e6fa 100644 --- a/js/common/GOColors.ts +++ b/js/common/GOColors.ts @@ -90,6 +90,22 @@ const GOColors = { default: 'black' } ), + arrow1FillProperty: new ProfileColorProperty( geometricOptics, 'arrow1Fill', { + default: 'green' + } ), + + arrow1StrokeProperty: new ProfileColorProperty( geometricOptics, 'arrow1Stroke', { + default: 'black' + } ), + + arrow2FillProperty: new ProfileColorProperty( geometricOptics, 'arrow2Fill', { + default: 'rgb( 255, 51, 51 )' //TODO same as secondPointFillProperty + } ), + + arrow2StrokeProperty: new ProfileColorProperty( geometricOptics, 'arrow2Stroke', { + default: 'black' + } ), + // Rays associated with the first Optical Object rays1StrokeProperty: new ProfileColorProperty( geometricOptics, 'realRayOneStroke', { default: 'rgb( 140, 140, 140 )' diff --git a/js/common/GOConstants.ts b/js/common/GOConstants.ts index 9334cfe4..a976bfca 100644 --- a/js/common/GOConstants.ts +++ b/js/common/GOConstants.ts @@ -90,6 +90,14 @@ const GOConstants = { headWidth: 12, headHeight: 8, tailWidth: 3 + }, + + ARROW_NODE_OPTIONS: { + headWidth: 24, + headHeight: 28, + tailWidth: 7, + isHeadDynamic: true, + fractionalHeadHeight: 0.5 } }; diff --git a/js/common/model/ArrowImage.ts b/js/common/model/ArrowImage.ts new file mode 100644 index 00000000..5982e366 --- /dev/null +++ b/js/common/model/ArrowImage.ts @@ -0,0 +1,35 @@ +// Copyright 2022, University of Colorado Boulder + +/** + * ArrowImage is the model of the optical image associated with an arrow object. + * + * @author Chris Malley (PixelZoom, Inc.) + */ + +import geometricOptics from '../../geometricOptics.js'; +import OpticalImage, { OpticalImageOptions } from './OpticalImage.js'; +import Optic from './Optic.js'; +import merge from '../../../../phet-core/js/merge.js'; +import ArrowObject from './ArrowObject.js'; + +class ArrowImage extends OpticalImage { + + /** + * @param arrowObject + * @param optic + * @param providedOptions + */ + constructor( arrowObject: ArrowObject, + optic: Optic, + providedOptions: OpticalImageOptions ) { + + const options = merge( {}, providedOptions ); + + super( arrowObject.positionProperty, optic, options ); + + //TODO more? + } +} + +geometricOptics.register( 'ArrowImage', ArrowImage ); +export default ArrowImage; \ No newline at end of file diff --git a/js/common/model/ArrowObject.ts b/js/common/model/ArrowObject.ts new file mode 100644 index 00000000..38e2a6bc --- /dev/null +++ b/js/common/model/ArrowObject.ts @@ -0,0 +1,41 @@ +// Copyright 2021-2022, University of Colorado Boulder + +/** + * ArrowObject is the model for arrow objects. + * + * @author Martin Veillette + * @author Chris Malley (PixelZoom, Inc.) + */ + +import geometricOptics from '../../geometricOptics.js'; +import OpticalObject, { OpticalObjectOptions } from './OpticalObject.js'; +import merge from '../../../../phet-core/js/merge.js'; + +type ArrowObjectOptions = { + fill: ColorDef + stroke: ColorDef +} & OpticalObjectOptions; + +class ArrowObject extends OpticalObject { + + public readonly fill: ColorDef; + public readonly stroke: ColorDef; + + /** + * @param providedOptions + */ + constructor( providedOptions: ArrowObjectOptions ) { + + const options = merge( {}, providedOptions ); + + super( options ); + + this.fill = options.fill; + this.stroke = options.stroke; + + //TODO more? + } +} + +geometricOptics.register( 'ArrowObject', ArrowObject ); +export default ArrowObject; \ No newline at end of file diff --git a/js/common/model/ArrowObjectScene.ts b/js/common/model/ArrowObjectScene.ts new file mode 100644 index 00000000..c2fbd04e --- /dev/null +++ b/js/common/model/ArrowObjectScene.ts @@ -0,0 +1,161 @@ +// Copyright 2022, University of Colorado Boulder + +//TODO lots of duplication with FramedObjectScene +/** + * ArrowObjectScene is a scene in which rays from two arrows interact with an optic and produce an Image. + * + * @author Chris Malley (PixelZoom, Inc.) + */ + +import Range from '../../../../dot/js/Range.js'; +import geometricOptics from '../../geometricOptics.js'; +import Optic from './Optic.js'; +import Vector2 from '../../../../dot/js/Vector2.js'; +import Tandem from '../../../../tandem/js/Tandem.js'; +import merge from '../../../../phet-core/js/merge.js'; +import { RaysType } from './RaysType.js'; +import NumberProperty from '../../../../axon/js/NumberProperty.js'; +import LightRays from './LightRays.js'; +import IReadOnlyProperty from '../../../../axon/js/IReadOnlyProperty.js'; +import Lens from '../../lens/model/Lens.js'; +import PhetioObject from '../../../../tandem/js/PhetioObject.js'; +import Guides from '../../lens/model/Guides.js'; +import ArrowObject from './ArrowObject.js'; +import ArrowImage from './ArrowImage.js'; +import GOColors from '../GOColors.js'; + +type ArrowObjectSceneOptions = { + + // initial positions of the arrow objects + arrowObject1Position: Vector2, + arrowObject2Position: Vector2, + + // phet-io options + tandem: Tandem +}; + +class ArrowObjectScene extends PhetioObject { + + readonly optic: Optic; + readonly arrowObject1: ArrowObject; + readonly arrowObject2: ArrowObject; + readonly arrowImage1: ArrowImage; + readonly arrowImage2: ArrowImage; + readonly lightRaysAnimationTimeProperty: NumberProperty; + readonly lightRays1: LightRays; + readonly lightRays2: LightRays; + readonly guides1: Guides | null; + readonly guides2: Guides | null; + private readonly resetArrowObjectScene: () => void; + + /** + * @param optic + * @param raysTypeProperty + * @param providedOptions + */ + constructor( optic: Optic, + raysTypeProperty: IReadOnlyProperty, + providedOptions: ArrowObjectSceneOptions ) { + + const options = merge( { + phetioState: false + }, providedOptions ); + + super( options ); + + this.optic = optic; + + this.addLinkedElement( optic, { + tandem: options.tandem.createTandem( 'optic' ) + } ); + + this.arrowObject1 = new ArrowObject( { + position: options.arrowObject1Position, + fill: GOColors.arrow1FillProperty, + stroke: GOColors.arrow1StrokeProperty, + tandem: options.tandem.createTandem( 'arrowObject1' ) + } ); + + this.arrowObject2 = new ArrowObject( { + position: options.arrowObject2Position, + fill: GOColors.arrow2FillProperty, + stroke: GOColors.arrow2StrokeProperty, + tandem: options.tandem.createTandem( 'arrowObject2' ) + } ); + + this.arrowImage1 = new ArrowImage( this.arrowObject1, this.optic, { + tandem: options.tandem.createTandem( 'arrowImage1' ), + phetioDocumentation: 'optical image associated with the first arrow object' + } ); + + this.arrowImage2 = new ArrowImage( this.arrowObject2, this.optic, { + tandem: options.tandem.createTandem( 'arrowImage2' ), + phetioDocumentation: 'optical image associated with the second arrow object' + } ); + + this.lightRaysAnimationTimeProperty = new NumberProperty( 0, { + units: 's', + range: new Range( 0, 10 ), // determines the duration of the light rays animation + tandem: options.tandem.createTandem( 'lightRaysAnimationTimeProperty' ), + phetioReadOnly: true + } ); + + this.lightRays1 = new LightRays( + this.lightRaysAnimationTimeProperty, + raysTypeProperty, + this.arrowObject1.positionProperty, + this.optic, + this.arrowImage1 + ); + + this.lightRays2 = new LightRays( + this.lightRaysAnimationTimeProperty, + raysTypeProperty, + this.arrowObject2.positionProperty, + this.optic, + this.arrowImage2 + ); + + // Guides + if ( optic instanceof Lens ) { + this.guides1 = new Guides( this.optic, this.arrowObject1.positionProperty, { + tandem: options.tandem.createTandem( 'guides1' ), + phetioDocumentation: 'guides associated with the first arrow object' + } ); + this.guides2 = new Guides( this.optic, this.arrowObject2.positionProperty, { + tandem: options.tandem.createTandem( 'guides2' ), + phetioDocumentation: 'guides associated with the second arrow object' + } ); + } + else { + this.guides1 = null; + this.guides2 = null; + } + + //TODO is this complete? + this.resetArrowObjectScene = () => { + this.arrowObject1.reset(); + this.arrowObject2.reset(); + this.lightRaysAnimationTimeProperty.reset(); + }; + } + + public reset(): void { + this.resetArrowObjectScene(); + } + + /** + * Steps the animation of light rays. + * @param dt - time step, in seconds + */ + public stepLightRays( dt: number ): void { + const t = Math.min( this.lightRaysAnimationTimeProperty.value + dt, this.lightRaysAnimationTimeProperty.range!.max ); + assert && assert( this.lightRaysAnimationTimeProperty.range ); // {Range|null} + if ( this.lightRaysAnimationTimeProperty.range!.contains( t ) ) { + this.lightRaysAnimationTimeProperty.value = t; + } + } +} + +geometricOptics.register( 'ArrowObjectScene', ArrowObjectScene ); +export default ArrowObjectScene; \ No newline at end of file diff --git a/js/common/model/FramedImage.ts b/js/common/model/FramedImage.ts index c0f84683..9cfdd96a 100644 --- a/js/common/model/FramedImage.ts +++ b/js/common/model/FramedImage.ts @@ -1,8 +1,7 @@ // Copyright 2021-2022, University of Colorado Boulder -//TODO this entire class needs to be reviewed/revised /** - * FramedImage is the model of an optical image associated with a framed object. + * FramedImage is the model of the optical image associated with a framed object. * * @author Martin Veillette * @author Chris Malley (PixelZoom, Inc.) diff --git a/js/common/model/FramedObjectScene.ts b/js/common/model/FramedObjectScene.ts index eaa5fd5e..31ef49ae 100644 --- a/js/common/model/FramedObjectScene.ts +++ b/js/common/model/FramedObjectScene.ts @@ -120,7 +120,6 @@ class FramedObjectScene extends PhetioObject { // Guides if ( optic instanceof Lens ) { - this.guides1 = new Guides( this.optic, this.framedObject.positionProperty, { tandem: options.tandem.createTandem( 'guides1' ), phetioDocumentation: 'guides associated with the first point-of-interest on the framed object' diff --git a/js/common/model/GOModel.ts b/js/common/model/GOModel.ts index 50715071..37afb8c3 100644 --- a/js/common/model/GOModel.ts +++ b/js/common/model/GOModel.ts @@ -20,11 +20,14 @@ import GORuler from './GORuler.js'; import Vector2 from '../../../../dot/js/Vector2.js'; import FramedObjectScene from './FramedObjectScene.js'; import OpticalObjectChoice from './OpticalObjectChoice.js'; +import ArrowObjectScene from './ArrowObjectScene.js'; -type GeometricOpticsModelOptions = { +type GOModelOptions = { - // initial position of the framed object + // initial positions of optical objects framedObjectPosition: Vector2, + arrowObject1Position: Vector2, + arrowObject2Position: Vector2, // optical object choices, in the order that they will appear in OpticalObjectChoiceComboBox opticalObjectChoices: OpticalObjectChoice[], @@ -44,6 +47,7 @@ class GOModel { readonly raysTypeProperty: Property; // scenes + readonly arrowObjectScene: ArrowObjectScene; readonly framedObjectScene: FramedObjectScene; // rulers @@ -64,7 +68,7 @@ class GOModel { * @param optic * @param providedOptions */ - constructor( optic: Optic, providedOptions: GeometricOpticsModelOptions ) { + constructor( optic: Optic, providedOptions: GOModelOptions ) { const options = merge( { //TODO @@ -85,6 +89,12 @@ class GOModel { this.scenesTandem = options.tandem.createTandem( 'scenes' ); + this.arrowObjectScene = new ArrowObjectScene( this.optic, this.raysTypeProperty, { + arrowObject1Position: options.arrowObject1Position, + arrowObject2Position: options.arrowObject2Position, + tandem: this.scenesTandem.createTandem( 'arrowObjectScene' ) + } ); + this.framedObjectScene = new FramedObjectScene( this.opticalObjectChoiceProperty, this.optic, this.raysTypeProperty, { framedObjectPosition: options.framedObjectPosition, tandem: this.scenesTandem.createTandem( 'framedObjectScene' ) @@ -111,6 +121,7 @@ class GOModel { this.optic.reset(); this.raysTypeProperty.reset(); this.framedObjectScene.reset(); + this.arrowObjectScene.reset(); this.horizontalRuler.reset(); this.verticalRuler.reset(); }; @@ -126,5 +137,4 @@ class GOModel { } geometricOptics.register( 'GOModel', GOModel ); -export default GOModel; -export type { GeometricOpticsModelOptions }; \ No newline at end of file +export default GOModel; \ No newline at end of file diff --git a/js/common/model/OpticalObjectChoice.ts b/js/common/model/OpticalObjectChoice.ts index 20b068f9..5dba6b56 100644 --- a/js/common/model/OpticalObjectChoice.ts +++ b/js/common/model/OpticalObjectChoice.ts @@ -31,6 +31,7 @@ import starRightFacingUpright_png from '../../../images/starRightFacingUpright_p import starRightFacingInverted_png from '../../../images/starRightFacingInverted_png.js'; import starLeftFacingUpright_png from '../../../images/starLeftFacingUpright_png.js'; import starLeftFacingInverted_png from '../../../images/starLeftFacingInverted_png.js'; +import arrowIcon_png from '../../../images/arrowIcon_png.js'; // Set of HTMLImageElements that depict a framed object and its associated optical image type ObjectHTMLImageElements = { @@ -42,6 +43,9 @@ type ObjectHTMLImageElements = { class OpticalObjectChoice extends EnumerationValue { + //TODO replace arrowIcon_png with new PNG file, or generate programmatically + static ARROW = new OpticalObjectChoice( geometricOpticsStrings.arrow, arrowIcon_png, 'arrow' ); + static PENCIL = new OpticalObjectChoice( geometricOpticsStrings.pencil, pencilIcon_png, 'pencil', { rightFacingUpright: pencilRightFacingUpright_png, rightFacingInverted: pencilRightFacingInverted_png, @@ -111,6 +115,14 @@ class OpticalObjectChoice extends EnumerationValue { this.objectHTMLImageElements = objectHTMLImageElements; } + /** + * Is the choice an arrow object? + * @param choice + */ + static isArrowObject( choice: OpticalObjectChoice ): boolean { + return ( choice === OpticalObjectChoice.ARROW ); + } + /** * Is the choice a framed object? * @param choice @@ -124,7 +136,7 @@ class OpticalObjectChoice extends EnumerationValue { * @param choice */ static isLightSource( choice: OpticalObjectChoice ): boolean { - return choice === OpticalObjectChoice.LIGHT; + return ( choice === OpticalObjectChoice.LIGHT ); } } diff --git a/js/common/view/ArrowImageNode.ts b/js/common/view/ArrowImageNode.ts new file mode 100644 index 00000000..0e6ce888 --- /dev/null +++ b/js/common/view/ArrowImageNode.ts @@ -0,0 +1,45 @@ +// Copyright 2022, University of Colorado Boulder + +/** + * ArrowImageNode is the visual representation of an arrow object. + * + * @author Chris Malley (PixelZoom, Inc.) + */ + +import IProperty from '../../../../axon/js/IProperty.js'; +import IReadOnlyProperty from '../../../../axon/js/IReadOnlyProperty.js'; +import merge from '../../../../phet-core/js/merge.js'; +import ModelViewTransform2 from '../../../../phetcommon/js/view/ModelViewTransform2.js'; +import { Node } from '../../../../scenery/js/imports.js'; +import Tandem from '../../../../tandem/js/Tandem.js'; +import geometricOptics from '../../geometricOptics.js'; +import ArrowImage from '../model/ArrowImage.js'; + +type ArrowImageNodeOptions = { + visibleProperty?: IProperty, + tandem: Tandem +}; + +class ArrowImageNode extends Node { + + /** + * @param arrowImage + * @param virtualImageVisibleProperty + * @param raysAndImagesVisibleProperty + * @param modelViewTransform + * @param providedOptions + */ + constructor( arrowImage: ArrowImage, + virtualImageVisibleProperty: IReadOnlyProperty, + raysAndImagesVisibleProperty: IReadOnlyProperty, + modelViewTransform: ModelViewTransform2, + providedOptions: ArrowImageNodeOptions ) { + + const options = merge( {}, providedOptions ); + + super( options ); + } +} + +geometricOptics.register( 'ArrowImageNode', ArrowImageNode ); +export default ArrowImageNode; \ No newline at end of file diff --git a/js/common/view/ArrowObjectNode.ts b/js/common/view/ArrowObjectNode.ts new file mode 100644 index 00000000..ad30c939 --- /dev/null +++ b/js/common/view/ArrowObjectNode.ts @@ -0,0 +1,67 @@ +// Copyright 2022, University of Colorado Boulder + +/** + * ArrowObjectNode is the visual representation of an arrow object. + * + * @author Chris Malley (PixelZoom, Inc.) + */ + +import IProperty from '../../../../axon/js/IProperty.js'; +import IReadOnlyProperty from '../../../../axon/js/IReadOnlyProperty.js'; +import Property from '../../../../axon/js/Property.js'; +import Bounds2 from '../../../../dot/js/Bounds2.js'; +import merge from '../../../../phet-core/js/merge.js'; +import ModelViewTransform2 from '../../../../phetcommon/js/view/ModelViewTransform2.js'; +import { Node } from '../../../../scenery/js/imports.js'; +import Tandem from '../../../../tandem/js/Tandem.js'; +import geometricOptics from '../../geometricOptics.js'; +import ArrowObject from '../model/ArrowObject.js'; +import Optic from '../model/Optic.js'; +import GOConstants from '../GOConstants.js'; +import ArrowNode from '../../../../scenery-phet/js/ArrowNode.js'; + +type ArrowObjectNodeOptions = { + visibleProperty?: IProperty, + tandem: Tandem +}; + +class ArrowObjectNode extends Node { + + /** + * @param arrowObject + * @param optic + * @param modelBoundsProperty + * @param modelViewTransform + * @param providedOptions + */ + constructor( arrowObject: ArrowObject, + optic: Optic, + modelBoundsProperty: IReadOnlyProperty, + modelViewTransform: ModelViewTransform2, + providedOptions: ArrowObjectNodeOptions ) { + + const options = merge( {}, providedOptions ); + + super( options ); + + const arrowNode = new ArrowNode( 0, 0, 0, 1, merge( {}, GOConstants.ARROW_NODE_OPTIONS, { + fill: arrowObject.fill, + stroke: arrowObject.stroke + } ) ); + this.addChild( arrowNode ); + + Property.multilink( [ arrowObject.positionProperty, optic.positionProperty ], + ( arrowObjectPosition, opticPosition ) => { + const tipPosition = modelViewTransform.modelToViewPosition( arrowObjectPosition ); + const tailY = modelViewTransform.modelToViewY( opticPosition.y ); + arrowNode.setTailAndTip( tipPosition.x, tailY, tipPosition.x, tipPosition.y ); + } ); + } + + reset() { + //TODO + } +} + +geometricOptics.register( 'ArrowObjectNode', ArrowObjectNode ); +export default ArrowObjectNode; \ No newline at end of file diff --git a/js/common/view/ArrowObjectSceneNode.ts b/js/common/view/ArrowObjectSceneNode.ts new file mode 100644 index 00000000..cf66d4a9 --- /dev/null +++ b/js/common/view/ArrowObjectSceneNode.ts @@ -0,0 +1,233 @@ +// Copyright 2022, University of Colorado Boulder + +//TODO lots of duplication with FramedObjectScene +/** + * ArrowObjectSceneNode is the view of ArrowObjectScene, the scene that uses a arrow objects. + * + * @author Chris Malley (PixelZoom, Inc.) + */ + +import merge from '../../../../phet-core/js/merge.js'; +import ModelViewTransform2 from '../../../../phetcommon/js/view/ModelViewTransform2.js'; +import { Node } from '../../../../scenery/js/imports.js'; +import geometricOptics from '../../geometricOptics.js'; +import VisibleProperties from './VisibleProperties.js'; +import Bounds2 from '../../../../dot/js/Bounds2.js'; +import IReadOnlyProperty from '../../../../axon/js/IReadOnlyProperty.js'; +import OpticalAxisNode from './OpticalAxisNode.js'; +import OpticVerticalAxisNode from './OpticVerticalAxisNode.js'; +import { RaysType } from '../model/RaysType.js'; +import FocalPointNode from './FocalPointNode.js'; +import TwoFPointNode from './TwoFPointNode.js'; +import GOColors from '../GOColors.js'; +import RealLightRaysNode from './RealLightRaysNode.js'; +import Tandem from '../../../../tandem/js/Tandem.js'; +import Optic from '../model/Optic.js'; +import BooleanProperty from '../../../../axon/js/BooleanProperty.js'; +import VirtualLightRaysNode from './VirtualLightRaysNode.js'; +import DerivedProperty from '../../../../axon/js/DerivedProperty.js'; +import GuidesNode from '../../lens/view/GuidesNode.js'; +import { RulerHotkeysData } from './GORulerNode.js'; +import BooleanIO from '../../../../tandem/js/types/BooleanIO.js'; +import ArrowObjectScene from '../model/ArrowObjectScene.js'; +import ArrowObjectNode from './ArrowObjectNode.js'; +import ArrowImageNode from './ArrowImageNode.js'; + +type ArrowObjectSceneNodeOptions = { + + // Creates the Node for the optic + createOpticNode: ( optic: Optic, modelBoundsProperty: IReadOnlyProperty, modelViewTransform: ModelViewTransform2, parentTandem: Tandem ) => Node, + + dragLockedProperty: BooleanProperty, + + tandem: Tandem +}; + +class ArrowObjectSceneNode extends Node { + + public readonly rulerHotkeysData: RulerHotkeysData; + private readonly resetFrameObjectSceneNode: () => void; + + /** + * @param scene + * @param visibleProperties + * @param modelViewTransform + * @param modelVisibleBoundsProperty + * @param modelBoundsProperty + * @param raysTypeProperty + * @param providedOptions + */ + constructor( scene: ArrowObjectScene, + visibleProperties: VisibleProperties, + modelViewTransform: ModelViewTransform2, + modelVisibleBoundsProperty: IReadOnlyProperty, + modelBoundsProperty: IReadOnlyProperty, + raysTypeProperty: IReadOnlyProperty, + providedOptions: ArrowObjectSceneNodeOptions ) { + + const options = merge( { + visiblePropertyOptions: { phetioReadOnly: true } + }, providedOptions ); + + super( options ); + + const opticNode = options.createOpticNode( scene.optic, modelBoundsProperty, modelViewTransform, options.tandem ); + + const opticalAxisNode = new OpticalAxisNode( + scene.optic.positionProperty, + modelVisibleBoundsProperty, + modelViewTransform, { + visibleProperty: visibleProperties.opticalAxisVisibleProperty, + tandem: options.tandem.createTandem( 'opticalAxisNode' ) + } ); + + const opticVerticalAxisNode = new OpticVerticalAxisNode( scene.optic, raysTypeProperty, modelViewTransform ); + + // focal points (F) + const focalPointsNodeTandem = options.tandem.createTandem( 'focalPointsNode' ); + const focalPointsNode = new Node( { + children: [ + new FocalPointNode( scene.optic.leftFocalPointProperty, modelViewTransform, { + tandem: focalPointsNodeTandem.createTandem( 'leftFocalPointNode' ) + } ), + new FocalPointNode( scene.optic.rightFocalPointProperty, modelViewTransform, { + tandem: focalPointsNodeTandem.createTandem( 'rightFocalPointNode' ) + } ) + ], + visibleProperty: visibleProperties.focalPointsVisibleProperty, + tandem: focalPointsNodeTandem + } ); + + // 2F points + const twoFPointsNodeTandem = options.tandem.createTandem( 'twoFPointsNode' ); + const twoFPointsNode = new Node( { + children: [ + new TwoFPointNode( scene.optic.left2FProperty, modelViewTransform, { + tandem: twoFPointsNodeTandem.createTandem( 'left2FPointNode' ) + } ), + new TwoFPointNode( scene.optic.right2FProperty, modelViewTransform, { + tandem: twoFPointsNodeTandem.createTandem( 'right2FPointNode' ) + } ) + ], + visibleProperty: visibleProperties.twoFPointsVisibleProperty, + tandem: twoFPointsNodeTandem + } ); + + const arrowObject1Node = new ArrowObjectNode( scene.arrowObject1, scene.optic, modelBoundsProperty, modelViewTransform, { + tandem: options.tandem.createTandem( 'arrowObject1Node' ) + } ); + + const arrowObject2Node = new ArrowObjectNode( scene.arrowObject2, scene.optic, modelBoundsProperty, modelViewTransform, { + visibleProperty: visibleProperties.secondPointVisibleProperty, + tandem: options.tandem.createTandem( 'arrowObject2Node' ) + } ); + + const arrowImage1Node = new ArrowImageNode( scene.arrowImage1, visibleProperties.virtualImageVisibleProperty, + visibleProperties.raysAndImagesVisibleProperty, modelViewTransform, { + tandem: options.tandem.createTandem( 'arrowImage1Node' ) + } ); + + const arrowImage2Node = new ArrowImageNode( scene.arrowImage2, visibleProperties.virtualImageVisibleProperty, + visibleProperties.raysAndImagesVisibleProperty, modelViewTransform, { + tandem: options.tandem.createTandem( 'arrowImage2Node' ) + } ); + + // Light rays (real & virtual) associated with the first point-of-interest (the framed object's position). + const realLightRays1Options = { + stroke: GOColors.rays1StrokeProperty, + visibleProperty: visibleProperties.raysAndImagesVisibleProperty + }; + const realLightRays1Node = new RealLightRaysNode( scene.lightRays1, modelViewTransform, realLightRays1Options ); + const virtualLightRays1Node = new VirtualLightRaysNode( scene.lightRays1, modelViewTransform, { + stroke: realLightRays1Options.stroke, + visibleProperty: DerivedProperty.and( [ + visibleProperties.virtualImageVisibleProperty, + visibleProperties.raysAndImagesVisibleProperty + ] ) + } ); + + // Light rays (real & virtual) associated with the second point-of-interest (also on the framed object). + const realLightRays2Options = { + stroke: GOColors.rays2StrokeProperty, + visibleProperty: DerivedProperty.and( [ + visibleProperties.secondPointVisibleProperty, + visibleProperties.raysAndImagesVisibleProperty + ] ) + }; + const realLightRays2Node = new RealLightRaysNode( scene.lightRays2, modelViewTransform, realLightRays2Options ); + const virtualLightRays2Node = new VirtualLightRaysNode( scene.lightRays2, modelViewTransform, { + stroke: realLightRays2Options.stroke, + visibleProperty: DerivedProperty.and( [ + visibleProperties.virtualImageVisibleProperty, + visibleProperties.secondPointVisibleProperty, + visibleProperties.raysAndImagesVisibleProperty + ] ) + } ); + + this.children = [ + opticalAxisNode, + arrowObject1Node, + arrowObject2Node, + arrowImage1Node, + arrowImage2Node, + opticNode, + opticVerticalAxisNode, + focalPointsNode, + twoFPointsNode, + realLightRays1Node, + virtualLightRays1Node, + realLightRays2Node, + virtualLightRays2Node + ]; + + if ( scene.guides1 ) { + const guides1Node = new GuidesNode( scene.guides1, GOColors.guideArm1FillProperty, modelViewTransform, { + visibleProperty: visibleProperties.guidesVisibleProperty, + tandem: options.tandem.createTandem( 'guides1Node' ), + phetioDocumentation: 'guides associated with the first point-of-interest on the framed object' + } ); + this.addChild( guides1Node ); + } + + if ( scene.guides2 ) { + const guides2Tandem = options.tandem.createTandem( 'guides2Node' ); + const guides2Node = new GuidesNode( scene.guides2, GOColors.guideArm2FillProperty, modelViewTransform, { + visibleProperty: DerivedProperty.and( + [ visibleProperties.guidesVisibleProperty, visibleProperties.secondPointVisibleProperty ], { + tandem: guides2Tandem.createTandem( 'visibleProperty' ), + phetioType: DerivedProperty.DerivedPropertyIO( BooleanIO ) + } ), + tandem: guides2Tandem, + phetioDocumentation: 'guides associated with the second point-of-interest on the framed object' + } ); + this.addChild( guides2Node ); + } + + this.rulerHotkeysData = { + opticPositionProperty: scene.optic.positionProperty, + opticalObject1PositionProperty: scene.arrowObject1.positionProperty, + opticalObject2PositionProperty: scene.arrowObject2.positionProperty, + opticalObject2VisibleProperty: arrowObject2Node.visibleProperty, + opticalImage1PositionProperty: scene.arrowImage1.positionProperty, + opticalImage1VisibleProperty: arrowImage1Node.visibleProperty + }; + + this.pdomOrder = [ + arrowObject1Node, + arrowObject2Node + ]; + + //TODO is this complete? + this.resetFrameObjectSceneNode = () => { + arrowObject1Node.reset(); + arrowObject2Node.reset(); + }; + } + + public reset(): void { + this.resetFrameObjectSceneNode(); + } +} + +geometricOptics.register( 'ArrowObjectSceneNode', ArrowObjectSceneNode ); +export default ArrowObjectSceneNode; \ No newline at end of file diff --git a/js/common/view/GOScreenView.ts b/js/common/view/GOScreenView.ts index bdb3dae7..265ea321 100644 --- a/js/common/view/GOScreenView.ts +++ b/js/common/view/GOScreenView.ts @@ -39,6 +39,7 @@ import { RaysType } from '../model/RaysType.js'; import GORulerNode from './GORulerNode.js'; import RulersToolbox from './RulersToolbox.js'; import FramedObjectSceneLabelsNode from './FramedObjectSceneLabelsNode.js'; +import ArrowObjectSceneNode from './ArrowObjectSceneNode.js'; // Zoom scale factors, in ascending order. // Careful! If you add values here, you may get undesirable tick intervals on rulers. @@ -252,6 +253,13 @@ class GOScreenView extends ScreenView { this.scenesTandem = options.tandem.createTandem( 'scenes' ); + const arrowObjectSceneNode = new ArrowObjectSceneNode( model.arrowObjectScene, visibleProperties, modelViewTransform, + modelVisibleBoundsProperty, modelBoundsProperty, model.raysTypeProperty, { + createOpticNode: options.createOpticNode, + dragLockedProperty: options.dragLockedProperty, + tandem: this.scenesTandem.createTandem( 'arrowObjectSceneNode' ) + } ); + const framedObjectSceneNode = new FramedObjectSceneNode( model.framedObjectScene, visibleProperties, modelViewTransform, modelVisibleBoundsProperty, modelBoundsProperty, model.raysTypeProperty, { createOpticNode: options.createOpticNode, @@ -260,7 +268,7 @@ class GOScreenView extends ScreenView { } ); const scenesNode = new Node( { - children: [ framedObjectSceneNode ] + children: [ arrowObjectSceneNode, framedObjectSceneNode ] } ); //TODO is experimentAreaNode still needed, or does scenesNode fill that role? @@ -308,6 +316,8 @@ class GOScreenView extends ScreenView { framedObjectSceneNode.visibleProperty ] ) } ); + //TODO arrowObjectSceneLabelsNode + const controlsLayer = new Node( { children: [ opticShapeRadioButtonGroup, @@ -340,6 +350,13 @@ class GOScreenView extends ScreenView { this.addChild( screenViewRootNode ); model.opticalObjectChoiceProperty.link( opticalObjectChoice => { + + arrowObjectSceneNode.visible = ( OpticalObjectChoice.isArrowObject( opticalObjectChoice ) ); + if ( arrowObjectSceneNode.visible ) { + horizontalRulerNode.setHotkeysData( arrowObjectSceneNode.rulerHotkeysData ); + verticalRulerNode.setHotkeysData( arrowObjectSceneNode.rulerHotkeysData ); + } + framedObjectSceneNode.visible = ( OpticalObjectChoice.isFramedObject( opticalObjectChoice ) ); if ( framedObjectSceneNode.visible ) { horizontalRulerNode.setHotkeysData( framedObjectSceneNode.rulerHotkeysData ); diff --git a/js/geometricOpticsStrings.ts b/js/geometricOpticsStrings.ts index 54b139b8..f2579e30 100644 --- a/js/geometricOpticsStrings.ts +++ b/js/geometricOpticsStrings.ts @@ -38,6 +38,7 @@ type StringsType = { 'many': string, 'secondPoint': string, 'valueCentimetersPattern': string, + 'arrow': string, 'pencil': string, 'penguin': string, 'planet': string, diff --git a/js/lens/model/LensModel.ts b/js/lens/model/LensModel.ts index 41127d8c..1abb32dd 100644 --- a/js/lens/model/LensModel.ts +++ b/js/lens/model/LensModel.ts @@ -36,8 +36,12 @@ class LensModel extends GOModel { // Initial position of the framed object, empirically set so that the optical axis goes through its center. framedObjectPosition: new Vector2( -170, 27 ), + arrowObject1Position: new Vector2( -150, 50 ), + arrowObject2Position: new Vector2( -150, -50 ), + // optical object choices, in the order that they will appear in OpticalObjectChoiceComboBox opticalObjectChoices: [ + OpticalObjectChoice.ARROW, OpticalObjectChoice.PENCIL, OpticalObjectChoice.PENGUIN, OpticalObjectChoice.PLANET, diff --git a/js/lens/model/LightSourceScene.ts b/js/lens/model/LightSourceScene.ts index d44b7273..5260778b 100644 --- a/js/lens/model/LightSourceScene.ts +++ b/js/lens/model/LightSourceScene.ts @@ -1,5 +1,6 @@ // Copyright 2022, University of Colorado Boulder +//TODO lots of duplication with FramedObjectScene /** * LightSourceScene is a scene in rays from 2 light sources interact with an optic, and project light spots on * a projection screen. diff --git a/js/lens/view/LensScreenView.ts b/js/lens/view/LensScreenView.ts index b8ea8a8f..bd3949f3 100644 --- a/js/lens/view/LensScreenView.ts +++ b/js/lens/view/LensScreenView.ts @@ -73,6 +73,9 @@ class LensScreenView extends GOScreenView { tandem: this.controlsTandem.createTandem( 'dragLockedButton' ) } ); this.controlsLayer.addChild( dragLockedButton ); + model.opticalObjectChoiceProperty.link( opticalObjectChoice => { + dragLockedButton.enabled = !OpticalObjectChoice.isArrowObject( opticalObjectChoice ); + } ); const lightSourceSceneNode = new LightSourceSceneNode( model.lightSourceScene, this.visibleProperties, this.modelViewTransform, this.modelVisibleBoundsProperty, this.modelBoundsProperty, model.raysTypeProperty, { diff --git a/js/mirror/model/MirrorModel.ts b/js/mirror/model/MirrorModel.ts index 621a575b..d5d41a70 100644 --- a/js/mirror/model/MirrorModel.ts +++ b/js/mirror/model/MirrorModel.ts @@ -32,8 +32,12 @@ class MirrorModel extends GOModel { // Initial position of the framed object, empirically set so that the optical axis goes through its center. framedObjectPosition: new Vector2( -170, 72.5 ), + arrowObject1Position: new Vector2( -150, 50 ), + arrowObject2Position: new Vector2( -150, -50 ), + // optical object choices, in the order that they will appear in OpticalObjectChoiceComboBox opticalObjectChoices: [ + OpticalObjectChoice.ARROW, OpticalObjectChoice.PENCIL, OpticalObjectChoice.PENGUIN, OpticalObjectChoice.PLANET,