nvg.lisp 9.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270
  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 load-test-objects ()
  31. (setf *objects* nil)
  32. (ocm.osm::with-related (obj "spb.osm.pbf" '(
  33. (nil "natural" "water")
  34. (nil "natural" "wood")
  35. (nil "natural" "scrub")))
  36. (push obj *objects*)))
  37. (defparameter +granularity+ (expt 10 8))
  38. (defun node-x (node)
  39. (float (/ (ocm.osm::node-lon node) +granularity+)))
  40. (defun node-y (node)
  41. (float (- (/ (ocm.osm::node-lat node) +granularity+))))
  42. (defun render-path (nodes)
  43. (%nvg:begin-path *context*)
  44. (let ((start (car nodes)))
  45. (%nvg:move-to *context* (node-x start) (node-y start))
  46. (loop for node in (cdr nodes)
  47. do (%nvg:line-to *context* (node-x node) (node-y node)))))
  48. (defparameter +highway-color+ '(250 0 0 255))
  49. (defparameter +strokes+
  50. `(
  51. (((:way "highway")) :color ,+highway-color+ :width 10)))
  52. (defun apply-stroke (tags)
  53. (when-let (options (find-match +strokes+ tags :way))
  54. (destructuring-bind (&key color width) options
  55. (when color
  56. (c-with ((n-color %nvg:color))
  57. (%nvg:stroke-color *context* (apply '%nvg:rgba (n-color &) color))))
  58. (when width (%nvg:stroke-width *context* width)))))
  59. (defun render-line (nodes tags)
  60. (render-path nodes)
  61. (apply-stroke tags)
  62. (%nvg:stroke *context*))
  63. (defparameter +water-color+ '(0 0 250 255))
  64. (defparameter +wood-color+ '(0 250 0 255))
  65. (defparameter +fills+
  66. `(
  67. (((nil "natural" "water")) :color ,+water-color+)
  68. (((nil "natural" "scrub")
  69. (nil "natural" "wood")) :color ,+wood-color+)
  70. ))
  71. (defun find-match (query-value tags type)
  72. (loop for (query . value) in query-value
  73. when (ocm.osm::query-match2 query type tags)
  74. do (return value)))
  75. (defun apply-fill (tags)
  76. (when-let (fill (find-match +fills+ tags :way))
  77. (destructuring-bind (&key color paint) fill
  78. (when color
  79. (c-with ((n-color %nvg:color))
  80. (%nvg:fill-color *context* (apply '%nvg:rgba (n-color &) color))))
  81. (when paint))))
  82. (defun render-poly (nodes tags &optional hole)
  83. (render-path nodes)
  84. (%nvg:close-path *context*)
  85. (when hole
  86. (%nvg:path-winding *context* (cffi:foreign-bitfield-value '%nvg:solidity :hole)))
  87. (apply-fill tags)
  88. (%nvg:fill *context*))
  89. (defparameter +closed-lines+
  90. '((:way "highway")))
  91. (defun is-area (nodes tags)
  92. (let ((first (first nodes))
  93. (last (car (last nodes))))
  94. (and first last
  95. (= (if (ocm.osm::obj-p first) (ocm.osm::obj-id first) first)
  96. (if (ocm.osm::obj-p last) (ocm.osm::obj-id last) last))
  97. (not (ocm.osm::query-match2 +closed-lines+ :way tags)))))
  98. (defun render-way (nodes tags)
  99. (if (is-area nodes tags)
  100. (render-poly nodes tags)
  101. (render-path nodes)))
  102. (defun render-multipoly (rel)
  103. (loop for mem in (ocm.osm::rel-memids rel)
  104. for mem-role in (ocm.osm::rel-roles rel)
  105. for mem-type in (ocm.osm::rel-types rel)
  106. when (ocm.osm::way-p mem) ;; skip non-ways
  107. do (render-poly (ocm.osm::way-refs mem)
  108. (merge-tags (tags mem) (tags rel))
  109. (equal "inner" mem-role))))
  110. (defun render-objects (objects)
  111. (loop for obj in objects
  112. do (typecase obj
  113. (ocm.osm::way
  114. (render-way (ocm.osm::way-refs obj) (tags obj)))
  115. (ocm.osm::rel
  116. (when (equal (aget (tags obj) "type")
  117. "multipolygon")
  118. (render-multipoly obj))))))
  119. (defun render ()
  120. (%nvg:begin-frame *context* *window-width* *window-height* *pixel-ratio*)
  121. (%nvg:transform *context* *scale* 0f0 0f0 *scale* *pos-x* *pos-y*)
  122. (render-objects *objects*)
  123. (%nvg:end-frame *context*))
  124. (defun process-keys ()
  125. (when (member %glfw:+key-left+ *pressed-keys*) (incf *pos-x* +translate-step+))
  126. (when (member %glfw:+key-right+ *pressed-keys*) (decf *pos-x* +translate-step+))
  127. (when (member %glfw:+key-up+ *pressed-keys*) (incf *pos-y* +translate-step+))
  128. (when (member %glfw:+key-down+ *pressed-keys*) (decf *pos-y* +translate-step+))
  129. (when (member %glfw:+key-minus+ *pressed-keys*) (decf *scale* +scale-step+))
  130. (when (member %glfw:+key-equal+ *pressed-keys*) (incf *scale* +scale-step+)))
  131. (defparameter +translate-step+ 80f0)
  132. (defparameter +scale-step+ 0.3)
  133. (defvar *pressed-keys* nil)
  134. (glfw:define-key-callback on-keys (window key scancode action mod-keys)
  135. (declare (ignorable window key scancode action mod-keys))
  136. (format t "KEY: ~{~A ~}~%" (list key scancode action mod-keys))
  137. (when (eql action %glfw:+press+)
  138. (pushnew key *pressed-keys*)
  139. (cond
  140. ((or
  141. (eql key %glfw:+key-escape+)
  142. (eql key %glfw:+key-q+))
  143. (%glfw:set-window-should-close window %glfw:+true+))
  144. ((eql key %glfw:+key-left+) (incf *pos-x* +translate-step+))
  145. ((eql key %glfw:+key-right+) (decf *pos-x* +translate-step+))
  146. ((eql key %glfw:+key-up+) (incf *pos-y* +translate-step+))
  147. ((eql key %glfw:+key-down+) (decf *pos-y* +translate-step+))
  148. ((eql key %glfw:+key-minus+) (decf *scale* +scale-step+))
  149. ((eql key %glfw:+key-equal+) (incf *scale* +scale-step+))))
  150. (when (eql action %glfw:+release+)
  151. (setf *pressed-keys* (delete key *pressed-keys*)))
  152. (print (list *scale* *pos-x* *pos-y*)))
  153. (glfw:define-framebuffer-size-callback on-fb-size (window width height)
  154. (declare (ignorable window))
  155. (setf *frame-width* (float width)
  156. *frame-height* (float height)
  157. *pixel-ratio* (/ *frame-width* *window-width*))
  158. (gl:viewport 0 0 *frame-width* *frame-height*))
  159. (glfw:define-window-size-callback on-win-size (window width height)
  160. (declare (ignorable window))
  161. (setf *window-width* (float width)
  162. *window-height* (float height)
  163. *pixel-ratio* (/ *frame-width* *window-width*)))
  164. (glfw::define-error-callback on-error (err desc)
  165. (format *error-output* "GLFW error ~A: ~A~%" err desc))
  166. (defun create-window ()
  167. (if (uiop:featurep :bodge-gl2)
  168. (glfw:with-window-hints ((%glfw:+context-version-major+ 2)
  169. (%glfw:+context-version-minor+ 1)
  170. (%glfw:+depth-bits+ 24)
  171. (%glfw:+stencil-bits+ 8))
  172. (%glfw:create-window 640 480 "Hello NanoVG 2" nil nil))
  173. (glfw:with-window-hints ((%glfw:+context-version-major+ 3)
  174. (%glfw:+context-version-minor+ 3)
  175. (%glfw:+opengl-profile+ %glfw:+opengl-core-profile+)
  176. (%glfw:+opengl-forward-compat+ %glfw:+true+)
  177. (%glfw:+depth-bits+ 24)
  178. (%glfw:+stencil-bits+ 8))
  179. (%glfw:create-window 640 480 "Hello NanoVG 3" nil nil))))
  180. (defun main ()
  181. ;; Initializing window and OpenGL context
  182. (glfw:with-init ()
  183. (%glfw:set-error-callback (cffi:callback on-error))
  184. (c-let ((window %glfw:window :from (create-window))
  185. (width :int :alloc t)
  186. (height :int :alloc t))
  187. (when (cffi:null-pointer-p (window &))
  188. (%glfw:terminate)
  189. (error "Failed to create GLFW window"))
  190. (%glfw:make-context-current (window &))
  191. ;; Mangling GL function pointers
  192. (glad:init)
  193. ;; Set event callbacks
  194. (%glfw:set-key-callback (window &) (cffi:callback on-keys))
  195. (%glfw:set-framebuffer-size-callback (window &) (cffi:callback on-fb-size))
  196. (%glfw:set-window-size-callback (window &) (cffi:callback on-win-size))
  197. ;; Get initial values
  198. (%glfw:get-framebuffer-size (window &) (width &) (height &))
  199. (setf *frame-width* (float width)
  200. *frame-height* (float height))
  201. (%glfw:get-window-size (window &) (width &) (height &))
  202. (setf *window-width* (float width)
  203. *window-height* (float height)
  204. *pixel-ratio* (/ *frame-width* *window-width*))
  205. (gl:viewport 0 0 *frame-width* *frame-height*)
  206. ;; reset
  207. (setf *scale* 300f0
  208. *pos-x* -90270.0
  209. *pos-y* 180380.0)
  210. ;; Creating NanoVG context
  211. (let ((*context* (nvg:make-context)))
  212. (unwind-protect
  213. (loop while (= (%glfw:window-should-close (window &)) 0) do
  214. (gl:clear-color 1f0 1f0 1f0 1f0)
  215. (gl:clear :color-buffer-bit :depth-buffer-bit :stencil-buffer-bit)
  216. ;; Step
  217. (process-keys)
  218. ;; Actually drawing into nanovg context and onto display
  219. (render)
  220. (%glfw:swap-buffers (window &))
  221. (%glfw:poll-events))
  222. ;; Cleaning up NanoVG context
  223. (nvg:destroy-context *context*))))))
  224. (defun run ()
  225. (main)
  226. ;; (trivial-main-thread:with-body-in-main-thread (:blocking t)
  227. ;; (float-features:with-float-traps-masked t
  228. ;; (main)))
  229. )