dust3d/third_party/libigl/include/igl/readBF.cpp

115 lines
3.5 KiB
C++

// This file is part of libigl, a simple c++ geometry processing library.
//
// Copyright (C) 2016 Alec Jacobson <alecjacobson@gmail.com>
//
// This Source Code Form is subject to the terms of the Mozilla Public License
// v. 2.0. If a copy of the MPL was not distributed with this file, You can
// obtain one at http://mozilla.org/MPL/2.0/.
#include "readBF.h"
#include "list_to_matrix.h"
#include <vector>
#include <cstdio>
#include <fstream>
#include <cassert>
#include <functional>
template <
typename DerivedWI,
typename DerivedP,
typename DerivedO>
IGL_INLINE bool igl::readBF(
const std::string & filename,
Eigen::PlainObjectBase<DerivedWI> & WI,
Eigen::PlainObjectBase<DerivedP> & P,
Eigen::PlainObjectBase<DerivedO> & O)
{
using namespace std;
ifstream is(filename);
if(!is.is_open())
{
return false;
}
string line;
std::vector<typename DerivedWI::Scalar> vWI;
std::vector<typename DerivedP::Scalar> vP;
std::vector<std::vector<typename DerivedO::Scalar> > vO;
while(getline(is, line))
{
int wi,p;
double cx,cy,cz;
if(sscanf(line.c_str(), "%d %d %lg %lg %lg",&wi,&p,&cx,&cy,&cz) != 5)
{
return false;
}
vWI.push_back(wi);
vP.push_back(p);
vO.push_back({cx,cy,cz});
}
list_to_matrix(vWI,WI);
list_to_matrix(vP,P);
list_to_matrix(vO,O);
return true;
}
template <
typename DerivedWI,
typename DerivedbfP,
typename DerivedO,
typename DerivedC,
typename DerivedBE,
typename DerivedP>
IGL_INLINE bool igl::readBF(
const std::string & filename,
Eigen::PlainObjectBase<DerivedWI> & WI,
Eigen::PlainObjectBase<DerivedbfP> & bfP,
Eigen::PlainObjectBase<DerivedO> & offsets,
Eigen::PlainObjectBase<DerivedC> & C,
Eigen::PlainObjectBase<DerivedBE> & BE,
Eigen::PlainObjectBase<DerivedP> & P)
{
using namespace Eigen;
using namespace std;
if(!readBF(filename,WI,bfP,offsets))
{
return false;
}
C.resize(WI.rows(),3);
vector<bool> computed(C.rows(),false);
// better not be cycles in bfP
std::function<Eigen::RowVector3d(const int)> locate_tip;
locate_tip =
[&offsets,&computed,&bfP,&locate_tip,&C](const int w)->Eigen::RowVector3d
{
if(w<0) return Eigen::RowVector3d(0,0,0);
if(computed[w]) return C.row(w);
computed[w] = true;
return C.row(w) = locate_tip(bfP(w)) + offsets.row(w);
};
int num_roots = (bfP.array() == -1).count();
BE.resize(WI.rows()-num_roots,2);
P.resize(BE.rows());
for(int c = 0;c<C.rows();c++)
{
locate_tip(c);
assert(c>=0);
// weight associated with this bone
const int wi = WI(c);
if(wi >= 0)
{
// index into C
const int p = bfP(c);
assert(p >= 0 && "No weights for roots allowed");
// index into BE
const int pwi = WI(p);
P(wi) = pwi;
BE(wi,0) = p;
BE(wi,1) = c;
}
}
return true;
}
#ifdef IGL_STATIC_LIBRARY
template bool igl::readBF<Eigen::Matrix<int, -1, 1, 0, -1, 1>, Eigen::Matrix<int, -1, 1, 0, -1, 1>, Eigen::Matrix<double, -1, -1, 0, -1, -1>, Eigen::Matrix<double, -1, -1, 0, -1, -1>, Eigen::Matrix<int, -1, -1, 0, -1, -1>, Eigen::Matrix<int, -1, 1, 0, -1, 1> >(std::basic_string<char, std::char_traits<char>, std::allocator<char> > const&, Eigen::PlainObjectBase<Eigen::Matrix<int, -1, 1, 0, -1, 1> >&, Eigen::PlainObjectBase<Eigen::Matrix<int, -1, 1, 0, -1, 1> >&, Eigen::PlainObjectBase<Eigen::Matrix<double, -1, -1, 0, -1, -1> >&, Eigen::PlainObjectBase<Eigen::Matrix<double, -1, -1, 0, -1, -1> >&, Eigen::PlainObjectBase<Eigen::Matrix<int, -1, -1, 0, -1, -1> >&, Eigen::PlainObjectBase<Eigen::Matrix<int, -1, 1, 0, -1, 1> >&);
#endif