input.c 23 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122
  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. // Portions Copyright (c) 2009, netwan.
  5. //
  6. // All rights reserved.
  7. //
  8. // This file is part of SDLPAL.
  9. //
  10. // SDLPAL is free software: you can redistribute it and/or modify
  11. // it under the terms of the GNU General Public License as published by
  12. // the Free Software Foundation, either version 3 of the License, or
  13. // (at your option) any later version.
  14. //
  15. // This program is distributed in the hope that it will be useful,
  16. // but WITHOUT ANY WARRANTY; without even the implied warranty of
  17. // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  18. // GNU General Public License for more details.
  19. //
  20. // You should have received a copy of the GNU General Public License
  21. // along with this program. If not, see <http://www.gnu.org/licenses/>.
  22. //
  23. #include "main.h"
  24. #include <math.h>
  25. volatile PALINPUTSTATE g_InputState;
  26. static SDL_Joystick *g_pJoy = NULL;
  27. BOOL g_fUseJoystick = TRUE;
  28. #if defined(GPH)
  29. #define MIN_DEADZONE -16384
  30. #define MAX_DEADZONE 16384
  31. #endif
  32. #if defined(__WINPHONE__)
  33. unsigned int g_uiLastBackKeyTime = 0;
  34. #endif
  35. static VOID
  36. PAL_KeyboardEventFilter(
  37. const SDL_Event *lpEvent
  38. )
  39. /*++
  40. Purpose:
  41. Handle keyboard events.
  42. Parameters:
  43. [IN] lpEvent - pointer to the event.
  44. Return value:
  45. None.
  46. --*/
  47. {
  48. switch (lpEvent->type)
  49. {
  50. case SDL_KEYDOWN:
  51. //
  52. // Pressed a key
  53. //
  54. if (lpEvent->key.keysym.mod & KMOD_ALT)
  55. {
  56. if (lpEvent->key.keysym.sym == SDLK_RETURN)
  57. {
  58. //
  59. // Pressed Alt+Enter (toggle fullscreen)...
  60. //
  61. VIDEO_ToggleFullscreen();
  62. return;
  63. }
  64. else if (lpEvent->key.keysym.sym == SDLK_F4)
  65. {
  66. //
  67. // Pressed Alt+F4 (Exit program)...
  68. //
  69. PAL_Shutdown();
  70. exit(0);
  71. }
  72. }
  73. switch (lpEvent->key.keysym.sym)
  74. {
  75. #ifdef __SYMBIAN32__
  76. //
  77. // Symbian-specific stuff
  78. //
  79. case SDLK_0:
  80. VIDEO_ToggleScaleScreen();
  81. break;
  82. case SDLK_1:
  83. SOUND_AdjustVolume(0);
  84. break;
  85. case SDLK_3:
  86. SOUND_AdjustVolume(1);
  87. break;
  88. #endif
  89. #ifdef __WINPHONE__
  90. case SDLK_AC_BACK:
  91. if (g_uiLastBackKeyTime != 0 && SDL_GetTicks() - g_uiLastBackKeyTime < 800)
  92. {
  93. PAL_Shutdown();
  94. exit(0);
  95. }
  96. g_uiLastBackKeyTime = SDL_GetTicks();
  97. break;
  98. #endif
  99. case SDLK_UP:
  100. case SDLK_KP8:
  101. g_InputState.prevdir = (gpGlobals->fInBattle ? kDirUnknown : g_InputState.dir);
  102. g_InputState.dir = kDirNorth;
  103. g_InputState.dwKeyPress |= kKeyUp;
  104. break;
  105. case SDLK_DOWN:
  106. case SDLK_KP2:
  107. g_InputState.prevdir = (gpGlobals->fInBattle ? kDirUnknown : g_InputState.dir);
  108. g_InputState.dir = kDirSouth;
  109. g_InputState.dwKeyPress |= kKeyDown;
  110. break;
  111. case SDLK_LEFT:
  112. case SDLK_KP4:
  113. g_InputState.prevdir = (gpGlobals->fInBattle ? kDirUnknown : g_InputState.dir);
  114. g_InputState.dir = kDirWest;
  115. g_InputState.dwKeyPress |= kKeyLeft;
  116. break;
  117. case SDLK_RIGHT:
  118. case SDLK_KP6:
  119. g_InputState.prevdir = (gpGlobals->fInBattle ? kDirUnknown : g_InputState.dir);
  120. g_InputState.dir = kDirEast;
  121. g_InputState.dwKeyPress |= kKeyRight;
  122. break;
  123. #if defined(DINGOO)
  124. case SDLK_SPACE:
  125. g_InputState.dwKeyPress = kKeyMenu;
  126. break;
  127. case SDLK_LCTRL:
  128. g_InputState.dwKeyPress = kKeySearch;
  129. break;
  130. #else
  131. case SDLK_ESCAPE:
  132. case SDLK_INSERT:
  133. case SDLK_LALT:
  134. case SDLK_RALT:
  135. case SDLK_KP0:
  136. g_InputState.dwKeyPress |= kKeyMenu;
  137. break;
  138. case SDLK_RETURN:
  139. case SDLK_SPACE:
  140. case SDLK_KP_ENTER:
  141. case SDLK_LCTRL:
  142. g_InputState.dwKeyPress |= kKeySearch;
  143. break;
  144. case SDLK_PAGEUP:
  145. case SDLK_KP9:
  146. g_InputState.dwKeyPress |= kKeyPgUp;
  147. break;
  148. case SDLK_PAGEDOWN:
  149. case SDLK_KP3:
  150. g_InputState.dwKeyPress |= kKeyPgDn;
  151. break;
  152. case SDLK_7: //7 for mobile device
  153. case SDLK_r:
  154. g_InputState.dwKeyPress |= kKeyRepeat;
  155. break;
  156. case SDLK_2: //2 for mobile device
  157. case SDLK_a:
  158. g_InputState.dwKeyPress |= kKeyAuto;
  159. break;
  160. case SDLK_d:
  161. g_InputState.dwKeyPress |= kKeyDefend;
  162. break;
  163. case SDLK_e:
  164. g_InputState.dwKeyPress |= kKeyUseItem;
  165. break;
  166. case SDLK_w:
  167. g_InputState.dwKeyPress |= kKeyThrowItem;
  168. break;
  169. case SDLK_q:
  170. g_InputState.dwKeyPress |= kKeyFlee;
  171. break;
  172. case SDLK_s:
  173. g_InputState.dwKeyPress |= kKeyStatus;
  174. break;
  175. case SDLK_f:
  176. case SDLK_5: // 5 for mobile device
  177. g_InputState.dwKeyPress |= kKeyForce;
  178. break;
  179. case SDLK_HASH: //# for mobile device
  180. case SDLK_p:
  181. VIDEO_SaveScreenshot();
  182. break;
  183. #endif
  184. default:
  185. break;
  186. }
  187. break;
  188. case SDL_KEYUP:
  189. //
  190. // Released a key
  191. //
  192. switch (lpEvent->key.keysym.sym)
  193. {
  194. case SDLK_UP:
  195. case SDLK_KP8:
  196. if (g_InputState.dir == kDirNorth)
  197. {
  198. g_InputState.dir = g_InputState.prevdir;
  199. }
  200. g_InputState.prevdir = kDirUnknown;
  201. break;
  202. case SDLK_DOWN:
  203. case SDLK_KP2:
  204. if (g_InputState.dir == kDirSouth)
  205. {
  206. g_InputState.dir = g_InputState.prevdir;
  207. }
  208. g_InputState.prevdir = kDirUnknown;
  209. break;
  210. case SDLK_LEFT:
  211. case SDLK_KP4:
  212. if (g_InputState.dir == kDirWest)
  213. {
  214. g_InputState.dir = g_InputState.prevdir;
  215. }
  216. g_InputState.prevdir = kDirUnknown;
  217. break;
  218. case SDLK_RIGHT:
  219. case SDLK_KP6:
  220. if (g_InputState.dir == kDirEast)
  221. {
  222. g_InputState.dir = g_InputState.prevdir;
  223. }
  224. g_InputState.prevdir = kDirUnknown;
  225. break;
  226. default:
  227. break;
  228. }
  229. break;
  230. }
  231. }
  232. static VOID
  233. PAL_MouseEventFilter(
  234. const SDL_Event *lpEvent
  235. )
  236. /*++
  237. Purpose:
  238. Handle mouse events.
  239. Parameters:
  240. [IN] lpEvent - pointer to the event.
  241. Return value:
  242. None.
  243. --*/
  244. {
  245. #ifdef PAL_HAS_MOUSE
  246. static short hitTest = 0; // Double click detect;
  247. const SDL_VideoInfo *vi;
  248. double screenWidth, gridWidth;
  249. double screenHeight, gridHeight;
  250. double mx, my;
  251. double thumbx;
  252. double thumby;
  253. INT gridIndex;
  254. BOOL isLeftMouseDBClick = FALSE;
  255. BOOL isLeftMouseClick = FALSE;
  256. BOOL isRightMouseClick = FALSE;
  257. static INT lastReleaseButtonTime, lastPressButtonTime, betweenTime;
  258. static INT lastPressx = 0;
  259. static INT lastPressy = 0;
  260. static INT lastReleasex = 0;
  261. static INT lastReleasey = 0;
  262. if (lpEvent->type!= SDL_MOUSEBUTTONDOWN && lpEvent->type != SDL_MOUSEBUTTONUP)
  263. return;
  264. vi = SDL_GetVideoInfo();
  265. screenWidth = vi->current_w;
  266. screenHeight = vi->current_h;
  267. gridWidth = screenWidth / 3;
  268. gridHeight = screenHeight / 3;
  269. mx = lpEvent->button.x;
  270. my = lpEvent->button.y;
  271. thumbx = ceil(mx / gridWidth);
  272. thumby = floor(my / gridHeight);
  273. gridIndex = thumbx + thumby * 3 - 1;
  274. switch (lpEvent->type)
  275. {
  276. case SDL_MOUSEBUTTONDOWN:
  277. lastPressButtonTime = SDL_GetTicks();
  278. lastPressx = lpEvent->button.x;
  279. lastPressy = lpEvent->button.y;
  280. switch (gridIndex)
  281. {
  282. case 2:
  283. g_InputState.prevdir = g_InputState.dir;
  284. g_InputState.dir = kDirNorth;
  285. break;
  286. case 6:
  287. g_InputState.prevdir = g_InputState.dir;
  288. g_InputState.dir = kDirSouth;
  289. break;
  290. case 0:
  291. g_InputState.prevdir = g_InputState.dir;
  292. g_InputState.dir = kDirWest;
  293. break;
  294. case 8:
  295. g_InputState.prevdir = g_InputState.dir;
  296. g_InputState.dir = kDirEast;
  297. break;
  298. case 1:
  299. //g_InputState.prevdir = g_InputState.dir;
  300. //g_InputState.dir = kDirNorth;
  301. g_InputState.dwKeyPress |= kKeyUp;
  302. break;
  303. case 7:
  304. //g_InputState.prevdir = g_InputState.dir;
  305. //g_InputState.dir = kDirSouth;
  306. g_InputState.dwKeyPress |= kKeyDown;
  307. break;
  308. case 3:
  309. //g_InputState.prevdir = g_InputState.dir;
  310. //g_InputState.dir = kDirWest;
  311. g_InputState.dwKeyPress |= kKeyLeft;
  312. break;
  313. case 5:
  314. //g_InputState.prevdir = g_InputState.dir;
  315. //g_InputState.dir = kDirEast;
  316. g_InputState.dwKeyPress |= kKeyRight;
  317. break;
  318. }
  319. break;
  320. case SDL_MOUSEBUTTONUP:
  321. lastReleaseButtonTime = SDL_GetTicks();
  322. lastReleasex = lpEvent->button.x;
  323. lastReleasey = lpEvent->button.y;
  324. hitTest ++;
  325. if (abs(lastPressx - lastReleasex) < 25 &&
  326. abs(lastPressy - lastReleasey) < 25)
  327. {
  328. betweenTime = lastReleaseButtonTime - lastPressButtonTime;
  329. if (betweenTime >500)
  330. {
  331. isRightMouseClick = TRUE;
  332. }
  333. else if (betweenTime >=0)
  334. {
  335. if((betweenTime < 100) && (hitTest >= 2))
  336. {
  337. isLeftMouseClick = TRUE;
  338. hitTest = 0;
  339. }
  340. else
  341. {
  342. isLeftMouseClick = TRUE;
  343. if(betweenTime > 100)
  344. {
  345. hitTest = 0;
  346. }
  347. }
  348. }
  349. }
  350. switch (gridIndex)
  351. {
  352. case 2:
  353. if( isLeftMouseDBClick )
  354. {
  355. SOUND_AdjustVolume(1);
  356. break;
  357. }
  358. case 6:
  359. case 0:
  360. if( isLeftMouseDBClick )
  361. {
  362. SOUND_AdjustVolume(0);
  363. break;
  364. }
  365. case 7:
  366. if (isRightMouseClick) //repeat attack
  367. {
  368. g_InputState.dwKeyPress |= kKeyRepeat;
  369. break;
  370. }
  371. case 8:
  372. g_InputState.dir = kDirUnknown;
  373. g_InputState.prevdir = kDirUnknown;
  374. break;
  375. case 1:
  376. if( isRightMouseClick )
  377. {
  378. g_InputState.dwKeyPress |= kKeyForce;
  379. }
  380. break;
  381. case 3:
  382. if( isRightMouseClick )
  383. {
  384. g_InputState.dwKeyPress |= kKeyAuto;
  385. }
  386. break;
  387. case 5:
  388. if( isRightMouseClick )
  389. {
  390. g_InputState.dwKeyPress |= kKeyDefend;
  391. }
  392. break;
  393. case 4:
  394. if (isRightMouseClick) // menu
  395. {
  396. g_InputState.dwKeyPress |= kKeyMenu;
  397. }
  398. else if (isLeftMouseClick) // search
  399. {
  400. g_InputState.dwKeyPress |= kKeySearch;
  401. }
  402. break;
  403. }
  404. break;
  405. }
  406. #endif
  407. }
  408. static VOID
  409. PAL_JoystickEventFilter(
  410. const SDL_Event *lpEvent
  411. )
  412. /*++
  413. Purpose:
  414. Handle joystick events.
  415. Parameters:
  416. [IN] lpEvent - pointer to the event.
  417. Return value:
  418. None.
  419. --*/
  420. {
  421. #ifdef PAL_HAS_JOYSTICKS
  422. switch (lpEvent->type)
  423. {
  424. #if defined (GEKKO)
  425. case SDL_JOYHATMOTION:
  426. switch (lpEvent->jhat.value)
  427. {
  428. case SDL_HAT_LEFT:
  429. g_InputState.prevdir = (gpGlobals->fInBattle ? kDirUnknown : g_InputState.dir);
  430. g_InputState.dir = kDirWest;
  431. g_InputState.dwKeyPress = kKeyLeft;
  432. break;
  433. case SDL_HAT_RIGHT:
  434. g_InputState.prevdir = (gpGlobals->fInBattle ? kDirUnknown : g_InputState.dir);
  435. g_InputState.dir = kDirEast;
  436. g_InputState.dwKeyPress = kKeyRight;
  437. break;
  438. case SDL_HAT_UP:
  439. g_InputState.prevdir = (gpGlobals->fInBattle ? kDirUnknown : g_InputState.dir);
  440. g_InputState.dir = kDirNorth;
  441. g_InputState.dwKeyPress = kKeyUp;
  442. break;
  443. case SDL_HAT_DOWN:
  444. g_InputState.prevdir = (gpGlobals->fInBattle ? kDirUnknown : g_InputState.dir);
  445. g_InputState.dir = kDirSouth;
  446. g_InputState.dwKeyPress = kKeyDown;
  447. break;
  448. }
  449. break;
  450. #else
  451. case SDL_JOYAXISMOTION:
  452. //
  453. // Moved an axis on joystick
  454. //
  455. switch (lpEvent->jaxis.axis)
  456. {
  457. case 0:
  458. //
  459. // X axis
  460. //
  461. #if defined(GPH)
  462. if (lpEvent->jaxis.value > MAX_DEADZONE) {
  463. g_InputState.prevdir = (gpGlobals->fInBattle ? kDirUnknown : g_InputState.dir);
  464. g_InputState.dir = kDirEast;
  465. g_InputState.dwKeyPress = kKeyRight;
  466. } else if (lpEvent->jaxis.value < MIN_DEADZONE) {
  467. g_InputState.prevdir = (gpGlobals->fInBattle ? kDirUnknown : g_InputState.dir);
  468. g_InputState.dir = kDirWest;
  469. g_InputState.dwKeyPress = kKeyLeft;
  470. } else {
  471. g_InputState.dir = kDirUnknown;
  472. }
  473. #else
  474. if (lpEvent->jaxis.value > 20000)
  475. {
  476. if (g_InputState.dir != kDirEast)
  477. {
  478. g_InputState.dwKeyPress |= kKeyRight;
  479. }
  480. g_InputState.prevdir = g_InputState.dir;
  481. g_InputState.dir = kDirEast;
  482. }
  483. else if (lpEvent->jaxis.value < -20000)
  484. {
  485. if (g_InputState.dir != kDirWest)
  486. {
  487. g_InputState.dwKeyPress |= kKeyLeft;
  488. }
  489. g_InputState.prevdir = g_InputState.dir;
  490. g_InputState.dir = kDirWest;
  491. }
  492. else
  493. {
  494. if (g_InputState.prevdir != kDirEast &&
  495. g_InputState.prevdir != kDirWest)
  496. {
  497. g_InputState.dir = g_InputState.prevdir;
  498. }
  499. g_InputState.prevdir = kDirUnknown;
  500. }
  501. #endif
  502. break;
  503. case 1:
  504. //
  505. // Y axis
  506. //
  507. #if defined(GPH)
  508. if (lpEvent->jaxis.value > MAX_DEADZONE) {
  509. g_InputState.prevdir = (gpGlobals->fInBattle ? kDirUnknown : g_InputState.dir);
  510. g_InputState.dir = kDirSouth;
  511. g_InputState.dwKeyPress = kKeyDown;
  512. } else if (lpEvent->jaxis.value < MIN_DEADZONE) {
  513. g_InputState.prevdir = (gpGlobals->fInBattle ? kDirUnknown : g_InputState.dir);
  514. g_InputState.dir = kDirNorth;
  515. g_InputState.dwKeyPress = kKeyUp;
  516. } else {
  517. g_InputState.dir = kDirUnknown;
  518. }
  519. #else
  520. if (lpEvent->jaxis.value > 20000)
  521. {
  522. if (g_InputState.dir != kDirSouth)
  523. {
  524. g_InputState.dwKeyPress |= kKeyDown;
  525. }
  526. g_InputState.prevdir = g_InputState.dir;
  527. g_InputState.dir = kDirSouth;
  528. }
  529. else if (lpEvent->jaxis.value < -20000)
  530. {
  531. if (g_InputState.dir != kDirNorth)
  532. {
  533. g_InputState.dwKeyPress |= kKeyUp;
  534. }
  535. g_InputState.prevdir = g_InputState.dir;
  536. g_InputState.dir = kDirNorth;
  537. }
  538. else
  539. {
  540. if (g_InputState.prevdir != kDirNorth &&
  541. g_InputState.prevdir != kDirSouth)
  542. {
  543. g_InputState.dir = g_InputState.prevdir;
  544. }
  545. g_InputState.prevdir = kDirUnknown;
  546. }
  547. #endif
  548. break;
  549. }
  550. break;
  551. #endif
  552. case SDL_JOYBUTTONDOWN:
  553. //
  554. // Pressed the joystick button
  555. //
  556. #if defined(GPH)
  557. switch (lpEvent->jbutton.button)
  558. {
  559. #if defined(GP2XWIZ)
  560. case 14:
  561. #elif defined(CAANOO)
  562. case 3:
  563. #endif
  564. g_InputState.dwKeyPress = kKeyMenu;
  565. break;
  566. #if defined(GP2XWIZ)
  567. case 13:
  568. #elif defined(CAANOO)
  569. case 2:
  570. #endif
  571. g_InputState.dwKeyPress = kKeySearch;
  572. break;
  573. #else
  574. #if defined(GEKKO)
  575. switch (lpEvent->jbutton.button)
  576. {
  577. case 2:
  578. g_InputState.dwKeyPress |= kKeyMenu;
  579. break;
  580. case 3:
  581. g_InputState.dwKeyPress |= kKeySearch;
  582. break;
  583. #else
  584. switch (lpEvent->jbutton.button & 1)
  585. {
  586. case 0:
  587. g_InputState.dwKeyPress |= kKeyMenu;
  588. break;
  589. case 1:
  590. g_InputState.dwKeyPress |= kKeySearch;
  591. break;
  592. #endif
  593. #endif
  594. }
  595. break;
  596. }
  597. #endif
  598. }
  599. #ifdef PAL_HAS_TOUCH
  600. #define TOUCH_NONE 0
  601. #define TOUCH_UP 1
  602. #define TOUCH_DOWN 2
  603. #define TOUCH_LEFT 3
  604. #define TOUCH_RIGHT 4
  605. #define TOUCH_BUTTON1 5
  606. #define TOUCH_BUTTON2 6
  607. #define TOUCH_BUTTON3 7
  608. #define TOUCH_BUTTON4 8
  609. static int
  610. PAL_GetTouchArea(
  611. float X,
  612. float Y
  613. )
  614. {
  615. if (Y < 0.5)
  616. {
  617. //
  618. // Upper area
  619. //
  620. return TOUCH_NONE;
  621. }
  622. else if (X < 1.0 / 3)
  623. {
  624. if (Y - 0.5 < (1.0 / 6 - fabs(X - 1.0 / 3 / 2)) * (0.5 / (1.0 / 3)))
  625. {
  626. return TOUCH_UP;
  627. }
  628. else if (Y - 0.75 > fabs(X - 1.0 / 3 / 2) * (0.5 / (1.0 / 3)))
  629. {
  630. return TOUCH_DOWN;
  631. }
  632. else if (X < 1.0 / 3 / 2 && fabs(Y - 0.75) < 0.25 - X * (0.5 / (1.0 / 3)))
  633. {
  634. return TOUCH_LEFT;
  635. }
  636. else
  637. {
  638. return TOUCH_RIGHT;
  639. }
  640. }
  641. else if (X > 1.0 - 1.0 / 3)
  642. {
  643. if (X < 1.0 - (1.0 / 3 / 2))
  644. {
  645. if (Y < 0.75)
  646. {
  647. return TOUCH_BUTTON1;
  648. }
  649. else
  650. {
  651. return TOUCH_BUTTON3;
  652. }
  653. }
  654. else
  655. {
  656. if (Y < 0.75)
  657. {
  658. return TOUCH_BUTTON2;
  659. }
  660. else
  661. {
  662. return TOUCH_BUTTON4;
  663. }
  664. }
  665. }
  666. return TOUCH_NONE;
  667. }
  668. static VOID
  669. PAL_SetTouchAction(
  670. int area
  671. )
  672. {
  673. switch (area)
  674. {
  675. case TOUCH_UP:
  676. g_InputState.dir = kDirNorth;
  677. g_InputState.dwKeyPress |= kKeyUp;
  678. break;
  679. case TOUCH_DOWN:
  680. g_InputState.dir = kDirSouth;
  681. g_InputState.dwKeyPress |= kKeyDown;
  682. break;
  683. case TOUCH_LEFT:
  684. g_InputState.dir = kDirWest;
  685. g_InputState.dwKeyPress |= kKeyLeft;
  686. break;
  687. case TOUCH_RIGHT:
  688. g_InputState.dir = kDirEast;
  689. g_InputState.dwKeyPress |= kKeyRight;
  690. break;
  691. case TOUCH_BUTTON1:
  692. if (gpGlobals->fInBattle)
  693. {
  694. g_InputState.dwKeyPress |= kKeyRepeat;
  695. }
  696. else
  697. {
  698. g_InputState.dwKeyPress |= kKeyForce;
  699. }
  700. break;
  701. case TOUCH_BUTTON2:
  702. g_InputState.dwKeyPress |= kKeyMenu;
  703. break;
  704. case TOUCH_BUTTON3:
  705. g_InputState.dwKeyPress |= kKeyUseItem;
  706. break;
  707. case TOUCH_BUTTON4:
  708. g_InputState.dwKeyPress |= kKeySearch;
  709. break;
  710. }
  711. }
  712. static VOID
  713. PAL_UnsetTouchAction(
  714. int area
  715. )
  716. {
  717. switch (area)
  718. {
  719. case TOUCH_UP:
  720. case TOUCH_DOWN:
  721. case TOUCH_LEFT:
  722. case TOUCH_RIGHT:
  723. g_InputState.dir = kDirUnknown;
  724. break;
  725. }
  726. }
  727. #endif
  728. static VOID
  729. PAL_TouchEventFilter(
  730. const SDL_Event *lpEvent
  731. )
  732. /*++
  733. Purpose:
  734. Handle touch events.
  735. Parameters:
  736. [IN] lpEvent - pointer to the event.
  737. Return value:
  738. None.
  739. --*/
  740. {
  741. #ifdef PAL_HAS_TOUCH
  742. static SDL_TouchID finger1 = -1, finger2 = -1;
  743. static int prev_touch1 = TOUCH_NONE;
  744. static int prev_touch2 = TOUCH_NONE;
  745. switch (lpEvent->type)
  746. {
  747. case SDL_FINGERDOWN:
  748. if (finger1 == -1)
  749. {
  750. int area = PAL_GetTouchArea(lpEvent->tfinger.x, lpEvent->tfinger.y);
  751. finger1 = lpEvent->tfinger.fingerId;
  752. prev_touch1 = area;
  753. PAL_SetTouchAction(area);
  754. }
  755. else if (finger2 == -1)
  756. {
  757. int area = PAL_GetTouchArea(lpEvent->tfinger.x, lpEvent->tfinger.y);
  758. finger2 = lpEvent->tfinger.fingerId;
  759. prev_touch2 = area;
  760. PAL_SetTouchAction(area);
  761. }
  762. break;
  763. case SDL_FINGERUP:
  764. if (lpEvent->tfinger.fingerId == finger1)
  765. {
  766. PAL_UnsetTouchAction(prev_touch1);
  767. finger1 = -1;
  768. prev_touch1 = TOUCH_NONE;
  769. }
  770. else if (lpEvent->tfinger.fingerId == finger2)
  771. {
  772. PAL_UnsetTouchAction(prev_touch2);
  773. finger2 = -1;
  774. prev_touch2 = TOUCH_NONE;
  775. }
  776. break;
  777. case SDL_FINGERMOTION:
  778. if (lpEvent->tfinger.fingerId == finger1)
  779. {
  780. int area = PAL_GetTouchArea(lpEvent->tfinger.x, lpEvent->tfinger.y);
  781. if (prev_touch1 != area && area != TOUCH_NONE)
  782. {
  783. PAL_UnsetTouchAction(prev_touch1);
  784. prev_touch1 = area;
  785. PAL_SetTouchAction(area);
  786. }
  787. }
  788. else if (lpEvent->tfinger.fingerId == finger2)
  789. {
  790. int area = PAL_GetTouchArea(lpEvent->tfinger.x, lpEvent->tfinger.y);
  791. if (prev_touch2 != area && area != TOUCH_NONE)
  792. {
  793. PAL_UnsetTouchAction(prev_touch2);
  794. prev_touch2 = area;
  795. PAL_SetTouchAction(area);
  796. }
  797. }
  798. break;
  799. }
  800. #endif
  801. }
  802. static int SDLCALL
  803. PAL_EventFilter(
  804. const SDL_Event *lpEvent
  805. )
  806. /*++
  807. Purpose:
  808. SDL event filter function. A filter to process all events.
  809. Parameters:
  810. [IN] lpEvent - pointer to the event.
  811. Return value:
  812. 1 = the event will be added to the internal queue.
  813. 0 = the event will be dropped from the queue.
  814. --*/
  815. {
  816. switch (lpEvent->type)
  817. {
  818. #if SDL_VERSION_ATLEAST(2,0,0)
  819. case SDL_WINDOWEVENT:
  820. if (lpEvent->window.event == SDL_WINDOWEVENT_RESIZED)
  821. {
  822. //
  823. // resized the window
  824. //
  825. VIDEO_Resize(lpEvent->window.data1, lpEvent->window.data2);
  826. }
  827. break;
  828. #ifdef __IOS__
  829. case SDL_APP_WILLENTERBACKGROUND:
  830. g_bRenderPaused = TRUE;
  831. break;
  832. case SDL_APP_DIDENTERFOREGROUND:
  833. g_bRenderPaused = FALSE;
  834. VIDEO_UpdateScreen(NULL);
  835. break;
  836. #endif
  837. #else
  838. case SDL_VIDEORESIZE:
  839. //
  840. // resized the window
  841. //
  842. VIDEO_Resize(lpEvent->resize.w, lpEvent->resize.h);
  843. break;
  844. #endif
  845. case SDL_QUIT:
  846. //
  847. // clicked on the close button of the window. Quit immediately.
  848. //
  849. PAL_Shutdown();
  850. exit(0);
  851. }
  852. PAL_KeyboardEventFilter(lpEvent);
  853. PAL_MouseEventFilter(lpEvent);
  854. PAL_JoystickEventFilter(lpEvent);
  855. PAL_TouchEventFilter(lpEvent);
  856. //
  857. // All events are handled here; don't put anything to the internal queue
  858. //
  859. return 0;
  860. }
  861. VOID
  862. PAL_ClearKeyState(
  863. VOID
  864. )
  865. /*++
  866. Purpose:
  867. Clear the record of pressed keys.
  868. Parameters:
  869. None.
  870. Return value:
  871. None.
  872. --*/
  873. {
  874. g_InputState.dwKeyPress = 0;
  875. }
  876. VOID
  877. PAL_InitInput(
  878. VOID
  879. )
  880. /*++
  881. Purpose:
  882. Initialize the input subsystem.
  883. Parameters:
  884. None.
  885. Return value:
  886. None.
  887. --*/
  888. {
  889. memset((void *)&g_InputState, 0, sizeof(g_InputState));
  890. g_InputState.dir = kDirUnknown;
  891. g_InputState.prevdir = kDirUnknown;
  892. //
  893. // Check for joystick
  894. //
  895. #ifdef PAL_HAS_JOYSTICKS
  896. if (SDL_NumJoysticks() > 0 && g_fUseJoystick)
  897. {
  898. g_pJoy = SDL_JoystickOpen(0);
  899. if (g_pJoy != NULL)
  900. {
  901. SDL_JoystickEventState(SDL_ENABLE);
  902. }
  903. }
  904. #endif
  905. }
  906. VOID
  907. PAL_ShutdownInput(
  908. VOID
  909. )
  910. /*++
  911. Purpose:
  912. Shutdown the input subsystem.
  913. Parameters:
  914. None.
  915. Return value:
  916. None.
  917. --*/
  918. {
  919. #ifdef PAL_HAS_JOYSTICKS
  920. #if SDL_VERSION_ATLEAST(2,0,0)
  921. if (g_pJoy != NULL)
  922. {
  923. SDL_JoystickClose(g_pJoy);
  924. g_pJoy = NULL;
  925. }
  926. #else
  927. if (SDL_JoystickOpened(0))
  928. {
  929. assert(g_pJoy != NULL);
  930. SDL_JoystickClose(g_pJoy);
  931. g_pJoy = NULL;
  932. }
  933. #endif
  934. #endif
  935. }
  936. VOID
  937. PAL_ProcessEvent(
  938. VOID
  939. )
  940. /*++
  941. Purpose:
  942. Process all events.
  943. Parameters:
  944. None.
  945. Return value:
  946. None.
  947. --*/
  948. {
  949. #ifdef PAL_HAS_NATIVEMIDI
  950. MIDI_CheckLoop();
  951. #endif
  952. while (PAL_PollEvent(NULL));
  953. }
  954. int
  955. PAL_PollEvent(
  956. SDL_Event *event
  957. )
  958. /*++
  959. Purpose:
  960. Poll and process one event.
  961. Parameters:
  962. [OUT] event - Events polled from SDL.
  963. Return value:
  964. Return value of PAL_PollEvent.
  965. --*/
  966. {
  967. SDL_Event evt;
  968. int ret = SDL_PollEvent(&evt);
  969. if (ret != 0)
  970. {
  971. PAL_EventFilter(&evt);
  972. }
  973. if (event != NULL)
  974. {
  975. *event = evt;
  976. }
  977. return ret;
  978. }