+ HandleKey(button_key, KEY_PRESSED);
+
+ Error(ERR_DEBUG, "---------- DROP STARTED ----------");
+ }
+ }
+ }
+}
+
+#else
+
+void HandleFingerEvent(FingerEvent *event)
+{
+#if 0
+ static int num_events = 0;
+ int max_events = 10;
+#endif
+
+#if 0
+#if DEBUG_EVENTS
+ Error(ERR_DEBUG, "FINGER EVENT: finger was %s, touch ID %lld, finger ID %lld, x/y %f/%f, dx/dy %f/%f, pressure %f",
+ event->type == EVENT_FINGERPRESS ? "pressed" :
+ event->type == EVENT_FINGERRELEASE ? "released" : "moved",
+ event->touchId,
+ event->fingerId,
+ event->x, event->y,
+ event->dx, event->dy,
+ event->pressure);
+#endif
+#endif
+
+#if 0
+ int x = (int)(event->x * video.width);
+ int y = (int)(event->y * video.height);
+ int button = MB_LEFTBUTTON;
+
+ Error(ERR_DEBUG, "=> screen x/y %d/%d", x, y);
+#endif
+
+#if 0
+ if (++num_events >= max_events)
+ CloseAllAndExit(0);
+#endif
+
+#if 1
+#if 0
+ if (event->type == EVENT_FINGERPRESS ||
+ event->type == EVENT_FINGERMOTION)
+ button_status = button;
+ else
+ button_status = MB_RELEASED;
+
+ int max_x = SX + SXSIZE;
+ int max_y = SY + SYSIZE;
+#endif
+
+#if 1
+ if (game_status == GAME_MODE_PLAYING)
+#else
+ if (game_status == GAME_MODE_PLAYING &&
+ x < max_x)
+#endif
+ {
+ int key_status = (event->type == EVENT_FINGERRELEASE ? KEY_RELEASED :
+ KEY_PRESSED);
+#if 1
+ Key key = (event->y < 1.0 / 3.0 ? setup.input[0].key.up :
+ event->y > 2.0 / 3.0 ? setup.input[0].key.down :
+ event->x < 1.0 / 3.0 ? setup.input[0].key.left :
+ event->x > 2.0 / 3.0 ? setup.input[0].key.right :
+ setup.input[0].key.drop);
+#else
+ Key key = (y < max_y / 3 ? setup.input[0].key.up :
+ y > 2 * max_y / 3 ? setup.input[0].key.down :
+ x < max_x / 3 ? setup.input[0].key.left :
+ x > 2 * max_x / 3 ? setup.input[0].key.right :
+ setup.input[0].key.drop);
+#endif
+
+ Error(ERR_DEBUG, "=> key == %d, key_status == %d", key, key_status);
+
+ HandleKey(key, key_status);
+ }
+ else
+ {
+#if 0
+ Error(ERR_DEBUG, "::: button_status == %d, button == %d\n",
+ button_status, button);
+
+ HandleButton(x, y, button_status, button);
+#endif
+ }
+#endif
+}
+
+#endif
+
+static boolean checkTextInputKeyModState()
+{
+ // when playing, only handle raw key events and ignore text input
+ if (game_status == GAME_MODE_PLAYING)
+ return FALSE;
+
+ return ((GetKeyModState() & KMOD_TextInput) != KMOD_None);
+}
+
+void HandleTextEvent(TextEvent *event)
+{
+ char *text = event->text;
+ Key key = getKeyFromKeyName(text);
+
+#if DEBUG_EVENTS
+ Error(ERR_DEBUG, "TEXT EVENT: text == '%s' [%d byte(s), '%c'/%d], resulting key == %d (%s) [%04x]",
+ text,
+ strlen(text),
+ text[0], (int)(text[0]),
+ key,
+ getKeyNameFromKey(key),
+ GetKeyModState());
+#endif
+
+ // if (game_status != GAME_MODE_PLAYING && GetKeyModState() != KMOD_None)
+ /*
+ if (game_status != GAME_MODE_PLAYING &&
+ (GetKeyModState() & KMOD_TextInput) != KMOD_None)
+ */
+ if (checkTextInputKeyModState())
+ {
+ HandleKey(key, KEY_PRESSED);
+ HandleKey(key, KEY_RELEASED);
+ }
+}
+#endif
+
+void HandleKeyEvent(KeyEvent *event)
+{
+ int key_status = (event->type == EVENT_KEYPRESS ? KEY_PRESSED : KEY_RELEASED);
+ boolean with_modifiers = (game_status == GAME_MODE_PLAYING ? FALSE : TRUE);
+ Key key = GetEventKey(event, with_modifiers);
+ Key keymod = (with_modifiers ? GetEventKey(event, FALSE) : key);
+
+#if DEBUG_EVENTS
+ Error(ERR_DEBUG, "KEY EVENT: key was %s, keysym.scancode == %d, keysym.sym == %d, keymod = %d, GetKeyModState() = 0x%04x, resulting key == %d (%s)",
+ event->type == EVENT_KEYPRESS ? "pressed" : "released",
+ event->keysym.scancode,
+ event->keysym.sym,
+ keymod,
+ GetKeyModState(),
+ key,
+ getKeyNameFromKey(key));
+#endif
+
+#if 0
+ if (key == KSYM_Menu)
+ Error(ERR_DEBUG, "menu key pressed");
+ else if (key == KSYM_Back)
+ Error(ERR_DEBUG, "back key pressed");
+#endif
+
+#if defined(PLATFORM_ANDROID)
+ // always map the "back" button to the "escape" key on Android devices
+ if (key == KSYM_Back)
+ key = KSYM_Escape;
+#endif
+
+ HandleKeyModState(keymod, key_status);
+
+#if defined(TARGET_SDL2)
+
+ // if (game_status == GAME_MODE_PLAYING || GetKeyModState() == KMOD_None)
+ /*
+ if (game_status == GAME_MODE_PLAYING ||
+ (GetKeyModState() & KMOD_TextInput) == KMOD_None)
+ */
+ if (!checkTextInputKeyModState())
+ HandleKey(key, key_status);
+#else
+ HandleKey(key, key_status);
+#endif
+}
+
+void HandleFocusEvent(FocusChangeEvent *event)
+{
+ static int old_joystick_status = -1;
+
+ if (event->type == EVENT_FOCUSOUT)
+ {
+ KeyboardAutoRepeatOn();
+ old_joystick_status = joystick.status;
+ joystick.status = JOYSTICK_NOT_AVAILABLE;
+
+ ClearPlayerAction();
+ }
+ else if (event->type == EVENT_FOCUSIN)
+ {
+ /* When there are two Rocks'n'Diamonds windows which overlap and
+ the player moves the pointer from one game window to the other,
+ a 'FocusOut' event is generated for the window the pointer is
+ leaving and a 'FocusIn' event is generated for the window the
+ pointer is entering. In some cases, it can happen that the
+ 'FocusIn' event is handled by the one game process before the
+ 'FocusOut' event by the other game process. In this case the
+ X11 environment would end up with activated keyboard auto repeat,
+ because unfortunately this is a global setting and not (which
+ would be far better) set for each X11 window individually.
+ The effect would be keyboard auto repeat while playing the game
+ (game_status == GAME_MODE_PLAYING), which is not desired.
+ To avoid this special case, we just wait 1/10 second before
+ processing the 'FocusIn' event.
+ */
+
+ if (game_status == GAME_MODE_PLAYING)
+ {
+ Delay(100);
+ KeyboardAutoRepeatOffUnlessAutoplay();
+ }
+
+ if (old_joystick_status != -1)
+ joystick.status = old_joystick_status;
+ }
+}
+
+void HandleClientMessageEvent(ClientMessageEvent *event)
+{
+ if (CheckCloseWindowEvent(event))
+ CloseAllAndExit(0);
+}
+
+void HandleWindowManagerEvent(Event *event)
+{
+#if defined(TARGET_SDL)
+ SDLHandleWindowManagerEvent(event);
+#endif
+}
+
+void HandleButton(int mx, int my, int button, int button_nr)
+{
+ static int old_mx = 0, old_my = 0;
+
+ if (button < 0)
+ {
+ mx = old_mx;
+ my = old_my;
+ button = -button;
+ }
+ else
+ {
+ old_mx = mx;
+ old_my = my;
+ }
+
+ if (HandleGadgets(mx, my, button))
+ {
+ /* do not handle this button event anymore */
+ mx = my = -32; /* force mouse event to be outside screen tiles */
+ }
+
+ /* do not use scroll wheel button events for anything other than gadgets */
+ if (IS_WHEEL_BUTTON(button_nr))
+ return;
+
+#if 0
+ Error(ERR_DEBUG, "::: game_status == %d", game_status);
+#endif
+
+ switch (game_status)
+ {
+ case GAME_MODE_TITLE:
+ HandleTitleScreen(mx, my, 0, 0, button);
+ break;
+
+ case GAME_MODE_MAIN:
+ HandleMainMenu(mx, my, 0, 0, button);
+ break;
+
+ case GAME_MODE_PSEUDO_TYPENAME:
+ HandleTypeName(0, KSYM_Return);
+ break;
+
+ case GAME_MODE_LEVELS:
+ HandleChooseLevelSet(mx, my, 0, 0, button);
+ break;
+
+ case GAME_MODE_LEVELNR:
+ HandleChooseLevelNr(mx, my, 0, 0, button);
+ break;
+
+ case GAME_MODE_SCORES:
+ HandleHallOfFame(0, 0, 0, 0, button);
+ break;
+
+ case GAME_MODE_EDITOR:
+ HandleLevelEditorIdle();
+ break;
+
+ case GAME_MODE_INFO:
+ HandleInfoScreen(mx, my, 0, 0, button);
+ break;
+
+ case GAME_MODE_SETUP:
+ HandleSetupScreen(mx, my, 0, 0, button);
+ break;
+
+ case GAME_MODE_PLAYING:
+#ifdef DEBUG
+ if (button == MB_PRESSED && !motion_status && IN_GFX_SCREEN(mx, my))
+ DumpTile(LEVELX((mx - SX) / TILEX), LEVELY((my - SY) / TILEY));
+#endif