sim*_*nwh 7 cocoa core-data objective-c nstokenfield
我在弄清楚如何在NSTokenField中表示多对多关系模型时遇到了问题.我有两个(相关)模型:
物品标签
一个项目可以有很多标签,一个标签可以有很多项目.所以这是一种反多种关系.
我想做的是在NSTokenField中表示这些标签.我想最终得到一个令牌字段自动建议匹配(找到一种方法来使用tokenfield:completionsForSubstring:indexOfToken:indexOfSelectedItem)并且能够添加新标记实体(如果它与现有标记不匹配).
好的,希望你还在我身边.我试图用绑定和数组控制器来做所有这些(因为这最有意义,对吧?)
我有一个数组控制器,"项目数组控制器",绑定到我的应用程序委托managedObjectContext.显示所有项目的tableview具有与此阵列控制器的绑定.
我的NSTokenField的值绑定了数组控制器选择键和模型键路径:标签.
使用此配置,NSTokenField将不显示标记.它只是给了我:
<NSTokenFieldCell: 0x10014dc60>: Unknown object type assigned (Relationship objects for {(
<NSManagedObject: 0x10059bdc0> (entity: Tag; id: 0x10016d6e0 <x-coredata://9D77D47A-1171-4397-9777-706F599D7E3B/Tag/p102> ; data: <fault>)
)} on 0x100169660). Ignoring...
Run Code Online (Sandbox Code Playgroud)
这对我来说很有意义,所以不用担心.我查看了一些NSTokenField委托方法,似乎我应该使用:
- (NSString *)tokenField:(NSTokenField *)tokenField displayStringForRepresentedObject:(id)representedObject
Run Code Online (Sandbox Code Playgroud)
问题是,这个方法没有被调用,我得到了和以前一样的错误.
好吧,所以我的下一步是尝试创建一个ValueTransformer.从带有标记实体的数组转换 - >带有字符串的数组(标记名称)都很好.另一种方式更具挑战性.
我试过的是在我的共享应用程序委托管理对象上下文中查找每个名称并返回匹配的标记.这显然给我一个不同的托管对象上下文的问题:
Illegal attempt to establish a relationship 'tags' between objects in different contexts (source = <NSManagedObject: 0x100156900> (entity: Item; id: 0x1003b22b0 <x-coredata://9D77D47A-1171-4397-9777-706F599D7E3B/Item/p106> ; data: {
author = "0x1003b1b30 <x-coredata://9D77D47A-1171-4397-9777-706F599D7E3B/Author/p103>";
createdAt = nil;
filePath = nil;
tags = (
);
title = "Great presentation";
type = "0x1003b1150 <x-coredata://9D77D47A-1171-4397-9777-706F599D7E3B/Type/p104>";
}) , destination = <NSManagedObject: 0x114d08100> (entity: Tag; id: 0x100146b40 <x-coredata://9D77D47A-1171-4397-9777-706F599D7E3B/Tag/p102> ; data: <fault>))
Run Code Online (Sandbox Code Playgroud)
我哪里错了?我该如何解决这个问题?它甚至是正确的方法(我觉得你必须使用ValueTransformer吗?)
提前致谢!
我已经编写了一个自定义NSValueTransformer来在令牌字段的边界NSManagedObject/Tag NSSet和NSString NSArray令牌字段之间进行映射.以下是两种方法:
- (id)transformedValue:(id)value {
if ([value isKindOfClass:[NSSet class]]) {
NSSet *set = (NSSet *)value;
NSMutableArray *ary = [NSMutableArray arrayWithCapacity:[set count]];
for (Tag *tag in [set allObjects]) {
[ary addObject:tag.name];
}
return ary;
}
return nil;
}
- (id)reverseTransformedValue:(id)value {
if ([value isKindOfClass:[NSArray class]]) {
NSArray *ary = (NSArray *)value;
// Check each NSString in the array representing a Tag name if a corresponding
// tag managed object already exists
NSMutableSet *tagSet = [NSMutableSet setWithCapacity:[ary count]];
for (NSString *tagName in ary) {
NSManagedObjectContext *context = [[NSApp delegate] managedObjectContext];
NSFetchRequest *request = [[NSFetchRequest alloc] init];
NSPredicate *searchFilter = [NSPredicate predicateWithFormat:@"name = %@", tagName];
NSEntityDescription *entity = [NSEntityDescription entityForName:[Tag className] inManagedObjectContext:context];
[request setEntity:entity];
[request setPredicate:searchFilter];
NSError *error = nil;
NSArray *results = [context executeFetchRequest:request error:&error];
if ([results count] > 0) {
[tagSet addObjectsFromArray:results];
}
else {
Tag *tag = [[Tag alloc] initWithEntity:entity insertIntoManagedObjectContext:context];
tag.name = tagName;
[tagSet addObject:tag];
[tag release];
}
}
return tagSet;
}
return nil;
}
Run Code Online (Sandbox Code Playgroud)
CoreData似乎在返回时自动建立对象关系(但我尚未完全验证这一点)
希望能帮助到你.
| 归档时间: |
|
| 查看次数: |
1126 次 |
| 最近记录: |