Mercurial > cortex
changeset 186:21816b27d7c8
moved sense.org to proper position.
author | Robert McIntyre <rlm@mit.edu> |
---|---|
date | Sat, 04 Feb 2012 09:35:15 -0700 (2012-02-04) |
parents | cfb71209ddc6 |
children | 6142e85f5825 |
files | org/sense.org sense.org |
diffstat | 2 files changed, 223 insertions(+), 223 deletions(-) [+] |
line wrap: on
line diff
1.1 --- /dev/null Thu Jan 01 00:00:00 1970 +0000 1.2 +++ b/org/sense.org Sat Feb 04 09:35:15 2012 -0700 1.3 @@ -0,0 +1,223 @@ 1.4 +#+title: General sense/effector utilities 1.5 +#+author: Robert McIntyre 1.6 +#+email: rlm@mit.edu 1.7 +#+description: sensory utilities 1.8 +#+keywords: simulation, jMonkeyEngine3, clojure, simulated senses 1.9 +#+SETUPFILE: ../../aurellem/org/setup.org 1.10 +#+INCLUDE: ../../aurellem/org/level-0.org 1.11 + 1.12 +#+name: sense 1.13 +#+begin_src clojure 1.14 +(ns cortex.sense 1.15 + "Here are functions useful in the construction of two or more 1.16 + sensors/effectors." 1.17 + {:author "Robert McInytre"} 1.18 + (:use (cortex world util)) 1.19 + (:import ij.process.ImageProcessor) 1.20 + (:import jme3tools.converters.ImageToAwt) 1.21 + (:import java.awt.image.BufferedImage) 1.22 + (:import com.jme3.collision.CollisionResults) 1.23 + (:import com.jme3.bounding.BoundingBox) 1.24 + (:import (com.jme3.scene Node Spatial)) 1.25 + (:import com.jme3.scene.control.AbstractControl) 1.26 + (:import (com.jme3.math Quaternion Vector3f))) 1.27 + 1.28 +(defn meta-data 1.29 + "Get the meta-data for a node created with blender." 1.30 + [blender-node key] 1.31 + (if-let [data (.getUserData blender-node "properties")] 1.32 + (.findValue data key) 1.33 + nil)) 1.34 + 1.35 +(defn closest-node 1.36 + "Return the node in creature which is closest to the given node." 1.37 + [#^Node creature #^Node eye] 1.38 + (loop [radius (float 0.01)] 1.39 + (let [results (CollisionResults.)] 1.40 + (.collideWith 1.41 + creature 1.42 + (BoundingBox. (.getWorldTranslation eye) 1.43 + radius radius radius) 1.44 + results) 1.45 + (if-let [target (first results)] 1.46 + (.getGeometry target) 1.47 + (recur (float (* 2 radius))))))) 1.48 + 1.49 +(defn bind-sense 1.50 + "Bind the sense to the Spatial such that it will maintain its 1.51 + current position relative to the Spatial no matter how the spatial 1.52 + moves. 'sense can be either a Camera or Listener object." 1.53 + [#^Spatial obj sense] 1.54 + (let [sense-offset (.subtract (.getLocation sense) 1.55 + (.getWorldTranslation obj)) 1.56 + initial-sense-rotation (Quaternion. (.getRotation sense)) 1.57 + base-anti-rotation (.inverse (.getWorldRotation obj))] 1.58 + (.addControl 1.59 + obj 1.60 + (proxy [AbstractControl] [] 1.61 + (controlUpdate [tpf] 1.62 + (let [total-rotation 1.63 + (.mult base-anti-rotation (.getWorldRotation obj))] 1.64 + (.setLocation 1.65 + sense 1.66 + (.add 1.67 + (.mult total-rotation sense-offset) 1.68 + (.getWorldTranslation obj))) 1.69 + (.setRotation 1.70 + sense 1.71 + (.mult total-rotation initial-sense-rotation)))) 1.72 + (controlRender [_ _]))))) 1.73 + 1.74 +(def white 0xFFFFFF) 1.75 + 1.76 +(defn white? [rgb] 1.77 + (= (bit-and white rgb) white)) 1.78 + 1.79 +(defn filter-pixels 1.80 + "List the coordinates of all pixels matching pred, within the bounds 1.81 + provided. 1.82 + bounds -> [x0 y0 width height]" 1.83 + {:author "Dylan Holmes"} 1.84 + ([pred #^BufferedImage image] 1.85 + (filter-pixels pred image [0 0 (.getWidth image) (.getHeight image)])) 1.86 + ([pred #^BufferedImage image [x0 y0 width height]] 1.87 + ((fn accumulate [x y matches] 1.88 + (cond 1.89 + (>= y (+ height y0)) matches 1.90 + (>= x (+ width x0)) (recur 0 (inc y) matches) 1.91 + (pred (.getRGB image x y)) 1.92 + (recur (inc x) y (conj matches [x y])) 1.93 + :else (recur (inc x) y matches))) 1.94 + x0 y0 []))) 1.95 + 1.96 +(defn white-coordinates 1.97 + "Coordinates of all the white pixels in a subset of the image." 1.98 + ([#^BufferedImage image bounds] 1.99 + (filter-pixels white? image bounds)) 1.100 + ([#^BufferedImage image] 1.101 + (filter-pixels white? image))) 1.102 + 1.103 +(defn points->image 1.104 + "Take a sparse collection of points and visuliaze it as a 1.105 + BufferedImage." 1.106 + [points] 1.107 + (if (empty? points) 1.108 + (BufferedImage. 1 1 BufferedImage/TYPE_BYTE_BINARY) 1.109 + (let [xs (vec (map first points)) 1.110 + ys (vec (map second points)) 1.111 + x0 (apply min xs) 1.112 + y0 (apply min ys) 1.113 + width (- (apply max xs) x0) 1.114 + height (- (apply max ys) y0) 1.115 + image (BufferedImage. (inc width) (inc height) 1.116 + BufferedImage/TYPE_INT_RGB)] 1.117 + (dorun 1.118 + (for [x (range (.getWidth image)) 1.119 + y (range (.getHeight image))] 1.120 + (.setRGB image x y 0xFF0000))) 1.121 + (dorun 1.122 + (for [index (range (count points))] 1.123 + (.setRGB image (- (xs index) x0) (- (ys index) y0) -1))) 1.124 + image))) 1.125 + 1.126 +(defn average [coll] 1.127 + (/ (reduce + coll) (count coll))) 1.128 + 1.129 +(defn collapse-1d 1.130 + "One dimensional analogue of collapse." 1.131 + [center line] 1.132 + (let [length (count line) 1.133 + num-above (count (filter (partial < center) line)) 1.134 + num-below (- length num-above)] 1.135 + (range (- center num-below) 1.136 + (+ center num-above)))) 1.137 + 1.138 +(defn collapse 1.139 + "Take a set of pairs of integers and collapse them into a 1.140 + contigous bitmap with no \"holes\"." 1.141 + [points] 1.142 + (if (empty? points) [] 1.143 + (let 1.144 + [num-points (count points) 1.145 + center (vector 1.146 + (int (average (map first points))) 1.147 + (int (average (map first points)))) 1.148 + flattened 1.149 + (reduce 1.150 + concat 1.151 + (map 1.152 + (fn [column] 1.153 + (map vector 1.154 + (map first column) 1.155 + (collapse-1d (second center) 1.156 + (map second column)))) 1.157 + (partition-by first (sort-by first points)))) 1.158 + squeezed 1.159 + (reduce 1.160 + concat 1.161 + (map 1.162 + (fn [row] 1.163 + (map vector 1.164 + (collapse-1d (first center) 1.165 + (map first row)) 1.166 + (map second row))) 1.167 + (partition-by second (sort-by second flattened)))) 1.168 + relocated 1.169 + (let [min-x (apply min (map first squeezed)) 1.170 + min-y (apply min (map second squeezed))] 1.171 + (map (fn [[x y]] 1.172 + [(- x min-x) 1.173 + (- y min-y)]) 1.174 + squeezed))] 1.175 + relocated))) 1.176 + 1.177 +(defn world-to-local 1.178 + "Convert the world coordinates into coordinates relative to the 1.179 + object (i.e. local coordinates), taking into account the rotation 1.180 + of object." 1.181 + [#^Spatial object world-coordinate] 1.182 + (.worldToLocal object world-coordinate nil)) 1.183 + 1.184 +(defn local-to-world 1.185 + "Convert the local coordinates into world relative coordinates" 1.186 + [#^Spatial object local-coordinate] 1.187 + (.localToWorld object local-coordinate nil)) 1.188 + 1.189 +(defn sense-nodes 1.190 + "For each sense there is a special blender node whose children are 1.191 + considered markers for an instance of a that sense. This function 1.192 + generates functions to find those children, given the name of the 1.193 + special parent node." 1.194 + [parent-name] 1.195 + (fn [#^Node creature] 1.196 + (if-let [sense-node (.getChild creature parent-name)] 1.197 + (seq (.getChildren sense-node)) 1.198 + (do (println-repl "could not find" parent-name "node") [])))) 1.199 + 1.200 +(defn load-image 1.201 + "Load an image as a BufferedImage using the asset-manager system." 1.202 + [asset-relative-path] 1.203 + (ImageToAwt/convert 1.204 + (.getImage (.loadTexture (asset-manager) asset-relative-path)) 1.205 + false false 0)) 1.206 + 1.207 +(defn jme-to-blender 1.208 + "Convert from JME coordinates to Blender coordinates" 1.209 + [#^Vector3f in] 1.210 + (Vector3f. (.getX in) 1.211 + (- (.getZ in)) 1.212 + (.getY in))) 1.213 + 1.214 +(defn blender-to-jme 1.215 + "Convert from Blender coordinates to JME coordinates" 1.216 + [#^Vector3f in] 1.217 + (Vector3f. (.getX in) 1.218 + (.getZ in) 1.219 + (- (.getY in)))) 1.220 +#+end_src 1.221 + 1.222 + 1.223 +* COMMENT generate source 1.224 +#+begin_src clojure :tangle ../src/cortex/sense.clj 1.225 +<<sense>> 1.226 +#+end_src
2.1 --- a/sense.org Sat Feb 04 09:33:13 2012 -0700 2.2 +++ /dev/null Thu Jan 01 00:00:00 1970 +0000 2.3 @@ -1,223 +0,0 @@ 2.4 -#+title: General sense/effector utilities 2.5 -#+author: Robert McIntyre 2.6 -#+email: rlm@mit.edu 2.7 -#+description: sensory utilities 2.8 -#+keywords: simulation, jMonkeyEngine3, clojure, simulated senses 2.9 -#+SETUPFILE: ../../aurellem/org/setup.org 2.10 -#+INCLUDE: ../../aurellem/org/level-0.org 2.11 - 2.12 -#+name: sense 2.13 -#+begin_src clojure 2.14 -(ns cortex.sense 2.15 - "Here are functions useful in the construction of two or more 2.16 - sensors/effectors." 2.17 - {:author "Robert McInytre"} 2.18 - (:use (cortex world util)) 2.19 - (:import ij.process.ImageProcessor) 2.20 - (:import jme3tools.converters.ImageToAwt) 2.21 - (:import java.awt.image.BufferedImage) 2.22 - (:import com.jme3.collision.CollisionResults) 2.23 - (:import com.jme3.bounding.BoundingBox) 2.24 - (:import (com.jme3.scene Node Spatial)) 2.25 - (:import com.jme3.scene.control.AbstractControl) 2.26 - (:import (com.jme3.math Quaternion Vector3f))) 2.27 - 2.28 -(defn meta-data 2.29 - "Get the meta-data for a node created with blender." 2.30 - [blender-node key] 2.31 - (if-let [data (.getUserData blender-node "properties")] 2.32 - (.findValue data key) 2.33 - nil)) 2.34 - 2.35 -(defn closest-node 2.36 - "Return the node in creature which is closest to the given node." 2.37 - [#^Node creature #^Node eye] 2.38 - (loop [radius (float 0.01)] 2.39 - (let [results (CollisionResults.)] 2.40 - (.collideWith 2.41 - creature 2.42 - (BoundingBox. (.getWorldTranslation eye) 2.43 - radius radius radius) 2.44 - results) 2.45 - (if-let [target (first results)] 2.46 - (.getGeometry target) 2.47 - (recur (float (* 2 radius))))))) 2.48 - 2.49 -(defn bind-sense 2.50 - "Bind the sense to the Spatial such that it will maintain its 2.51 - current position relative to the Spatial no matter how the spatial 2.52 - moves. 'sense can be either a Camera or Listener object." 2.53 - [#^Spatial obj sense] 2.54 - (let [sense-offset (.subtract (.getLocation sense) 2.55 - (.getWorldTranslation obj)) 2.56 - initial-sense-rotation (Quaternion. (.getRotation sense)) 2.57 - base-anti-rotation (.inverse (.getWorldRotation obj))] 2.58 - (.addControl 2.59 - obj 2.60 - (proxy [AbstractControl] [] 2.61 - (controlUpdate [tpf] 2.62 - (let [total-rotation 2.63 - (.mult base-anti-rotation (.getWorldRotation obj))] 2.64 - (.setLocation 2.65 - sense 2.66 - (.add 2.67 - (.mult total-rotation sense-offset) 2.68 - (.getWorldTranslation obj))) 2.69 - (.setRotation 2.70 - sense 2.71 - (.mult total-rotation initial-sense-rotation)))) 2.72 - (controlRender [_ _]))))) 2.73 - 2.74 -(def white 0xFFFFFF) 2.75 - 2.76 -(defn white? [rgb] 2.77 - (= (bit-and white rgb) white)) 2.78 - 2.79 -(defn filter-pixels 2.80 - "List the coordinates of all pixels matching pred, within the bounds 2.81 - provided. 2.82 - bounds -> [x0 y0 width height]" 2.83 - {:author "Dylan Holmes"} 2.84 - ([pred #^BufferedImage image] 2.85 - (filter-pixels pred image [0 0 (.getWidth image) (.getHeight image)])) 2.86 - ([pred #^BufferedImage image [x0 y0 width height]] 2.87 - ((fn accumulate [x y matches] 2.88 - (cond 2.89 - (>= y (+ height y0)) matches 2.90 - (>= x (+ width x0)) (recur 0 (inc y) matches) 2.91 - (pred (.getRGB image x y)) 2.92 - (recur (inc x) y (conj matches [x y])) 2.93 - :else (recur (inc x) y matches))) 2.94 - x0 y0 []))) 2.95 - 2.96 -(defn white-coordinates 2.97 - "Coordinates of all the white pixels in a subset of the image." 2.98 - ([#^BufferedImage image bounds] 2.99 - (filter-pixels white? image bounds)) 2.100 - ([#^BufferedImage image] 2.101 - (filter-pixels white? image))) 2.102 - 2.103 -(defn points->image 2.104 - "Take a sparse collection of points and visuliaze it as a 2.105 - BufferedImage." 2.106 - [points] 2.107 - (if (empty? points) 2.108 - (BufferedImage. 1 1 BufferedImage/TYPE_BYTE_BINARY) 2.109 - (let [xs (vec (map first points)) 2.110 - ys (vec (map second points)) 2.111 - x0 (apply min xs) 2.112 - y0 (apply min ys) 2.113 - width (- (apply max xs) x0) 2.114 - height (- (apply max ys) y0) 2.115 - image (BufferedImage. (inc width) (inc height) 2.116 - BufferedImage/TYPE_INT_RGB)] 2.117 - (dorun 2.118 - (for [x (range (.getWidth image)) 2.119 - y (range (.getHeight image))] 2.120 - (.setRGB image x y 0xFF0000))) 2.121 - (dorun 2.122 - (for [index (range (count points))] 2.123 - (.setRGB image (- (xs index) x0) (- (ys index) y0) -1))) 2.124 - image))) 2.125 - 2.126 -(defn average [coll] 2.127 - (/ (reduce + coll) (count coll))) 2.128 - 2.129 -(defn collapse-1d 2.130 - "One dimensional analogue of collapse." 2.131 - [center line] 2.132 - (let [length (count line) 2.133 - num-above (count (filter (partial < center) line)) 2.134 - num-below (- length num-above)] 2.135 - (range (- center num-below) 2.136 - (+ center num-above)))) 2.137 - 2.138 -(defn collapse 2.139 - "Take a set of pairs of integers and collapse them into a 2.140 - contigous bitmap with no \"holes\"." 2.141 - [points] 2.142 - (if (empty? points) [] 2.143 - (let 2.144 - [num-points (count points) 2.145 - center (vector 2.146 - (int (average (map first points))) 2.147 - (int (average (map first points)))) 2.148 - flattened 2.149 - (reduce 2.150 - concat 2.151 - (map 2.152 - (fn [column] 2.153 - (map vector 2.154 - (map first column) 2.155 - (collapse-1d (second center) 2.156 - (map second column)))) 2.157 - (partition-by first (sort-by first points)))) 2.158 - squeezed 2.159 - (reduce 2.160 - concat 2.161 - (map 2.162 - (fn [row] 2.163 - (map vector 2.164 - (collapse-1d (first center) 2.165 - (map first row)) 2.166 - (map second row))) 2.167 - (partition-by second (sort-by second flattened)))) 2.168 - relocated 2.169 - (let [min-x (apply min (map first squeezed)) 2.170 - min-y (apply min (map second squeezed))] 2.171 - (map (fn [[x y]] 2.172 - [(- x min-x) 2.173 - (- y min-y)]) 2.174 - squeezed))] 2.175 - relocated))) 2.176 - 2.177 -(defn world-to-local 2.178 - "Convert the world coordinates into coordinates relative to the 2.179 - object (i.e. local coordinates), taking into account the rotation 2.180 - of object." 2.181 - [#^Spatial object world-coordinate] 2.182 - (.worldToLocal object world-coordinate nil)) 2.183 - 2.184 -(defn local-to-world 2.185 - "Convert the local coordinates into world relative coordinates" 2.186 - [#^Spatial object local-coordinate] 2.187 - (.localToWorld object local-coordinate nil)) 2.188 - 2.189 -(defn sense-nodes 2.190 - "For each sense there is a special blender node whose children are 2.191 - considered markers for an instance of a that sense. This function 2.192 - generates functions to find those children, given the name of the 2.193 - special parent node." 2.194 - [parent-name] 2.195 - (fn [#^Node creature] 2.196 - (if-let [sense-node (.getChild creature parent-name)] 2.197 - (seq (.getChildren sense-node)) 2.198 - (do (println-repl "could not find" parent-name "node") [])))) 2.199 - 2.200 -(defn load-image 2.201 - "Load an image as a BufferedImage using the asset-manager system." 2.202 - [asset-relative-path] 2.203 - (ImageToAwt/convert 2.204 - (.getImage (.loadTexture (asset-manager) asset-relative-path)) 2.205 - false false 0)) 2.206 - 2.207 -(defn jme-to-blender 2.208 - "Convert from JME coordinates to Blender coordinates" 2.209 - [#^Vector3f in] 2.210 - (Vector3f. (.getX in) 2.211 - (- (.getZ in)) 2.212 - (.getY in))) 2.213 - 2.214 -(defn blender-to-jme 2.215 - "Convert from Blender coordinates to JME coordinates" 2.216 - [#^Vector3f in] 2.217 - (Vector3f. (.getX in) 2.218 - (.getZ in) 2.219 - (- (.getY in)))) 2.220 -#+end_src 2.221 - 2.222 - 2.223 -* COMMENT generate source 2.224 -#+begin_src clojure :tangle ../src/cortex/sense.clj 2.225 -<<sense>> 2.226 -#+end_src