mouse movement in raleigh, with colorpicker dragging example

This commit is contained in:
Benji Dial 2021-03-08 16:46:22 -05:00
parent af52ddac75
commit 920f1f0102
18 changed files with 117 additions and 7 deletions

View file

@ -61,6 +61,7 @@ table 1:
get timestamp | 0x19 | secs since 2000 | | | | |
file write | 0x1a | written | handle | file offset | count | buffer |
set file size | 0x1b | | handle | new file size | | |
want mouse moves | 0x1c | | window handle | | | |
table 2:

View file

@ -208,7 +208,8 @@ void const *syscall_table[] = {
&sc_is_task_running,
&sc_get_timestamp,
&sc_file_write,
&sc_file_set_size
&sc_file_set_size,
&window_wants_mouse_movements
};
//these aren't really void ()'s, but gcc complains if we take an address of a void, so we give it a type

View file

@ -26,7 +26,7 @@ extern exception_halt
extern pf_check_stack
extern dump
n_syscalls equ 0x1c
n_syscalls equ 0x1d
;section .bss
;_debug_is_start_task resb 1

View file

@ -38,6 +38,7 @@ static struct window {
struct window *below;
struct task_state *from_task;
bool wants_mouse_movements;
} windows[MAX_WINDOWS];
static struct window *bottom_window = 0;
@ -322,6 +323,7 @@ got_window:
send_action(top_window, (struct window_action){.action_type = FOCUS_ENTER});
w->from_task = active_task;
w->wants_mouse_movements = false;
paint_and_above(w);
return w;
@ -544,6 +546,16 @@ void move_mouse_by(int16_t y, int16_t x) {
blit(old_x, old_x + MOUSE_WIDTH > VBE_MODE_INFO->width ? VBE_MODE_INFO->width : old_x + MOUSE_WIDTH,
old_y, old_y + MOUSE_HEIGHT > VBE_MODE_INFO->height ? VBE_MODE_INFO->height : old_y + MOUSE_HEIGHT);
blit_mouse();
struct window *const moving_over = buffer[mouse_y * VBE_MODE_INFO->width + mouse_x].from_window;
if (moving_over && moving_over->wants_mouse_movements &&
(mouse_y >= moving_over->ypos) && (mouse_y < moving_over->ypos + moving_over->height) &&
(mouse_x >= moving_over->xpos) && (mouse_x < moving_over->xpos + moving_over->width))
send_action(moving_over, (struct window_action){
.action_type = MOUSE_MOVE, .moved_to = {
.x = mouse_x - moving_over->xpos,
.y = mouse_y - moving_over->ypos
}
});
}
switch_to_task_cr3();
//logf(LOG_INFO, "new mouse coords: (%d, %d)", mouse_x, mouse_y);
@ -628,4 +640,8 @@ void mouse_button(enum mouse_button which, bool up) {
}
send_action(clicked_on, (struct window_action){.action_type = up ? MOUSE_UP : MOUSE_DOWN, .as_mouse = {.y = mouse_y - clicked_on->ypos, .x = mouse_x - clicked_on->xpos, .which = which}});
}
}
void window_wants_mouse_movements(struct window *w) {
w->wants_mouse_movements = true;
}

View file

@ -36,4 +36,6 @@ void show_shutdown();
void move_mouse_by(int16_t y, int16_t x);
void mouse_button(enum mouse_button which, bool up);
void window_wants_mouse_movements(struct window *w);
#endif

View file

@ -7,7 +7,7 @@ struct mouse_packet {
uint16_t y;
uint16_t x;
enum mouse_button {LEFT, RIGHT, MIDDLE} which;
};
} __attribute__ ((__packed__));
struct window_action {
enum {
@ -17,11 +17,16 @@ struct window_action {
FOCUS_ENTER,
FOCUS_LEAVE,
MOUSE_DOWN,
MOUSE_UP
MOUSE_UP,
MOUSE_MOVE
} action_type;
union {
struct key_packet as_key;
struct mouse_packet as_mouse;
struct {
uint32_t y;
uint32_t x;
} moved_to;
};
} __attribute__ ((__packed__));

View file

@ -14,6 +14,7 @@ namespace raleigh {
void handle_click(coord window_coords, enum mouse_packet::mouse_button click_type, bool up) override;
void notify_child_size_change(widget &child, coord old_size) override;
void notify_has_opaque_parent(widget *parent) override;
void on_mouse_move(coord window_coords) override;
private:
widget &inner;
void (*on_click)(button &);

View file

@ -12,11 +12,14 @@ namespace raleigh {
void paint(_pixel_t *pixbuf, uint32_t pitch) override;
void handle_click(coord window_coords, enum mouse_packet::mouse_button click_type, bool up) override;
void notify_has_opaque_parent(widget *parent) override;
void on_mouse_move(coord window_coords) override;
private:
_pixel_t picked_color;
uint8_t resolution;
uint8_t inv_res;
enum {NO, R, G, B} selected;
};
}

View file

@ -13,6 +13,7 @@ namespace raleigh {
void handle_click(coord window_coords, enum mouse_packet::mouse_button click_type, bool up) override;
void notify_has_opaque_parent(widget *parent) override;
void notify_child_size_change(widget &child, coord old_size) override;
void on_mouse_move(coord window_coords) override;
private:
widget &inner;
uint32_t pad_by;

View file

@ -15,6 +15,7 @@ namespace raleigh {
void handle_click(coord window_coords, enum mouse_packet::mouse_button click_type, bool up) override;
void notify_has_opaque_parent(widget *parent) override;
void notify_child_size_change(widget &from, coord old_size) override;
void on_mouse_move(coord window_coords) override;
private:
dllist<widget &> widgets;
};

View file

@ -36,6 +36,7 @@ namespace raleigh {
virtual void handle_key(struct key_packet kp);
virtual void on_focus();
virtual void on_unfocus();
virtual void on_mouse_move(coord window_coords);
//this next one is not to be called by child widgets
//they should call window::notify_widget_size_change(widget &), which will call this if necessary
virtual void notify_child_size_change(widget &child, coord old_size);

View file

@ -57,7 +57,8 @@ enum _scn {
_SCN_IS_TASK_RUNNING,
_SCN_GET_TIMESTAMP,
_SCN_FILE_WRITE,
_SCN_SET_FILE_SIZE
_SCN_SET_FILE_SIZE,
_SCN_WANTS_MOUSE_MOVES
};
static inline uint32_t _sc0(enum _scn eax) {
@ -254,6 +255,10 @@ static inline void _set_file_size(_file_handle_t handle, uint32_t new_size) {
_sc2(_SCN_SET_FILE_SIZE, handle, new_size);
}
static inline void _wants_mouse_moves(_window_handle_t window) {
_sc1(_SCN_WANTS_MOUSE_MOVES, (uint32_t)window);
}
#ifdef __cplusplus
}
#endif

View file

@ -55,4 +55,12 @@ namespace raleigh {
}
void button::notify_has_opaque_parent(widget *parent) {}
void button::on_mouse_move(coord window_coords) {
if ((window_coords.x >= inner.window_offset.x) &&
(window_coords.y >= inner.window_offset.y) &&
(window_coords.x < inner.window_offset.x + inner.size.x) &&
(window_coords.y < inner.window_offset.y + inner.size.y))
inner.on_mouse_move(window_coords);
}
}

View file

@ -2,7 +2,8 @@
namespace raleigh {
colorpicker::colorpicker(_pixel_t default_color, uint8_t resolution)
: picked_color(default_color), resolution(resolution), inv_res(256 / resolution) {
: picked_color(default_color), resolution(resolution),
inv_res(256 / resolution), selected(NO) {
size = coord(inv_res * 2, inv_res * 2);
closest_opaque = this;
}
@ -32,19 +33,26 @@ namespace raleigh {
}
void colorpicker::handle_click(coord window_coords, enum mouse_packet::mouse_button click_type, bool up) {
if (up || (click_type != mouse_packet::LEFT))
if (click_type != mouse_packet::LEFT)
return;
if (up) {
selected = NO;
return;
}
window_coords.x -= window_offset.x;
window_coords.y -= window_offset.y;
if ((window_coords.x < inv_res) && (window_coords.y < inv_res)) {
selected = R;
picked_color.g = window_coords.x * resolution;
picked_color.b = window_coords.y * resolution;
}
else if (window_coords.y < inv_res) {
selected = G;
picked_color.b = (window_coords.x - inv_res) * resolution;
picked_color.r = window_coords.y * resolution;
}
else if (window_coords.x < inv_res) {
selected = B;
picked_color.r = window_coords.x * resolution;
picked_color.g = (window_coords.y - inv_res) * resolution;
}
@ -52,4 +60,38 @@ namespace raleigh {
}
void colorpicker::notify_has_opaque_parent(widget *parent) {}
void colorpicker::on_mouse_move(coord window_coords) {
uint32_t x = window_coords.x - window_offset.x;
uint32_t y = window_coords.y - window_offset.y;
switch (selected) {
case NO:
return;
case R:
if (x >= inv_res)
x = inv_res - 1;
if (y >= inv_res)
y = inv_res - 1;
picked_color.g = x * resolution;
picked_color.b = y * resolution;
break;
case G:
if (x < inv_res)
x = inv_res;
if (y >= inv_res)
y = inv_res - 1;
picked_color.b = x * resolution;
picked_color.r = y * resolution;
break;
case B:
if (x >= inv_res)
x = inv_res - 1;
if (y < inv_res)
y = inv_res;
picked_color.r = x * resolution;
picked_color.g = y * resolution;
break;
}
w->notify_needs_paint(*this);
}
}

View file

@ -34,4 +34,12 @@ namespace raleigh {
void padding::notify_child_size_change(widget &child, coord old_size) {
set_size(coord(inner.size.x + pad_by * 2, inner.size.y + pad_by * 2));
}
void padding::on_mouse_move(coord window_coords) {
if ((window_coords.x >= inner.window_offset.x) &&
(window_coords.y >= inner.window_offset.y) &&
(window_coords.x < inner.window_offset.x + inner.size.x) &&
(window_coords.y < inner.window_offset.y + inner.size.y))
inner.on_mouse_move(window_coords);
}
}

View file

@ -63,4 +63,15 @@ namespace raleigh {
notify_window_change();
}
}
void vbox::on_mouse_move(coord window_coords) {
for (dllist<widget &>::node *n = widgets.first; n; n = n->next)
if ((window_coords.x >= n->d.window_offset.x) &&
(window_coords.y >= n->d.window_offset.y) &&
(window_coords.x < n->d.window_offset.x + n->d.size.x) &&
(window_coords.y < n->d.window_offset.y + n->d.size.y)) {
n->d.on_mouse_move(window_coords);
return;
}
}
}

View file

@ -11,6 +11,7 @@ namespace raleigh {
void widget::handle_key(struct key_packet kp) {}
void widget::on_focus() {}
void widget::on_unfocus() {}
void widget::on_mouse_move(coord window_coords) {}
void widget::notify_child_size_change(widget &child, coord old_size) {}
void widget::set_size(coord new_size) {

View file

@ -57,6 +57,8 @@ namespace raleigh {
focussed->on_focus();
else if (wa.action_type == wa.FOCUS_LEAVE)
focussed->on_unfocus();
else if (wa.action_type == wa.MOUSE_MOVE)
root.on_mouse_move(coord(wa.moved_to.x, wa.moved_to.y));
}
}
@ -78,6 +80,7 @@ namespace raleigh {
if (handle)
return;
handle = _new_window(size.x, size.y, pixbuf);
_wants_mouse_moves(handle);
if (!handle)
show_error_and_quitf("Failed to get window handle for requested window.");
open_windows.add_front(*this);