/* San Angeles Observation OpenGL ES version example
* Copyright 2004-2005 Jetro Lauha
* All rights reserved.
* Web: http://iki.fi/jetro/
*
* This source is free software; you can redistribute it and/or
* modify it under the terms of EITHER:
* (1) The GNU Lesser General Public License as published by the Free
* Software Foundation; either version 2.1 of the License, or (at
* your option) any later version. The text of the GNU Lesser
* General Public License is included with this source in the
* file LICENSE-LGPL.txt.
* (2) The BSD-style license that is included with this source in
* the file LICENSE-BSD.txt.
*
* This source is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the files
* LICENSE-LGPL.txt and LICENSE-BSD.txt for more details.
*
* $Id$
* $Revision$
*/
package demos.es1.angeles;
import com.jogamp.common.nio.Buffers;
import com.jogamp.opengl.*;
import com.jogamp.opengl.glu.*;
import com.jogamp.opengl.math.FixedPoint;
import com.jogamp.opengl.util.*;
import java.nio.*;
public class AngelesES1 implements GLEventListener {
public AngelesES1() {
this(true);
}
public AngelesES1(boolean enableBlending) {
blendingEnabled = enableBlending;
quadVertices = Buffers.newDirectIntBuffer(12);
quadVertices.put(new int[]{
-0x10000, -0x10000,
0x10000, -0x10000,
-0x10000, 0x10000,
0x10000, -0x10000,
0x10000, 0x10000,
-0x10000, 0x10000
});
quadVertices.flip();
light0Position=Buffers.newDirectIntBuffer(4);
light0Diffuse=Buffers.newDirectIntBuffer(4);
light1Position=Buffers.newDirectIntBuffer(4);
light1Diffuse=Buffers.newDirectIntBuffer(4);
light2Position=Buffers.newDirectIntBuffer(4);
light2Diffuse=Buffers.newDirectIntBuffer(4);
materialSpecular=Buffers.newDirectIntBuffer(4);
light0Position.put(new int[] { -0x40000, 0x10000, 0x10000, 0 });
light0Diffuse.put(new int[] { 0x10000, 0x6666, 0, 0x10000 });
light1Position.put(new int[] { 0x10000, -0x20000, -0x10000, 0 });
light1Diffuse.put(new int[] { 0x11eb, 0x23d7, 0x5999, 0x10000 });
light2Position.put(new int[] { -0x10000, 0, -0x40000, 0 });
light2Diffuse.put(new int[] { 0x11eb, 0x2b85, 0x23d7, 0x10000 });
materialSpecular.put(new int[] { 0x10000, 0x10000, 0x10000, 0x10000 });
light0Position.flip();
light0Diffuse.flip();
light1Position.flip();
light1Diffuse.flip();
light2Position.flip();
light2Diffuse.flip();
materialSpecular.flip();
seedRandom(15);
width=0;
height=0;
x=0;
y=0;
}
public void init(GLAutoDrawable drawable) {
// FIXME: gl.setSwapInterval(1);
this.gl = drawable.getGL().getGLES1();
this.glu = GLU.createGLU();
gl.glEnable(gl.GL_NORMALIZE);
gl.glEnable(gl.GL_DEPTH_TEST);
gl.glDisable(gl.GL_CULL_FACE);
gl.glShadeModel(gl.GL_FLAT);
gl.glEnable(gl.GL_LIGHTING);
gl.glEnable(gl.GL_LIGHT0);
gl.glEnable(gl.GL_LIGHT1);
gl.glEnable(gl.GL_LIGHT2);
gl.glEnableClientState(gl.GL_VERTEX_ARRAY);
gl.glEnableClientState(gl.GL_COLOR_ARRAY);
for (int a = 0; a < SuperShape.COUNT; ++a)
{
sSuperShapeObjects[a] = createSuperShape(SuperShape.sParams[a]);
}
sGroundPlane = createGroundPlane();
gAppAlive = 1;
sStartTick = System.currentTimeMillis();
frames=0;
}
public void reshape(GLAutoDrawable drawable, int x, int y, int width, int height) {
this.width = width;
this.height=height;
this.x = x;
this.y = y;
this.gl = drawable.getGL().getGLES1();
gl.glMatrixMode(gl.GL_MODELVIEW);
gl.glLoadIdentity();
gl.glClearColorx((int)(0.1f * 65536),
(int)(0.2f * 65536),
(int)(0.3f * 65536), 0x10000);
gl.glCullFace(GL.GL_FRONT);
gl.glHint(gl.GL_PERSPECTIVE_CORRECTION_HINT, GL.GL_FASTEST);
//gl.glShadeModel(GL2ES1.GL_SMOOTH);
gl.glShadeModel(gl.GL_FLAT);
gl.glDisable(gl.GL_DITHER);
//gl.glMatrixMode(gl.GL_PROJECTION);
//gl.glLoadIdentity();
//glu.gluPerspective(45.0f, (float)width / (float)height, 0.5f, 150.0f);
System.out.println("reshape ..");
}
public void dispose(GLAutoDrawable drawable) {
}
public void display(GLAutoDrawable drawable) {
long tick = System.currentTimeMillis();
if (gAppAlive==0)
return;
this.gl = drawable.getGL().getGLES1();
// Actual tick value is "blurred" a little bit.
sTick = (sTick + tick - sStartTick) >> 1;
// Terminate application after running through the demonstration once.
if (sTick >= RUN_LENGTH)
{
gAppAlive = 0;
return;
}
gl.glClear(gl.GL_DEPTH_BUFFER_BIT | gl.GL_COLOR_BUFFER_BIT);
gl.glMatrixMode(gl.GL_PROJECTION);
gl.glLoadIdentity();
glu.gluPerspective(45.0f, (float)width / (float)height, 0.5f, 150.0f);
// Update the camera position and set the lookat.
camTrack();
// Configure environment.
configureLightAndMaterial();
if(blendingEnabled) {
// Draw the reflection by drawing models with negated Z-axis.
gl.glPushMatrix();
drawModels(-1);
gl.glPopMatrix();
}
// Draw the ground plane to the window. (opt. blending)
drawGroundPlane();
// Draw all the models normally.
drawModels(1);
if(blendingEnabled) {
// Draw fade quad over whole window (when changing cameras).
drawFadeQuad();
}
frames++;
tick = System.currentTimeMillis();
long dT = tick - sStartTick;
// System.out.println(frames+"f, "+dT+"ms "+ (frames*1000)/dT +"fps");
}
public void displayChanged(GLAutoDrawable drawable, boolean modeChanged, boolean deviceChanged) {
}
private boolean blendingEnabled = true;
private GLES1 gl;
private GLU glu;
// Total run length is 20 * camera track base unit length (see cams.h).
private int RUN_LENGTH = (20 * CamTrack.CAMTRACK_LEN) ;
private int RANDOM_UINT_MAX = 65535 ;
private long sRandomSeed = 0;
void seedRandom(long seed)
{
sRandomSeed = seed;
}
int randomUInt()
{
sRandomSeed = sRandomSeed * 0x343fd + 0x269ec3;
return Math.abs((int) (sRandomSeed >> 16));
}
// Definition of one GL object in this demo.
public class GLObject {
/* Vertex array and color array are enabled for all objects, so their
* pointers must always be valid and non-null. Normal array is not
* used by the ground plane, so when its pointer is null then normal
* array usage is disabled.
*
* Vertex array is supposed to use gl.GL_FIXED datatype and stride 0
* (i.e. tightly packed array). Color array is supposed to have 4
* components per color with gl.GL_UNSIGNED_BYTE datatype and stride 0.
* Normal array is supposed to use gl.GL_FIXED datatype and stride 0.
*/
IntBuffer vertexArray;
ByteBuffer colorArray;
IntBuffer normalArray;
int vertexComponents;
int count;
int vbo[];
public GLObject(int vertices, int vertexComponents,
boolean useNormalArray) {
this.count = vertices;
this.vertexComponents = vertexComponents;
this.vertexArray = Buffers.newDirectIntBuffer( vertices * vertexComponents );
this.colorArray = Buffers.newDirectByteBuffer (vertices * 4 );
if (useNormalArray)
{
this.normalArray = Buffers.newDirectIntBuffer (vertices * 3 );
} else {
this.normalArray = null;
}
}
void seal()
{
flip();
vbo = new int[3];
gl.glGenBuffers(3, vbo, 0);
gl.glBindBuffer(GL.GL_ARRAY_BUFFER, vbo[0]);
gl.glBufferData(GL.GL_ARRAY_BUFFER, vertexArray.capacity() * Buffers.SIZEOF_INT, vertexArray, GL.GL_STATIC_DRAW);
gl.glVertexPointer(vertexComponents, gl.GL_FIXED, 0, 0);
gl.glBindBuffer(GL.GL_ARRAY_BUFFER, vbo[1]);
gl.glBufferData(GL.GL_ARRAY_BUFFER, colorArray.capacity() * Buffers.SIZEOF_BYTE, colorArray, GL.GL_STATIC_DRAW);
gl.glColorPointer(4, gl.GL_UNSIGNED_BYTE, 0, 0);
if (null!=normalArray)
{
gl.glEnableClientState(gl.GL_NORMAL_ARRAY);
gl.glBindBuffer(GL.GL_ARRAY_BUFFER, vbo[2]);
gl.glBufferData(GL.GL_ARRAY_BUFFER, normalArray.capacity() * Buffers.SIZEOF_INT, normalArray, GL.GL_STATIC_DRAW);
gl.glNormalPointer(gl.GL_FIXED, 0, 0);
} else {
gl.glDisableClientState(gl.GL_NORMAL_ARRAY);
}
}
void draw()
{
gl.glBindBuffer(GL.GL_ARRAY_BUFFER, vbo[0]);
gl.glVertexPointer(vertexComponents, gl.GL_FIXED, 0, 0);
gl.glBindBuffer(GL.GL_ARRAY_BUFFER, vbo[1]);
gl.glColorPointer(4, gl.GL_UNSIGNED_BYTE, 0, 0);
if (null!=normalArray)
{
gl.glEnableClientState(gl.GL_NORMAL_ARRAY);
gl.glBindBuffer(GL.GL_ARRAY_BUFFER, vbo[2]);
gl.glNormalPointer(gl.GL_FIXED, 0, 0);
}
else
gl.glDisableClientState(gl.GL_NORMAL_ARRAY);
gl.glDrawArrays(gl.GL_TRIANGLES, 0, count);
}
void rewind() {
vertexArray.rewind();
colorArray.rewind();
if (normalArray != null) {
normalArray.rewind();
}
}
void flip() {
vertexArray.flip();
colorArray.flip();
if (normalArray != null) {
normalArray.flip();
}
}
}
long sStartTick = 0;
long sTick = 0;
int sCurrentCamTrack = 0;
long sCurrentCamTrackStartTick = 0;
long sNextCamTrackStartTick = 0x7fffffff;
GLObject sSuperShapeObjects[] = new GLObject[SuperShape.COUNT];
GLObject sGroundPlane;
public class VECTOR3 {
float x, y, z;
public VECTOR3() {
x=0f; y=0f; z=0f;
}
public VECTOR3(float x, float y, float z) {
this.x=x;
this.y=y;
this.z=z;
}
}
static void vector3Sub(VECTOR3 dest, VECTOR3 v1, VECTOR3 v2)
{
dest.x = v1.x - v2.x;
dest.y = v1.y - v2.y;
dest.z = v1.z - v2.z;
}
static void superShapeMap(VECTOR3 point, float r1, float r2, float t, float p)
{
// sphere-mapping of supershape parameters
point.x = (float)(Math.cos(t) * Math.cos(p) / r1 / r2);
point.y = (float)(Math.sin(t) * Math.cos(p) / r1 / r2);
point.z = (float)(Math.sin(p) / r2);
}
float ssFunc(final float t, final float p[])
{
return ssFunc(t, p, 0);
}
float ssFunc(final float t, final float p[], int pOff)
{
return (float)(Math.pow(Math.pow(Math.abs(Math.cos(p[0+pOff] * t / 4)) / p[1+pOff], p[4+pOff]) +
Math.pow(Math.abs(Math.sin(p[0+pOff] * t / 4)) / p[2+pOff], p[5+pOff]), 1 / p[3+pOff]));
}
// Creates and returns a supershape object.
// Based on Paul Bourke's POV-Ray implementation.
// http://astronomy.swin.edu.au/~pbourke/povray/supershape/
GLObject createSuperShape(final float params[])
{
final int resol1 = (int)params[SuperShape.PARAMS - 3];
final int resol2 = (int)params[SuperShape.PARAMS - 2];
// latitude 0 to pi/2 for no mirrored bottom
// (latitudeBegin==0 for -pi/2 to pi/2 originally)
final int latitudeBegin = resol2 / 4;
final int latitudeEnd = resol2 / 2; // non-inclusive
final int longitudeCount = resol1;
final int latitudeCount = latitudeEnd - latitudeBegin;
final int triangleCount = longitudeCount * latitudeCount * 2;
final int vertices = triangleCount * 3;
GLObject result;
float baseColor[] = new float[3];
int a, longitude, latitude;
int currentVertex, currentQuad;
result = new GLObject(vertices, 3, true);
if (result == null)
return null;
for (a = 0; a < 3; ++a)
baseColor[a] = ((randomUInt() % 155) + 100) / 255.f;
currentQuad = 0;
currentVertex = 0;
// longitude -pi to pi
for (longitude = 0; longitude < longitudeCount; ++longitude)
{
// latitude 0 to pi/2
for (latitude = latitudeBegin; latitude < latitudeEnd; ++latitude)
{
float t1 = (float) ( -Math.PI + longitude * 2 * Math.PI / resol1 );
float t2 = (float) ( -Math.PI + (longitude + 1) * 2 * Math.PI / resol1 );
float p1 = (float) ( -Math.PI / 2 + latitude * 2 * Math.PI / resol2 );
float p2 = (float) ( -Math.PI / 2 + (latitude + 1) * 2 * Math.PI / resol2 );
float r0, r1, r2, r3;
r0 = ssFunc(t1, params);
r1 = ssFunc(p1, params, 6);
r2 = ssFunc(t2, params);
r3 = ssFunc(p2, params, 6);
if (r0 != 0 && r1 != 0 && r2 != 0 && r3 != 0)
{
VECTOR3 pa=new VECTOR3(), pb=new VECTOR3(), pc=new VECTOR3(), pd=new VECTOR3();
VECTOR3 v1=new VECTOR3(), v2=new VECTOR3(), n=new VECTOR3();
float ca;
int i;
//float lenSq, invLenSq;
superShapeMap(pa, r0, r1, t1, p1);
superShapeMap(pb, r2, r1, t2, p1);
superShapeMap(pc, r2, r3, t2, p2);
superShapeMap(pd, r0, r3, t1, p2);
// kludge to set lower edge of the object to fixed level
if (latitude == latitudeBegin + 1)
pa.z = pb.z = 0;
vector3Sub(v1, pb, pa);
vector3Sub(v2, pd, pa);
// Calculate normal with cross product.
/* i j k i j
* v1.x v1.y v1.z | v1.x v1.y
* v2.x v2.y v2.z | v2.x v2.y
*/
n.x = v1.y * v2.z - v1.z * v2.y;
n.y = v1.z * v2.x - v1.x * v2.z;
n.z = v1.x * v2.y - v1.y * v2.x;
/* Pre-normalization of the normals is disabled here because
* they will be normalized anyway later due to automatic
* normalization (gl.GL_NORMALIZE). It is enabled because the
* objects are scaled with glScale.
*/
/*
lenSq = n.x * n.x + n.y * n.y + n.z * n.z;
invLenSq = (float)(1 / sqrt(lenSq));
n.x *= invLenSq;
n.y *= invLenSq;
n.z *= invLenSq;
*/
ca = pa.z + 0.5f;
for (i = currentVertex * 3;
i < (currentVertex + 6) * 3;
i += 3)
{
result.normalArray.put(i , FixedPoint.toFixed(n.x));
result.normalArray.put(i + 1, FixedPoint.toFixed(n.y));
result.normalArray.put(i + 2, FixedPoint.toFixed(n.z));
}
for (i = currentVertex * 4;
i < (currentVertex + 6) * 4;
i += 4)
{
int j, color[] = new int[3];
for (j = 0; j < 3; ++j)
{
color[j] = (int)(ca * baseColor[j] * 255);
if (color[j] > 255) color[j] = 255;
}
result.colorArray.put(i , (byte)color[0]);
result.colorArray.put(i + 1, (byte)color[1]);
result.colorArray.put(i + 2, (byte)color[2]);
result.colorArray.put(i + 3, (byte)0);
}
result.vertexArray.put(currentVertex * 3, FixedPoint.toFixed(pa.x));
result.vertexArray.put(currentVertex * 3 + 1, FixedPoint.toFixed(pa.y));
result.vertexArray.put(currentVertex * 3 + 2, FixedPoint.toFixed(pa.z));
++currentVertex;
result.vertexArray.put(currentVertex * 3, FixedPoint.toFixed(pb.x));
result.vertexArray.put(currentVertex * 3 + 1, FixedPoint.toFixed(pb.y));
result.vertexArray.put(currentVertex * 3 + 2, FixedPoint.toFixed(pb.z));
++currentVertex;
result.vertexArray.put(currentVertex * 3, FixedPoint.toFixed(pd.x));
result.vertexArray.put(currentVertex * 3 + 1, FixedPoint.toFixed(pd.y));
result.vertexArray.put(currentVertex * 3 + 2, FixedPoint.toFixed(pd.z));
++currentVertex;
result.vertexArray.put(currentVertex * 3, FixedPoint.toFixed(pb.x));
result.vertexArray.put(currentVertex * 3 + 1, FixedPoint.toFixed(pb.y));
result.vertexArray.put(currentVertex * 3 + 2, FixedPoint.toFixed(pb.z));
++currentVertex;
result.vertexArray.put(currentVertex * 3, FixedPoint.toFixed(pc.x));
result.vertexArray.put(currentVertex * 3 + 1, FixedPoint.toFixed(pc.y));
result.vertexArray.put(currentVertex * 3 + 2, FixedPoint.toFixed(pc.z));
++currentVertex;
result.vertexArray.put(currentVertex * 3, FixedPoint.toFixed(pd.x));
result.vertexArray.put(currentVertex * 3 + 1, FixedPoint.toFixed(pd.y));
result.vertexArray.put(currentVertex * 3 + 2, FixedPoint.toFixed(pd.z));
++currentVertex;
} // r0 && r1 && r2 && r3
++currentQuad;
} // latitude
} // longitude
// Set number of vertices in object to the actual amount created.
result.count = currentVertex;
result.seal();
return result;
}
GLObject createGroundPlane()
{
final int scale = 4;
final int yBegin = -15, yEnd = 15; // ends are non-inclusive
final int xBegin = -15, xEnd = 15;
final int triangleCount = (yEnd - yBegin) * (xEnd - xBegin) * 2;
final int vertices = triangleCount * 3;
GLObject result;
int x, y;
int currentVertex, currentQuad;
result = new GLObject(vertices, 2, false);
if (result == null)
return null;
currentQuad = 0;
currentVertex = 0;
for (y = yBegin; y < yEnd; ++y)
{
for (x = xBegin; x < xEnd; ++x)
{
byte color;
int i, a;
color = (byte)((randomUInt() & 0x5f) + 81); // 101 1111
for (i = currentVertex * 4; i < (currentVertex + 6) * 4; i += 4)
{
result.colorArray.put(i, color);
result.colorArray.put(i + 1, color);
result.colorArray.put(i + 2, color);
result.colorArray.put(i + 3, (byte)0);
}
// Axis bits for quad triangles:
// x: 011100 (0x1c), y: 110001 (0x31) (clockwise)
// x: 001110 (0x0e), y: 100011 (0x23) (counter-clockwise)
for (a = 0; a < 6; ++a)
{
final int xm = x + ((0x1c >> a) & 1);
final int ym = y + ((0x31 >> a) & 1);
final float m = (float)(Math.cos(xm * 2) * Math.sin(ym * 4) * 0.75f);
result.vertexArray.put(currentVertex * 2, FixedPoint.toFixed(xm * scale + m));
result.vertexArray.put(currentVertex * 2 + 1, FixedPoint.toFixed(ym * scale + m));
++currentVertex;
}
++currentQuad;
}
}
result.seal();
return result;
}
void drawGroundPlane()
{
gl.glDisable(gl.GL_LIGHTING);
gl.glDisable(gl.GL_DEPTH_TEST);
if(blendingEnabled) {
gl.glEnable(gl.GL_BLEND);
gl.glBlendFunc(gl.GL_ZERO, gl.GL_SRC_COLOR);
}
sGroundPlane.draw();
if(blendingEnabled) {
gl.glDisable(gl.GL_BLEND);
}
gl.glEnable(gl.GL_DEPTH_TEST);
gl.glEnable(gl.GL_LIGHTING);
}
void drawFadeQuad()
{
final int beginFade = (int) (sTick - sCurrentCamTrackStartTick);
final int endFade = (int) (sNextCamTrackStartTick - sTick);
final int minFade = beginFade < endFade ? beginFade : endFade;
if (minFade < 1024)
{
final int fadeColor = minFade << 7;
gl.glColor4x(fadeColor, fadeColor, fadeColor, 0);
gl.glDisable(gl.GL_DEPTH_TEST);
gl.glEnable(gl.GL_BLEND);
gl.glBlendFunc(gl.GL_ZERO, gl.GL_SRC_COLOR);
gl.glDisable(gl.GL_LIGHTING);
gl.glMatrixMode(gl.GL_MODELVIEW);
gl.glLoadIdentity();
gl.glMatrixMode(gl.GL_PROJECTION);
gl.glLoadIdentity();
gl.glBindBuffer(GL.GL_ARRAY_BUFFER, 0);
gl.glDisableClientState(gl.GL_COLOR_ARRAY);
gl.glDisableClientState(gl.GL_NORMAL_ARRAY);
gl.glVertexPointer(2, gl.GL_FIXED, 0, quadVertices);
gl.glDrawArrays(gl.GL_TRIANGLES, 0, 6);
gl.glEnableClientState(gl.GL_COLOR_ARRAY);
gl.glMatrixMode(gl.GL_MODELVIEW);
gl.glEnable(gl.GL_LIGHTING);
gl.glDisable(gl.GL_BLEND);
gl.glEnable(gl.GL_DEPTH_TEST);
}
}
IntBuffer quadVertices;
IntBuffer light0Position;
IntBuffer light0Diffuse;
IntBuffer light1Position;
IntBuffer light1Diffuse;
IntBuffer light2Position;
IntBuffer light2Diffuse;
IntBuffer materialSpecular;
void configureLightAndMaterial()
{
gl.glLightxv(gl.GL_LIGHT0, gl.GL_POSITION, light0Position);
gl.glLightxv(gl.GL_LIGHT0, gl.GL_DIFFUSE, light0Diffuse);
gl.glLightxv(gl.GL_LIGHT1, gl.GL_POSITION, light1Position);
gl.glLightxv(gl.GL_LIGHT1, gl.GL_DIFFUSE, light1Diffuse);
gl.glLightxv(gl.GL_LIGHT2, gl.GL_POSITION, light2Position);
gl.glLightxv(gl.GL_LIGHT2, gl.GL_DIFFUSE, light2Diffuse);
gl.glMaterialxv(gl.GL_FRONT_AND_BACK, gl.GL_SPECULAR, materialSpecular);
gl.glMaterialx(gl.GL_FRONT_AND_BACK, gl.GL_SHININESS, 60 << 16);
gl.glEnable(gl.GL_COLOR_MATERIAL);
}
void drawModels(float zScale)
{
final int translationScale = 9;
int x, y;
seedRandom(9);
gl.glScalex(1 << 16, 1 << 16, FixedPoint.toFixed(zScale));
for (y = -5; y <= 5; ++y)
{
for (x = -5; x <= 5; ++x)
{
int curShape = randomUInt() % SuperShape.COUNT;
float buildingScale = SuperShape.sParams[curShape][SuperShape.PARAMS - 1];
int fixedScale = FixedPoint.toFixed(buildingScale);
gl.glPushMatrix();
gl.glTranslatex(FixedPoint.toFixed(x * translationScale),
FixedPoint.toFixed(y * translationScale),
0);
gl.glRotatex(FixedPoint.toFixed((randomUInt() % 360) << 16), 0, 0, 1 << 16);
gl.glScalex(fixedScale, fixedScale, fixedScale);
sSuperShapeObjects[curShape].draw();
gl.glPopMatrix();
}
}
for (x = -2; x <= 2; ++x)
{
final int shipScale100 = translationScale * 500;
final int offs100 = x * shipScale100 + (int)(sTick % shipScale100);
float offs = offs100 * 0.01f;
int fixedOffs = FixedPoint.toFixed(offs);
gl.glPushMatrix();
gl.glTranslatex(fixedOffs, -4 * 65536, 2 << 16);
sSuperShapeObjects[SuperShape.COUNT - 1].draw();
gl.glPopMatrix();
gl.glPushMatrix();
gl.glTranslatex(-4 * 65536, fixedOffs, 4 << 16);
gl.glRotatex(90 << 16, 0, 0, 1 << 16);
sSuperShapeObjects[SuperShape.COUNT - 1].draw();
gl.glPopMatrix();
}
}
void camTrack()
{
float lerp[]= new float[5];
float eX, eY, eZ, cX, cY, cZ;
float trackPos;
CamTrack cam;
long currentCamTick;
int a;
if (sNextCamTrackStartTick <= sTick)
{
++sCurrentCamTrack;
sCurrentCamTrackStartTick = sNextCamTrackStartTick;
}
sNextCamTrackStartTick = sCurrentCamTrackStartTick +
CamTrack.sCamTracks[sCurrentCamTrack].len * CamTrack.CAMTRACK_LEN;
cam = CamTrack.sCamTracks[sCurrentCamTrack];
currentCamTick = sTick - sCurrentCamTrackStartTick;
trackPos = (float)currentCamTick / (CamTrack.CAMTRACK_LEN * cam.len);
for (a = 0; a < 5; ++a)
lerp[a] = (cam.src[a] + cam.dest[a] * trackPos) * 0.01f;
if (cam.dist>0)
{
float dist = cam.dist * 0.1f;
cX = lerp[0];
cY = lerp[1];
cZ = lerp[2];
eX = cX - (float)Math.cos(lerp[3]) * dist;
eY = cY - (float)Math.sin(lerp[3]) * dist;
eZ = cZ - lerp[4];
}
else
{
eX = lerp[0];
eY = lerp[1];
eZ = lerp[2];
cX = eX + (float)Math.cos(lerp[3]);
cY = eY + (float)Math.sin(lerp[3]);
cZ = eZ + lerp[4];
}
glu.gluLookAt(eX, eY, eZ, cX, cY, cZ, 0, 0, 1);
}
private int gAppAlive = 0;
private int width, height, x, y, frames;
}