OpenCVを使ってテンプレートマッチングをする
OpenCVをダウンロード
http://opencv.org/releases.html
フレームワークをプロジェクトに追加
GeneralのLinked Framework and Librariesに追加
ファイルを.mmに変更する
これでObjective-CとC++を混在させることができる。
ヘッダをインポート
#ifdef __cplusplus #import <opencv2/opencv.hpp> #import <opencv2/highgui/highgui_c.h> #import <opencv2/imgproc/imgproc_c.h> #import <opencv2/core/core_c.h> #endif #import <UIKit/UIKit.h>
プログラム
UIImageからcvMatに変換
- (cv::Mat)cvMatFromUIImage:(UIImage *)image { CGColorSpaceRef colorSpace = CGImageGetColorSpace(image.CGImage); CGFloat cols = image.size.width; CGFloat rows = image.size.height; cv::Mat cvMat(rows, cols, CV_8UC4); // 8 bits per component, 4 channels CGContextRef contextRef = CGBitmapContextCreate(cvMat.data, // Pointer to data cols, // Width of bitmap rows, // Height of bitmap 8, // Bits per component cvMat.step[0], // Bytes per row colorSpace, // Colorspace kCGImageAlphaNoneSkipLast | kCGBitmapByteOrderDefault); // Bitmap info flags CGContextDrawImage(contextRef, CGRectMake(0, 0, cols, rows), image.CGImage); CGContextRelease(contextRef); CGColorSpaceRelease(colorSpace); return cvMat; }
cvMatからUIImageに変換
- (UIImage *)UIImageFromCVMat:(cv::Mat)cvMat { NSData *data = [NSData dataWithBytes:cvMat.data length:cvMat.elemSize()*cvMat.total()]; CGColorSpaceRef colorSpace; if (cvMat.elemSize() == 1) { colorSpace = CGColorSpaceCreateDeviceGray(); } else { colorSpace = CGColorSpaceCreateDeviceRGB(); } CGDataProviderRef provider = CGDataProviderCreateWithCFData((__bridge CFDataRef)data); // Creating CGImage from cv::Mat CGImageRef imageRef = CGImageCreate(cvMat.cols, //width cvMat.rows, //height 8, //bits per component 8 * cvMat.elemSize(), //bits per pixel cvMat.step[0], //bytesPerRow colorSpace, //colorspace kCGImageAlphaNone|kCGBitmapByteOrderDefault,// bitmap info provider, //CGDataProviderRef NULL, //decode false, //should interpolate kCGRenderingIntentDefault //intent ); // Getting UIImage from CGImage UIImage *finalImage = [UIImage imageWithCGImage:imageRef]; CGImageRelease(imageRef); CGDataProviderRelease(provider); CGColorSpaceRelease(colorSpace); return finalImage; }
マッチング
-(UIImage*)match:(UIImage*)srcImage tempImage:(UIImage*)tempImage { cv::Mat srcMat = [self cvMatFromUIImage:srcImage]; cv::Mat tmpMat = [self cvMatFromUIImage:tempImage]; cv::Mat dst = srcMat.clone(); // マッチング cv::matchTemplate(srcMat, tmpMat, dst, cv::TM_CCOEFF); double min_val, max_val; cv::Point min_loc, max_loc; cv::minMaxLoc(dst, &min_val, &max_val, &min_loc, &max_loc); // 描画 cv::rectangle(srcMat, max_loc, cv::Point(max_loc.x + tmpMat.cols, max_loc.y + tmpMat.rows), CV_RGB(0, 255, 0), 2); return [self UIImageFromCVMat:srcMat]; }
ハマったところ
ヘッダファイルのインポートで、UIKitよりも先にopencv2をインポートしないとエラーが出た。
参考
http://qiita.com/qt6hy/items/ab4a887cd07fc679c6c6