Could Bevy support non-Euclidean geometry? #4207
Replies: 4 comments 14 replies
-
One of the simplest and most important bits of non-Euclidean geometry is simple 2D wrapping (think Joust). How do you think users should implement this sort of problem: both in terms of a current design, and under a hypothetical non-Euclidean future? |
Beta Was this translation helpful? Give feedback.
-
Firmly agree. I really appreciate how you've laid this out for folks without a math background. This sort of insight is critical to designing user-controllable transform types. We already have at least 3 extremely compelling and distinct use cases (2D, 3D and UI), with candidates such as fixed point and
Frustums and rays are the other key geometric primitive used for camera culling and occlusion; I'm not sure that they'll play nicely with non-Euclidean geometry out of the box.
Agreed. We should aim to optimize common case performance, but provide clean escape hatches for users trying to do very unusual things.
This is a helpful and clear insight: we should try to keep this in mind when redesigning. |
Beta Was this translation helpful? Give feedback.
-
I'll be the devil's advocate here and ask for a generalized use case. The split of the hierarchy from Transforms is largely being driven by the need to use it for more than just transforms, or non 3D Euclidean space where there are two existing very common use cases already in-engine (2D and UI). The desire to have a f64/fixed point transform system are nice and have their use cases, but they're likely to be something ecosystem crates provide, not bevy proper. Other than specialized games like Hyperbolica and Portal, I can't immediately think of big use cases for non-Euclidean transforms. While fun, those games are rather rare. Is there a compelling use case in typical game development for these kinds of spaces? If not, could this not live as an ecosystem crate? |
Beta Was this translation helpful? Give feedback.
-
It will be useful to have a couple of tests or examples to use as a reference if/when this is undertaken. Even might be worth making a fork with just the specific changes to transform and render required to get each example to function, as it will illustrate precisely where issues are and inform & expedite the design and review of a proper generalisation. |
Beta Was this translation helpful? Give feedback.
-
Euclidean geometry is the geometry we're used to. Triangles have angles adding to 180°, circles have circumference 2πr, et cetera. But there are also non-Euclidean geometries. In hyperbolic geometry, for example, there's more space than you expect — circles have circumference 2π*sinh(r), which grows exponentially! On the other hand, spherical geometry has less space than you expect, eventually curling around on itself; the whole universe has finite volume.
OpenGL, and similar systems, are made for rendering scenes in Euclidean geometry. Yet if you know what you're doing, it's no harder to render scenes in, say, spherical geometry!
(Proof: Here's the code for a simple Euclidean geometry renderer, and here's the code for a corresponding spherical geometry renderer. The latter is only fourteen lines longer, and that's mainly because I used a more complicated mesh.)
This is possible because GL doesn't really care about precise distances or angles. It matters which surfaces are in front of which others, but not how far in front. As a result, the differences between Euclidean and spherical geometry are unimportant to GL; it'll happily render either.
Geometry without a concept of distance and angle is called projective geometry. And OpenGL works with projective geometry the whole way through.
Could Bevy do the same? What would it take to generalize Bevy to support projective geometry, and hence simple non-Euclidean geometries?
Projective Geometry, and
Vec3
vs.Vec4
In projective geometry, we represent points as nonzero
Vec4
s. Two points are the same if one vector is a scaled copy of the other. For example,(3,4,5,1)
is a point.(6,8,10,2)
is the same point, but(3,4,5,2)
is a different point.Projective transformations are just invertible
Mat4
s. Since (for real numbersk
)M * (k * v) = k * (M * v)
, projective transformations correctly take equal points to equal points.This is why you see
vec4
s andmat4
s all over OpenGL.In the Euclidean case, it is convenient to always write a point as a vector whose last coordinate is one. (If you have
(x,y,z,w)
, rewrite it as(x/w, y/w, z/w, 1)
.) We then often drop that last coordinate, representing a point as aVec3
.But in other geometries, that convention is unhelpful. In spherical geometry, for instance, it is instead convenient to represent a point as a vector whose length is one. Other geometries will have other conventions, most of which don't yield a convenient simplification to
Vec3
.The upshot is that when generalizing to projective geometry, most instances of
Vec3
in the code should be replaced withVec4
.Transform
Bevy's
Transform
andGlobalTransform
types are currently tied to Euclidean geometry; they consist of a translation, rotation, and scale. But there is already appetite to change this. RFC 53, for instance, suggests dedicatedRectTransform
andTransform2D
types specialized for UIs and 2D games. I've also heard talk of a potentialf64
-basedTransform
, for when more precision is required. So we might as well allow users to define their ownTransform
types, including non-Euclidean ones.What's needed from a
Transform
type?Transform
must be composable, so that it can be propagated up the hierarchy.transform_propagate_system
doesn't depend on implementation details.Transform
, but it will probably exist. The same is true for inverses.Transform
must be convertible toMat4
. If we don't support non-Euclidean geometry, strengthen that toAffine3A
.Transform
s should be compatible with multiplication ofAffine3A
/Mat4
s.(Actually, Bevy's non-uniform scale breaks that last rule. Therefore, I don't like non-uniform scale. But that's a discussion for another time.)
In other words,
Transform
should be a group, equipped with a group homomorphism toAffine3A
/Mat4
.Issue with
GlobalTransform
GlobalTransform
seems to have two different purposes.From the user's point of view,
GlobalTransform
is a way to learn the absolute position of an entity, even if it's buried in a hierarchy. Maybe you want to position a bullet relative to a gun, but the gun is held in a hand, which is attached to an arm, which is attached to a player. So you set the bullet'sTransform
equal to the gun'sGlobalTransform
.From this perspective, we clearly want
GlobalTransform
to have the same representation asTransform
. In pull request #4168, @alice-i-cecile even suggests that it should be replaced by something likeInherited<Transform>
. So if we have severalTransform
types, we'll want severalGlobalTransform
s as well.But
GlobalTransform
serves another purpose. The renderer basically uses it as anAffine3A
. In this use-case, we definitely do not want multipleGlobalTransform
s; that'd force us to duplicate all of the render systems!Since there's currently only one
Transform
type, Bevy can get away with usingGlobalTransform
for both purposes. But this won't work when we have severalTransform
types. We'll need different types — maybe something likeInherited<Transform>
andRenderTransform
?Rendering
As I mentioned,
bevy_render
basically usesGlobalTransform
as anAffine3A
. If we can generalize that toMat4
,bevy_render
should work for non-Euclidean geometry as well.I believe this can be done, with some thought. Some
Vec3
s will becomeVec4
s; some computations will need rethinking. But it should be possible.A potential issue is that axis-aligned bounding boxes don't generalize well to non-Euclidean geometry. I don't understand Bevy's rendering process well enough to know if this is a problem.
Bounding ellipsoids would generalize, but I suspect those would be more computationally expensive.
bevy_pbr
will be trickier — some parts of it, like the shader, might just need to be rewritten for each geometry. But other parts, like materials, should be reusable. We'll want it to be easy to replace the geometry-specific parts without having to copy-paste the geometry-independent parts,More Info
There's certainly more I could say on this topic.
Transform
for spherical or hyperbolic geometry.And those are just the things I already wrote up for this post, before I realized I was including way too much detail.
If you'd like to know about those, or anything else about non-Euclidean geometry, feel free to ask! I'll be happy to answer your questions.
Beta Was this translation helpful? Give feedback.
All reactions