gpt4 book ai didi

objective-c - 以编程方式向 Cocoa 核心数据实体添加新记录

转载 作者:行者123 更新时间:2023-12-03 17:30:46 25 4
gpt4 key购买 nike

我正在 Cocoa 中制作一个简单的基于文档的应用程序。该应用程序的每个文档基本上应该管理一组日期和注释,因此每个记录都是一个日期和一个注释( TextView )。此外,每个文档都受密码保护。

为此,我创建了一个名为 HistoryElement 的核心数据实体(包含日期和注释属性),我还创建了一个设置实体,该实体应该只有一条记录,其中包含打开文件的密码(我没有找到了更好的方法,有一个吗?密码与每个文件绑定(bind),因此我无法使用首选项,因为它不是全局应用程序密码)。

我有一个首选项选项卡,其中包含绑定(bind)到“设置”实体的密码属性的“密码”文本字段。

好吧...现在的问题是这样的:当我创建一个新文档时,设置实体上没有记录,所以我希望以编程方式添加一个,以便用户可以放置(如果它想保护它的文件)密码文本字段中的密码。

相反,如果我打开一个现有文件,它应该发现设置实体的记录已经添加,并且不应该再次创建它,而密码文本字段应该使用这个记录。

我尝试了很多方法,但我无法做到这一点。我尝试过这样的例子:

if([[settingsArrayController arrangedObjects] count] == 0) {`

NSLog(@"Init settings");`

[settingsArrayController add:self];`

}

当我创建新文档时,它似乎添加了一条新记录,但是如果我在密码文本字段中输入密码,然后保存文档,当我再次打开文档时 [[settingsArrayControllerarrangedObjects] count] 返回 0 并再次创建一条新记录...

我该怎么做?有更好/简单/优雅的方法来使用密码保护文档吗?

最佳答案

在初始化 ManagedObjectContext 并加载数据后,您需要将代码添加到 NSPersistentDocument 中。如果不知道您发布的示例代码放在哪里,很难说它有什么问题。

您可以将其放置在 windowControllerDidLoadNib 中。例如,

- (void)windowControllerDidLoadNib:(NSWindowController *)windowController 
{
[super windowControllerDidLoadNib:windowController];
// user interface preparation code
NSManagedObjectContext *moc = [self managedObjectContext];
NSSet *settings = [moc fetchObjectsForEntityName:@"Settings"
withPredicate:nil];
if ([settings count] == 0)
{
NSManagedObject *obj = [NSEntityDescription insertNewObjectForEntityForName:@"Settings"
inManagedObjectContext:moc];
[obj setValue:@"myPass" forKey:@"password"];
} else {
// password record already exists, do something else.
}
}

请注意,我在 NSManagedObjectContext 上使用一个类别来执行所有样板查询操作:

// Convenience method to fetch the array of objects for a given Entity
// name in the context, optionally limiting by a predicate or by a predicate
// made from a format NSString and variable arguments.
//
- (NSSet *)fetchObjectsForEntityName:(NSString *)newEntityName
withPredicate:(id)stringOrPredicate, ...
{
NSEntityDescription *entity = [NSEntityDescription
entityForName:newEntityName inManagedObjectContext:self];

NSFetchRequest *request = [[[NSFetchRequest alloc] init] autorelease];
[request setEntity:entity];

if (stringOrPredicate)
{
NSPredicate *predicate;
if ([stringOrPredicate isKindOfClass:[NSString class]])
{
va_list variadicArguments;
va_start(variadicArguments, stringOrPredicate);
predicate = [NSPredicate predicateWithFormat:stringOrPredicate
arguments:variadicArguments];
va_end(variadicArguments);
}
else
{
NSAssert2([stringOrPredicate isKindOfClass:[NSPredicate class]],
@"Second parameter passed to %s is of unexpected class %@",
sel_getName(_cmd), [stringOrPredicate className]);
predicate = (NSPredicate *)stringOrPredicate;
}
[request setPredicate:predicate];
}

NSError *error = nil;
NSArray *results = [self executeFetchRequest:request error:&error];
if (error != nil)
{
[NSException raise:NSGenericException format:@"%@",[error description]];
}

return [NSSet setWithArray:results];
}

这应该就是全部内容了。

我创建了一个快速示例项目来说明您需要执行的操作:http://dl.dropbox.com/u/21359504/coredata-example.zip

关于objective-c - 以编程方式向 Cocoa 核心数据实体添加新记录,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/4962544/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com