All Downloads are FREE. Search and download functionalities are using the official Maven repository.

cn.nukkit.math.BVector3 Maven / Gradle / Ivy

package cn.nukkit.math;

import cn.nukkit.api.DoNotModify;
import cn.nukkit.api.PowerNukkitXDifference;
import cn.nukkit.api.PowerNukkitXOnly;
import cn.nukkit.api.Since;
import cn.nukkit.level.Location;

import static java.lang.StrictMath.*;

/**
 * 向量计算工具,同时整合了yaw和pitch与坐标空间的转换功能
 */
@PowerNukkitXOnly
@Since("1.6.0.0-PNX")
@PowerNukkitXDifference(info = "update Angle algorithm", since = "1.19.60-r1")
public final class BVector3 {
    /**
     * 向量的单位向量
     */
    private Vector3 vector3;//标准化的方向向量,模长为1
    private double yaw;//-90 270
    private double pitch;//-90 90
    /**
     * 向量的模
     */
    private double length;

    /**
     * 通过传入的Location的yaw与pitch初始化BVector3

* 此方法返回的BVector3的模长为1 * * @param location the location * @return the b vector 3 */ public static BVector3 fromLocation(Location location) { return fromLocation(location, 1); } /** * 通过传入的Location的yaw与pitch初始化BVector3

* 此方法返回的BVector3的模长为传入的length值 * * @param location the location * @return the b vector 3 */ public static BVector3 fromLocation(Location location, double length) { return new BVector3(location.getYaw(), location.getPitch(), length); } /** * 通过传入的yaw与pitch初始化BVector3

* 此方法返回的BVector3的模长为1 * * @param yaw the yaw (-90 270) * @param pitch the pitch * @return the b vector 3 */ public static BVector3 fromAngle(double yaw, double pitch) { return new BVector3(yaw, pitch, 1); } /** * 通过传入的向量坐标初始化BVector3 * * @param pos 向量坐标 * @return the b vector 3 */ public static BVector3 fromPos(Vector3 pos) { return new BVector3(pos); } /** * 通过传入的向量坐标初始化BVector3 * * @param x the x * @param y the y * @param z the z * @return the b vector 3 */ public static BVector3 fromPos(double x, double y, double z) { return fromPos(new Vector3(x, y, z)); } /** * 通过传入的yaw、pitch和向量的模初始化BVector3 * * @param yaw the yaw * @param pitch the pitch * @param length 向量模 */ private BVector3(double yaw, double pitch, double length) { this.vector3 = getDirectionVector(yaw, pitch); this.yaw = getYawFromVector(this.vector3); this.pitch = getPitchFromVector(this.vector3); this.length = length; } /** * 通过传入的向量坐标初始化BVector3 * * @param vector3 向量坐标 */ private BVector3(Vector3 vector3) { this.yaw = getYawFromVector(vector3); this.pitch = getPitchFromVector(vector3); this.vector3 = getDirectionVector(yaw, pitch); this.length = vector3.length(); } /** * 设置Yaw * * @param yaw the yaw * @return the yaw */ public BVector3 setYaw(double yaw) { this.vector3 = getDirectionVector(yaw, this.pitch); //重新计算在范围内的等价yaw值 this.yaw = getYawFromVector(this.vector3); return this; } /** * 设置 pitch. * * @param pitch the pitch * @return the pitch */ public BVector3 setPitch(double pitch) { this.vector3 = getDirectionVector(this.yaw, pitch); //重新计算在范围内的等价pitch值 this.pitch = getPitchFromVector(this.vector3); return this; } /** * 旋转Yaw * * @param yaw the yaw * @return the b vector 3 */ public BVector3 rotateYaw(double yaw) { this.yaw += yaw; this.vector3 = getDirectionVector(this.yaw, this.pitch); //重新计算在范围内的等价yaw值 this.yaw = getYawFromVector(this.vector3); return this; } /** * 旋转Pitch * * @param pitch the pitch * @return the b vector 3 */ public BVector3 rotatePitch(double pitch) { this.pitch += pitch; this.vector3 = getDirectionVector(this.yaw, this.pitch); //重新计算在范围内的等价pitch值 this.pitch = getPitchFromVector(this.vector3); return this; } /** * 旋转yaw和Pitch * * @param yaw the yaw * @param pitch the pitch * @return the b vector 3 */ public BVector3 rotate(double yaw, double pitch) { this.pitch += pitch; this.yaw += yaw; this.vector3 = getDirectionVector(this.yaw, this.pitch); //重新计算在范围内的等价pitch值 this.pitch = getPitchFromVector(this.vector3); this.pitch = getYawFromVector(this.vector3); return this; } /** * 向量加法 * * @return 结果向量 */ public BVector3 add(double x, double y, double z) { var pos = this.vector3.multiply(this.length); pos.add(x, y, z); this.yaw = getYawFromVector(pos); this.pitch = getPitchFromVector(pos); this.vector3 = pos.normalize(); this.length = pos.length(); return this; } /** * 向量加法 * * @return 结果向量 */ public BVector3 add(Vector3 vector3) { return add(vector3.x, vector3.y, vector3.z); } /** * 添加指定模长的方向向量到Vector3(0, 0, 0)

* 其实就是返回此向量的坐标 * * @return the vector 3 */ public Vector3 addToPos() { return new Vector3(this.vector3.x * this.length, this.vector3.y * this.length, this.vector3.z * this.length); } /** * 将此向量的坐标添加到pos上 * * @param pos the pos * @return the vector 3 */ public Vector3 addToPos(Vector3 pos) { return pos.add(this.vector3.x * this.length, this.vector3.y * this.length, this.vector3.z * this.length); } /** * 设置该向量的模 * * @param length the length * @return the length */ public BVector3 setLength(double length) { this.length = length; return this; } /** * 增加该向量的模

* 当然你也可以传入负数,但请确保最终长度要大于0! * * @param length 增加/减少的模 * @return 自身 */ public BVector3 extend(double length) { if ((this.length + length) <= 0) throw new IllegalArgumentException("Vector length must bigger than zero"); this.length += length; return this; } public double getYaw() { return yaw; } public double getPitch() { return pitch; } /** * 获取单位方向向量 * * @return the direction vector */ public Vector3 getDirectionVector() { return vector3.clone(); } /** * 获取未克隆的单位方向向量 * * @return the direction vector */ @DoNotModify public Vector3 getUnclonedDirectionVector() { return vector3; } /** * 通过yaw与pitch计算出等价的Vector3方向向量 * * @param yaw yaw * @param pitch pitch * @return Vector3方向向量 */ public static Vector3 getDirectionVector(double yaw, double pitch) { var pitch0 = toRadians(pitch + 90); var yaw0 = toRadians(yaw + 90); var x = sin(pitch0) * cos(yaw0); var z = sin(pitch0) * sin(yaw0); var y = cos(pitch0); return new Vector3(x, y, z).normalize(); } /** * 通过方向向量计算出yaw * * @param vector 方向向量 * @return yaw */ public static double getYawFromVector(Vector3 vector) { double yaw = toDegrees(asin(-vector.x / sqrt(vector.x * vector.x + vector.z * vector.z))); return -vector.z > 0.0D ? 180.0D - yaw : StrictMath.abs(yaw) < 1E-10 ? 0 : yaw; } /** * 通过方向向量计算出pitch * * @param vector 方向向量 * @return pitch */ public static double getPitchFromVector(Vector3 vector) { var pitch = toDegrees(asin(-vector.y / sqrt(vector.x * vector.x + vector.z * vector.z + vector.y * vector.y))); return StrictMath.abs(pitch) < 1E-10 ? 0 : pitch; } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy