TweetFollow Us on Twitter

The Eight Queens Problem

Volume Number: 13 (1997)
Issue Number: 12
Column Tag: Programming Techniques

Solving The Eight Queens Problem

by F.C. Kuechmann

Using the graphics of a Macintosh to explore the recursion and backtracking-based solution to this common puzzle

One of the oldest and most familiar intellectual puzzles whose solution is uniquely suited to the computer is the eight queens problem, in which the goal is to systematically determine each of the 92 different ways eight queens can be placed on a chessboard without conflict. Of the 92 solutions, 12 [numbers 1, 2, 5, 6, 7, 8, 9, 10, 11, 14, 17 and 18] are unique; the remaining 80 are variations on the twelve -- mirror images on the vertical, horizontal or diagonal axes, or 90, 180 or 270 degree rotations.

Since queens can move any number of squares in any direction, each queen must be placed on its own row, column and diagonals. Niklaus Wirth [1976, 1986] describes it as a familiar problem and states that Gauss considered it as early as 1850. No straightforward mathematical formula has ever been devised for its solution. The trial-and-error method, often combined with such problem solving techniques as recursion and backtracking, while tedious and error prone for humans, is well-suited to the computer. (A computer doesn't get bored, make mistakes, or have a cat that jumps onto the chess board.) Each time the active queen is moved to a new position the new location is tested for conflict unless that position is off the board. In that case, we backtrack to the previous column or row, advance that previously secure queen, and proceed. Thus a backtrack involves a move without testing the new position for conflict. 981 queen moves (876 position tests plus 105 backtracks) are required for the first solution alone. 16,704 moves (14,852 tests and 1852 backtracks) are needed to find all 92 solutions. If we continue testing until all possibilities are exhausted, we've made 17,684 moves -- 15,720 tests plus 1964 backtracks. Given those figures, it's easy to see why the solution is best left to computers.

While text-oriented computers can determine the solutions as well as any, a graphically-oriented computer like the Mac is ideally suited for illustrating the underlying algorithm.

The widespread use of the personal computer as a teaching tool has contributed to the appearance of the eight queens problem in several textbooks in the past 20-plus years, including especially Wirth [1976, 1986] and Budd [1987, 1991, 1996]. Niklaus Wirth's solutions are in Pascal and Modula-2. Timothy Budd has discussed object-oriented solutions, in languages ranging from SmallTalk to Object Pascal to Java, in at least three books. Microware furnished a structured BASIC version of Wirth's Pascal solution with their BASIC09 interpreter, which probably received its widest distribution packaged for the Radio Shack Color Computer 2 and 3.

A web search on the phrase "eight queens" will uncover several graphic Java solutions, some interactive, based on a problem posed by Budd [1996]. For some reason with my PowerPC Mac 6500/225 and Quadra 650, Microsoft's Internet Explorer browser works better in displaying the interactive Java versions on the web than does Netscape's Navigator.

WIRTH's Algorithm

The queen conflict tracking mechanism employed by Wirth consists of three Boolean arrays that track queen status for each row and diagonal. TRUE means no queen is on that row or diagonal; FALSE means a queen is already there.

Figure 1 shows the mapping of the arrays to the chess board for Pascal. All array elements are initialized to TRUE. The Row array elements 1-8 correspond to rows 1-8 on the board. A queen in row n sets rows array element n to FALSE.

Column-Row array elements are numbered from -7 to 7 and correspond to the difference between column and row numbers.

A queen at column 1, row 1 sets array element zero to FALSE. A queen at column 1, row 8 sets array element -7 to FALSE.

The Column+Row array elements are numbered 2-16 and correspond to the sum of the column and row. A queen placed in column 1, row 1 sets array element 2 to FALSE. A queen placed in column 3, row 5 sets array element 8 to FALSE.

Figures 2, 3 and 4 show the changes in array element values as 1, 2 and 3 queens are placed on the board.

In Figure 2, Row array element 1, Column-Row array element 0, and Column+Row array element 2 are all set to FALSE.

Figure 2. One conflict queen

Figure 3. Two conflict-free queens.

In Figure 3, Row array element 3, Column-Row array element -1, and Column+Row Array element 5 are set also set FALSE.

Figure 4. Three conflict-free queens.

In Figure 4, Row Array element 5, Column-Row array element -2, and Column+Row array element 8 are added to the FALSE list.

It would require hundreds of pages to show the entire move sequence just for the first solution, which is shown in Figure 5. All 981 moves can be easily visualized by stepping through the program, either in single-step mode or at slow speed, while observing changes in the Boolean array values displayed to the right of the board, as queens are set and removed.

Figure 5. The first solution.

The Trouble With C

In the C language, with its zero-based arrays, the mapping of the arrays to the board is a bit more complicated. Array elements are numbered 0-n and the mapping of board positions to the arrays is compensated accordingly. The Row array elements 0-7 correspond to rows 1-8 on the board. A queen in row n sets Row array element n-1 to FALSE. Column-Row array elements are numbered from 0 to 14 and correspond to the difference between column and row numbers, plus 7. A queen at column 1, row 1 sets array element 7 (1-1+7=7) to FALSE. A queen at column 1, row 8 sets array element 0 (1-8+7=0) to FALSE. The Column+Row array elements are numbered 0-14 and correspond to the sum of the column and row, minus 2. A queen placed in column 1, row 1 sets array element 0 (1+1-2=0) to FALSE. A queen placed in column 3, row 5 sets array element 6 (3+5-2=6) to FALSE.

Many Ways to Tango

Recursive and non-recursive variations of Wirth's method are easy to implement on the Macintosh and other computers in most popular languages, including Pascal, C and structured BASIC. The major code differences between recursive and non-recursive variations are these:

  1. The non-recursive version adds an 8-element integer array to hold the queen row for each column; with recursion this housekeeping is handled by the system.
  2. The recursive program's main procedure uses a single for loop, with backtracking handled by the system, whereas the non-recursive version uses two nested repeat loops and a separate procedure to handle backtracking.

With these exceptions, the code for the two variations is identical.

In Pascal, the main loop of the recursive solution looks like this:

procedure Try(column:integer);
  begin
    for row := 1 to 8 do
      begin
        { code }
        if column < 8 then
          Try(column + 1)  { procedure calls itself }
        else
          PrintSolution;
        { code }
      end;    {for}
  end;

The core of the non-recursive solution:

    repeat
      repeat
        { code }
      until row > 8;
      backtrack;
    until column < 1;

Listing 1. Try Try

This procedure is called with a column value of 1 after initializing the gRowe, gColPlusRow and gColMinusRow Boolean arrays to TRUE. When it finishes, all 92 solutions have been found.

procedure Try (column: integer);
  var
    row: integer;
    rowFlag,plusFlag,minusFlag,: Boolean;
  begin
    for row := 1 to 8 do
      begin
        rowFlag := gRowe[row];
        plusFlag := gColPlusRow[column + row];
        minusFlag := gColMinusRow[column - row];

        if rowFlag and plusFlag and 
                          minusFlag then
          begin
            gSolutionStash[column] := row;
            SetQueen(row, column);
            if column < 8 then
              Try(column + 1)
            else
              PrintSolution;

            RemoveQueen(row, column);
          end;  {if k}
      end;    {for}
  end;    {Try}

Listing 1 shows the complete Try() procedure for a simplified, non-interactive version of the recursive solution. The for loop repeatedly tests the Boolean arrays at indices consisting of the row, sum of row and column, and difference between row and column. The initial row and column values are both 1. If these three Boolean tests are TRUE, a queen is placed in the current column and row positions; if the column value is 8, we have a queen in each column and print the solution. After printing, the queen in column 8 is removed and the row increments at the top of the loop in pursuit of the next solution.

Otherwise when column < 8, the Try() procedure calls itself with an incremented column value and position testing starts anew at row 1. If any of three Boolean array locations are FALSE, the for loop row index increments and the tests repeated. If row exceeds a value of 8 (that is, the active queen drops off the board), execution continues after the Try(column+1) line in the previous iteration of the procedure unless the column value equals 1 when row exceeds 8. (In that case, all solutions have been found and operation passes back to whatever called Try initially.) The queen at the row position for the current column is removed, the row increments at the top of the loop, etc.

Listing 2a. SetQueen

SetQueen

procedure SetQueen (row, column: integer);
  begin
    gRowe[row] := false;
    gColPlusRow[column + row] := false;
    gColMinusRow[column - row] := false;
  end;

Listing 2b. RemoveQueen

RemoveQueen

procedure RemoveQueen (row, column: integer);
  begin
    gRowe[row] := true;
    gColPlusRow[column + row] := true;
    gColMinusRow[column - row] := true;
  end;

Listing 2 shows the SetQueen and RemoveQueen procedures that update the Boolean arrays.

Listing 3a. Try

Try

procedure Try;
  var
    row, column: integer;
    rowFlag,plusFlag,minusFlag: Boolean;
  begin
    row := 1;
    column := 1;
    repeat
      repeat
        rowFlag := gRowe[row];
        plusFlag := gColPlusRow[column + row];
        minusFlag := gColMinusRow[column - row];
        if rowFlag and plusFlag and minusFlag then
          begin
            gSolutionStash[column] := row;
            SetQueen(row, column);
            if column < 8 then
              begin
                gRowForCol[column] := row;
                row := 1;
                Inc(column);
                Leave;
              end
            else
              begin
                PrintSolution;
                RemoveQueen(row, column);
                Inc(row);
              end;
          end
        else
          Inc(row);
      until row > 8;

      if row > 8 then
        BakTrak(row, column);

  until column < 1;
end;    {procedure Try}

Listing 3b. BakTrak

BakTrak

procedure BakTrak (var row, column: integer);
  begin
    repeat
      Dec(column);
      if column > 0 then
        begin
          row := gRowForCol[column];
          RemoveQueen(row, column);
          Inc(row);
        end;
    until (row < 9) or (column < 1);
  end;

Listing 3 gives the Try and BakTrak procedures for a non-recursive solution. The biggest differences from Listing 1 are the two nested repeat loops the global array gRowForCol, which holds the row number of the queen in each column.

These listings generate only the row number for the queen in each column and do not show the event-handling calls or other fancifications needed to implement an interactive program with variable execution speeds, single-step mode, etc. For that, see Listing 4 and the full source code files.

Listing 4. DoColumns

DoColumns

I call this procedure DoColumns instead of Try in order to distinguish it from the corresponding procedure in my "sideways" solution, which is called DoRows.

procedure DoColumns (column: Integer);
  var
    row,n,rotNum,rotSize: integer;
    rowFlag,plusFlag,minusFlag,mirFlag,whiteDiagFlag,
      redDiagFlag,topBotFlipFlag,leftRtFlipFlag,
      rotFlag: boolean;
    elapsedTime,currentTime: longint;
    bn : byte;
  begin
    ggCol   := column;
    for row := 1 to 8 do
      begin
          {update active queen position unless in ultra-fast}
        if not ggUfastFlag then
          begin
            DrawQueen(row, column);
            if not ggVfastFlag then
              UpDateBoolean;
          end;
          
          {test active queen for conflicts}
        rowFlag     := ggRowFlags[row];
        plusFlag   := ggColPlusRow[column + row];
        minusFlag   := ggColMinusRow[Column-Row];
        Inc(ggTests);
        UpdateTestCount;

            { put this here to update boolean display before halting }
            { in step mode -- if no conflict }
        if rowFlag and plusFlag and minusFlag then
          begin
            SetQueen(row, column);
            UpDateBoolean;
          end;

        if not ggUfastFlag then
          gBoardNotClear := true;  
                  {flag used in ultra-fast mode to clear}
                  {board completely 1st solution after}
                  {ultra-fast mode is selected; only   }
                  {those queens that have changed pos-}
                  {ition are erased subsequently  }

        if ggStepModeFlag = true then
          ggStepFlag := true
        else
          ggStepFlag := false;

        if (not ggVfastFlag) and (not ggUfastFlag)
                            and (not ggStepModeFlag) then
              {delay according to speed and mode}
          Stall(ggStallVal);  
          
          {handle events except in very-fast or ultra-fast modes}
        if ((not ggUfastFlag) and (not ggVfastFlag))
                                or ggStepModeFlag then
          begin
            repeat
              HandleEvent;
            until (not ggStepFlag) or ggDoneFlag;
          end;
        if not ggStartClockFlag then
          begin
            ggStartClockFlag := true;
            GetDateTime(ggStartTime);
            if not gRunFlag then
              begin
                GetDateTime(ggStartTotalTime);
                gRunFlag := true;
              end;
          end;

        if rowFlag and plusFlag and minusFlag then
          begin
              {active queen position is ok, so save row}
            ggRowStash[ggSolNum, column] := row;
              {no solution yet; do next column}
            if column < 8 then
              begin
                Inc(column);
                ggCol := column;
                  {procedure calls itself}
                DoColumns(column);

                if not ggUfastFlag then
                  Stall(ggStallVal);
                Dec(column);
                  {ggCol is used in event-triggered board redraws}
                ggCol := column;  

                Inc(ggBakTrak);
                UpDateBakTrax;
                if ggStepModeFlag = true then
                  begin
                    ggStepFlag := true;
                    while ggStepFlag = true do
                      HandleEvent;
                  end
                else if not ggUfastFlag then
                  HandleEvent;

                if column > 0 then
                  begin
                    if not ggUfastFlag then
                      SnuffQueen(row, column);
                    RemoveQueen(row, column);
                  end;
              end
            else
              begin
                {we have a conflict-free queen in each column}
                {In ultra - fast mode we need to update the}
                {board and statistics}

                if ggUfastFlag then
                  begin
                    DoUfastUpdate;
                    UpDateBoolean;
                    UpdateTestCount;
                    UpdateBakTrax;
                    GetDateTime(currentTime);
                    elapsedTime := currentTime - 
                                        ggStartTime;
                    DrawElapsed(elapsedTime);
                  end;
                  
                  {get ready to test for unique solution}
                InitForMirrors(rotFlag, whiteDiagFlag, 
                  redDiagFlag,topBotFlipFlag,leftRtFlipFlag,
                   mirFlag, gMirrorNum);
                    {freeze time counter}
                ggStartClockFlag := false;
                    { no board redraws}     
                ggShowFlag := true;
                TestForMirrors(gMirrorNum,rotNum,rotSize,
                  mirFlag,whiteDiagFlag,redDiagFlag, 
                  topBotFlipFlag, leftRtFlipFlag, rotFlag);
                ggShowFlag := false;
                if (not mirFlag) and (not rotFlag) then
                  begin
                    ggUniqueSols[ggSolNum] := gUniqueSolNum;
                    Inc(gUniqueSolNum);
                  end;

                DrawSolStatus(mirFlag,whiteDiagFlag,
                  redDiagFlag,topBotFlipFlag,leftRtFlipFlag,
                  rotFlag,gMirrorNum,gUniqueSolNum,rotNum,
                  rotSize);

                ggSolFlag := TRUE;    {avoid board redraw}
                if not ggWaitFlag then
                  WaitDelaySecs
                else
                  begin    
                        {wait for run or step button push or quit}
                        {set up and call event handler}
                    ShowControl(ggStepButtonHdl);
                    HiliteControl(ggStepButtonHdl, 0);
                    ShowControl(ggRunButtonHdl);
                    HiliteControl(ggRunButtonHdl, 0);
                    ShowControl(ggStepButtonHdl);
                    HiliteControl(ggStepButtonHdl, 0);
                    SetControlTitle(ggRunButtonHdl, 'Run');
                    ggStepModeFlag := true;
                    ggStepFlag := true;
                    repeat
                      HandleEvent;
                    until (not ggStepModeFlag) or
                          (not ggStepFlag) or ggDoneFlag;

              {STEP button pushed in fast modes plus wait}
              { so drop to medium speed, step mode}
                    if (ggUfastFlag or ggVfastFlag) and
                                  (ggStepModeFlag and
                                  (not ggStepFlag)) then
                      begin
                        ggUfastFlag:= false;
                        ggVfastFlag:= false;
                        ggSpeedMenuHdl:=
                          GetMenuHandle(ggcSPEED_MENU_ID);
                        CheckItem(ggSpeedMenuHdl,
                                  ggOldSpeed,
                                  ggcREMOVE_CHECK_MARK);
                        CheckItem(ggSpeedMenuHdl,
                                  ggcMEDIUM_ITEM,
                                  ggcADD_CHECK_MARK);
                        ggOldSpeed := ggcMEDIUM_ITEM;
                        ggStallVal := gcMedium;
                      end;
                  end;

                    {init the next solution}
                for n := 1 to 8 do
                  begin
                    bn := ggRowStash[ggSolNum, n];
                    ggRowStash[ggSolNum + 1, n] := bn;
                  end;

                if not ggUfastFlag then
                  SnuffQueen(row, column);
                Inc(ggSolNum);
                DrawSolNum;
                GetDateTime(ggStartTime);
                ggStartClockFlag := true;  {start the clock}
                elapsedTime := 0;
                DrawElapsed(elapsedTime);
                RemoveQueen(row, column);
                EraseSolStat;
                gTotalTests:=gTotalTests+ggTests;
                gTotalMoves:=gTotalMoves+ggTests+ggBakTrak;
                gTotalTestsSol:=gTotalTestsSol+ggTests;
                gMovesForSol:=gMovesForSol+ggBakTrak+
                                        ggTests;
                EraseTestCount;
                ggTests := 0;
                UpdateTestCount;
                EraseBakTrax;
                ggBakTrak := 0;
                UpDateBakTrax;
                EraseTime;
                  {allow board redraws}
                ggSolFlag := FALSE;  
              end;
          end
        else
          begin
            if not ggUfastFlag then
              SnuffQueen(row, Column);
          end;

        if ggDoneFlag then
          Leave;
      end; {for}

  end;    {procedure DoColumns}

Listing 4 shows the recursive version of the Eight Queens program's main loop all dressed up for the event-driven Macintosh party with speed and mode variations.

Running the Program

The Macintosh programs EightQueens I and EightQueens II have two operating modes -- run and single-step. At startup, the chessboard is drawn and a startup window is displayed for 30 seconds or until the Go button is pushed. To the right of the chess board is an area giving the following information:

  • The time to achieve each solution.
  • The solution number 1-92.
  • The solution status -- unique or variation on a prior solution.
  • The number of position tests required to achieve each solution.
  • The number of backtracks.
  • The values of the elements of the Row or Column, Column plus Row, and Column minus Row Boolean arrays used to determine the conflict status of the queens.

All but the 2nd and 3rd are updated continuously as each solution progresses.

At bottom right are the Run and Step buttons. In single-step mode, pushing the Step button single-steps the currently-active queen. Pushing the Run button causes. Push the Step button to re-select single-step mode.

The program operates initially in single-step mode in which the active queen steps when the Step button is pressed. If the Run button is pressed, run mode is entered; the Step button disappears, the Run button is re-labeled Step, and the active queen steps continuously until a solution is achieved. It then delays (default delay is 10 seconds) before stepping to the next solution unless Wait is selected from the Delay menu. Both the step rate and the duration of the delay can be varied via the Speed and Delay menu. Speeds vary from about 1 step per second at the slow end to hundreds per second. Default speed is about 4 steps per second on a PowerPC Mac 6500/225. The delay between solutions can be varied from none to 30 seconds in 5-second increments, or Wait can be selected and the program will enter single-step mode after each solution. The Step button changes to Next, and the program waits for a click on the Run or Next button.

To determine whether a solution is unique or a variation on one of the twelve unique solutions, the program creates seven variations of each solution to compare with the previous ones. The variations are -- left-to-right flip (vertical axis mirror), top-to-bottom flip (horizontal axis mirror), upper-left-to-lower-right (red) diagonal mirror, lower-left-to-upper-right(white) diagonal mirror, and 90 degree, 180 degree and 270 degree clockwise rotations. To view these in sequence, click the Next button. The variations will continue to be displayed in sequence as long as the Next button is clicked.

If you click the Run button, the Next button is re-labeled Step, and the solution status line tells whether the solution is unique or variant. When the status line says, for example, at solution #12, Rotate 90 deg CW #10, it means "rotating solution #12 90 degrees clockwise gets solution #10"; at solution #21 "Left-Right Flip #11" means that, if solution #21 is flipped left-to-right, we get solution #11. At #13 , "Red diag mir #8" means that if solution #13 is flipped on the red (upper-left-to-lower-right) diagonal axis, we get solution #8. Solution #16 is a white (lower-left-to-upper-right) diagonal axis mirror of solution #8. Solution #75 is a top-to-bottom flip of #18, etc.

Clicking the Run button again steps the active queen continuously at the previous speed to the next solution, while clicking Step single-steps the active queen and sets the speed to Medium. When the program ceases pursuing solutions either because all possibilities have been exhausted or because Quit has been selected from the File menu (or Command-Q from the keyboard), the area to the right of the board clears and displays statistics on the number of solutions achieved, number of position tests and backtracks, etc. The right button appears, labeled Stop, while the left button is labeled Run. The user can then choose to either resume seeking solutions starting at the beginning by clicking Run, or terminate operation by clicking Stop.

Sourcecode

Sourcecode for two variations of Wirth's algorithm for solving the eight queens problem is supplied for CodeWarrior Professional Pascal. Those readers familiar with Dave Mark's books may notice some resemblances between the eight queens sourcecode and some of that found in Dave's books -- things like some of the names of constants and general structure of the event loop. The resemblance isn't accidental. I used the Timer project from the Macintosh Pascal Programming Primer, Vol. 1, by Dave Mark and Cartwright Reed, as a "skeleton" for EightQueens. While most of the code is mine, underneath there's still a bit of Mark and Reed code holding things together.

Variations

Wirth's recursive algorithm used in EightQueens I indexes rows in the single for loop and columns via recursion, but the method works equally well if the rows and columns are switched. The movement of the queens is then from left-to-right, starting at the top row. We get the same 92 solutions, but in a different order. The first solution with horizontal queen movement is the same as the fourth with vertical movement. Each solution, however, takes the same number of tests and backtracks as with Wirth's algorithm -- 876 tests and 105 backtracks for the first solution, 264 tests and 33 backtracks for the second, 200 tests and 25 backtracks for the third, etc. The reason becomes obvious if you think about it. Take the board set for vertical queen movement, with a single queen upper left. Flip the board on the vertical axis so the queen is in the upper right corner, then rotate it 90 degrees counter-clockwise to put the queen upper left. Start successive queens in column one, incrementing left-to-right. Test queens have exactly the same positions relative to the first queen as in Wirth's original approach. EightQueens II shows this "sideways" approach implemented non-recursively using two nested repeat loops.

Bibliography and References

  • Wirth, Niklaus, Algorithms + Data Structures = Programs, (Englewood Cliffs NJ: Prentice-Hall, 1976).
  • Wirth, Niklaus, Algorithm s and Data Structures, (Englewood Cliffs NJ: Prentice-Hall, 1986)
  • Budd, Timothy, A Little Smalltalk, (Reading, MA: Addison-Wesley, 1987).
  • Budd, Timothy, An Introduction to Object-Oriented Programming, (Reading, MA: Addison-Wesley, 1991).
  • Budd, Timothy, An Introduction to Object-Oriented Programming, 2nd Edition, (Reading, MA: Addison-Wesley, 1996).

F.C. Kuechmann, fk@aone.com, is a hardware designer, programmer and consultant with degrees from the University of Illinois at Chicago and Clark College who is currently building a programmers' clock that gives the time in hexadecimal.

 

Community Search:
MacTech Search:

Software Updates via MacUpdate

OmniGraffle 6.3 - Create diagrams, flow...
OmniGraffle helps you draw beautiful diagrams, family trees, flow charts, org charts, layouts, and (mathematically speaking) any other directed or non-directed graphs. We've had people use Graffle to... Read more
PDFKey Pro 4.3.2 - Edit and print passwo...
PDFKey Pro can unlock PDF documents protected for printing and copying when you've forgotten your password. It can now also protect your PDF files with a password to prevent unauthorized access and/... Read more
Ableton Live 9.2.2 - Record music using...
Ableton Live lets you create and record music on your Mac. Use digital instruments, pre-recorded sounds, and sampled loops to arrange, produce, and perform your music like never before. Ableton Live... Read more
Macs Fan Control 1.3.1.0 - Monitor and c...
Macs Fan Control allows you to monitor and control almost any aspect of your computer's fans, with support for controlling fan speed, temperature sensors pane, menu-bar icon, and autostart with... Read more
NetShade 6.3.1 - Browse privately using...
NetShade is an anonymous proxy and VPN app+service for Mac. Unblock your Internet through NetShade's high-speed proxy and VPN servers spanning seven countries. NetShade masks your IP address as you... Read more
Dragon Dictate 4.0.7 - Premium voice-rec...
With Dragon Dictate speech recognition software, you can use your voice to create and edit text or interact with your favorite Mac applications. Far more than just speech-to-text, Dragon Dictate lets... Read more
Persecond 1.0.2 - Timelapse video made e...
Persecond is the easy, fun way to create a beautiful timelapse video. Import an image sequence from any camera, trim the length of your video, adjust the speed and playback direction, and you’re done... Read more
GIMP 2.8.14p2 - Powerful, free image edi...
GIMP is a multi-platform photo manipulation tool. GIMP is an acronym for GNU Image Manipulation Program. The GIMP is suitable for a variety of image manipulation tasks, including photo retouching,... Read more
Sandvox 2.10.2 - Easily build eye-catchi...
Sandvox is for Mac users who want to create a professional looking website quickly and easily. With Sandvox, you don't need to be a Web genius to build a stylish, feature-rich, standards-compliant... Read more
LibreOffice 5.0.1.2 - Free, open-source...
LibreOffice is an office suite (word processor, spreadsheet, presentations, drawing tool) compatible with other major office suites. The Document Foundation is coordinating development and... Read more

ReBoard: Revolutionary Keyboard (Utilit...
ReBoard: Revolutionary Keyboard 1.0 Device: iOS Universal Category: Utilities Price: $1.99, Version: 1.0 (iTunes) Description: Do everything within the keyboard without switching apps! If you are in WhatsApp, how do you schedule a... | Read more »
Tiny Empire (Games)
Tiny Empire 1.1.3 Device: iOS Universal Category: Games Price: $2.99, Version: 1.1.3 (iTunes) Description: Launch cannonballs and blow tiny orcs into thousands of pieces in this intuitive fantasy-themed puzzle shooter! Embark on an... | Read more »
Astropad Mini (Productivity)
Astropad Mini 1.0 Device: iOS iPhone Category: Productivity Price: $4.99, Version: 1.0 (iTunes) Description: *** 50% off introductory price! ​*** Get the high-end experience of a Wacom tablet at a fraction of the price with Astropad... | Read more »
Emo Chorus (Music)
Emo Chorus 1.0.0 Device: iOS Universal Category: Music Price: $1.99, Version: 1.0.0 (iTunes) Description: Realistic Choir simulator ranging from simple Chorus emulation to full ensemble Choir with 128 members. ### introductory offer... | Read more »
Forest Spirit (Games)
Forest Spirit 1.0.5 Device: iOS Universal Category: Games Price: $2.99, Version: 1.0.5 (iTunes) Description: | Read more »
Ski Safari 2 (Games)
Ski Safari 2 1.0 Device: iOS Universal Category: Games Price: $1.99, Version: 1.0 (iTunes) Description: The world's most fantastical, fun, family-friendly skiing game is back and better than ever! Play as Sven's sister Evana, share... | Read more »
Lara Croft GO (Games)
Lara Croft GO 1.0.47768 Device: iOS Universal Category: Games Price: $4.99, Version: 1.0.47768 (iTunes) Description: Lara Croft GO is a turn based puzzle-adventure set in a long-forgotten world. Explore the ruins of an ancient... | Read more »
Whispering Willows (Games)
Whispering Willows 1.23 Device: iOS Universal Category: Games Price: $4.99, Version: 1.23 (iTunes) Description: **LAUNCH SALE 50% OFF** - Whispering Willows is on sale for 50% off ($4.99) until September 9th. | Read more »
Calvino Noir (Games)
Calvino Noir 1.1 Device: iOS iPhone Category: Games Price: $3.99, Version: 1.1 (iTunes) Description: The film noir stealth game. Calvino Noir is the exploratory, sneaking adventure through the 1930s European criminal underworld.... | Read more »
Angel Sword (Games)
Angel Sword 1.0 Device: iOS Universal Category: Games Price: $6.99, Version: 1.0 (iTunes) Description: Prepare to adventure in the most epic full scale multiplayer 3D RPG for mobile! Experience amazing detailed graphics in full HD.... | Read more »

Price Scanner via MacPrices.net

Apple offering refurbished 2015 13-inch Retin...
The Apple Store is offering Apple Certified Refurbished 2015 13″ Retina MacBook Pros for up to $270 (15%) off the cost of new models. An Apple one-year warranty is included with each model, and... Read more
Apple refurbished 2015 MacBook Airs available...
The Apple Store has Apple Certified Refurbished 2015 11″ and 13″ MacBook Airs (the latest models), available for up to $180 off the cost of new models. An Apple one-year warranty is included with... Read more
21-inch iMacs on sale for up to $120 off MSRP
B&H Photo has 21″ iMacs on sale for up to $120 off MSRP including free shipping plus NY sales tax only: - 21″ 1.4GHz iMac: $999.99 $100 off - 21″ 2.7GHz iMac: $1199.99 $100 off - 21″ 2.9GHz iMac... Read more
5K iMacs on sale for up to $150 off MSRP, fre...
B&H Photo has the 27″ 3.3GHz 5K iMac on sale for $1899.99 including free shipping plus NY tax only. Their price is $100 off MSRP. They have the 27″ 3.5GHz 5K iMac on sale for $2149.99 $2199.99, $... Read more
1.4GHz Mac mini, refurbished, available for $...
The Apple Store has Apple Certified Refurbished 1.4GHz Mac minis available for $419. Apple’s one-year warranty is included, and shipping is free. Their price is $80 off MSRP, and it’s the lowest... Read more
iPad Air 2 on sale for up to $100 off MSRP
Best Buy has iPad Air 2s on sale for up to $100 off MSRP on their online store for a limited time. Choose free shipping or free local store pickup (if available). Sale prices available for online... Read more
MacBook Airs on sale for $100 off MSRP
Best Buy has MacBook Airs on sale for $100 off MSRP on their online store. Choose free shipping or free local store pickup (if available). Sale prices for online orders only, in-store prices may vary... Read more
Big Grips Lift Handle For iPad Air and iPad A...
KEM Ventures, Inc. which pioneered the extra-large, super-protective iPad case market with the introduction of Big Grips Frame and Stand in 2011, is launching Big Grips Lift featuring a new super-... Read more
Samsung Launches Galaxy Tab S2, Its Most Powe...
Samsung Electronics America, Inc. has announced the U.S. release of the Galaxy Tab S2, its thinnest, lightest, ultra-fast tablet. Blending form and function, elegant design and multitasking power,... Read more
Tablet Screen Sizes Expanding as iPad Pro App...
Larger screen sizes are gaining favor as the tablet transforms into a productivity device, with shipments growing 185 percent year-over-year in 2015. According to a new Strategy Analytics’ Tablet... Read more

Jobs Board

*Apple* Retail - Multiple Positions (US) - A...
Sales Specialist - Retail Customer Service and Sales Transform Apple Store visitors into loyal Apple customers. When customers enter the store, you're also the Read more
*Apple* Desktop Analyst - KDS Staffing (Unit...
…field and consistent professional recruiting achievement. Job Description: Title: Apple Desktop AnalystPosition Type: Full-time PermanentLocation: White Plains, NYHot Read more
*Apple* Systems Engineer (Mclean, VA and NYC...
Title: Apple Systems Engineer (Mclean, VA and NYC) Location: United States-New York-New York-200 Park Ave (22005) Other Locations: United States-Virginia-Vienna-Towers Read more
*Apple* Systems Engineer (Mclean, VA and NYC...
…Assist in providing strategic direction and technical leadership within the Apple portfolio, including desktops, laptops, and printing environment. This person will Read more
*Apple* Subject Matter Expert - Experis (Uni...
We are seeking an Apple Subject Matter Expert to assist in developing the architecture, support and services for integration of Apple devices into the company's Read more
All contents are Copyright 1984-2011 by Xplain Corporation. All rights reserved. Theme designed by Icreon.