mirror of
https://github.com/fadden/ciderpress.git
synced 2024-12-24 17:29:22 +00:00
814872966e
The previous code was a stickler for only opening files whose type matched what was selected in the filter pop-up. The original goal was to allow you to choose whether a BXY or SDK file was interpreted as Binary II, ShrinkIt, or disk image, since they could go either way. Unfortunately, its refusal to consider types other than what was selected made it kind of annoying. The new code will start by trying to open the file with the selected filter, so that it's still possible to choose how SDK and BXY files are opened. However, it now continues on, trying all other types before finally giving up. If the generic ("*.*") filter is selected, CiderPress will start by trying to open the file as a disk image. This seems to produce good results with a variety of known and unknown files.
1027 lines
36 KiB
C++
1027 lines
36 KiB
C++
/*
|
|
* CiderPress
|
|
* Copyright (C) 2007 by faddenSoft, LLC. All Rights Reserved.
|
|
* See the file LICENSE for distribution terms.
|
|
*/
|
|
/*
|
|
* Generic Apple II archive handling. In the beginning we only handled
|
|
* NuFX archives, and the code continues to reflect that heritage.
|
|
*
|
|
* These are abstract base classes.
|
|
*/
|
|
#ifndef APP_GENERICARCHIVE_H
|
|
#define APP_GENERICARCHIVE_H
|
|
|
|
#include "Preferences.h"
|
|
#include "../util/UtilLib.h"
|
|
#include "../diskimg/DiskImg.h"
|
|
#include "../nufxlib/NufxLib.h"
|
|
#include "../reformat/Reformat.h"
|
|
#include <time.h>
|
|
#include <string.h>
|
|
|
|
// this shouldn't be in a header file, but in here it's probably okay
|
|
using namespace DiskImgLib;
|
|
|
|
class ActionProgressDialog;
|
|
class AddFilesDialog;
|
|
class RecompressOptionsDialog;
|
|
class SelectionSet;
|
|
struct Win32dirent;
|
|
class GenericArchive;
|
|
|
|
const int kFileTypeTXT = 0x04;
|
|
const int kFileTypeBIN = 0x06;
|
|
const int kFileTypeSRC = 0xb0;
|
|
const int kFileTypeINT = 0xfa;
|
|
const int kFileTypeBAS = 0xfc;
|
|
|
|
/*
|
|
* Set of data allowed in file property "set file info" calls.
|
|
*/
|
|
struct FileProps {
|
|
uint32_t fileType;
|
|
uint32_t auxType;
|
|
uint32_t access;
|
|
time_t createWhen;
|
|
time_t modWhen;
|
|
};
|
|
|
|
/*
|
|
* Options for converting between file archives and disk archives.
|
|
*/
|
|
class XferFileOptions {
|
|
public:
|
|
XferFileOptions() :
|
|
fTarget(NULL), fPreserveEmptyFolders(false), fpTargetFS(NULL)
|
|
{}
|
|
~XferFileOptions() {}
|
|
|
|
/* where the stuff is going */
|
|
GenericArchive* fTarget;
|
|
|
|
/* these really only have meaning when converting disk to file */
|
|
//bool fConvDOSText;
|
|
//bool fConvPascalText;
|
|
bool fPreserveEmptyFolders;
|
|
|
|
/* only useful when converting files to a disk image */
|
|
//CString fStoragePrefix;
|
|
DiskImgLib::DiskFS* fpTargetFS;
|
|
};
|
|
|
|
|
|
/*
|
|
* Generic description of an Apple II file.
|
|
*
|
|
* Everything returned by the basic "get" calls for display in the ContentList
|
|
* must be held in local storage (i.e. not just pointers into DiskFS data).
|
|
* Otherwise, we run the risk of doing some DiskFS updates and having weird
|
|
* things happen in the ContentList.
|
|
*/
|
|
class GenericEntry {
|
|
public:
|
|
GenericEntry();
|
|
virtual ~GenericEntry();
|
|
|
|
/* kinds of files found in archives */
|
|
enum RecordKind {
|
|
kRecordKindUnknown = 0,
|
|
kRecordKindDisk,
|
|
kRecordKindFile,
|
|
kRecordKindForkedFile,
|
|
kRecordKindDirectory,
|
|
kRecordKindVolumeDir,
|
|
};
|
|
/*
|
|
* Threads we will view or extract (threadMask). This is no longer used
|
|
* for viewing files, but still plays a role when extracting.
|
|
*/
|
|
enum {
|
|
// create one entry for each matching thread
|
|
kDataThread = 0x01,
|
|
kRsrcThread = 0x02,
|
|
kDiskImageThread = 0x04,
|
|
kCommentThread = 0x08,
|
|
|
|
// grab any of the above threads
|
|
kAnyThread = 0x10,
|
|
|
|
// set this if we allow matches on directory entries
|
|
kAllowDirectory = 0x20,
|
|
|
|
// and volume directory entries
|
|
kAllowVolumeDir = 0x40,
|
|
|
|
// set to include "damaged" files
|
|
kAllowDamaged = 0x80,
|
|
};
|
|
/* EOL conversion mode for threads being extracted */
|
|
enum ConvertEOL {
|
|
kConvertUnknown = 0, kConvertEOLOff, kConvertEOLOn, kConvertEOLAuto
|
|
};
|
|
enum EOLType {
|
|
kEOLUnknown = 0, kEOLCR, kEOLLF, kEOLCRLF
|
|
};
|
|
/* high ASCII conversion mode for threads being extracted */
|
|
enum ConvertHighASCII {
|
|
kConvertHAUnknown = 0, kConvertHAOff, kConvertHAOn, kConvertHAAuto
|
|
};
|
|
|
|
/* ProDOS access flags, used for all filesystems */
|
|
enum {
|
|
kAccessRead = 0x01,
|
|
kAccessWrite = 0x02,
|
|
kAccessInvisible = 0x04,
|
|
kAccessBackup = 0x20,
|
|
kAccessRename = 0x40,
|
|
kAccessDelete = 0x80
|
|
};
|
|
|
|
/*
|
|
* Features supported by underlying archive. Primarily of interest
|
|
* to EditPropsDialog, which needs to know what sort of attributes can
|
|
* be altered in the file type and access flags.
|
|
*/
|
|
enum Feature {
|
|
kFeatureCanChangeType,
|
|
kFeaturePascalTypes,
|
|
kFeatureDOSTypes,
|
|
kFeatureHFSTypes,
|
|
kFeatureHasFullAccess,
|
|
kFeatureHasSimpleAccess, // mutually exclusive with FullAccess
|
|
kFeatureHasInvisibleFlag,
|
|
};
|
|
|
|
/*
|
|
* Extract data from an archive (NuFX, disk image, etc).
|
|
*
|
|
* If "*ppText" is non-NULL, the data will be read into the pointed-to buffer
|
|
* so long as it's shorter than *pLength bytes. The value in "*pLength"
|
|
* will be set to the actual length used.
|
|
*
|
|
* If "*ppText" is NULL, the uncompressed data will be placed into a buffer
|
|
* allocated with "new[]".
|
|
*
|
|
* Returns IDOK on success, IDCANCEL if the operation was cancelled by the
|
|
* user, and -1 value on failure. On failure, "*pErrMsg" holds an error
|
|
* message.
|
|
*
|
|
* "which" is an anonymous GenericArchive enum (e.g. "kDataThread").
|
|
*/
|
|
virtual int ExtractThreadToBuffer(int which, char** ppText, long* pLength,
|
|
CString* pErrMsg) const = 0;
|
|
|
|
/*
|
|
* Extract data from a thread or disk file to a Windows file. Since we're
|
|
* not copying to a buffer we can't assume that we're able to hold the
|
|
* entire file in memory all at once.
|
|
*
|
|
* Returns IDOK on success, IDCANCEL if the operation was cancelled by the
|
|
* user, and -1 value on failure. On failure, "*pMsg" holds an
|
|
* error message.
|
|
*/
|
|
virtual int ExtractThreadToFile(int which, FILE* outfp, ConvertEOL conv,
|
|
ConvertHighASCII convHA, CString* pErrMsg) const = 0;
|
|
|
|
// This helps us retain the ContentList selection across a Reload(). Only
|
|
// necessary for read-write archives, since those are the only ones that
|
|
// ever need to be reloaded. Value must be nonzero to be used.
|
|
virtual long GetSelectionSerial() const = 0;
|
|
|
|
/* what operations are possible with this entry? */
|
|
virtual bool GetFeatureFlag(Feature feature) const = 0;
|
|
|
|
long GetIndex() const { return fIndex; }
|
|
void SetIndex(long idx) { fIndex = idx; }
|
|
|
|
/*
|
|
* Set the pathname. This comes from a file archive or disk image,
|
|
* so it's always in Mac OS Roman format.
|
|
*
|
|
* Calling this will invalidate any strings previously returned by
|
|
* GetPathName*(), GetFileName*(), and GetDisplayName().
|
|
*/
|
|
void SetPathNameMOR(const char* pathNameMOR);
|
|
|
|
const CStringA& GetPathNameMOR() const { return fPathNameMOR; }
|
|
const CString& GetPathNameUNI() const { return fPathNameUNI; }
|
|
const CString& GetFileName();
|
|
const CString& GetFileNameExtension(); // returns e.g. ".SHK"
|
|
const CStringA& GetFileNameExtensionMOR();
|
|
/*
|
|
* Returns the "display" name. This is a combination of the sub-volume
|
|
* name and the path name. This string is intended for display only,
|
|
* and may include characters that aren't legal on the filesystem.
|
|
*/
|
|
const CString& GetDisplayName() const;
|
|
|
|
void SetSubVolName(const WCHAR* name);
|
|
const CString& GetSubVolName() const { return fSubVolName; }
|
|
|
|
char GetFssep() const { return fFssep; }
|
|
void SetFssep(char fssep) { fFssep = fssep; }
|
|
uint32_t GetFileType() const { return fFileType; }
|
|
void SetFileType(uint32_t type) { fFileType = type; }
|
|
uint32_t GetAuxType() const { return fAuxType; }
|
|
void SetAuxType(uint32_t type) { fAuxType = type; }
|
|
uint32_t GetAccess() const { return fAccess; }
|
|
void SetAccess(uint32_t access) { fAccess = access; }
|
|
time_t GetCreateWhen() const { return fCreateWhen; }
|
|
void SetCreateWhen(time_t when) { fCreateWhen = when; }
|
|
time_t GetModWhen() const { return fModWhen; }
|
|
void SetModWhen(time_t when) { fModWhen = when; }
|
|
RecordKind GetRecordKind() const { return fRecordKind; }
|
|
void SetRecordKind(RecordKind recordKind) { fRecordKind = recordKind; }
|
|
const WCHAR* GetFormatStr() const { return fFormatStr; }
|
|
void SetFormatStr(const WCHAR* str) { fFormatStr = str; } // arg not copied, must be static!
|
|
LONGLONG GetCompressedLen() const { return fCompressedLen; }
|
|
void SetCompressedLen(LONGLONG len) { fCompressedLen = len; }
|
|
LONGLONG GetUncompressedLen() const {
|
|
return fDataForkLen + fRsrcForkLen;
|
|
}
|
|
LONGLONG GetDataForkLen() const { return fDataForkLen; }
|
|
void SetDataForkLen(LONGLONG len) { fDataForkLen = len; }
|
|
LONGLONG GetRsrcForkLen() const { return fRsrcForkLen; }
|
|
void SetRsrcForkLen(LONGLONG len) { fRsrcForkLen = len; }
|
|
|
|
DiskImg::FSFormat GetSourceFS() const { return fSourceFS; }
|
|
void SetSourceFS(DiskImg::FSFormat fmt) { fSourceFS = fmt; }
|
|
|
|
bool GetHasDataFork() const { return fHasDataFork; }
|
|
void SetHasDataFork(bool val) { fHasDataFork = val; }
|
|
bool GetHasRsrcFork() const { return fHasRsrcFork; }
|
|
void SetHasRsrcFork(bool val) { fHasRsrcFork = val; }
|
|
bool GetHasDiskImage() const { return fHasDiskImage; }
|
|
void SetHasDiskImage(bool val) { fHasDiskImage = val; }
|
|
bool GetHasComment() const { return fHasComment; }
|
|
void SetHasComment(bool val) { fHasComment = val; }
|
|
bool GetHasNonEmptyComment() const { return fHasNonEmptyComment; }
|
|
void SetHasNonEmptyComment(bool val) { fHasNonEmptyComment = val; }
|
|
|
|
bool GetDamaged() const { return fDamaged; }
|
|
void SetDamaged(bool val) { fDamaged = val; }
|
|
bool GetSuspicious() const { return fSuspicious; }
|
|
void SetSuspicious(bool val) { fSuspicious = val; }
|
|
|
|
GenericEntry* GetPrev() const { return fpPrev; }
|
|
void SetPrev(GenericEntry* pEntry) { fpPrev = pEntry; }
|
|
GenericEntry* GetNext() const { return fpNext; }
|
|
void SetNext(GenericEntry* pEntry) { fpNext = pEntry; }
|
|
|
|
/*
|
|
* Get a string for this entry's filetype.
|
|
*/
|
|
const WCHAR* GetFileTypeString() const;
|
|
|
|
/*
|
|
* Check to see if this is a high-ASCII file.
|
|
*/
|
|
static bool CheckHighASCII(const uint8_t* buffer, size_t count);
|
|
|
|
/*
|
|
* Decide, based on the contents of the buffer, whether we should do an
|
|
* EOL conversion on the data.
|
|
*
|
|
* Returns kConvEOLOff or kConvEOLOn.
|
|
*/
|
|
static ConvertEOL DetermineConversion(const uint8_t* buffer,
|
|
size_t count, EOLType* pSourceType, ConvertHighASCII* pConvHA);
|
|
|
|
/*
|
|
* Write data to a file, possibly converting EOL markers to Windows CRLF
|
|
* and stripping high ASCII.
|
|
*
|
|
* If "*pConv" is kConvertEOLAuto, this will try to auto-detect whether
|
|
* the input is a text file or not by scanning the input buffer.
|
|
*
|
|
* Ditto for "*pConvHA".
|
|
*
|
|
* "fp" is the output file, "buf" is the input, "len" is the buffer length.
|
|
* "*pLastCR" should initially be "false", and carried across invocations.
|
|
*
|
|
* Returns 0 on success, or an errno value on error.
|
|
*/
|
|
static int GenericEntry::WriteConvert(FILE* fp, const char* buf,
|
|
size_t len, ConvertEOL* pConv, ConvertHighASCII* pConvHA,
|
|
bool* pLastCR);
|
|
|
|
private:
|
|
/*
|
|
* Convert spaces to underscores, modifying the string.
|
|
*/
|
|
static void SpacesToUnderscores(CStringA* pStr);
|
|
|
|
private:
|
|
/*
|
|
* This represents a file from an archive or disk image, so the Mac OS
|
|
* Roman representation is the "true" version. The Unicode version
|
|
* is how we will use it on Windows (e.g. for file extraction), so
|
|
* it will be a CP-1252 conversion until the various libraries
|
|
* support UTF-16 filenames.
|
|
*
|
|
* The "display name" is only used for display, and should do a proper
|
|
* MOR to Unicode conversion so the file name looks right.
|
|
*/
|
|
|
|
CStringA fPathNameMOR; // original path name, Mac OS Roman chars
|
|
CString fPathNameUNI; // Unicode conversion
|
|
CString fFileName; // filename component of fPathNameUNI
|
|
CString fFileNameExtension; // filename extension from fPathNameUNI
|
|
CStringA fFileNameExtensionMOR;
|
|
char fFssep;
|
|
CString fSubVolName; // sub-volume prefix, or NULL if none
|
|
mutable CString fDisplayName; // combination of sub-vol and path
|
|
uint32_t fFileType;
|
|
uint32_t fAuxType;
|
|
uint32_t fAccess;
|
|
time_t fCreateWhen;
|
|
time_t fModWhen;
|
|
RecordKind fRecordKind; // forked file, disk image, ??
|
|
const WCHAR* fFormatStr; // static str; compression or fs format
|
|
LONGLONG fDataForkLen; // also for disk images
|
|
LONGLONG fRsrcForkLen; // set to 0 when nonexistent
|
|
LONGLONG fCompressedLen; // data/disk + rsrc
|
|
|
|
DiskImg::FSFormat fSourceFS; // if DOS3.3, text files have funky format
|
|
|
|
bool fHasDataFork;
|
|
bool fHasRsrcFork;
|
|
bool fHasDiskImage;
|
|
bool fHasComment;
|
|
bool fHasNonEmptyComment; // set if fHasComment and it isn't empty
|
|
|
|
bool fDamaged; // if set, don't try to open file
|
|
bool fSuspicious; // if set, file *might* be damaged
|
|
|
|
long fIndex; // serial index, for sorting "unsorted" view
|
|
GenericEntry* fpPrev;
|
|
GenericEntry* fpNext;
|
|
};
|
|
|
|
/*
|
|
* Generic representation of a collection of Apple II files.
|
|
*
|
|
* This raises the "reload" flag whenever its data is reloaded. Any code that
|
|
* keeps pointers to stuff (e.g. local copies of pointers to GenericEntry
|
|
* objects) needs to check the "reload" flag before dereferencing them.
|
|
*/
|
|
class GenericArchive {
|
|
public:
|
|
GenericArchive() {
|
|
fPathName = NULL;
|
|
fNumEntries = 0;
|
|
fEntryHead = fEntryTail = NULL;
|
|
fReloadFlag = true;
|
|
//fEntryIndex = NULL;
|
|
}
|
|
virtual ~GenericArchive() {
|
|
//LOGI("Deleting GenericArchive");
|
|
DeleteEntries();
|
|
delete fPathName;
|
|
}
|
|
|
|
virtual GenericEntry* GetEntries() const {
|
|
return fEntryHead;
|
|
}
|
|
virtual long GetNumEntries() const {
|
|
return fNumEntries;
|
|
}
|
|
|
|
enum OpenResult {
|
|
kResultUnknown = 0,
|
|
kResultSuccess, // open succeeded
|
|
kResultFailure, // open failed
|
|
kResultCancel, // open was cancelled by user
|
|
kResultFileArchive, // found a file archive rather than disk image
|
|
};
|
|
|
|
// Open an archive and do fun things with the innards.
|
|
virtual OpenResult Open(const WCHAR* filename, bool readOnly,
|
|
CString* pErrMsg) = 0;
|
|
// Create a new archive with the specified name.
|
|
virtual CString New(const WCHAR* filename, const void* options) = 0;
|
|
// Flush any unwritten data to disk
|
|
virtual CString Flush() = 0;
|
|
// Force a re-read from the underlying storage.
|
|
virtual CString Reload() = 0;
|
|
// Do we allow modification?
|
|
virtual bool IsReadOnly() const = 0;
|
|
// Does the underlying storage have un-flushed modifications?
|
|
virtual bool IsModified() const = 0;
|
|
|
|
virtual bool GetReloadFlag() { return fReloadFlag; }
|
|
virtual void ClearReloadFlag() { fReloadFlag = false; }
|
|
|
|
// One of these for every sub-class.
|
|
enum ArchiveKind {
|
|
kArchiveUnknown = 0,
|
|
kArchiveNuFX,
|
|
kArchiveBNY,
|
|
kArchiveACU,
|
|
kArchiveAppleSingle,
|
|
kArchiveDiskImage,
|
|
};
|
|
|
|
// Returns the kind of archive this is (disk image, NuFX, BNY, etc).
|
|
virtual ArchiveKind GetArchiveKind() = 0;
|
|
|
|
// Get a nice description for the title bar.
|
|
virtual CString GetDescription() const = 0;
|
|
|
|
// Do a bulk add.
|
|
virtual bool BulkAdd(ActionProgressDialog* pActionProgress,
|
|
const AddFilesDialog* pAddOpts) = 0;
|
|
|
|
// Add a single disk to the archive.
|
|
virtual bool AddDisk(ActionProgressDialog* pActionProgress,
|
|
const AddFilesDialog* pAddOpts) = 0;
|
|
|
|
// Create a subdirectory with name newName in pParentEntry.
|
|
virtual bool CreateSubdir(CWnd* pMsgWnd, GenericEntry* pParentEntry,
|
|
const WCHAR* newName) = 0;
|
|
|
|
// Test a set of files.
|
|
virtual bool TestSelection(CWnd* pMsgWnd, SelectionSet* pSelSet) = 0;
|
|
|
|
// Delete a set of files.
|
|
virtual bool DeleteSelection(CWnd* pMsgWnd, SelectionSet* pSelSet) = 0;
|
|
|
|
// Rename a set of files.
|
|
virtual bool RenameSelection(CWnd* pMsgWnd, SelectionSet* pSelSet) = 0;
|
|
|
|
// Verify that a name is suitable. Called by RenameEntryDialog and
|
|
// CreateSubdirDialog.
|
|
//
|
|
// Tests for context-specific syntax and checks for duplicates.
|
|
//
|
|
// Returns an empty string on success, or an error message on failure.
|
|
virtual CString TestPathName(const GenericEntry* pGenericEntry,
|
|
const CString& basePath, const CString& newName, char newFssep) const = 0;
|
|
|
|
// Rename a volume (or sub-volume)
|
|
virtual bool RenameVolume(CWnd* pMsgWnd, DiskFS* pDiskFS,
|
|
const WCHAR* newName) = 0;
|
|
virtual CString TestVolumeName(const DiskFS* pDiskFS,
|
|
const WCHAR* newName) const = 0;
|
|
|
|
// Recompress a set of files.
|
|
virtual bool RecompressSelection(CWnd* pMsgWnd, SelectionSet* pSelSet,
|
|
const RecompressOptionsDialog* pRecompOpts) = 0;
|
|
|
|
// return result from XferSelection()
|
|
enum XferStatus {
|
|
kXferOK = 0, kXferFailed = 1, kXferCancelled = 2, kXferOutOfSpace = 3
|
|
};
|
|
|
|
// Transfer selected files out of this archive and into another.
|
|
virtual XferStatus XferSelection(CWnd* pMsgWnd, SelectionSet* pSelSet,
|
|
ActionProgressDialog* pActionProgress,
|
|
const XferFileOptions* pXferOpts) = 0;
|
|
|
|
// Extract a comment from the archive, converting line terminators to CRLF.
|
|
virtual bool GetComment(CWnd* pMsgWnd, const GenericEntry* pEntry,
|
|
CString* pStr) = 0;
|
|
|
|
// Set a comment on an entry.
|
|
virtual bool SetComment(CWnd* pMsgWnd, GenericEntry* pEntry,
|
|
const CString& str) = 0;
|
|
|
|
// Delete the comment from the entry.
|
|
virtual bool DeleteComment(CWnd* pMsgWnd, GenericEntry* pEntry) = 0;
|
|
|
|
// Set ProDOS file properties (file type, aux type, access flags).
|
|
virtual bool SetProps(CWnd* pMsgWnd, GenericEntry* pEntry,
|
|
const FileProps* pProps) = 0;
|
|
|
|
// Preferences have changed, update library state as needed. Also called
|
|
// the first time though.
|
|
virtual void PreferencesChanged() = 0;
|
|
|
|
// Determine an archive's capabilities. This is specific to the object
|
|
// instance, so this must not be made a static function.
|
|
enum Capability {
|
|
kCapUnknown = 0,
|
|
|
|
kCapCanTest, // NuFX, BNY
|
|
kCapCanRenameFullPath, // NuFX, BNY
|
|
kCapCanRecompress, // NuFX, BNY
|
|
kCapCanEditComment, // NuFX
|
|
kCapCanAddDisk, // NuFX
|
|
kCapCanConvEOLOnAdd, // Disk
|
|
kCapCanCreateSubdir, // Disk
|
|
kCapCanRenameVolume, // Disk
|
|
};
|
|
virtual long GetCapability(Capability cap) = 0;
|
|
|
|
// Get the pathname of the file we opened.
|
|
const WCHAR* GetPathName() const { return fPathName; }
|
|
|
|
/*
|
|
* Generate a temp name from a file name.
|
|
*/
|
|
static CString GenDerivedTempName(const WCHAR* filename);
|
|
|
|
/*
|
|
* Do a strcasecmp-like comparison, taking equivalent fssep chars into
|
|
* account.
|
|
*
|
|
* The tricky part is with files like "foo:bar" ':' -- "foo:bar" '/'. The
|
|
* names appear to match, but the fssep chars are different, so they don't.
|
|
* If we just return (char1 - char2), though, we'll be returning 0 because
|
|
* the ASCII values match even if the character *meanings* don't.
|
|
*
|
|
* This assumes that the fssep char is not affected by tolower().
|
|
*
|
|
* [This may not sort correctly...haven't verified that I'm returning the
|
|
* right thing for ascending ASCII sort.]
|
|
*/
|
|
static int ComparePaths(const CString& name1, char fssep1,
|
|
const CString& name2, char fssep2);
|
|
|
|
/*
|
|
* Add a new entry to the end of the list.
|
|
*/
|
|
void AddEntry(GenericEntry* pEntry);
|
|
|
|
/*
|
|
* This class holds details about a file that we're adding from local disk.
|
|
*/
|
|
class LocalFileDetails {
|
|
public:
|
|
LocalFileDetails();
|
|
virtual ~LocalFileDetails() {}
|
|
|
|
/*
|
|
* Set the various fields, based on the pathname and characteristics
|
|
* of the file.
|
|
*/
|
|
NuError SetFields(const AddFilesDialog* pAddOpts, const WCHAR* pathname,
|
|
struct _stat* psb);
|
|
|
|
/*
|
|
* What kind of file this is. Files being added to NuFX from Windows
|
|
* can't be "BothForks" because the forks are stored in
|
|
* separate files. However, files being transferred from a NuFX
|
|
* archive, a disk image, or in from the clipboard can be both.
|
|
*
|
|
* (NOTE: this gets embedded into clipboard data. If you change
|
|
* these values, update the version number in Clipboard.cpp.)
|
|
*/
|
|
enum FileKind {
|
|
kFileKindUnknown = 0,
|
|
kFileKindDataFork,
|
|
kFileKindRsrcFork,
|
|
kFileKindDiskImage,
|
|
kFileKindBothForks,
|
|
kFileKindDirectory,
|
|
};
|
|
|
|
/*
|
|
* Provide operator= and copy constructor.
|
|
*/
|
|
LocalFileDetails& operator=(const LocalFileDetails& src) {
|
|
if (&src != this)
|
|
CopyFields(this, &src);
|
|
return *this;
|
|
}
|
|
LocalFileDetails(const LocalFileDetails& src) {
|
|
CopyFields(this, &src);
|
|
}
|
|
|
|
/*
|
|
* Returns a reference to a NuFileDetails structure with the contents
|
|
* of the LocalFileDetails.
|
|
*
|
|
* The returned structure may not be used after the LocalFileDetails
|
|
* is modified or released.
|
|
*/
|
|
const NuFileDetails& GetNuFileDetails();
|
|
|
|
/*
|
|
* Returns a reference to a NuFileDetails structure with the contents
|
|
* of the LocalFileDetails.
|
|
*
|
|
* The returned structure may not be used after the LocalFileDetails
|
|
* is modified or released.
|
|
*/
|
|
const DiskFS::CreateParms& GetCreateParms();
|
|
|
|
/*
|
|
* Returns the "local" pathname, i.e. the name of a Windows file.
|
|
*/
|
|
const CString& GetLocalPathName() const {
|
|
return fLocalPathName;
|
|
}
|
|
|
|
void SetLocalPathName(const CString& newName) {
|
|
fLocalPathName = newName;
|
|
}
|
|
|
|
/*
|
|
* This is the "local" pathname with any file type preservation
|
|
* strings removed.
|
|
*/
|
|
const CString& GetStrippedLocalPathName() const {
|
|
return fStrippedLocalPathName;
|
|
}
|
|
|
|
/*
|
|
* Sets the "stripped" local path name, and updates the MOR version.
|
|
* Does not alter the full local path name.
|
|
*/
|
|
void SetStrippedLocalPathName(const CString& newName) {
|
|
fStrippedLocalPathName = newName;
|
|
GenerateStoragePathName();
|
|
}
|
|
|
|
/*
|
|
* Returns a copy of the stripped local pathname that has been
|
|
* converted to Mac OS Roman.
|
|
*
|
|
* The returned string will be invalid if SetStrippedLocalPathName
|
|
* is subsequently called.
|
|
*/
|
|
const CStringA& GetStoragePathNameMOR() const {
|
|
return fStoragePathNameMOR;
|
|
}
|
|
|
|
FileKind GetEntryKind() const { return fEntryKind; }
|
|
void SetEntryKind(FileKind kind) { fEntryKind = kind; }
|
|
|
|
DiskImg::FSFormat GetFileSysFmt() const { return fFileSysFmt; }
|
|
void SetFileSysFmt(DiskImg::FSFormat fmt) { fFileSysFmt = fmt; }
|
|
|
|
char GetFssep() const { return fFssep; }
|
|
void SetFssep(char fssep) { fFssep = fssep; }
|
|
|
|
uint32_t GetFileType() const { return fFileType; }
|
|
void SetFileType(uint32_t type) { fFileType = type; }
|
|
|
|
uint32_t GetExtraType() const { return fExtraType; }
|
|
void SetExtraType(uint32_t type) { fExtraType = type; }
|
|
|
|
uint32_t GetAccess() const { return fAccess; }
|
|
void SetAccess(uint32_t access) { fAccess = access; }
|
|
|
|
uint16_t GetStorageType() const { return fStorageType; }
|
|
void SetStorageType(uint16_t type) { fStorageType = type; }
|
|
|
|
const NuDateTime& GetArchiveWhen() const { return fArchiveWhen; }
|
|
void SetArchiveWhen(const NuDateTime& when) { fArchiveWhen = when; }
|
|
|
|
const NuDateTime& GetModWhen() const { return fModWhen; }
|
|
void SetModWhen(const NuDateTime& when) { fModWhen = when; }
|
|
|
|
const NuDateTime& GetCreateWhen() const { return fCreateWhen; }
|
|
void SetCreateWhen(const NuDateTime& when) { fCreateWhen = when; }
|
|
|
|
private:
|
|
/*
|
|
* Copy the contents of our object to a new object, field by field,
|
|
* so the CStrings copy correctly.
|
|
*
|
|
* Useful for operator= and copy construction.
|
|
*/
|
|
static void CopyFields(LocalFileDetails* pDst,
|
|
const LocalFileDetails* pSrc);
|
|
|
|
/*
|
|
* Generates fStoragePathNameMOR from fStrippedLocalPathName.
|
|
*/
|
|
void GenerateStoragePathName();
|
|
|
|
/*
|
|
* These are the structs that the libraries want, so we provide calls
|
|
* that populate them and return a reference. These are "hairy"
|
|
* structures, so we have to place limitations on their lifetime.
|
|
*
|
|
* Ideally these would either be proper objects with destructors,
|
|
* so we could create them and not worry about who will free up the
|
|
* hairy bits, or we would omit the hairy bits from the structure and
|
|
* pass them separately.
|
|
*/
|
|
NuFileDetails fNuFileDetails;
|
|
DiskFS::CreateParms fCreateParms;
|
|
|
|
/*
|
|
* What does this entry represent?
|
|
*/
|
|
FileKind fEntryKind;
|
|
|
|
/*
|
|
* Full pathname of the Windows file.
|
|
*/
|
|
CString fLocalPathName; // was origName
|
|
|
|
/*
|
|
* "Stripped" pathname. This is the full path with any of our
|
|
* added bits removed (e.g. file type & fork identifiers).
|
|
*
|
|
* This is generated by PathProposal::LocalToArchive().
|
|
*/
|
|
CString fStrippedLocalPathName; // was storageName
|
|
|
|
/*
|
|
* The storage name, generated by converting strippedLocalPathName
|
|
* from Unicode to Mac OS Roman. This is what will be passed to
|
|
* DiskImg and NufxLib as the name of the file to create in the
|
|
* archive. For DiskImg there's an additional "normalization" pass
|
|
* make the filename suitable for use on the filesystem; that name
|
|
* is stored in a FileAddData object, not here.
|
|
*/
|
|
CStringA fStoragePathNameMOR;
|
|
|
|
DiskImg::FSFormat fFileSysFmt; // only set for xfers?
|
|
uint8_t fFssep;
|
|
uint32_t fFileType;
|
|
uint32_t fExtraType;
|
|
uint32_t fAccess;
|
|
uint16_t fStorageType; // "Unknown" or disk block size
|
|
NuDateTime fCreateWhen;
|
|
NuDateTime fModWhen;
|
|
NuDateTime fArchiveWhen;
|
|
};
|
|
|
|
// Prepare for file transfers.
|
|
virtual void XferPrepare(const XferFileOptions* pXferOpts) = 0;
|
|
|
|
// Transfer files, one at a time, into this archive from another. Called
|
|
// from XferSelection and clipboard "paste".
|
|
//
|
|
// "dataLen" and "rsrcLen" will be -1 if the corresponding fork doesn't exist.
|
|
// Returns 0 on success, nonzero on failure.
|
|
//
|
|
// On success, *pDataBuf and *pRsrcBuf are freed and set to NULL. (It's
|
|
// necessary for the interface to work this way because the NufxArchive
|
|
// version just tucks the pointers into NufxLib structures.)
|
|
virtual CString XferFile(LocalFileDetails* pDetails, uint8_t** pDataBuf,
|
|
long dataLen, uint8_t** pRsrcBuf, long rsrcLen) = 0;
|
|
|
|
// Abort progress. Not all subclasses are capable of "undo".
|
|
virtual void XferAbort(CWnd* pMsgWnd) = 0;
|
|
|
|
// Transfer is finished.
|
|
virtual void XferFinish(CWnd* pMsgWnd) = 0;
|
|
|
|
/*
|
|
* Convert from time in seconds to Apple IIgs DateTime format.
|
|
*/
|
|
static void UNIXTimeToDateTime(const time_t* pWhen, NuDateTime *pDateTime);
|
|
|
|
/*
|
|
* Reads a 16-bit big-endian value from a buffer. Does not guard
|
|
* against buffer overrun.
|
|
*/
|
|
static uint16_t Get16BE(const uint8_t* ptr) {
|
|
return ptr[1] | (ptr[0] << 8);
|
|
}
|
|
|
|
/*
|
|
* Reads a 32-bit big-endian value from a buffer. Does not guard
|
|
* against buffer overrun.
|
|
*/
|
|
static uint32_t Get32BE(const uint8_t* ptr) {
|
|
return ptr[3] | (ptr[2] << 8) | (ptr[1] << 16) | (ptr[0] << 24);
|
|
}
|
|
|
|
/*
|
|
* Reads a 16-bit little-endian value from a buffer. Does not guard
|
|
* against buffer overrun.
|
|
*/
|
|
static uint16_t Get16LE(const uint8_t* ptr) {
|
|
return ptr[0] | (ptr[1] << 8);
|
|
}
|
|
|
|
/*
|
|
* Reads a 32-bit little-endian value from a buffer. Does not guard
|
|
* against buffer overrun.
|
|
*/
|
|
static uint32_t Get32LE(const uint8_t* ptr) {
|
|
return ptr[0] | (ptr[1] << 8) | (ptr[2] << 16) | (ptr[3] << 24);
|
|
}
|
|
|
|
protected:
|
|
/*
|
|
* Delete the "entries" list.
|
|
*/
|
|
virtual void DeleteEntries();
|
|
|
|
void ReplaceFssep(WCHAR* str, char oldc, char newc, char newSubst);
|
|
|
|
/*
|
|
* Prepare a directory for reading.
|
|
*
|
|
* Allocates a Win32dirent struct that must be freed by the caller.
|
|
*/
|
|
Win32dirent* OpenDir(const WCHAR* name);
|
|
|
|
/*
|
|
* Get an entry from an open directory.
|
|
*
|
|
* Returns a NULL pointer after the last entry has been read.
|
|
*/
|
|
Win32dirent* ReadDir(Win32dirent* dir);
|
|
|
|
/*
|
|
* Close a directory.
|
|
*/
|
|
void CloseDir(Win32dirent* dir);
|
|
|
|
/*
|
|
* Win32 recursive directory descent. Scan the contents of a directory.
|
|
* If a subdirectory is found, follow it; otherwise, call Win32AddFile to
|
|
* add the file.
|
|
*/
|
|
NuError Win32AddDirectory(const AddFilesDialog* pAddOpts,
|
|
const WCHAR* dirName, CString* pErrMsg);
|
|
|
|
/*
|
|
* Add a file to the list we're adding to the archive. If it's a directory,
|
|
* and the recursive descent feature is enabled, call Win32AddDirectory to
|
|
* add the contents of the dir.
|
|
*
|
|
* Returns with an error if the file doesn't exist or isn't readable.
|
|
*/
|
|
NuError Win32AddFile(const AddFilesDialog* pAddOpts,
|
|
const WCHAR* pathname, CString* pErrMsg);
|
|
|
|
/*
|
|
* External entry point; just calls the system-specific version.
|
|
*/
|
|
NuError AddFile(const AddFilesDialog* pAddOpts, const WCHAR* pathname,
|
|
CString* pErrMsg);
|
|
|
|
/*
|
|
* Each implementation must provide this. It's called from the generic
|
|
* AddFile function with the high-level add options, a partial pathname,
|
|
* and a FileDetails structure filled in using Win32 calls.
|
|
*
|
|
* One call to AddFile can result in multiple calls to DoAddFile if
|
|
* the subject of the AddFile call is a directory (and fIncludeSubdirs
|
|
* is set).
|
|
*
|
|
* DoAddFile is not called for subdirectories. The underlying code must
|
|
* create directories as needed.
|
|
*
|
|
* In some cases (such as renaming a file as it is being added) the
|
|
* information in "*pDetails" may be modified.
|
|
*/
|
|
virtual NuError DoAddFile(const AddFilesDialog* pAddOpts,
|
|
LocalFileDetails* pDetails) = 0;
|
|
|
|
void SetPathName(const WCHAR* pathName) {
|
|
free(fPathName);
|
|
if (pathName != NULL) {
|
|
fPathName = _wcsdup(pathName);
|
|
} else {
|
|
fPathName = NULL;
|
|
}
|
|
}
|
|
|
|
bool fReloadFlag; // set after Reload called
|
|
|
|
private:
|
|
//virtual void CreateIndex();
|
|
|
|
//CString fNewPathHolder;
|
|
//CString fOrigPathHolder;
|
|
|
|
WCHAR* fPathName;
|
|
long fNumEntries;
|
|
GenericEntry* fEntryHead;
|
|
GenericEntry* fEntryTail;
|
|
//GenericEntry** fEntryIndex;
|
|
};
|
|
|
|
/*
|
|
* One entry in a SelectionSet.
|
|
*/
|
|
class SelectionEntry {
|
|
public:
|
|
SelectionEntry(GenericEntry* pEntry) {
|
|
fpEntry = pEntry;
|
|
//fThreadKind = threadKind;
|
|
//fFilter = filter;
|
|
//fReformatName = "";
|
|
fpPrev = fpNext = NULL;
|
|
}
|
|
~SelectionEntry() {}
|
|
|
|
int Reformat(ReformatHolder* pHolder);
|
|
|
|
GenericEntry* GetEntry() const { return fpEntry; }
|
|
//int GetThreadKind() const { return fThreadKind; }
|
|
//int GetFilter() const { return fFilter; }
|
|
//const char* GetReformatName() const { return fReformatName; }
|
|
|
|
SelectionEntry* GetPrev() const { return fpPrev; }
|
|
void SetPrev(SelectionEntry* pPrev) { fpPrev = pPrev; }
|
|
SelectionEntry* GetNext() const { return fpNext; }
|
|
void SetNext(SelectionEntry* pNext) { fpNext = pNext; }
|
|
|
|
private:
|
|
GenericEntry* fpEntry;
|
|
//int fThreadKind; // data, rsrc, etc (threadMask)
|
|
//int fFilter; // fAllowedFilters, really
|
|
//const char* fReformatName; // name of formatting actually applied
|
|
|
|
SelectionEntry* fpPrev;
|
|
SelectionEntry* fpNext;
|
|
};
|
|
|
|
class ContentList;
|
|
|
|
/*
|
|
* A set of selected files.
|
|
*
|
|
* Each entry represents one item that can be displayed, such as a data
|
|
* fork, resource fork, or comment thread. Thus, a single file may have
|
|
* multiple entries in the set.
|
|
*/
|
|
class SelectionSet {
|
|
public:
|
|
SelectionSet() {
|
|
fNumEntries = 0;
|
|
fEntryHead = fEntryTail = fIterCurrent = NULL;
|
|
}
|
|
~SelectionSet() {
|
|
DeleteEntries();
|
|
}
|
|
|
|
// create the set from the selected members of a ContentList
|
|
void CreateFromSelection(ContentList* pContentList, int threadMask);
|
|
// create the set from all members of a ContentList
|
|
void CreateFromAll(ContentList* pContentList, int threadMask);
|
|
|
|
// get the head of the list
|
|
SelectionEntry* GetEntries() const { return fEntryHead; }
|
|
|
|
void IterReset() {
|
|
fIterCurrent = NULL;
|
|
}
|
|
// move to the next or previous entry as part of iterating
|
|
SelectionEntry* IterPrev() {
|
|
if (fIterCurrent == NULL)
|
|
fIterCurrent = fEntryTail;
|
|
else
|
|
fIterCurrent = fIterCurrent->GetPrev();
|
|
return fIterCurrent;
|
|
}
|
|
SelectionEntry* IterNext() {
|
|
if (fIterCurrent == NULL)
|
|
fIterCurrent = fEntryHead;
|
|
else
|
|
fIterCurrent = fIterCurrent->GetNext();
|
|
return fIterCurrent;
|
|
}
|
|
SelectionEntry* IterCurrent() {
|
|
return fIterCurrent;
|
|
}
|
|
bool IterHasPrev() const {
|
|
if (fIterCurrent == NULL)
|
|
return fEntryTail != NULL;
|
|
else
|
|
return (fIterCurrent->GetPrev() != NULL);
|
|
}
|
|
bool IterHasNext() const {
|
|
if (fIterCurrent == NULL)
|
|
return fEntryHead != NULL;
|
|
else
|
|
return (fIterCurrent->GetNext() != NULL);
|
|
}
|
|
|
|
int GetNumEntries() const { return fNumEntries; }
|
|
|
|
// count the #of entries whose display name matches "prefix"
|
|
int CountMatchingPrefix(const WCHAR* prefix);
|
|
|
|
// debug dump the contents of the selection set
|
|
void Dump();
|
|
|
|
private:
|
|
/*
|
|
* Add a GenericEntry to the set, but only if we can find a thread that
|
|
* matches the flags in "threadMask".
|
|
*/
|
|
void AddToSet(GenericEntry* pEntry, int threadMask);
|
|
|
|
/*
|
|
* Add a new entry to the end of the list.
|
|
*/
|
|
void AddEntry(SelectionEntry* pEntry);
|
|
|
|
/*
|
|
* Delete the "entries" list.
|
|
*/
|
|
void DeleteEntries();
|
|
|
|
int fNumEntries;
|
|
SelectionEntry* fIterCurrent;
|
|
|
|
SelectionEntry* fEntryHead;
|
|
SelectionEntry* fEntryTail;
|
|
};
|
|
|
|
#endif /*APP_GENERICARCHIVE_H*/
|