changeset 25:775d97247dd0

cleaning up world.org
author Robert McIntyre <rlm@mit.edu>
date Mon, 24 Oct 2011 05:25:01 -0700 (2011-10-24)
parents e965675ec4d0
children bbffa41a12a9
files org/cortex.org org/eyes.org org/util.org org/world.org
diffstat 4 files changed, 410 insertions(+), 483 deletions(-) [+]
line wrap: on
line diff
     1.1 --- a/org/cortex.org	Mon Oct 24 01:19:15 2011 -0700
     1.2 +++ b/org/cortex.org	Mon Oct 24 05:25:01 2011 -0700
     1.3 @@ -78,6 +78,7 @@
     1.4  (cortex.import/mega-import-jme3)
     1.5  (use '[pokemon [lpsolve :only [constant-map]]])
     1.6  (use 'cortex.world)
     1.7 +(use 'cortex.util)
     1.8  #+end_src
     1.9  
    1.10  #+srcname: brick-wall-body
    1.11 @@ -744,250 +745,7 @@
    1.12  
    1.13  
    1.14      
    1.15 -* The Body
    1.16 -** Eyes
    1.17 -
    1.18 -Ultimately I want to make creatures with eyes. Each eye can be
    1.19 -independely moved and should see its own version of the world
    1.20 -depending on where it is.
    1.21 -#+srcname: eyes
    1.22 -#+begin_src clojure 
    1.23 -(ns body.eye)
    1.24 -(use 'cortex.world)
    1.25 -(use 'cortex.import)
    1.26 -(use 'clojure.contrib.def)
    1.27 -(cortex.import/mega-import-jme3)
    1.28 -(rlm.rlm-commands/help)
    1.29 -(import java.nio.ByteBuffer)
    1.30 -(import java.awt.image.BufferedImage)
    1.31 -(import java.awt.Color)
    1.32 -(import java.awt.Dimension)
    1.33 -(import java.awt.Graphics)
    1.34 -(import java.awt.Graphics2D)
    1.35 -(import java.awt.event.WindowAdapter)
    1.36 -(import java.awt.event.WindowEvent)
    1.37 -(import java.awt.image.BufferedImage)
    1.38 -(import java.nio.ByteBuffer)
    1.39 -(import javax.swing.JFrame)
    1.40 -(import javax.swing.JPanel)
    1.41 -(import javax.swing.SwingUtilities)
    1.42 -(import javax.swing.ImageIcon)
    1.43 -(import javax.swing.JOptionPane)
    1.44 -(import java.awt.image.ImageObserver)
    1.45 -
    1.46 -
    1.47 -
    1.48 -(defn scene-processor
    1.49 -  "deals with converting FrameBuffers to BufferedImages so
    1.50 -   that the continuation function can be defined only in terms
    1.51 -   of what it does with BufferedImages"
    1.52 -  [continuation]
    1.53 -  (let [byte-buffer (atom nil)
    1.54 -	renderer (atom nil)
    1.55 -	image (atom nil)]
    1.56 -  (proxy [SceneProcessor] []
    1.57 -    (initialize
    1.58 -     [renderManager viewPort]
    1.59 -     (let [cam (.getCamera viewPort)
    1.60 -	   width (.getWidth cam)
    1.61 -	   height (.getHeight cam)]
    1.62 -       (reset! renderer (.getRenderer renderManager))
    1.63 -       (reset! byte-buffer
    1.64 -	     (BufferUtils/createByteBuffer
    1.65 -	      (* width height 4)))
    1.66 -       (reset! image (BufferedImage. width height
    1.67 -				     BufferedImage/TYPE_4BYTE_ABGR))))
    1.68 -    (isInitialized [] (not (nil? @byte-buffer)))
    1.69 -    (reshape [_ _ _])
    1.70 -    (preFrame [_])
    1.71 -    (postQueue [_])
    1.72 -    (postFrame
    1.73 -     [#^FrameBuffer fb]
    1.74 -     (.clear @byte-buffer)
    1.75 -     (.readFrameBuffer @renderer fb @byte-buffer)
    1.76 -     (Screenshots/convertScreenShot @byte-buffer @image)
    1.77 -     (continuation @image))
    1.78 -    (cleanup []))))
    1.79 -    
    1.80 -(defn add-eye
    1.81 -  "Add an eye to the world, and call continuation on
    1.82 -   every frame produced"
    1.83 -  [world camera continuation]
    1.84 -  (let [width (.getWidth camera)
    1.85 -	height (.getHeight camera)
    1.86 -	render-manager (.getRenderManager world)
    1.87 -	viewport (.createMainView render-manager "eye-view" camera)]
    1.88 -    (doto viewport
    1.89 -      (.setBackgroundColor ColorRGBA/Black)
    1.90 -      (.setClearFlags true true true)
    1.91 -      (.addProcessor (scene-processor continuation))
    1.92 -      (.attachScene (.getRootNode world)))))
    1.93 -
    1.94 -(defn make-display-frame [display width height]
    1.95 -  (SwingUtilities/invokeLater
    1.96 -   (fn []
    1.97 -     (.setPreferredSize display (Dimension. width height))
    1.98 -     (doto (JFrame. "Eye Camera!")
    1.99 -       (-> (.getContentPane) (.add display))
   1.100 -       (.setDefaultCloseOperation JFrame/DISPOSE_ON_CLOSE)
   1.101 -       (.pack)
   1.102 -       (.setLocationRelativeTo nil)
   1.103 -       (.setResizable false)
   1.104 -       (.setVisible true)))))
   1.105 -	   
   1.106 -(defn image-monitor [#^BufferedImage image]
   1.107 -  (proxy [JPanel] []
   1.108 -    (paintComponent 
   1.109 -     [g]
   1.110 -     (proxy-super paintComponent g)
   1.111 -     (locking image
   1.112 -       (.drawImage g image 0 0
   1.113 -		   (proxy [ImageObserver]
   1.114 -		       []
   1.115 -		     (imageUpdate
   1.116 -		      []
   1.117 -		      (proxy-super imageUpdate))))))))
   1.118 -
   1.119 -(defn movie-image []
   1.120 -  (let [setup
   1.121 -	(runonce
   1.122 -	 (fn [#^BufferedImage image]
   1.123 -	   (let [width (.getWidth image)
   1.124 -		 height (.getHeight image)
   1.125 -		 display (image-monitor image)
   1.126 -		 frame (make-display-frame display width height)]
   1.127 -	     display)))]
   1.128 -    (fn [#^BufferedImage image]
   1.129 -      (.repaint (setup image)))))
   1.130 -	     
   1.131 -
   1.132 -(defn observer
   1.133 -  "place thy eye!"
   1.134 -  [world camera]
   1.135 -  (let [eye camera
   1.136 -	width (.getWidth eye)
   1.137 -	height (.getHeight eye)]
   1.138 -    (no-exceptions
   1.139 -     (add-eye
   1.140 -      world
   1.141 -      eye
   1.142 -      (movie-image)))))
   1.143 -#+end_src
   1.144 -
   1.145 -#+srcname: test-vision
   1.146 -#+begin_src clojure
   1.147 -
   1.148 -(ns test.vision)
   1.149 -(use 'cortex.world)
   1.150 -(use 'cortex.import)
   1.151 -(use 'clojure.contrib.def)
   1.152 -(use 'body.eye)
   1.153 -(cortex.import/mega-import-jme3)
   1.154 -(rlm.rlm-commands/help)
   1.155 -(import java.nio.ByteBuffer)
   1.156 -(import java.awt.image.BufferedImage)
   1.157 -(import java.awt.Color)
   1.158 -(import java.awt.Dimension)
   1.159 -(import java.awt.Graphics)
   1.160 -(import java.awt.Graphics2D)
   1.161 -(import java.awt.event.WindowAdapter)
   1.162 -(import java.awt.event.WindowEvent)
   1.163 -(import java.awt.image.BufferedImage)
   1.164 -(import java.nio.ByteBuffer)
   1.165 -(import javax.swing.JFrame)
   1.166 -(import javax.swing.JPanel)
   1.167 -(import javax.swing.SwingUtilities)
   1.168 -(import javax.swing.ImageIcon)
   1.169 -(import javax.swing.JOptionPane)
   1.170 -(import java.awt.image.ImageObserver)
   1.171 -
   1.172 -
   1.173 -(def width 200)
   1.174 -(def height 200)
   1.175 -
   1.176 -(defn camera []
   1.177 -  (doto (Camera. width height)
   1.178 -    (.setFrustumPerspective 45 1 1 1000)
   1.179 -    (.setLocation (Vector3f. -3 0 -5))
   1.180 -    (.lookAt Vector3f/ZERO Vector3f/UNIT_Y)))
   1.181 -
   1.182 -(defn camera2 []
   1.183 -  (doto (Camera. width height)
   1.184 -    (.setFrustumPerspective 45 1 1 1000)
   1.185 -    (.setLocation (Vector3f. 3 0 -5))
   1.186 -    (.lookAt Vector3f/ZERO Vector3f/UNIT_Y)))
   1.187 -
   1.188 -(defn setup-fn [world]
   1.189 -  (let [eye (camera)
   1.190 -	width (.getWidth eye)
   1.191 -	height (.getHeight eye)]
   1.192 -    (no-exceptions
   1.193 -     (add-eye
   1.194 -      world
   1.195 -      eye
   1.196 -      (runonce visual))
   1.197 -     (add-eye
   1.198 -      world
   1.199 -      (camera2)
   1.200 -      (runonce visual)))))
   1.201 -
   1.202 -(defn spider-eye [position]
   1.203 -  (doto (Camera. 200 200 )
   1.204 -    (.setFrustumPerspective 45 1 1 1000)
   1.205 -    (.setLocation position)
   1.206 -    (.lookAt Vector3f/ZERO Vector3f/UNIT_Y)))
   1.207 -  
   1.208 -(defn setup-fn* [world]
   1.209 -  (let [eye (camera)
   1.210 -	width (.getWidth eye)
   1.211 -	height (.getHeight eye)]
   1.212 -    ;;(.setClearFlags (.getViewPort world) true true true)
   1.213 -    (observer world (.getCamera world))
   1.214 -    (observer world (spider-eye (Vector3f.  3  0 -5)))
   1.215 -    ;;(observer world (spider-eye (Vector3f.  0  0 -5)))
   1.216 -    ;; (observer world (spider-eye (Vector3f. -3  0 -5)))
   1.217 -    ;; (observer world (spider-eye (Vector3f.  0  3 -5)))
   1.218 -    ;; (observer world (spider-eye (Vector3f.  0 -3 -5)))
   1.219 -    ;; (observer world (spider-eye (Vector3f.  3  3 -5)))
   1.220 -    ;; (observer world (spider-eye (Vector3f. -3  3 -5)))
   1.221 -    ;; (observer world (spider-eye (Vector3f.  3 -3 -5)))
   1.222 -    ;; (observer world (spider-eye (Vector3f. -3 -3 -5)))
   1.223 -
   1.224 -    )
   1.225 -  world)
   1.226 -
   1.227 -(defn test-world []
   1.228 -  (let [thing (box 1 1 1 :physical? false)]
   1.229 -    (world
   1.230 -     (doto (Node.)
   1.231 -       (.attachChild thing))
   1.232 -     {}
   1.233 -     setup-fn
   1.234 -     (fn [world tpf]
   1.235 -       (.rotate thing (* tpf 0.2) 0 0)
   1.236 -       ))))
   1.237 -
   1.238 -
   1.239 -#+end_src
   1.240 -
   1.241 -
   1.242 -#+results: eyes
   1.243 -: #'body.eye/test-world
   1.244 -
   1.245 -Note the use of continuation passing style for connecting the eye to a
   1.246 -function to process the output. The example code will create two
   1.247 -videos of the same rotating cube from different angles, sutiable for
   1.248 -stereoscopic vision.
   1.249 -
   1.250 -
   1.251 -
   1.252 -
   1.253 -
   1.254 -
   1.255  * COMMENT code generation
   1.256 -#+begin_src clojure :tangle ../src/cortex/import.clj
   1.257 -<<import>>
   1.258 -#+end_src
   1.259  
   1.260  #+begin_src clojure :tangle ../src/hello/brick_wall.clj
   1.261  <<brick-wall-header>>
   1.262 @@ -997,13 +755,6 @@
   1.263  #+begin_src clojure :tangle ../src/hello/hello_simple_app.clj
   1.264  <<hello-simple-app>>
   1.265  #+end_src
   1.266 -  
   1.267 -#+begin_src clojure :tangle ../src/cortex/world.clj
   1.268 -<<world-inputs>>
   1.269 -<<world>>
   1.270 -<<world-shapes>>
   1.271 -<<world-view>>
   1.272 -#+end_src
   1.273  
   1.274  #+begin_src clojure :tangle ../src/cortex/other_games.clj
   1.275  <<other-games>>
   1.276 @@ -1029,13 +780,6 @@
   1.277  <<material>>
   1.278  #+end_src
   1.279  
   1.280 -#+begin_src clojure :tangle ../src/body/eye.clj
   1.281 -<<eyes>>
   1.282 -#+end_src
   1.283 -
   1.284 -#+begin_src clojure :tangle ../src/test/vision.clj
   1.285 -<<test-vision>>
   1.286 -#+end_src
   1.287  
   1.288    
   1.289  
     2.1 --- a/org/eyes.org	Mon Oct 24 01:19:15 2011 -0700
     2.2 +++ b/org/eyes.org	Mon Oct 24 05:25:01 2011 -0700
     2.3 @@ -38,8 +38,6 @@
     2.4  (import javax.swing.JOptionPane)
     2.5  (import java.awt.image.ImageObserver)
     2.6  
     2.7 -
     2.8 -
     2.9  (defn scene-processor
    2.10    "deals with converting FrameBuffers to BufferedImages so
    2.11     that the continuation function can be defined only in terms
     3.1 --- a/org/util.org	Mon Oct 24 01:19:15 2011 -0700
     3.2 +++ b/org/util.org	Mon Oct 24 05:25:01 2011 -0700
     3.3 @@ -61,11 +61,13 @@
     3.4  #+results:
     3.5  : 955
     3.6  
     3.7 -** Simplification
     3.8 +
     3.9  #+srcname: world-view
    3.10  #+begin_src clojure :results silent
    3.11 -(in-ns 'cortex.debug)
    3.12 -
    3.13 +(ns cortex.util)
    3.14 +(require 'cortex.import)
    3.15 +(cortex.import/mega-import-jme3)
    3.16 +(use 'cortex.world)
    3.17  (defprotocol Viewable
    3.18    (view [something]))
    3.19  
    3.20 @@ -94,12 +96,6 @@
    3.21  			      (.setColor ColorRGBA/White))]
    3.22  		    (.addLight (.getRootNode world) sun)))
    3.23  		no-op))))
    3.24 -
    3.25 -(defn position-camera [game]
    3.26 -  (doto (.getCamera game)
    3.27 -    (.setLocation (Vector3f. 0 6 6))
    3.28 -    (.lookAt Vector3f/ZERO (Vector3f. 0 1 0))))
    3.29 -
    3.30  #+end_src
    3.31  
    3.32  Here I make the =Viewable= protocol and extend it to JME's types.  Now
    3.33 @@ -110,8 +106,131 @@
    3.34  #+end_src
    3.35  
    3.36  
    3.37 +#+srcname: util
    3.38 +#+begin_src clojure 
    3.39 +(in-ns 'cortex.util)
    3.40 +
    3.41 +(def println-repl (bound-fn [& args] (apply println args)))
    3.42 +
    3.43 +(defn position-camera [game]
    3.44 +  (doto (.getCamera game)
    3.45 +    (.setLocation (Vector3f. 0 6 6))
    3.46 +    (.lookAt Vector3f/ZERO (Vector3f. 0 1 0))))
    3.47 +
    3.48 +(defn set-gravity*
    3.49 +  [game gravity]
    3.50 +  (traverse
    3.51 +   (fn [geom]
    3.52 +     (if-let
    3.53 +	 [control (.getControl geom RigidBodyControl)]
    3.54 +       (do
    3.55 +	 (.setGravity control gravity)
    3.56 +	 (.applyImpulse control Vector3f/ZERO Vector3f/ZERO)
    3.57 +	 )))
    3.58 +   (.getRootNode game)))
    3.59 +#+end_src
    3.60 +
    3.61 +
    3.62 +#+srcname: shapes
    3.63 +#+begin_src clojure :results silent
    3.64 +(in-ns 'cortex.util)
    3.65 +(defrecord shape-description
    3.66 +  [name
    3.67 +   color
    3.68 +   mass
    3.69 +   friction
    3.70 +   texture
    3.71 +   material
    3.72 +   position
    3.73 +   rotation
    3.74 +   shape
    3.75 +   physical?])
    3.76 +
    3.77 +(def base-shape
    3.78 +     (shape-description.
    3.79 +      "default-shape"
    3.80 +      false
    3.81 +      ;;ColorRGBA/Blue
    3.82 +      1.0 ;; mass
    3.83 +      1.0 ;; friction
    3.84 +      ;; texture
    3.85 +      "Textures/Terrain/BrickWall/BrickWall.jpg"
    3.86 +      ;; material
    3.87 +      "Common/MatDefs/Misc/Unshaded.j3md"
    3.88 +      Vector3f/ZERO
    3.89 +      Quaternion/IDENTITY
    3.90 +      (Box. Vector3f/ZERO 0.5 0.5 0.5)
    3.91 +      true))
    3.92 +
    3.93 +(defn make-shape
    3.94 +  [#^shape-description d]
    3.95 +  (let [asset-manager (if (:asset-manager d) (:asset-manager d) (asset-manager))
    3.96 +        mat (Material. asset-manager (:material d))
    3.97 +	geom (Geometry. (:name d) (:shape d))]
    3.98 +    (if (:texture d)
    3.99 +      (let [key (TextureKey. (:texture d))]
   3.100 +	(.setGenerateMips key true)
   3.101 +	(.setTexture mat "ColorMap" (.loadTexture asset-manager key))))
   3.102 +    (if (:color d) (.setColor mat "Color" (:color d)))
   3.103 +    (.setMaterial geom mat)
   3.104 +    (if-let [rotation (:rotation d)] (.rotate geom rotation))
   3.105 +    (.setLocalTranslation geom (:position d))
   3.106 +    (if (:physical? d)
   3.107 +      (let [impact-shape (doto (GImpactCollisionShape. 
   3.108 +                                (.getMesh geom)) (.setMargin 0))
   3.109 +	    physics-control (RigidBodyControl.
   3.110 +			     ;;impact-shape ;; comment to disable 
   3.111 +			     (float (:mass d)))]
   3.112 +	(.createJmeMesh impact-shape)
   3.113 +	(.addControl geom physics-control)
   3.114 +	;;(.setSleepingThresholds physics-control (float 0) (float 0))
   3.115 +	(.setFriction physics-control (:friction d))))
   3.116 +    ;;the default is to keep this node in the physics engine forever.
   3.117 +    ;;these commands must come after the control is added to the geometry.
   3.118 +    ;;
   3.119 +    geom))
   3.120 +  
   3.121 +(defn box
   3.122 +  ([l w h & {:as options}]
   3.123 +     (let [options (merge base-shape options)]
   3.124 +       (make-shape (assoc options
   3.125 +		     :shape (Box. l w h)))))
   3.126 +  ([] (box 0.5 0.5 0.5)))
   3.127 +
   3.128 +(defn sphere
   3.129 +  ([r & {:as options}]
   3.130 +     (let [options (merge base-shape options)]
   3.131 +       (make-shape (assoc options
   3.132 +		     :shape (Sphere. 32 32 (float r)))))) 
   3.133 +  ([] (sphere 0.5)))
   3.134 +
   3.135 +(defn add-element
   3.136 +  ([game element node]
   3.137 +  (.addAll
   3.138 +   (.getPhysicsSpace
   3.139 +    (.getState
   3.140 +     (.getStateManager game)
   3.141 +     BulletAppState))
   3.142 +    element)
   3.143 +  (.attachChild node element))
   3.144 +  ([game element]
   3.145 +     (add-element game element (.getRootNode game))))
   3.146 +
   3.147 +
   3.148 +
   3.149 +#+end_src
   3.150 +
   3.151 +
   3.152  
   3.153  * COMMENT code generation
   3.154  #+begin_src clojure :tangle ../src/cortex/import.clj
   3.155  <<import>>
   3.156  #+end_src
   3.157 +
   3.158 +
   3.159 +#+begin_src clojure :tangle ../src/cortex/util.clj
   3.160 +<<world-view>>
   3.161 +<<util>>
   3.162 +<<shapes>>
   3.163 +#+end_src
   3.164 +
     4.1 --- a/org/world.org	Mon Oct 24 01:19:15 2011 -0700
     4.2 +++ b/org/world.org	Mon Oct 24 05:25:01 2011 -0700
     4.3 @@ -1,26 +1,83 @@
     4.4 -#+title: A world for the creatures to live
     4.5 +#+title: A World for the Creatures
     4.6  #+author: Robert McIntyre
     4.7  #+email: rlm@mit.edu
     4.8 -#+description: Simulating senses for AI research using JMonkeyEngine3
     4.9 +#+description: Creating a Virtual World for AI constructs using clojure and JME3
    4.10 +#+keywords: JME3, clojure, virtual world, exception handling
    4.11  #+SETUPFILE: ../../aurellem/org/setup.org
    4.12  #+INCLUDE: ../../aurellem/org/level-0.org
    4.13  #+babel: :mkdirp yes :noweb yes :exports both
    4.14  
    4.15 +* The  World
    4.16  
    4.17 +There's no point in having senses if there's nothing to experience. In
    4.18 +this section I make some tools with which to build virtual worlds for
    4.19 +my characters to inhabit. If you look at the tutorials at [[http://www.jmonkeyengine.org/wiki/doku.php/jme3:beginner][the jme3
    4.20 +website]], you will see a pattern in how virtual worlds are normally
    4.21 +built. I call this "the Java way" of making worlds.
    4.22  
    4.23 -*** World
    4.24 + - The Java way:  
    4.25 +   - Create a class that extends =SimpleApplication= or =Application=
    4.26 +   - Implement setup functions that create all the scene objects using
    4.27 +     the inherited =assetManager= and call them by overriding the
    4.28 +     =simpleInitApp= method.
    4.29 +   - Create =ActionListeners= and add them to the =inputManager=
    4.30 +     inherited from =Application= to handle key-bindings.
    4.31 +   - Override =simpleUpdate= to implement game logic.
    4.32 +   - Running/Testing an Application involves creating a new JVM,
    4.33 +     running the App, and then closing everything down.
    4.34  
    4.35 -It is comvienent to wrap the JME elements that deal with creating a
    4.36 -world, creation of basic objects, and Keyboard input with a nicer
    4.37 -interface (at least for my purposes).
    4.38  
    4.39 -#+srcname: world-inputs
    4.40 -#+begin_src clojure :results silent
    4.41 -(ns cortex.world)
    4.42 -(require 'cortex.import)
    4.43 -(use 'clojure.contrib.def)
    4.44 -(rlm.rlm-commands/help)
    4.45 -(cortex.import/mega-import-jme3)
    4.46 + - A more Clojureish way:
    4.47 +   - Use a map from keys->functions to specify key-bindings. 
    4.48 +   - Use functions to create objects separately from any particular
    4.49 +     application.
    4.50 +   - Use an REPL -- this means that there's only ever one JVM, and
    4.51 +     Applications come and go.
    4.52 +
    4.53 +Since most development work using jMonkeyEngine is done in Java, jme3
    4.54 +supports "the Java way" quite well out of the box. To work "the
    4.55 +clojure way", it necessary to wrap the jme3 elements that deal with
    4.56 +the Application life-cycle with a REPL driven interface.
    4.57 +
    4.58 +The most important modifications are:
    4.59 +
    4.60 + - Separation of Object life-cycles with the Application life-cycle.
    4.61 + - Functional interface to the underlying =Application= and
    4.62 +   =SimpleApplication= classes.
    4.63 +
    4.64 +** Header
    4.65 +#+srcname: header
    4.66 +#+begin_src clojure :results silent    
    4.67 +(ns cortex.world
    4.68 +  "World Creation, abstracion over jme3's input system, and REPL
    4.69 +  driven exception handling"
    4.70 +  {:author "Robert McIntyre"}
    4.71 +  
    4.72 +  (:use (clojure.contrib (def :only (defvar))))
    4.73 +  (:use [pokemon [lpsolve :only [constant-map]]])
    4.74 +  (:use [clojure.contrib [str-utils :only [re-gsub]]])
    4.75 +
    4.76 +  (:import com.jme3.math.Vector3f)
    4.77 +  (:import com.jme3.scene.Node)
    4.78 +  (:import com.jme3.system.AppSettings)
    4.79 +  (:import com.jme3.system.JmeSystem)
    4.80 +  (:import com.jme3.system.IsoTimer)
    4.81 +  (:import com.jme3.input.KeyInput)
    4.82 +  (:import com.jme3.input.controls.KeyTrigger)
    4.83 +  (:import com.jme3.input.controls.MouseButtonTrigger)
    4.84 +  (:import com.jme3.input.InputManager)
    4.85 +  (:import com.jme3.bullet.BulletAppState)
    4.86 +  (:import com.jme3.shadow.BasicShadowRenderer)
    4.87 +  (:import com.jme3.app.SimpleApplication)
    4.88 +  (:import com.jme3.input.controls.ActionListener)
    4.89 +  (:import com.jme3.renderer.queue.RenderQueue$ShadowMode)
    4.90 +  (:import org.lwjgl.input.Mouse))
    4.91 +#+end_src    
    4.92 +    
    4.93 +** General Settings    
    4.94 +#+srcname: settings    
    4.95 +#+begin_src clojure 
    4.96 +(in-ns 'cortex.world)
    4.97  
    4.98  (defvar *app-settings*
    4.99    (doto (AppSettings. true)
   4.100 @@ -31,7 +88,7 @@
   4.101      )
   4.102    "These settings control how the game is displayed on the screen for
   4.103     debugging purposes.  Use binding forms to change this if desired.
   4.104 -   Full-screen mode does not work on some computers.")
   4.105 +   Full-screen mode does not work on some computers.")    
   4.106  
   4.107  (defn asset-manager
   4.108    "returns a new, configured assetManager" []
   4.109 @@ -39,33 +96,58 @@
   4.110     (.getResource
   4.111      (.getContextClassLoader (Thread/currentThread))
   4.112      "com/jme3/asset/Desktop.cfg")))
   4.113 +#+end_src
   4.114 +
   4.115 +Normally, people just use the =AssetManager= inherited from
   4.116 +=Application= whenever they extend that class. However,
   4.117 +=AssetManagers= are useful on their own to create objects/ materials,
   4.118 +independent from any particular application. =(asset-manager)= makes
   4.119 +object creation less tightly bound to Application initialization.
   4.120 +
   4.121 +
   4.122 +** Exception Protection
   4.123 +#+srcname: exceptions
   4.124 +#+begin_src clojure
   4.125 +(in-ns 'cortex.world)
   4.126  
   4.127  (defmacro no-exceptions
   4.128    "Sweet relief like I never knew."
   4.129    [& forms]
   4.130    `(try ~@forms (catch Exception e# (.printStackTrace e#))))
   4.131  
   4.132 -(defn thread-exception-removal []
   4.133 -  (println "removing exceptions from " (Thread/currentThread))
   4.134 +(defn thread-exception-removal
   4.135 +  "Exceptions thrown in the graphics rendering thread generally cause
   4.136 +  the entire REPL to crash! It is good to suppress them while trying
   4.137 +  things out to shorten the debug loop."
   4.138 +  []
   4.139    (.setUncaughtExceptionHandler
   4.140     (Thread/currentThread)
   4.141     (proxy [Thread$UncaughtExceptionHandler] []
   4.142       (uncaughtException
   4.143 -      [thread thrown]
   4.144 -      (println "uncaught-exception thrown in " thread)
   4.145 -      (println (.getMessage thrown))))))
   4.146 +       [thread thrown]
   4.147 +       (println "uncaught-exception thrown in " thread)
   4.148 +       (println (.getMessage thrown))))))
   4.149  
   4.150 -(def println-repl (bound-fn [& args] (apply println args)))
   4.151 +#+end_src
   4.152  
   4.153 -(use '[pokemon [lpsolve :only [constant-map]]])
   4.154 +Exceptions thrown in the LWJGL render thread, if not caught, will
   4.155 +destroy the entire JVM process including the REPL and slow development
   4.156 +to a crawl.  It is better to try to continue on in the face of
   4.157 +exceptions and keep the REPL alive as long as possible.  Normally it
   4.158 +is possible to just exit the faulty Application, fix the bug,
   4.159 +reevaluate the appropriate forms, and be on your way, without
   4.160 +restarting the JVM.
   4.161  
   4.162 -(defn no-op [& _])
   4.163 +** Input
   4.164 +#+srcname: input
   4.165 +#+begin_src clojure
   4.166 +(in-ns 'cortex.world)
   4.167  
   4.168  (defn all-keys
   4.169 -  "Construct a map of strings representing all the manual inputs from
   4.170 -   either the keyboard or mouse." 
   4.171 +  "Uses reflection to generate a map of string names to jme3 trigger
   4.172 +  objects, which govern input from the keyboard and mouse"
   4.173    []
   4.174 -  (let [inputs (constant-map KeyInput)]
   4.175 +  (let [inputs (pokemon.lpsolve/constant-map KeyInput)]
   4.176      (assoc
   4.177  	(zipmap (map (fn [field]
   4.178  		       (.toLowerCase (re-gsub #"_" "-" field))) (vals inputs))
   4.179 @@ -76,21 +158,26 @@
   4.180        "mouse-right" (MouseButtonTrigger. 1))))
   4.181  
   4.182  (defn initialize-inputs
   4.183 -  "more java-interop cruft to establish keybindings for a particular virtual world"
   4.184 +  "Establish key-bindings for a particular virtual world."
   4.185    [game  input-manager key-map]
   4.186 -  (doall (map (fn [[name trigger]]
   4.187 -		(.addMapping ^InputManager input-manager
   4.188 -			     name (into-array (class trigger) [trigger]))) key-map))
   4.189 -  (doall (map (fn [name] 
   4.190 -		(.addListener ^InputManager input-manager game
   4.191 -			      (into-array String  [name]))) (keys key-map))))
   4.192 +  (doall
   4.193 +   (map (fn [[name trigger]]
   4.194 +          (.addMapping
   4.195 +           ^InputManager input-manager
   4.196 +           name (into-array (class trigger)
   4.197 +                            [trigger]))) key-map))
   4.198 +  (doall
   4.199 +   (map (fn [name] 
   4.200 +          (.addListener
   4.201 +           ^InputManager input-manager game
   4.202 +           (into-array String  [name]))) (keys key-map))))
   4.203  
   4.204  #+end_src
   4.205  
   4.206 -These functions are all for debug controlling of the world through
   4.207 -keyboard and mouse. 
   4.208 +These functions are for controlling the world through the keyboard and
   4.209 +mouse.
   4.210  
   4.211 -We reuse  =constant-map= from =pokemon.lpsolve= to get the numerical
   4.212 +I reuse =constant-map= from [[../../pokemon-types/html/lpsolve.html#sec-3-3-4][=pokemon.lpsolve=]] to get the numerical
   4.213  values for all the keys defined in the =KeyInput= class. The
   4.214  documentation for =constant-map= is:
   4.215  
   4.216 @@ -106,17 +193,63 @@
   4.217  :   fields with their names.  This helps with C wrappers where they have
   4.218  :   just defined a bunch of integer constants instead of enums
   4.219  
   4.220 +=(all-keys)= converts the constant names like =KEY_J= to the more
   4.221 +clojure-like =key-j=, and returns a map from these keys to
   4.222 +jMonkeyEngine =KeyTrigger= objects, which jMonkeyEngine3 uses as it's
   4.223 +abstraction over the physical keys. =all-keys= also adds the three
   4.224 +mouse button controls to the map.
   4.225  
   4.226 -Then, =all-keys= converts the constant names like =KEY_J= to the more
   4.227 -clojure-like =key-j=, and returns a map from these keys to
   4.228 -jMonkeyEngine KeyTrigger objects, the use of which will soon become
   4.229 -apparent. =all-keys= also adds the three mouse button controls to the
   4.230 -map.
   4.231  
   4.232 +#+begin_src clojure :exports both :results output
   4.233 +(require 'clojure.contrib.pprint)
   4.234 +(clojure.contrib.pprint/pprint
   4.235 + (take 10 (keys (cortex.world/all-keys))))
   4.236 +#+end_src
   4.237 +
   4.238 +#+results:
   4.239 +#+begin_example
   4.240 +("key-n"
   4.241 + "key-apps"
   4.242 + "key-pgup"
   4.243 + "key-f8"
   4.244 + "key-o"
   4.245 + "key-at"
   4.246 + "key-f9"
   4.247 + "key-0"
   4.248 + "key-p"
   4.249 + "key-subtract")
   4.250 +#+end_example
   4.251 +
   4.252 +#+begin_src clojure :exports both :results output
   4.253 +(clojure.contrib.pprint/pprint
   4.254 + (take 10 (vals (cortex.world/all-keys))))
   4.255 +#+end_src
   4.256 +
   4.257 +#+results:
   4.258 +#+begin_example
   4.259 +(#<KeyTrigger com.jme3.input.controls.KeyTrigger@6ec21e52>
   4.260 + #<KeyTrigger com.jme3.input.controls.KeyTrigger@a54d24d>
   4.261 + #<KeyTrigger com.jme3.input.controls.KeyTrigger@1ba5e91b>
   4.262 + #<KeyTrigger com.jme3.input.controls.KeyTrigger@296af9cb>
   4.263 + #<KeyTrigger com.jme3.input.controls.KeyTrigger@2e3593ab>
   4.264 + #<KeyTrigger com.jme3.input.controls.KeyTrigger@3f71d740>
   4.265 + #<KeyTrigger com.jme3.input.controls.KeyTrigger@4aeacb4a>
   4.266 + #<KeyTrigger com.jme3.input.controls.KeyTrigger@7cc88db2>
   4.267 + #<KeyTrigger com.jme3.input.controls.KeyTrigger@52cee11e>
   4.268 + #<KeyTrigger com.jme3.input.controls.KeyTrigger@c1da30b>)
   4.269 +#+end_example
   4.270 +
   4.271 +
   4.272 +
   4.273 +** World Creation
   4.274  #+srcname: world
   4.275  #+begin_src clojure :results silent
   4.276  (in-ns 'cortex.world)
   4.277  
   4.278 +(defn no-op
   4.279 +  "Takes any number of arguments and does nothing."
   4.280 +  [& _])
   4.281 +
   4.282  (defn traverse
   4.283    "apply f to every non-node, deeply"
   4.284    [f node]
   4.285 @@ -124,196 +257,129 @@
   4.286      (dorun (map (partial traverse f) (.getChildren node)))
   4.287      (f node)))
   4.288  
   4.289 -(def gravity (Vector3f. 0 -9.81 0))
   4.290 +(defn world
   4.291 +  "the =world= function takes care of the details of initializing a
   4.292 +  SimpleApplication.
   4.293  
   4.294 -(defn world
   4.295 +   ***** Arguments:
   4.296 +
   4.297 +   - root-node : a com.jme3.scene.Node object which contains all of
   4.298 +       the objects that should be in the simulation.
   4.299 +
   4.300 +   - key-map : a map from strings describing keys to functions that
   4.301 +       should be executed whenever that key is pressed.
   4.302 +       the functions should take a SimpleApplication object and a
   4.303 +       boolean value.  The SimpleApplication is the current simulation
   4.304 +       that is running, and the boolean is true if the key is being
   4.305 +       pressed, and false if it is being released. As an example,
   4.306 +
   4.307 +       {\"key-j\" (fn [game value] (if value (println \"key j pressed\")))}
   4.308 +
   4.309 +       is a valid key-map which will cause the simulation to print a
   4.310 +       message whenever the 'j' key on the keyboard is pressed.
   4.311 +
   4.312 +   - setup-fn : a function that takes a SimpleApplication object. It
   4.313 +       is called once when initializing the simulation. Use it to
   4.314 +       create things like lights, change the gravity, initialize debug
   4.315 +       nodes, etc.
   4.316 +
   4.317 +   - update-fn : this function takes a SimpleApplication object and a
   4.318 +       float and is called every frame of the simulation.  The float
   4.319 +       tells how many seconds is has been since the last frame was
   4.320 +       rendered, according to whatever clock jme is currently
   4.321 +       using. The default is to use IsoTimer which will result in this
   4.322 +       value always being the same.
   4.323 +  "
   4.324    [root-node key-map setup-fn update-fn]
   4.325    (let [physics-manager (BulletAppState.)
   4.326 -	shadow-renderer (BasicShadowRenderer. (asset-manager) (int 256))
   4.327 -	;;maybe use a better shadow renderer someday!
   4.328 -	;;shadow-renderer (PssmShadowRenderer. (asset-manager) 256 1)
   4.329 -	]   	
   4.330 -  (doto
   4.331 -      (proxy [SimpleApplication ActionListener] []
   4.332 -	(simpleInitApp
   4.333 -	 []
   4.334 -	 (no-exceptions
   4.335 -	  (.setTimer this (IsoTimer. 60))
   4.336 -	  ;; Create key-map.
   4.337 -	  (.setFrustumFar (.getCamera this) 300)
   4.338 -	  (initialize-inputs this (.getInputManager this) (all-keys))
   4.339 -	  ;; Don't take control of the mouse
   4.340 -	  (org.lwjgl.input.Mouse/setGrabbed false)
   4.341 -	  ;; add all objects to the world
   4.342 -	  (.attachChild (.getRootNode this) root-node)
   4.343 -	  ;; enable physics
   4.344 -	  ;; add a physics manager
   4.345 -	  (.attach (.getStateManager this) physics-manager)
   4.346 -	  (.setGravity (.getPhysicsSpace physics-manager) gravity)
   4.347 -
   4.348 -
   4.349 -	  ;; go through every object and add it to the physics manager
   4.350 -	  ;; if relavant.
   4.351 -	   (traverse (fn [geom]
   4.352 -		       (dorun
   4.353 -			(for [n (range (.getNumControls geom))]
   4.354 -			  (do
   4.355 -			    (println-repl "adding control " (.getControl geom n))
   4.356 -			    (.add (.getPhysicsSpace physics-manager)
   4.357 -				  (.getControl geom n))))))
   4.358 -		     (.getRootNode this))
   4.359 -	  ;;(.addAll (.getPhysicsSpace physics-manager) (.getRootNode this))
   4.360 -
   4.361 -	  (setup-fn this)
   4.362 -	  (.setDirection shadow-renderer
   4.363 -			 (.normalizeLocal (Vector3f. -1 -1 -1)))
   4.364 -	  (.addProcessor (.getViewPort this) shadow-renderer)
   4.365 -	  (.setShadowMode (.getRootNode this) RenderQueue$ShadowMode/Off)
   4.366 -	  ))
   4.367 -	(simpleUpdate
   4.368 -	 [tpf]
   4.369 -	 (no-exceptions
   4.370 -	  (update-fn this tpf))) 
   4.371 -	(onAction
   4.372 -	 [binding value tpf]
   4.373 -	 ;; whenever a key is pressed, call the function returned from
   4.374 -	 ;; key-map.
   4.375 -	 (no-exceptions
   4.376 -	  (if-let [react (key-map binding)]
   4.377 -	    (react this value)))))
   4.378 -    ;; don't show a menu to change options.
   4.379 -    
   4.380 -    (.setShowSettings false)
   4.381 -    (.setPauseOnLostFocus false)
   4.382 -    (.setSettings *app-settings*))))
   4.383 +	shadow-renderer (BasicShadowRenderer.
   4.384 +                         (asset-manager) (int 256))]   	
   4.385 +    (doto
   4.386 +        (proxy [SimpleApplication ActionListener] []
   4.387 +          (simpleInitApp
   4.388 +            []
   4.389 +            (no-exceptions
   4.390 +             ;; allow AI entities as much time as they need to think.
   4.391 +             (.setTimer this (IsoTimer. 60))
   4.392 +             (.setFrustumFar (.getCamera this) 300)
   4.393 +             ;; Create default key-map.
   4.394 +             (initialize-inputs this (.getInputManager this) (all-keys))
   4.395 +             ;; Don't take control of the mouse
   4.396 +             (org.lwjgl.input.Mouse/setGrabbed false)
   4.397 +             ;; add all objects to the world
   4.398 +             (.attachChild (.getRootNode this) root-node)
   4.399 +             ;; enable physics
   4.400 +             ;; add a physics manager
   4.401 +             (.attach (.getStateManager this) physics-manager)
   4.402 +             (.setGravity (.getPhysicsSpace physics-manager) 
   4.403 +                          (Vector3f. 0 -9.81 0))
   4.404 +             ;; go through every object and add it to the physics
   4.405 +             ;; manager if relevant.
   4.406 +             (traverse (fn [geom]
   4.407 +                         (dorun
   4.408 +                          (for [n (range (.getNumControls geom))]
   4.409 +                            (do
   4.410 +                              (.add (.getPhysicsSpace physics-manager)
   4.411 +                                    (.getControl geom n))))))
   4.412 +                       (.getRootNode this))
   4.413 +             ;;(.addAll (.getPhysicsSpace physics-manager) (.getRootNode this))
   4.414 +             
   4.415 +             ;; set some basic defaults for the shadow renderer.
   4.416 +             ;; these can be undone in the setup function
   4.417 +             (.setDirection shadow-renderer
   4.418 +                            (.normalizeLocal (Vector3f. -1 -1 -1)))
   4.419 +             (.addProcessor (.getViewPort this) shadow-renderer)
   4.420 +             (.setShadowMode (.getRootNode this)
   4.421 +                             RenderQueue$ShadowMode/Off)
   4.422 +             ;; call the supplied setup-fn
   4.423 +             (if setup-fn
   4.424 +               (setup-fn this))))
   4.425 +          (simpleUpdate
   4.426 +            [tpf]
   4.427 +            (no-exceptions
   4.428 +             (update-fn this tpf))) 
   4.429 +          (onAction
   4.430 +            [binding value tpf]
   4.431 +            ;; whenever a key is pressed, call the function returned
   4.432 +            ;; from key-map.
   4.433 +            (no-exceptions
   4.434 +             (if-let [react (key-map binding)]
   4.435 +               (react this value)))))
   4.436 +      ;; don't show a menu to change options.      
   4.437 +      (.setShowSettings false)
   4.438 +      ;; continue running simulation even if the window has lost
   4.439 +      ;; focus.
   4.440 +      (.setPauseOnLostFocus false)
   4.441 +      (.setSettings *app-settings*))))
   4.442  
   4.443  (defn apply-map
   4.444 -  "Like apply, but works for maps and functions that expect an implicit map
   4.445 -   and nothing else as in (fn [& {}]).
   4.446 -------- Example -------
   4.447 - (defn jjj [& {:keys [www] :or {www \"oh yeah\"} :as env}] (println www))
   4.448 - (apply-map jjj {:www \"whatever\"})
   4.449 -  -->\"whatever\""
   4.450 +  "Like apply, but works for maps and functions that expect an
   4.451 +   implicit map and nothing else as in (fn [& {}]).
   4.452 +   ------- Example -------
   4.453 +   (defn demo [& {:keys [www] :or {www \"oh yeah\"} :as env}] 
   4.454 +     (println www))
   4.455 +   (apply-map demo {:www \"hello!\"})
   4.456 +   -->\"hello\""
   4.457    [fn m]
   4.458    (apply fn (reduce #(into %1 %2) [] m)))
   4.459  
   4.460  #+end_src 
   4.461  
   4.462  
   4.463 -=world= is the most important function here.  
   4.464 -***  TODO more documentation
   4.465 -
   4.466 -#+srcname: world-shapes
   4.467 -#+begin_src clojure :results silent
   4.468 -(in-ns 'cortex.world)
   4.469 -(defrecord shape-description
   4.470 -  [name
   4.471 -   color
   4.472 -   mass
   4.473 -   friction
   4.474 -   texture
   4.475 -   material
   4.476 -   position
   4.477 -   rotation
   4.478 -   shape
   4.479 -   physical?])
   4.480 -
   4.481 -(def base-shape
   4.482 -     (shape-description.
   4.483 -      "default-shape"
   4.484 -      false
   4.485 -      ;;ColorRGBA/Blue
   4.486 -      1.0 ;; mass
   4.487 -      1.0 ;; friction
   4.488 -      ;; texture
   4.489 -      "Textures/Terrain/BrickWall/BrickWall.jpg"
   4.490 -      ;; material
   4.491 -      "Common/MatDefs/Misc/Unshaded.j3md"
   4.492 -      Vector3f/ZERO
   4.493 -      Quaternion/IDENTITY
   4.494 -      (Box. Vector3f/ZERO 0.5 0.5 0.5)
   4.495 -      true))
   4.496 -
   4.497 -(defn make-shape
   4.498 -  [#^shape-description d]
   4.499 -  (let [asset-manager (if (:asset-manager d) (:asset-manager d) (asset-manager))
   4.500 -        mat (Material. asset-manager (:material d))
   4.501 -	geom (Geometry. (:name d) (:shape d))]
   4.502 -    (if (:texture d)
   4.503 -      (let [key (TextureKey. (:texture d))]
   4.504 -	(.setGenerateMips key true)
   4.505 -	(.setTexture mat "ColorMap" (.loadTexture asset-manager key))))
   4.506 -    (if (:color d) (.setColor mat "Color" (:color d)))
   4.507 -    (.setMaterial geom mat)
   4.508 -    (if-let [rotation (:rotation d)] (.rotate geom rotation))
   4.509 -    (.setLocalTranslation geom (:position d))
   4.510 -    (if (:physical? d)
   4.511 -      (let [impact-shape (doto (GImpactCollisionShape. 
   4.512 -                                (.getMesh geom)) (.setMargin 0))
   4.513 -	    physics-control (RigidBodyControl.
   4.514 -			     ;;impact-shape ;; comment to disable 
   4.515 -			     (float (:mass d)))]
   4.516 -	(.createJmeMesh impact-shape)
   4.517 -	(.addControl geom physics-control)
   4.518 -	;;(.setSleepingThresholds physics-control (float 0) (float 0))
   4.519 -	(.setFriction physics-control (:friction d))))
   4.520 -    ;;the default is to keep this node in the physics engine forever.
   4.521 -    ;;these commands must come after the control is added to the geometry.
   4.522 -    ;;
   4.523 -    geom))
   4.524 -  
   4.525 -(defn box
   4.526 -  ([l w h & {:as options}]
   4.527 -     (let [options (merge base-shape options)]
   4.528 -       (make-shape (assoc options
   4.529 -		     :shape (Box. l w h)))))
   4.530 -  ([] (box 0.5 0.5 0.5)))
   4.531 -
   4.532 -(defn sphere
   4.533 -  ([r & {:as options}]
   4.534 -     (let [options (merge base-shape options)]
   4.535 -       (make-shape (assoc options
   4.536 -		     :shape (Sphere. 32 32 (float r)))))) 
   4.537 -  ([] (sphere 0.5)))
   4.538 -
   4.539 -(defn add-element
   4.540 -  ([game element node]
   4.541 -  (.addAll
   4.542 -   (.getPhysicsSpace
   4.543 -    (.getState
   4.544 -     (.getStateManager game)
   4.545 -     BulletAppState))
   4.546 -    element)
   4.547 -  (.attachChild node element))
   4.548 -  ([game element]
   4.549 -     (add-element game element (.getRootNode game))))
   4.550 -
   4.551 -
   4.552 -(defn set-gravity*
   4.553 -  [game gravity]
   4.554 -  (traverse
   4.555 -   (fn [geom]
   4.556 -     (if-let
   4.557 -	 [control (.getControl geom RigidBodyControl)]
   4.558 -       (do
   4.559 -	 (.setGravity control gravity)
   4.560 -	 (.applyImpulse control Vector3f/ZERO Vector3f/ZERO)
   4.561 -	 )))
   4.562 -   (.getRootNode game)))
   4.563 -
   4.564 -#+end_src
   4.565 -
   4.566 -These are convienence functions for creating JME objects and
   4.567 -manipulating a world.
   4.568 -
   4.569 -
   4.570 +=(world)= is the most important function here. It presents a more
   4.571 +functional interface to the Application life-cycle, and all it's
   4.572 +objects except =root-node= are plain clojure data structures. It's now
   4.573 +possible to extend functionally by composing multiple functions
   4.574 +together, and to add more keyboard-driven actions by combining clojure
   4.575 +maps.
   4.576  
   4.577  
   4.578  
   4.579  * COMMENT code generation
   4.580 -
   4.581  #+begin_src clojure :tangle ../src/cortex/world.clj
   4.582 -<<world-inputs>>
   4.583 +<<header>>
   4.584 +<<settings>>
   4.585 +<<exceptions>>
   4.586 +<<input>>
   4.587  <<world>>
   4.588 -<<world-shapes>>
   4.589  #+end_src