TweetFollow Us on Twitter

Object Toolbox
Volume Number:7
Issue Number:11
Column Tag:C Workshop

Related Info: List Manager Window Manager Control Manager

Objects and the Toolbox

By Rob Hafernik, Austin, TX

Note: Source code files accompanying article are located on MacTech CD-ROM or source code disks.

Think C Objects and the Toolbox

[Rob Hafernik is lucky enough to be paid to program the Macintosh, by Technology Works, in Austin, Texas.]

All Mac programmers should be using MacApp these days. No question about it, it’s where the action is in Mac programming. Once in a while, however, you come upon a project that MacApp isn’t suited for. It could be that MacApp’s application sizes are too large for some tight project or that the project has to be in C, or whatever. It must be a good reason, or you wouldn’t want to part with all the goodies MacApp gives you for free.

This is the position I found myself in recently, and I started thinking about ways to combine Object Programming with the Mac Toolbox, but without MacApp. This is a bit difficult, since none of the Toolbox routines know an object from a linked list. I did, however, come up with a couple of good ideas, and they’re what this article is about.

This particular article explains a method for combining objects with the Toolbox List Manager, but the method could be used equally well with the Window Manager or Control Manager. Anywhere that the Toolbox allows for definition procedures, you can sneak an object in. This will give you all of the great aspects of Object Programming while still letting the Mac Toolbox do lots of the work.

Let’s back up a little. The List Manager is great if you want to display a list or grid of text, but to display anything but text (such as pictures or icons), you have to write your own definition procedure for a custom list. You also have to write your own defproc if you want items in a list to be highlighted in some way other than the standard (with a check mark, for example). Writing a definition procedure is a nasty chore that most of us shy away from since a defproc is a code resource and code resources are a pain to write and a pain to debug. If you’re in the dark about defprocs for the List Manager, now would be a good time to refer to Inside Macintosh, vol IV, page 276. There is a nice explanation of the definition procedure for lists and how they work.

When you think about the List Manager, it’s only natural to think: “A list of what”? Well, a list of objects would certainly be nice. If each item in a list were an object, then each object in the list could also have methods to do things like draw itself or perform some action when it’s double-clicked. While we’re at it, let’s make the list itself an object with methods to manipulate the list and display it.

That’s just what we have here: A list object that holds information about the list and has methods to add items, display them, select them and so on. The List object also encapsulates the List Manager so that the programmer doesn’t need to deal with it or ever write another List Manager definition procedure. Each item in the list is also an object that is a sub-class of an Item object that we’ll define. The sub-class can do anything it needs to for a particular application.

The example program I’ll use to demonstrate these objects is very simple, almost to the point of being useless. The application, which I call Icon List, can open a file’s resource fork and display any icons it finds there in a list in a window. If you look at the inheritance diagram in figure 1, you’ll see the classes we’ll use in this example. I’ll talk about the CListObj object first.

Figure 1

A CListObj contains a handle to a regular Toolbox List Manager data structure, a rectangle for viewing the list in a window, and a flag to turn drawing on and off (for efficiency when adding a lot of items to a list - it’s a lot faster to turn off drawing, add them all in, then turn drawing back on).

The List Manager in the Mac Toolbox defines a data structure that includes a handle to the code that will actually draw the items in the list and highlight them. If you’re displaying a regular, text-based list (such as the one Standard File displays), then you’ll use the built-in code that is provided by the List Manager (‘LDEF’ resource id 0). If you’re customizing the list to display something else, you provide a block of code in a code resource (of type ‘LDEF’) and tell the List Manager to use that code.

Of course that means that the code you write has to be in a code resource and that’s a lot of trouble. There is a nice little trick that avoids all of this, however. I wish I could take credit for inventing this trick, but I can’t. I don’t even know who did it first, but it was at least three or four years ago. Here’s how the trick works: You have an LDEF resource, true, but it contains only a JMP instruction to some code in your regular application. Before you initialize a list manager list, you open this “fake defproc” resource and copy into it a JMP instruction and the address of the routine in your application code that will function as a list defproc (make sure that your code is in a locked segment!). The list manager will take the handle, dereference it and jump to it. Your JMP instruction will then jump to your code and off you’ll go. When your code returns, the return address of the List Manager is right there on the stack. You now have code in your application that the List Manager thinks is a regular, old defproc - that you can write, compile, link and debug just like any other code.

The Methods

Now for the methods of CListObj. The first is IList. This is always called first to initialize the list. It sets up the List Manager list, initializes it’s fields and sets up the handle for the “Fake” defproc. The Dispose method does just the reverse, throwing everything away. Notice that it does not dispose of the items in the list - the application code should do that, usually before disposing of the CListObj. The next few methods take care of the usual events you have in a Mac program: mouse clicks, update events and activate events. Actually, you can see that most of the work is done by the List Manager, with the CListObj just acting as glue code.

Then there are methods for setting flags (DrawingOn) and scrolling the list (Scroll) and counting the number of items in the list (NumObjs). These are pretty simple, as the code shows (the Toolbox does all the work, and that’s the way we want it). Next comes the Add method. Notice that Add takes a CItemObj as it’s argument. If you look at the definition of CItemObj, you’ll see that it’s nearly empty. It consists of just one method, DrawItem. In CItemObj, even DrawItem doesn’t do anything but go into the debugger with message. This is because each program that uses a CListObj must sub-class CItemObj to do whatever is specific to that program. In the example at hand, the sub-class is called CIconItemObj, and it holds information about an ‘icon’ resource. In addition, the CIconItemObjs’ DrawItem method overrides CItemObj->DrawItem to actually plot the icon and show it’s name and number. The Add method of a CListObj knows nothing of this, of course. It just gets the CIconItemObj and adds it to the list (notice that it just adds a reference to it, it’s handle, to the list). In like fashion, the Remove method removes a CItemObj reference from the list (without disposing the actual object).

Now for the fun part. If you’ve used MacApp, you know about that wonderful object, TList. A TList is like our CListObj here, but it does not display it’s list, it just manages it. And manage it does; there are dozens of useful methods. So useful, in fact, that I’ve stolen several of them for the CListObj object. The first is ForEach. ForEach takes a function pointer and calls the function once for each object in the list, passing each object to the function. It’s the equivalent of a for loop getting each item and calling a function with each one. You can use this all sorts of ways. If the items in the list were e-mail recipients, for example, the e-mail application would do something like:

recipientList->ForEach ( SendMessage );

Now, isn’t that a nice, readable bit of code? The next method, ForEachSelected, is the same except it only calls the function for items in the list that are currently selected.

After ForEachSelected come a couple of methods to manipulate and report on the current selection, SelectNone and AnySelected. Then comes FirstThat, which is similar to ForEach in that it walks down the list calling a function you supply and passing each item in the list. FirstThat, however, stops as soon as the function you’ve supplied returns true for one of the items. In other words, FirstThat does a search for an item in the list, using the criteria in the function you pass and returning the matching item.

The IconList application shows a CListObj in action (well, some of it). When the users selects the “Open ” menu pick, they get a standard file dialog. Once they select a file, the program opens a window and starts up a CListObj. Then it opens the file’s resource fork and looks for ‘icon’ resources. For each ‘icon’ resource it finds, the program allocates a CIconItemObj, fills it’s fields (we could have defined an IIconItem method to do initialization for us) and adds it to a list. It also sets the windowKind field of the window to something special and puts a reference to the CListObj in the window’s refCon field where the application can find it later. Although the Icon List application builds the list once and for all, another application could keep adding and removing items from the list all the time. Once the window is up and the list is built, the event loop takes care of it.

For example, if an update event comes along, the program checks to see if the window is an “Icon List” window (using it’s windowKind field). If it is, the program pulls the CListObj from the window’s refCon field and calls the UpdateList method of the CListObj. What happens next is a little complicated, so keep an eye on Figure 2. The event loop calls the UpdateList method. The UpdateList method calls the Toolbox List Manager with the handle to the list and tells it to update. The Toolbox List Manager calls our “fake” defproc to draw each item in the update region. The “fake” defproc calls a dispatching function, called ListFunction, each time. The ListFunction only knows that the list is made up of a sub-class of the CListObj. It pulls the object in question out of the list and calls it’s DrawItem function with a highlight flag set true or false. After that, it’s up to the DrawItem method to do whatever it wants to in it’s rectangle. Notice that we get all the selection logic and auto-scroll for free from the Mac Toolbox. Activate and MouseDown events are similar.

That’s all there is to it. You include the CListObj code in your application, sub-class CItemObj to do whatever you want and you’ve got a displayed list with a bunch of extras. Object Programming is really nice for programmers. Suppose we wanted to add support for color icons. All we would have to do is sub-class CItemObj again with a CColorIconItemObj and change FillIconList to also scan for ‘cicn’ resources and we would be done (note that CListObj and the event loop would not change at all). CListObj could also be sub-classed. If an application allowed users to double-click on a list item to do something to that item, CListObj could be sub-classed and support for double-clicks (timing and all) added with a “DoubleClick” method. That method could detect a double-click, get the CItemObj from the list and call a “DoubleClicked” method that’s been added to it. You wouldn’t rewrite CListObj, just add to it.

Figure 2.

A similar technique can be used for other definition functions. A Control Manager defproc, for example, would be an object with fields to hold it’s data and methods to respond to the Control Managers calls for drawing, highlighting, setting the value and so on. It would also use a “fake” defproc handle and a dispatching function to trick the Toolbox into using objects.

I always like to hear reactions to my work. I can be reached via e-mail at America On Line as ‘rob042’ or you can catch me as ‘shokwave@well.sf.ca.us’ on the internet.

/*+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-
 CItemObj.h - Defines an abstract List Item
class that must be overriden.
+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+- */

#ifndef CLASS_COBJECT
#include"CObject.h"
#endif

#define CLASS_CITEMOBJ

/* :::::::::: item object :::::::::: */
struct CItemObj:CObject {
 /*** no fields... */

 /*** methods */
 void DrawItem ( Boolean selected, 
 /* true if this item is to be
 drawn selected */
 Rect *dispR );  
 /* display rectangle */
 };

/*+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-
 CItemObj.c - Implements the abstract List Item
Class (must be overridden).
+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+- */

#include "CItemObj.h"

/* |||||||||||||||||||| */
 void CItemObj::DrawItem ( Boolean selected, 
 Rect *dispR )
 {
 DebugStr( "You MUST override CItemObj::DrawItem" );
 }

/*+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-
 CIconItemObj.h - Defines an Icon List Item 
subclass of CItemObj.
+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+- */

#ifndef CLASS_COBJECT
#include"CObject.h"
#endif

#ifndef CLASS_CITEMOBJ
#include"CItemObj.h"
#endif

#define CLASS_CICONITEMOBJ

/* :::::::::: list item object :::::::::: */
struct CIconItemObj:CItemObj {
 /*** fields... */
 int    id; /* icon res id number */
 Str255 name;    /* icon res name */
 Handle bits;    /* handle to icon image */
 
 /*** methods  */
 void DrawItem ( Boolean selected, 
 Rect *dispR );
 /* draw the icon item */
 };

/*+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-
 CIconItemObj.c - Implements an object that holds an icon from a resource 
file and draws it.
+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+- */

#include "CIconItemObj.h"
#include <color.h>

/* |||||||||||||||||||| */
 void CIconItemObj::DrawItem ( Boolean selected,
 Rect *dispR )
 {
 Rect   arect;
 Str255 numStr;
 
 /*** draw an icon item */
 EraseRect ( dispR );
 
 /*** draw the icon (inset by 4 pixels) */
 arect = *dispR;
 arect.top += 4;
 arect.bottom = arect.top +32;
 arect.left += 4;
 arect.right = arect.left + 32;
 PlotIcon ( &arect, bits );
 
 /*** display the name and id next to it */
 TextFont ( systemFont );
 TextSize ( 12 );
 TextFace ( 0 );
 arect.left = arect.right + 8;
 arect.right = arect.left + 40;
 arect.top += 9;
 arect.bottom -= 9;
 NumToString ( (long) id, numStr );
 TextBox ( &numStr[1], (int) numStr[0], 
 &arect, teJustLeft );
 
 TextFont ( geneva );
 TextSize ( 9 );
 arect.left = arect.right + 8;
 arect.right = dispR->right;
 arect.bottom += 16;
 TextBox ( &name[1], (int) name[0], 
 &arect, teJustLeft );
 
 /*** if selected, invert the rect (could use 
 hilite mode) */
 if ( selected ) {
 InvertRect ( dispR );
 }
 }

/*+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-
 CObjList.h - Defines an object that holds a
list of other objects and uses the List Manager
to display them.  Note that:
 * List is one-dimensional
 * Objects in list must be a sub-class of
 ItemObj object
 * ListObj is 1 based (not 0 based).
+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+- */

#define CLASS_OBJLIST

#ifndef CLASS_CITEMOBJ
#include"CItemObj.h"
#endif

#ifndef CLASS_COBJECT
#include"CObject.h"
#endif

#ifndef NULL
#define NULL0L
#endif

#define MYPROCID 42

/* :::: LDEF dispatch routine definition :::: */
pascal void ListFunction (int lMessage,
 Boolean lSelect, Rect *lRect, Cell lCell,
 int lDataOffSet, int lDataLen, 
 ListHandle lHandle );

/* :::::::::: list object :::::::::: */
struct CListObj:CObject {
 /*** fields */
 ListHandle theList; /* ROM List Manager list */
 Rect viewRect; /* location of the view in the
 window (incl scroll bar) */
 Boolean drawIt; /* drawing on or not */
 
 /*** methods */
 void IList ( Rect *viewR,
 /* view rectangle in win coords */
 int cellHeight, /* height of a row in pixels */
 Boolean hasScrollBar, /* true if vertical scroll bar */
 int selectMethod, /* List Manager selection flags */
 WindowPtr theWin ); /* window this list displays in */

 void Dispose ( void );  /* throw away everything (but 
 not objects in list) */
 
 Boolean MouseInList ( Point thePoint, 
 /* location of mouse click */
 int modifiers ); /* modifier keys */

 void UpdateList (RgnHandle updateRgn );
 /* update list(pass visRgn handle)*/

 void ActivateList ( Boolean activate );
 /* activate event for list,
 true or false */

 void DrawingOn ( Boolean drawing );
 /* turns drawing on and off */

 void Scroll ( int amount );
 /* scroll by amoumt (+ is up) */
 
 int NumObjs ( void );  
 /* return number of objs in list */

 void Add ( CItemObj *theObject );
 /* add a new object to the list */

 void Remove ( CItemObj *theObject );
 /* remove the obj from the list */

 void ForEach ( VoidFunc doThis ); 
 /* execute the function, passing
 each object in the list */

 void ForEachSelected ( VoidFunc doThis );
 /* same as ForEach, but selected */

 void SelectNone ( void );
 /* de-select all */

 Boolean AnySelected ( void );
 /* true if any objs are selected */

 CItemObj *FirstThat ( BooleanFunc test );
 /* return first object in list that
 satisfies test */

 CItemObj *GetIndObject ( int n ); 
 /* return nth object, if it any */
 };

/*+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-
 CListObj.c - Implements an object that holds
a list of other objects and uses the List Manager
to display them.
+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+- */

#ifndef CLASS_LISTOBJ
#include"CListObj.h"
#endif

/* |||||||||||||||||||| */
 void CListObj::IList ( Rect *viewR,
 int cellHeight, Boolean hasScrollBar,
 int selectMethod, WindowPtr theWin )
 {
 Rect   dataBounds, viewBounds;
 Point  cellSize;
 Handle fakeDefProc;
 int    jmpInstruction;
 long   jmpAddress;
 
 /*** create the "fake" defproc handle and put
 a jump to our code in it */
 jmpInstruction = 0x4EF9;
 jmpAddress = (long) ListFunction;
 fakeDefProc = GetResource( 'LDEF', MYPROCID );
 HLock ( fakeDefProc );
 BlockMove ( &jmpInstruction, *fakeDefProc,
 sizeof ( int ) );
 BlockMove ( &jmpAddress, *fakeDefProc + 2,
 sizeof ( long ) );
 HUnlock ( fakeDefProc );

 /*** initialize the list object, set up the
 corresponding list manager object */
 theList = NULL;
 viewRect = viewBounds = *viewR;
 drawIt = true;
 SetRect ( &dataBounds, 0, 0, 1, 0 );
 cellSize.h = viewR->right - viewR->left;
 cellSize.v = cellHeight;
 viewBounds.right -= 15;
 theList = LNew ( &viewBounds, &dataBounds,
 cellSize, MYPROCID, theWin, true,
 false, false, hasScrollBar );
 (*theList)->selFlags = (SignedByte) selectMethod;
 (*theList)->refCon = (long) this;
 }

/* |||||||||||||||||||| */
 void CListObj::Dispose ( void )
 {
 /*** dispose of the list, but not contents */
 LDispose ( theList );
 inherited::Dispose ( );
 }

/* |||||||||||||||||||| */
 BooleanCListObj::MouseInList ( Point thePoint,
 int modifiers )
 {
 /*** process a mouse click in the list by
 calling the toolbox List Manager */
 if ( PtInRect(thePoint,&viewRect)&&drawIt ) {
 LClick ( thePoint, modifiers, theList );
 return true; /* return true if mouse was in list */
 }
 return false; /* return false if it was a miss */
 }

/* |||||||||||||||||||| */
 void CListObj::UpdateList( RgnHandle updateRgn )
 {
 /*** process update event (only if drawing is on) */
 if ( drawIt ) LUpdate ( updateRgn, theList );
 }

/* |||||||||||||||||||| */
 void CListObj::ActivateList ( Boolean activate )
 {
 /*** process an activate event */
 LActivate ( activate, theList );  
 }

/* |||||||||||||||||||| */
 void CListObj::DrawingOn ( Boolean drawing )
 {
 /*** turn drawing on or off */
 drawIt = drawing;
 LDoDraw ( drawing, theList );
 }

/* |||||||||||||||||||| */
 void CListObj::Scroll ( int amount )
 {
 /*** scroll the list */
 LScroll ( 0, amount, theList );
 }

/* |||||||||||||||||||| */
 int CListObj::NumObjs ( void )
 {
 /*** return the # of objects in the list */
 return ( (*theList)->dataBounds.bottom );
 }

/* |||||||||||||||||||| */
void CListObj::Add ( CItemObj *theObject )
 {
 int  rowNum;
 long value;
 Cell theCell;
 
 /*** add an item to the bottom of the list */
 rowNum = LAddRow ( 1, 32760, theList );
 value = (long) theObject;
 theCell.h = 0;
 theCell.v = rowNum;
 LSetCell(&value, sizeof (long), theCell, theList);
 }

/* |||||||||||||||||||| */
void CListObj::Remove ( CItemObj *theObject )
 {
 int  nRows, len;
 long value;
 Cell theCell;
 
 /*** delete the object from the list, if it's
 in it (doesn't delete the object) */
 len = sizeof ( long );
 nRows = NumObjs ( );
 for ( theCell.h = 0, theCell.v = 0; 
 theCell.v < nRows; ++theCell.v ) {
 LGetCell( &value, &len, theCell, theList );
 if ( value == (long) theObject ) {
 LDelRow ( 1, theCell.v, theList );
 return;
 }
 }
 }

/* |||||||||||||||||||| */
void CListObj::ForEach ( VoidFunc doThis )
 {
 int    nRows, len;
 long   value;
 Cell   theCell;
 CItemObj *theObject;
 
 /*** call the doThis function, passing each
 object in the list */
 len = sizeof ( long );
 nRows = NumObjs ( );
 for ( theCell.h = 0, theCell.v = 0; 
 theCell.v < nRows; ++theCell.v ) {
 LGetCell ( &value, &len,theCell, theList );
 theObject = (CItemObj *) value;
 doThis ( theObject );
 }
 }

/* |||||||||||||||||||| */
void CListObj::ForEachSelected ( VoidFunc doThis )
 {
 int    nRows, len;
 long   value;
 Cell   theCell;
 CItemObj *theObject;
 
 /*** call the VoidFunc, passing each selected
 object in the list */
 len = sizeof ( long );
 nRows = NumObjs ( );
 theCell.h = theCell.v = 0;
 while (LGetSelect(true, &theCell, theList)) {
 LGetCell (&value, &len, theCell, theList);
 theObject = (CItemObj *) value;
 doThis ( theObject );
 ++theCell.v;
 }
 }

/* |||||||||||||||||||| */
void CListObj::SelectNone ( void )
 {
 int    nRows;
 long   value;
 Cell   theCell;
 CItemObj *theObject;
 
 /*** de-select all items */
 nRows = NumObjs ( );
 theCell.h = theCell.v = 0;
 while (LGetSelect(true,&theCell,theList)) {
 LSetSelect ( false, theCell, theList );
 ++theCell.v;
 }
 }

/* |||||||||||||||||||| */
Boolean CListObj::AnySelected ( void )
 {
 Cell   theCell;
 
 /*** return true if any item is selected */
 theCell.h = theCell.v = 0;
 return (LGetSelect(true,&theCell,theList));
 }

/* |||||||||||||||||||| */
CItemObj *CListObj::FirstThat ( BooleanFunc test )
 {
 int    nRows, len;
 long   value;
 Cell   theCell;
 CItemObj *theObject;
 
 /*** return the first item that passes the
 test in function test */
 len = sizeof ( long );
 nRows = NumObjs ( );
 for ( theCell.h = 0, theCell.v = 0; 
 theCell.v < nRows; ++theCell.v ) {
 LGetCell(&value, &len, theCell, theList);
 theObject = (CItemObj *) value;
 if ( test ( theObject ) ) 
 return ( theObject );
 }
 return ( NULL );
 }

/* |||||||||||||||||||| */
CItemObj * CListObj::GetIndObject ( int n )
 {
 int    len;
 long   value;
 Cell   theCell;
 CItemObj *theObject;

 /*** return the nth object in the list (list
 manager is 0 based, ListObj is 1 based) */
 --n;
 if ( (n < NumObjs ( )) && (n >= 0) ) {
 theCell.h = 0;
 theCell.v = n;
 len = sizeof ( long );
 LGetCell(&value, &len, theCell, theList);
 theObject = (CItemObj *) value;
 return ( theObject );
 }
 return ( NULL );
 }

/* |||||||||||||||||||| */
/* |||||||||||||||||||| */
pascal void ListFunction ( int lMessage, 
 Boolean lSelect, Rect *lRect, 
 Cell lCell, int lDataOffset, 
 int lDataLen, ListHandle lHandle )
 {
 CItemObj *item;
 long   *theData;
 
 /*** dispatch a message sent to a list by the
 list manager to the object it goes with */
 switch ( lMessage ) {
 case lInitMsg:
 break;
 case lDrawMsg:
 if ( lDataLen > 0 ) {
 theData = (long *) **(*lHandle)->cells;
 theData += lDataOffset/4;
 item = (CItemObj *) *theData;
 item->DrawItem ( lSelect, lRect );
 }
 break;
 case lHiliteMsg:
 if ( lDataLen > 0 ) {
 theData = (long *) **(*lHandle)->cells;
 theData += lDataOffset/4;
 item = (CItemObj *) *theData;
 item->DrawItem ( lSelect, lRect );
 }
 break;
 case lCloseMsg:
 break;
 }
 }

/*+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-
 main.c - main program for Icon List.
It’s function is to show off the List Object.
+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+- */

/* |||||||||| defines |||||||||| */
#define ICONWINKIND4242

#include<oops.h>

#ifndef CLASS_LISTOBJ
#include"CListObj.h"
#endif

#ifndef CLASS_ICONITEMOBJ
#include"CIconItemObj.h"
#endif

/* |||||||||| Globals |||||||||| */
intDoneFlag;
EventRecord TheEvent;
Point   TheMousePt;
Rect    DragRect;
MenuHandleAppleMenu;
MenuHandleFileMenu;
MenuHandleEditMenu;

/* |||||||||||||||||||| */
main ( )
 {
 WindowPtrwhichWindow;
 char   key;
 CListObj *anIconList;
 GrafPtroldPort;

 Initialize ( );
 while ( !DoneFlag ) {
 /*** the usual event loop... */
 GetNextEvent ( everyEvent, &TheEvent );
 switch ( TheEvent.what ) {
 case activateEvt:
 whichWindow = (WindowPtr) TheEvent.message;
 if ( ((WindowPeek)
 whichWindow)->windowKind ==
 ICONWINKIND ) {
 anIconList = 
  (CListObj*)GetWRefCon(whichWindow);
 anIconList->ActivateList(
 TheEvent.modifiers & activeFlag);
 }
 break;
 case updateEvt:
 whichWindow = (WindowPtr) TheEvent.message;
 if ( ((WindowPeek) 
 whichWindow)->windowKind ==
 ICONWINKIND ) {
 GetPort ( &oldPort );
 SetPort ( whichWindow );
 anIconList = 
  (CListObj*)GetWRefCon(whichWindow);
 BeginUpdate ( whichWindow );
 anIconList->UpdateList(
 whichWindow->visRgn);
 EndUpdate ( whichWindow );
 SetPort ( oldPort );
 }
 break;
 case mouseDown:
 switch ( FindWindow ( TheEvent.where,
 &whichWindow ) ) {
 case inMenuBar:
 DoMenu ( 
 MenuSelect ( TheEvent.where )
 );
 break;
 case inSysWindow:
 SystemClick ( &TheEvent,
 whichWindow );
 break;
 case inDrag:
 if ( ((WindowPeek)
 whichWindow)->windowKind ==
 ICONWINKIND ) {
 DragWindow ( whichWindow,
 TheEvent.where, &DragRect );
 }
 break;
 case inGoAway:
 if ( ((WindowPeek) 
 whichWindow)->windowKind == 
 ICONWINKIND ) {
 if ( TrackGoAway ( whichWindow, 
 TheEvent.where ) ) {
   CloseIconWindow(whichWindow);
   }
 }
 break;
 case inContent:
 if ( ((WindowPeek) 
 whichWindow)->windowKind == 
 ICONWINKIND ) {
 if (whichWindow!=FrontWindow())
 SelectWindow (whichWindow);
 else {
 SetPort ( whichWindow );
 TheMousePt = TheEvent.where;
 GlobalToLocal(&TheMousePt);
 anIconList = (CListObj *)
   GetWRefCon(whichWindow);
 anIconList->MouseInList (  TheMousePt, TheEvent.modifiers);
 }
 }
 break;
 }
 break;
 case autoKey:
 case keyDown:
 key = (char) 
 (TheEvent.message & charCodeMask);
 if ( TheEvent.modifiers & cmdKey )
 DoMenu ( MenuKey ( key ) );
 break;
 case nullEvent:
 SystemTask ( );
 break;
 }
 }
 }

/* |||||||||||||||||||| */
Initialize ( )
 {
 Handle h;
 Str255 volName;
 
 /*** init the mac rom stuff */
 InitGraf ( &thePort );
 InitFonts ();
 InitWindows ();
 InitMenus ();
 TEInit ();
 InitDialogs ( 0L );
 InitCursor ();
 FlushEvents ( everyEvent, 0 );
 
 MaxApplZone (); MoreMasters (); MoreMasters ();
 GetNextEvent ( everyEvent, &TheEvent );
 GetNextEvent ( everyEvent, &TheEvent );
 GetNextEvent ( everyEvent, &TheEvent );
 GetNextEvent ( everyEvent, &TheEvent );

 /*** set up a few globals ... */
 DoneFlag = false;
 DragRect = screenBits.bounds;

 /*** read in the menus */
 AppleMenu = GetMenu ( 1000 );
 InsertMenu ( AppleMenu, 0 );
 AddResMenu ( AppleMenu, 'DRVR' );
 FileMenu = GetMenu ( 1001 );
 InsertMenu ( FileMenu, 0 );
 EditMenu = GetMenu ( 1002 );
 InsertMenu ( EditMenu, 0 );
 DrawMenuBar ();
 }

/* |||||||||||||||||||| */
DoMenu ( long mResult )
 {
 int    theMenu, theItem, i;
 Str255 aname, freestr, maxstr;
 GrafPtrcurport;
 TEHandle te;
 MenuHandle new;
 long   sz, type;
 int    mark, *menuNum;
 WindowPtraWin;
 
 theItem = LoWord ( mResult );
 theMenu = HiWord ( mResult );
 switch ( theMenu ) {
 case 1000:
 if ( theItem == 1 ) DoAbout ( );
 else {
 GetItem ( AppleMenu, theItem, aname );
 GetPort ( &curport );
 OpenDeskAcc ( aname );
 SetPort ( curport );
 }
 break;
 case 1001:
 switch ( theItem ) {
 case 1:
 OpenIconWindow ( );
 break;
 case 2:
 aWin = FrontWindow ( );
 if ( aWin != 0L && 
 ((WindowPeek) aWin)->windowKind == 
 ICONWINKIND )
 CloseIconWindow ( aWin );
 break;
 case 4:
 DoneFlag = true;
 break;
 }
 break;
 case 1002:
 break;
 }

 HiliteMenu(0);
 }

/* |||||||||||||||||||| */
OpenIconWindow ( )
 {
 Point  where;
 SFReplytheFile;
 SFTypeList typs;
 WindowPtrtheWin;
 CListObj *theList;
 Rect   bounds;
 int    fileRef;
 
 /*** get the file */
 where.h = where.v = 42;
 SFGetFile (where,0L 0L,-1,typs,0L,&theFile);
 if ( theFile.good ) {
 
 /*** open the file */
 SetVol ( NULL, theFile.vRefNum );
 fileRef = OpenResFile (theFile.fName );
 if ( fileRef < 0 ) {
 StopAlert ( 1000, 0L );
 return;
 }

 /*** set up the window, start it's list
 object, fill it with icons */
 theWin = GetNewWindow ( 1000, 0L, -1L);
 ((WindowPeek) theWin)->windowKind = ICONWINKIND;
 SetWTitle ( theWin, theFile.fName );
 bounds = theWin->portRect;
 theList = new ( CListObj );
 theList->IList ( &bounds , 40, true, lOnlyOne, theWin );
 theList->DrawingOn ( false );

 /*** fill the list with ICONs */
 FillIconList ( fileRef, theList );
 theList->DrawingOn ( true );
 CloseResFile ( fileRef );
 SetWRefCon ( theWin, (long) theList );
 }
 }

/* |||||||||||||||||||| */
FillIconList ( int aFile, CListObj *aList )
 {
 int    i, iconID, numIcons;
 ResTypeiconType;
 Str255 iconName;
 CIconItemObj    *anIcon;
 Handle iconHan;
 
 /*** get each icon, create a new list item
 object, add it to the list */
 numIcons = Count1Resources ( 'ICON' );
 for ( i = 1; i <= numIcons; ++i ) {
 if ((iconHan = Get1IndResource ('ICON',i)) != 0L) {
 GetResInfo(iconHan, &iconID, &iconType, iconName);
 
 anIcon = new ( CIconItemObj );
 anIcon->id = iconID;
 BlockMove (iconName,anIcon->name,255L);
 DetachResource ( iconHan );
 anIcon->bits = iconHan;
 
 aList->Add ( anIcon );
 }
 }
 }

/* |||||||||||||||||||| */
CloseIconWindow ( WindowPtr theWin )
 {
 CListObj *theIconList;
 CIconItemObj    *anIcon;
 int    i, num;
 
 /*** throw away contents and close window */
 theIconList = (CListObj *)GetWRefCon(theWin);

 /*** throw away the CIconList objects, most efficient to go from the 
end of the list to the start */
 theIconList->DrawingOn ( false );
 num = theIconList->NumObjs();
 for ( i = num; i > 0; --i ) {
 anIcon = (CIconItemObj *) 
 theIconList->GetIndObject ( i );
 theIconList->Remove ( anIcon );
 DisposHandle ( anIcon->bits );
 delete ( anIcon );
 }
 theIconList->Dispose ( );
 DisposeWindow ( theWin );
 }

/* |||||||||||||||||||| */
DoAbout ( )
 {
 Alert ( 4242, 0L );
 }

/*+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-
 Icon List.Π.rsrc - all the resources for the
Icon List project.
+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+- */

resource 'ALRT' (4242, "About") {
 {52, 38, 168, 304},
 4242,
 { /* array: 4 elements */
 /* [1] */
 OK, visible, sound1,
 /* [2] */
 OK, visible, sound1,
 /* [3] */
 OK, visible, sound1,
 /* [4] */
 OK, visible, sound1
 }
};

resource 'ALRT' (1000, "No Res Fork") {
 {52, 34, 140, 314},
 128,
 { /* array: 4 elements */
 /* [1] */
 OK, visible, sound1,
 /* [2] */
 OK, visible, sound1,
 /* [3] */
 OK, visible, sound1,
 /* [4] */
 OK, visible, sound1
 }
};

resource 'DITL' (4242) {
 { /* array DITLarray: 3 elements */
 /* [1] */
 {77, 174, 97, 234},
 Button {
 enabled,
 "OK"
 },
 /* [2] */
 {17, 19, 49, 51},
 Icon {
 disabled,
 1001
 },
 /* [3] */
 {25, 68, 41, 242},
 StaticText {
 disabled,
 "Icon List by Rob Hafernik"
 }
 }
};

resource 'DITL' (128) {
 { /* array DITLarray: 2 elements */
 /* [1] */
 {55, 206, 75, 266},
 Button {
 enabled,
 "OK"
 },
 /* [2] */
 {16, 64, 49, 236},
 StaticText {
 disabled,
 "Can't open resource fork of file!"
 }
 }
};

resource 'ICON' (1001, "Application", purgeable) {
 $"0001 0000 0002 8000 0004 4000 0008 2000"
 $"0011 9000 0023 0800 0047 8400 0084 C200"
 $"0130 6100 0220 0080 043E 0040 0882 0020"
 $"10CE 0010 2060 0008 4030 3F04 8810 4082"
 $"4C40 8041 26C1 3022 1381 C814 090E 7F8F"
 $"0402 3007 0201 0007 0100 8007 0080 6007"
 $"0040 1FE7 0020 021F 0010 0407 0008 0800"
 $"0004 1000 0002 2000 0001 4000 0000 80"
};

resource 'ICN#' (128) {
 { /* array: 2 elements */
 /* [1] */
 $"0001 0000 0002 8000 0004 4000 0008 2000"
 $"0011 9000 0023 0800 0047 8400 0084 C200"
 $"0130 6100 0220 0080 043E 0040 0882 0020"
 $"10CE 0010 2060 0008 4030 3F04 8810 4082"
 $"4C40 8041 26C1 3022 1381 C814 090E 7F8F"
 $"0402 3007 0201 0007 0100 8007 0080 6007"
 $"0040 1FE7 0020 021F 0010 0407 0008 0800"
 $"0004 1000 0002 2000 0001 4000 0000 80",
 /* [2] */
 $"0001 0000 0003 8000 0007 C000 000F E000"
 $"001F F000 003F F800 007F FC00 00FF FE00"
 $"01FF FF00 03FF FF80 07FF FFC0 0FFF FFE0"
 $"1FFF FFF0 3FFF FFF8 7FFF FFFC FFFF FFFE"
 $"7FFF FFFF 3FFF FFFE 1FFF FFFC 0FFF FFFF"
 $"07FF FFFF 03FF FFFF 01FF FFFF 00FF FFFF"
 $"007F FFFF 003F FE1F 001F FC07 000F F800"
 $"0007 F000 0003 E000 0001 C000 0000 80"
 }
};

resource 'BNDL' (128) {
 'IcLi',
 0,
 { /* array TypeArray: 2 elements */
 /* [1] */
 'FREF',
 { /* array IDArray: 1 elements */
 /* [1] */
 0, 128
 },
 /* [2] */
 'ICN#',
 { /* array IDArray: 1 elements */
 /* [1] */
 0, 128
 }
 }
};

resource 'FREF' (128) {
 'APPL',
 0,
 ""
};

resource 'MENU' (1000, "Apple") {
 1000,
 textMenuProc,
 0x7FFFFFF9,
 enabled,
 apple,
 { /* array: 2 elements */
 /* [1] */
 "About Icon List ", noIcon, noKey, noMark, plain,
 /* [2] */
 "-", noIcon, noKey, noMark, plain
 }
};

resource 'MENU' (1001, "File") {
 1001,
 textMenuProc,
 0x7FFFFFFB,
 enabled,
 "File",
 { /* array: 4 elements */
 /* [1] */
 "Open ", noIcon, "O", noMark, plain,
 /* [2] */
 "Close ", noIcon, "W", noMark, plain,
 /* [3] */
 "-", noIcon, noKey, noMark, plain,
 /* [4] */
 "Quit", noIcon, "Q", noMark, plain
 }
};

resource 'MENU' (1002, "Edit") {
 1002,
 textMenuProc,
 0x7FFFFFE0,
 enabled,
 "Edit",
 { /* array: 5 elements */
 /* [1] */
 "Undo", noIcon, "Z", noMark, plain,
 /* [2] */
 "-", noIcon, noKey, noMark, plain,
 /* [3] */
 "Cut", noIcon, "X", noMark, plain,
 /* [4] */
 "Copy", noIcon, "C", noMark, plain,
 /* [5] */
 "Paste", noIcon, "V", noMark, plain
 }
};

resource 'WIND' (1000) {
 {60, 30, 260, 330},
 noGrowDocProc,
 visible,
 goAway,
 0x0,
 "New Window"
};

resource 'vers' (1) {
 0x1,
 0x0,
 0x0,
 0x1,
 verUs,
 "Icon List 1.0",
 "Version 1.0 - July 1990"
};

resource 'vers' (2) {
 0x1,
 0x0,
 0x0,
 0x1,
 verUs,
 "Icon List 1.0",
 "by Rob Hafernik"
};

data 'IcLi' (0, "Owner resource") {
 $"00"
};

data 'LDEF' (42) {
 $"0000 0000 0000"
};

 

Community Search:
MacTech Search:

Software Updates via MacUpdate

GarageSale 7.0.7 - Create outstanding eB...
GarageSale is a slick, full-featured client application for the eBay online auction system. Create and manage your auctions with ease. With GarageSale, you can create, edit, track, and manage... Read more
SpamSieve 2.9.28 - Robust spam filter fo...
SpamSieve is a robust spam filter for major email clients that uses powerful Bayesian spam filtering. SpamSieve understands what your spam looks like in order to block it all, but also learns what... Read more
Thunderbird 45.7.1 - Email client from M...
As of July 2012, Thunderbird has transitioned to a new governance model, with new features being developed by the broader free software and open source community, and security fixes and improvements... Read more
Opera 43.0.2442.991 - High-performance W...
Opera is a fast and secure browser trusted by millions of users. With the intuitive interface, Speed Dial and visual bookmarks for organizing favorite sites, news feature with fresh, relevant content... Read more
OnyX 3.2.4 - Maintenance and optimizatio...
OnyX is a multifunction utility that you can use to verify the startup disk and the structure of its system files, to run miscellaneous maintenance and cleaning tasks, to configure parameters in the... Read more
VueScan 9.5.71 - Scanner software with a...
VueScan is a scanning program that works with most high-quality flatbed and film scanners to produce scans that have excellent color fidelity and color balance. VueScan is easy to use, and has... Read more
Slack 2.5.1 - Collaborative communicatio...
Slack is a collaborative communication app that simplifies real-time messaging, archiving, and search for modern working teams. Version 2.5.1: New The way we load teams you don't view often has been... Read more
HandBrake 1.0.3 - Versatile video encode...
HandBrake is a tool for converting video from nearly any format to a selection of modern, widely supported codecs. Features Supported Sources VIDEO_TS folder, DVD image or real DVD (unencrypted... Read more
Vivaldi 1.7.735.46 - An advanced browser...
Vivaldi is a browser for our friends. In 1994, two programmers started working on a web browser. Our idea was to make a really fast browser, capable of running on limited hardware, keeping in mind... Read more
Vivaldi 1.7.735.46 - An advanced browser...
Vivaldi is a browser for our friends. In 1994, two programmers started working on a web browser. Our idea was to make a really fast browser, capable of running on limited hardware, keeping in mind... Read more

Mudd Masher arrives this week
Atooi Games, the minds behind Totes the Goat and Mutant Mudds, have a new game in the works -- Mudd Masher. The game, a hybrid of the independent studio's first two titles, is expected to launch this week on March 2. [Read more] | Read more »
The best sales on the App Store this wee...
The App Store has quite an exciting lineup of discount games this week that range across a variety of genres. It's a great opportunity to catch up on some of the premium games you may have been holding off on -- and some you can even grab for free... | Read more »
The best new games we played this week
Ah, here we are again at the close of another busy week. Don't rest too easy, though. We had a lot of great new releases in mobile games this week, and now you're going to have to spend all weekend playing them. That shouldn't be too much of a... | Read more »
Rollercoaster Tycoon Touch Guide: How to...
| Read more »
Rabbids Crazy Rush Guide: How to unlock...
The Rabbids are back in a new endless running adventure, Rabbids Crazy Rush. It's more ridiculous cartoon craziness as you help the little furballs gather enough fuel (soda) to get to the moon. Sure, it's a silly idea, but everyone has dreams --... | Read more »
Tavern Guardians (Games)
Tavern Guardians 1.0 Device: iOS Universal Category: Games Price: $2.99, Version: 1.0 (iTunes) Description: Tavern Guardians is a Hack-and-Slash action game played in the style of a match-three. You can experience high pace action... | Read more »
Slay your way to glory in idle RPG Endle...
It’s a golden age for idle games on the mobile market, and those addictive little clickers have a new best friend. South Korean developer Ekkorr released Endless Frontier last year, and players have been idling away the hours in the company of its... | Read more »
Tiny Striker: World Football Guide - How...
| Read more »
Good news everyone! Futurama: Worlds of...
Futurama is finding a new home on mobile in TinyCo and Fox Interactive's new game, Futurama: Worlds of Tomorrow. They're really doing it up, bringing on board Futurama creator Matt Groening along with the original cast and writers. TinyCo wants... | Read more »
MUL.MASH.TAB.BA.GAL.GAL (Games)
MUL.MASH.TAB.BA.GAL.GAL 1.0 Device: iOS Universal Category: Games Price: $2.99, Version: 1.0 (iTunes) Description: ENDLESS UPGRADES. CONSTANT DANGER. ANCIENT WISDOM. BOUNCY BALLS. Launch Sale, 40% OFF for a very limited time!!! MUL.... | Read more »

Price Scanner via MacPrices.net

27-inch 3.3GHz 5K iMac on sale for $2099, sav...
B&H Photo has the 27″ 3.3GHz 5K Apple iMac on sale for $2099.99 including free shipping plus NY sales tax only. Their price is $200 off MSRP. Amazon also has the 27″ 3.3GHz 5K iMac on sale for $... Read more
21-inch iMacs on sale for up to $111 off MSRP
B&H Photo has select 21″ Apple iMacs on sale for up to $110 off MSRP, each including free shipping plus NY sales tax only: - 21″ 2.8GHz iMac: $1189 $110 off MSRP - 21″ 1.6GHz iMac: $999 $100 off... Read more
12-inch 1.2GHz Retina MacBooks on sale for $2...
Newegg has the 12″ 1.2GHz Space Gray Retina MacBook (sku MLH82LL/A) on sale for $1349.99 including free shipping. Their price is $250 off MSRP, and it’s the lowest price available for this model.... Read more
13-inch MacBook Airs on sale for $100 off MSR...
B&H Photo has 13″ MacBook Airs on sale for $100 off MSRP. Shipping is free, and B&H charges NY sales tax only: - 13″ 1.6GHz/128GB MacBook Air (MMGF2LL/A): $899 $100 off MSRP - 13″ 1.6GHz/... Read more
9-inch 32GB Silver iPad Pro on sale for $549,...
B&H Photo has the 9.7″ 32GB Silver Apple iPad Pro on sale for $549 for a limited time. Shipping is free, and B&H charges NY sales tax only. Their price is $50 off standard MSRP for this model... Read more
13-inch 2.0GHz Apple MacBook Pros on sale for...
B&H has the non-Touch Bar 13″ 2.0GHz MacBook Pros in stock today and on sale for $100 off MSRP. Shipping is free, and B&H charges NY sales tax only: - 13″ 2.0GHz MacBook Pro Space Gray (... Read more
15-inch Touch Bar MacBook Pros on sale for up...
B&H Photo has the new 2016 15″ Apple Touch Bar MacBook Pros in stock today and on sale for up to $150 off MSRP. Shipping is free, and B&H charges NY sales tax only: - 15″ 2.7GHz Touch Bar... Read more
12-inch Retina MacBooks on sale for $1150, $1...
B&H has 12″ 1.1GHz Retina MacBooks on sale for $150 off MSRP. Shipping is free, and B&H charges NY sales tax only: - 12″ 1.1GHz Space Gray Retina MacBook: $1149 $150 off MSRP - 12″ 1.1GHz... Read more
Apple restocks refurbished 11-inch MacBook Ai...
Apple has Certified Refurbished 11″ MacBook Airs (the latest models recently discontinued by Apple), available for up to $170 off original MSRP. An Apple one-year warranty is included with each... Read more
Apple Park Opens to Employees in April With T...
Apple has announced that Apple Park, the company’s new 175-acre campus, will be ready for employees to begin occupying in April. The process of moving more than 12,000 people will take over six... Read more

Jobs Board

*Apple* Solutions Consultant - Apple (United...
# Apple Solutions Consultant Job Number: 55676865 Los Angeles, California, United States Posted: Feb. 22, 2017 Weekly Hours: 40.00 **Job Summary** As an Apple Read more
Programmer/Editor *Apple* Music Dance - App...
# Programmer/Editor Apple Music Dance Job Number: 55565967 Culver City, California, United States Posted: Feb. 23, 2017 Weekly Hours: **Job Summary** Apple Music Read more
Digital Marketing Specialist - *Apple* iClo...
# Digital Marketing Specialist - Apple iCloud Job Number: 54729233 Culver City, California, United States Posted: Feb. 22, 2017 Weekly Hours: 40.00 **Job Summary** Read more
Marketing Specialist, iTunes & *Apple*...
# Marketing Specialist, iTunes & Apple Music Job Number: 55704205 Culver City, California, United States Posted: Feb. 23, 2017 Weekly Hours: 40.00 **Job Summary** Read more
*Apple* Wireless Lead - T-ROC - The Retail O...
…of knowledge in wireless sales and activations to the Beautiful and NEW APPLE Experiencestore within MACYS. THIS role, APPLE Wireless Lead, isbrandnewas MACYS Read more
All contents are Copyright 1984-2011 by Xplain Corporation. All rights reserved. Theme designed by Icreon.