sim.portrayal3d.SimplePortrayal3D Maven / Gradle / Ivy
Show all versions of mason Show documentation
/*
Copyright 2006 by Sean Luke and George Mason University
Licensed under the Academic Free License version 3.0
See the file "LICENSE" for more information
*/
package sim.portrayal3d;
import sim.portrayal.*;
import sim.display.*;
import javax.media.j3d.*;
import com.sun.j3d.utils.image.*;
import java.awt.*;
import javax.vecmath.*;
import java.util.*;
import sim.display3d.*;
/** The superclass of all 3D Simple Portrayals which by default adds nothing to the 3D
scene. Since nothing is added to the scene, nothing is shown and you cannot
select anything. Subclasses add items to the scene, so they can then be selected
if they so choose. By default SimplePortrayal3Ds respond to requests for inspectors by
providing a basic LabelledList which shows all the portrayed object's
object properties (see sim.util.SimpleProperties). They also respond to inspector
update requests by updating this same LabelledList. No polygonAttributes are
provided by default, and setSelected always true by default.
SimplePortrayal3Ds have a getFieldPortrayal(), which is the FieldPortrayal3D
which houses them. This value can be null if the SimplePortrayal3D was added directly
into the Display3D's collection of portrayals rather than being used inside
a field portrayal. The contract SimplePortrayal3Ds may assume is that the getFieldPortrayal(),
if it exists, will have been set prior to getModel(...) being called.
Various utility functions are provided. setPickableFlags makes a Java3D object
pickable (necessary for inspectability and selection). clearPickableFlags does the opposite.
appearanceForColor creates a Java3D appearance that's a flat version of the color
which requires no lighting -- a very basic default appearance function. If the color
has a degree of transparency, the appearance will as well.
A default appearance is provided for subclasses which wish to draw themselves
using a default: flat white opaque.
*/
public class SimplePortrayal3D implements Portrayal3D
{
/** Flat white opaque. */
public static final Appearance DEFAULT_APPEARANCE = appearanceForColor(Color.white);
/** Creates an Appearance equivalent to a flat opaque surface of the provided color, needing no lighting.
Opacity is determined by the opacity of the unlit color. */
public static Appearance appearanceForColor(Color unlitColor)
{
Appearance appearance = new Appearance();
setAppearanceFlags(appearance);
float[] c = unlitColor.getRGBComponents(null);
ColoringAttributes ca = new ColoringAttributes(c[0], c[1], c[2], ColoringAttributes.SHADE_FLAT);
ca.setCapability(ColoringAttributes.ALLOW_COLOR_WRITE);
ca.setCapability(ColoringAttributes.ALLOW_COLOR_READ);
appearance.setColoringAttributes(ca);
if (c[3] < 1.0) // partially transparent
{
TransparencyAttributes tta = new TransparencyAttributes(TransparencyAttributes.BLENDED, 1.0f - c[3]); // duh, alpha's backwards
tta.setCapability(TransparencyAttributes.ALLOW_VALUE_WRITE);
tta.setCapability(TransparencyAttributes.ALLOW_VALUE_READ);
appearance.setTransparencyAttributes(tta);
}
return appearance;
}
static final Color3f BLACK = new Color3f(Color.black);
/** Creates an Appearance with the provided lit colors. Objects will not appear if the scene is unlit.
shininess and opacity both from 0.0 to 1.0. If any color is null, it's assumed to be black.
Note that even jet black ambient color will show up as a charcoal gray under the bright white
ambient light in MASON. That's Java3D for you, sorry. */
public static Appearance appearanceForColors(Color ambientColor,
Color emissiveColor, Color diffuseColor,
Color specularColor, double shininess, double opacity)
{
Appearance appearance = new Appearance();
setAppearanceFlags(appearance);
ColoringAttributes ca = new ColoringAttributes(BLACK, ColoringAttributes.SHADE_GOURAUD);
ca.setCapability(ColoringAttributes.ALLOW_COLOR_WRITE);
ca.setCapability(ColoringAttributes.ALLOW_COLOR_READ);
appearance.setColoringAttributes(ca);
if (opacity > 1.0f) opacity = 1.0f;
if (opacity < 0.0f) opacity = 0.0f;
if (shininess > 1.0f) shininess = 1.0f;
if (shininess < 0.0f) shininess = 0.0f;
shininess = shininess * 63.0f + 1.0f; // to go between 1.0 and 64.0
Material m = new Material();
m.setCapability(Material.ALLOW_COMPONENT_READ);
m.setCapability(Material.ALLOW_COMPONENT_WRITE);
if (ambientColor != null) m.setAmbientColor(new Color3f(ambientColor));
else m.setAmbientColor(BLACK);
if (emissiveColor != null) m.setEmissiveColor(new Color3f(emissiveColor));
else m.setEmissiveColor(BLACK);
if (diffuseColor != null) m.setDiffuseColor(new Color3f(diffuseColor));
else m.setDiffuseColor(BLACK);
if (specularColor != null) m.setSpecularColor(new Color3f(specularColor));
else m.setSpecularColor(BLACK);
m.setShininess((float)shininess);
appearance.setMaterial(m);
if (opacity < 1.0f) // partially transparent
{
TransparencyAttributes tta = new TransparencyAttributes(TransparencyAttributes.BLENDED, 1.0f - (float)opacity); // duh, alpha's backwards
tta.setCapability(TransparencyAttributes.ALLOW_VALUE_WRITE);
tta.setCapability(TransparencyAttributes.ALLOW_VALUE_READ);
appearance.setTransparencyAttributes(tta);
}
return appearance;
}
/** Creates an Appearance using the provided image. If the image should be entirely opaque, you
should definitely set opaque to true. If you want to use the image's built-in transparency information,
you should set opaque to false. Beware that there are bugs in Java3D's handling of transparent
textures: multiple such objects often will not draw in the correct order; thus objects in the back
may appear to be in the front. */
public static Appearance appearanceForImage(Image image, boolean opaque)
{
Appearance appearance = appearanceForColor(Color.black);
if (!opaque)
{
TransparencyAttributes tta = new TransparencyAttributes(TransparencyAttributes.BLENDED, 1.0f); // duh, alpha's backwards
tta.setCapability(TransparencyAttributes.ALLOW_VALUE_WRITE);
tta.setCapability(TransparencyAttributes.ALLOW_VALUE_READ);
appearance.setTransparencyAttributes(tta);
}
appearance.setTexture(new TextureLoader(image, TextureLoader.BY_REFERENCE, null).getTexture());
TextureAttributes ta = new TextureAttributes();
ta.setTextureMode(TextureAttributes.REPLACE);
PolygonAttributes pa = new PolygonAttributes();
pa.setCullFace(PolygonAttributes.CULL_NONE);
appearance.setPolygonAttributes(pa);
appearance.setTextureAttributes(ta);
return appearance;
}
public PolygonAttributes polygonAttributes() { return null; } // default
public TransformGroup getModel(Object object, TransformGroup prev)
{
// By default we'll not put anything in here. But we still have to
// create a TransformGroup if necessary.
if (prev==null) prev = new TransformGroup();
return prev;
}
public Inspector getInspector(LocationWrapper wrapper, GUIState state)
{
if (wrapper == null) return null;
return new SimpleInspector(wrapper.getObject(), state, "Properties");
}
public String getStatus(LocationWrapper wrapper) { return getName(wrapper); }
public String getName(LocationWrapper wrapper)
{
if (wrapper == null) return "";
return "" + wrapper.getObject();
}
FieldPortrayal3D fieldPortrayal = null;
public void setCurrentFieldPortrayal(FieldPortrayal3D p)
{
fieldPortrayal = p;
}
public FieldPortrayal3D getCurrentFieldPortrayal()
{
return fieldPortrayal;
}
Display3D display = null;
public void setCurrentDisplay(Display3D display)
{
this.display = display;
}
/** If the current display has been set, returns it.
Else if the field portrayal is null, returns null.
Else queries the field portrayal for its current
display and returns that. */
public Display3D getCurrentDisplay()
{
if (display == null)
{
FieldPortrayal3D f = getCurrentFieldPortrayal();
if (f == null) return null;
else return f.getCurrentDisplay();
}
else return display;
}
public GUIState getCurrentGUIState()
{
Display3D d = getCurrentDisplay();
return (d == null ? null : d.getSimulation());
}
public boolean isSelected(Object obj)
{
return selectedObjects != null && selectedObjects.containsKey(obj);
}
HashMap selectedObjects = null;
/** If the object is selected, adds it to a hash table of selected objects for which
this SimplePortrayal3D's isSelected() method will return TRUE. If the object is
deselected, removes it from the hash table. Always returns TRUE. The hash table
doesn't exist until this method is first called.
There are two implications to this approach. First, it means that after you've
selected an object, there's a hash table attached to its portrayal. If you're using
the same portrayal for lots of stuff, that's no big deal. But if you've got per-object
portrayals and a lot of objects (or if the objects are themselves SimplePortrayal3D
subclasses) then that's a fair number of hash tables. This is a minor memory issue but
if you don't care about testing for whether you've been selected or not, you could
just override this method to always return TRUE (and don't call super.selected(...) )
and the hash table will never be created. Note that isSelected will always return
FALSE for your portrayal in this situation.
Second, though you can test for selection with the isSelected() method, if you want
to, say, change the look of your portrayal based on whether or not it's selected, you
will need to test isSelected() each time getModel() is called and modify the model
accordingly. This could be a bit expensive. We're working on an approach for you to
be able to test if the object was RECENTLY selected or deselected so you could only test
then, but it's nontrivial to do without using up a lot of memory.
*/
public boolean setSelected(LocationWrapper wrapper, boolean selected)
{
if (selectedObjects == null)
selectedObjects = new HashMap(1); // be conservative
if (selected)
selectedObjects.put(wrapper.getObject(), wrapper);
else
{
selectedObjects.remove(wrapper.getObject());
if (selectedObjects.isEmpty())
selectedObjects = null;
}
return true;
}
/** Sets a variety of flags on an Appearance so that its features can be modified
when the scene is live. This method cannot be called on an Appearance presently
used in a live scene. */
public static void setAppearanceFlags(Appearance a)
{
a.setCapability(Appearance.ALLOW_COLORING_ATTRIBUTES_READ);
a.clearCapabilityIsFrequent(Appearance.ALLOW_COLORING_ATTRIBUTES_READ);
a.setCapability(Appearance.ALLOW_COLORING_ATTRIBUTES_WRITE);
a.clearCapabilityIsFrequent(Appearance.ALLOW_COLORING_ATTRIBUTES_WRITE);
a.setCapability(Appearance.ALLOW_MATERIAL_READ);
a.clearCapabilityIsFrequent(Appearance.ALLOW_MATERIAL_READ);
a.setCapability(Appearance.ALLOW_MATERIAL_WRITE);
a.clearCapabilityIsFrequent(Appearance.ALLOW_MATERIAL_WRITE);
a.setCapability(Appearance.ALLOW_POLYGON_ATTRIBUTES_READ);
a.clearCapabilityIsFrequent(Appearance.ALLOW_POLYGON_ATTRIBUTES_READ);
a.setCapability(Appearance.ALLOW_POLYGON_ATTRIBUTES_WRITE);
a.clearCapabilityIsFrequent(Appearance.ALLOW_POLYGON_ATTRIBUTES_WRITE);
a.setCapability(Appearance.ALLOW_TRANSPARENCY_ATTRIBUTES_READ);
a.clearCapabilityIsFrequent(Appearance.ALLOW_TRANSPARENCY_ATTRIBUTES_READ);
a.setCapability(Appearance.ALLOW_TRANSPARENCY_ATTRIBUTES_WRITE);
a.clearCapabilityIsFrequent(Appearance.ALLOW_TRANSPARENCY_ATTRIBUTES_WRITE);
a.setCapability(Appearance.ALLOW_TEXTURE_ATTRIBUTES_READ);
a.clearCapabilityIsFrequent(Appearance.ALLOW_TEXTURE_ATTRIBUTES_READ);
a.setCapability(Appearance.ALLOW_TEXTURE_ATTRIBUTES_WRITE);
a.clearCapabilityIsFrequent(Appearance.ALLOW_TEXTURE_ATTRIBUTES_WRITE);
a.setCapability(Appearance.ALLOW_TEXTURE_READ);
a.clearCapabilityIsFrequent(Appearance.ALLOW_TEXTURE_READ);
a.setCapability(Appearance.ALLOW_TEXTURE_WRITE);
a.clearCapabilityIsFrequent(Appearance.ALLOW_TEXTURE_WRITE);
}
/** Utility method which prepares the given Shape3D to be pickable (for selection and inspection). */
public static void setPickableFlags(Shape3D s3d)
{
s3d.setCapability(Shape3D.ALLOW_GEOMETRY_READ);
setPickableFlags(s3d.getGeometry());
// these are not going to be common, so we should state that they are infrequent
s3d.clearCapabilityIsFrequent(Shape3D.ALLOW_GEOMETRY_READ);
}
/** Utility method which prepares the given Geometry to be pickable (for selection and inspection). */
public static void setPickableFlags(Geometry geom)
{
geom.setCapability(GeometryArray.ALLOW_COUNT_READ);
geom.setCapability(GeometryArray.ALLOW_FORMAT_READ);
geom.setCapability(GeometryArray.ALLOW_COORDINATE_READ);
// these are not going to be common, so we should state that they are infrequent
geom.clearCapabilityIsFrequent(GeometryArray.ALLOW_COUNT_READ);
geom.clearCapabilityIsFrequent(GeometryArray.ALLOW_FORMAT_READ);
geom.clearCapabilityIsFrequent(GeometryArray.ALLOW_COORDINATE_READ);
}
/** Utility method which makes the given Node unpickable. */
static public void clearPickableFlags(Node n)
{
n.setPickable(false);
n.clearCapability(Group.ENABLE_PICK_REPORTING);
}
}