scummvm/backends/platform/android/events.cpp

643 lines
14 KiB
C++
Raw Normal View History

/* ScummVM - Graphic Adventure Engine
*
* ScummVM is the legal property of its developers, whose names
* are too numerous to list here. Please refer to the COPYRIGHT
* file distributed with this source distribution.
*
* This program is free software; you can redistribute it and/or
* modify it under the terms of the GNU General Public License
* as published by the Free Software Foundation; either version 2
* of the License, or (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
*
*/
#if defined(__ANDROID__)
// Allow use of stuff in <time.h>
#define FORBIDDEN_SYMBOL_EXCEPTION_time_h
// Disable printf override in common/forbidden.h to avoid
// clashes with log.h from the Android SDK.
// That header file uses
// __attribute__ ((format(printf, 3, 4)))
// which gets messed up by our override mechanism; this could
// be avoided by either changing the Android SDK to use the equally
// legal and valid
// __attribute__ ((format(printf, 3, 4)))
// or by refining our printf override to use a varadic macro
// (which then wouldn't be portable, though).
// Anyway, for now we just disable the printf override globally
// for the Android port
#define FORBIDDEN_SYMBOL_EXCEPTION_printf
#include "backends/platform/android/android.h"
#include "backends/platform/android/graphics.h"
#include "backends/platform/android/events.h"
#include "backends/platform/android/jni-android.h"
// floating point. use sparingly
template<class T>
static inline T scalef(T in, float numerator, float denominator) {
return static_cast<float>(in) * numerator / denominator;
}
static const int kQueuedInputEventDelay = 50;
void OSystem_Android::pushEvent(int type, int arg1, int arg2, int arg3,
int arg4, int arg5, int arg6) {
Common::Event e;
switch (type) {
case JE_SYS_KEY:
case JE_KEY:
2011-03-18 21:22:48 +01:00
switch (arg1) {
case JACTION_DOWN:
e.type = Common::EVENT_KEYDOWN;
break;
case JACTION_UP:
e.type = Common::EVENT_KEYUP;
break;
default:
LOGE("unhandled jaction on key: %d", arg1);
return;
}
if (arg2 < 1 || arg2 > ARRAYSIZE(jkeymap)) {
if (arg3 < 1) {
LOGE("received invalid keycode: %d (%d)", arg2, arg3);
return;
} else {
// lets bet on the ascii code
e.kbd.keycode = Common::KEYCODE_INVALID;
}
} else {
e.kbd.keycode = jkeymap[arg2];
2011-03-18 21:22:48 +01:00
}
if (arg5 > 0)
2017-09-09 16:43:55 +02:00
e.kbdRepeat = true;
2011-03-18 21:22:48 +01:00
// map special keys to 'our' ascii codes
switch (e.kbd.keycode) {
case Common::KEYCODE_BACKSPACE:
e.kbd.ascii = Common::ASCII_BACKSPACE;
break;
case Common::KEYCODE_TAB:
e.kbd.ascii = Common::ASCII_TAB;
break;
case Common::KEYCODE_RETURN:
e.kbd.ascii = Common::ASCII_RETURN;
break;
case Common::KEYCODE_ESCAPE:
e.kbd.ascii = Common::ASCII_ESCAPE;
break;
case Common::KEYCODE_SPACE:
e.kbd.ascii = Common::ASCII_SPACE;
break;
case Common::KEYCODE_F1:
e.kbd.ascii = Common::ASCII_F1;
break;
case Common::KEYCODE_F2:
e.kbd.ascii = Common::ASCII_F2;
break;
case Common::KEYCODE_F3:
e.kbd.ascii = Common::ASCII_F3;
break;
case Common::KEYCODE_F4:
e.kbd.ascii = Common::ASCII_F4;
break;
case Common::KEYCODE_F5:
e.kbd.ascii = Common::ASCII_F5;
break;
case Common::KEYCODE_F6:
e.kbd.ascii = Common::ASCII_F6;
break;
case Common::KEYCODE_F7:
e.kbd.ascii = Common::ASCII_F7;
break;
case Common::KEYCODE_F8:
e.kbd.ascii = Common::ASCII_F8;
break;
case Common::KEYCODE_F9:
e.kbd.ascii = Common::ASCII_F9;
break;
case Common::KEYCODE_F10:
e.kbd.ascii = Common::ASCII_F10;
break;
case Common::KEYCODE_F11:
e.kbd.ascii = Common::ASCII_F11;
break;
case Common::KEYCODE_F12:
e.kbd.ascii = Common::ASCII_F12;
break;
default:
e.kbd.ascii = arg3;
break;
}
2011-03-18 21:22:48 +01:00
if (arg4 & JMETA_SHIFT_MASK)
2011-03-18 21:22:48 +01:00
e.kbd.flags |= Common::KBD_SHIFT;
// JMETA_ALT is Fn on physical keyboards!
// when mapping this to ALT - as we know it from PC keyboards - all
// Fn combos will be broken (like Fn+q, which needs to end as 1 and
// not ALT+1). Do not want.
//if (arg4 & JMETA_ALT_MASK)
// e.kbd.flags |= Common::KBD_ALT;
if (arg4 & (JMETA_SYM_ON | JMETA_CTRL_MASK))
2011-03-18 21:22:48 +01:00
e.kbd.flags |= Common::KBD_CTRL;
pushEvent(e);
2011-03-18 21:22:48 +01:00
return;
case JE_DPAD:
switch (arg2) {
case JKEYCODE_DPAD_UP:
case JKEYCODE_DPAD_DOWN:
case JKEYCODE_DPAD_LEFT:
case JKEYCODE_DPAD_RIGHT:
2011-03-18 22:14:54 +01:00
if (arg1 != JACTION_DOWN)
return;
2011-03-18 22:14:54 +01:00
e.type = Common::EVENT_MOUSEMOVE;
e.mouse = dynamic_cast<AndroidGraphicsManager *>(_graphicsManager)->getMousePosition();
2011-03-18 22:14:54 +01:00
{
int16 *c;
int s;
if (arg2 == JKEYCODE_DPAD_UP || arg2 == JKEYCODE_DPAD_DOWN) {
c = &e.mouse.y;
s = _eventScaleY;
} else {
c = &e.mouse.x;
s = _eventScaleX;
}
// the longer the button held, the faster the pointer is
// TODO put these values in some option dlg?
int f = CLIP(arg5, 1, 8) * _dpad_scale * 100 / s;
2011-03-18 21:22:48 +01:00
if (arg2 == JKEYCODE_DPAD_UP || arg2 == JKEYCODE_DPAD_LEFT)
*c -= f;
else
*c += f;
2011-03-18 22:14:54 +01:00
}
pushEvent(e);
return;
case JKEYCODE_DPAD_CENTER:
switch (arg1) {
case JACTION_DOWN:
e.type = Common::EVENT_LBUTTONDOWN;
break;
case JACTION_UP:
e.type = Common::EVENT_LBUTTONUP;
break;
default:
LOGE("unhandled jaction on dpad key: %d", arg1);
return;
}
e.mouse = dynamic_cast<AndroidGraphicsManager *>(_graphicsManager)->getMousePosition();
pushEvent(e);
return;
}
case JE_DOWN:
_touch_pt_down = dynamic_cast<AndroidGraphicsManager *>(_graphicsManager)->getMousePosition();
_touch_pt_scroll.x = -1;
_touch_pt_scroll.y = -1;
break;
case JE_SCROLL:
e.type = Common::EVENT_MOUSEMOVE;
if (_touchpad_mode) {
if (_touch_pt_scroll.x == -1 && _touch_pt_scroll.y == -1) {
_touch_pt_scroll.x = arg3;
_touch_pt_scroll.y = arg4;
return;
}
e.mouse.x = (arg3 - _touch_pt_scroll.x) * 100 / _touchpad_scale;
e.mouse.y = (arg4 - _touch_pt_scroll.y) * 100 / _touchpad_scale;
e.mouse += _touch_pt_down;
} else {
e.mouse.x = arg3;
e.mouse.y = arg4;
}
pushEvent(e);
return;
case JE_TAP:
2011-04-06 18:08:38 +02:00
if (_fingersDown > 0) {
_fingersDown = 0;
return;
}
e.type = Common::EVENT_MOUSEMOVE;
if (_touchpad_mode) {
e.mouse = dynamic_cast<AndroidGraphicsManager *>(_graphicsManager)->getMousePosition();
} else {
e.mouse.x = arg1;
e.mouse.y = arg2;
}
{
Common::EventType down, up;
// TODO put these values in some option dlg?
if (arg3 > 1000) {
down = Common::EVENT_MBUTTONDOWN;
up = Common::EVENT_MBUTTONUP;
} else if (arg3 > 500) {
down = Common::EVENT_RBUTTONDOWN;
up = Common::EVENT_RBUTTONUP;
} else {
down = Common::EVENT_LBUTTONDOWN;
up = Common::EVENT_LBUTTONUP;
}
lockMutex(_event_queue_lock);
if (_queuedEventTime)
_event_queue.push(_queuedEvent);
if (!_touchpad_mode)
_event_queue.push(e);
e.type = down;
_event_queue.push(e);
e.type = up;
_queuedEvent = e;
_queuedEventTime = getMillis() + kQueuedInputEventDelay;
unlockMutex(_event_queue_lock);
}
return;
case JE_DOUBLE_TAP:
e.type = Common::EVENT_MOUSEMOVE;
if (_touchpad_mode) {
e.mouse = dynamic_cast<AndroidGraphicsManager *>(_graphicsManager)->getMousePosition();
} else {
e.mouse.x = arg1;
e.mouse.y = arg2;
}
{
Common::EventType dptype = Common::EVENT_INVALID;
switch (arg3) {
case JACTION_DOWN:
dptype = Common::EVENT_LBUTTONDOWN;
_touch_pt_dt.x = -1;
_touch_pt_dt.y = -1;
break;
case JACTION_UP:
dptype = Common::EVENT_LBUTTONUP;
break;
// held and moved
case JACTION_MULTIPLE:
if (_touch_pt_dt.x == -1 && _touch_pt_dt.y == -1) {
_touch_pt_dt.x = arg1;
_touch_pt_dt.y = arg2;
return;
}
dptype = Common::EVENT_MOUSEMOVE;
if (_touchpad_mode) {
e.mouse.x = (arg1 - _touch_pt_dt.x) * 100 / _touchpad_scale;
e.mouse.y = (arg2 - _touch_pt_dt.y) * 100 / _touchpad_scale;
e.mouse += _touch_pt_down;
}
break;
default:
LOGE("unhandled jaction on double tap: %d", arg3);
return;
}
lockMutex(_event_queue_lock);
_event_queue.push(e);
e.type = dptype;
_event_queue.push(e);
unlockMutex(_event_queue_lock);
}
return;
2011-04-06 18:08:38 +02:00
case JE_MULTI:
switch (arg2) {
case JACTION_POINTER_DOWN:
if (arg1 > _fingersDown)
_fingersDown = arg1;
return;
case JACTION_POINTER_UP:
if (arg1 != _fingersDown)
return;
{
Common::EventType up;
switch (_fingersDown) {
case 1:
e.type = Common::EVENT_RBUTTONDOWN;
up = Common::EVENT_RBUTTONUP;
break;
case 2:
e.type = Common::EVENT_MBUTTONDOWN;
up = Common::EVENT_MBUTTONUP;
break;
default:
LOGD("unmapped multi tap: %d", _fingersDown);
return;
}
e.mouse = dynamic_cast<AndroidGraphicsManager *>(_graphicsManager)->getMousePosition();
2011-04-06 18:08:38 +02:00
lockMutex(_event_queue_lock);
if (_queuedEventTime)
_event_queue.push(_queuedEvent);
2011-04-06 18:08:38 +02:00
_event_queue.push(e);
2011-04-06 18:08:38 +02:00
e.type = up;
_queuedEvent = e;
_queuedEventTime = getMillis() + kQueuedInputEventDelay;
2011-04-06 18:08:38 +02:00
unlockMutex(_event_queue_lock);
return;
default:
LOGE("unhandled jaction on multi tap: %d", arg2);
return;
}
}
return;
case JE_BALL:
e.mouse = dynamic_cast<AndroidGraphicsManager *>(_graphicsManager)->getMousePosition();
2011-03-18 21:05:46 +01:00
switch (arg1) {
case JACTION_DOWN:
e.type = Common::EVENT_LBUTTONDOWN;
break;
case JACTION_UP:
e.type = Common::EVENT_LBUTTONUP;
break;
case JACTION_MULTIPLE:
e.type = Common::EVENT_MOUSEMOVE;
// already multiplied by 100
e.mouse.x += arg2 * _trackball_scale / _eventScaleX;
e.mouse.y += arg3 * _trackball_scale / _eventScaleY;
2011-03-18 21:05:46 +01:00
break;
default:
LOGE("unhandled jaction on system key: %d", arg1);
return;
}
pushEvent(e);
return;
case JE_MOUSE_MOVE:
e.type = Common::EVENT_MOUSEMOVE;
e.mouse.x = arg1;
e.mouse.y = arg2;
pushEvent(e);
return;
case JE_LMB_DOWN:
e.type = Common::EVENT_LBUTTONDOWN;
e.mouse.x = arg1;
e.mouse.y = arg2;
pushEvent(e);
return;
case JE_LMB_UP:
e.type = Common::EVENT_LBUTTONUP;
e.mouse.x = arg1;
e.mouse.y = arg2;
pushEvent(e);
return;
case JE_RMB_DOWN:
e.type = Common::EVENT_RBUTTONDOWN;
e.mouse.x = arg1;
e.mouse.y = arg2;
pushEvent(e);
return;
case JE_RMB_UP:
e.type = Common::EVENT_RBUTTONUP;
e.mouse.x = arg1;
e.mouse.y = arg2;
pushEvent(e);
return;
2013-08-08 14:08:30 +02:00
case JE_GAMEPAD:
switch (arg1) {
case JACTION_DOWN:
e.type = Common::EVENT_KEYDOWN;
break;
case JACTION_UP:
e.type = Common::EVENT_KEYUP;
break;
default:
LOGE("unhandled jaction on gamepad key: %d", arg1);
return;
}
switch (arg2) {
case JKEYCODE_BUTTON_A:
case JKEYCODE_BUTTON_B:
switch (arg1) {
case JACTION_DOWN:
e.type = (arg2 == JKEYCODE_BUTTON_A?
Common::EVENT_LBUTTONDOWN :
Common::EVENT_RBUTTONDOWN);
break;
case JACTION_UP:
e.type = (arg2 == JKEYCODE_BUTTON_A?
Common::EVENT_LBUTTONUP :
Common::EVENT_RBUTTONUP);
break;
}
e.mouse = dynamic_cast<AndroidGraphicsManager *>(_graphicsManager)->getMousePosition();
2013-08-08 14:08:30 +02:00
break;
case JKEYCODE_BUTTON_X:
e.kbd.keycode = Common::KEYCODE_ESCAPE;
e.kbd.ascii = Common::ASCII_ESCAPE;
break;
case JKEYCODE_BUTTON_Y:
e.type = Common::EVENT_MAINMENU;
break;
2013-08-08 14:08:30 +02:00
default:
LOGW("unmapped gamepad key: %d", arg2);
return;
}
pushEvent(e);
2013-08-08 14:08:30 +02:00
break;
case JE_JOYSTICK:
e.mouse = dynamic_cast<AndroidGraphicsManager *>(_graphicsManager)->getMousePosition();
switch (arg1) {
case JACTION_MULTIPLE:
e.type = Common::EVENT_MOUSEMOVE;
// already multiplied by 100
e.mouse.x += arg2 * _joystick_scale / _eventScaleX;
e.mouse.y += arg3 * _joystick_scale / _eventScaleY;
break;
default:
LOGE("unhandled jaction on joystick: %d", arg1);
return;
}
pushEvent(e);
return;
case JE_MMB_DOWN:
e.type = Common::EVENT_MAINMENU;
pushEvent(e);
return;
case JE_MMB_UP:
// No action
return;
case JE_QUIT:
e.type = Common::EVENT_QUIT;
pushEvent(e);
return;
default:
LOGE("unknown jevent type: %d", type);
break;
}
}
bool OSystem_Android::pollEvent(Common::Event &event) {
//ENTER();
if (pthread_self() == _main_thread) {
if (_screen_changeid != JNI::surface_changeid) {
_screen_changeid = JNI::surface_changeid;
if (JNI::egl_surface_width > 0 && JNI::egl_surface_height > 0) {
// surface changed
dynamic_cast<AndroidGraphicsManager *>(_graphicsManager)->deinitSurface();
dynamic_cast<AndroidGraphicsManager *>(_graphicsManager)->initSurface();
event.type = Common::EVENT_SCREEN_CHANGED;
return true;
} else {
// surface lost
dynamic_cast<AndroidGraphicsManager *>(_graphicsManager)->deinitSurface();
}
}
if (JNI::pause) {
LOGD("main thread going to sleep");
sem_wait(&JNI::pause_sem);
LOGD("main thread woke up");
}
}
lockMutex(_event_queue_lock);
if (_queuedEventTime && (getMillis() > _queuedEventTime)) {
event = _queuedEvent;
_queuedEventTime = 0;
// unlockMutex(_event_queue_lock);
// return true;
} else if (_event_queue.empty()) {
unlockMutex(_event_queue_lock);
return false;
} else {
event = _event_queue.pop();
}
unlockMutex(_event_queue_lock);
switch (event.type) {
case Common::EVENT_MOUSEMOVE:
case Common::EVENT_LBUTTONDOWN:
case Common::EVENT_LBUTTONUP:
case Common::EVENT_RBUTTONDOWN:
case Common::EVENT_RBUTTONUP:
if (_graphicsManager)
return dynamic_cast<AndroidGraphicsManager *>(_graphicsManager)->notifyMousePosition(event.mouse);
break;
default:
break;
}
return true;
}
void OSystem_Android::pushEvent(const Common::Event &event) {
lockMutex(_event_queue_lock);
_event_queue.push(event);
unlockMutex(_event_queue_lock);
}
#endif