Неверное преобразование оси Z маркера OpenCV - PullRequest
0 голосов
/ 13 сентября 2018

Я использую C #, .NET Core 2.1 и библиотеку OpenCvSharp в OSX.

Я пытаюсь красиво нарисовать оси маркеров на кадрах с камеры, но, к сожалению, моя ось Z сходит с ума.Похоже, независимо от того, сколько калибровок я делаю (даже пробовал 100), проецируемые точки всегда находятся в пределах (-1, -1, -1).

Есть идеи, что может быть причиной этого?Код ниже.

Insane Z axis

private static async Task DetectMarkers(string output)
{
    var rms = 0.0;
    var calib = 100;
    var size = new Size(9, 6);
    var frameSize = Size.Zero;
    var distortion = new Mat();
    var imgPoints = new List<MatOfPoint2f>();
    var objPoints = new List<MatOfPoint3f>();
    var criteria = new TermCriteria(CriteriaType.Eps | CriteriaType.MaxIter, 30, 0.001);
    var objp = MatOfPoint3f.FromArray(Create3DChessboardCorners(size, 0.025f));
    using (var capture = new VideoCapture(0))
    using (var paramters = DetectorParameters.Create())
    using (var camera = new MatOfDouble(Mat.Eye(3, 3, MatType.CV_64FC1)))
    using (var dictionary = CvAruco.GetPredefinedDictionary(PredefinedDictionaryName.Dict4X4_50))
    {
        while (capture.Grab() && calib > 0)
        {
            using (var image = capture.RetrieveMat())
            using (var gray = new Mat())
            {
                frameSize = image.Size();
                Cv2.CvtColor(image, gray, ColorConversionCodes.BGR2GRAY);
                if (Cv2.FindChessboardCorners(gray, size, out Point2f[] corners))
                {
                    objPoints.Add(objp);
                    imgPoints.Add(MatOfPoint2f.FromArray(corners.ToArray()));
                    var corners2 = Cv2.CornerSubPix(gray, corners, new Size(11, 11), new Size(-1, -1), criteria);
                    Cv2.DrawChessboardCorners(image, size, corners2, true);
                    image.SaveImage(output);
                    calib--;
                    await Task.Delay(100);
                }
                image.SaveImage(output);
            }
        }
        rms = Cv2.CalibrateCamera(objPoints, imgPoints, frameSize, camera, distortion, out var rvectors, out var tvectors, CalibrationFlags.UseIntrinsicGuess | CalibrationFlags.FixK5);
        using (var newCamera = Cv2.GetOptimalNewCameraMatrix(camera, distortion, frameSize, 1, frameSize, out var roi))
        {
            await Task.Delay(1);
            while (capture.Grab())
            {
                using (var undistorted = new Mat())
                using (var image = capture.RetrieveMat())
                {
                    Cv2.Undistort(image, undistorted, camera, distortion, newCamera);
                    CvAruco.DetectMarkers(undistorted, dictionary, out Point2f[][] corners, out int[] ids, paramters, out Point2f[][] rejected);
                    if (ids.Any())
                    {
                        CvAruco.DrawDetectedMarkers(undistorted, corners, ids);
                        using (var rvecs = new Mat())
                        using (var tvecs = new Mat())
                        {
                            CvAruco.EstimatePoseSingleMarkers(corners, 0.065f, newCamera, distortion, rvecs, tvecs);
                            for (var i = 0; i < ids.Length; i++)
                            {
                                var rvec = rvecs.Get<Vec3d>(i);
                                var tvec = tvecs.Get<Vec3d>(i);
                                DrawAxis(undistorted, newCamera, distortion, rvec, tvec, 0.05f);
                            }
                        }
                    }
                    undistorted.SaveImage(output);
                }
            }
        }
    }
}

private static void DrawAxis(Mat image, InputArray camera, InputArray distortion, Vec3d rvec, Vec3d tvec, float length)
{
    if (image.Total() == 0 || (image.Channels() != 1 && image.Channels() != 3))
    {
        throw new ArgumentException(nameof(image));
    }
    if (length <= 0)
    {
        throw new ArgumentException(nameof(length));
    }
    // project axis points
    var axisPoints = new MatOfPoint3f()
    {
        new Point3f(0, 0, 0),
        new Point3f(length, 0, 0),
        new Point3f(0, length, 0),
        new Point3f(0, 0, length),
    };
    var imagePoints = new MatOfPoint2f();
    Cv2.ProjectPoints(axisPoints, InputArray.Create(new[] { rvec }), InputArray.Create(new[] { tvec }), camera, distortion, imagePoints);
    // draw axis lines
    Cv2.Line(image, imagePoints.Get<Point2f>(0), imagePoints.Get<Point2f>(1), new Scalar(0, 0, 255), 3);
    Cv2.Line(image, imagePoints.Get<Point2f>(0), imagePoints.Get<Point2f>(2), new Scalar(0, 255, 0), 3);
    Cv2.Line(image, imagePoints.Get<Point2f>(0), imagePoints.Get<Point2f>(3), new Scalar(255, 0, 0), 3);
}

private static IEnumerable<Point3f> Create3DChessboardCorners(Size boardSize, float squareSize)
{
    for (int y = 0; y < boardSize.Height; y++)
    {
        for (int x = 0; x < boardSize.Width; x++)
        {
            yield return new Point3f(x * squareSize, y * squareSize, 0);
        }
    }
}

1 Ответ

0 голосов
/ 13 сентября 2018

отсортировано, необходимо изменить с 0.025f на 1

var objp = MatOfPoint3f.FromArray(Create3DChessboardCorners(size, 1));
...