2

我有一个带有 NSTableView 的核心数据项目,其中列绑定到 NSArrayController。反过来,控制器的内容绑定到 AppDelegate 的主托管对象上下文。

我对 NSTextFieldCell 进行了细分,以提供一种更加定制化的数据显示方式。一切似乎都运行良好,但是当我枚举上下文中的对象并更改属性时,NSArrayController 似乎没有将新值传递给表。我知道它们正在随着 sortDescriptors 的工作而发生变化,并且表格根据新数据自行组织,但仍显示旧值。我猜这归结为显示数据的 NSCell 的回收:尽管核心数据数据库中的基础值已更改,但它们并没有重新绘制。

在尝试了各种 KVO 通知后[myArrayController willChangeValueFor: aKeyPath],相应didChange的也很简单[myTable reloadData]。我唯一的解决方案是解开控制器的 managedObjectContext 绑定,并在处理后重新钩住它,如下所示:

self.hasBackgroundOp = YES;
self.isCancelled = NO; //can get changed with a 'cancel' button in the UI
    if (self.managedObjectContext.hasChanges) {
        [self. managedObjectContext save:nil];
    }
    [myArrayController setManagedObjectContext:nil]; //brutal un-hooking of controller
    dispatch_group_t dispatchGroup = dispatch_group_create();
    dispatch_group_async(dispatchGroup, dispatch_get_global_queue(0, 0), ^{
        .
        .
        // create a thread context, set the persistent store co-ord
        // and do some processing on the
        // NSManagedObjects that are fetched into an array just
        // for the purpose of this thread
        .
        .
        [ctx save:&error];
        //and after the processing is finished then refresh the table
        dispatch_async(dispatch_get_main_queue(), ^{
            [myArrayController setManagedObjectContext:self.managedObjectContext]; //this forces re-drawing of the table
            self.hasBackgroundOp = NO;
            self.isCancelled = NO;
        });
    });

这样做似乎真的很残酷[myArrayController setManagedObjectContext:nil];,然后再次设置它只是为了刷新表格的内容。我简直不敢相信我已经正确地做到了这一点,尽管它工作得很好。有人可以建议更好的方法吗?谢谢!

更新:实际设置托管对象上下文并不能解决问题。它似乎已经停止工作。如果代码连续运行两次,那么应用程序就会挂起。注释掉这些-setManagedObjectContext:行,代码可以根据需要运行多次。

作为对评论的回应,我的自定义单元格使用以下内容来显示自身:

NSInteger index = [[self stringValue] integerValue] + 1;
NSAttributedString *cellText = [[NSAttributedString alloc] initWithString:[NSString stringWithFormat:@"%lu",index] attributes:attributes];
NSRect textRect;
textRect.origin = cellFrame.origin;
textRect.origin.x += (cellFrame.size.width - cellText.size.width) / 2;
textRect.origin.y += (cellFrame.size.height - cellText.size.height) / 2;
textRect.size.width = cellText.size.width;
textRect.size.height = cellText.size.height;
[cellText drawInRect:textRect];

阵列控制器设置在主 nib 中,并将其托管对象上下文绑定到 AppDelegate 的。nib 中没有获取请求或谓词 - AppDelegate 的代码中绑定了一个排序描述符:

[myArrayController setSortDescriptors:[NSArray arrayWithObject:[NSSortDescriptor sortDescriptorWithKey:kINDEX ascending:YES]]];

抓取控制器选项

4

1 回答 1

1

认为我找到了它 - 我将以下内容添加到线程处理的末尾并在主线程上执行它:

[self.managedObjectContext reset];
[myArrayController fetch:self];

重置 MOC 显然会清除它,从而迫使控制器重新绘制单元格。-fetch似乎是必要的,否则表格将变为空白......

于 2012-12-27T20:36:06.397 回答