Home | History | Annotate | Download | only in base
      1 // Copyright (c) 2012 The Chromium Authors. All rights reserved.
      2 // Use of this source code is governed by a BSD-style license that can be
      3 // found in the LICENSE file.
      4 
      5 #include "net/base/platform_mime_util.h"
      6 
      7 #import <Foundation/Foundation.h>
      8 
      9 #include <string>
     10 
     11 #include "base/mac/foundation_util.h"
     12 #include "base/mac/scoped_cftyperef.h"
     13 #include "base/strings/sys_string_conversions.h"
     14 
     15 #if defined(OS_IOS)
     16 #include <MobileCoreServices/MobileCoreServices.h>
     17 #else
     18 #include <CoreServices/CoreServices.h>
     19 #endif  // defined(OS_IOS)
     20 
     21 #if !defined(OS_IOS)
     22 // SPI declaration; see the commentary in GetPlatformExtensionsForMimeType.
     23 // iOS must not use any private API, per Apple guideline.
     24 
     25 @interface NSURLFileTypeMappings : NSObject
     26 + (NSURLFileTypeMappings*)sharedMappings;
     27 - (NSArray*)extensionsForMIMEType:(NSString*)mimeType;
     28 @end
     29 #endif  // !defined(OS_IOS)
     30 
     31 namespace net {
     32 
     33 bool PlatformMimeUtil::GetPlatformMimeTypeFromExtension(
     34     const base::FilePath::StringType& ext, std::string* result) const {
     35   std::string ext_nodot = ext;
     36   if (ext_nodot.length() >= 1 && ext_nodot[0] == L'.')
     37     ext_nodot.erase(ext_nodot.begin());
     38   base::ScopedCFTypeRef<CFStringRef> ext_ref(
     39       base::SysUTF8ToCFStringRef(ext_nodot));
     40   if (!ext_ref)
     41     return false;
     42   base::ScopedCFTypeRef<CFStringRef> uti(UTTypeCreatePreferredIdentifierForTag(
     43       kUTTagClassFilenameExtension, ext_ref, NULL));
     44   if (!uti)
     45     return false;
     46   base::ScopedCFTypeRef<CFStringRef> mime_ref(
     47       UTTypeCopyPreferredTagWithClass(uti, kUTTagClassMIMEType));
     48   if (!mime_ref)
     49     return false;
     50 
     51   *result = base::SysCFStringRefToUTF8(mime_ref);
     52   return true;
     53 }
     54 
     55 bool PlatformMimeUtil::GetPreferredExtensionForMimeType(
     56     const std::string& mime_type, base::FilePath::StringType* ext) const {
     57   base::ScopedCFTypeRef<CFStringRef> mime_ref(
     58       base::SysUTF8ToCFStringRef(mime_type));
     59   if (!mime_ref)
     60     return false;
     61   base::ScopedCFTypeRef<CFStringRef> uti(UTTypeCreatePreferredIdentifierForTag(
     62       kUTTagClassMIMEType, mime_ref, NULL));
     63   if (!uti)
     64     return false;
     65   base::ScopedCFTypeRef<CFStringRef> ext_ref(
     66       UTTypeCopyPreferredTagWithClass(uti, kUTTagClassFilenameExtension));
     67   if (!ext_ref)
     68     return false;
     69 
     70   *ext = base::SysCFStringRefToUTF8(ext_ref);
     71   return true;
     72 }
     73 
     74 void PlatformMimeUtil::GetPlatformExtensionsForMimeType(
     75     const std::string& mime_type,
     76     base::hash_set<base::FilePath::StringType>* extensions) const {
     77 #if defined(OS_IOS)
     78   NSArray* extensions_list = nil;
     79 #else
     80   // There is no API for this that uses UTIs. The WebKitSystemInterface call
     81   // WKGetExtensionsForMIMEType() is a thin wrapper around
     82   // [[NSURLFileTypeMappings sharedMappings] extensionsForMIMEType:], which is
     83   // used by Firefox as well.
     84   //
     85   // See:
     86   // http://mxr.mozilla.org/mozilla-central/search?string=extensionsForMIMEType
     87   // http://www.openradar.me/11384153
     88   // rdar://11384153
     89   NSArray* extensions_list =
     90       [[NSURLFileTypeMappings sharedMappings]
     91           extensionsForMIMEType:base::SysUTF8ToNSString(mime_type)];
     92 #endif  // defined(OS_IOS)
     93 
     94   if (extensions_list) {
     95     for (NSString* extension in extensions_list)
     96       extensions->insert(base::SysNSStringToUTF8(extension));
     97   } else {
     98     // Huh? Give up.
     99     base::FilePath::StringType ext;
    100     if (GetPreferredExtensionForMimeType(mime_type, &ext))
    101       extensions->insert(ext);
    102   }
    103 }
    104 
    105 }  // namespace net
    106