How to customize iOS camera in 30 minutes

How to customize iOS camera in 30 minutes

Recently, the company's project used a camera. Since the system camera was not used, the camera cutouts given by the UI had to be customized. I took some time to simply study the customization of the camera.

The camera belongs to the system hardware, which requires us to manually call the iPhone's camera hardware, which is divided into the following steps:

1. First declare the following objects

  1. #import < AVFoundation /AVFoundation.h >  
  2. //Capture device, usually front camera, rear camera, microphone (audio input)
  3. @property (nonatomic, strong) AVCaptureDevice *device;
  4.  
  5. //AVCaptureDeviceInput represents the input device, which is initialized using AVCaptureDevice
  6. @property (nonatomic, strong) AVCaptureDeviceInput *input;
  7.  
  8. // Output image
  9. @property (nonatomic, strong) AVCaptureStillImageOutput *imageOutput;
  10.  
  11. //session: It combines the input and output together and starts the capture device (camera)
  12. @property (nonatomic, strong) AVCaptureSession *session;
  13.  
  14. //Image preview layer, real-time display of captured images
  15. @property (nonatomic, strong) AVCaptureVideoPreviewLayer *previewLayer;

2. Initialize each object

  1. - (void)cameraDistrict
  2. {
  3. //AVCaptureDevicePositionBack rear camera
  4. //AVCaptureDevicePositionFront front camera
  5.     self.device = [self cameraWithPosition:AVCaptureDevicePositionFront];
  6.      self.input = [[AVCaptureDeviceInput alloc] initWithDevice:self.device error:nil];
  7.  
  8.      self.imageOutput = [[AVCaptureStillImageOutput alloc] init];
  9.  
  10.      self.session = [[AVCaptureSession alloc] init];
  11. // The size of the image obtained can be set by yourself
  12. //AVCaptureSessionPreset320x240
  13. //AVCaptureSessionPreset352x288
  14. //AVCaptureSessionPreset640x480
  15. //AVCaptureSessionPreset960x540
  16. //AVCaptureSessionPreset1280x720
  17. //AVCaptureSessionPreset1920x1080
  18. //AVCaptureSessionPreset3840x2160
  19.      self.session.sessionPreset = AVCaptureSessionPreset640x480 ;
  20. //Input and output device combination
  21. if ([self.session canAddInput:self.input]) {
  22. [self.session addInput:self.input];
  23. }
  24. if ([self.session canAddOutput:self.imageOutput]) {
  25. [self.session addOutput:self.imageOutput];
  26. }
  27. //Generate preview layer
  28.      self.previewLayer = [[AVCaptureVideoPreviewLayer alloc] initWithSession:self.session];
  29.      self.previewLayer.frame = CGRectMake (0, 64, SCREEN_WIDTH, SCREEN_HEIGHT-64);
  30.      self.previewLayer.videoGravity = AVLayerVideoGravityResizeAspectFill ;
  31. [self.view.layer addSublayer:self.previewLayer];
  32. //Device framing starts
  33. [self.session startRunning];
  34. if ([_device lockForConfiguration:nil]) {
  35. //Automatic flash,
  36. if ([_device isFlashModeSupported:AVCaptureFlashModeAuto]) {
  37. [_device setFlashMode:AVCaptureFlashModeAuto];
  38. }
  39. //Automatic white balance, but it seems that I can't get it in.
  40. if ([_device isWhiteBalanceModeSupported:AVCaptureWhiteBalanceModeAutoWhiteBalance]) {
  41. [_device setWhiteBalanceMode:AVCaptureWhiteBalanceModeAutoWhiteBalance];
  42. }
  43. [_device unlockForConfiguration];
  44. }
  45.  
  46. }

Get the corresponding camera according to the front and rear positions:

  1. - (AVCaptureDevice *)cameraWithPosition:(AVCaptureDevicePosition)position{
  2. NSArray * devices = [AVCaptureDevice devicesWithMediaType:AVMediaTypeVideo];
  3. for (AVCaptureDevice *device in devices)
  4. if ( device.position == position){
  5. return device;
  6. }
  7. return nil;
  8. }

3. Take a photo and get the corresponding picture:

  1. - (void)photoBtnDidClick
  2. {
  3. AVCaptureConnection * conntion = [self.imageOutput connectionWithMediaType:AVMediaTypeVideo];
  4. if (!conntion) {
  5. NSLog(@"Photo taking failed!");
  6. return;
  7. }
  8. [self.imageOutput captureStillImageAsynchronouslyFromConnection:conntion completionHandler:^(CMSampleBufferRef imageDataSampleBuffer, NSError *error) {
  9. if ( imageDataSampleBuffer == nil) {
  10. return ;
  11. }
  12. NSData * imageData = [AVCaptureStillImageOutput jpegStillImageNSDataRepresentation:imageDataSampleBuffer];
  13.          self.image = [UIImage imageWithData:imageData];
  14. [self.session stopRunning];
  15. [self.view addSubview:self.cameraImageView];
  16. }

4. Switch between front and rear cameras

  1. - (void)changeCamera{
  2. NSUInteger cameraCount = [[AVCaptureDevice devicesWithMediaType:AVMediaTypeVideo] count];
  3. if (cameraCount > 1) {
  4. NSError *error;
  5. //Add flip animation to the camera switch
  6. CATransition * animation = [CATransition animation];
  7.          animation.duration = .5f;
  8.          animation.timingFunction = [CAMediaTimingFunction functionWithName:kCAMediaTimingFunctionEaseInEaseOut];
  9.          animation.type = @"oglFlip";
  10.  
  11. AVCaptureDevice * newCamera = nil ;
  12. AVCaptureDeviceInput * newInput = nil ;
  13. //Get another camera position
  14. AVCaptureDevicePosition position = [[_input device] position];
  15. if ( position == AVCaptureDevicePositionFront){
  16.              newCamera = [self cameraWithPosition:AVCaptureDevicePositionBack];
  17.              animation.subtype = kCATransitionFromLeft ; // animation flip direction
  18. }
  19. else {
  20.              newCamera = [self cameraWithPosition:AVCaptureDevicePositionFront];
  21.              animation.subtype = kCATransitionFromRight ; // animation flip direction
  22. }
  23. //Generate new input
  24.          newInput = [AVCaptureDeviceInput deviceInputWithDevice:newCamera error:nil];
  25. [self.previewLayer addAnimation:animation forKey:nil];
  26. if (newInput != nil) {
  27. [self.session beginConfiguration];
  28. [self.session removeInput:self.input];
  29. if ([self.session canAddInput:newInput]) {
  30. [self.session addInput:newInput];
  31.                  self.input = newInput ;
  32.  
  33. } else {
  34. [self.session addInput:self.input];
  35. }
  36. [self.session commitConfiguration];
  37.  
  38. } else if (error) {
  39. NSLog(@"toggle carema failed, error = %@", error);
  40. }
  41. }
  42. }

5. Other camera parameter settings

  1. //AVCaptureFlashMode flash
  2. //AVCaptureFocusMode focus
  3. //AVCaptureExposureMode exposure
  4. //AVCaptureWhiteBalanceMode white balance
  5. //Flash and white balance can be set when generating the camera
  6. //Exposure is determined by the light conditions at the focus point, so write it together with the focus.
  7. //point is the click position
  8. - (void)focusAtPoint:(CGPoint)point{
  9. CGSize size = self .view.bounds.size;
  10. CGPoint focusPoint = CGPointMake ( point.y /size.height ,1-point.x/size.width );
  11. NSError *error;
  12. if ([self.device lockForConfiguration:&error]) {
  13. //Focus mode and focus point
  14. if ([self.device isFocusModeSupported:AVCaptureFocusModeAutoFocus]) {
  15. [self.device setFocusPointOfInterest:focusPoint];
  16. [self.device setFocusMode:AVCaptureFocusModeAutoFocus];
  17. }
  18. //Exposure mode and exposure point
  19. if ([self.device isExposureModeSupported:AVCaptureExposureModeAutoExpose]) {
  20. [self.device setExposurePointOfInterest:focusPoint];
  21. [self.device setExposureMode:AVCaptureExposureModeAutoExpose];
  22. }
  23.  
  24. [self.device unlockForConfiguration];
  25. //Set the focus animation
  26.          _focusView.center = point ;
  27.          _focusView.hidden = NO ;
  28. [UIView animateWithDuration:0.3 animations:^{
  29.              _focusView.transform = CGAffineTransformMakeScale (1.25, 1.25);
  30. }completion:^(BOOL finished) {
  31. [UIView animateWithDuration:0.5 animations:^{
  32.                  _focusView.transform = CGAffineTransformIdentity ;
  33. } completion:^(BOOL finished) {
  34.                  _focusView.hidden = YES ;
  35. }];
  36. }];
  37. }
  38.  
  39. }

6. Some pitfalls and solutions encountered

1) Switching between front and rear cameras

The front and back values ​​cannot be switched. I tried various ways to find the reason but couldn't find it. Later I found that I set the image size to 1080P [self.session canSetSessionPreset: AVCaptureSessionPreset1920x1080], and the front camera does not support such a large size, so the front camera cannot be switched. I verified that the front camera supports up to 720P, and can be switched freely within 720P.

Of course, you can also set different sizes according to the front and rear cameras when switching between them. I will not go into details here.

2) Focus position

CGPoint focusPoint = CGPointMake( point.y /size.height ,1-point.x/size.width );

The value range of Point after the setExposurePointOfInterest:focusPoint function is from the upper left corner of the viewfinder (0, 0) to the lower right corner of the viewfinder (1, 1). The official description is as follows:

The value of this property is a CGPoint that determines the receiver's focus point of interest, if it has one. A value of (0,0) indicates that the camera should focus on the top left corner of the image, while a value of (1,1) indicates that it should focus on the bottom right. The default value is (0.5,0.5).

I also tried to press this but the position is not right, I can only follow the above writing method. The front is the y of the click position/the height of the PreviewLayer, and the back is 1-the x of the click position/the width of the PreviewLayer

3) Focus and exposure

When I set the focus, I first set the mode setFocusMode, and then set the focus position, which will cause a very strange phenomenon. The focus position is the position you clicked last time. So you must set the position first, and then set the focus mode.

Exposure same as above

7. Final words

Attached demo: https://github.com/nanshanyi/photographDemo

These are the basic ones that are commonly used. This is not perfect. If there is anything wrong, you are welcome to criticize and correct me so that we can learn together.

<<:  Register as a mobile developer essential tool, get 100G Baichuan Multimedia traffic and 10 yuan recharge card for free

>>:  Tomorrow! China's first AR/VR Technology Conference invites you to participate!

Recommend

iOS developers should know: which data persistence solutions

Introduction The so-called persistence is to save...

How to apply for WeChat Small Store? The application process is as follows

WeChat officially announced on August 20 that WeC...

In the education industry, how should information flow advertising be used?

Recently, the investment guidelines for major ind...

Facebook and Twitter's "Growth Hackers" tell you about user growth!

Andy Jones was a growth hacker for companies such...

From 0 to 1 users, how to do a good job in seed-stage user growth operations?

The product operation cycle is generally divided ...

How to master fission growth?

There is an old saying: Three incompetent general...

App paid promotion, five factors that affect user registration conversion rate

With the development of mobile Internet, the numb...