Наблюдение за жестами мультитач в UITableView - PullRequest
19 голосов
/ 05 января 2010

Я хочу реализовать пинч-вход / выход поверх UITableView, я рассмотрел несколько методов, включая этот:

Аналогичный вопрос

Но хотя я могу создать объект UIViewTouch и наложить его на свой UITableView, события прокрутки не передаются на мой UITableView, я все же могу выбирать ячейки, и они правильно реагируют, инициируя переход к новому объекту ViewController. Но я не могу прокрутить UITableView, несмотря на прохождение событий touchesBegan, touchesMoved и touchesEnded.

Ответы [ 2 ]

41 голосов
/ 05 января 2010

Кажется, это классическая проблема. В моем случае я хотел перехватить некоторые события через UIWebView, которые нельзя разделить на подклассы и т. Д. И т. Д.

Я обнаружил, что лучший способ сделать это - перехватить события с помощью UIWindow:

EventInterceptWindow.h

@protocol EventInterceptWindowDelegate
- (BOOL)interceptEvent:(UIEvent *)event; // return YES if event handled
@end


@interface EventInterceptWindow : UIWindow {
    // It would appear that using the variable name 'delegate' in any UI Kit
    // subclass is a really bad idea because it can occlude the same name in a
    // superclass and silently break things like autorotation.
    id <EventInterceptWindowDelegate> eventInterceptDelegate;
}

@property(nonatomic, assign)
    id <EventInterceptWindowDelegate> eventInterceptDelegate;

@end

EventInterceptWindow.m:

#import "EventInterceptWindow.h"

@implementation EventInterceptWindow

@synthesize eventInterceptDelegate;

- (void)sendEvent:(UIEvent *)event {
    if ([eventInterceptDelegate interceptEvent:event] == NO)
        [super sendEvent:event];
}

@end

Создайте этот класс, измените класс вашего UIWindow в вашем MainWindow.xib на EventInterceptWindow, затем где-то установите eventInterceptDelegate в контроллер представления, который вы хотите перехватывать события. Пример, который перехватывает двойное касание:

- (BOOL)interceptEvent:(UIEvent *)event {
    NSSet *touches = [event allTouches];
    UITouch *oneTouch = [touches anyObject];
    UIView *touchView = [oneTouch view];
    //  NSLog(@"tap count = %d", [oneTouch tapCount]);
    // check for taps on the web view which really end up being dispatched to
    // a scroll view
    if (touchView && [touchView isDescendantOfView:webView]
            && touches && oneTouch.phase == UITouchPhaseBegan) {
        if ([oneTouch tapCount] == 2) {
            [self toggleScreenDecorations];
            return YES;
        }
    }   
    return NO;
}

Информация, связанная с данной здесь: http://iphoneincubator.com/blog/windows-views/360idev-iphone-developers-conference-presentation

5 голосов
/ 17 февраля 2011

Нимрод написал:

где-то установить eventInterceptDelegate в контроллер представления, который вы хотите перехватывать события

Я не сразу понял это утверждение. Для тех, у кого была такая же проблема, как у меня, я сделал это, добавив следующий код в мой подкласс UIView, который должен обнаруживать касания.

- (void) viewDidAppear:(BOOL)animated
{
    [super viewDidAppear:animated];

    // Register to receive touch events
    MyApplicationAppDelegate *appDelegate = (MyApplicationAppDelegate *) [[UIApplication sharedApplication] delegate];
    EventInterceptWindow *window = (EventInterceptWindow *) appDelegate.window;
    window.eventInterceptDelegate = self;
}


- (void) viewWillDisappear:(BOOL) animated
{
    // Deregister from receiving touch events
    MyApplicationAppDelegate *appDelegate = (MyApplicationAppDelegate *) [[UIApplication sharedApplication] delegate];
    EventInterceptWindow *window = (EventInterceptWindow *) appDelegate.window;
    window.eventInterceptDelegate = nil;

    [super viewWillDisappear:animated];
}


- (BOOL) interceptEvent:(UIEvent *) event
{
    NSLog(@"interceptEvent is being called...");
    return NO;
}


Эта версия interceptEvent: представляет собой простую реализацию обнаружения масштабирования. NB. Некоторая часть кода была взята с начала разработки iPhone 3 компанией Apress.

CGFloat initialDistance;

- (BOOL) interceptEvent:(UIEvent *) event
{
    NSSet *touches = [event allTouches];

    // Give up if user wasn't using two fingers
    if([touches count] != 2) return NO;

    UITouchPhase phase = ((UITouch *) [touches anyObject]).phase;
    CGPoint firstPoint = [[[touches allObjects] objectAtIndex:0] locationInView:self.view];
    CGPoint secondPoint = [[[touches allObjects] objectAtIndex:1] locationInView:self.view];

    CGFloat deltaX = secondPoint.x - firstPoint.x;
    CGFloat deltaY = secondPoint.y - firstPoint.y;
    CGFloat distance = sqrt(deltaX*deltaX + deltaY*deltaY);

    if(phase == UITouchPhaseBegan)
    {
        initialDistance = distance;
    }
    else if(phase == UITouchPhaseMoved)
    {
        CGFloat currentDistance = distance;
        if(initialDistance == 0) initialDistance = currentDistance;
        else if(currentDistance - initialDistance > kMinimumPinchDelta) NSLog(@"Zoom in");
        else if(initialDistance - currentDistance > kMinimumPinchDelta) NSLog(@"Zoom out");
    }
    else if(phase == UITouchPhaseEnded)
    {
        initialDistance = 0;
    }

    return YES;
}


Изменить: Хотя этот код работал на 100% отлично в симуляторе iPhone, когда я запустил его на устройстве iPhone, я обнаружил странные ошибки, связанные с прокруткой таблицы. Если это также произойдет с вами, то заставьте метод interceptEvent: вернуть NO во всех случаях. Это означает, что суперкласс также будет обрабатывать событие касания, но, к счастью, это не нарушило мой код.

...