- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
核心数据支持开箱即用的撤消/重做。但它的表现出乎意料。
为了使我的用户界面与我的模型保持同步,我发送了通知。我的用户界面收到通知消息并更新受影响的 View 。
@objc(Entity)
class Entity : NSManagedObject
{
var title : String? {
get {
self.willAccessValueForKey("title")
let text = self.primitiveValueForKey("title") as? String
self.didAccessValueForKey("title")
return text
}
set {
self.willChangeValueForKey("title")
self.setPrimitiveValue(newValue, forKey: "title")
self.didChangeValueForKey("title")
self.sendNotification(self, key:"title")
print("title did change: \(title)")
}
}
}
现在我想向应用程序添加撤消/重做支持。核心数据有一个 NSUndoManager,所以我认为不需要额外的工作。或者至少不多。为了测试这一假设,我制作了一个包含两个 NSTextField 和一个核心数据实体(恰本地命名为 Entity)的测试应用程序。
NSViewController 子类可以访问实体实例(恰本地命名为 testObject)。我观察每次击键以通过 controlTextDidChange: 更新 testObject。
override func controlTextDidChange(obj: NSNotification)
{
guard let value = self.textField?.stringValue else { return }
self.testObject?.setValue(value, forKey: "title")
}
func valueDidChange(sender: Entity, key: String)
{
self.textField?.stringValue = sender.valueForKey("title") as? String ?? ""
}
ManagedObjectContent 和两个文本字段具有相同的 NSUndoManager (调试控制台中的相同指针)。
当我编辑 NSTextField 并执行撤消/重做操作时,NSTextField 和底层 NSManagedObject 属性都保持同步。正如预期的那样。
但是当我将焦点(第一响应者)更改为第二个 NSTextField (没有任何编辑)并撤消/重做操作时,第一个 NSTextField 被(正确)更新,但底层 NSManagedObject 属性没有更新。 title 属性永远不会被调用。
因此第一个 NSTextField 和实体实例在撤消/重做操作后具有不同的值。
更新底层核心数据实例而不是用户界面对我来说更有意义。这里出了什么问题?
旁注:因为我正在观察 NSManagedObject 的任何更改,并且因为 controlTextDidChange: 正在发送通知(因为它更新了 NSManagedObject),所以我收到了对 valueDidChange 的不必要的调用嗯>。有什么技巧可以避免这种情况或者如何改进我的架构?
最佳答案
我做了类似的事情,我发现最好的方法是将 UI Controller 代码(MVC 中的 C)分成两个单独的“路径”。
通过监听来自核心数据模型的通知来观察核心数据模型的变化 NSManagedObjectContextObjectsDidChangeNotification
过滤掉变化是否影响 Controller UI 并相应地调整显示。这条“路径”盲目地跟随 coreData 的变化,不需要与用户交互,也不需要撤消知识。
另一条路径记录了用户请求的变化并相应地修改了核心数据模型。例如,如果我们有一个步进器控件和一个旁边有数字的标签。用户单击步进器。然后 Controller 通过加一或减一来更新核心数据对象上的相关属性。这会使用核心数据模型自动生成撤消操作。如果用户更改影响核心数据中的多个属性,则所有更改都会包含在撤消分组中。然后,对核心数据对象的更改将触发其他 Controller 路径来更新所有 UI 内容(示例中的标签)。
现在撤消功能会自动相反。通过在 MOC 撤消管理器上调用撤消,coreData 将恢复对对象的更改,这将再次触发第一条路径,并且 UI 将自动遵循。
如果用户正在编辑文本字段,我通常不会费心跟踪按键的更改,而是仅在文本字段通知编辑已结束时捕获结果。通过这种方法,编辑后撤消会删除先前编辑 session 中的所有更改,这通常是所需的。如果还需要在文本字段内进行撤消(例如,键入 aa 和 cmd-z 来撤消第二个 a),则可以通过在文本字段编辑时向窗口提供另一个撤消管理器来实现 - 从而避免在同一撤消中进行所有击键撤消堆栈作为核心数据操作。
要记住的一件事是 coreData 有时会等待执行一些操作,这使得事情看起来不同步。在结束撤消分组之前在 MOC 上调用 -processPendingChanges
将解决此问题。
要考虑的另一件事是您要撤消的内容。您是否希望能够撤消用户 key 条目或撤消数据模型中的更改。我有时发现两者,但不是同时发现,因此我发现多个撤消管理器很有用,如前所述。保留文档撤消管理器仅用于对数据模型的更改,这是用户可能长期关心的事情。然后创建一个新的撤消管理器,并在用户处于编辑模式时使用它来跟踪各个按键。一旦用户通过离开文本字段或在对话框中按“确定”等方式确认他对整个编辑感到满意,就丢弃撤消管理器并获取编辑的最终结果并使用文档将其填充到核心数据中撤消管理器。对我来说,这两种类型的撤消是根本不同的,不应该在撤消堆栈中交织在一起。
下面是一些代码,首先是更改监听器的示例(在收到 NSManagedObjectContextObjectsDidChangeNotification
后调用:
-(void)coreDataObjectsUpdated:(NSNotification *)notif {
// Filter for relevant change dicts
NSPredicate *isSectorObject = [NSPredicate predicateWithFormat: @"className == %@", @"Sector"];
NSSet *set;
BOOL changes = NO;
set = [[notif.userInfo objectForKey:NSDeletedObjectsKey] filteredSetUsingPredicate:isSectorObject];
if (set.count > 0) {
changes = YES;
}
else {
set = [[notif.userInfo objectForKey:NSInsertedObjectsKey] filteredSetUsingPredicate:isSectorObject];
if (set.count > 0) {
changes = YES;
}
else {
set = [[notif.userInfo objectForKey:NSUpdatedObjectsKey] filteredSetUsingPredicate:isSectorObject];
if (set.count > 0) {
changes = YES;
}
}
}
if (changes) {
[self.sectorTable reloadData];
}
}
这是创建复合撤消操作的示例,编辑是在单独的工作表中完成的,并且此代码片段将所有更改作为具有名称的单个可撤消操作移动到核心数据对象中。
-(IBAction) editCDObject:(id)sender{
NSManagedObject *stk = [self.objects objectAtIndex:self.objectTableView.clickedRow];
[self.editSheetController EditObject:stk attachToWindow:self.window completionHandler: ^(NSModalResponse returnCode){
if (returnCode == NSModalResponseOK) { // Write back the changes else do nothing
NSUndoManager *um = self.moc.undoManager;
[um beginUndoGrouping];
[um setActionName:[NSString stringWithFormat:@"Edit object"]];
stk.overrideName = self.editSheetController.overrideName;
stk.sector = self.editSheetController.sector;
[um endUndoGrouping];
}
}];
}
希望这能给一些想法。
关于cocoa - 如何将 NSUndoManager 与核心数据结合使用并保持用户界面和模型同步?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/35322590/
我是 Mac OS X 开发新手。在 XCode/Cocoa 开发环境中,每个 Objective-C 项目都以 开始 #import 它工作正常,但我对 Cocoa.h 文件位置感到困惑。我的文件
所以我开始阅读这本书: http://www.amazon.com/Cocoa-Design-Patterns-Erik-Buck/dp/0321535022 第 2 章解释了 MVC 设计模式,并给
我使用下面的代码来访问项目中的变量 appDelegate =(AppDelegate *)[[UIApplication sharedApplication] delegate]; UIApplic
我想从我的表格 View 中拖动一行并将其放入 Mac OS X 10.6 中的任何其他 NSTextField 中,并放置一串文本。 拖放已经在我的应用程序中工作(在 NSTableView 和 N
如何在另一个窗口中加载 Nib ? 我尝试了 initWithWindowName, if (mmController == NULL) mmController = [[mainMenu a
其中一个类使用#import 。所以,在我的Podspec ,我包括 framework Cocoa 我的Podspec是 Pod::Spec.new do |s| s.name
有没有可以让我创建简单的条形图和折线图的框架? 最佳答案 有更新的开源Core Plot。 关于cocoa - cocoa :创建图表,我们在Stack Overflow上找到一个类似的问题: htt
如何从 SIMBL 插件获取主应用程序中的单例?当我尝试调用诸如 [ProcessControl sharedInstance] 之类的内容时,我收到一条错误,指出 ProcessControl 未定
我正在尝试从我的 cocoa 应用程序(通过使用 Mail.app)发送电子邮件中的一些文本。最初我尝试使用 HTML 发送格式正确的文本。但 mailto: URL 不支持 html 标签(即使在设
我正在创建一个应用程序,该应用程序必须与服务器数据交互,然后相应地显示数据库中的结果。我正在用 Cocoa 编写客户端应用程序。 示例:用户登录到 Web 应用程序。他们有一些提交网络报告的选项。选项
我想创建一个可以在多个应用程序中使用的框架(如 coreData、CoreAudio 等)。 任何人都可以发布此链接或教程... 最佳答案 尝试苹果的 Framework Programming Gu
我正在使用 [[NSFontManager sharedFontManager] collectionNames] 获取所有集合,但我看到一些未翻译的字符串(例如“com.apple.AllFonts
我刚刚开始我的 cocoa 教育,我有一个简单的问题,我看到单击一个单词并使用 mac 文本转语音功能的能力表明文本是自动内置的。 (即 - 对于 hello world 应用程序,您可以单击 hel
我需要在加密中实现盐,但要做到这一点,我需要将其存储为我需要创建的文件格式,以便稍后检索它进行解密。在加密方面我是个菜鸟。文件格式规范如下: 密文:密文长度;salt:盐的长度; 然后将密文和盐写出来
有没有办法在Cocoa中以任意的能力创建贝塞尔路径?例如,对于我的应用程序的一部分,我需要一个起伏的单元格。因此,我想使用 10 到 50 个不同的点绘制一条曲线,形成一个循环。这些点将随机波动。我认
我想将声音存储在用户默认值中。既然我们不能直接存储语音,那么存储它的最佳方式是什么?安装新语音后,使用数组 [NSSpeechSynthesizer availableVoices] 中的索引可能会有
我一直在寻找解决方案,但不知道是否可以执行以下操作: 我有一个drawRect方法,我想要做的是将图形元素(例如矩形和线条)添加到当前 View 而不刷新它。我曾经调用 setNeedsDisplay
美好的一天! 我正在为 Mac OS X(不是 iPhone)开发提醒软件。它应该一天一次显示一个窗口。具体时间没有规定,只是一次。我怎样才能做到呢。它会自行注册以在登录后启动。我已经尝试过 NSTi
我只是想知道是否可以创建具有层次结构的下拉或弹出菜单?我目前正在开发的应用程序跟踪作业、类(class)和主题。当用户创建作业时,他们需要能够从下拉列表中选择它所属的类(class),但我也不希望通过
是否有任何 Cocoa Widget 可以用来构建典型的(除了 Interface Builder 中的)GUI 构建器属性检查器,例如 RealBasic 或 Delphi? 是否有一个网站列出了其
我是一名优秀的程序员,十分优秀!