Toshihiro Shimizu 890ddd
Toshihiro Shimizu 890ddd
Toshihiro Shimizu 890ddd
// TnzCore includes
Toshihiro Shimizu 890ddd
#include "tstream.h"
Toshihiro Shimizu 890ddd
Toshihiro Shimizu 890ddd
// TnzExt includes
Toshihiro Shimizu 890ddd
#include "ext/plasticskeletondeformation.h"
Toshihiro Shimizu 890ddd
Toshihiro Shimizu 890ddd
// STL includes
Toshihiro Shimizu 890ddd
#include <set></set>
Toshihiro Shimizu 890ddd
Toshihiro Shimizu 890ddd
// tcg includes
Toshihiro Shimizu 890ddd
#include "tcg/tcg_misc.h"
Toshihiro Shimizu 890ddd
#include "tcg/tcg_pool.h"
Toshihiro Shimizu 890ddd
#include "tcg/tcg_function_types.h"
Toshihiro Shimizu 890ddd
#include "tcg/tcg_iterator_ops.h"
Toshihiro Shimizu 890ddd
Toshihiro Shimizu 890ddd
#include "ext/plasticskeleton.h"
Toshihiro Shimizu 890ddd
Toshihiro Shimizu 890ddd
PERSIST_IDENTIFIER(PlasticSkeletonVertex, "PlasticSkeletonVertex")
Toshihiro Shimizu 890ddd
PERSIST_IDENTIFIER(PlasticSkeleton, "PlasticSkeleton")
Toshihiro Shimizu 890ddd
Toshihiro Shimizu 890ddd
DEFINE_CLASS_CODE(PlasticSkeleton, 122)
Toshihiro Shimizu 890ddd
Toshihiro Shimizu 890ddd
//************************************************************************************
Toshihiro Shimizu 890ddd
//    PlasticSkeletonVertex  implementation
Toshihiro Shimizu 890ddd
//************************************************************************************
Toshihiro Shimizu 890ddd
Toshihiro Shimizu 890ddd
PlasticSkeletonVertex::PlasticSkeletonVertex()
Shinya Kitaoka 120a6e
    : tcg::Vertex<tpointd>()</tpointd>
Shinya Kitaoka 120a6e
    , m_number(-1)
Shinya Kitaoka 120a6e
    , m_parent(-1)
Shinya Kitaoka 120a6e
    , m_minAngle(-(std::numeric_limits<double>::max)())</double>
Shinya Kitaoka 120a6e
    , m_maxAngle((std::numeric_limits<double>::max)())</double>
Shinya Kitaoka 120a6e
    , m_interpolate(true) {}
Toshihiro Shimizu 890ddd
Toshihiro Shimizu 890ddd
//-------------------------------------------------------------------------------
Toshihiro Shimizu 890ddd
Toshihiro Shimizu 890ddd
PlasticSkeletonVertex::PlasticSkeletonVertex(const TPointD &pos)
Shinya Kitaoka 120a6e
    : tcg::Vertex<tpointd>(pos)</tpointd>
Shinya Kitaoka 120a6e
    , m_number(-1)
Shinya Kitaoka 120a6e
    , m_parent(-1)
Shinya Kitaoka 120a6e
    , m_minAngle(-(std::numeric_limits<double>::max)())</double>
Shinya Kitaoka 120a6e
    , m_maxAngle((std::numeric_limits<double>::max)())</double>
Shinya Kitaoka 120a6e
    , m_interpolate(true) {}
Toshihiro Shimizu 890ddd
Toshihiro Shimizu 890ddd
//-------------------------------------------------------------------------------
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
PlasticSkeletonVertex::operator PlasticHandle() const {
Shinya Kitaoka 120a6e
  PlasticHandle result(P());
Shinya Kitaoka 120a6e
  result.m_interpolate = m_interpolate;
Shinya Kitaoka 120a6e
  return result;
Toshihiro Shimizu 890ddd
}
Toshihiro Shimizu 890ddd
Toshihiro Shimizu 890ddd
//-------------------------------------------------------------------------------
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
void PlasticSkeletonVertex::saveData(TOStream &os) {
Shinya Kitaoka 120a6e
  os.child("name") << m_name;
Shinya Kitaoka 120a6e
  os.child("number") << m_number;
Shinya Kitaoka 120a6e
  os.child("pos") << this->P().x << this->P().y;
Shinya Kitaoka 120a6e
  os.child("interpolate") << (int)this->m_interpolate;
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
  if (m_minAngle != -(std::numeric_limits<double>::max)())</double>
Shinya Kitaoka 120a6e
    os.child("minAngle") << m_minAngle;
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
  if (m_maxAngle != (std::numeric_limits<double>::max)())</double>
Shinya Kitaoka 120a6e
    os.child("maxAngle") << m_maxAngle;
Toshihiro Shimizu 890ddd
}
Toshihiro Shimizu 890ddd
Toshihiro Shimizu 890ddd
//-------------------------------------------------------------------------------
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
void PlasticSkeletonVertex::loadData(TIStream &is) {
Shinya Kitaoka 120a6e
  int val;
Shinya Kitaoka 120a6e
Shinya Kitaoka 120a6e
  std::string tagName;
Shinya Kitaoka 120a6e
  while (is.openChild(tagName)) {
Shinya Kitaoka 120a6e
    if (tagName == "name")
Shinya Kitaoka 120a6e
      is >> m_name, is.matchEndTag();
Shinya Kitaoka 120a6e
    else if (tagName == "number")
Shinya Kitaoka 120a6e
      is >> m_number, is.matchEndTag();
Shinya Kitaoka 120a6e
    else if (tagName == "pos")
Shinya Kitaoka 120a6e
      is >> this->P().x >> this->P().y, is.matchEndTag();
Shinya Kitaoka 120a6e
    else if (tagName == "interpolate")
Shinya Kitaoka 120a6e
      is >> val, m_interpolate = (bool)val, is.matchEndTag();
Shinya Kitaoka 120a6e
    else if (tagName == "minAngle")
Shinya Kitaoka 120a6e
      is >> m_minAngle, is.matchEndTag();
Shinya Kitaoka 120a6e
    else if (tagName == "maxAngle")
Shinya Kitaoka 120a6e
      is >> m_maxAngle, is.matchEndTag();
Shinya Kitaoka 120a6e
    else
Shinya Kitaoka 120a6e
      is.skipCurrentTag();
Shinya Kitaoka 120a6e
  }
Toshihiro Shimizu 890ddd
}
Toshihiro Shimizu 890ddd
Toshihiro Shimizu 890ddd
//************************************************************************************
Toshihiro Shimizu 890ddd
//    PlasticSkeleton::Imp  definition
Toshihiro Shimizu 890ddd
//************************************************************************************
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
class PlasticSkeleton::Imp {
Toshihiro Shimizu 890ddd
public:
Shinya Kitaoka 120a6e
  std::set<plasticskeletondeformation *=""> m_deformations;  //!< Registered</plasticskeletondeformation>
Shinya Kitaoka 120a6e
                                                          //!deformations for
Shinya Kitaoka 120a6e
                                                          //!this skeleton (not
Shinya Kitaoka 120a6e
                                                          //!owned)
Shinya Kitaoka 120a6e
  tcg::indices_pool<int></int>
Shinya Kitaoka 120a6e
      m_numbersPool;  //!< Vertex numbers pool (used for naming vertices only)
Toshihiro Shimizu 890ddd
Toshihiro Shimizu 890ddd
public:
Shinya Kitaoka 120a6e
  Imp() {}
Shinya Kitaoka 120a6e
  Imp(const Imp &other);
Shinya Kitaoka 120a6e
  Imp &operator=(const Imp &other);
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
  Imp(Imp &&other);
Shinya Kitaoka 120a6e
  Imp &operator=(Imp &&other);
Toshihiro Shimizu 890ddd
};
Toshihiro Shimizu 890ddd
Toshihiro Shimizu 890ddd
//===============================================================================
Toshihiro Shimizu 890ddd
Toshihiro Shimizu 890ddd
PlasticSkeleton::Imp::Imp(const Imp &other)
Shinya Kitaoka 120a6e
    : m_numbersPool(other.m_numbersPool) {}
Toshihiro Shimizu 890ddd
Toshihiro Shimizu 890ddd
//-------------------------------------------------------------------------------
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
PlasticSkeleton::Imp &PlasticSkeleton::Imp::operator=(const Imp &other) {
Shinya Kitaoka 120a6e
  m_numbersPool = other.m_numbersPool;
Shinya Kitaoka 120a6e
  return *this;
Toshihiro Shimizu 890ddd
}
Toshihiro Shimizu 890ddd
Toshihiro Shimizu 890ddd
//-------------------------------------------------------------------------------
Toshihiro Shimizu 890ddd
PlasticSkeleton::Imp::Imp(Imp &&other)
Shinya Kitaoka 120a6e
    : m_numbersPool(std::move(other.m_numbersPool)) {}
Toshihiro Shimizu 890ddd
Toshihiro Shimizu 890ddd
//-------------------------------------------------------------------------------
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
PlasticSkeleton::Imp &PlasticSkeleton::Imp::operator=(Imp &&other) {
Shinya Kitaoka 120a6e
  m_numbersPool = std::move(other.m_numbersPool);
Shinya Kitaoka 120a6e
  return *this;
Toshihiro Shimizu 890ddd
}
Toshihiro Shimizu 890ddd
Toshihiro Shimizu 890ddd
//************************************************************************************
Toshihiro Shimizu 890ddd
//    PlasticSkeleton  implementation
Toshihiro Shimizu 890ddd
//************************************************************************************
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
PlasticSkeleton::PlasticSkeleton() : m_imp(new Imp) {}
Toshihiro Shimizu 890ddd
Toshihiro Shimizu 890ddd
//------------------------------------------------------------------
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
PlasticSkeleton::~PlasticSkeleton() {}
Toshihiro Shimizu 890ddd
Toshihiro Shimizu 890ddd
//------------------------------------------------------------------
Toshihiro Shimizu 890ddd
Toshihiro Shimizu 890ddd
PlasticSkeleton::PlasticSkeleton(const PlasticSkeleton &other)
Shinya Kitaoka 120a6e
    : mesh_type(other), m_imp(new Imp(*other.m_imp)) {}
Toshihiro Shimizu 890ddd
Toshihiro Shimizu 890ddd
//------------------------------------------------------------------
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
PlasticSkeleton &PlasticSkeleton::operator=(const PlasticSkeleton &other) {
Shinya Kitaoka 120a6e
  assert(m_imp->m_deformations.empty());
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
  mesh_type::operator=(other);
Shinya Kitaoka 120a6e
  *m_imp             = *other.m_imp;
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
  return *this;
Toshihiro Shimizu 890ddd
}
Toshihiro Shimizu 890ddd
Toshihiro Shimizu 890ddd
//------------------------------------------------------------------
Toshihiro Shimizu 890ddd
PlasticSkeleton::PlasticSkeleton(PlasticSkeleton &&other)
Shinya Kitaoka 120a6e
    : mesh_type(std::forward<mesh_type>(other))</mesh_type>
Shinya Kitaoka 120a6e
    , m_imp(new Imp(std::move(*other.m_imp))) {}
Toshihiro Shimizu 890ddd
Toshihiro Shimizu 890ddd
//------------------------------------------------------------------
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
PlasticSkeleton &PlasticSkeleton::operator=(PlasticSkeleton &&other) {
Shinya Kitaoka 120a6e
  assert(m_imp->m_deformations.empty());
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
  mesh_type::operator=(std::forward<mesh_type>(other));</mesh_type>
Shinya Kitaoka 120a6e
  *m_imp             = std::move(*other.m_imp);
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
  return *this;
Toshihiro Shimizu 890ddd
}
Toshihiro Shimizu 890ddd
Toshihiro Shimizu 890ddd
//------------------------------------------------------------------
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
void PlasticSkeleton::addListener(PlasticSkeletonDeformation *deformation) {
Shinya Kitaoka 120a6e
  m_imp->m_deformations.insert(deformation);
Toshihiro Shimizu 890ddd
}
Toshihiro Shimizu 890ddd
Toshihiro Shimizu 890ddd
//------------------------------------------------------------------
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
void PlasticSkeleton::removeListener(PlasticSkeletonDeformation *deformation) {
Shinya Kitaoka 120a6e
  m_imp->m_deformations.erase(deformation);
Toshihiro Shimizu 890ddd
}
Toshihiro Shimizu 890ddd
Toshihiro Shimizu 890ddd
//------------------------------------------------------------------
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
bool PlasticSkeleton::setVertexName(int v, const QString &newName) {
Shinya Kitaoka 120a6e
  assert(!newName.isEmpty());
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
  if (m_vertices[v].m_name == newName) return true;
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
  // Traverse the vertices list - if the same name already exists, return false
Shinya Kitaoka 120a6e
  tcg::list<vertex_type>::iterator vt, vEnd(m_vertices.end());</vertex_type>
Shinya Kitaoka 120a6e
  for (vt = m_vertices.begin(); vt != vEnd; ++vt)
Shinya Kitaoka 120a6e
    if (vt.m_idx != v &&
Shinya Kitaoka 120a6e
        vt->m_name == newName)  // v should be skipped in the check
Shinya Kitaoka 120a6e
      return false;
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
  // Notify deformations before changing the name
Shinya Kitaoka 120a6e
  std::set<plasticskeletondeformation *="">::iterator dt,</plasticskeletondeformation>
Shinya Kitaoka 120a6e
      dEnd(m_imp->m_deformations.end());
Shinya Kitaoka 120a6e
  for (dt = m_imp->m_deformations.begin(); dt != dEnd; ++dt)
Shinya Kitaoka 120a6e
    (*dt)->vertexNameChange(this, v, newName);
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
  m_vertices[v].m_name = newName;
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
  return true;
Toshihiro Shimizu 890ddd
}
Toshihiro Shimizu 890ddd
Toshihiro Shimizu 890ddd
//------------------------------------------------------------------
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
void PlasticSkeleton::moveVertex(int v, const TPointD &pos) {
Shinya Kitaoka 120a6e
  mesh_type::vertex(v).P() = pos;  // Apply new position
Toshihiro Shimizu 890ddd
}
Toshihiro Shimizu 890ddd
Toshihiro Shimizu 890ddd
//------------------------------------------------------------------
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
int PlasticSkeleton::addVertex(const PlasticSkeletonVertex &vx, int parent) {
Shinya Kitaoka 120a6e
  // Add the vertex
Shinya Kitaoka 120a6e
  int v = mesh_type::addVertex(vx);
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
  // Retrieve a vertex index
Shinya Kitaoka 120a6e
  PlasticSkeletonVertex &vx_ = vertex(v);
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
  vx_.m_number = m_imp->m_numbersPool.acquire();
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
  // Assign a name to the vertex in case none was given
Shinya Kitaoka 120a6e
  QString name(vx.name());
Shinya Kitaoka 120a6e
  if (name.isEmpty())
Shinya Kitaoka 120a6e
    name = (v == 0) ? QString("Root")
Shinya Kitaoka 120a6e
                    : "Vertex " +
Shinya Kitaoka 120a6e
                          QString::number(vx_.m_number).rightJustified(3, '_');
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
  // Ensure the name is unique
Shinya Kitaoka 120a6e
  while (!setVertexName(v, name)) name += "_";
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
  if (parent >= 0) {
Shinya Kitaoka 120a6e
    // Link it to the parent
Shinya Kitaoka 120a6e
    mesh_type::addEdge(
Shinya Kitaoka 120a6e
        edge_type(parent, v));  // Observe that parent is always v0
Shinya Kitaoka 120a6e
    vx_.m_parent = parent;
Shinya Kitaoka 120a6e
    assert(parent != v);
Shinya Kitaoka 120a6e
  }
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
  // Notify deformations
Shinya Kitaoka 120a6e
  std::set<plasticskeletondeformation *="">::iterator dt,</plasticskeletondeformation>
Shinya Kitaoka 120a6e
      dEnd(m_imp->m_deformations.end());
Shinya Kitaoka 120a6e
  for (dt = m_imp->m_deformations.begin(); dt != dEnd; ++dt)
Shinya Kitaoka 120a6e
    (*dt)->addVertex(this, v);
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
  return v;
Toshihiro Shimizu 890ddd
}
Toshihiro Shimizu 890ddd
Toshihiro Shimizu 890ddd
//-------------------------------------------------------------------------------
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
int PlasticSkeleton::insertVertex(const PlasticSkeletonVertex &vx, int parent,
Shinya Kitaoka 120a6e
                                  const std::vector<int> &children) {</int>
Shinya Kitaoka 120a6e
  assert(parent >= 0);
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
  if (children.empty()) return addVertex(vx, parent);
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
  // Add the vertex
Shinya Kitaoka 120a6e
  int v = mesh_type::addVertex(vx);
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
  // Retrieve a vertex index
Shinya Kitaoka 120a6e
  PlasticSkeletonVertex &vx_ = vertex(v);
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
  vx_.m_number = m_imp->m_numbersPool.acquire();
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
  // Assign a name to the vertex in case none was given
Shinya Kitaoka 120a6e
  QString name(vx.name());
Shinya Kitaoka 120a6e
  if (name.isEmpty())
Shinya Kitaoka 120a6e
    name = "Vertex " + QString::number(vx_.m_number).rightJustified(3, '_');
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
  // Ensure the name is unique
Shinya Kitaoka 120a6e
  while (!setVertexName(v, name)) name += "_";
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
  // Link it to the parent
Shinya Kitaoka 120a6e
  {
Shinya Kitaoka 120a6e
    PlasticSkeletonVertex &vx_ = vertex(v);
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
    mesh_type::addEdge(
Shinya Kitaoka 120a6e
        edge_type(parent, v));  // Observe that parent is always v0
Shinya Kitaoka 120a6e
    vx_.m_parent = parent;
Shinya Kitaoka 120a6e
    assert(parent != v);
Shinya Kitaoka 120a6e
  }
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
  // Link it to children
Shinya Kitaoka 120a6e
  int c, cCount = int(children.size());
Shinya Kitaoka 120a6e
  for (c = 0; c != cCount; ++c) {
Shinya Kitaoka 120a6e
    int vChild                     = children[c];
Shinya Kitaoka 120a6e
    PlasticSkeletonVertex &vxChild = vertex(vChild);
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
    assert(vxChild.parent() == parent);
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
    // Remove the edge and substitute it with a new one
Shinya Kitaoka 120a6e
    int e = edgeInciding(parent, vChild);
Shinya Kitaoka 120a6e
    mesh_type::removeEdge(e);
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
    mesh_type::addEdge(edge_type(v, vChild));
Shinya Kitaoka 120a6e
    vxChild.m_parent = v;
Shinya Kitaoka 120a6e
    assert(v != vChild);
Shinya Kitaoka 120a6e
  }
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
  // Notify deformations
Shinya Kitaoka 120a6e
  std::set<plasticskeletondeformation *="">::iterator dt,</plasticskeletondeformation>
Shinya Kitaoka 120a6e
      dEnd(m_imp->m_deformations.end());
Shinya Kitaoka 120a6e
  for (dt = m_imp->m_deformations.begin(); dt != dEnd; ++dt)
Shinya Kitaoka 120a6e
    (*dt)->insertVertex(this, v);
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
  return v;
Toshihiro Shimizu 890ddd
}
Toshihiro Shimizu 890ddd
Toshihiro Shimizu 890ddd
//-------------------------------------------------------------------------------
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
int PlasticSkeleton::insertVertex(const PlasticSkeletonVertex &vx, int e) {
Shinya Kitaoka 120a6e
  const edge_type &ed = edge(e);
Shinya Kitaoka 120a6e
  return insertVertex(vx, ed.vertex(0), std::vector<int>(1, ed.vertex(1)));</int>
Toshihiro Shimizu 890ddd
}
Toshihiro Shimizu 890ddd
Toshihiro Shimizu 890ddd
//-------------------------------------------------------------------------------
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
void PlasticSkeleton::removeVertex(int v) {
Shinya Kitaoka 120a6e
  int vNumber;
Shinya Kitaoka 120a6e
  {
Shinya Kitaoka 120a6e
    // Reparent all of v's children to its parent's children, first. This is
Shinya Kitaoka 120a6e
    // needed
Shinya Kitaoka 120a6e
    // to ensure that deformations update vertex deformations correctly.
Shinya Kitaoka 120a6e
Shinya Kitaoka 120a6e
    PlasticSkeletonVertex vx(
Shinya Kitaoka 120a6e
        vertex(v));  // Note the COPY - not referencing. It's best since
Shinya Kitaoka 120a6e
    // we'll be iterating and erasing vx's edges at the same time
Shinya Kitaoka 120a6e
    int parent = vx.m_parent;
Shinya Kitaoka 120a6e
    if (parent < 0) {
Shinya Kitaoka 120a6e
      // Root case - clear the whole skeleton
Shinya Kitaoka 120a6e
      clear();  // Should ensure that the next inserted vertex has index 0
Shinya Kitaoka 120a6e
      return;
Shinya Kitaoka 120a6e
    }
Shinya Kitaoka 120a6e
Shinya Kitaoka 120a6e
    // Add edges from parent to vx's children
Shinya Kitaoka 120a6e
    vertex_type::edges_iterator et, eEnd = vx.edgesEnd();
Shinya Kitaoka 120a6e
    for (et = vx.edgesBegin(); et != eEnd; ++et) {
Shinya Kitaoka 120a6e
      int vChild = edge(*et).vertex(1);
Shinya Kitaoka 120a6e
      if (vChild == v) continue;
Shinya Kitaoka 120a6e
Shinya Kitaoka 120a6e
      mesh_type::removeEdge(*et);
Shinya Kitaoka 120a6e
Shinya Kitaoka 120a6e
      mesh_type::addEdge(edge_type(parent, vChild));
Shinya Kitaoka 120a6e
      vertex(vChild).m_parent = parent;
Shinya Kitaoka 120a6e
      assert(vChild != parent);
Shinya Kitaoka 120a6e
    }
Shinya Kitaoka 120a6e
Shinya Kitaoka 120a6e
    vNumber = vx.m_number;
Shinya Kitaoka 120a6e
  }
Shinya Kitaoka 120a6e
Shinya Kitaoka 120a6e
  // Notify deformations BEFORE removing the vertex, so the vertex deformations
Shinya Kitaoka 120a6e
  // are still accessible.
Shinya Kitaoka 120a6e
  std::set<plasticskeletondeformation *="">::iterator dt,</plasticskeletondeformation>
Shinya Kitaoka 120a6e
      dEnd(m_imp->m_deformations.end());
Shinya Kitaoka 120a6e
  for (dt = m_imp->m_deformations.begin(); dt != dEnd; ++dt)
Shinya Kitaoka 120a6e
    (*dt)->deleteVertex(this, v);
Shinya Kitaoka 120a6e
Shinya Kitaoka 120a6e
  // Then, erase v. This already ensures that edges connected with v are
Shinya Kitaoka 120a6e
  // destroyed.
Shinya Kitaoka 120a6e
  mesh_type::removeVertex(v);
Shinya Kitaoka 120a6e
Shinya Kitaoka 120a6e
  m_imp->m_numbersPool.release(vNumber);
Toshihiro Shimizu 890ddd
}
Toshihiro Shimizu 890ddd
Toshihiro Shimizu 890ddd
//-------------------------------------------------------------------------------
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
void PlasticSkeleton::clear() {
Shinya Kitaoka 120a6e
  mesh_type::clear();
Shinya Kitaoka 120a6e
  m_imp->m_numbersPool.clear();
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
  // Notify deformations
Shinya Kitaoka 120a6e
  std::set<plasticskeletondeformation *="">::iterator dt,</plasticskeletondeformation>
Shinya Kitaoka 120a6e
      dEnd(m_imp->m_deformations.end());
Shinya Kitaoka 120a6e
  for (dt = m_imp->m_deformations.begin(); dt != dEnd; ++dt) (*dt)->clear(this);
Toshihiro Shimizu 890ddd
}
Toshihiro Shimizu 890ddd
Toshihiro Shimizu 890ddd
//-------------------------------------------------------------------------------
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
void PlasticSkeleton::squeeze() {
Shinya Kitaoka 120a6e
  // Squeeze associated deformations first
Shinya Kitaoka 120a6e
  int v, e;
Shinya Kitaoka 120a6e
Shinya Kitaoka 120a6e
  // Update indices
Shinya Kitaoka 120a6e
  tcg::list<vertex_type>::iterator vt, vEnd(m_vertices.end());</vertex_type>
Shinya Kitaoka 120a6e
  for (v = 0, vt = m_vertices.begin(); vt != vEnd; ++vt, ++v) vt->setIndex(v);
Shinya Kitaoka 120a6e
Shinya Kitaoka 120a6e
  tcg::list<edge_type>::iterator et, eEnd(m_edges.end());</edge_type>
Shinya Kitaoka 120a6e
  for (e = 0, et = m_edges.begin(); et != eEnd; ++et, ++e) et->setIndex(e);
Shinya Kitaoka 120a6e
Shinya Kitaoka 120a6e
  // Update stored indices
Shinya Kitaoka 120a6e
  for (et = m_edges.begin(); et != eEnd; ++et) {
Shinya Kitaoka 120a6e
    edge_type &ed = *et;
Shinya Kitaoka 120a6e
    ed.setVertex(0, vertex(ed.vertex(0)).getIndex());
Shinya Kitaoka 120a6e
    ed.setVertex(1, vertex(ed.vertex(1)).getIndex());
Shinya Kitaoka 120a6e
  }
Shinya Kitaoka 120a6e
Shinya Kitaoka 120a6e
  for (vt = m_vertices.begin(); vt != vEnd; ++vt) {
Shinya Kitaoka 120a6e
    vertex_type &vx = *vt;
Shinya Kitaoka 120a6e
Shinya Kitaoka 120a6e
    if (vt->m_parent >= 0) vt->m_parent = vertex(vt->m_parent).getIndex();
Shinya Kitaoka 120a6e
Shinya Kitaoka 120a6e
    vertex_type::edges_iterator vet, veEnd(vx.edgesEnd());
Shinya Kitaoka 120a6e
    for (vet = vx.edgesBegin(); vet != veEnd; ++vet)
Shinya Kitaoka 120a6e
      *vet = edge(*vet).getIndex();
Shinya Kitaoka 120a6e
  }
Shinya Kitaoka 120a6e
Shinya Kitaoka 120a6e
  // Finally, rebuild the actual containers
Shinya Kitaoka 120a6e
  if (!m_edges.empty()) {
Shinya Kitaoka 120a6e
    tcg::list<edge_type> temp(m_edges.begin(), m_edges.end());</edge_type>
Shinya Kitaoka 120a6e
    std::swap(m_edges, temp);
Shinya Kitaoka 120a6e
  }
Shinya Kitaoka 120a6e
Shinya Kitaoka 120a6e
  if (!m_vertices.empty()) {
Shinya Kitaoka 120a6e
    tcg::list<vertex_type> temp(m_vertices.begin(), m_vertices.end());</vertex_type>
Shinya Kitaoka 120a6e
    std::swap(m_vertices, temp);
Shinya Kitaoka 120a6e
  }
Toshihiro Shimizu 890ddd
}
Toshihiro Shimizu 890ddd
Toshihiro Shimizu 890ddd
//-------------------------------------------------------------------------------
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
void PlasticSkeleton::saveData(TOStream &os) {
Shinya Kitaoka 120a6e
  // NOTE: Primitives saved by INDEX iteration is NOT COINCIDENTAL - since
Shinya Kitaoka 120a6e
  // the lists' internal linking could have been altered to mismatch the
Shinya Kitaoka 120a6e
  // natural indexing referred to by primitives' data.
Shinya Kitaoka 120a6e
Shinya Kitaoka 120a6e
  if (m_vertices.size() != m_vertices.nodesCount() ||
Shinya Kitaoka 120a6e
      m_edges.size() != m_edges.nodesCount()) {
Shinya Kitaoka 120a6e
    // Ensure that there are no holes in the representation
Shinya Kitaoka 120a6e
    PlasticSkeleton skel(*this);
Shinya Kitaoka 120a6e
Shinya Kitaoka 120a6e
    skel.squeeze();
Shinya Kitaoka 120a6e
    skel.saveData(os);
Shinya Kitaoka 120a6e
Shinya Kitaoka 120a6e
    return;
Shinya Kitaoka 120a6e
  }
Shinya Kitaoka 120a6e
Shinya Kitaoka 120a6e
  // Save vertices
Shinya Kitaoka 120a6e
  os.openChild("V");
Shinya Kitaoka 120a6e
  {
Shinya Kitaoka 120a6e
    int vCount = int(m_vertices.size());
Shinya Kitaoka 120a6e
    os << vCount;
Shinya Kitaoka 120a6e
Shinya Kitaoka 120a6e
    for (int v = 0; v != vCount; ++v) os.child("Vertex") << m_vertices[v];
Shinya Kitaoka 120a6e
  }
Shinya Kitaoka 120a6e
  os.closeChild();
Shinya Kitaoka 120a6e
Shinya Kitaoka 120a6e
  // Save edges
Shinya Kitaoka 120a6e
  os.openChild("E");
Shinya Kitaoka 120a6e
  {
Shinya Kitaoka 120a6e
    int eCount = int(m_edges.size());
Shinya Kitaoka 120a6e
    os << eCount;
Shinya Kitaoka 120a6e
Shinya Kitaoka 120a6e
    for (int e = 0; e != eCount; ++e) {
Shinya Kitaoka 120a6e
      PlasticSkeleton::edge_type &ed = m_edges[e];
Shinya Kitaoka 120a6e
      os << ed.vertex(0) << ed.vertex(1);
Shinya Kitaoka 120a6e
    }
Shinya Kitaoka 120a6e
  }
Shinya Kitaoka 120a6e
  os.closeChild();
Toshihiro Shimizu 890ddd
}
Toshihiro Shimizu 890ddd
Toshihiro Shimizu 890ddd
//-------------------------------------------------------------------------------
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
void PlasticSkeleton::loadData(TIStream &is) {
Shinya Kitaoka 120a6e
  // Ensure the skeleton is clean
Shinya Kitaoka 120a6e
  clear();
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
  // Load vertices
Shinya Kitaoka 120a6e
  std::string str;
Shinya Kitaoka 120a6e
  int i, size;
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
  while (is.openChild(str)) {
Shinya Kitaoka 120a6e
    if (str == "V") {
Shinya Kitaoka 120a6e
      is >> size;
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
      std::vector<int> acquiredVertexNumbers;</int>
Shinya Kitaoka 120a6e
      acquiredVertexNumbers.reserve(size);
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
      m_vertices.reserve(size);
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
      for (i = 0; i < size; ++i) {
Shinya Kitaoka 120a6e
        if (is.openChild(str) && (str == "Vertex")) {
Shinya Kitaoka 120a6e
          vertex_type vx;
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
          is >> vx;
Shinya Kitaoka 120a6e
          int idx = mesh_type::addVertex(vx);
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
          if (vx.m_number < 0) vertex(idx).m_number = vx.m_number = idx + 1;
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
          acquiredVertexNumbers.push_back(vx.m_number);
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
          is.matchEndTag();
Shinya Kitaoka 120a6e
        } else
Shinya Kitaoka 120a6e
          assert(false), is.skipCurrentTag();
Shinya Kitaoka 120a6e
      }
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
      m_imp->m_numbersPool = tcg::indices_pool<int>(</int>
Shinya Kitaoka 120a6e
          acquiredVertexNumbers.begin(), acquiredVertexNumbers.end());
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
      is.matchEndTag();
Shinya Kitaoka 120a6e
    } else if (str == "E") {
Shinya Kitaoka 120a6e
      is >> size;
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
      m_edges.reserve(size);
Shinya Kitaoka 120a6e
      int v0, v1;
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
      for (i = 0; i < size; ++i) {
Shinya Kitaoka 120a6e
        is >> v0 >> v1;
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
        addEdge(edge_type(v0, v1));
Shinya Kitaoka 120a6e
        vertex(v1).m_parent = v0;
Shinya Kitaoka 120a6e
      }
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
      is.matchEndTag();
Shinya Kitaoka 120a6e
    } else
Shinya Kitaoka 120a6e
      assert(false), is.skipCurrentTag();
Shinya Kitaoka 120a6e
  }
Toshihiro Shimizu 890ddd
}
Toshihiro Shimizu 890ddd
Toshihiro Shimizu 890ddd
//************************************************************************************
Toshihiro Shimizu 890ddd
//    PlasticSkeleton  utility functions
Toshihiro Shimizu 890ddd
//************************************************************************************
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
int PlasticSkeleton::closestVertex(const TPointD &pos, double *dist) const {
Shinya Kitaoka 120a6e
  // Traverse vertices
Shinya Kitaoka 120a6e
  double d2, minDist2 = (std::numeric_limits<double>::max)();</double>
Shinya Kitaoka 120a6e
  int v = -1;
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
  tcg::list<vertex_type>::const_iterator vt, vEnd(m_vertices.end());</vertex_type>
Shinya Kitaoka 120a6e
  for (vt = m_vertices.begin(); vt != vEnd; ++vt) {
Shinya Kitaoka 120a6e
    d2                          = tcg::point_ops::dist2(pos, vt->P());
Shinya Kitaoka 120a6e
    if (d2 < minDist2) minDist2 = d2, v = int(vt.m_idx);
Shinya Kitaoka 120a6e
  }
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
  if (dist && v >= 0) *dist = sqrt(minDist2);
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
  return v;
Toshihiro Shimizu 890ddd
}
Toshihiro Shimizu 890ddd
Toshihiro Shimizu 890ddd
//-------------------------------------------------------------------------------
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
int PlasticSkeleton::closestEdge(const TPointD &pos, double *dist) const {
Shinya Kitaoka 120a6e
  // Traverse edges
Shinya Kitaoka 120a6e
  double d, minDist = (std::numeric_limits<double>::max)();</double>
Shinya Kitaoka 120a6e
  int e = -1;
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
  tcg::list<edge_type>::const_iterator et, eEnd(m_edges.end());</edge_type>
Shinya Kitaoka 120a6e
  for (et = m_edges.begin(); et != eEnd; ++et) {
Shinya Kitaoka 120a6e
    const TPointD &vp0 = vertex(et->vertex(0)).P(),
Shinya Kitaoka 120a6e
                  &vp1 = vertex(et->vertex(1)).P();
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
    d                        = tcg::point_ops::segDist(vp0, vp1, pos);
Shinya Kitaoka 120a6e
    if (d < minDist) minDist = d, e = int(et.m_idx);
Shinya Kitaoka 120a6e
  }
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
  if (dist && e >= 0) *dist = minDist;
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
  return e;
Toshihiro Shimizu 890ddd
}
Toshihiro Shimizu 890ddd
Toshihiro Shimizu 890ddd
//-------------------------------------------------------------------------------
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
std::vector<plastichandle> PlasticSkeleton::verticesToHandles() const {</plastichandle>
Shinya Kitaoka 120a6e
  // Someway, PlasticHandle's EXPLICIT unary constructors are not enough
Shinya Kitaoka 120a6e
  // to disambiguate the direct construction of a vector of PlasticHandles
Shinya Kitaoka 120a6e
  // from m_vertices, at least with *gcc*. I guess it could be a compiler bug.
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
  // So, we'll convert them using an explicit casting iterator...
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
  typedef tcg::function < PlasticHandle (PlasticSkeletonVertex::*)() const,
Shinya Kitaoka 120a6e
      &PlasticSkeletonVertex::operator PlasticHandle> Func;
Toshihiro Shimizu 890ddd
Shinya Kitaoka 120a6e
  return std::vector<plastichandle>(</plastichandle>
Shinya Kitaoka 120a6e
      tcg::make_cast_it(m_vertices.begin(), Func()),
Shinya Kitaoka 120a6e
      tcg::make_cast_it(m_vertices.end(), Func()));
Toshihiro Shimizu 890ddd
}