1 /////////////////////////////////////////////////////////////////////////////
2 // Name: common/mimetype.cpp
3 // Purpose: classes and functions to manage MIME types
4 // Author: Vadim Zeitlin
8 // Copyright: (c) 1998 Vadim Zeitlin <zeitlin@dptmaths.ens-cachan.fr>
9 // Licence: wxWindows license (part of wxExtra library)
10 /////////////////////////////////////////////////////////////////////////////
13 #pragma implementation "mimetype.h"
16 // for compilers that support precompilation, includes "wx.h".
17 #include "wx/wxprec.h"
27 #if (wxUSE_FILE && wxUSE_TEXTFILE) || defined(__WXMSW__)
30 #include "wx/string.h"
34 // Doesn't compile in WIN16 mode
40 #include "wx/dynarray.h"
41 #include "wx/confbase.h"
44 #include "wx/msw/registry.h"
47 #include "wx/textfile.h"
50 #include "wx/mimetype.h"
52 // other standard headers
55 // ----------------------------------------------------------------------------
57 // ----------------------------------------------------------------------------
59 // implementation classes, platform dependent
62 // These classes use Windows registry to retrieve the required information.
64 // Keys used (not all of them are documented, so it might actually stop working
65 // in futur versions of Windows...):
66 // 1. "HKCR\MIME\Database\Content Type" contains subkeys for all known MIME
67 // types, each key has a string value "Extension" which gives (dot preceded)
68 // extension for the files of this MIME type.
70 // 2. "HKCR\.ext" contains
71 // a) unnamed value containing the "filetype"
72 // b) value "Content Type" containing the MIME type
74 // 3. "HKCR\filetype" contains
75 // a) unnamed value containing the description
76 // b) subkey "DefaultIcon" with single unnamed value giving the icon index in
78 // c) shell\open\command and shell\open\print subkeys containing the commands
79 // to open/print the file (the positional parameters are introduced by %1,
80 // %2, ... in these strings, we change them to %s ourselves)
86 wxFileTypeImpl() { m_info
= NULL
; }
88 // one of these Init() function must be called (ctor can't take any
89 // arguments because it's common)
91 // initialize us with our file type name and extension - in this case
92 // we will read all other data from the registry
93 void Init(const wxString
& strFileType
, const wxString
& ext
)
94 { m_strFileType
= strFileType
; m_ext
= ext
; }
96 // initialize us with a wxFileTypeInfo object - it contains all the
98 void Init(const wxFileTypeInfo
& info
)
101 // implement accessor functions
102 bool GetExtensions(wxArrayString
& extensions
);
103 bool GetMimeType(wxString
*mimeType
) const;
104 bool GetIcon(wxIcon
*icon
) const;
105 bool GetDescription(wxString
*desc
) const;
106 bool GetOpenCommand(wxString
*openCmd
,
107 const wxFileType::MessageParameters
& params
) const;
108 bool GetPrintCommand(wxString
*printCmd
,
109 const wxFileType::MessageParameters
& params
) const;
112 // helper function: reads the command corresponding to the specified verb
113 // from the registry (returns an empty string if not found)
114 wxString
GetCommand(const wxChar
*verb
) const;
116 // we use either m_info or read the data from the registry if m_info == NULL
117 const wxFileTypeInfo
*m_info
;
118 wxString m_strFileType
,
122 WX_DECLARE_OBJARRAY(wxFileTypeInfo
, wxArrayFileTypeInfo
);
123 #include "wx/arrimpl.cpp"
124 WX_DEFINE_OBJARRAY(wxArrayFileTypeInfo
);
126 class wxMimeTypesManagerImpl
129 // nothing to do here, we don't load any data but just go and fetch it from
130 // the registry when asked for
131 wxMimeTypesManagerImpl() { }
133 // implement containing class functions
134 wxFileType
*GetFileTypeFromExtension(const wxString
& ext
);
135 wxFileType
*GetFileTypeFromMimeType(const wxString
& mimeType
);
137 // this are NOPs under Windows
138 bool ReadMailcap(const wxString
& filename
, bool fallback
= TRUE
)
140 bool ReadMimeTypes(const wxString
& filename
)
143 void AddFallback(const wxFileTypeInfo
& ft
) { m_fallbacks
.Add(ft
); }
146 wxArrayFileTypeInfo m_fallbacks
;
151 // this class uses both mailcap and mime.types to gather information about file
154 // The information about mailcap file was extracted from metamail(1) sources and
157 // Format of mailcap file: spaces are ignored, each line is either a comment
158 // (starts with '#') or a line of the form <field1>;<field2>;...;<fieldN>.
159 // A backslash can be used to quote semicolons and newlines (and, in fact,
160 // anything else including itself).
162 // The first field is always the MIME type in the form of type/subtype (see RFC
163 // 822) where subtype may be '*' meaning "any". Following metamail, we accept
164 // "type" which means the same as "type/*", although I'm not sure whether this
167 // The second field is always the command to run. It is subject to
168 // parameter/filename expansion described below.
170 // All the following fields are optional and may not be present at all. If
171 // they're present they may appear in any order, although each of them should
172 // appear only once. The optional fields are the following:
173 // * notes=xxx is an uninterpreted string which is silently ignored
174 // * test=xxx is the command to be used to determine whether this mailcap line
175 // applies to our data or not. The RHS of this field goes through the
176 // parameter/filename expansion (as the 2nd field) and the resulting string
177 // is executed. The line applies only if the command succeeds, i.e. returns 0
179 // * print=xxx is the command to be used to print (and not view) the data of
180 // this type (parameter/filename expansion is done here too)
181 // * edit=xxx is the command to open/edit the data of this type
182 // * needsterminal means that a new console must be created for the viewer
183 // * copiousoutput means that the viewer doesn't interact with the user but
184 // produces (possibly) a lof of lines of output on stdout (i.e. "cat" is a
185 // good example), thus it might be a good idea to use some kind of paging
187 // * textualnewlines means not to perform CR/LF translation (not honored)
188 // * compose and composetyped fields are used to determine the program to be
189 // called to create a new message pert in the specified format (unused).
191 // Parameter/filename xpansion:
192 // * %s is replaced with the (full) file name
193 // * %t is replaced with MIME type/subtype of the entry
194 // * for multipart type only %n is replaced with the nnumber of parts and %F is
195 // replaced by an array of (content-type, temporary file name) pairs for all
196 // message parts (TODO)
197 // * %{parameter} is replaced with the value of parameter taken from
198 // Content-type header line of the message.
200 // FIXME any docs with real descriptions of these files??
202 // There are 2 possible formats for mime.types file, one entry per line (used
203 // for global mime.types) and "expanded" format where an entry takes multiple
204 // lines (used for users mime.types).
206 // For both formats spaces are ignored and lines starting with a '#' are
207 // comments. Each record has one of two following forms:
208 // a) for "brief" format:
209 // <mime type> <space separated list of extensions>
210 // b) for "expanded" format:
211 // type=<mime type> \ desc="<description>" \ exts="ext"
213 // We try to autodetect the format of mime.types: if a non-comment line starts
214 // with "type=" we assume the second format, otherwise the first one.
216 // there may be more than one entry for one and the same mime type, to
217 // choose the right one we have to run the command specified in the test
218 // field on our data.
223 MailCapEntry(const wxString
& openCmd
,
224 const wxString
& printCmd
,
225 const wxString
& testCmd
)
226 : m_openCmd(openCmd
), m_printCmd(printCmd
), m_testCmd(testCmd
)
232 const wxString
& GetOpenCmd() const { return m_openCmd
; }
233 const wxString
& GetPrintCmd() const { return m_printCmd
; }
234 const wxString
& GetTestCmd() const { return m_testCmd
; }
236 MailCapEntry
*GetNext() const { return m_next
; }
239 // prepend this element to the list
240 void Prepend(MailCapEntry
*next
) { m_next
= next
; }
241 // insert into the list at given position
242 void Insert(MailCapEntry
*next
, size_t pos
)
247 for ( cur
= next
; cur
!= NULL
; cur
= cur
->m_next
, n
++ ) {
252 wxASSERT_MSG( n
== pos
, T("invalid position in MailCapEntry::Insert") );
254 m_next
= cur
->m_next
;
257 // append this element to the list
258 void Append(MailCapEntry
*next
)
260 wxCHECK_RET( next
!= NULL
, T("Append()ing to what?") );
264 for ( cur
= next
; cur
->m_next
!= NULL
; cur
= cur
->m_next
)
269 wxASSERT_MSG( !m_next
, T("Append()ing element already in the list?") );
273 wxString m_openCmd
, // command to use to open/view the file
275 m_testCmd
; // only apply this entry if test yields
276 // true (i.e. the command returns 0)
278 MailCapEntry
*m_next
; // in the linked list
281 WX_DEFINE_ARRAY(MailCapEntry
*, ArrayTypeEntries
);
283 class wxMimeTypesManagerImpl
285 friend class wxFileTypeImpl
; // give it access to m_aXXX variables
288 // ctor loads all info into memory for quicker access later on
289 // TODO it would be nice to load them all, but parse on demand only...
290 wxMimeTypesManagerImpl();
292 // implement containing class functions
293 wxFileType
*GetFileTypeFromExtension(const wxString
& ext
);
294 wxFileType
*GetFileTypeFromMimeType(const wxString
& mimeType
);
296 bool ReadMailcap(const wxString
& filename
, bool fallback
= FALSE
);
297 bool ReadMimeTypes(const wxString
& filename
);
299 void AddFallback(const wxFileTypeInfo
& filetype
);
301 // add information about the given mimetype
302 void AddMimeTypeInfo(const wxString
& mimetype
,
303 const wxString
& extensions
,
304 const wxString
& description
);
305 void AddMailcapInfo(const wxString
& strType
,
306 const wxString
& strOpenCmd
,
307 const wxString
& strPrintCmd
,
308 const wxString
& strTest
,
309 const wxString
& strDesc
);
312 // get the string containing space separated extensions for the given
314 wxString
GetExtension(size_t index
) { return m_aExtensions
[index
]; }
317 wxArrayString m_aTypes
, // MIME types
318 m_aDescriptions
, // descriptions (just some text)
319 m_aExtensions
; // space separated list of extensions
320 ArrayTypeEntries m_aEntries
; // commands and tests for this file type
326 // initialization functions
327 void Init(wxMimeTypesManagerImpl
*manager
, size_t index
)
328 { m_manager
= manager
; m_index
= index
; }
331 bool GetExtensions(wxArrayString
& extensions
);
332 bool GetMimeType(wxString
*mimeType
) const
333 { *mimeType
= m_manager
->m_aTypes
[m_index
]; return TRUE
; }
334 bool GetIcon(wxIcon
* WXUNUSED(icon
)) const
335 { return FALSE
; } // TODO maybe with Gnome/KDE integration...
336 bool GetDescription(wxString
*desc
) const
337 { *desc
= m_manager
->m_aDescriptions
[m_index
]; return TRUE
; }
339 bool GetOpenCommand(wxString
*openCmd
,
340 const wxFileType::MessageParameters
& params
) const
342 return GetExpandedCommand(openCmd
, params
, TRUE
);
345 bool GetPrintCommand(wxString
*printCmd
,
346 const wxFileType::MessageParameters
& params
) const
348 return GetExpandedCommand(printCmd
, params
, FALSE
);
352 // get the entry which passes the test (may return NULL)
353 MailCapEntry
*GetEntry(const wxFileType::MessageParameters
& params
) const;
355 // choose the correct entry to use and expand the command
356 bool GetExpandedCommand(wxString
*expandedCmd
,
357 const wxFileType::MessageParameters
& params
,
360 wxMimeTypesManagerImpl
*m_manager
;
361 size_t m_index
; // in the wxMimeTypesManagerImpl arrays
366 // ============================================================================
368 // ============================================================================
370 // ----------------------------------------------------------------------------
372 // ----------------------------------------------------------------------------
374 wxFileTypeInfo::wxFileTypeInfo(const char *mimeType
,
376 const char *printCmd
,
379 : m_mimeType(mimeType
),
381 m_printCmd(printCmd
),
385 va_start(argptr
, desc
);
389 const char *ext
= va_arg(argptr
, const char *);
392 // NULL terminates the list
402 // ============================================================================
403 // implementation of the wrapper classes
404 // ============================================================================
406 // ----------------------------------------------------------------------------
408 // ----------------------------------------------------------------------------
410 wxString
wxFileType::ExpandCommand(const wxString
& command
,
411 const wxFileType::MessageParameters
& params
)
413 bool hasFilename
= FALSE
;
416 for ( const wxChar
*pc
= command
.c_str(); *pc
!= T('\0'); pc
++ ) {
417 if ( *pc
== T('%') ) {
420 // '%s' expands into file name (quoted because it might
421 // contain spaces) - except if there are already quotes
422 // there because otherwise some programs may get confused
423 // by double double quotes
425 if ( *(pc
- 2) == T('"') )
426 str
<< params
.GetFileName();
428 str
<< T('"') << params
.GetFileName() << T('"');
430 str
<< params
.GetFileName();
435 // '%t' expands into MIME type (quote it too just to be
437 str
<< T('\'') << params
.GetMimeType() << T('\'');
442 const wxChar
*pEnd
= wxStrchr(pc
, T('}'));
443 if ( pEnd
== NULL
) {
445 wxLogWarning(_("Unmatched '{' in an entry for "
447 params
.GetMimeType().c_str());
451 wxString
param(pc
+ 1, pEnd
- pc
- 1);
452 str
<< T('\'') << params
.GetParamValue(param
) << T('\'');
460 // TODO %n is the number of parts, %F is an array containing
461 // the names of temp files these parts were written to
462 // and their mime types.
466 wxLogDebug(T("Unknown field %%%c in command '%s'."),
467 *pc
, command
.c_str());
476 // metamail(1) man page states that if the mailcap entry doesn't have '%s'
477 // the program will accept the data on stdin: so give it to it!
478 if ( !hasFilename
&& !str
.IsEmpty() ) {
479 str
<< T(" < '") << params
.GetFileName() << T('\'');
485 wxFileType::wxFileType()
487 m_impl
= new wxFileTypeImpl
;
490 wxFileType::~wxFileType()
495 bool wxFileType::GetExtensions(wxArrayString
& extensions
)
497 return m_impl
->GetExtensions(extensions
);
500 bool wxFileType::GetMimeType(wxString
*mimeType
) const
502 return m_impl
->GetMimeType(mimeType
);
505 bool wxFileType::GetIcon(wxIcon
*icon
) const
507 return m_impl
->GetIcon(icon
);
510 bool wxFileType::GetDescription(wxString
*desc
) const
512 return m_impl
->GetDescription(desc
);
516 wxFileType::GetOpenCommand(wxString
*openCmd
,
517 const wxFileType::MessageParameters
& params
) const
519 return m_impl
->GetOpenCommand(openCmd
, params
);
523 wxFileType::GetPrintCommand(wxString
*printCmd
,
524 const wxFileType::MessageParameters
& params
) const
526 return m_impl
->GetPrintCommand(printCmd
, params
);
529 // ----------------------------------------------------------------------------
530 // wxMimeTypesManager
531 // ----------------------------------------------------------------------------
533 bool wxMimeTypesManager::IsOfType(const wxString
& mimeType
,
534 const wxString
& wildcard
)
536 wxASSERT_MSG( mimeType
.Find(T('*')) == wxNOT_FOUND
,
537 T("first MIME type can't contain wildcards") );
539 // all comparaisons are case insensitive (2nd arg of IsSameAs() is FALSE)
540 if ( wildcard
.BeforeFirst(T('/')).IsSameAs(mimeType
.BeforeFirst(T('/')), FALSE
) )
542 wxString strSubtype
= wildcard
.AfterFirst(T('/'));
544 if ( strSubtype
== T("*") ||
545 strSubtype
.IsSameAs(mimeType
.AfterFirst(T('/')), FALSE
) )
547 // matches (either exactly or it's a wildcard)
555 wxMimeTypesManager::wxMimeTypesManager()
557 m_impl
= new wxMimeTypesManagerImpl
;
560 wxMimeTypesManager::~wxMimeTypesManager()
566 wxMimeTypesManager::GetFileTypeFromExtension(const wxString
& ext
)
568 return m_impl
->GetFileTypeFromExtension(ext
);
572 wxMimeTypesManager::GetFileTypeFromMimeType(const wxString
& mimeType
)
574 return m_impl
->GetFileTypeFromMimeType(mimeType
);
577 bool wxMimeTypesManager::ReadMailcap(const wxString
& filename
, bool fallback
)
579 return m_impl
->ReadMailcap(filename
, fallback
);
582 bool wxMimeTypesManager::ReadMimeTypes(const wxString
& filename
)
584 return m_impl
->ReadMimeTypes(filename
);
587 void wxMimeTypesManager::AddFallbacks(const wxFileTypeInfo
*filetypes
)
589 for ( const wxFileTypeInfo
*ft
= filetypes
; ft
->IsValid(); ft
++ ) {
590 m_impl
->AddFallback(*ft
);
594 // ============================================================================
595 // real (OS specific) implementation
596 // ============================================================================
600 wxString
wxFileTypeImpl::GetCommand(const wxChar
*verb
) const
602 // suppress possible error messages
605 strKey
<< m_strFileType
<< T("\\shell\\") << verb
<< T("\\command");
606 wxRegKey
key(wxRegKey::HKCR
, strKey
);
610 // it's the default value of the key
611 if ( key
.QueryValue(T(""), command
) ) {
612 // transform it from '%1' to '%s' style format string
614 // NB: we don't make any attempt to verify that the string is valid,
615 // i.e. doesn't contain %2, or second %1 or .... But we do make
616 // sure that we return a string with _exactly_ one '%s'!
617 bool foundFilename
= FALSE
;
618 size_t len
= command
.Len();
619 for ( size_t n
= 0; (n
< len
) && !foundFilename
; n
++ ) {
620 if ( command
[n
] == T('%') &&
621 (n
+ 1 < len
) && command
[n
+ 1] == T('1') ) {
622 // replace it with '%s'
623 command
[n
+ 1] = T('s');
625 foundFilename
= TRUE
;
629 if ( !foundFilename
) {
630 // we didn't find any '%1'!
631 // HACK: append the filename at the end, hope that it will do
637 // no such file type or no value
642 wxFileTypeImpl::GetOpenCommand(wxString
*openCmd
,
643 const wxFileType::MessageParameters
& params
)
648 cmd
= m_info
->GetOpenCommand();
651 cmd
= GetCommand(T("open"));
654 *openCmd
= wxFileType::ExpandCommand(cmd
, params
);
656 return !openCmd
->IsEmpty();
660 wxFileTypeImpl::GetPrintCommand(wxString
*printCmd
,
661 const wxFileType::MessageParameters
& params
)
666 cmd
= m_info
->GetPrintCommand();
669 cmd
= GetCommand(T("print"));
672 *printCmd
= wxFileType::ExpandCommand(cmd
, params
);
674 return !printCmd
->IsEmpty();
677 // TODO this function is half implemented
678 bool wxFileTypeImpl::GetExtensions(wxArrayString
& extensions
)
681 extensions
= m_info
->GetExtensions();
685 else if ( m_ext
.IsEmpty() ) {
686 // the only way to get the list of extensions from the file type is to
687 // scan through all extensions in the registry - too slow...
692 extensions
.Add(m_ext
);
694 // it's a lie too, we don't return _all_ extensions...
699 bool wxFileTypeImpl::GetMimeType(wxString
*mimeType
) const
702 // we already have it
703 *mimeType
= m_info
->GetMimeType();
708 // suppress possible error messages
710 wxRegKey
key(wxRegKey::HKCR
, /*m_strFileType*/ T(".") + m_ext
);
711 if ( key
.Open() && key
.QueryValue(T("Content Type"), *mimeType
) ) {
719 bool wxFileTypeImpl::GetIcon(wxIcon
*icon
) const
722 // we don't have icons in the fallback resources
727 strIconKey
<< m_strFileType
<< T("\\DefaultIcon");
729 // suppress possible error messages
731 wxRegKey
key(wxRegKey::HKCR
, strIconKey
);
735 // it's the default value of the key
736 if ( key
.QueryValue(T(""), strIcon
) ) {
737 // the format is the following: <full path to file>, <icon index>
738 // NB: icon index may be negative as well as positive and the full
739 // path may contain the environment variables inside '%'
740 wxString strFullPath
= strIcon
.BeforeLast(T(',')),
741 strIndex
= strIcon
.AfterLast(T(','));
743 // index may be omitted, in which case BeforeLast(',') is empty and
744 // AfterLast(',') is the whole string
745 if ( strFullPath
.IsEmpty() ) {
746 strFullPath
= strIndex
;
750 wxString strExpPath
= wxExpandEnvVars(strFullPath
);
751 int nIndex
= wxAtoi(strIndex
);
753 HICON hIcon
= ExtractIcon(GetModuleHandle(NULL
), strExpPath
, nIndex
);
754 switch ( (int)hIcon
) {
755 case 0: // means no icons were found
756 case 1: // means no such file or it wasn't a DLL/EXE/OCX/ICO/...
757 wxLogDebug(T("incorrect registry entry '%s': no such icon."),
758 key
.GetName().c_str());
762 icon
->SetHICON((WXHICON
)hIcon
);
768 // no such file type or no value or incorrect icon entry
772 bool wxFileTypeImpl::GetDescription(wxString
*desc
) const
775 // we already have it
776 *desc
= m_info
->GetDescription();
781 // suppress possible error messages
783 wxRegKey
key(wxRegKey::HKCR
, m_strFileType
);
786 // it's the default value of the key
787 if ( key
.QueryValue(T(""), *desc
) ) {
795 // extension -> file type
797 wxMimeTypesManagerImpl::GetFileTypeFromExtension(const wxString
& ext
)
799 // add the leading point if necessary
801 if ( ext
[0u] != T('.') ) {
806 // suppress possible error messages
809 wxString strFileType
;
810 wxRegKey
key(wxRegKey::HKCR
, str
);
812 // it's the default value of the key
813 if ( key
.QueryValue(T(""), strFileType
) ) {
814 // create the new wxFileType object
815 wxFileType
*fileType
= new wxFileType
;
816 fileType
->m_impl
->Init(strFileType
, ext
);
822 // check the fallbacks
823 // TODO linear search is potentially slow, perhaps we should use a sorted
825 size_t count
= m_fallbacks
.GetCount();
826 for ( size_t n
= 0; n
< count
; n
++ ) {
827 if ( m_fallbacks
[n
].GetExtensions().Index(ext
) != wxNOT_FOUND
) {
828 wxFileType
*fileType
= new wxFileType
;
829 fileType
->m_impl
->Init(m_fallbacks
[n
]);
839 // MIME type -> extension -> file type
841 wxMimeTypesManagerImpl::GetFileTypeFromMimeType(const wxString
& mimeType
)
843 // HACK I don't know of any official documentation which mentions this
844 // location, but as a matter of fact IE uses it, so why not we?
845 static const wxChar
*szMimeDbase
= T("MIME\\Database\\Content Type\\");
847 wxString strKey
= szMimeDbase
;
850 // suppress possible error messages
854 wxRegKey
key(wxRegKey::HKCR
, strKey
);
856 if ( key
.QueryValue(T("Extension"), ext
) ) {
857 return GetFileTypeFromExtension(ext
);
861 // check the fallbacks
862 // TODO linear search is potentially slow, perhaps we should use a sorted
864 size_t count
= m_fallbacks
.GetCount();
865 for ( size_t n
= 0; n
< count
; n
++ ) {
866 if ( wxMimeTypesManager::IsOfType(mimeType
,
867 m_fallbacks
[n
].GetMimeType()) ) {
868 wxFileType
*fileType
= new wxFileType
;
869 fileType
->m_impl
->Init(m_fallbacks
[n
]);
882 wxFileTypeImpl::GetEntry(const wxFileType::MessageParameters
& params
) const
885 MailCapEntry
*entry
= m_manager
->m_aEntries
[m_index
];
886 while ( entry
!= NULL
) {
887 // notice that an empty command would always succeed (it's ok)
888 command
= wxFileType::ExpandCommand(entry
->GetTestCmd(), params
);
890 if ( command
.IsEmpty() || (wxSystem(command
) == 0) ) {
892 wxLogTrace(T("Test '%s' for mime type '%s' succeeded."),
893 command
.c_str(), params
.GetMimeType().c_str());
897 wxLogTrace(T("Test '%s' for mime type '%s' failed."),
898 command
.c_str(), params
.GetMimeType().c_str());
901 entry
= entry
->GetNext();
908 wxFileTypeImpl::GetExpandedCommand(wxString
*expandedCmd
,
909 const wxFileType::MessageParameters
& params
,
912 MailCapEntry
*entry
= GetEntry(params
);
913 if ( entry
== NULL
) {
914 // all tests failed...
918 wxString cmd
= open
? entry
->GetOpenCmd() : entry
->GetPrintCmd();
919 if ( cmd
.IsEmpty() ) {
920 // may happen, especially for "print"
924 *expandedCmd
= wxFileType::ExpandCommand(cmd
, params
);
928 bool wxFileTypeImpl::GetExtensions(wxArrayString
& extensions
)
930 wxString strExtensions
= m_manager
->GetExtension(m_index
);
933 // one extension in the space or comma delimitid list
935 for ( const wxChar
*p
= strExtensions
; ; p
++ ) {
936 if ( *p
== T(' ') || *p
== T(',') || *p
== T('\0') ) {
937 if ( !strExt
.IsEmpty() ) {
938 extensions
.Add(strExt
);
941 //else: repeated spaces (shouldn't happen, but it's not that
942 // important if it does happen)
947 else if ( *p
== T('.') ) {
948 // remove the dot from extension (but only if it's the first char)
949 if ( !strExt
.IsEmpty() ) {
952 //else: no, don't append it
962 // read system and user mailcaps (TODO implement mime.types support)
963 wxMimeTypesManagerImpl::wxMimeTypesManagerImpl()
965 // directories where we look for mailcap and mime.types by default
966 // (taken from metamail(1) sources)
967 static const wxChar
*aStandardLocations
[] =
976 // first read the system wide file(s)
977 for ( size_t n
= 0; n
< WXSIZEOF(aStandardLocations
); n
++ ) {
978 wxString dir
= aStandardLocations
[n
];
980 wxString file
= dir
+ T("/mailcap");
981 if ( wxFile::Exists(file
) ) {
985 file
= dir
+ T("/mime.types");
986 if ( wxFile::Exists(file
) ) {
991 wxString strHome
= wxGetenv(T("HOME"));
993 // and now the users mailcap
994 wxString strUserMailcap
= strHome
+ T("/.mailcap");
995 if ( wxFile::Exists(strUserMailcap
) ) {
996 ReadMailcap(strUserMailcap
);
999 // read the users mime.types
1000 wxString strUserMimeTypes
= strHome
+ T("/.mime.types");
1001 if ( wxFile::Exists(strUserMimeTypes
) ) {
1002 ReadMimeTypes(strUserMimeTypes
);
1007 wxMimeTypesManagerImpl::GetFileTypeFromExtension(const wxString
& ext
)
1009 size_t count
= m_aExtensions
.GetCount();
1010 for ( size_t n
= 0; n
< count
; n
++ ) {
1011 wxString extensions
= m_aExtensions
[n
];
1012 while ( !extensions
.IsEmpty() ) {
1013 wxString field
= extensions
.BeforeFirst(T(' '));
1014 extensions
= extensions
.AfterFirst(T(' '));
1016 // consider extensions as not being case-sensitive
1017 if ( field
.IsSameAs(ext
, FALSE
/* no case */) ) {
1019 wxFileType
*fileType
= new wxFileType
;
1020 fileType
->m_impl
->Init(this, n
);
1032 wxMimeTypesManagerImpl::GetFileTypeFromMimeType(const wxString
& mimeType
)
1034 // mime types are not case-sensitive
1035 wxString
mimetype(mimeType
);
1036 mimetype
.MakeLower();
1038 // first look for an exact match
1039 int index
= m_aTypes
.Index(mimetype
);
1040 if ( index
== wxNOT_FOUND
) {
1041 // then try to find "text/*" as match for "text/plain" (for example)
1042 // NB: if mimeType doesn't contain '/' at all, BeforeFirst() will return
1043 // the whole string - ok.
1044 wxString strCategory
= mimetype
.BeforeFirst(T('/'));
1046 size_t nCount
= m_aTypes
.Count();
1047 for ( size_t n
= 0; n
< nCount
; n
++ ) {
1048 if ( (m_aTypes
[n
].BeforeFirst(T('/')) == strCategory
) &&
1049 m_aTypes
[n
].AfterFirst(T('/')) == T("*") ) {
1056 if ( index
!= wxNOT_FOUND
) {
1057 wxFileType
*fileType
= new wxFileType
;
1058 fileType
->m_impl
->Init(this, index
);
1068 void wxMimeTypesManagerImpl::AddFallback(const wxFileTypeInfo
& filetype
)
1070 wxString extensions
;
1071 const wxArrayString
& exts
= filetype
.GetExtensions();
1072 size_t nExts
= exts
.GetCount();
1073 for ( size_t nExt
= 0; nExt
< nExts
; nExt
++ ) {
1075 extensions
+= T(' ');
1077 extensions
+= exts
[nExt
];
1080 AddMimeTypeInfo(filetype
.GetMimeType(),
1082 filetype
.GetDescription());
1084 AddMailcapInfo(filetype
.GetMimeType(),
1085 filetype
.GetOpenCommand(),
1086 filetype
.GetPrintCommand(),
1088 filetype
.GetDescription());
1091 void wxMimeTypesManagerImpl::AddMimeTypeInfo(const wxString
& strMimeType
,
1092 const wxString
& strExtensions
,
1093 const wxString
& strDesc
)
1095 int index
= m_aTypes
.Index(strMimeType
);
1096 if ( index
== wxNOT_FOUND
) {
1098 m_aTypes
.Add(strMimeType
);
1099 m_aEntries
.Add(NULL
);
1100 m_aExtensions
.Add(strExtensions
);
1101 m_aDescriptions
.Add(strDesc
);
1104 // modify an existing one
1105 if ( !strDesc
.IsEmpty() ) {
1106 m_aDescriptions
[index
] = strDesc
; // replace old value
1108 m_aExtensions
[index
] += ' ' + strExtensions
;
1112 void wxMimeTypesManagerImpl::AddMailcapInfo(const wxString
& strType
,
1113 const wxString
& strOpenCmd
,
1114 const wxString
& strPrintCmd
,
1115 const wxString
& strTest
,
1116 const wxString
& strDesc
)
1118 MailCapEntry
*entry
= new MailCapEntry(strOpenCmd
, strPrintCmd
, strTest
);
1120 int nIndex
= m_aTypes
.Index(strType
);
1121 if ( nIndex
== wxNOT_FOUND
) {
1123 m_aTypes
.Add(strType
);
1125 m_aEntries
.Add(entry
);
1126 m_aExtensions
.Add(T(""));
1127 m_aDescriptions
.Add(strDesc
);
1130 // always append the entry in the tail of the list - info added with
1131 // this function can only come from AddFallbacks()
1132 MailCapEntry
*entryOld
= m_aEntries
[nIndex
];
1134 entry
->Append(entryOld
);
1136 m_aEntries
[nIndex
] = entry
;
1140 bool wxMimeTypesManagerImpl::ReadMimeTypes(const wxString
& strFileName
)
1142 wxLogTrace(T("--- Parsing mime.types file '%s' ---"), strFileName
.c_str());
1144 wxTextFile
file(strFileName
);
1148 // the information we extract
1149 wxString strMimeType
, strDesc
, strExtensions
;
1151 size_t nLineCount
= file
.GetLineCount();
1152 const wxChar
*pc
= NULL
;
1153 for ( size_t nLine
= 0; nLine
< nLineCount
; nLine
++ ) {
1155 // now we're at the start of the line
1156 pc
= file
[nLine
].c_str();
1159 // we didn't finish with the previous line yet
1164 while ( wxIsspace(*pc
) )
1168 if ( *pc
== T('#') ) {
1169 // skip the whole line
1174 // detect file format
1175 const wxChar
*pEqualSign
= wxStrchr(pc
, T('='));
1176 if ( pEqualSign
== NULL
) {
1180 // first field is mime type
1181 for ( strMimeType
.Empty(); !wxIsspace(*pc
) && *pc
!= T('\0'); pc
++ ) {
1186 while ( wxIsspace(*pc
) )
1189 // take all the rest of the string
1192 // no description...
1199 // the string on the left of '=' is the field name
1200 wxString
strLHS(pc
, pEqualSign
- pc
);
1203 for ( pc
= pEqualSign
+ 1; wxIsspace(*pc
); pc
++ )
1207 if ( *pc
== T('"') ) {
1208 // the string is quoted and ends at the matching quote
1209 pEnd
= wxStrchr(++pc
, T('"'));
1210 if ( pEnd
== NULL
) {
1211 wxLogWarning(_("Mime.types file %s, line %d: unterminated "
1213 strFileName
.c_str(), nLine
+ 1);
1217 // unquoted string ends at the first space
1218 for ( pEnd
= pc
; !wxIsspace(*pEnd
); pEnd
++ )
1222 // now we have the RHS (field value)
1223 wxString
strRHS(pc
, pEnd
- pc
);
1225 // check what follows this entry
1226 if ( *pEnd
== T('"') ) {
1231 for ( pc
= pEnd
; wxIsspace(*pc
); pc
++ )
1234 // if there is something left, it may be either a '\\' to continue
1235 // the line or the next field of the same entry
1236 bool entryEnded
= *pc
== T('\0'),
1237 nextFieldOnSameLine
= FALSE
;
1238 if ( !entryEnded
) {
1239 nextFieldOnSameLine
= ((*pc
!= T('\\')) || (pc
[1] != T('\0')));
1242 // now see what we got
1243 if ( strLHS
== T("type") ) {
1244 strMimeType
= strRHS
;
1246 else if ( strLHS
== T("desc") ) {
1249 else if ( strLHS
== T("exts") ) {
1250 strExtensions
= strRHS
;
1253 wxLogWarning(_("Unknown field in file %s, line %d: '%s'."),
1254 strFileName
.c_str(), nLine
+ 1, strLHS
.c_str());
1257 if ( !entryEnded
) {
1258 if ( !nextFieldOnSameLine
)
1260 //else: don't reset it
1262 // as we don't reset strMimeType, the next field in this entry
1263 // will be interpreted correctly.
1269 // although it doesn't seem to be covered by RFCs, some programs
1270 // (notably Netscape) create their entries with several comma
1271 // separated extensions (RFC mention the spaces only)
1272 strExtensions
.Replace(T(","), T(" "));
1274 // also deal with the leading dot
1275 if ( !strExtensions
.IsEmpty() && strExtensions
[0] == T('.') ) {
1276 strExtensions
.erase(0, 1);
1279 AddMimeTypeInfo(strMimeType
, strExtensions
, strDesc
);
1281 // finished with this line
1285 // check our data integriry
1286 wxASSERT( m_aTypes
.Count() == m_aEntries
.Count() &&
1287 m_aTypes
.Count() == m_aExtensions
.Count() &&
1288 m_aTypes
.Count() == m_aDescriptions
.Count() );
1293 bool wxMimeTypesManagerImpl::ReadMailcap(const wxString
& strFileName
,
1296 wxLogTrace(T("--- Parsing mailcap file '%s' ---"), strFileName
.c_str());
1298 wxTextFile
file(strFileName
);
1302 // see the comments near the end of function for the reason we need these
1303 // variables (search for the next occurence of them)
1304 // indices of MIME types (in m_aTypes) we already found in this file
1305 wxArrayInt aEntryIndices
;
1306 // aLastIndices[n] is the index of last element in
1307 // m_aEntries[aEntryIndices[n]] from this file
1308 wxArrayInt aLastIndices
;
1310 size_t nLineCount
= file
.GetLineCount();
1311 for ( size_t nLine
= 0; nLine
< nLineCount
; nLine
++ ) {
1312 // now we're at the start of the line
1313 const wxChar
*pc
= file
[nLine
].c_str();
1316 while ( wxIsspace(*pc
) )
1319 // comment or empty string?
1320 if ( *pc
== T('#') || *pc
== T('\0') )
1325 // what field are we currently in? The first 2 are fixed and there may
1326 // be an arbitrary number of other fields -- currently, we are not
1327 // interested in any of them, but we should parse them as well...
1333 } currentToken
= Field_Type
;
1335 // the flags and field values on the current line
1336 bool needsterminal
= FALSE
,
1337 copiousoutput
= FALSE
;
1343 curField
; // accumulator
1344 for ( bool cont
= TRUE
; cont
; pc
++ ) {
1347 // interpret the next character literally (notice that
1348 // backslash can be used for line continuation)
1349 if ( *++pc
== T('\0') ) {
1350 // fetch the next line.
1352 // pc currently points to nowhere, but after the next
1353 // pc++ in the for line it will point to the beginning
1354 // of the next line in the file
1355 pc
= file
[++nLine
].c_str() - 1;
1358 // just a normal character
1364 cont
= FALSE
; // end of line reached, exit the loop
1369 // store this field and start looking for the next one
1371 // trim whitespaces from both sides
1372 curField
.Trim(TRUE
).Trim(FALSE
);
1374 switch ( currentToken
) {
1377 if ( strType
.Find(T('/')) == wxNOT_FOUND
) {
1378 // we interpret "type" as "type/*"
1382 currentToken
= Field_OpenCmd
;
1386 strOpenCmd
= curField
;
1388 currentToken
= Field_Other
;
1393 // "good" mailcap entry?
1396 // is this something of the form foo=bar?
1397 const wxChar
*pEq
= wxStrchr(curField
, T('='));
1398 if ( pEq
!= NULL
) {
1399 wxString lhs
= curField
.BeforeFirst(T('=')),
1400 rhs
= curField
.AfterFirst(T('='));
1402 lhs
.Trim(TRUE
); // from right
1403 rhs
.Trim(FALSE
); // from left
1405 if ( lhs
== T("print") )
1407 else if ( lhs
== T("test") )
1409 else if ( lhs
== T("description") ) {
1410 // it might be quoted
1411 if ( rhs
[0u] == T('"') &&
1412 rhs
.Last() == T('"') ) {
1413 strDesc
= wxString(rhs
.c_str() + 1,
1420 else if ( lhs
== T("compose") ||
1421 lhs
== T("composetyped") ||
1422 lhs
== T("notes") ||
1430 // no, it's a simple flag
1431 // TODO support the flags:
1432 // 1. create an xterm for 'needsterminal'
1433 // 2. append "| $PAGER" for 'copiousoutput'
1434 if ( curField
== T("needsterminal") )
1435 needsterminal
= TRUE
;
1436 else if ( curField
== T("copiousoutput") )
1437 copiousoutput
= TRUE
;
1438 else if ( curField
== T("textualnewlines") )
1446 // don't flood the user with error messages
1447 // if we don't understand something in his
1448 // mailcap, but give them in debug mode
1449 // because this might be useful for the
1453 T("Mailcap file %s, line %d: unknown "
1454 "field '%s' for the MIME type "
1456 strFileName
.c_str(),
1464 // it already has this value
1465 //currentToken = Field_Other;
1469 wxFAIL_MSG(T("unknown field type in mailcap"));
1472 // next token starts immediately after ';'
1481 // check that we really read something reasonable
1482 if ( currentToken
== Field_Type
|| currentToken
== Field_OpenCmd
) {
1483 wxLogWarning(_("Mailcap file %s, line %d: incomplete entry "
1485 strFileName
.c_str(), nLine
+ 1);
1488 MailCapEntry
*entry
= new MailCapEntry(strOpenCmd
,
1492 // NB: because of complications below (we must get entries priority
1493 // right), we can't use AddMailcapInfo() here, unfortunately.
1494 strType
.MakeLower();
1495 int nIndex
= m_aTypes
.Index(strType
);
1496 if ( nIndex
== wxNOT_FOUND
) {
1498 m_aTypes
.Add(strType
);
1500 m_aEntries
.Add(entry
);
1501 m_aExtensions
.Add(T(""));
1502 m_aDescriptions
.Add(strDesc
);
1505 // modify the existing entry: the entries in one and the same
1506 // file are read in top-to-bottom order, i.e. the entries read
1507 // first should be tried before the entries below. However,
1508 // the files read later should override the settings in the
1509 // files read before (except if fallback is TRUE), thus we
1510 // Insert() the new entry to the list if it has already
1511 // occured in _this_ file, but Prepend() it if it occured in
1512 // some of the previous ones and Append() to it in the
1516 // 'fallback' parameter prevents the entries from this
1517 // file from overriding the other ones - always append
1518 MailCapEntry
*entryOld
= m_aEntries
[nIndex
];
1520 entry
->Append(entryOld
);
1522 m_aEntries
[nIndex
] = entry
;
1525 int entryIndex
= aEntryIndices
.Index(nIndex
);
1526 if ( entryIndex
== wxNOT_FOUND
) {
1527 // first time in this file
1528 aEntryIndices
.Add(nIndex
);
1529 aLastIndices
.Add(0);
1531 entry
->Prepend(m_aEntries
[nIndex
]);
1532 m_aEntries
[nIndex
] = entry
;
1535 // not the first time in _this_ file
1536 size_t nEntryIndex
= (size_t)entryIndex
;
1537 MailCapEntry
*entryOld
= m_aEntries
[nIndex
];
1539 entry
->Insert(entryOld
, aLastIndices
[nEntryIndex
]);
1541 m_aEntries
[nIndex
] = entry
;
1543 // the indices were shifted by 1
1544 aLastIndices
[nEntryIndex
]++;
1548 if ( !strDesc
.IsEmpty() ) {
1549 // replace the old one - what else can we do??
1550 m_aDescriptions
[nIndex
] = strDesc
;
1555 // check our data integriry
1556 wxASSERT( m_aTypes
.Count() == m_aEntries
.Count() &&
1557 m_aTypes
.Count() == m_aExtensions
.Count() &&
1558 m_aTypes
.Count() == m_aDescriptions
.Count() );
1568 // wxUSE_FILE && wxUSE_TEXTFILE