1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
use static_assertions::assert_not_impl_all;
use std::{
cell::{Cell, RefCell},
collections::HashMap,
ffi::c_void,
ptr::addr_of_mut,
};
use crate::{bind, geo::Point, Result};
use super::Window;
#[derive(Debug, Clone, Copy, PartialEq, Eq)]
#[non_exhaustive]
pub enum HitTestResult {
Normal,
Draggable,
ResizeTopLeft,
ResizeTop,
ResizeTopRight,
ResizeRight,
ResizeBottomRight,
ResizeBottom,
ResizeBottomLeft,
ResizeLeft,
}
impl HitTestResult {
fn into_arg(self) -> bind::SDL_HitTestResult {
match self {
HitTestResult::Normal => bind::SDL_HITTEST_NORMAL,
HitTestResult::Draggable => bind::SDL_HITTEST_DRAGGABLE,
HitTestResult::ResizeTopLeft => bind::SDL_HITTEST_RESIZE_TOPLEFT,
HitTestResult::ResizeTop => bind::SDL_HITTEST_RESIZE_TOP,
HitTestResult::ResizeTopRight => bind::SDL_HITTEST_RESIZE_TOPRIGHT,
HitTestResult::ResizeRight => bind::SDL_HITTEST_RESIZE_RIGHT,
HitTestResult::ResizeBottomRight => bind::SDL_HITTEST_RESIZE_BOTTOMRIGHT,
HitTestResult::ResizeBottom => bind::SDL_HITTEST_RESIZE_BOTTOM,
HitTestResult::ResizeBottomLeft => bind::SDL_HITTEST_RESIZE_BOTTOMLEFT,
HitTestResult::ResizeLeft => bind::SDL_HITTEST_RESIZE_LEFT,
}
}
}
pub trait HitTester<'window>: FnMut(Point) -> HitTestResult + 'window {}
#[derive(Debug)]
pub struct HitTest<'window, T> {
window: &'window Window<'window>,
tester: T,
}
impl<'window, T: HitTester<'window>> HitTest<'window, T> {
pub fn new(window: &'window Window<'window>, mut tester: T) -> Result<Self> {
let data = addr_of_mut!(tester);
let ret = unsafe {
bind::SDL_SetWindowHitTest(
window.as_ptr(),
Some(hit_test_wrap_handler::<T>),
data.cast(),
)
};
if ret == 0 {
Ok(Self { window, tester })
} else {
Err(crate::SdlError::UnsupportedFeature)
}
}
}
impl<T> Drop for HitTest<'_, T> {
fn drop(&mut self) {
let _ =
unsafe { bind::SDL_SetWindowHitTest(self.window.as_ptr(), None, std::ptr::null_mut()) };
}
}
unsafe extern "C" fn hit_test_wrap_handler<'window, T: HitTester<'window>>(
win: *mut bind::SDL_Window,
area: *const bind::SDL_Point,
data: *mut c_void,
) -> bind::SDL_HitTestResult {
let callback = unsafe { &mut *data.cast::<T>() };
callback((*area).into()).into_arg()
}