zoukankan      html  css  js  c++  java
  • ios 人脸检测

    #import "OpenCVTestViewController.h"

    #import <opencv2/imgproc/imgproc_c.h>
    #import <opencv2/objdetect/objdetect.hpp>

    @implementation OpenCVTestViewController
    @synthesize imageView;

    - (void)dealloc {
    AudioServicesDisposeSystemSoundID(alertSoundID);
    [imageView dealloc];
    [super dealloc];
    }

    #pragma mark -
    #pragma mark OpenCV Support Methods

    // NOTE you SHOULD cvReleaseImage() for the return value when end of the code.
    - (IplImage *)CreateIplImageFromUIImage:(UIImage *)image {
    CGImageRef imageRef = image.CGImage;

    CGColorSpaceRef colorSpace = CGColorSpaceCreateDeviceRGB();
    IplImage *iplimage = cvCreateImage(cvSize(image.size.width, image.size.height), IPL_DEPTH_8U, 4);
    CGContextRef contextRef = CGBitmapContextCreate(iplimage->imageData, iplimage->width, iplimage->height,
    iplimage->depth, iplimage->widthStep,
    colorSpace, kCGImageAlphaPremultipliedLast|kCGBitmapByteOrderDefault);
    CGContextDrawImage(contextRef, CGRectMake(0, 0, image.size.width, image.size.height), imageRef);
    CGContextRelease(contextRef);
    CGColorSpaceRelease(colorSpace);

    IplImage *ret = cvCreateImage(cvGetSize(iplimage), IPL_DEPTH_8U, 3);
    cvCvtColor(iplimage, ret, CV_RGBA2BGR);
    cvReleaseImage(&iplimage);

    return ret;
    }

    // NOTE You should convert color mode as RGB before passing to this function
    - (UIImage *)UIImageFromIplImage:(IplImage *)image {
    NSLog(@"IplImage (%d, %d) %d bits by %d channels, %d bytes/row %s", image->width, image->height, image->depth, image->nChannels, image->widthStep, image->channelSeq);

    CGColorSpaceRef colorSpace = CGColorSpaceCreateDeviceRGB();
    NSData *data = [NSData dataWithBytes:image->imageData length:image->imageSize];
    CGDataProviderRef provider = CGDataProviderCreateWithCFData((CFDataRef)data);
    CGImageRef imageRef = CGImageCreate(image->width, image->height,
    image->depth, image->depth * image->nChannels, image->widthStep,
    colorSpace, kCGImageAlphaNone|kCGBitmapByteOrderDefault,
    provider, NULL, false, kCGRenderingIntentDefault);
    UIImage *ret = [UIImage imageWithCGImage:imageRef];
    CGImageRelease(imageRef);
    CGDataProviderRelease(provider);
    CGColorSpaceRelease(colorSpace);
    return ret;
    }

    #pragma mark -
    #pragma mark Utilities for intarnal use

    - (void)showProgressIndicator:(NSString *)text {
    //[UIApplication sharedApplication].networkActivityIndicatorVisible = YES;
    self.view.userInteractionEnabled = FALSE;
    if(!progressHUD) {
    CGFloat w = 160.0f, h = 120.0f;
    progressHUD = [[UIProgressHUD alloc] initWithFrame:CGRectMake((self.view.frame.size.width-w)/2, (self.view.frame.size.height-h)/2, w, h)];
    [progressHUD setText:text];
    [progressHUD showInView:self.view];
    }
    }

    - (void)hideProgressIndicator {
    //[UIApplication sharedApplication].networkActivityIndicatorVisible = NO;
    self.view.userInteractionEnabled = TRUE;
    if(progressHUD) {
    [progressHUD hide];
    [progressHUD release];
    progressHUD = nil;

    AudioServicesPlaySystemSound(alertSoundID);
    }
    }

    - (void)opencvEdgeDetect {
    NSAutoreleasePool * pool = [[NSAutoreleasePool alloc] init];

    if(imageView.image) {
    cvSetErrMode(CV_ErrModeParent);

    // Create grayscale IplImage from UIImage
    IplImage *img_color = [self CreateIplImageFromUIImage:imageView.image];
    IplImage *img = cvCreateImage(cvGetSize(img_color), IPL_DEPTH_8U, 1);
    cvCvtColor(img_color, img, CV_BGR2GRAY);
    cvReleaseImage(&img_color);

    // Detect edge
    IplImage *img2 = cvCreateImage(cvGetSize(img), IPL_DEPTH_8U, 1);
    cvCanny(img, img2, 64, 128, 3);
    cvReleaseImage(&img);

    // Convert black and whilte to 24bit image then convert to UIImage to show
    IplImage *image = cvCreateImage(cvGetSize(img2), IPL_DEPTH_8U, 3);
    for(int y=0; y<img2->height; y++) {
    for(int x=0; x<img2->width; x++) {
    char *p = image->imageData + y * image->widthStep + x * 3;
    *p = *(p+1) = *(p+2) = img2->imageData[y * img2->widthStep + x];
    }
    }
    cvReleaseImage(&img2);
    imageView.image = [self UIImageFromIplImage:image];
    cvReleaseImage(&image);

    [self hideProgressIndicator];
    }

    [pool release];
    }

    - (void) opencvFaceDetect:(UIImage *)overlayImage {
    NSAutoreleasePool * pool = [[NSAutoreleasePool alloc] init];

    if(imageView.image) {
    cvSetErrMode(CV_ErrModeParent);

    IplImage *image = [self CreateIplImageFromUIImage:imageView.image];

    // Scaling down
    IplImage *small_image = cvCreateImage(cvSize(image->width/2,image->height/2), IPL_DEPTH_8U, 3);
    cvPyrDown(image, small_image, CV_GAUSSIAN_5x5);
    int scale = 2;

    // Load XML
    NSString *path = [[NSBundle mainBundle] pathForResource:@"haarcascade_frontalface_default" ofType:@"xml"];
    CvHaarClassifierCascade* cascade = (CvHaarClassifierCascade*)cvLoad([path cStringUsingEncoding:NSASCIIStringEncoding], NULL, NULL, NULL);
    CvMemStorage* storage = cvCreateMemStorage(0);

    // Detect faces and draw rectangle on them
    CvSeq* faces = cvHaarDetectObjects(small_image, cascade, storage, 1.2f, 2, CV_HAAR_DO_CANNY_PRUNING, cvSize(0,0), cvSize(20, 20));
    cvReleaseImage(&small_image);

    // Create canvas to show the results
    CGImageRef imageRef = imageView.image.CGImage;
    CGColorSpaceRef colorSpace = CGColorSpaceCreateDeviceRGB();
    CGContextRef contextRef = CGBitmapContextCreate(NULL, imageView.image.size.width, imageView.image.size.height,
    8, imageView.image.size.width * 4,
    colorSpace, kCGImageAlphaPremultipliedLast|kCGBitmapByteOrderDefault);
    CGContextDrawImage(contextRef, CGRectMake(0, 0, imageView.image.size.width, imageView.image.size.height), imageRef);

    CGContextSetLineWidth(contextRef, 4);
    CGContextSetRGBStrokeColor(contextRef, 0.0, 0.0, 1.0, 0.5);

    // Draw results on the iamge
    for(int i = 0; i < faces->total; i++) {
    NSAutoreleasePool * pool = [[NSAutoreleasePool alloc] init];

    // Calc the rect of faces
    CvRect cvrect = *(CvRect*)cvGetSeqElem(faces, i);
    CGRect face_rect = CGContextConvertRectToDeviceSpace(contextRef, CGRectMake(cvrect.x * scale, cvrect.y * scale, cvrect.width * scale, cvrect.height * scale));

    if(overlayImage) {
    CGContextDrawImage(contextRef, face_rect, overlayImage.CGImage);
    } else {
    CGContextStrokeRect(contextRef, face_rect);
    }

    [pool release];
    }

    imageView.image = [UIImage imageWithCGImage:CGBitmapContextCreateImage(contextRef)];
    CGContextRelease(contextRef);
    CGColorSpaceRelease(colorSpace);

    cvReleaseMemStorage(&storage);
    cvReleaseHaarClassifierCascade(&cascade);

    [self hideProgressIndicator];
    }

    [pool release];
    }


    #pragma mark -
    #pragma mark IBAction

    - (IBAction)loadImage:(id)sender {
    if(!actionSheetAction) {
    UIActionSheet *actionSheet = [[UIActionSheet alloc] initWithTitle:@""
    delegate:self cancelButtonTitle:@"Cancel" destructiveButtonTitle:nil
    otherButtonTitles:@"Use Photo from Library", @"Take Photo with Camera", @"Use Default Lena", nil];
    actionSheet.actionSheetStyle = UIActionSheetStyleDefault;
    actionSheetAction = ActionSheetToSelectTypeOfSource;
    [actionSheet showInView:self.view];
    [actionSheet release];
    }
    }

    - (IBAction)saveImage:(id)sender {
    if(imageView.image) {
    [self showProgressIndicator:@"Saving"];
    UIImageWriteToSavedPhotosAlbum(imageView.image, self, @selector(finishUIImageWriteToSavedPhotosAlbum:didFinishSavingWithError:contextInfo:), nil);
    }
    }

    - (void)finishUIImageWriteToSavedPhotosAlbum:(UIImage *)image didFinishSavingWithError:(NSError *)error contextInfo:(void *)contextInfo {
    [self hideProgressIndicator];
    }

    - (IBAction)edgeDetect:(id)sender {
    [self showProgressIndicator:@"Detecting"];
    [self performSelectorInBackground:@selector(opencvEdgeDetect) withObject:nil];
    }

    - (IBAction)faceDetect:(id)sender {
    cvSetErrMode(CV_ErrModeParent);
    if(imageView.image && !actionSheetAction) {
    UIActionSheet *actionSheet = [[UIActionSheet alloc] initWithTitle:@""
    delegate:self cancelButtonTitle:@"Cancel" destructiveButtonTitle:nil
    otherButtonTitles:@"Bounding Box", @"Laughing Man", nil];
    actionSheet.actionSheetStyle = UIActionSheetStyleDefault;
    actionSheetAction = ActionSheetToSelectTypeOfMarks;
    [actionSheet showInView:self.view];
    [actionSheet release];
    }
    }

    #pragma mark -
    #pragma mark UIViewControllerDelegate

    - (void)viewDidLoad {
    [super viewDidLoad];
    [[UIApplication sharedApplication] setStatusBarStyle:UIStatusBarStyleBlackOpaque animated:YES];
    [self loadImage:nil];

    NSURL *url = [NSURL fileURLWithPath:[[NSBundle mainBundle] pathForResource:@"Tink" ofType:@"aiff"] isDirectory:NO];
    AudioServicesCreateSystemSoundID((CFURLRef)url, &alertSoundID);
    }

    - (BOOL)shouldAutorotateToInterfaceOrientation:(UIInterfaceOrientation)interfaceOrientation {
    return NO;
    }

    #pragma mark -
    #pragma mark UIActionSheetDelegate

    - (void)actionSheet:(UIActionSheet *)actionSheet clickedButtonAtIndex:(NSInteger)buttonIndex {
    switch(actionSheetAction) {
    case ActionSheetToSelectTypeOfSource: {
    UIImagePickerControllerSourceType sourceType;
    if (buttonIndex == 0) {
    sourceType = UIImagePickerControllerSourceTypePhotoLibrary;
    } else if(buttonIndex == 1) {
    sourceType = UIImagePickerControllerSourceTypeCamera;
    } else if(buttonIndex == 2) {
    NSString *path = [[NSBundle mainBundle] pathForResource:@"lena" ofType:@"jpg"];
    imageView.image = [UIImage imageWithContentsOfFile:path];
    break;
    } else {
    // Cancel
    break;
    }
    if([UIImagePickerController isSourceTypeAvailable:sourceType]) {
    UIImagePickerController *picker = [[UIImagePickerController alloc] init];
    picker.sourceType = sourceType;
    picker.delegate = self;
    picker.allowsImageEditing = NO;
    [self presentModalViewController:picker animated:YES];
    [picker release];
    }
    break;
    }
    case ActionSheetToSelectTypeOfMarks: {
    if(buttonIndex != 0 && buttonIndex != 1) {
    break;
    }

    UIImage *image = nil;
    if(buttonIndex == 1) {
    NSString *path = [[NSBundle mainBundle] pathForResource:@"laughing_man" ofType:@"png"];
    image = [UIImage imageWithContentsOfFile:path];
    }

    [self showProgressIndicator:@"Detecting"];
    [self performSelectorInBackground:@selector(opencvFaceDetect:) withObject:image];
    break;
    }
    }
    actionSheetAction = 0;
    }

    #pragma mark -
    #pragma mark UIImagePickerControllerDelegate

    - (UIImage *)scaleAndRotateImage:(UIImage *)image {
    static int kMaxResolution = 640;

    CGImageRef imgRef = image.CGImage;
    CGFloat width = CGImageGetWidth(imgRef);
    CGFloat height = CGImageGetHeight(imgRef);

    CGAffineTransform transform = CGAffineTransformIdentity;
    CGRect bounds = CGRectMake(0, 0, width, height);
    if (width > kMaxResolution || height > kMaxResolution) {
    CGFloat ratio = width/height;
    if (ratio > 1) {
    bounds.size.width = kMaxResolution;
    bounds.size.height = bounds.size.width / ratio;
    } else {
    bounds.size.height = kMaxResolution;
    bounds.size.width = bounds.size.height * ratio;
    }
    }

    CGFloat scaleRatio = bounds.size.width / width;
    CGSize imageSize = CGSizeMake(CGImageGetWidth(imgRef), CGImageGetHeight(imgRef));
    CGFloat boundHeight;

    UIImageOrientation orient = image.imageOrientation;
    switch(orient) {
    case UIImageOrientationUp:
    transform = CGAffineTransformIdentity;
    break;
    case UIImageOrientationUpMirrored:
    transform = CGAffineTransformMakeTranslation(imageSize.width, 0.0);
    transform = CGAffineTransformScale(transform, -1.0, 1.0);
    break;
    case UIImageOrientationDown:
    transform = CGAffineTransformMakeTranslation(imageSize.width, imageSize.height);
    transform = CGAffineTransformRotate(transform, M_PI);
    break;
    case UIImageOrientationDownMirrored:
    transform = CGAffineTransformMakeTranslation(0.0, imageSize.height);
    transform = CGAffineTransformScale(transform, 1.0, -1.0);
    break;
    case UIImageOrientationLeftMirrored:
    boundHeight = bounds.size.height;
    bounds.size.height = bounds.size.width;
    bounds.size.width = boundHeight;
    transform = CGAffineTransformMakeTranslation(imageSize.height, imageSize.width);
    transform = CGAffineTransformScale(transform, -1.0, 1.0);
    transform = CGAffineTransformRotate(transform, 3.0 * M_PI / 2.0);
    break;
    case UIImageOrientationLeft:
    boundHeight = bounds.size.height;
    bounds.size.height = bounds.size.width;
    bounds.size.width = boundHeight;
    transform = CGAffineTransformMakeTranslation(0.0, imageSize.width);
    transform = CGAffineTransformRotate(transform, 3.0 * M_PI / 2.0);
    break;
    case UIImageOrientationRightMirrored:
    boundHeight = bounds.size.height;
    bounds.size.height = bounds.size.width;
    bounds.size.width = boundHeight;
    transform = CGAffineTransformMakeScale(-1.0, 1.0);
    transform = CGAffineTransformRotate(transform, M_PI / 2.0);
    break;
    case UIImageOrientationRight:
    boundHeight = bounds.size.height;
    bounds.size.height = bounds.size.width;
    bounds.size.width = boundHeight;
    transform = CGAffineTransformMakeTranslation(imageSize.height, 0.0);
    transform = CGAffineTransformRotate(transform, M_PI / 2.0);
    break;
    default:
    [NSException raise:NSInternalInconsistencyException format:@"Invalid image orientation"];
    }

    UIGraphicsBeginImageContext(bounds.size);
    CGContextRef context = UIGraphicsGetCurrentContext();
    if (orient == UIImageOrientationRight || orient == UIImageOrientationLeft) {
    CGContextScaleCTM(context, -scaleRatio, scaleRatio);
    CGContextTranslateCTM(context, -height, 0);
    } else {
    CGContextScaleCTM(context, scaleRatio, -scaleRatio);
    CGContextTranslateCTM(context, 0, -height);
    }
    CGContextConcatCTM(context, transform);
    CGContextDrawImage(UIGraphicsGetCurrentContext(), CGRectMake(0, 0, width, height), imgRef);
    UIImage *imageCopy = UIGraphicsGetImageFromCurrentImageContext();
    UIGraphicsEndImageContext();

    return imageCopy;
    }

    - (void)imagePickerController:(UIImagePickerController *)picker
    didFinishPickingImage:(UIImage *)image
    editingInfo:(NSDictionary *)editingInfo
    {
    imageView.image = [self scaleAndRotateImage:image];
    [[picker parentViewController] dismissModalViewControllerAnimated:YES];
    }

    - (void)imagePickerControllerDidCancel:(UIImagePickerController *)picker {
    [[picker parentViewController] dismissModalViewControllerAnimated:YES];
    }
    @end

  • 相关阅读:
    快速开始
    阿里为什么选择RocketMQ
    4 分布式消息队列的协调者
    9 首个Apache中间件顶级项目
    3、用适合的方式发送和接收消息
    2 生产环境下的配置和使用
    TOMCAT加载两次war包(重复加载)
    Google Protocol Buffer 的使用(二)
    Google Protocol Buffer 的使用(一)
    PostgreSQL及PostGIS使用
  • 原文地址:https://www.cnblogs.com/mfryf/p/2377945.html
Copyright © 2011-2022 走看看