[3.x] Expose determinant in Transform2D, rename internal method

This commit is contained in:
Aaron Franke 2023-04-21 01:28:55 -05:00
parent 17e0a0205d
commit 141783d90f
No known key found for this signature in database
GPG Key ID: 40A1750B977E56BF
11 changed files with 32 additions and 9 deletions

View File

@ -44,7 +44,7 @@ Transform2D Transform2D::inverse() const {
} }
void Transform2D::affine_invert() { void Transform2D::affine_invert() {
real_t det = basis_determinant(); real_t det = determinant();
#ifdef MATH_CHECKS #ifdef MATH_CHECKS
ERR_FAIL_COND(det == 0); ERR_FAIL_COND(det == 0);
#endif #endif
@ -93,7 +93,7 @@ Transform2D::Transform2D(real_t p_rot, const Vector2 &p_pos) {
} }
Size2 Transform2D::get_scale() const { Size2 Transform2D::get_scale() const {
real_t det_sign = SGN(basis_determinant()); real_t det_sign = SGN(determinant());
return Size2(elements[0].length(), det_sign * elements[1].length()); return Size2(elements[0].length(), det_sign * elements[1].length());
} }
@ -217,7 +217,7 @@ Transform2D Transform2D::rotated(real_t p_angle) const {
return copy; return copy;
} }
real_t Transform2D::basis_determinant() const { real_t Transform2D::determinant() const {
return elements[0].x * elements[1].y - elements[0].y * elements[1].x; return elements[0].x * elements[1].y - elements[0].y * elements[1].x;
} }

View File

@ -79,7 +79,7 @@ struct _NO_DISCARD_CLASS_ Transform2D {
void translate(real_t p_tx, real_t p_ty); void translate(real_t p_tx, real_t p_ty);
void translate(const Vector2 &p_translation); void translate(const Vector2 &p_translation);
real_t basis_determinant() const; real_t determinant() const;
Size2 get_scale() const; Size2 get_scale() const;
void set_scale(const Size2 &p_scale); void set_scale(const Size2 &p_scale);

View File

@ -886,6 +886,7 @@ struct _VariantCall {
VCALL_PTR1R(Transform2D, translated); VCALL_PTR1R(Transform2D, translated);
VCALL_PTR2R(Transform2D, interpolate_with); VCALL_PTR2R(Transform2D, interpolate_with);
VCALL_PTR1R(Transform2D, is_equal_approx); VCALL_PTR1R(Transform2D, is_equal_approx);
VCALL_PTR0R(Transform2D, determinant);
static void _call_Transform2D_xform(Variant &r_ret, Variant &p_self, const Variant **p_args) { static void _call_Transform2D_xform(Variant &r_ret, Variant &p_self, const Variant **p_args) {
switch (p_args[0]->type) { switch (p_args[0]->type) {
@ -2151,6 +2152,7 @@ void register_variant_methods() {
ADDFUNC1R(TRANSFORM2D, TRANSFORM2D, Transform2D, translated, VECTOR2, "offset", varray()); ADDFUNC1R(TRANSFORM2D, TRANSFORM2D, Transform2D, translated, VECTOR2, "offset", varray());
ADDFUNC1R(TRANSFORM2D, NIL, Transform2D, xform, NIL, "v", varray()); ADDFUNC1R(TRANSFORM2D, NIL, Transform2D, xform, NIL, "v", varray());
ADDFUNC1R(TRANSFORM2D, NIL, Transform2D, xform_inv, NIL, "v", varray()); ADDFUNC1R(TRANSFORM2D, NIL, Transform2D, xform_inv, NIL, "v", varray());
ADDFUNC0R(TRANSFORM2D, REAL, Transform2D, determinant, varray());
ADDFUNC1R(TRANSFORM2D, VECTOR2, Transform2D, basis_xform, VECTOR2, "v", varray()); ADDFUNC1R(TRANSFORM2D, VECTOR2, Transform2D, basis_xform, VECTOR2, "v", varray());
ADDFUNC1R(TRANSFORM2D, VECTOR2, Transform2D, basis_xform_inv, VECTOR2, "v", varray()); ADDFUNC1R(TRANSFORM2D, VECTOR2, Transform2D, basis_xform_inv, VECTOR2, "v", varray());
ADDFUNC2R(TRANSFORM2D, TRANSFORM2D, Transform2D, interpolate_with, TRANSFORM2D, "transform", REAL, "weight", varray()); ADDFUNC2R(TRANSFORM2D, TRANSFORM2D, Transform2D, interpolate_with, TRANSFORM2D, "transform", REAL, "weight", varray());

View File

@ -60,6 +60,13 @@
This method does not account for translation (the origin vector). This method does not account for translation (the origin vector).
</description> </description>
</method> </method>
<method name="determinant">
<return type="float" />
<description>
Returns the determinant of the basis matrix. If the basis is uniformly scaled, then its determinant equals the square of the scale factor.
A negative determinant means the basis was flipped, so one part of the scale is negative. A zero determinant means the basis isn't invertible, and is usually considered invalid.
</description>
</method>
<method name="get_origin"> <method name="get_origin">
<return type="Vector2" /> <return type="Vector2" />
<description> <description>

View File

@ -837,7 +837,7 @@ void RasterizerCanvasBaseGLES2::canvas_light_shadow_buffer_update(RID p_buffer,
VS::CanvasOccluderPolygonCullMode transformed_cull_cache = instance->cull_cache; VS::CanvasOccluderPolygonCullMode transformed_cull_cache = instance->cull_cache;
if (transformed_cull_cache != VS::CANVAS_OCCLUDER_POLYGON_CULL_DISABLED && if (transformed_cull_cache != VS::CANVAS_OCCLUDER_POLYGON_CULL_DISABLED &&
(p_light_xform.basis_determinant() * instance->xform_cache.basis_determinant()) < 0) { (p_light_xform.determinant() * instance->xform_cache.determinant()) < 0) {
transformed_cull_cache = (transformed_cull_cache == VS::CANVAS_OCCLUDER_POLYGON_CULL_CLOCKWISE) transformed_cull_cache = (transformed_cull_cache == VS::CANVAS_OCCLUDER_POLYGON_CULL_CLOCKWISE)
? VS::CANVAS_OCCLUDER_POLYGON_CULL_COUNTER_CLOCKWISE ? VS::CANVAS_OCCLUDER_POLYGON_CULL_COUNTER_CLOCKWISE
: VS::CANVAS_OCCLUDER_POLYGON_CULL_CLOCKWISE; : VS::CANVAS_OCCLUDER_POLYGON_CULL_CLOCKWISE;

View File

@ -895,7 +895,7 @@ void RasterizerCanvasBaseGLES3::canvas_light_shadow_buffer_update(RID p_buffer,
VS::CanvasOccluderPolygonCullMode transformed_cull_cache = instance->cull_cache; VS::CanvasOccluderPolygonCullMode transformed_cull_cache = instance->cull_cache;
if (transformed_cull_cache != VS::CANVAS_OCCLUDER_POLYGON_CULL_DISABLED && if (transformed_cull_cache != VS::CANVAS_OCCLUDER_POLYGON_CULL_DISABLED &&
(p_light_xform.basis_determinant() * instance->xform_cache.basis_determinant()) < 0) { (p_light_xform.determinant() * instance->xform_cache.determinant()) < 0) {
transformed_cull_cache = (transformed_cull_cache == VS::CANVAS_OCCLUDER_POLYGON_CULL_CLOCKWISE) transformed_cull_cache = (transformed_cull_cache == VS::CANVAS_OCCLUDER_POLYGON_CULL_CLOCKWISE)
? VS::CANVAS_OCCLUDER_POLYGON_CULL_COUNTER_CLOCKWISE ? VS::CANVAS_OCCLUDER_POLYGON_CULL_COUNTER_CLOCKWISE
: VS::CANVAS_OCCLUDER_POLYGON_CULL_CLOCKWISE; : VS::CANVAS_OCCLUDER_POLYGON_CULL_CLOCKWISE;

View File

@ -1623,7 +1623,7 @@ void CanvasItemEditor::_solve_IK(Node2D *leaf_node, Point2 target_position) {
Point2 current = (joints_list[node_id - 1]->get_global_position() - joints_list[node_id]->get_global_position()).normalized(); Point2 current = (joints_list[node_id - 1]->get_global_position() - joints_list[node_id]->get_global_position()).normalized();
Point2 target = (joints_pos[node_id - 1] - joints_list[node_id]->get_global_position()).normalized(); Point2 target = (joints_pos[node_id - 1] - joints_list[node_id]->get_global_position()).normalized();
float rot = current.angle_to(target); float rot = current.angle_to(target);
if (joints_list[node_id]->get_global_transform().basis_determinant() < 0) { if (joints_list[node_id]->get_global_transform().determinant() < 0) {
rot = -rot; rot = -rot;
} }
joints_list[node_id]->rotate(rot); joints_list[node_id]->rotate(rot);

View File

@ -93,6 +93,11 @@ godot_vector2 GDAPI godot_transform2d_get_scale(const godot_transform2d *p_self)
return dest; return dest;
} }
godot_real GDAPI godot_transform2d_determinant(const godot_transform2d *p_self) {
const Transform2D *self = (const Transform2D *)p_self;
return self->determinant();
}
godot_transform2d GDAPI godot_transform2d_orthonormalized(const godot_transform2d *p_self) { godot_transform2d GDAPI godot_transform2d_orthonormalized(const godot_transform2d *p_self) {
godot_transform2d dest; godot_transform2d dest;
const Transform2D *self = (const Transform2D *)p_self; const Transform2D *self = (const Transform2D *)p_self;

View File

@ -4431,6 +4431,13 @@
["const godot_transform2d *", "p_self"] ["const godot_transform2d *", "p_self"]
] ]
}, },
{
"name": "godot_transform2d_determinant",
"return_type": "godot_real",
"arguments": [
["const godot_transform2d *", "p_self"]
]
},
{ {
"name": "godot_transform2d_orthonormalized", "name": "godot_transform2d_orthonormalized",
"return_type": "godot_transform2d", "return_type": "godot_transform2d",

View File

@ -74,6 +74,8 @@ godot_vector2 GDAPI godot_transform2d_get_origin(const godot_transform2d *p_self
godot_vector2 GDAPI godot_transform2d_get_scale(const godot_transform2d *p_self); godot_vector2 GDAPI godot_transform2d_get_scale(const godot_transform2d *p_self);
godot_real GDAPI godot_transform2d_determinant(const godot_transform2d *p_self);
godot_transform2d GDAPI godot_transform2d_orthonormalized(const godot_transform2d *p_self); godot_transform2d GDAPI godot_transform2d_orthonormalized(const godot_transform2d *p_self);
godot_transform2d GDAPI godot_transform2d_rotated(const godot_transform2d *p_self, const godot_real p_phi); godot_transform2d GDAPI godot_transform2d_rotated(const godot_transform2d *p_self, const godot_real p_phi);

View File

@ -1786,8 +1786,8 @@ Control *Viewport::_gui_find_control_at_pos(CanvasItem *p_node, const Point2 &p_
} }
Transform2D matrix = p_xform * p_node->get_transform(); Transform2D matrix = p_xform * p_node->get_transform();
// matrix.basis_determinant() == 0.0f implies that node does not exist on scene // matrix.determinant() == 0.0f implies that node does not exist on scene
if (matrix.basis_determinant() == 0.0f) { if (matrix.determinant() == 0.0f) {
return nullptr; return nullptr;
} }