Skip to content

Latest commit

 

History

History
2736 lines (2225 loc) · 160 KB

CHANGELOG.md

File metadata and controls

2736 lines (2225 loc) · 160 KB

Change Log

All notable changes to this project will be documented in this file. This project adheres to Semantic Versioning.

Breaking Changes

  • System.priority is refactored to be static.
  • ex.Timer now only takes the option bag constructor
  • PreDrawEvent, PostDrawEvent, PreTransformDrawEvent, PostTransformDrawEvent, PreUpdateEvent, PostUpdateEvent now use elapsedMs instead of delta for the elapsed milliseconds between the last frame.460696
  • Trigger API has been slightly changed:
    • action now returns the triggering entity: (entity: Entity) => void
    • target now works in conjunction with filter instead of overwriting it.
    • EnterTriggerEvent and ExitTriggerEvent now contain a entity: Entity property instead of actor: Actor
  • ex.Vector.normalize() return zero-vector ((0,0)) instead of (0,1) when normalizing a vector with a magnitude of 0
  • ex.Gif transparent color constructor arg is removed in favor of the built in Gif file mechanism
  • Remove core-js dependency, it is no longer necessary in modern browsers. Technically a breaking change for older browsers
  • ex.Particle and ex.ParticleEmitter now have an API that looks like modern Excalibur APIs
    • particleSprite is renamed to graphic
    • particleRotationalVelocity is renamed to angularVelocity
    • fadeFlag is renamed to fade
    • acceleration is renamed to acc
    • particleLife is renamed to life
    • ParticleEmitter now takes a separate particle: ParticleConfig parameter to disambiguate between particles parameters and emitter ones
      const emitter =  new ex.ParticleEmitter({
          width: 10,
          height: 10,
          radius: 5,
          emitterType: ex.EmitterType.Rectangle,
          emitRate: 300,
          isEmitting: true,
          particle: {
            transform: ex.ParticleTransform.Global,
            opacity: 0.5,
            life: 1000,
            acc: ex.vec(10, 80),
            beginColor: ex.Color.Chartreuse,
            endColor: ex.Color.Magenta,
            startSize: 5,
            endSize: 100,
            minVel: 100,
            maxVel: 200,
            minAngle: 5.1,
            maxAngle: 6.2,
            fade: true,
            maxSize: 10,
            graphic: swordImg.toSprite(),
            randomRotation: true,
            minSize: 1
          }
        });

Deprecated

  • Vector.size is deprecated, use Vector.magnitude instead
  • ScreenShader v_texcoord is deprecated, use v_uv. This is changed to match the materials shader API
  • actor.getGlobalPos() - use actor.globalPos instead
  • actor.getGlobalRotation() - use actor.globalRotation instead
  • actor.getGlobalScale() - use actor.globalScale instead

Added

  • Child ex.Actor inherits opacity of parents
  • ex.Engine.timeScale values of 0 are now supported
  • ex.Trigger now supports all valid actor constructor parameters from ex.ActorArgs in addition to ex.TriggerOptions
  • ex.Gif can now handle default embedded GIF frame timings
  • New ex.Screen.worldToPagePixelRatio API that will return the ratio between excalibur pixels and the HTML pixels.
    • Additionally excalibur will now decorate the document root with this same value as a CSS variable --ex-pixel-ratio
    • Useful for scaling HTML UIs to match your game
      .ui-container {
        pointer-events: none;
        position: absolute;
        transform-origin: 0 0;
        transform: scale(
          calc(var(--pixel-conversion)),
          calc(var(--pixel-conversion)));
      }
  • New updates to ex.coroutine(...)
    • New ex.CoroutineInstance is returned (still awaitable)
    • Control coroutine autostart with ex.coroutine(function*(){...}, {autostart: false})
    • .start() and .cancel() coroutines
    • Nested coroutines!
  • Excalibur will now clean up WebGL textures that have not been drawn in a while, which improves stability for long game sessions
    • If a graphic is drawn again it will be reloaded into the GPU seamlessly
  • You can now query for colliders on the physics world
      const scene = ...;
      const colliders = scene.physics.query(ex.BoundingBox.fromDimensions(...));
  • actor.oldGlobalPos returns the globalPosition from the previous frame
  • create development builds of excalibur that bundlers can use in dev mode
  • show warning in development when Entity hasn't been added to a scene after a few seconds
  • New RentalPool type for sparse object pooling
  • New ex.SparseHashGridCollisionProcessor which is a simpler (and faster) implementation for broadphase pair generation. This works by bucketing colliders into uniform sized square buckets and using that to generate pairs.
  • CollisionContact can be biased toward a collider by using contact.bias(collider). This adjusts the contact so that the given collider is colliderA, and is helpful if you are doing mtv adjustments during precollision.

Fixed

  • Fixed issue where ex.Label where setting the opacity of caused a multiplicative opacity effect when actor opacity set
  • Fixed issue where the ex.Loader would have a low res logo on small configured resolution sizes
  • Fixed issue where ex.Gif was not parsing certain binary formats correctly
  • Fixed issue where the boot ex.Loader was removing pixelRatio override
  • Fixed ex.RasterOptions, it now extends ex.GraphicsOptions which is the underlying truth
  • Fixed issue where rayCast filter would not be called in hit order
  • Fixed issue where rayCasts would return inconsistent orderings with the ex.SparseHashGridCollisionProcessor strategy
  • Fixed issue where CircleCollider tangent raycast did not work correctly
  • Fixed issue where you were required to provide a transition if you provided a loader in the ex.Engine.start('scene', { loader })
  • Fixed issue where ex.Scene.onPreLoad(loader: ex.DefaultLoader) would lock up the engine if there was an empty loader
  • Fixed issue where ex.Scene scoped input events would preserve state and get stuck causing issues when switching back to the original scene.
  • Fixed issue where not all physical keys from the spec were present in ex.Keys including the reported ex.Keys.Tab
  • Fixed invalid graphics types around ex.Graphic.tint
  • improve types to disallow invalid combo of collider/width/height/radius in actor args
  • only add default color graphic for the respective collider used
  • Fixed issue where ex.SpriteFont did not respect scale when measuring text
  • Fixed issue where negative transforms would cause collision issues because polygon winding would change.
  • Fixed issue where removing and re-adding an actor would cause subsequent children added not to function properly with regards to their parent/child transforms
  • Fixed issue where ex.GraphicsSystem would crash if a parent entity did not have a ex.TransformComponent
  • Fixed a bug in the new physics config merging, and re-arranged to better match the existing pattern

Updates

  • Non-breaking parameters that reference delta to elapsedMs to better communicate intent and units

  • Perf improvements to ex.ParticleEmitter

    • Use the same integrator as the MotionSystem in the tight loop
    • Leverage object pools to increase performance and reduce allocations
  • Perf improvements to collision narrowphase and solver steps

    • Working in the local polygon space as much as possible speeds things up
    • Add another pair filtering condition on the SparseHashGridCollisionProcessor which reduces pairs passed to narrowphase
    • Switching to c-style loops where possible
    • Caching get component calls
    • Removing allocations where it makes sense
  • Perf Side.fromDirection(direction: Vector): Side - thanks @ikudrickiy!

  • Perf improvements to PointerSystem by using new spatial hash grid data structure

  • Perf improvements: Hot path allocations

    • Reduce State/Transform stack hot path allocations in graphics context
    • Reduce Transform allocations
    • Reduce AffineMatrix allocations
  • Perf improvements to CircleCollider bounds calculations

  • Switch from iterators to c-style loops which bring more speed

    • Entity component iteration
    • EntityManager iteration
    • EventEmitters
    • GraphicsSystem entity iteration
    • PointerSystem entity iteration

Changed

  • Applied increased TS strictness:
    • Resource API subtree
    • Graphics API subtree
    • TileMap API subtree

[v0.29.3]

Breaking Changes

  • ex.Action now requires a unique id property
  • Z-indexes are now relative to the parent's Z-index. You can get the global Z-index with the globalZ property on the Actor or TransformComponent.

Deprecated

Added

  • Built in actions now have a unique id property
  • globalZ property to Actor and TransformComponent

Fixed

  • Fixed animation glitch caused by uninitialized state in ImageRenderer
  • Fixed issue where ex.Loader.suppressPlayButton = true did not work. Only using the ex.Engine({suppressPlayButton: true}) worked

Updates

Changed

  • ex.Vector.toAngle() now returns angles from [0 - 2 PI)

[v0.29.2]

Breaking Changes

Deprecated

  • `

Added

  • Added ability to configure image wrapping on ex.ImageSource with the new ex.ImageWrapping.Clamp (default), ex.ImageWrapping.Repeat, and ex.ImageWrapping.Mirror.
    const image = new ex.ImageSource('path/to/image.png', {
      filtering: ex.ImageFiltering.Pixel,
      wrapping: {
        x: ex.ImageWrapping.Repeat,
        y: ex.ImageWrapping.Repeat,
      }
    });
  • Added pointer event support to ex.TileMap's and individual ex.Tile's
  • Added pointer event support to ex.IsometricMap's and individual ex.IsometricTile's
  • Added useAnchor parameter to ex.GraphicsGroup to allow users to opt out of anchor based positioning, if set to false all graphics members will be positioned with the top left of the graphic at the actor's position.
    const graphicGroup = new ex.GraphicsGroup({
      useAnchor: false,
      members: [
        {
          graphic: heartImage.toSprite(),
          offset: ex.vec(0, 0),
        },
        {
          graphic: heartImage.toSprite(),
          offset: ex.vec(0, 16),
        },
        {
          graphic: heartImage.toSprite(),
          offset: ex.vec(16, 16),
        },
        {
          graphic: heartImage.toSprite(),
          offset: ex.vec(16, 0),
        },
      ],
    });
  • Added simplified ex.coroutine overloads, you need not pass engine as long as you are in an Excalibur lifecycle
    const result = ex.coroutine(function* () {...});
  • Added way to bind 'this' to ex.coroutine overloads, you need not pass engine as long as you are in an Excalibur lifecycle
    const result = ex.coroutine({myThis: 'cool'}, function* () {...});
  • Added optional ex.coroutine timing parameter to schedule when they are updated
    const result = ex.coroutine(engine, function * () {...}, { timing: 'postupdate' })
  • Added GraphicsComponent.bounds which will report the world bounds of the graphic if applicable!
  • Added ex.Vector.EQUALS_EPSILON to configure the ex.Vector.equals(v) threshold
  • Added way to add custom WebGL context lost/recovered handlers for your game
    const game = new ex.Engine({
      handleContextLost: (e) => {...},
      handleContextRestored: (e) => {...}
    })

Fixed

  • Fixed issue where ex.TileMap culling did not work properly when using fixed updates lower than refresh rate
  • Fixed incomplete types for font options in ex.FontSource().toFont(options)
  • Fixed issue with ex.Loader start button position when using CSS transforms
  • Fixed issue where adding scenes with the same name did not work when it was previously removed
  • Fixed issue when WebGL context lost occurs where there was no friendly output to the user
  • Fixed issue where HiDPI scaling could accidentally scale past the 4k mobile limit, if the context would scale too large it will now attempt to recover by backing off.
  • Fixed issue where logo was sometimes not loaded during ex.Loader
  • Fixed issue where unbounded containers would grow infinitely when using the following display modes:
    • DisplayMode.FillContainer
    • DisplayMode.FitContainer
    • DisplayMode.FitContainerAndFill
    • DisplayMode.FitContainerAndZoom
  • Fixed issue where ex.ParticleEmitter z-index did not propagate to particles
  • Fixed incongruent behavior as small scales when setting transform.scale = v and transform.scale.setTo(x, y)
  • Fixed ex.coroutine TypeScript type to include yielding undefined
  • Fixed issue where Firefox on Linux would throw an error when using custom Materials due to unused attributes caused by glsl compiler optimization.
  • Fixed issue where start transition did not work properly if deferred
  • Fixed issue where transitions did not cover the whole screen if camera was zoomed
  • Fixed issue where Color.toHex() produced invalid strings if the channel values are negative or fractional, or if the alpha channel was different than 1

Updates

Changed

  • Significant 2x performance improvement to image drawing in Excalibur
  • Simplified ex.Loader viewport/resolution internal configuration

[v0.29.0]

Breaking Changes

  • ex.Entity.tags is now a javascript Set instead of an Array this will affect methods that inspected tags as an array before.

  • ex.Engine.goToScene's second argument now takes GoToOptions instead of just scene activation data

    {
      /**
       * Optionally supply scene activation data passed to Scene.onActivate
      */
      sceneActivationData?: TActivationData,
      /**
       * Optionally supply destination scene "in" transition, this will override any previously defined transition
      */
      destinationIn?: Transition,
      /**
       * Optionally supply source scene "out" transition, this will override any previously defined transition
      */
      sourceOut?: Transition,
      /**
       * Optionally supply a different loader for the destination scene, this will override any previously defined loader
      */
      loader?: DefaultLoader
    }
  • ex.Physics static is marked as deprecated, configuring these setting will move to the ex.Engine({...}) constructor

    const engine = new ex.Engine({
      ...
      physics: {
        solver: ex.SolverStrategy.Realistic,
        gravity: ex.vec(0, 20),
        arcade: {
          contactSolveBias: ex.ContactSolveBias.VerticalFirst
        },
      }
    })
  • Changed the Font default base align to Top this is more in line with user expectations. This does change the default rendering to the top left corner of the font instead of the bottom left.

  • Remove confusing Graphics Layering from ex.GraphicsComponent, recommend we use the ex.GraphicsGroup to manage this behavior

    • Update ex.GraphicsGroup to be consistent and use offset instead of pos for graphics relative positioning
  • ECS implementation has been updated to remove the "stringly" typed nature of components & systems

    • For average users of Excalibur folks shouldn't notice any difference
    • For folks leveraging the ECS, Systems/Components no longer have type parameters based on strings. The type itself is used to track changes.
    • class MySystem extends System<'ex.component'> becomes class MySystem extends System
    • class MyComponent extends Component<'ex.component'> becomes class MyComponent extends Component
    • ex.System.update(elapsedMs: number) is only passed an elapsed time
  • Prevent people from inadvertently overriding update() in ex.Scene and ex.Actor. This method can still be overridden with the //@ts-ignore pragma

  • ex.SpriteSheet.getSprite(...) will now throw on invalid sprite coordinates, this is likely always an error and a warning is inappropriate. This also has the side benefit that you will always get a definite type out of the method.

Deprecated

Added

  • Added new ex.Tilemap.getOnScreenTiles() method to help users access onscreen tiles for logic or other concerns.

  • Added ex.FontSource resource type

    const fontSource = new ex.FontSource('/my-font.ttf', 'My Font')
    loader.addResource(fontSource)
    
    game.start(loader).then(() => {
      const font = fontSource.toFont() // returns ex.Font
    })

    Font options can be defined either at the source or at the toFont() call. If defined in both, toFont(options) will override the options in the FontSource.

    const fontSource = new ex.FontSource('/my-font.ttf', 'My Font', { 
      filtering: ex.ImageFiltering.Pixel,
      size: 16, // set a default size
    })
    const font = fontSource.toFont({
      // override just the size
      size: 20,
    })
  • Added fullscreen after load feature! You can optionally provide a fullscreenContainer with a string id or an instance of the HTMLElement

    new ex.Loader({
      fullscreenAfterLoad: true,
      fullscreenContainer: document.getElementById('container')
    });
  • Added new ex.Debug static for more convenient debug drawing where you might not have a graphics context accessible to you. This works by batching up all the debug draw requests and flushing them during the debug draw step.

    • ex.Debug.drawRay(ray: Ray, options?: { distance?: number, color?: Color })
    • ex.Debug.drawBounds(boundingBox: BoundingBox, options?: { color?: Color })
    • ex.Debug.drawCircle(center: Vector, radius: number, options?: ...)
    • ex.Debug.drawPolygon(points: Vector[], options?: { color?: Color })
    • ex.Debug.drawText(text: string, pos: Vector)
    • ex.Debug.drawLine(start: Vector, end: Vector, options?: LineGraphicsOptions)
    • ex.Debug.drawLines(points: Vector[], options?: LineGraphicsOptions)
    • drawPoint(point: Vector, options?: PointGraphicsOptions)
  • Experimental ex.coroutine for running code that changes over time, useful for modeling complex animation code. Coroutines return a promise when they are complete. You can think of each yield as a frame.

    • The result of a yield is the current elapsed time
    • You can yield a number in milliseconds and it will wait that long before resuming
    • You can yield a promise and it will wait until it resolves before resuming
      const completePromise = coroutine(engine, function * () {
        let elapsed = 0;
        elapsed = yield 200; // frame 1 wait 200 ms before resuming
        elapsed = yield fetch('./some/data.json'); // frame 2
        elapsed = yield; // frame 3
      });
  • Added additional options in rayCast options

    • ignoreCollisionGroupAll: boolean will ignore testing against anything with the CollisionGroup.All which is the default for all
    • filter: (hit: RayCastHit) => boolean will allow people to do arbitrary filtering on raycast results, this runs very last after all other collision group/collision mask decisions have been made
  • Added additional data side and lastContact to onCollisionEnd and collisionend events

  • Added configuration option to ex.PhysicsConfig to configure composite collider onCollisionStart/End behavior

  • Added configuration option to ex.TileMap({ meshingLookBehind: Infinity }) which allows users to configure how far the TileMap looks behind for matching colliders (default is 10).

  • Added Arcade Collision Solver bias to help mitigate seams in geometry that can cause problems for certain games.

    • ex.ContactSolveBias.None No bias, current default behavior collisions are solved in the default distance order
    • ex.ContactSolveBias.VerticalFirst Vertical collisions are solved first (useful for platformers with up/down gravity)
    • ex.ContactSolveBias.HorizontalFirst Horizontal collisions are solved first (useful for games with left/right predominant forces)
      const engine = new ex.Engine({
        ...
        physics: {
          solver: ex.SolverStrategy.Realistic,
          arcade: {
            contactSolveBias: ex.ContactSolveBias.VerticalFirst
          },
        }
      })
  • Added Graphics opacity on the Actor constructor new ex.Actor({opacity: .5})

  • Added Graphics pixel offset on the Actor constructor new ex.Actor({offset: ex.vec(-15, -15)})

  • Added new new ex.Engine({uvPadding: .25}) option to allow users using texture atlases in their sprite sheets to configure this to avoid texture bleed. This can happen if you're sampling from images meant for pixel art

  • Added new antialias settings for pixel art! This allows for smooth subpixel rendering of pixel art without shimmer/fat-pixel artifacts.

    • Use new ex.Engine({pixelArt: true}) to opt in to all the right defaults to make this work!
  • Added new antialias configuration options to deeply configure how Excalibur does any antialiasing, or you can provide antialiasing: true/antialiasing: false to use the old defaults.

    • Example;
    const game = new ex.Engine({
       antialiasing: {
           pixelArtSampler: false,
           filtering: ex.ImageFiltering.Pixel,
           nativeContextAntialiasing: false,
           canvasImageRendering: 'pixelated'
       }
    })
  • Added new lineHeight property on SpriteFont and Font to manually adjust the line height when rendering text.

  • Added missing dual of ex.GraphicsComponent.add(), you can now ex.GraphicsComponent.remove(name);

  • Added additional options to ex.Animation.fromSpriteSheetCoordinates() you can now pass any valid ex.GraphicOptions to influence the sprite per frame

    const anim = ex.Animation.fromSpriteSheetCoordinates({
      spriteSheet: ss,
      frameCoordinates: [
        {x: 0, y: 0, duration: 100, options: { flipHorizontal: true }},
        {x: 1, y: 0, duration: 100, options: { flipVertical: true }},
        {x: 2, y: 0, duration: 100},
        {x: 3, y: 0, duration: 100}
      ],
      strategy: ex.AnimationStrategy.Freeze
    });
  • Added additional options to ex.SpriteSheet.getSprite(..., options). You can pass any valid ex.GraphicOptions to modify a copy of the sprite from the spritesheet.

    const sprite = ss.getSprite(0, 0, {
      flipHorizontal: true,
      flipVertical: true,
      width: 200,
      height: 201,
      opacity: .5,
      scale: ex.vec(2, 2),
      origin: ex.vec(0, 1),
      tint: ex.Color.Red,
      rotation: 4
    });
  • New simplified way to query entities ex.World.query([MyComponentA, MyComponentB])

  • New way to query for tags on entities ex.World.queryTags(['A', 'B'])

  • Systems can be added as a constructor to a world, if they are the world will construct and pass a world instance to them

    world.add(MySystem);
    ...
    
    class MySystem extends System {
      query: Query;
      constructor(world: World) {
        super()
        this.query = world.query([MyComponent]);
      }
    
      update
    }
  • Added RayCastHitas part of every raycast not just the physics world query!

    • Additionally added the ray distance and the contact normal for the surface
  • Added the ability to log a message once to all log levels

    • debugOnce
    • infoOnce
    • warnOnce
    • errorOnce
    • fatalOnce
  • Added ability to load additional images into ex.Materials!

    const noise = new ex.ImageSource('./noise.avif');
    loader.addResource(noise);
    
    var waterMaterial = game.graphicsContext.createMaterial({
      name: 'water',
      fragmentSource: waterFrag,
      color: ex.Color.fromRGB(55, 0, 200, .6),
      images: {
        u_noise: noise
      }
    });
  • Scene Transition & Loader API, this gives you the ability to have first class support for individual scene resource loading and scene transitions.

    • Add or remove scenes by constructor

    • Add loaders by constructor

    • New ex.DefaultLoader type that allows for easier custom loader creation

    • New ex.Transition type for building custom transitions

    • New scene lifecycle to allow scene specific resource loading

      • onTransition(direction: "in" | "out") {...}
      • onPreLoad(loader: DefaultLoader) {...}
    • New async goToScene() API that allows overriding loaders/transitions between scenes

    • Scenes now can have async onInitialize and async onActivate!

    • New scenes director API that allows upfront definition of scenes/transitions/loaders

    • Example: Defining scenes upfront

      const game = new ex.Engine({
        scenes: {
          scene1: {
            scene: scene1,
            transitions: {
              out: new ex.FadeInOut({duration: 1000, direction: 'out', color: ex.Color.Black}),
              in: new ex.FadeInOut({duration: 1000, direction: 'in'})
            }
          },
          scene2: {
            scene: scene2,
            loader: ex.DefaultLoader, // Constructor only option!
            transitions: {
              out: new ex.FadeInOut({duration: 1000, direction: 'out'}),
              in: new ex.FadeInOut({duration: 1000, direction: 'in', color: ex.Color.Black })
            }
          },
        scene3: ex.Scene // Constructor only option!
        } 
      })
      
      // Specify the boot loader & first scene transition from loader
      game.start('scene1',
      {
        inTransition: new ex.FadeInOut({duration: 500, direction: 'in', color: ex.Color.ExcaliburBlue})
        loader: boot,
      });
    • Scene specific input API so that you can add input handlers that only fire when a scene is active!
      class SceneWithInput extends ex.Scene {
        onInitialize(engine: ex.Engine<any>): void {
          this.input.pointers.on('down', () => {
            console.log('pointer down from scene1');
          });
        }
      }
      class OtherSceneWithInput extends ex.Scene {
        onInitialize(engine: ex.Engine<any>): void {
          this.input.pointers.on('down', () => {
            console.log('pointer down from scene2');
          });
        }
      }

Fixed

  • Performance improvement in ex.TileMap finding onscreen tiles is now BLAZINGLY FAST thanks to a suggestion from Kristen Maeyvn in the Discord.
    • TileMaps no longer need a quad tree, we can calculate the onscreen tiles with math by converting the screen into tilemap space 😎
  • Fixed bug where ex.TileMap.getTileByPoint() did not take into account the rotation/scale of the tilemap.
  • Fixes issue where mis-matched coordinate planes on parent/children caused bizarre issues. Now children are forced to inherit their parent's coordinate plane, it will always be the coordinate plane of the top most parent.
  • Fixed issue with Log ScreenAppender utility where it was not positioned correctly, you can now deeply configure it!
    export interface ScreenAppenderOptions {
      engine: Engine;
      /**
       * Optionally set the width of the overlay canvas
      */
      width?: number;
      /**
       * Optionally set the height of the overlay canvas
      */
      height?: number;
      /**
       * Adjust the text offset from the left side of the screen
      */
      xPos?: number;
      /**
       * Provide a text color
      */
      color?: Color;
      /**
       * Optionally set the CSS zindex of the overlay canvas
      */
      zIndex?: number;
    }
  • Fixed errant warning about resolution when using pixelRatio on low res games to upscale
  • Fixes an issue where a collider that was part of a contact that was deleted did not fire a collision end event, this was unexpected
  • Fixes an issue where you may want to have composite colliders behave as constituent colliders for the purposes of start/end collision events. A new property is added to physics config, the current behavior is the default which is 'together', this means the whole composite collider is treated as 1 collider for onCollisionStart/onCollisionEnd. Now you can configure a separate which will fire onCollisionStart/onCollisionEnd for every separate collider included in the composite (useful if you are building levels or things with gaps that you need to disambiguate). You can also configure this on a per composite level to mix and match CompositeCollider.compositeStrategy
  • Fixed issue where particles would have an errant draw if using a particle sprite
  • Fixed issue where a null/undefined graphics group member graphic would cause a crash, now logs a warning.
  • Fixed issue where Actor built in components could not be extended because of the way the Actor based type was built.
    • Actors now use instance properties for built-ins instead of getters
    • With the ECS refactor you can now subtype built-in Components and .get(Builtin) will return the correct subtype.
    class MyBodyComponent extends ex.BodyComponent {}
    
    class MyActor extends ex.Actor {
        constructor() {
          super({})
          this.removeComponent(ex.BodyComponent);
          this.addComponent(new MyBodyComponent())
        }
    }
    
    const myActor = new MyActor();
    const myBody = myActor.get(ex.BodyComponent); // Returns the new MyBodyComponent subtype!
  • Fixed issue with snapToPixel where the ex.Camera was not snapping correctly
  • Fixed issue where using CSS transforms on the canvas confused Excalibur pointers
  • Fixed issue with *AndFill suffixed [[DisplayModes]]s where content area offset was not accounted for in world space
  • Fixed issue where ex.Sound.getTotalPlaybackDuration() would crash if not loaded, now logs friendly warning
  • Fixed issue where an empty constructor on new ex.Label() would crash

Updates

Changed

[v0.28.7]

Breaking Changes

Deprecated

Added

Fixed

  • Fixed issue where pointer events did not work properly when using [[ScreenElement]]s
  • Fixed issue where debug draw was not accurate when using *AndFill suffixed [[DisplayMode]]s

Updates

Changed

  • Changed the default ex.PointerComponent.useGraphicsBounds = true, users expect this to just work by default.
  • Changed a rough edge in the ex.Material API, if a material was created with a constructor it was lazily initialized. However this causes confusion because now the two ways of creating a material behave differently (the shader is not available immediately on the lazy version). Now ex.Material requires the GL graphics context to make sure it always works the same.
  • Changed a rough edge in the ex.Material API, if a material was created with a constructor it was lazily initialized. However this causes confusion because now the two ways of creating a material behave differently (the shader is not available immediately on the lazy version). Now ex.Material requires the GL graphics context to make sure it always works the same.

[v0.28.6]

Breaking Changes

Deprecated

Added

  • Added arbitrary data storage in isometric tiles, ex.IsometricTile.data this brings it into feature parity with normal ex.Tile.data
  • New graphics events and hooks that allow you to hook into graphics drawing before or after any drawing transformations have been applied
    • Actor.graphics.onPreTransformDraw with the corresponding event .on('pretransformdraw')
    • Actor.graphics.onPostTransformDraw with the corresponding event .on('posttransformdraw')
  • New property and methods overloads to ex.Animation
    • ex.Animation.currentFrameTimeLeft will return the current time in milliseconds left in the current
    • ex.Animation.goToFrame(frameNumber: number, duration?: number) now accepts an optional duration for the target frame
    • ex.Animation.speed can set the speed multiplier on an animation 1 = 1x speed, 2 = 2x speed.

Fixed

  • Fixed issue where nesting ex.CompositeColliders inside one another would cause a crash on collision
  • Fixed issue where ex.CompositeColliders did not respect collider offset
  • Fixed issue where parenting a entity with fixed updates on would cause a drawing flicker, transform interpolation now is aware of changing parents so it interpolates drawing continuously to prevent any flickering
  • ex.Animation.reset() did not properly reset all internal state

Updates

Changed

[v0.28.5]

Breaking Changes

Deprecated

Added

  • Added collision lifecycle convenience methods to Actor, you can now override the following events
    class MyActor extends ex.Actor {
      constructor(args: ex.ActorArgs) {
        super(args);
      }
      onPreCollisionResolve(self: ex.Collider, other: ex.Collider, side: ex.Side, contact: ex.CollisionContact): void {
        
      }
      onPostCollisionResolve(self: ex.Collider, other: ex.Collider, side: ex.Side, contact: ex.CollisionContact): void {
        
      }
      onCollisionStart(self: ex.Collider, other: ex.Collider, side: ex.Side, contact: ex.CollisionContact): void {
        
      }
      onCollisionEnd(self: ex.Collider, other: ex.Collider): void {
        
      }
    }
  • Added Scene specific background color
  • Added ability to apply draw offset to ex.IsometricMap and ex.Tilemap
  • Added visibility and opacity to ex.IsometricMap
  • Added base elevation for ex.IsometricMap so multiple maps can sort correctly
  • Added method to suppress convex polygon warning for library code usage
  • Added more configuration options to debug draw flags, including isometric map controls
  • Added actionstart and actioncomplete events to the Actor that are fired when an action starts and completes

Fixed

  • Fixed issue where the Camera wasn't interpolated during fixed update, which is very noticeable when using camera locked strategies
  • Fixed issue where IsometricMap would debug draw collision geometry on non-solid tiles
  • Fixed issue where CompositeCollider offset was undefined if not set
  • Fixed Actor so it receives predraw/postdraw events per the advertised strongly typed events
  • Fixed infinite loop 💣 when certain degenerate polygons were attempted to be triangulated!
  • Fixed incorrect type on ex.Tilemap.getTileByPoint()
  • Fixed TS type on GraphicsComponent and allow .material to be null to unset, current workaround is using .material = null as any

Updates

Changed

  • All debug geometry settings are controlled from debug.collider now
  • Removed dunder prefixed parameters from overrideable methods
  • Tweaked debug draw to be less noisy by default
  • Removed dependency on ex.IsometricMap in the ex.IsometricEntityComponent, this allows for greater flexibility when using the component when a map may not be known or constructed.

[v0.28.4]

Breaking Changes

Deprecated

Added

  • Ability to configure TileMap debug drawing with the ex.Engine.debug.tilemap property.
  • Materials have a new convenience method for updating uniforms
    game.input.pointers.primary.on('move', evt => {
      heartActor.pos = evt.worldPos;
      swirlMaterial.update(shader => {
        shader.trySetUniformFloatVector('iMouse', evt.worldPos);
      });
    });

Fixed

  • Fixed issue where TileMap solid tiles tile packing algorithm would incorrectly merge tiles in certain situations.
  • Sprite tint was not respected when supplied in the constructor, this has been fixed!
  • Adjusted the FontCache font timeout to 400 ms and makes it configurable as a static FontCache.FONT_TIMEOUT. This is to help prevent a downward spiral on mobile devices that might take a long while to render a few starting frames causing the cache to repeatedly clear and never recover.

Updates

  • Materials can now reference a new uniform for the screen texture and a screen uv attribute in their fragment shaders

    • u_screen_texture - This is the texture of the screen right before the material draw call
    • a_screenuv - The vertex attribute corresponding to the screen uv relative to the current graphic
    • v_screenuv - The fragment varying corresponding to the screen uv relative to the current graphic
  • Materials can now reference the current time in their shaders

    • u_time_ms - This is the ms since page navigation (performance.now() under the hood)

Changed

  • TileMap debug draw is now less verbose by default to save draw cycles when toggling to debug

[v0.28.3]

Breaking Changes

Deprecated

Added

  • Added new feature to collision group raycasting, directly provide a collisionMask that you want to search for.
const playerGroup = ex.CollisionGroupManager.create('playerGroup');
const notPlayersMask = ~playersGroup.category;
const hits = engine.currentScene.physics.rayCast(
  new ex.Ray(player.pos, playerDir),
  {
    maxDistance: playerSightDistance,
    // Search for all categories that match the mask
    collisionMask: notPlayers,
    searchAllColliders: false
  });

Fixed

  • Fixed issue where rendering multiple materials at once would crash the renderer
  • Fixed issue where raycasting with more complex collision groups was not working as expected

Updates

Changed

[v0.28.2]

Breaking Changes

Deprecated

Added

  • Added ex.Engine.version to report the current excalibur version build string
  • Added new ex.Screen.events
    • screen.events.on('resize', (evt) => ) Will emit when the screen is resized
    • screen.events.on('fullscreen', (evt) => ) Will emit when the screen is changed into browser fullscreen mode
    • screen.events.on('pixelratio', (evt) => ) Will emit when the screen's pixel ratio changes (moving from a hidpi screen to a non, or vice versa)

Fixed

  • Fixed issue where removing handlers by function reference only removed the first registered one
  • Fixed issue where play button was hidden when going fullscreen mode
  • Fixed issue where screen resizing caused artifacts on the loading screen
  • Fixed bug in useCanvas2DFallback() where antialiasing settings could be lost
  • Fixed bug in useCanvas2DFallback() where opacity was not respected in `save
  • Fixed typo in trigger event signature entertrigger should have been enter
  • Fixed typo in trigger event signature exittrigger should have been exit
  • Fixed typo in animation event signature ended should have been end
  • Fixed issue where some excalibur clear() implementations modified the collection they were iterating over
  • Fixed async issue where sound could not be stopped if stop()/start() were called in rapid succession
  • Fixed issue with input mapper where keyboard.wasPressed(...) did not fire
  • Fixed issue issue where TileMaps would not properly draw Tiles when setup in screen space coordinates
  • Fixed issue where the ex.Line graphics bounds were incorrect causing erroneous offscreen culling
  • Fixed event type signature on ex.Engine.input.pointers.primary.on('wheel', ...) for wheel events

Updates

  • Improved performance in TileMaps when drawing tiles using QuadTree data structure

Changed

  • Changed the canvas 2d fallback default, no longer is enabled by default. Developers must opt in.
  • Allow entity names to be set after construction! Entities will now default to a name "Entity#1234" followed by an id.

[v0.28.0]

Breaking Changes

  • Removed ex.Class base class type, this was a common base class for many excalibur types that provided old on/off event functionality. This functionality has been preserved on the types that had it before using ex.EventEmitter

Deprecated

  • The ex.Input.* import site is deprecated, will be removed in v0.29.0. All the imports are still available on ex. now
  • [[ex.Input.Gamepad]] isButtonPressed has been renamed to isButtonHeld
  • ex.EventDispatcher is marked deprecated, will eventually be removed in v0.29.0

Added

  • Added new ex.InputMapper for mapping multiple input sources into actions! This can be useful for providing accessibility into your games and allowing users to map inputs to different game commands.

     const moveRight = (amount: number) => { actor.vel.x = 100 * amount }
     const moveLeft = (amount: number) => { actor.vel.x = -100 * amount }
     const moveUp = (amount: number) => { actor.vel.y = -100 * amount }
     const moveDown = (amount: number) => { actor.vel.y = 100 * amount }
     engine.inputMapper.on(({keyboard}) => keyboard.isHeld(ex.Keys.ArrowRight) ? 1 : 0, moveRight);
     engine.inputMapper.on(({gamepads}) => gamepads.at(0).isButtonPressed(ex.Buttons.DpadRight) ? 1 : 0, moveRight);
     engine.inputMapper.on(({gamepads}) => gamepads.at(0).getAxes(ex.Axes.LeftStickX) > 0 ? gamepads.at(0).getAxes(ex.Axes.LeftStickX) : 0, moveRight);
  • Added strongly typed events with ex.EventEmitter<TEventMap>

  • Added new convenience properties for flipping all the graphics on an Actor

    • ex.Actor.graphics.flipHorizontal - Flips all the graphics horizontally
    • ex.Actor.graphics.flipVertical - Flips all the graphics vertically
  • Added new ex.Scene.transfer(actor) method for transferring actors between scenes, useful if you want to only have an actor in 1 scene at a time.

  • Added new ex.Material to add custom shaders per ex.Actor!

    • This feature cant be applied using the ex.Actor.graphics.material = material property or by setting the material property on the ex.ExcaliburGraphicsContext.material = material with .save()/.restore()
    • This feature opt out of batch rendering and issues a separate draw call
    • A custom vertex shader can be provided, otherwise a default will be provided
    • A number of default uniforms are available to shaders
      • Pre-built varyings:
        • in vec2 v_uv - UV coordinate
      • Pre-built uniforms:
        • uniform sampler2D u_graphic - The current graphic displayed by the GraphicsComponent
        • uniform vec2 u_resolution - The current resolution of the screen
        • uniform vec2 u_size; - The current size of the graphic
        • uniform vec4 u_color - The current color of the material
        • uniform float u_opacity - The current opacity of the graphics context
    const material = new ex.Material({
      name: 'test',
      color: ex.Color.Red,
      fragmentSource: `#version 300 es
      precision mediump float;
      // UV coord
      in vec2 v_uv;
      uniform sampler2D u_graphic;
      uniform vec4 u_color;
      uniform float u_opacity;
      out vec4 fragColor;
      void main() {
        vec4 color = u_color;
        color = texture(u_graphic, v_uv);
        color.rgb = color.rgb * u_opacity;
        color.a = color.a * u_opacity;
        fragColor = color * u_color;
      }`
    });
  • Added updates to ex.PostProcessor

    • New optional ex.PostProcessor.onUpdate hook for updating custom uniforms
    • Added default uniforms that are automatically added
      • uniform float u_time_ms - total playback time in milliseconds
      • uniform float u_elapsed_ms - the elapsed time from the last frame in milliseconds
      • uniform vec2 u_resolution - the resolution of the canvas (in pixels)
  • Added new helper called ex.Animation.fromSpriteSheetCoordinates to help build animations more tersely from SpriteSheets

     const spriteSheet = SpriteSheet.fromImageSource({...});
        const anim = Animation.fromSpriteSheetCoordinates({
      spriteSheet,
      frameCoordinates: [
        {x: 0, y: 5, duration: 100},
        {x: 1, y: 5, duration: 200},
        {x: 2, y: 5, duration: 100},
        {x: 3, y: 5, duration: 500}
      ],
      strategy: AnimationStrategy.PingPong
     });
  • Added new FrameEvent to ex.Animation which includes the frame index of the current frame!

      const anim = new Animation();
    
      // TS autocompletes the handler
      anim.on('frame', (frame: FrameEvent) => {
        // Do stuff on frame
      });
  • Added new typed ex.EventEmitter which will eventually replace the old ex.EventDispatcher, this gives users a way of strongly typing the possible events that can be emitted using a type map. This is loosely typed you can still emit any event you want, you only get type completion suggestions for the type map.

    export type AnimationEvents = {
      frame: FrameEvent;
      loop: Animation;
      ended: Animation;
    };
    
    export class Animation {
      public events = new EventEmitter<AnimationEvents>();
      ...
    }
    
    const anim = new Animation();
    
    // TS autocompletes the handler
    anim.on('frame', (frame: FrameEvent) => {
      // Do stuff on frame
    });
  • Added ability to perform arbitrary ray casts into ex.Scene, the ex.PhysicsWorld can be passed a variety of options to influence the types of ray cast hits that are returned

      const engine = new ex.Engine({...});
      const enemyGroup = ex.CollisionGroupManager.create('enemy');
      const ray = new ex.Ray(ex.vec(0, 0), ex.Vector.Right);
      const hits = engine.currentScene.physics.rayCast(ray, {
        /**
         * Optionally specify to search for all colliders that intersect the ray cast, not just the first which is the default
         */
        searchAllColliders: true,
        /**
         * Optionally specify the maximum distance in pixels to ray cast, default is Infinity
         */
        maxDistance: 100,
        /**
         * Optionally specify a collision group to consider in the ray cast, default is All
         */
        collisionGroup: enemyGroup
      });
  • Added word-wrap support for ex.Text using the optional parameter maxWidth

  • Added the emitted particle transform style as part of ex.ParticleEmitter({particleTransform: ex.ParticleTransform.Global}), [[ParticleTransform.Global]] is the default and emits particles as if they were world space objects, useful for most effects. If set to [[ParticleTransform.Local]] particles are children of the emitter and move relative to the emitter as they would in a parent/child actor relationship.

  • Added wasButtonReleased and wasButtonPressed methods to [[ex.Input.Gamepad]]

  • Added clone() method to ex.SpriteSheet

Fixed

  • Fixed issue with ex.TileMap collider consolidation where custom colliders would prevent normal solid tile colliders from being included.
  • Fixed memory leak in the internal ex.EntityManager, it did not properly clear internal state when removing entities
  • Fixed issue where scaling a ex.TileMap didn't properly offscreen cull due to the bounds not scaling properly.
  • Fixed issue where ex.Text.flipHorizontal or ex.Text.flipVertical would not work
  • Fixed issue where overriding existing components did not work properly because of deferred component removal
  • Fixed issue where ex.ScreenElement pointer events were not working by default.
  • Fixed issue where setting lineWidth on ex.Circle was not accounted for in the bitmap
  • Fixed issue in macos where the meta key would prevent keyup's from firing correctly
  • Fixed issue when excalibur was hosted in a x-origin iframe, the engine will grab window focus by default if in an iframe. This can be suppressed with new ex.Engine({grabWindowFocus: false})
  • Fixed issue where ex.Camera.rotation = ... did not work to rotate the camera, also addressed offscreen culling issues that were revealed by this fix.
  • Fixed issue where the ex.ScreenElement anchor was not being accounted for properly when passed as a constructor parameter.
  • Fixed issue where you could not use multiple instances of Excalibur on the same page, you can now have as many Excalibur's as you want (up to the webgl context limit).
  • Fixed issue where ex.ScreenElement would log a warning when created without a height or width
  • Fixed issue where ex.Sound would get confused parsing and playing sound files with a querystring in their path
  • Fixed issue where ex.ColliderComponent was not deeply cloning the stored ex.Collider causing them to be shared across clones.
  • Fixed issue where ex.GraphicsComponent was not deeploy cloning the stored ex.Graphics causing them to be shared across clones.
  • Fixed issue where Actor.clone() and Entity.clone() crashed.
  • Fixed issue where zero mtv collisions cause erroneous precollision events to be fired in the ArcadeSolver and RealisticSolver
  • Fixed issue where calling .kill() on a child entity would not remove it from the parent Entity
  • Fixed issue where calling .removeAllChildren() would not remove all the children from the parent Entity
  • Fixed issue where world origin was inconsistent when the using ex.DisplayMode.FitScreenAndFill when the screen was resized.
  • Fixed issue where context opacity was not respected when set in a preDraw
  • Fixed issue where ex.Sound.loop was not working, and switching tab visibility would cause odd behavior with looping ex.Sound
  • Fixed issue where adding a ex.ParticleEmitter as a child did not position particles according to the parent
  • Fixed issue where screenshots from ex.Engine.screenshot() did not match the smoothing set on the engine.
  • Fixed incorrect event type returned when ex.Actor.on('postupdate', (event) => {...}).
  • Fixed issue where using numerous ex.Text instances would cause Excalibur to crash webgl by implementing a global font cache.
  • Fixed issue where child entities did not inherit the scene from their parent
  • Fixed issue where ex.Font would become corrupted when re-used by multiple ex.Text instances
  • Fixed engine.on('visible') event not firing
  • Fixed EventDispatcher.emit converting falsy values to ex.GameEvent. It will only convert undefined or null values now.

Updates

Changed

  • Excalibur will now use ex.EventEmitter to broadcast events, Excalibur types that have events support will also have an .events member.
  • Excalibur resources by default no longer add cache busting query string to resources. All built in resources now expose a bustCache property to allow setting this before loading, for example ex.Sound.bustCache.

[0.27.0] - 2022-07-08

Breaking Changes

  • ex.Engine.snapToPixel now defaults to false, it was unexpected to have pixel snapping on by default it has now been switched.
  • The ex.Physics.useRealisticPhysics() physics solver has been updated to fix a bug in bounciness to be more physically accurate, this does change how physics behaves. Setting ex.Body.bounciness = 0 will simulate the old behavior.
  • ex.TransformComponent.posChanged$ has been removed, it incurs a steep performance cost
  • ex.EventDispatcher meta events 'subscribe' and 'unsubscribe' were unused and undocumented and have been removed
  • ex.TileMap tlies are now drawn from the lower left by default to match with ex.IsometricMap and Tiled, but can be configured with renderFromTopOfGraphic to restore the previous behavior.
  • Scene onActivate and onDeactivate methods have been changed to receive a single parameter, an object containing the previousScene, nextScene, and optional data passed in from goToScene()

Deprecated

Added

  • Added new configurable ex.TileMap option for rendering from the bottom or the top of the graphic, this matches with ex.IsometricMap and how Tiled renders renderFromTopOfGraphic, by default false and renders from the bottom.

    const tileMap = new ex.TileMap({
      renderFromTopOfGraphic: false
    })
  • Added new ex.Future type which is a convenient way of wrapping a native browser promise and resolving/rejecting later

    const future = new ex.Future();
    const promise = future.promise; // returns promise
    promise.then(() => {
      console.log('Resolved!');
    });
    future.resolve(); // resolved promise
  • Added new ex.Semaphore type to limit the number of concurrent cans in a section of code, this is used internally to work around a chrome browser limitation, but can be useful for throttling network calls or even async game events.

    const semaphore = new ex.Semaphore(10); // Only allow 10 concurrent between enter() and exit()
    ...
    
    await semaphore.enter();
    await methodToBeLimited();
    semaphore.exit();
  • Added new ex.WatchVector type that can observe changes to x/y more efficiently than ex.watch()

  • Added performance improvements

    • ex.Vector.distance improvement
    • ex.BoundingBox.transform improvement
  • Added ability to clone ex.Vector.clone(destVector) into a destination vector

  • Added new ex.Transform type that is a light weight container for transformation data. This logic has been extracted from the ex.TransformComponent, this makes it easy to pass ex.Transforms around. Additionally the extracted ex.Transform logic has been refactored for performance.

  • Added new ex.AffineMatrix that is meant for 2D affine transformations, it uses less memory and performs less calculations than the ex.Matrix which uses a 4x4 Float32 matrix.

  • Added new fixed update step to Excalibur! This allows developers to configure a fixed FPS for the update loop. One advantage of setting a fix update is that you will have a more consistent and predictable physics simulation. Excalibur graphics will be interpolated automatically to avoid any jitter in the fixed update.

    • If the fixed update FPS is greater than the display FPS, excalibur will run multiple updates in a row (at the configured update elapsed) to catch up, for example there could be X updates and 1 draw each clock step.
    • If the fixed update FPS is less than the display FPS, excalibur will skip updates until it meets the desired FPS, for example there could be no update for 1 draw each clock step.
    const game = new ex.Engine({
      fixedUpdateFps: 20 // 20 fps fixed update, or a fixed update delta of 50 milliseconds
    });
    // turn off interpolation on a per actor basis
    const actor = new ex.Actor({...});
    actor.body.enableFixedUpdateInterpolate = false;
    game.add(game);
  • Allowed setting playback ex.Sound.duration which will limit the amount of time that a clip plays from the current playback position.

  • Added a new lightweight ex.StateMachine type for building finite state machines

    const machine = ex.StateMachine.create({
      start: 'STOPPED',
      states: {
        PLAYING: {
          onEnter: () => {
            console.log("playing");
          },
          transitions: ['STOPPED', 'PAUSED']
        },
        STOPPED: {
          onEnter: () => {
            console.log("stopped");
          },
          transitions: ['PLAYING', 'SEEK']
        },
        SEEK: {
          transitions: ['*']
        },
        PAUSED: {
          onEnter: () => {
            console.log("paused")
          },
          transitions: ['PLAYING', 'STOPPED']
        }
      }
    });
  • Added ex.Sound.seek(positionInSeconds) which will allow you to see to a place in the sound, this will implicitly pause the sound

  • Added ex.Sound.getTotalPlaybackDuration() which will return the total time in the sound in seconds.

  • Allow tinting of ex.Sprite's by setting a new tint property, renderers must support the tint property in order to function.

    const imageSource = new ex.ImageSource('./path/to/image.png');
    await imageSource.load();
    const sprite = imageSource.toSprite();
    sprite.tint = ex.Color.Red;
  • Added ex.Sound.getPlaybackPosition() which returns the current playback position in seconds of the currently playing sound.

  • Added ex.Sound.playbackRate which allows developers to get/set the current rate of playback. 1.0 is the default playback rate, 2.0 is twice the speed, and 0.5 is half speed.

  • Added missing ex.EaseBy action type, uses ex.EasingFunctions to move relative from the current entity position.

  • Added 2 new Action types to enable running parallel actions. ex.ActionSequence which allows developers to specify a sequence of actions to run in order, and ex.ParallelActions to run multiple actions at the same time.

    const actor = new ex.Actor();
    const parallel = new ex.ParallelActions([
      new ex.ActionSequence(actor, ctx => ctx.moveTo(ex.vec(100, 0), 100)),
      new ex.ActionSequence(actor, ctx => ctx.rotateTo(Math.PI/2, Math.PI/2))
    ]);
    actor.actions.runAction(parallel);
    // actor will now move to (100, 100) and rotate to Math.PI/2 at the same time!!
  • Add target element id to ex.Screen.goFullScreen('some-element-id') to influence the fullscreen element in the fullscreen browser API.

  • Added optional data parameter to goToScene, which gets passed to the target scene's onActivate method.

    class SceneA extends ex.Scene {
      /* ... */
    
      onActivate(context: ex.SceneActivationContext<{ foo: string }>) {
        console.log(context.data.foo); // bar
      }
    }
    
    engine.goToScene('sceneA', { foo: 'bar' })
    • Added the ability to select variable duration into Timer constructor.
    const random = new ex.Random(1337);
    const timer = new ex.Timer({
      random,
      interval: 500,
      randomRange: [0, 500]
    })

Fixed

  • Fixed issue with ex.Canvas and ex.Raster graphics that forced their dimensions to the next highest power of two.

  • Fixed issue with ex.Engine.snapToPixel where positions very close to pixel boundary created jarring 1 pixel oscillations.

  • Fixed bug where a deferred goToScene would preserve the incorrect scene so engine.add(someActor) would place actors in the wrong scene after transitioning to another.

  • Fixed usability issue and log warning if the ex.ImageSource is not loaded and a draw was attempted.

  • Fixed bug in ex.Physics.useRealisticPhysics() solver where ex.Body.bounciness was not being respected in the simulation

  • Fixed bug in ex.Physics.useRealisticPhysics() solver where ex.Body.limitDegreeOfFreedom was not working all the time.

  • Fixed bug in Clock.schedule where callbacks would not fire at the correct time, this was because it was scheduling using browser time and not the clock's internal time.

  • Fixed issue in Chromium browsers where Excalibur crashes if more than 256 Image.decode() calls are happening in the same frame.

  • Fixed issue where ex.EdgeCollider were not working properly in ex.CompositeCollider for ex.TileMap's

  • Fixed issue where ex.BoundingBox overlap return false due to floating point rounding error causing multiple collisions to be evaluated sometimes

  • Fixed issue with ex.EventDispatcher where removing a handler that didn't already exist would remove another handler by mistake

  • Fixed issue with ex.EventDispatcher where concurrent modifications of the handler list where handlers would or would not fire correctly and throw

  • Tweak to the ex.ArcadeSolver to produce more stable results by adjusting by an infinitesimal epsilon

    • Contacts with overlap smaller than the epsilon are ignored
    • Colliders with bounds that overlap smaller than the epsilon are ignored
  • Fixed issue with ex.ArcadeSolver based collisions where colliders were catching on seams when sliding along a floor of multiple colliders. This was by sorting contacts by distance between bodies. sorted-collisions

  • Fixed issue with ex.ArcadeSolver where corner contacts would zero out velocity even if the bodies were already moving away from the contact "divergent contacts". cancel-velocity-fix

  • Fixed issue where ex.Sound wasn't being paused when the browser window lost focus

Updates

  • Updated the collision system to improve performance
    • Cache computed values where possible
    • Avoid calculating transformations until absolutely necessary
    • Avoid calling methods in tight loops

Changed

  • ex.Engine.configurePerformanceCanvas2DFallback no longer requires threshold or showPlayerMessage
  • ex.Engine.snapToPixel now defaults to false
  • Most places where ex.Matrix was used have been switched to ex.AffineMatrix
  • Most places where ex.TransformComponent was used have been switched to ex.Transform

[0.26.0] - 2022-05-20

Breaking Changes

  • ex.Line has be replaced with a new Graphics type, the old geometric behavior is now under the type ex.LineSegment
  • Notable deprecated types removed
    • ex.SortedList old sorted list is removed
    • ex.Collection old collection type is removed
    • ex.Util import site, exported code promoted ex.*
    • ex.DisplayMode.Position is removed, use CSS to position the canvas
    • ex.Trait interface, traits are not longer supported
    • ex.Promises old promise implementation is removed in favor of browser promises
  • Notable method & property removals
    • ex.Actor
      • .getZIndex() and .setZIndex() removed use .z
    • ex.Scene
      • .screenElements removed in favor of .entities
      • .addScreenElement(...) removed use .add(...)
      • .addTileMap(...) removed use .add(...)
      • .removeTileMap(...) removed use .remove(...)
    • ex.Timer
      • .unpause() removed use .resume()
    • ex.Camera
      • .rx removed use .angularVelocity
    • ex.BodyComponent
      • .sx removed use .scaleFactor
      • .rx removed use .angularVelocity
    • ex.ActionsComponent
      • .asPromise() removed use .toPromise()
    • ex.ActionContext
      • .asPromise() removed use .toPromise()
    • ex.Color
      • Misspellings corrected
  • The old drawing API had been removed from excalibur, this should not affect you unless you were using the ex.Flags.useLegacyDrawing() or ex.Flags.useCanvasGraphicsContext().
    • Notably all implementations of Drawable are removed, use the new Graphics API
    • Methods on actor ex.Actor.setDrawing(...), ex.Actor.addDrawing(...) are removed, use the ex.Actor.graphics.add(...), ex.Actor.graphics.show(...) and ex.Actor.graphics.use(...)
    • The ex.Actor.onPreDraw(...) and ex.Actor.onPostDraw(...) are removed, use ex.Actor.graphics.onPreDraw(...) and ex.Actor.graphics.onPostDraw(...)
    • The events predraw and postdraw are removed
    • ex.Scene.onPreDraw() and ex.Scene.onPostDraw() are now called with the ExcaliburGraphicsContext instead of an CanvasRenderingContext2D
  • ex.TileMap has several breaking changes around naming, but brings it consistent with Tiled terminology and the new ex.IsometricMap. Additionally the new names are easier to follow.
    • Constructor has been changed to the following
       new ex.TileMap({
        pos: ex.vec(100, 100),
        tileWidth: 64,
        tileHeight: 48,
        rows: 20,
        columns: 20
      });
    • ex.Cell has been renamed to ex.Tile
      • ex.Tile now uses addGraphic(...), removeGraphic(...), clearGraphics() and getGraphics() instead of having an accessible ex.Tile.graphics array.
    • ex.TileMap.data has been renamed to ex.TileMap.tiles
    • ex.TileMap.getCell(..) has been renamed to ex.TileMap.getTile(...)
    • ex.TileMap.getCellByIndex(...) has been renamed to ex.TileMap.getTileByIndex(...)
    • ex.TileMap.getCellByPoint(...) has been renamed to ex.TileMap.getTileByPoint(...)

Deprecated

Added

  • Added new parameter to ex.Raster({quality: 4}) to specify the internal scaling for the bitmap, this is useful for improving the rendering quality of small rasters due to sampling error.

  • Added new ex.Line graphics object for drawing lines!

    const lineActor = new ex.Actor({
      pos: ex.vec(100, 0)
    });
    lineActor.graphics.anchor = ex.Vector.Zero;
    lineActor.graphics.use(new ex.Line({
      start: ex.vec(0, 0),
      end: ex.vec(200, 200),
      color: ex.Color.Green,
      thickness: 10
    }));
    game.add(lineActor);
  • Added new performance fallback configuration to ex.Engine for developers to help players experiencing poor performance in non-standard browser configurations

    • This will fallback to the Canvas2D rendering graphics context which usually performs better on non hardware accelerated browsers, currently postprocessing effects are unavailable in this fallback.
    • By default if a game is running at 20fps or lower for 100 frames or more after the game has started it will be triggered, the developer can optionally show a player message that is off by default.
      var game = new ex.Engine({
        ...
        configurePerformanceCanvas2DFallback: {
          allow: true, // opt-out of the fallback
          showPlayerMessage: true, // opt-in to a player pop-up message
          threshold: { fps: 20, numberOfFrames: 100 } // configure the threshold to trigger the fallback
        }
      });
  • Added new ex.ParallaxComponent for creating parallax effects on the graphics, entities with this component are drawn differently and a collider will not be where you expect. It is not recommended you use colliders with parallax entities.

    const actor = new ex.Actor();
    // The actor will be drawn shifted based on the camera position scaled by the parallax factor
    actor.addComponent(new ParallaxComponent(ex.vec(0.5, 0.5)));
  • Added feature to build SpriteSheets from a list of different sized source views using ex.SpriteSheet.fromImageSourceWithSourceViews(...)

      const ss = ex.SpriteSheet.fromImageSourceWithSourceViews({
        image,
        sourceViews: [
          {x: 0, y: 0, width: 20, height: 30},
          {x: 20, y: 0, width: 40, height: 50},
        ]
      });
  • Added draw call sorting new ex.Engine({useDrawSorting: true}) to efficiently draw render plugins in batches to avoid expensive renderer switching as much as possible. By default this is turned on, but can be opted out of.

  • Added the ability to clone into a target Matrix this is useful to save allocations and in turn garbage collection pauses.

  • ex.Engine now support setting the pixel ratio in the constructor new ex.Engine({pixelRatio: 2}), this is useful for smooth ex.Text rendering when antialiasing: false and rendering pixel art type graphics

  • ex.TileMap now supports per Tile custom colliders!

    const tileMap = new ex.TileMap(...);
    const tile = tileMap.getTile(0, 0);
    tile.solid = true;
    tile.addCollider(...); // add your custom collider!
  • New ex.IsometricMap for drawing isometric grids! (They also support custom colliders via the same mechanism as ex.TileMap)

    new ex.IsometricMap({
        pos: ex.vec(250, 10),
        tileWidth: 32,
        tileHeight: 16,
        columns: 15,
        rows: 15
      });
    • ex.IsometricTile now come with a ex.IsometricEntityComponent which can be applied to any entity that needs to be correctly sorted to preserve the isometric illusion
    • ex.IsometricEntitySystem generates a new z-index based on the elevation and y position of an entity with ex.IsometricEntityComponent
  • Added arbitrary non-convex polygon support (only non-self intersecting) with ex.PolygonCollider(...).triangulate() which builds a new ex.CompositeCollider composed of triangles.

  • Added faster ex.BoundingBox.transform(...) implementation.

  • Added faster ex.BoundingBox.overlap(...) implementation.

  • Added ex.Vector.min(...) and ex.Vector.max(...) to find the min/max of each vector component between 2 vectors.

  • Added ex.TransformComponent.zIndexChange$ observable to watch when z index changes.

  • Added new display mode ex.DisplayMode.FitContainerAndFill.

  • Added new display mode ex.DisplayMode.FitScreenAndFill.

  • Added new display mode ex.DisplayMode.FitContainerAndZoom.

  • Added new display mode ex.DisplayMode.FitScreenAndZoom.

Fixed

  • Fixed unreleased issue where fixed update interpolation was incorrect with child actors
  • Fixed unreleased bug where CompositeCollider components would not collide appropriately because contacts did not have unique ids
  • Fixed issue where CompositeColliders treat separate constituents as separate collisionstart/collisionend which is unexpected
  • Fixed issue where resources that failed to load would silently fail making debugging challenging
  • Fixed issue where large pieces of Text were rendered as black rectangles on mobile, excalibur now internally breaks these into smaller chunks in order to render them.
  • Fixed issue #2263 where keyboard input wasPressed was not working in the onPostUpdate lifecycle
  • Fixed issue #2263 where there were some keys missing from the ex.Input.Keys enum, including Enter
  • Fixed issue where Rectangle line renderer did not respect z order

Updates

  • Performance improvement to the ex.Loader screen keeping frame rates higher by only updating the backing ex.Canvas when there are changes
  • Improved collision broadphase by swapping to a more efficient ex.BoundingBox.overlaps check
  • Improved collision narrowphase by improving ex.PolygonCollider calculations for localBounds, bounds, and transformed point geometry
  • Improved Text/Font performance by internally caching expensive native measureText() calls
  • Performance improvement to GraphicsSystem
  • Performance improvement to the transform capture of the previous frame transform and motion

Changed

  • Split offscreen detection into a separate system
  • Renamed ex.Matrix.multv() and ex.Matrix.multm() to ex.Matrix.multiply() which matches our naming conventions

[0.25.3] - 2022-02-05

Breaking Changes

  • Small breaking change to engine.screenshot() you must now use await engine.screenshot(). This avoids copy buffer performance impact of preserveDrawingBuffer: true by capturing a screen shot request on the next frame when the buffer has not yet been cleared.

Deprecated

Added

Fixed

  • Fixed issue where collision normals are inaccurate on polygon colliders that offset from their origin
  • Fixed issue where only Pixel 6 devices crash when using their MAX_TEXTURE_IMAGE_UNITS, artificially cap Excalibur to 125 textures max
  • Fixed issue [#2224] where pointer events sometimes didn't work in mobile platforms due to touch-action not being set to none
  • Fixed issue [#2203] where engine.screenshot() did not work in the WebGL implementation
  • Fixed issue [#1528] where screenshots didn't match the displayed game's size in HiDPI displays, images are now consistent with the game. If you want the full scaled image pass engine.screenshot(true) to preserve HiDPI Resolution.
  • Fixed issue [#2206] error and warning logs for large images to help developers identify error situations in the webgl implementation

Updates

Changed

[0.25.2] - 2022-01-21

Breaking Changes

  • ex.Util.extend() is removed, modern js spread operator {...someobject, ...someotherobject} handles this better.
  • Excalibur post processing is now moved to the engine.graphicsContext.addPostProcessor()
  • Breaking change to ex.PostProcessor, all post processors must now now implement this interface
    export interface PostProcessor {
      intialize(gl: WebGLRenderingContext): void;
      getShader(): Shader;
    }

Deprecated

  • The static Engine.createMainLoop is now marked deprecated and will be removed in v0.26.0, it is replaced by the Clock api
  • Mark legacy draw routines in ex.Engine, ex.Scene, and ex.Actor deprecated

Added

  • Added ability to build custom renderer plugins that are accessible to the ex.ExcaliburGraphicsContext.draw<TCustomRenderer>(...) after registering them ex.ExcaliburGraphicsContext.register(new LineRenderer())

  • Added ability to draw circles and rectangles with outlines! ex.ExcaliburGraphicsContext.drawCircle(...) and ex.ExcaliburGraphicsContext.drawRectangle(...)

  • Added ex.CoordPlane can be set in the new ex.Actor({coordPlane: CoordPlane.Screen}) constructor

  • Added convenience feature, setting the color, sets the color on default graphic if applicable

  • Added a DebugGraphicsComponent for doing direct debug draw in the DebugSystem

  • Added back TileMap debug draw

  • Added ex.Scene.timers to expose the list of timers

  • Added support for different webgl texture blending modes as ex.ImageFiltering :

    • ex.ImageFiltering.Blended - Blended is useful when you have high resolution artwork and would like it blended and smoothed
    • ex.ImageFiltering.Pixel - Pixel is useful when you do not want smoothing aka antialiasing applied to your graphics.
  • Excalibur will set a "default" blend mode based on the ex.EngineOption antialiasing property, but can be overridden per graphic

    • antialiasing: true, then the blend mode defaults to ex.ImageFiltering.Blended
    • antialiasing: false, then the blend mode defaults to ex.ImageFiltering.Pixel
  • ex.Text/ex.Font defaults to blended which improves the default look of text rendering dramatically!

  • ex.Circle and ex.Polygon also default to blended which improves the default look dramatically!

  • ex.ImageSource can now specify a blend mode before the Image is loaded, otherwise

  • Added new measureText method to the ex.SpriteFont and ex.Font to return the bounds of any particular text

  • Added new Clock api to manage the core main loop. Clocks hide the implementation detail of how the mainloop runs, users just knows that it ticks somehow. Clocks additionally encapsulate any related browser timing, like performance.now()

    1. StandardClock encapsulates the existing requestAnimationFrame api logic
    2. TestClock allows a user to manually step the mainloop, this can be useful for frame by frame debugging #1170
    3. The base abstract clock implements the specifics of elapsed time
  • Added a new feature to Engine options to set a maximum fps new ex.Engine({...options, maxFps: 30}). This can be useful when needing to deliver a consistent experience across devices.

  • Pointers can now be configured to use the collider or the graphics bounds as the target for pointers with the ex.PointerComponent

    • useColliderShape - (default true) uses the collider component geometry for pointer events
    • useGraphicsBounds - (default false) uses the graphics bounds for pointer events

Fixed

  • Fixed issue [#2192] where Actor.center was not correct in child actors
  • Fixed issue where ex.CircleColliders did not respect rotation/scale when offset
  • Fixed issue [#2157] when compiling in TS strict mode complaining about ex.Poolable
  • Fixed issue where scaled graphics were not calculating the correct bounds
  • Fixed unreleased issue where clock implementation was not updating frame id
  • Fixed alpha pre-multiply math in multiple shaders
  • Fixed label initialization of fonts, passing a font in the constructor work
  • Fixed bug in sprite bounds calculations not taking scale into account
  • Fixed bug with pointer api where clicking on screen coordinate actors didn't work
  • Fixed [#1815] issue where Camera would jitter when using a strategies based off of actors in the previous frame.
  • Fixed issue where TileMaps would sometimes have a geometry seam that may not fall on an actual screen pixel causing a visible gap between tiles and the background -- image
  • Fixed unreleased issue where SpriteFonts log every frame they detect a misconfigured font.
  • Fixed unreleased issue where clock when constraining fps would pass larger than expected elapsed times to the simulation causing things to "speed up" bizarrely
  • Fixed unreleased issue where games with no resources would crash
  • Fixed issue [#2152] where shared state in ex.Font and ex.SpriteFont prevented text from aligning properly when re-used
  • Fixed issue where fast moving CompositeColliders were erroneously generating pairs for their constituent parts
  • Fixed Safari 13.1 crash when booting Excalibur because of they odd MediaQuery API in older Safari
  • Fixed issue where pointers did not work because of missing types
  • Fixed issue with ArcadeSolver where stacked/overlapped tiles would double solve the position of the collider for the same overlap
  • Fixed issue where changing the ex.Sprite.width or ex.Sprite.height did not resize the graphic.
  • Fixed issue where initial Actor anchors set in the constructor were not being set in the graphics component
  • EventDispatcher
    • EventDispatcher - doesn't require the target object. The context of this is not tampered anymore.
  • Pointers
    • PointerAbstraction - is fixed to maintain reference

Updates

  • The following Engine's pieces: Collision Graphics Resources Trigger are updated to reflect the new EventDispatcher behavior.
  • Refactor camera/screen interaction to utilize transforms instead of bespoke coordinate conversion

Changed

  • Updated Graphics to improve general performance
  • Updated the webgl primitives to make building ex.Shaders, ex.VertexBuffers, and ex.VertexLayouts much easier
  • Broke up the internal monolithic shader into separate internal renderer plugins
  • Changed the debug system to separate displaying the debug position point (game.debug.transform.showPosition = true) and debug position label (game.debug.transform.showPositionLabel = true)
  • ex.ColorBlindCorrector is renamed to ex.ColorBlindnessPostProcessor, and ex.ColorBlindness is renamed to ex.ColorBlindnessMode
    • Color blindness can still be corrected or simulated:
      • game.debug.colorBlindMode.correct(ex.ColorBlindnessMode.Deuteranope)
      • game.debug.colorBlindMode.simulate(ex.ColorBlindnessMode.Deuteranope)
  • Excalibur now uses pre-multiplied alpha automatically, images will be unpacked into memory using gl.pixelStorei(gl.UNPACK_PREMULTIPLY_ALPHA_WEBGL, true)
  • Excalibur FPS is now sampled over 100ms blocks, this gives a more usable fps in the stats. The sampler is available off of the engine clock engine.clock.fpsSampler.fps
  • Pointer Events:
    • Event types (up, down, move, etc) now all exist in 2 types ex.Input.PointerEvent and ex.Input.WheelEvent
    • The stopPropagation() method used to cancel further dispatches has been renamed to cancel() to match other events API.
    • Events no longer have a reference to the pointer but now have all of the same information that was availabe on the pointer worldPos, screenPos, pagePos

0.25.1 - 2021-11-05

Added

  • Experimental: Native ES module bundle distribution in package esm/excalibur.js entrypoint (#2064)
  • withEngine utils support an aditional options parameter to override the Engine default options.
  • Story to show a play / pause implementation.
  • ex.Animation now support totalDuration that will calculate automatically each frame duration based on how many frames have.
  • ex.Animation now supports .reverse() to reverse the direction of play in an animation, use the ex.Animation.direction to inspect if the animation is playing in the ex.AnimationDirection.Forward direction or the ex.AnimationDirection.Backward direction.

Changed

  • Pointer system refactored into 2 parts:
    • First is an ECS style system ex.PointerSystem that dispatches events to Entities/Actors
    • Second is an event receiver ex.PointerEventReceiver which is responsible for collecting the native browser events
    • The API is mostly backwards compatible breaking changes are listed in the breaking change section, event types have been simplified, and stopPropagation() and been renamed to cancel()
  • Internal Actions implementation converted to ECS system and component, this is a backwards compatible change with v0.25.0
    • ex.ActionsSystem and ex.ActionsComponent now wrap the existing ex.ActionContext
    • Actions can be shared with all entities now!
  • Dispatch the hidePlayButton on the Button Event to prevent that keep on the screen on some situations [#1431].
  • Revert VSCode Workbench Colors

Deprecated

  • Actions asPromise() renamed to toPromise()

Fixed

  • Fixed loader button position on window resize
  • Fixed issue with setting ex.TileMap.z to a value
  • Fixed crash in debug system if there is no collider geometry
  • Fixed ImageSource loading error message [#2049]

0.25.0 - 2021-10-03

Breaking Changes

  • Actor Drawing: ex.Actor.addDrawing, ex.Actor.setDrawing, onPostDraw(), and onPreDraw() are no longer on by default and will be removed in v0.26.0, they are available behind a flag ex.Flags.useLegacyDrawing()

    • For custom drawing use the ex.Canvas
  • ex.Actor.rx has been renamed to ex.Actor.angularVelocity

  • Rename ex.Edge to ex.EdgeCollider and ex.ConvexPolygon to ex.PolygonCollider to avoid confusion and maintian consistency

  • ex.Label constructor now only takes the option bag constructor and the font properties have been replaced with ex.Font

    const label = new ex.Label({
      text: 'My Text',
      x: 100,
      y: 100,
      font: new ex.Font({
        family: 'Consolas',
        size: 32
      })
    });
  • ex.Physics.debug properties for Debug drawing are now moved to engine.debug.physics, engine.debug.collider, and engine.debug.body.

    • Old debugDraw(ctx: CanvasRenderingContext2D) methods are removed.
  • Collision Pair's are now between Collider's and not bodies

  • PerlinNoise has been removed from the core repo will now be offered as a plugin

  • Legacy drawing implementations are moved behind ex.LegacyDrawing new Graphics implemenations of Sprite, SpriteSheet, Animation are now the default import.

    • To use any of the ex.LegacyDrawing.* implementations you must opt-in with the ex.Flags.useLegacyDrawing() note: new graphics do not work in this egacy mode
  • Renames CollisionResolutionStrategy.Box collision resolution strategy to Arcade

  • Renames CollisionResolutionStrategy.RigidBody collision resolution strategy to Realistic

  • Collider is now a first class type and encapsulates what Shape used to be. Collider is no longer a member of the Body

  • CollisionType and CollisionGroup are now a member of the Body component, the reasoning is they define how the simulated physics body will behave in simulation.

  • Timer's no longer automatically start when added to a Scene, this Timer.start() must be called. (#1865)

  • Timer.complete is now read-only to prevent odd bugs, use reset(), stop(), and start() to manipulate timers.

  • Actor.actions.repeat() and Actor.actions.repeatForever() now require a handler that specifies the actions to repeat. This is more clear and helps prevent bugs like #1891

    const actor = new ex.Actor();
    
    actor.actions
      // Move up in a zig-zag by repeating 5 times
      .repeat((ctx) => {
        ctx.moveBy(10, 0, 10);
        ctx.moveBy(0, 10, 10);
      }, 5)
      .callMethod(() => {
        console.log('Done repeating!');
      });
  • Removes Entity.components as a way to access, add, and remove components

  • ex.Camera.z has been renamed to property ex.Camera.zoom which is the zoom factor

  • ex.Camera.zoom(...) has been renamed to function ex.Camera.zoomOverTime()

  • TileMap no longer needs registered SpriteSheets, Sprite's can be added directly to Cell's with addGraphic

  • Directly changing debug drawing by engine.isDebug = value has been replaced by engine.showDebug(value) and engine.toggleDebug() (#1655)

  • UIActor Class instances need to be replaced to ScreenElement (This Class it's marked as Obsolete) (#1656)

  • Switch to browser based promise, the Excalibur implementation ex.Promise is marked deprecated (#994)

  • DisplayMode's have changed (#1733) & (#1928):

    • DisplayMode.FitContainer fits the screen to the available width/height in the canvas parent element, while maintaining aspect ratio and resolution
    • DisplayMode.FillContainer update the resolution and viewport dyanmically to fill the available space in the canvas parent element, DOES NOT preserve aspectRatio
    • DisplayMode.FitScreen fits the screen to the available browser window space, while maintaining aspect ratio and resolution
    • DisplayMode.FillScreen now does what DisplayMode.FullScreen used to do, the resolution and viewport dynamically adjust to fill the available space in the window, DOES NOT preserve aspectRatio (#1733)
    • DisplayMode.FullScreen is now removed, use Screen.goFullScreen().
  • SpriteSheet now is immutable after creation to reduce chance of bugs if you modified a public field. The following properties are read-only: columns, rows, spWidth, spHeight, image, sprites and spacing.

  • Engine.pointerScope now defaults to a more expected ex.Input.PointerScope.Canvas instead of ex.Input.PointerScope.Document which can cause frustrating bugs if building an HTML app with Excalibur

Added

  • New property center to Screen to encapsulate screen center coordinates calculation considering zoom and device pixel ratio
  • New ex.Shape.Capsule(width, height) helper for defining capsule colliders, these are useful for ramps or jagged floor colliders.
  • New collision group constructor argument added to Actornew Actor({collisionGroup: collisionGroup})
  • SpriteSheet.getSprite(x, y) can retrieve a sprite from the SpriteSheet by x and y coordinate. For example, getSprite(0, 0) returns the top left sprite in the sheet.
    • SpriteSheet's now have dimensionality with rows and columns optionally specified, if not there is always 1 row, and sprites.length columns
  • new Actor({radius: 10}) can now take a radius parameter to help create circular actors
  • The ExcaliburGraphicsContext now supports drawing debug text
  • Entity may also now optionally have a name, this is useful for finding entities by name or when displaying in debug mode.
  • New DebugSystem ECS system will show debug drawing output for things toggled on/off in the engine.debug section, this allows for a less cluttered debug experience.
    • Each debug section now has a configurable color.
  • Turn on WebGL support with ex.Flags.useWebGL()
  • Added new helpers to CollisionGroup to define groups that collide with specified groups CollisionGroup.collidesWith([groupA, groupB])
    • Combine groups with const groupAandB = CollisionGroup.combine([groupA, groupB])
    • Invert a group instance const everthingButGroupA = groupA.invert()
  • Improved Collision Simulation
    • New ECS based CollisionSystem and MotionSystem
    • Rigid body's can now sleep for improved performance
    • Multiple contacts now supported which improves stability
    • Iterative solver for improved stability
  • Added ColliderComponent to hold individual Collider implementations like Circle, Box, or CompositeCollider
    • Actor.collider.get() will get the current collider
    • Actor.collider.set(someCollider) allows you to set a specific collider
  • New CompositeCollider type to combine multiple colliders together into one for an entity
    • Composite colliders flatten into their individual colliders in the collision system
    • Composite collider keeps it's internal colliders in a DynamicTree for fast .collide checks
  • New TransformComponent to encapsulate Entity transform, that is to say position, rotation, and scale
  • New MotionComponent to encapsulate Entity transform values changing over time like velocity and acceleration
  • Added multi-line support to Text graphics (#1866)
  • Added TileMap arbitrary graphics support with .addGraphic() (#1862)
  • Added TileMap row and column accessors getRows() and getColumns() (#1859)
  • Added the ability to store arbitrary data in TileMap cells with Cell.data.set('key', 'value') and Cell.data.get('key') (#1861)
  • Actions moveTo(), moveBy(), easeTo(), scaleTo(), and scaleBy() now have vector overloads
  • Animation.fromSpriteSheet will now log a warning if an index into the SpriteSheet is invalid (#1856)
  • new ImageSource() will now log a warning if an image type isn't fully supported. (#1855)
  • Timer.start() to explicitly start timers, and Timer.stop() to stop timers and "rewind" them.
  • Timer.timeToNextAction will return the milliseconds until the next action callback
  • Timer.timeElapsedTowardNextAction will return the milliseconds counted towards the next action callback
  • BoundingBox now has a method for detecting zero dimensions in width or height hasZeroDimensions()
  • BoundingBox's can now by transform'd by a Matrix
  • Added new Entity(components: Component[]) constructor overload to create entities with components quickly.
  • Added Entity.get(type: ComponentType) to get strongly typed components if they exist on the entity.
  • Added Entity.has(type: ComponentType) overload to check if an entity has a component of that type.
  • Added Entity.hasTag(tag: string), Entity.addTag(tag: string), and Entity.removeTag(tag: string, force: boolean).
    • Tag offscreen is now added to entities that are offscreen
  • Added Entity.componentAdded$ and Entity.componentRemoved$ for observing component changes on an entity.
  • For child/parent entities:
    • Added Entity.addChild(entity: Entity), Entity.removeChild(entity: Entity), Entity.removeAllChildren() for managing child entities
    • Added Entity.addTemplate(templateEntity: Entity) for adding template entities or "prefab".
    • Added Entity.parent readonly accessor to the parent (if exists), and Entity.unparent() to unparent an entity.
    • Added Entity.getAncestors() is a sorted list of parents starting with the topmost parent.
    • Added Entity.children readonly accessor to the list of children.
  • Add the ability to press enter to start the game after loaded
  • Add Excalibur Feature Flag implementation for releasing experimental or preview features (#1673)
  • Color now can parse RGB/A string using Color.fromRGBString('rgb(255, 255, 255)') or Color.fromRGBString('rgb(255, 255, 255, 1)')
  • DisplayMode.FitScreen will now scale the game to fit the available space, preserving the aspectRatio. (#1733)
  • SpriteSheet.spacing now accepts a structure { top: number, left: number, margin: number } for custom spacing dimensions (#1788)
  • SpriteSheet.ctor now has an overload that accepts spacing for consistency although the object constructor is recommended (#1788)
  • Add SpriteSheet.getSpacingDimensions() method to retrieve calculated spacing dimensions (#1788)
  • Add KeyEvent.value?: string which is the key value (or "typed" value) that the browser detected. For example, holding Shift and pressing 9 will have a value of ( which is the typed character.
  • Add KeyEvent.originalEvent?: KeyboardEvent which exposes the raw keyboard event handled from the browser.
  • Added a new getter to GraphicsComponent.ts called currentKeys that will return the names of the graphics shown in all layers
  • Added a new getter to GraphicsLayer called currentKeys that will the names of the graphics shown in this layer

Changed

  • Gif now supports new graphics component
  • Algebra.ts refactored into separate files in Math/
  • Engine/Scene refactored to make use of the new ECS world which simplifies their logic
  • TileMap now uses the built in Collider component instead of custom collision code.
  • Updates the Excalibur ECS implementation for ease of use and Excalibur draw system integration
    • Adds "ex." namespace to built in component types like "ex.transform"
    • Adds ex.World to encapsulate all things ECS
    • Adds ex.CanvasDrawSystem to handle all HTML Canvas 2D drawing via ECS
    • Updates ex.Actor to use new ex.TransformComponent and ex.CanvasDrawComponent

Deprecated

  • Timer.unpause() has be deprecated in favor of Timer.resume() (#1864)
  • Removed UIActor Stub in favor of ScreenElement (#1656)
  • ex.SortedList as deprecated
  • ex.Promise is marked deprecated (#994)
  • ex.DisplayMode.Position CSS can accomplish this task better than Excalibur (#1733)

Fixed

  • Fixed allow ex.ColliderComponent to not have a collider
  • Fixed issue where collision events were not being forwarded from individual colliders in a ex.CompositeCollider
  • Fixed issue where ex.CompositeCollider's individual colliders were erroneously generating pairs
  • Fixed issue where GraphicsOptions width/height could not be used to define a ex.Sprite with equivalent sourceView and destSize (#1863)
  • Fixed issue where ex.Scene.onActivate/onDeactivate were called with the wrong arguments (#1850)
  • Fixed issue where no width/height argmunents to engine throws an error
  • Fixed issue where zero dimension image draws on the ExcaliburGraphicsContext throw an error
  • Fixed issue where the first scene onInitialize fires at Engine contructor time and before the "play button" clicked (#1900)
  • Fixed issue where the "play button" click was being interpreted as an input event excalibur needed to handle (#1854)
  • Fixed issue where pointer events were not firing at the ex.Engine.input.pointers level (#1439)
  • Fixed issue where pointer events propagate in an unexpected order, now they go from high z-index to low z-index (#1922)
  • Fixed issue with Raster padding which caused images to grow over time (#1897)
  • Fixed N+1 repeat/repeatForever bug (#1891)
  • Fixed repeat/repeatForever issue with rotateTo (#635)
  • Entity update lifecycle is now called correctly
  • Fixed GraphicsSystem enterviewport and exitviewport event
  • Fixed DOM element leak when restarting games, play button elements piled up in the DOM.
  • Fixed issues with ex.Sprite not rotating/scaling correctly around the anchor (Related to TileMap plugin updates excaliburjs/excalibur-tiled#4, excaliburjs/excalibur-tiled#23, excaliburjs/excalibur-tiled#108)
    • Optionally specify whether to draw around the anchor or not drawAroundAnchor
  • Fixed in the browser "FullScreen" api, coordinates are now correctly mapped from page space to world space (#1734)
  • Fix audio decoding bug introduced in excaliburjs#1707
  • Fixed issue with promise resolve on double resource load (#1434)
  • Fixed Firefox bug where scaled graphics with anti-aliasing turned off are not pixelated (#1676)
  • Fixed z-index regression where actors did not respect z-index (#1678)
  • Fixed Animation flicker bug when switching to an animation (#1636)
  • Fixed ex.Actor.easeTo actions, they now use velocity to move Actors (#1638)
  • Fixed Scene constructor signature to make the Engine argument optional (#1363)
  • Fixed anchor properly of single shape Actor #1535
  • Fixed Safari bug where Sound resources would fail to load (#1848)

0.24.5 - 2020-09-07

Breaking Changes

  • [#1361] Makes use of proxies, Excalibur longer supports IE11 💥 ([#1361]excaliburjs#1361)

Added

  • Adds new ECS Foundations API, which allows excalibur core behavior to be manipulated with ECS style code ([#1361]excaliburjs#1361)
    • Adds new ex.Entity & ex.EntityManager which represent anything that can do something in a Scene and are containers for Components
    • Adds new ex.Component type which allows encapsulation of state on entities
    • Adds new ex.Query & ex.QueryManager which allows queries over entities that match a component list
    • Adds new ex.System type which operates on matching Entities to do some behavior in Excalibur.
    • Adds new ex.Observable a small observable implementation for observing Entity component changes over time

Fixed

  • Fixed Animation flicker bug on the first frame when using animations with scale, anchors, or rotation. (#1636)

0.24.4 - 2020-09-02

Added

  • Add new ex.Screen abstraction to manage viewport size and resolution independently and all other screen related logic. (#1617)
    • New support for the browser fullscreen API
  • Add color blind mode simulation and correction in debug object. (#390)
  • Add LimitCameraBoundsStrategy, which always keeps the camera locked to within the given bounds. (#1498)
  • Add mechanisms to manipulate the Loader screen. (#1417)
    • Logo position Loader.logoPosition
    • Play button position Loader.playButtonPosition
    • Loading bar position Loader.loadingBarPosition
    • Loading bar color Loader.loadingBarColor by default is white, but can be any excalibur ex.Color

Changed

  • Remove usage of mock.engine from the tests. Use real engine instead.
  • Upgrade Excalibur to TypeScript 3.9.2
  • Upgrade Excalibur to Node 12 LTS

Fixed

  • Fixed Loader play button markup and styles are now cleaned up after clicked (#1431)
  • Fixed Excalibur crashing when embedded within a cross-origin IFrame (#1151)
  • Fixed performance issue where uneccessary effect processing was occurring for opacity changes (#1549)
  • Fixed issue when loading images from a base64 strings that would crash the loader (#1543)
  • Fixed issue where actors that were not in scene still received pointer events (#1555)
  • Fixed Scene initialization order when using the lifecycle overrides (#1553)

0.24.0 - 2020-04-23

Breaking Changes

  • Remove obsolete .extend() semantics in Class.ts as as well as related test cases.

Added

  • Added new option for constructing bounding boxes. You can now construct with an options object rather than only individual coordinate parameters. (#1151)
  • Added new interface for specifying the type of the options object passed to the bounding box constructor.
  • Added the ex.vec(x, y) shorthand for creating vectors. (#1340)
  • Added new event processed to Sound that passes processed string | AudioBuffer data. (#1474)
  • Added new property duration to Sound and AudioInstance that exposes the track's duration in seconds when Web Audio API is used. (#1474)

Changed

  • Animation no longer mutate underlying sprites, instead they draw the sprite using the animations parameters. This allows more robust flipping at runtime. (#1258)
  • Changed obsolete decorator to only log the same message 5 times. (#1281)
  • Switched to core-js based polyfills instead of custom written ones (#1214)
  • Updated to TypeScript@3.6.4 and node 10 LTS build
  • Sound.stop() now always rewinds the track, even when the sound is paused. (#1474)

Deprecated

  • ex.Vector.magnitude() will be removed in v0.25.0, use ex.Vector.size(). (#1277)

Fixed

  • Fixed Excalibur crashing when displaying both a tilemap and a zero-size actor (#1418)
  • Fixed animation flipping behavior (#1172)
  • Fixed actors being drawn when their opacity is 0 (#875)
  • Fixed iframe event handling, excalibur will respond to keyboard events from the top window (#1294)
  • Fixed camera to be vector backed so ex.Camera.x = ? and ex.Camera.pos.setTo(...) both work as expected(#1299)
  • Fixed missing on/once/off signatures on ex.Pointer (#1345)
  • Fixed sounds not being stopped when Engine.stop() is called. (#1476)

0.23.0 - 2019-06-08

Breaking Changes

  • ex.Actor.scale, ex.Actor.sx/sy, ex.Actor.actions.scaleTo/scaleBy will not work as expected with new collider implementation, set width and height directly. These features will be completely removed in v0.24.0.

Added

  • New collision group implementation (#1091, #862)
  • New ex.Collider type which is the container for all collision related behavior and state. Actor is now extracted from collision.
  • New interface Clonable<T> to indicate if an object contains a clone method
  • New interface Eventable<T> to indicated if an object can emit and receive events
  • ex.Vector.scale now also works with vector input
  • ex.BoundingBox.fromDimension(width: number, height: number) can generate a bounding box from a width and height
  • ex.BoundingBox.translate(pos: Vector) will create a new bounding box shifted by pos
  • ex.BoundingBox.scale(scale: Vector) will create a new bounding box scaled by scale
  • Added isActor() and isCollider() type guards
  • Added ex.CollisionShape.draw collision shapes can now be drawn, actor's will use these shapes if no other drawing is specified
  • Added a getClosestLineBetween method to CollisionShape's for returning the closest line between 2 shapes (#1071)

Changed

  • Change ex.Actor.within to use surface of object geometry instead of the center to make judgements (#1071)
  • Changed moveBy, rotateBy, and scaleBy to operate relative to the current actor position at a speed, instead of moving to an absolute by a certain time.
  • Changed event handlers in excalibur to expect non-null event objects, before hander: (event?: GameEvent) => void implied that event could be null. This change addresses (#1147) making strict null/function checks compatible with new TypeScript.
  • Changed collision system to remove actor coupling, in addition ex.Collider is a new type that encapsulates all collision behavior. Use ex.Actor.body.collider to interact with collisions in Excalibur (#1119)
    • Add new ex.Collider type that is the housing for all collision related code
      • The source of truth for ex.CollisionType is now on collider, with a convenience getter on actor
      • The collision system now operates on ex.Collider's not ex.Actor's
    • ex.CollisionType has been moved to a separate file outside of Actor
      • CollisionType is switched to a string enum, style guide also updated
    • ex.CollisionPair now operates on a pair of ex.Colliders's instead of ex.Actors's
    • ex.CollisionContact now operates on a pair of ex.Collider's instead of ex.Actors's
    • ex.Body has been modified to house all the physical position/transform information
      • Integration has been moved from actor to Body as a physical concern
      • useBoxCollision has been renamed to useBoxCollider
      • useCircleCollision has been renamed to useCircleCollider
      • usePolygonCollision has been renamed to usePolygonCollider
      • useEdgeCollision has been renamed to useEdgeCollider
    • Renamed ex.CollisionArea to ex.CollisionShape
      • ex.CircleArea has been renamed to ex.Circle
      • ex.PolygonArea has been renamed to ex.ConvexPolygon
      • ex.EdgeArea has been renamed to ex.Edge
    • Renamed getWidth() & setWidth() to property width
      • Actor and BoundingBox are affected
    • Renamed getHeight() & setHeight() to property height
      • Actor and BoundingBox are affected
    • Renamed getCenter() to the property center
      • Actor, BoundingBox, and Cell are affected
    • Renamed getBounds() to the property bounds
      • Actor, Collider, and Shapes are affected
    • Renamed getRelativeBounds() to the property localBounds
      • Actor, Collider, and Shapes are affected
    • Renamed moi() to the property inertia (moment of inertia)
    • Renamed restitution to the property bounciness
    • Moved collisionType to Actor.body.collider.type
    • Moved Actor.integrate to Actor.body.integrate

Deprecated

  • Legacy groups ex.Group will be removed in v0.24.0, use collision groups as a replacement #1091
  • Legacy collision groups off Actor will be removed in v0.24.0, use Actor.body.collider.collisionGroup #1091
  • Removed NaiveCollisionBroadphase as it was no longer used
  • Renamed methods and properties will be available until v0.24.0
  • Deprecated collision attributes on actor, use Actor.body.collider
    • Actor.x & Actor.y will be removed in v0.24.0 use Actor.pos.x & Actor.pos.y
    • Actor.collisionArea will be removed in v0.24.0 use Actor.body.collider.shape
    • Actor.getLeft(), Actor.getRight(), Actor.getTop(), and Actor.getBottom are deprecated
      • Use Actor.body.collider.bounds.(left|right|top|bottom)
    • Actor.getGeometry() and Actor.getRelativeGeometry() are removed, use Collider
    • Collision related properties on Actor moved to Collider, use Actor.body.collider
      • Actor.torque
      • Actor.mass
      • Actor.moi
      • Actor.friction
      • Actor.restitution
    • Collision related methods on Actor moved to Collider, use Actor.body.collider or Actor.body.collider.bounds
      • Actor.getSideFromIntersect(intersect) -> BoundingBox.sideFromIntersection
      • Actor.collidesWithSide(actor) -> Actor.body.collider.bounds.intersectWithSide
      • Actor.collides(actor) -> Actor.body.collider.bounds.intersect

Fixed

  • Fixed issue where leaking window/document handlers was possible when calling ex.Engine.stop() and ex.Engine.start() (#1063)
  • Fixed wrong Camera and Loader scaling on HiDPI screens when option suppressHiDPIScaling is set. (#1120)
  • Fixed polyfill application by exporting a polyfill() function that can be called. (#1132)
  • Fixed Color.lighten() (#1084)

0.22.0 - 2019-04-06

Breaking Changes

  • ex.BaseCamera replaced with Camera (#1087)

Added

  • Added enableCanvasTransparency property that can enable/disable canvas transparency (#1096)

Changed

  • Upgraded Excalibur to TypeScript 3.3.3333 (#1052)
  • Added exceptions on SpriteSheetImpl constructor to check if the source texture dimensions are valid (#1108)

0.21.0 - 2019-02-02

Added

  • Added ability to automatically convert .gif files to SpriteSheet, Animations, and Sprites (#153)
  • New viewport property on camera to return a world space bounding box of the current visible area (#1078)

Changed

  • Updated ex.Color and ex.Vector constants to be static getters that return new instances each time, eliminating a common source of bugs (#1085)
  • Remove optionality of engine in constructor of Scene and _engine private with an underscore prefix (#1067)

Deprecated

  • Rename ex.BaseCamera to Camera, ex.BaseCamera will be removed in v0.22.0 (#1087)

Fixed

  • Fixed issue of early offscreen culling related to zooming in and out (#1078)
  • Fixed issue where setting suppressPlayButton: true blocks load in certain browsers (#1079)
  • Fixed issue where the absence of a pointer button caused an error in the console(#1153)

0.20.0 - 2018-12-10

Breaking Changes

  • ex.PauseAfterLoader removed, use ex.Loader instead (#1031)

Added

  • Added strongly-typed EventTypes enum to Events.ts to avoid magic strings (#1066)

Changed

  • Added parameter on SpriteSheet constructor so you can define how many pixels of space are between sprites (#1058)

0.19.1 - 2018-10-22

Fixed

  • Fixed issue where there were missing files in the dist (Loader.css, Loader.logo.png) (#1057)

0.19.0 - 2018-10-13

Changed

  • Excalibur user documentation has now moved to excaliburjs.com/docs
  • Excalibur will now prompt for user input before starting the game to be inline with the new webaudio requirements from chrome/mobile browsers (#1031)

Deprecated

  • PauseAfterLoader for iOS in favor of new click-to-play functionality built into the default Loader (#1031)

Fixed

  • Fixed issue where Edge web audio playback was breaking (#1047)
  • Fixed issue where pointer events do not work in mobile (#1044)
  • Fixed issue where iOS was not loading by including the right polyfills (#1043)
  • Fixed issue where sprites do not work in Firefox (#980)
  • Fixed issue where collision pairs could sometimes be incorrect (#975)
  • Fixed box collision velocity resolution so that objects resting on a surface do not accumulate velocity (#986)

0.18.0 - 2018-08-04

Breaking Changes

  • Sound.setVolume() replaced with Sound.volume
  • Sound.setLoop() replaced with Sound.loop

Added

  • Add Scene.isActorInDrawTree method to determine if an actor is in the scene's draw tree.

Fixed

  • Fixed missing exitviewport/enterviewport events on Actors.on/once/off signatures (#978)
  • Fix issue where Actors would not be properly added to a scene if they were removed from that scene during the same frame (#979)

0.17.0 - 2018-06-04

Breaking Changes

  • Property scope Pointer.actorsUnderPointer changed to private
  • Sprite.sx replaced with Sprite.x
  • Sprite.sy replaced with Sprite.y
  • Sprite.swidth replaced with Sprite.width
  • Sprite.sheight replaced with Sprite.height

Added

  • Allow timers to limit repeats to a finite number of times (#957)
  • Convenience method on Scene to determine whether it is the current scene. Scene.isCurrentScene() (#982)
  • New PointerEvent.stopPropagation() method added. Works the same way as (https://developer.mozilla.org/en-US/docs/Web/API/Event/stopPropagation) (#912)
  • New Actor.getAncestors() method, which retrieves full array of current Actor ancestors
  • Static Actor.defaults prop, which implements IActorDefaults.
  • Native sound events now exposed
    • volumechange - on playing sound volume change;
    • pause - on playback pause;
    • stop - on playback stop;
    • emptied - on data cleanup(f.e. when setting new data);
    • resume - on playback resume;
    • playbackstart - on playback start;
    • playbackend - on playback end;
  • Added Sound.instances getter, which returns active tracks. Playing or paused
  • Added Sound.getTrackId(track: [[AudioInstance]]) method. Which returns id of track provided, if it is in list of active tracks.

Changed

  • Refactored Easing functions to be reversable (#944)
  • Now at creation every Actor.anchor prop is set to default Actor.defaults.anchor.
  • Scene.remove(Actor) now starts the Actor.Kill event cycle (#981)

Deprecated

  • CapturePointer.update() method now doesn't propagate event to actor, just verifies pointer events for actor.
  • Added Sound.volume & Sound.loop properties as a replacement for Sound.setVolume() and Sound.setLoop(). The methods setVolume and setLoop have been marked obsolete.

Fixed

  • Added missing variable assignments to TileMapImpl constructor (#957)
  • Correct setting audio volume level from value to setValueAtTime to comply with deprecation warning in Chrome 59 (#953)
  • Force HiDPI scaling to always be at least 1 to prevent visual artifacts in some browsers
  • Recalculate physics geometry when width/height change on Actor (#948)
  • Fix camera move chaining (#944)
  • Fix pickSet(allowDuplicates: true) now returns the proper length array with correct elements (#977)
  • Index export order to prevent almond.js from creation of corrupted modules loading order.
  • Sound.pause() now saves correct timings.
  • Fix ex.Vector.isValid edgecase at Infinity (#1006)

0.16.0 - 2018-03-31

Added

  • New typesafe and override safe event lifecycle overriding, all onEventName handlers will no longer be dangerous to override (#582)
    • New lifecycle event onPreKill and onPostKill
  • SpriteSheets can now produce animations from custom sprite coordinates SpriteSheet.getAnimationByCoords(engine, coords[], speed) (#918)
  • Added drag and drop support for Actors (#134)
    • New Event enter
    • New Event leave
    • New Event pointerenter
    • New Event pointerleave
    • New Event pointerdragstart
    • New Event pointerdragend
    • New Event pointerdragmove
    • New Event pointerdragenter
    • New Event pointerdragleave
    • New Class PointerDragEvent which extends PointerEvent
    • New Class GlobalCoordinates that contains Vectors for the world, the page, and the screen.
    • Added property ICapturePointerConfig.captureDragEvents which controls whether to emit drag events to the actor
    • Added property PointerEvent.pointer which equals the original pointer object

Deprecated

  • Sprite.sx, Sprite.sy, Sprite.swidth, Sprite.sheight have been deprecated in favor of Sprite.x, Sprite.y, Sprite.width, Sprite.height (#918)

Fixed

  • Added missing lifecycle event handlers on Actors, Triggers, Scenes, Engine, and Camera (#582)
  • Tile Maps now correctly render negative x-axis coordinates (#904)
  • Offscreen culling in HiDPI mode (#949)
    • Correct bounds check to check drawWidth/drawHeight for HiDPI
    • suppressHiDPIScaling now also suppresses pixel ratio based scaling
  • Extract and separate Sprite width/height from drawWidth/drawHeight to prevent context corruption (#951)

0.15.0 - 2018-02-16

Breaking Changes

  • LockedCamera replaced with BaseCamera.strategy.lockToActor
  • SideCamera replaced with BaseCamera.strategy.lockToActorAxis
  • Body.wasTouching replaced with event type CollisionEnd

Added

  • Option bag constructors have been added for commonly-used classes (see Constructors.md) (#410)

0.14.0 - 2017-12-02

Breaking Changes

  • Triggers now have a new option bag constructor using the ITriggerOptions interface. (#863).
  • update event replaced with postupdate event
  • CollisionEvent replaced by PreCollisionEvent
  • getDrawWidth() and getDrawHeight() replaced with the getters drawWidth and drawHeight
  • PointerEvent.x and PointerEvent.y replaced with PointerEvent.pos

Added

  • Automatic HiDPI screen detection and scaling in excalibur internals to correct blurry bitmap rendering on HiDPI screens. This feature can optionally be suppressed with IEngineOptions.suppressHiDPIScaling.
  • Added new line utility Line.normal() and Line.distanceToPoint (#703)
  • Added new PolygonArea utility PolygonArea.getClosestFace(point) (#703)
  • Triggers now fire an EnterTriggerEvent when an actor enters the trigger, and an ExitTriggerEvent when an actor exits the trigger. (#863)
  • Actors have a new events CollisionStart which when 2 actors first start colliding and CollisionEnd when 2 actors are no longer colliding. (#863)
  • New camera strategies implementation for following targets in a scene. Allows for custom strategies to be implemented on top of some prebuilt
    • LockCameraToActorStrategy which behaves like LockedCamera and can be switched on with Camera.strategy.lockToActor(actor).
    • LockCameraToActorAxisStrategy which behaves like SideCamera and can be switched on with Camera.strategy.lockToActorAxis(actor, ex.Axis.X)
    • ElasticToActorStrategy which is a new strategy that elastically moves the camera to an actor and can be switched on with Camera.strategy.elasticToActor(actor, cameraElasticity, cameraFriction)
    • CircleAroundActorStrategy which is a new strategy that will follow an actor when a certain radius from the camera focus and can be switched on with Camera.strategy.circleAroundActor(actor)

Changed

  • Trigger has been rebuilt to provide a better experience
    • The trigger action only fires when an actor enters the designated area instead of every frame of collision. (#863)
    • Triggers can now draw like other Actors, but are still not visible by default (#863)

Deprecated

  • Body.wasTouching has been deprecated in favor of a new event type CollisionEnd (#863)
  • SideCamera and LockedCamera are deprecated in favor of camera strategies

Fixed

  • Fixed odd jumping behavior when polygons collided with the end of an edge (#703)

0.13.0 - 2017-10-07

Breaking Changes

  • Scene.children replaced with Scene.actors

Added

  • Convenience getters implemented halfDrawWidth, halfDrawHeight, halfCanvasWidth, halfCanvasHeight, canvasWidth, and canvasHeight.
  • New pause/unpause feature for timers to help with more robust pausing (#885)
  • New event listening feature to listen to events only .once(...) then unsubscribe automatically (#745)
  • New collision event postcollision to indicate if collision resolution occured (#880)

Deprecated

  • PointerEvent.x and PointerEvent.y, in favor of PointerEvent.pos (#612)
  • CollisionEvent has been deprecated in favor of the more clear PreCollisionEvent (#880)
  • getDrawWidth() and getDrawHeight() have been marked obsolete and changed into the getters drawWidth and drawHeight respectively in order to progressively make getters/setters consistent (#861)

Fixed

  • Fixed same instance of color potentially being shared, and thus mutated, between instance actors (#840)
  • Fixed bug where active and passive type collisions would resolve when they shouldn't in rigid body physics mode (#880)

0.12.0 2017-08-12

Breaking Changes

  • CollisionType.Elastic has been removed
  • Promises.wrap has been replaced with Promise.resolve

Added

  • Added new hsl and hex format options in Color.toString(format). rgb is the default to maintain backwards compatibility (#852)

Changed

  • Animation.loop property now to set to true by default (#583)
  • Added backgroundColor to engine options as part of Engine constructor (#846)

Deprecated

  • ex.Scene.children is now ex.Scene.actors (#796)

0.11.0 2017-06-10

Breaking Changes

  • Renamed Utils.removeItemToArray() to Utils.removeItemFromArray() (#798)

Added

  • Added optional volume argument to Sound.play(volume?: number), which will play the Audio file at anywhere from mute (volume is 0.0) to full volume (volume is 1.0). (#801)
  • Added another DisplayMode option: DisplayMode.Position. When this is selected as the displayMode type, the user must specify a new position option (#781)
  • Added a static method distance to the Vector class (#517)
  • Added WheelEvent event type for the wheel browser event, Excalibur now supports scroll wheel (#808)

Changed

  • Camera zoom over time now returns a promise that resolves on completion (#800)
  • Edge builds have more descriptive versions now containing build number and Git commit hash (e.g. 0.10.0-alpha.105#commit) (#777)

Fixed

  • Fixed camera zoom over time, before it did not work at all (#800)
  • Fixed semi-colon key not being detected on Firefox and Opera. (#789)

0.10.0 2017-04-07

Breaking Changes

  • Rename Engine.width and Engine.height to be Engine.canvasWidth and Engine.canvasHeight (#591)
  • Rename Engine.getWidth and Engine.getHeight to be Engine.getDrawWidth and Engine.getDrawHeight (#591)
  • Changed GameEvent to be a generic type for TypeScript, allowing strongly typing the target property. (#724)
  • Removed Body.useEdgeCollision() parameter center (#724)

Added

  • Added Engine.isPaused to retrieve the running status of Engine (#750)
  • Added Engine.getWorldBounds to provide a quick way to get the top left corner and bottom right corner of the screen (#729)
  • Added predraw and postdraw events to Engine class. These events happen when prior to and after a draw (#744)
  • Added Perlin noise generation helper ex.PerlinGenerator for 1d, 2d, and 3d noise, along with drawing utilities (#491)
  • Added font styles support for normal, italic, and oblique in addition to bold text support (#563)

Changed

  • Update project to use TypeScript 2.2.2 (#762)
  • Changed Util.extend to include Object.assign functionality (#763)

Fixed

  • Update the order of the affine transformations to fix bug when scaling and rotating Actors (#770)

0.9.0 2017-02-09

Added

  • Added preupdate, postupdate, predraw, postdraw events to TileMap
  • Added ex.Random with seed support via Mersenne Twister algorithm (#538)
  • Added extended feature detection and reporting to ex.Detector (#707)
    • ex.Detector.getBrowserFeatures() to retrieve the support matrix of the current browser
    • ex.Detector.logBrowserFeatures() to log the support matrix to the console (runs at startup when in Debug mode)
  • Added @obsolete decorator to help give greater visibility to deprecated methods (#684)
  • Added better support for module loaders and TypeScript importing. See Installation docs for more info. (#606)
  • Added new Excalibur example project templates (#706, #733):
  • Added Pointer.lastPagePos, Pointer.lastScreenPos and Pointer.lastWorldPos that store the last pointer move coordinates (#509)

Changed

  • Changed Util.clamp to use math libraries (#536)
  • Upgraded to TypeScript 2.1.4 (#726)

Fixed

  • Fixed Scene/Actor activation and initialization order, actors were not being initialized before scene activation causing bugs (#661)
  • Fixed bug where the engine would not load if a loader was provided without any resources (#565)
  • Fixed bug where an Actor/UIActor/TileMap added during a Timer callback would not initialize before running draw loop. (#584)
  • Fixed bug where on slower systems a Sprite may not be drawn on the first draw frame (#748)

0.8.0 2016-12-04

Added

  • ex.Vector.magnitude alias that calls ex.Vector.distance() to get magnitude of Vector (#663)
  • Added new ex.Line utilities (#662):
    • ex.Line.slope for the raw slope (m) value
    • ex.Line.intercept for the Y intercept (b) value
    • ex.Line.findPoint(x?, y?) to find a point given an X or a Y value
    • ex.Line.hasPoint(x, y, threshold) to determine if given point lies on the line
  • Added Vector.One and Vector.Half constants (#649)
  • Added Vector.isValid to check for null, undefined, Infinity, or NaN vectors method as part of (#665)
  • Added ex.Promise.resolve and ex.Promise.reject static methods (#501)
  • PhantomJS based testing infrastructure to accurately test browser features such as image diffs on canvas drawing (#521)
  • Added some basic debug stat collection to Excalibur (#97):
    • Added ex.Engine.stats to hold frame statistic information
    • Added ex.Engine.debug to hold debug flags and current frame stats
    • Added preframe and postframe events to Engine as hooks
    • Added ex.Physics statistics to the Excalibur statistics collection
  • Added new fast body collision detection to Excalibur to prevent fast moving objects from tunneling through other objects (#665)
    • Added DynamicTree raycast to query the scene for bounds that intersect a ray
    • Added fast BoundingBox raycast test

Changed

  • Internal physics names refactored to be more readable and to use names more in line with game engine terminology (explicit broadphase and narrowphase called out)

Deprecated

  • ex.Promise.wrap (#501)

Fixed

  • Fix Actor.oldPos and Actor.oldVel values on update (#666)
  • Fix Label.getTextWidth returns incorrect result (#679)
  • Fix semi-transparent PNGs appear garbled (#687)
  • Fix incorrect code coverage metrics, previously our test process was reporting higher than actual code coverage (#521)
  • Fix Actor.getBounds() and Actor.getRelativeBounds() to return accurate bounding boxes based on the scale and rotation of actors. (#692)

0.7.1 - 2016-10-03

Breaking Changes

  • Refactored and modified Sound API (#644)
    • Sound.setData now returns a Promise which differs from previous API
    • Removed internal FallbackAudio and Sound classes and replaced with single Sound class
    • Added AudioTagInstance and WebAudioInstance internal classes

Added

  • ex.Promise.join(Promise[]) support (in addition to ...promises support) (#642)
  • Moved build artifacts to separate excalibur-dist repository (#648)
  • ex.Events namespace and typed event handler .on(...) overloads for default events on core excalibur objects (#639)
  • Engine.timescale property (default: 1.0) to add time-scaling to the engine for time-based movements (#543)
  • Two new parameters to ex.Util.DrawUtil.line that accept a line thickness and end-cap style (#658)

Fixed

  • Actor.actions.fade properly supporting fading between 0 and 1 and vice versa (#640)
  • Fix issues with audio offset tracking and muting while game is invisible (#644)
  • Actor.getHeight() and Actor.getWidth() now take into account parent scaling (#645)
  • Actor.debugDraw now works properly for child actors (#505, #645)
  • Sprite culling was double scaling calculations (#646)
  • Fix negative zoom sprite culling (#539)
  • Fix Actor updates happening more than once per frame, causing multiple pointer events to trigger (#643)
  • Fix Actor.on('pointerup') capturePointer events opt-in on event handler. The opt-in was triggering correctly for handlers on 'pointerdown' and 'pointermove', but not 'pointerup'.

0.7.0 - 2016-08-29

Breaking Changes

  • Code marked 'Obsolete' has been removed (#625, #603)
    • Actor
      • addEventListener
      • getWorldX, getWorldY
      • clearActions, easeTo, moveTo, moveBy, rotateTo, rotateBy, scaleTo, scaleBy, blink, fade, delay, die, callMethod, asPromise, repeat, repeatForever, follow, meet
    • Class
      • addEventListener, removeEventListener
    • Engine
      • parameterized constructor
      • addChild, removeChild
    • UpdateEvent removed
  • Scene.addChild and Scene.removeChild are now protected
  • Removed ex.Template and ex.Binding (#627)

Added

  • New physics system, physical properties for Actors (#557, #472)
  • Read The Docs support for documentation (#558)
  • Continuous integration builds unstable packages and publishes them (#567)
  • Sound and Texture resources can now process data (#574)
  • Actors now throw an event when they are killed (#585)
  • "Tap to Play" button for iOS to fulfill platform audio requirements (#262)
  • Generic lerp/easing functions (#320)
  • Whitespace checking for conditional statements (#634)
  • Initial support for Yeoman generator (#578)

Changed

  • Upgraded Jasmine testing framework to version 2.4 (#126)
  • Updated TypeScript to 1.8 (#596)
  • Improved contributing document (#560)
  • Improved local and global coordinate tracking for Actors (#60)
  • Updated loader image to match new logo and theme (#615)
  • Ignored additional files for Bower publishing (#614)

Fixed

  • Actions on the action context threw an error (#564)
  • Actor getLeft(), getTop(), getBottom() and getRight() did not respect anchors (#568)
  • Actor.actions.rotateTo and rotateBy were missing RotationType (#575)
  • Actors didn't behave correctly when killed and re-added to game (#586)
  • Default fontFamily for Label didn't work with custom FontSize or FontUnit (#471)
  • Fixed issues with testing sandbox (#609)
  • Issue with camera lerp (#555)
  • Issue setting initial opacity on Actors (#511)
  • Children were not being updated by their parent Actors (#616)
  • Center-anchored Actors were not drawn at the correct canvas coordinates when scaled (#618)

0.6.0 - 2016-01-19

Added

  • GamePads now have a connection event (#473)
  • Unit circle drawing for debug mode (#467)
  • Engine now fails gracefully in unsupported browsers (#386)
  • Global fatal error catching (#381)
  • MockEngine for testing (#360)
  • Code coverage reports via Coveralls (#169)
  • SpriteFonts now support different target colors (#148)
  • Cameras now have position, velocity, and acceleration properties (#490)

Changed

  • Actor.addChild() changed to Actor.add() (#519)
  • Actor.removeChild() changed to Actor.remove() (#519)
  • Documentation is only deployed on changes to the main git branch (#483)
  • A warning message is now displayed if no supported audio format is provided for a browser (#476)
  • Updated TSLint directory scanning (#442, #443, #447)
  • Deprecated older methods (#399)
  • Changed API for Key events (#502)

Fixed

  • Actors now properly collide with TileMaps (#541)
  • Gamepad detection is fixed (#460, #518)
  • Actor scale now correctly occurs after translation (#514)
  • Actors now respect the visible property of their children (#513)
  • Fixed centered sprite drawing on Actors (#507)
  • Animation freezeframe is now properly set to last Animation frame by default (#506)
  • It is no longer possible to add the same Actor to a scene multiple times (#504)
  • Text alignment on SpriteFonts with Labels is fixed (#484)
  • Engine pointer events properly fire when a camera is zoomed (#480)
  • Fixed a small bug in rotateTo (#469)
  • Setting Label colors now works (#468)
  • Labels now respect set font (#372)
  • UIActor now respects visibility (#368)
  • Solid color Actors now respect opacity (#364)
  • TileMap culling uses proper width and height values (#293)
  • Font API changed while fixing font size issue

0.5.1 - 2015-06-26

Added

  • Actors can now recursively check the containment of their children (#453)
  • RotateTo and RotateBy now support ShortestPath, LongestPath, Clockwise, and Counterclockwise rotation (#461)

Fixed

  • Actor.contains() did not work for child actors (#147)
  • Unexpected placement occasionally occurred for Actors with certain collision types (#319)
  • Velocity wasn’t updating properly when fixed and active Actors collided (#454)
  • Actors removed with actor.kill() were not being removed from the draw tree (#458)
  • RotateTo and RotateBy weren’t using the shortest angle by default (#282)
  • Sprite width and height didn’t take scaling into account (#437)
  • Fixed error message when calling Actor.setDrawing() on a non-existent key (#456)

0.5.0 - 2015-06-03

Added

  • resource cache busting (#280)
  • HTML5 Gamepad API support (#15)
  • Browserify support (#312)
  • ‘blur’ and ‘visible’ events to detect when the browser window a game is in has focus (#385)
  • Z-index support for Actors, allowing for specific ordered drawing (#356)
  • unlocked drawing for UI elements (#354)
  • Promise.join() to return a new promise when promises passed to it have been resolved (#341, #340)
  • ability to skip a frame in an animation (#313)
  • You can now remove effects from IDrawable objects (#303)
  • generic Resource type to allow for XHR loading (#297)
  • gray Color constants (#209)

Changed

  • Renamed engine.addChild() to engine.add() (#288)
  • Renamed setSpriteTransformationPoint() to setAnchor() (#269)
  • Renamed TopCamera to LockedCamera (#184)
  • Renamed Actor.pipeline to Actor.traits (#351)
  • Actor anchoring now uses center origin by default (#299)
  • Actor updates (movement, collision, etc.) now use a pipeline (#330)
  • Organized classes, files, and project structure (#182, #347)
  • Improvements to collision detection (#345, #332)
  • Loop optimizations for performance improvements (#296)
  • Updated to TypeScript 1.4 (#393)
  • Improved pointer event handling so touch and mouse events can be captured together (#334)
  • Improved Point and Vector methods and rotation (#323, #302)
  • Color is now treated as a vector to allow for changes (#298)
  • Cleaned up event type consistency (#273)
  • There is now a default instance of a Camera (#270)
  • TSLint now used to enforce code quality

Fixed

  • A Sprite’s dimensions weren’t validated against the size of its texture (#318)
  • Improved sprite drawing performance issues (#316)
  • Actors were sometimes throwing duplicate collision events (#284)
  • Actors were not setting their initial opacity correctly (#307)
  • Particle emitters couldn’t emit less than 60 particles per second (#301)
  • Fixed issue with TileMap collisions (#286)
  • Animations with duplicate frames weren’t being created correctly (#283)
  • Separated drawing and collision logic for CollisionMaps (now TileMap) (#285)
  • Errors in promises were being swallowed if no error callback was supplied (#337)
  • A null promise was being returned if no loader was given to Engine.start() (#335)
  • Changed default collisionType to ‘PreventCollision’ (#324)
  • Color didn’t handle alpha = 0 correctly (#257)
  • Blink action usage was confusing (#279)
  • Couldn’t use the width and height properties of a Texture after it loaded (#355)
  • Using on(‘pointerdown’) would not automatically enable pointer capturing (#398)
  • Unsubscribing from an event sometimes removed other event handlers (#366)
  • Actor.setCenterDrawing() was hard-coded to true (#375)
  • Console was undefined in IE9. (#378)
  • Pointers were not handling mobile Safari touch events (#382)
  • Fixed debug mode drawing (#274)
  • Flipping a sprite didn’t factor in scaling (#401)
  • Sound continued to play when the game was paused (#383)
  • UIActor.kill() didn’t remove the actor (#373)
  • Passing an empty array to ex.Promise.join resulted in unresolved promises (#365)
  • MouseUp / TouchEnd events weren’t capture correctly if outside of canvas (#374)
  • Clearing actions from an empty action queue caused problems (#409)
  • Scene.onActivate() was being called before Scene.onInitialize() (#418)
  • New z-indexing wasn’t cleaning up after itself (#433)
  • Fixed issue with world / screen coordinates in UIActors (#371)
  • Fade action didn’t work for text (#261)
  • Fade action didn’t work for plain-color actors (#256)
  • Collision events weren’t being published for both collision participants (#254)
  • The loading bar was misrepresenting the time taken to decode audio files (#106)
  • actor.getCenter() wasn’t returning the correct value (#438)
  • Cameras were on the engine instead of the scene, resulting in scene transition problems (#277)
  • Actors with sprites larger than the actor would disappear prematurely from the screen (#287)
  • Derived classes can now use offscreen culling (#294)
  • Fixed issue with TileMap culling (#444)

0.2.2 - 2014-04-15

Fixed

  • Removed extra declarations file from package that was causing visual studio build problems

0.2.0 - 2014-04-09

Added

  • Visual Studio 2013 template support (#139)
  • Collision Map for building large static collidable levels (#33)
  • Redundant fallback sound sources for cross browser support (#125)
  • Particle Emitter implementation (#52)
  • Trigger implementation (#91)
  • Timer implementation (#76)
  • Camera Effects: zoom, shake (#55)
  • Polygon IDrawable (#93)
  • Alias 'on' and 'off' for 'addEventListener' and 'removeEventListener' (#229)
  • Optimized draw so only on screen elements are drawn (#239)
  • Support Scale in the x and y directions for actors (#118)
  • Added notion of collision grouping (#100)
  • New Events like 'enterviewport', 'exitviewport', and 'initialize' (#215, #224)
  • Textures allow direct pixel manipulation (#155)
  • Static Logger improvements with '.debug()', '.info()', '.warn()' and '.error()' (#81)
  • Added callMethod() action to actor (#244)
  • Added fade() action to actor (#104)
  • Added follow() and meet() action to actor (#77)

Changed

  • 'engine.goToScene()' replaces push and pop (#168)
  • More intuitive starting workflow (#149)
  • Collisions are now more concrete on actors with CollisionType (#241)
  • Namespace all types with 'ex' to prevent Excalibur from polluting the global (#87)
  • Refactor SceneNode to Scene (#135)
  • Refactor keys (#115)
  • Build system with Grunt (#92)

Fixed

  • Collision event was firing after other actor has been killed (#228)
  • Additional actor was killed when actor.kill() is called (#226)
  • Fixed loading bar (#195)
  • ex.Color.Yellow constant was wrong (#122)
  • removeEventListener did not exist off of engine (#175)
  • Excalibur promises should not swallow exceptions in promise callbacks (#176)
  • Actor.extend did not work on actor subclasses (#103)

0.1.1 - 2013-12-19

Changed

  • Refactored Keys to be less confusing (#115)
  • Refactored ActorEvent to be less confusing (#113)

Fixed

  • 'update' event on the Engine now fires correctly (#105)
  • Actor.extend works on subclasses now (#103)

0.1.0 - 2013-12-11

Added

  • Actor based paradigm for managing game objects
  • Built-in scripting for actors, allowing objects to move, rotate, blink, scale, and repeat actions
  • Entity-entity collision detection
  • Event support to react to events happening in the game
  • Camera abstraction to easily think about the view port
  • Multiple display modes including fixed size, full screen, and dynamic container
  • Scene stack support to create multiple game levels
  • Sprite sheet and animation support
  • Simple sound library for game audio, supporting the Web Audio API and the HTML Audio API
  • Promise implementation for managing asynchronous behavior
  • Resource loading with optional custom progress bars