//---------------------------------------------------------------------
// Ukazkovy priklad k serii clanku OpenGL a GLU
//
// Autor: Pavel Tisnovsky
// Cislo clanku: 11
// Cislo prikladu: 2
//
// Zobrazeni prubehu NURB krivky ctvrteho stupne, ktera ma osm ridicich
// bodu a uzlovy vektor obsahuje celkem dvanact polozek. Krome samotne
// krivky jsou zobrazeny i polohy jejich ridicich bodu.
// Pomoci klavesy 'f' lze provest prepnuti do celeho okna, klavesou 'w'
// se provede nastaveni puvodni velikosti okna, tj. 450x450 pixelu.
// Klavesou ESC je mozne program ukoncit.
//---------------------------------------------------------------------
#include <GL/gl.h>
#include <GL/glu.h>
#include <GL/glut.h>
#define WINDOW_WIDTH 450 // velikost okna
#define WINDOW_HEIGHT 450
#define WINDOW_TITLE "OpenGL a GLU, priklad 11.2"// titulek okna
#define POINTS 8
// ridici body
GLfloat ctlpoints[8][3]={
{ 50, 50, 0},
{400, 100, 0},
{ 50, 150, 0},
{400, 200, 0},
{ 50, 250, 0},
{400, 300, 0},
{ 50, 350, 0},
{400, 400, 0}
};
// uzlovy vektor
GLfloat knots[12] = {0.0, 0.0, 0.0, 0.0,
0.2, 0.4, 0.6, 0.8,
1.0, 1.0, 1.0, 1.0};
GLUnurbs *nurbs;
//---------------------------------------------------------------------
// Funkce pro inicializaci vykreslovani
//---------------------------------------------------------------------
void onInit(void)
{
glClearColor(0.0f, 0.0f, 0.0f, 1.0f);
glEnable(GL_LINE_SMOOTH);
glEnable(GL_POINT_SMOOTH);
glPointSize(5.0f);
nurbs=gluNewNurbsRenderer(); // vytvoreni NURBS
}
//--------------------------------------------------------------------
// Tato funkce je volana pri kazdem prekresleni okna
//--------------------------------------------------------------------
void onDisplay(void)
{
int d;
glClear(GL_COLOR_BUFFER_BIT); // smazani barvoveho bufferu
// vykreslit krivku
glColor3f(1.0f, 0.0f, 0.0f);
gluBeginCurve(nurbs); // zacatek specifikace NURBS
gluNurbsCurve(nurbs,
12 , knots, // pocet slozek v uzlovem vektoru
3, // pocet floatu pro jeden ridici bod
&ctlpoints[0][0], // ridici body
4, // pocet ridicich bodu
GL_MAP1_VERTEX_3 // funkce ridicich bodu
);
gluEndCurve(nurbs); // konec specifikace NURBS
// propojit ridici body polycarou
glColor3f(0.6f, 0.6f, 0.6f);
glBegin(GL_LINE_STRIP);
for (d=0; d<POINTS; d++)
glVertex2f(ctlpoints[d][0], ctlpoints[d][1]);
glEnd();
// vykreslit ridici body
glColor3f(0.5f, 0.5f, 1.0f);
glBegin(GL_POINTS);
for (d=0; d<POINTS; d++)
glVertex2f(ctlpoints[d][0], ctlpoints[d][1]);
glEnd();
glFlush();
glutSwapBuffers();
}
//---------------------------------------------------------------------
// Nastaveni souradneho systemu v zavislosti na velikosti okna
//---------------------------------------------------------------------
void onResize(GLsizei w, GLsizei h)
{
glViewport(0, 0, w, h); // viditelna oblast pres cele okno
glMatrixMode(GL_PROJECTION); // zacatek modifikace projekcni matice
glLoadIdentity(); // vymazani projekcni matice (=identita)
glOrtho(0, w, 0, h, -1, 1); // mapovani abstraktnich souradnic do souradnic okna
glScalef(1, -1, 1); // inverze y-ove osy, aby se y zvetsovalo smerem dolu
glTranslatef(0, -h, 0); // posun pocatku do leveho horniho rohu
}
//---------------------------------------------------------------------
// Tato funkce je volana pri stlaceni ASCII klavesy
//---------------------------------------------------------------------
#ifdef __BORLANDC__
#pragma option -w-par // zabranit warningum pri prekladu
#endif // u borlandskych prekladacu
void onKeyboard(unsigned char key, int x, int y)
{
if (key>='A' && key<='Z') // uprava velkych pismen na mala
key+=(unsigned char)('a'-'A'); // pro zjednoduseni prikazu switch
switch (key) {
case 27: exit(0); break; // ukonceni aplikace
case 'q': exit(0); break; // ukonceni aplikace
case 'f': glutFullScreen(); break; // prepnuti na celou obrazovku
case 'w': glutReshapeWindow(WINDOW_WIDTH, WINDOW_HEIGHT); break;
default: break;
}
}
#ifdef __BORLANDC__
#pragma option -w+par
#endif
//---------------------------------------------------------------------
// Hlavni funkce konzolove aplikace
//---------------------------------------------------------------------
int main(int argc, char** argv)
{
glutInit(&argc, argv); // inicializace knihovny GLUT
glutInitDisplayMode(GLUT_RGB | GLUT_DOUBLE); // nastaveni dvou barvovych bufferu a pameti hloubky
glutInitWindowPosition(30, 30); // pocatecni pozice leveho horniho rohu okna
glutInitWindowSize(WINDOW_WIDTH, WINDOW_HEIGHT);// pocatecni velikost okna
glutCreateWindow(WINDOW_TITLE); // vytvoreni okna pro kresleni
glutDisplayFunc(onDisplay); // registrace funkce volane pri prekreslovani okna
glutReshapeFunc(onResize); // registrace funkce volane pri zmene velikosti okna
glutKeyboardFunc(onKeyboard); // registrace funkce volane pri stlaceni klavesy
onInit(); // inicializace vykreslovani
glutMainLoop(); // nekonecna smycka, kde se volaji zaregistrovane funkce
return 0; // navratova hodnota vracena operacnimu systemu
}
//---------------------------------------------------------------------
// Konec zdrojoveho souboru
//---------------------------------------------------------------------