video.c 22 KB

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