The Java 3D API Specification |
A P P E N D I X H |
The Example Programs |
THIS appendix describes the example programs on the CD-ROM.
H.1
Before you can compile Java 3D applications or run the example programs, you need to have installed or you need to install the following software on your system: TheIntroduction demo/java3d
directory contains 37 subdirectories. All but two of these directories (geometry
andimages
) contain at least one example program. Some directories contain several example programs.H.2
All of the example programs can be run as standalone applications from a UNIX shell or a DOS window. The syntax shown in this appendix is for UNIX. In DOS windows, you will need to replace "Running the Example Programs /
" with "\
" and specify the correct drive letter when referring to<jdkhome>
(for example, "c:\jdk1.2.2
").
% java HelloUniverse
Each of the other example programs can be run in a similar manner.
% java -Xmx64m HelloUniverse
H.2.1
Java 2 applets, including many Java 3D example programs, can be run within a browser using Java Plug-in. Special HTML tags are required to cause Netscape or Internet Explorer to use the correct version of the Java 2 platform via Java Plug-in. All of the Java 3D example programs that can be run as applets include HTML files that have been converted to use Java Plug-in. Refer to the following URL for information on using Java Plug-in 1.2.2 HTML Converter to convert your own applets to run in a browser:Running within a Browser http://java.sun.com/products/plugin/On Windows, the Java Plug-in is automatically installed along with the Java 2 run-time environment. Applets can be run in either Netscape Communicator or Internet Explorer.http://www.sun.com/solaris/netscape/Additional patches may be required (see the website for details).The following page contains links to all of the Java 3D demos that can be run as applets:
<jdkhome>/demo/java3d/index.htmlJust click on the link for a given demo to run that demo within your browser.H.2.2
To run Java 3D applets withinRunning within Appletviewer appletviewer
, open the original, unconverted version of the associated HTML file (not the "_plugin
" version). For example,
% appletviewer HelloUniverse.html
% appletviewer -J-Xmx64m HelloUniverse.html
H.3
Several example programs are included in theProgram Descriptions demo/java3d
directory. All of the example programs are described here. Code fragments are listed for a few of the example programs.H.3.1
Directory:AWT_Interaction demo/java3d/AWT_Interaction
The relevant source code fragments from
AWT_Interaction.java
follow:This code creates instance variables for the current angle, the TransformGroup that will be modified, and a button that will trigger the modification. The AWTInteraction class implements ActionListener so that it can receive button press events. The
public class AWTInteraction extends Applet implements ActionListener { TransformGroup objTrans; float angle = 0.0f; Transform3D trans = new Transform3D(); Button rotateB = new Button("Rotate");createSceneGraph
method (not shown here) creates a root BranchGroup, an object TransformGroup, and a color cube, much as in HelloUniverse.The constructor puts the Rotate button in a panel and adds the AWTInteraction class as an action listener for the button.
public AWTInteraction() { ... Panel p = new Panel(); p.add(rotateB); add("North", p); rotateB.addActionListener(this); ... }The
public void actionPerformed(ActionEvent e) { if (e.getSource() == rotateB) { angle += Math.toRadians(10); trans.rotY(angle); objTrans.setTransform(trans); } }actionPerformed
method increments theangle
, computes a new rotation matrix, and updates the object's TransformGroup. Since this is in an AWT event listener method rather than in a behavior, the transform update is not synchronized with the Java 3D renderer. In particular, if such an event listener method modifies two objects in the Java 3D scene graph, there is no guarantee that the effects of those two updates will be seen in the same frame. Also remember that it is never safe for two threads to modify the same Java 3D object simultaneously. This means that an object that is being updated by a behavior must not be modified by an AWT event listener.H.3.2
Directory:AlternateAppearance demo/java3d/AlternateAppearance
H.3.3
Directory:Appearance demo/java3d/Appearance
The above code, extracted from the
int row, col; Appearance[][] app = new Appearance[3][3]; for (row = 0; row < 3; row++) for (col = 0; col < 3; col++) app[row][col] = createAppearance(row * 3 + col); for (int i = 0; i < 3; i++) { double ypos = (double)(i - 1) * 0.6; for (int j = 0; j < 3; j++) { double xpos = (double)(j - 1) * 0.6; objRoot.addChild(createObject(app[i][j], 0.12, xpos, ypos)); } }createSceneGraph
method, creates a 3 x 3 array of objects, each with its own Appearance, and adds them to the scene graph.
private Appearance createAppearance(int idx) { Appearance app = new Appearance(); // Globally used colorsColor3f black = new Color3f(0.0f, 0.0f, 0.0f); Color3f white = new Color3f(1.0f, 1.0f, 1.0f); switch (idx) { ... case 4: // Set up the texture mapTextureLoader tex = new TextureLoader(texImage, this); app.setTexture(tex.getTexture()); TextureAttributes texAttr = new TextureAttributes(); texAttr.setTextureMode(TextureAttributes.MODULATE); app.setTextureAttributes(texAttr); app.setMaterial(new Material(white, black, white, black, 1.0f)); break;For Appearance number 4, the TextureLoader utility is used to load a JPEG image from a file and create a Texture object. TextureAttributes are set up so that the lit color is blended with the texture map (MODULATE
). Finally, a lighting Material object is created with a default color of white.
case 5: // Set up the transparency propertiesTransparencyAttributes ta = new TransparencyAttributes(); ta.setTransparencyMode(ta.BLENDED); ta.setTransparency(0.6f); app.setTransparencyAttributes(ta); // Set up the polygon attributesPolygonAttributes pa = new PolygonAttributes(); pa.setCullFace(pa.CULL_NONE); app.setPolygonAttributes(pa); // Set up the material propertiesColor3f objColor = new Color3f(0.7f, 0.8f, 1.0f); app.setMaterial(new Material(objColor, black, objColor, black, 1.0f)); break; ... return app; }For Appearance number 5, TransparencyAttributes are set up to use blended transparency with the object being 60 percent transparent. Back face culling is disabled in the PolygonAttributes object so that the front and back faces of the transparent object are visible. Finally, a lighting Material object is created with the specified object color.H.3.4
Directory:AppearanceMixed demo/java3d/AppearanceMixed
The
static class MyCanvas3D extends Canvas3D { private GraphicsContext3D gc; ... private IndexedTriangleArray tri = new IndexedTriangleArray(4, IndexedTriangleArray.COORDINATES | IndexedTriangleArray.NORMALS, 6); private Point3f vert[]; private Vector3f normals[]; public void renderField(int fieldDesc) { computeVert(); computeNormals(); gc.draw(tri); } private void computeVert() { <modify vert[] array> ... tri.setCoordinates(0, vert); } private void computeNormals() { <compute new normals[] based on vert[] values> ... tri.setNormals(0, normals); }renderField
method is called by Java 3D during the rendering loop for each frame. The AppearanceMixed example overrides this Canvas3D method to compute a new set of vertices and normals for the pair of triangles and to draw the triangles to the canvas. ThecomputeVert
andcomputeNormals
update thevert
andnormals
array and then call the methods to copy these changes to the IndexedTriangleArray object.
public MyCanvas3D(GraphicsConfiguration gcfg) { super(gcfg); ... // Set up the graphics contextgc = getGraphicsContext3D(); // Create the appearance for the triangle fanAppearance app = new Appearance(); ... gc.setAppearance(app); // Set up the global lightsColor3f lColor1 = new Color3f(0.7f, 0.7f, 0.7f); Vector3f lDir1 = new Vector3f(-1.0f, -1.0f, -1.0f); Color3f alColor = new Color3f(0.2f, 0.2f, 0.2f); gc.addLight(new AmbientLight(alColor)); gc.addLight(new DirectionalLight(lColor1, lDir1)); }The constructor for MyCanvas creates a Graphics3D object and initializes its appearance and lights. Note that even though the scene graph also contains light objects, they are not used for immediate mode rendering-lights must be created and added to the graphics context in order for immediate mode geometry to be lit.H.3.5
Directory:Background demo/java3d/Background
H.3.6
Directory:Billboard demo/java3d/Billboard
Note: Billboard's functionality has largely been superseded by OrientedShape3D.
H.3.7
Directory:ConicWorld demo/java3d/ConicWorld
This directory contains a README file and six demonstration programs:
These programs demonstrate the use of the geometry primitives in thecom.sun.j3d.utils.geometry
package.H.3.8
Directory:FourByFour demo/java3d/FourByFour
H.3.9
Directory:GearTest demo/java3d/GearTest
H.3.10
Directory:GeometryByReference demo/java3d/GeometryByReference
H.3.11
Directory:GeometryCompression demo/java3d/GeometryCompression
java cgview <.cg file> [object-number]You can use one of the .cg resource files in thedemo/java3d/geometry
directory. The following example will display a galleon (ship):
% java cgview ../geometry/galleon.cgjava obj2cg <.obj file> [<.obj file> ...] <.cg file>If the .cg file does not exist, it is created. If the file does exist and is a valid .cg resource file, the new object(s) are appended to the objects in the existing file. If it is not a valid .cg file, an exception is thrown.H.3.12
Directory:HelloUniverse demo/java3d/HelloUniverse
The HelloUniverse program creates a cube and a RotationInterpolator behavior object that rotates the cube at a constant rate of /2 radians per second. The code for this program is described in Section 1.6.3, "HelloUniverse: A Sample Java 3D Program."
H.3.13
Directory:LOD demo/java3d/LOD
H.3.14
Directory:Lightwave demo/java3d/Lightwave
directory
% java Viewer ballcone.lws
will display a red cone moving behind a stationary green ball.
H.3.15
Directory:ModelClip demo/java3d/ModelClip
H.3.16
Directory:Morphing demo/java3d/Morphing
H.3.17
Directory:ObjLoad demo/java3d/ObjLoad
The ObjLoad program loads Wavefront object files. Run the program with the following command:
java ObjLoad [-s] [-n] [-t] [-c degrees] <.obj file>You can use one of the .obj files in the
- where the options are
-s
Spin (no user interaction)
-n
No triangulation
-t
No stripification
-c
Set crease angle for normal generation (default is 60 without smoothing group info, otherwise 180 within smoothing groups)
demo/java3d/geometry
directory. The following example will display a galleon (ship):
% java ObjLoad ../geometry/galleon.objThe relevant source code fragment from
ObjLoad.java
follows:The above code fragment creates an ObjectFile loader with the desired flags and crease angle, loads the specified filename (checking for file and parsing exceptions), and adds the loaded objects to the scene graph. This code fragment could easily be modified to accommodate a variety of loaders.
public BranchGroup createSceneGraph(String args[]) { ... int flags = ObjectFile.RESIZE; ... ObjectFile f = new ObjectFile(flags, (float)(creaseAngle * Math.PI / 180.0)); Scene s = null; try { s = f.load(filename); } catch (FileNotFoundException e) { System.err.println(e); System.exit(1); } catch (ParsingErrorException e) { System.err.println(e); System.exit(1); } catch (IncorrectFormatException e) { System.err.println(e); System.exit(1); } objTrans.addChild(s.getSceneGroup());H.3.18
Directory:OffScreenCanvas3D demo/java3d/OffScreenCanvas3D
H.3.19
Directory:OrientedShape3D demo/java3d/OrientedShape3D
H.3.20
Directory:PackageInfo demo/java3d/PackageInfo
H.3.21
Directory:PickTest demo/java3d/PickTest
The IntersectInfoBehavior class constructor creates a new PickCanvas object, initializes the PickCanvas with the desired tolerance, and sets the mode to allow geometry intersection information to be retrieved.
PickCanvas pickCanvas; PickResult[] pickResult; public IntersectInfoBehavior(Canvas3D canvas3D, BranchGroup branchGroup, float size) { pickCanvas = new PickCanvas(canvas3D, branchGroup); pickCanvas.setTolerance(5.0f); pickCanvas.setMode(PickCanvas.GEOMETRY_INTERSECT_INFO); ...
public void processStimulus(Enumeration criteria) { ... <check for mouse event>if (eventId == MouseEvent.MOUSE_PRESSED) { int x = ((MouseEvent)event[i]).getX(); int y = ((MouseEvent)event[i]).getY(); pickCanvas.setShapeLocation(x, y); Point3d eyePos = pickCanvas.getStartPosition(); pickResult = pickCanvas.pickAllSorted(); if (pickResult != null) { // Get node infoNode curNode = pickResult[0].getObject(); Geometry curGeom = ((Shape3D)curNode).getGeometry(); GeometryArray curGeomArray = (GeometryArray) curGeom; // Get closest intersection resultsPickIntersection pi = pickResult[0].getClosestIntersection(eyePos); <get specific info from PickIntersection>} } ...The processStimulus method checks for a mouse event and initiates a pick, via the PickCanvas object, at the selected location. It then looks for pick results and extracts the intersection information from the pick result (if any).H.3.22
Directory:PickText3D demo/java3d/PickText3D
H.3.23
Directory:PlatformGeometry demo/java3d/PlatformGeometry
H.3.24
Directory:PureImmediate demo/java3d/PureImmediate
The relevant source code fragments from
PureImmediate.java
follow:
public class PureImmediate extends Applet implements Runnable { private Canvas3D canvas; private GraphicsContext3D gc = null; private Geometry cube = null; private Transform3D cmt = new Transform3D(); // One rotation (2*PI radians) every 6 secondsprivate Alpha rotAlpha = new Alpha(-1, 6000);The above code creates instance variables for a Canvas3D, the GraphicsContext associated with the canvas, a geometry object for drawing, a Transform3D object for rotation, and an alpha object to allow the rotation to be time-based. The PureImmediate class implements Runnable so that it can be run by a user-created drawing thread.
public void render() { if (gc == null) { // Set up Graphics contextgc = canvas.getGraphicsContext3D(); gc.setAppearance(new Appearance()); // Set up geometrycube = new ColorCube(0.4).getGeometry(); } // Compute angle of rotation based on alpha valuedouble angle = rotAlpha.value() * 2.0*Math.PI; cmt.rotY(angle); // Render the geometry for this framegc.clear(); gc.setModelTransform(cmt); gc.draw(cube); canvas.swap(); } public void run() { while (true) { render(); Thread.yield(); } }Therender
method renders a single frame. After ensuring that the graphics context is set up and the geometry is created, it computes the new rotation matrix, clears the canvas, draws the cube, and swaps the draw and display buffer. Therun
method is the entry point for our drawing thread. It calls therender
method in a loop, yielding control to other threads once per frame.
public PureImmediate() { <set layout of applet, get best graphics config>canvas = new Canvas3D(config); canvas.stopRenderer(); add("Center", canvas); // Create the universe and viewing branchSimpleUniverse u = new SimpleUniverse(canvas); u.getViewingPlatform().setNominalViewingTransform(); // Start a new thread that will continuously rendernew Thread(this).start(); }The constructor creates the 3D canvas, stops the Java 3D renderer, sets up the viewing branch (necessary even in pure immediate mode), and starts up the drawing thread.H.3.25
Directory:ReadRaster demo/java3d/ReadRaster
H.3.26
Directory:Sound demo/java3d/Sound
H.3.27
Directory:SphereMotion demo/java3d/SphereMotion
java SphereMotion [-dir | -point | -spot]H.3.28
Directory:SplineAnim demo/java3d/SplineAnim
The SplineAnim program demonstrates the use of KBRotPosScaleSplinePathInterpolator (see the description in Table F-10) to do spline animation paths using Kochanek-Bartels splines. A red cone is animated along a spline path specified by five knot points, which are shown as cyan spheres. Use the mouse to manipulate the scene.
H.3.29
Directory:Text2D demo/java3d/Text2D
The Text2DTest program shows three different types of 2D text using the Text2D utility class.
H.3.30
Directory:Text3D demo/java3d/Text3D
java Text3DLoadThe[-f fontname] [-t tesselation] <text>
fontname
variable allows you to specify one of the Java Font names, such as Helvetica, Times Roman, and Courier. Thetesselation
variable specifies how finely to tessellate the font glyphs. Once the text displays, the left mouse button rotates the text, the middle button zooms, and the right button translates.H.3.31
Directory:TextureByReference demo/java3d/TextureByReference
H.3.32
Directory:TextureTest demo/java3d/TextureTest
java TextureImageThe<image-filename> [-f ImageComponent format]
ImageComponent format
variable allows you to specify the format of the pixel data. If you do not specify an image file, the rotating cube will appear white. You can use one of the image files in thedemo/java3d/images
directory. For example:
% java TextureImage ../images/earth.jpgwill display the rotating cube with an image of earth mapped onto it.
H.3.33
Directory:TickTockCollision demo/java3d/TickTockCollision
H.3.34
Directory:TickTockPicking demo/java3d/TickTockPicking
H.3.35
Directory:VirtualInputDevice demo/java3d/VirtualInputDevice
The Java 3D API Specification |