// Copyright (c) 2012 The Chromium Authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. #include "net/base/platform_mime_util.h" #import #include #include "base/mac/foundation_util.h" #include "base/mac/scoped_cftyperef.h" #include "base/strings/sys_string_conversions.h" #if defined(OS_IOS) #include #else #include #endif // defined(OS_IOS) #if !defined(OS_IOS) // SPI declaration; see the commentary in GetPlatformExtensionsForMimeType. // iOS must not use any private API, per Apple guideline. @interface NSURLFileTypeMappings : NSObject + (NSURLFileTypeMappings*)sharedMappings; - (NSArray*)extensionsForMIMEType:(NSString*)mimeType; @end #endif // !defined(OS_IOS) namespace net { bool PlatformMimeUtil::GetPlatformMimeTypeFromExtension( const base::FilePath::StringType& ext, std::string* result) const { std::string ext_nodot = ext; if (ext_nodot.length() >= 1 && ext_nodot[0] == L'.') ext_nodot.erase(ext_nodot.begin()); base::ScopedCFTypeRef ext_ref( base::SysUTF8ToCFStringRef(ext_nodot)); if (!ext_ref) return false; base::ScopedCFTypeRef uti(UTTypeCreatePreferredIdentifierForTag( kUTTagClassFilenameExtension, ext_ref, NULL)); if (!uti) return false; base::ScopedCFTypeRef mime_ref( UTTypeCopyPreferredTagWithClass(uti, kUTTagClassMIMEType)); if (!mime_ref) return false; *result = base::SysCFStringRefToUTF8(mime_ref); return true; } bool PlatformMimeUtil::GetPlatformPreferredExtensionForMimeType( const std::string& mime_type, base::FilePath::StringType* ext) const { base::ScopedCFTypeRef mime_ref( base::SysUTF8ToCFStringRef(mime_type)); if (!mime_ref) return false; base::ScopedCFTypeRef uti(UTTypeCreatePreferredIdentifierForTag( kUTTagClassMIMEType, mime_ref, NULL)); if (!uti) return false; base::ScopedCFTypeRef ext_ref( UTTypeCopyPreferredTagWithClass(uti, kUTTagClassFilenameExtension)); if (!ext_ref) return false; *ext = base::SysCFStringRefToUTF8(ext_ref); return true; } void PlatformMimeUtil::GetPlatformExtensionsForMimeType( const std::string& mime_type, std::unordered_set* extensions) const { #if defined(OS_IOS) NSArray* extensions_list = nil; #else // There is no API for this that uses UTIs. The WebKitSystemInterface call // WKGetExtensionsForMIMEType() is a thin wrapper around // [[NSURLFileTypeMappings sharedMappings] extensionsForMIMEType:], which is // used by Firefox as well. // // See: // http://mxr.mozilla.org/mozilla-central/search?string=extensionsForMIMEType // http://www.openradar.me/11384153 // rdar://11384153 NSArray* extensions_list = [[NSURLFileTypeMappings sharedMappings] extensionsForMIMEType:base::SysUTF8ToNSString(mime_type)]; #endif // defined(OS_IOS) if (extensions_list) { for (NSString* extension in extensions_list) extensions->insert(base::SysNSStringToUTF8(extension)); } else { // Huh? Give up. base::FilePath::StringType ext; if (GetPlatformPreferredExtensionForMimeType(mime_type, &ext)) extensions->insert(ext); } } } // namespace net