L'exécutable

Fichier source : DisqueAccretionBlur.cpp

/* 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 <stdio.h>
#include <stdlib.h>
#include <math.h>

#include <GL/glut.h>
#include <GL/glu.h>
#include <GL/gl.h>

/* 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);
}

RETOUR