您最多选择25个主题 主题必须以中文或者字母或数字开头,可以包含连字符 (-),并且长度不得超过35个字符
 
 
 
 
 
 

626 行
19 KiB

using System;
using System.Collections.Generic;
using Unity.UIWidgets.foundation;
using UnityEngine;
namespace Unity.UIWidgets.ui {
public enum TileMode : int {
clamp = 0,
mirror = 1,
repeated = 2,
}
public abstract class PaintShader {
}
public class Gradient : PaintShader {
public static Gradient linear(
Offset start, Offset end, List<Color> colors,
List<float> colorStops = null, TileMode tileMode = TileMode.clamp,
Matrix3 matrix = null) {
D.assert(PaintingUtils._offsetIsValid(start));
D.assert(PaintingUtils._offsetIsValid(end));
D.assert(colors != null && colors.Count >= 2);
_validateColorStops(ref colors, ref colorStops);
return new _LinearGradient(start, end, colors, colorStops, tileMode, matrix);
}
public static Gradient radial(
Offset center, float radius, List<Color> colors,
List<float> colorStops = null, TileMode tileMode = TileMode.clamp,
Matrix3 matrix = null) {
D.assert(PaintingUtils._offsetIsValid(center));
D.assert(colors != null && colors.Count >= 2);
_validateColorStops(ref colors, ref colorStops);
return new _RadialGradient(center, radius, colors, colorStops, tileMode, matrix);
}
public static Gradient sweep(
Offset center, List<Color> colors,
List<float> colorStops = null, TileMode tileMode = TileMode.clamp,
float startAngle = 0.0f, float endAngle = Mathf.PI * 2,
Matrix3 matrix = null) {
D.assert(PaintingUtils._offsetIsValid(center));
D.assert(colors != null && colors.Count >= 2);
D.assert(startAngle < endAngle);
_validateColorStops(ref colors, ref colorStops);
return new _SweepGradient(center, colors, colorStops, tileMode, startAngle, endAngle, matrix);
}
static void _validateColorStops(ref List<Color> colors, ref List<float> colorStops) {
if (colorStops == null) {
colors = new List<Color>(colors);
colorStops = new List<float>(colors.Count);
colorStops.Add(0);
var stepCount = colors.Count - 1;
var step = 1.0f / stepCount;
for (int i = 1; i < stepCount; i++) {
colorStops.Add(colorStops[i - 1] + step);
}
colorStops.Add(1);
return;
}
if (colors.Count != colorStops.Count) {
throw new ArgumentException("\"colors\" and \"colorStops\" arguments must have equal length.");
}
var dummyFirst = colorStops[0] != 0;
var dummyLast = colorStops[colorStops.Count - 1] != 1;
var count = colors.Count + (dummyFirst ? 1 : 0) + (dummyFirst ? 1 : 0);
var newColors = new List<Color>(count);
if (dummyFirst) {
newColors.Add(colors[0]);
}
for (int i = 0; i < colors.Count; i++) {
newColors.Add(colors[i]);
}
if (dummyLast) {
newColors.Add(colors[colors.Count - 1]);
}
var newColorStops = new List<float>(count);
if (dummyFirst) {
newColorStops.Add(0.0f);
}
var prevStop = 0.0f;
for (int i = 0; i < colorStops.Count; i++) {
var stop = Mathf.Max(Mathf.Min(colorStops[i], 1.0f), prevStop);
newColorStops.Add(stop);
prevStop = stop;
}
if (dummyLast) {
newColorStops.Add(1.0f);
}
colors = newColors;
colorStops = newColorStops;
}
static readonly GradientBitmapCache _cache = new GradientBitmapCache();
internal static Image makeTexturedColorizer(List<Color> colors, List<float> positions) {
int count = colors.Count;
D.assert(count >= 2);
bool bottomHardStop = ScalarUtils.ScalarNearlyEqual(positions[0], positions[1]);
bool topHardStop =
ScalarUtils.ScalarNearlyEqual(positions[count - 2], positions[count - 1]);
int offset = 0;
if (bottomHardStop) {
offset += 1;
count--;
}
if (topHardStop) {
count--;
}
if (offset != 0 || count != colors.Count) {
colors = colors.GetRange(offset, count);
positions = positions.GetRange(offset, count);
}
return _cache.getGradient(colors, positions);
}
}
class GradientBitmapCache : IDisposable {
public GradientBitmapCache(int maxEntries = 32, int resolution = 256) {
this.maxEntries = maxEntries;
this.resolution = resolution;
this._entryCount = 0;
this._head = this._tail = null;
D.assert(this.validate);
}
public readonly int maxEntries;
public readonly int resolution;
int _entryCount;
_Entry _head;
_Entry _tail;
public Image getGradient(List<Color> colors, List<float> positions) {
var key = new _Key(colors, positions);
if (!this.find(key, out var image)) {
image = this.fillGradient(colors, positions);
this.add(key, image);
}
return image;
}
public void Dispose() {
D.assert(this.validate);
// just remove the references, Image will dispose by themselves.
this._entryCount = 0;
this._head = this._tail = null;
}
_Entry release(_Entry entry) {
if (entry.prev != null) {
D.assert(this._head != entry);
entry.prev.next = entry.next;
}
else {
D.assert(this._head == entry);
this._head = entry.next;
}
if (entry.next != null) {
D.assert(this._tail != entry);
entry.next.prev = entry.prev;
}
else {
D.assert(this._tail == entry);
this._tail = entry.prev;
}
return entry;
}
void attachToHead(_Entry entry) {
entry.prev = null;
entry.next = this._head;
if (this._head != null) {
this._head.prev = entry;
}
else {
this._tail = entry;
}
this._head = entry;
}
bool find(_Key key, out Image image) {
D.assert(this.validate);
var entry = this._head;
while (entry != null) {
if (entry.key == key) {
image = entry.image;
// move to the head of our list, so we purge it last
this.release(entry);
this.attachToHead(entry);
D.assert(this.validate);
return true;
}
entry = entry.next;
}
D.assert(this.validate);
image = null;
return false;
}
void add(_Key key, Image image) {
if (this._entryCount == this.maxEntries) {
D.assert(this._tail != null);
this.release(this._tail);
this._entryCount--;
}
var entry = new _Entry {key = key, image = image};
this.attachToHead(entry);
this._entryCount++;
}
Image fillGradient(List<Color> colors, List<float> positions) {
Texture2D tex = new Texture2D(this.resolution, 1, TextureFormat.RGBA32, false);
tex.hideFlags = HideFlags.HideAndDontSave;
tex.wrapMode = TextureWrapMode.Clamp;
var bytes = new byte[this.resolution * 4];
int count = colors.Count;
int prevIndex = 0;
for (int i = 1; i < count; i++) {
// Historically, stops have been mapped to [0, 256], with 256 then nudged to the next
// smaller value, then truncate for the texture index. This seems to produce the best
// results for some common distributions, so we preserve the behavior.
int nextIndex = (int) Mathf.Min(positions[i] * this.resolution, this.resolution - 1);
if (nextIndex > prevIndex) {
var c0 = colors[i - 1];
var c1 = colors[i];
var step = 1.0f / (nextIndex - prevIndex);
var t = 0.0f;
for (int curIndex = prevIndex; curIndex <= nextIndex; ++curIndex) {
var c = Color.lerp(c0, c1, t);
var baseIndex = curIndex << 2;
bytes[baseIndex] = (byte) c.red;
bytes[baseIndex + 1] = (byte) c.green;
bytes[baseIndex + 2] = (byte) c.blue;
bytes[baseIndex + 3] = (byte) c.alpha;
t += step;
}
}
prevIndex = nextIndex;
}
D.assert(prevIndex == this.resolution - 1);
tex.LoadRawTextureData(bytes);
tex.Apply();
return new Image(tex);
}
bool validate() {
D.assert(this._entryCount >= 0 && this._entryCount <= this.maxEntries);
if (this._entryCount > 0) {
D.assert(null == this._head.prev);
D.assert(null == this._tail.next);
if (this._entryCount == 1) {
D.assert(this._head == this._tail);
}
else {
D.assert(this._head != this._tail);
}
var entry = this._head;
int count = 0;
while (entry != null) {
count += 1;
entry = entry.next;
}
D.assert(count == this._entryCount);
entry = this._tail;
while (entry != null) {
count -= 1;
entry = entry.prev;
}
D.assert(0 == count);
}
else {
D.assert(null == this._head);
D.assert(null == this._tail);
}
return true;
}
class _Entry {
public _Entry prev;
public _Entry next;
public _Key key;
public Image image;
}
class _Key : IEquatable<_Key> {
public _Key(List<Color> colors, List<float> positions) {
D.assert(colors != null);
D.assert(positions != null);
D.assert(colors.Count == positions.Count);
this.colors = colors;
this.positions = positions;
this._hashCode = _getHashCode(this.colors) ^ _getHashCode(this.positions);
;
}
public readonly List<Color> colors;
public readonly List<float> positions;
readonly int _hashCode;
public bool Equals(_Key other) {
if (ReferenceEquals(null, other)) {
return false;
}
if (ReferenceEquals(this, other)) {
return true;
}
return _listEquals(this.colors, other.colors) &&
_listEquals(this.positions, other.positions);
}
public override bool Equals(object obj) {
if (ReferenceEquals(null, obj)) {
return false;
}
if (ReferenceEquals(this, obj)) {
return true;
}
if (obj.GetType() != this.GetType()) {
return false;
}
return this.Equals((_Key) obj);
}
public override int GetHashCode() {
return this._hashCode;
}
public static bool operator ==(_Key left, _Key right) {
return Equals(left, right);
}
public static bool operator !=(_Key left, _Key right) {
return !Equals(left, right);
}
static int _getHashCode<T>(List<T> list) {
unchecked {
var hashCode = 0;
foreach (var item in list) {
hashCode = (hashCode * 397) ^ item.GetHashCode();
}
return hashCode;
}
}
static bool _listEquals<T>(List<T> left, List<T> right) {
if (left.Count != right.Count) {
return false;
}
for (int i = 0; i < left.Count; i++) {
if (!left[i].Equals(right[i])) {
return false;
}
}
return true;
}
}
}
class _LinearGradient : Gradient {
public _LinearGradient(
Offset start, Offset end, List<Color> colors,
List<float> colorStops, TileMode tileMode,
Matrix3 matrix = null) {
this.start = start;
this.end = end;
this.colors = colors;
this.colorStops = colorStops;
this.tileMode = tileMode;
this.matrix = matrix;
this.ptsToUnit = ptsToUnitMatrix(start, end);
this.gradientTex = makeTexturedColorizer(colors, colorStops);
}
public readonly Offset start;
public readonly Offset end;
public readonly List<Color> colors;
public readonly List<float> colorStops;
public readonly TileMode tileMode;
public readonly Matrix3 matrix;
public readonly Matrix3 ptsToUnit;
public readonly Image gradientTex;
public Color leftColor {
get { return this.colors[0]; }
}
public Color rightColor {
get { return this.colors[this.colors.Count - 1]; }
}
public Matrix3 getGradientMat(Matrix3 ctm) {
var mat = Matrix3.I();
ctm.invert(mat); // just use I() if not invertible.
if (this.matrix != null) {
mat.postConcat(this.matrix);
}
mat.postConcat(this.ptsToUnit);
return mat;
}
static Matrix3 ptsToUnitMatrix(Offset start, Offset end) {
var vec = end - start;
var mag = vec.distance;
var inv = mag != 0 ? 1 / mag : 0;
vec = vec.scale(inv);
var matrix = Matrix3.I();
matrix.setSinCos(-vec.dy, vec.dx, start.dx, start.dy);
matrix.postTranslate(-start.dx, -start.dy);
matrix.postScale(inv, inv);
return matrix;
}
}
class _RadialGradient : Gradient {
public _RadialGradient(
Offset center, float radius, List<Color> colors,
List<float> colorStops = null, TileMode tileMode = TileMode.clamp,
Matrix3 matrix = null
) {
this.center = center;
this.radius = radius;
this.colors = colors;
this.colorStops = colorStops;
this.tileMode = tileMode;
this.matrix = matrix;
this.ptsToUnit = radToUnitMatrix(center, radius);
this.gradientTex = makeTexturedColorizer(colors, colorStops);
}
public readonly Offset center;
public readonly float radius;
public readonly List<Color> colors;
public readonly List<float> colorStops;
public readonly TileMode tileMode;
public readonly Matrix3 matrix;
public readonly Matrix3 ptsToUnit;
public readonly Image gradientTex;
public Color leftColor {
get { return this.colors[0]; }
}
public Color rightColor {
get { return this.colors[this.colors.Count - 1]; }
}
public Matrix3 getGradientMat(Matrix3 ctm) {
var mat = Matrix3.I();
ctm.invert(mat); // just use I() if not invertible.
if (this.matrix != null) {
mat.postConcat(this.matrix);
}
mat.postConcat(this.ptsToUnit);
return mat;
}
static Matrix3 radToUnitMatrix(Offset center, float radius) {
var inv = radius != 0 ? 1 / radius : 0;
var matrix = Matrix3.I();
matrix.setTranslate(-center.dx, -center.dy);
matrix.postScale(inv, inv);
return matrix;
}
}
class _SweepGradient : Gradient {
public _SweepGradient(
Offset center, List<Color> colors,
List<float> colorStops = null, TileMode tileMode = TileMode.clamp,
float startAngle = 0.0f, float endAngle = Mathf.PI * 2,
Matrix3 matrix = null
) {
this.center = center;
this.colors = colors;
this.colorStops = colorStops;
this.tileMode = tileMode;
this.startAngle = startAngle;
this.endAngle = endAngle;
this.matrix = matrix;
var t0 = startAngle / (Mathf.PI * 2f);
var t1 = endAngle / (Mathf.PI * 2f);
this.bias = -t0;
this.scale = 1f / (t1 - t0);
var ptsToUnit = Matrix3.I();
ptsToUnit.setTranslate(-center.dx, -center.dy);
this.ptsToUnit = ptsToUnit;
this.gradientTex = makeTexturedColorizer(colors, colorStops);
}
public readonly Offset center;
public readonly List<Color> colors;
public readonly List<float> colorStops;
public readonly TileMode tileMode;
public readonly float startAngle;
public readonly float endAngle;
public readonly Matrix3 matrix;
public readonly Matrix3 ptsToUnit;
public readonly Image gradientTex;
public readonly float bias;
public readonly float scale;
public Color leftColor {
get { return this.colors[0]; }
}
public Color rightColor {
get { return this.colors[this.colors.Count - 1]; }
}
public Matrix3 getGradientMat(Matrix3 ctm) {
var mat = Matrix3.I();
ctm.invert(mat); // just use I() if not invertible.
if (this.matrix != null) {
mat.postConcat(this.matrix);
}
mat.postConcat(this.ptsToUnit);
return mat;
}
}
public class ImageShader : PaintShader {
public ImageShader(Image image,
TileMode tileMode = TileMode.clamp, Matrix3 matrix = null) {
this.image = image;
this.tileMode = tileMode;
this.matrix = matrix;
}
public readonly Image image;
public readonly TileMode tileMode;
public readonly Matrix3 matrix;
public Matrix3 getShaderMat(Matrix3 ctm) {
var mat = Matrix3.I();
ctm.invert(mat); // just use I() if not invertible.
if (this.matrix != null) {
mat.postConcat(this.matrix);
}
mat.postScale(1f / this.image.width, 1f / this.image.height);
return mat;
}
}
}