iOS: AVCaptureOutput перестает работать после навигации

Я использую вид камеры как сканер штрих-кода, который сканирует штрих-код и запускает переход к другому контроллеру представления с веб-просмотром. Это работает нормально, и я могу вернуться к сканеру из веб-просмотра и сканировать другой штрих-код без проблем. Однако, если я отойду от контроллера вида с камерой и вернусь к нему, изображение с камеры загружается, но больше не обнаруживает штрих-коды.

@implementation ProductScanViewController

NSString *loadUrl;     
AVCaptureSession *_captureSession;
AVCaptureDevice *_videoDevice;
AVCaptureDeviceInput *_videoInput;
AVCaptureVideoPreviewLayer *_previewLayer;
BOOL _running;
AVCaptureMetadataOutput *_metadataOutput;
@synthesize mWebView;


- (void)viewDidLoad {
    [super viewDidLoad];
NSURL *url = [NSURL URLWithString:loadUrl];
NSURLRequest *request = [NSURLRequest requestWithURL:url];
NSOperationQueue *queue = [[NSOperationQueue alloc]init];
[NSURLConnection sendAsynchronousRequest:request queue:queue completionHandler:^(NSURLResponse *response, NSData *data, NSError *error)
 {if([data length] > 0 && error == nil)[mWebView loadRequest:request];         else if (error != nil) NSLog(@"Error: %", error);}
 ];

    [self setupCaptureSession];
}

- (void)setupCaptureSession {

    // 1
    if (_captureSession){        
        return;
    }

    // 2
    _videoDevice = [AVCaptureDevice defaultDeviceWithMediaType:AVMediaTypeVideo];
    if (!_videoDevice) {
        return;
    }

    // 3
    _captureSession = [[AVCaptureSession alloc] init];

    // 4
    _videoInput = [[AVCaptureDeviceInput alloc] initWithDevice:_videoDevice error:nil];

    // 5
    if ([_captureSession canAddInput:_videoInput]) {
        [_captureSession addInput:_videoInput];
    }

    // 6
    _previewLayer = [[AVCaptureVideoPreviewLayer alloc] initWithSession:_captureSession];
    _previewLayer.videoGravity = AVLayerVideoGravityResizeAspectFill;

    // capture and process the metadata
    _metadataOutput = [[AVCaptureMetadataOutput alloc] init];
    dispatch_queue_t metadataQueue =
    dispatch_queue_create("com.1337labz.featurebuild.metadata", 0);
    [_metadataOutput setMetadataObjectsDelegate:self
                                      queue:metadataQueue];
    if ([_captureSession canAddOutput:_metadataOutput]) {
        [_captureSession addOutput:_metadataOutput];
    }
}

#pragma mark - Delegate functions

- (void)captureOutput:(AVCaptureOutput *)captureOutput
    didOutputMetadataObjects:(NSArray *)metadataObjects
       fromConnection:(AVCaptureConnection *)connection {

    [metadataObjects enumerateObjectsUsingBlock:^(AVMetadataObject *obj,
                               NSUInteger idx,
                              BOOL *stop) {
         if ([obj isKindOfClass: [AVMetadataMachineReadableCodeObject class]]) {
             //NSLog(@"Capture Output started");

            // 3
            AVMetadataMachineReadableCodeObject *code = (AVMetadataMachineReadableCodeObject*)
            [_previewLayer transformedMetadataObjectForMetadataObject:obj];

            // 4
            Barcode * barcode = [Barcode processMetadataObject:code];

            for (NSString * str in self.allowedBarcodeTypes) {
                 if([barcode.getBarcodeType isEqualToString:str]){
                     [self validBarcodeFound:(barcode)];
                     return;
                 }
             }
         }
    }];
}

- (void) validBarcodeFound:(Barcode *)barcode{
NSLog(@"Found Barcode");

[self stopRunning];
[self.foundBarcodes addObject:barcode];
//[self showBarcodeAlert:barcode];
NSString *alertMessage = @"";
alertMessage = [alertMessage stringByAppendingString:[barcode getBarcodeType]];
NSLog([barcode getBarcodeData]);
NSLog(alertMessage);

NSLog([NSString stringWithFormat:@"%@", barcode.getBarcodeData]);
if ([barcode.getBarcodeType isEqualToString:@"org.iso.QRCode"])
{
 if ([[NSString stringWithFormat:@"%lu",(unsigned long)[self.foundBarcodes count]-1] length] > 0){

    NSString *input = [barcode getBarcodeData];
    [NSString stringWithFormat:@"%lu",(unsigned long)[self.foundBarcodes count]-1];
    NSLog(input);


    if ([input length] >= 13)
    {
        input = [input substringToIndex:12];
    }
    loadUrl = [[@"http://www.mywebsite.co.uk/" stringByAppendingString:input] stringByAppendingString:@"?utm_source=iphone"];
    NSLog(loadUrl);        
    dispatch_sync(dispatch_get_main_queue(), ^{
        [self performSegueWithIdentifier:@"toWebView" sender:self];
    });
}


}

1 ответ

Решение

Так что мне удалось это исправить, но я не могу дать реального понимания того, почему это работает.

Вместо асинхронной загрузки URL во время viewDidLoadЯ передал URL с помощью segue и загрузил его из ViewController, содержащего WebView.

В дополнение к этому, объявления переменных были заключены в фигурные скобки {} и @synthesize mWebView был удален. Я понятия не имею, почему это вызывает проблемы, поэтому любые возможные объяснения будут оценены

Другие вопросы по тегам