]> git.saurik.com Git - wxWidgets.git/blame - src/common/fs_mem.cpp
fix the menu item bitmaps for wxMSW;
[wxWidgets.git] / src / common / fs_mem.cpp
CommitLineData
dcb86da0 1/////////////////////////////////////////////////////////////////////////////
0bca0373 2// Name: src/common/fs_mem.cpp
dcb86da0
VS
3// Purpose: in-memory file system
4// Author: Vaclav Slavik
0bca0373 5// RCS-ID: $Id$
dcb86da0 6// Copyright: (c) 2000 Vaclav Slavik
65571936 7// Licence: wxWindows licence
dcb86da0
VS
8/////////////////////////////////////////////////////////////////////////////
9
dcb86da0
VS
10#include "wx/wxprec.h"
11
2b5f62a0 12#ifdef __BORLANDC__
0bca0373 13 #pragma hdrstop
dcb86da0
VS
14#endif
15
24528b0c 16#if wxUSE_FILESYSTEM && wxUSE_STREAMS
dcb86da0 17
e2478fde
VZ
18#include "wx/fs_mem.h"
19
b4f4d3dd 20#ifndef WX_PRECOMP
04dbb646
VZ
21 #include "wx/intl.h"
22 #include "wx/log.h"
6b24b421 23 #include "wx/wxcrtvararg.h"
0bca0373 24 #if wxUSE_GUI
155ecd4c 25 #include "wx/image.h"
0bca0373 26 #endif // wxUSE_GUI
dcb86da0
VS
27#endif
28
dcb86da0
VS
29#include "wx/mstream.h"
30
8c9d7602
VZ
31// represents a file entry in wxMemoryFS
32class wxMemoryFSFile
dcb86da0 33{
cb564879 34public:
8c9d7602 35 wxMemoryFSFile(const void *data, size_t len, const wxString& mime)
cb564879
VZ
36 {
37 m_Data = new char[len];
38 memcpy(m_Data, data, len);
39 m_Len = len;
40 m_MimeType = mime;
41 InitTime();
42 }
43
8c9d7602 44 wxMemoryFSFile(const wxMemoryOutputStream& stream, const wxString& mime)
cb564879
VZ
45 {
46 m_Len = stream.GetSize();
47 m_Data = new char[m_Len];
48 stream.CopyTo(m_Data, m_Len);
49 m_MimeType = mime;
50 InitTime();
51 }
52
8c9d7602 53 virtual ~wxMemoryFSFile()
cb564879
VZ
54 {
55 delete[] m_Data;
56 }
57
58 char *m_Data;
59 size_t m_Len;
60 wxString m_MimeType;
e2b87f38 61#if wxUSE_DATETIME
cb564879 62 wxDateTime m_Time;
e2b87f38 63#endif // wxUSE_DATETIME
22f3361e 64
cb564879
VZ
65private:
66 void InitTime()
67 {
e2b87f38 68#if wxUSE_DATETIME
cb564879 69 m_Time = wxDateTime::Now();
a62848fd 70#endif // wxUSE_DATETIME
cb564879 71 }
dcb86da0 72
8c9d7602
VZ
73 DECLARE_NO_COPY_CLASS(wxMemoryFSFile)
74};
0eb2e510 75
ec67cff1 76#if wxUSE_BASE
60c0a8db 77
dcb86da0
VS
78
79//--------------------------------------------------------------------------------
80// wxMemoryFSHandler
81//--------------------------------------------------------------------------------
82
83
8c9d7602 84wxMemoryFSHash wxMemoryFSHandlerBase::m_Hash;
dcb86da0
VS
85
86
e2478fde 87wxMemoryFSHandlerBase::wxMemoryFSHandlerBase() : wxFileSystemHandler()
dcb86da0 88{
dcb86da0
VS
89}
90
e2478fde 91wxMemoryFSHandlerBase::~wxMemoryFSHandlerBase()
dcb86da0
VS
92{
93 // as only one copy of FS handler is supposed to exist, we may silently
94 // delete static data here. (There is no way how to remove FS handler from
95 // wxFileSystem other than releasing _all_ handlers.)
8c9d7602 96 WX_CLEAR_HASH_MAP(wxMemoryFSHash, m_Hash);
dcb86da0
VS
97}
98
e2478fde 99bool wxMemoryFSHandlerBase::CanOpen(const wxString& location)
dcb86da0 100{
cb564879 101 return GetProtocol(location) == "memory";
dcb86da0
VS
102}
103
cb564879
VZ
104wxFSFile * wxMemoryFSHandlerBase::OpenFile(wxFileSystem& WXUNUSED(fs),
105 const wxString& location)
106{
8c9d7602
VZ
107 wxMemoryFSHash::const_iterator i = m_Hash.find(GetRightLocation(location));
108 if ( i == m_Hash.end() )
cb564879 109 return NULL;
dcb86da0 110
8c9d7602 111 const wxMemoryFSFile * const obj = i->second;
0eb2e510
VZ
112
113 return new wxFSFile
114 (
115 new wxMemoryInputStream(obj->m_Data, obj->m_Len),
116 location,
117 obj->m_MimeType,
118 GetAnchor(location)
e2b87f38 119#if wxUSE_DATETIME
0eb2e510 120 , obj->m_Time
a62848fd 121#endif // wxUSE_DATETIME
0eb2e510 122 );
dcb86da0
VS
123}
124
fcc65883 125wxString wxMemoryFSHandlerBase::FindFirst(const wxString& spec, int flags)
dcb86da0 126{
fcc65883
VZ
127 if ( (flags & wxDIR) && !(flags & wxFILE) )
128 {
129 // we only store files, not directories, so we don't risk finding
130 // anything
131 return wxString();
132 }
133
134 if ( spec.find_first_of("?*") == wxString::npos )
135 {
136 // simple case: there are no wildcard characters so we can return
137 // either 0 or 1 results and we can find the potential match quickly
138 return m_Hash.count(spec) ? spec : wxString();
139 }
140 //else: deal with wildcards in FindNext()
bc1dcfc1 141
fcc65883
VZ
142 m_findArgument = spec;
143 m_findIter = m_Hash.begin();
144
145 return FindNext();
dcb86da0
VS
146}
147
e2478fde 148wxString wxMemoryFSHandlerBase::FindNext()
dcb86da0 149{
fcc65883
VZ
150 // m_findArgument is used to indicate that search is in progress, we reset
151 // it to empty string after iterating over all elements
152 while ( !m_findArgument.empty() )
153 {
154 // advance m_findIter before checking the value at the current position
155 // as we need to do it anyhow, whether it matches or not
156 const wxMemoryFSHash::const_iterator current = m_findIter;
157
158 if ( ++m_findIter == m_Hash.end() )
159 m_findArgument.clear();
160
161 if ( current->first.Matches(m_findArgument) )
162 return current->first;
163 }
bc1dcfc1 164
fcc65883 165 return wxString();
dcb86da0
VS
166}
167
8c9d7602 168bool wxMemoryFSHandlerBase::CheckDoesntExist(const wxString& filename)
dcb86da0 169{
8c9d7602 170 if ( m_Hash.count(filename) )
dcb86da0 171 {
cb564879 172 wxLogError(_("Memory VFS already contains file '%s'!"), filename);
a62848fd 173 return false;
dcb86da0 174 }
cb564879
VZ
175
176 return true;
dcb86da0
VS
177}
178
179
c5d7b81e
VS
180/*static*/
181void wxMemoryFSHandlerBase::AddFileWithMimeType(const wxString& filename,
182 const wxString& textdata,
183 const wxString& mimetype)
dcb86da0 184{
c5d7b81e
VS
185 AddFileWithMimeType(filename,
186 (const void*) textdata.mb_str(), textdata.length(),
187 mimetype);
dcb86da0
VS
188}
189
190
c5d7b81e
VS
191/*static*/
192void wxMemoryFSHandlerBase::AddFileWithMimeType(const wxString& filename,
193 const void *binarydata, size_t size,
194 const wxString& mimetype)
dcb86da0 195{
8c9d7602 196 if ( !CheckDoesntExist(filename) )
cb564879
VZ
197 return;
198
8c9d7602 199 m_Hash[filename] = new wxMemoryFSFile(binarydata, size, mimetype);
c5d7b81e
VS
200}
201
202/*static*/
203void wxMemoryFSHandlerBase::AddFile(const wxString& filename,
204 const wxString& textdata)
205{
206 AddFileWithMimeType(filename, textdata, wxEmptyString);
207}
208
209
210/*static*/
211void wxMemoryFSHandlerBase::AddFile(const wxString& filename,
212 const void *binarydata, size_t size)
213{
214 AddFileWithMimeType(filename, binarydata, size, wxEmptyString);
dcb86da0
VS
215}
216
217
218
e2478fde 219/*static*/ void wxMemoryFSHandlerBase::RemoveFile(const wxString& filename)
dcb86da0 220{
8c9d7602
VZ
221 wxMemoryFSHash::iterator i = m_Hash.find(filename);
222 if ( i == m_Hash.end() )
dcb86da0 223 {
8c9d7602
VZ
224 wxLogError(_("Trying to remove file '%s' from memory VFS, "
225 "but it is not loaded!"),
226 filename);
227 return;
cb564879 228 }
0eb2e510 229
8c9d7602
VZ
230 delete i->second;
231 m_Hash.erase(i);
dcb86da0
VS
232}
233
ec67cff1 234#endif // wxUSE_BASE
e2478fde
VZ
235
236#if wxUSE_GUI
237
2de5a6ee
VS
238#if wxUSE_IMAGE
239/*static*/ void
989a2421
VZ
240wxMemoryFSHandler::AddFile(const wxString& filename,
241 const wxImage& image,
d75a69e8 242 wxBitmapType type)
2de5a6ee 243{
8c9d7602 244 if ( !CheckDoesntExist(filename) )
cb564879 245 return;
2de5a6ee 246
2de5a6ee 247 wxMemoryOutputStream mems;
0eb2e510 248 if ( image.Ok() && image.SaveFile(mems, type) )
c5d7b81e 249 {
8c9d7602
VZ
250 m_Hash[filename] = new wxMemoryFSFile
251 (
0eb2e510
VZ
252 mems,
253 wxImage::FindHandler(type)->GetMimeType()
8c9d7602 254 );
c5d7b81e 255 }
2de5a6ee
VS
256 else
257 {
cb564879 258 wxLogError(_("Failed to store image '%s' to memory VFS!"), filename);
2de5a6ee
VS
259 }
260}
2de5a6ee 261
989a2421
VZ
262/*static*/ void
263wxMemoryFSHandler::AddFile(const wxString& filename,
264 const wxBitmap& bitmap,
d75a69e8 265 wxBitmapType type)
e2478fde
VZ
266{
267 wxImage img = bitmap.ConvertToImage();
268 AddFile(filename, img, type);
269}
270
1904aa72
DS
271#endif // wxUSE_IMAGE
272
273#endif // wxUSE_GUI
dcb86da0
VS
274
275
24528b0c 276#endif // wxUSE_FILESYSTEM && wxUSE_FS_ZIP