2015-01-17 21:51:04 +01:00
|
|
|
#include "rwtest.h"
|
|
|
|
|
|
|
|
using namespace std;
|
|
|
|
|
|
|
|
int screenWidth = 640, screenHeight= 480;
|
|
|
|
bool running = true;
|
|
|
|
Camera *camera;
|
|
|
|
|
|
|
|
GLint program;
|
|
|
|
GLuint vbo;
|
|
|
|
|
2015-01-25 22:27:03 +01:00
|
|
|
rw::Clump *clump;
|
2015-01-17 21:51:04 +01:00
|
|
|
|
|
|
|
char *filename;
|
|
|
|
|
|
|
|
void
|
2015-01-25 22:27:03 +01:00
|
|
|
renderAtomic(rw::Atomic *atomic)
|
2015-01-17 21:51:04 +01:00
|
|
|
{
|
2015-01-25 22:27:03 +01:00
|
|
|
using namespace rw;
|
2015-01-17 21:51:04 +01:00
|
|
|
static GLenum prim[] = {
|
|
|
|
GL_TRIANGLES, GL_TRIANGLE_STRIP
|
|
|
|
};
|
|
|
|
Geometry *geo = atomic->geometry;
|
2015-08-03 18:30:10 +02:00
|
|
|
atomic->getPipeline()->instance(atomic);
|
2015-01-25 22:27:03 +01:00
|
|
|
gl::InstanceDataHeader *inst = (gl::InstanceDataHeader*)geo->instData;
|
2015-01-17 21:51:04 +01:00
|
|
|
MeshHeader *meshHeader = geo->meshHeader;
|
|
|
|
|
|
|
|
Frame *frm = atomic->frame;
|
|
|
|
frm->updateLTM();
|
|
|
|
glUniformMatrix4fv(glGetUniformLocation(program, "worldMat"),
|
|
|
|
1, GL_FALSE, frm->ltm);
|
|
|
|
|
2015-01-20 15:25:53 +01:00
|
|
|
glVertexAttrib4f(3, 0.0f, 0.0f, 0.0f, 1.0f);
|
2015-01-22 08:40:44 +01:00
|
|
|
glVertexAttrib3f(2, 0.0f, 0.0f, 0.0f);
|
2015-01-17 21:51:04 +01:00
|
|
|
if(inst->vbo == 0 && inst->ibo == 0)
|
2015-01-25 22:27:03 +01:00
|
|
|
gl::uploadGeo(geo);
|
2015-01-17 21:51:04 +01:00
|
|
|
glBindBuffer(GL_ARRAY_BUFFER, inst->vbo);
|
|
|
|
glBindBuffer(GL_ELEMENT_ARRAY_BUFFER, inst->ibo);
|
2015-01-25 22:27:03 +01:00
|
|
|
gl::setAttribPointers(inst);
|
2015-01-17 21:51:04 +01:00
|
|
|
|
|
|
|
uint64 offset = 0;
|
|
|
|
for(uint32 i = 0; i < meshHeader->numMeshes; i++){
|
|
|
|
Mesh *mesh = &meshHeader->mesh[i];
|
2015-07-11 23:48:11 +02:00
|
|
|
Material *mat = mesh->material;
|
2015-01-20 15:25:53 +01:00
|
|
|
float color[4];
|
2015-07-11 23:48:11 +02:00
|
|
|
uint8 *col = mat->color;
|
2015-01-20 15:25:53 +01:00
|
|
|
color[0] = col[0] / 255.0f;
|
|
|
|
color[1] = col[1] / 255.0f;
|
|
|
|
color[2] = col[2] / 255.0f;
|
|
|
|
color[3] = col[3] / 255.0f;
|
|
|
|
glUniform4fv(glGetUniformLocation(program, "matColor"),
|
|
|
|
1, color);
|
2015-07-11 23:48:11 +02:00
|
|
|
rw::gl::Texture *tex =(rw::gl::Texture*)mat->texture;
|
2015-01-20 19:22:57 +01:00
|
|
|
if(tex)
|
|
|
|
tex->bind(0);
|
|
|
|
else
|
2015-01-20 14:48:18 +01:00
|
|
|
glBindTexture(GL_TEXTURE_2D, 0);
|
2015-01-17 21:51:04 +01:00
|
|
|
glDrawElements(prim[meshHeader->flags], mesh->numIndices,
|
|
|
|
GL_UNSIGNED_SHORT, (void*)offset);
|
|
|
|
offset += mesh->numIndices*2;
|
|
|
|
}
|
|
|
|
glDisableVertexAttribArray(0);
|
|
|
|
glDisableVertexAttribArray(1);
|
|
|
|
glDisableVertexAttribArray(2);
|
|
|
|
glDisableVertexAttribArray(3);
|
|
|
|
glBindBuffer(GL_ARRAY_BUFFER, 0);
|
|
|
|
glBindBuffer(GL_ELEMENT_ARRAY_BUFFER, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
render(void)
|
|
|
|
{
|
2015-01-20 19:22:57 +01:00
|
|
|
static Mat4 worldMat(1.0f);
|
2015-01-17 21:51:04 +01:00
|
|
|
glUseProgram(program);
|
|
|
|
glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT);
|
|
|
|
|
|
|
|
camera->look();
|
|
|
|
glUniformMatrix4fv(glGetUniformLocation(program, "projMat"),
|
|
|
|
1, GL_FALSE, camera->projMat.cr);
|
|
|
|
glUniformMatrix4fv(glGetUniformLocation(program, "viewMat"),
|
|
|
|
1, GL_FALSE, camera->viewMat.cr);
|
2015-01-20 19:22:57 +01:00
|
|
|
glUniformMatrix4fv(glGetUniformLocation(program, "worldMat"),
|
|
|
|
1, GL_FALSE, worldMat.cr);
|
2015-01-17 21:51:04 +01:00
|
|
|
|
|
|
|
glVertexAttrib3f(2, -0.5f, 0.5f, 0.70710f);
|
|
|
|
glBindBuffer(GL_ARRAY_BUFFER, vbo);
|
|
|
|
glEnableVertexAttribArray(0);
|
|
|
|
glEnableVertexAttribArray(3);
|
|
|
|
glVertexAttribPointer(0, 3, GL_FLOAT, GL_FALSE, 28, (GLvoid*)0);
|
|
|
|
glVertexAttribPointer(3, 3, GL_FLOAT, GL_FALSE, 28, (GLvoid*)12);
|
|
|
|
glDrawArrays(GL_LINES, 0, 6);
|
|
|
|
glDisableVertexAttribArray(0);
|
|
|
|
glDisableVertexAttribArray(3);
|
|
|
|
|
2015-01-25 22:27:03 +01:00
|
|
|
for(rw::uint32 i = 0; i < clump->numAtomics; i++){
|
2015-01-17 21:51:04 +01:00
|
|
|
char *name = PLUGINOFFSET(char, clump->atomicList[i]->frame,
|
2015-01-25 22:27:03 +01:00
|
|
|
gta::nodeNameOffset);
|
2015-01-17 21:51:04 +01:00
|
|
|
if(strstr(name, "_dam") || strstr(name, "_vlo"))
|
|
|
|
continue;
|
|
|
|
renderAtomic(clump->atomicList[i]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
init(void)
|
|
|
|
{
|
|
|
|
glClearColor(0.3f, 0.3f, 0.3f, 1.0f);
|
|
|
|
glEnable(GL_DEPTH_TEST);
|
2015-01-20 15:25:53 +01:00
|
|
|
glEnable(GL_BLEND);
|
|
|
|
glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA);
|
|
|
|
glEnable(GL_ALPHA_TEST);
|
|
|
|
glAlphaFunc(GL_GEQUAL, 0.5f);
|
2015-01-17 21:51:04 +01:00
|
|
|
const char *shadersrc =
|
|
|
|
"#ifdef VERTEX\n"
|
|
|
|
"uniform mat4 projMat;"
|
|
|
|
"uniform mat4 viewMat;"
|
|
|
|
"uniform mat4 worldMat;"
|
2015-01-20 15:25:53 +01:00
|
|
|
"uniform vec4 matColor;"
|
2015-01-17 21:51:04 +01:00
|
|
|
"attribute vec3 in_vertex;"
|
2015-01-20 14:48:18 +01:00
|
|
|
"attribute vec2 in_texCoord;"
|
2015-01-17 21:51:04 +01:00
|
|
|
"attribute vec3 in_normal;"
|
|
|
|
"attribute vec4 in_color;"
|
|
|
|
"varying vec4 v_color;"
|
2015-01-20 14:48:18 +01:00
|
|
|
"varying vec2 v_texCoord;"
|
2015-01-17 21:51:04 +01:00
|
|
|
"vec3 lightdir = vec3(0.5, -0.5, -0.70710);"
|
2015-01-20 15:25:53 +01:00
|
|
|
"vec4 amblight = vec4(20, 20, 20, 0)/255;"
|
2015-01-17 21:51:04 +01:00
|
|
|
"void main()"
|
|
|
|
"{"
|
|
|
|
" gl_Position = projMat * viewMat * worldMat * vec4(in_vertex, 1.0);"
|
2015-08-27 16:45:54 +02:00
|
|
|
|
|
|
|
" vec3 N = normalize(mat3(viewMat * worldMat) * in_normal);"
|
|
|
|
" vec3 P = vec3(viewMat * worldMat * vec4(in_vertex, 1.0));"
|
|
|
|
" vec3 V = normalize(-P);"
|
|
|
|
" float rim = 1.0 - max(dot(V, N), 0.0);"
|
|
|
|
|
2015-01-17 21:51:04 +01:00
|
|
|
" vec3 n = mat3(worldMat) * in_normal;"
|
|
|
|
" float l = max(0.0, dot(n, -lightdir));"
|
2015-08-27 16:45:54 +02:00
|
|
|
// " l = l*0.4 + rim;"
|
2015-01-20 15:25:53 +01:00
|
|
|
" v_color = (in_color+vec4(l,l,l,0)+amblight)*matColor;"
|
2015-01-20 14:48:18 +01:00
|
|
|
" v_texCoord = in_texCoord;"
|
2015-01-17 21:51:04 +01:00
|
|
|
"}\n"
|
|
|
|
"#endif\n"
|
|
|
|
"#ifdef FRAGMENT\n"
|
2015-01-20 14:48:18 +01:00
|
|
|
"uniform sampler2D u_texture0;"
|
2015-01-17 21:51:04 +01:00
|
|
|
"varying vec4 v_color;"
|
2015-01-20 14:48:18 +01:00
|
|
|
"varying vec2 v_texCoord;"
|
2015-01-17 21:51:04 +01:00
|
|
|
"void main()"
|
|
|
|
"{"
|
2015-01-22 08:40:44 +01:00
|
|
|
" vec4 c0 = texture2D(u_texture0, v_texCoord/512.0);"
|
2015-01-20 14:48:18 +01:00
|
|
|
" gl_FragColor = v_color*c0;"
|
2015-01-17 21:51:04 +01:00
|
|
|
"}\n"
|
|
|
|
"#endif\n";
|
2015-06-10 22:29:48 +02:00
|
|
|
const char *srcarr[] = { "#version 120\n",
|
|
|
|
"#define VERTEX\n", shadersrc };
|
|
|
|
GLint vertshader = rw::gl::compileShader(srcarr, 3, GL_VERTEX_SHADER);
|
2015-01-17 21:51:04 +01:00
|
|
|
assert(vertshader != 0);
|
2015-06-10 22:29:48 +02:00
|
|
|
srcarr[1] = "#define FRAGMENT\n";
|
|
|
|
GLint fragshader = rw::gl::compileShader(srcarr, 3, GL_FRAGMENT_SHADER);
|
2015-01-17 21:51:04 +01:00
|
|
|
assert(fragshader != 0);
|
2015-01-25 22:27:03 +01:00
|
|
|
program = rw::gl::linkProgram(vertshader, fragshader);
|
2015-01-17 21:51:04 +01:00
|
|
|
assert(program != 0);
|
|
|
|
|
|
|
|
GLfloat vertarray[] = {
|
|
|
|
0.0f, 0.0f, 0.0f,
|
|
|
|
1.0f, 0.0f, 0.0f, 1.0f,
|
|
|
|
1.0f, 0.0f, 0.0f,
|
|
|
|
1.0f, 0.0f, 0.0f, 1.0f,
|
|
|
|
|
|
|
|
0.0f, 0.0f, 0.0f,
|
|
|
|
0.0f, 1.0f, 0.0f, 1.0f,
|
|
|
|
0.0f, 1.0f, 0.0f,
|
|
|
|
0.0f, 1.0f, 0.0f, 1.0f,
|
|
|
|
|
|
|
|
0.0f, 0.0f, 0.0f,
|
|
|
|
0.0f, 0.0f, 1.0f, 1.0f,
|
|
|
|
0.0f, 0.0f, 1.0f,
|
|
|
|
0.0f, 0.0f, 1.0f, 1.0f,
|
|
|
|
};
|
|
|
|
glGenBuffers(1, &vbo);
|
|
|
|
glBindBuffer(GL_ARRAY_BUFFER, vbo);
|
|
|
|
glBufferData(GL_ARRAY_BUFFER, sizeof(vertarray),
|
|
|
|
vertarray, GL_STATIC_DRAW);
|
|
|
|
glBindBuffer(GL_ARRAY_BUFFER, 0);
|
|
|
|
|
|
|
|
camera = new Camera;
|
|
|
|
camera->setAspectRatio(1.0f*screenWidth/screenHeight);
|
|
|
|
camera->setNearFar(0.1f, 450.0f);
|
|
|
|
camera->setTarget(Vec3(0.0f, 0.0f, 0.0f));
|
2015-08-27 16:45:54 +02:00
|
|
|
// camera->setPosition(Vec3(0.0f, 5.0f, 0.0f));
|
|
|
|
camera->setPosition(Vec3(0.0f, -1.0f, 3.0f));
|
2015-01-17 21:51:04 +01:00
|
|
|
|
2015-09-09 23:26:16 +02:00
|
|
|
gta::attachPlugins();
|
|
|
|
|
2015-01-25 22:27:03 +01:00
|
|
|
rw::currentTexDictionary = new rw::TexDictionary;
|
|
|
|
// rw::Image::setSearchPath("/home/aap/gamedata/ps2/gtasa/models/gta3_archive/txd_extracted/");
|
|
|
|
// rw::Image::setSearchPath("/home/aap/gamedata/ps2/gtavc/MODELS/gta3_archive/txd_extracted/");
|
|
|
|
rw::Image::setSearchPath(
|
2015-01-22 08:40:44 +01:00
|
|
|
"/home/aap/gamedata/ps2/gta3/MODELS/gta3_archive/txd_extracted/;//home/aap/gamedata/ps2/gtavc/MODELS/gta3_archive/txd_extracted/;/home/aap/gamedata/ps2/gtasa/models/gta3_archive/txd_extracted/");
|
2015-01-20 21:49:57 +01:00
|
|
|
//"D:\\rockstargames\\ps2\\gtavc\\MODELS\\gta3_archive\\txd_extracted\\;D:\\rockstargames\\ps2\\gtasa\\models\\gta3_archive\\txd_extracted\\");
|
2015-01-26 20:23:45 +01:00
|
|
|
|
2015-08-03 18:30:10 +02:00
|
|
|
printf("platform: %d\n", rw::platform);
|
2015-01-25 22:27:03 +01:00
|
|
|
|
|
|
|
rw::StreamFile in;
|
2015-01-20 21:49:57 +01:00
|
|
|
if(in.open(filename, "rb") == NULL)
|
|
|
|
printf("couldn't open file\n");
|
2015-01-25 22:27:03 +01:00
|
|
|
rw::findChunk(&in, rw::ID_CLUMP, NULL, NULL);
|
|
|
|
clump = rw::Clump::streamRead(&in);
|
2015-01-17 21:51:04 +01:00
|
|
|
assert(clump);
|
|
|
|
in.close();
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
keypress(GLFWwindow *w, int key, int scancode, int action, int mods)
|
|
|
|
{
|
|
|
|
if(action != GLFW_PRESS)
|
|
|
|
return;
|
|
|
|
|
|
|
|
switch(key){
|
|
|
|
case 'Q':
|
|
|
|
case GLFW_KEY_ESCAPE:
|
|
|
|
running = false;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static int lastX, lastY;
|
|
|
|
static int clickX, clickY;
|
|
|
|
static bool isLDown, isMDown, isRDown;
|
|
|
|
static bool isShiftDown, isCtrlDown, isAltDown;
|
|
|
|
|
|
|
|
void
|
|
|
|
mouseButton(GLFWwindow *w, int button, int action, int mods)
|
|
|
|
{
|
|
|
|
double x, y;
|
|
|
|
glfwGetCursorPos(w, &x, &y);
|
|
|
|
if(action == GLFW_PRESS){
|
|
|
|
lastX = clickX = x;
|
|
|
|
lastY = clickY = y;
|
|
|
|
if(button == GLFW_MOUSE_BUTTON_LEFT)
|
|
|
|
isLDown = true;
|
|
|
|
if(button == GLFW_MOUSE_BUTTON_MIDDLE)
|
|
|
|
isMDown = true;
|
|
|
|
if(button == GLFW_MOUSE_BUTTON_RIGHT)
|
|
|
|
isRDown = true;
|
|
|
|
}else if(action == GLFW_RELEASE){
|
|
|
|
if(button == GLFW_MOUSE_BUTTON_LEFT)
|
|
|
|
isLDown = false;
|
|
|
|
if(button == GLFW_MOUSE_BUTTON_MIDDLE)
|
|
|
|
isMDown = false;
|
|
|
|
if(button == GLFW_MOUSE_BUTTON_RIGHT)
|
|
|
|
isRDown = false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
mouseMotion(GLFWwindow *w, double x, double y)
|
|
|
|
{
|
|
|
|
GLfloat dx, dy;
|
|
|
|
static int xoff = 0, yoff = 0;
|
|
|
|
static bool wrappedLast = false;
|
|
|
|
int width, height;
|
|
|
|
|
|
|
|
glfwGetWindowSize(w, &width, &height);
|
|
|
|
|
|
|
|
dx = float(lastX - x) / float(width);
|
|
|
|
dy = float(lastY - y) / float(height);
|
|
|
|
/* Wrap the mouse if it goes over the window border.
|
|
|
|
* Unfortunately, after glfwSetMousePos is done, there can be old
|
|
|
|
* events with an old mouse position,
|
|
|
|
* hence the check if the pointer was wrapped the last time. */
|
|
|
|
if((isLDown || isMDown || isRDown) &&
|
|
|
|
(x < 0 || y < 0 || x >= width || y >= height)){
|
|
|
|
if(wrappedLast){
|
|
|
|
dx = float(lastX-xoff - x) / float(width);
|
|
|
|
dy = float(lastY-yoff - y) / float(height);
|
|
|
|
}
|
|
|
|
xoff = yoff = 0;
|
|
|
|
while (x+xoff >= width) xoff -= width;
|
|
|
|
while (y+yoff >= height) yoff -= height;
|
|
|
|
while (x+xoff < 0) xoff += width;
|
|
|
|
while (y+yoff < 0) yoff += height;
|
|
|
|
glfwSetCursorPos(w, x+xoff, y+yoff);
|
|
|
|
wrappedLast = true;
|
|
|
|
}else{
|
|
|
|
wrappedLast = false;
|
|
|
|
xoff = yoff = 0;
|
|
|
|
}
|
|
|
|
lastX = x+xoff;
|
|
|
|
lastY = y+yoff;
|
|
|
|
if(isLDown){
|
|
|
|
if(isShiftDown)
|
|
|
|
camera->turn(dx*2.0f, dy*2.0f);
|
|
|
|
else
|
|
|
|
camera->orbit(dx*2.0f, -dy*2.0f);
|
|
|
|
}
|
|
|
|
if(isMDown){
|
|
|
|
if(isShiftDown)
|
|
|
|
;
|
|
|
|
else
|
|
|
|
camera->pan(dx*8.0f, -dy*8.0f);
|
|
|
|
}
|
|
|
|
if(isRDown){
|
|
|
|
if(isShiftDown)
|
|
|
|
;
|
|
|
|
else
|
|
|
|
camera->zoom(dx*12.0f);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
resize(GLFWwindow*, int width, int height)
|
|
|
|
{
|
|
|
|
screenWidth = width;
|
|
|
|
screenHeight = height;
|
|
|
|
glViewport(0, 0, screenWidth, screenHeight);
|
|
|
|
camera->setAspectRatio(1.0f*screenWidth/screenHeight);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
closewindow(GLFWwindow*)
|
|
|
|
{
|
|
|
|
running = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
int
|
|
|
|
main(int argc, char *argv[])
|
|
|
|
{
|
|
|
|
if(argc < 2)
|
|
|
|
return 1;
|
|
|
|
filename = argv[1];
|
|
|
|
|
|
|
|
if(!glfwInit()){
|
|
|
|
fprintf(stderr, "Error: could not initialize GLFW\n");
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
GLFWwindow *window = glfwCreateWindow(screenWidth, screenHeight,
|
|
|
|
"OpenGL", 0, 0);
|
|
|
|
if(!window){
|
|
|
|
fprintf(stderr, "Error: could not create GLFW window\n");
|
|
|
|
glfwTerminate();
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
glfwMakeContextCurrent(window);
|
|
|
|
GLenum status = glewInit();
|
|
|
|
if(status != GLEW_OK){
|
|
|
|
fprintf(stderr, "Error: %s\n", glewGetErrorString(status));
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
if(!GLEW_VERSION_2_0){
|
|
|
|
fprintf(stderr, "Error: OpenGL 2.0 needed\n");
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
init();
|
|
|
|
|
|
|
|
glfwSetWindowSizeCallback(window, resize);
|
|
|
|
glfwSetWindowCloseCallback(window, closewindow);
|
|
|
|
glfwSetMouseButtonCallback(window, mouseButton);
|
|
|
|
glfwSetCursorPosCallback(window, mouseMotion);
|
|
|
|
glfwSetKeyCallback(window, keypress);
|
|
|
|
while(running){
|
|
|
|
glfwPollEvents();
|
|
|
|
isShiftDown = glfwGetKey(window, GLFW_KEY_LEFT_SHIFT) == GLFW_PRESS;
|
|
|
|
isCtrlDown = glfwGetKey(window, GLFW_KEY_LEFT_CONTROL) == GLFW_PRESS;
|
|
|
|
isAltDown = glfwGetKey(window, GLFW_KEY_LEFT_ALT) == GLFW_PRESS;
|
|
|
|
|
|
|
|
render();
|
|
|
|
glfwSwapBuffers(window);
|
|
|
|
}
|
|
|
|
|
|
|
|
glfwTerminate();
|
|
|
|
return 0;
|
|
|
|
}
|