-
-
Notifications
You must be signed in to change notification settings - Fork 21.4k
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
Physics interpolation (2D) #88424
Physics interpolation (2D) #88424
Conversation
Transform2D parent_xform = p_parent_xform; | ||
|
||
if (snapping_2d_transforms_to_pixel) { | ||
xform.columns[2] = xform.columns[2].round(); | ||
final_xform.columns[2] = final_xform.columns[2].round(); | ||
parent_xform.columns[2] = parent_xform.columns[2].round(); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Note: If you are doing snapping here, you may want the snapping to apply to the local object, but NOT to propagate to the children, so they can do their snapping independently. This may entail creating an extra intermediate e.g. snapped_xform
and use this for rendering, while passing the unchanged final_xform
to the children.
This certainly needs consideration because snapping is done differently in 3.x.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I tried multiple times snap the children independently (ask @markdibarry) but it always resulted with snapping issues. I think it's best to snap the parent as is and pass the parent to the children.
Great work @rburing . This is something I was not looking forward to 😁 as I'm not so familiar with master and there will inevitably be a lot of 4.x specific bugs / ongoing maintenance work. Particles in particular have the potential to be tricky because (at least in 3.x) they were previously not specified in world space. They (and MultiMesh which CPUParticles used under the hood) were the most time consuming feature in 3.x, though more so for 3D. Also we should pay particularly attention at whether CPUParticles are planned to be deprecated in 4.x (in which case you will have to get them working with GPUParticles). Also I noted that, at least in 3D, particles in 4.x already use some kind of interpolation so you will need to get both systems to play nice with each other. Also as I noted above there may end up being some rejigging for snapping purposes. This is something I'm anticipating may need adjustment in 3.x based on feedback, as we haven't yet released 3.6. So in some ways this code may get more testing in 4.x (also bearing in mind there are now many more users on 4.x than 3.x). The paradigm is intended to be that transforms are interpolated and passed through the tree without snapping, and snapping is applied on a per object basis. Consider if a child node has a scaling applied, then trying to interpolate from snapped parent transforms would look incorrect otherwise. BTW an option (if other reviewers are agreeable to this) that I used in 3.x is to leave particles to a followup PR. They will probably have a lot more bikeshedding / bugs, and it makes it easier to review both sections in depth. This does mean that realistically users will have to wait for particle PR to make finished games, but it does mean they can start testing etc more rapidly. |
Only GPU particles have built-in interpolation in 4.x, and it's not complete yet. |
Thanks for the feedback @lawnjelly ! I think it makes sense to leave particles for a follow-up PR, if reviewers can agree. I'll have to look at the snapping (propagation) more carefully, and I'll ask for some more eyes on this on RocketChat. The latest push also fixes this issue with TileMaps: physics-interpolation-streaking.mp4 |
The test from the project PixelPerfectTest.zip from #84380 seems to pass with a physics tick rate of 30 (also lower ones) plus the interpolation from this PR, after commenting out the physics-interpolation-pixel-test.mp4This was recorded using Peek because movie writer mode is bugged when using the settings of the test project. The potential issue #88424 (comment) with propagating snapped transforms to children should be present in |
336b826
to
406b409
Compare
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Tested locally (rebased on top of master
b8f106a), it works as expected. Amazing work!
Here's the 2D platformer demo with Physics Ticks per Second set to 10
:
platformer.mp4
I noticed an issue when resizing the viewport when using the viewport
stretch mode and a Camera2D. If the aspect ratio changes, the camera motion will be interpolated, even though this doesn't occur in discrete steps when physics interpolation is disabled. The Camera2D in the project has its Process Callback set to Idle (the default).
This is hardly noticeable with tickrates greater than 30 Hz, so it's not a huge issue, but it might be worth looking into nonetheless.
See these videos with Physics Ticks per Second set to 5
:
Physics interpolation disabled
simplescreenrecorder-2024-02-20_18.15.04.mp4
Physics interpolation enabled
simplescreenrecorder-2024-02-20_18.15.37.mp4
In the bullet shower demo with Physics Ticks per Second set to 5
, something strange happens with the sprite movement. It's interpolated even though we're moving it every frame in _input()
based on mouse movement, so it can move faster than the physics tick rate.
The individual white sprites (that are not nodes but created via low-level servers) are also not interpolated:
Physics interpolation disabled
simplescreenrecorder-2024-02-20_18.26.11.mp4
Physics interpolation enabled
simplescreenrecorder-2024-02-20_18.25.52.mp4
Parallax backgrounds look strange when wrapping over with low physics tick rate (10 here), and they don't interpolate as smoothly as they could. They seem to interpolate in visible "steps", as if the interpolation was only performed for half of the physics tick length:
simplescreenrecorder-2024-02-20_18.37.21.mp4
Testing project: physics_platformer_modified.zip
Also, I strongly suggest enabling physics interpolation by default. We'll need to stick to a default value once this is merged, and with high refresh rate displays slowly becoming the norm among players, it's expected that games should adapt automatically without requiring extra work from the developer's side. While this will require updating some tutorials, I believe it's worth the effort.
@Calinou Thanks a lot for your testing! I've rebased the PR and fixed the issue with parallax (it should not be interpolated, as it is already updated every frame). I'm not sure what to do about the player sprite in the bullet shower demo. The bullets, with their transforms set manually every physics tick, are rendered using func _draw():
var offset = -bullet_image.get_size() * 0.5
for bullet in bullets:
var interpolated_position = lerp(bullet.position + Vector2(bullet.speed / Engine.physics_ticks_per_second, 0), bullet.position, Engine.get_physics_interpolation_fraction())
draw_texture(bullet_image, interpolated_position + offset) (It could be improved a bit further by storing the current and previous position, thereby avoiding the hack to compute I haven't yet looked into the Camera2D interpolation when resizing the viewport with stretch mode set to It's easy to make physics interpolation the default (should it also be a basic setting then?), but currently this would also override the |
I'm not sure if it should be made a basic setting, considering we want to encourage leaving physics interpolation enabled as much as possible. In the long run, physics interpolation is something you should only disable for troubleshooting, or for games intended to run at a fixed framerate (which is generally a bad idea, but can minimize input lag in this particular situation). |
I've finally had a chance to test this (sorry was away from home when writing originally so was working from memory), and it doesn't seem to cause the problem that I envisaged, because the scaling turns out to be applied after the translation. (The problem I was thinking of was e.g. moving parent node by 0.9 pixels, then child being scaled 10x and this movement being lost, when it should have moved 9 pixels.) But this was based on the incorrect assumption of scaling being applied first, so it seems fine so far in some tests. 👍 |
EDIT: It's the same in 3.x, I'll try and work out why. It's an order of operations bug, the transform notification is coming in after the reset, even if the reset is called after setting the transform in the script. I'll see if I can fix it in 3.x, and we can carry fix over. Ok I have it: The This won't be too involved to fix but I can't guarantee I'll be able to do before going off on holiday tomorrow for a week, but I don't think this should be a showstopper for this PR (after all it is already merged in 3.x with this bug 😄 ). This same bug could be affecting some of the other non-canvas items, but with any luck they can all be fixed in similar way. UPDATE: Fixed in 3.x by #89577 . Should be fixed in this PR soon. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
What a great port. I tested it summarily (and it seems to work amazingly), but I think others tested it thoroughly.
The code seems OK! Didn't find anything requiring major changes, but this is a hefty PR, so I suggest that we merge for dev6 then we fix issues afterwards.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Looks fantastic, really good work!
@rburing From your comment earlier, you probably want to add your new method to void Parallax2D::_update_repeat() {
if (!is_inside_tree()) {
return;
}
Point2 repeat_scale = repeat_size * get_scale();
RenderingServer::get_singleton()->canvas_set_item_repeat(get_canvas_item(), repeat_scale, repeat_times);
RenderingServer::get_singleton()->canvas_item_set_interpolated(get_canvas_item(), false);
} |
136338c
to
e07c2f9
Compare
Thanks for the reviews! And thanks @markdibarry, it's done. This should be ready to merge now. |
Adds fixed timestep interpolation to the rendering server (2D only). Switchable on and off with a project setting (default is off). Co-authored-by: lawnjelly <lawnjelly@gmail.com>
06abc86
Thanks! Amazing work! |
#endif // _3D_DISABLED | ||
|
||
// Prepare the fixed timestep interpolated nodes BEFORE they are updated | ||
// by the physics server, otherwise the current and previous transforms | ||
// may be the same, and no interpolation takes place. | ||
OS::get_singleton()->get_main_loop()->iteration_prepare(); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
If the goal is to do this before physics, why isn't it done before all physics (including 3D physics?)
It would make sense to me to open this up to 3D in the future, so either this should be moved before 3D, or the comment needs to be adjusted explaining why things are the way they are.
As-is, this code also leaves me with the question: If I want to disable 2D physics, should I keep this line enabled? I don't know.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
@aaronfranke You are also absolutely right, I didn't spot this in the PR, the prepare should probably be before the 3D here (although it shouldn't make any difference until the 3D is done, so we can fix in 3D PR).
(This is actually something I need to double check in 3.x actually, because there was some rejigging here when I added 2D and it may have affected 3D, so thanks for pointing to this! 👍 3.x has a flush_queries()
here but no sync()
.)
To address the confusion a little - "physics interpolation" (somewhat confusingly) is nothing to do with physics engine. The more correct name is fixed timestep interpolation.
The only relation is to the physics ticks, rather than the physics itself. It can work with or without physics engine. Juan preferred the name "physics interpolation", because even though incorrect, he thought it would be easier for users to understand (and admittedly in many cases their eyes would glaze over with "fixed timestep interpolation" 😁 ).
This is something that we as developers will be expected to learn the difference, but it will inevitably cause confusion in some way (if we e.g. called it "fixed timestep interpolation" in the c++, and "physics interpolation" for user facing code, this could be confusing, as many c++ funcs are bound etc, so it is called "physics interpolation" throughout).
As it is not related to the physics engines, it should not be disabled when disabling physics engines (as _physics_process()
still occurs).
Just to try and explain what it is doing .. the VisualServer
items have to be prepared before each iteration (tick) before any changes have been made to them (server side at least).
If you follow the code it calls update_interpolation_tick()
.
This does a bunch of things:
- Detect any objects that are no longer being moved, and ensure their previous transform = current transform so they cease vibrating
- Store the current transform (from the last tick) to the previous transform, so they can be interpolated after the tick
This is a port of @lawnjelly's
to the master branch. It also takes bits
TODO:
TODO
in the codeLeft for a follow-up PR (if it is acceptable):
Thanks to @lawnjelly for the original PR.
Testing welcome! Make sure to enable the project setting
physics/common/physics_interpolation
and setphysics/common/physics_jitter_fix
to0.0
when testing.This PR is sponsored by My Spare Time™