2021-01-03 12:38:02 +00:00
|
|
|
using System;
|
|
|
|
using System.Collections.Generic;
|
|
|
|
using UnityEngine;
|
|
|
|
using UnityEngine.Rendering;
|
|
|
|
|
|
|
|
namespace Popcron
|
|
|
|
{
|
2021-01-04 17:31:58 +00:00
|
|
|
public class GizmosInstance : MonoBehaviour
|
|
|
|
{
|
|
|
|
private const int DefaultQueueSize = 4096;
|
|
|
|
|
|
|
|
private static GizmosInstance instance;
|
|
|
|
private static Material defaultMaterial;
|
|
|
|
|
|
|
|
private Material overrideMaterial;
|
|
|
|
private int queueIndex = 0;
|
|
|
|
private int lastFrame;
|
|
|
|
private Element[] queue = new Element[DefaultQueueSize];
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
/// The material being used to render
|
|
|
|
/// </summary>
|
|
|
|
public static Material Material
|
|
|
|
{
|
|
|
|
get
|
|
|
|
{
|
2021-01-05 15:58:12 +00:00
|
|
|
var inst = GetOrCreate();
|
2021-01-04 17:31:58 +00:00
|
|
|
if (inst.overrideMaterial)
|
|
|
|
{
|
|
|
|
return inst.overrideMaterial;
|
|
|
|
}
|
|
|
|
|
|
|
|
return DefaultMaterial;
|
|
|
|
}
|
|
|
|
set
|
|
|
|
{
|
2021-01-05 15:58:12 +00:00
|
|
|
var inst = GetOrCreate();
|
2021-01-04 17:31:58 +00:00
|
|
|
inst.overrideMaterial = value;
|
|
|
|
}
|
|
|
|
}
|
2021-01-03 12:38:02 +00:00
|
|
|
|
2021-01-04 17:31:58 +00:00
|
|
|
/// <summary>
|
|
|
|
/// The default line renderer material
|
|
|
|
/// </summary>
|
|
|
|
public static Material DefaultMaterial
|
|
|
|
{
|
|
|
|
get
|
|
|
|
{
|
|
|
|
if (!defaultMaterial)
|
|
|
|
{
|
|
|
|
// Unity has a built-in shader that is useful for drawing
|
|
|
|
// simple colored things.
|
2021-01-06 11:52:43 +00:00
|
|
|
var shader = Shader.Find("UI/Default");
|
2021-01-04 17:31:58 +00:00
|
|
|
defaultMaterial = new Material(shader)
|
|
|
|
{
|
|
|
|
hideFlags = HideFlags.HideAndDontSave
|
|
|
|
};
|
|
|
|
|
|
|
|
// Turn on alpha blending
|
2021-01-06 11:52:43 +00:00
|
|
|
defaultMaterial.SetInt("unity_GUIZTestMode", (int)CompareFunction.Always);
|
2021-01-04 17:31:58 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return defaultMaterial;
|
|
|
|
}
|
|
|
|
}
|
2021-01-03 12:38:02 +00:00
|
|
|
|
2021-01-04 17:31:58 +00:00
|
|
|
internal static GizmosInstance GetOrCreate()
|
|
|
|
{
|
2021-01-06 11:52:43 +00:00
|
|
|
if (!instance)
|
2021-01-04 17:31:58 +00:00
|
|
|
{
|
2021-01-05 15:58:12 +00:00
|
|
|
var gizmosInstances = FindObjectsOfType<GizmosInstance>();
|
|
|
|
for (var i = 0; i < gizmosInstances.Length; i++)
|
2021-01-04 17:31:58 +00:00
|
|
|
{
|
|
|
|
instance = gizmosInstances[i];
|
|
|
|
|
|
|
|
//destroy any extra gizmo instances
|
|
|
|
if (i > 0)
|
|
|
|
{
|
2021-01-06 11:52:43 +00:00
|
|
|
Destroy(gizmosInstances[i]);
|
2021-01-04 17:31:58 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//none were found, create a new one
|
|
|
|
if (!instance)
|
|
|
|
{
|
|
|
|
instance = new GameObject(typeof(GizmosInstance).FullName).AddComponent<GizmosInstance>();
|
|
|
|
instance.gameObject.hideFlags = HideFlags.HideInHierarchy | HideFlags.HideInInspector;
|
2021-01-05 15:56:14 +00:00
|
|
|
}
|
2021-01-04 17:31:58 +00:00
|
|
|
}
|
2021-01-03 12:38:02 +00:00
|
|
|
|
2021-01-04 17:31:58 +00:00
|
|
|
return instance;
|
|
|
|
}
|
2021-01-03 12:38:02 +00:00
|
|
|
|
2021-01-06 11:52:43 +00:00
|
|
|
private float CurrentTime => Time.time;
|
2021-01-03 12:38:02 +00:00
|
|
|
|
2021-01-04 17:31:58 +00:00
|
|
|
/// <summary>
|
|
|
|
/// Submits an array of points to draw into the queue.
|
|
|
|
/// </summary>
|
|
|
|
internal static void Submit(Vector3[] points, Color? color, bool dashed)
|
|
|
|
{
|
2021-01-05 15:58:12 +00:00
|
|
|
var inst = GetOrCreate();
|
2021-01-04 17:31:58 +00:00
|
|
|
|
|
|
|
//if new frame, reset index
|
|
|
|
if (inst.lastFrame != Time.frameCount)
|
|
|
|
{
|
|
|
|
inst.lastFrame = Time.frameCount;
|
|
|
|
inst.queueIndex = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
//excedeed the length, so make it even bigger
|
|
|
|
if (inst.queueIndex >= inst.queue.Length)
|
|
|
|
{
|
2021-01-05 15:58:12 +00:00
|
|
|
var bigger = new Element[inst.queue.Length + DefaultQueueSize];
|
|
|
|
for (var i = inst.queue.Length; i < bigger.Length; i++)
|
2021-01-04 17:31:58 +00:00
|
|
|
{
|
|
|
|
bigger[i] = new Element();
|
|
|
|
}
|
|
|
|
|
|
|
|
Array.Copy(inst.queue, 0, bigger, 0, inst.queue.Length);
|
|
|
|
inst.queue = bigger;
|
|
|
|
}
|
|
|
|
|
|
|
|
inst.queue[inst.queueIndex].color = color ?? Color.white;
|
|
|
|
inst.queue[inst.queueIndex].points = points;
|
|
|
|
inst.queue[inst.queueIndex].dashed = dashed;
|
|
|
|
|
|
|
|
inst.queueIndex++;
|
|
|
|
}
|
2021-01-03 12:38:02 +00:00
|
|
|
|
2021-01-04 17:31:58 +00:00
|
|
|
private void OnEnable()
|
|
|
|
{
|
|
|
|
//populate queue with empty elements
|
|
|
|
queue = new Element[DefaultQueueSize];
|
2021-01-05 15:58:12 +00:00
|
|
|
for (var i = 0; i < DefaultQueueSize; i++)
|
2021-01-04 17:31:58 +00:00
|
|
|
{
|
|
|
|
queue[i] = new Element();
|
|
|
|
}
|
|
|
|
|
2021-01-06 11:52:43 +00:00
|
|
|
Camera.onPostRender += OnRendered;
|
2021-01-04 17:31:58 +00:00
|
|
|
}
|
2021-01-03 12:38:02 +00:00
|
|
|
|
2021-01-18 12:33:07 +00:00
|
|
|
private void OnDisable() => Camera.onPostRender -= OnRendered;
|
2021-01-03 12:38:02 +00:00
|
|
|
|
2021-03-07 09:27:23 +00:00
|
|
|
private void Update() =>
|
2021-01-03 12:38:02 +00:00
|
|
|
//always render something
|
|
|
|
Gizmos.Line(default, default);
|
|
|
|
|
2021-01-04 17:31:58 +00:00
|
|
|
private void OnRendered(Camera camera)
|
|
|
|
{
|
2021-01-06 11:52:43 +00:00
|
|
|
Material.SetPass(0);
|
2021-01-04 17:31:58 +00:00
|
|
|
|
2021-01-05 15:58:12 +00:00
|
|
|
var offset = Gizmos.Offset;
|
2021-01-04 17:31:58 +00:00
|
|
|
|
|
|
|
GL.PushMatrix();
|
|
|
|
GL.MultMatrix(Matrix4x4.identity);
|
|
|
|
GL.Begin(GL.LINES);
|
|
|
|
|
2021-01-05 15:58:12 +00:00
|
|
|
var alt = CurrentTime % 1 > 0.5f;
|
|
|
|
var dashGap = Mathf.Clamp(Gizmos.DashGap, 0.01f, 32f);
|
|
|
|
var points = new List<Vector3>();
|
2021-01-04 17:31:58 +00:00
|
|
|
|
|
|
|
//draw le elements
|
2021-01-05 15:58:12 +00:00
|
|
|
for (var e = 0; e < queueIndex; e++)
|
2021-01-04 17:31:58 +00:00
|
|
|
{
|
|
|
|
//just in case
|
|
|
|
if (queue.Length <= e)
|
|
|
|
{
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2021-01-05 15:58:12 +00:00
|
|
|
var element = queue[e];
|
2021-01-04 17:31:58 +00:00
|
|
|
|
|
|
|
points.Clear();
|
|
|
|
if (element.dashed)
|
|
|
|
{
|
|
|
|
//subdivide
|
2021-01-05 15:58:12 +00:00
|
|
|
for (var i = 0; i < element.points.Length - 1; i++)
|
2021-01-04 17:31:58 +00:00
|
|
|
{
|
2021-01-05 15:58:12 +00:00
|
|
|
var pointA = element.points[i];
|
|
|
|
var pointB = element.points[i + 1];
|
|
|
|
var direction = pointB - pointA;
|
2021-01-04 17:31:58 +00:00
|
|
|
if (direction.sqrMagnitude > dashGap * dashGap * 2f)
|
|
|
|
{
|
2021-01-05 15:58:12 +00:00
|
|
|
var magnitude = direction.magnitude;
|
|
|
|
var amount = Mathf.RoundToInt(magnitude / dashGap);
|
2021-01-04 17:31:58 +00:00
|
|
|
direction /= magnitude;
|
|
|
|
|
2021-01-05 15:58:12 +00:00
|
|
|
for (var p = 0; p < amount - 1; p++)
|
2021-01-04 17:31:58 +00:00
|
|
|
{
|
|
|
|
if (p % 2 == (alt ? 1 : 0))
|
|
|
|
{
|
2021-01-05 15:58:12 +00:00
|
|
|
var startLerp = p / (amount - 1f);
|
|
|
|
var endLerp = (p + 1) / (amount - 1f);
|
|
|
|
var start = Vector3.Lerp(pointA, pointB, startLerp);
|
|
|
|
var end = Vector3.Lerp(pointA, pointB, endLerp);
|
2021-01-04 17:31:58 +00:00
|
|
|
points.Add(start);
|
|
|
|
points.Add(end);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
points.Add(pointA);
|
|
|
|
points.Add(pointB);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
points.AddRange(element.points);
|
|
|
|
}
|
|
|
|
|
|
|
|
GL.Color(element.color);
|
2021-01-05 15:58:12 +00:00
|
|
|
for (var i = 0; i < points.Count; i++)
|
2021-01-04 17:31:58 +00:00
|
|
|
{
|
|
|
|
GL.Vertex(points[i] + offset);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
GL.End();
|
|
|
|
GL.PopMatrix();
|
|
|
|
}
|
|
|
|
}
|
2021-01-03 12:38:02 +00:00
|
|
|
}
|