A complete 3D game development suite written purely in Java.
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 
jmonkeyengine/sdk/jme3-scenecomposer/src/com/jme3/gde/scenecomposer/SceneComposerToolController...

503 lines
16 KiB

/*
* To change this template, choose Tools | Templates
* and open the template in the editor.
*/
package com.jme3.gde.scenecomposer;
import com.jme3.asset.AssetManager;
import com.jme3.audio.AudioNode;
import com.jme3.gde.core.scene.SceneApplication;
import com.jme3.gde.core.scene.controller.SceneToolController;
import com.jme3.gde.core.sceneexplorer.nodes.JmeNode;
import com.jme3.gde.core.sceneexplorer.nodes.JmeSpatial;
import com.jme3.input.event.KeyInputEvent;
import com.jme3.light.Light;
import com.jme3.light.PointLight;
import com.jme3.light.SpotLight;
import com.jme3.material.Material;
import com.jme3.material.RenderState.BlendMode;
import com.jme3.math.Vector2f;
import com.jme3.math.Vector3f;
import com.jme3.renderer.Camera;
import com.jme3.renderer.RenderManager;
import com.jme3.renderer.ViewPort;
import com.jme3.renderer.queue.RenderQueue.Bucket;
import com.jme3.scene.Geometry;
import com.jme3.scene.Node;
import com.jme3.scene.Spatial;
import com.jme3.scene.control.BillboardControl;
import com.jme3.scene.control.Control;
import com.jme3.scene.shape.Quad;
import com.jme3.texture.Texture;
import java.util.concurrent.Callable;
/**
*
* @author Brent Owens
*/
public class SceneComposerToolController extends SceneToolController {
private JmeNode rootNode;
private SceneEditTool editTool;
private SceneEditorController editorController;
private ComposerCameraController cameraController;
private ViewPort overlayView;
private Node onTopToolsNode;
private Node nonSpatialMarkersNode;
private Material lightMarkerMaterial;
private Material audioMarkerMaterial;
private JmeSpatial selectedSpatial;
private boolean snapToGrid = false;
private boolean snapToScene = false;
private boolean selectTerrain = false;
private boolean selectGeometries = false;
public SceneComposerToolController(final Node toolsNode, AssetManager manager, JmeNode rootNode) {
super(toolsNode, manager);
this.rootNode = rootNode;
nonSpatialMarkersNode = new Node("lightMarkersNode");
SceneApplication.getApplication().enqueue(new Callable<Object>() {
public Object call() throws Exception {
toolsNode.attachChild(nonSpatialMarkersNode);
return null;
}
});
}
public SceneComposerToolController(AssetManager manager) {
super(manager);
}
public void setEditorController(SceneEditorController editorController) {
this.editorController = editorController;
}
public void setCameraController(ComposerCameraController cameraController) {
this.cameraController = cameraController;
// a node in a viewport that will always render on top
onTopToolsNode = new Node("OverlayNode");
overlayView = SceneApplication.getApplication().getOverlayView();
SceneApplication.getApplication().enqueue(new Callable<Void>() {
public Void call() throws Exception {
overlayView.attachScene(onTopToolsNode);
return null;
}
});
}
@Override
public void cleanup() {
super.cleanup();
cameraController = null;
editorController = null;
SceneApplication.getApplication().enqueue(new Callable<Void>() {
public Void call() throws Exception {
overlayView.detachScene(onTopToolsNode);
onTopToolsNode.detachAllChildren();
return null;
}
});
}
@Override
public void update(float tpf) {
super.update(tpf);
if (editTool != null) {
editTool.doUpdateToolsTransformation();
}
if (onTopToolsNode != null) {
onTopToolsNode.updateLogicalState(tpf);
onTopToolsNode.updateGeometricState();
}
}
@Override
public void render(RenderManager rm) {
super.render(rm);
}
public boolean isEditToolEnabled() {
return editTool != null;
}
/**
* If the current tool overrides camera zoom/pan controls
*/
public boolean isOverrideCameraControl() {
if (editTool != null) {
return editTool.isOverrideCameraControl();
} else {
return false;
}
}
/**
* Scene composer edit tool activated. Pass in null to remove tools.
*
* @param sceneEditButton pass in null to hide any existing tool markers
*/
public void showEditTool(final SceneEditTool sceneEditTool) {
SceneApplication.getApplication().enqueue(new Callable<Object>() {
public Object call() throws Exception {
doEnableEditTool(sceneEditTool);
return null;
}
});
}
private void doEnableEditTool(SceneEditTool sceneEditTool) {
if (editTool != null) {
editTool.hideMarker();
}
editTool = sceneEditTool;
editTool.activate(manager, toolsNode, onTopToolsNode, selected, this);
}
public void selectedSpatialTransformed() {
if (editTool != null) {
editTool.updateToolsTransformation();
}
}
public void setSelected(Spatial selected) {
this.selected = selected;
}
public void setNeedsSave(boolean needsSave) {
editorController.setNeedsSave(needsSave);
}
/**
* Primary button activated, send command to the tool
* for appropriate action.
*/
public void doEditToolActivatedPrimary(Vector2f mouseLoc, boolean pressed, Camera camera) {
if (editTool != null) {
editTool.setCamera(camera);
editTool.actionPrimary(mouseLoc, pressed, rootNode, editorController.getCurrentDataObject());
}
}
/**
* Secondary button activated, send command to the tool
* for appropriate action.
*/
public void doEditToolActivatedSecondary(Vector2f mouseLoc, boolean pressed, Camera camera) {
if (editTool != null) {
editTool.setCamera(camera);
editTool.actionSecondary(mouseLoc, pressed, rootNode, editorController.getCurrentDataObject());
}
}
public void doEditToolMoved(Vector2f mouseLoc, Camera camera) {
if (editTool != null) {
editTool.setCamera(camera);
editTool.mouseMoved(mouseLoc, rootNode, editorController.getCurrentDataObject(), selectedSpatial);
}
}
public void doEditToolDraggedPrimary(Vector2f mouseLoc, boolean pressed, Camera camera) {
if (editTool != null) {
editTool.setCamera(camera);
editTool.draggedPrimary(mouseLoc, pressed, rootNode, editorController.getCurrentDataObject());
}
}
public void doEditToolDraggedSecondary(Vector2f mouseLoc, boolean pressed, Camera camera) {
if (editTool != null) {
editTool.setCamera(camera);
editTool.draggedSecondary(mouseLoc, pressed, rootNode, editorController.getCurrentDataObject());
}
}
void doKeyPressed(KeyInputEvent kie) {
if (editTool != null) {
editTool.keyPressed(kie);
}
}
/**
* Adds a marker for the light to the scene if it does not exist yet
*/
public void addLightMarker(Light light) {
if (!(light instanceof PointLight) && !(light instanceof SpotLight))
return; // only handle point and spot lights
Spatial s = nonSpatialMarkersNode.getChild(light.getName());
if (s != null) {
// update location maybe? Remove old and replace with new?
return;
}
LightMarker lm = new LightMarker(light);
nonSpatialMarkersNode.attachChild(lm);
}
public void addAudioMarker(AudioNode audio) {
Spatial s = nonSpatialMarkersNode.getChild(audio.getName());
if (s != null) {
// update location maybe? Remove old and replace with new?
return;
}
AudioMarker am = new AudioMarker(audio);
nonSpatialMarkersNode.attachChild(am);
}
/**
* Removes a light marker from the scene's tool node
*/
public void removeLightMarker(Light light) {
Spatial s = nonSpatialMarkersNode.getChild(light.getName());
s.removeFromParent();
}
private Material getLightMarkerMaterial() {
if (lightMarkerMaterial == null) {
Material mat = new Material(manager, "Common/MatDefs/Misc/Unshaded.j3md");
Texture tex = manager.loadTexture("com/jme3/gde/scenecomposer/lightbulb32.png");
mat.setTexture("ColorMap", tex);
mat.getAdditionalRenderState().setBlendMode(BlendMode.Alpha);
lightMarkerMaterial = mat;
}
return lightMarkerMaterial;
}
private Material getAudioMarkerMaterial() {
if (audioMarkerMaterial == null) {
Material mat = new Material(manager, "Common/MatDefs/Misc/Unshaded.j3md");
Texture tex = manager.loadTexture("com/jme3/gde/scenecomposer/audionode.gif");
mat.setTexture("ColorMap", tex);
mat.getAdditionalRenderState().setBlendMode(BlendMode.Alpha);
audioMarkerMaterial = mat;
}
return audioMarkerMaterial;
}
protected void refreshNonSpatialMarkers() {
SceneApplication.getApplication().enqueue(new Callable<Void>() {
public Void call() throws Exception {
nonSpatialMarkersNode.detachAllChildren();
addMarkers(rootNode.getLookup().lookup(Node.class));
return null;
}
});
}
private void addMarkers(Node parent) {
for (Light light : parent.getLocalLightList())
addLightMarker(light);
if (parent instanceof AudioNode) {
addAudioMarker((AudioNode)parent);
}
for (Spatial s : parent.getChildren()) {
if (s instanceof Node)
addMarkers((Node)s);
else {
//TODO later if we include other types of non-spatial markers
}
}
}
public boolean isSnapToGrid() {
return snapToGrid;
}
public void setSnapToGrid(boolean snapToGrid) {
this.snapToGrid = snapToGrid;
}
public void setSnapToScene(boolean snapToScene) {
this.snapToScene = snapToScene;
}
public boolean isSnapToScene() {
return snapToScene;
}
public boolean selectTerrain() {
return selectTerrain;
}
public void setSelectTerrain(boolean selectTerrain) {
this.selectTerrain = selectTerrain;
}
public boolean isSelectGeometries() {
return selectGeometries;
}
public void setSelectGeometries(boolean selectGeometries) {
this.selectGeometries = selectGeometries;
}
/**
* A marker on the screen that shows where a point light or
* a spot light is. This marker is not part of the scene,
* but is part of the tools node.
*/
protected class LightMarker extends Geometry {
private Light light;
protected LightMarker() {}
protected LightMarker(Light light) {
this.light = light;
Quad q = new Quad(0.5f, 0.5f);
this.setMesh(q);
this.setMaterial(getLightMarkerMaterial());
this.addControl(new LightMarkerControl());
this.setQueueBucket(Bucket.Transparent);
}
protected Light getLight() {
return light;
}
@Override
public void setLocalTranslation(Vector3f location) {
super.setLocalTranslation(location);
if (light instanceof PointLight)
((PointLight)light).setPosition(location);
else if (light instanceof SpotLight)
((SpotLight)light).setPosition(location);
}
@Override
public void setLocalTranslation(float x, float y, float z) {
super.setLocalTranslation(x, y, z);
if (light instanceof PointLight)
((PointLight)light).setPosition(new Vector3f(x,y,z));
else if (light instanceof SpotLight)
((SpotLight)light).setPosition(new Vector3f(x,y,z));
}
}
/**
* Updates the marker's position whenever the light has moved.
* It is also a BillboardControl, so this marker always faces
* the camera
*/
protected class LightMarkerControl extends BillboardControl {
LightMarkerControl(){
super();
}
@Override
protected void controlUpdate(float f) {
super.controlUpdate(f);
LightMarker marker = (LightMarker) getSpatial();
if (marker != null) {
if (marker.getLight() instanceof PointLight) {
marker.setLocalTranslation(((PointLight)marker.getLight()).getPosition());
} else if (marker.getLight() instanceof SpotLight) {
marker.setLocalTranslation(((SpotLight)marker.getLight()).getPosition());
}
}
}
@Override
protected void controlRender(RenderManager rm, ViewPort vp) {
super.controlRender(rm, vp);
}
@Override
public Control cloneForSpatial(Spatial sptl) {
LightMarkerControl c = new LightMarkerControl();
c.setSpatial(sptl);
//TODO this isn't correct, none of BillboardControl is copied over
return c;
}
}
/**
* A marker on the screen that shows where an audio node is.
* This marker is not part of the scene, but is part of the tools node.
*/
protected class AudioMarker extends Geometry {
private AudioNode audio;
protected AudioMarker() {}
protected AudioMarker(AudioNode audio) {
this.audio = audio;
Quad q = new Quad(0.5f, 0.5f);
this.setMesh(q);
this.setMaterial(getAudioMarkerMaterial());
this.addControl(new AudioMarkerControl());
this.setQueueBucket(Bucket.Transparent);
}
protected AudioNode getAudioNode() {
return audio;
}
@Override
public void setLocalTranslation(Vector3f location) {
super.setLocalTranslation(location);
}
@Override
public void setLocalTranslation(float x, float y, float z) {
super.setLocalTranslation(x, y, z);
}
}
/**
* Updates the marker's position whenever the audio node has moved.
* It is also a BillboardControl, so this marker always faces
* the camera
*/
protected class AudioMarkerControl extends BillboardControl {
AudioMarkerControl(){
super();
}
@Override
protected void controlUpdate(float f) {
super.controlUpdate(f);
AudioMarker marker = (AudioMarker) getSpatial();
if (marker != null) {
marker.setLocalTranslation(marker.getAudioNode().getWorldTranslation());
}
}
@Override
protected void controlRender(RenderManager rm, ViewPort vp) {
super.controlRender(rm, vp);
}
@Override
public Control cloneForSpatial(Spatial sptl) {
AudioMarkerControl c = new AudioMarkerControl();
c.setSpatial(sptl);
//TODO this isn't correct, none of BillboardControl is copied over
return c;
}
}
public JmeNode getRootNode() {
return rootNode;
}
public void setSelectedSpatial(JmeSpatial selectedSpatial) {
this.selectedSpatial = selectedSpatial;
}
}