TweetFollow Us on Twitter

MACINTOSH C

Demonstration Program

Go to Contents
// 
// Appearance.c
// 
// 
// This program opens two kWindowDocumentProc windows containing:
//
//   In the first window, a theme-compliant list view.
//
//   In the second window, various images drawn with Appearance primitives and window
//    header text drawn in the correct theme colour.
//
// Two of the images in the second window are edit text field frames and one is a list
// box frame.  At any one time, one of these will have a keyboard focus frame drawn
// around it.  Clicking in one of the other frames will move the keyboard focus frame
// to that frame.
//
// The program is terminated by the choosing the Quit item in the File menu. 
//
// The program utilises the following resources:
//
//   An 'MBAR' resource, and 'MENU' resources for Apple, File, Edit, and Demonstration
//     menus, and the pop-up menus (preload, non-purgeable).  
//
//   Two 'WIND' resources (purgeable) (initially not visible).  
//
//   'hrct' and 'hwin' resources (both purgeable), which provide help balloons 
//    describing the contents of the windows.  
//
//   A 'SIZE' resource with the acceptSuspendResumeEvents, doesActivateOnFGSwitch,
//    and is32BitCompatible flags set.    
//
// 

// ............................................................................. includes

#include <Appearance.h>
#include <Devices.h>
#include <Gestalt.h>
#include <LowMem.h>
#include <Sound.h>
#include <ToolUtils.h>

// .............................................................................. defines

#define rMenubar     128
#define rNewWindow1  128
#define rNewWindow2  129
#define mApple       128
#define  iAbout      1
#define mFile        129
#define  iQuit       11

#define MAXLONG      0x7FFFFFFF
#define topLeft(r)   (((Point *) &(r))[0])

// ..................................................................... global variables

Boolean    gAppearancePresent     = false;
Boolean    gInCompatibilityMode   = false;
Boolean    gAppearance101present  = false;
Boolean    gAppearance110present  = false;
Boolean    gDone;
Boolean    gInBackground;
WindowPtr  gWindowPtr1, gWindowPtr2;
SInt16     gPixelDepth;  
Boolean    gIsColourDevice        = false;
Rect       gCurrentRect;

// .................................................................. function prototypes

void  main                       (void);
void  doInitManagers             (void);
void  doEvents                   (EventRecord *);
void  doUpdate                   (EventRecord *);
void  doActivate                 (EventRecord *);
void  doActivateWindow           (WindowPtr,Boolean);
void  doOSEvent                  (EventRecord *);
void  doDrawThemePrimitives      (ThemeDrawState);
void  doDrawThemeCompliantText   (WindowPtr,ThemeDrawState);
void  doDrawListView             (WindowPtr);
void  doChangeKeyBoardFocus      (Point);
void  doGetDepthAndDevice        (void);

//  main

void  main(void)
{
  OSErr         osError;
  SInt32        response;
  Handle        menubarHdl;
  MenuHandle    menuHdl;
  EventRecord   EventStructure;

  // ......... check for Appearance and functions, compatibility mode, Appearance version

  osError = Gestalt(gestaltAppearanceAttr,&response);

  if(osError == noErr && (BitTst(&response,31 - gestaltAppearanceExists)))
  {
    gAppearancePresent = true;

    if(BitTst(&response,31 - gestaltAppearanceCompatMode))
      gInCompatibilityMode = true;

    Gestalt(gestaltAppearanceVersion,&response);

    if(response == 0x00000101)
      gAppearance101present = true;
    else if(response >= 0x00000110)
      gAppearance110present = true;
  }
  else
  {
    SysBeep(10);
    ExitToShell();
  }

  // ................................................................ initialise managers

  doInitManagers();
  
  // .......................................................... set up menu bar and menus
  
  menubarHdl = GetNewMBar(rMenubar);
  if(menubarHdl == NULL)
    ExitToShell();
  SetMenuBar(menubarHdl);
  DrawMenuBar();

  menuHdl = GetMenuHandle(mApple);
  if(menuHdl == NULL)
    ExitToShell();
  else
    AppendResMenu(menuHdl,'DRVR');

  // ............................ open windows, set font size, show windows, move windows

  if(!(gWindowPtr1 = GetNewCWindow(rNewWindow1,NULL,(WindowPtr)-1)))
    ExitToShell();

  SetPort(gWindowPtr1);
  TextSize(10);
  ShowWindow(gWindowPtr1);
    
  if(!(gWindowPtr2 = GetNewCWindow(rNewWindow2,NULL,(WindowPtr)-1)))
    ExitToShell();

  SetPort(gWindowPtr2);
  TextSize(10);
  ShowWindow(gWindowPtr2);
  
  // ............................... set theme-compliant colour/pattern for second window
  
  SetThemeWindowBackground(gWindowPtr2,kThemeBrushDialogBackgroundActive,true);

  // ......... get pixel depth and whether colour device for certain Appearance functions  
  
  doGetDepthAndDevice();

    // ..... set top edit text field rectangle as target for initial keyboard focus frame
    
  SetRect(&gCurrentRect,20,141,239,162);

  // .................................................................... enter eventLoop

  gDone = false;

  while(!gDone)
  {
    if(WaitNextEvent(everyEvent,&EventStructure,MAXLONG,NULL))
      doEvents(&EventStructure);
  }
}

//  doInitManagers

void  doInitManagers(void)
{
  MaxApplZone();
  MoreMasters();

  InitGraf(&qd.thePort);
  InitFonts();
  InitWindows();
  InitMenus();
  TEInit();
  InitDialogs(NULL);

  InitCursor();  
  FlushEvents(everyEvent,0);

  RegisterAppearanceClient();
}

//  doEvents

void  doEvents(EventRecord *eventStrucPtr)
{
  SInt8      charCode;
  SInt32     menuChoice;
  SInt16     menuID, menuItem;
  SInt16     partCode;
  WindowPtr  windowPtr;
  Str255     itemName;
  SInt16     daDriverRefNum;

  switch(eventStrucPtr->what)
  {
    case keyDown:
    case autoKey:
      charCode = eventStrucPtr->message & charCodeMask;
      if((eventStrucPtr->modifiers & cmdKey) != 0)
      {
        menuChoice = MenuEvent(eventStrucPtr);
        menuID = HiWord(menuChoice);
        menuItem = LoWord(menuChoice);
        if(menuID == mFile && menuItem  == iQuit)
          gDone = true;
      }
      break;
  
    case mouseDown:
      if(partCode = FindWindow(eventStrucPtr->where,&windowPtr))
      {
        switch(partCode)
        {
          case inMenuBar:
            menuChoice = MenuSelect(eventStrucPtr->where);
            menuID = HiWord(menuChoice);
            menuItem = LoWord(menuChoice);

            if(menuID == 0)
              return;

            switch(menuID)
            {
              case mApple:
                if(menuItem == iAbout)
                  SysBeep(10);
                else
                {
                  GetMenuItemText(GetMenuHandle(mApple),menuItem,itemName);
                  daDriverRefNum = OpenDeskAcc(itemName);
                }
                break;

              case mFile:
                if(menuItem == iQuit)
                  gDone = true;
                break;
            }
            HiliteMenu(0);
            break;
        
          case inContent:
            if(windowPtr != FrontWindow())
              SelectWindow(windowPtr);
            else
            {
              if(FrontWindow() == gWindowPtr2)
              {
                SetPort(gWindowPtr2);
                doChangeKeyBoardFocus(eventStrucPtr->where);
              }
            }
            break;
          
          case inDrag:
            DragWindow(windowPtr,eventStrucPtr->where,&qd.screenBits.bounds);
            break;
        }
      }
      break;
      
    case updateEvt:
      doUpdate(eventStrucPtr);
      break;

    case activateEvt:
      doActivate(eventStrucPtr);
      break;

    case osEvt:
      doOSEvent(eventStrucPtr);
      HiliteMenu(0);
      break;
  }
}

//  doUpdate

void  doUpdate(EventRecord *eventStrucPtr)
{
  WindowPtr  windowPtr;
  
  windowPtr = (WindowPtr) eventStrucPtr->message;

  BeginUpdate(windowPtr);

  SetPort(windowPtr);
  
  if(windowPtr == gWindowPtr2)
  {
    if(gWindowPtr2 == FrontWindow() && !gInBackground)
    {
      doDrawThemePrimitives(kThemeStateActive);
      doDrawThemeCompliantText(windowPtr,kThemeStateActive);
      DrawThemeFocusRect(&gCurrentRect,true);
    }
    else
    {
      doDrawThemePrimitives(kThemeStateDisabled);
      doDrawThemeCompliantText(windowPtr,kThemeStateDisabled);
    }  
  }    
    
  if(windowPtr == gWindowPtr1)
    doDrawListView(windowPtr);

  EndUpdate(windowPtr);
}

//  doActivate

void  doActivate(EventRecord *eventStrucPtr)
{
  WindowPtr  windowPtr;
  Boolean    becomingActive;

  windowPtr = (WindowPtr) eventStrucPtr->message;
  becomingActive = ((eventStrucPtr->modifiers & activeFlag) == activeFlag);
  doActivateWindow(windowPtr,becomingActive);
}

//  doActivateWindow

void  doActivateWindow(WindowPtr windowPtr,Boolean becomingActive)
{  
  if(windowPtr == gWindowPtr2)
  {
    SetPort(gWindowPtr2);

    doDrawThemePrimitives(becomingActive);
    doDrawThemeCompliantText(windowPtr,becomingActive);
    DrawThemeFocusRect(&gCurrentRect,becomingActive);
  }
}

//  doOSEvent

void  doOSEvent(EventRecord *eventStrucPtr)
{
  switch((eventStrucPtr->message >> 24) & 0x000000FF)
  {
    case suspendResumeMessage:
      gInBackground = (eventStrucPtr->message & resumeFlag) == 0;
      doActivateWindow(FrontWindow(),!gInBackground);
      break;
  }
}

//  doDrawThemePrimitives

void  doDrawThemePrimitives(ThemeDrawState inState)
{
  Rect  theRect;
      
  SetRect(&theRect,-1,-1,261,26);
  DrawThemeWindowHeader(&theRect,inState);

  SetRect(&theRect,20,46,119,115);
  DrawThemePrimaryGroup(&theRect,inState);

  SetRect(&theRect,140,46,239,115);
  DrawThemeSecondaryGroup(&theRect,inState);

  SetRect(&theRect,20,127,240,128);
  DrawThemeSeparator(&theRect,inState);

  SetRect(&theRect,20,141,239,162);
  DrawThemeEditTextFrame(&theRect,inState);

  SetRect(&theRect,20,169,239,190);
  DrawThemeEditTextFrame(&theRect,inState);

  if(gAppearance101present || gAppearance110present)
  {
    SetRect(&theRect,20,203,62,245);
    DrawThemeGenericWell(&theRect,inState,false);
  }

  SetRect(&theRect,20,258,62,300);
  DrawThemeGenericWell(&theRect,inState,true);

  SetRect(&theRect,75,202,76,302);
  DrawThemeSeparator(&theRect,inState);

  SetRect(&theRect,90,203,239,300);
  DrawThemeListBoxFrame(&theRect,inState);
  
  SetRect(&theRect,-1,321,261,337);
  DrawThemePlacard(&theRect,inState);
}  

//  doDrawThemeCompliantText

void  doDrawThemeCompliantText(WindowPtr windowPtr,ThemeDrawState inState)
{
  SInt16  windowWidth, stringWidth;
  Str255  message = "\pBalloon help is available";

  if(inState == kThemeStateActive)
    SetThemeTextColor(kThemeTextColorWindowHeaderActive,gPixelDepth,gIsColourDevice);
  else
    SetThemeTextColor(kThemeTextColorWindowHeaderInactive,gPixelDepth,gIsColourDevice);

  windowWidth = (windowPtr)->portRect.right - (windowPtr)->portRect.left;
  stringWidth = StringWidth(message);
  MoveTo((windowWidth / 2) - (stringWidth / 2), 17);
  DrawString("\pBalloon help is available");  
}

//  doDrawListView

void  doDrawListView(WindowPtr windowPtr)
{
  Rect    theRect;
  SInt16  a;
  
  theRect = windowPtr->portRect;

  SetThemeBackground(kThemeBrushListViewBackground,gPixelDepth,gIsColourDevice);
  EraseRect(&theRect);

  theRect.left += 130;
  
  SetThemeBackground(kThemeBrushListViewSortColumnBackground,gPixelDepth,gIsColourDevice);
  EraseRect(&theRect);
  
  SetThemePen(kThemeBrushListViewSeparator,gPixelDepth,gIsColourDevice);

  theRect = windowPtr->portRect;
  for(a=theRect.top;a<=theRect.bottom;a+=18)
  {
    MoveTo(theRect.left,a);
    LineTo(theRect.right - 1,a);
  }

  SetThemeTextColor(kThemeTextColorListView,gPixelDepth,gIsColourDevice);
  
  for(a=theRect.top;a<=theRect.bottom +18;a+=18)
  {
    MoveTo(theRect.left,a - 5);
    DrawString("\p   List View Background        List View Sort Column");
  }
}

//  doChangeKeyBoardFocus

void  doChangeKeyBoardFocus(Point mouseXY)
{
  Rect  edit1Rect, edit2Rect, listRec;
  
  DrawThemeFocusRect(&gCurrentRect,false);
  DrawThemeEditTextFrame(&gCurrentRect,kThemeStateActive);

  SetRect(&edit1Rect,20,141,239,162);
  SetRect(&edit2Rect,20,169,239,190);
  SetRect(&listRec,90,203,239,300);

  SetPort(gWindowPtr2);
  GlobalToLocal(&mouseXY);

  if(PtInRect(mouseXY,&edit1Rect))
    SetRect(&gCurrentRect,20,141,239,162);
  else if(PtInRect(mouseXY,&edit2Rect))
    SetRect(&gCurrentRect,20,169,239,190);
  else if(PtInRect(mouseXY,&listRec))
    SetRect(&gCurrentRect,90,203,239,300);
  
  DrawThemeFocusRect(&gCurrentRect,true);
}

//  doGetDepthAndDevice

void doGetDepthAndDevice(void)
{
  GDHandle  deviceHdl;

  deviceHdl = LMGetMainDevice();
  gPixelDepth = (*(*deviceHdl)->gdPMap)->pixelSize;
  if(BitTst(&(*deviceHdl)->gdFlags,gdDevType))
    gIsColourDevice = true;
}

// 

Demonstration Program Comments

When this program is run, the user should:

*   First drag the top window to a position where the content of the bottom window is
    visible.

*   Choose Show Balloons from the Help menu and move the cursor over the frames in the
    window titled "Drawing With Primitives" window (when active), and the left and right
    sides of the window titled "Theme-Compliant List View" (when active), noting the 
    descriptions in the balloons.

*   With the "Drawing With Primitives" window frontmost, click in the edit text field
    frame not currently outlined with the keyboard focus frame, or in the list box frame,
    so as to move the keyboard focus frame to that rectangle.

*   Click on the desktop to send the application to the background and note the changed
    appearance of the frames and text in the "Drawing With Primitives" window.  Note also
    that there is no change to the appearance of the content region of the 
    "Theme-Compliant List View" window.  Click on the "Drawing With Primitives" 
    window to bring the application to the foreground with that window active, noting the 
    changed appearance of the frames and text.

In the following, reference is made to graphics devices and pixel depth.  Graphics devices
and pixel depth are explained at Chapter 11 - QuickDraw Preliminaries.

#define

The first block establishes constants representing menu IDs, resources, and menu items,
and window and menu bar resources.

MAXLONG is defined as the maximum possible long value, and is used in the WaitNextEvent
function.  The last line defines a common macro which converts the top and left fields of
a Rect structure to a Point.

Global Variables

gAppearancePresent will be assigned true if at least Version 1.0 of the Appearance Manager
is present.  gInCompatibilityMode will be assigned true if the machine on which the 
demonstration is running is in compatibility mode (applicable only to Versions 1.0 through
1.0.3 only). gAppearance101present will be assigned true if Versions 1.0.1, 1.0.2, or 
1.0.3 are present. gAppearance110present will be assigned true if Version 1.1 is present.

gDone, when set to true, causes the main event loop to be exited and the program to
terminate.  gInBackground relates to foreground/background switching.  gWindowPtr1 and
gWindowPtr2 will be assigned window pointers.

gPixelDepth will be assigned the pixel depth of the main device. gIsColourDevice will be
assigned true if the graphics device is a colour device and false if it is a monochrome
device.  The values in these two variables are required by certain Appearance functions. 
gCurrentRect will be assigned the rectangle which is to be the current target for the
keyboard focus frame.

main

Gestalt is called to determine whether some version of the Appearance Manager is present.
If so, bit 1 in response is tested to determine whether the machine on which the program 
is running is currently in compatibility mode (relevant only where Appearance Manager 
Versions 1.0 through 1.0.3 are present), and Gestalt is called again to determine whether 
Version 1.0.1 through 1.0.3, or Version 1.1 or later, is present.  If the Appearance 
Manager is not present, the system alert sound is played and the program simply 
terminates.

Note that the assignment to the global variable gInCompatibilityMode is for demonstration 
purposes only; the program does not use this variables for any purpose.

After the menus are set up, each window is created.  After each window is created, its 
graphics port is set as the current port and the text size for that port is set to 10pt, 
the window is shown.

SetThemeWindowBackground sets a theme-compliant colour/pattern for the "Drawing With
Primitives" window's content area.  This means that the content area will be automatically
repainted with that colour/pattern when required with no further assistance from the
application.  When true is passed in the third parameter, the content region of the window
is invalidated and the content region is repainted immediately.

The call to the application-defined function doGetDepthAndDevice determines the current
pixel depth of the graphics port, and whether the current graphics device is a colour
device, and assigns the results to the global variables gPixelDepth and gIsColourDevice.

The call to SetRect establishes the initial target for the keyboard focus frame.  This is
the rectangle used by the first edit text field frame.

doInitManagers

DoInitManagers is called from main immediately after it has been determined that the 
Appearance Manager is present.  In this demonstration program, and in all subsequent 
demonstration programs, a call to RegisterAppearanceClient has been added to this 
function.

If this program is run under Appearance Manager Versions 1.0 through 1.0.3, one effect of 
the call to RegisterAppearanceClient is that the new theme-compliant menu bar 
definition function (resource ID 63) will be used regardless of whether system-wide 
Appearance is selected on or off in the Appearance control panel.

doEvents

At the mouseDown case, the inContent case within the partCode switch is of relevance to
the demonstration.

If the mouse-down was within the content region of a window, and if that window is not the
front window, SelectWindow is called to bring that window to the front and activate it.

However, if the window is the front window, and if that window is the "Drawing With
Primitives" window, that window's graphics port is set as the current graphics port for
drawing, and the application-defined function doChangeKeyBoardFocus is called.  That
function determines whether the mouse-down was within one of the edit text field frames or
the list box frame, and moves the keyboard focus if necessary.

doUpdate

Within the doUpdate function, if the window to which the update event relates is the
"Drawing With Primitives" window, and if that window is currently the front window:

*   Application-defined functions are called to draw the primitives and the window header
    text in the active mode.

*   DrawThemeFocusRect is called to draw the keyboard focus frame using the rectangle
    currently assigned to the global variable gCurrentRect.

If, however, the "Drawing With Primitives" window is not the front window, the same calls
are made but with the primitives and text being drawn in the inactive mode.  Note that no
call is required to erase the keyboard focus frame because this will already have been
erased when the window was deactivated (see below).

If the window to which update event relates is the "Theme-Compliant List View" 
window, an application-defined function for drawing the window's content area is called.
Note that, for this window, there is no differentiation between active and inactive modes.
This is because, for list views, the same brush type constants are used regardless of 
whether the window is active or inactive.


doActivateWindow

When an activate event is received for the "Drawing With Primitives" window, the
application-defined functions for drawing the primitives and the window header text,
together with the Appearance function which draws and erases the keyboard focus rectangle,
are called.  To eliminate the necessity for if/else coding, the becomingActive value is
used to ensure that, firstly, the primitives and text are drawn in the appropriate mode
and, secondly, that the keyboard focus frame is either drawn or erased, depending on
whether the window is coming to the front or being sent to the back.

Once again, the "Theme-Compliant List View" window is treated differently because the
list view brush constants to be used are the same regardless of whether the window is 
activated and deactivated.

doDrawAppearancePrimitives

doDrawAppearancePrimitives uses Appearance Manager functions for drawing Appearance
primitives, and is called to draw the various frames in the "Drawing With Primitives" window.
The mode in which the primitives are drawn (active or inactive) is determined by the
Boolean value passed in the inState parameter.

Note that DrawThemeGenericWell, which was introduced with Version 1.0.1 of the Appearance
Manager, is called only if Versions 1.0.1 through 1.0.3, or Version 1.1, are present.

doDrawAppearanceCompliantText

doDrawAppearanceCompliantText is called to draw some advisory text in the window header
of the "Drawing With Primitives" window.  The QuickDraw drawing function DrawString does
the drawing; however, before the drawing begins, the Appearance function SetThemeTextColor
is used to set the foreground colour for drawing text, in either the active or inactive
modes, so as to comply with the current appearance.

At the first two lines, if "Drawing With Primitives" is the active window,
SetThemeTextColor is called with the kThemeActiveWindowHeaderTextColor text colour
constant passed in the first parameter.  At the next two lines, if the window is inactive,
SetThemeTextColor is called with kThemeInctiveWindowHeaderTextColor passed in the first
parameter.  Note that SetThemeTextColor requires the pixel depth of the graphics port, and
whether the graphics device is a colour device or a monochrome device, passed in the
second and third parameters.

The next three lines simply adjust QuickDraw's pen location so that the text is drawn
centered laterally in the window header frame.  The call to DrawString draws the specified
text.

doDrawListView

doDrawListView draws a theme-compliant list view background in the specified window.

The first line copies the window's port rectangle to a local variable of type Rect.

The call to SetThemeBackground sets the background colour/pattern to the colour/pattern
represented by the theme-compliant brush type constant kThemeListViewBackgroundBrush. 
The QuickDraw function EraseRect fills the whole of the port rectangle with this
colour/pattern.

The next line adjusts the Rect variable's left field so that the rectangle now represents
the right half of the port rectangle.  The same drawing process is then repeated, but this
time with kThemeListViewSortColumnBackgroundBrush passed in the first parameter of the
SetThemeBackground call.

SetThemePen is then called with the colour/pattern represented by the constant
kThemeListViewSeparatorBrush passed in the first parameter.  The rectangle for drawing is
then expanded to equate with the port rectangle before the following five lines draw
one-pixel-wide horizontal lines, at 18-pixel intervals, from the top to the bottom of the
port rectangle.

Finally, some text is drawn in the list view in the theme-compliant colour for list
views.  SetThemeTextColour is called with the kThemeListViewTextColor passed in, following
which a for loop draws some text, at 18-pixel intervals, from the top to the bottom of the
port rectangle.

doChangeKeyBoardFocus

doChangeKeyBoardFocus is called when a mouse-down occurs in the content region of the
"Drawing With Primitives" window.

At the first two lines, Appearance functions are used to, firstly, erase the keyboard
focus frame from the rectangle around which it is currently drawn and, secondly, redraw an
edit text field frame around that rectangle.

The next three lines make three local variables of type Rect equal to the rectangles for
the two edit text field frames and the list box frame.

The call to GlobalToLocal converts the coordinates of the mouse-down to the local
coordinates required by the following calls to PtInRect.  PtInRect returns true if the
mouse-down is within the rectangle passed in the second parameter.  If one of the calls to
PtInRect returns true, that rectangle is made the current rectangle for keyboard focus by
assigning it to the global variable gCurrentRect.

Whatever rectangle is assigned to gCurrentRect, the call to DrawThemeFocusRect draws a
theme-compliant keyboard focus frame around that rectangle.

doGetDepthAndDevice

doGetDepthAndDevice determines the pixel depth of the graphics port, and whether the
graphics device is a colour device or a monochrome device, and assigns the results to two
global variables.  This information is required by certain Appearance functions.
 

Community Search:
MacTech Search:

Software Updates via MacUpdate

Latest Forum Discussions

See All

Top Mobile Game Discounts
Every day, we pick out a curated list of the best mobile discounts on the App Store and post them here. This list won't be comprehensive, but it every game on it is recommended. Feel free to check out the coverage we did on them in the links... | Read more »
Price of Glory unleashes its 1.4 Alpha u...
As much as we all probably dislike Maths as a subject, we do have to hand it to geometry for giving us the good old Hexgrid, home of some of the best strategy games. One such example, Price of Glory, has dropped its 1.4 Alpha update, stocked full... | Read more »
The SLC 2025 kicks off this month to cro...
Ever since the Solo Leveling: Arise Championship 2025 was announced, I have been looking forward to it. The promotional clip they released a month or two back showed crowds going absolutely nuts for the previous competitions, so imagine the... | Read more »
Dive into some early Magicpunk fun as Cr...
Excellent news for fans of steampunk and magic; the Precursor Test for Magicpunk MMORPG Crystal of Atlan opens today. This rather fancy way of saying beta test will remain open until March 5th and is available for PC - boo - and Android devices -... | Read more »
Prepare to get your mind melted as Evang...
If you are a fan of sci-fi shooters and incredibly weird, mind-bending anime series, then you are in for a treat, as Goddess of Victory: Nikke is gearing up for its second collaboration with Evangelion. We were also treated to an upcoming... | Read more »
Square Enix gives with one hand and slap...
We have something of a mixed bag coming over from Square Enix HQ today. Two of their mobile games are revelling in life with new events keeping them alive, whilst another has been thrown onto the ever-growing discard pile Square is building. I... | Read more »
Let the world burn as you have some fest...
It is time to leave the world burning once again as you take a much-needed break from that whole “hero” lark and enjoy some celebrations in Genshin Impact. Version 5.4, Moonlight Amidst Dreams, will see you in Inazuma to attend the Mikawa Flower... | Read more »
Full Moon Over the Abyssal Sea lands on...
Aether Gazer has announced its latest major update, and it is one of the loveliest event names I have ever heard. Full Moon Over the Abyssal Sea is an amazing name, and it comes loaded with two side stories, a new S-grade Modifier, and some fancy... | Read more »
Open your own eatery for all the forest...
Very important question; when you read the title Zoo Restaurant, do you also immediately think of running a restaurant in which you cook Zoo animals as the course? I will just assume yes. Anyway, come June 23rd we will all be able to start up our... | Read more »
Crystal of Atlan opens registration for...
Nuverse was prominently featured in the last month for all the wrong reasons with the USA TikTok debacle, but now it is putting all that behind it and preparing for the Crystal of Atlan beta test. Taking place between February 18th and March 5th,... | Read more »

Price Scanner via MacPrices.net

AT&T is offering a 65% discount on the ne...
AT&T is offering the new iPhone 16e for up to 65% off their monthly finance fee with 36-months of service. No trade-in is required. Discount is applied via monthly bill credits over the 36 month... Read more
Use this code to get a free iPhone 13 at Visi...
For a limited time, use code SWEETDEAL to get a free 128GB iPhone 13 Visible, Verizon’s low-cost wireless cell service, Visible. Deal is valid when you purchase the Visible+ annual plan. Free... Read more
M4 Mac minis on sale for $50-$80 off MSRP at...
B&H Photo has M4 Mac minis in stock and on sale right now for $50 to $80 off Apple’s MSRP, each including free 1-2 day shipping to most US addresses: – M4 Mac mini (16GB/256GB): $549, $50 off... Read more
Buy an iPhone 16 at Boost Mobile and get one...
Boost Mobile, an MVNO using AT&T and T-Mobile’s networks, is offering one year of free Unlimited service with the purchase of any iPhone 16. Purchase the iPhone at standard MSRP, and then choose... Read more
Get an iPhone 15 for only $299 at Boost Mobil...
Boost Mobile, an MVNO using AT&T and T-Mobile’s networks, is offering the 128GB iPhone 15 for $299.99 including service with their Unlimited Premium plan (50GB of premium data, $60/month), or $20... Read more
Unreal Mobile is offering $100 off any new iP...
Unreal Mobile, an MVNO using AT&T and T-Mobile’s networks, is offering a $100 discount on any new iPhone with service. This includes new iPhone 16 models as well as iPhone 15, 14, 13, and SE... Read more
Apple drops prices on clearance iPhone 14 mod...
With today’s introduction of the new iPhone 16e, Apple has discontinued the iPhone 14, 14 Pro, and SE. In response, Apple has dropped prices on unlocked, Certified Refurbished, iPhone 14 models to a... Read more
B&H has 16-inch M4 Max MacBook Pros on sa...
B&H Photo is offering a $360-$410 discount on new 16-inch MacBook Pros with M4 Max CPUs right now. B&H offers free 1-2 day shipping to most US addresses: – 16″ M4 Max MacBook Pro (36GB/1TB/... Read more
Amazon is offering a $100 discount on the M4...
Amazon has the M4 Pro Mac mini discounted $100 off MSRP right now. Shipping is free. Their price is the lowest currently available for this popular mini: – Mac mini M4 Pro (24GB/512GB): $1299, $100... Read more
B&H continues to offer $150-$220 discount...
B&H Photo has 14-inch M4 MacBook Pros on sale for $150-$220 off MSRP. B&H offers free 1-2 day shipping to most US addresses: – 14″ M4 MacBook Pro (16GB/512GB): $1449, $150 off MSRP – 14″ M4... Read more

Jobs Board

All contents are Copyright 1984-2011 by Xplain Corporation. All rights reserved. Theme designed by Icreon.