rstar/algorithm/
selection_functions.rsuse crate::envelope::Envelope;
use crate::object::PointDistance;
use crate::object::RTreeObject;
use crate::Point;
pub trait SelectionFunction<T>
where
T: RTreeObject,
{
fn should_unpack_parent(&self, envelope: &T::Envelope) -> bool;
fn should_unpack_leaf(&self, _leaf: &T) -> bool {
true
}
}
pub struct SelectInEnvelopeFunction<T>
where
T: RTreeObject,
{
envelope: T::Envelope,
}
impl<T> SelectInEnvelopeFunction<T>
where
T: RTreeObject,
{
pub fn new(envelope: T::Envelope) -> Self {
SelectInEnvelopeFunction { envelope }
}
}
impl<T> SelectionFunction<T> for SelectInEnvelopeFunction<T>
where
T: RTreeObject,
{
fn should_unpack_parent(&self, parent_envelope: &T::Envelope) -> bool {
self.envelope.intersects(parent_envelope)
}
fn should_unpack_leaf(&self, leaf: &T) -> bool {
self.envelope.contains_envelope(&leaf.envelope())
}
}
pub struct SelectInEnvelopeFuncIntersecting<T>
where
T: RTreeObject,
{
envelope: T::Envelope,
}
impl<T> SelectInEnvelopeFuncIntersecting<T>
where
T: RTreeObject,
{
pub fn new(envelope: T::Envelope) -> Self {
SelectInEnvelopeFuncIntersecting { envelope }
}
}
impl<T> SelectionFunction<T> for SelectInEnvelopeFuncIntersecting<T>
where
T: RTreeObject,
{
fn should_unpack_parent(&self, envelope: &T::Envelope) -> bool {
self.envelope.intersects(envelope)
}
fn should_unpack_leaf(&self, leaf: &T) -> bool {
leaf.envelope().intersects(&self.envelope)
}
}
pub struct SelectAllFunc;
impl<T> SelectionFunction<T> for SelectAllFunc
where
T: RTreeObject,
{
fn should_unpack_parent(&self, _: &T::Envelope) -> bool {
true
}
}
pub struct SelectAtPointFunction<T>
where
T: RTreeObject,
{
point: <T::Envelope as Envelope>::Point,
}
impl<T> SelectAtPointFunction<T>
where
T: PointDistance,
{
pub fn new(point: <T::Envelope as Envelope>::Point) -> Self {
SelectAtPointFunction { point }
}
}
impl<T> SelectionFunction<T> for SelectAtPointFunction<T>
where
T: PointDistance,
{
fn should_unpack_parent(&self, envelope: &T::Envelope) -> bool {
envelope.contains_point(&self.point)
}
fn should_unpack_leaf(&self, leaf: &T) -> bool {
leaf.contains_point(&self.point)
}
}
pub struct SelectEqualsFunction<'a, T>
where
T: RTreeObject + PartialEq + 'a,
{
object_to_remove: &'a T,
}
impl<'a, T> SelectEqualsFunction<'a, T>
where
T: RTreeObject + PartialEq,
{
pub fn new(object_to_remove: &'a T) -> Self {
SelectEqualsFunction { object_to_remove }
}
}
impl<'a, T> SelectionFunction<T> for SelectEqualsFunction<'a, T>
where
T: RTreeObject + PartialEq,
{
fn should_unpack_parent(&self, parent_envelope: &T::Envelope) -> bool {
parent_envelope.contains_envelope(&self.object_to_remove.envelope())
}
fn should_unpack_leaf(&self, leaf: &T) -> bool {
leaf == self.object_to_remove
}
}
pub struct SelectWithinDistanceFunction<T>
where
T: RTreeObject + PointDistance,
{
circle_origin: <T::Envelope as Envelope>::Point,
squared_max_distance: <<T::Envelope as Envelope>::Point as Point>::Scalar,
}
impl<T> SelectWithinDistanceFunction<T>
where
T: RTreeObject + PointDistance,
{
pub fn new(
circle_origin: <T::Envelope as Envelope>::Point,
squared_max_distance: <<T::Envelope as Envelope>::Point as Point>::Scalar,
) -> Self {
SelectWithinDistanceFunction {
circle_origin,
squared_max_distance,
}
}
}
impl<T> SelectionFunction<T> for SelectWithinDistanceFunction<T>
where
T: RTreeObject + PointDistance,
{
fn should_unpack_parent(&self, parent_envelope: &T::Envelope) -> bool {
let envelope_distance = parent_envelope.distance_2(&self.circle_origin);
envelope_distance <= self.squared_max_distance
}
fn should_unpack_leaf(&self, leaf: &T) -> bool {
leaf.distance_2_if_less_or_equal(&self.circle_origin, self.squared_max_distance)
.is_some()
}
}
pub struct SelectByAddressFunction<T>
where
T: RTreeObject,
{
envelope: T::Envelope,
element_address: *const T,
}
impl<T> SelectByAddressFunction<T>
where
T: RTreeObject,
{
pub fn new(envelope: T::Envelope, element_address: &T) -> Self {
Self {
envelope,
element_address,
}
}
}
impl<T> SelectionFunction<T> for SelectByAddressFunction<T>
where
T: RTreeObject,
{
fn should_unpack_parent(&self, parent_envelope: &T::Envelope) -> bool {
parent_envelope.contains_envelope(&self.envelope)
}
fn should_unpack_leaf(&self, leaf: &T) -> bool {
core::ptr::eq(self.element_address, leaf)
}
}