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
|
|
|
|
}
|
2023-05-11 22:39:25 +00:00
|
|
|
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]
|
2023-05-11 22:39:25 +00:00
|
|
|
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
|
|
|
};
|
2023-05-11 22:39:25 +00:00
|
|
|
let color_string = match color {
|
|
|
|
Some(c) => c.clone(),
|
2023-05-13 23:13:03 +00:00
|
|
|
None => "None".to_string(),
|
2023-05-11 22:39:25 +00:00
|
|
|
};
|
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",
|
2023-05-11 22:39:25 +00:00
|
|
|
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-21 01:38:52 +00:00
|
|
|
div {
|
|
|
|
class: "plate_cell_inner"
|
|
|
|
}
|
2023-05-11 21:49:03 +00:00
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
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
|
|
|
}
|
|
|
|
|
2023-05-11 22:39:25 +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 }
|
|
|
|
})
|
|
|
|
}
|
2023-05-11 22:39:25 +00:00
|
|
|
|
|
|
|
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
|
2023-05-13 23:13:03 +00:00
|
|
|
let c1 = (1, 1);
|
|
|
|
let c2 = (10, 10);
|
|
|
|
let pt = (5, 5);
|
2023-05-12 00:39:43 +00:00
|
|
|
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
|
2023-05-13 23:13:03 +00:00
|
|
|
let c1 = (1, 1);
|
|
|
|
let c2 = (10, 10);
|
|
|
|
let pt1 = (1, 5);
|
|
|
|
let pt2 = (10, 5);
|
2023-05-12 00:39:43 +00:00
|
|
|
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
|
2023-05-13 23:13:03 +00:00
|
|
|
let c1 = (1, 1);
|
|
|
|
let c2 = (10, 10);
|
|
|
|
let pt1 = (5, 1);
|
|
|
|
let pt2 = (5, 10);
|
2023-05-12 00:39:43 +00:00
|
|
|
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() {
|
2023-05-13 23:13:03 +00:00
|
|
|
// Test cases that should fail
|
|
|
|
let c1 = (1, 1);
|
|
|
|
let c2 = (10, 10);
|
|
|
|
let pt1 = (0, 0);
|
|
|
|
let pt2 = (15, 15);
|
2023-05-12 00:39:43 +00:00
|
|
|
assert_eq!(false, in_rect(Some(c1), Some(c2), pt1));
|
|
|
|
assert_eq!(false, in_rect(Some(c1), Some(c2), pt2));
|
|
|
|
}
|
|
|
|
}
|