TweetFollow Us on Twitter

HyperArrays
Volume Number:6
Issue Number:6
Column Tag:HyperChat™

HyperArrays

By Fred Stauder, Scott Vore, Indianapolis, IN

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

on HyperChat

HyperMedicine: Using Hypercard to Solve Problems

This month we will look at ways people have solved problems in the medical field (my origin) using Hypercard. One of the reasons Hypercard is so popular in the medical field is that after spending so many years learning to be a doctor people feel that investing more to be a computer programmer is not justified. There is very much a “We want solutions now!” mentality.

The first problem was how do you get the patient more involved in his/her case decision making process without tying up too much of your surgeon’s time? Harold Lyon from Dartmouth Medical School used Hypercard and a laserdisc to form an interactive decision making tool. The topic is “Prostatectomy or Watchful Waiting”. The disc shows the risks and rewards of having the operation. Patients are interviewed and even doctors that have been patients. It gives you all the pros and cons of having the operation. The other purpose of the laserdisc is to gather data about patients and follow up treatments. This program that they have created has legal, ethical, educational, and research implications. It also protects the doctor by documenting the patients informed consent.

Harold Lyon gives some useful tips in preparing a videodisc>

- Always put SMPTE on the source tape.

- Flowchart the disc

- Use the same microphones at the same distance to maintain sound quality

- Train the speakers not to speak at the same time

- Work with one studio and film group

- Use small organized teams

- Never try to insert single words in audio

- Reach final script consensus prior to off-line edit

- Use colorbars on original source tapes

- Keep good track of script versions for team

- Plan the important details first

- Prepare more before on-line edit

- The video editor must be part of the team not someone hired for piecemeal work

- Consider carefully the use of one versus two camera shoots

- Ask interviewee the questions on video. Tape record these questions, play them back, and write them down. Take some head shots for cutaways. Then take the interviewer asking the same questions with a delay between each. Also take shots of interviewer sitting quietly for cutaways.

- Allow some spontinaity on line

- Film making and videodisc film making require different skills- be careful who you hire.

- Evaluate the disc before it is pressed changes are hard to make after pressing

- MacRecorder from Farralon was invaluable

- Consult experts

- Study other potential markets for your videodisc before shooting. With minor additions different applications can be made.

These tips are typical of the type of problems you will encounter when you make a videodisc.

The second interesting application is an article on XCMD’s written By Scott Vore MD. He is an anesthesiologist who wanted to track patients blood pressure, heart rate, drugs administered etc. He decided to do it in Hypercard however he found that he had to keep track of an array of data. So he wrote an XCMD called Hyperarray to do just that.

If you hear of any interesting applications Hypercard has been put to send them to us.

Send articles ideas, comments etc to Applelink: STAUDER

end HyperChat

HYPER ARRAYS

[Scott’s background is entirely self taught but he has been at it for 3 plus years now]

First of all, before you read any further I feel compelled to make the following disclaimers:

I am not, never have been, never plan to be a ‘real’ programmer. I’m totally self taught, thanks in most part to a handful of books and one excellent journal (we know which one)-consequently my programming ‘style’, if ‘style’ can describe it, contains bits and pieces of code examples I’ve seen elsewhere and often; in my haste to get things to work, I’ll leave in bits and pieces of code that serve no real purpose but should be removed if programming correctness were to be maintained. I’m sorry if this has happened here- with more time I could make it all prettier but I hope I’ve given someone enough to work with that they too can write and make things neat. My only motivation has been to make this machine work for me ,(with no formal training I have never felt competent to try and teach others). In that process, however I have learned a few things and some of those things seem important ( I guess I’ll let the editors decide just how important). Oh well, on with the show...

In my programming experience with Hypercard, I have never ceased to be amazed at the power , elegance, and simplicity inherent in this product. Every now and again, however, we all run up against ‘the wall’ and are forced to either change our approach or find another method to solving a particular problem.

Currently I am writing a stack that will be used in an Operating Room environment to allow an Anesthesiologist to record various pieces of data at specific time intervals throughout a case. Typically, the blood pressure, heart rate, and various other measurements are kept track of throughout a case so that the Anesthesiologist can keep track of the trend in these parameters as time passes, additional drugs are administered, and as the surgery progresses.

My particular problem was in storing the information in a way that was accessible in a random fashion, retrievable between calls to different stacks and between calls to shutdown that stack. In other words I wanted a data structure that was ‘global’ in nature and that acted as an array.

Being a faithful MacTutor reader I am aware that we are being admonished to always try and solve our problems in Hypertalk before resorting to writing an XCMD, so in all fairness I must say that the XCMD described below does have a Hypertalk counterpart although somewhat more cumbersome. The example stack I have enclosed will compare the two methods.

Ok, enough of all this, what is my problem?

I wanted to create an array that was accessible throughout Hypercard. Fine. How? One method is to create a field somewhere in my stack that has each line representing a different time point and each item on that line representing a different array variable.

 e.g.   put 10 into line 10 of cd fld “array”

While this approach does serve its purpose well a multi-dimensional array must rely on scripting to enter the various data points with commas interspersed between items and those same commas must be stripped off as the data is retrievable- doable but messy.

My approach was to create a one dimensional array (though a multidimensional array can easily be created) as a resource, attach that resource to the stack in question and access that array with calls to an XCMD. The array is very quickly accessed, is ‘global’ in nature and can be accessed easily through Hypercard.

The first step consists in determining the size of the array. For the example stack I have created a 60 element array since in a time based environment 60 works out well. Now, the array Resource must be created. There are two options here- the resource can be created from the program (from the XCMD) or it can be created with Rmaker and pasted into the stack in question. I chose the latter approach ( to me, it was the easiest way).

At this point decide on a Resource type too.

I chose ANES for the simple reason that I am an anesthesiologist .

The Rmaker source code is as follows:

/* 1 */

 ANESTH.RSRC

 type ANES = GNRL
 ,1005
 .H
 0000 00000000 0000
 0000 00000000 0000
 0000 00000000 0000
 0000 00000000 0000
 0000 00000000 0000
 0000 00000000 0000
 0000 00000000 0000
 0000 00000000 0000
 0000 00000000 0000
 0000 00000000 0000
 0000 00000000 0000
 0000 00000000 0000
 0000 00000000 0000
 0000 00000000 0000
 0000 00000000 0000

This, obviously initializes the values to zero and sets the values in hexadecimal format though integer or any other Rmaker legal format could have been used. If the hex format is maintained and you want to store character values then be sure and use some variation of the numtostring function to convert the character to its ASCI value and the reciprocal function to convert it back on retrieval.

After running Rmaker, the resource would exist with id 1005 and could then be pasted into the stack of your choice with Resedit. So, assuming this has all been done the next step is to write the XCMD’s.

At this point knowing exactly what you want the XCMD to do will save all sorts of frustration and reworking later (trust me). So, a small digression is in order as we set up the stack and decide what it is we want to accomplish with our array.

First, and foremost, the book that everyone is talking about, Gary Bond’s XCMD’s for Hypercard, should be your computer side companion if you want quick access to terminology, examples and good programming style while writing your commands. Next, depending on the development system you are using open the HyperX... interface files to see the exact format that the procedures are written in since they aren’t all written the same way that the examples in Gary Bond’s book assumes (some pass the paramblock ptr in all procedures).

For the sake of illustration and simplicity we will set up our stack in such a way that we will

(1) fill the array with 60 values from some source (a quick and dirty method will be to use Hypercard’s random number generator to generate 60 values) although these values could come from a field, an ask dialog, anywhere.

(2) we will access these values in two ways

(a) randomly-just to prove it works and

(b) in sequence to fill in a graph.

(3) as an added feature we will also continually update the graph described in 2b above so that new points can be added and the old ones removed.

The above stack will require at least three but probably four XCMD’s to fulfill it’s requirements. The first will fill the array RSRC with values on a random access basis. The second will retrieve values from the array and the third will create a ‘buffer array’ that will serve to save 60 data points- in the example stack this will hold the previous 60 points that the program generated so that they can be erased as the new points are plotted. This means that an additional ‘ANES’ RSRC must be created with a different id to serve as a buffer- but that is simply done with Resedit(copy/paste), and it’s get info menu item. The fourth XCMD will then access the buffer array and not the working array to retrieve previous ‘saved’ values’

Again, a slight digression is in order here, to stand back and see exactly what these items can allow us to do. In the example stack, the array is somewhat simple minded, but with a little imagination, the possibilities of such a Resource are wide. For example, the id of all cards can be kept in the array to give a ‘recording’ of movement throughout the stack, a record of any and all users of a stack can be kept, initialization values can be kept here, the buffer array can be huge so that it can be updated every time the working array is filled. Ok, maybe I’m the only one excited about all these possibilities, but hopefully some of you out there will be turned on too.

So, the first XCMD to write will be the one to access the array at any point and fill in the data at that point.

I have chosen to call it (in a fit of imagination and creativity- PUTDATA). The XCMD expects to receive two parameters- the time or array index and the value at that point. Below is the source code of the PUTDATA xcmd:

{2}
unit putdataXcmd;
interface
 uses MemTypes, QuickDraw, OSIntf, ToolIntf, PackIntf, HyperXCMD, QDAccess;

procedure putdata(ParamPtr: XCMDPtr);

implementation
 type
   timeDarray=array[0..59] of integer;
procedure arrayrsc(ParamPtr:XCMDPtr);forward;

  procedure putdata(paramptr:xcmdptr);
 begin
   arrayrsc(paramptr);
 end;

procedure arrayrsc(ParamPtr: XCMDPtr);
  var
 MYTIMEHAND:HANDLE;
 REFNUM:INTEGER;
 TIMEARRAY:TIMEDARRAY;
 temphandle:handle;
 tempstr:str255;
 horiz,vert:longint;
{*************** get points ************}
procedure getpoints( paramptr :xcmdptr;var  horiz,vert:longint);
begin zerotopas(paramptr,paramptr^.params[1]^,tempstr);
 horiz:=strtonum(paramptr,tempstr);
zerotopas(paramptr,paramptr^.params[2]^,tempstr);
 vert:=strtonum(paramptr,tempstr);
end;
{************************************}
begin
 mytimehand :=(getresource(‘ANES’,1005));
 hlock(mytimehand);
 blockmove(mytimehand^,@timearray, sizeof(timearray));
 getpoints(paramptr,horiz,vert);
 if  horiz > 59 then horiz := 59;
 timearray[horiz] := vert;
 blockmove(@timearray,mytimehand^,sizeof(timearray));
 refnum:=curresfile;
 changedresource(mytimehand);
 writeresource(mytimehand);
 releaseresource(mytimehand);
 end;
end.

The XCMD is lacking in many things, I didn’t error check after the call to getresource so the user will have no way of knowing whether the call was successful or not. Also there is no check on the number of parameters passed to the XCMD- shear laziness I guess. But, assuming that the user is the one writing the XCMD it should be his/her decision as to whether or not to make this XCMD general enough for the public at large or specific to his/her application (the latter approach implies that the programmer using the XCMD would be versed in it’s parameter requirements and would pass the correct values).

The second XCMD is very similar to the first, and in fact the two could be combined in such a way that a parameter check would tell if two parameters were passed indicating the user wanted to set the array value of param[1] to param[2] or if one parameter were passed indicating the user desired to retrieve the array value of param[1]. Again I’ve stuck with my original version of writing these things, not because they are better, but because they serve to indicate in some small sense, the evolution that these XCMD’s have undergone and they serve to suggest a number of ways to improve upon and write better XCMD’s. The second XCMD is titled ‘GetData’ and expects one argument as a parameter- the value of the array index to be read. The result is passed back to Hypercard in the result field but could easily be placed in a global variable or field.(See Gary Bond’s book for the gory details).

{3}

unit getdataXcmd;
interface
 uses MemTypes, QuickDraw, OSIntf, ToolIntf, PackIntf, HyperXCMD, QDAccess;
 procedure getdata(ParamPtr: XCMDPtr);
implementation
type
timeDarray=array[0..59] of integer;
procedure arrayrsc(ParamPtr: XCMDPtr);forward;
procedure getdata(paramptr:xcmdptr);
 begin
   arrayrsc(paramptr);
 end;
procedure arrayrsc(ParamPtr: XCMDPtr);
var
 MYTIMEHAND:HANDLE;
 REFNUM:INTEGER;
 TIMEARRAY:TIMEDARRAY;
 temphandle:handle;
 a:integer; 
 tempstr:str255;
 horiz:longint;
procedure getpoints(Paramptr:xcmdPtr;var num:longint);
var tempstr1:str255;
begin
  zerotopas(paramptr,paramptr^.params[1]^,tempstr1);
 horiz:=strtonum(paramptr,tempstr1);
end;

begin
 mytimehand :=(getresource(‘ANES’,1005));
 hlock(mytimehand);
  blockmove(mytimehand^,@timearray,sizeof(timearray));
 getpoints(paramptr,horiz);
 a:=timearray[horiz];
 longtostr(paramptr,a,tempstr);
 paramptr^.returnvalue:= pastozero(paramptr,tempstr);
  hunlock(mytimehand);
 releaseresource(mytimehand);
 end;
end.

Again, no error checking on getresource or number of params.

By themselves, or combined as a single XCMD, these examples will allow random access to a 60 point array from within Hypercard. The array size can easily be changed in the above examples and, in the initial resource definition, to allow any size array to be used--though I think the 32K limit also applies to resources.

The third XCMD I’ve written is used to set up a buffer so that the initial array can continue to be updated while the ‘old values’ are saved. An example of it’s use is in the example stack and has been described above. It’s call ‘DATABUFF’ and requires no parameters- assuming that the two array rsrc’s are present in the stack file and that their id’s are known. Clearly this approach does not allow much room for error but,with careful planning ,should not be much of a problem.

{4}

unit databuffXcmd;
interface
 uses MemTypes, QuickDraw, OSIntf, ToolIntf, PackIntf, HyperXCMD, QDAccess;

procedure databuff(ParamPtr: XCMDPtr);

implementation
type
 timeDarray=array[0..59] of integer;
procedure arrayrsc(ParamPtr: XCMDPtr);forward;
procedure databuff(paramptr:xcmdptr);
 begin
   arrayrsc(paramptr);
 end;

procedure arrayrsc(ParamPtr: XCMDPtr);
var 
  mytimehand,buffarrayH:HANDLE;
  REFNUM:INTEGER;
  TIMEARRAY,buffarray:TIMEDARRAY;
  temphandle:handle;
  tempstr:str255;
  horiz,vert:longint;
  numparams :integer;

begin
mytimehand := (getresource(‘ANES”,1005));
hlock(mytimehand); BLOCKMOVE(MYTIMEHAND^,@timearray,
                                 SIZEOF(TIMEARRAY));
releaseresource(mytimehand);
buffarrayH:=(getresource(‘ANES’,1010));
hlock(buffarrayH);
blockmove(buffarrayH^,@buffarray, sizeof(buffarray));
buffarray := timearray;
blockmove(@buffarray,buffarrayH^, sizeof(buffarray));
 
REFNUM:=CURRESFILE;
changedresource(buffarrayH);
writeresource(buffarrayH);
hunlock(buffarrayH);
releaseresource(buffarrayH);
end;
end.

The fourth XCMD must be used with the databuff XCMD. It is actually a copy of the XCMD ‘getdata’ with the exception that it accesses the buffer array (with a different id number) so that saved values can be read back and acted upon accordingly.

{5}

unit getprevdataXcmd;
interface
 uses MemTypes, QuickDraw, OSIntf, ToolIntf, PackIntf, HyperXCMD, QDAccess;
 procedure getprevdata(ParamPtr: XCMDPtr);
implementation
type
timeDarray=array[0..59] of integer;
procedure arrayrsc(ParamPtr: XCMDPtr);forward;
procedure getprevdata(paramptr:xcmdptr);
 begin
     arrayrsc(paramptr);
 end;
procedure arrayrsc(ParamPtr: XCMDPtr);
 var
 MYTIMEHAND:HANDLE;
 REFNUM:INTEGER;
 TIMEARRAY:TIMEDARRAY;
 temphandle:handle;
 tempstr:str255;
 a :integer;
 vert,b:integer;
 horiz:longint;
procedure getpoints(Paramptr:xcmdPtr;var;num:longint);
 var tempstr1:str255;
begin
 zerotopas(paramptr,paramptr^.params[1]^,tempstr1);
 horiz:=strtonum(paramptr,tempstr1);
end;
 begin
 mytimehand:=(getresource(‘ANES’,1010));
 HLOCK(MYTIMEHAND);
 blockmove(mytimehand^,@timearray,sizeof(timearray));
 getpoints(paramptr,horiz);
 a:=timearray[horiz];
 longtostr(paramptr,a,tempstr);
 paramptr^.returnvalue:=pastozero(paramptr, tempstr);
 releaseresource(mytimehand);
 end;
end.

Figure 1 is a table summarizing the syntax of the XCMD’s and the id’s of the array resources.

Well, that’s about it for the XCMD’s. There is a lot of room for improvement here and I believe a lot of room to explore. I’ve included the XCMD’s and an example stack that demonstrates their use as well as the Hypertalk method of using an array. I apologize again for my lack of programming expertise and hope that it doesn’t detract from the overall article. I would like to write another article in the near future that gives a few tricks to use the third party product “Programmer’s Extender” in XCMD’s in order to save even more time and effort in writing XCMD’s- let me know if anyone is interested in this.

The example stack contains three buttons:

The first will demonstrate the use of a Hypertalk array in the best way I know how - it’s entitled “the old way” and basically initializes 50 lines of an invisible card field to some random number .

It’s script is as follows:

--6

on mouseup
put 1 into linecount
 repeat(50)
  put random(50) into line linecount of cd fld “array
  put linecount + 1 into linecount
 end repeat
end mouseup

The user can then access any point in the array with the button “get data”- it’s sript is as follows:

--7
on mouseup

 ask “what point”
 put line it of cd fld “array” 
end mouseup

The next card will do basically the same thing except it will use the XCMD’s Putdata and Getdata.

The script of cd btn “new way” is:

--8

on mouseup
 put 1 into linecount
  repeat(50)
   putdata linecount,random(50)
   put 1 + linecount into linecount
 end repeat
end mouseup

The script of card button “get data” is :

--9

on mouseup
 ask “what point”
 put the result into pointvar
 put getdata pointvar
end mouseup

The final card will draw a graph of points - the horizontal coordinates will represent the array index and the vertical will represent the array index value.

It demonstrates the use of the databuff XCMD in that the graph will be continuously updated until the user presses the mouse button.

Well, that’s it for my first attempt at sharing the few things I’ve learned about Macintosh programming. If there are questions my address is

915 N. Bolton Ave

Indpls., In. 46219.

 

Community Search:
MacTech Search:

Software Updates via MacUpdate

Espresso 5.1 - Powerful HTML, XML, CSS,...
Note from the developer: For the new Espresso, we changed our versioning and licensing approach with more consistent pricing and a simpler development timeline: "X+1". Each new update would increase... Read more
MacFamilyTree 8.2.7 - Create and explore...
MacFamilyTree gives genealogy a facelift: modern, interactive, convenient and fast. Explore your family tree and your family history in a way generations of chroniclers before you would have loved.... Read more
VueScan 9.6.04 - Scanner software with a...
VueScan is a scanning program that works with most high-quality flatbed and film scanners to produce scans that have excellent color fidelity and color balance. VueScan is easy to use, and has... Read more
WhatsApp 0.2.8000 - Desktop client for W...
WhatsApp is the desktop client for WhatsApp Messenger, a cross-platform mobile messaging app which allows you to exchange messages without having to pay for SMS. WhatsApp Messenger is available for... Read more
Vivaldi 1.13.1008.40 - An advanced brows...
Vivaldi is a browser for our friends. In 1994, two programmers started working on a web browser. Our idea was to make a really fast browser, capable of running on limited hardware, keeping in mind... Read more
TotalFinder 1.10.7 - Adds tabs, hotkeys,...
TotalFinder is a universally acclaimed navigational companion for your Mac. Enhance your Mac's Finder with features so smart and convenient, you won't believe you ever lived without them. Features... Read more
Box Sync 4.0.7886 - Online synchronizati...
Box Sync gives you a hard-drive in the Cloud for online storage. Note: You must first sign up to use Box. What if the files you need are on your laptop -- but you're on the road with your iPhone? No... Read more
iExplorer 4.1.14 - View and transfer fil...
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
WhatRoute 2.1.1 - Geographically trace o...
WhatRoute is designed to find the names of all the routers an IP packet passes through on its way from your Mac to a destination host. It also measures the round-trip time from your Mac to the router... Read more
TunnelBear 3.5.1 - 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

Latest Forum Discussions

See All

Everything about Hero Academy 2 - The co...
It's fair to say we've spent a good deal of time on Hero Academy 2. So much so, that we think we're probably in a really good place to give you some advice about how to get the most out of the game. And in this guide, that's exactly what you're... | Read more »
Everything about Hero Academy 2: Part 3...
In the third part of our Hero Academy 2 guide we're going to take a look at the different modes you can play in the game. We'll explain what you need to do in each of them, and tell you why it's important that you do. [Read more] | Read more »
Everything about Hero Academy 2: Part 2...
In this second part of our guide to Hero Academy 2, we're going to have a look at the different card types that you're going to be using in the game. We'll split them up into different sections too, to make sure you're getting the most information... | Read more »
Everything about Hero Academy 2: Part 1...
So you've started playing Hero Academy 2, and you're feeling a little bit lost. Don't worry, we've got your back. So we've come up with a series of guides that are going to help you get to grips with everything that's going on in the game. [Read... | Read more »
What mobile gaming can learn from the Ni...
While Nintendo might not have had things all its own way since it began developing for mobile, one thing it has got right is the release of the Switch. After the disappointment of the WiiU, which I still can't really explain, the Switch felt a... | Read more »
Programmer of Sonic The Hedgehog launche...
Japanese programmer Yuji Naka is best known for leading the team that created the original Sonic The Hedgehog. He’s moved on from the speedy blue hero since then, launching his own company based in Tokyo – Prope Games. Legend of Coin is the... | Read more »
Why doesn't mobile gaming have its...
The Overwatch League is a pretty big deal. It's an attempt to really push eSports into the mainstream, by turning them into, well, regular sports. But slightly less sweaty. It's a lavish affair with teams from all around the world, and more... | Read more »
Give Webzen’s new billiard game PoolTime...
Best known for producing hugely popular MMO titles, South Korean publisher Webzen is now taking aim at a different genre altogether. PoolTime is a realistic eight ball pool simulator, allowing you to compete in real-time matches against players... | Read more »
Let Them Come Guide - How to survive aga...
Let Them Come is all about making it as far as possible against overwhelming odds. Check out some of these tips to help you last a little longer in your unwinnable fight: [Read more] | Read more »
All the best games on sale for iPhone an...
Happy last day of the week. I hope you've been having a good one. I have. I saw ten doggos today. So because I'm in a good mood, I thought I'd round up all of the best games that are currently on sale on the App Store. [Read more] | Read more »

Price Scanner via MacPrices.net

9.7-inch 2017 WiFi iPads on sale starting at...
B&H Photo has 9.7″ 2017 WiFi #Apple #iPads on sale for $30 off MSRP for a limited time. Shipping is free, and pay sales tax in NY & NJ only: – 32GB iPad WiFi: $299, $30 off – 128GB iPad WiFi... Read more
Wednesday deal: 13″ MacBook Pros for $100-$15...
B&H Photo has 13″ #Apple #MacBook Pros on sale for up to $100-$150 off MSRP. Shipping is free, and B&H charges sales tax for NY & NJ residents only: – 13-inch 2.3GHz/128GB Space Gray... Read more
Apple now offering Certified Refurbished 2017...
Apple has Certified Refurbished 9.7″ WiFi iPads available for $50-$80 off the cost of new models. An Apple one-year warranty is included with each iPad, and shipping is free: – 9″ 32GB WiFi iPad: $... Read more
10″ iPad Pros on sale for $50-$75 off MSRP, n...
B&H Photo has 10″ and #Apple #iPad Pros on sale for up to $75 off MSRP. Shipping is free, and B&H charges sales tax in NY & NJ only. Note that some sale prices are restricted to certain... Read more
Apple refurbished Mac minis available startin...
Apple has restocked Certified Refurbished Mac minis 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
Amazon offers Silver 13″ Apple MacBook Pros f...
Amazon has new Silver 2017 13″ #Apple #MacBook Pros on sale today for up to $150 off MSRP, each including free shipping: – 13″ 2.3GHz/128GB Silver MacBook Pro (MPXR2LL/A): $1199.99 $100 off MSRP – 13... Read more
Sale: 12″ 1.3GHz MacBooks on sale for $1499,...
B&H Photo has Space Gray and Rose Gold 12″ 1.3GHz #Apple MacBooks on sale for $100 off MSRP. Shipping is free, and B&H charges sales tax for NY & NJ residents only: – 12″ 1.3GHz Space... Read more
Apple offers Certified Refurbished 2017 iMacs...
Apple has a full line of Certified Refurbished iMacs available for up to $350 off original MSRP. Apple’s one-year warranty is standard, and shipping is free. The following models are available: – 27... Read more
13″ MacBook Airs on sale for $120-$100 off MS...
B&H Photo has 2017 13″ 128GB MacBook Airs on sale for $120 off MSRP. Shipping is free, and B&H charges sales tax for NY & NJ residents only: – 13″ 1.8GHz/128GB MacBook Air (MQD32LL/A): $... Read more
15″ Touch Bar MacBook Pros on sale for up to...
Adorama has Space Gray 15″ MacBook Pros on sale for $200 off MSRP. Shipping is free, and Adorama charges sales tax in NJ and NY only: – 15″ 2.8GHz MacBook Pro Space Gray (MPTR2LL/A): $2199, $200 off... Read more

Jobs Board

Commerce Engineer, *Apple* Media Products -...
# Commerce Engineer, Apple Media Products Job Number: 113161479 Santa Clara Valley, California, United States Posted: 01-Nov-2017 Weekly Hours: 40.00 **Job Summary** Read more
*Apple* Retail - Multiple Positions - Apple,...
Job Description:SalesSpecialist - Retail Customer Service and SalesTransform Apple Store visitors into loyal Apple customers. When customers enter the store, Read more
Site Reliability Engineer, *Apple* Pay - Ap...
# Site Reliability Engineer, Apple Pay Job Number: 113356036 Santa Clara Valley, California, United States Posted: 12-Jan-2018 Weekly Hours: 40.00 **Job Summary** Read more
UI Tools and Automation Engineer, *Apple* M...
# UI Tools and Automation Engineer, Apple Media Products Job Number: 86351939 Santa Clara Valley, California, United States Posted: 11-Jan-2018 Weekly Hours: 40.00 Read more
*Apple* Retail - Multiple Positions - Apple,...
Job Description: Sales Specialist - Retail Customer Service and Sales Transform Apple Store visitors into loyal Apple customers. When customers enter the store, Read more
All contents are Copyright 1984-2011 by Xplain Corporation. All rights reserved. Theme designed by Icreon.