nvg.lisp 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332
  1. (cl:defpackage :ocm.nvg
  2. (:use :cl :cffi-c-ref :alexandria)
  3. (:export run))
  4. (cl:in-package :ocm.nvg)
  5. (defun aget (alist key)
  6. (cdr (assoc key alist :test #'equal)))
  7. (defun agets (alist &rest keys)
  8. (reduce #'aget keys :initial-value alist))
  9. (defun agetter (&rest keys)
  10. (lambda (alist)
  11. (apply 'agets alist keys)))
  12. (defun tags (obj)
  13. (let ((tags (ocm.osm::obj-tags obj)))
  14. (when (functionp tags)
  15. (setf tags (funcall tags)
  16. (ocm.osm::obj-tags obj) tags))
  17. tags))
  18. (defun merge-tags (obj parent)
  19. (append obj parent))
  20. (defvar *context* nil "NanoVG Context")
  21. (defvar *window-width* 640.0)
  22. (defvar *window-height* 480.0)
  23. (defvar *frame-width* 640.0)
  24. (defvar *frame-height* 480.0)
  25. (defvar *pixel-ratio* 1.0)
  26. (defvar *scale* 3f0)
  27. (defvar *pos-x* 0f0)
  28. (defvar *pos-y* 0f0)
  29. (defvar *objects* nil "List of OSM referenced objects to render")
  30. (defun fix-multipoly (rel)
  31. (labels ((id (obj) (ocm.osm::obj-id obj))
  32. (try-attach (way ways)
  33. (loop
  34. with way-nodes = (ocm.osm::way-refs way)
  35. with way-first = (car way-nodes)
  36. with way-last = (car (last way-nodes))
  37. for prev in ways
  38. for prev-nodes = (ocm.osm::way-refs prev)
  39. for prev-first = (car prev-nodes)
  40. for prev-last = (car (last prev-nodes))
  41. when (and way-first way-last prev-first prev-last)
  42. do (when-let (nodes (cond
  43. ((= (id way-first) (id prev-last))
  44. (append prev-nodes (cdr way-nodes)))
  45. ((= (id way-last) (id prev-first))
  46. (append way-nodes (cdr prev-nodes)))
  47. ((= (id way-first) (id prev-first))
  48. (append (reverse way-nodes) (cdr prev-nodes)))
  49. ((= (id way-last) (id prev-last))
  50. (append prev-nodes (cdr (reverse way-nodes))))
  51. ))
  52. (setf (ocm.osm::way-refs prev) nodes)
  53. (return prev)))))
  54. (loop with inner-ways and outer-ways
  55. for mem in (ocm.osm::rel-memids rel)
  56. for mem-role in (ocm.osm::rel-roles rel)
  57. for mem-type in (ocm.osm::rel-types rel)
  58. for parts = (if (equal "inner" mem-role) inner-ways outer-ways)
  59. when (ocm.osm::way-p mem)
  60. unless (try-attach mem parts)
  61. do (let ((new-way (ocm.osm::make-way :id (id mem)
  62. :refs (copy-list (ocm.osm::way-refs mem)))))
  63. (if (equal "inner" mem-role) (push new-way inner-ways) (push new-way outer-ways)))
  64. finally (return (ocm.osm::make-rel
  65. :id (id rel) :tags (ocm.osm::obj-tags rel)
  66. :memids (append outer-ways inner-ways)
  67. :roles (append (loop for w in outer-ways collect "outer")
  68. (loop for w in inner-ways collect "inner"))
  69. :types (append (loop for w in outer-ways collect :way)
  70. (loop for w in inner-ways collect :way)))))))
  71. (defun osm-fix (obj)
  72. (cond
  73. ((multipoly-p obj) (fix-multipoly obj))
  74. (:otherwise obj)))
  75. (defun load-test-objects ()
  76. (setf *objects* nil)
  77. (ocm.osm::with-related (obj "spb.osm.pbf" '(
  78. (nil "natural" "water")
  79. (nil "natural" "wood")
  80. (nil "natural" "scrub")
  81. (nil "highway" "primary")
  82. (nil "highway" "secondary")))
  83. (push (osm-fix obj) *objects*)))
  84. (defparameter +granularity+ (expt 10 8))
  85. (defun node-x (node)
  86. (float (/ (ocm.osm::node-lon node) +granularity+)))
  87. (defun node-y (node)
  88. (float (- (/ (ocm.osm::node-lat node) +granularity+))))
  89. (defun render-path (nodes)
  90. (%nvg:begin-path *context*)
  91. (let ((start (car nodes)))
  92. (%nvg:move-to *context* (node-x start) (node-y start))
  93. (loop for node in (cdr nodes)
  94. do (%nvg:line-to *context* (node-x node) (node-y node)))))
  95. (defparameter +highway-color+ '(150 100 0 255))
  96. (defparameter +highway-primary-color+ '(150 30 0 255))
  97. (defparameter +strokes+
  98. `(
  99. (((:way "lanes" "5")) :color ,+highway-primary-color+ :width 0.005)
  100. (((:way "lanes" "3")) :color ,+highway-primary-color+ :width 0.004)
  101. (((:way "highway" "primary")) :color ,+highway-primary-color+ :width 0.003)
  102. (((:way "highway")) :color ,+highway-color+ :width 0.001)))
  103. (defun apply-stroke (tags)
  104. (when-let (options (find-match +strokes+ tags :way))
  105. (destructuring-bind (&key color width) options
  106. (when color
  107. (c-with ((n-color %nvg:color))
  108. (%nvg:stroke-color *context* (apply '%nvg:rgba (n-color &) color))))
  109. (when width (%nvg:stroke-width *context* width)))))
  110. (defun render-line (nodes tags)
  111. (render-path nodes)
  112. (apply-stroke tags)
  113. (%nvg:stroke *context*))
  114. (defparameter +water-color+ '(0 0 250 255))
  115. (defparameter +wood-color+ '(0 250 0 255))
  116. (defparameter +fills+
  117. `(
  118. (((nil "natural" "water")) :color ,+water-color+)
  119. (((nil "natural" "scrub")
  120. (nil "natural" "wood")) :color ,+wood-color+)
  121. ))
  122. (defun find-match (query-value tags type)
  123. (loop for (query . value) in query-value
  124. when (ocm.osm::query-match2 query type tags)
  125. do (return value)))
  126. (defun apply-fill (tags)
  127. (when-let (fill (find-match +fills+ tags :way))
  128. (destructuring-bind (&key color paint) fill
  129. (when color
  130. (c-with ((n-color %nvg:color))
  131. (%nvg:fill-color *context* (apply '%nvg:rgba (n-color &) color))))
  132. (when paint))))
  133. (defun render-poly (nodes tags &optional hole)
  134. (render-path nodes)
  135. (%nvg:close-path *context*)
  136. (if hole
  137. (%nvg:path-winding *context* (cffi:foreign-bitfield-value '%nvg:solidity :hole))
  138. (apply-fill tags))
  139. (%nvg:fill *context*))
  140. (defparameter +closed-lines+
  141. '((:way "highway")))
  142. (defun closed-p (nodes)
  143. (let ((first (first nodes))
  144. (last (car (last nodes))))
  145. (and first last
  146. (= (if (ocm.osm::obj-p first) (ocm.osm::obj-id first) first)
  147. (if (ocm.osm::obj-p last) (ocm.osm::obj-id last) last)))))
  148. (defun area-p (nodes tags)
  149. (and (closed-p nodes)
  150. (not (ocm.osm::query-match2 +closed-lines+ :way tags))))
  151. (defun multipoly-p (rel)
  152. (and (ocm.osm::rel-p rel)
  153. (equal (aget (tags rel) "type")
  154. "multipolygon")))
  155. (defun render-way (nodes tags)
  156. (if (area-p nodes tags)
  157. (render-poly nodes tags)
  158. (render-line nodes tags)))
  159. (defun render-multipoly (rel)
  160. (loop for mem in (ocm.osm::rel-memids rel)
  161. for mem-role in (ocm.osm::rel-roles rel)
  162. for mem-type in (ocm.osm::rel-types rel)
  163. when (ocm.osm::way-p mem) ;; skip non-ways
  164. do (render-poly (ocm.osm::way-refs mem)
  165. (tags rel)
  166. (equal "inner" mem-role))))
  167. (defun render-objects (objects)
  168. (loop for obj in objects
  169. do (typecase obj
  170. (ocm.osm::way
  171. (render-way (ocm.osm::way-refs obj) (tags obj)))
  172. (ocm.osm::rel
  173. (when (multipoly-p obj)
  174. (render-multipoly obj))))))
  175. (defun render ()
  176. (%nvg:begin-frame *context* *window-width* *window-height* *pixel-ratio*)
  177. (%nvg:transform *context* *scale* 0f0 0f0 *scale* *pos-x* *pos-y*)
  178. (render-objects *objects*)
  179. (%nvg:end-frame *context*))
  180. (defvar *pressed-keys* nil)
  181. (defun process-keys ()
  182. (when (member %glfw:+key-left+ *pressed-keys*) (incf *pos-x* +translate-step+))
  183. (when (member %glfw:+key-right+ *pressed-keys*) (decf *pos-x* +translate-step+))
  184. (when (member %glfw:+key-up+ *pressed-keys*) (incf *pos-y* +translate-step+))
  185. (when (member %glfw:+key-down+ *pressed-keys*) (decf *pos-y* +translate-step+))
  186. (when (member %glfw:+key-minus+ *pressed-keys*) (decf *scale* +scale-step+))
  187. (when (member %glfw:+key-equal+ *pressed-keys*) (incf *scale* +scale-step+)))
  188. (defparameter +translate-step+ 80f0)
  189. (defparameter +scale-step+ 0.3)
  190. (glfw:define-key-callback on-keys (window key scancode action mod-keys)
  191. (declare (ignorable window key scancode action mod-keys))
  192. (format t "KEY: ~{~A ~}~%" (list key scancode action mod-keys))
  193. (when (eql action %glfw:+press+)
  194. (pushnew key *pressed-keys*)
  195. (cond
  196. ((or
  197. (eql key %glfw:+key-escape+)
  198. (eql key %glfw:+key-q+))
  199. (%glfw:set-window-should-close window %glfw:+true+))
  200. ((eql key %glfw:+key-left+) (incf *pos-x* +translate-step+))
  201. ((eql key %glfw:+key-right+) (decf *pos-x* +translate-step+))
  202. ((eql key %glfw:+key-up+) (incf *pos-y* +translate-step+))
  203. ((eql key %glfw:+key-down+) (decf *pos-y* +translate-step+))
  204. ((eql key %glfw:+key-minus+) (decf *scale* +scale-step+))
  205. ((eql key %glfw:+key-equal+) (incf *scale* +scale-step+))))
  206. (when (eql action %glfw:+release+)
  207. (setf *pressed-keys* (delete key *pressed-keys*)))
  208. (print (list *scale* *pos-x* *pos-y*)))
  209. (glfw:define-framebuffer-size-callback on-fb-size (window width height)
  210. (declare (ignorable window))
  211. (setf *frame-width* (float width)
  212. *frame-height* (float height)
  213. *pixel-ratio* (/ *frame-width* *window-width*))
  214. (gl:viewport 0 0 *frame-width* *frame-height*))
  215. (glfw:define-window-size-callback on-win-size (window width height)
  216. (declare (ignorable window))
  217. (setf *window-width* (float width)
  218. *window-height* (float height)
  219. *pixel-ratio* (/ *frame-width* *window-width*)))
  220. (glfw::define-error-callback on-error (err desc)
  221. (format *error-output* "GLFW error ~A: ~A~%" err desc))
  222. (defun create-window ()
  223. (if (uiop:featurep :bodge-gl2)
  224. (glfw:with-window-hints ((%glfw:+context-version-major+ 2)
  225. (%glfw:+context-version-minor+ 1)
  226. (%glfw:+depth-bits+ 24)
  227. (%glfw:+stencil-bits+ 8))
  228. (%glfw:create-window 640 480 "Hello NanoVG 2" nil nil))
  229. (glfw:with-window-hints ((%glfw:+context-version-major+ 3)
  230. (%glfw:+context-version-minor+ 3)
  231. (%glfw:+opengl-profile+ %glfw:+opengl-core-profile+)
  232. (%glfw:+opengl-forward-compat+ %glfw:+true+)
  233. (%glfw:+depth-bits+ 24)
  234. (%glfw:+stencil-bits+ 8))
  235. (%glfw:create-window 640 480 "Hello NanoVG 3" nil nil))))
  236. (defun main ()
  237. ;; Initializing window and OpenGL context
  238. (glfw:with-init ()
  239. (%glfw:set-error-callback (cffi:callback on-error))
  240. (c-let ((window %glfw:window :from (create-window))
  241. (width :int :alloc t)
  242. (height :int :alloc t))
  243. (when (cffi:null-pointer-p (window &))
  244. (%glfw:terminate)
  245. (error "Failed to create GLFW window"))
  246. (%glfw:make-context-current (window &))
  247. ;; Mangling GL function pointers
  248. (glad:init)
  249. ;; Set event callbacks
  250. (%glfw:set-key-callback (window &) (cffi:callback on-keys))
  251. (%glfw:set-framebuffer-size-callback (window &) (cffi:callback on-fb-size))
  252. (%glfw:set-window-size-callback (window &) (cffi:callback on-win-size))
  253. ;; Get initial values
  254. (%glfw:get-framebuffer-size (window &) (width &) (height &))
  255. (setf *frame-width* (float width)
  256. *frame-height* (float height))
  257. (%glfw:get-window-size (window &) (width &) (height &))
  258. (setf *window-width* (float width)
  259. *window-height* (float height)
  260. *pixel-ratio* (/ *frame-width* *window-width*))
  261. (gl:viewport 0 0 *frame-width* *frame-height*)
  262. ;; reset
  263. (setf *scale* 540f0
  264. *pos-x* -162910.0
  265. *pos-y* 324540.0)
  266. ;; Creating NanoVG context
  267. (let ((*context* (nvg:make-context)))
  268. (unwind-protect
  269. (loop while (= (%glfw:window-should-close (window &)) 0) do
  270. (gl:clear-color 1f0 1f0 1f0 1f0)
  271. (gl:clear :color-buffer-bit :depth-buffer-bit :stencil-buffer-bit)
  272. ;; Step
  273. (process-keys)
  274. ;; Actually drawing into nanovg context and onto display
  275. (render)
  276. (%glfw:swap-buffers (window &))
  277. (%glfw:poll-events))
  278. ;; Cleaning up NanoVG context
  279. (nvg:destroy-context *context*))))))
  280. (defun run ()
  281. (main)
  282. ;; (trivial-main-thread:with-body-in-main-thread (:blocking t)
  283. ;; (float-features:with-float-traps-masked t
  284. ;; (main)))
  285. )