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

Algebraic quaternions vs geometric quaternions #10

Closed
andyferris opened this issue May 3, 2016 · 7 comments
Closed

Algebraic quaternions vs geometric quaternions #10

andyferris opened this issue May 3, 2016 · 7 comments

Comments

@andyferris
Copy link

Hi @SimonDanisch, we at Roames (myself, @awbsmith and @c42f) have been using and thinking about quaternions, and there seems to be a few conceptual blockers to e.g. using * as in you suggest in #3.

Primarily, a quaternion is a algebraic object, Quaternion{T} <: Number, much like Complex{T} <: Number. They form a ring, and algebraic operations + and * have a single unambiguous meaning. However, some quaternions, especially normalized quaternions, are useful for geometry, and it sure would be useful to implement rotations of 3-vectors.

I'm proposing that we break up the structure a little bit, so that Quaternion is a pure algebraic number and UnitQuaternion or RotationQuaternion or similar is an object that either inherits from AbstractQuaternion or else is a simple struct containing a Quaternion.

As an exaple, we would want

Quaternion(1.0,2.0,3.0,4.0) * Vec(1.0,2.0,3.0) <: Vec{3,Quaternion{Float64}} # elementwise multiplication with standard multiplication rules

(you could replace Vec with Vector, Point, any FixedVector, etc).

and

RotationQuaternion(cos(t), sin(t), 0.0, 0.0) * Vec(1.0,0.0,0.0) == Vec(cos(t), sin(t), 0.0) # or whatever it is for this rotation...

For a simple gist, see here.

This would help us in Rotations.jl, plus clean up the whole interface for people who want to do either just algebra or just geometry. In fact, there is some small argument to say the algebraic part belongs outside of JuliaGeometry, but the simplest thing is to make a separation.

@Orbots
Copy link

Orbots commented Jan 10, 2019

I don't think having the flag to indicate a quaternion is normalized is useful. I'd like to use this, but it just seems wasteful so I'm writing my own Quaternion type. The reason the norm flag is not useful is that it doesn't adequately answer the question "when" do we renormalize/project. Well, it appears that it does answer that question with ? (q.norm) never : always which will give you problems with drift ( you need to renormalize unit quaternions after awhile ).

Also this layout isn't the best for potential SIMD optimization.

@SimonDanisch
Copy link
Member

I admit I also rolled my own Quaternion type because of this :D But I have no idea about quaternions, so I dont feel able to actually refactor this package, to be mathematical sound...

@Orbots
Copy link

Orbots commented Jan 11, 2019

I initially didn't intend to hijack this issue with an somewhat orthogonal issue ( representation/implementation of unit quaternions ).

To bring it back to the original issue.

Quaternions are an extension of complex numbers right? So it's more Quaternion{T} <: Complex{T}.

The nicest representation of Quaternions is via Geometric Algebra/Grassmann Algebra/Clifford Algebra ( history and relationship between these are confusing ). But then you potentially would have some performance hit?

Anyways, I'm in support of having a more algebraic context-free Quaternion type as the "go to" Quaternion type. Although they are kind of geometric objects by nature.

@andyferris
Copy link
Author

So it's more Quaternion{T} <: Complex{T}

Don't think about subsets or isomorphisms with <:.

From the perspective of dispatch, subtypes inherit the interface (and importantly - all the guarantees - if you want to make generic code possible) of the supertype.

@Orbots
Copy link

Orbots commented Jan 11, 2019

The interface guarantee that making a Quaternion a subtype of Complex is that they both can have a set of functions with a notion of relative "angles"*. Also a guarantee that they can not be ordered ( less useful ). So if you had a function foo that you wanted to operate on all Numbers which have "angles" as conceptualized by Complex numbers. You could explicitly constrain the inputs with foo( ::Complex, ::Complex ). This could operate happily on Quaternion <: Complex type.

But yeah, abstract type inheritance is a bit of a goofy thing especially when you consider contra/covariance etc. So maybe these commonalities would be better expressed via a traits mechanism or just left as a MethodError. I think it could have some utility though.

*IANAM: I am not a mathematician.

@chakravala

This comment was marked as abuse.

@hyrodium
Copy link
Collaborator

I'll close this issue because the original issue seems solved.

This would help us in Rotations.jl, plus clean up the whole interface for people who want to do either just algebra or just geometry. In fact, there is some small argument to say the algebraic part belongs outside of JuliaGeometry, but the simplest thing is to make a separation.

Now Rotations.jl has a dependency on Quaternions.jl with JuliaGeometry/Rotations.jl#175.

I don't think having the flag to indicate a quaternion is normalized is useful.

This will be solved in #75, and there is a new discussion in #60.

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

No branches or pull requests

5 participants