cursive/src/views/button.rs
Alexandre Bury 2610b697a8 Rustfmt
2018-07-19 19:44:59 -07:00

199 lines
4.9 KiB
Rust

use align::HAlign;
use direction::Direction;
use event::*;
use rect::Rect;
use theme::ColorStyle;
use unicode_width::UnicodeWidthStr;
use vec::Vec2;
use view::View;
use {Cursive, Printer, With};
/// Simple text label with a callback when <Enter> is pressed.
///
/// A button shows its content in a single line and has a fixed size.
///
/// # Examples
///
/// ```
/// # use cursive::views::Button;
/// let quit_button = Button::new("Quit", |s| s.quit());
/// ```
pub struct Button {
label: String,
callback: Callback,
enabled: bool,
last_size: Vec2,
}
impl Button {
/// Creates a new button with the given content and callback.
pub fn new<F, S>(label: S, cb: F) -> Self
where
F: 'static + Fn(&mut Cursive),
S: Into<String>,
{
let label = label.into();
Self::new_raw(format!("<{}>", label), cb)
}
/// Creates a new button without angle brackets.
pub fn new_raw<F, S: Into<String>>(label: S, cb: F) -> Self
where
F: 'static + Fn(&mut Cursive),
{
Button {
label: label.into(),
callback: Callback::from_fn(cb),
enabled: true,
last_size: Vec2::zero(),
}
}
/// Sets the function to be called when the button is pressed.
///
/// Replaces the previous callback.
pub fn set_callback<F>(&mut self, cb: F)
where
F: Fn(&mut Cursive) + 'static,
{
self.callback = Callback::from_fn(cb);
}
/// Disables this view.
///
/// A disabled view cannot be selected.
pub fn disable(&mut self) {
self.enabled = false;
}
/// Disables this view.
///
/// Chainable variant.
pub fn disabled(self) -> Self {
self.with(Self::disable)
}
/// Re-enables this view.
pub fn enable(&mut self) {
self.enabled = true;
}
/// Enable or disable this view.
pub fn set_enabled(&mut self, enabled: bool) {
self.enabled = enabled;
}
/// Returns `true` if this view is enabled.
pub fn is_enabled(&self) -> bool {
self.enabled
}
/// Returns the label for this button.
///
/// Includes brackets.
///
/// # Examples
///
/// ```rust
/// # use cursive::views::Button;
/// let button = Button::new("Quit", |s| s.quit());
/// assert_eq!(button.label(), "<Quit>");
/// ```
pub fn label(&self) -> &str {
&self.label
}
/// Sets the label to the given value.
///
/// This will include brackets.
pub fn set_label<S>(&mut self, label: S)
where
S: Into<String>,
{
self.set_label_raw(format!("<{}>", label.into()));
}
/// Sets the label exactly to the given value.
///
/// This will not include brackets.
pub fn set_label_raw<S>(&mut self, label: S)
where
S: Into<String>,
{
self.label = label.into();
}
fn req_size(&self) -> Vec2 {
Vec2::new(self.label.width(), 1)
}
}
impl View for Button {
fn draw(&self, printer: &Printer) {
if printer.size.x == 0 {
return;
}
let style = if !self.enabled {
ColorStyle::secondary()
} else if !printer.focused {
ColorStyle::primary()
} else {
ColorStyle::highlight()
};
let offset =
HAlign::Center.get_offset(self.label.width(), printer.size.x);
printer.with_color(style, |printer| {
printer.print((offset, 0), &self.label);
});
}
fn layout(&mut self, size: Vec2) {
self.last_size = size;
}
fn required_size(&mut self, _: Vec2) -> Vec2 {
// Meh. Fixed size we are.
self.req_size()
}
fn on_event(&mut self, event: Event) -> EventResult {
// eprintln!("{:?}", event);
// eprintln!("{:?}", self.req_size());
let width = self.label.width();
let self_offset = HAlign::Center.get_offset(width, self.last_size.x);
match event {
// 10 is the ascii code for '\n', that is the return key
Event::Key(Key::Enter) => {
EventResult::Consumed(Some(self.callback.clone()))
}
Event::Mouse {
event: MouseEvent::Release(MouseButton::Left),
position,
offset,
}
if position.fits_in_rect(
offset + (self_offset, 0),
self.req_size(),
) =>
{
EventResult::Consumed(Some(self.callback.clone()))
}
_ => EventResult::Ignored,
}
}
fn take_focus(&mut self, _: Direction) -> bool {
self.enabled
}
fn important_area(&self, view_size: Vec2) -> Rect {
let width = self.label.width();
let offset = HAlign::Center.get_offset(width, view_size.x);
Rect::from_size((offset, 0), (width, 1))
}
}