TweetFollow Us on Twitter

Sprocket Menus 1
Volume Number:11
Issue Number:5
Column Tag:Getting Started

Sprocket Menus, Part 1

By Dave Mark, MacTech Magazine Regular Contributing Author

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

My February ’93 Getting Started column featured a program called MenuMaster. MenuMaster constructed a menu bar consisting of four menus: The traditional Apple, File, and Edit menus, as well as a special Options menu (Figure 1). Selecting the first item changes it from Change My Name to Change Me Back Again. Selecting the first item again changes it back to Change My Name.

Selecting Disable Me disables the second item and enables the third item. If you then select the newly enabled Enable Previous Item, it gets disabled and Disable Me is reenabled.

If you select Add Extra Menu, a new menu is inserted in the menu bar and Add Extra Menu is disabled. The new menu, titled Extra Menu, features a single item, Delete Me. Selecting Delete Me deletes the extra menu from the menu bar and reenables Add Extra Menu.

Finally, selecting Append Item adds an extra item (Can’t Delete Me...) to the end of the menu. As its names implies, there’s no way to delete this extra item.

Fig. 1. MenuMaster’s Options menu.

A Sprocket Version of MenuMaster

This month we’re going to use Sprocket to implement most of MenuMaster’s functionality. We’ll skip the ability to append an item to the end of a menu for two reasons. First, appending a single item to the end of a menu just isn’t done that often and isn’t particularly useful. More importantly (and probably for the same reason), Sprocket doesn’t give you an easy way to append a new item to a menu.

If you come up with a good reason to add this functionality to Sprocket (or if you have any comments or bugs to report), send e-mail to sprocket@hax.com.

As I mentioned last month, Sprocket based its menu-handling model on that used by OpenDoc. At the heart of this model is a replacement for the MENU resource type. A CMNU resource is just like a MENU resource, with one important addition. Each menu item features a command number. You’ll use this command number to refer to the item, instead of the more traditional method of specifying the menu the item belongs to, along with the item’s position in the menu (e.g., menu 129, item 4).

Figure 2. The CMNU resource, featuring a Cmd-Num field for each menu item.

Check out the ResEdit snapshot in Figure 2. It shows the CMNU resource that represents our new Options menu. The first menu item, Change My Name, is selected. The command number for this item is 1000. When the user selects this item, Sprocket will pass the associated command number (in this case, 1000) as a parameter to the routine HandleMenuCommand() (it’s in the file SprocketStarter.cp). Instead of creating a separate item dispatch routine for each menu (HandleAppleMenu(), HandleFileMenu(), etc.), you’ll create a single switch statement containing cases for all your commands.

Sprocket automatically creates a menu bar at application startup. In C++ terms, Sprocket constructs a TMenuBar object, which is implemented in the files TMenuBar.cp and TMenuBar.h. Here’s the TMenuBar class definition:

class TMenuBar
 {
public:
Resource ('MBAR' and 'CMNU') Utilities
 
 OSErr  GetNewMenuBar(short whichMBAR);
 MenuRefGetMenuFromCMNU(short whichMenu);
Menu command mapping functions

 MenuCommandID   GetCommand(MenuID menu, MenuItemID item);
 void   GetMenuAndItem(MenuCommandID commandNum, 
 MenuID * returnedMenu, MenuItemID * returnedItem);
 OSErr  RegisterCommand(  MenuCommandID commandNum, 
   MenuID menu, MenuItemID item);
 OSErr  UnregisterCommand(MenuCommandID commandNum);
Menu enable/disable routines for menu items

 void   EnableCommand(MenuCommandID commandNum, 
 Boolean enable);
 void   EnableAndCheckCommand(MenuCommandID commandNum, 
 Boolean enable, Boolean check);
 void   GetItemString(MenuCommandID commandNum,
 StringPtr itemString);
 void   SetItemString(MenuCommandID commandNum,
 StringPtr itemString);
helpful utility functions
 void   HideMenuBar();
 void   ShowMenuBar();
 void   RedrawIfNeeded();
 void   Invalidate();
 void   Validate();

private:
"globals"
 static Boolean  fgMenuBarNeedsRedraw;
 static Boolean  fgMenuBarHidden;
mapping tables
 TMenuCommandTable fCommandTable;
 TMenuItemTable  fMenuItemTable;
internal methods
 MenuHandle GetMenuHandleAndItemFromCommand(
 MenuCommandID commandNum, 
 MenuID *menu,MenuItemID *item);
 };

The first member function, GetNewMenuBar() uses the specified MBAR resource to build a new menu bar. Though this version of Sprocket only creates a single menu bar, this might not be the case in the future. For now, a pointer to the menu bar object is stored in the global gMenuBar. Take a minute to open up the file SprocketMain.cp and check out the code around line 363. This is where Sprocket creates the TMenuBar object based on the MBAR resource in SprocketStarter.rsrc.

The member function GetMenuFromCMNU() loads a CMNU resource and walks through it, one item at a time. It builds a traditional menu structure, passing each item’s command number to the RegisterCommand() member function, which adds the command to Sprocket’s menu command table. If you are going to take advantage of Sprocket’s menu command mechanism, you must register your menu item commands. If you base your menus on a CMNU resource, GetMenuFromCMNU() will register your menu items automatically. If the menus in your MBAR resource correspond to a CMNU resource, Sprocket will register the menu items automatically.

If you don’t want to use a CMNU resource, you can still add and delete your menus to and from the global menu bar yourself. For example, since a font or size menu will have a dynamic number of items, the CMNU resource just doesn’t make sense. We’ll look at that process in a future column.

The member function GetCommand() takes a menu and item ID and returns the associated command. GetMenuAndItem() takes a command and returns the associated menu and item ID.

If you want to delete a menu whose commands have been registered, you can use the UnregisterCommand() member function to, one-at-a-time, unregister the commands in that menu. Otherwise, you’ll orphan commands in the command table.

EnableCommand() and EnableAndCheckCommand() let you enable, disable, check, and uncheck a menu command. GetItemString() and SetItemString() allow you to retrieve and set an items name using its command.

HideMenuBar() and ShowMenuBar() let you hide and show the menu bar (what a concept!). Invalidate() marks the menu bar as needing to be redrawn. Validate() sets the menu bar as up to date. RedrawIfNeeded() redraws the menu bar if the invalid flag has been set. Note that RedrawIfNeeded() is called in Sprocket’s main event loop, so there’s no need for you to call it yourself.

This Month’s Resources

Sprocket gets its resources from four different resource files. CreditsBox.rsrc contains the resources used to build the Sprocket about box. StandardMenus.rsrc contains some standard MENU and CMNU resources. If you want to change any of these menus, copy the appropriate resource from StandardMenus.rsrc into SprocketStarter.rsrc and delete the original from StandardMenus.rsrc. Modify the version you copied into SprocketStarter.rsrc.

Sprocket.rsrc contains various resources used by Sprocket and should not be modified. SprocketStarter.rsrc is your resource center. Put all the resources you add to Sprocket there.

You’ll need to modify one resource and add three new ones to SprocketStarter.rsrc. First, open up MBAR 128 and add menu ID 1000 to the list already in place.

If you’re not using Projector (the sourced code control system), you might want to delete the ckid resources you’ll find in each of the resource files. That will get rid of the annoying link error complaining about the multiply-defined resource.

Next, you’ll create your three CMNU resources. The first represents the Options menu we want to add to the end of the menu bar. In general, when you add a new resource to Sprocket, you’ll start numbering your resources from 1000, instead of at 128 the way you normally would. This is just a convention, and might change as Sprocket grows up.

When you create CMNU 1000, be sure to change the resource ID in both places: once in the “Get Info” box and also in the “Edit Menu and MDEF ID” dialog.

Figure 3. CMNU 1000

Enter a command number of 1000 for the item Change My Name, 1001 for Disable Me, 1002 for Enable Previous Item, 1003 for Add Extra Menu, and 1004 for Beeps. Next, disable the item Enable Previous Item. After that, click on the Beeps item, check the has SubMenu checkbox and enter 100 as the submenu ID (Figure 4). Since submenu IDs are limited to a single byte, we won’t be able to give the submenu CMNU resource an ID greater than 1000. So much for sticking to conventions!

Figure 4. The Beeps item, with its submenu ID of 100 entered.

Next, create a new CMNU resource with an ID of 1001 (Once again, be sure to change the ID in both places). The menu will have a title of Extra Menu and a single item, Delete This Menu. Give the item Delete This Menu a command of 1007 (Figure 5).

Figure 5. CMNU 1001

Finally, create a CMNU resource with an ID of 100. Add two items, Beep Once with a command ID of 1005 and Beep Twice with a command ID of 1006 (Figure 6).

Figure 6. CMNU 100.

Save your changes and quit your resource editor.

Modifying the Source Code

Now launch CodeWarrior or Symantec C++ and edit SprocketStarter.h. Start by adding this global reference to the file:

extern Boolean   gItemNameChanged;

gItemNameChanged is a Boolean that indicates whether the item Change My Name has been selected. It tells us whether the item should read Change My Name or Change Me Back Again.

Next, add this enum to the file:

enum
{
 mSubMenu = 100,
 mExtraMenu = 1001,
 
 cChangeName= 1000,
 cDisableMe = 1001,
 cEnablePrevious = 1002,
 cAddExtraMenu   = 1003,
 cBeeps = 1004,
 cBeepOnce= 1005,
 cBeepTwice = 1006,
 cDeleteExtraMenu= 1007
};

The first two constants specify the two CMNU resource IDs. The next 8 specify the menu command IDs. Notice that the menu constants start with a lower case “m” and the commands start with a lower case “c”. Unfortunately, the Apple event registry starts all its class names with a lower-case “c”, so be on the lookout for name collisions.

Next, add these three constants to the file:

const StringPtr kUnchangedName = "\pChange My Name";
const StringPtr kChangedName = "\pChange Me Back Again";
const short kLastMenu = 0;

The first two are just Pascal strings we used for the menu names. We really should have implemented these strings as ‘STR ’ resources to make the code easier to localize. In general, I try never to specify strings in code, but I guess I was just feeling a bit lazy.

The last constant will be used in our call of InsertMenu(), telling InsertMenu() to insert the menu at the end of the menu bar.

Next, edit the file SprocketStarter.cp. Start by adding this global definition at the top of the file:

Boolean gItemNameChanged = false;

Next, add these three lines to the beginning of the routine SetUpApplication():

MenuRef hierMenu;

hierMenu = gMenuBar->GetMenuFromCMNU( mSubMenu );
InsertMenu( hierMenu, -1 ); 

GetMenuFromCMNU() loads CMNU 100, registers all the commands, and returns a MenuHandle to a standard menu based on the CMNU resource. InsertMenu() inserts the resulting menu in the menu bar.

Finally, add the cases to handle our new commands to the switch in HandleMenuCommand() further down in SprocketStarter.cp. Here’s my edited copy of HandleMenuCommand():

HandleMenuCommand
void
HandleMenuCommand(MenuCommandID theCommand)
 {
 MenuRefextraMenu;
 OSErr  err;
 
 switch (theCommand)
 {
 case cAbout:
 AboutBox();
 break;
 
 case cNew:
 CreateNewDocument();
 break;
 
 case cOpen:
 OpenExistingDocument();
 break;
 
 case cPreferences:
 TPreferencesDialogWindow * prefsDialog = 
 new TPreferencesDialogWindow;
 break;
 
#ifqAOCEAware
 case cNewMailableWindow:
 TMailableDocWindow *aWackyThing = new TMailableDocWindow;
 break;
#endif

Here come the new commands. This first one switches the first menu item between Change My Name and Change Me Back Again. Notice that we’re using the global TMenuBar object to change the menus. If Sprocket ever gets modified to use more than one menu bar, we’ll have to modify this code to be sure we use the menu bar that contains the menu we want to work with. Of course, if that happens, you can count on some sample code in this column to show you how to do that.

 case cChangeName:
 if ( gItemNameChanged )
 gMenuBar->SetItemString( cChangeName, kUnchangedName );
 else
 gMenuBar->SetItemString( cChangeName, kChangedName );
 gItemNameChanged = ! gItemNameChanged;
 break;

This command disables Disable Me and enables Enable Previous Item.

 case cDisableMe:
 gMenuBar->EnableCommand( cDisableMe, false );
 gMenuBar->EnableCommand( cEnablePrevious, true );
 break;

This command does just the opposite.

 case cEnablePrevious:
 gMenuBar->EnableCommand( cDisableMe, true );
 gMenuBar->EnableCommand( cEnablePrevious, false );
 break;

This command disables the item that spawned this command in the first place (Add Extra Menu), then builds a new menu from the extra menu CMNU resource. We add the new menu to the end of the menu bar, then call Invalidate() to force the menu bar to get redrawn.

 case cAddExtraMenu:
 gMenuBar->EnableCommand( cAddExtraMenu, false );
 extraMenu = gMenuBar->GetMenuFromCMNU( mExtraMenu );
 InsertMenu( extraMenu, kLastMenu );

 gMenuBar->Invalidate();
 break;

This command deletes the extra menu we added with the previous command. First, we reenable the Add Extra Menu item. Notice that we didn’t have to retrieve the menu that this item belongs to. All we needed was the command. This definitely makes life a lot simpler.

 case cDeleteExtraMenu:
 gMenuBar->EnableCommand( cAddExtraMenu, true );

Since the TMenuBar class doesn’t support a DeleteCMNU() method, we’ll have to deregister the command by hand. A DeleteCMNU() method would step through all the items in the specified menu, calling UnregisterCommand() for each item. It would then delete the menu for us. Since our extra menu only contains a single item, it’s no big deal to do this by hand. Once we are done, we’ll force a menu bar redraw. Look for a DeleteCMNU() method in a future version of Sprocket.

 err = gMenuBar->UnregisterCommand( cDeleteExtraMenu );
 DeleteMenu( mExtraMenu );
 
 gMenuBar->Invalidate();
 break;

This next command corresponds to the parent menu of our hierarchical submenu. Normally, this command will never get called because the menu manager won’t detect a selection of the parent item of a submenu (Figure 7). There are times when this is useful, however. For example, imagine if you built a menu of applications, where each application item had a submenu listing some frequently used documents that can be opened by that application (NowMenus does this). If you select a document from a submenu, its parent application gets launched and opens the selected document. If you release the mouse with the application selected (without selecting a document from the submenu), you might want to launch the application without specifying a document.

The point here is this: Specify commands for all your menu items, even the hierarchical parent menus. A future version of Sprocket might include a workaround to execute commands associated with these currently orphaned items.

 case cBeeps:
 break;

Figure 7. A menu item with its submenu showing.
If the mouse button was released at this point, the Beeps item
(rather than either of the submenu items) would be selected.

These next two items are horribly technical. They beep either once or twice, depending on the item selected.

 case cBeepOnce:
 SysBeep( 20 );
 break;
 case cBeepTwice:
 SysBeep( 20 ); SysBeep( 20 );
 break;
 default:
 break;
 }
 }

Till Next Month

There are still some concepts that we need to get into regarding Sprocket and menus. For example, when do you make the decision about which menu items you will enable and disable to ensure that things are set up correctly before a user makes a selection from a menu. How should the frontmost window affect which menu items are enabled or disabled? We’ll explore these important issues in next month’s column. See you then!

 

Community Search:
MacTech Search:

Software Updates via MacUpdate

FileZilla 3.23.0.2 - Fast and reliable F...
FileZilla (ported from Windows) is a fast and reliable FTP client and server with lots of useful features and an intuitive interface. Version 3.23.0.2: Bug Fixes and Minor Changes Speed up icon... Read more
PDFpen 8.3 - $74.95
PDFpen allows users to easily edit PDF's. Add text, images and signatures. Fill out PDF forms. Merge or split PDF documents. Reorder and delete pages. Even correct text and edit graphics! Features... Read more
TunnelBear 3.0.8 - Subscription-based pr...
TunnelBear is a subscription-based virtual private network (VPN) service and companion app, enabling you to browse the internet privately and securely. Features Browse privately - Secure your data... Read more
Safari Technology Preview 10.1 - The new...
Safari Technology Preview contains the most recent additions and improvements to WebKit and the latest advances in Safari web technologies. And once installed, you will receive notifications of... Read more
Ableton Live 9.7.1 - Record music using...
Ableton Live lets you create and record music on your Mac. Use digital instruments, pre-recorded sounds, and sampled loops to arrange, produce, and perform your music like never before. Ableton Live... Read more
BetterTouchTool 1.963 - Customize Multi-...
BetterTouchTool adds many new, fully customizable gestures to the Magic Mouse, Multi-Touch MacBook trackpad, and Magic Trackpad. These gestures are customizable: Magic Mouse: Pinch in / out (zoom... Read more
NeoFinder 7.0 - Catalog your external me...
NeoFinder (formerly CDFinder) rapidly organizes your data, either on external or internal disks, or any other volumes. It catalogs all your data, so you stay in control of your data archive or disk... Read more
Coda 2.6 - One-window Web development su...
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
PDFpenPro 8.3 - $124.95
PDFpenPro allows users to edit PDF's easily. Add text, images and signatures. Fill out PDF forms. Merge or split PDF documents. Reorder and delete pages. Create fillable forms and tables of content... Read more
File Juicer 4.51 - Extract images, video...
File Juicer is a drag-and-drop can opener and data archaeologist. Its specialty is to find and extract images, video, audio, or text from files which are hard to open in other ways. In computer... Read more

Latest Forum Discussions

See All

Niantic teases new Pokémon announcement...
After rumors started swirling yesterday, it turns out there is an official Pokémon GO update on its way. We’ll find out what’s in store for us and our growing Pokémon collections tomorrow during the Starbucks event, but Niantic will be revealing... | Read more »
3 reasons why Nicki Minaj: The Empire is...
Nicki Minaj is as business-savvy as she is musically talented and she’s proved that by launching her own game. Designed by Glu, purveyors of other fine celebrity games like cult favorite Kim Kardashian: Hollywood, Nicki Minaj: The Empire launched... | Read more »
Clash of Clans is getting its own animat...
Riding on its unending wave of fame and success, Clash of Clans is getting an animated web series based on its Clash-A-Rama animated shorts.As opposed to the current shorts' 60 second run time, the new and improved Clash-A-Rama will be comprised of... | Read more »
Leaks hint at Pokémon GO and Starbucks C...
Leaked images from a hub for Starbucks employees suggests that a big Pokémon GO event with the coffee giant could begin this very week. The images appeared on Reddit and hint at some exciting new things to come for Niantic's smash hit game. | Read more »
Silent Depth Submarine Simulation (Game...
Silent Depth Submarine Simulation 1.0 Device: iOS Universal Category: Games Price: $7.99, Version: 1.0 (iTunes) Description: | Read more »
Enneas Saga lets you lead your own demon...
Defend the land of Enneas Continent from the forces of evil in the new fantasy MMORPG from Lyto Mobi: Enneas Saga. Can’t wait? No problem. It’s available to download now on Android devices. | Read more »
Great zombie games in the spirit of Dead...
Dead Rising 4 arrives tomorrow, giving enthusiasts a fresh chance to take selfies with zombies and get up to other ridiculous end-of-the-world shenanigans. To really get into the spirit of things, we've gone and gathered the best zombie games that... | Read more »
Amateur Surgeon 4 Guide: Advanced tips a...
Amateur Surgeon 4 is still tackling the competition at the top of the App Store charts, so if you haven't tried it out yet, you should probably do that right away. If you've been at it for a while, though, perhaps you're ready to start expanding... | Read more »
Amateur Surgeon 4 Guide: Become the worl...
It's time to wield your trusty pizza cutter again, as Amateur Surgeon has returned with a whole fresh set of challenges (and some old, familiar ones, too). Starting anew isn't easy, especially when all you have at your disposal is a lighter, the... | Read more »
Le Parker: Sous Chef Extraordinaire (Ga...
Le Parker: Sous Chef Extraordinaire 1.0 Device: iOS Universal Category: Games Price: $2.99, Version: 1.0 (iTunes) Description: | Read more »

Price Scanner via MacPrices.net

Back in stock: Apple refurbished Mac minis fr...
Apple has Certified Refurbished Mac minis available starting at $419. Apple’s one-year warranty is included with each mini, and shipping is free: - 1.4GHz Mac mini: $419 $80 off MSRP - 2.6GHz Mac... Read more
Twenty-Five Years Of Apple Laptops – A person...
Among many other things, the often tumultuous 16th year of the new century marked the 25th anniversary of Apple laptop computers, not counting the optimistically named 16-pound Mac Portable of 1989.... Read more
Landlordy iOS App Adds Support For Appliances...
Riga, Latvia based E-protect SIA is releasing major update (version 1.8) to its Landlordy app for managing rental business financials on the go. Landlordy is iPhone and iPad app designed for self-... Read more
Holiday sale, Apple iMacs for up to $200 off...
B&H Photo has 21″ and 27″ Apple iMacs on sale for up to $200 off MSRP including free shipping plus NY sales tax only: - 27″ 3.3GHz iMac 5K: $2099 $200 off MSRP - 27″ 3.2GHz/1TB Fusion iMac 5K: $... Read more
Holiday sale: Mac minis for $50 to $100 off M...
B&H Photo has Mac minis on sale for up to $100 off MSRP free shipping plus NY sales tax only: - 1.4GHz Mac mini: $449 $50 off MSRP - 2.6GHz Mac mini: $629 $70 off MSRP - 2.8GHz Mac mini: $899 $... Read more
Mac Pros on sale for up to $300 off MSRP, ref...
B&H Photo has Mac Pros on sale for up to $300 off MSRP. Shipping is free, and B&H charges sales tax in NY only: - 3.7GHz 4-core Mac Pro: $2799, $200 off MSRP - 3.5GHz 6-core Mac Pro: $3699, $... Read more
12-inch WiFi Apple iPad Pros on sale for up t...
B&H Photo has 12″ WiFi Apple iPad Pros on sale for up to $50 off MSRP, each including free shipping. B&H charges sales tax in NY only: - 12″ Space Gray 32GB WiFi iPad Pro: $749 $50 off MSRP... Read more
9-inch Apple WiFi iPad Pros on sale for $20-$...
B&H Photo has 9.7″ Apple WiFi iPad Pros on sale for $20-$50 off MSRP, each including free shipping. B&H charges sales tax in NY only: - 9″ Space Gray 256GB WiFi iPad Pro: $779.95 $20 off MSRP... Read more
Apple refurbished 2015 15-inch MacBook Pros a...
Apple has Certified Refurbished 2015 15″ Retina MacBook Pros available starting at $1699. An Apple one-year warranty is included with each model, and shipping is free: - 15″ 2.2GHz Retina MacBook Pro... Read more
Back in stock! 13-inch 2.7GHz Retina MacBook...
Apple has Apple Certified Refurbished 2015 13″ 2.7GHz/128GB Retina MacBook Pros (MF839LL/A) available again for $1099 including free shipping. That’s $200 off MSRP, and it’s the lowest price... Read more

Jobs Board

*Apple* Retail - Multiple Positions- Philade...
Job Description: Sales Specialist - Retail Customer Service and Sales Transform Apple Store visitors into loyal Apple customers. When customers enter the store, Read more
*Apple* Retail - Multiple Positions- San Ant...
Job Description:SalesSpecialist - Retail Customer Service and SalesTransform Apple Store visitors into loyal Apple customers. When customers enter the store, Read more
*Apple* Products Tester Needed - Apple (Unit...
…we therefore look forward to put out products to quality test for durability. Apple leads the digital music revolution with its iPods and iTunes online store, Read more
SW Engineer *Apple* TV Frameworks - Apple I...
The Apple TV team is looking for a software...create features that reflect the look and feel of Apple TV. Description: Were looking for someone who is Read more
Hardware Design Validation Engineer - *Apple...
The Apple Watch team is looking for a Hardware Design Validation Engineer. This person will be part of the Apple Watch hardware team with responsibilities for Read more
All contents are Copyright 1984-2011 by Xplain Corporation. All rights reserved. Theme designed by Icreon.