include "freeglut.inc"
indexbase 0
'function Main at the end of the file
' GLOBAL VARIABLES
'escape to exit
% GLUT_KEY_ESCAPE 27
'Arrow keys to rotate cube
'A zoom in
% GLUT_KEY_IN 97
'Z zoom out
% GLUT_KEY_OUT 122
sys flagOnceForLastTime = 0
const GLsizei windowWidth = 1024
const GLsizei windowHeight = 768
const GLfloat speed = 1
const GLfloat zoomSpeed = 0.05
GLfloat cubeRotateX = 45
GLfloat cubeRotateY = 45
GLfloat camMovZ = -5
bool keys[255]
' NON CALLBACK FUNCTIONS AND SUBS
sub establishProjectionMatrix( GLsizei width, GLsizei height )
glViewport 0, 0, width, height
glMatrixMode GL_PROJECTION
glLoadIdentity
gluPerspective 45, ( width / height ), 0.1, 200
end sub
sub initGL( GLsizei width, GLsizei height )
establishProjectionMatrix width, height
glShadeModel GL_SMOOTH
glClearColor 0, 0, 0, 1
glEnable GL_DEPTH_TEST
glDepthFunc GL_LEQUAL
glHint GL_PERSPECTIVE_CORRECTION_HINT, GL_NICEST
glEnable GL_PERSPECTIVE_CORRECTION_HINT
end sub
sub displayFPS()
static sys lastTime = 0
if flagOnceForLastTime = 0 then
lastTime = glutGet(GLUT_ELAPSED_TIME)
flagOnceForLastTime = 1
end if
' this keeps track of the loops or frames
static sys loops = 0
' we will average out the fps to make it less volatile in what it shows
static GLfloat fps = 0
sys newTime = glutGet GLUT_ELAPSED_TIME
sys t = newTime - lastTime
'glutSetWindowTitle str(lastTime) + " " + str(newTime) + " t: " + str(t)
if t > 100 then
GLfloat newFPS = loops / ( newTime - lastTime ) * 1000
fps = ( fps + newFPS ) / 2
string s = "FPS : " + str(fps,2)
glutSetWindowTitle s
lastTime = newTime
loops = 0
end if
loops ++
end sub
function checkKeys() as GLboolean
if keys[GLUT_KEY_ESCAPE] = GL_TRUE then
return GL_TRUE
end if
if keys[GLUT_KEY_LEFT] = GL_TRUE then
cubeRotateY -= speed
end if
if keys[GLUT_KEY_RIGHT] = GL_TRUE then
cubeRotateY += speed
end if
if keys[GLUT_KEY_UP] = GL_TRUE then
cubeRotateX -= speed
end if
if keys[GLUT_KEY_DOWN] = GL_TRUE then
cubeRotateX += speed
end if
if keys[GLUT_KEY_IN] = GL_TRUE then
camMovZ += zoomSpeed
end if
if keys[GLUT_KEY_OUT] = GL_TRUE then
camMovZ -= zoomSpeed
end if
return GL_FALSE
end function
'===========
extern cdecl ' because of this syntax, no need to add callback to each sub or function below
'===========
'CALLBACK FUNCTIONS AND SUBS
' by declaring and defining here, we don't need to do it twice
sub drawSceneCB()
' for 3D we clear both bits, if 2D only the one on the left
glClear GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT
glMatrixMode GL_MODELVIEW
glLoadIdentity()
' moves the camera back
glTranslatef 0, 0, camMovZ
glRotatef cubeRotateX, 1, 0, 0
glRotatef cubeRotateY, 0, 1, 0
' Draw cube
glBegin GL_QUADS
' top face
glColor3f 1, 0.5, 0
glVertex3f 1, 1, -1
glVertex3f -1, 1, -1
glVertex3f -1, 1, 1
glVertex3f 1, 1, 1
' bottom face
glColor3f 0, 1, 0
glVertex3f 1, -1, -1
glVertex3f -1, -1, -1
glVertex3f -1, -1, 1
glVertex3f 1, -1, 1
' front face
glColor3f 1, 0, 0
glVertex3f 1, 1, 1
glVertex3f -1, 1, 1
glVertex3f -1, -1, 1
glVertex3f 1, -1, 1
' back face
glColor3f 1, 1, 0
glVertex3f 1, 1, -1
glVertex3f -1, 1, -1
glVertex3f -1, -1, -1
glVertex3f 1, -1, -1
' left face
glColor3f 0, 0, 1
glVertex3f -1, 1, 1
glVertex3f -1, 1, -1
glVertex3f -1, -1, -1
glVertex3f -1, -1, 1
' right face
glColor3f 1, 0, 1
glVertex3f 1, 1, 1
glVertex3f 1, 1, -1
glVertex3f 1, -1, -1
glVertex3f 1, -1, 1
glEnd
glFlush
' this line will change depending on which window interface
' is being used. Like SDL and GLUT
glutSwapBuffers
displayFPS
end sub
'timerLoop must match glutTimerFunc's requirements
sub timerLoopCB (int value)
if checkKeys() = GL_TRUE then
glutExit
end if
' glut will call the drawscene function when it ready
' very important for this to be allowed
glutPostRedisplay
' by putting this here, we created a loop.
' SDL doesn't need something like this
glutTimerFunc 1, @timerLoopCB, 0
end sub
sub keyboardCB(int key, int x, int y)
keys[key] = GL_TRUE
end sub
sub keyboardUpCB(int key, int x, int y)
keys[key] = GL_FALSE
end sub
sub keyboardSpecialCB(int key, int x, int y)
keys[key] = GL_TRUE
end sub
sub keyboardSpecialUpCB(int key, int x, int y)
keys[key] = GL_FALSE
end sub
'=========
end extern
'=========
function main(int argc, char **argv) as sys
' initialize glut, passing the above parameters to gluInit
glutInit argc, argv
' enable double frame buffering, so we don't get flicker
glutInitDisplayMode GLUT_DOUBLE
sys windowID = glutCreateWindow "OpenGL FreeGlut Cube Demo"
' by default, glut knows the last window used,
'so we can resize it after we create it
glutReshapeWindow windowWidth, windowHeight
' our procedure
initGL windowWidth, windowHeight
glutDisplayFunc @drawSceneCB
glutKeyboardFunc @keyboardCB
glutKeyboardUpFunc @keyboardUpCB
glutSpecialFunc @keyboardSpecialCB
glutSpecialUpFunc @keyboardSpecialUpCB
glutTimerFunc 1, @timerLoopCB, 0
' this goes to the main loop and stays there till the end
glutMainLoop
'should never get here, but just in case
return 0
end function
' THIS IS THE PROGRAM
main null,null
'====================