Hide Objects Blocking Player View

Published August 10, 2013 by Christopher Rolfs, posted by irrationalistic
Do you see issues with this article? Let us know.
Advertisement
In most isometric-esque RTS-style game views, sometimes pesky walls can pop into view, hindering the player's connection with the protagonist. This isn't good but is easily solvable with many different solutions. One, highlight the player's outline when it is behind objects. Two, hide the interfering objects. Three, masking shaders to overlay over the screen. There are, of course, many other solutions, but I'll only explore these in this post through the use of Unity.

Highlight The Player

This method already has several existing code examples. Found on the UnifyCommunity wiki, the shader will draw an outline for the player or fill when part of the player is hidden: Silhouette-Outlined Diffuse This method is effective since it requires very little calculation and handles all possible situations. The con to this system is that you don't get a very good viewpoint of anything else on the other side of the wall including obstacles or enemies.

Hide The Interfering Objects

Another very simple execution with the right use of C# and/or shaders, this method will create a shader that adds transparency to overlapping objects and maintains visual understanding of the scene. First your script needs to detect all blocking objects. Just attach this to your camera and tell it to watch for the player, using walls and other hideable objects as the layermask. This will disable rendering the walls in the way, but you'll still collide with them. using UnityEngine; using System.Collections; using System.Collections.Generic; public class HideObjects : MonoBehaviour { public Transform WatchTarget; public LayerMask OccluderMask; private List _LastTransforms; void Start () { _LastTransforms = new List(); } void Update () { //reset and clear all the previous objects if(_LastTransforms.Count > 0){ foreach(Transform t in _LastTransforms) t.GetComponent().enabled = true; _LastTransforms.Clear(); } //Cast a ray from this object's transform the the watch target's transform. RaycastHit[] hits = Physics.RaycastAll( transform.position, WatchTarget.transform.position - transform.position, Vector3.Distance(WatchTarget.transform.position, transform.position), OccluderMask ); //Loop through all overlapping objects and disable their mesh renderer if(hits.Length > 0){ foreach(RaycastHit hit in hits){ if(hit.collider.gameObject.transform != WatchTarget && hit.collider.transform.root != WatchTarget){ hit.collider.gameObject.GetComponent().enabled = false; _LastTransforms.Add(hit.collider.gameObject.transform); } } } } }

Use Shaders

Finally, a nicer-looking option that retains shadows is to use shaders to de-occlude your walls. The shader is essentially adding transparency to the render while the object continues to cast shadows. The shader, which works wonders, was supplied by user ScroodgeM on UnityAnswers: Shader "Transparent/Diffuse with Shadow" { Properties { _Color ("Main Color", Color) = (1,1,1,1) _MainTex ("Base (RGB) Trans (A)", 2D) = "white" {} } SubShader { Tags { "Queue"="Transparent" "IgnoreProjector"="True" "RenderType"="Transparent" } LOD 200 Blend SrcAlpha OneMinusSrcAlpha CGPROGRAM #pragma surface surf Lambert addshadow sampler2D _MainTex; fixed4 _Color; struct Input { float2 uv_MainTex; }; void surf (Input IN, inout SurfaceOutput o) { fixed4 c = tex2D(_MainTex, IN.uv_MainTex) * _Color; o.Albedo = c.rgb; o.Alpha = c.a; } ENDCG } Fallback "Transparent/VertexLit" } And the code that will toggle the shader in its basic entirety: using UnityEngine; using System.Collections; using System.Collections.Generic; public class HideObjects : MonoBehaviour { public Transform WatchTarget; public LayerMask OccluderMask; //This is the material with the Transparent/Diffuse With Shadow shader public Material HiderMaterial; private Dictionary _LastTransforms; void Start () { _LastTransforms = new Dictionary(); } void Update () { //reset and clear all the previous objects if(_LastTransforms.Count > 0){ foreach(Transform t in _LastTransforms.Keys){ t.renderer.material = _LastTransforms[t]; } _LastTransforms.Clear(); } //Cast a ray from this object's transform the the watch target's transform. RaycastHit[] hits = Physics.RaycastAll( transform.position, WatchTarget.transform.position - transform.position, Vector3.Distance(WatchTarget.transform.position, transform.position), OccluderMask ); //Loop through all overlapping objects and disable their mesh renderer if(hits.Length > 0){ foreach(RaycastHit hit in hits){ if(hit.collider.gameObject.transform != WatchTarget && hit.collider.transform.root != WatchTarget){ _LastTransforms.Add(hit.collider.gameObject.transform, hit.collider.gameObject.renderer.material); hit.collider.gameObject.renderer.material = HiderMaterial; } } } } } Just attach the script to your main camera, telling it to watch the player, use the walls and other occluding objects as the layermask, and pass it a material with the hiding shader attached. It will hotswap the materials on the fly. The result is a very clear indication of where walls are and how they are hiding the player. Don't forget to set the opacity down on your material! HideObjects_4.jpg HideObjects_3.jpg

Conclusion

I prefer the shader version since I want to be able to animate the fade and maintain visuals of obstacles. Either way, there are many more options to choose from such as forcing the viewpoint or changing the visuals with a blended shader. As always, check out the options and pick the one that best fits your project and timeline. 12 Jun 2013: Initial release
Cancel Save
0 Likes 0 Comments

Comments

Nobody has left a comment. You can be the first!
You must log in to join the conversation.
Don't have a GameDev.net account? Sign up!

In most isometric-esque RTS-style game views, sometimes pesky walls can pop into view, hindering the player

Advertisement

Other Tutorials by irrationalistic

irrationalistic has not posted any other tutorials. Encourage them to write more!
Advertisement