This VREnvironment is independent from the JMonkey stuff and enables to check and initialize VR specific capabilities before initializing VRAppState. The procedure is now to initialize a VR environment and, if the initialization is ok, to attach a VRAppstate to the main application. Some class has been refactored: - System classes are within the com.jme3.system package - VR related utility classes are in the package com.jme3.util.fix-456
parent
33e6574387
commit
6cf1b57e00
@ -0,0 +1,485 @@ |
||||
package com.jme3.app; |
||||
|
||||
import java.util.Locale; |
||||
import java.util.logging.Level; |
||||
import java.util.logging.Logger; |
||||
|
||||
import com.jme3.app.state.AppState; |
||||
import com.jme3.input.vr.OSVR; |
||||
import com.jme3.input.vr.OpenVR; |
||||
import com.jme3.input.vr.VRAPI; |
||||
import com.jme3.input.vr.VRInputAPI; |
||||
import com.jme3.renderer.Camera; |
||||
import com.jme3.scene.Spatial; |
||||
import com.jme3.system.AppSettings; |
||||
import com.jme3.system.jopenvr.JOpenVRLibrary; |
||||
import com.jme3.util.VRGuiManager; |
||||
import com.jme3.util.VRMouseManager; |
||||
import com.jme3.util.VRViewManager; |
||||
import com.jme3.util.VRViewManagerOSVR; |
||||
import com.jme3.util.VRViewManagerOpenVR; |
||||
|
||||
public class VREnvironment { |
||||
|
||||
private static final Logger logger = Logger.getLogger(VREnvironment.class.getName()); |
||||
|
||||
private VRAPI hardware = null; |
||||
private VRGuiManager guiManager = null; |
||||
private VRMouseManager mouseManager = null; |
||||
private VRViewManager viewmanager = null; |
||||
|
||||
/** |
||||
* The underlying system VR API. By default set to {@link VRConstants#SETTING_VRAPI_OPENVR_VALUE}. |
||||
*/ |
||||
public int vrBinding = VRConstants.SETTING_VRAPI_OPENVR_VALUE; |
||||
|
||||
private boolean seated = false; |
||||
|
||||
private Spatial observer = null; |
||||
|
||||
private boolean forceVR = false; |
||||
|
||||
private boolean vrSupportedOS = false; |
||||
|
||||
private boolean nogui = false; |
||||
|
||||
private boolean compositorOS; |
||||
|
||||
private boolean useCompositor = true; |
||||
|
||||
private boolean instanceRendering = false; |
||||
|
||||
private boolean disableSwapBuffers = true; |
||||
|
||||
private float defaultFOV = 108f; |
||||
|
||||
private float defaultAspect = 1f; |
||||
|
||||
private AppSettings settings = null; |
||||
|
||||
private Application application = null; |
||||
|
||||
private Camera dummyCam = null; |
||||
|
||||
private AppState app = null; |
||||
|
||||
private boolean initialized = false; |
||||
|
||||
private boolean attached = false; |
||||
|
||||
public VREnvironment(AppSettings settings){ |
||||
|
||||
this.settings = settings; |
||||
|
||||
guiManager = new VRGuiManager(this); |
||||
mouseManager = new VRMouseManager(this); |
||||
dummyCam = new Camera(); |
||||
|
||||
processSettings(); |
||||
} |
||||
|
||||
/** |
||||
* Get the VR underlying hardware. |
||||
* @return the VR underlying hardware. |
||||
*/ |
||||
public VRAPI getVRHardware() { |
||||
return hardware; |
||||
} |
||||
|
||||
/** |
||||
* Get the VR dedicated input. |
||||
* @return the VR dedicated input. |
||||
*/ |
||||
public VRInputAPI getVRinput() { |
||||
if( hardware == null ){ |
||||
return null; |
||||
} |
||||
|
||||
return hardware.getVRinput(); |
||||
} |
||||
|
||||
/** |
||||
* Get the VR view manager. |
||||
* @return the VR view manager. |
||||
*/ |
||||
public VRViewManager getVRViewManager() { |
||||
return viewmanager; |
||||
} |
||||
|
||||
/** |
||||
* Get the GUI manager attached to this environment. |
||||
* @return the GUI manager attached to this environment. |
||||
*/ |
||||
public VRGuiManager getVRGUIManager(){ |
||||
return guiManager; |
||||
} |
||||
|
||||
/** |
||||
* Get the VR mouse manager attached to this environment. |
||||
* @return the VR mouse manager attached to this environment. |
||||
*/ |
||||
public VRMouseManager getVRMouseManager(){ |
||||
return mouseManager; |
||||
} |
||||
|
||||
/** |
||||
* Can be used to change seated experience during runtime. |
||||
* @param isSeated <code>true</code> if designed for sitting, <code>false</code> for standing/roomscale |
||||
* @see #isSeatedExperience() |
||||
*/ |
||||
public void setSeatedExperience(boolean isSeated) { |
||||
seated = isSeated; |
||||
if( hardware instanceof OpenVR ) { |
||||
if( hardware.getCompositor() == null ) { |
||||
return; |
||||
} |
||||
|
||||
if( seated ) { |
||||
((OpenVR)hardware).getCompositor().SetTrackingSpace.apply(JOpenVRLibrary.ETrackingUniverseOrigin.ETrackingUniverseOrigin_TrackingUniverseSeated); |
||||
} else { |
||||
((OpenVR)hardware).getCompositor().SetTrackingSpace.apply(JOpenVRLibrary.ETrackingUniverseOrigin.ETrackingUniverseOrigin_TrackingUniverseStanding); |
||||
} |
||||
} |
||||
} |
||||
|
||||
/** |
||||
* Check if the application is configured as a seated experience. |
||||
* @return <code>true</code> if the application is configured as a seated experience and <code>false</code> otherwise. |
||||
* @see #setSeatedExperience(boolean) |
||||
*/ |
||||
public boolean isSeatedExperience() { |
||||
return seated; |
||||
} |
||||
|
||||
/** |
||||
* Set the VR headset height from the ground. |
||||
* @param amount the VR headset height from the ground. |
||||
* @see #getVRHeightAdjustment() |
||||
*/ |
||||
public void setVRHeightAdjustment(float amount) { |
||||
if( viewmanager != null ){ |
||||
viewmanager.setHeightAdjustment(amount); |
||||
} |
||||
} |
||||
|
||||
/** |
||||
* Get the VR headset height from the ground. |
||||
* @return the VR headset height from the ground. |
||||
* @see #setVRHeightAdjustment(float) |
||||
*/ |
||||
public float getVRHeightAdjustment() { |
||||
if( viewmanager != null ){ |
||||
return viewmanager.getHeightAdjustment(); |
||||
} |
||||
return 0f; |
||||
} |
||||
|
||||
/** |
||||
* Get the scene observer. If no observer has been set, this method return the application {@link #getCamera() camera}. |
||||
* @return the scene observer. |
||||
* @see #setObserver(Spatial) |
||||
*/ |
||||
public Object getObserver() { |
||||
if( observer == null ) { |
||||
|
||||
if (application != null){ |
||||
return application.getCamera(); |
||||
} else { |
||||
throw new IllegalStateException("VR environment is not attached to any application."); |
||||
} |
||||
} |
||||
return observer; |
||||
} |
||||
|
||||
/** |
||||
* Set the scene observer. The VR headset will be linked to it. If no observer is set, the VR headset is linked to the the application {@link #getCamera() camera}. |
||||
* @param observer the scene observer. |
||||
*/ |
||||
public void setObserver(Spatial observer) { |
||||
this.observer = observer; |
||||
} |
||||
|
||||
/** |
||||
* Get the default Field Of View (FOV) value. |
||||
* @return the default Field Of View (FOV) value. |
||||
* @see #setDefaultFOV(float) |
||||
*/ |
||||
public float getDefaultFOV() { |
||||
return defaultFOV; |
||||
} |
||||
|
||||
/** |
||||
* Set the default Field Of View (FOV) value. |
||||
* @param defaultFOV the default Field Of View (FOV) value. |
||||
* @see #getDefaultFOV() |
||||
*/ |
||||
public void setDefaultFOV(float defaultFOV) { |
||||
this.defaultFOV = defaultFOV; |
||||
} |
||||
|
||||
/** |
||||
* Get the default aspect ratio. |
||||
* @return the default aspect ratio. |
||||
* @see #setDefaultAspect(float) |
||||
*/ |
||||
public float getDefaultAspect() { |
||||
return defaultAspect; |
||||
} |
||||
|
||||
/** |
||||
* Set the default aspect ratio. |
||||
* @param defaultAspect the default aspect ratio. |
||||
* @see #getDefaultAspect() |
||||
*/ |
||||
public void setDefaultAspect(float defaultAspect) { |
||||
this.defaultAspect = defaultAspect; |
||||
} |
||||
|
||||
/** |
||||
* Get the {@link AppSettings settings} attached to this environment. |
||||
* @return the {@link AppSettings settings} attached to this environment. |
||||
* @see #setSettings(AppSettings) |
||||
*/ |
||||
public AppSettings getSettings(){ |
||||
return settings; |
||||
} |
||||
|
||||
/** |
||||
* Set the {@link AppSettings settings} attached to this environment. |
||||
* @param settings the {@link AppSettings settings} attached to this environment. |
||||
* @see #getSettings() |
||||
*/ |
||||
public void setSettings(AppSettings settings){ |
||||
this.settings = settings; |
||||
processSettings(); |
||||
} |
||||
|
||||
/** |
||||
* Get if the system currently support VR. |
||||
* @return <code>true</code> if the system currently support VR and <code>false</Code> otherwise. |
||||
*/ |
||||
public boolean isVRSupported() { |
||||
return vrSupportedOS; |
||||
} |
||||
|
||||
/** |
||||
* Check if the VR mode is enabled. |
||||
* @return <code>true</code> if the VR mode is enabled and <code>false</code> otherwise. |
||||
*/ |
||||
public boolean isInVR() { |
||||
return (forceVR || vrSupportedOS && hardware != null && hardware.isInitialized() && isInitialized()); |
||||
} |
||||
|
||||
/** |
||||
* Check if the rendering is instanced (see <a href="https://en.wikipedia.org/wiki/Geometry_instancing">Geometry instancing</a>). |
||||
* @return <code>true</code> if the rendering is instanced and <code>false</code> otherwise. |
||||
*/ |
||||
public boolean isInstanceRendering() { |
||||
return instanceRendering; |
||||
} |
||||
|
||||
public boolean isSwapBuffers(){ |
||||
return disableSwapBuffers; |
||||
} |
||||
|
||||
/** |
||||
* Check if the application has a GUI overlay attached. |
||||
* @return <code>true</code> if the application has a GUI overlay attached and <code>false</code> otherwise. |
||||
*/ |
||||
public boolean hasTraditionalGUIOverlay() { |
||||
return !nogui; |
||||
} |
||||
|
||||
/** |
||||
* Check if the VR environment is initialized. A call to the {@link #initialize() initialize()} method should set this value to <code>true</code> |
||||
* @return <code>true</code> if the VR environment is initialized and <code>false</code> otherwise. |
||||
*/ |
||||
public boolean isInitialized(){ |
||||
return initialized; |
||||
} |
||||
|
||||
/** |
||||
* Is the VR compositor is active. |
||||
* @return <code>true</code> if the VR compositor is active and <code>false</code> otherwise. |
||||
*/ |
||||
public boolean compositorAllowed() { |
||||
return useCompositor && compositorOS; |
||||
} |
||||
|
||||
/** |
||||
* Reset headset pose if seating experience. |
||||
*/ |
||||
public void resetSeatedPose(){ |
||||
if( vrSupportedOS == false || isSeatedExperience() == false ){ |
||||
return; |
||||
} |
||||
getVRHardware().reset(); |
||||
} |
||||
|
||||
public AppState getAppState(){ |
||||
return app; |
||||
} |
||||
|
||||
public Application getApplication(){ |
||||
return application; |
||||
} |
||||
|
||||
/** |
||||
* Get the {@link Camera camera} used for rendering. |
||||
* If the VR mode is {@link #isInVR() active}, this method return a dummy camera, otherwise, |
||||
* this method return the camera of the attached application. |
||||
* @return the camera attached used for rendering. |
||||
*/ |
||||
public Camera getCamera() { |
||||
if( isInVR() && getVRViewManager() != null && getVRViewManager().getLeftCamera() != null ) { |
||||
return dummyCam; |
||||
} |
||||
|
||||
return application.getCamera(); |
||||
} |
||||
|
||||
public Camera getDummyCamera(){ |
||||
|
||||
if (dummyCam == null){ |
||||
|
||||
if (application != null){ |
||||
|
||||
if (application.getCamera() != null){ |
||||
dummyCam = application.getCamera().clone(); |
||||
} else { |
||||
return new Camera(); |
||||
} |
||||
} else { |
||||
throw new IllegalStateException("VR environment is not attached to any application."); |
||||
} |
||||
} |
||||
|
||||
return dummyCam; |
||||
} |
||||
|
||||
/** |
||||
* Attach the VR environment to the given app state and application. |
||||
* This method should be called within the {@link AppState#stateAttached(com.jme3.app.state.AppStateManager) stateAttached(com.jme3.app.state.AppStateManager)} method |
||||
* from the app state. |
||||
* @param appState the app state to attach. |
||||
* @param application the application to attach. |
||||
*/ |
||||
public void atttach(AppState appState, Application application){ |
||||
this.application = application; |
||||
this.app = appState; |
||||
|
||||
// Instanciate view manager
|
||||
if (vrBinding == VRConstants.SETTING_VRAPI_OPENVR_VALUE){ |
||||
viewmanager = new VRViewManagerOpenVR(this); |
||||
} else if (vrBinding == VRConstants.SETTING_VRAPI_OSVR_VALUE){ |
||||
viewmanager = new VRViewManagerOSVR(this); |
||||
} else { |
||||
logger.severe("Cannot instanciate view manager, unknown VRAPI type: "+vrBinding); |
||||
} |
||||
} |
||||
|
||||
/** |
||||
* Initialize this VR environment. This method enable the system bindings and configure all the VR system modules. |
||||
* A call to this method has to be made before any use of VR capabilities. |
||||
* @return <code>true</code> if the VR environment is successfully initialized and <code>false</code> otherwise. |
||||
*/ |
||||
public boolean initialize(){ |
||||
|
||||
logger.config("Initializing VR environment."); |
||||
|
||||
initialized = false; |
||||
|
||||
// we are going to use OpenVR now, not the Oculus Rift
|
||||
// OpenVR does support the Rift
|
||||
String OS = System.getProperty("os.name", "generic").toLowerCase(Locale.ENGLISH); |
||||
vrSupportedOS = !OS.contains("nux") && System.getProperty("sun.arch.data.model").equalsIgnoreCase("64"); //for the moment, linux/unix causes crashes, 64-bit only
|
||||
compositorOS = OS.contains("indows"); |
||||
|
||||
if( vrSupportedOS) { |
||||
if( vrBinding == VRConstants.SETTING_VRAPI_OSVR_VALUE ) { |
||||
hardware = new OSVR(this); |
||||
initialized = true; |
||||
logger.config("Creating OSVR wrapper [SUCCESS]"); |
||||
} else if( vrBinding == VRConstants.SETTING_VRAPI_OPENVR_VALUE ) { |
||||
hardware = new OpenVR(this); |
||||
initialized = true; |
||||
logger.config("Creating OpenVR wrapper [SUCCESS]"); |
||||
} else { |
||||
logger.config("Cannot create VR binding: "+vrBinding+" [FAILED]"); |
||||
logger.log(Level.SEVERE, "Cannot initialize VR environment [FAILED]"); |
||||
} |
||||
|
||||
if( hardware.initialize() ) { |
||||
initialized &= true; |
||||
logger.config("VR native wrapper initialized [SUCCESS]"); |
||||
} else { |
||||
initialized &= false; |
||||
logger.warning("VR native wrapper initialized [FAILED]"); |
||||
logger.log(Level.SEVERE, "Cannot initialize VR environment [FAILED]"); |
||||
} |
||||
} else { |
||||
logger.log(Level.SEVERE, "System does not support VR capabilities."); |
||||
logger.log(Level.SEVERE, "Cannot initialize VR environment [FAILED]"); |
||||
} |
||||
|
||||
return initialized; |
||||
} |
||||
|
||||
private void processSettings(){ |
||||
if (settings != null){ |
||||
|
||||
if (settings.get(VRConstants.SETTING_USE_COMPOSITOR) != null){ |
||||
useCompositor = settings.getBoolean(VRConstants.SETTING_USE_COMPOSITOR); |
||||
if( useCompositor == false ){ |
||||
disableSwapBuffers = false; |
||||
} |
||||
} |
||||
|
||||
if (settings.get(VRConstants.SETTING_ENABLE_MIRROR_WINDOW) != null){ |
||||
if( useCompositor == false ) { |
||||
disableSwapBuffers = false; |
||||
} else { |
||||
disableSwapBuffers = !settings.getBoolean(VRConstants.SETTING_ENABLE_MIRROR_WINDOW); |
||||
} |
||||
} |
||||
|
||||
if (settings.get(VRConstants.SETTING_GUI_OVERDRAW) != null){ |
||||
getVRGUIManager().setGuiOverdraw(settings.getBoolean(VRConstants.SETTING_GUI_OVERDRAW)); |
||||
} |
||||
|
||||
if (settings.get(VRConstants.SETTING_GUI_CURVED_SURFACE) != null){ |
||||
getVRGUIManager().setCurvedSurface(settings.getBoolean(VRConstants.SETTING_GUI_CURVED_SURFACE)); |
||||
} |
||||
|
||||
if (settings.get(VRConstants.SETTING_NO_GUI) != null){ |
||||
nogui = settings.getBoolean(VRConstants.SETTING_NO_GUI); |
||||
} |
||||
|
||||
if (settings.get(VRConstants.SETTING_VRAPI) != null){ |
||||
vrBinding = settings.getInteger(VRConstants.SETTING_VRAPI); |
||||
} |
||||
|
||||
if (settings.get(VRConstants.SETTING_SEATED_EXPERIENCE) != null){ |
||||
seated = settings.getBoolean(VRConstants.SETTING_SEATED_EXPERIENCE); |
||||
} |
||||
|
||||
if (settings.get(VRConstants.SETTING_INSTANCE_RENDERING) != null){ |
||||
instanceRendering = settings.getBoolean(VRConstants.SETTING_INSTANCE_RENDERING); |
||||
} |
||||
|
||||
if (settings.get(VRConstants.SETTING_DEFAULT_FOV) != null){ |
||||
defaultFOV = settings.getFloat(VRConstants.SETTING_DEFAULT_FOV); |
||||
} |
||||
|
||||
if (settings.get(VRConstants.SETTING_DEFAULT_ASPECT_RATIO) != null){ |
||||
defaultAspect = settings.getFloat(VRConstants.SETTING_DEFAULT_ASPECT_RATIO); |
||||
} |
||||
|
||||
if (settings.get(VRConstants.SETTING_FLIP_EYES) != null){ |
||||
if( getVRHardware() != null ){ |
||||
getVRHardware().setFlipEyes(settings.getBoolean(VRConstants.SETTING_FLIP_EYES)); |
||||
} |
||||
} |
||||
} |
||||
} |
||||
} |
@ -1,4 +1,4 @@ |
||||
package osvrclientkit; |
||||
package com.jme3.system.osvr.osvrclientkit; |
||||
import com.sun.jna.Callback; |
||||
import com.sun.jna.Library; |
||||
import com.sun.jna.Native; |
@ -1,4 +1,4 @@ |
||||
package osvrclientreporttypes; |
||||
package com.jme3.system.osvr.osvrclientreporttypes; |
||||
import com.sun.jna.Pointer; |
||||
import com.sun.jna.Structure; |
||||
import java.util.Arrays; |
@ -1,4 +1,4 @@ |
||||
package osvrclientreporttypes; |
||||
package com.jme3.system.osvr.osvrclientreporttypes; |
||||
import com.sun.jna.Pointer; |
||||
import com.sun.jna.Structure; |
||||
import java.util.Arrays; |
@ -1,4 +1,4 @@ |
||||
package osvrclientreporttypes; |
||||
package com.jme3.system.osvr.osvrclientreporttypes; |
||||
import com.sun.jna.Pointer; |
||||
import com.sun.jna.Structure; |
||||
import java.util.Arrays; |
@ -1,4 +1,4 @@ |
||||
package osvrclientreporttypes; |
||||
package com.jme3.system.osvr.osvrclientreporttypes; |
||||
import com.sun.jna.Pointer; |
||||
import com.sun.jna.Structure; |
||||
import java.util.Arrays; |
@ -1,4 +1,4 @@ |
||||
package osvrclientreporttypes; |
||||
package com.jme3.system.osvr.osvrclientreporttypes; |
||||
import com.sun.jna.Pointer; |
||||
import com.sun.jna.Structure; |
||||
import java.util.Arrays; |
@ -1,4 +1,4 @@ |
||||
package osvrclientreporttypes; |
||||
package com.jme3.system.osvr.osvrclientreporttypes; |
||||
import com.sun.jna.Pointer; |
||||
import com.sun.jna.Structure; |
||||
import java.util.Arrays; |
@ -1,4 +1,4 @@ |
||||
package osvrclientreporttypes; |
||||
package com.jme3.system.osvr.osvrclientreporttypes; |
||||
import com.sun.jna.Pointer; |
||||
import com.sun.jna.Structure; |
||||
import java.util.Arrays; |
@ -1,4 +1,4 @@ |
||||
package osvrclientreporttypes; |
||||
package com.jme3.system.osvr.osvrclientreporttypes; |
||||
import com.sun.jna.Pointer; |
||||
import com.sun.jna.Structure; |
||||
import java.util.Arrays; |
@ -1,4 +1,4 @@ |
||||
package osvrclientreporttypes; |
||||
package com.jme3.system.osvr.osvrclientreporttypes; |
||||
import com.sun.jna.Pointer; |
||||
import com.sun.jna.Structure; |
||||
import java.util.Arrays; |
@ -1,4 +1,4 @@ |
||||
package osvrclientreporttypes; |
||||
package com.jme3.system.osvr.osvrclientreporttypes; |
||||
import com.sun.jna.Pointer; |
||||
import com.sun.jna.Structure; |
||||
import java.util.Arrays; |
@ -1,4 +1,4 @@ |
||||
package osvrclientreporttypes; |
||||
package com.jme3.system.osvr.osvrclientreporttypes; |
||||
import com.sun.jna.Pointer; |
||||
import com.sun.jna.Structure; |
||||
import java.util.Arrays; |
@ -1,4 +1,4 @@ |
||||
package osvrclientreporttypes; |
||||
package com.jme3.system.osvr.osvrclientreporttypes; |
||||
import com.sun.jna.Pointer; |
||||
import com.sun.jna.Structure; |
||||
import java.util.Arrays; |
@ -1,4 +1,4 @@ |
||||
package osvrclientreporttypes; |
||||
package com.jme3.system.osvr.osvrclientreporttypes; |
||||
import com.sun.jna.Pointer; |
||||
import com.sun.jna.Structure; |
||||
import java.util.Arrays; |
@ -1,4 +1,4 @@ |
||||
package osvrclientreporttypes; |
||||
package com.jme3.system.osvr.osvrclientreporttypes; |
||||
import com.sun.jna.Pointer; |
||||
import com.sun.jna.Structure; |
||||
import java.util.Arrays; |
@ -1,4 +1,4 @@ |
||||
package osvrclientreporttypes; |
||||
package com.jme3.system.osvr.osvrclientreporttypes; |
||||
import com.sun.jna.Pointer; |
||||
import com.sun.jna.Structure; |
||||
import java.util.Arrays; |
@ -1,4 +1,4 @@ |
||||
package osvrclientreporttypes; |
||||
package com.jme3.system.osvr.osvrclientreporttypes; |
||||
import com.sun.jna.Pointer; |
||||
import com.sun.jna.Structure; |
||||
import java.util.Arrays; |
@ -1,4 +1,4 @@ |
||||
package osvrclientreporttypes; |
||||
package com.jme3.system.osvr.osvrclientreporttypes; |
||||
import com.sun.jna.Pointer; |
||||
import com.sun.jna.Structure; |
||||
import java.util.Arrays; |
@ -1,4 +1,4 @@ |
||||
package osvrclientreporttypes; |
||||
package com.jme3.system.osvr.osvrclientreporttypes; |
||||
import com.sun.jna.Pointer; |
||||
import com.sun.jna.Structure; |
||||
import java.util.Arrays; |
@ -1,4 +1,4 @@ |
||||
package osvrclientreporttypes; |
||||
package com.jme3.system.osvr.osvrclientreporttypes; |
||||
import com.sun.jna.Pointer; |
||||
import com.sun.jna.Structure; |
||||
import java.util.Arrays; |
@ -1,4 +1,4 @@ |
||||
package osvrclientreporttypes; |
||||
package com.jme3.system.osvr.osvrclientreporttypes; |
||||
import com.sun.jna.Pointer; |
||||
import com.sun.jna.Structure; |
||||
import java.util.Arrays; |
@ -1,4 +1,4 @@ |
||||
package osvrclientreporttypes; |
||||
package com.jme3.system.osvr.osvrclientreporttypes; |
||||
import com.sun.jna.Pointer; |
||||
import com.sun.jna.Structure; |
||||
import java.util.Arrays; |
@ -1,4 +1,4 @@ |
||||
package osvrmatrixconventions; |
||||
package com.jme3.system.osvr.osvrclientreporttypes; |
||||
import com.sun.jna.Pointer; |
||||
import com.sun.jna.Structure; |
||||
import java.util.Arrays; |
@ -1,4 +1,4 @@ |
||||
package osvrclientreporttypes; |
||||
package com.jme3.system.osvr.osvrclientreporttypes; |
||||
import com.sun.jna.Pointer; |
||||
import com.sun.jna.Structure; |
||||
import java.util.Arrays; |
@ -1,4 +1,4 @@ |
||||
package osvrclientreporttypes; |
||||
package com.jme3.system.osvr.osvrclientreporttypes; |
||||
import com.sun.jna.Pointer; |
||||
import com.sun.jna.Structure; |
||||
import java.util.Arrays; |
@ -1,4 +1,4 @@ |
||||
package osvrclientreporttypes; |
||||
package com.jme3.system.osvr.osvrclientreporttypes; |
||||
import com.sun.jna.Pointer; |
||||
import com.sun.jna.Structure; |
||||
import java.util.Arrays; |
@ -1,4 +1,4 @@ |
||||
package osvrclientreporttypes; |
||||
package com.jme3.system.osvr.osvrclientreporttypes; |
||||
import com.sun.jna.Pointer; |
||||
import com.sun.jna.Structure; |
||||
import java.util.Arrays; |
@ -1,4 +1,4 @@ |
||||
package osvrclientreporttypes; |
||||
package com.jme3.system.osvr.osvrclientreporttypes; |
||||
import com.sun.jna.Library; |
||||
import com.sun.jna.Native; |
||||
import com.sun.jna.NativeLibrary; |
@ -1,10 +1,10 @@ |
||||
package osvrinterface; |
||||
package com.jme3.system.osvr.osvrinterface; |
||||
import com.jme3.system.osvr.osvrclientkit.OsvrClientKitLibrary.OSVR_ClientInterface; |
||||
import com.jme3.system.osvr.osvrclientreporttypes.OSVR_Pose3; |
||||
import com.jme3.system.osvr.osvrtimevalue.OSVR_TimeValue; |
||||
import com.sun.jna.Library; |
||||
import com.sun.jna.Native; |
||||
import com.sun.jna.NativeLibrary; |
||||
import osvrclientkit.OsvrClientKitLibrary.OSVR_ClientInterface; |
||||
import osvrclientreporttypes.OSVR_Pose3; |
||||
import osvrtimevalue.OSVR_TimeValue; |
||||
/** |
||||
* JNA Wrapper for library <b>osvrInterface</b><br> |
||||
* This file was autogenerated by <a href="http://jnaerator.googlecode.com/">JNAerator</a>,<br> |
@ -1,4 +1,4 @@ |
||||
package osvrmatrixconventions; |
||||
package com.jme3.system.osvr.osvrmatrixconventions; |
||||
import com.sun.jna.Pointer; |
||||
import com.sun.jna.Structure; |
||||
import java.util.Arrays; |
@ -1,4 +1,4 @@ |
||||
package osvrclientreporttypes; |
||||
package com.jme3.system.osvr.osvrmatrixconventions; |
||||
import com.sun.jna.Pointer; |
||||
import com.sun.jna.Structure; |
||||
import java.util.Arrays; |
@ -1,4 +1,4 @@ |
||||
package osvrmatrixconventions; |
||||
package com.jme3.system.osvr.osvrmatrixconventions; |
||||
import com.sun.jna.Pointer; |
||||
import com.sun.jna.Structure; |
||||
import java.util.Arrays; |
@ -1,4 +1,4 @@ |
||||
package osvrmatrixconventions; |
||||
package com.jme3.system.osvr.osvrmatrixconventions; |
||||
import com.sun.jna.Library; |
||||
import com.sun.jna.Native; |
||||
import com.sun.jna.NativeLibrary; |
@ -1,4 +1,4 @@ |
||||
package osvrrendermanager; |
||||
package com.jme3.system.osvr.osvrrendermanager; |
||||
import com.sun.jna.Pointer; |
||||
import com.sun.jna.Structure; |
||||
import java.util.Arrays; |
@ -1,4 +1,4 @@ |
||||
package osvrrendermanager; |
||||
package com.jme3.system.osvr.osvrrendermanager; |
||||
import com.sun.jna.Pointer; |
||||
import com.sun.jna.Structure; |
||||
import java.util.Arrays; |
@ -1,4 +1,4 @@ |
||||
package osvrrendermanageropengl; |
||||
package com.jme3.system.osvr.osvrrendermanager; |
||||
import com.sun.jna.Pointer; |
||||
import com.sun.jna.Structure; |
||||
import java.util.Arrays; |
@ -1,4 +1,4 @@ |
||||
package osvrrendermanager; |
||||
package com.jme3.system.osvr.osvrrendermanager; |
||||
import com.sun.jna.Pointer; |
||||
import com.sun.jna.Structure; |
||||
import java.util.Arrays; |
@ -1,4 +1,4 @@ |
||||
package osvrrendermanageropengl; |
||||
package com.jme3.system.osvr.osvrrendermanageropengl; |
||||
import com.sun.jna.Pointer; |
||||
import com.sun.jna.Structure; |
||||
import java.util.Arrays; |
@ -1,4 +1,4 @@ |
||||
package osvrrendermanageropengl; |
||||
package com.jme3.system.osvr.osvrrendermanageropengl; |
||||
import com.ochafik.lang.jnaerator.runtime.NativeSize; |
||||
import com.sun.jna.Callback; |
||||
import com.sun.jna.Pointer; |
@ -1,4 +1,4 @@ |
||||
package osvrrendermanageropengl; |
||||
package com.jme3.system.osvr.osvrrendermanageropengl; |
||||
import com.sun.jna.Pointer; |
||||
import com.sun.jna.Structure; |
||||
import java.util.Arrays; |
@ -1,4 +1,4 @@ |
||||
package osvrrendermanageropengl; |
||||
package com.jme3.system.osvr.osvrrendermanageropengl; |
||||
import com.sun.jna.Pointer; |
||||
import com.sun.jna.Structure; |
||||
import java.util.Arrays; |
@ -1,4 +1,4 @@ |
||||
package osvrrendermanageropengl; |
||||
package com.jme3.system.osvr.osvrrendermanageropengl; |
||||
import com.sun.jna.Pointer; |
||||
import com.sun.jna.Structure; |
||||
import java.util.Arrays; |
@ -1,4 +1,4 @@ |
||||
package osvrrendermanageropengl; |
||||
package com.jme3.system.osvr.osvrrendermanageropengl; |
||||
import com.sun.jna.Pointer; |
||||
import com.sun.jna.Structure; |
||||
import java.util.Arrays; |
@ -1,9 +1,9 @@ |
||||
package osvrrendermanageropengl; |
||||
package com.jme3.system.osvr.osvrrendermanageropengl; |
||||
import com.jme3.system.osvr.osvrmatrixconventions.OSVR_Pose3; |
||||
import com.sun.jna.Pointer; |
||||
import com.sun.jna.Structure; |
||||
import java.util.Arrays; |
||||
import java.util.List; |
||||
import osvrmatrixconventions.OSVR_Pose3; |
||||
/** |
||||
* This file was autogenerated by <a href="http://jnaerator.googlecode.com/">JNAerator</a>,<br> |
||||
* a tool written by <a href="http://ochafik.com/">Olivier Chafik</a> that <a href="http://code.google.com/p/jnaerator/wiki/CreditsAndLicense">uses a few opensource projects.</a>.<br> |
@ -1,4 +1,4 @@ |
||||
package osvrrendermanager; |
||||
package com.jme3.system.osvr.osvrrendermanageropengl; |
||||
import com.sun.jna.Pointer; |
||||
import com.sun.jna.Structure; |
||||
import java.util.Arrays; |
@ -1,4 +1,4 @@ |
||||
package osvrrendermanageropengl; |
||||
package com.jme3.system.osvr.osvrrendermanageropengl; |
||||
import com.sun.jna.Pointer; |
||||
import com.sun.jna.Structure; |
||||
import java.util.Arrays; |
@ -1,4 +1,4 @@ |
||||
package osvrtimevalue; |
||||
package com.jme3.system.osvr.osvrtimevalue; |
||||
import com.sun.jna.Pointer; |
||||
import com.sun.jna.Structure; |
||||
import java.util.Arrays; |
@ -1,4 +1,4 @@ |
||||
package osvrtimevalue; |
||||
package com.jme3.system.osvr.osvrtimevalue; |
||||
import com.sun.jna.Library; |
||||
import com.sun.jna.Native; |
||||
import com.sun.jna.NativeLibrary; |
@ -0,0 +1,223 @@ |
||||
package com.jme3.util; |
||||
|
||||
import com.jme3.app.VREnvironment; |
||||
import com.jme3.post.CartoonSSAO; |
||||
import com.jme3.post.Filter; |
||||
import com.jme3.post.FilterPostProcessor; |
||||
import com.jme3.post.FilterUtil; |
||||
import com.jme3.post.SceneProcessor; |
||||
import com.jme3.post.filters.FogFilter; |
||||
import com.jme3.post.filters.TranslucentBucketFilter; |
||||
import com.jme3.post.ssao.SSAOFilter; |
||||
import com.jme3.renderer.Camera; |
||||
import com.jme3.renderer.ViewPort; |
||||
import com.jme3.shadow.DirectionalLightShadowFilter; |
||||
import com.jme3.shadow.VRDirectionalLightShadowRenderer; |
||||
import com.jme3.texture.Texture2D; |
||||
|
||||
/** |
||||
* A VR view manager. This class holds methods that enable to submit 3D views to the VR compositor. |
||||
* System dependent classes should extends from this one. |
||||
* @author Julien Seinturier - (c) 2016 - JOrigin project - <a href="http://www.jorigin.org">http:/www.jorigin.org</a> |
||||
*/ |
||||
public abstract class AbstractVRViewManager implements VRViewManager { |
||||
|
||||
|
||||
//private static final Logger logger = Logger.getLogger(AbstractVRViewManager.class.getName());
|
||||
|
||||
protected VREnvironment environment = null; |
||||
|
||||
protected Camera leftCamera; |
||||
protected ViewPort leftViewport; |
||||
protected FilterPostProcessor leftPostProcessor; |
||||
protected Texture2D leftEyeTexture; |
||||
protected Texture2D leftEyeDepth; |
||||
|
||||
protected Camera rightCamera; |
||||
protected ViewPort rightViewport; |
||||
protected FilterPostProcessor rightPostProcessor; |
||||
protected Texture2D rightEyeTexture; |
||||
protected Texture2D rightEyeDepth; |
||||
|
||||
private float resMult = 1f; |
||||
|
||||
private float heightAdjustment; |
||||
|
||||
@Override |
||||
public Camera getLeftCamera() { |
||||
return leftCamera; |
||||
} |
||||
|
||||
@Override |
||||
public Camera getRightCamera() { |
||||
return rightCamera; |
||||
} |
||||
|
||||
@Override |
||||
public ViewPort getLeftViewport() { |
||||
return leftViewport; |
||||
} |
||||
|
||||
@Override |
||||
public ViewPort getRightViewport() { |
||||
return rightViewport; |
||||
} |
||||
|
||||
@Override |
||||
public Texture2D getLeftTexture(){ |
||||
return leftEyeTexture; |
||||
} |
||||
|
||||
@Override |
||||
public Texture2D getRightTexture(){ |
||||
return rightEyeTexture; |
||||
} |
||||
|
||||
@Override |
||||
public Texture2D getLeftDepth(){ |
||||
return leftEyeDepth; |
||||
} |
||||
|
||||
@Override |
||||
public Texture2D getRightDepth(){ |
||||
return rightEyeDepth; |
||||
} |
||||
|
||||
@Override |
||||
public FilterPostProcessor getLeftPostProcessor(){ |
||||
return leftPostProcessor; |
||||
} |
||||
|
||||
@Override |
||||
public FilterPostProcessor getRightPostProcessor(){ |
||||
return rightPostProcessor; |
||||
} |
||||
|
||||
@Override |
||||
public float getResolutionMuliplier() { |
||||
return resMult; |
||||
} |
||||
|
||||
@Override |
||||
public void setResolutionMultiplier(float resMult) { |
||||
this.resMult = resMult; |
||||
} |
||||
|
||||
@Override |
||||
public float getHeightAdjustment() { |
||||
return heightAdjustment; |
||||
} |
||||
|
||||
@Override |
||||
public void setHeightAdjustment(float amount) { |
||||
heightAdjustment = amount; |
||||
} |
||||
|
||||
@Override |
||||
public VREnvironment getVREnvironment(){ |
||||
return environment; |
||||
} |
||||
|
||||
/** |
||||
* Handles moving filters from the main view to each eye |
||||
*/ |
||||
public void moveScreenProcessingToEyes() { |
||||
|
||||
if (environment != null){ |
||||
if( getRightViewport() == null ){ |
||||
return; |
||||
} |
||||
|
||||
if (environment.getApplication() != null){ |
||||
syncScreenProcessing(environment.getApplication().getViewPort()); |
||||
environment.getApplication().getViewPort().clearProcessors(); |
||||
} else { |
||||
throw new IllegalStateException("The VR environment is not attached to any application."); |
||||
} |
||||
|
||||
|
||||
} else { |
||||
throw new IllegalStateException("This VR view manager is not attached to any VR environment."); |
||||
} |
||||
|
||||
|
||||
} |
||||
|
||||
/** |
||||
* Sets the two views to use the list of {@link SceneProcessor processors}. |
||||
* @param sourceViewport the {@link ViewPort viewport} that contains the processors to use. |
||||
*/ |
||||
public void syncScreenProcessing(ViewPort sourceViewport) { |
||||
|
||||
if (environment != null){ |
||||
if( getRightViewport() == null ){ |
||||
return; |
||||
} |
||||
|
||||
if (environment.getApplication() != null){ |
||||
// setup post processing filters
|
||||
if( getRightPostProcessor() == null ) { |
||||
rightPostProcessor = new FilterPostProcessor(environment.getApplication().getAssetManager()); |
||||
leftPostProcessor = new FilterPostProcessor(environment.getApplication().getAssetManager()); |
||||
} |
||||
// clear out all filters & processors, to start from scratch
|
||||
getRightPostProcessor().removeAllFilters(); |
||||
getLeftPostProcessor().removeAllFilters(); |
||||
getLeftViewport().clearProcessors(); |
||||
getRightViewport().clearProcessors(); |
||||
// if we have no processors to sync, don't add the FilterPostProcessor
|
||||
if( sourceViewport.getProcessors().isEmpty() ) return; |
||||
// add post processors we just made, which are empty
|
||||
getLeftViewport().addProcessor(getLeftPostProcessor()); |
||||
getRightViewport().addProcessor(getRightPostProcessor()); |
||||
// go through all of the filters in the processors list
|
||||
// add them to the left viewport processor & clone them to the right
|
||||
for(SceneProcessor sceneProcessor : sourceViewport.getProcessors()) { |
||||
if (sceneProcessor instanceof FilterPostProcessor) { |
||||
for(Filter f : ((FilterPostProcessor)sceneProcessor).getFilterList() ) { |
||||
if( f instanceof TranslucentBucketFilter ) { |
||||
// just remove this filter, we will add it at the end manually
|
||||
((FilterPostProcessor)sceneProcessor).removeFilter(f); |
||||
} else { |
||||
getLeftPostProcessor().addFilter(f); |
||||
// clone to the right
|
||||
Filter f2; |
||||
if(f instanceof FogFilter){ |
||||
f2 = FilterUtil.cloneFogFilter((FogFilter)f); |
||||
} else if (f instanceof CartoonSSAO ) { |
||||
f2 = new CartoonSSAO((CartoonSSAO)f); |
||||
} else if (f instanceof SSAOFilter){ |
||||
f2 = FilterUtil.cloneSSAOFilter((SSAOFilter)f); |
||||
} else if (f instanceof DirectionalLightShadowFilter){ |
||||
f2 = FilterUtil.cloneDirectionalLightShadowFilter(environment.getApplication().getAssetManager(), (DirectionalLightShadowFilter)f); |
||||
} else { |
||||
f2 = f; // dof, bloom, lightscattering etc.
|
||||
} |
||||
getRightPostProcessor().addFilter(f2); |
||||
} |
||||
} |
||||
} else if (sceneProcessor instanceof VRDirectionalLightShadowRenderer) { |
||||
// shadow processing
|
||||
// TODO: make right shadow processor use same left shadow maps for performance
|
||||
VRDirectionalLightShadowRenderer dlsr = (VRDirectionalLightShadowRenderer) sceneProcessor; |
||||
VRDirectionalLightShadowRenderer dlsrRight = dlsr.clone(); |
||||
dlsrRight.setLight(dlsr.getLight()); |
||||
getRightViewport().getProcessors().add(0, dlsrRight); |
||||
getLeftViewport().getProcessors().add(0, sceneProcessor); |
||||
} |
||||
} |
||||
// make sure each has a translucent filter renderer
|
||||
getLeftPostProcessor().addFilter(new TranslucentBucketFilter()); |
||||
getRightPostProcessor().addFilter(new TranslucentBucketFilter()); |
||||
} else { |
||||
throw new IllegalStateException("The VR environment is not attached to any application."); |
||||
} |
||||
|
||||
|
||||
} else { |
||||
throw new IllegalStateException("This VR view manager is not attached to any VR environment."); |
||||
} |
||||
|
||||
|
||||
} |
||||
} |
@ -0,0 +1,15 @@ |
||||
package com.jme3.util; |
||||
|
||||
/** |
||||
* A enumeration that describes the GUI display positioning modes. |
||||
* @author Julien Seinturier - (c) 2016 - JOrigin project - <a href="http://www.jorigin.org">http:/www.jorigin.org</a> |
||||
* |
||||
*/ |
||||
public enum VRGUIPositioningMode { |
||||
MANUAL, |
||||
AUTO_CAM_ALL, |
||||
AUTO_CAM_ALL_SKIP_PITCH, |
||||
AUTO_OBSERVER_POS_CAM_ROTATION, |
||||
AUTO_OBSERVER_ALL, |
||||
AUTO_OBSERVER_ALL_CAMHEIGHT |
||||
} |
@ -0,0 +1,474 @@ |
||||
/* |
||||
* To change this template, choose Tools | Templates |
||||
* and open the template in the editor. |
||||
*/ |
||||
package com.jme3.util; |
||||
|
||||
import com.jme3.app.VREnvironment; |
||||
import com.jme3.material.Material; |
||||
import com.jme3.material.RenderState.BlendMode; |
||||
import com.jme3.math.ColorRGBA; |
||||
import com.jme3.math.Matrix3f; |
||||
import com.jme3.math.Quaternion; |
||||
import com.jme3.math.Vector2f; |
||||
import com.jme3.math.Vector3f; |
||||
import com.jme3.renderer.Camera; |
||||
import com.jme3.renderer.ViewPort; |
||||
import com.jme3.renderer.queue.RenderQueue.Bucket; |
||||
import com.jme3.scene.Spatial; |
||||
import com.jme3.scene.CenterQuad; |
||||
import com.jme3.scene.Geometry; |
||||
import com.jme3.scene.Node; |
||||
import com.jme3.system.AppSettings; |
||||
import com.jme3.texture.FrameBuffer; |
||||
import com.jme3.texture.Image.Format; |
||||
import com.jme3.texture.Texture; |
||||
import com.jme3.texture.Texture2D; |
||||
import java.awt.GraphicsEnvironment; |
||||
import java.util.Iterator; |
||||
|
||||
/** |
||||
* A class dedicated to the management and the display of a Graphical User Interface (GUI) within a VR environment. |
||||
* @author reden - phr00t - https://github.com/phr00t
|
||||
* @author Julien Seinturier - (c) 2016 - JOrigin project - <a href="http://www.jorigin.org">http:/www.jorigin.org</a> |
||||
* |
||||
*/ |
||||
public class VRGuiManager { |
||||
|
||||
private Camera camLeft, camRight; |
||||
private float guiDistance = 1.5f; |
||||
private float guiScale = 1f; |
||||
private float guiPositioningElastic; |
||||
|
||||
private VRGUIPositioningMode posMode = VRGUIPositioningMode.AUTO_CAM_ALL; |
||||
|
||||
private final Matrix3f orient = new Matrix3f(); |
||||
private Vector2f screenSize; |
||||
protected boolean wantsReposition; |
||||
|
||||
private Vector2f ratio; |
||||
|
||||
private final Vector3f EoldPos = new Vector3f(); |
||||
|
||||
private final Quaternion EoldDir = new Quaternion(); |
||||
|
||||
private final Vector3f look = new Vector3f(); |
||||
private final Vector3f left = new Vector3f(); |
||||
private final Vector3f temppos = new Vector3f(); |
||||
private final Vector3f up = new Vector3f(); |
||||
|
||||
private boolean useCurvedSurface = false; |
||||
private boolean overdraw = false; |
||||
private Geometry guiQuad; |
||||
private Node guiQuadNode; |
||||
private ViewPort offView; |
||||
private Texture2D guiTexture; |
||||
|
||||
private final Quaternion tempq = new Quaternion(); |
||||
|
||||
private VREnvironment environment = null; |
||||
|
||||
/** |
||||
* Create a new GUI manager attached to the given app state. |
||||
* @param environment the VR environment to which this manager is attached to. |
||||
*/ |
||||
public VRGuiManager(VREnvironment environment){ |
||||
this.environment = environment; |
||||
} |
||||
|
||||
/** |
||||
* |
||||
* Makes auto GUI positioning happen not immediately, but like an |
||||
* elastic connected to the headset. Setting to 0 disables (default) |
||||
* Higher settings make it track the headset quicker. |
||||
* |
||||
* @param elastic amount of elasticity |
||||
*/ |
||||
public void setPositioningElasticity(float elastic) { |
||||
guiPositioningElastic = elastic; |
||||
} |
||||
|
||||
public float getPositioningElasticity() { |
||||
return guiPositioningElastic; |
||||
} |
||||
|
||||
/** |
||||
* Get the GUI {@link VRGUIPositioningMode positioning mode}. |
||||
* @return the GUI {@link VRGUIPositioningMode positioning mode}. |
||||
* @see #setPositioningMode(VRGUIPositioningMode) |
||||
*/ |
||||
public VRGUIPositioningMode getPositioningMode() { |
||||
return posMode; |
||||
} |
||||
|
||||
/** |
||||
* Set the GUI {@link VRGUIPositioningMode positioning mode}. |
||||
* @param mode the GUI {@link VRGUIPositioningMode positioning mode}. |
||||
* @see #getPositioningMode() |
||||
*/ |
||||
public void setPositioningMode(VRGUIPositioningMode mode) { |
||||
posMode = mode; |
||||
} |
||||
|
||||
/** |
||||
* Get the GUI canvas size. This method return the size in pixels of the GUI available area within the VR view. |
||||
* @return the GUI canvas size. This method return the size in pixels of the GUI available area within the VR view. |
||||
*/ |
||||
public Vector2f getCanvasSize() { |
||||
|
||||
if (environment != null){ |
||||
|
||||
if (environment.getApplication() != null){ |
||||
if( screenSize == null ) { |
||||
if( environment.isInVR() && environment.getVRHardware() != null ) { |
||||
screenSize = new Vector2f(); |
||||
environment.getVRHardware().getRenderSize(screenSize); |
||||
screenSize.multLocal(environment.getVRViewManager().getResolutionMuliplier()); |
||||
} else { |
||||
AppSettings as = environment.getApplication().getContext().getSettings(); |
||||
screenSize = new Vector2f(as.getWidth(), as.getHeight()); |
||||
} |
||||
} |
||||
return screenSize; |
||||
} else { |
||||
throw new IllegalStateException("VR GUI manager underlying environment is not attached to any application."); |
||||
} |
||||
} else { |
||||
throw new IllegalStateException("VR GUI manager is not attached to any environment."); |
||||
} |
||||
|
||||
} |
||||
|
||||
/** |
||||
* Get the ratio between the {@link #getCanvasSize() GUI canvas size} and the application main windows (if available) or the screen size. |
||||
* @return the ratio between the {@link #getCanvasSize() GUI canvas size} and the application main windows (if available). |
||||
* @see #getCanvasSize() |
||||
*/ |
||||
public Vector2f getCanvasToWindowRatio() { |
||||
|
||||
if (environment != null){ |
||||
|
||||
if (environment.getApplication() != null){ |
||||
if( ratio == null ) { |
||||
ratio = new Vector2f(); |
||||
Vector2f canvas = getCanvasSize(); |
||||
int width = Integer.min(GraphicsEnvironment.getLocalGraphicsEnvironment().getDefaultScreenDevice().getDisplayMode().getWidth(), |
||||
environment.getApplication().getContext().getSettings().getWidth()); |
||||
int height = Integer.min(GraphicsEnvironment.getLocalGraphicsEnvironment().getDefaultScreenDevice().getDisplayMode().getHeight(), |
||||
environment.getApplication().getContext().getSettings().getHeight()); |
||||
ratio.x = Float.max(1f, canvas.x / width); |
||||
ratio.y = Float.max(1f, canvas.y / height); |
||||
} |
||||
return ratio; |
||||
|
||||
} else { |
||||
throw new IllegalStateException("VR GUI manager underlying environment is not attached to any application."); |
||||
} |
||||
} else { |
||||
throw new IllegalStateException("VR GUI manager is not attached to any environment."); |
||||
} |
||||
} |
||||
|
||||
/** |
||||
* Inform this manager that it has to position the GUI. |
||||
*/ |
||||
public void positionGui() { |
||||
wantsReposition = true; |
||||
} |
||||
|
||||
/** |
||||
* Position the GUI to the given location. |
||||
* @param pos the position of the GUI. |
||||
* @param dir the rotation of the GUI. |
||||
* @param tpf the time per frame. |
||||
*/ |
||||
private void positionTo(Vector3f pos, Quaternion dir, float tpf) { |
||||
|
||||
if (environment != null){ |
||||
Vector3f guiPos = guiQuadNode.getLocalTranslation(); |
||||
guiPos.set(0f, 0f, guiDistance); |
||||
dir.mult(guiPos, guiPos); |
||||
guiPos.x += pos.x; |
||||
guiPos.y += pos.y + environment.getVRHeightAdjustment(); |
||||
guiPos.z += pos.z; |
||||
if( guiPositioningElastic > 0f && posMode != VRGUIPositioningMode.MANUAL ) { |
||||
// mix pos & dir with current pos & dir
|
||||
guiPos.interpolateLocal(EoldPos, guiPos, Float.min(1f, tpf * guiPositioningElastic)); |
||||
EoldPos.set(guiPos); |
||||
} |
||||
} else { |
||||
throw new IllegalStateException("VR GUI manager is not attached to any environment."); |
||||
} |
||||
} |
||||
|
||||
/** |
||||
* Update the GUI geometric state. This method should be called after GUI modification. |
||||
*/ |
||||
protected void updateGuiQuadGeometricState() { |
||||
guiQuadNode.updateGeometricState(); |
||||
} |
||||
|
||||
/** |
||||
* Position the GUI without delay. |
||||
* @param tpf the time per frame. |
||||
*/ |
||||
protected void positionGuiNow(float tpf) { |
||||
|
||||
if (environment != null){ |
||||
wantsReposition = false; |
||||
if( environment.isInVR() == false ){ |
||||
return; |
||||
} |
||||
|
||||
guiQuadNode.setLocalScale(guiDistance * guiScale * 4f, 4f * guiDistance * guiScale, 1f); |
||||
|
||||
switch( posMode ) { |
||||
case MANUAL: |
||||
case AUTO_CAM_ALL_SKIP_PITCH: |
||||
case AUTO_CAM_ALL: |
||||
if( camLeft != null && camRight != null ) { |
||||
// get middle point
|
||||
temppos.set(camLeft.getLocation()).interpolateLocal(camRight.getLocation(), 0.5f); |
||||
positionTo(temppos, camLeft.getRotation(), tpf); |
||||
} |
||||
rotateScreenTo(camLeft.getRotation(), tpf); |
||||
|
||||
break; |
||||
case AUTO_OBSERVER_POS_CAM_ROTATION: |
||||
Object obs = environment.getObserver(); |
||||
if( obs != null ) { |
||||
if( obs instanceof Camera ) { |
||||
positionTo(((Camera)obs).getLocation(), camLeft.getRotation(), tpf); |
||||
} else { |
||||
positionTo(((Spatial)obs).getWorldTranslation(), camLeft.getRotation(), tpf); |
||||
} |
||||
} |
||||
rotateScreenTo(camLeft.getRotation(), tpf); |
||||
|
||||
break; |
||||
case AUTO_OBSERVER_ALL: |
||||
case AUTO_OBSERVER_ALL_CAMHEIGHT: |
||||
obs = environment.getObserver(); |
||||
if( obs != null ) { |
||||
Quaternion q; |
||||
if( obs instanceof Camera ) { |
||||
q = ((Camera)obs).getRotation(); |
||||
temppos.set(((Camera)obs).getLocation()); |
||||
} else { |
||||
q = ((Spatial)obs).getWorldRotation(); |
||||
temppos.set(((Spatial)obs).getWorldTranslation()); |
||||
} |
||||
if( posMode == VRGUIPositioningMode.AUTO_OBSERVER_ALL_CAMHEIGHT ) { |
||||
temppos.y = camLeft.getLocation().y; |
||||
} |
||||
positionTo(temppos, q, tpf); |
||||
rotateScreenTo(q, tpf); |
||||
|
||||
} |
||||
break; |
||||
} |
||||
} else { |
||||
throw new IllegalStateException("VR GUI manager is not attached to any environment."); |
||||
} |
||||
} |
||||
|
||||
/** |
||||
* Rotate the GUI to the given direction. |
||||
* @param dir the direction to rotate to. |
||||
* @param tpf the time per frame. |
||||
*/ |
||||
private void rotateScreenTo(Quaternion dir, float tpf) { |
||||
dir.getRotationColumn(2, look).negateLocal(); |
||||
dir.getRotationColumn(0, left).negateLocal(); |
||||
orient.fromAxes(left, dir.getRotationColumn(1, up), look); |
||||
Quaternion rot = tempq.fromRotationMatrix(orient); |
||||
if( posMode == VRGUIPositioningMode.AUTO_CAM_ALL_SKIP_PITCH ){ |
||||
VRUtil.stripToYaw(rot); |
||||
} |
||||
|
||||
if( guiPositioningElastic > 0f && posMode != VRGUIPositioningMode.MANUAL ) { |
||||
// mix pos & dir with current pos & dir
|
||||
EoldDir.nlerp(rot, tpf * guiPositioningElastic); |
||||
guiQuadNode.setLocalRotation(EoldDir); |
||||
} else { |
||||
guiQuadNode.setLocalRotation(rot); |
||||
} |
||||
} |
||||
|
||||
/** |
||||
* Get the GUI distance from the observer. |
||||
* @return the GUI distance from the observer. |
||||
* @see #setGuiDistance(float) |
||||
*/ |
||||
public float getGuiDistance() { |
||||
return guiDistance; |
||||
} |
||||
|
||||
/** |
||||
* Set the GUI distance from the observer. |
||||
* @param newGuiDistance the GUI distance from the observer. |
||||
* @see #getGuiDistance() |
||||
*/ |
||||
public void setGuiDistance(float newGuiDistance) { |
||||
guiDistance = newGuiDistance; |
||||
} |
||||
|
||||
/** |
||||
* Get the GUI scale. |
||||
* @return the GUI scale. |
||||
* @see #setGuiScale(float) |
||||
*/ |
||||
public float getGUIScale(){ |
||||
return guiScale; |
||||
} |
||||
|
||||
/** |
||||
* Set the GUI scale. |
||||
* @param scale the GUI scale. |
||||
* @see #getGUIScale() |
||||
*/ |
||||
public void setGuiScale(float scale) { |
||||
guiScale = scale; |
||||
} |
||||
|
||||
/** |
||||
* Adjust the GUI distance from the observer. |
||||
* This method increment / decrement the {@link #getGuiDistance() GUI distance} by the given value. |
||||
* @param adjustAmount the increment (if positive) / decrement (if negative) value of the GUI distance. |
||||
*/ |
||||
public void adjustGuiDistance(float adjustAmount) { |
||||
guiDistance += adjustAmount; |
||||
} |
||||
|
||||
/** |
||||
* Set up the GUI. |
||||
* @param leftcam the left eye camera. |
||||
* @param rightcam the right eye camera. |
||||
* @param left the left eye viewport. |
||||
* @param right the right eye viewport. |
||||
*/ |
||||
protected void setupGui(Camera leftcam, Camera rightcam, ViewPort left, ViewPort right) { |
||||
|
||||
if (environment != null){ |
||||
if( environment.hasTraditionalGUIOverlay() ) { |
||||
camLeft = leftcam; |
||||
camRight = rightcam; |
||||
Spatial guiScene = getGuiQuad(camLeft); |
||||
left.attachScene(guiScene); |
||||
if( right != null ) right.attachScene(guiScene); |
||||
setPositioningMode(posMode); |
||||
} |
||||
} else { |
||||
throw new IllegalStateException("VR GUI manager is not attached to any environment."); |
||||
} |
||||
} |
||||
|
||||
/** |
||||
* Get if the GUI has to use curved surface. |
||||
* @return <code>true</code> if the GUI has to use curved surface and <code>false</code> otherwise. |
||||
* @see #setCurvedSurface(boolean) |
||||
*/ |
||||
public boolean isCurverSurface(){ |
||||
return useCurvedSurface; |
||||
} |
||||
|
||||
/** |
||||
* Set if the GUI has to use curved surface. |
||||
* @param set <code>true</code> if the GUI has to use curved surface and <code>false</code> otherwise. |
||||
* @see #isCurverSurface() |
||||
*/ |
||||
public void setCurvedSurface(boolean set) { |
||||
useCurvedSurface = set; |
||||
} |
||||
|
||||
/** |
||||
* Get if the GUI has to be displayed even if it is behind objects. |
||||
* @return <code>true</code> if the GUI has to use curved surface and <code>false</code> otherwise. |
||||
* @see #setGuiOverdraw(boolean) |
||||
*/ |
||||
public boolean isGuiOverdraw(){ |
||||
return overdraw; |
||||
} |
||||
|
||||
/** |
||||
* Set if the GUI has to be displayed even if it is behind objects. |
||||
* @param set <code>true</code> if the GUI has to use curved surface and <code>false</code> otherwise. |
||||
* @see #isGuiOverdraw() |
||||
*/ |
||||
public void setGuiOverdraw(boolean set) { |
||||
overdraw = set; |
||||
} |
||||
|
||||
/** |
||||
* Create a GUI quad for the given camera. |
||||
* @param sourceCam the camera |
||||
* @return a GUI quad for the given camera. |
||||
*/ |
||||
private Spatial getGuiQuad(Camera sourceCam){ |
||||
|
||||
if (environment != null){ |
||||
|
||||
if (environment.getApplication() != null){ |
||||
if( guiQuadNode == null ) { |
||||
Vector2f guiCanvasSize = getCanvasSize(); |
||||
Camera offCamera = sourceCam.clone(); |
||||
offCamera.setParallelProjection(true); |
||||
offCamera.setLocation(Vector3f.ZERO); |
||||
offCamera.lookAt(Vector3f.UNIT_Z, Vector3f.UNIT_Y); |
||||
|
||||
offView = environment.getApplication().getRenderManager().createPreView("GUI View", offCamera); |
||||
offView.setClearFlags(true, true, true); |
||||
offView.setBackgroundColor(ColorRGBA.BlackNoAlpha); |
||||
|
||||
// create offscreen framebuffer
|
||||
FrameBuffer offBuffer = new FrameBuffer((int)guiCanvasSize.x, (int)guiCanvasSize.y, 1); |
||||
|
||||
//setup framebuffer's texture
|
||||
guiTexture = new Texture2D((int)guiCanvasSize.x, (int)guiCanvasSize.y, Format.RGBA8); |
||||
guiTexture.setMinFilter(Texture.MinFilter.BilinearNoMipMaps); |
||||
guiTexture.setMagFilter(Texture.MagFilter.Bilinear); |
||||
|
||||
//setup framebuffer to use texture
|
||||
offBuffer.setDepthBuffer(Format.Depth); |
||||
offBuffer.setColorTexture(guiTexture); |
||||
|
||||
//set viewport to render to offscreen framebuffer
|
||||
offView.setOutputFrameBuffer(offBuffer); |
||||
|
||||
// setup framebuffer's scene
|
||||
Iterator<Spatial> spatialIter = environment.getApplication().getGuiViewPort().getScenes().iterator(); |
||||
while(spatialIter.hasNext()){ |
||||
offView.attachScene(spatialIter.next()); |
||||
} |
||||
|
||||
|
||||
if( useCurvedSurface ) { |
||||
guiQuad = (Geometry)environment.getApplication().getAssetManager().loadModel("Common/Util/gui_mesh.j3o"); |
||||
} else { |
||||
guiQuad = new Geometry("guiQuad", new CenterQuad(1f, 1f)); |
||||
} |
||||
|
||||
Material mat = new Material(environment.getApplication().getAssetManager(), "Common/MatDefs/VR/GuiOverlay.j3md"); |
||||
mat.getAdditionalRenderState().setDepthTest(!overdraw); |
||||
mat.getAdditionalRenderState().setBlendMode(BlendMode.Alpha); |
||||
mat.getAdditionalRenderState().setDepthWrite(false); |
||||
mat.setTexture("ColorMap", guiTexture); |
||||
guiQuad.setQueueBucket(Bucket.Translucent); |
||||
guiQuad.setMaterial(mat); |
||||
|
||||
guiQuadNode = new Node("gui-quad-node"); |
||||
guiQuadNode.setQueueBucket(Bucket.Translucent); |
||||
guiQuadNode.attachChild(guiQuad); |
||||
} |
||||
return guiQuadNode; |
||||
} else { |
||||
throw new IllegalStateException("VR GUI manager underlying environment is not attached to any application."); |
||||
} |
||||
} else { |
||||
throw new IllegalStateException("VR GUI manager is not attached to any environment."); |
||||
} |
||||
|
||||
|
||||
|
||||
} |
||||
} |
@ -0,0 +1,334 @@ |
||||
/* |
||||
* To change this license header, choose License Headers in Project Properties. |
||||
* To change this template file, choose Tools | Templates |
||||
* and open the template in the editor. |
||||
*/ |
||||
package com.jme3.util; |
||||
|
||||
import java.util.logging.Logger; |
||||
|
||||
import org.lwjgl.glfw.GLFW; |
||||
|
||||
import com.jme3.app.VREnvironment; |
||||
import com.jme3.input.MouseInput; |
||||
import com.jme3.input.controls.AnalogListener; |
||||
import com.jme3.input.lwjgl.GlfwMouseInputVR; |
||||
import com.jme3.input.vr.VRInputType; |
||||
import com.jme3.material.RenderState.BlendMode; |
||||
import com.jme3.math.Vector2f; |
||||
import com.jme3.scene.Node; |
||||
import com.jme3.system.AppSettings; |
||||
import com.jme3.system.lwjgl.LwjglWindow; |
||||
import com.jme3.texture.Texture; |
||||
import com.jme3.texture.Texture2D; |
||||
import com.jme3.ui.Picture; |
||||
|
||||
/** |
||||
* A class dedicated to the handling of the mouse within VR environment. |
||||
* @author Julien Seinturier - (c) 2016 - JOrigin project - <a href="http://www.jorigin.org">http:/www.jorigin.org</a> |
||||
* |
||||
*/ |
||||
public class VRMouseManager { |
||||
|
||||
private static final Logger logger = Logger.getLogger(VRMouseManager.class.getName()); |
||||
|
||||
|
||||
private VREnvironment environment = null; |
||||
|
||||
private final int AVERAGE_AMNT = 4; |
||||
private int avgCounter; |
||||
|
||||
private Picture mouseImage; |
||||
private int recentCenterCount = 0; |
||||
private final Vector2f cursorPos = new Vector2f(); |
||||
private float ySize, sensitivity = 8f, acceleration = 2f; |
||||
private final float[] lastXmv = new float[AVERAGE_AMNT], lastYmv = new float[AVERAGE_AMNT]; |
||||
private boolean thumbstickMode; |
||||
private float moveScale = 1f; |
||||
|
||||
private float avg(float[] arr) { |
||||
float amt = 0f; |
||||
for(float f : arr) amt += f; |
||||
return amt / arr.length; |
||||
} |
||||
|
||||
/** |
||||
* Create a new VR mouse manager within the given {@link VREnvironment VR environment}. |
||||
* @param environment the VR environment of the mouse manager. |
||||
*/ |
||||
public VRMouseManager(VREnvironment environment){ |
||||
this.environment = environment; |
||||
} |
||||
|
||||
/** |
||||
* Initialize the VR mouse manager. |
||||
*/ |
||||
protected void initialize() { |
||||
|
||||
logger.config("Initializing VR mouse manager."); |
||||
|
||||
// load default mouseimage
|
||||
mouseImage = new Picture("mouse"); |
||||
setImage("Common/Util/mouse.png"); |
||||
// hide default cursor by making it invisible
|
||||
|
||||
MouseInput mi = environment.getApplication().getContext().getMouseInput(); |
||||
if( mi instanceof GlfwMouseInputVR ){ |
||||
((GlfwMouseInputVR)mi).hideActiveCursor(); |
||||
} |
||||
centerMouse(); |
||||
|
||||
logger.config("Initialized VR mouse manager [SUCCESS]"); |
||||
} |
||||
|
||||
public void setThumbstickMode(boolean set) { |
||||
thumbstickMode = set; |
||||
} |
||||
|
||||
public boolean isThumbstickMode() { |
||||
return thumbstickMode; |
||||
} |
||||
|
||||
/** |
||||
* Set the speed of the mouse. |
||||
* @param sensitivity the sensitivity of the mouse. |
||||
* @param acceleration the acceleration of the mouse. |
||||
* @see #getSpeedAcceleration() |
||||
* @see #getSpeedSensitivity() |
||||
*/ |
||||
public void setSpeed(float sensitivity, float acceleration) { |
||||
this.sensitivity = sensitivity; |
||||
this.acceleration = acceleration; |
||||
} |
||||
|
||||
/** |
||||
* Get the sensitivity of the mouse. |
||||
* @return the sensitivity of the mouse. |
||||
* @see #setSpeed(float, float) |
||||
*/ |
||||
public float getSpeedSensitivity() { |
||||
return sensitivity; |
||||
} |
||||
|
||||
/** |
||||
* Get the acceleration of the mouse. |
||||
* @return the acceleration of the mouse. |
||||
* @see #setSpeed(float, float) |
||||
*/ |
||||
public float getSpeedAcceleration() { |
||||
return acceleration; |
||||
} |
||||
|
||||
/** |
||||
* Set the mouse move scale. |
||||
* @param set the mouse move scale. |
||||
*/ |
||||
public void setMouseMoveScale(float set) { |
||||
moveScale = set; |
||||
} |
||||
|
||||
/** |
||||
* Set the image to use as mouse cursor. The given string describe an asset that the underlying application asset manager has to load. |
||||
* @param texture the image to use as mouse cursor. |
||||
*/ |
||||
public void setImage(String texture) { |
||||
|
||||
if (environment != null){ |
||||
|
||||
if (environment.getApplication() != null){ |
||||
if( environment.isInVR() == false ){ |
||||
Texture tex = environment.getApplication().getAssetManager().loadTexture(texture); |
||||
mouseImage.setTexture(environment.getApplication().getAssetManager(), (Texture2D)tex, true); |
||||
ySize = tex.getImage().getHeight(); |
||||
mouseImage.setHeight(ySize); |
||||
mouseImage.setWidth(tex.getImage().getWidth()); |
||||
mouseImage.getMaterial().getAdditionalRenderState().setBlendMode(BlendMode.Alpha); |
||||
mouseImage.getMaterial().getAdditionalRenderState().setDepthWrite(false); |
||||
} else { |
||||
Texture tex = environment.getApplication().getAssetManager().loadTexture(texture); |
||||
mouseImage.setTexture(environment.getApplication().getAssetManager(), (Texture2D)tex, true); |
||||
ySize = tex.getImage().getHeight(); |
||||
mouseImage.setHeight(ySize); |
||||
mouseImage.setWidth(tex.getImage().getWidth()); |
||||
mouseImage.getMaterial().getAdditionalRenderState().setBlendMode(BlendMode.Alpha); |
||||
mouseImage.getMaterial().getAdditionalRenderState().setDepthWrite(false); |
||||
} |
||||
} else { |
||||
throw new IllegalStateException("This VR environment is not attached to any application."); |
||||
} |
||||
|
||||
} else { |
||||
throw new IllegalStateException("This VR view manager is not attached to any VR environment."); |
||||
} |
||||
} |
||||
|
||||
/** |
||||
* Update analog controller as it was a mouse controller. |
||||
* @param inputIndex the index of the controller attached to the VR system. |
||||
* @param mouseListener the JMonkey mouse listener to trigger. |
||||
* @param mouseXName the mouseX identifier. |
||||
* @param mouseYName the mouseY identifier |
||||
* @param tpf the time per frame. |
||||
*/ |
||||
public void updateAnalogAsMouse(int inputIndex, AnalogListener mouseListener, String mouseXName, String mouseYName, float tpf) { |
||||
|
||||
if (environment != null){ |
||||
if (environment.getApplication() != null){ |
||||
// got a tracked controller to use as the "mouse"
|
||||
if( environment.isInVR() == false || |
||||
environment.getVRinput() == null || |
||||
environment.getVRinput().isInputDeviceTracking(inputIndex) == false ){ |
||||
return; |
||||
} |
||||
|
||||
Vector2f tpDelta; |
||||
if( thumbstickMode ) { |
||||
tpDelta = environment.getVRinput().getAxis(inputIndex, VRInputType.ViveTrackpadAxis); |
||||
} else { |
||||
tpDelta = environment.getVRinput().getAxisDeltaSinceLastCall(inputIndex, VRInputType.ViveTrackpadAxis); |
||||
} |
||||
|
||||
float Xamount = (float)Math.pow(Math.abs(tpDelta.x) * sensitivity, acceleration); |
||||
float Yamount = (float)Math.pow(Math.abs(tpDelta.y) * sensitivity, acceleration); |
||||
|
||||
if( tpDelta.x < 0f ){ |
||||
Xamount = -Xamount; |
||||
} |
||||
|
||||
if( tpDelta.y < 0f ){ |
||||
Yamount = -Yamount; |
||||
} |
||||
|
||||
Xamount *= moveScale; Yamount *= moveScale; |
||||
if( mouseListener != null ) { |
||||
if( tpDelta.x != 0f && mouseXName != null ) mouseListener.onAnalog(mouseXName, Xamount * 0.2f, tpf); |
||||
if( tpDelta.y != 0f && mouseYName != null ) mouseListener.onAnalog(mouseYName, Yamount * 0.2f, tpf); |
||||
} |
||||
|
||||
if( environment.getApplication().getInputManager().isCursorVisible() ) { |
||||
int index = (avgCounter+1) % AVERAGE_AMNT; |
||||
lastXmv[index] = Xamount * 133f; |
||||
lastYmv[index] = Yamount * 133f; |
||||
cursorPos.x -= avg(lastXmv); |
||||
cursorPos.y -= avg(lastYmv); |
||||
Vector2f maxsize = environment.getVRGUIManager().getCanvasSize(); |
||||
|
||||
if( cursorPos.x > maxsize.x ){ |
||||
cursorPos.x = maxsize.x; |
||||
} |
||||
|
||||
if( cursorPos.x < 0f ){ |
||||
cursorPos.x = 0f; |
||||
} |
||||
|
||||
if( cursorPos.y > maxsize.y ){ |
||||
cursorPos.y = maxsize.y; |
||||
} |
||||
|
||||
if( cursorPos.y < 0f ){ |
||||
cursorPos.y = 0f; |
||||
} |
||||
} |
||||
} else { |
||||
throw new IllegalStateException("This VR environment is not attached to any application."); |
||||
} |
||||
} else { |
||||
throw new IllegalStateException("This VR view manager is not attached to any VR environment."); |
||||
} |
||||
} |
||||
|
||||
/** |
||||
* Get the actual cursor position. |
||||
* @return the actual cursor position. |
||||
*/ |
||||
public Vector2f getCursorPosition() { |
||||
|
||||
if (environment != null){ |
||||
if (environment.getApplication() != null){ |
||||
if( environment.isInVR() ) { |
||||
return cursorPos; |
||||
} |
||||
|
||||
return environment.getApplication().getInputManager().getCursorPosition(); |
||||
} else { |
||||
throw new IllegalStateException("This VR environment is not attached to any application."); |
||||
} |
||||
} else { |
||||
throw new IllegalStateException("This VR view manager is not attached to any VR environment."); |
||||
} |
||||
} |
||||
|
||||
/** |
||||
* Center the mouse on the display. |
||||
*/ |
||||
public void centerMouse() { |
||||
|
||||
if (environment != null){ |
||||
if (environment.getApplication() != null){ |
||||
// set mouse in center of the screen if newly added
|
||||
Vector2f size = environment.getVRGUIManager().getCanvasSize(); |
||||
MouseInput mi = environment.getApplication().getContext().getMouseInput(); |
||||
AppSettings as = environment.getApplication().getContext().getSettings(); |
||||
if( mi instanceof GlfwMouseInputVR ) ((GlfwMouseInputVR)mi).setCursorPosition((int)(as.getWidth() / 2f), (int)(as.getHeight() / 2f)); |
||||
if( environment.isInVR() ) { |
||||
cursorPos.x = size.x / 2f; |
||||
cursorPos.y = size.y / 2f; |
||||
recentCenterCount = 2; |
||||
} |
||||
} else { |
||||
throw new IllegalStateException("This VR environment is not attached to any application."); |
||||
} |
||||
} else { |
||||
throw new IllegalStateException("This VR view manager is not attached to any VR environment."); |
||||
} |
||||
|
||||
} |
||||
|
||||
/** |
||||
* Update the mouse manager. This method should not be called manually. |
||||
* The standard behavior for this method is to be called from the {@link VRViewManager#update(float) update method} of the attached {@link VRViewManager VR view manager}. |
||||
* @param tpf the time per frame. |
||||
*/ |
||||
protected void update(float tpf) { |
||||
// if we are showing the cursor, add our picture as it
|
||||
|
||||
if( environment.getApplication().getInputManager().isCursorVisible() ) { |
||||
if( mouseImage.getParent() == null ) { |
||||
|
||||
environment.getApplication().getGuiViewPort().attachScene(mouseImage); |
||||
centerMouse(); |
||||
// the "real" mouse pointer should stay hidden
|
||||
if (environment.getApplication().getContext() instanceof LwjglWindow){ |
||||
GLFW.glfwSetInputMode(((LwjglWindow)environment.getApplication().getContext()).getWindowHandle(), GLFW.GLFW_CURSOR, GLFW.GLFW_CURSOR_DISABLED); |
||||
} |
||||
} |
||||
// handle mouse movements, which may be in addition to (or exclusive from) tracked movement
|
||||
MouseInput mi = environment.getApplication().getContext().getMouseInput(); |
||||
if( mi instanceof GlfwMouseInputVR ) { |
||||
if( recentCenterCount <= 0 ) { |
||||
//Vector2f winratio = VRGuiManager.getCanvasToWindowRatio();
|
||||
cursorPos.x += ((GlfwMouseInputVR)mi).getLastDeltaX();// * winratio.x;
|
||||
cursorPos.y += ((GlfwMouseInputVR)mi).getLastDeltaY();// * winratio.y;
|
||||
if( cursorPos.x < 0f ) cursorPos.x = 0f; |
||||
if( cursorPos.y < 0f ) cursorPos.y = 0f; |
||||
if( cursorPos.x > environment.getVRGUIManager().getCanvasSize().x ) cursorPos.x = environment.getVRGUIManager().getCanvasSize().x; |
||||
if( cursorPos.y > environment.getVRGUIManager().getCanvasSize().y ) cursorPos.y = environment.getVRGUIManager().getCanvasSize().y; |
||||
} else recentCenterCount--; |
||||
((GlfwMouseInputVR)mi).clearDeltas(); |
||||
} |
||||
// ok, update the cursor graphic position
|
||||
Vector2f currentPos = getCursorPosition(); |
||||
mouseImage.setLocalTranslation(currentPos.x, currentPos.y - ySize, environment.getVRGUIManager().getGuiDistance() + 1f); |
||||
|
||||
mouseImage.updateGeometricState(); |
||||
|
||||
} else if( mouseImage.getParent() != null ) { |
||||
Node n = mouseImage.getParent(); |
||||
mouseImage.removeFromParent(); |
||||
|
||||
if (n != null){ |
||||
n.updateGeometricState(); |
||||
} |
||||
} |
||||
} |
||||
} |
@ -0,0 +1,156 @@ |
||||
package com.jme3.util; |
||||
|
||||
import com.jme3.app.VRAppState; |
||||
import com.jme3.app.VREnvironment; |
||||
import com.jme3.app.state.AppState; |
||||
import com.jme3.post.FilterPostProcessor; |
||||
import com.jme3.renderer.Camera; |
||||
import com.jme3.renderer.ViewPort; |
||||
import com.jme3.texture.Texture2D; |
||||
|
||||
/** |
||||
* A VR view manager. This interface describes methods that enable to submit 3D views to the VR compositor. |
||||
* @author reden - phr00t - https://github.com/phr00t
|
||||
* @author Julien Seinturier - (c) 2016 - JOrigin project - <a href="http://www.jorigin.org">http:/www.jorigin.org</a> |
||||
*/ |
||||
public interface VRViewManager { |
||||
|
||||
/** |
||||
* The name of the left view. |
||||
*/ |
||||
public final static String LEFT_VIEW_NAME = "Left View"; |
||||
|
||||
/** |
||||
* The name of the right view. |
||||
*/ |
||||
public final static String RIGHT_VIEW_NAME = "Right View"; |
||||
|
||||
/** |
||||
* Get the {@link Camera camera} attached to the left eye. |
||||
* @return the {@link Camera camera} attached to the left eye. |
||||
* @see #getRightCamera() |
||||
*/ |
||||
public Camera getLeftCamera(); |
||||
|
||||
/** |
||||
* Get the {@link Camera camera} attached to the right eye. |
||||
* @return the {@link Camera camera} attached to the right eye. |
||||
* @see #getLeftCamera() |
||||
*/ |
||||
public Camera getRightCamera(); |
||||
|
||||
/** |
||||
* Get the {@link ViewPort viewport} attached to the left eye. |
||||
* @return the {@link ViewPort viewport} attached to the left eye. |
||||
* @see #getRightViewport() |
||||
*/ |
||||
public ViewPort getLeftViewport(); |
||||
|
||||
|
||||
/** |
||||
* Get the {@link ViewPort viewport} attached to the right eye. |
||||
* @return the {@link ViewPort viewport} attached to the right eye. |
||||
* @see #getLeftViewport() |
||||
*/ |
||||
public ViewPort getRightViewport(); |
||||
|
||||
/** |
||||
* Get the texture attached to the left eye. |
||||
* @return the texture attached to the left eye. |
||||
* @see #getRightTexture() |
||||
*/ |
||||
public Texture2D getLeftTexture(); |
||||
|
||||
/** |
||||
* Get the texture attached to the right eye. |
||||
* @return the texture attached to the right eye. |
||||
* @see #getLeftTexture() |
||||
*/ |
||||
public Texture2D getRightTexture(); |
||||
|
||||
/** |
||||
* Get the depth texture attached to the left eye. |
||||
* @return the texture attached to the left eye. |
||||
* @see #getRightTexture() |
||||
*/ |
||||
public Texture2D getLeftDepth(); |
||||
|
||||
/** |
||||
* Get the depth texture attached to the right eye. |
||||
* @return the texture attached to the right eye. |
||||
* @see #getLeftTexture() |
||||
*/ |
||||
public Texture2D getRightDepth(); |
||||
|
||||
/** |
||||
* Get the {@link FilterPostProcessor filter post processor} attached to the left eye. |
||||
* @return the {@link FilterPostProcessor filter post processor} attached to the left eye. |
||||
* @see #getRightPostProcessor() |
||||
*/ |
||||
public FilterPostProcessor getLeftPostProcessor(); |
||||
|
||||
/** |
||||
* Get the {@link FilterPostProcessor filter post processor} attached to the right eye. |
||||
* @return the {@link FilterPostProcessor filter post processor} attached to the right eye. |
||||
* @see #getLeftPostProcessor() |
||||
*/ |
||||
public FilterPostProcessor getRightPostProcessor(); |
||||
|
||||
/** |
||||
* Get the resolution multiplier. |
||||
* @return the resolution multiplier. |
||||
* @see #setResolutionMultiplier(float) |
||||
*/ |
||||
public float getResolutionMuliplier(); |
||||
|
||||
/** |
||||
* Set the resolution multiplier. |
||||
* @param resMult the resolution multiplier. |
||||
* @see #getResolutionMuliplier() |
||||
*/ |
||||
public void setResolutionMultiplier(float resMult); |
||||
|
||||
/** |
||||
* Get the height adjustment to apply to the cameras before rendering. |
||||
* @return the height adjustment to apply to the cameras before rendering. |
||||
* @see #setHeightAdjustment(float) |
||||
*/ |
||||
public float getHeightAdjustment(); |
||||
|
||||
/** |
||||
* Set the height adjustment to apply to the cameras before rendering. |
||||
* @param amount the height adjustment to apply to the cameras before rendering. |
||||
* @see #getHeightAdjustment() |
||||
*/ |
||||
public void setHeightAdjustment(float amount); |
||||
|
||||
/** |
||||
* Get the {@link VREnvironment VR environment} to which the view manager is attached. |
||||
* @return the {@link VREnvironment VR environment} to which the view manager is attached. |
||||
*/ |
||||
public VREnvironment getVREnvironment(); |
||||
|
||||
/** |
||||
* Initialize the VR view manager. This method should be called after the attachment of a {@link VREnvironment VR environment} to an application. |
||||
*/ |
||||
public void initialize(); |
||||
|
||||
/** |
||||
* Update the VR view manager. |
||||
* This method is called by the attached {@link VRAppState app state} and should not be called manually. |
||||
* @param tpf the time per frame. |
||||
*/ |
||||
public void update(float tpf); |
||||
|
||||
/** |
||||
* Send the rendering result as textures to the two eyes. |
||||
* This method should be called after all the rendering operations |
||||
* (for example at the end of the {@link AppState#postRender() postRender()} method of the attached app state.) |
||||
*/ |
||||
public void postRender(); |
||||
|
||||
/** |
||||
* Handles moving filters from the main view to each eye. |
||||
*/ |
||||
public void moveScreenProcessingToEyes(); |
||||
} |
@ -0,0 +1,957 @@ |
||||
package com.jme3.util; |
||||
|
||||
import java.awt.GraphicsEnvironment; |
||||
import java.util.Iterator; |
||||
import java.util.logging.Logger; |
||||
|
||||
import com.jme3.app.VREnvironment; |
||||
import com.jme3.input.vr.OSVR; |
||||
import com.jme3.input.vr.VRAPI; |
||||
import com.jme3.material.Material; |
||||
import com.jme3.math.ColorRGBA; |
||||
import com.jme3.math.Quaternion; |
||||
import com.jme3.math.Vector2f; |
||||
import com.jme3.math.Vector3f; |
||||
import com.jme3.post.CartoonSSAO; |
||||
import com.jme3.post.Filter; |
||||
import com.jme3.post.FilterPostProcessor; |
||||
import com.jme3.post.FilterUtil; |
||||
import com.jme3.post.SceneProcessor; |
||||
import com.jme3.post.filters.FogFilter; |
||||
import com.jme3.post.filters.TranslucentBucketFilter; |
||||
import com.jme3.post.ssao.SSAOFilter; |
||||
import com.jme3.renderer.Camera; |
||||
import com.jme3.renderer.ViewPort; |
||||
import com.jme3.renderer.queue.RenderQueue.Bucket; |
||||
import com.jme3.scene.Geometry; |
||||
import com.jme3.scene.Mesh; |
||||
import com.jme3.scene.Node; |
||||
import com.jme3.scene.Spatial; |
||||
import com.jme3.scene.VertexBuffer; |
||||
import com.jme3.shadow.DirectionalLightShadowFilter; |
||||
import com.jme3.shadow.VRDirectionalLightShadowRenderer; |
||||
import com.jme3.system.jopenvr.DistortionCoordinates_t; |
||||
import com.jme3.system.jopenvr.JOpenVRLibrary; |
||||
import com.jme3.system.jopenvr.OpenVRUtil; |
||||
import com.jme3.system.jopenvr.Texture_t; |
||||
import com.jme3.system.jopenvr.VR_IVRSystem_FnTable; |
||||
import com.jme3.system.lwjgl.LwjglWindow; |
||||
import com.jme3.system.osvr.osvrrendermanageropengl.OSVR_RenderBufferOpenGL; |
||||
import com.jme3.system.osvr.osvrrendermanageropengl.OSVR_ViewportDescription; |
||||
import com.jme3.system.osvr.osvrrendermanageropengl.OsvrRenderManagerOpenGLLibrary; |
||||
import com.jme3.texture.FrameBuffer; |
||||
import com.jme3.texture.Image; |
||||
import com.jme3.texture.Texture; |
||||
import com.jme3.texture.Texture2D; |
||||
import com.jme3.ui.Picture; |
||||
import com.sun.jna.Pointer; |
||||
import com.sun.jna.ptr.PointerByReference; |
||||
|
||||
public class VRViewManagerOSVR extends AbstractVRViewManager{ |
||||
private static final Logger logger = Logger.getLogger(VRViewManagerOpenVR.class.getName()); |
||||
|
||||
private Camera leftCamera; |
||||
private ViewPort leftViewport; |
||||
private FilterPostProcessor leftPostProcessor; |
||||
private Texture2D leftEyeTexture; |
||||
private Texture2D leftEyeDepth; |
||||
|
||||
private Camera rightCamera; |
||||
private ViewPort rightViewport; |
||||
private FilterPostProcessor rightPostProcessor; |
||||
private Texture2D rightEyeTexture; |
||||
private Texture2D rightEyeDepth; |
||||
|
||||
// OpenVR values
|
||||
private Texture_t leftTextureType; |
||||
private Texture_t rightTextureType; |
||||
|
||||
// OSVR values
|
||||
OSVR_RenderBufferOpenGL.ByValue[] osvr_renderBuffer; |
||||
OSVR_ViewportDescription.ByValue osvr_viewDescFull; |
||||
OSVR_ViewportDescription.ByValue osvr_viewDescLeft; |
||||
OSVR_ViewportDescription.ByValue osvr_viewDescRight; |
||||
Pointer osvr_rmBufferState; |
||||
|
||||
//private static boolean useCustomDistortion;
|
||||
private float heightAdjustment; |
||||
|
||||
private Texture2D dualEyeTex; |
||||
|
||||
private final PointerByReference grabRBS = new PointerByReference(); |
||||
|
||||
private float resMult = 1f; |
||||
|
||||
//final & temp values for camera calculations
|
||||
private final Vector3f finalPosition = new Vector3f(); |
||||
private final Quaternion finalRotation = new Quaternion(); |
||||
private final Vector3f hmdPos = new Vector3f(); |
||||
private final Quaternion hmdRot = new Quaternion(); |
||||
|
||||
/** |
||||
* Create a new VR view manager attached to the given {@link VREnvironment VR environment}. |
||||
* @param environment the {@link VREnvironment VR environment} to which this view manager is attached. |
||||
*/ |
||||
public VRViewManagerOSVR(VREnvironment environment){ |
||||
this.environment = environment; |
||||
} |
||||
|
||||
/** |
||||
* Get the {@link Camera camera} attached to the left eye. |
||||
* @return the {@link Camera camera} attached to the left eye. |
||||
* @see #getRightCamera() |
||||
*/ |
||||
public Camera getLeftCamera() { |
||||
return leftCamera; |
||||
} |
||||
|
||||
/** |
||||
* Get the {@link Camera camera} attached to the right eye. |
||||
* @return the {@link Camera camera} attached to the right eye. |
||||
* @see #getLeftCamera() |
||||
*/ |
||||
public Camera getRightCamera() { |
||||
return rightCamera; |
||||
} |
||||
|
||||
/** |
||||
* Get the {@link ViewPort viewport} attached to the left eye. |
||||
* @return the {@link ViewPort viewport} attached to the left eye. |
||||
* @see #getRightViewport() |
||||
*/ |
||||
public ViewPort getLeftViewport() { |
||||
return leftViewport; |
||||
} |
||||
|
||||
/** |
||||
* Get the {@link ViewPort viewport} attached to the right eye. |
||||
* @return the {@link ViewPort viewport} attached to the right eye. |
||||
* @see #getLeftViewport() |
||||
*/ |
||||
public ViewPort getRightViewport() { |
||||
return rightViewport; |
||||
} |
||||
|
||||
/** |
||||
* Get the identifier of the left eye texture. |
||||
* @return the identifier of the left eye texture. |
||||
* @see #getRightTexId() |
||||
* @see #getFullTexId() |
||||
*/ |
||||
protected int getLeftTexId() { |
||||
return (int)leftEyeTexture.getImage().getId(); |
||||
} |
||||
|
||||
/** |
||||
* Get the identifier of the right eye texture. |
||||
* @return the identifier of the right eye texture. |
||||
* @see #getLeftTexId() |
||||
* @see #getFullTexId() |
||||
*/ |
||||
protected int getRightTexId() { |
||||
return (int)rightEyeTexture.getImage().getId(); |
||||
} |
||||
|
||||
/** |
||||
* Get the identifier of the full (dual eye) texture. |
||||
* @return the identifier of the full (dual eye) texture. |
||||
* @see #getLeftTexId() |
||||
* @see #getRightTexId() |
||||
*/ |
||||
private int getFullTexId() { |
||||
return (int)dualEyeTex.getImage().getId(); |
||||
} |
||||
|
||||
/** |
||||
* Get the height adjustment to apply to the cameras before rendering. |
||||
* @return the height adjustment to apply to the cameras before rendering. |
||||
* @see #setHeightAdjustment(float) |
||||
*/ |
||||
public float getHeightAdjustment() { |
||||
return heightAdjustment; |
||||
} |
||||
|
||||
/** |
||||
* Set the height adjustment to apply to the cameras before rendering. |
||||
* @param amount the height adjustment to apply to the cameras before rendering. |
||||
* @see #getHeightAdjustment() |
||||
*/ |
||||
public void setHeightAdjustment(float amount) { |
||||
heightAdjustment = amount; |
||||
} |
||||
|
||||
/** |
||||
* Get the resolution multiplier. |
||||
* @return the resolution multiplier. |
||||
* @see #setResolutionMultiplier(float) |
||||
*/ |
||||
public float getResolutionMuliplier() { |
||||
return resMult; |
||||
} |
||||
|
||||
/** |
||||
* Set the resolution multiplier. |
||||
* @param resMult the resolution multiplier. |
||||
* @see #getResolutionMuliplier() |
||||
*/ |
||||
public void setResolutionMultiplier(float resMult) { |
||||
this.resMult = resMult; |
||||
} |
||||
|
||||
/** |
||||
* Initialize the system binds of the textures. |
||||
*/ |
||||
private void initTextureSubmitStructs() { |
||||
leftTextureType = new Texture_t(); |
||||
rightTextureType = new Texture_t(); |
||||
|
||||
// must be OSVR
|
||||
osvr_renderBuffer = new OSVR_RenderBufferOpenGL.ByValue[2]; |
||||
osvr_renderBuffer[OSVR.EYE_LEFT] = new OSVR_RenderBufferOpenGL.ByValue(); |
||||
osvr_renderBuffer[OSVR.EYE_RIGHT] = new OSVR_RenderBufferOpenGL.ByValue(); |
||||
osvr_renderBuffer[OSVR.EYE_LEFT].setAutoSynch(false); |
||||
osvr_renderBuffer[OSVR.EYE_RIGHT].setAutoSynch(false); |
||||
osvr_viewDescFull = new OSVR_ViewportDescription.ByValue(); |
||||
osvr_viewDescFull.setAutoSynch(false); |
||||
osvr_viewDescFull.left = osvr_viewDescFull.lower = 0.0; |
||||
osvr_viewDescFull.width = osvr_viewDescFull.height = 1.0; |
||||
osvr_viewDescLeft = new OSVR_ViewportDescription.ByValue(); |
||||
osvr_viewDescLeft.setAutoSynch(false); |
||||
osvr_viewDescLeft.left = osvr_viewDescLeft.lower = 0.0; |
||||
osvr_viewDescLeft.width = 0.5; |
||||
osvr_viewDescLeft.height = 1.0; |
||||
osvr_viewDescRight = new OSVR_ViewportDescription.ByValue(); |
||||
osvr_viewDescRight.setAutoSynch(false); |
||||
osvr_viewDescRight.left = 0.5; |
||||
osvr_viewDescRight.lower = 0.0; |
||||
osvr_viewDescRight.width = 0.5; |
||||
osvr_viewDescRight.height = 1.0; |
||||
osvr_viewDescRight.write(); |
||||
osvr_viewDescLeft.write(); |
||||
osvr_viewDescFull.write(); |
||||
osvr_renderBuffer[OSVR.EYE_LEFT].depthStencilBufferName = -1; |
||||
osvr_renderBuffer[OSVR.EYE_LEFT].colorBufferName = -1; |
||||
osvr_renderBuffer[OSVR.EYE_RIGHT].depthStencilBufferName = -1; |
||||
osvr_renderBuffer[OSVR.EYE_RIGHT].colorBufferName = -1; |
||||
} |
||||
|
||||
/** |
||||
* Register the OSVR OpenGL buffer. |
||||
* @param buf the OSVR OpenGL buffer. |
||||
*/ |
||||
private void registerOSVRBuffer(OSVR_RenderBufferOpenGL.ByValue buf) { |
||||
|
||||
if (environment != null){ |
||||
OsvrRenderManagerOpenGLLibrary.osvrRenderManagerStartRegisterRenderBuffers(grabRBS); |
||||
OsvrRenderManagerOpenGLLibrary.osvrRenderManagerRegisterRenderBufferOpenGL(grabRBS.getValue(), buf); |
||||
OsvrRenderManagerOpenGLLibrary.osvrRenderManagerFinishRegisterRenderBuffers(((OSVR)environment.getVRHardware()).getCompositor(), grabRBS.getValue(), (byte)0); |
||||
|
||||
} else { |
||||
throw new IllegalStateException("This VR view manager is not attached to any VR environment."); |
||||
} |
||||
} |
||||
|
||||
/** |
||||
* Send the textures to the two eyes. |
||||
*/ |
||||
public void postRender() { |
||||
|
||||
if (environment != null){ |
||||
if( environment.isInVR() ) { |
||||
VRAPI api = environment.getVRHardware(); |
||||
if( api.getCompositor() != null ) { |
||||
// using the compositor...
|
||||
int errl = 0, errr = 0; |
||||
if( environment.isInstanceRendering() ) { |
||||
if( leftTextureType.handle == -1 || leftTextureType.handle != getFullTexId() ) { |
||||
leftTextureType.handle = getFullTexId(); |
||||
if( leftTextureType.handle != -1 ) { |
||||
leftTextureType.write(); |
||||
if( api instanceof OSVR ) { |
||||
osvr_renderBuffer[OSVR.EYE_LEFT].colorBufferName = leftTextureType.handle; |
||||
osvr_renderBuffer[OSVR.EYE_LEFT].depthStencilBufferName = dualEyeTex.getImage().getId(); |
||||
osvr_renderBuffer[OSVR.EYE_LEFT].write(); |
||||
registerOSVRBuffer(osvr_renderBuffer[OSVR.EYE_LEFT]); |
||||
} |
||||
} |
||||
} else { |
||||
if( api instanceof OSVR ) { |
||||
((OSVR)api).handleRenderBufferPresent(osvr_viewDescLeft, osvr_viewDescRight, |
||||
osvr_renderBuffer[OSVR.EYE_LEFT], osvr_renderBuffer[OSVR.EYE_LEFT]); |
||||
} |
||||
} |
||||
} else if( leftTextureType.handle == -1 || rightTextureType.handle == -1 || |
||||
leftTextureType.handle != getLeftTexId() || rightTextureType.handle != getRightTexId() ) { |
||||
leftTextureType.handle = getLeftTexId(); |
||||
if( leftTextureType.handle != -1 ) { |
||||
logger.fine("Writing Left texture to native memory at " + leftTextureType.getPointer()); |
||||
leftTextureType.write(); |
||||
if( api instanceof OSVR ) { |
||||
osvr_renderBuffer[OSVR.EYE_LEFT].colorBufferName = leftTextureType.handle; |
||||
if( leftEyeDepth != null ) osvr_renderBuffer[OSVR.EYE_LEFT].depthStencilBufferName = leftEyeDepth.getImage().getId(); |
||||
osvr_renderBuffer[OSVR.EYE_LEFT].write(); |
||||
registerOSVRBuffer(osvr_renderBuffer[OSVR.EYE_LEFT]); |
||||
} |
||||
} |
||||
rightTextureType.handle = getRightTexId(); |
||||
if( rightTextureType.handle != -1 ) { |
||||
logger.fine("Writing Right texture to native memory at " + leftTextureType.getPointer()); |
||||
rightTextureType.write(); |
||||
if( api instanceof OSVR ) { |
||||
osvr_renderBuffer[OSVR.EYE_RIGHT].colorBufferName = rightTextureType.handle; |
||||
if( rightEyeDepth != null ) osvr_renderBuffer[OSVR.EYE_RIGHT].depthStencilBufferName = rightEyeDepth.getImage().getId(); |
||||
osvr_renderBuffer[OSVR.EYE_RIGHT].write(); |
||||
registerOSVRBuffer(osvr_renderBuffer[OSVR.EYE_RIGHT]); |
||||
} |
||||
} |
||||
} else { |
||||
if( api instanceof OSVR ) { |
||||
((OSVR)api).handleRenderBufferPresent(osvr_viewDescFull, osvr_viewDescFull, |
||||
osvr_renderBuffer[OSVR.EYE_LEFT], osvr_renderBuffer[OSVR.EYE_RIGHT]); |
||||
} |
||||
} |
||||
|
||||
if( errl != 0 ){ |
||||
logger.severe("Submit to left compositor error: " + OpenVRUtil.getEVRCompositorErrorString(errl)+" ("+Integer.toString(errl)+")"); |
||||
logger.severe(" Texture color space: "+OpenVRUtil.getEColorSpaceString(leftTextureType.eColorSpace)); |
||||
logger.severe(" Texture type: "+OpenVRUtil.getETextureTypeString(leftTextureType.eType)); |
||||
logger.severe(" Texture handle: "+leftTextureType.handle); |
||||
|
||||
logger.severe(" Left eye texture "+leftEyeTexture.getName()+" ("+leftEyeTexture.getImage().getId()+")"); |
||||
logger.severe(" Type: "+leftEyeTexture.getType()); |
||||
logger.severe(" Size: "+leftEyeTexture.getImage().getWidth()+"x"+leftEyeTexture.getImage().getHeight()); |
||||
logger.severe(" Image depth: "+leftEyeTexture.getImage().getDepth()); |
||||
logger.severe(" Image format: "+leftEyeTexture.getImage().getFormat()); |
||||
logger.severe(" Image color space: "+leftEyeTexture.getImage().getColorSpace()); |
||||
|
||||
} |
||||
|
||||
if( errr != 0 ){ |
||||
logger.severe("Submit to right compositor error: " + OpenVRUtil.getEVRCompositorErrorString(errl)+" ("+Integer.toString(errl)+")"); |
||||
logger.severe(" Texture color space: "+OpenVRUtil.getEColorSpaceString(rightTextureType.eColorSpace)); |
||||
logger.severe(" Texture type: "+OpenVRUtil.getETextureTypeString(rightTextureType.eType)); |
||||
logger.severe(" Texture handle: "+rightTextureType.handle); |
||||
|
||||
logger.severe(" Right eye texture "+rightEyeTexture.getName()+" ("+rightEyeTexture.getImage().getId()+")"); |
||||
logger.severe(" Type: "+rightEyeTexture.getType()); |
||||
logger.severe(" Size: "+rightEyeTexture.getImage().getWidth()+"x"+rightEyeTexture.getImage().getHeight()); |
||||
logger.severe(" Image depth: "+rightEyeTexture.getImage().getDepth()); |
||||
logger.severe(" Image format: "+rightEyeTexture.getImage().getFormat()); |
||||
logger.severe(" Image color space: "+rightEyeTexture.getImage().getColorSpace()); |
||||
} |
||||
} |
||||
} |
||||
} else { |
||||
throw new IllegalStateException("This VR view manager is not attached to any VR environment."); |
||||
} |
||||
} |
||||
|
||||
|
||||
/** |
||||
* Initialize the VR view manager. |
||||
*/ |
||||
public void initialize() { |
||||
|
||||
logger.config("Initializing VR view manager."); |
||||
|
||||
if (environment != null){ |
||||
initTextureSubmitStructs(); |
||||
setupCamerasAndViews(); |
||||
setupVRScene(); |
||||
moveScreenProcessingToEyes(); |
||||
if( environment.hasTraditionalGUIOverlay() ) { |
||||
|
||||
environment.getVRMouseManager().initialize(); |
||||
|
||||
// update the pose to position the gui correctly on start
|
||||
update(0f); |
||||
environment.getVRGUIManager().positionGui(); |
||||
} |
||||
|
||||
if (environment.getApplication() != null){ |
||||
// if we are OSVR, our primary mirror window needs to be the same size as the render manager's output...
|
||||
if( environment.getVRHardware() instanceof OSVR ) { |
||||
int origWidth = GraphicsEnvironment.getLocalGraphicsEnvironment().getDefaultScreenDevice().getDisplayMode().getWidth(); |
||||
int origHeight = GraphicsEnvironment.getLocalGraphicsEnvironment().getDefaultScreenDevice().getDisplayMode().getHeight(); |
||||
long window = ((LwjglWindow)environment.getApplication().getContext()).getWindowHandle(); |
||||
Vector2f windowSize = new Vector2f(); |
||||
((OSVR)environment.getVRHardware()).getRenderSize(windowSize); |
||||
windowSize.x = Math.max(windowSize.x * 2f, leftCamera.getWidth()); |
||||
org.lwjgl.glfw.GLFW.glfwSetWindowSize(window, (int)windowSize.x, (int)windowSize.y); |
||||
environment.getApplication().getContext().getSettings().setResolution((int)windowSize.x, (int)windowSize.y); |
||||
|
||||
if (environment.getApplication().getRenderManager() != null) { |
||||
environment.getApplication().getRenderManager().notifyReshape((int)windowSize.x, (int)windowSize.y); |
||||
} |
||||
|
||||
org.lwjgl.glfw.GLFW.glfwSetWindowPos(window, origWidth - (int)windowSize.x, 32); |
||||
|
||||
org.lwjgl.glfw.GLFW.glfwFocusWindow(window); |
||||
|
||||
org.lwjgl.glfw.GLFW.glfwSetCursorPos(window, origWidth / 2.0, origHeight / 2.0); |
||||
|
||||
logger.config("Initialized VR view manager [SUCCESS]"); |
||||
} else { |
||||
throw new IllegalStateException("Underlying VR hardware should be "+OSVR.class.getSimpleName()); |
||||
} |
||||
} else { |
||||
throw new IllegalStateException("This VR environment is not attached to any application."); |
||||
} |
||||
} else { |
||||
throw new IllegalStateException("This VR view manager is not attached to any VR environment."); |
||||
} |
||||
|
||||
|
||||
} |
||||
|
||||
/** |
||||
* Prepare the size of the given {@link Camera camera} to adapt it to the underlying rendering context. |
||||
* @param cam the {@link Camera camera} to prepare. |
||||
* @param xMult the camera width multiplier. |
||||
*/ |
||||
private void prepareCameraSize(Camera cam, float xMult) { |
||||
|
||||
if (environment != null){ |
||||
|
||||
if (environment.getApplication() != null){ |
||||
|
||||
} else { |
||||
throw new IllegalStateException("This VR environment is not attached to any application."); |
||||
} |
||||
|
||||
Vector2f size = new Vector2f(); |
||||
VRAPI vrhmd = environment.getVRHardware(); |
||||
|
||||
if( vrhmd == null ) { |
||||
size.x = 1280f; |
||||
size.y = 720f; |
||||
} else { |
||||
vrhmd.getRenderSize(size); |
||||
} |
||||
|
||||
if( size.x < environment.getApplication().getContext().getSettings().getWidth() ) { |
||||
size.x = environment.getApplication().getContext().getSettings().getWidth(); |
||||
} |
||||
if( size.y < environment.getApplication().getContext().getSettings().getHeight() ) { |
||||
size.y = environment.getApplication().getContext().getSettings().getHeight(); |
||||
} |
||||
|
||||
if( environment.isInstanceRendering() ){ |
||||
size.x *= 2f; |
||||
} |
||||
|
||||
// other adjustments
|
||||
size.x *= xMult; |
||||
size.x *= resMult; |
||||
size.y *= resMult; |
||||
|
||||
if( cam.getWidth() != size.x || cam.getHeight() != size.y ){ |
||||
cam.resize((int)size.x, (int)size.y, false); |
||||
} |
||||
} else { |
||||
throw new IllegalStateException("This VR view manager is not attached to any VR environment."); |
||||
} |
||||
} |
||||
|
||||
/** |
||||
* Replaces rootNode as the main cameras scene with the distortion mesh |
||||
*/ |
||||
private void setupVRScene(){ |
||||
|
||||
if (environment != null){ |
||||
if (environment.getApplication() != null){ |
||||
// no special scene to setup if we are doing instancing
|
||||
if( environment.isInstanceRendering() ) { |
||||
// distortion has to be done with compositor here... we want only one pass on our end!
|
||||
if( environment.getApplication().getContext().getSettings().isSwapBuffers() ) { |
||||
setupMirrorBuffers(environment.getCamera(), dualEyeTex, true); |
||||
} |
||||
return; |
||||
} |
||||
|
||||
leftEyeTexture = (Texture2D) leftViewport.getOutputFrameBuffer().getColorBuffer().getTexture(); |
||||
rightEyeTexture = (Texture2D)rightViewport.getOutputFrameBuffer().getColorBuffer().getTexture(); |
||||
leftEyeDepth = (Texture2D) leftViewport.getOutputFrameBuffer().getDepthBuffer().getTexture(); |
||||
rightEyeDepth = (Texture2D)rightViewport.getOutputFrameBuffer().getDepthBuffer().getTexture(); |
||||
|
||||
// main viewport is either going to be a distortion scene or nothing
|
||||
// mirroring is handled by copying framebuffers
|
||||
Iterator<Spatial> spatialIter = environment.getApplication().getViewPort().getScenes().iterator(); |
||||
while(spatialIter.hasNext()){ |
||||
environment.getApplication().getViewPort().detachScene(spatialIter.next()); |
||||
} |
||||
|
||||
spatialIter = environment.getApplication().getGuiViewPort().getScenes().iterator(); |
||||
while(spatialIter.hasNext()){ |
||||
environment.getApplication().getGuiViewPort().detachScene(spatialIter.next()); |
||||
} |
||||
|
||||
// only setup distortion scene if compositor isn't running (or using custom mesh distortion option)
|
||||
if( environment.getVRHardware().getCompositor() == null ) { |
||||
Node distortionScene = new Node(); |
||||
Material leftMat = new Material(environment.getApplication().getAssetManager(), "Common/MatDefs/VR/OpenVR.j3md"); |
||||
leftMat.setTexture("Texture", leftEyeTexture); |
||||
Geometry leftEye = new Geometry("box", setupDistortionMesh(JOpenVRLibrary.EVREye.EVREye_Eye_Left, environment.getVRHardware())); |
||||
leftEye.setMaterial(leftMat); |
||||
distortionScene.attachChild(leftEye); |
||||
|
||||
Material rightMat = new Material(environment.getApplication().getAssetManager(), "Common/MatDefs/VR/OpenVR.j3md"); |
||||
rightMat.setTexture("Texture", rightEyeTexture); |
||||
Geometry rightEye = new Geometry("box", setupDistortionMesh(JOpenVRLibrary.EVREye.EVREye_Eye_Right, environment.getVRHardware())); |
||||
rightEye.setMaterial(rightMat); |
||||
distortionScene.attachChild(rightEye); |
||||
|
||||
distortionScene.updateGeometricState(); |
||||
|
||||
environment.getApplication().getViewPort().attachScene(distortionScene); |
||||
|
||||
//if( useCustomDistortion ) setupFinalFullTexture(app.getViewPort().getCamera());
|
||||
} |
||||
|
||||
if( environment.getApplication().getContext().getSettings().isSwapBuffers() ) { |
||||
setupMirrorBuffers(environment.getCamera(), leftEyeTexture, false); |
||||
} |
||||
} else { |
||||
throw new IllegalStateException("This VR environment is not attached to any application."); |
||||
} |
||||
} else { |
||||
throw new IllegalStateException("This VR view manager is not attached to any VR environment."); |
||||
} |
||||
|
||||
|
||||
} |
||||
|
||||
/** |
||||
* Update the VR view manager. |
||||
* This method is called by the attached {@link VRApplication VR application} and should not be called manually. |
||||
* @param tpf the time per frame. |
||||
*/ |
||||
public void update(float tpf) { |
||||
|
||||
if (environment != null){ |
||||
// grab the observer
|
||||
Object obs = environment.getObserver(); |
||||
Quaternion objRot; |
||||
Vector3f objPos; |
||||
if( obs instanceof Camera ) { |
||||
objRot = ((Camera)obs).getRotation(); |
||||
objPos = ((Camera)obs).getLocation(); |
||||
} else { |
||||
objRot = ((Spatial)obs).getWorldRotation(); |
||||
objPos = ((Spatial)obs).getWorldTranslation(); |
||||
} |
||||
// grab the hardware handle
|
||||
VRAPI dev = environment.getVRHardware(); |
||||
if( dev != null ) { |
||||
// update the HMD's position & orientation
|
||||
dev.updatePose(); |
||||
dev.getPositionAndOrientation(hmdPos, hmdRot); |
||||
if( obs != null ) { |
||||
// update hmdPos based on obs rotation
|
||||
finalRotation.set(objRot); |
||||
finalRotation.mult(hmdPos, hmdPos); |
||||
finalRotation.multLocal(hmdRot); |
||||
} |
||||
|
||||
finalizeCamera(dev.getHMDVectorPoseLeftEye(), objPos, leftCamera); |
||||
finalizeCamera(dev.getHMDVectorPoseRightEye(), objPos, rightCamera); |
||||
} else { |
||||
leftCamera.setFrame(objPos, objRot); |
||||
rightCamera.setFrame(objPos, objRot); |
||||
} |
||||
|
||||
if( environment.hasTraditionalGUIOverlay() ) { |
||||
// update the mouse?
|
||||
environment.getVRMouseManager().update(tpf); |
||||
|
||||
// update GUI position?
|
||||
if( environment.getVRGUIManager().wantsReposition || environment.getVRGUIManager().getPositioningMode() != VRGUIPositioningMode.MANUAL ) { |
||||
environment.getVRGUIManager().positionGuiNow(tpf); |
||||
environment.getVRGUIManager().updateGuiQuadGeometricState(); |
||||
} |
||||
} |
||||
} else { |
||||
throw new IllegalStateException("This VR view manager is not attached to any VR environment."); |
||||
} |
||||
} |
||||
|
||||
/** |
||||
* Place the camera within the scene. |
||||
* @param eyePos the eye position. |
||||
* @param obsPosition the observer position. |
||||
* @param cam the camera to place. |
||||
*/ |
||||
private void finalizeCamera(Vector3f eyePos, Vector3f obsPosition, Camera cam) { |
||||
finalRotation.mult(eyePos, finalPosition); |
||||
finalPosition.addLocal(hmdPos); |
||||
if( obsPosition != null ){ |
||||
finalPosition.addLocal(obsPosition); |
||||
} |
||||
finalPosition.y += heightAdjustment; |
||||
cam.setFrame(finalPosition, finalRotation); |
||||
} |
||||
|
||||
/** |
||||
* Handles moving filters from the main view to each eye |
||||
*/ |
||||
public void moveScreenProcessingToEyes() { |
||||
if( rightViewport == null ){ |
||||
return; |
||||
} |
||||
|
||||
if (environment != null){ |
||||
if (environment.getApplication() != null){ |
||||
|
||||
syncScreenProcessing(environment.getApplication().getViewPort()); |
||||
environment.getApplication().getViewPort().clearProcessors(); |
||||
} else { |
||||
throw new IllegalStateException("This VR environment is not attached to any application."); |
||||
} |
||||
} else { |
||||
throw new IllegalStateException("This VR view manager is not attached to any VR environment."); |
||||
} |
||||
} |
||||
|
||||
/** |
||||
* Sets the two views to use the list of {@link SceneProcessor processors}. |
||||
* @param sourceViewport the {@link ViewPort viewport} that contains the processors to use. |
||||
*/ |
||||
public void syncScreenProcessing(ViewPort sourceViewport) { |
||||
if( rightViewport == null ){ |
||||
return; |
||||
} |
||||
|
||||
if (environment != null){ |
||||
if (environment.getApplication() != null){ |
||||
// setup post processing filters
|
||||
if( rightPostProcessor == null ) { |
||||
rightPostProcessor = new FilterPostProcessor(environment.getApplication().getAssetManager()); |
||||
leftPostProcessor = new FilterPostProcessor(environment.getApplication().getAssetManager()); |
||||
} |
||||
// clear out all filters & processors, to start from scratch
|
||||
rightPostProcessor.removeAllFilters(); |
||||
leftPostProcessor.removeAllFilters(); |
||||
leftViewport.clearProcessors(); |
||||
rightViewport.clearProcessors(); |
||||
// if we have no processors to sync, don't add the FilterPostProcessor
|
||||
if( sourceViewport.getProcessors().isEmpty() ) return; |
||||
// add post processors we just made, which are empty
|
||||
leftViewport.addProcessor(leftPostProcessor); |
||||
rightViewport.addProcessor(rightPostProcessor); |
||||
// go through all of the filters in the processors list
|
||||
// add them to the left viewport processor & clone them to the right
|
||||
for(SceneProcessor sceneProcessor : sourceViewport.getProcessors()) { |
||||
if (sceneProcessor instanceof FilterPostProcessor) { |
||||
for(Filter f : ((FilterPostProcessor)sceneProcessor).getFilterList() ) { |
||||
if( f instanceof TranslucentBucketFilter ) { |
||||
// just remove this filter, we will add it at the end manually
|
||||
((FilterPostProcessor)sceneProcessor).removeFilter(f); |
||||
} else { |
||||
leftPostProcessor.addFilter(f); |
||||
// clone to the right
|
||||
Filter f2; |
||||
if(f instanceof FogFilter){ |
||||
f2 = FilterUtil.cloneFogFilter((FogFilter)f); |
||||
} else if (f instanceof CartoonSSAO ) { |
||||
f2 = new CartoonSSAO((CartoonSSAO)f); |
||||
} else if (f instanceof SSAOFilter){ |
||||
f2 = FilterUtil.cloneSSAOFilter((SSAOFilter)f); |
||||
} else if (f instanceof DirectionalLightShadowFilter){ |
||||
f2 = FilterUtil.cloneDirectionalLightShadowFilter(environment.getApplication().getAssetManager(), (DirectionalLightShadowFilter)f); |
||||
} else { |
||||
f2 = f; // dof, bloom, lightscattering etc.
|
||||
} |
||||
rightPostProcessor.addFilter(f2); |
||||
} |
||||
} |
||||
} else if (sceneProcessor instanceof VRDirectionalLightShadowRenderer) { |
||||
// shadow processing
|
||||
// TODO: make right shadow processor use same left shadow maps for performance
|
||||
VRDirectionalLightShadowRenderer dlsr = (VRDirectionalLightShadowRenderer) sceneProcessor; |
||||
VRDirectionalLightShadowRenderer dlsrRight = dlsr.clone(); |
||||
dlsrRight.setLight(dlsr.getLight()); |
||||
rightViewport.getProcessors().add(0, dlsrRight); |
||||
leftViewport.getProcessors().add(0, sceneProcessor); |
||||
} |
||||
} |
||||
// make sure each has a translucent filter renderer
|
||||
leftPostProcessor.addFilter(new TranslucentBucketFilter()); |
||||
rightPostProcessor.addFilter(new TranslucentBucketFilter()); |
||||
} else { |
||||
throw new IllegalStateException("This VR environment is not attached to any application."); |
||||
} |
||||
} else { |
||||
throw new IllegalStateException("This VR view manager is not attached to any VR environment."); |
||||
} |
||||
} |
||||
|
||||
private void setupCamerasAndViews() { |
||||
|
||||
if (environment != null){ |
||||
if (environment.getApplication() != null){ |
||||
// get desired frustrum from original camera
|
||||
Camera origCam = environment.getCamera(); |
||||
float fFar = origCam.getFrustumFar(); |
||||
float fNear = origCam.getFrustumNear(); |
||||
|
||||
// if we are using OSVR get the eye info here
|
||||
if( environment.getVRHardware() instanceof OSVR ) { |
||||
((OSVR)environment.getVRHardware()).getEyeInfo(); |
||||
} |
||||
|
||||
// restore frustrum on distortion scene cam, if needed
|
||||
if( environment.isInstanceRendering() ) { |
||||
leftCamera = origCam; |
||||
} else if( environment.compositorAllowed() == false ) { |
||||
origCam.setFrustumFar(100f); |
||||
origCam.setFrustumNear(1f); |
||||
leftCamera = origCam.clone(); |
||||
prepareCameraSize(origCam, 2f); |
||||
} else { |
||||
leftCamera = origCam.clone(); |
||||
} |
||||
|
||||
leftCamera.setFrustumPerspective(environment.getDefaultFOV(), environment.getDefaultAspect(), fNear, fFar); |
||||
|
||||
prepareCameraSize(leftCamera, 1f); |
||||
if( environment.getVRHardware() != null ) leftCamera.setProjectionMatrix(environment.getVRHardware().getHMDMatrixProjectionLeftEye(leftCamera)); |
||||
//org.lwjgl.opengl.GL11.glEnable(org.lwjgl.opengl.GL30.GL_FRAMEBUFFER_SRGB);
|
||||
|
||||
if( !environment.isInstanceRendering()) { |
||||
leftViewport = setupViewBuffers(leftCamera, LEFT_VIEW_NAME); |
||||
rightCamera = leftCamera.clone(); |
||||
if( environment.getVRHardware() != null ){ |
||||
rightCamera.setProjectionMatrix(environment.getVRHardware().getHMDMatrixProjectionRightEye(rightCamera)); |
||||
} |
||||
rightViewport = setupViewBuffers(rightCamera, RIGHT_VIEW_NAME); |
||||
} else { |
||||
|
||||
System.err.println("[VRViewManager] THIS CODE NEED CHANGES !!!"); |
||||
leftViewport = environment.getApplication().getViewPort(); |
||||
//leftViewport.attachScene(app.getRootNode());
|
||||
rightCamera = leftCamera.clone(); |
||||
if( environment.getVRHardware() != null ){ |
||||
rightCamera.setProjectionMatrix(environment.getVRHardware().getHMDMatrixProjectionRightEye(rightCamera)); |
||||
} |
||||
|
||||
org.lwjgl.opengl.GL11.glEnable(org.lwjgl.opengl.GL30.GL_CLIP_DISTANCE0); |
||||
|
||||
//FIXME: [jme-vr] Fix with JMonkey next release
|
||||
//RenderManager._VRInstancing_RightCamProjection = camRight.getViewProjectionMatrix();
|
||||
setupFinalFullTexture(environment.getApplication().getViewPort().getCamera()); |
||||
} |
||||
|
||||
// setup gui
|
||||
environment.getVRGUIManager().setupGui(leftCamera, rightCamera, leftViewport, rightViewport); |
||||
|
||||
if( environment.getVRHardware() != null ) { |
||||
// call these to cache the results internally
|
||||
environment.getVRHardware().getHMDMatrixPoseLeftEye(); |
||||
environment.getVRHardware().getHMDMatrixPoseRightEye(); |
||||
} |
||||
} else { |
||||
throw new IllegalStateException("This VR environment is not attached to any application."); |
||||
} |
||||
} else { |
||||
throw new IllegalStateException("This VR view manager is not attached to any VR environment."); |
||||
} |
||||
} |
||||
|
||||
private ViewPort setupMirrorBuffers(Camera cam, Texture tex, boolean expand) { |
||||
|
||||
if (environment != null){ |
||||
if (environment.getApplication() != null){ |
||||
Camera clonecam = cam.clone(); |
||||
ViewPort viewPort = environment.getApplication().getRenderManager().createPostView("MirrorView", clonecam); |
||||
clonecam.setParallelProjection(true); |
||||
viewPort.setClearFlags(true, true, true); |
||||
viewPort.setBackgroundColor(ColorRGBA.Black); |
||||
Picture pic = new Picture("fullscene"); |
||||
pic.setLocalTranslation(-0.75f, -0.5f, 0f); |
||||
if( expand ) { |
||||
pic.setLocalScale(3f, 1f, 1f); |
||||
} else { |
||||
pic.setLocalScale(1.5f, 1f, 1f); |
||||
} |
||||
pic.setQueueBucket(Bucket.Opaque); |
||||
pic.setTexture(environment.getApplication().getAssetManager(), (Texture2D)tex, false); |
||||
viewPort.attachScene(pic); |
||||
viewPort.setOutputFrameBuffer(null); |
||||
|
||||
pic.updateGeometricState(); |
||||
|
||||
return viewPort; |
||||
} else { |
||||
throw new IllegalStateException("This VR environment is not attached to any application."); |
||||
} |
||||
} else { |
||||
throw new IllegalStateException("This VR view manager is not attached to any VR environment."); |
||||
} |
||||
} |
||||
|
||||
private void setupFinalFullTexture(Camera cam) { |
||||
|
||||
if (environment != null){ |
||||
if (environment.getApplication() != null){ |
||||
// create offscreen framebuffer
|
||||
FrameBuffer out = new FrameBuffer(cam.getWidth(), cam.getHeight(), 1); |
||||
//offBuffer.setSrgb(true);
|
||||
|
||||
//setup framebuffer's texture
|
||||
dualEyeTex = new Texture2D(cam.getWidth(), cam.getHeight(), Image.Format.RGBA8); |
||||
dualEyeTex.setMinFilter(Texture.MinFilter.BilinearNoMipMaps); |
||||
dualEyeTex.setMagFilter(Texture.MagFilter.Bilinear); |
||||
|
||||
logger.config("Dual eye texture "+dualEyeTex.getName()+" ("+dualEyeTex.getImage().getId()+")"); |
||||
logger.config(" Type: "+dualEyeTex.getType()); |
||||
logger.config(" Size: "+dualEyeTex.getImage().getWidth()+"x"+dualEyeTex.getImage().getHeight()); |
||||
logger.config(" Image depth: "+dualEyeTex.getImage().getDepth()); |
||||
logger.config(" Image format: "+dualEyeTex.getImage().getFormat()); |
||||
logger.config(" Image color space: "+dualEyeTex.getImage().getColorSpace()); |
||||
|
||||
//setup framebuffer to use texture
|
||||
out.setDepthBuffer(Image.Format.Depth); |
||||
out.setColorTexture(dualEyeTex); |
||||
|
||||
ViewPort viewPort = environment.getApplication().getViewPort(); |
||||
viewPort.setClearFlags(true, true, true); |
||||
viewPort.setBackgroundColor(ColorRGBA.Black); |
||||
viewPort.setOutputFrameBuffer(out); |
||||
} else { |
||||
throw new IllegalStateException("This VR environment is not attached to any application."); |
||||
} |
||||
} else { |
||||
throw new IllegalStateException("This VR view manager is not attached to any VR environment."); |
||||
} |
||||
} |
||||
|
||||
private ViewPort setupViewBuffers(Camera cam, String viewName){ |
||||
|
||||
if (environment != null){ |
||||
if (environment.getApplication() != null){ |
||||
// create offscreen framebuffer
|
||||
FrameBuffer offBufferLeft = new FrameBuffer(cam.getWidth(), cam.getHeight(), 1); |
||||
//offBufferLeft.setSrgb(true);
|
||||
|
||||
//setup framebuffer's texture
|
||||
Texture2D offTex = new Texture2D(cam.getWidth(), cam.getHeight(), Image.Format.RGBA8); |
||||
offTex.setMinFilter(Texture.MinFilter.BilinearNoMipMaps); |
||||
offTex.setMagFilter(Texture.MagFilter.Bilinear); |
||||
|
||||
//setup framebuffer to use texture
|
||||
offBufferLeft.setDepthBuffer(Image.Format.Depth); |
||||
offBufferLeft.setColorTexture(offTex); |
||||
|
||||
ViewPort viewPort = environment.getApplication().getRenderManager().createPreView(viewName, cam); |
||||
viewPort.setClearFlags(true, true, true); |
||||
viewPort.setBackgroundColor(ColorRGBA.Black); |
||||
|
||||
Iterator<Spatial> spatialIter = environment.getApplication().getViewPort().getScenes().iterator(); |
||||
while(spatialIter.hasNext()){ |
||||
viewPort.attachScene(spatialIter.next()); |
||||
} |
||||
|
||||
//set viewport to render to offscreen framebuffer
|
||||
viewPort.setOutputFrameBuffer(offBufferLeft); |
||||
return viewPort; |
||||
} else { |
||||
throw new IllegalStateException("This VR environment is not attached to any application."); |
||||
} |
||||
} else { |
||||
throw new IllegalStateException("This VR view manager is not attached to any VR environment."); |
||||
} |
||||
} |
||||
|
||||
/** |
||||
* Setup a distortion mesh for the stereo view. |
||||
* @param eye the eye to apply. |
||||
* @param api the underlying VR api |
||||
* @return the distorted mesh. |
||||
*/ |
||||
public static Mesh setupDistortionMesh(int eye, VRAPI api) { |
||||
Mesh distortionMesh = new Mesh(); |
||||
float m_iLensGridSegmentCountH = 43, m_iLensGridSegmentCountV = 43; |
||||
|
||||
float w = 1f / (m_iLensGridSegmentCountH - 1f); |
||||
float h = 1f / (m_iLensGridSegmentCountV - 1f); |
||||
|
||||
float u, v; |
||||
|
||||
float verts[] = new float[(int) (m_iLensGridSegmentCountV * m_iLensGridSegmentCountH) * 3]; |
||||
|
||||
float texcoordR[] = new float[(int) (m_iLensGridSegmentCountV * m_iLensGridSegmentCountH) * 2]; |
||||
float texcoordG[] = new float[(int) (m_iLensGridSegmentCountV * m_iLensGridSegmentCountH) * 2]; |
||||
float texcoordB[] = new float[(int) (m_iLensGridSegmentCountV * m_iLensGridSegmentCountH) * 2]; |
||||
|
||||
int vertPos = 0, coordPos = 0; |
||||
|
||||
float Xoffset = eye == JOpenVRLibrary.EVREye.EVREye_Eye_Left ? -1f : 0; |
||||
for (int y = 0; y < m_iLensGridSegmentCountV; y++) { |
||||
for (int x = 0; x < m_iLensGridSegmentCountH; x++) { |
||||
u = x * w; |
||||
v = 1 - y * h; |
||||
verts[vertPos] = Xoffset + u; // x
|
||||
verts[vertPos + 1] = -1 + 2 * y * h; // y
|
||||
verts[vertPos + 2] = 0f; // z
|
||||
vertPos += 3; |
||||
|
||||
DistortionCoordinates_t dc0 = new DistortionCoordinates_t(); |
||||
if( api.getVRSystem() == null ) { |
||||
// default to no distortion
|
||||
texcoordR[coordPos] = u; |
||||
texcoordR[coordPos + 1] = 1 - v; |
||||
texcoordG[coordPos] = u; |
||||
texcoordG[coordPos + 1] = 1 - v; |
||||
texcoordB[coordPos] = u; |
||||
texcoordB[coordPos + 1] = 1 - v; |
||||
} else { |
||||
((VR_IVRSystem_FnTable)api.getVRSystem()).ComputeDistortion.apply(eye, u, v, dc0); |
||||
|
||||
texcoordR[coordPos] = dc0.rfRed[0]; |
||||
texcoordR[coordPos + 1] = 1 - dc0.rfRed[1]; |
||||
texcoordG[coordPos] = dc0.rfGreen[0]; |
||||
texcoordG[coordPos + 1] = 1 - dc0.rfGreen[1]; |
||||
texcoordB[coordPos] = dc0.rfBlue[0]; |
||||
texcoordB[coordPos + 1] = 1 - dc0.rfBlue[1]; |
||||
} |
||||
|
||||
coordPos += 2; |
||||
} |
||||
} |
||||
|
||||
// have UV coordinates & positions, now to setup indices
|
||||
|
||||
int[] indices = new int[(int) ((m_iLensGridSegmentCountV - 1) * (m_iLensGridSegmentCountH - 1)) * 6]; |
||||
int indexPos = 0; |
||||
int a, b, c, d; |
||||
|
||||
int offset = 0; |
||||
for (int y = 0; y < m_iLensGridSegmentCountV - 1; y++) { |
||||
for (int x = 0; x < m_iLensGridSegmentCountH - 1; x++) { |
||||
a = (int) (m_iLensGridSegmentCountH * y + x + offset); |
||||
b = (int) (m_iLensGridSegmentCountH * y + x + 1 + offset); |
||||
c = (int) ((y + 1) * m_iLensGridSegmentCountH + x + 1 + offset); |
||||
d = (int) ((y + 1) * m_iLensGridSegmentCountH + x + offset); |
||||
|
||||
indices[indexPos] = a; |
||||
indices[indexPos + 1] = b; |
||||
indices[indexPos + 2] = c; |
||||
|
||||
indices[indexPos + 3] = a; |
||||
indices[indexPos + 4] = c; |
||||
indices[indexPos + 5] = d; |
||||
|
||||
indexPos += 6; |
||||
} |
||||
} |
||||
|
||||
// OK, create the mesh
|
||||
distortionMesh.setBuffer(VertexBuffer.Type.Position, 3, verts); |
||||
distortionMesh.setBuffer(VertexBuffer.Type.Index, 1, indices); |
||||
distortionMesh.setBuffer(VertexBuffer.Type.TexCoord, 2, texcoordR); |
||||
distortionMesh.setBuffer(VertexBuffer.Type.TexCoord2, 2, texcoordG); |
||||
distortionMesh.setBuffer(VertexBuffer.Type.TexCoord3, 2, texcoordB); |
||||
distortionMesh.setStatic(); |
||||
return distortionMesh; |
||||
} |
||||
} |
@ -0,0 +1,732 @@ |
||||
/* |
||||
* To change this template, choose Tools | Templates |
||||
* and open the template in the editor. |
||||
*/ |
||||
package com.jme3.util; |
||||
|
||||
import com.jme3.app.VREnvironment; |
||||
import com.jme3.input.vr.OpenVR; |
||||
import com.jme3.input.vr.VRAPI; |
||||
import com.jme3.material.Material; |
||||
import com.jme3.math.ColorRGBA; |
||||
import com.jme3.math.Quaternion; |
||||
import com.jme3.math.Vector2f; |
||||
import com.jme3.math.Vector3f; |
||||
import com.jme3.renderer.Camera; |
||||
import com.jme3.renderer.ViewPort; |
||||
import com.jme3.renderer.queue.RenderQueue.Bucket; |
||||
import com.jme3.scene.Geometry; |
||||
import com.jme3.scene.Mesh; |
||||
import com.jme3.scene.Node; |
||||
import com.jme3.scene.Spatial; |
||||
import com.jme3.scene.VertexBuffer; |
||||
import com.jme3.system.jopenvr.DistortionCoordinates_t; |
||||
import com.jme3.system.jopenvr.JOpenVRLibrary; |
||||
import com.jme3.system.jopenvr.OpenVRUtil; |
||||
import com.jme3.system.jopenvr.Texture_t; |
||||
import com.jme3.system.jopenvr.VRTextureBounds_t; |
||||
import com.jme3.system.jopenvr.VR_IVRSystem_FnTable; |
||||
import com.jme3.texture.FrameBuffer; |
||||
import com.jme3.texture.Image; |
||||
import com.jme3.texture.Texture; |
||||
import com.jme3.texture.Texture2D; |
||||
import com.jme3.ui.Picture; |
||||
|
||||
import java.util.Iterator; |
||||
import java.util.logging.Logger; |
||||
|
||||
/** |
||||
* A VR view manager based on OpenVR. This class enable to submit 3D views to the VR compositor. |
||||
* @author reden - phr00t - https://github.com/phr00t
|
||||
* @author Julien Seinturier - (c) 2016 - JOrigin project - <a href="http://www.jorigin.org">http:/www.jorigin.org</a> |
||||
*/ |
||||
public class VRViewManagerOpenVR extends AbstractVRViewManager { |
||||
|
||||
private static final Logger logger = Logger.getLogger(VRViewManagerOpenVR.class.getName()); |
||||
|
||||
// OpenVR values
|
||||
private VRTextureBounds_t leftTextureBounds; |
||||
private Texture_t leftTextureType; |
||||
|
||||
private VRTextureBounds_t rightTextureBounds; |
||||
private Texture_t rightTextureType; |
||||
|
||||
private Texture2D dualEyeTex; |
||||
|
||||
//final & temp values for camera calculations
|
||||
private final Vector3f finalPosition = new Vector3f(); |
||||
private final Quaternion finalRotation = new Quaternion(); |
||||
private final Vector3f hmdPos = new Vector3f(); |
||||
private final Quaternion hmdRot = new Quaternion(); |
||||
|
||||
/** |
||||
* Create a new VR view manager attached to the given {@link VREnvironment VR environment}. |
||||
* @param environment the {@link VREnvironment VR environment} to which this view manager is attached. |
||||
*/ |
||||
public VRViewManagerOpenVR(VREnvironment environment){ |
||||
this.environment = environment; |
||||
} |
||||
|
||||
/** |
||||
* Get the identifier of the left eye texture. |
||||
* @return the identifier of the left eye texture. |
||||
* @see #getRightTexId() |
||||
* @see #getFullTexId() |
||||
*/ |
||||
protected int getLeftTexId() { |
||||
return (int)getLeftTexture().getImage().getId(); |
||||
} |
||||
|
||||
/** |
||||
* Get the identifier of the right eye texture. |
||||
* @return the identifier of the right eye texture. |
||||
* @see #getLeftTexId() |
||||
* @see #getFullTexId() |
||||
*/ |
||||
protected int getRightTexId() { |
||||
return (int)getRightTexture().getImage().getId(); |
||||
} |
||||
|
||||
/** |
||||
* Get the identifier of the full (dual eye) texture. |
||||
* @return the identifier of the full (dual eye) texture. |
||||
* @see #getLeftTexId() |
||||
* @see #getRightTexId() |
||||
*/ |
||||
private int getFullTexId() { |
||||
return (int)dualEyeTex.getImage().getId(); |
||||
} |
||||
|
||||
/** |
||||
* Initialize the system binds of the textures. |
||||
*/ |
||||
private void initTextureSubmitStructs() { |
||||
leftTextureType = new Texture_t(); |
||||
rightTextureType = new Texture_t(); |
||||
|
||||
if (environment != null){ |
||||
if( environment.getVRHardware() instanceof OpenVR ) { |
||||
leftTextureBounds = new VRTextureBounds_t(); |
||||
rightTextureBounds = new VRTextureBounds_t(); |
||||
// left eye
|
||||
leftTextureBounds.uMax = 0.5f; |
||||
leftTextureBounds.uMin = 0f; |
||||
leftTextureBounds.vMax = 1f; |
||||
leftTextureBounds.vMin = 0f; |
||||
leftTextureBounds.setAutoSynch(false); |
||||
leftTextureBounds.setAutoRead(false); |
||||
leftTextureBounds.setAutoWrite(false); |
||||
leftTextureBounds.write(); |
||||
// right eye
|
||||
rightTextureBounds.uMax = 1f; |
||||
rightTextureBounds.uMin = 0.5f; |
||||
rightTextureBounds.vMax = 1f; |
||||
rightTextureBounds.vMin = 0f; |
||||
rightTextureBounds.setAutoSynch(false); |
||||
rightTextureBounds.setAutoRead(false); |
||||
rightTextureBounds.setAutoWrite(false); |
||||
rightTextureBounds.write(); |
||||
// texture type
|
||||
leftTextureType.eColorSpace = JOpenVRLibrary.EColorSpace.EColorSpace_ColorSpace_Gamma; |
||||
leftTextureType.eType = JOpenVRLibrary.ETextureType.ETextureType_TextureType_OpenGL; |
||||
leftTextureType.setAutoSynch(false); |
||||
leftTextureType.setAutoRead(false); |
||||
leftTextureType.setAutoWrite(false); |
||||
leftTextureType.handle = -1; |
||||
rightTextureType.eColorSpace = JOpenVRLibrary.EColorSpace.EColorSpace_ColorSpace_Gamma; |
||||
rightTextureType.eType = JOpenVRLibrary.ETextureType.ETextureType_TextureType_OpenGL; |
||||
rightTextureType.setAutoSynch(false); |
||||
rightTextureType.setAutoRead(false); |
||||
rightTextureType.setAutoWrite(false); |
||||
rightTextureType.handle = -1; |
||||
|
||||
|
||||
logger.config("Init eyes native texture binds"); |
||||
logger.config(" Left eye texture"); |
||||
logger.config(" address: "+leftTextureType.getPointer()); |
||||
logger.config(" size: "+leftTextureType.size()+" bytes"); |
||||
logger.config(" color space: "+OpenVRUtil.getEColorSpaceString(leftTextureType.eColorSpace)); |
||||
logger.config(" type: "+OpenVRUtil.getETextureTypeString(leftTextureType.eType)); |
||||
logger.config(" auto read: "+leftTextureType.getAutoRead()); |
||||
logger.config(" auto write: "+leftTextureType.getAutoWrite()); |
||||
logger.config(" handle address: "+leftTextureType.handle); |
||||
logger.config(" handle value: "+leftTextureType.handle); |
||||
logger.config(""); |
||||
logger.config(" Right eye texture"); |
||||
logger.config(" address: "+rightTextureType.getPointer()); |
||||
logger.config(" size: "+rightTextureType.size()+" bytes"); |
||||
logger.config(" color space: "+OpenVRUtil.getEColorSpaceString(rightTextureType.eColorSpace)); |
||||
logger.config(" type: "+OpenVRUtil.getETextureTypeString(rightTextureType.eType)); |
||||
logger.config(" auto read: "+rightTextureType.getAutoRead()); |
||||
logger.config(" auto write: "+rightTextureType.getAutoWrite()); |
||||
logger.config(" handle address: "+rightTextureType.handle); |
||||
logger.config(" handle value: "+rightTextureType.handle); |
||||
} |
||||
} else { |
||||
throw new IllegalStateException("This VR view manager is not attached to any VR environment."); |
||||
} |
||||
} |
||||
|
||||
@Override |
||||
public void postRender() { |
||||
|
||||
if (environment != null){ |
||||
if( environment.isInVR() ) { |
||||
VRAPI api = environment.getVRHardware(); |
||||
if( api.getCompositor() != null ) { |
||||
// using the compositor...
|
||||
int errl = 0, errr = 0; |
||||
if( environment.isInstanceRendering() ) { |
||||
if( leftTextureType.handle == -1 || leftTextureType.handle != getFullTexId() ) { |
||||
leftTextureType.handle = getFullTexId(); |
||||
if( leftTextureType.handle != -1 ) { |
||||
leftTextureType.write(); |
||||
} |
||||
} else { |
||||
if( api instanceof OpenVR ) { |
||||
int submitFlag = JOpenVRLibrary.EVRSubmitFlags.EVRSubmitFlags_Submit_Default; |
||||
errr = ((OpenVR)api).getCompositor().Submit.apply(JOpenVRLibrary.EVREye.EVREye_Eye_Right, leftTextureType, rightTextureBounds, submitFlag); |
||||
errl = ((OpenVR)api).getCompositor().Submit.apply(JOpenVRLibrary.EVREye.EVREye_Eye_Left, leftTextureType, leftTextureBounds, submitFlag); |
||||
} |
||||
} |
||||
} else if( leftTextureType.handle == -1 || rightTextureType.handle == -1 || |
||||
leftTextureType.handle != getLeftTexId() || rightTextureType.handle != getRightTexId() ) { |
||||
leftTextureType.handle = getLeftTexId(); |
||||
if( leftTextureType.handle != -1 ) { |
||||
logger.fine("Writing Left texture to native memory at " + leftTextureType.getPointer()); |
||||
leftTextureType.write(); |
||||
} |
||||
rightTextureType.handle = getRightTexId(); |
||||
if( rightTextureType.handle != -1 ) { |
||||
logger.fine("Writing Right texture to native memory at " + leftTextureType.getPointer()); |
||||
rightTextureType.write(); |
||||
} |
||||
} else { |
||||
if( api instanceof OpenVR ) { |
||||
errl = ((OpenVR)api).getCompositor().Submit.apply(JOpenVRLibrary.EVREye.EVREye_Eye_Left, leftTextureType, null, |
||||
JOpenVRLibrary.EVRSubmitFlags.EVRSubmitFlags_Submit_Default); |
||||
errr = ((OpenVR)api).getCompositor().Submit.apply(JOpenVRLibrary.EVREye.EVREye_Eye_Right, rightTextureType, null, |
||||
JOpenVRLibrary.EVRSubmitFlags.EVRSubmitFlags_Submit_Default); |
||||
} else { |
||||
|
||||
} |
||||
} |
||||
|
||||
if( errl != 0 ){ |
||||
logger.severe("Submit to left compositor error: " + OpenVRUtil.getEVRCompositorErrorString(errl)+" ("+Integer.toString(errl)+")"); |
||||
logger.severe(" Texture color space: "+OpenVRUtil.getEColorSpaceString(leftTextureType.eColorSpace)); |
||||
logger.severe(" Texture type: "+OpenVRUtil.getETextureTypeString(leftTextureType.eType)); |
||||
logger.severe(" Texture handle: "+leftTextureType.handle); |
||||
|
||||
logger.severe(" Left eye texture "+leftEyeTexture.getName()+" ("+leftEyeTexture.getImage().getId()+")"); |
||||
logger.severe(" Type: "+leftEyeTexture.getType()); |
||||
logger.severe(" Size: "+leftEyeTexture.getImage().getWidth()+"x"+leftEyeTexture.getImage().getHeight()); |
||||
logger.severe(" Image depth: "+leftEyeTexture.getImage().getDepth()); |
||||
logger.severe(" Image format: "+leftEyeTexture.getImage().getFormat()); |
||||
logger.severe(" Image color space: "+leftEyeTexture.getImage().getColorSpace()); |
||||
|
||||
} |
||||
|
||||
if( errr != 0 ){ |
||||
logger.severe("Submit to right compositor error: " + OpenVRUtil.getEVRCompositorErrorString(errl)+" ("+Integer.toString(errl)+")"); |
||||
logger.severe(" Texture color space: "+OpenVRUtil.getEColorSpaceString(rightTextureType.eColorSpace)); |
||||
logger.severe(" Texture type: "+OpenVRUtil.getETextureTypeString(rightTextureType.eType)); |
||||
logger.severe(" Texture handle: "+rightTextureType.handle); |
||||
|
||||
logger.severe(" Right eye texture "+rightEyeTexture.getName()+" ("+rightEyeTexture.getImage().getId()+")"); |
||||
logger.severe(" Type: "+rightEyeTexture.getType()); |
||||
logger.severe(" Size: "+rightEyeTexture.getImage().getWidth()+"x"+rightEyeTexture.getImage().getHeight()); |
||||
logger.severe(" Image depth: "+rightEyeTexture.getImage().getDepth()); |
||||
logger.severe(" Image format: "+rightEyeTexture.getImage().getFormat()); |
||||
logger.severe(" Image color space: "+rightEyeTexture.getImage().getColorSpace()); |
||||
} |
||||
} |
||||
} |
||||
} else { |
||||
throw new IllegalStateException("This VR view manager is not attached to any VR environment."); |
||||
} |
||||
|
||||
|
||||
|
||||
} |
||||
|
||||
|
||||
@Override |
||||
public void initialize() { |
||||
|
||||
logger.config("Initializing VR view manager."); |
||||
|
||||
if (environment != null){ |
||||
|
||||
initTextureSubmitStructs(); |
||||
setupCamerasAndViews(); |
||||
setupVRScene(); |
||||
moveScreenProcessingToEyes(); |
||||
|
||||
if( environment.hasTraditionalGUIOverlay() ) { |
||||
|
||||
environment.getVRMouseManager().initialize(); |
||||
|
||||
// update the pose to position the gui correctly on start
|
||||
update(0f); |
||||
environment.getVRGUIManager().positionGui(); |
||||
} |
||||
|
||||
logger.config("Initialized VR view manager [SUCCESS]"); |
||||
|
||||
} else { |
||||
throw new IllegalStateException("This VR view manager is not attached to any VR environment."); |
||||
} |
||||
} |
||||
|
||||
/** |
||||
* Prepare the size of the given {@link Camera camera} to adapt it to the underlying rendering context. |
||||
* @param cam the {@link Camera camera} to prepare. |
||||
* @param xMult the camera width multiplier. |
||||
*/ |
||||
private void prepareCameraSize(Camera cam, float xMult) { |
||||
|
||||
if (environment != null){ |
||||
|
||||
if (environment.getApplication() != null){ |
||||
Vector2f size = new Vector2f(); |
||||
VRAPI vrhmd = environment.getVRHardware(); |
||||
|
||||
if( vrhmd == null ) { |
||||
size.x = 1280f; |
||||
size.y = 720f; |
||||
} else { |
||||
vrhmd.getRenderSize(size); |
||||
} |
||||
|
||||
if( size.x < environment.getApplication().getContext().getSettings().getWidth() ) { |
||||
size.x = environment.getApplication().getContext().getSettings().getWidth(); |
||||
} |
||||
if( size.y < environment.getApplication().getContext().getSettings().getHeight() ) { |
||||
size.y = environment.getApplication().getContext().getSettings().getHeight(); |
||||
} |
||||
|
||||
if( environment.isInstanceRendering() ){ |
||||
size.x *= 2f; |
||||
} |
||||
|
||||
// other adjustments
|
||||
size.x *= xMult; |
||||
size.x *= getResolutionMuliplier(); |
||||
size.y *= getResolutionMuliplier(); |
||||
|
||||
if( cam.getWidth() != size.x || cam.getHeight() != size.y ){ |
||||
cam.resize((int)size.x, (int)size.y, false); |
||||
} |
||||
} else { |
||||
throw new IllegalStateException("This VR environment is not attached to any application."); |
||||
} |
||||
|
||||
} else { |
||||
throw new IllegalStateException("This VR view manager is not attached to any VR environment."); |
||||
} |
||||
|
||||
|
||||
} |
||||
|
||||
/** |
||||
* Replaces rootNode as the main cameras scene with the distortion mesh |
||||
*/ |
||||
private void setupVRScene(){ |
||||
|
||||
|
||||
if (environment != null){ |
||||
if (environment.getApplication() != null){ |
||||
// no special scene to setup if we are doing instancing
|
||||
if( environment.isInstanceRendering() ) { |
||||
// distortion has to be done with compositor here... we want only one pass on our end!
|
||||
if( environment.getApplication().getContext().getSettings().isSwapBuffers() ) { |
||||
setupMirrorBuffers(environment.getCamera(), dualEyeTex, true); |
||||
} |
||||
return; |
||||
} |
||||
|
||||
leftEyeTexture = (Texture2D) getLeftViewport().getOutputFrameBuffer().getColorBuffer().getTexture(); |
||||
rightEyeTexture = (Texture2D)getRightViewport().getOutputFrameBuffer().getColorBuffer().getTexture(); |
||||
leftEyeDepth = (Texture2D) getLeftViewport().getOutputFrameBuffer().getDepthBuffer().getTexture(); |
||||
rightEyeDepth = (Texture2D)getRightViewport().getOutputFrameBuffer().getDepthBuffer().getTexture(); |
||||
|
||||
// main viewport is either going to be a distortion scene or nothing
|
||||
// mirroring is handled by copying framebuffers
|
||||
Iterator<Spatial> spatialIter = environment.getApplication().getViewPort().getScenes().iterator(); |
||||
while(spatialIter.hasNext()){ |
||||
environment.getApplication().getViewPort().detachScene(spatialIter.next()); |
||||
} |
||||
|
||||
spatialIter = environment.getApplication().getGuiViewPort().getScenes().iterator(); |
||||
while(spatialIter.hasNext()){ |
||||
environment.getApplication().getGuiViewPort().detachScene(spatialIter.next()); |
||||
} |
||||
|
||||
// only setup distortion scene if compositor isn't running (or using custom mesh distortion option)
|
||||
if( environment.getVRHardware().getCompositor() == null ) { |
||||
Node distortionScene = new Node(); |
||||
Material leftMat = new Material(environment.getApplication().getAssetManager(), "Common/MatDefs/VR/OpenVR.j3md"); |
||||
leftMat.setTexture("Texture", leftEyeTexture); |
||||
Geometry leftEye = new Geometry("box", setupDistortionMesh(JOpenVRLibrary.EVREye.EVREye_Eye_Left, environment.getVRHardware())); |
||||
leftEye.setMaterial(leftMat); |
||||
distortionScene.attachChild(leftEye); |
||||
|
||||
Material rightMat = new Material(environment.getApplication().getAssetManager(), "Common/MatDefs/VR/OpenVR.j3md"); |
||||
rightMat.setTexture("Texture", rightEyeTexture); |
||||
Geometry rightEye = new Geometry("box", setupDistortionMesh(JOpenVRLibrary.EVREye.EVREye_Eye_Right, environment.getVRHardware())); |
||||
rightEye.setMaterial(rightMat); |
||||
distortionScene.attachChild(rightEye); |
||||
|
||||
distortionScene.updateGeometricState(); |
||||
|
||||
environment.getApplication().getViewPort().attachScene(distortionScene); |
||||
|
||||
//if( useCustomDistortion ) setupFinalFullTexture(app.getViewPort().getCamera());
|
||||
} |
||||
|
||||
if( environment.getApplication().getContext().getSettings().isSwapBuffers() ) { |
||||
setupMirrorBuffers(environment.getCamera(), leftEyeTexture, false); |
||||
} |
||||
} else { |
||||
throw new IllegalStateException("This VR environment is not attached to any application."); |
||||
} |
||||
} else { |
||||
throw new IllegalStateException("This VR view manager is not attached to any VR environment."); |
||||
} |
||||
} |
||||
|
||||
@Override |
||||
public void update(float tpf) { |
||||
|
||||
if (environment != null){ |
||||
// grab the observer
|
||||
Object obs = environment.getObserver(); |
||||
Quaternion objRot; |
||||
Vector3f objPos; |
||||
if( obs instanceof Camera ) { |
||||
objRot = ((Camera)obs).getRotation(); |
||||
objPos = ((Camera)obs).getLocation(); |
||||
} else { |
||||
objRot = ((Spatial)obs).getWorldRotation(); |
||||
objPos = ((Spatial)obs).getWorldTranslation(); |
||||
} |
||||
// grab the hardware handle
|
||||
VRAPI dev = environment.getVRHardware(); |
||||
if( dev != null ) { |
||||
// update the HMD's position & orientation
|
||||
dev.updatePose(); |
||||
dev.getPositionAndOrientation(hmdPos, hmdRot); |
||||
if( obs != null ) { |
||||
// update hmdPos based on obs rotation
|
||||
finalRotation.set(objRot); |
||||
finalRotation.mult(hmdPos, hmdPos); |
||||
finalRotation.multLocal(hmdRot); |
||||
} |
||||
|
||||
finalizeCamera(dev.getHMDVectorPoseLeftEye(), objPos, getLeftCamera()); |
||||
finalizeCamera(dev.getHMDVectorPoseRightEye(), objPos, getRightCamera()); |
||||
} else { |
||||
getLeftCamera().setFrame(objPos, objRot); |
||||
getRightCamera().setFrame(objPos, objRot); |
||||
} |
||||
|
||||
if( environment.hasTraditionalGUIOverlay() ) { |
||||
// update the mouse?
|
||||
environment.getVRMouseManager().update(tpf); |
||||
|
||||
// update GUI position?
|
||||
if( environment.getVRGUIManager().wantsReposition || environment.getVRGUIManager().getPositioningMode() != VRGUIPositioningMode.MANUAL ) { |
||||
environment.getVRGUIManager().positionGuiNow(tpf); |
||||
environment.getVRGUIManager().updateGuiQuadGeometricState(); |
||||
} |
||||
} |
||||
} else { |
||||
throw new IllegalStateException("This VR view manager is not attached to any VR environment."); |
||||
} |
||||
} |
||||
|
||||
/** |
||||
* Place the camera within the scene. |
||||
* @param eyePos the eye position. |
||||
* @param obsPosition the observer position. |
||||
* @param cam the camera to place. |
||||
*/ |
||||
private void finalizeCamera(Vector3f eyePos, Vector3f obsPosition, Camera cam) { |
||||
finalRotation.mult(eyePos, finalPosition); |
||||
finalPosition.addLocal(hmdPos); |
||||
if( obsPosition != null ) finalPosition.addLocal(obsPosition); |
||||
finalPosition.y += getHeightAdjustment(); |
||||
cam.setFrame(finalPosition, finalRotation); |
||||
} |
||||
|
||||
|
||||
private void setupCamerasAndViews() { |
||||
|
||||
if (environment != null){ |
||||
// get desired frustrum from original camera
|
||||
Camera origCam = environment.getCamera(); |
||||
float fFar = origCam.getFrustumFar(); |
||||
float fNear = origCam.getFrustumNear(); |
||||
|
||||
// restore frustrum on distortion scene cam, if needed
|
||||
if( environment.isInstanceRendering() ) { |
||||
leftCamera = origCam; |
||||
} else if( environment.compositorAllowed() == false ) { |
||||
origCam.setFrustumFar(100f); |
||||
origCam.setFrustumNear(1f); |
||||
leftCamera = origCam.clone(); |
||||
prepareCameraSize(origCam, 2f); |
||||
} else { |
||||
leftCamera = origCam.clone(); |
||||
} |
||||
|
||||
getLeftCamera().setFrustumPerspective(environment.getDefaultFOV(), environment.getDefaultAspect(), fNear, fFar); |
||||
|
||||
prepareCameraSize(getLeftCamera(), 1f); |
||||
if( environment.getVRHardware() != null ) { |
||||
getLeftCamera().setProjectionMatrix(environment.getVRHardware().getHMDMatrixProjectionLeftEye(getLeftCamera())); |
||||
} |
||||
//org.lwjgl.opengl.GL11.glEnable(org.lwjgl.opengl.GL30.GL_FRAMEBUFFER_SRGB);
|
||||
|
||||
if( !environment.isInstanceRendering()) { |
||||
leftViewport = setupViewBuffers(getLeftCamera(), LEFT_VIEW_NAME); |
||||
rightCamera = getLeftCamera().clone(); |
||||
if( environment.getVRHardware() != null ){ |
||||
getRightCamera().setProjectionMatrix(environment.getVRHardware().getHMDMatrixProjectionRightEye(getRightCamera())); |
||||
} |
||||
rightViewport = setupViewBuffers(getRightCamera(), RIGHT_VIEW_NAME); |
||||
} else { |
||||
|
||||
if (environment.getApplication() != null){ |
||||
|
||||
logger.severe("THIS CODE NEED CHANGES !!!"); |
||||
leftViewport = environment.getApplication().getViewPort(); |
||||
//leftViewport.attachScene(app.getRootNode());
|
||||
rightCamera = getLeftCamera().clone(); |
||||
if( environment.getVRHardware() != null ){ |
||||
getRightCamera().setProjectionMatrix(environment.getVRHardware().getHMDMatrixProjectionRightEye(getRightCamera())); |
||||
} |
||||
|
||||
org.lwjgl.opengl.GL11.glEnable(org.lwjgl.opengl.GL30.GL_CLIP_DISTANCE0); |
||||
|
||||
//FIXME: [jme-vr] Fix with JMonkey next release
|
||||
//RenderManager._VRInstancing_RightCamProjection = camRight.getViewProjectionMatrix();
|
||||
setupFinalFullTexture(environment.getApplication().getViewPort().getCamera()); |
||||
} else { |
||||
throw new IllegalStateException("This VR environment is not attached to any application."); |
||||
} |
||||
|
||||
} |
||||
|
||||
// setup gui
|
||||
environment.getVRGUIManager().setupGui(getLeftCamera(), getRightCamera(), getLeftViewport(), getRightViewport()); |
||||
|
||||
if( environment.getVRHardware() != null ) { |
||||
// call these to cache the results internally
|
||||
environment.getVRHardware().getHMDMatrixPoseLeftEye(); |
||||
environment.getVRHardware().getHMDMatrixPoseRightEye(); |
||||
} |
||||
} else { |
||||
throw new IllegalStateException("This VR view manager is not attached to any VR environment."); |
||||
} |
||||
} |
||||
|
||||
private ViewPort setupMirrorBuffers(Camera cam, Texture tex, boolean expand) { |
||||
|
||||
if (environment != null){ |
||||
if (environment.getApplication() != null){ |
||||
Camera clonecam = cam.clone(); |
||||
ViewPort viewPort = environment.getApplication().getRenderManager().createPostView("MirrorView", clonecam); |
||||
clonecam.setParallelProjection(true); |
||||
viewPort.setClearFlags(true, true, true); |
||||
viewPort.setBackgroundColor(ColorRGBA.Black); |
||||
Picture pic = new Picture("fullscene"); |
||||
pic.setLocalTranslation(-0.75f, -0.5f, 0f); |
||||
if( expand ) { |
||||
pic.setLocalScale(3f, 1f, 1f); |
||||
} else { |
||||
pic.setLocalScale(1.5f, 1f, 1f); |
||||
} |
||||
pic.setQueueBucket(Bucket.Opaque); |
||||
pic.setTexture(environment.getApplication().getAssetManager(), (Texture2D)tex, false); |
||||
viewPort.attachScene(pic); |
||||
viewPort.setOutputFrameBuffer(null); |
||||
|
||||
pic.updateGeometricState(); |
||||
|
||||
return viewPort; |
||||
} else { |
||||
throw new IllegalStateException("This VR environment is not attached to any application."); |
||||
} |
||||
} else { |
||||
throw new IllegalStateException("This VR view manager is not attached to any VR environment."); |
||||
} |
||||
} |
||||
|
||||
private void setupFinalFullTexture(Camera cam) { |
||||
|
||||
if (environment != null){ |
||||
if (environment.getApplication() != null){ |
||||
// create offscreen framebuffer
|
||||
FrameBuffer out = new FrameBuffer(cam.getWidth(), cam.getHeight(), 1); |
||||
//offBuffer.setSrgb(true);
|
||||
|
||||
//setup framebuffer's texture
|
||||
dualEyeTex = new Texture2D(cam.getWidth(), cam.getHeight(), Image.Format.RGBA8); |
||||
dualEyeTex.setMinFilter(Texture.MinFilter.BilinearNoMipMaps); |
||||
dualEyeTex.setMagFilter(Texture.MagFilter.Bilinear); |
||||
|
||||
logger.config("Dual eye texture "+dualEyeTex.getName()+" ("+dualEyeTex.getImage().getId()+")"); |
||||
logger.config(" Type: "+dualEyeTex.getType()); |
||||
logger.config(" Size: "+dualEyeTex.getImage().getWidth()+"x"+dualEyeTex.getImage().getHeight()); |
||||
logger.config(" Image depth: "+dualEyeTex.getImage().getDepth()); |
||||
logger.config(" Image format: "+dualEyeTex.getImage().getFormat()); |
||||
logger.config(" Image color space: "+dualEyeTex.getImage().getColorSpace()); |
||||
|
||||
//setup framebuffer to use texture
|
||||
out.setDepthBuffer(Image.Format.Depth); |
||||
out.setColorTexture(dualEyeTex); |
||||
|
||||
ViewPort viewPort = environment.getApplication().getViewPort(); |
||||
viewPort.setClearFlags(true, true, true); |
||||
viewPort.setBackgroundColor(ColorRGBA.Black); |
||||
viewPort.setOutputFrameBuffer(out); |
||||
} else { |
||||
throw new IllegalStateException("This VR environment is not attached to any application."); |
||||
} |
||||
} else { |
||||
throw new IllegalStateException("This VR view manager is not attached to any VR environment."); |
||||
} |
||||
} |
||||
|
||||
private ViewPort setupViewBuffers(Camera cam, String viewName){ |
||||
|
||||
if (environment != null){ |
||||
if (environment.getApplication() != null){ |
||||
// create offscreen framebuffer
|
||||
FrameBuffer offBufferLeft = new FrameBuffer(cam.getWidth(), cam.getHeight(), 1); |
||||
//offBufferLeft.setSrgb(true);
|
||||
|
||||
//setup framebuffer's texture
|
||||
Texture2D offTex = new Texture2D(cam.getWidth(), cam.getHeight(), Image.Format.RGBA8); |
||||
offTex.setMinFilter(Texture.MinFilter.BilinearNoMipMaps); |
||||
offTex.setMagFilter(Texture.MagFilter.Bilinear); |
||||
|
||||
//setup framebuffer to use texture
|
||||
offBufferLeft.setDepthBuffer(Image.Format.Depth); |
||||
offBufferLeft.setColorTexture(offTex); |
||||
|
||||
ViewPort viewPort = environment.getApplication().getRenderManager().createPreView(viewName, cam); |
||||
viewPort.setClearFlags(true, true, true); |
||||
viewPort.setBackgroundColor(ColorRGBA.Black); |
||||
|
||||
Iterator<Spatial> spatialIter = environment.getApplication().getViewPort().getScenes().iterator(); |
||||
while(spatialIter.hasNext()){ |
||||
viewPort.attachScene(spatialIter.next()); |
||||
} |
||||
|
||||
//set viewport to render to offscreen framebuffer
|
||||
viewPort.setOutputFrameBuffer(offBufferLeft); |
||||
return viewPort; |
||||
} else { |
||||
throw new IllegalStateException("This VR environment is not attached to any application."); |
||||
} |
||||
} else { |
||||
throw new IllegalStateException("This VR view manager is not attached to any VR environment."); |
||||
} |
||||
} |
||||
|
||||
/** |
||||
* Setup a distortion mesh for the stereo view. |
||||
* @param eye the eye to apply. |
||||
* @param api the underlying VR api |
||||
* @return the distorted mesh. |
||||
*/ |
||||
public static Mesh setupDistortionMesh(int eye, VRAPI api) { |
||||
Mesh distortionMesh = new Mesh(); |
||||
float m_iLensGridSegmentCountH = 43, m_iLensGridSegmentCountV = 43; |
||||
|
||||
float w = 1f / (m_iLensGridSegmentCountH - 1f); |
||||
float h = 1f / (m_iLensGridSegmentCountV - 1f); |
||||
|
||||
float u, v; |
||||
|
||||
float verts[] = new float[(int) (m_iLensGridSegmentCountV * m_iLensGridSegmentCountH) * 3]; |
||||
|
||||
float texcoordR[] = new float[(int) (m_iLensGridSegmentCountV * m_iLensGridSegmentCountH) * 2]; |
||||
float texcoordG[] = new float[(int) (m_iLensGridSegmentCountV * m_iLensGridSegmentCountH) * 2]; |
||||
float texcoordB[] = new float[(int) (m_iLensGridSegmentCountV * m_iLensGridSegmentCountH) * 2]; |
||||
|
||||
int vertPos = 0, coordPos = 0; |
||||
|
||||
float Xoffset = eye == JOpenVRLibrary.EVREye.EVREye_Eye_Left ? -1f : 0; |
||||
for (int y = 0; y < m_iLensGridSegmentCountV; y++) { |
||||
for (int x = 0; x < m_iLensGridSegmentCountH; x++) { |
||||
u = x * w; |
||||
v = 1 - y * h; |
||||
verts[vertPos] = Xoffset + u; // x
|
||||
verts[vertPos + 1] = -1 + 2 * y * h; // y
|
||||
verts[vertPos + 2] = 0f; // z
|
||||
vertPos += 3; |
||||
|
||||
DistortionCoordinates_t dc0 = new DistortionCoordinates_t(); |
||||
if( api.getVRSystem() == null ) { |
||||
// default to no distortion
|
||||
texcoordR[coordPos] = u; |
||||
texcoordR[coordPos + 1] = 1 - v; |
||||
texcoordG[coordPos] = u; |
||||
texcoordG[coordPos + 1] = 1 - v; |
||||
texcoordB[coordPos] = u; |
||||
texcoordB[coordPos + 1] = 1 - v; |
||||
} else { |
||||
((VR_IVRSystem_FnTable)api.getVRSystem()).ComputeDistortion.apply(eye, u, v, dc0); |
||||
|
||||
texcoordR[coordPos] = dc0.rfRed[0]; |
||||
texcoordR[coordPos + 1] = 1 - dc0.rfRed[1]; |
||||
texcoordG[coordPos] = dc0.rfGreen[0]; |
||||
texcoordG[coordPos + 1] = 1 - dc0.rfGreen[1]; |
||||
texcoordB[coordPos] = dc0.rfBlue[0]; |
||||
texcoordB[coordPos + 1] = 1 - dc0.rfBlue[1]; |
||||
} |
||||
|
||||
coordPos += 2; |
||||
} |
||||
} |
||||
|
||||
// have UV coordinates & positions, now to setup indices
|
||||
|
||||
int[] indices = new int[(int) ((m_iLensGridSegmentCountV - 1) * (m_iLensGridSegmentCountH - 1)) * 6]; |
||||
int indexPos = 0; |
||||
int a, b, c, d; |
||||
|
||||
int offset = 0; |
||||
for (int y = 0; y < m_iLensGridSegmentCountV - 1; y++) { |
||||
for (int x = 0; x < m_iLensGridSegmentCountH - 1; x++) { |
||||
a = (int) (m_iLensGridSegmentCountH * y + x + offset); |
||||
b = (int) (m_iLensGridSegmentCountH * y + x + 1 + offset); |
||||
c = (int) ((y + 1) * m_iLensGridSegmentCountH + x + 1 + offset); |
||||
d = (int) ((y + 1) * m_iLensGridSegmentCountH + x + offset); |
||||
|
||||
indices[indexPos] = a; |
||||
indices[indexPos + 1] = b; |
||||
indices[indexPos + 2] = c; |
||||
|
||||
indices[indexPos + 3] = a; |
||||
indices[indexPos + 4] = c; |
||||
indices[indexPos + 5] = d; |
||||
|
||||
indexPos += 6; |
||||
} |
||||
} |
||||
|
||||
// OK, create the mesh
|
||||
distortionMesh.setBuffer(VertexBuffer.Type.Position, 3, verts); |
||||
distortionMesh.setBuffer(VertexBuffer.Type.Index, 1, indices); |
||||
distortionMesh.setBuffer(VertexBuffer.Type.TexCoord, 2, texcoordR); |
||||
distortionMesh.setBuffer(VertexBuffer.Type.TexCoord2, 2, texcoordG); |
||||
distortionMesh.setBuffer(VertexBuffer.Type.TexCoord3, 2, texcoordB); |
||||
distortionMesh.setStatic(); |
||||
return distortionMesh; |
||||
} |
||||
} |
@ -1,107 +0,0 @@ |
||||
/* |
||||
* To change this template, choose Tools | Templates |
||||
* and open the template in the editor. |
||||
*/ |
||||
package jmevr.util; |
||||
|
||||
import com.jme3.app.VRApplication; |
||||
import com.jme3.input.vr.VRAPI; |
||||
import com.jme3.scene.Mesh; |
||||
import com.jme3.scene.VertexBuffer; |
||||
import com.jme3.system.jopenvr.DistortionCoordinates_t; |
||||
import com.jme3.system.jopenvr.JOpenVRLibrary; |
||||
import com.jme3.system.jopenvr.VR_IVRSystem_FnTable; |
||||
|
||||
/** |
||||
* |
||||
* @author reden |
||||
*/ |
||||
public class MeshUtil { |
||||
|
||||
public static Mesh setupDistortionMesh(int eye, VRAPI api) { |
||||
Mesh distortionMesh = new Mesh(); |
||||
float m_iLensGridSegmentCountH = 43, m_iLensGridSegmentCountV = 43; |
||||
|
||||
float w = 1f / (m_iLensGridSegmentCountH - 1f); |
||||
float h = 1f / (m_iLensGridSegmentCountV - 1f); |
||||
|
||||
float u, v; |
||||
|
||||
float verts[] = new float[(int) (m_iLensGridSegmentCountV * m_iLensGridSegmentCountH) * 3]; |
||||
|
||||
float texcoordR[] = new float[(int) (m_iLensGridSegmentCountV * m_iLensGridSegmentCountH) * 2]; |
||||
float texcoordG[] = new float[(int) (m_iLensGridSegmentCountV * m_iLensGridSegmentCountH) * 2]; |
||||
float texcoordB[] = new float[(int) (m_iLensGridSegmentCountV * m_iLensGridSegmentCountH) * 2]; |
||||
|
||||
int vertPos = 0, coordPos = 0; |
||||
|
||||
float Xoffset = eye == JOpenVRLibrary.EVREye.EVREye_Eye_Left ? -1f : 0; |
||||
for (int y = 0; y < m_iLensGridSegmentCountV; y++) { |
||||
for (int x = 0; x < m_iLensGridSegmentCountH; x++) { |
||||
u = x * w; |
||||
v = 1 - y * h; |
||||
verts[vertPos] = Xoffset + u; // x
|
||||
verts[vertPos + 1] = -1 + 2 * y * h; // y
|
||||
verts[vertPos + 2] = 0f; // z
|
||||
vertPos += 3; |
||||
|
||||
DistortionCoordinates_t dc0 = new DistortionCoordinates_t(); |
||||
if( api.getVRSystem() == null ) { |
||||
// default to no distortion
|
||||
texcoordR[coordPos] = u; |
||||
texcoordR[coordPos + 1] = 1 - v; |
||||
texcoordG[coordPos] = u; |
||||
texcoordG[coordPos + 1] = 1 - v; |
||||
texcoordB[coordPos] = u; |
||||
texcoordB[coordPos + 1] = 1 - v; |
||||
} else { |
||||
((VR_IVRSystem_FnTable)api.getVRSystem()).ComputeDistortion.apply(eye, u, v, dc0); |
||||
|
||||
texcoordR[coordPos] = dc0.rfRed[0]; |
||||
texcoordR[coordPos + 1] = 1 - dc0.rfRed[1]; |
||||
texcoordG[coordPos] = dc0.rfGreen[0]; |
||||
texcoordG[coordPos + 1] = 1 - dc0.rfGreen[1]; |
||||
texcoordB[coordPos] = dc0.rfBlue[0]; |
||||
texcoordB[coordPos + 1] = 1 - dc0.rfBlue[1]; |
||||
} |
||||
|
||||
coordPos += 2; |
||||
} |
||||
} |
||||
|
||||
// have UV coordinates & positions, now to setup indices
|
||||
|
||||
int[] indices = new int[(int) ((m_iLensGridSegmentCountV - 1) * (m_iLensGridSegmentCountH - 1)) * 6]; |
||||
int indexPos = 0; |
||||
int a, b, c, d; |
||||
|
||||
int offset = 0; |
||||
for (int y = 0; y < m_iLensGridSegmentCountV - 1; y++) { |
||||
for (int x = 0; x < m_iLensGridSegmentCountH - 1; x++) { |
||||
a = (int) (m_iLensGridSegmentCountH * y + x + offset); |
||||
b = (int) (m_iLensGridSegmentCountH * y + x + 1 + offset); |
||||
c = (int) ((y + 1) * m_iLensGridSegmentCountH + x + 1 + offset); |
||||
d = (int) ((y + 1) * m_iLensGridSegmentCountH + x + offset); |
||||
|
||||
indices[indexPos] = a; |
||||
indices[indexPos + 1] = b; |
||||
indices[indexPos + 2] = c; |
||||
|
||||
indices[indexPos + 3] = a; |
||||
indices[indexPos + 4] = c; |
||||
indices[indexPos + 5] = d; |
||||
|
||||
indexPos += 6; |
||||
} |
||||
} |
||||
|
||||
// OK, create the mesh
|
||||
distortionMesh.setBuffer(VertexBuffer.Type.Position, 3, verts); |
||||
distortionMesh.setBuffer(VertexBuffer.Type.Index, 1, indices); |
||||
distortionMesh.setBuffer(VertexBuffer.Type.TexCoord, 2, texcoordR); |
||||
distortionMesh.setBuffer(VertexBuffer.Type.TexCoord2, 2, texcoordG); |
||||
distortionMesh.setBuffer(VertexBuffer.Type.TexCoord3, 2, texcoordB); |
||||
distortionMesh.setStatic(); |
||||
return distortionMesh; |
||||
} |
||||
} |
@ -1,334 +0,0 @@ |
||||
/* |
||||
* To change this template, choose Tools | Templates |
||||
* and open the template in the editor. |
||||
*/ |
||||
package jmevr.util; |
||||
|
||||
import com.jme3.app.Application; |
||||
import com.jme3.app.VRAppState; |
||||
import com.jme3.app.VRApplication; |
||||
import com.jme3.app.state.AppState; |
||||
import com.jme3.material.Material; |
||||
import com.jme3.material.RenderState.BlendMode; |
||||
import com.jme3.math.ColorRGBA; |
||||
import com.jme3.math.Matrix3f; |
||||
import com.jme3.math.Quaternion; |
||||
import com.jme3.math.Vector2f; |
||||
import com.jme3.math.Vector3f; |
||||
import com.jme3.renderer.Camera; |
||||
import com.jme3.renderer.ViewPort; |
||||
import com.jme3.renderer.queue.RenderQueue.Bucket; |
||||
import com.jme3.scene.Spatial; |
||||
import com.jme3.scene.Geometry; |
||||
import com.jme3.scene.Node; |
||||
import com.jme3.system.AppSettings; |
||||
import com.jme3.texture.FrameBuffer; |
||||
import com.jme3.texture.Image.Format; |
||||
import com.jme3.texture.Texture; |
||||
import com.jme3.texture.Texture2D; |
||||
import java.awt.GraphicsEnvironment; |
||||
import java.util.Iterator; |
||||
|
||||
/** |
||||
* |
||||
* @author |
||||
* phr00t |
||||
*/ |
||||
public class VRGuiManager { |
||||
|
||||
public enum POSITIONING_MODE { |
||||
MANUAL, AUTO_CAM_ALL, AUTO_CAM_ALL_SKIP_PITCH, AUTO_OBSERVER_POS_CAM_ROTATION, AUTO_OBSERVER_ALL, AUTO_OBSERVER_ALL_CAMHEIGHT |
||||
} |
||||
|
||||
private Camera camLeft, camRight; |
||||
private float guiDistance = 1.5f, guiScale = 1f, guiPositioningElastic; |
||||
private POSITIONING_MODE posMode = POSITIONING_MODE.AUTO_CAM_ALL; |
||||
|
||||
private final Matrix3f orient = new Matrix3f(); |
||||
private Vector2f screenSize; |
||||
protected boolean wantsReposition; |
||||
|
||||
private VRAppState app = null; |
||||
private Application application = null; |
||||
|
||||
/** |
||||
* Create a new GUI manager attached to the given app state. |
||||
* @param app the VR app state that this manager is attached to. |
||||
*/ |
||||
public VRGuiManager(){ |
||||
} |
||||
|
||||
/** |
||||
* Get the VR app state to which this GUI manager is attached. |
||||
* @return the VR app state to which this GUI manager is attached. |
||||
*/ |
||||
public VRAppState getVRAppState(){ |
||||
return app; |
||||
} |
||||
|
||||
/** |
||||
* Attach the GUI manager to an app state and an Application. |
||||
* The application has to be the one that the app state is attached. |
||||
* This method should be called from the {@link AppState#initialize(com.jme3.app.state.AppStateManager, Application) initialize} |
||||
* method of the {@link AppState} instance. |
||||
* @param app the VR app state that this manager is attached to. |
||||
* @param application the application to whitch the app state is attcached. |
||||
*/ |
||||
public void attach(VRAppState app, Application application){ |
||||
this.app = app; |
||||
this.application = application; |
||||
} |
||||
|
||||
/** |
||||
* |
||||
* Makes auto GUI positioning happen not immediately, but like an |
||||
* elastic connected to the headset. Setting to 0 disables (default) |
||||
* Higher settings make it track the headset quicker. |
||||
* |
||||
* @param elastic amount of elasticity |
||||
*/ |
||||
public void setPositioningElasticity(float elastic) { |
||||
guiPositioningElastic = elastic; |
||||
} |
||||
|
||||
public float getPositioningElasticity() { |
||||
return guiPositioningElastic; |
||||
} |
||||
|
||||
public void setPositioningMode(POSITIONING_MODE mode) { |
||||
posMode = mode; |
||||
} |
||||
|
||||
public Vector2f getCanvasSize() { |
||||
if( screenSize == null ) { |
||||
if( app.isInVR() && app.getVRHardware() != null ) { |
||||
screenSize = new Vector2f(); |
||||
app.getVRHardware().getRenderSize(screenSize); |
||||
screenSize.multLocal(app.getVRViewManager().getResolutionMuliplier()); |
||||
} else { |
||||
AppSettings as = application.getContext().getSettings(); |
||||
screenSize = new Vector2f(as.getWidth(), as.getHeight()); |
||||
} |
||||
} |
||||
return screenSize; |
||||
} |
||||
|
||||
private Vector2f ratio; |
||||
|
||||
public Vector2f getCanvasToWindowRatio() { |
||||
if( ratio == null ) { |
||||
ratio = new Vector2f(); |
||||
Vector2f canvas = getCanvasSize(); |
||||
int width = Integer.min(GraphicsEnvironment.getLocalGraphicsEnvironment().getDefaultScreenDevice().getDisplayMode().getWidth(), |
||||
application.getContext().getSettings().getWidth()); |
||||
int height = Integer.min(GraphicsEnvironment.getLocalGraphicsEnvironment().getDefaultScreenDevice().getDisplayMode().getHeight(), |
||||
application.getContext().getSettings().getHeight()); |
||||
ratio.x = Float.max(1f, canvas.x / width); |
||||
ratio.y = Float.max(1f, canvas.y / height); |
||||
} |
||||
return ratio; |
||||
} |
||||
|
||||
public POSITIONING_MODE getPositioningMode() { |
||||
return posMode; |
||||
} |
||||
|
||||
public void positionGui() { |
||||
wantsReposition = true; |
||||
} |
||||
|
||||
private final Vector3f EoldPos = new Vector3f(); |
||||
private final Quaternion EoldDir = new Quaternion(); |
||||
private void positionTo(Vector3f pos, Quaternion dir, float tpf) { |
||||
Vector3f guiPos = guiQuadNode.getLocalTranslation(); |
||||
guiPos.set(0f, 0f, guiDistance); |
||||
dir.mult(guiPos, guiPos); |
||||
guiPos.x += pos.x; |
||||
guiPos.y += pos.y + app.getVRHeightAdjustment(); |
||||
guiPos.z += pos.z; |
||||
if( guiPositioningElastic > 0f && posMode != POSITIONING_MODE.MANUAL ) { |
||||
// mix pos & dir with current pos & dir
|
||||
guiPos.interpolateLocal(EoldPos, guiPos, Float.min(1f, tpf * guiPositioningElastic)); |
||||
EoldPos.set(guiPos); |
||||
} |
||||
} |
||||
|
||||
protected void updateGuiQuadGeometricState() { |
||||
guiQuadNode.updateGeometricState(); |
||||
} |
||||
|
||||
protected void positionGuiNow(float tpf) { |
||||
wantsReposition = false; |
||||
if( app.isInVR() == false ) return; |
||||
guiQuadNode.setLocalScale(guiDistance * guiScale * 4f, 4f * guiDistance * guiScale, 1f); |
||||
|
||||
switch( posMode ) { |
||||
case MANUAL: |
||||
case AUTO_CAM_ALL_SKIP_PITCH: |
||||
case AUTO_CAM_ALL: |
||||
if( camLeft != null && camRight != null ) { |
||||
// get middle point
|
||||
temppos.set(camLeft.getLocation()).interpolateLocal(camRight.getLocation(), 0.5f); |
||||
positionTo(temppos, camLeft.getRotation(), tpf); |
||||
} |
||||
rotateScreenTo(camLeft.getRotation(), tpf); |
||||
|
||||
break; |
||||
case AUTO_OBSERVER_POS_CAM_ROTATION: |
||||
Object obs = app.getObserver(); |
||||
if( obs != null ) { |
||||
if( obs instanceof Camera ) { |
||||
positionTo(((Camera)obs).getLocation(), camLeft.getRotation(), tpf); |
||||
} else { |
||||
positionTo(((Spatial)obs).getWorldTranslation(), camLeft.getRotation(), tpf); |
||||
} |
||||
} |
||||
rotateScreenTo(camLeft.getRotation(), tpf); |
||||
|
||||
break; |
||||
case AUTO_OBSERVER_ALL: |
||||
case AUTO_OBSERVER_ALL_CAMHEIGHT: |
||||
obs = app.getObserver(); |
||||
if( obs != null ) { |
||||
Quaternion q; |
||||
if( obs instanceof Camera ) { |
||||
q = ((Camera)obs).getRotation(); |
||||
temppos.set(((Camera)obs).getLocation()); |
||||
} else { |
||||
q = ((Spatial)obs).getWorldRotation(); |
||||
temppos.set(((Spatial)obs).getWorldTranslation()); |
||||
} |
||||
if( posMode == POSITIONING_MODE.AUTO_OBSERVER_ALL_CAMHEIGHT ) { |
||||
temppos.y = camLeft.getLocation().y; |
||||
} |
||||
positionTo(temppos, q, tpf); |
||||
rotateScreenTo(q, tpf); |
||||
|
||||
} |
||||
break; |
||||
|
||||
} |
||||
|
||||
|
||||
} |
||||
|
||||
private final Vector3f look = new Vector3f(), left = new Vector3f(), temppos = new Vector3f(), up = new Vector3f(); |
||||
private final Quaternion tempq = new Quaternion(); |
||||
|
||||
private void rotateScreenTo(Quaternion dir, float tpf) { |
||||
dir.getRotationColumn(2, look).negateLocal(); |
||||
dir.getRotationColumn(0, left).negateLocal(); |
||||
orient.fromAxes(left, dir.getRotationColumn(1, up), look); |
||||
Quaternion rot = tempq.fromRotationMatrix(orient); |
||||
if( posMode == POSITIONING_MODE.AUTO_CAM_ALL_SKIP_PITCH ) VRUtil.stripToYaw(rot); |
||||
if( guiPositioningElastic > 0f && posMode != POSITIONING_MODE.MANUAL ) { |
||||
// mix pos & dir with current pos & dir
|
||||
EoldDir.nlerp(rot, tpf * guiPositioningElastic); |
||||
guiQuadNode.setLocalRotation(EoldDir); |
||||
} else { |
||||
guiQuadNode.setLocalRotation(rot); |
||||
} |
||||
} |
||||
|
||||
public void setGuiDistance(float newGuiDistance) { |
||||
guiDistance = newGuiDistance; |
||||
} |
||||
|
||||
public void setGuiScale(float scale) { |
||||
guiScale = scale; |
||||
} |
||||
|
||||
public float getGuiDistance() { |
||||
return guiDistance; |
||||
} |
||||
|
||||
public void adjustGuiDistance(float adjustAmount) { |
||||
guiDistance += adjustAmount; |
||||
} |
||||
|
||||
protected void setupGui(Camera leftcam, Camera rightcam, ViewPort left, ViewPort right) { |
||||
if( app.hasTraditionalGUIOverlay() ) { |
||||
camLeft = leftcam; |
||||
camRight = rightcam; |
||||
Spatial guiScene = getGuiQuad(camLeft); |
||||
left.attachScene(guiScene); |
||||
if( right != null ) right.attachScene(guiScene); |
||||
setPositioningMode(posMode); |
||||
} |
||||
} |
||||
|
||||
/* |
||||
do not use, set by preconfigure routine in VRApplication |
||||
*/ |
||||
public void _enableCurvedSuface(boolean set) { |
||||
useCurvedSurface = set; |
||||
} |
||||
|
||||
/* |
||||
do not use, set by preconfigure routine in VRApplication |
||||
*/ |
||||
public void _enableGuiOverdraw(boolean set) { |
||||
overdraw = set; |
||||
} |
||||
|
||||
private boolean useCurvedSurface = false, overdraw = false; |
||||
private Geometry guiQuad; |
||||
private Node guiQuadNode; |
||||
private ViewPort offView; |
||||
private Texture2D guiTexture; |
||||
|
||||
private Spatial getGuiQuad(Camera sourceCam){ |
||||
if( guiQuadNode == null ) { |
||||
Vector2f guiCanvasSize = getCanvasSize(); |
||||
Camera offCamera = sourceCam.clone(); |
||||
offCamera.setParallelProjection(true); |
||||
offCamera.setLocation(Vector3f.ZERO); |
||||
offCamera.lookAt(Vector3f.UNIT_Z, Vector3f.UNIT_Y); |
||||
|
||||
offView = application.getRenderManager().createPreView("GUI View", offCamera); |
||||
offView.setClearFlags(true, true, true); |
||||
offView.setBackgroundColor(ColorRGBA.BlackNoAlpha); |
||||
|
||||
// create offscreen framebuffer
|
||||
FrameBuffer offBuffer = new FrameBuffer((int)guiCanvasSize.x, (int)guiCanvasSize.y, 1); |
||||
|
||||
//setup framebuffer's texture
|
||||
guiTexture = new Texture2D((int)guiCanvasSize.x, (int)guiCanvasSize.y, Format.RGBA8); |
||||
guiTexture.setMinFilter(Texture.MinFilter.BilinearNoMipMaps); |
||||
guiTexture.setMagFilter(Texture.MagFilter.Bilinear); |
||||
|
||||
//setup framebuffer to use texture
|
||||
offBuffer.setDepthBuffer(Format.Depth); |
||||
offBuffer.setColorTexture(guiTexture); |
||||
|
||||
//set viewport to render to offscreen framebuffer
|
||||
offView.setOutputFrameBuffer(offBuffer); |
||||
|
||||
// setup framebuffer's scene
|
||||
Iterator<Spatial> spatialIter = application.getGuiViewPort().getScenes().iterator(); |
||||
while(spatialIter.hasNext()){ |
||||
offView.attachScene(spatialIter.next()); |
||||
} |
||||
|
||||
|
||||
if( useCurvedSurface ) { |
||||
guiQuad = (Geometry)application.getAssetManager().loadModel("Common/Util/gui_mesh.j3o"); |
||||
} else { |
||||
guiQuad = new Geometry("guiQuad", new CenterQuad(1f, 1f)); |
||||
} |
||||
|
||||
Material mat = new Material(application.getAssetManager(), "Common/MatDefs/VR/GuiOverlay.j3md"); |
||||
mat.getAdditionalRenderState().setDepthTest(!overdraw); |
||||
mat.getAdditionalRenderState().setBlendMode(BlendMode.Alpha); |
||||
mat.getAdditionalRenderState().setDepthWrite(false); |
||||
mat.setTexture("ColorMap", guiTexture); |
||||
guiQuad.setQueueBucket(Bucket.Translucent); |
||||
guiQuad.setMaterial(mat); |
||||
|
||||
guiQuadNode = new Node("gui-quad-node"); |
||||
guiQuadNode.setQueueBucket(Bucket.Translucent); |
||||
guiQuadNode.attachChild(guiQuad); |
||||
} |
||||
return guiQuadNode; |
||||
} |
||||
} |
@ -1,234 +0,0 @@ |
||||
/* |
||||
* To change this license header, choose License Headers in Project Properties. |
||||
* To change this template file, choose Tools | Templates |
||||
* and open the template in the editor. |
||||
*/ |
||||
package jmevr.util; |
||||
|
||||
import java.util.logging.Level; |
||||
import java.util.logging.Logger; |
||||
|
||||
import org.lwjgl.glfw.GLFW; |
||||
|
||||
import com.jme3.app.Application; |
||||
import com.jme3.app.VRAppState; |
||||
import com.jme3.app.VRApplication; |
||||
import com.jme3.app.state.AppState; |
||||
import com.jme3.input.MouseInput; |
||||
import com.jme3.input.controls.AnalogListener; |
||||
import com.jme3.input.lwjgl.GlfwMouseInputVR; |
||||
import com.jme3.input.vr.VRInputType; |
||||
import com.jme3.material.RenderState.BlendMode; |
||||
import com.jme3.math.Vector2f; |
||||
import com.jme3.scene.Node; |
||||
import com.jme3.system.AppSettings; |
||||
import com.jme3.system.lwjgl.LwjglWindow; |
||||
import com.jme3.system.lwjgl.LwjglWindowVR; |
||||
import com.jme3.texture.Texture; |
||||
import com.jme3.texture.Texture2D; |
||||
import com.jme3.ui.Picture; |
||||
|
||||
/** |
||||
* |
||||
* @author Phr00t |
||||
*/ |
||||
public class VRMouseManager { |
||||
|
||||
private static final Logger logger = Logger.getLogger(VRMouseManager.class.getName()); |
||||
|
||||
private Application application = null; |
||||
private VRAppState app = null; |
||||
|
||||
private final int AVERAGE_AMNT = 4; |
||||
private int avgCounter; |
||||
|
||||
private Picture mouseImage; |
||||
private int recentCenterCount = 0; |
||||
private final Vector2f cursorPos = new Vector2f(); |
||||
private float ySize, sensitivity = 8f, acceleration = 2f; |
||||
private final float[] lastXmv = new float[AVERAGE_AMNT], lastYmv = new float[AVERAGE_AMNT]; |
||||
private boolean thumbstickMode; |
||||
private float moveScale = 1f; |
||||
|
||||
private float avg(float[] arr) { |
||||
float amt = 0f; |
||||
for(float f : arr) amt += f; |
||||
return amt / arr.length; |
||||
} |
||||
|
||||
public VRMouseManager(){ |
||||
} |
||||
|
||||
/** |
||||
* Attach the mouse manager to an app state and an Application. |
||||
* The application has to be the one that the app state is attached. |
||||
* This method should be called from the {@link AppState#initialize(com.jme3.app.state.AppStateManager, Application) initialize} |
||||
* method of the {@link AppState} instance. |
||||
* @param app the VR app state that this manager is attached to. |
||||
* @param application the application to whitch the app state is attcached. |
||||
*/ |
||||
public void attach(VRAppState app, Application application){ |
||||
this.app = app; |
||||
this.application = application; |
||||
} |
||||
|
||||
protected void init() { |
||||
|
||||
logger.config("Initializing VR mouse manager."); |
||||
|
||||
// load default mouseimage
|
||||
mouseImage = new Picture("mouse"); |
||||
setImage("Common/Util/mouse.png"); |
||||
// hide default cursor by making it invisible
|
||||
MouseInput mi = application.getContext().getMouseInput(); |
||||
if( mi instanceof GlfwMouseInputVR ){ |
||||
((GlfwMouseInputVR)mi).hideActiveCursor(); |
||||
} |
||||
centerMouse(); |
||||
|
||||
logger.config("Initialized VR mouse manager [SUCCESS]"); |
||||
} |
||||
|
||||
public void setThumbstickMode(boolean set) { |
||||
thumbstickMode = set; |
||||
} |
||||
|
||||
public boolean isThumbstickMode() { |
||||
return thumbstickMode; |
||||
} |
||||
|
||||
public void setSpeed(float sensitivity, float acceleration) { |
||||
this.sensitivity = sensitivity; |
||||
this.acceleration = acceleration; |
||||
} |
||||
|
||||
public float getSpeedSensitivity() { |
||||
return sensitivity; |
||||
} |
||||
|
||||
public float getSpeedAcceleration() { |
||||
return acceleration; |
||||
} |
||||
|
||||
public void setMouseMoveScale(float set) { |
||||
moveScale = set; |
||||
} |
||||
|
||||
public void setImage(String texture) { |
||||
if( app.isInVR() == false ){ |
||||
Texture tex = application.getAssetManager().loadTexture(texture); |
||||
mouseImage.setTexture(application.getAssetManager(), (Texture2D)tex, true); |
||||
ySize = tex.getImage().getHeight(); |
||||
mouseImage.setHeight(ySize); |
||||
mouseImage.setWidth(tex.getImage().getWidth()); |
||||
mouseImage.getMaterial().getAdditionalRenderState().setBlendMode(BlendMode.Alpha); |
||||
mouseImage.getMaterial().getAdditionalRenderState().setDepthWrite(false); |
||||
} else { |
||||
Texture tex = application.getAssetManager().loadTexture(texture); |
||||
mouseImage.setTexture(application.getAssetManager(), (Texture2D)tex, true); |
||||
ySize = tex.getImage().getHeight(); |
||||
mouseImage.setHeight(ySize); |
||||
mouseImage.setWidth(tex.getImage().getWidth()); |
||||
mouseImage.getMaterial().getAdditionalRenderState().setBlendMode(BlendMode.Alpha); |
||||
mouseImage.getMaterial().getAdditionalRenderState().setDepthWrite(false); |
||||
} |
||||
|
||||
} |
||||
|
||||
public void updateAnalogAsMouse(int inputIndex, AnalogListener mouseListener, String mouseXName, String mouseYName, float tpf) { |
||||
// got a tracked controller to use as the "mouse"
|
||||
if( app.isInVR() == false || |
||||
app.getVRinput() == null || |
||||
app.getVRinput().isInputDeviceTracking(inputIndex) == false ) return; |
||||
Vector2f tpDelta; |
||||
if( thumbstickMode ) { |
||||
tpDelta = app.getVRinput().getAxis(inputIndex, VRInputType.ViveTrackpadAxis); |
||||
} else { |
||||
tpDelta = app.getVRinput().getAxisDeltaSinceLastCall(inputIndex, VRInputType.ViveTrackpadAxis); |
||||
} |
||||
float Xamount = (float)Math.pow(Math.abs(tpDelta.x) * sensitivity, acceleration); |
||||
float Yamount = (float)Math.pow(Math.abs(tpDelta.y) * sensitivity, acceleration); |
||||
if( tpDelta.x < 0f ) Xamount = -Xamount; |
||||
if( tpDelta.y < 0f ) Yamount = -Yamount; |
||||
Xamount *= moveScale; Yamount *= moveScale; |
||||
if( mouseListener != null ) { |
||||
if( tpDelta.x != 0f && mouseXName != null ) mouseListener.onAnalog(mouseXName, Xamount * 0.2f, tpf); |
||||
if( tpDelta.y != 0f && mouseYName != null ) mouseListener.onAnalog(mouseYName, Yamount * 0.2f, tpf); |
||||
} |
||||
if( application.getInputManager().isCursorVisible() ) { |
||||
int index = (avgCounter+1) % AVERAGE_AMNT; |
||||
lastXmv[index] = Xamount * 133f; |
||||
lastYmv[index] = Yamount * 133f; |
||||
cursorPos.x -= avg(lastXmv); |
||||
cursorPos.y -= avg(lastYmv); |
||||
Vector2f maxsize = app.getVRGUIManager().getCanvasSize(); |
||||
if( cursorPos.x > maxsize.x ) cursorPos.x = maxsize.x; |
||||
if( cursorPos.x < 0f ) cursorPos.x = 0f; |
||||
if( cursorPos.y > maxsize.y ) cursorPos.y = maxsize.y; |
||||
if( cursorPos.y < 0f ) cursorPos.y = 0f; |
||||
} |
||||
} |
||||
|
||||
public Vector2f getCursorPosition() { |
||||
if( app.isInVR() ) { |
||||
return cursorPos; |
||||
} |
||||
return application.getInputManager().getCursorPosition(); |
||||
} |
||||
|
||||
public void centerMouse() { |
||||
// set mouse in center of the screen if newly added
|
||||
Vector2f size = app.getVRGUIManager().getCanvasSize(); |
||||
MouseInput mi = application.getContext().getMouseInput(); |
||||
AppSettings as = application.getContext().getSettings(); |
||||
if( mi instanceof GlfwMouseInputVR ) ((GlfwMouseInputVR)mi).setCursorPosition((int)(as.getWidth() / 2f), (int)(as.getHeight() / 2f)); |
||||
if( app.isInVR() ) { |
||||
cursorPos.x = size.x / 2f; |
||||
cursorPos.y = size.y / 2f; |
||||
recentCenterCount = 2; |
||||
} |
||||
} |
||||
|
||||
protected void update(float tpf) { |
||||
// if we are showing the cursor, add our picture as it
|
||||
|
||||
if( application.getInputManager().isCursorVisible() ) { |
||||
if( mouseImage.getParent() == null ) { |
||||
|
||||
application.getGuiViewPort().attachScene(mouseImage); |
||||
centerMouse(); |
||||
// the "real" mouse pointer should stay hidden
|
||||
if (application.getContext() instanceof LwjglWindow){ |
||||
GLFW.glfwSetInputMode(((LwjglWindow)application.getContext()).getWindowHandle(), GLFW.GLFW_CURSOR, GLFW.GLFW_CURSOR_DISABLED); |
||||
} |
||||
} |
||||
// handle mouse movements, which may be in addition to (or exclusive from) tracked movement
|
||||
MouseInput mi = application.getContext().getMouseInput(); |
||||
if( mi instanceof GlfwMouseInputVR ) { |
||||
if( recentCenterCount <= 0 ) { |
||||
//Vector2f winratio = VRGuiManager.getCanvasToWindowRatio();
|
||||
cursorPos.x += ((GlfwMouseInputVR)mi).getLastDeltaX();// * winratio.x;
|
||||
cursorPos.y += ((GlfwMouseInputVR)mi).getLastDeltaY();// * winratio.y;
|
||||
if( cursorPos.x < 0f ) cursorPos.x = 0f; |
||||
if( cursorPos.y < 0f ) cursorPos.y = 0f; |
||||
if( cursorPos.x > app.getVRGUIManager().getCanvasSize().x ) cursorPos.x = app.getVRGUIManager().getCanvasSize().x; |
||||
if( cursorPos.y > app.getVRGUIManager().getCanvasSize().y ) cursorPos.y = app.getVRGUIManager().getCanvasSize().y; |
||||
} else recentCenterCount--; |
||||
((GlfwMouseInputVR)mi).clearDeltas(); |
||||
} |
||||
// ok, update the cursor graphic position
|
||||
Vector2f currentPos = getCursorPosition(); |
||||
mouseImage.setLocalTranslation(currentPos.x, currentPos.y - ySize, app.getVRGUIManager().getGuiDistance() + 1f); |
||||
|
||||
mouseImage.updateGeometricState(); |
||||
|
||||
} else if( mouseImage.getParent() != null ) { |
||||
Node n = mouseImage.getParent(); |
||||
mouseImage.removeFromParent(); |
||||
|
||||
if (n != null){ |
||||
n.updateGeometricState(); |
||||
} |
||||
} |
||||
} |
||||
} |
@ -1,863 +0,0 @@ |
||||
/* |
||||
* To change this template, choose Tools | Templates |
||||
* and open the template in the editor. |
||||
*/ |
||||
package jmevr.util; |
||||
|
||||
import com.jme3.app.Application; |
||||
import com.jme3.app.VRAppState; |
||||
import com.jme3.app.VRApplication; |
||||
import com.jme3.app.state.AppState; |
||||
import com.jme3.input.vr.OSVR; |
||||
import com.jme3.input.vr.OpenVR; |
||||
import com.jme3.input.vr.VRAPI; |
||||
import com.jme3.material.Material; |
||||
import com.jme3.math.ColorRGBA; |
||||
import com.jme3.math.Quaternion; |
||||
import com.jme3.math.Vector2f; |
||||
import com.jme3.math.Vector3f; |
||||
import com.jme3.post.CartoonSSAO; |
||||
import com.jme3.post.Filter; |
||||
import com.jme3.post.FilterPostProcessor; |
||||
import com.jme3.post.SceneProcessor; |
||||
import com.jme3.post.filters.FogFilter; |
||||
import com.jme3.post.filters.TranslucentBucketFilter; |
||||
import com.jme3.post.ssao.SSAOFilter; |
||||
import com.jme3.renderer.Camera; |
||||
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.shadow.DirectionalLightShadowFilter; |
||||
import com.jme3.shadow.VRDirectionalLightShadowRenderer; |
||||
import com.jme3.system.jopenvr.JOpenVRLibrary; |
||||
import com.jme3.system.jopenvr.OpenVRUtil; |
||||
import com.jme3.system.jopenvr.Texture_t; |
||||
import com.jme3.system.jopenvr.VRTextureBounds_t; |
||||
import com.jme3.system.lwjgl.LwjglWindow; |
||||
import com.jme3.texture.FrameBuffer; |
||||
import com.jme3.texture.Image; |
||||
import com.jme3.texture.Texture; |
||||
import com.jme3.texture.Texture2D; |
||||
import com.jme3.ui.Picture; |
||||
import com.sun.jna.Pointer; |
||||
import com.sun.jna.ptr.PointerByReference; |
||||
|
||||
import java.awt.GraphicsEnvironment; |
||||
import java.util.Iterator; |
||||
import java.util.logging.Logger; |
||||
|
||||
import osvrrendermanageropengl.OSVR_RenderBufferOpenGL; |
||||
import osvrrendermanageropengl.OSVR_ViewportDescription; |
||||
import osvrrendermanageropengl.OsvrRenderManagerOpenGLLibrary; |
||||
|
||||
/** |
||||
* A VR view manager. This class enable to submit 3D views to the VR compositor. |
||||
* @author reden - phr00t - https://github.com/phr00t
|
||||
* @author Julien Seinturier - (c) 2016 - JOrigin project - <a href="http://www.jorigin.org">http:/www.jorigin.org</a> |
||||
*/ |
||||
public class VRViewManager { |
||||
|
||||
private static final Logger logger = Logger.getLogger(VRViewManager.class.getName()); |
||||
|
||||
/** |
||||
* The name of the left view. |
||||
*/ |
||||
public final static String LEFT_VIEW_NAME = "Left View"; |
||||
|
||||
/** |
||||
* The name of the right view. |
||||
*/ |
||||
public final static String RIGHT_VIEW_NAME = "Right View"; |
||||
|
||||
private VRAppState app; |
||||
private Application application; |
||||
|
||||
private Camera leftCamera; |
||||
private ViewPort leftViewport; |
||||
private FilterPostProcessor leftPostProcessor; |
||||
private Texture2D leftEyeTexture; |
||||
private Texture2D leftEyeDepth; |
||||
|
||||
private Camera rightCamera; |
||||
private ViewPort rightViewport; |
||||
private FilterPostProcessor rightPostProcessor; |
||||
private Texture2D rightEyeTexture; |
||||
private Texture2D rightEyeDepth; |
||||
|
||||
// OpenVR values
|
||||
private VRTextureBounds_t leftTextureBounds; |
||||
private Texture_t leftTextureType; |
||||
|
||||
private VRTextureBounds_t rightTextureBounds; |
||||
private Texture_t rightTextureType; |
||||
|
||||
// OSVR values
|
||||
OSVR_RenderBufferOpenGL.ByValue[] osvr_renderBuffer; |
||||
OSVR_ViewportDescription.ByValue osvr_viewDescFull; |
||||
OSVR_ViewportDescription.ByValue osvr_viewDescLeft; |
||||
OSVR_ViewportDescription.ByValue osvr_viewDescRight; |
||||
Pointer osvr_rmBufferState; |
||||
|
||||
//private static boolean useCustomDistortion;
|
||||
private float heightAdjustment; |
||||
|
||||
private Texture2D dualEyeTex; |
||||
|
||||
private final PointerByReference grabRBS = new PointerByReference(); |
||||
|
||||
private float resMult = 1f; |
||||
|
||||
//final & temp values for camera calculations
|
||||
private final Vector3f finalPosition = new Vector3f(); |
||||
private final Quaternion finalRotation = new Quaternion(); |
||||
private final Vector3f hmdPos = new Vector3f(); |
||||
private final Quaternion hmdRot = new Quaternion(); |
||||
|
||||
/** |
||||
* Create a new VR view manager attached to the given {@link VRAppState VR app state}.<br> |
||||
* in order to be used, this manager has to be attached to an app state and to an application. |
||||
*/ |
||||
public VRViewManager(){ |
||||
} |
||||
|
||||
/** |
||||
* Attach this manager to the given {@link VRAppState app state} and the given {@link Application application}. |
||||
* The application has to be the one that the app state is attached. |
||||
* This method should be called from the {@link AppState#initialize(com.jme3.app.state.AppStateManager, Application) initialize} |
||||
* method of the {@link AppState} instance. |
||||
* @param app the {@link VRAppState VR app state} to which this manager is linked. |
||||
* @param application the {@link Application} which the app state is attached. |
||||
*/ |
||||
public void attach(VRAppState app, Application application){ |
||||
this.app = app; |
||||
this.application = application; |
||||
} |
||||
|
||||
/** |
||||
* Get the {@link Camera camera} attached to the left eye. |
||||
* @return the {@link Camera camera} attached to the left eye. |
||||
* @see #getRightCamera() |
||||
*/ |
||||
public Camera getLeftCamera() { |
||||
return leftCamera; |
||||
} |
||||
|
||||
/** |
||||
* Get the {@link Camera camera} attached to the right eye. |
||||
* @return the {@link Camera camera} attached to the right eye. |
||||
* @see #getLeftCamera() |
||||
*/ |
||||
public Camera getRightCamera() { |
||||
return rightCamera; |
||||
} |
||||
|
||||
/** |
||||
* Get the {@link ViewPort viewport} attached to the left eye. |
||||
* @return the {@link ViewPort viewport} attached to the left eye. |
||||
* @see #getRightViewport() |
||||
*/ |
||||
public ViewPort getLeftViewport() { |
||||
return leftViewport; |
||||
} |
||||
|
||||
/** |
||||
* Get the {@link ViewPort viewport} attached to the right eye. |
||||
* @return the {@link ViewPort viewport} attached to the right eye. |
||||
* @see #getLeftViewport() |
||||
*/ |
||||
public ViewPort getRightViewport() { |
||||
return rightViewport; |
||||
} |
||||
|
||||
/** |
||||
* Get the identifier of the left eye texture. |
||||
* @return the identifier of the left eye texture. |
||||
* @see #getRightTexId() |
||||
* @see #getFullTexId() |
||||
*/ |
||||
private int getLeftTexId() { |
||||
return (int)leftEyeTexture.getImage().getId(); |
||||
} |
||||
|
||||
/** |
||||
* Get the identifier of the right eye texture. |
||||
* @return the identifier of the right eye texture. |
||||
* @see #getLeftTexId() |
||||
* @see #getFullTexId() |
||||
*/ |
||||
private int getRightTexId() { |
||||
return (int)rightEyeTexture.getImage().getId(); |
||||
} |
||||
|
||||
/** |
||||
* Get the identifier of the full (dual eye) texture. |
||||
* @return the identifier of the full (dual eye) texture. |
||||
* @see #getLeftTexId() |
||||
* @see #getRightTexId() |
||||
*/ |
||||
private int getFullTexId() { |
||||
return (int)dualEyeTex.getImage().getId(); |
||||
} |
||||
|
||||
/** |
||||
* Get the height adjustment to apply to the cameras before rendering. |
||||
* @return the height adjustment to apply to the cameras before rendering. |
||||
* @see #setHeightAdjustment(float) |
||||
*/ |
||||
public float getHeightAdjustment() { |
||||
return heightAdjustment; |
||||
} |
||||
|
||||
/** |
||||
* Set the height adjustment to apply to the cameras before rendering. |
||||
* @param amount the height adjustment to apply to the cameras before rendering. |
||||
* @see #getHeightAdjustment() |
||||
*/ |
||||
public void setHeightAdjustment(float amount) { |
||||
heightAdjustment = amount; |
||||
} |
||||
|
||||
/** |
||||
* Get the resolution multiplier. |
||||
* @return the resolution multiplier. |
||||
* @see #setResolutionMultiplier(float) |
||||
*/ |
||||
public float getResolutionMuliplier() { |
||||
return resMult; |
||||
} |
||||
|
||||
/** |
||||
* Set the resolution multiplier. |
||||
* @param resMult the resolution multiplier. |
||||
* @see #getResolutionMuliplier() |
||||
*/ |
||||
public void setResolutionMultiplier(float resMult) { |
||||
this.resMult = resMult; |
||||
} |
||||
|
||||
/** |
||||
* Initialize the system binds of the textures. |
||||
*/ |
||||
private void initTextureSubmitStructs() { |
||||
leftTextureType = new Texture_t(); |
||||
rightTextureType = new Texture_t(); |
||||
|
||||
|
||||
if( app.getVRHardware() instanceof OpenVR ) { |
||||
leftTextureBounds = new VRTextureBounds_t(); |
||||
rightTextureBounds = new VRTextureBounds_t(); |
||||
// left eye
|
||||
leftTextureBounds.uMax = 0.5f; |
||||
leftTextureBounds.uMin = 0f; |
||||
leftTextureBounds.vMax = 1f; |
||||
leftTextureBounds.vMin = 0f; |
||||
leftTextureBounds.setAutoSynch(false); |
||||
leftTextureBounds.setAutoRead(false); |
||||
leftTextureBounds.setAutoWrite(false); |
||||
leftTextureBounds.write(); |
||||
// right eye
|
||||
rightTextureBounds.uMax = 1f; |
||||
rightTextureBounds.uMin = 0.5f; |
||||
rightTextureBounds.vMax = 1f; |
||||
rightTextureBounds.vMin = 0f; |
||||
rightTextureBounds.setAutoSynch(false); |
||||
rightTextureBounds.setAutoRead(false); |
||||
rightTextureBounds.setAutoWrite(false); |
||||
rightTextureBounds.write(); |
||||
// texture type
|
||||
// FIXME: Synchronize with JMonkey given texture (at this time is linear but was Gamma with phr00t implementation)
|
||||
leftTextureType.eColorSpace = JOpenVRLibrary.EColorSpace.EColorSpace_ColorSpace_Gamma; |
||||
//leftTextureType.eColorSpace = JOpenVRLibrary.EColorSpace.EColorSpace_ColorSpace_Linear;
|
||||
leftTextureType.eType = JOpenVRLibrary.ETextureType.ETextureType_TextureType_OpenGL; |
||||
leftTextureType.setAutoSynch(false); |
||||
leftTextureType.setAutoRead(false); |
||||
leftTextureType.setAutoWrite(false); |
||||
leftTextureType.handle = -1; |
||||
// FIXME: Synchronize with JMonkey given texture (at this time is linear but was Gamma with phr00t implementation)
|
||||
rightTextureType.eColorSpace = JOpenVRLibrary.EColorSpace.EColorSpace_ColorSpace_Gamma; |
||||
//rightTextureType.eColorSpace = JOpenVRLibrary.EColorSpace.EColorSpace_ColorSpace_Linear;
|
||||
rightTextureType.eType = JOpenVRLibrary.ETextureType.ETextureType_TextureType_OpenGL; |
||||
rightTextureType.setAutoSynch(false); |
||||
rightTextureType.setAutoRead(false); |
||||
rightTextureType.setAutoWrite(false); |
||||
rightTextureType.handle = -1; |
||||
|
||||
|
||||
logger.config("Init eyes native texture binds"); |
||||
logger.config(" Left eye texture"); |
||||
logger.config(" address: "+leftTextureType.getPointer()); |
||||
logger.config(" size: "+leftTextureType.size()+" bytes"); |
||||
logger.config(" color space: "+OpenVRUtil.getEColorSpaceString(leftTextureType.eColorSpace)); |
||||
logger.config(" type: "+OpenVRUtil.getETextureTypeString(leftTextureType.eType)); |
||||
logger.config(" auto read: "+leftTextureType.getAutoRead()); |
||||
logger.config(" auto write: "+leftTextureType.getAutoWrite()); |
||||
logger.config(" handle address: "+leftTextureType.handle); |
||||
logger.config(" handle value: "+leftTextureType.handle); |
||||
logger.config(""); |
||||
logger.config(" Right eye texture"); |
||||
logger.config(" address: "+rightTextureType.getPointer()); |
||||
logger.config(" size: "+rightTextureType.size()+" bytes"); |
||||
logger.config(" color space: "+OpenVRUtil.getEColorSpaceString(rightTextureType.eColorSpace)); |
||||
logger.config(" type: "+OpenVRUtil.getETextureTypeString(rightTextureType.eType)); |
||||
logger.config(" auto read: "+rightTextureType.getAutoRead()); |
||||
logger.config(" auto write: "+rightTextureType.getAutoWrite()); |
||||
logger.config(" handle address: "+rightTextureType.handle); |
||||
logger.config(" handle value: "+rightTextureType.handle); |
||||
|
||||
|
||||
} else if( app.getVRHardware() instanceof OSVR ) { |
||||
// must be OSVR
|
||||
osvr_renderBuffer = new OSVR_RenderBufferOpenGL.ByValue[2]; |
||||
osvr_renderBuffer[OSVR.EYE_LEFT] = new OSVR_RenderBufferOpenGL.ByValue(); |
||||
osvr_renderBuffer[OSVR.EYE_RIGHT] = new OSVR_RenderBufferOpenGL.ByValue(); |
||||
osvr_renderBuffer[OSVR.EYE_LEFT].setAutoSynch(false); |
||||
osvr_renderBuffer[OSVR.EYE_RIGHT].setAutoSynch(false); |
||||
osvr_viewDescFull = new OSVR_ViewportDescription.ByValue(); |
||||
osvr_viewDescFull.setAutoSynch(false); |
||||
osvr_viewDescFull.left = osvr_viewDescFull.lower = 0.0; |
||||
osvr_viewDescFull.width = osvr_viewDescFull.height = 1.0; |
||||
osvr_viewDescLeft = new OSVR_ViewportDescription.ByValue(); |
||||
osvr_viewDescLeft.setAutoSynch(false); |
||||
osvr_viewDescLeft.left = osvr_viewDescLeft.lower = 0.0; |
||||
osvr_viewDescLeft.width = 0.5; |
||||
osvr_viewDescLeft.height = 1.0; |
||||
osvr_viewDescRight = new OSVR_ViewportDescription.ByValue(); |
||||
osvr_viewDescRight.setAutoSynch(false); |
||||
osvr_viewDescRight.left = 0.5; |
||||
osvr_viewDescRight.lower = 0.0; |
||||
osvr_viewDescRight.width = 0.5; |
||||
osvr_viewDescRight.height = 1.0; |
||||
osvr_viewDescRight.write(); |
||||
osvr_viewDescLeft.write(); |
||||
osvr_viewDescFull.write(); |
||||
osvr_renderBuffer[OSVR.EYE_LEFT].depthStencilBufferName = -1; |
||||
osvr_renderBuffer[OSVR.EYE_LEFT].colorBufferName = -1; |
||||
osvr_renderBuffer[OSVR.EYE_RIGHT].depthStencilBufferName = -1; |
||||
osvr_renderBuffer[OSVR.EYE_RIGHT].colorBufferName = -1; |
||||
} |
||||
} |
||||
|
||||
/** |
||||
* Register the OSVR OpenGL buffer. |
||||
* @param buf the OSVR OpenGL buffer. |
||||
*/ |
||||
private void registerOSVRBuffer(OSVR_RenderBufferOpenGL.ByValue buf) { |
||||
OsvrRenderManagerOpenGLLibrary.osvrRenderManagerStartRegisterRenderBuffers(grabRBS); |
||||
OsvrRenderManagerOpenGLLibrary.osvrRenderManagerRegisterRenderBufferOpenGL(grabRBS.getValue(), buf); |
||||
OsvrRenderManagerOpenGLLibrary.osvrRenderManagerFinishRegisterRenderBuffers(((OSVR)app.getVRHardware()).getCompositor(), grabRBS.getValue(), (byte)0); |
||||
} |
||||
|
||||
/** |
||||
* Send the textures to the two eyes. |
||||
*/ |
||||
public void sendTextures() { |
||||
if( app.isInVR() ) { |
||||
VRAPI api = app.getVRHardware(); |
||||
if( api.getCompositor() != null ) { |
||||
// using the compositor...
|
||||
int errl = 0, errr = 0; |
||||
if( app.isInstanceVRRendering() ) { |
||||
if( leftTextureType.handle == -1 || leftTextureType.handle != getFullTexId() ) { |
||||
leftTextureType.handle = getFullTexId(); |
||||
if( leftTextureType.handle != -1 ) { |
||||
leftTextureType.write(); |
||||
if( api instanceof OSVR ) { |
||||
osvr_renderBuffer[OSVR.EYE_LEFT].colorBufferName = leftTextureType.handle; |
||||
osvr_renderBuffer[OSVR.EYE_LEFT].depthStencilBufferName = dualEyeTex.getImage().getId(); |
||||
osvr_renderBuffer[OSVR.EYE_LEFT].write(); |
||||
registerOSVRBuffer(osvr_renderBuffer[OSVR.EYE_LEFT]); |
||||
} |
||||
} |
||||
} else { |
||||
if( api instanceof OpenVR ) { |
||||
int submitFlag = JOpenVRLibrary.EVRSubmitFlags.EVRSubmitFlags_Submit_Default; |
||||
errr = ((OpenVR)api).getCompositor().Submit.apply(JOpenVRLibrary.EVREye.EVREye_Eye_Right, leftTextureType, rightTextureBounds, submitFlag); |
||||
errl = ((OpenVR)api).getCompositor().Submit.apply(JOpenVRLibrary.EVREye.EVREye_Eye_Left, leftTextureType, leftTextureBounds, submitFlag); |
||||
} else if( api instanceof OSVR ) { |
||||
((OSVR)api).handleRenderBufferPresent(osvr_viewDescLeft, osvr_viewDescRight, |
||||
osvr_renderBuffer[OSVR.EYE_LEFT], osvr_renderBuffer[OSVR.EYE_LEFT]); |
||||
} |
||||
} |
||||
} else if( leftTextureType.handle == -1 || rightTextureType.handle == -1 || |
||||
leftTextureType.handle != getLeftTexId() || rightTextureType.handle != getRightTexId() ) { |
||||
leftTextureType.handle = getLeftTexId(); |
||||
if( leftTextureType.handle != -1 ) { |
||||
logger.fine("Writing Left texture to native memory at " + leftTextureType.getPointer()); |
||||
leftTextureType.write(); |
||||
if( api instanceof OSVR ) { |
||||
osvr_renderBuffer[OSVR.EYE_LEFT].colorBufferName = leftTextureType.handle; |
||||
if( leftEyeDepth != null ) osvr_renderBuffer[OSVR.EYE_LEFT].depthStencilBufferName = leftEyeDepth.getImage().getId(); |
||||
osvr_renderBuffer[OSVR.EYE_LEFT].write(); |
||||
registerOSVRBuffer(osvr_renderBuffer[OSVR.EYE_LEFT]); |
||||
} |
||||
} |
||||
rightTextureType.handle = getRightTexId(); |
||||
if( rightTextureType.handle != -1 ) { |
||||
logger.fine("Writing Right texture to native memory at " + leftTextureType.getPointer()); |
||||
rightTextureType.write(); |
||||
if( api instanceof OSVR ) { |
||||
osvr_renderBuffer[OSVR.EYE_RIGHT].colorBufferName = rightTextureType.handle; |
||||
if( rightEyeDepth != null ) osvr_renderBuffer[OSVR.EYE_RIGHT].depthStencilBufferName = rightEyeDepth.getImage().getId(); |
||||
osvr_renderBuffer[OSVR.EYE_RIGHT].write(); |
||||
registerOSVRBuffer(osvr_renderBuffer[OSVR.EYE_RIGHT]); |
||||
} |
||||
} |
||||
} else { |
||||
if( api instanceof OpenVR ) { |
||||
errl = ((OpenVR)api).getCompositor().Submit.apply(JOpenVRLibrary.EVREye.EVREye_Eye_Left, leftTextureType, null, |
||||
JOpenVRLibrary.EVRSubmitFlags.EVRSubmitFlags_Submit_Default); |
||||
errr = ((OpenVR)api).getCompositor().Submit.apply(JOpenVRLibrary.EVREye.EVREye_Eye_Right, rightTextureType, null, |
||||
JOpenVRLibrary.EVRSubmitFlags.EVRSubmitFlags_Submit_Default); |
||||
} else if( api instanceof OSVR ) { |
||||
((OSVR)api).handleRenderBufferPresent(osvr_viewDescFull, osvr_viewDescFull, |
||||
osvr_renderBuffer[OSVR.EYE_LEFT], osvr_renderBuffer[OSVR.EYE_RIGHT]); |
||||
} |
||||
} |
||||
|
||||
if( errl != 0 ){ |
||||
logger.severe("Submit to left compositor error: " + OpenVRUtil.getEVRCompositorErrorString(errl)+" ("+Integer.toString(errl)+")"); |
||||
logger.severe(" Texture color space: "+OpenVRUtil.getEColorSpaceString(leftTextureType.eColorSpace)); |
||||
logger.severe(" Texture type: "+OpenVRUtil.getETextureTypeString(leftTextureType.eType)); |
||||
logger.severe(" Texture handle: "+leftTextureType.handle); |
||||
|
||||
logger.severe(" Left eye texture "+leftEyeTexture.getName()+" ("+leftEyeTexture.getImage().getId()+")"); |
||||
logger.severe(" Type: "+leftEyeTexture.getType()); |
||||
logger.severe(" Size: "+leftEyeTexture.getImage().getWidth()+"x"+leftEyeTexture.getImage().getHeight()); |
||||
logger.severe(" Image depth: "+leftEyeTexture.getImage().getDepth()); |
||||
logger.severe(" Image format: "+leftEyeTexture.getImage().getFormat()); |
||||
logger.severe(" Image color space: "+leftEyeTexture.getImage().getColorSpace()); |
||||
|
||||
} |
||||
|
||||
if( errr != 0 ){ |
||||
logger.severe("Submit to right compositor error: " + OpenVRUtil.getEVRCompositorErrorString(errl)+" ("+Integer.toString(errl)+")"); |
||||
logger.severe(" Texture color space: "+OpenVRUtil.getEColorSpaceString(rightTextureType.eColorSpace)); |
||||
logger.severe(" Texture type: "+OpenVRUtil.getETextureTypeString(rightTextureType.eType)); |
||||
logger.severe(" Texture handle: "+rightTextureType.handle); |
||||
|
||||
logger.severe(" Right eye texture "+rightEyeTexture.getName()+" ("+rightEyeTexture.getImage().getId()+")"); |
||||
logger.severe(" Type: "+rightEyeTexture.getType()); |
||||
logger.severe(" Size: "+rightEyeTexture.getImage().getWidth()+"x"+rightEyeTexture.getImage().getHeight()); |
||||
logger.severe(" Image depth: "+rightEyeTexture.getImage().getDepth()); |
||||
logger.severe(" Image format: "+rightEyeTexture.getImage().getFormat()); |
||||
logger.severe(" Image color space: "+rightEyeTexture.getImage().getColorSpace()); |
||||
} |
||||
} |
||||
} |
||||
} |
||||
|
||||
|
||||
/** |
||||
* Initialize the VR view manager. |
||||
*/ |
||||
public void initialize() { |
||||
|
||||
logger.config("Initializing VR view manager."); |
||||
|
||||
initTextureSubmitStructs(); |
||||
setupCamerasAndViews(); |
||||
setupVRScene(); |
||||
moveScreenProcessingToEyes(); |
||||
if( app.hasTraditionalGUIOverlay() ) { |
||||
|
||||
app.getVRMouseManager().init(); |
||||
|
||||
// update the pose to position the gui correctly on start
|
||||
update(0f); |
||||
app.getVRGUIManager().positionGui(); |
||||
} |
||||
// if we are OSVR, our primary mirror window needs to be the same size as the render manager's output...
|
||||
if( app.getVRHardware() instanceof OSVR ) { |
||||
int origWidth = GraphicsEnvironment.getLocalGraphicsEnvironment().getDefaultScreenDevice().getDisplayMode().getWidth(); |
||||
int origHeight = GraphicsEnvironment.getLocalGraphicsEnvironment().getDefaultScreenDevice().getDisplayMode().getHeight(); |
||||
long window = ((LwjglWindow)application.getContext()).getWindowHandle(); |
||||
Vector2f windowSize = new Vector2f(); |
||||
((OSVR)app.getVRHardware()).getRenderSize(windowSize); |
||||
windowSize.x = Math.max(windowSize.x * 2f, leftCamera.getWidth()); |
||||
org.lwjgl.glfw.GLFW.glfwSetWindowSize(window, (int)windowSize.x, (int)windowSize.y); |
||||
application.getContext().getSettings().setResolution((int)windowSize.x, (int)windowSize.y); |
||||
|
||||
if (application.getRenderManager() != null) { |
||||
application.getRenderManager().notifyReshape((int)windowSize.x, (int)windowSize.y); |
||||
} |
||||
|
||||
org.lwjgl.glfw.GLFW.glfwSetWindowPos(window, origWidth - (int)windowSize.x, 32); |
||||
|
||||
org.lwjgl.glfw.GLFW.glfwFocusWindow(window); |
||||
|
||||
org.lwjgl.glfw.GLFW.glfwSetCursorPos(window, origWidth / 2.0, origHeight / 2.0); |
||||
} |
||||
|
||||
logger.config("Initialized VR view manager [SUCCESS]"); |
||||
} |
||||
|
||||
/** |
||||
* Prepare the size of the given {@link Camera camera} to adapt it to the underlying rendering context. |
||||
* @param cam the {@link Camera camera} to prepare. |
||||
* @param xMult the camera width multiplier. |
||||
*/ |
||||
private void prepareCameraSize(Camera cam, float xMult) { |
||||
Vector2f size = new Vector2f(); |
||||
VRAPI vrhmd = app.getVRHardware(); |
||||
|
||||
if( vrhmd == null ) { |
||||
size.x = 1280f; |
||||
size.y = 720f; |
||||
} else { |
||||
vrhmd.getRenderSize(size); |
||||
} |
||||
|
||||
if( size.x < application.getContext().getSettings().getWidth() ) { |
||||
size.x = application.getContext().getSettings().getWidth(); |
||||
} |
||||
if( size.y < application.getContext().getSettings().getHeight() ) { |
||||
size.y = application.getContext().getSettings().getHeight(); |
||||
} |
||||
|
||||
if( app.isInstanceVRRendering() ) size.x *= 2f; |
||||
|
||||
// other adjustments
|
||||
size.x *= xMult; |
||||
size.x *= resMult; |
||||
size.y *= resMult; |
||||
|
||||
if( cam.getWidth() != size.x || cam.getHeight() != size.y ) cam.resize((int)size.x, (int)size.y, false); |
||||
} |
||||
|
||||
/** |
||||
* Replaces rootNode as the main cameras scene with the distortion mesh |
||||
*/ |
||||
private void setupVRScene(){ |
||||
// no special scene to setup if we are doing instancing
|
||||
if( app.isInstanceVRRendering() ) { |
||||
// distortion has to be done with compositor here... we want only one pass on our end!
|
||||
if( application.getContext().getSettings().isSwapBuffers() ) { |
||||
setupMirrorBuffers(app.getCamera(), dualEyeTex, true); |
||||
} |
||||
return; |
||||
} |
||||
|
||||
leftEyeTexture = (Texture2D) leftViewport.getOutputFrameBuffer().getColorBuffer().getTexture(); |
||||
rightEyeTexture = (Texture2D)rightViewport.getOutputFrameBuffer().getColorBuffer().getTexture(); |
||||
leftEyeDepth = (Texture2D) leftViewport.getOutputFrameBuffer().getDepthBuffer().getTexture(); |
||||
rightEyeDepth = (Texture2D)rightViewport.getOutputFrameBuffer().getDepthBuffer().getTexture(); |
||||
|
||||
// main viewport is either going to be a distortion scene or nothing
|
||||
// mirroring is handled by copying framebuffers
|
||||
Iterator<Spatial> spatialIter = application.getViewPort().getScenes().iterator(); |
||||
while(spatialIter.hasNext()){ |
||||
application.getViewPort().detachScene(spatialIter.next()); |
||||
} |
||||
|
||||
spatialIter = application.getGuiViewPort().getScenes().iterator(); |
||||
while(spatialIter.hasNext()){ |
||||
application.getGuiViewPort().detachScene(spatialIter.next()); |
||||
} |
||||
|
||||
// only setup distortion scene if compositor isn't running (or using custom mesh distortion option)
|
||||
if( app.getVRHardware().getCompositor() == null ) { |
||||
Node distortionScene = new Node(); |
||||
Material leftMat = new Material(application.getAssetManager(), "Common/MatDefs/VR/OpenVR.j3md"); |
||||
leftMat.setTexture("Texture", leftEyeTexture); |
||||
Geometry leftEye = new Geometry("box", MeshUtil.setupDistortionMesh(JOpenVRLibrary.EVREye.EVREye_Eye_Left, app.getVRHardware())); |
||||
leftEye.setMaterial(leftMat); |
||||
distortionScene.attachChild(leftEye); |
||||
|
||||
Material rightMat = new Material(application.getAssetManager(), "Common/MatDefs/VR/OpenVR.j3md"); |
||||
rightMat.setTexture("Texture", rightEyeTexture); |
||||
Geometry rightEye = new Geometry("box", MeshUtil.setupDistortionMesh(JOpenVRLibrary.EVREye.EVREye_Eye_Right, app.getVRHardware())); |
||||
rightEye.setMaterial(rightMat); |
||||
distortionScene.attachChild(rightEye); |
||||
|
||||
distortionScene.updateGeometricState(); |
||||
|
||||
application.getViewPort().attachScene(distortionScene); |
||||
|
||||
//if( useCustomDistortion ) setupFinalFullTexture(app.getViewPort().getCamera());
|
||||
} |
||||
|
||||
if( application.getContext().getSettings().isSwapBuffers() ) { |
||||
setupMirrorBuffers(app.getCamera(), leftEyeTexture, false); |
||||
} |
||||
} |
||||
|
||||
/** |
||||
* Update the VR view manager. |
||||
* This method is called by the attached {@link VRApplication VR application} and should not be called manually. |
||||
* @param tpf the time per frame. |
||||
*/ |
||||
public void update(float tpf) { |
||||
|
||||
// grab the observer
|
||||
Object obs = app.getObserver(); |
||||
Quaternion objRot; |
||||
Vector3f objPos; |
||||
if( obs instanceof Camera ) { |
||||
objRot = ((Camera)obs).getRotation(); |
||||
objPos = ((Camera)obs).getLocation(); |
||||
} else { |
||||
objRot = ((Spatial)obs).getWorldRotation(); |
||||
objPos = ((Spatial)obs).getWorldTranslation(); |
||||
} |
||||
// grab the hardware handle
|
||||
VRAPI dev = app.getVRHardware(); |
||||
if( dev != null ) { |
||||
// update the HMD's position & orientation
|
||||
dev.updatePose(); |
||||
dev.getPositionAndOrientation(hmdPos, hmdRot); |
||||
if( obs != null ) { |
||||
// update hmdPos based on obs rotation
|
||||
finalRotation.set(objRot); |
||||
finalRotation.mult(hmdPos, hmdPos); |
||||
finalRotation.multLocal(hmdRot); |
||||
} |
||||
|
||||
finalizeCamera(dev.getHMDVectorPoseLeftEye(), objPos, leftCamera); |
||||
finalizeCamera(dev.getHMDVectorPoseRightEye(), objPos, rightCamera); |
||||
} else { |
||||
leftCamera.setFrame(objPos, objRot); |
||||
rightCamera.setFrame(objPos, objRot); |
||||
} |
||||
|
||||
if( app.hasTraditionalGUIOverlay() ) { |
||||
// update the mouse?
|
||||
app.getVRMouseManager().update(tpf); |
||||
|
||||
// update GUI position?
|
||||
if( app.getVRGUIManager().wantsReposition || app.getVRGUIManager().getPositioningMode() != VRGuiManager.POSITIONING_MODE.MANUAL ) { |
||||
app.getVRGUIManager().positionGuiNow(tpf); |
||||
app.getVRGUIManager().updateGuiQuadGeometricState(); |
||||
} |
||||
} |
||||
} |
||||
|
||||
/** |
||||
* Place the camera within the scene. |
||||
* @param eyePos the eye position. |
||||
* @param obsPosition the observer position. |
||||
* @param cam the camera to place. |
||||
*/ |
||||
private void finalizeCamera(Vector3f eyePos, Vector3f obsPosition, Camera cam) { |
||||
finalRotation.mult(eyePos, finalPosition); |
||||
finalPosition.addLocal(hmdPos); |
||||
if( obsPosition != null ) finalPosition.addLocal(obsPosition); |
||||
finalPosition.y += heightAdjustment; |
||||
cam.setFrame(finalPosition, finalRotation); |
||||
} |
||||
|
||||
/** |
||||
* Handles moving filters from the main view to each eye |
||||
*/ |
||||
public void moveScreenProcessingToEyes() { |
||||
if( rightViewport == null ) return; |
||||
syncScreenProcessing(application.getViewPort()); |
||||
application.getViewPort().clearProcessors(); |
||||
} |
||||
|
||||
/** |
||||
* Sets the two views to use the list of {@link SceneProcessor processors}. |
||||
* @param sourceViewport the {@link ViewPort viewport} that contains the processors to use. |
||||
*/ |
||||
public void syncScreenProcessing(ViewPort sourceViewport) { |
||||
if( rightViewport == null ) return; |
||||
// setup post processing filters
|
||||
if( rightPostProcessor == null ) { |
||||
rightPostProcessor = new FilterPostProcessor(application.getAssetManager()); |
||||
leftPostProcessor = new FilterPostProcessor(application.getAssetManager()); |
||||
} |
||||
// clear out all filters & processors, to start from scratch
|
||||
rightPostProcessor.removeAllFilters(); |
||||
leftPostProcessor.removeAllFilters(); |
||||
leftViewport.clearProcessors(); |
||||
rightViewport.clearProcessors(); |
||||
// if we have no processors to sync, don't add the FilterPostProcessor
|
||||
if( sourceViewport.getProcessors().isEmpty() ) return; |
||||
// add post processors we just made, which are empty
|
||||
leftViewport.addProcessor(leftPostProcessor); |
||||
rightViewport.addProcessor(rightPostProcessor); |
||||
// go through all of the filters in the processors list
|
||||
// add them to the left viewport processor & clone them to the right
|
||||
for(SceneProcessor sceneProcessor : sourceViewport.getProcessors()) { |
||||
if (sceneProcessor instanceof FilterPostProcessor) { |
||||
for(Filter f : ((FilterPostProcessor)sceneProcessor).getFilterList() ) { |
||||
if( f instanceof TranslucentBucketFilter ) { |
||||
// just remove this filter, we will add it at the end manually
|
||||
((FilterPostProcessor)sceneProcessor).removeFilter(f); |
||||
} else { |
||||
leftPostProcessor.addFilter(f); |
||||
// clone to the right
|
||||
Filter f2; |
||||
if(f instanceof FogFilter){ |
||||
f2 = FilterUtil.cloneFogFilter((FogFilter)f); |
||||
} else if (f instanceof CartoonSSAO ) { |
||||
f2 = new CartoonSSAO((CartoonSSAO)f); |
||||
} else if (f instanceof SSAOFilter){ |
||||
f2 = FilterUtil.cloneSSAOFilter((SSAOFilter)f); |
||||
} else if (f instanceof DirectionalLightShadowFilter){ |
||||
f2 = FilterUtil.cloneDirectionalLightShadowFilter(application.getAssetManager(), (DirectionalLightShadowFilter)f); |
||||
} else { |
||||
f2 = f; // dof, bloom, lightscattering etc.
|
||||
} |
||||
rightPostProcessor.addFilter(f2); |
||||
} |
||||
} |
||||
} else if (sceneProcessor instanceof VRDirectionalLightShadowRenderer) { |
||||
// shadow processing
|
||||
// TODO: make right shadow processor use same left shadow maps for performance
|
||||
VRDirectionalLightShadowRenderer dlsr = (VRDirectionalLightShadowRenderer) sceneProcessor; |
||||
VRDirectionalLightShadowRenderer dlsrRight = dlsr.clone(); |
||||
dlsrRight.setLight(dlsr.getLight()); |
||||
rightViewport.getProcessors().add(0, dlsrRight); |
||||
leftViewport.getProcessors().add(0, sceneProcessor); |
||||
} |
||||
} |
||||
// make sure each has a translucent filter renderer
|
||||
leftPostProcessor.addFilter(new TranslucentBucketFilter()); |
||||
rightPostProcessor.addFilter(new TranslucentBucketFilter()); |
||||
} |
||||
|
||||
private void setupCamerasAndViews() { |
||||
// get desired frustrum from original camera
|
||||
Camera origCam = app.getCamera(); |
||||
float fFar = origCam.getFrustumFar(); |
||||
float fNear = origCam.getFrustumNear(); |
||||
|
||||
// if we are using OSVR get the eye info here
|
||||
if( app.getVRHardware() instanceof OSVR ) { |
||||
((OSVR)app.getVRHardware()).getEyeInfo(); |
||||
} |
||||
|
||||
// restore frustrum on distortion scene cam, if needed
|
||||
if( app.isInstanceVRRendering() ) { |
||||
leftCamera = origCam; |
||||
} else if( app.compositorAllowed() == false ) { |
||||
origCam.setFrustumFar(100f); |
||||
origCam.setFrustumNear(1f); |
||||
leftCamera = origCam.clone(); |
||||
prepareCameraSize(origCam, 2f); |
||||
} else { |
||||
leftCamera = origCam.clone(); |
||||
} |
||||
|
||||
leftCamera.setFrustumPerspective(app.getDefaultFOV(), app.getDefaultAspect(), fNear, fFar); |
||||
|
||||
prepareCameraSize(leftCamera, 1f); |
||||
if( app.getVRHardware() != null ) leftCamera.setProjectionMatrix(app.getVRHardware().getHMDMatrixProjectionLeftEye(leftCamera)); |
||||
//org.lwjgl.opengl.GL11.glEnable(org.lwjgl.opengl.GL30.GL_FRAMEBUFFER_SRGB);
|
||||
|
||||
if( !app.isInstanceVRRendering()) { |
||||
leftViewport = setupViewBuffers(leftCamera, LEFT_VIEW_NAME); |
||||
rightCamera = leftCamera.clone(); |
||||
if( app.getVRHardware() != null ){ |
||||
rightCamera.setProjectionMatrix(app.getVRHardware().getHMDMatrixProjectionRightEye(rightCamera)); |
||||
} |
||||
rightViewport = setupViewBuffers(rightCamera, RIGHT_VIEW_NAME); |
||||
} else { |
||||
|
||||
System.err.println("[VRViewManager] THIS CODE NEED CHANGES !!!"); |
||||
leftViewport = application.getViewPort(); |
||||
//leftViewport.attachScene(app.getRootNode());
|
||||
rightCamera = leftCamera.clone(); |
||||
if( app.getVRHardware() != null ){ |
||||
rightCamera.setProjectionMatrix(app.getVRHardware().getHMDMatrixProjectionRightEye(rightCamera)); |
||||
} |
||||
|
||||
org.lwjgl.opengl.GL11.glEnable(org.lwjgl.opengl.GL30.GL_CLIP_DISTANCE0); |
||||
|
||||
//FIXME: [jme-vr] Fix with JMonkey next release
|
||||
//RenderManager._VRInstancing_RightCamProjection = camRight.getViewProjectionMatrix();
|
||||
setupFinalFullTexture(application.getViewPort().getCamera()); |
||||
} |
||||
|
||||
// setup gui
|
||||
app.getVRGUIManager().setupGui(leftCamera, rightCamera, leftViewport, rightViewport); |
||||
|
||||
if( app.getVRHardware() != null ) { |
||||
// call these to cache the results internally
|
||||
app.getVRHardware().getHMDMatrixPoseLeftEye(); |
||||
app.getVRHardware().getHMDMatrixPoseRightEye(); |
||||
} |
||||
|
||||
} |
||||
|
||||
private ViewPort setupMirrorBuffers(Camera cam, Texture tex, boolean expand) { |
||||
Camera clonecam = cam.clone(); |
||||
ViewPort viewPort = application.getRenderManager().createPostView("MirrorView", clonecam); |
||||
clonecam.setParallelProjection(true); |
||||
viewPort.setClearFlags(true, true, true); |
||||
viewPort.setBackgroundColor(ColorRGBA.Black); |
||||
Picture pic = new Picture("fullscene"); |
||||
pic.setLocalTranslation(-0.75f, -0.5f, 0f); |
||||
if( expand ) { |
||||
pic.setLocalScale(3f, 1f, 1f); |
||||
} else { |
||||
pic.setLocalScale(1.5f, 1f, 1f); |
||||
} |
||||
pic.setQueueBucket(Bucket.Opaque); |
||||
pic.setTexture(application.getAssetManager(), (Texture2D)tex, false); |
||||
viewPort.attachScene(pic); |
||||
viewPort.setOutputFrameBuffer(null); |
||||
|
||||
pic.updateGeometricState(); |
||||
|
||||
return viewPort; |
||||
} |
||||
|
||||
private void setupFinalFullTexture(Camera cam) { |
||||
// create offscreen framebuffer
|
||||
FrameBuffer out = new FrameBuffer(cam.getWidth(), cam.getHeight(), 1); |
||||
//offBuffer.setSrgb(true);
|
||||
|
||||
//setup framebuffer's texture
|
||||
dualEyeTex = new Texture2D(cam.getWidth(), cam.getHeight(), Image.Format.RGBA8); |
||||
dualEyeTex.setMinFilter(Texture.MinFilter.BilinearNoMipMaps); |
||||
dualEyeTex.setMagFilter(Texture.MagFilter.Bilinear); |
||||
|
||||
logger.config("Dual eye texture "+dualEyeTex.getName()+" ("+dualEyeTex.getImage().getId()+")"); |
||||
logger.config(" Type: "+dualEyeTex.getType()); |
||||
logger.config(" Size: "+dualEyeTex.getImage().getWidth()+"x"+dualEyeTex.getImage().getHeight()); |
||||
logger.config(" Image depth: "+dualEyeTex.getImage().getDepth()); |
||||
logger.config(" Image format: "+dualEyeTex.getImage().getFormat()); |
||||
logger.config(" Image color space: "+dualEyeTex.getImage().getColorSpace()); |
||||
|
||||
//setup framebuffer to use texture
|
||||
out.setDepthBuffer(Image.Format.Depth); |
||||
out.setColorTexture(dualEyeTex); |
||||
|
||||
ViewPort viewPort = application.getViewPort(); |
||||
viewPort.setClearFlags(true, true, true); |
||||
viewPort.setBackgroundColor(ColorRGBA.Black); |
||||
viewPort.setOutputFrameBuffer(out); |
||||
|
||||
} |
||||
|
||||
private ViewPort setupViewBuffers(Camera cam, String viewName){ |
||||
// create offscreen framebuffer
|
||||
FrameBuffer offBufferLeft = new FrameBuffer(cam.getWidth(), cam.getHeight(), 1); |
||||
//offBufferLeft.setSrgb(true);
|
||||
|
||||
//setup framebuffer's texture
|
||||
Texture2D offTex = new Texture2D(cam.getWidth(), cam.getHeight(), Image.Format.RGBA8); |
||||
offTex.setMinFilter(Texture.MinFilter.BilinearNoMipMaps); |
||||
offTex.setMagFilter(Texture.MagFilter.Bilinear); |
||||
|
||||
//setup framebuffer to use texture
|
||||
offBufferLeft.setDepthBuffer(Image.Format.Depth); |
||||
offBufferLeft.setColorTexture(offTex); |
||||
|
||||
ViewPort viewPort = application.getRenderManager().createPreView(viewName, cam); |
||||
viewPort.setClearFlags(true, true, true); |
||||
viewPort.setBackgroundColor(ColorRGBA.Black); |
||||
|
||||
Iterator<Spatial> spatialIter = application.getViewPort().getScenes().iterator(); |
||||
while(spatialIter.hasNext()){ |
||||
viewPort.attachScene(spatialIter.next()); |
||||
} |
||||
|
||||
//set viewport to render to offscreen framebuffer
|
||||
viewPort.setOutputFrameBuffer(offBufferLeft); |
||||
return viewPort; |
||||
} |
||||
} |
Loading…
Reference in new issue