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 // ============================================================================
18 // ============================================================================
20 // ----------------------------------------------------------------------------
22 // ----------------------------------------------------------------------------
24 // for compilers that support precompilation, includes "wx.h".
25 #include "wx/wxprec.h"
33 #include "wx/string.h"
37 // Doesn't compile in WIN16 mode
42 #include "wx/dynarray.h"
43 #include "wx/confbase.h"
46 #include "wx/msw/registry.h"
49 #include "wx/textfile.h"
52 #include "wx/mimetype.h"
54 // other standard headers
57 // ----------------------------------------------------------------------------
59 // ----------------------------------------------------------------------------
61 // implementation classes, platform dependent
64 // These classes use Windows registry to retrieve the required information.
66 // Keys used (not all of them are documented, so it might actually stop working
67 // in futur versions of Windows...):
68 // 1. "HKCR\MIME\Database\Content Type" contains subkeys for all known MIME
69 // types, each key has a string value "Extension" which gives (dot preceded)
70 // extension for the files of this MIME type.
72 // 2. "HKCR\.ext" contains
73 // a) unnamed value containing the "filetype"
74 // b) value "Content Type" containing the MIME type
76 // 3. "HKCR\filetype" contains
77 // a) unnamed value containing the description
78 // b) subkey "DefaultIcon" with single unnamed value giving the icon index in
80 // c) shell\open\command and shell\open\print subkeys containing the commands
81 // to open/print the file (the positional parameters are introduced by %1,
82 // %2, ... in these strings, we change them to %s ourselves)
90 // initialize us with our file type name
91 void SetFileType(const wxString
& strFileType
)
92 { m_strFileType
= strFileType
; }
93 void SetExt(const wxString
& ext
)
96 // implement accessor functions
97 bool GetExtensions(wxArrayString
& extensions
);
98 bool GetMimeType(wxString
*mimeType
) const;
99 bool GetIcon(wxIcon
*icon
) const;
100 bool GetDescription(wxString
*desc
) const;
101 bool GetOpenCommand(wxString
*openCmd
,
102 const wxFileType::MessageParameters
&) const
103 { return GetCommand(openCmd
, "open"); }
104 bool GetPrintCommand(wxString
*printCmd
,
105 const wxFileType::MessageParameters
&) const
106 { return GetCommand(printCmd
, "print"); }
110 bool GetCommand(wxString
*command
, const char *verb
) const;
112 wxString m_strFileType
, m_ext
;
115 class wxMimeTypesManagerImpl
118 // nothing to do here, we don't load any data but just go and fetch it from
119 // the registry when asked for
120 wxMimeTypesManagerImpl() { }
122 // implement containing class functions
123 wxFileType
*GetFileTypeFromExtension(const wxString
& ext
);
124 wxFileType
*GetFileTypeFromMimeType(const wxString
& mimeType
);
126 // this are NOPs under Windows
127 void ReadMailcap(const wxString
& filename
) { }
128 void ReadMimeTypes(const wxString
& filename
) { }
133 // this class uses both mailcap and mime.types to gather information about file
136 // The information about mailcap file was extracted from metamail(1) sources and
139 // Format of mailcap file: spaces are ignored, each line is either a comment
140 // (starts with '#') or a line of the form <field1>;<field2>;...;<fieldN>.
141 // A backslash can be used to quote semicolons and newlines (and, in fact,
142 // anything else including itself).
144 // The first field is always the MIME type in the form of type/subtype (see RFC
145 // 822) where subtype may be '*' meaning "any". Following metamail, we accept
146 // "type" which means the same as "type/*", although I'm not sure whether this
149 // The second field is always the command to run. It is subject to
150 // parameter/filename expansion described below.
152 // All the following fields are optional and may not be present at all. If
153 // they're present they may appear in any order, although each of them should
154 // appear only once. The optional fields are the following:
155 // * notes=xxx is an uninterpreted string which is silently ignored
156 // * test=xxx is the command to be used to determine whether this mailcap line
157 // applies to our data or not. The RHS of this field goes through the
158 // parameter/filename expansion (as the 2nd field) and the resulting string
159 // is executed. The line applies only if the command succeeds, i.e. returns 0
161 // * print=xxx is the command to be used to print (and not view) the data of
162 // this type (parameter/filename expansion is done here too)
163 // * edit=xxx is the command to open/edit the data of this type
164 // * needsterminal means that a new console must be created for the viewer
165 // * copiousoutput means that the viewer doesn't interact with the user but
166 // produces (possibly) a lof of lines of output on stdout (i.e. "cat" is a
167 // good example), thus it might be a good idea to use some kind of paging
169 // * textualnewlines means not to perform CR/LF translation (not honored)
170 // * compose and composetyped fields are used to determine the program to be
171 // called to create a new message pert in the specified format (unused).
173 // Parameter/filename xpansion:
174 // * %s is replaced with the (full) file name
175 // * %t is replaced with MIME type/subtype of the entry
176 // * for multipart type only %n is replaced with the nnumber of parts and %F is
177 // replaced by an array of (content-type, temporary file name) pairs for all
178 // message parts (TODO)
179 // * %{parameter} is replaced with the value of parameter taken from
180 // Content-type header line of the message.
182 // FIXME any docs with real descriptions of these files??
184 // There are 2 possible formats for mime.types file, one entry per line (used
185 // for global mime.types) and "expanded" format where an entry takes multiple
186 // lines (used for users mime.types).
188 // For both formats spaces are ignored and lines starting with a '#' are
189 // comments. Each record has one of two following forms:
190 // a) for "brief" format:
191 // <mime type> <space separated list of extensions>
192 // b) for "expanded" format:
193 // type=<mime type> \ desc="<description>" \ exts="ext"
195 // We try to autodetect the format of mime.types: if a non-comment line starts
196 // with "type=" we assume the second format, otherwise the first one.
198 // there may be more than one entry for one and the same mime type, to
199 // choose the right one we have to run the command specified in the test
200 // field on our data.
205 MailCapEntry(const wxString
& openCmd
,
206 const wxString
& printCmd
,
207 const wxString
& testCmd
)
208 : m_openCmd(openCmd
), m_printCmd(printCmd
), m_testCmd(testCmd
)
214 const wxString
& GetOpenCmd() const { return m_openCmd
; }
215 const wxString
& GetPrintCmd() const { return m_printCmd
; }
216 const wxString
& GetTestCmd() const { return m_testCmd
; }
218 MailCapEntry
*GetNext() const { return m_next
; }
221 // prepend this element to the list
222 void Prepend(MailCapEntry
*next
) { m_next
= next
; }
223 // append to the list
224 void Append(MailCapEntry
*next
)
228 for ( cur
= next
; cur
->m_next
!= NULL
; cur
= cur
->m_next
)
233 // we initialize it in the ctor and there is no reason to both Prepend()
234 // and Append() one and the same entry
235 wxASSERT( m_next
== NULL
);
239 wxString m_openCmd
, // command to use to open/view the file
241 m_testCmd
; // only apply this entry if test yields
242 // true (i.e. the command returns 0)
244 MailCapEntry
*m_next
; // in the linked list
247 WX_DEFINE_ARRAY(MailCapEntry
*, ArrayTypeEntries
);
249 class wxMimeTypesManagerImpl
251 friend class wxFileTypeImpl
; // give it access to m_aXXX variables
254 // ctor loads all info into memory for quicker access later on
255 // @@ it would be nice to load them all, but parse on demand only...
256 wxMimeTypesManagerImpl();
258 // implement containing class functions
259 wxFileType
*GetFileTypeFromExtension(const wxString
& ext
);
260 wxFileType
*GetFileTypeFromMimeType(const wxString
& mimeType
);
262 void ReadMailcap(const wxString
& filename
);
263 void ReadMimeTypes(const wxString
& filename
);
266 // get the string containing space separated extensions for the given
268 wxString
GetExtension(size_t index
) { return m_aExtensions
[index
]; }
271 wxArrayString m_aTypes
, // MIME types
272 m_aDescriptions
, // descriptions (just some text)
273 m_aExtensions
; // space separated list of extensions
274 ArrayTypeEntries m_aEntries
; // commands and tests for this file type
280 // initialization functions
281 void Init(wxMimeTypesManagerImpl
*manager
, size_t index
)
282 { m_manager
= manager
; m_index
= index
; }
285 bool GetExtensions(wxArrayString
& extensions
);
286 bool GetMimeType(wxString
*mimeType
) const
287 { *mimeType
= m_manager
->m_aTypes
[m_index
]; return TRUE
; }
288 bool GetIcon(wxIcon
* WXUNUSED(icon
)) const
289 { return FALSE
; } // @@ maybe with Gnome/KDE integration...
290 bool GetDescription(wxString
*desc
) const
291 { *desc
= m_manager
->m_aDescriptions
[m_index
]; return TRUE
; }
293 bool GetOpenCommand(wxString
*openCmd
,
294 const wxFileType::MessageParameters
& params
) const
296 return GetExpandedCommand(openCmd
, params
, TRUE
);
299 bool GetPrintCommand(wxString
*printCmd
,
300 const wxFileType::MessageParameters
& params
) const
302 return GetExpandedCommand(printCmd
, params
, FALSE
);
306 // get the entry which passes the test (may return NULL)
307 MailCapEntry
*GetEntry(const wxFileType::MessageParameters
& params
) const;
309 // choose the correct entry to use and expand the command
310 bool GetExpandedCommand(wxString
*expandedCmd
,
311 const wxFileType::MessageParameters
& params
,
314 wxMimeTypesManagerImpl
*m_manager
;
315 size_t m_index
; // in the wxMimeTypesManagerImpl arrays
320 // ============================================================================
321 // implementation of the wrapper classes
322 // ============================================================================
324 // ----------------------------------------------------------------------------
326 // ----------------------------------------------------------------------------
328 wxString
wxFileType::ExpandCommand(const wxString
& command
,
329 const wxFileType::MessageParameters
& params
)
331 bool hasFilename
= FALSE
;
334 for ( const char *pc
= command
.c_str(); *pc
!= '\0'; pc
++ ) {
338 // '%s' expands into file name (quoted because it might
339 // contain spaces) - except if there are already quotes
340 // there because otherwise some programs may get confused
341 // by double double quotes
343 if ( *(pc
- 2) == '"' )
344 str
<< params
.GetFileName();
346 str
<< '"' << params
.GetFileName() << '"';
348 str
<< params
.GetFileName();
353 // '%t' expands into MIME type (quote it too just to be
355 str
<< '\'' << params
.GetMimeType() << '\'';
360 const char *pEnd
= strchr(pc
, '}');
361 if ( pEnd
== NULL
) {
363 wxLogWarning(_("Unmatched '{' in an entry for "
365 params
.GetMimeType().c_str());
369 wxString
param(pc
+ 1, pEnd
- pc
- 1);
370 str
<< '\'' << params
.GetParamValue(param
) << '\'';
378 // TODO %n is the number of parts, %F is an array containing
379 // the names of temp files these parts were written to
380 // and their mime types.
384 wxLogDebug("Unknown field %%%c in command '%s'.",
385 *pc
, command
.c_str());
394 // metamail(1) man page states that if the mailcap entry doesn't have '%s'
395 // the program will accept the data on stdin: so give it to it!
396 if ( !hasFilename
&& !str
.IsEmpty() ) {
397 str
<< " < '" << params
.GetFileName() << '\'';
403 wxFileType::wxFileType()
405 m_impl
= new wxFileTypeImpl
;
408 wxFileType::~wxFileType()
413 bool wxFileType::GetExtensions(wxArrayString
& extensions
)
415 return m_impl
->GetExtensions(extensions
);
418 bool wxFileType::GetMimeType(wxString
*mimeType
) const
420 return m_impl
->GetMimeType(mimeType
);
423 bool wxFileType::GetIcon(wxIcon
*icon
) const
425 return m_impl
->GetIcon(icon
);
428 bool wxFileType::GetDescription(wxString
*desc
) const
430 return m_impl
->GetDescription(desc
);
434 wxFileType::GetOpenCommand(wxString
*openCmd
,
435 const wxFileType::MessageParameters
& params
) const
437 return m_impl
->GetOpenCommand(openCmd
, params
);
441 wxFileType::GetPrintCommand(wxString
*printCmd
,
442 const wxFileType::MessageParameters
& params
) const
444 return m_impl
->GetPrintCommand(printCmd
, params
);
447 // ----------------------------------------------------------------------------
448 // wxMimeTypesManager
449 // ----------------------------------------------------------------------------
451 bool wxMimeTypesManager::IsOfType(const wxString
& mimeType
,
452 const wxString
& wildcard
)
454 wxASSERT_MSG( mimeType
.Find('*') == wxNOT_FOUND
,
455 "first MIME type can't contain wildcards" );
457 // all comparaisons are case insensitive (2nd arg of IsSameAs() is FALSE)
458 if ( wildcard
.BeforeFirst('/').IsSameAs(mimeType
.BeforeFirst('/'), FALSE
) )
460 wxString strSubtype
= wildcard
.AfterFirst('/');
462 if ( strSubtype
== '*' ||
463 strSubtype
.IsSameAs(mimeType
.AfterFirst('/'), FALSE
) )
465 // matches (either exactly or it's a wildcard)
473 wxMimeTypesManager::wxMimeTypesManager()
475 m_impl
= new wxMimeTypesManagerImpl
;
478 wxMimeTypesManager::~wxMimeTypesManager()
484 wxMimeTypesManager::GetFileTypeFromExtension(const wxString
& ext
)
486 return m_impl
->GetFileTypeFromExtension(ext
);
490 wxMimeTypesManager::GetFileTypeFromMimeType(const wxString
& mimeType
)
492 return m_impl
->GetFileTypeFromMimeType(mimeType
);
495 void wxMimeTypesManager::ReadMailcap(const wxString
& filename
)
497 m_impl
->ReadMailcap(filename
);
500 void wxMimeTypesManager::ReadMimeTypes(const wxString
& filename
)
502 m_impl
->ReadMimeTypes(filename
);
505 // ============================================================================
506 // real (OS specific) implementation
507 // ============================================================================
511 bool wxFileTypeImpl::GetCommand(wxString
*command
, const char *verb
) const
513 // suppress possible error messages
516 strKey
<< m_strFileType
<< "\\shell\\" << verb
<< "\\command";
517 wxRegKey
key(wxRegKey::HKCR
, strKey
);
520 // it's the default value of the key
521 if ( key
.QueryValue("", *command
) ) {
522 // transform it from '%1' to '%s' style format string
523 // @@ we don't make any attempt to verify that the string is valid,
524 // i.e. doesn't contain %2, or second %1 or .... But we do make
525 // sure that we return a string with _exactly_ one '%s'!
526 size_t len
= command
->Len();
527 for ( size_t n
= 0; n
< len
; n
++ ) {
528 if ( command
->GetChar(n
) == '%' &&
529 (n
+ 1 < len
) && command
->GetChar(n
+ 1) == '1' ) {
530 // replace it with '%s'
531 command
->SetChar(n
+ 1, 's');
537 // we didn't find any '%1'!
538 // @@@ hack: append the filename at the end, hope that it will do
545 // no such file type or no value
549 // @@ this function is half implemented
550 bool wxFileTypeImpl::GetExtensions(wxArrayString
& extensions
)
552 if ( m_ext
.IsEmpty() ) {
553 // the only way to get the list of extensions from the file type is to
554 // scan through all extensions in the registry - too slow...
559 extensions
.Add(m_ext
);
561 // it's a lie too, we don't return _all_ extensions...
566 bool wxFileTypeImpl::GetMimeType(wxString
*mimeType
) const
568 // suppress possible error messages
570 wxRegKey
key(wxRegKey::HKCR
, /*m_strFileType*/ "." + m_ext
);
571 if ( key
.Open() && key
.QueryValue("Content Type", *mimeType
) ) {
579 bool wxFileTypeImpl::GetIcon(wxIcon
*icon
) const
582 strIconKey
<< m_strFileType
<< "\\DefaultIcon";
584 // suppress possible error messages
586 wxRegKey
key(wxRegKey::HKCR
, strIconKey
);
590 // it's the default value of the key
591 if ( key
.QueryValue("", strIcon
) ) {
592 // the format is the following: <full path to file>, <icon index>
593 // NB: icon index may be negative as well as positive and the full
594 // path may contain the environment variables inside '%'
595 wxString strFullPath
= strIcon
.BeforeLast(','),
596 strIndex
= strIcon
.AfterLast(',');
598 // index may be omitted, in which case BeforeLast(',') is empty and
599 // AfterLast(',') is the whole string
600 if ( strFullPath
.IsEmpty() ) {
601 strFullPath
= strIndex
;
605 wxString strExpPath
= wxExpandEnvVars(strFullPath
);
606 int nIndex
= atoi(strIndex
);
608 HICON hIcon
= ExtractIcon(GetModuleHandle(NULL
), strExpPath
, nIndex
);
609 switch ( (int)hIcon
) {
610 case 0: // means no icons were found
611 case 1: // means no such file or it wasn't a DLL/EXE/OCX/ICO/...
612 wxLogDebug("incorrect registry entry '%s': no such icon.",
613 key
.GetName().c_str());
617 icon
->SetHICON((WXHICON
)hIcon
);
623 // no such file type or no value or incorrect icon entry
627 bool wxFileTypeImpl::GetDescription(wxString
*desc
) const
629 // suppress possible error messages
631 wxRegKey
key(wxRegKey::HKCR
, m_strFileType
);
634 // it's the default value of the key
635 if ( key
.QueryValue("", *desc
) ) {
643 // extension -> file type
645 wxMimeTypesManagerImpl::GetFileTypeFromExtension(const wxString
& ext
)
647 // add the leading point if necessary
649 if ( ext
[0u] != '.' ) {
654 // suppress possible error messages
657 wxString strFileType
;
658 wxRegKey
key(wxRegKey::HKCR
, str
);
660 // it's the default value of the key
661 if ( key
.QueryValue("", strFileType
) ) {
662 // create the new wxFileType object
663 wxFileType
*fileType
= new wxFileType
;
664 fileType
->m_impl
->SetFileType(strFileType
);
665 fileType
->m_impl
->SetExt(ext
);
675 // MIME type -> extension -> file type
677 wxMimeTypesManagerImpl::GetFileTypeFromMimeType(const wxString
& mimeType
)
679 // @@@ I don't know of any official documentation which mentions this
680 // location, but as a matter of fact IE uses it, so why not we?
681 static const char *szMimeDbase
= "MIME\\Database\\Content Type\\";
683 wxString strKey
= szMimeDbase
;
686 // suppress possible error messages
690 wxRegKey
key(wxRegKey::HKCR
, strKey
);
692 if ( key
.QueryValue("Extension", ext
) ) {
693 return GetFileTypeFromExtension(ext
);
704 wxFileTypeImpl::GetEntry(const wxFileType::MessageParameters
& params
) const
707 MailCapEntry
*entry
= m_manager
->m_aEntries
[m_index
];
708 while ( entry
!= NULL
) {
709 // notice that an empty command would always succeed (@@ is it ok?)
710 command
= wxFileType::ExpandCommand(entry
->GetTestCmd(), params
);
712 if ( command
.IsEmpty() || (system(command
) == 0) ) {
714 wxLogTrace("Test '%s' for mime type '%s' succeeded.",
715 command
.c_str(), params
.GetMimeType().c_str());
719 wxLogTrace("Test '%s' for mime type '%s' failed.",
720 command
.c_str(), params
.GetMimeType().c_str());
723 entry
= entry
->GetNext();
730 wxFileTypeImpl::GetExpandedCommand(wxString
*expandedCmd
,
731 const wxFileType::MessageParameters
& params
,
734 MailCapEntry
*entry
= GetEntry(params
);
735 if ( entry
== NULL
) {
736 // all tests failed...
740 wxString cmd
= open
? entry
->GetOpenCmd() : entry
->GetPrintCmd();
741 if ( cmd
.IsEmpty() ) {
742 // may happen, especially for "print"
746 *expandedCmd
= wxFileType::ExpandCommand(cmd
, params
);
750 bool wxFileTypeImpl::GetExtensions(wxArrayString
& extensions
)
752 wxString strExtensions
= m_manager
->GetExtension(m_index
);
755 // one extension in the space or comma delimitid list
757 for ( const char *p
= strExtensions
; ; p
++ ) {
758 if ( *p
== ' ' || *p
== ',' || *p
== '\0' ) {
759 if ( !strExt
.IsEmpty() ) {
760 extensions
.Add(strExt
);
763 //else: repeated spaces (shouldn't happen, but it's not that
764 // important if it does happen)
769 else if ( *p
== '.' ) {
770 // remove the dot from extension (but only if it's the first char)
771 if ( !strExt
.IsEmpty() ) {
774 //else: no, don't append it
784 // read system and user mailcaps (TODO implement mime.types support)
785 wxMimeTypesManagerImpl::wxMimeTypesManagerImpl()
787 // directories where we look for mailcap and mime.types by default
788 // (taken from metamail(1) sources)
789 static const char *aStandardLocations
[] =
798 // first read the system wide file(s)
799 for ( size_t n
= 0; n
< WXSIZEOF(aStandardLocations
); n
++ ) {
800 wxString dir
= aStandardLocations
[n
];
802 wxString file
= dir
+ "/mailcap";
803 if ( wxFile::Exists(file
) ) {
807 file
= dir
+ "/mime.types";
808 if ( wxFile::Exists(file
) ) {
813 wxString strHome
= getenv("HOME");
815 // and now the users mailcap
816 wxString strUserMailcap
= strHome
+ "/.mailcap";
817 if ( wxFile::Exists(strUserMailcap
) ) {
818 ReadMailcap(strUserMailcap
);
821 // read the users mime.types
822 wxString strUserMimeTypes
= strHome
+ "/.mime.types";
823 if ( wxFile::Exists(strUserMimeTypes
) ) {
824 ReadMimeTypes(strUserMimeTypes
);
829 wxMimeTypesManagerImpl::GetFileTypeFromExtension(const wxString
& ext
)
831 size_t count
= m_aExtensions
.GetCount();
832 for ( size_t n
= 0; n
< count
; n
++ ) {
833 wxString extensions
= m_aExtensions
[n
];
834 while ( !extensions
.IsEmpty() ) {
835 wxString field
= extensions
.BeforeFirst(' ');
836 extensions
= extensions
.AfterFirst(' ');
838 // consider extensions as not being case-sensitive
839 if ( field
.IsSameAs(ext
, FALSE
/* no case */) ) {
841 wxFileType
*fileType
= new wxFileType
;
842 fileType
->m_impl
->Init(this, n
);
854 wxMimeTypesManagerImpl::GetFileTypeFromMimeType(const wxString
& mimeType
)
856 // mime types are not case-sensitive
857 wxString
mimetype(mimeType
);
858 mimetype
.MakeLower();
860 // first look for an exact match
861 int index
= m_aTypes
.Index(mimetype
);
862 if ( index
== wxNOT_FOUND
) {
863 // then try to find "text/*" as match for "text/plain" (for example)
864 // NB: if mimeType doesn't contain '/' at all, BeforeFirst() will return
865 // the whole string - ok.
866 wxString strCategory
= mimetype
.BeforeFirst('/');
868 size_t nCount
= m_aTypes
.Count();
869 for ( size_t n
= 0; n
< nCount
; n
++ ) {
870 if ( (m_aTypes
[n
].BeforeFirst('/') == strCategory
) &&
871 m_aTypes
[n
].AfterFirst('/') == "*" ) {
878 if ( index
!= wxNOT_FOUND
) {
879 wxFileType
*fileType
= new wxFileType
;
880 fileType
->m_impl
->Init(this, index
);
890 void wxMimeTypesManagerImpl::ReadMimeTypes(const wxString
& strFileName
)
892 wxLogTrace("--- Parsing mime.types file '%s' ---", strFileName
.c_str());
894 wxTextFile
file(strFileName
);
898 // the information we extract
899 wxString strMimeType
, strDesc
, strExtensions
;
901 size_t nLineCount
= file
.GetLineCount();
902 const char *pc
= NULL
;
903 for ( size_t nLine
= 0; nLine
< nLineCount
; nLine
++ ) {
905 // now we're at the start of the line
906 pc
= file
[nLine
].c_str();
909 // we didn't finish with the previous line yet
914 while ( isspace(*pc
) )
919 // skip the whole line
924 // detect file format
925 const char *pEqualSign
= strchr(pc
, '=');
926 if ( pEqualSign
== NULL
) {
930 // first field is mime type
931 for ( strMimeType
.Empty(); !isspace(*pc
) && *pc
!= '\0'; pc
++ ) {
936 while ( isspace(*pc
) )
939 // take all the rest of the string
949 // the string on the left of '=' is the field name
950 wxString
strLHS(pc
, pEqualSign
- pc
);
953 for ( pc
= pEqualSign
+ 1; isspace(*pc
); pc
++ )
958 // the string is quoted and ends at the matching quote
959 pEnd
= strchr(++pc
, '"');
960 if ( pEnd
== NULL
) {
961 wxLogWarning(_("Mime.types file %s, line %d: unterminated "
963 strFileName
.c_str(), nLine
+ 1);
967 // unquoted string ends at the first space
968 for ( pEnd
= pc
; !isspace(*pEnd
); pEnd
++ )
972 // now we have the RHS (field value)
973 wxString
strRHS(pc
, pEnd
- pc
);
975 // check what follows this entry
976 if ( *pEnd
== '"' ) {
981 for ( pc
= pEnd
; isspace(*pc
); pc
++ )
984 // if there is something left, it may be either a '\\' to continue
985 // the line or the next field of the same entry
986 bool entryEnded
= *pc
== '\0',
987 nextFieldOnSameLine
= FALSE
;
989 nextFieldOnSameLine
= ((*pc
!= '\\') || (pc
[1] != '\0'));
992 // now see what we got
993 if ( strLHS
== "type" ) {
994 strMimeType
= strRHS
;
996 else if ( strLHS
== "desc" ) {
999 else if ( strLHS
== "exts" ) {
1000 strExtensions
= strRHS
;
1003 wxLogWarning(_("Unknown field in file %s, line %d: '%s'."),
1004 strFileName
.c_str(), nLine
+ 1, strLHS
.c_str());
1007 if ( !entryEnded
) {
1008 if ( !nextFieldOnSameLine
)
1010 //else: don't reset it
1012 // as we don't reset strMimeType, the next field in this entry
1013 // will be interpreted correctly.
1019 // although it doesn't seem to be covered by RFCs, some programs
1020 // (notably Netscape) create their entries with several comma
1021 // separated extensions (RFC mention the spaces only)
1022 strExtensions
.Replace(",", " ");
1024 // also deal with the leading dot
1025 if ( !strExtensions
.IsEmpty() && strExtensions
[0] == '.' ) {
1026 strExtensions
.erase(0, 1);
1029 int index
= m_aTypes
.Index(strMimeType
);
1030 if ( index
== wxNOT_FOUND
) {
1032 m_aTypes
.Add(strMimeType
);
1033 m_aEntries
.Add(NULL
);
1034 m_aExtensions
.Add(strExtensions
);
1035 m_aDescriptions
.Add(strDesc
);
1038 // modify an existing one
1039 if ( !strDesc
.IsEmpty() ) {
1040 m_aDescriptions
[index
] = strDesc
; // replace old value
1042 m_aExtensions
[index
] += strExtensions
;
1045 // finished with this line
1049 // check our data integriry
1050 wxASSERT( m_aTypes
.Count() == m_aEntries
.Count() &&
1051 m_aTypes
.Count() == m_aExtensions
.Count() &&
1052 m_aTypes
.Count() == m_aDescriptions
.Count() );
1055 void wxMimeTypesManagerImpl::ReadMailcap(const wxString
& strFileName
)
1057 wxLogTrace("--- Parsing mailcap file '%s' ---", strFileName
.c_str());
1059 wxTextFile
file(strFileName
);
1063 // see the comments near the end of function for the reason we need this
1064 wxArrayInt aEntryIndices
;
1066 size_t nLineCount
= file
.GetLineCount();
1067 for ( size_t nLine
= 0; nLine
< nLineCount
; nLine
++ ) {
1068 // now we're at the start of the line
1069 const char *pc
= file
[nLine
].c_str();
1072 while ( isspace(*pc
) )
1075 // comment or empty string?
1076 if ( *pc
== '#' || *pc
== '\0' )
1081 // what field are we currently in? The first 2 are fixed and there may
1082 // be an arbitrary number of other fields -- currently, we are not
1083 // interested in any of them, but we should parse them as well...
1089 } currentToken
= Field_Type
;
1091 // the flags and field values on the current line
1092 bool needsterminal
= FALSE
,
1093 copiousoutput
= FALSE
;
1099 curField
; // accumulator
1100 for ( bool cont
= TRUE
; cont
; pc
++ ) {
1103 // interpret the next character literally (notice that
1104 // backslash can be used for line continuation)
1105 if ( *++pc
== '\0' ) {
1106 // fetch the next line.
1108 // pc currently points to nowhere, but after the next
1109 // pc++ in the for line it will point to the beginning
1110 // of the next line in the file
1111 pc
= file
[++nLine
].c_str() - 1;
1114 // just a normal character
1120 cont
= FALSE
; // end of line reached, exit the loop
1125 // store this field and start looking for the next one
1127 // trim whitespaces from both sides
1128 curField
.Trim(TRUE
).Trim(FALSE
);
1130 switch ( currentToken
) {
1133 if ( strType
.Find('/') == wxNOT_FOUND
) {
1134 // we interpret "type" as "type/*"
1138 currentToken
= Field_OpenCmd
;
1142 strOpenCmd
= curField
;
1144 currentToken
= Field_Other
;
1149 // "good" mailcap entry?
1152 // is this something of the form foo=bar?
1153 const char *pEq
= strchr(curField
, '=');
1154 if ( pEq
!= NULL
) {
1155 wxString lhs
= curField
.BeforeFirst('='),
1156 rhs
= curField
.AfterFirst('=');
1158 lhs
.Trim(TRUE
); // from right
1159 rhs
.Trim(FALSE
); // from left
1161 if ( lhs
== "print" )
1163 else if ( lhs
== "test" )
1165 else if ( lhs
== "description" ) {
1166 // it might be quoted
1167 if ( rhs
[0u] == '"' &&
1168 rhs
.Last() == '"' ) {
1169 strDesc
= wxString(rhs
.c_str() + 1,
1176 else if ( lhs
== "compose" ||
1177 lhs
== "composetyped" ||
1186 // no, it's a simple flag
1187 // TODO support the flags:
1188 // 1. create an xterm for 'needsterminal'
1189 // 2. append "| $PAGER" for 'copiousoutput'
1190 if ( curField
== "needsterminal" )
1191 needsterminal
= TRUE
;
1192 else if ( curField
== "copiousoutput" )
1193 copiousoutput
= TRUE
;
1194 else if ( curField
== "textualnewlines" )
1202 // don't flood the user with error messages
1203 // if we don't understand something in his
1204 // mailcap, but give them in debug mode
1205 // because this might be useful for the
1209 "Mailcap file %s, line %d: unknown "
1210 "field '%s' for the MIME type "
1212 strFileName
.c_str(),
1220 // it already has this value
1221 //currentToken = Field_Other;
1225 wxFAIL_MSG("unknown field type in mailcap");
1228 // next token starts immediately after ';'
1237 // check that we really read something reasonable
1238 if ( currentToken
== Field_Type
|| currentToken
== Field_OpenCmd
) {
1239 wxLogWarning(_("Mailcap file %s, line %d: incomplete entry "
1241 strFileName
.c_str(), nLine
+ 1);
1244 MailCapEntry
*entry
= new MailCapEntry(strOpenCmd
,
1248 strType
.MakeLower();
1249 int nIndex
= m_aTypes
.Index(strType
);
1250 if ( nIndex
== wxNOT_FOUND
) {
1252 m_aTypes
.Add(strType
);
1254 m_aEntries
.Add(entry
);
1255 m_aExtensions
.Add("");
1256 m_aDescriptions
.Add(strDesc
);
1259 // modify the existing entry: the entry in one and the same file
1260 // are read in top-to-bottom order, i.e. the entries read first
1261 // should be tried before the entries below. However, the files
1262 // read later should override the settings in the files read
1263 // before, thus we Append() the new entry to the list if it has
1264 // already occured in _this_ file, but Prepend() it if it
1265 // occured in some of the previous ones.
1266 if ( aEntryIndices
.Index(nIndex
) == wxNOT_FOUND
) {
1267 // first time in this file
1268 aEntryIndices
.Add(nIndex
);
1269 entry
->Prepend(m_aEntries
[nIndex
]);
1270 m_aEntries
[nIndex
] = entry
;
1273 // not the first time in _this_ file
1274 entry
->Append(m_aEntries
[nIndex
]);
1277 if ( !strDesc
.IsEmpty() ) {
1278 // @@ replace the old one - what else can we do??
1279 m_aDescriptions
[nIndex
] = strDesc
;
1284 // check our data integriry
1285 wxASSERT( m_aTypes
.Count() == m_aEntries
.Count() &&
1286 m_aTypes
.Count() == m_aExtensions
.Count() &&
1287 m_aTypes
.Count() == m_aDescriptions
.Count() );