I’m trying to adapt the brackey portal tutorial to VR, and I’m cloth to it but I still got deformations when I turn my head left or right in front of the portal.
I also got deformations when I look forward standing on the left or right of the portal.
It seems to be a parallax problem that I can’t grasp.
When I’m turning my head, you can see the green pillars of the portal structure appearing behind the grey one.
Here are some explanations of the set up of my scene.
For rendering a correct image inside the portal in VR, I cloned the LeftEyeAnchor & RighEyeAnchor in an other gameobject who hold them.
I’m capturing the original LeftEyeAnchor & RightEyeAnchor rotational datas for applying them to my clones
I render each of this clones view on one plane who have the same position in space as the other, creating a stereoscopic renderTexture.
Here is the script I attached to my clones holder (the game object who holds my two cloned cameras)
using System.Collections;
using System.Collections.Generic;
using UnityEngine;
using UnityEngine.UI;
public class PortalCamera : MonoBehaviour {
public Transform player;
public GameObject headtracker_LEFT;
public GameObject headtracker_RIGHT;
public GameObject CameraB_RIGHT;
public GameObject CameraB_LEFT;
public Vector3 CameraB_Pos;
// LateUpdate is called once after each frame
void LateUpdate()
{
//Parenting B Cameras to player movements
transform.position = player.transform.TransformPoint(CameraB_Pos.x, PlayerScript.globalCameraHeight, CameraB_Pos.z);
//Parenting B Cameras to OVR left and right cameras roation
//Left
CameraB_LEFT.transform.eulerAngles = new Vector3(headtracker_LEFT.transform.eulerAngles.x, headtracker_LEFT.transform.eulerAngles.y, headtracker_LEFT.transform.eulerAngles.z);
//Right
CameraB_RIGHT.transform.eulerAngles = new Vector3(headtracker_RIGHT.transform.eulerAngles.x, headtracker_RIGHT.transform.eulerAngles.y, headtracker_RIGHT.transform.eulerAngles.z);
}
}
I also share my unity project here (enable the Fly Control script attached to the player for moving in the scene in unity, and don’t forget disabled it befor building the app for viewing it in the GO ) fromsmash.com/govrportal
Hey @Untoldecay ,
Tried downloading your project and made sure all the settings were the same as what you told me over on my thread and this is what i saw when I had ran it in editor using the Oculus Rift Headest. Any idea why this isnt working for me?
I have no clues, I’m pretty new to unity so I don’t have a large experience but I’ll try to help.
What are the white areas on each side of the left eye? A bug I suppose but I’m not sure.
Could you take an other screenshot of the scene but at some distance of the portal?
In few days when I’ll have some time and when I’ll solve all my portal related problems, I’ll post my process and all the ressources I found helpful, but here are some leads to begin with.
All the position / rotation tracking who are replicated from the controller(s) or the headset should be placed in the Application.onBeforeRender() function for preventing any lags or delays. I found a german article illustrating it here
The portal image distorsion have something to do with the cameras projection matrix, on which I understand the concept, but not really their integration. I found some code exemples here
Here is a great article about portals, but more in depth than the brackey one ( keep in mind that for VR and stereo you have to use sets of 2 cameras instead of 1 )
Tracking the headset or the remote have to be fast enough to prevent lags, and LateUpdate is not fast enough. I should have used Application.onBeforeRender()
Parallax problems due to stereoscopic view and cameras projection matrix. I needed to clip the renderTexture plane from the concerned camera views ( It was clearly over my head, and it still is ).
There are other points who differs from the Brackey tutorial or the on ( more in depth ) made by Tom Hulton
It could be obvious for the majority, but It came to me after I began the tutorials : VR needs stereoscopic render for create depth, so I had to 2 cameras instead of one for each place where a camera was needed. I just copied the LeftEyeAnchor and RightEyeAnchor cameras for keeping the same settings and used the layer system + culling mask for displaying different content to each eyes.
I you need to cameras, you need 2 overlapping renderTextures for this to work
! You can move the cameras parent position using the player position but you have to rotate each cameras apart. For getting my player cameras rotation &/or position, I placed a dummy camera ( not rendering anything ) as a tracker inside my LeftEyeAnchor & RightEyeAnchor.
By default, disable the MeshRenderer of the destination portal if not needed, and enable it once the teleportation happened.
And overall, I’m not sure but you should use MultiPass Stereo Rendering Method ( in the Player settings of your project ) for this to word.
In the previous version of my code I was calculating the position of player’s camera clones directly, using only transform.TransformPoint(), it seemed shorter and better, but now I came back to calculate it relative to the portal ( as Brackey or Tom Hulton are doing ).
So here is the code on my cameras holders ( the gameobject containing my cloned cameras )
using System.Collections;
using System.Collections.Generic;
using UnityEngine;
using UnityEngine.UI;
public class RelativePortalCamera : MonoBehaviour {
//Referring to the dummy cameras placed inside LeftEyeAnchor & RightEyeAnchor used as position / rotation trackers
public GameObject Player;
public Camera MainCamera_L;
public Camera MainCamera_R;
//Pairs of cameras used for the renderTexture
public GameObject PortalCamera_Holder;
public Camera PortalCamera_L;
public Camera PortalCamera_R;
//Portals
public Transform Source;
public Transform Destination;
//Debug hud
public static Text debugText_PosCompare;
public static Text debugText_EulerRightCompare;
public static Text debugText_EulerLeftCompare;
public static Text debugText_Headset;
void Start()
{
Application.onBeforeRender += OnBeforeRender;
//Debug
debugText_PosCompare = GameObject.Find("debugText_PosCompare").GetComponent<Text>();
debugText_EulerRightCompare = GameObject.Find("debugText_EulerRightCompare").GetComponent<Text>();
debugText_EulerLeftCompare = GameObject.Find("debugText_EulerLeftCompare").GetComponent<Text>();
}
// OneBeforeRender is used for better pos / rotation tracking.
// Putting all the tracking stuff inside should kill all lags.
// https://bit.ly/2OTVi65
void OnBeforeRender() {
/*
----------
TRACKING & CLIP
----------
*/
// Holder position
Vector3 cameraPositionInSourceSpace_L = Source.InverseTransformPoint(MainCamera_L.transform.position);
PortalCamera_Holder.transform.position = Destination.TransformPoint(cameraPositionInSourceSpace_L);
// Left rotation
Quaternion cameraRotationInSourceSpace_L = Quaternion.Inverse(Source.rotation) * MainCamera_L.transform.rotation;
PortalCamera_L.transform.rotation = Destination.rotation * cameraRotationInSourceSpace_L;
// Right rotation
// Vector3 cameraPositionInSourceSpace_R = Source.InverseTransformPoint(MainCamera_R.transform.position);
// PortalCamera_R.transform.position = Destination.TransformPoint(cameraPositionInSourceSpace_R);
Quaternion cameraRotationInSourceSpace_R = Quaternion.Inverse(Source.rotation) * MainCamera_R.transform.rotation;
PortalCamera_R.transform.rotation = Destination.rotation * cameraRotationInSourceSpace_R;
// Cliplane projection matrix
// All the code relative to this part come from https://bit.ly/2Kc6VGN and here https://bit.ly/2WUpwZx
Vector4 clipPlaneB_L = CameraSpacePlane(PortalCamera_L.GetComponent<Camera>(), Destination.transform.position, Destination.transform.forward * -1, 1.0f);
Matrix4x4 projectionB_L = MainCamera_L.GetComponent<Camera>().GetStereoProjectionMatrix(Camera.StereoscopicEye.Left);
CalculateObliqueMatrix(ref projectionB_L, clipPlaneB_L);
PortalCamera_L.GetComponent<Camera>().projectionMatrix = MainCamera_L.GetComponent<Camera>().CalculateObliqueMatrix(clipPlaneB_L);
Vector4 clipPlaneB_R = CameraSpacePlane(PortalCamera_R.GetComponent<Camera>(), Destination.transform.position, Destination.transform.forward * -1, 1.0f);
Matrix4x4 projectionB_R = MainCamera_R.GetComponent<Camera>().GetStereoProjectionMatrix(Camera.StereoscopicEye.Right);
CalculateObliqueMatrix(ref projectionB_R, clipPlaneB_R);
PortalCamera_R.GetComponent<Camera>().projectionMatrix = MainCamera_R.GetComponent<Camera>().CalculateObliqueMatrix(clipPlaneB_R);
/*
----------
DEBUG
----------
*/
//hud
debugText_PosCompare.text = "PlayerPos : " + Player.transform.position + " || HolderPos : " + PortalCamera_Holder.transform.position;
debugText_EulerLeftCompare.text = "MainL : " + MainCamera_L.transform.eulerAngles + " || CloneL : " + PortalCamera_L.transform.eulerAngles;
debugText_EulerRightCompare.text = "MainR : " + MainCamera_R.transform.eulerAngles + " || CloneR : " + PortalCamera_R.transform.eulerAngles;
}
/*
----------
FUNCTIONS
----------
*/
// Extended sign: returns -1, 0 or 1 based on sign of a
private static float sgn(float a)
{
if (a > 0.0f) return 1.0f;
if (a < 0.0f) return -1.0f;
return 0.0f;
}
// Given position/normal of the plane, calculates plane in camera space.
private Vector4 CameraSpacePlane(Camera cam, Vector3 pos, Vector3 normal, float sideSign)
{
Vector3 offsetPos = pos + normal;
Matrix4x4 m = cam.worldToCameraMatrix;
Vector3 cpos = m.MultiplyPoint(offsetPos);
Vector3 cnormal = m.MultiplyVector(normal).normalized * sideSign;
return new Vector4(cnormal.x, cnormal.y, cnormal.z, -Vector3.Dot(cpos, cnormal));
}
// Adjusts the given projection matrix so that near plane is the given clipPlane
// clipPlane is given in camera space. See article in Game Programming Gems 5 and
// http://aras-p.info/texts/obliqueortho.html
private static void CalculateObliqueMatrix(ref Matrix4x4 projection, Vector4 clipPlane)
{
Vector4 q = projection.inverse * new Vector4(
sgn(clipPlane.x),
sgn(clipPlane.y),
1.0f,
1.0f
);
Vector4 c = clipPlane * (2.0F / (Vector4.Dot(clipPlane, q)));
// third row = clip plane - fourth row
projection[2] = c.x - projection[3];
projection[6] = c.y - projection[7];
projection[10] = c.z - projection[11];
projection[14] = c.w - projection[15];
}
}
Is there a reason why your RelativePortalCamera script only works on one portal at a time? I’m trying to have multiple portals in my scene, but the effect only happens to one of the portals - the rest of the other portals are static.
Sorry folks I totally stopped this project.
It was partly working, and the other side of the protal, once the player teleported, wasn’t displaying the rend texture right.
I never continued, but I will post a link to my project files if I got the time ( I have to search / find them in a 2 years mess of files haha )