403 lines
8.8 KiB
Rust
403 lines
8.8 KiB
Rust
use std::ops::Range;
|
|
|
|
use crate::{draw_scaled_texture, theme::*, MouseInput, Scroll, Textures};
|
|
use raylib::prelude::*;
|
|
|
|
#[derive(Debug)]
|
|
pub struct ShapedText {
|
|
text: String,
|
|
max_width: i32,
|
|
font_size: i32,
|
|
lines: Vec<Range<usize>>,
|
|
}
|
|
|
|
impl ShapedText {
|
|
pub fn new(font_size: i32) -> Self {
|
|
Self {
|
|
text: String::new(),
|
|
font_size,
|
|
max_width: 500,
|
|
lines: Vec::new(),
|
|
}
|
|
}
|
|
|
|
pub fn set_text(&mut self, text: &str) {
|
|
if text != self.text {
|
|
self.text = text.to_owned();
|
|
self.lines.clear();
|
|
}
|
|
}
|
|
|
|
pub fn height(&self) -> i32 {
|
|
self.font_size * self.lines.len() as i32
|
|
}
|
|
|
|
pub fn update_width(&mut self, d: &RaylibHandle, width: i32) {
|
|
if self.max_width == width && !self.lines.is_empty() {
|
|
return;
|
|
}
|
|
self.max_width = width;
|
|
self.lines.clear();
|
|
// todo remove leading space on broken lines
|
|
// todo fix splitting very long words
|
|
let mut line_start = 0;
|
|
let mut line_end = 0;
|
|
for (i, c) in self.text.char_indices() {
|
|
if c == ' ' {
|
|
let line = &self.text[line_start..i];
|
|
let new_line_length = d.measure_text(line, self.font_size);
|
|
if new_line_length <= self.max_width {
|
|
line_end = i;
|
|
} else {
|
|
self.lines.push(line_start..line_end);
|
|
line_start = line_end;
|
|
}
|
|
}
|
|
if c == '\n' {
|
|
let line = &self.text[line_start..i];
|
|
let new_line_length = d.measure_text(line, self.font_size);
|
|
if new_line_length <= self.max_width {
|
|
self.lines.push(line_start..i);
|
|
line_end = i + 1;
|
|
line_start = i + 1;
|
|
} else {
|
|
self.lines.push(line_start..line_end);
|
|
self.lines.push(line_end..(i + 1));
|
|
line_start = i + 1;
|
|
line_end = i + 1;
|
|
}
|
|
}
|
|
}
|
|
let i = self.text.len();
|
|
let line = &self.text[line_start..i];
|
|
let new_line_length = d.measure_text(line, self.font_size);
|
|
if new_line_length <= self.max_width {
|
|
self.lines.push(line_start..i);
|
|
} else {
|
|
self.lines.push(line_start..line_end);
|
|
self.lines.push(line_end..i);
|
|
}
|
|
}
|
|
|
|
pub fn draw(&self, d: &mut RaylibDrawHandle, x: i32, y: i32) {
|
|
// d.draw_rectangle(x, y, self.max_width, 4, Color::RED);
|
|
for (i, line) in self.lines.iter().enumerate() {
|
|
let line = &self.text[line.clone()];
|
|
let line_y = y + self.font_size * i as i32;
|
|
d.draw_text(line, x, line_y, self.font_size, Color::WHITE);
|
|
}
|
|
}
|
|
}
|
|
|
|
#[derive(Debug, Default)]
|
|
pub struct Tooltip {
|
|
text: Option<&'static str>,
|
|
mouse_x: i32,
|
|
mouse_y: i32,
|
|
screen_width: i32,
|
|
screen_height: i32,
|
|
}
|
|
|
|
impl Tooltip {
|
|
pub fn init_frame(&mut self, d: &RaylibHandle) {
|
|
let p = d.get_mouse_position();
|
|
*self = Self {
|
|
text: None,
|
|
mouse_x: p.x as i32,
|
|
mouse_y: p.y as i32,
|
|
screen_width: d.get_screen_width(),
|
|
screen_height: d.get_screen_height(),
|
|
};
|
|
}
|
|
|
|
pub fn add(&mut self, x: i32, y: i32, width: i32, height: i32, text: &'static str) {
|
|
if self.mouse_x >= x
|
|
&& self.mouse_y >= y
|
|
&& self.mouse_x <= (x + width)
|
|
&& self.mouse_y <= (y + height)
|
|
{
|
|
self.text = Some(text);
|
|
}
|
|
}
|
|
|
|
pub fn reset(&mut self) {
|
|
self.text = None;
|
|
}
|
|
|
|
pub fn add_rec(&mut self, bounds: Rectangle, text: &'static str) {
|
|
self.add(
|
|
bounds.x as i32,
|
|
bounds.y as i32,
|
|
bounds.width as i32,
|
|
bounds.height as i32,
|
|
text,
|
|
);
|
|
}
|
|
|
|
pub fn draw(&self, d: &mut RaylibDrawHandle) {
|
|
if let Some(text) = self.text {
|
|
let font_size = 20;
|
|
let margin = 4;
|
|
let text_width = d.measure_text(text, font_size);
|
|
let x = self
|
|
.mouse_x
|
|
.min(self.screen_width - text_width - margin * 2);
|
|
let y = self
|
|
.mouse_y
|
|
.min(self.screen_height - font_size - margin * 2);
|
|
d.draw_rectangle(
|
|
x,
|
|
y,
|
|
text_width + margin * 2,
|
|
font_size + margin * 2,
|
|
BG_LIGHT,
|
|
);
|
|
d.draw_text(text, x + margin, y + margin, font_size, Color::WHITE);
|
|
}
|
|
}
|
|
}
|
|
|
|
pub fn simple_button(
|
|
d: &mut RaylibDrawHandle,
|
|
mouse: &MouseInput,
|
|
x: i32,
|
|
y: i32,
|
|
width: i32,
|
|
height: i32,
|
|
) -> bool {
|
|
let mouse_pos = mouse.pos();
|
|
let bounds = Rectangle {
|
|
x: x as f32,
|
|
y: y as f32,
|
|
width: width as f32,
|
|
height: height as f32,
|
|
};
|
|
let hover = bounds.check_collision_point_rec(mouse_pos);
|
|
let pressed = hover && mouse.left_click();
|
|
d.draw_rectangle(x, y, width, height, widget_bg(hover));
|
|
pressed
|
|
}
|
|
|
|
pub fn text_button(
|
|
d: &mut RaylibDrawHandle,
|
|
mouse: &MouseInput,
|
|
x: i32,
|
|
y: i32,
|
|
width: i32,
|
|
text: &str,
|
|
) -> bool {
|
|
let font_size = 20;
|
|
let margin = font_size / 4;
|
|
let height = font_size + margin * 2;
|
|
let clicked = simple_button(d, mouse, x, y, width, height);
|
|
d.draw_text(text, x + margin, y + margin, font_size, Color::WHITE);
|
|
clicked
|
|
}
|
|
|
|
pub fn simple_option_button<T>(
|
|
d: &mut RaylibDrawHandle,
|
|
mouse: &MouseInput,
|
|
x: i32,
|
|
y: i32,
|
|
width: i32,
|
|
height: i32,
|
|
option: T,
|
|
current: &mut T,
|
|
) -> bool
|
|
where
|
|
T: PartialEq,
|
|
{
|
|
let bounds = Rectangle::new(x as f32, y as f32, width as f32, height as f32);
|
|
d.draw_rectangle_rec(bounds, widget_bg(&option == current));
|
|
let mut changed = false;
|
|
if mouse.left_click() && mouse.is_over(bounds) && current != &option {
|
|
*current = option;
|
|
changed = true;
|
|
}
|
|
changed
|
|
}
|
|
|
|
pub fn text_input(
|
|
d: &mut RaylibDrawHandle,
|
|
mouse: &MouseInput,
|
|
bounds: Rectangle,
|
|
text: &mut String,
|
|
is_selected: &mut bool,
|
|
max_len: usize,
|
|
editable: bool,
|
|
) -> bool {
|
|
let mut changed = false;
|
|
d.draw_rectangle_rec(bounds, widget_bg(*is_selected));
|
|
d.draw_rectangle_rec(
|
|
Rectangle::new(
|
|
bounds.x + 2.,
|
|
bounds.y + bounds.height - 5.,
|
|
bounds.width - 4.,
|
|
3.,
|
|
),
|
|
BG_DARK,
|
|
);
|
|
let drawn_text = if *is_selected {
|
|
&format!("{text}_")
|
|
} else {
|
|
text.as_str()
|
|
};
|
|
d.draw_text(
|
|
drawn_text,
|
|
bounds.x as i32 + 4,
|
|
bounds.y as i32 + 4,
|
|
20,
|
|
Color::WHITE,
|
|
);
|
|
if editable && mouse.left_click() && (mouse.is_over(bounds) || *is_selected) {
|
|
*is_selected = !*is_selected;
|
|
}
|
|
|
|
if *is_selected {
|
|
if d.is_key_pressed(KeyboardKey::KEY_ESCAPE) {
|
|
*is_selected = false;
|
|
}
|
|
if d.is_key_pressed(KeyboardKey::KEY_BACKSPACE) && !text.is_empty() {
|
|
changed = true;
|
|
text.pop();
|
|
}
|
|
if text.len() < max_len {
|
|
let char_code = unsafe { ffi::GetCharPressed() };
|
|
let c = if char_code > 0 {
|
|
char::from_u32(char_code as u32)
|
|
} else {
|
|
None
|
|
};
|
|
if let Some(c) = c {
|
|
changed = true;
|
|
text.push(c);
|
|
}
|
|
}
|
|
}
|
|
changed
|
|
}
|
|
|
|
pub fn scrollable_texture_option_button<T>(
|
|
d: &mut RaylibDrawHandle,
|
|
mouse: &MouseInput,
|
|
pos: Vector2,
|
|
texture: &Texture2D,
|
|
option: T,
|
|
current: &mut T,
|
|
tex_size: f32,
|
|
border: f32,
|
|
) -> Option<Scroll>
|
|
where
|
|
T: PartialEq,
|
|
{
|
|
let bounds = Rectangle {
|
|
x: pos.x,
|
|
y: pos.y,
|
|
width: tex_size + border * 2.,
|
|
height: tex_size + border * 2.,
|
|
};
|
|
d.draw_rectangle_rec(
|
|
bounds,
|
|
if &option == current {
|
|
BG_WIDGET_ACTIVE
|
|
} else {
|
|
gray(16)
|
|
},
|
|
);
|
|
d.draw_texture_ex(
|
|
texture,
|
|
pos + Vector2::new(border, border),
|
|
0.,
|
|
tex_size / texture.width as f32,
|
|
Color::WHITE,
|
|
);
|
|
if mouse.is_over(bounds) {
|
|
if mouse.left_click() {
|
|
*current = option;
|
|
}
|
|
return mouse.scroll();
|
|
}
|
|
None
|
|
}
|
|
|
|
pub fn draw_usize(
|
|
d: &mut RaylibDrawHandle,
|
|
textures: &Textures,
|
|
number: usize,
|
|
x: i32,
|
|
y: i32,
|
|
digits: u8,
|
|
scale: u8,
|
|
) {
|
|
let digits = digits as i32;
|
|
let scale = scale as i32;
|
|
for i in 0..digits {
|
|
d.draw_rectangle(x + 10 * i * scale, y, 8 * scale, 16 * scale, BG_LIGHT);
|
|
}
|
|
let mut num = number;
|
|
let mut i = 0;
|
|
while (num != 0 || i == 0) && i < digits {
|
|
let texture = textures.get(&format!("digit_{}", num % 10));
|
|
let x = x + (digits - i - 1) * 10 * scale;
|
|
draw_scaled_texture(d, texture, x, y, scale as f32);
|
|
num /= 10;
|
|
i += 1;
|
|
}
|
|
}
|
|
|
|
pub fn draw_usize_small(
|
|
d: &mut RaylibDrawHandle,
|
|
textures: &Textures,
|
|
mut num: usize,
|
|
mut x: i32,
|
|
y: i32,
|
|
scale: f32,
|
|
) {
|
|
const MAX_DIGITS: usize = 8;
|
|
let mut digits = [0; MAX_DIGITS];
|
|
let mut i = 0;
|
|
while (num != 0 || i == 0) && i < MAX_DIGITS {
|
|
digits[MAX_DIGITS - i - 1] = num % 10;
|
|
num /= 10;
|
|
i += 1;
|
|
}
|
|
let texture = textures.get("digits_small");
|
|
for &digit in &digits[(MAX_DIGITS - i)..] {
|
|
let source = Rectangle::new(4. * digit as f32, 0., 4., 6.);
|
|
let dest = Rectangle::new(x as f32, y as f32, 4. * scale, 6. * scale);
|
|
d.draw_texture_pro(texture, source, dest, Vector2::zero(), 0., FG_MARBLE_VALUE);
|
|
x += 4 * scale as i32;
|
|
}
|
|
}
|
|
|
|
pub fn slider(
|
|
d: &mut RaylibDrawHandle,
|
|
mouse: &MouseInput,
|
|
value: &mut u8,
|
|
min: u8,
|
|
max: u8,
|
|
x: i32,
|
|
y: i32,
|
|
width: i32,
|
|
height: i32,
|
|
) -> bool {
|
|
// the +1 makes the lowest state look slightly filled and the max state fully filled
|
|
let percent = (*value - min + 1) as f32 / (max - min + 1) as f32;
|
|
d.draw_rectangle(x, y, width, height, BG_WIDGET);
|
|
d.draw_rectangle(x, y, (width as f32 * percent) as i32, height, Color::CYAN);
|
|
let bounds = Rectangle::new(x as f32, y as f32, width as f32, height as f32);
|
|
if mouse.is_over(bounds) {
|
|
if mouse.left_hold() {
|
|
let percent = (mouse.pos().x - bounds.x) / bounds.width;
|
|
let new_value = min + (percent * (max - min + 1) as f32) as u8;
|
|
if *value != new_value {
|
|
*value = new_value;
|
|
}
|
|
} else if mouse.scroll() == Some(Scroll::Up) && *value < max {
|
|
*value += 1;
|
|
} else if mouse.scroll() == Some(Scroll::Down) && *value > min {
|
|
*value -= 1;
|
|
}
|
|
}
|
|
false
|
|
}
|