Implementing UIScrollView programmatically
Since you're not loading the interface from a nib file, you should set up your UIScrollView
in your PhoneContentController
's init
method:
- (id)init
{
[super init];
if (self) {
scrollView = [[UIScrollView alloc] initWithFrame:CGRectMake(0, 20, 320, 440)];
pageControl = [[UIPageControl alloc] initWithFrame:CGRectMake(<#CGFloat x#>, <#CGFloat y#>, <#CGFloat width#>, <#CGFloat height#>)]; // Place it where you want it.
viewControllers = [[NSMutableArray alloc] init];
// load our data from a plist file inside our app bundle
NSString *path = [[NSBundle mainBundle] pathForResource:@"content_iPhone" ofType:@"plist"];
self.contentList = [NSArray arrayWithContentsOfFile:path];
// 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];
}
return self;
}
In your AppDelegate
, make the following changes:
- (void)applicationDidFinishLaunching:(UIApplication *)application
{
self.window = [[UIWindow alloc] initWithFrame:[[UIScreen mainScreen] bounds]];
if (UI_USER_INTERFACE_IDIOM() == UIUserInterfaceIdiomPhone) {
contentController = [[PhoneContentController alloc] init];
} else {
contentController = [[PadContentController alloc] init];
}
[self.window addSubview:contentController.view];
[window makeKeyAndVisible];
}
Dollarslice
Updated on June 04, 2022Comments
-
Dollarslice almost 2 years
In the page control sample from apple there is a ScrollView in the interface builder. It is linked with the corresponding IBOutlet. I want to change the code so this is all done programatically. I delete the interface builder object, I delete the IBOutlet keyword. I alloc and init the scrollView, but nothing appears when I run the program.
I assume this is because I need to assign it as a subView to the main view. Or do I? I still don't really understand how all the views work and interact with each other. If I do
[self.view addSubView:ScrollView];
I get a runtime error (or something, it usually just says something like BAD ACCESS or SIGABRT).What am I doing wrong? Am I on the wrong path completely? (only two days in to ios programming, still a bit lost in the woods)
awakeFromNib in phone content controller:
- (void)awakeFromNib { scrollView = [[UIScrollView alloc] initWithFrame:CGRectMake(0, 0, self.view.frame.size.width, self.view.frame.size.height)]; // load our data from a plist file inside our app bundle NSString *path = [[NSBundle mainBundle] pathForResource:@"content_iPhone" ofType:@"plist"]; self.contentList = [NSArray arrayWithContentsOfFile:path]; // 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]; }
header file:
#import <UIKit/UIKit.h> #import <Foundation/Foundation.h> #import "ContentController.h" @interface PhoneContentController : ContentController <UIScrollViewDelegate> { UIScrollView *scrollView; UIPageControl *pageControl; NSMutableArray *viewControllers; // To be used when scrolls originate from the UIPageControl BOOL pageControlUsed; } @property (nonatomic, retain) UIScrollView *scrollView; @property (nonatomic, retain) IBOutlet UIPageControl *pageControl; @property (nonatomic, retain) NSMutableArray *viewControllers; - (IBAction)changePage:(id)sender; @end
appDelegate:
#import "AppDelegate.h" #import "ContentController.h" @implementation AppDelegate @synthesize window, contentController; - (void)dealloc { [window release]; [contentController release]; [super dealloc]; } - (void)applicationDidFinishLaunching:(UIApplication *)application { NSString *nibTitle = @"PadContent"; if (UI_USER_INTERFACE_IDIOM() == UIUserInterfaceIdiomPhone) { nibTitle = @"PhoneContent"; } [[NSBundle mainBundle] loadNibNamed:nibTitle owner:self options:nil]; [self.window addSubview:self.contentController.view]; [window makeKeyAndVisible]; } @end
and the scrollView has been deleted from the xib file. Note: this is a new version of the downloaded program where all I have changed is deleting the IBOutlet keyword for the scrollView, deleted the scroll from the xib and added the alloc, init line in awake from nib.
I've had suggestions to change the appDelegate and change the awakeFromNib to an init method, i've tried all this but it still doesn't work.
-
Dollarslice over 12 yearsIt doesn't break but it doesn't display what it did before I removed the nib stuff. Just a blank screen now. Also my init method is called awakeFromNib. Is that significant?
-
Dollarslice over 12 yearsoh no sorry, my bad. It does break. I get EXC_BAD_ACCESS on the add subview line
-
Dollarslice over 12 yearsthank you, I'm not sure I fully understand the view hierarchy still. Also it still doesn't work. It doesn't break but nothing is displayed.
-
Dollarslice over 12 yearsI changed the background colour to red, and the screen is red so I guess it is displaying, but the things that used to draw on it aren't there anymore and it doesn't seem to be scrolling... there must be some link somewhere that is missing..
-
Mudit Bajpai over 12 yearsi have done the same thing and it's working correct.it mean's you have change some other code also.
-
Iñigo Beitia over 12 years@SirYakalot, I've made changes to the answer, make sure you remove the
addSubview
line from the init method. I would recommend usinginit
instead ofawakeFromNib
. -
Dollarslice over 12 yearsthe only other thing I changed is the UILabel called numberTitle inside of MyViewController. I did the same thing - removing it from the xib file and coding it in programatically. It worked before I changed the scrollView. In LoadScrollViewWithPage, at the end of the second if statement I am doing ` [controller.view addSubview:controller.numberTitle];` It is also now a UITextView, not a UILabel
-
Dollarslice over 12 yearsI can just change the name and the program will still work? why is this?
-
Dollarslice over 12 yearsI suppose what i'm asking is how can I then change numberTitle to ALSO be done programatically?
-
Dollarslice over 12 yearsactually I just re-downloaded the code, deleted the scrollView in xib, deleted the IBOutlet keyword and added the alloc init and it doesn't work.
-
Dollarslice over 12 yearsstill doesn't work for me. I get a non-scrollable black screen. the content is gone.
-
Iñigo Beitia over 12 yearsSince you're not loading from nib anymore from the
AppDelegate
'sapplicationDidFinishLaunching:
, the methodawakeFromNib
won't be called. Instead notice how you callinit
on the newPhoneContentController
instance. -
Hiren over 12 years@SirYakalot: so you have add the scrollview and now you want to add UItextView in the scrollview?
-
Dollarslice over 12 yearsIt says that Phone and Pad content controller are unknown receivers. It suggests that I meant content controller?
-
Mudit Bajpai over 12 years@SirYakalot:are You saying you are using initWithFrame and it's not working?....How can it's possible bcoz same thing i have done and it's working fine.
-
Dollarslice over 12 yearshave you still got the scrollView inside your xib file?
-
Mudit Bajpai over 12 yearsno, inside xib there are no scrollView.... i am only using alloc initwithframe inside awakeFromNib method nothing else
-
Dollarslice over 12 yearsseems there is an extra bracket in your code before alloc. plus you need to include the appropriate headers. After that's done you run it and it breaks. EXC_BAD_ACCESS
-
Dollarslice over 12 yearswell If I make those three simple changes. Deleting the IBOutlet keyword from the property line of phonecontentcontroller, deleting the scroll view from the nib file and adding the alloc init at the top of awakeFromNib - the code runs but nothing displays but a blank white screen.
-
Mudit Bajpai over 12 yearsyes i agree with this answer because you are using alloc init only, use alloc initwithframe .
-
Caleb over 12 yearsThere's no extra bracket before alloc: the first
[
corresponds to the]
afterinit
. It's typical Obj-C style to combine calls to alloc and init:[[alloc Foo] init];
That means allocate an instance of Foo, and then initialize the new object. You should always follow this pattern when creating objects. -
Iñigo Beitia over 12 yearsSorry for the typo. I've updated the answer and tested it, it now works.
-
Dollarslice over 12 yearsThank you! That worked. Sorry for needing so much help with this. would you mind very briefly explaining why you made the changes you did and what they do? I understand if you don't have time / can't be bothered. Thanks again!
-
Iñigo Beitia over 12 yearsWhen you use Outlets in InterfaceBuilder, the initializations are done for you, that's why I needed to add init calls for the scrollView, the page controller and the array holding the pages. Since you no longer load from nib, the awakeFromNib method is never called, so you need to put all the setup code of the view in the Class' init.