]>
git.saurik.com Git - wxWidgets.git/blob - src/common/filesys.cpp
d154b75f7334a2b9c37e95b150194398f052c8cd
1 /////////////////////////////////////////////////////////////////////////////
3 // Purpose: wxFileSystem class - interface for opening files
4 // Author: Vaclav Slavik
5 // Copyright: (c) 1999 Vaclav Slavik
7 // Licence: wxWindows Licence
8 /////////////////////////////////////////////////////////////////////////////
11 #pragma implementation
14 #include "wx/wxprec.h"
23 #include "wx/wfstream.h"
24 #include "wx/module.h"
25 #include "wx/filesys.h"
26 #include "wx/mimetype.h"
31 //--------------------------------------------------------------------------------
32 // wxFileSystemHandler
33 //--------------------------------------------------------------------------------
35 IMPLEMENT_ABSTRACT_CLASS(wxFileSystemHandler
, wxObject
)
38 static wxFileTypeInfo
*gs_FSMimeFallbacks
= NULL
;
40 wxString
wxFileSystemHandler::GetMimeTypeFromExt(const wxString
& location
)
42 wxString ext
= wxEmptyString
, mime
= wxEmptyString
;
43 wxString loc
= GetRightLocation(location
);
45 int l
= loc
.Length(), l2
;
49 for (int i
= l
-1; i
>= 0; i
--) {
50 c
= loc
[(unsigned int) i
];
51 if (c
== wxT('#')) l2
= i
+ 1;
52 if (c
== wxT('.')) {ext
= loc
.Right(l2
-i
-1); break;}
53 if ((c
== wxT('/')) || (c
== wxT('\\')) || (c
== wxT(':'))) {return wxEmptyString
;}
56 static bool s_MinimalMimeEnsured
= FALSE
;
57 if (!s_MinimalMimeEnsured
) {
58 wxTheMimeTypesManager
-> AddFallbacks(gs_FSMimeFallbacks
);
59 s_MinimalMimeEnsured
= TRUE
;
62 ft
= wxTheMimeTypesManager
-> GetFileTypeFromExtension(ext
);
63 if ( !ft
|| !ft
-> GetMimeType(&mime
) ) {
74 wxString
wxFileSystemHandler::GetProtocol(const wxString
& location
) const
76 wxString s
= wxEmptyString
;
77 int i
, l
= location
.Length();
81 for (i
= l
-1; (i
>= 0) && ((location
[i
] != wxT('#')) || (!fnd
)); i
--) {
82 if ((location
[i
] == wxT(':')) && (i
!= 1 /*win: C:\path*/)) fnd
= TRUE
;
84 if (!fnd
) return wxT("file");
85 for (++i
; (i
< l
) && (location
[i
] != wxT(':')); i
++) s
<< location
[i
];
90 wxString
wxFileSystemHandler::GetLeftLocation(const wxString
& location
) const
96 for (i
= location
.Length()-1; i
>= 0; i
--) {
97 if ((location
[i
] == wxT(':')) && (i
!= 1 /*win: C:\path*/)) fnd
= TRUE
;
98 else if (fnd
&& (location
[i
] == wxT('#'))) return location
.Left(i
);
100 return wxEmptyString
;
103 wxString
wxFileSystemHandler::GetRightLocation(const wxString
& location
) const
105 int i
, l
= location
.Length();
107 for (i
= l
-1; (i
>= 0) && ((location
[i
] != wxT(':')) || (i
== 1) || (location
[i
-2] == wxT(':'))); i
--) {if (location
[i
] == wxT('#')) l2
= i
+ 1;}
108 if (i
== 0) return wxEmptyString
;
109 else return location
.Mid(i
+ 1, l2
- i
- 2);
112 wxString
wxFileSystemHandler::GetAnchor(const wxString
& location
) const
115 int l
= location
.Length();
117 for (int i
= l
-1; i
>= 0; i
--) {
119 if (c
== wxT('#')) return location
.Right(l
-i
-1);
120 else if ((c
== wxT('.')) || (c
== wxT('/')) || (c
== wxT('\\')) || (c
== wxT(':'))) return wxEmptyString
;
122 return wxEmptyString
;
126 wxString
wxFileSystemHandler::FindFirst(const wxString
& WXUNUSED(spec
),
129 return wxEmptyString
;
132 wxString
wxFileSystemHandler::FindNext()
134 return wxEmptyString
;
137 //--------------------------------------------------------------------------------
139 //--------------------------------------------------------------------------------
141 class wxLocalFSHandler
: public wxFileSystemHandler
144 virtual bool CanOpen(const wxString
& location
);
145 virtual wxFSFile
* OpenFile(wxFileSystem
& fs
, const wxString
& location
);
146 virtual wxString
FindFirst(const wxString
& spec
, int flags
= 0);
147 virtual wxString
FindNext();
151 bool wxLocalFSHandler::CanOpen(const wxString
& location
)
153 return GetProtocol(location
) == wxT("file");
156 wxFSFile
* wxLocalFSHandler::OpenFile(wxFileSystem
& WXUNUSED(fs
), const wxString
& location
)
158 wxString right
= GetRightLocation(location
);
160 if ( right
[0] != '.' && right
[0] != '/' && right
.Find( '/' ) != wxNOT_FOUND
) {
161 right
= "./" + right
;
163 right
= wxUnix2MacFilename( right
) ;
165 if (!wxFileExists(right
))
166 return (wxFSFile
*) NULL
;
168 return new wxFSFile(new wxFileInputStream(right
),
170 GetMimeTypeFromExt(location
),
172 wxDateTime(wxFileModificationTime(right
)));
176 wxString
wxLocalFSHandler::FindFirst(const wxString
& spec
, int flags
)
178 wxString right
= GetRightLocation(spec
);
180 if ( right
[0] != '.' && right
[0] != '/' && right
.Find( '/' ) != wxNOT_FOUND
) {
181 right
= "./" + right
;
183 right
= wxUnix2MacFilename( right
) ;
185 return wxFindFirstFile(right
, flags
);
188 wxString
wxLocalFSHandler::FindNext()
190 return wxFindNextFile();
195 //-----------------------------------------------------------------------------
197 //-----------------------------------------------------------------------------
199 IMPLEMENT_DYNAMIC_CLASS(wxFileSystem
, wxObject
)
202 wxList
wxFileSystem::m_Handlers
;
205 static wxString
MakeCorrectPath(const wxString
& path
)
212 for (i
= 0; i
< cnt
; i
++)
213 if (p
.GetChar(i
) == wxT('\\')) p
.GetWritableChar(i
) = wxT('/'); // Want to be windows-safe
215 if (p
.Left(2) == wxT("./")) { p
= p
.Mid(2); cnt
-= 2; }
217 if (cnt
< 3) return p
;
219 r
<< p
.GetChar(0) << p
.GetChar(1);
221 // skip trailing ../.., if any
222 for (i
= 2; i
< cnt
&& (p
.GetChar(i
) == wxT('/') || p
.GetChar(i
) == wxT('.')); i
++) r
<< p
.GetChar(i
);
224 // remove back references: translate dir1/../dir2 to dir2
228 if (p
.GetChar(i
) == wxT('/') && p
.GetChar(i
-1) == wxT('.') && p
.GetChar(i
-2) == wxT('.'))
230 for (j
= r
.Length() - 2; j
>= 0 && r
.GetChar(j
) != wxT('/') && r
.GetChar(j
) != wxT(':'); j
--) {}
231 if (j
>= 0 && r
.GetChar(j
) != wxT(':'))
233 for (j
= j
- 1; j
>= 0 && r
.GetChar(j
) != wxT('/') && r
.GetChar(j
) != wxT(':'); j
--) {}
239 for (; i
< cnt
; i
++) r
<< p
.GetChar(i
);
245 void wxFileSystem::ChangePathTo(const wxString
& location
, bool is_dir
)
249 m_Path
= MakeCorrectPath(location
);
253 if (m_Path
.Length() > 0 && m_Path
.Last() != wxT('/') && m_Path
.Last() != wxT(':'))
259 for (i
= m_Path
.Length()-1; i
>= 0; i
--)
261 if (m_Path
[(unsigned int) i
] == wxT('/'))
263 if ((i
> 1) && (m_Path
[(unsigned int) (i
-1)] == wxT('/')) && (m_Path
[(unsigned int) (i
-2)] == wxT(':')))
274 else if (m_Path
[(unsigned int) i
] == wxT(':')) {
281 for (i
= 0; i
< (int) m_Path
.Length(); i
++)
283 if (m_Path
[(unsigned int) i
] == wxT(':'))
289 if (i
== (int) m_Path
.Length())
290 m_Path
= wxEmptyString
;
294 m_Path
.Remove(pathpos
+1);
301 wxFSFile
* wxFileSystem::OpenFile(const wxString
& location
)
303 wxString loc
= MakeCorrectPath(location
);
311 for (i
= 0; i
< ln
; i
++)
315 case wxT('/') : case wxT(':') : case wxT('#') :
319 if (meta
!= 0) break;
321 m_LastName
= wxEmptyString
;
323 // try relative paths first :
324 if (meta
!= wxT(':'))
326 node
= m_Handlers
.GetFirst();
329 wxFileSystemHandler
*h
= (wxFileSystemHandler
*) node
-> GetData();
330 if (h
->CanOpen(m_Path
+ loc
))
332 s
= h
->OpenFile(*this, m_Path
+ loc
);
333 if (s
) { m_LastName
= m_Path
+ loc
; break; }
335 node
= node
->GetNext();
339 // if failed, try absolute paths :
342 node
= m_Handlers
.GetFirst();
345 wxFileSystemHandler
*h
= (wxFileSystemHandler
*) node
->GetData();
348 s
= h
->OpenFile(*this, loc
);
349 if (s
) { m_LastName
= loc
; break; }
351 node
= node
->GetNext();
359 wxString
wxFileSystem::FindFirst(const wxString
& spec
, int flags
)
362 wxString
spec2(spec
);
364 m_FindFileHandler
= NULL
;
366 for (int i
= spec2
.Length()-1; i
>= 0; i
--)
367 if (spec2
[(unsigned int) i
] == wxT('\\')) spec2
.GetWritableChar(i
) = wxT('/'); // Want to be windows-safe
369 node
= m_Handlers
.GetFirst();
372 m_FindFileHandler
= (wxFileSystemHandler
*) node
-> GetData();
373 if (m_FindFileHandler
-> CanOpen(m_Path
+ spec2
))
374 return m_FindFileHandler
-> FindFirst(m_Path
+ spec2
, flags
);
375 node
= node
->GetNext();
378 node
= m_Handlers
.GetFirst();
381 m_FindFileHandler
= (wxFileSystemHandler
*) node
-> GetData();
382 if (m_FindFileHandler
-> CanOpen(spec2
))
383 return m_FindFileHandler
-> FindFirst(spec2
, flags
);
384 node
= node
->GetNext();
387 return wxEmptyString
;
392 wxString
wxFileSystem::FindNext()
394 if (m_FindFileHandler
== NULL
) return wxEmptyString
;
395 else return m_FindFileHandler
-> FindNext();
400 void wxFileSystem::AddHandler(wxFileSystemHandler
*handler
)
402 m_Handlers
.Append(handler
);
406 void wxFileSystem::CleanUpHandlers()
408 m_Handlers
.DeleteContents(TRUE
);
417 class wxFileSystemModule
: public wxModule
419 DECLARE_DYNAMIC_CLASS(wxFileSystemModule
)
422 virtual bool OnInit()
424 wxFileSystem::AddHandler(new wxLocalFSHandler
);
426 gs_FSMimeFallbacks
= new wxFileTypeInfo
[6];
427 gs_FSMimeFallbacks
[0] =
428 wxFileTypeInfo("image/jpeg",
431 "JPEG image (from fallback)",
432 "jpg", "jpeg", NULL
);
433 gs_FSMimeFallbacks
[1] =
434 wxFileTypeInfo("image/gif",
437 "GIF image (from fallback)",
439 gs_FSMimeFallbacks
[2] =
440 wxFileTypeInfo("image/png",
443 "PNG image (from fallback)",
445 gs_FSMimeFallbacks
[3] =
446 wxFileTypeInfo("image/bmp",
449 "windows bitmap image (from fallback)",
451 gs_FSMimeFallbacks
[4] =
452 wxFileTypeInfo("text/html",
455 "HTML document (from fallback)",
456 "htm", "html", NULL
);
457 gs_FSMimeFallbacks
[5] =
458 // must terminate the table with this!
463 virtual void OnExit()
465 delete [] gs_FSMimeFallbacks
;
466 wxFileSystem::CleanUpHandlers();
470 IMPLEMENT_DYNAMIC_CLASS(wxFileSystemModule
, wxModule
)