isolines.h 3.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990
  1. #include <vector>
  2. #include <limits>
  3. #include <stdlib.h>
  4. #include <igl/remove_unreferenced.h>
  5. static void isolines(const Eigen::MatrixXd& V,
  6. const Eigen::MatrixXi& F,
  7. const Eigen::VectorXd& z,
  8. const int grads,
  9. Eigen::MatrixXd& isoV,
  10. Eigen::MatrixXi& isoE)
  11. {
  12. const double min = z.minCoeff(), max = z.maxCoeff();
  13. //Following http://www.alecjacobson.com/weblog/?p=2529
  14. Eigen::VectorXd iso(grads+1);
  15. for(int i=0; i<iso.size(); ++i)
  16. iso(i) = double(i)/double(grads)*(max-min) + min;
  17. Eigen::MatrixXd t12(F.rows(), iso.size()), t23(F.rows(), iso.size()),
  18. t31(F.rows(), iso.size());
  19. for(int i=0; i<t12.rows(); ++i) {
  20. const double z1=z(F(i,0)), z2=z(F(i,1)), z3=z(F(i,2));
  21. const double s12 = z2-z1;
  22. const double s23 = z3-z2;
  23. const double s31 = z1-z3;
  24. for(int j=0; j<t12.cols(); ++j) {
  25. t12(i,j) = (iso(j)-z1) / s12;
  26. t23(i,j) = (iso(j)-z2) / s23;
  27. t31(i,j) = (iso(j)-z3) / s31;
  28. if(t12(i,j)<0 || t12(i,j)>1)
  29. t12(i,j) = std::numeric_limits<double>::quiet_NaN();
  30. if(t23(i,j)<0 || t23(i,j)>1)
  31. t23(i,j) = std::numeric_limits<double>::quiet_NaN();
  32. if(t31(i,j)<0 || t31(i,j)>1)
  33. t31(i,j) = std::numeric_limits<double>::quiet_NaN();
  34. }
  35. }
  36. std::vector<int> F12, F23, F31, I12, I23, I31;
  37. for(int i=0; i<t12.rows(); ++i) {
  38. for(int j=0; j<t12.cols(); ++j) {
  39. if(std::isfinite(t23(i,j)) && std::isfinite(t31(i,j))) {
  40. F12.push_back(i);
  41. I12.push_back(j);
  42. }
  43. if(std::isfinite(t31(i,j)) && std::isfinite(t12(i,j))) {
  44. F23.push_back(i);
  45. I23.push_back(j);
  46. }
  47. if(std::isfinite(t12(i,j)) && std::isfinite(t23(i,j))) {
  48. F31.push_back(i);
  49. I31.push_back(j);
  50. }
  51. }
  52. }
  53. const int K = F12.size()+F23.size()+F31.size();
  54. isoV.resize(2*K, 3);
  55. int b = 0;
  56. for(int i=0; i<F12.size(); ++i) {
  57. isoV.row(b+i) = (1.-t23(F12[i],I12[i]))*V.row(F(F12[i],1))
  58. + t23(F12[i],I12[i])*V.row(F(F12[i],2));
  59. isoV.row(K+b+i) = (1.-t31(F12[i],I12[i]))*V.row(F(F12[i],2))
  60. + t31(F12[i],I12[i])*V.row(F(F12[i],0));
  61. }
  62. b += F12.size();
  63. for(int i=0; i<F23.size(); ++i) {
  64. isoV.row(b+i) = (1.-t31(F23[i],I23[i]))*V.row(F(F23[i],2))
  65. + t31(F23[i],I23[i])*V.row(F(F23[i],0));
  66. isoV.row(K+b+i) = (1.-t12(F23[i],I23[i]))*V.row(F(F23[i],0))
  67. + t12(F23[i],I23[i])*V.row(F(F23[i],1));
  68. }
  69. b += F23.size();
  70. for(int i=0; i<F31.size(); ++i) {
  71. isoV.row(b+i) = (1.-t12(F31[i],I31[i]))*V.row(F(F31[i],0))
  72. + t12(F31[i],I31[i])*V.row(F(F31[i],1));
  73. isoV.row(K+b+i) = (1.-t23(F31[i],I31[i]))*V.row(F(F31[i],1))
  74. + t23(F31[i],I31[i])*V.row(F(F31[i],2));
  75. }
  76. isoE.resize(K,2);
  77. for(int i=0; i<K; ++i)
  78. isoE.row(i) << i, K+i;
  79. }