- android - 多次调用 OnPrimaryClipChangedListener
- android - 无法更新 RecyclerView 中的 TextView 字段
- android.database.CursorIndexOutOfBoundsException : Index 0 requested, 光标大小为 0
- android - 使用 AppCompat 时,我们是否需要明确指定其 UI 组件(Spinner、EditText)颜色
几周前,我决定制作自己的简单 3D 游戏引擎。我在网上学习了一些教程,也看了很多视频,到目前为止,我已经得到了相当不错的引擎。但我有一个问题。当我旋转或缩放对象时,它的中心似乎是屏幕的左上角而不是其真正的中心。例如,当我旋转一个对象时,它会围绕屏幕的左边缘旋转,而当我缩放它时,它会向屏幕的一角或远离屏幕的一角缩放。这是我的代码:
网格类
package com.cub.cubefront.mesh;
import com.cub.cubefront.math.Matrix4f;
import com.cub.cubefront.math.Vector3f;
public class Mesh {
private Vector3f[] verts;
private Vector3f center;
private double rotX = 0;
private double rotY = 0;
private double rotZ = 0;
private double scaleX = 1;
private double scaleY = 1;
private double scaleZ = 1;
private double translateX = 0;
private double translateY = 0;
private double translateZ = 0;
public Mesh(int arg0) {
verts = new Vector3f[arg0];
}
public Vector3f getVertexAt(int arg0) {
return verts[arg0 - 1];
}
public Vector3f[] getVerticies() {
return verts;
}
public int getVerticiesCount() {
return verts.length;
}
public void setVertexAt(int arg0, Vector3f arg1) {
verts[arg0 - 1] = arg1;
}
public void setRotationPoint(Vector3f arg0) {
this.center = arg0;
}
public Vector3f getRotationPoint() {
return center;
}
public Vector3f getCenter() {
int arg0 = verts.length;
double centerX = 0;
double centerY = 0;
double centerZ = 0;
for (int i = 0; i < arg0; i++) {
centerX += verts[i].getX();
centerY += verts[i].getY();
centerZ += verts[i].getZ();
}
return new Vector3f((float)(centerX / arg0), (float)(centerY / arg0), (float)(centerZ / arg0));
}
public void rotateX(double arg0) {
this.rotX += Math.toRadians(arg0);
}
public void setRotationX(double arg0) {
this.rotX = Math.toRadians(arg0);
}
public Matrix4f getXRotationAsMatrix() {
return new Matrix4f(new double[][] { // YZ rotation matrix (X)
{ 1, 0, 0, 0 },
{ 0, Math.cos(rotX), Math.sin(rotX), 0 },
{ 0, -Math.sin(rotX), Math.cos(rotX), 0 },
{ 0, 0, 0, 1 }
});
}
public Matrix4f getZRotationAsMatrix() {
return new Matrix4f(new double[][] { // XY rotation matrix (Z)
{ Math.cos(rotZ), -Math.sin(rotZ), 0, 0 },
{ Math.sin(rotZ), Math.cos(rotZ), 0, 0 },
{ 0, 0, 1, 0 },
{ 0, 0, 0, 1 }
});
}
public void rotateY(double arg0) {
this.rotY += Math.toRadians(arg0);
}
public void setRotationY(double arg0) {
this.rotY = Math.toRadians(arg0);
}
public Matrix4f getYRotationAsMatrix() {
return new Matrix4f(new double[][] { // XZ rotation matrix (Y)
{ Math.cos(rotY), 0, Math.sin(rotY), 0 },
{ 0, 1, 0, 0 },
{ -Math.sin(rotY), 0, Math.cos(rotY), 0},
{ 0, 0, 0, 1 }
});
}
public void setRotationZ(double arg0) {
this.rotZ = Math.toRadians(arg0);
}
public void rotateZ(double arg0) {
this.rotZ += Math.toRadians(arg0);
}
public Matrix4f getRotation() {
return getZRotationAsMatrix().multiply(getXRotationAsMatrix()).multiply(getYRotationAsMatrix());
}
public void setScaleX(double arg0) {
this.scaleX = arg0;
}
public void scaleX(double arg0) {
this.scaleX += arg0;
}
public double getScaleX() {
return scaleX;
}
public void setScaleY(double arg0) {
this.scaleY = arg0;
}
public void scaleY(double arg0) {
this.scaleY += arg0;
}
public double getScaleY() {
return scaleY;
}
public void setScaleZ(double arg0) {
this.scaleZ = arg0;
}
public void scaleZ(double arg0) {
this.scaleZ += arg0;
}
public double getScaleZ() {
return scaleZ;
}
public void setScale(double arg0) {
setScaleX(arg0);
setScaleY(arg0);
setScaleZ(arg0);
}
public void setScale(double[][] arg0) {
this.scaleX = arg0[0][0];
this.scaleY = arg0[1][1];
this.scaleZ = arg0[2][2];
}
public void setScale(Matrix4f arg0) {
this.scaleX = arg0.getValueAt(0, 0);
this.scaleY = arg0.getValueAt(1, 1);
this.scaleZ = arg0.getValueAt(2, 2);
}
public void scale(double arg0) {
scaleX(arg0);
scaleY(arg0);
scaleZ(arg0);
}
public Matrix4f getScale() {
return new Matrix4f(new double[][] {
{ getScaleX(), 0, 0, 0 },
{ 0, getScaleY(), 0, 0 },
{ 0, 0, getScaleZ(), 0 },
{ 0, 0, 0, 1 }
});
}
public void translateX(double arg0) {
this.translateX += arg0;
}
public void translateY(double arg0) {
this.translateY += arg0;
}
public void translateZ(double arg0) {
this.translateZ += arg0;
}
public Matrix4f getTranslation() {
return new Matrix4f(new double[][] {
{ 1, 0, 0, translateX },
{ 0, 1, 0, translateY },
{ 0, 0, 1, translateZ },
{ 0, 0, 0, 1 }
});
}
}
场景类
package com.cub.cubefront;
import com.cub.cubefront.graphics.Bitmap;
import com.cub.cubefront.input.InputHandler;
import com.cub.cubefront.math.Matrix4f;
import com.cub.cubefront.math.Vector2f;
import com.cub.cubefront.math.Vector3f;
import com.cub.cubefront.mesh.Camera;
import com.cub.cubefront.mesh.Mesh;
public class Scene extends Bitmap {
private Camera defaultCam;
public InputHandler input = new InputHandler();
public Scene() {
super(MainEngine.getWidth(), MainEngine.getHeight());
defaultCam = new Camera();
}
public void update() { }
public void start() { }
public void render() { }
public void render(Mesh arg0) {
Matrix4f trans = arg0.getRotation().multiply(arg0.getScale().multiply(arg0.getTranslation()));
for (int i = 1; i < arg0.getVerticiesCount()+1; i++) { // Depth: Manipulate x and y with z
Vector3f v1 = trans.transform(arg0.getVertexAt(i));
for (int n = 1; n < i; n++) {
Vector3f v2 = trans.transform(arg0.getVertexAt(n));
drawLine(
new Vector2f((v1.getX()), (v1.getY())),
new Vector2f((v2.getX()), (v2.getY()))
);
}
}
}
public void clear() {
for (int i = 0; i < pixels.length; i++) {
pixels[i] = 0;
}
}
public Camera getCamera() {
return defaultCam;
}
public void setCamera(Camera defaultCam) {
this.defaultCam = defaultCam;
}
}
Matrix4f 类
package com.cub.cubefront.math;
public class Matrix4f {
private double[][] values;
public Matrix4f() {
values = new double[4][4];
setValues();
}
public Matrix4f(double[][] arg0) {
setValues(arg0);
}
private Matrix4f setValues() {
values[0][0] = 1; values[0][1] = 0; values[0][2] = 0; values[0][3] = 0;
values[1][0] = 0; values[1][1] = 1; values[1][2] = 0; values[1][3] = 0;
values[2][0] = 0; values[2][1] = 0; values[2][2] = 1; values[2][3] = 0;
values[3][0] = 0; values[3][1] = 0; values[3][2] = 0; values[3][3] = 1;
return this;
}
public Matrix4f multiply(Matrix4f arg0) {
double res[][] = new double[4][4];
for (int i = 0; i < 4; i++) {
for (int j = 0; j < 4; j++) {
for (int k = 0; k < 4; k++) {
res[i][j] += values[i][k] * arg0.getValueAt(k, j);
}
}
}
return new Matrix4f(res);
}
public double[][] getValues() {
return values;
}
public double getValueAt(int arg0, int arg1) {
return values[arg0][arg1];
}
public void setValues(double[][] arg0) {
this.values = arg0;
}
public void setValueAt(int arg0, int arg1, double arg2) {
values[arg0][arg1] = arg2;
}
public Vector3f transform(Vector3f arg0) {
return new Vector3f(
(float)(arg0.getX() * values[0][0] + arg0.getY() * values[1][0] + arg0.getZ() * values[2][0]),
(float)(arg0.getX() * values[0][1] + arg0.getY() * values[1][1] + arg0.getZ() * values[2][1]),
(float)(arg0.getY() * values[0][2] + arg0.getY() * values[1][2] + arg0.getZ() * values[2][2])
);
}
}
如果您仍然不明白我的意思,请看看这些图片。
图片#1:
图片#2:
你看到它是如何绕角而不是中心旋转的吗?感谢您努力提供帮助!
最佳答案
当您进行变换而没有在影响对象之前将其重置为原点时,通常会发生这种情况。请记住,矩阵顺序很重要。旋转是相对于原点进行的,因此如果您的对象不在原点上,那么它会奇怪地旋转。
将其转换为(0,0,0)
后尝试旋转它。然后对三角形应用所需的平移(即平移到原点、旋转、返回到原始位置)。
希望有帮助。
编辑
一般来说,如果您想围绕固定点旋转多边形,您应该平移多边形,使所需点位于 (0,0,0)
中。然后继续旋转,最后回到原来的位置。
假设您有一个带有顶点 (1,1)
、(3,1)
和 (2,3)
的三角形。如果您想围绕该三角形的中心(即(2,2)
)旋转,您应该:
(2,2)
(使 (2,2)
作为原点)(2,2)
添加到每个顶点,返回到原始位置。同样的原则也适用于 3 维。
关于java - 为什么我的旋转矩阵会产生额外的平移?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/37758837/
我正在编写一个具有以下签名的 Java 方法。 void Logger(Method method, Object[] args); 如果一个方法(例如 ABC() )调用此方法 Logger,它应该
我是 Java 新手。 我的问题是我的 Java 程序找不到我试图用作的图像文件一个 JButton。 (目前这段代码什么也没做,因为我只是得到了想要的外观第一的)。这是我的主课 代码: packag
好的,今天我在接受采访,我已经编写 Java 代码多年了。采访中说“Java 垃圾收集是一个棘手的问题,我有几个 friend 一直在努力弄清楚。你在这方面做得怎么样?”。她是想骗我吗?还是我的一生都
我的 friend 给了我一个谜语让我解开。它是这样的: There are 100 people. Each one of them, in his turn, does the following
如果我将使用 Java 5 代码的应用程序编译成字节码,生成的 .class 文件是否能够在 Java 1.4 下运行? 如果后者可以工作并且我正在尝试在我的 Java 1.4 应用程序中使用 Jav
有关于why Java doesn't support unsigned types的问题以及一些关于处理无符号类型的问题。我做了一些搜索,似乎 Scala 也不支持无符号数据类型。限制是Java和S
我只是想知道在一个 java 版本中生成的字节码是否可以在其他 java 版本上运行 最佳答案 通常,字节码无需修改即可在 较新 版本的 Java 上运行。它不会在旧版本上运行,除非您使用特殊参数 (
我有一个关于在命令提示符下执行 java 程序的基本问题。 在某些机器上我们需要指定 -cp 。 (类路径)同时执行java程序 (test为java文件名与.class文件存在于同一目录下) jav
我已经阅读 StackOverflow 有一段时间了,现在我才鼓起勇气提出问题。我今年 20 岁,目前在我的家乡(罗马尼亚克卢日-纳波卡)就读 IT 大学。足以介绍:D。 基本上,我有一家提供簿记应用
我有 public JSONObject parseXML(String xml) { JSONObject jsonObject = XML.toJSONObject(xml); r
我已经在 Java 中实现了带有动态类型的简单解释语言。不幸的是我遇到了以下问题。测试时如下代码: def main() { def ks = Map[[1, 2]].keySet()
一直提示输入 1 到 10 的数字 - 结果应将 st、rd、th 和 nd 添加到数字中。编写一个程序,提示用户输入 1 到 10 之间的任意整数,然后以序数形式显示该整数并附加后缀。 public
我有这个 DownloadFile.java 并按预期下载该文件: import java.io.*; import java.net.URL; public class DownloadFile {
我想在 GUI 上添加延迟。我放置了 2 个 for 循环,然后重新绘制了一个标签,但这 2 个 for 循环一个接一个地执行,并且标签被重新绘制到最后一个。 我能做什么? for(int i=0;
我正在对对象 Student 的列表项进行一些测试,但是我更喜欢在 java 类对象中创建硬编码列表,然后从那里提取数据,而不是连接到数据库并在结果集中选择记录。然而,自从我这样做以来已经很长时间了,
我知道对象创建分为三个部分: 声明 实例化 初始化 classA{} classB extends classA{} classA obj = new classB(1,1); 实例化 它必须使用
我有兴趣使用 GPRS 构建车辆跟踪系统。但是,我有一些问题要问以前做过此操作的人: GPRS 是最好的技术吗?人们意识到任何问题吗? 我计划使用 Java/Java EE - 有更好的技术吗? 如果
我可以通过递归方法反转数组,例如:数组={1,2,3,4,5} 数组结果={5,4,3,2,1}但我的结果是相同的数组,我不知道为什么,请帮助我。 public class Recursion { p
有这样的标准方式吗? 包括 Java源代码-测试代码- Ant 或 Maven联合单元持续集成(可能是巡航控制)ClearCase 版本控制工具部署到应用服务器 最后我希望有一个自动构建和集成环境。
我什至不知道这是否可能,我非常怀疑它是否可能,但如果可以,您能告诉我怎么做吗?我只是想知道如何从打印机打印一些文本。 有什么想法吗? 最佳答案 这里有更简单的事情。 import javax.swin
我是一名优秀的程序员,十分优秀!