4

我的应用程序是一个自定义格式的查看器,一个带有明确定义的 XML 清单和资源(如图像和电影)的 zip 文件。我使用 zlib 打开内存中的 zip 文件,然后继续显示所述资源。

我遇到的一个问题是我无法正确显示视频,显然是因为 QTMovie 无法确定 mime 类型。从文件([QTMovie movieWithFile])加载的电影效果很好。从内存加载([QTMovie movieWithData])拒绝工作。

这是有道理的,因为缺少文件扩展名,QTMovie 无法确定 mime 类型信息。经过一番搜索后,我采用了以下方式使用 QTDataReference:

NSData *movieData = ...read from memory...;
QTDataReference *movieDataReference = [[QTDataReference alloc] initWithReferenceToData:movieData name:fileName MIMEType:@"video/x-m4v"];
QTMovie *mov = [QTMovie movieWithDataReference:movieDataReference error:&err];

这很好用,但是硬编码 MIMEType 远非理想。我可以访问文件名和扩展名,所以我尝试使用 UTI 找到 mime 类型(感谢#macdev 上的好人):

- (NSString*)mimeTypeForExtension:(NSString*)ext {

    CFStringRef UTI = UTTypeCreatePreferredIdentifierForTag(kUTTagClassFilenameExtension,(CFStringRef)ext,NULL);
    return NSMakeCollectable(UTTypeCopyPreferredTagWithClass((CFStringRef)UTI,kUTTagClassMIMEType));
}

然而这不起作用。显然,在某个地方有一个内部 OS X 扩展数据库和相应的 mime 类型。否则从磁盘电影将无法正常工作。我如何访问它?

谢谢!

4

3 回答 3

8

您遇到的问题是 m4v 和 m4p 没有向 Launch Services 注册的 mime 类型(可能是因为 m4v 和 m4p 的 mime 类型不是标准的)。无论如何,您可能应该做的是处理这样的边缘情况,然后在函数返回时检查 nil (以防扩展既未注册也未由您处理)。

另一件事是您当前的使用正在泄漏内存。我假设您正在使用垃圾收集,但第一次调用会创建一个永远不会释放的 CFString。这是您的方法的建议实现:

-(NSString*)mimeTypeForExtension:(NSString*)ext
{
    NSAssert( ext, @"Extension cannot be nil" );
    NSString* mimeType = nil;

    CFStringRef UTI = UTTypeCreatePreferredIdentifierForTag(kUTTagClassFilenameExtension,
        (CFStringRef)ext, NULL);
    if( !UTI ) return nil;

    CFStringRef registeredType = UTTypeCopyPreferredTagWithClass(UTI, kUTTagClassMIMEType);
    if( !registeredType ) // check for edge case
    {
        if( [ext isEqualToString:@"m4v"] )
            mimeType = @"video/x-m4v";
        else if( [ext isEqualToString:@"m4p"] )
            mimeType = @"audio/x-m4p";
        // handle anything else here that you know is not registered
    } else {
        mimeType = NSMakeCollectable(registeredType);
    }

    CFRelease(UTI);
    return mimeType;
}
于 2008-11-19T22:54:40.593 回答
6

您可以使用 NSWorkspace 让系统猜测文件的 UTI。

-(NSString *)mimeTypeForFileAtPath:(NSString *)path error:(NSError **)err {
  NSString *uti, *mimeType = nil;

  if (!(uti = [[NSWorkspace sharedWorkspace] typeOfFile:path error:err]))
    return nil;
  if (err)
    *err = nil;

  if ((mimeType = (NSString *)UTTypeCopyPreferredTagWithClass((CFStringRef)uti, kUTTagClassMIMEType)))
    mimeType = NSMakeCollectable(mimeType);

  return mimeType;
}
于 2009-11-19T15:05:26.233 回答
1

建议大家把return改成[mimeType autorelease];- 我们中的一些人仍然使用古老的方式!

谢谢!这是一个很大的帮助。

于 2009-07-18T20:34:19.890 回答