QuadrilateralInCamera.cs 5.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135
  1. using System;
  2. using System.Collections.Generic;
  3. using o0.Geometry2D;
  4. using o0.Geometry2D.Float;
  5. using UnityEngine;
  6. namespace ZIM.Unity
  7. {
  8. // 屏幕识别使用,记录摄像机画面内的四边形
  9. public class QuadrilateralInCamera
  10. {
  11. public Vector CameraSize;
  12. public OrdinalQuadrilateral Quad;
  13. // 固定的顶点顺序: 左下,右下,左上,右上
  14. public QuadrilateralInCamera(Vector2 a, Vector2 b, Vector2 c, Vector2 d, Vector2 cameraSize)
  15. {
  16. CameraSize = cameraSize.o0Vector();
  17. Quad = new OrdinalQuadrilateral(a.o0Vector(), b.o0Vector(), c.o0Vector(), d.o0Vector());
  18. }
  19. // 固定的顶点顺序: 左下,右下,左上,右上
  20. public QuadrilateralInCamera(Vector a, Vector b, Vector c, Vector d, Vector cameraSize)
  21. {
  22. CameraSize = cameraSize;
  23. Quad = new OrdinalQuadrilateral(a, b, c, d);
  24. }
  25. // 固定的顶点顺序: 左下,右下,左上,右上
  26. public QuadrilateralInCamera(IEnumerable<Vector> enumable, Vector cameraSize)
  27. {
  28. CameraSize = cameraSize;
  29. Quad = new OrdinalQuadrilateral(enumable);
  30. }
  31. public QuadrilateralInCamera(OrdinalQuadrilateral quad, Vector cameraSize)
  32. {
  33. CameraSize = cameraSize;
  34. Quad = quad;
  35. }
  36. public string SizeString => $"({CameraSize.x}, {CameraSize.y})";
  37. public string QuadString => $"[({Quad.A.x}, {Quad.A.y}), ({Quad.B.x}, {Quad.B.y}), ({Quad.C.x}, {Quad.C.y}), ({Quad.D.x}, {Quad.D.y})]";
  38. // 需要标准化的坐标(即数值范围0-1)时使用,会自动根据当前的CameraSize转换
  39. private List<Vector> ScreenVertexListNormalized
  40. {
  41. get
  42. {
  43. return new List<Vector> {
  44. Quad.A / CameraSize,
  45. Quad.B / CameraSize,
  46. Quad.C / CameraSize,
  47. Quad.D / CameraSize
  48. };
  49. }
  50. set
  51. {
  52. for (int i = 0; i < Quad.Count; i++)
  53. Quad[i] = value[i] * CameraSize;
  54. }
  55. }
  56. // 这里是标准化坐标(即数值范围0-1)
  57. public List<Vector2> GetUnityVertexList()
  58. {
  59. return new List<Vector2>() {
  60. new Vector2(Quad.A.x / CameraSize.x, Quad.A.y / CameraSize.y),
  61. new Vector2(Quad.B.x / CameraSize.x, Quad.B.y / CameraSize.y),
  62. new Vector2(Quad.C.x / CameraSize.x, Quad.C.y / CameraSize.y),
  63. new Vector2(Quad.D.x / CameraSize.x, Quad.D.y / CameraSize.y)
  64. };
  65. }
  66. // 摄像机分辨率变化时调用
  67. public void ReSize(Vector sizeNew, AspectRatioSetting viewAspectRatioSetting)
  68. {
  69. float ratio;
  70. Func<Vector, Vector> Translate;
  71. List<Vector> vertices;
  72. switch (viewAspectRatioSetting)
  73. {
  74. case AspectRatioSetting.FixedAll:
  75. var scale = new Vector(sizeNew.x / CameraSize.x, sizeNew.y / CameraSize.y);
  76. CameraSize = sizeNew;
  77. for (int i = 0; i < Quad.Count; i++)
  78. Quad[i] *= scale;
  79. break;
  80. case AspectRatioSetting.FixedHeight:
  81. var wOld = CameraSize.x / CameraSize.y; // 宽度比高度
  82. var wNew = sizeNew.x / sizeNew.y;
  83. ratio = wOld / wNew;
  84. Translate = (v) =>
  85. {
  86. var x = v.x - 0.5f;
  87. return new Vector(x * ratio + 0.5f, v.y);
  88. };
  89. vertices = ScreenVertexListNormalized;
  90. for (int i = 0; i < vertices.Count; i++)
  91. vertices[i] = Translate(vertices[i]);
  92. CameraSize = sizeNew; // 这里要先设置Size
  93. ScreenVertexListNormalized = vertices;
  94. break;
  95. case AspectRatioSetting.FixedWidth:
  96. var hOld = CameraSize.y / CameraSize.x; // 高度比宽度
  97. var hNew = sizeNew.y / sizeNew.x;
  98. ratio = hOld / hNew;
  99. Translate = (v) =>
  100. {
  101. var y = v.y - 0.5f;
  102. return new Vector(v.x, y * ratio + 0.5f);
  103. };
  104. vertices = ScreenVertexListNormalized;
  105. for (int i = 0; i < vertices.Count; i++)
  106. vertices[i] = Translate(vertices[i]);
  107. CameraSize = sizeNew;
  108. ScreenVertexListNormalized = vertices;
  109. break;
  110. }
  111. }
  112. // 四边形是否完整的在屏幕内
  113. public bool IsQuadComplete()
  114. {
  115. if (Quad.A.x < Quad.B.x && Quad.A.y < Quad.C.y && Quad.B.y < Quad.D.y && Quad.C.x < Quad.D.x && Quad.A.x > 0 && Quad.A.y > 0 &&
  116. Quad.B.x < CameraSize.x && Quad.B.y > 0 && Quad.C.x > 0 && Quad.C.y < CameraSize.y && Quad.D.x < CameraSize.x && Quad.D.y < CameraSize.y)
  117. return true;
  118. return false;
  119. }
  120. }
  121. }