您最多选择25个主题
主题必须以中文或者字母或数字开头,可以包含连字符 (-),并且长度不得超过35个字符
1440 行
49 KiB
1440 行
49 KiB
using System;
|
|
using Unity.UIWidgets.foundation;
|
|
using UnityEngine;
|
|
|
|
namespace Unity.UIWidgets.ui {
|
|
public class Matrix3 : IEquatable<Matrix3> {
|
|
public static Matrix3 makeScale(float sx, float sy) {
|
|
var m = new Matrix3();
|
|
m.setScale(sx, sy);
|
|
return m;
|
|
}
|
|
|
|
public static Matrix3 makeScale(float scale) {
|
|
var m = new Matrix3();
|
|
m.setScale(scale, scale);
|
|
return m;
|
|
}
|
|
|
|
public static Matrix3 makeTrans(float dx, float dy) {
|
|
var m = new Matrix3();
|
|
m.setTranslate(dx, dy);
|
|
return m;
|
|
}
|
|
|
|
public static Matrix3 makeRotate(float degree) {
|
|
var m = new Matrix3();
|
|
m.setRotate(degree);
|
|
return m;
|
|
}
|
|
|
|
public static Matrix3 makeRotate(float degree, float px, float py) {
|
|
var m = new Matrix3();
|
|
m.setRotate(degree, px, py);
|
|
return m;
|
|
}
|
|
|
|
public static Matrix3 makeTrans(Offset offset) {
|
|
var m = new Matrix3();
|
|
m.setTranslate((float) offset.dx, (float) offset.dy);
|
|
return m;
|
|
}
|
|
|
|
public static Matrix3 makeAll(
|
|
float scaleX, float skewX, float transX,
|
|
float skewY, float scaleY, float transY,
|
|
float pers0, float pers1, float pers2) {
|
|
var m = new Matrix3();
|
|
m.setAll(scaleX, skewX, transX, skewY, scaleY, transY, pers0, pers1, pers2);
|
|
return m;
|
|
}
|
|
|
|
public Matrix3(Matrix3 other) {
|
|
D.assert(other != null);
|
|
this.fMat = (float[]) other.fMat.Clone();
|
|
this.fTypeMask = other.fTypeMask;
|
|
}
|
|
|
|
public void copyFrom(Matrix3 other) {
|
|
Array.Copy(other.fMat, this.fMat, 9);
|
|
this.fTypeMask = other.fTypeMask;
|
|
}
|
|
|
|
[Flags]
|
|
public enum TypeMask {
|
|
kIdentity_Mask = 0, //!< identity SkMatrix; all bits clear
|
|
kTranslate_Mask = 0x01, //!< translation SkMatrix
|
|
kScale_Mask = 0x02, //!< scale SkMatrix
|
|
kAffine_Mask = 0x04, //!< skew or rotate SkMatrix
|
|
kPerspective_Mask = 0x08, //!< perspective SkMatrix
|
|
}
|
|
|
|
public TypeMask getType() {
|
|
if ((this.fTypeMask & kUnknown_Mask) != 0) {
|
|
this.fTypeMask = this.computeTypeMask();
|
|
}
|
|
|
|
// only return the public masks
|
|
return (TypeMask) (this.fTypeMask & 0xF);
|
|
}
|
|
|
|
public bool isIdentity() {
|
|
return this.getType() == 0;
|
|
}
|
|
|
|
public bool isScaleTranslate() {
|
|
return (this.getType() & ~(TypeMask.kScale_Mask | TypeMask.kTranslate_Mask)) == 0;
|
|
}
|
|
|
|
public bool isTranslate() {
|
|
return (this.getType() & ~(TypeMask.kTranslate_Mask)) == 0;
|
|
}
|
|
|
|
public bool rectStaysRect() {
|
|
if ((this.fTypeMask & kUnknown_Mask) != 0) {
|
|
this.fTypeMask = this.computeTypeMask();
|
|
}
|
|
|
|
return (this.fTypeMask & kRectStaysRect_Mask) != 0;
|
|
}
|
|
|
|
public bool preservesAxisAlignment() {
|
|
return this.rectStaysRect();
|
|
}
|
|
|
|
public bool hasPerspective() {
|
|
return (this.getPerspectiveTypeMaskOnly() & TypeMask.kPerspective_Mask) != 0;
|
|
}
|
|
|
|
public bool isSimilarity(float tol = ScalarUtils.kScalarNearlyZero) {
|
|
TypeMask mask = this.getType();
|
|
if (mask <= TypeMask.kTranslate_Mask) {
|
|
return true;
|
|
}
|
|
|
|
if ((mask & TypeMask.kPerspective_Mask) != 0) {
|
|
return false;
|
|
}
|
|
|
|
float mx = this.fMat[kMScaleX];
|
|
float my = this.fMat[kMScaleY];
|
|
// if no skew, can just compare scale factors
|
|
if ((mask & TypeMask.kAffine_Mask) == 0) {
|
|
return !ScalarUtils.ScalarNearlyZero(mx) &&
|
|
ScalarUtils.ScalarNearlyEqual(Mathf.Abs(mx), Mathf.Abs(my));
|
|
}
|
|
|
|
float sx = this.fMat[kMSkewX];
|
|
float sy = this.fMat[kMSkewY];
|
|
|
|
if (ScalarUtils.is_degenerate_2x2(mx, sx, sy, my)) {
|
|
return false;
|
|
}
|
|
|
|
// upper 2x2 is rotation/reflection + uniform scale if basis vectors
|
|
// are 90 degree rotations of each other
|
|
return (ScalarUtils.ScalarNearlyEqual(mx, my, tol) && ScalarUtils.ScalarNearlyEqual(sx, -sy, tol))
|
|
|| (ScalarUtils.ScalarNearlyEqual(mx, -my, tol) && ScalarUtils.ScalarNearlyEqual(sx, sy, tol));
|
|
}
|
|
|
|
public bool preservesRightAngles(float tol = ScalarUtils.kScalarNearlyZero) {
|
|
TypeMask mask = this.getType();
|
|
|
|
if (mask <= TypeMask.kTranslate_Mask) {
|
|
// identity, translate and/or scale
|
|
return true;
|
|
}
|
|
|
|
if ((mask & TypeMask.kPerspective_Mask) != 0) {
|
|
return false;
|
|
}
|
|
|
|
D.assert((mask & (TypeMask.kAffine_Mask | TypeMask.kScale_Mask)) != 0);
|
|
|
|
float mx = this.fMat[kMScaleX];
|
|
float my = this.fMat[kMScaleY];
|
|
float sx = this.fMat[kMSkewX];
|
|
float sy = this.fMat[kMSkewY];
|
|
|
|
if (ScalarUtils.is_degenerate_2x2(mx, sx, sy, my)) {
|
|
return false;
|
|
}
|
|
|
|
// upper 2x2 is scale + rotation/reflection if basis vectors are orthogonal
|
|
var dot = mx * sx + sy * my;
|
|
return ScalarUtils.ScalarNearlyZero(dot, tol * tol);
|
|
}
|
|
|
|
public const int kMScaleX = 0; //!< horizontal scale factor
|
|
public const int kMSkewX = 1; //!< horizontal skew factor
|
|
public const int kMTransX = 2; //!< horizontal translation
|
|
public const int kMSkewY = 3; //!< vertical skew factor
|
|
public const int kMScaleY = 4; //!< vertical scale factor
|
|
public const int kMTransY = 5; //!< vertical translation
|
|
public const int kMPersp0 = 6; //!< input x perspective factor
|
|
public const int kMPersp1 = 7; //!< input y perspective factor
|
|
public const int kMPersp2 = 8; //!< perspective bias
|
|
|
|
public float this[int index] {
|
|
get {
|
|
D.assert((uint) index < 9);
|
|
return this.fMat[index];
|
|
}
|
|
|
|
set {
|
|
D.assert((uint) index < 9);
|
|
this.fMat[index] = value;
|
|
this.setTypeMask(kUnknown_Mask);
|
|
}
|
|
}
|
|
|
|
public float getScaleX() {
|
|
return this.fMat[kMScaleX];
|
|
}
|
|
|
|
public float getScaleY() {
|
|
return this.fMat[kMScaleY];
|
|
}
|
|
|
|
public float getSkewY() {
|
|
return this.fMat[kMSkewY];
|
|
}
|
|
|
|
public float getSkewX() {
|
|
return this.fMat[kMSkewX];
|
|
}
|
|
|
|
public float getTranslateX() {
|
|
return this.fMat[kMTransX];
|
|
}
|
|
|
|
public float getTranslateY() {
|
|
return this.fMat[kMTransY];
|
|
}
|
|
|
|
public float getPerspX() {
|
|
return this.fMat[kMPersp0];
|
|
}
|
|
|
|
public float getPerspY() {
|
|
return this.fMat[kMPersp1];
|
|
}
|
|
|
|
public void setScaleX(float v) {
|
|
this[kMScaleX] = v;
|
|
}
|
|
|
|
public void setScaleY(float v) {
|
|
this[kMScaleY] = v;
|
|
}
|
|
|
|
public void setSkewY(float v) {
|
|
this[kMSkewY] = v;
|
|
}
|
|
|
|
public void setSkewX(float v) {
|
|
this[kMSkewX] = v;
|
|
}
|
|
|
|
public void setTranslateX(float v) {
|
|
this[kMTransX] = v;
|
|
}
|
|
|
|
public void setTranslateY(float v) {
|
|
this[kMTransY] = v;
|
|
}
|
|
|
|
public void setPerspX(float v) {
|
|
this[kMPersp0] = v;
|
|
}
|
|
|
|
public void setPerspY(float v) {
|
|
this[kMPersp1] = v;
|
|
}
|
|
|
|
public void setAll(
|
|
float scaleX, float skewX, float transX,
|
|
float skewY, float scaleY, float transY,
|
|
float persp0, float persp1, float persp2) {
|
|
this.fMat[kMScaleX] = scaleX;
|
|
this.fMat[kMSkewX] = skewX;
|
|
this.fMat[kMTransX] = transX;
|
|
this.fMat[kMSkewY] = skewY;
|
|
this.fMat[kMScaleY] = scaleY;
|
|
this.fMat[kMTransY] = transY;
|
|
this.fMat[kMPersp0] = persp0;
|
|
this.fMat[kMPersp1] = persp1;
|
|
this.fMat[kMPersp2] = persp2;
|
|
this.setTypeMask(kUnknown_Mask);
|
|
}
|
|
|
|
public void get9(float[] buffer) {
|
|
Array.Copy(this.fMat, buffer, 9);
|
|
}
|
|
|
|
public void set9(float[] buffer) {
|
|
Array.Copy(buffer, this.fMat, 9);
|
|
this.setTypeMask(kUnknown_Mask);
|
|
}
|
|
|
|
public void reset() {
|
|
this.fMat[kMScaleX] = this.fMat[kMScaleY] = this.fMat[kMPersp2] = 1;
|
|
this.fMat[kMSkewX] = this.fMat[kMSkewY] = this.fMat[kMTransX] =
|
|
this.fMat[kMTransY] = this.fMat[kMPersp0] = this.fMat[kMPersp1] = 0;
|
|
this.setTypeMask((int) TypeMask.kIdentity_Mask | kRectStaysRect_Mask);
|
|
}
|
|
|
|
public void setIdentity() {
|
|
this.reset();
|
|
}
|
|
|
|
public void setTranslate(float dx, float dy) {
|
|
if ((dx != 0) | (dy != 0)) {
|
|
this.fMat[kMTransX] = dx;
|
|
this.fMat[kMTransY] = dy;
|
|
|
|
this.fMat[kMScaleX] = this.fMat[kMScaleY] = this.fMat[kMPersp2] = 1;
|
|
this.fMat[kMSkewX] = this.fMat[kMSkewY] =
|
|
this.fMat[kMPersp0] = this.fMat[kMPersp1] = 0;
|
|
|
|
this.setTypeMask((int) TypeMask.kTranslate_Mask | kRectStaysRect_Mask);
|
|
} else {
|
|
this.reset();
|
|
}
|
|
}
|
|
|
|
public void setScale(float sx, float sy, float px, float py) {
|
|
if (1 == sx && 1 == sy) {
|
|
this.reset();
|
|
} else {
|
|
this.setScaleTranslate(sx, sy, px - sx * px, py - sy * py);
|
|
}
|
|
}
|
|
|
|
public void setScale(float sx, float sy) {
|
|
if (1 == sx && 1 == sy) {
|
|
this.reset();
|
|
} else {
|
|
this.fMat[kMScaleX] = sx;
|
|
this.fMat[kMScaleY] = sy;
|
|
this.fMat[kMPersp2] = 1;
|
|
|
|
this.fMat[kMTransX] = this.fMat[kMTransY] = this.fMat[kMSkewX] =
|
|
this.fMat[kMSkewY] = this.fMat[kMPersp0] = this.fMat[kMPersp1] = 0;
|
|
|
|
this.setTypeMask((int) TypeMask.kScale_Mask | kRectStaysRect_Mask);
|
|
}
|
|
}
|
|
|
|
public void setRotate(float degrees, float px, float py) {
|
|
float sinV, cosV;
|
|
sinV = ScalarUtils.ScalarSinCos(ScalarUtils.DegreesToRadians(degrees), out cosV);
|
|
this.setSinCos(sinV, cosV, px, py);
|
|
}
|
|
|
|
public void setRotate(float degrees) {
|
|
float sinV, cosV;
|
|
sinV = ScalarUtils.ScalarSinCos(ScalarUtils.DegreesToRadians(degrees), out cosV);
|
|
this.setSinCos(sinV, cosV);
|
|
}
|
|
|
|
public void setSinCos(float sinV, float cosV, float px, float py) {
|
|
var oneMinusCosV = 1 - cosV;
|
|
|
|
this.fMat[kMScaleX] = cosV;
|
|
this.fMat[kMSkewX] = -sinV;
|
|
this.fMat[kMTransX] = ScalarUtils.sdot(sinV, py, oneMinusCosV, px);
|
|
|
|
this.fMat[kMSkewY] = sinV;
|
|
this.fMat[kMScaleY] = cosV;
|
|
this.fMat[kMTransY] = ScalarUtils.sdot(-sinV, px, oneMinusCosV, py);
|
|
|
|
this.fMat[kMPersp0] = this.fMat[kMPersp1] = 0;
|
|
this.fMat[kMPersp2] = 1;
|
|
|
|
this.setTypeMask(kUnknown_Mask | kOnlyPerspectiveValid_Mask);
|
|
}
|
|
|
|
public void setSinCos(float sinV, float cosV) {
|
|
this.fMat[kMScaleX] = cosV;
|
|
this.fMat[kMSkewX] = -sinV;
|
|
this.fMat[kMTransX] = 0;
|
|
|
|
this.fMat[kMSkewY] = sinV;
|
|
this.fMat[kMScaleY] = cosV;
|
|
this.fMat[kMTransY] = 0;
|
|
|
|
this.fMat[kMPersp0] = this.fMat[kMPersp1] = 0;
|
|
this.fMat[kMPersp2] = 1;
|
|
|
|
this.setTypeMask(kUnknown_Mask | kOnlyPerspectiveValid_Mask);
|
|
}
|
|
|
|
public void setSkew(float kx, float ky, float px, float py) {
|
|
this.fMat[kMScaleX] = 1;
|
|
this.fMat[kMSkewX] = kx;
|
|
this.fMat[kMTransX] = -kx * py;
|
|
|
|
this.fMat[kMSkewY] = ky;
|
|
this.fMat[kMScaleY] = 1;
|
|
this.fMat[kMTransY] = -ky * px;
|
|
|
|
this.fMat[kMPersp0] = this.fMat[kMPersp1] = 0;
|
|
this.fMat[kMPersp2] = 1;
|
|
|
|
this.setTypeMask(kUnknown_Mask | kOnlyPerspectiveValid_Mask);
|
|
}
|
|
|
|
public void setSkew(float kx, float ky) {
|
|
this.fMat[kMScaleX] = 1;
|
|
this.fMat[kMSkewX] = kx;
|
|
this.fMat[kMTransX] = 0;
|
|
|
|
this.fMat[kMSkewY] = ky;
|
|
this.fMat[kMScaleY] = 1;
|
|
this.fMat[kMTransY] = 0;
|
|
|
|
this.fMat[kMPersp0] = this.fMat[kMPersp1] = 0;
|
|
this.fMat[kMPersp2] = 1;
|
|
|
|
this.setTypeMask(kUnknown_Mask | kOnlyPerspectiveValid_Mask);
|
|
}
|
|
|
|
static bool only_scale_and_translate(int mask) {
|
|
return 0 == (mask & (int) (TypeMask.kAffine_Mask | TypeMask.kPerspective_Mask));
|
|
}
|
|
|
|
static float rowcol3(float[] row, int i, float[] col, int j) {
|
|
return row[i] * col[j] + row[i + 1] * col[j + 3] + row[i + 2] * col[j + 6];
|
|
}
|
|
|
|
static float muladdmul(float a, float b, float c, float d) {
|
|
return (float) ((double) a * b + (double) c * d);
|
|
}
|
|
|
|
public void setConcat(Matrix3 a, Matrix3 b) {
|
|
TypeMask aType = a.getType();
|
|
TypeMask bType = b.getType();
|
|
|
|
if (a.isTriviallyIdentity()) {
|
|
this.copyFrom(b);
|
|
} else if (b.isTriviallyIdentity()) {
|
|
this.copyFrom(a);
|
|
} else if (only_scale_and_translate((int) aType | (int) bType)) {
|
|
this.setScaleTranslate(a.fMat[kMScaleX] * b.fMat[kMScaleX],
|
|
a.fMat[kMScaleY] * b.fMat[kMScaleY],
|
|
a.fMat[kMScaleX] * b.fMat[kMTransX] + a.fMat[kMTransX],
|
|
a.fMat[kMScaleY] * b.fMat[kMTransY] + a.fMat[kMTransY]);
|
|
} else {
|
|
Matrix3 tmp = new Matrix3();
|
|
|
|
if (((aType | bType) & TypeMask.kPerspective_Mask) != 0) {
|
|
tmp.fMat[kMScaleX] = rowcol3(a.fMat, 0, b.fMat, 0);
|
|
tmp.fMat[kMSkewX] = rowcol3(a.fMat, 0, b.fMat, 1);
|
|
tmp.fMat[kMTransX] = rowcol3(a.fMat, 0, b.fMat, 2);
|
|
tmp.fMat[kMSkewY] = rowcol3(a.fMat, 3, b.fMat, 0);
|
|
tmp.fMat[kMScaleY] = rowcol3(a.fMat, 3, b.fMat, 1);
|
|
tmp.fMat[kMTransY] = rowcol3(a.fMat, 3, b.fMat, 2);
|
|
tmp.fMat[kMPersp0] = rowcol3(a.fMat, 6, b.fMat, 0);
|
|
tmp.fMat[kMPersp1] = rowcol3(a.fMat, 6, b.fMat, 1);
|
|
tmp.fMat[kMPersp2] = rowcol3(a.fMat, 6, b.fMat, 2);
|
|
|
|
tmp.setTypeMask(kUnknown_Mask);
|
|
} else {
|
|
tmp.fMat[kMScaleX] = muladdmul(a.fMat[kMScaleX],
|
|
b.fMat[kMScaleX],
|
|
a.fMat[kMSkewX],
|
|
b.fMat[kMSkewY]);
|
|
|
|
tmp.fMat[kMSkewX] = muladdmul(a.fMat[kMScaleX],
|
|
b.fMat[kMSkewX],
|
|
a.fMat[kMSkewX],
|
|
b.fMat[kMScaleY]);
|
|
|
|
tmp.fMat[kMTransX] = muladdmul(a.fMat[kMScaleX],
|
|
b.fMat[kMTransX],
|
|
a.fMat[kMSkewX],
|
|
b.fMat[kMTransY]) + a.fMat[kMTransX];
|
|
|
|
tmp.fMat[kMSkewY] = muladdmul(a.fMat[kMSkewY],
|
|
b.fMat[kMScaleX],
|
|
a.fMat[kMScaleY],
|
|
b.fMat[kMSkewY]);
|
|
|
|
tmp.fMat[kMScaleY] = muladdmul(a.fMat[kMSkewY],
|
|
b.fMat[kMSkewX],
|
|
a.fMat[kMScaleY],
|
|
b.fMat[kMScaleY]);
|
|
|
|
tmp.fMat[kMTransY] = muladdmul(a.fMat[kMSkewY],
|
|
b.fMat[kMTransX],
|
|
a.fMat[kMScaleY],
|
|
b.fMat[kMTransY]) + a.fMat[kMTransY];
|
|
|
|
tmp.fMat[kMPersp0] = 0;
|
|
tmp.fMat[kMPersp1] = 0;
|
|
tmp.fMat[kMPersp2] = 1;
|
|
tmp.setTypeMask(kUnknown_Mask | kOnlyPerspectiveValid_Mask);
|
|
}
|
|
|
|
this.copyFrom(tmp);
|
|
}
|
|
}
|
|
|
|
public void preTranslate(float dx, float dy) {
|
|
var mask = this.getType();
|
|
|
|
if (mask <= TypeMask.kTranslate_Mask) {
|
|
this.fMat[kMTransX] += dx;
|
|
this.fMat[kMTransY] += dy;
|
|
} else if ((mask & TypeMask.kPerspective_Mask) != 0) {
|
|
var m = new Matrix3();
|
|
m.setTranslate(dx, dy);
|
|
this.preConcat(m);
|
|
return;
|
|
} else {
|
|
this.fMat[kMTransX] += this.fMat[kMScaleX] * dx + this.fMat[kMSkewX] * dy;
|
|
this.fMat[kMTransY] += this.fMat[kMSkewY] * dx + this.fMat[kMScaleY] * dy;
|
|
}
|
|
|
|
this.updateTranslateMask();
|
|
}
|
|
|
|
public void preScale(float sx, float sy, float px, float py) {
|
|
if (1 == sx && 1 == sy) {
|
|
return;
|
|
}
|
|
|
|
var m = new Matrix3();
|
|
m.setScale(sx, sy, px, py);
|
|
this.preConcat(m);
|
|
}
|
|
|
|
public void preScale(float sx, float sy) {
|
|
if (1 == sx && 1 == sy) {
|
|
return;
|
|
}
|
|
|
|
// the assumption is that these multiplies are very cheap, and that
|
|
// a full concat and/or just computing the matrix type is more expensive.
|
|
// Also, the fixed-point case checks for overflow, but the float doesn't,
|
|
// so we can get away with these blind multiplies.
|
|
|
|
this.fMat[kMScaleX] *= sx;
|
|
this.fMat[kMSkewY] *= sx;
|
|
this.fMat[kMPersp0] *= sx;
|
|
|
|
this.fMat[kMSkewX] *= sy;
|
|
this.fMat[kMScaleY] *= sy;
|
|
this.fMat[kMPersp1] *= sy;
|
|
|
|
// Attempt to simplify our type when applying an inverse scale.
|
|
// TODO: The persp/affine preconditions are in place to keep the mask consistent with
|
|
// what computeTypeMask() would produce (persp/skew always implies kScale).
|
|
// We should investigate whether these flag dependencies are truly needed.
|
|
if (this.fMat[kMScaleX] == 1 && this.fMat[kMScaleY] == 1
|
|
&& (this.fTypeMask &
|
|
(int) (TypeMask.kPerspective_Mask | TypeMask.kAffine_Mask)) == 0) {
|
|
this.clearTypeMask((int) TypeMask.kScale_Mask);
|
|
} else {
|
|
this.orTypeMask((int) TypeMask.kScale_Mask);
|
|
}
|
|
}
|
|
|
|
public void preRotate(float degrees, float px, float py) {
|
|
var m = new Matrix3();
|
|
m.setRotate(degrees, px, py);
|
|
this.preConcat(m);
|
|
}
|
|
|
|
public void preRotate(float degrees) {
|
|
var m = new Matrix3();
|
|
m.setRotate(degrees);
|
|
this.preConcat(m);
|
|
}
|
|
|
|
public void preSkew(float kx, float ky, float px, float py) {
|
|
var m = new Matrix3();
|
|
m.setSkew(kx, ky, px, py);
|
|
this.preConcat(m);
|
|
}
|
|
|
|
public void preSkew(float kx, float ky) {
|
|
var m = new Matrix3();
|
|
m.setSkew(kx, ky);
|
|
this.preConcat(m);
|
|
}
|
|
|
|
public void preConcat(Matrix3 other) {
|
|
// check for identity first, so we don't do a needless copy of ourselves
|
|
// to ourselves inside setConcat()
|
|
if (!other.isIdentity()) {
|
|
this.setConcat(this, other);
|
|
}
|
|
}
|
|
|
|
public void postTranslate(float dx, float dy) {
|
|
if (this.hasPerspective()) {
|
|
var m = new Matrix3();
|
|
m.setTranslate(dx, dy);
|
|
this.postConcat(m);
|
|
} else {
|
|
this.fMat[kMTransX] += dx;
|
|
this.fMat[kMTransY] += dy;
|
|
this.updateTranslateMask();
|
|
}
|
|
}
|
|
|
|
public void postScale(float sx, float sy, float px, float py) {
|
|
if (1 == sx && 1 == sy) {
|
|
return;
|
|
}
|
|
|
|
var m = new Matrix3();
|
|
m.setScale(sx, sy, px, py);
|
|
this.postConcat(m);
|
|
}
|
|
|
|
public void postScale(float sx, float sy) {
|
|
if (1 == sx && 1 == sy) {
|
|
return;
|
|
}
|
|
|
|
var m = new Matrix3();
|
|
m.setScale(sx, sy);
|
|
this.postConcat(m);
|
|
}
|
|
|
|
public void postRotate(float degrees, float px, float py) {
|
|
var m = new Matrix3();
|
|
m.setRotate(degrees, px, py);
|
|
this.postConcat(m);
|
|
}
|
|
|
|
public void postRotate(float degrees) {
|
|
var m = new Matrix3();
|
|
m.setRotate(degrees);
|
|
this.postConcat(m);
|
|
}
|
|
|
|
public void postSkew(float kx, float ky, float px, float py) {
|
|
var m = new Matrix3();
|
|
m.setSkew(kx, ky, px, py);
|
|
this.postConcat(m);
|
|
}
|
|
|
|
public void postSkew(float kx, float ky) {
|
|
var m = new Matrix3();
|
|
m.setSkew(kx, ky);
|
|
this.postConcat(m);
|
|
}
|
|
|
|
public void postConcat(Matrix3 mat) {
|
|
if (!mat.isIdentity()) {
|
|
this.setConcat(mat, this);
|
|
}
|
|
}
|
|
|
|
public bool invert(Matrix3 inverse) {
|
|
if (this.isIdentity()) {
|
|
if (inverse != null) {
|
|
inverse.reset();
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
return this.invertNonIdentity(inverse);
|
|
}
|
|
|
|
public void mapPoints(Offset[] dst, Offset[] src) {
|
|
D.assert(dst != null && src != null && dst.Length == src.Length);
|
|
this.getMapPtsProc()(this, dst, src, src.Length);
|
|
}
|
|
|
|
public void mapPoints(Offset[] pts) {
|
|
this.mapPoints(pts, pts);
|
|
}
|
|
|
|
public Offset mapXY(float x, float y) {
|
|
Offset result;
|
|
this.getMapXYProc()(this, x, y, out result);
|
|
return result;
|
|
}
|
|
|
|
public bool mapRect(out Rect dst, Rect src) {
|
|
if (this.getType() <= TypeMask.kTranslate_Mask) {
|
|
var tx = this.fMat[kMTransX];
|
|
var ty = this.fMat[kMTransY];
|
|
|
|
dst = Rect.fromLTRB(
|
|
src.left + tx,
|
|
src.top + ty,
|
|
src.right + tx,
|
|
src.bottom + ty
|
|
);
|
|
|
|
return true;
|
|
}
|
|
|
|
if (this.isScaleTranslate()) {
|
|
this.mapRectScaleTranslate(out dst, src);
|
|
return true;
|
|
} else {
|
|
var points = new[] {
|
|
new Offset(src.left, src.top),
|
|
new Offset(src.right, src.top),
|
|
new Offset(src.right, src.bottom),
|
|
new Offset(src.left, src.bottom),
|
|
};
|
|
|
|
this.mapPoints(points);
|
|
|
|
var minX = points[0].dx;
|
|
var minY = points[0].dy;
|
|
var maxX = points[0].dx;
|
|
var maxY = points[0].dy;
|
|
|
|
for (int i = 1; i < 4; ++i) {
|
|
minX = Math.Min(minX, points[i].dx);
|
|
minY = Math.Min(minY, points[i].dy);
|
|
maxX = Math.Max(maxX, points[i].dx);
|
|
maxY = Math.Max(maxY, points[i].dy);
|
|
}
|
|
|
|
dst = Rect.fromLTRB(minX, minY, maxX, maxY);
|
|
return this.rectStaysRect(); // might still return true if rotated by 90, etc.
|
|
}
|
|
}
|
|
|
|
public bool mapRect(ref Rect rect) {
|
|
return this.mapRect(out rect, rect);
|
|
}
|
|
|
|
public Rect mapRect(Rect src) {
|
|
Rect dst;
|
|
this.mapRect(out dst, src);
|
|
return dst;
|
|
}
|
|
|
|
public void mapRectScaleTranslate(out Rect dst, Rect src) {
|
|
D.assert(this.isScaleTranslate());
|
|
|
|
var sx = this.fMat[kMScaleX];
|
|
var sy = this.fMat[kMScaleY];
|
|
var tx = this.fMat[kMTransX];
|
|
var ty = this.fMat[kMTransY];
|
|
|
|
dst = Rect.fromLTRB(
|
|
src.left * sx + tx,
|
|
src.top * sy + ty,
|
|
src.right * sx + tx,
|
|
src.bottom * sy + ty
|
|
);
|
|
}
|
|
|
|
public static bool operator ==(Matrix3 a, Matrix3 b) {
|
|
if (ReferenceEquals(a, null) && ReferenceEquals(b, null)) {
|
|
return true;
|
|
}
|
|
|
|
if (ReferenceEquals(a, null) || ReferenceEquals(b, null)) {
|
|
return false;
|
|
}
|
|
|
|
var ma = a.fMat;
|
|
var mb = b.fMat;
|
|
|
|
return ma[0] == mb[0] && ma[1] == mb[1] && ma[2] == mb[2] &&
|
|
ma[3] == mb[3] && ma[4] == mb[4] && ma[5] == mb[5] &&
|
|
ma[6] == mb[6] && ma[7] == mb[7] && ma[8] == mb[8];
|
|
}
|
|
|
|
public static bool operator !=(Matrix3 a, Matrix3 b) {
|
|
return !(a == b);
|
|
}
|
|
|
|
public static Matrix3 I() {
|
|
var m = new Matrix3();
|
|
m.reset();
|
|
return m;
|
|
}
|
|
|
|
public static Matrix3 concat(Matrix3 a, Matrix3 b) {
|
|
Matrix3 result = new Matrix3();
|
|
result.setConcat(a, b);
|
|
return result;
|
|
}
|
|
|
|
public void dirtyMatrixTypeCache() {
|
|
this.setTypeMask(kUnknown_Mask);
|
|
}
|
|
|
|
public void setScaleTranslate(float sx, float sy, float tx, float ty) {
|
|
this.fMat[kMScaleX] = sx;
|
|
this.fMat[kMSkewX] = 0;
|
|
this.fMat[kMTransX] = tx;
|
|
|
|
this.fMat[kMSkewY] = 0;
|
|
this.fMat[kMScaleY] = sy;
|
|
this.fMat[kMTransY] = ty;
|
|
|
|
this.fMat[kMPersp0] = 0;
|
|
this.fMat[kMPersp1] = 0;
|
|
this.fMat[kMPersp2] = 1;
|
|
|
|
int mask = 0;
|
|
if (sx != 1 || sy != 1) {
|
|
mask |= (int) TypeMask.kScale_Mask;
|
|
}
|
|
|
|
if (tx != 0 || ty != 0) {
|
|
mask |= (int) TypeMask.kTranslate_Mask;
|
|
}
|
|
|
|
this.setTypeMask(mask | kRectStaysRect_Mask);
|
|
}
|
|
|
|
public bool isFinite() {
|
|
return ScalarUtils.ScalarsAreFinite(this.fMat, 9);
|
|
}
|
|
|
|
// PRIVATE
|
|
|
|
const int kRectStaysRect_Mask = 0x10;
|
|
|
|
const int kOnlyPerspectiveValid_Mask = 0x40;
|
|
|
|
const int kUnknown_Mask = 0x80;
|
|
|
|
const int kORableMasks =
|
|
(int)
|
|
(TypeMask.kTranslate_Mask |
|
|
TypeMask.kScale_Mask |
|
|
TypeMask.kAffine_Mask |
|
|
TypeMask.kPerspective_Mask);
|
|
|
|
const int kAllMasks =
|
|
(int)
|
|
(TypeMask.kTranslate_Mask |
|
|
TypeMask.kScale_Mask |
|
|
TypeMask.kAffine_Mask |
|
|
TypeMask.kPerspective_Mask) |
|
|
kRectStaysRect_Mask;
|
|
|
|
readonly float[] fMat = new float[9];
|
|
int fTypeMask = 0;
|
|
|
|
Matrix3() {
|
|
}
|
|
|
|
enum TypeShift {
|
|
kTranslate_Shift,
|
|
kScale_Shift,
|
|
kAffine_Shift,
|
|
kPerspective_Shift,
|
|
kRectStaysRect_Shift
|
|
}
|
|
|
|
static void ComputeInv(float[] dst, float[] src, double invDet, bool isPersp) {
|
|
D.assert(src != dst);
|
|
D.assert(src != null && dst != null);
|
|
|
|
if (isPersp) {
|
|
dst[kMScaleX] =
|
|
ScalarUtils.scross_dscale(src[kMScaleY], src[kMPersp2], src[kMTransY],
|
|
src[kMPersp1], invDet);
|
|
dst[kMSkewX] =
|
|
ScalarUtils.scross_dscale(src[kMTransX], src[kMPersp1], src[kMSkewX],
|
|
src[kMPersp2], invDet);
|
|
dst[kMTransX] =
|
|
ScalarUtils.scross_dscale(src[kMSkewX], src[kMTransY], src[kMTransX],
|
|
src[kMScaleY], invDet);
|
|
|
|
dst[kMSkewY] =
|
|
ScalarUtils.scross_dscale(src[kMTransY], src[kMPersp0], src[kMSkewY],
|
|
src[kMPersp2], invDet);
|
|
dst[kMScaleY] =
|
|
ScalarUtils.scross_dscale(src[kMScaleX], src[kMPersp2], src[kMTransX],
|
|
src[kMPersp0], invDet);
|
|
dst[kMTransY] =
|
|
ScalarUtils.scross_dscale(src[kMTransX], src[kMSkewY], src[kMScaleX],
|
|
src[kMTransY], invDet);
|
|
|
|
dst[kMPersp0] =
|
|
ScalarUtils.scross_dscale(src[kMSkewY], src[kMPersp1], src[kMScaleY],
|
|
src[kMPersp0], invDet);
|
|
dst[kMPersp1] =
|
|
ScalarUtils.scross_dscale(src[kMSkewX], src[kMPersp0], src[kMScaleX],
|
|
src[kMPersp1], invDet);
|
|
dst[kMPersp2] =
|
|
ScalarUtils.scross_dscale(src[kMScaleX], src[kMScaleY], src[kMSkewX],
|
|
src[kMSkewY], invDet);
|
|
} else {
|
|
// not perspective
|
|
dst[kMScaleX] = (float) (src[kMScaleY] * invDet);
|
|
dst[kMSkewX] = (float) (-src[kMSkewX] * invDet);
|
|
dst[kMTransX] =
|
|
ScalarUtils.dcross_dscale(src[kMSkewX], src[kMTransY], src[kMScaleY],
|
|
src[kMTransX], invDet);
|
|
|
|
dst[kMSkewY] = (float) (-src[kMSkewY] * invDet);
|
|
dst[kMScaleY] = (float) (src[kMScaleX] * invDet);
|
|
dst[kMTransY] =
|
|
ScalarUtils.dcross_dscale(src[kMSkewY], src[kMTransX], src[kMScaleX],
|
|
src[kMTransY], invDet);
|
|
|
|
dst[kMPersp0] = 0;
|
|
dst[kMPersp1] = 0;
|
|
dst[kMPersp2] = 1;
|
|
}
|
|
}
|
|
|
|
const int kScalar1Int = 0x3f800000;
|
|
|
|
int computeTypeMask() {
|
|
int mask = 0;
|
|
|
|
if (this.fMat[kMPersp0] != 0 || this.fMat[kMPersp1] != 0 ||
|
|
this.fMat[kMPersp2] != 1) {
|
|
// Once it is determined that that this is a perspective transform,
|
|
// all other flags are moot as far as optimizations are concerned.
|
|
return kORableMasks;
|
|
}
|
|
|
|
if (this.fMat[kMTransX] != 0 || this.fMat[kMTransY] != 0) {
|
|
mask |= (int) TypeMask.kTranslate_Mask;
|
|
}
|
|
|
|
int m00 = ScalarUtils.ScalarAs2sCompliment(this.fMat[kMScaleX]);
|
|
int m01 = ScalarUtils.ScalarAs2sCompliment(this.fMat[kMSkewX]);
|
|
int m10 = ScalarUtils.ScalarAs2sCompliment(this.fMat[kMSkewY]);
|
|
int m11 = ScalarUtils.ScalarAs2sCompliment(this.fMat[kMScaleY]);
|
|
|
|
if ((m01 != 0) | (m10 != 0)) {
|
|
// The skew components may be scale-inducing, unless we are dealing
|
|
// with a pure rotation. Testing for a pure rotation is expensive,
|
|
// so we opt for being conservative by always setting the scale bit.
|
|
// along with affine.
|
|
// By doing this, we are also ensuring that matrices have the same
|
|
// type masks as their inverses.
|
|
mask |= (int) TypeMask.kAffine_Mask | (int) TypeMask.kScale_Mask;
|
|
|
|
// For rectStaysRect, in the affine case, we only need check that
|
|
// the primary diagonal is all zeros and that the secondary diagonal
|
|
// is all non-zero.
|
|
|
|
// map non-zero to 1
|
|
m01 = m01 != 0 ? 1 : 0;
|
|
m10 = m10 != 0 ? 1 : 0;
|
|
|
|
int dp0 = 0 == (m00 | m11) ? 1 : 0; // true if both are 0
|
|
int ds1 = m01 & m10; // true if both are 1
|
|
|
|
mask |= (dp0 & ds1) << (int) TypeShift.kRectStaysRect_Shift;
|
|
} else {
|
|
// Only test for scale explicitly if not affine, since affine sets the
|
|
// scale bit.
|
|
if (((m00 ^ kScalar1Int) | (m11 ^ kScalar1Int)) != 0) {
|
|
mask |= (int) TypeMask.kScale_Mask;
|
|
}
|
|
|
|
// Not affine, therefore we already know secondary diagonal is
|
|
// all zeros, so we just need to check that primary diagonal is
|
|
// all non-zero.
|
|
|
|
// map non-zero to 1
|
|
m00 = m00 != 0 ? 1 : 0;
|
|
m11 = m11 != 0 ? 1 : 0;
|
|
|
|
// record if the (p)rimary diagonal is all non-zero
|
|
mask |= (m00 & m11) << (int) TypeShift.kRectStaysRect_Shift;
|
|
}
|
|
|
|
return mask;
|
|
}
|
|
|
|
int computePerspectiveTypeMask() {
|
|
// Benchmarking suggests that replacing this set of floatAs2sCompliment
|
|
// is a win, but replacing those below is not. We don't yet understand
|
|
// that result.
|
|
if (this.fMat[kMPersp0] != 0 || this.fMat[kMPersp1] != 0 ||
|
|
this.fMat[kMPersp2] != 1) {
|
|
// If this is a perspective transform, we return true for all other
|
|
// transform flags - this does not disable any optimizations, respects
|
|
// the rule that the type mask must be conservative, and speeds up
|
|
// type mask computation.
|
|
return kORableMasks;
|
|
}
|
|
|
|
return kOnlyPerspectiveValid_Mask | kUnknown_Mask;
|
|
}
|
|
|
|
|
|
void setTypeMask(int mask) {
|
|
D.assert(kUnknown_Mask == mask || (mask & kAllMasks) == mask ||
|
|
((kUnknown_Mask | kOnlyPerspectiveValid_Mask) & mask)
|
|
== (kUnknown_Mask | kOnlyPerspectiveValid_Mask));
|
|
this.fTypeMask = mask;
|
|
}
|
|
|
|
void orTypeMask(int mask) {
|
|
D.assert((mask & kORableMasks) == mask);
|
|
this.fTypeMask |= mask;
|
|
}
|
|
|
|
void clearTypeMask(int mask) {
|
|
// only allow a valid mask
|
|
D.assert((mask & kAllMasks) == mask);
|
|
this.fTypeMask &= ~mask;
|
|
}
|
|
|
|
TypeMask getPerspectiveTypeMaskOnly() {
|
|
if ((this.fTypeMask & kUnknown_Mask) != 0 &&
|
|
(this.fTypeMask & kOnlyPerspectiveValid_Mask) == 0) {
|
|
this.fTypeMask = this.computePerspectiveTypeMask();
|
|
}
|
|
|
|
return (TypeMask) (this.fTypeMask & 0xF);
|
|
}
|
|
|
|
bool isTriviallyIdentity() {
|
|
if ((this.fTypeMask & kUnknown_Mask) != 0) {
|
|
return false;
|
|
}
|
|
|
|
return (this.fTypeMask & 0xF) == 0;
|
|
}
|
|
|
|
void updateTranslateMask() {
|
|
if ((this.fMat[kMTransX] != 0) | (this.fMat[kMTransY] != 0)) {
|
|
this.fTypeMask |= (int) TypeMask.kTranslate_Mask;
|
|
} else {
|
|
this.fTypeMask &= ~(int) TypeMask.kTranslate_Mask;
|
|
}
|
|
}
|
|
|
|
delegate void MapXYProc(Matrix3 mat, float x, float y, out Offset result);
|
|
|
|
static readonly MapXYProc[] gMapXYProcs = {
|
|
Identity_xy, Trans_xy,
|
|
Scale_xy, ScaleTrans_xy,
|
|
Rot_xy, RotTrans_xy,
|
|
Rot_xy, RotTrans_xy,
|
|
// repeat the persp proc 8 times
|
|
Persp_xy, Persp_xy,
|
|
Persp_xy, Persp_xy,
|
|
Persp_xy, Persp_xy,
|
|
Persp_xy, Persp_xy
|
|
};
|
|
|
|
static MapXYProc GetMapXYProc(TypeMask mask) {
|
|
D.assert(((int) mask & ~kAllMasks) == 0);
|
|
return gMapXYProcs[(int) mask & kAllMasks];
|
|
}
|
|
|
|
MapXYProc getMapXYProc() {
|
|
return GetMapXYProc(this.getType());
|
|
}
|
|
|
|
static void Identity_xy(Matrix3 m, float sx, float sy, out Offset result) {
|
|
D.assert(0 == m.getType());
|
|
|
|
result = new Offset(sx, sy);
|
|
}
|
|
|
|
static void Trans_xy(Matrix3 m, float sx, float sy, out Offset result) {
|
|
D.assert(m.getType() == TypeMask.kTranslate_Mask);
|
|
|
|
result = new Offset(sx + m.fMat[kMTransX], sy + m.fMat[kMTransY]);
|
|
}
|
|
|
|
static void Scale_xy(Matrix3 m, float sx, float sy, out Offset result) {
|
|
D.assert((m.getType() & (TypeMask.kScale_Mask | TypeMask.kAffine_Mask | TypeMask.kPerspective_Mask))
|
|
== TypeMask.kScale_Mask);
|
|
D.assert(0 == m.fMat[kMTransX]);
|
|
D.assert(0 == m.fMat[kMTransY]);
|
|
|
|
result = new Offset(sx * m.fMat[kMScaleX], sy * m.fMat[kMScaleY]);
|
|
}
|
|
|
|
static void ScaleTrans_xy(Matrix3 m, float sx, float sy, out Offset result) {
|
|
D.assert((m.getType() & (TypeMask.kScale_Mask | TypeMask.kAffine_Mask | TypeMask.kPerspective_Mask))
|
|
== TypeMask.kScale_Mask);
|
|
|
|
result = new Offset(sx * m.fMat[kMScaleX] + m.fMat[kMTransX],
|
|
sy * m.fMat[kMScaleY] + m.fMat[kMTransY]);
|
|
}
|
|
|
|
static void Rot_xy(Matrix3 m, float sx, float sy, out Offset result) {
|
|
D.assert((m.getType() & (TypeMask.kAffine_Mask | TypeMask.kPerspective_Mask)) == TypeMask.kAffine_Mask);
|
|
D.assert(0 == m.fMat[kMTransX]);
|
|
D.assert(0 == m.fMat[kMTransY]);
|
|
|
|
result = new Offset(
|
|
ScalarUtils.sdot(sx, m.fMat[kMScaleX], sy, m.fMat[kMSkewX]),
|
|
ScalarUtils.sdot(sx, m.fMat[kMSkewY], sy, m.fMat[kMScaleY]));
|
|
}
|
|
|
|
static void RotTrans_xy(Matrix3 m, float sx, float sy, out Offset result) {
|
|
D.assert((m.getType() & (TypeMask.kAffine_Mask | TypeMask.kPerspective_Mask)) == TypeMask.kAffine_Mask);
|
|
|
|
result = new Offset(
|
|
ScalarUtils.sdot(sx, m.fMat[kMScaleX], sy, m.fMat[kMSkewX]) + m.fMat[kMTransX],
|
|
ScalarUtils.sdot(sx, m.fMat[kMSkewY], sy, m.fMat[kMScaleY]) + m.fMat[kMTransY]);
|
|
}
|
|
|
|
static void Persp_xy(Matrix3 m, float sx, float sy, out Offset result) {
|
|
D.assert(m.hasPerspective());
|
|
|
|
float x = ScalarUtils.sdot(sx, m.fMat[kMScaleX], sy, m.fMat[kMSkewX]) +
|
|
m.fMat[kMTransX];
|
|
float y = ScalarUtils.sdot(sx, m.fMat[kMSkewY], sy, m.fMat[kMScaleY]) +
|
|
m.fMat[kMTransY];
|
|
float z = ScalarUtils.sdot(sx, m.fMat[kMPersp0], sy, m.fMat[kMPersp1]) +
|
|
m.fMat[kMPersp2];
|
|
if (z != 0) {
|
|
z = 1 / z;
|
|
}
|
|
|
|
result = new Offset(x * z, y * z);
|
|
}
|
|
|
|
delegate void MapPtsProc(Matrix3 mat, Offset[] dst, Offset[] src, int count);
|
|
|
|
static readonly MapPtsProc[] gMapPtsProcs = {
|
|
Identity_pts, Trans_pts,
|
|
Scale_pts, Scale_pts,
|
|
Affine_pts, Affine_pts,
|
|
Affine_pts, Affine_pts,
|
|
// repeat the persp proc 8 times
|
|
Persp_pts, Persp_pts,
|
|
Persp_pts, Persp_pts,
|
|
Persp_pts, Persp_pts,
|
|
Persp_pts, Persp_pts
|
|
};
|
|
|
|
static MapPtsProc GetMapPtsProc(TypeMask mask) {
|
|
D.assert(((int) mask & ~kAllMasks) == 0);
|
|
return gMapPtsProcs[(int) mask & kAllMasks];
|
|
}
|
|
|
|
MapPtsProc getMapPtsProc() {
|
|
return GetMapPtsProc(this.getType());
|
|
}
|
|
|
|
static void Identity_pts(Matrix3 m, Offset[] dst, Offset[] src, int count) {
|
|
D.assert(m.getType() == 0);
|
|
|
|
if (dst != src && count > 0) {
|
|
Array.Copy(src, dst, count);
|
|
}
|
|
}
|
|
|
|
static void Trans_pts(Matrix3 m, Offset[] dst, Offset[] src, int count) {
|
|
D.assert(m.getType() <= TypeMask.kTranslate_Mask);
|
|
if (count > 0) {
|
|
var tx = m.getTranslateX();
|
|
var ty = m.getTranslateY();
|
|
for (int i = 0; i < count; ++i) {
|
|
dst[i] = new Offset(src[i].dx + tx, src[i].dy + ty);
|
|
}
|
|
}
|
|
}
|
|
|
|
static void Scale_pts(Matrix3 m, Offset[] dst, Offset[] src, int count) {
|
|
D.assert(m.getType() <= (TypeMask.kScale_Mask | TypeMask.kTranslate_Mask));
|
|
if (count > 0) {
|
|
var tx = m.getTranslateX();
|
|
var ty = m.getTranslateY();
|
|
var sx = m.getScaleX();
|
|
var sy = m.getScaleY();
|
|
|
|
for (int i = 0; i < count; ++i) {
|
|
dst[i] = new Offset(src[i].dx * sx + tx, src[i].dy * sy + ty);
|
|
}
|
|
}
|
|
}
|
|
|
|
static void Persp_pts(Matrix3 m, Offset[] dst, Offset[] src, int count) {
|
|
D.assert(m.hasPerspective());
|
|
|
|
if (count > 0) {
|
|
for (int i = 0; i < count; ++i) {
|
|
var sy = (float) src[i].dy;
|
|
var sx = (float) src[i].dx;
|
|
var x = ScalarUtils.sdot(sx, m.fMat[kMScaleX], sy, m.fMat[kMSkewX]) +
|
|
m.fMat[kMTransX];
|
|
var y = ScalarUtils.sdot(sx, m.fMat[kMSkewY], sy, m.fMat[kMScaleY]) +
|
|
m.fMat[kMTransY];
|
|
var z = ScalarUtils.sdot(sx, m.fMat[kMPersp0], sy, m.fMat[kMPersp1]) +
|
|
m.fMat[kMPersp2];
|
|
if (z != 0) {
|
|
z = 1 / z;
|
|
}
|
|
|
|
dst[i] = new Offset(x * z, y * z);
|
|
}
|
|
}
|
|
}
|
|
|
|
static void Affine_pts(Matrix3 m, Offset[] dst, Offset[] src, int count) {
|
|
D.assert(m.getType() != TypeMask.kPerspective_Mask);
|
|
if (count > 0) {
|
|
var tx = m.getTranslateX();
|
|
var ty = m.getTranslateY();
|
|
var sx = m.getScaleX();
|
|
var sy = m.getScaleY();
|
|
var kx = m.getSkewX();
|
|
var ky = m.getSkewY();
|
|
|
|
for (int i = 0; i < count; ++i) {
|
|
dst[i] = new Offset(
|
|
src[i].dx * sx + src[i].dy * kx + tx,
|
|
src[i].dx * ky + src[i].dy * sy + ty);
|
|
}
|
|
}
|
|
}
|
|
|
|
bool invertNonIdentity(Matrix3 inv) {
|
|
D.assert(!this.isIdentity());
|
|
|
|
TypeMask mask = this.getType();
|
|
|
|
if (0 == (mask & ~(TypeMask.kScale_Mask | TypeMask.kTranslate_Mask))) {
|
|
bool invertible = true;
|
|
if (inv != null) {
|
|
if ((mask & TypeMask.kScale_Mask) != 0) {
|
|
var invX = this.fMat[kMScaleX];
|
|
var invY = this.fMat[kMScaleY];
|
|
if (0 == invX || 0 == invY) {
|
|
return false;
|
|
}
|
|
|
|
invX = 1f / invX;
|
|
invY = 1f / invY;
|
|
|
|
// Must be careful when writing to inv, since it may be the
|
|
// same memory as this.
|
|
|
|
inv.fMat[kMSkewX] = inv.fMat[kMSkewY] =
|
|
inv.fMat[kMPersp0] = inv.fMat[kMPersp1] = 0;
|
|
|
|
inv.fMat[kMScaleX] = invX;
|
|
inv.fMat[kMScaleY] = invY;
|
|
inv.fMat[kMPersp2] = 1;
|
|
inv.fMat[kMTransX] = -this.fMat[kMTransX] * invX;
|
|
inv.fMat[kMTransY] = -this.fMat[kMTransY] * invY;
|
|
|
|
inv.setTypeMask((int) mask | kRectStaysRect_Mask);
|
|
} else {
|
|
// translate only
|
|
inv.setTranslate(-this.fMat[kMTransX], -this.fMat[kMTransY]);
|
|
}
|
|
} else {
|
|
// inv is nullptr, just check if we're invertible
|
|
if (this.fMat[kMScaleX] == 0 || this.fMat[kMScaleY] == 0) {
|
|
invertible = false;
|
|
}
|
|
}
|
|
|
|
return invertible;
|
|
}
|
|
|
|
int isPersp = (int) (mask & TypeMask.kPerspective_Mask);
|
|
double invDet = ScalarUtils.inv_determinant(this.fMat, isPersp);
|
|
|
|
if (invDet == 0) {
|
|
// underflow
|
|
return false;
|
|
}
|
|
|
|
bool applyingInPlace = (inv == this);
|
|
|
|
var tmp = inv;
|
|
|
|
if (applyingInPlace || null == tmp) {
|
|
tmp = new Matrix3(); // we either need to avoid trampling memory or have no memory
|
|
}
|
|
|
|
ComputeInv(tmp.fMat, this.fMat, invDet, isPersp != 0);
|
|
if (!tmp.isFinite()) {
|
|
return false;
|
|
}
|
|
|
|
tmp.setTypeMask(this.fTypeMask);
|
|
|
|
if (applyingInPlace) {
|
|
Array.Copy(tmp.fMat, inv.fMat, 9); // need to copy answer back
|
|
inv.fTypeMask = tmp.fTypeMask;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
public bool Equals(Matrix3 other) {
|
|
if (ReferenceEquals(null, other)) {
|
|
return false;
|
|
}
|
|
if (ReferenceEquals(this, other)) {
|
|
return true;
|
|
}
|
|
return this == other;
|
|
}
|
|
|
|
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((Matrix3) obj);
|
|
}
|
|
|
|
public override int GetHashCode() {
|
|
unchecked {
|
|
int hash = 17;
|
|
foreach (var element in this.fMat) {
|
|
hash = hash * 31 + element.GetHashCode();
|
|
}
|
|
|
|
return hash;
|
|
}
|
|
}
|
|
|
|
public override string ToString() {
|
|
return "Matrix3(" + string.Join(",", Array.ConvertAll(this.fMat, i => i.ToString())) + ")";
|
|
}
|
|
|
|
|
|
public Offset mapPoint(Offset point) {
|
|
return this.mapXY((float) point.dx, (float) point.dy);
|
|
}
|
|
}
|
|
|
|
class ScalarUtils {
|
|
public const float kScalarNearlyZero = 1f / (1 << 12);
|
|
|
|
public static bool ScalarNearlyZero(float x, float tolerance = kScalarNearlyZero) {
|
|
D.assert(tolerance >= 0);
|
|
return Mathf.Abs(x) <= tolerance;
|
|
}
|
|
|
|
public static bool ScalarNearlyEqual(float x, float y, float tolerance = kScalarNearlyZero) {
|
|
D.assert(tolerance >= 0);
|
|
return Mathf.Abs(x - y) <= tolerance;
|
|
}
|
|
|
|
public static float DegreesToRadians(float degrees) {
|
|
return degrees * (Mathf.PI / 180);
|
|
}
|
|
|
|
public static float RadiansToDegrees(float radians) {
|
|
return radians * (180 / Mathf.PI);
|
|
}
|
|
|
|
public static float ScalarSinCos(float radians, out float cosValue) {
|
|
float sinValue = Mathf.Sin(radians);
|
|
|
|
cosValue = Mathf.Cos(radians);
|
|
if (ScalarNearlyZero(cosValue)) {
|
|
cosValue = 0;
|
|
}
|
|
|
|
if (ScalarNearlyZero(sinValue)) {
|
|
sinValue = 0;
|
|
}
|
|
|
|
return sinValue;
|
|
}
|
|
|
|
public static bool ScalarsAreFinite(float[] array, int count) {
|
|
float prod = 0;
|
|
for (int i = 0; i < count; ++i) {
|
|
prod *= array[i];
|
|
}
|
|
|
|
// At this point, prod will either be NaN or 0
|
|
return prod == 0; // if prod is NaN, this check will return false
|
|
}
|
|
|
|
public static int ScalarAs2sCompliment(float x) {
|
|
var result = BitConverter.ToInt32(BitConverter.GetBytes(x), 0);
|
|
if (result < 0) {
|
|
result &= 0x7FFFFFFF;
|
|
result = -result;
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
public static float sdot(float a, float b, float c, float d) {
|
|
return a * b + c * d;
|
|
}
|
|
|
|
public static float sdot(float a, float b, float c, float d, float e, float f) {
|
|
return a * b + c * d + e * f;
|
|
}
|
|
|
|
public static float scross(float a, float b, float c, float d) {
|
|
return a * b - c * d;
|
|
}
|
|
|
|
public static double dcross(double a, double b, double c, double d) {
|
|
return a * b - c * d;
|
|
}
|
|
|
|
public static float scross_dscale(float a, float b,
|
|
float c, float d, double scale) {
|
|
return (float) (scross(a, b, c, d) * scale);
|
|
}
|
|
|
|
public static float dcross_dscale(double a, double b,
|
|
double c, double d, double scale) {
|
|
return (float) (dcross(a, b, c, d) * scale);
|
|
}
|
|
|
|
public static bool is_degenerate_2x2(
|
|
float scaleX, float skewX,
|
|
float skewY, float scaleY) {
|
|
float perp_dot = scaleX * scaleY - skewX * skewY;
|
|
return ScalarNearlyZero(perp_dot,
|
|
kScalarNearlyZero * kScalarNearlyZero);
|
|
}
|
|
|
|
public static double inv_determinant(float[] mat, int isPerspective) {
|
|
double det;
|
|
|
|
if (isPerspective != 0) {
|
|
det = mat[Matrix3.kMScaleX] *
|
|
dcross(mat[Matrix3.kMScaleY], mat[Matrix3.kMPersp2],
|
|
mat[Matrix3.kMTransY], mat[Matrix3.kMPersp1])
|
|
+
|
|
mat[Matrix3.kMSkewX] *
|
|
dcross(mat[Matrix3.kMTransY], mat[Matrix3.kMPersp0],
|
|
mat[Matrix3.kMSkewY], mat[Matrix3.kMPersp2])
|
|
+
|
|
mat[Matrix3.kMTransX] *
|
|
dcross(mat[Matrix3.kMSkewY], mat[Matrix3.kMPersp1],
|
|
mat[Matrix3.kMScaleY], mat[Matrix3.kMPersp0]);
|
|
} else {
|
|
det = dcross(mat[Matrix3.kMScaleX], mat[Matrix3.kMScaleY],
|
|
mat[Matrix3.kMSkewX], mat[Matrix3.kMSkewY]);
|
|
}
|
|
|
|
// Since the determinant is on the order of the cube of the matrix members,
|
|
// compare to the cube of the default nearly-zero constant (although an
|
|
// estimate of the condition number would be better if it wasn't so expensive).
|
|
if (ScalarNearlyZero((float) det,
|
|
kScalarNearlyZero * kScalarNearlyZero * kScalarNearlyZero)) {
|
|
return 0;
|
|
}
|
|
|
|
return 1.0 / det;
|
|
}
|
|
}
|
|
}
|