icu_datetime/provider/pattern/runtime/helpers.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
// This file is part of ICU4X. For terms of use, please see the file
// called LICENSE at the top level of the ICU4X source tree
// (online at: https://github.com/unicode-org/icu4x/blob/main/LICENSE ).
#[cfg(feature = "datagen")]
use super::{super::PatternItem, Pattern};
#[cfg(feature = "datagen")]
use zerovec::ule::AsULE;
/// Helper function which takes a runtime `Pattern` and calls
/// the callback function passing each item until the callback
/// returns a new value.
///
/// When that happens, the `Pattern` is turned into an owned one,
/// and the old item gets replaced with the new one.
///
/// The utility of this function is to allow for a single
/// item to be replaced allocating the `Pattern` only if needed.
///
/// For a variant that replaces all matching instances, see `maybe_replace`.
#[cfg(feature = "datagen")]
pub fn maybe_replace_first(pattern: &mut Pattern, f: impl Fn(&PatternItem) -> Option<PatternItem>) {
let result = pattern
.items
.iter()
.enumerate()
.find_map(|(i, item)| f(&item).map(|result| (i, result)));
#[allow(clippy::indexing_slicing)] // i was produced by enumerate
if let Some((i, result)) = result {
pattern.items.to_mut_slice()[i] = result.to_unaligned();
}
}
/// Helper function which takes a runtime `Pattern` and calls
/// the callback function passing each item.
/// If the callback returns a new value, the old one gets
/// replaced with it.
///
/// The utility of this function is to allow for a pattern
/// to be altered, allocating the `Pattern` only if needed.
///
/// For a variant that replaces just the first matching instance,
/// see `maybe_replace_first`.
#[cfg(feature = "datagen")]
pub fn maybe_replace(pattern: &mut Pattern, f: impl Fn(&PatternItem) -> Option<PatternItem>) {
let result = pattern
.items
.iter()
.enumerate()
.find_map(|(i, item)| f(&item).map(|result| (i, result)));
#[allow(clippy::indexing_slicing)] // i was produced by enumerate
if let Some((i, result)) = result {
let owned = pattern.items.to_mut_slice();
owned[i] = result.to_unaligned();
owned.iter_mut().skip(i).for_each(|item| {
if let Some(new_item) = f(&PatternItem::from_unaligned(*item)) {
*item = new_item.to_unaligned();
}
});
}
}