Sprite Kit - Determine vector of swipe gesture to flick sprite

房东的猫 提交于 2019-12-02 19:48:44

Here's an example of how to detect a swipe gesture:

First, define instance variables to store the starting location and time .

    CGPoint start;
    NSTimeInterval startTime;

In touchesBegan, save the location/time of a touch event.

-(void)touchesBegan:(NSSet *)touches withEvent:(UIEvent *)event {

    /* Avoid multi-touch gestures (optional) */
    if ([touches count] > 1) {
        return;
    }
    UITouch *touch = [touches anyObject];
    CGPoint location = [touch locationInNode:self];
    // Save start location and time
    start = location;
    startTime = touch.timestamp;
}

Define parameters of the swipe gesture. Adjust these accordingly.

#define kMinDistance    25
#define kMinDuration    0.1
#define kMinSpeed       100
#define kMaxSpeed       500

In touchesEnded, determine if the user's gesture was a swipe by comparing the differences between starting and ending locations and time stamps.

- (void) touchesEnded:(NSSet *)touches withEvent:(UIEvent *)event {

    UITouch *touch = [touches anyObject];
    CGPoint location = [touch locationInNode:self];
    // Determine distance from the starting point
    CGFloat dx = location.x - start.x;
    CGFloat dy = location.y - start.y;
    CGFloat magnitude = sqrt(dx*dx+dy*dy);
    if (magnitude >= kMinDistance) {
        // Determine time difference from start of the gesture
        CGFloat dt = touch.timestamp - startTime;
        if (dt > kMinDuration) {
            // Determine gesture speed in points/sec
            CGFloat speed = magnitude / dt;
            if (speed >= kMinSpeed && speed <= kMaxSpeed) {
                // Calculate normalized direction of the swipe
                dx = dx / magnitude;
                dy = dy / magnitude;
                NSLog(@"Swipe detected with speed = %g and direction (%g, %g)",speed, dx, dy);
            }
        }
    }
}

There is another way to do it, you can add a pan gesture and then get the velocity from it:

First add pan gesture in your view:

UIPanGestureRecognizer *gestureRecognizer = [[UIPanGestureRecognizer alloc] initWithTarget:self action:@selector(handlePanFrom:)];
    [self.view addGestureRecognizer:gestureRecognizer];

Then handle the gesture:

- (void)handlePanFrom:(UIPanGestureRecognizer *)recognizer
{
    if (recognizer.state == UIGestureRecognizerStateBegan) {
        CGPoint location = [recognizer locationInView:recognizer.view];
        if ([_object containsPoint:location]){
            self.movingObject = YES;
            <.. object start moving ..>
        }

    } else if (recognizer.state == UIGestureRecognizerStateChanged) {

        if (!self.movingObject)
            return;

        CGPoint translation = [recognizer translationInView:recognizer.view];
        object.position = CGPointMake(object.position.x + translation.x, object.position.y + translation.y);

        [recognizer setTranslation:CGPointZero inView:recognizer.view];

    } else if (recognizer.state == UIGestureRecognizerStateEnded) {

        if (!self.movingObject)
            return;

        self.movingObject = NO;
        float force = 1.0f;
        CGPoint gestureVelocity = [recognizer velocityInView:recognizer.view];
        CGVector impulse = CGVectorMake(gestureVelocity.x * force, gestureVelocity.y * force);

        <.. Move object with that impulse using an animation..>
    }
}

In touchesBegan save the touch location as a CGPoint you can access throughout your app.

In touchesEnded calculate the distance and direction of your initial touch (touchesBegan) and ending touch (touchesEnded). Then apply the appropriate Impulse.

To refrain from double hitting, add a bool canHit that you set to NO when the impulse is applied and set back to YES when you are ready to hit again. Before applying the impulse, make sure canHit is set to YES.

易学教程内所有资源均来自网络或用户发布的内容,如有违反法律规定的内容欢迎反馈
该文章没有解决你所遇到的问题?点击提问,说说你的问题,让更多的人一起探讨吧!