Search code examples
xamarinxamarin.formspinchzoom

Xamarin Forms pinch and pan together


I have implemented both pan and pinch individually, and it works fine. I'm now trying to use pinch and pan together and I'm seeing some issues. Here's my code:

XAML:

<AbsoluteLayout x:Name="PinchZoomContainer">
  <controls:NavBar x:Name="NavBar" ShowPrevNext="true" ShowMenu="false" IsModal="true" />
  <controls:PanContainer  x:Name="PinchToZoomContainer">
    <Image x:Name="ImageMain" />
  </controls:PanContainer>
</AbsoluteLayout>

Pinch/Pan Gesture Add's:

var panGesture = new PanGestureRecognizer();
panGesture.PanUpdated += OnPanUpdated;
GestureRecognizers.Add(panGesture);

var pinchGesture = new PinchGestureRecognizer();
pinchGesture.PinchUpdated += OnPinchUpdated;
GestureRecognizers.Add(pinchGesture);

Pan Method:

void OnPanUpdated(object sender, PanUpdatedEventArgs e)
{
    switch (e.StatusType)
    {
        case GestureStatus.Started:
            startX = e.TotalX;
            startY = e.TotalY;
            Content.AnchorX = 0;
            Content.AnchorY = 0;

            break;
        case GestureStatus.Running:
            // Translate and ensure we don't pan beyond the wrapped user interface element bounds.
            Content.TranslationX = Math.Max(Math.Min(0, x + e.TotalX), -Math.Abs(Content.Width - App.ScreenWidth));
            Content.TranslationY = Math.Max(Math.Min(0, y + e.TotalY), -Math.Abs(Content.Height - App.ScreenHeight));
            break;

        case GestureStatus.Completed:
            // Store the translation applied during the pan
            x = Content.TranslationX;
            y = Content.TranslationY;
            break;
    }
}

Pinch Method:

void OnPinchUpdated(object sender, PinchGestureUpdatedEventArgs e)
{
    if (e.Status == GestureStatus.Started)
    {
        // Store the current scale factor applied to the wrapped user interface element,
        // and zero the components for the center point of the translate transform.
        startScale = Content.Scale;
        //ImageMain.AnchorX = 0;
        //ImageMain.AnchorY = 0;
    }
    if (e.Status == GestureStatus.Running)
    {
        // Calculate the scale factor to be applied.
        currentScale += (e.Scale - 1) * startScale;
        currentScale = Math.Max(1, currentScale);
        currentScale = Math.Min(currentScale, 2.5);
        // The ScaleOrigin is in relative coordinates to the wrapped user interface element,
        // so get the X pixel coordinate.
        double renderedX = Content.X + xOffset;
        double deltaX = renderedX / Width;
        double deltaWidth = Width / (Content.Width * startScale);
        double originX = (e.ScaleOrigin.X - deltaX) * deltaWidth;

        // The ScaleOrigin is in relative coordinates to the wrapped user interface element,
        // so get the Y pixel coordinate.
        double renderedY = Content.Y + yOffset;
        double deltaY = renderedY / Height;
        double deltaHeight = Height / (Content.Height * startScale);
        double originY = (e.ScaleOrigin.Y - deltaY) * deltaHeight;

        // Calculate the transformed element pixel coordinates.
        double targetX = xOffset - (originX * Content.Width) * (currentScale - startScale);
        double targetY = yOffset - (originY * Content.Height) * (currentScale - startScale);

        // Apply translation based on the change in origin.
        Content.TranslationX = targetX.Clamp(-Content.Width * (currentScale - 1), 0);
        Content.TranslationY = targetY.Clamp(-Content.Height * (currentScale - 1), 0);

        // Apply scale factor
        Content.Scale = currentScale;
    }
    if (e.Status == GestureStatus.Completed)
    {
        // Store the translation delta's of the wrapped user interface element.
        xOffset = Content.TranslationX;
        yOffset = Content.TranslationY;
    }
}

If I turn off either gesture and only use the other then the functionality works perfectly. The issue arises when I add the pan AND pinch gestures. What seems to be happening is this:

1) The pan actually seems to be working as expected 2) When you pan on the image initially, let's say, move the image to Y-center and X-center, and then you try to zoom, the image gets set back to it's initial state. Then, when you pan, it moves you back to where you were before you tried to zoom (which is why I say the pan is working fine).

From what I'm understanding from my debugging is that when you zoom it's not taking into consideration the position you are currently at. So when you pan first, and then zoom, it doesn't zoom on the position you're at but the beginning point of the image. Then when you try to pan from there, the pan method still remembers where you were, and it moves you back to where you were before you tried to zoom.

Hoping some insight on this. Obviously, there's an issue with my pinch method. I just think (obviously can't figure out) I need to add logic into it that takes into consideration where you're currently at.


Solution

  • Went with a completely different method of handling this. For anyone who is having issues, this works 100%.

    OnPanUpdated

    void OnPanUpdated(object sender, PanUpdatedEventArgs e)
    {
        var s = (ContentView)sender;
    
        // do not allow pan if the image is in its intial size
        if (currentScale == 1)
            return;
    
        switch (e.StatusType)
        {
            case GestureStatus.Running:
                double xTrans = xOffset + e.TotalX, yTrans = yOffset + e.TotalY;
                // do not allow verical scorlling unless the image size is bigger than the screen
                s.Content.TranslateTo(xTrans, yTrans, 0, Easing.Linear);
                break;
    
            case GestureStatus.Completed:
                // Store the translation applied during the pan
                xOffset = s.Content.TranslationX;
                yOffset = s.Content.TranslationY;
    
                // center the image if the width of the image is smaller than the screen width
                if (originalWidth * currentScale < ScreenWidth && ScreenWidth > ScreenHeight)
                    xOffset = (ScreenWidth - originalWidth * currentScale) / 2 - s.Content.X;
                else
                    xOffset = System.Math.Max(System.Math.Min(0, xOffset), -System.Math.Abs(originalWidth * currentScale - ScreenWidth));
    
                // center the image if the height of the image is smaller than the screen height
                if (originalHeight * currentScale < ScreenHeight && ScreenHeight > ScreenWidth)
                    yOffset = (ScreenHeight - originalHeight * currentScale) / 2 - s.Content.Y;
                else
                    //yOffset = System.Math.Max(System.Math.Min((originalHeight - (ScreenHeight)) / 2, yOffset), -System.Math.Abs((originalHeight * currentScale - ScreenHeight - (originalHeight - ScreenHeight) / 2)) + (NavBar.Height + App.StatusBarHeight));
                    yOffset = System.Math.Max(System.Math.Min((originalHeight - (ScreenHeight)) / 2, yOffset), -System.Math.Abs((originalHeight * currentScale - ScreenHeight - (originalHeight - ScreenHeight) / 2)));
    
                // bounce the image back to inside the bounds
                s.Content.TranslateTo(xOffset, yOffset, 500, Easing.BounceOut);
                break;
        }
    }
    

    OnPinchUpdated

    void OnPinchUpdated(object sender, PinchGestureUpdatedEventArgs e)
    {
        var s = (ContentView)sender;
    
        if (e.Status == GestureStatus.Started)
        {
            // Store the current scale factor applied to the wrapped user interface element,
            // and zero the components for the center point of the translate transform.
            startScale = s.Content.Scale;
    
            s.Content.AnchorX = 0;
            s.Content.AnchorY = 0;
        }
        if (e.Status == GestureStatus.Running)
        {
    
            // Calculate the scale factor to be applied.
            currentScale += (e.Scale - 1) * startScale;
            currentScale = System.Math.Max(1, currentScale);
            currentScale = System.Math.Min(currentScale, 5);
    
            //scaleLabel.Text = "Scale: " + currentScale.ToString ();
    
            // The ScaleOrigin is in relative coordinates to the wrapped user interface element,
            // so get the X pixel coordinate.
            double renderedX = s.Content.X + xOffset;
            double deltaX = renderedX / App.ScreenWidth;
            double deltaWidth = App.ScreenWidth / (s.Content.Width * startScale);
            double originX = (e.ScaleOrigin.X - deltaX) * deltaWidth;
    
            // The ScaleOrigin is in relative coordinates to the wrapped user interface element,
            // so get the Y pixel coordinate.
            double renderedY = s.Content.Y + yOffset;
    
            double deltaY = renderedY / App.ScreenHeight;
            double deltaHeight = App.ScreenHeight / (s.Content.Height * startScale);
            double originY = (e.ScaleOrigin.Y - deltaY) * deltaHeight;
    
            // Calculate the transformed element pixel coordinates.
            double targetX = xOffset - (originX * s.Content.Width) * (currentScale - startScale);
            double targetY = yOffset - (originY * s.Content.Height) * (currentScale - startScale);
    
            // Apply translation based on the change in origin.
            var transX = targetX.Clamp(-s.Content.Width * (currentScale - 1), 0);
            var transY = targetY.Clamp(-s.Content.Height * (currentScale - 1), 0);
    
    
            s.Content.TranslateTo(transX, transY, 0, Easing.Linear);
            // Apply scale factor.
            s.Content.Scale = currentScale;
        }
        if (e.Status == GestureStatus.Completed)
        {
            // Store the translation applied during the pan
            xOffset = s.Content.TranslationX;
            yOffset = s.Content.TranslationY;
    
            // center the image if the width of the image is smaller than the screen width
            if (originalWidth * currentScale < ScreenWidth && ScreenWidth > ScreenHeight)
                xOffset = (ScreenWidth - originalWidth * currentScale) / 2 - s.Content.X;
            else
                xOffset = System.Math.Max(System.Math.Min(0, xOffset), -System.Math.Abs(originalWidth * currentScale - ScreenWidth));
    
            // center the image if the height of the image is smaller than the screen height
            if (originalHeight * currentScale < ScreenHeight && ScreenHeight > ScreenWidth)
                yOffset = (ScreenHeight - originalHeight * currentScale) / 2 - s.Content.Y;
            else
                yOffset = System.Math.Max(System.Math.Min((originalHeight - ScreenHeight) / 2, yOffset), -System.Math.Abs(originalHeight * currentScale - ScreenHeight - (originalHeight - ScreenHeight) / 2));
    
            // bounce the image back to inside the bounds
            s.Content.TranslateTo(xOffset, yOffset, 500, Easing.BounceOut);
        }
    }
    

    OnSizeAllocated (most of this you probably dont need, but some you do. consider ScreenWidth, ScreenHeight, yOffset, xOffset, currentScale)

    protected override void OnSizeAllocated(double width, double height)
    {            
        base.OnSizeAllocated(width, height); //must be called
    
        if (width != -1 &&  (ScreenWidth != width || ScreenHeight != height))
        {
            ResetLayout(width, height);
    
            originalWidth = initialLoad ?
                ImageWidth >= 960 ?
                   App.ScreenWidth > 320 
                        ? 768 
                        : 320 
                    :  ImageWidth / 3
                : imageContainer.Content.Width / imageContainer.Content.Scale;
    
            var normalizedHeight = ImageWidth >= 960 ?
                    App.ScreenWidth > 320 ? ImageHeight / (ImageWidth / 768) 
                    : ImageHeight / (ImageWidth / 320) 
                : ImageHeight / 3;
    
            originalHeight = initialLoad ? 
                normalizedHeight : (imageContainer.Content.Height / imageContainer.Content.Scale);
    
            ScreenWidth = width;
            ScreenHeight = height;
    
            xOffset = imageContainer.TranslationX;
            yOffset = imageContainer.TranslationY;
    
            currentScale = imageContainer.Scale;
    
            if (initialLoad)
                initialLoad = false;
        }
    }
    

    Layout (XAML in C#)

    ImageMain = new Image
    {
        HorizontalOptions = LayoutOptions.CenterAndExpand,
        VerticalOptions = LayoutOptions.CenterAndExpand,
        Aspect = Aspect.AspectFill,
        Source = ImageMainSource
    };
    
    imageContainer = new ContentView
    {
        Content = ImageMain,
        BackgroundColor = Xamarin.Forms.Color.Black,
        WidthRequest = App.ScreenWidth - 250
    };
    
    var panGesture = new PanGestureRecognizer();
    panGesture.PanUpdated += OnPanUpdated;
    imageContainer.GestureRecognizers.Add(panGesture);
    
    var pinchGesture = new PinchGestureRecognizer();
    pinchGesture.PinchUpdated += OnPinchUpdated;
    imageContainer.GestureRecognizers.Add(pinchGesture);
    
    double smallImageHeight = ImageHeight / (ImageWidth / 320);
    
    absoluteLayout = new AbsoluteLayout
    {
        HeightRequest = App.ScreenHeight,
        BackgroundColor = Xamarin.Forms.Color.Black,
    };
    
    AbsoluteLayout.SetLayoutFlags(imageContainer, AbsoluteLayoutFlags.All);
    AbsoluteLayout.SetLayoutBounds(imageContainer, new Rectangle(0f, 0f, AbsoluteLayout.AutoSize, AbsoluteLayout.AutoSize));
    absoluteLayout.Children.Add(imageContainer, new Rectangle(0, 0, 1, 1), AbsoluteLayoutFlags.All);
    Content = absoluteLayout;