home *** CD-ROM | disk | FTP | other *** search
/ Tricks of the Windows Gam…ming Gurus (2nd Edition) / Disc2.iso / msdn_vcb / samples / vc98 / sdk / graphics / directx / flip2d / flipcube.cpp < prev    next >
C/C++ Source or Header  |  1997-07-14  |  38KB  |  1,271 lines

  1. /**************************************************************************
  2.  
  3.     FLIPCUBE.CPP - A spinning cube demo for DirectDraw
  4.  
  5.     basic page fliping app, just render to the back buffer and flip
  6.     that is all I do.
  7.  
  8.  **************************************************************************/
  9. /**************************************************************************
  10.  
  11.     (C) Copyright 1995-1997 Microsoft Corp.  All rights reserved.
  12.  
  13.     You have a royalty-free right to use, modify, reproduce and
  14.     distribute the Sample Files (and/or any modified version) in
  15.     any way you find useful, provided that you agree that
  16.     Microsoft has no warranty obligations or liability for any
  17.     Sample Application Files which are modified.
  18.  
  19.  **************************************************************************/
  20.  
  21. #define INITGUID
  22.  
  23. #include <windows.h>
  24. #include <windowsx.h>
  25. #include <mmsystem.h>
  26. #include <ddraw.h>
  27. #include <dinput.h>
  28. #include <math.h>
  29.  
  30. #include "flipcube.h"
  31. #include "dumb3d.h"
  32.  
  33. // code is in tri.cpp
  34. extern void Triangle8(BYTE *p, int next_scan, POINT P0, POINT P1, POINT P2, DWORD c);
  35.  
  36. /**************************************************************************
  37.   Global Variables
  38.  **************************************************************************/
  39.  
  40. static char szAppName[]="DirectDraw Spinning Cube";
  41.  
  42. static HINSTANCE  hInstApp;
  43. static BOOL       fAppActive;
  44. static BOOL       fAppPaused;
  45. static HWND       hwndApp;
  46. static HACCEL     hAccelApp;
  47. static HFONT      AppFont;
  48. static SIZE       ScreenSize;
  49. static BOOL       fDrawWithGDI;
  50.  
  51. /**************************************************************************
  52.   DirectDraw Globals
  53.  **************************************************************************/
  54.  
  55. IDirectDraw            *dd;
  56. IDirectDrawSurface     *FrontBuffer;
  57. IDirectDrawSurface     *BackBuffer;
  58. IDirectDrawPalette     *Palette;
  59.  
  60. /**************************************************************************
  61.   DirectInput Globals
  62.  **************************************************************************/
  63. LPDIRECTINPUT            lpdi;
  64. LPDIRECTINPUTDEVICE             lpdiZoom;       // Used for zooming
  65. LPDIRECTINPUTDEVICE             lpdiRot;        // Use for rotation
  66. BOOL                            fMouseAcquired = FALSE; // Acquired for rot'n
  67.  
  68. /**************************************************************************
  69.   dumb 3D Globals
  70.  **************************************************************************/
  71.  
  72. //*** Cube vertices, normals, shades, and modeling transform
  73. static point_4 CubeVertices[8] =
  74. {
  75.   point_4( -10,  10, -10 ),
  76.   point_4( -10,  10,  10 ),
  77.   point_4(  10,  10,  10 ),
  78.   point_4(  10,  10, -10 ),
  79.   point_4(  10, -10, -10 ),
  80.   point_4(  10, -10,  10 ),
  81.   point_4( -10, -10,  10 ),
  82.   point_4( -10, -10, -10 )
  83. };
  84. static vector_4   CubeSurfaceNormals[6];
  85. static real       CubeSurfaceShades[6];
  86. static matrix_4x4 CubeTransform;
  87.  
  88. //*** Cube edges - ordered indices into the vertex array
  89. const int CubeFaces[6][4] =
  90. {
  91.   0, 1, 2, 3,
  92.   2, 1, 6, 5,
  93.   3, 2, 5, 4,
  94.   0, 3, 4, 7,
  95.   1, 0, 7, 6,
  96.   4, 5, 6, 7
  97. };
  98.  
  99. //*** Cube colors - one RGB color per surface
  100. const unsigned char CubeColors[6][3] =
  101. {
  102.   240,  20,  20,    // Unsaturated Red
  103.    20, 240,  20,    // Unsaturated Green
  104.    20,  20, 240,    // Unsaturated Blue
  105.   128,  64,   0,    // Brown
  106.   240,  20, 240,    // Unsaturated Magenta
  107.   240, 240,  20     // Unsaturated Yellow
  108. };
  109.  
  110. //*** Lighting
  111. vector_4   LightSourceDirection;
  112. const real AmbientLight = 0.2;
  113.  
  114. //*** Viewing and perspective
  115. static matrix_4x4  ViewPerspective;
  116. static point_4     Viewpoint(60, 60, 60);
  117. static vector_4    Up(0, 1, 0);
  118. static point_4     Origin;
  119.  
  120. /**************************************************************************
  121.    Internal function declarations
  122.  **************************************************************************/
  123.  
  124. LONG  CALLBACK AppWndProc(HWND hwnd,UINT msg,WPARAM wParam,LPARAM lParam);
  125. BOOL  InitDInput(void);
  126. BOOL  AppIdle(void);
  127. void  RenderFrame(void);
  128.  
  129. void  TransformCube(matrix_4x4 const &Transform);
  130. BOOL  ProjectAndDrawCube(HDC hdc, int XOffset, int YOffset);
  131. BOOL  ProjectAndDrawCube(IDirectDrawSurface *pdds, int XOffset, int YOffset);
  132.  
  133. /**************************************************************************
  134.   AppAbout
  135.  
  136.   Description:
  137.     This function handles messages belonging to the "About" dialog box.
  138.   The only message that it looks for is WM_COMMAND, indicating the user
  139.   has pressed the "OK" button.
  140.  **************************************************************************/
  141.  
  142. BOOL FAR PASCAL AppAbout(HWND hwnd,UINT msg,WPARAM wParam,LPARAM lParam)
  143. {
  144.   switch (msg)
  145.   {
  146.     case WM_COMMAND:
  147.       if (LOWORD(wParam) == IDOK)
  148.         EndDialog(hwnd, TRUE);
  149.       break;
  150.  
  151.     case WM_INITDIALOG:
  152.       return TRUE;
  153.   }
  154.   return FALSE;
  155. }
  156.  
  157. /**************************************************************************
  158.   DDInit
  159.  
  160.   Description:
  161.     initialize all the DirectDraw specific stuff
  162.  **************************************************************************/
  163.  
  164. BOOL DDInit()
  165. {
  166.     HRESULT err;
  167.  
  168.     err = DirectDrawCreate(NULL, &dd, NULL);
  169.  
  170.     if (err != DD_OK)
  171.         return FALSE;
  172.  
  173.     err = dd->SetCooperativeLevel(hwndApp,
  174.         DDSCL_EXCLUSIVE | DDSCL_FULLSCREEN | DDSCL_ALLOWMODEX);
  175.  
  176.     if (err != DD_OK)
  177.         return FALSE;
  178.  
  179.     // NEWNEW init DirectInput iff DDraw inits ok
  180.     if(!InitDInput())
  181.         return FALSE;
  182.  
  183.     return TRUE;
  184. }
  185.  
  186. /**************************************************************************
  187.   CreateMouse
  188.  **************************************************************************/
  189.  
  190. BOOL CreateMouse(GUID &guid, LPDIRECTINPUTDEVICE& lpdiMouse, DWORD dwAccess)
  191. {
  192.     HRESULT err;
  193.  
  194.     err = lpdi->CreateDevice(guid, &lpdiMouse, NULL);
  195.  
  196.     if(err != DI_OK)
  197.     {
  198.             MessageBox(NULL, "Unable to Create DirectInput Mouse Device",
  199.                     "DirectDraw Spinning Cube", MB_OK);
  200.             goto fail;
  201.     }
  202.  
  203.     // Tell DirectInput that we want to receive data in mouse format
  204.     err = lpdiMouse->SetDataFormat(&c_dfDIMouse);
  205.  
  206.     if(err != DI_OK)
  207.     {
  208.             MessageBox(NULL, "Unable to Access DirectInput Device as a mouse",
  209.                     "DirectDraw Spinning Cube", MB_OK);
  210.             goto fail;
  211.     }
  212.  
  213.     // set desired access mode
  214.     err = lpdiMouse->SetCooperativeLevel(hwndApp, dwAccess);
  215.     if(err != DI_OK)
  216.     {
  217.         // Don't complain if asking for DISCL_NONEXCLUSIVE since
  218.         // not all systems support it.  (The caller will gracefully
  219.         // degrade.)
  220.         if (dwAccess & DISCL_EXCLUSIVE) {
  221.             MessageBox(NULL, "Unable to set cooperativity level",
  222.                     "DirectDraw Spinning Cube", MB_OK);
  223.         }
  224.         goto fail;
  225.     }
  226.  
  227.     return TRUE;
  228.  
  229. fail:;
  230.     if (lpdiMouse)      lpdiMouse->Release(),       lpdiMouse = 0;
  231.     return FALSE;
  232. }
  233.  
  234. /**************************************************************************
  235.   InitDInput
  236.  **************************************************************************/
  237. BOOL InitDInput(void)
  238. {
  239.     HRESULT        err;
  240.     GUID     guid = GUID_SysMouse;
  241.     // Need DirectX 3 or better
  242.     err = DirectInputCreate(hInstApp, 0x0300, &lpdi, NULL);
  243.  
  244.     if(err != DI_OK)
  245.     {
  246.         MessageBox(NULL, "Unable to Create DirectInput Object",
  247.             "DirectDraw Spinning Cube", MB_OK);
  248.         return FALSE;
  249.     }
  250.  
  251.         // Create a mouse for zooming.  Zooming is done non-exclusively.
  252.         if (!CreateMouse(guid, lpdiZoom, DISCL_NONEXCLUSIVE | DISCL_FOREGROUND))
  253.         {
  254.                 /* Note!  This is not a fatal error */
  255.         } else {
  256.  
  257.             if (fAppActive) lpdiZoom->Acquire();
  258.  
  259.         }
  260.  
  261.         // Create a mouse for rotation.  Rotation is done exclusively.
  262.         if (!CreateMouse(guid, lpdiRot, DISCL_EXCLUSIVE | DISCL_FOREGROUND))
  263.         {
  264.                 goto fail;
  265.         }
  266.  
  267.         // if we get here, all DirectInput objects were created ok
  268.     return TRUE;
  269.  
  270. fail:
  271.         if (lpdiZoom)  lpdiZoom->Release(), lpdiZoom = NULL;
  272.         if (lpdiRot)   lpdiRot ->Release(), lpdiRot  = NULL;
  273.         if (lpdi)      lpdi    ->Release(), lpdi     = NULL;
  274.         return FALSE;
  275.  
  276. }
  277.  
  278.  
  279. /**************************************************************************
  280.   DDSetMode
  281.  **************************************************************************/
  282.  
  283. BOOL DDSetMode(int width, int height, int bpp)
  284. {
  285.     HRESULT err;
  286.  
  287.     err = dd->SetDisplayMode(width, height, bpp);
  288.  
  289.     if (err != DD_OK)
  290.         return FALSE;
  291.  
  292.     ScreenSize.cx = width;
  293.     ScreenSize.cy = height;
  294.  
  295.     // get rid of any previous surfaces.
  296.     if (BackBuffer)  BackBuffer->Release(),     BackBuffer = NULL;
  297.     if (FrontBuffer) FrontBuffer->Release(),    FrontBuffer = NULL;
  298.     if (Palette)     Palette->Release(),        Palette = NULL;
  299.  
  300.     //
  301.     // Create surfaces
  302.     //
  303.     // what we want is a tripple buffered surface in video memory
  304.     // so we try to create this first.
  305.     //
  306.     // if we cant get a triple buffered surface, we try again
  307.     // for a double buffered surface (still must be in video memory)
  308.     //
  309.     // if we cant get a double buffered surface, we try for a double
  310.     // buffered surface not being specific about video memory, we will
  311.     // get back a main-memory surface, that work use HW page flipping
  312.     // but at least we run.
  313.     //
  314.     // NOTE you need to recreate the surfaces for a new display mode
  315.     // they wont work when/if the mode is changed.
  316.     //
  317.     DDSURFACEDESC ddsd;
  318.  
  319.     ZeroMemory(&ddsd, sizeof(ddsd));
  320.     ddsd.dwSize = sizeof(ddsd);
  321.     ddsd.dwFlags = DDSD_CAPS | DDSD_BACKBUFFERCOUNT;
  322.     ddsd.dwBackBufferCount = 2;
  323.     ddsd.ddsCaps.dwCaps = DDSCAPS_PRIMARYSURFACE |
  324.                           DDSCAPS_FLIP |
  325.                           DDSCAPS_COMPLEX |
  326.                           DDSCAPS_VIDEOMEMORY;
  327.  
  328.     // try to get a triple buffered video memory surface.
  329.     err = dd->CreateSurface(&ddsd, &FrontBuffer, NULL);
  330.  
  331.     if (err != DD_OK)
  332.     {
  333.         // try to get a double buffered video memory surface.
  334.         ddsd.dwBackBufferCount = 1;
  335.         err = dd->CreateSurface(&ddsd, &FrontBuffer, NULL);
  336.     }
  337.  
  338.     if (err != DD_OK)
  339.     {
  340.         // settle for a main memory surface.
  341.         ddsd.ddsCaps.dwCaps &= ~DDSCAPS_VIDEOMEMORY;
  342.         err = dd->CreateSurface(&ddsd, &FrontBuffer, NULL);
  343.     }
  344.  
  345.     if (err != DD_OK)
  346.         return FALSE;
  347.  
  348.     // get a pointer to the back buffer
  349.     DDSCAPS caps;
  350.     caps.dwCaps = DDSCAPS_BACKBUFFER;
  351.     err = FrontBuffer->GetAttachedSurface(&caps, &BackBuffer);
  352.  
  353.     if (err != DD_OK)
  354.         return FALSE;
  355.  
  356.     // create a palette if we are in a paletized display mode.
  357.     //
  358.     // NOTE because we want to be able to show dialog boxs and
  359.     // use our menu, we leave the windows reserved colors as is
  360.     // so things dont look ugly.
  361.     //
  362.     // palette is setup like so:
  363.     //
  364.     //      10      windows system colors
  365.     //      64      red wash
  366.     //      64      grn wash
  367.     //      64      blu wash
  368.     //
  369.     PALETTEENTRY ape[256];
  370.     HDC hdc = GetDC(NULL);
  371.     if (GetDeviceCaps(hdc, RASTERCAPS) & RC_PALETTE)
  372.     {
  373.         // get the current windows colors.
  374.         GetSystemPaletteEntries(hdc, 0, 256, ape);
  375.  
  376.         // make a red, grn, and blu wash for our cube.
  377.         for (int i=0; i<64; i++)
  378.         {
  379.             ape[10+64*0+i].peRed   = i * 255/63;
  380.             ape[10+64*0+i].peGreen = 0;
  381.             ape[10+64*0+i].peBlue  = 0;
  382.  
  383.             ape[10+64*1+i].peRed   = 0;
  384.             ape[10+64*1+i].peGreen = i * 255/63;
  385.             ape[10+64*1+i].peBlue  = 0;
  386.  
  387.             ape[10+64*2+i].peRed   = 0;
  388.             ape[10+64*2+i].peGreen = 0;
  389.             ape[10+64*2+i].peBlue  = i * 255/63;
  390.         }
  391.  
  392.         // create the palette.
  393.         err = dd->CreatePalette(DDPCAPS_8BIT, ape, &Palette, NULL);
  394.  
  395.         if (err == DD_OK)
  396.         {
  397.             FrontBuffer->SetPalette(Palette);
  398.         }
  399.     }
  400.     ReleaseDC(NULL, hdc);
  401.  
  402.     if (AppFont)
  403.         DeleteObject(AppFont);
  404.  
  405.     AppFont = CreateFont(width < 640 ? 24 : 48,
  406.         0, 0, 0, FW_NORMAL, FALSE, FALSE, FALSE,
  407.         ANSI_CHARSET,
  408.         OUT_DEFAULT_PRECIS,
  409.         CLIP_DEFAULT_PRECIS,
  410.         NONANTIALIASED_QUALITY,
  411.         VARIABLE_PITCH,
  412.         "Comic Sans MS");
  413.  
  414.     return TRUE;
  415. }
  416.  
  417.  
  418. /**************************************************************************
  419.   DDTerm
  420.  **************************************************************************/
  421.  
  422. void DDTerm()
  423. {
  424.     if (BackBuffer)  BackBuffer->Release(),     BackBuffer = NULL;
  425.     if (FrontBuffer) FrontBuffer->Release(),    FrontBuffer = NULL;
  426.     if (Palette)     Palette->Release(),        Palette = NULL;
  427.     if (dd)          dd->Release(),             dd = NULL;
  428.     if (lpdi)        lpdi->Release(),           lpdi = NULL;
  429. }
  430.  
  431. /**************************************************************************
  432.   ModeCallback
  433.  **************************************************************************/
  434.  
  435. #define MAKEMENUITEMDATA(width, height, bpp) \
  436.         (width) | ((height) << 12) | ((bpp) << 24)
  437.  
  438. HRESULT CALLBACK ModeCallback(LPDDSURFACEDESC pdds, LPVOID lParam)
  439. {
  440.     HMENU hmenu = (HMENU)lParam;
  441.     char ach[80];
  442.     int n;
  443.     int width  = pdds->dwWidth;
  444.     int height = pdds->dwHeight;
  445.     int bpp    = pdds->ddpfPixelFormat.dwRGBBitCount;
  446.  
  447.     n = GetMenuItemCount(hmenu);
  448.     wsprintf(ach,"%dx%dx%d",width,height,bpp);
  449.     AppendMenu(hmenu,MF_STRING,MENU_MODE+n,ach);
  450.  
  451.     MENUITEMINFO mii;
  452.  
  453.     // pack the mode info into a DWORD and set the extra item data.
  454.     mii.cbSize = sizeof(mii);
  455.     mii.fMask = MIIM_DATA;
  456.     mii.dwItemData = MAKEMENUITEMDATA(width, height, bpp);
  457.     SetMenuItemInfo(hmenu, MENU_MODE+n, MF_BYCOMMAND, &mii);
  458.  
  459.     //return S_TRUE to stop enuming modes, S_FALSE to continue
  460.     return S_FALSE;
  461. }
  462.  
  463. /**************************************************************************
  464.   AppInit
  465.  
  466.   Description:
  467.     This is called when the application is first loaded. It initializes
  468.   all variables, registers the window class, and creates the main app
  469.   window.
  470.  **************************************************************************/
  471.  
  472. BOOL AppInit(HINSTANCE hInst,HINSTANCE hPrev,int sw,LPSTR szCmdLine)
  473. {
  474.   WNDCLASS cls;
  475.  
  476.   /* Save instance handle for DialogBoxes */
  477.   hInstApp = hInst;
  478.  
  479.   if (!hPrev)
  480.   {
  481.     //***  Register a class for the main application window
  482.     cls.hCursor        = LoadCursor(0,IDC_ARROW);
  483.  
  484.     //*** Just for fun, we'll draw our own spinning cube icon.
  485.     cls.hIcon          = LoadIcon(hInst, "AppIcon");
  486.     cls.lpszMenuName   = "AppMenu";
  487.     cls.lpszClassName  = szAppName;
  488.     cls.hbrBackground  = (HBRUSH)GetStockObject(BLACK_BRUSH);
  489.     cls.hInstance      = hInst;
  490.     cls.style          = CS_VREDRAW | CS_HREDRAW;
  491.     cls.lpfnWndProc    = (WNDPROC)AppWndProc;
  492.     cls.cbClsExtra     = 0;
  493.     cls.cbWndExtra     = 0;
  494.  
  495.     if (!RegisterClass(&cls))
  496.       return FALSE;
  497.   }
  498.  
  499.   hAccelApp = LoadAccelerators(hInst, "AppAccel");
  500.  
  501.   //*** Set and normalize the light source
  502.   LightSourceDirection = vector_4(50, 30, -15);
  503.   LightSourceDirection.Normalize();
  504.  
  505.   //*** Distance to view plane:
  506.   ViewPerspective.SetElement(3, 2, 1/300.0);
  507.   ViewPerspective.SetElement(3, 3, 0);
  508.  
  509.   //*** Viewport scaling - some arbitrary number like 3.5 will do
  510.   ViewPerspective.SetElement(0, 0, 3.5);
  511.   ViewPerspective.SetElement(1, 1, 3.5);
  512.  
  513.   //*** Calculate the initial normals and shades
  514.   TransformCube(CubeTransform);
  515.  
  516.   //*** Then generate an interesting rotation for the spin
  517.   CubeTransform.ConcatenateYRotation(6.0);
  518.   CubeTransform.ConcatenateXRotation(3.5);
  519.   CubeTransform.ConcatenateZRotation(2.0);
  520.  
  521.   hwndApp = CreateWindowEx(
  522.                   WS_EX_APPWINDOW,
  523.                   szAppName,           // Class name
  524.                   szAppName,           // Caption
  525.                   WS_POPUP |
  526.                   WS_SYSMENU |
  527.                   WS_CAPTION,
  528.           0, 0,            // Position
  529.           640,480,           // Size
  530.                   0,                   // Parent window (no parent)
  531.                   0,                   // use class menu
  532.                   hInst,               // handle to window instance
  533.                   0                    // no params to pass on
  534.                   );
  535.   ShowWindow(hwndApp,sw);
  536.   UpdateWindow(hwndApp);
  537.  
  538.   if (!DDInit())
  539.     return FALSE;
  540.  
  541.   // Enumerate all posible display modes, and stick them in our menu.
  542.   // we use the extra item DWORD of a menu item to store the mode info
  543.   HMENU hmenu = CreatePopupMenu();
  544.   dd->EnumDisplayModes(0,NULL,(LPVOID)hmenu,ModeCallback);
  545.   AppendMenu(GetMenu(hwndApp),MF_POPUP,(UINT)hmenu,"&Modes");
  546.  
  547.   if (!DDSetMode(640,480,8) &&
  548.       !DDSetMode(640,480,16))
  549.     return FALSE;
  550.  
  551.   return TRUE;
  552. }
  553.  
  554. /**************************************************************************
  555.   WinMain
  556.  
  557.   Description:
  558.     The main procedure for the App.  After initializing, it just goes
  559.   into a message-processing loop until it gets a WM_QUIT message.
  560.  **************************************************************************/
  561.  
  562. int PASCAL WinMain(HINSTANCE hInst, HINSTANCE hPrev, LPSTR szCmdLine, int sw)
  563. {
  564.   MSG     msg;
  565.  
  566.   //*** Call initialization procedure
  567.   if (!AppInit(hInst,hPrev,sw,szCmdLine))
  568.     return FALSE;
  569.  
  570.     //*** Polling messages from event queue until quit
  571.   for (;;)
  572.   {
  573.     if (PeekMessage(&msg, 0, 0, 0, PM_REMOVE))
  574.     {
  575.       if (msg.message == WM_QUIT)
  576.         break;
  577.  
  578.       if (!hwndApp || !TranslateAccelerator(hwndApp, hAccelApp, &msg))
  579.       {
  580.         TranslateMessage(&msg);
  581.         DispatchMessage(&msg);
  582.       }
  583.     }
  584.     else
  585.     {
  586.       if (AppIdle())
  587.         WaitMessage();
  588.     }
  589.   }
  590.  
  591.   DDTerm();
  592.   return msg.wParam;
  593. }
  594.  
  595. /**************************************************************************
  596.   AppPause
  597.  
  598.  **************************************************************************/
  599.  
  600. void AppPause(BOOL f)
  601. {
  602.     if (f)
  603.     {
  604.         DDSCAPS caps;
  605.         FrontBuffer->GetCaps(&caps);
  606.  
  607.         // if we are in ModeX go back to a windows mode
  608.         // so we can see the menu or dialog box.
  609.  
  610.         if (caps.dwCaps & DDSCAPS_MODEX)
  611.         {
  612.             DDSetMode(640,480,8);
  613.         }
  614.  
  615.         // turn off rotation while paused
  616.         if (lpdiRot) lpdiRot->Unacquire();
  617.         fMouseAcquired = FALSE;
  618.  
  619.         fAppPaused = TRUE;
  620.         dd->FlipToGDISurface();
  621.         DrawMenuBar(hwndApp);
  622.         RedrawWindow(hwndApp, NULL, NULL, RDW_FRAME);
  623.     }
  624.     else
  625.     {
  626.         fAppPaused = FALSE;
  627.  
  628.     }
  629. }
  630.  
  631. /**************************************************************************
  632.   MagnifyCube
  633.  
  634.   Description:
  635.     Magnify the cube the indicated number of times.  A negative number
  636.     makes it smaller.
  637.  **************************************************************************/
  638.  
  639. void MagnifyCube(double times)
  640. {
  641.     matrix_4x4 m;
  642.     double factor = pow(1.5, times);
  643.     m.SetElement(0,0,factor);
  644.     m.SetElement(1,1,factor);
  645.     m.SetElement(2,2,factor);
  646.     TransformCube(m);
  647. }
  648.  
  649. /**************************************************************************
  650.   AppIdle
  651.  
  652.   return TRUE if the app is idle
  653.   return FALSE if the app is not idle.
  654.  
  655.   Description:
  656.  **************************************************************************/
  657.  
  658. BOOL AppIdle()
  659. {
  660.   DIMOUSESTATE dims;
  661.  
  662.   //*** Spin while the app is active, lbutton is up, and spinning is on.
  663.  
  664.   //*** Spin while the app is iconized.
  665.   if (fAppActive && !fAppPaused)
  666.   {
  667.     //*** If the app is active, spin the cube and redraw
  668.  
  669.     // See if any zooming needs to be done.
  670.     if(lpdiZoom &&
  671.        lpdiZoom->GetDeviceState(sizeof(DIMOUSESTATE), &dims) == DI_OK) {
  672.         // 240 units of motion in the Z-axis equals one unit of
  673.         // magnification / shrinkage.
  674.         if(dims.lZ) {
  675.            MagnifyCube(dims.lZ / 240.0);
  676.         }
  677.     }
  678.  
  679.     if(fMouseAcquired)
  680.      {
  681.         //** If we have the mouse acquired...
  682.  
  683.         // user spins cube if GetDeviceState succeeds and if the left button (button 0) is held
  684.                 if(lpdiRot->GetDeviceState(sizeof(DIMOUSESTATE), &dims) == DI_OK)
  685.         {
  686.          if(dims.rgbButtons[0] & 0x80)
  687.          {
  688.             if(dims.lX || dims.lY)
  689.             {
  690.                matrix_4x4 Movement;
  691.                Movement.ConcatenateYRotation(dims.lX);
  692.                Movement.ConcatenateXRotation(dims.lY);
  693.                TransformCube(Movement);
  694.             }
  695.  
  696.          }
  697.          else
  698.            {
  699.                // unacquire the mouse
  700.                            lpdiRot->Unacquire();
  701.                fMouseAcquired = FALSE;
  702.            }
  703.       }
  704.     }
  705.     else
  706.     {
  707.          TransformCube(CubeTransform);
  708.     }
  709.     RenderFrame();
  710.     return FALSE;
  711.   }
  712.   else
  713.   {
  714.     //*** Don't do anything when not the active app
  715.     return TRUE;
  716.   }
  717. }
  718.  
  719. /**************************************************************************
  720.   RenderFrame
  721.  
  722.   render the frame into the back buffer and do a page flip.
  723.  
  724.   things to NOTE:
  725.  
  726.     we use the blter to clear the backbuffer, this usualy is a big
  727.     win blters are real fast.
  728.  
  729.     we use GDI to draw the frame rate, and info text
  730.  
  731.     we either use GDI to draw the faces of the cube, or our own code
  732.     based on the fDrawWithGDI global variable.
  733.  
  734.  **************************************************************************/
  735.  
  736. int FrameRate;
  737. int FrameCount;
  738. int FrameCount0;
  739. DWORD FrameTime;
  740. DWORD FrameTime0;
  741.  
  742. void RenderFrame()
  743. {
  744.   HDC hdc;
  745.  
  746.   //*** always need to handle DDERR_SURFACELOST, this will happen
  747.   //*** when we get switched away from.
  748.  
  749.   if (FrontBuffer->IsLost() == DDERR_SURFACELOST)
  750.     FrontBuffer->Restore();
  751.  
  752.   //*** use the blter to do a color fill to clear the back buffer
  753.  
  754.   DDBLTFX ddbltfx;
  755.   ddbltfx.dwSize = sizeof(ddbltfx);
  756.   ddbltfx.dwFillColor = 0;
  757.   BackBuffer->Blt(NULL,NULL,NULL,DDBLT_COLORFILL | DDBLT_WAIT,&ddbltfx);
  758.  
  759.   //*** based on the fDrawWithGDI global variable, we either
  760.   //*** render the polygons ourself or let GDI do it
  761.  
  762.   BOOL fGDI = fDrawWithGDI;
  763.  
  764.   //*** render the cube with our own code.
  765.   //*** we need to do this outside of the GetDC, because we cant
  766.   //*** lock a buffer while we have a DC on it.
  767.   //*** if ProjectAndDrawCube returns FALSE it did not want to
  768.   //*** draw for some reason, so go ahead and use GDI
  769.  
  770.   if (!fGDI)
  771.      fGDI = !ProjectAndDrawCube(BackBuffer, ScreenSize.cx/2, ScreenSize.cy/2);
  772.  
  773.   if (BackBuffer->GetDC(&hdc) == DD_OK)
  774.   {
  775.     //*** use GDI to draw the cube.
  776.     if (fGDI)
  777.       ProjectAndDrawCube(hdc, ScreenSize.cx/2, ScreenSize.cy/2);
  778.  
  779.     //*** draw stats, like frame number and frame rate
  780.  
  781.     char ach[128];
  782.     int len;
  783.     static char szHelp[] = "F10=Menu F7=Smaller F8=Larger";
  784.  
  785.     SetBkMode(hdc, TRANSPARENT);
  786.     SelectObject(hdc, AppFont);
  787.  
  788.     len = wsprintf(ach, "FPS %02d Frame %05d", FrameRate, FrameCount);
  789.  
  790.     SetTextColor(hdc, RGB(255, 255, 0));
  791.     TextOut(hdc, 0, 0, ach, len);
  792.     TextOut(hdc, 0, ScreenSize.cy-(ScreenSize.cx<640 ? 24:48),szHelp,sizeof(szHelp)-1);
  793.  
  794.     BackBuffer->ReleaseDC(hdc);
  795.   }
  796.  
  797.   //*** we have rendered the backbuffer, call flip so we can see it
  798.   FrontBuffer->Flip(NULL, DDFLIP_WAIT);
  799.  
  800.   FrameCount++;
  801.   FrameTime = timeGetTime();
  802.  
  803.   if (FrameTime - FrameTime0 > 1000)
  804.   {
  805.     FrameRate = (FrameCount - FrameCount0) * 1000 / (FrameTime - FrameTime0);
  806.     FrameTime0 = FrameTime;
  807.     FrameCount0 = FrameCount;
  808.   }
  809. }
  810.  
  811. /**************************************************************************
  812.   AppInitMenuPopup
  813.  
  814.     If it's the Cube menu, then check or uncheck the GDI item accordingly.
  815.  
  816.     If it's the mode list popup, then add a MENUBARBREAK as necessary.
  817.  
  818.  **************************************************************************/
  819.  
  820. void AppInitMenuPopup(HWND hwnd, HMENU hmenuPopup, UINT uPos, BOOL fSys)
  821. {
  822.     if (fSys) return;       /* Don't mess with the sysmenu */
  823.  
  824.     switch (uPos) {
  825.     case 0:                 /* Cube menu */
  826.         CheckMenuItem(hmenuPopup, MENU_GDI,
  827.                       fDrawWithGDI ? MF_CHECKED : MF_UNCHECKED);
  828.         break;
  829.  
  830.     case 1:                 /* Mode menu */
  831.         /*
  832.          *  Compute how many menu items fit on the screen.
  833.          *
  834.          *  Note that we use ScreenSize instead of SM_CYSCREEN.
  835.          *  This allows us to do the right thing in the face of
  836.          *  multiple monitors.  (ScreenSize is the size of our
  837.          *  monitor.)
  838.          */
  839.         int cmi;
  840.  
  841.         cmi = GetMenuItemCount(hmenuPopup);
  842.         if (cmi) {
  843.  
  844.             RECT rcClient;
  845.             GetClientRect(hwnd, &rcClient);
  846.  
  847.             RECT rc;
  848.             GetMenuItemRect(hwnd, hmenuPopup, 0, &rc);
  849.  
  850.             int dyMenuItem = rc.bottom - rc.top;
  851.  
  852.             /*
  853.              *  Aargh.  If the menu has never appeared yet, USER
  854.              *  returns an empty rectangle.  DON'T DIVIDE BY ZERO!
  855.              *
  856.              *  In such case, we use the height of the menu bar.
  857.              *  Not perfect, but close enough.  And it happens only
  858.              *  once.
  859.              */
  860.             if (dyMenuItem == 0) {
  861.                 dyMenuItem = GetSystemMetrics(SM_CYMENU);
  862.             }
  863.  
  864.             /*
  865.              *  Aargh.  You can't change MF_MENUBARBREAK without
  866.              *  also changing the text, so we have to get the
  867.              *  old text and set it back.
  868.              *
  869.              *  While we're here, we may as well put a check-box
  870.              *  next to the item that matches our current screen res.
  871.              */
  872.  
  873.             HDC hdc = GetDC(0);
  874.  
  875.             DWORD dwItemDataMatch = MAKEMENUITEMDATA(
  876.                             GetDeviceCaps(hdc, HORZRES),
  877.                             GetDeviceCaps(hdc, VERTRES),
  878.                             GetDeviceCaps(hdc, BITSPIXEL) *
  879.                             GetDeviceCaps(hdc, PLANES));
  880.             ReleaseDC(0, hdc);
  881.  
  882.  
  883.             MENUITEMINFO mii;
  884.             TCHAR tsz[80];
  885.  
  886.             mii.cbSize = sizeof(mii);
  887.             mii.fMask = MIIM_TYPE | MIIM_DATA | MIIM_STATE;
  888.             mii.dwTypeData = tsz;
  889.  
  890.             /*
  891.              *  Compute the number of MI's that fit in our client area.
  892.              *  Note: Client area, not screen size.  This ensures that
  893.              *  the menu pops in a reasonable location.
  894.              */
  895.  
  896.             int cmiScreen = rcClient.bottom / dyMenuItem;
  897.  BOOL bModeMatch = FALSE;
  898.             for (int imi = 0; imi < cmi; imi++) {
  899.                 mii.cch = sizeof(tsz) / sizeof(tsz[0]);
  900.                 if (GetMenuItemInfo(hmenuPopup, imi, MF_BYPOSITION, &mii)) {
  901.                     if (imi > 0 && imi % cmiScreen == 0) {
  902.                         mii.fType |= MFT_MENUBARBREAK;
  903.                     } else {
  904.                         mii.fType &= ~MFT_MENUBARBREAK;
  905.                     }
  906.  
  907.                     if (mii.dwItemData == dwItemDataMatch) {
  908.                         mii.fState |= MFS_CHECKED;
  909. bModeMatch = TRUE;
  910.                     } else {
  911.                         mii.fState &= ~MFS_CHECKED;
  912.                     }
  913.  
  914.                     SetMenuItemInfo(hmenuPopup, imi, MF_BYPOSITION, &mii);
  915.                 }
  916.             }
  917. if(bModeMatch == FALSE) _asm int 3
  918.  
  919.         }
  920.         break;
  921.     }
  922.  
  923. }
  924.  
  925. /**************************************************************************
  926.   AppWndProc
  927.  
  928.   Description:
  929.     Main window proc. Standard Windows fare.
  930.  **************************************************************************/
  931.  
  932. LONG CALLBACK AppWndProc(HWND hwnd,UINT msg,WPARAM wParam,LPARAM lParam)
  933. {
  934.   RECT Rect;
  935.  
  936.   switch (msg)
  937.   {
  938.     case WM_CREATE:
  939.       break;
  940.  
  941.     case WM_ACTIVATEAPP:
  942.       //*** Keep track of whether or not the app is in the foreground
  943.       fAppActive = (BOOL)wParam;
  944.           // re-acquire the zooming controller when we are activated
  945.           if (fAppActive) {
  946.                 if (lpdiZoom) lpdiZoom->Acquire();
  947.           } else{           // unacquire everything if app is not active
  948.                 if (lpdiZoom) lpdiZoom->Unacquire();
  949.                 if (lpdiRot) lpdiRot->Unacquire();
  950.         fMouseAcquired = FALSE;
  951.       }
  952.       break;
  953.  
  954.     case WM_SETCURSOR:
  955.       if (fAppActive && !fAppPaused)
  956.       {
  957.         SetCursor(NULL);
  958.         return 1;
  959.       }
  960.       break;
  961.  
  962.     case WM_ENTERMENULOOP:
  963.       AppPause(TRUE);
  964.       break;
  965.  
  966.     case WM_EXITMENULOOP:
  967.       AppPause(FALSE);
  968.       break;
  969.  
  970.     case WM_INITMENUPOPUP:
  971.         AppInitMenuPopup(hwnd, (HMENU)wParam,
  972.                          (UINT)LOWORD(lParam), (BOOL)HIWORD(lParam));
  973.         break;
  974.  
  975.     case WM_COMMAND:
  976.       switch(LOWORD(wParam))
  977.       {
  978.         case MENU_ABOUT:
  979.           AppPause(TRUE);
  980.           DialogBox(hInstApp, "AppAbout", hwnd, (DLGPROC)AppAbout);
  981.           AppPause(FALSE);
  982.           break;
  983.  
  984.         case MENU_EXIT:
  985.           PostMessage(hwnd, WM_CLOSE, 0, 0L);
  986.           break;
  987.  
  988.         case MENU_LARGER:
  989.           MagnifyCube(+1.0);
  990.           break;
  991.  
  992.         case MENU_SMALLER:
  993.           MagnifyCube(-1.0);
  994.           break;
  995.  
  996.         case MENU_GDI:
  997.           fDrawWithGDI = !fDrawWithGDI;
  998.           break;
  999.       }
  1000.       if (LOWORD(wParam) >= MENU_MODE && LOWORD(wParam) < MENU_MODE+100)
  1001.       {
  1002.         MENUITEMINFO mii;
  1003.  
  1004.         mii.cbSize = sizeof(mii);
  1005.         mii.fMask = MIIM_DATA;
  1006.         GetMenuItemInfo(GetMenu(hwnd), LOWORD(wParam), MF_BYCOMMAND, &mii);
  1007.  
  1008.         DDSetMode(
  1009.             (mii.dwItemData >> 0)  & 0xFFF,
  1010.             (mii.dwItemData >> 12) & 0xFFF,
  1011.             (mii.dwItemData >> 24) & 0x0FF);
  1012.       }
  1013.       return 0L;
  1014.  
  1015.     case WM_DESTROY:
  1016.       // clean up DirectInput objects
  1017.           if (fMouseAcquired) lpdiRot->Unacquire();
  1018.           if (lpdiZoom)  lpdiZoom->Release(), lpdiZoom = NULL;
  1019.           if (lpdiRot)   lpdiRot ->Release(), lpdiRot  = NULL;
  1020.           if (lpdi)      lpdi    ->Release(), lpdi     = NULL;
  1021.  
  1022.       hwndApp = NULL;
  1023.       PostQuitMessage(0);
  1024.       break;
  1025.  
  1026.     case WM_PAINT:
  1027.       break;
  1028.  
  1029.     case WM_MOVE:
  1030.     case WM_SIZE:
  1031.     case WM_DISPLAYCHANGE:
  1032.       if (fAppActive && !IsIconic(hwnd))
  1033.       {
  1034.         SetRect(&Rect, 0, GetSystemMetrics(SM_CYCAPTION), GetSystemMetrics(SM_CXSCREEN), GetSystemMetrics(SM_CYSCREEN));
  1035.         AdjustWindowRectEx(&Rect, WS_POPUP | WS_CAPTION, FALSE, 0);
  1036.         SetWindowPos(hwnd, NULL, Rect.left, Rect.top, Rect.right-Rect.left, Rect.bottom-Rect.top, SWP_NOACTIVATE | SWP_NOZORDER);
  1037.       }
  1038.       break;
  1039.  
  1040.     case WM_LBUTTONDOWN:
  1041.     case WM_RBUTTONDOWN:
  1042.     case WM_MBUTTONDOWN:
  1043.           if(lpdiRot->Acquire() == DI_OK)
  1044.      {
  1045.         fMouseAcquired = TRUE;
  1046.      }
  1047.       else
  1048.       {
  1049.         // not acquired, mouse will not do anything
  1050.       }
  1051.       break;
  1052.  
  1053.   }
  1054.  
  1055.   return DefWindowProc(hwnd,msg,wParam,lParam);
  1056. }
  1057.  
  1058. /**************************************************************************
  1059.   TransformCube
  1060.  
  1061.   Description:
  1062.     Transforms the cube vertices by the current rotation matrix.
  1063.     Recalculates normals and flat shade values for the
  1064.   directional light source.
  1065.  **************************************************************************/
  1066.  
  1067. void TransformCube(matrix_4x4 const &Transform)
  1068. {
  1069.   int i;
  1070.  
  1071.   //*** Transform the cube by the matrix
  1072.   for (i = 0; i < 8; ++i)
  1073.     CubeVertices[i] = Transform * CubeVertices[i];
  1074.  
  1075.   //*** Recalculate normals and shades
  1076.   for (i = 0; i < 6; ++i)
  1077.   {
  1078.     //*** Normals are perpendicular to two edges of the cube
  1079.     vector_4 Edge1, Edge2;
  1080.     Edge1 = CubeVertices[CubeFaces[i][1]] - CubeVertices[CubeFaces[i][0]];
  1081.     Edge2 = CubeVertices[CubeFaces[i][3]] - CubeVertices[CubeFaces[i][0]];
  1082.     CubeSurfaceNormals[i] = CrossProduct(Edge1, Edge2);
  1083.     CubeSurfaceNormals[i].Normalize();
  1084.  
  1085.     //*** Cosine shading based on the surface normal, clamped to [0, 1]
  1086.     real Shade = DotProduct(CubeSurfaceNormals[i], LightSourceDirection);
  1087.     Shade = Shade + AmbientLight;
  1088.     if (Shade < 0) Shade = 0;
  1089.     else if (Shade > 1.0) Shade = 1.0;
  1090.     CubeSurfaceShades[i] = Shade;
  1091.   }
  1092. }
  1093.  
  1094. /**************************************************************************
  1095.   ProjectAndDrawCube
  1096.  
  1097.   Description:
  1098.     Projects the cube vertices for the current viewpoint then culls
  1099.   in screen space and draws into the DC via GDI.
  1100.  **************************************************************************/
  1101.  
  1102. BOOL ProjectAndDrawCube(HDC hdc, int XOffset, int YOffset)
  1103. {
  1104.   //*** Create a viewing transform for the current eye position
  1105.   vector_4 ViewDirection = Origin - Viewpoint;
  1106.   ViewDirection.Normalize();
  1107.   view_transform View(Viewpoint, ViewDirection, Up);
  1108.  
  1109.   //*** Transform and project the vertices into screen space
  1110.   int i;
  1111.   POINT aScreenVertices[8];
  1112.   for (i = 0; i < 8; ++i)
  1113.   {
  1114.     point_4 Temp = View * CubeVertices[i];
  1115.     Temp = ViewPerspective * Temp;
  1116.     Temp.Homogenize();
  1117.  
  1118.     aScreenVertices[i].x = (int)Temp.GetX() + XOffset;
  1119.     aScreenVertices[i].y = (int)Temp.GetY() + YOffset;
  1120.   }
  1121.  
  1122.   SelectPen(hdc, GetStockPen(NULL_PEN));
  1123.  
  1124.   for (i = 0; i < 6; ++i)
  1125.   {
  1126.     //*** Standard culling operation based on the z value of the
  1127.     //*** cross product of the edges: are the vertices oriented in the
  1128.     //*** counterclockwise or clockwise direction?
  1129.     real v1 = aScreenVertices[ CubeFaces[i][2] ].x -
  1130.       aScreenVertices[ CubeFaces[i][1] ].x;
  1131.     real w1 = aScreenVertices[ CubeFaces[i][0] ].x -
  1132.       aScreenVertices[ CubeFaces[i][1] ].x;
  1133.     real v2 = aScreenVertices[ CubeFaces[i][2] ].y -
  1134.       aScreenVertices[ CubeFaces[i][1] ].y;
  1135.     real w2 = aScreenVertices[ CubeFaces[i][0] ].y -
  1136.       aScreenVertices[ CubeFaces[i][1] ].y;
  1137.     if ((v1*w2 - v2*w1) <= 0)
  1138.       continue;
  1139.  
  1140.     //*** Create a brush for the shaded face color using the selected dither
  1141.  
  1142.     HBRUSH hbr;
  1143.  
  1144.     //*** Get the shading colors
  1145.  
  1146.     int Red, Green, Blue;
  1147.  
  1148.     Red   = (int)(CubeColors[i][0] * CubeSurfaceShades[i]);
  1149.     Green = (int)(CubeColors[i][1] * CubeSurfaceShades[i]);
  1150.     Blue  = (int)(CubeColors[i][2] * CubeSurfaceShades[i]);
  1151.  
  1152.     //*** Create the dithered or PALETTERGB brush
  1153.  
  1154.     COLORREF cr;
  1155.  
  1156.     cr = RGB(Red, Green, Blue);
  1157.     hbr = CreateSolidBrush(cr);
  1158.  
  1159.     //*** Collect the correct points in an array
  1160.     POINT aQuadVertices[4];
  1161.     for (int j = 0; j < 4; ++j)
  1162.       aQuadVertices[j] = aScreenVertices[ CubeFaces[i][j] ];
  1163.  
  1164.     //*** Use GDI to draw the face
  1165.     hbr = SelectBrush(hdc, hbr);
  1166.     Polygon(hdc, aQuadVertices, 4);
  1167.     hbr = SelectBrush(hdc, hbr);
  1168.     DeleteObject(hbr);
  1169.   }
  1170.  
  1171.   return TRUE;
  1172. }
  1173.  
  1174. /**************************************************************************
  1175.   ProjectAndDrawCube
  1176.  
  1177.   Description:
  1178.     Projects the cube vertices for the current viewpoint then culls
  1179.   in screen space and draws them into a DirectDrawSurface via custom code
  1180.  **************************************************************************/
  1181.  
  1182. BOOL ProjectAndDrawCube(IDirectDrawSurface *pdds, int XOffset, int YOffset)
  1183. {
  1184.   //*** Lock the DirectDraw surface
  1185.   DDSURFACEDESC ddsd;
  1186.   ddsd.dwSize = sizeof(ddsd);
  1187.  
  1188.   if (pdds->Lock(NULL, &ddsd, DDLOCK_WAIT, NULL) != DD_OK)
  1189.     return FALSE;
  1190.  
  1191.   //*** This code only works for 8bpp
  1192.   if (ddsd.ddpfPixelFormat.dwRGBBitCount != 8)
  1193.   {
  1194.     pdds->Unlock(NULL);
  1195.     return FALSE;
  1196.   }
  1197.  
  1198.   //*** Create a viewing transform for the current eye position
  1199.   vector_4 ViewDirection = Origin - Viewpoint;
  1200.   ViewDirection.Normalize();
  1201.   view_transform View(Viewpoint, ViewDirection, Up);
  1202.  
  1203.   //*** Transform and project the vertices into screen space
  1204.   int i;
  1205.   POINT aScreenVertices[8];
  1206.   for (i = 0; i < 8; ++i)
  1207.   {
  1208.     point_4 Temp = View * CubeVertices[i];
  1209.     Temp = ViewPerspective * Temp;
  1210.     Temp.Homogenize();
  1211.  
  1212.     aScreenVertices[i].x = (int)Temp.GetX() + XOffset;
  1213.     aScreenVertices[i].y = (int)Temp.GetY() + YOffset;
  1214.  
  1215.     //*** !!! OUR CODE DOES NOT CLIP, SO FAIL IF WE NEED CLIPPING
  1216.     if (aScreenVertices[i].x < 0 || aScreenVertices[i].x >= ScreenSize.cx ||
  1217.         aScreenVertices[i].y < 0 || aScreenVertices[i].y >= ScreenSize.cy)
  1218.     {
  1219.       pdds->Unlock(NULL);
  1220.       return FALSE;
  1221.     }
  1222.   }
  1223.  
  1224.   for (i = 0; i < 6; ++i)
  1225.   {
  1226.     //*** Standard culling operation based on the z value of the
  1227.     //*** cross product of the edges: are the vertices oriented in the
  1228.     //*** counterclockwise or clockwise direction?
  1229.     real v1 = aScreenVertices[ CubeFaces[i][2] ].x -
  1230.       aScreenVertices[ CubeFaces[i][1] ].x;
  1231.     real w1 = aScreenVertices[ CubeFaces[i][0] ].x -
  1232.       aScreenVertices[ CubeFaces[i][1] ].x;
  1233.     real v2 = aScreenVertices[ CubeFaces[i][2] ].y -
  1234.       aScreenVertices[ CubeFaces[i][1] ].y;
  1235.     real w2 = aScreenVertices[ CubeFaces[i][0] ].y -
  1236.       aScreenVertices[ CubeFaces[i][1] ].y;
  1237.     if ((v1*w2 - v2*w1) <= 0)
  1238.       continue;
  1239.  
  1240.     //*** Get the shading color, palette is setup like so:
  1241.     //*** 10 system, 64 red, 64 green, 64 blue
  1242.  
  1243.     BYTE color;
  1244.  
  1245.     if (CubeColors[i][0] >= 128)
  1246.         color = (BYTE)(10 + 0*64 + (63 * CubeSurfaceShades[i]));
  1247.     else if (CubeColors[i][1] >= 128)
  1248.         color = (BYTE)(10 + 1*64 + (63 * CubeSurfaceShades[i]));
  1249.     else
  1250.         color = (BYTE)(10 + 2*64 + (63 * CubeSurfaceShades[i]));
  1251.  
  1252.     //*** Use code in tri.cpp draw the face
  1253.  
  1254.     Triangle8((BYTE*)ddsd.lpSurface, ddsd.lPitch,
  1255.         aScreenVertices[CubeFaces[i][0]],
  1256.         aScreenVertices[CubeFaces[i][1]],
  1257.         aScreenVertices[CubeFaces[i][2]],
  1258.         color);
  1259.  
  1260.     Triangle8((BYTE*)ddsd.lpSurface, ddsd.lPitch,
  1261.         aScreenVertices[CubeFaces[i][2]],
  1262.         aScreenVertices[CubeFaces[i][3]],
  1263.         aScreenVertices[CubeFaces[i][0]],
  1264.         color);
  1265.   }
  1266.  
  1267.   //*** Never ever forget to unlock!
  1268.   pdds->Unlock(NULL);
  1269.   return TRUE;
  1270. }
  1271.