TweetFollow Us on Twitter

Extend Modal Dialog
Volume Number:6
Issue Number:4
Column Tag:C Workshop

Related Info: Dialog Manager Event Manager

Extending Modal Dialogs

By Paul Potts, Wooster, OH

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

The Gauntlet is Thrown Down

A few months ago I was developing a program that used a modal dialog. I showed the results to my roommate. He ran the program and brought up the dialog, but was then interrupted. When he got back, the screen saver had come on. He moved the mouse to refresh the screen, and then scrutinized my display.

“Which button is default?” he asked .

“That one. The program drew the outline, but it got erased.”

“It should be there.”

“Oh, come on! The Finder doesn’t even keep the outline around its default buttons when you use a screen saver!”

“If it’s in Inside Macintosh, it should be there. Go fix it. Here, make me a copy first.” He popped in a blank disk. Nothing happened.

“How come it didn’t ask if I want to initialize the disk?”

“That doesn’t work when a modal dialog is active,” I explained lamely.

He then pressed the “Q” key. Nothing happened. “There’s a dialog button called Quit -- why didn’t it quit?” he asked. “What kind of user-friendly program is this, anyway?”

The Gauntlet is Picked Up

My roommate was right. We are used to putting up with the limitations of modal dialogs, but ModalDialog can be extended using filter procedures. The parameters of ModalDialog can be declared in C as follows:

/* 1 */

void ModalDialog (filterProc, itemHit)
 ProcPtr  filterProc;
 int  *itemHit;

The address of your filter procedure gets passed by casting it to a procPtr. If you don’t want to use a filterProc, you pass NIL as your first parameter. If you do use a filterProc, Modal dialog then gets each event for you, using a mask which excludes disk events, and sends them to your filterProc. It is then up to your filterProc to decide what to do with them. ModalDialog expects your filterProc to be declared as follows:

/* 2 */

pascal Boolean FProc(Dialog, Event, itemHit)
 DialogPtrtheDialog;
 EventRecord*theEvent;
 int  *itemHit;

A dialog filterProc is actually a function that returns a Boolean value. This value should be TRUE if you want ModalDialog to exit, and FALSE otherwise. ModalDialog passes your filterProc a pointer to the current event. You then can do what you want with it. If your function returns FALSE, ModalDialog will handle the event after you. Thus, you can get a crack at each event even before ModalDialog does. Your filterProc can then do any of the following:

1) Handle the event, then send it to ModalDialog

2) Handle the event and tell ModalDialog to exit

3) Change the event and return to ModalDialog

4) Read the event queue itself and act on it.

My filterProc uses the first technique to handle updateEvents before ModalDialog gets them. This allows me to redraw the default box around a button and then tell ModalDialog to do its own updating. I use the second technique to exit ModalDialog if my filterProc receives a key event that it understands. I use the third technique to handle the <return> key by changing the event from a keypress to a click in the default button. You don’t actually have to do this for <enter> and <return> keypresses, since ModalDialog handles key events, but I wanted to illustrate the technique of altering an event.

Scott Knaster, in Macintosh Programming Secrets, suggests that command-key equivalents be provided for buttons in modal dialog boxes whenever possible. Command-key equivalents aren’t always necessary, however. Some popular applications, such as Microsoft Word, provide single-key equivalents for button clicks, if no editable text fields are present in the dialog, such as in the Save Changes Before Closing dialog. I suggest that commnad-key equivalents be used in modal dialogs with editable text fields, and single-key equivalents (or both single-key and command-key equivalents) be used when no editable text fields are present. My code illustrates both techniques, but your application should use only one. Even though my modal dialog has an editable text field, I allows the user to press a 1, 2, or 3 to choose one of my three buttons. The user can also use <command>-F, S, and T to choose the first, second, and third dialog items, respectively. NOTE: You must make certain that the editText item(s) in the DITL resource that you use are set to Disabled, or else my technique will not work.

When drawing your dialogs with ResEdit, keep in mind that the command-key character is ASCII 17 in the Chicago font, but cannot be typed directly from the keyboard. Microsoft Word will generate this character if you use the Change function to replace a specified character with a ^17. You can then copy this character and paste it into your ResEdit fields. RMaker can generate this character by using \11.

Since ModalDialog does not pass disk events to the filterProc, I use the fourth technique to allow the user to insert a blank disk while the ModalDialog is active. It gets initialized by DIBadMount, which then mounts it (or ejects it, if initialization failed). This might be useful in a modal dialog which displays on-line volumes. SFPutFile uses this technique to allow disks to be mounted while its dialog is active.

There are other ways you can handle events in a modal dialog using filterProcs. You can do anything you want during null events. You can plot a series of icons in your dialog to give the appearance of animation, or draw the current time using DrawString. After all, you have the current grafPort, and can draw directly with QuickDraw. Remember, though, that your drawing operations should be as fast as possible, and must take much less than a single tick to work effectively. Also, remember your poor end-user: don’t make your dialogs overly confusing or complex. We want to extend the Macintosh interface, not bury it.

If you like my code, you are welcome to use it “as-is,” modify it, or completely rewrite it. I’d appreciate it if you’d put my name somewhere in your application, but you can drop me a postcard instead.

Listing:  driveDialog.c

/* This Think C 3.02/4.0 code should be compiled with MacHeaders turned 
on.  Create a new project called FilterProc.Π, and add this file, driveDialog.c, 
along with HandleDialog.c and MacTraps.  Prototypes are provided. Compile 
the resource file, FilterProc.r, with RMaker and put it in the folder 
with the FilterProc project.*/

/*********************************************/
/* File: driveDialog.c */
/* A simple driver application for the dialog
handler.  Your own application would call
it instead. */
 
void main (void);
int HandleDialog(short ID);

void main()
{
 int  itemHit,
/* Dialog item hit, returned by HandleDialog */
 counter;
 
 InitGraf(&thePort);
 InitFonts();
 InitWindows();
 InitMenus();
 TEInit();
 InitDialogs(0L);
 FlushEvents(everyEvent, 0);
 InitCursor();
 
/* Now send the HandleDialog function the resource ID of the DLOG to 
use */
 itemHit = HandleDialog ((short)1);
 
/* Now beep to tell us what item number the HandleDialog call sent back 
(i.e., what item was hit to exit ModalDialog */
 for (counter = 1; counter<= itemHit;  counter++)
 SysBeep(40);
}
Listing:  HandleDialog.c

/*****************************************/
/* File HandleDialog.c */
/*****************************************/
/* Prototypes */
 
int HandleDialog(short Dialog_ID);
/* Called whenever you want to put up a modal dialog */

pascal Boolean FProc(DialogPtr theDialog, 
 EventRecord *theEvent, 
 int *itemHit);
/* Filter procedure called by ModalDialog to screen dialog events */
 
Point CenterDialogItem(DialogPtr theDialog,              int   item_number);
/* Returns the center of the rect of a dialog item, to be called from 
a filter proc while a modal dialog is active. */

void FlashDialogItem(DialogPtr theDialog, int                  item_number);
/* Flashes an item of a dialog, to be called from a filter proc while 
a modal dialog is active. */

void HandleUpdate(DialogPtr theDialog);
/* Used to handle update events while a modal 
dialog is active, called from a filter proc.*/

/********************************************/
/* This procedure returns the center of a dialog item as a point.  It 
is designed to be used with buttons, but will work with any type of dialog 
item.  */
/* Input: theDialog (a DialogPtr), item number (an integer) */
/* Output: a point */

Point CenterDialogItem(theDialog, item_number)
 DialogPtr  theDialog;
 int    item_number;

{
 Point  the_center; /* Center of item, to return */
 int    itemType; /* Returned by GetDItem but not used */
 Handle theItem; /* Returned by GetDItem but not used */
 Rect the_Rect; /* Returned by GetDItem */

GetDItem(theDialog, item_number, &itemType,        &theItem, &the_Rect);
the_center.h = the_Rect.left +   ((the_Rect.right -      the_Rect.left)/ 
2);
the_center.v = the_Rect.top + ((the_Rect.bottom - the_Rect.top)/ 2);
return the_center;
}

/********************************************/
/* This procedure returns the center of the rectangle of the default 
dialog item as a single point.  */
/* Input: theDialog (a DialogPtr), item_number (an integer) */
/* Output:  none */

void FlashDialogItem(theDialog, item_number)
 DialogPtr  theDialog;
 int    item_number;
{
 long tickScratch; /* returned by Delay, unused */
 int  itemType; /* Returned by GetDItem but not used */
 Handle the_item; /* Handle to default button */
 Rect controlRect; /* Returned by GetDItem but not used */

GetDItem(theDialog, item_number, &itemType,  &the_item, &controlRect);
HiliteControl(the_item, 1); 
Delay(6, &tickScratch);
}

/********************************************/
/* The following function handles update events for your dialog box. 
 It is called whenever the filterProc receives an update event.  You 
can put whatever you want to be drawn in the dialog in it.  Right now 
it redraws the rounded rect around the default button (number one). */

/* Constants for drawing roundRect */
#define edgeCurve16
#define gapBetween -4
#define lineSize 3

void HandleUpdate(theDialog)
 DialogPtr theDialog;
{
 Rect controlRect; /* rectangle of the default control.  We
need this to draw the round rect. */
 int    itemType; /* Returned by GetDItem but not used */
 Handle the_item; /* ditto */
 Rect border; /* Rect of the thick border */
 
PenSize(lineSize, lineSize);
/* Get control #1’s rectangle and grow it a little bit, then call FrameRoundRect 
to draw it. */
GetDItem(theDialog, (int)1, &itemType, &the_item, &controlRect);
border = controlRect;
InsetRect(&border, gapBetween, gapBetween);  
FrameRoundRect(&border, edgeCurve, edgeCurve);     
}

/********************************************/
/* The following filterProc is used to handle extra events during the 
modal dialog loop, such as update events.  Its original purpose was to 
keep the default round-rect drawn around the control even after a screen 
saver has redrawn the screen. It also handles insertion of blank disks 
and button-keyboard equivalents. 
NOTE:  If you have one or more editable text fields within your modal 
dialog, your key equivalents should use the command key.  If you don’t 
have any editable text fields, your key equivalents can be handled straight. 
 The names of your buttons should all begin with unique first letters, 
and if you use command-key entry you should provide a legend of command-key 
equivalents next to the buttons. */
 
pascal Boolean FProc(theDialog,  theEvent, itemHit)
 DialogPtrtheDialog;
 EventRecord   *theEvent;
 int    *itemHit;
{
 long key; /* Holds the key code */
 Point  ctr; /* where the mouse click will go */
 int  DIResult; /* Returned by DIBadMount */
 EventRecorddiskEvent; /* Returned by EventAvail, below */

/* Since ModalDialog doesn’t handle bad disk mounts, we have a handler 
that will allow the user to format new disks with a modal dialog still 
on the screen. */

if (GetNextEvent(diskMask, &diskEvent) ==    TRUE)
{
 if (HiWord(diskEvent.message) != 0) 
 {
 diskEvent.where.h = ((screenBits.bounds.right
 - screenBits.bounds.left)/ 2) - (304 / 2);
 diskEvent.where.v = ((screenBits.bounds.bottom
 - screenBits.bounds.top)/  3) - (104 / 2);
 InitCursor();
 DIResult = DIBadMount(diskEvent.where, 
 diskEvent.message);
 }

} /* end of if GetNextEvent test for disk events */

switch (theEvent->what)
{
 case (updateEvt):
 HandleUpdate(theDialog);
 return FALSE;
/*Do not return an item hit value.  Returning
FALSE tells modalDialog to handle the update
event.  We have just added our own handling of the event. */
 break; 
 
 case (keyDown):
 key = theEvent->message & charCodeMask;
 switch(key)
 { 
/* key has been pressed, we want to interpret it properly. */
 case 13: /* Return key */
 { 
 FlashDialogItem(theDialog, 1);
 ctr = CenterDialogItem(theDialog, 1);

/* We can do it this way: change the event record to fool ModalDialog. 
 ModalDialog doesn’t flash the button long enough for my taste, so I 
flash it some more myself. */
 theEvent->what = mouseDown;
 LocalToGlobal (&ctr);
 theEvent->where = ctr;
 theEvent->message = 0;

/* Now we tell ModalDialog to handle the
event.  It doesn’t suspect a thing! */
 *itemHit = 1;
 return FALSE;
 break;
 }

 case 3: /* the Enter key */
 { 

/* Or we can do it our own way by flashing the button and telling ModalDialog 
to exit. */
 FlashDialogItem(theDialog, 1);
 *itemHit = 1;
 return TRUE;
 break;
 }
 
/* These key equivalents do not rely on the use of the command key.  
You would use these in your modal dialog only if you had no editable 
text fields. */
 case 49: /* 1 key */
 { 
 FlashDialogItem(theDialog, 1);
 *itemHit = 1;
 return TRUE;
 break;
 }

 case 50: /* 2 key */
 { 
 FlashDialogItem(theDialog, 2);
 *itemHit = 2;
 return TRUE;
 break;
 }

 case 51: /* 3 key */
 { 
 FlashDialogItem(theDialog, 3);
 *itemHit = 3;
 return TRUE;
 break;
 }
 
/* These key equivalents for buttons use
the command key, and are appropriate even 
for use in a modal dialog with editable text. */
 
 case 102:/* ASCII F */
 { 
 if (theEvent->modifiers & cmdKey)
 {
 FlashDialogItem(theDialog, 1);
 *itemHit = 1;
 return TRUE;
 break;
 }
 }
 
 case 115:/* ASCII S */
 { 
 if (theEvent->modifiers & cmdKey)
 {
 FlashDialogItem(theDialog, 2);
 *itemHit = 2;
 return TRUE;
 break;
 }
 }

 case 116:/* ASCII T */
 { 
 if (theEvent->modifiers & cmdKey)
 {
 FlashDialogItem(theDialog, 3);
 *itemHit = 3;
 return TRUE;
 break;
 }
 }
 
 default:
 {
 return FALSE; /* Do nothing if another key is chosen. */
 break;
 }
 } /* end of key code switch */
 
case (mouseDown):
 
/* You can insert your own mouse click
handlers here.  ModalDialog takes care of mouse events, but you can do 
do special processing.  For example, ModalDialog does nothing if a mouse 
click
occurs inside a modal dialog but outside of a control, but you could 
do something.  */
 
 return FALSE;
 break;
case (mouseUp):
 return FALSE;
 break;
default:
 return FALSE;
 break; 
/* We don’t handle any other types of events, so these get sent to ModalDialog 
unchanged. */    
 } /* end of switch */
} /* end of filterproc function */

/********************************************/
/* HandleDialog is a function to draw and dispose of a simple modal dialog. 
 It tells ModalDialog to use a filter proc to screen events for the dialog. 


Input: short Dialog_ID - the resource ID
Output: item number of the control that was hit to exit ModalDialog. 
*/

int HandleDialog(dialog_ID)
 short  dialog_ID; 
{
 int  itemHit; /* returned by ModalDialog */
 DialogPtrtheDialog; /* The dialog we will work with */
 GrafPtroldWindow; /* Saves the previous window */       

GetPort(&oldWindow);  /* save current grafPort */
SetDAFont(0);    /* use system font */
/* First, get the dialog from the resource and prepare to execute ModalDialog 
*/
theDialog = GetNewDialog(dialog_ID, (Ptr)0, (WindowPtr)-1);
if (ResError() != noErr)
/* Called if GetNewDialog returns an error.  You can put your own error 
handler here.  In this example we are assuming that the dialog is available 
in the application resource fork */
 {
 SysBeep(40);
 ExitToShell();
 }
 
ShowWindow(theDialog);
SetPort(theDialog);
FlushEvents(everyEvent, 0);
 
/*Modaldialog call should wait until event in active control*/
ModalDialog((ProcPtr)FProc, &itemHit);
DisposDialog(theDialog);
FlushEvents (everyEvent, 0);
SetPort(oldWindow);
/* restore drawing state */
return itemHit;
}
Listing:  Filterproc.r

**************************
* RMaker source file for
* filterProc application
**************************

Filterproc.Π.RSRC
RSRCRSED

Type DITL ;;Text for dialog
,1 ;;Resource number
5;;Number of items

Button Enabled
17 27 37 95
First \11F;;The non-printable character is * ASCII 17 (see text)

Button Enabled
17 108 37 182
Second \11S

Button Enabled
17 200 37 262
Third \11T

editText Disabled
137 86 157 222
This is editable text

staticText Enabled
61 43 115 246
This is an example of a modal dialog with command-key equivalents for 
buttons.
* The above text should be all on one line

Type DLOG ;;modal dialog
,1

90 120 266 420
Visible NoGoAway
5;;proc id
0;;refCon
1;;DITL

 
AAPL
$98.15
Apple Inc.
-0.23
MSFT
$43.58
Microsoft Corpora
-0.31
GOOG
$587.42
Google Inc.
+1.81

MacTech Search:
Community Search:

Software Updates via MacUpdate

Knock 1.1.7 - Unlock your Mac by knockin...
Knock is a faster, safer way to sign in. You keep your iPhone with you all the time. Now you can use it as a password. You never have to open the app -- just knock on your phone twice, even when it's... Read more
Mellel 3.3.6 - Powerful word processor w...
Mellel is the leading word processor for OS X and has been widely considered the industry standard since its inception. Mellel focuses on writers and scholars for technical writing and multilingual... Read more
LibreOffice 4.3.0.4 - Free Open Source o...
LibreOffice is an office suite (word processor, spreadsheet, presentations, drawing tool) compatible with other major office suites. The Document Foundation is coordinating development and... Read more
Freeway Pro 7.0 - Drag-and-drop Web desi...
Freeway Pro lets you build websites with speed and precision... without writing a line of code! With it's user-oriented drag-and-drop interface, Freeway Pro helps you piece together the website of... Read more
Drive Genius 3.2.4 - Powerful system uti...
Drive Genius is an OS X utility designed to provide unsurpassed storage management. Featuring an easy-to-use interface, Drive Genius is packed with powerful tools such as a drive optimizer, a... Read more
Vitamin-R 2.15 - Personal productivity t...
Vitamin-R creates the optimal conditions for your brain to work at its best by structuring your work into short bursts of distraction-free, highly focused activity alternating with opportunities for... Read more
Toast Titanium 12.0 - The ultimate media...
Toast Titanium goes way beyond the very basic burning in the Mac OS and iLife software, and sets the standard for burning CDs, DVDs, and now Blu-ray discs on the Mac. Create superior sounding audio... Read more
OS X Yosemite Wallpaper 1.0 - Desktop im...
OS X Yosemite Wallpaper is the gorgeous new background image for Apple's upcoming OS X 10.10 Yosemite. This wallpaper is available for all screen resolutions with a source file that measures 5,418... Read more
Acorn 4.4 - Bitmap image editor. (Demo)
Acorn is a new image editor built with one goal in mind - simplicity. Fast, easy, and fluid, Acorn provides the options you'll need without any overhead. Acorn feels right, and won't drain your bank... Read more
Bartender 1.2.20 - Organize your menu ba...
Bartender lets you organize your menu bar apps. Features: Lets you tidy your menu bar apps how you want. See your menu bar apps when you want. Hide the apps you need to run, but do not need to... Read more

Latest Forum Discussions

See All

Ice Wings Plus (Games)
Ice Wings Plus 1.0 Device: iOS Universal Category: Games Price: $1.99, Version: 1.0 (iTunes) Description: THE GREAT ENDLESS RUNNER OF COMBAT JETS IS BACK !! With more than 680.000 downloads in the App Store, Ice Wings: Skies of Steel... | Read more »
Murl the Squirrel (Games)
Murl the Squirrel 1.0 Device: iOS Universal Category: Games Price: $.99, Version: 1.0 (iTunes) Description: Meet Murl. He is teased by a group of flying squirrels because he can't fly. Determined to show them he's can fly, he meets... | Read more »
Celleste (Games)
Celleste 0.1 Device: iOS Universal Category: Games Price: $2.99, Version: 0.1 (iTunes) Description: Lots of cute action with amazing 3D graphics and a new type of gameplay! Take control over the forces of the universe to help a group... | Read more »
Super Heavy Sword (Games)
Super Heavy Sword 0.0.1 Device: iOS Universal Category: Games Price: $.99, Version: 0.0.1 (iTunes) Description: Get Ready to Get HEAVY! Monster Robot Studios presents SUPER Heavy Sword! The sequel to the smash hit HEAVY sword which... | Read more »
Angels In The Sky (Games)
Angels In The Sky 1.00 Device: iOS Universal Category: Games Price: $6.99, Version: 1.00 (iTunes) Description: - A.I.S will only run smoothly on iPhone 5s. It's NOT compatible with iPad, iPhone 5 or earlier devices.- In order to... | Read more »
80 Days (Games)
80 Days 1.0.2 Device: iOS Universal Category: Games Price: $4.99, Version: 1.0.2 (iTunes) Description: 1872, with a steampunk twist. Phileas Fogg has wagered he can circumnavigate the world in just eighty days. Choose your own route... | Read more »
Micromon (Games)
Micromon 1.0 Device: iOS Universal Category: Games Price: $.99, Version: 1.0 (iTunes) Description: 130+ Animated Monsters to Catch & Battle! No waiting, play at your own pace! Embark on an epic monster capture RPG like none... | Read more »
Empire Manager (Games)
Empire Manager 1.0 Device: iOS iPhone Category: Games Price: $3.99, Version: 1.0 (iTunes) Description: Become ruler of an empire. Manage your economy, develop technology, hire an army and conquer the world in this addictive turn-... | Read more »
Empire Manager HD (Games)
Empire Manager HD 1.0 Device: iOS Universal Category: Games Price: $7.99, Version: 1.0 (iTunes) Description: Become ruler of an empire. Manage your economy, develop technology, hire an army and conquer the world in this addictive... | Read more »
Star Admiral Review
Star Admiral Review By Rob Thomas on July 30th, 2014 Our Rating: :: ADMIRABLE ADMIRALSUniversal App - Designed for iPhone and iPad While this new digital CCG may feel a bit familiar, Star Admiral offers a sci-fi twist and galaxy’s... | Read more »

Price Scanner via MacPrices.net

iPad Cannibalization Threat “Overblown”
Seeking Alpha’s Kevin Greenhalgh observes that while many commentators think Apple’s forthcoming 5.5-inch panel iPhone 6 will cannibalize iPad sales, in his estimation, these concerns are being... Read more
Primate Labs Releases July 2014 MacBook Pro P...
Primate Labs’ John Poole has posted Geekbench 3 results for most of the new MacBook Pro models that Apple released on Tuesday. Poole observes that overall performance improvements for the new MacBook... Read more
Apple Re-Releases Bugfixed MacBook Air EFI Fi...
Apple has posted a bugfixed version EFI Firmware Update 2.9 a for MacBook Air (Mid 2011) models. The update addresses an issue where systems may take longer to wake from sleep than expected, and... Read more
Save $50 on the 2.5GHz Mac mini, plus free sh...
B&H Photo has the 2.5GHz Mac mini on sale for $549.99 including free shipping. That’s $50 off MSRP, and B&H will also include a free copy of Parallels Desktop software. NY sales tax only. Read more
Save up to $140 on an iPad Air with Apple ref...
Apple is offering Certified Refurbished iPad Airs for up to $140 off MSRP. Apple’s one-year warranty is included with each model, and shipping is free. Stock tends to come and go with some of these... Read more
$250 price drop on leftover 15-inch Retina Ma...
B&H Photo has dropped prices on 2013 15″ Retina MacBook Pros by $250 off original MSRP. Shipping is free, and B&H charges NY sales tax only: - 15″ 2.3GHz Retina MacBook Pro: $2249, $250 off... Read more
More iPad Upgrade Musings – The ‘Book Mystiqu...
Much discussed recently, what with Apple reporting iPad sales shrinkage over two consecutive quarters, is that it had apparently been widely assumed that tablet users would follow a two-year hardware... Read more
13-inch 2.5GHz MacBook Pro on sale for $999,...
Best Buy has the 13″ 2.5GHz MacBook Pro available for $999.99 on their online store. Choose free shipping or free instant local store pickup (if available). Their price is $100 off MSRP. Price is... Read more
Save up to $300 on an iMac with Apple refurbi...
The Apple Store has Apple Certified Refurbished iMacs available for up to $300 off the cost of new models. Apple’s one-year warranty is standard, and shipping is free. These are the best prices on... Read more
WaterField Unveils 15″ Outback Solo & 13″...
Hard on the heels of Apple’s refreshed MacBook Pro Retina laptops announcement, WaterField Designs has unveiled a 15-inch version of the waxed-canvas and leather Outback Solo and a 13-inch version of... Read more

Jobs Board

Sr. Product Leader, *Apple* Store Apps - Ap...
**Job Summary** Imagine what you could do here. At Apple , great ideas have a way of becoming great products, services, and customer experiences very quickly. Bring Read more
Sr Software Lead Engineer, *Apple* Online S...
Sr Software Lead Engineer, Apple Online Store Publishing Systems Keywords: Company: Apple Job Code: E3PCAK8MgYYkw Location (City or ZIP): Santa Clara Status: Full Read more
Sr Software Lead Engineer, *Apple* Online S...
Sr Software Lead Engineer, Apple Online Store Publishing Systems Keywords: Company: Apple Job Code: E3PCAK8MgYYkw Location (City or ZIP): Santa Clara Status: Full Read more
*Apple* Solutions Consultant (ASC) - Apple (...
**Job Summary** The ASC is an Apple employee who serves as an Apple brand ambassador and influencer in a Reseller's store. The ASC's role is to grow Apple Read more
Sr. Product Leader, *Apple* Store Apps - Ap...
**Job Summary** Imagine what you could do here. At Apple , great ideas have a way of becoming great products, services, and customer experiences very quickly. Bring Read more
All contents are Copyright 1984-2011 by Xplain Corporation. All rights reserved. Theme designed by Icreon.