目前,我试图在LWJGL中创建一个Camera类,但我一直遇到投影矩阵的问题。出于某种原因,当我尝试将顶点乘以投影矩阵时,屏幕上什么都没有出现。
相机课程
public class Camera {
private Vector3f position, rotation;
private Matrix4f view;
private final Vector3f xAxis, yAxis, zAxis;
private float fov, aspect, zNear, zFar;
private Matrix4f projection;
public Camera(float fov, float aspect, float zNear, float zFar){
this.fov = fov;
this.aspect = aspect;
this.zNear = zNear;
this.zFar = zFar;
projection = createPerspectiveProjection(fov, aspect, zNear, zFar);
position = new Vector3f();
rotation = new Vector3f();
view = new Matrix4f();
view.setIdentity();
xAxis = new Vector3f(1, 0, 0);
yAxis = new Vector3f(0, 1, 0);
zAxis = new Vector3f(0, 0, 1);
}
public void addRotation(float x, float y, float z){
rotation.x += x;
rotation.y += y;
rotation.z += z;
apply();
}
public void move(float x, float y, float z){
position.x += x;
position.y += y;
position.z += z;
apply();
}
public Matrix4f getView(){
return view;
}
public Matrix4f getProjection(){
return projection;
}
private void apply(){
view.setIdentity();
view.rotate(rotation.x, xAxis);
view.rotate(rotation.y, yAxis);
view.rotate(rotation.z, zAxis);
view.translate(position);
}
private Matrix4f createPerspectiveProjection(float fov, float aspect, float zNear, float zFar){
Matrix4f mat = new Matrix4f();
float yScale = (float) (1 / (Math.tan(Math.toRadians(fov / 2))));
float xScale = yScale / aspect;
float frustrumLength = zFar - zNear;
mat.m00 = xScale;
mat.m11 = yScale;
mat.m22 = -((zFar + zNear) / frustrumLength);
mat.m23 = -1;
mat.m32 = -((2 * zFar * zNear) / frustrumLength);
mat.m33 = 0;
return mat;
}
}
主要课程
public class Game implements Runnable {
public static final int WIDTH = 800;
public static final int HEIGHT = 600;
public static final DisplayMode dm = new DisplayMode(WIDTH, HEIGHT);
int vaoID;
ShaderProgram program;
Camera camera;
Model model;
public Game(){
new Thread(this).start();
}
public void run(){
init();
while(true){
if(Display.isCloseRequested())
break;
update(Timer.getElapsedTime());
render();
Display.sync(60);
Display.update();
}
Display.destroy();
}
public void init(){
try{
Display.setTitle("Ludum Dare!");
Display.setDisplayMode(dm);
Display.create();
}catch(LWJGLException e){
e.printStackTrace();
System.exit(1);
}
Timer.start();
camera = new Camera(60.0f, WIDTH / HEIGHT, 0.1f, 100.0f);
program = new ShaderProgram("res/shader/defaultshader.vert", "res/shader/defaultshader.frag");
glClearColor(0.0f, 0.0f, 0.4f, 0.0f);
vaoID = glGenVertexArrays();
glBindVertexArray(vaoID);
model = new Model(new float[] {
-1, -1, 1,
1, -1, 1,
0, 1, 1
});
}
public void update(float delta){
}
public void render(){
glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT);
program.bind();
program.setUniform("model_matrix", model.getModel());
program.setUniform("view_matrix", camera.getView());
program.setUniform("projection_matrix", camera.getProjection());
model.render();
program.unbind();
}
public static void main(String[] args){
new Game();
}
}
顶点着色器
#version 330 core
layout(location = 0) in vec3 vertex_modelspace;
uniform mat4 model_matrix;
uniform mat4 view_matrix;
uniform mat4 projection_matrix;
void main(){
mat4 modelviewprojection_matrix = projection_matrix * view_matrix * model_matrix;
vec4 vertex = vec4(vertex_modelspace, 1.0);
gl_Position = modelviewprojection_matrix * vertex;
}
ShaderProgram类
public class ShaderProgram {
private int vertexShaderID, fragmentShaderID;
private int programID;
public ShaderProgram(String vertPath, String fragPath){
programID = glCreateProgram();
vertexShaderID = attachShader(vertPath, GL_VERTEX_SHADER);
fragmentShaderID = attachShader(fragPath, GL_FRAGMENT_SHADER);
link();
}
private int attachShader(String path, int type){
StringBuilder shaderSource = new StringBuilder();
try{
BufferedReader reader = new BufferedReader(new FileReader(new File(path)));
String line;
while((line = reader.readLine()) != null){
shaderSource.append(line).append("\n");
}
reader.close();
}catch(IOException e){
e.printStackTrace();
System.out.println("Error reading from shader " + path);
System.exit(1);
}
System.out.println("Compiling shader " + path);
int id = glCreateShader(type);
glShaderSource(id, shaderSource);
glCompileShader(id);
if(glGetShaderi(id, GL_COMPILE_STATUS) == GL_FALSE){
System.out.println(glGetShaderInfoLog(id, 1000));
System.exit(1);
}
return id;
}
private void link(){
System.out.println("Linking program...");
glAttachShader(programID, vertexShaderID);
glAttachShader(programID, fragmentShaderID);
glLinkProgram(programID);
}
public void bind(){
glUseProgram(programID);
}
public void unbind(){
glUseProgram(0);
}
public void setUniform(String name, Matrix4f value){
FloatBuffer matrix = BufferUtils.createFloatBuffer(16);
value.store(matrix); matrix.flip();
glUniformMatrix4(glGetUniformLocation(programID, name), false, matrix);
}
}
当我运行它时,我创建的VBO(一个三角形)不会出现,但是当我将投影矩阵从顶点着色器中的乘法中移出时,它运行得很好。我错过了什么吗?
答案 0 :(得分:3)
我假设model.getModel()
将返回单位矩阵(就像您的视图矩阵是标识一样)。在这种情况下,您有以下情况:在平面z = 1.0中绘制一个三角形。如果你也使用identity作为投影矩阵,你可以直接在剪辑空间中绘制,而trianlge将在far平面上,因此它是可见的。
但是,您的createPerspectiveProjection
函数似乎是考虑到标准的OpenGL约定,因此它几乎与glFrustum()
一样。 (您的代码缺少该联机帮助页中标记为A和B的部分,因此您只能使用对称的平截头体,但在大多数情况下这样做很好。)
用于此矩阵的约定是摄像机正在查看-z方向,而zNear
和zFar
参数实际上是maped,以便z_eye=-zNear
处的点投射到近处平面(z_ndc=-1
),z_eye=-zFar
处的点投射到远平面(z_ndc=1
)。因此,如果应用投影矩阵,z = 1处的三角形就在摄像机后面。