2022-07-14 16:47:30 +08:00
|
|
|
#include "PikaMath_Quaternion.h"
|
|
|
|
#include "PikaStdData_List.h"
|
|
|
|
#include "math.h"
|
|
|
|
|
2022-07-30 18:26:58 +08:00
|
|
|
void PikaMath_Quaternion___init__(PikaObj* self) {
|
|
|
|
obj_setFloat(self, "x", 0.0);
|
|
|
|
obj_setFloat(self, "y", 0.0);
|
|
|
|
obj_setFloat(self, "z", 0.0);
|
|
|
|
obj_setFloat(self, "w", 1.0);
|
2022-07-14 16:47:30 +08:00
|
|
|
}
|
|
|
|
|
2022-07-30 18:26:58 +08:00
|
|
|
void PikaMath_Quaternion_set(PikaObj* self,
|
|
|
|
double x,
|
|
|
|
double y,
|
|
|
|
double z,
|
|
|
|
double w) {
|
|
|
|
obj_setFloat(self, "x", x);
|
|
|
|
obj_setFloat(self, "y", y);
|
|
|
|
obj_setFloat(self, "z", z);
|
|
|
|
obj_setFloat(self, "w", w);
|
2022-07-14 16:47:30 +08:00
|
|
|
}
|
|
|
|
|
2022-07-30 18:26:58 +08:00
|
|
|
double PikaMath_Quaternion_get(PikaObj* self, int key) {
|
|
|
|
if (key == 0) {
|
|
|
|
return obj_getFloat(self, "x");
|
2022-07-14 16:47:30 +08:00
|
|
|
}
|
2022-07-30 18:26:58 +08:00
|
|
|
if (key == 1) {
|
|
|
|
return obj_getFloat(self, "y");
|
2022-07-14 16:47:30 +08:00
|
|
|
}
|
2022-07-30 18:26:58 +08:00
|
|
|
if (key == 2) {
|
|
|
|
return obj_getFloat(self, "z");
|
2022-07-14 16:47:30 +08:00
|
|
|
}
|
2022-07-30 18:26:58 +08:00
|
|
|
if (key == 3) {
|
|
|
|
return obj_getFloat(self, "w");
|
|
|
|
} else {
|
|
|
|
obj_setErrorCode(self, 1);
|
2022-07-14 16:47:30 +08:00
|
|
|
return 0.0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-07-30 18:26:58 +08:00
|
|
|
void PikaMath_Quaternion_add(PikaObj* self, PikaObj* quat) {
|
|
|
|
float x_ = obj_getFloat(quat, "x");
|
|
|
|
float y_ = obj_getFloat(quat, "y");
|
|
|
|
float z_ = obj_getFloat(quat, "z");
|
|
|
|
float w_ = obj_getFloat(quat, "w");
|
|
|
|
|
|
|
|
float x = obj_getFloat(self, "x");
|
|
|
|
float y = obj_getFloat(self, "y");
|
|
|
|
float z = obj_getFloat(self, "z");
|
|
|
|
float w = obj_getFloat(self, "w");
|
|
|
|
|
|
|
|
float x_sum = x + x_;
|
|
|
|
float y_sum = y + y_;
|
|
|
|
float z_sum = z + z_;
|
|
|
|
float w_sum = w + w_;
|
|
|
|
|
|
|
|
obj_setFloat(self, "x", x_sum);
|
|
|
|
obj_setFloat(self, "y", y_sum);
|
|
|
|
obj_setFloat(self, "z", z_sum);
|
|
|
|
obj_setFloat(self, "w", w_sum);
|
2022-07-14 16:47:30 +08:00
|
|
|
}
|
|
|
|
|
2022-07-30 18:26:58 +08:00
|
|
|
void PikaMath_Quaternion_sub(PikaObj* self, PikaObj* quat) {
|
|
|
|
float x_ = obj_getFloat(quat, "x");
|
|
|
|
float y_ = obj_getFloat(quat, "y");
|
|
|
|
float z_ = obj_getFloat(quat, "z");
|
|
|
|
float w_ = obj_getFloat(quat, "w");
|
|
|
|
|
|
|
|
float x = obj_getFloat(self, "x");
|
|
|
|
float y = obj_getFloat(self, "y");
|
|
|
|
float z = obj_getFloat(self, "z");
|
|
|
|
float w = obj_getFloat(self, "w");
|
|
|
|
|
|
|
|
float x_sub = x - x_;
|
|
|
|
float y_sub = y - y_;
|
|
|
|
float z_sub = z - z_;
|
|
|
|
float w_sub = w - w_;
|
|
|
|
|
|
|
|
obj_setFloat(self, "x", x_sub);
|
|
|
|
obj_setFloat(self, "y", y_sub);
|
|
|
|
obj_setFloat(self, "z", z_sub);
|
|
|
|
obj_setFloat(self, "w", w_sub);
|
2022-07-14 16:47:30 +08:00
|
|
|
}
|
|
|
|
|
2022-07-30 18:26:58 +08:00
|
|
|
void PikaMath_Quaternion_mul(PikaObj* self, PikaObj* quat) {
|
|
|
|
float x_ = obj_getFloat(quat, "x");
|
|
|
|
float y_ = obj_getFloat(quat, "y");
|
|
|
|
float z_ = obj_getFloat(quat, "z");
|
|
|
|
float w_ = obj_getFloat(quat, "w");
|
2022-07-14 16:47:30 +08:00
|
|
|
|
2022-07-30 18:26:58 +08:00
|
|
|
float x = obj_getFloat(self, "x");
|
|
|
|
float y = obj_getFloat(self, "y");
|
|
|
|
float z = obj_getFloat(self, "z");
|
|
|
|
float w = obj_getFloat(self, "w");
|
2022-07-14 16:47:30 +08:00
|
|
|
|
|
|
|
float x_mul = w * x_ + x * w_ - y * z_ + z * y_;
|
2022-07-30 18:26:58 +08:00
|
|
|
float y_mul = w * y_ + y * w_ + x * z_ - z * x_;
|
2022-07-14 16:47:30 +08:00
|
|
|
float z_mul = w * z_ + z * w_ - x * y_ + y * x_;
|
|
|
|
float w_mul = w * w_ - x * x_ - y * y_ - z * z_;
|
|
|
|
|
2022-07-30 18:26:58 +08:00
|
|
|
obj_setFloat(self, "x", x_mul);
|
|
|
|
obj_setFloat(self, "y", y_mul);
|
|
|
|
obj_setFloat(self, "z", z_mul);
|
|
|
|
obj_setFloat(self, "w", w_mul);
|
2022-07-14 16:47:30 +08:00
|
|
|
}
|
2022-07-30 18:26:58 +08:00
|
|
|
double PikaMath_Quaternion_magnituded(PikaObj* self) {
|
2022-07-14 16:47:30 +08:00
|
|
|
float magnituded = PikaMath_Quaternion_magnitudedsquare(self);
|
|
|
|
|
2022-07-30 18:26:58 +08:00
|
|
|
return pow(magnituded, 0.5);
|
2022-07-14 16:47:30 +08:00
|
|
|
}
|
2022-07-30 18:26:58 +08:00
|
|
|
double PikaMath_Quaternion_magnitudedsquare(PikaObj* self) {
|
|
|
|
float x = obj_getFloat(self, "x");
|
|
|
|
float y = obj_getFloat(self, "y");
|
|
|
|
float z = obj_getFloat(self, "z");
|
|
|
|
float w = obj_getFloat(self, "w");
|
2022-07-14 16:47:30 +08:00
|
|
|
|
2022-07-30 18:26:58 +08:00
|
|
|
float magnitudedsquare = pow(x, 2) + pow(y, 2) + pow(z, 2) + pow(w, 2);
|
2022-07-14 16:47:30 +08:00
|
|
|
|
|
|
|
return magnitudedsquare;
|
|
|
|
}
|
|
|
|
|
2022-07-30 18:26:58 +08:00
|
|
|
void PikaMath_Quaternion_reverse(PikaObj* self) {
|
|
|
|
float x = obj_getFloat(self, "x");
|
|
|
|
float y = obj_getFloat(self, "y");
|
|
|
|
float z = obj_getFloat(self, "z");
|
|
|
|
float w = obj_getFloat(self, "w");
|
2022-07-14 16:47:30 +08:00
|
|
|
|
2022-07-30 18:26:58 +08:00
|
|
|
obj_setFloat(self, "x", -x);
|
|
|
|
obj_setFloat(self, "y", -y);
|
|
|
|
obj_setFloat(self, "z", -z);
|
|
|
|
obj_setFloat(self, "w", w);
|
2022-07-14 16:47:30 +08:00
|
|
|
}
|
|
|
|
|
2022-07-30 18:26:58 +08:00
|
|
|
void PikaMath_Quaternion_inverse(PikaObj* self) {
|
|
|
|
float x = obj_getFloat(self, "x");
|
|
|
|
float y = obj_getFloat(self, "y");
|
|
|
|
float z = obj_getFloat(self, "z");
|
|
|
|
float w = obj_getFloat(self, "w");
|
2022-07-14 16:47:30 +08:00
|
|
|
|
|
|
|
float mag = PikaMath_Quaternion_magnitudedsquare(self);
|
2022-07-30 18:26:58 +08:00
|
|
|
if (mag > 0.0001) {
|
|
|
|
obj_setFloat(self, "x", -x / mag);
|
|
|
|
obj_setFloat(self, "y", -y / mag);
|
|
|
|
obj_setFloat(self, "z", -z / mag);
|
|
|
|
obj_setFloat(self, "w", w / mag);
|
|
|
|
} else {
|
|
|
|
obj_setFloat(self, "x", 0.0);
|
|
|
|
obj_setFloat(self, "y", 0.0);
|
|
|
|
obj_setFloat(self, "z", 0.0);
|
|
|
|
obj_setFloat(self, "w", 1.0);
|
2022-07-14 16:47:30 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-07-30 18:26:58 +08:00
|
|
|
void PikaMath_Quaternion_crossproduct(PikaObj* self, PikaObj* quat) {
|
|
|
|
float x_ = obj_getFloat(quat, "x");
|
|
|
|
float y_ = obj_getFloat(quat, "y");
|
|
|
|
float z_ = obj_getFloat(quat, "z");
|
2022-07-14 16:47:30 +08:00
|
|
|
|
2022-07-30 18:26:58 +08:00
|
|
|
float x = obj_getFloat(self, "x");
|
|
|
|
float y = obj_getFloat(self, "y");
|
|
|
|
float z = obj_getFloat(self, "z");
|
2022-07-14 16:47:30 +08:00
|
|
|
|
2022-07-30 18:26:58 +08:00
|
|
|
float x_cross = y * z_ - z * y_;
|
|
|
|
float y_cross = z * x_ - x * z_;
|
|
|
|
float z_cross = x * y_ - y * x_;
|
2022-07-14 16:47:30 +08:00
|
|
|
|
2022-07-30 18:26:58 +08:00
|
|
|
obj_setFloat(self, "x", x_cross);
|
|
|
|
obj_setFloat(self, "y", y_cross);
|
|
|
|
obj_setFloat(self, "z", z_cross);
|
|
|
|
obj_setFloat(self, "w", 0.0);
|
2022-07-14 16:47:30 +08:00
|
|
|
}
|
|
|
|
|
2022-07-30 18:26:58 +08:00
|
|
|
double PikaMath_Quaternion_dot(PikaObj* self, PikaObj* quat) {
|
|
|
|
float x_ = obj_getFloat(quat, "x");
|
|
|
|
float y_ = obj_getFloat(quat, "y");
|
|
|
|
float z_ = obj_getFloat(quat, "z");
|
|
|
|
float w_ = obj_getFloat(quat, "w");
|
2022-07-14 16:47:30 +08:00
|
|
|
|
2022-07-30 18:26:58 +08:00
|
|
|
float x = obj_getFloat(self, "x");
|
|
|
|
float y = obj_getFloat(self, "y");
|
|
|
|
float z = obj_getFloat(self, "z");
|
|
|
|
float w = obj_getFloat(self, "w");
|
2022-07-14 16:47:30 +08:00
|
|
|
|
2022-07-30 18:26:58 +08:00
|
|
|
return (x_ * x + y_ * y + z_ * z + w_ * w);
|
2022-07-14 16:47:30 +08:00
|
|
|
}
|
|
|
|
|
2022-07-30 18:26:58 +08:00
|
|
|
void PikaMath_Quaternion_normalize(PikaObj* self) {
|
|
|
|
float x = obj_getFloat(self, "x");
|
|
|
|
float y = obj_getFloat(self, "y");
|
|
|
|
float z = obj_getFloat(self, "z");
|
|
|
|
float w = obj_getFloat(self, "w");
|
2022-07-14 16:47:30 +08:00
|
|
|
|
|
|
|
float mag = PikaMath_Quaternion_magnituded(self);
|
2022-07-30 18:26:58 +08:00
|
|
|
if (mag > 0.0001) {
|
|
|
|
obj_setFloat(self, "x", x / mag);
|
|
|
|
obj_setFloat(self, "y", y / mag);
|
|
|
|
obj_setFloat(self, "z", z / mag);
|
|
|
|
obj_setFloat(self, "w", w / mag);
|
|
|
|
} else {
|
|
|
|
obj_setFloat(self, "x", 0.0);
|
|
|
|
obj_setFloat(self, "y", 0.0);
|
|
|
|
obj_setFloat(self, "z", 0.0);
|
|
|
|
obj_setFloat(self, "w", 1.0);
|
2022-07-14 16:47:30 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-07-30 18:26:58 +08:00
|
|
|
int PikaMath_Quaternion_isnormalize(PikaObj* self) {
|
2022-07-14 16:47:30 +08:00
|
|
|
float mag = PikaMath_Quaternion_magnituded(self);
|
|
|
|
|
2022-07-30 18:26:58 +08:00
|
|
|
if (1.0 == mag) {
|
2022-07-14 16:47:30 +08:00
|
|
|
return 1;
|
2022-07-30 18:26:58 +08:00
|
|
|
} else {
|
2022-07-14 16:47:30 +08:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#define PI_DIV_180 (0.017453292519943296)
|
2022-07-30 18:26:58 +08:00
|
|
|
#define DegToRad(x) ((x)*PI_DIV_180)
|
|
|
|
|
|
|
|
void PikaMath_Quaternion_fromEuler(PikaObj* self,
|
|
|
|
double yaw,
|
|
|
|
double pitch,
|
|
|
|
double roll,
|
|
|
|
int mode) {
|
|
|
|
if (mode) {
|
2022-07-14 16:47:30 +08:00
|
|
|
pitch = DegToRad(pitch);
|
|
|
|
roll = DegToRad(roll);
|
|
|
|
yaw = DegToRad(yaw);
|
|
|
|
}
|
|
|
|
|
2022-07-30 18:26:58 +08:00
|
|
|
float sinp = sin(pitch / 2);
|
|
|
|
float siny = sin(yaw / 2);
|
|
|
|
float sinr = sin(roll / 2);
|
2022-07-14 16:47:30 +08:00
|
|
|
|
2022-07-30 18:26:58 +08:00
|
|
|
float cosp = cos(pitch / 2);
|
|
|
|
float cosy = cos(yaw / 2);
|
|
|
|
float cosr = cos(roll / 2);
|
2022-07-14 16:47:30 +08:00
|
|
|
|
2022-07-30 18:26:58 +08:00
|
|
|
float x = sinr * cosp * cosy - cosr * sinp * siny;
|
|
|
|
float y = cosr * sinp * cosy + sinr * cosp * siny;
|
|
|
|
float z = cosr * cosp * siny - sinr * sinp * cosy;
|
|
|
|
float w = cosr * cosp * cosy + sinr * sinp * siny;
|
2022-07-14 16:47:30 +08:00
|
|
|
|
2022-07-30 18:26:58 +08:00
|
|
|
obj_setFloat(self, "x", x);
|
|
|
|
obj_setFloat(self, "y", y);
|
|
|
|
obj_setFloat(self, "z", z);
|
|
|
|
obj_setFloat(self, "w", w);
|
2022-07-14 16:47:30 +08:00
|
|
|
}
|
|
|
|
|
2022-07-30 18:26:58 +08:00
|
|
|
PikaObj* PikaMath_Quaternion_toEuler(PikaObj* self) {
|
|
|
|
float x = obj_getFloat(self, "x");
|
|
|
|
float y = obj_getFloat(self, "y");
|
|
|
|
float z = obj_getFloat(self, "z");
|
|
|
|
float w = obj_getFloat(self, "w");
|
2022-07-14 16:47:30 +08:00
|
|
|
|
|
|
|
float roll = atan2(2 * (w * x + y * z), 1 - 2 * (x * x + y * y));
|
2022-07-30 18:26:58 +08:00
|
|
|
float pitch = asin(2 * (w * y - z * x));
|
2022-07-14 16:47:30 +08:00
|
|
|
float yaw = atan2(2 * (w * z + x * y), 1 - 2 * (y * y + z * z));
|
|
|
|
|
2022-07-30 18:26:58 +08:00
|
|
|
float rpy[3] = {roll, pitch, yaw};
|
2022-07-14 16:47:30 +08:00
|
|
|
|
|
|
|
PikaObj* list = newNormalObj(New_PikaStdData_List);
|
|
|
|
PikaStdData_List___init__(list);
|
|
|
|
|
2022-07-30 18:26:58 +08:00
|
|
|
for (int i = 0; i < 3; i++) {
|
2022-07-20 10:32:01 +08:00
|
|
|
Arg* token_arg = arg_newFloat(rpy[i]);
|
2022-07-14 16:47:30 +08:00
|
|
|
/* 添加到 list 对象 */
|
|
|
|
PikaStdData_List_append(list, token_arg);
|
|
|
|
/* 销毁 arg */
|
|
|
|
arg_deinit(token_arg);
|
|
|
|
}
|
|
|
|
|
|
|
|
return list;
|
|
|
|
}
|