plate-tool/src/components/source_plate.rs

176 lines
4.8 KiB
Rust
Raw Normal View History

2023-05-11 21:49:03 +00:00
#![allow(non_snake_case)]
use dioxus::prelude::*;
static STYLE: &'static str = include_str!("plate.css");
#[derive(PartialEq, Props)]
pub struct SourcePlateProps {
width: u8,
height: u8,
}
struct SelectionState {
2023-05-11 21:51:09 +00:00
m_start: Option<(u8, u8)>,
m_end: Option<(u8, u8)>,
m_stat: bool,
2023-05-11 21:49:03 +00:00
}
pub fn SourcePlate(cx: Scope<SourcePlateProps>) -> Element {
use_shared_state_provider(cx, || SelectionState {
m_start: None,
m_end: None,
2023-05-11 21:51:09 +00:00
m_stat: false,
2023-05-11 21:49:03 +00:00
});
2023-05-11 21:51:09 +00:00
cx.render(rsx! {
2023-05-11 21:49:03 +00:00
style {
vec![STYLE].into_iter().map(|s| rsx!{s}) // This is stupid
}
SourcePlateSelectionIndicator {}
SourcePlateSelectionController {}
2023-05-11 21:49:03 +00:00
table {
draggable: "false",
for i in 1..=cx.props.height {
tr {
draggable: "false",
for j in 1..=cx.props.width {
SourcePlateCell {i: i, j: j}
}
}
},
}
})
}
#[inline_props]
fn SourcePlateCell(cx: Scope<PlateCellProps>, i: u8, j: u8, color: Option<String>) -> Element {
2023-05-11 21:49:03 +00:00
let selection_state = use_shared_state::<SelectionState>(cx).unwrap();
2023-05-12 00:39:43 +00:00
let selected = in_rect(
2023-05-11 21:51:09 +00:00
selection_state.read().m_start,
selection_state.read().m_end,
(*i, *j),
);
2023-05-11 21:49:03 +00:00
let selected_class = match selected {
true => "current_select",
2023-05-11 21:51:09 +00:00
false => "",
2023-05-11 21:49:03 +00:00
};
let color_string = match color {
Some(c) => c.clone(),
None => "None".to_string()
};
2023-05-11 21:49:03 +00:00
2023-05-11 21:51:09 +00:00
cx.render(rsx! {
2023-05-11 21:49:03 +00:00
td {
class: "plate_cell {selected_class}",
draggable: "false",
style: "background: {color_string}",
2023-05-11 21:49:03 +00:00
onmousedown: move |_| {
selection_state.write().m_start = Some((*i,*j));
selection_state.write().m_end = None;
selection_state.write().m_stat = true;
},
onmouseover: move |_| {
if selection_state.read().m_stat {
selection_state.write().m_end = Some((*i,*j))
}
},
onmouseup: move |_| {
selection_state.write().m_stat = false
},
}
})
}
2023-05-12 00:39:43 +00:00
fn in_rect(corner1: Option<(u8, u8)>, corner2: Option<(u8, u8)>, pt: (u8, u8)) -> bool {
2023-05-11 21:49:03 +00:00
if let (Some(c1), Some(c2)) = (corner1, corner2) {
2023-05-11 21:51:09 +00:00
return pt.0 <= u8::max(c1.0, c2.0)
&& pt.0 >= u8::min(c1.0, c2.0)
&& pt.1 <= u8::max(c1.1, c2.1)
&& pt.1 >= u8::min(c1.1, c2.1);
} else {
return false;
}
2023-05-11 21:49:03 +00:00
}
// This is a dummy component for design purposes only
fn SourcePlateSelectionIndicator(cx: Scope) -> Element {
2023-05-11 21:49:03 +00:00
let selection_state = use_shared_state::<SelectionState>(cx).unwrap();
let start_str = match selection_state.read().m_start {
Some(start) => format!("{},{}", start.0, start.1),
2023-05-11 21:51:09 +00:00
None => "None".to_string(),
2023-05-11 21:49:03 +00:00
};
2023-05-11 21:51:09 +00:00
let end_str = match selection_state.read().m_end {
2023-05-11 21:49:03 +00:00
Some(end) => format!("{},{}", end.0, end.1),
2023-05-11 21:51:09 +00:00
None => "None".to_string(),
2023-05-11 21:49:03 +00:00
};
2023-05-11 21:51:09 +00:00
cx.render(rsx! {
2023-05-11 21:49:03 +00:00
p { start_str ", and " end_str }
})
}
fn SourcePlateSelectionController(cx: Scope) -> Element {
cx.render(rsx! {
div {
button {
"Select"
}
}
})
}
2023-05-12 00:39:43 +00:00
#[cfg(test)]
mod tests {
use super::in_rect;
// in_rect tests
#[test]
fn test_in_rect1() {
// Test in center of rect
let c1 = (1,1);
let c2 = (10,10);
let pt = (5,5);
assert!(in_rect(Some(c1), Some(c2), pt));
// Order of the corners should not matter:
assert!(in_rect(Some(c2), Some(c1), pt));
}
#[test]
fn test_in_rect2() {
// Test on top/bottom edges of rect
let c1 = (1,1);
let c2 = (10,10);
let pt1 = (1,5);
let pt2 = (10,5);
assert!(in_rect(Some(c1), Some(c2), pt1));
assert!(in_rect(Some(c1), Some(c2), pt2));
// Order of the corners should not matter:
assert!(in_rect(Some(c2), Some(c1), pt1));
assert!(in_rect(Some(c2), Some(c1), pt2));
}
#[test]
fn test_in_rect3() {
// Test on left/right edges of rect
let c1 = (1,1);
let c2 = (10,10);
let pt1 = (5,1);
let pt2 = (5,10);
assert!(in_rect(Some(c1), Some(c2), pt1));
assert!(in_rect(Some(c1), Some(c2), pt2));
// Order of the corners should not matter:
assert!(in_rect(Some(c2), Some(c1), pt1));
assert!(in_rect(Some(c2), Some(c1), pt2));
}
#[test]
fn test_in_rect4() {
// Test cases that should fail
let c1 = (1,1);
let c2 = (10,10);
let pt1 = (0,0);
let pt2 = (15,15);
assert_eq!(false, in_rect(Some(c1), Some(c2), pt1));
assert_eq!(false, in_rect(Some(c1), Some(c2), pt2));
}
}