TweetFollow Us on Twitter

MicroApp 2
Volume Number:6
Issue Number:1
Column Tag:Jörg's Folder

C++ Micro-application

By Jörg Langowski, MacTutor Editorial Board

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

“C++ micro-application, part 2”

We’ll expand the micro-application that I presented last month by adding more functionality to the windows this time. But first, let me review some discussions that started on our Calvacom bulletin board after my first introduction to C++. The way I defined my matrix operations, it seems, wasn’t totally in the C++ spirit; there was a reason for this, which I forgot to explain.

The problem is that our matrix operators must return a result somehow. If you define the operator the intuitive way:

// 1

matrix  matrix::operator* (matrix& a)
{
 if (cols != a.rows)
 error(“class matrix: size mismatch in *”);
 matrix c(rows,a.cols);
 for (register int i=0 ; i<rows; i++)
 for (register int j=0 ; j<a.cols; j++)
 {
 register float sum = 0;
 for (register int k=0 ; k<cols; k++)
 sum = sum + elem(i,k)*a.elem(k,j);
 c(i,j) = sum;
 }
 return c;
}

the result matrix is allocated locally first, the product computed and stored into the local object, and the whole matrix is copied as soon as it is assigned to another variable of type matrix- like in matrix x = a*b, where x is newly created in the assignment, or in x = a*b, where x has been defined previously. That the local object is destroyed as soon as one leaves the operator’s scope and a copy be made for use in the scope that called the operator is required by C++. But the copy operation may take a long time with big arrays, and in most cases it won’t be necessary because the result has already been computed and the memory allocated.

One solution - which I chose - was not to return an object as a result, but a reference to it:

//2

matrix& matrix::operator* (matrix& a)
{
 if (cols != a.rows)
 error(“class matrix: size mismatch in *”);
 matrix& c = *new matrix(rows,a.cols);
 for (register int i=0 ; i<rows; i++)
 for (register int j=0 ; j<a.cols; j++)
 {
 register float sum = 0;
 for (register int k=0 ; k<cols; k++)
 sum = sum + elem(i,k)*a.elem(k,j);
 c(i,j) = sum;
 }
 return c;
}

Here, the space for the result is allocated through the new() operation, which creates memory space that is not automatically destroyed as one leaves the operator’s context. Now, however, the system will never know when to free the space associated with the matrix, unless you tell it to do so by calling delete(). This precludes usage of those operators in any complex formulas, because intermediate results will be stored in matrix-type objects that are never accessible and can therefore never be deleted.

Alain Richard (on Calvacom) has raised these points, and others:

“ Constructors/destructors allow the use of data structures without having to care about initialization/ termination code. For instance, a first version of a class X might not need such code while the second version might need it; a program using class X can be recompiled without modification in both cases.

When a result of type matrix is produced, a 500*500 matrix is not passed back through the stack; only the data structure which defines it and contains a pointer to the data. The only problem is that the constructor will often copy the data unnecessarily; but the current definition of C++ doesn’t allow a better solution.”

Alain proposes to use procedures instead of operators for functions in which the programmer has to clean up intermediate results explicitly. One possibility, but this gives up the last advantage of operators, shortness of notation.

He has some more comments on bugs and deficiencies:

“ In C++, the only way to pass a result is through the return statement, even though the result is a function parameter. However, since one doesn’t know its name, one has to go through an intermediate variable R and then copy R into the actual result.

One should also note that the code that CFront generates isn’t very well optimized (a euphemism), or that the preprocessor relies too much on the efficiency of the C compiler, in our case MPW C 3.1b1. Some observations I made in the code generated by CFront:

CFront doesn’t optimize the use of intermediate variables and creates them for each new expression. Then, these variables are freed only at the end of the block containing the expression, which is stupid because it blocks memory.

A lot of dead code is generated by CFront. Usually, such code should be removed by the C compiler, but that is unfortunately not the case. For instance:

//3

struct A { A() {} };

struct B:A { B() {} };

main()
{
 B b;
}

The declaration of b causes 30 lines of assembler code to be generated which don’t do anything [in fact, when I checked it there seemed to be even more dead code - JL].

For the maintenance of virtual methods of a class A, CFront creates a method dispatch table. The table, named __A_vtbl, is stored together with the application code. In addition, CFront creates a global variable __A_ptbl, a pointer to __A_vtbl. This global is later used to initialize a hidden instance of each variable of type A. __A_vtbl and __A_ptbl are stored as globals, which prohibits the use of classes for DAs, CDEFs, LDEFs, etc.

I have found one or two bugs: it is not always possible to declare a variable inside an expression (produces a syntax error). There is also a problem with pointers to base class member functions in the case of multiple inheritance. That error is easy to circumvent but causes a run time, not a compilation error.

This is not an exhaustive list, but unfortunately shows that the MPW compilers are not quite mature yet. But even with the bad code quality, CFront does at least exist. At any rate, C++ is going to be the most important development language for the next years.”

I almost agree with the last statement - however, I am also getting very curious about Eiffel, a new OOP-language which is rumored to be available for the Mac next year.

Zoom, grow and scroll for MacTutorApp

The skeleton application that I presented last month did not do much; the window could not be resized or zoomed, and there were no controls such as scroll bars present. This month we’ll expand the application by adding these functions. We’ll use the definitions that we made last time, and create a new subclass of MacTutorDocument, MacTutorGrow.

The interest of the object-oriented approach is, of course, that we can reuse most of the code that we have already defined in the previous example. Only those functions which are specific to our new document will have to be redefined.

Placeholders for these functions exist already in the TDocument class definition, which is part of the C++ examples that come with Apple’s C++ system. That definition is quite long, an we don’t need to reprint it here; all I show (listing 1) is the header file that constitutes the interface. You see that entries exist for methods such as DoContent, DoGrow and DoZoom. These methods will be called from the main event loop, so if you define your own document class with a non-empty DoContent method and a mouse click occurs inside the content region of the document window, that method will be called automatically.

Therefore, the changes we have to do to previous month’s example are quite simple: the main program will still be an object of type TMacTutorApp, but the document opened will be of a new class, which we call TMacTutorGrow. Then we only have to change a few lines in the main program definition of TMacTutorApp, as you see in listing 3. If someone had already defined a document class for us that handles scrolling and resizing, that is all we would have to do to make the application work with that new type of document window.

Unfortunately, we still have to do that work, and I won’t claim I did a perfect job of scrolling the display message around in the example window; window resizing and drawing of the controls works OK, but the scrolling still leaves something to be desired. Only when the window actually gets redrawn on an update event is the contents displayed correctly. You’re, of course, invited to improve on my example.

Listing 2 shows the definition of the TMacTutorGrow class - the header file - followed by the actual code. We derive TMacTutorGrow from TMacTutorDocument (see last column), adding some private variables and methods, and overriding some of the public methods in TDocument.

The first thing to note is that we define some functions that are not methods of any particular class for scroll bar handling (this is the approach Apple took for its TEDocument example). To indicate to the compiler that these functions are defined in C, their prototype definitions are enclosed in curly brackets and prefixed by extern “C”. This is a feature of the 2.0 implementation of C++ and not mentioned in Stroustrup’s book.

The class implementation follows, first we need to define the constructor and destructor. The constructor of a subclass is called after the constructor of its ancestor; therefore all that is left to do is to get the scroll bar resources, insert the controls into the window, and initialize their values. The display string (our document’s “contents”) is passed on to the ancestor’s constructor. The destructor will dispose the controls; disposing of the window is taken care of by the superclass.

For the DoUpdate method, we replicate the definition from the TMacTutorDocument class, because our DrawWindow method has changed. In this case, the display string and the controls are redrawn. I had to change the fDisplayString variable from the TMacTutorDocument class from private to protected, so that the derived class could access it. The other possibility would have been to go through the interface that I provided and use the method GetDisplayString.

The DoActivate method invalidates the control and grow rectangles when the window is activated, so that they get redrawn on the next update event. On deactivation, the controls are hidden.

Several methods are provided for redrawing the scroll bars in their correct position; they are needed by the DoGrow and DoZoom methods. The main work is done by the routine AdjustScrollSizes, which calculates the control positions from the port rectangle of the window and moves the controls into their places. AdjustScrollValues simply resets the controls to their initial values; in a real-world application, one would insert code here that calculates the correct maximum, minimum and control settings from the properties of the window’s contents (e.g. number of lines in a text, size of a picture).

DoZoom zooms the window, then repositions the scroll bars and invalidates all of the window’s contents except for the control rectangles. The update event will then take care of redrawing. DoGrow works in the same way, only the update region is calculated through an intersection of the old and the new window’s port rectangles, the standard method that’s already been described in IM Vol.I.

DoContent is where scrolling is handled. I implemented scrolling through a call to ScrollRect and a resetting of the origin of the GrafPort. This means that the controls have to be repositioned after the scrolling, otherwise they are redrawn outside the window (the SetOrigin made them scroll with the rest of the contents). Therefore, a call to AdjustScrollSizes() is made after the control value has changed and the window has been scrolled.

The thumb region and up/down arrows with page regions are handled separately; for the thumb region, the window adjustment can be made after the control has been released, while the arrows and page fields have to respond continuously while the mouse button is down. This is done through a control action procedure, which is defined as an external routine and not a class method. (This is the way Apple implemented it in their example; I don’t know whether a pointer to a class method can be passed as a filter procedure to a toolbox routine). The control action routines, whose prototypes had been defined at the start of the program, are implemented at the end of listing 2.

There are certainly a few bugs left in my example, and the scrolling leaves much to be desired but take this as a challenge for improving the example. We’ll expand this application with other features as we proceed in our tutorials. Until then.

Listing 1:  Apple’s TDocument class standard definitions

class TDocument : HandleObject {
protected:
 WindowPtr fDocWindow;

public:
 TDocument(short resID);  virtual ~TDocument();
 // you will need to override these in your subclasses,
 // since they are do-nothing routines by default...
 virtual void DoZoom(short partCode) {}
 virtual void DoGrow(EventRecord* theEvent) {}
 virtual void DoContent(EventRecord* theEvent) {}
 virtual void DoKeyDown(EventRecord* theEvent) {}
 virtual void DoActivate(Boolean becomingActive) {}
 virtual void DoUpdate(void) {}
 // file handling routines
 virtual void DoOpen(void) {};
 virtual void DoClose(void) { delete this; };
 // by default, we just delete ourself 
 // & let destructor do cleanup
 virtual void DoSave(void) {};
 virtual void DoSaveAs(void) {};
 virtual void DoRevert(void) {};
 virtual void DoPrint(void) {};
 // do standard edit menu actions
 virtual void DoUndo(void) {};
 virtual void DoCut(void) {};
 virtual void DoCopy(void) {};
 virtual void DoPaste(void) {};
 virtual void DoClear(void) {};
 virtual void DoSelectAll(void) {};

 // idle time routines: 
 // you can use these to do cursor handling,
 // TE caret blinking, marquee effects, etc...
 virtual void DoIdle(void) {};
 virtual unsigned long CalcIdle(void) 
 { return kMaxSleepTime; };
 // by default, we don’t need idle
 virtual void AdjustCursor(Point where) {};
 // where is in local coords

 // query state of document - 
 // useful for adjusting menu state
 virtual Boolean HaveUndo(void) { return false; };
 virtual Boolean HaveSelection(void) { return false; };
 virtual Boolean HavePaste(void) { return false; };
 virtual Boolean CanClose(void) { return true; };
 virtual Boolean CanSave(void) { return false; };
 virtual Boolean CanSaveAs(void) { return true; };
 virtual Boolean CanRevert(void) { return false; };
 virtual Boolean CanPrint(void) { return false; };

 // utility routine to get window pointer for document
 inline WindowPtr GetDocWindow(void) 
 { return fDocWindow; }
};
Listing 2: Our MacTutorGrow class, overriding some of TDocument’s methods

// MacTutorGrow definitions
// subclass of MacTutorDocument that adds 
// scroll, grow and zoom

#define rVScroll 128 /* vertical scrollbar control */
#define rHScroll 129 /* horizontal scrollbar control*/
class TMacTutorGrow : public TMacTutorDocument {
 
  private:
 ControlHandle fDocVScroll; // vertical scrollbar
 ControlHandle fDocHScroll; // horizontal scrollbar
 void AdjustViewRect(void);
 void ResizeWindow(void);
 void AdjustScrollSizes(void);
 void AdjustScrollbars(Boolean needsResize);
 void AdjustScrollValues(Boolean mustRedraw);
 void DrawWindow(void);
 
  public:
 TMacTutorGrow(short resID,StringPtr s);
 ~TMacTutorGrow(void);
 // new methods, override previous ones
 void DoZoom(short partCode);
 void DoGrow(EventRecord* theEvent);
 void DoContent(EventRecord* theEvent);
 void DoUpdate(void);
 void DoActivate(Boolean becomingActive);
};

#include <Types.h>
#include <QuickDraw.h>
#include <Fonts.h>
#include <Events.h>
#include <OSEvents.h>
#include <Controls.h>
#include <Windows.h>
#include <Menus.h>
#include <TextEdit.h>
#include <Dialogs.h>
#include <Desk.h>
#include <Scrap.h>
#include <ToolUtils.h>
#include <Memory.h>
#include <SegLoad.h>
#include <Files.h>
#include <OSUtils.h>
#include <Traps.h>
#include <StdLib.h>

#include “TDocument.h”
#include “TApplication.h”
#include “MacTutorApp.h”
#include “MacTutorDoc.h”
#include “MacTutorGrow.h”

// consts for scroll bar and grow box size adjustment
const short kScrollbarAdjust = 15;
const short kGrowboxAdjust = 15;
const short kScrollbarWidth = 16;
const short kScrollTweek = 2;
const short kControlVisible = 0xFF;

extern “C” { 
 // prototypes for functions outside of classes
 void CommonAction (ControlHandle control,
 short* amount);
 pascal void VActionProc(ControlHandle control,
 short part);
 pascal void HActionProc(ControlHandle control,
 short part);
};

// methods for the MacTutorGrow class

// create and delete document windows
// override methods from MacTutorDocument
// The base class creates the window and 
// sets the display string.
// We try to get the controls, and display an error if we can’t
//
TMacTutorGrow::TMacTutorGrow
 (short resID, StringPtr s) : (resID,s)
{
 Boolean good;
 fDocVScroll = GetNewControl(rVScroll, fDocWindow);
 good = (fDocVScroll != nil);
 if ( good)
   {
 fDocHScroll = GetNewControl (rHScroll, fDocWindow);
 good = (fDocHScroll != nil);
   }
 if ( good )// good? -- set & draw the controls
   {
 AdjustScrollValues(true);
   }
 else
   {
 // tell user we failed
 HideWindow(fDocWindow);
 AlertUser(kErrStrings,eNoWindow); 
   }
}

TMacTutorGrow::~TMacTutorGrow(void)
{
 HideWindow(fDocWindow);
 if ( fDocVScroll != nil )
   DisposeControl(fDocVScroll);
 if ( fDocHScroll != nil )
   DisposeControl(fDocHScroll);
}

void TMacTutorGrow::DoUpdate(void)
{

 BeginUpdate(fDocWindow);
 if ( ! EmptyRgn(fDocWindow->visRgn) ) 
   DrawWindow();
 EndUpdate(fDocWindow);
}

void TMacTutorGrow::DrawWindow(void)
{
 Rect tRect;
 SetPort(fDocWindow);
 tRect = fDocWindow->portRect;
 tRect.bottom = tRect.bottom - kScrollbarAdjust;
 tRect.right = tRect.right - kScrollbarAdjust;
 EraseRect(&tRect);
 
 MoveTo(100,100);
 TextSize(18); TextFont(monaco);
 DrawString(fDisplayString);
 DrawControls(fDocWindow);
 DrawGrowIcon(fDocWindow);

} // DrawWindow

void TMacTutorGrow::DoActivate
 (Boolean becomingActive)
{
 if ( becomingActive )
   {
 Rect growRect;
 Rect tRect;

 /* the controls must be redrawn on activation: */
 (*fDocVScroll)->contrlVis = kControlVisible;
 (*fDocHScroll)->contrlVis = kControlVisible;
 // copy rectangles to avoid 
 // unsafe object field references!
 tRect = (*fDocVScroll)->contrlRect; 
 InvalRect(&tRect);
 tRect = (*fDocHScroll)->contrlRect; 
 InvalRect(&tRect);
 // the growbox needs to be redrawn on activation:
 growRect = fDocWindow->portRect;
 // adjust for the scrollbars
 growRect.top = growRect.bottom - kScrollbarAdjust;
 growRect.left = growRect.right - kScrollbarAdjust;
 InvalRect(&growRect);
   }
 else
   {    
 /* the controls must be hidden on deactivation: */
 HideControl(fDocVScroll);
 HideControl(fDocHScroll);
 // we draw grow icon immediately, 
 // since we deactivate controls
 // immediately, and the update delay looks funny
 DrawGrowIcon(fDocWindow);
   }
}

void TMacTutorGrow::AdjustScrollSizes(void)
{
 MoveControl(fDocVScroll, 
 fDocWindow->portRect.right - kScrollbarAdjust, 
 fDocWindow->portRect.top);
 SizeControl(fDocVScroll, kScrollbarWidth,
 fDocWindow->portRect.bottom - 
 fDocWindow->portRect.top -
 kGrowboxAdjust + kScrollTweek);
 MoveControl(fDocHScroll, 
 fDocWindow->portRect.left, fDocWindow->portRect.bottom - kScrollbarAdjust);
 SizeControl(fDocHScroll,
 fDocWindow->portRect.right - 
 fDocWindow->portRect.left -
 kGrowboxAdjust + kScrollTweek,
 kScrollbarWidth);
} // AdjustScrollSizes

void TMacTutorGrow::AdjustScrollbars
 (Boolean needsResize)
{
 (*fDocVScroll)->contrlVis = 0;
 (*fDocHScroll)->contrlVis = 0;
 if ( needsResize )  AdjustScrollSizes();
 AdjustScrollValues(needsResize);
 (*fDocVScroll)->contrlVis = 0xff;
 (*fDocHScroll)->contrlVis = 0xff;
} // AdjustScrollbars 

void TMacTutorGrow::AdjustScrollValues
 (Boolean mustRedraw)
// always reset the controls to the same values. In reality,
// you would calculate reasonable control values here 
// from the contents of your document
{
 SetCtlMin(fDocVScroll,0);
 SetCtlMax(fDocVScroll,1000);
 SetCtlValue(fDocVScroll,500);
 SetCtlMin(fDocHScroll,0);
 SetCtlMax(fDocHScroll,1000);
 SetCtlValue(fDocHScroll,500);
 if ( mustRedraw )
 {
 ShowControl(fDocVScroll);
 ShowControl(fDocHScroll);
 }
} // AdjustScrollValues

void TMacTutorGrow::DoZoom(short partCode)
{
 Rect tRect;

 tRect = fDocWindow->portRect;
 EraseRect(&tRect);
 ZoomWindow(fDocWindow, partCode, 
 fDocWindow == FrontWindow());
 AdjustScrollbars(true);  // adjust, redraw anyway 
 InvalRect(&tRect);// invalidate the whole content 
 // revalidate scroll bars 
 tRect = (*fDocVScroll)->contrlRect;
 ValidRect(&tRect);
 tRect = (*fDocHScroll)->contrlRect;
 ValidRect(&tRect);
}

void TMacTutorGrow::DoGrow(EventRecord* theEvent)
{
 long growResult;
 Rect tRect, tRect2;
 
 tRect = qd.screenBits.bounds;
 tRect.left = kMinDocDim;
 tRect.top = kMinDocDim;
 tRect2 = fDocWindow->portRect;
 tRect2.bottom = tRect2.bottom - kScrollbarAdjust;
 tRect2.right = tRect2.right - kScrollbarAdjust;
 growResult = GrowWindow
 (fDocWindow, theEvent->where, &tRect);
 // see if it really changed size 
 if ( growResult != 0 )
   {
 SizeWindow (fDocWindow, 
 LoWrd(growResult), HiWrd(growResult), true);
 AdjustScrollbars(true);
 // calculate & validate the region 
 // that hasn’t changed so it won’t get redrawn
 // Note: we copy rectangles so that we don’t 
 // take address of object fields.
 tRect = fDocWindow->portRect; 
 (void) SectRect(&tRect, &tRect2, &tRect2);
 InvalRect(&tRect); ValidRect(&tRect2);
 tRect2 = (*fDocVScroll)->contrlRect;
 ValidRect(&tRect2);
 tRect2 = (*fDocHScroll)->contrlRect; 
 ValidRect(&tRect2);
 // redraw grow icon
 tRect.top = tRect.bottom - kScrollbarAdjust;
 tRect.left = tRect.right - kScrollbarAdjust;
 InvalRect(&tRect);
   }
}

void TMacTutorGrow::DoContent(EventRecord* theEvent)
{
 Point mouse;
 ControlHandle control;
 short part, value;
 Rect tRect;
 
 SetPort(fDocWindow);
 mouse = theEvent->where;
 GlobalToLocal(&mouse);
 
 tRect = fDocWindow->portRect; 
 tRect.bottom = tRect.bottom - kScrollbarAdjust;
 tRect.right = tRect.right - kScrollbarAdjust;
 part = FindControl(mouse, fDocWindow, &control);
 switch ( part )
 {
 case 0: break;
 case inThumb:
 value = GetCtlValue(control);
 part = TrackControl(control, mouse, nil);
 if ( part != 0 )
 {
 value -= GetCtlValue(control);
 // value now has CHANGE in value; 
 // if value changed, scroll 
 if ( value != 0 )
 {
 if ( control == fDocVScroll )
 {
   ScrollRect (&tRect, 0, value, nil);
   SetOrigin (tRect.left,tRect.top-value);
 }
 else 
 {
   ScrollRect (&tRect, value, 0, nil);
   SetOrigin (tRect.left-value,tRect.top);
 }
 AdjustScrollSizes();
 }
 }
 break; 
 default: // clicked in an arrow, so track & scroll 
 if ( control == fDocVScroll )
 value = TrackControl
 (control, mouse, (ProcPtr) VActionProc);
 else value = TrackControl
 (control, mouse, (ProcPtr) HActionProc);
 AdjustScrollSizes();
 break;
 }
}

// routines that do not belong to any particular class. 

// Common algorithm for changing the value of a control. 
// It returns the value by which the control changed.

void CommonAction(ControlHandle control,short* amount)
{
 short value, max;
 
 value = GetCtlValue(control);
 max = GetCtlMax(control);
 *amount = value - *amount;
 if ( *amount <= 0 ) *amount = 0;
 else if ( *amount >= max ) *amount = max;
 SetCtlValue(control, *amount);
 *amount = value - *amount;
} // CommonAction 

pascal void VActionProc(ControlHandle control,short part)
{
 short  amount;
 WindowPtrwindow;
 Rect tRect;

 if ( part != 0 )
   {
 window = (*control)->contrlOwner;
 tRect = window->portRect; 
 tRect.bottom = tRect.bottom - kScrollbarAdjust;
 tRect.right = tRect.right - kScrollbarAdjust;
 switch ( part )
   {
 case inUpButton:
 case inDownButton:  // 5 pixels
 amount = 5;
 break;
 case inPageUp:  // 50 pixels
 case inPageDown:
 amount = 50;
 break;
   }
 if ((part == inDownButton) || (part == inPageDown))
 amount = -amount;  // reverse direction 
 CommonAction(control, &amount);
 if ( amount != 0 )
 {
 ScrollRect (&tRect, 0, amount, nil);
 SetOrigin(tRect.left,tRect.top-amount);
 }
   }
} // VActionProc 

pascal void HActionProc(ControlHandle control,short part)
{
 short  amount;
 WindowPtrwindow;
 Rect tRect;

 if ( part != 0 )
   {
 window = (*control)->contrlOwner;
 tRect = window->portRect; 
 tRect.bottom = tRect.bottom - kScrollbarAdjust;
 tRect.right = tRect.right - kScrollbarAdjust;
 switch ( part )
   {
 case inUpButton:
 case inDownButton:  // 5 pixels
 amount = 5;
 break;
 case inPageUp:  // 50 pixels
 case inPageDown:
 amount = 50;
 break;
   }
 if ((part == inDownButton) || (part == inPageDown))
 amount = -amount;  // reverse direction 
 CommonAction(control, &amount);
 if ( amount != 0 )
 {
 ScrollRect (&tRect, amount, 0, nil);
 SetOrigin(tRect.left-amount,tRect.top);
 }
   }
} // HActionProc 
Listing 3:  Adjustments to the MacTutorApp.cp and MacTutorApp.r files

Changes in MacTutorApp.cp:

#include “MacTutorGrow.h” 
 // insert behind the other #includes

// change the definition of the main() routine:
TMacTutorApp *gTheApplication;
int main(void)
{
 gTheApplication = new TMacTutorApp;
 if (gTheApplication == nil)return 0;
 gTheApplication->EventLoop();
 return 0;
}

// insert the following definitions to MacTutorApp.r:

#define kErrStrings 129
#define eNoMemory1
#define eNoWindow2

resource ‘CNTL’ (rVScroll, preload, purgeable) {
 {-1, 385, 236, 401},
 0, visible, 0, 0, scrollBarProc, 0, “” };

resource ‘CNTL’ (rHScroll, preload, purgeable) {
 {235, -1, 251, 386},
 0, visible, 0, 0, scrollBarProc, 0, “” };

// change the WIND definition:

resource ‘WIND’ (rDocWindow, preload, purgeable) {
 {64, 60, 314, 460},
 zoomDocProc, invisible, goAway, 0x0, 
 “MacTutor C++ demo” };

 

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

13-inch 2.7GHz Retina MacBook Pro on sale for...
B&H Photo has the 2015 13″ 2.7GHz/128GB Retina Apple MacBook Pro on sale for $150 off MSRP. Shipping is free, and B&H charges NY tax only: - 13″ 2.7GHz/128GB Retina MacBook Pro (MF839LL/A): $... Read more
13-inch 1.6GHz/256GB MacBook Air on sale for...
Newegg has the 13″ 1.6GHz/256GB MacBook Air (MMGG2LL/A) on sale for $1029.99 including free shipping. Their price is $170 off MSRP, and it’s the lowest price available for this model. Choose Newegg... Read more
Apple refurbished Apple TVs available for up...
Apple has Certified Refurbished 32GB and 64GB Apple TVs available for up to $30 off the cost of new models. Apple’s standard one-year warranty is included with each model, and shipping is free: -... Read more
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

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.