354 lines
9.8 KiB
C
354 lines
9.8 KiB
C
|
/////////////////////////////////////////////////////////////////////////////
|
||
|
// Name: archive.h
|
||
|
// Purpose: Streams for archive formats
|
||
|
// Author: Mike Wetherell
|
||
|
// RCS-ID: $Id$
|
||
|
// Copyright: (c) 2004 Mike Wetherell
|
||
|
// Licence: wxWindows licence
|
||
|
/////////////////////////////////////////////////////////////////////////////
|
||
|
|
||
|
#ifndef _WX_ARCHIVE_H__
|
||
|
#define _WX_ARCHIVE_H__
|
||
|
|
||
|
#if defined(__GNUG__) && !defined(NO_GCC_PRAGMA)
|
||
|
#pragma interface "archive.h"
|
||
|
#endif
|
||
|
|
||
|
#include "wx/defs.h"
|
||
|
|
||
|
#if wxUSE_ZLIB && wxUSE_STREAMS && wxUSE_ZIPSTREAM
|
||
|
|
||
|
#include "wx/stream.h"
|
||
|
#include "wx/filename.h"
|
||
|
|
||
|
|
||
|
/////////////////////////////////////////////////////////////////////////////
|
||
|
// wxArchiveNotifier
|
||
|
|
||
|
class WXDLLIMPEXP_BASE wxArchiveNotifier
|
||
|
{
|
||
|
public:
|
||
|
virtual ~wxArchiveNotifier() { }
|
||
|
|
||
|
virtual void OnEntryUpdated(class wxArchiveEntry& entry) = 0;
|
||
|
};
|
||
|
|
||
|
|
||
|
/////////////////////////////////////////////////////////////////////////////
|
||
|
// wxArchiveEntry
|
||
|
//
|
||
|
// Holds an entry's meta data, such as filename and timestamp.
|
||
|
|
||
|
class WXDLLIMPEXP_BASE wxArchiveEntry : public wxObject
|
||
|
{
|
||
|
public:
|
||
|
virtual ~wxArchiveEntry() { }
|
||
|
|
||
|
virtual wxDateTime GetDateTime() const = 0;
|
||
|
virtual wxFileOffset GetSize() const = 0;
|
||
|
virtual wxFileOffset GetOffset() const = 0;
|
||
|
virtual bool IsDir() const = 0;
|
||
|
virtual bool IsReadOnly() const = 0;
|
||
|
virtual wxString GetInternalName() const = 0;
|
||
|
virtual wxPathFormat GetInternalFormat() const = 0;
|
||
|
virtual wxString GetName(wxPathFormat format = wxPATH_NATIVE) const = 0;
|
||
|
|
||
|
virtual void SetDateTime(const wxDateTime& dt) = 0;
|
||
|
virtual void SetSize(wxFileOffset size) = 0;
|
||
|
virtual void SetIsDir(bool isDir = true) = 0;
|
||
|
virtual void SetIsReadOnly(bool isReadOnly = true) = 0;
|
||
|
virtual void SetName(const wxString& name,
|
||
|
wxPathFormat format = wxPATH_NATIVE) = 0;
|
||
|
|
||
|
wxArchiveEntry *Clone() const { return DoClone(); }
|
||
|
|
||
|
void SetNotifier(wxArchiveNotifier& notifier);
|
||
|
virtual void UnsetNotifier() { m_notifier = NULL; }
|
||
|
|
||
|
protected:
|
||
|
wxArchiveEntry() : m_notifier(NULL) { }
|
||
|
|
||
|
virtual void SetOffset(wxFileOffset offset) = 0;
|
||
|
virtual wxArchiveEntry* DoClone() const = 0;
|
||
|
|
||
|
wxArchiveNotifier *GetNotifier() const { return m_notifier; }
|
||
|
wxArchiveEntry& operator=(const wxArchiveEntry& entry);
|
||
|
|
||
|
private:
|
||
|
wxArchiveNotifier *m_notifier;
|
||
|
|
||
|
DECLARE_ABSTRACT_CLASS(wxArchiveEntry)
|
||
|
};
|
||
|
|
||
|
|
||
|
/////////////////////////////////////////////////////////////////////////////
|
||
|
// wxArchiveInputStream
|
||
|
//
|
||
|
// GetNextEntry() returns an wxArchiveEntry object containing the meta-data
|
||
|
// for the next entry in the archive (and gives away ownership). Reading from
|
||
|
// the wxArchiveInputStream then returns the entry's data. Eof() becomes true
|
||
|
// after an attempt has been made to read past the end of the entry's data.
|
||
|
//
|
||
|
// When there are no more entries, GetNextEntry() returns NULL and sets Eof().
|
||
|
|
||
|
class WXDLLIMPEXP_BASE wxArchiveInputStream : public wxFilterInputStream
|
||
|
{
|
||
|
public:
|
||
|
typedef wxArchiveEntry entry_type;
|
||
|
|
||
|
virtual ~wxArchiveInputStream() { }
|
||
|
|
||
|
virtual bool OpenEntry(wxArchiveEntry& entry) = 0;
|
||
|
virtual bool CloseEntry() = 0;
|
||
|
|
||
|
wxArchiveEntry *GetNextEntry() { return DoGetNextEntry(); }
|
||
|
|
||
|
virtual char Peek() { return wxInputStream::Peek(); }
|
||
|
|
||
|
protected:
|
||
|
wxArchiveInputStream(wxInputStream& stream, wxMBConv& conv);
|
||
|
|
||
|
virtual wxArchiveEntry *DoGetNextEntry() = 0;
|
||
|
|
||
|
wxMBConv& GetConv() const { return m_conv; }
|
||
|
|
||
|
private:
|
||
|
wxMBConv& m_conv;
|
||
|
};
|
||
|
|
||
|
|
||
|
/////////////////////////////////////////////////////////////////////////////
|
||
|
// wxArchiveOutputStream
|
||
|
//
|
||
|
// PutNextEntry is used to create a new entry in the output archive, then
|
||
|
// the entry's data is written to the wxArchiveOutputStream.
|
||
|
//
|
||
|
// Only one entry can be open for output at a time; another call to
|
||
|
// PutNextEntry closes the current entry and begins the next.
|
||
|
//
|
||
|
// The overload 'bool PutNextEntry(wxArchiveEntry *entry)' takes ownership
|
||
|
// of the entry object.
|
||
|
|
||
|
class WXDLLIMPEXP_BASE wxArchiveOutputStream : public wxFilterOutputStream
|
||
|
{
|
||
|
public:
|
||
|
virtual ~wxArchiveOutputStream() { }
|
||
|
|
||
|
virtual bool PutNextEntry(wxArchiveEntry *entry) = 0;
|
||
|
|
||
|
virtual bool PutNextEntry(const wxString& name,
|
||
|
const wxDateTime& dt = wxDateTime::Now(),
|
||
|
wxFileOffset size = wxInvalidOffset) = 0;
|
||
|
|
||
|
virtual bool PutNextDirEntry(const wxString& name,
|
||
|
const wxDateTime& dt = wxDateTime::Now()) = 0;
|
||
|
|
||
|
virtual bool CopyEntry(wxArchiveEntry *entry,
|
||
|
wxArchiveInputStream& stream) = 0;
|
||
|
|
||
|
virtual bool CopyArchiveMetaData(wxArchiveInputStream& stream) = 0;
|
||
|
|
||
|
virtual bool CloseEntry() = 0;
|
||
|
virtual bool Close() = 0;
|
||
|
|
||
|
protected:
|
||
|
wxArchiveOutputStream(wxOutputStream& stream, wxMBConv& conv);
|
||
|
|
||
|
wxMBConv& GetConv() const { return m_conv; }
|
||
|
|
||
|
private:
|
||
|
wxMBConv& m_conv;
|
||
|
};
|
||
|
|
||
|
|
||
|
/////////////////////////////////////////////////////////////////////////////
|
||
|
// wxArchiveClassFactory
|
||
|
//
|
||
|
// A wxArchiveClassFactory instance for a particular archive type allows
|
||
|
// the creation of the other classes that may be needed.
|
||
|
|
||
|
class WXDLLIMPEXP_BASE wxArchiveClassFactory : public wxObject
|
||
|
{
|
||
|
public:
|
||
|
virtual ~wxArchiveClassFactory() { }
|
||
|
|
||
|
wxArchiveEntry *NewEntry() const
|
||
|
{ return DoNewEntry(); }
|
||
|
wxArchiveInputStream *NewStream(wxInputStream& stream) const
|
||
|
{ return DoNewStream(stream); }
|
||
|
wxArchiveOutputStream *NewStream(wxOutputStream& stream) const
|
||
|
{ return DoNewStream(stream); }
|
||
|
|
||
|
virtual wxString GetInternalName(
|
||
|
const wxString& name,
|
||
|
wxPathFormat format = wxPATH_NATIVE) const = 0;
|
||
|
|
||
|
void SetConv(wxMBConv& conv) { m_pConv = &conv; }
|
||
|
wxMBConv& GetConv() const { return *m_pConv; }
|
||
|
|
||
|
protected:
|
||
|
virtual wxArchiveEntry *DoNewEntry() const = 0;
|
||
|
virtual wxArchiveInputStream *DoNewStream(wxInputStream& stream) const = 0;
|
||
|
virtual wxArchiveOutputStream *DoNewStream(wxOutputStream& stream) const = 0;
|
||
|
|
||
|
wxArchiveClassFactory() : m_pConv(&wxConvLocal) { }
|
||
|
wxArchiveClassFactory& operator=(const wxArchiveClassFactory& WXUNUSED(f))
|
||
|
{ return *this; }
|
||
|
|
||
|
private:
|
||
|
wxMBConv *m_pConv;
|
||
|
|
||
|
DECLARE_ABSTRACT_CLASS(wxArchiveClassFactory)
|
||
|
};
|
||
|
|
||
|
|
||
|
/////////////////////////////////////////////////////////////////////////////
|
||
|
// wxArchiveIterator
|
||
|
//
|
||
|
// An input iterator that can be used to transfer an archive's catalog to
|
||
|
// a container.
|
||
|
|
||
|
#if wxUSE_STL || defined WX_TEST_ARCHIVE_ITERATOR
|
||
|
#include <iterator>
|
||
|
#include <utility>
|
||
|
|
||
|
template <class X, class Y>
|
||
|
void WXDLLIMPEXP_BASE _wxSetArchiveIteratorValue(
|
||
|
X& val, Y entry, void *WXUNUSED(d))
|
||
|
{
|
||
|
val = X(entry);
|
||
|
}
|
||
|
template <class X, class Y, class Z>
|
||
|
void WXDLLIMPEXP_BASE _wxSetArchiveIteratorValue(
|
||
|
std::pair<X, Y>& val, Z entry, Z WXUNUSED(d))
|
||
|
{
|
||
|
val = std::make_pair(X(entry->GetInternalName()), Y(entry));
|
||
|
}
|
||
|
|
||
|
#if defined _MSC_VER && _MSC_VER < 1300
|
||
|
template <class Arc, class T = Arc::entry_type*>
|
||
|
#else
|
||
|
template <class Arc, class T = typename Arc::entry_type*>
|
||
|
#endif
|
||
|
class WXDLLIMPEXP_BASE wxArchiveIterator
|
||
|
{
|
||
|
public:
|
||
|
typedef std::input_iterator_tag iterator_category;
|
||
|
typedef T value_type;
|
||
|
typedef ptrdiff_t difference_type;
|
||
|
typedef T* pointer;
|
||
|
typedef T& reference;
|
||
|
|
||
|
wxArchiveIterator() : m_rep(NULL) { }
|
||
|
|
||
|
wxArchiveIterator(Arc& arc) {
|
||
|
typename Arc::entry_type* entry = arc.GetNextEntry();
|
||
|
m_rep = entry ? new Rep(arc, entry) : NULL;
|
||
|
}
|
||
|
|
||
|
wxArchiveIterator(const wxArchiveIterator& it) : m_rep(it.m_rep) {
|
||
|
if (m_rep)
|
||
|
m_rep->AddRef();
|
||
|
}
|
||
|
|
||
|
~wxArchiveIterator() {
|
||
|
if (m_rep)
|
||
|
m_rep->UnRef();
|
||
|
}
|
||
|
|
||
|
const T& operator *() const {
|
||
|
return m_rep->GetValue();
|
||
|
}
|
||
|
|
||
|
const T* operator ->() const {
|
||
|
return &**this;
|
||
|
}
|
||
|
|
||
|
wxArchiveIterator& operator =(const wxArchiveIterator& it) {
|
||
|
if (it.m_rep)
|
||
|
it.m_rep.AddRef();
|
||
|
if (m_rep)
|
||
|
m_rep.UnRef();
|
||
|
m_rep = it.m_rep;
|
||
|
return *this;
|
||
|
}
|
||
|
|
||
|
wxArchiveIterator& operator ++() {
|
||
|
m_rep = m_rep->Next();
|
||
|
return *this;
|
||
|
}
|
||
|
|
||
|
wxArchiveIterator operator ++(int) {
|
||
|
wxArchiveIterator it(*this);
|
||
|
++(*this);
|
||
|
return it;
|
||
|
}
|
||
|
|
||
|
friend bool operator ==(const wxArchiveIterator& i,
|
||
|
const wxArchiveIterator& j) {
|
||
|
return i.m_rep == j.m_rep;
|
||
|
}
|
||
|
|
||
|
friend bool operator !=(const wxArchiveIterator& i,
|
||
|
const wxArchiveIterator& j) {
|
||
|
return !(i == j);
|
||
|
}
|
||
|
|
||
|
private:
|
||
|
class Rep {
|
||
|
Arc& m_arc;
|
||
|
typename Arc::entry_type* m_entry;
|
||
|
T m_value;
|
||
|
int m_ref;
|
||
|
|
||
|
public:
|
||
|
Rep(Arc& arc, typename Arc::entry_type* entry)
|
||
|
: m_arc(arc), m_entry(entry), m_value(), m_ref(1) { }
|
||
|
~Rep()
|
||
|
{ delete m_entry; }
|
||
|
|
||
|
void AddRef() {
|
||
|
m_ref++;
|
||
|
}
|
||
|
|
||
|
void UnRef() {
|
||
|
if (--m_ref == 0)
|
||
|
delete this;
|
||
|
}
|
||
|
|
||
|
Rep *Next() {
|
||
|
typename Arc::entry_type* entry = m_arc.GetNextEntry();
|
||
|
if (!entry) {
|
||
|
UnRef();
|
||
|
return NULL;
|
||
|
}
|
||
|
if (m_ref > 1) {
|
||
|
m_ref--;
|
||
|
return new Rep(m_arc, entry);
|
||
|
}
|
||
|
delete m_entry;
|
||
|
m_entry = entry;
|
||
|
m_value = T();
|
||
|
return this;
|
||
|
}
|
||
|
|
||
|
const T& GetValue() {
|
||
|
if (m_entry) {
|
||
|
_wxSetArchiveIteratorValue(m_value, m_entry, m_entry);
|
||
|
m_entry = NULL;
|
||
|
}
|
||
|
return m_value;
|
||
|
}
|
||
|
} *m_rep;
|
||
|
};
|
||
|
|
||
|
typedef wxArchiveIterator<wxArchiveInputStream> wxArchiveIter;
|
||
|
typedef wxArchiveIterator<wxArchiveInputStream,
|
||
|
std::pair<wxString, wxArchiveEntry*> > wxArchivePairIter;
|
||
|
|
||
|
#endif // wxUSE_STL || defined WX_TEST_ARCHIVE_ITERATOR
|
||
|
|
||
|
#endif // wxUSE_STREAMS
|
||
|
|
||
|
#endif // _WX_ARCHIVE_H__
|