Go to most recent revision | Blame | Compare with Previous | Last modification | View Log | RSS feed
#include "bob_dom_bob.h"//---------------------------------------------------------------------------------// BOB filebool bob_dom_bob::load(unsigned char *data, size_t size){int pos = 0;if ( data[pos++] != hdr_begin )return false;// info sectionif ( data[pos] != bob_dom_info::hdr_begin )return false;return false;}bool bob_dom_bob::load(bob_dom_ibinaryfilestream& is){int hdr;is >> hdr;if(hdr!=hdr_begin){error(e_badHeader);return false;}if(peek(is)==bob_dom_info::hdr_begin){if(!info.load(is)){for(ErrorIterator &it=info.errors.begin(); it!=info.errors.end(); ++it){error(it->code, "info: %s", it->text);}return false;}}if(materials.load(is)==false){for(ErrorIterator &it=materials.errors.begin(); it!=materials.errors.end(); ++it){error(it->code, "materials->%s", it->text);return false;}}/*switch(m_settings->X3BOB()){case Settings::SwitchValue::on:bobType=bobX3;break;case Settings::SwitchValue::off:bobType=bobX2;break;default:if(materials.size() && materials.front()->type >= bob_dom_material::mat6)bobType=bobX3;elsebobType=bobX2;}*/if(bodies.load(is)==false){for(ErrorIterator &it=bodies.errors.begin(); it!=bodies.errors.end(); ++it){error(it->code, "bodies->%s", it->text);}return false;}is >> hdr;if(hdr!=hdr_end)error(e_badEndHeader);return(hdr==hdr_end && !is.fail());}//---------------------------------------------------------------------------------bool bob_dom_bob::toFile(bob_dom_obinaryfilestream& os){os << hdr_begin;info.toFile(os);materials.toFile(os);bodies.toFile(os);os << hdr_end;return os.good();}//---------------------------------------------------------------------------------bool bob_dom_bob::toFile(bob_dom_otextfilestream& os){info.toFile(os);os << endl;materials.toFile(os);if(materials.size())os << endl;bodies.toFile(os, m_settings, &materials);return os.good();}//---------------------------------------------------------------------------------// MATERIAL - objectbool bob_dom_material1::load(bob_dom_ibinaryfilestream& is){is >> index;is >> textureID; // int?is >> ambient;is >> diffuse;is >> specular;errorCode=is.fail() ? e_notEnoughData : e_noError;return !is.fail();}//---------------------------------------------------------------------------------bool bob_dom_material1::toFile(bob_dom_obinaryfilestream& os){os << index << textureID << ambient << diffuse << specular;return os.good();}//---------------------------------------------------------------------------------bool bob_dom_material1::toFile(bob_dom_otextfilestream& os){os << noSemicolons << "MATERIAL";switch(type){case mat3:os << '3';break;case mat5:os << '5';break;}os << ": " << autoSemicolons;os << index << textureID;os << ambient << diffuse << specular;return os.good();}//---------------------------------------------------------------------------------// MATERIAL 3 - objectbool bob_dom_material3::load(bob_dom_ibinaryfilestream& is){base::load(is);if(type > mat1){is >> transparency; // int?is >> selfIllumination;is >> shininess;short s;is >> s;destinationBlend = (s & 0x2) > 0;twoSided = (s & 0x10) > 0;wireframe = (s & 0x8) > 0;is >> textureValue;is >> enviromentMap;is >> bumpMap;}errorCode=is.fail() ? e_notEnoughData : e_noError;return !is.fail();}//---------------------------------------------------------------------------------bool bob_dom_material3::toFile(bob_dom_obinaryfilestream& os){base::toFile(os);os << transparency << selfIllumination << shininess;short s=0;if(destinationBlend) s|=0x2;if(twoSided) s|=0x10;if(wireframe) s|=0x8;os << s;os << textureValue << enviromentMap << bumpMap;return os.good();}//---------------------------------------------------------------------------------bool bob_dom_material3::toFile(bob_dom_otextfilestream& os){base::toFile(os);os << autoSemicolons;os << transparency << selfIllumination << shininess;os << noSemicolons << destinationBlend << ';' << twoSided << ';' << autoSemicolons << wireframe;os << textureValue;os << enviromentMap << bumpMap;return os.good();}//---------------------------------------------------------------------------------// MATERIAL5bool bob_dom_material5::load(bob_dom_ibinaryfilestream& is){base::load(is);if(type > mat3)is >> lightMap;errorCode=is.fail() ? e_notEnoughData : e_noError;return !is.fail();}//---------------------------------------------------------------------------------bool bob_dom_material5::toFile(bob_dom_obinaryfilestream& os){base::toFile(os);os << lightMap;return os.good();}//---------------------------------------------------------------------------------bool bob_dom_material5::toFile(bob_dom_otextfilestream& os){base::toFile(os);if(type > mat3)os << autoSemicolons << lightMap;return os.good();}//---------------------------------------------------------------------------------char * material6_value::m_stringTypes[]={"SPTYPE_LONG","SPTYPE_BOOL","SPTYPE_FLOAT","","","SPTYPE_FLOAT4","","","SPTYPE_STRING"};int material6_value::m_stringTypesCount=9;bob_error_codes material6_value::load(bob_dom_ibinaryfilestream &is){is >> name;if(is.fail()) return e_error;short t;is >> t;type=(Type)t;switch(type){case typeBool: // no breakcase typeLong:is >> val.i;break;case typeFloat:is >> val.f;break;case typeFloat4:is >> val.f4;break;case typeString:is >> val.psz;break;default:return e_unkMaterialValueType;}return is.fail() ? e_error : e_noError;}//---------------------------------------------------------------------------------bool material6_value::toFile(bob_dom_obinaryfilestream& os){os << name << (short) type;switch(type){case typeBool: // no breakcase typeLong:os << val.i;break;case typeFloat:os << val.f;break;case typeFloat4:os << val.f4;break;case typeString:os << val.psz;break;default:return false;}return os.good();}//---------------------------------------------------------------------------------bool material6_value::toFile(bob_dom_otextfilestream& os){int old=os.flags();os << noSemicolons << name << ';';os << typeName() << ';';switch(type){case typeLong:os << val.i << ';';break;case typeBool:os << val.i << ';';break;case typeFloat:os << val.f << ';';break;case typeFloat4:for(int i=0; i < 4; i++){os << val.f4.f[i] << ';';}break;case typeString:os << val.psz << ';';break;default:return false;}os << ' ';os.flags(old);return os.good();}//---------------------------------------------------------------------------------// MATERIAL6_VALUESbob_error_codes bob_dom_material6_values::load(bob_dom_ibinaryfilestream& is){short count;material6_value *v;bob_error_codes res;is >> count;for(int i=0; i < count; i++){v=new material6_value();if((res=v->load(is))!=e_noError) {delete v;return res;}push_back(v);}return is.fail() ? e_error : e_noError;}//---------------------------------------------------------------------------------bool bob_dom_material6_values::toFile(bob_dom_obinaryfilestream& os){os << (short)size();for(iterator &it=begin(); it!=end(); ++it){it->toFile(os);}return os.good();}//---------------------------------------------------------------------------------bool bob_dom_material6_values::toFile(bob_dom_otextfilestream& os){os << (int)size();for(iterator &it=begin(); it!=end(); ++it){it->toFile(os);}return os.good();}//---------------------------------------------------------------------------------// MATERIAL6 - bigbob_error_codes bob_dom_material6::Big::load(bob_dom_ibinaryfilestream& is){bob_error_codes errorCode;is >> technique >> effect;errorCode=values.load(is);return errorCode;}//---------------------------------------------------------------------------------bool bob_dom_material6::Big::toFile(bob_dom_obinaryfilestream& os){os << technique << effect;values.toFile(os);return os.good();}//---------------------------------------------------------------------------------bool bob_dom_material6::Big::toFile(bob_dom_otextfilestream& os){int old=os.flags();os << autoSemicolons << technique << effect;values.toFile(os);os.flags(old);return os.good();}//---------------------------------------------------------------------------------// MATERIAL 6 - smallbob_error_codes bob_dom_material6::Small::load(bob_dom_ibinaryfilestream& is, int flags){is >> textureFile;is >> ambient >> diffuse >> specular;is >> transparency >> selfIllumination >> shininess;short s=flags;//is >> s;destinationBlend = (s & 0x2) > 0;twoSided = (s & 0x10) > 0;wireframe = (s & 0x8) > 0;is >> textureValue >> enviromentMap >> bumpMap >> lightMap;is >> map4 >> map5;return !is.fail() ? e_noError : e_notEnoughData;}//---------------------------------------------------------------------------------bool bob_dom_material6::Small::toFile(bob_dom_obinaryfilestream& os){static char *empty="";os << (textureFile ? textureFile : empty);os << ambient << diffuse << specular;os << transparency << selfIllumination << shininess;os << textureValue << enviromentMap << bumpMap << lightMap;os << map4 << map5;return os.good();}//---------------------------------------------------------------------------------bool bob_dom_material6::Small::toFile(bob_dom_otextfilestream& os){os << (*textureFile==0 ? "NULL" : textureFile);os << ambient << diffuse << specular;os << transparency << selfIllumination << shininess;os << noSemicolons << destinationBlend << ';' << twoSided << ';' << autoSemicolons << wireframe;os << textureValue;os << enviromentMap << bumpMap << lightMap << map4 << map5;return os.good();}//---------------------------------------------------------------------------------// MATERIAL 6bool bob_dom_material6::load(bob_dom_ibinaryfilestream& is){is >> index;is >> flags;if(flags==Big::flag) {big=new Big();errorCode=big->load(is);}else{/*FILE *f=fopen("d:\\mat56.log", "a+");fprintf(f, "%s - flags: %d, offset 0x%x\n", is.name(), flags, is.tell());fclose(f);*/small=new Small();errorCode=small->load(is, flags);}return errorCode==e_noError;}//---------------------------------------------------------------------------------bool bob_dom_material6::toFile(bob_dom_obinaryfilestream& os){bool bRes;os << index << flags;if(big)bRes=big->toFile(os);elsebRes=small->toFile(os);return bRes;}//---------------------------------------------------------------------------------bool bob_dom_material6::toFile(bob_dom_otextfilestream& os){int old=os.flags();os << noSemicolons << "MATERIAL6: "<< autoSemicolons << index << noSemicolons << "0x" << hex << autoSemicolons << flags << dec;if(big)big->toFile(os);else if(small)small->toFile(os);os.flags(old);return os.good();}//---------------------------------------------------------------------------------// MATERIAL - sectionbool bob_dom_materials::load(bob_dom_ibinaryfilestream& is){int hdr;bob_dom_material::materialType type;is >> hdr;switch(hdr){case hdr_mat5_begin:type=bob_dom_material::mat5;break;case hdr_mat6_begin:type=bob_dom_material::mat6;break;default:error(e_badHeader);return false;}int matCount;is >> matCount;bob_dom_material *m;for(int i=0; i < matCount; i++){switch(type) {case bob_dom_material::mat5:m=new bob_dom_material5();break;case bob_dom_material::mat6:m=new bob_dom_material6();break;}if(m->load(is)==false){error(m->errorCode, "material[%d]: %s", i, bob_traslate_error(m->errorCode));delete m;return false;}elsepush_back(m);}is >> hdr;if(hdr!=hdr_end)error(e_badEndHeader);return (hdr==hdr_end && !is.fail());}//---------------------------------------------------------------------------------bool bob_dom_materials::toFile(bob_dom_obinaryfilestream& os){int hdr;if(size() && front()->type==bob_dom_material::mat6)hdr=hdr_mat6_begin;elsehdr=hdr_mat5_begin;os << hdr << (int)size();for(iterator &it=begin(); it!=end(); ++it){it->toFile(os);}os << hdr_end;return os.good();}//---------------------------------------------------------------------------------bool bob_dom_materials::toFile(bob_dom_otextfilestream& os){for(iterator &it=begin(); it!=end(); ++it){it->toFile(os);os << endl;}return os.good();}//---------------------------------------------------------------------------------// BODIES - sectionbool bob_dom_bodies::load(bob_dom_ibinaryfilestream& is){int hdr;is >> hdr;if(hdr!=hdr_begin){error(e_badHeader);return false;}short bodyCount;is >> bodyCount;child_type *ch;for(int i=0; i < bodyCount; i++){ch=createChild();if(ch->load(is)==false){for(ErrorIterator &it=ch->errors.begin(); it!=ch->errors.end(); ++it){error(it->code, "body[%d]->%s", i, it->text);}return false;}}is >> hdr;if(is.fail())error(e_notEnoughData);if(hdr!=hdr_end)error(e_badEndHeader);return hdr==hdr_end;}//---------------------------------------------------------------------------------bool bob_dom_bodies::toFile(bob_dom_obinaryfilestream& os){os << hdr_begin << (short)size();for(iterator &it=begin(); it!=end(); ++it){it->toFile(os);}os << hdr_end;return os.good();}//---------------------------------------------------------------------------------bool bob_dom_bodies::toFile(bob_dom_otextfilestream& os, const Settings *settings, const bob_dom_materials *materials){bool bRes=true;int i=1;os << noSemicolons << "// beginning of bodies (" << (int)size() << ')' << endl;for(iterator &it=begin(); it!=end(); ++it, ++i){bRes&=it->toFile(os, settings, materials, i);os << endl;}os << "// end of bodies" << endl;return bRes;}//---------------------------------------------------------------------------------// POINTS - sectionbool bob_dom_points::load(bob_dom_ibinaryfilestream& is){int hdr;is >> hdr;if(hdr!=hdr_begin){error(e_badHeader);return false;}int pointCount;is >> pointCount;map.create(pointCount);bob_dom_point *ch;for(int i=0; i < pointCount; i++){ch=new bob_dom_point();if(ch->load(is)==false){error(ch->errorCode, "point[%d]: %s", i, bob_traslate_error(ch->errorCode));delete ch;return false;}map.addPoint(ch);}//printf("AVL Points: %d, compares: %d, average comp to find pnt: %d\n", pointCount, map.map2.m_compCount, map.map2.m_compCount / pointCount);//printf("TST: Points: %d, compares: %d, average comp to find pnt: %d\n", pointCount,map.m_compCount, map.m_compCount / pointCount);is >> hdr;if(hdr!=hdr_end)error(e_badEndHeader);return hdr==hdr_end && !is.fail();}//---------------------------------------------------------------------------------bool bob_dom_points::toFile(bob_dom_obinaryfilestream& os){os << hdr_begin;if(new_points.size()){os << (int)new_points.size();for(PointIterator &it=new_points.begin(); it!=new_points.end(); ++it){it->toFile(os);}}else{os << (int)map.pointsSize();for(bob_point_map::iterator &it=map.begin(); it!=map.end(); ++it){it->toFile(os);}}os << hdr_end;return os.good();}//---------------------------------------------------------------------------------bool bob_dom_points::toFile(bob_dom_otextfilestream& os){int oldf=os.flags();int i=0;os << noSemicolons << "// beginning of points (" << (int)map.uniquePointsSize() << ')' << endl << autoSemicolons;bob_dom_point *p;while(p=map.nextUniquePoint()){p->toFile(os, i++);}os << noSemicolons << "-1; -1; -1; // points end" << endl << endl;os.flags(oldf);return os.good();}//---------------------------------------------------------------------------------// POINT - objectdouble bob_dom_point::multiplier=1.52587890625; // == 100000 / 65535bool bob_dom_point::load(bob_dom_ibinaryfilestream& is){int size, v;is >> type;switch(type){case 0x1F:size=11;break;case 0x1B:size=9;break;case 0x19:size=7;break;default:errorCode=e_unkPointHeader;return false;}is >> x >> y >> z;is >> textureCoords.left >> textureCoords.right;for(int i=5; i < size; i++){is >> v;values.push_back(v);}if(is.fail())errorCode=e_notEnoughData;return !is.fail();}//---------------------------------------------------------------------------------bool bob_dom_point::toFile(bob_dom_obinaryfilestream& os){os << type << x << y << z << textureCoords.left << textureCoords.right;for(iterator &it=values.begin(); it!=values.end(); ++it){os << *it;}return os.good();}//---------------------------------------------------------------------------------bool bob_dom_point::toFile(bob_dom_otextfilestream& os, int idx){// autoSemicolons is set by parent (points)int x,y,z;__asm{mov eax, this/* fld multiplierfimul [eax].xfistp x*/fld multiplierfimul [eax].yfistp yfld multiplierfimul [eax].zfistp z}os << x << y << z;os << noSemicolons << "// " << idx << endl << autoSemicolons;return os.good();}//---------------------------------------------------------------------------------// X3 UV recordbool bob_x3uv::load(bob_dom_ibinaryfilestream& is){is >> idx;for(int i=0; i < 6; i++){is >> values[i];}return !is.fail();}//---------------------------------------------------------------------------------bool bob_x3uv::toFile(bob_dom_obinaryfilestream& os){os << idx;for(int i=0; i < 6; i++){os << values[i];}return os.good();}//---------------------------------------------------------------------------------bool bob_x3uv::toFile(bob_dom_otextfilestream& os, int matIdx){int old=os.flags();os << autoSemicolons;os << matIdx << idx;for(int i=0; i < 6; i++){os << values[i];}os.flags(old);return os.good();}//---------------------------------------------------------------------------------int compare_uv(const void *a, const void *b){const bob_x3uv *uv1=*((bob_x3uv**)a), *uv2=*((bob_x3uv**)b);return uv1->idx - uv2->idx;}//---------------------------------------------------------------------------------// X3 uv listbool bob_x3uv_list::load(bob_dom_ibinaryfilestream& is){int count;bob_x3uv *uv;is >> count;reserve(count);for(int i=0; i < count; i++){uv=new bob_x3uv();if(!uv->load(is)){delete uv;return false;}push_back(uv);}sort(compare_uv);return !is.fail();}//---------------------------------------------------------------------------------bool bob_x3uv_list::toFile(bob_dom_obinaryfilestream& os){os << (int) size();for(iterator &it=begin(); it!=end(); ++it){it->toFile(os);}return os.good();}//---------------------------------------------------------------------------------bool bob_x3uv_list::toFile(bob_dom_otextfilestream& os, int matIdx){for(iterator &it=begin(); it!=end(); ++it){it->toFile(os, matIdx);os << endl;}return os.good();}//---------------------------------------------------------------------------------// FACE LISTbool bob_face_list::load(bob_dom_ibinaryfilestream& is, bool x3data){int count;is >> materialIndex >> count;for(int i=0; i < count; i++){bob_dom_face *face=new bob_dom_face();if(face->load(is)==false){error(face->errorCode, "face[%d]: %s", i, bob_traslate_error(face->errorCode));delete face;return false;}push_back(face);}if(x3data && x3uvlist.load(is)==false) return false;return !is.fail();}//---------------------------------------------------------------------------------bool bob_face_list::toFile(bob_dom_obinaryfilestream& os, bool x3data){os << materialIndex << (int)size();for(iterator &it=begin(); it!=end(); ++it){it->toFile(os);}if(x3data)x3uvlist.toFile(os);return os.good();}//---------------------------------------------------------------------------------// PARTS - section PARTbool bob_dom_parts::load(bob_dom_ibinaryfilestream& is){int hdr, count;is >> hdr;if(hdr!=hdr_begin){error(e_badHeader);return false;}is >> count;bob_dom_part *part;for(int i=0; i < count; i++){part=new bob_dom_part();if(!part->load(is)){for(ErrorIterator &it=part->errors.begin(); it!=part->errors.end(); ++it){error(it->severity, it->code, "part[%d]->%s", i, it->text);}delete part;return false;}push_back(part);}is >> hdr;if(hdr!=hdr_end)error(e_badEndHeader);return hdr==hdr_end && !is.fail();}//---------------------------------------------------------------------------------bool bob_dom_parts::toFile(bob_dom_obinaryfilestream& os){os << hdr_begin << (int) size();for(iterator &it=begin(); it!=end(); ++it){it->toFile(os);}os << hdr_end;return os.good();}//---------------------------------------------------------------------------------bool bob_dom_parts::toFile(bob_dom_otextfilestream& os, const Settings *settings, const bob_dom_materials *materials, const bob_point_map *pointMap){int i=1;for(iterator &it=begin(); it!=end(); ++it, ++i){it->toFile(os, settings, materials, pointMap, i);}return os.good();}//---------------------------------------------------------------------------------bool bob_dom_part_x3vals::load(bob_dom_ibinaryfilestream& is){for(int i=0; i < 10; i++){is >> m_values[i];}return !is.fail();}//---------------------------------------------------------------------------------bool bob_dom_part_x3vals::toFile(bob_dom_obinaryfilestream& os){for(int i=0; i < 10; i++){os << m_values[i];}return os.good();}//---------------------------------------------------------------------------------bool bob_dom_part_x3vals::toFile(bob_dom_otextfilestream& os, bool bCommentItself){int old=os.flags();os << noSemicolons;if(bCommentItself)os << "// ";os << "PART_VALUES_RAW: " << autoSemicolons;for(int i=0; i < 10; i++){os << m_values[i];}os << endl;/*os << noSemicolons << "// raw:" << endl;os << "// " << autoSemicolons;for(int i=0; i < 10; i++){os << m_values[i];}os << endl << */os << noSemicolons << "// point:" << endl;os << "// " << autoSemicolons;for(int i=0; i < 10; i++){os << (int)(m_values[i] / bob_dom_point::multiplier);}os << endl << noSemicolons << "// float:" << endl;static const double multiplier=1.52587890625E-05; // 1/65535os << "// " << autoSemicolons;for(int i=0; i < 10; i++){os << (float)(m_values[i] * multiplier);}os << endl;os.flags(old);return os.good();}//---------------------------------------------------------------------------------// PARTS - objectconst double bob_dom_part::multiplier=1.52587890625E-05; // == 1 / 65636bool bob_dom_part::load(bob_dom_ibinaryfilestream& is){short count;is >> flags >> count;for(int i=0; i < count; i++){bob_face_list *faces=new bob_face_list();if(faces->load(is, (flags & x3flag) > 0)==false){for(ErrorIterator &it=faces->errors.begin(); it!=faces->errors.end(); ++it){error(it->severity, it->code, "face list[%d]->%s", i, it->text);}delete faces;return false;}push_back(faces);}if(flags & x3flag)values.load(is);return !is.fail();}//---------------------------------------------------------------------------------bool bob_dom_part::toFile(bob_dom_obinaryfilestream& os){os << flags << (short)size();for(iterator &it=begin(); it!=end(); ++it){it->toFile(os, (flags & x3flag) > 0);}if(flags & x3flag)values.toFile(os);return os.good();}//---------------------------------------------------------------------------------bool bob_dom_part::toFile(bob_dom_otextfilestream& os, const Settings *settings, const bob_dom_materials *materials, const bob_point_map *pointMap, int idx){bob_dom_face *part;const bob_dom_point *pnt;int line=0;int materialIndex;int oldf=os.flags();size_t count=0;for(iterator &it=begin(); it!=end(); ++it){count+=it->size();}os << noSemicolons << "// ----- part " << idx << " (" << (int)count << " faces) -----" << endl << autoSemicolons;for(iterator &it=begin(); it!=end(); ++it){materialIndex=it->materialIndex;for(bob_face_list::iterator &it2=it->begin(); it2!=it->end(); ++it2, ++line){part=*it2;os << autoSemicolons << materialIndex;for(int i=0; i < 3; i++){os << pointMap->uniqueIndex(part->values[i]);}pnt=(*pointMap)[part->values[0]];if(pnt==NULL){error(e_pointNotFound);return false;}int SGBits=pnt->values.back();int magic;/*meaning of "magic" number:-25: smoothing bits and UV coords-17: smoothing bits and no UV coords-9: no smoothing and UV coords-1: no smoothing and no UV coords--obsolete----------------------old text from Checker's sources:if there are no materials used, magic is -17if there are materials and last value of the first point is non zero,magic is -25, otherwise it's -9*/const bob_dom_point *points_ar[3];for(int i=0; i < 3; i++){pnt=(*pointMap)[part->values[i]];if(pnt==NULL){error(e_pointNotFound);return false;}points_ar[i]=pnt;}bool bHasUV=false;for(int i=0; i < 3; i++){if(bHasUV=(points_ar[i]->textureCoords.left!=0 || points_ar[i]->textureCoords.right!=0))break;}if(SGBits) {if(bHasUV)magic=-25;elsemagic=-17;}else{if(bHasUV)magic=-9;elsemagic=-1;}os << magic;if(SGBits!=0) os << SGBits;// we have uv coords so lets output themif(bHasUV){double u, v;for(int i=0; i < 3; i++){u=points_ar[i]->textureCoords.left * multiplier;v=points_ar[i]->textureCoords.right * multiplier;os << (float) u << (float) v;}// those are the unknown values stored in point after texture coordinates// I place them after bod part definition and behind comment, so they are invisible to// bod viewer and x2if(settings->extraPntInfo()){bool bEmpty=true;for(int i=0; i < 3; i++){bEmpty&=(points_ar[i]->values.size() <= 1);}if(!bEmpty){os << noSemicolons << "/! ";for(int i=0; i < 3; i++){os << "{ " << autoSemicolons;bob_dom_point::const_iterator &last=--points_ar[i]->values.end();double d;for(bob_dom_point::const_iterator &it=points_ar[i]->values.begin(); it!=last; ++it){d=*it * multiplier;if(d==0)os << 0;elseos << (float)d;}os << noSemicolons << "} ";}os << "!/ " << autoSemicolons;}}}os << noSemicolons << "// " << line << endl;}}// x3 dataif(flags & x3flag){os << "// x3 part values" << endl;if(settings->extraX3BobInfo())os << "/!" << endl;values.toFile(os, settings->extraX3BobInfo()==false);if(settings->extraX3BobInfo()) {os << endl << "// secondary UVs (or whatever this is)" << endl;for(iterator &it=begin(); it!=end(); ++it){it->x3uvlist.toFile(os, it->materialIndex);}os << "!/" << endl;}}int bit=1;char str[21];for(int i=19; i >= 0; i--){str[i]=flags & bit ? '1' : '0';bit<<=1;}str[sizeof(str)-1]=0;os << autoSemicolons << "-99" << str << noSemicolons << "// part " << idx << " end" << endl;os.flags(oldf);return os.good();}//---------------------------------------------------------------------------------// PART - objectbool bob_dom_face::load(bob_dom_ibinaryfilestream& is){for(int i=0; i < 4; i++){is >> values[i];}if(is.fail())errorCode=e_notEnoughData;return !is.fail();}//---------------------------------------------------------------------------------bool bob_dom_face::toFile(bob_dom_obinaryfilestream& os){for(int i=0; i < 4; i++){os << values[i];}return os.good();}//---------------------------------------------------------------------------------bool bob_dom_body::load(bob_dom_ibinaryfilestream& is){is >> bodySize >> bodyFlags;if(peek(is)==bob_dom_bones::hdr_begin){if(!bones.load(is)){for(ErrorIterator &it=bones.errors.begin(); it!=bones.errors.end(); ++it){error(it->code, "bones: %s", it->text);}return false;}}if(peek(is)==bob_dom_points::hdr_begin){if(points.load(is)==false){for(ErrorIterator &it=points.errors.begin(); it!=points.errors.end(); ++it){error(it->code, "points->%s", it->text);}return false;}}if(peek(is)==bob_dom_weights::hdr_begin){if(!weights.load(is)){for(ErrorIterator &it=weights.errors.begin(); it!=weights.errors.end(); ++it){error(it->code, "weights->%s", it->text);}return false;}}if(peek(is)==bob_dom_parts::hdr_begin){if(parts.load(is)==false){for(ErrorIterator &it=parts.errors.begin(); it!=parts.errors.end(); ++it){error(it->code, "parts->%s", it->text);}return false;}}return !is.fail();}//---------------------------------------------------------------------------------bool bob_dom_body::toFile(bob_dom_obinaryfilestream& os){os << bodySize << bodyFlags;bones.toFile(os);points.toFile(os);weights.toFile(os);parts.toFile(os);return os.good();}//---------------------------------------------------------------------------------bool bob_dom_body::toFile(bob_dom_otextfilestream& os, const Settings *settings, const bob_dom_materials *materials, int idx){os << "// beginning of body " << idx << endl;os << autoSemicolons << bodySize << endl << endl;bones.toFile(os);points.toFile(os);weights.toFile(os, &points.map);parts.toFile(os, settings, materials, &points.map);char str[17]; int bit=1;for(int i=15; i >= 0; i--){str[i]=bodyFlags & bit ? '1' : '0';bit<<=1;}str[sizeof(str)-1]=0;os << "-99" << str << noSemicolons << "// body " << idx << " end" << endl;return os.good();}//---------------------------------------------------------------------------------// BONESbool bob_dom_bones::load(bob_dom_ibinaryfilestream& is){int hdr, count;is >> hdr;if(hdr!=hdr_begin){error(e_badHeader);return false;}is >> count;char *name;for(int i=0; i < count; i++){is >> name;if(name==0){error(e_notEnoughData);delete[] name;return false;}push_back(name);}is >> hdr;if(hdr!=hdr_end){error(e_badEndHeader);return false;}return true;}//---------------------------------------------------------------------------------bool bob_dom_bones::toFile(bob_dom_obinaryfilestream& os){if(size()==0) return true;os << hdr_begin << (int) size();for(iterator &it=begin(); it!=end(); ++it){os << *it;}os << hdr_end;return os.good();}//---------------------------------------------------------------------------------bool bob_dom_bones::toFile(bob_dom_otextfilestream& os){if(size()==0) return true;os << noSemicolons << "BONES: " << autoSemicolons << (int)size() << endl;for(iterator &it=begin(); it!=end(); ++it){os << *it << endl;}os << endl;return os.good();}//---------------------------------------------------------------------------------// WEIGHTS - sectionbool bob_dom_weights::load(bob_dom_ibinaryfilestream& is){int hdr;is >> hdr;if(hdr!=hdr_begin){error(e_badHeader);return false;}int count;is >> count;resize(count);bob_dom_weight *ch;for(int i=0; i < count; i++){ch=new bob_dom_weight();if(!ch->load(is)){error(ch->errorCode, "weight[%d]: %s", i, bob_traslate_error(ch->errorCode));delete ch;return false;}(*this)[i]=ch;}is >> hdr;if(hdr!=hdr_end){error(e_badEndHeader);return false;}return true;}//---------------------------------------------------------------------------------bool bob_dom_weights::toFile(bob_dom_obinaryfilestream& os){if(new_weights.size()==0 && size()==0) return true;os << hdr_begin;if(new_weights.size()){os << (int)new_weights.size();for(WeightList::iterator &it=new_weights.begin(); it!=new_weights.end(); ++it){it->toFile(os);}}else{os << (int)size();for(iterator &it=begin(); it!=end(); ++it){it->toFile(os);}}os << hdr_end;return os.good();}//---------------------------------------------------------------------------------bool bob_dom_weights::toFile(bob_dom_otextfilestream& os, const bob_point_map *pointMap){if(size()==0) return true;os << noSemicolons << "WEIGHTS " << autoSemicolons << (int) pointMap->uniquePointsSize() << endl;int i=0;for(bob_point_map::const_index_iterator &it=pointMap->indexBegin(); it!=pointMap->indexEnd(); ++it){(*this)[*it]->toFile(os, i++);}os << autoSemicolons << "-1" << endl << endl;return os.good();}//---------------------------------------------------------------------------------// WEIGHT - objectbool bob_dom_weight::load(bob_dom_ibinaryfilestream& is){short count;value v;is >> count;for(int i=0; i < count; i++){is >> v.boneIdx >> v.boneCoefficient;values.push_back(v);}return !is.fail();}//---------------------------------------------------------------------------------bool bob_dom_weight::toFile(bob_dom_obinaryfilestream& os){os << (short) values.size();value v;for(iterator &it=values.begin(); it!=values.end(); ++it){v=*it;os << v.boneIdx << v.boneCoefficient;}return os.good();}//---------------------------------------------------------------------------------bool bob_dom_weight::toFile(bob_dom_otextfilestream& os, int idx){os << autoSemicolons;for(iterator &it=values.begin(); it!=values.end(); ++it){(*it).toFile(os);}os << noSemicolons << "-1; // " << idx << endl;return os.good();}//---------------------------------------------------------------------------------