обнаружить длительное нажатие на кнопку "Назад" UINavigationItem

Я хочу добавить функциональность к моим задним кнопкам через мое приложение на основе UINavigationController, где длительное нажатие кнопки "Назад" появится в root. Однако я не могу понять, куда прикрепить распознаватель жестов. Должен ли я подкласс UINavigationBar и попытаться определить, находится ли длительное нажатие в области левой кнопки?

Я слышал о людях, добавляющих подобную функциональность раньше. У кого-нибудь есть идеи?

4 ответов


Я считаю, что UIGestureRecognizers могут быть добавлены только в UIViews и подклассы UIViews.

http://developer.apple.com/library/ios/#documentation/uikit/reference/UIView_Class/UIView/UIView.html

кнопка "назад" - это UIBarButtonItem, который спускается из NSObject. Таким образом, вы не сможете прикрепить распознаватель жестов к стандартной кнопке "Назад" с помощью

UILongPressGestureRecognizer *longPressGesture =
            [[[UILongPressGestureRecognizer alloc]
              initWithTarget:self action:@selector(longPress:)] autorelease];

[self.navigationItem.backBarButtonItem addGestureRecognizer:longPressGesture];

однако вы можете добавить пользовательское представление в UIBarButtonItem. Ля пользовательский вид может быть также легко UIView, UIButton, UILabel и т. д.

пример:

UIView *myTransparentGestureView = [[UIView alloc] initWithFrame:CGRectMake(0,0,40,30)];
[myTransparentGestureView addGestureRecognizer:longPressGesture];
[self.navigationItem.backBarButtonItem setCustomView:myTransparentGestureView];
// Or you could set it like this
// self.navigationItem.backBarButtonItem.customView = myTransparentGestureView;
[myTransparentGestureView release];

вы должны быть осторожны, однако, так как установка свойств на backBarButtonItem применяется к следующему представлению, которое вы нажимаете. Поэтому, если у вас есть представление A, которое толкает к представлению B, и вы хотите, чтобы жест был распознан при нажатии назад в представлении B. Вы должны настроить его в представлении A.


Я знаю, этот вопрос старый, но я придумал решение. Вместо того, чтобы пытаться добавить распознаватель жестов к самой кнопке (что было бы идеально), я добавил его в self.navigationController.navigationBar и затем в методе действия используйте locationInView чтобы увидеть, если я за кнопку "Назад". Я не был полностью уверен в том, как точно идентифицировать кнопку "назад", поэтому я неуклюже просто хватаю первое подвидение с координатой x меньше некоторого произвольного значения, но это кажется многообещающим. Если у кого-то лучше способ определить рамку кнопки "Назад", дайте мне знать.

- (void)longPress:(UILongPressGestureRecognizer *)sender 
{
    if (sender.state == UIGestureRecognizerStateEnded)
    {
        // set a default rectangle in case we don't find the back button for some reason

        CGRect rect = CGRectMake(0, 0, 100, 40);

        // iterate through the subviews looking for something that looks like it might be the right location to be the back button

        for (UIView *subview in self.navigationController.navigationBar.subviews)
        {
            if (subview.frame.origin.x < 30) 
            {
                rect = subview.frame;
                break;
            }
        }

        // ok, let's get the point of the long press

        CGPoint longPressPoint = [sender locationInView:self.navigationController.navigationBar];

        // if the long press point in the rectangle then do whatever

        if (CGRectContainsPoint(rect, longPressPoint))
            [self doWhatever];
    }
}

- (void)addLongPressGesture
{
    if (NSClassFromString(@"UILongPressGestureRecognizer"))
    {
        UILongPressGestureRecognizer *longPress = [[UILongPressGestureRecognizer alloc] initWithTarget:self action:@selector(longPress:)];
        [self.navigationController.navigationBar addGestureRecognizer:longPress];
        [longPress release];
    }
}

Я пошел немного другим путем, решил, что разделю его. Приведенные выше ответы прекрасны, но на самом деле, если долгое нажатие находится в ведущей 1/3 навигационной панели, этого достаточно для меня:

- (void)longPress:(UILongPressGestureRecognizer *)gr
{
    NSLog(@"longPress:");
    UINavigationBar *navBar = [self navigationBar];
    CGFloat height = navBar.bounds.size.height;
    CGPoint pt = [gr locationOfTouch:0 inView:navBar];
    //NSLog(@"PT=%@ height=%f", NSStringFromCGPoint(pt), height);
    if(CGRectContainsPoint(CGRectMake(0,0,100,height), pt)) {
        [self popToViewController:self.viewControllers[0] animated:YES];
    }
}

вот мое решение:

в appDelegate ("владелец" панели навигации в моем приложении), в applicationDidFinishLaunchingWithoptions:

получите вид панели навигации и добавьте распознаватель жестов ко всему виду:

// Get the nav bar view
UINavigationBar *myNavBar = nil;
for (UIView *view in [self.window.rootViewController.view subviews]) {
    if ([view isKindOfClass:[UINavigationBar class]]) {
        NSLog(@"Found Nav Bar!!!");
        myNavBar = (UINavigationBar *)view;
    }
}

UILongPressGestureRecognizer *longPress = [[UILongPressGestureRecognizer alloc] initWithTarget:self
                                                                                        action:@selector(backButtonLongPress:)];
[myNavBar addGestureRecognizer:longPress];
NSLog(@"Gesture Recognizer Added.");

затем в appDelegate, в - (void) backButtonLongPress: (id) отправитель

Проверьте, происходит ли жест в рамке кнопки "Назад":

if ([sender state] == UIGestureRecognizerStateBegan) {

    // Get the nav bar view
    UINavigationBar *myNavBar = nil;
    for (UIView *view in [self.window.rootViewController.view subviews]) {
        if ([view isKindOfClass:[UINavigationBar class]]) {
            NSLog(@"Found Nav Bar!!!");
            myNavBar = (UINavigationBar *)view;
        }
    }

    // Get the back button view
    UIView *backButtonView = nil;
    for (UIView *view in [myNavBar subviews]) {
        if ([[[view class] description] isEqualToString:@"UINavigationItemButtonView"]) {
            backButtonView = view;
            NSLog(@"Found It: %@", backButtonView);
            NSLog(@"Back Button View Frame: %f, %f; %f, %f", backButtonView.frame.origin.x, backButtonView.frame.origin.y, backButtonView.frame.size.width, backButtonView.frame.size.height);
        }
    }

    CGPoint longPressPoint = [sender locationInView:myNavBar];
    NSLog(@"Touch is in back button: %@", CGRectContainsPoint(backButtonView.frame, longPressPoint) ? @"YES" : @"NO");
    if (CGRectContainsPoint(backButtonView.frame, longPressPoint)) {
        // Place your action here
    }

    // Do nothing if outside the back button frame

}