ref: 3cbc44156098b3e1d2a5fce0bdb25aaa991efe2b
author: sirjofri <sirjofri@sirjofri.de>
date: Fri Jan 31 12:04:04 EST 2025
adds files
--- /dev/null
+++ b/dat.h
@@ -1,0 +1,18 @@
+typedef struct GPos GPos;
+typedef struct GBundle GBundle;
+typedef struct GProvider GProvider;
+
+struct GPos {
+ double lon;
+ double lat;
+};
+
+struct GBundle {
+ int x;
+ int y;
+ int z;
+};
+
+struct GProvider {
+ char* (*formaturl)(GBundle);
+};
--- /dev/null
+++ b/fns.h
@@ -1,0 +1,9 @@
+int lon2tilex(double lon, int z);
+int lat2tiley(double lat, int z);
+double tilex2long(int x, int z);
+double tiley2lat(int y, int z);
+
+GBundle getbundle(GPos pos, int zoom);
+GPos getlocation(void);
+
+GProvider *getprovider(void);
--- /dev/null
+++ b/gps.c
@@ -1,0 +1,43 @@
+#include <u.h>
+#include <libc.h>
+#include "dat.h"
+#include "fns.h"
+
+GPos
+getlocation()
+{
+ int fd;
+ GPos ret;
+ char buf[256];
+ char *fields[9];
+ int n, quality;
+ char *nop;
+
+ ret.lon = 0.;
+ ret.lat = 0.;
+
+ fd = open("/mnt/gps/position", OREAD);
+ if (fd < 0)
+ return ret;
+
+ if ((n = read(fd, buf, 256)) <= 0) {
+ fprint(2, "read error: %r\n");
+ close(fd);
+ return ret;
+ }
+ close(fd);
+ buf[n] = 0;
+
+ quality = atoi(buf);
+ if (!quality)
+ return ret;
+
+ if (getfields(buf, fields, 9, 1, " ") != 9) {
+ fprint(2, "read error: invalid fields\n");
+ return ret;
+ }
+
+ ret.lon = strtod(fields[3], &nop);
+ ret.lat = strtod(fields[4], &nop);
+ return ret;
+}
--- /dev/null
+++ b/map.c
@@ -1,0 +1,124 @@
+#include <u.h>
+#include <libc.h>
+#include <draw.h>
+#include <event.h>
+#include <nate/nate.h>
+#include <nate/n_window.h>
+#include <nate/n_hbox.h>
+#include <nate/n_vbox.h>
+#include <nate/n_box.h>
+#include <nate/n_label.h>
+#include <nate/n_button.h>
+#include <nate/n_image.h>
+#include "dat.h"
+#include "fns.h"
+
+NWindow *mainwindow;
+NImage *nimage;
+
+Image *testimage;
+
+void
+inittestimage(void)
+{
+ Image *r, *g;
+ testimage = allocimage(display, Rect(0, 0, 100, 100), screen->chan, 1, DWhite);
+ r = allocimage(display, Rect(0, 0, 1, 1), screen->chan, 1, DRed);
+ g = allocimage(display, Rect(0, 0, 1, 1), screen->chan, 1, DGreen);
+ draw(testimage, Rect(0, 0, 50, 50), r, nil, ZP);
+ draw(testimage, Rect(50, 50, 100, 100), g, nil, ZP);
+ freeimage(r);
+ freeimage(g);
+}
+
+void
+eresized(int new)
+{
+ if (new && getwindow(display, Refnone) < 0)
+ sysfatal("%r");
+
+ nateredraw();
+}
+
+int
+exitclicked(Mouse, Nelem *el, void*)
+{
+ exits(nil);
+}
+
+void
+main(int argc, char **argv)
+{
+ Event ev;
+ int z = 0;
+ char *url;
+
+ if (initdraw(nil, nil, "map") < 0)
+ sysfatal("%r");
+
+ einit(Emouse);
+ nateinit();
+
+ inittestimage();
+
+ NAssign(NWindow, &mainwindow, New_Window())
+ ->MakeRoot()
+ ->Slot(
+ New_VBox()
+ ->Slot(
+ New_HBox()
+ ->SizeToContent(1)
+ ->Slot(
+ New_Button()
+ ->SizeToContent(1)
+ ->Border(1, display->black)
+ ->Label("Exit")
+ ->OnClick(exitclicked, nil)
+ )
+ ->Slot(
+ New_Button()
+ ->SizeToContent(1)
+ ->Label("Test")
+ ->Border(1, display->black)
+ )
+ )
+ ->Slot(
+ New_Box()
+ ->Border(1, display->black)
+ ->Size(Pt(200, 300))
+ ->Slot(
+ NAssign(NImage, &nimage, New_Image())
+ ->Image(testimage)
+ )
+ )
+ ->Slot(
+ New_Box()
+ ->Border(1, display->black)
+ ->Size(Pt(500, 30))
+ ->Slot(
+ New_Label()
+ ->Label("Status Line")
+ )
+ )
+ );
+
+ nateredraw();
+
+ for (;;) {
+ switch (event(&ev)) {
+ case Emouse:
+ if (ev.mouse.buttons & 4)
+ exits(nil);
+ else
+ natemouseevent(ev.mouse);
+ break;
+ }
+ }
+
+ GPos loc = getlocation();
+ GBundle bundle = getbundle(loc, z);
+ GProvider *prov = getprovider();
+
+ url = prov->formaturl(bundle);
+ print("%s\n", url);
+}
--- /dev/null
+++ b/mkfile
@@ -1,0 +1,11 @@
+</$objtype/mkfile
+
+TARG=map
+OFILES=\
+ map.$O\
+ osm.$O\
+ gps.$O\
+
+HFILES=fns.h
+
+</sys/src/cmd/mkone
--- /dev/null
+++ b/osm.c
@@ -1,0 +1,61 @@
+#include <u.h>
+#include <libc.h>
+#include "dat.h"
+#include "fns.h"
+
+static double
+asinh(double x)
+{
+ double s = sqrt(x*x + 1);
+ return log(x + s);
+}
+
+int
+lon2tilex(double lon, int z)
+{
+ return (int)(floor((lon + 180.0) / 360.0 * (1<<z)));
+}
+
+int
+lat2tiley(double lat, int z)
+{
+ double latrad = lat * PI/180.0;
+ return (int)(floor((1.0 - asinh(tan(latrad)) / PI) / 2.0 * (1<<z)));
+}
+
+double tilex2long(int x, int z)
+{
+ return x / (double)(1<<z) * 360.0 - 180.0;
+}
+
+double tiley2lat(int y, int z)
+{
+ double n = PI - 2.0 * PI * y / (double)(1<<z);
+ return 180.0 / PI * atan(0.5 * (exp(n) - exp(-n)));
+}
+
+GBundle
+getbundle(GPos pos, int z)
+{
+ GBundle ret;
+ ret.x = lon2tilex(pos.lon, z);
+ ret.y = lat2tiley(pos.lat, z);
+ ret.z = z;
+ return ret;
+}
+
+static char*
+defaultformaturl(GBundle b)
+{
+ return smprint("https://tile.openstreetmap.org/%d/%d/%d.png", b.z, b.x, b.y);
+}
+
+static GProvider defaultprovider = {
+ .formaturl = defaultformaturl,
+};
+
+GProvider*
+getprovider()
+{
+ return &defaultprovider;
+}