Support complex linear gradients, gradient zoom, and a switch of whether to modify vertices

release
David Gileadi 2017-08-12 15:41:24 -07:00
parent 1f16a65f0f
commit 48acafbb85
1 changed files with 465 additions and 193 deletions

View File

@ -5,257 +5,529 @@
/// -Offset is now limited to -1,1 /// -Offset is now limited to -1,1
/// -Multiple color blend modes /// -Multiple color blend modes
/// ///
/// Remember that the colors are applied per-vertex so if you have multiple points on your gradient where the color changes and there aren't enough vertices, you won't see all of the colors. /// Remember that for radial and diamond gradients, colors are applied per-vertex so if you have multiple points on your gradient where the color changes and there aren't enough vertices, you won't see all of the colors.
/// </summary> /// </summary>
using System;
using System.Collections.Generic; using System.Collections.Generic;
namespace UnityEngine.UI.Extensions namespace UnityEngine.UI.Extensions
{ {
[AddComponentMenu("UI/Effects/Extensions/Gradient2")] [AddComponentMenu("UI/Effects/Extensions/Gradient2")]
public class Gradient2 : BaseMeshEffect { public class Gradient2 : BaseMeshEffect
[SerializeField] {
Type _gradientType; [SerializeField]
Type _gradientType;
[SerializeField] [SerializeField]
Blend _blendMode = Blend.Multiply; Blend _blendMode = Blend.Multiply;
[SerializeField] [SerializeField]
[Range(-1, 1)] [Tooltip("Add vertices to display complex gradients. Turn off if your shape is already very complex, like text.")]
float _offset = 0f; bool _modifyVertices = true;
[SerializeField] [SerializeField]
UnityEngine.Gradient _effectGradient = new UnityEngine.Gradient() { colorKeys = new GradientColorKey[] { new GradientColorKey(Color.black, 0), new GradientColorKey(Color.white, 1) } }; [Range(-1, 1)]
float _offset = 0f;
#region Properties [SerializeField]
public Blend BlendMode { [Range(0.1f, 10)]
get { return _blendMode; } float _zoom = 1f;
set
{
_blendMode = value;
graphic.SetVerticesDirty();
}
}
public UnityEngine.Gradient EffectGradient { [SerializeField]
get { return _effectGradient; } UnityEngine.Gradient _effectGradient = new UnityEngine.Gradient() { colorKeys = new GradientColorKey[] { new GradientColorKey(Color.black, 0), new GradientColorKey(Color.white, 1) } };
set
{
_effectGradient = value;
graphic.SetVerticesDirty();
}
}
public Type GradientType { #region Properties
get { return _gradientType; } public Blend BlendMode
set {
{ get { return _blendMode; }
_gradientType = value; set
graphic.SetVerticesDirty(); {
} _blendMode = value;
} graphic.SetVerticesDirty();
}
}
public float Offset { public UnityEngine.Gradient EffectGradient
get { return _offset; } {
set get { return _effectGradient; }
{ set
_offset = value; {
graphic.SetVerticesDirty(); _effectGradient = value;
} graphic.SetVerticesDirty();
} }
#endregion }
public override void ModifyMesh(VertexHelper helper) { public Type GradientType
if(!IsActive() || helper.currentVertCount == 0) {
return; get { return _gradientType; }
set
{
_gradientType = value;
graphic.SetVerticesDirty();
}
}
List<UIVertex> _vertexList = new List<UIVertex>(); public bool ModifyVertices
{
get { return _modifyVertices; }
set
{
_modifyVertices = value;
graphic.SetVerticesDirty();
}
}
helper.GetUIVertexStream(_vertexList); public float Offset
{
get { return _offset; }
set
{
_offset = value;
graphic.SetVerticesDirty();
}
}
int nCount = _vertexList.Count; public float Zoom
switch(GradientType) { {
case Type.Horizontal: { get { return _zoom; }
float left = _vertexList[0].position.x; set
float right = _vertexList[0].position.x; {
float x = 0f; _zoom = value;
graphic.SetVerticesDirty();
}
}
#endregion
for(int i = nCount - 1; i >= 1; --i) { public override void ModifyMesh(VertexHelper helper)
x = _vertexList[i].position.x; {
if (!IsActive() || helper.currentVertCount == 0)
return;
if(x > right) right = x; List<UIVertex> _vertexList = new List<UIVertex>();
else if(x < left) left = x;
}
float width = 1f / (right - left); helper.GetUIVertexStream(_vertexList);
UIVertex vertex = new UIVertex();
for(int i = 0; i < helper.currentVertCount; i++) { int nCount = _vertexList.Count;
helper.PopulateUIVertex(ref vertex, i); switch (GradientType)
{
case Type.Horizontal:
case Type.Vertical:
{
Rect bounds = GetBounds(_vertexList);
float min = bounds.xMin;
float w = bounds.width;
Func<UIVertex, float> GetPosition = v => v.position.x;
vertex.color = BlendColor(vertex.color, EffectGradient.Evaluate((vertex.position.x - left) * width - Offset)); if (GradientType == Type.Vertical)
{
min = bounds.yMin;
w = bounds.height;
GetPosition = v => v.position.y;
}
helper.SetUIVertex(vertex, i); float width = 1f / w / Zoom;
} float zoomOffset = ((w * Zoom) - w) * width * 0.5f;
} float offset = Offset - zoomOffset;
break;
case Type.Vertical: { if (ModifyVertices)
float bottom = _vertexList[0].position.y; {
float top = _vertexList[0].position.y; SplitTrianglesAtGradientStops(_vertexList, bounds, zoomOffset, helper);
float y = 0f; }
for(int i = nCount - 1; i >= 1; --i) { UIVertex vertex = new UIVertex();
y = _vertexList[i].position.y; for (int i = 0; i < helper.currentVertCount; i++)
{
helper.PopulateUIVertex(ref vertex, i);
vertex.color = BlendColor(vertex.color, EffectGradient.Evaluate((GetPosition(vertex) - min) * width - offset));
helper.SetUIVertex(vertex, i);
}
}
break;
if(y > top) top = y; case Type.Diamond:
else if(y < bottom) bottom = y; {
} Rect bounds = GetBounds(_vertexList);
float height = 1f / (top - bottom); float height = 1f / bounds.height / Zoom;
UIVertex vertex = new UIVertex(); float radius = bounds.center.y / 2f;
Vector3 center = (Vector3.right + Vector3.up) * radius + Vector3.forward * _vertexList[0].position.z;
for(int i = 0; i < helper.currentVertCount; i++) { if (ModifyVertices)
helper.PopulateUIVertex(ref vertex, i); {
helper.Clear();
for (int i = 0; i < nCount; i++) helper.AddVert(_vertexList[i]);
vertex.color = BlendColor(vertex.color, EffectGradient.Evaluate((vertex.position.y - bottom) * height - Offset)); UIVertex centralVertex = new UIVertex();
centralVertex.position = center;
centralVertex.normal = _vertexList[0].normal;
centralVertex.uv0 = new Vector2(0.5f, 0.5f);
centralVertex.color = Color.white;
helper.AddVert(centralVertex);
helper.SetUIVertex(vertex, i); for (int i = 1; i < nCount; i++) helper.AddTriangle(i - 1, i, nCount);
} helper.AddTriangle(0, nCount - 1, nCount);
} }
break;
case Type.Diamond: { UIVertex vertex = new UIVertex();
float bottom = _vertexList[0].position.y; for (int i = 0; i < helper.currentVertCount; i++)
float top = _vertexList[0].position.y; {
float y = 0f; helper.PopulateUIVertex(ref vertex, i);
for(int i = nCount - 1; i >= 1; --i) { vertex.color = BlendColor(vertex.color, EffectGradient.Evaluate(
y = _vertexList[i].position.y; Vector3.Distance(vertex.position, center) * height - Offset));
if(y > top) top = y; helper.SetUIVertex(vertex, i);
else if(y < bottom) bottom = y; }
} }
break;
float height = 1f / (top - bottom); case Type.Radial:
{
Rect bounds = GetBounds(_vertexList);
helper.Clear(); float width = 1f / bounds.width / Zoom;
for (int i = 0; i < nCount; i++) helper.AddVert(_vertexList[i]); float height = 1f / bounds.height / Zoom;
float center = (bottom + top) / 2f; if (ModifyVertices)
UIVertex centralVertex = new UIVertex(); {
centralVertex.position = (Vector3.right + Vector3.up) * center + Vector3.forward * _vertexList[0].position.z; helper.Clear();
centralVertex.normal = _vertexList[0].normal;
centralVertex.color = Color.white;
helper.AddVert(centralVertex);
for (int i = 1; i < nCount; i++) helper.AddTriangle(i-1,i,nCount); float radiusX = bounds.width / 2f;
helper.AddTriangle(0,nCount-1,nCount); float radiusY = bounds.height / 2f;
UIVertex centralVertex = new UIVertex();
centralVertex.position = Vector3.right * bounds.center.x + Vector3.up * bounds.center.y + Vector3.forward * _vertexList[0].position.z;
centralVertex.normal = _vertexList[0].normal;
centralVertex.uv0 = new Vector2(0.5f, 0.5f);
centralVertex.color = Color.white;
UIVertex vertex = new UIVertex(); int steps = 64;
for (int i = 0; i < steps; i++)
{
UIVertex curVertex = new UIVertex();
float angle = (float)i * 360f / (float)steps;
float cosX = Mathf.Cos(Mathf.Deg2Rad * angle);
float cosY = Mathf.Sin(Mathf.Deg2Rad * angle);
for(int i = 0; i < helper.currentVertCount; i++) { curVertex.position = Vector3.right * cosX * radiusX + Vector3.up * cosY * radiusY + Vector3.forward * _vertexList[0].position.z;
helper.PopulateUIVertex(ref vertex, i); curVertex.normal = _vertexList[0].normal;
curVertex.uv0 = new Vector2((cosX + 1) * 0.5f, (cosY + 1) * 0.5f);
curVertex.color = Color.white;
helper.AddVert(curVertex);
}
vertex.color = BlendColor(vertex.color, EffectGradient.Evaluate( helper.AddVert(centralVertex);
Vector3.Distance(vertex.position, centralVertex.position) * height - Offset));
helper.SetUIVertex(vertex, i); for (int i = 1; i < steps; i++) helper.AddTriangle(i - 1, i, steps);
} helper.AddTriangle(0, steps - 1, steps);
} }
break;
case Type.Radial: { UIVertex vertex = new UIVertex();
float left = _vertexList[0].position.x; for (int i = 0; i < helper.currentVertCount; i++)
float right = _vertexList[0].position.x; {
float bottom = _vertexList[0].position.y; helper.PopulateUIVertex(ref vertex, i);
float top = _vertexList[0].position.y;
float x = 0f; vertex.color = BlendColor(vertex.color, EffectGradient.Evaluate(
float y = 0f; Mathf.Sqrt(
Mathf.Pow(Mathf.Abs(vertex.position.x - bounds.center.x) * width, 2f) +
Mathf.Pow(Mathf.Abs(vertex.position.y - bounds.center.y) * height, 2f)) * 2f - Offset));
for(int i = nCount - 1; i >= 1; --i) { helper.SetUIVertex(vertex, i);
x = _vertexList[i].position.x; }
}
break;
}
}
if(x > right) right = x; Rect GetBounds(List<UIVertex> vertices)
else if(x < left) left = x; {
float left = vertices[0].position.x;
float right = left;
float bottom = vertices[0].position.y;
float top = bottom;
y = _vertexList[i].position.y; for (int i = vertices.Count - 1; i >= 1; --i)
{
float x = vertices[i].position.x;
float y = vertices[i].position.y;
if(y > top) top = y; if (x > right) right = x;
else if(y < bottom) bottom = y; else if (x < left) left = x;
}
float width = 1f / (right - left); if (y > top) top = y;
float height = 1f / (top - bottom); else if (y < bottom) bottom = y;
}
helper.Clear(); return new Rect(left, bottom, right - left, top - bottom);
}
float centerX = (right + left) / 2f; void SplitTrianglesAtGradientStops(List<UIVertex> _vertexList, Rect bounds, float zoomOffset, VertexHelper helper)
float centerY = (bottom + top) / 2f; {
float radiusX = (right - left) / 2f; List<float> stops = FindStops(zoomOffset, bounds);
float radiusY = (top - bottom) / 2f; if (stops.Count > 0)
UIVertex centralVertex = new UIVertex(); {
centralVertex.position = Vector3.right * centerX + Vector3.up * centerY + Vector3.forward * _vertexList[0].position.z; helper.Clear();
centralVertex.normal = _vertexList[0].normal;
centralVertex.color = Color.white;
int steps = 64; int nCount = _vertexList.Count;
for (int i = 0; i < steps; i++) for (int i = 0; i < nCount; i += 3)
{ {
UIVertex curVertex = new UIVertex(); float[] positions = GetPositions(_vertexList, i);
float angle = (float)i * 360f / (float)steps; List<int> originIndices = new List<int>(3);
float curX = Mathf.Cos(Mathf.Deg2Rad * angle) * radiusX; List<UIVertex> starts = new List<UIVertex>(3);
float curY = Mathf.Sin(Mathf.Deg2Rad * angle) * radiusY; List<UIVertex> ends = new List<UIVertex>(2);
curVertex.position = Vector3.right * curX + Vector3.up * curY + Vector3.forward * _vertexList[0].position.z; for (int s = 0; s < stops.Count; s++)
curVertex.normal = _vertexList[0].normal; {
curVertex.color = Color.white; int initialCount = helper.currentVertCount;
helper.AddVert(curVertex); bool hadEnds = ends.Count > 0;
} bool earlyStart = false;
helper.AddVert(centralVertex); // find any start vertices for this stop
for (int p = 0; p < 3; p++)
{
if (!originIndices.Contains(p) && positions[p] < stops[s])
{
// make sure the first index crosses the stop
int p1 = (p + 1) % 3;
var start = _vertexList[p + i];
if (positions[p1] > stops[s])
{
originIndices.Insert(0, p);
starts.Insert(0, start);
earlyStart = true;
}
else
{
originIndices.Add(p);
starts.Add(start);
}
}
}
for (int i = 1; i < steps; i++) helper.AddTriangle(i-1,i,steps); // bail if all before or after the stop
helper.AddTriangle(0,steps-1,steps); if (originIndices.Count == 0)
continue;
if (originIndices.Count == 3)
break;
UIVertex vertex = new UIVertex(); // report any start vertices
foreach (var start in starts)
helper.AddVert(start);
for(int i = 0; i < helper.currentVertCount; i++) { // make two ends, splitting at the stop
helper.PopulateUIVertex(ref vertex, i); ends.Clear();
foreach (int index in originIndices)
{
int oppositeIndex = (index + 1) % 3;
if (positions[oppositeIndex] < stops[s])
oppositeIndex = (oppositeIndex + 1) % 3;
ends.Add(CreateSplitVertex(_vertexList[index + i], _vertexList[oppositeIndex + i], stops[s]));
}
if (ends.Count == 1)
{
int oppositeIndex = (originIndices[0] + 2) % 3;
ends.Add(CreateSplitVertex(_vertexList[originIndices[0] + i], _vertexList[oppositeIndex + i], stops[s]));
}
vertex.color = BlendColor(vertex.color, EffectGradient.Evaluate( // report end vertices
Mathf.Sqrt( foreach (var end in ends)
Mathf.Pow(Mathf.Abs(vertex.position.x - centerX) * width, 2f) + helper.AddVert(end);
Mathf.Pow(Mathf.Abs(vertex.position.y - centerY) * height, 2f)) * 2f - Offset));
helper.SetUIVertex(vertex, i); // make triangles
} if (hadEnds)
} {
break; helper.AddTriangle(initialCount - 2, initialCount, initialCount + 1);
} helper.AddTriangle(initialCount - 2, initialCount + 1, initialCount - 1);
} if (starts.Count > 0)
{
if (earlyStart)
helper.AddTriangle(initialCount - 2, initialCount + 3, initialCount);
else
helper.AddTriangle(initialCount + 1, initialCount + 3, initialCount - 1);
}
}
else
{
int vertexCount = helper.currentVertCount;
helper.AddTriangle(initialCount, vertexCount - 2, vertexCount - 1);
if (starts.Count > 1)
helper.AddTriangle(initialCount, vertexCount - 1, initialCount + 1);
}
Color BlendColor(Color colorA, Color colorB) { starts.Clear();
switch(BlendMode) { }
default: return colorB;
case Blend.Add: return colorA + colorB;
case Blend.Multiply: return colorA * colorB;
}
}
public enum Type { // clean up after looping through gradient stops
Horizontal, if (ends.Count > 0)
Vertical, {
Radial, // find any final vertices after the gradient stops
Diamond if (starts.Count == 0)
} {
for (int p = 0; p < 3; p++)
{
if (!originIndices.Contains(p) && positions[p] > stops[stops.Count - 1])
{
int p1 = (p + 1) % 3;
UIVertex end = _vertexList[p + i];
if (positions[p1] > stops[stops.Count - 1])
starts.Insert(0, end);
else
starts.Add(end);
}
}
}
public enum Blend { // report final vertices
Override, foreach (var start in starts)
Add, helper.AddVert(start);
Multiply
} // make final triangle(s)
} int vertexCount = helper.currentVertCount;
if (starts.Count > 1)
{
helper.AddTriangle(vertexCount - 4, vertexCount - 2, vertexCount - 1);
helper.AddTriangle(vertexCount - 4, vertexCount - 1, vertexCount - 3);
}
else if (starts.Count > 0)
{
helper.AddTriangle(vertexCount - 3, vertexCount - 1, vertexCount - 2);
}
}
else
{
// if the triangle wasn't split, add it as-is
helper.AddVert(_vertexList[i]);
helper.AddVert(_vertexList[i + 1]);
helper.AddVert(_vertexList[i + 2]);
int vertexCount = helper.currentVertCount;
helper.AddTriangle(vertexCount - 3, vertexCount - 2, vertexCount - 1);
}
}
}
}
float[] GetPositions(List<UIVertex> _vertexList, int index)
{
float[] positions = new float[3];
if (GradientType == Type.Horizontal)
{
positions[0] = _vertexList[index].position.x;
positions[1] = _vertexList[index + 1].position.x;
positions[2] = _vertexList[index + 2].position.x;
}
else
{
positions[0] = _vertexList[index].position.y;
positions[1] = _vertexList[index + 1].position.y;
positions[2] = _vertexList[index + 2].position.y;
}
return positions;
}
List<float> FindStops(float zoomOffset, Rect bounds)
{
List<float> stops = new List<float>();
var scaledOffset = Offset * Zoom;
foreach (var color in EffectGradient.colorKeys)
{
if (color.time >= (1 - (zoomOffset + scaledOffset)))
break;
if (color.time > (zoomOffset - scaledOffset))
stops.Add(((color.time - 0.5f) * Zoom) + 0.5f + scaledOffset);
}
foreach (var alpha in EffectGradient.alphaKeys)
{
if (alpha.time >= (1 - (zoomOffset + scaledOffset)))
break;
if (alpha.time > (zoomOffset - scaledOffset))
stops.Add(((alpha.time - 0.5f) * Zoom) + 0.5f + scaledOffset);
}
float min = bounds.xMin;
float size = bounds.width;
if (GradientType == Type.Vertical)
{
min = bounds.yMin;
size = bounds.height;
}
stops.Sort();
for (int i = 0; i < stops.Count; i++)
{
stops[i] = (stops[i] * size) + min;
if (i > 0 && Math.Abs(stops[i] - stops[i - 1]) < 2)
{
stops.RemoveAt(i);
--i;
}
}
return stops;
}
UIVertex CreateSplitVertex(UIVertex vertex1, UIVertex vertex2, float stop)
{
if (GradientType == Type.Horizontal)
{
float sx = vertex1.position.x - stop;
float dx = vertex1.position.x - vertex2.position.x;
float dy = vertex1.position.y - vertex2.position.y;
float uvx = vertex1.uv0.x - vertex2.uv0.x;
float uvy = vertex1.uv0.y - vertex2.uv0.y;
float ratio = sx / dx;
float splitY = vertex1.position.y - (dy * ratio);
UIVertex splitVertex = new UIVertex();
splitVertex.position = new Vector3(stop, splitY, vertex1.position.z);
splitVertex.normal = vertex1.normal;
splitVertex.uv0 = new Vector2(vertex1.uv0.x - (uvx * ratio), vertex1.uv0.y - (uvy * ratio));
splitVertex.color = Color.white;
return splitVertex;
}
else
{
float sy = vertex1.position.y - stop;
float dy = vertex1.position.y - vertex2.position.y;
float dx = vertex1.position.x - vertex2.position.x;
float uvx = vertex1.uv0.x - vertex2.uv0.x;
float uvy = vertex1.uv0.y - vertex2.uv0.y;
float ratio = sy / dy;
float splitX = vertex1.position.x - (dx * ratio);
UIVertex splitVertex = new UIVertex();
splitVertex.position = new Vector3(splitX, stop, vertex1.position.z);
splitVertex.normal = vertex1.normal;
splitVertex.uv0 = new Vector2(vertex1.uv0.x - (uvx * ratio), vertex1.uv0.y - (uvy * ratio));
splitVertex.color = Color.white;
return splitVertex;
}
}
Color BlendColor(Color colorA, Color colorB)
{
switch (BlendMode)
{
default: return colorB;
case Blend.Add: return colorA + colorB;
case Blend.Multiply: return colorA * colorB;
}
}
public enum Type
{
Horizontal,
Vertical,
Radial,
Diamond
}
public enum Blend
{
Override,
Add,
Multiply
}
}
} }