Simple DOOM style navigation of a 3D scene using Java 3D : 3D Animation « 3D « Java

Java
1. 2D Graphics GUI
2. 3D
3. Advanced Graphics
4. Ant
5. Apache Common
6. Chart
7. Collections Data Structure
8. Database SQL JDBC
9. Design Pattern
10. Development Class
11. Email
12. Event
13. File Input Output
14. Game
15. Hibernate
16. J2EE
17. J2ME
18. JDK 6
19. JSP
20. JSTL
21. Language Basics
22. Network Protocol
23. PDF RTF
24. Regular Expressions
25. Security
26. Servlets
27. Spring
28. Swing Components
29. Swing JFC
30. SWT JFace Eclipse
31. Threads
32. Tiny Application
33. Velocity
34. Web Services SOA
35. XML
Microsoft Office Word 2007 Tutorial
Java Tutorial
Java Source Code / Java Documentation
Java Open Source
Jar File Download
Java Articles
Java Products
Java by API
C# / C Sharp
C# / CSharp Tutorial
ASP.Net
JavaScript DHTML
JavaScript Tutorial
JavaScript Reference
HTML / CSS
HTML CSS Reference
C / ANSI-C
C Tutorial
C++
C++ Tutorial
PHP
Python
SQL Server / T-SQL
Oracle PL / SQL
Oracle PL/SQL Tutorial
PostgreSQL
SQL / MySQL
MySQL Tutorial
VB.Net
VB.Net Tutorial
Java » 3D » 3D AnimationScreenshots 
Simple DOOM style navigation of a 3D scene using Java 3D
Simple DOOM style navigation of a 3D scene using Java 3D


/**********************************************************
 Copyright (C) 2001   Daniel Selman

 First distributed with the book "Java 3D Programming"
 by Daniel Selman and published by Manning Publications.
 http://manning.com/selman

 This program is free software; you can redistribute it and/or
 modify it under the terms of the GNU General Public License
 as published by the Free Software Foundation, version 2.

 This program is distributed in the hope that it will be useful,
 but WITHOUT ANY WARRANTY; without even the implied warranty of
 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 GNU General Public License for more details.

 The license can be found on the WWW at:
 http://www.fsf.org/copyleft/gpl.html

 Or by writing to:
 Free Software Foundation, Inc.,
 59 Temple Place - Suite 330, Boston, MA  02111-1307, USA.

 Author can be contacted at:
 Daniel Selman: daniel@selman.org

 If you make changes you think others would like please
 contact Daniel Selman.
 **************************************************************/

import java.applet.Applet;
import java.awt.AWTEvent;
import java.awt.BorderLayout;
import java.awt.Color;
import java.awt.Component;
import java.awt.GraphicsConfigTemplate;
import java.awt.GraphicsDevice;
import java.awt.GraphicsEnvironment;
import java.awt.event.KeyEvent;
import java.awt.image.BufferedImage;
import java.io.File;
import java.io.InputStream;
import java.net.URL;
import java.util.Enumeration;
import java.util.Vector;

import javax.media.j3d.Alpha;
import javax.media.j3d.Appearance;
import javax.media.j3d.AudioDevice;
import javax.media.j3d.Background;
import javax.media.j3d.BackgroundSound;
import javax.media.j3d.Behavior;
import javax.media.j3d.BoundingSphere;
import javax.media.j3d.Bounds;
import javax.media.j3d.BranchGroup;
import javax.media.j3d.Canvas3D;
import javax.media.j3d.ColoringAttributes;
import javax.media.j3d.GeometryArray;
import javax.media.j3d.GraphicsConfigTemplate3D;
import javax.media.j3d.Group;
import javax.media.j3d.ImageComponent2D;
import javax.media.j3d.Locale;
import javax.media.j3d.MediaContainer;
import javax.media.j3d.Node;
import javax.media.j3d.PhysicalBody;
import javax.media.j3d.PhysicalEnvironment;
import javax.media.j3d.PointSound;
import javax.media.j3d.PolygonAttributes;
import javax.media.j3d.QuadArray;
import javax.media.j3d.Shape3D;
import javax.media.j3d.Sound;
import javax.media.j3d.Texture;
import javax.media.j3d.TextureAttributes;
import javax.media.j3d.Transform3D;
import javax.media.j3d.TransformGroup;
import javax.media.j3d.TransparencyAttributes;
import javax.media.j3d.View;
import javax.media.j3d.ViewPlatform;
import javax.media.j3d.VirtualUniverse;
import javax.media.j3d.WakeupCondition;
import javax.media.j3d.WakeupCriterion;
import javax.media.j3d.WakeupOnAWTEvent;
import javax.media.j3d.WakeupOnCollisionEntry;
import javax.media.j3d.WakeupOnCollisionExit;
import javax.media.j3d.WakeupOnElapsedTime;
import javax.media.j3d.WakeupOr;
import javax.vecmath.Color3f;
import javax.vecmath.Color4f;
import javax.vecmath.Point2d;
import javax.vecmath.Point2f;
import javax.vecmath.Point3d;
import javax.vecmath.Point3f;
import javax.vecmath.Quat4f;
import javax.vecmath.Vector3d;
import javax.vecmath.Vector3f;
import javax.vecmath.Vector4f;

import com.sun.j3d.audioengines.javasound.JavaSoundMixer;
import com.sun.j3d.loaders.Scene;
import com.sun.j3d.loaders.objectfile.ObjectFile;
import com.sun.j3d.utils.applet.MainFrame;
import com.sun.j3d.utils.behaviors.interpolators.RotPosScaleTCBSplinePathInterpolator;
import com.sun.j3d.utils.behaviors.interpolators.TCBKeyFrame;
import com.sun.j3d.utils.geometry.Cone;
import com.sun.j3d.utils.geometry.Primitive;
import com.sun.j3d.utils.image.TextureLoader;

/**
 * Simple DOOM style navigation of a 3D scene using Java 3D. The scene
 * description is loaded from a GIF file where different colors denote objects
 * in the 3D scene. The example includes: simple (manual) collision detection,
 * texture animation, keyboard navigation.
 */
public class KeyNavigateTest extends Java3dApplet implements CollisionDetector {
  private static int m_kWidth = 300;

  private static int m_kHeight = 300;

  private static final String m_szMapName = new String("small_map.gif");

  private float FLOOR_WIDTH = 256;

  private float FLOOR_LENGTH = 256;

  private final int m_ColorWall = new Color(000).getRGB();

  private final int m_ColorGuard = new Color(25500).getRGB();

  private final int m_ColorLight = new Color(2552550).getRGB();

  private final int m_ColorBookcase = new Color(02550).getRGB();

  private final int m_ColorWater = new Color(00255).getRGB();

  private Vector3d m_MapSquareSize = null;

  private Appearance m_WallAppearance = null;

  private Appearance m_GuardAppearance = null;

  private Appearance m_BookcaseAppearance = null;

  private Appearance m_WaterAppearance = null;

  private BufferedImage m_MapImage = null;

  private final double m_kFloorLevel = -20;

  private final double m_kCeilingHeight = 50;

  private final double m_kCeilingLevel = m_kFloorLevel + m_kCeilingHeight;

  private Vector3d m_Translation = new Vector3d();

  public KeyNavigateTest() {
    initJava3d();
  }

  protected double getScale() {
    return 0.05;
  }

  protected int getCanvas3dWidth(Canvas3D c3d) {
    return m_kWidth - 10;
  }

  protected int getCanvas3dHeight(Canvas3D c3d) {
    return m_kHeight - 10;
  }

  // edit the positions of the clipping
  // planes so we don't clip on the front
  // plane prematurely
  protected double getBackClipDistance() {
    return 20.0;
  }

  protected double getFrontClipDistance() {
    return 0.05;
  }

  // we create 2 TransformGroups above the ViewPlatform:
  // the first merely applies a scale, while the second is
  // manipulated using the KeyBehavior
  public TransformGroup[] getViewTransformGroupArray() {
    TransformGroup[] tgArray = new TransformGroup[2];
    tgArray[0new TransformGroup();
    tgArray[1new TransformGroup();

    Transform3D t3d = new Transform3D();
    t3d.setScale(getScale());
    t3d.invert();
    tgArray[0].setTransform(t3d);

    // ensure the Behavior can access the TG
    tgArray[1].setCapability(TransformGroup.ALLOW_TRANSFORM_WRITE);

    // create the KeyBehavior and attach
    KeyCollisionBehavior keyBehavior = new KeyCollisionBehavior(tgArray[1],
        this);
    keyBehavior.setSchedulingBounds(getApplicationBounds());
    keyBehavior.setMovementRate(0.7);
    tgArray[1].addChild(keyBehavior);

    return tgArray;
  }

  protected BranchGroup createSceneBranchGroup() {
    BranchGroup objRoot = super.createSceneBranchGroup();

    TransformGroup objTrans = new TransformGroup();
    objTrans.setCapability(TransformGroup.ALLOW_TRANSFORM_WRITE);
    objTrans.setCapability(TransformGroup.ALLOW_TRANSFORM_READ);

    createMap(objTrans);
    createFloor(objTrans);
    createCeiling(objTrans);

    objRoot.addChild(objTrans);

    return objRoot;
  }

  public Group createFloor(Group g) {
    System.out.println("Creating floor");

    Group floorGroup = new Group();
    Land floorTile = null;

    // use a shared Appearance so we only store 1 copy of the texture
    Appearance app = new Appearance();
    g.addChild(floorGroup);

    final double kNumTiles = 6;

    for (double x = -FLOOR_WIDTH + FLOOR_WIDTH / (* kNumTiles); x < FLOOR_WIDTH; x = x
        + FLOOR_WIDTH / kNumTiles) {
      for (double z = -FLOOR_LENGTH + FLOOR_LENGTH / (* kNumTiles); z < FLOOR_LENGTH; z = z
          + FLOOR_LENGTH / kNumTiles) {
        floorTile = new Land(this, g, ComplexObject.GEOMETRY
            | ComplexObject.TEXTURE);
        floorTile.createObject(app, new Vector3d(x, m_kFloorLevel, z),
            new Vector3d(FLOOR_WIDTH / (* kNumTiles)1,
                FLOOR_LENGTH / (* kNumTiles))"floor.gif",
            null, null);
      }
    }

    return floorGroup;
  }

  public Group createCeiling(Group g) {
    System.out.println("Creating ceiling");

    Group ceilingGroup = new Group();
    Land ceilingTile = null;

    // because we are technically viewing the ceiling from below
    // we want to switch the normals using a PolygonAttributes.
    Appearance app = new Appearance();
    app.setPolygonAttributes(new PolygonAttributes(
        PolygonAttributes.POLYGON_FILL, PolygonAttributes.CULL_NONE, 0,
        false));

    g.addChild(ceilingGroup);

    final double kNumTiles = 6;

    for (double x = -FLOOR_WIDTH + FLOOR_WIDTH / (* kNumTiles); x < FLOOR_WIDTH; x = x
        + FLOOR_WIDTH / kNumTiles) {
      for (double z = -FLOOR_LENGTH + FLOOR_LENGTH / (* kNumTiles); z < FLOOR_LENGTH; z = z
          + FLOOR_LENGTH / kNumTiles) {
        ceilingTile = new Land(this, g, ComplexObject.GEOMETRY
            | ComplexObject.TEXTURE);
        ceilingTile.createObject(app, new Vector3d(x, m_kCeilingLevel,
            z)new Vector3d(FLOOR_WIDTH / (* kNumTiles)1,
            FLOOR_LENGTH / (* kNumTiles))"ceiling.gif", null,
            null);
      }
    }

    return ceilingGroup;
  }

  Point3d convertToWorldCoordinate(int nPixelX, int nPixelY) {
    Point3d point3d = new Point3d();
    Vector3d squareSize = getMapSquareSize();

    // range from 0 to 1
    point3d.x = nPixelX * squareSize.x;
    point3d.x -= FLOOR_WIDTH;

    point3d.z = nPixelY * squareSize.z;
    point3d.z -= FLOOR_LENGTH;

    point3d.y = 0;

    return point3d;
  }

  Point3d convertToWorldCoordinatesPixelCenter(int nPixelX, int nPixelY) {
    Point3d point3d = convertToWorldCoordinate(nPixelX, nPixelY);
    Vector3d squareSize = getMapSquareSize();

    point3d.x += squareSize.x / 2;
    point3d.z += squareSize.z / 2;

    return point3d;
  }

  Point2d convertToMapCoordinate(Vector3d worldCoord) {
    Point2d point2d = new Point2d();

    Vector3d squareSize = getMapSquareSize();

    point2d.x = (worldCoord.x + FLOOR_WIDTH/ squareSize.x;
    point2d.y = (worldCoord.z + FLOOR_LENGTH/ squareSize.z;

    return point2d;
  }

  public boolean isCollision(Transform3D t3d, boolean bViewSide) {
    // get the translation
    t3d.get(m_Translation);

    // we need to scale up by the scale that was
    // applied to the root TG on the view side of the scenegraph
    if (bViewSide != false)
      m_Translation.scale(1.0 / getScale());

    Vector3d mapSquareSize = getMapSquareSize();

    // first check that we are still inside the "world"
    if (m_Translation.x < -FLOOR_WIDTH + mapSquareSize.x
        || m_Translation.x > FLOOR_WIDTH - mapSquareSize.x
        || m_Translation.y < -FLOOR_LENGTH + mapSquareSize.y
        || m_Translation.y > FLOOR_LENGTH - mapSquareSize.y)
      return true;

    if (bViewSide != false)
      // then do a pixel based look up using the map
      return isCollision(m_Translation);

    return false;
  }

  // returns true if the given x,z location in the world corresponds to a wall
  // section
  protected boolean isCollision(Vector3d worldCoord) {
    Point2d point = convertToMapCoordinate(worldCoord);
    int nImageWidth = m_MapImage.getWidth();
    int nImageHeight = m_MapImage.getHeight();

    // outside of image
    if (point.x < || point.x >= nImageWidth || point.y < 0
        || point.y >= nImageHeight)
      return true;

    int color = m_MapImage.getRGB((intpoint.x, (intpoint.y);

    // we can't walk through walls or bookcases
    return (color == m_ColorWall || color == m_ColorBookcase);
  }

  public Group createMap(Group g) {
    System.out.println("Creating map items");

    Group mapGroup = new Group();
    g.addChild(mapGroup);

    Texture tex = new TextureLoader(m_szMapName, this).getTexture();
    m_MapImage = ((ImageComponent2Dtex.getImage(0)).getImage();

    float imageWidth = m_MapImage.getWidth();
    float imageHeight = m_MapImage.getHeight();

    FLOOR_WIDTH = imageWidth * 8;
    FLOOR_LENGTH = imageHeight * 8;

    for (int nPixelX = 1; nPixelX < imageWidth - 1; nPixelX++) {
      for (int nPixelY = 1; nPixelY < imageWidth - 1; nPixelY++)
        createMapItem(mapGroup, nPixelX, nPixelY);

      float percentDone = 100 (floatnPixelX
          (float) (imageWidth - 2);
      System.out.println("   " (int) (percentDone"%");
    }

    createExternalWall(mapGroup);

    return mapGroup;
  }

  void createMapItem(Group mapGroup, int nPixelX, int nPixelY) {
    int color = m_MapImage.getRGB((intnPixelX, (intnPixelY);

    if (color == m_ColorWall)
      createWall(mapGroup, nPixelX, nPixelY);

    else if (color == m_ColorGuard)
      createGuard(mapGroup, nPixelX, nPixelY);

    else if (color == m_ColorLight)
      createLight(mapGroup, nPixelX, nPixelY);

    else if (color == m_ColorBookcase)
      createBookcase(mapGroup, nPixelX, nPixelY);

    else if (color == m_ColorWater)
      createWater(mapGroup, nPixelX, nPixelY);
  }

  Vector3d getMapSquareSize() {
    if (m_MapSquareSize == null) {
      double imageWidth = m_MapImage.getWidth();
      double imageHeight = m_MapImage.getHeight();
      m_MapSquareSize = new Vector3d(* FLOOR_WIDTH / imageWidth, 02
          * FLOOR_LENGTH / imageHeight);
    }

    return m_MapSquareSize;
  }

  void createWall(Group mapGroup, int nPixelX, int nPixelY) {
    Point3d point = convertToWorldCoordinatesPixelCenter(nPixelX, nPixelY);

    if (m_WallAppearance == null)
      m_WallAppearance = new Appearance();

    Vector3d squareSize = getMapSquareSize();

    Cuboid wall = new Cuboid(this, mapGroup, ComplexObject.GEOMETRY
        | ComplexObject.TEXTURE);
    wall
        .createObject(m_WallAppearance, new Vector3d(point.x,
            m_kFloorLevel, point.z)new Vector3d(squareSize.x / 2,
            m_kCeilingHeight / 2, squareSize.z / 2)"wall.gif",
            null, null);
  }

  void createExternalWall(Group mapGroup) {
    Vector3d squareSize = getMapSquareSize();

    Appearance app = new Appearance();
    app.setColoringAttributes(new ColoringAttributes(new Color3f(
        132f 255f066f 255f), ColoringAttributes.FASTEST));

    int imageWidth = m_MapImage.getWidth();
    int imageHeight = m_MapImage.getHeight();

    Point3d topLeft = convertToWorldCoordinatesPixelCenter(00);
    Point3d bottomRight = convertToWorldCoordinatesPixelCenter(
        imageWidth - 1, imageHeight - 1);

    // top
    Cuboid wall = new Cuboid(this, mapGroup, ComplexObject.GEOMETRY);
    wall.createObject(app, new Vector3d(0, m_kFloorLevel, topLeft.z),
        new Vector3d(squareSize.x * imageWidth / 2,
            m_kCeilingHeight / 2, squareSize.z / 2), null, null,
        null);

    // bottom
    wall = new Cuboid(this, mapGroup, ComplexObject.GEOMETRY);
    wall.createObject(app, new Vector3d(0, m_kFloorLevel, bottomRight.z),
        new Vector3d(squareSize.x * imageWidth / 2,
            m_kCeilingHeight / 2, squareSize.z / 2), null, null,
        null);

    // left
    wall = new Cuboid(this, mapGroup, ComplexObject.GEOMETRY);
    wall.createObject(app, new Vector3d(topLeft.x, m_kFloorLevel, 0),
        new Vector3d(squareSize.x / 2, m_kCeilingHeight / 2,
            squareSize.z * imageHeight / 2), null, null, null);

    // right
    wall = new Cuboid(this, mapGroup, ComplexObject.GEOMETRY);
    wall.createObject(app, new Vector3d(bottomRight.x, m_kFloorLevel, 0),
        new Vector3d(squareSize.x / 2, m_kCeilingHeight / 2,
            squareSize.z * imageHeight / 2), null, null, null);
  }

  void createGuard(Group mapGroup, int nPixelX, int nPixelY) {
    Point3d point = convertToWorldCoordinatesPixelCenter(nPixelX, nPixelY);

    if (m_GuardAppearance == null)
      m_GuardAppearance = new Appearance();

    Vector3d squareSize = getMapSquareSize();

    Guard guard = new Guard(this, mapGroup, ComplexObject.GEOMETRY, this);
    guard.createObject(m_GuardAppearance, new Vector3d(point.x,
        (m_kFloorLevel + m_kCeilingLevel2, point.z)new Vector3d(
        111), null, null, null);
  }

  void createLight(Group mapGroup, int nPixelX, int nPixelY) {
    Point3d point = convertToWorldCoordinatesPixelCenter(nPixelX, nPixelY);

    // we do not share an Appearance for lights
    // or they all animate in synch...
    final double lightHeight = m_kCeilingHeight / 1.5;

    Light light = new Light(this, mapGroup, ComplexObject.GEOMETRY
        | ComplexObject.TEXTURE);
    light.createObject(new Appearance()new Vector3d(point.x,
        m_kFloorLevel + lightHeight / 2, point.z)new Vector3d(5,
        lightHeight, 5)"light.gif", null, null);
  }

  void createWater(Group mapGroup, int nPixelX, int nPixelY) {
    Point3d point = convertToWorldCoordinatesPixelCenter(nPixelX, nPixelY);

    if (m_WaterAppearance == null) {
      m_WaterAppearance = new Appearance();
      m_WaterAppearance.setPolygonAttributes(new PolygonAttributes(
          PolygonAttributes.POLYGON_FILL,
          PolygonAttributes.CULL_NONE, 0false));
      m_WaterAppearance
          .setTransparencyAttributes(new TransparencyAttributes(
              TransparencyAttributes.BLENDED, 1.0f));
      m_WaterAppearance.setTextureAttributes(new TextureAttributes(
          TextureAttributes.REPLACE, new Transform3D()new Color4f(
              0001), TextureAttributes.FASTEST));

    }

    Land water = new Land(this, mapGroup, ComplexObject.GEOMETRY
        | ComplexObject.TEXTURE);
    water.createObject(m_WaterAppearance, new Vector3d(point.x,
        m_kFloorLevel + 0.1, point.z)new Vector3d(40140),
        "water.gif", null, null);
  }

  void createBookcase(Group mapGroup, int nPixelX, int nPixelY) {
    Point3d point = convertToWorldCoordinatesPixelCenter(nPixelX, nPixelY);

    if (m_BookcaseAppearance == null)
      m_BookcaseAppearance = new Appearance();

    Vector3d squareSize = getMapSquareSize();

    Cuboid bookcase = new Cuboid(this, mapGroup, ComplexObject.GEOMETRY
        | ComplexObject.TEXTURE);
    bookcase.createObject(m_BookcaseAppearance, new Vector3d(point.x,
        m_kFloorLevel, point.z)new Vector3d(squareSize.x / 2,
        m_kCeilingHeight / 2.7, squareSize.z / 2)"bookcase.gif",
        null, null);

  }

  public static void main(String[] args) {
    KeyNavigateTest keyTest = new KeyNavigateTest();
    keyTest.saveCommandLineArguments(args);

    new MainFrame(keyTest, m_kWidth, m_kHeight);
  }
}

/*******************************************************************************
 * Copyright (C) 2001 Daniel Selman
 
 * First distributed with the book "Java 3D Programming" by Daniel Selman and
 * published by Manning Publications. http://manning.com/selman
 
 * This program is free software; you can redistribute it and/or modify it under
 * the terms of the GNU General Public License as published by the Free Software
 * Foundation, version 2.
 
 * This program is distributed in the hope that it will be useful, but WITHOUT
 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
 * FOR A PARTICULAR PURPOSE. See the GNU General Public License for more
 * details.
 
 * The license can be found on the WWW at: http://www.fsf.org/copyleft/gpl.html
 
 * Or by writing to: Free Software Foundation, Inc., 59 Temple Place - Suite
 * 330, Boston, MA 02111-1307, USA.
 
 * Authors can be contacted at: Daniel Selman: daniel@selman.org
 
 * If you make changes you think others would like, please contact one of the
 * authors or someone at the www.j3d.org web site.
 ******************************************************************************/

//*****************************************************************************
/**
 * Java3dApplet
 
 * Base class for defining a Java 3D applet. Contains some useful methods for
 * defining views and scenegraphs etc.
 
 @author Daniel Selman
 @version 1.0
 */
//*****************************************************************************

abstract class Java3dApplet extends Applet {
  public static int m_kWidth = 300;

  public static int m_kHeight = 300;

  protected String[] m_szCommandLineArray = null;

  protected VirtualUniverse m_Universe = null;

  protected BranchGroup m_SceneBranchGroup = null;

  protected Bounds m_ApplicationBounds = null;

  //  protected com.tornadolabs.j3dtree.Java3dTree m_Java3dTree = null;

  public Java3dApplet() {
  }

  public boolean isApplet() {
    try {
      System.getProperty("user.dir");
      System.out.println("Running as Application.");
      return false;
    catch (Exception e) {
    }

    System.out.println("Running as Applet.");
    return true;
  }

  public URL getWorkingDirectory() throws java.net.MalformedURLException {
    URL url = null;

    try {
      File file = new File(System.getProperty("user.dir"));
      System.out.println("Running as Application:");
      System.out.println("   " + file.toURL());
      return file.toURL();
    catch (Exception e) {
    }

    System.out.println("Running as Applet:");
    System.out.println("   " + getCodeBase());

    return getCodeBase();
  }

  public VirtualUniverse getVirtualUniverse() {
    return m_Universe;
  }

  //public com.tornadolabs.j3dtree.Java3dTree getJ3dTree() {
  //return m_Java3dTree;
  //  }

  public Locale getFirstLocale() {
    java.util.Enumeration e = m_Universe.getAllLocales();

    if (e.hasMoreElements() != false)