using BBIWARG.Utility; using Emgu.CV; using Emgu.CV.Structure; using System; using System.Drawing; namespace BBIWARG.Images { /// /// DepthImage stores a processed version of the depth image read by the camera as an /// public class DepthImage { /// /// the processed depth image /// public Image Image { get; private set; } /// /// the maximum depth which is considered important /// public Int16 MaxDepth { get; private set; } /// /// the minimum depth in the raw depth image /// public Int16 MinDepth { get; private set; } /// /// image size of the depth image /// public ImageSize Size { get; private set; } /// /// Construct a DepthImage from the raw depth data, the image size and a confidenceImage. /// Filters the image using confidenceImage and also applies thresholding and smoothing. /// /// the raw depth data /// the image size /// the confidence image public DepthImage(IntPtr rawDepthData, ImageSize size, ConfidenceImage confidenceImage) { Size = size; Image rawDepthImage = new Image(Size.Width, Size.Height, Size.Width * 2, rawDepthData); // filter with confidenceImage mask rawDepthImage = rawDepthImage.Or((1 - confidenceImage.Mask).Convert().Mul(Int16.MaxValue)); // smooth with median filter rawDepthImage = rawDepthImage.SmoothMedian(Parameters.DepthImageMedianSize); // threshold min&maxDepth MinDepth = findMinDepth(rawDepthImage); MaxDepth = (Int16)(MinDepth + Parameters.DepthImageDepthRange); // threshold (dst = (src > (MaxDepth - MinDepth)) ? MaxDepth - MinDepth : src) Image = (rawDepthImage - MinDepth).ThresholdTrunc(new Gray(MaxDepth - MinDepth)).Convert(); // smooth with median filter Image = Image.SmoothMedian(Parameters.DepthImageMedianSize); } /// /// Returns the depth in the processed image at a given point. /// /// the point /// depth at the point public Int16 getDepthAt(Point point) { return getDepthAt(point.X, point.Y); } /// /// Returns the depth in the processed image at a given position. /// /// x coordinate of the position /// y coordinate of the position /// the depth at the position public Int16 getDepthAt(int x, int y) { try { return (Int16)(MinDepth + Image.Data[y, x, 0]); } catch (IndexOutOfRangeException e) { return (Int16)(MinDepth + Image.Data[0, 0, 0]); } } /// /// Returns the depth in the processed image at a given position. /// It will move the x,y more to the center of the palm! /// /// x coordinate of the position /// y coordinate of the position /// the depth at the position public Int16 getDepthAtFixed(int x0, int y0, int x1, int y1) { try { int rx = (int) ((x1 - x0) * 0.15f + x0); int ry = (int) ((y1 - y0) * 0.15f + y0); return (Int16)(MinDepth + Image.Data[ry, rx, 0]); } catch (IndexOutOfRangeException e) { return (Int16)(MinDepth + Image.Data[0, 0, 0]); } } /// /// Sets the depth in the processed image. /// /// point where the depth is set /// new depth value public void setDepthAt(Point point, Int16 depth) { setDepthAt(point.X, point.Y, depth); } /// /// Sets the depth in the processed image. /// /// x coordinate of position to set depth /// y coordinate of position to set depth /// new depth value public void setDepthAt(int x, int y, Int16 depth) { Image.Data[y, x, 0] = (byte)(depth - MinDepth); } /// /// Returns the minimum depth in an /// /// the image /// the minimum depth private Int16 findMinDepth(Image image) { // min and max values double[] min, max; // min and max locations Point[] minLoc, maxLoc; image.MinMax(out min, out max, out minLoc, out maxLoc); return (Int16)min[0]; } } }