问题是页面大小已经改变(即滚动视图的宽度),但图像的宽度没有改变。
一种解决方案是,当您旋转时,您重新计算 UIImageViews 的大小并计算滚动视图的 contentSize。也就是说,将 for 循环和对 scrollView.contentSize 的赋值从viewDidLoad移到私有方法中,比如 sizeImagesAndSetContentSize。从willAnimateRotationToInterfaceOrientation:duration调用它:类似于以下内容:
- (void)viewDidLoad
{
[super viewDidLoad];
imageArray = [[NSArray alloc] initWithObjects:@"image1.jpg", @"image2.jpg", @"image3.jpg", nil];
[self sizeImagesAndSetContentSize];
scrollView.autoresizingMask = UIViewAutoresizingFlexibleWidth | UIViewAutoresizingFlexibleTopMargin | UIViewAutoresizingFlexibleLeftMargin | UIViewAutoresizingFlexibleRightMargin;
pageControl.autoresizingMask = UIViewAutoresizingFlexibleWidth | UIViewAutoresizingFlexibleTopMargin;
}
- (void) sizeImagesAndSetContentSize {
for (int i = 0; i < [imageArray count]; i++) {
CGRect frame;
frame.origin.x = self.scrollView.frame.size.width * i;
frame.origin.y = 0;
frame.size = self.scrollView.frame.size;
UIImageView *imageView = [[UIImageView alloc] initWithFrame:frame];
imageView.image = [UIImage imageNamed:[imageArray objectAtIndex:i]];
[self.scrollView addSubview:imageView];
}
scrollView.contentSize = CGSizeMake(scrollView.frame.size.width * [imageArray count], scrollView.frame.size.height);
}
- (void)willAnimateRotationToInterfaceOrientation:(UIInterfaceOrientation)interfaceOrientation duration:(NSTimeInterval)duration {
[self sizeImagesAndSetContentSize];
}
顺便说一句,viewDidLoad不是进行几何计算的地方——就像涉及到 scrollView 框架的事情一样,因为那时还没有设置几何。在viewDidLoad中,事物的大小是它们在故事板(或笔尖)中的大小,并且尚未根据实际设备和实际方向进行调整。如果到目前为止它对您有用,那是因为您从故事板布局的方向开始。可能是这样做的最佳位置,因此调用sizeImagesAndSetContentSize是在方法viewDidLayoutSubviews中。事实上,如果你把调用移到那里,你就不需要在旋转时显式调用它,因为当旋转发生时,视图会布置子视图和viewDidLayoutSubviews将被自动调用。所以我建议:
- (void)viewDidLoad
{
[super viewDidLoad];
imageArray = [[NSArray alloc] initWithObjects:@"image1.jpg", @"image2.jpg", @"image3.jpg", nil];
scrollView.autoresizingMask = UIViewAutoresizingFlexibleWidth | UIViewAutoresizingFlexibleTopMargin | UIViewAutoresizingFlexibleLeftMargin | UIViewAutoresizingFlexibleRightMargin;
pageControl.autoresizingMask = UIViewAutoresizingFlexibleWidth | UIViewAutoresizingFlexibleTopMargin;
}
- (void) sizeImagesAndSetContentSize {
for (int i = 0; i < [imageArray count]; i++) {
CGRect frame;
frame.origin.x = self.scrollView.frame.size.width * i;
frame.origin.y = 0;
frame.size = self.scrollView.frame.size;
UIImageView *imageView = [[UIImageView alloc] initWithFrame:frame];
imageView.image = [UIImage imageNamed:[imageArray objectAtIndex:i]];
[self.scrollView addSubview:imageView];
}
scrollView.contentSize = CGSizeMake(scrollView.frame.size.width * [imageArray count], scrollView.frame.size.height);
}
- (void)viewDidLayoutSubviews {
[self sizeImagesAndSetContentSize];
}
附录:
在图像中添加文字:
要在图像顶部添加文本——您可以在滚动视图的“顶部”添加文本元素,如标签。考虑以下应用程序的屏幕截图。在顶级 Cloud Slide Show View 中有一个全屏滚动视图,在相同的层次结构级别但在子视图列表的更下方(因此在滚动视图的“顶部”)是描述标签,3 个按钮,和一个 UIPageControl。标签设置为通过支柱和弹簧(在 iOS 5 上运行)在旋转时调整大小,因此您不必弄乱它。
此应用程序更改描述标签以匹配当前显示在滚动视图页面中的图像——因此它必须检测页面更改。此应用程序在滚动视图委托协议方法scrollViewDidEndDecelerating:中执行此操作:它决定滚动停止时屏幕上显示的图像并填写正确的描述。
另一种方法是在每个 UIImage 上放置一个标签,但这很混乱,这就是我的应用程序不这样做的原因。您可能希望视图同时包含 UIImage 和标签,以便重新定位标签是有意义的。可能在代码中创建它......对我来说,太麻烦了。
顺便说一句,我应该说在滚动视图中使用页面模式可能不再是执行此操作的“最佳实践”方式。我正在将我的应用程序转换为使用 UIPageViewController 来简化应用程序。例如,您不必执行 resize-on-rotate 的操作,因为 UIPageViewControl 的各个子 VC 可以自动调整自己的大小。