Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat: Add a method to adapt the camera bounds to the world #2769

Merged
merged 29 commits into from
Nov 8, 2023
Merged

feat: Add a method to adapt the camera bounds to the world #2769

merged 29 commits into from
Nov 8, 2023

Conversation

Skyost
Copy link
Contributor

@Skyost Skyost commented Sep 24, 2023

Description

This PR adds a method to the CameraComponent class that allows it to set its bounds according to the current world. In fact, this was the behavior of the old camera object.

Checklist

  • I have followed the Contributor Guide when preparing my PR.
  • I have updated/added tests for ALL new/updated/fixed functionality.
  • I have updated/added relevant documentation in docs and added dartdoc comments with ///.
  • I have updated/added relevant examples in examples or docs.

Breaking Change?

  • Yes, this PR is a breaking change.
  • No, this PR is not a breaking change.

Related Issues

Closes #2601.

@Skyost Skyost changed the title feat: Add a method to adapt the camera bounds to the world. feat: Add a method to adapt the camera bounds to the world Sep 24, 2023
@Skyost
Copy link
Contributor Author

Skyost commented Sep 25, 2023

@spydon I see, but calculating the worldSize may be a bit more difficult if the shape is not a Rectangle, isn't it ? Also, does strictBounds sounds good to you ?

@spydon
Copy link
Member

spydon commented Oct 2, 2023

@spydon I see, but calculating the worldSize may be a bit more difficult if the shape is not a Rectangle, isn't it ? Also, does strictBounds sounds good to you ?

Maybe:

/// When [considerViewport] is true none of the viewport can go outside
/// of the bounds, when it is false only the viewfinder anchor is considered.
void setBounds(Shape? bounds. {bool considerViewport = false}}) {

@Skyost
Copy link
Contributor Author

Skyost commented Oct 3, 2023

Okay, I've made the changes you've suggested :

if (considerViewport && bounds != null) {
  final halfViewportSize = viewport.size / 2;
  bounds = Rectangle.fromCenter(
    center: bounds.center,
    size: Vector2(bounds.support(Vector2(1, 0)), bounds.support(Vector2(0, 1))) - halfViewportSize,
  );
}

One drawback is that the bounds are converted to a Rectangle. I don't know if it's possible to just "adapt" the given shape to the viewport as Shape is an abstract class.

@spydon
Copy link
Member

spydon commented Oct 3, 2023

One drawback is that the bounds are converted to a Rectangle. I don't know if it's possible to just "adapt" the given shape to the viewport as Shape is an abstract class.

Much better! Except that it doesn't compile. 😅
I think we should check at least support circular viewports and circular bounds in the future too, but maybe that could be a follow-up.

@Skyost
Copy link
Contributor Author

Skyost commented Oct 3, 2023

Yep, it still wip. More commits to follow 🙂

This method allows to support various other viewport shapes.
@Skyost
Copy link
Contributor Author

Skyost commented Oct 3, 2023

Alright, so I've added the following method :

Shape? _calculateViewportAwareBounds(Shape originalBounds) {
  final worldSize = Vector2(
    originalBounds.support(Vector2(1, 0)).x,
    originalBounds.support(Vector2(0, 1)).y,
  );
  final halfViewportSize = viewport.size / 2;
  if (originalBounds is Rectangle) {
    return Rectangle.fromCenter(
      center: originalBounds.center,
      size: worldSize - halfViewportSize,
    );
  } else if (originalBounds is RoundedRectangle) {
    final halfSize = (worldSize - halfViewportSize) / 2;
    return RoundedRectangle.fromPoints(
      originalBounds.center - halfSize,
      originalBounds.center + halfSize,
      originalBounds.radius,
    );
  } else if (originalBounds is Circle) {
    return Circle(
      originalBounds.center,
      worldSize.x - max(halfViewportSize.x, halfViewportSize.y),
    );
  }
  return null;
}

Which is being used in setBounds :

if (considerViewport && bounds != null) {
  bounds = _calculateViewportAwareBounds(bounds) ?? bounds;
}

Tell me what you think about it 🙂

@spydon
Copy link
Member

spydon commented Oct 4, 2023

Tell me what you think about it 🙂

Looks good! I think we need to take the zoom into consideration too though, because this doesn't work when the zoom level is changed right?

@Skyost
Copy link
Contributor Author

Skyost commented Oct 11, 2023

Let's do that !

@Skyost
Copy link
Contributor Author

Skyost commented Oct 11, 2023

@spydon Just checked the source code and it seems that BoundedPositionBehavior doesn't take any viewfinder.transformation into account. Maybe this belongs to a separate PR ?

@spydon
Copy link
Member

spydon commented Oct 11, 2023

@spydon Just checked the source code and it seems that BoundedPositionBehavior doesn't take any viewfinder.transformation into account. Maybe this belongs to a separate PR ?

The BoundedPositionBehavior doesn't need to take any viewfinder transformations into account, since it only checks on a position and a bounds. This PR on the other hand that adds bounds aware of the viewport needs to take into consideration when the CameraComponent.visibleWorldRect changes (i.e. when the viewport size changes after a resize or the zoom level of the viewfinder makes a smaller/bigger part of the world visible), otherwise if you for example first play the game in landscape mode and then flip it to portrait the bounds won't be correctly calculated.

@Skyost
Copy link
Contributor Author

Skyost commented Oct 12, 2023

Alright. Maybe the best approach is to add a new behavior to the viewfinder so that it can update the bounds based on the visible rectangle.

That's what I did in the latest commit. I haven't implemented the "update" part, but can you tell me what you think about it please ?

@spydon
Copy link
Member

spydon commented Oct 18, 2023

Alright. Maybe the best approach is to add a new behavior to the viewfinder so that it can update the bounds based on the visible rectangle.

That's what I did in the latest commit. I haven't implemented the "update" part, but can you tell me what you think about it please ?

The idea is good, but the bounds should not have to be recalculated on every tick, but only when the transforms that are affecting it are changing. You can add listeners to those transforms and run the update method when they trigger.

@Skyost
Copy link
Contributor Author

Skyost commented Oct 18, 2023

The idea is good, but the bounds should not have to be recalculated on every tick, but only when the transforms that are affecting it are changing. You can add listeners to those transforms and run the update method when they trigger.

Yes, I was thinking about that. The thing is, from what I know, these transforms don't have any method that allow to register a listener. I'll see what's the best approach.

@spydon
Copy link
Member

spydon commented Oct 18, 2023

Yes, I was thinking about that. The thing is, from what I know, these transforms don't have any method that allow to register a listener. I'll see what's the best approach.

The FixedResolutionViewport has it (called transform), we could just put that field directly on the Viewport and override it in the subclasses. And the viewfinder already has it.

@Skyost
Copy link
Contributor Author

Skyost commented Oct 18, 2023

Yes, what I mean is I didn't know that the Transform2D class had a addListener method or something like that. But I just seen that it overrides ChangeNotifier.

@Skyost
Copy link
Contributor Author

Skyost commented Oct 19, 2023

Okay, the problem with listening for Transformation2D updates, it that when there is a FollowBehavior, the following line,

owner.position = delta..add(owner.position);

triggers an update at every tick.

@Skyost Skyost requested a review from spydon October 19, 2023 09:40
@spydon
Copy link
Member

spydon commented Oct 21, 2023

Okay, the problem with listening for Transformation2D updates, it that when there is a FollowBehavior, the following line,

owner.position = delta..add(owner.position);

triggers an update at every tick.

scale is exposed as a NotifyingVector2 inside of Transform2D so that you can listen to only scale changes, are there more things from the transform that you need?

@Skyost
Copy link
Contributor Author

Skyost commented Oct 23, 2023

Oh yes, sure.

Copy link
Member

@spydon spydon left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Starting to look good!
Just some tests missing, the easiest would be to do golden tests for this I think, you can get some inspiration from here for example:
https://github.com/flame-engine/flame/blob/main/packages/flame/test/camera/viewports/fixed_aspect_ratio_viewport_test.dart

@spydon
Copy link
Member

spydon commented Nov 3, 2023

Any updates on this @Skyost? :)

@Skyost
Copy link
Contributor Author

Skyost commented Nov 6, 2023

Hey @spydon,

Sorry, I've never written any golden test in my life and have been unactive for the past two weeks. I'll try to make some progress today 🙂

@Skyost
Copy link
Contributor Author

Skyost commented Nov 6, 2023

I've added a test that checks whether the viewport has been correctly updated. It's not a golden test, but I think it does the trick. Do you think it needs more tests ? If so, what parts of the ViewportAwareBoundsBehavior class need to be tested ?

Copy link
Member

@spydon spydon left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Lgtm, the test doesn't really ensure things are rendered correctly, but since the other camera behaviour doesn't have a goldens test either I think we can let this pass for now.

@spydon spydon merged commit 87b69df into flame-engine:main Nov 8, 2023
8 checks passed
spydon pushed a commit that referenced this pull request Sep 16, 2024
# Description
The behavior for `CameraComponent.setBounds` was not behaving correctly.
The viewport-aware behavior was not triggering until a window resize
event. All supported bound variants were not respecting the fixed
resolution viewport, or in some cases, behaving unpredictably.

The issue was with the fact the viewport-aware behavior component
depends on the bounded position component to be in its parent, but was
returning `null` during `onMount`. The viewport math was using the
logical size instead of the virtual size. I made some math changes to be
accurate.

This PR adds new a getter `CameraComponent.considerViewport`.
`ViewportAwareBoundsBehavior` is now added as a side effect of mounting
`BoundedPositionBehavior` by waiting for the `mounted` future. This
guarantees that
Flames life cycle is respected and the components behave as expected. 

Tests pass on my own local project. 

Videos are linked on the discord thread
[here](https://discord.com/channels/509714518008528896/1275814019235709003/1275888732481785925).

Melos dry run completed successfully.

## Checklist
- [x] I have followed the [Contributor Guide] when preparing my PR.
- [x] I have updated/added tests for ALL new/updated/fixed
functionality.
- [x] I have updated/added relevant documentation in `docs` and added
dartdoc comments with `///`.
- [ ] I have updated/added relevant examples in `examples` or `docs`.

## Breaking Change?
- [ ] Yes, this PR is a breaking change.
- [x] No, this PR is not a breaking change.

### Migration instructions
No work to be done.

## Related Issues
#2769

## QUESTIONS FOR THE DEVS
So I'm not satisfied with the fixes for `Circle` bounds. The bounds uses
the maxima of the viewport, and yes, creates a circle that stays within
the size of the bounds circle, but I'd expect that `Circle` bounds
should keep my viewport _inside_ the circle as it is documented to stay
_inside_ the desired bounds shape. Therefore, I think `Circle` case
should have 4 incident points to fit the viewport. That is to say, the
viewport is not the maxima of the newly calculated `Circle`, but will be
fully contained by this new `Circle`. This way, you'll never see outside
of the bounds as it, seems to me anyway, implies. However I do not know
if this is intuitive for others and expected behavior. Thoughts?
luanpotter pushed a commit that referenced this pull request Oct 15, 2024
# Description
The behavior for `CameraComponent.setBounds` was not behaving correctly.
The viewport-aware behavior was not triggering until a window resize
event. All supported bound variants were not respecting the fixed
resolution viewport, or in some cases, behaving unpredictably.

The issue was with the fact the viewport-aware behavior component
depends on the bounded position component to be in its parent, but was
returning `null` during `onMount`. The viewport math was using the
logical size instead of the virtual size. I made some math changes to be
accurate.

This PR adds new a getter `CameraComponent.considerViewport`.
`ViewportAwareBoundsBehavior` is now added as a side effect of mounting
`BoundedPositionBehavior` by waiting for the `mounted` future. This
guarantees that
Flames life cycle is respected and the components behave as expected. 

Tests pass on my own local project. 

Videos are linked on the discord thread
[here](https://discord.com/channels/509714518008528896/1275814019235709003/1275888732481785925).

Melos dry run completed successfully.

## Checklist
- [x] I have followed the [Contributor Guide] when preparing my PR.
- [x] I have updated/added tests for ALL new/updated/fixed
functionality.
- [x] I have updated/added relevant documentation in `docs` and added
dartdoc comments with `///`.
- [ ] I have updated/added relevant examples in `examples` or `docs`.

## Breaking Change?
- [ ] Yes, this PR is a breaking change.
- [x] No, this PR is not a breaking change.

### Migration instructions
No work to be done.

## Related Issues
#2769

## QUESTIONS FOR THE DEVS
So I'm not satisfied with the fixes for `Circle` bounds. The bounds uses
the maxima of the viewport, and yes, creates a circle that stays within
the size of the bounds circle, but I'd expect that `Circle` bounds
should keep my viewport _inside_ the circle as it is documented to stay
_inside_ the desired bounds shape. Therefore, I think `Circle` case
should have 4 incident points to fit the viewport. That is to say, the
viewport is not the maxima of the newly calculated `Circle`, but will be
fully contained by this new `Circle`. This way, you'll never see outside
of the bounds as it, seems to me anyway, implies. However I do not know
if this is intuitive for others and expected behavior. Thoughts?
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

Add the ability to calculate make the camera component bounds to behave like the old camera object
2 participants