TweetFollow Us on Twitter

Children of the Revolution

Volume Number: 19 (2003)
Issue Number: 10
Column Tag: Programming

QuickTime Toolkit

Children of the Revolution

by Tim Monroe

Editing QuickTime Movies with Revolution

Introduction

In the previous QuickTime Toolkit article ("Revolution" in MacTech, September 2003), we took a first look at Revolution, a rapid application development tool published by Runtime Revolution Ltd. We saw how to create a new application -- which we called RunRevVeez -- that can open and display QuickTime movies. We saw how to set things up so that the user can have several movies open at once, and we saw how to use a few of the built-in Revolution commands to modify the appearance of a movie player object at runtime. In terms of movie playback, RunRevVeez is just about complete.

The situation with movie editing is somewhat different, however. As I mentioned last time, Revolution has no built-in support for editing QuickTime movies. In addition (as far as I can tell), it provides no support for tracking changes to a window or document, and it provides no way to save an edited movie. We'd certainly like our application to be able to handle these tasks, so we'll have to go beyond the built-in capabilities of Revolution. We need to write a Revolution plug-in.

Happily, Runtime Revolution provides a software development kit (SDK) for writing Revolution plug-ins, and this makes writing our plug-in a snap. With just a few dozen lines of new C code and a handful of routines borrowed from our existing C-based sample application QTShell, we'll be able to handle all the basic editing operations, keep track of the modification state of a movie window, and save edited movies into new files.

Unhappily, even with this plug-in, there are a few things we won't be able to accomplish with Revolution. The Revolution runtime engine opens QuickTime movie files with read-only permission, which effectively prevents us from saving any changes to a movie into the file we opened the movie from. We will be able to write an edited movie into a new file. (In a nutshell, we'll be able to implement the "Save As" menu item but not the Save menu item.) Also, the Revolution runtime engine installs a movie controller action filter procedure, which effectively prevents us from installing our own procedure. This restricts our ability to access many important QuickTime capabilities. (You may recall that REALbasic currently has this same limitation; see "Basic Instinct" in MacTech, February, 2003.)

In this article, we'll continue our development of RunRevVeez. We'll implement the editing operations on a movie, which requires us to develop a plug-in and then to call the plug-in from within our scripts. We'll look at the file-handling operations (principally, "Save As" and Close) in the next article.

One final note before we begin: Runtime Revolution has recently released Revolution version 2.1. In these articles, I've used version 2.0.2. I would assume that the plug-in and Revolution project will work unchanged under 2.1, but I have not actually verified that.

Revolution Plug-Ins

The Revolution runtime engine is based largely on an existing product called MetaCard, which was introduced in 1990 as a competitor to Apple's HyperCard. Not surprisingly, the plug-in architecture used by MetaCard, and hence Revolution, is identical to that introduced by HyperCard. HyperCard can be extended by adding modules of commands and functions called externals. A set of external commands is called an XCMD and a set of external functions is called an XFCN.

Originally, XCMDs and XFCNs were packaged as executable code resources that were added to the resource fork of the application or to the resource fork of a stack. MetaCard and Revolution followed this example through Revolution version 1.1.1. In version 2.0 and later, the packaging of externals was changed; in current versions, externals on Mac OS X are packaged as bundles, which can be copied into the application bundle.

The packaging actually doesn't really matter all that much, since it will be taken care of by the project files provided with the plug-in SDK. The current SDK provides project files for both CodeWarrior and Project Builder. In this article, we'll work with the Project Builder version, whose project window is shown in Figure 1. (Notice that I've renamed the project as "QTExternal".) We'll need to modify only one file here, external.c. The file XCmdGlue.c contains a number of support routines for the external; we won't need to call any of those routines.


Figure 1: The Project Builder project

Connecting to the Runtime Engine

Our Revolution external will define a number of procedures and functions that can be called by RunRevVeez scripts. To expose those routines to the runtime engine, we need to declare two global variables, Xname and Xtable. The Xname variable specifies the name of the external:

char   Xname[] = "QuickTime Revolution External";

The Xtable variable contains an array of procedure specifiers. Each entry in the array specifies information about a single external function or command. Here's our array:

Xternal Xtable[] = {
   {"mcInitialize", XCOMMAND, 0, XCMD_MCInitialize, 
                                                            XCMD_Abort},
   {"mcUndo", XFUNCTION, 0, XCMD_MCUndo, XCMD_Abort},
   {"mcCut", XFUNCTION, 0, XCMD_MCCut, XCMD_Abort},
   {"mcCopy", XCOMMAND, 0, XCMD_MCCopy, XCMD_Abort},
   {"mcPaste", XFUNCTION, 0, XCMD_MCPaste, XCMD_Abort},
   {"mcClear", XFUNCTION, 0, XCMD_MCClear, XCMD_Abort},
   {"selectAll", XCOMMAND, 0, XCMD_SelectAll, XCMD_Abort},
   {"selectNone", XCOMMAND, 0, XCMD_SelectNone, 
                                                            XCMD_Abort},
   
   {"mcEnableEditMenuItem", XFUNCTION, 0, 
                           XCMD_MCEnableEditMenuItem, XCMD_Abort},
   
   {"windowSetModified", XFUNCTION, 0, 
                        XCMD_SetWindowModified, XCMD_Abort},
   {"saveAs", XFUNCTION, 0, XCMD_SaveAs, XCMD_Abort},
   {"", XNONE, 0, NULL, NULL}
};

The first item in a procedure specifier is the name of the routine that we'll use in our scripts. The second item indicates the type of routine; it's XCOMMAND for commands (which do not return a value to the caller) and XFUNCTION for functions (which do return a value to the caller). The third entry is used by the runtime engine and should be set to 0 by our external. The fourth entry is the name of the corresponding C language routine in the external. (In other words, it's the routine that is called when our script executes the first item.) Finally, the fifth item is the name of an abort routine, which is called when the user cancels the execution of an external routine. All our external routines will use the same abort routine, shown in Listing 1.

Listing 1: Handling user cancellations

XCMD_Abort
void XCMD_Abort()
{
   DebugStr("\pQuickTime Revolution External abort");
}

Our abort routine just prints a diagnostic message on the standard error output.

Handling Commands

When a script calls the mcInitialize command (for instance), the external function XCMD_MCInitialize is executed; XCMD_MCInitialize has this declaration:

void XCMD_MCInitialize (char *args[], int nargs, 
            char **retstring, Bool *pass, Bool *error);

The first parameter passed to XCMD_MCInitialize is an array of C strings that specifies the parameters that were passed to the mcInitialize command. The second parameter specifies the number of items in that array. We'll call mcInitialize with only one parameter, like this:

put the movieControllerID of player "MoviePlayer" \ 
               of stack newStackName into mc
mcInitialize(mc)

The third parameter, retstring, is a pointer to a C string that contains the results of the external routine. For procedures, this is ignored by the runtime engine; for functions, this string is returned to the script as the command result. The buffer for this string must be allocated by the external and is disposed of by the runtime engine.

The fourth and fifth parameters are used to pass other information back to the runtime engine. The pass parameter indicates whether we want the command (in this case, mcInitialize) to be passed up the message hierarchy after it is executed. In general, we shall return false in this parameter. The error parameter indicates the success or failure of the external routine. Once again, we'll always pass back false, to indicate that no error occurred. (Errors may indeed occur within our external routines, but RunRevVeez will have no capability to work around errors; so there's little point in letting it know that something went wrong.)

Configuring the Movie Controller

So let's see how we can implement the handler for the mcInitialize command. As we've seen, the args parameter will contain a single C string, which is the movie controller identifier encoded as a string. To get a value of type MovieController, we need to convert the string to a long.

mc = (MovieController)atol(args[0]);

Once we've got the movie controller identifier, we can call any QuickTime APIs that operate on a movie controller. In RunRevVeez, we need to enable editing (by calling MCEnableEditing) and enable keyboard event handling (by calling MCDoAction with the mcActionSetKeysEnabled selector). Listing 2 shows our complete handler for the mcInitialize command.

Listing 2: Initializing the movie controller

XCMD_MCInitialize
void XCMD_MCInitialize (char *args[], int nargs, 
            char **retstring, Bool *pass, Bool *error)
{
   MovieController mc = NULL;
   ComponentResult result = noErr;
   char *retstr = NULL;
   
   // initialize the movie controller as desired
   *pass = false;
   *error = false;
   if (nargs == 1) {
      mc = (MovieController)atol(args[0]);
      
      if (mc != NULL) {
         // enable editing
         result = MCEnableEditing(mc, true);
         
         // enable keyboard event handling
         MCDoAction(mc, mcActionSetKeysEnabled, (void *)true);
         
         // disable drag support
         MCDoAction(mc, mcActionSetDragEnabled, 
                                                            (void *)false);
      }
   }
   
   retstr = calloc(1, 2);
   if (retstr != NULL)
      retstr[0] = (result == noErr) ? '0': '1';
    
   *retstring = retstr;
}

As indicated just above, we set both pass and error to false. And we pass back, via retstring, a C string of length 1 that contains either "0" or "1". RunRevVeez ignores that value.

Once we've successfully called mcInitialize, the thumb in the controller bar will change to reflect that editing is enabled (as seen in Figure 2).


Figure 2: A movie window with editing enabled

Handling Edit Operations

So, we've enabled movie controller editing. Now we need to handle the various editing operations. In these cases, we need to pass a value back to the caller, indicating whether the operation completed successfully. That's so RunRevVeez can know to set the movie window as modified and that the movie has changed since last opened or saved. We'll return the string "1" if the edit operation fails and "0" if it succeeds. Listing 3 shows how we'll handle the mcUndo command.

Listing 3: Undoing a movie edit

XCMD_MCUndo
void XCMD_MCUndo (char *args[], int nargs, 
            char **retstring, Bool *pass, Bool *error)
{
   MovieController mc = NULL;
   ComponentResult result = noErr;
   char *retstr = NULL;
   
   *pass = false;
   *error = false;
   if (nargs == 1) {
      mc = (MovieController)atol(args[0]);
      if (mc != NULL)
         result = MCUndo(mc);
   }
   
   retstr = calloc(1, 2);
   if (retstr != NULL)
      retstr[0] = (result == noErr) ? '0': '1';
    
   *retstring = retstr;
}

We simply retrieve the movie controller identifier and call MCUndo. Then we call calloc to allocate a 2-byte buffer, to hold the returned character and the null terminating byte.

The other editing operations are quite similar. Listing 4 shows how we handle the mcCut command, and Listing 5 shows how we handle the mcCopy command. Notice in both cases that we call PutMovieOnScrap to place the cut or copied movie segment onto the scrap.

Listing 4: Cutting a movie selection

XCMD_MCCut
void XCMD_MCCut (char *args[], int nargs, 
            char **retstring, Bool *pass, Bool *error)
{
   MovieController mc = NULL;
   ComponentResult result = noErr;
   Movie editmovie = NULL;
   char *retstr = NULL;
   
   *pass = false;
   *error = false;
   if (nargs == 1) {
      mc = (MovieController)atol(args[0]);
      if (mc != NULL) {
         editmovie = MCCut(mc);
         result = (editmovie != NULL) ? result: invalidMovie;
      }
   }
   
   // place the cut movie segment onto the scrap
   if (editmovie != NULL) {
      PutMovieOnScrap(editmovie, 0L);
      DisposeMovie(editmovie);
   }
   retstr = calloc(1, 2);
   if (retstr != NULL)
      retstr[0] = (result == noErr) ? '0': '1';
      
   *retstring = retstr;
}

Listing 5: Copying a movie selection

XCMD_MCCopy
void XCMD_MCCopy (char *args[], int nargs, 
            char **retstring, Bool *pass, Bool *error)
{
   MovieController mc = NULL;
   ComponentResult result = noErr;
   Movie editmovie = NULL;
   char *retstr = NULL;
   
   *pass = false;
   *error = false;
   if (nargs == 1) {
      mc = (MovieController)atol(args[0]);
   
      if (mc != NULL) {
         editmovie = MCCopy(mc);
         result = (editmovie != NULL) ? result: invalidMovie;
      }
   }
   
   // place the copied movie segment onto the scrap
   if (editmovie != NULL) {
      PutMovieOnScrap(editmovie, 0L);
      DisposeMovie(editmovie);
   }
   retstr = calloc(1, 2);
   if (retstr != NULL)
      retstr[0] = (result == noErr) ? '0': '1';
      
   *retstring = retstr;
}

Implementation of XCMD_MCPaste and XCMD_MCClear is left as an easy exercise for the reader. (The complete code for the QuickTime external is of course contained in the source code accompanying this article.)

Selecting All or None of a Movie

Our Edit menu contains two further items, "Select All" and "Select None", which are once again easy to implement. In earlier articles, we've seen how to handle these items by calling MCDoAction with the mcActionSetSelectionDuration selector. Listing 6 shows how our Revolution external handles the selectAll command, and Listing 7 shows how our Revolution external handles the selectNone command.

Listing 6: Selecting all of a movie

XCMD_SelectAll
void XCMD_SelectAll (char *args[], int nargs, 
            char **retstring, Bool *pass, Bool *error)
{
   MovieController mc = NULL;
   Movie mv = NULL;
   ComponentResult result = noErr;
   TimeRecord tr;
   char *retstr = NULL;
   
   *pass = false;
   *error = false;
   if (nargs == 1) {
      mc = (MovieController)atol(args[0]);
   
      if (mc != NULL) {
         mv = MCGetMovie(mc);
         if (mv) {
            tr.value.hi = 0;
            tr.value.lo = 0;
            tr.base = 0;
            tr.scale = GetMovieTimeScale(mv);   
            result = MCDoAction(mc, 
                              mcActionSetSelectionBegin, &tr);
            
            tr.value.hi = 0;
            tr.value.lo = GetMovieDuration(mv);   
            tr.base = 0;
            tr.scale = GetMovieTimeScale(mv);   
            result = MCDoAction(mc, 
                              mcActionSetSelectionDuration, &tr);
         }
      }
   }
   
   retstr = calloc(1, 2);
   if (retstr != NULL)
      retstr[0] = (result == noErr) ? '0': '1';
      
   *retstring = retstr;
}

Listing 7: Selecting none of a movie

XCMD_SelectNone
void XCMD_SelectNone (char *args[], int nargs, 
            char **retstring, Bool *pass, Bool *error)
{
   MovieController mc = NULL;
   Movie mv = NULL;
   ComponentResult result = noErr;
   TimeRecord tr;
   char *retstr = NULL;
   
   *pass = false;
   *error = false;
   if (nargs == 1) {
      mc = (MovieController)atol(args[0]);
   
      if (mc != NULL) {
         mv = MCGetMovie(mc);
         if (mv) {
            tr.value.hi = 0;
            tr.value.lo = 0;   
            tr.base = 0;
            tr.scale = GetMovieTimeScale(mv);   
            result = MCDoAction(mc, 
                              mcActionSetSelectionDuration, &tr);
         }
      }
   }
   
   retstr = calloc(1, 2);
   if (retstr != NULL)
      retstr[0] = (result == noErr) ? '0': '1';
      
   *retstring = retstr;
}

Enabling and Disabling Edit Menu Items

RunRevVeez needs to enable and disable the Edit menu items according to the edit state of the movie in a movie window. For instance, when a movie is first opened and no edit operations have yet occurred, the Undo item should be disabled. QuickTime provides the MCGetControllerInfo function, which we've used in the past to adjust the states of our edit menu items. We'll use it again here, as shown in Listing 8.

Listing 8: Adjusting the Edit menu

XCMD_MCEnableEditMenuItem
void XCMD_MCEnableEditMenuItem (char *args[], int nargs, 
            char **retstring, Bool *pass, Bool *error)
{
   MovieController mc = NULL;
   ComponentResult result = noErr;
   long mcInfo = 0L;
   short index = 0;
   char *retstr = NULL;
   
   *pass = false;
   *error = false;
   retstr = malloc(2);      // either "0" or "1", plus the terminating null byte
   if (nargs == 2) {
      mc = (MovieController)atol(args[0]);
      index = (short)atoi(args[1]);
      
      if (mc != NULL)
         result = MCGetControllerInfo(mc, &mcInfo);
   }
   
   switch (index) {
      case kUndoItemIndex:
         retstr[0] = mcInfo & mcInfoUndoAvailable ? '1': '0';
         break;
   
      case kCutItemIndex:
         retstr[0] = mcInfo & mcInfoCutAvailable ? '1': '0';
         break;
   
      case kCopyItemIndex:
         retstr[0] = mcInfo & mcInfoCopyAvailable ? '1': '0';
         break;
   
      case kPasteItemIndex:
         retstr[0] = mcInfo & mcInfoPasteAvailable ? '1': '0';
         break;
   
      case kClearItemIndex:
         retstr[0] = mcInfo & mcInfoClearAvailable ? '1': '0';
         break;
   
      case kSelectAllItemIndex:
      case kSelectNoneItemIndex:
         retstr[0] = mcInfo & mcInfoEditingEnabled ? '1': '0';
         break;
         
      default:
         DebugStr("\pGOT AN INDEX WE DIDN'T EXPECT!");
         break;
   }   
   
   // tack on the terminating null byte
   retstr[1] = 0;
    
   *retstring = retstr;
}

Notice that our code here looks for two parameters, which are the movie controller identifier and the index of the menu item we want information about. If, according to MCGetControllerInfo, the menu item with that index should be enabled, XCMD_MCEnableEditMenuItem passes back the string "1"; otherwise it passes back the string "0".

In RunRevVeez, the code that enables or disables the menu items is contained in the script attached to the menu item group (and not to any particular menu or item). That's because, when the user clicks on the menu bar, a mouseDown message is sent to the menu item group. We want to call mcEnableEditMenuItem for each menu item index and adjust the menu item according to the value returned by it.

Listing 9: Adjusting the Edit menu

mouseDown
on mouseDown
   put first line of the openStacks into theTopStack
   put exists(player "MoviePlayer" of stack theTopStack) \
                  into gotPlayer
  
   repeat for each item itemIndex in "1,3,4,5,6,8,9"
      if gotPlayer then
         if mcEnableEditMenuItem(the movieControllerID of \
                  player "MoviePlayer" of stack theTopStack, \
                     itemIndex) is "1" then
            enable menuItem itemIndex of menu "Edit"
         else
            disable menuItem itemIndex of menu "Edit"
         end if
      else
         disable menuItem itemIndex of menu "Edit"
      end if
   end repeat
  
end mouseDown

We also need to adjust the states of the items in the File menu and the Movie menu. We'll postpone our consideration of the File menu to the next article. We can handle the Movie menu as shown in Listing 10.

Listing 10: Adjusting the Movie menu

mouseDown
if gotPlayer then
   enable menuItem kShowBarItemIndex of menu "Movie"
   enable menuItem kHideSpeakerItemIndex of menu "Movie"
else
   disable menuItem kShowBarItemIndex of menu "Movie"
   disable menuItem kHideSpeakerItemIndex of menu "Movie"
end if

Here we use a few constants that we've defined in our message handler:

constant kShowBarItemIndex = 1
constant kHideSpeakerItemIndex = 2
Setting the Window Status

In the Aqua interface, a window's close button contains a dot if the movie in the window has been modified since opened or last saved (compare Figure 3 with Figure 2).


Figure 3: A modified movie window

In earlier QuickTime Toolkit articles, we've seen that we can set the window modification state by calling SetWindowModified. With Revolution, we need to call into our external to do this. Listing 11 shows our definition of the XCMD_SetWindowModified function.

Listing 11: Setting the window modification state

XCMD_SetWindowModified
void XCMD_SetWindowModified (char *args[], int nargs, 
            char **retstring, Bool *pass, Bool *error)
{
   WindowPtr wID = NULL;
   Boolean state;
   OSErr result = noErr;
   char *retstr = NULL;
   *pass = false;
   *error = false;
   if (nargs == 2) {
      wID = (WindowPtr)atol(args[0]);
      state = (Boolean)atoi(args[1]);
      if (wID != NULL)
         result = SetWindowModified(wID, state);
   }
      
   retstr = calloc(1, 2);
   if (retstr != NULL)
      retstr[0] = (result == noErr) ? '0': '1';
      
   *retstring = retstr;
}

This is pretty much like all the external procedures we've seen so far, except that the first parameter here is of type WindowPtr. Our call to windowSetModified looks like this:

get windowSetModified(windowID of stack theTopStack, 1)

A stack's windowID property contains the operating system ID of the window containing the stack; on Mac OS, this ID is a window pointer. (By the way, notice that we invoke the windowSetModified command by passing it as an expression to the get command. The "get expr" command is a shortcut for the expression:

put expr into it

We need to treat windowSetModified as a function, since that's how we declared it. If we had declared it as a command, we would omit the get.)

We also need to keep track of a window's modification state within our scripts in RunRevVeez (so, for instance, we know whether to enable or disable some of the items in the File menu). We could implement yet another function in our external that calls IsWindowModified. Or we can define a custom property associated with the movie window stack that keeps track of this modification state. Let's use a custom property. Open the movie window's property inspector palette and select the "Custom Properties" panel in the pop-up menu. The original panel looks like Figure 4.


Figure 4: The movie window's custom properties (original)

Click the "+" icon to add a new property. Let's call the new property movieChanged. When a movie window is first opened, this property should be set to 0, so set the property contents accordingly. The property inspector palette now looks like Figure 5.


Figure 5: The movie window's custom properties (final)

Once we've done this, we can access the movieChanged property just like we access any of the built-in properties, for example like this:

set the movieChanged of stack theTopStack to true

We'll see some examples of this in the next section.

Movie Editing

We're now finished constructing the movie editing portions of our Revolution plug-in module. It's very easy to put them to work. When the user selects an item in the Edit menu, the menuPick message handler of the Edit menu is called. Listing 12 shows our complete menuPick handler. Notice that we check to make sure that the value returned by the editing operations (for example, mcCut) is the string "0", which indicates that the operation completed successfully.

Listing 12: Handling the Edit menu items

menuPick
on menuPick pWhich
  
   put first line of the openStacks into theTopStack
   if exists(player "MoviePlayer" of stack theTopStack) then
      put the movieControllerID of player "MoviePlayer" of \
                                 stack theTopStack into mc
      put false into changed
    
      switch pWhich
      case "Undo"
         if mcUndo(mc) = "0" then put true into changed
         break
      case "Cut"
         if mcCut(mc) = "0" then put true into changed
         break
      case "Copy"
         mcCopy(mc)
         break
      case "Paste"
         if mcPaste(mc) = "0" then put true into changed
         break
      case "Clear"
         if mcClear(mc) = "0" then put true into changed
         break
      case "Select All"
          selectAll(mc)
         break
      case "Select None"
         selectNone(mc)
         break
      end switch
     
      if changed then
         set the movieChanged of stack theTopStack to true
         get windowSetModified \
                                    (windowID of stack theTopStack, 1)
         sizeStackToMovie the short name of stack theTopStack
      end if
    
  end if
end menuPick

We also call the sizeStackToMovie method if the movie has been edited, since the size of the movie may have changed.

Conclusion

In this article, we've focused mainly on adding the ability to edit movies to our application RunRevVeez. We've seen how to construct a plug-in that allows our Revolution scripts to invoke external code modules. This is the primary avenue by which we can enhance the built-in behaviors and capabilities of Revolution.

We've got a little bit more work to do to get RunRevVeez to operate precisely as desired. We still need to handle the "Save As" and Close menu items in the File menu, and we need to tie up a few remaining loose ends. We'll tackle all that in the next article.

Credits

Thanks are due once again to Kevin Miller and Tuviah Snyder at Runtime Revolution Ltd. Tuviah was especially helpful with the plug-in. And a special thanks is again due to Geoff Canyon of Inspired Logic, LLC.


Tim Monroe is a member of the QuickTime engineering team at Apple. You can contact him at monroe@mactech.com. The views expressed here are not necessarily shared by his employer.

 

Community Search:
MacTech Search:

Software Updates via MacUpdate

Latest Forum Discussions

See All

Fresh From the Land Down Under – The Tou...
After a two week hiatus, we are back with another episode of The TouchArcade Show. Eli is fresh off his trip to Australia, which according to him is very similar to America but more upside down. Also kangaroos all over. Other topics this week... | Read more »
TouchArcade Game of the Week: ‘Dungeon T...
I’m a little conflicted on this week’s pick. Pretty much everyone knows the legend of Dungeon Raid, the match-3 RPG hybrid that took the world by storm way back in 2011. Everyone at the time was obsessed with it, but for whatever reason the... | Read more »
SwitchArcade Round-Up: Reviews Featuring...
Hello gentle readers, and welcome to the SwitchArcade Round-Up for July 19th, 2024. In today’s article, we finish up the week with the unusual appearance of a review. I’ve spent my time with Hot Lap Racing, and I’m ready to give my verdict. After... | Read more »
Draknek Interview: Alan Hazelden on Thin...
Ever since I played my first release from Draknek & Friends years ago, I knew I wanted to sit down with Alan Hazelden and chat about the team, puzzle games, and much more. | Read more »
The Latest ‘Marvel Snap’ OTA Update Buff...
I don’t know about all of you, my fellow Marvel Snap (Free) players, but these days when I see a balance update I find myself clenching my… teeth and bracing for the impact to my decks. They’ve been pretty spicy of late, after all. How will the... | Read more »
‘Honkai Star Rail’ Version 2.4 “Finest D...
HoYoverse just announced the Honkai Star Rail (Free) version 2.4 “Finest Duel Under the Pristine Blue" update alongside a surprising collaboration. Honkai Star Rail 2.4 follows the 2.3 “Farewell, Penacony" update. Read about that here. | Read more »
‘Vampire Survivors+’ on Apple Arcade Wil...
Earlier this month, Apple revealed that poncle’s excellent Vampire Survivors+ () would be heading to Apple Arcade as a new App Store Great. I reached out to poncle to check in on the DLC for Vampire Survivors+ because only the first two DLCs were... | Read more »
Homerun Clash 2: Legends Derby opens for...
Since launching in 2018, Homerun Clash has performed admirably for HAEGIN, racking up 12 million players all eager to prove they could be the next baseball champions. Well, the title will soon be up for grabs again, as Homerun Clash 2: Legends... | Read more »
‘Neverness to Everness’ Is a Free To Pla...
Perfect World Games and Hotta Studio (Tower of Fantasy) announced a new free to play open world RPG in the form of Neverness to Everness a few days ago (via Gematsu). Neverness to Everness has an urban setting, and the two reveal trailers for it... | Read more »
Meditative Puzzler ‘Ouros’ Coming to iOS...
Ouros is a mediative puzzle game from developer Michael Kamm that launched on PC just a couple of months back, and today it has been revealed that the title is now heading to iOS and Android devices next month. Which is good news I say because this... | Read more »

Price Scanner via MacPrices.net

Amazon is still selling 16-inch MacBook Pros...
Prime Day in July is over, but Amazon is still selling 16-inch Apple MacBook Pros for $500-$600 off MSRP. Shipping is free. These are the lowest prices available this weekend for new 16″ Apple... Read more
Walmart continues to sell clearance 13-inch M...
Walmart continues to offer clearance, but new, Apple 13″ M1 MacBook Airs (8GB RAM, 256GB SSD) online for $699, $300 off original MSRP, in Space Gray, Silver, and Gold colors. These are new MacBooks... Read more
Apple is offering steep discounts, up to $600...
Apple has standard-configuration 16″ M3 Max MacBook Pros available, Certified Refurbished, starting at $2969 and ranging up to $600 off MSRP. Each model features a new outer case, shipping is free,... Read more
Save up to $480 with these 14-inch M3 Pro/M3...
Apple has 14″ M3 Pro and M3 Max MacBook Pros in stock today and available, Certified Refurbished, starting at $1699 and ranging up to $480 off MSRP. Each model features a new outer case, shipping is... Read more
Amazon has clearance 9th-generation WiFi iPad...
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
Apple is offering a $50 discount on 2nd-gener...
Apple has Certified Refurbished White and Midnight HomePods available for $249, Certified Refurbished. That’s $50 off MSRP and the lowest price currently available for a full-size Apple HomePod today... Read more
The latest MacBook Pro sale at Amazon: 16-inc...
Amazon is offering instant discounts on 16″ M3 Pro and 16″ M3 Max MacBook Pros ranging up to $400 off MSRP as part of their early July 4th sale. Shipping is free. These are the lowest prices... Read more
14-inch M3 Pro MacBook Pros with 36GB of RAM...
B&H Photo has 14″ M3 Pro MacBook Pros with 36GB of RAM and 512GB or 1TB SSDs in stock today and on sale for $200 off Apple’s MSRP, each including free 1-2 day shipping: – 14″ M3 Pro MacBook Pro (... Read more
14-inch M3 MacBook Pros with 16GB of RAM on s...
B&H Photo has 14″ M3 MacBook Pros with 16GB of RAM and 512GB or 1TB SSDs in stock today and on sale for $150-$200 off Apple’s MSRP, each including free 1-2 day shipping: – 14″ M3 MacBook Pro (... Read more
Amazon is offering $170-$200 discounts on new...
Amazon is offering a $170-$200 discount on every configuration and color of Apple’s M3-powered 15″ MacBook Airs. Prices start at $1129 for models with 8GB of RAM and 256GB of storage: – 15″ M3... Read more

Jobs Board

*Apple* Systems Engineer - Chenega Corporati...
…LLC,** a **Chenega Professional Services** ' company, is looking for a ** Apple Systems Engineer** to support the Information Technology Operations and Maintenance Read more
Solutions Engineer - *Apple* - SHI (United...
**Job Summary** An Apple Solution Engineer's primary role is tosupport SHI customers in their efforts to select, deploy, and manage Apple operating systems and Read more
*Apple* / Mac Administrator - JAMF Pro - Ame...
Amentum is seeking an ** Apple / Mac Administrator - JAMF Pro** to provide support with the Apple Ecosystem to include hardware and software to join our team and 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
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
All contents are Copyright 1984-2011 by Xplain Corporation. All rights reserved. Theme designed by Icreon.