обнаружить длительное нажатие на кнопку "Назад" UINavigationItem
Я хочу добавить функциональность к моим задним кнопкам через мое приложение на основе UINavigationController, где длительное нажатие кнопки "Назад" появится в root. Однако я не могу понять, куда прикрепить распознаватель жестов. Должен ли я подкласс UINavigationBar и попытаться определить, находится ли длительное нажатие в области левой кнопки?
Я слышал о людях, добавляющих подобную функциональность раньше. У кого-нибудь есть идеи?
4 ответов
Я считаю, что UIGestureRecognizers могут быть добавлены только в UIViews и подклассы UIViews.
кнопка "назад" - это 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
}