/* Auteur: Nicolas JANEY */ /* nicolas.janey@univ-fcomte.fr */ /* Octobre 2007 */ /* Un programme OpenGL d'animation automatique */ /* d'un ensemble de particules ponctuelles */ /* selon des orbites circulaires de periodes */ /* en pow(r,1.5) (orbite keplerienne). */ /* r est le rayon de l'orbite et decroit */ /* au cours du temps */ #include #include #include #include #include #include /* Variables globales */ static float *r; static float *v; static float *a; static float *rold; static float *aold; static int image = 0; static int nouvellesEtoilesParImage = 10; static int imagesParCycle = 1100; static float rayon = 40.0F; static float rayonMin = 0.1F; static int nombreEtoiles = nouvellesEtoilesParImage * imagesParCycle; static float vitesse = 1.0F; /* Variables globales de gestion */ /* de l'interactivite clavier et souris */ static int clic = 0; static int mx; static int my; static float rx = 0.0F; static float ry = 0.0F; static float zoom = 0.1F; /* Fonction d'affichage d'une eventuelle */ /* erreur OpenGL */ static void checkError(int line) { GLenum err = glGetError(); if (err) { printf("GL Error %s (0x%x) at line %d\n", gluErrorString(err), (int) err, line); } } /* Fonction de calcul d'un nombre reel */ /* aleatoire compris entre 0.0 et 1.0 */ static float random(void) { float v = ((float) rand()/RAND_MAX); return(v); } /* Fonction de creation d'une particule */ void creationParticule(int i) { r[i] = rayon+random(); v[i] = 1.0F/pow((double) r[i],1.5); a[i] = random()*2*3.14159; } /* Fonction de dessin du systeme de particules */ /* blurees */ static void drawDisqueAccretion(void) { glBegin(GL_LINES); for ( int i = 0 ; i < nombreEtoiles ; i++ ) { if ( ( i < nouvellesEtoilesParImage*image ) && ( rold[i] != 0.0F ) ) { float inc = a[i]-aold[i]; int ite = 1+(int) (inc/0.05); float inca = inc/ite; float incr = (r[i]-rold[i])/ite; for ( int j = 0 ; j < ite ; j++ ) { float aa = aold[i]+j*inca; float rr = rold[i]+j*incr; { float x = rr*cos(aa); float y = rr*sin(aa); float c = (rayon-rr)/rayon+0.2F; glColor3f(c,c,c); glVertex3f(x,y,0.0F); } { float x = (rr+incr)*cos(aa+inca); float y = (rr+incr)*sin(aa+inca); float c = (rayon-rr)/rayon+0.2F; glColor3f(c,c,c); glVertex3f(x,y,0.0F); } } } } glEnd(); } /* Fonction executee lors d'un rafraichissement */ /* de la fenetre de dessin */ static void display(void) { glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT); glPushMatrix(); glScalef(zoom,zoom,zoom); glRotatef(rx,1.0F,0.0F,0.0F); glRotatef(ry,0.0F,1.0F,0.0F); drawDisqueAccretion(); glPopMatrix(); glFlush(); checkError(__LINE__); glutSwapBuffers(); } /* Fonction executee lorsqu'aucun evenement */ /* n'est en attente de gestion */ static void idle(void) { int i; int ne = (image%imagesParCycle)*nouvellesEtoilesParImage; for ( i = ne ; i < ne+nouvellesEtoilesParImage ; i++ ) creationParticule(i); for ( i = 0 ; i < nombreEtoiles ; i++ ) { if ( i < nouvellesEtoilesParImage*image ) { aold[i] = a[i]; rold[i] = r[i]; a[i] += v[i]*vitesse; r[i] -= (rayon-rayonMin)/imagesParCycle; v[i] = 1.0F/pow((double) r[i],1.5); } } image++; glutPostRedisplay(); } /* Fonction executee lors d'un changement */ /* de la taille de la fenetre OpenGL */ /* Configuration d'une camera de visualisation */ /* en projection en perspective */ void reshape(int tx,int ty) { glViewport(0,0,tx,ty); glMatrixMode(GL_PROJECTION); glLoadIdentity(); gluPerspective(60,(double) tx/ty,0.1,50.0); glMatrixMode(GL_MODELVIEW); glLoadIdentity(); glTranslatef(0.0F,0.0F,-6.0F); } /* Fonction executee lors de la frappe */ /* d'une touche alphanumerique du clavier */ void key(unsigned char key,int x,int y) { switch ( key ) { case ' ' : { static int anim = 0; anim = !anim; glutIdleFunc(anim ? idle : NULL); } break; case 0x1B : free(r); free(v); free(a); exit(0); break; } } /* Fonction executee lors de la frappe */ /* d'une touche non alphanumerique du clavier */ void special(int key,int x,int y) { switch ( key ) { case GLUT_KEY_UP : zoom *= 0.99F; glutPostRedisplay(); break; case GLUT_KEY_DOWN : zoom /= 0.99F; glutPostRedisplay(); break; case GLUT_KEY_RIGHT : vitesse += 0.001F; glutPostRedisplay(); break; case GLUT_KEY_LEFT : vitesse -= 0.001F; glutPostRedisplay(); break; } } /* Fonction d'initialisation de certains */ /* parametres de l'environnement OpenGL */ /* et allocation des tableaux de stockage */ /* des informations relatives aux particules */ static void init(void) { r =(float *) calloc(nombreEtoiles,sizeof(float)); v =(float *) calloc(nombreEtoiles,sizeof(float)); a =(float *) calloc(nombreEtoiles,sizeof(float)); rold =(float *) calloc(nombreEtoiles,sizeof(float)); aold =(float *) calloc(nombreEtoiles,sizeof(float)); glClearColor(0.0F,0.0F,0.0F,0.0F); glEnable(GL_DEPTH_TEST); } /* Fonction executee lors d'un clic de souris */ /* dans la fenetre */ void mouse(int bouton,int etat,int x,int y) { if ( bouton == GLUT_LEFT_BUTTON ) { if ( etat == GLUT_DOWN ) { clic = 1; mx = x; my = y; } if ( etat == GLUT_UP ) { clic = 0; } } } /* Fonction executee lors d'un deplacement */ /* de la souris sur la fenetre */ /* avec un bouton appuye */ void motion(int x,int y) { if ( clic ) { ry += (x-mx); rx += (y-my); mx = x; my = y; glutPostRedisplay(); } } /* Fonction principale */ int main(int argc,char *argv[]) { glutInit(&argc,argv); glutInitWindowPosition(10,10); glutInitWindowSize(480,240); glutInitDisplayMode(GLUT_RGBA|GLUT_DOUBLE|GLUT_DEPTH); glutCreateWindow("Disque d'accrétion"); init(); glutReshapeFunc(reshape); glutKeyboardFunc(key); glutMouseFunc(mouse); glutMotionFunc(motion); glutSpecialFunc(special); glutIdleFunc(idle); glutDisplayFunc(display); glutMainLoop(); return(0); }