# HG changeset patch # User Robert McIntyre # Date 1320334106 25200 # Node ID 183744c179e6f8c375587397616420b00d26a1f2 # Parent c377222528e62c6d16363f9a54793b627e98fbb2 MASSIVE cleanup, especially in the vision code diff -r c377222528e6 -r 183744c179e6 org/eyes.org --- a/org/eyes.org Wed Nov 02 11:03:12 2011 -0700 +++ b/org/eyes.org Thu Nov 03 08:28:26 2011 -0700 @@ -1,47 +1,37 @@ -#+title: Eyes +#+title: Simulated Sense of Sight #+author: Robert McIntyre #+email: rlm@mit.edu -#+description: Simulating senses for AI research using JMonkeyEngine3 +#+description: Simulated sight for AI research using JMonkeyEngine3 +#+keywords: computer vision, jMonkeyEngine3, clojure #+SETUPFILE: ../../aurellem/org/setup.org #+INCLUDE: ../../aurellem/org/level-0.org #+babel: :mkdirp yes :noweb yes :exports both +* Vision - -** Eyes - -Ultimately I want to make creatures with eyes. Each eye can be -independely moved and should see its own version of the world -depending on where it is. +I want to make creatures with eyes. Each eye can be independely moved +and should see its own version of the world depending on where it is. #+srcname: eyes #+begin_src clojure -(ns body.eye) -(use 'cortex.world) -(use 'cortex.import) -(use 'clojure.contrib.def) -(cortex.import/mega-import-jme3) -(rlm.rlm-commands/help) -(import java.nio.ByteBuffer) -(import java.awt.image.BufferedImage) -(import java.awt.Color) -(import java.awt.Dimension) -(import java.awt.Graphics) -(import java.awt.Graphics2D) -(import java.awt.event.WindowAdapter) -(import java.awt.event.WindowEvent) -(import java.awt.image.BufferedImage) -(import java.nio.ByteBuffer) -(import javax.swing.JFrame) -(import javax.swing.JPanel) -(import javax.swing.SwingUtilities) -(import javax.swing.ImageIcon) -(import javax.swing.JOptionPane) -(import java.awt.image.ImageObserver) +(ns cortex.vision + "Simulate the sense of vision in jMonkeyEngine3. Enables multiple + eyes from different positions to observe the same world, and pass + the observed data to any arbitray function." + {:author "Robert McIntyre"} + (:use cortex.world) + (:import com.jme3.post.SceneProcessor) + (:import (com.jme3.util Screenshots BufferUtils)) + (:import java.nio.ByteBuffer) + (:import java.awt.image.BufferedImage) + (:import com.jme3.renderer.ViewPort) + (:import com.jme3.math.ColorRGBA)) (defn scene-processor - "deals with converting FrameBuffers to BufferedImages so - that the continuation function can be defined only in terms - of what it does with BufferedImages" + "Create a SceneProcessor object which wraps a vision processing + continuation function. The SceneProcessor will take care of + converting the rendered frame to a BufferedImage and passing that + BufferedImage to the continuation. The continuation should be a + function that takes a BufferedImage." [continuation] (let [byte-buffer (atom nil) renderer (atom nil) @@ -56,8 +46,9 @@ (reset! byte-buffer (BufferUtils/createByteBuffer (* width height 4))) - (reset! image (BufferedImage. width height - BufferedImage/TYPE_4BYTE_ABGR)))) + (reset! image (BufferedImage. + width height + BufferedImage/TYPE_4BYTE_ABGR)))) (isInitialized [] (not (nil? @byte-buffer))) (reshape [_ _ _]) (preFrame [_]) @@ -71,179 +62,108 @@ (cleanup [])))) (defn add-eye - "Add an eye to the world, and call continuation on - every frame produced" + "Add an eye to the world, calling continuation on every frame + produced." [world camera continuation] (let [width (.getWidth camera) height (.getHeight camera) render-manager (.getRenderManager world) viewport (.createMainView render-manager "eye-view" camera)] (doto viewport - (.setBackgroundColor ColorRGBA/Black) (.setClearFlags true true true) + (.setBackgroundColor ColorRGBA/Gray) (.addProcessor (scene-processor continuation)) (.attachScene (.getRootNode world))))) -(defn make-display-frame [display width height] - (SwingUtilities/invokeLater - (fn [] - (.setPreferredSize display (Dimension. width height)) - (doto (JFrame. "Eye Camera!") - (-> (.getContentPane) (.add display)) - (.setDefaultCloseOperation JFrame/DISPOSE_ON_CLOSE) - (.pack) - (.setLocationRelativeTo nil) - (.setResizable false) - (.setVisible true))))) - -(defn image-monitor [#^BufferedImage image] - (proxy [JPanel] [] - (paintComponent - [g] - (proxy-super paintComponent g) - (locking image - (.drawImage g image 0 0 - (proxy [ImageObserver] - [] - (imageUpdate - [] - (proxy-super imageUpdate)))))))) +#+end_src -(defn movie-image [] - (let [setup - (runonce - (fn [#^BufferedImage image] - (let [width (.getWidth image) - height (.getHeight image) - display (image-monitor image) - frame (make-display-frame display width height)] - display)))] - (fn [#^BufferedImage image] - (.repaint (setup image))))) - +Note the use of continuation passing style for connecting the eye to a +function to process the output. You can create any number of eyes, and +each of them will see the world from their own =Camera=. Once every +frame, the rendered image is copied to a =BufferedImage=, and that +data is sent off to the continuation function. Moving the =Camera= +which was used to create the eye will change what the eye sees. -(defn observer - "place thy eye!" - [world camera] - (let [eye camera - width (.getWidth eye) - height (.getHeight eye)] - (no-exceptions - (add-eye - world - eye - (movie-image))))) -#+end_src +* Example #+srcname: test-vision #+begin_src clojure +(ns test.vision + (:use (cortex world util vision)) + (:import java.awt.image.BufferedImage) + (:import javax.swing.JPanel) + (:import javax.swing.SwingUtilities) + (:import java.awt.Dimension) + (:import javax.swing.JFrame) + (:import com.jme3.math.ColorRGBA) + (:import com.jme3.scene.Node)) -(ns test.vision) -(use 'cortex.world) -(use 'cortex.import) -(use 'clojure.contrib.def) -(use 'body.eye) -(cortex.import/mega-import-jme3) -(rlm.rlm-commands/help) -(import java.nio.ByteBuffer) -(import java.awt.image.BufferedImage) -(import java.awt.Color) -(import java.awt.Dimension) -(import java.awt.Graphics) -(import java.awt.Graphics2D) -(import java.awt.event.WindowAdapter) -(import java.awt.event.WindowEvent) -(import java.awt.image.BufferedImage) -(import java.nio.ByteBuffer) -(import javax.swing.JFrame) -(import javax.swing.JPanel) -(import javax.swing.SwingUtilities) -(import javax.swing.ImageIcon) -(import javax.swing.JOptionPane) -(import java.awt.image.ImageObserver) +(defn view-image + "Initailizes a JPanel on which you may draw a BufferedImage of the + given width and height. Returns a function that accepts a + BufferedImage and draws it to the JPanel." + [width height] + (let [image + (atom + (BufferedImage. width height BufferedImage/TYPE_4BYTE_ABGR)) + panel + (proxy [JPanel] [] + (paint + [graphics] + (proxy-super paintComponent graphics) + (.drawImage graphics @image 0 0 nil) + ))] + + (SwingUtilities/invokeLater + (fn [] + (.setPreferredSize panel (Dimension. width height)) + (doto (JFrame. "Eye Camera!") + (-> (.getContentPane) (.add panel)) + (.pack) + (.setLocationRelativeTo nil) + (.setResizable false) + (.setVisible true)))) + (fn [#^BufferedImage i] + (reset! image i) + (.repaint panel)))) - -(def width 200) -(def height 200) - -(defn camera [] - (doto (Camera. width height) - (.setFrustumPerspective 45 1 1 1000) - (.setLocation (Vector3f. -3 0 -5)) - (.lookAt Vector3f/ZERO Vector3f/UNIT_Y))) - -(defn camera2 [] - (doto (Camera. width height) - (.setFrustumPerspective 45 1 1 1000) - (.setLocation (Vector3f. 3 0 -5)) - (.lookAt Vector3f/ZERO Vector3f/UNIT_Y))) - -(defn setup-fn [world] - (let [eye (camera) - width (.getWidth eye) - height (.getHeight eye)] - (no-exceptions - (add-eye - world - eye - (runonce visual)) - (add-eye - world - (camera2) - (runonce visual))))) - -(defn spider-eye [position] - (doto (Camera. 200 200 ) - (.setFrustumPerspective 45 1 1 1000) - (.setLocation position) - (.lookAt Vector3f/ZERO Vector3f/UNIT_Y))) - -(defn setup-fn* [world] - (let [eye (camera) - width (.getWidth eye) - height (.getHeight eye)] - ;;(.setClearFlags (.getViewPort world) true true true) - (observer world (.getCamera world)) - (observer world (spider-eye (Vector3f. 3 0 -5))) - ;;(observer world (spider-eye (Vector3f. 0 0 -5))) - ;; (observer world (spider-eye (Vector3f. -3 0 -5))) - ;; (observer world (spider-eye (Vector3f. 0 3 -5))) - ;; (observer world (spider-eye (Vector3f. 0 -3 -5))) - ;; (observer world (spider-eye (Vector3f. 3 3 -5))) - ;; (observer world (spider-eye (Vector3f. -3 3 -5))) - ;; (observer world (spider-eye (Vector3f. 3 -3 -5))) - ;; (observer world (spider-eye (Vector3f. -3 -3 -5))) - - ) - world) - -(defn test-world [] - (let [thing (box 1 1 1 :physical? false)] - (world - (doto (Node.) - (.attachChild thing)) - {} - setup-fn - (fn [world tpf] - (.rotate thing (* tpf 0.2) 0 0) - )))) - - +(defn test-vision + "Tests the vision system by creating two views of the same rotating + object from different angles and displaying both of those views in + JFrames." [] + (.start + (let [candy + (box 1 1 1 :physical? false :color ColorRGBA/Blue)] + (world (doto (Node.) + (.attachChild candy)) + {} + (fn [world] + (let [cam (.clone (.getCamera world)) + width (.getWidth cam) + height (.getHeight cam)] + (add-eye world + (doto (.clone cam) + (.setLocation (Vector3f. -10 0 0)) + (.lookAt Vector3f/ZERO Vector3f/UNIT_Y)) + (view-image width height)) + (add-eye world cam + (view-image width height)) + ;; this is here to restore the main view + ;; after the other views have completed processing + (add-eye world (.getCamera world) no-op) + )) + (fn [world tpf] + (.rotate candy (* tpf 0.2) 0 0)))))) #+end_src - -#+results: eyes -: #'body.eye/test-world - -Note the use of continuation passing style for connecting the eye to a -function to process the output. The example code will create two -videos of the same rotating cube from different angles, sutiable for -stereoscopic vision. - +The example code will create two videos of the same rotating object +from different angles. It can be used both for stereoscopic vision +simulation or for simulating multiple creatures, each with their own +sense of vision. * COMMENT code generation -#+begin_src clojure :tangle ../src/body/eye.clj +#+begin_src clojure :tangle ../src/cortex/vision.clj <> #+end_src diff -r c377222528e6 -r 183744c179e6 org/games.org --- a/org/games.org Wed Nov 02 11:03:12 2011 -0700 +++ b/org/games.org Thu Nov 03 08:28:26 2011 -0700 @@ -1,16 +1,18 @@ -#+title: Games! Games! Games! <3 +#+title: jMonkeyEngine3 from Clojure #+author: Robert McIntyre #+email: rlm@mit.edu -#+description: Simulating senses for AI research using JMonkeyEngine3 +#+description: Using jMonkeyEngine3 from clojure +#+keywords: clojure, jMonkeyEngine3, tutorial, examples #+SETUPFILE: ../../aurellem/org/setup.org #+INCLUDE: ../../aurellem/org/level-0.org #+babel: :mkdirp yes :noweb yes :exports both +[TABLE-OF-CONTENTS] -* Games! Here are the jMonkeyEngine "Hello" programs translated to clojure. -** Hello Simple App + +* Hello Simple App Here is the hello world example for jme3 in clojure. It's a more or less direct translation from the java source [[http://jmonkeyengine.org/wiki/doku.php/jme3:beginner:hello_simpleapplication][here]]. @@ -20,13 +22,14 @@ #+srcname: hello-simple-app #+begin_src clojure :results silent -(ns hello.hello-simple-app) -(require 'cortex.import) -(use 'clojure.contrib.def) -(rlm.rlm-commands/help) -(cortex.import/mega-import-jme3) -(use 'cortex.world) - +(ns hello.hello-simple-app + (:use cortex.world) + (:import com.jme3.math.Vector3f) + (:import com.jme3.material.Material) + (:import com.jme3.scene.Geometry) + (:import com.jme3.math.ColorRGBA) + (:import com.jme3.app.SimpleApplication) + (:import com.jme3.scene.shape.Box)) (def cube (Box. Vector3f/ZERO 1 1 1)) @@ -62,8 +65,31 @@ #+caption: the simplest JME game. [[../images/simple-app.jpg]] +* Simpler HelloSimpleApp -** Hello Physics +#+srcname: hello-simpler-app +#+begin_src clojure +(ns hello.hello-simpler-app + (:use cortex.world) + (:use cortex.util) + (:import com.jme3.math.ColorRGBA) + (:import com.jme3.scene.Node)) + +(defn simpler-world + "The jMonkeyEngine3 Hello World program. Displays a blue 3D cube in + a basic 3D world." + [] + (world (doto (Node.) + (.attachChild + (box 1 1 1 + :color ColorRGBA/Blue :physical? false))) + {} no-op no-op)) +#+end_src + +More information about the jMonkeyEngine3 hello world program can be +found [[http://jmonkeyengine.org/wiki/doku.php/starter:hello_world][here]]. + +* COMMENT Hello Physics From http://jmonkeyengine.org/wiki/doku.php/jme3:beginner:hello_physics #+srcname: brick-wall-header @@ -90,7 +116,7 @@ (def brick-length 0.48) (def brick-width 0.24) (def brick-height 0.12) - +(def gravity (Vector3f. 0 -9.81 0)) (defn brick* [position] (doto (box brick-length brick-height brick-width @@ -126,8 +152,8 @@ (fn [game value] (println-repl "set gravity to " new-value) (if value - (set-gravity* game new-value) - (set-gravity* game gravity)))) + (set-gravity game new-value) + (set-gravity game gravity)))) (defn fire-cannon-ball ([node] @@ -216,10 +242,10 @@ #+caption: the brick wall after it has been knocked over by a "pok\eacute{}ball" [[../images/brick-wall-knocked-down.jpg]] -** Other Brick Games +* COMMENT Other Brick Games #+srcname: other-games #+begin_src clojure :results silent -(ns cortex.other-games +(ns hello.other-games {:author "Dylan Holmes"}) (use 'cortex.world) (use 'hello.brick-wall) @@ -364,15 +390,15 @@ #+caption: floating dominos [[../images/dominos.jpg]] -** Hello Loop +* Hello Loop #+srcname: hello-loop #+begin_src clojure :results silent -(ns hello.loop) -(use 'cortex.world) -(use 'cortex.import) -(cortex.import/mega-import-jme3) -(rlm.rlm-commands/help) - +(ns hello.loop + (:use cortex.world) + (:use cortex.util) + (:import com.jme3.math.ColorRGBA) + (:import com.jme3.scene.Node)) + (defn blue-cube [] (box 1 1 1 :color ColorRGBA/Blue @@ -391,7 +417,7 @@ (.rotate cube 0.0 (* 2 tpf) 0.0))))) #+end_src -** Hello Collision +* COMMENT Hello Collision #+srcname: hello-collision #+begin_src clojure :results silent @@ -513,7 +539,7 @@ update-fn))) #+end_src -** Hello Terrain +* COMMENT Hello Terrain #+srcname: hello-terrain #+begin_src clojure :results silent (ns hello.terrain) @@ -663,12 +689,13 @@ no-op)))) #+end_src -** Hello Materials +* COMMENT Hello Materials #+srcname: material #+begin_src clojure :results silent (ns hello.material) (use 'cortex.world) (use 'cortex.import) +(use 'cortex.util) (use 'clojure.contrib.def) (cortex.import/mega-import-jme3) (rlm.rlm-commands/help) @@ -717,7 +744,7 @@ (doto (.setTexture "DiffuseMap" (.loadTexture (asset-manager) - "Textures/Terrain/Pond/Pond.png")) + "Textures/Terrain/Pond/Pond.jpg")) (.setTexture "NormalMap" (.loadTexture (asset-manager) "Textures/Terrain/Pond/Pond_normal.png")) @@ -755,7 +782,12 @@ <> #+end_src -#+begin_src clojure :tangle ../src/cortex/other_games.clj +#+begin_src clojure :tangle ../src/hello/hello_simpler_app.clj +<> +#+end_src + + +#+begin_src clojure :tangle ../src/hello/other_games.clj <> #+end_src diff -r c377222528e6 -r 183744c179e6 org/intro.org --- a/org/intro.org Wed Nov 02 11:03:12 2011 -0700 +++ b/org/intro.org Thu Nov 03 08:28:26 2011 -0700 @@ -2,6 +2,7 @@ #+author: Robert McIntyre #+email: rlm@mit.edu #+description: Simulating senses for AI research using JMonkeyEngine3 +#+keywords: Alan Turing, AI, sinulated senses, jMonkeyEngine3, virtual world #+SETUPFILE: ../../aurellem/org/setup.org #+INCLUDE: ../../aurellem/org/level-0.org #+babel: :mkdirp yes :noweb yes @@ -14,7 +15,7 @@ already. What idea could be missing? When Turing first proposed his famous "Turing Test" in the -groundbreaking paper [[./sources/turing.pdf][/Computing Machines and Intelligence/]], he gave +groundbreaking paper [[../sources/turing.pdf][/Computing Machines and Intelligence/]], he gave little importance to how a computer program might interact with the world: @@ -42,25 +43,24 @@ your life from birth, you would never learn anything, and could never become intelligent. Actual humans placed in sensory deprivation chambers experience hallucinations and can begin to loose their sense -of reality in as little as 15 minutes[sensory-deprivation]. Most of -the time, the programs we write are in exactly this situation. They do -not interface with cameras and microphones, and they do not control a -real or simulated body or interact with any sort of world. +of reality. Most of the time, the programs we write are in exactly +this situation. They do not interface with cameras and microphones, +and they do not control a real or simulated body or interact with any +sort of world. * Simulation vs. Reality I want demonstrate that multiple senses are what enable intelligence. There are two ways of playing around with senses and computer programs: + +** Simulation The first is to go entirely with simulation: virtual world, virtual character, virtual senses. The advantages are that when everything is a simulation, experiments in that simulation are absolutely reproducible. It's also easier to change the character and world to explore new situations and different sensory combinations. - -** Issues with Simulation - If the world is to be simulated on a computer, then not only do you have to worry about whether the character's senses are rich enough to learn from the world, but whether the world itself is rendered with @@ -75,7 +75,7 @@ dust. Maybe a simulated world with today's limitations doesn't provide enough richness for real intelligence to evolve. -** Issues with Reality +** Reality The other approach for playing with senses is to hook your software up to real cameras, microphones, robots, etc., and let it loose in the @@ -107,16 +107,15 @@ * Choose a Simulation Engine Mainly because of issues with controlling the flow of time, I chose to -simulate both the world and the character. I set out to make a minimal -world in which I could embed a character with multiple senses. My main -goal is to make an environment where I can perform further experiments -in simulated senses. +simulate both the world and the character. I set out to make a world +in which I could embed a character with multiple senses. My main goal +is to make an environment where I can perform further experiments in +simulated senses. -As Carl Sagan once said, "If you wish to make an apple pie from -scratch, you must first invent the universe." I examined many -different 3D environments to try and find something I would use as the -base for my simulation; eventually the choice came down to three -engines: the Quake II engine, the Source Engine, and jMonkeyEngine. +I examined many different 3D environments to try and find something I +would use as the base for my simulation; eventually the choice came +down to three engines: the Quake II engine, the Source Engine, and +jMonkeyEngine. ** [[http://www.idsoftware.com][Quake II]]/[[http://www.bytonic.de/html/jake2.html][Jake2]] @@ -172,7 +171,7 @@ others for about 2 months I settled on jMonkeyEngine. I chose it because it had the most features out of all the open projects I looked at, and because I could then write my code in Clojure, an -implementation of LISP that runs on the JVM... +implementation of LISP that runs on the JVM. @@ -206,4 +205,3 @@ - diff -r c377222528e6 -r 183744c179e6 org/setup.org --- a/org/setup.org Wed Nov 02 11:03:12 2011 -0700 +++ b/org/setup.org Thu Nov 03 08:28:26 2011 -0700 @@ -2,6 +2,7 @@ #+author: Robert McIntyre #+email: rlm@mit.edu #+description: Simulating senses for AI research using JMonkeyEngine3 +#+keywords: JMonkeyEngine3, clojure, java, setup #+SETUPFILE: ../../aurellem/org/setup.org #+INCLUDE: ../../aurellem/org/level-0.org #+babel: :mkdirp yes :noweb yes :exports both @@ -12,7 +13,8 @@ #+srcname: checkout #+begin_src sh :results verbatim -svn checkout http://jmonkeyengine.googlecode.com/svn/trunk/engine jme3 +svn checkout http://jmonkeyengine.googlecode.com/svn/trunk/engine \ + /home/r/proj/jMonkeyEngine3 #+end_src #+results: checkout @@ -23,114 +25,34 @@ #+srcname: build #+begin_src sh :results verbatim -cd jme3 +cd /home/r/proj/jMonkeyEngine3 ant jar | tail -n 2 #+end_src #+results: build : BUILD SUCCESSFUL -: Total time: 15 seconds +: Total time: 5 seconds Also build the javadoc: #+srcname: javadoc #+begin_src sh :results verbatim -cd jme3 +cd /home/r/proj/jMonkeyEngine3 ant javadoc | tail -n 2 #+end_src #+results: javadoc : BUILD SUCCESSFUL -: Total time: 12 seconds +: Total time: 10 seconds -Now, move the jars from the compilation into the project's lib folder. -#+srcname: move-jars -#+begin_src sh :results verbatim -mkdir -p lib -mkdir -p src -cp jme3/dist/jMonkeyEngine3.jar lib/ -cp jme3/dist/lib/* lib/ -ls lib -#+end_src - -#+results: move-jars -#+begin_example -eventbus-1.4.jar -jbullet.jar -jheora-jst-debug-0.6.0.jar -jinput.jar -jME3-jbullet.jar -jME3-lwjgl-natives.jar -jME3-testdata.jar -jME3-test.jar -jMonkeyEngine3.jar -j-ogg-oggd.jar -j-ogg-vorbisd.jar -lwjgl.jar -nifty-1.3.jar -nifty-default-controls-1.3.jar -nifty-examples-1.3.jar -nifty-lwjgl-renderer-1.3.jar -nifty-openal-soundsystem-1.0.jar -nifty-style-black-1.3.jar -nifty-style-grey-1.0.jar -noise-0.0.1-SNAPSHOT.jar -stack-alloc.jar -vecmath.jar -xmlpull-xpp3-1.1.4c.jar -#+end_example - -It's good to create a =assets= directory in the style that the -=AssetManager= will like. - -#+srcname: create-assets -#+begin_src sh :results verbatim -mkdir -p assets -mkdir -p assets/Interface -mkdir -p assets/Materials -mkdir -p assets/MatDefs -mkdir -p assets/Models -mkdir -p assets/Scenes -mkdir -p assets/Shaders -mkdir -p assets/Sounds -mkdir -p assets/Textures -tree -L 1 assets -#+end_src - -#+results: create-assets -#+begin_example -assets -|-- Interface -|-- MatDefs -|-- Materials -|-- Models -|-- Scenes -|-- Shaders -|-- Sounds -`-- Textures - -8 directories, 0 files -#+end_example - - -The java classpath should have all the jars contained in the =lib= -directory as well as the src directory. +The java classpath should have all the jars from the jMonkeyEngine +directory. For example, here is the file I use to run my REPL for clojure. #+include: "/home/r/bin/swank-all" src sh :exports code -The important thing here is that =cortex/lib/*=, =cortex/src=, and -=cortex/assets= appear on the classpath. (=cortex= is the base -directory of this project.) -#+srcname: pwd -#+begin_src sh -pwd -#+end_src -#+results: pwd -: /home/r/proj/cortex - diff -r c377222528e6 -r 183744c179e6 org/util.org --- a/org/util.org Wed Nov 02 11:03:12 2011 -0700 +++ b/org/util.org Thu Nov 03 08:28:26 2011 -0700 @@ -6,12 +6,12 @@ #+SETUPFILE: ../../aurellem/org/setup.org #+INCLUDE: ../../aurellem/org/level-0.org -* Utilities +[TABLE-OF-CONTENTS] These are a collection of functions to make programming jMonkeyEngine in clojure easier. -** Imports +* Imports #+srcname: import #+begin_src clojure :results silent @@ -53,7 +53,7 @@ importing only the classes it actually needs. The =mega-import-jme3= is quite usefull for debugging purposes since -it allows completion for almost all of JME's classes. +it allows completion for almost all of JME's classes from the REPL. Out of curiousity, let's see just how many classes =mega-import-jme3= imports: @@ -66,7 +66,7 @@ : 955 classes -** Utilities +* Utilities The utilities here come in three main groups: - Changing settings in a running =Application= @@ -79,8 +79,8 @@ #+srcname: util #+begin_src clojure (ns cortex.util - "Utility functions for making jMonkeyEngine easier to program from - clojure" + "Utility functions for making jMonkeyEngine3 easier to program from + clojure." {:author "Robert McIntyre"} (:use cortex.world) (:use clojure.contrib.def) @@ -105,16 +105,17 @@ will always output to the REPL") (defn position-camera - ([game position direction up] - (doto (.getCamera game) + "Change the position of the in-world camera." + ([world position direction up] + (doto (.getCamera world) (.setLocation ) (.lookAt direction up))) - ([game position direction] + ([world position direction] (position-camera - game position direction Vector3f/UNIT_Y))) + world position direction Vector3f/UNIT_Y))) (defn enable-debug - "Turn on the debug wireframes for every object in this simulation" + "Turn on debug wireframes for every object in this simulation." [world] (.enableDebug (.getPhysicsSpace @@ -123,11 +124,11 @@ BulletAppState)) (asset-manager))) -(defn set-gravity +(defn set-gravity "In order to change the gravity of a scene, it is not only necessary to set the gravity variable, but to \"tap\" every physics object in the scene to reactivate physics calculations." - [game gravity] + [world gravity] (traverse (fn [geom] (if-let @@ -137,20 +138,20 @@ (.setGravity control gravity) ;; tappsies! (.applyImpulse control Vector3f/ZERO Vector3f/ZERO)))) - (.getRootNode game))) + (.getRootNode world))) (defn add-element - "Add the Spatial to the game's environment" - ([game element node] + "Add the Spatial to the world's environment" + ([world element node] (.addAll (.getPhysicsSpace (.getState - (.getStateManager game) + (.getStateManager world) BulletAppState)) element) (.attachChild node element)) - ([game element] - (add-element game element (.getRootNode game)))) + ([world element] + (add-element world element (.getRootNode world)))) (defn apply-map "Like apply, but works for maps and functions that expect an @@ -186,7 +187,7 @@ GImpact? ]) -(def base-shape +(defvar base-shape (shape-description. "default-shape" false @@ -201,7 +202,8 @@ Quaternion/IDENTITY (Box. Vector3f/ZERO 0.5 0.5 0.5) true - false)) + false) + "Basic settings for shapes.") (defn make-shape [#^shape-description d] @@ -210,8 +212,9 @@ geom (Geometry. (:name d) (:shape d))] (if (:texture d) (let [key (TextureKey. (:texture d))] - (.setGenerateMips key true) - (.setTexture mat "ColorMap" (.loadTexture asset-manager key)))) + ;;(.setGenerateMips key true) + ;;(.setTexture mat "ColorMap" (.loadTexture asset-manager key)) + )) (if (:color d) (.setColor mat "Color" (:color d))) (.setMaterial geom mat) (if-let [rotation (:rotation d)] (.rotate geom rotation)) @@ -248,7 +251,6 @@ ([] (sphere 0.5))) #+end_src - *** Viewing Objects #+srcname: world-view @@ -279,12 +281,13 @@ (.setDirection (.normalizeLocal (Vector3f. 1 0 -2))) (.setColor ColorRGBA/White))] + ;; lights are required to view some objects. (.addLight (.getRootNode world) sun))) no-op)))) #+end_src Here I make the =Viewable= protocol and extend it to JME's types. Now -hello-world can be written as easily as: +JME3's =hello-world= can be written as easily as: #+begin_src clojure :results silent (cortex.util/view (cortex.util/box)) diff -r c377222528e6 -r 183744c179e6 org/world.org --- a/org/world.org Wed Nov 02 11:03:12 2011 -0700 +++ b/org/world.org Thu Nov 03 08:28:26 2011 -0700 @@ -1,4 +1,4 @@ -#+title: A World for the Creatures +#+title: A Virtual World for Sensate Creatures #+author: Robert McIntyre #+email: rlm@mit.edu #+description: Creating a Virtual World for AI constructs using clojure and JME3 @@ -31,12 +31,12 @@ - Use a map from keys->functions to specify key-bindings. - Use functions to create objects separately from any particular application. - - Use an REPL -- this means that there's only ever one JVM, and + - Use a REPL -- this means that there's only ever one JVM, and Applications come and go. Since most development work using jMonkeyEngine is done in Java, jme3 supports "the Java way" quite well out of the box. To work "the -clojure way", it necessary to wrap the jme3 elements that deal with +clojure way", it necessary to wrap the JME3 elements that deal with the Application life-cycle with a REPL driven interface. The most important modifications are: @@ -57,11 +57,12 @@ (:use [pokemon [lpsolve :only [constant-map]]]) (:use [clojure.contrib [str-utils :only [re-gsub]]]) + (:import com.aurellem.capture.IsoTimer) + (:import com.jme3.math.Vector3f) (:import com.jme3.scene.Node) (:import com.jme3.system.AppSettings) (:import com.jme3.system.JmeSystem) - (:import com.jme3.system.IsoTimer) (:import com.jme3.input.KeyInput) (:import com.jme3.input.controls.KeyTrigger) (:import com.jme3.input.controls.MouseButtonTrigger) @@ -83,9 +84,8 @@ (doto (AppSettings. true) (.setFullscreen false) (.setTitle "Aurellem.") - ;; disable 32 bit stuff for now - ;;(.setAudioRenderer "Send") - ) + ;; The "Send" AudioRenderer supports sumulated hearing. + (.setAudioRenderer "Send")) "These settings control how the game is displayed on the screen for debugging purposes. Use binding forms to change this if desired. Full-screen mode does not work on some computers.") @@ -102,7 +102,8 @@ =Application= whenever they extend that class. However, =AssetManagers= are useful on their own to create objects/ materials, independent from any particular application. =(asset-manager)= makes -object creation less tightly bound to Application initialization. +object creation less tightly bound to a particular Application +Instance. ** Exception Protection @@ -170,18 +171,18 @@ (map (fn [name] (.addListener ^InputManager input-manager game - (into-array String [name]))) (keys key-map)))) + (into-array String [name]))) (keys key-map)))) #+end_src These functions are for controlling the world through the keyboard and mouse. -I reuse =constant-map= from [[../../pokemon-types/html/lpsolve.html#sec-3-3-4][=pokemon.lpsolve=]] to get the numerical +I reuse =constant-map= from [[../../pokemon-types/html/lpsolve.html#sec-3-2-4][=pokemon.lpsolve=]] to get the numerical values for all the keys defined in the =KeyInput= class. The documentation for =constant-map= is: -#+begin_src clojure :results output +#+begin_src clojure :results output :exports both (doc pokemon.lpsolve/constant-map) #+end_src @@ -220,7 +221,6 @@ : ["key-o" #] : ["key-at" #]) - ** World Creation #+srcname: world #+begin_src clojure :results silent @@ -271,9 +271,7 @@ value always being the same. " [root-node key-map setup-fn update-fn] - (let [physics-manager (BulletAppState.) - shadow-renderer (BasicShadowRenderer. - (asset-manager) (int 256))] + (let [physics-manager (BulletAppState.)] (doto (proxy [SimpleApplication ActionListener] [] (simpleInitApp @@ -302,15 +300,6 @@ (.add (.getPhysicsSpace physics-manager) (.getControl geom n)))))) (.getRootNode this)) - ;;(.addAll (.getPhysicsSpace physics-manager) (.getRootNode this)) - - ;; set some basic defaults for the shadow renderer. - ;; these can be undone in the setup function - (.setDirection shadow-renderer - (.normalizeLocal (Vector3f. -1 -1 -1))) - (.addProcessor (.getViewPort this) shadow-renderer) - (.setShadowMode (.getRootNode this) - RenderQueue$ShadowMode/Off) ;; call the supplied setup-fn (if setup-fn (setup-fn this))))