57 if (formulation ==
"Helmholtz")
58 return std::make_shared<Helmholtz>();
59 else if (formulation ==
"Laplacian")
60 return std::make_shared<Laplacian>();
61 else if (formulation ==
"Electrostatics")
62 return std::make_shared<Electrostatics>();
64 else if (formulation ==
"Bilaplacian")
65 return std::make_shared<BilaplacianMain>();
66 else if (formulation ==
"BilaplacianAux")
67 return std::make_shared<BilaplacianAux>();
69 else if (formulation ==
"LinearElasticity")
70 return std::make_shared<LinearElasticity>();
71 else if (formulation ==
"HookeLinearElasticity")
72 return std::make_shared<HookeLinearElasticity>();
73 else if (formulation ==
"IncompressibleLinearElasticity")
74 return std::make_shared<IncompressibleLinearElasticityDispacement>();
75 else if (formulation ==
"IncompressibleLinearElasticityPressure")
76 return std::make_shared<IncompressibleLinearElasticityPressure>();
78 else if (formulation ==
"SaintVenant")
79 return std::make_shared<SaintVenantElasticity>();
80 else if (formulation ==
"NeoHookean")
81 return std::make_shared<NeoHookeanElasticity>();
82 else if (formulation ==
"IsochoricNeoHookean")
83 return std::make_shared<IsochoricNeoHookean>();
84 else if (formulation ==
"MooneyRivlin")
85 return std::make_shared<MooneyRivlinElasticity>();
86 else if (formulation ==
"MooneyRivlin3Param")
87 return std::make_shared<MooneyRivlin3ParamElasticity>();
88 else if (formulation ==
"MooneyRivlin3ParamSymbolic")
89 return std::make_shared<MooneyRivlin3ParamSymbolic>();
90 else if (formulation ==
"MultiModels")
91 return std::make_shared<MultiModel>();
92 else if (formulation ==
"MaterialSum")
93 return std::make_shared<SumModel>();
94 else if (formulation ==
"UnconstrainedOgden")
95 return std::make_shared<UnconstrainedOgdenElasticity>();
96 else if (formulation ==
"IncompressibleOgden")
97 return std::make_shared<IncompressibleOgdenElasticity>();
98 else if (formulation ==
"VolumePenalty")
99 return std::make_shared<VolumePenalty>();
101 else if (formulation ==
"HGOFiber")
102 return std::make_shared<HGOFiber>();
104 else if (formulation ==
"ActiveFiber")
105 return std::make_shared<ActiveFiber>();
107 else if (formulation ==
"Stokes")
108 return std::make_shared<StokesVelocity>();
109 else if (formulation ==
"StokesPressure")
110 return std::make_shared<StokesPressure>();
111 else if (formulation ==
"NavierStokes")
112 return std::make_shared<NavierStokesVelocity>();
113 else if (formulation ==
"OperatorSplitting")
114 return std::make_shared<OperatorSplitting>();
116 else if (formulation ==
"AMIPS")
117 return std::make_shared<AMIPSEnergy>();
118 else if (formulation ==
"FixedCorotational")
119 return std::make_shared<FixedCorotational>();
137 const int n_bases,
const int n_pressure_bases,
const int problem_dim,
const bool add_average,
141 assert(velocity_stiffness.rows() == velocity_stiffness.cols());
142 assert(velocity_stiffness.rows() == n_bases * problem_dim);
144 assert(mixed_stiffness.size() == 0 || mixed_stiffness.rows() == n_bases * problem_dim);
145 assert(mixed_stiffness.size() == 0 || mixed_stiffness.cols() == n_pressure_bases);
147 assert(pressure_stiffness.size() == 0 || pressure_stiffness.rows() == n_pressure_bases);
148 assert(pressure_stiffness.size() == 0 || pressure_stiffness.cols() == n_pressure_bases);
150 const int avg_offset = add_average ? 1 : 0;
152 std::vector<Eigen::Triplet<double>> blocks;
153 blocks.reserve(velocity_stiffness.nonZeros() + 2 * mixed_stiffness.nonZeros() + pressure_stiffness.nonZeros() + 2 * avg_offset * velocity_stiffness.rows());
155 for (
int k = 0; k < velocity_stiffness.outerSize(); ++k)
157 for (StiffnessMatrix::InnerIterator it(velocity_stiffness, k); it; ++it)
159 blocks.emplace_back(it.row(), it.col(), it.value());
163 for (
int k = 0; k < mixed_stiffness.outerSize(); ++k)
165 for (StiffnessMatrix::InnerIterator it(mixed_stiffness, k); it; ++it)
167 blocks.emplace_back(it.row(), n_bases * problem_dim + it.col(), it.value());
168 blocks.emplace_back(it.col() + n_bases * problem_dim, it.row(), it.value());
172 for (
int k = 0; k < pressure_stiffness.outerSize(); ++k)
174 for (StiffnessMatrix::InnerIterator it(pressure_stiffness, k); it; ++it)
176 blocks.emplace_back(n_bases * problem_dim + it.row(), n_bases * problem_dim + it.col(), it.value());
182 const double val = 1.0 / n_pressure_bases;
183 for (
int i = 0; i < n_pressure_bases; ++i)
185 blocks.emplace_back(n_bases * problem_dim + i, n_bases * problem_dim + n_pressure_bases,
val);
186 blocks.emplace_back(n_bases * problem_dim + n_pressure_bases, n_bases * problem_dim + i,
val);
190 stiffness.resize(n_bases * problem_dim + n_pressure_bases + avg_offset, n_bases * problem_dim + n_pressure_bases + avg_offset);
191 stiffness.setFromTriplets(blocks.begin(), blocks.end());
192 stiffness.makeCompressed();
204 if (assembler ==
"Mass")
208 return std::max(basis_degree * 2, 1);
210 return basis_degree * 2 + 1;
212 else if (assembler ==
"NavierStokes")
215 return std::max((basis_degree - 1) + basis_degree, 1);
217 return std::max(basis_degree * 2, 1);
219 return basis_degree * 2 + 1;
227 return std::max((basis_degree - 1) * 2, 1);
238 return std::max(basis_degree * 2, 1);
242 return (basis_degree - 1) * 2 + 1;
static int quadrature_order(const std::string &assembler, const int basis_degree, const BasisType &b_type, const int dim)
utility for retrieving the needed quadrature order to precisely integrate the given form on the given...
static void merge_mixed_matrices(const int n_bases, const int n_pressure_bases, const int problem_dim, const bool add_average, const StiffnessMatrix &velocity_stiffness, const StiffnessMatrix &mixed_stiffness, const StiffnessMatrix &pressure_stiffness, StiffnessMatrix &stiffness)
utility to merge 3 blocks of mixed matrices, A=velocity_stiffness, B=mixed_stiffness,...