我正在创建一些示例应用程序来理解cocoa中的视图导航,绑定等概念。 这是场景: 我有一个窗口,在MainMenu.Xib中有一个标签视图(2个标签)。 我在第一个标签中有一个文本字段,在第二个标签中有标签。我希望它们都反映相同的值,我想使用绑定来做到这一点。另外,我不想使用提供给我的视图以及选项卡视图。
这些是我已经完成的步骤。
使用以下代码在applicationDidFinishLaunching:方法中单独设置每个选项卡视图项的视图:
- (void)applicationDidFinishLaunching:(NSNotification *)aNotification
{
// Insert code here to initialize your application
//initialize view controllers
view1=[[ViewTab1 alloc] initWithNibName:@"ViewTab1" bundle:nil];
view2=[[ViewTab2 alloc] initWithNibName:@"ViewTab2" bundle:nil];
//set views
[[[myTabView tabViewItems] objectAtIndex:0]setView:view1.view];
[[[myTabView tabViewItems] objectAtIndex:1]setView:view2.view];
}
ViewTab1只有一个文本字段(以及一个关联的标签)。我已经将它绑定到AppDelegate中声明的变量(名称)。 ViewTab2有一个标签。我也将它绑定到AppDelegate中的同一个变量。
变量'name'在AppDelegate的init方法中初始化。
AppDelegate.h
....
NSString *name;
....
@property(strong) ViewTab1 *view1;
@property(strong) ViewTab2 *view2;
@property (assign) IBOutlet NSTabView *myTabView;
@property (strong) NSString *name;
....
AppDelegate.m
....
@synthesize myTabView;
@synthesize view1,view2;
@synthesize name;
....
- (id)init {
self = [super init];
if (self) {
name=@"dummy";
}
return self;
....
除此之外,我还没有在我的程序中进行任何编码。
在ViewTab1.xib中,我得到了一个对象并使其成为AppDelegate的一个实例,然后将Application对象(NSApplication)的委托引用连接到同一个对象。 (我希望这是获取AppDelegate对象的正确方法。)
我在ViewTab2.xib中做了同样的事情
然后我将ViewTab1中的文本字段绑定到ViewTab2中并将其绑定到AppDelegate中的此变量。
当我运行程序时,文本字段和标签都显示“dummy”。但是当我更改文本字段中的值时,它不会反映在第二个选项卡中的标签中(即ViewTab2)。
请告诉我我做错了什么。
答案 0 :(得分:2)
如何从任何已加载的Nib建立与同一App委托对象的绑定?
是的,我知道这个令人沮丧的情况如上所述...经过数周和数百页的KVO文档 - 通知 - 绑定我认为有一个非常简单的解决方案。
正如我们在一些信息源中所发现的那样,nib加载过程会产生新的成员实例......我们需要使用绑定连接到旧的实例。
请注意,在加载nib后,InterfaceBuilder中的绑定会自动重定向到这些新实例
为什么不将App委托的指针重定向到旧实例?
在加载nib的方法中,您可以在nib加载之前和之后测试哪个对象是app委托。 如果新的与前一个不同,您可以根据需要重定向。
这个简单的例子适用于Xcode3 10.5.8以下,目标是OSX10.5 / i386:
// ***** SOMEWHERE IN DEFAULT APP-DELEGATE.m IMPLEMENTATION
- (IBAction) createOtherWindowFromNib: (id)sender
{
// ensure that app delegate is set as you want...
[NSApp setDelegate:self];
NSLog(@"APP-DELEGAT **** CREATE-TEST-WINDOW ***** WHO IS APP-DELEGATE BEFORE NIB LOAD: %@ ", [[NSApp delegate] description]);
// we can bind members of the nib to this controller over proxy object named "File’s Owner"
NSWindowController *otherWinCapo = [[NSWindowController alloc] initWithWindowNibName: @"OtherTestWindow"];
NSLog(@"APP-DELEGAT **** CREATE-TEST-WINDOW ***** WHO IS APP-DELEGATE AFTER NIB LOAD: %@ ", [[NSApp delegate] description]);
// make some test for delegates before/after here if you need ...
// usually your bindings made inside "OtherTestWindow.xib" by IB doesn’t works in this moment
// ... and some redirection if needed
[NSApp setDelegate:self];
// afer that the bind made in IB inside "OtherTestWindow.xib"
// referred to (proxy object) "Application.delegate.myBOOL" (Bind to:Application, Model Key Path:delegate.myBOOL)
// react to changes of myBOOL placed in default app delegate object as expected
// simultaneously in every open instance of "OtherTestWindow.xib"
[otherWinCapo showWindow: otherWinCapo.window]; // we need populate the window instance on screen to see it
}
答案 1 :(得分:0)
您是否在NSTextField
控件中启用了“持续更新价值”?
请参阅this example。
答案 2 :(得分:0)
我认为问题在于您设置为app delegate类的xib中的对象会创建2个不同的app委托实例,因此更改text字段的值会在一个实例中更改name的值,但不会更改另一个。不幸的是,这就是你做错了,我现在想不出解决方案。