Finger.cs 4.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158
  1. using System;
  2. using System.Collections.Generic;
  3. using System.Drawing;
  4. using System.Linq;
  5. using System.Text;
  6. using System.Threading.Tasks;
  7. using Emgu.CV;
  8. using Emgu.CV.Structure;
  9. using bbiwarg.Images;
  10. namespace bbiwarg.Detectors.Fingers
  11. {
  12. class Finger
  13. {
  14. private List<FingerPoint> fingerPoints;
  15. private bool lineUpToDate = false;
  16. private PointF direction;
  17. private PointF pointOnLine;
  18. private PointF lineEndPoint1;
  19. private PointF lineEndPoint2;
  20. private FingerPoint tipPoint;
  21. private FingerPoint handPoint;
  22. private float length;
  23. public Finger(FingerPoint fingerPoint)
  24. {
  25. fingerPoints = new List<FingerPoint>();
  26. addFingerPoint(fingerPoint);
  27. }
  28. public PointF getLineEndPoint1()
  29. {
  30. if (!lineUpToDate) updateLine();
  31. return lineEndPoint1;
  32. }
  33. public PointF getLineEndPoint2()
  34. {
  35. if (!lineUpToDate) updateLine();
  36. return lineEndPoint2;
  37. }
  38. public FingerPoint getTipPoint()
  39. {
  40. if (!lineUpToDate) updateLine();
  41. return tipPoint;
  42. }
  43. public PointF getDirection() {
  44. if (!lineUpToDate) updateLine();
  45. return direction;
  46. }
  47. public float getLength()
  48. {
  49. if (!lineUpToDate) updateLine();
  50. return length;
  51. }
  52. public void addFingerPoint(FingerPoint fingerPoint)
  53. {
  54. fingerPoints.Add(fingerPoint);
  55. lineUpToDate = false;
  56. }
  57. public float getMinDistance(FingerPoint fingerPoint)
  58. {
  59. float minDinstance = float.MaxValue;
  60. foreach (FingerPoint fp in fingerPoints)
  61. {
  62. float distance = fp.getDistanceTo(fingerPoint);
  63. if (distance < minDinstance)
  64. {
  65. minDinstance = distance;
  66. }
  67. }
  68. return minDinstance;
  69. }
  70. public bool isSimilarTo(Finger compareFinger) {
  71. PointF center = new PointF((lineEndPoint1.X + lineEndPoint2.X) / 2, (lineEndPoint1.Y + lineEndPoint2.Y) / 2);
  72. PointF compareCenter = new PointF((compareFinger.getLineEndPoint1().X+compareFinger.getLineEndPoint2().X)/2, (compareFinger.getLineEndPoint1().Y+compareFinger.getLineEndPoint2().Y)/2);
  73. float xDiff = center.X - compareCenter.X;
  74. float yDiff = center.Y - compareCenter.Y;
  75. float centerDistance = (float)Math.Sqrt(xDiff * xDiff + yDiff * yDiff);
  76. float maxCenterDistance = 40;
  77. if (centerDistance > maxCenterDistance)
  78. return false;
  79. return true;
  80. }
  81. private void updateLine() {
  82. //update direction+pointonline
  83. PointF[] pointArray = new PointF[fingerPoints.Count];
  84. int i = 0;
  85. foreach (FingerPoint fp in fingerPoints)
  86. {
  87. pointArray[i] = new PointF(fp.getX(), fp.getY());
  88. ++i;
  89. }
  90. PointCollection.Line2DFitting(pointArray, Emgu.CV.CvEnum.DIST_TYPE.CV_DIST_L2, out direction, out pointOnLine);
  91. FingerPoint fp1 = fingerPoints[0];
  92. FingerPoint fp2 = fingerPoints[0];
  93. length = 0.0f;
  94. foreach (FingerPoint fp in fingerPoints) {
  95. float distanceToFP1 = fp.getDistanceTo(fp1);
  96. float distanceToFP2 = fp.getDistanceTo(fp2);
  97. if (length < distanceToFP1 && distanceToFP1 >= distanceToFP2)
  98. {
  99. fp2 = fp;
  100. length = distanceToFP1;
  101. }
  102. else if (length < distanceToFP2 && distanceToFP2 > distanceToFP1)
  103. {
  104. fp1 = fp;
  105. length = distanceToFP2;
  106. }
  107. }
  108. if (fp1.getY() < fp2.getY())
  109. {
  110. tipPoint = fp1;
  111. handPoint = fp2;
  112. }
  113. else
  114. {
  115. tipPoint = fp2;
  116. handPoint = fp1;
  117. }
  118. //update start+end
  119. lineEndPoint1 = projectToLine(new PointF(tipPoint.getX(), tipPoint.getY()));
  120. lineEndPoint2 = projectToLine(new PointF(handPoint.getX(), handPoint.getY()));
  121. lineUpToDate = true;
  122. }
  123. private PointF projectToLine(PointF p)
  124. {
  125. float px = p.X, py = p.Y, dx = direction.X, dy = direction.Y, ox = pointOnLine.X, oy = pointOnLine.Y;
  126. float diffx = px - ox;
  127. float diffy = py - oy;
  128. float diff_d = (diffx * dx + diffy * dy);
  129. float d_d = (dx * dx + dy * dy);
  130. float q = diff_d / d_d;
  131. return new PointF(ox + q * dx, oy + q * dy);
  132. }
  133. }
  134. }