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
*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 by
341 // 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 wxMimeTypesManager::wxMimeTypesManager()
453 m_impl
= new wxMimeTypesManagerImpl
;
456 wxMimeTypesManager::~wxMimeTypesManager()
462 wxMimeTypesManager::GetFileTypeFromExtension(const wxString
& ext
)
464 return m_impl
->GetFileTypeFromExtension(ext
);
468 wxMimeTypesManager::GetFileTypeFromMimeType(const wxString
& mimeType
)
470 return m_impl
->GetFileTypeFromMimeType(mimeType
);
473 // ============================================================================
474 // real (OS specific) implementation
475 // ============================================================================
479 bool wxFileTypeImpl::GetCommand(wxString
*command
, const char *verb
) const
481 // suppress possible error messages
484 strKey
<< m_strFileType
<< "\\shell\\" << verb
<< "\\command";
485 wxRegKey
key(wxRegKey::HKCR
, strKey
);
488 // it's the default value of the key
489 if ( key
.QueryValue("", *command
) ) {
490 // transform it from '%1' to '%s' style format string
491 // @@ we don't make any attempt to verify that the string is valid,
492 // i.e. doesn't contain %2, or second %1 or .... But we do make
493 // sure that we return a string with _exactly_ one '%s'!
494 size_t len
= command
->Len();
495 for ( size_t n
= 0; n
< len
; n
++ ) {
496 if ( command
->GetChar(n
) == '%' &&
497 (n
+ 1 < len
) && command
->GetChar(n
+ 1) == '1' ) {
498 // replace it with '%s'
499 command
->SetChar(n
+ 1, 's');
505 // we didn't find any '%1'!
506 // @@@ hack: append the filename at the end, hope that it will do
513 // no such file type or no value
517 // @@ this function is half implemented
518 bool wxFileTypeImpl::GetExtensions(wxArrayString
& extensions
)
520 if ( m_ext
.IsEmpty() ) {
521 // the only way to get the list of extensions from the file type is to
522 // scan through all extensions in the registry - too slow...
527 extensions
.Add(m_ext
);
529 // it's a lie too, we don't return _all_ extensions...
534 bool wxFileTypeImpl::GetMimeType(wxString
*mimeType
) const
536 // suppress possible error messages
538 wxRegKey
key(wxRegKey::HKCR
, m_strFileType
);
539 if ( key
.Open() && key
.QueryValue("Content Type", *mimeType
) ) {
547 bool wxFileTypeImpl::GetIcon(wxIcon
*icon
) const
550 strIconKey
<< m_strFileType
<< "\\DefaultIcon";
552 // suppress possible error messages
554 wxRegKey
key(wxRegKey::HKCR
, strIconKey
);
558 // it's the default value of the key
559 if ( key
.QueryValue("", strIcon
) ) {
560 // the format is the following: <full path to file>, <icon index>
561 // NB: icon index may be negative as well as positive and the full
562 // path may contain the environment variables inside '%'
563 wxString strFullPath
= strIcon
.BeforeLast(','),
564 strIndex
= strIcon
.AfterLast(',');
566 // index may be omitted, in which case BeforeLast(',') is empty and
567 // AfterLast(',') is the whole string
568 if ( strFullPath
.IsEmpty() ) {
569 strFullPath
= strIndex
;
573 wxString strExpPath
= wxExpandEnvVars(strFullPath
);
574 int nIndex
= atoi(strIndex
);
576 HICON hIcon
= ExtractIcon(GetModuleHandle(NULL
), strExpPath
, nIndex
);
577 switch ( (int)hIcon
) {
578 case 0: // means no icons were found
579 case 1: // means no such file or it wasn't a DLL/EXE/OCX/ICO/...
580 wxLogDebug("incorrect registry entry '%s': no such icon.",
581 key
.GetName().c_str());
585 icon
->SetHICON((WXHICON
)hIcon
);
591 // no such file type or no value or incorrect icon entry
595 bool wxFileTypeImpl::GetDescription(wxString
*desc
) const
597 // suppress possible error messages
599 wxRegKey
key(wxRegKey::HKCR
, m_strFileType
);
602 // it's the default value of the key
603 if ( key
.QueryValue("", *desc
) ) {
611 // extension -> file type
613 wxMimeTypesManagerImpl::GetFileTypeFromExtension(const wxString
& ext
)
615 // add the leading point if necessary
617 if ( ext
[0u] != '.' ) {
622 // suppress possible error messages
625 wxString strFileType
;
626 wxRegKey
key(wxRegKey::HKCR
, str
);
628 // it's the default value of the key
629 if ( key
.QueryValue("", strFileType
) ) {
630 // create the new wxFileType object
631 wxFileType
*fileType
= new wxFileType
;
632 fileType
->m_impl
->SetFileType(strFileType
);
633 fileType
->m_impl
->SetExt(ext
);
643 // MIME type -> extension -> file type
645 wxMimeTypesManagerImpl::GetFileTypeFromMimeType(const wxString
& mimeType
)
647 // @@@ I don't know of any official documentation which mentions this
648 // location, but as a matter of fact IE uses it, so why not we?
649 static const char *szMimeDbase
= "MIME\\Database\\Content Type\\";
651 wxString strKey
= szMimeDbase
;
654 // suppress possible error messages
658 wxRegKey
key(wxRegKey::HKCR
, strKey
);
660 if ( key
.QueryValue("Extension", ext
) ) {
661 return GetFileTypeFromExtension(ext
);
672 wxFileTypeImpl::GetEntry(const wxFileType::MessageParameters
& params
) const
675 MailCapEntry
*entry
= m_manager
->m_aEntries
[m_index
];
676 while ( entry
!= NULL
) {
677 // notice that an empty command would always succeed (@@ is it ok?)
678 command
= wxFileType::ExpandCommand(entry
->GetTestCmd(), params
);
680 if ( command
.IsEmpty() || (system(command
) == 0) ) {
682 wxLogTrace("Test '%s' for mime type '%s' succeeded.",
683 command
.c_str(), params
.GetMimeType().c_str());
687 wxLogTrace("Test '%s' for mime type '%s' failed.",
688 command
.c_str(), params
.GetMimeType().c_str());
691 entry
= entry
->GetNext();
698 wxFileTypeImpl::GetExpandedCommand(wxString
*expandedCmd
,
699 const wxFileType::MessageParameters
& params
,
702 MailCapEntry
*entry
= GetEntry(params
);
703 if ( entry
== NULL
) {
704 // all tests failed...
708 wxString cmd
= open
? entry
->GetOpenCmd() : entry
->GetPrintCmd();
709 if ( cmd
.IsEmpty() ) {
710 // may happen, especially for "print"
714 *expandedCmd
= wxFileType::ExpandCommand(cmd
, params
);
718 bool wxFileTypeImpl::GetExtensions(wxArrayString
& extensions
)
720 wxString strExtensions
= m_manager
->GetExtension(m_index
);
723 // one extension in the space or comma delimitid list
725 for ( const char *p
= strExtensions
; ; p
++ ) {
726 if ( *p
== ' ' || *p
== ',' || *p
== '\0' ) {
727 if ( !strExt
.IsEmpty() ) {
728 extensions
.Add(strExt
);
731 //else: repeated spaces (shouldn't happen, but it's not that
732 // important if it does happen)
737 else if ( *p
== '.' ) {
738 // remove the dot from extension (but only if it's the first char)
739 if ( !strExt
.IsEmpty() ) {
742 //else: no, don't append it
752 // read system and user mailcaps (TODO implement mime.types support)
753 wxMimeTypesManagerImpl::wxMimeTypesManagerImpl()
755 // directories where we look for mailcap and mime.types by default
756 // (taken from metamail(1) sources)
757 static const char *aStandardLocations
[] =
766 // first read the system wide file(s)
767 for ( size_t n
= 0; n
< WXSIZEOF(aStandardLocations
); n
++ ) {
768 wxString dir
= aStandardLocations
[n
];
770 wxString file
= dir
+ "/mailcap";
771 if ( wxFile::Exists(file
) ) {
775 file
= dir
+ "/mime.types";
776 if ( wxFile::Exists(file
) ) {
781 wxString strHome
= getenv("HOME");
783 // and now the users mailcap
784 wxString strUserMailcap
= strHome
+ "/.mailcap";
785 if ( wxFile::Exists(strUserMailcap
) ) {
786 ReadMailcap(strUserMailcap
);
789 // read the users mime.types
790 wxString strUserMimeTypes
= strHome
+ "/.mime.types";
791 if ( wxFile::Exists(strUserMimeTypes
) ) {
792 ReadMimeTypes(strUserMimeTypes
);
797 wxMimeTypesManagerImpl::GetFileTypeFromExtension(const wxString
& ext
)
799 size_t count
= m_aExtensions
.GetCount();
800 for ( size_t n
= 0; n
< count
; n
++ ) {
801 wxString extensions
= m_aExtensions
[n
];
802 while ( !extensions
.IsEmpty() ) {
803 wxString field
= extensions
.BeforeFirst(' ');
804 extensions
= extensions
.AfterFirst(' ');
806 // consider extensions as not being case-sensitive
807 if ( field
.IsSameAs(ext
, FALSE
/* no case */) ) {
809 wxFileType
*fileType
= new wxFileType
;
810 fileType
->m_impl
->Init(this, n
);
822 wxMimeTypesManagerImpl::GetFileTypeFromMimeType(const wxString
& mimeType
)
824 // mime types are not case-sensitive
825 wxString
mimetype(mimeType
);
826 mimetype
.MakeLower();
828 // first look for an exact match
829 int index
= m_aTypes
.Index(mimetype
);
830 if ( index
== wxNOT_FOUND
) {
831 // then try to find "text/*" as match for "text/plain" (for example)
832 // NB: if mimeType doesn't contain '/' at all, BeforeFirst() will return
833 // the whole string - ok.
834 wxString strCategory
= mimetype
.BeforeFirst('/');
836 size_t nCount
= m_aTypes
.Count();
837 for ( size_t n
= 0; n
< nCount
; n
++ ) {
838 if ( (m_aTypes
[n
].BeforeFirst('/') == strCategory
) &&
839 m_aTypes
[n
].AfterFirst('/') == "*" ) {
846 if ( index
!= wxNOT_FOUND
) {
847 wxFileType
*fileType
= new wxFileType
;
848 fileType
->m_impl
->Init(this, index
);
858 void wxMimeTypesManagerImpl::ReadMimeTypes(const wxString
& strFileName
)
860 wxLogTrace("--- Parsing mime.types file '%s' ---", strFileName
.c_str());
862 wxTextFile
file(strFileName
);
866 // the information we extract
867 wxString strMimeType
, strDesc
, strExtensions
;
869 size_t nLineCount
= file
.GetLineCount();
870 for ( size_t nLine
= 0; nLine
< nLineCount
; nLine
++ ) {
871 // now we're at the start of the line
872 const char *pc
= file
[nLine
].c_str();
875 while ( isspace(*pc
) )
882 // detect file format
883 const char *pEqualSign
= strchr(pc
, '=');
884 if ( pEqualSign
== NULL
) {
888 // first field is mime type
889 for ( strMimeType
.Empty(); !isspace(*pc
) && *pc
!= '\0'; pc
++ ) {
894 while ( isspace(*pc
) )
897 // take all the rest of the string
907 // the string on the left of '=' is the field name
908 wxString
strLHS(pc
, pEqualSign
- pc
);
911 for ( pc
= pEqualSign
+ 1; isspace(*pc
); pc
++ )
916 // the string is quoted and ends at the matching quote
917 pEnd
= strchr(++pc
, '"');
918 if ( pEnd
== NULL
) {
919 wxLogWarning(_("Mime.types file %s, line %d: unterminated "
921 strFileName
.c_str(), nLine
+ 1);
925 // unquoted stringends at the first space
926 for ( pEnd
= pc
; !isspace(*pEnd
); pEnd
++ )
930 // now we have the RHS (field value)
931 wxString
strRHS(pc
, pEnd
- pc
);
933 // check that it's more or less what we're waiting for, i.e. that
934 // only '\' is left on the line
935 if ( *pEnd
== '"' ) {
940 for ( pc
= pEnd
; isspace(*pc
); pc
++ )
943 // only '\\' may be left on the line normally
944 bool entryEnded
= *pc
== '\0';
945 if ( !entryEnded
&& ((*pc
!= '\\') || (*++pc
!= '\0')) ) {
946 wxLogWarning(_("Mime.types file %s, line %d: extra characters "
947 "after the field value ignored."),
948 strFileName
.c_str(), nLine
+ 1);
950 // if there is a trailing backslash entryEnded = FALSE
952 // now see what we got
953 if ( strLHS
== "type" ) {
954 strMimeType
= strRHS
;
956 else if ( strLHS
== "desc" ) {
959 else if ( strLHS
== "exts" ) {
960 strExtensions
= strRHS
;
963 wxLogWarning(_("Unknown field in file %s, line %d: '%s'."),
964 strFileName
.c_str(), nLine
+ 1, strLHS
.c_str());
968 // as we don't reset strMimeType, the next line in this entry
969 // will be interpreted correctly.
974 int index
= m_aTypes
.Index(strMimeType
);
975 if ( index
== wxNOT_FOUND
) {
977 m_aTypes
.Add(strMimeType
);
978 m_aEntries
.Add(NULL
);
979 m_aExtensions
.Add(strExtensions
);
980 m_aDescriptions
.Add(strDesc
);
983 // modify an existing one
984 if ( !strDesc
.IsEmpty() ) {
985 m_aDescriptions
[index
] = strDesc
; // replace old value
987 m_aExtensions
[index
] += strExtensions
;
991 // check our data integriry
992 wxASSERT( m_aTypes
.Count() == m_aEntries
.Count() &&
993 m_aTypes
.Count() == m_aExtensions
.Count() &&
994 m_aTypes
.Count() == m_aDescriptions
.Count() );
997 void wxMimeTypesManagerImpl::ReadMailcap(const wxString
& strFileName
)
999 wxLogTrace("--- Parsing mailcap file '%s' ---", strFileName
.c_str());
1001 wxTextFile
file(strFileName
);
1005 // see the comments near the end of function for the reason we need this
1006 wxArrayInt aEntryIndices
;
1008 size_t nLineCount
= file
.GetLineCount();
1009 for ( size_t nLine
= 0; nLine
< nLineCount
; nLine
++ ) {
1010 // now we're at the start of the line
1011 const char *pc
= file
[nLine
].c_str();
1014 while ( isspace(*pc
) )
1017 // comment or empty string?
1018 if ( *pc
== '#' || *pc
== '\0' )
1023 // what field are we currently in? The first 2 are fixed and there may
1024 // be an arbitrary number of other fields -- currently, we are not
1025 // interested in any of them, but we should parse them as well...
1031 } currentToken
= Field_Type
;
1033 // the flags and field values on the current line
1034 bool needsterminal
= false,
1035 copiousoutput
= false;
1041 curField
; // accumulator
1042 for ( bool cont
= TRUE
; cont
; pc
++ ) {
1045 // interpret the next character literally (notice that
1046 // backslash can be used for line continuation)
1047 if ( *++pc
== '\0' ) {
1048 // fetch the next line.
1050 // pc currently points to nowhere, but after the next
1051 // pc++ in the for line it will point to the beginning
1052 // of the next line in the file
1053 pc
= file
[++nLine
].c_str() - 1;
1056 // just a normal character
1062 cont
= FALSE
; // end of line reached, exit the loop
1067 // store this field and start looking for the next one
1069 // trim whitespaces from both sides
1070 curField
.Trim(TRUE
).Trim(FALSE
);
1072 switch ( currentToken
) {
1075 if ( strType
.Find('/') == wxNOT_FOUND
) {
1076 // we interpret "type" as "type/*"
1080 currentToken
= Field_OpenCmd
;
1084 strOpenCmd
= curField
;
1086 currentToken
= Field_Other
;
1091 // "good" mailcap entry?
1094 // is this something of the form foo=bar?
1095 const char *pEq
= strchr(curField
, '=');
1096 if ( pEq
!= NULL
) {
1097 wxString lhs
= curField
.Before('='),
1098 rhs
= curField
.After('=');
1100 lhs
.Trim(TRUE
); // from right
1101 rhs
.Trim(FALSE
); // from left
1103 if ( lhs
== "print" )
1105 else if ( lhs
== "test" )
1107 else if ( lhs
== "description" ) {
1108 // it might be quoted
1109 if ( rhs
[0u] == '"' &&
1110 rhs
.Last() == '"' ) {
1111 strDesc
= wxString(rhs
.c_str() + 1,
1118 else if ( lhs
== "compose" ||
1119 lhs
== "composetyped" ||
1128 // no, it's a simple flag
1129 // TODO support the flags:
1130 // 1. create an xterm for 'needsterminal'
1131 // 2. append "| $PAGER" for 'copiousoutput'
1132 if ( curField
== "needsterminal" )
1133 needsterminal
= TRUE
;
1134 else if ( curField
== "copiousoutput" )
1135 copiousoutput
= TRUE
;
1136 else if ( curField
== "textualnewlines" )
1144 // don't flood the user with error messages
1145 // if we don't understand something in his
1149 _("Mailcap file %s, line %d: unknown "
1150 "field '%s' for the MIME type "
1152 strFileName
.c_str(),
1160 // it already has this value
1161 //currentToken = Field_Other;
1165 wxFAIL_MSG("unknown field type in mailcap");
1168 // next token starts immediately after ';'
1177 // check that we really read something reasonable
1178 if ( currentToken
== Field_Type
|| currentToken
== Field_OpenCmd
) {
1179 wxLogWarning(_("Mailcap file %s, line %d: incomplete entry "
1181 strFileName
.c_str(), nLine
+ 1);
1184 MailCapEntry
*entry
= new MailCapEntry(strOpenCmd
,
1188 strType
.MakeLower();
1189 int nIndex
= m_aTypes
.Index(strType
);
1190 if ( nIndex
== wxNOT_FOUND
) {
1192 m_aTypes
.Add(strType
);
1194 m_aEntries
.Add(entry
);
1195 m_aExtensions
.Add("");
1196 m_aDescriptions
.Add(strDesc
);
1199 // modify the existing entry: the entry in one and the same file
1200 // are read in top-to-bottom order, i.e. the entries read first
1201 // should be tried before the entries below. However, the files
1202 // read later should override the settings in the files read
1203 // before, thus we Append() the new entry to the list if it has
1204 // already occured in _this_ file, but Prepend() it if it
1205 // occured in some of the previous ones.
1206 if ( aEntryIndices
.Index(nIndex
) == wxNOT_FOUND
) {
1207 // first time in this file
1208 aEntryIndices
.Add(nIndex
);
1209 entry
->Prepend(m_aEntries
[nIndex
]);
1210 m_aEntries
[nIndex
] = entry
;
1213 // not the first time in _this_ file
1214 entry
->Append(m_aEntries
[nIndex
]);
1217 if ( !strDesc
.IsEmpty() ) {
1218 // @@ replace the old one - what else can we do??
1219 m_aDescriptions
[nIndex
] = strDesc
;
1224 // check our data integriry
1225 wxASSERT( m_aTypes
.Count() == m_aEntries
.Count() &&
1226 m_aTypes
.Count() == m_aExtensions
.Count() &&
1227 m_aTypes
.Count() == m_aDescriptions
.Count() );