Center content of UIScrollView when smaller
Asked Answered
N

28

156

I have a UIImageView inside a UIScrollView which I use for zooming and scrolling. If the image / content of the scroll view is bigger than the scroll view, everything works fine. However, when the image becomes smaller than the scroll view, it sticks to the top left corner of the scroll view. I would like to keep it centered, like the Photos app.

Any ideas or examples about keeping the content of the UIScrollView centered when it's smaller?

I am working with iPhone 3.0.

The following code almost works. The image returns to the top left corner if I pinch it after reaching the minimum zoom level.

- (void)loadView {
    [super loadView];

    // set up main scroll view
    imageScrollView = [[UIScrollView alloc] initWithFrame:[[self view] bounds]];
    [imageScrollView setBackgroundColor:[UIColor blackColor]];
    [imageScrollView setDelegate:self];
    [imageScrollView setBouncesZoom:YES];
    [[self view] addSubview:imageScrollView];

    UIImageView *imageView = [[UIImageView alloc] initWithImage:[UIImage imageNamed:@"WeCanDoIt.png"]];
    [imageView setTag:ZOOM_VIEW_TAG];
    [imageScrollView setContentSize:[imageView frame].size];
    [imageScrollView addSubview:imageView];

    CGSize imageSize = imageView.image.size;
    [imageView release];

    CGSize maxSize = imageScrollView.frame.size;
    CGFloat widthRatio = maxSize.width / imageSize.width;
    CGFloat heightRatio = maxSize.height / imageSize.height;
    CGFloat initialZoom = (widthRatio > heightRatio) ? heightRatio : widthRatio;

    [imageScrollView setMinimumZoomScale:initialZoom];
    [imageScrollView setZoomScale:1];

    float topInset = (maxSize.height - imageSize.height) / 2.0;
    float sideInset = (maxSize.width - imageSize.width) / 2.0;
    if (topInset < 0.0) topInset = 0.0;
    if (sideInset < 0.0) sideInset = 0.0;
    [imageScrollView setContentInset:UIEdgeInsetsMake(topInset, sideInset, -topInset, -sideInset)];
}

- (UIView *)viewForZoomingInScrollView:(UIScrollView *)scrollView {
    return [imageScrollView viewWithTag:ZOOM_VIEW_TAG];
}

/************************************** NOTE **************************************/
/* The following delegate method works around a known bug in zoomToRect:animated: */
/* In the next release after 3.0 this workaround will no longer be necessary      */
/**********************************************************************************/
- (void)scrollViewDidEndZooming:(UIScrollView *)scrollView withView:(UIView *)view atScale:(float)scale {
    [scrollView setZoomScale:scale+0.01 animated:NO];
    [scrollView setZoomScale:scale animated:NO];
    // END Bug workaround

    CGSize maxSize = imageScrollView.frame.size;
    CGSize viewSize = view.frame.size;
    float topInset = (maxSize.height - viewSize.height) / 2.0;
    float sideInset = (maxSize.width - viewSize.width) / 2.0;
    if (topInset < 0.0) topInset = 0.0;
    if (sideInset < 0.0) sideInset = 0.0;
    [imageScrollView setContentInset:UIEdgeInsetsMake(topInset, sideInset, -topInset, -sideInset)];
}
Necropsy answered 22/8, 2009 at 16:55 Comment(2)
Did you ever solve this problem completely? I'm struggling with the same issue.Owner
Attention: Use the initialZoom value to calculate the inset if it is NOT one (no zooming). E.g. use these lines: float topInset = (maxSize.height - imageSize.height * initialZoom) / 2.0; float sideInset = (maxSize.width - imageSize.width * initialZoom) / 2.0; and finally set the initial zoom value [imageScrollView setZoomScale: initialZoom];Vermiculate
N
21

Currently I'm subclassing UIScrollView and overriding setContentOffset: to adjust the offset based on contentSize. It works both with pinch and programatic zooming.

@implementation HPCenteringScrollView

- (void)setContentOffset:(CGPoint)contentOffset
{
    const CGSize contentSize = self.contentSize;
    const CGSize scrollViewSize = self.bounds.size;

    if (contentSize.width < scrollViewSize.width)
    {
        contentOffset.x = -(scrollViewSize.width - contentSize.width) / 2.0;
    }

    if (contentSize.height < scrollViewSize.height)
    {
        contentOffset.y = -(scrollViewSize.height - contentSize.height) / 2.0;
    }

    [super setContentOffset:contentOffset];
}

@end

In addition to being short and sweet, this code produces a much smoother zoom than @Erdemus solution. You can see it in action in the RMGallery demo.

Necropsy answered 21/3, 2014 at 12:6 Comment(2)
You don't need to subclass UIScrollView to implement this method. Apple allow you to see scroll and zoom events in the delegate methods. Specifically: - (void)scrollViewDidZoom:(UIScrollView *)scrollView;Detoxicate
Best solution I’ve seen yet (work even when using constraints).Karolekarolina
L
236

I've got very simple solution! All you need is to update the center of your subview (imageview) while zooming in the ScrollViewDelegate. If zoomed image is smaller than scrollview then adjust subview.center else center is (0,0).

- (void)scrollViewDidZoom:(UIScrollView *)scrollView 
{
    UIView *subView = [scrollView.subviews objectAtIndex:0];

    CGFloat offsetX = MAX((scrollView.bounds.size.width - scrollView.contentSize.width) * 0.5, 0.0);
    CGFloat offsetY = MAX((scrollView.bounds.size.height - scrollView.contentSize.height) * 0.5, 0.0);

    subView.center = CGPointMake(scrollView.contentSize.width * 0.5 + offsetX, 
                                 scrollView.contentSize.height * 0.5 + offsetY);
}
Lapstrake answered 22/5, 2010 at 7:25 Comment(7)
This works somewhat better than the contentInset adjustment one, at least, for the usages I have tried. The contentInset adjustment option is quite jerky, but this is extremely smooth.Pantelegraph
For me this solution is more jerky than using Liam's NYOBetterZoom. Maybe it depend on image size etc. The moral; use the solution that best suits your needsRochdale
Of all the "solutions" I've tried for this problem, this is the only one I can get to work properly and it's sooo much simpler to implement than some of the others I've seen out there. Thanks a lot for this time saver!Surefire
to simplify a bit: CGFloat offsetX = MAX((scrollView.bounds.size.width - scrollView.contentSize.width) * 0.5, 0.0);Meso
This adjustment helped me out when the scrollview had insets: CGFloat offsetX = MAX((scrollView.bounds.size.width - scrollView.contentInset.left - scrollView.contentInset.right - scrollView.contentSize.width) * 0.5, 0.0); CGFloat offsetY = MAX((scrollView.bounds.size.height - scrollView.contentInset.top - scrollView.contentInset.bottom - scrollView.contentSize.height) * 0.5, 0.0);Ferino
I noticed that this, like many other centering techniques, seems to suffer from problems when using zoomToRect:. Using the contentInset approach works better, if you happen to need that functionality. See petersteinberger.com/blog/2013/how-to-center-uiscrollview for more details.Toponym
This actually also helps keeping the image in the right spot when zooming in, which was a bigger problem in my particular caseBacklash
O
76

@EvelynCordner's answer was the one that worked best in my app. A lot less code than the other options too.

Here's the Swift version if anyone needs it:

func scrollViewDidZoom(_ scrollView: UIScrollView) {
    let offsetX = max((scrollView.bounds.width - scrollView.contentSize.width) * 0.5, 0)
    let offsetY = max((scrollView.bounds.height - scrollView.contentSize.height) * 0.5, 0)
    scrollView.contentInset = UIEdgeInsets(top: offsetY, left: offsetX, bottom: 0, right: 0)
}
Odoriferous answered 23/3, 2016 at 5:57 Comment(6)
This one works great! The view even animated properly after shrinking.Foothill
I put this code in a func and also called it in viewDidLayoutSubviews to make sure it was properly set up initially.Foothill
Good call @CarterMedlin helped me so much with my Swift 3 initial load.Detroit
This seems to work, but I can't understand why, since it always seems to calculate the same insets: the scroll view's bounds are fixed, as is the content size.Lupien
contentSize changes when zooming scrollView size is only fixedLombok
This solution has the least movement when zooming into areas close to the image border and (mostly) keeps the initial zoom point centered between the fingersDissipate
D
26

Okay, I've been fighting this for the past two days on and off and having finally come to a pretty reliable (so far...) solution I thought I should share it and save others some pain. :) If you do find a problem with this solution please shout!

I've basically gone through what everyone else has: searching StackOverflow, the Apple Developer Forums, looked at the code for three20, ScrollingMadness, ScrollTestSuite, etc. I've tried enlarging the UIImageView frame, playing with the UIScrollView's offset and/or insets from the ViewController, etc. but nothing worked great (as everyone else has found out too).

After sleeping on it, I tried a couple of alternative angles:

  1. Subclassing the UIImageView so it alters it's own size dynamically - this didn't work well at all.
  2. Subclassing the UIScrollView so it alters it's own contentOffset dynamically - this is the one that seems to be a winner for me.

With this subclassing UIScrollView method I'm overriding the contentOffset mutator so it isn't setting {0,0} when the image is scaled smaller than the viewport - instead it's setting the offset such that the image will be kept centred in the viewport. So far, it always seems to work. I've checked it with wide, tall, tiny & large images and doesn't have the "works but pinch at minimum zoom breaks it" issue.

I've uploaded an example project to github that uses this solution, you can find it here: http://github.com/nyoron/NYOBetterZoom

Downpour answered 14/4, 2010 at 14:28 Comment(7)
For those interested - I've updated the above linked project so it's a bit less reliant on the ViewController doing 'the right thing', the custom UIScrollView itself takes care of more of the detail.Downpour
Liam, you rock. I'm saying this as an author of ScrollingMadness. BTW on iPad in 3.2+ setting contentInset in scrollViewDidZoom (a new 3.2+ delegate method) Just Works.Loosing
Very neat piece of code. I've been scratching my head with this for a while. Added the project to handyiphonecode.comHazem
This is great. Thank you. It wasn't compensating for my UIStatusBar being hidden though, so I changed the line anOffset.y = -(scrollViewSize.height - zoomViewSize.height) / 2.0 to anOffset.y = (-(scrollViewSize.height - zoomViewSize.height) / 2.0) + 10;Redstart
The solution given bi Erdemus is much more simple in my opinion.Selfseeker
@LiamJones you don't have a license in your GitHub repo? That ways it's All Rights Reserved. Would you open it up to MIT or something similar?Backlash
apparently @LiamJones is not on SO anymore, with just this answer. Still i say, you've done and shared great work with that scrollView - that still helps people after 7 years, and works flawlessly. Applauding.Kwabena
S
24

For a solution better suited for scroll views that use autolayout, use content insets of the scroll view rather than updating the frames of your scroll view's subviews.

- (void)scrollViewDidZoom:(UIScrollView *)scrollView
{
    CGFloat offsetX = MAX((scrollView.bounds.size.width - scrollView.contentSize.width) * 0.5, 0.0);
    CGFloat offsetY = MAX((scrollView.bounds.size.height - scrollView.contentSize.height) * 0.5, 0.0);

    self.scrollView.contentInset = UIEdgeInsetsMake(offsetY, offsetX, 0.f, 0.f);
}
Sites answered 15/7, 2015 at 4:45 Comment(3)
This worked for me, however if the image is smaller to begin with somehow this code (when called directly) didn't update the scrollview. What I needed todo was first put the UIView that is inside the UIScrollview to the same center (view.center = scrollview.center;) and then in the scrollViewDidZoom set the view.frame.origin x and y to 0 again.Squishy
Works well for me as well, but I did a dispatch_async to the main queue in viewWillAppear where I call scrollViewDidZoom: on my main scroll view. This makes the view appear with centered images.Euhemerus
Place a call to this code in viewDidLayoutSubviews to make sure it is properly set up when the view shows the first time.Foothill
P
23

This code should work on most versions of iOS (and has been tested to work on 3.1 upwards).

It's based on the Apple WWDC code for the photoscoller.

Add the below to your subclass of UIScrollView, and replace tileContainerView with the view containing your image or tiles:

- (void)layoutSubviews {
    [super layoutSubviews];

    // center the image as it becomes smaller than the size of the screen
    CGSize boundsSize = self.bounds.size;
    CGRect frameToCenter = tileContainerView.frame;

    // center horizontally
    if (frameToCenter.size.width < boundsSize.width)
        frameToCenter.origin.x = (boundsSize.width - frameToCenter.size.width) / 2;
    else
        frameToCenter.origin.x = 0;

    // center vertically
    if (frameToCenter.size.height < boundsSize.height)
        frameToCenter.origin.y = (boundsSize.height - frameToCenter.size.height) / 2;
    else
        frameToCenter.origin.y = 0;

    tileContainerView.frame = frameToCenter;
}
Plugugly answered 13/8, 2010 at 16:56 Comment(3)
this should be the accepted answer, cause it is simpler. Thanks. You saved my life!!!!!! :DPecten
After removing all the iOS 3.2+ API calls, the centering logic only seems to work on devices with iOS 3.2+ and not 3.1.3 (jumpy, flickery, gets random offset). I've compared the outputs of frame origin and size between the 3.1.3 and 3.2+ and even though they do match, for some reason, the child view still gets positioned incorrectly. Very odd. Only Liam Jone's answer worked out for me.Promptitude
Both @Lapstrake and @Plugugly solutions work, but the UIScrollView subclass approach seems preferable: it is invoked when the view is first displayed + continuously while zooming is in progress (whereas scrollViewDidZoom is only invoked once per scroll and after the fact)Orlina
N
21

Currently I'm subclassing UIScrollView and overriding setContentOffset: to adjust the offset based on contentSize. It works both with pinch and programatic zooming.

@implementation HPCenteringScrollView

- (void)setContentOffset:(CGPoint)contentOffset
{
    const CGSize contentSize = self.contentSize;
    const CGSize scrollViewSize = self.bounds.size;

    if (contentSize.width < scrollViewSize.width)
    {
        contentOffset.x = -(scrollViewSize.width - contentSize.width) / 2.0;
    }

    if (contentSize.height < scrollViewSize.height)
    {
        contentOffset.y = -(scrollViewSize.height - contentSize.height) / 2.0;
    }

    [super setContentOffset:contentOffset];
}

@end

In addition to being short and sweet, this code produces a much smoother zoom than @Erdemus solution. You can see it in action in the RMGallery demo.

Necropsy answered 21/3, 2014 at 12:6 Comment(2)
You don't need to subclass UIScrollView to implement this method. Apple allow you to see scroll and zoom events in the delegate methods. Specifically: - (void)scrollViewDidZoom:(UIScrollView *)scrollView;Detoxicate
Best solution I’ve seen yet (work even when using constraints).Karolekarolina
A
12

I've spent a day fighting with this issue, and ended up implementing the scrollViewDidEndZooming:withView:atScale: as follows:

- (void)scrollViewDidEndZooming:(UIScrollView *)scrollView withView:(UIView *)view atScale:(float)scale {
    CGFloat screenWidth = [[UIScreen mainScreen] bounds].size.width;
    CGFloat screenHeight = [[UIScreen mainScreen] bounds].size.height;
    CGFloat viewWidth = view.frame.size.width;
    CGFloat viewHeight = view.frame.size.height;

    CGFloat x = 0;
    CGFloat y = 0;

    if(viewWidth < screenWidth) {
        x = screenWidth / 2;
    }
    if(viewHeight < screenHeight) {
        y = screenHeight / 2 ;
    }

    self.scrollView.contentInset = UIEdgeInsetsMake(y, x, y, x);
}

This ensures that when the image is smaller than the screen, there's still adequate space around it so you can position it to the exact place you want.

(assuming that your UIScrollView contains an UIImageView to hold the image)

Essentially, what this does is check whether your image view's width / height is smaller that the screen's width / height, and if so, create an inset of half the screen's width / height (you could probably make this larger if you want the image to go out of the screen bounds).

Note that since this is a UIScrollViewDelegate method, don't forget to add it to your view controller's declaration, so to avoid getting a build issue.

Auramine answered 25/1, 2011 at 11:31 Comment(0)
F
9

If contentInset is not needed for anything else, it can be used to center scrollview's content.

class ContentCenteringScrollView: UIScrollView {

    override var bounds: CGRect {
        didSet { updateContentInset() }
    }

    override var contentSize: CGSize {
        didSet { updateContentInset() }
    }

    private func updateContentInset() {
        var top = CGFloat(0)
        var left = CGFloat(0)
        if contentSize.width < bounds.width {
            left = (bounds.width - contentSize.width) / 2
        }
        if contentSize.height < bounds.height {
            top = (bounds.height - contentSize.height) / 2
        }
        contentInset = UIEdgeInsets(top: top, left: left, bottom: top, right: left)
    }
}

Advantage if this approach is that you can still use contentLayoutGuide to place content inside scrollview

scrollView.addSubview(imageView)
imageView.translatesAutoresizingMaskIntoConstraints = false
NSLayoutConstraint.activate([
    imageView.leadingAnchor.constraint(equalTo: scrollView.contentLayoutGuide.leadingAnchor),
    imageView.trailingAnchor.constraint(equalTo: scrollView.contentLayoutGuide.trailingAnchor),
    imageView.topAnchor.constraint(equalTo: scrollView.contentLayoutGuide.topAnchor),
    imageView.bottomAnchor.constraint(equalTo: scrollView.contentLayoutGuide.bottomAnchor)
])

or just drag and drop the content in Xcode's Interface Builder.

Flournoy answered 14/4, 2020 at 11:36 Comment(0)
O
8

Apple has released the 2010 WWDC session videos to all members of the iphone developer program. One of the topics discussed is how they created the photos app!!! They build a very similar app step by step and have made all the code available for free.

It does not use private api either. I can't put any of the code here because of the non disclosure agreement, but here is a link to the sample code download. You will probably need to login to gain access.

http://connect.apple.com/cgi-bin/WebObjects/MemberSite.woa/wa/getSoftware?code=y&source=x&bundleID=20645

And, here is a link to the iTunes WWDC page:

http://insideapple.apple.com/redir/cbx-cgi.do?v=2&la=en&lc=&a=kGSol9sgPHP%2BtlWtLp%2BEP%2FnxnZarjWJglPBZRHd3oDbACudP51JNGS8KlsFgxZto9X%2BTsnqSbeUSWX0doe%2Fzv%2FN5XV55%2FomsyfRgFBysOnIVggO%2Fn2p%2BiweDK%2F%2FmsIXj

Owner answered 20/6, 2010 at 19:51 Comment(4)
The example in question is MyImagePicker and it, hilariously, exhibits this same problem.Conservancy
I should have been more clear. The example in question is actually "PhotoScroller" not "MyImagePicker". You are right that "MyImagePicker" doesn't function properly. But, "PhotoScroller" does. Try it.Owner
Do you remember perhaps the title of the WWDC video where they discuss the Photoscroller?Awesome
It's called "Designing Apps With Scroll Views".Owner
W
2

The way I've done this is to add an extra view into the hierarchy:

UIScrollView -> UIView -> UIImageView

Give your UIView the same aspect ratio as your UIScrollView, and centre your UIImageView into that.

Wigeon answered 24/8, 2009 at 19:5 Comment(2)
Thanks, hatfinch. Will try this as well. Can you post sample code or change my sample code to show how you construct the view hierarchy?Necropsy
This sort of works. Except for the fact that because the UIView is the size of the UIScrollView, if the image is smaller (i.e. landscape instead of portrait) you can scroll part of the image off the screen. The photos app does not allow this and looks much nicer.Owner
R
2

Ok, this solution is working for me. I have a subclass of UIScrollView with a reference to the UIImageView it is displaying. Whenever the UIScrollView zooms, the contentSize property is adjusted. It is in the setter that I scale the UIImageView appropriately and also adjust its center position.

-(void) setContentSize:(CGSize) size{
CGSize lSelfSize = self.frame.size;
CGPoint mid;
if(self.zoomScale >= self.minimumZoomScale){
    CGSize lImageSize = cachedImageView.initialSize;
    float newHeight = lImageSize.height * self.zoomScale;

    if (newHeight < lSelfSize.height ) {
        newHeight = lSelfSize.height;
    }
    size.height = newHeight;

    float newWidth = lImageSize.width * self.zoomScale;
    if (newWidth < lSelfSize.width ) {
        newWidth = lSelfSize.width;
    }
    size.width = newWidth;
    mid = CGPointMake(size.width/2, size.height/2);

}
else {
    mid = CGPointMake(lSelfSize.width/2, lSelfSize.height/2);
}

cachedImageView.center = mid;
[super  setContentSize:size];
[self printLocations];
NSLog(@"zoom %f setting size %f x %f",self.zoomScale,size.width,size.height);
}

Evertime I set the image on the UIScrollView I resize it. The UIScrollView in the scrollview is also a custom class I created.

-(void) resetSize{
    if (!scrollView){//scroll view is view containing imageview
        return;
    }

    CGSize lSize = scrollView.frame.size;

    CGSize lSelfSize = self.image.size; 
    float lWidth = lSize.width/lSelfSize.width;
    float lHeight = lSize.height/lSelfSize.height;

    // choose minimum scale so image width fits screen
    float factor  = (lWidth<lHeight)?lWidth:lHeight;

    initialSize.height = lSelfSize.height  * factor;
    initialSize.width = lSelfSize.width  * factor;

    [scrollView setContentSize:lSize];
    [scrollView setContentOffset:CGPointZero];
    scrollView.userInteractionEnabled = YES;
}

With these two methods I am able to have a view that behaves just like the photos app.

Regeneration answered 16/3, 2010 at 5:8 Comment(2)
This seems to do the trick, and it's a fairly simple solution. Some zooming transitions are not perfect, but I believe it can be fixed. I will experiment some more.Necropsy
The solution was clear before the update. Now it's a bit confusing. Can you clarify?Necropsy
G
2

Just the approved answer in swift, but without subclassing using the delegate

func centerScrollViewContents(scrollView: UIScrollView) {
    let contentSize = scrollView.contentSize
    let scrollViewSize = scrollView.frame.size;
    var contentOffset = scrollView.contentOffset;

    if (contentSize.width < scrollViewSize.width) {
        contentOffset.x = -(scrollViewSize.width - contentSize.width) / 2.0
    }

    if (contentSize.height < scrollViewSize.height) {
        contentOffset.y = -(scrollViewSize.height - contentSize.height) / 2.0
    }

    scrollView.setContentOffset(contentOffset, animated: false)
}

// UIScrollViewDelegate    
func scrollViewDidZoom(scrollView: UIScrollView) {
    centerScrollViewContents(scrollView)
}
Grieg answered 12/2, 2016 at 10:25 Comment(0)
G
2

I know some answers above are right, but I just want to give my answer with some explanation, the comments will make you understand why we do like this.

When I load the scrollView for the first time, I write the following code to make it center, please notice we set contentOffset first, then contentInset

    scrollView.maximumZoomScale = 8
    scrollView.minimumZoomScale = 1

    // set vContent frame
    vContent.frame = CGRect(x: 0,
                            y: 0  ,
                            width: vContentWidth,
                            height: vContentWidth)
    // set scrollView.contentSize
    scrollView.contentSize = vContent.frame.size

    //on the X direction, if contentSize.width > scrollView.bounds.with, move scrollView from 0 to offsetX to make it center(using `scrollView.contentOffset`)
    // if not, don't need to set offset, but we need to set contentInset to make it center.(using `scrollView.contentInset`)
    // so does the Y direction.
    let offsetX = max((scrollView.contentSize.width - scrollView.bounds.width) * 0.5, 0)
    let offsetY = max((scrollView.contentSize.height - scrollView.bounds.height) * 0.5, 0)
    scrollView.contentOffset = CGPoint(x: offsetX, y: offsetY)

    let topX = max((scrollView.bounds.width - scrollView.contentSize.width) * 0.5, 0)
    let topY = max((scrollView.bounds.height - scrollView.contentSize.height) * 0.5, 0)
    scrollView.contentInset = UIEdgeInsets(top: topY, left: topX, bottom: 0, right: 0)

Then, when I pinch vContent, I write the following code to make it center.

func scrollViewDidZoom(_ scrollView: UIScrollView) {
    //we just need to ensure that the content is in the center when the contentSize is less than scrollView.size.
    let topX = max((scrollView.bounds.width - scrollView.contentSize.width) * 0.5, 0)
    let topY = max((scrollView.bounds.height - scrollView.contentSize.height) * 0.5, 0)
    scrollView.contentInset = UIEdgeInsets(top: topY, left: topX, bottom: 0, right: 0)
}
Guzman answered 1/7, 2019 at 8:25 Comment(0)
O
1

You could watch the contentSize property of the UIScrollView (using key-value observing or similar), and automatically adjust the contentInset whenever the contentSize changes to be less than the size of the scroll view.

Ongoing answered 22/8, 2009 at 17:17 Comment(8)
Will try. Is this be possible to do with the UIScrollViewDelegate methods instead of observing contentSize?Necropsy
Most likely; you'd use - (void)scrollViewDidEndZooming:(UIScrollView *)scrollView withView:(UIView *)view atScale:(float)scale (described at developer.apple.com/iphone/library/documentation/UIKit/…), but I'd prefer observing contentSize because otherwise you wind up discarding the new zoom scale and finding it from the view anyway. (Unless you can pull off some awesome math based on the relative sizes of your views.)Ongoing
Thanks, Tim. scrollViewDidEndZooming is what I'm using. See code in question (I've added it after your first reply). It almost works. The only problem that remains if that if I pinch the image after minimumZoomScale has been reached, it returns to the top left corner.Necropsy
Do you mean that it works (centers the image) for pinching from any scale other than the minimum zoom scale, and only breaks if you try to pinch it again once you're at the minimum scale? Or does it not work for pinching to the minimum scale at all?Ongoing
The first one. It centers the image for pinching from any scale other than the minimum zoom scale, and it breaks if you try to pinch it again once you're at the minimum scale. Also, when it reaches the minimum zoom scale the first time, the content is briefly animated like coming from the top, which cause a brief weird effect. This happens on the 3.0 simulator at least.Necropsy
Huh. Maybe you can use a BOOL ivar to remember when you're already at the minimum zoom scale, and not apply your custom inset logic (or indeed do much of anything) when it receives a pinch at the minimum scale? And as for the animation effect, that could either be a Simulator quirk or an actual issue - can you try it on a device and see which it is?Ongoing
Tried to do nothing if minimumZoomScale has been reached, but it breaks anyway. Whatever code makes it break it's not coming from scrollViewDidEndZooming. Will try on the device to confirm it's not a simulator quirk.Necropsy
In that case, hgpc, I'm kind of stuck for ideas :) Sorry!Ongoing
T
1

One elegant way to center the content of UISCrollView is this.

Add one observer to the contentSize of your UIScrollView, so this method will be called everytime the content change...

[myScrollView addObserver:delegate 
               forKeyPath:@"contentSize"
                  options:(NSKeyValueObservingOptionNew) 
                  context:NULL];

Now on your observer method:

- (void)observeValueForKeyPath:(NSString *)keyPath   ofObject:(id)object   change:(NSDictionary *)change   context:(void *)context { 

    // Correct Object Class.
    UIScrollView *pointer = object;

    // Calculate Center.
    CGFloat topCorrect = ([pointer bounds].size.height - [pointer viewWithTag:100].bounds.size.height * [pointer zoomScale])  / 2.0 ;
            topCorrect = ( topCorrect < 0.0 ? 0.0 : topCorrect );

    topCorrect = topCorrect - (  pointer.frame.origin.y - imageGallery.frame.origin.y );

    // Apply Correct Center.
    pointer.center = CGPointMake(pointer.center.x,
                                 pointer.center.y + topCorrect ); }
  • You should change the [pointer viewWithTag:100]. Replace by your content view UIView.

    • Also change imageGallery pointing to your window size.

This will correct the center of the content everytime his size change.

NOTE: The only way this content don't works very well is with standard zoom functionality of the UIScrollView.

Throughput answered 3/11, 2009 at 17:9 Comment(1)
Couldn't make this to work. It seems you're centering the scrollView and not its content. Why does the window size matter? Shouldn't the code correct the x position as well?Necropsy
P
1

You'll find that the solution posted by Erdemus does work, but… There are some cases where the scrollViewDidZoom method does not get invoked & your image is stuck to the top left corner. A simple solution is to explicitly invoke the method when you initially display an image, like this:

[self scrollViewDidZoom: scrollView];

In many cases, you may be invoking this method twice, but this is a cleaner solution than some of the other answers in this topic.

Preterition answered 7/9, 2011 at 1:13 Comment(0)
F
1

This is my solution to that problem which works pretty fine for any kind of view inside a scrollview.

-(void)scrollViewDidZoom:(__unused UIScrollView *)scrollView 
    {
    CGFloat top;
    CGFloat left;
    CGFloat bottom;
    CGFloat right;

    if (_scrollView.contentSize.width < scrollView.bounds.size.width) {
        DDLogInfo(@"contentSize %@",NSStringFromCGSize(_scrollView.contentSize));

        CGFloat width = (_scrollView.bounds.size.width-_scrollView.contentSize.width)/2.0;

        left = width;
        right = width;


    }else {
        left = kInset;
        right = kInset;
    }

    if (_scrollView.contentSize.height < scrollView.bounds.size.height) {

        CGFloat height = (_scrollView.bounds.size.height-_scrollView.contentSize.height)/2.0;

        top = height;
        bottom = height;

    }else {
        top = kInset;
        right = kInset;
    }

    _scrollView.contentInset = UIEdgeInsetsMake(top, left, bottom, right);



  if ([self.tiledScrollViewDelegate respondsToSelector:@selector(tiledScrollViewDidZoom:)])
  {
        [self.tiledScrollViewDelegate tiledScrollViewDidZoom:self];
  }
}
Fibrilla answered 7/1, 2013 at 0:13 Comment(0)
U
1

There are a plenty of solutions here, but I'd risk putting here my own. It's good for two reasons: it doesn't mess zooming experience, as would do updating image view frame in progress, and also it respects original scroll view insets (say, defined in xib or storyboard for graceful handling of semi-transparent toolbars etc).

First, define a small helper:

CGSize CGSizeWithAspectFit(CGSize containerSize, CGSize contentSize) {
    CGFloat containerAspect = containerSize.width / containerSize.height,
            contentAspect = contentSize.width / contentSize.height;

    CGFloat scale = containerAspect > contentAspect
                    ? containerSize.height / contentSize.height
                    : containerSize.width / contentSize.width;

    return CGSizeMake(contentSize.width * scale, contentSize.height * scale);
}

To retain original insets, defined field:

UIEdgeInsets originalScrollViewInsets;

And somewhere in viewDidLoad fill it:

originalScrollViewInsets = self.scrollView.contentInset;

To place UIImageView into UIScrollView (assuming UIImage itself is in loadedImage var):

CGSize containerSize = self.scrollView.bounds.size;
containerSize.height -= originalScrollViewInsets.top + originalScrollViewInsets.bottom;
containerSize.width -= originalScrollViewInsets.left + originalScrollViewInsets.right;

CGSize contentSize = CGSizeWithAspectFit(containerSize, loadedImage.size);

UIImageView *imageView = [[UIImageView alloc] initWithFrame:(CGRect) { CGPointZero, contentSize }];
imageView.autoresizingMask = UIViewAutoresizingNone;
imageView.contentMode = UIViewContentModeScaleAspectFit;
imageView.image = loadedImage;

[self.scrollView addSubview:imageView];
self.scrollView.contentSize = contentSize;

[self centerImageViewInScrollView];

scrollViewDidZoom: from UIScrollViewDelegate for that scroll view:

- (void)scrollViewDidZoom:(UIScrollView *)scrollView {
    if (scrollView == self.scrollView) {
        [self centerImageViewInScrollView];
    }
}

An finally, centering itself:

- (void)centerImageViewInScrollView {
    CGFloat excessiveWidth = MAX(0.0, self.scrollView.bounds.size.width - self.scrollView.contentSize.width),
            excessiveHeight = MAX(0.0, self.scrollView.bounds.size.height - self.scrollView.contentSize.height),
            insetX = excessiveWidth / 2.0,
            insetY = excessiveHeight / 2.0;

    self.scrollView.contentInset = UIEdgeInsetsMake(
            MAX(insetY, originalScrollViewInsets.top),
            MAX(insetX, originalScrollViewInsets.left),
            MAX(insetY, originalScrollViewInsets.bottom),
            MAX(insetX, originalScrollViewInsets.right)
    );
}

I didn't test orientation change yet (i.e. proper reaction for resizing UIScrollView itself), but fix for that should be relatively easy.

Underthrust answered 20/3, 2013 at 19:40 Comment(0)
O
1

Apple's Photo Scroller Example does exactly what you are looking for. Put this in your UIScrollView Subclass and change _zoomView to be your UIImageView.

-(void)layoutSubviews{
  [super layoutSubviews];
  // center the zoom view as it becomes smaller than the size of the screen
  CGSize boundsSize = self.bounds.size;
  CGRect frameToCenter = self.imageView.frame;
  // center horizontally
  if (frameToCenter.size.width < boundsSize.width){
     frameToCenter.origin.x = (boundsSize.width - frameToCenter.size.width) / 2;
  }else{
    frameToCenter.origin.x = 0;
  }
  // center vertically
  if (frameToCenter.size.height < boundsSize.height){
     frameToCenter.origin.y = (boundsSize.height - frameToCenter.size.height) / 2;
  }else{
    frameToCenter.origin.y = 0;
  }
  self.imageView.frame = frameToCenter; 
}

Apple's Photo Scroller Sample Code

Outstrip answered 1/5, 2013 at 17:35 Comment(0)
U
1

To make the animation flow nicely, set

self.scrollview.bouncesZoom = NO;

and use this function (finding the center using the method at this answer)

- (void)scrollViewDidEndZooming:(UIScrollView *)scrollView withView:(UIView *)view atScale:(CGFloat)scale {
    [UIView animateWithDuration:0.2 animations:^{
        float offsetX = MAX((scrollView.bounds.size.width-scrollView.contentSize.width)/2, 0);
        float offsetY = MAX((scrollView.bounds.size.height-scrollView.contentSize.height)/2, 0);
        self.imageCoverView.center = CGPointMake(scrollView.contentSize.width*0.5+offsetX, scrollView.contentSize.height*0.5+offsetY);
    }];
}

This creates the bouncing effect but doesn't involve any sudden movements beforehand.

Unsettled answered 18/3, 2015 at 3:40 Comment(0)
A
1

In case your inner imageView has initial specific width(eg 300) and you just want to center its width only on zoom smaller than its initial width this might help you also.

 func scrollViewDidZoom(scrollView: UIScrollView){
    if imageView.frame.size.width < 300{
        imageView.center.x = self.view.frame.width/2
    }
  }
Aldo answered 26/9, 2016 at 14:6 Comment(0)
O
0

Okay, I think I've found a pretty good solution to this problem. The trick is to constantly readjust the imageView's frame. I find this works much better than constantly adjusting the contentInsets or contentOffSets. I had to add a bit of extra code to accommodate both portrait and landscape images.

Here's the code:

- (void) scrollViewDidEndZooming:(UIScrollView *)scrollView withView:(UIView *)view atScale:(float)scale {

CGSize screenSize = [[self view] bounds].size;

if (myScrollView.zoomScale <= initialZoom +0.01) //This resolves a problem with the code not working correctly when zooming all the way out.
{
    imageView.frame = [[self view] bounds];
    [myScrollView setZoomScale:myScrollView.zoomScale +0.01];
}

if (myScrollView.zoomScale > initialZoom)
{
    if (CGImageGetWidth(temporaryImage.CGImage) > CGImageGetHeight(temporaryImage.CGImage)) //If the image is wider than tall, do the following...
    {
        if (screenSize.height >= CGImageGetHeight(temporaryImage.CGImage) * [myScrollView zoomScale]) //If the height of the screen is greater than the zoomed height of the image do the following...
        {
            imageView.frame = CGRectMake(0, 0, 320*(myScrollView.zoomScale), 368);
        }
        if (screenSize.height < CGImageGetHeight(temporaryImage.CGImage) * [myScrollView zoomScale]) //If the height of the screen is less than the zoomed height of the image do the following...
        {
            imageView.frame = CGRectMake(0, 0, 320*(myScrollView.zoomScale), CGImageGetHeight(temporaryImage.CGImage) * [myScrollView zoomScale]);
        }
    }
    if (CGImageGetWidth(temporaryImage.CGImage) < CGImageGetHeight(temporaryImage.CGImage)) //If the image is taller than wide, do the following...
    {
        CGFloat portraitHeight;
        if (CGImageGetHeight(temporaryImage.CGImage) * [myScrollView zoomScale] < 368)
        { portraitHeight = 368;}
        else {portraitHeight = CGImageGetHeight(temporaryImage.CGImage) * [myScrollView zoomScale];}

        if (screenSize.width >= CGImageGetWidth(temporaryImage.CGImage) * [myScrollView zoomScale]) //If the width of the screen is greater than the zoomed width of the image do the following...
        {
            imageView.frame = CGRectMake(0, 0, 320, portraitHeight);
        }
        if (screenSize.width < CGImageGetWidth (temporaryImage.CGImage) * [myScrollView zoomScale]) //If the width of the screen is less than the zoomed width of the image do the following...
        {
            imageView.frame = CGRectMake(0, 0, CGImageGetWidth(temporaryImage.CGImage) * [myScrollView zoomScale], portraitHeight);
        }
    }
    [myScrollView setZoomScale:myScrollView.zoomScale -0.01];
}
Owner answered 4/10, 2009 at 1:50 Comment(0)
O
0

Here's the current way I'm making this work. It's better but still not perfect. Try setting:

 myScrollView.bouncesZoom = YES; 

to fix the problem with the view not centering when at minZoomScale.

- (void)scrollViewDidScroll:(UIScrollView *)scrollView
{
CGSize screenSize = [[self view] bounds].size;//[[UIScreen mainScreen] bounds].size;//
CGSize photoSize = [yourImage size];
CGFloat topInset = (screenSize.height - photoSize.height * [myScrollView zoomScale]) / 2.0;
CGFloat sideInset = (screenSize.width - photoSize.width * [myScrollView zoomScale]) / 2.0;

if (topInset < 0.0)
{ topInset = 0.0; }
if (sideInset < 0.0)
{ sideInset = 0.0; } 
[myScrollView setContentInset:UIEdgeInsetsMake(topInset, sideInset, -topInset, -sideInset)];
ApplicationDelegate *appDelegate = (ApplicationDelegate *)[[UIApplication sharedApplication] delegate];

CGFloat scrollViewHeight; //Used later to calculate the height of the scrollView
if (appDelegate.navigationController.navigationBar.hidden == YES) //If the NavBar is Hidden, set scrollViewHeight to 480
{ scrollViewHeight = 480; }
if (appDelegate.navigationController.navigationBar.hidden == NO) //If the NavBar not Hidden, set scrollViewHeight to 360
{ scrollViewHeight = 368; }

imageView.frame = CGRectMake(0, 0, CGImageGetWidth(yourImage)* [myScrollView zoomScale], CGImageGetHeight(yourImage)* [myScrollView zoomScale]);

[imageView setContentMode:UIViewContentModeCenter];
}

Also, I do the following to prevent the image from sticking a the side after zooming out.

- (void) scrollViewDidEndZooming:(UIScrollView *)scrollView withView:(UIView *)view atScale:(float)scale {
myScrollView.frame = CGRectMake(0, 0, 320, 420);
 //put the correct parameters for your scroll view width and height above
}
Owner answered 5/1, 2010 at 5:19 Comment(3)
Hi Jonah. It seems we've been dealing with the same issue for a while. Will check your two solutions and reply soon.Necropsy
Hi Jonah, I tried your latest solution. However, what is temporaryImage ? I tried putting temporaryImage = imageView.image ; however, once I zoom, the image disappears. Thanks, PannagStag
Pannag, temporaryImage was poorly named. It should be called myImage as it is just whatever picture you are using. Sorry for the confusion.Owner
S
0

Just disable the pagination, so it'll work fine:

scrollview.pagingEnabled = NO;
Sarinasarine answered 21/3, 2014 at 5:26 Comment(0)
C
0

I had the exact same problem. Here is how I solved

This code should get called as the result of scrollView:DidScroll:

CGFloat imageHeight = self.imageView.frame.size.width * self.imageView.image.size.height / self.imageView.image.size.width;
BOOL imageSmallerThanContent = (imageHeight < self.scrollview.frame.size.height) ? YES : NO;
CGFloat topOffset = (self.imageView.frame.size.height - imageHeight) / 2;

// If image is not large enough setup content offset in a way that image is centered and not vertically scrollable
if (imageSmallerThanContent) {
     topOffset = topOffset - ((self.scrollview.frame.size.height - imageHeight)/2);
}

self.scrollview.contentInset = UIEdgeInsetsMake(topOffset * -1, 0, topOffset * -1, 0);
Chatham answered 3/2, 2015 at 2:31 Comment(0)
A
0

Although the question is a bit old yet the problem still exists. I solved it in Xcode 7 by making the vertical space constraint of the uppermost item (in this case the topLabel) to the superViews (the scrollView) top an IBOutlet and then recalculating its constant every time the content changes depending on the height of the scrollView's subviews (topLabel and bottomLabel).

class MyViewController: UIViewController {

    @IBOutlet weak var scrollView: UIScrollView!
    @IBOutlet weak var topLabel: UILabel!
    @IBOutlet weak var bottomLabel: UILabel!
    @IBOutlet weak var toTopConstraint: NSLayoutConstraint!

    override func viewDidLayoutSubviews() {
        let heightOfScrollViewContents = (topLabel.frame.origin.y + topLabel.frame.size.height - bottomLabel.frame.origin.y)
        // In my case abs() delivers the perfect result, but you could also check if the heightOfScrollViewContents is greater than 0.
        toTopConstraint.constant = abs((scrollView.frame.height - heightOfScrollViewContents) / 2)
    }

    func refreshContents() {
        // Set the label's text …

        self.view.layoutIfNeeded()
    }
}
Aquiver answered 5/10, 2015 at 10:20 Comment(0)
S
0

A Swift version to just subclass UIScrollView and lauout the subview by yourself. It works pretty smooth.

import UIKit

class CenteringScrollView: UIScrollView {
    override func layoutSubviews() {
        super.layoutSubviews()

        if zoomScale < 1.0 {
            if let subview = self.subviews.first {
                subview.center.x = self.center.x
            }
        }
    }
}

Sussex answered 13/1, 2022 at 23:55 Comment(1)
However, this method repeatedly calls viewForZooming(in:)Sussex
R
0

Everybody who wants to center all content inside UIScrollView should take a look at the center of content (any UIView) inside UIScrollView.

I've written some extension for UIScrollView and can share it with anyone who would like to try it in a project.

You can simply use it like below:

// center your content inside scroll view by x-axis:
self.yourScrollView.centralizeContentByXIfAvailableSpaceExists()

// also you can center your content inside scroll view by y-axis:
self.yourScrollView.centralizeContentByYIYAvailableSpaceExists()

// and you can center your content by x and y calling one function:
self.yourScrollView.centralizeContentIfAvailableSpaceExists()
Rucksack answered 30/11, 2023 at 9:15 Comment(0)

© 2022 - 2024 — McMap. All rights reserved.