Периодический сбой при переходе от пользовательского контроллера представления

У меня есть UIViewController, который представляет / отклоняет другой контроллер представления с настраиваемой анимацией, используя шаблон transitioningDelegate, представленный в iOS 7.

В своих файлах журнала Crashlytics я заметил, что этот раздел кода иногда дает сбой в производственной среде, и я не мог понять, почему. В моем тестировании он ни разу не дал сбоев и, похоже, затронул только около 10% наших пользователей. iOS 7 и iOS 8 затронуты, а также iPad, iPhone и iPod.

Вот трассировка стека от Crashlytics:

Crashed: com.apple.main-thread
EXC_BAD_ACCESS KERN_INVALID_ADDRESS at 0x00014000

Thread : Crashed: com.apple.main-thread
0  libobjc.A.dylib                0x303c3f76 objc_msgSend + 21
1  UIKit                          0x25ed0ac5 -[UIViewController _customAnimatorForDismissedController:] + 64
2  UIKit                          0x25ed0927 -[UIViewController _dismissViewControllerWithTransition:from:completion:] + 538
3  UIKit                          0x25e7e4b3 -[UIViewController dismissViewControllerWithTransition:completion:] + 822
4  UIKit                          0x25e7e4b3 -[UIViewController dismissViewControllerWithTransition:completion:] + 822
5  UIKit                          0x25e7e127 -[UIViewController dismissViewControllerAnimated:completion:] + 222
6  Skin Creator                   0x000f38e7 -[SEUSBodySideChooserViewController imageEditViewController:didFinishWithImage:] (SEUSBodySideChooserViewController.m:251)
7  Skin Creator                   0x000ae2e9 -[SEUSImageEditViewController sendFinishedDelegateMethod] (SEUSImageEditViewController.m:409)
8  Skin Creator                   0x000d585f -[SEUSCloseMenuController circleMenu:didSelectItemAtIndex:] (SEUSCloseMenuController.m:69)
9  Skin Creator                   0x000ceb75 -[SEUSCircleMenu menuButtonPressed:] (SEUSCircleMenu.m:203)
10 UIKit                          0x25dec427 -[UIApplication sendAction:to:from:forEvent:] + 70
11 UIKit                          0x25dec3c9 -[UIControl sendAction:to:forEvent:] + 44
12 UIKit                          0x25dd6fcd -[UIControl _sendActionsForEvents:withEvent:] + 584
13 UIKit                          0x25debdf9 -[UIControl touchesEnded:withEvent:] + 584
14 UIKit                          0x25db0b47 _UIGestureRecognizerUpdate + 10158
15 UIKit                          0x25de5afd -[UIWindow _sendGesturesForEvent:] + 784
16 UIKit                          0x25de53cd -[UIWindow sendEvent:] + 520
17 UIKit                          0x25dbbb5d -[UIApplication sendEvent:] + 196
18 UIKit                          0x2602f4e3 _UIApplicationHandleEventFromQueueEvent + 13874
19 UIKit                          0x25dba59f _UIApplicationHandleEventQueue + 1294
20 CoreFoundation                 0x228dd5e7 __CFRUNLOOP_IS_CALLING_OUT_TO_A_SOURCE0_PERFORM_FUNCTION__ + 14
21 CoreFoundation                 0x228dc9fb __CFRunLoopDoSources0 + 222
22 CoreFoundation                 0x228db079 __CFRunLoopRun + 768
23 CoreFoundation                 0x22828981 CFRunLoopRunSpecific + 476
24 CoreFoundation                 0x22828793 CFRunLoopRunInMode + 106
25 GraphicsServices               0x29ba9051 GSEventRunModal + 136
26 UIKit                          0x25e1a981 UIApplicationMain + 1440
27 Skin Creator                   0x000ead6f main (main.m:16)

Просто чтобы вы могли понять это немного больше [SEUSBodySideChooserViewController imageEditViewController: didFinishWithImage:] - это обратный вызов делегата, в котором я отклоняю контроллер представления.

Код, который я использую для представления контроллера представления

_toPixelEditTransitioningDelegate = [[SEUSFadeInTransitioningDelegate alloc] init];
_imageEditViewController.transitioningDelegate = _toPixelEditTransitioningDelegate;
[self presentViewController:_imageEditViewController animated:YES completion:nil];

Код, который я использую для закрытия контроллера представления

[viewController dismissViewControllerAnimated:YES completion:nil];

Мой переходный делегат

- (id<UIViewControllerAnimatedTransitioning>)animationControllerForPresentedController:(UIViewController *)presented presentingController:(UIViewController *)presenting sourceController:(UIViewController *)source
{
    SEUSFadeInTransitioning *transitioning = [[SEUSFadeInTransitioning alloc] init];
    return transitioning;
}

- (id<UIViewControllerAnimatedTransitioning>)animationControllerForDismissedController:(UIViewController *)dismissed
{
    SEUSFadeInTransitioning *transitioning = [[SEUSFadeInTransitioning alloc] init];
    transitioning.reverse = YES;
    return transitioning;
}

Объект перехода, который реализует протокол UIViewControllerAnimatedTransitioning

- (instancetype)init
{
    if (self = [super init])
    {
        _reverse = NO;
        _duration = 0.25f;
    }
    return self;
}

- (void)animateToPixelEdit:(id<UIViewControllerContextTransitioning>)transitionContext
{
    UIViewController* overviewVC = [transitionContext viewControllerForKey:UITransitionContextFromViewControllerKey];
    UIViewController* pixelVC = [transitionContext viewControllerForKey:UITransitionContextToViewControllerKey];
    [[transitionContext containerView] addSubview:overviewVC.view];
    [[transitionContext containerView] addSubview:pixelVC.view];

    pixelVC.view.alpha = 0;

    [UIView animateWithDuration:[self transitionDuration:transitionContext] animations:^
    {
        pixelVC.view.alpha = 1;
    }
    completion:^(BOOL finished)
    {
        [transitionContext completeTransition:![transitionContext transitionWasCancelled]];
    }];
}

- (void)animateToOverview:(id<UIViewControllerContextTransitioning>)transitionContext
{
    UIViewController* pixelVC = [transitionContext viewControllerForKey:UITransitionContextFromViewControllerKey];
    UIViewController* overviewVC = [transitionContext viewControllerForKey:UITransitionContextToViewControllerKey];
    [[transitionContext containerView] addSubview:pixelVC.view];
    [[transitionContext containerView] addSubview:overviewVC.view];

    overviewVC.view.alpha = 0;

    [UIView animateWithDuration:[self transitionDuration:transitionContext] animations:^
    {
        overviewVC.view.alpha = 1;
    }
    completion:^(BOOL finished)
    {
        [transitionContext completeTransition:![transitionContext transitionWasCancelled]];
    }];
}

- (NSTimeInterval)transitionDuration:(id<UIViewControllerContextTransitioning>)transitionContext
{
    return _duration;
}

- (void)animateTransition:(id<UIViewControllerContextTransitioning>)transitionContext
{
    if (self.reverse == NO)
    {
        [self animateToPixelEdit:transitionContext];
    }
    else
    {
        [self animateToOverview:transitionContext];
    }
}

Кто-нибудь заметил что-нибудь, что могло пойти не так? Это действительно сложно устранить, так как я не смог воспроизвести сбой.


person Joel Bell    schedule 13.11.2014    source источник
comment
Возможно ли, что imageEditVC дважды вызывает свой метод делегата didFinish, если пользователь нажимает несколько раз?   -  person Acey    schedule 13.11.2014
comment
О чувак. Если все так просто, я не уверен, должен ли я чувствовать себя глупым или взволнованным. Гунна, попробуй прямо сейчас!   -  person Joel Bell    schedule 13.11.2014
comment
Пока не удалось воспроизвести это, я закодирую для него проверку, просто вставлю и вытолкну. А пока кто-нибудь видит что-нибудь еще, что потенциально могло бы его вызвать?   -  person Joel Bell    schedule 13.11.2014
comment
Хорошо, еще одно предположение ... toPixelEditTransitioningDelegate сильно где-то сохранен?   -  person Acey    schedule 13.11.2014
comment
Думаю, я разобрался, благодаря вашим комментариям. Похоже, быстрое нажатие могло бы представить imageEditVC дважды, а затем, когда пользователь попытался бы закрыть его после того, как он был представлен сам по себе, это вызвало бы этот сбой. Спасибо за вашу помощь!   -  person Joel Bell    schedule 13.11.2014
comment
Не стесняйтесь отвечать, если хотите, я приму.   -  person Joel Bell    schedule 13.11.2014


Ответы (2)


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

person Acey    schedule 13.11.2014
comment
Это правда. В моем случае переключение между вкладками контроллера сегмента и анимация контроллера представления при касаниях вызывали сбой. Простым исправлением было бы отключить взаимодействие с элементом управления пользовательским интерфейсом или просто UIApplication.shared.beginIgnoringInteractionEvents () и повторно включить, когда анимация завершена UIApplication.shared.endIgnoringInteractionEvents () - person Raj D; 11.01.2021

Я исправил ошибку, похожую на эту, она оказалась удаленной ссылкой transitioningDelegate (_toPixelEditTransitioningDelegate в примере вопроса).

Если вы видите EXEC_BAD_ACCESS, вы можете включить NSZombies, что поможет вам отследить объект с плохой ссылкой.

Надеюсь, это окажется полезным для кого-то еще!

person Oliver Atkinson    schedule 25.07.2016