37 #ifndef COAL_CONTACT_PATCH_SOLVER_HXX
38 #define COAL_CONTACT_PATCH_SOLVER_HXX
77 template <
typename ShapeType1,
typename ShapeType2>
88 contact_patch.
points().clear();
113 this->
reset(s1, tf1, s2, tf2, contact_patch);
129 this->support_set_shape2.size() <= 1) {
135 const Scalar eps = Eigen::NumTraits<Scalar>::dummy_precision();
139 (this->support_set_shape2.size() == 2)) {
144 const Vec2s& a = pts1[0];
145 const Vec2s& b = pts1[1];
148 const Vec2s& c = pts2[0];
149 const Vec2s& d = pts2[1];
152 (b(0) - a(0)) * (d(1) - c(1)) >= (b(1) - a(1)) * (d(0) - c(0));
153 if ((std::abs(det) > eps) || ((c - d).squaredNorm() < eps) ||
154 ((b - a).squaredNorm() < eps)) {
159 const Vec2s cd = (d - c);
160 const Scalar l = cd.squaredNorm();
161 Polygon& patch = contact_patch.
points();
164 Scalar t1 = (a - c).dot(cd);
165 t1 = (t1 >= l) ? 1.0 : ((t1 <= 0) ? 0.0 : (t1 / l));
166 const Vec2s p1 = c + t1 * cd;
167 patch.emplace_back(p1);
170 Scalar t2 = (b - c).dot(cd);
171 t2 = (t2 >= l) ? 1.0 : ((t2 <= 0) ? 0.0 : (t2 / l));
172 const Vec2s p2 = c + t2 * cd;
173 if ((p1 - p2).squaredNorm() >= eps) {
174 patch.emplace_back(p2);
194 const size_t max_result_size =
200 const Polygon* clipper_ptr =
nullptr;
201 Polygon* current_ptr =
nullptr;
214 const Polygon& clipper = *(clipper_ptr);
215 const size_t clipper_size = clipper.size();
216 for (
size_t i = 0; i < clipper_size; ++i) {
220 Polygon* tmp_ptr = previous_ptr;
221 previous_ptr = current_ptr;
222 current_ptr = tmp_ptr;
224 const Polygon& previous = *(previous_ptr);
225 Polygon& current = *(current_ptr);
228 const Vec2s& a = clipper[i];
229 const Vec2s& b = clipper[(i + 1) % clipper_size];
230 const Vec2s ab = b - a;
232 if (previous.size() == 2) {
236 const Vec2s& p1 = previous[0];
237 const Vec2s& p2 = previous[1];
239 const Vec2s ap1 = p1 - a;
240 const Vec2s ap2 = p2 - a;
242 const Scalar det1 = ab(0) * ap1(1) - ab(1) * ap1(0);
243 const Scalar det2 = ab(0) * ap2(1) - ab(1) * ap2(0);
245 if (det1 < 0 && det2 < 0) {
251 if (det1 >= 0 && det2 >= 0) {
263 current.emplace_back(p1);
264 current.emplace_back(p);
269 current.emplace_back(p1);
275 current.emplace_back(p2);
276 current.emplace_back(p);
281 current.emplace_back(p2);
290 this->added_to_patch.end(),
293 const size_t previous_size = previous.size();
294 for (
size_t j = 0; j < previous_size; ++j) {
295 const Vec2s& p1 = previous[j];
296 const Vec2s& p2 = previous[(j + 1) % previous_size];
298 const Vec2s ap1 = p1 - a;
299 const Vec2s ap2 = p2 - a;
301 const Scalar det1 = ab(0) * ap1(1) - ab(1) * ap1(0);
302 const Scalar det2 = ab(0) * ap2(1) - ab(1) * ap2(0);
304 if (det1 < 0 && det2 < 0) {
309 if (det1 >= 0 && det2 >= 0) {
313 current.emplace_back(p1);
323 current.emplace_back(p1);
327 current.emplace_back(p);
331 current.emplace_back(p1);
338 current.emplace_back(p);
341 current.emplace_back(p2);
350 if (current.size() <= 1) {
358 this->
getResult(contact, current_ptr, contact_patch);
365 if (result_ptr->size() <= 1) {
376 template <
typename ShapeType1,
typename ShapeType2>
379 const ShapeType2& shape2,
414 const Vec2s ab = b - a;
415 const Vec2s n(-ab(1), ab(0));
416 const Scalar denominator = n.dot(c - d);
417 if (std::abs(denominator) < std::numeric_limits<Scalar>::epsilon()) {
420 const Scalar nominator = n.dot(a - d);
421 Scalar alpha = nominator / denominator;
422 alpha = std::min<Scalar>(1.0, std::max<Scalar>(0.0, alpha));
423 return alpha * c + (1 - alpha) * d;
Main namespace.
Definition: broadphase_bruteforce.h:44
double Scalar
Definition: data_types.h:68
Eigen::Matrix< Scalar, 2, 1 > Vec2s
Definition: data_types.h:71
void constructContactPatchFrameFromContact(const Contact &contact, ContactPatch &contact_patch)
Construct a frame from a Contact's position and normal. Because both Contact's position and normal ar...
Definition: collision_data.h:703
Definition: geometric_shapes_traits.h:55
#define COAL_TRACY_ZONE_SCOPED_N(x)
Definition: tracy.hh:26