在触摸模式下无法启动iPhone X闪存

laz*_*rev 5 camera objective-c avfoundation ios flashlight

我在火炬模式下运行iPhone X闪存时遇到问题.

返回选择AVCaptureDeviceTypeBuiltInTelephotoCamera作为捕获设备:

com.apple.avfoundation.avcapturedevice.built-in_video:2' -
AVCaptureDeviceTypeBuiltInTelephotoCamera
Run Code Online (Sandbox Code Playgroud)

检查触摸模式可用性后:

[self.captureDevice isTorchModeSupported:AVCaptureTorchModeOn]
Run Code Online (Sandbox Code Playgroud)

我正试图用闪光灯将闪光灯切换到火炬模式

[self.captureDevice lockForConfiguration:nil];
BOOL result = [self.captureDevice setTorchModeOnWithLevel:1 error:&error];
[self.captureDevice unlockForConfiguration];
Run Code Online (Sandbox Code Playgroud)

此通话成功.结果==是和错误==无.但闪光灯闪烁一次然后熄灭.

我自己在iPhone X上看到了这种行为,并且有报告显示iPhone 8和iPhone 8 Plus用户的行为相同.有些用户说在更新到iOS 11.1后出现此问题.但我自己无法用iPhone 8重现它.

有任何想法如何修复或调试此问题?

以下列出的应用中的完整代码段:

// Retrieve the back camera
    if ([AVCaptureDeviceDiscoverySession class]) {
        DDLogDebug(@"Search camera with AVCaptureDeviceDiscoverySession");
        AVCaptureDevice* camera =
        [AVCaptureDeviceDiscoverySession
         discoverySessionWithDeviceTypes: @[AVCaptureDeviceTypeBuiltInTelephotoCamera]
         mediaType:AVMediaTypeVideo
         position:AVCaptureDevicePositionBack].devices.firstObject;

        if (!camera) {
            camera = [AVCaptureDeviceDiscoverySession
                      discoverySessionWithDeviceTypes: @[AVCaptureDeviceTypeBuiltInTelephotoCamera]
                      mediaType:AVMediaTypeVideo
                      position:AVCaptureDevicePositionBack].devices.firstObject;
        }
        DDLogDebug(@"Did find %@ camera", camera);
        self.captureDevice = camera;
    } else {
        DDLogDebug(@"Haven't found camera device with AVCaptureDeviceDiscoverySession");
    }

    if (!self.captureDevice) {
        DDLogDebug(@"Searching at [AVCaptureDevice devices], where %lu devices available", (unsigned long)AVCaptureDevice.devices.count);
        for (AVCaptureDevice *device in [AVCaptureDevice devices]) {
            if ([device hasMediaType:AVMediaTypeVideo] && [device hasTorch]) {
                self.captureDevice = device;
                break;
            }
        }
    }

    if (!self.captureDevice) {
        NSError* error = [NSError buildError:^(MRErrorBuilder *builder) {
            builder.localizedDescription = NSLocalizedString(@"There is no camera devices able to measure heart rate", nil);
            builder.domain               = kWTCameraHeartRateMonitorError;
            builder.code                 = 27172;
        }];
        DDLogError(@"%@", error);
        self.session = nil;
        self.handler(0, 0, error);
        return NO;
    }

    NSError *error;
    AVCaptureDeviceInput *input = [[AVCaptureDeviceInput alloc] initWithDevice:self.captureDevice
                                                                         error:&error];
    if (error) {
        DDLogError(@"%@", error);
        self.session = nil;
        self.handler(0, 0, error);
        return NO;
    }


    NSString* deviceType = [self.captureDevice respondsToSelector:@selector(deviceType)] ? self.captureDevice.deviceType : @"Unknown";

    DDLogDebug(@"Configurating camera '%@'/'%@' - %@ id %@ at %ld connected: %@", self.captureDevice.localizedName, self.captureDevice.modelID, deviceType, self.captureDevice.uniqueID, (long)self.captureDevice.position, self.captureDevice.connected?@"YES":@"NO");

    self.session               = [[AVCaptureSession alloc] init];
    NSString* preset = [self.session canSetSessionPreset:AVCaptureSessionPresetLow] ? AVCaptureSessionPresetLow : nil;
    if (preset) {
        self.session.sessionPreset = preset;
    }

    [self.session beginConfiguration];
    [self.session addInput:input];

    // Find the max frame rate we can get from the given device
    AVCaptureDeviceFormat *currentFormat;
    for (AVCaptureDeviceFormat *format in self.captureDevice.formats)
    {
        NSArray *ranges = format.videoSupportedFrameRateRanges;
        AVFrameRateRange *frameRates = ranges[0];

        // Find the lowest resolution format at the frame rate we want.
        if (frameRates.maxFrameRate == FRAMES_PER_SECOND && (!currentFormat || (CMVideoFormatDescriptionGetDimensions(format.formatDescription).width < CMVideoFormatDescriptionGetDimensions(currentFormat.formatDescription).width && CMVideoFormatDescriptionGetDimensions(format.formatDescription).height < CMVideoFormatDescriptionGetDimensions(currentFormat.formatDescription).height)))
        {
            currentFormat = format;
        }
    }

    if (![self.captureDevice isTorchModeSupported:AVCaptureTorchModeOn]) {
        NSError* error = [NSError buildError:^(MRErrorBuilder *builder) {
            builder.localizedDescription = NSLocalizedString(@"Torch mode is not supported for your camera", nil);
            builder.domain               = kWTCameraHeartRateMonitorError;
            builder.code                 = 28633;
        }];
        self.session = nil;
        DDLogError(@"%@", error);
        self.session = nil;
        self.handler(0, 0, error);
        return NO;
    }

    // Tell the device to use the max frame rate.
    [self.captureDevice lockForConfiguration:nil];
    DDLogVerbose(@"Turn on tourch mode with level 0.5");
    self.captureDevice.flashMode = AVCaptureFlashModeOff;
    BOOL result = [self.captureDevice setTorchModeOnWithLevel:0.5 error:&error];
    if (!result) {
        DDLogError(@"%@", error);
        self.session = nil;
        self.handler(0, 0, error);
        return NO;
    }
    [self.captureDevice setFocusMode:AVCaptureFocusModeLocked];
    [self.captureDevice setFocusModeLockedWithLensPosition:1.0
                                         completionHandler:nil];
    self.captureDevice.activeFormat = currentFormat;
    self.captureDevice.activeVideoMinFrameDuration = CMTimeMake(1, FRAMES_PER_SECOND);
    self.captureDevice.activeVideoMaxFrameDuration = CMTimeMake(1, FRAMES_PER_SECOND);
    [self.captureDevice unlockForConfiguration];

    // Set the output
    AVCaptureVideoDataOutput* videoOutput = [AVCaptureVideoDataOutput new];

    // create a queue to run the capture on
    dispatch_queue_t captureQueue=dispatch_queue_create("catpureQueue", DISPATCH_QUEUE_SERIAL);

    // setup our delegate
    [videoOutput setSampleBufferDelegate:self queue:captureQueue];

    // configure the pixel format

    videoOutput.videoSettings = @{(id)kCVPixelBufferPixelFormatTypeKey: @(kCVPixelFormatType_32BGRA)};
    videoOutput.alwaysDiscardsLateVideoFrames = NO;

    [self.session addOutput:videoOutput];

    if (debugPath) {
        NSError* error;
        [[NSFileManager defaultManager] removeItemAtPath:debugPath
                                                   error:nil];

        BOOL result =
        [[NSFileManager defaultManager] createDirectoryAtPath:debugPath
                                  withIntermediateDirectories:YES
                                                   attributes:nil
                                                        error:&error];
        if (result) {
            [self setupDebugRecordAt:debugPath withFormat:currentFormat];
        } else {
            DDLogError(@"%@", error);
        }

        const char* path = [debugPath cStringUsingEncoding:NSUTF8StringEncoding];
        self.filter->setDebugPath(path);
    }


    // Start the video session
    [self.session commitConfiguration];

    self.frameNumber = 0;
    [self.assetWriter startWriting];
    [self.assetWriter startSessionAtSourceTime:kCMTimeZero];
    [self.session startRunning];
Run Code Online (Sandbox Code Playgroud)

laz*_*rev 6

最后,问题得到解决.我不确定确切的原因.任何与此问题相关的信息都表示赞赏.

在运行iOS 11.1的iPhone 8,8 +和iPhone X上存在此​​问题.在将iOS从11.0更新到11.1之后,我在iPhone 8上重现了这种行为.

我注意到火炬在打电话后开启了

BOOL result = [self.captureDevice setTorchModeOnWithLevel:0.5 error:&error];
Run Code Online (Sandbox Code Playgroud)

然后关掉

[self.captureDevice setFocusMode:AVCaptureFocusModeLocked];
Run Code Online (Sandbox Code Playgroud)

要么

[self.session commitConfiguration];
Run Code Online (Sandbox Code Playgroud)

因此解决方案是执行火炬配置,其中所有其他会话和设备配置完成并启动会话.

我目前的实施是:

// Session configuration ...

[self.session startRunning];

if (![self.captureDevice isTorchModeSupported:AVCaptureTorchModeOn]) {
    NSError* error = [NSError buildError:^(MRErrorBuilder *builder) {
        builder.localizedDescription = NSLocalizedString(@"Torch mode is not supported for your camera", nil);
        builder.domain               = kWTCameraHeartRateMonitorError;
        builder.code                 = 28633;
    }];
    DDLogError(@"%@", error);
    if (self.session) {
       [self.session stopRunning];
    }
    self.session = nil;
    self.handler(0, 0, error);
    return NO;
}

[self.captureDevice lockForConfiguration:nil];
self.captureDevice.flashMode = AVCaptureFlashModeOff;
[self.captureDevice setFocusMode:AVCaptureFocusModeLocked];
[self.captureDevice setFocusModeLockedWithLensPosition:1.0
                                         completionHandler:nil];
self.captureDevice.activeFormat = currentFormat;
self.captureDevice.activeVideoMinFrameDuration = CMTimeMake(1, FRAMES_PER_SECOND);
self.captureDevice.activeVideoMaxFrameDuration = CMTimeMake(1, FRAMES_PER_SECOND);

// This call should be placed AFTER all other configurations

BOOL result = [self.captureDevice setTorchModeOnWithLevel:0.5 error:&error];
if (!result) {
    DDLogError(@"%@", error);
    self.session = nil;
    self.handler(0, 0, error);
    return NO;
}
[self.captureDevice unlockForConfiguration];
Run Code Online (Sandbox Code Playgroud)