// Purpose: input stream for ZIP archive access
// Author: Vaclav Slavik
// Copyright: (c) 1999 Vaclav Slavik
-// Licence: wxWindows Licence
+// Licence: wxWindows licence
/////////////////////////////////////////////////////////////////////////////
-#ifdef __GNUG__
+#if defined(__GNUG__) && !defined(NO_GCC_PRAGMA)
#pragma implementation "zipstrm.h"
#endif
#include "wx/stream.h"
#include "wx/wfstream.h"
#include "wx/zipstrm.h"
+#include "wx/utils.h"
/* Not the right solution (paths in makefiles) but... */
#ifdef __BORLANDC__
m_Pos = 0;
m_Size = 0;
- m_Archive = (void*) unzOpen(archive.mb_str());
+ m_Archive = (void*) unzOpen(archive.mb_str(wxConvFile));
if (m_Archive == NULL)
{
- m_lasterror = wxStream_READ_ERR;
+ m_lasterror = wxSTREAM_READ_ERROR;
return;
}
- if (unzLocateFile((unzFile)m_Archive, file.mb_str(), 0) != UNZ_OK)
+ // TODO what encoding does ZIP use?
+ if (unzLocateFile((unzFile)m_Archive, file.ToAscii(), 0) != UNZ_OK)
{
- m_lasterror = wxStream_READ_ERR;
+ m_lasterror = wxSTREAM_READ_ERROR;
return;
}
if (unzOpenCurrentFile((unzFile)m_Archive) != UNZ_OK)
{
- m_lasterror = wxStream_READ_ERR;
+ m_lasterror = wxSTREAM_READ_ERROR;
return;
}
m_Size = (size_t)zinfo.uncompressed_size;
bool wxZipInputStream::Eof() const
{
- wxASSERT_MSG( m_Pos <= (off_t)m_Size,
+ wxASSERT_MSG( m_Pos <= (wxFileOffset)m_Size,
_T("wxZipInputStream: invalid current position") );
- return m_Pos >= (off_t)m_Size;
+ return m_Pos >= (wxFileOffset)m_Size;
}
size_t wxZipInputStream::OnSysRead(void *buffer, size_t bufsize)
{
- wxASSERT_MSG( m_Pos <= (off_t)m_Size,
+ wxASSERT_MSG( m_Pos <= (wxFileOffset)m_Size,
_T("wxZipInputStream: invalid current position") );
- if ( m_Pos >= (off_t)m_Size )
+ if ( m_Pos >= (wxFileOffset)m_Size )
{
- m_lasterror = wxStream_EOF;
+ m_lasterror = wxSTREAM_EOF;
return 0;
}
-off_t wxZipInputStream::OnSysSeek(off_t seek, wxSeekMode mode)
+wxFileOffset wxZipInputStream::OnSysSeek(wxFileOffset seek, wxSeekMode mode)
{
- off_t nextpos;
- void *buf;
+ // NB: since ZIP files don't natively support seeking, we have to
+ // implement a brute force workaround -- reading all the data
+ // between current and the new position (or between beginning of
+ // the file and new position...)
- switch (mode)
+ wxFileOffset nextpos;
+
+ switch ( mode )
{
case wxFromCurrent : nextpos = seek + m_Pos; break;
case wxFromStart : nextpos = seek; break;
default : nextpos = m_Pos; break; /* just to fool compiler, never happens */
}
- // cheated seeking :
- if (nextpos > m_Pos)
+ size_t toskip;
+ if ( nextpos > m_Pos )
{
- buf = malloc(nextpos - m_Pos);
- unzReadCurrentFile((unzFile)m_Archive, buf, nextpos - m_Pos);
- free(buf);
+ toskip = nextpos - m_Pos;
}
- else if (nextpos < m_Pos) {
+ else
+ {
unzCloseCurrentFile((unzFile)m_Archive);
if (unzOpenCurrentFile((unzFile)m_Archive) != UNZ_OK)
{
- m_lasterror = wxStream_READ_ERR;
+ m_lasterror = wxSTREAM_READ_ERROR;
return m_Pos;
}
- buf = malloc(nextpos);
- unzReadCurrentFile((unzFile)m_Archive, buf, nextpos);
- free(buf);
+ toskip = nextpos;
+ }
+
+ if ( toskip > 0 )
+ {
+ const size_t BUFSIZE = 4096;
+ size_t sz;
+ char buffer[BUFSIZE];
+ while ( toskip > 0 )
+ {
+ sz = wxMin(toskip, BUFSIZE);
+ unzReadCurrentFile((unzFile)m_Archive, buffer, sz);
+ toskip -= sz;
+ }
}
m_Pos = nextpos;