LCOV - code coverage report
Current view: top level - engine/model - animmodel.cpp (source / functions) Hit Total Coverage
Test: Libprimis Test Coverage Lines: 144 1033 13.9 %
Date: 2025-01-07 07:51:37 Functions: 34 111 30.6 %

          Line data    Source code
       1             : /* animmodel.cpp: implementation for animated models
       2             :  *
       3             :  * animmodel.cpp implements the animmodel object in animmodel.h and is the
       4             :  * working form of an an animated model loaded from a file. The animmodel object
       5             :  * supports skeletal animation along with the basics it inherits from model.h;
       6             :  * see that file for non-animated functionality (e.g. normal, specular, etc mapping,
       7             :  * position, orientation, etc).
       8             :  *
       9             :  */
      10             : #include "../libprimis-headers/cube.h"
      11             : #include "../../shared/geomexts.h"
      12             : #include "../../shared/glemu.h"
      13             : #include "../../shared/glexts.h"
      14             : 
      15             : #include <memory>
      16             : #include <optional>
      17             : 
      18             : #include "interface/console.h"
      19             : #include "interface/control.h"
      20             : 
      21             : #include "render/radiancehints.h"
      22             : #include "render/rendergl.h"
      23             : #include "render/renderlights.h"
      24             : #include "render/rendermodel.h"
      25             : #include "render/renderparticles.h"
      26             : #include "render/shader.h"
      27             : #include "render/shaderparam.h"
      28             : #include "render/texture.h"
      29             : 
      30             : #include "world/entities.h"
      31             : #include "world/bih.h"
      32             : 
      33             : #include "model.h"
      34             : #include "ragdoll.h"
      35             : #include "animmodel.h"
      36             : 
      37             : //animmodel
      38             : VARP(fullbrightmodels, 0, 0, 200); //sets minimum amount of brightness for a model: 200 is 100% brightness
      39             : VAR(testtags, 0, 0, 1);            //not used by animmodel object, used by children vert/skelmodel
      40           0 : VARF(debugcolmesh, 0, 0, 1,
      41             : {
      42             :     cleanupmodels();
      43             : });
      44             : 
      45             : VAR(animationinterpolationtime, 0, 200, 1000);
      46             : 
      47             : std::unordered_map<std::string, animmodel::meshgroup *> animmodel::meshgroups;
      48             : 
      49             : bool animmodel::enabletc = false,
      50             :      animmodel::enabletangents = false,
      51             :      animmodel::enablebones = false,
      52             :      animmodel::enablecullface = true,
      53             :      animmodel::enabledepthoffset = false;
      54             : 
      55             : float animmodel::sizescale = 1;
      56             : 
      57             : vec4<float> animmodel::colorscale(1, 1, 1, 1);
      58             : 
      59             : GLuint animmodel::lastvbuf = 0,
      60             :        animmodel::lasttcbuf = 0,
      61             :        animmodel::lastxbuf = 0,
      62             :        animmodel::lastbbuf = 0,
      63             :        animmodel::lastebuf = 0;
      64             : 
      65             : const Texture *animmodel::lasttex = nullptr,
      66             :               *animmodel::lastdecal = nullptr,
      67             :               *animmodel::lastmasks = nullptr,
      68             :               *animmodel::lastnormalmap = nullptr;
      69             : 
      70             : std::stack<matrix4> animmodel::matrixstack;
      71             : 
      72             : template <>
      73             : struct std::hash<animmodel::shaderparams>
      74             : {
      75          11 :     size_t operator()(const animmodel::shaderparams &k) const
      76             :     {
      77          11 :         return memhash(&k, sizeof(k));
      78             :     }
      79             : };
      80             : 
      81             : std::unordered_map<animmodel::shaderparams, animmodel::skin::ShaderParamsKey> animmodel::skin::ShaderParamsKey::keys;
      82             : 
      83             : int animmodel::skin::ShaderParamsKey::firstversion = 0,
      84             :     animmodel::skin::ShaderParamsKey::lastversion = 1;
      85             : 
      86             : //animmodel
      87             : 
      88          20 : animmodel::animmodel(std::string name) : model(std::move(name))
      89             : {
      90          20 : }
      91             : 
      92          20 : animmodel::~animmodel()
      93             : {
      94          39 :     for(part * i : parts)
      95             :     {
      96          19 :         delete i;
      97             :     }
      98          20 :     parts.clear();
      99          20 : }
     100             : 
     101             : // AnimPos
     102             : 
     103           0 : void animmodel::AnimPos::setframes(const animinfo &info)
     104             : {
     105           0 :     anim = info.anim;
     106           0 :     if(info.range<=1)
     107             :     {
     108           0 :         fr1 = 0;
     109           0 :         t = 0;
     110             :     }
     111             :     else
     112             :     {
     113           0 :         int time = info.anim & Anim_SetTime ? info.basetime : lastmillis - info.basetime;
     114           0 :         fr1 = static_cast<int>(time/info.speed); // round to full frames
     115           0 :         t = (time-fr1*info.speed)/info.speed; // progress of the frame, value from 0.0f to 1.0f
     116             :     }
     117           0 :     if(info.anim & Anim_Loop)
     118             :     {
     119           0 :         fr1 = fr1%info.range+info.frame;
     120           0 :         fr2 = fr1+1;
     121           0 :         if(fr2>=info.frame+info.range)
     122             :         {
     123           0 :             fr2 = info.frame;
     124             :         }
     125             :     }
     126             :     else
     127             :     {
     128           0 :         fr1 = std::min(fr1, info.range-1)+info.frame;
     129           0 :         fr2 = std::min(fr1+1, info.frame+info.range-1);
     130             :     }
     131           0 :     if(info.anim & Anim_Reverse)
     132             :     {
     133           0 :         fr1 = (info.frame+info.range-1)-(fr1-info.frame);
     134           0 :         fr2 = (info.frame+info.range-1)-(fr2-info.frame);
     135             :     }
     136           0 : }
     137             : 
     138           6 : bool animmodel::AnimPos::operator==(const AnimPos &a) const
     139             : {
     140           6 :     return fr1==a.fr1 && fr2==a.fr2 && (fr1==fr2 || t==a.t);
     141             : }
     142           6 : bool animmodel::AnimPos::operator!=(const AnimPos &a) const
     143             : {
     144           6 :     return fr1!=a.fr1 || fr2!=a.fr2 || (fr1!=fr2 && t!=a.t);
     145             : }
     146             : 
     147             : // AnimState
     148             : 
     149           3 : bool animmodel::AnimState::operator==(const AnimState &a) const
     150             : {
     151           3 :     return cur==a.cur && (interp<1 ? interp==a.interp && prev==a.prev : a.interp>=1);
     152             : }
     153           3 : bool animmodel::AnimState::operator!=(const AnimState &a) const
     154             : {
     155           3 :     return cur!=a.cur || (interp<1 ? interp!=a.interp || prev!=a.prev : a.interp<1);
     156             : }
     157             : 
     158             : // ShaderParamsKey
     159             : 
     160           0 : bool animmodel::skin::ShaderParamsKey::checkversion()
     161             : {
     162           0 :     if(version >= firstversion)
     163             :     {
     164           0 :         return true;
     165             :     }
     166           0 :     version = lastversion;
     167           0 :     if(++lastversion <= 0)
     168             :     {
     169           0 :         for(auto &[k, t] : keys)
     170             :         {
     171           0 :             t.version = -1;
     172             :         }
     173           0 :         firstversion = 0;
     174           0 :         lastversion = 1;
     175           0 :         version = 0;
     176             :     }
     177           0 :     return false;
     178             : }
     179             : 
     180             : //skin
     181             : 
     182           0 : bool animmodel::skin::masked() const
     183             : {
     184           0 :     return masks != notexture;
     185             : }
     186             : 
     187           0 : bool animmodel::skin::bumpmapped() const
     188             : {
     189           0 :     return normalmap != nullptr;
     190             : }
     191             : 
     192           1 : bool animmodel::skin::alphatested() const
     193             : {
     194           1 :     return alphatest > 0 && tex->type&Texture::ALPHA;
     195             : }
     196             : 
     197           0 : bool animmodel::skin::decaled() const
     198             : {
     199           0 :     return decal != nullptr;
     200             : }
     201             : 
     202          11 : void animmodel::skin::setkey()
     203             : {
     204          11 :     key = &ShaderParamsKey::keys[*this];
     205          11 : }
     206             : 
     207           0 : void animmodel::skin::setshaderparams(Mesh &m, const AnimState *as, bool skinned)
     208             : {
     209           0 :     if(!Shader::lastshader)
     210             :     {
     211           0 :         return;
     212             :     }
     213           0 :     if(key->checkversion() && Shader::lastshader->owner == key)
     214             :     {
     215           0 :         return;
     216             :     }
     217           0 :     Shader::lastshader->owner = key;
     218             : 
     219           0 :     LOCALPARAMF(texscroll, scrollu*lastmillis/1000.0f, scrollv*lastmillis/1000.0f);
     220           0 :     if(alphatested())
     221             :     {
     222           0 :         LOCALPARAMF(alphatest, alphatest);
     223             :     }
     224             : 
     225           0 :     if(!skinned)
     226             :     {
     227           0 :         return;
     228             :     }
     229             : 
     230           0 :     if(color.r() < 0)
     231             :     {
     232           0 :         LOCALPARAM(colorscale, colorscale);
     233             :     }
     234             :     else
     235             :     {
     236           0 :         LOCALPARAMF(colorscale, color.r(), color.g(), color.b(), colorscale.a());
     237             :     }
     238           0 :     if(fullbright)
     239             :     {
     240           0 :         LOCALPARAMF(fullbright, 0.0f, fullbright);
     241             :     }
     242             :     else
     243             :     {
     244           0 :         LOCALPARAMF(fullbright, 1.0f, as->cur.anim & Anim_FullBright ? 0.5f * fullbrightmodels / 100.0f : 0.0f);
     245             :     }
     246           0 :     float curglow = glow;
     247           0 :     if(glowpulse > 0)
     248             :     {
     249           0 :         float curpulse = lastmillis*glowpulse;
     250           0 :         curpulse -= std::floor(curpulse);
     251           0 :         curglow += glowdelta*2*std::fabs(curpulse - 0.5f);
     252             :     }
     253           0 :     LOCALPARAMF(maskscale, spec, gloss, curglow);
     254             : }
     255             : 
     256           0 : Shader *animmodel::skin::loadshader()
     257             : {
     258           0 :     if(shadowmapping == ShadowMap_Reflect)
     259             :     {
     260           0 :         if(rsmshader)
     261             :         {
     262           0 :             return rsmshader;
     263             :         }
     264           0 :         std::string opts;
     265           0 :         if(alphatested())
     266             :         {
     267           0 :             opts.push_back('a');
     268             :         }
     269           0 :         if(!cullface)
     270             :         {
     271           0 :             opts.push_back('c');
     272             :         }
     273             : 
     274           0 :         DEF_FORMAT_STRING(name, "rsmmodel%s", opts.c_str());
     275           0 :         rsmshader = generateshader(name, "rsmmodelshader \"%s\"", opts.c_str());
     276           0 :         return rsmshader;
     277           0 :     }
     278           0 :     if(shader)
     279             :     {
     280           0 :         return shader;
     281             :     }
     282           0 :     std::string opts;
     283           0 :     if(alphatested())
     284             :     {
     285           0 :         opts.push_back('a');
     286             :     }
     287           0 :     if(decaled())
     288             :     {
     289           0 :         opts.push_back(decal->type&Texture::ALPHA ? 'D' : 'd');
     290             :     }
     291           0 :     if(bumpmapped())
     292             :     {
     293           0 :         opts.push_back('n');
     294             :     }
     295           0 :     else if(masked())
     296             :     {
     297           0 :         opts.push_back('m');
     298             :     }
     299           0 :     if(!cullface)
     300             :     {
     301           0 :         opts.push_back('c');
     302             :     }
     303             : 
     304           0 :     DEF_FORMAT_STRING(name, "model%s", opts.c_str());
     305           0 :     shader = generateshader(name, "modelshader \"%s\"", opts.c_str());
     306           0 :     return shader;
     307           0 : }
     308             : 
     309           0 : void animmodel::skin::cleanup()
     310             : {
     311           0 :     if(shader && shader->standard)
     312             :     {
     313           0 :         shader = nullptr;
     314             :     }
     315           0 : }
     316             : 
     317           1 : void animmodel::skin::preloadBIH() const
     318             : {
     319           1 :     if(alphatested() && !tex->alphamask)
     320             :     {
     321           0 :         tex->loadalphamask();
     322             :     }
     323           1 : }
     324             : 
     325           0 : void animmodel::skin::preloadshader()
     326             : {
     327           0 :     loadshader();
     328           0 :     useshaderbyname(alphatested() && owner->model->alphashadow ? "alphashadowmodel" : "shadowmodel");
     329           0 :     if(useradiancehints())
     330             :     {
     331           0 :         useshaderbyname(alphatested() ? "rsmalphamodel" : "rsmmodel");
     332             :     }
     333           0 : }
     334             : 
     335           0 : void animmodel::skin::setshader(Mesh &m, const AnimState *as, bool usegpuskel, int vweights)
     336             : {
     337           0 :     m.setshader(loadshader(), usegpuskel, vweights, gbuf.istransparentlayer());
     338           0 : }
     339             : 
     340           0 : void animmodel::skin::bind(Mesh &b, const AnimState *as, bool usegpuskel, int vweights)
     341             : {
     342           0 :     if(cullface > 0)
     343             :     {
     344           0 :         if(!enablecullface)
     345             :         {
     346           0 :             glEnable(GL_CULL_FACE);
     347           0 :             enablecullface = true;
     348             :         }
     349             :     }
     350           0 :     else if(enablecullface)
     351             :     {
     352           0 :         glDisable(GL_CULL_FACE);
     353           0 :         enablecullface = false;
     354             :     }
     355             : 
     356           0 :     if(as->cur.anim & Anim_NoSkin)
     357             :     {
     358           0 :         if(alphatested() && owner->model->alphashadow)
     359             :         {
     360           0 :             if(tex!=lasttex)
     361             :             {
     362           0 :                 glBindTexture(GL_TEXTURE_2D, tex->id);
     363           0 :                 lasttex = tex;
     364             :             }
     365             :             static Shader *alphashadowmodelshader = nullptr;
     366           0 :             if(!alphashadowmodelshader)
     367             :             {
     368           0 :                 alphashadowmodelshader = useshaderbyname("alphashadowmodel");
     369             :             }
     370           0 :             b.setshader(alphashadowmodelshader, usegpuskel, vweights);
     371           0 :             setshaderparams(b, as, false);
     372             :         }
     373             :         else
     374             :         {
     375             :             static Shader *shadowmodelshader = nullptr;
     376           0 :             if(!shadowmodelshader)
     377             :             {
     378           0 :                 shadowmodelshader = useshaderbyname("shadowmodel");
     379             :             }
     380           0 :             b.setshader(shadowmodelshader, usegpuskel, vweights);
     381             :         }
     382           0 :         return;
     383             :     }
     384           0 :     int activetmu = 0;
     385           0 :     if(tex!=lasttex)
     386             :     {
     387           0 :         glBindTexture(GL_TEXTURE_2D, tex->id);
     388           0 :         lasttex = tex;
     389             :     }
     390           0 :     if(bumpmapped() && normalmap!=lastnormalmap)
     391             :     {
     392           0 :         glActiveTexture(GL_TEXTURE3);
     393           0 :         activetmu = 3;
     394           0 :         glBindTexture(GL_TEXTURE_2D, normalmap->id);
     395           0 :         lastnormalmap = normalmap;
     396             :     }
     397           0 :     if(decaled() && decal!=lastdecal)
     398             :     {
     399           0 :         glActiveTexture(GL_TEXTURE4);
     400           0 :         activetmu = 4;
     401           0 :         glBindTexture(GL_TEXTURE_2D, decal->id);
     402           0 :         lastdecal = decal;
     403             :     }
     404           0 :     if(masked() && masks!=lastmasks)
     405             :     {
     406           0 :         glActiveTexture(GL_TEXTURE1);
     407           0 :         activetmu = 1;
     408           0 :         glBindTexture(GL_TEXTURE_2D, masks->id);
     409           0 :         lastmasks = masks;
     410             :     }
     411           0 :     if(activetmu != 0)
     412             :     {
     413           0 :         glActiveTexture(GL_TEXTURE0);
     414             :     }
     415           0 :     setshader(b, as, usegpuskel, vweights);
     416           0 :     setshaderparams(b, as);
     417             : }
     418             : 
     419           0 : void animmodel::skin::invalidateshaderparams()
     420             : {
     421           0 :     ShaderParamsKey::invalidate();
     422           0 : }
     423             : 
     424             : //Mesh
     425             : 
     426           0 : void animmodel::Mesh::genBIH(const skin &s, std::vector<BIH::mesh> &bih, const matrix4x3 &t)
     427             : {
     428           0 :     bih.emplace_back();
     429           0 :     BIH::mesh &m = bih.back();
     430           0 :     m.xform = t;
     431           0 :     m.tex = s.tex;
     432           0 :     if(canrender)
     433             :     {
     434           0 :         m.flags |= BIH::Mesh_Render;
     435             :     }
     436           0 :     if(cancollide)
     437             :     {
     438           0 :         m.flags |= BIH::Mesh_Collide;
     439             :     }
     440           0 :     if(s.alphatested())
     441             :     {
     442           0 :         m.flags |= BIH::Mesh_Alpha;
     443             :     }
     444           0 :     if(noclip)
     445             :     {
     446           0 :         m.flags |= BIH::Mesh_NoClip;
     447             :     }
     448           0 :     if(s.cullface > 0)
     449             :     {
     450           0 :         m.flags |= BIH::Mesh_CullFace;
     451             :     }
     452           0 :     genBIH(m);
     453           0 :     while(bih.back().numtris > BIH::mesh::maxtriangles)
     454             :     {
     455           0 :         bih.push_back(bih.back());
     456           0 :         BIH::mesh &overflow = bih.back();
     457           0 :         overflow.tris += BIH::mesh::maxtriangles;
     458           0 :         overflow.numtris -= BIH::mesh::maxtriangles;
     459           0 :         bih[bih.size()-2].numtris = BIH::mesh::maxtriangles;
     460             :     }
     461           0 : }
     462             : 
     463         438 : void animmodel::Mesh::fixqtangent(quat &q, float bt)
     464             : {
     465             :     static constexpr float bias = -1.5f/65535;
     466             :     static const float biasscale = sqrtf(1 - bias*bias); //cannot be constexpr, sqrtf is not compile time
     467         438 :     if(bt < 0)
     468             :     {
     469          23 :         if(q.w >= 0)
     470             :         {
     471          21 :             q.neg();
     472             :         }
     473          23 :         if(q.w > bias)
     474             :         {
     475           0 :             q.mul3(biasscale);
     476           0 :             q.w = bias;
     477             :         }
     478             :     }
     479         415 :     else if(q.w < 0)
     480             :     {
     481          71 :         q.neg();
     482             :     }
     483         438 : }
     484             : 
     485             : //meshgroup
     486             : 
     487           4 : animmodel::meshgroup::meshgroup()
     488             : {
     489           4 : }
     490             : 
     491           2 : animmodel::meshgroup::~meshgroup()
     492             : {
     493           2 :     for(Mesh * i : meshes)
     494             :     {
     495           0 :         delete i;
     496             :     }
     497           2 :     meshes.clear();
     498           2 : }
     499             : 
     500           0 : std::vector<std::vector<animmodel::Mesh *>::const_iterator> animmodel::meshgroup::getmeshes(std::string_view meshname) const
     501             : {
     502           0 :     std::vector<std::vector<animmodel::Mesh *>::const_iterator> meshlist;
     503           0 :     for(std::vector<animmodel::Mesh *>::const_iterator i = meshes.begin(); i != meshes.end(); ++i)
     504             :     {
     505           0 :         const animmodel::Mesh &tempmesh = **i;
     506           0 :         if((meshname == "*") || (tempmesh.name.size() && (tempmesh.name == meshname)))
     507             :         {
     508           0 :             meshlist.push_back(i);
     509             :         }
     510             :     }
     511           0 :     return meshlist;
     512           0 : }
     513             : 
     514           8 : std::vector<size_t> animmodel::meshgroup::getskins(std::string_view meshname) const
     515             : {
     516           8 :     std::vector<size_t> skinlist;
     517          16 :     for(uint i = 0; i < meshes.size(); i++)
     518             :     {
     519           8 :         const animmodel::Mesh &m = *(meshes[i]);
     520           8 :         if((meshname == "*") || (m.name.size() && (m.name == meshname)))
     521             :         {
     522           8 :             skinlist.push_back(i);
     523             :         }
     524             :     }
     525           8 :     return skinlist;
     526           0 : }
     527             : 
     528           0 : void animmodel::meshgroup::calcbb(vec &bbmin, vec &bbmax, const matrix4x3 &t) const
     529             : {
     530           0 :     auto rendermeshes = getrendermeshes();
     531           0 :     for(auto i : rendermeshes)
     532             :     {
     533           0 :         (*i)->calcbb(bbmin, bbmax, t);
     534             :     }
     535           0 : }
     536             : 
     537           0 : void animmodel::meshgroup::genBIH(const std::vector<skin> &skins, std::vector<BIH::mesh> &bih, const matrix4x3 &t) const
     538             : {
     539           0 :     for(uint i = 0; i < meshes.size(); i++)
     540             :     {
     541           0 :         meshes[i]->genBIH(skins[i], bih, t);
     542             :     }
     543           0 : }
     544             : 
     545           0 : void animmodel::meshgroup::genshadowmesh(std::vector<triangle> &tris, const matrix4x3 &t) const
     546             : {
     547           0 :     auto rendermeshes = getrendermeshes();
     548           0 :     for(auto i : rendermeshes)
     549             :     {
     550           0 :         (*i)->genshadowmesh(tris, t);
     551             :     }
     552           0 : }
     553             : 
     554           0 : bool animmodel::meshgroup::hasframe(int i) const
     555             : {
     556           0 :     return i>=0 && i<totalframes();
     557             : }
     558             : 
     559           2 : bool animmodel::meshgroup::hasframes(int i, int n) const
     560             : {
     561           2 :     return i>=0 && i+n<=totalframes();
     562             : }
     563             : 
     564           0 : int animmodel::meshgroup::clipframes(int i, int n) const
     565             : {
     566           0 :     return std::min(n, totalframes() - i);
     567             : }
     568             : 
     569           1 : const std::string &animmodel::meshgroup::groupname() const
     570             : {
     571           1 :     return name;
     572             : }
     573             : 
     574           0 : std::vector<std::vector<animmodel::Mesh *>::const_iterator> animmodel::meshgroup::getrendermeshes() const
     575             : {
     576           0 :     std::vector<std::vector<animmodel::Mesh *>::const_iterator> rendermeshes;
     577           0 :     for(std::vector<animmodel::Mesh *>::const_iterator i = meshes.begin(); i != meshes.end(); ++i)
     578             :     {
     579           0 :         if((*i)->canrender || debugcolmesh)
     580             :         {
     581           0 :             rendermeshes.push_back(i);
     582             :         }
     583             :     }
     584           0 :     return rendermeshes;
     585           0 : }
     586             : 
     587             : //identical to above but non-const iterator and non const this
     588           1 : std::vector<std::vector<animmodel::Mesh *>::iterator> animmodel::meshgroup::getrendermeshes()
     589             : {
     590           1 :     std::vector<std::vector<animmodel::Mesh *>::iterator> rendermeshes;
     591           2 :     for(std::vector<animmodel::Mesh *>::iterator i = meshes.begin(); i != meshes.end(); ++i)
     592             :     {
     593           1 :         if((*i)->canrender || debugcolmesh)
     594             :         {
     595           1 :             rendermeshes.push_back(i);
     596             :         }
     597             :     }
     598           1 :     return rendermeshes;
     599           0 : }
     600             : 
     601           0 : void animmodel::meshgroup::bindpos(GLuint ebuf, GLuint vbuf, const void *v, int stride, int type, int size)
     602             : {
     603           0 :     if(lastebuf!=ebuf)
     604             :     {
     605           0 :         gle::bindebo(ebuf);
     606           0 :         lastebuf = ebuf;
     607             :     }
     608           0 :     if(lastvbuf!=vbuf)
     609             :     {
     610           0 :         gle::bindvbo(vbuf);
     611           0 :         if(!lastvbuf)
     612             :         {
     613           0 :             gle::enablevertex();
     614             :         }
     615           0 :         gle::vertexpointer(stride, v, type, size);
     616           0 :         lastvbuf = vbuf;
     617             :     }
     618           0 : }
     619           0 : void animmodel::meshgroup::bindpos(GLuint ebuf, GLuint vbuf, const vec *v, int stride)
     620             : {
     621           0 :     bindpos(ebuf, vbuf, v, stride, GL_FLOAT, 3);
     622           0 : }
     623             : 
     624           0 : void animmodel::meshgroup::bindpos(GLuint ebuf, GLuint vbuf, const vec4<half> *v, int stride)
     625             : {
     626           0 :     bindpos(ebuf, vbuf, v, stride, GL_HALF_FLOAT, 4);
     627           0 : }
     628             : 
     629           0 : void animmodel::meshgroup::bindtc(const void *v, int stride)
     630             : {
     631           0 :     if(!enabletc)
     632             :     {
     633           0 :         gle::enabletexcoord0();
     634           0 :         enabletc = true;
     635             :     }
     636           0 :     if(lasttcbuf!=lastvbuf)
     637             :     {
     638           0 :         gle::texcoord0pointer(stride, v, GL_HALF_FLOAT);
     639           0 :         lasttcbuf = lastvbuf;
     640             :     }
     641           0 : }
     642             : 
     643           0 : void animmodel::meshgroup::bindtangents(const void *v, int stride)
     644             : {
     645           0 :     if(!enabletangents)
     646             :     {
     647           0 :         gle::enabletangent();
     648           0 :         enabletangents = true;
     649             :     }
     650           0 :     if(lastxbuf!=lastvbuf)
     651             :     {
     652           0 :         gle::tangentpointer(stride, v, GL_SHORT);
     653           0 :         lastxbuf = lastvbuf;
     654             :     }
     655           0 : }
     656             : 
     657           0 : void animmodel::meshgroup::bindbones(const void *wv, const void *bv, int stride)
     658             : {
     659           0 :     if(!enablebones)
     660             :     {
     661           0 :         gle::enableboneweight();
     662           0 :         gle::enableboneindex();
     663           0 :         enablebones = true;
     664             :     }
     665           0 :     if(lastbbuf!=lastvbuf)
     666             :     {
     667           0 :         gle::boneweightpointer(stride, wv);
     668           0 :         gle::boneindexpointer(stride, bv);
     669           0 :         lastbbuf = lastvbuf;
     670             :     }
     671           0 : }
     672             : 
     673             : //part
     674             : 
     675          19 : animmodel::part::part(const animmodel *model, int index) : model(model), index(index), meshes(nullptr), numanimparts(1), pitchscale(1), pitchoffset(0), pitchmin(0), pitchmax(0)
     676             : {
     677          76 :     for(int i = 0; i < maxanimparts; ++i)
     678             :     {
     679          57 :         anims[i] = nullptr;
     680             :     }
     681          19 : }
     682             : 
     683          19 : animmodel::part::~part()
     684             : {
     685          76 :     for(int i = 0; i < maxanimparts; ++i)
     686             :     {
     687          59 :         delete[] anims[i];
     688             :     }
     689          19 : }
     690             : 
     691           0 : void animmodel::part::cleanup()
     692             : {
     693           0 :     if(meshes)
     694             :     {
     695           0 :         meshes->cleanup();
     696             :     }
     697           0 :     for(skin &i : skins)
     698             :     {
     699           0 :         i.cleanup();
     700             :     }
     701           0 : }
     702             : 
     703          11 : void animmodel::part::disablepitch()
     704             : {
     705          11 :     pitchscale = pitchoffset = pitchmin = pitchmax = 0;
     706          11 : }
     707             : 
     708           0 : void animmodel::part::calcbb(vec &bbmin, vec &bbmax, const matrix4x3 &m, float modelscale) const
     709             : {
     710           0 :     matrix4x3 t = m;
     711           0 :     t.scale(modelscale);
     712           0 :     meshes->calcbb(bbmin, bbmax, t);
     713           0 :     for(const linkedpart &i : links)
     714             :     {
     715           0 :         matrix4x3 n;
     716           0 :         meshes->concattagtransform(i.tag, m, n);
     717           0 :         n.translate(i.translate, modelscale);
     718           0 :         i.p->calcbb(bbmin, bbmax, n, modelscale);
     719             :     }
     720           0 : }
     721             : 
     722           0 : void animmodel::part::genBIH(std::vector<BIH::mesh> &bih, const matrix4x3 &m, float modelscale) const
     723             : {
     724           0 :     matrix4x3 t = m;
     725           0 :     t.scale(modelscale);
     726           0 :     meshes->genBIH(skins, bih, t);
     727           0 :     for(const linkedpart &i : links)
     728             :     {
     729           0 :         matrix4x3 n;
     730           0 :         meshes->concattagtransform(i.tag, m, n);
     731           0 :         n.translate(i.translate, modelscale);
     732           0 :         i.p->genBIH(bih, n, modelscale);
     733             :     }
     734           0 : }
     735             : 
     736           0 : void animmodel::part::genshadowmesh(std::vector<triangle> &tris, const matrix4x3 &m, float modelscale) const
     737             : {
     738           0 :     matrix4x3 t = m;
     739           0 :     t.scale(modelscale);
     740           0 :     meshes->genshadowmesh(tris, t);
     741           0 :     for(const linkedpart &i : links)
     742             :     {
     743           0 :         matrix4x3 n;
     744           0 :         meshes->concattagtransform(i.tag, m, n);
     745           0 :         n.translate(i.translate, modelscale);
     746           0 :         i.p->genshadowmesh(tris, n, modelscale);
     747             :     }
     748           0 : }
     749             : 
     750           0 : bool animmodel::part::link(part *p, std::string_view tag, const vec &translate, int anim, int basetime, vec *pos)
     751             : {
     752           0 :     std::optional<size_t> i = meshes ? meshes->findtag(tag) : std::nullopt;
     753           0 :     if(i<0)
     754             :     {
     755           0 :         for(const linkedpart &i : links)
     756             :         {
     757           0 :             if(i.p && i.p->link(p, tag, translate, anim, basetime, pos))
     758             :             {
     759           0 :                 return true;
     760             :             }
     761             :         }
     762           0 :         return false;
     763             :     }
     764           0 :     links.emplace_back(p, *i, anim, basetime, translate, pos, matrix4());
     765           0 :     return true;
     766             : }
     767             : 
     768           0 : bool animmodel::part::unlink(const part *p)
     769             : {
     770           0 :     for(int i = links.size(); --i >=0;) //note reverse iteration
     771             :     {
     772           0 :         if(links[i].p==p)
     773             :         {
     774           0 :             links.erase(links.begin() + i);
     775           0 :             return true;
     776             :         }
     777             :     }
     778           0 :     for(const linkedpart &i : links)
     779             :     {
     780           0 :         if(i.p && i.p->unlink(p))
     781             :         {
     782           0 :             return true;
     783             :         }
     784             :     }
     785           0 :     return false;
     786             : }
     787             : 
     788          21 : void animmodel::part::initskins(Texture *tex, Texture *masks, uint limit)
     789             : {
     790          21 :     if(!limit)
     791             :     {
     792          20 :         if(!meshes)
     793             :         {
     794           0 :             return;
     795             :         }
     796          20 :         limit = meshes->meshes.size();
     797             :     }
     798          38 :     while(skins.size() < limit)
     799             :     {
     800          17 :         skins.emplace_back(this, tex, masks);
     801             :     }
     802             : }
     803             : 
     804           0 : bool animmodel::part::alphatested() const
     805             : {
     806           0 :     for(const skin &i : skins)
     807             :     {
     808           0 :         if(i.alphatested())
     809             :         {
     810           0 :             return true;
     811             :         }
     812             :     }
     813           0 :     return false;
     814             : }
     815             : 
     816           1 : void animmodel::part::preloadBIH() const
     817             : {
     818           2 :     for(const skin &i : skins)
     819             :     {
     820           1 :         i.preloadBIH();
     821             :     }
     822           1 : }
     823             : 
     824           0 : void animmodel::part::preloadshaders()
     825             : {
     826           0 :     for(skin &i : skins)
     827             :     {
     828           0 :         i.preloadshader();
     829             :     }
     830           0 : }
     831             : 
     832           0 : void animmodel::part::preloadmeshes()
     833             : {
     834           0 :     if(meshes)
     835             :     {
     836           0 :         meshes->preload();
     837             :     }
     838           0 : }
     839             : 
     840           0 : void animmodel::part::getdefaultanim(animinfo &info) const
     841             : {
     842           0 :     info.frame = 0;
     843           0 :     info.range = 1;
     844           0 : }
     845             : 
     846           0 : bool animmodel::part::calcanim(int animpart, int anim, int basetime, int basetime2, dynent *d, int interp, animinfo &info, int &animinterptime) const
     847             : {
     848             :     //varseed uses an UGLY reinterpret cast from a pointer address to a size_t int
     849             :     //presumably the address should be a fairly random value
     850           0 :     uint varseed = static_cast<uint>(reinterpret_cast<size_t>(d));
     851           0 :     info.anim = anim;
     852           0 :     info.basetime = basetime;
     853           0 :     info.varseed = varseed;
     854           0 :     info.speed = anim & Anim_SetSpeed ? basetime2 : 100.0f;
     855           0 :     if((anim & Anim_Index) == Anim_All)
     856             :     {
     857           0 :         info.frame = 0;
     858           0 :         info.range = meshes->totalframes();
     859             :     }
     860             :     else
     861             :     {
     862           0 :         const animspec *spec = nullptr;
     863           0 :         if(anims[animpart])
     864             :         {
     865           0 :             const std::vector<animspec> &primary = anims[animpart][anim & Anim_Index];
     866           0 :             if(primary.size())
     867             :             {
     868           0 :                 spec = &primary[(varseed + static_cast<uint>(basetime))%primary.size()];
     869             :             }
     870           0 :             if((anim >> Anim_Secondary) & (Anim_Index | Anim_Dir))
     871             :             {
     872           0 :                 const std::vector<animspec> &secondary = anims[animpart][(anim >> Anim_Secondary) & Anim_Index];
     873           0 :                 if(secondary.size())
     874             :                 {
     875           0 :                     const animspec &spec2 = secondary[(varseed + static_cast<uint>(basetime2))%secondary.size()];
     876           0 :                     if(!spec || spec2.priority > spec->priority)
     877             :                     {
     878           0 :                         spec = &spec2;
     879           0 :                         info.anim >>= Anim_Secondary;
     880           0 :                         info.basetime = basetime2;
     881             :                     }
     882             :                 }
     883             :             }
     884             :         }
     885           0 :         if(spec)
     886             :         {
     887           0 :             info.frame = spec->frame;
     888           0 :             info.range = spec->range;
     889           0 :             if(spec->speed>0)
     890             :             {
     891           0 :                 info.speed = 1000.0f/spec->speed;
     892             :             }
     893             :         }
     894             :         else
     895             :         {
     896           0 :             getdefaultanim(info);
     897             :         }
     898             :     }
     899             : 
     900           0 :     info.anim &= (1 << Anim_Secondary) - 1;
     901           0 :     info.anim |= anim & Anim_Flags;
     902           0 :     if(info.anim & Anim_Loop)
     903             :     {
     904           0 :         info.anim &= ~Anim_SetTime;
     905           0 :         if(!info.basetime)
     906             :         {
     907           0 :             info.basetime = -(static_cast<int>(reinterpret_cast<size_t>(d)) & 0xFFF);
     908             :         }
     909           0 :         if(info.anim & Anim_Clamp)
     910             :         {
     911           0 :             if(info.anim & Anim_Reverse)
     912             :             {
     913           0 :                 info.frame += info.range-1;
     914             :             }
     915           0 :             info.range = 1;
     916             :         }
     917             :     }
     918             : 
     919           0 :     if(!meshes->hasframes(info.frame, info.range))
     920             :     {
     921           0 :         if(!meshes->hasframe(info.frame))
     922             :         {
     923           0 :             return false;
     924             :         }
     925           0 :         info.range = meshes->clipframes(info.frame, info.range);
     926             :     }
     927             : 
     928           0 :     if(d && interp>=0)
     929             :     {
     930           0 :         animinterpinfo &animationinterpolation = d->animinterp[interp];
     931           0 :         if((info.anim&(Anim_Loop | Anim_Clamp)) == Anim_Clamp)
     932             :         {
     933           0 :             animinterptime = std::min(animinterptime, static_cast<int>(info.range*info.speed*0.5e-3f));
     934             :         }
     935           0 :         void *ak = meshes->animkey();
     936           0 :         if(d->ragdoll && d->ragdoll->millis != lastmillis)
     937             :         {
     938           0 :             animationinterpolation.prev.range = animationinterpolation.cur.range = 0;
     939           0 :             animationinterpolation.lastswitch = -1;
     940             :         }
     941           0 :         else if(animationinterpolation.lastmodel!=ak || animationinterpolation.lastswitch<0 || lastmillis-d->lastrendered>animinterptime)
     942             :         {
     943           0 :             animationinterpolation.prev = animationinterpolation.cur = info;
     944           0 :             animationinterpolation.lastswitch = lastmillis-animinterptime*2;
     945             :         }
     946           0 :         else if(animationinterpolation.cur!=info)
     947             :         {
     948           0 :             if(lastmillis-animationinterpolation.lastswitch>animinterptime/2)
     949             :             {
     950           0 :                 animationinterpolation.prev = animationinterpolation.cur;
     951             :             }
     952           0 :             animationinterpolation.cur = info;
     953           0 :             animationinterpolation.lastswitch = lastmillis;
     954             :         }
     955           0 :         else if(info.anim & Anim_SetTime)
     956             :         {
     957           0 :             animationinterpolation.cur.basetime = info.basetime;
     958             :         }
     959           0 :         animationinterpolation.lastmodel = ak;
     960             :     }
     961           0 :     return true;
     962             : }
     963             : 
     964           0 : void animmodel::part::intersect(int anim, int basetime, int basetime2, float pitch, const vec &axis, const vec &forward, dynent *d, const vec &o, const vec &ray)
     965             : {
     966             :     AnimState as[maxanimparts];
     967           0 :     intersect(anim, basetime, basetime2, pitch, axis, forward, d, o, ray, as);
     968           0 : }
     969             : 
     970           0 : void animmodel::part::intersect(int anim, int basetime, int basetime2, float pitch, const vec &axis, const vec &forward, dynent *d, const vec &o, const vec &ray, AnimState *as)
     971             : {
     972           0 :     if((anim & Anim_Reuse) != Anim_Reuse)
     973             :     {
     974           0 :         for(int i = 0; i < numanimparts; ++i)
     975             :         {
     976           0 :             animinfo info;
     977           0 :             int interp = d && index+numanimparts<=maxanimparts ? index+i : -1,
     978           0 :                 animinterptime = animationinterpolationtime;
     979           0 :             if(!calcanim(i, anim, basetime, basetime2, d, interp, info, animinterptime))
     980             :             {
     981           0 :                 return;
     982             :             }
     983           0 :             AnimState &p = as[i];
     984           0 :             p.owner = this;
     985           0 :             p.cur.setframes(info);
     986           0 :             p.interp = 1;
     987           0 :             if(interp>=0 && d->animinterp[interp].prev.range>0)
     988             :             {
     989           0 :                 int diff = lastmillis-d->animinterp[interp].lastswitch;
     990           0 :                 if(diff<animinterptime)
     991             :                 {
     992           0 :                     p.prev.setframes(d->animinterp[interp].prev);
     993           0 :                     p.interp = diff/static_cast<float>(animinterptime);
     994             :                 }
     995             :             }
     996             :         }
     997             :     }
     998             : 
     999           0 :     float resize = model->scale * sizescale;
    1000           0 :     size_t oldpos = matrixstack.size();
    1001           0 :     vec oaxis, oforward, oo, oray;
    1002           0 :     matrixstack.top().transposedtransformnormal(axis, oaxis);
    1003           0 :     float pitchamount = pitchscale*pitch + pitchoffset;
    1004           0 :     if(pitchmin || pitchmax)
    1005             :     {
    1006           0 :         pitchamount = std::clamp(pitchamount, pitchmin, pitchmax);
    1007             :     }
    1008           0 :     if(as->cur.anim & Anim_NoPitch || (as->interp < 1 && as->prev.anim & Anim_NoPitch))
    1009             :     {
    1010           0 :         pitchamount *= (as->cur.anim & Anim_NoPitch ? 0 : as->interp) + (as->interp < 1 && as->prev.anim & Anim_NoPitch ? 0 : 1 - as->interp);
    1011             :     }
    1012           0 :     if(pitchamount)
    1013             :     {
    1014           0 :         matrixstack.push(matrixstack.top());
    1015           0 :         matrixstack.top().rotate(pitchamount/RAD, oaxis);
    1016             :     }
    1017           0 :     if(this == model->parts[0] && !model->translate.iszero())
    1018             :     {
    1019           0 :         if(oldpos == matrixstack.size())
    1020             :         {
    1021           0 :             matrixstack.push(matrixstack.top());
    1022             :         }
    1023           0 :         matrixstack.top().translate(model->translate, resize);
    1024             :     }
    1025           0 :     matrixstack.top().transposedtransformnormal(forward, oforward);
    1026           0 :     matrixstack.top().transposedtransform(o, oo);
    1027           0 :     oo.div(resize);
    1028           0 :     matrixstack.top().transposedtransformnormal(ray, oray);
    1029             : 
    1030           0 :     if((anim & Anim_Reuse) != Anim_Reuse)
    1031             :     {
    1032           0 :         for(linkedpart &link : links)
    1033             :         {
    1034           0 :             if(!link.p)
    1035             :             {
    1036           0 :                 continue;
    1037             :             }
    1038           0 :             link.matrix.translate(link.translate, resize);
    1039           0 :             matrix4 mul;
    1040           0 :             mul.mul(matrixstack.top(), link.matrix);
    1041           0 :             matrixstack.push(mul);
    1042             : 
    1043           0 :             int nanim = anim,
    1044           0 :                 nbasetime  = basetime,
    1045           0 :                 nbasetime2 = basetime2;
    1046           0 :             if(link.anim>=0)
    1047             :             {
    1048           0 :                 nanim = link.anim | (anim & Anim_Flags);
    1049           0 :                 nbasetime = link.basetime;
    1050           0 :                 nbasetime2 = 0;
    1051             :             }
    1052           0 :             link.p->intersect(nanim, nbasetime, nbasetime2, pitch, axis, forward, d, o, ray);
    1053             : 
    1054           0 :             matrixstack.pop();
    1055             :         }
    1056             :     }
    1057             : 
    1058           0 :     while(matrixstack.size() > oldpos)
    1059             :     {
    1060           0 :         matrixstack.pop();
    1061             :     }
    1062             : }
    1063             : 
    1064           0 : void animmodel::part::render(int anim, int basetime, int basetime2, float pitch, const vec &axis, const vec &forward, dynent *d)
    1065             : {
    1066             :     AnimState as[maxanimparts];
    1067           0 :     render(anim, basetime, basetime2, pitch, axis, forward, d, as);
    1068           0 : }
    1069             : 
    1070           0 : void animmodel::part::render(int anim, int basetime, int basetime2, float pitch, const vec &axis, const vec &forward, dynent *d, AnimState *as)
    1071             : {
    1072           0 :     if((anim & Anim_Reuse) != Anim_Reuse)
    1073             :     {
    1074           0 :         for(int i = 0; i < numanimparts; ++i)
    1075             :         {
    1076           0 :             animinfo info;
    1077           0 :             int interp = d && index+numanimparts<=maxanimparts ? index+i : -1, animinterptime = animationinterpolationtime;
    1078           0 :             if(!calcanim(i, anim, basetime, basetime2, d, interp, info, animinterptime))
    1079             :             {
    1080           0 :                 return;
    1081             :             }
    1082           0 :             AnimState &p = as[i];
    1083           0 :             p.owner = this;
    1084           0 :             p.cur.setframes(info);
    1085           0 :             p.interp = 1;
    1086           0 :             if(interp>=0 && d->animinterp[interp].prev.range>0)
    1087             :             {
    1088           0 :                 int diff = lastmillis-d->animinterp[interp].lastswitch;
    1089           0 :                 if(diff<animinterptime)
    1090             :                 {
    1091           0 :                     p.prev.setframes(d->animinterp[interp].prev);
    1092           0 :                     p.interp = diff/static_cast<float>(animinterptime);
    1093             :                 }
    1094             :             }
    1095             :         }
    1096             :     }
    1097             : 
    1098           0 :     float resize = model->scale * sizescale;
    1099           0 :     size_t oldpos = matrixstack.size();
    1100           0 :     vec oaxis, oforward;
    1101           0 :     matrixstack.top().transposedtransformnormal(axis, oaxis);
    1102           0 :     float pitchamount = pitchscale*pitch + pitchoffset;
    1103           0 :     if(pitchmin || pitchmax)
    1104             :     {
    1105           0 :         pitchamount = std::clamp(pitchamount, pitchmin, pitchmax);
    1106             :     }
    1107           0 :     if(as->cur.anim & Anim_NoPitch || (as->interp < 1 && as->prev.anim & Anim_NoPitch))
    1108             :     {
    1109           0 :         pitchamount *= (as->cur.anim & Anim_NoPitch ? 0 : as->interp) + (as->interp < 1 && as->prev.anim & Anim_NoPitch ? 0 : 1 - as->interp);
    1110             :     }
    1111           0 :     if(pitchamount)
    1112             :     {
    1113           0 :         matrixstack.push(matrixstack.top());
    1114           0 :         matrixstack.top().rotate(pitchamount/RAD, oaxis);
    1115             :     }
    1116           0 :     if(this == model->parts[0] && !model->translate.iszero())
    1117             :     {
    1118           0 :         if(oldpos == matrixstack.size())
    1119             :         {
    1120           0 :             matrixstack.push(matrixstack.top());
    1121             :         }
    1122           0 :         matrixstack.top().translate(model->translate, resize);
    1123             :     }
    1124           0 :     matrixstack.top().transposedtransformnormal(forward, oforward);
    1125             : 
    1126           0 :     if(!(anim & Anim_NoRender))
    1127             :     {
    1128           0 :         matrix4 modelmatrix;
    1129           0 :         modelmatrix.mul(shadowmapping ? shadowmatrix : camprojmatrix, matrixstack.top());
    1130           0 :         if(resize!=1)
    1131             :         {
    1132           0 :             modelmatrix.scale(resize);
    1133             :         }
    1134           0 :         GLOBALPARAM(modelmatrix, modelmatrix);
    1135           0 :         if(!(anim & Anim_NoSkin))
    1136             :         {
    1137           0 :             GLOBALPARAM(modelworld, matrix3(matrixstack.top()));
    1138             : 
    1139           0 :             vec modelcamera;
    1140           0 :             matrixstack.top().transposedtransform(camera1->o, modelcamera);
    1141           0 :             modelcamera.div(resize);
    1142           0 :             GLOBALPARAM(modelcamera, modelcamera);
    1143             :         }
    1144             :     }
    1145             : 
    1146           0 :     meshes->render(as, pitch, oaxis, oforward, d, this);
    1147             : 
    1148           0 :     if((anim & Anim_Reuse) != Anim_Reuse)
    1149             :     {
    1150           0 :         for(linkedpart &link : links)
    1151             :         {
    1152           0 :             link.matrix.translate(link.translate, resize);
    1153           0 :             matrix4 mul;
    1154           0 :             mul.mul(matrixstack.top(), link.matrix);
    1155           0 :             matrixstack.push(mul);
    1156           0 :             if(link.pos)
    1157             :             {
    1158           0 :                 *link.pos = matrixstack.top().gettranslation();
    1159             :             }
    1160           0 :             if(!link.p)
    1161             :             {
    1162           0 :                 matrixstack.pop();
    1163           0 :                 continue;
    1164             :             }
    1165           0 :             int nanim = anim,
    1166           0 :                 nbasetime = basetime,
    1167           0 :                 nbasetime2 = basetime2;
    1168           0 :             if(link.anim>=0)
    1169             :             {
    1170           0 :                 nanim = link.anim | (anim & Anim_Flags);
    1171           0 :                 nbasetime = link.basetime;
    1172           0 :                 nbasetime2 = 0;
    1173             :             }
    1174           0 :             link.p->render(nanim, nbasetime, nbasetime2, pitch, axis, forward, d);
    1175           0 :             matrixstack.pop();
    1176             :         }
    1177             :     }
    1178             : 
    1179           0 :     while(matrixstack.size() > oldpos)
    1180             :     {
    1181           0 :         matrixstack.pop();
    1182             :     }
    1183             : }
    1184             : 
    1185           2 : void animmodel::part::setanim(int animpart, int num, int frame, int range, float speed, int priority)
    1186             : {
    1187           2 :     if(animpart<0 || animpart>=maxanimparts || num<0 || num >= static_cast<int>(animnames.size()))
    1188             :     {
    1189           0 :         return;
    1190             :     }
    1191           2 :     if(frame<0 || range<=0 || !meshes || !meshes->hasframes(frame, range))
    1192             :     {
    1193           0 :         conoutf("invalid frame %d, range %d in model %s", frame, range, model->modelname().c_str());
    1194           0 :         return;
    1195             :     }
    1196           2 :     if(!anims[animpart])
    1197             :     {
    1198           4 :         anims[animpart] = new std::vector<animspec>[animnames.size()];
    1199             :     }
    1200           2 :     anims[animpart][num].push_back({frame, range, speed, priority});
    1201             : }
    1202             : 
    1203           2 : bool animmodel::part::animated() const
    1204             : {
    1205           2 :     for(int i = 0; i < maxanimparts; ++i)
    1206             :     {
    1207           2 :         if(anims[i])
    1208             :         {
    1209           2 :             return true;
    1210             :         }
    1211             :     }
    1212           0 :     return false;
    1213             : }
    1214             : 
    1215          11 : void animmodel::part::loaded()
    1216             : {
    1217          22 :     for(skin &i : skins)
    1218             :     {
    1219          11 :         i.setkey();
    1220             :     }
    1221          11 : }
    1222             : 
    1223           0 : int animmodel::linktype(const animmodel *, const part *) const
    1224             : {
    1225           0 :     return Link_Tag;
    1226             : }
    1227             : 
    1228           0 : void animmodel::intersect(int anim, int basetime, int basetime2, float pitch, const vec &axis, const vec &forward, dynent *d, modelattach *a, const vec &o, const vec &ray) const
    1229             : {
    1230           0 :     int numtags = 0;
    1231           0 :     if(a)
    1232             :     {
    1233           0 :         int index = parts.back()->index + parts.back()->numanimparts;
    1234           0 :         for(int i = 0; a[i].tag; i++)
    1235             :         {
    1236           0 :             numtags++;
    1237           0 :             animmodel *m = static_cast<animmodel *>(a[i].m);
    1238           0 :             if(!m)
    1239             :             {
    1240           0 :                 continue;
    1241             :             }
    1242           0 :             part *p = m->parts[0];
    1243           0 :             switch(linktype(m, p))
    1244             :             {
    1245           0 :                 case Link_Tag:
    1246             :                 {
    1247           0 :                     p->index = link(p, a[i].tag, vec(0, 0, 0), a[i].anim, a[i].basetime, a[i].pos) ? index : -1;
    1248           0 :                     break;
    1249             :                 }
    1250           0 :                 default:
    1251             :                 {
    1252           0 :                     continue;
    1253             :                 }
    1254             :             }
    1255           0 :             index += p->numanimparts;
    1256             :         }
    1257             :     }
    1258             : 
    1259             :     AnimState as[maxanimparts];
    1260           0 :     parts[0]->intersect(anim, basetime, basetime2, pitch, axis, forward, d, o, ray, as);
    1261             : 
    1262           0 :     for(part *p : parts)
    1263             :     {
    1264           0 :         switch(linktype(this, p))
    1265             :         {
    1266           0 :             case Link_Reuse:
    1267             :             {
    1268           0 :                 p->intersect(anim | Anim_Reuse, basetime, basetime2, pitch, axis, forward, d, o, ray, as);
    1269           0 :                 break;
    1270             :             }
    1271             :         }
    1272             :     }
    1273             : 
    1274           0 :     if(a)
    1275             :     {
    1276           0 :         for(int i = numtags-1; i >= 0; i--)
    1277             :         {
    1278           0 :             animmodel *m = static_cast<animmodel *>(a[i].m);
    1279           0 :             if(!m)
    1280             :             {
    1281           0 :                 continue;
    1282             :             }
    1283             : 
    1284           0 :             part *p = m->parts[0];
    1285           0 :             switch(linktype(m, p))
    1286             :             {
    1287           0 :                 case Link_Tag:
    1288             :                 {
    1289           0 :                     if(p->index >= 0)
    1290             :                     {
    1291           0 :                         unlink(p);
    1292             :                     }
    1293           0 :                     p->index = 0;
    1294           0 :                     break;
    1295             :                 }
    1296           0 :                 case Link_Reuse:
    1297             :                 {
    1298           0 :                     p->intersect(anim | Anim_Reuse, basetime, basetime2, pitch, axis, forward, d, o, ray, as);
    1299           0 :                     break;
    1300             :                 }
    1301             :             }
    1302             :         }
    1303             :     }
    1304           0 : }
    1305             : 
    1306           0 : int animmodel::intersect(int anim, int basetime, int basetime2, const vec &pos, float yaw, float pitch, float roll, dynent *d, modelattach *a, float size, const vec &o, const vec &ray, float &dist) const
    1307             : {
    1308           0 :     vec axis(1, 0, 0),
    1309           0 :         forward(0, 1, 0);
    1310             : 
    1311           0 :     matrixstack.push(matrix4());
    1312           0 :     matrixstack.top().identity();
    1313           0 :     if(!d || !d->ragdoll || d->ragdoll->millis == lastmillis)
    1314             :     {
    1315           0 :         const float secs = lastmillis/1000.0f;
    1316           0 :         yaw += spin.x*secs;
    1317           0 :         pitch += spin.y*secs;
    1318           0 :         roll += spin.z*secs;
    1319             : 
    1320           0 :         matrixstack.top().settranslation(pos);
    1321           0 :         matrixstack.top().rotate_around_z(yaw/RAD);
    1322           0 :         const bool usepitch = pitched();
    1323           0 :         if(roll && !usepitch)
    1324             :         {
    1325           0 :             matrixstack.top().rotate_around_y(-roll/RAD);
    1326             :         }
    1327           0 :         matrixstack.top().transformnormal(vec(axis), axis);
    1328           0 :         matrixstack.top().transformnormal(vec(forward), forward);
    1329           0 :         if(roll && usepitch)
    1330             :         {
    1331           0 :             matrixstack.top().rotate_around_y(-roll/RAD);
    1332             :         }
    1333           0 :         if(orientation.x)
    1334             :         {
    1335           0 :             matrixstack.top().rotate_around_z(orientation.x/RAD);
    1336             :         }
    1337           0 :         if(orientation.y)
    1338             :         {
    1339           0 :             matrixstack.top().rotate_around_x(orientation.y/RAD);
    1340             :         }
    1341           0 :         if(orientation.z)
    1342             :         {
    1343           0 :             matrixstack.top().rotate_around_y(-orientation.z/RAD);
    1344             :         }
    1345           0 :     }
    1346             :     else
    1347             :     {
    1348           0 :         matrixstack.top().settranslation(d->ragdoll->center);
    1349           0 :         pitch = 0;
    1350             :     }
    1351           0 :     sizescale = size;
    1352             : 
    1353           0 :     intersect(anim, basetime, basetime2, pitch, axis, forward, d, a, o, ray);
    1354           0 :     return -1;
    1355             : }
    1356             : 
    1357           0 : void animmodel::render(int anim, int basetime, int basetime2, float pitch, const vec &axis, const vec &forward, dynent *d, modelattach *a) const
    1358             : {
    1359           0 :     int numtags = 0;
    1360           0 :     if(a)
    1361             :     {
    1362           0 :         int index = parts.back()->index + parts.back()->numanimparts;
    1363           0 :         for(int i = 0; a[i].tag; i++)
    1364             :         {
    1365           0 :             numtags++;
    1366             : 
    1367           0 :             animmodel *m = static_cast<animmodel *>(a[i].m);
    1368           0 :             if(!m)
    1369             :             {
    1370           0 :                 if(a[i].pos)
    1371             :                 {
    1372           0 :                     link(nullptr, a[i].tag, vec(0, 0, 0), 0, 0, a[i].pos);
    1373             :                 }
    1374           0 :                 continue;
    1375             :             }
    1376           0 :             part *p = m->parts[0];
    1377           0 :             switch(linktype(m, p))
    1378             :             {
    1379           0 :                 case Link_Tag:
    1380             :                 {
    1381           0 :                     p->index = link(p, a[i].tag, vec(0, 0, 0), a[i].anim, a[i].basetime, a[i].pos) ? index : -1;
    1382           0 :                     break;
    1383             :                 }
    1384           0 :                 default:
    1385             :                 {
    1386           0 :                     continue;
    1387             :                 }
    1388             :             }
    1389           0 :             index += p->numanimparts;
    1390             :         }
    1391             :     }
    1392             : 
    1393             :     AnimState as[maxanimparts];
    1394           0 :     parts[0]->render(anim, basetime, basetime2, pitch, axis, forward, d, as);
    1395             : 
    1396           0 :     for(uint i = 1; i < parts.size(); i++)
    1397             :     {
    1398           0 :         part *p = parts[i];
    1399           0 :         switch(linktype(this, p))
    1400             :         {
    1401           0 :             case Link_Reuse:
    1402             :             {
    1403           0 :                 p->render(anim | Anim_Reuse, basetime, basetime2, pitch, axis, forward, d, as);
    1404           0 :                 break;
    1405             :             }
    1406             :         }
    1407             :     }
    1408             : 
    1409           0 :     if(a)
    1410             :     {
    1411           0 :         for(int i = numtags-1; i >= 0; i--)
    1412             :         {
    1413           0 :             animmodel *m = static_cast<animmodel *>(a[i].m);
    1414           0 :             if(!m)
    1415             :             {
    1416           0 :                 if(a[i].pos)
    1417             :                 {
    1418           0 :                     unlink(nullptr);
    1419             :                 }
    1420           0 :                 continue;
    1421             :             }
    1422           0 :             part *p = m->parts[0];
    1423           0 :             switch(linktype(m, p))
    1424             :             {
    1425           0 :                 case Link_Tag:
    1426             :                 {
    1427           0 :                     if(p->index >= 0)
    1428             :                     {
    1429           0 :                         unlink(p);
    1430             :                     }
    1431           0 :                     p->index = 0;
    1432           0 :                     break;
    1433             :                 }
    1434           0 :                 case Link_Reuse:
    1435             :                 {
    1436           0 :                     p->render(anim | Anim_Reuse, basetime, basetime2, pitch, axis, forward, d, as);
    1437           0 :                     break;
    1438             :                 }
    1439             :             }
    1440             :         }
    1441             :     }
    1442           0 : }
    1443             : 
    1444           0 : void animmodel::render(int anim, int basetime, int basetime2, const vec &o, float yaw, float pitch, float roll, dynent *d, modelattach *a, float size, const vec4<float> &color) const
    1445             : {
    1446           0 :     vec axis(1, 0, 0),
    1447           0 :         forward(0, 1, 0);
    1448             : 
    1449           0 :     matrixstack = {};
    1450           0 :     matrixstack.push(matrix4());
    1451           0 :     matrixstack.top().identity();
    1452           0 :     if(!d || !d->ragdoll || d->ragdoll->millis == lastmillis)
    1453             :     {
    1454           0 :         float secs = lastmillis/1000.0f;
    1455           0 :         yaw += spin.x*secs;
    1456           0 :         pitch += spin.y*secs;
    1457           0 :         roll += spin.z*secs;
    1458             : 
    1459           0 :         matrixstack.top().settranslation(o);
    1460           0 :         matrixstack.top().rotate_around_z(yaw/RAD);
    1461           0 :         bool usepitch = pitched();
    1462           0 :         if(roll && !usepitch)
    1463             :         {
    1464           0 :             matrixstack.top().rotate_around_y(-roll/RAD);
    1465             :         }
    1466           0 :         matrixstack.top().transformnormal(vec(axis), axis);
    1467           0 :         matrixstack.top().transformnormal(vec(forward), forward);
    1468           0 :         if(roll && usepitch)
    1469             :         {
    1470           0 :             matrixstack.top().rotate_around_y(-roll/RAD);
    1471             :         }
    1472           0 :         if(orientation.x)
    1473             :         {
    1474           0 :             matrixstack.top().rotate_around_z(orientation.x/RAD);
    1475             :         }
    1476           0 :         if(orientation.y)
    1477             :         {
    1478           0 :             matrixstack.top().rotate_around_x(orientation.y/RAD);
    1479             :         }
    1480           0 :         if(orientation.z)
    1481             :         {
    1482           0 :             matrixstack.top().rotate_around_y(-orientation.z/RAD);
    1483             :         }
    1484           0 :     }
    1485             :     else
    1486             :     {
    1487           0 :         matrixstack.top().settranslation(d->ragdoll->center);
    1488           0 :         pitch = 0;
    1489             :     }
    1490             : 
    1491           0 :     sizescale = size;
    1492             : 
    1493           0 :     if(anim & Anim_NoRender)
    1494             :     {
    1495           0 :         render(anim, basetime, basetime2, pitch, axis, forward, d, a);
    1496           0 :         if(d)
    1497             :         {
    1498           0 :             d->lastrendered = lastmillis;
    1499             :         }
    1500           0 :         return;
    1501             :     }
    1502             : 
    1503           0 :     if(!(anim & Anim_NoSkin))
    1504             :     {
    1505           0 :         if(colorscale != color)
    1506             :         {
    1507           0 :             colorscale = color;
    1508           0 :             skin::invalidateshaderparams();
    1509             :         }
    1510             :     }
    1511             : 
    1512           0 :     if(depthoffset && !enabledepthoffset)
    1513             :     {
    1514           0 :         enablepolygonoffset(GL_POLYGON_OFFSET_FILL);
    1515           0 :         enabledepthoffset = true;
    1516             :     }
    1517             : 
    1518           0 :     render(anim, basetime, basetime2, pitch, axis, forward, d, a);
    1519             : 
    1520           0 :     if(d)
    1521             :     {
    1522           0 :         d->lastrendered = lastmillis;
    1523             :     }
    1524             : }
    1525             : 
    1526           0 : void animmodel::cleanup()
    1527             : {
    1528           0 :     for(part *p : parts)
    1529             :     {
    1530           0 :         p->cleanup();
    1531             :     }
    1532           0 : }
    1533             : 
    1534           0 : animmodel::part &animmodel::addpart()
    1535             : {
    1536           0 :     part *p = new part(this, parts.size());
    1537           0 :     parts.push_back(p);
    1538           0 :     return *p;
    1539             : }
    1540             : 
    1541           2 : matrix4x3 animmodel::initmatrix() const
    1542             : {
    1543           2 :     matrix4x3 m;
    1544           2 :     m.identity();
    1545           2 :     if(orientation.x)
    1546             :     {
    1547           0 :         m.rotate_around_z(orientation.x/RAD);
    1548             :     }
    1549           2 :     if(orientation.y)
    1550             :     {
    1551           0 :         m.rotate_around_x(orientation.y/RAD);
    1552             :     }
    1553           2 :     if(orientation.z)
    1554             :     {
    1555           0 :         m.rotate_around_y(-orientation.z/RAD);
    1556             :     }
    1557           2 :     m.translate(translate, scale);
    1558           2 :     return m;
    1559             : }
    1560             : 
    1561           2 : void animmodel::genBIH(std::vector<BIH::mesh> &bih)
    1562             : {
    1563           2 :     if(parts.empty())
    1564             :     {
    1565           0 :         return;
    1566             :     }
    1567           2 :     matrix4x3 m = initmatrix();
    1568           4 :     for(const skin &s : parts[0]->skins)
    1569             :     {
    1570           2 :         s.tex->loadalphamask();
    1571             :     }
    1572           2 :     for(uint i = 1; i < parts.size(); i++)
    1573             :     {
    1574           0 :         const part *p = parts[i];
    1575           0 :         switch(linktype(this, p))
    1576             :         {
    1577           0 :             case Link_Reuse:
    1578             :             {
    1579           0 :                 for(skin &s : parts[i]->skins)
    1580             :                 {
    1581           0 :                     s.tex->loadalphamask();
    1582             :                 }
    1583           0 :                 p->genBIH(bih, m, scale);
    1584           0 :                 break;
    1585             :             }
    1586             :         }
    1587             :     }
    1588             : }
    1589             : 
    1590           0 : bool animmodel::link(part *p, std::string_view tag, const vec &translate, int anim, int basetime, vec *pos) const
    1591             : {
    1592           0 :     if(parts.empty())
    1593             :     {
    1594           0 :         return false;
    1595             :     }
    1596           0 :     return parts[0]->link(p, tag, translate, anim, basetime, pos);
    1597             : }
    1598             : 
    1599           7 : void animmodel::loaded()
    1600             : {
    1601          18 :     for(part *p : parts)
    1602             :     {
    1603          11 :         p->loaded();
    1604             :     }
    1605           7 : }
    1606             : 
    1607           0 : bool animmodel::unlink(const part *p) const
    1608             : {
    1609           0 :     if(parts.empty())
    1610             :     {
    1611           0 :         return false;
    1612             :     }
    1613           0 :     return parts[0]->unlink(p);
    1614             : }
    1615             : 
    1616           0 : void animmodel::genshadowmesh(std::vector<triangle> &tris, const matrix4x3 &orient) const
    1617             : {
    1618           0 :     if(parts.empty())
    1619             :     {
    1620           0 :         return;
    1621             :     }
    1622           0 :     matrix4x3 m = initmatrix();
    1623           0 :     m.mul(orient, matrix4x3(m));
    1624           0 :     parts[0]->genshadowmesh(tris, m, scale);
    1625           0 :     for(uint i = 1; i < parts.size(); i++)
    1626             :     {
    1627           0 :         const part *p = parts[i];
    1628           0 :         switch(linktype(this, p))
    1629             :         {
    1630           0 :             case Link_Reuse:
    1631             :             {
    1632           0 :                 p->genshadowmesh(tris, m, scale);
    1633           0 :                 break;
    1634             :             }
    1635             :         }
    1636             :     }
    1637             : }
    1638             : 
    1639           1 : void animmodel::preloadBIH()
    1640             : {
    1641           1 :     if(!bih)
    1642             :     {
    1643           1 :         setBIH();
    1644             :     }
    1645           1 :     if(bih)
    1646             :     {
    1647           2 :         for(const part *i : parts)
    1648             :         {
    1649           1 :             i->preloadBIH();
    1650             :         }
    1651             :     }
    1652           1 : }
    1653             : 
    1654             : //always will return true (note that this overloads model::setBIH())
    1655           2 : void animmodel::setBIH()
    1656             : {
    1657           2 :     if(bih)
    1658             :     {
    1659           0 :         return;
    1660             :     }
    1661           2 :     std::vector<BIH::mesh> meshes;
    1662           2 :     genBIH(meshes);
    1663           2 :     bih = std::make_unique<BIH>(meshes);
    1664           2 : }
    1665             : 
    1666           1 : bool animmodel::animated() const
    1667             : {
    1668           1 :     if(spin.x || spin.y || spin.z)
    1669             :     {
    1670           0 :         return true;
    1671             :     }
    1672           1 :     for(const part *i : parts)
    1673             :     {
    1674           1 :         if(i->animated())
    1675             :         {
    1676           1 :             return true;
    1677             :         }
    1678             :     }
    1679           0 :     return false;
    1680             : }
    1681             : 
    1682           1 : bool animmodel::pitched() const
    1683             : {
    1684           1 :     return parts[0]->pitchscale != 0;
    1685             : }
    1686             : 
    1687           0 : bool animmodel::alphatested() const
    1688             : {
    1689           0 :     for(const part *i : parts)
    1690             :     {
    1691           0 :         if(i->alphatested())
    1692             :         {
    1693           0 :             return true;
    1694             :         }
    1695             :     }
    1696           0 :     return false;
    1697             : }
    1698             : 
    1699           6 : bool animmodel::load()
    1700             : {
    1701           6 :     startload();
    1702           6 :     bool success = loadconfig(modelname()) && parts.size(); // configured model, will call the model commands below
    1703           6 :     if(!success)
    1704             :     {
    1705           6 :         success = loaddefaultparts(); // model without configuration, try default tris and skin
    1706             :     }
    1707           6 :     endload();
    1708           6 :     if(flipy())
    1709             :     {
    1710           0 :         translate.y = -translate.y;
    1711             :     }
    1712             : 
    1713           6 :     if(!success)
    1714             :     {
    1715           1 :         return false;
    1716             :     }
    1717          14 :     for(const part *i : parts)
    1718             :     {
    1719           9 :         if(!i->meshes)
    1720             :         {
    1721           0 :             return false;
    1722             :         }
    1723             :     }
    1724           5 :     loaded();
    1725           5 :     return true;
    1726             : }
    1727             : 
    1728           0 : void animmodel::preloadshaders()
    1729             : {
    1730           0 :     for(part *i : parts)
    1731             :     {
    1732           0 :         i->preloadshaders();
    1733             :     }
    1734           0 : }
    1735             : 
    1736           0 : void animmodel::preloadmeshes()
    1737             : {
    1738           0 :     for(part *i : parts)
    1739             :     {
    1740           0 :         i->preloadmeshes();
    1741             :     }
    1742           0 : }
    1743             : 
    1744           0 : void animmodel::setshader(Shader *shader)
    1745             : {
    1746           0 :     if(parts.empty())
    1747             :     {
    1748           0 :         loaddefaultparts();
    1749             :     }
    1750           0 :     for(part *i : parts)
    1751             :     {
    1752           0 :         for(skin &j : i->skins)
    1753             :         {
    1754           0 :             j.shader = shader;
    1755             :         }
    1756             :     }
    1757           0 : }
    1758             : 
    1759           0 : void animmodel::setspec(float spec)
    1760             : {
    1761           0 :     if(parts.empty())
    1762             :     {
    1763           0 :         loaddefaultparts();
    1764             :     }
    1765           0 :     for(part *i : parts)
    1766             :     {
    1767           0 :         for(skin &j : i->skins)
    1768             :         {
    1769           0 :             j.spec = spec;
    1770             :         }
    1771             :     }
    1772           0 : }
    1773             : 
    1774           0 : void animmodel::setgloss(int gloss)
    1775             : {
    1776           0 :     if(parts.empty())
    1777             :     {
    1778           0 :         loaddefaultparts();
    1779             :     }
    1780           0 :     for(part *i : parts)
    1781             :     {
    1782           0 :         for(skin &j : i->skins)
    1783             :         {
    1784           0 :             j.gloss = gloss;
    1785             :         }
    1786             :     }
    1787           0 : }
    1788             : 
    1789           0 : void animmodel::setglow(float glow, float delta, float pulse)
    1790             : {
    1791           0 :     if(parts.empty())
    1792             :     {
    1793           0 :         loaddefaultparts();
    1794             :     }
    1795           0 :     for(part *i : parts)
    1796             :     {
    1797           0 :         for(skin &s : i->skins)
    1798             :         {
    1799           0 :             s.glow = glow;
    1800           0 :             s.glowdelta = delta;
    1801           0 :             s.glowpulse = pulse;
    1802             :         }
    1803             :     }
    1804           0 : }
    1805             : 
    1806           0 : void animmodel::setalphatest(float alphatest)
    1807             : {
    1808           0 :     if(parts.empty())
    1809             :     {
    1810           0 :         loaddefaultparts();
    1811             :     }
    1812           0 :     for(part *i : parts)
    1813             :     {
    1814           0 :         for(skin &j : i->skins)
    1815             :         {
    1816           0 :             j.alphatest = alphatest;
    1817             :         }
    1818             :     }
    1819           0 : }
    1820             : 
    1821           0 : void animmodel::setfullbright(float fullbright)
    1822             : {
    1823           0 :     if(parts.empty())
    1824             :     {
    1825           0 :         loaddefaultparts();
    1826             :     }
    1827           0 :     for(part *i : parts)
    1828             :     {
    1829           0 :         for(skin &j : i->skins)
    1830             :         {
    1831           0 :             j.fullbright = fullbright;
    1832             :         }
    1833             :     }
    1834           0 : }
    1835             : 
    1836           0 : void animmodel::setcullface(int cullface)
    1837             : {
    1838           0 :     if(parts.empty())
    1839             :     {
    1840           0 :         loaddefaultparts();
    1841             :     }
    1842           0 :     for(part *i : parts)
    1843             :     {
    1844           0 :         for(skin &j : i->skins)
    1845             :         {
    1846           0 :             j.cullface = cullface;
    1847             :         }
    1848             :     }
    1849           0 : }
    1850             : 
    1851           0 : void animmodel::setcolor(const vec &color)
    1852             : {
    1853           0 :     if(parts.empty())
    1854             :     {
    1855           0 :         loaddefaultparts();
    1856             :     }
    1857           0 :     for(part *i : parts)
    1858             :     {
    1859           0 :         for(skin &j : i->skins)
    1860             :         {
    1861           0 :             j.color = color;
    1862             :         }
    1863             :     }
    1864           0 : }
    1865             : 
    1866           0 : void animmodel::settransformation(const std::optional<vec> pos,
    1867             :                                   const std::optional<vec> rotate,
    1868             :                                   const std::optional<vec> orient,
    1869             :                                   const std::optional<float> size)
    1870             : {
    1871           0 :     if(pos)
    1872             :     {
    1873           0 :         translate = pos.value();
    1874             :     }
    1875           0 :     if(rotate)
    1876             :     {
    1877           0 :         spin = rotate.value();
    1878             :     }
    1879           0 :     if(orient)
    1880             :     {
    1881           0 :         orientation = orient.value();
    1882             :     }
    1883           0 :     if(size)
    1884             :     {
    1885           0 :         scale = size.value();
    1886             :     }
    1887           0 : }
    1888             : 
    1889           0 : vec4<float> animmodel::locationsize() const
    1890             : {
    1891           0 :     return vec4<float>(translate.x, translate.y, translate.z, scale);
    1892             : }
    1893             : 
    1894           0 : void animmodel::calcbb(vec &center, vec &radius) const
    1895             : {
    1896           0 :     if(parts.empty())
    1897             :     {
    1898           0 :         return;
    1899             :     }
    1900           0 :     vec bbmin(1e16f, 1e16f, 1e16f),
    1901           0 :         bbmax(-1e16f, -1e16f, -1e16f);
    1902           0 :     matrix4x3 m = initmatrix();
    1903           0 :     parts[0]->calcbb(bbmin, bbmax, m, scale);
    1904           0 :     for(const part *p : parts)
    1905             :     {
    1906           0 :         switch(linktype(this, p))
    1907             :         {
    1908           0 :             case Link_Reuse:
    1909             :             {
    1910           0 :                 p->calcbb(bbmin, bbmax, m, scale);
    1911           0 :                 break;
    1912             :             }
    1913             :         }
    1914             :     }
    1915           0 :     radius = bbmax;
    1916           0 :     radius.sub(bbmin);
    1917           0 :     radius.mul(0.5f);
    1918           0 :     center = bbmin;
    1919           0 :     center.add(radius);
    1920             : }
    1921             : 
    1922           0 : void animmodel::startrender() const
    1923             : {
    1924           0 :     enabletc = enabletangents = enablebones = enabledepthoffset = false;
    1925           0 :     enablecullface = true;
    1926           0 :     lastvbuf = lasttcbuf = lastxbuf = lastbbuf = lastebuf =0;
    1927           0 :     lastmasks = lastnormalmap = lastdecal = lasttex = nullptr;
    1928           0 :     skin::invalidateshaderparams();
    1929           0 : }
    1930             : 
    1931           0 : void animmodel::endrender() const
    1932             : {
    1933           0 :     if(lastvbuf || lastebuf)
    1934             :     {
    1935           0 :         disablevbo();
    1936             :     }
    1937           0 :     if(!enablecullface)
    1938             :     {
    1939           0 :         glEnable(GL_CULL_FACE);
    1940             :     }
    1941           0 :     if(enabledepthoffset)
    1942             :     {
    1943           0 :         disablepolygonoffset(GL_POLYGON_OFFSET_FILL);
    1944             :     }
    1945           0 : }
    1946             : 
    1947           0 : void animmodel::boundbox(vec &center, vec &radius)
    1948             : {
    1949           0 :     if(bbradius.x < 0)
    1950             :     {
    1951           0 :         calcbb(bbcenter, bbradius);
    1952           0 :         bbradius.add(bbextend);
    1953             :     }
    1954           0 :     center = bbcenter;
    1955           0 :     radius = bbradius;
    1956           0 : }
    1957             : 
    1958           0 : float animmodel::collisionbox(vec &center, vec &radius)
    1959             : {
    1960           0 :     if(collideradius.x < 0)
    1961             :     {
    1962           0 :         boundbox(collidecenter, collideradius);
    1963           0 :         if(collidexyradius)
    1964             :         {
    1965           0 :             collidecenter.x = collidecenter.y = 0;
    1966           0 :             collideradius.x = collideradius.y = collidexyradius;
    1967             :         }
    1968           0 :         if(collideheight)
    1969             :         {
    1970           0 :             collidecenter.z = collideradius.z = collideheight/2;
    1971             :         }
    1972           0 :         rejectradius = collideradius.magnitude();
    1973             :     }
    1974           0 :     center = collidecenter;
    1975           0 :     radius = collideradius;
    1976           0 :     return rejectradius;
    1977             : }
    1978             : 
    1979         131 : const std::string &animmodel::modelname() const
    1980             : {
    1981         131 :     return name;
    1982             : }
    1983             : 
    1984           0 : void animmodel::disablebones()
    1985             : {
    1986           0 :     gle::disableboneweight();
    1987           0 :     gle::disableboneindex();
    1988           0 :     enablebones = false;
    1989           0 : }
    1990             : 
    1991           0 : void animmodel::disabletangents()
    1992             : {
    1993           0 :     gle::disabletangent();
    1994           0 :     enabletangents = false;
    1995           0 : }
    1996             : 
    1997           0 : void animmodel::disabletc()
    1998             : {
    1999           0 :     gle::disabletexcoord0();
    2000           0 :     enabletc = false;
    2001           0 : }
    2002             : 
    2003           0 : void animmodel::disablevbo()
    2004             : {
    2005           0 :     if(lastebuf)
    2006             :     {
    2007           0 :         gle::clearebo();
    2008             :     }
    2009           0 :     if(lastvbuf)
    2010             :     {
    2011           0 :         gle::clearvbo();
    2012           0 :         gle::disablevertex();
    2013             :     }
    2014           0 :     if(enabletc)
    2015             :     {
    2016           0 :         disabletc();
    2017             :     }
    2018           0 :     if(enabletangents)
    2019             :     {
    2020           0 :         disabletangents();
    2021             :     }
    2022           0 :     if(enablebones)
    2023             :     {
    2024           0 :         disablebones();
    2025             :     }
    2026           0 :     lastvbuf = lasttcbuf = lastxbuf = lastbbuf = lastebuf = 0;
    2027           0 : }

Generated by: LCOV version 1.14