Make Quat distinct, replace GameObject.rotation by a union (with quat and euler).
Replace `@` quat multiplication by `*` to avoid confusion.
This commit is contained in:
parent
17470f4c02
commit
a5c9bb3c70
26 changed files with 103 additions and 66 deletions
|
@ -61,7 +61,7 @@ import ./screen
|
|||
import ./types
|
||||
import ./util
|
||||
import std/tables
|
||||
import vmath except Quat
|
||||
import vmath except Quat, quat
|
||||
|
||||
export attributes
|
||||
export blend
|
||||
|
@ -89,7 +89,7 @@ export texture
|
|||
export types
|
||||
export ubo
|
||||
export util
|
||||
export vmath except Quat
|
||||
export vmath except Quat, quat
|
||||
|
||||
import platform/platform
|
||||
when defined(android):
|
||||
|
|
|
@ -37,7 +37,7 @@ proc swap_lines(p: pointer, line_stride, line_count: int) {.gcsafe.}
|
|||
# End forward declarations
|
||||
|
||||
import std/strformat
|
||||
import vmath except Quat
|
||||
import vmath except Quat, quat
|
||||
import arr_ref
|
||||
import float16
|
||||
|
||||
|
|
|
@ -33,7 +33,7 @@
|
|||
import ../types
|
||||
import std/options
|
||||
import std/tables
|
||||
import vmath except Quat
|
||||
import vmath except Quat, quat
|
||||
|
||||
# Forward declarations
|
||||
proc newFramebuffer*(engine: MyouEngine,
|
||||
|
|
|
@ -33,7 +33,7 @@
|
|||
import ../types
|
||||
import ../platform/gl
|
||||
import arr_ref
|
||||
import vmath except Quat
|
||||
import vmath except Quat, quat
|
||||
|
||||
# Forward declarations
|
||||
func newRenderCameraData*(world_matrix, proj_matrix: Mat4, cull_planes: array[6, Vec4], viewport_size: Vec2): RenderCameraData
|
||||
|
|
|
@ -31,7 +31,7 @@
|
|||
# version of this file under either the CPAL or the [AGPL-3] License.
|
||||
|
||||
import ../types
|
||||
import vmath except Quat
|
||||
import vmath except Quat, quat
|
||||
import arr_ref
|
||||
# import tinyre
|
||||
import std/tables
|
||||
|
|
|
@ -33,7 +33,7 @@
|
|||
import ../types
|
||||
import std/tables
|
||||
import std/strformat
|
||||
import vmath except Quat
|
||||
import vmath except Quat, quat
|
||||
import ../platform/gl
|
||||
import arr_ref
|
||||
export arr_ref
|
||||
|
|
|
@ -30,7 +30,7 @@
|
|||
# License. If you do not delete the provisions above, a recipient may use your
|
||||
# version of this file under either the CPAL or the [AGPL-3] License.
|
||||
|
||||
import vmath except Quat
|
||||
import vmath except Quat, quat
|
||||
|
||||
type KeyCode* = enum
|
||||
KeyUnknown = - 1, KeySpace = 32, KeyApostrophe = 39, KeyComma = 44, KeyMinus,
|
||||
|
|
|
@ -45,8 +45,9 @@ import std/strformat
|
|||
import std/bitops
|
||||
import std/options
|
||||
import std/json
|
||||
import vmath except Quat
|
||||
import vmath except Quat, quat
|
||||
import ../quat
|
||||
import ../util
|
||||
|
||||
import ../../libs/loadable/loadable
|
||||
import ../graphics/material
|
||||
|
@ -432,7 +433,7 @@ proc loadObjectImpl(self: BlendLoader, scene: Scene, obn: FNode): (GameObject, s
|
|||
ubos = scene.get_lighting_UBOs,
|
||||
double_sided = not backface_culling,
|
||||
)
|
||||
|
||||
|
||||
for v in varyings:
|
||||
if v.vtype == Tangent and v.attname notin tangents:
|
||||
tangents.add v.attname
|
||||
|
@ -563,13 +564,13 @@ proc loadObjectImpl(self: BlendLoader, scene: Scene, obn: FNode): (GameObject, s
|
|||
of BEulerZYX: EulerZYX
|
||||
ob.rotation = if ob.rotation_order == Quaternion:
|
||||
let q = obn.quat.f32
|
||||
quat(q[1], q[2], q[3], q[0])
|
||||
Rotation(quat: quat(q[1], q[2], q[3], q[0]))
|
||||
elif ob.rotation_order == AxisAngle:
|
||||
assert false, "not implemented"
|
||||
quat()
|
||||
Rotation()
|
||||
else:
|
||||
let r = obn.rot.f32
|
||||
quat(r[0], r[1], r[2], 1)
|
||||
Rotation(euler: vec3(r[0], r[1], r[2]))
|
||||
ob.scale = obn.size.f32.vec3
|
||||
let restrictflag = obn.restrictflag.i16[0]
|
||||
ob.visible = (restrictflag and 4) == 0
|
||||
|
|
|
@ -36,7 +36,7 @@ import std/strformat
|
|||
import std/strutils
|
||||
import std/tables
|
||||
import tinyre
|
||||
import vmath except Quat
|
||||
import vmath except Quat, quat
|
||||
# import sugar
|
||||
|
||||
import ../types
|
||||
|
|
|
@ -47,7 +47,7 @@ import ../types
|
|||
import ./blend_curve_maps
|
||||
import ./blend_format
|
||||
import ./glsl_functions
|
||||
import vmath except Quat
|
||||
import vmath except Quat, quat
|
||||
|
||||
type Expr = object
|
||||
str: string
|
||||
|
|
|
@ -31,7 +31,7 @@
|
|||
# version of this file under either the CPAL or the [AGPL-3] License.
|
||||
|
||||
|
||||
import vmath except Quat
|
||||
import vmath except Quat, quat
|
||||
|
||||
func plane_from_norm_point*(normal, point: Vec3): Vec4 =
|
||||
## Calculates a plane from a normal and a point in the plane.
|
||||
|
|
|
@ -32,7 +32,7 @@
|
|||
|
||||
import ../types
|
||||
import std/options
|
||||
import vmath except Quat
|
||||
import vmath except Quat, quat
|
||||
import ../quat
|
||||
import ../util
|
||||
|
||||
|
@ -122,7 +122,7 @@ proc get_ray_direction_local*(self: Camera, x, y: float32): Vec3 =
|
|||
## relative to the camera. The upper left corner of the viewport has
|
||||
## coordinates (0,0), and the lower right corner (1,1).
|
||||
assert self.rotation_order == Quaternion
|
||||
return self.rotation * (self.projection_matrix_inverse *
|
||||
return self.rotation.quat * (self.projection_matrix_inverse *
|
||||
vec3(x * 2 - 1, 1 - y * 2, 1))
|
||||
|
||||
# proc get_position_at_depth*(self: camera, x, y: float32): Vec3 =
|
||||
|
|
|
@ -32,7 +32,7 @@
|
|||
|
||||
import ../types
|
||||
import std/strformat
|
||||
import vmath except Quat
|
||||
import vmath except Quat, quat
|
||||
|
||||
when defined(nimdoc):
|
||||
type TYPES* = ProbeInfluenceType | ProbeParallaxType | CubemapProbe | CubemapProbeUniform | SH9Uniform
|
||||
|
|
|
@ -31,7 +31,7 @@
|
|||
# version of this file under either the CPAL or the [AGPL-3] License.
|
||||
|
||||
import ../types
|
||||
import vmath except Quat
|
||||
import vmath except Quat, quat
|
||||
import ../quat
|
||||
|
||||
when defined(nimdoc):
|
||||
|
@ -123,7 +123,7 @@ proc initGameObject*(self: GameObject, engine: MyouEngine, name: string="", scen
|
|||
# Remember to add any new mutable reference to clone()
|
||||
self.engine = engine
|
||||
self.name = name
|
||||
self.rotation = quat()
|
||||
self.rotation.quat = quat()
|
||||
self.rotation_order = EulerXYZ
|
||||
self.scale = vec3(1, 1, 1)
|
||||
self.object_color = vec4(1, 1, 1, 1)
|
||||
|
@ -179,9 +179,9 @@ proc update_matrices*(self: GameObject) =
|
|||
## and parent matrix. It assumes the parent object and/or bone has its world
|
||||
## matrix already up to date.
|
||||
var q = if self.rotation_order == Quaternion:
|
||||
self.rotation
|
||||
self.rotation.quat
|
||||
else:
|
||||
to_quat(self.rotation.xyz, self.rotation_order)
|
||||
to_quat(self.rotation.euler, self.rotation_order)
|
||||
# var q = self.rotation
|
||||
q = normalize(q)
|
||||
let (x,y,z,w) = q.toTuple
|
||||
|
@ -238,17 +238,18 @@ proc set_rotation_order*(self: GameObject, order: RotationOrder) =
|
|||
## Change the rotation mode and order of the object.
|
||||
if order == self.rotation_order:
|
||||
return
|
||||
var q = self.rotation
|
||||
if self.rotation_order != Quaternion:
|
||||
q = q.xyz.to_quat(self.rotation_order)
|
||||
var q = if self.rotation_order != Quaternion:
|
||||
self.rotation.euler.to_quat(self.rotation_order)
|
||||
else:
|
||||
self.rotation.quat
|
||||
self.rotation = case order:
|
||||
of Quaternion: q
|
||||
of EulerXYZ: vec4(q.to_euler_XYZ, 0)
|
||||
of EulerXZY: vec4(q.to_euler_XZY, 0)
|
||||
of EulerYXZ: vec4(q.to_euler_YXZ, 0)
|
||||
of EulerYZX: vec4(q.to_euler_YZX, 0)
|
||||
of EulerZXY: vec4(q.to_euler_ZXY, 0)
|
||||
of EulerZYX: vec4(q.to_euler_ZYX, 0)
|
||||
of Quaternion: Rotation(quat: q)
|
||||
of EulerXYZ: Rotation(euler: q.to_euler_XYZ)
|
||||
of EulerXZY: Rotation(euler: q.to_euler_XZY)
|
||||
of EulerYXZ: Rotation(euler: q.to_euler_YXZ)
|
||||
of EulerYZX: Rotation(euler: q.to_euler_YZX)
|
||||
of EulerZXY: Rotation(euler: q.to_euler_ZXY)
|
||||
of EulerZYX: Rotation(euler: q.to_euler_ZYX)
|
||||
of AxisAngle:
|
||||
raise newException(ValueError, "axis angle not supported yet")
|
||||
self.rotation_order = order
|
||||
|
@ -263,9 +264,9 @@ proc get_local_matrix*(self: GameObject): Mat4 =
|
|||
## Calculates and returns the transformation matrix in local space. If the
|
||||
## object has no parent, this is equivalent to the world matrix.
|
||||
var q = if self.rotation_order == Quaternion:
|
||||
self.rotation
|
||||
self.rotation.quat
|
||||
else:
|
||||
to_quat(self.rotation.xyz, self.rotation_order)
|
||||
to_quat(self.rotation.euler, self.rotation_order)
|
||||
q = normalize(q)
|
||||
var (x,y,z,w) = q.toTuple
|
||||
let scl = self.scale
|
||||
|
@ -368,7 +369,7 @@ proc rotate_quat*(self: GameObject, q: Quat, relative_object: GameObject=nil): G
|
|||
let rotation_order = self.rotation_order
|
||||
if rotation_order != Quaternion:
|
||||
self.set_rotation_order(Quaternion)
|
||||
self.rotation = inv_par @ rel @ q @ inv_rel @ par @ self.rotation
|
||||
self.rotation.quat = inv_par * rel * q * inv_rel * par * self.rotation.quat
|
||||
if rotation_order != Quaternion:
|
||||
self.set_rotation_order(rotation_order)
|
||||
return self
|
||||
|
@ -415,10 +416,10 @@ proc set_world_position_rotation*(
|
|||
# TODO: does it work well with parents?
|
||||
let wm = inverse(self.parent.get_world_matrix * self.matrix_parent_inverse)
|
||||
self.position = wm * position
|
||||
self.rotation = wm.to_mat3_rotation.to_quat @ rotation
|
||||
self.rotation.quat = wm.to_mat3_rotation.to_quat * rotation
|
||||
else:
|
||||
self.position = position
|
||||
self.rotation = rotation
|
||||
self.rotation.quat = rotation
|
||||
self.rotation_order = Quaternion
|
||||
|
||||
proc set_world_position*(self: GameObject, position: Vec3) =
|
||||
|
|
|
@ -31,7 +31,7 @@
|
|||
# version of this file under either the CPAL or the [AGPL-3] License.
|
||||
|
||||
import ../types
|
||||
import vmath except Quat
|
||||
import vmath except Quat, quat
|
||||
import ../util
|
||||
import ../graphics/material
|
||||
|
||||
|
|
|
@ -35,7 +35,7 @@
|
|||
# TODO: Split mesh object and mesh data
|
||||
|
||||
import ../types
|
||||
import vmath except Quat
|
||||
import vmath except Quat, quat
|
||||
import arr_ref
|
||||
|
||||
when defined(nimdoc):
|
||||
|
|
|
@ -33,7 +33,7 @@
|
|||
|
||||
|
||||
import ../types
|
||||
import vmath except Quat
|
||||
import vmath except Quat, quat
|
||||
import std/strutils
|
||||
type Window* = ref object of RootObj
|
||||
|
||||
|
|
|
@ -33,7 +33,7 @@
|
|||
|
||||
|
||||
import ../types
|
||||
import vmath except Quat
|
||||
import vmath except Quat, quat
|
||||
import std/strutils
|
||||
type Window* = ref object of RootObj
|
||||
|
||||
|
|
|
@ -33,7 +33,7 @@
|
|||
|
||||
|
||||
import ../types
|
||||
import vmath except Quat
|
||||
import vmath except Quat, quat
|
||||
import glfm
|
||||
type Window* = ptr GLFMDisplay
|
||||
|
||||
|
|
|
@ -33,7 +33,7 @@
|
|||
|
||||
|
||||
import ../types
|
||||
import vmath except Quat
|
||||
import vmath except Quat, quat
|
||||
|
||||
import nglfw as glfw
|
||||
type Window* = glfw.Window
|
||||
|
|
55
src/quat.nim
55
src/quat.nim
|
@ -2,7 +2,7 @@
|
|||
|
||||
# TODO: make own vmath module with proper quats and more matrix types
|
||||
|
||||
import vmath except Quat
|
||||
import vmath except Quat, quat
|
||||
|
||||
type RotationOrder* = enum
|
||||
Quaternion
|
||||
|
@ -16,20 +16,33 @@ type RotationOrder* = enum
|
|||
|
||||
|
||||
type
|
||||
GQuat*[T] = GVec4[T]
|
||||
GQuat*[T] = object
|
||||
x*, y*, z*, w*: float32
|
||||
Quat* = GQuat[float32]
|
||||
DQuat* = GQuat[float64]
|
||||
|
||||
template gquat*[T](x,y,z,w: T): GQuat[T] =
|
||||
gvec4[T](x,y,z,w)
|
||||
proc gquat*[T](x,y,z,w: T): GQuat[T] {.inline.} =
|
||||
GQuat[T](x:x, y:y, z:z, w:w)
|
||||
|
||||
proc `@`*[T](a,b: GQuat[T]): GQuat[T] =
|
||||
return gquat[T](
|
||||
a.x * b.w + a.w * b.x + a.y * b.z - a.z * b.y,
|
||||
a.y * b.w + a.w * b.y + a.z * b.x - a.x * b.z,
|
||||
a.z * b.w + a.w * b.z + a.x * b.y - a.y * b.x,
|
||||
a.w * b.w - a.x * b.x - a.y * b.y - a.z * b.z,
|
||||
)
|
||||
proc quat*(x,y,z,w: float32): Quat {.inline.} =
|
||||
Quat(x:x, y:y, z:z, w:w)
|
||||
|
||||
proc dquat*(x,y,z,w: float64): DQuat {.inline.} =
|
||||
DQuat(x:x, y:y, z:z, w:w)
|
||||
|
||||
template quat*(): Quat = quat(0,0,0,1)
|
||||
template dquat*(): DQuat = dquat(0,0,0,1)
|
||||
template quat*(v: Vec4): Quat = cast[Quat](v)
|
||||
template dquat*(v: DVec4): DQuat = cast[DQuat](v)
|
||||
|
||||
template `[]`*[T](q: GQuat[T], i: int): T = cast[array[4, T]](q)[i]
|
||||
template `[]=`*[T](q: var GQuat[T], i: int, v: T) =
|
||||
case i:
|
||||
of 0: q.x = v
|
||||
of 1: q.y = v
|
||||
of 2: q.z = v
|
||||
of 3: q.w = v
|
||||
else: raise RangeDefect.newException "Index out of range"
|
||||
|
||||
# https://stackoverflow.com/questions/28673777/convert-quaternion-from-right-handed-to-left-handed-coordinate-system
|
||||
# S=s, B=−b, C=−d and D=−c.
|
||||
|
@ -73,6 +86,12 @@ func inverse*[T](a: GQuat[T]): GQuat[T] =
|
|||
return
|
||||
return gquat[T](-a.x, -a.y, -a.z, a.w) * (1/dot)
|
||||
|
||||
proc normalize*[T](q: GQuat[T]): GQuat[T] =
|
||||
let length = sqrt(q.x * q.x + q.y * q.y + q.z * q.z + q.w * q.w)
|
||||
if length == 0:
|
||||
return q
|
||||
return gquat[T](q.x / length , q.y / length, q.z / length, q.w / length)
|
||||
|
||||
func to_mat3*[T](q: GQuat[T]): GMat3[T] =
|
||||
let xx = q.x * q.x * 2
|
||||
let yx = q.y * q.x * 2
|
||||
|
@ -109,6 +128,16 @@ func `*`*[T](q: GQuat[T], a: GVec3[T]): GVec3[T] =
|
|||
result.y = iy * q.w + iw * -q.y + iz * -q.x - ix * -q.z
|
||||
result.z = iz * q.w + iw * -q.z + ix * -q.y - iy * -q.x
|
||||
|
||||
proc `*`*[T](a, b: GQuat[T]): GQuat[T] =
|
||||
GQuat[T](
|
||||
x: a.w * b.x + a.x * b.w + a.y * b.z - a.z * b.y,
|
||||
y: a.w * b.y - a.x * b.z + a.y * b.w + a.z * b.x,
|
||||
z: a.w * b.z + a.x * b.y - a.y * b.x + a.z * b.w,
|
||||
w: a.w * b.w - a.x * b.x - a.y * b.y - a.z * b.z,
|
||||
)
|
||||
|
||||
proc `*`*[T](q: GQuat[T], s: T): GQuat[T] =
|
||||
GQuat[T](x: q.x*s, y: q.y*s, z: q.z*s, w: q.w*s)
|
||||
|
||||
template to_tuple*[T](v: GQuat[T]): (T, T, T, T) =
|
||||
# (v.x, v.y, v.z, v.w)
|
||||
|
@ -232,10 +261,10 @@ proc rotationTo*(a,b: Vec3): Quat =
|
|||
var v = cross(vec3(1,0,0), a)
|
||||
if v.length < 0.000001:
|
||||
v = cross(vec3(0,1,0), a)
|
||||
return fromAxisAngle(v.normalize, PI)
|
||||
return cast[Quat](fromAxisAngle(v.normalize, PI))
|
||||
elif dot > 0.999999:
|
||||
return quat(0,0,0,1)
|
||||
else:
|
||||
let v = cross(a, b)
|
||||
return quat(v.x, v.y, v.z, 1.0+dot)
|
||||
return quat(v.x, v.y, v.z, 1.0+dot).normalize
|
||||
|
||||
|
|
|
@ -62,7 +62,7 @@ proc ensure_cubemaps*(self: Scene)
|
|||
proc render_all_cubemaps*(self: Scene, use_roughness_prefiltering: bool, mipmap_shader: Material = nil)
|
||||
# End forward declarations
|
||||
|
||||
import vmath except Quat
|
||||
import vmath except Quat, quat
|
||||
import ./quat
|
||||
import std/algorithm
|
||||
import std/math
|
||||
|
@ -256,7 +256,7 @@ proc make_parent*(self: Scene, parent: GameObject, child: GameObject,
|
|||
let rotation = parent.get_world_rotation
|
||||
let rotation_order = child.rotation_order
|
||||
child.set_rotation_order(Quaternion)
|
||||
var rot = inverse(rotation) * child.rotation
|
||||
var rot = inverse(rotation) * child.rotation.quat
|
||||
let rot_inv = inverse(rot)
|
||||
var scale = child.scale
|
||||
# get local rotation matrix and scale it with parent world scale vector
|
||||
|
@ -294,7 +294,7 @@ proc clear_parent*(self: Scene, child: GameObject, keep_transform = true) =
|
|||
let rotation_order = child.rotation_order
|
||||
let (position, rotation) = child.get_world_position_rotation
|
||||
child.position = position
|
||||
child.rotation = rotation
|
||||
child.rotation.quat = rotation
|
||||
child.rotation_order = Quaternion
|
||||
var scale = child.scale
|
||||
let world_matrix = child.world_matrix
|
||||
|
|
|
@ -44,7 +44,7 @@ proc emulateMouseWithTouch*(screen: Screen, touch: int32, ending: bool, x, y: fl
|
|||
import std/bitops
|
||||
import std/sequtils
|
||||
import std/math
|
||||
import vmath except Quat
|
||||
import vmath except Quat, quat
|
||||
import ./graphics/framebuffer
|
||||
import ./objects/camera
|
||||
import ./platform/platform
|
||||
|
|
|
@ -32,7 +32,7 @@
|
|||
|
||||
import std/strutils
|
||||
import ../types
|
||||
import vmath except Quat
|
||||
import vmath except Quat, quat
|
||||
import ../graphics/framebuffer
|
||||
import ../graphics/material
|
||||
import ../graphics/render
|
||||
|
|
|
@ -32,7 +32,7 @@
|
|||
|
||||
import std/tables
|
||||
import std/options
|
||||
import vmath except Quat
|
||||
import vmath except Quat, quat
|
||||
import json
|
||||
import arr_ref
|
||||
import ./platform/gl
|
||||
|
@ -90,6 +90,12 @@ type
|
|||
|
||||
# gameobject.nim
|
||||
|
||||
Rotation* {.union.} = object
|
||||
euler*: Vec3
|
||||
quat*: Quat
|
||||
axis_angle*: tuple[axis: Vec3, angle: float32]
|
||||
vec4*: Vec4
|
||||
|
||||
ObjectType* = enum ## private
|
||||
TGameObject
|
||||
TMesh
|
||||
|
@ -103,7 +109,7 @@ type
|
|||
engine* {.cursor.}: MyouEngine ## private
|
||||
debug*: bool ## private
|
||||
position*: Vec3
|
||||
rotation*: Quat
|
||||
rotation*: Rotation
|
||||
radius*: float
|
||||
rotation_order*: RotationOrder
|
||||
scale*: Vec3
|
||||
|
@ -949,7 +955,7 @@ type BlendLoader* = ref object of Loader
|
|||
(string, seq[Varying], OrderedTable[string, string], OrderedTable[string, TexturePixels])] ## private
|
||||
resource*: LoadableResource
|
||||
textures*: Table[string, Texture]
|
||||
|
||||
|
||||
template enqueue*(renderer: RenderManager, fun: untyped) =
|
||||
## Run a proc after the renderer has been initialized.
|
||||
##
|
||||
|
|
|
@ -31,7 +31,7 @@
|
|||
# version of this file under either the CPAL or the [AGPL-3] License.
|
||||
|
||||
import ./platform/gl
|
||||
import vmath except Quat
|
||||
import vmath except Quat, quat
|
||||
import std/math
|
||||
import std/algorithm
|
||||
import arr_ref
|
||||
|
|
Loading…
Reference in a new issue