fix prop-filter when multiple properties of a name exist

This commit is contained in:
Lennart
2025-12-31 13:46:43 +01:00
parent b632ff6fe8
commit e5d6541ffb
3 changed files with 53 additions and 44 deletions

View File

@@ -30,18 +30,8 @@ pub struct PropFilterElement {
} }
impl PropFilterElement { impl PropFilterElement {
pub fn match_component(&self, comp: &impl PropFilterable) -> bool { #[must_use]
let property = comp.get_property(&self.name); pub fn match_property(&self, property: &Property) -> bool {
let property = match (self.is_not_defined.is_some(), property) {
// We are the component that's not supposed to be defined
(true, Some(_))
// We don't match
| (false, None) => return false,
// We shall not be and indeed we aren't
(true, None) => return true,
(false, Some(property)) => property
};
if let Some(TimeRangeElement { start, end }) = &self.time_range { if let Some(TimeRangeElement { start, end }) = &self.time_range {
// TODO: Respect timezones // TODO: Respect timezones
let Ok(timestamp) = CalDateTime::parse_prop(property, &HashMap::default()) else { let Ok(timestamp) = CalDateTime::parse_prop(property, &HashMap::default()) else {
@@ -77,54 +67,65 @@ impl PropFilterElement {
true true
} }
pub fn match_component(&self, comp: &impl PropFilterable) -> bool {
let properties = comp.get_named_properties(&self.name);
if self.is_not_defined.is_some() {
return properties.is_empty();
}
// The filter matches when one property instance matches
// Example where this matters: We have multiple attendees and want to match one
properties.iter().any(|prop| self.match_property(prop))
}
} }
pub trait PropFilterable { pub trait PropFilterable {
fn get_property(&self, name: &str) -> Option<&Property>; fn get_named_properties(&self, name: &str) -> Vec<&Property>;
} }
impl PropFilterable for CalendarObject { impl PropFilterable for CalendarObject {
fn get_property(&self, name: &str) -> Option<&Property> { fn get_named_properties(&self, name: &str) -> Vec<&Property> {
Self::get_property(self, name) Self::get_named_properties(self, name)
} }
} }
impl PropFilterable for IcalEvent { impl PropFilterable for IcalEvent {
fn get_property(&self, name: &str) -> Option<&Property> { fn get_named_properties(&self, name: &str) -> Vec<&Property> {
Component::get_property(self, name) Component::get_named_properties(self, name)
} }
} }
impl PropFilterable for IcalTodo { impl PropFilterable for IcalTodo {
fn get_property(&self, name: &str) -> Option<&Property> { fn get_named_properties(&self, name: &str) -> Vec<&Property> {
Component::get_property(self, name) Component::get_named_properties(self, name)
} }
} }
impl PropFilterable for IcalJournal { impl PropFilterable for IcalJournal {
fn get_property(&self, name: &str) -> Option<&Property> { fn get_named_properties(&self, name: &str) -> Vec<&Property> {
Component::get_property(self, name) Component::get_named_properties(self, name)
} }
} }
impl PropFilterable for IcalCalendar { impl PropFilterable for IcalCalendar {
fn get_property(&self, name: &str) -> Option<&Property> { fn get_named_properties(&self, name: &str) -> Vec<&Property> {
Component::get_property(self, name) Component::get_named_properties(self, name)
} }
} }
impl PropFilterable for IcalTimeZone { impl PropFilterable for IcalTimeZone {
fn get_property(&self, name: &str) -> Option<&Property> { fn get_named_properties(&self, name: &str) -> Vec<&Property> {
Component::get_property(self, name) Component::get_named_properties(self, name)
} }
} }
impl PropFilterable for CalendarObjectComponent { impl PropFilterable for CalendarObjectComponent {
fn get_property(&self, name: &str) -> Option<&Property> { fn get_named_properties(&self, name: &str) -> Vec<&Property> {
match self { match self {
Self::Event(event, _) => PropFilterable::get_property(&event.event, name), Self::Event(event, _) => PropFilterable::get_named_properties(&event.event, name),
Self::Todo(todo, _) => PropFilterable::get_property(todo, name), Self::Todo(todo, _) => PropFilterable::get_named_properties(todo, name),
Self::Journal(journal, _) => PropFilterable::get_property(journal, name), Self::Journal(journal, _) => PropFilterable::get_named_properties(journal, name),
} }
} }
} }

View File

@@ -33,18 +33,8 @@ pub struct PropFilterElement {
} }
impl PropFilterElement { impl PropFilterElement {
pub fn match_component(&self, comp: &impl PropFilterable) -> bool { #[must_use]
let property = comp.get_property(&self.name); pub fn match_property(&self, property: &Property) -> bool {
let property = match (self.is_not_defined.is_some(), property) {
// We are the component that's not supposed to be defined
(true, Some(_))
// We don't match
| (false, None) => return false,
// We shall not be and indeed we aren't
(true, None) => return true,
(false, Some(property)) => property
};
let allof = match (self.allof.is_some(), self.anyof.is_some()) { let allof = match (self.allof.is_some(), self.anyof.is_some()) {
(true, false) => true, (true, false) => true,
(false, _) => false, (false, _) => false,
@@ -68,14 +58,24 @@ impl PropFilterElement {
matches.any(|a| a) matches.any(|a| a)
} }
} }
pub fn match_component(&self, comp: &impl PropFilterable) -> bool {
let properties = comp.get_named_properties(&self.name);
if self.is_not_defined.is_some() {
return properties.is_empty();
}
// The filter matches when one property instance matches
properties.iter().any(|prop| self.match_property(prop))
}
} }
pub trait PropFilterable { pub trait PropFilterable {
fn get_property(&self, name: &str) -> Option<&Property>; fn get_named_properties(&self, name: &str) -> Vec<&Property>;
} }
impl PropFilterable for AddressObject { impl PropFilterable for AddressObject {
fn get_property(&self, name: &str) -> Option<&Property> { fn get_named_properties(&self, name: &str) -> Vec<&Property> {
self.get_vcard().get_property(name) self.get_vcard().get_named_properties(name)
} }
} }

View File

@@ -328,4 +328,12 @@ impl CalendarObject {
.iter() .iter()
.find(|property| property.name == name) .find(|property| property.name == name)
} }
#[must_use]
pub fn get_named_properties(&self, name: &str) -> Vec<&Property> {
self.properties
.iter()
.filter(|property| property.name == name)
.collect()
}
} }