Объекты интерфейса UISplitView Interface Builder не выделены - PullRequest
0 голосов
/ 04 февраля 2011

Я сейчас играю с UISplitView Controller, так как некоторые из них должны работать в UITabBarController.После нескольких попыток я наконец нашел удобный способ сделать это, единственная проблема, которую я получаю, заключается в том, что мне приходится вручную создавать экземпляры моих деталей и основного вида, хотя они настроены в IB и хорошо связаны между собой.

Вот как я это делаю

Я инициализирую UITabBarCOntroller в моем MainWindow.xib и устанавливаю элементы панели вкладок.

Мой первый контроллер вкладок наследуется от UISplitViewController и настроен с xib.Вот реализация этого класса FirstViewController

#import "FirstSplitViewController.h"
#import "MasterSplitViewController.h"
#import "DetailSplitViewController.h"


@implementation FirstSplitViewController

@synthesize detailSplitViewController,masterSplitViewController;



// Implement viewDidLoad to do additional setup after loading the view, typically from a nib.
- (void)viewDidLoad {
[super viewDidLoad];

/*  
masterSplitViewController = [[[MasterSplitViewController alloc] initWithNibName:@"MasterSplitViewController" bundle:nil] autorelease];
detailSplitViewController = [[[DetailSplitViewController alloc] initWithNibName:@"DetailSplitViewController" bundle:nil] autorelease];
*/

self.viewControllers = [NSArray arrayWithObjects:masterSplitViewController, detailSplitViewController , nil];
self.delegate = detailSplitViewController;
}

- (BOOL)shouldAutorotateToInterfaceOrientation:(UIInterfaceOrientation)interfaceOrientation {
// Overriden to allow any orientation.
return YES;
}


- (void)didReceiveMemoryWarning {
// Releases the view if it doesn't have a superview.
[super didReceiveMemoryWarning];

// Release any cached data, images, etc. that aren't in use.
}


- (void)viewDidUnload {
[super viewDidUnload];
// Release any retained subviews of the main view.
// e.g. self.myOutlet = nil;
}


- (void)dealloc {
[super dealloc];
}


@end

Вот моя реализация MasterSplitview

#import "MasterSplitViewController.h"


@implementation MasterSplitViewController


// The designated initializer.  Override if you create the controller programmatically and want to perform customization that is not appropriate for viewDidLoad.
/*
- (id)initWithNibName:(NSString *)nibNameOrNil bundle:(NSBundle *)nibBundleOrNil {
self = [super initWithNibName:nibNameOrNil bundle:nibBundleOrNil];
if (self) {
    // Custom initialization.
}
return self;
}
*/


// Implement viewDidLoad to do additional setup after loading the view, typically from a nib.
- (void)viewDidLoad {
[super viewDidLoad];
}



- (BOOL)shouldAutorotateToInterfaceOrientation:(UIInterfaceOrientation)interfaceOrientation {
// Overriden to allow any orientation.
return YES;
}


- (void)didReceiveMemoryWarning {
// Releases the view if it doesn't have a superview.
[super didReceiveMemoryWarning];

// Release any cached data, images, etc. that aren't in use.
}


- (void)viewDidUnload {
[super viewDidUnload];
// Release any retained subviews of the main view.
// e.g. self.myOutlet = nil;
}


- (void)dealloc {
[super dealloc];
}


@end

и моя реализация DetailSplitViewController

#import "DetailSplitViewController.h"

@interface DetailSplitViewController ()
@property (nonatomic, retain) UIPopoverController *popoverController;
- (void)configureView;
@end

@implementation DetailSplitViewController

@synthesize toolbar, popoverController, detailItem, detailDescriptionLabel;

/*
When setting the detail item, update the view and dismiss the popover controller if it's showing.
*/
- (void)setDetailItem:(id)newDetailItem {
if (detailItem != newDetailItem) {
    [detailItem release];
    detailItem = [newDetailItem retain];

    // Update the view.
    [self configureView];
}

if (self.popoverController != nil) {
    [self.popoverController dismissPopoverAnimated:YES];
}        
}

- (void)configureView {
// Update the user interface for the detail item.
// detailDescriptionLabel.text = [detailItem description];   
}

- (void)splitViewController: (UISplitViewController*)svc willHideViewController:(UIViewController *)aViewController withBarButtonItem:(UIBarButtonItem*)barButtonItem forPopoverController: (UIPopoverController*)pc 
{
barButtonItem.title = @"Root List";
NSMutableArray *items = [[toolbar items] mutableCopy];
[items insertObject:barButtonItem atIndex:0];
[toolbar setItems:items animated:YES];
[items release];
self.popoverController = pc;
}

// Called when the view is shown again in the split view, invalidating the button and popover controller.
- (void)splitViewController: (UISplitViewController*)svc willShowViewController:(UIViewController *)aViewController invalidatingBarButtonItem:(UIBarButtonItem *)barButtonItem {

NSMutableArray *items = [[toolbar items] mutableCopy];
[items removeObjectAtIndex:0];
[toolbar setItems:items animated:YES];
[items release];
self.popoverController = nil;
}


- (BOOL)shouldAutorotateToInterfaceOrientation:(UIInterfaceOrientation)interfaceOrientation {
// Overriden to allow any orientation.
return YES;
}


- (void)didReceiveMemoryWarning {
// Releases the view if it doesn't have a superview.
[super didReceiveMemoryWarning];

// Release any cached data, images, etc. that aren't in use.
}


- (void)viewDidUnload {
[super viewDidUnload];
// Release any retained subviews of the main view.
// e.g. self.myOutlet = nil;
}


- (void)dealloc {
//[toolbar release];
[super dealloc];
} 

@end

Everiting подключен в XIB, и проблема, которую я получаю, состоит в том, что, когда мой FirstSplitViewController загружается из его xib, мой основной и подробный контроллеры splitview не выделяются (они связаны в IB).Если я распределяю их вручную, все работает как талисман (не комментируя строки инициализации alloc ниже в моем FirstSplitViewController.m)

// Implement viewDidLoad to do additional setup after loading the view, typically from a nib.
- (void)viewDidLoad {
[super viewDidLoad];

/*  
masterSplitViewController = [[[MasterSplitViewController alloc] initWithNibName:@"MasterSplitViewController" bundle:nil] autorelease];
detailSplitViewController = [[[DetailSplitViewController alloc] initWithNibName:@"DetailSplitViewController" bundle:nil] autorelease];
*/

self.viewControllers = [NSArray arrayWithObjects:masterSplitViewController, detailSplitViewController , nil];
self.delegate = detailSplitViewController;

}

Так что мой вопрос, почему эти объекты не загружаются, когда xib находится?Это действительно первый раз, когда я должен сделать это вручную.Может быть, я что-то упускаю.

Спасибо за любые ответы или советы

s-mart

1 Ответ

1 голос
/ 22 марта 2011

Я только что столкнулся с этим же явлением (я думаю).Я только начинаю полностью понимать, как Heirarchy / view Interface Builder работает для iOS.Похоже, переменные-члены, которые связаны через IBOutlet, не инициализируются до тех пор, пока не будет получен доступ к экземпляру контроллера.Мой код был таким:

    if(self.sectionOneViewController == nil)
{
    SectionOneViewController *sectionOneView = [[SectionOneViewController alloc]
                        initWithNibName:@"SectionOne"
                        bundle:[NSBundle mainBundle]];
    self.sectionOneViewController = sectionOneView;

    [sectionOneView release];
    //[self showSectionOne:sender];
}

    [self.navigationController pushViewController:self.sectionOneViewController animated:YES];

[[UIApplication sharedApplication].keyWindow  addSubview:self.sectionOneViewController.sectionOneTabController.view];

, если бы я поменял местами последние 2 строки, у него был бы нулевой указатель для sectionOneTabController, если я не пересмотрю представление.Я думаю, что наличие контроллера, добавляющего ваше представление, необходимо до того, как ссылки на .xib будут доступны.

...