23
#include "PreCompiled.h"
29
#include "MeshFlatteningNurbs.h"
35
double divide(double a, double b)
43
Eigen::VectorXd NurbsBase1D::getKnotSequence(double u_min, double u_max, int u_deg, int num_u_poles)
46
std::vector<double> u_knots;
47
for (int i=0; i < u_deg; i++)
48
u_knots.push_back(u_min);
49
for (int i=0; i < num_u_poles; i++)
50
u_knots.push_back(u_min + (u_max - u_min) * i / (num_u_poles - 1));
51
for (int i=0; i < u_deg; i++)
52
u_knots.push_back(u_max);
53
return Eigen::Map<Eigen::VectorXd>(u_knots.data(), u_knots.size());
56
Eigen::VectorXd NurbsBase1D::getWeightList(Eigen::VectorXd knots, int u_deg)
58
Eigen::VectorXd weights;
59
weights.resize(knots.rows() - u_deg - 1);
66
std::function<double(double)> get_basis(int degree, int i, Eigen::VectorXd knots)
71
return [degree, i, knots](double t)
75
double t_this = knots[i];
76
double t_next = knots[i+1];
78
return (double)(t_next >= t && t >= t_this);
79
return (double)(t_next >= t && t > t_this);
84
return [degree, i, knots](double t)
88
return get_basis(degree, i, knots)(knots[0]);
89
else if (t > knots[knots.rows() - 1])
90
return get_basis(degree, i, knots)(knots[knots.rows() - 1]);
92
double t_this = knots[i];
93
double t_next = knots[i + 1];
94
double t_precog = knots[i + degree];
95
double t_horizon = knots[i + degree + 1];
96
if (t_this == t_horizon)
99
double bottom = (t_precog - t_this);
100
out = divide(t - t_this, bottom) * get_basis(degree - 1, i, knots)(t);
102
bottom = (t_horizon - t_next);
103
out += divide(t_horizon - t, bottom) * get_basis(degree - 1, i + 1, knots)(t);
110
std::function<double(double)> get_basis_derivative(int order, int degree, int i, Eigen::VectorXd knots)
119
return [degree, i, knots, order](double t)
123
if (!(knots[i + degree] - knots[i] == 0))
125
out += get_basis(degree - 1, i, knots)(t) *
126
degree / (knots[i + degree] - knots[i]);
128
if (!(knots[i + degree + 1] - knots[i + 1] == 0))
130
out -= get_basis(degree - 1, i + 1, knots)(t) *
131
degree / (knots[i + degree + 1] - knots[i + 1]);
138
return [degree, i, knots, order](double t)
141
if (!(knots[i + degree] - knots[i] == 0))
143
out += get_basis_derivative(order - 1, degree - 1, i, knots)(t) *
144
degree / (knots[i + degree] - knots[i]);
146
if (!(knots[i + degree + 1] - knots[i + 1] == 0))
148
out -= get_basis_derivative(order - 1, degree - 1, i + 1, knots)(t) *
149
degree / (knots[i + degree + 1] - knots[i + 1]);
157
NurbsBase2D::NurbsBase2D(Eigen::VectorXd u_knots, Eigen::VectorXd v_knots,
158
Eigen::VectorXd weights,
159
int degree_u, int degree_v)
162
this->u_knots = u_knots;
163
this->v_knots = v_knots;
164
this->weights = weights;
165
this->degree_u = degree_u;
166
this->degree_v = degree_v;
167
for (int u_i = 0; u_i < u_knots.size() - degree_u - 1; u_i ++)
168
this->u_functions.push_back(get_basis(degree_u, u_i, u_knots));
169
for (int v_i = 0; v_i < v_knots.size() - degree_v - 1; v_i ++)
170
this->v_functions.push_back(get_basis(degree_v, v_i, v_knots));
173
Eigen::VectorXd NurbsBase2D::getInfluenceVector(Eigen::Vector2d u)
175
Eigen::VectorXd n_u, n_v;
176
double sum_weights = 0;
177
Eigen::VectorXd infl(this->u_functions.size() * this->v_functions.size());
180
n_u.resize(this->u_functions.size());
181
n_v.resize(this->v_functions.size());
183
for (unsigned int i = 0; i < this->u_functions.size(); i ++)
184
n_u[i] = this->u_functions[i](u.x());
185
for (unsigned int i = 0; i < this->v_functions.size(); i ++)
186
n_v[i] = this->v_functions[i](u.y());
188
for (unsigned int u_i = 0; u_i < this->u_functions.size(); u_i++)
190
for (unsigned int v_i = 0; v_i < this->v_functions.size(); v_i++)
192
sum_weights += weights[i] * n_u[u_i] * n_v[v_i];
193
infl[i] = weights[i] * n_u[u_i] * n_v[v_i];
197
return infl / sum_weights;
200
void add_triplets(Eigen::VectorXd values, double row, std::vector<trip> &triplets)
202
for (unsigned int i=0; i < values.size(); i++)
204
if (values(i) != 0.) {
205
triplets.emplace_back(trip(row, i, values(i)));
210
spMat NurbsBase2D::getInfluenceMatrix(Eigen::Matrix<double, Eigen::Dynamic, 2> U)
212
std::vector<trip> triplets;
213
for (unsigned int row_index = 0; row_index < U.rows(); row_index++)
214
add_triplets(this->getInfluenceVector(U.row(row_index)), row_index, triplets);
215
spMat mat(U.rows(), this->u_functions.size() * this->v_functions.size());
216
mat.setFromTriplets(triplets.begin(), triplets.end());
220
void NurbsBase2D::computeFirstDerivatives()
222
for (unsigned int u_i = 0; u_i < u_functions.size(); u_i ++)
223
this->Du_functions.push_back(get_basis_derivative(1, this->degree_u, u_i, this->u_knots));
224
for (unsigned int v_i = 0; v_i < v_functions.size(); v_i ++)
225
this->Dv_functions.push_back(get_basis_derivative(1, this->degree_v, v_i, this->v_knots));
228
void NurbsBase2D::computeSecondDerivatives()
230
for (unsigned int u_i = 0; u_i < u_functions.size(); u_i ++)
231
this->DDu_functions.push_back(get_basis_derivative(2, this->degree_u, u_i, this->u_knots));
232
for (unsigned int v_i = 0; v_i < v_functions.size(); v_i ++)
233
this->DDv_functions.push_back(get_basis_derivative(2, this->degree_v, v_i, this->v_knots));
236
Eigen::VectorXd NurbsBase2D::getDuVector(Eigen::Vector2d u)
238
Eigen::VectorXd A1, A2;
240
A1.resize(this->u_functions.size() * this->v_functions.size());
241
A2.resize(this->u_functions.size() * this->v_functions.size());
245
Eigen::VectorXd n_u, n_v, Dn_u;
246
n_u.resize(this->u_functions.size());
247
Dn_u.resize(this->u_functions.size());
248
n_v.resize(this->v_functions.size());
249
for (unsigned int u_i=0; u_i < this->u_functions.size(); u_i++)
253
n_u[u_i] = this->u_functions[u_i](u.x());
254
Dn_u[u_i] = this->Du_functions[u_i](u.x());
256
for (unsigned int v_i=0; v_i < this->v_functions.size(); v_i++)
258
n_v[v_i] = this->v_functions[v_i](u.y());
262
for (unsigned int u_i=0; u_i < this->u_functions.size(); u_i++)
264
for (unsigned int v_i=0; v_i < this->v_functions.size(); v_i++)
266
C1 = weights[i] * Dn_u[u_i] * n_v[v_i];
267
C2 = weights[i] * n_u[u_i] * n_v[v_i];
275
return (A1 * A3 - A2 * A5) / A3 / A3;
278
Eigen::VectorXd NurbsBase2D::getDvVector(Eigen::Vector2d u)
280
Eigen::VectorXd A1, A2;
282
A1.resize(this->u_functions.size() * this->v_functions.size());
283
A2.resize(this->u_functions.size() * this->v_functions.size());
287
Eigen::VectorXd n_u, n_v, Dn_v;
288
n_u.resize(this->u_functions.size());
289
Dn_v.resize(this->v_functions.size());
290
n_v.resize(this->v_functions.size());
291
for (unsigned int u_i=0; u_i < this->u_functions.size(); u_i++)
293
n_u[u_i] = this->u_functions[u_i](u.x());
295
for (unsigned int v_i=0; v_i < this->v_functions.size(); v_i++)
297
n_v[v_i] = this->v_functions[v_i](u.y());
298
Dn_v[v_i] = this->Dv_functions[v_i](u.y());
301
for (unsigned int u_i=0; u_i < this->u_functions.size(); u_i++)
303
for (unsigned int v_i=0; v_i < this->v_functions.size(); v_i++)
305
C1 = weights[i] * Dn_v[v_i] * n_u[u_i];
306
C2 = weights[i] * n_v[v_i] * n_u[u_i];
314
return (A1 * A3 - A2 * A5) / A3 / A3;
318
spMat NurbsBase2D::getDuMatrix(Eigen::Matrix<double, Eigen::Dynamic, 2> U)
320
std::vector<trip> triplets;
321
for (unsigned int row_index = 0; row_index < U.rows(); row_index++)
322
add_triplets(this->getDuVector(U.row(row_index)), row_index, triplets);
323
spMat mat(U.rows(), this->u_functions.size() * this->v_functions.size());
324
mat.setFromTriplets(triplets.begin(), triplets.end());
328
spMat NurbsBase2D::getDvMatrix(Eigen::Matrix<double, Eigen::Dynamic, 2> U)
330
std::vector<trip> triplets;
331
for (unsigned int row_index = 0; row_index < U.rows(); row_index++)
332
add_triplets(this->getDvVector(U.row(row_index)), row_index, triplets);
333
spMat mat(U.rows(), this->u_functions.size() * this->v_functions.size());
334
mat.setFromTriplets(triplets.begin(), triplets.end());
339
std::tuple<NurbsBase2D, Eigen::MatrixXd> NurbsBase2D::interpolateUBS(
340
Eigen::Matrix<double, Eigen::Dynamic, 3> poles,
348
double u_min = this->u_knots(0);
349
double u_max = this->u_knots(this->u_knots.size() - 1);
350
double v_min = this->v_knots(0);
351
double v_max = this->v_knots(this->v_knots.size() - 1);
352
Eigen::VectorXd weights, u_knots, v_knots;
353
u_knots = NurbsBase1D::getKnotSequence(u_min, u_max, degree_u, num_u_poles);
354
v_knots = NurbsBase1D::getKnotSequence(v_min, v_max, degree_v, num_v_poles);
356
weights.resize((u_knots.rows() - degree_u - 1) * (v_knots.rows() - degree_v - 1));
358
NurbsBase2D new_base(u_knots, v_knots, weights, degree_u, degree_v);
359
Eigen::Matrix<double, Eigen::Dynamic, 2> uv_points = this->getUVMesh(num_u_points, num_v_points);
360
Eigen::Matrix<double, Eigen::Dynamic, 3> xyz_points = this->getInfluenceMatrix(uv_points) * poles;
361
spMat A = new_base.getInfluenceMatrix(uv_points);
362
Eigen::LeastSquaresConjugateGradient<spMat > solver;
364
Eigen::Matrix<double, Eigen::Dynamic, 3> new_poles = solver.solve(xyz_points);
365
return std::tuple<NurbsBase2D, Eigen::MatrixXd >(new_base, new_poles);
368
Eigen::Matrix<double, Eigen::Dynamic, 2> NurbsBase2D::getUVMesh(int num_u_points, int num_v_points)
370
double u_min = this->u_knots(0);
371
double u_max = this->u_knots(this->u_knots.size() - 1);
372
double v_min = this->v_knots(0);
373
double v_max = this->v_knots(this->v_knots.size() - 1);
374
Eigen::Matrix<double, Eigen::Dynamic, 2> uv_points;
375
uv_points.resize(num_u_points * num_v_points, 2);
377
for (int u = 0; u < num_u_points; u++)
379
for (int v = 0; v < num_v_points; v++)
381
uv_points(i, 0) = u_min + (u_max - u_min) * u / (num_u_points - 1);
382
uv_points(i, 1) = v_min + (v_max - v_min) * v / (num_v_points - 1);
391
NurbsBase1D::NurbsBase1D(Eigen::VectorXd u_knots, Eigen::VectorXd weights, int degree_u)
393
this->u_knots = u_knots;
394
this->weights = weights;
395
this->degree_u = degree_u;
396
for (int u_i = 0; u_i < u_knots.size() - degree_u - 1; u_i ++)
397
this->u_functions.push_back(get_basis(degree_u, u_i, u_knots));
400
Eigen::VectorXd NurbsBase1D::getInfluenceVector(double u)
403
double sum_weights = 0;
404
Eigen::VectorXd infl(this->u_functions.size());
406
n_u.resize(this->u_functions.size());
408
for (unsigned int i = 0; i < this->u_functions.size(); i ++)
409
n_u[i] = this->u_functions[i](u);
411
for (unsigned int u_i = 0; u_i < this->u_functions.size(); u_i++)
413
sum_weights += weights[u_i] * n_u[u_i];
414
infl[u_i] = weights[u_i] * n_u[u_i];
416
return infl / sum_weights;
419
spMat NurbsBase1D::getInfluenceMatrix(Eigen::VectorXd u)
421
std::vector<trip> triplets;
422
for (unsigned int row_index = 0; row_index < u.size(); row_index++)
423
add_triplets(this->getInfluenceVector(u[row_index]), row_index, triplets);
424
spMat mat(u.size(), this->u_functions.size());
425
mat.setFromTriplets(triplets.begin(), triplets.end());
429
void NurbsBase1D::computeFirstDerivatives()
431
for (unsigned int u_i = 0; u_i < u_functions.size(); u_i ++)
432
this->Du_functions.push_back(get_basis_derivative(1, this->degree_u, u_i, this->u_knots));
435
void NurbsBase1D::computeSecondDerivatives()
437
for (unsigned int u_i = 0; u_i < u_functions.size(); u_i ++)
438
this->DDu_functions.push_back(get_basis_derivative(2, this->degree_u, u_i, this->u_knots));
442
Eigen::VectorXd NurbsBase1D::getDuVector(double u)
444
Eigen::VectorXd A1, A2;
449
A1.resize(this->u_functions.size());
450
A2.resize(this->u_functions.size());
451
Eigen::VectorXd n_u, Dn_u;
452
n_u.resize(this->u_functions.size());
453
Dn_u.resize(this->u_functions.size());
455
for (unsigned u_i=0; u_i < this->u_functions.size(); u_i++)
457
n_u[u_i] = this->u_functions[u_i](u);
458
Dn_u[u_i] = this->Du_functions[u_i](u);
461
for (unsigned int u_i=0; u_i < this->Du_functions.size(); u_i++)
463
C1 = weights[i] * Dn_u[u_i];
464
C2 = weights[i] * n_u[u_i];
472
return (A1 * C4 - A2 * C3) / C4 / C4 ;
476
spMat NurbsBase1D::getDuMatrix(Eigen::VectorXd U)
478
std::vector<trip> triplets;
479
for (unsigned int row_index = 0; row_index < U.size(); row_index++)
480
add_triplets(this->getDuVector(U[row_index]), row_index, triplets);
481
spMat mat(U.size(), this->u_functions.size());
482
mat.setFromTriplets(triplets.begin(), triplets.end());
486
std::tuple<NurbsBase1D, Eigen::Matrix<double, Eigen::Dynamic, 3>> NurbsBase1D::interpolateUBS(
487
Eigen::Matrix<double,
488
Eigen::Dynamic, 3> poles,
493
double u_min = this->u_knots(0);
494
double u_max = this->u_knots(this->u_knots.size() - 1);
495
Eigen::VectorXd u_knots, weights;
496
u_knots = NurbsBase1D::getKnotSequence(u_min, u_max, degree, num_poles);
497
weights = NurbsBase1D::getWeightList(u_knots, degree);
498
NurbsBase1D new_base(u_knots, weights, degree);
499
Eigen::Matrix<double, Eigen::Dynamic, 1> u_points = this->getUMesh(num_points);
500
Eigen::Matrix<double, Eigen::Dynamic, 3> xyz_points;
501
xyz_points = this->getInfluenceMatrix(u_points) * poles;
502
spMat A = new_base.getInfluenceMatrix(u_points);
503
Eigen::LeastSquaresConjugateGradient<spMat > solver;
505
Eigen::Matrix<double, Eigen::Dynamic, 3> new_poles = solver.solve(xyz_points);
506
return std::tuple<NurbsBase1D, Eigen::Matrix<double, Eigen::Dynamic, 3> >(new_base, new_poles);
509
Eigen::VectorXd NurbsBase1D::getUMesh(int num_u_points)
511
double u_min = this->u_knots(0);
512
double u_max = this->u_knots(this->u_knots.size() - 1);
513
Eigen::Matrix<double, Eigen::Dynamic, 1> u_points;
514
u_points.setLinSpaced(num_u_points, u_min, u_max);