2020-04-24 16:47:14 +00:00
|
|
|
#![allow(clippy::new_without_default)]
|
2019-02-10 14:30:48 +00:00
|
|
|
|
2020-04-25 09:01:57 +00:00
|
|
|
use std::ops::RangeInclusive;
|
|
|
|
|
2019-01-10 09:55:38 +00:00
|
|
|
use crate::{
|
2020-04-17 21:44:14 +00:00
|
|
|
layout::{Direction, GuiResponse},
|
2020-04-17 13:29:48 +00:00
|
|
|
*,
|
2019-01-10 09:55:38 +00:00
|
|
|
};
|
|
|
|
|
2020-04-29 19:25:49 +00:00
|
|
|
mod text_edit;
|
|
|
|
pub use text_edit::*;
|
|
|
|
|
2019-01-10 09:55:38 +00:00
|
|
|
// ----------------------------------------------------------------------------
|
|
|
|
|
|
|
|
/// Anything implementing Widget can be added to a Region with Region::add
|
|
|
|
pub trait Widget {
|
2020-04-29 19:58:41 +00:00
|
|
|
// TODO: rename .ui(
|
2019-01-10 09:55:38 +00:00
|
|
|
fn add_to(self, region: &mut Region) -> GuiResponse;
|
|
|
|
}
|
|
|
|
|
|
|
|
// ----------------------------------------------------------------------------
|
|
|
|
|
|
|
|
pub struct Label {
|
|
|
|
text: String,
|
2020-04-12 10:07:51 +00:00
|
|
|
text_style: TextStyle, // TODO: Option<TextStyle>, where None means "use the default for the region"
|
2019-01-16 15:28:43 +00:00
|
|
|
text_color: Option<Color>,
|
2019-01-10 09:55:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
impl Label {
|
2020-04-24 16:47:14 +00:00
|
|
|
pub fn new(text: impl Into<String>) -> Self {
|
2019-01-12 23:55:56 +00:00
|
|
|
Label {
|
|
|
|
text: text.into(),
|
|
|
|
text_style: TextStyle::Body,
|
2019-01-16 15:28:43 +00:00
|
|
|
text_color: None,
|
2019-01-12 23:55:56 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn text_style(mut self, text_style: TextStyle) -> Self {
|
|
|
|
self.text_style = text_style;
|
|
|
|
self
|
2019-01-10 09:55:38 +00:00
|
|
|
}
|
2019-01-16 15:28:43 +00:00
|
|
|
|
|
|
|
pub fn text_color(mut self, text_color: Color) -> Self {
|
|
|
|
self.text_color = Some(text_color);
|
|
|
|
self
|
|
|
|
}
|
2019-01-10 09:55:38 +00:00
|
|
|
}
|
|
|
|
|
2019-01-21 07:48:32 +00:00
|
|
|
/// Usage: label!("Foo: {}", bar)
|
|
|
|
#[macro_export]
|
|
|
|
macro_rules! label {
|
|
|
|
($fmt:expr) => (Label::new($fmt));
|
|
|
|
($fmt:expr, $($arg:tt)*) => (Label::new(format!($fmt, $($arg)*)));
|
2019-01-10 09:55:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
impl Widget for Label {
|
|
|
|
fn add_to(self, region: &mut Region) -> GuiResponse {
|
2019-01-12 23:55:56 +00:00
|
|
|
let font = ®ion.fonts()[self.text_style];
|
2020-04-19 22:48:54 +00:00
|
|
|
let (text, text_size) = font.layout_multiline(&self.text, region.available_width());
|
2019-01-14 13:26:02 +00:00
|
|
|
let interact = region.reserve_space(text_size, None);
|
2020-04-25 13:45:38 +00:00
|
|
|
region.add_text(interact.rect.min, self.text_style, text, self.text_color);
|
2019-01-10 09:55:38 +00:00
|
|
|
region.response(interact)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// ----------------------------------------------------------------------------
|
|
|
|
|
2020-04-23 17:15:17 +00:00
|
|
|
pub struct Hyperlink {
|
|
|
|
url: String,
|
|
|
|
text: String,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Hyperlink {
|
|
|
|
pub fn new(url: impl Into<String>) -> Self {
|
|
|
|
let url = url.into();
|
|
|
|
Self {
|
|
|
|
text: url.clone(),
|
|
|
|
url,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Widget for Hyperlink {
|
|
|
|
fn add_to(self, region: &mut Region) -> GuiResponse {
|
|
|
|
let color = color::LIGHT_BLUE;
|
|
|
|
let text_style = TextStyle::Body;
|
|
|
|
let id = region.make_child_id(&self.url);
|
|
|
|
let font = ®ion.fonts()[text_style];
|
|
|
|
let line_spacing = font.line_spacing();
|
|
|
|
// TODO: underline
|
|
|
|
let (text, text_size) = font.layout_multiline(&self.text, region.available_width());
|
|
|
|
let interact = region.reserve_space(text_size, Some(id));
|
|
|
|
if interact.hovered {
|
|
|
|
region.ctx().output.lock().cursor_icon = CursorIcon::PointingHand;
|
|
|
|
}
|
|
|
|
if interact.clicked {
|
2020-04-24 16:47:14 +00:00
|
|
|
region.ctx().output.lock().open_url = Some(self.url);
|
2020-04-23 17:15:17 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if interact.hovered {
|
|
|
|
// Underline:
|
|
|
|
for fragment in &text {
|
2020-04-25 13:45:38 +00:00
|
|
|
let pos = interact.rect.min;
|
2020-04-23 17:15:17 +00:00
|
|
|
let y = pos.y + fragment.y_offset + line_spacing;
|
|
|
|
let y = region.round_to_pixel(y);
|
|
|
|
let min_x = pos.x + fragment.min_x();
|
|
|
|
let max_x = pos.x + fragment.max_x();
|
|
|
|
region.add_paint_cmd(PaintCmd::Line {
|
|
|
|
points: vec![pos2(min_x, y), pos2(max_x, y)],
|
|
|
|
color,
|
|
|
|
width: region.style().line_width,
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-04-25 13:45:38 +00:00
|
|
|
region.add_text(interact.rect.min, text_style, text, Some(color));
|
2020-04-23 17:15:17 +00:00
|
|
|
|
|
|
|
region.response(interact)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// ----------------------------------------------------------------------------
|
|
|
|
|
2019-01-10 09:55:38 +00:00
|
|
|
pub struct Button {
|
|
|
|
text: String,
|
2019-01-16 15:28:43 +00:00
|
|
|
text_color: Option<Color>,
|
2019-01-10 09:55:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
impl Button {
|
2020-04-24 16:47:14 +00:00
|
|
|
pub fn new(text: impl Into<String>) -> Self {
|
2019-01-16 15:28:43 +00:00
|
|
|
Button {
|
|
|
|
text: text.into(),
|
|
|
|
text_color: None,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn text_color(mut self, text_color: Color) -> Self {
|
|
|
|
self.text_color = Some(text_color);
|
|
|
|
self
|
2019-01-10 09:55:38 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Widget for Button {
|
|
|
|
fn add_to(self, region: &mut Region) -> GuiResponse {
|
2020-04-19 09:13:24 +00:00
|
|
|
let id = region.make_position_id();
|
2019-01-12 23:55:56 +00:00
|
|
|
let text_style = TextStyle::Button;
|
|
|
|
let font = ®ion.fonts()[text_style];
|
2020-04-19 22:48:54 +00:00
|
|
|
let (text, text_size) = font.layout_multiline(&self.text, region.available_width());
|
2019-03-11 14:39:54 +00:00
|
|
|
let padding = region.style().button_padding;
|
2019-01-19 16:09:00 +00:00
|
|
|
let mut size = text_size + 2.0 * padding;
|
2019-03-11 14:39:54 +00:00
|
|
|
size.y = size.y.max(region.style().clickable_diameter);
|
2019-01-19 16:09:00 +00:00
|
|
|
let interact = region.reserve_space(size, Some(id));
|
2020-04-21 14:50:56 +00:00
|
|
|
let mut text_cursor = interact.rect.left_center() + vec2(padding.x, -0.5 * text_size.y);
|
|
|
|
text_cursor.y += 2.0; // TODO: why is this needed?
|
2019-03-11 14:59:49 +00:00
|
|
|
region.add_paint_cmd(PaintCmd::Rect {
|
2020-04-22 16:16:06 +00:00
|
|
|
corner_radius: region.style().interact_corner_radius(&interact),
|
2020-04-19 21:34:34 +00:00
|
|
|
fill_color: region.style().interact_fill_color(&interact),
|
2020-04-22 16:16:06 +00:00
|
|
|
outline: region.style().interact_outline(&interact),
|
2019-03-11 14:59:49 +00:00
|
|
|
rect: interact.rect,
|
|
|
|
});
|
2020-04-21 14:50:56 +00:00
|
|
|
let stroke_color = region.style().interact_stroke_color(&interact);
|
|
|
|
let text_color = self.text_color.unwrap_or(stroke_color);
|
|
|
|
region.add_text(text_cursor, text_style, text, Some(text_color));
|
2019-01-10 09:55:38 +00:00
|
|
|
region.response(interact)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// ----------------------------------------------------------------------------
|
|
|
|
|
|
|
|
#[derive(Debug)]
|
|
|
|
pub struct Checkbox<'a> {
|
|
|
|
checked: &'a mut bool,
|
|
|
|
text: String,
|
2019-01-16 15:28:43 +00:00
|
|
|
text_color: Option<Color>,
|
2019-01-10 09:55:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
impl<'a> Checkbox<'a> {
|
2020-04-24 16:47:14 +00:00
|
|
|
pub fn new(checked: &'a mut bool, text: impl Into<String>) -> Self {
|
2019-01-10 09:55:38 +00:00
|
|
|
Checkbox {
|
|
|
|
checked,
|
|
|
|
text: text.into(),
|
2019-01-16 15:28:43 +00:00
|
|
|
text_color: None,
|
2019-01-10 09:55:38 +00:00
|
|
|
}
|
|
|
|
}
|
2019-01-16 15:28:43 +00:00
|
|
|
|
|
|
|
pub fn text_color(mut self, text_color: Color) -> Self {
|
|
|
|
self.text_color = Some(text_color);
|
|
|
|
self
|
|
|
|
}
|
2019-01-10 09:55:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
impl<'a> Widget for Checkbox<'a> {
|
|
|
|
fn add_to(self, region: &mut Region) -> GuiResponse {
|
2020-04-19 09:13:24 +00:00
|
|
|
let id = region.make_position_id();
|
2019-01-12 23:55:56 +00:00
|
|
|
let text_style = TextStyle::Button;
|
|
|
|
let font = ®ion.fonts()[text_style];
|
2020-04-19 22:48:54 +00:00
|
|
|
let (text, text_size) = font.layout_multiline(&self.text, region.available_width());
|
2019-01-14 13:26:02 +00:00
|
|
|
let interact = region.reserve_space(
|
2019-03-11 14:39:54 +00:00
|
|
|
region.style().button_padding
|
|
|
|
+ vec2(region.style().start_icon_width, 0.0)
|
2019-01-10 09:55:38 +00:00
|
|
|
+ text_size
|
2019-03-11 14:39:54 +00:00
|
|
|
+ region.style().button_padding,
|
2019-01-10 09:55:38 +00:00
|
|
|
Some(id),
|
|
|
|
);
|
2020-04-25 13:45:38 +00:00
|
|
|
let text_cursor = interact.rect.min
|
2019-03-11 14:39:54 +00:00
|
|
|
+ region.style().button_padding
|
|
|
|
+ vec2(region.style().start_icon_width, 0.0);
|
2019-01-10 09:55:38 +00:00
|
|
|
if interact.clicked {
|
|
|
|
*self.checked = !*self.checked;
|
|
|
|
}
|
2019-03-11 14:59:49 +00:00
|
|
|
let (small_icon_rect, big_icon_rect) = region.style().icon_rectangles(&interact.rect);
|
|
|
|
region.add_paint_cmd(PaintCmd::Rect {
|
|
|
|
corner_radius: 3.0,
|
2020-04-19 21:34:34 +00:00
|
|
|
fill_color: region.style().interact_fill_color(&interact),
|
2019-03-11 14:59:49 +00:00
|
|
|
outline: None,
|
|
|
|
rect: big_icon_rect,
|
2019-01-10 09:55:38 +00:00
|
|
|
});
|
2019-03-11 14:59:49 +00:00
|
|
|
|
|
|
|
let stroke_color = region.style().interact_stroke_color(&interact);
|
|
|
|
|
|
|
|
if *self.checked {
|
|
|
|
region.add_paint_cmd(PaintCmd::Line {
|
|
|
|
points: vec![
|
2020-04-22 17:38:38 +00:00
|
|
|
pos2(small_icon_rect.left(), small_icon_rect.center().y),
|
|
|
|
pos2(small_icon_rect.center().x, small_icon_rect.bottom()),
|
|
|
|
pos2(small_icon_rect.right(), small_icon_rect.top()),
|
2019-03-11 14:59:49 +00:00
|
|
|
],
|
|
|
|
color: stroke_color,
|
|
|
|
width: region.style().line_width,
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2020-04-21 14:50:56 +00:00
|
|
|
let text_color = self.text_color.unwrap_or(stroke_color);
|
|
|
|
region.add_text(text_cursor, text_style, text, Some(text_color));
|
2019-01-10 09:55:38 +00:00
|
|
|
region.response(interact)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// ----------------------------------------------------------------------------
|
|
|
|
|
|
|
|
#[derive(Debug)]
|
|
|
|
pub struct RadioButton {
|
|
|
|
checked: bool,
|
|
|
|
text: String,
|
2019-01-16 15:28:43 +00:00
|
|
|
text_color: Option<Color>,
|
2019-01-10 09:55:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
impl RadioButton {
|
2020-04-24 16:47:14 +00:00
|
|
|
pub fn new(checked: bool, text: impl Into<String>) -> Self {
|
2019-01-10 09:55:38 +00:00
|
|
|
RadioButton {
|
|
|
|
checked,
|
|
|
|
text: text.into(),
|
2019-01-16 15:28:43 +00:00
|
|
|
text_color: None,
|
2019-01-10 09:55:38 +00:00
|
|
|
}
|
|
|
|
}
|
2019-01-16 15:28:43 +00:00
|
|
|
|
|
|
|
pub fn text_color(mut self, text_color: Color) -> Self {
|
|
|
|
self.text_color = Some(text_color);
|
|
|
|
self
|
|
|
|
}
|
2019-01-10 09:55:38 +00:00
|
|
|
}
|
|
|
|
|
2020-04-24 16:47:14 +00:00
|
|
|
pub fn radio(checked: bool, text: impl Into<String>) -> RadioButton {
|
2019-01-10 09:55:38 +00:00
|
|
|
RadioButton::new(checked, text)
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Widget for RadioButton {
|
|
|
|
fn add_to(self, region: &mut Region) -> GuiResponse {
|
2020-04-19 09:13:24 +00:00
|
|
|
let id = region.make_position_id();
|
2019-01-12 23:55:56 +00:00
|
|
|
let text_style = TextStyle::Button;
|
|
|
|
let font = ®ion.fonts()[text_style];
|
2020-04-19 22:48:54 +00:00
|
|
|
let (text, text_size) = font.layout_multiline(&self.text, region.available_width());
|
2019-01-14 13:26:02 +00:00
|
|
|
let interact = region.reserve_space(
|
2019-03-11 14:39:54 +00:00
|
|
|
region.style().button_padding
|
|
|
|
+ vec2(region.style().start_icon_width, 0.0)
|
2019-01-10 09:55:38 +00:00
|
|
|
+ text_size
|
2019-03-11 14:39:54 +00:00
|
|
|
+ region.style().button_padding,
|
2019-01-10 09:55:38 +00:00
|
|
|
Some(id),
|
|
|
|
);
|
2020-04-25 13:45:38 +00:00
|
|
|
let text_cursor = interact.rect.min
|
2019-03-11 14:39:54 +00:00
|
|
|
+ region.style().button_padding
|
|
|
|
+ vec2(region.style().start_icon_width, 0.0);
|
2019-03-11 14:59:49 +00:00
|
|
|
|
|
|
|
let fill_color = region.style().interact_fill_color(&interact);
|
|
|
|
let stroke_color = region.style().interact_stroke_color(&interact);
|
|
|
|
|
|
|
|
let (small_icon_rect, big_icon_rect) = region.style().icon_rectangles(&interact.rect);
|
|
|
|
|
|
|
|
region.add_paint_cmd(PaintCmd::Circle {
|
|
|
|
center: big_icon_rect.center(),
|
2020-04-19 21:34:34 +00:00
|
|
|
fill_color,
|
2019-03-11 14:59:49 +00:00
|
|
|
outline: None,
|
2020-04-22 22:17:37 +00:00
|
|
|
radius: big_icon_rect.width() / 2.0,
|
2019-01-10 09:55:38 +00:00
|
|
|
});
|
2019-03-11 14:59:49 +00:00
|
|
|
|
|
|
|
if self.checked {
|
|
|
|
region.add_paint_cmd(PaintCmd::Circle {
|
|
|
|
center: small_icon_rect.center(),
|
|
|
|
fill_color: Some(stroke_color),
|
|
|
|
outline: None,
|
2020-04-22 22:17:37 +00:00
|
|
|
radius: small_icon_rect.width() / 2.0,
|
2019-03-11 14:59:49 +00:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2020-04-21 14:50:56 +00:00
|
|
|
let text_color = self.text_color.unwrap_or(stroke_color);
|
|
|
|
region.add_text(text_cursor, text_style, text, Some(text_color));
|
2019-01-10 09:55:38 +00:00
|
|
|
region.response(interact)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// ----------------------------------------------------------------------------
|
|
|
|
|
2020-04-19 09:13:24 +00:00
|
|
|
/// Combined into one function (rather than two) to make it easier
|
|
|
|
/// for the borrow checker.
|
|
|
|
type SliderGetSet<'a> = Box<dyn 'a + FnMut(Option<f32>) -> f32>;
|
|
|
|
|
2019-01-10 09:55:38 +00:00
|
|
|
pub struct Slider<'a> {
|
2020-04-19 09:13:24 +00:00
|
|
|
get_set_value: SliderGetSet<'a>,
|
2020-04-25 09:01:57 +00:00
|
|
|
range: RangeInclusive<f32>,
|
2020-04-29 19:58:41 +00:00
|
|
|
// TODO: label: Option<Label>
|
2019-01-10 09:55:38 +00:00
|
|
|
text: Option<String>,
|
2019-02-10 15:10:08 +00:00
|
|
|
precision: usize,
|
2019-01-16 15:28:43 +00:00
|
|
|
text_color: Option<Color>,
|
2019-01-10 09:55:38 +00:00
|
|
|
text_on_top: Option<bool>,
|
2020-04-23 07:27:08 +00:00
|
|
|
id: Option<Id>,
|
2019-01-10 09:55:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
impl<'a> Slider<'a> {
|
2020-04-25 09:01:57 +00:00
|
|
|
fn from_get_set(
|
|
|
|
range: RangeInclusive<f32>,
|
|
|
|
get_set_value: impl 'a + FnMut(Option<f32>) -> f32,
|
|
|
|
) -> Self {
|
2020-04-19 09:13:24 +00:00
|
|
|
Slider {
|
|
|
|
get_set_value: Box::new(get_set_value),
|
2020-04-25 09:01:57 +00:00
|
|
|
range,
|
2020-04-19 09:13:24 +00:00
|
|
|
text: None,
|
|
|
|
precision: 3,
|
|
|
|
text_on_top: None,
|
|
|
|
text_color: None,
|
2020-04-23 07:27:08 +00:00
|
|
|
id: None,
|
2020-04-19 09:13:24 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-04-25 09:01:57 +00:00
|
|
|
pub fn f32(value: &'a mut f32, range: RangeInclusive<f32>) -> Self {
|
2019-01-10 09:55:38 +00:00
|
|
|
Slider {
|
2020-04-19 09:13:24 +00:00
|
|
|
precision: 3,
|
2020-04-25 09:01:57 +00:00
|
|
|
..Self::from_get_set(range, move |v: Option<f32>| {
|
2019-02-10 15:10:08 +00:00
|
|
|
if let Some(v) = v {
|
|
|
|
*value = v
|
|
|
|
}
|
|
|
|
*value
|
2020-04-19 09:13:24 +00:00
|
|
|
})
|
2019-02-10 15:10:08 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-04-25 09:01:57 +00:00
|
|
|
pub fn i32(value: &'a mut i32, range: RangeInclusive<i32>) -> Self {
|
|
|
|
let range = (*range.start() as f32)..=(*range.end() as f32);
|
2019-02-10 15:10:08 +00:00
|
|
|
Slider {
|
2020-04-19 09:13:24 +00:00
|
|
|
precision: 0,
|
2020-04-25 09:01:57 +00:00
|
|
|
..Self::from_get_set(range, move |v: Option<f32>| {
|
2019-02-10 15:10:08 +00:00
|
|
|
if let Some(v) = v {
|
|
|
|
*value = v.round() as i32
|
|
|
|
}
|
|
|
|
*value as f32
|
2020-04-19 09:13:24 +00:00
|
|
|
})
|
2019-01-10 09:55:38 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-04-25 09:01:57 +00:00
|
|
|
pub fn usize(value: &'a mut usize, range: RangeInclusive<usize>) -> Self {
|
|
|
|
let range = (*range.start() as f32)..=(*range.end() as f32);
|
2019-03-11 12:31:55 +00:00
|
|
|
Slider {
|
2020-04-19 09:13:24 +00:00
|
|
|
precision: 0,
|
2020-04-25 09:01:57 +00:00
|
|
|
..Self::from_get_set(range, move |v: Option<f32>| {
|
2019-03-11 12:31:55 +00:00
|
|
|
if let Some(v) = v {
|
|
|
|
*value = v.round() as usize
|
|
|
|
}
|
|
|
|
*value as f32
|
2020-04-19 09:13:24 +00:00
|
|
|
})
|
2019-03-11 12:31:55 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-04-24 16:47:14 +00:00
|
|
|
pub fn text(mut self, text: impl Into<String>) -> Self {
|
2019-01-10 09:55:38 +00:00
|
|
|
self.text = Some(text.into());
|
|
|
|
self
|
|
|
|
}
|
2019-01-16 15:28:43 +00:00
|
|
|
|
|
|
|
pub fn text_color(mut self, text_color: Color) -> Self {
|
|
|
|
self.text_color = Some(text_color);
|
|
|
|
self
|
|
|
|
}
|
2019-04-25 16:07:36 +00:00
|
|
|
|
|
|
|
pub fn precision(mut self, precision: usize) -> Self {
|
|
|
|
self.precision = precision;
|
|
|
|
self
|
|
|
|
}
|
2020-04-12 10:07:51 +00:00
|
|
|
|
|
|
|
fn get_value_f32(&mut self) -> f32 {
|
|
|
|
(self.get_set_value)(None)
|
|
|
|
}
|
|
|
|
|
|
|
|
fn set_value_f32(&mut self, mut value: f32) {
|
|
|
|
if self.precision == 0 {
|
|
|
|
value = value.round();
|
|
|
|
}
|
|
|
|
(self.get_set_value)(Some(value));
|
|
|
|
}
|
2019-01-10 09:55:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
impl<'a> Widget for Slider<'a> {
|
2019-02-10 15:10:08 +00:00
|
|
|
fn add_to(mut self, region: &mut Region) -> GuiResponse {
|
2019-01-12 23:55:56 +00:00
|
|
|
let text_style = TextStyle::Button;
|
|
|
|
let font = ®ion.fonts()[text_style];
|
|
|
|
|
2019-01-10 09:55:38 +00:00
|
|
|
if let Some(text) = &self.text {
|
2020-04-23 07:27:08 +00:00
|
|
|
if self.id.is_none() {
|
|
|
|
self.id = Some(Id::new(text));
|
|
|
|
}
|
|
|
|
|
2019-01-10 09:55:38 +00:00
|
|
|
let text_on_top = self.text_on_top.unwrap_or_default();
|
2019-01-16 15:28:43 +00:00
|
|
|
let text_color = self.text_color;
|
2020-04-12 10:07:51 +00:00
|
|
|
let value = (self.get_set_value)(None);
|
|
|
|
let full_text = format!("{}: {:.*}", text, self.precision, value);
|
2020-04-19 09:13:24 +00:00
|
|
|
|
2020-04-22 17:39:51 +00:00
|
|
|
let slider_sans_text = Slider { text: None, ..self };
|
2019-01-10 09:55:38 +00:00
|
|
|
|
|
|
|
if text_on_top {
|
2020-04-19 22:48:54 +00:00
|
|
|
let (text, text_size) = font.layout_multiline(&full_text, region.available_width());
|
2019-01-14 13:54:06 +00:00
|
|
|
let pos = region.reserve_space_without_padding(text_size);
|
2019-01-16 15:28:43 +00:00
|
|
|
region.add_text(pos, text_style, text, text_color);
|
2020-04-22 17:39:51 +00:00
|
|
|
slider_sans_text.add_to(region)
|
2019-01-10 09:55:38 +00:00
|
|
|
} else {
|
|
|
|
region.columns(2, |columns| {
|
2020-04-22 17:39:51 +00:00
|
|
|
// Slider on the left:
|
|
|
|
let slider_response = columns[0].add(slider_sans_text);
|
2019-01-14 13:54:06 +00:00
|
|
|
|
2020-04-22 17:39:51 +00:00
|
|
|
// Place the text in line with the slider on the left:
|
|
|
|
columns[1]
|
|
|
|
.desired_rect
|
2020-04-22 22:17:37 +00:00
|
|
|
.set_height(slider_response.rect.height());
|
2019-01-14 13:54:06 +00:00
|
|
|
columns[1].horizontal(Align::Center, |region| {
|
2019-01-21 07:48:32 +00:00
|
|
|
region.add(Label::new(full_text));
|
2019-01-14 13:54:06 +00:00
|
|
|
});
|
|
|
|
|
2020-04-22 17:39:51 +00:00
|
|
|
slider_response
|
2019-01-10 09:55:38 +00:00
|
|
|
})
|
|
|
|
}
|
|
|
|
} else {
|
2019-03-11 14:39:54 +00:00
|
|
|
let height = font.line_spacing().max(region.style().clickable_diameter);
|
2020-04-22 22:17:37 +00:00
|
|
|
let handle_radius = height / 2.5;
|
|
|
|
|
2020-04-23 07:27:08 +00:00
|
|
|
let id = self.id.unwrap_or_else(|| region.make_position_id());
|
2019-01-14 13:26:02 +00:00
|
|
|
|
|
|
|
let interact = region.reserve_space(
|
2019-01-10 09:55:38 +00:00
|
|
|
Vec2 {
|
2020-04-19 22:48:54 +00:00
|
|
|
x: region.available_width(),
|
2019-01-14 13:26:02 +00:00
|
|
|
y: height,
|
2019-01-10 09:55:38 +00:00
|
|
|
},
|
2020-04-19 09:13:24 +00:00
|
|
|
Some(id),
|
2019-01-10 09:55:38 +00:00
|
|
|
);
|
|
|
|
|
2020-04-22 17:39:51 +00:00
|
|
|
let left = interact.rect.left() + handle_radius;
|
|
|
|
let right = interact.rect.right() - handle_radius;
|
|
|
|
|
2020-04-25 09:11:44 +00:00
|
|
|
let range = self.range.clone();
|
|
|
|
debug_assert!(range.start() <= range.end());
|
2020-04-22 17:39:51 +00:00
|
|
|
|
2019-02-10 14:30:48 +00:00
|
|
|
if let Some(mouse_pos) = region.input().mouse_pos {
|
|
|
|
if interact.active {
|
2020-04-25 09:11:44 +00:00
|
|
|
self.set_value_f32(remap_clamp(mouse_pos.x, left..=right, range.clone()));
|
2019-02-10 14:30:48 +00:00
|
|
|
}
|
2019-01-10 09:55:38 +00:00
|
|
|
}
|
|
|
|
|
2019-03-11 14:59:49 +00:00
|
|
|
// Paint it:
|
|
|
|
{
|
2020-04-12 10:07:51 +00:00
|
|
|
let value = self.get_value_f32();
|
2019-03-11 14:59:49 +00:00
|
|
|
|
|
|
|
let rect = interact.rect;
|
2020-04-23 17:15:17 +00:00
|
|
|
let rail_radius = region.round_to_pixel((height / 8.0).max(2.0));
|
2020-04-22 17:39:51 +00:00
|
|
|
let rail_rect = Rect::from_min_max(
|
2020-04-22 22:17:37 +00:00
|
|
|
pos2(interact.rect.left(), rect.center().y - rail_radius),
|
|
|
|
pos2(interact.rect.right(), rect.center().y + rail_radius),
|
2020-04-22 17:39:51 +00:00
|
|
|
);
|
2020-04-25 09:11:44 +00:00
|
|
|
let marker_center_x = remap_clamp(value, range, left..=right);
|
2019-03-11 14:59:49 +00:00
|
|
|
|
|
|
|
region.add_paint_cmd(PaintCmd::Rect {
|
2020-04-22 22:17:37 +00:00
|
|
|
rect: rail_rect,
|
2020-04-22 17:39:51 +00:00
|
|
|
corner_radius: rail_radius,
|
2019-03-11 14:59:49 +00:00
|
|
|
fill_color: Some(region.style().background_fill_color()),
|
2020-04-19 09:11:41 +00:00
|
|
|
outline: Some(Outline::new(1.0, color::gray(200, 255))), // TODO
|
2019-03-11 14:59:49 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
region.add_paint_cmd(PaintCmd::Circle {
|
2020-04-22 17:39:51 +00:00
|
|
|
center: pos2(marker_center_x, rail_rect.center().y),
|
2020-04-22 22:17:37 +00:00
|
|
|
radius: handle_radius,
|
2020-04-19 21:34:34 +00:00
|
|
|
fill_color: region.style().interact_fill_color(&interact),
|
2020-04-19 09:11:41 +00:00
|
|
|
outline: Some(Outline::new(
|
2020-04-22 17:39:51 +00:00
|
|
|
region.style().interact_stroke_width(&interact),
|
2020-04-19 09:11:41 +00:00
|
|
|
region.style().interact_stroke_color(&interact),
|
|
|
|
)),
|
2019-03-11 14:59:49 +00:00
|
|
|
});
|
|
|
|
}
|
2019-01-10 09:55:38 +00:00
|
|
|
|
|
|
|
region.response(interact)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// ----------------------------------------------------------------------------
|
2019-01-13 18:15:11 +00:00
|
|
|
|
|
|
|
pub struct Separator {
|
|
|
|
line_width: f32,
|
2020-04-19 21:51:38 +00:00
|
|
|
min_length: f32,
|
|
|
|
extra: f32,
|
|
|
|
color: Color,
|
2019-01-13 18:15:11 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
impl Separator {
|
|
|
|
pub fn new() -> Separator {
|
|
|
|
Separator {
|
|
|
|
line_width: 2.0,
|
2020-04-19 21:51:38 +00:00
|
|
|
min_length: 6.0,
|
|
|
|
extra: 0.0,
|
|
|
|
color: color::WHITE,
|
2019-01-13 18:15:11 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-04-19 21:51:38 +00:00
|
|
|
pub fn line_width(mut self, line_width: f32) -> Self {
|
2019-01-13 18:15:11 +00:00
|
|
|
self.line_width = line_width;
|
|
|
|
self
|
|
|
|
}
|
|
|
|
|
2020-04-19 21:51:38 +00:00
|
|
|
pub fn min_length(mut self, min_length: f32) -> Self {
|
|
|
|
self.min_length = min_length;
|
|
|
|
self
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Draw this much longer on each side
|
|
|
|
pub fn extra(mut self, extra: f32) -> Self {
|
|
|
|
self.extra = extra;
|
|
|
|
self
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn color(mut self, color: Color) -> Self {
|
|
|
|
self.color = color;
|
2019-01-13 18:15:11 +00:00
|
|
|
self
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Widget for Separator {
|
|
|
|
fn add_to(self, region: &mut Region) -> GuiResponse {
|
2020-04-19 22:48:54 +00:00
|
|
|
let available_space = region.available_space();
|
2020-04-22 17:38:38 +00:00
|
|
|
let extra = self.extra;
|
2019-01-13 18:15:11 +00:00
|
|
|
let (points, interact) = match region.direction() {
|
|
|
|
Direction::Horizontal => {
|
2020-04-19 21:51:38 +00:00
|
|
|
let interact = region.reserve_space(vec2(self.min_length, available_space.y), None);
|
2019-01-13 18:15:11 +00:00
|
|
|
(
|
|
|
|
vec![
|
2020-04-22 17:38:38 +00:00
|
|
|
pos2(interact.rect.center().x, interact.rect.top() - extra),
|
|
|
|
pos2(interact.rect.center().x, interact.rect.bottom() + extra),
|
2019-01-13 18:15:11 +00:00
|
|
|
],
|
|
|
|
interact,
|
|
|
|
)
|
|
|
|
}
|
|
|
|
Direction::Vertical => {
|
2020-04-19 21:51:38 +00:00
|
|
|
let interact = region.reserve_space(vec2(available_space.x, self.min_length), None);
|
2019-01-13 18:15:11 +00:00
|
|
|
(
|
|
|
|
vec![
|
2020-04-22 17:38:38 +00:00
|
|
|
pos2(interact.rect.left() - extra, interact.rect.center().y),
|
|
|
|
pos2(interact.rect.right() + extra, interact.rect.center().y),
|
2019-01-13 18:15:11 +00:00
|
|
|
],
|
|
|
|
interact,
|
|
|
|
)
|
|
|
|
}
|
|
|
|
};
|
2019-03-11 14:59:49 +00:00
|
|
|
region.add_paint_cmd(PaintCmd::Line {
|
2019-01-13 18:15:11 +00:00
|
|
|
points,
|
2020-04-19 21:51:38 +00:00
|
|
|
color: self.color,
|
2019-01-13 18:15:11 +00:00
|
|
|
width: self.line_width,
|
2019-03-11 14:59:49 +00:00
|
|
|
});
|
2019-01-13 18:15:11 +00:00
|
|
|
region.response(interact)
|
|
|
|
}
|
|
|
|
}
|