summaryrefslogtreecommitdiff
path: root/sources/main.c
diff options
context:
space:
mode:
authordweller <dweller@cabin.digital>2025-03-28 19:52:47 +0200
committerdweller <dweller@cabin.digital>2025-03-28 19:52:47 +0200
commitb41ec911812e66931f01939378979845716b6119 (patch)
tree08be727857d9881182a723af382680c48fb89434 /sources/main.c
parent2bcb97cade32e4781135ff4c1500b95fcf351889 (diff)
experimenting with UI
Diffstat (limited to 'sources/main.c')
-rw-r--r--sources/main.c343
1 files changed, 342 insertions, 1 deletions
diff --git a/sources/main.c b/sources/main.c
index 5bc1c42..791cf43 100644
--- a/sources/main.c
+++ b/sources/main.c
@@ -5,7 +5,6 @@
* See LICENSE for details
*/
-
#define _DEFAULT_SOURCE
#include <stdlib.h>
#include <stdio.h>
@@ -13,19 +12,360 @@
#include <assert.h>
#include <time.h>
#include <stdarg.h>
+#include <ctype.h>
#include <endian.h>
+#include <unistd.h>
+#include <sys/mman.h>
+#include <sys/stat.h>
+#include <fcntl.h>
+
+#include <X11/Xlib.h>
+#include <X11/Xatom.h>
+#include <X11/Xutil.h>
+#include <X11/Xresource.h>
+
#include "bits.c"
#include "log.c"
+#include "tga.c"
+#include "x11.c"
#include "chip8.c"
#include "meta/disasm.c"
#include "meta/exec.c"
+typedef struct
+{
+ void* usr;
+
+ usize hot;
+ usize active;
+ usize kbdfocus;
+
+ /* mouse */
+ u16 x;
+ u16 y;
+
+ u8 pbtns;
+ u8 btns;
+
+} iui;
+
+void iui_start(iui* ui, void* usr)
+{
+ assert(ui);
+
+ ui->usr = usr;
+ ui->hot = 0;
+}
+
+void iui_end(iui* ui)
+{
+ assert(ui);
+
+ if(!ui->btns) ui->active = 0;
+ else
+ {
+ if(!ui->active) ui->kbdfocus = 0;
+ }
+
+ ui->pbtns = ui->btns;
+}
+
+bool iui_mouse_inside(iui* ui, u16 x, u16 y, u16 w, u16 h)
+{
+ return !(ui->x < x || ui->y < y || ui->x >= x + w || ui->y >= y + h);
+}
+
+int iui_button(iui* ui, u16 x, u16 y, int w, int h, char* label)
+{
+ xctx* x11 = (xctx*)ui->usr;
+ usize id = (usize)label;
+ bool imhot, imactive;
+
+ if(iui_mouse_inside(ui, x, y, w, h))
+ {
+ bool click = ui->btns & bit(1) && !(ui->pbtns & bit(1));
+ if(!ui->active && click) ui->active = id;
+ ui->hot = id;
+ }
+
+ imhot = ui->hot == id;
+ imactive = ui->active == id;
+
+ {
+ XGCValues gcvals = {0};
+
+ if(imhot) gcvals.line_width = 2;
+ else gcvals.line_width = 1;
+
+ if(imactive) gcvals.foreground = x11->cols.as.fg;
+ else gcvals.foreground = x11->cols.as.bg;
+
+ XChangeGC(x11->disp, x11->gc, GCForeground | GCLineWidth, &gcvals);
+ XFillRectangle(x11->disp, x11->wnd, x11->gc, x, y, w, h);
+
+ XSetForeground(x11->disp, x11->gc, x11->cols.as.fg);
+ XDrawLine(x11->disp, x11->wnd, x11->gc, x, y, x, y + h);
+ XDrawLine(x11->disp, x11->wnd, x11->gc, x + w, y, x, y);
+
+ XSetForeground(x11->disp, x11->gc, x11->cols.as.fg);
+ XDrawLine(x11->disp, x11->wnd, x11->gc, x, y + h, x + w, y + h);
+ XDrawLine(x11->disp, x11->wnd, x11->gc, x + w, y + h, x + w, y);
+ }
+
+ {
+ usize len;
+ int xx, yy;
+ int di, fa, fd;
+ int width, height;
+ XCharStruct chs = {0};
+
+ len = strlen(label);
+ XTextExtents(x11->xfs, label, len, &di, &fa, &fd, &chs);
+ width = chs.rbearing - chs.lbearing;
+ height = chs.ascent + chs.descent;
+
+ xx = x + w / 2 - width / 2;
+ yy = y + h / 2 + height / 2;
+
+ if(imactive) XSetForeground(x11->disp, x11->gc, x11->cols.as.bg);
+ else XSetForeground(x11->disp, x11->gc, x11->cols.as.fg);
+ XDrawString(x11->disp, x11->wnd, x11->gc, xx, yy, label, len);
+ }
+
+ return imhot && imactive && !(ui->btns & bit(1));
+}
+
+int iui_textbox(iui* ui, u16 x, u16 y, int w, int h, char* buf, usize buflen)
+{
+ xctx* x11 = (xctx*)ui->usr;
+ usize id = (usize)buf;
+ bool imhot, imactive;
+ usize len;
+
+ len = strlen(buf);
+
+ if(iui_mouse_inside(ui, x, y, w, h))
+ {
+ bool click = ui->btns & bit(1) && !(ui->pbtns & bit(1));
+ if(!ui->active && click) ui->active = id;
+ ui->hot = id;
+ }
+
+ imhot = ui->hot == id;
+ imactive = ui->active == id;
+
+ {
+ XGCValues gcvals = {0};
+
+ if(imhot) gcvals.line_width = 2;
+ else gcvals.line_width = 1;
+
+ gcvals.foreground = x11->cols.as.bg;
+
+ XChangeGC(x11->disp, x11->gc, GCForeground | GCLineWidth, &gcvals);
+ XFillRectangle(x11->disp, x11->wnd, x11->gc, x, y, w, h);
+
+ XSetForeground(x11->disp, x11->gc, x11->cols.as.fg);
+ XDrawLine(x11->disp, x11->wnd, x11->gc, x, y, x, y + h);
+ XDrawLine(x11->disp, x11->wnd, x11->gc, x + w, y, x, y);
+
+ XSetForeground(x11->disp, x11->gc, x11->cols.as.fg);
+ XDrawLine(x11->disp, x11->wnd, x11->gc, x, y + h, x + w, y + h);
+ XDrawLine(x11->disp, x11->wnd, x11->gc, x + w, y + h, x + w, y);
+ }
+
+ if(imactive) ui->kbdfocus = id;
+ if(ui->kbdfocus == id)
+ {
+
+ XGCValues gcvals = {0};
+ gcvals.foreground = x11->cols.as.fg;
+
+ XChangeGC(x11->disp, x11->gc, GCForeground, &gcvals);
+ XFillRectangle(x11->disp, x11->wnd, x11->gc, x + 6 + (6 * len), y + 6, 10, h - 12);
+ }
+ else
+ {
+ XGCValues gcvals = {0};
+ gcvals.foreground = x11->cols.as.fg;
+ gcvals.line_width = 1;
+
+ XChangeGC(x11->disp, x11->gc, GCForeground | GCLineWidth, &gcvals);
+ XDrawRectangle(x11->disp, x11->wnd, x11->gc, x + 6 + (6 * len), y + 6, 9, h - 13);
+ }
+
+ {
+ int xx, yy;
+ int di, fa, fd;
+ int width, height;
+ XCharStruct chs = {0};
+
+ XTextExtents(x11->xfs, buf, buflen, &di, &fa, &fd, &chs);
+ height = chs.ascent + chs.descent;
+
+ xx = x + 6;
+ yy = y + h / 2 + height / 2;
+
+
+ if(imactive) XSetForeground(x11->disp, x11->gc, x11->cols.as.bg);
+ else XSetForeground(x11->disp, x11->gc, x11->cols.as.fg);
+ XDrawString(x11->disp, x11->wnd, x11->gc, xx, yy, buf, len);
+ }
+
+ return imhot && imactive && !(ui->btns & bit(1));
+}
+
+
int main(int argc, char** argv)
{
+ xctx x11 = {0};
+ iui ui = {0};
+ bool run = true;
+ char txtbuf[256] = {0};
+
+ x11_init(800, 600, "xip-8", argc, argv, &x11);
+
+ while(run)
+ {
+ XEvent ev;
+
+ while(XCheckTypedWindowEvent(x11.disp, x11.wnd, ClientMessage, &ev))
+ if((Atom)ev.xclient.data.l[0] == x11.xa_delwnd) run = false;
+
+ while(XCheckWindowEvent(x11.disp, x11.wnd, max_u32, &ev))
+ {
+ bool keyup = false;
+ switch(ev.type)
+ {
+ case KeymapNotify:
+ XRefreshKeyboardMapping(&ev.xmapping);
+ break;
+
+ case KeyRelease:
+ keyup = true;
+ /* fall through */
+ case KeyPress:
+ {
+ KeySym ks;
+ char name[32] = {0};
+ int len = 0;
+
+ len = XLookupString(&ev.xkey, name, sizeof(name) - 1, &ks, NULL);
+ if(keyup && ks == XK_Escape) run = false;
+ if(!keyup && len == 1)
+ {
+ if(ui.kbdfocus && (isgraph(name[0]) || name[0] == ' '))
+ strcat((char*)ui.kbdfocus, name);
+ }
+ if(!keyup && ks == XK_BackSpace)
+ {
+ if(ui.kbdfocus)
+ {
+ ssize i = strlen((char*)ui.kbdfocus) - 1;
+ if(i >= 0) ((char*)ui.kbdfocus)[i] = '\0';
+ }
+ }
+ }
+ break;
+
+ case ButtonPress:
+ if(ev.xbutton.button == Button1) ui.btns |= bit(1);
+ if(ev.xbutton.button == Button2) ui.btns |= bit(2);
+ if(ev.xbutton.button == Button3) ui.btns |= bit(3);
+ break;
+
+ case ButtonRelease:
+ if(ev.xbutton.button == Button1) ui.btns &= ~bit(1);
+ if(ev.xbutton.button == Button2) ui.btns &= ~bit(2);
+ if(ev.xbutton.button == Button3) ui.btns &= ~bit(3);
+ break;
+
+ case MotionNotify:
+ ui.x = ev.xmotion.x;
+ ui.y = ev.xmotion.y;
+ break;
+
+ case Expose:
+ {
+ XWindowAttributes wa = {0};
+ XGCValues gcvals = {0};
+
+ gcvals.line_width = 1;
+ gcvals.foreground = x11.cols.as.bg;
+
+ XGetWindowAttributes(x11.disp, x11.wnd, &wa);
+
+ XChangeGC(x11.disp, x11.gc, GCForeground | GCLineWidth, &gcvals);
+ XFillRectangle(x11.disp, x11.wnd, x11.gc, 0, 0, wa.width, wa.height);
+
+ /* TODO: draw ui
+ * for ui draw cmds
+ * draw cmd
+ */
+ }
+ break;
+
+ default:
+ break;
+ }
+ }
+
+ {
+ int len = 0;
+ char buf[512] = {0};
+ XGCValues gcvals = {0};
+
+ gcvals.foreground = x11.cols.as.bg;
+ XChangeGC(x11.disp, x11.gc, GCForeground | GCLineWidth, &gcvals);
+ XFillRectangle(x11.disp, x11.wnd, x11.gc, 0, 0, 800, 600);
+
+ XSetForeground(x11.disp, x11.gc, x11.cols.as.fg);
+ len = snprintf(buf, lengthof(buf), "hot: %p", (void*)ui.hot);
+ XDrawString(x11.disp, x11.wnd, x11.gc, 650, 25, buf, len);
+
+ len = snprintf(buf, lengthof(buf), "act: %p", (void*)ui.active);
+ XDrawString(x11.disp, x11.wnd, x11.gc, 650, 45, buf, len);
+
+ len = snprintf(buf, lengthof(buf), "cur: %dx%d", ui.x, ui.y);
+ XDrawString(x11.disp, x11.wnd, x11.gc, 650, 65, buf, len);
+
+ len = snprintf(buf, lengthof(buf), "btn: %d%d%d",
+ ui.btns & bit(1) ? 1 : 0,
+ ui.btns & bit(2) ? 2 : 0,
+ ui.btns & bit(3) ? 3 : 0);
+ XDrawString(x11.disp, x11.wnd, x11.gc, 650, 85, buf, len);
+
+ len = snprintf(buf, lengthof(buf), "old: %d%d%d",
+ ui.pbtns & bit(1) ? 1 : 0,
+ ui.pbtns & bit(2) ? 2 : 0,
+ ui.pbtns & bit(3) ? 3 : 0);
+ XDrawString(x11.disp, x11.wnd, x11.gc, 650, 105, buf, len);
+
+ len = snprintf(buf, lengthof(buf), "kbd: %p", (void*)ui.kbdfocus);
+ XDrawString(x11.disp, x11.wnd, x11.gc, 650, 125, buf, len);
+ }
+
+ iui_start(&ui, &x11);
+ if(iui_button(&ui, 25, 25, 100, 35, "Click ME!")) report(DBG, "CLICK!\n");
+ if(iui_button(&ui, 150, 25, 100, 35, "NOT ME!!!")) report(DBG, "WHYYYYYY!\n");
+
+ iui_textbox(&ui, 25, 75, 225, 35, txtbuf, lengthof(txtbuf));
+ if(iui_button(&ui, 275, 75, 100, 35, "OK")) report(DBG, "textbox: '%s'!\n", txtbuf);
+ iui_end(&ui);
+
+ usleep(10000);
+ }
+
+ x11_cleanup(&x11);
+ return 0;
+}
+
+int main2(int argc, char** argv)
+{
u64 sz = 0;
FILE* f = NULL;
chip8 c8 = {0};
@@ -93,6 +433,7 @@ int main(int argc, char** argv)
{
char buf[512] = {0};
+
c8_disasm(c8.RAM, c8.pc, buf);
printf("%04X: %s\n", c8.pc, buf);