iphone - 将数据从我的详细 View 发送回我的 UITableView

标签 iphone objective-c cocoa-touch uitableview uinavigationcontroller

首先,感谢上帝提供了 Stack Overflow。我是 Objective-C/Cocoa/iPhone 开发的新手,这个网站是一个很棒的帮助。

我有一个窗口,有一个选项卡栏,有一个导航 Controller ,加载 UITableViewController。单击导航栏上的“添加”按钮(以编程方式创建),推送 UITableViewController,如下所示:

InvoiceAddViewController *addController = [[InvoiceAddViewController alloc] initWithNibName:@"InvoiceAddViewController" bundle:[NSBundle mainBundle]];
[self.navigationController pushViewController:addController animated:YES];

这个 TableView Controller 推送它自己的详细 View :

    UITableViewCell *targetCell = [self.tableView cellForRowAtIndexPath:indexPath];

    GenericTextFieldDetailViewController *dvController = [[GenericTextFieldDetailViewController alloc] initWithNibName:@"GenericTextFieldDetailViewController" bundle:[NSBundle mainBundle]];

    dvController.fieldName = targetCell.textLabel.text;
    dvController.fieldValue = targetCell.detailTextLabel.text;

    [self.navigationController pushViewController:dvController animated:YES];
    [dvController release];

这个概念是,您单击 TableView Controller 中的一个单元格,例如“注释”。这将使用您单击的“字段”的名称和值(如果已存在)推送“GenericTextFieldDetailViewController”。这使我能够重复使用详细信息 View ,而不是为每个字段创建一个令人厌恶的广告。

为了将数据推回,我在“Add”UITableViewController 上创建了一个方法:

- (void) updateField:(NSString*) fieldName value:(NSString*) fieldValue
{
    UITableViewCell *targetCell;
    if([fieldName isEqualToString:@"Invoice ID"])
    {
        NSUInteger indexArr[] = {1,1};
        targetCell = [[self tableView] cellForRowAtIndexPath:[NSIndexPath indexPathWithIndexes:indexArr length:2]];
        targetCell.detailTextLabel.text = fieldValue;
    }
    else if([fieldName isEqualToString:@"P.O. Number"])
    {
        NSUInteger indexArr[] = {1,2};
        targetCell = [[self tableView] cellForRowAtIndexPath:[NSIndexPath indexPathWithIndexes:indexArr length:2]];
        targetCell.detailTextLabel.text = fieldValue;
    }
    else if([fieldName isEqualToString:@"Add Note"])
    {
        NSUInteger indexArr[] = {3,0};
        targetCell = [[self tableView] cellForRowAtIndexPath:[NSIndexPath indexPathWithIndexes:indexArr length:2]];
        targetCell.detailTextLabel.text = fieldValue;
    }
}

此方法旨在接收我在“Generic”中使用此方法推送的数据:

- (IBAction)saveField:(id)sender
{
    self.fieldValue = theTextField.text;
    InvoiceAddViewController *parentController = (InvoiceAddViewController*)self.view.superview;
    [parentController updateField:self.fieldName value:self.fieldValue];
}

这给我们带来了问题: 当 save 方法触发时,它会抛出无效选择器错误,因为 self.view.superview 不是推送“通用”详细信息 View 的 UITableView。

我尝试了以下组合(来自 GDB):

(gdb) po [[self view] superview]
<UIViewControllerWrapperView: 0x4b6d4d0; frame = (0 64; 320 367); autoresize = W+H; layer = <CALayer: 0x4b6c090>>
(gdb) po [[self navigationController] parentViewController]
<UITabBarController: 0x4d2fa90>
(gdb) po [self parentViewController]
<UINavigationController: 0x4d2fdc0>

我感觉我正落在我想要调用的 UITableView 周围,但找不到它。

我做错了什么?

不要拔出更多头发

最佳答案

问题在于您混淆了导航堆栈的 View 层次结构。您的详细 View Controller 想要向将其推送到堆栈上的 Controller 发送一条消息,该 Controller 是导航 Controller 的 viewControllers 数组中倒数第二个对象。

尝试将 saveField: 方法更改为:

- (IBAction)saveField:(id)sender
{
    self.fieldValue = theTextField.text;
    NSArray *navigationStack = self.navigationController.viewControllers;
    InvoiceAddViewController *parentController = (InvoiceAddViewController*)[navigationStack objectAtIndex:navigationSTack.count - 2];
    [parentController updateField:self.fieldName value:self.fieldValue];
}

编辑:我应该注意到这个设计非常脆弱。更好的方法是应用模型- View - Controller 并创建一个表示字段和标题值的对象。然后,您的 InvoiceAddViewController 可以将这些对象的实例传递到您的详细信息 Controller ,并且当您的详细信息 Controller 更改它们时,这些更改可以轻松反射(reflect)在您的其他 Controller 中。

编辑 2:以下是其工作原理的提示。

   UITableViewCell *targetCell = [self.tableView cellForRowAtIndexPath:indexPath];

    GenericTextFieldDetailViewController *dvController = [[GenericTextFieldDetailViewController alloc] initWithNibName:@"GenericTextFieldDetailViewController" bundle:[NSBundle mainBundle]];

    dvController.dataObject = [self dataObjectForIndexPath:indexPath];
    [self.navigationController pushViewController:dvController animated:YES];
    [dvController release];

我假设您已经实现了 dataObjectForIndexPath: 方法。据推测 tableView:cellForRowAtIndexPath: 也会使用此方法来配置其单元格。

现在,您可以消除 saveField: 和 updateField: 方法。在您的 InvoiceAddViewController 中, viewWillAppear: 可用于刷新您的 View ,如下所示:

- (void)viewWillAppear:(BOOL)animated
{
    [super viewWillAppear:animated];
    [self.tableView reloadData]
}

这里有一个充满可能性的世界。 reloadData 是一个非常严厉的方法。尝试使用 KVO 之类的东西来使其更加自动化。

在您的详细 Controller 中,当然,不要忘记更新对象,比如说 View 将消失,执行类似的操作

self.dataObject.fieldValue = theTextField.text;

这只是为了让您开始。有很多可能性和细节需要考虑。您确实应该查看大量示例代码,这种模式被大量使用。开发人员门户上的 CoreDataBooks 示例使用类似的模式,几乎肯定还有其他模式。

关于iphone - 将数据从我的详细 View 发送回我的 UITableView,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/5037811/

相关文章:

iphone - Objective-C 中的十六进制值

cocoa - KVO 的添加和删除观察者方法——计数还是幂等?

iphone - webkit img 100% 宽度问题

ios - iPhone 无法通过 WiFi 连接到本地 HTTP 服务器

ios - 由于 XCode 6.3 -> Nib 无法加载

iphone - 多个 UIWebViews 问题

javascript - 在 iOS 应用程序的 Parse Cloud Code 上运行 crypto-js 时出现问题

objective-c - 使用 NSTimer 的秒表错误地在显示中包含暂停时间

ios - NSRegularExpression 如何给我一个 NSRange 越界?

swift - 在循环内使用dispatch_after延迟