Mercurial > cortex
changeset 108:92b857b6145d
slow implementation of UV-mapped touch is complete
author | Robert McIntyre <rlm@mit.edu> |
---|---|
date | Sun, 15 Jan 2012 04:08:31 -0700 (2012-01-15) |
parents | 53fb379ac678 |
children | c05d8d222166 |
files | assets/Models/creature1/tip.png assets/Models/creature1/try-again.blend org/test-creature.org org/util.org |
diffstat | 4 files changed, 168 insertions(+), 200 deletions(-) [+] |
line wrap: on
line diff
1.1 Binary file assets/Models/creature1/tip.png has changed
2.1 Binary file assets/Models/creature1/try-again.blend has changed
3.1 --- a/org/test-creature.org Sun Jan 15 00:40:49 2012 -0700 3.2 +++ b/org/test-creature.org Sun Jan 15 04:08:31 2012 -0700 3.3 @@ -65,6 +65,7 @@ 3.4 (import java.awt.Dimension) 3.5 (import com.aurellem.capture.RatchetTimer) 3.6 (declare joint-create) 3.7 +(use 'clojure.contrib.def) 3.8 3.9 (defn view-image 3.10 "Initailizes a JPanel on which you may draw a BufferedImage. 3.11 @@ -142,40 +143,41 @@ 3.12 "Take a set of pairs of integers and collapse them into a 3.13 contigous bitmap." 3.14 [points] 3.15 - (let 3.16 - [num-points (count points) 3.17 - center (vector 3.18 - (int (average (map first points))) 3.19 - (int (average (map first points)))) 3.20 - flattened 3.21 - (reduce 3.22 - concat 3.23 - (map 3.24 - (fn [column] 3.25 - (map vector 3.26 - (map first column) 3.27 - (collapse-1d (second center) 3.28 - (map second column)))) 3.29 - (partition-by first (sort-by first points)))) 3.30 - squeezed 3.31 - (reduce 3.32 - concat 3.33 - (map 3.34 - (fn [row] 3.35 - (map vector 3.36 - (collapse-1d (first center) 3.37 - (map first row)) 3.38 - (map second row))) 3.39 - (partition-by second (sort-by second flattened)))) 3.40 - relocate 3.41 - (let [min-x (apply min (map first squeezed)) 3.42 - min-y (apply min (map second squeezed))] 3.43 - (map (fn [[x y]] 3.44 - [(- x min-x) 3.45 - (- y min-y)]) 3.46 - squeezed))] 3.47 - relocate 3.48 - )) 3.49 + (if (empty? points) [] 3.50 + (let 3.51 + [num-points (count points) 3.52 + center (vector 3.53 + (int (average (map first points))) 3.54 + (int (average (map first points)))) 3.55 + flattened 3.56 + (reduce 3.57 + concat 3.58 + (map 3.59 + (fn [column] 3.60 + (map vector 3.61 + (map first column) 3.62 + (collapse-1d (second center) 3.63 + (map second column)))) 3.64 + (partition-by first (sort-by first points)))) 3.65 + squeezed 3.66 + (reduce 3.67 + concat 3.68 + (map 3.69 + (fn [row] 3.70 + (map vector 3.71 + (collapse-1d (first center) 3.72 + (map first row)) 3.73 + (map second row))) 3.74 + (partition-by second (sort-by second flattened)))) 3.75 + relocate 3.76 + (let [min-x (apply min (map first squeezed)) 3.77 + min-y (apply min (map second squeezed))] 3.78 + (map (fn [[x y]] 3.79 + [(- x min-x) 3.80 + (- y min-y)]) 3.81 + squeezed))] 3.82 + relocate 3.83 + ))) 3.84 3.85 (defn load-bullet [] 3.86 (let [sim (world (Node.) {} no-op no-op)] 3.87 @@ -417,33 +419,10 @@ 3.88 3.89 (import ij.ImagePlus) 3.90 3.91 -(defn triangle-indices 3.92 - "Get the triangle vertex indices of a given triangle from a given 3.93 - mesh." 3.94 - [#^Mesh mesh triangle-index] 3.95 - (let [indices (int-array 3)] 3.96 - (.getTriangle mesh triangle-index indices) 3.97 - (vec indices))) 3.98 +;; Every Mesh has many triangles, each with its own index. 3.99 +;; Every vertex has its own index as well. 3.100 3.101 -(defn uv-coord 3.102 - "Get the uv-coordinates of the vertex named by vertex-index" 3.103 - [#^Mesh mesh vertex-index] 3.104 - (let [UV-buffer 3.105 - (.getData 3.106 - (.getBuffer 3.107 - mesh 3.108 - VertexBuffer$Type/TexCoord))] 3.109 - (Vector2f. 3.110 - (.get UV-buffer (* vertex-index 2)) 3.111 - (.get UV-buffer (+ 1 (* vertex-index 2)))))) 3.112 - 3.113 -(defn tri-uv-coord 3.114 - "Get the uv-cooridnates of the triangle's verticies." 3.115 - [#^Mesh mesh #^Triangle triangle] 3.116 - (map (partial uv-coord mesh) 3.117 - (triangle-indices mesh (.getIndex triangle)))) 3.118 - 3.119 -(defn touch-receptor-image 3.120 +(defn tactile-sensor-image 3.121 "Return the touch-sensor distribution image in ImagePlus format, or 3.122 nil if it does not exist." 3.123 [#^Geometry obj] 3.124 @@ -456,9 +435,7 @@ 3.125 [texture 3.126 (.getTextureValue texture-param) 3.127 im (.getImage texture)] 3.128 - (ImagePlus. 3.129 - "UV-map" 3.130 - (ImageToAwt/convert im false false 0)))))) 3.131 + (ImageToAwt/convert im false false 0))))) 3.132 3.133 (import ij.process.ImageProcessor) 3.134 (import java.awt.image.BufferedImage) 3.135 @@ -466,25 +443,58 @@ 3.136 (def white -1) 3.137 3.138 (defn filter-pixels 3.139 - "List the coordinates of all pixels matching pred." 3.140 + "List the coordinates of all pixels matching pred, within the bounds 3.141 + provided. Bounds -> [x0 y0 width height]" 3.142 {:author "Dylan Holmes"} 3.143 - [pred #^ImageProcessor ip] 3.144 - (let 3.145 - [width (.getWidth ip) 3.146 - height (.getHeight ip)] 3.147 - ((fn accumulate [x y matches] 3.148 - (cond 3.149 - (>= y height) matches 3.150 - (>= x width) (recur 0 (inc y) matches) 3.151 - (pred (.getPixel ip x y)) 3.152 - (recur (inc x) y (conj matches (Vector2f. x y))) 3.153 - :else (recur (inc x) y matches))) 3.154 - 0 0 []))) 3.155 + ([pred #^BufferedImage image] 3.156 + (filter-pixels pred image [0 0 (.getWidth image) (.getHeight image)])) 3.157 + ([pred #^BufferedImage image [x0 y0 width height]] 3.158 + ((fn accumulate [x y matches] 3.159 + (cond 3.160 + (>= y (+ height y0)) matches 3.161 + (>= x (+ width x0)) (recur 0 (inc y) matches) 3.162 + (pred (.getRGB image x y)) 3.163 + (recur (inc x) y (conj matches [x y])) 3.164 + :else (recur (inc x) y matches))) 3.165 + x0 y0 []))) 3.166 3.167 (defn white-coordinates 3.168 - "List the coordinates of all the white pixels in an image." 3.169 - [#^ImageProcessor ip] 3.170 - (filter-pixels #(= % white) ip)) 3.171 + "Coordinates of all the white pixels in a subset of the image." 3.172 + [#^BufferedImage image bounds] 3.173 + (filter-pixels #(= % white) image bounds)) 3.174 + 3.175 +(defn triangle 3.176 + "Get the triangle specified by triangle-index from the mesh" 3.177 + [#^Mesh mesh triangle-index] 3.178 + (let [scratch (Triangle.)] 3.179 + (.getTriangle mesh triangle-index scratch) 3.180 + scratch)) 3.181 + 3.182 +(defn triangle-vertex-indices 3.183 + "Get the triangle vertex indices of a given triangle from a given 3.184 + mesh." 3.185 + [#^Mesh mesh triangle-index] 3.186 + (let [indices (int-array 3)] 3.187 + (.getTriangle mesh triangle-index indices) 3.188 + (vec indices))) 3.189 + 3.190 +(defn vertex-UV-coord 3.191 + "Get the uv-coordinates of the vertex named by vertex-index" 3.192 + [#^Mesh mesh vertex-index] 3.193 + (let [UV-buffer 3.194 + (.getData 3.195 + (.getBuffer 3.196 + mesh 3.197 + VertexBuffer$Type/TexCoord))] 3.198 + [(.get UV-buffer (* vertex-index 2)) 3.199 + (.get UV-buffer (+ 1 (* vertex-index 2)))])) 3.200 + 3.201 +(defn triangle-UV-coord 3.202 + "Get the uv-cooridnates of the triangle's verticies." 3.203 + [#^Mesh mesh width height triangle-index] 3.204 + (map (fn [[u v]] (vector (* width u) (* height v))) 3.205 + (map (partial vertex-UV-coord mesh) 3.206 + (triangle-vertex-indices mesh triangle-index)))) 3.207 3.208 (defn same-side? 3.209 "Given the points p1 and p2 and the reference point ref, is point p 3.210 @@ -496,11 +506,26 @@ 3.211 (.cross (.subtract p2 p1) (.subtract p p1)) 3.212 (.cross (.subtract p2 p1) (.subtract ref p1))))) 3.213 3.214 +(defn triangle-seq [#^Triangle tri] 3.215 + [(.get1 tri) (.get2 tri) (.get3 tri)]) 3.216 + 3.217 +(defn vector3f-seq [#^Vector3f v] 3.218 + [(.getX v) (.getY v) (.getZ v)]) 3.219 + 3.220 +(defn inside-triangle? 3.221 + "Is the point inside the triangle?" 3.222 + {:author "Dylan Holmes"} 3.223 + [#^Triangle tri #^Vector3f p] 3.224 + (let [[vert-1 vert-2 vert-3] (triangle-seq tri)] 3.225 + (and 3.226 + (same-side? vert-1 vert-2 vert-3 p) 3.227 + (same-side? vert-2 vert-3 vert-1 p) 3.228 + (same-side? vert-3 vert-1 vert-2 p)))) 3.229 + 3.230 (defn triangle->matrix4f 3.231 - "Converts the triangle into a 4x4 matrix of vertices: The first 3.232 - three columns contain the vertices of the triangle; the last 3.233 - contains the unit normal of the triangle. The bottom row is filled 3.234 - with 1s." 3.235 + "Converts the triangle into a 4x4 matrix: The first three columns 3.236 + contain the vertices of the triangle; the last contains the unit 3.237 + normal of the triangle. The bottom row is filled with 1s." 3.238 [#^Triangle t] 3.239 (let [mat (Matrix4f.) 3.240 [vert-1 vert-2 vert-3] 3.241 @@ -523,150 +548,85 @@ 3.242 (triangle->matrix4f tri-2) 3.243 (.invert (triangle->matrix4f tri-1)))) 3.244 3.245 -(def death (Triangle. 3.246 - (Vector3f. 1 1 1) 3.247 - (Vector3f. 1 2 3) 3.248 - (Vector3f. 5 6 7))) 3.249 - 3.250 -(def death-2 (Triangle. 3.251 - (Vector3f. 2 2 2) 3.252 - (Vector3f. 1 1 1) 3.253 - (Vector3f. 0 1 0))) 3.254 +(defn point->vector2f [[u v]] 3.255 + (Vector2f. u v)) 3.256 3.257 (defn vector2f->vector3f [v] 3.258 (Vector3f. (.getX v) (.getY v) 0)) 3.259 3.260 -(extend-type Triangle 3.261 - Textual 3.262 - (text [t] 3.263 - (println "Triangle: " \newline (.get1 t) \newline 3.264 - (.get2 t) \newline (.get3 t)))) 3.265 - 3.266 (defn map-triangle [f #^Triangle tri] 3.267 (Triangle. 3.268 (f 0 (.get1 tri)) 3.269 (f 1 (.get2 tri)) 3.270 (f 2 (.get3 tri)))) 3.271 3.272 -(defn triangle-seq [#^Triangle tri] 3.273 - [(.get1 tri) (.get2 tri) (.get3 tri)]) 3.274 +(defn points->triangle 3.275 + "Convert a list of points into a triangle." 3.276 + [points] 3.277 + (apply #(Triangle. %1 %2 %3) 3.278 + (map (fn [point] 3.279 + (let [point (vec point)] 3.280 + (Vector3f. (get point 0 0) 3.281 + (get point 1 0) 3.282 + (get point 2 0)))) 3.283 + (take 3 points)))) 3.284 3.285 -(defn vector3f-seq [#^Vector3f v] 3.286 - [(.getX v) (.getY v) (.getZ v)]) 3.287 - 3.288 -(defn inside-triangle? 3.289 - "Is the point inside the triangle? Now what do we do? 3.290 - You might want to hold on there" 3.291 - {:author "Dylan Holmes"} 3.292 - [tri p] 3.293 - (let [[vert-1 vert-2 vert-3] (triangle-seq tri)] 3.294 - (and 3.295 - (same-side? vert-1 vert-2 vert-3 p) 3.296 - (same-side? vert-2 vert-3 vert-1 p) 3.297 - (same-side? vert-3 vert-1 vert-2 p)))) 3.298 - 3.299 -(defn uv-triangle 3.300 - "Convert the mesh triangle into the cooresponding triangle in 3.301 - UV-space. Z-component of these triangles is always zero." 3.302 - [#^Mesh mesh #^Triangle tri] 3.303 - (apply #(Triangle. %1 %2 %3) 3.304 - (map vector2f->vector3f 3.305 - (tri-uv-coord mesh tri)))) 3.306 - 3.307 -(defn pixel-triangle 3.308 - "Convert the mesh triangle into the corresponding triangle in 3.309 - UV-pixel-space. Z compenent will be zero." 3.310 - [#^Mesh mesh #^Triangle tri width height] 3.311 - (map-triangle (fn [_ v] 3.312 - (Vector3f. (* width (.getX v)) 3.313 - (* height (.getY v)) 3.314 - 0)) 3.315 - (uv-triangle mesh tri))) 3.316 - 3.317 -(def rasterize pixel-triangle) 3.318 - 3.319 - 3.320 -(defn triangle-bounds 3.321 - "Dimensions of the bounding square of the triangle in the form 3.322 - [x y width height]. 3.323 - Assumes that the triangle lies in the XY plane." 3.324 - [#^Triangle tri] 3.325 - (let [verts (map vector3f-seq (triangle-seq tri)) 3.326 - x (apply min (map first verts)) 3.327 - y (apply min (map second verts))] 3.328 - [x y 3.329 - (- (apply max (map first verts)) x) 3.330 - (- (apply max (map second verts)) y) 3.331 - ])) 3.332 - 3.333 +(defn convex-bounds 3.334 + "Dimensions of the smallest integer bounding square of the list of 3.335 + 2D verticies in the form: [x y width height]." 3.336 + [uv-verts] 3.337 + (let [xs (map first uv-verts) 3.338 + ys (map second uv-verts) 3.339 + x0 (Math/floor (apply min xs)) 3.340 + y0 (Math/floor (apply min ys)) 3.341 + x1 (Math/ceil (apply max xs)) 3.342 + y1 (Math/ceil (apply max ys))] 3.343 + [x0 y0 (- x1 x0) (- y1 y0)])) 3.344 3.345 (defn sensors-in-triangle 3.346 "Find the locations of the touch sensors within a triangle in both 3.347 UV and gemoetry relative coordinates." 3.348 [image mesh tri-index] 3.349 (let [width (.getWidth image) 3.350 - height (.getHeight image)] 3.351 + height (.getHeight image) 3.352 + UV-vertex-coords (triangle-UV-coord mesh width height tri-index) 3.353 + bounds (convex-bounds UV-vertex-coords) 3.354 + 3.355 + cutout-triangle (points->triangle UV-vertex-coords) 3.356 + UV-sensor-coords 3.357 + (filter (comp (partial inside-triangle? cutout-triangle) 3.358 + (fn [[u v]] (Vector3f. u v 0))) 3.359 + (white-coordinates image bounds)) 3.360 + UV->geometry (triangle-transformation 3.361 + cutout-triangle 3.362 + (triangle mesh tri-index)) 3.363 + geometry-sensor-coords 3.364 + (map (fn [[u v]] (.mult UV->geometry (Vector3f. u v 0))) 3.365 + UV-sensor-coords)] 3.366 + {:UV UV-sensor-coords :geometry geometry-sensor-coords})) 3.367 3.368 - 3.369 - 3.370 - 3.371 - ) 3.372 - 3.373 - 3.374 -(defn locate-feelers 3.375 +(defn-memo locate-feelers 3.376 "Search the geometry's tactile UV image for touch sensors, returning 3.377 their positions in geometry-relative coordinates." 3.378 [#^Geometry geo] 3.379 - (if-let [image (touch-receptor-image geo)] 3.380 - (let [mesh (.getMesh geo) 3.381 - tris (triangles geo) 3.382 - 3.383 - width (.getWidth image) 3.384 - height (.getHeight image) 3.385 - 3.386 - ;; for each triangle 3.387 - sensor-coords 3.388 - (fn [tri] 3.389 - ;; translate triangle to uv-pixel-space 3.390 - (let [uv-tri 3.391 - (pixel-triangle mesh tri width height) 3.392 - bounds (vec (triangle-bounds uv-tri))] 3.393 - 3.394 - ;; get that part of the picture 3.395 - 3.396 - (apply #(.setRoi image %1 %2 %3 %4) bounds) 3.397 - (let [cutout (.crop (.getProcessor image)) 3.398 - ;; extract white pixels inside triangle 3.399 - cutout-tri 3.400 - (map-triangle 3.401 - (fn [_ v] 3.402 - (.subtract 3.403 - v 3.404 - (Vector3f. (bounds 0) (bounds 1) (float 0)))) 3.405 - uv-tri) 3.406 - whites (filter (partial inside-triangle? cutout-tri) 3.407 - (map vector2f->vector3f 3.408 - (white-coordinates cutout))) 3.409 - ;; translate pixel coordinates to world-space 3.410 - transform (triangle-transformation cutout-tri tri)] 3.411 - (map #(.mult transform %) whites))))] 3.412 - (vec (map sensor-coords tris))) 3.413 - (repeat (count (triangles geo)) []))) 3.414 + (let [mesh (.getMesh geo) 3.415 + num-triangles (.getTriangleCount mesh)] 3.416 + (if-let [image (tactile-sensor-image geo)] 3.417 + (map 3.418 + (partial sensors-in-triangle image mesh) 3.419 + (range num-triangles)) 3.420 + (repeat (.getTriangleCount mesh) {:UV nil :geometry nil})))) 3.421 3.422 (use 'clojure.contrib.def) 3.423 3.424 (defn-memo touch-topology [#^Gemoetry geo] 3.425 - (if-let [image (touch-receptor-image geo)] 3.426 - (let [feeler-coords 3.427 - (map 3.428 - #(vector (int (.getX %)) (int (.getY %))) 3.429 - (white-coordinates 3.430 - (.getProcessor image)))] 3.431 - (vec (collapse feeler-coords))) 3.432 - [])) 3.433 + (vec (collapse (reduce concat (map :UV (locate-feelers geo)))))) 3.434 + 3.435 +(defn-memo feeler-coordinates [#^Geometry geo] 3.436 + (vec (map :geometry (locate-feelers geo)))) 3.437 3.438 (defn enable-touch [#^Geometry geo] 3.439 - (let [feeler-coords (locate-feelers geo) 3.440 + (let [feeler-coords (feeler-coordinates geo) 3.441 tris (triangles geo) 3.442 limit 0.1] 3.443 (fn [node] 3.444 @@ -744,6 +704,7 @@ 3.445 ;;(com.aurellem.capture.Capture/captureVideo 3.446 ;; world (file-str "/home/r/proj/ai-videos/hand")) 3.447 (.setTimer world (RatchetTimer. 60)) 3.448 + ;;(speed-up world) 3.449 ;;(set-gravity world (Vector3f. 0 0 0)) 3.450 ) 3.451 (fn [world tpf]
4.1 --- a/org/util.org Sun Jan 15 00:40:49 2012 -0700 4.2 +++ b/org/util.org Sun Jan 15 04:08:31 2012 -0700 4.3 @@ -453,6 +453,13 @@ 4.4 Textual 4.5 (text [control] 4.6 (println "...geo..."))) 4.7 + 4.8 +(extend-type Triangle 4.9 + Textual 4.10 + (text [t] 4.11 + (println "Triangle: " \newline (.get1 t) \newline 4.12 + (.get2 t) \newline (.get3 t)))) 4.13 + 4.14 #+end_src 4.15 4.16 Here I make the =Viewable= protocol and extend it to JME's types. Now