main.cpp 3.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156
  1. #include <igl/avg_edge_length.h>
  2. #include <igl/barycenter.h>
  3. #include <igl/jet.h>
  4. #include <igl/local_basis.h>
  5. #include <igl/n_polyvector.h>
  6. #include <igl/readDMAT.h>
  7. #include <igl/readOBJ.h>
  8. #include <igl/viewer/Viewer.h>
  9. #include <cstdlib>
  10. #include <iostream>
  11. #include <vector>
  12. #include "tutorial_shared_path.h"
  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. Eigen::VectorXi samples;
  25. void readSamples(const std::string &fname, Eigen::VectorXi &samples)
  26. {
  27. int numSamples;
  28. FILE *fp = fopen(fname.c_str(),"r");
  29. if (fscanf(fp, "%d", &numSamples)!=1)
  30. {
  31. fclose(fp);
  32. return;
  33. }
  34. samples.resize(numSamples,1);
  35. int vali;
  36. for (int i =0; i<numSamples; ++i)
  37. {
  38. if (fscanf(fp, "%d", &vali)!=1 || vali<0)
  39. {
  40. fclose(fp);
  41. samples.resize(0,1);
  42. return;
  43. }
  44. samples[i]=vali;
  45. }
  46. fclose(fp);
  47. }
  48. // Create a random set of tangent vectors
  49. Eigen::VectorXd random_constraints(const
  50. Eigen::VectorXd& b1, const
  51. Eigen::VectorXd& b2, int n)
  52. {
  53. Eigen::VectorXd r(n*3);
  54. for (unsigned i=0; i<n;++i)
  55. {
  56. double a = (double(rand())/RAND_MAX)*2*M_PI;
  57. double s = 1 + ((double(rand())/RAND_MAX)) * rand_factor;
  58. Eigen::Vector3d t = s * (cos(a) * b1 + sin(a) * b2);
  59. r.block(i*3,0,3,1) = t;
  60. }
  61. return r;
  62. }
  63. bool key_down(igl::viewer::Viewer& viewer, unsigned char key, int modifier)
  64. {
  65. using namespace std;
  66. using namespace Eigen;
  67. if (key <'1' || key >'8')
  68. return false;
  69. viewer.data.lines.resize(0,9);
  70. int num = key - '0';
  71. // Interpolate
  72. std::cerr << "Interpolating " << num * 2 << "-PolyVector field" << std::endl;
  73. VectorXi b(4);
  74. b << 4550, 2321, 5413, 5350;
  75. MatrixXd bc(b.size(),num*3);
  76. for (unsigned i=0; i<b.size(); ++i)
  77. {
  78. VectorXd t = random_constraints(B1.row(b(i)),B2.row(b(i)),num);
  79. bc.row(i) = t;
  80. }
  81. // Interpolated PolyVector field
  82. Eigen::MatrixXd pvf;
  83. igl::n_polyvector(V, F, b, bc, pvf);
  84. // Highlight in red the constrained faces
  85. MatrixXd C = MatrixXd::Constant(F.rows(),3,1);
  86. for (unsigned i=0; i<b.size();++i)
  87. C.row(b(i)) << 1, 0, 0;
  88. viewer.data.set_colors(C);
  89. for (int n=0; n<num; ++n)
  90. {
  91. MatrixXd VF = MatrixXd::Zero(F.rows(),3);
  92. for (unsigned i=0; i<b.size(); ++i)
  93. VF.row(b[i]) = bc.row(i);
  94. for (int i=0; i<samples.rows(); ++i)
  95. VF.row(samples[i]) = pvf.block(samples[i],n*3,1,3);
  96. // MatrixXd VF = pvf.block(0,n*3,F.rows(),3);
  97. VectorXd c = VF.rowwise().norm();
  98. MatrixXd C2;
  99. igl::jet(c,1,1+rand_factor,C2);
  100. viewer.data.add_edges(B - global_scale*VF, B + global_scale*VF , C2);
  101. }
  102. return false;
  103. }
  104. int main(int argc, char *argv[])
  105. {
  106. using namespace Eigen;
  107. using namespace std;
  108. // Load a mesh in OBJ format
  109. igl::readOBJ(TUTORIAL_SHARED_PATH "/lilium.obj", V, F);
  110. readSamples(TUTORIAL_SHARED_PATH "/lilium.samples.0.2", samples);
  111. // Compute local basis for faces
  112. igl::local_basis(V,F,B1,B2,B3);
  113. // Compute face barycenters
  114. igl::barycenter(V, F, B);
  115. // Compute scale for visualizing fields
  116. global_scale = .2*igl::avg_edge_length(V, F);
  117. // Make the example deterministic
  118. srand(0);
  119. igl::viewer::Viewer viewer;
  120. viewer.data.set_mesh(V, F);
  121. viewer.callback_key_down = &key_down;
  122. viewer.core.show_lines = false;
  123. viewer.core.line_width = 10000;// this does not work, why?
  124. key_down(viewer,'2',0);
  125. viewer.launch();
  126. }