3D Graphics Java

/* Author Claude G. Schwab 
 * Copyright (c) 2005  University of applied sciences 
 * Biel School of Engineering and Architecture, Switzerland.
 * http://www.hta-bi.bfh.ch
 * All Rights Reserved.
 * Compiled with SDK 1.4.1 and Java3D API Version 1.3
 *
 * This Demo class is a demonstration software (code) 
 * for my introduction to Java3D.
 */
import java.awt.AWTEvent;
import java.awt.BorderLayout;
import java.awt.Color;
import java.awt.Dimension;
import java.awt.GraphicsConfiguration;
import java.awt.GraphicsEnvironment;
import java.awt.Toolkit;
import java.awt.event.KeyEvent;
import java.awt.event.WindowAdapter;
import java.awt.event.WindowEvent;
import java.awt.event.WindowListener;
import java.text.NumberFormat;
import java.util.Enumeration;
import javax.media.j3d.Alpha;
import javax.media.j3d.AmbientLight;
import javax.media.j3d.Appearance;
import javax.media.j3d.Background;
import javax.media.j3d.Behavior;
import javax.media.j3d.BoundingSphere;
import javax.media.j3d.BranchGroup;
import javax.media.j3d.Canvas3D;
import javax.media.j3d.ColoringAttributes;
import javax.media.j3d.DirectionalLight;
import javax.media.j3d.GeometryArray;
import javax.media.j3d.GraphicsConfigTemplate3D;
import javax.media.j3d.Group;
import javax.media.j3d.LineArray;
import javax.media.j3d.Locale;
import javax.media.j3d.Material;
import javax.media.j3d.PhysicalBody;
import javax.media.j3d.PhysicalEnvironment;
import javax.media.j3d.PointLight;
import javax.media.j3d.PolygonAttributes;
import javax.media.j3d.QuadArray;
import javax.media.j3d.RotationInterpolator;
import javax.media.j3d.Shape3D;
import javax.media.j3d.Switch;
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.WakeupOnAWTEvent;
import javax.media.j3d.WakeupOnElapsedFrames;
import javax.swing.JFrame;
import javax.swing.JLabel;
import javax.swing.JPanel;
import javax.swing.SwingUtilities;
import javax.vecmath.AxisAngle4f;
import javax.vecmath.Color3f;
import javax.vecmath.Point3d;
import javax.vecmath.Point3f;
import javax.vecmath.TexCoord2f;
import javax.vecmath.Vector3d;
import javax.vecmath.Vector3f;
import com.sun.j3d.utils.behaviors.picking.PickRotateBehavior;
import com.sun.j3d.utils.behaviors.picking.PickTranslateBehavior;
import com.sun.j3d.utils.behaviors.picking.PickZoomBehavior;
import com.sun.j3d.utils.geometry.GeometryInfo;
import com.sun.j3d.utils.geometry.NormalGenerator;
import com.sun.j3d.utils.geometry.Sphere;
import com.sun.j3d.utils.geometry.Stripifier;
import com.sun.j3d.utils.image.TextureLoader;
/**
 * This class is the main class of the program Demo. It creates a Virtual
 * Universe and a Locale and attaches to the Locale: a right-handed 3D
 * coordinate system, a color cube, a tetrahedron, the earth and the ViewBranch.
 * It also overrides the addBranchGraph methode.
 */
public class Demo3D extends JFrame implements Runnable {
  static int screenwidth;
  static int screenheight;
  static Thread fpsThread;
  long sleepDuration = 200; // in msec
  int decimalForAllFps = 1;
  JLabel jLabel;
  // Create a virtual universe.
  VirtualUniverse universe = new VirtualUniverse();
  // A single hi-res. Locale node is created and attached to the
  // virtual universe.
  Locale locale = new Locale(universe);
  Canvas3D canvas3D;
  ViewBranch viewBr;
  /**
   * Constructor that allows to specify the desired initial instances.
   */
  public Demo3D() {
    // Set the best GraphicsConfiguration
    GraphicsConfigTemplate3D template = new GraphicsConfigTemplate3D();
    GraphicsConfiguration graphConf = GraphicsEnvironment
        .getLocalGraphicsEnvironment().getDefaultScreenDevice()
        .getBestConfiguration(template);
    canvas3D = new Canvas3D(graphConf); // The used Canvas3D
    // Construction of the main frame.
    setTitle("Demo");
    JPanel jMainPanel = new JPanel(true);
    jMainPanel.setLayout(new BorderLayout(0, 5)); // hor_gap and ver_gap
    JPanel jFpsPanel = new JPanel(true);
    jFpsPanel.setBackground(Color.white);
    jLabel = new JLabel("");
    jLabel.setText("Wait for informations");
    jFpsPanel.add(jLabel);
    jMainPanel.add(canvas3D, BorderLayout.CENTER);
    /*
     * // For the stereo-mode with an "Head Monted Display" (HMD). JPanel
     * jScene_Stereo_Panel = new JPanel(true);
     * jScene_Stereo_Panel.setLayout(new GridLayout(1, 2, 0, 0)); // rows,
     * col, hor_gap and ver_gap jScene_Stereo_Panel.add(canvas3D);
     * jScene_Stereo_Panel.add(canvas3D);
     * jMainPanel.add(jScene_Stereo_Panel, BorderLayout.CENTER);
     */
    jMainPanel.add(jFpsPanel, BorderLayout.SOUTH);
    setContentPane(jMainPanel);
    // The ViewBranch class creates the instances of ViewPlatform, View,
    // etc.
    viewBr = new ViewBranch(canvas3D);
    fpsThread = new Thread(this);
    myScene();
  }
  /**
   * Assembling of all components of the scene.
   */
  public void myScene() {
    // Necessary to use NewTextureLoader in other classes.
    NewTextureLoader.setImageObserver(this); // AWT Component
    // Attach the subgraphs SceneBuilder1, SceneBuilder2, SceneBuilder3
    // and the ViewBranch to the Locale node.
    addBranchGraph(new SceneBuilder1().mySubGraph1());
    addBranchGraph(new SceneBuilder2().mySubGraph2());
    addBranchGraph(new SceneBuilder3(canvas3D).mySubGraph3());
    addBranchGraph(viewBr.myViewBranch());
  }
  /**
   * Allows to attach all subgraphs of the scene and the ViewBranch to the
   * Locale node.
   * 
   * @param javax.media.j3d.BranchGroup
   *            brGr - the root of the subgraph
   */
  public void addBranchGraph(BranchGroup brGr) {
    locale.addBranchGraph(brGr);
  }
  ///////////////////////////// Framemeter /////////////////////////////
  /**
   * This start method allows to start the thread of the framemeter.
   */
  public void start() {
    SwingUtilities.invokeLater(fpsThread);
  }
  /**
   * This run method allows to launch the computation of all frames per second
   * for the framemeter.
   */
  public void run() {
    long lastFrameTime;
    double fps;
    double min = Double.MAX_VALUE;
    double max = Double.MIN_VALUE;
    long count = 0;
    double sum = 0;
    double mean = 0;
    while (true) {
      lastFrameTime = viewBr.view.getLastFrameDuration();
      if (lastFrameTime > 0) {
        fps = 1000 / (double) lastFrameTime;
        count += 1;
        sum += fps;
        mean = sum / count;
        // To format all fps-informations.
        NumberFormat numbForm;
        numbForm = NumberFormat.getInstance();
        numbForm.setMaximumFractionDigits(decimalForAllFps);
        if (min > fps && fps != 0 && count > 4)
          min = fps;
        if (max < fps)
          max = fps;
        jLabel.setText("Frames/sec = " + numbForm.format(fps)
            + "  ;    minFrames/sec = " + numbForm.format(min)
            + "  ;    maxFrames/sec = " + numbForm.format(max)
            + "  ;    meanFrames/sec = " + numbForm.format(mean));
        // System.out.println("Frames per second = " + fps);
      }
      try {
        Thread.sleep(sleepDuration);
      } catch (InterruptedException e) {
      }
    }
  }
  ///////////////////////// End of the framemeter /////////////////////////
  /**
   * Main of the Demo program. Take the graphic environment of the
   * workstation.
   */
  public static void main(String args[]) {
    JFrame jFrameDemo = new Demo3D();
    // To be sure to stop the application when the frame is closed.
    WindowListener winListener = new WindowAdapter() {
      public void windowClosing(WindowEvent event) {
        System.exit(0);
      }
    };
    jFrameDemo.addWindowListener(winListener);
    Dimension screenSize = Toolkit.getDefaultToolkit().getScreenSize();
    screenwidth = (int) screenSize.getWidth();
    screenheight = (int) screenSize.getHeight();
    jFrameDemo.setSize(screenwidth, screenheight);
    // Turn on the visibility of the frame.
    jFrameDemo.setVisible(true);
    fpsThread.start();
  }
}
/*
 * Author Claude G. Schwab Copyright (c) 2002 University of applied sciences
 * Biel School of Engineering and Architecture, Switzerland.
 * http://www.hta-bi.bfh.ch All Rights Reserved. Compiled with SDK 1.4.1 and
 * Java3D API Version 1.3
 * 
 * This ViewBranch class is a demonstration software (code) for my introduction
 * to Java3D.
 */
/**
 * This class creates all necessary objects on the "View Branch" side of the
 * scene graph.
 */
class ViewBranch {
  public BranchGroup vbBrGr;
  private Canvas3D canvas3D;
  private PhysicalBody body;
  private PhysicalEnvironment environment;
  public static View view; // static for the Framemeter
  private ViewPlatform viewPlat;
  private TransformGroup vpTrGrStart, vpTrGrKeys_Transl_Turn,
      vpTrGrKeys_Rot_Up_Down;
  private Transform3D trStart;
  private BoundingSphere cameraBounds;
  private Camera_Transl_Turn camera_Transl_Turn;
  private Camera_Rot_Up_Down camera_Rot_Up_Down;
  private Aimer aimer;
  /**
   * Constructor that allows to specify the desired Canvas3D.
   * 
   * @param javax.media.j3d.Canvas3D
   *            canv - the Canvas3D being used
   */
  public ViewBranch(Canvas3D canv) // The instance canv of Canvas3D class is
  { // created in the constructor of the Demo class.
    canvas3D = canv;
  }
  /**
   * Create the ViewBranch
   * 
   * @return javax.media.j3d.BranchGroup vbBrGr - the root of the ViewBranch
   */
  public BranchGroup myViewBranch() {
    // Create the minimal PhysicalBody and PhysicalEnvironnement
    // instances with default parameters.
    body = new PhysicalBody();
    environment = new PhysicalEnvironment();
    // Create a View instance and attach the Canvas3D, the PhysicalBody
    // and the PhysicalEnvironment to it.
    view = new View();
    view.setFrontClipDistance(0.02); // Default value is 0.1 m
    view.setBackClipDistance(40.0); // Default value is 10 m
    // Rem.: BackClipDistance / FrontClipDistance = 2000 > 1000 but < 3000
    view.addCanvas3D(canvas3D);
    view.setPhysicalBody(body);
    view.setPhysicalEnvironment(environment);
    /*
     * // Choices of the projection type. They are 2 possibilities, namely: //
     * PERSPECTIVE_PROJECTION and PARALLEL_PROJECTION. // Note: the default
     * value is PERSPECTIVE_PROJECTION
     * view.setProjectionPolicy(View.PARALLEL_PROJECTION);
     */
    // Create a ViewPlatform instance and bind it with the View instance.
    viewPlat = new ViewPlatform();
    viewPlat.setActivationRadius(40.0f); // Default value is 62 m
    view.attachViewPlatform(viewPlat);
    // Create the action volume for the camera's navigation.
    cameraBounds = new BoundingSphere(new Point3d(0.0, 0.0, 0.0), 100.0);
    // Create the two necessary TransformGroups for the ViewPlatform's
    // motion (6 translations and 4 rotations).
    vpTrGrKeys_Rot_Up_Down = new TransformGroup();
    vpTrGrKeys_Transl_Turn = new TransformGroup();
    // With the ALLOW_TRANSFORM_READ and ALLOW_TRANSFORM_WRITE
    // capabilities, we allow the modification of the TransformGroup's
    // code by the Behavior's code at run time.
    vpTrGrKeys_Transl_Turn
        .setCapability(TransformGroup.ALLOW_TRANSFORM_READ);
    vpTrGrKeys_Transl_Turn
        .setCapability(TransformGroup.ALLOW_TRANSFORM_WRITE);
    vpTrGrKeys_Rot_Up_Down
        .setCapability(TransformGroup.ALLOW_TRANSFORM_READ);
    vpTrGrKeys_Rot_Up_Down
        .setCapability(TransformGroup.ALLOW_TRANSFORM_WRITE);
    // Attach the ViewPlatform to the vpTrGrKeys_Rot_Up_Down node.
    vpTrGrKeys_Rot_Up_Down.addChild(viewPlat);
    // Create and attach an aimer to the TransformGroup node
    // vpTrGrKeys_Rot_Up_Down.
    aimer = new Aimer(1.5f);
    vpTrGrKeys_Rot_Up_Down.addChild(aimer.myAimer());
    // View-platform's motion ==> camera's navigation: 6 translations and 4
    // rotations.
    // Create and attach the camera's rotation on the vpTrGrKeys_Rot_Up_Down
    // node.
    camera_Rot_Up_Down = new Camera_Rot_Up_Down(vpTrGrKeys_Rot_Up_Down);
    camera_Rot_Up_Down.setSchedulingBounds(cameraBounds);
    vpTrGrKeys_Rot_Up_Down.addChild(camera_Rot_Up_Down);
    // Create and attach the camera's translation and rotation instances
    // on the vpTrGrKeys_Transl_Turn node.
    camera_Transl_Turn = new Camera_Transl_Turn(vpTrGrKeys_Transl_Turn);
    camera_Transl_Turn.setSchedulingBounds(cameraBounds);
    vpTrGrKeys_Transl_Turn.addChild(camera_Transl_Turn);
    // Attach the vpTrGrKeys_Rot_Up_Down node to the vpTrGrKeys_Transl_Turn
    // node.
    vpTrGrKeys_Transl_Turn.addChild(vpTrGrKeys_Rot_Up_Down);
    // Give the starting position of the ViewPlatform.
    trStart = new Transform3D(); // Identity matrix
    trStart.set(new Vector3f(0.0f, 0.0f, 10.0f)); // Translation of the
    // camera (0,0,10)
    // Create the TransformGroup node for the ViewPlatform's
    // starting position.
    vpTrGrStart = new TransformGroup(trStart);
    // Attach the vpTrGrKeys_Transl_Turn node to the TransformGroup
    // node vpTrGrStart.
    vpTrGrStart.addChild(vpTrGrKeys_Transl_Turn);
    // Add the TransformGroup node vpTrGrStart to the view
    // BranchGroup node vbBrGr.
    vbBrGr = new BranchGroup();
    vbBrGr.addChild(vpTrGrStart);
    // Compile the ViewBranch to optimize the performances.
    vbBrGr.compile();
    // Return the final version of the view branch BranchGroup node vbBrGr.
    return vbBrGr;
  }
}
/**
 * This class serves to put a locale right-handed 3D coordinate system as
 * reference system into the scene. It also creates a background, the
 * illumination of the scene and the borders of the virtual universe.
 */
class SceneBuilder1 {
  public BranchGroup brGr1;
  private CoordSyst coordSyst;
  private BoundingSphere boundsBackGr, boundsGen;
  private Background backGr;
  private NewTextureLoader newTextureLoader;
  private AmbientLight ambientLight;
  private PointLight pointLight;
  private DirectionalLight directionalLight;
  private BordersIn bordersIn;
  private BordersOut bordersOut;
  private static final float dimUniverse = 5.0f; // dimensions of the virtual
  // universe are:
  // dimUniverse x dimUniverse x dimUniverse
  /**
   * Create the subgraph #1
   * 
   * @return javax.media.j3d.BranchGroup brGr1 - the root of the subgraph #1
   */
  public BranchGroup mySubGraph1() {
    // Create the BranchGroup brGr1 of the subgraph #1.
    brGr1 = new BranchGroup();
    // Create and attach the coordinate system to the brGr1.
    coordSyst = new CoordSyst(1.0f, 1.0f, 0.0f, // Color of the x-axis
        0.0f, 0.0f, 1.0f, // Color of the y-axis
        1.0f, 0.0f, 0.0f, // Color of the z-axis
        0.75f); // Lenght of the 3 axes
    brGr1.addChild(coordSyst);
    // Background setting for the scene.
    newTextureLoader = new NewTextureLoader("Images/Ciel_Out.jpg");
    newTextureLoader.setImageObserver(newTextureLoader.getImageObserver());
    backGr = new Background(newTextureLoader.getImage());
    backGr.setImageScaleMode(Background.SCALE_FIT_ALL);
    boundsBackGr = new BoundingSphere(new Point3d(0.0, 0.0, 0.0), 1000.0);
    backGr.setApplicationBounds(boundsBackGr);
    brGr1.addChild(backGr);
    // A BoundingSphere instance as general bounding region.
    boundsGen = new BoundingSphere(new Point3d(0.0, 0.0, 0.0), 100.0);
    // Lighting of the scene.
    // Create and attach an ambient light.
    ambientLight = new AmbientLight(true, new Color3f(0.2f, 0.2f, 0.2f));
    ambientLight.setInfluencingBounds(boundsGen);
    brGr1.addChild(ambientLight);
    // Create and attach a point light.
    pointLight = new PointLight(true, new Color3f(1.0f, 1.0f, 0.3f),
        new Point3f(-100.0f, 0.0f, 100.0f), new Point3f(0.0f, 0.05f,
            0.1f));
    pointLight.setInfluencingBounds(boundsGen);
    brGr1.addChild(pointLight);
    // Create and attach a directional light.
    directionalLight = new DirectionalLight(true, new Color3f(0.8f, 1.0f,
        1.0f), new Vector3f(-0.5f, -0.5f, -0.5f));
    directionalLight.setInfluencingBounds(boundsGen);
    brGr1.addChild(directionalLight);
    // Create the borders of the virtual universe for the inside view of the
    // scene.
    bordersIn = new BordersIn(dimUniverse);
    brGr1.addChild(bordersIn.myInternalUniverse());
    // Create the borders of the virtual universe for the outside view of
    // the scene.
    bordersOut = new BordersOut(dimUniverse);
    brGr1.addChild(bordersOut.myExternalUniverse());
    // Compile the subgraph to optimize the performances.
    brGr1.compile();
    // Return the final version of the BranchGroup node brGr1
    return brGr1;
  }
}
/**
 * This class serves to put a colored cube into the scene graph. It also
 * produces a "static rotation" and a "dynamic rotation" of the colored cube.
 */
class SceneBuilder2 {
  public BranchGroup brGr2;
  private BoundingSphere boundsGen;
  private TransformGroup trGr2_1, trGr2_2;
  private CoordSyst coordSyst;
  private ColorCube colorCube;
  private Transform3D trans1, rot1;
  private Alpha rotationAlpha;
  private AxisAngle4f axe_rot;
  private RotationInterpolator rotator;
  /**
   * Create the subgraph #2
   * 
   * @return javax.media.j3d.BranchGroup brGr2 - the root of the subgraph #2
   */
  public BranchGroup mySubGraph2() {
    // Create the BranchGroup node brGr2 of the second subgraph.
    brGr2 = new BranchGroup();
    // A BoundingSphere instance as general bounding region.
    boundsGen = new BoundingSphere(new Point3d(0.0, 0.0, 0.0), 100.0);
    // Create a Transform3D instance rot1 to perform the necessary
    // "static rotation" for the desired cube's position.
    rot1 = new Transform3D();
    // Rotation of Pi/2 - arctan(1/sqrt(2)) = 0.955 rad about the
    // (1,0,-1)-axis passing through the origin.
    axe_rot = new AxisAngle4f(1.0f, 0.0f, -1.0f, 0.955f);
    rot1.setRotation(axe_rot);
    // Create the first TransformGroup node trGr2_1 and attach the
    // "static rotation" rot1 instance to it.
    trGr2_1 = new TransformGroup(rot1);
    // Create and attach a coordinate system to the TransformGroup node
    // trGr2_1 of the subgraph #2, that is to the cube.
    coordSyst = new CoordSyst(1.0f, 1.0f, 0.0f, // Color of the x-axis
        0.0f, 0.0f, 1.0f, // Color of the y-axis
        1.0f, 0.0f, 0.0f, // Color of the z-axis
        0.4f); // Lenght of the 3 axes
    trGr2_1.addChild(coordSyst);
    // Create the ColorCube (Shape3D) and attach it to the
    // TransformGroup node trGr2_1 of the subgraph #2.
    colorCube = new ColorCube(0.5f);
    trGr2_1.addChild(colorCube);
    // Create the second TransformGroup node trGr2_2.
    trGr2_2 = new TransformGroup();
    // With the ALLOW_TRANSFORM_WRITE capability, we allow the
    // modification of the TransformGroup's code by the behavior's
    // code at run time.
    trGr2_2.setCapability(TransformGroup.ALLOW_TRANSFORM_WRITE);
    // Attach the first node trGr2_1 to the second node trGr2_2.
    trGr2_2.addChild(trGr2_1);
    // Prepare the RotationInterpolator (Behavior) for the
    // cube's rotation about the y-axis.
    trans1 = new Transform3D();
    // Create the alpha(t) function.
    rotationAlpha = new Alpha(-1, Alpha.INCREASING_ENABLE, 0, 0, 10000, 0,
        0, 0, 0, 0);
    // Create the cube's rotation about the y-axis.
    rotator = new RotationInterpolator(rotationAlpha, trGr2_2, trans1,
        0.0f, (float) Math.PI * 2.0f);
    rotator.setSchedulingBounds(boundsGen);
    trGr2_2.addChild(rotator);
    brGr2.addChild(trGr2_2);
    // Compile the subgraph to optimize the performances.
    brGr2.compile();
    // Return the final version of the BranchGroup node brGr2
    return brGr2;
  }
}
/**
 * This class serves to attache both the BranchGraph31 and BranchGraph32 to the
 * Locale and to pick the tetrahedron.
 */
/*
 * Note: It is not always necessary to use "detach" and "add" to add/remove a
 * subgraph from a scene graph. In many cases the using of the setEnable()
 * method, to turn a subgraph on and off, is adequate.
 */
class SceneBuilder3 {
  public BranchGroup brGr3;
  private AddDetachEarthBehavior addDetachEarthBehavior;
  private BoundingSphere boundsGen, pickBounds;
  private Canvas3D canvas3D; // needed 3 times for the Picking of the
  // tetrahedron
  private PickRotateBehavior pickRotBehavior;
  private PickZoomBehavior pickZoomBehavior;
  private PickTranslateBehavior pickTransBehavior;
  private SceneBuilder31 sceneBuilder31;
  private SceneBuilder32 sceneBuilder32;
  private Tetrahedron tetrahedron;
  /**
   * Constructor that allows to specify the desired Canvas3D.
   * 
   * @param javax.media.j3d.Canvas3D
   *            canv - the active Canvas3D
   */
  public SceneBuilder3(Canvas3D canv) {
    canvas3D = canv;
  }
  /**
   * Create the subgraph #3
   * 
   * @return javax.media.j3d.BranchGroup brGr3 - the root of the subgraph #3
   */
  public BranchGroup mySubGraph3() {
    // Create the BranchGroup node brGr3, in other words the root of
    // the subgraph31 and subgraph32.
    brGr3 = new BranchGroup();
    // To allow the detach/add process of the subgraph 32 from the
    // BranchGroup node brGr3.
    brGr3.setCapability(Group.ALLOW_CHILDREN_READ);
    brGr3.setCapability(Group.ALLOW_CHILDREN_WRITE);
    brGr3.setCapability(Group.ALLOW_CHILDREN_EXTEND);
    // A BoundingSphere instance as picking bound region.
    pickBounds = new BoundingSphere(new Point3d(0.0, 0.0, 0.0), 5.0);
    // A BoundingSphere instance as general bounding region.
    boundsGen = new BoundingSphere(new Point3d(0.0, 0.0, 0.0), 100.0);
    // Create and attach the subgraph31 with the tetrahedron
    // to the BranchGroup node brGr3.
    tetrahedron = new Tetrahedron(1.0f);
    sceneBuilder31 = new SceneBuilder31(tetrahedron.myTetrahedron());
    brGr3.addChild(sceneBuilder31.mySubGraph31());
    // Picking of the tetrahedron
    // Note:It's the instruction:
    //      trGr31.setCapability(TransformGroup.ENABLE_PICK_REPORTING)
    //      in the class SceneBuilder31 that determines if the
    //      tetrahedron is pickable or not.
    // Pick and translate the tetrahedron parallel to the z-axis if the
    // mouse pointer is over it.
    pickZoomBehavior = new PickZoomBehavior(brGr3, canvas3D, pickBounds);
    // pickZoomBehavior.setEnable(ctrlDown);
    brGr3.addChild(pickZoomBehavior);
    // Pick and translate the tetrahedron in the (x-y)-plane if the
    // mouse pointer is over it.
    pickTransBehavior = new PickTranslateBehavior(brGr3, canvas3D,
        pickBounds);
    // pickTransBehavior.setEnable(ctrlDown);
    brGr3.addChild(pickTransBehavior);
    // Pick and rotate the tetrahedron if the mouse pointer is over it.
    pickRotBehavior = new PickRotateBehavior(brGr3, canvas3D, pickBounds);
    // pickRotBehavior.setEnable(ctrlDown);
    brGr3.addChild(pickRotBehavior);
    // Create the subgraph32 ===> the earth in double rotation.
    sceneBuilder32 = new SceneBuilder32();
    brGr3.addChild(sceneBuilder32.mySubGraph32());
    // Create an instance of the AddDetachEarthBehavior class to
    // allow the detach/add process of the subgraph32.
    addDetachEarthBehavior = new AddDetachEarthBehavior(this,
        sceneBuilder32);
    addDetachEarthBehavior.setSchedulingBounds(boundsGen);
    brGr3.addChild(addDetachEarthBehavior);
    // Compile the subgraph to optimize the performances.
    brGr3.compile();
    // Return the final version of the BranchGroup node brGr3
    return brGr3;
  }
  /**
   * This method is called up in the DetachEathBehavior class to add a new
   * representation of the earth.
   */
  public void addEarth() {
    brGr3.addChild(sceneBuilder32.mySubGraph32());
  }
}
/**
 * This class serves to put the objet (a tetrahedron) into the scene. It also
 * produces a "static translation" of the tetrahedron and allows its picking.
 */
class SceneBuilder31 {
  public TransformGroup trGr31;
  private Shape3D myObject;
  private Transform3D transl;
  private Vector3d vectransl; // translation
  /**
   * Constructor that allows to specify the desired object.
   * 
   * @param javax.media.j3d.Shape3D
   *            objet - the Shape3D instance which will be attached to the
   *            subgraph #31
   */
  public SceneBuilder31(Shape3D object) {
    myObject = object;
  }
  /**
   * Create the subgraph #31 and prepare the TransformGroup node trGr31 for
   * the tetrahedron's picking.
   * 
   * @return javax.media.j3d.TransformGroup trGr31_2 - the root of the
   *         subgraph #31
   */
  public TransformGroup mySubGraph31() {
    // Create a Transform3D node to execute the desired "static translation"
    // of the tetrahedron ===> start position.
    transl = new Transform3D();
    vectransl = new Vector3d(0.0, -2.0, 0.0); // translation
    transl.set(vectransl);
    // Create the TransformGroup node trGr31, attach into it the "static
    // translation" instance and prepare it for the picking.
    trGr31 = new TransformGroup(transl);
    trGr31.setCapability(TransformGroup.ALLOW_TRANSFORM_READ);
    trGr31.setCapability(TransformGroup.ALLOW_TRANSFORM_WRITE);
    trGr31.setCapability(TransformGroup.ENABLE_PICK_REPORTING);
    // Attach myObject (Shape3D leaf) to the TransformGroup node trGr31.
    trGr31.addChild(myObject);
    // Return the final version of the TransformGroup node trGr31.
    return trGr31;
  }
}
/**
 * This class serves to put the rendering of five different earth's
 * representations (each with a right-handed 3D coordinate system) into the
 * scene graph. It also produces a "static translation" and a double "dynamic
 * rotation" of each earth's representations and allows to detach and then add
 * again this subgraph32 from to the entire scene graph.
 */
/*
 * Note: It is not always necessary to use "detach" and "add" to detach/add a
 * subgraph from a scene graph. In many cases the using of the setEnable()
 * method, to turn a subgraph on and off, is adequate.
 */
class SceneBuilder32 {
  public BranchGroup brGr32;
  private TransformGroup trGr32_1, trGr32_2, trGr32_3;
  private int thisEarth;
  private BoundingSphere boundsGen;
  private CoordSyst coordSyst;
  private SwitchBehavior switchBehavior;
  private Switch switchEarths; // the Switch for the 5 different earth's
  // representations
  private Transform3D transl;
  private Vector3d vectTransl; // translation
  private Alpha rotationAlpha_1, rotationAlpha_2;
  private RotationInterpolator rotator_1, rotator_2;
  // The five different earth's representations
  private Earth earth_Points, earth_Lines, earth_Polygons, earth_Gouraud,
      earth_Texture;
  /**
   * Create the subgraph #32
   * 
   * @return javax.media.j3d.TransformGroup trGr32_3 - the root of the
   *         subgraph #32
   */
  public BranchGroup mySubGraph32() {
    // A BoundingSphere instance as general bounding region.
    boundsGen = new BoundingSphere(new Point3d(0.0, 0.0, 0.0), 100.0);
    // Create the first TransformGroup node trGr32_1 to:
    // 1) attach the Switch node with the five different earth's
    //    representations to the subgraph32
    // 2) attach a coordinate system to each earth's representation
    // 3) rotate each earth about its own y-axis.
    trGr32_1 = new TransformGroup();
    // With the ALLOW_TRANSFORM_WRITE capability, we allow the
    // modification of the TransformGroup's code by the behavior's
    // code at run time.
    trGr32_1.setCapability(TransformGroup.ALLOW_TRANSFORM_WRITE);
    // SwitchBehavior is the class which controls the fonctioning of
    // the switchEarths node.
    switchBehavior = new SwitchBehavior(this);
    switchBehavior.setSchedulingBounds(boundsGen);
    trGr32_1.addChild(switchBehavior);
    // The Switch which allows the rendering of the five different
    // earth's representations.
    switchEarths = new Switch();
    // With the ALLOW_TRANSFORM_WRITE, ALLOW_SWITCH_WRITE and
    // ALLOW_CHILDREN_READ
    // capabilities we allow to get or set new capabilities.
    switchEarths.setCapability(Switch.ALLOW_SWITCH_READ);
    switchEarths.setCapability(Switch.ALLOW_SWITCH_WRITE);
    switchEarths.setCapability(Switch.ALLOW_CHILDREN_READ);
    // Attach the different earth's representations to the Switch node.
    // Increasing
    earth_Points = new Earth("points", 0.4f);
    switchEarths.addChild(earth_Points.myEarth()); // # 0
    earth_Lines = new Earth("lines", 0.4f);
    switchEarths.addChild(earth_Lines.myEarth()); // # 1
    earth_Polygons = new Earth("polygons", 0.4f);
    switchEarths.addChild(earth_Polygons.myEarth()); // # 2
    earth_Gouraud = new Earth("gouraud", 0.4f);
    switchEarths.addChild(earth_Gouraud.myEarth()); // # 3
    earth_Texture = new Earth("texture", 0.4f);
    switchEarths.addChild(earth_Texture.myEarth()); // # 4
    // Decreasing
    switchEarths.addChild(earth_Texture.myEarth()); // # 4
    switchEarths.addChild(earth_Gouraud.myEarth()); // # 3
    switchEarths.addChild(earth_Polygons.myEarth()); // # 2
    switchEarths.addChild(earth_Lines.myEarth()); // # 1
    switchEarths.addChild(earth_Points.myEarth()); // # 0
    // Attach the Switch node with the five different earth's
    // representations to the TransformGroup node trGr32_1.
    trGr32_1.addChild(switchEarths);
    // Create and attach a coordinate system to the TransformGroup node
    // trGr32_1, that is to each earth's representation.
    coordSyst = new CoordSyst(1.0f, 1.0f, 0.0f, // Color of the x-axis
        0.0f, 0.0f, 1.0f, // Color of the y-axis
        1.0f, 0.0f, 0.0f, // Color of the z-axis
        0.6f); // Lenght of the 3 axes
    trGr32_1.addChild(coordSyst);
    // Create the alpha(t) function for the earth's rotation about
    // its own y-axis.
    rotationAlpha_1 = new Alpha(-1, Alpha.INCREASING_ENABLE, 0, 0, 10000,
        0, 0, 0, 0, 0);
    // Create the earth's rotation about its own y-axis.
    rotator_1 = new RotationInterpolator(rotationAlpha_1, trGr32_1,
        new Transform3D(), 0.0f, (float) Math.PI * 2.0f);
    rotator_1.setSchedulingBounds(boundsGen);
    trGr32_1.addChild(rotator_1);
    // Create a Transform3D instance to execute the desired "static
    // translation" of the earth, that is the rotation radius around
    // the sun.
    transl = new Transform3D();
    vectTransl = new Vector3d(2.5, 0.0, 0.0);
    transl.set(vectTransl);
    // Create the second TransformGroup node trGr32_2 and attach the
    // "static translation" transl to it.
    trGr32_2 = new TransformGroup(transl);
    // Attach the trGr32_1 node to the trGr32_2 node.
    trGr32_2.addChild(trGr32_1);
    // Create the third TransformGroup node trGr32_3 for the earth's
    // rotation around the sun.
    trGr32_3 = new TransformGroup();
    // With the ALLOW_TRANSFORM_WRITE capability, we allow the
    // modification of the TransformGroup's code by the behavior's
    // code at run time.
    trGr32_3.setCapability(TransformGroup.ALLOW_TRANSFORM_WRITE);
    // Attach the trGr32_2 node to the trGr32_3 node.
    trGr32_3.addChild(trGr32_2);
    // Create the alpha(t) function for the earth's rotation around the sun.
    rotationAlpha_2 = new Alpha(-1, Alpha.INCREASING_ENABLE, 0, 0, 20000,
        0, 0, 0, 0, 0);
    // To restart correctly the rotation of the earth around the
    // sun after a detach/add process of the subgraph32 from the
    // BranchGroup node brGr3.
    rotationAlpha_2.setStartTime(System.currentTimeMillis());
    // Create the earth's rotation around the sun.
    rotator_2 = new RotationInterpolator(rotationAlpha_2, trGr32_3,
        new Transform3D(), 0.0f, (float) Math.PI * 2.0f);
    rotator_2.setSchedulingBounds(boundsGen);
    trGr32_3.addChild(rotator_2);
    // To allow the detaching of this subgraph32 from the
    // BranchGroup node brGr3.
    brGr32 = new BranchGroup();
    brGr32.setCapability(BranchGroup.ALLOW_DETACH);
    brGr32.addChild(trGr32_3);
    // Return the final version of the BranchGroup node brGr32.
    return brGr32;
  }
  /**
   * This method is called up in the SwitchBehavior class and gets the new
   * earth's representation which has to be drawn.
   * 
   * @param thisEarth -
   *            the new earth's representation to draw.
   */
  public void setNewEarth(int thisEarth) {
    switchEarths.setWhichChild(thisEarth);
  }
  // This method is called up in the DetachEathBehavior class to
  // detach the momentary representation of the earth.
  public void detachEarth() {
    brGr32.detach();
  }
}
/*
 * @(#)NewTextureLoades.java 1.0 99/10/21
 * 
 * Copyright (c) 1996-1999 Sun Microsystems, Inc. All Rights Reserved.
 * 
 * Sun grants you ("Licensee") a non-exclusive, royalty free, license to use,
 * modify and redistribute this software in source and binary code form,
 * provided that i) this copyright notice and license appear on all copies of
 * the software; and ii) Licensee does not utilize the software in a manner
 * which is disparaging to Sun.
 * 
 * This software is provided "AS IS," without a warranty of any kind. ALL
 * EXPRESS OR IMPLIED CONDITIONS, REPRESENTATIONS AND WARRANTIES, INCLUDING ANY
 * IMPLIED WARRANTY OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE OR
 * NON-INFRINGEMENT, ARE HEREBY EXCLUDED. SUN AND ITS LICENSORS SHALL NOT BE
 * LIABLE FOR ANY DAMAGES SUFFERED BY LICENSEE AS A RESULT OF USING, MODIFYING
 * OR DISTRIBUTING THE SOFTWARE OR ITS DERIVATIVES. IN NO EVENT WILL SUN OR ITS
 * LICENSORS BE LIABLE FOR ANY LOST REVENUE, PROFIT OR DATA, OR FOR DIRECT,
 * INDIRECT, SPECIAL, CONSEQUENTIAL, INCIDENTAL OR PUNITIVE DAMAGES, HOWEVER
 * CAUSED AND REGARDLESS OF THE THEORY OF LIABILITY, ARISING OUT OF THE USE OF
 * OR INABILITY TO USE SOFTWARE, EVEN IF SUN HAS BEEN ADVISED OF THE POSSIBILITY
 * OF SUCH DAMAGES.
 * 
 * This software is not designed or intended for use in on-line control of
 * aircraft, air traffic, aircraft navigation or aircraft communications; or in
 * the design, construction, operation or maintenance of any nuclear facility.
 * Licensee represents and warrants that it will not use or redistribute the
 * Software for such purposes.
 */
/**
 * A texture loading utility that doesn't require an image observer for
 * constructing objects. This class extends the TextureLoader class of the
 * com.sun.j3d.utils.image package.
 *  
 */
class NewTextureLoader extends TextureLoader {
  static java.awt.Component observer;
  /**
   * Specify an object to server as the image observer. Use this method once
   * before constructing any texture loaders.
   * 
   * @param imageObserver
   *            the object to be used in subsequent NewTextureLoader
   *            constuctions
   */
  public static void setImageObserver(java.awt.Component imageObserver) {
    observer = imageObserver;
  }
  /**
   * Retreve the object used as the image observer for NewTextureLoader
   * objects. Use this method when the image observer is needed.
   * 
   * @return the object used in as the image observer in subsequent
   *         NewTextureLoader constuctions
   */
  public static java.awt.Component getImageObserver() {
    return observer;
  }
  // constructors without an image observer argument
  /**
   * Constructs a NewTextureLoader object loading the specified iamge in
   * default (RGBA) format. The an image observer must be set using the
   * setImageObserver() method before using this constructor.
   * 
   * @param image
   *            the image object to load
   */
  public NewTextureLoader(java.awt.Image image) {
    super(image, observer);
  }
  /**
   * Constructs a NewTextureLoader object loading the specified image and
   * option flags in the default (RGBA) format. The an image observer must be
   * set using the setImageObserver() method before using this constructor.
   * 
   * @param image
   *            the image object to load
   * @param flags
   *            the flags to use in construction (e.g. generate mipmap)
   */
  public NewTextureLoader(java.awt.Image image, int flags) {
    super(image, flags, observer);
  }
  /**
   * Constructs a NewTextureLoader object loading the specified file using the
   * specified format. The an image observer must be set using the
   * setImageObserver() method before using this constructor.
   * 
   * @param image
   *            the image object to load
   * @param format
   *            specificaiton of which channels to use (e.g. RGB)
   */
  public NewTextureLoader(java.awt.Image image, java.lang.String format) {
    super(image, format, observer);
  }
  /**
   * Constructs a NewTextureLoader object loading the specified file with
   * specified format and flags. The an image observer must be set using the
   * setImageObserver() method before using this constructor.
   * 
   * @param image
   *            the image object to load
   * @param format
   *            specificaiton of which channels to use (e.g. RGB)
   * @param flags
   *            the flags to use in construction (e.g. generate mipmap)
   */
  public NewTextureLoader(java.awt.Image image, java.lang.String format,
      int flags) {
    super(image, format, flags, observer);
  }
  /**
   * Constructs a NewTextureLoader object loading the specified file using the
   * default format (RGBA). The an image observer must be set using the
   * setImageObserver() method before using this constructor.
   * 
   * @param fname
   *            the name of the file to load
   */
  public NewTextureLoader(java.lang.String fname) {
    super(fname, observer);
  }
  /**
   * Constructs a NewTextureLoader object loading the specified file with the
   * specified flags. The an image observer must be set using the
   * setImageObserver() method before using this constructor.
   * 
   * @param fname
   *            the name of the file to load
   * @param flags
   *            the flags to use in construction (e.g. generate mipmap)
   */
  public NewTextureLoader(java.lang.String fname, int flags) {
    super(fname, flags, observer);
  }
  /**
   * Constructs a NewTextureLoader object loading the specified file using the
   * specified format. The an image observer must be set using the
   * setImageObserver() method before using this constructor.
   * 
   * @param fname
   *            the name of the file to load
   * @param format
   *            specificaiton of which channels to use (e.g. RGB)
   */
  public NewTextureLoader(java.lang.String fname, java.lang.String format) {
    super(fname, format, observer);
  }
  /**
   * Constructs a NewTextureLoader object loading the specified file using the
   * specified format and flags. The an image observer must be set using the
   * setImageObserver() method before using this constructor.
   * 
   * @param fname
   *            the name of the file to load
   * @param format
   *            specificaiton of which channels to use (e.g. RGB)
   * @param flags
   *            the flags to use in construction (e.g. generate mipmap)
   */
  public NewTextureLoader(java.lang.String fname, java.lang.String format,
      int flags) {
    super(fname, format, flags, observer);
  }
  /**
   * Constructs a NewTextureLoader object loading the specified URL using the
   * default format. The an image observer must be set using the
   * setImageObserver() method before using this constructor.
   * 
   * @param url
   *            specifies the URL of the image to load
   */
  public NewTextureLoader(java.net.URL url) {
    super(url, observer);
  }
  /**
   * Constructs a NewTextureLoader object loading the specified URL using the
   * specified flags. The an image observer must be set using the
   * setImageObserver() method before using this constructor.
   * 
   * @param url
   *            specifies the URL of the image to load
   * @param flags
   *            the flags to use in construction (e.g. generate mipmap)
   */
  public NewTextureLoader(java.net.URL url, int flags) {
    super(url, flags, observer);
  }
  /**
   * Constructs a NewTextureLoader object loading the specified URL using the
   * specified format. The an image observer must be set using the
   * setImageObserver() method before using this constructor.
   * 
   * @param url
   *            specifies the URL of the image to load
   * @param format
   *            specificaiton of which channels to use (e.g. RGB)
   */
  public NewTextureLoader(java.net.URL url, java.lang.String format) {
    super(url, format, observer);
  }
  /**
   * Constructs a NewTextureLoader object loading the specified URL using the
   * specified format and flags. The an image observer must be set using the
   * setImageObserver() method before using this constructor.
   * 
   * @param url
   *            specifies the URL of the image to load
   * @param format
   *            specificaiton of which channels to use (e.g. RGB)
   * @param flags
   *            the flags to use in construction (e.g. generate mipmap)
   */
  public NewTextureLoader(java.net.URL url, java.lang.String format, int flags) {
    super(url, format, flags, observer);
  }
} // end of TexturedPlane class
/**
 * This class is a keyboard behavior to control the partly rotation (rotate up
 * and rotate down) of the camera.
 */
class Camera_Rot_Up_Down extends Behavior {
  // The TransformGroup node to modify by the keyboard interaction.
  private TransformGroup target_trGr;
  // Wake up event when a key is pressed.
  private WakeupOnAWTEvent wakeUp = new WakeupOnAWTEvent(KeyEvent.KEY_PRESSED);
  // The key event
  private KeyEvent keyEvent;
  // The angle to turn when the Home key or End key is pressed.
  private float angle = (float) Math.PI / 36;
  private Transform3D myKeyNavTransf3D = new Transform3D();
  private Transform3D rotation = new Transform3D();
  /**
   * Constructor that allows to specify the desired target transform group.
   * 
   * @param javax.media.j3d.TransformGroup
   *            target - the target transform group
   */
  public Camera_Rot_Up_Down(TransformGroup target) {
    target_trGr = target;
  }
  /**
   * Override Behavior's initialize method to setup wakeup criteria.
   */
  public void initialize() {
    wakeupOn(wakeUp);
  }
  /**
   * Override Behavior's stimulus method to handle the event. This method is
   * called when a key on the keyboard has been pressed and operates on the
   * specified transform group to move the camera.
   * 
   * @param Enumeration
   *            criteria - all pressed keys in a list. This will be passed by
   *            the system.
   */
  public void processStimulus(Enumeration criteria) {
    WakeupOnAWTEvent eventToWakeUp;
    AWTEvent[] events;
    if (criteria.hasMoreElements()) {
      // Decode the wakeup criteria
      eventToWakeUp = (WakeupOnAWTEvent) criteria.nextElement();
      events = eventToWakeUp.getAWTEvent();
      keyEvent = (KeyEvent) events[0];
      int keyCode = keyEvent.getKeyCode();
      // Perform our processing
      // Get the initial transformation from target and put it
      // into myKeyNavTransf3D
      target_trGr.getTransform(myKeyNavTransf3D);
      // Not any of the 2 rotations don't act simultaneously.
      switch (keyCode) {
      case KeyEvent.VK_HOME: // Home - rotate up
        rotation.rotX(angle);
        break;
      case KeyEvent.VK_END: // End - rotate down
        rotation.rotX(-angle);
        break;
      default:
        rotation.rotX(0.0f);
      }
      myKeyNavTransf3D.mul(rotation);
      // Return the final transformation myKeyNavTransf3D to target
      target_trGr.setTransform(myKeyNavTransf3D);
    }
    // Set wakeup criteria for next time.
    wakeupOn(wakeUp);
  }
}
/**
 * This class creates an aimer in the ViewBranch.
 */
class Aimer extends Shape3D {
  private float scale_XYZ;
  private LineArray aimer;
  private float scaledExtremites[];
  /**
   * Constructor that allows to specify the desired initial aimer's
   * dimensions.
   * 
   * @param type
   *            float s_XYZ - the scale factor to adjust the aimer's
   *            dimensions
   */
  public Aimer(float s_XYZ) {
    scale_XYZ = s_XYZ;
  }
  /**
   * Construct an aimer.
   * 
   * @return javax.media.j3d.Shape3D myAimer - the constructed aimer.
   */
  public Shape3D myAimer() {
    // Construction of the aimer (LineArray).
    aimer = new LineArray(20, LineArray.COORDINATES | LineArray.COLOR_3);
    // Scalling of the vertices of the aimer using scale_XYZ.
    scaledExtremites = new float[extremites.length];
    for (int i = 0; i < extremites.length; i++)
      scaledExtremites[i] = extremites[i] * scale_XYZ;
    aimer.setCoordinates(0, scaledExtremites);
    aimer.setColors(0, color);
    this.setGeometry(aimer);
    return this;
  }
  // Aimer's geometry
  private static final float extremites[] = {
  // top-front
      0.075f, 0.05f, -1.0f, -0.075f, 0.05f, -1.0f,
      // left-front
      -0.075f, 0.05f, -1.0f, -0.075f, -0.05f, -1.0f,
      // bottom-front
      -0.075f, -0.05f, -1.0f, 0.075f, -0.05f, -1.0f,
      // right-front
      0.075f, -0.05f, -1.0f, 0.075f, 0.05f, -1.0f,
      // top-back
      0.04f, 0.025f, -1.0f, -0.04f, 0.025f, -1.0f,
      // left-back
      -0.04f, 0.025f, -1.0f, -0.04f, -0.025f, -1.0f,
      // bottom-back
      -0.04f, -0.025f, -1.0f, 0.04f, -0.025f, -1.0f,
      // right-back
      0.04f, -0.025f, -1.0f, 0.04f, 0.025f, -1.0f,
      // cross
      -0.04f, 0.025f, -1.0f, 0.04f, -0.025f, -1.0f, -0.04f, -0.025f,
      -1.0f, 0.04f, 0.025f, -1.0f };
  // Colors of the aimer (each vertex in aimer is red).
  float color[] = { 1.0f, 0.0f, 0.0f, 1.0f, 0.0f, 0.0f, // front-frame
      1.0f, 0.0f, 0.0f, 1.0f, 0.0f, 0.0f, 1.0f, 0.0f, 0.0f, 1.0f, 0.0f,
      0.0f, 1.0f, 0.0f, 0.0f, 1.0f, 0.0f, 0.0f,
      1.0f, 0.0f, 0.0f, 1.0f, 0.0f, 0.0f, // back-frame
      1.0f, 0.0f, 0.0f, 1.0f, 0.0f, 0.0f, 1.0f, 0.0f, 0.0f, 1.0f, 0.0f,
      0.0f, 1.0f, 0.0f, 0.0f, 1.0f, 0.0f, 0.0f,
      1.0f, 0.0f, 0.0f, 1.0f, 0.0f, 0.0f, // cross
      1.0f, 0.0f, 0.0f, 1.0f, 0.0f, 0.0f };
}
/**
 * This class creates a simple cube with a given texture to simulate the inside
 * of the virtual universe.
 */
class BordersIn extends Shape3D {
  private QuadArray cube;
  private float scale_XYZ;
  private NewTextureLoader newTextureLoader;
  private Texture texture;
  private TextureAttributes textAttr;
  /**
   * Constructor that allows to specify the desired scale factor of the
   * virtual universe.
   * 
   * @param type
   *            float s_XYZ - the scale factor to adjust the borders of the
   *            virtual universe
   */
  public BordersIn(float s_XYZ) {
    scale_XYZ = s_XYZ;
  }
  /**
   * Construction of the desired borders of the virtual universe (cube).
   * 
   * @return javax.media.j3d.Shape3D myUniverse - the constructed borders of
   *         the virtual universe
   */
  public Shape3D myInternalUniverse() {
    cube = new QuadArray(cubeFaces.length, QuadArray.COORDINATES
        | QuadArray.TEXTURE_COORDINATE_2);
    ////////////////////// Geometric part ///////////////////////////
    // Scaling of the faces.
    for (int i = 0; i < cubeFaces.length; i++)
      cubeFaces[i].scale(scale_XYZ);
    cube.setCoordinates(0, cubeFaces);
    for (int i = 0; i < cubeFaces.length; i++) {
      // With i mod 4 ==> 0 1 2 3 0 1 2 3 0 1 2 3 0 1 2 3 for
      // the 4 vertices of the 6 faces, thus each vertex has
      // a point in the texture space. In this case, each cube's
      // face has the same texture coordinates.
      cube.setTextureCoordinate(0, i, textCoord[i % 4]);
    }
    // The geometry is passed to the instance this of the cube.
    this.setGeometry(cube);
    ////////////////////// Appearance part ///////////////////////////
    Appearance appearance = new Appearance();
    // This code block is only necessary to insure, in all cases, the
    // correct
    // rendering of the 6 faces of the cube (bug in Java3D version 1.2.0 !).
    // Set up the polygon's rendering-mode
    PolygonAttributes polygonAttributes = new PolygonAttributes();
    polygonAttributes.setPolygonMode(PolygonAttributes.POLYGON_FILL);
    appearance.setPolygonAttributes(polygonAttributes);
    // Loading the texture for the 6 cube's faces.
    newTextureLoader = new NewTextureLoader("Images/Galaxies.gif");
    newTextureLoader.setImageObserver(newTextureLoader.getImageObserver());
    texture = newTextureLoader.getTexture();
    appearance.setTexture(texture);
    // Application modes of the texture
    textAttr = new TextureAttributes();
    textAttr.setTextureMode(TextureAttributes.MODULATE); // there still are:
    // BLEND, COMBINE,
    // DECAL, and REPLACE
    appearance.setTextureAttributes(textAttr);
    // The appearance is passed to the instance this of the cube.
    this.setAppearance(appearance);
    return this;
  }
  // The 8 vertices p1, p2, ..., p8 of the cube.
  private static final Point3f p1 = new Point3f(1.0f, 1.0f, 1.0f);
  private static final Point3f p2 = new Point3f(1.0f, -1.0f, 1.0f);
  private static final Point3f p3 = new Point3f(-1.0f, -1.0f, 1.0f);
  private static final Point3f p4 = new Point3f(-1.0f, 1.0f, 1.0f);
  private static final Point3f p5 = new Point3f(-1.0f, 1.0f, -1.0f);
  private static final Point3f p6 = new Point3f(-1.0f, -1.0f, -1.0f);
  private static final Point3f p7 = new Point3f(1.0f, -1.0f, -1.0f);
  private static final Point3f p8 = new Point3f(1.0f, 1.0f, -1.0f);
  // The 6 faces of the cube.
  private static final Point3f cubeFaces[] = { // internal front face
  p5, p6, p7, p8,
  // internal right face
      p1, p8, p7, p2,
      // internal back face
      p1, p2, p3, p4,
      // internal left face
      p4, p3, p6, p5,
      // internal top face
      p1, p4, p5, p8,
      // internal bottom face
      p3, p2, p7, p6 };
  // Coordinates in the texture space. Each cube's face has the
  // same texture coordinates.
  private TexCoord2f textCoord[] = { new TexCoord2f(0.0f, 0.0f),
      new TexCoord2f(0.0f, 1.0f), new TexCoord2f(1.0f, 1.0f),
      new TexCoord2f(1.0f, 0.0f) };
}
/**
 * This class creates a simple cube with a given texture to simulate the outside
 * of the virtual universe.
 */
class BordersOut extends Shape3D {
  private QuadArray cube;
  private float scale_XYZ;
  private NewTextureLoader newTextureLoader;
  private Texture texture;
  private TextureAttributes textAttr;
  /**
   * Constructor that allows to specify the desired scale factor of the
   * virtual universe.
   * 
   * @param type
   *            float s_XYZ - the scale factor to adjust the borders of the
   *            virtual universe
   */
  public BordersOut(float s_XYZ) {
    scale_XYZ = s_XYZ;
  }
  /**
   * Construction of the desired borders of the virtual universe (cube).
   * 
   * @return javax.media.j3d.Shape3D myUniverse - the constructed borders of
   *         the virtual universe
   */
  public Shape3D myExternalUniverse() {
    cube = new QuadArray(cubeFaces.length, QuadArray.COORDINATES
        | QuadArray.TEXTURE_COORDINATE_2);
    ////////////////////// Geometric part ///////////////////////////
    // Scaling of the faces.
    for (int i = 0; i < cubeFaces.length; i++)
      cubeFaces[i].scale(scale_XYZ);
    cube.setCoordinates(0, cubeFaces);
    for (int i = 0; i < cubeFaces.length; i++) {
      // With i mod 4 ==> 0 1 2 3 0 1 2 3 0 1 2 3 0 1 2 3 for
      // the 4 vertices of the 6 faces, thus each vertex has
      // a point in the texture space. In this case, each cube's
      // face has the same texture coordinates.
      cube.setTextureCoordinate(0, i, textCoord[i % 4]);
    }
    // The geometry is passed to the instance this of the cube.
    this.setGeometry(cube);
    ////////////////////// Appearance part ///////////////////////////
    Appearance appearance = new Appearance();
    // This code block is only necessary to insure, in all cases, the
    // correct
    // rendering of the 6 faces of the cube (bug in Java3D version 1.2.0 !).
    // Set up the polygon's rendering-mode
    PolygonAttributes polygonAttributes = new PolygonAttributes();
    polygonAttributes.setPolygonMode(PolygonAttributes.POLYGON_FILL);
    appearance.setPolygonAttributes(polygonAttributes);
    // Loading the texture for the 6 cube's faces.
    newTextureLoader = new NewTextureLoader("Images/Ciel_Outside.jpg");
    newTextureLoader.setImageObserver(newTextureLoader.getImageObserver());
    texture = newTextureLoader.getTexture();
    appearance.setTexture(texture);
    // Application modes of the texture
    textAttr = new TextureAttributes();
    textAttr.setTextureMode(TextureAttributes.MODULATE); // there still are:
    // BLEND, COMBINE,
    // DECAL, and REPLACE
    appearance.setTextureAttributes(textAttr);
    // The appearance is passed to the instance this of the cube.
    this.setAppearance(appearance);
    return this;
  }
  // The 8 vertices p1, p2, ..., p8 of the cube.
  private static final Point3f p1 = new Point3f(1.0f, 1.0f, 1.0f);
  private static final Point3f p2 = new Point3f(1.0f, -1.0f, 1.0f);
  private static final Point3f p3 = new Point3f(-1.0f, -1.0f, 1.0f);
  private static final Point3f p4 = new Point3f(-1.0f, 1.0f, 1.0f);
  private static final Point3f p5 = new Point3f(-1.0f, 1.0f, -1.0f);
  private static final Point3f p6 = new Point3f(-1.0f, -1.0f, -1.0f);
  private static final Point3f p7 = new Point3f(1.0f, -1.0f, -1.0f);
  private static final Point3f p8 = new Point3f(1.0f, 1.0f, -1.0f);
  // The 6 faces of the cube.
  private static final Point3f cubeFaces[] = { // external front face
  p5, p8, p7, p6,
  // external right face
      p8, p1, p2, p7,
      // external back face
      p1, p4, p3, p2,
      // external left face
      p4, p5, p6, p3,
      // external top face
      p8, p5, p4, p1,
      // external bottom face
      p2, p3, p6, p7 };
  // Coordinates in the texture space. Each cube's face has the
  // same texture coordinates.
  private TexCoord2f textCoord[] = { new TexCoord2f(1.0f, 1.0f),
      new TexCoord2f(0.0f, 1.0f), new TexCoord2f(0.0f, 0.0f),
      new TexCoord2f(1.0f, 0.0f) };
}
/**
 * This class is a switch behavior to control the rendering of the five
 * different earth's representations.
 */
class SwitchBehavior extends Behavior {
  // The Alpha class which gives the alpha values to command the Switch node.
  private Alpha switchAlpha;
  private float switchAlphaValue;
  // The class which contains the Switch node for the rendering of
  // the 5 different earth's representations.
  private SceneBuilder32 sceneBuilder32 = null;
  // The earth which will be rendered.
  private int thisEarth = 0;
  // Wakeup event after each frame.
  private WakeupOnElapsedFrames wakeUp = new WakeupOnElapsedFrames(0);
  /**
   * Constructor that allows to specify the reference of the SceneBuilder32's
   * instance.
   * 
   * @param sceneBuilder32 -
   *            the SceneBuilder32 instance
   */
  public SwitchBehavior(SceneBuilder32 sceneBuilder32) {
    super();
    // Create the alpha(t) function to automaticaly switch between the
    // five different earth's representations.
    switchAlpha = new Alpha(-1, Alpha.INCREASING_ENABLE
        | Alpha.DECREASING_ENABLE, 0, 0, 10000, 0, 0, 10000, 0, 0);
    // Get the SceneBuilder32 reference
    this.sceneBuilder32 = sceneBuilder32;
  }
  /**
   * Override Behavior's initialize method to setup wakeup criteria.
   */
  public void initialize() {
    wakeupOn(wakeUp);
  }
  /**
   * Override Behavior's stimulus method to handle the event. This method is
   * called up when the define number of frames is draw.
   * 
   * @param criteria -
   *            the wake-up criteria
   */
  public void processStimulus(Enumeration criteria) {
    switchAlphaValue = switchAlpha.value();
    if (switchAlphaValue <= 0.15f)
      thisEarth = 0;
    else if (0.15f < switchAlphaValue && switchAlphaValue <= 0.4f)
      thisEarth = 1;
    else if (0.4f < switchAlphaValue && switchAlphaValue <= 0.6f)
      thisEarth = 2;
    else if (0.6f < switchAlphaValue && switchAlphaValue <= 0.8f)
      thisEarth = 3;
    else if (0.8f < switchAlphaValue)
      thisEarth = 4;
    sceneBuilder32.setNewEarth(thisEarth);
    // Set wakeup criteria for next time.
    wakeupOn(wakeUp);
  }
}
/**
 * This class creates a simple colored cube.
 */
class ColorCube extends Shape3D {
  private QuadArray cube;
  // The constant colors of each cube's face
  private static final Color3f red = new Color3f(1.0f, 0.0f, 0.0f);
  private static final Color3f green = new Color3f(0.0f, 1.0f, 0.0f);
  private static final Color3f blue = new Color3f(0.0f, 0.0f, 1.0f);
  private static final Color3f yellow = new Color3f(1.0f, 1.0f, 0.0f);
  private static final Color3f magenta = new Color3f(1.0f, 0.0f, 1.0f);
  private static final Color3f cyan = new Color3f(0.0f, 1.0f, 1.0f);
  /**
   * Constructor that allows to specify the desired scale factor for the
   * colored cube.
   * 
   * @param type
   *            float s_XYZ - the scale factor to adjust the edges's length of
   *            the colored cube
   */
  public ColorCube(float scale_XYZ) { // 24
    cube = new QuadArray(cubeFaces.length, QuadArray.COORDINATES
        | QuadArray.COLOR_3);
    // Scaling of vertices
    for (int i = 0; i < cubeFaces.length; i++)
      cubeFaces[i].scale(scale_XYZ);
    cube.setCoordinates(0, cubeFaces);
    cube.setColors(0, colorsFaces);
    this.setGeometry(cube);
  }
  // The 8 vertices p1, ..., p8 of the cube.
  private static final Point3f p1 = new Point3f(1.0f, 1.0f, 1.0f);
  private static final Point3f p2 = new Point3f(-1.0f, 1.0f, 1.0f);
  private static final Point3f p3 = new Point3f(-1.0f, -1.0f, 1.0f);
  private static final Point3f p4 = new Point3f(1.0f, -1.0f, 1.0f);
  private static final Point3f p5 = new Point3f(1.0f, -1.0f, -1.0f);
  private static final Point3f p6 = new Point3f(1.0f, 1.0f, -1.0f);
  private static final Point3f p7 = new Point3f(-1.0f, 1.0f, -1.0f);
  private static final Point3f p8 = new Point3f(-1.0f, -1.0f, -1.0f);
  // The 6 faces of the cube.
  private static final Point3f cubeFaces[] = { // front face
  p1, p2, p3, p4,
  // right face
      p1, p4, p5, p6,
      // back face
      p8, p7, p6, p5,
      // left face
      p2, p7, p8, p3,
      // top face
      p1, p6, p7, p2,
      // bottom face
      p5, p4, p3, p8 };
  // The constant colors for the 6 faces.
  private static final Color3f[] colorsFaces = {// front face
  red, red, red, red,
  // back face
      green, green, green, green,
      // right face
      blue, blue, blue, blue,
      // left face
      yellow, yellow, yellow, yellow,
      // top face
      magenta, magenta, magenta, magenta,
      // bottom face
      cyan, cyan, cyan, cyan };
}
/**
 * This class is used to detach then add again the subgraph 32 from the scene
 * graph.
 */
/*
 * Note: It is not always necessary to use "detach" and "add" to add/remove a
 * subgraph from a scene graph. In many cases the using of the setEnable()
 * method, to turn a subgraph on and off, is adequate.
 */
class AddDetachEarthBehavior extends Behavior {
  // The classe which contains the addEarth method.
  private SceneBuilder3 sceneBuilder3;
  // The classe which contains the detachEarth method.
  private SceneBuilder32 sceneBuilder32;
  // To control the detach / add sequence.
  private boolean validation = false;
  // Wake up event when a key is pressed.
  private WakeupOnAWTEvent wakeUp = new WakeupOnAWTEvent(KeyEvent.KEY_PRESSED);
  // The key event
  private KeyEvent keyEvent;
  /**
   * Constructor that allows to specify the references of the SceneBuilder3's
   * and SceneBuilder32's instances.
   * 
   * @param sceneBuilder3 -
   *            the SceneBuilder3 instance
   * @param sceneBuilder32 -
   *            the SceneBuilder32 instance
   */
  public AddDetachEarthBehavior(SceneBuilder3 sceneBuilder3,
      SceneBuilder32 sceneBuilder32) {
    super();
    // Get the SceneBuilder3's and SceneBuilder32's references
    this.sceneBuilder3 = sceneBuilder3;
    this.sceneBuilder32 = sceneBuilder32;
  }
  /**
   * Override Behavior's initialize method to setup wakeup criteria.
   */
  public void initialize() {
    wakeupOn(wakeUp);
  }
  /**
   * Override Behavior's stimulus method to handle the event. This method is
   * called when a key on the keyboard has been pressed and operates on the
   * specified transform group to move the camera.
   * 
   * @param Enumeration
   *            criteria - all pressed keys in a list. This will be passed by
   *            the system.
   */
  public void processStimulus(Enumeration criteria) {
    WakeupOnAWTEvent eventToWakeUp;
    AWTEvent[] events;
    if (criteria.hasMoreElements()) {
      // Decode the wakeup criteria
      eventToWakeUp = (WakeupOnAWTEvent) criteria.nextElement();
      events = eventToWakeUp.getAWTEvent();
      keyEvent = (KeyEvent) events[0];
      int keyCode = keyEvent.getKeyCode();
      // Perform our processing
      switch (keyCode) {
      case KeyEvent.VK_A:
        if (validation) {
          sceneBuilder3.addEarth();
          System.out.println("===>    Add Earth");
          validation = false;
        }
        break;
      case KeyEvent.VK_D:
        if (!validation) {
          sceneBuilder32.detachEarth();
          System.out.println("===>    Detach Earth");
          validation = true;
        }
        break;
      default:
      }
    }
    // Set wakeup criteria for next time.
    wakeupOn(wakeUp);
  }
}
/**
 * This class is a keyboard behavior to control the translation and the partly
 * rotation (turn left and turn right) of the camera.
 * 
 * After a translation or rotation performed by using this class, the
 * observation direction of the virtual camera will always stay parallel to the
 * coordinate system's (x-y)-plane of the parent node.
 * 
 * In the case of this Demo application, it will be the coordinate system
 * associated with the Locale node.
 */
class Camera_Transl_Turn extends Behavior {
  // The TransformGroup node to modify by the keyboard interaction.
  private TransformGroup target_trGr;
  // Wake up event when a key is pressed.
  private WakeupOnAWTEvent wakeUp = new WakeupOnAWTEvent(KeyEvent.KEY_PRESSED);
  // The key event
  private KeyEvent keyEvent;
  // The angle to turn when a Direction Key and the Shift key are pressed.
  private float angle = (float) Math.PI / 36;
  // The step size to move when a Direction Key or the PageDown or
  // PageUp key is pressed.
  private float step = 1f;
  private Transform3D myKeyNavTransf3D = new Transform3D();
  private Vector3f translation = new Vector3f();
  private Vector3f dtrans = new Vector3f();
  private Vector3f dtrans0 = new Vector3f();
  private Transform3D trans_rot = new Transform3D();
  /**
   * Constructor that allows to specify the desired target transform group.
   * 
   * @param javax.media.j3d.TransformGroup
   *            target - the target transform group
   */
  public Camera_Transl_Turn(TransformGroup target) {
    target_trGr = target;
  }
  /**
   * Override Behavior's initialize method to setup wakeup criteria.
   */
  public void initialize() {
    wakeupOn(wakeUp);
  }
  /**
   * Override Behavior's stimulus method to handle the event. This method is
   * called when a key on the keyboard has been pressed and operates on the
   * specified transform group to move the camera.
   * 
   * @param Enumeration
   *            criteria - all pressed keys in a list. This will be passed by
   *            the system.
   */
  public void processStimulus(Enumeration criteria) {
    WakeupOnAWTEvent eventToWakeUp;
    AWTEvent[] events;
    if (criteria.hasMoreElements()) {
      // Decode the wakeup criteria
      eventToWakeUp = (WakeupOnAWTEvent) criteria.nextElement();
      events = eventToWakeUp.getAWTEvent();
      keyEvent = (KeyEvent) events[0];
      int keyCode = keyEvent.getKeyCode();
      // Perform our processing
      // Get the initial transformation from target and put it into
      // myKeyNavTransf3D
      target_trGr.getTransform(myKeyNavTransf3D);
      // Set the translational components of myKeyNavTransf3D in
      // translation
      myKeyNavTransf3D.get(translation);
      // Not any of the 8 motions (6 translations and 2 rotations)
      // don't act simultaneously.
      switch (keyCode) {
      case KeyEvent.VK_UP: // Up Arrow - to move forward
        trans_rot.set(new Vector3f(0.0f, 0.0f, -step));
        break;
      case KeyEvent.VK_DOWN: // Down Arrow - to move backwards
        trans_rot.set(new Vector3f(0.0f, 0.0f, step));
        break;
      case KeyEvent.VK_LEFT: // Left Arrow -to turn left or move left
        if (keyEvent.isShiftDown())
          trans_rot.set(new Vector3f(-step, 0.0f, 0.0f));
        else
          trans_rot.rotY(angle);
        break;
      case KeyEvent.VK_RIGHT: // Right Arrow - to turn right or move right
        if (keyEvent.isShiftDown())
          trans_rot.set(new Vector3f(step, 0.0f, 0.0f));
        else
          trans_rot.rotY(-angle);
        break;
      case KeyEvent.VK_PAGE_DOWN: // Page Down - to move down
        trans_rot.set(new Vector3f(0.0f, -step, 0.0f));
        break;
      case KeyEvent.VK_PAGE_UP: // Page Up - to move up
        trans_rot.set(new Vector3f(0.0f, step, 0.0f));
        break;
      default:
        trans_rot.set(new Vector3f(0.0f, 0.0f, 0.0f));
      }
      myKeyNavTransf3D.mul(trans_rot);
      // Return the final transformation myKeyNavTransf3D to target
      target_trGr.setTransform(myKeyNavTransf3D);
    }
    // Set wakeup criteria for next time.
    wakeupOn(wakeUp);
  }
}
/**
 * This class creates the Earth by using the Sphere class and a given texture
 * (Earth.jpg).
 */
class Earth {
  public Sphere earth;
  private String renderingType;
  private float scale_XYZ;
  private Color3f diffAmb, reflDiff, reflSpec, emittedLight;
  private float c; //  shininess;
  private Appearance appearance;
  private Material material;
  private ColoringAttributes coloringAttributes;
  private PolygonAttributes polygonAttributes;
  private NewTextureLoader newTextureLoader;
  private Texture texture;
  private TextureAttributes textAttr;
  /**
   * Constructor that allows to specify the desired rendering's mode and the
   * desired scale factor.
   * 
   * @param java.lang.String
   *            nom - the desired type of earth's rendering,
   * @param type
   *            float s_XYZ - the scale factor to adjust the earth's radius
   */
  public Earth(String name, float s_XYZ) {
    renderingType = name;
    scale_XYZ = s_XYZ;
  }
  /**
   * This methode serves to construct the earth.
   * 
   * @return com.sun.j3d.utils.geometry.Sphere earth - the constructed earth
   */
  public Sphere myEarth() {
    // Optical properties of the earth.
    // Ambient-diffuse-reflection coefficient
    diffAmb = new Color3f(1.0f, 1.0f, 1.0f);
    // Diffuse-reflection coefficient
    reflDiff = new Color3f(1.0f, 1.0f, 1.0f);
    // Specular-reflection coefficient (reflectance function)
    reflSpec = new Color3f(0.0f, 0.0f, 0.1f);
    // c = shininess: cos^c in the specular reflection
    c = 1;
    // Emitted light
    emittedLight = new Color3f(0.0f, 0.0f, 0.0f);
    appearance = new Appearance();
    // Create the material and set up the optical properties.
    material = new Material(diffAmb, emittedLight, reflDiff, reflSpec, c);
    appearance.setMaterial(material);
    // Set up the polygon's rendering-mode (with the polygonAttributes) and
    // the shading-mode (with the coloringAttributes).
    polygonAttributes = new PolygonAttributes();
    coloringAttributes = new ColoringAttributes();
    // Points
    if (renderingType.compareTo("points") == 0) {
      polygonAttributes.setPolygonMode(PolygonAttributes.POLYGON_POINT);
    }
    /* Lines*/
    else if (renderingType.compareTo("lines") == 0) {
      polygonAttributes.setPolygonMode(PolygonAttributes.POLYGON_LINE);
    }
    /* Polygons */
    else if (renderingType.compareTo("polygons") == 0) {
      /* is the default value*/
      polygonAttributes.setPolygonMode(PolygonAttributes.POLYGON_FILL); 
      coloringAttributes.setShadeModel(ColoringAttributes.SHADE_FLAT);
    }
    /* Gouraud */
    else if (renderingType.compareTo("gouraud") == 0) {
      polygonAttributes.setPolygonMode(PolygonAttributes.POLYGON_FILL); /* is the default value*/
      coloringAttributes.setShadeModel(ColoringAttributes.SHADE_GOURAUD); /* is the default value*/
    }
    else if (renderingType.compareTo("texture") == 0) {
      polygonAttributes.setPolygonMode(PolygonAttributes.POLYGON_FILL); /* is the default value*/
      coloringAttributes.setShadeModel(ColoringAttributes.SHADE_GOURAUD); /* is the default value*/
      /* Loading of the texture*/
      newTextureLoader = new NewTextureLoader("Images/Earth.jpg");
      newTextureLoader.setImageObserver(newTextureLoader
          .getImageObserver());
      texture = newTextureLoader.getTexture();
      appearance.setTexture(texture);
      /* Application mode of the texture */
      textAttr = new TextureAttributes();
      textAttr.setTextureMode(TextureAttributes.REPLACE); /* there still are:
       BLEND, COMBINE,
       DECAL, and MODULATE*/
      appearance.setTextureAttributes(textAttr);
    }
    appearance.setPolygonAttributes(polygonAttributes);
    appearance.setColoringAttributes(coloringAttributes);
    /* Construction of the earth with all its features.*/
        earth = new Sphere(scale_XYZ, Sphere.GENERATE_NORMALS
        | Sphere.GENERATE_TEXTURE_COORDS, 10, appearance);
    return earth;
  }
}
/**
 * This class creates a tetrahedron with a given texture (Claude.jpg).
 */
class Tetrahedron extends Shape3D {
  private float scale_XYZ;
  private Point3f vertices[];
  private int lengthVertices = 4, lengthTetraFaceIndices = 12; // 4 faces x 3
                                 // vertices
  private int tetraFaceIndices[], textCoordIndices[];
  private GeometryInfo tetra_GeometryInfo;
  private NormalGenerator normalGenerator;
  private Stripifier stripifier;
  private GeometryArray tetra_GeometryArray;
  private TexCoord2f textCoord2f[]; // for triangles
  private boolean crAngle;
  private Appearance appearance;
  private Color3f diffAmb, reflDiff, reflSpec, emittedLight;
  private Material material;
  private TransparencyAttributes trAttr;
  private NewTextureLoader newTextureLoader;
  private Texture texture;
  private TextureAttributes textAttr;
  /**
   * Constructor that allows to specify the desired scale factor.
   * 
   * @param type
   *            float s_XYZ - the scale factor to adjust the edges's length of
   *            the tetrahedron
   */
  public Tetrahedron(float s_XYZ) {
    scale_XYZ = s_XYZ;
  }
  /**
   * Construction of the desired tetrahedron.
   * 
   * @return javax.media.j3d.Shape3D myTetrahedron - the constructed
   *         tetrahedron
   */
  public Shape3D myTetrahedron() {
    ////////////////////// Geometric part ///////////////////////////
    // The 4 vertices p0, p1, p2 and p3 of the tetrahedron.
    vertices = new Point3f[lengthVertices]; // 4
    vertices[0] = new Point3f(0.0f, 0.0f, 0.0f);
    vertices[1] = new Point3f(1.0f, 0.0f, 0.0f);
    vertices[2] = new Point3f(0.0f, 1.0f, 0.0f);
    vertices[3] = new Point3f(0.0f, 0.0f, 1.0f);
    // Scaling of vertices
    for (int i = 0; i < lengthVertices; i++)
      // lengthVertices = 4
      vertices[i].scale(scale_XYZ);
    // Set the face's indices for the tetrahedron (referenced to the array
    // of vertices
    // by setCoordinates(vertices) and
    // setCoordinateIndices(tetraFaceIndices)).
    tetraFaceIndices = new int[lengthTetraFaceIndices]; // 12
    // From the camera in the view coordinate system
    // bottom
    tetraFaceIndices[0] = 0;
    tetraFaceIndices[1] = 1;
    tetraFaceIndices[2] = 3;
    // back-left face
    tetraFaceIndices[3] = 0;
    tetraFaceIndices[4] = 3;
    tetraFaceIndices[5] = 2;
    // back face
    tetraFaceIndices[6] = 0;
    tetraFaceIndices[7] = 2;
    tetraFaceIndices[8] = 1;
    // front face
    tetraFaceIndices[9] = 1;
    tetraFaceIndices[10] = 2;
    tetraFaceIndices[11] = 3;
    // Create the GeometryInfo instance and set the vertices
    tetra_GeometryInfo = new GeometryInfo(GeometryInfo.TRIANGLE_ARRAY);
    tetra_GeometryInfo.setCoordinates(vertices);
    tetra_GeometryInfo.setCoordinateIndices(tetraFaceIndices);
    //      triangulator = new Triangulator(); // only for polygons:
    // POLYGON_ARRAY
    //      triangulator.triangulate(tetra_GeometryInfo); // and with: int
    // stripCounts[]
    //           gi.setStripCounts(...)
    //           int contourCounts[]
    // Set the parameters (1 texture with dimension 2) for the texture's
    // coordinates
    tetra_GeometryInfo.setTextureCoordinateParams(1, 2);
    //    case #1: each face of the tetrahedron has the same texture portion.
    // The coordinates of the 3 points in the 2D texture space.
    textCoord2f = new TexCoord2f[3];
    textCoord2f[0] = new TexCoord2f(0.0f, 0.2f);
    textCoord2f[1] = new TexCoord2f(0.5f, 1.0f);
    textCoord2f[2] = new TexCoord2f(1.0f, 0.5f);
    // Set the texture coordinate's indices (referenced to the array of 2D
    // points
    // in the texture space by setTextureCoordinates(0, textCoord2f) and
    // setTextureCoordinateIndices(0, textCoordIndices)).
    textCoordIndices = new int[lengthTetraFaceIndices]; // 12
    // From the camera in the view coordinate system (inverse of
    // tetraFaceIndices !!!)
    // front face
    textCoordIndices[0] = 0;
    textCoordIndices[1] = 1;
    textCoordIndices[2] = 2;
    // back face
    textCoordIndices[3] = 0;
    textCoordIndices[4] = 1;
    textCoordIndices[5] = 2;
    // back-left face
    textCoordIndices[6] = 2;
    textCoordIndices[7] = 0;
    textCoordIndices[8] = 1;
    // bottom
    textCoordIndices[9] = 0;
    textCoordIndices[10] = 1;
    textCoordIndices[11] = 2;
    /*
     * // case #2: each face of the tetrahedron has a different part of the
     * texture.
     *  // The coordinates of the 4 points in the 2D texture space.
     * textCoord2f = new TexCoord2f[4]; textCoord2f[0] = new
     * TexCoord2f(0.0f, 0.5f); textCoord2f[1] = new TexCoord2f(1.0f, 0.5f);
     * textCoord2f[2] = new TexCoord2f(0.6f, 0.7f); textCoord2f[3] = new
     * TexCoord2f(0.6f, 0.3f);
     * 
     *  // Set the texture coordinate's indices (referenced to the array of
     * 2D points // in the texture space by setTextureCoordinates(0,
     * textCoord2f) and // setTextureCoordinateIndices(0,
     * textCoordIndices)). textCoordIndices = new
     * int[lengthTetraFaceIndices]; // 12
     *  // From the camera in the view coordinate system (inverse of
     * tetraFaceIndices !!!) // front face textCoordIndices[0] = 3;
     * textCoordIndices[1] = 2; textCoordIndices[2] = 0; // back face
     * textCoordIndices[3] = 1; textCoordIndices[4] = 2; textCoordIndices[5] =
     * 3; // back-left face textCoordIndices[6] = 1; textCoordIndices[7] =
     * 0; textCoordIndices[8] = 2; // bottom textCoordIndices[9] = 1;
     * textCoordIndices[10]= 3; textCoordIndices[11]= 0;
     */
    // just one set
    tetra_GeometryInfo.setTextureCoordinates(0, textCoord2f);
    // just one set
    tetra_GeometryInfo.setTextureCoordinateIndices(0, textCoordIndices);
    normalGenerator = new NormalGenerator();
    normalGenerator.generateNormals(tetra_GeometryInfo);
    if (crAngle)
      normalGenerator.setCreaseAngle(0.0f); // with 0 radian ===> creased
    stripifier = new Stripifier();
    stripifier.stripify(tetra_GeometryInfo);
    tetra_GeometryArray = tetra_GeometryInfo.getGeometryArray();
    // The geonometry is passed to the instance this of the tetrahedron.
    this.setGeometry(tetra_GeometryArray);
    ////////////////////// Appearance part ///////////////////////////
    appearance = new Appearance();
    // Optical properties of the tetrahedron.
    // Ambient-diffuse-reflection coefficient
    diffAmb = new Color3f(1.0f, 0.5f, 1.0f);
    // Diffuse-reflection coefficient
    reflDiff = new Color3f(1.0f, 0.5f, 1.0f);
    // Specular-reflection coefficient (reflectance function)
    reflSpec = new Color3f(1.0f, 0.5f, 1.0f);
    // c = shininess: cos^c in the specular reflection
    float c = 15;
    // Emitted light
    emittedLight = new Color3f(0.0f, 0.0f, 0.0f);
    material = new Material(diffAmb, emittedLight, reflDiff, reflSpec, c);
    appearance.setMaterial(material);
    // This instance acts only on the tetrahedron and not on its texture.
    trAttr = new TransparencyAttributes(TransparencyAttributes.NICEST, 0.0f);
    // 0.0 = fully opaque
    // 1.0 = fully transparent
    appearance.setTransparencyAttributes(trAttr);
    // Loading the texture
    newTextureLoader = new NewTextureLoader("Images/Claude.jpg");
    newTextureLoader.setImageObserver(newTextureLoader.getImageObserver());
    texture = newTextureLoader.getTexture();
    appearance.setTexture(texture);
    // Application mode of the texture
    textAttr = new TextureAttributes();
    textAttr.setTextureMode(TextureAttributes.MODULATE); // there still are:
    // BLEND, COMBINE,
    // DECAL, and REPLACE
    appearance.setTextureAttributes(textAttr);
    // The appearance is passed to the instance this of the tetrahedron.
    this.setAppearance(appearance);
    return this;
  }
}
/**
 * This class creates a right-handed 3D coordinate system.
 */
class CoordSyst extends Shape3D {
  private float rX, gX, bX, rY, gY, bY, rZ, gZ, bZ, scale_XYZ;
  private LineArray axes;
  private float scaledExtremites[];
  /**
   * Constructor that allows to specify the desired initial colors and the
   * length of the axes.
   * 
   * @param type
   *            float rougeX, vertX, bleuX, rougeY, vertY, bleuY, rougeZ,
   *            vertZ, bleuZ - the colors of the axes
   * @param type
   *            float s_XYZ - the scale factor to adjust the axes's length
   */
  public CoordSyst(float rougeX, float vertX, float bleuX, float rougeY,
      float vertY, float bleuY, float rougeZ, float vertZ, float bleuZ,
      float s_XYZ) {
    rX = rougeX;
    rY = rougeY;
    rZ = rougeZ;
    gX = vertX;
    gY = vertY;
    gZ = vertZ;
    bX = bleuX;
    bY = bleuY;
    bZ = bleuZ;
    scale_XYZ = s_XYZ;
    // Colors of the three axes.
    float color[] = { rX, gX, bX, rX, gX, bX, // the x axis
        rY, gY, bY, rY, gY, bY, // the y axis
        rZ, gZ, bZ, rZ, gZ, bZ }; // the z axis
    // Construction of the axes (LineArray).
    axes = new LineArray(6, LineArray.COORDINATES | LineArray.COLOR_3);
    // Scalling of the vertices of the 3 axes using scale_XYZ.
    scaledExtremites = new float[extremites.length];
    for (int i = 0; i < extremites.length; i++)
      scaledExtremites[i] = extremites[i] * scale_XYZ;
    axes.setCoordinates(0, scaledExtremites);
    axes.setColors(0, color);
    this.setGeometry(axes);
  }
  // Definition of the geometry of the three axes.
  private static final float extremites[] = {
  // x-axis
      0.0f, 0.0f, 0.0f, 1.0f, 0.0f, 0.0f,
      // y-axis
      0.0f, 0.0f, 0.0f, 0.0f, 1.0f, 0.0f,
      // z-axis
      0.0f, 0.0f, 0.0f, 0.0f, 0.0f, 1.0f };
}