X-Git-Url: https://git.saurik.com/wxWidgets.git/blobdiff_plain/df467a9d7475917264cc5cffe68163effc50233c..5595181f48d45d2c277c7d40a4d24bea42d6847b:/docs/latex/wx/arc.tex diff --git a/docs/latex/wx/arc.tex b/docs/latex/wx/arc.tex index eee37b4d52..836828ae1b 100644 --- a/docs/latex/wx/arc.tex +++ b/docs/latex/wx/arc.tex @@ -10,8 +10,8 @@ \section{Archive formats such as zip}\label{wxarc} The archive classes handle archive formats such as zip, tar, rar and cab. -Currently only the wxZip classes are included. wxTar classes are under -development at \urlref{wxCode}{http://wxcode.sf.net}. +Currently \helpref{wxZip}{wxzipinputstream} +and \helpref{wxTar}{wxtarinputstream} classes are included. For each archive type, there are the following classes (using zip here as an example): @@ -52,15 +52,19 @@ For example: wxFFileOutputStream out(_T("test.zip")); wxZipOutputStream zip(out); wxTextOutputStream txt(zip); + wxString sep(wxFileName::GetPathSeparator()); zip.PutNextEntry(_T("entry1.txt")); - txt << _T("Some text for entry1\n"); + txt << _T("Some text for entry1.txt\n"); - zip.PutNextEntry(_T("entry2.txt")); - txt << _T("Some text for entry2\n"); + zip.PutNextEntry(_T("subdir") + sep + _T("entry2.txt")); + txt << _T("Some text for subdir/entry2.txt\n"); \end{verbatim} +The name of each entry can be a full path, which makes it possible to +store entries in subdirectories. + \subsection{Extracting an archive}\label{wxarcextract} @@ -75,8 +79,7 @@ the end of the entry's data. When there are no more entries, GetNextEntry() returns NULL and sets Eof(). \begin{verbatim} - // 'smart pointer' type created with wxDEFINE_SCOPED_PTR_TYPE - wxZipEntryPtr entry; + auto_ptr entry; wxFFileInputStream in(_T("test.zip")); wxZipInputStream zip(in); @@ -90,15 +93,6 @@ When there are no more entries, GetNextEntry() returns NULL and sets Eof(). \end{verbatim} -The \helpref{smart pointer}{wxscopedptr} type {\em wxZipEntryPtr} -can be created like this: - -\begin{verbatim} - #include - wxDEFINE_SCOPED_PTR_TYPE(wxZipEntry); - -\end{verbatim} - \subsection{Modifying an archive}\label{wxarcmodify} @@ -121,14 +115,13 @@ be helpful to do this. For example to delete all entries matching the pattern "*.txt": \begin{verbatim} - wxFFileInputStreamPtr in(new wxFFileInputStream(_T("test.zip"))); + auto_ptr in(new wxFFileInputStream(_T("test.zip"))); wxTempFileOutputStream out(_T("test.zip")); wxZipInputStream inzip(*in); wxZipOutputStream outzip(out); - // 'smart pointer' type created with wxDEFINE_SCOPED_PTR_TYPE - wxZipEntryPtr entry; + auto_ptr entry; // transfer any meta-data for the archive as a whole (the zip comment // in the case of zip) @@ -149,16 +142,6 @@ For example to delete all entries matching the pattern "*.txt": \end{verbatim} -The \helpref{smart pointer}{wxscopedptr} types {\em wxZipEntryPtr} -and {\em wxFFileInputStreamPtr} can be created like this: - -\begin{verbatim} - #include - wxDEFINE_SCOPED_PTR_TYPE(wxZipEntry); - wxDEFINE_SCOPED_PTR_TYPE(wxFFileInputStream); - -\end{verbatim} - \subsection{Looking up an archive entry by name}\label{wxarcbyname} @@ -187,8 +170,7 @@ it is better to convert the local name to the archive's internal format and search for that: \begin{verbatim} - // 'smart pointer' type created with wxDEFINE_SCOPED_PTR_TYPE - wxZipEntryPtr entry; + auto_ptr entry; // convert the local name we are looking for into the internal format wxString name = wxZipEntry::GetInternalName(localname); @@ -275,8 +257,8 @@ entry (e.g. filename)} In order to able to write generic code it's necessary to be able to create instances of the classes without knowing which archive type is being used. -So there is a class factory for each archive type, derived from - \helpref{wxArchiveClassFactory}{wxarchiveclassfactory}, which can create +To allow this there is a class factory for each archive type, derived from + \helpref{wxArchiveClassFactory}{wxarchiveclassfactory}, that can create the other classes. For example, given {\it wxArchiveClassFactory* factory}, streams and @@ -284,52 +266,68 @@ entries can be created like this: \begin{verbatim} // create streams without knowing their type - wxArchiveInputStreamPtr inarc(factory->NewStream(in)); - wxArchiveOutputStreamPtr outarc(factory->NewStream(out)); + auto_ptr inarc(factory->NewStream(in)); + auto_ptr outarc(factory->NewStream(out)); // create an empty entry object - wxArchiveEntryPtr entry(factory->NewEntry()); + auto_ptr entry(factory->NewEntry()); \end{verbatim} -The \helpref{smart pointer}{wxscopedptr} types {\em wxArchiveInputStreamPtr}, -{\em wxArchiveOutputStreamPtr} and {\em wxArchiveEntryPtr} would need to -have already have been defined, which could be done like this: +For the factory itself, the static member + \helpref{wxArchiveClassFactory::Find()}{wxarchiveclassfactoryfind}. +can be used to find a class factory that can handle a given file +extension or mime type. For example, given {\it filename}: \begin{verbatim} - #include - wxDEFINE_SCOPED_PTR_TYPE(wxArchiveInputStream); - wxDEFINE_SCOPED_PTR_TYPE(wxArchiveOutputStream); - wxDEFINE_SCOPED_PTR_TYPE(wxArchiveEntry); + const wxArchiveClassFactory *factory; + factory = wxArchiveClassFactory::Find(filename, wxSTREAM_FILEEXT); + + if (factory) + stream = factory->NewStream(new wxFFileInputStream(filename)); \end{verbatim} -The class factory itself can either be created explicitly: +{\it Find} does not give away ownership of the returned pointer, so it +does not need to be deleted. -\begin{verbatim} - wxArchiveClassFactory *factory = new wxZipClassFactory; - -\end{verbatim} +There are similar class factories for the filter streams that handle the +compression and decompression of a single stream, such as wxGzipInputStream. +These can be found using + \helpref{wxFilterClassFactory::Find()}{wxfilterclassfactoryfind}. -or using wxWidgets' \helpref{RTTI}{runtimeclassoverview}: +For example, to list the contents of archive {\it filename}: \begin{verbatim} -wxArchiveClassFactory *MakeFactory(const wxString& type) -{ - wxString name = _T("wx") + type.Left(1).Upper() + - type.Mid(1).Lower() + _T("ClassFactory"); - - wxObject *pObj = wxCreateDynamicObject(name); - wxArchiveClassFactory *pcf = wxDynamicCast(pObj, wxArchiveClassFactory); + auto_ptr in(new wxFFileInputStream(filename)); - if (!pcf) { - wxLogError(_T("can't handle '%s' archives"), type.c_str()); - delete pObj; + if (in->IsOk()) + { + // look for a filter handler, e.g. for '.gz' + const wxFilterClassFactory *fcf; + fcf = wxFilterClassFactory::Find(filename, wxSTREAM_FILEEXT); + if (fcf) { + in.reset(fcf->NewStream(in.release())); + // pop the extension, so if it was '.tar.gz' it is now just '.tar' + filename = fcf->PopExtension(filename); + } + + // look for a archive handler, e.g. for '.zip' or '.tar' + const wxArchiveClassFactory *acf; + acf = wxArchiveClassFactory::Find(filename, wxSTREAM_FILEEXT); + if (acf) { + auto_ptr arc(acf->NewStream(in.release())); + auto_ptr entry; + + // list the contents of the archive + while ((entry.reset(arc->GetNextEntry())), entry.get() != NULL) + std::wcout << entry->GetName().c_str() << "\n"; + } + else { + wxLogError(_T("can't handle '%s'"), filename.c_str()); + } } - return pcf; -} - \end{verbatim} @@ -410,9 +408,9 @@ required field before writing it with \helpref{CopyEntry()}{wxarchiveoutputstreamcopyentry}: \begin{verbatim} - wxArchiveInputStreamPtr arc(factory->NewStream(in)); - wxArchiveOutputStreamPtr outarc(factory->NewStream(out)); - wxArchiveEntryPtr entry; + auto_ptr arc(factory->NewStream(in)); + auto_ptr outarc(factory->NewStream(out)); + auto_ptr entry; outarc->CopyArchiveMetaData(*arc); @@ -448,10 +446,10 @@ then \helpref{SetNotifier()}{wxarchiveentrynotifier} is called before CopyEntry(): \begin{verbatim} - wxArchiveInputStreamPtr arc(factory->NewStream(in)); - wxArchiveOutputStreamPtr outarc(factory->NewStream(out)); - wxArchiveEntryPtr entry; - MyNotifier notifier; + auto_ptr arc(factory->NewStream(in)); + auto_ptr outarc(factory->NewStream(out)); + auto_ptr entry; + MyNotifier notifier; outarc->CopyArchiveMetaData(*arc);