Methods of keeping player in view

Hi!

Currently I’m messing around with a procedurally generated world with the character running around a forest. There’s also a primitive inventory system, and a way to cut down trees. The trees are automatically populated, along with the monsters.

Here’s what it looks like for now:

However, I’ve run into an issue: Sometimes the camera gets blocked by a tree. Since it’s always following the player, if the player runs behind a tree, you can’t see the player. For example:


Looking at that, you’d never guess that the player is behind that tree, and if you’re fighting an enemy, this is very annoying as you can’t see what you’re doing. I’ve made a script that will raise the camera over the tree to keep the player in view. However, this creates a strange, jarring experience:

So, currently, I’ve thought of 3 paths to go down:
1: Shorten the tall trees so none of them completely block the camera. However, this would mostly kill the “in the forest” effect I’m going for.
2: Make a point-based camera system like in the Walking Dead Telltale games. I’m afraid it might be confusing for players though.
3: Just do nothing and let the camera clip through trees.

What are your ideas on this?

Always a thorny issues. Here are some solutions I’ve seen that aren’t already on your list:

  • When a tree is in the way (detectable via raycast), make it see-through (lower its alpha value).
  • When something is in the way, move the camera closer to the player until it’s not.
  • When something’s in the way, move the camera off to the side and rotate it. For this I imagine the camera is attached to the player via an invisible rod, and you basically do collision detection and response between this rod and the environment.

3 is certainly the trickiest, and might work in only certain kinds of environments. I think in your case I’d like option 1 best, which happily is also the easiest to implement.

5 Likes

This is what most games do in my experience.

Yeah I noticed that, but for me game as shown in the screenshots, that wouldn’t really be an option as its a top/down angled view.

I’m going to try a different camera movement system; if that fails then I’ll probably go with #1 from JoeStrout. This should be a fun challenge :slight_smile:

Then you either scrap the camera for a better one or construct environments around the limitation.

1 Like

Isn’t this a perfect problem for the new cinemachine to deal with?

edit: example, have a second orbit cam at slightly higher angle.

1 Like

Never used Cinemachine, but I should look into it.

I also noticed outlines can be helpful:
3208098--245539--upload_2017-9-4_13-12-56.png

That doesn’t fix the problem, but it makes a hidden player 50% less annoying.

I’m currently working on a lerping camera controller with raycasts; if it’s close to a tree it moves to the side so you can see the player. In theory I have it working but I want to add accelerating lerps to make it smooth. I have free time today, so I’ll try to finish it today and post the result! Fingers crossed.

I wasn’t sure if you were making the camera fixed in some way. In that case I would definitely concur that #1 from JoeStrout is the best idea. I’ll be honest, I might be annoyed if a camera suddenly swung around without my telling it to do so (but then, I also don’t care for it in traditional third-person cameras).

1 Like

Well, after spending most of the day on this, here’s what I came up with:

I like 80% of it; specifically how you can walk around a bit without the camera moving, and also how the camera can move between trees. So far I know of 2 bugs: If the view to the player is blocked by a tree farther than the TreeDodgeDistance away, the camera won’t move. Also, the player can sometimes move out of view to the sides and can only be seen after letting go of the movement keys.

It usually avoids trees, but leaves the small ones alone so the player can still be behind them without the camera moving. I really like that effect and apart from a few bugs, I’m pretty satisfied with the result. I think I need to increase the TreeDodgeDistance so it’ll try to avoid trees farther away, and maybe scale the gigantic ones down slightly.

Also, this was my 1st time using a Tween engine (LeanTween), and I have to say it’s pretty awesome :). I might still look into Cinemachine just to learn it though.

Here’s the rough code, over the next few days, once I have a mostly finished sample I’ll update the code to a polished and well-commented version:

//NOTE: REQUIRES LEANTWEEN. Get it from the Asset Store!
using cakeslice;
using System.Collections;
using UnityEngine;

public class CameraControllerTweenV2 : MonoBehaviour
{
    #region "Variables"
    public PlayerController PlayerControllerScript;
    LeanTween TweenScript;
    LeanTweenType TweenType;
    public Transform Player;
    public Vector3 StartOffset;
    public Vector3 PlayerToCameraDifference;
    public float OffsetToKeepX;
    public float OffsetToKeepZ;
    public float MoveTime;
    public Vector3 PreviousFrameVector;
    public Vector3 DifferecePerFrame;
    public Vector3 ToPlayer;
    private Ray _Ray;
    private RaycastHit _RayHit;
    public bool CoolingDownX;
    public bool CoolingDownZ;
    public bool EnableDirectControlMaster;
    public bool EnableDirectControlX;
    public bool EnableDirectControlZ;
    public bool DodgingTree;
    public bool Override;
    public bool Test;
    public float RayHitDistance;
    public bool PlayerIsVisible;
    public float TreeDodgeDistance;
    #endregion
    void Start()
    {
        StartOffset = transform.position - Player.position;
    }

    void FixedUpdate()
    {
        PlayerToCameraDifference = transform.position - Player.position;
        if (Mathf.Abs(PlayerToCameraDifference.x) > 5f && !CoolingDownX && !EnableDirectControlX)
        {
            LeanTween.cancelAll();
            EnableDirectControlMaster = true;
            EnableDirectControlX = true;
            OffsetToKeepX = transform.position.x - Player.position.x;
        }

        if (Mathf.Abs(PlayerToCameraDifference.z) < 4f  && !CoolingDownZ && !EnableDirectControlZ)
            PrepareZ();
        if (Mathf.Abs(PlayerToCameraDifference.z) > 12f && !CoolingDownZ && !EnableDirectControlZ)
            PrepareZ();

        if (EnableDirectControlMaster)
        {
            if (Mathf.Abs(PlayerToCameraDifference.x) > 5f && EnableDirectControlX)
            {
                //if (TestMovePointForTree(new Vector3((Player.position.x + OffsetToKeepX), transform.position.y, transform.position.z)))
                    transform.position = new Vector3((Player.position.x + OffsetToKeepX), transform.position.y, transform.position.z);
            }
            if (Mathf.Abs(PlayerToCameraDifference.z) < 4f || Mathf.Abs(PlayerToCameraDifference.z) > 12f && EnableDirectControlZ)
            {
                //if (TestMovePointForTree(new Vector3(transform.position.x, transform.position.y, (Player.position.z + OffsetToKeepZ))))
                    transform.position = new Vector3(transform.position.x, transform.position.y, (Player.position.z + OffsetToKeepZ));
            }
            //if stopped moving
            if (!PlayerControllerScript.IsMoving && !Override)
            {
                AttemptToCenter();
            }
        }
        ToPlayer = (Player.position - transform.position).normalized;
        #region "Raycasting"
        if (Physics.Raycast(transform.position, ToPlayer, out _RayHit) && _RayHit.transform.name != "Player")
        {
            GetComponent<OutlineEffect>().enabled = true;
            PlayerIsVisible = false;
        }
        else
        {
            GetComponent<OutlineEffect>().enabled = false;
            PlayerIsVisible = true;
        }
        RayHitDistance = _RayHit.distance;
        if (_RayHit.transform.tag == "Choppable" && _RayHit.distance < TreeDodgeDistance && !Override && !PlayerIsVisible)
        {
            AttemptToCenter();
            Override = true;
            LeanTween.cancelAll();
            //DodgingTree = true;
            if (TestMovePointForTree(new Vector3(_RayHit.transform.position.x + 5, transform.position.y, _RayHit.transform.position.z)) && _RayHit.transform.name != "Player")
            {
                StartLerp(new Vector3(_RayHit.transform.position.x + 5, transform.position.y, _RayHit.transform.position.z));
                Debug.Log("Dodging right " + _RayHit.transform.name + " " + new Vector3(_RayHit.transform.position.x + 5, transform.position.y, _RayHit.transform.position.z));
                return;
            }
            if (TestMovePointForTree(new Vector3(_RayHit.transform.position.x - 5, transform.position.y, _RayHit.transform.position.z)) && _RayHit.transform.name != "Player")
            {
                StartLerp(new Vector3(_RayHit.transform.position.x - 5, transform.position.y, _RayHit.transform.position.z));
                Debug.Log("Dodging right " + _RayHit.transform.name + " " + new Vector3(_RayHit.transform.position.x - 5, transform.position.y, _RayHit.transform.position.z));
                return;
            }
            else
            {
                Debug.Log("Can't dodge this tree because " + new Vector3(_RayHit.transform.position.x + 5, transform.position.y, _RayHit.transform.position.z) + " && "
                    + new Vector3(_RayHit.transform.position.x - 5, transform.position.y, _RayHit.transform.position.z) + " don't pass tree bool inspection");
                Override = false;
            }
        }
        else
            Override = false;
        #endregion
    }

    //when dodging a tree, try this FIRST. That way, you don't have to move to an OK position then immediatly center the camera
    //because centered was also an OK position
    void AttemptToCenter()
    {
        if (TestMovePointForTree(new Vector3(Player.position.x, transform.position.y, Player.position.z + StartOffset.z)))
        {
            Debug.Log("Attempted to center, Override no longer needed");
            LeanTween.cancelAll();
            Override = false;
            StopLerp();
        }
        else
            Debug.Log("Stopped centering due to tree near " + new Vector3(Player.position.x, transform.position.y, Player.position.z + StartOffset.z));
    }

    bool TestMovePointForTree(Vector3 TargetPosition)
    {
        //Debug.Log("Testing: " + TargetPosition);
        if (Physics.Raycast(TargetPosition, (Player.position - TargetPosition).normalized, out _RayHit) && _RayHit.transform.name != "Player")
        {
            if (_RayHit.transform.tag == "Choppable" && _RayHit.distance < TreeDodgeDistance)
            {
                Debug.Log(_RayHit.transform.name + " was in the way!");
                return false;
            }
            else
            {
                Debug.Log(TargetPosition + " was deemed a safe place for the camera");
                return true;
            }
        }
        else
        {
            //Debug.Log("No ray hit from " + TargetPosition);
            return true;
        }
    }

    void PrepareZ()
    {
        LeanTween.cancelAll();
        EnableDirectControlMaster = true;
        EnableDirectControlZ = true;
        OffsetToKeepZ = transform.position.z - Player.position.z;
        //OffsetToKeepZ = Player.position.z - 7;
    }

    void StartLerp(Vector3 Destination)
    {
        float MoveTime = 3f;
        if (Override)
        {
            MoveTime = 5f;
        }
        LeanTween.moveX(gameObject, Destination.x, MoveTime).setEase(TweenType);
        //need start offset for Z so camera won't be directly above player
        if(Override)
            LeanTween.moveZ(gameObject, Destination.z, MoveTime).setEase(TweenType);
        else
            LeanTween.moveZ(gameObject, Destination.z + StartOffset.z, MoveTime).setEase(TweenType);
    }

    void StopLerp()
    {
        TweenType = LeanTweenType.easeOutBack;
        if(!Override)
            StartLerp(Player.position);
        EnableDirectControlMaster = false;
        EnableDirectControlX = false;
        EnableDirectControlZ = false;
    }

    IEnumerator CoolDownXTimer()
    {
        CoolingDownX = true;
        yield return new WaitForSeconds(0.2f);
        CoolingDownX = false;
    }

    IEnumerator CoolDownZTimer()
    {
        CoolingDownZ = true;
        yield return new WaitForSeconds(0.2f);
        CoolingDownZ = false;
    }
}

What do you think?

1 Like

Is this still possible with terrain trees?

Since it’s a procedural system, I’m spawning tree prefabs that are tagged “Choppable”. I don’t know of a way to interact with individual Terrain trees, so with this script, no.

Unless you try one of the rather complicated solutions here:
https://forum.unity3d.com/threads/interact-with-terrain-trees.301470/

And that is why I never use trees in the Terrain system :face_with_spiral_eyes:

1 Like

I haven’t either, and I should also look into it. Just haven’t had the use for it yet.
But it would be interesting to see how it used, considering how they (unity) described it to me.

  • Clear Shot Real-time shot evaluation. Setup any number of cameras and give them a priority. If the camera becomes occluded or can’t make a good shot, Cinemachine will cut (or blend) to the next highest priority shot.
1 Like

OH MY GOD. I don’t think anyone developing in Unity is ever going to write a camera controller again.

After 2 hours of watching Unite 2017 Austin Cinemachine demos to convince myself to dive in, I spent another 20 minutes watching tutorials and came up with this:

https://www.youtube.com/watch?v=Ex0GqGjRCGU

For so little work and absolutely ZERO (null) (0) scripting, I was able to achieve such an amazing result. My script above, I probably spent 15 hours on 5 different revisions, even using a tweening engine, and even if it worked bug-free it still wouldn’t achieve this result. I’m not the world’s best programmer so I’m sure it would be possible to do this without Cinemachine, but this is just so easy and so fast.

With the Curb Feelers option, you have to try hard to get the camera to clip something. I’m just amazed. Huge thanks @Kemonono

5 Likes

You can do that occluding with a very simple shader. You don’t want to do what you are doing. Sorry don’t have the code in front of me right now I’ll try and post it tomorrow.

Shader "Custom/StandardOccluded"
{
    Properties {
        _Color ("Color", Color) = (1,1,1,1)
        _MainTex ("Albedo (RGB)", 2D) = "white" {}
        _Glossiness ("Smoothness", Range(0,1)) = 0.5
        _Metallic ("Metallic", Range(0,1)) = 0.0
        _OccludedColor("Occluded Color", Color) = (1,1,1,1)
    }
    SubShader {
  
        Pass
        {
            Tags { "Queue"="Geometry+1" }
            Blend SrcAlpha OneMinusSrcAlpha
            ZTest Greater
            ZWrite Off

            CGPROGRAM
            #pragma vertex vert           
            #pragma fragment frag
            #pragma fragmentoption ARB_precision_hint_fastest

            half4 _OccludedColor;

            float4 vert(float4 pos : POSITION) : SV_POSITION
            {
                float4 viewPos = UnityObjectToClipPos(pos);
                return viewPos;
            }

            half4 frag(float4 pos : SV_POSITION) : COLOR
            {
                return _OccludedColor;
            }

            ENDCG
        }

        Tags { "RenderType"="Opaque" "Queue"="Geometry+1"}
        LOD 200
        ZWrite On
        ZTest LEqual
      
        CGPROGRAM
        // Physically based Standard lighting model, and enable shadows on all light types
        #pragma surface surf Standard fullforwardshadows

        // Use shader model 3.0 target, to get nicer looking lighting
        #pragma target 3.0

        sampler2D _MainTex;

        struct Input {
            float2 uv_MainTex;
        };

        half _Glossiness;
        half _Metallic;
        fixed4 _Color;

        void surf (Input IN, inout SurfaceOutputStandard o) {
            // Albedo comes from a texture tinted by color
            fixed4 c = tex2D (_MainTex, IN.uv_MainTex) * _Color;
            o.Albedo = c.rgb;
            // Metallic and smoothness come from slider variables
            o.Metallic = _Metallic;
            o.Smoothness = _Glossiness;
            o.Alpha = c.a;
        }
        ENDCG
    }
    FallBack "Diffuse"
}

@snacktime , I have to confess I don’t understand how this shader works or how to use it. Can you explain?

Thanks,

  • Joe

Personally, here how I would have tackled this problem by design.

I would have made the trees into 3 possible versions :

  1. The normal Tree,
  2. A partial version of the tree,
  3. The cut version of the tree.

The way it can be handle depends on what you want to do.

It could be via raycast between the camera and the player or simply by making a system that detect all trees between the player and the camera. Then depending on what method you use, you can switch the “normal” trees by their “partial” version.

Both ways requires a bit of finesse as to how to handle them in an optimal way, but they are still relatively reliable and easy to do.

The key part in this method I’m writing about is that you can control how the tree that are between the player and the camera looks. If you want, you could even make the upper part of them kinda shadowy and semi-transparent or something.

Personally, I would avoid forcing the camera around when in collision with any trees. Not that it’s should never be done, but simply that it should mainly only be done when the space around the trees or any collision is sufficient. When the camera turn or the player move, if the camera “hit” a tree too often and start moving, it can get really chaotic and the could even make your players sick. (You know those game that are critiqued for their bad camera? Same comment would come if your camera keeps moving around because it hit tons of stuff in the world.)

This is why, usually, you find camera that “avoid” obstacles between the player and the said camera in games that have really short distance between both by default. Games in 3rd person like Skyrim, Legend of Zelda, Dark Souls and so on. Your game seem more closer to games like Diablo and Grim Dawn in how the view is presented.

Just a tip here. :slight_smile:

2 Likes

Correct me if I’m wrong, but don’t those games tend to avoid the issue rather than solve it? I don’t remember there being a lot of large, camera occluding objects in any of the Diablo games I’ve played, or any of their clones. The camera is at a fairly sharp downward angle to minimise the effect, it’s possibly ortho rather than perspective, and there are few if any objects so tall as to occlude more than a character’s worth of space in the play area.

When there is stuff that would otherwise occlude the camera, it’s stuff like roofs that can be turned off or made really transparent when you go under it.

1 Like

If I’m reading it right (and I admit my reading of shaders is still sketchy) this shader should be applied to the player.

If the player is hidden, it will draw them in a flat _OcculdedColour. If the player is not hidden it will draw them normally.

You could use the same principle to draw the player in any number of ways.

1 Like