TweetFollow Us on Twitter

Animation Contest
Volume Number:1
Issue Number:5
Column Tag:Mousehole

Animation Contest

By David E. Smith

Last month we presented a complete application shell program in assembly that supported desk accessories, cut and past and “about” dialog boxes. But the application itself was rather boring, simply drawing nested boxes in the window. This month, we concentrate on the application itself. To eliminate confusion, we will use the simple shell program from the December issue of MacTutor that simply opens a window. This allows us to concentrate on the application, which is to animate a paddle and ball. Once we understand animation, we can combine our paddle and ball program with the shell program published last month to create a complete Mac game application. But first, the problem of animation.

Animation Contest

The Mousehole recently held a programming contest sponsored by MacTutor on the problem of moving a paddle with the mouse. “Brett” offered an interesting solution: “Have the user blink a lot and pretend!” Another interesting solution offered by “Lone Falcon” and “Chief Wizard” and expanded on by “The Jerk” was to make the cursor a paddle and constrain the cursor to a given rectangle. The trap call “_PinRect” can be used to change the rectangle in which the cursor is allowed to move. Or the rectangle can be poked into the global variable at $834 as mentioned by “The Jerk”. One problem with this solution is that of providing an “escape” for the cursor when you want to go to the menu bar to quit the game!

A simple technique to move the paddle is to draw the paddle rectange where the mouse now is and erase it from where the paddle used to be. The following example code shows how this might be done:

MOVEPADDLE:

; Copy current paddle to old paddle

LEApaddle, A0
LEAoldpaddle, A1
MOVE.W  0(A0), 0(A1) ;top of rect
MOVE.W  2(A0), 2(A1) ;left of rect
MOVE.W  4(A0), 4(A1) ;bottom 
MOVE.W  6(A0), 6(A1) ;right

; Update paddle position from mouse

; _GetMouse trap previously called
; and mouse coord. saved in Mouse.
; Since paddle moves vertically, only
; the top and bottom coordinates need
; be updated.

LEAMouse, A0;get mouse
LEApaddle, A1  ;get paddle
MOVE.W  (A0), (A1) ;update top
MOVE.W  (A0), D0 ;get top
ADD.W Paddlelength, D0  ;add length
MOVE.W  D0, 4(A1);update bottom

; Erase old paddle position

PEAoldpaddle;push old pad.
_EraseRect;erase it

; Draw new paddle

PEApaddle ;push paddle
PEAPaddlepattern ;push fill pat.
_FillRect ;draw paddle
RTS

The Paddle Blinks!

The problem with the above solution, and hence the contest, is that the paddle will blink as it moves up and down. The problem is to improve on the above technique to eliminate the blinking. The reason for the blinking is shown in figure 1.

As shown in figure 1, the paddle blinks because we are erasing and drawing part of the paddle that overlaps, causing it to blink on and off. The solution is to determine the non-overlapping part and erase only that part. One way to do this is to replace the bottom of the old paddle with the top of the new paddle. Then the modified old paddle rectange would be only the cross-hatched area shown in figure 1 as the non-over-lapping part. But that only works for the paddle moving down. If the paddle moves up, then the opposite condition applies as shown in the following code segment:

ERASEPADDLE:

; create difference rectange

LEApaddle, A0  ;new paddle
LEAoldpaddle, A1 ;old paddle
MOVE.W  (A0), D0 ;get new paddle
MOVE.W  (A1), D1 ;get old paddle
SUB.W D1, D0;top difference
BMIUP ;moving up?

DN: MOVE.W (A0), 4(A1)  ;update bottom
     JMP Eraseit

UP: MOVE.W 4(A0), (A1)  ;update top

Eraseit:  PEA oldpaddle
 _EraseRect

CONTEST WINNER!

The above method works fine for a rectangular paddle moving up or down. But what about the ball? The winning solution to our animation contest was posted by “Don L.” in which he explained how regions could be used in a manner similar to what was done above for rectangles. Moreover, the toolbox has a special trap for finding the difference of two regions but does not have a similar trap for rectangles. The solution then is to form a union region of the old and new paddle regions, and then take the difference between the new paddle and this union region. The difference region is then erased and the paddle re-drawn. The result is a non-blinking paddle but using regions instead of rectangles.

In our example program we use “Don L.”’s method to move the ball as a region, and the rectangular method for moving the less complicated paddle rectangle. See the subroutine MOVEPADDLE for the rectangular implmentation discussed above and the subroutine MOVEBALL for the region solution submitted by Don. The tricky part is getting the regions properly defined with NewRgn, OpenRgn and CloseRgn.

Detecting Collisions

Once we have a moving paddle responding to the mouse, and a ball moving towards us, we need to detect if the ball has hit the paddle. A simple way to do this is to use the trap _RectInRgn, which tells if the given rectange intersects with the given region. We have used this trap by pushing the paddle rectange as the given rectange, and the ball region as the given region and tested the returning byte for true or false. If true, then the ball has “hit” the paddle, and we erase the ball and start a new one. If not, then the paddle has “missed” the ball and the game ends on a click of the mouse. Of course, in the final game, we really want the ball to “bounce” and move away from the paddle if there is a hit. We will leave the problem of bouncing for next month.

ANIMATE EXAMPLE

Our sample program includes all of the topics we’ve discussed this month in a complete program. To conserve space, none of the fancy user interface support we did last month is included. The program opens a window and displays our game field with a rectangular paddle, and launches a ball. If the user hits the ball, it disappears and another ball is launched. If the user misses the ball, the game stops until the button is pressed, at which point the finder is invoked. This is the basic animation required for a pinball type of game. Make note of the INITBALL routine that sets up the three temporary regions for use with the ball, and the MOVEBALL routine which erases the ball by only erasing the non-over-lapping region. Using these techniques, the paddle and ball should move smoothly over the shaded playing field.

BALL SPEED

The speed of the ball is determined by the increment used to update the ball region. The trap “_OffSetRgn” is used to incrementaly move the ball a small distance (dh, dv). In our example, dh is set at -1 and dv is zero, making the ball move horizontally one pixel left at a time. For a faster moving ball, the horiztonal increment should be doubled in a loop, until the user misses. Try this and see how fast the ball appears to move. Note that moving the paddle very quickly does produce an effect on the speed of the ball slightly.

In figure 2 we see the output from this month’s program. The white ball moves on a field of gray towards the black paddle. The white area was added in MacPaint to reduce the amount of gray, which does not print well.

Resources and Linker Files

The resource and linker files included are the minimum necessary to get an icon to pop up on the desktop. Our icon is shown at the top of the column header, and was produced using the Icon Converter Utility from issue 2 and “included” into our resource source code.

; EXAMPLE ASSEMBLY PROGRAM 
; ANIMATE (MacTutor 1-5)
; VERSION 27 FEB 1985
; (C) 1985 MacTutor by David E. Smith

;    Macro subset for Toolbox stuff

MACRO _InitGraf =  DC.W $A86E|
MACRO _InitWind =DC.W $A912| 
MACRO _NewWindow = DC.W $A913|
MACRO _setport = DC.W $A873|
MACRO _InitFont =DC.W $A8FE|
MACRO _InitMenu =DC.W $A930|
MACRO _InitDialog =DC.W $A97B|
MACRO _TEInit   =DC.W $A9CC|
MACRO _Initpack =         DC.W $A9E5|
MACRO _FlushEvents = DC.W $A032|
MACRO _InitCursor =DC.W $A850|
MACRO _GetNextEvent =DC.W $A970|
MACRO _FrameRect = DC.W $A8A1|
MACRO _BackPat = DC.W $A87C|
MACRO _EraseRect = DC.W $A8A3|
MACRO _NewRgn =  DC.W $A8D8|
MACRO _OpenRgn = DC.W $A8DA|
MACRO _CloseRgn =DC.W $A8DB|
MACRO _FrameRgn =  DC.W $A8D2|
MACRO _FrameRoundRect =   DC.W $A8B0|
MACRO _FillRect =DC.W $A8A5|
MACRO _FillRgn = DC.W $A8D6|
MACRO _BeginUpdate = DC.W $A922|
MACRO _EndUpdate = DC.W $A923|
MACRO _ClipRect =  DC.W $A87B|
MACRO _GetMouse =  DC.W $A972|
MACRO _CopyRgn = DC.W $A8DC|
MACRO _OffsetRgn = DC.W $A8E0|
MACRO _OffsetRect =  DC.W $A8A8|
MACRO _UnionRgn =  DC.W $A8E5|
MACRO _DiffRgn = DC.W $A8E6|
MACRO _MoveTo =  DC.W $A893|
MACRO _DrawChar =  DC.W $A883|
MACRO _RectInRgn = DC.W $A8E9|

;      DECLARE LABELS EXTERNAL

XDEF  START           ; required for linker 
 
;     LOCAL EQUATES
 
MouseDown equ  1
AllEvents equ  $0000FFFF
UpdateEvt equ  6 

; SET UP MANAGERS

START:

PEA -4(A5);push qd global ptr
_InitGraf        ;init quickdraw global 
_InitFont        ; init font manager
_InitWind   ; init window manager
_InitMenu   ; init menu manager  
CLR.L -(SP) ; kill the restart 
_InitDialog ; init dialog manager
_TEInit ; init text edit (ROM) 
MOVE.W  #2,-(SP) ;  set-up
_Initpack ; init package mgr
MOVE.L  #AllEvents,D0    ;all events
_FlushEvents      ;flushed
_InitCursor      ; make cursor the arrow

;-- SET UP NEW WINDOW ON HEAP ----
 
CLR.L -(SP)          ;return window ptr
CLR.L -(SP)          ;window record ptr.
PEA WBOUNDS               ;window rectangle 
PEA WINDTITLE              ; window title
MOVE.W #$100,-(SP)   ; true = visible 
MOVE.W #0,-(SP)           ; doc type window
MOVE.L #-1,-(SP)          ; window in front
MOVE.W #$100,-(SP)   ; true=closebox
MOVE.L #0, -(SP)         ; reference value 
_NewWindow                ; make new window
 
; --  ACTIVATE THIS NEW WINDOW ------

LEA WPOINTER,A0         ; copy window ptr 
MOVE.L (SP),(A0)        ; to stacksave
_setport                  ;current window

;   INIT ALL VARIABLES

JSRINITVAR

; --EVENT LOOP ------------

GetEvent:

JSRDOGAME ;move paddle with mouse

CLR-(SP);returned event 
MOVE  #AllEvents,-(SP)    ;mask all events
PEAEventRecord   ; event record block
_GetNextEvent    ;go check the mouse 
MOVE  (SP)+,D0   ;get event result 
CMP#0,D0;if 0 then no event 
BEQGetEvent ;loop until it happens

 ; JUMP TABLE OF EVENT PROCESSING
 
MOVE  What,D0    ;what to do!
CMP#MouseDown,D0     ; button down?
BEQEXIT ;yes so exit...
CMP#UpdateEvt, D0;is it an update event?
BEQUPDATE ;yes, so do it
JMP    GetEvent  ;get next event

UPDATE:

; window needs refreshing

MOVE.L  WPOINTER,-(SP)  ;push window ptr
_BeginUpdate
MOVE.L  WPOINTER,-(SP)  ;push our window ptr.
_SetPort;restore our port
 ;least an update draw in
 ;the wrong window.
BSRQDSTUFF;re-draw everything
MOVE.L  WPOINTER,-(SP)
_EndUpdate
JMP   GetEvent

; ---------- END OF MAIN ----------
INITVAR:

;     INIT GAME VARIABLES 

LEAField.of.play(A5),A0 ;set-up appl. window 
MOVE.W #5,    (A0) ;Field.of.play TOP
MOVE.W #5,   2(A0) ;LEFT
MOVE.W #245, 4(A0) ;BOTTOM
MOVE.W #500, 6(A0) ;RIGHT

PEA Field.of.play(A5)
_ClipRect ;set clip region

LEAPaddle(A5),A0 ;set-up paddle
MOVE.W #6,    (A0) ;TOP
MOVE.W #8,   2(A0) ;LEFT
MOVE.W #40, 4(A0);BOTTOM
MOVE.W #15, 6(A0);RIGHT

LEAPaddlelength(A5),A1  ;update paddle bottom
MOVE.W  (A0),D0
MOVE.W  4(A0),D1
SUB.W   D0,D1
MOVE.W  D1, (A1)

;  set up regions for ball

CLR.L -(SP)
_NewRgn ;set new region
LEABallrgn(A5), A1 ;make ball a region
MOVE.L  (SP)+,(A1) ;save handle in ballrgn

CLR.L -(SP)
_NewRgn ;set new region
LEAoldballrgn(A5), A1;make old ball a region
MOVE.L  (SP)+,(A1) ;save handle in oldballrg 

CLR.L -(SP)
_NewRgn ;set new region
LEAunionrgn(A5), A1;make union ball a region
MOVE.L  (SP)+,(A1) ;save handle in unionrgn

JSRINITBALL ;set-up ball

RTS

; ------   init ball subroutine ------

INITBALL:

LEABall(A5),A0 ;set-up ball rect
MOVE.W #100,    (A0) ;TOP
MOVE.W #300,   2(A0) ;LEFT
MOVE.W #109, 4(A0) ;BOTTOM
MOVE.W #309, 6(A0) ;RIGHT

LEABallwide(A5),A0 ;set up rounded corners
MOVE.W #6, (A0)
LEABallhigh(A5), A0
MOVE.W #6, (A0)

_OpenRgn;init ball region
PEABall(A5)
MOVE.W  Ballwide(A5),-(SP)
MOVE.W  Ballhigh(A5),-(SP)
_FrameRoundRect
MOVE.L  Ballrgn(A5),A0
MOVE.L  A0,-(SP)
_CloseRgn

_OpenRgn;init oldball region
PEABall(A5)
MOVE.W  Ballwide(A5),-(SP)
MOVE.W  Ballhigh(A5),-(SP)
_FrameRoundRect
MOVE.L  oldballrgn(A5),A0
MOVE.L   A0,-(SP)
_CloseRgn

_OpenRgn;init union region
PEABall(A5)
MOVE.W  Ballwide(A5),-(SP)
MOVE.W  Ballhigh(A5),-(SP)
_FrameRoundRect
MOVE.L  unionrgn(A5),A0
MOVE.L  A0,-(SP)
_CloseRgn
RTS

DOGAME:

; update game field during null event

PEAmouse(A5)
_GetMouse 
LEAmouse(A5),A0  ;current mouse
LEAoldmouse(A5),A1 ;old mouse;find where mouse is
MOVE.W  (A0),D0  ;new mouse
MOVE.W  (A1),D1  ;old mouse
CMPD1,D0;has y mouse changed?
BEQstatck ;no, don’t update paddle
 ;yes, make new paddle
JSRMOVEPADDLE    ;update paddle position

statck:
JSRMOVEBALL ;advance ball
JSRBOUNCE ;change ball direction

RTS
; ----  MOVE PADDLE SUBROUTINE ----

MOVEPADDLE:

;      copy paddle position to oldpaddle

LEApaddle(A5), A0
LEAoldpaddle(A5), A1
MOVE.W  (A0), (A1) ;top
MOVE.W  2(A0), 2(A1) ;left
MOVE.W  4(A0), 4(A1) ;bottom
MOVE.W  6(A0), 6(A1) ;right

; update paddle position

LEAmouse(A5),A0  ;get current mouse position
LEApaddle(A5),A1 ;update paddle position
MOVE.W  (A0), (A1) ;update TOP
MOVE.W  (A0),D0  ;calculate BOTTOM
ADD.W Paddlelength(A5),D0
MOVE.W  D0,4(A1) ;update BOTTOM

;   draw new paddle

PEApaddle(A5)
PEAPaddlePat
_FillRect

;    calculate difference rectangle

LEApaddle(A5),A0 ;current paddle
LEAoldpaddle(A5),A1;old paddle
CLR.L   D0
CLR.L   D1
MOVE.W  (A0), D0
MOVE.W  (A1),D1
LEApaddle(A5), A0
LEAoldpaddle(A5), A1
SUB.W D1, D0
BMIup
dn:   MOVE.W(A0), 4(A1) ;moved down
      BRA wipeout
up: MOVE.W4(A0), (A1);moved up
    BRA wipeout
    
wipeout:
PEAoldpaddle(A5)
PEABackPat0
_FillRect

; update mouse

LEAmouse(A5),A0  ;current mouse
LEAoldmouse(A5),A1 ;old mouse
MOVE.L  (A0),(A1);update old mouse
RTS

; ----      MOVE BALL SUBROUTINE ------
MOVEBALL:

;      copy Ballrgn to oldballrgn

MOVE.L  Ballrgn(A5),A0
MOVE.L  A0,-(SP)
MOVE.L  oldballrgn(A5),A0
MOVE.L  A0,-(SP)
_CopyRgn

; move ball region

MOVE.L  Ballrgn(A5),A0
MOVE.L  A0,-(SP)
MOVE.W  dh,-(SP) ;horiz. increment
MOVE.W  dv, -(SP);vert. increment
_OffsetRgn;offset ball rgn

;   draw new BALL

MOVE.L  Ballrgn(A5),A0
MOVE.L  A0,-(SP)
PEABallPat
_FillRgn

; erase old ball difference

MOVE.L  Ballrgn(A5),A0
MOVE.L  oldballrgn(A5),A1
MOVE.L  unionrgn(A5),A3
MOVE.L  A0,-(SP)
MOVE.L  A1,-(SP)
MOVE.L  A3,-(SP)
_Unionrgn

MOVE.L  unionrgn(A5),A0
MOVE.L  Ballrgn(A5),A1
MOVE.L  unionrgn(A5),A3
MOVE.L  A0,-(SP)
MOVE.L  A1,-(SP)
MOVE.L  A3,-(SP)
_Diffrgn

MOVE.L  unionrgn(A5),A0
MOVE.L  A0,-(SP)
PEABackPat0
_FillRgn

RTS

; ---------- BOUNCE BALL --------
BOUNCE:

CLR.B -(SP)
PEAPaddle(A5)    ;push rect
MOVE.L  Ballrgn(A5), A0
MOVE.L  A0, -(SP);push rgn handle
_RectInRgn
MOVE.B  (SP)+, D0
CMP#0, D0 ;false?
BEQRETBOUNCE;yes...

MOVE.L  Ballrgn(A5),A0  ;no...
MOVE.L  A0,-(SP) ;erase ball
PEABackPat0
_FillRgn
JSR INITBALL;start new ball

RETBOUNCE:
RTS

; ------ QDSTUFF SUBROUTINE ----
QDSTUFF:

;       DRAW GAME FIELD

_PenNormal

PEABACKPAT0
_BackPat
PEA Field.of.play(A5)   ;frame rectangle
_EraseRect
PEA Field.of.play(A5)
_FrameRect;draw game rectangle

;      DRAW PADDLE

PEAPaddle(A5)
PEAPaddlePat
_FillRect
PEAPaddle(A5)
_FrameRect

RTS
; ---- RETURN TO FINDER --------

EXIT:        RTS      ; return to finder

; ----LOCAL DATA AREA ----------

WPOINTER: DC.L     0      ;store window pt
WBOUNDS:DC.W   40  ;rectangle top 
 DC.W    2;left
 DC.W    335;bottom
 DC.W    508;right
WINDTITLE:     DC.B    24        ; title length
                 DC.B    ‘BALL AND PADDLE EXAMPLE ‘,0      

EventRecord:
 
 What:  DC.W     0 ; what event
 Message: DC.L     0 ; ptr. to msg
 When:  DC.L     0
 Point:   DC.L     0
 Modify:  DC.W     0

BackPat0: DC.L $AA55AA55  ;playing field
 DC.L $AA55AA55

BallPat:DC.L$00000000
 DC.L $00000000
 
PaddlePat:DC.L $FFFFFFFF
 DC.L $FFFFFFFF

dh:DC.W -1;BALL INCREMENTS
dv:DC.W 0
 
; ------   APPLICATION GLOBALS  ----
 
Field.of.play: DS.W1 ;frame size storage
 DS.W 1 ;for game field rectangle
 DS.W 1
 DS.W 1
 
Paddle: DS.W1  ;frame size storage
 DS.W 1 ;for paddle rectangle
 DS.W 1
 DS.W 1
 
oldpaddle:DS.W 1 ;frame size storage
 DS.W 1 ;for paddle rectangle
 DS.W 1
 DS.W 1
 
Paddlelength:  DS.W1
 
Ball:   DS.W1  ;frame size storage
 DS.W 1 ;for ball rectangle
 DS.W 1
 DS.W 1 
 
Ballwide: DS.W 1
Ballhigh: DS.W 1

Ballrgn:DS.L1  ;ball region handle
oldballrgn: DS.L 1 ;old ball region handle
unionrgn: DS.L 1 ;temp region

mouse:  DS.L1
oldmouse: DS.L 1


; ------   END OF PROGRAM -------


!START
[
)
/OUTPUT Animate Example

Animate

/TYPE ‘APPL’ ‘ANIM’
/BUNDLE
/RESOURCES
Animate_rscs

$



;ANIMATE_rscs.asm
; resource file for the animate example
; created using the assembler
; signiture is creator tag 
;
RESOURCE ‘ANIM’ 0 ‘IDENTIFICATION’

 DC.B 30, ‘animate example--Feb. 27, 1985’
 
.ALIGN 2
RESOURCE ‘BNDL’ 128 ‘BUNDLE’

 DC.L ‘ANIM’;NAME OF SIGNATURE
 DC.W 0,1 ;DATA (DOESN’T CHANGE)
 DC.L ‘ICN#’;ICON MAPPINGS
    DC.W0 ;NUMBER OF MAPPINGS-1
    DC.W 0,128 ;MAP 0 TO ICON 128
    
 DC.L ‘FREF’;FREF MAPPINGS
    DC.W0 ;NUMBER OF MAPPINGS-1
    DC.W 0,128 ;MAP 0 TO FREF 128

RESOURCE ‘FREF’ 128 ‘FREF 1’
 
 DC.B ‘APPL’, 0, 0, 0
 
.ALIGN 2
RESOURCE ‘ICN#’ 128 ‘MY ICON’

; FIRST APPLICATION ICON BIT MAP

INCLUDE animate.icon.asm

 

Community Search:
MacTech Search:

Software Updates via MacUpdate

Tor Browser 12.5.5 - Anonymize Web brows...
Using Tor Browser you can protect yourself against tracking, surveillance, and censorship. Tor was originally designed, implemented, and deployed as a third-generation onion-routing project of the U.... Read more
Malwarebytes 4.21.9.5141 - Adware remova...
Malwarebytes (was AdwareMedic) helps you get your Mac experience back. Malwarebytes scans for and removes code that degrades system performance or attacks your system. Making your Mac once again your... Read more
TinkerTool 9.5 - Expanded preference set...
TinkerTool is an application that gives you access to additional preference settings Apple has built into Mac OS X. This allows to activate hidden features in the operating system and in some of the... Read more
Paragon NTFS 15.11.839 - Provides full r...
Paragon NTFS breaks down the barriers between Windows and macOS. Paragon NTFS effectively solves the communication problems between the Mac system and NTFS. Write, edit, copy, move, delete files on... Read more
Apple Safari 17 - Apple's Web brows...
Apple Safari is Apple's web browser that comes bundled with the most recent macOS. Safari is faster and more energy efficient than other browsers, so sites are more responsive and your notebook... Read more
Firefox 118.0 - Fast, safe Web browser.
Firefox offers a fast, safe Web browsing experience. Browse quickly, securely, and effortlessly. With its industry-leading features, Firefox is the choice of Web development professionals and casual... Read more
ClamXAV 3.6.1 - Virus checker based on C...
ClamXAV is a popular virus checker for OS X. Time to take control ClamXAV keeps threats at bay and puts you firmly in charge of your Mac’s security. Scan a specific file or your entire hard drive.... Read more
SuperDuper! 3.8 - Advanced disk cloning/...
SuperDuper! is an advanced, yet easy to use disk copying program. It can, of course, make a straight copy, or "clone" - useful when you want to move all your data from one machine to another, or do a... Read more
Alfred 5.1.3 - Quick launcher for apps a...
Alfred is an award-winning productivity application for OS X. Alfred saves you time when you search for files online or on your Mac. Be more productive with hotkeys, keywords, and file actions at... Read more
Sketch 98.3 - Design app for UX/UI for i...
Sketch is an innovative and fresh look at vector drawing. Its intentionally minimalist design is based upon a drawing space of unlimited size and layers, free of palettes, panels, menus, windows, and... Read more

Latest Forum Discussions

See All

Listener Emails and the iPhone 15! – The...
In this week’s episode of The TouchArcade Show we finally get to a backlog of emails that have been hanging out in our inbox for, oh, about a month or so. We love getting emails as they always lead to interesting discussion about a variety of topics... | Read more »
TouchArcade Game of the Week: ‘Cypher 00...
This doesn’t happen too often, but occasionally there will be an Apple Arcade game that I adore so much I just have to pick it as the Game of the Week. Well, here we are, and Cypher 007 is one of those games. The big key point here is that Cypher... | Read more »
SwitchArcade Round-Up: ‘EA Sports FC 24’...
Hello gentle readers, and welcome to the SwitchArcade Round-Up for September 29th, 2023. In today’s article, we’ve got a ton of news to go over. Just a lot going on today, I suppose. After that, there are quite a few new releases to look at... | Read more »
‘Storyteller’ Mobile Review – Perfect fo...
I first played Daniel Benmergui’s Storyteller (Free) through its Nintendo Switch and Steam releases. Read my original review of it here. Since then, a lot of friends who played the game enjoyed it, but thought it was overpriced given the short... | Read more »
An Interview with the Legendary Yu Suzuk...
One of the cool things about my job is that every once in a while, I get to talk to the people behind the games. It’s always a pleasure. Well, today we have a really special one for you, dear friends. Mr. Yu Suzuki of Ys Net, the force behind such... | Read more »
New ‘Marvel Snap’ Update Has Balance Adj...
As we wait for the information on the new season to drop, we shall have to content ourselves with looking at the latest update to Marvel Snap (Free). It’s just a balance update, but it makes some very big changes that combined with the arrival of... | Read more »
‘Honkai Star Rail’ Version 1.4 Update Re...
At Sony’s recently-aired presentation, HoYoverse announced the Honkai Star Rail (Free) PS5 release date. Most people speculated that the next major update would arrive alongside the PS5 release. | Read more »
‘Omniheroes’ Major Update “Tide’s Cadenc...
What secrets do the depths of the sea hold? Omniheroes is revealing the mysteries of the deep with its latest “Tide’s Cadence" update, where you can look forward to scoring a free Valkyrie and limited skin among other login rewards like the 2nd... | Read more »
Recruit yourself some run-and-gun royalt...
It is always nice to see the return of a series that has lost a bit of its global staying power, and thanks to Lilith Games' latest collaboration, Warpath will be playing host the the run-and-gun legend that is Metal Slug 3. [Read more] | Read more »
‘The Elder Scrolls: Castles’ Is Availabl...
Back when Fallout Shelter (Free) released on mobile, and eventually hit consoles and PC, I didn’t think it would lead to something similar for The Elder Scrolls, but here we are. The Elder Scrolls: Castles is a new simulation game from Bethesda... | Read more »

Price Scanner via MacPrices.net

Clearance M1 Max Mac Studio available today a...
Apple has clearance M1 Max Mac Studios available in their Certified Refurbished store for $270 off original MSRP. Each Mac Studio comes with Apple’s one-year warranty, and shipping is free: – Mac... Read more
Apple continues to offer 24-inch iMacs for up...
Apple has a full range of 24-inch M1 iMacs available today in their Certified Refurbished store. Models are available starting at only $1099 and range up to $260 off original MSRP. Each iMac is in... Read more
Final weekend for Apple’s 2023 Back to School...
This is the final weekend for Apple’s Back to School Promotion 2023. It remains active until Monday, October 2nd. Education customers receive a free $150 Apple Gift Card with the purchase of a new... Read more
Apple drops prices on refurbished 13-inch M2...
Apple has dropped prices on standard-configuration 13″ M2 MacBook Pros, Certified Refurbished, to as low as $1099 and ranging up to $230 off MSRP. These are the cheapest 13″ M2 MacBook Pros for sale... Read more
14-inch M2 Max MacBook Pro on sale for $300 o...
B&H Photo has the Space Gray 14″ 30-Core GPU M2 Max MacBook Pro in stock and on sale today for $2799 including free 1-2 day shipping. Their price is $300 off Apple’s MSRP, and it’s the lowest... Read more
Apple is now selling Certified Refurbished M2...
Apple has added a full line of standard-configuration M2 Max and M2 Ultra Mac Studios available in their Certified Refurbished section starting at only $1699 and ranging up to $600 off MSRP. Each Mac... Read more
New sale: 13-inch M2 MacBook Airs starting at...
B&H Photo has 13″ MacBook Airs with M2 CPUs in stock today and on sale for $200 off Apple’s MSRP with prices available starting at only $899. Free 1-2 day delivery is available to most US... Read more
Apple has all 15-inch M2 MacBook Airs in stoc...
Apple has Certified Refurbished 15″ M2 MacBook Airs in stock today starting at only $1099 and ranging up to $230 off MSRP. These are the cheapest M2-powered 15″ MacBook Airs for sale today at Apple.... Read more
In stock: Clearance M1 Ultra Mac Studios for...
Apple has clearance M1 Ultra Mac Studios available in their Certified Refurbished store for $540 off original MSRP. Each Mac Studio comes with Apple’s one-year warranty, and shipping is free: – Mac... Read more
Back on sale: Apple’s M2 Mac minis for $100 o...
B&H Photo has Apple’s M2-powered Mac minis back in stock and on sale today for $100 off MSRP. Free 1-2 day shipping is available for most US addresses: – Mac mini M2/256GB SSD: $499, save $100 –... Read more

Jobs Board

Licensed Dental Hygienist - *Apple* River -...
Park Dental Apple River in Somerset, WI is seeking a compassionate, professional Dental Hygienist to join our team-oriented practice. COMPETITIVE PAY AND SIGN-ON Read more
Sublease Associate Optometrist- *Apple* Val...
Sublease Associate Optometrist- Apple Valley, CA- Target Optical Date: Sep 30, 2023 Brand: Target Optical Location: Apple Valley, CA, US, 92307 **Requisition Read more
*Apple* / Mac Administrator - JAMF - Amentum...
Amentum is seeking an ** Apple / Mac Administrator - JAMF** to provide support with the Apple Ecosystem to include hardware and software to join our team and Read more
Child Care Teacher - Glenda Drive/ *Apple* V...
Child Care Teacher - Glenda Drive/ Apple ValleyTeacher Share by Email Share on LinkedIn Share on Twitter 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.