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

Latest Forum Discussions

See All

Whitethorn Games combines two completely...
If you have ever gone fishing then you know that it is a lesson in patience, sitting around waiting for a bite that may never come. Well, that's because you have been doing it wrong, since as Whitehorn Games now demonstrates in new release Skate... | Read more »
Call of Duty Warzone is a Waiting Simula...
It's always fun when a splashy multiplayer game comes to mobile because they are few and far between, so I was excited to see the notification about Call of Duty: Warzone Mobile (finally) launching last week and wanted to try it out. As someone who... | Read more »
Albion Online introduces some massive ne...
Sandbox Interactive has announced an upcoming update to its flagship MMORPG Albion Online, containing massive updates to its existing guild Vs guild systems. Someone clearly rewatched the Helms Deep battle in Lord of the Rings and spent the next... | Read more »
Chucklefish announces launch date of the...
Chucklefish, the indie London-based team we probably all know from developing Terraria or their stint publishing Stardew Valley, has revealed the mobile release date for roguelike deck-builder Wildfrost. Developed by Gaziter and Deadpan Games, the... | Read more »
Netmarble opens pre-registration for act...
It has been close to three years since Netmarble announced they would be adapting the smash series Solo Leveling into a video game, and at last, they have announced the opening of pre-orders for Solo Leveling: Arise. [Read more] | Read more »
PUBG Mobile celebrates sixth anniversary...
For the past six years, PUBG Mobile has been one of the most popular shooters you can play in the palm of your hand, and Krafton is celebrating this milestone and many years of ups by teaming up with hit music man JVKE to create a special song for... | Read more »
ASTRA: Knights of Veda refuse to pump th...
In perhaps the most recent example of being incredibly eager, ASTRA: Knights of Veda has dropped its second collaboration with South Korean boyband Seventeen, named so as it consists of exactly thirteen members and a video collaboration with Lee... | Read more »
Collect all your cats and caterpillars a...
If you are growing tired of trying to build a town with your phone by using it as a tiny, ineffectual shover then fear no longer, as Independent Arts Software has announced the upcoming release of Construction Simulator 4, from the critically... | Read more »
Backbone complete its lineup of 2nd Gene...
With all the ports of big AAA games that have been coming to mobile, it is becoming more convenient than ever to own a good controller, and to help with this Backbone has announced the completion of their 2nd generation product lineup with their... | Read more »
Zenless Zone Zero opens entries for its...
miHoYo, aka HoYoverse, has become such a big name in mobile gaming that it's hard to believe that arguably their flagship title, Genshin Impact, is only three and a half years old. Now, they continue the road to the next title in their world, with... | Read more »

Price Scanner via MacPrices.net

B&H has Apple’s 13-inch M2 MacBook Airs o...
B&H Photo has 13″ MacBook Airs with M2 CPUs and 256GB of storage in stock and on sale for up to $150 off Apple’s new MSRP, starting at only $849. Free 1-2 day delivery is available to most US... Read more
M2 Mac minis on sale for $100-$200 off MSRP,...
B&H Photo has Apple’s M2-powered Mac minis back in stock and on sale today for $100-$200 off MSRP. Free 1-2 day shipping is available for most US addresses: – Mac mini M2/256GB SSD: $499, save $... Read more
Mac Studios with M2 Max and M2 Ultra CPUs on...
B&H Photo has standard-configuration Mac Studios with Apple’s M2 Max & Ultra CPUs in stock today and on Easter sale for $200 off MSRP. Their prices are the lowest available for these models... Read more
Deal Alert! B&H Photo has Apple’s 14-inch...
B&H Photo has new Gray and Black 14″ M3, M3 Pro, and M3 Max MacBook Pros on sale for $200-$300 off MSRP, starting at only $1399. B&H offers free 1-2 day delivery to most US addresses: – 14″ 8... Read more
Department Of Justice Sets Sights On Apple In...
NEWS – The ball has finally dropped on the big Apple. The ball (metaphorically speaking) — an antitrust lawsuit filed in the U.S. on March 21 by the Department of Justice (DOJ) — came down following... Read more
New 13-inch M3 MacBook Air on sale for $999,...
Amazon has Apple’s new 13″ M3 MacBook Air on sale for $100 off MSRP for the first time, now just $999 shipped. Shipping is free: – 13″ MacBook Air (8GB RAM/256GB SSD/Space Gray): $999 $100 off MSRP... Read more
Amazon has Apple’s 9th-generation WiFi iPads...
Amazon has Apple’s 9th generation 10.2″ WiFi iPads on sale for $80-$100 off MSRP, starting only $249. Their prices are the lowest available for new iPads anywhere: – 10″ 64GB WiFi iPad (Space Gray or... Read more
Discounted 14-inch M3 MacBook Pros with 16GB...
Apple retailer Expercom has 14″ MacBook Pros with M3 CPUs and 16GB of standard memory discounted by up to $120 off Apple’s MSRP: – 14″ M3 MacBook Pro (16GB RAM/256GB SSD): $1691.06 $108 off MSRP – 14... Read more
Clearance 15-inch M2 MacBook Airs on sale for...
B&H Photo has Apple’s 15″ MacBook Airs with M2 CPUs (8GB RAM/256GB SSD) in stock today and on clearance sale for $999 in all four colors. Free 1-2 delivery is available to most US addresses.... Read more
Clearance 13-inch M1 MacBook Airs drop to onl...
B&H has Apple’s base 13″ M1 MacBook Air (Space Gray, Silver, & Gold) in stock and on clearance sale today for $300 off MSRP, only $699. Free 1-2 day shipping is available to most addresses in... Read more

Jobs Board

Medical Assistant - Surgical Oncology- *Apple...
Medical Assistant - Surgical Oncology- Apple Hill Location: WellSpan Medical Group, York, PA Schedule: Full Time Sign-On Bonus Eligible Remote/Hybrid Regular Apply Read more
Omnichannel Associate - *Apple* Blossom Mal...
Omnichannel Associate - Apple Blossom Mall Location:Winchester, VA, United States (https://jobs.jcp.com/jobs/location/191170/winchester-va-united-states) - Apple Read more
Cashier - *Apple* Blossom Mall - JCPenney (...
Cashier - Apple Blossom Mall Location:Winchester, VA, United States (https://jobs.jcp.com/jobs/location/191170/winchester-va-united-states) - Apple Blossom Mall Read more
Operations Associate - *Apple* Blossom Mall...
Operations Associate - Apple Blossom Mall Location:Winchester, VA, United States (https://jobs.jcp.com/jobs/location/191170/winchester-va-united-states) - Apple Read more
Business Analyst | *Apple* Pay - Banco Popu...
Business Analyst | Apple PayApply now " Apply now + Apply Now + Start applying with LinkedIn Start + Please wait Date:Mar 19, 2024 Location: San Juan-Cupey, PR Read more
All contents are Copyright 1984-2011 by Xplain Corporation. All rights reserved. Theme designed by Icreon.