/* Copyright (c) 2017, Computer History Museum All rights reserved. Redistribution and use in source and binary forms, with or without modification, are permitted (subject to the limitations in the disclaimer below) provided that the following conditions are met: * Redistributions of source code must retain the above copyright notice, this list of conditions and the following disclaimer. * Redistributions in binary form must reproduce the above copyright notice, this list of conditions and the following disclaimer in the documentation and/or other materials provided with the distribution. * Neither the name of Computer History Museum nor the names of its contributors may be used to endorse or promote products derived from this software without specific prior written permission. NO EXPRESS OR IMPLIED LICENSES TO ANY PARTY'S PATENT RIGHTS ARE GRANTED BY THIS LICENSE. THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */ #include "export.h" #define FILE_NUM 27 OSErr ExportMailFromMenu(MenuHandle mh,FSSpecPtr mailFolderSpec); OSErr ExportMailboxFromItem(MenuHandle mh,short item, FSSpecPtr mailFolderSpec); OSErr ExportMailFolderFromItem(MenuHandle mh,short item, FSSpecPtr mailFolderSpec); OSErr ExportEnvelope(TOCHandle tocH,short sumNum,short refN); OSErr ExportMozillaHeaders(TOCHandle tocH,short sumNum,short refN); /************************************************************************ * DoFileExport - export Eudora's data to a folder of the user's choosing. ************************************************************************/ void DoFileExport(void) { FSSpec exportFolder; Str63 folderName; Str63 volName; OSErr err = FindFolder(kOnSystemDisk,kDesktopFolderType,False,&exportFolder.vRefNum,&exportFolder.parID); if (err) return; // Bail. Should warn user that the system can't find the desktop, but geez... { if (GetFolder(volName,&exportFolder.vRefNum,&exportFolder.parID,true,true,true,true)) { // Get the name of the folder chosen GetDirName(volName,exportFolder.vRefNum,exportFolder.parID,folderName); // If the user has chosen a folder with the magic name, use that. if (EqualStrRes(folderName,EXPORT_FOLDER_NAME)) { // Pop up a level and then fill in the filename if (!(err = ParentSpec(&exportFolder,&exportFolder))) PCopy(exportFolder.name,folderName); else { FileSystemError(EXPORT_MAIL_ERR,folderName,err); return; } } // If the user didn't choose a magically named folder, make the magically named folder else FSMakeFSSpec(exportFolder.vRefNum,exportFolder.parID,GetRString(folderName,EXPORT_FOLDER_NAME),&exportFolder); OpenProgress(); ExportTo(&exportFolder); CloseProgress(); } } } /************************************************************************ * ExportTo - export Eudora's data to a specified folder * Exports all data by calling the various sub-exporters ************************************************************************/ OSErr ExportTo(FSSpecPtr exportFolderSpec) { OSErr err = noErr; long parID; FSSpec subFolderSpec; ProgressMessageR(kpTitle,EXPORTING_MAIL); // Check to see that we either have or can create a folder if (!FSpIsItAFolder(exportFolderSpec)) { err = FSpDirCreate(exportFolderSpec,smSystemScript,&parID); if (err) return FileSystemError(EXPORT_MAIL_ERR,exportFolderSpec->name,err); } // Now that we've verified the folder exists, descend into it for the various // exporters { short folderIDs[] = { EXPORT_MAIL_FOLDER, 0 }; ExporterFunc *funcs[] = { ExportMailTo, 0 }; short i; Accumulator errors; // Record errors, don't present them AccuInit(&errors); ExportErrors = &errors; for (i=0; !err && folderIDs[i] && !CommandPeriod ;i++) { // Point at the appropriate subfolder subFolderSpec = *exportFolderSpec; subFolderSpec.parID = SpecDirId(exportFolderSpec); GetRString(subFolderSpec.name,EXPORT_MAIL_FOLDER); // Delete what's there already if ((err = FSpTrash(&subFolderSpec))) { if (err == fnfErr) err = noErr; // ok if it's not there else return (FileSystemError(EXPORT_MAIL_ERR,subFolderSpec.name,err)); } // Create the subfolder if ((err= FSpDirCreate(&subFolderSpec,smSystemScript,&parID))) return (FileSystemError(EXPORT_MAIL_ERR,subFolderSpec.name,err)); // Descend into it... subFolderSpec.parID = parID; *subFolderSpec.name = 0; // Now, call the exporter err = funcs[i](&subFolderSpec); } AccuZap(errors); ExportErrors = nil; } return err; } /************************************************************************ * ExportMailTo - Export mail to a particular folder ************************************************************************/ OSErr ExportMailTo(FSSpecPtr mailFolderSpec) { MenuHandle mh = GetMHandle(MAILBOX_MENU); return ExportMailFromMenu(mh,mailFolderSpec); } /************************************************************************ * ExportMailFrom - Export mail from a particular menu, including submenus ************************************************************************/ OSErr ExportMailFromMenu(MenuHandle mh,FSSpecPtr mailFolderSpec) { short item; Boolean topLevel = GetMenuID(mh)==MAILBOX_MENU; OSErr err = noErr; short nItems = CountMenuItems(mh); // Do In/Out/Junk/Trash if (topLevel) { for (item=1;itemcount) ByteProgress(nil,-1,(*tocH)->count); for (i=0;i<(*tocH)->count && !CommandPeriod;i++) { MyWindowPtr win; UHandle text; Str15 oldNewLine; PCopy(oldNewLine,NewLine); PCopy(NewLine,Lf); openedMessage = !(*tocH)->sums[i].messH; win = GetAMessage(tocH,i,nil,nil,false); if (!win || !(*tocH)->sums[i].messH || (PETEGetRawText(PETE,(*(*tocH)->sums[i].messH)->bodyPTE,&text),!text)) ; // not sure what's useful to do here... else { ExportEnvelope(tocH,i,refN); ExportMozillaHeaders(tocH,i,refN); err = SpoolMessage((*tocH)->sums[i].messH,&exportSpec,refN); } if (openedMessage) { CloseMyWindow(GetMyWindowWindowPtr(win)); ZapHandle((*tocH)->sums[i].cache); } ByteProgress(nil,1,0); PCopy(NewLine,oldNewLine); } FSClose(refN); if (openedBox) CloseMyWindow((*tocH)->win); return noErr; } /************************************************************************ * ExportEnvelope - write an envelope for a message to the export file ************************************************************************/ OSErr ExportEnvelope(TOCHandle tocH,short sumNum,short refN) { OSErr err = noErr; long count = 1; UHandle cache; Str63 envelope; CacheMessage(tocH,sumNum); cache = (*tocH)->sums[sumNum].cache; if (cache && *cache) { UPtr spot; count = GetHandleSize(cache); MakePStr(envelope,*cache,count); spot = PIndex(envelope,Cr[1]); if (spot) { *envelope = spot - envelope - 1; if (sumNum>0) PInsert(envelope,sizeof(envelope)-1,NewLine,envelope+1); PCat(envelope,NewLine); count = *envelope; err = AWrite(refN,&count,envelope+1); } } return err; } /************************************************************************ * ExportMozillaHeaders - add Mozilla-specific headers to an export file ************************************************************************/ OSErr ExportMozillaHeaders(TOCHandle tocH,short sumNum,short refN) { OSErr err = noErr; Str63 flagStr; long flags; Accumulator keywords; // first set of flags flags = 0; // status switch ((*tocH)->sums[sumNum].state) { case UNREAD: break; // zero works here case READ: flags |= MOZ_MSG_FLAG_READ; break; case REPLIED: flags |= MOZ_MSG_FLAG_REPLIED; break; case QUEUED: flags |= MOZ_MSG_FLAG_FORWARDED; break; case FORWARDED: flags |= MOZ_MSG_FLAG_FORWARDED; break; case BUSY_SENDING: break; // irrelevant case SENT: case UNSENT: case TIMED: case REDIST: case UNSENDABLE: case SENDABLE: case MESG_ERR: case REBUILT: // none of these are currently supported by mozilla ; } // Mozilla stores priority high up in the flags if (Prior2Display((*tocH)->sums[sumNum].priority) != 3) flags |= (1L+Prior2Display((*tocH)->sums[sumNum].priority)) << 13; // write out the flag if it's nonzero if (flags) err = AWriteP(refN,ComposeRString(flagStr,MOZILLA_FLAGS_FMT,flags,NewLine)); if (err) return err; // Second set of flags flags = 0; // We record the a receipt is requested, and then clear the record. // Mozilla rediscovers the request when the message is opened, and remembers once // you've sent the receipt. So, if our receipt request flag is NOT set, // we will set mozilla's receipt-sent flag. // It may be wise to only do this if there is an actual receipt request inside the // message, but I'll leave that for another day. if (!SumOptIsSet(tocH,sumNum,OPT_RECEIPT)) flags |= MOZ_MSG_FLAG_MDN_REPORT_SENT; // write out the flag if it's nonzero if (flags) err = AWriteP(refN,ComposeRString(flagStr,MOZILLA_FLAGS2_FMT,flags,NewLine)); // Write label out as a keyword // We record the a receipt is requested, and then clear the record. // Mozilla rediscovers the request when the message is opened, and remembers once // you've sent the receipt. So, if our receipt request flag is NOT set, // we will set mozilla's receipt-sent flag. // It may be wise to only do this if there is an actual receipt request inside the // message, but I'll leave that for another day. if (!SumOptIsSet(tocH,sumNum,OPT_RECEIPT)) flags |= MOZ_MSG_FLAG_MDN_REPORT_SENT; // write out the flag if it's nonzero if (flags) err = AWriteP(refN,ComposeRString(flagStr,MOZILLA_FLAGS2_FMT,flags,NewLine)); return err; } OSErr ExportAddressBookTo(FSSpecPtr addressBookFolderSpec) { return unimpErr; } OSErr ExportFiltersTo(FSSpecPtr filterFileSpec) { return unimpErr; } OSErr ExportStationeryTo(FSSpecPtr stationeryFolderSpec) { return unimpErr; }