Рецепт приготовления пакетов Cocoa NSDocument хорошо сочетается с SVN?

У меня есть инструмент, который пишет документы в стиле пакета. Он реализован с использованием NSDocument и переопределяет следующие методы NSDocument:

- (NSFileWrapper *)fileWrapperOfType:(NSString *)typeName
                              error:(NSError **)outError;

- (BOOL)readFromFileWrapper:(NSFileWrapper *)fileWrapper
                    ofType:(NSString *)typeName
                     error:(NSError **)outError;

Это все прекрасно, за исключением случаев, когда я сохраняю документ, который находится под контролем версий. (Каталоги.svn не сохраняются и т. Д.)

Есть ли где-нибудь хороший рецепт, чтобы мои документы хорошо играли с SVN?

2 ответа

Решение

Я предполагаю, что ваш код работает, каждый раз создавая новую обертку файла -fileWrapperOfType:error: называется. Это удобно, но игнорирует любые дополнительные файлы, которые могут существовать на диске внутри пакета.

Так что вместо этого, что если вы начнете с создания / использования оболочки файла, которая ссылается на существующее содержимое на диске. Измените эту оболочку в соответствии с текущим состоянием документа и верните результат. Когда эта оболочка записывается на диск, файлы SVN должны поддерживаться должным образом.

Вот мое решение, основанное на ответе Майка!

Мои пакеты документов являются пакетами, с обычной иерархической структурой... Итак, во время сохранения я изменяю четыре каталога:

  1. Сохранение создает новый верхний уровень (My.bundle)
  2. Каталог содержимого изменен (My.bundle/Contents)
  3. Каталог ресурсов изменен (My.bundle/Contents/Resources)
  4. Локализованные ресурсы обновляются (My.bundle/Contents/Resources/en.lproj)

Рецепт начинается с добавления изменяемого слота словаря в ваш класс документов, чтобы сохранить содержимое каждого из этих каталогов.

@interface LMDocument : NSDocument {
@private
    // All this is for preserving SCM artifacts across saves…
    NSMutableDictionary * bundleWrappers;
    NSMutableDictionary * contentsWrappers;
    NSMutableDictionary * resourcesWrappers;
    NSMutableDictionary * localizedWrappers;
}

При создании нового документа они начинаются как пустые словари.

- (id)init;
{
    if ((self = [super init]) != nil) {
        bundleWrappers = [[NSMutableDictionary alloc] initWithCapacity:0];
        contentsWrappers = [[NSMutableDictionary alloc] initWithCapacity:0];
        resourcesWrappers = [[NSMutableDictionary alloc] initWithCapacity:0];
        localizedWrappers = [[NSMutableDictionary alloc] initWithCapacity:0];
    }
    return self;
}

При чтении в существующем документе замените их изменчивыми копиями соответствующего содержимого fileWrapper.

- (BOOL)readFromFileWrapper:(NSFileWrapper *)fileWrapper
                     ofType:(NSString *)typeName
                      error:(NSError **)outError;
{
    NSAutoreleasePool * pool = [[NSAutoreleasePool alloc] init];
    bundleWrappers = [[fileWrapper fileWrappers] mutableCopy];
    contentsWrappers = [[[bundleWrappers objectForKey:@"Contents"] fileWrappers] mutableCopy];
    resourcesWrappers = [[[contentsWrappers objectForKey:@"Resources"] fileWrappers] mutableCopy];
    localizedWrappers = [[[resourcesWrappers objectForKey:@"en.lproj"] fileWrappers] mutableCopy];
    NSFileWrapper * infoPlistWrapper = [contentsWrappers objectForKey:@"Info.plist"];
    [contentsWrappers removeObjectForKey:@"Info.plist"]; // Replaced during save…
    // …
    NSMutableDictionary * localizedWrappersCopy = [localizedWrappers mutableCopy];
    [localizedWrappers enumerateKeysAndObjectsUsingBlock:^(id key,
                                                           id obj,
                                                           BOOL * stop)
     {
         if (mumble) { // If it's a file that will be replaced during save…
             [localizedWrappersCopy removeObjectForKey:key]; // Replaced during save…
             // …
         }
     }];
    localizedWrappers = localizedWrappersCopy;
    [pool drain];
    return YES;
}

И, наконец, при сохранении документа используйте словари, которые так тщательно подготовлены.

- (NSFileWrapper *)fileWrapperOfType:(NSString *)typeName
                               error:(NSError **)outError;
{
    NSAutoreleasePool * pool = [[NSAutoreleasePool alloc] init];
    NSFileWrapper * localizedWrapper =
    [[NSFileWrapper alloc] initDirectoryWithFileWrappers:localizedWrappers];
    [resourcesWrappers setObject:localizedWrapper
                          forKey:@"en.lproj"];
    NSFileWrapper * resourcesWrapper =
    [[NSFileWrapper alloc] initDirectoryWithFileWrappers:resourcesWrappers];
    [contentsWrappers setObject:resourcesWrapper
                         forKey:@"Resources"];
    NSFileWrapper * contentsWrapper =
    [[NSFileWrapper alloc] initDirectoryWithFileWrappers:contentsWrappers];
    // …
    for (id item in mumble) {
        NSString * filename = [item filename];
        NSData * data = [item data];
        [localizedWrapper addRegularFileWithContents:data
                                   preferredFilename:filename];
    }
    [contentsWrapper addRegularFileWithContents:[self infoPlistData]
                              preferredFilename:@"Info.plist"];
    [pool drain];
    [bundleWrappers setObject:contentsWrapper
                       forKey:@"Contents"];
    NSFileWrapper * bundleWrapper =
    [[[NSFileWrapper alloc] initDirectoryWithFileWrappers:bundleWrappers] autorelease];
    return bundleWrapper;
}

Теперь, когда документ пакета редактируется, приложение сохраняет все файлы, которые не были добавлены в пакет, включая артефакты SCM и "другие" локализации!

Другие вопросы по тегам