eudora-mac/stationerywin.c

1 line
21 KiB
C
Executable File

/* 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 "stationerywin.h"
#define FILE_NUM 88
/* Copyright (c) 1996 by QUALCOMM Incorporated */
#pragma segment StationeryWin
// Stationery Window controls
enum
{
kctlNew=0,
kctlEdit,
kctlRemove
};
typedef struct
{
ViewList list;
MyWindowPtr win;
ControlHandle ctlEdit,ctlNew,ctlRemove;
Boolean inited;
Boolean dontOpenAfterAll; // Don't open; set when deleting stationery that may
// just have been created
} WinData;
static WinData gWin;
/************************************************************************
* prototypes
************************************************************************/
static void DoDidResize(MyWindowPtr win, Rect *oldContR);
static void DoZoomSize(MyWindowPtr win,Rect *zoom);
static void DoUpdate(MyWindowPtr win);
static Boolean DoClose(MyWindowPtr win);
static void DoClick(MyWindowPtr win,EventRecord *event);
static void DoCursor(Point mouse);
static void DoActivate(MyWindowPtr win);
static Boolean DoKey(MyWindowPtr win, EventRecord *event);
static void DoShowHelp(MyWindowPtr win,Point mouse);
static Boolean DoMenuSelect(MyWindowPtr win, int menu, int item, short modifiers);
static long ViewListCallBack(ViewListPtr pView, VLCallbackMessage message, long data);
static void EditStationery(void);
static void SetControls(void);
static void DeleteStationery(void);
static Boolean GetListData(VLNodeInfo *data,short selectedItem);
static void GetStationerySpec(FSSpecPtr pSpec, StringPtr name);
static MyWindowPtr NewStationeryWindow(FSSpecPtr spec);
static void NewStationeryFile(void);
static OSErr DoDragHandler(MyWindowPtr win,DragTrackingMessage which,DragReference drag);
static void DoGrow(MyWindowPtr win,Point *newSize);
static void CloseStationeryWindow(FSSpecPtr spec);
static Boolean StnyFind(MyWindowPtr win,PStr what);
/************************************************************************
* OpenStationeryWin - open the stationery window
************************************************************************/
void OpenStationeryWin(void)
{
WindowPtr gWinWinWP;
if (!HasFeature (featureStationery))
return;
if (SelectOpenWazoo(STA_WIN)) return; // Already opened in a wazoo
if (!gWin.inited)
{
short err=0;
Rect r;
if (!(gWin.win=GetNewMyWindow(STATIONERY_WIND,nil,nil,BehindModal,false,false,STA_WIN)))
{err=MemError(); goto fail;}
gWinWinWP = GetMyWindowWindowPtr (gWin.win);
SetWinMinSize(gWin.win,-1,-1);
SetPort_(GetWindowPort(gWinWinWP));
ConfigFontSetup(gWin.win);
MySetThemeWindowBackground(gWin.win,kThemeListViewBackgroundBrush,False);
// list
SetRect(&r,0,0,20,20);
if (LVNew(&gWin.list,gWin.win,&r,1,ViewListCallBack,kStationeryDragType))
{err=MemError(); goto fail;}
// controls
if (!(gWin.ctlNew = NewIconButton(STNY_NEW_CNTL,gWinWinWP)) ||
!(gWin.ctlRemove = NewIconButton(STNY_REMOVE_CNTL,gWinWinWP)) ||
!(gWin.ctlEdit = NewIconButton(STNY_EDIT_CNTL,gWinWinWP)))
goto fail;
gWin.win->didResize = DoDidResize;
gWin.win->close = DoClose;
gWin.win->update = DoUpdate;
gWin.win->position = PositionPrefsTitle;
gWin.win->click = DoClick;
gWin.win->bgClick = DoClick;
gWin.win->dontControl = true;
gWin.win->cursor = DoCursor;
gWin.win->activate = DoActivate;
gWin.win->help = DoShowHelp;
gWin.win->menu = DoMenuSelect;
gWin.win->key = DoKey;
gWin.win->app1 = DoKey;
gWin.win->zoomSize = DoZoomSize;
gWin.win->drag = DoDragHandler;
gWin.win->grow = DoGrow;
gWin.win->find = StnyFind;
gWin.win->showsSponsorAd = true;
ShowMyWindow(gWinWinWP);
MyWindowDidResize(gWin.win,&gWin.win->contR);
gWin.inited = true;
return;
fail:
if (gWin.win) CloseMyWindow(GetMyWindowWindowPtr(gWin.win));
if (err) WarnUser(COULDNT_WIN,err);
return;
}
UserSelectWindow(GetMyWindowWindowPtr(gWin.win));
}
/************************************************************************
* DoDidResize - resize the window
************************************************************************/
static void DoDidResize(MyWindowPtr win, Rect *oldContR)
{
#define kListInset 10
#pragma unused(oldContR)
Rect r;
short htAdjustment;
ControlHandle ctlList[3];
// buttons
ctlList[0] = gWin.ctlNew;
ctlList[1] = gWin.ctlRemove;
ctlList[2] = gWin.ctlEdit;
PositionBevelButtons(win,3,ctlList,kListInset,gWin.win->contR.bottom-INSET-kHtCtl,kHtCtl,RectWi(win->contR));
// list
SetRect(&r,kListInset,win->topMargin+kListInset,gWin.win->contR.right-kListInset,gWin.win->contR.bottom - 2*INSET - kHtCtl);
if (gWin.win->sponsorAdExists && r.bottom >= gWin.win->sponsorAdRect.top - kSponsorBorderMargin)
r.bottom = gWin.win->sponsorAdRect.top - kSponsorBorderMargin;
LVSize(&gWin.list,&r,&htAdjustment);
// enable/disble controls
SetControls();
// redraw
InvalContent(win);
}
/************************************************************************
* DoZoomSize - zoom to only the maximum size of list
************************************************************************/
static void DoZoomSize(MyWindowPtr win,Rect *zoom)
{
short zoomHi = zoom->bottom-zoom->top;
short zoomWi = zoom->right-zoom->left;
short hi, wi;
LVMaxSize(&gWin.list, &wi, &hi);
wi += 2*kListInset;
hi += 2*kListInset + INSET + kHtCtl;
wi = MIN(wi,zoomWi); wi = MAX(wi,win->minSize.h);
hi = MIN(hi,zoomHi); hi = MAX(hi,win->minSize.v);
zoom->right = zoom->left+wi;
zoom->bottom = zoom->top+hi;
}
/************************************************************************
* DoGrow - adjust grow size
************************************************************************/
static void DoGrow(MyWindowPtr win,Point *newSize)
{
Rect r;
short htControl = ControlHi(gWin.ctlNew);
short bottomMargin,sponsorMargin;
// Get list position
bottomMargin = INSET*2 + htControl;
if (win->sponsorAdExists)
{
CGrafPtr winPort = GetMyWindowCGrafPtr(win);
Rect rPort;
GetPortBounds(winPort,&rPort);
sponsorMargin = rPort.bottom - win->sponsorAdRect.top + kSponsorBorderMargin;
if (sponsorMargin > bottomMargin) bottomMargin = sponsorMargin;
}
SetRect(&r,kListInset,win->topMargin+kListInset,newSize->h-kListInset,newSize->v - bottomMargin);
LVCalcSize(&gWin.list,&r);
// Calculate new window height
newSize->v = r.bottom + bottomMargin;
}
/************************************************************************
* DoClose - close the window
************************************************************************/
static Boolean DoClose(MyWindowPtr win)
{
#pragma unused(win)
if (gWin.inited)
{
// Dispose of list
LVDispose(&gWin.list);
gWin.inited = false;
}
return(True);
}
/************************************************************************
* DoUpdate - draw the window
************************************************************************/
static void DoUpdate(MyWindowPtr win)
{
CGrafPtr winPort = GetMyWindowCGrafPtr(win);
Rect r;
r = gWin.list.bounds;
DrawThemeListBoxFrame(&r,kThemeStateActive);
LVDraw(&gWin.list,nil, true, false);
}
/************************************************************************
* DoActivate - activate the window
************************************************************************/
static void DoActivate(MyWindowPtr win)
{
#pragma unused(win)
LVActivate(&gWin.list, gWin.win->isActive);
}
/************************************************************************
* DoKey - key stroke
************************************************************************/
static Boolean DoKey(MyWindowPtr win, EventRecord *event)
{
#pragma unused(win)
short key = (event->message & 0xff);
Boolean fResult;
fResult = LVKey(&gWin.list,event);
SetControls();
return fResult;
}
/************************************************************************
* DoClick - click in window
************************************************************************/
void DoClick(MyWindowPtr win,EventRecord *event)
{
WindowPtr winWP = GetMyWindowWindowPtr (win);
Point pt;
ControlHandle hCtl;
SetPort(GetWindowPort(winWP));
pt = event->where;
GlobalToLocal(&pt);
if (PtInControl(pt,gWin.ctlRemove)) gWin.dontOpenAfterAll = True; // don't open file before we delete it
if (!LVClick(&gWin.list,event))
{
if (!win->isActive)
{
SelectWindow_(winWP);
UpdateMyWindow(winWP); // Have to update manually since no events are processed
}
if (FindControl(pt, winWP, &hCtl))
{
if (TrackControl(hCtl,pt,(void *)(-1)))
{
long controlId;
if (hCtl == gWin.ctlEdit && !DateWarning(true))
{
EditStationery();
controlId = kctlEdit;
}
else if (hCtl == gWin.ctlNew && !DateWarning(true))
{
NewStationeryFile();
controlId = kctlNew;
}
else if (hCtl == gWin.ctlRemove)
{
DeleteStationery();
controlId = kctlRemove;
}
else
{
ASSERT(0);
}
AuditHit((event->modifiers&shiftKey)!=0, (event->modifiers&controlKey)!=0, (event->modifiers&optionKey)!=0, (event->modifiers&cmdKey)!=0, false, GetWindowKind(winWP), AUDITCONTROLID(GetWindowKind(winWP),controlId), event->what);
}
}
}
gWin.dontOpenAfterAll = False;
SetControls();
}
/**********************************************************************
* StnyFind - find in the window
**********************************************************************/
static Boolean StnyFind(MyWindowPtr win,PStr what)
{
return FindListView(win,&gWin.list,what);
}
/************************************************************************
* DoCursor - set the cursor properly for the window
************************************************************************/
static void DoCursor(Point mouse)
{
if (!PeteCursorList(gWin.win->pteList,mouse))
SetMyCursor(arrowCursor);
}
/************************************************************************
* DoShowHelp - provide help for the window
************************************************************************/
static void DoShowHelp(MyWindowPtr win,Point mouse)
{
if (PtInRect(mouse,&gWin.list.bounds))
MyBalloon(&gWin.list.bounds,100,0,STNRY_HELP_STRN+1,0,nil);
else
ShowControlHelp(mouse,STNRY_HELP_STRN+2,gWin.ctlNew,gWin.ctlRemove,gWin.ctlEdit,nil);
}
/************************************************************************
* DoMenuSelect - menu choice in the window
************************************************************************/
static Boolean DoMenuSelect(MyWindowPtr win, int menu, int item, short modifiers)
{
#pragma unused(win,modifiers)
switch (menu)
{
case FILE_MENU:
switch(item)
{
case FILE_OPENSEL_ITEM:
EditStationery();
return(True);
break;
}
break;
case EDIT_MENU:
switch(item)
{
case EDIT_SELECT_ITEM:
if (LVSelectAll(&gWin.list))
{
SetControls();
return(true);
}
break;
case EDIT_COPY_ITEM:
LVCopy(&gWin.list);
return true;
}
break;
}
return(False);
}
/************************************************************************
* GetListData - get data for selected item
************************************************************************/
static Boolean GetListData(VLNodeInfo *data,short selectedItem)
{
return LVGetItem(&gWin.list,selectedItem,data,true);
}
/************************************************************************
* SetControls - enable or disable the controls, based on current situation
************************************************************************/
static void SetControls(void)
{
Boolean fSelect;
fSelect = LVCountSelection(&gWin.list)!=0;
gWin.win->hasSelection = fSelect;
// Determine if "Remove" and "Edit" button needs to be greyed
SetGreyControl(gWin.ctlRemove,!fSelect);
SetGreyControl(gWin.ctlEdit,!fSelect);
}
/**********************************************************************
* DeleteStationery - delete stationery file(s)
**********************************************************************/
static void DeleteStationery(void)
{
FSSpec spec;
short i;
VLNodeInfo info;
Handle textH;
MSumType sum;
for (i=1;i<=LVCountSelection(&gWin.list);i++)
{
GetListData(&info,i);
GetStationerySpec(&spec, info.name);
CloseStationeryWindow(&spec);
// check for spool folder
if (!Snarf(&spec,&textH,0))
{
if (!GetStationerySum(textH,&sum))
RemSpoolFolder(sum.uidHash);
ZapHandle(textH);
}
// move stationery to trash
FSpTrash(&spec);
}
BuildStationeryList();
}
/**********************************************************************
* CloseStationeryWindow - find and close any Stationery window associated with the file
**********************************************************************/
static void CloseStationeryWindow(FSSpecPtr spec)
{
WindowPtr winWP;
MyWindowPtr win;
for (winWP=FrontWindow_();winWP;winWP=GetNextWindow(winWP))
{
if (GetWindowKind(winWP) == COMP_WIN)
{
MessHandle messH;
win = GetWindowMyWindowPtr(winWP);
messH = Win2MessH(win);
if ((*messH)->hStationerySpec)
{
FSSpec winSpec = **(*messH)->hStationerySpec;
if (SameSpec(spec,&winSpec))
{
// Close the window. Don't save changes
win->isDirty = False;
PeteCleanList(win->pte);
CloseMyWindow(winWP);
break;
}
}
}
}
}
/************************************************************************
* ViewListCallBack - callback function for List View
************************************************************************/
static long ViewListCallBack(ViewListPtr pView, VLCallbackMessage message, long data)
{
VLNodeInfo info,*pInfo;
OSErr err = noErr;
short i,count;
FSSpec spec;
MenuHandle mh;
SendDragDataInfo *pSendData;
Boolean anyStationery = false;
switch (message)
{
case kLVAddNodeItems:
// Add stationery names to list
mh = GetMHandle(NEW_WITH_HIER_MENU); // Get names from stationery menu
count = CountMenuItems(mh);
if (count>1 || IsMenuItemEnabled(mh,1))
for (i=1;i<=count;i++)
{
Zero(info);
info.useLevelZero = true;
info.iconID = STATIONERY_ICON;
MyGetItem(mh, i, info.name);
LVAdd(pView, &info);
anyStationery = true;
}
if (anyStationery)
UseFeature (featureStationery);
break;
case kLVOpenItem:
if (!gWin.dontOpenAfterAll)
{
if (CurrentModifiers()&optionKey)
// Edit if option key down
EditStationery();
else
{
// New message with stationery
for (i=1;i<=LVCountSelection(&gWin.list);i++)
{
GetListData(&info,i);
NewMessageWith(info.rowNum);
}
}
}
break;
case kLVRenameItem:
{
Str31 newName;
SanitizeFN(newName,(StringPtr)data,MAC_FN_BAD,MAC_FN_REP,false);
GetListData(&info,1);
GetStationerySpec(&spec, info.name);
if (err = HRename(spec.vRefNum,spec.parID,spec.name,newName))
FileSystemError(CANT_RENAME_ERR,spec.name,err);
BuildStationeryList();
break;
}
case kLVQueryItem:
pInfo = ( VLNodeInfo *)data;
switch (pInfo->query)
{
case kQuerySelect:
case kQueryRename:
case kQueryDrop:
case kQueryDrag:
case kQueryDCOpens:
return true;
}
return false;
break;
case kLVDeleteItem:
DeleteStationery();
break;
case kLVSendDragData:
pSendData = (SendDragDataInfo *)data;
GetStationerySpec(&spec,pSendData->info->name);
err = SetDragItemFlavorData(pSendData->drag, pSendData->itemRef, pSendData->flavor,&spec, sizeof(spec), 0L);
break;
}
return err;
}
/************************************************************************
* GetStationerySpec - get the file spec for a stationery file
************************************************************************/
static void GetStationerySpec(FSSpecPtr pSpec, StringPtr name)
{
SubFolderSpec(STATION_FOLDER,pSpec);
PCopy(pSpec->name,name);
}
/************************************************************************
* EditStationery - edit the selected stationery
************************************************************************/
static void EditStationery(void)
{
// Open selected stationery
VLNodeInfo info;
short i;
for (i=1;i<=LVCountSelection(&gWin.list);i++)
{
MyWindowPtr newWin;
FSSpec spec;
GetListData(&info,i);
GetStationerySpec(&spec, info.name);
if (newWin = NewStationeryWindow(&spec))
ShowMyWindow(GetMyWindowWindowPtr(newWin));
}
}
/************************************************************************
* BuildStationeryList - rebuild the entire stationery list
************************************************************************/
void BuildStationeryList(void)
{
BuildStationMenu();
if (gWin.inited)
InvalidListView(&gWin.list); // Regenerate list
}
/**********************************************************************
* NewStationeryWindow - create a new stationery window
**********************************************************************/
static MyWindowPtr NewStationeryWindow(FSSpecPtr spec)
{
MyWindowPtr newWin;
if (!HasFeature (featureStationery))
return (nil);
if (newWin=DoComposeNew(0))
{
MessHandle messH =(MessHandle)GetMyWindowPrivateData(newWin);
if (spec)
{
FSSpecHandle hSpec;
if (hSpec = NuHandle(sizeof(FSSpec)))
{
BMD(spec,*hSpec,sizeof(FSSpec));
(*messH)->hStationerySpec = hSpec;
}
ApplyStationeryLo(newWin,spec,True,True,true);
}
else
ApplyDefaultStationery(newWin,True,True);
UpdateSum(messH,SumOf(messH)->offset,SumOf(messH)->length);
}
return newWin;
}
/**********************************************************************
* NewStationeryFile - create a new stationery file
**********************************************************************/
static void NewStationeryFile(void)
{
MyWindowPtr newWin;
FSSpec spec,folderSpec;
OSErr err;
if (!HasFeature (featureStationery))
return;
UseFeature (featureStationery);
SubFolderSpec(STATION_FOLDER,&folderSpec);
// Make a unique "untitled" name
MakeUniqueUntitledSpec (folderSpec.vRefNum, folderSpec.parID, UNTITLED, &spec);
if (err=FSpCreate(&spec,CREATOR,STATIONERY_TYPE,smSystemScript))
{
FileSystemError(CREATE_STA,&spec.name,err);
return;
}
// Write default info to file
if (newWin = NewStationeryWindow(nil))
{
MessHandle messH =(MessHandle)GetMyWindowPrivateData(newWin);
short refN;
if (!FSpOpenDF(&spec,fsRdWrPerm,&refN))
{
Stationery = True;
err = SaveAsToOpenFile(refN,messH);
Stationery = false;
MyFSClose(refN);
}
CloseMyWindow(GetMyWindowWindowPtr(newWin));
BuildStationeryList();
SelectWindow_(GetMyWindowWindowPtr(gWin.win));
LVRename(&gWin.list,0,spec.name,false,false); // Allow user to rename the stationery file
}
}
/**********************************************************************
* DoDragHandler - handle drags
**********************************************************************/
static OSErr DoDragHandler(MyWindowPtr win,DragTrackingMessage which,DragReference drag)
{
#pragma unused(win)
OSErr err = noErr;
VLNodeInfo targetInfo;
if (!DragIsInteresting(drag,MESS_FLAVOR,TOC_FLAVOR,nil))
return(dragNotAcceptedErr); // Nothing here we want
switch (which)
{
case kDragTrackingEnterWindow:
case kDragTrackingLeaveWindow:
case kDragTrackingInWindow:
err = LVDrag(&gWin.list,which,drag);
break;
case 0xfff:
// Drop
if (LVDrop(&gWin.list,&targetInfo))
{
TOCHandle tocH;
UHandle data=nil;
if (!(err=MyGetDragItemData(drag,1,MESS_FLAVOR,(void*)&data)))
{
Boolean all, quote, self;
MessHandle messH;
messH = **(MessHandle**)data;
tocH = (*messH)->tocH;
ReplyDefaults(0,&all,&self,&quote);
DoReplyMessage((*messH)->win,all,self,quote,True,targetInfo.rowNum,True,True,True);
}
else if (!(err=MyGetDragItemData(drag,1,TOC_FLAVOR,(void*)&data)))
{
tocH = **(TOCHandle**)data;
DoIterativeThingy(tocH,MESSAGE_REPLY_ITEM,0,(void*)targetInfo.rowNum);
}
ZapHandle(data);
return(err);
}
else
return dragNotAcceptedErr;
break;
}
return err;
}