video.c 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990
  1. /* -*- mode: c; tab-width: 4; c-basic-offset: 3; c-file-style: "linux" -*- */
  2. //
  3. // Copyright (c) 2009, Wei Mingzhi <whistler_wmz@users.sf.net>.
  4. // All rights reserved.
  5. //
  6. // This file is part of SDLPAL.
  7. //
  8. // SDLPAL is free software: you can redistribute it and/or modify
  9. // it under the terms of the GNU General Public License as published by
  10. // the Free Software Foundation, either version 3 of the License, or
  11. // (at your option) any later version.
  12. //
  13. // This program is distributed in the hope that it will be useful,
  14. // but WITHOUT ANY WARRANTY; without even the implied warranty of
  15. // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  16. // GNU General Public License for more details.
  17. //
  18. // You should have received a copy of the GNU General Public License
  19. // along with this program. If not, see <http://www.gnu.org/licenses/>.
  20. //
  21. #include "main.h"
  22. // Screen buffer
  23. SDL_Surface *gpScreen = NULL;
  24. // Backup screen buffer
  25. SDL_Surface *gpScreenBak = NULL;
  26. // The real screen surface
  27. #if SDL_VERSION_ATLEAST(2,0,0)
  28. static SDL_Window *gpWindow = NULL;
  29. static SDL_Renderer *gpRenderer = NULL;
  30. #else
  31. static SDL_Surface *gpScreenReal = NULL;
  32. #endif
  33. #if (defined (__SYMBIAN32__) && !defined (__S60_5X__)) || defined (PSP) || defined (GEKKO)
  34. static BOOL bScaleScreen = FALSE;
  35. #else
  36. static BOOL bScaleScreen = TRUE;
  37. #endif
  38. // Initial screen size
  39. static WORD g_wInitialWidth = 640;
  40. static WORD g_wInitialHeight = 400;
  41. // Shake times and level
  42. static WORD g_wShakeTime = 0;
  43. static WORD g_wShakeLevel = 0;
  44. INT
  45. #ifdef GEKKO // Rikku2000: Crash on compile, allready define on WIISDK
  46. VIDEO_Init_GEKKO(
  47. #else
  48. VIDEO_Init(
  49. #endif
  50. WORD wScreenWidth,
  51. WORD wScreenHeight,
  52. BOOL fFullScreen
  53. )
  54. /*++
  55. Purpose:
  56. Initialze the video subsystem.
  57. Parameters:
  58. [IN] wScreenWidth - width of the screen.
  59. [IN] wScreenHeight - height of the screen.
  60. [IN] fFullScreen - TRUE to use full screen mode, FALSE to use windowed mode.
  61. Return value:
  62. 0 = success, -1 = fail to create the screen surface,
  63. -2 = fail to create screen buffer.
  64. --*/
  65. {
  66. g_wInitialWidth = wScreenWidth;
  67. g_wInitialHeight = wScreenHeight;
  68. #if SDL_VERSION_ATLEAST(2,0,0)
  69. //
  70. // Before we can render anything, we need a window and a renderer.
  71. //
  72. gpWindow = SDL_CreateWindow("SDL_RenderCopy Example",
  73. SDL_WINDOWPOS_UNDEFINED, SDL_WINDOWPOS_UNDEFINED, 640, 400,
  74. SDL_WINDOW_SHOWN | SDL_WINDOW_RESIZABLE);
  75. if (gpWindow == NULL)
  76. {
  77. return -1;
  78. }
  79. gpRenderer = SDL_CreateRenderer(gpWindow, -1, SDL_RENDERER_ACCELERATED);
  80. if (gpRenderer == NULL)
  81. {
  82. return -1;
  83. }
  84. //
  85. // Create the screen buffer and the backup screen buffer.
  86. //
  87. gpScreen = SDL_CreateRGBSurface(gpScreenReal->flags & ~SDL_HWSURFACE, 320, 200, 8,
  88. gpScreenReal->format->Rmask, gpScreenReal->format->Gmask,
  89. gpScreenReal->format->Bmask, gpScreenReal->format->Amask);
  90. gpScreenBak = SDL_CreateRGBSurface(gpScreenReal->flags & ~SDL_HWSURFACE, 320, 200, 8,
  91. gpScreenReal->format->Rmask, gpScreenReal->format->Gmask,
  92. gpScreenReal->format->Bmask, gpScreenReal->format->Amask);
  93. //
  94. // Failed?
  95. //
  96. if (gpScreen == NULL || gpScreenBak == NULL)
  97. {
  98. if (gpScreen != NULL)
  99. {
  100. SDL_FreeSurface(gpScreen);
  101. gpScreen = NULL;
  102. }
  103. if (gpScreenBak != NULL)
  104. {
  105. SDL_FreeSurface(gpScreenBak);
  106. gpScreenBak = NULL;
  107. }
  108. SDL_DestroyRenderer(gpRenderer);
  109. gpRenderer = NULL;
  110. SDL_DestroyWindow(gpWindow);
  111. gpWindow = NULL;
  112. return -2;
  113. }
  114. #else
  115. //
  116. // Create the screen surface.
  117. //
  118. #if defined (NDS)
  119. gpScreenReal = SDL_SetVideoMode(293, 196, 8, SDL_SWSURFACE | SDL_FULLSCREEN);
  120. #elif defined (__SYMBIAN32__)
  121. #ifdef __S60_5X__
  122. gpScreenReal = SDL_SetVideoMode(640, 360, 8,
  123. SDL_SWSURFACE | (fFullScreen ? SDL_FULLSCREEN : 0));
  124. #else
  125. gpScreenReal = SDL_SetVideoMode(320, 240, 8,
  126. SDL_SWSURFACE | (fFullScreen ? SDL_FULLSCREEN : 0));
  127. #endif
  128. #elif defined (GEKKO)
  129. gpScreenReal = SDL_SetVideoMode(640, 480, 8,
  130. SDL_SWSURFACE | (fFullScreen ? SDL_FULLSCREEN : 0));
  131. #elif defined (PSP)
  132. gpScreenReal = SDL_SetVideoMode(320, 240, 8, SDL_SWSURFACE | SDL_FULLSCREEN);
  133. #else
  134. gpScreenReal = SDL_SetVideoMode(wScreenWidth, wScreenHeight, 8,
  135. SDL_HWSURFACE | SDL_RESIZABLE | (fFullScreen ? SDL_FULLSCREEN : 0));
  136. #endif
  137. if (gpScreenReal == NULL)
  138. {
  139. //
  140. // Fall back to 640x480 software mode.
  141. //
  142. gpScreenReal = SDL_SetVideoMode(640, 480, 8,
  143. SDL_SWSURFACE | (fFullScreen ? SDL_FULLSCREEN : 0));
  144. }
  145. //
  146. // Still fail?
  147. //
  148. if (gpScreenReal == NULL)
  149. {
  150. return -1;
  151. }
  152. //
  153. // Create the screen buffer and the backup screen buffer.
  154. //
  155. gpScreen = SDL_CreateRGBSurface(gpScreenReal->flags & ~SDL_HWSURFACE, 320, 200, 8,
  156. gpScreenReal->format->Rmask, gpScreenReal->format->Gmask,
  157. gpScreenReal->format->Bmask, gpScreenReal->format->Amask);
  158. gpScreenBak = SDL_CreateRGBSurface(gpScreenReal->flags & ~SDL_HWSURFACE, 320, 200, 8,
  159. gpScreenReal->format->Rmask, gpScreenReal->format->Gmask,
  160. gpScreenReal->format->Bmask, gpScreenReal->format->Amask);
  161. //
  162. // Failed?
  163. //
  164. if (gpScreen == NULL || gpScreenBak == NULL)
  165. {
  166. if (gpScreen != NULL)
  167. {
  168. SDL_FreeSurface(gpScreen);
  169. gpScreen = NULL;
  170. }
  171. if (gpScreenBak != NULL)
  172. {
  173. SDL_FreeSurface(gpScreenBak);
  174. gpScreenBak = NULL;
  175. }
  176. SDL_FreeSurface(gpScreenReal);
  177. gpScreenReal = NULL;
  178. return -2;
  179. }
  180. #endif
  181. if (fFullScreen)
  182. {
  183. SDL_ShowCursor(FALSE);
  184. }
  185. return 0;
  186. }
  187. VOID
  188. VIDEO_Shutdown(
  189. VOID
  190. )
  191. /*++
  192. Purpose:
  193. Shutdown the video subsystem.
  194. Parameters:
  195. None.
  196. Return value:
  197. None.
  198. --*/
  199. {
  200. if (gpScreen != NULL)
  201. {
  202. SDL_FreeSurface(gpScreen);
  203. }
  204. gpScreen = NULL;
  205. if (gpScreenBak != NULL)
  206. {
  207. SDL_FreeSurface(gpScreenBak);
  208. }
  209. gpScreenBak = NULL;
  210. #if SDL_VERSION_ATLEAST(2,0,0)
  211. if (gpRenderer)
  212. {
  213. SDL_DestroyRenderer(gpRenderer);
  214. }
  215. gpRenderer = NULL;
  216. if (gpWindow)
  217. {
  218. SDL_DestroyWindow(gpWindow);
  219. }
  220. gpWindow = NULL;
  221. #else
  222. if (gpScreenReal != NULL)
  223. {
  224. SDL_FreeSurface(gpScreenReal);
  225. }
  226. gpScreenReal = NULL;
  227. #endif
  228. }
  229. VOID
  230. VIDEO_UpdateScreen(
  231. const SDL_Rect *lpRect
  232. )
  233. /*++
  234. Purpose:
  235. Update the screen area specified by lpRect.
  236. Parameters:
  237. [IN] lpRect - Screen area to update.
  238. Return value:
  239. None.
  240. --*/
  241. {
  242. SDL_Rect srcrect, dstrect;
  243. short offset = 240 - 200;
  244. short screenRealHeight = gpScreenReal->h;
  245. short screenRealY = 0;
  246. //
  247. // Lock surface if needed
  248. //
  249. if (SDL_MUSTLOCK(gpScreenReal))
  250. {
  251. if (SDL_LockSurface(gpScreenReal) < 0)
  252. return;
  253. }
  254. if (!bScaleScreen)
  255. {
  256. screenRealHeight -= offset;
  257. screenRealY = offset / 2;
  258. }
  259. if (lpRect != NULL)
  260. {
  261. dstrect.x = (SHORT)((INT)(lpRect->x) * gpScreenReal->w / gpScreen->w);
  262. dstrect.y = (SHORT)((INT)(screenRealY + lpRect->y) * screenRealHeight / gpScreen->h);
  263. dstrect.w = (WORD)((DWORD)(lpRect->w) * gpScreenReal->w / gpScreen->w);
  264. dstrect.h = (WORD)((DWORD)(lpRect->h) * screenRealHeight / gpScreen->h);
  265. SDL_SoftStretch(gpScreen, (SDL_Rect *)lpRect, gpScreenReal, &dstrect);
  266. if (SDL_MUSTLOCK(gpScreenReal))
  267. {
  268. SDL_UnlockSurface(gpScreenReal);
  269. }
  270. SDL_UpdateRect(gpScreenReal, dstrect.x, dstrect.y, dstrect.w, dstrect.h);
  271. }
  272. else if (g_wShakeTime != 0)
  273. {
  274. //
  275. // Shake the screen
  276. //
  277. srcrect.x = 0;
  278. srcrect.y = 0;
  279. srcrect.w = 320;
  280. srcrect.h = 200 - g_wShakeLevel;
  281. dstrect.x = 0;
  282. dstrect.y = screenRealY;
  283. dstrect.w = 320 * gpScreenReal->w / gpScreen->w;
  284. dstrect.h = (200 - g_wShakeLevel) * screenRealHeight / gpScreen->h;
  285. if (g_wShakeTime & 1)
  286. {
  287. srcrect.y = g_wShakeLevel;
  288. }
  289. else
  290. {
  291. dstrect.y = (screenRealY + g_wShakeLevel) * screenRealHeight / gpScreen->h;
  292. }
  293. SDL_SoftStretch(gpScreen, &srcrect, gpScreenReal, &dstrect);
  294. if (g_wShakeTime & 1)
  295. {
  296. dstrect.y = (screenRealY + screenRealHeight - g_wShakeLevel) * screenRealHeight / gpScreen->h;
  297. }
  298. else
  299. {
  300. dstrect.y = screenRealY;
  301. }
  302. dstrect.h = g_wShakeLevel * screenRealHeight / gpScreen->h;
  303. SDL_FillRect(gpScreenReal, &dstrect, 0);
  304. if (SDL_MUSTLOCK(gpScreenReal))
  305. {
  306. SDL_UnlockSurface(gpScreenReal);
  307. }
  308. SDL_UpdateRect(gpScreenReal, 0, 0, gpScreenReal->w, gpScreenReal->h);
  309. g_wShakeTime--;
  310. }
  311. else
  312. {
  313. dstrect.x = 0;
  314. dstrect.y = screenRealY;
  315. dstrect.w = gpScreenReal->w;
  316. dstrect.h = screenRealHeight;
  317. SDL_SoftStretch(gpScreen, NULL, gpScreenReal, &dstrect);
  318. if (SDL_MUSTLOCK(gpScreenReal))
  319. {
  320. SDL_UnlockSurface(gpScreenReal);
  321. }
  322. SDL_UpdateRect(gpScreenReal, 0, 0, gpScreenReal->w, gpScreenReal->h);
  323. }
  324. }
  325. VOID
  326. VIDEO_SetPalette(
  327. SDL_Color rgPalette[256]
  328. )
  329. /*++
  330. Purpose:
  331. Set the palette of the screen.
  332. Parameters:
  333. [IN] rgPalette - array of 256 colors.
  334. Return value:
  335. None.
  336. --*/
  337. {
  338. SDL_SetPalette(gpScreenReal, SDL_LOGPAL | SDL_PHYSPAL, rgPalette, 0, 256);
  339. #if (defined (__SYMBIAN32__))
  340. {
  341. static UINT32 time = 0;
  342. if (SDL_GetTicks() - time > 50)
  343. {
  344. SDL_UpdateRect(gpScreenReal, 0, 0, gpScreenReal->w, gpScreenReal->h);
  345. time = SDL_GetTicks();
  346. }
  347. }
  348. #endif
  349. }
  350. VOID
  351. VIDEO_Resize(
  352. INT w,
  353. INT h
  354. )
  355. /*++
  356. Purpose:
  357. This function is called when user resized the window.
  358. Parameters:
  359. [IN] w - width of the window after resizing.
  360. [IN] h - height of the window after resizing.
  361. Return value:
  362. None.
  363. --*/
  364. {
  365. DWORD flags;
  366. PAL_LARGE SDL_Color palette[256];
  367. int i;
  368. //
  369. // Get the original palette.
  370. //
  371. for (i = 0; i < gpScreenReal->format->palette->ncolors; i++)
  372. {
  373. palette[i] = gpScreenReal->format->palette->colors[i];
  374. }
  375. //
  376. // Create the screen surface.
  377. //
  378. flags = gpScreenReal->flags;
  379. SDL_FreeSurface(gpScreenReal);
  380. gpScreenReal = SDL_SetVideoMode(w, h, 8, flags);
  381. if (gpScreenReal == NULL)
  382. {
  383. #ifdef __SYMBIAN32__
  384. #ifdef __S60_5X__
  385. gpScreenReal = SDL_SetVideoMode(640, 360, 8, SDL_SWSURFACE);
  386. #else
  387. gpScreenReal = SDL_SetVideoMode(320, 240, 8, SDL_SWSURFACE);
  388. #endif
  389. #else
  390. //
  391. // Fall back to 640x480 software windowed mode.
  392. //
  393. gpScreenReal = SDL_SetVideoMode(640, 480, 8, SDL_SWSURFACE);
  394. #endif
  395. }
  396. SDL_SetPalette(gpScreenReal, SDL_PHYSPAL | SDL_LOGPAL, palette, 0, i);
  397. VIDEO_UpdateScreen(NULL);
  398. }
  399. SDL_Color *
  400. VIDEO_GetPalette(
  401. VOID
  402. )
  403. /*++
  404. Purpose:
  405. Get the current palette of the screen.
  406. Parameters:
  407. None.
  408. Return value:
  409. Pointer to the current palette.
  410. --*/
  411. {
  412. return gpScreenReal->format->palette->colors;
  413. }
  414. VOID
  415. VIDEO_ToggleScaleScreen(
  416. VOID
  417. )
  418. /*++
  419. Purpose:
  420. Toggle scalescreen mode.
  421. Parameters:
  422. None.
  423. Return value:
  424. None.
  425. --*/
  426. {
  427. #ifdef __SYMBIAN32__
  428. bScaleScreen = !bScaleScreen;
  429. VIDEO_Resize(320, 240);
  430. VIDEO_UpdateScreen(NULL);
  431. #endif
  432. }
  433. VOID
  434. VIDEO_ToggleFullscreen(
  435. VOID
  436. )
  437. /*++
  438. Purpose:
  439. Toggle fullscreen mode.
  440. Parameters:
  441. None.
  442. Return value:
  443. None.
  444. --*/
  445. {
  446. DWORD flags;
  447. PAL_LARGE SDL_Color palette[256];
  448. int i;
  449. //
  450. // Get the original palette.
  451. //
  452. for (i = 0; i < gpScreenReal->format->palette->ncolors; i++)
  453. {
  454. palette[i] = gpScreenReal->format->palette->colors[i];
  455. }
  456. //
  457. // Get the flags of the original screen surface
  458. //
  459. flags = gpScreenReal->flags;
  460. if (flags & SDL_FULLSCREEN)
  461. {
  462. //
  463. // Already in fullscreen mode. Remove the fullscreen flag.
  464. //
  465. flags &= ~SDL_FULLSCREEN;
  466. flags |= SDL_RESIZABLE;
  467. SDL_ShowCursor(TRUE);
  468. }
  469. else
  470. {
  471. //
  472. // Not in fullscreen mode. Set the fullscreen flag.
  473. //
  474. flags |= SDL_FULLSCREEN;
  475. SDL_ShowCursor(FALSE);
  476. }
  477. //
  478. // Free the original screen surface
  479. //
  480. SDL_FreeSurface(gpScreenReal);
  481. //
  482. // ... and create a new one
  483. //
  484. if (g_wInitialWidth == 640 && g_wInitialHeight == 400 && (flags & SDL_FULLSCREEN))
  485. {
  486. gpScreenReal = SDL_SetVideoMode(640, 480, 8, flags);
  487. }
  488. else if (g_wInitialWidth == 640 && g_wInitialHeight == 480 && !(flags & SDL_FULLSCREEN))
  489. {
  490. gpScreenReal = SDL_SetVideoMode(640, 400, 8, flags);
  491. }
  492. else
  493. {
  494. gpScreenReal = SDL_SetVideoMode(g_wInitialWidth, g_wInitialHeight, 8, flags);
  495. }
  496. VIDEO_SetPalette(palette);
  497. //
  498. // Update the screen
  499. //
  500. VIDEO_UpdateScreen(NULL);
  501. }
  502. VOID
  503. VIDEO_SaveScreenshot(
  504. VOID
  505. )
  506. /*++
  507. Purpose:
  508. Save the screenshot of current screen to a BMP file.
  509. Parameters:
  510. None.
  511. Return value:
  512. None.
  513. --*/
  514. {
  515. int iNumBMP = 0;
  516. FILE *fp;
  517. //
  518. // Find a usable BMP filename.
  519. //
  520. for (iNumBMP = 0; iNumBMP <= 9999; iNumBMP++)
  521. {
  522. fp = fopen(va("%sscrn%.4d.bmp", PAL_PREFIX, iNumBMP), "rb");
  523. if (fp == NULL)
  524. {
  525. break;
  526. }
  527. fclose(fp);
  528. }
  529. if (iNumBMP > 9999)
  530. {
  531. return;
  532. }
  533. //
  534. // Save the screenshot.
  535. //
  536. SDL_SaveBMP(gpScreenReal, va("%sscrn%.4d.bmp", PAL_PREFIX, iNumBMP));
  537. }
  538. VOID
  539. VIDEO_BackupScreen(
  540. VOID
  541. )
  542. /*++
  543. Purpose:
  544. Backup the screen buffer.
  545. Parameters:
  546. None.
  547. Return value:
  548. None.
  549. --*/
  550. {
  551. SDL_BlitSurface(gpScreen, NULL, gpScreenBak, NULL);
  552. }
  553. VOID
  554. VIDEO_RestoreScreen(
  555. VOID
  556. )
  557. /*++
  558. Purpose:
  559. Restore the screen buffer which has been saved with VIDEO_BackupScreen().
  560. Parameters:
  561. None.
  562. Return value:
  563. None.
  564. --*/
  565. {
  566. SDL_BlitSurface(gpScreenBak, NULL, gpScreen, NULL);
  567. }
  568. VOID
  569. VIDEO_ShakeScreen(
  570. WORD wShakeTime,
  571. WORD wShakeLevel
  572. )
  573. /*++
  574. Purpose:
  575. Set the screen shake time and level.
  576. Parameters:
  577. [IN] wShakeTime - how many times should we shake the screen.
  578. [IN] wShakeLevel - level of shaking.
  579. Return value:
  580. None.
  581. --*/
  582. {
  583. g_wShakeTime = wShakeTime;
  584. g_wShakeLevel = wShakeLevel;
  585. }
  586. VOID
  587. VIDEO_SwitchScreen(
  588. WORD wSpeed
  589. )
  590. /*++
  591. Purpose:
  592. Switch the screen from the backup screen buffer to the current screen buffer.
  593. NOTE: This will destroy the backup buffer.
  594. Parameters:
  595. [IN] wSpeed - speed of fading (the larger value, the slower).
  596. Return value:
  597. None.
  598. --*/
  599. {
  600. int i, j;
  601. const int rgIndex[6] = {0, 3, 1, 5, 2, 4};
  602. SDL_Rect dstrect;
  603. short offset = 240 - 200;
  604. short screenRealHeight = gpScreenReal->h;
  605. short screenRealY = 0;
  606. if (!bScaleScreen)
  607. {
  608. screenRealHeight -= offset;
  609. screenRealY = offset / 2;
  610. }
  611. wSpeed++;
  612. wSpeed *= 10;
  613. for (i = 0; i < 6; i++)
  614. {
  615. for (j = rgIndex[i]; j < gpScreen->pitch * gpScreen->h; j += 6)
  616. {
  617. ((LPBYTE)(gpScreenBak->pixels))[j] = ((LPBYTE)(gpScreen->pixels))[j];
  618. }
  619. //
  620. // Draw the backup buffer to the screen
  621. //
  622. dstrect.x = 0;
  623. dstrect.y = screenRealY;
  624. dstrect.w = gpScreenReal->w;
  625. dstrect.h = screenRealHeight;
  626. SDL_SoftStretch(gpScreenBak, NULL, gpScreenReal, &dstrect);
  627. SDL_UpdateRect(gpScreenReal, 0, 0, gpScreenReal->w, gpScreenReal->h);
  628. UTIL_Delay(wSpeed);
  629. }
  630. }
  631. VOID
  632. VIDEO_FadeScreen(
  633. WORD wSpeed
  634. )
  635. /*++
  636. Purpose:
  637. Fade from the backup screen buffer to the current screen buffer.
  638. NOTE: This will destroy the backup buffer.
  639. Parameters:
  640. [IN] wSpeed - speed of fading (the larger value, the slower).
  641. Return value:
  642. None.
  643. --*/
  644. {
  645. int i, j, k;
  646. DWORD time;
  647. BYTE a, b;
  648. const int rgIndex[6] = {0, 3, 1, 5, 2, 4};
  649. SDL_Rect dstrect;
  650. short offset = 240 - 200;
  651. short screenRealHeight = gpScreenReal->h;
  652. short screenRealY = 0;
  653. //
  654. // Lock surface if needed
  655. //
  656. if (SDL_MUSTLOCK(gpScreenReal))
  657. {
  658. if (SDL_LockSurface(gpScreenReal) < 0)
  659. return;
  660. }
  661. if (!bScaleScreen)
  662. {
  663. screenRealHeight -= offset;
  664. screenRealY = offset / 2;
  665. }
  666. time = SDL_GetTicks();
  667. wSpeed++;
  668. wSpeed *= 10;
  669. for (i = 0; i < 12; i++)
  670. {
  671. for (j = 0; j < 6; j++)
  672. {
  673. PAL_ProcessEvent();
  674. while (SDL_GetTicks() <= time)
  675. {
  676. PAL_ProcessEvent();
  677. SDL_Delay(5);
  678. }
  679. time = SDL_GetTicks() + wSpeed;
  680. //
  681. // Blend the pixels in the 2 buffers, and put the result into the
  682. // backup buffer
  683. //
  684. for (k = rgIndex[j]; k < gpScreen->pitch * gpScreen->h; k += 6)
  685. {
  686. a = ((LPBYTE)(gpScreen->pixels))[k];
  687. b = ((LPBYTE)(gpScreenBak->pixels))[k];
  688. if (i > 0)
  689. {
  690. if ((a & 0x0F) > (b & 0x0F))
  691. {
  692. b++;
  693. }
  694. else if ((a & 0x0F) < (b & 0x0F))
  695. {
  696. b--;
  697. }
  698. }
  699. ((LPBYTE)(gpScreenBak->pixels))[k] = ((a & 0xF0) | (b & 0x0F));
  700. }
  701. //
  702. // Draw the backup buffer to the screen
  703. //
  704. if (g_wShakeTime != 0)
  705. {
  706. //
  707. // Shake the screen
  708. //
  709. SDL_Rect srcrect, dstrect;
  710. srcrect.x = 0;
  711. srcrect.y = 0;
  712. srcrect.w = 320;
  713. srcrect.h = 200 - g_wShakeLevel;
  714. dstrect.x = 0;
  715. dstrect.y = screenRealY;
  716. dstrect.w = 320 * gpScreenReal->w / gpScreen->w;
  717. dstrect.h = (200 - g_wShakeLevel) * screenRealHeight / gpScreen->h;
  718. if (g_wShakeTime & 1)
  719. {
  720. srcrect.y = g_wShakeLevel;
  721. }
  722. else
  723. {
  724. dstrect.y = (screenRealY + g_wShakeLevel) * screenRealHeight / gpScreen->h;
  725. }
  726. SDL_SoftStretch(gpScreenBak, &srcrect, gpScreenReal, &dstrect);
  727. if (g_wShakeTime & 1)
  728. {
  729. dstrect.y = (screenRealY + screenRealHeight - g_wShakeLevel) * screenRealHeight / gpScreen->h;
  730. }
  731. else
  732. {
  733. dstrect.y = screenRealY;
  734. }
  735. dstrect.h = g_wShakeLevel * screenRealHeight / gpScreen->h;
  736. SDL_FillRect(gpScreenReal, &dstrect, 0);
  737. SDL_UpdateRect(gpScreenReal, 0, 0, gpScreenReal->w, gpScreenReal->h);
  738. g_wShakeTime--;
  739. }
  740. else
  741. {
  742. dstrect.x = 0;
  743. dstrect.y = screenRealY;
  744. dstrect.w = gpScreenReal->w;
  745. dstrect.h = screenRealHeight;
  746. SDL_SoftStretch(gpScreenBak, NULL, gpScreenReal, &dstrect);
  747. SDL_UpdateRect(gpScreenReal, 0, 0, gpScreenReal->w, gpScreenReal->h);
  748. }
  749. }
  750. }
  751. if (SDL_MUSTLOCK(gpScreenReal))
  752. {
  753. SDL_UnlockSurface(gpScreenReal);
  754. }
  755. //
  756. // Draw the result buffer to the screen as the final step
  757. //
  758. VIDEO_UpdateScreen(NULL);
  759. }
  760. #if SDL_VERSION_ATLEAST(2,0,0)
  761. /*++
  762. Purpose:
  763. Set the caption of the window. For compatibility with SDL2 only.
  764. Parameters:
  765. [IN] lpszCaption - the new caption of the window.
  766. [IN] lpReserved - not used, for compatibility only.
  767. Return value:
  768. None.
  769. --*/
  770. VOID
  771. SDL_WM_SetCaption(
  772. LPCSTR lpszCaption,
  773. LPVOID lpReserved
  774. )
  775. {
  776. if (gpWindow != NULL)
  777. {
  778. SDL_SetWindowTitle(gpWindow, lpszCaption);
  779. }
  780. }
  781. #endif