Определение количества прокрутки с помощью UIScrollView и PageControl

Для приложения, над которым я работаю, мне нужна прокрутка с разбиением на страницы вместе с несколькими представлениями (в данном случае табличными). Настроить это (используя пример Apple PageControl) достаточно просто, и у меня нет проблем с его работой в отношении прокрутки и привязки между страницами. Я хочу, чтобы представления таблиц были почти как миниатюры представлений таблиц в просмотре прокрутки. Как только вы щелкнете по одному из них, он откроется или увеличится, чтобы вид таблицы покрыл весь экран. Опять же, настройка этого не является проблемой, я могу сделать представления внутри scrollview любого размера, который захочу, и щелкнув мышью, я уверен, что смогу приступить к работе, когда доберусь до этой части. :)

Моя проблема заключается в следующем: в любом данном представлении, независимо от того, на какой странице вы находитесь, я хочу, чтобы следующая страница и предыдущая страница были слегка видны с правой и с левой стороны соответственно. Конечно, на первой странице будет только одна страница, показанная с правой стороны, или с левой стороны для последней страницы. Опять же, установив x-origin для представленных представлений (табличных представлений), я могу заставить первую страницу выглядеть правильно. Однако как только вы прокручиваете страницу, проблема становится очевидной, а именно то, что экран прокручивается слишком далеко.

Разбивая это на части, я полагаю, что моя главная проблема в том, что я не могу изменить объем прокрутки, выполняемой при переключении страниц.

Буду признателен за любую помощь, которую я могу получить, спасибо.

AppDelegate.h

#import <UIKit/UIKit.h>

@interface AppDelegate : NSObject <UIApplicationDelegate, UIScrollViewDelegate> {
UIWindow *window;
UIView *background;
UIScrollView *scrollView;
UIPageControl *pageControl;
NSMutableArray *viewControllers;

// To be used when scrolls originate from the UIPageControl
// I've commented this out (and all references to it) since I've disabled user interaction with the UIPageControl
//BOOL pageControlUsed;
}

@property (nonatomic, retain) IBOutlet UIWindow *window;
@property (nonatomic, retain) IBOutlet UIView *background;

@property (nonatomic, retain) IBOutlet UIScrollView *scrollView;
@property (nonatomic, retain) IBOutlet UIPageControl *pageControl;
@property (nonatomic, retain) NSMutableArray *viewControllers;

- (IBAction)changePage:(id)sender;

@end  

AppDelegate.m

#import "AppDelegate.h"
#import "MyViewController.h"

static NSUInteger kNumberOfPages = 4;

@interface AppDelegate (PrivateMethods)

- (void)loadScrollViewWithPage:(int)page;
- (void)scrollViewDidScroll:(UIScrollView *)sender;

@end

@implementation AppDelegate

@synthesize window, scrollView, pageControl, viewControllers, background;

- (void)dealloc {
[background release];
[viewControllers release];
[scrollView release];
[pageControl release];
[window release];
[super dealloc];
}

- (void)applicationDidFinishLaunching:(UIApplication *)application {

background.backgroundColor = [UIColor colorWithPatternImage:[UIImage imageNamed:@"Background.png"]];

// view controllers are created lazily
// in the meantime, load the array with placeholders which will be replaced on demand
NSMutableArray *controllers = [[NSMutableArray alloc] init];
for (unsigned i = 0; i < kNumberOfPages; i++) {
    [controllers addObject:[NSNull null]];
}
self.viewControllers = controllers;
[controllers release];

// a page is the width of the scroll view
scrollView.pagingEnabled = YES;
scrollView.contentSize = CGSizeMake(scrollView.frame.size.width * kNumberOfPages, scrollView.frame.size.height);
scrollView.showsHorizontalScrollIndicator = NO;
scrollView.showsVerticalScrollIndicator = NO;
scrollView.scrollsToTop = NO;
scrollView.delegate = self;

pageControl.numberOfPages = kNumberOfPages;
pageControl.currentPage = 0;

// pages are created on demand
// load the visible page
// load the page on either side to avoid flashes when the user starts scrolling
[self loadScrollViewWithPage:0];
[self loadScrollViewWithPage:1];
}

- (void)loadScrollViewWithPage:(int)page {
if (page < 0) return;
if (page >= kNumberOfPages) return;

// replace the placeholder if necessary
MyViewController *controller = [viewControllers objectAtIndex:page];
if ((NSNull *)controller == [NSNull null]) {
    controller = [[MyViewController alloc] initWithPageNumber:page];
    [viewControllers replaceObjectAtIndex:page withObject:controller];
    [controller release];
}

// add the controller's view to the scroll view
if (nil == controller.view.superview) {
    CGRect frame = CGRectMake(50, 50, 220, 330);
    frame.origin.x = 50 + (250 * page);
    frame.origin.y = 50;
    controller.view.frame = frame;
    [scrollView addSubview:controller.view];
}
}

- (void)scrollViewDidScroll:(UIScrollView *)sender {
// We don't want a "feedback loop" between the UIPageControl and the scroll delegate in
// which a scroll event generated from the user hitting the page control triggers updates from
// the delegate method. We use a boolean to disable the delegate logic when the page control is used.
/*if (pageControlUsed) {
    // do nothing - the scroll was initiated from the page control, not the user dragging
    return;
}*/

// Switch the indicator when more than 50% of the previous/next page is visible
CGFloat pageWidth = scrollView.frame.size.width;
int page = floor((scrollView.contentOffset.x - pageWidth / 2) / pageWidth) + 1;
pageControl.currentPage = page;

// load the visible page and the page on either side of it (to avoid flashes when the user starts scrolling)
[self loadScrollViewWithPage:page - 1];
[self loadScrollViewWithPage:page];
[self loadScrollViewWithPage:page + 1];

// A possible optimization would be to unload the views+controllers which are no longer visible
}

// At the begin of scroll dragging, reset the boolean used when scrolls originate from the UIPageControl
- (void)scrollViewWillBeginDragging:(UIScrollView *)scrollView {
//pageControlUsed = NO;
}

// At the end of scroll animation, reset the boolean used when scrolls originate from the UIPageControl
- (void)scrollViewDidEndDecelerating:(UIScrollView *)scrollView {
//pageControlUsed = NO;
}

- (IBAction)changePage:(id)sender {
int page = pageControl.currentPage;

// load the visible page and the page on either side of it (to avoid flashes when the user starts scrolling)
[self loadScrollViewWithPage:page - 1];
[self loadScrollViewWithPage:page];
[self loadScrollViewWithPage:page + 1];

// update the scroll view to the appropriate page
CGRect frame = scrollView.frame;
frame.origin.x = frame.size.width * page;
frame.origin.y = 0;
[scrollView scrollRectToVisible:frame animated:YES];

// Set the boolean used when scrolls originate from the UIPageControl. See       scrollViewDidScroll: above.
//pageControlUsed = YES;
}

@end

person Glitch    schedule 09.09.2010    source источник
comment
Мне нужна помощь по этой самой проблеме ... изменение размера прокрутки на что-то меньшее, чем размер экрана. У меня отображаются предыдущие / следующие страницы, но при прокрутке к следующей странице она прокручивается далеко. Поскольку следующая / предыдущая страницы немного видны, прокрутка страницы очень небольшая. У кого-нибудь есть идея?   -  person Valerie    schedule 04.02.2011
comment
В итоге я использовал этот метод: blog.proculo.de/index.php?url=archives/. Надеюсь, поможет   -  person Glitch    schedule 10.02.2011


Ответы (1)


Похоже, вы по-разному устанавливаете размер кадра scrollView и кадра controller.view. ScrollView - это IBOutlet, это означает, что вы создаете scrollView с помощью конструктора интерфейса, но размещаете контент, используя следующие строки:

CGRect frame = CGRectMake(50, 50, 220, 330);
frame.origin.x = 50 + (250 * page);
frame.origin.y = 50;
controller.view.frame = frame;
[scrollView addSubview:controller.view];

и поместив его в scrollView. Если вы не хотите видеть предыдущую / следующую страницу сбоку от scrollView, вам нужно изменить числа, которые вы используете в первой строке. Попробуйте использовать scrollView.frame.size или проверьте размер кадра scrollView, созданного вами из IB, и используйте их вместо магических чисел, которых вам следует избегать.

person Yunus Nedim Mehel    schedule 02.11.2012