main.cpp 3.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138
  1. #include <igl/readOBJ.h>
  2. #include <igl/readDMAT.h>
  3. #include <igl/viewer/Viewer.h>
  4. #include <igl/barycenter.h>
  5. #include <igl/avg_edge_length.h>
  6. #include <vector>
  7. #include <igl/n_polyvector_general.h>
  8. #include <igl/n_polyvector.h>
  9. #include <igl/local_basis.h>
  10. #include <stdlib.h>
  11. #include <igl/jet.h>
  12. #include <fstream>
  13. // Input mesh
  14. Eigen::MatrixXd V;
  15. Eigen::MatrixXi F;
  16. // Per face bases
  17. Eigen::MatrixXd B1,B2,B3;
  18. // Face barycenters
  19. Eigen::MatrixXd B;
  20. // Scale for visualizing the fields
  21. double global_scale;
  22. // Random length factor
  23. double rand_factor = 5;
  24. // Create a random set of tangent vectors
  25. Eigen::VectorXd random_constraints(const
  26. Eigen::VectorXd& b1, const
  27. Eigen::VectorXd& b2, int n)
  28. {
  29. Eigen::VectorXd r(n*3);
  30. for (unsigned i=0; i<n;++i)
  31. {
  32. double a = (double(rand())/RAND_MAX)*2*M_PI;
  33. double s = 1 + ((double(rand())/RAND_MAX)) * rand_factor;
  34. Eigen::Vector3d t = s * (cos(a) * b1 + sin(a) * b2);
  35. r.block(i*3,0,3,1) = t;
  36. }
  37. return r;
  38. }
  39. bool key_down(igl::Viewer& viewer, unsigned char key, int modifier)
  40. {
  41. using namespace std;
  42. using namespace Eigen;
  43. if (key <'1' || key >'8')
  44. return false;
  45. viewer.data.lines.resize(0,9);
  46. int num = key - '0';
  47. // Interpolate
  48. cerr << "Interpolating " << num << "-PolyVector field" << endl;
  49. VectorXi b(3);
  50. b << 1511, 603, 506;
  51. int numConstraintsToGenerate;
  52. // if it's not a 2-PV or a 1-PV, include a line direction (2 opposite vectors)
  53. // in the field
  54. if (num>=5)
  55. numConstraintsToGenerate = num-2;
  56. else
  57. if (num>=3)
  58. numConstraintsToGenerate = num-1;
  59. else
  60. numConstraintsToGenerate = num;
  61. MatrixXd bc(b.size(),numConstraintsToGenerate*3);
  62. for (unsigned i=0; i<b.size(); ++i)
  63. {
  64. VectorXd t = random_constraints(B1.row(b(i)),B2.row(b(i)),numConstraintsToGenerate);
  65. bc.row(i) = t;
  66. }
  67. VectorXi rootsIndex(num);
  68. for (int i =0; i<numConstraintsToGenerate; ++i)
  69. rootsIndex[i] = i+1;
  70. if (num>=5)
  71. rootsIndex[num-2] = -2;
  72. if (num>=3)
  73. rootsIndex[num-1] = -1;
  74. // Interpolated PolyVector field
  75. Eigen::MatrixXd pvf;
  76. igl::n_polyvector_general(V, F, b, bc, rootsIndex, pvf);
  77. // Highlight in red the constrained faces
  78. MatrixXd C = MatrixXd::Constant(F.rows(),3,1);
  79. for (unsigned i=0; i<b.size();++i)
  80. C.row(b(i)) << 1, 0, 0;
  81. viewer.data.set_colors(C);
  82. for (int n=0; n<num; ++n)
  83. {
  84. const MatrixXd &VF = pvf.block(0,n*3,F.rows(),3);
  85. VectorXd c = VF.rowwise().norm();
  86. MatrixXd C2;
  87. igl::jet(c,1,1+rand_factor,C2);
  88. // viewer.data.add_edges(B - global_scale*VF, B + global_scale*VF , C2);
  89. viewer.data.add_edges(B, B + global_scale*VF , C2);
  90. }
  91. return false;
  92. }
  93. int main(int argc, char *argv[])
  94. {
  95. using namespace Eigen;
  96. using namespace std;
  97. // Load a mesh in OBJ format
  98. igl::readOBJ("../../shared/snail.obj", V, F);
  99. // Compute local basis for faces
  100. igl::local_basis(V,F,B1,B2,B3);
  101. // Compute face barycenters
  102. igl::barycenter(V, F, B);
  103. // Compute scale for visualizing fields
  104. global_scale = .1*igl::avg_edge_length(V, F);
  105. // Make the example deterministic
  106. srand(0);
  107. igl::Viewer viewer;
  108. viewer.data.set_mesh(V, F);
  109. viewer.callback_key_down = &key_down;
  110. viewer.core.show_lines = false;
  111. key_down(viewer,'3',0);
  112. viewer.launch();
  113. }