using BBIWARG.Utility; using Emgu.CV; using System; using System.Collections.Generic; using System.Drawing; namespace BBIWARG.Recognition.FingerRecognition { /// /// Each Finger consists of multiple FingerSlices, a FingerSliceTrail is a collection of these slices. /// public class FingerSliceTrail { /// /// the fitted direction through all slice's mid points [end to start] /// private Vector2D fittedDirection; /// /// indicates whether the variable fittedDirection is up to date (outdates if new slices are added) /// private bool fittedDirectionUpToDate; /// /// the line segment connection the first slice's mid to the last slice's mid /// private LineSegment2D lineSegment; /// /// indicates whether the variable lineSegment is up to date (outdates if new start or end slices are added) /// private bool lineSegmentUpToDate; /// /// the last slice /// public FingerSlice EndSlice { get { return Slices[Slices.Count - 1]; } } /// /// the fitted direction through all slice's mid points [end to start] /// public Vector2D FittedDirection { get { if (!fittedDirectionUpToDate) updateFittedDirection(); return fittedDirection; } } /// /// the line segment connecting the start slice's mid to the end slice's mid /// public LineSegment2D LineSegment { get { if (!lineSegmentUpToDate) updateLineSegment(); return lineSegment; } } /// /// the middle slice /// public FingerSlice MidSlice { get { return Slices[NumSlices / 2]; } } /// /// the number of slices /// public int NumSlices { get { return Slices.Count; } } /// /// the finger slices /// public List Slices { get; private set; } /// /// the first slice /// public FingerSlice StartSlice { get { return Slices[0]; } } /// /// Initializes a new instance of the FingerSliceTrail class. /// /// The initial slice. public FingerSliceTrail(FingerSlice slice) { Slices = new List(); addSlice(slice); lineSegmentUpToDate = false; fittedDirectionUpToDate = false; } /// /// The slice at the given index. /// /// the index /// the slice at the given index public FingerSlice this[int index] { get { return Slices[index]; } } /// /// Adds a slice to the end of the slice trail and outdates the lineSegment and fittedDirection. /// /// the slice that should be added public void addSlice(FingerSlice slice) { Slices.Add(slice); lineSegmentUpToDate = false; fittedDirectionUpToDate = false; } /// /// Gets the contour of the finger with a given margin /// /// the margin around the finger (in pixels) /// the contour of the finger public Contour getContour(float margin) { List pointsA = new List(); List pointsB = new List(); foreach (FingerSlice slice in Slices) { Vector2D direction = slice.Direction; pointsA.Add(slice.Start + margin * direction.getInverse()); pointsB.Add(slice.End + margin * direction); } pointsA.Reverse(); pointsA.AddRange(pointsB); Contour contour = new Contour(new MemStorage()); contour.PushMulti(pointsA.ToArray(), Emgu.CV.CvEnum.BACK_OR_FRONT.FRONT); return contour; } /// /// Calculates the direction of the last few slices. /// /// the end direction (direction towards hand) public Vector2D getEndDirection() { int innerEndIndex = Math.Max(0, NumSlices - Parameters.FingerNumSlicesForRelativeDirection); return (EndSlice.Mid - Slices[innerEndIndex].Mid).normalize(); } /// /// Calculates the direction of the first few slices /// /// the start direction (pointing direction) public Vector2D getStartDirection() { int innerStartIndex = Math.Min(NumSlices - 1, Parameters.FingerNumSlicesForRelativeDirection); return (StartSlice.Mid - Slices[innerStartIndex].Mid).normalize(); } /// /// Removes the first few slices. /// /// the number of slices that should be removed public void removeFirstSlices(int numSlices) { Slices.RemoveRange(0, numSlices); lineSegmentUpToDate = false; fittedDirectionUpToDate = false; } /// /// Reverses the trail and updates the line segment and the fitted direction /// public void reverse() { Slices.Reverse(); if (lineSegmentUpToDate) lineSegment = new LineSegment2D(lineSegment.P2, lineSegment.P1); if (fittedDirectionUpToDate) fittedDirection = fittedDirection.getInverse(); } /// /// updates the fitted direction (line fitting through all slices mid points) /// private void updateFittedDirection() { List midPoints = new List(); foreach (FingerSlice slice in Slices) midPoints.Add(slice.Mid); PointF pointOnLine, direction; PointCollection.Line2DFitting(midPoints.ToArray(), Emgu.CV.CvEnum.DIST_TYPE.CV_DIST_FAIR, out direction, out pointOnLine); fittedDirection = new Vector2D(direction).normalize(); if (fittedDirection.isInOppositeDirection(LineSegment.Direction)) fittedDirection = fittedDirection.getInverse(); fittedDirectionUpToDate = true; } /// /// updates the line segment (new line segment from end to start) /// private void updateLineSegment() { lineSegment = new LineSegment2D(EndSlice.Mid, StartSlice.Mid); lineSegmentUpToDate = true; } } }