resizable widgets, default widget implementation for some functions, reimplementing meminfo in raleigh

This commit is contained in:
Benji Dial 2021-03-08 14:46:19 -05:00
parent 1a5ece4f52
commit 8221fd5451
19 changed files with 215 additions and 88 deletions

View file

@ -103,9 +103,10 @@ obj/popups.so: obj/popups/info.o obj/popups/popup.o
ld ${partlink} $^ -o $@ ld ${partlink} $^ -o $@
obj/raleigh.so: obj/raleigh/runtime.po obj/raleigh/window.po \ obj/raleigh.so: obj/raleigh/runtime.po obj/raleigh/window.po \
obj/raleigh/w/label.po obj/raleigh/util.po \ obj/raleigh/widget.po obj/raleigh/util.po \
obj/raleigh/w/padding.po obj/raleigh/w/button.po \ obj/raleigh/w/padding.po obj/raleigh/w/button.po \
obj/raleigh/w/vbox.po obj/raleigh/w/entry.po obj/raleigh/w/vbox.po obj/raleigh/w/entry.po \
obj/raleigh/w/label.po
ld ${partlink} $^ -o $@ ld ${partlink} $^ -o $@
obj/init.elf: obj/init/init.o obj/knob.so obj/c.rto obj/init.elf: obj/init/init.o obj/knob.so obj/c.rto
@ -116,9 +117,9 @@ obj/highway.elf: obj/highway/main.o obj/highway/cmds.o obj/highway/line.o \
obj/c.rto obj/c.rto
ld ${clink} $^ -o $@ ld ${clink} $^ -o $@
obj/meminfo.elf: obj/meminfo/meminfo.o obj/popups.so obj/libfont.so \ obj/meminfo.elf: obj/meminfo/meminfo.po obj/raleigh.so obj/popups.so \
obj/knob.so obj/c.rto obj/libfont.so obj/knob.so obj/cxx.rto
ld ${clink} $^ -o $@ ld ${cxxlink} $^ -o $@
obj/terminal.elf: obj/terminal/main.o obj/libfont.so obj/knob.so \ obj/terminal.elf: obj/terminal/main.o obj/libfont.so obj/knob.so \
obj/c.rto obj/c.rto

View file

@ -12,10 +12,8 @@ namespace raleigh {
void notify_window_change() override; void notify_window_change() override;
void paint(_pixel_t *pixbuf, uint32_t pitch) override; 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 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 notify_has_opaque_parent(widget *parent) override;
void handle_key(struct key_packet kp) override;
void on_focus() override;
void on_unfocus() override;
private: private:
widget &inner; widget &inner;
void (*on_click)(button &); void (*on_click)(button &);

View file

@ -7,19 +7,17 @@
namespace raleigh { namespace raleigh {
class label : public widget { class label : public widget {
public: public:
//this pointer is used directly, and the contents of the string should not be changed afterward //value's data is copied
label(const char *value, const char *font="fixed-10", bool bg_transparent=true, label(const char *value, const char *font="fixed-10", bool bg_transparent=true,
_pixel_t fg=RGB(00, 00, 00), _pixel_t bg=RGB(bf, bf, bf)); _pixel_t fg=RGB(00, 00, 00), _pixel_t bg=RGB(bf, bf, bf));
void notify_window_change() override; void change_value(const char *new_value);
void paint(_pixel_t *pixbuf, uint32_t pitch) override; 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 notify_has_opaque_parent(widget *parent) override;
void handle_key(struct key_packet kp) override;
void on_focus() override;
void on_unfocus() override;
private: private:
const char *const value; char *value;
uint32_t v_size;
const struct font_info *const fi; const struct font_info *const fi;
bool bg_transparent; bool bg_transparent;
const _pixel_t fg; const _pixel_t fg;

View file

@ -12,9 +12,7 @@ namespace raleigh {
void paint(_pixel_t *pixbuf, uint32_t pitch) override; 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 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_has_opaque_parent(widget *parent) override;
void handle_key(struct key_packet kp) override; void notify_child_size_change(widget &child, coord old_size) override;
void on_focus() override;
void on_unfocus() override;
private: private:
widget &inner; widget &inner;
uint32_t pad_by; uint32_t pad_by;

View file

@ -14,9 +14,7 @@ namespace raleigh {
void paint(_pixel_t *pixbuf, uint32_t pitch) override; 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 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_has_opaque_parent(widget *parent) override;
void handle_key(struct key_packet kp) override; void notify_child_size_change(widget &from, coord old_size) override;
void on_focus() override;
void on_unfocus() override;
private: private:
dllist<widget &> widgets; dllist<widget &> widgets;
}; };

View file

@ -12,12 +12,15 @@ namespace raleigh {
namespace raleigh { namespace raleigh {
class widget { class widget {
public: public:
coord size; //these three are set by window class (or parent widget)
widget *parent;//set to zero when root widget
//set by window class (or parent widget)
window *w; window *w;
coord window_offset; coord window_offset;
//derived classes should not set this outside of the initializer
//instead, they should call widget::set_size(coord)
coord size;
//fewest steps up that a widget can be redrawn without needing its parents //fewest steps up that a widget can be redrawn without needing its parents
//if a widget is opaque, it will set this to a pointer to itself, and then call //if a widget is opaque, it will set this to a pointer to itself, and then call
// notify_has_opaque_parent on any children, passing itself as an argument. // notify_has_opaque_parent on any children, passing itself as an argument.
@ -25,14 +28,21 @@ namespace raleigh {
// and then call notify_has_opaque_parent on any children (with the opaque parent). // and then call notify_has_opaque_parent on any children (with the opaque parent).
widget *closest_opaque; widget *closest_opaque;
//called by window class (or parent widget) //these are called by window class (or parent widgets)
virtual void notify_window_change() = 0; virtual void notify_window_change();
virtual void paint(_pixel_t *pixbuf, uint32_t pitch) = 0; virtual void paint(_pixel_t *pixbuf, uint32_t pitch) = 0;
virtual void handle_click(coord window_coords, enum mouse_packet::mouse_button click_type, bool up) = 0; virtual void handle_click(coord window_coords, enum mouse_packet::mouse_button click_type, bool up);
virtual void notify_has_opaque_parent(widget *parent) = 0; virtual void notify_has_opaque_parent(widget *parent) = 0;
virtual void handle_key(struct key_packet kp) = 0; virtual void handle_key(struct key_packet kp);
virtual void on_focus() = 0; virtual void on_focus();
virtual void on_unfocus() = 0; virtual void on_unfocus();
//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);
protected:
widget();
void set_size(coord to);
}; };
} }

View file

@ -7,6 +7,8 @@ namespace raleigh {
#include <raleigh/runtime.h> #include <raleigh/runtime.h>
#include <raleigh/widget.h> #include <raleigh/widget.h>
#include <structs/dllist.h>
#include <structs/duple.h>
#include <pland/syscall.h> #include <pland/syscall.h>
#include <raleigh/util.h> #include <raleigh/util.h>
@ -16,7 +18,10 @@ namespace raleigh {
public: public:
//pass on_close to specify a close handler. if on_close returns false, the window will not be closed. //pass on_close to specify a close handler. if on_close returns false, the window will not be closed.
window(widget &root, _pixel_t bg_color=RGB(bf, bf, bf), bool (*on_close)(window &)=0); window(widget &root, _pixel_t bg_color=RGB(bf, bf, bf), bool (*on_close)(window &)=0);
void add_keybind(struct key_packet kp, void (*handler)(window &));
void notify_needs_paint(widget &from); void notify_needs_paint(widget &from);
void notify_widget_size_change(widget &from, coord old_size);
enum try_actions_return_t {NONE, GOOD, DELETE}; enum try_actions_return_t {NONE, GOOD, DELETE};
try_actions_return_t try_actions(); try_actions_return_t try_actions();
void show(); void show();
@ -31,6 +36,7 @@ namespace raleigh {
bool needs_repaint; bool needs_repaint;
void paint_full(); void paint_full();
bool (*on_close)(window &); bool (*on_close)(window &);
dllist<duple<struct key_packet, void (*)(window &)>> keybinds;
}; };
} }

View file

@ -0,0 +1,13 @@
#ifndef STRUCTS_DUPLE_H
#define STRUCTS_DUPLE_H
template<class at, class bt>
class duple {
public:
duple(at a, bt b)
: a(a), b(b) {}
at a;
bt b;
};
#endif

View file

@ -7,7 +7,8 @@ extern "C" {
#include <keypack.h> #include <keypack.h>
char key_to_char(struct key_packet kp) __attribute__ ((pure)); char key_to_char(struct key_packet kp) __attribute__ ((const));
bool match_side_agnostic(struct key_packet a, struct key_packet b) __attribute__ ((const));
#ifdef __cplusplus #ifdef __cplusplus
} }

View file

@ -1,4 +1,5 @@
#include <keypack.h> #include <keypack.h>
#include <stdbool.h>
static const char no_mod[] = { static const char no_mod[] = {
0, 0, 0, 0, 0, 0, 0, 0, 0, '\t','\n', 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, '\t','\n', 0, 0, 0, 0, 0,
@ -48,7 +49,7 @@ static const char num[] = {
'0', '1', '2', '3', '4', '5', '6', '7', '8', '9', '*', '+', '\n', '-', '.', '/' '0', '1', '2', '3', '4', '5', '6', '7', '8', '9', '*', '+', '\n', '-', '.', '/'
}; };
__attribute__ ((pure)) __attribute__ ((const))
char key_to_char(struct key_packet kp) { char key_to_char(struct key_packet kp) {
if (kp.key_id < 0x80) { if (kp.key_id < 0x80) {
const char ch = (kp.modifiers & SHIFTS const char ch = (kp.modifiers & SHIFTS
@ -65,3 +66,12 @@ char key_to_char(struct key_packet kp) {
else else
return 0; return 0;
} }
__attribute__ ((const))
bool match_side_agnostic(struct key_packet a, struct key_packet b) {
return (a.key_id == b.key_id) &&
((bool)(a.modifiers & SHIFTS) == (bool)(b.modifiers & SHIFTS)) &&
((bool)(a.modifiers & CTRLS) == (bool)(b.modifiers & CTRLS)) &&
((bool)(a.modifiers & ALTS) == (bool)(b.modifiers & ALTS)) &&
((bool)(a.modifiers & WINS) == (bool)(b.modifiers & WINS));
}

View file

@ -1,31 +0,0 @@
#include <popups/info.h>
#include <pland/syscall.h>
static const struct key_packet meminfo_quits[] = {
{ .key_id = KEY_ESCAPE, .modifiers = NO_MODS },
{ .key_id = KEY_F5, .modifiers = NO_MODS },
{ .key_id = 0 }
};
void main() {
struct popup p;
redo:
info_popupf(&p,
"kernel memory free: %uk\n"
"userspace memory free: %uk / %uk\n"
"Escape to quit, F5 to refresh.",
(_pixel_t){.r = 0, .g = 0, .b = 0},
(_pixel_t){.r = 0xbf, .g = 0xbf, .b = 0xbf},
_kernel_dynamic_area_left() * 4,
_total_userspace_left() * 4,
_total_userspace_size() * 4
);
//hacky, i should really make info_popup take an arg
p.quit_binds = meminfo_quits;
make_modal(&p);
if (p.quit_as.key_id == KEY_F5)
//i should make popups have changable text
//(make a new pixbuf but reuse the window)
goto redo;
}

View file

@ -0,0 +1,51 @@
#include <raleigh/w/padding.h>
#include <raleigh/w/vbox.h>
#include <raleigh/w/label.h>
#include <raleigh/runtime.h>
#include <raleigh/window.h>
#include <pland/syscall.h>
#include <knob/format.h>
#include <pland/pcrt.h>
using namespace raleigh;
label *kmem;
label *umem;
void refresh(window &w) {
char *const kstr = format("kernel memory free: %uk", _kernel_dynamic_area_left() * 4);
char *const ustr = format("userspace memory free: %uk / %uk", _total_userspace_left() * 4, _total_userspace_size() * 4);
kmem->change_value(kstr);
umem->change_value(ustr);
free_block(kstr);
free_block(ustr);
}
void main() {
kmem = new label("");
umem = new label("");
label msg("press Alt+F4 to quit, or F5 to refresh");
padding pkmem(*kmem, 1);
padding pumem(*umem, 1);
padding pmsg(msg, 1);
dllist<widget &> ll;
ll.add_front(pmsg);
ll.add_front(pumem);
ll.add_front(pkmem);
vbox box(ll);
padding pbox(box, 3);
window w(pbox, RGB(bf, bf, bf), (bool (*)(window &))&__pcrt_quit);
w.add_keybind((struct key_packet){.key_id = key_packet::KEY_F5, .modifiers = key_packet::NO_MODS}, &refresh);
refresh(w);
w.show();
start_runtime();
}

View file

@ -8,6 +8,7 @@ namespace raleigh {
size = coord(inner.size.x + 2, inner.size.y + 2); size = coord(inner.size.x + 2, inner.size.y + 2);
closest_opaque = this; closest_opaque = this;
inner.notify_has_opaque_parent(this); inner.notify_has_opaque_parent(this);
inner.parent = this;
} }
void button::notify_window_change() { void button::notify_window_change() {
@ -49,11 +50,9 @@ namespace raleigh {
} }
} }
void button::notify_has_opaque_parent(widget *parent) {} void button::notify_child_size_change(widget &child, coord old_size) {
set_size(coord(inner.size.x + 2, inner.size.y + 2));
void button::handle_key(struct key_packet kp) {} }
void button::on_focus() {} void button::notify_has_opaque_parent(widget *parent) {}
void button::on_unfocus() {}
} }

View file

@ -4,12 +4,27 @@
namespace raleigh { namespace raleigh {
label::label(const char *value, const char *font, bool bg_transparent, _pixel_t fg, _pixel_t bg) label::label(const char *value, const char *font, bool bg_transparent, _pixel_t fg, _pixel_t bg)
: value(value), fi(get_font(font)), bg_transparent(bg_transparent), fg(fg), bg(bg) { : fi(get_font(font)), bg_transparent(bg_transparent), fg(fg), bg(bg) {
size = coord(fi->space_width * (strlen(value) - 1) + fi->char_width, fi->char_height);
v_size = strlen(value) + 1;
this->value = new char[v_size];
blockcpy(this->value, value, v_size);
size = coord(fi->space_width * (v_size - 2) + fi->char_width, fi->char_height);
closest_opaque = 0; closest_opaque = 0;
} }
void label::notify_window_change() {} void label::change_value(const char *new_value) {
delete[] value;
const uint32_t ns = strlen(new_value) + 1;
if (ns != v_size) {
v_size = ns;
value = new char[ns];
set_size(coord(fi->space_width * (ns - 2) + fi->char_width, fi->char_height));
}
blockcpy(value, new_value, ns);
w->notify_needs_paint(*this);
}
void label::paint(_pixel_t *pixbuf, uint32_t pitch) { void label::paint(_pixel_t *pixbuf, uint32_t pitch) {
if (!bg_transparent) if (!bg_transparent)
@ -23,13 +38,7 @@ namespace raleigh {
} }
} }
void label::handle_click(coord window_coords, enum mouse_packet::mouse_button click_type, bool up) { }
void label::notify_has_opaque_parent(widget *parent) { void label::notify_has_opaque_parent(widget *parent) {
closest_opaque = parent; closest_opaque = parent;
} }
void label::handle_key(struct key_packet kp) {};
void label::on_focus() {};
void label::on_unfocus() {};
} }

View file

@ -5,6 +5,7 @@ namespace raleigh {
: inner(inner), pad_by(pad_by) { : inner(inner), pad_by(pad_by) {
size = coord(inner.size.x + pad_by * 2, inner.size.y + pad_by * 2); size = coord(inner.size.x + pad_by * 2, inner.size.y + pad_by * 2);
closest_opaque = 0; closest_opaque = 0;
inner.parent = this;
} }
void padding::notify_window_change() { void padding::notify_window_change() {
@ -30,7 +31,7 @@ namespace raleigh {
inner.notify_has_opaque_parent(parent); inner.notify_has_opaque_parent(parent);
} }
void padding::handle_key(struct key_packet kp) {}; void padding::notify_child_size_change(widget &child, coord old_size) {
void padding::on_focus() {}; set_size(coord(inner.size.x + pad_by * 2, inner.size.y + pad_by * 2));
void padding::on_unfocus() {}; }
} }

View file

@ -4,6 +4,7 @@ namespace raleigh {
vbox::vbox(dllist<widget &> widgets) : widgets(widgets) { vbox::vbox(dllist<widget &> widgets) : widgets(widgets) {
uint32_t w = 0, h = 0; uint32_t w = 0, h = 0;
for (dllist<widget &>::node *n = widgets.first; n; n = n->next) { for (dllist<widget &>::node *n = widgets.first; n; n = n->next) {
n->d.parent = this;
h += n->d.size.y; h += n->d.size.y;
if (n->d.size.x > w) if (n->d.size.x > w)
w = n->d.size.x; w = n->d.size.x;
@ -45,7 +46,21 @@ namespace raleigh {
n->d.notify_has_opaque_parent(parent); n->d.notify_has_opaque_parent(parent);
} }
void vbox::handle_key(struct key_packet kp) {}; void vbox::notify_child_size_change(widget &from, coord old_size) {
void vbox::on_focus() {}; if ((old_size.y == from.size.y) && (from.size.x <= size.x)) {
void vbox::on_unfocus() {}; from.window_offset.x = window_offset.x + size.x / 2 - from.size.x / 2;
from.notify_window_change();
}
else {//lazy, less efficient approach
uint32_t h = 0, w = 0;
for (dllist<widget &>::node *n = widgets.first; n; n = n->next) {
h += n->d.size.y;
if (n->d.size.x > w)
w = n->d.size.x;
}
set_size(coord(w, h));
notify_window_change();
}
}
} }

View file

@ -0,0 +1,22 @@
#include <raleigh/widget.h>
#include <knob/format.h>
namespace raleigh {
widget::widget()
: parent(0) {}
void widget::notify_window_change() {}
void widget::handle_click(coord window_coords, enum mouse_packet::mouse_button click_type, bool up) {}
void widget::handle_key(struct key_packet kp) {}
void widget::on_focus() {}
void widget::on_unfocus() {}
void widget::notify_child_size_change(widget &child, coord old_size) {}
void widget::set_size(coord new_size) {
coord old_size = size;
size = new_size;
if (w)
w->notify_widget_size_change(*this, old_size);
}
}

View file

@ -1,7 +1,6 @@
#include <raleigh/runtime.h> #include <raleigh/runtime.h>
#include <raleigh/window.h> #include <raleigh/window.h>
#include <popups/info.h> #include <knob/key.h>
#include <knob/heap.h>
#include <knob/format.h> #include <knob/format.h>
@ -44,8 +43,16 @@ namespace raleigh {
root.handle_click(coord(wa.as_mouse.x, wa.as_mouse.y), wa.as_mouse.which, false); root.handle_click(coord(wa.as_mouse.x, wa.as_mouse.y), wa.as_mouse.which, false);
else if (wa.action_type == wa.MOUSE_UP) else if (wa.action_type == wa.MOUSE_UP)
root.handle_click(coord(wa.as_mouse.x, wa.as_mouse.y), wa.as_mouse.which, true); root.handle_click(coord(wa.as_mouse.x, wa.as_mouse.y), wa.as_mouse.which, true);
else if (wa.action_type == wa.KEY_DOWN) else if (wa.action_type == wa.KEY_DOWN) {
for (dllist<duple<struct key_packet, void (*)(window &)>>::node *n = keybinds.first; n; n = n->next)
if (match_side_agnostic(wa.as_key, n->d.a)) {
n->d.b(*this);
goto next_loop;
}
focussed->handle_key(wa.as_key); focussed->handle_key(wa.as_key);
next_loop:
;
}
else if (wa.action_type == wa.FOCUS_ENTER) else if (wa.action_type == wa.FOCUS_ENTER)
focussed->on_focus(); focussed->on_focus();
else if (wa.action_type == wa.FOCUS_LEAVE) else if (wa.action_type == wa.FOCUS_LEAVE)
@ -83,4 +90,22 @@ namespace raleigh {
focussed->on_focus(); focussed->on_focus();
} }
} }
void window::notify_widget_size_change(widget &from, coord old_size) {
if (from.parent)
from.parent->notify_child_size_change(from, old_size);
else {
size = root.size;
delete[] pixbuf;
pixbuf = new _pixel_t[size.x * size.y];
if (!pixbuf)
show_error_and_quitf("Failed to allocate %u byte buffer while\nresizing window to %ux%u pixels.", size.x * size.y, size.x, size.y);
paint_full();
_resize_window(handle, size.x, size.y, pixbuf);
}
}
void window::add_keybind(struct key_packet kp, void (*handler)(window &)) {
keybinds.add_front(duple<struct key_packet, void (*)(window &)>(kp, handler));
}
} }

View file

@ -6,3 +6,6 @@ _Znaj = get_block;
/* void operator delete(void *, size_t) */ /* void operator delete(void *, size_t) */
_ZdlPvj = free_block; _ZdlPvj = free_block;
/* void operator delete[](void *) */
_ZdaPv = free_block;