using System;
using System.Collections.Generic;
using System.Text;
using Matrox.MatroxImagingLibrary;
namespace Mcal
{
class Program
{
private const int RUN_LINEAR_CALIBRATION_EXAMPLE = MIL.M_YES;
private const int RUN_TSAI_CALIBRATION_EXAMPLE = MIL.M_YES;
private const int RUN_PARTIAL_GRID_CALIBRATION_EXAMPLE = MIL.M_YES;
private const int GRID_OFFSET_X = 0;
private const int GRID_OFFSET_Y = 0;
private const int GRID_OFFSET_Z = 0;
static void Main(string[] args)
{
MIL_ID MilApplication = MIL.M_NULL;
MIL_ID MilSystem = MIL.M_NULL;
MIL_ID MilDisplay = MIL.M_NULL;
MIL.MappAllocDefault(MIL.M_DEFAULT, ref MilApplication, ref MilSystem, ref MilDisplay, MIL.M_NULL, MIL.M_NULL);
Console.WriteLine("CALIBRATION MODULE:");
Console.WriteLine("-------------------");
Console.WriteLine();
if (RUN_LINEAR_CALIBRATION_EXAMPLE == MIL.M_YES)
{
LinearInterpolationCalibration(MilSystem, MilDisplay);
}
if (RUN_TSAI_CALIBRATION_EXAMPLE == MIL.M_YES)
{
TsaiCalibration(MilSystem, MilDisplay);
}
if (RUN_PARTIAL_GRID_CALIBRATION_EXAMPLE == MIL.M_YES)
{
PartialGridCalibration(MilSystem, MilDisplay);
}
MIL.MappFreeDefault(MilApplication, MilSystem, MilDisplay, MIL.M_NULL, MIL.M_NULL);
}
private const string GRID_IMAGE_FILE = MIL.M_IMAGE_PATH + "CalGrid.mim";
private const string BOARD_IMAGE_FILE = MIL.M_IMAGE_PATH + "CalBoard.mim";
private const int GRID_ROW_SPACING = 1;
private const int GRID_COLUMN_SPACING = 1;
private const int GRID_ROW_NUMBER = 18;
private const int GRID_COLUMN_NUMBER = 25;
private const int MEAS_BOX_POS_X1 = 55;
private const int MEAS_BOX_POS_Y1 = 24;
private const int MEAS_BOX_WIDTH1 = 7;
private const int MEAS_BOX_HEIGHT1 = 425;
private const int MEAS_BOX_POS_X2 = 225;
private const int MEAS_BOX_POS_Y2 = 11;
private const int MEAS_BOX_WIDTH2 = 7;
private const int MEAS_BOX_HEIGHT2 = 450;
private const int WIDTH_APPROXIMATION = 410;
private const int WIDTH_VARIATION = 25;
private const int MIN_EDGE_VALUE = 5;
static void LinearInterpolationCalibration(MIL_ID MilSystem, MIL_ID MilDisplay)
{
MIL_ID MilImage = MIL.M_NULL;
MIL_ID MilOverlayImage = MIL.M_NULL;
MIL_ID MilCalibration = MIL.M_NULL;
MIL_ID MeasMarker1 = MIL.M_NULL;
MIL_ID MeasMarker2 = MIL.M_NULL;
double WorldDistance1 = 0.0;
double WorldDistance2 = 0.0;
double PixelDistance1 = 0.0;
double PixelDistance2 = 0.0;
double PosX1 = 0.0;
double PosY1 = 0.0;
double PosX2 = 0.0;
double PosY2 = 0.0;
double PosX3 = 0.0;
double PosY3 = 0.0;
double PosX4 = 0.0;
double PosY4 = 0.0;
MIL_INT CalibrationStatus = 0;
MIL.MdispControl(MilDisplay, MIL.M_OVERLAY_CLEAR, MIL.M_DEFAULT);
MIL.MbufRestore(GRID_IMAGE_FILE, MilSystem, ref MilImage);
MIL.MdispSelect(MilDisplay, MilImage);
MIL.MdispControl(MilDisplay, MIL.M_OVERLAY, MIL.M_ENABLE);
MIL.MdispInquire(MilDisplay, MIL.M_OVERLAY_ID, ref MilOverlayImage);
Console.WriteLine();
Console.WriteLine("LINEAR INTERPOLATION CALIBRATION:");
Console.WriteLine("---------------------------------");
Console.WriteLine();
Console.WriteLine("The displayed grid has been grabbed with a high distortion");
Console.WriteLine("camera and will be used to calibrate the camera.");
Console.WriteLine("Press <Enter> to continue.");
Console.WriteLine();
Console.ReadKey();
MIL.McalAlloc(MilSystem, MIL.M_DEFAULT, MIL.M_DEFAULT, ref MilCalibration);
MIL.McalGrid(MilCalibration, MilImage, GRID_OFFSET_X, GRID_OFFSET_Y, GRID_OFFSET_Z, GRID_ROW_NUMBER, GRID_COLUMN_NUMBER, GRID_ROW_SPACING, GRID_COLUMN_SPACING, MIL.M_DEFAULT, MIL.M_DEFAULT);
MIL.McalInquire(MilCalibration, MIL.M_CALIBRATION_STATUS + MIL.M_TYPE_MIL_INT, ref CalibrationStatus);
if (CalibrationStatus == MIL.M_CALIBRATED)
{
MIL.McalTransformImage(MilImage, MilImage, MilCalibration, MIL.M_BILINEAR | MIL.M_OVERSCAN_CLEAR, MIL.M_DEFAULT, MIL.M_DEFAULT);
Console.WriteLine("The camera has been calibrated and the image of the grid");
Console.WriteLine("has been transformed to remove its distortions.");
Console.WriteLine("Press <Enter> to continue.");
Console.WriteLine();
Console.ReadKey();
MIL.MbufLoad(BOARD_IMAGE_FILE, MilImage);
MIL.McalAssociate(MilCalibration, MilImage, MIL.M_DEFAULT);
MIL.MmeasAllocMarker(MilSystem, MIL.M_STRIPE, MIL.M_DEFAULT, ref MeasMarker1);
MIL.MmeasAllocMarker(MilSystem, MIL.M_STRIPE, MIL.M_DEFAULT, ref MeasMarker2);
MIL.MmeasSetMarker(MeasMarker1, MIL.M_BOX_ORIGIN, MEAS_BOX_POS_X1, MEAS_BOX_POS_Y1);
MIL.MmeasSetMarker(MeasMarker1, MIL.M_BOX_SIZE, MEAS_BOX_WIDTH1, MEAS_BOX_HEIGHT1);
MIL.MmeasSetMarker(MeasMarker2, MIL.M_BOX_ORIGIN, MEAS_BOX_POS_X2, MEAS_BOX_POS_Y2);
MIL.MmeasSetMarker(MeasMarker2, MIL.M_BOX_SIZE, MEAS_BOX_WIDTH2, MEAS_BOX_HEIGHT2);
MIL.MmeasSetMarker(MeasMarker1, MIL.M_ORIENTATION, MIL.M_HORIZONTAL, MIL.M_NULL);
MIL.MmeasSetMarker(MeasMarker2, MIL.M_ORIENTATION, MIL.M_HORIZONTAL, MIL.M_NULL);
MIL.MmeasSetMarker(MeasMarker1, MIL.M_EDGEVALUE_MIN, MIN_EDGE_VALUE, MIL.M_NULL);
MIL.MmeasSetScore(MeasMarker1, MIL.M_STRENGTH_SCORE,
0.0,
0.0,
MIL.M_MAX_POSSIBLE_VALUE,
MIL.M_MAX_POSSIBLE_VALUE,
MIL.M_DEFAULT,
MIL.M_DEFAULT,
MIL.M_DEFAULT);
MIL.MmeasSetScore(MeasMarker1, MIL.M_STRIPE_WIDTH_SCORE,
WIDTH_APPROXIMATION - WIDTH_VARIATION,
WIDTH_APPROXIMATION + WIDTH_VARIATION,
WIDTH_APPROXIMATION + WIDTH_VARIATION,
WIDTH_APPROXIMATION + WIDTH_VARIATION,
MIL.M_DEFAULT,
MIL.M_PIXEL,
MIL.M_DEFAULT);
MIL.MmeasSetMarker(MeasMarker2, MIL.M_EDGEVALUE_MIN, MIN_EDGE_VALUE, MIL.M_NULL);
MIL.MmeasSetScore(MeasMarker2, MIL.M_STRENGTH_SCORE,
0.0,
0.0,
MIL.M_MAX_POSSIBLE_VALUE,
MIL.M_MAX_POSSIBLE_VALUE,
MIL.M_DEFAULT,
MIL.M_DEFAULT,
MIL.M_DEFAULT);
MIL.MmeasSetScore(MeasMarker2, MIL.M_STRIPE_WIDTH_SCORE,
WIDTH_APPROXIMATION - WIDTH_VARIATION,
WIDTH_APPROXIMATION + WIDTH_VARIATION,
WIDTH_APPROXIMATION + WIDTH_VARIATION,
WIDTH_APPROXIMATION + WIDTH_VARIATION,
MIL.M_DEFAULT,
MIL.M_PIXEL,
MIL.M_DEFAULT);
MIL.MmeasFindMarker(MIL.M_DEFAULT, MilImage, MeasMarker1, MIL.M_STRIPE_WIDTH + MIL.M_POSITION);
MIL.MmeasFindMarker(MIL.M_DEFAULT, MilImage, MeasMarker2, MIL.M_STRIPE_WIDTH + MIL.M_POSITION);
MIL.MmeasGetResult(MeasMarker1, MIL.M_STRIPE_WIDTH, ref WorldDistance1);
MIL.MmeasGetResult(MeasMarker2, MIL.M_STRIPE_WIDTH, ref WorldDistance2);
MIL.MmeasSetMarker(MeasMarker1, MIL.M_RESULT_OUTPUT_UNITS, MIL.M_PIXEL, MIL.M_NULL);
MIL.MmeasSetMarker(MeasMarker2, MIL.M_RESULT_OUTPUT_UNITS, MIL.M_PIXEL, MIL.M_NULL);
MIL.MmeasGetResult(MeasMarker1, MIL.M_STRIPE_WIDTH, ref PixelDistance1);
MIL.MmeasGetResult(MeasMarker2, MIL.M_STRIPE_WIDTH, ref PixelDistance2);
MIL.MmeasGetResult(MeasMarker1, MIL.M_POSITION + MIL.M_EDGE_FIRST, ref PosX1, ref PosY1);
MIL.MmeasGetResult(MeasMarker1, MIL.M_POSITION + MIL.M_EDGE_SECOND, ref PosX2, ref PosY2);
MIL.MmeasGetResult(MeasMarker2, MIL.M_POSITION + MIL.M_EDGE_FIRST, ref PosX3, ref PosY3);
MIL.MmeasGetResult(MeasMarker2, MIL.M_POSITION + MIL.M_EDGE_SECOND, ref PosX4, ref PosY4);
MIL.MgraColor(MIL.M_DEFAULT, MIL.M_COLOR_YELLOW);
MIL.MmeasDraw(MIL.M_DEFAULT, MeasMarker1, MilOverlayImage, MIL.M_DRAW_WIDTH, MIL.M_DEFAULT, MIL.M_RESULT);
MIL.MmeasDraw(MIL.M_DEFAULT, MeasMarker2, MilOverlayImage, MIL.M_DRAW_WIDTH, MIL.M_DEFAULT, MIL.M_RESULT);
MIL.MgraBackColor(MIL.M_DEFAULT, MIL.M_COLOR_BLACK);
MIL.MgraText(MIL.M_DEFAULT, MilOverlayImage, (int)(PosX1 + 0.5 - 40), (int)((PosY1 + 0.5) + ((PosY2 - PosY1) / 2.0)), " Distance 1 ");
MIL.MgraText(MIL.M_DEFAULT, MilOverlayImage, (int)(PosX3 + 0.5 - 40), (int)((PosY3 + 0.5) + ((PosY4 - PosY3) / 2.0)), " Distance 2 ");
Console.WriteLine("A distorted image grabbed with the same camera was loaded and");
Console.WriteLine("calibrated measurements were done to evaluate the board dimensions.");
Console.WriteLine();
Console.WriteLine("========================================================");
Console.WriteLine(" Distance 1 Distance 2 ");
Console.WriteLine("--------------------------------------------------------");
Console.WriteLine(" Calibrated unit: {0,8:0.00} cm {1,6:0.00} cm ", WorldDistance1, WorldDistance2);
Console.WriteLine(" Uncalibrated unit: {0,8:0.00} pixels {1,6:0.00} pixels", PixelDistance1, PixelDistance2);
Console.WriteLine("========================================================");
Console.WriteLine();
Console.WriteLine("Press <Enter> to continue.");
Console.WriteLine();
Console.ReadKey();
MIL.MdispControl(MilDisplay, MIL.M_OVERLAY_CLEAR, MIL.M_DEFAULT);
MIL.MbufLoad(BOARD_IMAGE_FILE, MilImage);
MIL.McalTransformImage(MilImage, MilImage, MilCalibration, MIL.M_BILINEAR + MIL.M_OVERSCAN_CLEAR, MIL.M_DEFAULT, MIL.M_DEFAULT);
Console.WriteLine("The image was corrected to remove its distortions.");
MIL.MmeasFree(MeasMarker1);
MIL.MmeasFree(MeasMarker2);
}
else
{
Console.WriteLine("Calibration generated an exception.");
Console.WriteLine("See User Guide to resolve the situation.");
Console.WriteLine();
}
Console.WriteLine("Press <Enter> to continue.");
Console.WriteLine();
Console.ReadKey();
MIL.McalFree(MilCalibration);
MIL.MbufFree(MilImage);
}
private const string GRID_ORIGINAL_IMAGE_FILE = MIL.M_IMAGE_PATH + "CalGridOriginal.mim";
private const string OBJECT_ORIGINAL_IMAGE_FILE = MIL.M_IMAGE_PATH + "CalObjOriginal.mim";
private const string OBJECT_MOVED_IMAGE_FILE = MIL.M_IMAGE_PATH + "CalObjMoved.mim";
private const double GRID_ORG_ROW_SPACING = 1.5;
private const double GRID_ORG_COLUMN_SPACING = 1.5;
private const int GRID_ORG_ROW_NUMBER = 12;
private const int GRID_ORG_COLUMN_NUMBER = 13;
private const int GRID_ORG_OFFSET_X = 0;
private const int GRID_ORG_OFFSET_Y = 0;
private const int GRID_ORG_OFFSET_Z = 0;
private const double ENCODER_TRANSLATION_X = 0.817;
private const double ENCODER_TRANSLATION_Y = 13.293;
private const double ENCODER_TRANSLATION_Z = -14.060;
private const double ENCODER_ROTATION_Y = 354.207;
private const double ENCODER_ROTATION_X = 317.467;
private const double ENCODER_ROTATION_Z = 281.969;
private const int MEASURED_CIRCLE_LABEL = 1;
private const double RING1_POS1_X = 2.3;
private const double RING1_POS1_Y = 3.9;
private const double RING2_POS1_X = 10.7;
private const double RING2_POS1_Y = 11.1;
private const double RING1_POS2_X = 8.6;
private const double RING1_POS2_Y = 2.5;
private const double RING2_POS2_X = 7.9;
private const double RING2_POS2_Y = 13.2;
private const double RING_START_RADIUS = 1.25;
private const double RING_END_RADIUS = 2.3;
private const double RING_THICKNESS = 0.175;
private const double STEP_THICKNESS = 4.0;
static readonly int REGION_COLOR = MIL.M_RGB888(0, 100, 255);
static readonly int FEATURE_COLOR = MIL.M_RGB888(255, 0, 255);
static void TsaiCalibration(MIL_ID MilSystem, MIL_ID MilDisplay)
{
MIL_ID MilImage = MIL.M_NULL;
MIL_ID MilCalibration = MIL.M_NULL;
MIL_INT CalibrationStatus = 0;
MIL.MbufRestore(GRID_ORIGINAL_IMAGE_FILE, MilSystem, ref MilImage);
MIL.MdispSelect(MilDisplay, MilImage);
Console.WriteLine();
Console.WriteLine("TSAI BASED CALIBRATION:");
Console.WriteLine("-----------------------");
Console.WriteLine();
Console.WriteLine("The displayed grid has been grabbed with a high perspective");
Console.WriteLine("camera and will be used to calibrate the camera.");
Console.WriteLine("Press <Enter> to continue.");
Console.WriteLine();
Console.ReadKey();
MIL.McalAlloc(MilSystem, MIL.M_TSAI_BASED, MIL.M_DEFAULT, ref MilCalibration);
MIL.McalGrid(MilCalibration, MilImage, GRID_ORG_OFFSET_X, GRID_ORG_OFFSET_Y, GRID_ORG_OFFSET_Z, GRID_ORG_ROW_NUMBER, GRID_ORG_COLUMN_NUMBER, GRID_ORG_ROW_SPACING, GRID_ORG_COLUMN_SPACING, MIL.M_DEFAULT, MIL.M_DEFAULT);
MIL.McalInquire(MilCalibration, MIL.M_CALIBRATION_STATUS + MIL.M_TYPE_MIL_INT, ref CalibrationStatus);
if (CalibrationStatus == MIL.M_CALIBRATED)
{
Console.WriteLine("The camera has been calibrated.");
Console.WriteLine();
ShowCameraInformation(MilCalibration);
MIL.MbufLoad(OBJECT_ORIGINAL_IMAGE_FILE, MilImage);
SetCalibrationOffset(MilCalibration, -RING_THICKNESS, MilImage);
Console.Write("First measured circle at z = {0:f3} cm. ", -RING_THICKNESS);
MeasureRing(MilSystem, MilDisplay, MilImage, RING1_POS1_X, RING1_POS1_Y);
SetCalibrationOffset(MilCalibration, -(STEP_THICKNESS + RING_THICKNESS), MilImage);
Console.Write("Second measured circle at z = {0:f3} cm. ", -(STEP_THICKNESS + RING_THICKNESS));
MeasureRing(MilSystem, MilDisplay, MilImage, RING2_POS1_X, RING2_POS1_Y);
Console.WriteLine("Press <Enter> to continue.");
Console.WriteLine();
Console.ReadKey();
MIL.MdispControl(MilDisplay, MIL.M_OVERLAY_CLEAR, MIL.M_DEFAULT);
MIL.McalSetCoordinateSystem(MilCalibration,
MIL.M_TOOL_COORDINATE_SYSTEM,
MIL.M_ABSOLUTE_COORDINATE_SYSTEM,
MIL.M_ROTATION_YXZ | MIL.M_ASSIGN,
MIL.M_NULL,
ENCODER_ROTATION_Y,
ENCODER_ROTATION_X,
ENCODER_ROTATION_Z,
MIL.M_DEFAULT);
MIL.McalSetCoordinateSystem(MilCalibration,
MIL.M_TOOL_COORDINATE_SYSTEM,
MIL.M_ABSOLUTE_COORDINATE_SYSTEM,
MIL.M_TRANSLATION | MIL.M_COMPOSE_WITH_CURRENT,
MIL.M_NULL,
ENCODER_TRANSLATION_X,
ENCODER_TRANSLATION_Y,
ENCODER_TRANSLATION_Z,
MIL.M_DEFAULT);
MIL.MbufLoad(OBJECT_MOVED_IMAGE_FILE, MilImage);
Console.WriteLine("Calibration module moved camera position and camera ");
Console.WriteLine("orientation according to known encoder transformation.");
Console.WriteLine();
ShowCameraInformation(MilCalibration);
SetCalibrationOffset(MilCalibration, -RING_THICKNESS, MilImage);
Console.Write("First measured circle at z = {0:f3} cm. ", -RING_THICKNESS);
MeasureRing(MilSystem, MilDisplay, MilImage, RING1_POS2_X, RING1_POS2_Y);
SetCalibrationOffset(MilCalibration, -(STEP_THICKNESS + RING_THICKNESS), MilImage);
Console.Write("Second measured circle at z = {0:f3} cm. ", -(STEP_THICKNESS + RING_THICKNESS));
MeasureRing(MilSystem, MilDisplay, MilImage, RING2_POS2_X, RING2_POS2_Y);
Console.WriteLine("Press <Enter> to continue.");
Console.WriteLine();
Console.ReadKey();
}
else
{
Console.WriteLine("Calibration generated an exception.");
Console.WriteLine("See User Guide to resolve the situation.");
Console.WriteLine();
}
MIL.McalFree(MilCalibration);
MIL.MbufFree(MilImage);
}
static void SetCalibrationOffset(MIL_ID MilCalibration, double ZOffset, MIL_ID MilImage)
{
MIL.McalSetCoordinateSystem(MilCalibration, MIL.M_RELATIVE_COORDINATE_SYSTEM, MIL.M_ABSOLUTE_COORDINATE_SYSTEM, MIL.M_TRANSLATION | MIL.M_ASSIGN, MIL.M_NULL, 0, 0, ZOffset, MIL.M_DEFAULT);
MIL.McalAssociate(MilCalibration, MilImage, MIL.M_DEFAULT);
}
static void MeasureRing(MIL_ID MilSystem, MIL_ID MilDisplay, MIL_ID MilImage, double MeasureRingX, double MeasureRingY)
{
MIL_ID MilMetrolContext = MIL.M_NULL;
MIL_ID MilMetrolResult = MIL.M_NULL;
MIL_ID MilOverlayImage = MIL.M_NULL;
double Value = 0.0;
MIL.MdispControl(MilDisplay, MIL.M_OVERLAY, MIL.M_ENABLE);
MIL.MdispInquire(MilDisplay, MIL.M_OVERLAY_ID, ref MilOverlayImage);
MIL.MmetAlloc(MilSystem, MIL.M_DEFAULT, ref MilMetrolContext);
MIL.MmetAllocResult(MilSystem, MIL.M_DEFAULT, ref MilMetrolResult);
MIL.MmetAddFeature(MilMetrolContext, MIL.M_MEASURED, MIL.M_CIRCLE, MEASURED_CIRCLE_LABEL, MIL.M_DEFAULT, MIL.M_NULL, MIL.M_NULL, 0, MIL.M_DEFAULT);
MIL.MmetSetRegion(MilMetrolContext, MIL.M_FEATURE_LABEL(MEASURED_CIRCLE_LABEL), MIL.M_DEFAULT, MIL.M_RING, MeasureRingX, MeasureRingY, RING_START_RADIUS, RING_END_RADIUS, MIL.M_NULL, MIL.M_NULL);
MIL.MmetCalculate(MilMetrolContext, MilImage, MilMetrolResult, MIL.M_DEFAULT);
MIL.MgraColor(MIL.M_DEFAULT, REGION_COLOR);
MIL.MmetDraw(MIL.M_DEFAULT, MilMetrolResult, MilOverlayImage, MIL.M_DRAW_REGION, MIL.M_DEFAULT, MIL.M_DEFAULT);
MIL.MgraColor(MIL.M_DEFAULT, FEATURE_COLOR);
MIL.MmetDraw(MIL.M_DEFAULT, MilMetrolResult, MilOverlayImage, MIL.M_DRAW_FEATURE, MIL.M_DEFAULT, MIL.M_DEFAULT);
MIL.MmetGetResult(MilMetrolResult, MIL.M_FEATURE_LABEL(MEASURED_CIRCLE_LABEL), MIL.M_RADIUS, ref Value);
Console.WriteLine("Measured Radius: {0:0.000} cm", Value);
MIL.MmetFree(MilMetrolResult);
MIL.MmetFree(MilMetrolContext);
}
static void ShowCameraInformation(MIL_ID MilCalibration)
{
double CameraPosX = 0.0;
double CameraPosY = 0.0;
double CameraPosZ = 0.0;
double CameraYaw = 0.0;
double CameraPitch = 0.0;
double CameraRoll = 0.0;
MIL.McalGetCoordinateSystem(MilCalibration, MIL.M_CAMERA_COORDINATE_SYSTEM, MIL.M_ABSOLUTE_COORDINATE_SYSTEM, MIL.M_TRANSLATION, MIL.M_NULL, ref CameraPosX, ref CameraPosY, ref CameraPosZ, MIL.M_NULL);
MIL.McalGetCoordinateSystem(MilCalibration, MIL.M_CAMERA_COORDINATE_SYSTEM, MIL.M_ABSOLUTE_COORDINATE_SYSTEM, MIL.M_ROTATION_YXZ, MIL.M_NULL, ref CameraYaw, ref CameraPitch, ref CameraRoll, MIL.M_NULL);
Console.WriteLine("Camera Position in cm: (x, y, z) ({0:0.00}, {1:0.00}, {2:0.00})", CameraPosX, CameraPosY, CameraPosZ);
Console.WriteLine("Camera Orientation in degrees: (yaw, pitch, roll) ({0:0.00}, {1:0.00}, {2:0.00})", CameraYaw, CameraPitch, CameraRoll);
Console.WriteLine("Press <Enter> to continue.");
Console.WriteLine();
Console.ReadKey();
}
private const string PARTIAL_GRID_IMAGE_FILE = MIL.M_IMAGE_PATH + "PartialGrid.mim";
private const double CORRECTED_SIZE_X = 60.0;
private const double CORRECTED_SIZE_Y = 50.0;
private const double CORRECTED_OFFSET_X = -35.0;
private const double CORRECTED_OFFSET_Y = -5.0;
private const int CORRECTED_IMAGE_SIZE_X = 512;
static void PartialGridCalibration(MIL_ID MilSystem, MIL_ID MilDisplay)
{
MIL_ID MilImage = MIL.M_NULL;
MIL_ID MilCorrectedImage = MIL.M_NULL;
MIL_ID MilGraList = MIL.M_NULL;
MIL_ID MilCalibration = MIL.M_NULL;
MIL_INT CalibrationStatus = 0;
MIL_INT ImageType = 0;
MIL_INT CorrectedImageSizeY = 0;
double RowSpacing = 0.0;
double ColumnSpacing = 0.0;
double CorrectedPixelSize = 0.0;
StringBuilder UnitName = new StringBuilder();
MIL.MdispControl(MilDisplay, MIL.M_OVERLAY_CLEAR, MIL.M_DEFAULT);
MIL.MgraAllocList(MilSystem, MIL.M_DEFAULT, ref MilGraList);
MIL.MdispControl(MilDisplay, MIL.M_ASSOCIATED_GRAPHIC_LIST_ID, MilGraList);
MIL.MbufRestore(PARTIAL_GRID_IMAGE_FILE, MilSystem, ref MilImage);
MIL.MbufInquire(MilImage, MIL.M_TYPE, ref ImageType);
MIL.MdispSelect(MilDisplay, MilImage);
Console.WriteLine();
Console.WriteLine("PARTIAL GRID CALIBRATION:");
Console.WriteLine("-------------------------");
Console.WriteLine();
Console.WriteLine("A camera will be calibrated using a rectangular grid that");
Console.WriteLine("is only partially visible in the camera's field of view.");
Console.WriteLine("The 2D code in the center is used as a fiducial to retrieve");
Console.WriteLine("the characteristics of the calibration grid.");
Console.WriteLine("Press <Enter> to continue.");
Console.ReadKey();
MIL.McalAlloc(MilSystem, MIL.M_TSAI_BASED, MIL.M_DEFAULT, ref MilCalibration);
MIL.McalControl(MilCalibration, MIL.M_GRID_PARTIAL, MIL.M_ENABLE);
MIL.McalControl(MilCalibration, MIL.M_GRID_FIDUCIAL, MIL.M_DATAMATRIX);
MIL.McalGrid(MilCalibration, MilImage,
GRID_OFFSET_X, GRID_OFFSET_Y, GRID_OFFSET_Z,
MIL.M_UNKNOWN, MIL.M_UNKNOWN, MIL.M_FROM_FIDUCIAL, MIL.M_FROM_FIDUCIAL,
MIL.M_DEFAULT, MIL.M_CHESSBOARD_GRID);
MIL.McalInquire(MilCalibration, MIL.M_CALIBRATION_STATUS + MIL.M_TYPE_MIL_INT, ref CalibrationStatus);
if (CalibrationStatus == MIL.M_CALIBRATED)
{
MIL.MgraColor(MIL.M_DEFAULT, MIL.M_COLOR_RED);
MIL.McalDraw(MIL.M_DEFAULT, MilCalibration, MilGraList, MIL.M_DRAW_ABSOLUTE_COORDINATE_SYSTEM,
MIL.M_DEFAULT, MIL.M_DEFAULT);
MIL.MgraColor(MIL.M_DEFAULT, MIL.M_COLOR_CYAN);
MIL.McalDraw(MIL.M_DEFAULT, MilCalibration, MilGraList, MIL.M_DRAW_FIDUCIAL_BOX,
MIL.M_DEFAULT, MIL.M_DEFAULT);
MIL.McalInquire(MilCalibration, MIL.M_ROW_SPACING, ref RowSpacing);
MIL.McalInquire(MilCalibration, MIL.M_COLUMN_SPACING, ref ColumnSpacing);
MIL.McalInquire(MilCalibration, MIL.M_GRID_UNIT_SHORT_NAME, UnitName);
MIL.MgraColor(MIL.M_DEFAULT, MIL.M_COLOR_RED);
MIL.MgraControl(MIL.M_DEFAULT, MIL.M_INPUT_UNITS, MIL.M_DISPLAY);
DrawGridInfo(MilGraList, "Row spacing", RowSpacing, 0, UnitName.ToString());
DrawGridInfo(MilGraList, "Col spacing", ColumnSpacing, 1, UnitName.ToString());
Console.WriteLine();
Console.WriteLine("The camera has been calibrated.");
Console.WriteLine();
Console.WriteLine("The grid information read is displayed.");
Console.WriteLine("Press <Enter> to continue.");
Console.WriteLine();
Console.ReadKey();
CorrectedPixelSize = CORRECTED_SIZE_X / CORRECTED_IMAGE_SIZE_X;
CorrectedImageSizeY = (MIL_INT)(CORRECTED_SIZE_Y / CorrectedPixelSize);
MIL.MbufAlloc2d(MilSystem, CORRECTED_IMAGE_SIZE_X, CorrectedImageSizeY, ImageType,
MIL.M_IMAGE + MIL.M_PROC + MIL.M_DISP, ref MilCorrectedImage);
MIL.McalUniform(MilCorrectedImage, CORRECTED_OFFSET_X, CORRECTED_OFFSET_Y,
CorrectedPixelSize, CorrectedPixelSize, 0.0, MIL.M_DEFAULT);
MIL.McalTransformImage(MilImage, MilCorrectedImage, MilCalibration,
MIL.M_BILINEAR + MIL.M_OVERSCAN_CLEAR, MIL.M_DEFAULT,
MIL.M_WARP_IMAGE + MIL.M_USE_DESTINATION_CALIBRATION);
MIL.MgraClear(MIL.M_DEFAULT, MilGraList);
MIL.MdispSelect(MilDisplay, MilCorrectedImage);
Console.WriteLine("A sub-region of the grid was selected and transformed");
Console.WriteLine("to remove the distortions.");
Console.WriteLine("The sub-region dimensions and position are:");
Console.WriteLine(" Size X : {0,3:g3} {1}", CORRECTED_SIZE_X, UnitName);
Console.WriteLine(" Size Y : {0,3:g3} {1}", CORRECTED_SIZE_Y, UnitName);
Console.WriteLine(" Offset X: {0,3:g3} {1}", CORRECTED_OFFSET_X, UnitName);
Console.WriteLine(" Offset Y: {0,3:g3} {1}", CORRECTED_OFFSET_Y, UnitName);
Console.WriteLine("Press <Enter> to quit.");
Console.WriteLine();
Console.ReadKey();
MIL.MbufFree(MilCorrectedImage);
}
else
{
Console.WriteLine("Calibration generated an exception.");
Console.WriteLine("See User Guide to resolve the situation.");
Console.WriteLine();
Console.WriteLine("Press <Enter> to quit.");
Console.WriteLine();
Console.ReadKey();
}
MIL.McalFree(MilCalibration);
MIL.MbufFree(MilImage);
MIL.MgraFree(MilGraList);
}
private const int LINE_HEIGHT = 16;
static void DrawGridInfo(MIL_ID MilGraList, string InfoTag, double Value, MIL_INT LineOffsetY, string Units)
{
string Info = string.Format("{0}: {1:g3} {2}", InfoTag, Value, Units);
MIL.MgraText(MIL.M_DEFAULT, MilGraList, 0, LineOffsetY * LINE_HEIGHT, Info);
}
}
}