use crate::builder::{Config, Context, Error, Resolvable};
use crate::{
direction::Direction,
event::{Event, EventResult, Key, MouseButton, MouseEvent},
style::PaletteStyle,
utils::markup::StyledString,
view::{CannotFocus, View},
Cursive, Printer, Vec2,
};
use std::any::Any;
use std::any::TypeId;
use std::collections::BTreeMap;
use std::sync::{Arc, Mutex};
type Callback<T> = dyn Fn(&mut Cursive, &T) + Send + Sync;
static GROUPS: Mutex<BTreeMap<(String, TypeId), Box<dyn Any + Send + Sync>>> =
Mutex::new(BTreeMap::new());
struct SharedState<T> {
selection: usize,
values: Vec<Arc<T>>,
on_change: Option<Arc<Callback<T>>>,
}
impl<T> SharedState<T> {
pub fn selection(&self) -> Arc<T> {
Arc::clone(&self.values[self.selection])
}
}
pub struct RadioGroup<T> {
state: Arc<Mutex<SharedState<T>>>,
}
impl<T> Clone for RadioGroup<T> {
fn clone(&self) -> Self {
Self {
state: Arc::clone(&self.state),
}
}
}
impl<T: 'static + Send + Sync> Default for RadioGroup<T> {
fn default() -> Self {
Self::new()
}
}
impl<T: 'static + Send + Sync> RadioGroup<T> {
pub fn new() -> Self {
RadioGroup {
state: Arc::new(Mutex::new(SharedState {
selection: 0,
values: Vec::new(),
on_change: None,
})),
}
}
pub fn global<S: Into<String>>(key: S) -> Self {
Self::with_global(key, |group| group.clone())
}
pub fn with_global<F, R, S: Into<String>>(key: S, f: F) -> R
where
F: FnOnce(&mut RadioGroup<T>) -> R,
{
let type_id = TypeId::of::<T>();
let mut groups = GROUPS.lock().unwrap();
let group = groups
.entry((key.into(), type_id))
.or_insert(Box::new(RadioGroup::<T>::new()));
let group = group.downcast_mut().unwrap();
f(group)
}
pub fn button<S: Into<StyledString>>(&mut self, value: T, label: S) -> RadioButton<T> {
let mut state = self.state.lock().unwrap();
let count = state.values.len();
state.values.push(Arc::new(value));
RadioButton::new(Arc::clone(&self.state), count, label.into())
}
pub fn selected_id(&self) -> usize {
self.state.lock().unwrap().selection
}
pub fn selection(&self) -> Arc<T> {
self.state.lock().unwrap().selection()
}
pub fn set_on_change<F: 'static + Fn(&mut Cursive, &T) + Send + Sync>(&mut self, on_change: F) {
self.state.lock().unwrap().on_change = Some(Arc::new(on_change));
}
#[must_use]
pub fn on_change<F: 'static + Fn(&mut Cursive, &T) + Send + Sync>(self, on_change: F) -> Self {
crate::With::with(self, |s| s.set_on_change(on_change))
}
}
impl RadioGroup<String> {
pub fn button_str<S: Into<String>>(&mut self, text: S) -> RadioButton<String> {
let text = text.into();
self.button(text.clone(), text)
}
}
impl RadioButton<String> {
pub fn global_str<S: Into<String>>(key: &str, text: S) -> Self {
RadioGroup::with_global(key, move |group| group.button_str(text))
}
}
pub struct RadioButton<T> {
state: Arc<Mutex<SharedState<T>>>,
id: usize,
enabled: bool,
label: StyledString,
}
impl<T: 'static + Send + Sync> RadioButton<T> {
impl_enabled!(self.enabled);
fn new(state: Arc<Mutex<SharedState<T>>>, id: usize, label: StyledString) -> Self {
RadioButton {
state,
id,
enabled: true,
label,
}
}
pub fn global<S: Into<StyledString>>(key: &str, value: T, label: S) -> Self {
RadioGroup::with_global(key, move |group| group.button(value, label))
}
pub fn is_selected(&self) -> bool {
self.state.lock().unwrap().selection == self.id
}
pub fn select(&mut self) -> EventResult {
let mut state = self.state.lock().unwrap();
state.selection = self.id;
if let Some(ref on_change) = state.on_change {
let on_change = Arc::clone(on_change);
let value = state.selection();
EventResult::with_cb(move |s| on_change(s, &value))
} else {
EventResult::Consumed(None)
}
}
#[must_use]
pub fn selected(self) -> Self {
crate::With::with(self, |s| {
s.select();
})
}
fn draw_internal(&self, printer: &Printer) {
printer.print((0, 0), "( )");
if self.is_selected() {
printer.print((1, 0), "X");
}
if !self.label.is_empty() {
printer.print((3, 0), " ");
printer.print_styled((4, 0), &self.label);
}
}
fn req_size(&self) -> Vec2 {
if self.label.is_empty() {
Vec2::new(3, 1)
} else {
Vec2::new(3 + 1 + self.label.width(), 1)
}
}
pub fn from_group<S: Into<String>>(group: &mut RadioGroup<T>, value: T, label: S) -> Self {
group.button(value, label)
}
}
impl RadioButton<String> {
pub fn from_group_str<S: Into<String>>(group: &mut RadioGroup<String>, label: S) -> Self {
group.button_str(label)
}
}
impl<T: 'static + Send + Sync> View for RadioButton<T> {
fn required_size(&mut self, _: Vec2) -> Vec2 {
self.req_size()
}
fn take_focus(&mut self, _: Direction) -> Result<EventResult, CannotFocus> {
self.enabled.then(EventResult::consumed).ok_or(CannotFocus)
}
fn draw(&self, printer: &Printer) {
if self.enabled && printer.enabled {
printer.with_selection(printer.focused, |printer| self.draw_internal(printer));
} else {
printer.with_style(PaletteStyle::Secondary, |printer| {
self.draw_internal(printer)
});
}
}
fn on_event(&mut self, event: Event) -> EventResult {
if !self.enabled {
return EventResult::Ignored;
}
match event {
Event::Key(Key::Enter) | Event::Char(' ') => self.select(),
Event::Mouse {
event: MouseEvent::Release(MouseButton::Left),
position,
offset,
} if position.fits_in_rect(offset, self.req_size()) => self.select(),
_ => EventResult::Ignored,
}
}
}
impl Resolvable for RadioGroup<String> {
fn from_config(config: &Config, context: &Context) -> Result<Self, Error> {
let name: String = context.resolve(config)?;
Ok(Self::global(name))
}
}
#[crate::blueprint(RadioButton::from_group_str(&mut group, label))]
struct Blueprint {
group: RadioGroup<String>,
label: String,
}