我不知道如何在nstokenfield中表示多对多关系模型。我有两个(相关)模型:
项目
标签
一个项目可以有多个标记,一个标记可以有多个项目。所以这和很多关系是相反的。
我想做的是在nstokenfield中表示这些标记。我想结束一个TutkFieldfield自动建议匹配(找到了一个与Tokfield的方法:FultExoSfFraseNo:DexOfToMe:DexOfSeopDead项目),并且如果它与现有的标签实体不匹配,则可以添加新的标签实体。
好吧,希望你还和我在一起。我试图用绑定和数组控制器来完成这一切(因为这是最有意义的,对吧?)
我有一个数组控制器,“项目数组控制器”,它绑定到我的应用程序委托管理对象上下文。显示所有项的TabLVIEW对该数组控制器具有绑定。
我的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...
这对我来说很有意义,所以不用担心。我看过一些nstokenfield委托方法,似乎应该使用:
- (NSString *)tokenField:(NSTokenField *)tokenField displayStringForRepresentedObject:(id)representedObject
问题是,这个方法没有被调用,并且我得到了和以前一样的错误。
好吧,我的下一步就是做一个价值转换器。从带有标记实体的数组->带有字符串的数组(标记名)转换都很好。另一种方式更具挑战性。
我尝试的是在我的共享应用委托托管对象上下文中查找每个名称并返回匹配的标记。这给了我不同的被管理对象上下文的问题:
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>))
我哪里做错了?我该如何解决?这是不是正确的方法(在我看来,你不得不使用一个值转换器似乎很奇怪?)
提前感谢!
最佳答案
我编写了一个自定义的NSValueTransformer
来映射token字段的绑定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;
}
coredata似乎在返回时自动建立对象关系(但我还没有完全验证这一点)
希望有帮助。
关于objective-c - NSTokenField表示核心数据对多关系,我们在Stack Overflow上找到一个类似的问题:https://stackoverflow.com/questions/3880961/