]> git.saurik.com Git - wxWidgets.git/blame_incremental - src/unix/mimetype.cpp
fix crash in DoGetBestSize() if default widths are used (corrects patch 1386199 appli...
[wxWidgets.git] / src / unix / mimetype.cpp
... / ...
CommitLineData
1/////////////////////////////////////////////////////////////////////////////
2// Name: src/unix/mimetype.cpp
3// Purpose: classes and functions to manage MIME types
4// Author: Vadim Zeitlin
5// Modified by:
6// Created: 23.09.98
7// RCS-ID: $Id$
8// Copyright: (c) 1998 Vadim Zeitlin <zeitlin@dptmaths.ens-cachan.fr>
9// Licence: wxWindows licence (part of wxExtra library)
10/////////////////////////////////////////////////////////////////////////////
11
12// known bugs; there may be others!! chris elliott, biol75@york.ac.uk 27 Mar 01
13
14// 1) .mailcap and .mimetypes can be either in a netscape or metamail format
15// and entries may get confused during writing (I've tried to fix this; please let me know
16// any files that fail)
17// 2) KDE and Gnome do not yet fully support international read/write
18// 3) Gnome key lines like open.latex."LaTeX this file"=latex %f will have odd results
19// 4) writing to files comments out the existing data; I hope this avoids losing
20// any data which we could not read, and data which we did not store like test=
21// 5) results from reading files with multiple entries (especially matches with type/* )
22// may (or may not) work for getXXX commands
23// 6) Loading the png icons in Gnome doesn't work for me...
24// 7) In Gnome, if keys.mime exists but keys.users does not, there is
25// an error message in debug mode, but the file is still written OK
26// 8) Deleting entries is only allowed from the user file; sytem wide entries
27// will be preserved during unassociate
28// 9) KDE does not yet handle multiple actions; Netscape mode never will
29
30// TODO: this file is a mess, we need to split it and review everything (VZ)
31
32// for compilers that support precompilation, includes "wx.h".
33#include "wx/wxprec.h"
34
35#ifdef __BORLANDC__
36 #pragma hdrstop
37#endif
38
39#if wxUSE_MIMETYPE && wxUSE_FILE && wxUSE_TEXTFILE
40
41#include "wx/unix/mimetype.h"
42
43#ifndef WX_PRECOMP
44 #include "wx/dynarray.h"
45 #include "wx/string.h"
46 #include "wx/intl.h"
47 #include "wx/log.h"
48 #include "wx/utils.h"
49#endif
50
51#include "wx/file.h"
52#include "wx/confbase.h"
53
54#include "wx/ffile.h"
55#include "wx/textfile.h"
56#include "wx/dir.h"
57#include "wx/tokenzr.h"
58#include "wx/iconloc.h"
59#include "wx/filename.h"
60#include "wx/app.h"
61#include "wx/apptrait.h"
62
63#if wxUSE_LIBGNOMEVFS
64 // Not GUI dependent
65 #include "wx/gtk/gnome/gvfs.h"
66#endif
67
68// other standard headers
69#include <ctype.h>
70
71// this class is a wxTextFile specialization for dealing with files storing
72// various MIME-related information
73//
74// it should be used instead of wxTextFile even if none of its additional
75// methods are used just because it handles files with mixed encodings (often
76// the case for MIME files which contain strings for different languages)
77// correctly, see OnRead()
78class wxMimeTextFile : public wxTextFile
79{
80public:
81 // constructors
82 wxMimeTextFile () : wxTextFile () { }
83 wxMimeTextFile(const wxString& strFile) : wxTextFile(strFile) { }
84
85 int pIndexOf(const wxString& sSearch,
86 bool bIncludeComments = false,
87 int iStart = 0)
88 {
89 wxString sTest = sSearch;
90 sTest.MakeLower();
91 for(size_t i = iStart; i < GetLineCount(); i++)
92 {
93 wxString sLine = GetLine(i).Trim(false);
94 if(bIncludeComments || ! sLine.StartsWith(wxT("#")))
95 {
96 sLine.MakeLower();
97 if(sLine.StartsWith(sTest))
98 return (int)i;
99 }
100 }
101 return wxNOT_FOUND;
102 }
103
104 bool CommentLine(int nIndex)
105 {
106 if (nIndex < 0)
107 return false;
108 if (nIndex >= (int)GetLineCount() )
109 return false;
110
111 GetLine(nIndex) = GetLine(nIndex).Prepend(wxT("#"));
112 return true;
113 }
114
115 bool CommentLine(const wxString & sTest)
116 {
117 int nIndex = pIndexOf(sTest);
118 if (nIndex < 0)
119 return false;
120 if (nIndex >= (int)GetLineCount() )
121 return false;
122
123 GetLine(nIndex) = GetLine(nIndex).Prepend(wxT("#"));
124 return true;
125 }
126
127 wxString GetVerb(size_t i)
128 {
129 if (i > GetLineCount() )
130 return wxEmptyString;
131
132 wxString sTmp = GetLine(i).BeforeFirst(wxT('='));
133 return sTmp;
134 }
135
136 wxString GetCmd(size_t i)
137 {
138 if (i > GetLineCount() )
139 return wxEmptyString;
140
141 wxString sTmp = GetLine(i).AfterFirst(wxT('='));
142 return sTmp;
143 }
144};
145
146// ----------------------------------------------------------------------------
147// constants
148// ----------------------------------------------------------------------------
149
150// MIME code tracing mask
151#define TRACE_MIME wxT("mime")
152
153// give trace messages about the results of mailcap tests
154#define TRACE_MIME_TEST wxT("mimetest")
155
156// ----------------------------------------------------------------------------
157// private functions
158// ----------------------------------------------------------------------------
159
160// there are some fields which we don't understand but for which we don't give
161// warnings as we know that they're not important - this function is used to
162// test for them
163static bool IsKnownUnimportantField(const wxString& field);
164
165// ----------------------------------------------------------------------------
166// private classes
167// ----------------------------------------------------------------------------
168
169
170// This class uses both mailcap and mime.types to gather information about file
171// types.
172//
173// The information about mailcap file was extracted from metamail(1) sources
174// and documentation and subsequently revised when I found the RFC 1524
175// describing it.
176//
177// Format of mailcap file: spaces are ignored, each line is either a comment
178// (starts with '#') or a line of the form <field1>;<field2>;...;<fieldN>.
179// A backslash can be used to quote semicolons and newlines (and, in fact,
180// anything else including itself).
181//
182// The first field is always the MIME type in the form of type/subtype (see RFC
183// 822) where subtype may be '*' meaning "any". Following metamail, we accept
184// "type" which means the same as "type/*", although I'm not sure whether this
185// is standard.
186//
187// The second field is always the command to run. It is subject to
188// parameter/filename expansion described below.
189//
190// All the following fields are optional and may not be present at all. If
191// they're present they may appear in any order, although each of them should
192// appear only once. The optional fields are the following:
193// * notes=xxx is an uninterpreted string which is silently ignored
194// * test=xxx is the command to be used to determine whether this mailcap line
195// applies to our data or not. The RHS of this field goes through the
196// parameter/filename expansion (as the 2nd field) and the resulting string
197// is executed. The line applies only if the command succeeds, i.e. returns 0
198// exit code.
199// * print=xxx is the command to be used to print (and not view) the data of
200// this type (parameter/filename expansion is done here too)
201// * edit=xxx is the command to open/edit the data of this type
202// * needsterminal means that a new interactive console must be created for
203// the viewer
204// * copiousoutput means that the viewer doesn't interact with the user but
205// produces (possibly) a lof of lines of output on stdout (i.e. "cat" is a
206// good example), thus it might be a good idea to use some kind of paging
207// mechanism.
208// * textualnewlines means not to perform CR/LF translation (not honored)
209// * compose and composetyped fields are used to determine the program to be
210// called to create a new message pert in the specified format (unused).
211//
212// Parameter/filename expansion:
213// * %s is replaced with the (full) file name
214// * %t is replaced with MIME type/subtype of the entry
215// * for multipart type only %n is replaced with the nnumber of parts and %F is
216// replaced by an array of (content-type, temporary file name) pairs for all
217// message parts (TODO)
218// * %{parameter} is replaced with the value of parameter taken from
219// Content-type header line of the message.
220//
221//
222// There are 2 possible formats for mime.types file, one entry per line (used
223// for global mime.types and called Mosaic format) and "expanded" format where
224// an entry takes multiple lines (used for users mime.types and called
225// Netscape format).
226//
227// For both formats spaces are ignored and lines starting with a '#' are
228// comments. Each record has one of two following forms:
229// a) for "brief" format:
230// <mime type> <space separated list of extensions>
231// b) for "expanded" format:
232// type=<mime type> BACKSLASH
233// desc="<description>" BACKSLASH
234// exts="<comma separated list of extensions>"
235//
236// (where BACKSLASH is a literal '\\' which we can't put here because cpp
237// misinterprets it)
238//
239// We try to autodetect the format of mime.types: if a non-comment line starts
240// with "type=" we assume the second format, otherwise the first one.
241
242// there may be more than one entry for one and the same mime type, to
243// choose the right one we have to run the command specified in the test
244// field on our data.
245
246// ----------------------------------------------------------------------------
247// wxGNOME
248// ----------------------------------------------------------------------------
249
250// GNOME stores the info we're interested in in several locations:
251// 1. xxx.keys files under /usr/share/mime-info
252// 2. xxx.keys files under ~/.gnome/mime-info
253//
254// Update (Chris Elliott): apparently there may be an optional "[lang]" prefix
255// just before the field name.
256
257
258void wxMimeTypesManagerImpl::LoadGnomeDataFromKeyFile(const wxString& filename,
259 const wxArrayString& dirs)
260{
261 wxMimeTextFile textfile(filename);
262 if ( !textfile.Open() )
263 return;
264
265 wxLogTrace(TRACE_MIME, wxT("--- Opened Gnome file %s ---"),
266 filename.c_str());
267
268 wxArrayString search_dirs( dirs );
269
270 // values for the entry being parsed
271 wxString curMimeType, curIconFile;
272 wxMimeTypeCommands * entry = new wxMimeTypeCommands;
273
274 wxArrayString strExtensions;
275 wxString strDesc;
276
277 const wxChar *pc;
278 size_t nLineCount = textfile.GetLineCount();
279 size_t nLine = 0;
280 while ( nLine < nLineCount )
281 {
282 pc = textfile[nLine].c_str();
283 if ( *pc != wxT('#') )
284 {
285
286 wxLogTrace(TRACE_MIME, wxT("--- Reading from Gnome file %s '%s' ---"),
287 filename.c_str(), pc);
288
289 // trim trailing space and tab
290 while ((*pc == wxT(' ')) || (*pc == wxT('\t')))
291 pc++;
292
293 wxString sTmp(pc);
294 int equal_pos = sTmp.Find( wxT('=') );
295 if (equal_pos > 0)
296 {
297 wxString left_of_equal = sTmp.Left( equal_pos );
298 const wxChar *right_of_equal = pc;
299 right_of_equal += equal_pos+1;
300
301 if (left_of_equal == wxT("icon_filename"))
302 {
303 // GNOME 2:
304 curIconFile = right_of_equal;
305
306 wxFileName newFile( curIconFile );
307 if (newFile.IsRelative() || newFile.FileExists())
308 {
309 size_t nDirs = search_dirs.GetCount();
310
311 for (size_t nDir = 0; nDir < nDirs; nDir++)
312 {
313 newFile.SetPath( search_dirs[nDir] );
314 newFile.AppendDir( wxT("pixmaps") );
315 newFile.AppendDir( wxT("document-icons") );
316 newFile.SetExt( wxT("png") );
317 if (newFile.FileExists())
318 {
319 curIconFile = newFile.GetFullPath();
320 // reorder search_dirs for speedup (fewer
321 // calls to FileExist() required)
322 if (nDir != 0)
323 {
324 const wxString &tmp = search_dirs[nDir];
325 search_dirs.RemoveAt( nDir );
326 search_dirs.Insert( tmp, 0 );
327 }
328 break;
329 }
330 }
331 }
332 }
333 else if (left_of_equal == wxT("open"))
334 {
335 sTmp = right_of_equal;
336 sTmp.Replace( wxT("%f"), wxT("%s") );
337 sTmp.Prepend( wxT("open=") );
338 entry->Add(sTmp);
339 }
340 else if (left_of_equal == wxT("view"))
341 {
342 sTmp = right_of_equal;
343 sTmp.Replace( wxT("%f"), wxT("%s") );
344 sTmp.Prepend( wxT("view=") );
345 entry->Add(sTmp);
346 }
347 else if (left_of_equal == wxT("print"))
348 {
349 sTmp = right_of_equal;
350 sTmp.Replace( wxT("%f"), wxT("%s") );
351 sTmp.Prepend( wxT("print=") );
352 entry->Add(sTmp);
353 }
354 else if (left_of_equal == wxT("description"))
355 {
356 strDesc = right_of_equal;
357 }
358 else if (left_of_equal == wxT("short_list_application_ids_for_novice_user_level"))
359 {
360 sTmp = right_of_equal;
361 if (sTmp.Contains( wxT(",") ))
362 sTmp = sTmp.BeforeFirst( wxT(',') );
363 sTmp.Prepend( wxT("open=") );
364 sTmp.Append( wxT(" %s") );
365 entry->Add(sTmp);
366 }
367
368 } // emd of has an equals sign
369 else
370 {
371 // not a comment and not an equals sign
372 if (sTmp.Contains(wxT('/')))
373 {
374 // this is the start of the new mimetype
375 // overwrite any existing data
376 if (! curMimeType.empty())
377 {
378 AddToMimeData( curMimeType, curIconFile, entry, strExtensions, strDesc );
379
380 // now get ready for next bit
381 entry = new wxMimeTypeCommands;
382 }
383
384 curMimeType = sTmp.BeforeFirst(wxT(':'));
385 }
386 }
387 } // end of not a comment
388
389 // ignore blank lines
390 nLine++;
391 } // end of while, save any data
392
393 if ( curMimeType.empty() )
394 delete entry;
395 else
396 AddToMimeData( curMimeType, curIconFile, entry, strExtensions, strDesc);
397}
398
399void wxMimeTypesManagerImpl::LoadGnomeMimeTypesFromMimeFile(const wxString& filename)
400{
401 wxMimeTextFile textfile(filename);
402 if ( !textfile.Open() )
403 return;
404
405 wxLogTrace(TRACE_MIME,
406 wxT("--- Opened Gnome file %s ---"),
407 filename.c_str());
408
409 // values for the entry being parsed
410 wxString curMimeType, curExtList;
411
412 const wxChar *pc;
413 size_t nLineCount = textfile.GetLineCount();
414 for ( size_t nLine = 0; /* nothing */; nLine++ )
415 {
416 if ( nLine < nLineCount )
417 {
418 pc = textfile[nLine].c_str();
419 if ( *pc == wxT('#') )
420 {
421 // skip comments
422 continue;
423 }
424 }
425 else
426 {
427 // so that we will fall into the "if" below
428 pc = NULL;
429 }
430
431 if ( !pc || !*pc )
432 {
433 // end of the entry
434 if ( !curMimeType.empty() && !curExtList.empty() )
435 {
436 wxLogTrace(TRACE_MIME,
437 wxT("--- At end of Gnome file finding mimetype %s ---"),
438 curMimeType.c_str());
439
440 AddMimeTypeInfo(curMimeType, curExtList, wxEmptyString);
441 }
442
443 if ( !pc )
444 {
445 // the end: this can only happen if nLine == nLineCount
446 break;
447 }
448
449 curExtList.Empty();
450
451 continue;
452 }
453
454 // what do we have here?
455 if ( *pc == wxT('\t') )
456 {
457 // this is a field=value ling
458 pc++; // skip leading TAB
459
460 static const int lenField = 5; // strlen("ext: ")
461 if ( wxStrncmp(pc, wxT("ext: "), lenField) == 0 )
462 {
463 // skip it and take everything left until the end of line
464 curExtList = pc + lenField;
465 }
466 //else: some other field, we don't care
467 }
468 else
469 {
470 // this is the start of the new section
471 wxLogTrace(TRACE_MIME,
472 wxT("--- In Gnome file finding mimetype %s ---"),
473 curMimeType.c_str());
474
475 if (! curMimeType.empty())
476 AddMimeTypeInfo(curMimeType, curExtList, wxEmptyString);
477
478 curMimeType.Empty();
479
480 while ( *pc != wxT(':') && *pc != wxT('\0') )
481 {
482 curMimeType += *pc++;
483 }
484 }
485 }
486}
487
488
489void wxMimeTypesManagerImpl::LoadGnomeMimeFilesFromDir(
490 const wxString& dirbase, const wxArrayString& dirs)
491{
492 wxASSERT_MSG( !dirbase.empty() && !wxEndsWithPathSeparator(dirbase),
493 wxT("base directory shouldn't end with a slash") );
494
495 wxString dirname = dirbase;
496 dirname << wxT("/mime-info");
497
498 // Don't complain if we don't have permissions to read - it confuses users
499 wxLogNull logNull;
500
501 if ( !wxDir::Exists(dirname) )
502 return;
503
504 wxDir dir(dirname);
505 if ( !dir.IsOpened() )
506 return;
507
508 // we will concatenate it with filename to get the full path below
509 dirname += wxT('/');
510
511 wxString filename;
512 bool cont;
513
514 cont = dir.GetFirst(&filename, wxT("*.mime"), wxDIR_FILES);
515 while ( cont )
516 {
517 LoadGnomeMimeTypesFromMimeFile(dirname + filename);
518
519 cont = dir.GetNext(&filename);
520 }
521
522 cont = dir.GetFirst(&filename, wxT("*.keys"), wxDIR_FILES);
523 while ( cont )
524 {
525 LoadGnomeDataFromKeyFile(dirname + filename, dirs);
526
527 cont = dir.GetNext(&filename);
528 }
529
530 // FIXME: Hack alert: We scan all icons and deduce the
531 // mime-type from the file name.
532 dirname = dirbase;
533 dirname << wxT("/pixmaps/document-icons");
534
535 // these are always empty in this file
536 wxArrayString strExtensions;
537 wxString strDesc;
538
539 if ( !wxDir::Exists(dirname) )
540 {
541 // Just test for default GPE dir also
542 dirname = wxT("/usr/share/gpe/pixmaps/default/filemanager/document-icons");
543
544 if ( !wxDir::Exists(dirname) )
545 return;
546 }
547
548 wxDir dir2( dirname );
549
550 cont = dir2.GetFirst(&filename, wxT("gnome-*.png"), wxDIR_FILES);
551 while ( cont )
552 {
553 wxString mimeType = filename;
554 mimeType.Remove( 0, 6 ); // remove "gnome-"
555 mimeType.Remove( mimeType.Len() - 4, 4 ); // remove ".png"
556 int pos = mimeType.Find( wxT("-") );
557 if (pos != wxNOT_FOUND)
558 {
559 mimeType.SetChar( pos, wxT('/') );
560 wxString iconFile = dirname;
561 iconFile << wxT("/");
562 iconFile << filename;
563 AddToMimeData( mimeType, iconFile, NULL, strExtensions, strDesc, true );
564 }
565
566 cont = dir2.GetNext(&filename);
567 }
568}
569
570void wxMimeTypesManagerImpl::GetGnomeMimeInfo(const wxString& sExtraDir)
571{
572 wxArrayString dirs;
573
574 wxString gnomedir = wxGetenv( wxT("GNOMEDIR") );
575 if (!gnomedir.empty())
576 {
577 gnomedir << wxT("/share");
578 dirs.Add( gnomedir );
579 }
580
581 dirs.Add(wxT("/usr/share"));
582 dirs.Add(wxT("/usr/local/share"));
583
584 gnomedir = wxGetHomeDir();
585 gnomedir << wxT("/.gnome");
586 dirs.Add( gnomedir );
587
588 if (!sExtraDir.empty())
589 dirs.Add( sExtraDir );
590
591 size_t nDirs = dirs.GetCount();
592 for ( size_t nDir = 0; nDir < nDirs; nDir++ )
593 {
594 LoadGnomeMimeFilesFromDir(dirs[nDir], dirs);
595 }
596}
597
598// ----------------------------------------------------------------------------
599// KDE
600// ----------------------------------------------------------------------------
601
602
603// KDE stores the icon info in its .kdelnk files. The file for mimetype/subtype
604// may be found in either of the following locations
605//
606// 1. $KDEDIR/share/mimelnk/mimetype/subtype.kdelnk
607// 2. ~/.kde/share/mimelnk/mimetype/subtype.kdelnk
608//
609// The format of a .kdelnk file is almost the same as the one used by
610// wxFileConfig, i.e. there are groups, comments and entries. The icon is the
611// value for the entry "Type"
612
613// kde writing; see http://webcvs.kde.org/cgi-bin/cvsweb.cgi/~checkout~/kdelibs/kio/DESKTOP_ENTRY_STANDARD
614// for now write to .kdelnk but should eventually do .desktop instead (in preference??)
615
616bool wxMimeTypesManagerImpl::CheckKDEDirsExist( const wxString &sOK, const wxString &sTest )
617{
618 if (sTest.empty())
619 {
620 return wxDir::Exists(sOK);
621 }
622 else
623 {
624 wxString sStart = sOK + wxT("/") + sTest.BeforeFirst(wxT('/'));
625 if (!wxDir::Exists(sStart))
626 wxMkdir(sStart);
627 wxString sEnd = sTest.AfterFirst(wxT('/'));
628 return CheckKDEDirsExist(sStart, sEnd);
629 }
630}
631
632bool wxMimeTypesManagerImpl::WriteKDEMimeFile(int index, bool delete_index)
633{
634 wxMimeTextFile appoutfile, mimeoutfile;
635 wxString sHome = wxGetHomeDir();
636 wxString sTmp = wxT(".kde/share/mimelnk/");
637 wxString sMime = m_aTypes[index];
638 CheckKDEDirsExist(sHome, sTmp + sMime.BeforeFirst(wxT('/')) );
639 sTmp = sHome + wxT('/') + sTmp + sMime + wxT(".kdelnk");
640
641 bool bTemp;
642 bool bMimeExists = mimeoutfile.Open(sTmp);
643 if (!bMimeExists)
644 {
645 bTemp = mimeoutfile.Create(sTmp);
646 // some unknown error eg out of disk space
647 if (!bTemp)
648 return false;
649 }
650
651 sTmp = wxT(".kde/share/applnk/");
652 CheckKDEDirsExist(sHome, sTmp + sMime.AfterFirst(wxT('/')) );
653 sTmp = sHome + wxT('/') + sTmp + sMime.AfterFirst(wxT('/')) + wxT(".kdelnk");
654
655 bool bAppExists;
656 bAppExists = appoutfile.Open(sTmp);
657 if (!bAppExists)
658 {
659 bTemp = appoutfile.Create(sTmp);
660 // some unknown error eg out of disk space
661 if (!bTemp)
662 return false;
663 }
664
665 // fixed data; write if new file
666 if (!bMimeExists)
667 {
668 mimeoutfile.AddLine(wxT("#KDE Config File"));
669 mimeoutfile.AddLine(wxT("[KDE Desktop Entry]"));
670 mimeoutfile.AddLine(wxT("Version=1.0"));
671 mimeoutfile.AddLine(wxT("Type=MimeType"));
672 mimeoutfile.AddLine(wxT("MimeType=") + sMime);
673 }
674
675 if (!bAppExists)
676 {
677 mimeoutfile.AddLine(wxT("#KDE Config File"));
678 mimeoutfile.AddLine(wxT("[KDE Desktop Entry]"));
679 appoutfile.AddLine(wxT("Version=1.0"));
680 appoutfile.AddLine(wxT("Type=Application"));
681 appoutfile.AddLine(wxT("MimeType=") + sMime + wxT(';'));
682 }
683
684 // variable data
685 // ignore locale
686 mimeoutfile.CommentLine(wxT("Comment="));
687 if (!delete_index)
688 mimeoutfile.AddLine(wxT("Comment=") + m_aDescriptions[index]);
689 appoutfile.CommentLine(wxT("Name="));
690 if (!delete_index)
691 appoutfile.AddLine(wxT("Comment=") + m_aDescriptions[index]);
692
693 sTmp = m_aIcons[index];
694 // we can either give the full path, or the shortfilename if its in
695 // one of the directories we search
696 mimeoutfile.CommentLine(wxT("Icon=") );
697 if (!delete_index)
698 mimeoutfile.AddLine(wxT("Icon=") + sTmp );
699 appoutfile.CommentLine(wxT("Icon=") );
700 if (!delete_index)
701 appoutfile.AddLine(wxT("Icon=") + sTmp );
702
703 sTmp = wxT(" ") + m_aExtensions[index];
704
705 wxStringTokenizer tokenizer(sTmp, wxT(" "));
706 sTmp = wxT("Patterns=");
707 mimeoutfile.CommentLine(sTmp);
708 while ( tokenizer.HasMoreTokens() )
709 {
710 // holds an extension; need to change it to *.ext;
711 wxString e = wxT("*.") + tokenizer.GetNextToken() + wxT(";");
712 sTmp += e;
713 }
714
715 if (!delete_index)
716 mimeoutfile.AddLine(sTmp);
717
718 wxMimeTypeCommands * entries = m_aEntries[index];
719 // if we don't find open just have an empty string ... FIX this
720 sTmp = entries->GetCommandForVerb(wxT("open"));
721 sTmp.Replace( wxT("%s"), wxT("%f") );
722
723 mimeoutfile.CommentLine(wxT("DefaultApp=") );
724 if (!delete_index)
725 mimeoutfile.AddLine(wxT("DefaultApp=") + sTmp);
726
727 sTmp.Replace( wxT("%f"), wxT("") );
728 appoutfile.CommentLine(wxT("Exec="));
729 if (!delete_index)
730 appoutfile.AddLine(wxT("Exec=") + sTmp);
731
732 if (entries->GetCount() > 1)
733 {
734 //other actions as well as open
735 }
736
737 bTemp = false;
738 if (mimeoutfile.Write())
739 bTemp = true;
740 mimeoutfile.Close();
741 if (appoutfile.Write())
742 bTemp = true;
743 appoutfile.Close();
744
745 return bTemp;
746}
747
748void wxMimeTypesManagerImpl::LoadKDELinksForMimeSubtype(const wxString& dirbase,
749 const wxString& subdir,
750 const wxString& filename,
751 const wxArrayString& icondirs)
752{
753 wxFileName fullname(dirbase, filename);
754 wxLogTrace(TRACE_MIME, wxT("loading KDE file %s"),
755 fullname.GetFullPath().c_str());
756
757 wxMimeTextFile file;
758 if ( !file.Open(fullname.GetFullPath()) )
759 return;
760
761 wxMimeTypeCommands * entry = new wxMimeTypeCommands;
762 wxArrayString sExts;
763 wxString mimetype, mime_desc, strIcon;
764
765 int nIndex = file.pIndexOf( wxT("MimeType=") );
766 if (nIndex == wxNOT_FOUND)
767 {
768 // construct mimetype from the directory name and the basename of the
769 // file (it always has .kdelnk extension)
770 mimetype << subdir << wxT('/') << filename.BeforeLast( wxT('.') );
771 }
772 else
773 mimetype = file.GetCmd(nIndex);
774
775 // first find the description string: it is the value in either "Comment="
776 // line or "Comment[<locale_name>]=" one
777 nIndex = wxNOT_FOUND;
778
779 wxString comment;
780
781#if wxUSE_INTL
782 wxLocale *locale = wxGetLocale();
783 if ( locale )
784 {
785 // try "Comment[locale name]" first
786 comment << wxT("Comment[") + locale->GetName() + wxT("]=");
787 nIndex = file.pIndexOf(comment);
788 }
789#endif
790
791 if ( nIndex == wxNOT_FOUND )
792 {
793 comment = wxT("Comment=");
794 nIndex = file.pIndexOf(comment);
795 }
796
797 if ( nIndex != wxNOT_FOUND )
798 mime_desc = file.GetCmd(nIndex);
799 //else: no description
800
801 // next find the extensions
802 wxString mime_extension;
803
804 nIndex = file.pIndexOf(wxT("Patterns="));
805 if ( nIndex != wxNOT_FOUND )
806 {
807 wxString exts = file.GetCmd(nIndex);
808
809 wxStringTokenizer tokenizer(exts, wxT(";"));
810 while ( tokenizer.HasMoreTokens() )
811 {
812 wxString e = tokenizer.GetNextToken();
813
814 // don't support too difficult patterns
815 if ( e.Left(2) != wxT("*.") )
816 continue;
817
818 if ( !mime_extension.empty() )
819 {
820 // separate from the previous ext
821 mime_extension << wxT(' ');
822 }
823
824 mime_extension << e.Mid(2);
825 }
826 }
827
828 sExts.Add(mime_extension);
829
830 // ok, now we can take care of icon:
831
832 nIndex = file.pIndexOf(wxT("Icon="));
833 if ( nIndex != wxNOT_FOUND )
834 {
835 strIcon = file.GetCmd(nIndex);
836
837 wxLogTrace(TRACE_MIME, wxT(" icon %s"), strIcon.c_str());
838
839 // it could be the real path, but more often a short name
840 if (!wxFileExists(strIcon))
841 {
842 // icon is just the short name
843 if ( !strIcon.empty() )
844 {
845 // we must check if the file exists because it may be stored
846 // in many locations, at least ~/.kde and $KDEDIR
847 size_t nDir, nDirs = icondirs.GetCount();
848 for ( nDir = 0; nDir < nDirs; nDir++ )
849 {
850 wxFileName fnameIcon( strIcon );
851 wxFileName fname( icondirs[nDir], fnameIcon.GetName() );
852 fname.SetExt( wxT("png") );
853 if (fname.FileExists())
854 {
855 strIcon = fname.GetFullPath();
856 wxLogTrace(TRACE_MIME, wxT(" iconfile %s"), strIcon.c_str());
857 break;
858 }
859 }
860 }
861 }
862 }
863
864 // now look for lines which know about the application
865 // exec= or DefaultApp=
866
867 nIndex = file.pIndexOf(wxT("DefaultApp"));
868
869 if ( nIndex == wxNOT_FOUND )
870 {
871 // no entry try exec
872 nIndex = file.pIndexOf(wxT("Exec"));
873 }
874
875 if ( nIndex != wxNOT_FOUND )
876 {
877 // we expect %f; others including %F and %U and %u are possible
878 wxString sTmp = file.GetCmd(nIndex);
879 if (0 == sTmp.Replace( wxT("%f"), wxT("%s") ))
880 sTmp += wxT(" %s");
881 entry->AddOrReplaceVerb(wxString(wxT("open")), sTmp );
882 }
883
884 AddToMimeData(mimetype, strIcon, entry, sExts, mime_desc);
885}
886
887void wxMimeTypesManagerImpl::LoadKDELinksForMimeType(const wxString& dirbase,
888 const wxString& subdir,
889 const wxArrayString& icondirs)
890{
891 wxFileName dirname(dirbase, wxEmptyString);
892 dirname.AppendDir(subdir);
893
894 // Don't complain if we don't have permissions to read - it confuses users
895 wxLogNull logNull;
896
897 wxDir dir(dirname.GetPath());
898 if(! dir.IsOpened())
899 return;
900
901 wxLogTrace(TRACE_MIME, wxT("--- Loading from KDE directory %s ---"),
902 dirname.GetPath().c_str());
903
904 wxString filename;
905 bool cont = dir.GetFirst(&filename, wxT("*.kdelnk"), wxDIR_FILES);
906 while(cont) {
907 LoadKDELinksForMimeSubtype(dirname.GetPath(), subdir,
908 filename, icondirs);
909 cont = dir.GetNext(&filename);
910 }
911
912 // new standard for Gnome and KDE
913 cont = dir.GetFirst(&filename, wxT("*.desktop"), wxDIR_FILES);
914 while(cont) {
915 LoadKDELinksForMimeSubtype(dirname.GetPath(), subdir,
916 filename, icondirs);
917 cont = dir.GetNext(&filename);
918 }
919}
920
921void wxMimeTypesManagerImpl::LoadKDELinkFilesFromDir(const wxString& dirname,
922 const wxArrayString& icondirs)
923{
924 // Don't complain if we don't have permissions to read - it confuses users
925 wxLogNull logNull;
926
927 if(! wxDir::Exists(dirname))
928 return;
929
930 wxDir dir(dirname);
931 if ( !dir.IsOpened() )
932 return;
933
934 wxString subdir;
935 bool cont = dir.GetFirst(&subdir, wxEmptyString, wxDIR_DIRS);
936 while ( cont )
937 {
938 LoadKDELinksForMimeType(dirname, subdir, icondirs);
939
940 cont = dir.GetNext(&subdir);
941 }
942}
943
944// Read a KDE .desktop file of type 'Application'
945void wxMimeTypesManagerImpl::LoadKDEApp(const wxString& filename)
946{
947 wxLogTrace(TRACE_MIME, wxT("loading KDE file %s"), filename.c_str());
948
949 wxMimeTextFile file;
950 if ( !file.Open(filename) )
951 return;
952
953 // Here, only type 'Application' should be considered.
954 int nIndex = file.pIndexOf( wxT("Type=") );
955 if (nIndex != wxNOT_FOUND &&
956 file.GetCmd(nIndex).Lower() != wxT("application"))
957 return;
958
959 // The hidden entry specifies a file to be ignored.
960 nIndex = file.pIndexOf( wxT("Hidden=") );
961 if (nIndex != wxNOT_FOUND && file.GetCmd(nIndex).Lower() == wxT("true"))
962 return;
963
964 // Semicolon separated list of mime types handled by the application.
965 nIndex = file.pIndexOf( wxT("MimeType=") );
966 if (nIndex == wxNOT_FOUND)
967 return;
968 wxString mimetypes = file.GetCmd (nIndex);
969
970 // Name of the application
971 wxString nameapp;
972 nIndex = wxNOT_FOUND;
973#if wxUSE_INTL // try "Name[locale name]" first
974 wxLocale *locale = wxGetLocale();
975 if ( locale )
976 nIndex = file.pIndexOf(_T("Name[")+locale->GetName()+_T("]="));
977#endif // wxUSE_INTL
978 if(nIndex == wxNOT_FOUND)
979 nIndex = file.pIndexOf( wxT("Name=") );
980 if(nIndex != wxNOT_FOUND)
981 nameapp = file.GetCmd(nIndex);
982
983 // Icon of the application.
984 wxString nameicon, namemini;
985 nIndex = wxNOT_FOUND;
986#if wxUSE_INTL // try "Icon[locale name]" first
987 if ( locale )
988 nIndex = file.pIndexOf(_T("Icon[")+locale->GetName()+_T("]="));
989#endif // wxUSE_INTL
990 if(nIndex == wxNOT_FOUND)
991 nIndex = file.pIndexOf( wxT("Icon=") );
992 if(nIndex != wxNOT_FOUND) {
993 nameicon = wxString(wxT("--icon ")) + file.GetCmd(nIndex);
994 namemini = wxString(wxT("--miniicon ")) + file.GetCmd(nIndex);
995 }
996
997 // Replace some of the field code in the 'Exec' entry.
998 // TODO: deal with %d, %D, %n, %N, %k and %v (but last one is deprecated)
999 nIndex = file.pIndexOf( wxT("Exec=") );
1000 if (nIndex == wxNOT_FOUND)
1001 return;
1002 wxString sCmd = file.GetCmd(nIndex);
1003 // we expect %f; others including %F and %U and %u are possible
1004 sCmd.Replace(wxT("%F"), wxT("%f"));
1005 sCmd.Replace(wxT("%U"), wxT("%f"));
1006 sCmd.Replace(wxT("%u"), wxT("%f"));
1007 if (0 == sCmd.Replace ( wxT("%f"), wxT("%s") ))
1008 sCmd = sCmd + wxT(" %s");
1009 sCmd.Replace(wxT("%c"), nameapp);
1010 sCmd.Replace(wxT("%i"), nameicon);
1011 sCmd.Replace(wxT("%m"), namemini);
1012
1013 wxStringTokenizer tokenizer(mimetypes, _T(";"));
1014 while(tokenizer.HasMoreTokens()) {
1015 wxString mimetype = tokenizer.GetNextToken().Lower();
1016 nIndex = m_aTypes.Index(mimetype);
1017 if(nIndex != wxNOT_FOUND) { // is this a known MIME type?
1018 wxMimeTypeCommands* entry = m_aEntries[nIndex];
1019 entry->AddOrReplaceVerb(wxT("open"), sCmd);
1020 }
1021 }
1022}
1023
1024void wxMimeTypesManagerImpl::LoadKDEAppsFilesFromDir(const wxString& dirname)
1025{
1026 // Don't complain if we don't have permissions to read - it confuses users
1027 wxLogNull logNull;
1028
1029 if(! wxDir::Exists(dirname))
1030 return;
1031 wxDir dir(dirname);
1032 if ( !dir.IsOpened() )
1033 return;
1034
1035 wxString filename;
1036 // Look into .desktop files
1037 bool cont = dir.GetFirst(&filename, _T("*.desktop"), wxDIR_FILES);
1038 while(cont) {
1039 wxFileName p(dirname, filename);
1040 LoadKDEApp( p.GetFullPath() );
1041 cont = dir.GetNext(&filename);
1042 }
1043 // Look recursively into subdirs
1044 cont = dir.GetFirst(&filename, wxEmptyString, wxDIR_DIRS);
1045 while(cont) {
1046 wxFileName p(dirname, wxEmptyString);
1047 p.AppendDir(filename);
1048 LoadKDEAppsFilesFromDir( p.GetPath() );
1049 cont = dir.GetNext(&filename);
1050 }
1051}
1052
1053// Return base KDE directories.
1054// 1) Environment variable $KDEHOME, or "~/.kde" if not set.
1055// 2) List of directories in colon separated environment variable $KDEDIRS.
1056// 3) Environment variable $KDEDIR in case $KDEDIRS is not set.
1057// Notice at least the local kde directory is added to the list. If it is the
1058// only one, use later the application 'kde-config' to get additional paths.
1059static void GetKDEBaseDirs(wxArrayString& basedirs)
1060{
1061 wxString env = wxGetenv( wxT("KDEHOME") );
1062 if(env.IsEmpty())
1063 env = wxGetHomeDir() + wxT("/.kde");
1064 basedirs.Add(env);
1065
1066 env = wxGetenv( wxT("KDEDIRS") );
1067 if(env.IsEmpty()) {
1068 env = wxGetenv( wxT("KDEDIR") );
1069 if(! env.IsEmpty())
1070 basedirs.Add(env);
1071 } else {
1072 wxStringTokenizer tokenizer(env, wxT(":"));
1073 while(tokenizer.HasMoreTokens())
1074 basedirs.Add( tokenizer.GetNextToken() );
1075 }
1076}
1077
1078static wxString ReadPathFromKDEConfig(const wxString& request)
1079{
1080 wxString str;
1081 wxArrayString output;
1082 if(wxExecute(wxT("kde-config --path ")+request, output) == 0 &&
1083 output.GetCount() > 0)
1084 str = output.Item(0);
1085 return str;
1086}
1087
1088// Try to find the "Theme" entry in the configuration file, provided it exists.
1089static wxString GetKDEThemeInFile(const wxFileName& filename)
1090{
1091 wxString theme;
1092 wxMimeTextFile config;
1093 if ( filename.FileExists() && config.Open(filename.GetFullPath()) )
1094 {
1095 size_t cnt = config.GetLineCount();
1096 for ( size_t i = 0; i < cnt; i++ )
1097 {
1098 if ( config[i].StartsWith(wxT("Theme="), &theme) )
1099 break;
1100 }
1101 }
1102
1103 return theme;
1104}
1105
1106// Try to find a file "kdeglobals" in one of the directories and read the
1107// "Theme" entry there.
1108static wxString GetKDETheme(const wxArrayString& basedirs)
1109{
1110 wxString theme;
1111 for(size_t i = 0; i < basedirs.GetCount(); i++) {
1112 wxFileName filename(basedirs.Item(i), wxEmptyString);
1113 filename.AppendDir( wxT("share") );
1114 filename.AppendDir( wxT("config") );
1115 filename.SetName( wxT("kdeglobals") );
1116 theme = GetKDEThemeInFile(filename);
1117 if(! theme.IsEmpty())
1118 return theme;
1119 }
1120 // If $KDEDIRS and $KDEDIR were set, we try nothing more. Otherwise, we
1121 // try to get the configuration file with 'kde-config'.
1122 if(basedirs.GetCount() > 1)
1123 return theme;
1124 wxString paths = ReadPathFromKDEConfig(wxT("config"));
1125 if(! paths.IsEmpty()) {
1126 wxStringTokenizer tokenizer(paths, wxT(":"));
1127 while( tokenizer.HasMoreTokens() ) {
1128 wxFileName filename(tokenizer.GetNextToken(), wxT("kdeglobals"));
1129 theme = GetKDEThemeInFile(filename);
1130 if(! theme.IsEmpty())
1131 return theme;
1132 }
1133 }
1134 return theme;
1135}
1136
1137// Get list of directories of icons.
1138static void GetKDEIconDirs(const wxArrayString& basedirs,
1139 wxArrayString& icondirs)
1140{
1141 wxString theme = GetKDETheme(basedirs);
1142 if(theme.IsEmpty())
1143 theme = wxT("default.kde");
1144
1145 for(size_t i = 0; i < basedirs.GetCount(); i++) {
1146 wxFileName dirname(basedirs.Item(i), wxEmptyString);
1147 dirname.AppendDir( wxT("share") );
1148 dirname.AppendDir( wxT("icons") );
1149 dirname.AppendDir(theme);
1150 dirname.AppendDir( wxT("32x32") );
1151 dirname.AppendDir( wxT("mimetypes") );
1152 if( wxDir::Exists( dirname.GetPath() ) )
1153 icondirs.Add( dirname.GetPath() );
1154 }
1155
1156 // If $KDEDIRS and $KDEDIR were not set, use 'kde-config'
1157 if(basedirs.GetCount() > 1)
1158 return;
1159 wxString paths = ReadPathFromKDEConfig(wxT("icon"));
1160 if(! paths.IsEmpty()) {
1161 wxStringTokenizer tokenizer(paths, wxT(":"));
1162 while( tokenizer.HasMoreTokens() ) {
1163 wxFileName dirname(tokenizer.GetNextToken(), wxEmptyString);
1164 dirname.AppendDir(theme);
1165 dirname.AppendDir( wxT("32x32") );
1166 dirname.AppendDir( wxT("mimetypes") );
1167 if(icondirs.Index(dirname.GetPath()) == wxNOT_FOUND &&
1168 wxDir::Exists( dirname.GetPath() ) )
1169 icondirs.Add( dirname.GetPath() );
1170 }
1171 }
1172}
1173
1174// Get list of directories of mime types.
1175static void GetKDEMimeDirs(const wxArrayString& basedirs,
1176 wxArrayString& mimedirs)
1177{
1178 for(size_t i = 0; i < basedirs.GetCount(); i++) {
1179 wxFileName dirname(basedirs.Item(i), wxEmptyString);
1180 dirname.AppendDir( wxT("share") );
1181 dirname.AppendDir( wxT("mimelnk") );
1182 if( wxDir::Exists( dirname.GetPath() ) )
1183 mimedirs.Add( dirname.GetPath() );
1184 }
1185
1186 // If $KDEDIRS and $KDEDIR were not set, use 'kde-config'
1187 if(basedirs.GetCount() > 1)
1188 return;
1189 wxString paths = ReadPathFromKDEConfig(wxT("mime"));
1190 if(! paths.IsEmpty()) {
1191 wxStringTokenizer tokenizer(paths, wxT(":"));
1192 while( tokenizer.HasMoreTokens() ) {
1193 wxFileName p(tokenizer.GetNextToken(), wxEmptyString);
1194 wxString dirname = p.GetPath(); // To remove possible trailing '/'
1195 if(mimedirs.Index(dirname) == wxNOT_FOUND &&
1196 wxDir::Exists(dirname) )
1197 mimedirs.Add(dirname);
1198 }
1199 }
1200}
1201
1202// Get list of directories of application desktop files.
1203static void GetKDEAppsDirs(const wxArrayString& basedirs,
1204 wxArrayString& appsdirs)
1205{
1206 for(size_t i = 0; i < basedirs.GetCount(); i++) {
1207 wxFileName dirname(basedirs.Item(i), wxEmptyString);
1208 dirname.AppendDir( wxT("share") );
1209 dirname.AppendDir( wxT("applnk") );
1210 if( wxDir::Exists( dirname.GetPath() ) )
1211 appsdirs.Add( dirname.GetPath() );
1212 }
1213
1214 // If $KDEDIRS and $KDEDIR were not set, use 'kde-config'
1215 if(basedirs.GetCount() > 1)
1216 return;
1217 wxString paths = ReadPathFromKDEConfig(wxT("apps"));
1218 if(! paths.IsEmpty()) {
1219 wxStringTokenizer tokenizer(paths, wxT(":"));
1220 while( tokenizer.HasMoreTokens() ) {
1221 wxFileName p(tokenizer.GetNextToken(), wxEmptyString);
1222 wxString dirname = p.GetPath(); // To remove possible trailing '/'
1223 if(appsdirs.Index(dirname) == wxNOT_FOUND &&
1224 wxDir::Exists(dirname) )
1225 appsdirs.Add(dirname);
1226 }
1227 }
1228 paths = ReadPathFromKDEConfig(wxT("xdgdata-apps"));
1229 if(! paths.IsEmpty()) {
1230 wxStringTokenizer tokenizer(paths, wxT(":"));
1231 while( tokenizer.HasMoreTokens() ) {
1232 wxFileName p(tokenizer.GetNextToken(), wxEmptyString);
1233 wxString dirname = p.GetPath(); // To remove possible trailing '/'
1234 if(appsdirs.Index(dirname) == wxNOT_FOUND &&
1235 wxDir::Exists(dirname) )
1236 appsdirs.Add(dirname);
1237 }
1238 }
1239}
1240
1241// Fill database with all mime types.
1242void wxMimeTypesManagerImpl::GetKDEMimeInfo(const wxString& sExtraDir)
1243{
1244 wxArrayString basedirs;
1245 GetKDEBaseDirs(basedirs);
1246
1247 wxArrayString icondirs;
1248 GetKDEIconDirs(basedirs, icondirs);
1249 wxArrayString mimedirs;
1250 GetKDEMimeDirs(basedirs, mimedirs);
1251 wxArrayString appsdirs;
1252 GetKDEAppsDirs(basedirs, appsdirs);
1253
1254 if(! sExtraDir.IsEmpty()) {
1255 icondirs.Add(sExtraDir + wxT("/icons"));
1256 mimedirs.Add(sExtraDir + wxT("/mimelnk"));
1257 appsdirs.Add(sExtraDir + wxT("/applnk"));
1258 }
1259
1260 // Load mime types
1261 size_t nDirs = mimedirs.GetCount(), nDir;
1262 for(nDir = 0; nDir < nDirs; nDir++)
1263 LoadKDELinkFilesFromDir(mimedirs[nDir], icondirs);
1264
1265 // Load application files and associate them to corresponding mime types.
1266 nDirs = appsdirs.GetCount();
1267 for(nDir = 0; nDir < nDirs; nDir++)
1268 LoadKDEAppsFilesFromDir(appsdirs[nDir]);
1269}
1270
1271// ----------------------------------------------------------------------------
1272// wxFileTypeImpl (Unix)
1273// ----------------------------------------------------------------------------
1274
1275wxString wxFileTypeImpl::GetExpandedCommand(const wxString & verb, const wxFileType::MessageParameters& params) const
1276{
1277 wxString sTmp;
1278 size_t i = 0;
1279 while ( (i < m_index.GetCount() ) && sTmp.empty() )
1280 {
1281 sTmp = m_manager->GetCommand( verb, m_index[i] );
1282 i++;
1283 }
1284
1285 return wxFileType::ExpandCommand(sTmp, params);
1286}
1287
1288bool wxFileTypeImpl::GetIcon(wxIconLocation *iconLoc) const
1289{
1290 wxString sTmp;
1291 size_t i = 0;
1292 while ( (i < m_index.GetCount() ) && sTmp.empty() )
1293 {
1294 sTmp = m_manager->m_aIcons[m_index[i]];
1295 i++;
1296 }
1297
1298 if ( sTmp.empty() )
1299 return false;
1300
1301 if ( iconLoc )
1302 {
1303 iconLoc->SetFileName(sTmp);
1304 }
1305
1306 return true;
1307}
1308
1309bool wxFileTypeImpl::GetMimeTypes(wxArrayString& mimeTypes) const
1310{
1311 mimeTypes.Clear();
1312 size_t nCount = m_index.GetCount();
1313 for (size_t i = 0; i < nCount; i++)
1314 mimeTypes.Add(m_manager->m_aTypes[m_index[i]]);
1315
1316 return true;
1317}
1318
1319size_t wxFileTypeImpl::GetAllCommands(wxArrayString *verbs,
1320 wxArrayString *commands,
1321 const wxFileType::MessageParameters& params) const
1322{
1323 wxString vrb, cmd, sTmp;
1324 size_t count = 0;
1325 wxMimeTypeCommands * sPairs;
1326
1327 // verbs and commands have been cleared already in mimecmn.cpp...
1328 // if we find no entries in the exact match, try the inexact match
1329 for (size_t n = 0; ((count == 0) && (n < m_index.GetCount())); n++)
1330 {
1331 // list of verb = command pairs for this mimetype
1332 sPairs = m_manager->m_aEntries [m_index[n]];
1333 size_t i;
1334 for ( i = 0; i < sPairs->GetCount(); i++ )
1335 {
1336 vrb = sPairs->GetVerb(i);
1337 // some gnome entries have "." inside
1338 vrb = vrb.AfterLast(wxT('.'));
1339 cmd = sPairs->GetCmd(i);
1340 if (! cmd.empty() )
1341 {
1342 cmd = wxFileType::ExpandCommand(cmd, params);
1343 count++;
1344 if ( vrb.IsSameAs(wxT("open")))
1345 {
1346 if ( verbs )
1347 verbs->Insert(vrb, 0u);
1348 if ( commands )
1349 commands ->Insert(cmd, 0u);
1350 }
1351 else
1352 {
1353 if ( verbs )
1354 verbs->Add(vrb);
1355 if ( commands )
1356 commands->Add(cmd);
1357 }
1358 }
1359 }
1360 }
1361
1362 return count;
1363}
1364
1365bool wxFileTypeImpl::GetExtensions(wxArrayString& extensions)
1366{
1367 const wxString strExtensions = m_manager->GetExtension(m_index[0]);
1368 extensions.Empty();
1369
1370 // one extension in the space or comma-delimited list
1371 wxString strExt;
1372 wxString::const_iterator end = strExtensions.end();
1373 for ( wxString::const_iterator p = strExtensions.begin(); /* nothing */; ++p )
1374 {
1375 if ( p == end || *p == wxT(' ') || *p == wxT(',') )
1376 {
1377 if ( !strExt.empty() )
1378 {
1379 extensions.Add(strExt);
1380 strExt.Empty();
1381 }
1382 //else: repeated spaces
1383 // (shouldn't happen, but it's not that important if it does happen)
1384
1385 if ( p == end )
1386 break;
1387 }
1388 else if ( *p == wxT('.') )
1389 {
1390 // remove the dot from extension (but only if it's the first char)
1391 if ( !strExt.empty() )
1392 {
1393 strExt += wxT('.');
1394 }
1395 //else: no, don't append it
1396 }
1397 else
1398 {
1399 strExt += *p;
1400 }
1401 }
1402
1403 return true;
1404}
1405
1406// set an arbitrary command:
1407// could adjust the code to ask confirmation if it already exists and
1408// overwriteprompt is true, but this is currently ignored as *Associate* has
1409// no overwrite prompt
1410bool
1411wxFileTypeImpl::SetCommand(const wxString& cmd,
1412 const wxString& verb,
1413 bool WXUNUSED(overwriteprompt))
1414{
1415 wxArrayString strExtensions;
1416 wxString strDesc, strIcon;
1417
1418 wxArrayString strTypes;
1419 GetMimeTypes(strTypes);
1420 if ( strTypes.IsEmpty() )
1421 return false;
1422
1423 wxMimeTypeCommands *entry = new wxMimeTypeCommands();
1424 entry->Add(verb + wxT("=") + cmd + wxT(" %s "));
1425
1426 bool ok = false;
1427 size_t nCount = strTypes.GetCount();
1428 for ( size_t i = 0; i < nCount; i++ )
1429 {
1430 if ( m_manager->DoAssociation
1431 (
1432 strTypes[i],
1433 strIcon,
1434 entry,
1435 strExtensions,
1436 strDesc
1437 ) )
1438 {
1439 // DoAssociation() took ownership of entry, don't delete it below
1440 ok = true;
1441 }
1442 }
1443
1444 if ( !ok )
1445 delete entry;
1446
1447 return ok;
1448}
1449
1450// ignore index on the grounds that we only have one icon in a Unix file
1451bool wxFileTypeImpl::SetDefaultIcon(const wxString& strIcon, int WXUNUSED(index))
1452{
1453 if (strIcon.empty())
1454 return false;
1455
1456 wxArrayString strExtensions;
1457 wxString strDesc;
1458
1459 wxArrayString strTypes;
1460 GetMimeTypes(strTypes);
1461 if ( strTypes.IsEmpty() )
1462 return false;
1463
1464 wxMimeTypeCommands *entry = new wxMimeTypeCommands();
1465 bool ok = false;
1466 size_t nCount = strTypes.GetCount();
1467 for ( size_t i = 0; i < nCount; i++ )
1468 {
1469 if ( m_manager->DoAssociation
1470 (
1471 strTypes[i],
1472 strIcon,
1473 entry,
1474 strExtensions,
1475 strDesc
1476 ) )
1477 {
1478 // we don't need to free entry now, DoAssociation() took ownership
1479 // of it
1480 ok = true;
1481 }
1482 }
1483
1484 if ( !ok )
1485 delete entry;
1486
1487 return ok;
1488}
1489
1490// ----------------------------------------------------------------------------
1491// wxMimeTypesManagerImpl (Unix)
1492// ----------------------------------------------------------------------------
1493
1494wxMimeTypesManagerImpl::wxMimeTypesManagerImpl()
1495{
1496 m_initialized = false;
1497 m_mailcapStylesInited = 0;
1498}
1499
1500void wxMimeTypesManagerImpl::InitIfNeeded()
1501{
1502 if ( !m_initialized )
1503 {
1504 // set the flag first to prevent recursion
1505 m_initialized = true;
1506
1507 wxString wm = wxTheApp->GetTraits()->GetDesktopEnvironment();
1508
1509 if (wm == wxT("KDE"))
1510 Initialize( wxMAILCAP_KDE );
1511 else if (wm == wxT("GNOME"))
1512 Initialize( wxMAILCAP_GNOME );
1513 else
1514 Initialize();
1515 }
1516}
1517
1518// read system and user mailcaps and other files
1519void wxMimeTypesManagerImpl::Initialize(int mailcapStyles,
1520 const wxString& sExtraDir)
1521{
1522 // read mimecap amd mime.types
1523 if ( (mailcapStyles & wxMAILCAP_NETSCAPE) ||
1524 (mailcapStyles & wxMAILCAP_STANDARD) )
1525 GetMimeInfo(sExtraDir);
1526
1527 // read GNOME tables
1528 if (mailcapStyles & wxMAILCAP_GNOME)
1529 GetGnomeMimeInfo(sExtraDir);
1530
1531 // read KDE tables which are never installed on OpenVMS
1532#ifndef __VMS
1533 if (mailcapStyles & wxMAILCAP_KDE)
1534 GetKDEMimeInfo(sExtraDir);
1535#endif
1536
1537 m_mailcapStylesInited |= mailcapStyles;
1538}
1539
1540// clear data so you can read another group of WM files
1541void wxMimeTypesManagerImpl::ClearData()
1542{
1543 m_aTypes.Clear();
1544 m_aIcons.Clear();
1545 m_aExtensions.Clear();
1546 m_aDescriptions.Clear();
1547
1548 WX_CLEAR_ARRAY(m_aEntries);
1549 m_aEntries.Empty();
1550
1551 m_mailcapStylesInited = 0;
1552}
1553
1554wxMimeTypesManagerImpl::~wxMimeTypesManagerImpl()
1555{
1556 ClearData();
1557}
1558
1559void wxMimeTypesManagerImpl::GetMimeInfo(const wxString& sExtraDir)
1560{
1561 // read this for netscape or Metamail formats
1562
1563 // directories where we look for mailcap and mime.types by default
1564 // used by netscape and pine and other mailers, using 2 different formats!
1565
1566 // (taken from metamail(1) sources)
1567 //
1568 // although RFC 1524 specifies the search path of
1569 // /etc/:/usr/etc:/usr/local/etc only, it doesn't hurt to search in more
1570 // places - OTOH, the RFC also says that this path can be changed with
1571 // MAILCAPS environment variable (containing the colon separated full
1572 // filenames to try) which is not done yet (TODO?)
1573
1574 wxString strHome = wxGetenv(wxT("HOME"));
1575
1576 wxArrayString dirs;
1577 dirs.Add( strHome + wxT("/.") );
1578 dirs.Add( wxT("/etc/") );
1579 dirs.Add( wxT("/usr/etc/") );
1580 dirs.Add( wxT("/usr/local/etc/") );
1581 dirs.Add( wxT("/etc/mail/") );
1582 dirs.Add( wxT("/usr/public/lib/") );
1583 if (!sExtraDir.empty())
1584 dirs.Add( sExtraDir + wxT("/") );
1585
1586 wxString file;
1587 size_t nDirs = dirs.GetCount();
1588 for ( size_t nDir = 0; nDir < nDirs; nDir++ )
1589 {
1590 file = dirs[nDir];
1591 file += wxT("mailcap");
1592 if ( wxFile::Exists(file) )
1593 {
1594 ReadMailcap(file);
1595 }
1596
1597 file = dirs[nDir];
1598 file += wxT("mime.types");
1599 if ( wxFile::Exists(file) )
1600 ReadMimeTypes(file);
1601 }
1602}
1603
1604bool wxMimeTypesManagerImpl::WriteToMimeTypes(int index, bool delete_index)
1605{
1606 // check we have the right manager
1607 if (! ( m_mailcapStylesInited & wxMAILCAP_STANDARD) )
1608 return false;
1609
1610 bool bTemp;
1611 wxString strHome = wxGetenv(wxT("HOME"));
1612
1613 // and now the users mailcap
1614 wxString strUserMailcap = strHome + wxT("/.mime.types");
1615
1616 wxMimeTextFile file;
1617 if ( wxFile::Exists(strUserMailcap) )
1618 {
1619 bTemp = file.Open(strUserMailcap);
1620 }
1621 else
1622 {
1623 if (delete_index)
1624 return false;
1625
1626 bTemp = file.Create(strUserMailcap);
1627 }
1628
1629 if (bTemp)
1630 {
1631 int nIndex;
1632 // test for netscape's header and return false if its found
1633 nIndex = file.pIndexOf(wxT("#--Netscape"));
1634 if (nIndex != wxNOT_FOUND)
1635 {
1636 wxFAIL_MSG(wxT("Error in .mime.types\nTrying to mix Netscape and Metamail formats\nFile not modified"));
1637 return false;
1638 }
1639
1640 // write it in alternative format
1641 // get rid of unwanted entries
1642 wxString strType = m_aTypes[index];
1643 nIndex = file.pIndexOf(strType);
1644
1645 // get rid of all the unwanted entries...
1646 if (nIndex != wxNOT_FOUND)
1647 file.CommentLine(nIndex);
1648
1649 if (!delete_index)
1650 {
1651 // add the new entries in
1652 wxString sTmp = strType.Append( wxT(' '), 40 - strType.Len() );
1653 sTmp += m_aExtensions[index];
1654 file.AddLine(sTmp);
1655 }
1656
1657 bTemp = file.Write();
1658 file.Close();
1659 }
1660
1661 return bTemp;
1662}
1663
1664bool wxMimeTypesManagerImpl::WriteToNSMimeTypes(int index, bool delete_index)
1665{
1666 //check we have the right managers
1667 if (! ( m_mailcapStylesInited & wxMAILCAP_NETSCAPE) )
1668 return false;
1669
1670 bool bTemp;
1671 wxString strHome = wxGetenv(wxT("HOME"));
1672
1673 // and now the users mailcap
1674 wxString strUserMailcap = strHome + wxT("/.mime.types");
1675
1676 wxMimeTextFile file;
1677 if ( wxFile::Exists(strUserMailcap) )
1678 {
1679 bTemp = file.Open(strUserMailcap);
1680 }
1681 else
1682 {
1683 if (delete_index)
1684 return false;
1685
1686 bTemp = file.Create(strUserMailcap);
1687 }
1688
1689 if (bTemp)
1690 {
1691 // write it in the format that Netscape uses
1692 int nIndex;
1693 // test for netscape's header and insert if required...
1694 // this is a comment so use true
1695 nIndex = file.pIndexOf(wxT("#--Netscape"), true);
1696 if (nIndex == wxNOT_FOUND)
1697 {
1698 // either empty file or metamail format
1699 // at present we can't cope with mixed formats, so exit to preseve
1700 // metamail entreies
1701 if (file.GetLineCount() > 0)
1702 {
1703 wxFAIL_MSG(wxT(".mime.types File not in Netscape format\nNo entries written to\n.mime.types or to .mailcap"));
1704 return false;
1705 }
1706
1707 file.InsertLine(wxT( "#--Netscape Communications Corporation MIME Information" ), 0);
1708 nIndex = 0;
1709 }
1710
1711 wxString strType = wxT("type=") + m_aTypes[index];
1712 nIndex = file.pIndexOf(strType);
1713
1714 // get rid of all the unwanted entries...
1715 if (nIndex != wxNOT_FOUND)
1716 {
1717 wxString sOld = file[nIndex];
1718 while ( (sOld.Contains(wxT("\\"))) && (nIndex < (int) file.GetLineCount()) )
1719 {
1720 file.CommentLine(nIndex);
1721 sOld = file[nIndex];
1722
1723 wxLogTrace(TRACE_MIME, wxT("--- Deleting from mime.types line '%d %s' ---"), nIndex, sOld.c_str());
1724
1725 nIndex++;
1726 }
1727
1728 if (nIndex < (int) file.GetLineCount())
1729 file.CommentLine(nIndex);
1730 }
1731 else
1732 nIndex = (int) file.GetLineCount();
1733
1734 wxString sTmp = strType + wxT(" \\");
1735 if (!delete_index)
1736 file.InsertLine(sTmp, nIndex);
1737
1738 if ( ! m_aDescriptions.Item(index).empty() )
1739 {
1740 sTmp = wxT("desc=\"") + m_aDescriptions[index]+ wxT("\" \\"); //.trim ??
1741 if (!delete_index)
1742 {
1743 nIndex++;
1744 file.InsertLine(sTmp, nIndex);
1745 }
1746 }
1747
1748 wxString sExts = m_aExtensions.Item(index);
1749 sTmp = wxT("exts=\"") + sExts.Trim(false).Trim() + wxT("\"");
1750 if (!delete_index)
1751 {
1752 nIndex++;
1753 file.InsertLine(sTmp, nIndex);
1754 }
1755
1756 bTemp = file.Write();
1757 file.Close();
1758 }
1759
1760 return bTemp;
1761}
1762
1763bool wxMimeTypesManagerImpl::WriteToMailCap(int index, bool delete_index)
1764{
1765 //check we have the right managers
1766 if ( !( ( m_mailcapStylesInited & wxMAILCAP_NETSCAPE) ||
1767 ( m_mailcapStylesInited & wxMAILCAP_STANDARD) ) )
1768 return false;
1769
1770 bool bTemp = false;
1771 wxString strHome = wxGetenv(wxT("HOME"));
1772
1773 // and now the users mailcap
1774 wxString strUserMailcap = strHome + wxT("/.mailcap");
1775
1776 wxMimeTextFile file;
1777 if ( wxFile::Exists(strUserMailcap) )
1778 {
1779 bTemp = file.Open(strUserMailcap);
1780 }
1781 else
1782 {
1783 if (delete_index)
1784 return false;
1785
1786 bTemp = file.Create(strUserMailcap);
1787 }
1788
1789 if (bTemp)
1790 {
1791 // now got a file we can write to ....
1792 wxMimeTypeCommands * entries = m_aEntries[index];
1793 size_t iOpen;
1794 wxString sCmd = entries->GetCommandForVerb(wxT("open"), &iOpen);
1795 wxString sTmp;
1796
1797 sTmp = m_aTypes[index];
1798 wxString sOld;
1799 int nIndex = file.pIndexOf(sTmp);
1800
1801 // get rid of all the unwanted entries...
1802 if (nIndex == wxNOT_FOUND)
1803 {
1804 nIndex = (int) file.GetLineCount();
1805 }
1806 else
1807 {
1808 sOld = file[nIndex];
1809 wxLogTrace(TRACE_MIME, wxT("--- Deleting from mailcap line '%d' ---"), nIndex);
1810
1811 while ( (sOld.Contains(wxT("\\"))) && (nIndex < (int) file.GetLineCount()) )
1812 {
1813 file.CommentLine(nIndex);
1814 if (nIndex < (int) file.GetLineCount())
1815 sOld = sOld + file[nIndex];
1816 }
1817
1818 if (nIndex < (int)
1819 file.GetLineCount()) file.CommentLine(nIndex);
1820 }
1821
1822 sTmp += wxT(";") + sCmd; //includes wxT(" %s ");
1823
1824 // write it in the format that Netscape uses (default)
1825 if (! ( m_mailcapStylesInited & wxMAILCAP_STANDARD ) )
1826 {
1827 if (! delete_index)
1828 file.InsertLine(sTmp, nIndex);
1829 nIndex++;
1830 }
1831 else
1832 {
1833 // write extended format
1834
1835 // TODO - FIX this code:
1836 // ii) lost entries
1837 // sOld holds all the entries, but our data store only has some
1838 // eg test= is not stored
1839
1840 // so far we have written the mimetype and command out
1841 wxStringTokenizer sT(sOld, wxT(";\\"));
1842 if (sT.CountTokens() > 2)
1843 {
1844 // first one mimetype; second one command, rest unknown...
1845 wxString s;
1846 s = sT.GetNextToken();
1847 s = sT.GetNextToken();
1848
1849 // first unknown
1850 s = sT.GetNextToken();
1851 while ( ! s.empty() )
1852 {
1853 bool bKnownToken = false;
1854 if (s.Contains(wxT("description=")))
1855 bKnownToken = true;
1856 if (s.Contains(wxT("x11-bitmap=")))
1857 bKnownToken = true;
1858
1859 size_t i;
1860 size_t nCount = entries->GetCount();
1861 for (i=0; i < nCount; i++)
1862 {
1863 if (s.Contains(entries->GetVerb(i)))
1864 bKnownToken = true;
1865 }
1866
1867 if (!bKnownToken)
1868 {
1869 sTmp += wxT("; \\");
1870 file.InsertLine(sTmp, nIndex);
1871 sTmp = s;
1872 }
1873
1874 s = sT.GetNextToken();
1875 }
1876 }
1877
1878 if (! m_aDescriptions[index].empty() )
1879 {
1880 sTmp += wxT("; \\");
1881 file.InsertLine(sTmp, nIndex);
1882 nIndex++;
1883 sTmp = wxT(" description=\"") + m_aDescriptions[index] + wxT("\"");
1884 }
1885
1886 if (! m_aIcons[index].empty() )
1887 {
1888 sTmp += wxT("; \\");
1889 file.InsertLine(sTmp, nIndex);
1890 nIndex++;
1891 sTmp = wxT(" x11-bitmap=\"") + m_aIcons[index] + wxT("\"");
1892 }
1893
1894 if ( entries->GetCount() > 1 )
1895 {
1896 size_t i;
1897 for (i=0; i < entries->GetCount(); i++)
1898 if ( i != iOpen )
1899 {
1900 sTmp += wxT("; \\");
1901 file.InsertLine(sTmp, nIndex);
1902 nIndex++;
1903 sTmp = wxT(" ") + entries->GetVerbCmd(i);
1904 }
1905 }
1906
1907 file.InsertLine(sTmp, nIndex);
1908 nIndex++;
1909 }
1910
1911 bTemp = file.Write();
1912 file.Close();
1913 }
1914
1915 return bTemp;
1916}
1917
1918wxFileType * wxMimeTypesManagerImpl::Associate(const wxFileTypeInfo& ftInfo)
1919{
1920 InitIfNeeded();
1921
1922 wxString strType = ftInfo.GetMimeType();
1923 wxString strDesc = ftInfo.GetDescription();
1924 wxString strIcon = ftInfo.GetIconFile();
1925
1926 wxMimeTypeCommands *entry = new wxMimeTypeCommands();
1927
1928 if ( ! ftInfo.GetOpenCommand().empty())
1929 entry->Add(wxT("open=") + ftInfo.GetOpenCommand() + wxT(" %s "));
1930 if ( ! ftInfo.GetPrintCommand().empty())
1931 entry->Add(wxT("print=") + ftInfo.GetPrintCommand() + wxT(" %s "));
1932
1933 // now find where these extensions are in the data store and remove them
1934 wxArrayString sA_Exts = ftInfo.GetExtensions();
1935 wxString sExt, sExtStore;
1936 size_t i, nIndex;
1937 size_t nExtCount = sA_Exts.GetCount();
1938 for (i=0; i < nExtCount; i++)
1939 {
1940 sExt = sA_Exts.Item(i);
1941
1942 // clean up to just a space before and after
1943 sExt.Trim().Trim(false);
1944 sExt = wxT(' ') + sExt + wxT(' ');
1945 size_t nCount = m_aExtensions.GetCount();
1946 for (nIndex = 0; nIndex < nCount; nIndex++)
1947 {
1948 sExtStore = m_aExtensions.Item(nIndex);
1949 if (sExtStore.Replace(sExt, wxT(" ") ) > 0)
1950 m_aExtensions.Item(nIndex) = sExtStore;
1951 }
1952 }
1953
1954 if ( !DoAssociation(strType, strIcon, entry, sA_Exts, strDesc) )
1955 return NULL;
1956
1957 return GetFileTypeFromMimeType(strType);
1958}
1959
1960bool wxMimeTypesManagerImpl::DoAssociation(const wxString& strType,
1961 const wxString& strIcon,
1962 wxMimeTypeCommands *entry,
1963 const wxArrayString& strExtensions,
1964 const wxString& strDesc)
1965{
1966 int nIndex = AddToMimeData(strType, strIcon, entry, strExtensions, strDesc, true);
1967
1968 if ( nIndex == wxNOT_FOUND )
1969 return false;
1970
1971 return WriteMimeInfo(nIndex, false);
1972}
1973
1974bool wxMimeTypesManagerImpl::WriteMimeInfo(int nIndex, bool delete_mime )
1975{
1976 bool ok = true;
1977
1978 if ( m_mailcapStylesInited & wxMAILCAP_STANDARD )
1979 {
1980 // write in metamail format;
1981 if (WriteToMimeTypes(nIndex, delete_mime) )
1982 if ( WriteToMailCap(nIndex, delete_mime) )
1983 ok = false;
1984 }
1985
1986 if ( m_mailcapStylesInited & wxMAILCAP_NETSCAPE )
1987 {
1988 // write in netsacpe format;
1989 if (WriteToNSMimeTypes(nIndex, delete_mime) )
1990 if ( WriteToMailCap(nIndex, delete_mime) )
1991 ok = false;
1992 }
1993
1994 // Don't write GNOME files here as this is not
1995 // allowed and simply doesn't work
1996
1997 if (m_mailcapStylesInited & wxMAILCAP_KDE)
1998 {
1999 // write in KDE format;
2000 if (WriteKDEMimeFile(nIndex, delete_mime) )
2001 ok = false;
2002 }
2003
2004 return ok;
2005}
2006
2007int wxMimeTypesManagerImpl::AddToMimeData(const wxString& strType,
2008 const wxString& strIcon,
2009 wxMimeTypeCommands *entry,
2010 const wxArrayString& strExtensions,
2011 const wxString& strDesc,
2012 bool replaceExisting)
2013{
2014 InitIfNeeded();
2015
2016 // ensure mimetype is always lower case
2017 wxString mimeType = strType.Lower();
2018
2019 // is this a known MIME type?
2020 int nIndex = m_aTypes.Index(mimeType);
2021 if ( nIndex == wxNOT_FOUND )
2022 {
2023 // new file type
2024 m_aTypes.Add(mimeType);
2025 m_aIcons.Add(strIcon);
2026 m_aEntries.Add(entry ? entry : new wxMimeTypeCommands);
2027
2028 // change nIndex so we can use it below to add the extensions
2029 m_aExtensions.Add(wxEmptyString);
2030 nIndex = m_aExtensions.size() - 1;
2031
2032 m_aDescriptions.Add(strDesc);
2033 }
2034 else // yes, we already have it
2035 {
2036 if ( replaceExisting )
2037 {
2038 // if new description change it
2039 if ( !strDesc.empty())
2040 m_aDescriptions[nIndex] = strDesc;
2041
2042 // if new icon change it
2043 if ( !strIcon.empty())
2044 m_aIcons[nIndex] = strIcon;
2045
2046 if ( entry )
2047 {
2048 delete m_aEntries[nIndex];
2049 m_aEntries[nIndex] = entry;
2050 }
2051 }
2052 else // add data we don't already have ...
2053 {
2054 // if new description add only if none
2055 if ( m_aDescriptions[nIndex].empty() )
2056 m_aDescriptions[nIndex] = strDesc;
2057
2058 // if new icon and no existing icon
2059 if ( m_aIcons[nIndex].empty() )
2060 m_aIcons[nIndex] = strIcon;
2061
2062 // add any new entries...
2063 if ( entry )
2064 {
2065 wxMimeTypeCommands *entryOld = m_aEntries[nIndex];
2066
2067 size_t count = entry->GetCount();
2068 for ( size_t i = 0; i < count; i++ )
2069 {
2070 const wxString& verb = entry->GetVerb(i);
2071 if ( !entryOld->HasVerb(verb) )
2072 {
2073 entryOld->AddOrReplaceVerb(verb, entry->GetCmd(i));
2074 }
2075 }
2076
2077 // as we don't store it anywhere, it won't be deleted later as
2078 // usual -- do it immediately instead
2079 delete entry;
2080 }
2081 }
2082 }
2083
2084 // always add the extensions to this mimetype
2085 wxString& exts = m_aExtensions[nIndex];
2086
2087 // add all extensions we don't have yet
2088 wxString ext;
2089 size_t count = strExtensions.GetCount();
2090 for ( size_t i = 0; i < count; i++ )
2091 {
2092 ext = strExtensions[i];
2093 ext += wxT(' ');
2094
2095 if ( exts.Find(ext) == wxNOT_FOUND )
2096 {
2097 exts += ext;
2098 }
2099 }
2100
2101 // check data integrity
2102 wxASSERT( m_aTypes.GetCount() == m_aEntries.GetCount() &&
2103 m_aTypes.GetCount() == m_aExtensions.GetCount() &&
2104 m_aTypes.GetCount() == m_aIcons.GetCount() &&
2105 m_aTypes.GetCount() == m_aDescriptions.GetCount() );
2106
2107 return nIndex;
2108}
2109
2110wxFileType * wxMimeTypesManagerImpl::GetFileTypeFromExtension(const wxString& ext)
2111{
2112 if (ext.empty() )
2113 return NULL;
2114
2115 InitIfNeeded();
2116
2117 size_t count = m_aExtensions.GetCount();
2118 for ( size_t n = 0; n < count; n++ )
2119 {
2120 wxStringTokenizer tk(m_aExtensions[n], wxT(' '));
2121
2122 while ( tk.HasMoreTokens() )
2123 {
2124 // consider extensions as not being case-sensitive
2125 if ( tk.GetNextToken().IsSameAs(ext, false /* no case */) )
2126 {
2127 // found
2128 wxFileType *fileType = new wxFileType;
2129 fileType->m_impl->Init(this, n);
2130
2131 return fileType;
2132 }
2133 }
2134 }
2135
2136 return NULL;
2137}
2138
2139wxFileType * wxMimeTypesManagerImpl::GetFileTypeFromMimeType(const wxString& mimeType)
2140{
2141 InitIfNeeded();
2142
2143 wxFileType * fileType = NULL;
2144 // mime types are not case-sensitive
2145 wxString mimetype(mimeType);
2146 mimetype.MakeLower();
2147
2148 // first look for an exact match
2149 int index = m_aTypes.Index(mimetype);
2150 if ( index != wxNOT_FOUND )
2151 {
2152 fileType = new wxFileType;
2153 fileType->m_impl->Init(this, index);
2154 }
2155
2156 // then try to find "text/*" as match for "text/plain" (for example)
2157 // NB: if mimeType doesn't contain '/' at all, BeforeFirst() will return
2158 // the whole string - ok.
2159
2160 index = wxNOT_FOUND;
2161 wxString strCategory = mimetype.BeforeFirst(wxT('/'));
2162
2163 size_t nCount = m_aTypes.GetCount();
2164 for ( size_t n = 0; n < nCount; n++ )
2165 {
2166 if ( (m_aTypes[n].BeforeFirst(wxT('/')) == strCategory ) &&
2167 m_aTypes[n].AfterFirst(wxT('/')) == wxT("*") )
2168 {
2169 index = n;
2170 break;
2171 }
2172 }
2173
2174 if ( index != wxNOT_FOUND )
2175 {
2176 // don't throw away fileType that was already found
2177 if (!fileType)
2178 fileType = new wxFileType;
2179 fileType->m_impl->Init(this, index);
2180 }
2181
2182 return fileType;
2183}
2184
2185wxString wxMimeTypesManagerImpl::GetCommand(const wxString & verb, size_t nIndex) const
2186{
2187 wxString command, testcmd, sV, sTmp;
2188 sV = verb + wxT("=");
2189
2190 // list of verb = command pairs for this mimetype
2191 wxMimeTypeCommands * sPairs = m_aEntries [nIndex];
2192
2193 size_t i;
2194 size_t nCount = sPairs->GetCount();
2195 for ( i = 0; i < nCount; i++ )
2196 {
2197 sTmp = sPairs->GetVerbCmd (i);
2198 if ( sTmp.Contains(sV) )
2199 command = sTmp.AfterFirst(wxT('='));
2200 }
2201
2202 return command;
2203}
2204
2205void wxMimeTypesManagerImpl::AddFallback(const wxFileTypeInfo& filetype)
2206{
2207 InitIfNeeded();
2208
2209 wxString extensions;
2210 const wxArrayString& exts = filetype.GetExtensions();
2211 size_t nExts = exts.GetCount();
2212 for ( size_t nExt = 0; nExt < nExts; nExt++ )
2213 {
2214 if ( nExt > 0 )
2215 extensions += wxT(' ');
2216
2217 extensions += exts[nExt];
2218 }
2219
2220 AddMimeTypeInfo(filetype.GetMimeType(),
2221 extensions,
2222 filetype.GetDescription());
2223
2224 AddMailcapInfo(filetype.GetMimeType(),
2225 filetype.GetOpenCommand(),
2226 filetype.GetPrintCommand(),
2227 wxT(""),
2228 filetype.GetDescription());
2229}
2230
2231void wxMimeTypesManagerImpl::AddMimeTypeInfo(const wxString& strMimeType,
2232 const wxString& strExtensions,
2233 const wxString& strDesc)
2234{
2235 // reading mailcap may find image/* , while
2236 // reading mime.types finds image/gif and no match is made
2237 // this means all the get functions don't work fix this
2238 wxString strIcon;
2239 wxString sTmp = strExtensions;
2240
2241 wxArrayString sExts;
2242 sTmp.Trim().Trim(false);
2243
2244 while (!sTmp.empty())
2245 {
2246 sExts.Add(sTmp.AfterLast(wxT(' ')));
2247 sTmp = sTmp.BeforeLast(wxT(' '));
2248 }
2249
2250 AddToMimeData(strMimeType, strIcon, NULL, sExts, strDesc, true);
2251}
2252
2253void wxMimeTypesManagerImpl::AddMailcapInfo(const wxString& strType,
2254 const wxString& strOpenCmd,
2255 const wxString& strPrintCmd,
2256 const wxString& strTest,
2257 const wxString& strDesc)
2258{
2259 InitIfNeeded();
2260
2261 wxMimeTypeCommands *entry = new wxMimeTypeCommands;
2262 entry->Add(wxT("open=") + strOpenCmd);
2263 entry->Add(wxT("print=") + strPrintCmd);
2264 entry->Add(wxT("test=") + strTest);
2265
2266 wxString strIcon;
2267 wxArrayString strExtensions;
2268
2269 AddToMimeData(strType, strIcon, entry, strExtensions, strDesc, true);
2270}
2271
2272bool wxMimeTypesManagerImpl::ReadMimeTypes(const wxString& strFileName)
2273{
2274 wxLogTrace(TRACE_MIME, wxT("--- Parsing mime.types file '%s' ---"),
2275 strFileName.c_str());
2276
2277 wxMimeTextFile file(strFileName);
2278 if ( !file.Open() )
2279 return false;
2280
2281 // the information we extract
2282 wxString strMimeType, strDesc, strExtensions;
2283
2284 size_t nLineCount = file.GetLineCount();
2285 const wxChar *pc = NULL;
2286 for ( size_t nLine = 0; nLine < nLineCount; nLine++ )
2287 {
2288 if ( pc == NULL )
2289 {
2290 // now we're at the start of the line
2291 pc = file[nLine].c_str();
2292 }
2293 else
2294 {
2295 // we didn't finish with the previous line yet
2296 nLine--;
2297 }
2298
2299 // skip whitespace
2300 while ( wxIsspace(*pc) )
2301 pc++;
2302
2303 // comment or blank line?
2304 if ( *pc == wxT('#') || !*pc )
2305 {
2306 // skip the whole line
2307 pc = NULL;
2308 continue;
2309 }
2310
2311 // detect file format
2312 const wxChar *pEqualSign = wxStrchr(pc, wxT('='));
2313 if ( pEqualSign == NULL )
2314 {
2315 // brief format
2316 // ------------
2317
2318 // first field is mime type
2319 for ( strMimeType.Empty(); !wxIsspace(*pc) && *pc != wxT('\0'); pc++ )
2320 {
2321 strMimeType += *pc;
2322 }
2323
2324 // skip whitespace
2325 while ( wxIsspace(*pc) )
2326 pc++;
2327
2328 // take all the rest of the string
2329 strExtensions = pc;
2330
2331 // no description...
2332 strDesc.Empty();
2333 }
2334 else
2335 {
2336 // expanded format
2337 // ---------------
2338
2339 // the string on the left of '=' is the field name
2340 wxString strLHS(pc, pEqualSign - pc);
2341
2342 // eat whitespace
2343 for ( pc = pEqualSign + 1; wxIsspace(*pc); pc++ )
2344 ;
2345
2346 const wxChar *pEnd;
2347 if ( *pc == wxT('"') )
2348 {
2349 // the string is quoted and ends at the matching quote
2350 pEnd = wxStrchr(++pc, wxT('"'));
2351 if ( pEnd == NULL )
2352 {
2353 wxLogWarning(wxT("Mime.types file %s, line %lu: unterminated quoted string."),
2354 strFileName.c_str(), nLine + 1L);
2355 }
2356 }
2357 else
2358 {
2359 // unquoted string ends at the first space or at the end of
2360 // line
2361 for ( pEnd = pc; *pEnd && !wxIsspace(*pEnd); pEnd++ )
2362 ;
2363 }
2364
2365 // now we have the RHS (field value)
2366 wxString strRHS(pc, pEnd - pc);
2367
2368 // check what follows this entry
2369 if ( *pEnd == wxT('"') )
2370 {
2371 // skip this quote
2372 pEnd++;
2373 }
2374
2375 for ( pc = pEnd; wxIsspace(*pc); pc++ )
2376 ;
2377
2378 // if there is something left, it may be either a '\\' to continue
2379 // the line or the next field of the same entry
2380 bool entryEnded = *pc == wxT('\0');
2381 bool nextFieldOnSameLine = false;
2382 if ( !entryEnded )
2383 {
2384 nextFieldOnSameLine = ((*pc != wxT('\\')) || (pc[1] != wxT('\0')));
2385 }
2386
2387 // now see what we got
2388 if ( strLHS == wxT("type") )
2389 {
2390 strMimeType = strRHS;
2391 }
2392 else if ( strLHS.StartsWith(wxT("desc")) )
2393 {
2394 strDesc = strRHS;
2395 }
2396 else if ( strLHS == wxT("exts") )
2397 {
2398 strExtensions = strRHS;
2399 }
2400 else if ( strLHS == wxT("icon") )
2401 {
2402 // this one is simply ignored: it usually refers to Netscape
2403 // built in icons which are useless for us anyhow
2404 }
2405 else if ( !strLHS.StartsWith(wxT("x-")) )
2406 {
2407 // we suppose that all fields starting with "X-" are
2408 // unregistered extensions according to the standard practice,
2409 // but it may be worth telling the user about other junk in
2410 // his mime.types file
2411 wxLogWarning(wxT("Unknown field in file %s, line %lu: '%s'."),
2412 strFileName.c_str(), nLine + 1L, strLHS.c_str());
2413 }
2414
2415 if ( !entryEnded )
2416 {
2417 if ( !nextFieldOnSameLine )
2418 pc = NULL;
2419 //else: don't reset it
2420
2421 // as we don't reset strMimeType, the next field in this entry
2422 // will be interpreted correctly.
2423
2424 continue;
2425 }
2426 }
2427
2428 // depending on the format (Mosaic or Netscape) either space or comma
2429 // is used to separate the extensions
2430 strExtensions.Replace(wxT(","), wxT(" "));
2431
2432 // also deal with the leading dot
2433 if ( !strExtensions.empty() && strExtensions[0u] == wxT('.') )
2434 {
2435 strExtensions.erase(0, 1);
2436 }
2437
2438 wxLogTrace(TRACE_MIME, wxT("mime.types: '%s' => '%s' (%s)"),
2439 strExtensions.c_str(),
2440 strMimeType.c_str(),
2441 strDesc.c_str());
2442
2443 AddMimeTypeInfo(strMimeType, strExtensions, strDesc);
2444
2445 // finished with this line
2446 pc = NULL;
2447 }
2448
2449 return true;
2450}
2451
2452// ----------------------------------------------------------------------------
2453// UNIX mailcap files parsing
2454// ----------------------------------------------------------------------------
2455
2456// the data for a single MIME type
2457struct MailcapLineData
2458{
2459 // field values
2460 wxString type,
2461 cmdOpen,
2462 test,
2463 icon,
2464 desc;
2465
2466 wxArrayString verbs,
2467 commands;
2468
2469 // flags
2470 bool testfailed,
2471 needsterminal,
2472 copiousoutput;
2473
2474 MailcapLineData() { testfailed = needsterminal = copiousoutput = false; }
2475};
2476
2477// process a non-standard (i.e. not the first or second one) mailcap field
2478bool
2479wxMimeTypesManagerImpl::ProcessOtherMailcapField(MailcapLineData& data,
2480 const wxString& curField)
2481{
2482 if ( curField.empty() )
2483 {
2484 // we don't care
2485 return true;
2486 }
2487
2488 // is this something of the form foo=bar?
2489 if ( curField.find('=') != wxString::npos )
2490 {
2491 // split "LHS = RHS" in 2
2492 wxString lhs = curField.BeforeFirst(wxT('=')),
2493 rhs = curField.AfterFirst(wxT('='));
2494
2495 lhs.Trim(true); // from right
2496 rhs.Trim(false); // from left
2497
2498 // it might be quoted
2499 if ( !rhs.empty() && rhs[0u] == wxT('"') && rhs.Last() == wxT('"') )
2500 {
2501 rhs = rhs.Mid(1, rhs.length() - 2);
2502 }
2503
2504 // is it a command verb or something else?
2505 if ( lhs == wxT("test") )
2506 {
2507 if ( wxSystem(rhs) == 0 )
2508 {
2509 // ok, test passed
2510 wxLogTrace(TRACE_MIME_TEST,
2511 wxT("Test '%s' for mime type '%s' succeeded."),
2512 rhs.c_str(), data.type.c_str());
2513 }
2514 else
2515 {
2516 wxLogTrace(TRACE_MIME_TEST,
2517 wxT("Test '%s' for mime type '%s' failed, skipping."),
2518 rhs.c_str(), data.type.c_str());
2519
2520 data.testfailed = true;
2521 }
2522 }
2523 else if ( lhs == wxT("desc") )
2524 {
2525 data.desc = rhs;
2526 }
2527 else if ( lhs == wxT("x11-bitmap") )
2528 {
2529 data.icon = rhs;
2530 }
2531 else if ( lhs == wxT("notes") )
2532 {
2533 // ignore
2534 }
2535 else // not a (recognized) special case, must be a verb (e.g. "print")
2536 {
2537 data.verbs.Add(lhs);
2538 data.commands.Add(rhs);
2539 }
2540 }
2541 else // '=' not found
2542 {
2543 // so it must be a simple flag
2544 if ( curField == wxT("needsterminal") )
2545 {
2546 data.needsterminal = true;
2547 }
2548 else if ( curField == wxT("copiousoutput"))
2549 {
2550 // copiousoutput impies that the viewer is a console program
2551 data.needsterminal =
2552 data.copiousoutput = true;
2553 }
2554 else if ( !IsKnownUnimportantField(curField) )
2555 {
2556 return false;
2557 }
2558 }
2559
2560 return true;
2561}
2562
2563bool wxMimeTypesManagerImpl::ReadMailcap(const wxString& strFileName,
2564 bool fallback)
2565{
2566 wxLogTrace(TRACE_MIME, wxT("--- Parsing mailcap file '%s' ---"),
2567 strFileName.c_str());
2568
2569 wxMimeTextFile file(strFileName);
2570 if ( !file.Open() )
2571 return false;
2572
2573 // indices of MIME types (in m_aTypes) we already found in this file
2574 //
2575 // (see the comments near the end of function for the reason we need this)
2576 wxArrayInt aIndicesSeenHere;
2577
2578 // accumulator for the current field
2579 wxString curField;
2580 curField.reserve(1024);
2581
2582 const wxChar *pPagerEnv = wxGetenv(wxT("PAGER"));
2583
2584 const wxArrayString empty_extensions_list;
2585
2586 size_t nLineCount = file.GetLineCount();
2587 for ( size_t nLine = 0; nLine < nLineCount; nLine++ )
2588 {
2589 // now we're at the start of the line
2590 const wxChar *pc = file[nLine].c_str();
2591
2592 // skip whitespace
2593 while ( wxIsspace(*pc) )
2594 pc++;
2595
2596 // comment or empty string?
2597 if ( *pc == wxT('#') || *pc == wxT('\0') )
2598 continue;
2599
2600 // no, do parse
2601 // ------------
2602
2603 // what field are we currently in? The first 2 are fixed and there may
2604 // be an arbitrary number of other fields parsed by
2605 // ProcessOtherMailcapField()
2606 //
2607 // the first field is the MIME type
2608 enum
2609 {
2610 Field_Type,
2611 Field_OpenCmd,
2612 Field_Other
2613 } currentToken = Field_Type;
2614
2615 // the flags and field values on the current line
2616 MailcapLineData data;
2617
2618 bool cont = true;
2619 while ( cont )
2620 {
2621 switch ( *pc )
2622 {
2623 case wxT('\\'):
2624 // interpret the next character literally (notice that
2625 // backslash can be used for line continuation)
2626 if ( *++pc == wxT('\0') )
2627 {
2628 // fetch the next line if there is one
2629 if ( nLine == nLineCount - 1 )
2630 {
2631 // something is wrong, bail out
2632 cont = false;
2633
2634 wxLogDebug(wxT("Mailcap file %s, line %lu: '\\' on the end of the last line ignored."),
2635 strFileName.c_str(),
2636 nLine + 1L);
2637 }
2638 else
2639 {
2640 // pass to the beginning of the next line
2641 pc = file[++nLine].c_str();
2642
2643 // skip pc++ at the end of the loop
2644 continue;
2645 }
2646 }
2647 else
2648 {
2649 // just a normal character
2650 curField += *pc;
2651 }
2652 break;
2653
2654 case wxT('\0'):
2655 cont = false; // end of line reached, exit the loop
2656
2657 // fall through to still process this field
2658
2659 case wxT(';'):
2660 // trim whitespaces from both sides
2661 curField.Trim(true).Trim(false);
2662
2663 switch ( currentToken )
2664 {
2665 case Field_Type:
2666 data.type = curField.Lower();
2667 if ( data.type.empty() )
2668 {
2669 // I don't think that this is a valid mailcap
2670 // entry, but try to interpret it somehow
2671 data.type = wxT('*');
2672 }
2673
2674 if ( data.type.Find(wxT('/')) == wxNOT_FOUND )
2675 {
2676 // we interpret "type" as "type/*"
2677 data.type += wxT("/*");
2678 }
2679
2680 currentToken = Field_OpenCmd;
2681 break;
2682
2683 case Field_OpenCmd:
2684 data.cmdOpen = curField;
2685
2686 currentToken = Field_Other;
2687 break;
2688
2689 case Field_Other:
2690 if ( !ProcessOtherMailcapField(data, curField) )
2691 {
2692 // don't flood the user with error messages if
2693 // we don't understand something in his
2694 // mailcap, but give them in debug mode because
2695 // this might be useful for the programmer
2696 wxLogDebug
2697 (
2698 wxT("Mailcap file %s, line %lu: unknown field '%s' for the MIME type '%s' ignored."),
2699 strFileName.c_str(),
2700 nLine + 1L,
2701 curField.c_str(),
2702 data.type.c_str()
2703 );
2704 }
2705 else if ( data.testfailed )
2706 {
2707 // skip this entry entirely
2708 cont = false;
2709 }
2710
2711 // it already has this value
2712 //currentToken = Field_Other;
2713 break;
2714
2715 default:
2716 wxFAIL_MSG(wxT("unknown field type in mailcap"));
2717 }
2718
2719 // next token starts immediately after ';'
2720 curField.Empty();
2721 break;
2722
2723 default:
2724 curField += *pc;
2725 }
2726
2727 // continue in the same line
2728 pc++;
2729 }
2730
2731 // we read the entire entry, check what have we got
2732 // ------------------------------------------------
2733
2734 // check that we really read something reasonable
2735 if ( currentToken < Field_Other )
2736 {
2737 wxLogWarning(wxT("Mailcap file %s, line %lu: incomplete entry ignored."),
2738 strFileName.c_str(), nLine + 1L);
2739
2740 continue;
2741 }
2742
2743 // if the test command failed, it's as if the entry were not there at all
2744 if ( data.testfailed )
2745 {
2746 continue;
2747 }
2748
2749 // support for flags:
2750 // 1. create an xterm for 'needsterminal'
2751 // 2. append "| $PAGER" for 'copiousoutput'
2752 //
2753 // Note that the RFC says that having both needsterminal and
2754 // copiousoutput is probably a mistake, so it seems that running
2755 // programs with copiousoutput inside an xterm as it is done now
2756 // is a bad idea (FIXME)
2757 if ( data.copiousoutput )
2758 {
2759 data.cmdOpen << wxT(" | ") << (pPagerEnv ? pPagerEnv : wxT("more"));
2760 }
2761
2762 if ( data.needsterminal )
2763 {
2764 data.cmdOpen.insert(0, wxT("xterm -e sh -c '"));
2765 data.cmdOpen.append(wxT("'"));
2766 }
2767
2768 if ( !data.cmdOpen.empty() )
2769 {
2770 data.verbs.Insert(wxT("open"), 0);
2771 data.commands.Insert(data.cmdOpen, 0);
2772 }
2773
2774 // we have to decide whether the new entry should replace any entries
2775 // for the same MIME type we had previously found or not
2776 bool overwrite;
2777
2778 // the fall back entries have the lowest priority, by definition
2779 if ( fallback )
2780 {
2781 overwrite = false;
2782 }
2783 else
2784 {
2785 // have we seen this one before?
2786 int nIndex = m_aTypes.Index(data.type);
2787
2788 // and if we have, was it in this file? if not, we should
2789 // overwrite the previously seen one
2790 overwrite = nIndex == wxNOT_FOUND ||
2791 aIndicesSeenHere.Index(nIndex) == wxNOT_FOUND;
2792 }
2793
2794 wxLogTrace(TRACE_MIME, wxT("mailcap %s: %s [%s]"),
2795 data.type.c_str(), data.cmdOpen.c_str(),
2796 overwrite ? wxT("replace") : wxT("add"));
2797
2798 int n = AddToMimeData
2799 (
2800 data.type,
2801 data.icon,
2802 new wxMimeTypeCommands(data.verbs, data.commands),
2803 empty_extensions_list,
2804 data.desc,
2805 overwrite
2806 );
2807
2808 if ( overwrite )
2809 {
2810 aIndicesSeenHere.Add(n);
2811 }
2812 }
2813
2814 return true;
2815}
2816
2817size_t wxMimeTypesManagerImpl::EnumAllFileTypes(wxArrayString& mimetypes)
2818{
2819 InitIfNeeded();
2820
2821 mimetypes.Empty();
2822
2823 size_t count = m_aTypes.GetCount();
2824 for ( size_t n = 0; n < count; n++ )
2825 {
2826 // don't return template types from here (i.e. anything containg '*')
2827 const wxString &type = m_aTypes[n];
2828 if ( type.Find(wxT('*')) == wxNOT_FOUND )
2829 {
2830 mimetypes.Add(type);
2831 }
2832 }
2833
2834 return mimetypes.GetCount();
2835}
2836
2837// ----------------------------------------------------------------------------
2838// writing to MIME type files
2839// ----------------------------------------------------------------------------
2840
2841bool wxMimeTypesManagerImpl::Unassociate(wxFileType *ft)
2842{
2843 InitIfNeeded();
2844
2845 wxArrayString sMimeTypes;
2846 ft->GetMimeTypes(sMimeTypes);
2847
2848 size_t i;
2849 size_t nCount = sMimeTypes.GetCount();
2850 for (i = 0; i < nCount; i ++)
2851 {
2852 const wxString &sMime = sMimeTypes.Item(i);
2853 int nIndex = m_aTypes.Index(sMime);
2854 if ( nIndex == wxNOT_FOUND)
2855 {
2856 // error if we get here ??
2857 return false;
2858 }
2859 else
2860 {
2861 WriteMimeInfo(nIndex, true);
2862 m_aTypes.RemoveAt(nIndex);
2863 m_aEntries.RemoveAt(nIndex);
2864 m_aExtensions.RemoveAt(nIndex);
2865 m_aDescriptions.RemoveAt(nIndex);
2866 m_aIcons.RemoveAt(nIndex);
2867 }
2868 }
2869 // check data integrity
2870 wxASSERT( m_aTypes.GetCount() == m_aEntries.GetCount() &&
2871 m_aTypes.GetCount() == m_aExtensions.GetCount() &&
2872 m_aTypes.GetCount() == m_aIcons.GetCount() &&
2873 m_aTypes.GetCount() == m_aDescriptions.GetCount() );
2874
2875 return true;
2876}
2877
2878// ----------------------------------------------------------------------------
2879// private functions
2880// ----------------------------------------------------------------------------
2881
2882static bool IsKnownUnimportantField(const wxString& fieldAll)
2883{
2884 static const wxChar * const knownFields[] =
2885 {
2886 wxT("x-mozilla-flags"),
2887 wxT("nametemplate"),
2888 wxT("textualnewlines"),
2889 };
2890
2891 wxString field = fieldAll.BeforeFirst(wxT('='));
2892 for ( size_t n = 0; n < WXSIZEOF(knownFields); n++ )
2893 {
2894 if ( field.CmpNoCase(knownFields[n]) == 0 )
2895 return true;
2896 }
2897
2898 return false;
2899}
2900
2901#endif
2902 // wxUSE_MIMETYPE && wxUSE_FILE && wxUSE_TEXTFILE