Leaked source code of windows server 2003
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

1665 lines
51 KiB

  1. //-----------------------------------------------------------------------------
  2. // File: FlowerBox.cpp
  3. //
  4. // Desc: Fun screen saver
  5. // Original OpenGL version by Drew Bliss
  6. // Ported to Direct3D 8.0 by Mike Anderson
  7. //
  8. // Copyright (c) 2000-2001 Microsoft Corporation. All rights reserved.
  9. //-----------------------------------------------------------------------------
  10. #include <windows.h>
  11. #include <tchar.h>
  12. #include <d3d8.h>
  13. #include <d3dx8.h>
  14. #include <d3d8rgbrast.h>
  15. #include <stdio.h>
  16. #include <commctrl.h>
  17. #include <scrnsave.h>
  18. #include "d3dsaver.h"
  19. #include "FlowerBox.h"
  20. #include "Resource.h"
  21. #include "dxutil.h"
  22. #ifndef GL_FRONT
  23. #define GL_FRONT 0x0404
  24. #endif
  25. #ifndef GL_FRONT_AND_BACK
  26. #define GL_FRONT_AND_BACK 0x0408
  27. #endif
  28. // Minimum and maximum image sizes
  29. #define MINIMAGESIZE 10
  30. #define MAXIMAGESIZE 100
  31. // Color tables for checkboard, per-side and single color modes
  32. FLOAT base_checker_cols[MAXSIDES][NCCOLS][4] =
  33. {
  34. 1.0f, 0.0f, 0.0f, 1.0f,
  35. 0.0f, 1.0f, 0.0f, 1.0f,
  36. 0.0f, 1.0f, 0.0f, 1.0f,
  37. 0.0f, 0.0f, 1.0f, 1.0f,
  38. 0.0f, 0.0f, 1.0f, 1.0f,
  39. 1.0f, 0.0f, 1.0f, 1.0f,
  40. 1.0f, 0.0f, 1.0f, 1.0f,
  41. 0.0f, 1.0f, 1.0f, 1.0f,
  42. 0.0f, 1.0f, 1.0f, 1.0f,
  43. 1.0f, 1.0f, 0.0f, 1.0f,
  44. 1.0f, 1.0f, 0.0f, 1.0f,
  45. 0.5f, 0.5f, 1.0f, 1.0f,
  46. 0.5f, 0.5f, 1.0f, 1.0f,
  47. 1.0f, 0.5f, 0.5f, 1.0f,
  48. 1.0f, 0.5f, 0.5f, 1.0f,
  49. 1.0f, 0.0f, 0.0f, 1.0f
  50. };
  51. FLOAT checker_cols[MAXSIDES][NCCOLS][4];
  52. FLOAT base_side_cols[MAXSIDES][4] =
  53. {
  54. 1.0f, 0.0f, 0.0f, 1.0f,
  55. 0.0f, 1.0f, 0.0f, 1.0f,
  56. 0.0f, 0.0f, 1.0f, 1.0f,
  57. 1.0f, 0.0f, 1.0f, 1.0f,
  58. 0.0f, 1.0f, 1.0f, 1.0f,
  59. 1.0f, 1.0f, 0.0f, 1.0f,
  60. 0.5f, 0.5f, 1.0f, 1.0f,
  61. 1.0f, 0.5f, 0.5f, 1.0f
  62. };
  63. FLOAT side_cols[MAXSIDES][4];
  64. FLOAT base_solid_cols[4] =
  65. {
  66. 1.0f, 1.0f, 1.0f, 1.0f
  67. };
  68. FLOAT solid_cols[4];
  69. // Default configuration
  70. CONFIG config =
  71. {
  72. TRUE, FALSE, FALSE, TRUE, TRUE, MAXSUBDIV, ID_COL_PER_SIDE,
  73. (MAXIMAGESIZE+MINIMAGESIZE)/2, GEOM_CUBE, GL_FRONT
  74. };
  75. // A slider range
  76. typedef struct _RANGE
  77. {
  78. INT min_val;
  79. INT max_val;
  80. INT step;
  81. INT page_step;
  82. } RANGE;
  83. RANGE complexity_range = {MINSUBDIV, MAXSUBDIV, 1, 2};
  84. RANGE image_size_range = {MINIMAGESIZE, MAXIMAGESIZE, 1, 10};
  85. // Values to map a 2D point onto a 3D plane
  86. // Base point and axes to map X and Y coordinates onto
  87. struct PLANE_MAP
  88. {
  89. D3DXVECTOR3 base, x_axis, y_axis;
  90. PLANE_MAP(D3DXVECTOR3 tbase, D3DXVECTOR3 tx_axis, D3DXVECTOR3 ty_axis)
  91. { base = tbase; x_axis = tx_axis; y_axis = ty_axis; }
  92. PLANE_MAP(VOID) {};
  93. };
  94. // Data area used by the current geometry
  95. // Base points and generated points
  96. D3DXVECTOR3 pts[MAXPTS], npts[MAXPTS];
  97. MYVERTEX vertices[MAXPTS];
  98. // Scaling factor for spherical projection
  99. FLOAT vlen[MAXPTS];
  100. // Normals
  101. D3DXVECTOR3 normals[MAXPTS];
  102. // Vertex data indices
  103. WORD index[MAXPTS*2];
  104. // Triangle strip sizes
  105. INT strip_size[MAXSIDES*MAXSUBDIV];
  106. VOID InitCube(GEOMETRY *geom);
  107. VOID InitTetra(GEOMETRY *geom);
  108. VOID InitPyramids(GEOMETRY *geom);
  109. VOID InitCylinder(GEOMETRY *geom);
  110. VOID InitSpring(GEOMETRY *geom);
  111. GEOMETRY cube_geom = {InitCube};
  112. GEOMETRY tetra_geom = {InitTetra};
  113. GEOMETRY pyramids_geom = {InitPyramids};
  114. GEOMETRY cylinder_geom = {InitCylinder};
  115. GEOMETRY spring_geom = {InitSpring};
  116. GEOMETRY *geom_table[] =
  117. {
  118. &cube_geom,
  119. &tetra_geom,
  120. &pyramids_geom,
  121. &cylinder_geom,
  122. &spring_geom
  123. };
  124. #define BUF_SIZE 255
  125. TCHAR g_szSectName[BUF_SIZE];
  126. TCHAR g_szFname[BUF_SIZE];
  127. CFlowerBoxScreensaver* g_pMyFlowerBoxScreensaver = NULL;
  128. //-----------------------------------------------------------------------------
  129. // Name: WinMain()
  130. // Desc: Entry point to the program. Initializes everything, and goes into a
  131. // message-processing loop. Idle time is used to render the scene.
  132. //-----------------------------------------------------------------------------
  133. INT WINAPI WinMain( HINSTANCE hInst, HINSTANCE, LPSTR, INT )
  134. {
  135. HRESULT hr;
  136. CFlowerBoxScreensaver flowerboxSS;
  137. if( FAILED( hr = flowerboxSS.Create( hInst ) ) )
  138. {
  139. flowerboxSS.DisplayErrorMsg( hr );
  140. return 0;
  141. }
  142. return flowerboxSS.Run();
  143. }
  144. //-----------------------------------------------------------------------------
  145. // Name: CFlowerBoxScreensaver()
  146. // Desc: Constructor
  147. //-----------------------------------------------------------------------------
  148. CFlowerBoxScreensaver::CFlowerBoxScreensaver( )
  149. {
  150. g_pMyFlowerBoxScreensaver = this;
  151. LoadString( NULL, IDS_DESCRIPTION, m_strWindowTitle, 200 );
  152. m_bUseDepthBuffer = TRUE;
  153. lstrcpy( m_strRegPath, TEXT("Software\\Microsoft\\Screensavers\\Flowerbox") );
  154. m_xr = 0;
  155. m_yr = 0;
  156. m_zr = 0;
  157. m_sf = 0;
  158. m_sfi = 0;
  159. m_phase = 0.0f;
  160. m_pGeomCur = NULL;
  161. m_floatrect.xSize = 0.0f;
  162. InitCommonControls();
  163. }
  164. //-----------------------------------------------------------------------------
  165. // Name: RegisterSoftwareDevice()
  166. // Desc: This can register the D3D8RGBRasterizer or any other
  167. // pluggable software rasterizer.
  168. //-----------------------------------------------------------------------------
  169. HRESULT CFlowerBoxScreensaver::RegisterSoftwareDevice()
  170. {
  171. m_pD3D->RegisterSoftwareDevice( D3D8RGBRasterizer );
  172. return S_OK;
  173. }
  174. //-----------------------------------------------------------------------------
  175. // Name: FrameMove()
  176. // Desc: Called once per frame, the call is the entry point for animating
  177. // the scene.
  178. //-----------------------------------------------------------------------------
  179. HRESULT CFlowerBoxScreensaver::FrameMove()
  180. {
  181. // update floatrect
  182. RECT rcBounceBounds;
  183. if( m_floatrect.xSize == 0.0f )
  184. {
  185. // Initialize floatrect
  186. RECT rcBounds;
  187. DWORD dwParentWidth;
  188. DWORD dwParentHeight;
  189. rcBounds = m_rcRenderTotal;
  190. dwParentWidth = rcBounds.right - rcBounds.left;
  191. dwParentHeight = rcBounds.bottom - rcBounds.top;
  192. FLOAT sizeFact;
  193. FLOAT sizeScale;
  194. DWORD size;
  195. sizeScale = (FLOAT)config.image_size / 150.0f;
  196. // sizeFact = 0.25f + (0.5f * sizeScale); // range 25-75%
  197. // size = (DWORD) (sizeFact * ( ((FLOAT)(dwParentWidth + dwParentHeight)) / 2.0f ));
  198. sizeFact = 0.25f + (0.75f * sizeScale); // range 25-100%
  199. size = (DWORD) (sizeFact * ( dwParentWidth > dwParentHeight ? dwParentHeight : dwParentWidth ) );
  200. if( size > dwParentWidth )
  201. size = dwParentWidth;
  202. if( size > dwParentHeight )
  203. size = dwParentHeight;
  204. // Start floatrect centered on primary
  205. m_floatrect.xMin = rcBounds.left + (dwParentWidth - size) / 2.0f;
  206. m_floatrect.yMin = rcBounds.top + (dwParentHeight - size) / 2.0f;
  207. m_floatrect.xSize = (FLOAT)size;
  208. m_floatrect.ySize = (FLOAT)size;
  209. m_floatrect.xVel = 0.01f * (FLOAT) size;
  210. m_floatrect.yVel = 0.01f * (FLOAT) size;
  211. }
  212. rcBounceBounds = m_rcRenderTotal;
  213. FLOAT xMinOld = m_floatrect.xMin;
  214. FLOAT yMinOld = m_floatrect.yMin;
  215. m_floatrect.xMin += m_floatrect.xVel * 20.0f * m_fElapsedTime;
  216. m_floatrect.yMin += m_floatrect.yVel * 20.0f * m_fElapsedTime;
  217. if( m_floatrect.xVel < 0 && m_floatrect.xMin < rcBounceBounds.left ||
  218. m_floatrect.xVel > 0 && (m_floatrect.xMin + m_floatrect.xSize) > rcBounceBounds.right )
  219. {
  220. m_floatrect.xMin = xMinOld; // undo last move
  221. m_floatrect.xVel = -m_floatrect.xVel; // change direction
  222. }
  223. if( m_floatrect.yVel < 0 && m_floatrect.yMin < rcBounceBounds.top ||
  224. m_floatrect.yVel > 0 && (m_floatrect.yMin + m_floatrect.ySize) > rcBounceBounds.bottom )
  225. {
  226. m_floatrect.yMin = yMinOld; // undo last move
  227. m_floatrect.yVel = -m_floatrect.yVel; // change direction
  228. }
  229. if (config.spin)
  230. {
  231. m_xr += 3 * 20.0f * m_fElapsedTime;
  232. m_yr += 2 * 20.0f * m_fElapsedTime;
  233. }
  234. if (config.bloom)
  235. {
  236. m_sf += m_sfi * 20.0f * m_fElapsedTime;
  237. if (m_sfi > 0.0f && m_sf > m_pGeomCur->max_sf ||
  238. m_sfi < 0.0f && m_sf < m_pGeomCur->min_sf )
  239. {
  240. m_sfi = -m_sfi;
  241. }
  242. UpdatePts(m_pGeomCur, m_sf);
  243. }
  244. if (config.cycle_colors)
  245. {
  246. ComputeHsvColors();
  247. m_phase += (FLOAT)(2.5 * 20.0f * m_fElapsedTime * D3DX_PI/180.);
  248. }
  249. return S_OK;
  250. }
  251. //-----------------------------------------------------------------------------
  252. // Name: Render()
  253. // Desc: Called once per frame, the call is the entry point for 3d
  254. // rendering. This function sets up render states, clears the
  255. // viewport, and renders the scene.
  256. //-----------------------------------------------------------------------------
  257. HRESULT CFlowerBoxScreensaver::Render()
  258. {
  259. D3DVIEWPORT8 vp;
  260. // First, clear the entire back buffer to the background color
  261. vp.X = 0;
  262. vp.Y = 0;
  263. vp.Width = m_rcRenderCurDevice.right - m_rcRenderCurDevice.left;
  264. vp.Height = m_rcRenderCurDevice.bottom - m_rcRenderCurDevice.top;
  265. vp.MinZ = 0.0f;
  266. vp.MaxZ = 1.0f;
  267. m_pd3dDevice->SetViewport( &vp );
  268. m_pd3dDevice->Clear( 0L, NULL, D3DCLEAR_TARGET | D3DCLEAR_ZBUFFER, 0xff000000, 1.0f, 0L );
  269. // Now determine what part of the floatrect, if any, intersects the current screen
  270. RECT rcFloatThisScreen;
  271. RECT rcFloatThisScreenClipped;
  272. rcFloatThisScreen.left = (INT)m_floatrect.xMin;
  273. rcFloatThisScreen.top = (INT)m_floatrect.yMin;
  274. rcFloatThisScreen.right = rcFloatThisScreen.left + (INT)m_floatrect.xSize;
  275. rcFloatThisScreen.bottom = rcFloatThisScreen.top + (INT)m_floatrect.ySize;
  276. if( !IntersectRect(&rcFloatThisScreenClipped, &rcFloatThisScreen, &m_rcRenderCurDevice) )
  277. {
  278. return S_OK; // no intersection, so nothing further to render on this screen
  279. }
  280. // Convert rcFloatThisScreen from screen to window coordinates
  281. OffsetRect(&rcFloatThisScreen, -m_rcRenderCurDevice.left, -m_rcRenderCurDevice.top);
  282. OffsetRect(&rcFloatThisScreenClipped, -m_rcRenderCurDevice.left, -m_rcRenderCurDevice.top);
  283. // Now set up the viewport to render to the clipped rect
  284. vp.X = rcFloatThisScreenClipped.left;
  285. vp.Y = rcFloatThisScreenClipped.top;
  286. vp.Width = rcFloatThisScreenClipped.right - rcFloatThisScreenClipped.left;
  287. vp.Height = rcFloatThisScreenClipped.bottom - rcFloatThisScreenClipped.top;
  288. vp.MinZ = 0.0f;
  289. vp.MaxZ = 1.0f;
  290. m_pd3dDevice->SetViewport( &vp );
  291. // Now set up the projection matrix to only render the onscreen part of the
  292. // rect to the viewport
  293. D3DXMATRIX matProj;
  294. FLOAT l,r,b,t;
  295. l = -0.8f;
  296. r = 0.8f;
  297. b = 0.8f;
  298. t = -0.8f;
  299. FLOAT cxUnclipped = (rcFloatThisScreen.right + rcFloatThisScreen.left) / 2.0f;
  300. FLOAT cyUnclipped = (rcFloatThisScreen.bottom + rcFloatThisScreen.top) / 2.0f;
  301. l *= (rcFloatThisScreenClipped.left - cxUnclipped) / (rcFloatThisScreen.left - cxUnclipped);
  302. r *= (rcFloatThisScreenClipped.right - cxUnclipped) / (rcFloatThisScreen.right - cxUnclipped);
  303. t *= (rcFloatThisScreenClipped.top - cyUnclipped) / (rcFloatThisScreen.top - cyUnclipped);
  304. b *= (rcFloatThisScreenClipped.bottom - cyUnclipped) / (rcFloatThisScreen.bottom - cyUnclipped);
  305. D3DXMatrixPerspectiveOffCenterLH( &matProj, l, r, b, t, 2.0f, 5.0f );
  306. m_pd3dDevice->SetTransform( D3DTS_PROJECTION , &matProj );
  307. // Begin the scene
  308. if( SUCCEEDED( m_pd3dDevice->BeginScene() ) )
  309. {
  310. D3DXMATRIX mat, matx, maty, matz;
  311. D3DXMatrixRotationX( &matx, D3DXToRadian( (FLOAT)m_xr ) );
  312. D3DXMatrixRotationY( &maty, D3DXToRadian( (FLOAT)m_yr ) );
  313. D3DXMatrixRotationZ( &matz, D3DXToRadian( (FLOAT)m_zr ) );
  314. mat = matx * maty * matz;
  315. m_pd3dDevice->SetTransform( D3DTS_WORLD, &mat );
  316. DrawGeom(m_pGeomCur);
  317. // End the scene.
  318. m_pd3dDevice->EndScene();
  319. }
  320. return S_OK;
  321. }
  322. //-----------------------------------------------------------------------------
  323. // Name: RestoreDeviceObjects()
  324. // Desc:
  325. //-----------------------------------------------------------------------------
  326. HRESULT CFlowerBoxScreensaver::RestoreDeviceObjects()
  327. {
  328. if( m_pd3dDevice == NULL )
  329. return S_OK;
  330. D3DXMATRIX matView;
  331. D3DXVECTOR3 vUpVec( 0.0f, 1.0f, 0.0f );
  332. D3DXVECTOR3 vEyePt(0, 0, 3.5f);
  333. D3DXVECTOR3 vLookatPt(0, 0, 0);
  334. D3DXMatrixLookAtLH( &matView, &vEyePt, &vLookatPt, &vUpVec );
  335. m_pd3dDevice->SetTransform( D3DTS_VIEW , &matView );
  336. D3DLIGHT8 light;
  337. ZeroMemory( &light, sizeof(D3DLIGHT8) );
  338. light.Type = D3DLIGHT_POINT;
  339. light.Diffuse.r = 1.0f;
  340. light.Diffuse.g = 1.0f;
  341. light.Diffuse.b = 1.0f;
  342. light.Diffuse.a = 1.0f;
  343. light.Specular.r = 1.0f;
  344. light.Specular.g = 1.0f;
  345. light.Specular.b = 1.0f;
  346. light.Specular.a = 1.0f;
  347. light.Position.x = 2.0;
  348. light.Position.y = 2.0;
  349. light.Position.z = 10.0;
  350. light.Range = 1000.0f;
  351. light.Attenuation0 = 1.0f;
  352. m_pd3dDevice->SetLight(0, &light);
  353. m_pd3dDevice->LightEnable(0, TRUE);
  354. // Set some basic renderstates
  355. m_pd3dDevice->SetRenderState( D3DRS_DITHERENABLE , TRUE );
  356. m_pd3dDevice->SetRenderState( D3DRS_SPECULARENABLE , TRUE );
  357. m_pd3dDevice->SetRenderState( D3DRS_LIGHTING, TRUE );
  358. m_pd3dDevice->SetRenderState( D3DRS_AMBIENT, D3DCOLOR(0x40404040 ) );
  359. if( config.two_sided == GL_FRONT_AND_BACK )
  360. m_pd3dDevice->SetRenderState( D3DRS_CULLMODE, D3DCULL_NONE );
  361. else
  362. m_pd3dDevice->SetRenderState( D3DRS_CULLMODE, D3DCULL_CW );
  363. NewConfig(&config);
  364. return S_OK;
  365. }
  366. //-----------------------------------------------------------------------------
  367. // Name: InvalidateDeviceObjects()
  368. // Desc:
  369. //-----------------------------------------------------------------------------
  370. HRESULT CFlowerBoxScreensaver::InvalidateDeviceObjects()
  371. {
  372. return S_OK;
  373. }
  374. //-----------------------------------------------------------------------------
  375. // Name: InitVlen()
  376. // Desc: Precomputes scaling factor for spherical projection
  377. //-----------------------------------------------------------------------------
  378. VOID CFlowerBoxScreensaver::InitVlen(GEOMETRY *geom, INT npts, D3DXVECTOR3 *pts)
  379. {
  380. const FLOAT RADIUS = 1;
  381. FLOAT d;
  382. FLOAT *vl;
  383. vl = vlen;
  384. while (npts-- > 0)
  385. {
  386. d = D3DXVec3Length(pts);
  387. // Don't allow really close points because this leads to
  388. // numeric instability and really large objects
  389. // assert(d > 0.01f);
  390. // Geometries are created with size one, filling the area
  391. // from -.5 to .5. This leads to distances generally less
  392. // than one, which leaves off half of the interesting morphing
  393. // effects due to the projection
  394. // Scaling up the scaling factor allows the values to
  395. // be both above and below one
  396. d *= geom->init_sf;
  397. // assert(d > 0.0001f);
  398. *vl++ = (RADIUS-d)/d;
  399. pts++;
  400. }
  401. }
  402. //-----------------------------------------------------------------------------
  403. // Name: MapToSide()
  404. // Desc: Takes x,y coordinates in the range 0-1 and maps them onto the given
  405. // side plane for the current geometry
  406. //-----------------------------------------------------------------------------
  407. VOID MapToSide(PLANE_MAP *map, FLOAT x, FLOAT y, D3DXVECTOR3 *pt)
  408. {
  409. pt->x = x*map->x_axis.x+y*map->y_axis.x+map->base.x;
  410. pt->y = x*map->x_axis.y+y*map->y_axis.y+map->base.y;
  411. pt->z = x*map->x_axis.z+y*map->y_axis.z+map->base.z;
  412. }
  413. //-----------------------------------------------------------------------------
  414. // Name: InitCube()
  415. // Desc: Initializes the cube's geometry
  416. //-----------------------------------------------------------------------------
  417. VOID InitCube(GEOMETRY *geom)
  418. {
  419. const INT CUBE_SIDES = 6;
  420. PLANE_MAP cube_planes[CUBE_SIDES] =
  421. {
  422. PLANE_MAP(D3DXVECTOR3(-0.5f, -0.5f, 0.5f), D3DXVECTOR3( 1.0f, 0.0f, 0.0f), D3DXVECTOR3(0.0f, 1.0f, 0.0f)),
  423. PLANE_MAP(D3DXVECTOR3( 0.5f, -0.5f, -0.5f), D3DXVECTOR3(-1.0f, 0.0f, 0.0f), D3DXVECTOR3(0.0f, 1.0f, 0.0f)),
  424. PLANE_MAP(D3DXVECTOR3( 0.5f, 0.5f, -0.5f), D3DXVECTOR3(-1.0f, 0.0f, 0.0f), D3DXVECTOR3(0.0f, 0.0f, 1.0f)),
  425. PLANE_MAP(D3DXVECTOR3(-0.5f, -0.5f, -0.5f), D3DXVECTOR3( 1.0f, 0.0f, 0.0f), D3DXVECTOR3(0.0f, 0.0f, 1.0f)),
  426. PLANE_MAP(D3DXVECTOR3( 0.5f, -0.5f, -0.5f), D3DXVECTOR3( 0.0f, 1.0f, 0.0f), D3DXVECTOR3(0.0f, 0.0f, 1.0f)),
  427. PLANE_MAP(D3DXVECTOR3(-0.5f, 0.5f, -0.5f), D3DXVECTOR3( 0.0f, -1.0f, 0.0f), D3DXVECTOR3(0.0f, 0.0f, 1.0f))
  428. };
  429. INT side, x, y;
  430. D3DXVECTOR3 *pt;
  431. unsigned short/*INT*/ *idx;
  432. INT *sz;
  433. INT side_pts;
  434. side_pts = (config.subdiv+1)*(config.subdiv+1);
  435. geom->nsides = CUBE_SIDES;
  436. geom->pts = &pts[0];
  437. geom->npts = &npts[0];
  438. geom->normals = &normals[0];
  439. geom->pVertices = &vertices[0];
  440. geom->min_sf = -1.1f;
  441. geom->max_sf = 5.1f;
  442. geom->sf_inc = 0.05f;
  443. geom->init_sf = 2.0f;
  444. // Generate triangle strip data
  445. sz = &strip_size[0];
  446. idx = &index[0];
  447. for (side = 0; side < geom->nsides; side++)
  448. {
  449. geom->sides[side].nstrips = config.subdiv;
  450. geom->sides[side].strip_size = sz;
  451. geom->sides[side].strip_index = idx;
  452. for (x = 0; x < config.subdiv; x++)
  453. {
  454. *sz++ = (config.subdiv+1)*2;
  455. for (y = 0; y < config.subdiv+1; y++)
  456. {
  457. *idx++ = side * side_pts + x * (config.subdiv + 1) + y;
  458. *idx++ = side * side_pts + (x + 1) * (config.subdiv + 1) + y;
  459. }
  460. }
  461. }
  462. // Generate base vertices
  463. pt = geom->pts;
  464. for (side = 0; side < geom->nsides; side++)
  465. {
  466. for (x = 0; x < config.subdiv+1; x++)
  467. {
  468. for (y = 0; y < config.subdiv+1; y++)
  469. {
  470. MapToSide(&cube_planes[side],
  471. (FLOAT)x/config.subdiv, (FLOAT)y/config.subdiv,
  472. pt);
  473. pt++;
  474. }
  475. }
  476. }
  477. geom->total_pts = geom->nsides*side_pts;
  478. }
  479. //-----------------------------------------------------------------------------
  480. // Name: InitTetra()
  481. // Desc: Initializes the tetrahedron's geometry
  482. //-----------------------------------------------------------------------------
  483. VOID InitTetra(GEOMETRY *geom)
  484. {
  485. const INT TETRA_SIDES = 4;
  486. const FLOAT SQRT3 = 1.73205f;
  487. const FLOAT SQRT3_2 = (SQRT3/2.0f);
  488. const FLOAT SQRT3_3 = (SQRT3/3.0f);
  489. const FLOAT SQRT3_6 = (SQRT3/6.0f);
  490. const FLOAT SQRT3_12 = (SQRT3/12.0f);
  491. const FLOAT TETRA_BASE = (-SQRT3/8.0f);
  492. PLANE_MAP tetra_planes[TETRA_SIDES] =
  493. {
  494. PLANE_MAP(D3DXVECTOR3(-0.5f, TETRA_BASE, SQRT3_6), D3DXVECTOR3(1.0f, 0.0f, 0.0f), D3DXVECTOR3(0.0f, SQRT3_2, -SQRT3_6)),
  495. PLANE_MAP(D3DXVECTOR3(0.0f, TETRA_BASE, -SQRT3_3), D3DXVECTOR3(-0.5f, 0.0f, SQRT3_2), D3DXVECTOR3(0.25f, SQRT3_2, SQRT3_12)),
  496. PLANE_MAP(D3DXVECTOR3(0.5f, TETRA_BASE, SQRT3_6), D3DXVECTOR3(-0.5f, 0.0f, -SQRT3_2), D3DXVECTOR3(-0.25f, SQRT3_2, SQRT3_12)),
  497. PLANE_MAP(D3DXVECTOR3(0.5f, TETRA_BASE, SQRT3_6), D3DXVECTOR3(-1.0f, 0.0f, 0.0f), D3DXVECTOR3(0.0f, 0.0f, -SQRT3_2)),
  498. };
  499. INT side, x, y;
  500. D3DXVECTOR3 *pt;
  501. unsigned short/*INT*/ *idx;
  502. INT *sz;
  503. unsigned short side_pts;
  504. unsigned short base_pt;
  505. unsigned short row_pts;
  506. FLOAT fx;
  507. side_pts = (config.subdiv+2)*(config.subdiv+1)/2;
  508. geom->nsides = TETRA_SIDES;
  509. geom->pts = &pts[0];
  510. geom->npts = &npts[0];
  511. geom->normals = &normals[0];
  512. geom->pVertices = &vertices[0];
  513. geom->min_sf = -1.1f;
  514. geom->max_sf = 5.2f;
  515. geom->sf_inc = 0.05f;
  516. geom->init_sf = 3.75f;
  517. // Generate triangle strip data
  518. sz = &strip_size[0];
  519. idx = &index[0];
  520. base_pt = 0;
  521. for (side = 0; side < geom->nsides; side++)
  522. {
  523. geom->sides[side].nstrips = config.subdiv;
  524. geom->sides[side].strip_size = sz;
  525. geom->sides[side].strip_index = idx;
  526. for (x = 0; x < config.subdiv; x++)
  527. {
  528. row_pts = config.subdiv-x+1;
  529. *sz++ = row_pts*2-1;
  530. *idx++ = base_pt;
  531. for (y = 0; y < row_pts-1; y++)
  532. {
  533. *idx++ = base_pt+row_pts+y;
  534. *idx++ = base_pt+1+y;
  535. }
  536. base_pt += row_pts;
  537. }
  538. base_pt++;
  539. }
  540. // Generate base vertices
  541. pt = geom->pts;
  542. for (side = 0; side < geom->nsides; side++)
  543. {
  544. for (x = 0; x < config.subdiv+1; x++)
  545. {
  546. fx = (FLOAT)x/config.subdiv;
  547. for (y = 0; y < config.subdiv-x+1; y++)
  548. {
  549. MapToSide(&tetra_planes[side],
  550. fx+(FLOAT)y/(config.subdiv*2),
  551. (FLOAT)y/config.subdiv,
  552. pt);
  553. pt++;
  554. }
  555. }
  556. }
  557. geom->total_pts = geom->nsides*side_pts;
  558. }
  559. //-----------------------------------------------------------------------------
  560. // Name: InitPyramids()
  561. // Desc: Initializes double pyramid geometry
  562. //-----------------------------------------------------------------------------
  563. VOID InitPyramids(GEOMETRY *geom)
  564. {
  565. const INT PYRAMIDS_SIDES = 8;
  566. PLANE_MAP pyramids_planes[PYRAMIDS_SIDES] =
  567. {
  568. PLANE_MAP(D3DXVECTOR3(-0.5f, 0.0f, 0.5f), D3DXVECTOR3( 1.0f, 0.0f, 0.0f), D3DXVECTOR3( 0.0f, 0.5f, -0.5f)),
  569. PLANE_MAP(D3DXVECTOR3( 0.5f, 0.0f, 0.5f), D3DXVECTOR3(-1.0f, 0.0f, 0.0f), D3DXVECTOR3( 0.0f, -0.5f, -0.5f)),
  570. PLANE_MAP(D3DXVECTOR3( 0.5f, 0.0f, 0.5f), D3DXVECTOR3( 0.0f, 0.0f, -1.0f), D3DXVECTOR3(-0.5f, 0.5f, 0.0f)),
  571. PLANE_MAP(D3DXVECTOR3( 0.5f, 0.0f, -0.5f), D3DXVECTOR3( 0.0f, 0.0f, 1.0f), D3DXVECTOR3(-0.5f, -0.5f, 0.0f)),
  572. PLANE_MAP(D3DXVECTOR3( 0.5f, 0.0f, -0.5f), D3DXVECTOR3(-1.0f, 0.0f, 0.0f), D3DXVECTOR3( 0.0f, 0.5f, 0.5f)),
  573. PLANE_MAP(D3DXVECTOR3(-0.5f, 0.0f, -0.5f), D3DXVECTOR3( 1.0f, 0.0f, 0.0f), D3DXVECTOR3( 0.0f, -0.5f, 0.5f)),
  574. PLANE_MAP(D3DXVECTOR3(-0.5f, 0.0f, -0.5f), D3DXVECTOR3( 0.0f, 0.0f, 1.0f), D3DXVECTOR3( 0.5f, 0.5f, 0.0f)),
  575. PLANE_MAP(D3DXVECTOR3(-0.5f, 0.0f, 0.5f), D3DXVECTOR3( 0.0f, 0.0f, -1.0f), D3DXVECTOR3( 0.5f, -0.5f, 0.0f))
  576. };
  577. INT side, x, y;
  578. D3DXVECTOR3 *pt;
  579. unsigned short/*INT*/ *idx;
  580. INT *sz;
  581. unsigned short side_pts;
  582. unsigned short base_pt;
  583. unsigned short row_pts;
  584. FLOAT fx;
  585. side_pts = (config.subdiv+2)*(config.subdiv+1)/2;
  586. geom->nsides = PYRAMIDS_SIDES;
  587. geom->pts = &pts[0];
  588. geom->npts = &npts[0];
  589. geom->normals = &normals[0];
  590. geom->pVertices = &vertices[0];
  591. geom->min_sf = -1.1f;
  592. geom->max_sf = 5.2f;
  593. geom->sf_inc = 0.05f;
  594. geom->init_sf = 3.0f;
  595. // Generate triangle strip data
  596. sz = &strip_size[0];
  597. idx = &index[0];
  598. base_pt = 0;
  599. for (side = 0; side < geom->nsides; side++) {
  600. geom->sides[side].nstrips = config.subdiv;
  601. geom->sides[side].strip_size = sz;
  602. geom->sides[side].strip_index = idx;
  603. for (x = 0; x < config.subdiv; x++) {
  604. row_pts = config.subdiv-x+1;
  605. *sz++ = row_pts*2-1;
  606. *idx++ = base_pt;
  607. for (y = 0; y < row_pts-1; y++) {
  608. *idx++ = base_pt+row_pts+y;
  609. *idx++ = base_pt+1+y;
  610. }
  611. base_pt += row_pts;
  612. }
  613. base_pt++;
  614. }
  615. // Generate base vertices
  616. pt = geom->pts;
  617. for (side = 0; side < geom->nsides; side++)
  618. {
  619. for (x = 0; x < config.subdiv+1; x++)
  620. {
  621. fx = (FLOAT)x/config.subdiv;
  622. for (y = 0; y < config.subdiv-x+1; y++)
  623. {
  624. MapToSide(&pyramids_planes[side],
  625. fx+(FLOAT)y/(config.subdiv*2),
  626. (FLOAT)y/config.subdiv,
  627. pt);
  628. pt++;
  629. }
  630. }
  631. }
  632. geom->total_pts = geom->nsides*side_pts;
  633. }
  634. //-----------------------------------------------------------------------------
  635. // Name: InitCylinder()
  636. // Desc: Initializes the cylinder geometry
  637. //-----------------------------------------------------------------------------
  638. VOID InitCylinder(GEOMETRY *geom)
  639. {
  640. unsigned short side, x, y;
  641. D3DXVECTOR3 *pt;
  642. unsigned short/*INT*/ *idx;
  643. INT *sz;
  644. unsigned short base_pt;
  645. unsigned short row_pts;
  646. FLOAT fx, fz;
  647. double ang;
  648. geom->nsides = 1;
  649. geom->pts = &pts[0];
  650. geom->npts = &npts[0];
  651. geom->normals = &normals[0];
  652. geom->pVertices = &vertices[0];
  653. geom->min_sf = -2.5f;
  654. geom->max_sf = 8.5f;
  655. geom->sf_inc = 0.05f;
  656. geom->init_sf = 2.1f;
  657. // Generate triangle strip data
  658. // If version 1.1 then allocate the index buffer for glDrawElements
  659. sz = &strip_size[0];
  660. idx = &index[0];
  661. side = 0;
  662. geom->sides[side].nstrips = config.subdiv;
  663. geom->sides[side].strip_size = sz;
  664. geom->sides[side].strip_index = idx;
  665. row_pts = config.subdiv+1;
  666. base_pt = 0;
  667. for (x = 0; x < config.subdiv; x++) {
  668. *sz++ = row_pts*2;
  669. for (y = 0; y < row_pts; y++) {
  670. // Wrap around at the edge so the cylinder normals
  671. // are properly averaged
  672. if (x == config.subdiv-1) {
  673. *idx++ = y;
  674. }
  675. else {
  676. *idx++ = base_pt+row_pts+y;
  677. }
  678. *idx++ = base_pt+y;
  679. }
  680. base_pt += row_pts;
  681. }
  682. // Generate base vertices
  683. pt = geom->pts;
  684. ang = 0;
  685. for (x = 0; x < config.subdiv; x++)
  686. {
  687. fx = (FLOAT)cos(ang)*0.5f;
  688. fz = (FLOAT)sin(ang)*0.5f;
  689. for (y = 0; y < config.subdiv+1; y++)
  690. {
  691. pt->x = fx;
  692. pt->y = (FLOAT)y/config.subdiv-0.5f;
  693. pt->z = fz;
  694. pt++;
  695. }
  696. ang += (2*D3DX_PI)/config.subdiv;
  697. }
  698. geom->total_pts = geom->nsides*(config.subdiv+1)*config.subdiv;
  699. }
  700. //-----------------------------------------------------------------------------
  701. // Name: InitSpring()
  702. // Desc: Initializes the spring geometry
  703. //-----------------------------------------------------------------------------
  704. VOID InitSpring(GEOMETRY *geom)
  705. {
  706. const FLOAT SPRING_RADIUS = 0.1f;
  707. const FLOAT SPRING_CENTER = (0.5f-SPRING_RADIUS);
  708. INT side, x, y;
  709. D3DXVECTOR3 *pt;
  710. unsigned short/*INT*/ *idx;
  711. INT *sz;
  712. double ang_center, ang_surf;
  713. FLOAT cs, sn;
  714. FLOAT rad;
  715. PLANE_MAP plane;
  716. INT spin_pts;
  717. INT row_pts;
  718. geom->nsides = 1;
  719. geom->pts = &pts[0];
  720. geom->npts = &npts[0];
  721. geom->normals = &normals[0];
  722. geom->pVertices = &vertices[0];
  723. geom->min_sf = -2.2f;
  724. geom->max_sf = 0.2f;
  725. geom->sf_inc = 0.05f;
  726. geom->init_sf = 1.0f;
  727. // Generate triangle strip data
  728. // If version 1.1 then allocate the index buffer for glDrawElements
  729. sz = &strip_size[0];
  730. idx = &index[0];
  731. side = 0;
  732. geom->sides[side].nstrips = config.subdiv;
  733. geom->sides[side].strip_size = sz;
  734. geom->sides[side].strip_index = idx;
  735. row_pts = config.subdiv;
  736. spin_pts = 4*config.subdiv+1;
  737. for (x = 0; x < config.subdiv; x++) {
  738. *sz++ = spin_pts*2;
  739. for (y = 0; y < spin_pts; y++) {
  740. *idx++ = x+row_pts*y;
  741. // Wrap around at the edge so the cylindrical surface
  742. // of the tube is seamless. Without this the normal
  743. // averaging would be incorrect and a seam would be visible
  744. if (x == config.subdiv-1) {
  745. *idx++ = row_pts*y;
  746. }
  747. else {
  748. *idx++ = x+row_pts*y+1;
  749. }
  750. }
  751. }
  752. // Generate base vertices
  753. pt = geom->pts;
  754. ang_center = 0;
  755. plane.y_axis.x = 0.0f;
  756. plane.y_axis.y = SPRING_RADIUS;
  757. plane.y_axis.z = 0.0f;
  758. plane.x_axis.y = 0.0f;
  759. for (x = 0; x < spin_pts; x++)
  760. {
  761. cs = (FLOAT)cos(ang_center);
  762. sn = (FLOAT)sin(ang_center);
  763. rad = 0.5f-(FLOAT)x/(spin_pts-1)*(SPRING_CENTER/2);
  764. plane.base.x = cs*rad;
  765. plane.base.y = -0.5f+(FLOAT)x/(spin_pts-1);
  766. plane.base.z = sn*rad;
  767. plane.x_axis.x = cs*SPRING_RADIUS;
  768. plane.x_axis.z = sn*SPRING_RADIUS;
  769. ang_surf = 0;
  770. for (y = 0; y < config.subdiv; y++)
  771. {
  772. MapToSide(&plane,
  773. (FLOAT)cos(ang_surf), (FLOAT)sin(ang_surf),
  774. pt);
  775. pt++;
  776. ang_surf += (2*D3DX_PI)/config.subdiv;
  777. }
  778. ang_center += (4*D3DX_PI)/(spin_pts-1);
  779. }
  780. geom->total_pts = geom->nsides*spin_pts*config.subdiv;
  781. }
  782. //-----------------------------------------------------------------------------
  783. // Name: DrawGeom()
  784. // Desc: Draw the current geometry
  785. //-----------------------------------------------------------------------------
  786. VOID CFlowerBoxScreensaver::DrawGeom(GEOMETRY *geom)
  787. {
  788. if (config.smooth_colors)
  789. m_pd3dDevice->SetRenderState( D3DRS_SHADEMODE, D3DSHADE_GOURAUD );
  790. else
  791. m_pd3dDevice->SetRenderState( D3DRS_SHADEMODE, D3DSHADE_FLAT );
  792. if (config.color_pick == ID_COL_SINGLE)
  793. SetMaterialColor(solid_cols);
  794. for( INT i = 0; i < geom->total_pts; i++ )
  795. {
  796. geom->pVertices[i].p = *(D3DXVECTOR3*)&geom->npts[i];
  797. D3DXVec3Normalize( &geom->pVertices[i].n, (D3DXVECTOR3*)&geom->normals[i] );
  798. }
  799. m_pd3dDevice->SetVertexShader( D3DFVF_MYVERTEX );
  800. WORD* col0Indices = NULL;
  801. INT numCol0Indices;
  802. WORD* col1Indices = NULL;
  803. INT numCol1Indices;
  804. col0Indices = new WORD[5000];
  805. if( col0Indices == NULL )
  806. return;
  807. col1Indices = new WORD[5000];
  808. if( col1Indices == NULL )
  809. {
  810. delete[] col0Indices;
  811. return;
  812. }
  813. INT side;
  814. INT strip;
  815. INT triangle;
  816. unsigned short *idx;
  817. BOOL bBackwards; // triangle backwards for original strip implementation
  818. BOOL bCol0; // col0 or col1 for this triangle?
  819. INT numThisColor; // num tris done in this color; helps determine bCol0
  820. for (side = 0; side < geom->nsides; side++)
  821. {
  822. numCol0Indices = 0;
  823. numCol1Indices = 0;
  824. bCol0 = TRUE;
  825. idx = geom->sides[side].strip_index;
  826. for (strip = 0; strip < geom->sides[side].nstrips; strip++, idx += 2)
  827. {
  828. numThisColor = 0;
  829. if( !config.triangle_colors )
  830. numThisColor++;
  831. bBackwards = FALSE;
  832. if( config.color_pick == ID_COL_CHECKER )
  833. bCol0 = ( strip & 1 );
  834. for( triangle = 0; triangle < geom->sides[side].strip_size[strip] - 2; triangle++ )
  835. {
  836. if( config.color_pick == ID_COL_CHECKER )
  837. {
  838. numThisColor++;
  839. if( numThisColor == 2 )
  840. {
  841. bCol0 = !bCol0;
  842. numThisColor = 0;
  843. }
  844. }
  845. if( bCol0 )
  846. {
  847. if( bBackwards )
  848. {
  849. col0Indices[numCol0Indices++] = *(idx+1);
  850. col0Indices[numCol0Indices++] = *(idx+0);
  851. col0Indices[numCol0Indices++] = *(idx+2);
  852. }
  853. else
  854. {
  855. col0Indices[numCol0Indices++] = *(idx+0);
  856. col0Indices[numCol0Indices++] = *(idx+1);
  857. col0Indices[numCol0Indices++] = *(idx+2);
  858. }
  859. }
  860. else
  861. {
  862. if( bBackwards )
  863. {
  864. col1Indices[numCol1Indices++] = *(idx+1);
  865. col1Indices[numCol1Indices++] = *(idx+0);
  866. col1Indices[numCol1Indices++] = *(idx+2);
  867. }
  868. else
  869. {
  870. col1Indices[numCol1Indices++] = *(idx+0);
  871. col1Indices[numCol1Indices++] = *(idx+1);
  872. col1Indices[numCol1Indices++] = *(idx+2);
  873. }
  874. }
  875. idx ++;
  876. bBackwards = !bBackwards;
  877. }
  878. }
  879. // Draw this side's col0 primitives
  880. if (config.color_pick == ID_COL_PER_SIDE)
  881. SetMaterialColor(side_cols[side]);
  882. else if (config.color_pick == ID_COL_CHECKER)
  883. SetMaterialColor(checker_cols[side][0]);
  884. m_pd3dDevice->DrawIndexedPrimitiveUP( D3DPT_TRIANGLELIST, 0,
  885. geom->total_pts, numCol0Indices / 3,
  886. col0Indices, D3DFMT_INDEX16, &geom->pVertices[0], sizeof(MYVERTEX) );
  887. // Draw this side's col1 primitives, if any
  888. if (config.color_pick == ID_COL_CHECKER)
  889. {
  890. SetMaterialColor(checker_cols[side][1]);
  891. m_pd3dDevice->DrawIndexedPrimitiveUP( D3DPT_TRIANGLELIST, 0,
  892. geom->total_pts, numCol1Indices / 3,
  893. col1Indices, D3DFMT_INDEX16, &geom->pVertices[0], sizeof(MYVERTEX) );
  894. }
  895. }
  896. if( col0Indices != NULL )
  897. delete[] col0Indices;
  898. if( col1Indices != NULL )
  899. delete[] col1Indices;
  900. }
  901. //-----------------------------------------------------------------------------
  902. // Name: ComputeAveragedNormals()
  903. // Desc: Compute face-averaged normals for each vertex
  904. //-----------------------------------------------------------------------------
  905. VOID ComputeAveragedNormals(GEOMETRY *geom)
  906. {
  907. INT side, strip;
  908. INT *sz;
  909. unsigned short/*INT*/ *idx;
  910. INT idx1, idx2, idx3;
  911. INT tc, idc;
  912. D3DXVECTOR3 v1, v2, n1;
  913. memset(geom->normals, 0, sizeof(D3DXVECTOR3)*geom->total_pts);
  914. for (side = 0; side < geom->nsides; side++)
  915. {
  916. idx = geom->sides[side].strip_index;
  917. sz = geom->sides[side].strip_size;
  918. for (strip = 0; strip < geom->sides[side].nstrips; strip++)
  919. {
  920. idx1 = *idx++;
  921. idx2 = *idx++;
  922. // assert(idx1 >= 0 && idx1 < geom->total_pts &&
  923. // idx2 >= 0 && idx2 < geom->total_pts);
  924. tc = (*sz++)-2;
  925. for (idc = 0; idc < tc; idc++)
  926. {
  927. idx3 = *idx++;
  928. // assert(idx3 >= 0 && idx3 < geom->total_pts);
  929. v1 = geom->npts[idx3] - geom->npts[idx1];
  930. v2 = geom->npts[idx2] - geom->npts[idx1];
  931. D3DXVec3Cross( &n1, &v1, &v2 );
  932. // Triangle strip ordering causes half of the triangles
  933. // to be oriented oppositely from the others
  934. // Those triangles need to have their normals flipped
  935. // so the whole triangle strip has consistent normals
  936. if ((idc & 1) == 0)
  937. {
  938. n1.x = -n1.x;
  939. n1.y = -n1.y;
  940. n1.z = -n1.z;
  941. }
  942. geom->normals[idx1] += n1;
  943. geom->normals[idx2] += n1;
  944. geom->normals[idx3] += n1;
  945. idx1 = idx2;
  946. idx2 = idx3;
  947. }
  948. }
  949. }
  950. }
  951. //-----------------------------------------------------------------------------
  952. // Name: UpdatePts()
  953. // Desc:
  954. //-----------------------------------------------------------------------------
  955. VOID CFlowerBoxScreensaver::UpdatePts(GEOMETRY *geom, FLOAT sf)
  956. {
  957. INT pt;
  958. FLOAT f, *vl;
  959. D3DXVECTOR3 *v;
  960. D3DXVECTOR3 *p;
  961. vl = vlen;
  962. p = &geom->pts[0];
  963. v = &geom->npts[0];
  964. for (pt = 0; pt < geom->total_pts; pt++)
  965. {
  966. f = (*vl++)*sf+1;
  967. v->x = p->x*f;
  968. v->y = p->y*f;
  969. v->z = p->z*f;
  970. p++;
  971. v++;
  972. }
  973. ComputeAveragedNormals(geom);
  974. }
  975. //-----------------------------------------------------------------------------
  976. // Name: ComputeHsvColors()
  977. // Desc:
  978. //-----------------------------------------------------------------------------
  979. VOID CFlowerBoxScreensaver::ComputeHsvColors(VOID)
  980. {
  981. FLOAT *cols;
  982. INT ncols;
  983. FLOAT ang, da;
  984. INT hex;
  985. FLOAT fhex, frac;
  986. FLOAT p, q, t;
  987. FLOAT sat, val;
  988. switch(config.color_pick)
  989. {
  990. case ID_COL_CHECKER:
  991. ncols = MAXSIDES*NCCOLS;
  992. cols = &checker_cols[0][0][0];
  993. break;
  994. case ID_COL_PER_SIDE:
  995. ncols = MAXSIDES;
  996. cols = &side_cols[0][0];
  997. break;
  998. case ID_COL_SINGLE:
  999. ncols = 1;
  1000. cols = &solid_cols[0];
  1001. break;
  1002. }
  1003. ang = m_phase;
  1004. da = (FLOAT)((2*D3DX_PI)/ncols);
  1005. val = sat = 1.0f;
  1006. while (ncols > 0)
  1007. {
  1008. fhex = (FLOAT)(6*ang/(2*D3DX_PI));
  1009. hex = (INT)fhex;
  1010. frac = fhex-hex;
  1011. hex = hex % 6;
  1012. p = val*(1-sat);
  1013. q = val*(1-sat*frac);
  1014. t = val*(1-sat*(1-frac));
  1015. switch(hex)
  1016. {
  1017. case 0:
  1018. cols[0] = val;
  1019. cols[1] = t;
  1020. cols[2] = p;
  1021. break;
  1022. case 1:
  1023. cols[0] = q;
  1024. cols[1] = val;
  1025. cols[2] = p;
  1026. break;
  1027. case 2:
  1028. cols[0] = p;
  1029. cols[1] = val;
  1030. cols[2] = t;
  1031. break;
  1032. case 3:
  1033. cols[0] = p;
  1034. cols[1] = q;
  1035. cols[2] = val;
  1036. break;
  1037. case 4:
  1038. cols[0] = t;
  1039. cols[1] = p;
  1040. cols[2] = val;
  1041. break;
  1042. case 5:
  1043. cols[0] = val;
  1044. cols[1] = p;
  1045. cols[2] = q;
  1046. break;
  1047. }
  1048. ang += da;
  1049. cols += 4;
  1050. ncols--;
  1051. }
  1052. }
  1053. //-----------------------------------------------------------------------------
  1054. // Name: NewConfig()
  1055. // Desc:
  1056. //-----------------------------------------------------------------------------
  1057. VOID CFlowerBoxScreensaver::NewConfig(CONFIG *cnf)
  1058. {
  1059. // Set new config
  1060. config = *cnf;
  1061. HKEY hkey;
  1062. if( ERROR_SUCCESS == RegCreateKeyEx( HKEY_CURRENT_USER, m_strRegPath,
  1063. 0, NULL, REG_OPTION_NON_VOLATILE, KEY_ALL_ACCESS, NULL, &hkey, NULL ) )
  1064. {
  1065. DXUtil_WriteBoolRegKey( hkey, TEXT("Smooth"), config.smooth_colors );
  1066. DXUtil_WriteBoolRegKey( hkey, TEXT("Slanted"), config.triangle_colors );
  1067. DXUtil_WriteBoolRegKey( hkey, TEXT("Cycle"), config.cycle_colors );
  1068. DXUtil_WriteBoolRegKey( hkey, TEXT("Spin"), config.spin );
  1069. DXUtil_WriteBoolRegKey( hkey, TEXT("Bloom"), config.bloom );
  1070. DXUtil_WriteIntRegKey( hkey, TEXT("Subdiv"), config.subdiv );
  1071. DXUtil_WriteIntRegKey( hkey, TEXT("ColorPick"), config.color_pick );
  1072. DXUtil_WriteIntRegKey( hkey, TEXT("ImageSize"), config.image_size );
  1073. DXUtil_WriteIntRegKey( hkey, TEXT("Geom"), config.geom );
  1074. DXUtil_WriteIntRegKey( hkey, TEXT("TwoSided"), config.two_sided );
  1075. WriteScreenSettings( hkey );
  1076. RegCloseKey( hkey );
  1077. }
  1078. // Reset colors
  1079. memcpy(checker_cols, base_checker_cols, sizeof(checker_cols));
  1080. memcpy(side_cols, base_side_cols, sizeof(side_cols));
  1081. memcpy(solid_cols, base_solid_cols, sizeof(solid_cols));
  1082. // Reset geometry
  1083. m_pGeomCur = geom_table[config.geom];
  1084. m_pGeomCur->init(m_pGeomCur);
  1085. // assert(m_pGeomCur->total_pts <= MAXPTS);
  1086. InitVlen(m_pGeomCur, m_pGeomCur->total_pts, m_pGeomCur->pts);
  1087. m_sf = 0.0f;
  1088. m_sfi = m_pGeomCur->sf_inc;
  1089. UpdatePts(m_pGeomCur, m_sf);
  1090. }
  1091. //-----------------------------------------------------------------------------
  1092. // Name: SetupTrackbar()
  1093. // Desc: Setup a common control trackbar
  1094. //-----------------------------------------------------------------------------
  1095. VOID SetupTrackbar( HWND hDlg, INT item, INT lo, INT hi, INT lineSize,
  1096. INT pageSize, INT pos )
  1097. {
  1098. SendDlgItemMessage(
  1099. hDlg,
  1100. item,
  1101. TBM_SETRANGE,
  1102. (WPARAM) TRUE,
  1103. (LPARAM) MAKELONG( lo, hi )
  1104. );
  1105. SendDlgItemMessage(
  1106. hDlg,
  1107. item,
  1108. TBM_SETPOS,
  1109. (WPARAM) TRUE,
  1110. (LPARAM) pos
  1111. );
  1112. SendDlgItemMessage(
  1113. hDlg,
  1114. item,
  1115. TBM_SETPAGESIZE,
  1116. (WPARAM) 0,
  1117. (LPARAM) pageSize
  1118. );
  1119. SendDlgItemMessage(
  1120. hDlg,
  1121. item,
  1122. TBM_SETLINESIZE,
  1123. (WPARAM) 0,
  1124. (LPARAM) lineSize
  1125. );
  1126. }
  1127. //-----------------------------------------------------------------------------
  1128. // Name: GetTrackbarPos()
  1129. // Desc: Get the current position of a common control trackbar
  1130. //-----------------------------------------------------------------------------
  1131. INT GetTrackbarPos( HWND hDlg, INT item )
  1132. {
  1133. return
  1134. (INT)SendDlgItemMessage(
  1135. hDlg,
  1136. item,
  1137. TBM_GETPOS,
  1138. 0,
  1139. 0
  1140. );
  1141. }
  1142. // Temporary configuration for when the configuration dialog is active
  1143. // If the dialog is ok'ed then this becomes the current configuration,
  1144. // otherwise it is discarded
  1145. CONFIG temp_config;
  1146. //-----------------------------------------------------------------------------
  1147. // Name: ScreenSaverConfigureDialog()
  1148. // Desc:
  1149. //-----------------------------------------------------------------------------
  1150. INT_PTR CALLBACK CFlowerBoxScreensaver::ScreenSaverConfigureDialog(HWND hdlg, UINT msg,
  1151. WPARAM wpm, LPARAM lpm)
  1152. {
  1153. HWND hCtrl;
  1154. INT i;
  1155. switch(msg)
  1156. {
  1157. case WM_INITDIALOG:
  1158. InitCommonControls();
  1159. g_pMyFlowerBoxScreensaver->ReadSettings();
  1160. temp_config = config;
  1161. CheckRadioButton(hdlg, ID_COL_PICK_FIRST, ID_COL_PICK_LAST,
  1162. config.color_pick);
  1163. CheckDlgButton(hdlg, ID_COL_SMOOTH, config.smooth_colors);
  1164. CheckDlgButton(hdlg, ID_COL_TRIANGLE, config.triangle_colors);
  1165. CheckDlgButton(hdlg, ID_COL_CYCLE, config.cycle_colors);
  1166. CheckDlgButton(hdlg, ID_SPIN, config.spin);
  1167. CheckDlgButton(hdlg, ID_BLOOM, config.bloom);
  1168. CheckDlgButton(hdlg, ID_TWO_SIDED,
  1169. config.two_sided == GL_FRONT_AND_BACK);
  1170. SetupTrackbar( hdlg, ID_COMPLEXITY, MINSUBDIV, MAXSUBDIV,
  1171. complexity_range.step,
  1172. complexity_range.page_step,
  1173. config.subdiv);
  1174. SetupTrackbar( hdlg, ID_IMAGE_SIZE, MINIMAGESIZE, MAXIMAGESIZE,
  1175. image_size_range.step,
  1176. image_size_range.page_step,
  1177. config.image_size);
  1178. hCtrl = GetDlgItem(hdlg, ID_GEOM);
  1179. SendMessage(hCtrl, CB_RESETCONTENT, 0, 0);
  1180. // String storage
  1181. TCHAR geom_names[IDS_GEOM_COUNT][20];
  1182. for (i = 0; i < IDS_GEOM_COUNT; i++)
  1183. {
  1184. LoadString( NULL, i+IDS_GEOM_FIRST, geom_names[i],
  1185. sizeof(geom_names)/IDS_GEOM_COUNT );
  1186. SendMessage(hCtrl, CB_ADDSTRING, 0, (LPARAM)geom_names[i]);
  1187. }
  1188. SendMessage(hCtrl, CB_SETCURSEL, config.geom, 0);
  1189. SetFocus(GetDlgItem(hdlg, ID_COMPLEXITY));
  1190. return FALSE;
  1191. case WM_COMMAND:
  1192. switch(LOWORD(wpm))
  1193. {
  1194. case ID_COL_CHECKER:
  1195. case ID_COL_PER_SIDE:
  1196. case ID_COL_SINGLE:
  1197. temp_config.color_pick = LOWORD(wpm);
  1198. break;
  1199. case ID_COL_SMOOTH:
  1200. temp_config.smooth_colors = !temp_config.smooth_colors;
  1201. break;
  1202. case ID_COL_TRIANGLE:
  1203. temp_config.triangle_colors = !temp_config.triangle_colors;
  1204. break;
  1205. case ID_COL_CYCLE:
  1206. temp_config.cycle_colors = !temp_config.cycle_colors;
  1207. break;
  1208. case ID_SPIN:
  1209. temp_config.spin = !temp_config.spin;
  1210. break;
  1211. case ID_BLOOM:
  1212. temp_config.bloom = !temp_config.bloom;
  1213. break;
  1214. case ID_TWO_SIDED:
  1215. temp_config.two_sided =
  1216. temp_config.two_sided == GL_FRONT_AND_BACK ? GL_FRONT :
  1217. GL_FRONT_AND_BACK;
  1218. break;
  1219. case IDC_SCREENSETTINGS:
  1220. g_pMyFlowerBoxScreensaver->DoScreenSettingsDialog(hdlg);
  1221. break;
  1222. case IDOK:
  1223. temp_config.subdiv =
  1224. GetTrackbarPos(hdlg, ID_COMPLEXITY);
  1225. temp_config.image_size =
  1226. GetTrackbarPos(hdlg, ID_IMAGE_SIZE);
  1227. temp_config.geom =
  1228. (INT)SendMessage(GetDlgItem(hdlg, ID_GEOM), CB_GETCURSEL, 0, 0);
  1229. // NewConfig(&temp_config);
  1230. // Fall through
  1231. case IDCANCEL:
  1232. EndDialog(hdlg, LOWORD(wpm));
  1233. break;
  1234. }
  1235. return TRUE;
  1236. }
  1237. return FALSE;
  1238. }
  1239. //-----------------------------------------------------------------------------
  1240. // Name: SetMaterialColor()
  1241. // Desc:
  1242. //-----------------------------------------------------------------------------
  1243. HRESULT CFlowerBoxScreensaver::SetMaterialColor(FLOAT* pfColors)
  1244. {
  1245. D3DMATERIAL8 mtrl;
  1246. ZeroMemory( &mtrl, sizeof(D3DMATERIAL8) );
  1247. mtrl.Diffuse.r = mtrl.Ambient.r = pfColors[0];
  1248. mtrl.Diffuse.g = mtrl.Ambient.g = pfColors[1];
  1249. mtrl.Diffuse.b = mtrl.Ambient.b = pfColors[2];
  1250. mtrl.Diffuse.a = mtrl.Ambient.a = pfColors[3];
  1251. mtrl.Specular.r = 0.8f;
  1252. mtrl.Specular.g = 0.8f;
  1253. mtrl.Specular.b = 0.8f;
  1254. mtrl.Specular.a = 1.0f;
  1255. mtrl.Power = 30.0f;
  1256. return m_pd3dDevice->SetMaterial(&mtrl);
  1257. }
  1258. //-----------------------------------------------------------------------------
  1259. // Name: ReadSettings()
  1260. // Desc: Read user preferences from registry
  1261. //-----------------------------------------------------------------------------
  1262. VOID CFlowerBoxScreensaver::ReadSettings()
  1263. {
  1264. HKEY hkey;
  1265. // Read OpenGL settings first, so OS upgrade cases work
  1266. ss_ReadSettings();
  1267. if( ERROR_SUCCESS == RegCreateKeyEx( HKEY_CURRENT_USER, m_strRegPath,
  1268. 0, NULL, REG_OPTION_NON_VOLATILE, KEY_ALL_ACCESS, NULL, &hkey, NULL ) )
  1269. {
  1270. DXUtil_ReadBoolRegKey( hkey, TEXT("Smooth"), &config.smooth_colors, config.smooth_colors );
  1271. DXUtil_ReadBoolRegKey( hkey, TEXT("Slanted"), &config.triangle_colors, config.triangle_colors );
  1272. DXUtil_ReadBoolRegKey( hkey, TEXT("Cycle"), &config.cycle_colors, config.cycle_colors );
  1273. DXUtil_ReadBoolRegKey( hkey, TEXT("Spin"), &config.spin, config.spin );
  1274. DXUtil_ReadBoolRegKey( hkey, TEXT("Bloom"), &config.bloom, config.bloom );
  1275. DXUtil_ReadIntRegKey( hkey, TEXT("Subdiv"), (DWORD*)&config.subdiv, config.subdiv );
  1276. if( config.subdiv < MINSUBDIV )
  1277. config.subdiv = MINSUBDIV;
  1278. if( config.subdiv > MAXSUBDIV )
  1279. config.subdiv = MAXSUBDIV;
  1280. DXUtil_ReadIntRegKey( hkey, TEXT("ColorPick"), (DWORD*)&config.color_pick, config.color_pick );
  1281. DXUtil_ReadIntRegKey( hkey, TEXT("ImageSize"), (DWORD*)&config.image_size, config.image_size );
  1282. DXUtil_ReadIntRegKey( hkey, TEXT("Geom"), (DWORD*)&config.geom, config.geom );
  1283. DXUtil_ReadIntRegKey( hkey, TEXT("TwoSided"), (DWORD*)&config.two_sided, config.two_sided );
  1284. ReadScreenSettings( hkey );
  1285. RegCloseKey( hkey );
  1286. }
  1287. }
  1288. //-----------------------------------------------------------------------------
  1289. // Name: ss_ReadSettings()
  1290. // Desc:
  1291. //-----------------------------------------------------------------------------
  1292. VOID CFlowerBoxScreensaver::ss_ReadSettings()
  1293. {
  1294. if( !ss_RegistrySetup( IDS_INI_SECTION, IDS_INIFILE ) )
  1295. return;
  1296. config.smooth_colors =
  1297. ss_GetRegistryInt( IDS_CONFIG_SMOOTH_COLORS, config.smooth_colors );
  1298. config.triangle_colors =
  1299. ss_GetRegistryInt( IDS_CONFIG_TRIANGLE_COLORS, config.triangle_colors );
  1300. config.cycle_colors =
  1301. ss_GetRegistryInt( IDS_CONFIG_CYCLE_COLORS, config.cycle_colors );
  1302. config.spin =
  1303. ss_GetRegistryInt( IDS_CONFIG_SPIN, config.spin );
  1304. config.bloom =
  1305. ss_GetRegistryInt( IDS_CONFIG_BLOOM, config.bloom );
  1306. config.subdiv =
  1307. ss_GetRegistryInt( IDS_CONFIG_SUBDIV, config.subdiv );
  1308. config.color_pick =
  1309. ss_GetRegistryInt( IDS_CONFIG_COLOR_PICK, config.color_pick );
  1310. config.image_size =
  1311. ss_GetRegistryInt( IDS_CONFIG_IMAGE_SIZE, config.image_size );
  1312. config.geom =
  1313. ss_GetRegistryInt( IDS_CONFIG_GEOM, config.geom );
  1314. config.two_sided =
  1315. ss_GetRegistryInt( IDS_CONFIG_TWO_SIDED, config.two_sided );
  1316. }
  1317. //-----------------------------------------------------------------------------
  1318. // Name: ss_GetRegistryString()
  1319. // Desc:
  1320. //-----------------------------------------------------------------------------
  1321. BOOL CFlowerBoxScreensaver::ss_RegistrySetup( int section, int file )
  1322. {
  1323. if( LoadString(m_hInstance, section, g_szSectName, BUF_SIZE) &&
  1324. LoadString(m_hInstance, file, g_szFname, BUF_SIZE) )
  1325. {
  1326. TCHAR pBuffer[100];
  1327. DWORD dwRealSize = GetPrivateProfileSection( g_szSectName, pBuffer, 100, g_szFname );
  1328. if( dwRealSize > 0 )
  1329. return TRUE;
  1330. }
  1331. return FALSE;
  1332. }
  1333. //-----------------------------------------------------------------------------
  1334. // Name: ss_GetRegistryString()
  1335. // Desc:
  1336. //-----------------------------------------------------------------------------
  1337. int CFlowerBoxScreensaver::ss_GetRegistryInt( int name, int iDefault )
  1338. {
  1339. TCHAR szItemName[BUF_SIZE];
  1340. if( LoadString( m_hInstance, name, szItemName, BUF_SIZE ) )
  1341. return GetPrivateProfileInt(g_szSectName, szItemName, iDefault, g_szFname);
  1342. return 0;
  1343. }
  1344. //-----------------------------------------------------------------------------
  1345. // Name: ss_GetRegistryString()
  1346. // Desc:
  1347. //-----------------------------------------------------------------------------
  1348. VOID CFlowerBoxScreensaver::ss_GetRegistryString( int name, LPTSTR lpDefault,
  1349. LPTSTR lpDest, int bufSize )
  1350. {
  1351. TCHAR szItemName[BUF_SIZE];
  1352. if( LoadString( m_hInstance, name, szItemName, BUF_SIZE ) )
  1353. GetPrivateProfileString(g_szSectName, szItemName, lpDefault, lpDest,
  1354. bufSize, g_szFname);
  1355. return;
  1356. }
  1357. //-----------------------------------------------------------------------------
  1358. // Name: DoConfig()
  1359. // Desc:
  1360. //-----------------------------------------------------------------------------
  1361. VOID CFlowerBoxScreensaver::DoConfig()
  1362. {
  1363. if( IDOK == DialogBox( NULL, MAKEINTRESOURCE( DLG_SCRNSAVECONFIGURE ),
  1364. m_hWndParent, ScreenSaverConfigureDialog ) )
  1365. {
  1366. NewConfig(&temp_config);
  1367. }
  1368. }