您最多选择25个主题
主题必须以中文或者字母或数字开头,可以包含连字符 (-),并且长度不得超过35个字符
482 行
13 KiB
482 行
13 KiB
using System;
|
|
using Unity.UIWidgets.foundation;
|
|
using UnityEngine;
|
|
|
|
namespace Unity.UIWidgets.ui {
|
|
class PaintingUtils {
|
|
internal static bool _offsetIsValid(Offset offset) {
|
|
D.assert(offset != null, "Offset argument was null.");
|
|
D.assert(!offset.dx.isNaN() && !offset.dy.isNaN(), "Offset argument contained a NaN value.");
|
|
return true;
|
|
}
|
|
|
|
internal static bool _radiusIsValid(Radius radius) {
|
|
D.assert(radius != null, "Radius argument was null.");
|
|
D.assert(!radius.x.isNaN() && !radius.y.isNaN(), "Radius argument contained a NaN value.");
|
|
return true;
|
|
}
|
|
|
|
internal static Color _scaleAlpha(Color a, float factor) {
|
|
return a.withAlpha((a.alpha * factor).round().clamp(0, 255));
|
|
}
|
|
}
|
|
|
|
public class Color : IEquatable<Color> {
|
|
public Color(long value) {
|
|
this.value = value & 0xFFFFFFFF;
|
|
}
|
|
|
|
public static readonly Color clear = new Color(0x00000000);
|
|
|
|
public static readonly Color black = new Color(0xFF000000);
|
|
|
|
public static readonly Color white = new Color(0xFFFFFFFF);
|
|
|
|
public static Color fromARGB(int a, int r, int g, int b) {
|
|
return new Color(
|
|
(((a & 0xff) << 24) |
|
|
((r & 0xff) << 16) |
|
|
((g & 0xff) << 8) |
|
|
((b & 0xff) << 0)) & 0xFFFFFFFF);
|
|
}
|
|
|
|
public static Color fromRGBO(int r, int g, int b, float opacity) {
|
|
return new Color(
|
|
((((int) (opacity * 0xff) & 0xff) << 24) |
|
|
((r & 0xff) << 16) |
|
|
((g & 0xff) << 8) |
|
|
((b & 0xff) << 0)) & 0xFFFFFFFF);
|
|
}
|
|
|
|
public readonly long value;
|
|
|
|
public int alpha {
|
|
get { return (int) ((0xff000000 & this.value) >> 24); }
|
|
}
|
|
|
|
public float opacity {
|
|
get { return this.alpha / 255.0f; }
|
|
}
|
|
|
|
public int red {
|
|
get { return (int) ((0x00ff0000 & this.value) >> 16); }
|
|
}
|
|
|
|
public int green {
|
|
get { return (int) ((0x0000ff00 & this.value) >> 8); }
|
|
}
|
|
|
|
public int blue {
|
|
get { return (int) ((0x000000ff & this.value) >> 0); }
|
|
}
|
|
|
|
public Color withAlpha(int a) {
|
|
return fromARGB(a, this.red, this.green, this.blue);
|
|
}
|
|
|
|
public Color withOpacity(float opacity) {
|
|
return this.withAlpha((int) (opacity * 255));
|
|
}
|
|
|
|
public Color withRed(int r) {
|
|
return fromARGB(this.alpha, r, this.green, this.blue);
|
|
}
|
|
|
|
public Color withGreen(int g) {
|
|
return fromARGB(this.alpha, this.red, g, this.blue);
|
|
}
|
|
|
|
public Color withBlue(int b) {
|
|
return fromARGB(this.alpha, this.red, this.green, b);
|
|
}
|
|
|
|
static float _linearizeColorComponent(float component) {
|
|
if (component <= 0.03928f) {
|
|
return component / 12.92f;
|
|
}
|
|
|
|
return Mathf.Pow((component + 0.055f) / 1.055f, 2.4f);
|
|
}
|
|
|
|
public float computeLuminance() {
|
|
float R = _linearizeColorComponent(this.red / 0xFF);
|
|
float G = _linearizeColorComponent(this.green / 0xFF);
|
|
float B = _linearizeColorComponent(this.blue / 0xFF);
|
|
return 0.2126f * R + 0.7152f * G + 0.0722f * B;
|
|
}
|
|
|
|
public static Color lerp(Color a, Color b, float t) {
|
|
if (a == null && b == null) {
|
|
return null;
|
|
}
|
|
|
|
if (a == null) {
|
|
return b._scaleAlpha(t);
|
|
}
|
|
|
|
if (b == null) {
|
|
return a._scaleAlpha(1.0f - t);
|
|
}
|
|
|
|
return fromARGB(
|
|
((int) MathUtils.lerpFloat(a.alpha, b.alpha, t)).clamp(0, 255),
|
|
((int) MathUtils.lerpFloat(a.red, b.red, t)).clamp(0, 255),
|
|
((int) MathUtils.lerpFloat(a.green, b.green, t)).clamp(0, 255),
|
|
((int) MathUtils.lerpFloat(a.blue, b.blue, t)).clamp(0, 255)
|
|
);
|
|
}
|
|
|
|
public bool Equals(Color other) {
|
|
if (ReferenceEquals(null, other)) {
|
|
return false;
|
|
}
|
|
|
|
if (ReferenceEquals(this, other)) {
|
|
return true;
|
|
}
|
|
|
|
return this.value == other.value;
|
|
}
|
|
|
|
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((Color) obj);
|
|
}
|
|
|
|
public override int GetHashCode() {
|
|
return this.value.GetHashCode();
|
|
}
|
|
|
|
public static bool operator ==(Color a, Color b) {
|
|
return ReferenceEquals(a, null) ? ReferenceEquals(b, null) : a.Equals(b);
|
|
}
|
|
|
|
public static bool operator !=(Color a, Color b) {
|
|
return !(a == b);
|
|
}
|
|
|
|
public override string ToString() {
|
|
return $"Color(0x{this.value:X8})";
|
|
}
|
|
}
|
|
|
|
public enum Clip {
|
|
none,
|
|
hardEdge,
|
|
antiAlias,
|
|
antiAliasWithSaveLayer,
|
|
}
|
|
|
|
public enum PaintingStyle {
|
|
fill,
|
|
stroke,
|
|
}
|
|
|
|
public enum StrokeCap {
|
|
butt,
|
|
round,
|
|
square,
|
|
}
|
|
|
|
public enum StrokeJoin {
|
|
miter,
|
|
round,
|
|
bevel,
|
|
}
|
|
|
|
public enum BlurStyle {
|
|
normal, // only normal for now.
|
|
solid,
|
|
outer,
|
|
inner,
|
|
}
|
|
|
|
public class MaskFilter : IEquatable<MaskFilter> {
|
|
MaskFilter(BlurStyle style, float sigma) {
|
|
this.style = style;
|
|
this.sigma = sigma;
|
|
}
|
|
|
|
public static MaskFilter blur(BlurStyle style, float sigma) {
|
|
return new MaskFilter(style, sigma);
|
|
}
|
|
|
|
public readonly BlurStyle style;
|
|
public readonly float sigma;
|
|
|
|
public bool Equals(MaskFilter other) {
|
|
if (ReferenceEquals(null, other)) {
|
|
return false;
|
|
}
|
|
|
|
if (ReferenceEquals(this, other)) {
|
|
return true;
|
|
}
|
|
|
|
return this.style == other.style && this.sigma.Equals(other.sigma);
|
|
}
|
|
|
|
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((MaskFilter) obj);
|
|
}
|
|
|
|
public override int GetHashCode() {
|
|
unchecked {
|
|
return ((int) this.style * 397) ^ this.sigma.GetHashCode();
|
|
}
|
|
}
|
|
|
|
public static bool operator ==(MaskFilter left, MaskFilter right) {
|
|
return Equals(left, right);
|
|
}
|
|
|
|
public static bool operator !=(MaskFilter left, MaskFilter right) {
|
|
return !Equals(left, right);
|
|
}
|
|
|
|
public override string ToString() {
|
|
return $"MaskFilter.blur(${this.style}, ${this.sigma:F1})";
|
|
}
|
|
}
|
|
|
|
public class ColorFilter : IEquatable<ColorFilter> {
|
|
ColorFilter(Color color, BlendMode blendMode) {
|
|
D.assert(color != null);
|
|
this.color = color;
|
|
this.blendMode = blendMode;
|
|
}
|
|
|
|
public static ColorFilter mode(Color color, BlendMode blendMode) {
|
|
return new ColorFilter(color, blendMode);
|
|
}
|
|
|
|
public readonly Color color;
|
|
|
|
public readonly BlendMode blendMode;
|
|
|
|
public bool Equals(ColorFilter other) {
|
|
if (ReferenceEquals(null, other)) {
|
|
return false;
|
|
}
|
|
|
|
if (ReferenceEquals(this, other)) {
|
|
return true;
|
|
}
|
|
|
|
return Equals(this.color, other.color) && this.blendMode == other.blendMode;
|
|
}
|
|
|
|
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((ColorFilter) obj);
|
|
}
|
|
|
|
public override int GetHashCode() {
|
|
unchecked {
|
|
return ((this.color != null ? this.color.GetHashCode() : 0) * 397) ^ (int) this.blendMode;
|
|
}
|
|
}
|
|
|
|
public static bool operator ==(ColorFilter left, ColorFilter right) {
|
|
return Equals(left, right);
|
|
}
|
|
|
|
public static bool operator !=(ColorFilter left, ColorFilter right) {
|
|
return !Equals(left, right);
|
|
}
|
|
|
|
public override string ToString() {
|
|
return $"ColorFilter({this.color}, {this.blendMode})";
|
|
}
|
|
}
|
|
|
|
public abstract class ImageFilter {
|
|
public static ImageFilter blur(float sigmaX = 0.0f, float sigmaY = 0.0f) {
|
|
return new _BlurImageFilter(sigmaX, sigmaY);
|
|
}
|
|
|
|
public static ImageFilter matrix(Matrix3 transform, FilterMode filterMode = FilterMode.Point) {
|
|
return new _MatrixImageFilter(transform, filterMode);
|
|
}
|
|
}
|
|
|
|
class _BlurImageFilter : ImageFilter {
|
|
public _BlurImageFilter(float sigmaX, float sigmaY) {
|
|
this.sigmaX = sigmaX;
|
|
this.sigmaY = sigmaY;
|
|
}
|
|
|
|
public readonly float sigmaX;
|
|
public readonly float sigmaY;
|
|
}
|
|
|
|
class _MatrixImageFilter : ImageFilter {
|
|
public _MatrixImageFilter(Matrix3 transform, FilterMode filterMode) {
|
|
D.assert(transform != null);
|
|
this.transform = transform;
|
|
this.filterMode = filterMode;
|
|
}
|
|
|
|
public readonly Matrix3 transform;
|
|
public readonly FilterMode filterMode;
|
|
}
|
|
|
|
public class Paint {
|
|
static readonly Color _kColorDefault = new Color(0xFFFFFFFF);
|
|
|
|
public Color color = _kColorDefault;
|
|
|
|
public BlendMode blendMode = BlendMode.srcOver;
|
|
|
|
public PaintingStyle style = PaintingStyle.fill;
|
|
|
|
public float strokeWidth = 0;
|
|
|
|
public StrokeCap strokeCap = StrokeCap.butt;
|
|
|
|
public StrokeJoin strokeJoin = StrokeJoin.miter;
|
|
|
|
public float strokeMiterLimit = 4.0f;
|
|
|
|
public FilterMode filterMode = FilterMode.Point;
|
|
|
|
public ColorFilter colorFilter = null;
|
|
|
|
public MaskFilter maskFilter = null;
|
|
|
|
public ImageFilter backdrop = null;
|
|
|
|
public PaintShader shader = null;
|
|
|
|
public bool invertColors = false;
|
|
|
|
public Paint() {
|
|
}
|
|
|
|
public Paint(Paint paint) {
|
|
D.assert(paint != null);
|
|
|
|
this.color = paint.color;
|
|
this.blendMode = paint.blendMode;
|
|
this.style = paint.style;
|
|
this.strokeWidth = paint.strokeWidth;
|
|
this.strokeCap = paint.strokeCap;
|
|
this.strokeJoin = paint.strokeJoin;
|
|
this.strokeMiterLimit = paint.strokeMiterLimit;
|
|
this.filterMode = paint.filterMode;
|
|
this.colorFilter = paint.colorFilter;
|
|
this.maskFilter = paint.maskFilter;
|
|
this.backdrop = paint.backdrop;
|
|
this.shader = paint.shader;
|
|
this.invertColors = paint.invertColors;
|
|
}
|
|
|
|
public static Paint shapeOnly(Paint paint) {
|
|
return new Paint {
|
|
style = paint.style,
|
|
strokeWidth = paint.strokeWidth,
|
|
strokeCap = paint.strokeCap,
|
|
strokeJoin = paint.strokeJoin,
|
|
strokeMiterLimit = paint.strokeMiterLimit,
|
|
};
|
|
}
|
|
}
|
|
|
|
public static class Conversions {
|
|
public static UnityEngine.Color toColor(this Color color) {
|
|
return new UnityEngine.Color(
|
|
color.red / 255f, color.green / 255f, color.blue / 255f, color.alpha / 255f);
|
|
}
|
|
|
|
public static Color32 toColor32(this Color color) {
|
|
return new Color32(
|
|
(byte) color.red, (byte) color.green, (byte) color.blue, (byte) color.alpha);
|
|
}
|
|
|
|
public static Vector2 toVector(this Offset offset) {
|
|
return new Vector2(offset.dx, offset.dy);
|
|
}
|
|
|
|
public static UnityEngine.Rect toRect(this Rect rect) {
|
|
return new UnityEngine.Rect(rect.left, rect.top, rect.width, rect.height);
|
|
}
|
|
|
|
public static float alignToPixel(this float v, float devicePixelRatio) {
|
|
return Mathf.Round(v * devicePixelRatio) / devicePixelRatio;
|
|
}
|
|
|
|
internal static Color _scaleAlpha(this Color a, float factor) {
|
|
return a.withAlpha((a.alpha * factor).round().clamp(0, 255));
|
|
}
|
|
}
|
|
|
|
public enum BlendMode {
|
|
clear,
|
|
src,
|
|
dst,
|
|
srcOver,
|
|
dstOver,
|
|
srcIn,
|
|
dstIn,
|
|
srcOut,
|
|
dstOut,
|
|
srcATop,
|
|
dstATop,
|
|
xor,
|
|
plus,
|
|
|
|
// REF: https://www.w3.org/TR/compositing-1/#blendingseparable
|
|
modulate,
|
|
screen,
|
|
overlay,
|
|
darken,
|
|
lighten,
|
|
colorDodge,
|
|
colorBurn,
|
|
hardLight,
|
|
softLight,
|
|
difference,
|
|
exclusion,
|
|
multiply,
|
|
|
|
// REF: https://www.w3.org/TR/compositing-1/#blendingnonseparable
|
|
hue,
|
|
saturation,
|
|
color,
|
|
luminosity,
|
|
}
|
|
}
|