]> git.saurik.com Git - wxWidgets.git/blame - src/common/fs_zip.cpp
don't crash on weird line endings like \r\r\n
[wxWidgets.git] / src / common / fs_zip.cpp
CommitLineData
5526e819
VS
1/////////////////////////////////////////////////////////////////////////////
2// Name: fs_zip.cpp
3// Purpose: ZIP file system
4// Author: Vaclav Slavik
5// Copyright: (c) 1999 Vaclav Slavik
de0702d0 6// CVS-ID: $Id$
65571936 7// Licence: wxWindows licence
5526e819
VS
8/////////////////////////////////////////////////////////////////////////////
9
3096bd2f 10#include "wx/wxprec.h"
5526e819 11
2b5f62a0 12#ifdef __BORLANDC__
5526e819
VS
13#pragma hdrstop
14#endif
15
9777274a 16#if wxUSE_FILESYSTEM && wxUSE_FS_ZIP && wxUSE_ZIPSTREAM && wxUSE_ZLIB
e3e717ec 17
5526e819 18#ifndef WXPRECOMP
04dbb646
VZ
19 #include "wx/intl.h"
20 #include "wx/log.h"
5526e819
VS
21#endif
22
23#include "wx/filesys.h"
00375592 24#include "wx/wfstream.h"
6001e347 25#include "wx/zipstrm.h"
5526e819
VS
26#include "wx/fs_zip.h"
27
aaa66113 28
08335000
MW
29//---------------------------------------------------------------------------
30// wxZipFSInputStream
31//---------------------------------------------------------------------------
32// Helper class for wxZipFSHandler
33
34class wxZipFSInputStream : public wxZipInputStream
35{
36 public:
37 wxZipFSInputStream(wxFSFile *file)
38 : wxZipInputStream(*file->GetStream())
39 {
40 m_file = file;
41#if 1 //WXWIN_COMPATIBILITY_2_6
42 m_allowSeeking = true;
43#endif
44 }
45
46 virtual ~wxZipFSInputStream() { delete m_file; }
47
48 private:
49 wxFSFile *m_file;
50};
51
ddaf5566 52//----------------------------------------------------------------------------
5526e819 53// wxZipFSHandler
ddaf5566 54//----------------------------------------------------------------------------
5526e819 55
aaa66113
VS
56wxZipFSHandler::wxZipFSHandler() : wxFileSystemHandler()
57{
58 m_Archive = NULL;
59 m_ZipFile = m_Pattern = m_BaseDir = wxEmptyString;
a62848fd 60 m_AllowDirs = m_AllowFiles = true;
de0702d0 61 m_DirsFound = NULL;
aaa66113
VS
62}
63
64
65
66wxZipFSHandler::~wxZipFSHandler()
67{
f60b1d82 68 Cleanup();
aaa66113
VS
69}
70
71
f60b1d82
VZ
72void wxZipFSHandler::Cleanup()
73{
74 wxDELETE(m_Archive);
75 wxDELETE(m_DirsFound);
76}
77
78
aaa66113 79
5526e819
VS
80bool wxZipFSHandler::CanOpen(const wxString& location)
81{
82 wxString p = GetProtocol(location);
08335000 83 return (p == wxT("zip"));
5526e819
VS
84}
85
86
ba75c6bb 87wxFSFile* wxZipFSHandler::OpenFile(wxFileSystem& WXUNUSED(fs), const wxString& location)
5526e819
VS
88{
89 wxString right = GetRightLocation(location);
90 wxString left = GetLeftLocation(location);
08335000 91 wxZipInputStream *s;
5526e819 92
ddaf5566
VS
93 if (right.Contains(wxT("./")))
94 {
95 if (right.GetChar(0) != wxT('/')) right = wxT('/') + right;
96 wxFileName rightPart(right, wxPATH_UNIX);
97 rightPart.Normalize(wxPATH_NORM_DOTS, wxT("/"), wxPATH_UNIX);
98 right = rightPart.GetFullPath(wxPATH_UNIX);
99 }
a62848fd 100
505710ca
KB
101 if (right.GetChar(0) == wxT('/')) right = right.Mid(1);
102
ba75c6bb
MW
103 // a new wxFileSystem object is needed here to avoid infinite recursion
104 wxFSFile *leftFile = wxFileSystem().OpenFile(left);
08335000
MW
105 if (!leftFile)
106 return NULL;
2b5f62a0 107
08335000
MW
108 s = new wxZipFSInputStream(leftFile);
109 if (s && s->IsOk())
095472c0 110 {
468c4693
VZ
111#if wxUSE_DATETIME
112 wxDateTime dtMod;
113#endif // wxUSE_DATETIME
114
08335000 115 bool found = false;
49178e65
WS
116 while (!found)
117 {
118 wxZipEntry *ent = s->GetNextEntry();
119 if (!ent)
120 break;
468c4693 121
08335000 122 if (ent->GetInternalName() == right)
468c4693 123 {
08335000 124 found = true;
468c4693
VZ
125 dtMod = ent->GetDateTime();
126 }
127
08335000
MW
128 delete ent;
129 }
130 if (found)
f7b06c8c 131 {
08335000 132 return new wxFSFile(s,
58c837a4 133 left + wxT("#zip:") + right,
5526e819 134 GetMimeTypeFromExt(location),
e2b87f38
VZ
135 GetAnchor(location)
136#if wxUSE_DATETIME
468c4693 137 , dtMod
e2b87f38
VZ
138#endif // wxUSE_DATETIME
139 );
f7b06c8c 140 }
5526e819 141 }
3ca6a5f0
BP
142
143 delete s;
144 return NULL;
5526e819
VS
145}
146
147
148
aaa66113
VS
149wxString wxZipFSHandler::FindFirst(const wxString& spec, int flags)
150{
151 wxString right = GetRightLocation(spec);
152 wxString left = GetLeftLocation(spec);
505710ca 153
ba75c6bb 154 if (!right.empty() && right.Last() == wxT('/')) right.RemoveLast();
aaa66113 155
505710ca 156 if (m_Archive)
095472c0 157 {
08335000 158 delete m_Archive;
aaa66113
VS
159 m_Archive = NULL;
160 }
161
505710ca 162 switch (flags)
095472c0 163 {
505710ca 164 case wxFILE:
a62848fd 165 m_AllowDirs = false, m_AllowFiles = true; break;
505710ca 166 case wxDIR:
a62848fd 167 m_AllowDirs = true, m_AllowFiles = false; break;
505710ca 168 default:
a62848fd 169 m_AllowDirs = m_AllowFiles = true; break;
aaa66113
VS
170 }
171
172 m_ZipFile = left;
e06cac58 173
08335000
MW
174 wxFSFile *leftFile = wxFileSystem().OpenFile(left);
175 if (leftFile)
176 m_Archive = new wxZipFSInputStream(leftFile);
e06cac58 177
aaa66113
VS
178 m_Pattern = right.AfterLast(wxT('/'));
179 m_BaseDir = right.BeforeLast(wxT('/'));
ba75c6bb
MW
180 if (m_BaseDir.StartsWith(wxT("/")))
181 m_BaseDir = m_BaseDir.Mid(1);
aaa66113 182
505710ca 183 if (m_Archive)
095472c0 184 {
00375592 185 if (m_AllowDirs)
de0702d0 186 {
00375592 187 delete m_DirsFound;
62d9eab6 188 m_DirsFound = new wxZipFilenameHashMap();
ba75c6bb
MW
189 if (right.empty()) // allow "/" to match the archive root
190 return spec;
de0702d0 191 }
00375592 192 return DoFind();
aaa66113 193 }
095472c0 194 return wxEmptyString;
aaa66113
VS
195}
196
197
198
199wxString wxZipFSHandler::FindNext()
5526e819 200{
aaa66113
VS
201 if (!m_Archive) return wxEmptyString;
202 return DoFind();
5526e819
VS
203}
204
aaa66113
VS
205
206
207wxString wxZipFSHandler::DoFind()
208{
de0702d0 209 wxString namestr, dir, filename;
aaa66113 210 wxString match = wxEmptyString;
aaa66113
VS
211
212 while (match == wxEmptyString)
213 {
00375592
VZ
214 wxZipEntry *entry = m_Archive->GetNextEntry();
215 if (!entry)
216 {
08335000 217 delete m_Archive;
00375592
VZ
218 m_Archive = NULL;
219 break;
220 }
221 namestr = entry->GetName(wxPATH_UNIX);
222 delete entry;
aaa66113 223
de0702d0 224 if (m_AllowDirs)
095472c0 225 {
505710ca 226 dir = namestr.BeforeLast(wxT('/'));
489f6cf7 227 while (!dir.empty())
de0702d0 228 {
62d9eab6 229 if( m_DirsFound->find(dir) == m_DirsFound->end() )
de0702d0 230 {
62d9eab6 231 (*m_DirsFound)[dir] = 1;
de0702d0
VS
232 filename = dir.AfterLast(wxT('/'));
233 dir = dir.BeforeLast(wxT('/'));
489f6cf7 234 if (!filename.empty() && m_BaseDir == dir &&
a62848fd 235 wxMatchWild(m_Pattern, filename, false))
de0702d0
VS
236 match = m_ZipFile + wxT("#zip:") + dir + wxT("/") + filename;
237 }
505710ca 238 else
de0702d0
VS
239 break; // already tranversed
240 }
aaa66113 241 }
de0702d0
VS
242
243 filename = namestr.AfterLast(wxT('/'));
244 dir = namestr.BeforeLast(wxT('/'));
489f6cf7 245 if (m_AllowFiles && !filename.empty() && m_BaseDir == dir &&
a62848fd 246 wxMatchWild(m_Pattern, filename, false))
de0702d0 247 match = m_ZipFile + wxT("#zip:") + namestr;
aaa66113 248 }
505710ca 249
aaa66113
VS
250 return match;
251}
252
253
254
505710ca 255#endif
24528b0c 256 //wxUSE_FILESYSTEM && wxUSE_FS_ZIP && wxUSE_ZIPSTREAM