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?
   STOP
END IF
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

DO
   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?
      END SELECT
   END IF
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))
NEXT 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
END SUB

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
    END SELECT
    CALL HiliteMenu(0)
END SUB

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
END SUB

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
END SUB

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
    ELSE
       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
       END SELECT
       CALL MoveTo(xlocation,ylocation)
       CALL TextFont(0)    ! Set font to Chicago (System)
       CALL Drawstring(PopItem$(PopItem))   ! draw selected popup item
    END IF
END SUB
END

 

Community Search:
MacTech Search:

Software Updates via MacUpdate

NetNewsWire 5.1 - RSS and Atom news read...
NetNewsWire is the best way to keep up with the sites and authors you read most regularly. Let NetNewsWire pull down the latest articles, and read them in a distraction-free and Mac-like way. Native... Read more
FontExplorer X Pro 7.1.3 - Font manageme...
FontExplorer X Pro is optimized for professional use; it's the solution that gives you the power you need to manage all your fonts. Now you can more easily manage, activate and organize your... Read more
DiskCatalogMaker 8.2.5 - Catalog your di...
DiskCatalogMaker is a simple disk management tool which catalogs disks. Simple, light-weight, and fast Finder-like intuitive look and feel Super-fast search algorithm Can compress catalog data for... Read more
Skim 1.5.12 - PDF reader and note-taker...
Skim is a PDF reader and note-taker for OS X. It is designed to help you read and annotate scientific papers in PDF, but is also great for viewing any PDF file. Skim includes many features and has a... Read more
rekordbox 6.1.0.0030 - Professional DJ m...
rekordbox is the best way of preparing and managing your tracks, be it at home, in the studio, or even on the plane! It allows you to import music from other music-management software using the... Read more
iExplorer 4.4.0 - View and transfer file...
iExplorer is an iPhone browser for Mac lets you view the files on your iOS device. By using a drag and drop interface, you can quickly copy files and folders between your Mac and your iPhone or... Read more
OmniGraffle 7.17.5 - Create diagrams, fl...
OmniGraffle helps you draw beautiful diagrams, family trees, flow charts, org charts, layouts, and (mathematically speaking) any other directed or non-directed graphs. We've had people use Graffle to... Read more
Apple Configurator 2.13.1 - Configure an...
Apple Configurator makes it easy to deploy iPad, iPhone, iPod touch, and Apple TV devices in your school or business. Use Apple Configurator to quickly configure large numbers of devices connected to... Read more
OmniGraffle Pro 7.17.5 - Create diagrams...
OmniGraffle Pro helps you draw beautiful diagrams, family trees, flow charts, org charts, layouts, and (mathematically speaking) any other directed or non-directed graphs. We've had people use... Read more
FoldersSynchronizer 5.1.6 - Synchronize...
FoldersSynchronizer is a popular and useful utility that synchronizes and backs-up files, folders, disks and boot disks. On each session you can apply special options like Timers, Multiple Folders,... Read more

Latest Forum Discussions

See All

Steam Link Spotlight - Hades
Steam Link Spotlight is a feature where we look at PC games that play exceptionally well using the Steam Link app. Our last entry was on Disco Elysium. Read about how it plays using Steam Link over here. | Read more »
Microsoft has acquired ZeniMax Media and...
In the latest of a series of blockbuster moves, Microsoft has now acquired Zenimax Media and its subsidiary, Bethesda Softworks, for $7.5 billion. [Read more] | Read more »
Infinity Mechs is an upcoming idle game...
Indie developer SkullStar studio has announced an upcoming idle mech game called Infinity Mechs. It draws inspiration from the mobile game Iron Saga and has been officially licensed by Game Duchy. It's set to launch for both iOS and Android on... | Read more »
PUBG Mobile Lite's latest update se...
PUBG Mobile Lite, the streamlined version of the popular battle royale that's designed to work on less powerful devices, sees the return of a popular game variant today, Survive Till Dawn mode. It arrives as part of the 0.19.0 content update. [... | Read more »
Matchy Catch, Jyamma Games’ new hyper-ca...
Matchy Catch is a new hyper-casual puzzler from Jyamma Games, the Italian studio behind the Pong-inspired puzzle-adventure Hi-Ball Rush. It’s only the developer’s second game for iOS and Android devices, but it promises to be every bit as fun and... | Read more »
Among Us! Imposter Guide - How to be a s...
Among Us! continues to be getting a lot of play in these parts, and since our first guide we've learned a thing or two about the game. This is especially true regarding the imposter role, as its a relatively rare opportunity that we've now put... | Read more »
Paladin's Story is an upcoming fant...
Paladin's Story is an upcoming fantasy RPG with an off-kilter sense of humour that's heading for iOS and Android. It will officially launch for both on September 16th though the game is already available on Google Play in Early Access. [Read more... | Read more »
Among Us! Guide - Tips for the uninitiat...
| Read more »
A Pretty Odd Bunny is a stealth-platform...
A Pretty Odd Bunny is a stealth-platformer from two-man team AJ Ordaz and René Rivera. It follows the story of a red-eyed rabbit who is allergic to carrots and instead has a penchant for devouring pigs. It's available now for Android devices. [... | Read more »
Apple Arcade: Ranked - Top 25 [Updated 9...
In case you missed it, I am on a quest to rank every Apple Arcade game there is. [Read more] | Read more »

Price Scanner via MacPrices.net

The cheapest Macs are back in stock today at...
Apple has restocked clearance, previous-generation, Certified Refurbished Mac minis starting at only $599. Each mini comes with free shipping plus Apple’s standard one-year warranty. These are the... Read more
Sale! Amazon has 2020 13″ 2.0GHz MacBook Pros...
Amazon has 2020 13″ MacBook Pros with 10th generation Intel CPUs back in stock on sale again today for $150-$200 off Apple’s MSRP. Shipping is free. Be sure to purchase the MacBook Pro from Amazon,... Read more
Base 13″ 1.4GHz Apple MacBook Pros on sale fo...
Apple reseller Expercom is offering a $65-$75 discount on new 2020 13″ 1.4GHz MacBook Pros, depending on configuration. Shipping is free. Expercom estimates shipping in 3-5 days, as stock of Apple’s... Read more
Price drop! Get a 44mm Apple Watch Series 5 G...
Amazon has dropped their price on the 44mm Apple Watch Series 5 GPS + Cellular by $100 to $429 shipped. That’s $100 off Apple’s original MSRP for this model. For the latest prices and sales, see our... Read more
Verizon offers $200 discount on new Apple Wat...
Verizon will take up to $200 off the purchase of a new GPS + Cellular Apple Watch Series 6 or Apple Watch SE with select trade-in and the purchase of a new iPhone with service. The fine print: “Get... Read more
Verizon offers $250 discount on new 8th gener...
Verizon will take up to $250 off the price of an 8th generation 2020 Apple Cellular iPad with select trade-ins and a new iPhone purchase. Plus get Apple News+ free for 6 months. The fine print: “Save... Read more
Apple’s Implementation Of COVID-19 Exposure...
NEWS: 09.18.20 – The latest effort by Apple to embed exposure notifications for COVID-19 contact tracing right into its mobile operating system has some iPhone users weary of being exposed to... Read more
Here’s how to get a 16″ MacBook Pro for $300...
B&H Photo has new 16″ MacBook Pros on sale today for $250-$300 off Apple’s MSRP, starting at $2099. Expedited shipping is free to many addresses in the US: – 2019 16″ 2.6GHz 6-Core MacBook Pro... Read more
Apple has Certified Refurbished 16″ MacBook P...
Apple has Certified Refurbished 2019 16″ MacBook Pros available for up to $420 off the cost of new models, starting at $2039. Each model features a new outer case, shipping is free, and an Apple 1-... Read more
Price drops! Apple reseller B&H drops App...
B&H Photo has dropped prices on Apple Watch Series 5 models by $50-$70 off Apple’s original MSRP. Shipping is free. These are the same Apple Watch models sold by Apple in their retail and online... Read more

Jobs Board

Security Officer ($23.00/Hourly) - *Apple*...
**Security Officer \($23\.00/Hourly\) \- Apple Store** **Description** About NMS Built on a culture of safety and integrity, NMSdelivers award\-winning, integrated Read more
Security Officer ($23.00/Hourly) - *Apple*...
**Security Officer \($23\.00/Hourly\) \- Apple Store** **Description** About NMS Built on a culture of safety and integrity, NMSdelivers award\-winning, integrated Read more
Platform - Workplace Eng - *Apple* Enterpri...
MORE ABOUT THIS JOB We are looking for an Apple Platform Engineer who will bring a unique engineering skill set, support, clarity, organization and above all else, Read more
*Apple* Certified Repair Technician - Utah S...
…selected candidate will work in the USU Campus Store Tech Department as an Apple Certified Repair Technician and floor associate. This position is for both summer Read more
Senior Data Engineer - *Apple* - Theorem, L...
Job Summary Apple is seeking an experienced, detail-minded data engineeringconsultant to join our worldwide business development and strategy team. If you are Read more
All contents are Copyright 1984-2011 by Xplain Corporation. All rights reserved. Theme designed by Icreon.