TweetFollow Us on Twitter

Popup Menu BASIC
Volume Number:5
Issue Number:4
Column Tag:BASIC School

Related Info: Menu Manager

Pop Up Menus in True BASIC

By Dave Kelly, MacTutor Editorial Board

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

New users of the Macintosh are sometimes fascinated with the use of pull-down menus. Even experienced users head straight for the menu bar when they get a new program to see just what it does. Usually the menus get checked out before the manual does. Hierarchical menus added additional capabilities for the user. A little over a year ago, we began seeing the appearance of pop-up menus as another interface for the user. The various versions of BASIC cover the regular pull-down menus pretty easily. Hierarchical menus and pop-up menus are now being used in most of the software which is now being released. This column will attempt to explain how to set up pop-up menus using True Basic. Similar methods for implementing may be used in ZBasic or any other language for that matter. The method is about the same in whatever language is used and since many of the commands come from the Macintosh Toolbox. It should be pointed out that you must be using System 4.1 or greater for Hierarchical or pop-up menus to work. You should upgrade to the latest version (6.02).

It is advisable to know when it is appropriate to use pop-up menus. Apple recommends that pop-up menus be used for setting values or choosing from lists of related items. A pop-up menu could “pop up” anywhere (its location is global), but usually used in a dialog. The indication that there is a pop-up menu is that there is a box with a one-pixel thick drop shadow which is drawn around the current value. When the user presses the mouse button which pointing within the box, the pop-up menu appears with the current value checked and highlighted. Other than that, the pop-up menu acts just like any other menu.

The Macintosh ROM takes care of the pop-up menu action, but you must take care of drawing the shadowed box which indicates that there is a pop-up menu. You must take care of inverting the title when the menu is showing. The current value should appear in the pop-up menu when it is selected. Apple also recommends that you NOT use Hierarchical pop-up menus. You should consider very carefully if you really need to use a pop-up menu. Some features could better be implemented with icons or with the normal menus in the menu bar. Don’t use commands in your pop-up menus unless you have to. If you do use commands, you should duplicate the item in the menu bar menus. “All commands that are in general use throughout the application should appear in the menu bar. This assures that the most important commands are always visible and available to the user”. (refer to “Commands in pop-up Menu” from HI Update #13).

How it works....

Inside Macintosh Volume. V menu manager routines added one new ROM function for doing pop-up menus. The function PopUpMenuSelect allows pop-up menus to be handled and created anywhere on the screen. Since the pop-up menu resources are handled the same as for other menus, they may contain color and submenus. The True Basic demo program has a basic event loop structure which can be used as a basic skeleton for other applications if you want. The libraries which are used come from the True Basic Macintosh Programmer’s Kit which is essential for any serious Macintosh program development.

To implement pop-up menus, the program uses an array to set up the menu items. These items are declared in the PopItem$ array which is dimensioned near the beginning of the program. SysEnvirons is called to see if the system has the pop-up menu routines. If it doesn’t, the program quits. It would be better to display a dialog to warn the user why the program quit. In addition to setting up the menu bar, the menu used in the pop-up is created and stored in a resource where MyMenus$(4) is the handle to the menu resource. After the menu resource is created, the only thing left to do is draw the pop-up title and shadowed box then wait for an event.

The top and left corner of the pop-up are assigned at the beginning of the Drawpopup routine so that you may place the pop-up wherever you like. Be sure to leave room for the title. The routine figures out the width of the Title and longest menu item and draws them at the appropriate locations. When the pop-up menu comes up, the shadowed box should always line up with the pop-up menu. This is simple, but you can get confused unless you realize that the shadowed box and title are drawn with local window coordinates, but the pop-up menu location is determined by global screen coordinates. It is necessary to convert this location from local to global to tell the pop-up select routine where the menu should go. If the window gets moved (by someone dragging it), remember that the global coordinate is now different, but the local coordinate stayed the same.

You determine that a pop-up menu needs to be handled by checking to see if the mouse was clicked within the shadowed box. The GetNextEvent function returns the point where the click occurred. This point is compared with the PtInRect routine to determine if the point is within the shadowed rectangle where the pop-up will appear. If so then the program handles the pop-up event by calling the PopUpMenuSelect routine. PopUpMenuSelect takes the menu handle, the left and top coordinates where the menu will appear, and the currently selected menu item and handles the menu by displaying the menu then returns the item number of the menu item that was selected. That item is then redrawn into the shadowed box and becomes the newly selected item. The program should then handle whatever action or flag the menu item should do.

This process is simple enough after seeing a few examples. The main problem with pop-up menus as with other controls that you may want to use is that you must be careful not to clutter up the window too much. The graphic design of the interface is just as important as the code that is written. That’s why there will always be a demand for good programs even though HyperCard and other tools make developing programs easier.

By the way, some of the programming in True Basic can be simplified by using the TrueWindows Library. The problem comes when you want to create an application using the Runtime package. The Binder utility asks for the compiled libraries and then the compiled program and outputs them to an application that contains the True Basic interpreter. If you use libraries that have calls or functions with the same name, they may work fine until you bind them together into an application with Binder. Binder gives an error when putting the libraries together which says that you have more than one call with the same name. This could happen if you are using TrueWindows and the other Macintosh Programmer’s Kit routines (like the ones used in this program). You may be able to get around this by going to the source code files and removing the duplicate name and recompiling the library. Be sure to keep it separate from the original so that they don’t get mixed up or someday you may need a routine that you deleted and wonder why things don’t work.

I set up my copy of the True Basic runtime application with my own icon and when I create an application, the information is already set up provided that I want to use the same one each time I make a new application. Usually you will want them to be different. I use the Prof. Mac icon for any applications that I make for MacTutor.

! PopUp Menu Demo
! True Basic version 2.01
! Requires True Basic Macintosh Developer’s ToolKit Libraries
! by Dave Kelly
! ©1989 MacTutor

REM Open up libraries
LIBRARY “MenuLib*”                ! Menu Manager
LIBRARY “WindowLib*”              ! Window Manager
LIBRARY “DeskLib*”                ! Desk Manager
LIBRARY “EventLib*”               ! Event Manager
LIBRARY “QuickLib*”               ! Quickdraw
LIBRARY “DataLib*”                ! Desk Acc and system calls
LIBRARY “MacLib*”                 ! True Basic event control
LIBRARY “System*”                 ! System Calls

REM following variables are used globally throughout program
DECLARE DEF NIL$, POINTER$,screenBits$,bounds$,top,left,bottom,right,TopLeft$,H,V
DECLARE DEF OpenDeskAcc,NewWindow$,SystemEdit
DECLARE DEF MenuSelect,PtInRect,TrackGoAway

DIM MyMenus$(1:4)
DIM PopItem$(1:3)
CALL SysEnvirons(sysEnvRec$,status)    ! Get current system revision
CALL UnpackEnvirons(sysEnvRec$,envversion, machine,sysversion,processor, 
hasFPU,hasColorQD,keyboardtype,atversion, sysvrefnum)
IF sysversion=0 then              ! Do we have the right ROM?
CALL TakeMac   ! turn off True Basic and let the program do its own thing
LET everyevent=-1                 ! event mask for all events
LET doneFlag=0                    ! this flag is set when program ending 
has been selected.
LET z$=bounds$(screenBits$)       ! Get the size of the screen.
CALL setrect(r$,left(z$)+4,top(z$)+44,right(z$)-4,bottom(z$)-4)
CALL setrect(dragrect$,4,24,right(z$)-4,bottom(z$)-4)
LET myWindow$=NewWindow$(NIL$,r$,”Sample”,1,0,POINTER$(-1),1,0)      

! Create a window
CALL SetPort(myWindow$)           ! Access the new window
CALL SetUpMenus                   ! Turn on menus
CALL Drawwindow                   ! Set up window info

! Main Event Loop

   CALL SystemTask                ! Handle System tasks/DAs
   CALL GetNextEvent(everyevent,theEvent$,eResult)    
! check for events
   IF eResult<>0 then    ! if no event error occurred then...
      CALL UnpackEvent(theEvent$,what,mess,when,where$,mod)
      SELECT CASE what 
! what represents the kind of event that occurred.
      CASE 1                      ! mouse down event occurred
           CALL FindWindow(where$,whichWindow$,wResult)
           SELECT CASE wResult
           CASE 1         ! Event was in the menu bar
                LET mResult=MenuSelect(where$)
                CALL DoMenu(mResult)
           CASE 2         ! Event was in a system window
                CALL SystemClick(theEvent$,whichWindow$)   
 ! Pass the event to the system
           CASE 3   ! Event was in content region of a window
                CALL GlobalToLocal(where$)  
 ! convert coordinates for the window
                IF PtInRect(where$,PopRect$)=1 THEN   
 ! see if popup was selected
                   CALL PopUpEvent     ! if so, handle the popup event
                END IF
           CASE 4   ! Event in the window’s drag region
                CALL DragWindow(whichWindow$,where$,dragrect$)
           CASE 6   ! Event in go-away region of active window
                LET doneFlag=TrackGoAway(whichWindow$,where$)
           CASE else
           END SELECT
      CASE 6                      ! update event occurred
           CALL Packb(w$,1,32,mess)
           CALL BeginUpdate(w$)
           CALL Drawwindow
           CALL DrawPopUp
           CALL EndUpdate(w$)
      CASE else                   ! anything else?
LOOP until doneFlag<>0

CALL DisposeWindow(myWindow$)     ! Throw away window handle
CALL ClearMenuBar                 ! Clear Menus
FOR i=Lbound(MyMenus$) to Ubound(MyMenus$)
    CALL DisposeMenu(MyMenus$(i))
CALL GiveMac                      ! Return control back to True Basic
STOP                              ! End the program

SUB DrawWindow                    ! Draw message in window
    CALL textfont(2)              ! Set font to New York font
    CALL textsize(12)             ! Set size to 12 point
    CALL textface(1)              ! Set text to bold
    CALL textmode(0)              ! Set to copy mode
    CALL moveto(10,20)
    CALL DrawString(“True BASIC Version 2.0 PopUp Menu demo”)
    CALL textface(0)              ! Set text to plain

SUB DoMenu(code)                  ! handle Menu events
    CALL Packb(s$,1,32,code)
    LET MenuNumber=Unpackb(s$,1,-16)
    LET Menuitem = Unpackb(s$,17,-16)
    SELECT CASE MenuNumber
    CASE 1                        ! Apple Menu
         CALL GetItem(MyMenus$(1),MenuItem,name$)
         LET mrefNum=OpenDeskAcc(name$)
         CALL SetPort(mywindow$)
    CASE 2                        ! File Menu
         LET doneFlag=-1
    CASE 3                        ! Edit Menu
         LET z=SystemEdit(Menuitem+1)
    CASE else
    CALL HiliteMenu(0)

SUB SetUpMenus
    DECLARE DEF NewMenu$,StringWidth ! Declare variables used
    DECLARE DEF GetFontInfo$      ! in toolbox functions

    LET MyMenus$(1)=NewMenu$(1,chr$(20))  ! The first menu is 
    CALL AddResMenu(MyMenus$(1),”DRVR”)   ! Apple menu.
    LET MyMenus$(2)=NewMenu$(2,”File”)   ! File menu is second
    CALL AppendMenu(MyMenus$(2),”Quit”)
    LET MyMenus$(3)=NewMenu$(3,”Edit”)   ! Next the Edit menu
    CALL AppendMenu(MyMenus$(3),”Cut”)
    CALL AppendMenu(MyMenus$(3),”Copy”)
    CALL AppendMenu(MyMenus$(3),”Paste”)
    LET PopTitle$=”PopUp Menu Title:  “   ! Save pop up title
    LET MyMenus$(4)=NewMenu$(4,PopTitle$) ! Create pop up menu
    LET Popitem$(1)=”Item 1"
    LET Popitem$(2)=”Item 2"
    LET Popitem$(3)=”Item 3"
    LET NoOfPopItems=3
    LET PopItem=1
    FOR i=1 to 3
        CALL AppendMenu(MyMenus$(4),Popitem$(i))  ! Add popup items
    NEXT i
    FOR i=lbound(MyMenus$) to Ubound(MyMenus$)-1  ! put the menus into
        CALL insertMenu(MyMenus$(i),0)      ! the menu bar
    NEXT i
    CALL InsertMenu(MyMenus$(4),-1)    ! Add pop up menu
    CALL CheckItem(MyMenus$(4),PopItem,1)   ! check default item

    REM Get maximum length of PopUp items
    CALL TextFont(0)              ! Set to system font
    CALL TextSize(12)             ! Set to 12 point size
    CALL GetFontInfo(FontInfo$)
    LET ascent = Unpackb(fontinfo$,1,-16)
    LET descent = Unpackb(fontinfo$,17,-16)
    LET widMax = Unpackb(fontinfo$,33,-16)
    LET leading = Unpackb(fontinfo$,49,-16)
    LET MaxItemLength=0

    FOR i=1 to NoOfPopItems
        LET strwidth=StringWidth(Popitem$(i))
        IF StrWidth>MaxItemLength then LET MaxItemLength=StrWidth
    NEXT i

    CALL DrawPopUp
    CALL DrawMenuBar

SUB DrawPopUp
    CALL TextFont(0)       ! Set Font to Chicago (System)
    CALL TextSize(12)             ! Set Size to 12 point
    LET Popuptop=100              ! Top of Popup menu
    LET Popupleft=200             ! Left of Popup menu
    CALL SetRect(PopRect$, Popupleft,Popuptop, Popupleft+5+MaxItemLength+13, 
Popuptop+ascent+ descent+ leading+1)
    CALL FrameRect(PopRect$)   ! Draw currently selected item
    CALL MoveTo(Right(PopRect$),Top(PopRect$)+1)
    CALL LineTo(Right(PopRect$),Bottom(PopRect$))
    CALL MoveTo(Left(PopRect$)+1,Bottom(PopRect$))
    CALL LineTo(Right(PopRect$),Bottom(PopRect$))
    LET StrWidth=StringWidth(PopTitle$)
    LET xlocation=Left(PopRect$)-StrWidth
    LET ylocation=(Top(PopRect$)+Bottom(PopRect$))/2+(ascent-descent)/2
    CALL MoveTo(xlocation,ylocation)
    CALL Drawstring(PopTitle$)    ! Draw the Popup menu title
    CALL SetRect(InvertTitleRect$,xlocation-8,Top(PopRect$)+1,Left(PopRect$),Bottom(PopRect$))
    LET xlocation=Left(PopRect$)+13
    CALL MoveTo(xlocation,ylocation)
    CALL Drawstring(PopItem$(PopItem))      
 ! Draw the currently selected item

SUB PopUpEvent
    DECLARE DEF PopUpMenuSelect   ! Declare function
    CALL InvertRect(InvertTitleRect$)  ! invert popup title
    LET TempPoint$=TopLeft$(PopRect$)
    CALL LocalToGlobal(TempPoint$)   ! Change to global coords
    LET PopTop=V(TempPoint$)+1
    LET PopLeft=H(TempPoint$)+1
    LET Result=PopUpMenuSelect(MyMenus$(4),PopTop,PopLeft,PopItem)
   ! Do the Popup
    CALL Packb(s$,1,32,Result)    ! Get the menu result
    LET MenuNumber=Unpackb(s$,1,-16) 
  ! Ignore Menunumber, we know which menu this is
    LET Menuitem = Unpackb(s$,17,-16)  ! Get the menu item

    IF MenuItem=PopItem THEN
       CALL InvertRect(InvertTitleRect$) 
    ! Invert the title to normal if old item selected
       CALL CheckItem(MyMenus$(4),PopItem,0) ! uncheck last item
       CALL CheckItem(MyMenus$(4),MenuItem,1) ! check new item
       CALL EraseRect(PopRect$)   ! Draw the current menu item
       CALL FrameRect(PopRect$)
       CALL MoveTo(xlocation,ylocation)
       CALL Drawstring(PopItem$(PopItem))
       CALL InvertRect(InvertTitleRect$)
       SELECT CASE MenuItem       ! Handle menu event
       CASE 1
            REM Do Item 1
            LET PopItem=MenuItem
       CASE 2
            REM Do Item 2
            LET PopItem=MenuItem
       CASE 3
            REM Do Item 3
            LET PopItem=MenuItem
       CASE ELSE
       CALL MoveTo(xlocation,ylocation)
       CALL TextFont(0)    ! Set font to Chicago (System)
       CALL Drawstring(PopItem$(PopItem))   ! draw selected popup item
    END IF


Community Search:
MacTech Search:

Software Updates via MacUpdate

Opera 40.0.2308.90 - High-performance We...
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
BetterTouchTool 1.93 - Customize Multi-T...
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
Backblaze - Online backup serv...
Backblaze is an online backup service designed from the ground-up for the Mac. With unlimited storage available for $5 per month, as well as a free 15-day trial, peace of mind is within reach with... Read more
Postbox 5.0.5 - Powerful and flexible em...
Postbox is a new email application that helps you organize your work life and get stuff done. It has all the elegance and simplicity of Apple Mail, but with more power and flexibility to manage even... Read more
Coda 2.5.19 - 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
Toast Titanium 15.1 - $99.99
Roxio Toast 15 Titanium, the leading DVD burner for Mac, makes burning even better, adding Roxio Secure Burn to protect your files on disc and USB in Mac- or Windows-compatible formats. Get more... Read more
Firetask 3.8.1 - Innovative task managem...
Firetask uniquely combines the advantages of classical priority-and-due-date-based task management with GTD. Stay focused and on top of your commitments - Firetask's "Today" view shows all relevant... Read more
Chromium 54.0.2840.71 - Fast and stable...
Chromium is an open-source browser project that aims to build a safer, faster, and more stable way for all Internet users to experience the web. Version 54.0.2840.71: Release notes were unavailable... Read more
Chromium 54.0.2840.71 - Fast and stable...
Chromium is an open-source browser project that aims to build a safer, faster, and more stable way for all Internet users to experience the web. Version 54.0.2840.71: Release notes were unavailable... Read more
Firetask 3.8.1 - Innovative task managem...
Firetask uniquely combines the advantages of classical priority-and-due-date-based task management with GTD. Stay focused and on top of your commitments - Firetask's "Today" view shows all relevant... Read more

Latest Forum Discussions

See All

WitchSpring2 (Games)
WitchSpring2 1.27 Device: iOS Universal Category: Games Price: $3.99, Version: 1.27 (iTunes) Description: This is the story of Luna, the Moonlight Witch as she sets out into the world. This is a sequel to Witch Spring. Witch Spring 2... | Read more »
Best Fiends Forever Guide: How to collec...
The fiendship in Seriously's hit Best Fiends has been upgraded this time around in Best Fiends Forever. It’s a fast-paced clicker with lots of color and style--kind of reminiscent of a ‘90s animal mascot game like Crash Bandicoot. The game... | Read more »
5 apps for the budding mixologist
Creating your own cocktails is something of an art form, requiring a knack for unique tastes and devising interesting combinations. It's easy to get started right in your own kitchen, though, even if you're a complete beginner. Try using one of... | Read more »
5 mobile strategy games to try when you...
Strategy enthusiasts everywhere are celebrating the release of Civilization VI this week, and so far everyone seems pretty satisfied with the first full release in the series since 2010. The series has always been about ultra-addictive gameplay... | Read more »
Popclaire talk to us about why The Virus...
Humanity has succumbed to a virus that’s spread throughout the world. Now the dead have risen with a hunger for human flesh, and all that remain are a few survivors. One of those survivors has just called you for help. That’s the plot in POPCLAIRE’... | Read more »
Oceans & Empires preview build sets...
Hugely ambitious sea battler Oceans & Empires is available to play in preview form now on Google Play - but download it quickly, as it’s setting sail away in just a few days. [Read more] | Read more »
Rusty Lake: Roots (Games)
Rusty Lake: Roots 1.1.4 Device: iOS Universal Category: Games Price: $2.99, Version: 1.1.4 (iTunes) Description: James Vanderboom's life drastically changes when he plants a special seed in the garden of the house he has inherited.... | Read more »
Flippy Bottle Extreme! and 3 other physi...
Flippy Bottle Extreme! takes on the bottle flipping craze with a bunch of increasingly tricky physics platforming puzzles. It's difficult and highly frustrating, but also addictive. When you begin to master the game, the sense of achievement is... | Read more »
Plants vs. Zombies Heroes guide: How to...
Plants vs. Zombies Heroes surprised us all, presenting a deep deck building experience. It's a great CCG that stands up well to the competition. There are a lot of CCGs vying for players' attention at the moment, but PvZ Heroes is definitely one... | Read more »
Arcane Online takes Online RPG’s to anot...
If you think that you need a desktop to enjoy high quality MMO gaming then Arcane Online hopes to prove you emphatically wrong. An epic fantasy Online RPG set in the land of Eldine, Arcane Online offers an abundance of features and content that... | Read more »

Price Scanner via

Apple’s Thursday “Hello Again” Event A Largel...
KGI Securities analyst Ming-Chi Kuo, who has a strong record of Apple hardware prediction accuracy, forecasts in a new note to investors released late last week that a long-overdue redo of the... Read more
12-inch Retina MacBooks on sale for $100 off...
Amazon has 2016 12″ Apple Retina MacBooks on sale for $100 off MSRP. Shipping is free: - 12″ 1.1GHz Silver Retina MacBook: $1199.99 $100 off MSRP - 12″ 1.1GHz Gold Retina MacBook: $1199.99 $100 off... Read more
Save up to $600 with Apple refurbished Mac Pr...
Apple has Certified Refurbished Mac Pros available for up to $600 off the cost of new models. An Apple one-year warranty is included with each Mac Pro, and shipping is free. The following... Read more
PixelStyle Inexpensive Photo Editor For Mac W...
PixelStyle is an all-in-one Mac Photo Editor with a huge range of high-end filters including lighting, blurs, distortions, tilt-shift, shadows, glows and so forth. PixelStyle Photo Editor for Mac... Read more
13-inch MacBook Airs on sale for $100-$140 of...
B&H has 13″ MacBook Airs on sale for $100-$140 off MSRP for a limited time. Shipping is free, and B&H charges NY sales tax only: - 13″ 1.6GHz/128GB MacBook Air (sku MMGF2LL/A): $899 $100 off... Read more
2.8GHz Mac mini available for $988, includes...
Adorama has the 2.8GHz Mac mini available for $988, $11 off MSRP, including a free copy of Apple’s 3-Year AppleCare Protection Plan. Shipping is free, and Adorama charges sales tax in NY & NJ... Read more
21-inch 3.1GHz 4K on sale for $1379, $120 off...
Adorama has the 21″ 3.1GHz 4K iMac on sale $1379.99. Shipping is free, and Adorama charges NY & NJ sales tax only. Their price is $120 off MSRP. To purchase an iMac at this price, you must first... Read more
Check Apple prices on any device with the iTr...
MacPrices is proud to offer readers a free iOS app (iPhones, iPads, & iPod touch) and Android app (Google Play and Amazon App Store) called iTracx, which allows you to glance at today’s lowest... Read more
Apple, Samsung, Lead J.D. Power Smartphone Sa...
Customer satisfaction is much higher among smartphone owners currently subscribing to full-service wireless carriers, compared with those purchasing service through a non-contract carrier, according... Read more
Select 9-inch Apple WiFi iPad Pros on sale fo...
B&H Photo has select 9.7″ Apple WiFi iPad Pros on sale for up to $50 off MSRP, each including free shipping. B&H charges sales tax in NY only: - 9″ Space Gray 256GB WiFi iPad Pro: $799 $0 off... Read more

Jobs Board

*Apple* Retail - Multiple Positions- Nashua,...
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- Napervi...
Job Description:SalesSpecialist - Retail Customer Service and SalesTransform Apple Store visitors into loyal Apple customers. When customers enter the store, Read more
Security Data Analyst - *Apple* Information...
…data sources need to be collected to allow Information Security to better protect Apple employees and customers from a wide range of threats.Act as the subject Read more
*Apple* Retail - Multiple Positions (Multi-L...
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- New Yor...
Sales Specialist - Retail Customer Service and Sales Transform Apple Store visitors into loyal Apple customers. When customers enter the store, you're also the Read more
All contents are Copyright 1984-2011 by Xplain Corporation. All rights reserved. Theme designed by Icreon.