rapier3d/geometry/interaction_groups.rs
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170
#![allow(clippy::bad_bit_mask)] // Clippy will complain about the bitmasks due to Group::NONE being 0.
/// Pairwise filtering using bit masks.
///
/// This filtering method is based on two 32-bit values:
/// - The interaction groups memberships.
/// - The interaction groups filter.
///
/// An interaction is allowed between two filters `a` and `b` when two conditions
/// are met simultaneously:
/// - The groups membership of `a` has at least one bit set to `1` in common with the groups filter of `b`.
/// - The groups membership of `b` has at least one bit set to `1` in common with the groups filter of `a`.
///
/// In other words, interactions are allowed between two filter iff. the following condition is met:
/// ```ignore
/// (self.memberships & rhs.filter) != 0 && (rhs.memberships & self.filter) != 0
/// ```
#[cfg_attr(feature = "serde-serialize", derive(Serialize, Deserialize))]
#[derive(Copy, Clone, Debug, Hash, PartialEq, Eq)]
#[repr(C)]
pub struct InteractionGroups {
/// Groups memberships.
pub memberships: Group,
/// Groups filter.
pub filter: Group,
}
impl InteractionGroups {
/// Initializes with the given interaction groups and interaction mask.
pub const fn new(memberships: Group, filter: Group) -> Self {
Self {
memberships,
filter,
}
}
/// Allow interaction with everything.
pub const fn all() -> Self {
Self::new(Group::ALL, Group::ALL)
}
/// Prevent all interactions.
pub const fn none() -> Self {
Self::new(Group::NONE, Group::NONE)
}
/// Sets the group this filter is part of.
pub const fn with_memberships(mut self, memberships: Group) -> Self {
self.memberships = memberships;
self
}
/// Sets the interaction mask of this filter.
pub const fn with_filter(mut self, filter: Group) -> Self {
self.filter = filter;
self
}
/// Check if interactions should be allowed based on the interaction memberships and filter.
///
/// An interaction is allowed iff. the memberships of `self` contain at least one bit set to 1 in common
/// with the filter of `rhs`, and vice-versa.
#[inline]
pub const fn test(self, rhs: Self) -> bool {
// NOTE: since const ops is not stable, we have to convert `Group` into u32
// to use & operator in const context.
(self.memberships.bits() & rhs.filter.bits()) != 0
&& (rhs.memberships.bits() & self.filter.bits()) != 0
}
}
impl Default for InteractionGroups {
fn default() -> Self {
Self {
memberships: Group::GROUP_1,
filter: Group::ALL,
}
}
}
bitflags::bitflags! {
/// A bit mask identifying groups for interaction.
#[cfg_attr(feature = "serde-serialize", derive(Serialize, Deserialize))]
#[derive(Copy, Clone, PartialEq, Eq, Debug, Hash)]
pub struct Group: u32 {
/// The group n°1.
const GROUP_1 = 1 << 0;
/// The group n°2.
const GROUP_2 = 1 << 1;
/// The group n°3.
const GROUP_3 = 1 << 2;
/// The group n°4.
const GROUP_4 = 1 << 3;
/// The group n°5.
const GROUP_5 = 1 << 4;
/// The group n°6.
const GROUP_6 = 1 << 5;
/// The group n°7.
const GROUP_7 = 1 << 6;
/// The group n°8.
const GROUP_8 = 1 << 7;
/// The group n°9.
const GROUP_9 = 1 << 8;
/// The group n°10.
const GROUP_10 = 1 << 9;
/// The group n°11.
const GROUP_11 = 1 << 10;
/// The group n°12.
const GROUP_12 = 1 << 11;
/// The group n°13.
const GROUP_13 = 1 << 12;
/// The group n°14.
const GROUP_14 = 1 << 13;
/// The group n°15.
const GROUP_15 = 1 << 14;
/// The group n°16.
const GROUP_16 = 1 << 15;
/// The group n°17.
const GROUP_17 = 1 << 16;
/// The group n°18.
const GROUP_18 = 1 << 17;
/// The group n°19.
const GROUP_19 = 1 << 18;
/// The group n°20.
const GROUP_20 = 1 << 19;
/// The group n°21.
const GROUP_21 = 1 << 20;
/// The group n°22.
const GROUP_22 = 1 << 21;
/// The group n°23.
const GROUP_23 = 1 << 22;
/// The group n°24.
const GROUP_24 = 1 << 23;
/// The group n°25.
const GROUP_25 = 1 << 24;
/// The group n°26.
const GROUP_26 = 1 << 25;
/// The group n°27.
const GROUP_27 = 1 << 26;
/// The group n°28.
const GROUP_28 = 1 << 27;
/// The group n°29.
const GROUP_29 = 1 << 28;
/// The group n°30.
const GROUP_30 = 1 << 29;
/// The group n°31.
const GROUP_31 = 1 << 30;
/// The group n°32.
const GROUP_32 = 1 << 31;
/// All of the groups.
const ALL = u32::MAX;
/// None of the groups.
const NONE = 0;
}
}
impl From<u32> for Group {
#[inline]
fn from(val: u32) -> Self {
Self::from_bits_retain(val)
}
}
impl From<Group> for u32 {
#[inline]
fn from(val: Group) -> Self {
val.bits()
}
}