Simplify theming for QRCode widget

This commit is contained in:
Héctor Ramón Jiménez 2024-03-05 21:16:22 +01:00
parent 704ec9cb5c
commit 330a625205
No known key found for this signature in database
GPG key ID: 7CC46565708259A7
5 changed files with 48 additions and 84 deletions

View file

@ -396,7 +396,7 @@ where
#[cfg(feature = "qr_code")]
pub fn qr_code<Theme>(data: &crate::qr_code::Data) -> crate::QRCode<'_, Theme>
where
Theme: crate::qr_code::StyleSheet,
Theme: crate::qr_code::Style,
{
crate::QRCode::new(data)
}

View file

@ -5,41 +5,37 @@ use crate::core::mouse;
use crate::core::renderer::{self, Renderer as _};
use crate::core::widget::tree::{self, Tree};
use crate::core::{
Element, Layout, Length, Point, Rectangle, Size, Vector, Widget,
Color, Element, Layout, Length, Point, Rectangle, Size, Vector, Widget,
};
use crate::graphics::geometry::Renderer as _;
use crate::style::Theme;
use crate::Renderer;
use std::cell::RefCell;
use thiserror::Error;
pub use crate::style::qr_code::{Appearance, StyleSheet};
const DEFAULT_CELL_SIZE: u16 = 4;
const QUIET_ZONE: usize = 2;
/// A type of matrix barcode consisting of squares arranged in a grid which
/// can be read by an imaging device, such as a camera.
#[derive(Debug)]
pub struct QRCode<'a, Theme = crate::Theme>
where
Theme: StyleSheet,
{
pub struct QRCode<'a, Theme = crate::Theme> {
data: &'a Data,
cell_size: u16,
style: Theme::Style,
style: fn(&Theme) -> Appearance,
}
impl<'a, Theme> QRCode<'a, Theme>
where
Theme: StyleSheet,
{
impl<'a, Theme> QRCode<'a, Theme> {
/// Creates a new [`QRCode`] with the provided [`Data`].
pub fn new(data: &'a Data) -> Self {
pub fn new(data: &'a Data) -> Self
where
Theme: Style,
{
Self {
data,
cell_size: DEFAULT_CELL_SIZE,
style: Default::default(),
style: Theme::style(),
}
}
@ -50,15 +46,14 @@ where
}
/// Sets the style of the [`QRCode`].
pub fn style(mut self, style: impl Into<Theme::Style>) -> Self {
self.style = style.into();
pub fn style(mut self, style: fn(&Theme) -> Appearance) -> Self {
self.style = style;
self
}
}
impl<'a, Message, Theme> Widget<Message, Theme, Renderer> for QRCode<'a, Theme>
where
Theme: StyleSheet,
impl<'a, Message, Theme> Widget<Message, Theme, Renderer>
for QRCode<'a, Theme>
{
fn tag(&self) -> tree::Tag {
tree::Tag::of::<State>()
@ -102,7 +97,7 @@ where
let bounds = layout.bounds();
let side_length = self.data.width + 2 * QUIET_ZONE;
let appearance = theme.appearance(&self.style);
let appearance = (self.style)(theme);
let mut last_appearance = state.last_appearance.borrow_mut();
if Some(appearance) != *last_appearance {
@ -156,7 +151,7 @@ where
impl<'a, Message, Theme> From<QRCode<'a, Theme>>
for Element<'a, Message, Theme, Renderer>
where
Theme: StyleSheet + 'a,
Theme: 'a,
{
fn from(qr_code: QRCode<'a, Theme>) -> Self {
Self::new(qr_code)
@ -330,3 +325,34 @@ impl From<qrcode::types::QrError> for Error {
struct State {
last_appearance: RefCell<Option<Appearance>>,
}
/// The appearance of a QR code.
#[derive(Debug, Clone, Copy, PartialEq)]
pub struct Appearance {
/// The color of the QR code data cells
pub cell: Color,
/// The color of the QR code background
pub background: Color,
}
/// The definiton of the default style of a [`QRCode`].
pub trait Style {
/// Returns the default style of a [`QRCode`].
fn style() -> fn(&Self) -> Appearance;
}
impl Style for Theme {
fn style() -> fn(&Self) -> Appearance {
default
}
}
/// The default style of a [`QRCode`].
pub fn default(theme: &Theme) -> Appearance {
let palette = theme.palette();
Appearance {
cell: palette.text,
background: palette.background,
}
}