274 lines
7.9 KiB
C
Raw Normal View History

2022-07-14 16:47:30 +08:00
#include "PikaMath_Quaternion.h"
#include "PikaStdData_List.h"
#include "math.h"
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
}
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
}
double PikaMath_Quaternion_get(PikaObj* self, int key) {
if (key == 0) {
return obj_getFloat(self, "x");
2022-07-14 16:47:30 +08:00
}
if (key == 1) {
return obj_getFloat(self, "y");
2022-07-14 16:47:30 +08:00
}
if (key == 2) {
return obj_getFloat(self, "z");
2022-07-14 16:47:30 +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;
}
}
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
}
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
}
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
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_;
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_;
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
}
double PikaMath_Quaternion_magnituded(PikaObj* self) {
2022-07-14 16:47:30 +08:00
float magnituded = PikaMath_Quaternion_magnitudedsquare(self);
return pow(magnituded, 0.5);
2022-07-14 16:47:30 +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
float magnitudedsquare = pow(x, 2) + pow(y, 2) + pow(z, 2) + pow(w, 2);
2022-07-14 16:47:30 +08:00
return magnitudedsquare;
}
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
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
}
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);
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
}
}
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
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
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
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
}
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
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
return (x_ * x + y_ * y + z_ * z + w_ * w);
2022-07-14 16:47:30 +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);
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
}
}
int PikaMath_Quaternion_isnormalize(PikaObj* self) {
2022-07-14 16:47:30 +08:00
float mag = PikaMath_Quaternion_magnituded(self);
if (1.0 == mag) {
2022-07-14 16:47:30 +08:00
return 1;
} else {
2022-07-14 16:47:30 +08:00
return 0;
}
}
#define PI_DIV_180 (0.017453292519943296)
#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);
}
float sinp = sin(pitch / 2);
float siny = sin(yaw / 2);
float sinr = sin(roll / 2);
2022-07-14 16:47:30 +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
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
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
}
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));
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));
float rpy[3] = {roll, pitch, yaw};
2022-07-14 16:47:30 +08:00
PikaObj* list = newNormalObj(New_PikaStdData_List);
PikaStdData_List___init__(list);
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;
}