TweetFollow Us on Twitter

Fat Bits
Volume Number:2
Issue Number:6
Column Tag:C Workshop

Mouse DA shows off Fat Bits

By Rick Flott, FlottWare, Chandler, AZ

Mouse Position Desk Accessory

Introduction

Setting up screen graphics on the Mac can be a very tedious and time consuming job. Previously I have used "Mouse Position" desk accessories (DA's) to show the position of the mouse, but none of them had the features I wanted. So as a true programmer I wrote one myself. I borrowed some ideas from others and added a few new ones of my own.

First, I must give credit where it is due. I used an old DA called Magnifying Glass to get a "Fat Bits" view of the screen. I really thought the "Fat Bits" was neat, but wanted other information. This Mouse Position DA also has this feature, but uses it in a little different way.

Using the Mouse Position DA

The window created by the Mouse Position DA looks like this -

The top line gives the name of the window the cursor is currently over. The next two lines (labeled 'L' and 'G') give the local and global coordinates of the mouse (both horizontal and vertical portions are given). The bottom graphics give a "Fat Bits" view of the mouse position. There are a few differences between these "Fat Bits" and others you may have seen.

First, the actual position of the cursor is in the center of the screen (not in the top left corner). Second, the gray lines symbolize the real point (as you all know from reading Inside Macintosh - points do not occupy space, they are at the intersection of infinitely thin horizontal and vertical grid lines). The "hot spot" of the cursor is the point immediately below and to the right of the gray "crosshairs" (the tip of the paintbrush of the MacPaint Icon shown in the window). The "crosshairs" do not cover up any pixels, they just split the rectangle surrounding the cursor into four planes.

There are also a few other features in this DA -

• when the mouse is over the desktop, the window name is set to "DeskTop" and no local coordinates are shown (since they don't exist).

• the window name, local, and global coordinates can be "Cut" or "Copied" to the clipboard and "Pasted" into your program.

• when the CapsLock key is down and the coordinates are Cut or Copied, they are appended to the clipboard. This is very handy when setting up rectangles or other complex graphics that require multiple points. Just press CapsLock and start Copying!

• remember - in most applications Cut and Copy only work on the topmost window, hence the Mouse Position window must be in front for these commands to work.

Code Description

As in any desk accessory, 5 routines must be present -

• open (initializes the DA)

• close (stops the DA)

• control (receives commands from system)

• prime

• status

The latter two routines do not need to perform any functions, but they must be present.

The open routine in the Mouse Position DA is the "main()" of this C program. It allocates the DA window from the heap, sets up the font for this window, and draws the "static" portion of the window. It also stores the reference number of the DA in the windowKind field of the window. This is very important since this the only way the Mac knows that this is a "system" window and to pass the proper events to it and not to the running application. In addition to this, the pointer to the window is kept in the device control entry for retrieval later on. Remember, a desk accessory is viewed as 5 separate routines called by the system (unlike an application).

The Close routine does the opposite of the open. It releases the memory used by the window and resets the proper fields in the device control entry record.

The Control routine does all the work. It takes commands from the system (passed to it when the application calls SystemTask, SystemEdit, or SystemClick) and processes them. In this DA, only two commands are processed - accRun (periodic command telling the DA to run) and accEvent (command telling the DA to handle an event). As you will see later on, this DA is set up to run as often as possible so that the mouse position coordinates appear to be updated continuously in the window.

This routine starts off by obtaining the window pointer from the device control entry and parsing the command (CSCode) sent to it. If it is a run command (accRun) then the new mouse position is displayed (by calling the routine dspMousePos). If it is the event command (accEvent) then the doEvent routine is called.

The doEvent routine handles 4 types of events - keyDown, autoKey, updateEvt, and activateEvt. The updateEvt will re-draw the coordinates and the window name. The activateEvt only re-draws the coordinates. The keyDown and autoKey events do most of the work in this code. When either of these events occur, a string is allocated, the coordinates and window name placed into the string (with the Munger ROM call), and moved to the clipboard.

Why did I append perfectly good strings into another string? I'm lazy I guess (plus the string is temporarily allocated off the heap and is released immediately). Notice that if the Cut or Copy keys are capitals (the CapsLock key is down) then the current TEXT contents of the clipboard are placed in the string. This allows the user to "append" coordinates together as previously described.

The dspMousePos routine first gets the current coordinates of the mouse. If the mouse has moved from the last time this routine was called, then processing continues, otherwise this routine just returns. This keeps down the "flicker" of the display, since the window is only updated when the mouse moves. It also keeps the DA from being a CPU hog.

Next, the name of the window the mouse is currently over is retrieved. If it is a different window than the last time, the new name is displayed (by the routine dspWindowTitle).

If the mouse is not on the desktop, then the local coordinates must be recomputed. GetMouse gives the mouse in local coordinates of the active window, not the window the mouse is currently over (which may be inactive or even invisible). Hence, the local coordinates must be recomputed with respect to the window the mouse is over. The rest of this routine is straightforward.

The coordinates are converted to strings, the "h" and "v" characters are appended to them, and these strings are displayed right justified in the window. Next, the four "Fat Bit" rectangles are displayed. Notice that 24 pixels surrounding the cursor in the horizontal direction and 16 pixels surrounding the cursor in the vertical direction are displayed (using CopyBits) in the window.

The dspWindowTitle routine first determines if the window passed to it is the desktop or a real window. If it is a real window, then its name is retrieved. Otherwise the name "DeskTop" is used. It then displays this name center justified in the DA window. It also adds a carriage return ('\n') to the name so that when the name is Cut or Copied to the clipboard, the coordinates will be on the next line (like in the window).

The drawWindow routine just draws the "static" portion of the window. This includes the "G" and "L" characters, the horizontal dividing lines, and the gray "crosshairs".

Consulair's DeskMaker

Now all of you desk accessory veterans are saying - "this stuff will never work, look at all of the global variables this guy has in a DA". Well, an application called DeskMaker by Consulair (included with their Mac C development system) takes alot of the headache out of writing DA's. After talking with Bill Duvall, he informed me that the way DeskMaker works is that it takes all of the global variables (referenced off of A4 by the #Options R=4 line) and makes them part of the DRVR resource (he appends them to the end of the code). This allows them to be global and still be accessed by the code in the DA. Of course, I didn't call him until I had already stayed up all night allocating my globals from the heap and accessing them through a handle. Oh well, it wasn't the first time I threw out code, and it won't be the last.

In addition to this, DeskMaker takes commands from a ".desk" text file to set up the DA's flags and header constants they require. You can turn any flag on or off, set up the other constants (like drvrDelay, drvrMask, etc.), and specify the names of the standard routines (Open, Close, Control, etc.) He has also added another command "Test" which allows you temporarily install the DA in the system menu to try it out. This is nice since you do not have to use Font/DA Mover or any of the other applications/DA's that run DA's from a file.

The "MousePos.Desk" listing shows the commands sent to DeskMaker for this DA. First, the filename, map, and name of the DA is set up. Next, the names of the 5 routines are defined. Finally, the drvrEMask, drvrDelay, and drvrFlags are set up. This DA was given the ID of 22 and the test flag was set to allow debugging. Since most of the C compilers handle DAs differently, this information should allow you to recreate this DA on other compilers.

MousePos.c Lisiting

/*************************************************************

  MousePos.c

     This is a desk accessory that shows three things about the position 
of the mouse cursor -

     - Name of the window the cursor is currently over
     - Local and global coordinates of the cursor
     - "Fat Bits" display of a rectangle around the cursor

      It also allows the user to Copy or Cut the window name and the 
local & global coordinates to the clipboard (CapsLock-Copy or CapsLock-Cut 
will append the coordinates to the clipboard).

 Written by: Rick Flott                           Mac C (Consulair) V 
4.5
*************************************************************/

#Options R=4 L=500 F=8000 Z Q=0 O=200

#include "MacCDefs.h"// Mac ROM data structure def's
#include "Events.h"
#include "Window.h"
#include "Font.h"
#include "TextEdit.h"
#include "Osmisc.h"
#include "Osio.h"
#include "Desk.h"

/*----------------------------------------------------------------
    Global Data
-----------------------------------------------------------------*/

/* ------ Constants ----- */

#define FALSE 0
#define TRUE  0xFF

/* ------- Types ------- */

typedef struct    // 6 char strings for the coord's
 {
 char count;
 char s[6];
 } Str6;

/* ---- Rectangles ---- */

Rect titleRect      = {  0,  0, 10,100},   // Window title rect
     localStrRect   = { 11,  1, 21,  6},
     localHRect     = { 11,  6, 21, 49},     // Local horiz coord's rect
     localVRect     = { 11, 52, 21, 94},   // Local vert  coord's rect
     globalStrRect  = { 21,  1, 31,  6},
     globalHRect    = { 21,  6, 31, 49},   // Global horiz coord's rect
     globalVRect    = { 21, 52, 31, 94},  // Global vert  coord's rect

     fBTopLeftRect  = { 32,  0, 64, 48}, // "Fat Bits" rect's
     fBBotLeftRect  = { 68,  0,100, 48},
     fBTopRightRect = { 32, 52, 64,100},
     fBBotRightRect = { 68, 52,100,100},

     windowRect     = { 50,  5,150,105}; // DA window rect

/* ----- Strings ------ */

char   deskTopTitle[] = {"\pDeskTop"}; // Constant desktop str

Str6   localVStr,  localHStr, // Local coord strings
       globalVStr, globalHStr;// Global coord strings

Str255 windowT;    // Window name string

/* --- Global Var's --- */

WindowPtr oldFrontWindow = 0; // Last front window ptr
Point     oldPt = {0,0};   // Last position of mouse

struct QDVar *getQD();


/*------------------------------------------------------------------------
    main()     (Open routine)

      The Open routine opens the desk accessory window and intializes 
any global data before the desk accessory is used.
-------------------------------------------------------------*/

int main(parameterBlock,DeviceControlEntry)
 CntrlParam       *parameterBlock;
 DeviceControl *DeviceControlEntry;
 {
 WindowPtr windowPtr;
 GrafPtr port;

 if ((windowPtr = DeviceControlEntry->dCtlWindow) == 0)
 {
 GetPort(&port); // Preserve appl window
    // Open DA window
 windowPtr = NewWindow(0, &windowRect,
 "\pMouse Pos",0,
 rDocProc, -1, 1, 0);
 if (windowPtr == 0)
      return(-1);

   SetPort(windowPtr);       // Use this new window

   // Set it as a system window
   ((WindowPeek)windowPtr)->windowKind =     
                                                  DeviceControlEntry->dCtlRefNum;

  // Save DA window ptr
   DeviceControlEntry->dCtlWindow=windowPtr;

   TextFont(monaco);    // Set up DA font
   TextSize(9);

   drawWindow();   // Draw static portion of window

   SetPort(port);   // Restore application window
   }

 return 0;

 } // end main()

/*----------------------------------------------------------------
    Close()

      The Close routine disposes of the desk accessory window and any 
data allocated on the heap.
--------------------------------------------------------------*/

int Close(parameterBlock,DeviceControlEntry)
 CntrlParam       *parameterBlock;
 DeviceControl *DeviceControlEntry;
 {
 WindowPtr windowPtr;
 // Get DA window ptr
 windowPtr = DeviceControlEntry->dCtlWindow; 

 DisposeWindow(windowPtr);   // Release DA window
 DeviceControlEntry->dCtlWindow = 0;

 return 0;

 } // end Close()

/*--------------------------------------------------------------
   Prime()

      This desk accessory does not use a Prime routine.
----------------------------------------------------------------*/

Prime(parameterBlock,DeviceControlEntry)
 CntrlParam       *parameterBlock;
 DeviceControl *DeviceControlEntry;
 {
 }

/*--------------------------------------------------------------
   Status(parameterBlock,DeviceControlEntry)

      This desk accessory does not use a Status routine.
---------------------------------------------------------------*/

Status(parameterBlock,DeviceControlEntry)
 CntrlParam       *parameterBlock;
 DeviceControl *DeviceControlEntry;
 {
 }

/*----------------------------------------------------------------
    Control()

      The Control routine parses the desk accessory command sent from 
the system and routes the data to the proper routine. The commands currently 
used are -

 accRun - Display the mouse position.
 accEvent - Handle Cut and Copy menu commands.

----------------------------------------------------------------*/

Control(parameterBlock,DeviceControlEntry)
 CntrlParam       *parameterBlock;
 DeviceControl *DeviceControlEntry;
 {
 GrafPtr port;
 WindowPtr windowPtr;
 // Get DA window ptr
 windowPtr = DeviceControlEntry->dCtlWindow; 

 GetPort(&port); // Preserve application window
 SetPort(windowPtr);    // Use DA window

 switch (parameterBlock->CSCode) // What cmd was sent?
 {
 case accRun:      // cmd = RUN
 dspMousePos(windowPtr);      // Display new  position
 break;
 case accEvent:   // cmd = HANDLE EVENT
 doEvent(parameterBlock->csp.event,windowPtr);
 break;
 } // end switch

 SetPort(port);   // Restore appl window

  } // end Control()


/*----------------------------------------------------------------
    doEvent()

      This routine parses the event sent from the system. The events 
currently used are -

 keyDown, autoKey  - Cut or Copy the window name &                   
                                                  mouse coord's to clipboard.

 updateEvt         - Redraw the entire desk acc window.

 activateEvt       - Redraw the only the mouse coord's.

--------------------------------------------------------------*/

doEvent(event,windowPtr)
 EventRecord *event;
 WindowPtr    windowPtr;
 {
 Handle strHandle;
 Ptr    strPtr;
 int    scrapOffset;
 long   offset = 0;

 switch (event->what)// Which event occurred?
 {
 case keyDown:
 case autoKey:   // Event = KEY PUSH
 if ((event->modifiers&cmdKey))      // Was it a cmd key?
 {
 strHandle = (Handle)NewHandle(0); // Allocate a string

 // Only allow Cut,Copy
 switch ((char)(event->message))
 {
 case 'C':         // If Shift/CapsLock, copy previous clip
 case 'X':
 offset = GetScrap(strHandle,
  'TEXT',&scrapOffset);
 case 'c':
 case 'x':
 if (offset < 0)   // Was there a scrap error?
 {
 SysBeep(2);// Y - return
 break;
 }

 if (ZeroScrap())// Clear the clipboard
 {
 SysBeep(2);
 break;
 }

        // Grow string to place clipboard stuff in

   SetHandleSize(strHandle,offset+
 (sizeof windowT)+
 (sizeof localHStr)+
 (sizeof localVStr)+
 (sizeof globalHStr)+
 (sizeof globalVStr));

        // Place window name, local, global coord's into string

        offset = Munger(strHandle, offset,0,0, windowT.s,      
 (long) windowT.count);
           offset = Munger(strHandle, offset,0,0, localHStr.s, 
 (long) localHStr.count);
        offset = Munger(strHandle, offset,0,0, localVStr.s,    
 (long) localVStr.count);
        offset = Munger(strHandle, offset,0,0, globalHStr.s,
 (long) globalHStr.count);
        offset = Munger(strHandle,offset,0,0, globalVStr.s,
 (long) globalVStr.count);

        HLock(strHandle); // Lock the string down

        strPtr = (Ptr)*strHandle;
        // Put string into clipboard
        if (PutScrap((long)offset,'TEXT',strPtr))
 SysBeep(20);

   HUnlock(strHandle);  // Unlock the string
   break;

        default: // Beep on other cmd keys
        SysBeep(2);
    break;

 } // end switch

 DisposHandle(strHandle); // Release the string
   } // end if

   return;

 case updateEvt: // Event = UPDATE EVENT
 SetPort(windowPtr);  // Use DA window
 BeginUpdate(windowPtr);
 drawWindow();      // Redraw the window
 // Display  new window title
 dspWindowTitle(oldFrontWindow); 
 EndUpdate(windowPtr);
 return;

 case activateEvt: // Event = ACTIVATE EVENT
 dspMousePos(windowPtr);   // Display  new mouse pos
 return;

    } // end switch
  } // end doEvent()

/*-----------------------------------------------------------
   dspMousePos()

      This routine displays the position of the mouse. It displays the 
following information -

 - Local and global coordinates of the cursor
 - "Fat Bits" display of a rectangle around the cursor

------------------------------------------------------------*/

dspMousePos(windowPtr)
  WindowPtr windowPtr;
  {
  Rect      cursorRect;
  WindowPtr mouseWindow;
  short     windowCode;
  struct    QDVar *myQD;  // Place for copy of QD pointer
  Point     localPt,globalPt;

  myQD = getQD();// Get a copy of QD pointer

  GetMouse(&localPt);// Get the new mouse position
  globalPt = localPt;
  LocalToGlobal(&globalPt); // Convert it to global coord's

  if (!EqualPt(&globalPt,&oldPt))  // Has the mouse moved?
 {
    oldPt = globalPt;// Y - remember where it now is

  // Determine the window the mouse is now in
    windowCode = FindWindow(&globalPt, 
   &mouseWindow);

 // Is mouse in a different window?
    if (oldFrontWindow != mouseWindow)
            // Y - Display  title of new window
      dspWindowTitle(oldFrontWindow = mouseWindow);

    if (mouseWindow) // Is the mouse on the Desktop?
      { // N - get, display  local coord's
 // Get local coord's of window mouse is in
      SetPort(mouseWindow);
      localPt = globalPt;
      GlobalToLocal(&localPt);
           // Convert local coord's to strings
      NumToString(localPt.h,&localHStr);
      NumToString(localPt.v,&localVStr);

      localHStr.s[localHStr.count++]='h';    // Add in 'h' and 'v'
      localVStr.s[localVStr.count++]='v';
      localVStr.s[localVStr.count++]='\n';   // Add CR
      }
    else    // Y - don't display local coord's when on desktop
      localHStr.count=localVStr.count=0;

    SetPort(windowPtr);      // Draw in desk acc window
 // Convert global coord's to strings
    NumToString(globalPt.h,&globalHStr);
    NumToString(globalPt.v,&globalVStr);

    globalHStr.s[globalHStr.count++]='h';    // Add in 'h' and 'v'
    globalVStr.s[globalVStr.count++]='v';
    globalVStr.s[globalVStr.count++]='\n';   // Add CR

 // Display global coord's
    TextBox(globalHStr.s,globalHStr.count,
                      &globalHRect,-1);
    TextBox(globalVStr.s,globalVStr.count,
                      &globalVRect,-1);
 // Display local coord's
    TextBox(localHStr.s,localHStr.count,     
            &localHRect,-1);
    TextBox(localVStr.s,localVStr.count,
            &localVRect,-1);


    SetRect(&cursorRect,  // Set up top left "Fat Bits" rect
                      globalPt.h-12,globalPt.v-8,
           globalPt.h, globalPt.v);
    CopyBits(&myQD->screenBits, // Display  top left "Fat Bits"
                       &windowPtr->portBits,
    &cursorRect,&fBTopLeftRect,srcCopy,0);


    SetRect(&cursorRect,  // Set up bottom left "Fat Bits" rect
                     globalPt.h-12,globalPt.v,
  globalPt.h, globalPt.v+8);
    CopyBits(&myQD->screenBits,    // Display bot left "Fat Bits"
             &windowPtr->portBits,
    &cursorRect,&fBBotLeftRect,srcCopy,0);

    SetRect(&cursorRect,  // Set up bottom right "Fat Bits" rect
                      globalPt.h, globalPt.v,
   globalPt.h+12,globalPt.v+8);
    CopyBits(&myQD->screenBits,  // Display bot right "Fat Bits"
                        &windowPtr->portBits,
 &cursorRect,&fBBotRightRect,srcCopy,0);


    SetRect(&cursorRect,  // Set up top right "Fat Bits" rect
                     globalPt.h, globalPt.v-8,
 globalPt.h+12,globalPt.v);
    CopyBits(&myQD->screenBits,  // Display top right "Fat Bits"
                        &windowPtr->portBits,
 &cursorRect,&fBTopRightRect,srcCopy,0);

    } // endif
  } // end dspMousePos()

/*----------------------------------------------------------------
   dspWindowTitle()

      This routine displays the name of the window passed to it in the 
window title rectangle.

---------------------------------------------------------------*/
dspWindowTitle(windowPtr)
  WindowPtr windowPtr;
  {
  if (windowPtr) // Is the mouse in a real window?
  // Y - display  window's name
    GetWTitle(windowPtr,&windowT);
  else   // N - display the desktop name
    BlockMove(&deskTopTitle[0],&windowT,deskTopTitle[0]+1);

  TextBox(windowT.s,windowT.count,&titleRect,1);
  windowT.s[windowT.count++] = '\n'; // Add CR

 } // end dspWindowTitle()

/*----------------------------------------------------------------------
  drawWindow()

 This routine draws the "static" portion of the window.

  ----------------------------------------------------------------------*/
drawWindow()
  {
  struct    QDVar *myQD;  // Place for copy of QD pointer

  myQD = getQD();  // Get a copy of QD pointer

  MoveTo(globalStrRect.left,// Draw "G"
                   globalStrRect.bottom - 1 );
  DrawChar('G');

 MoveTo(localStrRect.left,  // Draw "L"
                   localStrRect.bottom - 1);
  DrawChar('L');

  MoveTo(0,titleRect.bottom);  // Draw the horiz dividing lines
  Line(titleRect.right,0);
  MoveTo(0,globalStrRect.bottom);
  Line(titleRect.right,0);

  PenSize(4,4);
  PenPat(&myQD->gray);

  MoveTo(fBTopLeftRect.right,    // Draw crosshairs
                   fBTopLeftRect.bottom);
  LineTo(fBTopRightRect.right,fBTopRightRect.bottom);

  MoveTo(fBTopLeftRect.right,
                   fBTopLeftRect.bottom);
  LineTo(fBBotLeftRect.right,fBBotLeftRect.bottom);

  MoveTo(fBTopLeftRect.right,
                   fBTopLeftRect.bottom);
  LineTo(fBTopLeftRect.left,fBTopLeftRect.bottom);

  MoveTo(fBTopLeftRect.right,
                   fBTopLeftRect.bottom);
  LineTo(fBTopLeftRect.right,fBTopLeftRect.top);

  PenNormal();

  } // end drawWindow()

/*--------------------------------------------------------------
      getQD()

      This routine returns the pointer used by Quikdraw to point to its 
global data.

-------------------------------------------------------------*/
 {
  #asm

grafSizeEQU $CA

  MOVE.L0(A5),A0
  SUB.L #grafSize,A0

  #endasm
  } // end getQD()

/*-------------------------------------------------------------
      NumToString()  (Package glue routine)

---------------------------------------------------------------*/

NumToString(theNum,theString)
  long         theNum;
  struct PStr *theString;

  {
  #asm
 MOVE.L D1,A0    ; theString
 MOVE   #0,-(SP) ; NumToString selector
 DC.W   $A9EE    ; PACK7
  #endasm
  }

MousePos.link Listing

/NoAnimate
/Output MousePos
/Type 'DFIL' 'DMOV'

MousePos

$

MousePos.desk Listing

File MousePos

Map MousePos.map

Name "Mouse Position"

Open main

Close close

Control control

Status status

Prime prime

EventMask 362
Delay 0

+ Periodic
+ Control
+ Status

DeskID 22

+ Test
 
AAPL
$112.65
Apple Inc.
+3.24
MSFT
$47.52
Microsoft Corpora
+1.78
GOOG
$511.10
Google Inc.
+6.21

MacTech Search:
Community Search:

Software Updates via MacUpdate

BBEdit 11.0.2 - Powerful text and HTML e...
BBEdit is the leading professional HTML and text editor for the Mac. Specifically crafted in response to the needs of Web authors and software developers, this award-winning product provides a... Read more
ExpanDrive 4.2.1 - Access cloud storage...
ExpanDrive builds cloud storage in every application, acts just like a USB drive plugged into your Mac. With ExpanDrive, you can securely access any remote file server directly from the Finder or... Read more
Adobe After Effects CC 2014 13.2 - Creat...
After Effects CC 2014 is available as part of Adobe Creative Cloud for as little as $19.99/month (or $9.99/month if you're a previous After Effects customer). After Effects CS6 is still available... Read more
Command-C 1.1.7 - Clipboard sharing tool...
Command-C is a revolutionary app which makes easy to share your clipboard between iOS and OS X using your local WiFi network, even if the app is not currently opened. Copy anything (text, pictures,... Read more
Tidy Up 4.0.2 - Find duplicate files and...
Tidy Up is a complete duplicate finder and disk-tidiness utility. With Tidy Up you can search for duplicate files and packages by the owner application, content, type, creator, extension, time... Read more
Typinator 6.3 - Speedy and reliable text...
Typinator turbo-charges your typing productivity. Type a little. Typinator does the rest. We've all faced projects that require repetitive typing tasks. With Typinator, you can store commonly used... Read more
GraphicConverter 9.5 - Graphics editor w...
GraphicConverter is an all-purpose image-editing program that can import 200 different graphic-based formats, edit the image, and export it to any of 80 available file formats. The high-end editing... Read more
Toast Titanium 12.0.1 - The ultimate med...
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
QuickBooks 2015 16.0.2.1422 R3 - Financi...
Save 20% on QuickBooks Pro for Mac today through this special discount link QuickBooks Pro 2013 helps you manage your business easily and efficiently. Organize your finances all in one place, track... Read more
Remotix 3.0.6 - Access all your computer...
Remotix is a fast and powerful application to easily access multiple Macs (and PCs) from your own Mac. Features: Complete Apple Screen Sharing support - including Mac OS X login, clipboard... Read more

Latest Forum Discussions

See All

The Hit List — Simply Powerful Tasks, To...
The Hit List — Simply Powerful Tasks, To-Dos, Projects, & Reminders 2.0 Device: iOS iPhone Category: Productivity Price: $9.99, Version: 2.0 (iTunes) Description: >> LAUNCH SPECIAL: The Hit List 2 for iPhone is ONLY $9.99... | Read more »
Mahjong Journey Review
Mahjong Journey Review By Jennifer Allen on December 18th, 2014 Our Rating: :: STEADY MATCHINGiPad Only App - Designed for the iPad Aimed at the more laid back gamer, Mahjong Journey isn’t for everyone, but those looking for some... | Read more »
Emoji Type - custom keyboard with predic...
Emoji Type - custom keyboard with predictive emojis 0.4.0 Device: iOS iPhone Category: Utilities Price: $.99, Version: 0.4.0 (iTunes) Description: Emoji Type is custom keyboard for iOS 8 that auto suggests emojis as you type. ABOUT... | Read more »
Game of the Year 2014 – 148Apps Staff Pi...
The end of 2014 is almost here, which can only mean one thing. Okay it can mean a lot of things, but in this specific context it means Game of the Year lists! Which is why the 148Apps staff have all picked their favorites from the past year. And why... | Read more »
UponPixels Review
UponPixels Review By Jennifer Allen on December 18th, 2014 Our Rating: :: CREATIVE TYPOGRAPHYUniversal App - Designed for iPhone and iPad Add cool typography and objects to your photos with the easy to use UponPixels.   | Read more »
The Vikings are Coming! CastleStorm’s Ne...
The Vikings are Coming! CastleStorm’s New Update Adds a Survival Mode Posted by Jessica Fisher on December 18th, 2014 [ permalink ] | Read more »
Duet Display (Productivity)
Duet Display 0.3.3 Device: iOS Universal Category: Productivity Price: $9.99, Version: 0.3.3 (iTunes) Description: Duet Display allows you to use your iPad or iPhone as an extra display. Developed by a team of ex-Apple engineers,... | Read more »
Dragon Quest III Review
Dragon Quest III Review By Jennifer Allen on December 18th, 2014 Our Rating: :: DATED BUT NOT WITHOUT MERITUniversal App - Designed for iPhone and iPad A walk down memory lane isn’t foolproof for Dragon Quest III, but it has its... | Read more »
8 KEMCO JRPGs Are Just $0.99 in Celebrat...
8 KEMCO JRPGs Are Just $0.99 in Celebration of the Holidays Posted by Jessica Fisher on December 18th, 2014 [ permalink ] KEMCO has announc | Read more »
Brothers in Arms 3: Sons of War Review
Brothers in Arms 3: Sons of War Review By Jennifer Allen on December 18th, 2014 Our Rating: :: FUN BUT PUSHYUniversal App - Designed for iPhone and iPad Brothers in Arms 3: Sons of War could be great fun, but its plethora of... | Read more »

Price Scanner via MacPrices.net

Save up to $400 on MacBooks with Apple Certif...
The Apple Store has Apple Certified Refurbished 2014 MacBook Pros and MacBook Airs available for up to $400 off the cost of new models. An Apple one-year warranty is included with each model, and... Read more
Save up to $300 on Macs, $30 on iPads with Ap...
Purchase a new Mac or iPad at The Apple Store for Education and take up to $300 off MSRP. All teachers, students, and staff of any educational institution qualify for the discount. Shipping is free,... Read more
iOS and Android OS Targeted by Man-in-the-Mid...
Cloud services security provider Akamai Technologies, Inc. has released, through the company’s Prolexic Security Engineering & Research Team (PLXsert), a new cybersecurity threat advisory. The... Read more
KMI MIDI K-Board Great Gift for Amateur &...
The K-Board is a MIDI Nano keyboard for music creation for iPad, Android, And computers; the easiest way to make music with iPads & Android tablets, and Mac, Windows, or Linux computers. Ultra-... Read more
Amazon offers 15-inch 2.2GHz Retina MacBook P...
 Amazon.com has the 15″ 2.2GHz Retina MacBook Pro on sale for $1699 including free shipping. Their price is $300 off MSRP. Stock is limited, so act now if you’re interested. Read more
Holiday sales continue: MacBook Pros for up t...
 B&H Photo has new MacBook Pros on sale for up to $300 off MSRP as part of their Holiday pricing. Shipping is free, and B&H charges NY sales tax only: - 15″ 2.2GHz Retina MacBook Pro: $1699... Read more
Holiday sale: Mac minis available for up to $...
 B&H Photo has new 2014 Mac minis on sale for up to $80 off MSRP. Shipping is free, and B&H charges NY sales tax only: - 1.4GHz Mac mini: $459 $40 off MSRP - 2.6GHz Mac mini: $629 $70 off... Read more
Google Search App For iOS Gets A Major Makeov...
Google has given iOS users an early Christmas present with a substantial update of it’s not-very-often-upgraded Google Search app. Google Search has been my go-to tool for Web searches since it was... Read more
ShopKeep Apple Pay And Chip Card Reader Avail...
ShopKeep, a cloud-based technology provider to more than 10,000 small business owners to manage retail shops and restaurants with iPads, has released its new Apple Pay and chip card reader. This... Read more
Holiday sale! 27-inch 5K iMac for $2299, save...
 B&H Photo has the 27″ 3.5GHz 5K iMac in stock today and on sale for $2299 including free shipping plus NY sales tax only. Their price is $200 off MSRP, and it’s the lowest price available for... Read more

Jobs Board

*Apple* Retail - Multiple Positions (US) - A...
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
*Apple* Retail - Multiple Positions (US) - A...
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
*Apple* Retail - Multiple Positions (US) - A...
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 (US) - A...
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
Project Manager / Business Analyst, WW *Appl...
…a senior project manager / business analyst to work within our Worldwide Apple Fulfillment Operations and the Business Process Re-engineering team. This role will work Read more
All contents are Copyright 1984-2011 by Xplain Corporation. All rights reserved. Theme designed by Icreon.