devela/code/result/opt_res/ext_option.rs
1// devela::code::result::option::ext
2//
3//!
4//
5
6use super::{OptionFmt, OptionFmtOr, OptionFmtOrElse};
7use crate::Display;
8
9/// Marker trait to prevent downstream implementations of the [`ExtOption`] trait.
10trait Sealed {}
11impl<T> Sealed for Option<T> {}
12
13/// Extension trait providing additional methods for [`Option`].
14///
15/// This trait is sealed and cannot be implemented for any other type.
16///
17/// See also [`ExtResult`][crate::ExtResult],
18/// [`ExtOptRes`][crate::ExtOptRes].
19///
20/// Based on work from:
21/// - <https://github.com/rust-lang/rust/issues/62358> (contains).
22/// - <https://github.com/rust-lang/rust/pull/87036> (reduce).
23#[cfg_attr(feature = "nightly_doc", doc(notable_trait))]
24#[expect(private_bounds, reason = "Sealed")]
25pub trait ExtOption<T>: Sealed {
26 /// Returns `true` if the option is a [`Some`] value containing the given value.
27 ///
28 /// # Example
29 /// ```
30 /// # use devela::ExtOption;
31 /// assert_eq!(Some(1).contains(&1), true);
32 /// assert_eq!(Some(1).contains(&2), false);
33 /// assert_eq!(None::<u8>.contains(&1), false);
34 /// ```
35 #[must_use]
36 fn contains<U: PartialEq<T>>(&self, x: &U) -> bool;
37
38 /// Merges `self` with another `Option`.
39 ///
40 /// Returns
41 /// - `Some(f(l, r))` if both options are `Some(_)`.
42 /// - `Some(x)` if either of the options is `Some(x)` and the other is `None`.
43 /// - `None` if both options are `None`.
44 ///
45 /// # Example
46 /// ```
47 /// # use devela::ExtOption;
48 /// # use core::{cmp::min, ops::Add};
49 /// let (x, y) = (Some(2), Some(4));
50 ///
51 /// assert_eq!(x.reduce(y, Add::add), Some(6));
52 /// assert_eq!(x.reduce(y, min), Some(2));
53 ///
54 /// assert_eq!(x.reduce(None, Add::add), x);
55 /// assert_eq!(None.reduce(y, min), y);
56 ///
57 /// assert_eq!(None.reduce(None, i32::add), None);
58 /// ```
59 #[must_use]
60 fn reduce<F: FnOnce(T, T) -> T>(self, other: Option<T>, f: F) -> Option<T>;
61
62 /// Format some value, or an alternative if it's `None`.
63 ///
64 /// The alternative value must implement [`Display`]
65 /// regardless of which formatting is used originally.
66 ///
67 /// # Example
68 /// ```
69 /// # use devela::ExtOption;
70 /// assert_eq!("42", format!("{}", Some(Box::new(42)).fmt_or("Nothing")));
71 /// assert_eq!("Nothing", format!("{}", None::<u8>.fmt_or("Nothing")));
72 /// ```
73 #[doc = crate::doc_!(vendor: "fmtor")]
74 #[must_use]
75 fn fmt_or<U: Display>(&self, u: U) -> OptionFmtOr<T, U>;
76
77 /// Format some value, or run an alternative closure if it's `None`.
78 ///
79 /// The value returned from the closure must implement [`Display`]
80 /// regardless of which formatting is used originally.
81 ///
82 /// The value returned from the closure is not stored after use.
83 /// Therefore, using a single [`OptionFmtOrElse`] object for multiple
84 /// formatting operations will run the closure multiple times.
85 ///
86 /// # Example
87 /// ```
88 /// # use devela::ExtOption;
89 /// assert_eq!("42", format!("{}", Some(42).fmt_or_else(|| "Nothing")));
90 /// assert_eq!("Nothing", format!("{}", None::<u8>.fmt_or_else(|| "Nothing")));
91 /// ```
92 #[must_use]
93 fn fmt_or_else<U: Display, F: Fn() -> U>(&self, f: F) -> OptionFmtOrElse<T, F>;
94
95 /// Format some value, or display an empty string if it's `None`.
96 ///
97 /// # Example
98 /// ```
99 /// # use devela::ExtOption;
100 /// assert_eq!("0x42", format!("{:#x}", Some(0x42).fmt_or_empty()));
101 /// assert_eq!("", format!("{:#x}", None::<u8>.fmt_or_empty()));
102 /// ```
103 #[must_use]
104 fn fmt_or_empty(&self) -> OptionFmt<T>;
105}
106
107impl<T> ExtOption<T> for Option<T> {
108 fn contains<U: PartialEq<T>>(&self, x: &U) -> bool {
109 self.as_ref().is_some_and(|y| x == y)
110 }
111
112 fn reduce<F: FnOnce(T, T) -> T>(self, other: Option<T>, f: F) -> Option<T> {
113 match (self, other) {
114 (Some(l), Some(r)) => Some(f(l, r)),
115 (x @ Some(_), None) | (None, x @ Some(_)) => x,
116 (None, None) => None,
117 }
118 }
119
120 fn fmt_or_empty(&self) -> OptionFmt<T> {
121 OptionFmt(self)
122 }
123
124 fn fmt_or<U: Display>(&self, u: U) -> OptionFmtOr<T, U> {
125 OptionFmtOr(self, u)
126 }
127
128 fn fmt_or_else<U: Display, F: Fn() -> U>(&self, f: F) -> OptionFmtOrElse<T, F> {
129 OptionFmtOrElse(self, f)
130 }
131}