In our project, we decided to implement portals. A portal can have many definitions, but for this project, we took inspiration from the popular video game series developed by Valve: Portal and Portal 2. Here, portals are essentially an open doorway that transports you to a different location or environment. They allow you to peer into the environment as seen by the other portal, without needing to actually enter it. Our overall goals for this project include experimenting with ray tracing and ray casting for the 3D objects in the scene. We wanted to see how the portals would handle rendering these objects. We also wanted to implement portal teleportation, to experiment with manipulating the player's position. Furthermore, we wanted to make the portals more vibrant than just a simple projection and wanted to inject more life into the scene. Because of this, we implemented a couple of custom shaders and applied them to the portal and to 3D objects.
The portal placement is calculated by casting a ray from where the player clicks. We check for collisions from the ray to an object. If the ray collides with an object, then we place a portal down on the object with a slight offset, so that the portal does not get buried in the object.
If the portal placement is targeted on a portal, the portal will not be placed. However, our current implementation does allow portals to overlap if targeted right next to the portal, which may cause the portals looks to mix together.
The left mouse button spawns the orange portal, while the right mouse button spawns the blue portal. The portals are deactivated and hidden at the start, then activated at the first click of either. If a portal has already spawned in the scene, it will despawn from the original location and respawn to the new location.
For being able to see through the portals, we employ a couple of camera tricks. We have two cameras set up, one for each portal. Each of these cameras relays what they are viewing to its partner portal. We must update both camera's position in relation to the player's position. This works by flipping the portal camera's and positioning the camera to the inverse of the player's current position. We must then rotate the camera afterwards. This allows us to maintain a nice view in each portal that does not look disjointed.
The math for one portal's camera position can be summarized using the following matrix multiplication formula: $$ C_{portalCam} = P_{out} T_{yaxis\_180\_flip} P_{in}^{-1} C_{player} $$ where:
It's rotation can then be determined by the following, with the formula being similar to position but for Rotation instead (noted by the "R" for rotation): $$ R_{portalCam} = R_{outportal} T_{yaxis\_180\_flip} R_{inportal}^{-1} R_{player} $$
When the player enters the portal, it should teleport them to the other portal. Our implementation uses a collider to detect when the player touches the portal and then teleports the player to the other portal, while keeping the orientation of the player.
When teleporting the player through the portal, two things are calculated: position and rotation.
For position, we take the player's coordinates in terms of the entry portal and transform that to the destination portal with a small forward offset applied.
The math for calculating the player's teleported position can be summarized using the following matrix multiplication formula: $$ O_{position\_inportal} = P_{in}^{-1} O_{prev} $$ $$ smallFwdOffset = {teleportOffset} * {outPortal\_forward\_vector} $$ $$ O_{new} = P_{out} O_{position\_inportal} + smallFwdOffset $$ where:
We create recursive portals by rendering a subportal up to a limit set by us (we used 7). The implementation for this is very similar to that of Portal Viewing, but in the perspective of the portal that is being recursed upon. The math is the same thing as when viewing the portal. We do this up to the current recursion depth so that the camera is angled and positioned properly for that specific step. We could have saved the matrix calculations to save on time, but we didn't have time to make that.
The math will be as follows: $$ C_{{inportalCam}_{(t + 1)}} = P_{out} T_{yaxis\_180\_flip} P_{in}^{-1} C_{{inportalCam}_{(t)}} $$ where \(t = 0\) starts off from \(C_{{inportalCam}_{(0)}}\) calculated starting from the Player Camera as written in Portal Viewing. We stop once we reach \(t\) = recursiveLimit.
The equation we used for the Blinn-Phong shader: $$L = (k_a * I_a) + (k_d * (I /r^2) * max(0, n * l)) + (k_s * (I/r^2) * max(0, n * h) ^ p)$$
For the shaders, we apply what we learned in project 4 to create a Blinn-Phong Unlit Shader. We have also created a surface shader that wraps objects in light. We first calculate a Fresnel coefficient by taking the dot product of the view direction and the normal of the object's vertex. ka, kd, and ks are the ambiance, diffuse, and spectural coefficients respectively. \(I/r^2\) is the irradiance from the light source to the vertex we are sampling. n is the normal of the point and l is the magnitude of the light source to the vertex.
The formula used to determine the Fresnel term is: $$R = max(0, min(1, bias + scale * (1.0 + I \cdot N) * power))$$
We then raise it to a power that is tuned manually. We linearly interpolate the texture and emission color using this coefficient and set our object's emission equal to that. This works well for spherical/cylindrical objects with smooth, round edges, but not as well for shapes such as planes or cubes. You can see the result of this work applied to the capsule of the scene.
We implemented a second emissive shader that gives each of the portals a glowing orange or blue rim that becomes lighter towards the center of the portal. We find the distance between the center of the object and each vertex, and use this distance to interpolate between different emitted colors. This shader works well for flat-faced objects like planes and cubes.
One problem we encountered when implementing the emissive portal shader was that the glowing fresnel shader carried over poorly directly from the capsule to the flat portal plane. The fresnel shader was effective on round capsules because the glowing effect becomes more pronounced as the surface curves away from the viewer. However, it was ineffective on flat portals that generally face the player directly. To resolve this, we implemented a second shader for portals that uses UV coordinates to detect proximity to an edge and adjust emitted color accordingly.
Another problem overall was the lack of resources on how to write unity shaders. Many of the online examples of shaders we wanted to implement or tried to gain inspiration from were done using the built in shader lab in unity. They didn't provide any equations that they used to base their shader off of. Even in tutorials that had well defined equations, we struggled due to lack of documentation. We weren't sure what parameters were needed and what their types were, which made shaders a lot harder to implement than expected. We did get help from Mark Zhang, bless his soul. All these issues we ran into caused us to only be able to implement three shaders in total. It's a bit disappointing we couldn't implement more, but regardless we are still happy and proud with what we accomplished.
Occasionally, the player would travel straight through a portal and not teleport out of the other connected portal. One cause of this problem was that teleportation checks every frame for input and updates the player's position based on the input. Thus, the framerate of the game will affect whether or not the player will teleport. Our solution to this problem was to run the teleport code in Unity's FixedUpdate which runs at a fixed interval and can run several times per frame. This is much better as the framerate does not affect whether or not the player will teleport.
Speaking of framerate, our portal views were causing performance issues on some of our machines. The performance issue was further compounded with recursive portal rendering. Thus, we had to implement a few compromises to our portals. One compromise was to reduce the recursion depth on our recursive portal rendering. At a loss of being able to see more of the same portal, the player will notice an increase in performance. Another compromise was to lower the resolution on the views being rendered on the portals. While a drastic increase in performance, there is a definite noticeable decrease in quality when looking at the rendered views on the portals.
There were many problems that arose during the implementation of recursive portals. One problem was getting the sub portals to render in the proper position in real-time because otherwise the deeper sub-portals would move out of sync with the closer portals. Our solution was to reassign the original camera rendering the portal view multiple times with the same offset applied each time.
As a result of working on this project, we have learned many lessons. One major lesson was learning how to develop in Unity with version control. As a result, we learned how to handle merge conflicts with Unity assets and especially Unity scenes. Additionally, we learned how to utilize software engineering principles to organize and prioritize tasks. We would setup a document that plans what we need to prioritize along with stretch goals to split the work among us. Furthermore, we learned how to implement shaders in Unity and experimented with converting our Blinn-Phong shader from project 4 into Unity. We learned to develop in C# to implement character movement, portal placement, recursive portal rendering, and portal teleportation.
Our final interactive portals demo is linked below. We have implemented an interactive, real-time 3D portal simulation in Unity, where the first-person player can move around the scene using WASD keys and place portals on surfaces by left or right clicking. The player can teleport between portals by walking through them. Recursive portal images can be seen when they are placed facing each other.
We also implemented a variety of shaders to enhance the appearance of the scene. For round objects like capsules and spheres, we created a glowing fresnel shader that takes in the surface normal and the view direction, and increases brightness closer to the outer edges of the object. The result is a glowing appearance, regardless of the angle the object is viewed from.
For the orange and blue edges of the portals, we implemented a shader that gradually shifts the emitted color as it approaches the edge of the portal face.
Interactive demoWASD to move, left/right click to place portals. If the cursor is off-center or behaving abnormally, press Esc and then click within the Unity player to realign the cursor with the mouse. Chrome is recommended.
Oscar worked on portal design, recursive portals, and portals shader. Additional contributions were made to each of the deliverables such as the proposal, milestone status report, final presentation, and final report webpage.
Raymond worked on implementing portal placement, portal teleportation, and updating the portal views. Additional contributions were made to each of the deliverables such as the proposal, milestone status report, final presentation, and final report webpage.
Daniel worked on movement with Ashley, implementing shaders, and despawning portals out of view/respawning them when in view again as an optimization. Additional contributions were made to each of the deliverables such as the proposal, milestone status report, final presentation, and final report webpage.
Ashley worked on implementing player and camera movement, implementing shaders with Daniel, constructing the demo Unity scene, and recording and editing all videos. Additional contributions were made to each of the deliverables such as the proposal, milestone status report, final presentation, and final report webpage.