/*
* %Z%%M% %I% %E% %U%
*
* ************************************************************** "Copyright (c)
* 2001 Sun Microsystems, Inc. All Rights Reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions are met:
*
* -Redistributions of source code must retain the above copyright notice, this
* list of conditions and the following disclaimer.
*
* -Redistribution in binary form must reproduce the above copyright notice,
* this list of conditions and the following disclaimer in the documentation
* and/or other materials provided with the distribution.
*
* Neither the name of Sun Microsystems, Inc. or the names of contributors may
* be used to endorse or promote products derived from this software without
* specific prior written permission.
*
* 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.
*
* You acknowledge that Software is not designed,licensed or intended for use in
* the design, construction, operation or maintenance of any nuclear facility."
*
* ***************************************************************************
*/
import java.applet.Applet;
import java.awt.BorderLayout;
import java.awt.Component;
import java.awt.Dimension;
import java.awt.Font;
import java.awt.GraphicsConfiguration;
import java.awt.GridLayout;
import java.awt.Point;
import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;
import java.awt.image.BufferedImage;
import java.io.BufferedOutputStream;
import java.io.FileOutputStream;
import java.text.NumberFormat;
import java.util.Enumeration;
import java.util.EventListener;
import java.util.EventObject;
import java.util.Hashtable;
import java.util.Vector;
import javax.media.j3d.AmbientLight;
import javax.media.j3d.Appearance;
import javax.media.j3d.BoundingSphere;
import javax.media.j3d.BranchGroup;
import javax.media.j3d.Canvas3D;
import javax.media.j3d.DirectionalLight;
import javax.media.j3d.Font3D;
import javax.media.j3d.ImageComponent;
import javax.media.j3d.ImageComponent2D;
import javax.media.j3d.Link;
import javax.media.j3d.Material;
import javax.media.j3d.OrientedShape3D;
import javax.media.j3d.Screen3D;
import javax.media.j3d.SharedGroup;
import javax.media.j3d.Switch;
import javax.media.j3d.Text3D;
import javax.media.j3d.Transform3D;
import javax.media.j3d.TransformGroup;
import javax.media.j3d.View;
import javax.swing.Box;
import javax.swing.BoxLayout;
import javax.swing.JButton;
import javax.swing.JCheckBox;
import javax.swing.JLabel;
import javax.swing.JPanel;
import javax.swing.JSlider;
import javax.swing.JTabbedPane;
import javax.swing.event.ChangeEvent;
import javax.swing.event.ChangeListener;
import javax.vecmath.AxisAngle4f;
import javax.vecmath.Color3f;
import javax.vecmath.Point3d;
import javax.vecmath.Point3f;
import javax.vecmath.Vector3d;
import javax.vecmath.Vector3f;
import com.sun.image.codec.jpeg.JPEGCodec;
import com.sun.image.codec.jpeg.JPEGEncodeParam;
import com.sun.image.codec.jpeg.JPEGImageEncoder;
import com.sun.j3d.utils.applet.MainFrame;
import com.sun.j3d.utils.behaviors.vp.OrbitBehavior;
import com.sun.j3d.utils.geometry.Cone;
import com.sun.j3d.utils.geometry.Cylinder;
import com.sun.j3d.utils.universe.SimpleUniverse;
import com.sun.j3d.utils.universe.ViewingPlatform;
/*
*
*/
public class TransformExplorer extends Applet implements
Java3DExplorerConstants {
SimpleUniverse u;
boolean isApplication;
Canvas3D canvas;
OffScreenCanvas3D offScreenCanvas;
View view;
TransformGroup coneTG;
// transformation factors for the cone
Vector3f coneTranslation = new Vector3f(0.0f, 0.0f, 0.0f);
float coneScale = 1.0f;
Vector3d coneNUScale = new Vector3d(1.0f, 1.0f, 1.0f);
Vector3f coneRotateAxis = new Vector3f(1.0f, 0.0f, 0.0f);
Vector3f coneRotateNAxis = new Vector3f(1.0f, 0.0f, 0.0f);
float coneRotateAngle = 0.0f;
AxisAngle4f coneRotateAxisAngle = new AxisAngle4f(coneRotateAxis,
coneRotateAngle);
Vector3f coneRefPt = new Vector3f(0.0f, 0.0f, 0.0f);
// this tells whether to use the compound transformation
boolean useCompoundTransform = true;
// These are Transforms are used for the compound transformation
Transform3D translateTrans = new Transform3D();
Transform3D scaleTrans = new Transform3D();
Transform3D rotateTrans = new Transform3D();
Transform3D refPtTrans = new Transform3D();
Transform3D refPtInvTrans = new Transform3D();
// this tells whether to use the uniform or non-uniform scale when
// updating the compound transform
boolean useUniformScale = true;
// The size of the cone
float coneRadius = 1.0f;
float coneHeight = 2.0f;
// The axis indicator, used to show the rotation axis
RotAxis rotAxis;
boolean showRotAxis = false;
float rotAxisLength = 3.0f;
// The coord sys used to show the coordinate system
CoordSys coordSys;
boolean showCoordSys = true;
float coordSysLength = 5.0f;
// GUI elements
String rotAxisString = "Rotation Axis";
String coordSysString = "Coord Sys";
JCheckBox rotAxisCheckBox;
JCheckBox coordSysCheckBox;
String snapImageString = "Snap Image";
String outFileBase = "transform";
int outFileSeq = 0;
float offScreenScale;
JLabel coneRotateNAxisXLabel;
JLabel coneRotateNAxisYLabel;
JLabel coneRotateNAxisZLabel;
// Temporaries that are reused
Transform3D tmpTrans = new Transform3D();
Vector3f tmpVector = new Vector3f();
AxisAngle4f tmpAxisAngle = new AxisAngle4f();
// geometric constant
Point3f origin = new Point3f();
Vector3f yAxis = new Vector3f(0.0f, 1.0f, 0.0f);
// Returns the TransformGroup we will be editing to change the transform
// on the cone
TransformGroup createConeTransformGroup() {
// create a TransformGroup for the cone, allow tranform changes,
coneTG = new TransformGroup();
coneTG.setCapability(TransformGroup.ALLOW_TRANSFORM_WRITE);
coneTG.setCapability(TransformGroup.ALLOW_TRANSFORM_READ);
// Set up an appearance to make the Cone with red ambient,
// black emmissive, red diffuse and white specular coloring
Material material = new Material(red, black, red, white, 64);
// These are the colors used for the book figures:
//Material material = new Material(white, black, white, black, 64);
Appearance appearance = new Appearance();
appearance.setMaterial(material);
// create the cone and add it to the coneTG
Cone cone = new Cone(coneRadius, coneHeight, appearance);
coneTG.addChild(cone);
return coneTG;
}
void setConeTranslation() {
coneTG.getTransform(tmpTrans); // get the old transform
tmpTrans.setTranslation(coneTranslation); // set only translation
coneTG.setTransform(tmpTrans); // set the new transform
}
void setConeUScale() {
coneTG.getTransform(tmpTrans); // get the old transform
tmpTrans.setScale(coneScale); // set only scale
coneTG.setTransform(tmpTrans); // set the new transform
}
void setConeNUScale() {
coneTG.getTransform(tmpTrans); // get the old transform
System.out.println("coneNUScale.x = " + coneNUScale.x);
tmpTrans.setScale(coneNUScale);// set only scale
coneTG.setTransform(tmpTrans); // set the new transform
}
void setConeRotation() {
coneTG.getTransform(tmpTrans); // get the old transform
tmpTrans.setRotation(coneRotateAxisAngle); // set only rotation
coneTG.setTransform(tmpTrans); // set the new transform
}
void updateUsingCompoundTransform() {
// set the component transformations
translateTrans.set(coneTranslation);
if (useUniformScale) {
scaleTrans.set(coneScale);
} else {
scaleTrans.setIdentity();
scaleTrans.setScale(coneNUScale);
}
rotateTrans.set(coneRotateAxisAngle);
// translate from ref pt to origin
tmpVector.sub(origin, coneRefPt); // vector from ref pt to origin
refPtTrans.set(tmpVector);
// translate from origin to ref pt
tmpVector.sub(coneRefPt, origin); // vector from origin to ref pt
refPtInvTrans.set(tmpVector);
// now build up the transfomation
// trans = translate * refPtInv * scale * rotate * refPt;
tmpTrans.set(translateTrans);
tmpTrans.mul(refPtInvTrans);
tmpTrans.mul(scaleTrans);
tmpTrans.mul(rotateTrans);
tmpTrans.mul(refPtTrans);
// Copy the transform to the TransformGroup
coneTG.setTransform(tmpTrans);
}
// ensure that the cone rotation axis is a unit vector
void normalizeConeRotateAxis() {
// normalize, watch for length == 0, if so, then use default
float lengthSquared = coneRotateAxis.lengthSquared();
if (lengthSquared > 0.0001) {
coneRotateNAxis.scale((float) (1.0 / Math.sqrt(lengthSquared)),
coneRotateAxis);
} else {
coneRotateNAxis.set(1.0f, 0.0f, 0.0f);
}
}
// copy the current axis and angle to the axis angle, convert angle
// to radians
void updateConeAxisAngle() {
coneRotateAxisAngle.set(coneRotateNAxis, (float) Math
.toRadians(coneRotateAngle));
}
void updateConeRotateNormalizedLabels() {
nf.setMinimumFractionDigits(2);
nf.setMaximumFractionDigits(2);
coneRotateNAxisXLabel.setText("X: " + nf.format(coneRotateNAxis.x));
coneRotateNAxisYLabel.setText("Y: " + nf.format(coneRotateNAxis.y));
coneRotateNAxisZLabel.setText("Z: " + nf.format(coneRotateNAxis.z));
}
BranchGroup createSceneGraph() {
// Create the root of the branch graph
BranchGroup objRoot = new BranchGroup();
// Create a TransformGroup to scale the scene down by 3.5x
TransformGroup objScale = new TransformGroup();
Transform3D scaleTrans = new Transform3D();
scaleTrans.set(1 / 3.5f); // scale down by 3.5x
objScale.setTransform(scaleTrans);
objRoot.addChild(objScale);
// Create a TransformGroup and initialize it to the
// identity. Enable the TRANSFORM_WRITE capability so that
// the mouse behaviors code can modify it at runtime. Add it to the
// root of the subgraph.
TransformGroup objTrans = new TransformGroup();
objTrans.setCapability(TransformGroup.ALLOW_TRANSFORM_WRITE);
objTrans.setCapability(TransformGroup.ALLOW_TRANSFORM_READ);
objScale.addChild(objTrans);
// Add the primitives to the scene
objTrans.addChild(createConeTransformGroup()); // the cone
rotAxis = new RotAxis(rotAxisLength); // the axis
objTrans.addChild(rotAxis);
coordSys = new CoordSys(coordSysLength); // the coordSys
objTrans.addChild(coordSys);
BoundingSphere bounds = new BoundingSphere(new Point3d(), 100.0);
// The book used a white background for the figures
//Background bg = new Background(new Color3f(1.0f, 1.0f, 1.0f));
//bg.setApplicationBounds(bounds);
//objTrans.addChild(bg);
// Set up the ambient light
Color3f ambientColor = new Color3f(0.1f, 0.1f, 0.1f);
AmbientLight ambientLightNode = new AmbientLight(ambientColor);
ambientLightNode.setInfluencingBounds(bounds);
objRoot.addChild(ambientLightNode);
// Set up the directional lights
Color3f light1Color = new Color3f(1.0f, 1.0f, 1.0f);
Vector3f light1Direction = new Vector3f(0.0f, -0.2f, -1.0f);
DirectionalLight light1 = new DirectionalLight(light1Color,
light1Direction);
light1.setInfluencingBounds(bounds);
objRoot.addChild(light1);
return objRoot;
}
public TransformExplorer() {
this(false, 1.0f);
}
public TransformExplorer(boolean isApplication, float initOffScreenScale) {
this.isApplication = isApplication;
this.offScreenScale = initOffScreenScale;
}
public void init() {
setLayout(new BorderLayout());
GraphicsConfiguration config = SimpleUniverse
.getPreferredConfiguration();
canvas = new Canvas3D(config);
add("Center", canvas);
u = new SimpleUniverse(canvas);
if (isApplication) {
offScreenCanvas = new OffScreenCanvas3D(config, true);
// set the size of the off-screen canvas based on a scale
// of the on-screen size
Screen3D sOn = canvas.getScreen3D();
Screen3D sOff = offScreenCanvas.getScreen3D();
Dimension dim = sOn.getSize();
dim.width *= offScreenScale;
dim.height *= offScreenScale;
sOff.setSize(dim);
sOff.setPhysicalScreenWidth(sOn.getPhysicalScreenWidth()
* offScreenScale);
sOff.setPhysicalScreenHeight(sOn.getPhysicalScreenHeight()
* offScreenScale);
// attach the offscreen canvas to the view
u.getViewer().getView().addCanvas3D(offScreenCanvas);
}
// Create a simple scene and attach it to the virtual universe
BranchGroup scene = createSceneGraph();
// get the view
view = u.getViewer().getView();
// This will move the ViewPlatform back a bit so the
// objects in the scene can be viewed.
ViewingPlatform viewingPlatform = u.getViewingPlatform();
viewingPlatform.setNominalViewingTransform();
// add an orbit behavior to move the viewing platform
OrbitBehavior orbit = new OrbitBehavior(canvas, OrbitBehavior.STOP_ZOOM);
BoundingSphere bounds = new BoundingSphere(new Point3d(0.0, 0.0, 0.0),
100.0);
orbit.setSchedulingBounds(bounds);
viewingPlatform.setViewPlatformBehavior(orbit);
u.addBranchGraph(scene);
add("East", guiPanel());
}
// create a panel with a tabbed pane holding each of the edit panels
JPanel guiPanel() {
JPanel panel = new JPanel();
panel.setLayout(new BorderLayout());
JTabbedPane tabbedPane = new JTabbedPane();
tabbedPane.addTab("Translation", translationPanel());
tabbedPane.addTab("Scaling", scalePanel());
tabbedPane.addTab("Rotation", rotationPanel());
tabbedPane.addTab("Reference Point", refPtPanel());
panel.add("Center", tabbedPane);
panel.add("South", configPanel());
return panel;
}
Box translationPanel() {
Box panel = new Box(BoxLayout.Y_AXIS);
panel.add(new LeftAlignComponent(new JLabel("Translation Offset")));
// X translation label, slider, and value label
FloatLabelJSlider coneTranslateXSlider = new FloatLabelJSlider("X",
0.1f, -2.0f, 2.0f, coneTranslation.x);
coneTranslateXSlider.setMajorTickSpacing(1.0f);
coneTranslateXSlider.setPaintTicks(true);
coneTranslateXSlider.addFloatListener(new FloatListener() {
public void floatChanged(FloatEvent e) {
coneTranslation.x = e.getValue();
if (useCompoundTransform) {
updateUsingCompoundTransform();
} else {
setConeTranslation();
}
}
});
panel.add(coneTranslateXSlider);
// Y translation label, slider, and value label
FloatLabelJSlider coneTranslateYSlider = new FloatLabelJSlider("Y",
0.1f, -2.0f, 2.0f, coneTranslation.y);
coneTranslateYSlider.setMajorTickSpacing(1.0f);
coneTranslateYSlider.setPaintTicks(true);
coneTranslateYSlider.addFloatListener(new FloatListener() {
public void floatChanged(FloatEvent e) {
coneTranslation.y = e.getValue();
if (useCompoundTransform) {
updateUsingCompoundTransform();
} else {
setConeTranslation();
}
}
});
panel.add(coneTranslateYSlider);
// Z translation label, slider, and value label
FloatLabelJSlider coneTranslateZSlider = new FloatLabelJSlider("Z",
0.1f, -2.0f, 2.0f, coneTranslation.z);
coneTranslateZSlider.setMajorTickSpacing(1.0f);
coneTranslateZSlider.setPaintTicks(true);
coneTranslateZSlider.addFloatListener(new FloatListener() {
public void floatChanged(FloatEvent e) {
coneTranslation.z = e.getValue();
if (useCompoundTransform) {
updateUsingCompoundTransform();
} else {
setConeTranslation();
}
}
});
panel.add(coneTranslateZSlider);
return panel;
}
Box scalePanel() {
Box panel = new Box(BoxLayout.Y_AXIS);
// Uniform Scale
JLabel uniform = new JLabel("Uniform Scale");
panel.add(new LeftAlignComponent(uniform));
FloatLabelJSlider coneScaleSlider = new FloatLabelJSlider("S:", 0.1f,
0.0f, 3.0f, coneScale);
coneScaleSlider.setMajorTickSpacing(1.0f);
coneScaleSlider.setPaintTicks(true);
coneScaleSlider.addFloatListener(new FloatListener() {
public void floatChanged(FloatEvent e) {
coneScale = e.getValue();
useUniformScale = true;
if (useCompoundTransform) {
updateUsingCompoundTransform();
} else {
setConeUScale();
}
}
});
panel.add(coneScaleSlider);
JLabel nonUniform = new JLabel("Non-Uniform Scale");
panel.add(new LeftAlignComponent(nonUniform));
// Non-Uniform Scale
FloatLabelJSlider coneNUScaleXSlider = new FloatLabelJSlider("X: ",
0.1f, 0.0f, 3.0f, (float) coneNUScale.x);
coneNUScaleXSlider.setMajorTickSpacing(1.0f);
coneNUScaleXSlider.setPaintTicks(true);
coneNUScaleXSlider.addFloatListener(new FloatListener() {
public void floatChanged(FloatEvent e) {
coneNUScale.x = (double) e.getValue();
useUniformScale = false;
if (useCompoundTransform) {
updateUsingCompoundTransform();
} else {
setConeNUScale();
}
}
});
panel.add(coneNUScaleXSlider);
FloatLabelJSlider coneNUScaleYSlider = new FloatLabelJSlider("Y: ",
0.1f, 0.0f, 3.0f, (float) coneNUScale.y);
coneNUScaleYSlider.setMajorTickSpacing(1.0f);
coneNUScaleYSlider.setPaintTicks(true);
coneNUScaleYSlider.addFloatListener(new FloatListener() {
public void floatChanged(FloatEvent e) {
coneNUScale.y = (double) e.getValue();
useUniformScale = false;
if (useCompoundTransform) {
updateUsingCompoundTransform();
} else {
setConeNUScale();
}
}
});
panel.add(coneNUScaleYSlider);
FloatLabelJSlider coneNUScaleZSlider = new FloatLabelJSlider("Z: ",
0.1f, 0.0f, 3.0f, (float) coneNUScale.z);
coneNUScaleZSlider.setMajorTickSpacing(1.0f);
coneNUScaleZSlider.setPaintTicks(true);
coneNUScaleZSlider.addFloatListener(new FloatListener() {
public void floatChanged(FloatEvent e) {
coneNUScale.z = (double) e.getValue();
useUniformScale = false;
if (useCompoundTransform) {
updateUsingCompoundTransform();
} else {
setConeNUScale();
}
}
});
panel.add(coneNUScaleZSlider);
return panel;
}
JPanel rotationPanel() {
JPanel panel = new JPanel();
panel.setLayout(new GridLayout(0, 1));
panel.add(new LeftAlignComponent(new JLabel("Rotation Axis")));
FloatLabelJSlider coneRotateAxisXSlider = new FloatLabelJSlider("X: ",
0.01f, -1.0f, 1.0f, (float) coneRotateAxis.x);
coneRotateAxisXSlider.addFloatListener(new FloatListener() {
public void floatChanged(FloatEvent e) {
coneRotateAxis.x = e.getValue();
normalizeConeRotateAxis();
updateConeAxisAngle();
if (useCompoundTransform) {
updateUsingCompoundTransform();
} else {
setConeRotation();
}
rotAxis.setRotationAxis(coneRotateAxis);
updateConeRotateNormalizedLabels();
}
});
panel.add(coneRotateAxisXSlider);
FloatLabelJSlider coneRotateAxisYSlider = new FloatLabelJSlider("Y: ",
0.01f, -1.0f, 1.0f, (float) coneRotateAxis.y);
coneRotateAxisYSlider.addFloatListener(new FloatListener() {
public void floatChanged(FloatEvent e) {
coneRotateAxis.y = e.getValue();
normalizeConeRotateAxis();
updateConeAxisAngle();
if (useCompoundTransform) {
updateUsingCompoundTransform();
} else {
setConeRotation();
}
rotAxis.setRotationAxis(coneRotateAxis);
updateConeRotateNormalizedLabels();
}
});
panel.add(coneRotateAxisYSlider);
FloatLabelJSlider coneRotateAxisZSlider = new FloatLabelJSlider("Z: ",
0.01f, -1.0f, 1.0f, (float) coneRotateAxis.y);
coneRotateAxisZSlider.addFloatListener(new FloatListener() {
public void floatChanged(FloatEvent e) {
coneRotateAxis.z = e.getValue();
normalizeConeRotateAxis();
updateConeAxisAngle();
if (useCompoundTransform) {
updateUsingCompoundTransform();
} else {
setConeRotation();
}
rotAxis.setRotationAxis(coneRotateAxis);
updateConeRotateNormalizedLabels();
}
});
panel.add(coneRotateAxisZSlider);
JLabel normalizedLabel = new JLabel("Normalized Rotation Axis");
panel.add(new LeftAlignComponent(normalizedLabel));
;
coneRotateNAxisXLabel = new JLabel("X: 1.000");
panel.add(new LeftAlignComponent(coneRotateNAxisXLabel));
coneRotateNAxisYLabel = new JLabel("Y: 0.000");
panel.add(new LeftAlignComponent(coneRotateNAxisYLabel));
coneRotateNAxisZLabel = new JLabel("Z: 0.000");
panel.add(new LeftAlignComponent(coneRotateNAxisZLabel));
normalizeConeRotateAxis();
updateConeRotateNormalizedLabels();
FloatLabelJSlider coneRotateAxisAngleSlider = new FloatLabelJSlider(
"Angle: ", 1.0f, -180.0f, 180.0f, (float) coneRotateAngle);
coneRotateAxisAngleSlider.addFloatListener(new FloatListener() {
public void floatChanged(FloatEvent e) {
coneRotateAngle = e.getValue();
updateConeAxisAngle();
if (useCompoundTransform) {
updateUsingCompoundTransform();
} else {
setConeRotation();
}
}
});
panel.add(coneRotateAxisAngleSlider);
return panel;
}
Box refPtPanel() {
Box panel = new Box(BoxLayout.Y_AXIS);
panel.add(new LeftAlignComponent(new JLabel(
"Reference Point Coordinates")));
// X Ref Pt
FloatLabelJSlider coneRefPtXSlider = new FloatLabelJSlider("X", 0.1f,
-2.0f, 2.0f, coneRefPt.x);
coneRefPtXSlider.setMajorTickSpacing(1.0f);
coneRefPtXSlider.setPaintTicks(true);
coneRefPtXSlider.addFloatListener(new FloatListener() {
public void floatChanged(FloatEvent e) {
coneRefPt.x = e.getValue();
useCompoundTransform = true;
updateUsingCompoundTransform();
rotAxis.setRefPt(coneRefPt);
}
});
panel.add(coneRefPtXSlider);
// Y Ref Pt
FloatLabelJSlider coneRefPtYSlider = new FloatLabelJSlider("Y", 0.1f,
-2.0f, 2.0f, coneRefPt.y);
coneRefPtYSlider.setMajorTickSpacing(1.0f);
coneRefPtYSlider.setPaintTicks(true);
coneRefPtYSlider.addFloatListener(new FloatListener() {
public void floatChanged(FloatEvent e) {
coneRefPt.y = e.getValue();
useCompoundTransform = true;
updateUsingCompoundTransform();
rotAxis.setRefPt(coneRefPt);
}
});
panel.add(coneRefPtYSlider);
// Z Ref Pt
FloatLabelJSlider coneRefPtZSlider = new FloatLabelJSlider("Z", 0.1f,
-2.0f, 2.0f, coneRefPt.z);
coneRefPtZSlider.setMajorTickSpacing(1.0f);
coneRefPtZSlider.setPaintTicks(true);
coneRefPtZSlider.addFloatListener(new FloatListener() {
public void floatChanged(FloatEvent e) {
coneRefPt.z = e.getValue();
useCompoundTransform = true;
updateUsingCompoundTransform();
rotAxis.setRefPt(coneRefPt);
}
});
panel.add(coneRefPtZSlider);
return panel;
}
JPanel configPanel() {
JPanel panel = new JPanel();
panel.setLayout(new GridLayout(0, 1));
panel.add(new JLabel("Display annotation:"));
// create the check boxes
rotAxisCheckBox = new JCheckBox(rotAxisString);
rotAxisCheckBox.setSelected(showRotAxis);
rotAxisCheckBox.addActionListener(new ActionListener() {
public void actionPerformed(ActionEvent e) {
Object source = e.getSource();
showRotAxis = ((JCheckBox) source).isSelected();
if (showRotAxis) {
rotAxis.setWhichChild(Switch.CHILD_ALL);
} else {
rotAxis.setWhichChild(Switch.CHILD_NONE);
}
}
});
panel.add(rotAxisCheckBox);
coordSysCheckBox = new JCheckBox(coordSysString);
coordSysCheckBox.setSelected(showCoordSys);
coordSysCheckBox.addActionListener(new ActionListener() {
public void actionPerformed(ActionEvent e) {
Object source = e.getSource();
showCoordSys = ((JCheckBox) source).isSelected();
if (showCoordSys) {
coordSys.setWhichChild(Switch.CHILD_ALL);
} else {
coordSys.setWhichChild(Switch.CHILD_NONE);
}
}
});
panel.add(coordSysCheckBox);
if (isApplication) {
JButton snapButton = new JButton(snapImageString);
snapButton.addActionListener(new ActionListener() {
public void actionPerformed(ActionEvent e) {
Point loc = canvas.getLocationOnScreen();
offScreenCanvas.setOffScreenLocation(loc);
Dimension dim = canvas.getSize();
dim.width *= offScreenScale;
dim.height *= offScreenScale;
nf.setMinimumIntegerDigits(3);
nf.setMaximumFractionDigits(0);
offScreenCanvas.snapImageFile(outFileBase
+ nf.format(outFileSeq++), dim.width, dim.height);
nf.setMinimumIntegerDigits(0);
}
});
panel.add(snapButton);
}
return panel;
}
public void destroy() {
u.removeAllLocales();
}
// The following allows TransformExplorer to be run as an application
// as well as an applet
//
public static void main(String[] args) {
float initOffScreenScale = 2.5f;
for (int i = 0; i < args.length; i++) {
if (args[i].equals("-s")) {
if (args.length >= (i + 1)) {
initOffScreenScale = Float.parseFloat(args[i + 1]);
i++;
}
}
}
new MainFrame(new TransformExplorer(true, initOffScreenScale), 950, 600);
}
}
interface Java3DExplorerConstants {
// colors
static Color3f black = new Color3f(0.0f, 0.0f, 0.0f);
static Color3f red = new Color3f(1.0f, 0.0f, 0.0f);
static Color3f green = new Color3f(0.0f, 1.0f, 0.0f);
static Color3f blue = new Color3f(0.0f, 0.0f, 1.0f);
static Color3f skyBlue = new Color3f(0.6f, 0.7f, 0.9f);
static Color3f cyan = new Color3f(0.0f, 1.0f, 1.0f);
static Color3f magenta = new Color3f(1.0f, 0.0f, 1.0f);
static Color3f yellow = new Color3f(1.0f, 1.0f, 0.0f);
static Color3f brightWhite = new Color3f(1.0f, 1.5f, 1.5f);
static Color3f white = new Color3f(1.0f, 1.0f, 1.0f);
static Color3f darkGrey = new Color3f(0.15f, 0.15f, 0.15f);
static Color3f medGrey = new Color3f(0.3f, 0.3f, 0.3f);
static Color3f grey = new Color3f(0.5f, 0.5f, 0.5f);
static Color3f lightGrey = new Color3f(0.75f, 0.75f, 0.75f);
// infinite bounding region, used to make env nodes active everywhere
BoundingSphere infiniteBounds = new BoundingSphere(new Point3d(),
Double.MAX_VALUE);
// common values
static final String nicestString = "NICEST";
static final String fastestString = "FASTEST";
static final String antiAliasString = "Anti-Aliasing";
static final String noneString = "NONE";
// light type constants
static int LIGHT_AMBIENT = 1;
static int LIGHT_DIRECTIONAL = 2;
static int LIGHT_POSITIONAL = 3;
static int LIGHT_SPOT = 4;
// screen capture constants
static final int USE_COLOR = 1;
static final int USE_BLACK_AND_WHITE = 2;
// number formatter
NumberFormat nf = NumberFormat.getInstance();
}
class OffScreenCanvas3D extends Canvas3D {
OffScreenCanvas3D(GraphicsConfiguration graphicsConfiguration,
boolean offScreen) {
super(graphicsConfiguration, offScreen);
}
private BufferedImage doRender(int width, int height) {
BufferedImage bImage = new BufferedImage(width, height,
BufferedImage.TYPE_INT_RGB);
ImageComponent2D buffer = new ImageComponent2D(
ImageComponent.FORMAT_RGB, bImage);
//buffer.setYUp(true);
setOffScreenBuffer(buffer);
renderOffScreenBuffer();
waitForOffScreenRendering();
bImage = getOffScreenBuffer().getImage();
return bImage;
}
void snapImageFile(String filename, int width, int height) {
BufferedImage bImage = doRender(width, height);
/*
* JAI: RenderedImage fImage = JAI.create("format", bImage,
* DataBuffer.TYPE_BYTE); JAI.create("filestore", fImage, filename +
* ".tif", "tiff", null);
*/
/* No JAI: */
try {
FileOutputStream fos = new FileOutputStream(filename + ".jpg");
BufferedOutputStream bos = new BufferedOutputStream(fos);
JPEGImageEncoder jie = JPEGCodec.createJPEGEncoder(bos);
JPEGEncodeParam param = jie.getDefaultJPEGEncodeParam(bImage);
param.setQuality(1.0f, true);
jie.setJPEGEncodeParam(param);
jie.encode(bImage);
bos.flush();
fos.close();
} catch (Exception e) {
System.out.println(e);
}
}
}
class FloatLabelJSlider extends JPanel implements ChangeListener,
Java3DExplorerConstants {
JSlider slider;
JLabel valueLabel;
Vector listeners = new Vector();
float min, max, resolution, current, scale;
int minInt, maxInt, curInt;;
int intDigits, fractDigits;
float minResolution = 0.001f;
// default slider with name, resolution = 0.1, min = 0.0, max = 1.0 inital
// 0.5
FloatLabelJSlider(String name) {
this(name, 0.1f, 0.0f, 1.0f, 0.5f);
}
FloatLabelJSlider(String name, float resolution, float min, float max,
float current) {
this.resolution = resolution;
this.min = min;
this.max = max;
this.current = current;
if (resolution < minResolution) {
resolution = minResolution;
}
// round scale to nearest integer fraction. i.e. 0.3 => 1/3 = 0.33
scale = (float) Math.round(1.0f / resolution);
resolution = 1.0f / scale;
// get the integer versions of max, min, current
minInt = Math.round(min * scale);
maxInt = Math.round(max * scale);
curInt = Math.round(current * scale);
// sliders use integers, so scale our floating point value by "scale"
// to make each slider "notch" be "resolution". We will scale the
// value down by "scale" when we get the event.
slider = new JSlider(JSlider.HORIZONTAL, minInt, maxInt, curInt);
slider.addChangeListener(this);
valueLabel = new JLabel(" ");
// set the initial value label
setLabelString();
// add min and max labels to the slider
Hashtable labelTable = new Hashtable();
labelTable.put(new Integer(minInt), new JLabel(nf.format(min)));
labelTable.put(new Integer(maxInt), new JLabel(nf.format(max)));
slider.setLabelTable(labelTable);
slider.setPaintLabels(true);
/* layout to align left */
setLayout(new BorderLayout());
Box box = new Box(BoxLayout.X_AXIS);
add(box, BorderLayout.WEST);
box.add(new JLabel(name));
box.add(slider);
box.add(valueLabel);
}
public void setMinorTickSpacing(float spacing) {
int intSpacing = Math.round(spacing * scale);
slider.setMinorTickSpacing(intSpacing);
}
public void setMajorTickSpacing(float spacing) {
int intSpacing = Math.round(spacing * scale);
slider.setMajorTickSpacing(intSpacing);
}
public void setPaintTicks(boolean paint) {
slider.setPaintTicks(paint);
}
public void addFloatListener(FloatListener listener) {
listeners.add(listener);
}
public void removeFloatListener(FloatListener listener) {
listeners.remove(listener);
}
public void stateChanged(ChangeEvent e) {
JSlider source = (JSlider) e.getSource();
// get the event type, set the corresponding value.
// Sliders use integers, handle floating point values by scaling the
// values by "scale" to allow settings at "resolution" intervals.
// Divide by "scale" to get back to the real value.
curInt = source.getValue();
current = curInt / scale;
valueChanged();
}
public void setValue(float newValue) {
boolean changed = (newValue != current);
current = newValue;
if (changed) {
valueChanged();
}
}
private void valueChanged() {
// update the label
setLabelString();
// notify the listeners
FloatEvent event = new FloatEvent(this, current);
for (Enumeration e = listeners.elements(); e.hasMoreElements();) {
FloatListener listener = (FloatListener) e.nextElement();
listener.floatChanged(event);
}
}
void setLabelString() {
// Need to muck around to try to make sure that the width of the label
// is wide enough for the largest value. Pad the string
// be large enough to hold the largest value.
int pad = 5; // fudge to make up for variable width fonts
float maxVal = Math.max(Math.abs(min), Math.abs(max));
intDigits = Math.round((float) (Math.log(maxVal) / Math.log(10))) + pad;
if (min < 0) {
intDigits++; // add one for the '-'
}
// fractDigits is num digits of resolution for fraction. Use base 10 log
// of scale, rounded up, + 2.
fractDigits = (int) Math.ceil((Math.log(scale) / Math.log(10)));
nf.setMinimumFractionDigits(fractDigits);
nf.setMaximumFractionDigits(fractDigits);
String value = nf.format(current);
while (value.length() < (intDigits + fractDigits)) {
value = value + " ";
}
valueLabel.setText(value);
}
}
class FloatEvent extends EventObject {
float value;
FloatEvent(Object source, float newValue) {
super(source);
value = newValue;
}
float getValue() {
return value;
}
}
interface FloatListener extends EventListener {
void floatChanged(FloatEvent e);
}
class LogFloatLabelJSlider extends JPanel implements ChangeListener,
Java3DExplorerConstants {
JSlider slider;
JLabel valueLabel;
Vector listeners = new Vector();
float min, max, resolution, current, scale;
double minLog, maxLog, curLog;
int minInt, maxInt, curInt;;
int intDigits, fractDigits;
NumberFormat nf = NumberFormat.getInstance();
float minResolution = 0.001f;
double logBase = Math.log(10);
// default slider with name, resolution = 0.1, min = 0.0, max = 1.0 inital
// 0.5
LogFloatLabelJSlider(String name) {
this(name, 0.1f, 100.0f, 10.0f);
}
LogFloatLabelJSlider(String name, float min, float max, float current) {
this.resolution = resolution;
this.min = min;
this.max = max;
this.current = current;
if (resolution < minResolution) {
resolution = minResolution;
}
minLog = log10(min);
maxLog = log10(max);
curLog = log10(current);
// resolution is 100 steps from min to max
scale = 100.0f;
resolution = 1.0f / scale;
// get the integer versions of max, min, current
minInt = (int) Math.round(minLog * scale);
maxInt = (int) Math.round(maxLog * scale);
curInt = (int) Math.round(curLog * scale);
slider = new JSlider(JSlider.HORIZONTAL, minInt, maxInt, curInt);
slider.addChangeListener(this);
valueLabel = new JLabel(" ");
// Need to muck around to make sure that the width of the label
// is wide enough for the largest value. Pad the initial string
// be large enough to hold the largest value.
int pad = 5; // fudge to make up for variable width fonts
intDigits = (int) Math.ceil(maxLog) + pad;
if (min < 0) {
intDigits++; // add one for the '-'
}
if (minLog < 0) {
fractDigits = (int) Math.ceil(-minLog);
} else {
fractDigits = 0;
}
nf.setMinimumFractionDigits(fractDigits);
nf.setMaximumFractionDigits(fractDigits);
String value = nf.format(current);
while (value.length() < (intDigits + fractDigits)) {
value = value + " ";
}
valueLabel.setText(value);
// add min and max labels to the slider
Hashtable labelTable = new Hashtable();
labelTable.put(new Integer(minInt), new JLabel(nf.format(min)));
labelTable.put(new Integer(maxInt), new JLabel(nf.format(max)));
slider.setLabelTable(labelTable);
slider.setPaintLabels(true);
// layout to align left
setLayout(new BorderLayout());
Box box = new Box(BoxLayout.X_AXIS);
add(box, BorderLayout.WEST);
box.add(new JLabel(name));
box.add(slider);
box.add(valueLabel);
}
public void setMinorTickSpacing(float spacing) {
int intSpacing = Math.round(spacing * scale);
slider.setMinorTickSpacing(intSpacing);
}
public void setMajorTickSpacing(float spacing) {
int intSpacing = Math.round(spacing * scale);
slider.setMajorTickSpacing(intSpacing);
}
public void setPaintTicks(boolean paint) {
slider.setPaintTicks(paint);
}
public void addFloatListener(FloatListener listener) {
listeners.add(listener);
}
public void removeFloatListener(FloatListener listener) {
listeners.remove(listener);
}
public void stateChanged(ChangeEvent e) {
JSlider source = (JSlider) e.getSource();
curInt = source.getValue();
curLog = curInt / scale;
current = (float) exp10(curLog);
valueChanged();
}
public void setValue(float newValue) {
boolean changed = (newValue != current);
current = newValue;
if (changed) {
valueChanged();
}
}
private void valueChanged() {
String value = nf.format(current);
valueLabel.setText(value);
// notify the listeners
FloatEvent event = new FloatEvent(this, current);
for (Enumeration e = listeners.elements(); e.hasMoreElements();) {
FloatListener listener = (FloatListener) e.nextElement();
listener.floatChanged(event);
}
}
double log10(double value) {
return Math.log(value) / logBase;
}
double exp10(double value) {
return Math.exp(value * logBase);
}
}
class CoordSys extends Switch {
// Temporaries that are reused
Transform3D tmpTrans = new Transform3D();
Vector3f tmpVector = new Vector3f();
AxisAngle4f tmpAxisAngle = new AxisAngle4f();
// colors for use in the shapes
Color3f black = new Color3f(0.0f, 0.0f, 0.0f);
Color3f grey = new Color3f(0.3f, 0.3f, 0.3f);
Color3f white = new Color3f(1.0f, 1.0f, 1.0f);
// geometric constants
Point3f origin = new Point3f();
Vector3f yAxis = new Vector3f(0.0f, 1.0f, 0.0f);
CoordSys(float axisLength) {
super(Switch.CHILD_ALL);
float coordSysLength = axisLength;
float labelOffset = axisLength / 20.0f;
float axisRadius = axisLength / 500.0f;
float arrowRadius = axisLength / 125.0f;
float arrowHeight = axisLength / 50.0f;
float tickRadius = axisLength / 125.0f;
float tickHeight = axisLength / 250.0f;
// Set the Switch to allow changes
setCapability(Switch.ALLOW_SWITCH_READ);
setCapability(Switch.ALLOW_SWITCH_WRITE);
// Set up an appearance to make the Axis have
// grey ambient, black emmissive, grey diffuse and grey specular
// coloring.
//Material material = new Material(grey, black, grey, white, 64);
Material material = new Material(white, black, white, white, 64);
Appearance appearance = new Appearance();
appearance.setMaterial(material);
// Create a shared group to hold one axis of the coord sys
SharedGroup coordAxisSG = new SharedGroup();
// create a cylinder for the central line of the axis
Cylinder cylinder = new Cylinder(axisRadius, coordSysLength, appearance);
// cylinder goes from -coordSysLength/2 to coordSysLength in y
coordAxisSG.addChild(cylinder);
// create the shared arrowhead
Cone arrowHead = new Cone(arrowRadius, arrowHeight, appearance);
SharedGroup arrowHeadSG = new SharedGroup();
arrowHeadSG.addChild(arrowHead);
// Create a TransformGroup to move the arrowhead to the top of the
// axis
// The arrowhead goes from -arrowHeight/2 to arrowHeight/2 in y.
// Put it at the top of the axis, coordSysLength / 2
tmpVector.set(0.0f, coordSysLength / 2 + arrowHeight / 2, 0.0f);
tmpTrans.set(tmpVector);
TransformGroup topTG = new TransformGroup();
topTG.setTransform(tmpTrans);
topTG.addChild(new Link(arrowHeadSG));
coordAxisSG.addChild(topTG);
// create the minus arrowhead
// Create a TransformGroup to turn the cone upside down:
// Rotate 180 degrees around Z axis
tmpAxisAngle.set(0.0f, 0.0f, 1.0f, (float) Math.toRadians(180));
tmpTrans.set(tmpAxisAngle);
// Put the arrowhead at the bottom of the axis
tmpVector.set(0.0f, -coordSysLength / 2 - arrowHeight / 2, 0.0f);
tmpTrans.setTranslation(tmpVector);
TransformGroup bottomTG = new TransformGroup();
bottomTG.setTransform(tmpTrans);
bottomTG.addChild(new Link(arrowHeadSG));
coordAxisSG.addChild(bottomTG);
// Now add "ticks" at 1, 2, 3, etc.
// create a shared group for the tick
Cylinder tick = new Cylinder(tickRadius, tickHeight, appearance);
SharedGroup tickSG = new SharedGroup();
tickSG.addChild(tick);
// transform each instance and add it to the coord axis group
int maxTick = (int) (coordSysLength / 2);
int minTick = -maxTick;
for (int i = minTick; i <= maxTick; i++) {
if (i == 0)
continue; // no tick at 0
// use a TransformGroup to offset to the tick location
TransformGroup tickTG = new TransformGroup();
tmpVector.set(0.0f, (float) i, 0.0f);
tmpTrans.set(tmpVector);
tickTG.setTransform(tmpTrans);
// then link to an instance of the Tick shared group
tickTG.addChild(new Link(tickSG));
// add the TransformGroup to the coord axis
coordAxisSG.addChild(tickTG);
}
// add a Link to the axis SharedGroup to the coordSys
addChild(new Link(coordAxisSG)); // Y axis
// Create TransformGroups for the X and Z axes
TransformGroup xAxisTG = new TransformGroup();
// rotate 90 degrees around Z axis
tmpAxisAngle.set(0.0f, 0.0f, 1.0f, (float) Math.toRadians(90));
tmpTrans.set(tmpAxisAngle);
xAxisTG.setTransform(tmpTrans);
xAxisTG.addChild(new Link(coordAxisSG));
addChild(xAxisTG); // X axis
TransformGroup zAxisTG = new TransformGroup();
// rotate 90 degrees around X axis
tmpAxisAngle.set(1.0f, 0.0f, 0.0f, (float) Math.toRadians(90));
tmpTrans.set(tmpAxisAngle);
zAxisTG.setTransform(tmpTrans);
zAxisTG.addChild(new Link(coordAxisSG));
addChild(zAxisTG); // Z axis
// Add the labels. First we need a Font3D for the Text3Ds
// select the default font, plain style, 0.5 tall. Use null for
// the extrusion so we get "flat" text since we will be putting it
// into an oriented Shape3D
Font3D f3d = new Font3D(new Font("Default", Font.PLAIN, 1), null);
// set up the +X label
Text3D plusXText = new Text3D(f3d, "+X", origin, Text3D.ALIGN_CENTER,
Text3D.PATH_RIGHT);
// orient around the local origin
OrientedShape3D plusXTextShape = new OrientedShape3D(plusXText,
appearance, OrientedShape3D.ROTATE_ABOUT_POINT, origin);
// transform to scale down to 0.15 in height, locate at end of axis
TransformGroup plusXTG = new TransformGroup();
tmpVector.set(coordSysLength / 2 + labelOffset, 0.0f, 0.0f);
tmpTrans.set(0.15f, tmpVector);
plusXTG.setTransform(tmpTrans);
plusXTG.addChild(plusXTextShape);
addChild(plusXTG);
// set up the -X label
Text3D minusXText = new Text3D(f3d, "-X", origin, Text3D.ALIGN_CENTER,
Text3D.PATH_RIGHT);
// orient around the local origin
OrientedShape3D minusXTextShape = new OrientedShape3D(minusXText,
appearance, OrientedShape3D.ROTATE_ABOUT_POINT, origin);
// transform to scale down to 0.15 in height, locate at end of axis
TransformGroup minusXTG = new TransformGroup();
tmpVector.set(-coordSysLength / 2 - labelOffset, 0.0f, 0.0f);
tmpTrans.set(0.15f, tmpVector);
minusXTG.setTransform(tmpTrans);
minusXTG.addChild(minusXTextShape);
addChild(minusXTG);
// set up the +Y label
Text3D plusYText = new Text3D(f3d, "+Y", origin, Text3D.ALIGN_CENTER,
Text3D.PATH_RIGHT);
// orient around the local origin
OrientedShape3D plusYTextShape = new OrientedShape3D(plusYText,
appearance, OrientedShape3D.ROTATE_ABOUT_POINT, origin);
// transform to scale down to 0.15 in height, locate at end of axis
TransformGroup plusYTG = new TransformGroup();
tmpVector.set(0.0f, coordSysLength / 2 + labelOffset, 0.0f);
tmpTrans.set(0.15f, tmpVector);
plusYTG.setTransform(tmpTrans);
plusYTG.addChild(plusYTextShape);
addChild(plusYTG);
// set up the -Y label
Text3D minusYText = new Text3D(f3d, "-Y", origin, Text3D.ALIGN_CENTER,
Text3D.PATH_RIGHT);
// orient around the local origin
OrientedShape3D minusYTextShape = new OrientedShape3D(minusYText,
appearance, OrientedShape3D.ROTATE_ABOUT_POINT, origin);
// transform to scale down to 0.15 in height, locate at end of axis
TransformGroup minusYTG = new TransformGroup();
tmpVector.set(0.0f, -coordSysLength / 2 - labelOffset, 0.0f);
tmpTrans.set(0.15f, tmpVector);
minusYTG.setTransform(tmpTrans);
minusYTG.addChild(minusYTextShape);
addChild(minusYTG);
// set up the +Z label
Text3D plusZText = new Text3D(f3d, "+Z", origin, Text3D.ALIGN_CENTER,
Text3D.PATH_RIGHT);
// orient around the local origin
OrientedShape3D plusZTextShape = new OrientedShape3D(plusZText,
appearance, OrientedShape3D.ROTATE_ABOUT_POINT, origin);
// transform to scale down to 0.15 in height, locate at end of axis
TransformGroup plusZTG = new TransformGroup();
tmpVector.set(0.0f, 0.0f, coordSysLength / 2 + labelOffset);
tmpTrans.set(0.15f, tmpVector);
plusZTG.setTransform(tmpTrans);
plusZTG.addChild(plusZTextShape);
addChild(plusZTG);
// set up the -Z label
Text3D minusZText = new Text3D(f3d, "-Z", origin, Text3D.ALIGN_CENTER,
Text3D.PATH_RIGHT);
// orient around the local origin
OrientedShape3D minusZTextShape = new OrientedShape3D(minusZText,
appearance, OrientedShape3D.ROTATE_ABOUT_POINT, origin);
// transform to scale down to 0.15 in height, locate at end of axis
TransformGroup minusZTG = new TransformGroup();
tmpVector.set(0.0f, 0.0f, -coordSysLength / 2 - labelOffset);
tmpTrans.set(0.15f, tmpVector);
minusZTG.setTransform(tmpTrans);
minusZTG.addChild(minusZTextShape);
addChild(minusZTG);
}
}
class LeftAlignComponent extends JPanel {
LeftAlignComponent(Component c) {
setLayout(new BorderLayout());
add(c, BorderLayout.WEST);
}
}
class RotAxis extends Switch implements Java3DExplorerConstants {
// axis to align with
Vector3f rotAxis = new Vector3f(1.0f, 0.0f, 0.0f);
// offset to ref point
Vector3f refPt = new Vector3f(0.0f, 0.0f, 0.0f);
TransformGroup axisTG; // the transform group used to align the axis
// Temporaries that are reused
Transform3D tmpTrans = new Transform3D();
Vector3f tmpVector = new Vector3f();
AxisAngle4f tmpAxisAngle = new AxisAngle4f();
// geometric constants
Point3f origin = new Point3f();
Vector3f yAxis = new Vector3f(0.0f, 1.0f, 0.0f);
RotAxis(float axisLength) {
super(Switch.CHILD_NONE);
setCapability(Switch.ALLOW_SWITCH_READ);
setCapability(Switch.ALLOW_SWITCH_WRITE);
// set up the proportions for the arrow
float axisRadius = axisLength / 120.0f;
float arrowRadius = axisLength / 50.0f;
float arrowHeight = axisLength / 30.0f;
// create the TransformGroup which will be used to orient the axis
axisTG = new TransformGroup();
axisTG.setCapability(TransformGroup.ALLOW_TRANSFORM_WRITE);
axisTG.setCapability(TransformGroup.ALLOW_TRANSFORM_READ);
addChild(axisTG);
// Set up an appearance to make the Axis have
// blue ambient, black emmissive, blue diffuse and white specular
// coloring.
Material material = new Material(blue, black, blue, white, 64);
Appearance appearance = new Appearance();
appearance.setMaterial(material);
// create a cylinder for the central line of the axis
Cylinder cylinder = new Cylinder(axisRadius, axisLength, appearance);
// cylinder goes from -length/2 to length/2 in y
axisTG.addChild(cylinder);
// create a SharedGroup for the arrowHead
Cone arrowHead = new Cone(arrowRadius, arrowHeight, appearance);
SharedGroup arrowHeadSG = new SharedGroup();
arrowHeadSG.addChild(arrowHead);
// Create a TransformGroup to move the cone to the top of the
// cylinder
tmpVector.set(0.0f, axisLength/2 + arrowHeight / 2, 0.0f);
tmpTrans.set(tmpVector);
TransformGroup topTG = new TransformGroup();
topTG.setTransform(tmpTrans);
topTG.addChild(new Link(arrowHeadSG));
axisTG.addChild(topTG);
// create the bottom of the arrow
// Create a TransformGroup to move the cone to the bottom of the
// axis so that its pushes into the bottom of the cylinder
tmpVector.set(0.0f, -(axisLength / 2), 0.0f);
tmpTrans.set(tmpVector);
TransformGroup bottomTG = new TransformGroup();
bottomTG.setTransform(tmpTrans);
bottomTG.addChild(new Link(arrowHeadSG));
axisTG.addChild(bottomTG);
updateAxisTransform();
}
public void setRotationAxis(Vector3f setRotAxis) {
rotAxis.set(setRotAxis);
float magSquared = rotAxis.lengthSquared();
if (magSquared > 0.0001) {
rotAxis.scale((float)(1.0 / Math.sqrt(magSquared)));
} else {
rotAxis.set(1.0f, 0.0f, 0.0f);
}
updateAxisTransform();
}
public void setRefPt(Vector3f setRefPt) {
refPt.set(setRefPt);
updateAxisTransform();
}
// set the transform on the axis so that it aligns with the rotation
// axis and goes through the reference point
private void updateAxisTransform() {
// We need to rotate the axis, which is defined along the y-axis,
// to the direction indicated by the rotAxis.
// We can do this using a neat trick. To transform a vector to align
// with another vector (assuming both vectors have unit length), take
// the cross product the the vectors. The direction of the cross
// product is the axis, and the length of the cross product is the
// the sine of the angle, so the inverse sine of the length gives
// us the angle
tmpVector.cross(yAxis, rotAxis);
float angle = (float)Math.asin(tmpVector.length());
tmpAxisAngle.set(tmpVector, angle);
tmpTrans.set(tmpAxisAngle);
tmpTrans.setTranslation(refPt);
axisTG.setTransform(tmpTrans);
}
}
|