Подтвердить что ты не робот

Можно ли программно изменить значок тома на смонтированном диске в Mac OS X?

Я хочу программно изменить значок тома для сложной файловой системы, реализованной с помощью OSXFUSE (ранее MacFUSE). Значок должен отражать состояние смонтированной файловой системы.

Подход, который я пытался получить, - это сопоставить запросы для /.VolumeIcon.icns с соответствующим значком в наборе приложений. Затем отправьте уведомления об изменении в файловую систему для фактического пути (пути) и пути монтирования (mountPath).

    [[NSWorkspace sharedWorkspace] noteFileSystemChanged: @"/Volumes"]; 
    [[NSWorkspace sharedWorkspace] noteFileSystemChanged: [mountPath stringByDeletingLastPathComponent]];
    [[NSWorkspace sharedWorkspace] noteFileSystemChanged: mountPath];
    [[NSWorkspace sharedWorkspace] noteFileSystemChanged: [path stringByDeletingLastPathComponent]];
    [[NSWorkspace sharedWorkspace] noteFileSystemChanged: path];

    FNNotifyByPath([[[mountPath stringByDeletingLastPathComponent] dataUsingEncoding:NSUTF8StringEncoding] bytes], kFNDirectoryModifiedMessage, kNilOptions);
    FNNotifyByPath([[[path stringByDeletingLastPathComponent] dataUsingEncoding:NSUTF8StringEncoding] bytes], kFNDirectoryModifiedMessage, kNilOptions);
    FNNotifyByPath([[@"/Volumes" dataUsingEncoding:NSUTF8StringEncoding] bytes], kFNDirectoryModifiedMessage, kNilOptions);

Пройдя через отладчик, я вижу, что этот код попадает, но код для отображения /.VolumeIcon.icns вызывается нечасто и никогда не реагирует на эти уведомления.

4b9b3361

Ответ 1

Я думаю, что короткий ответ: вам не повезло. Длительный ответ заключается в том, что проект OSXFUSE отличается от проекта Fuse4X, оба они получены из одного источника, и Fuse4X говорит об этом в своих FAQ:

Q 4.1. Почему объемы Fuse4X отображаются с "сервером" (или "сетью" тома ")?

A: Если быть точным, то по умолчанию объемы Fuse4X отображаются как нелокальные тома, к которым, к сожалению, поисковик относится к" серверам ". Хороший вопрос о том, почему Fuse4X обычно отмечает свои объемы как нелокальной. Некоторые считают, что в случае с дисковым файлом систем, Fuse4X должен маркировать том как локальный. Давайте посмотрим.

Для того, чтобы vfs был локальным в Mac OS X, вам нужно" реальное" устройство на диске - a /dev/disk * style node. Такое реальное дисковое устройство node в корпусе Fuse4X проблематично: во время монтирования для локального тома ядро сам откройте устройство node и передайте его в Fuse4X. При этом ядро будет убедиться, что устройство в настоящее время не используется (для один, чтобы запретить несколько монтировок одного и того же устройства). Бывает до того, как управление перейдет к Fuse4X, и установка может продолжаться. Это были в порядке, если вся файловая система была в ядре, но в Fuse4X, программа файловой системы пользовательского пространства также хотела бы (исключительно) открыть дисковое устройство.

Ответ 2

Посмотрите исходный код поиска пути.

- (BOOL)setAsCustomIconForVolume:(NString *)path;
{
    FSref FSRefpath = convertoFsref(path);
    // filename for custom icon is ".VolumeIcon.icns"
    NSString *iconPath = [path stringByAppendingPathComponent:@".VolumeIcon.icns"];

    // remove any existing file first.

    [self writeToFile:iconPath];
    FSSetHasCustomIcon(FSRefpath);

    // rebuild volumeList


    return YES;
}
OSErr FSSetHasCustomIcon(
                   const FSRef *ref)
{
    return ( FSChangeFinderFlags(ref, true, kHasCustomIcon) );
}
OSErr FSChangeFinderFlags(
                    const FSRef *ref,
                    Boolean setBits,
                    UInt16 flagBits)
{
    OSErr           result;
    FSCatalogInfo   catalogInfo;
    FSRef           parentRef;

    /* get the current finderInfo */
    result = FSGetCatalogInfo(ref, kFSCatInfoFinderInfo, &catalogInfo, NULL, NULL, &parentRef);
    require_noerr(result, FSGetCatalogInfo);

    /* set or clear the appropriate bits in the finderInfo.finderFlags */
    if ( setBits )
    {
        /* OR in the bits */
        ((FileInfo *)&catalogInfo.finderInfo)->finderFlags |= flagBits;
    }
    else
    {
        /* AND out the bits */
        ((FileInfo *)&catalogInfo.finderInfo)->finderFlags &= ~flagBits;
    }

    /* save the modified finderInfo */
    result = FSSetCatalogInfo(ref, kFSCatInfoFinderInfo, &catalogInfo);
    require_noerr(result, FSSetCatalogInfo);

FSSetCatalogInfo:
FSGetCatalogInfo:

        return ( result );
}  

NTVolumeNotificationMgr
NTIconFamily