20 void scaled_jacobian(
const Eigen::MatrixXd &
V,
const Eigen::MatrixXi &
F, Eigen::VectorXd &quality)
22 const int dim =
F.cols() - 1;
24 quality.setZero(
F.rows());
27 for (
int i = 0; i <
F.rows(); i++)
29 Eigen::RowVector3d e0;
31 e0.head(2) =
V.row(
F(i, 2)) -
V.row(
F(i, 1));
32 Eigen::RowVector3d e1;
34 e1.head(2) =
V.row(
F(i, 0)) -
V.row(
F(i, 2));
35 Eigen::RowVector3d e2;
37 e2.head(2) =
V.row(
F(i, 1)) -
V.row(
F(i, 0));
39 double l0 = e0.norm();
40 double l1 = e1.norm();
41 double l2 = e2.norm();
43 double A = 0.5 * (e0.cross(e1)).
norm();
44 double Lmax = std::max(l0 * l1, std::max(l1 * l2, l0 * l2));
46 quality(i) = 2 * A * (2 / sqrt(3)) / Lmax;
51 for (
int i = 0; i <
F.rows(); i++)
53 Eigen::RowVector3d e0 =
V.row(
F(i, 1)) -
V.row(
F(i, 0));
54 Eigen::RowVector3d e1 =
V.row(
F(i, 2)) -
V.row(
F(i, 1));
55 Eigen::RowVector3d e2 =
V.row(
F(i, 0)) -
V.row(
F(i, 2));
56 Eigen::RowVector3d e3 =
V.row(
F(i, 3)) -
V.row(
F(i, 0));
57 Eigen::RowVector3d e4 =
V.row(
F(i, 3)) -
V.row(
F(i, 1));
58 Eigen::RowVector3d e5 =
V.row(
F(i, 3)) -
V.row(
F(i, 2));
60 double l0 = e0.norm();
61 double l1 = e1.norm();
62 double l2 = e2.norm();
63 double l3 = e3.norm();
64 double l4 = e4.norm();
65 double l5 = e5.norm();
67 double J = std::abs((e0.cross(e3)).dot(e2));
69 double a1 = l0 * l2 * l3;
70 double a2 = l0 * l1 * l4;
71 double a3 = l1 * l2 * l5;
72 double a4 = l3 * l4 * l5;
74 double a = std::max({a1, a2, a3, a4, J});
75 quality(i) = J * sqrt(2) / a;
83 const bool is_volume =
state_->mesh->is_volume();
84 double min_jacs = std::numeric_limits<double>::max();
85 for (
size_t e = 0; e <
state_->geom_bases().size(); ++e)
87 if (
state_->mesh->is_polytope(e))
90 const auto &gbasis =
state_->geom_bases()[e];
91 const int n_local_bases = int(gbasis.bases.size());
94 gbasis.compute_quadrature(quad);
96 std::vector<assembler::AssemblyValues> tmp;
98 Eigen::MatrixXd dx = Eigen::MatrixXd::Zero(quad.
points.rows(), quad.
points.cols());
99 Eigen::MatrixXd dy = Eigen::MatrixXd::Zero(quad.
points.rows(), quad.
points.cols());
102 dz = Eigen::MatrixXd::Zero(quad.
points.rows(), quad.
points.cols());
104 gbasis.evaluate_grads(quad.
points, tmp);
106 for (
int j = 0; j < n_local_bases; ++j)
110 for (std::size_t ii = 0; ii < b.global().size(); ++ii)
112 dx += tmp[j].grad.col(0) * b.global()[ii].node * b.global()[ii].val;
113 dy += tmp[j].grad.col(1) * b.global()[ii].node * b.global()[ii].val;
115 dz += tmp[j].grad.col(2) * b.global()[ii].node * b.global()[ii].val;
119 for (
long i = 0; i < dx.rows(); ++i)
124 tmp << dx.row(i), dy.row(i), dz.row(i);
125 min_jacs = std::min(min_jacs, tmp.determinant());
130 tmp << dx.row(i), dy.row(i);
131 min_jacs = std::min(min_jacs, tmp.determinant());
146 state_(std::move(state))
152 use_rest[
"use_rest_pose"] =
true;
173 Eigen::MatrixXd grad;
174 amips_energy_->assemble_gradient(
state_->mesh->is_volume(),
state_->n_geom_bases,
init_geom_bases_,
init_geom_bases_,
init_ass_vals_cache_, 0, 0,
AdjointTools::map_primitive_to_node_order(*
state_, X -
X_rest), Eigen::VectorXd(), grad);
static std::shared_ptr< Assembler > make_assembler(const std::string &formulation)
Represents one basis function and its gradient.
Eigen::VectorXd apply_parametrization_jacobian(ParameterType type, const legacy::State &target, const Eigen::VectorXd &x, const std::function< Eigen::VectorXd()> &grad) const
Compute parametrization jacobian for all var2sim matching parameter type and output to target state.
bool scaled_jacobian(Mesh3DStorage &hmi, Mesh_Quality &mq)
bool is_flipped(const Eigen::MatrixXd &V, const Eigen::MatrixXi &F)
Determine if any simplex is inverted or collapses.
Eigen::MatrixXd unflatten(const Eigen::VectorXd &x, int dim)
Unflatten rowwises, so every dim elements in x become a row.
Eigen::VectorXd flatten(const Eigen::MatrixXd &X)
Flatten rowwises.
spdlog::logger & adjoint_logger()
Retrieves the current logger for adjoint.
void log_and_throw_adjoint_error(const std::string &msg)