]>
git.saurik.com Git - wxWidgets.git/blob - src/common/filesys.cpp
   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" 
  27 #include "wx/filename.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 //-------------------------------------------------------------------------------- 
 142 wxString 
wxLocalFSHandler::ms_root
; 
 144 bool wxLocalFSHandler::CanOpen(const wxString
& location
) 
 146     return GetProtocol(location
) == wxT("file"); 
 149 wxFSFile
* wxLocalFSHandler::OpenFile(wxFileSystem
& WXUNUSED(fs
), const wxString
& location
) 
 151     // location has Unix path separators 
 152     wxString right 
= ms_root 
+ GetRightLocation(location
); 
 153     wxFileName 
fn(right
, wxPATH_UNIX
); 
 155     if (!wxFileExists(fn
.GetFullPath())) 
 156         return (wxFSFile
*) NULL
; 
 158     return new wxFSFile(new wxFileInputStream(fn
.GetFullPath()), 
 160                         GetMimeTypeFromExt(location
), 
 162                         wxDateTime(wxFileModificationTime(fn
.GetFullPath()))); 
 166 wxString 
wxLocalFSHandler::FindFirst(const wxString
& spec
, int flags
) 
 168     wxString right 
= ms_root 
+ GetRightLocation(spec
); 
 169     return wxFindFirstFile(right
, flags
); 
 172 wxString 
wxLocalFSHandler::FindNext() 
 174     return wxFindNextFile(); 
 179 //----------------------------------------------------------------------------- 
 181 //----------------------------------------------------------------------------- 
 183 IMPLEMENT_DYNAMIC_CLASS(wxFileSystem
, wxObject
) 
 186 wxList 
wxFileSystem::m_Handlers
; 
 189 static wxString 
MakeCorrectPath(const wxString
& path
) 
 196     for (i 
= 0; i 
< cnt
; i
++) 
 197       if (p
.GetChar(i
) == wxT('\\')) p
.GetWritableChar(i
) = wxT('/'); // Want to be windows-safe 
 199     if (p
.Left(2) == wxT("./")) { p 
= p
.Mid(2); cnt 
-= 2; } 
 201     if (cnt 
< 3) return p
; 
 203     r 
<< p
.GetChar(0) << p
.GetChar(1); 
 205     // skip trailing ../.., if any 
 206     for (i 
= 2; i 
< cnt 
&& (p
.GetChar(i
) == wxT('/') || p
.GetChar(i
) == wxT('.')); i
++) r 
<< p
.GetChar(i
); 
 208     // remove back references: translate dir1/../dir2 to dir2 
 212         if (p
.GetChar(i
) == wxT('/') && p
.GetChar(i
-1) == wxT('.') && p
.GetChar(i
-2) == wxT('.')) 
 214             for (j 
= r
.Length() - 2; j 
>= 0 && r
.GetChar(j
) != wxT('/') && r
.GetChar(j
) != wxT(':'); j
--) {} 
 215             if (j 
>= 0 && r
.GetChar(j
) != wxT(':')) 
 217                 for (j 
= j 
- 1; j 
>= 0 && r
.GetChar(j
) != wxT('/') && r
.GetChar(j
) != wxT(':'); j
--) {} 
 223     for (; i 
< cnt
; i
++) r 
<< p
.GetChar(i
); 
 229 void wxFileSystem::ChangePathTo(const wxString
& location
, bool is_dir
) 
 233     m_Path 
= MakeCorrectPath(location
); 
 237         if (m_Path
.Length() > 0 && m_Path
.Last() != wxT('/') && m_Path
.Last() != wxT(':')) 
 243         for (i 
= m_Path
.Length()-1; i 
>= 0; i
--) 
 245             if (m_Path
[(unsigned int) i
] == wxT('/')) 
 247                 if ((i 
> 1) && (m_Path
[(unsigned int) (i
-1)] == wxT('/')) && (m_Path
[(unsigned int) (i
-2)] == wxT(':'))) 
 258             else if (m_Path
[(unsigned int) i
] == wxT(':')) { 
 265             for (i 
= 0; i 
< (int) m_Path
.Length(); i
++) 
 267                 if (m_Path
[(unsigned int) i
] == wxT(':')) 
 273             if (i 
== (int) m_Path
.Length()) 
 274                 m_Path 
= wxEmptyString
; 
 278             m_Path
.Remove(pathpos
+1); 
 285 wxFSFile
* wxFileSystem::OpenFile(const wxString
& location
) 
 287     wxString loc 
= MakeCorrectPath(location
); 
 295     for (i 
= 0; i 
< ln
; i
++) 
 299             case wxT('/') : case wxT(':') : case wxT('#') : 
 303         if (meta 
!= 0) break; 
 305     m_LastName 
= wxEmptyString
; 
 307     // try relative paths first : 
 308     if (meta 
!= wxT(':')) 
 310         node 
= m_Handlers
.GetFirst(); 
 313             wxFileSystemHandler 
*h 
= (wxFileSystemHandler
*) node 
-> GetData(); 
 314             if (h
->CanOpen(m_Path 
+ loc
)) 
 316                 s 
= h
->OpenFile(*this, m_Path 
+ loc
); 
 317                 if (s
) { m_LastName 
= m_Path 
+ loc
; break; } 
 319             node 
= node
->GetNext(); 
 323     // if failed, try absolute paths : 
 326         node 
= m_Handlers
.GetFirst(); 
 329             wxFileSystemHandler 
*h 
= (wxFileSystemHandler
*) node
->GetData(); 
 332                 s 
= h
->OpenFile(*this, loc
); 
 333                 if (s
) { m_LastName 
= loc
; break; } 
 335             node 
= node
->GetNext(); 
 343 wxString 
wxFileSystem::FindFirst(const wxString
& spec
, int flags
) 
 346     wxString 
spec2(spec
); 
 348     m_FindFileHandler 
= NULL
; 
 350     for (int i 
= spec2
.Length()-1; i 
>= 0; i
--) 
 351         if (spec2
[(unsigned int) i
] == wxT('\\')) spec2
.GetWritableChar(i
) = wxT('/'); // Want to be windows-safe 
 353     node 
= m_Handlers
.GetFirst(); 
 356         m_FindFileHandler 
= (wxFileSystemHandler
*) node 
-> GetData(); 
 357         if (m_FindFileHandler 
-> CanOpen(m_Path 
+ spec2
)) 
 358             return m_FindFileHandler 
-> FindFirst(m_Path 
+ spec2
, flags
); 
 359         node 
= node
->GetNext(); 
 362     node 
= m_Handlers
.GetFirst(); 
 365         m_FindFileHandler 
= (wxFileSystemHandler
*) node 
-> GetData(); 
 366         if (m_FindFileHandler 
-> CanOpen(spec2
)) 
 367             return m_FindFileHandler 
-> FindFirst(spec2
, flags
); 
 368         node 
= node
->GetNext(); 
 371     return wxEmptyString
; 
 376 wxString 
wxFileSystem::FindNext() 
 378     if (m_FindFileHandler 
== NULL
) return wxEmptyString
; 
 379     else return m_FindFileHandler 
-> FindNext(); 
 384 void wxFileSystem::AddHandler(wxFileSystemHandler 
*handler
) 
 386     m_Handlers
.Append(handler
); 
 390 void wxFileSystem::CleanUpHandlers() 
 392     m_Handlers
.DeleteContents(TRUE
); 
 401 class wxFileSystemModule 
: public wxModule
 
 403     DECLARE_DYNAMIC_CLASS(wxFileSystemModule
) 
 406         virtual bool OnInit() 
 408             wxFileSystem::AddHandler(new wxLocalFSHandler
); 
 410             gs_FSMimeFallbacks 
= new wxFileTypeInfo
[6]; 
 411             gs_FSMimeFallbacks
[0] = 
 412             wxFileTypeInfo("image/jpeg", 
 415                            "JPEG image (from fallback)", 
 416                            "jpg", "jpeg", NULL
); 
 417             gs_FSMimeFallbacks
[1] = 
 418             wxFileTypeInfo("image/gif", 
 421                            "GIF image (from fallback)", 
 423             gs_FSMimeFallbacks
[2] = 
 424             wxFileTypeInfo("image/png", 
 427                            "PNG image (from fallback)", 
 429             gs_FSMimeFallbacks
[3] = 
 430             wxFileTypeInfo("image/bmp", 
 433                            "windows bitmap image (from fallback)", 
 435             gs_FSMimeFallbacks
[4] = 
 436             wxFileTypeInfo("text/html", 
 439                            "HTML document (from fallback)", 
 440                            "htm", "html", NULL
); 
 441             gs_FSMimeFallbacks
[5] = 
 442             // must terminate the table with this! 
 447         virtual void OnExit() 
 449             delete [] gs_FSMimeFallbacks
; 
 450             wxFileSystem::CleanUpHandlers(); 
 454 IMPLEMENT_DYNAMIC_CLASS(wxFileSystemModule
, wxModule
)