449 lines
13 KiB
C++
449 lines
13 KiB
C++
//
|
|
// SuperTuxKart - a fun racing game with go-kart
|
|
// Copyright (C) 2013-2015 SuperTuxKart-Team
|
|
//
|
|
// 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 3
|
|
// 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., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
|
|
|
|
#include <cassert>
|
|
#include <algorithm>
|
|
|
|
#include "config/user_config.hpp"
|
|
#include "input/multitouch_device.hpp"
|
|
#include "karts/abstract_kart.hpp"
|
|
#include "karts/controller/controller.hpp"
|
|
#include "graphics/irr_driver.hpp"
|
|
#include "guiengine/modaldialog.hpp"
|
|
#include "guiengine/screen_keyboard.hpp"
|
|
|
|
// ----------------------------------------------------------------------------
|
|
/** The multitouch device constructor
|
|
*/
|
|
MultitouchDevice::MultitouchDevice()
|
|
{
|
|
m_configuration = NULL;
|
|
m_type = DT_MULTITOUCH;
|
|
m_name = "Multitouch";
|
|
m_player = NULL;
|
|
m_controller = NULL;
|
|
#ifdef ANDROID
|
|
m_android_device = dynamic_cast<CIrrDeviceAndroid*>(
|
|
irr_driver->getDevice());
|
|
assert(m_android_device != NULL);
|
|
#endif
|
|
|
|
reset();
|
|
updateConfigParams();
|
|
} // MultitouchDevice
|
|
|
|
// ----------------------------------------------------------------------------
|
|
/** The multitouch device destructor
|
|
*/
|
|
MultitouchDevice::~MultitouchDevice()
|
|
{
|
|
clearButtons();
|
|
}
|
|
|
|
// ----------------------------------------------------------------------------
|
|
/** Returns a number of fingers that are currently in use
|
|
*/
|
|
unsigned int MultitouchDevice::getActiveTouchesCount()
|
|
{
|
|
unsigned int count = 0;
|
|
|
|
for (MultitouchEvent event : m_events)
|
|
{
|
|
if (event.touched)
|
|
count++;
|
|
}
|
|
|
|
return count;
|
|
} // getActiveTouchesCount
|
|
|
|
// ----------------------------------------------------------------------------
|
|
/** Creates a button of specified type and position. The button is then updated
|
|
* when touch event occurs and proper action is sent to player controller.
|
|
* Note that it just determines the screen area that is considered as button
|
|
* and it doesn't draw the GUI element on a screen.
|
|
* \param type The button type that determines its behaviour.
|
|
* \param x Vertical position of the button.
|
|
* \param y Horizontal position of the button.
|
|
* \param width Width of the button.
|
|
* \param height Height of the button.
|
|
*/
|
|
void MultitouchDevice::addButton(MultitouchButtonType type, int x, int y,
|
|
int width, int height)
|
|
{
|
|
assert(width > 0 && height > 0);
|
|
|
|
MultitouchButton* button = new MultitouchButton();
|
|
button->type = type;
|
|
button->event_id = 0;
|
|
button->pressed = false;
|
|
button->x = x;
|
|
button->y = y;
|
|
button->width = width;
|
|
button->height = height;
|
|
button->axis_x = 0.0f;
|
|
button->axis_y = 0.0f;
|
|
|
|
switch (button->type)
|
|
{
|
|
case MultitouchButtonType::BUTTON_FIRE:
|
|
button->action = PA_FIRE;
|
|
break;
|
|
case MultitouchButtonType::BUTTON_NITRO:
|
|
button->action = PA_NITRO;
|
|
break;
|
|
case MultitouchButtonType::BUTTON_SKIDDING:
|
|
button->action = PA_DRIFT;
|
|
break;
|
|
case MultitouchButtonType::BUTTON_LOOK_BACKWARDS:
|
|
button->action = PA_LOOK_BACK;
|
|
break;
|
|
case MultitouchButtonType::BUTTON_RESCUE:
|
|
button->action = PA_RESCUE;
|
|
break;
|
|
case MultitouchButtonType::BUTTON_ESCAPE:
|
|
button->action = PA_PAUSE_RACE;
|
|
break;
|
|
case MultitouchButtonType::BUTTON_UP:
|
|
button->action = PA_ACCEL;
|
|
break;
|
|
case MultitouchButtonType::BUTTON_DOWN:
|
|
button->action = PA_BRAKE;
|
|
break;
|
|
case MultitouchButtonType::BUTTON_LEFT:
|
|
button->action = PA_STEER_LEFT;
|
|
break;
|
|
case MultitouchButtonType::BUTTON_RIGHT:
|
|
button->action = PA_STEER_RIGHT;
|
|
break;
|
|
default:
|
|
button->action = PA_BEFORE_FIRST;
|
|
break;
|
|
}
|
|
|
|
m_buttons.push_back(button);
|
|
} // addButton
|
|
|
|
// ----------------------------------------------------------------------------
|
|
/** Deletes all previously created buttons
|
|
*/
|
|
void MultitouchDevice::clearButtons()
|
|
{
|
|
for (MultitouchButton* button : m_buttons)
|
|
{
|
|
delete button;
|
|
}
|
|
|
|
m_buttons.clear();
|
|
} // clearButtons
|
|
|
|
// ----------------------------------------------------------------------------
|
|
/** Sets all buttons and events to default state
|
|
*/
|
|
void MultitouchDevice::reset()
|
|
{
|
|
for (MultitouchButton* button : m_buttons)
|
|
{
|
|
button->pressed = false;
|
|
button->event_id = 0;
|
|
button->axis_x = 0.0f;
|
|
button->axis_y = 0.0f;
|
|
}
|
|
|
|
for (MultitouchEvent& event : m_events)
|
|
{
|
|
event.id = 0;
|
|
event.touched = false;
|
|
event.x = 0;
|
|
event.y = 0;
|
|
}
|
|
} // reset
|
|
|
|
// ----------------------------------------------------------------------------
|
|
/** Activates accelerometer
|
|
*/
|
|
void MultitouchDevice::activateAccelerometer()
|
|
{
|
|
#ifdef ANDROID
|
|
if (!m_android_device->isAccelerometerActive())
|
|
{
|
|
m_android_device->activateAccelerometer(1.0f / 30);
|
|
}
|
|
#endif
|
|
}
|
|
|
|
// ----------------------------------------------------------------------------
|
|
/** Deativates accelerometer
|
|
*/
|
|
void MultitouchDevice::deactivateAccelerometer()
|
|
{
|
|
#ifdef ANDROID
|
|
if (m_android_device->isAccelerometerActive())
|
|
{
|
|
m_android_device->deactivateAccelerometer();
|
|
}
|
|
#endif
|
|
}
|
|
|
|
// ----------------------------------------------------------------------------
|
|
/** Get accelerometer state
|
|
* \return true if accelerometer is active
|
|
*/
|
|
bool MultitouchDevice::isAccelerometerActive()
|
|
{
|
|
#ifdef ANDROID
|
|
return m_android_device->isAccelerometerActive();
|
|
#endif
|
|
|
|
return false;
|
|
}
|
|
|
|
// ----------------------------------------------------------------------------
|
|
/** The function that is executed when touch event occurs. It updates the
|
|
* buttons state when it's needed.
|
|
* \param event_id The id of touch event that should be processed.
|
|
*/
|
|
void MultitouchDevice::updateDeviceState(unsigned int event_id)
|
|
{
|
|
assert(event_id < m_events.size());
|
|
|
|
MultitouchButton* pressed_button = NULL;
|
|
|
|
for (MultitouchButton* button : m_buttons)
|
|
{
|
|
if (button->pressed && button->event_id == event_id)
|
|
{
|
|
pressed_button = button;
|
|
break;
|
|
}
|
|
}
|
|
|
|
for (MultitouchButton* button : m_buttons)
|
|
{
|
|
if (pressed_button != NULL && button != pressed_button)
|
|
continue;
|
|
|
|
bool update_controls = false;
|
|
bool prev_button_state = button->pressed;
|
|
MultitouchEvent event = m_events[event_id];
|
|
|
|
if (pressed_button != NULL ||
|
|
(event.x >= button->x && event.x <= button->x + button->width &&
|
|
event.y >= button->y && event.y <= button->y + button->height))
|
|
{
|
|
button->pressed = event.touched;
|
|
button->event_id = event_id;
|
|
|
|
if (button->type == MultitouchButtonType::BUTTON_STEERING)
|
|
{
|
|
float prev_axis_x = button->axis_x;
|
|
|
|
if (button->pressed == true)
|
|
{
|
|
button->axis_x =
|
|
(float)(event.x - button->x) / (button->width/2) - 1;
|
|
}
|
|
else
|
|
{
|
|
button->axis_x = 0.0f;
|
|
}
|
|
|
|
if (prev_axis_x != button->axis_x)
|
|
{
|
|
update_controls = true;
|
|
}
|
|
}
|
|
|
|
if (button->type == MultitouchButtonType::BUTTON_STEERING ||
|
|
button->type == MultitouchButtonType::BUTTON_UP_DOWN)
|
|
{
|
|
float prev_axis_y = button->axis_y;
|
|
|
|
if (button->pressed == true)
|
|
{
|
|
button->axis_y =
|
|
(float)(event.y - button->y) / (button->height/2) - 1;
|
|
}
|
|
else
|
|
{
|
|
button->axis_y = 0.0f;
|
|
}
|
|
|
|
if (prev_axis_y != button->axis_y)
|
|
{
|
|
update_controls = true;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (prev_button_state != button->pressed)
|
|
{
|
|
update_controls = true;
|
|
}
|
|
|
|
if (update_controls)
|
|
{
|
|
handleControls(button);
|
|
}
|
|
|
|
}
|
|
} // updateDeviceState
|
|
|
|
// ----------------------------------------------------------------------------
|
|
/** Updates config parameters i.e. when they are modified in options
|
|
*/
|
|
void MultitouchDevice::updateConfigParams()
|
|
{
|
|
m_deadzone_center = UserConfigParams::m_multitouch_deadzone_center;
|
|
m_deadzone_center = std::min(std::max(m_deadzone_center, 0.0f), 0.5f);
|
|
|
|
m_deadzone_edge = UserConfigParams::m_multitouch_deadzone_edge;
|
|
m_deadzone_edge = std::min(std::max(m_deadzone_edge, 0.0f), 0.5f);
|
|
} // updateConfigParams
|
|
|
|
// ----------------------------------------------------------------------------
|
|
/** Helper function that returns a steering factor for steering button.
|
|
* \param value The axis value from 0 to 1.
|
|
*/
|
|
float MultitouchDevice::getSteeringFactor(float value)
|
|
{
|
|
if (m_deadzone_edge + m_deadzone_center >= 1.0f)
|
|
return 1.0f;
|
|
|
|
assert(m_deadzone_edge + m_deadzone_center != 1.0f);
|
|
|
|
return std::min((value - m_deadzone_center) / (1.0f - m_deadzone_edge -
|
|
m_deadzone_center), 1.0f);
|
|
}
|
|
|
|
// ----------------------------------------------------------------------------
|
|
|
|
void MultitouchDevice::updateAxisX(float value)
|
|
{
|
|
if (m_controller == NULL)
|
|
return;
|
|
|
|
if (value < -m_deadzone_center)
|
|
{
|
|
float factor = getSteeringFactor(std::abs(value));
|
|
m_controller->action(PA_STEER_LEFT, int(factor * Input::MAX_VALUE));
|
|
}
|
|
else if (value > m_deadzone_center)
|
|
{
|
|
float factor = getSteeringFactor(std::abs(value));
|
|
m_controller->action(PA_STEER_RIGHT, int(factor * Input::MAX_VALUE));
|
|
}
|
|
else
|
|
{
|
|
m_controller->action(PA_STEER_LEFT, 0);
|
|
m_controller->action(PA_STEER_RIGHT, 0);
|
|
}
|
|
}
|
|
|
|
// ----------------------------------------------------------------------------
|
|
|
|
void MultitouchDevice::updateAxisY(float value)
|
|
{
|
|
if (m_controller == NULL)
|
|
return;
|
|
|
|
if (value < -m_deadzone_center)
|
|
{
|
|
float factor = getSteeringFactor(std::abs(value));
|
|
m_controller->action(PA_ACCEL, int(factor * Input::MAX_VALUE));
|
|
}
|
|
else if (value > m_deadzone_center)
|
|
{
|
|
float factor = getSteeringFactor(std::abs(value));
|
|
m_controller->action(PA_BRAKE, int(factor * Input::MAX_VALUE));
|
|
}
|
|
else
|
|
{
|
|
m_controller->action(PA_BRAKE, 0);
|
|
m_controller->action(PA_ACCEL, 0);
|
|
}
|
|
}
|
|
|
|
// ----------------------------------------------------------------------------
|
|
|
|
/** Sends proper action for player controller depending on the button type
|
|
* and state.
|
|
* \param button The button that should be handled.
|
|
*/
|
|
void MultitouchDevice::handleControls(MultitouchButton* button)
|
|
{
|
|
if (m_controller == NULL)
|
|
return;
|
|
|
|
if (button->type == MultitouchButtonType::BUTTON_STEERING)
|
|
{
|
|
updateAxisX(button->axis_x);
|
|
updateAxisY(button->axis_y);
|
|
}
|
|
else if (button->type == MultitouchButtonType::BUTTON_UP_DOWN)
|
|
{
|
|
updateAxisY(button->axis_y);
|
|
}
|
|
else if (button->type == MultitouchButtonType::BUTTON_ESCAPE)
|
|
{
|
|
StateManager::get()->escapePressed();
|
|
}
|
|
else
|
|
{
|
|
if (button->action != PA_BEFORE_FIRST)
|
|
{
|
|
int value = button->pressed ? Input::MAX_VALUE : 0;
|
|
m_controller->action(button->action, value);
|
|
}
|
|
}
|
|
}
|
|
|
|
// ----------------------------------------------------------------------------
|
|
|
|
void MultitouchDevice::updateController()
|
|
{
|
|
if (m_player == NULL)
|
|
{
|
|
m_controller = NULL;
|
|
return;
|
|
}
|
|
|
|
// Handle multitouch events only when race is running. It avoids to process
|
|
// it when pause dialog is active during the race. And there is no reason
|
|
// to use it for GUI navigation.
|
|
if (StateManager::get()->getGameState() != GUIEngine::GAME ||
|
|
GUIEngine::ModalDialog::isADialogActive() ||
|
|
GUIEngine::ScreenKeyboard::isActive() ||
|
|
race_manager->isWatchingReplay())
|
|
{
|
|
m_controller = NULL;
|
|
return;
|
|
}
|
|
|
|
AbstractKart* pk = m_player->getKart();
|
|
|
|
if (pk == NULL)
|
|
{
|
|
m_controller = NULL;
|
|
return;
|
|
}
|
|
|
|
m_controller = pk->getController();
|
|
}
|
|
|
|
// ----------------------------------------------------------------------------
|