Planet
navi homePPSaboutscreenshotsdownloaddevelopmentforum

source: orxonox.OLD/orxonox/trunk/src/importer/model.cc @ 3426

Last change on this file since 3426 was 3426, checked in by bensch, 20 years ago

orxonox/trunk/importer: added cylinderModel to importer

File size: 24.5 KB
RevLine 
[2823]1/*
2   orxonox - the future of 3D-vertical-scrollers
3
4   Copyright (C) 2004 orx
5
6   This program is free software; you can redistribute it and/or modify
7   it under the terms of the GNU General Public License as published by
8   the Free Software Foundation; either version 2, or (at your option)
9   any later version.
10
11   ### File Specific:
12   main-programmer: Benjamin Grauer
13   co-programmer: ...
14*/
15
[3360]16#include "model.h"
[3400]17#include <math.h>
[3418]18#include <stdarg.h>
[3398]19
[3140]20using namespace std;
[2776]21
[2842]22/**
[3398]23   \brief Creates a 3D-Model.
[3186]24
[3398]25   This only initializes a 3D-Model, but does not cleanup the Faces.
[2842]26*/
[3398]27Model::Model(void)
[2748]28{
[3398]29  this->initialize();
30}
[2765]31
[3398]32/**
33   \brief Creates a 3D-Model of Primitive-Type type
34
35   if you want to just display a Cube/Sphere/Cylinder/... without any material.
36   
37   \todo implement Cube/Sphere/Cylinder/...
38*/
39Model::Model(PRIMITIVE type)
40{
[3195]41  this->initialize();
[2847]42
[3400]43  switch (type)
44    {
45    default:
46    case CUBE:
47      this->cubeModel();
48      break;
49    case SPHERE:
[3418]50      this->sphereModel();
[3400]51      break;
52    case CYLINDER:
[3426]53      this->cylinderModel();
[3400]54      break;
[2765]55
[3400]56    }
[3195]57  this->importToGL ();
[3064]58
[3195]59  this->cleanup();
[2767]60}
61
[2842]62/**
[3398]63   \brief Creates a 3D-Model. and assigns it a Name.
64*/
65Model::Model(char* modelName)
66{
67  this->initialize();
68  this->setName(modelName);
69}
70
71/**
[3360]72   \brief deletes an Model.
[3186]73
[3360]74   Looks if any from model allocated space is still in use, and if so deleted it.
[2847]75*/
[3398]76Model::~Model(void)
[2847]77{
[3397]78  PRINTF(3)("Deleting Model ");
[3396]79  if (this->name)
80    {
[3397]81      PRINT(3)("%s\n", this->name);
[3396]82      delete []this->name;
83    }
84  else
[3397]85      PRINT(3)("\n");
[3396]86
[3397]87  PRINTF(3)("Deleting display Lists.\n");
[3195]88  Group* walker = this->firstGroup;
[2850]89  while (walker != NULL)
90    {
91      glDeleteLists (walker->listNumber, 1);
[3066]92      Group* delWalker = walker;
[3063]93      walker = walker->next;
[3066]94      delete delWalker;
[3140]95    }
96
[3397]97  PRINTF(3)("Deleting Materials.\n");
[3195]98  if (this->material)
99    delete this->material;
[2847]100}
101
[2842]102/**
[3398]103   \brief Finalizes an Object. This can be done outside of the Class.
104*/
105void Model::finalize(void)
106{
107  this->importToGL ();
108 
109  this->cleanup();
110
111  this->finalized = true;
112}
113
114/**
[3360]115   \brief Draws the Models of all Groups.
[2851]116   It does this by just calling the Lists that must have been created earlier.
[2842]117*/
[3360]118void Model::draw (void) const
[2748]119{
[3360]120  PRINTF(2)("drawing the 3D-Models\n"); 
[3195]121  Group* walker = this->firstGroup;
[2850]122  while (walker != NULL)
123    {
[3360]124      PRINTF(3)("Drawing model %s\n", walker->name);
[2850]125      glCallList (walker->listNumber);
[3063]126      walker = walker->next;
[2850]127    }
[2748]128}
[2754]129
[2842]130/**
[3360]131   \brief Draws the Model number groupNumber
[3186]132   \param groupNumber The number of the group that will be displayed.
133
[2851]134   It does this by just calling the List that must have been created earlier.
135*/
[3360]136void Model::draw (int groupNumber) const 
[2851]137{
[3195]138  if (groupNumber >= this->groupCount)
[2852]139    {
[3360]140      PRINTF(1)("You requested model number %i, but this File only contains of %i Models.\n", groupNumber-1, this->groupCount);
[2852]141      return;
142    }
[3360]143  PRINTF(2)("drawing the requested 3D-Models if found.\n"); 
[3195]144  Group* walker = this->firstGroup;
[2851]145  int counter = 0;
146  while (walker != NULL)
147    {
148      if (counter == groupNumber)
149        {
[3360]150          PRINTF(2)("Drawing model number %i named %s\n", counter, walker->name);
[2851]151          glCallList (walker->listNumber);
152          return;
153        }
154      ++counter;
[3063]155      walker = walker->next;
[2851]156    }
[3396]157  PRINTF(1)("Model number %i in %s not Found.\n", groupNumber, this->name);
[2851]158  return;
159
160}
[2852]161
162/**
[3360]163   \brief Draws the Model with a specific groupName
[3186]164   \param groupName The name of the group that will be displayed.
165
[2852]166   It does this by just calling the List that must have been created earlier.
167*/
[3360]168void Model::draw (char* groupName) const
[2851]169{
[3360]170  PRINTF(2)("drawing the requested 3D-Models if found.\n"); 
[3195]171  Group* walker = this->firstGroup;
[2851]172  while (walker != NULL)
173    {
174      if (!strcmp(walker->name, groupName))
175        {
[3360]176          PRINTF(2)("Drawing model %s\n", walker->name);
[2851]177          glCallList (walker->listNumber);
178          return;
179        }
[3063]180      walker = walker->next;
[2851]181    }
[3396]182  PRINTF(1)("Model Named %s in %s not Found.\n", groupName, this->name);
[2851]183  return;
184}
185
[2852]186/**
[3360]187   \returns Count of the Models in this File
[2852]188*/
[3360]189int Model::getGroupCount (void) const
[2852]190{
[3195]191  return this->groupCount;
[2852]192}
[2851]193
194/**
[3360]195    \brief initializes the Model.
[3186]196
197    This Function initializes all the needed arrays, Lists and clientStates.
198    It also defines default values.
[3066]199*/
[3360]200bool Model::initialize (void)
[3066]201{
[3360]202  PRINTF(2)("new 3D-Model is being created\n"); 
[3066]203
[3396]204  this->name = NULL;
[3398]205  this->finalized = false;
[3066]206  // setting the start group;
[3195]207  this->firstGroup = new Group;
208  this->currentGroup = this->firstGroup;
209  this->groupCount = 0;
[3066]210 
[3195]211  this->initGroup (this->currentGroup);
212  this->scaleFactor = 1;
213  this->material = new Material();
[3066]214
[3195]215  this->vertices = new Array();
216  this->vTexture = new Array();
217  this->normals = new Array();
[3066]218
219  return true;
220}
221
[3398]222void Model::setName(const char* name)
223{
224  if (this->name) 
225    delete this->name;
226  this->name = new char[strlen(name)+1];
227  strcpy(this->name, name);
228}
[3066]229/**
[3360]230   \brief initializes a new Group model
[3186]231   \param group the group that should be initialized.
232   \todo Maybe Group should be a Class, because it does a lot of stuff
233   
[2850]234*/
[3360]235bool Model::initGroup(Group* group)
[2850]236{
[3207]237  PRINTF(3)("Adding new Group\n");
[2863]238  group->name = "";
[2850]239  group->faceMode = -1;
[2967]240  group->faceCount = 0; 
[3063]241  group->next = NULL;
[2850]242
[3063]243  group->firstFace = new Face;
[3195]244  this->initFace (group->firstFace);
[3063]245  group->currentFace = group->firstFace;
[2850]246}
247
248/**
[3066]249   \brief initializes a new Face. (sets default Values)
[3186]250   \param face The face to initialize
[2850]251*/
[3360]252bool Model::initFace (Face* face)
[2850]253{
[3066]254  face->vertexCount = 0;
[2967]255
[3066]256  face->firstElem = NULL;
257 
258  face->materialString = NULL;
259 
260  face->next = NULL;
261
262  return true;
[2850]263}
[2863]264
[2850]265/**
[3360]266   \brief finalizes an Model.
[3066]267   This funcion is needed, to delete all the Lists, and arrays that are no more needed because they are already imported into openGL. This will be applied at the end of the importing Process.
268*/
[3360]269bool Model::cleanup(void)
[3066]270{
[3360]271  PRINTF(3)("cleaning up the 3D-Model to save Memory.\n");
[3066]272
[3195]273  if (this->vertices)
274    delete this->vertices;
275  if (this->vTexture)
276    delete this->vTexture;
277  if (this->normals)
278    delete this->normals;
[3066]279
[3195]280  this->cleanupGroup(this->firstGroup);
[3066]281  return true; 
282}
283
284/**
[3068]285   \brief Cleans up all groups starting from group.
286   \param group the first Group to clean
287*/
[3360]288bool Model::cleanupGroup (Group* group)
[3068]289{
[3207]290  PRINTF(3)("Cleaning up group\n");
[3068]291  if (group->firstFace != NULL)
292    {
293      cleanupFace (group->firstFace);
294      delete group->firstFace;
295    }
296
297  if (group->next !=NULL)
298    cleanupGroup (group->next);
299  return true;
300}
301
302/**
[3186]303   \brief Cleans up all Faces starting from face until NULL is reached.
304   \param face the first face to clean.
[3068]305*/
[3360]306bool Model::cleanupFace (Face* face)
[3068]307{
[3207]308  PRINTF(3)("Cleaning up Face\n");
[3068]309
310  if (face->materialString != NULL)
[3080]311      delete []face->materialString;
[3068]312  if (face->firstElem != NULL)
313    {
[3195]314      this->cleanupFaceElement(face->firstElem);
[3068]315      delete face->firstElem;
316    }
317     
318  if (face->next != NULL)
319    {
[3195]320      this->cleanupFace (face->next);
[3068]321      delete face->next;
322    }
323     
324}
325
326
327/**
328   \brief Cleans up all FaceElements starting from faceElem.
329   \param faceElem the first FaceElement to clean.
330*/
[3360]331bool Model::cleanupFaceElement(FaceElement* faceElem)
[3068]332{
333  if (faceElem->next != NULL)
334    {
[3195]335      this->cleanupFaceElement (faceElem->next);
[3072]336      delete faceElem->next;
[3068]337    }
338}
339
340/**
[3066]341   \brief parses a group String
[3186]342   \param groupString the new Group to create
343
[3066]344   This function initializes a new Group.
[3360]345   With it you should be able to import .obj-files with more than one Models inside.
[3066]346*/
[3396]347bool Model::addGroup (char* groupString)
[3066]348{
[3207]349  PRINTF(3)("Read Group: %s.\n", groupString);
[3195]350  if (this->groupCount != 0 && this->currentGroup->faceCount>0)
[3066]351    {
352      //      finalizeGroup(currentGroup);
[3195]353      this->currentGroup = this->currentGroup->next = new Group;
354      this->initGroup(this->currentGroup);
[3066]355    }
[3140]356  // setting the group name if not default.
357  if (strcmp(groupString, "default"))
358    {
[3195]359      this->currentGroup->name = new char [strlen(groupString)+1];
360      strcpy(this->currentGroup->name, groupString);
[3140]361    }
[3195]362  ++this->groupCount;
[3066]363
364}
365
366/**
[2842]367   \brief parses a vertex-String
[3186]368   \param vertexString The String that will be parsed.
369
[2842]370   If a vertex line is found this function will inject it into the vertex-Array
371*/
[3396]372bool Model::addVertex (char* vertexString)
[2767]373{
[3071]374  float subbuffer1;
375  float subbuffer2;
376  float subbuffer3;
377  sscanf (vertexString, "%f %f %f", &subbuffer1, &subbuffer2, &subbuffer3);
[3207]378  PRINTF(3)("reading in a vertex: %f %f %f\n", &subbuffer1, &subbuffer2, &subbuffer3);
[3195]379  this->vertices->addEntry(subbuffer1*scaleFactor, subbuffer2*scaleFactor, subbuffer3*scaleFactor);
[2767]380  return true;
381}
382
[2842]383/**
[3400]384   \brief parses a vertex-String
385   \param x the X-coordinate of the Vertex to add.
386   \param y the Y-coordinate of the Vertex to add.
387   \param z the Z-coordinate of the Vertex to add.
388   
389*/
390bool Model::addVertex(const float x, const float y, const float z)
391{
392  PRINTF(4)("reading in a vertex: %f %f %f\n", x, y, z);
393  this->vertices->addEntry(x*scaleFactor, y*scaleFactor, z*scaleFactor);
394  return true;
395}
396
397/**
[2842]398   \brief parses a face-string
[3186]399   \param faceString The String that will be parsed.
400
[2842]401   If a face line is found this function will add it to the glList.
402*/
[3396]403bool Model::addFace (char* faceString)
[2767]404{
[3195]405  if (this->currentGroup->faceCount >0)
406    this->currentGroup->currentFace = this->currentGroup->currentFace->next = new Face;
407  this->initFace (this->currentGroup->currentFace);
[2767]408
[3195]409  FaceElement* tmpElem = this->currentGroup->currentFace->firstElem = new FaceElement;
[3068]410  tmpElem->next = NULL;
[2934]411  while(strcmp (faceString, "\0"))
[2767]412    {
[3195]413      if (this->currentGroup->currentFace->vertexCount>0)
[2934]414          tmpElem = tmpElem->next = new FaceElement;
415      tmpElem->next = NULL;
416
[3064]417      char tmpValue [50];
[3072]418      int tmpLen;
419      char* vertex = NULL;
420      char* texture = NULL;
421      char* normal = NULL;
422
[3063]423      sscanf (faceString, "%s", tmpValue);
[3072]424      tmpLen = strlen(tmpValue);
425      vertex = tmpValue;
[2934]426
[3072]427      if ((texture = strstr (vertex, "/")) != NULL)
428        {
429          texture[0] = '\0';
430          texture ++;
431         
432          if ((normal = strstr (texture, "/")) !=NULL)
433            {
434              normal[0] = '\0';
435              normal ++;
436            }     
437        }
438      if (vertex)
[3073]439        tmpElem->vertexNumber = atoi(vertex)-1;
[3072]440      else
[3073]441        tmpElem->vertexNumber = -1;
[3072]442      if (texture)
[3073]443        tmpElem->texCoordNumber = atoi(texture)-1;
[3072]444      else
[3073]445        tmpElem->texCoordNumber = -1;
[3072]446      if (normal)
[3073]447        tmpElem->normalNumber = atoi(normal)-1;
[3072]448      else
[3073]449        tmpElem->normalNumber = -1;
[3072]450
451      faceString += tmpLen;
[2934]452      if (strcmp (faceString, "\0"))
453        faceString++;
[3195]454      this->currentGroup->currentFace->vertexCount++;
[2934]455    }
456
[3195]457  this->currentGroup->faceCount += this->currentGroup->currentFace->vertexCount -2;
[2754]458}
[2768]459
[2842]460/**
[3418]461   \brief adds a new Face
462   \param faceElemCount the number of Vertices to add to the Face.
463   \param type 0: vertex only, 1: vertex and normal, 2: vertex and Texture, 3 vertex, normal and texture
[3400]464*/
[3418]465bool Model::addFace(const float faceElemCount, int type, ...)
[3400]466{
[3418]467   if (this->currentGroup->faceCount > 0)
468    this->currentGroup->currentFace = this->currentGroup->currentFace->next = new Face;
469  this->initFace (this->currentGroup->currentFace);
[3400]470
[3418]471  FaceElement* tmpElem = this->currentGroup->currentFace->firstElem = new FaceElement;
472  tmpElem->next = NULL;
473 
474  va_list itemlist;
475  va_start (itemlist, type);
476
477  for (int i = 0; i < faceElemCount; i++)
478    {
479      if (this->currentGroup->currentFace->vertexCount>0)
480          tmpElem = tmpElem->next = new FaceElement;
481      tmpElem->next = NULL;
482
483      tmpElem->vertexNumber = va_arg (itemlist, int) -1;
484      if (type >= 2)
485        tmpElem->texCoordNumber = va_arg (itemlist, int) -1;
486      if (type == 1 || type ==3)
487        tmpElem->normalNumber = va_arg(itemlist, int) -1;
488      this->currentGroup->currentFace->vertexCount++;
489    }
490  va_end(itemlist);
491
492  this->currentGroup->faceCount += this->currentGroup->currentFace->vertexCount - 2;
[3400]493}
494
495/**
[2842]496   \brief parses a vertexNormal-String
[3186]497   \param normalString The String that will be parsed.
498
[2842]499   If a vertexNormal line is found this function will inject it into the vertexNormal-Array
500*/
[3396]501bool Model::addVertexNormal (char* normalString)
[2794]502{
[3071]503  float subbuffer1;
504  float subbuffer2;
505  float subbuffer3;
506  sscanf (normalString, "%f %f %f", &subbuffer1, &subbuffer2, &subbuffer3);
[3207]507  PRINTF(3)("found vertex-Normal %f, %f, %f\n", &subbuffer1,&subbuffer2,&subbuffer3);
[3195]508  this->normals->addEntry(subbuffer1, subbuffer2, subbuffer3);
[2794]509  return true;
510}
511
[2842]512/**
[3400]513   \brief adds a VertexNormal.
514   \param x The x coordinate of the Normal.
515   \param y The y coordinate of the Normal.
516   \param z The z coordinate of the Normal.
517
518   If a vertexNormal line is found this function will inject it into the vertexNormal-Array
519*/
520bool Model::addVertexNormal(const float x, const float y, const float z)
521{
522  PRINTF(3)("found vertex-Normal %f, %f, %f\n", x, y, z);
523  this->normals->addEntry(x, y, z);
524}
525
526/**
[2842]527   \brief parses a vertexTextureCoordinate-String
528   \param vTextureString The String that will be parsed.
[3186]529
530   If a vertexTextureCoordinate line is found,
531   this function will inject it into the vertexTexture-Array
[2842]532*/
[3396]533bool Model::addVertexTexture (char* vTextureString)
[2820]534{
[3071]535  float subbuffer1;
536  float subbuffer2;
537  sscanf (vTextureString, "%f %f", &subbuffer1, &subbuffer2);
[3207]538  PRINTF(3)("found vertex-Texture %f, %f\n", &subbuffer1, &subbuffer2);
[3195]539  this->vTexture->addEntry(subbuffer1);
540  this->vTexture->addEntry(subbuffer2);
[2820]541  return true;
542}
543
[3066]544/**
[3400]545   \brief adds a Texture Coordinate
546   \param u The u coordinate of the TextureCoordinate.
547   \param v The y coordinate of the TextureCoordinate.
548
549   If a TextureCoordinate line is found this function will inject it into the TextureCoordinate-Array
550*/
551bool Model::addVertexTexture(const float u, const float v)
552{
553  PRINTF(3)("found vertex-Texture %f, %f\n", u, v);
554  this->vTexture->addEntry(u);
555  this->vTexture->addEntry(v);
556}
557
558/**
[3066]559   \brief Function that selects a material, if changed in the obj file.
560   \param matString the Material that will be set.
561*/
[3396]562bool Model::addUseMtl (char* matString)
[3063]563{
[3396]564  /*
[3195]565  if (!this->mtlFileName)
[3066]566    {
[3207]567      PRINTF(4)("Not using new defined material, because no mtlFile found yet\n");
[3066]568      return false;
569    }
[3396]570  */     
[3195]571  if (this->currentGroup->faceCount >0)
572    this->currentGroup->currentFace = this->currentGroup->currentFace->next = new Face;
573  this->initFace (this->currentGroup->currentFace);
[3063]574 
[3195]575  this->currentGroup->currentFace->materialString = new char[strlen(matString)+1];
576  strcpy (this->currentGroup->currentFace->materialString, matString);
[3063]577 
[3195]578  if (this->currentGroup->faceCount == 0)
579    this->currentGroup->faceCount ++;
[3063]580
581}
582
[3066]583/**
584   \brief reads and includes the Faces/Materials into the openGL state Machine
585*/
[3360]586bool Model::importToGL (void)
[3063]587{
588
589  // finalize the Arrays
[3195]590  this->vertices->finalizeArray();
591  this->vTexture->finalizeArray();
[3426]592  if (normals->getCount() == 0) // vertices-Array must be built for this
[3195]593    this->buildVertexNormals();
594  this->normals->finalizeArray();
[3063]595
[3195]596  this->currentGroup = this->firstGroup;
[3063]597
[3195]598  while (this->currentGroup != NULL)
[3063]599    {
600
601      // creating a glList for the Group
[3195]602      if ((this->currentGroup->listNumber = glGenLists(1)) == 0)
[3063]603        {
[3360]604          PRINTF(1)("list could not be created for this Model\n");
[3063]605          return false;
606        }
[3195]607      glNewList (this->currentGroup->listNumber, GL_COMPILE);
[3063]608
609      // Putting Faces to GL
[3195]610      Face* tmpFace = this->currentGroup->firstFace;
[3063]611      while (tmpFace != NULL)
612        {
[3065]613          if (tmpFace->vertexCount == 0 && tmpFace->materialString != NULL)
[3063]614            {
[3195]615              if (this->currentGroup->faceMode != -1)
[3065]616                glEnd();
[3195]617              this->currentGroup->faceMode = 0;
[3069]618              Material* tmpMat;
619              if ((tmpMat = material->search(tmpFace->materialString)) != NULL)
[3396]620                {
621                  tmpMat->select();
622                  PRINTF(2)("using material %s for coming Faces.\n", tmpFace->materialString);
623                }
624              else 
625                PRINTF(1)("material %s not found.\n", tmpFace->materialString);
[3065]626
[3396]627
[3065]628            }
629
630          else if (tmpFace->vertexCount == 3)
631            {
[3195]632              if (this->currentGroup->faceMode != 3)
[3063]633                {
[3195]634                  if (this->currentGroup->faceMode != -1)
[3063]635                    glEnd();
636                  glBegin(GL_TRIANGLES);
637                }
638             
[3195]639              this->currentGroup->faceMode = 3;
[3207]640              PRINTF(3)("found triag.\n");
[3063]641            }
642         
643          else if (tmpFace->vertexCount == 4)
644            {
[3195]645              if (this->currentGroup->faceMode != 4)
[3063]646                {
[3195]647                  if (this->currentGroup->faceMode != -1)
[3063]648                    glEnd();
649                  glBegin(GL_QUADS);
650                }
[3195]651              this->currentGroup->faceMode = 4;
[3207]652              PRINTF(3)("found quad.\n");
[3063]653            }
654         
655          else if (tmpFace->vertexCount > 4)
656            {
[3195]657              if (this->currentGroup->faceMode != -1)
[3063]658                glEnd();
659              glBegin(GL_POLYGON);
[3207]660              PRINTF(3)("Polygon with %i faces found.", tmpFace->vertexCount);
[3195]661              this->currentGroup->faceMode = tmpFace->vertexCount;
[3063]662            }
663         
664          FaceElement* tmpElem = tmpFace->firstElem;
665          while (tmpElem != NULL)
666            {
[3207]667              //      PRINTF(2)("%s\n", tmpElem->value);
[3195]668              this->addGLElement(tmpElem);
[3063]669              tmpElem = tmpElem->next;
670            }
671          tmpFace = tmpFace->next;
672        }
673      glEnd();
674      glEndList();
[3195]675
676      this->currentGroup = this->currentGroup->next;
[3063]677    } 
678}
679
[3066]680/**
681   \brief Adds a Face-element (one vertex of a face) with all its information.
[3186]682   \param elem The FaceElement to add to the OpenGL-environment.
683
[3066]684   It does this by searching:
685   1. The Vertex itself
686   2. The VertexNormale
687   3. The VertexTextureCoordinate
688   merging this information, the face will be drawn.
[2842]689*/
[3360]690bool Model::addGLElement (FaceElement* elem)
[2776]691{
[3207]692  PRINTF(3)("importing grafical Element to openGL.\n");
[3066]693
[3073]694  if (elem->texCoordNumber != -1)
[3195]695    glTexCoord2fv(this->vTexture->getArray() + elem->texCoordNumber * 2);
[3073]696  if (elem->normalNumber != -1)
[3195]697    glNormal3fv(this->normals->getArray() + elem->normalNumber * 3);
[3073]698  if (elem->vertexNumber != -1)
[3195]699    glVertex3fv(this->vertices->getArray() + elem->vertexNumber * 3);
[2769]700
[2776]701}
702
[3079]703/**
704   \brief A routine that is able to create normals.
[3186]705
[3079]706   The algorithm does the following:
707   1. It calculates creates Vectors for each normale, and sets them to zero.
708   2. It then Walks through a) all the Groups b) all the Faces c) all the FaceElements
709   3. It searches for a points two neighbours per Face, takes Vecotrs to them calculates FaceNormals and adds it to the Points Normal.
710   4. It goes through all the normale-Points and calculates the VertexNormale and includes it in the normals-Array.
711*/
[3360]712bool Model::buildVertexNormals ()
[3075]713{
[3081]714 
[3207]715  PRINTF(2)("Normals are being calculated.\n");
[3075]716
717  Vector* normArray = new Vector [vertices->getCount()/3];
[3079]718  for (int i=0; i<vertices->getCount()/3;i++)
[3075]719    normArray[i] = Vector(.0,.0,.0);
[3081]720 
[3075]721  int firstTouch;
722  int secondTouch;
723  Vector prevV;
724  Vector nextV;
725  Vector curV;
726
727  Group* tmpGroup = firstGroup;
728  while (tmpGroup)
729    {
730      Face* tmpFace = tmpGroup->firstFace;
731      while (tmpFace)
732        {
733          if (tmpFace->firstElem)
734            {
735              FaceElement* firstElem = tmpFace->firstElem;
[3082]736              FaceElement* prevElem;
737              FaceElement* curElem = firstElem;
738              FaceElement* nextElem;
[3081]739              FaceElement* lastElem;
[3082]740              // find last Element of the Chain. !! IMPORTANT:the last Element of the Chain must point to NULL, or it will resolv into an infinity-loop.
[3081]741              while (curElem)
742                {
743                  prevElem = curElem;
744                  curElem = curElem->next;
745                }
746              lastElem = prevElem;
[3075]747             
[3081]748              curElem = firstElem;
749              for (int j=0; j<tmpFace->vertexCount; j++)
[3075]750                {
[3082]751                  if (!(nextElem = curElem->next))
752                    nextElem = firstElem;
[3075]753                  curElem->normalNumber = curElem->vertexNumber;
754                 
755                  curV = Vector (vertices->getArray()[curElem->vertexNumber*3], vertices->getArray()[curElem->vertexNumber*3+1], vertices->getArray()[curElem->vertexNumber*3+2]);
756                  prevV = Vector (vertices->getArray()[prevElem->vertexNumber*3], vertices->getArray()[prevElem->vertexNumber*3+1], vertices->getArray()[prevElem->vertexNumber*3+2]) - curV;
[3082]757                  nextV = Vector (vertices->getArray()[nextElem->vertexNumber*3], vertices->getArray()[nextElem->vertexNumber*3+1], vertices->getArray()[nextElem->vertexNumber*3+2]) - curV;
[3081]758                  normArray[curElem->vertexNumber] = normArray[curElem->vertexNumber] + nextV.cross(prevV);
[3075]759
760                  prevElem = curElem;
761                  curElem = curElem->next;
762                }
763            }
764          tmpFace = tmpFace->next;
765        }
766      tmpGroup = tmpGroup->next;
767    }
768
769  for (int i=0; i<vertices->getCount()/3;i++)
770    {
771      normArray[i].normalize();
[3207]772      PRINTF(3)("Found Normale number %d: (%f; %f, %f).\n", i, normArray[i].x, normArray[i].y, normArray[i].z);
[3426]773     
[3195]774      this->normals->addEntry(normArray[i].x, normArray[i].y, normArray[i].z);
[3075]775
776    }
[3080]777  delete []normArray; 
[3081]778 
[3075]779}
780
781
[2842]782/**
[3360]783   \brief Includes a default model
[3186]784
[3360]785   This will inject a Cube, because this is the most basic model.
[2842]786*/
[3400]787void Model::cubeModel(void)
[2821]788{
[3396]789  this->addVertex ("-0.5 -0.5 0.5");
790  this->addVertex ("0.5 -0.5 0.5");
791  this->addVertex ("-0.5 0.5 0.5");
792  this->addVertex ("0.5 0.5 0.5");
793  this->addVertex ("-0.5 0.5 -0.5");
794  this->addVertex ("0.5 0.5 -0.5");
795  this->addVertex ("-0.5 -0.5 -0.5");
796  this->addVertex ("0.5 -0.5 -0.5");
[2967]797
[3396]798  this->addVertexTexture ("0.0 0.0");
799  this->addVertexTexture ("1.0 0.0");
800  this->addVertexTexture ("0.0 1.0");
801  this->addVertexTexture ("1.0 1.0");
802  this->addVertexTexture ("0.0 2.0");
803  this->addVertexTexture ("1.0 2.0");
804  this->addVertexTexture ("0.0 3.0");
805  this->addVertexTexture ("1.0 3.0");
806  this->addVertexTexture ("0.0 4.0");
807  this->addVertexTexture ("1.0 4.0");
808  this->addVertexTexture ("2.0 0.0");
809  this->addVertexTexture ("2.0 1.0");
810  this->addVertexTexture ("-1.0 0.0");
811  this->addVertexTexture ("-1.0 1.0");
[3081]812
[3396]813  this->addVertexNormal ("0.0 0.0 1.0");
814  this->addVertexNormal ("0.0 0.0 1.0");
815  this->addVertexNormal ("0.0 0.0 1.0");
816  this->addVertexNormal ("0.0 0.0 1.0");
817  this->addVertexNormal ("0.0 1.0 0.0");
818  this->addVertexNormal ("0.0 1.0 0.0");
819  this->addVertexNormal ("0.0 1.0 0.0");
820  this->addVertexNormal ("0.0 1.0 0.0");
821  this->addVertexNormal ("0.0 0.0 -1.0");
822  this->addVertexNormal ("0.0 0.0 -1.0");
823  this->addVertexNormal ("0.0 0.0 -1.0");
824  this->addVertexNormal ("0.0 0.0 -1.0");
825  this->addVertexNormal ("0.0 -1.0 0.0");
826  this->addVertexNormal ("0.0 -1.0 0.0");
827  this->addVertexNormal ("0.0 -1.0 0.0");
828  this->addVertexNormal ("0.0 -1.0 0.0");
829  this->addVertexNormal ("1.0 0.0 0.0");
830  this->addVertexNormal ("1.0 0.0 0.0");
831  this->addVertexNormal ("1.0 0.0 0.0");
832  this->addVertexNormal ("1.0 0.0 0.0");
833  this->addVertexNormal ("-1.0 0.0 0.0");
834  this->addVertexNormal ("-1.0 0.0 0.0");
835  this->addVertexNormal ("-1.0 0.0 0.0");
836  this->addVertexNormal ("-1.0 0.0 0.0");
[2821]837
[3083]838  /* normaleLess-testingMode
[3396]839  this->addFace ("1 2 4 3");
840  this->addFace ("3 4 6 5");
841  this->addFace ("5 6 8 7");
842  this->addFace ("7 8 2 1");
843  this->addFace ("2 8 6 4");
844  this->addFace ("7 1 3 5");
[3083]845  */
[3081]846
[3396]847  this->addFace ("1/1/1 2/2/2 4/4/3 3/3/4");
848  this->addFace ("3/3/5 4/4/6 6/6/7 5/5/8");
849  this->addFace ("5/5/9 6/6/10 8/8/11 7/7/12");
850  this->addFace ("7/7/13 8/8/14 2/10/15 1/9/16");
851  this->addFace ("2/2/17 8/11/18 6/12/19 4/4/20");
852  this->addFace ("7/13/21 1/1/22 3/3/23 5/14/24");
[3083]853
[2821]854}
[3400]855
856
[3418]857void Model::sphereModel()
858{
859  int detail = 30;
860  if (detail <= 0) detail = 1;
861  float df = (float)detail;
862  for (float i = 0.0; i < df/2; i+=1.0)
863    {
864      for (float j = 0.0; j < df; j+=1.0)
865      {
866        float vz = i/df *2.0*PI - PI/2.0;
867        this->addVertex(cos(j/df*2.0*PI) * cos(vz) ,
868                        sin(j/df*2.0*PI) * cos(vz),
869                        sin(vz));
870        //if (j==0.0)
871        //printf ("%f %f\n", vz, sin (vz));
872        if (i==0.0)
873                  printf("%f, %f\n", j/df*2.0*PI, cos(j/df*PI));
874      }
875    }
876  vertices->debug();
877  for (int i = 0; i < detail/2; i++)
878    for (int j = 1; j < detail; j++)
879      {
880        unsigned int v1,v2,v3,v4;
881        v1 = i*detail +j;
882
883        /*      if (j+1 == detail)
884          {
885            v2 = i*detail +1;
886            v3 = i*detail+detail + 1;
887          }
888          else*/
889          {
890            v2 = i*detail +j+1;
891            v3 = i*detail+detail + j+1;
892          }
893        v4 = i*detail+detail + j;
894        //printf("%i %i %i %i\n", v1, v2, v3, v4);
895        this->addFace(4, 0, v1, v2, v3, v4);
896      }
897}
898
[3426]899/**
900   \brief Creates a Cylinder.
901*/
902void Model::cylinderModel(void)
[3400]903{
[3426]904  unsigned int detail = 20;
905  float size = 1.0;
906
907  // check if devision by zero
908  if (detail <= 3)
909    detail = 3;
910  int count = 0;
911  // defining Points of the Cylinder.
912  for (float phi = 0.0; phi < 2.0*PI; phi += 2.0*PI/(float)detail)
[3400]913    {
[3426]914      this->addVertex(size*cos(phi), size*sin(phi), -size);
915      this->addVertex(size*cos(phi), size*sin(phi), size);
916      count ++;
[3400]917    }
[3426]918  this->addVertex(0, 0, -size);
919  this->addVertex(0, 0, size);
920
921 
922  if (count != detail)
923    cout << "calculation error, count should be " << detail << " but is " << count << endl;
924  vertices->debug();
925
926  // adding Faces
927  for (int i = 0; i < detail-1; i++)
928    {
929      int p1, p2, p3, p4;
930      p1 = 2*i+1;
931      p2 = 2*i+2;
932      if (i <= detail);
933      p3 = 2*i+4;
934      p4 = 2*i+3;
935      cout <<i+1 <<": "<< p1 <<" "<< p2 <<" "<< p3 <<" "<< p4 <<endl;
936      this->addFace(4, 0, p1, p2, p3, p4);
937      this->addFace(3, 0, p4, p1, 2*detail+1);
938      this->addFace(3, 0, p2, p3, 2*detail+2);
939    }
940  addFace(4,0, 2*detail-1, 2*detail, 2, 1);
941  this->addFace(3, 0, 1, 2*detail-1, 2*detail+1);
942  this->addFace(3, 0, 2*detail, 2, 2*detail+2);
943
[3400]944}
Note: See TracBrowser for help on using the repository browser.