TweetFollow Us on Twitter

Color Basics
Volume9
Number11
Column TabGetting Started

Related Info: Color QuickDraw Control Panel Gestalt Manager

The Basics of Color Quickdraw

Using a color grafport and multiple monitors

By Dave Mark, MacTech Magazine Regular Contributing Author

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

Last month’s column introduced some of the basics of C++ and object programming. This month, we’ll turn our attention back to C and the Macintosh Toolbox. This month’s column introduces one of the most enjoyable parts of the Mac Toolbox, Color QuickDraw. Along the way, we’ll learn how to use Gestalt(), the Toolbox function that knows everything there is to know about the current state of your Macintosh.

So far, about the only real experience we’ve had with color has been with a program that displayed a color PICT in a window. Since any possible color work was handled by the routine DrawPicture(), we won’t even count that as color experience. This month’s program is called ColorMondrian, a colorized version of an early Mac Primer program called Mondrian.

Mondrian created a window and drew a never-ending series of randomly generated QuickDraw shapes in the window. All this was done in the plain old black-and-white of QuickDraw version 1, the version of QuickDraw that shipped with the original 128K Macintosh.

Ever since the introduction of the Macintosh II, all Macs have shipped with a new version of QuickDraw known as Color QuickDraw, also known as 8-bit Color QuickDraw. As the technology marched on, 24-bit and then 32-bit Color QuickDraw made their way into the Toolbox. For now, we’ll concentrate on the features common to all flavors of Color QuickDraw.

ColorMondrian starts by checking to see if Color QuickDraw is available on the current Macintosh. If so, it looks at each of the displays connected to the Mac (remember, your Mac can have multiple monitors hooked up at the same time!) to determine which display is the deepest, that is, can display the largest number of colors. For example, if you have an 8-bit monitor (256 simultaneous colors) and a 1-bit monitor (black and white only), ColorMondrian will identify the 8-bit monitor as the deepest.

Next, ColorMondrian will open a color window on the specified monitor and start drawing randomly sized QuickDraw shapes in randomly selected colors. If your deepest monitor is gray-scale, don’t worry. Shades of gray count as colors too!

Let’s get started. As has been the case of late, we’ll get the program running this month, then walk through the code next month.

Creating the ColorMondrian Resources

Start by creating a folder labeled ColorMondrian in your Development folder. Next, launch ResEdit and create a new file called ColorMondrian.Π.rsrc in the ColorMondrian folder.

Create a new MBAR resource with a resource id of 128. Enter the MENU ids 128, 129, 130, and 131 as shown in Figure 1.

Figure 1. The MBAR resource.

Next, create four MENU resources. The first, having a resource id of 128, will match the menu shown in Figure 2. It’s an • menu with a single item, About ColorMondrian....

Figure 2. The • menu.

The second menu should match the one shown in Figure 3. The title is File with a single item, Be sure to add the command-key equivalent, Q.

Figure 3. The File menu.

The third menu should match the one shown in Figure 4. It’s a standard Edit menu with a single separator line and all the standard command-key equivalents.

Figure 4. The Edit menu.

The fourth menu consists of a title and no items. The title is Devices, as shown in Figure 5.

Figure 5. The Devices menu.

Next you’ll create two ALRT resources along with their respective DITLs, one for displaying error messages, and one to bring up when About ColorMondrian... is selected from the • menu. First the error ALRT. Create an ALRT resource with a resource id of 128, using the sizing info in Figure 6.

Figure 6. Specifications for the error ALRT.

Double-click on the alert window to bring up a DITL editor. Create two items, an OK button (Figure 7) and a static text item (Figure 8).

Figure 7. Specifications for the OK button.

Figure 8. Specification for the static text item.

Now create a second ALRT with a resource id of 129. using the sizing info in Figure 9.

Figure 9. Specifications for the about ALRT.

Double-click on the alert window to bring up a DITL editor. Create two items, an OK button (Figure 10) and a static text item (Figure 11).

Figure 10. Specifications for the OK button.

Figure 11. Specification for the static text item.

Creating the ColorMondrian Project

Quit ResEdit, being sure to Save your changes. Now launch THINK C and create a new project named ColorMondrian.Π in the ColorMondrian folder. Add the MacTraps and ANSI libraries to the project (In THINK 6, ANSI is inthe Standard Libraries folder - add it in its own segment), then create a new source code window. Save the window as ColorMondrian.c and Add it to the project as well.

Type in this source code, being sure to save periodically:

/* 1 */

#include <stdio.h>
#include <GestaltEqu.h>


#define kMBARResID 128
#define kErrorAlertID128
#define kAboutALRTid 129

#define kSleep   0L

#define kAutoStorage NULL
#define kVisible true
#define kWindowTitle "\pColor Mondrian"
#define kMoveToFront (WindowPtr)-1
#define kNoGoAwayfalse
#define kNULLRefCon60L

#define mApple   128
#define iAbout   1

#define mFile    129
#define iQuit    1

#define mDevice  131

#define kWindowMargin5

#define kRandomUpperLimit 32768

#define kEmptyString "\p"
#define kNULLFilterProc   NULL

/*************/
/*  Globals  */
/*************/

Boolean gDone;

/***************/
/*  Functions  */
/***************/

void    ToolboxInit( void );
void    MenuBarInit( void );
void    CreateWindow( GDHandle device );
void    EventLoop( void );
void    DoEvent( EventRecord *eventPtr );
void    HandleMouseDown( EventRecord *eventPtr );
void    HandleMenuChoice( long menuChoice );
void    HandleAppleChoice( short item );
void    HandleFileChoice( short item );
void    HandleDeviceChoice( short item );
Boolean HasColorQD( void );
GDHandleGetDeepestDevice( void );
short   GetDeviceDepth( GDHandle device );
void    DrawRandomRect( void );
void    RandomColor( RGBColor *colorPtr );
void    RandomRect( Rect *rectPtr );
short Randomize( short range );
void    DoError( Str255 errorString );


/****************** main ***************************/

void main( void )
{
 ToolboxInit();
 
 if ( ! HasColorQD() )
 DoError( 
 "\pThis machine doesn't support Color Quickdraw!" );
 
 MenuBarInit();
 
 CreateWindow( GetDeepestDevice() );
 
 EventLoop();
}


/****************** ToolboxInit *********************/

void ToolboxInit( void )
{
 InitGraf( &thePort );
 InitFonts();
 InitWindows();
 InitMenus();
 TEInit();
 InitDialogs( nil );
 InitCursor();
}


/****************** MenuBarInit ***********************/

void MenuBarInit( void )
{
 Handle menuBar;
 MenuHandle menu;
 GDHandle device, deepestDevice;
 Str255 itemStr;
 short  curDeviceNumber = 1;
 
 menuBar = GetNewMBar( kMBARResID );
 SetMenuBar( menuBar );

 menu = GetMHandle( mApple );
 AddResMenu( menu, 'DRVR' );
 
 menu = GetMHandle( mDevice );
 
 deepestDevice = GetDeepestDevice();
 
 device = GetDeviceList();
 
 while ( device != NULL )
 {
 itemStr[0] = 10;
 sprintf( (char *)(&(itemStr[1])), "0x%08lX", 
 (unsigned long)device );
 AppendMenu( menu, itemStr );
 
 if ( device == deepestDevice )
 CheckItem( menu, curDeviceNumber, true );
 
 device = GetNextDevice( device );
 curDeviceNumber++;
 }
 
 DrawMenuBar();
}


/****************** CreateWindow ***********************/

void CreateWindow( GDHandle device )
{
 WindowPtrwindow;
 Rect   wBounds;
 
 wBounds = (**device).gdRect;
 
 if ( device == GetMainDevice() )
 wBounds.top += GetMBarHeight();
 
 InsetRect( &wBounds, kWindowMargin, kWindowMargin );
 
 window = NewCWindow( kAutoStorage, &wBounds, kWindowTitle, 
 kVisible, altDBoxProc, kMoveToFront, 
 kNoGoAway, kNULLRefCon );
 
 if ( window == nil )
 {
 DoError( "\pCouldn't create window!" );
 }
 else
 {
 ShowWindow( window );
 SetPort( window );
 }
}


/******************************** EventLoop *********/

void EventLoop( void )
{
 EventRecordevent;
 
 GetDateTime( (unsigned long *)(&randSeed) );
 
 gDone = false;
 while ( gDone == false )
 {
 if ( WaitNextEvent( everyEvent, &event, kSleep, nil ) )
 DoEvent( &event );
 
 DrawRandomRect();
 }
}


/************************************* DoEvent *********/

void DoEvent( EventRecord *eventPtr )
{
 char   theChar;
 
 switch ( eventPtr->what )
 {
 case mouseDown: 
 HandleMouseDown( eventPtr );
 break;
 case keyDown:
 case autoKey:
 theChar = eventPtr->message & charCodeMask;
 if ( (eventPtr->modifiers & cmdKey) != 0 ) 
 HandleMenuChoice( MenuKey( theChar ) );
 break;
 }
}


/****************************** HandleMouseDown *********/

void HandleMouseDown( EventRecord *eventPtr )
{
 WindowPtrwindow;
 short  thePart;
 long   menuChoice;
 
 thePart = FindWindow( eventPtr->where, &window );

 switch ( thePart )
 {
 case inMenuBar:
 menuChoice = MenuSelect( eventPtr->where );
 HandleMenuChoice( menuChoice );
 break;
 case inSysWindow : 
 SystemClick( eventPtr, window );
 break;
 }
}


/****************** HandleMenuChoice ***********************/

void HandleMenuChoice( long menuChoice )
{
 short  menu;
 short  item;
 
 if ( menuChoice != 0 )
 {
 menu = HiWord( menuChoice );
 item = LoWord( menuChoice );
 switch ( menu )
 {
 case mApple:
 HandleAppleChoice( item );
 break;
 case mFile:
 HandleFileChoice( item );
 break;
 case mDevice:
 HandleDeviceChoice( item );
 break;
 }
 HiliteMenu( 0 );
 }
}


/****************** HandleAppleChoice ***********************/

void HandleAppleChoice( short item )
{
 MenuHandle appleMenu;
 Str255 accName;
 short  accNumber;
 
 switch ( item )
 {
 case iAbout:
 NoteAlert( kAboutALRTid, kNULLFilterProc );
 break;
 default:
 appleMenu = GetMHandle( mApple );
 GetItem( appleMenu, item, accName );
 accNumber = OpenDeskAcc( accName );
 break;
 }
}


/****************** HandleFileChoice ***********************/

void HandleFileChoice( short item )
{
 switch ( item )
 {
 case iQuit :
 gDone = true;
 break;
 }
}


/****************** HandleDeviceChoice **********************/

void HandleDeviceChoice( short item )
{
/* Try this: 
 Modify the program so that when a device is selected
 from the Device menu, the current window gets closed and a
 new window is opened on the selected device. Be careful when
 you translate the menu item back into an address. Debug your
 program thoroughly before you try to use the address as an
 address. You don't want to accidentally reformat your hard 
 drive, right?
 
 Also, don't forget to update the check mark!
*/
}


/****************** HasColorQD *****************/

Boolean HasColorQD( void )
{
 unsigned char   version[ 4 ];
 OSErr  err;
 
 err = Gestalt( gestaltQuickdrawVersion, (long *)version );
 
 if ( err != noErr )
 {
 SysBeep( 10 );  /*  Error calling Gestalt!!!  */
 ExitToShell();
 }
 
 if ( version[ 2 ] > 0 )
 return( true );
 else
 return( false );
}


/****************** GetDeepestDevice *****************/

GDHandle GetDeepestDevice( void )
{
 GDHandle curDevice, maxDevice = NULL;
 short  curDepth, maxDepth = 0;
 
 curDevice = GetDeviceList();
 
 while ( curDevice != NULL )
 {
 curDepth = GetDeviceDepth( curDevice );
 
 if ( curDepth > maxDepth )
 {
 maxDepth = curDepth;
 maxDevice = curDevice;
 }

 curDevice = GetNextDevice( curDevice );
 }
 
 return( maxDevice );
}


/****************** GetDeviceDepth *****************/

short GetDeviceDepth( GDHandle device )
{
 PixMapHandle  screenPixMapH;
 
 screenPixMapH = (**device).gdPMap;
 
 return( (**screenPixMapH).pixelSize );
}


/****************** DrawRandomRect *****************/

void DrawRandomRect( void )
{
 Rect   randomRect;
 RGBColor color;
 
 RandomRect( &randomRect );
 RandomColor( &color );
 RGBForeColor( &color );
 PaintOval( &randomRect );
}


/****************** RandomColor *********************/

void RandomColor( RGBColor *colorPtr )
{
 colorPtr->red = Random() + 32767;
 colorPtr->blue = Random() + 32767;
 colorPtr->green = Random() + 32767;
}


/****************** RandomRect *********************/

void RandomRect( Rect *rectPtr )
{
 WindowPtrwindow;

 window = FrontWindow();
 
 rectPtr->left = Randomize( window->portRect.right
 - window->portRect.left );
 rectPtr->right = Randomize( window->portRect.right
 - window->portRect.left );
 rectPtr->top = Randomize( window->portRect.bottom
 - window->portRect.top );
 rectPtr->bottom = Randomize( window->portRect.bottom
 - window->portRect.top );
}


/****************** Randomize **********************/

short Randomize( short range )
{
 long   randomNumber;
 
 randomNumber = Random();
 
 if ( randomNumber < 0 )
 randomNumber *= -1;
 
 return( (randomNumber * range) / kRandomUpperLimit );
}


/***************** DoError ********************/

void DoError( Str255 errorString )
{
 ParamText( errorString, kEmptyString, 
 kEmptyString, kEmptyString );
 
 StopAlert( kErrorAlertID, kNULLFilterProc );
}

Running ColorMondrian

Save your changes, then select Run from the Project menu. If ColorQuickDraw is not available on your machine (unlikely), an error message will appear. Otherwise, a window, similar to the one shown in Figure 12, will appear on the monitor with the deepest pixel settings.

Figure 12. ColorMondrian in action.

Bring the Finder to the front by selecting Finder from the applications menu at the right end of the menu bar. Notice that ColorMondrian stops dead in its tracks. Go back to ColorMondrian and select Quit from the File menu. Back in THINK C, select Set Project Type... from the Project menu. Set the value in the SIZE Flags field to 5880 (you don’t have to fiddle with the popup menu - you can just type the number in the field).

You’ve just set ColorMondrian up to continue processing events even when it is running in the background. To prove this, select Run from the Project menu, then bring the Finder to the front again. This time, ColorMondrian will continue running, even in the background.

Next, click on the Devices menu. A hex address will appear for every monitor attached to your Macintosh. A check-mark will appear next to the address representing the monitor with the deepest settings. You’ll find out what the address or addresses are for in next month’s column. Figure 13 shows my Devices menu. As you can see, I’ve got two monitors attached to my Mac.

Figure 13. My devices menu.

If you’ve got more than one monitor on your machine, try using the Monitors control panel to set one monitor to 1-bit and the other to a deeper setting. Run ColorMondrian. The window should appear on the deeper monitor. Now switch the settings so that the second monitor has the deeper settings. Run ColorMondrian again. Now the window should appear on the other monitor.

Till Next Month...

Next, month we’ll get into all the theory behind ColorMondrian. Till then, take a look through the code, then read up on Color QuickDraw in the pages of Inside Macintosh or on your Mac, courtesy of THINK Reference.

 

Community Search:
MacTech Search:

Software Updates via MacUpdate

Coda 2.5.11 - One-window Web development...
Coda is a powerful Web editor that puts everything in one place. An editor. Terminal. CSS. Files. With Coda 2, we went beyond expectations. With loads of new, much-requested features, a few surprises... Read more
Bookends 12.5.7 - Reference management a...
Bookends is a full-featured bibliography/reference and information-management system for students and professionals. Access the power of Bookends directly from Mellel, Nisus Writer Pro, or MS Word (... Read more
Maya 2016 - Professional 3D modeling and...
Maya is an award-winning software and powerful, integrated 3D modeling, animation, visual effects, and rendering solution. Because Maya is based on an open architecture, all your work can be scripted... Read more
RapidWeaver 6.2.3 - Create template-base...
RapidWeaver is a next-generation Web design application to help you easily create professional-looking Web sites in minutes. No knowledge of complex code is required, RapidWeaver will take care of... Read more
MacFamilyTree 7.5.2 - Create and explore...
MacFamilyTree gives genealogy a facelift: it's modern, interactive, incredibly fast, and easy to use. We're convinced that generations of chroniclers would have loved to trade in their genealogy... Read more
Paragraphs 1.0.1 - Writing tool just for...
Paragraphs is an app just for writers. It was built for one thing and one thing only: writing. It gives you everything you need to create brilliant prose and does away with the rest. Everything in... Read more
BlueStacks App Player 0.9.21 - Run Andro...
BlueStacks App Player lets you run your Android apps fast and fullscreen on your Mac. Version 0.9.21: Note: Now requires OS X 10.8 or later running on a 64-bit Intel processor. Initial stable... Read more
Tweetbot 2.0.2 - Popular Twitter client....
Tweetbot is a full-featured OS X Twitter client with a lot of personality. Whether it's the meticulously-crafted interface, sounds and animation, or features like multiple timelines and column views... Read more
Apple iBooks Author 2.3 - Create and pub...
Apple iBooks Author helps you create and publish amazing Multi-Touch books for iPad. Now anyone can create stunning iBooks textbooks, cookbooks, history books, picture books, and more for iPad. All... Read more
NeoOffice 2014.12 - Mac-tailored, OpenOf...
NeoOffice is a complete office suite for OS X. With NeoOffice, users can view, edit, and save OpenOffice documents, PDF files, and most Microsoft Word, Excel, and PowerPoint documents. NeoOffice 3.x... Read more

Rage of Bahamut is Giving Almost All of...
The App Store isn't what it used to be back in 2012, so it's not unexpected to see some games changing their structures with the times. Now we can add Rage of Bahamut to that list with the recent announcement that the game is severely cutting back... | Read more »
Adventures of Pip (Games)
Adventures of Pip 1.0 Device: iOS iPhone Category: Games Price: $4.99, Version: 1.0 (iTunes) Description: ** ONE WEEK ONLY — 66% OFF! *** “Adventures of Pip is a delightful little platformer full of charm, challenge and impeccable... | Read more »
Divide By Sheep - Tips, Tricks, and Stre...
Who would have thought splitting up sheep could be so involved? Anyone who’s played Divide by Sheep, that’s who! While we’re not about to give you complete solutions to everything (because that’s just cheating), we will happily give you some... | Read more »
NaturalMotion and Zynga Have Started Tea...
An official sequel to 2012's CSR Racing is officially on the way, with Zynga and NaturalMotion releasing a short teaser trailer to get everyone excited. Well, as excited as one can get from a trailer with no gameplay footage, anyway. [Read more] | Read more »
Grab a Friend and Pick up Overkill 3, Be...
Overkill 3 is a pretty enjoyable third-person shooter that was sort of begging for some online multiplayer. Fortunately the begging can stop, because its newest update has added an online co-op mode. [Read more] | Read more »
Scanner Pro's Newest Update Adds Au...
Scanner Pro is one of the most popular document scanning apps on iOS, thanks in no small part to its near-constant updates, I'm sure. Now we're up to update number six, and it adds some pretty handy new features. [Read more] | Read more »
Heroki (Games)
Heroki 1.0 Device: iOS Universal Category: Games Price: $7.99, Version: 1.0 (iTunes) Description: CLEAR THE SKIES FOR A NEW HERO!The peaceful sky village of Levantia is in danger! The dastardly Dr. N. Forchin and his accomplice,... | Read more »
Wars of the Roses (Games)
Wars of the Roses 1.0 Device: iOS Universal Category: Games Price: $4.99, Version: 1.0 (iTunes) Description: | Read more »
TapMon Battle (Games)
TapMon Battle 1.0 Device: iOS Universal Category: Games Price: $.99, Version: 1.0 (iTunes) Description: It's time to battle!Tap! Tap! Tap! Try tap a egg to hatch a Tapmon!Do a battle with another tapmons using your hatched tapmons! *... | Read more »
Alchemic Dungeons (Games)
Alchemic Dungeons 1.0 Device: iOS Universal Category: Games Price: $.99, Version: 1.0 (iTunes) Description: ### Release Event! ### 2.99$->0.99$ for limited time! ### Roguelike Role Playing Game! ### Alchemic Dungeons is roguelike... | Read more »

Price Scanner via MacPrices.net

Canon PIXMA MG3620 Wireless Inkjet All-in-One...
Canon U.S.A., Inc. has announced the PIXMA MG3620 Wireless (1) Inkjet All-in-One (AIO) printer for high-quality photo and document printing. Built with convenience in mind for the everyday home user... Read more
July 4th Holiday Weekend 13-inch MacBook Pro...
Save up to $150 on the purchase of a new 2015 13″ Retina MacBook Pro at the following resellers this weekend. Shipping is free with each model: 2.7GHz/128GB MSRP $1299 2.7GHz/... Read more
27-inch 3.5GHz 5K iMac on sale for $2149, sav...
Best Buy has the 27″ 3.5GHz 5K iMac on sale for $2149.99. Choose free shipping or free local store pickup (if available). Sale price for online orders only, in-store prices may vary. Their price is $... Read more
Apple now offering refurbished 2015 11-inch...
The Apple Store is now offering Apple Certified Refurbished 2015 11″ MacBook Airs as well as 13″ MacBook Airs (the latest models), available for up to $180 off the cost of new models. An Apple one-... Read more
15-inch 2.5GHz Retina MacBook Pro on sale for...
Amazon.com has the 15″ 2.5GHz Retina MacBook Pro on sale for $2274 including free shipping. Their price is $225 off MSRP, and it’s the lowest price available for this model. Read more
Finally Safe To Upgrade To Yosemite’?
The reason I’ve held back from upgrading my MacBook Air from OS X 10.9 Mavericks to 10.10 Yosemite for nearly a year isn’t just procrastination. Among other bugs reported, there have been persistent... Read more
Logo Pop Free Vector Logo Design App For OS X...
128bit Technologies has released of Logo Pop Free 1.2 for Mac OS X, a vector based, full-fledged, logo design app available exclusively on the Mac App Store for the agreeable price of absolutely free... Read more
21-inch 1.4GHz iMac on sale for $999, save $1...
B&H Photo has new 21″ 1.4GHz iMac on sale for $999 including free shipping plus NY sales tax only. Their price is $100 off MSRP. Best Buy has the 21″ 1.4GHz iMac on sale for $999.99 on their... Read more
16GB iPad mini 3 on sale for $339, save $60
B&H Photo has the 16GB iPad mini 3 WiFi on sale for $339 including free shipping plus NY tax only. Their price is $60 off MSRP. Read more
Save up to $40 on iPad Air 2, NY tax only, fr...
B&H Photo has iPad Air 2s on sale for up to $40 off MSRP including free shipping plus NY sales tax only: - 16GB iPad Air 2 WiFi: $489 $10 off - 64GB iPad Air 2 WiFi: $559 $40 off - 128GB iPad Air... Read more

Jobs Board

*Apple* Solutions Consultant - Retail Sales...
**Job Summary** As an Apple Solutions Consultant (ASC) you are the link between our customers and our products. Your role is to drive the Apple business in a retail Read more
*Apple* Fulfillment Operations Execution Ana...
**Job Summary** The AMR Apple Fulfillment Operations Team is seeking a talented team player to drive the Apple Online Store (AOS) fulfillment performance to ensure a Read more
Localization Producer - *Apple* HR and Reta...
…project manager to support the Retail Globalization team. You will participate in Apple exponential inte ational growth and drive global project initiatives for the Read more
*Apple* Online Store UAT Lead - Apple (Unite...
**Job Summary** The Apple Online Store is a fast paced and ever evolving business environment. A UAT lead in this organization is able to have a direct impact on one of Read more
Senior Payments Security Manager - *Apple*...
**Job Summary** Apple , Inc. is looking for a highly motivated, innovative and hands-on senior payments security manager to join the Apple Pay security team. You will Read more
All contents are Copyright 1984-2011 by Xplain Corporation. All rights reserved. Theme designed by Icreon.