TweetFollow Us on Twitter

MACINTOSH C

Demonstration Program

Go to Contents
// 
// Scrap.c
// 
// 
// This program utilises the desk scrap and Scrap Manager routines to allow the user to:
//
//   Cut, copy and clear pictures from, and paste pictures to, two windows opened by the
//    program.
//
//   Paste pictures cut or copied from another application to the two windows opened
//    by the program.
//
//   Open and close a Clipboard window, in which the current contents of the desk scrap
//    are displayed.
//
// In addition to the pictures cut and copied from either the program's windows or from
// another application's windows, the Clipboard window will display text copied to the
// desk scrap as a result of text cut and copy operations in another application.  The
// program, however, does not support the pasting of this text to documents displayed in
// the program's windows.  (The demonstration program at Chapter 19 - Text and TextEdit
// shows how to cut, copy and paste text from and to a TextEdit edit structure using the
// desk scrap.)   
//
// The program utilises the following resources:
//
//   An 'MBAR' resource, and 'MENU' resources for Apple, File, and Edit menus (preload, 
//    non-purgeable).  
//
//   Three 'WIND' resources (purgeable) (initially visible), two for the program's main
//    windows and one for the Clipboard window.  
//
//   A 'PICT' resource (non-purgeable) containing a picture which may be cut, copied,
//    and pasted between the windows.
//
//   A 'STR#' resource (purgeable) containing strings to be displayed in the error
//    Alert.  
//
//   A 'SIZE' resource with the acceptSuspendResumeEvents, doesActivateOnFGSwitch, and
//    is32BitCompatible flags set.
//
// 

// ............................................................................. includes

#include <Appearance.h>
#include <Devices.h>
#include <LowMem.h>
#include <Scrap.h>
#include <Sound.h>
#include <ToolUtils.h>

// .............................................................................. defines

#define mApple            128
#define  iAbout           1
#define mFile             129
#define  iClose           4
#define  iQuit            11
#define mEdit             130
#define  iCut             3
#define  iCopy            4
#define  iPaste           5
#define  iClear           6
#define  iClipboard       9
#define rMenubar          128
#define rWindow           128
#define rClipboardWindow  130
#define rPicture          128
#define rErrorStrings     128
#define  eFailMenu        1
#define  eFailWindow      2
#define  eFailDocStruc    3
#define  eZeroScrap       4
#define  ePutScrap        5
#define  eNoPictInScrap   6
#define kDocumentType     1
#define kClipboardType    2
#define MAXLONG           0x7FFFFFFF

// ............................................................................. typedefs

typedef struct
{
  PicHandle pictureHdl;
  Boolean   selectFlag;
  SInt16    windowType;
} docStructure, **docStructureHandle;

// ..................................................................... global variables

Boolean   gDone;
Boolean   gInBackground;
SInt16    gPixelDepth;  
Boolean   gIsColourDevice      = false;
WindowPtr gWindowPtrs[2];
WindowPtr gClipboardWindowPtr  = NULL;
Boolean   gClipboardShowing    = false;

// .................................................................. function prototypes

void  main                  (void);
void  doInitManagers        (void);
void  doEvents              (EventRecord *);
void  doUpdate              (EventRecord *);
void  doOSEvent             (EventRecord *);
void  doAdjustMenus         (void);
void  doMenuChoice          (SInt32);
void  doErrorAlert          (SInt16);
void  doOpenWindows         (void);
void  doCloseWindow         (void);
void  doInContent           (Point);
void  doCutCopyCommand      (Boolean);
void  doPasteCommand        (void);
void  doClearCommand        (void);
void  doClipboardCommand    (void);
void  doDrawClipboardWindow (void);
void  doDrawPictureWindow   (WindowPtr);
Rect  doSetDestRect         (Rect *,WindowPtr);
void  doGetDepthAndDevice   (void);

//  main

void  main(void)
{
  Handle      menubarHdl;
  MenuHandle  menuHdl;
  Boolean     gotEvent;
  EventRecord eventStructure;
  
  // ................................................................ initialise managers

  doInitManagers();

  // .......................................................... set up menu bar and menus
  
  menubarHdl = GetNewMBar(rMenubar);
  if(menubarHdl == NULL)
    doErrorAlert(eFailMenu);
  SetMenuBar(menubarHdl);
  DrawMenuBar();

  menuHdl = GetMenuHandle(mApple);
  if(menuHdl == NULL)
    doErrorAlert(eFailMenu);
  else
    AppendResMenu(menuHdl,'DRVR');

  // ....................................................................... open windows 

  doOpenWindows();

  // ........... get pixel depth and whether colour device for function SetThemeTextColor   
  
  doGetDepthAndDevice();

  // .................................................................... enter eventLoop
  
  gDone = false;

  while(!gDone)
  {
    gotEvent = WaitNextEvent(everyEvent,&eventStructure,MAXLONG,NULL);

    if(gotEvent)
      doEvents(&eventStructure);
  }
}

//  doInitManagers

void  doInitManagers(void)
{
  MaxApplZone();
  MoreMasters();

  InitGraf(&qd.thePort);
  InitFonts();
  InitWindows();
  InitMenus();
  TEInit();
  InitDialogs(NULL);

  InitCursor();
  FlushEvents(everyEvent,0);
  
  RegisterAppearanceClient();
}

//  doEvents

void  doEvents(EventRecord *eventStrucPtr)
{
  WindowPtr windowPtr;
  SInt16    partCode;
  SInt8      charCode;

  switch(eventStrucPtr->what)
  {
    case mouseDown:
      partCode = FindWindow(eventStrucPtr->where,&windowPtr);

      switch(partCode)
      {
        case inMenuBar:
          doAdjustMenus();
          doMenuChoice(MenuSelect(eventStrucPtr->where));
          break;

        case inContent:
          if(windowPtr != FrontWindow())
            SelectWindow(windowPtr);
          else
            doInContent(eventStrucPtr->where);
          break;

        case inDrag:
          DragWindow(windowPtr,eventStrucPtr->where,&qd.screenBits.bounds);
          break;

        case inGoAway:
          if(TrackGoAway(windowPtr,eventStrucPtr->where) == true)
            doCloseWindow();
          break;
      }
      break;

    case keyDown:
    case autoKey:
      charCode = eventStrucPtr->message & charCodeMask;
      if((eventStrucPtr->modifiers & cmdKey) != 0)
      {
        doAdjustMenus();
        doMenuChoice(MenuEvent(eventStrucPtr));
      }
      break;

    case updateEvt:
      doUpdate(eventStrucPtr);
      break;

    case activateEvt:
      windowPtr = (WindowPtr) eventStrucPtr->message;
      if(windowPtr == gClipboardWindowPtr)
        doDrawClipboardWindow();
      break;

    case osEvt:
      doOSEvent(eventStrucPtr);
      HiliteMenu(0);
      break;
  }
}

//  doUpdate

void  doUpdate(EventRecord *eventStrucPtr)
{
  WindowPtr           windowPtr;
  docStructureHandle  docStrucHdl;
  SInt32              windowType;

  windowPtr = (WindowPtr) eventStrucPtr->message;
  docStrucHdl = (docStructureHandle) GetWRefCon(windowPtr);
  windowType = (*docStrucHdl)->windowType;

  BeginUpdate(windowPtr);

  if(windowType == kDocumentType)
  {
    if((*docStrucHdl)->pictureHdl != NULL)
      doDrawPictureWindow(windowPtr);
  }
  else if(windowType == kClipboardType)
    doDrawClipboardWindow();

  EndUpdate(windowPtr);
}

//  doOSEvent

void  doOSEvent(EventRecord *eventStrucPtr)
{
  switch((eventStrucPtr->message >> 24) & 0x000000FF)
  {
    case suspendResumeMessage:
      gInBackground = (eventStrucPtr->message & resumeFlag) == 0;
      if(gClipboardWindowPtr && gClipboardShowing)
      {
        if(gInBackground)
          ShowHide(gClipboardWindowPtr);
        else
          ShowWindow(gClipboardWindowPtr);
      }
      break;

    case mouseMovedMessage:
      break;
  }
}

//  doAdjustMenus

void  doAdjustMenus(void)
{
  MenuHandle          fileMenuHdl, editMenuHdl;
  docStructureHandle  docStrucHdl;
  SInt32              scrapOffset;

  fileMenuHdl = GetMenuHandle(mFile);
  editMenuHdl = GetMenuHandle(mEdit);

  docStrucHdl = (docStructureHandle) GetWRefCon(FrontWindow());

  if((*docStrucHdl)->windowType == kClipboardType)
    EnableItem(fileMenuHdl,iClose);
  else
    DisableItem(fileMenuHdl,iClose);

  if((*docStrucHdl)->pictureHdl && (*docStrucHdl)->selectFlag)
  {
    EnableItem(editMenuHdl,iCut);
    EnableItem(editMenuHdl,iCopy);
    EnableItem(editMenuHdl,iClear);
  }
  else
  {
    DisableItem(editMenuHdl,iCut);
    DisableItem(editMenuHdl,iCopy);
    DisableItem(editMenuHdl,iClear);
  }

  if(GetScrap(NULL,'PICT',&scrapOffset)  && (*docStrucHdl)->windowType != kClipboardType)
    EnableItem(editMenuHdl,iPaste);
  else
    DisableItem(editMenuHdl,iPaste);

  DrawMenuBar();
}

//  doMenuChoice

void  doMenuChoice(SInt32 menuChoice)
{
  SInt16  menuID, menuItem;
  Str255  itemName;
  SInt16  daDriverRefNum;

  menuID = HiWord(menuChoice);
  menuItem = LoWord(menuChoice);

  if(menuID == 0)
    return;

  switch(menuID)
  {
    case mApple:
      if(menuItem == iAbout)
        SysBeep(10);
      else
      {
        GetMenuItemText(GetMenuHandle(mApple),menuItem,itemName);
        daDriverRefNum = OpenDeskAcc(itemName);
      }
      break;

    case mFile:
      if(menuItem == iClose)
        doCloseWindow();
      else if(menuItem == iQuit)
        gDone = true;
      break;

    case mEdit:
      switch(menuItem)
      {
        case iCut:
          doCutCopyCommand(true);
          break;

        case iCopy:
          doCutCopyCommand(false);
          break;

        case iPaste:
          doPasteCommand();
          break;

        case iClear:
          doClearCommand();
          break;

        case iClipboard:
          doClipboardCommand();
          break;
      }
      break;    
  }

  HiliteMenu(0);
}

//  doErrorAlert

void  doErrorAlert(SInt16 errorCode)
{
  AlertStdAlertParamRec paramRec;
  Str255                errorString;
  SInt16                itemHit;

  paramRec.movable        = true;
  paramRec.helpButton     = false;
  paramRec.filterProc     = NULL;
  paramRec.defaultText    = (StringPtr) kAlertDefaultOKText;
  paramRec.cancelText     = NULL;
  paramRec.otherText      = NULL;
  paramRec.defaultButton  = kAlertStdAlertOKButton;
  paramRec.cancelButton   = 0;
  paramRec.position       = kWindowDefaultPosition;

  GetIndString(errorString,rErrorStrings,errorCode);

  if(errorCode < ePutScrap)
  {
    StandardAlert(kAlertStopAlert,errorString,NULL,¶mRec,&itemHit);
    ExitToShell();
  }
  else
    StandardAlert(kAlertCautionAlert,errorString,NULL,¶mRec,&itemHit);
}

//  doOpenWindows

void  doOpenWindows(void)
{
  SInt16              a;
  WindowPtr           windowPtr;
  docStructureHandle  docStrucHdl;

  for(a=0;a<2;a++)
  {
    if(!(windowPtr = GetNewCWindow(rWindow + a,NULL,(WindowPtr)-1)))
      doErrorAlert(eFailWindow);
    gWindowPtrs[a] = windowPtr;

    if(!(docStrucHdl = (docStructureHandle) NewHandle(sizeof(docStructure))))
      doErrorAlert(eFailDocStruc);
    SetWRefCon(windowPtr,(SInt32) docStrucHdl);

    (*docStrucHdl)->pictureHdl = NULL;
    (*docStrucHdl)->windowType = kDocumentType;
    (*docStrucHdl)->selectFlag = false;
  }

  SetPort(windowPtr);

  (*docStrucHdl)->pictureHdl = GetPicture(rPicture);
}

//  doCloseWindow

void  doCloseWindow(void)
{
  WindowPtr           windowPtr;
  docStructureHandle  docStrucHdl;
  MenuHandle          editMenuHdl;

  windowPtr = FrontWindow();
  docStrucHdl = (docStructureHandle) GetWRefCon(windowPtr);

  if((*docStrucHdl)->windowType == kClipboardType)
  {
    DisposeWindow(windowPtr);
    gClipboardWindowPtr = NULL;
    gClipboardShowing = false;
    editMenuHdl = GetMenu(mEdit);
    SetMenuItemText(editMenuHdl,iClipboard,"\pShow Clipboard");
  }
}

//  doInContent

void  doInContent(Point mouseXY)
{
  WindowPtr           windowPtr;
  docStructureHandle  docStrucHdl;
  GrafPtr             oldPort;
  Rect                pictRect;

  windowPtr = FrontWindow();
  docStrucHdl = (docStructureHandle) GetWRefCon(windowPtr);

  if((*docStrucHdl)->windowType == kClipboardType)
    return;

  GetPort(&oldPort);
  SetPort(windowPtr);

  if((*docStrucHdl)->pictureHdl != NULL)
  {
    pictRect = doSetDestRect(&(*(*docStrucHdl)->pictureHdl)->picFrame,windowPtr);

    GlobalToLocal(&mouseXY);

    if(PtInRect(mouseXY,&pictRect) && (*docStrucHdl)->selectFlag == false)
    {
      (*docStrucHdl)->selectFlag = true;
      InvertRect(&pictRect);
    }
    else if(!PtInRect(mouseXY,&pictRect) && (*docStrucHdl)->selectFlag == true)
    {
      (*docStrucHdl)->selectFlag = false;
      InvertRect(&pictRect);
    }
  }

  SetPort(oldPort);
}

//  doCutCopyCommand

void  doCutCopyCommand(Boolean cutFlag)
{
  WindowPtr           windowPtr;
  docStructureHandle  docStrucHdl;  
  Size                dataLength;
  SInt32              errorCode;
  GrafPtr             oldPort;

  windowPtr = FrontWindow();
  docStrucHdl = (docStructureHandle) GetWRefCon(windowPtr);

  if((*docStrucHdl)->selectFlag == false)
    return;

  if(ZeroScrap() == noErr)
  {
    dataLength = GetHandleSize((Handle) (*docStrucHdl)->pictureHdl);
    HLock((Handle) (*docStrucHdl)->pictureHdl);

    errorCode = PutScrap((SInt32) dataLength,'PICT',
                         *((Handle) (*docStrucHdl)->pictureHdl));
    if(errorCode != noErr)
      doErrorAlert(ePutScrap);

    HUnlock((Handle) (*docStrucHdl)->pictureHdl);
  }
  else
    doErrorAlert(eZeroScrap);

  if(cutFlag)
  {
    GetPort(&oldPort);
    SetPort(windowPtr);

    DisposeHandle((Handle) (*docStrucHdl)->pictureHdl);
    (*docStrucHdl)->pictureHdl = NULL;
    (*docStrucHdl)->selectFlag = false;
    EraseRect(&windowPtr->portRect);

    SetPort(oldPort);
  }

  if(gClipboardWindowPtr != NULL)
    doDrawClipboardWindow();
}

//  doPasteCommand

void  doPasteCommand(void)
{
  WindowPtr           windowPtr;
  docStructureHandle  docStrucHdl;  
  GrafPtr             oldPort;
  SInt32              sizeOfPictData, scrapOffset;
  Handle              tempHdl;
  Rect                destRect;  

  windowPtr = FrontWindow();
  docStrucHdl = (docStructureHandle) GetWRefCon(windowPtr);

  GetPort(&oldPort);
  SetPort(windowPtr);  

  sizeOfPictData = GetScrap(NULL,'PICT',&scrapOffset);
  if(sizeOfPictData > 0)
  {
    tempHdl = NewHandle((Size) sizeOfPictData);  
    HLock(tempHdl);

    sizeOfPictData = GetScrap(tempHdl,'PICT',&scrapOffset);

    EraseRect(&windowPtr->portRect);
    (*docStrucHdl)->selectFlag = false;
    destRect = doSetDestRect(&(*(PicHandle) tempHdl)->picFrame,windowPtr);

    DrawPicture((PicHandle) tempHdl,&destRect);

    if((*docStrucHdl)->pictureHdl != NULL)
      DisposeHandle((Handle) (*docStrucHdl)->pictureHdl);

    (*docStrucHdl)->pictureHdl = (PicHandle) NewHandle((Size) sizeOfPictData);
    BlockMoveData(*tempHdl,*((*docStrucHdl)->pictureHdl),(Size) sizeOfPictData);

    HUnlock(tempHdl);
    DisposeHandle(tempHdl);
  }

  SetPort(oldPort);
}

//  doClearCommand

void  doClearCommand(void)
{
  WindowPtr           windowPtr;
  docStructureHandle  docStrucHdl;  
  GrafPtr             oldPort;

  windowPtr = FrontWindow();
  docStrucHdl = (docStructureHandle) GetWRefCon(windowPtr);

  GetPort(&oldPort);
  SetPort(windowPtr);

  DisposeHandle((Handle) (*docStrucHdl)->pictureHdl);
  (*docStrucHdl)->pictureHdl = NULL;
  (*docStrucHdl)->selectFlag = false;
  EraseRect(&windowPtr->portRect);

  SetPort(oldPort);
}

//  doClipboardCommand

void  doClipboardCommand(void)
{
  MenuHandle          editMenuHdl;
  docStructureHandle  docStrucHdl;  

  editMenuHdl = GetMenu(mEdit);

  if(gClipboardWindowPtr == NULL)
  {
    if(!(gClipboardWindowPtr = GetNewCWindow(rClipboardWindow,NULL,(WindowPtr)-1)))
      doErrorAlert(eFailWindow);
    if(!(docStrucHdl = (docStructureHandle) NewHandle(sizeof(docStructure))))
      doErrorAlert(eFailDocStruc);
    SetWRefCon(gClipboardWindowPtr,(SInt32) docStrucHdl);
    (*docStrucHdl)->windowType = kClipboardType;

    gClipboardShowing = true;

    SetMenuItemText(editMenuHdl,iClipboard,"\pHide Clipboard");
  }
  else
  {
    if(gClipboardShowing)
    {
      HideWindow(gClipboardWindowPtr);
      gClipboardShowing = false;
      SetMenuItemText(editMenuHdl,iClipboard,"\pShow Clipboard");
    }
    else
    {
      ShowWindow(gClipboardWindowPtr);
      gClipboardShowing = true;
      SetMenuItemText(editMenuHdl,iClipboard,"\pHide Clipboard");        
    }
  }
}

//  doDrawClipboardWindow

void  doDrawClipboardWindow(void)
{
  GrafPtr   oldPort;
  Rect      theRect, destRect;  
  RGBColor  blackColour = { 0x0000, 0x0000, 0x0000 };
  SInt32    sizeOfPictData, sizeOfTextData, scrapOffset;
  Handle    tempHdl;
  
  GetPort(&oldPort);
  SetPort(gClipboardWindowPtr);  

  EraseRect(&gClipboardWindowPtr->portRect);

  SetRect(&theRect,-1,-1,597,24);
  DrawThemeWindowHeader(&theRect,gClipboardWindowPtr == FrontWindow());

  if(gClipboardWindowPtr == FrontWindow())
    SetThemeTextColor(kThemeActiveWindowHeaderTextColor,gPixelDepth,gIsColourDevice);
  else
    SetThemeTextColor(kThemeInactiveWindowHeaderTextColor,gPixelDepth,gIsColourDevice);

  TextSize(10);
  MoveTo(7,16);
  DrawString("\pClipboard Contents:");

  sizeOfPictData = GetScrap(NULL,'PICT',&scrapOffset);
  if(sizeOfPictData > 0)
  {
    MoveTo(105,16);
    DrawString("\pPicture");

    tempHdl = NewHandle((Size) sizeOfPictData);  
    HLock(tempHdl);

    sizeOfPictData = GetScrap(tempHdl,'PICT',&scrapOffset);

    destRect = (*(PicHandle) tempHdl)->picFrame;
    OffsetRect(&destRect,-((*(PicHandle) tempHdl)->picFrame.left - 2),
                         -((*(PicHandle) tempHdl)->picFrame.top - 26));
    DrawPicture((PicHandle) tempHdl,&destRect);

    HUnlock(tempHdl);
    DisposeHandle(tempHdl);
  }

  sizeOfTextData = GetScrap(NULL,'TEXT',&scrapOffset);
  if(sizeOfTextData > 0)
  {
    MoveTo(105,16);
    DrawString("\pText");
  
    tempHdl = NewHandle((Size) sizeOfTextData);  
    HLock(tempHdl);
  
    sizeOfTextData = GetScrap(tempHdl,'TEXT',&scrapOffset);

    destRect = gClipboardWindowPtr->portRect;
    destRect.top += 24;
    InsetRect(&destRect,2,2);

    RGBForeColor(&blackColour);
    
    TETextBox(*tempHdl,sizeOfTextData,&destRect,0);

    HUnlock(tempHdl);
    DisposeHandle(tempHdl);
  }

  SetPort(oldPort);
}

//  doDrawPictureWindow

void  doDrawPictureWindow(WindowPtr windowPtr)
{
  GrafPtr             oldPort;
  Rect                destRect;
  docStructureHandle  docStrucHdl;

  GetPort(&oldPort);
  SetPort(windowPtr);

  docStrucHdl = (docStructureHandle) GetWRefCon(windowPtr);
  destRect = doSetDestRect(&(*(*docStrucHdl)->pictureHdl)->picFrame,windowPtr);

  DrawPicture((*docStrucHdl)->pictureHdl,&destRect);

  if((*docStrucHdl)->selectFlag)
    InvertRect(&destRect);

  SetPort(oldPort);
}

//  doSetDestRect

Rect  doSetDestRect(Rect *picFrame,WindowPtr windowPtr)
{
  Rect    destRect;
  SInt16  diffX, diffY;

  destRect = *picFrame;

  OffsetRect(&destRect,-(*picFrame).left,-(*picFrame).top);

  diffX = (windowPtr->portRect.right - windowPtr->portRect.left) - 
          ((*picFrame).right - (*picFrame).left);
  diffY = (windowPtr->portRect.bottom - windowPtr->portRect.top) - 
          ((*picFrame).bottom - (*picFrame).top);
            
  OffsetRect(&destRect,diffX / 2,diffY / 2);
  
  return(destRect);
}

//  doGetDepthAndDevice

void doGetDepthAndDevice(void)
{
  GDHandle  deviceHdl;

  deviceHdl = LMGetMainDevice();
  gPixelDepth = (*(*deviceHdl)->gdPMap)->pixelSize;
  if(BitTst(&(*deviceHdl)->gdFlags,gdDevType))
    gIsColourDevice = true;
}

// 

Demonstration Program Comments

When this program is run, the user should choose the Edit menu's Show Clipboard command
to open the Clipboard window.  The user should then cut, copy, clear and paste the
supplied picture from/to the two windows opened by the program, noting the effect on the
desk scrap as displayed in the Clipboard window.  (The picture inverts when the user
selects it with the mouse.)  The user should also copy some text from another
application's window and observe the changes to the contents of the Clipboard window.

The user should note that, when the Clipboard window is open and showing, it will be
hidden when the program is sent to the background and shown again when the program is
brought to the foreground.

The user may also copy pictures from another application's window and paste them in the
demonstration program's windows.

#define

kDocumentType and kClipboardType will enable the program to distinguish between the two
"document" windows opened by the program and the Clipboard window.

#typedef

Document structures will be attached to each of the two document windows.  docStructure
is the associated data type.

Global Variables

gDone controls program termination.  gInBackground relates to foreground/background
switching. The values assigned to gPixelDepth and gIsColourDevice will be used by the
Appearance Manager function SetThemeTextColor.

The WindowPtrs for the two document windows will be copied into the elements of
gWindowPtrs.  gClipBoardWindowPtr will be assigned the WindowPtr for the Clipboard window
when it is opened by the user.  gClipBoardShowing will keep track of whether the
Clipboard window is currently hidden or showing.

doEvents

Note that, in the case of an activate event for the Clipboard window, the
application-defined function doDrawClipBoardWindow is called.

doUpdate

If the window is of the document type (as opposed to the Clipboard type), and if the
window's document structure currently contains a picture, the application-defined
function doDrawPicture is called to draw that picture.  If the window is the Clipboard
window, the application-defined function doDrawClipboardWindow is called to draw the
Clipboard window.

doOSEvent

The if statement tests whether the Clipboard window has been opened by the user and
whether it should be showing when the demonstration program is in the foreground.  If the
window has previously been opened and gClipboardShowing contains true, and if the event
is a suspend event, the window is hidden.  If the event is a resume event, the window is
shown..  Note that ShowHide, rather than HideWindow is called to hide the window.  This 
is to prevent activation of the first document window in the list when the Clipboard 
window is in front and the application is switched out.

doAdjustMenus

If the front window is the Clipboard window, the Close item is enabled, otherwise it is
disabled.  If the document contains a picture and that picture is currently selected, the
Cut, Copy, and Clear items are enabled, otherwise they are disabled.  If the desk scrap
contains data of type 'PICT' and the front window is not the Clipboard window, the Paste
item is enabled, otherwise it is disabled.

doOpenWindows

doOpenWindows opens the two document windows, creates document structures for each
window, attaches the document structures to the windows and initialises the fields of the
document structures.  The graphics port of the second window created is then set as the
current port and a picture is read in from a resource, its handle being assigned to the
pictureHdl field of the second window's document structure.

doCloseWindow

doCloseWindow closes the Clipboard window (the only window that can be closed from within
the program).

If the window is the Clipboard window, The window is disposed of, the global variable
which contains its pointer is set to NULL, the global variable which keeps track of
whether the window is showing or hidden is set to false, and the text of the Show/Hide
Clipboard menu item is set to Show Clipboard.

doInContent

doInContent handles mouse-down events in the content region of a document window.  If the
window contains a picture, and if the mouse-down was inside the picture, the picture is
selected.  If the window contains a picture, and if the mouse-down was outside the
picture, the picture is deselected.

The first two lines get a pointer the front window and a handle to its document
structure.  If the front window is the Clipboard window, the function returns
immediately.  GetPort and SetPort save the current graphics port and make the graphics
port associated with the front window the current graphics port.

If the front window contains a picture the following occurs.  The application-defined
function doSetDestRect is called to return a rectangle of the same dimensions as that
contained in the picture structure's picFrame field, but centred laterally and vertically
in the window.  GlobalToLocal converts the mouse-down coordinates to local coordinates
preparatory to the call to PtInRect.  If the mouse-down occurred within the rectangle and
the picture has not yet been selected, the document structure's selectFlag field is set
to true and the picture is inverted.  If the mouse-down occurred outside the rectangle
and the picture is currently selected, the document structure's selectFlag field is set
to false, and the picture is re-inverted.

doCutCopyCommand

doCutCopyCommand handles the user's choice of the Cut and Copy items in the Edit menu.

The first two lines get a pointer to the front window and a handle to that window's
document structure.

If the selectFlag field of the document structure contains false (meaning that the
picture has not been selected), the function returns immediately.  (Note that no check is
made as to whether the front window is the Clipboard window because the menu adjustment
function disables the Cut and Copy items when the Clipboard window is the front window,
meaning that this function can never be called when the Clipboard window is in front.)

ZeroScrap attempts to purge the desk scrap.  If the call is successful, GetHandleSize
gets the size of the picture structure, HLock locks the picture structure, PutScrap
copies the picture to the desk scrap, and HUnlock unlocks the picture structure.  If the
calls to ZeroScrap and PutScrap are not successful, a caution alert is displayed to
advise the user of the error.

If the menu choice was the Cut item, additional action is taken.  Preparatory to a call
to EraseRect, the current graphics port is saved and the front window's port is made the
current port.  DisposeHandle is called to dispose of the picture structure and the
document structure's pictureHdl and selectFlag fields are set to NULL and false
respectively.  EraseRect then erases the picture.

Finally, and importantly, if the Clipboard window has previously been opened by the user,
an application defined function is called to draw the current contents of the desk scrap
in the Clipboard window.

doPasteCommand

doPasteCommand handles the user's choice of the Paste item from the Edit menu.  Note that
no check is made as to whether the front window is the Clipboard window because the menu
adjustment function disables the Paste item when the Clipboard window is the front
window, meaning that this function can never be called when the Clipboard window is in
front.

In order to determine whether the desk scrap contains data of type 'PICT', GetScrap is
called with the destHandle parameter set to NULL.  The following occurs if data of type
'PICT' is present in the desk scrap.

NewHandle and HLock create and lock a relocatable block of a size equivalent to the
'PICT' data in the scrap.  GetScrap is called again to copy the 'PICT' data in the scrap
to the newly-created relocatable block.  EraseRect erases the front window and the next
line sets the selectFlag field of the document structure associated with the front window
to false.  The call to the application-defined function doSetDestRect returns a
destination rectangle of the same dimensions as the picFrame rectangle but centred in the
front window.  DrawPicture draws the picture in this rectangle.

If the document structure currently contains a picture, DisposeHandle is called to
dispose of the picture structure.  NewHandle creates a new relocatable block the size of 
'PICT' data and assigns its handle to the pictureHdl field of the document structure. 
BlockMoveData then copies the bytes in the relocatable block created at the first line in
the if block to this new relocatable block.  HUnlock and DisposeHandle then dispose of
the block created at the first line in the if block.

doClearCommand

doClearCommand handles the user's choice of the Clear item in the Edit menu.

Note that, as was the case in the doCutCopyCommand function, no check is made as to
whether the front window is the Clipboard window because the menu adjustment function
disables the Clear item when the Clipboard window is the front window.

DisposeHandle dispose of the picture structure.  The next two three lines set the
pictureHdl field of the document structure to NULL, set the selectFlag field of the
document structure to false, and erase the window's port rectangle.

doClipboardCommand

doClipboardCommand handles the user's choice of the Show/Hide Clipboard command in the
Edit menu.

The first line gets the handle to the Edit menu.  This will be required in order to
toggle the Show/Hide Clipboard item's text between Show Clipboard and Hide Clipboard.

The if statement checks whether the Clipboard window has been created.  If not, the
Clipboard window is created by the call to GetNewCWindow, a document structure is created
and attached to the window, the windowType field of the document structure is set to
indicate that the window is of the Clipboard type, a global variable which keeps track of
whether the Clipboard window is currently showing or hidden is set to true, and the text
of the menu item is set to Hide Clipboard.

If the Clipboard window has previously been created, and if the window is currently
showing, the window is hidden, the Clipboard-showing flag is set to false, and the item's
text is set to Show Clipboard.  If the window is not currently showing, the window is
made visible, the Clipboard-showing flag is set to true, and the item's text is set to
Hide Clipboard.

doDrawClipboardWindow

doDrawClipboardWindow draws the contents of the desk scrap in the Clipboard window.  It
supports the drawing of both 'PICT' and 'TEXT' data.

The first three lines save the current graphics port, make the Clipboard window's
graphics port the current graphics port and erase the window's content region.

DrawThemeWindowHeader draws a window header in the top of the window.  Text describing
the type of data in the desk scrap will be drawn in this header.  The theme-compliant
colour for this text is set at the next four lines.  The following three lines set the
text size to 10pt and draw some text in the header.

The call to GetScrap, with NULL passed as the destHandle parameter, checks whether data
of type 'PICT' exists in the desk scrap.  If so, the following occurs.  The word
"Picture" is drawn in the window header.  A relocatable block the size of the 'PICT' data
is created and locked and GetScrap is called once again to copy the 'PICT' data from the
scrap into the newly-created block.  A destination rectangle, based on the rectangle in
the picFrame field of the picture structure, is created with its left and top fields set
to two pixels inside that part of the content area not occupied by the window header. 
The picture is then drawn in this destination rectangle, following which the relocatable
block created earlier is unlocked and disposed of.

The next call to GetScrap checks whether data of type 'TEXT' exists in the desk scrap. 
If so, much the same procedure is followed, the differences being that the word "Text" is
drawn in the window header, the destination rectangle is set to two pixels inside that
part of the content area not occupied by the window header., and the text is drawn in
this rectangle using TETextBox.  (TETextBox is a TextEdit routine, and is described at
Chapter 19 - Text and TextEdit.)

doDrawPictureWindow

doDrawPictureWindow draws the picture belonging to a document window in that window.

The third line gets the handle to the window's document structure.  The call to the
application-defined function doSetDestRect returns a rectangle of the same dimensions as
that contained in the picFrame field of the picture structure (the handle to which is
contained in the pictureHdl field of the document structure), but centred in the window. 
DrawPicture draws the picture specified in the window's document structure in this
rectangle.

If the selectFlag field of the document structure indicates that the picture is currently
selected, the call to InvertRect inverts the picture.

doSetDestRect

doSetDestRect takes the rectangle contained in the picFrame field of a picture structure
and returns a rectangle of the same dimensions but centred in the window's port
rectangle.

The first line makes a local Rect variable equal to the rectangle in the picFrame field. 
OffsetRect then offsets this rectangle so that its left and top fields both contain 0. 
The next four lines calculate the differences between the widths and heights of the
rectangle and the window's port rectangle.  This is used at the next call to OffsetRect
to further offset the rectangle to the middle of the port rectangle.  The rectangle is
then returned to the calling function.
 

Community Search:
MacTech Search:

Software Updates via MacUpdate

Latest Forum Discussions

See All

Top Mobile Game Discounts
Every day, we pick out a curated list of the best mobile discounts on the App Store and post them here. This list won't be comprehensive, but it every game on it is recommended. Feel free to check out the coverage we did on them in the links... | Read more »
Price of Glory unleashes its 1.4 Alpha u...
As much as we all probably dislike Maths as a subject, we do have to hand it to geometry for giving us the good old Hexgrid, home of some of the best strategy games. One such example, Price of Glory, has dropped its 1.4 Alpha update, stocked full... | Read more »
The SLC 2025 kicks off this month to cro...
Ever since the Solo Leveling: Arise Championship 2025 was announced, I have been looking forward to it. The promotional clip they released a month or two back showed crowds going absolutely nuts for the previous competitions, so imagine the... | Read more »
Dive into some early Magicpunk fun as Cr...
Excellent news for fans of steampunk and magic; the Precursor Test for Magicpunk MMORPG Crystal of Atlan opens today. This rather fancy way of saying beta test will remain open until March 5th and is available for PC - boo - and Android devices -... | Read more »
Prepare to get your mind melted as Evang...
If you are a fan of sci-fi shooters and incredibly weird, mind-bending anime series, then you are in for a treat, as Goddess of Victory: Nikke is gearing up for its second collaboration with Evangelion. We were also treated to an upcoming... | Read more »
Square Enix gives with one hand and slap...
We have something of a mixed bag coming over from Square Enix HQ today. Two of their mobile games are revelling in life with new events keeping them alive, whilst another has been thrown onto the ever-growing discard pile Square is building. I... | Read more »
Let the world burn as you have some fest...
It is time to leave the world burning once again as you take a much-needed break from that whole “hero” lark and enjoy some celebrations in Genshin Impact. Version 5.4, Moonlight Amidst Dreams, will see you in Inazuma to attend the Mikawa Flower... | Read more »
Full Moon Over the Abyssal Sea lands on...
Aether Gazer has announced its latest major update, and it is one of the loveliest event names I have ever heard. Full Moon Over the Abyssal Sea is an amazing name, and it comes loaded with two side stories, a new S-grade Modifier, and some fancy... | Read more »
Open your own eatery for all the forest...
Very important question; when you read the title Zoo Restaurant, do you also immediately think of running a restaurant in which you cook Zoo animals as the course? I will just assume yes. Anyway, come June 23rd we will all be able to start up our... | Read more »
Crystal of Atlan opens registration for...
Nuverse was prominently featured in the last month for all the wrong reasons with the USA TikTok debacle, but now it is putting all that behind it and preparing for the Crystal of Atlan beta test. Taking place between February 18th and March 5th,... | Read more »

Price Scanner via MacPrices.net

AT&T is offering a 65% discount on the ne...
AT&T is offering the new iPhone 16e for up to 65% off their monthly finance fee with 36-months of service. No trade-in is required. Discount is applied via monthly bill credits over the 36 month... Read more
Use this code to get a free iPhone 13 at Visi...
For a limited time, use code SWEETDEAL to get a free 128GB iPhone 13 Visible, Verizon’s low-cost wireless cell service, Visible. Deal is valid when you purchase the Visible+ annual plan. Free... Read more
M4 Mac minis on sale for $50-$80 off MSRP at...
B&H Photo has M4 Mac minis in stock and on sale right now for $50 to $80 off Apple’s MSRP, each including free 1-2 day shipping to most US addresses: – M4 Mac mini (16GB/256GB): $549, $50 off... Read more
Buy an iPhone 16 at Boost Mobile and get one...
Boost Mobile, an MVNO using AT&T and T-Mobile’s networks, is offering one year of free Unlimited service with the purchase of any iPhone 16. Purchase the iPhone at standard MSRP, and then choose... Read more
Get an iPhone 15 for only $299 at Boost Mobil...
Boost Mobile, an MVNO using AT&T and T-Mobile’s networks, is offering the 128GB iPhone 15 for $299.99 including service with their Unlimited Premium plan (50GB of premium data, $60/month), or $20... Read more
Unreal Mobile is offering $100 off any new iP...
Unreal Mobile, an MVNO using AT&T and T-Mobile’s networks, is offering a $100 discount on any new iPhone with service. This includes new iPhone 16 models as well as iPhone 15, 14, 13, and SE... Read more
Apple drops prices on clearance iPhone 14 mod...
With today’s introduction of the new iPhone 16e, Apple has discontinued the iPhone 14, 14 Pro, and SE. In response, Apple has dropped prices on unlocked, Certified Refurbished, iPhone 14 models to a... Read more
B&H has 16-inch M4 Max MacBook Pros on sa...
B&H Photo is offering a $360-$410 discount on new 16-inch MacBook Pros with M4 Max CPUs right now. B&H offers free 1-2 day shipping to most US addresses: – 16″ M4 Max MacBook Pro (36GB/1TB/... Read more
Amazon is offering a $100 discount on the M4...
Amazon has the M4 Pro Mac mini discounted $100 off MSRP right now. Shipping is free. Their price is the lowest currently available for this popular mini: – Mac mini M4 Pro (24GB/512GB): $1299, $100... Read more
B&H continues to offer $150-$220 discount...
B&H Photo has 14-inch M4 MacBook Pros on sale for $150-$220 off MSRP. B&H offers free 1-2 day shipping to most US addresses: – 14″ M4 MacBook Pro (16GB/512GB): $1449, $150 off MSRP – 14″ M4... Read more

Jobs Board

All contents are Copyright 1984-2011 by Xplain Corporation. All rights reserved. Theme designed by Icreon.