TweetFollow Us on Twitter

BBEdit Plug-In Programming

Volume Number: 14 (1998)
Issue Number: 6
Column Tag: Tools Of The Trade

BBEdit Plug-In Programming

by Steve Sheets

Creating Extensions for Bare Bones Software's BBEdit Text Editor

The Right Tool for the Right Job

As a well known starship engineer would say, "always use the right tool for the right job". It's true for dilithium chambers and it is true for Macintosh development. When working with resources, ResEdit and Resorcerer are the best tools available. Hypercard and Macromedia Director have been the favorites of two generations of multimedia authors. Part of what makes these tools so powerful is the potential to customize them through ResEdit templates, Resorcerer "apprentice" plug-ins, Hypercard XCMDs, and Director Xtras. This plug-in extendibility let's you customize and focus their features to turn a great tool into the perfect tool for your job. The text editor BBEdit is one of the best at what it does and its basic functions can also be extended with third party BBEdit extensions. This article will describe how to create your own BBEdit extensions so you can apply the power of BBEdit to the unique challenges in your projects.

The Right Tool for the Right Editing

Bare Bones Software's BBEdit is a powerful text editor. A text editor is not a word processor (where the output is intended to be a printed page). BBEdit is just for text files -- the source file format for almost all compilers and interpreters. Many developers prefer using BBEdit or another third party editor instead of the text tool built into standard development environments (like Apple's MPW, Metrowerk's CodeWarrior, Symantec C++ or Symantec's Visual Cafe). This preference has become wide spread enough that most IDEs now have explicit support for external editors. BBEdit is also a popular tool for web page design (text being the source format of all HTML web pages) or any other job that requires text manipulation.

While the built in features of BBEdit are impressive to begin with, BBEdit has an additional method to add functions -- the extensions API. Each plug-in is a code resource, of type 'BBXT', that is loaded in, executed and unloaded when the user selects the corresponding item in the extension menu. Bare Bones Software uses the API themselves and BBEdit ships with a couple dozen plug-ins.

When executed, the plug-ins can do any number of text manipulation tasks. The plug-in can even have limited user interaction, as in the case of modal dialogs or Standard File dialogs. However, the plug-in does not have access to any global memory nor to any settings of the BBEdit application itself.

A plug-in also has access to a large list of callback services. These services are functions that Bare Bones Software provides to do common tasks such as accessing or changing the currently selected text, creating or opening new windows, and creating message, progress or error windows. Additional functions make it easy to handle undo, AppleEvents and grep type commands.

Beyond these user interface and single use limitations, BBEdit plug-ins can do almost anything with or to a text file. A plug-in can be a simple tool to automate some changes in the selected text, or it can be a sophisticated compiler (C++, Java, or Resource) that processes a text file into another format. The only limit is the developers imagination.

This article will explain the format of a BBEdit plug-in (resource and file). The extension interface will be described, as will the plug-in control flags resource. Several, but not all, of the callback services will be described. Two examples of BBEdit plug-ins are included for you to learn from. Bare Bones Software provides a complete SDK with all commercial versions of their software and you can download this kit from the BareBones web site at http://www.barebones.com/. The SDK includes a manual explaining all the calls, as well as the required interface files.

Extension Resource and File

A BBEdit plug-in is an old fashion code resource, like an desk accessory, FKEY or menu proc. The code resource can be a traditional 68K resource or an accelerated PowerPC code resource. The code resource can even be a 'fat' code resource that contains both. For most developers, a 68K version of their plug-in is best since it can run on both platforms. You only need to create a PowerPC version to take advantage of the increased speed of native code.

Any number of plug-ins can exist inside a single BBEdit plug-in file. This file must reside inside the "BBEdit Plug-Ins" folder, which is located beside the BBEdit Application. The file can also reside in a sub folder in the BBEdit Plug-ins folder, which would cause the plug-in to appear inside a submenu in the application. This is a good way to group plug-ins. This nesting only goes down one level. The BBEdit file must be of file type 'BBXT'. It's creator type can be anything (allowing custom icon bundles) but if the type is 'R*ch', then the file will have the generic BBEdit plug-in icon. BBEdit will work correctly if aliases to the actual plug-in file are placed in the plug-ins folder.

A BBEdit plug-in is a code resource of type 'BBXT'. The ID can be any number, while the name should describe the plug-in. BBEdit will use this name when placing the plug-in in the extension menu. BBEdit opens the plug-in file as a resource file when the call is executing, so the plug-in can access other resources in the file.

Along with the code resource, the plug-in file can contain an optional "plug-in control flags" resource. This resource is of type 'BBXF' and has the same ID as the plug-in it is associated with. The resource contains 4 bytes of data, each bit signifying a specific flag. The easiest way to create and modify this resource is with the "BBXF" resource template provided by Bare Bones Software in their SDK. A Rez resource header would have been nice, but is not currently available. The flags resource provides the BBEdit application with information about what the plug-in does, and when it can be invoked. These flags will be explained in more detail below. Technically, this flag resource is not required for an old style plug-in, but is necessary if you wish to signify that your plug-in is a new style one.

Calling Conventions

The original or old calling convention had a pascal style function as the main entry point. The function was passed a callback pointer, and the window pointer of the top most window. BBEdit version 3.5 and later support a newer calling convention. While the older convention remains, it has all but been replaced by the new format. The new format provides more information, as well as supporting Apple Events. The new style does require the BBEdit application to run under System 7.0, but this is no longer very limiting. The new style calling convention has a main entry point defined as:

pascal OSErr main(ExternalCallbackBlock *callbacks, 
WindowPtr w, long flags, AppleEvent *event, AppleEvent *reply);

The routine returns an OSErr result indicating the errors, if any. If the plug-in runs successfully, this should be set to noErr (0). The callback parameter points to a structure used by the BBEdit application and the BBEdit callback routines. While the plug-in has access to the structure directly, the structure may change in future revisions of the application. Therefore, it is strongly recommend you only access the callback using the callback services routines or macros.

The w parameter is the WindowPtr of the top most window, while the flags parameter provides information about the applications current state. The flag parameter is a combination of logically ORed flags, each flag having an associated mask defined for it. The xfWindowOpen flag will be set if the front window is a edit or text window. If the xfWindowChangable flag is set, the text in the front window can be modified. If the xfHasSelection window is set, then the front window has some text selected. This text can be accessed directly from the callback services routines. The xfUseDefault flag allows the plug-in to invoke a non-user version of the plug-in. It will only be set if the correct plug-in control flag is set, and the user holds down the option key when invoking the plug-in. In this case, the plug-in should not place up a dialog or provide any other user interface. If settings are required, use the default setting of the plug-in (what ever you want to make them). The xfIsBBEditLite and the xfIsBBEditDemo flags indicate if the BBEdit application is a Lite version (non-commercial) or a demo version. The last two flags, xfWindowHasMailer and xfWindowHasActiveMailer relate to the unsuccessful Apple PowerTalk Mail system. If the xfWindowHasMailer flag is set, then the front window contains an PowerTalk mailer. In that case, if the xfWindowHasActiveMailer flag is also set, then the mailer is currently active.

If the plug-in is invoked from an AppleEvent (or OSA Script), then the event and reply parameters are the standard AE Event Handler parameters. If the AE parameters are not NIL (for example, if an Apple Event was used to invoke the plug-in), then the plug-in does not require user interaction (no modal dialog). The script may be invoked by someone across the network. To allow the plug-in to be scriptable, a standard 'aete' resource with the same ID as the plug-in is needed in the plug-in file. This 'aete' resource can only support one AE suite, but can have any number of events as you want. Bare Bones Software suggests using their BBEdit extension suite.

Extension Control Flags

The plug-in control flags provides information to the BBEdit application about your plug-in. This information allows BBEdit to determine whether to enable or disable your extensions menu item depending on the state of the application. For example, if the plug-in requires PowerPC flag is set, then the plug-in will be disabled when it is run on a 68K Macintosh.

The first three flags deal with handling an undoable plug-in. Bare Bones Software provides several callback services that make undoing most text editing plug-ins trivial. To signify that the extension know about undoing it's actions, the undo-savvy flag needs to be set. If the Can Be Undone flag is also set, then BBEdit expects your plug-in to use the callback routines to handle this. Otherwise, if this flag is not set, when the user selects the menu item, an "This action can not be undone -- do you wish to continue?" alert will appear when the menu item is selected. The plug-in is only called if the user then clicks "Yes". To disable this alert, set the "Can't Undo" alert flag.

The next five flags control whether or not the plug-in is active in the menu. The Requires Non-Empty Window flag indicates the plug-in is disabled if there is no top most window, or if the top most window has no text in it. The Requires Changeable Window indicates the plug-in is disabled if there is no top most window, or if the window is read only. The Requires Edit Window requires exactly that before the plug-in is enabled. The Requires Selection flag indicates the top most window is there, and some text is selected. The Requires PowerPC flag was already explained.

The last four flags are miscellaneous ones. The Support New Interface flag tells the BBEdit application if the plug-in support the new or old conventions. For this article, I strongly recommend setting this flag, and only using the new style. The Use Option Key for Default flag means the user can hold the option key down to invoke the plug-in with the default setting. This flag is only supported on the new style interface. The Place on "Internet" Menu flag indicates where the plug-in wishes to be placed in the menubar. Assuming the user has an Internet menu and this flag is set, the plug-in is place on the Internet menu. The Is a Tool flag indicates the BBEdit plug-in is a BBEdit Tool

BBEdit Tools are beyond the scope of this article. Tools are special plug-ins that function very similarly to old fashion desk accessories. They provide dragable, resizeable, floating windows that are drawn by the plug-in, and can handle mouse down events in the window. BBEdit Tools also support Drag and Drop. Finally, they have access to global memory and are persistent. The Bare Bones Software's SDK provides more information, including examples, of BBEdit Tools.

Callback Services

BBEdit callback services are an essential part of the API. If the callback services routines did not exist, then the BBEdit plug-ins would not be more than application specific FKEYs. Using callbacks gives a plug-in complete control of the text in the application. All callback routines begin with 'bbxt', making them easy to identify in your code. These routines use the Pascal calling conventions and pass the main function's callback parameter as their first parameter. For example, you pass the bbxtGetCallbackVersion routine the single callback parameter, and it returns the version number of the BBEdit API. Using this, you can insure which callback routine is available for a given version of the application (check the header and documentation).

Several routines use either windowptrs as parameters or pass them as results. These are standard Macintosh windowptrs. The plug-in itself receives the windowptr of the top most active window. If the window's Kind field is of type userKind, then the window is a standard BBEdit text window. Be aware that there are a number of other window kinds. Use the callback routines, or the control flags, to make sure your plug-in is only invoked when the correct window is on top.

The first set of callback routines duplicate the standard feature of the Edit window. Since an plug-in can both pass data into and take data out of the clipboard, this is a good method to modify the text in a window. These routines include bbxtCopy, bbxtPaste, and bbxtDelete. Other text editing routines include bbxtGetSelection and bbxtSetSelection which can be used to get and set the selection of the text in a window. The bbxtGetWindowContent call can used to directly access the text in a window, while bbxtSetWindowContent can be used to change it. If you use the bbxtGetWindowContent call to examine the text handle, and then change the content of that text handle, then you should call the bbxtContentsChanged routine to inform the application of the changes. The bbxtInsert call can be used to place text into the current selection space of a window. Between these routines, you can control of changes to the text in a window.

The routines bbxtGetLastLine, bbxtGetLineNumber, bbxtGetLinePos, bbxtGetLineStart, and bbxtGetLineEnd can be used to find the number of lines in a window, the line number of a specific character, the offset of a specific line and the offset of a given character from the beginning or end of it's line. These functions can be useful in formatting lines of text.

Bare Bones Software also provides several file io routines. Given the file name, volume id and directory id, bbxtGetFileText will load that file into a given handle. The bbxtOpenFile call loads a text file into a standard edit window, and returns the windowptr of that window. The bbxtGetFolder call displays a Standard File dialog for choosing a folder (I wish the Mac OS would provide that call). The bbxtOpenSeveral call provides a Standard File dialog for multiple selections of files (another one Apple could do to duplicate). The bbxtNewDocument, bbxtOpenDocument and bbxtSave calls invoke the same actions as if the user had selected the items from the File menu. Many of these calls return the WindowPtr of the created window, which can be then used to view or modify the text. You may also want information about a given window/file. The bbxtGetDocInfo call can return the name, volume id and directory id of a window/file, while bbxtGetModDate returns the last modification date of the file/window. For those who wish to use CodeWarrior or Think project files, bbxtGetProjectList will parse these files for you, while bbxtProjectTextList will generate a textual listing of the project document's content. Finally, bbxtOpenFileByName will open the specified file, using the same search logic as the "Open Selection Command".

Several user interface calls are available for plug-in developers. The simplest is bbxtReportOSError which puts up an alert box with the proper OS Error message, based on the OSErr parameter. Dialog management can be handled by bbxtStandardFilter (which can be used as a filter proc for modal dialogs), bbxtFrameDialogItem (which draws a rectangle around the dialog item), and bbxtCenterDialog (which centers the dialog). For those who need to place a Progress Bar window on the screen, bbxtStartProgress, bbxtDoProgress and bbxtDoneProgress provide exactly that. The bbxttConfirmSave provides the standard "Do you want to save?" Alert to the plug-in. BareBones Software provided this long before Apple thought to add it as a standard call to System 8.0. Finally, bbxtCreateResults will provide an error browser similar to a compiler error window. This is the perfect feedback for errors if your extension is come sort of compiler

The BBEdit API provides a way for plug-ins to store and retrieve plug-in-specific preference data. Using bbxtGetPreference and bbxtSetPreference, a plug-in stores its own persistent preference data in the BBEdit preferences file.

As mentioned previously, BBEdit provides a built-in Undo feature for plug-ins. If you are going to change a range of text in the current window, BBEdit keeps track of the old and new states of the text. First, call the bbxtPresetUndo or bbxtPrepareUndo routines before the changes start, and then call the bbxtSetUndo or bbxtCommitUndo routines after the changes are done. BBEdit will then handle undoing and/or redoing the text. One set of routines is based on the current selection of text, while the other set is based on a given selection range.

Other Apple Event, AppleScript and Process manager calls are available. The bbxtRunScript provides a simple way to execute a given AppleScript file. The bbxtSendAppleEvent call allows you to send the AE, and BBEdit application will handle the idling phase of the transaction. Simpler Apple Events can be generated by bbxtSendOpenDoc and bbxtOpenWithFinder which cause an application or the Finder to open a document. The bbxtFindApplication call uses the Desktop manager to locate an application, while the bbxtLaunchApplication call can be used to launch it. The bbxtApplicationRunning call can tell you if that application is already running.

Finally, several routines provide pattern searching and Grep feature for the plug-in. The bbxtFindPattern call does exactly that; it searches some text for the given text pattern. A host of Grep calls (bbxtPrepareGrep, bbxtDoneGrep, bbxtReportGrepError, bbxtGrepSearch, bbxtGrepRepleace) provide direct control of all the features of a standard Grep search and replace.

The list of callback routines evolved over time, and it shows. These are routines that real plug-in developers needed, requested and received. Bare Bones Software has been successful with marketing their product as web designer tool based on how well the HTML plug-ins are designed. They could not have done this, unless the SDK that comes with BBEdit was also a success.

Examples

The first example provided here demonstrates a good way to display about or help information for your plug-in. After a text resource is loaded and detached, the bbxtNewDocument call creates an edit window. Then the bbxtSetWindowContents call sets the window content to a given text handle. This will place a simple window, containing the text, for the user to view. ResEdit was used to create the required resource with the given flags set. Notice the EnterCodeResource and the ExitCodeResource calls, which are required for code resource.

Listing 1: About MacTech Examples.c

Main
//      About MacTech Examples.c
//      by Steve Sheets for MacTech Magazine
//      Sample of BBEdit extension
//      
//      Requires 'BBXF' Resource 128 with Support New Interface, 
//      and Undo-Savy flags set.
//      Also requires 'TEXT' Resource 128 which contains
//      "MacTech BBEdit Examples by Steve Sheets".

//   Required includes

#include <ExternalInterface.h>
#include <A4Stuff.h>

//   Main entry point of extension

pascal OSErr main(ExternalCallbackBlock *callbacks, 
                  WindowPtr w, 
                  long flags, 
                  AppleEvent *event, 
                  AppleEvent *reply)
{
   OSErr a_result = noErr;
   WindowPtr a_window_ptr;
   Handle a_text_hdl;
   
//   Start Call

   EnterCodeResource();
   
//   Load Text

   a_text_hdl = GetResource('TEXT', 128);
   
//   If have Text, detatch it

   if (a_text_hdl!=NULL) {
   
//   Detatch Text

      DetachResource(a_text_hdl);
   
//   Create new window

      a_window_ptr = bbxtNewDocument(callbacks);
   
//   If Window correct
   
      if (a_window_ptr!=NULL) {

//   Set Content (must not dispose handle then!)

      bbxtSetWindowContents(callbacks, a_window_ptr, a_text_hdl);
      }
      
//   Dispose Text (if can not create new Doc)

      else
         DisposeHandle(a_text_hdl);
   }
   
//   Finish call

   ExitCodeResource();
   
   return a_result;
}

The second example demonstrates how to examine, and then modify the selected text inside an editable text window. The routine copies the selected text using bbxtCopy, then the copy is modified. Modification for this extension consists of stepping through each character, shifting it to upper or lower case. The first shift decides whether or not to change the remaining text up or down. Then the bbxtPaste changes the text in the current selected window. Notice the bbxtPresetUndo and bbxtSetUndo calls are used to allow this plug-in to be undoable.

Listing 2: Shift Uppercase/Lowercase.c

Main
//   Shift Uppercase/Lowercase.c
//   by Steve Sheets for MacTech Magazine
//   Sample of BBEdit extension
//      
//   Requires 'BBXF' Resource 128 with Support New Interface,
//   Undo-Savy, Requires Changeable Window, Can Be Undone,
//   and Requires Selection flags set.

//   Required includes

#include <ExternalInterface.h>
#include <A4Stuff.h>
#include <Sound.h>

//   Main entry point of extension

pascal OSErr main(ExternalCallbackBlock *callbacks, 
                  WindowPtr w, 
                  long flags, 
                  AppleEvent *event, 
                  AppleEvent *reply)
{
   OSErr a_result = noErr;
   Handle a_text_hdl;
   long a_length;
   char a_char;
   long a_count;
   short a_status = 0;   
// 0 - Case Unknown, 1 - shift to Upper, 2 - shirt to Lower
   
//   Start Call

   EnterCodeResource();
   
//   Copy Selection

   a_text_hdl = bbxtCopy(callbacks);   
   
//   Check it is not null

   if (a_text_hdl!=NULL) {
      a_length = GetHandleSize(a_text_hdl);
      
      if (a_length>0) {

//   Set Undoable

         if (bbxtPresetUndo(callbacks)) {
         
//   Set status to 0 (unknown)
//      1 indicates to shift lower case to upper
//      2 indicates to shift upper case to lower

            a_status = 0;

//   Loop through all the text, char at time
            HLock(a_text_hdl);
         
            for (a_count = 0; a_count<a_length; a_count++) {
               a_char = *(*a_text_hdl+a_count);
            
//   If status is not set

               if (a_status==0) {

//   If char is lower case, set status to Upper

                  if ((a_char>='a') && (a_char<='z'))
                     a_status = 1;

//   If char is upper case, set status to Lower

                  else if ((a_char>='A') && (a_char<='Z'))
                     a_status = 2;
               }
            
//   If char is lower & state is Upper, shift it

               if (a_status==1) {
                  if ((a_char>='a') && (a_char<='z'))
                     *(*a_text_hdl+a_count) = a_char - 'a' + 'A';
               }

//   If char is upper & state is lower, shift it

               else if (a_status==2) {
                  if ((a_char>='A') && (a_char<='Z'))
                     *(*a_text_hdl+a_count) = a_char - 'A' + 'a';
               }
            }
         
            HUnlock(a_text_hdl);

//   Return results

            bbxtPaste(callbacks, a_text_hdl);
         
//   Set Undoable

            bbxtSetUndo(callbacks);
         }
      }
      
//   Dispose Text

      DisposeHandle(a_text_hdl);
   }
   
//   Finish call

   ExitCodeResource();
   
   return a_result;
}

The Last Word

While BBEdit is a tremendously powerful and feature rich tool, perhaps it's finest selling point is the insight and humility of the Bare Bones team in realizing that they couldn't imagine or provide all the features every user would require -- so they provided us with the option to add our own features. They've given us the chance to make BBEdit the perfect tool for each of us. Hopefully, this brief survey of the plug-in architecture for BBEdit will help you to begin the challenge of tuning the power of BBEdit to the demands of your own work.


Steve Sheets has been happily programming the Macintosh since 1983, which makes him older than he wishes, but not as young as he acts. A native Californian, his wanderings have led him to northern Virginia. For those interested, his non-computer interests involve his family (wife and two daughters), the Society for Creative Anachronism (medieval reenactment) and the martial arts (Fencing, Tai Chi). Steve is currently an independent developer, taking on any and all projects and can be reached at MageSteve@aol.com.

 

Community Search:
MacTech Search:

Software Updates via MacUpdate

OpenOffice 4.1.7 - Free and open-source...
OpenOffice.org is both an Open Source product and a project. The product is a multi-platform office productivity suite. It includes the key desktop applications, such as a word processor, spreadsheet... Read more
Backup and Sync 3.46 - File backup and s...
Backup and Sync (was Google Drive) is a place where you can create, share, collaborate, and keep all of your stuff. Whether you're working with a friend on a joint research project, planning a... Read more
iClock 5.5 - Customizable menu bar clock...
iClock replaces the old Apple's default menu bar clock with more features, customization and increases your productivity. Features: Have your Apple or Google calendar instantly available from the... Read more
Garmin Express 6.18.0.0 - Manage your Ga...
Garmin Express is your essential tool for managing your Garmin devices. Update maps, golf courses and device software. You can even register your device. Update maps Update software Register your... Read more
MarsEdit 4.3.5 - Quick and convenient bl...
MarsEdit is a blog editor for OS X that makes editing your blog like writing email, with spell-checking, drafts, multiple windows, and even AppleScript support. It works with with most blog services... Read more
Xcode 11.0 - Integrated development envi...
Xcode includes everything developers need to create great applications for Mac, iPhone, iPad, and Apple Watch. Xcode provides developers a unified workflow for user interface design, coding, testing... Read more
DaisyDisk 4.8 - $9.99
DaisyDisk allows you to visualize your disk usage and free up disk space by quickly finding and deleting big unused files. The program scans your disk and displays its content as a sector diagram... Read more
VMware Fusion 11.5.0 - Run Windows apps...
VMware Fusion and Fusion Pro - virtualization software for running Windows, Linux, and other systems on a Mac without rebooting. The latest version includes full support for Windows 10, macOS Mojave... Read more
Apple Configurator 2.10 - Configure and...
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
Spotify 1.1.15.448. - Stream music, crea...
Spotify is a streaming music service that gives you on-demand access to millions of songs. Whether you like driving rock, silky R&B, or grandiose classical music, Spotify's massive catalogue puts... Read more

Latest Forum Discussions

See All

Marvel Strike Force is adding Agent Coul...
Marvel Strike Force, the popular squad-based RPG, is set to receive a bunch of new content over the next few weeks. [Read more] | Read more »
Lots of premium games are going free (so...
You may have seen over the past couple weeks a that a bunch of premium games have suddenly become free. This isn’t a mistake, nor is it some last hurrah before Apple Arcade hits, and it’s important to know that these games aren’t actually becoming... | Read more »
Yoozoo Games launches Saint Seiya Awaken...
If you’re into your anime, you’ve probably seen or heard of Saint Seiya. Based on a shonen manga by Masami Kurumada, the series was massively popular in the 1980s – especially in its native Japan. Since then, it’s grown into a franchise of all... | Read more »
Five Nights at Freddy's AR: Special...
Five Nights at Freddy's AR: Special Delivery is a terrifying new nightmare from developer Illumix. Last week, FNAF fans were sent into a frenzy by a short teaser for what we now know to be Special Delivery. Those in the comments were quick to... | Read more »
Rush Rally 3's new live events are...
Last week, Rush Rally 3 got updated with live events, and it’s one of the best things to happen to racing games on mobile. Prior to this update, the game already had multiplayer, but live events are more convenient in the sense that it’s somewhat... | Read more »
Why your free-to-play racer sucks
It’s been this way for a while now, but playing Hot Wheels Infinite Loop really highlights a big issue with free-to-play mobile racing games: They suck. It doesn’t matter if you’re trying going for realism, cart racing, or arcade nonsense, they’re... | Read more »
Steam Link Spotlight - The Banner Saga 3
Steam Link Spotlight is a new feature where we take a look at PC games that play exceptionally well using the Steam Link app. Our last entry talked about Terry Cavanaugh’s incredible Dicey Dungeons. Read about how it’s a great mobile experience... | Read more »
Combo Quest (Games)
Combo Quest 1.0 Device: iOS Universal Category: Games Price: $.99, Version: 1.0 (iTunes) Description: Combo Quest is an epic, time tap role-playing adventure. In this unique masterpiece, you are a knight on a heroic quest to retrieve... | Read more »
Hero Emblems (Games)
Hero Emblems 1.0 Device: iOS Universal Category: Games Price: $2.99, Version: 1.0 (iTunes) Description: ** 25% OFF for a limited time to celebrate the release ** ** Note for iPhone 6 user: If it doesn't run fullscreen on your device... | Read more »
Puzzle Blitz (Games)
Puzzle Blitz 1.0 Device: iOS Universal Category: Games Price: $1.99, Version: 1.0 (iTunes) Description: Puzzle Blitz is a frantic puzzle solving race against the clock! Solve as many puzzles as you can, before time runs out! You have... | Read more »

Price Scanner via MacPrices.net

11″ WiFi iPad Pros on sale today for up to $2...
Amazon has new 2018 Apple 11″ WiFi iPad Pros in stock today and on sale for up to $200 off Apple’s MSRP. These are the same iPad Pros sold by Apple in its retail and online stores. Be sure to select... Read more
Select 12″ iPad Pros on sale for $200 off App...
Amazon has select 2018 Apple 12″ iPad Pros in stock today and on sale for $200 off Apple’s MSRP. These are the same iPad Pros sold by Apple in its retail and online stores. Be sure to select Amazon... Read more
Get one of Apple’s new 2019 iPhone 11 models...
Boost Mobile is offering the new 2019 Apple iPhone 11, iPhone 11 Pro, and 11 Pro Max for $100 off MSRP. Their discount reduces the cost of an iPhone 11 to $599 for the 64GB models, $899 for the 64GB... Read more
13″ 1.4GHz Silver MacBook Pros on sale for $1...
B&H Photo has new 2019 13″ 1.4GHz 4-Core Touch Bar Silver MacBook Pros on sale for $100 off Apple’s MSRP. Overnight shipping is free to many addresses in the US. These are the same MacBook Pros... Read more
4-core and 6-core 2018 Mac minis available at...
Apple has Certified Refurbished 2018 Mac minis available on their online store for $120-$170 off the cost of new models. Each mini comes with a new outer case plus a standard Apple one-year warranty... Read more
$250 prepaid Visa card with any Apple iPhone,...
Xfinity Mobile will include a free $250 prepaid Visa card with the purchase of any new iPhone, new line activation, and transfer of phone number to Xfinity Mobile. Offer is valid through October 27,... Read more
Sprint is offering the 64GB Apple iPhone 11 P...
Sprint has the new 64GB iPhone 11 Pro available for $12.50 per month for new customers with an eligible trade-in in of iPhone 7 or newer. That’s down from their standard monthly lease of $41.67. The... Read more
Final week: Apple’s 2019 Back to School Promo...
Purchase a new Mac using Apple’s Education discount, and take up to $400 off MSRP. All teachers, students, and staff of any educational institution with a .edu email address qualify for the discount... Read more
Save $30 on Apple’s AirPods at these reseller...
Amazon is offering discounts on new 2019 Apple AirPods ranging up to $30 off MSRP as part of their Labor Day sale. Shipping is free: – AirPods with Charging Case: $144.95 $15 off MSRP – AirPods with... Read more
Preorder your Apple Watch Series 5 today at A...
Amazon has Apple Watch Series 5 GPS models available for preorder and on sale today for $15 off Apple’s MSRP. Shipping is free and starts on September 20th: – 40mm Apple Watch Series 5 GPS: $384.99 $... Read more

Jobs Board

Systems Analyst ( *Apple* & Android) (Jo...
Systems Analyst ( Apple & Android) (Job ID: 572513) + 11751 Meadowville Ln, Chester, VA 23836, USA + Full-time Company Description Computer Consultants International, Read more
*Apple* Mobile App Developer - eiWorkflow So...
…eiWorkflow Solutions, LLC is currently looking for a consultant for the following role. Apple Mobile App Developer Tasks the role will be performing: ? Mobile App Read more
Essbase Developer - *Apple* - Theorem, LLC...
Job Summary Apple is seeking an experienced, detail-minded Essbase developer to join our worldwide business development and strategy team. If you are someone who Read more
Student Employment (Blue *Apple* Cafe) Spri...
Student Employment (Blue Apple Cafe) Spring 2019 Penn State University Campus/Location: Penn State Brandywine Campus City: Media, PA Date Announced: 12/20/2018 Date Read more
Best Buy *Apple* Computing Master - Best Bu...
**732093BR** **Job Title:** Best Buy Apple Computing Master **Job Category:** Store Associates **Location Number:** 001441-Beaumont-Store **Job Description:** The Read more
All contents are Copyright 1984-2011 by Xplain Corporation. All rights reserved. Theme designed by Icreon.