feat: Add AsRef impls for widget types (#2297)

This commit is contained in:
Josh McKinney
2025-12-29 11:16:27 -08:00
committed by GitHub
parent 1c9f56aa4b
commit 01a15f9809
3 changed files with 125 additions and 0 deletions

View File

@@ -10,6 +10,8 @@ GitHub with a [breaking change] label.
This is a quick summary of the sections below: This is a quick summary of the sections below:
- [v0.30.1](#v0301)
- Adding `AsRef` impls for widgets may affect type inference in rare cases
- [v0.30.0](#v0300) - [v0.30.0](#v0300)
- `Flex::SpaceAround` now mirrors flexbox: space between items is twice the size of the outer gaps - `Flex::SpaceAround` now mirrors flexbox: space between items is twice the size of the outer gaps
are twice the size of first and last elements are twice the size of first and last elements
@@ -93,6 +95,17 @@ This is a quick summary of the sections below:
- MSRV is now 1.63.0 - MSRV is now 1.63.0
- `List` no longer ignores empty strings - `List` no longer ignores empty strings
## [v0.30.1](https://github.com/ratatui/ratatui/releases/tag/ratatui-v0.30.1)
### Adding `AsRef` impls for widgets may affect type inference ([#2297])
[#2297]: https://github.com/ratatui/ratatui/pull/2297
Adding `AsRef<Self>` for built-in widgets can change type inference outcomes in rare cases where
`AsRef` is part of a trait bound, and can also conflict with downstream blanket or manual `AsRef`
impls for widget types. If you hit new ambiguity errors, add explicit type annotations or specify
the concrete widget type to guide inference, and remove any redundant `AsRef` impls.
## [v0.30.0](https://github.com/ratatui/ratatui/releases/tag/ratatui-v0.30.0) ## [v0.30.0](https://github.com/ratatui/ratatui/releases/tag/ratatui-v0.30.0)
### `Marker` is now non-exhaustive ([#2236]) ### `Marker` is now non-exhaustive ([#2236])

View File

@@ -0,0 +1,111 @@
/// Implement `AsRef<Self>` for widget types to enable `as_ref()` in generic contexts.
///
/// This keeps widget rendering ergonomic when APIs accept `AsRef<WidgetType>` bounds, avoiding
/// the need for `(&widget).render(...)` just to satisfy a trait bound.
///
/// # Example
///
/// ```rust
/// use ratatui_widgets::block::Block;
///
/// let block = Block::default();
/// let block_ref: &Block<'_> = block.as_ref();
/// ```
///
/// # Generated impls
///
/// ```rust,ignore
/// // Non-generic widgets (e.g. Clear, RatatuiLogo).
/// impl AsRef<Clear> for Clear {
/// fn as_ref(&self) -> &Clear {
/// self
/// }
/// }
///
/// // Generic widgets (e.g. Block with a lifetime, Canvas with a lifetime + type parameter).
/// impl<'a> AsRef<Block<'a>> for Block<'a> {
/// fn as_ref(&self) -> &Block<'a> {
/// self
/// }
/// }
///
/// impl<'a, F> AsRef<Canvas<'a, F>> for Canvas<'a, F>
/// where
/// F: Fn(&mut Context),
/// {
/// fn as_ref(&self) -> &Canvas<'a, F> {
/// self
/// }
/// }
/// ```
macro_rules! impl_as_ref {
($type:ty, <$($gen:tt),+> $(where $($bounds:tt)+)?) => {
impl<$($gen),+> AsRef<$type> for $type $(where $($bounds)+)? {
fn as_ref(&self) -> &$type {
self
}
}
};
($type:ty) => {
impl AsRef<$type> for $type {
fn as_ref(&self) -> &$type {
self
}
}
};
}
impl_as_ref!(crate::barchart::BarChart<'a>, <'a>);
impl_as_ref!(crate::block::Block<'a>, <'a>);
impl_as_ref!(crate::canvas::Canvas<'a, F>, <'a, F> where F: Fn(&mut crate::canvas::Context));
impl_as_ref!(crate::chart::Chart<'a>, <'a>);
impl_as_ref!(crate::clear::Clear);
impl_as_ref!(crate::gauge::Gauge<'a>, <'a>);
impl_as_ref!(crate::gauge::LineGauge<'a>, <'a>);
impl_as_ref!(crate::list::List<'a>, <'a>);
impl_as_ref!(crate::logo::RatatuiLogo);
impl_as_ref!(crate::mascot::RatatuiMascot);
impl_as_ref!(crate::paragraph::Paragraph<'a>, <'a>);
impl_as_ref!(crate::scrollbar::Scrollbar<'a>, <'a>);
impl_as_ref!(crate::sparkline::Sparkline<'a>, <'a>);
impl_as_ref!(crate::table::Table<'a>, <'a>);
impl_as_ref!(crate::tabs::Tabs<'a>, <'a>);
#[cfg(feature = "calendar")]
impl_as_ref!(
crate::calendar::Monthly<'a, DS>,
<'a, DS> where DS: crate::calendar::DateStyler
);
#[cfg(test)]
mod tests {
use alloc::vec;
#[test]
fn widgets_implement_as_ref() {
let _ = crate::barchart::BarChart::default().as_ref();
let _ = crate::block::Block::new().as_ref();
let _ = crate::canvas::Canvas::default().paint(|_| {}).as_ref();
let _ = crate::chart::Chart::new(vec![]).as_ref();
let _ = crate::clear::Clear.as_ref();
let _ = crate::gauge::Gauge::default().as_ref();
let _ = crate::gauge::LineGauge::default().as_ref();
let _ = crate::list::List::new(["foo"]).as_ref();
let _ = crate::logo::RatatuiLogo::default().as_ref();
let _ = crate::mascot::RatatuiMascot::default().as_ref();
let _ = crate::paragraph::Paragraph::new("").as_ref();
let _ = crate::scrollbar::Scrollbar::default().as_ref();
let _ = crate::sparkline::Sparkline::default().as_ref();
let _ = crate::table::Table::default().as_ref();
let _ = crate::tabs::Tabs::default().as_ref();
}
#[cfg(feature = "calendar")]
#[test]
fn calendar_widget_implements_as_ref() {
use time::{Date, Month};
let date = Date::from_calendar_date(2024, Month::January, 1).unwrap();
let _ = crate::calendar::Monthly::new(date, crate::calendar::CalendarEventStore::default())
.as_ref();
}
}

View File

@@ -130,5 +130,6 @@ pub mod tabs;
mod polyfills; mod polyfills;
mod reflow; mod reflow;
mod as_ref;
#[cfg(feature = "calendar")] #[cfg(feature = "calendar")]
pub mod calendar; pub mod calendar;