2001-04-15 07:18:04 +00:00
|
|
|
/*
|
|
|
|
in_common.c
|
|
|
|
|
|
|
|
general input driver
|
|
|
|
|
|
|
|
Copyright (C) 1996-1997 Id Software, Inc.
|
|
|
|
Copyright (C) 2000 Marcus Sundberg [mackan@stacken.kth.se]
|
|
|
|
Copyright (C) 1999,2000 contributors of the QuakeForge project
|
|
|
|
Please see the file "AUTHORS" for a list of contributors
|
|
|
|
|
|
|
|
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:
|
|
|
|
|
|
|
|
Free Software Foundation, Inc.
|
|
|
|
59 Temple Place - Suite 330
|
|
|
|
Boston, MA 02111-1307, USA
|
|
|
|
|
|
|
|
*/
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
|
|
# include "config.h"
|
|
|
|
#endif
|
2003-01-15 15:31:36 +00:00
|
|
|
|
2001-04-15 07:18:04 +00:00
|
|
|
#ifdef HAVE_STRING_H
|
|
|
|
# include <string.h>
|
|
|
|
#endif
|
|
|
|
#ifdef HAVE_STRINGS_H
|
|
|
|
# include <strings.h>
|
|
|
|
#endif
|
|
|
|
#ifdef HAVE_UNISTD_H
|
|
|
|
# include <unistd.h>
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#define _BSD
|
|
|
|
#include <ctype.h>
|
|
|
|
#include <errno.h>
|
2001-08-27 01:00:03 +00:00
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
2001-04-15 07:18:04 +00:00
|
|
|
|
2012-02-01 12:52:47 +00:00
|
|
|
#include "QF/cbuf.h"
|
2001-04-15 07:18:04 +00:00
|
|
|
#include "QF/cvar.h"
|
2021-08-27 00:10:21 +00:00
|
|
|
#include "QF/darray.h"
|
2001-08-14 00:03:03 +00:00
|
|
|
#include "QF/in_event.h"
|
2001-04-15 07:18:04 +00:00
|
|
|
#include "QF/input.h"
|
|
|
|
#include "QF/joystick.h"
|
|
|
|
#include "QF/keys.h"
|
|
|
|
#include "QF/mathlib.h"
|
2007-11-06 10:17:14 +00:00
|
|
|
#include "QF/sys.h"
|
2001-09-28 16:54:31 +00:00
|
|
|
#include "QF/vid.h"
|
2001-04-15 07:18:04 +00:00
|
|
|
|
2021-09-28 01:53:51 +00:00
|
|
|
#include "qfselect.h"
|
|
|
|
|
2021-08-30 00:54:36 +00:00
|
|
|
typedef struct {
|
|
|
|
in_driver_t driver;
|
|
|
|
void *data;
|
|
|
|
} in_regdriver_t;
|
|
|
|
|
2021-08-30 05:40:19 +00:00
|
|
|
static struct DARRAY_TYPE (in_regdriver_t) in_drivers = { .grow = 8 };
|
2021-09-20 06:21:11 +00:00
|
|
|
static struct DARRAY_TYPE (in_device_t) in_devices = { .grow = 8 };
|
2021-08-27 00:10:21 +00:00
|
|
|
|
2007-03-10 12:00:59 +00:00
|
|
|
VISIBLE viewdelta_t viewdelta;
|
2001-10-23 16:55:23 +00:00
|
|
|
|
2001-08-30 20:04:13 +00:00
|
|
|
cvar_t *in_grab;
|
2007-03-10 12:00:59 +00:00
|
|
|
VISIBLE cvar_t *in_amp;
|
|
|
|
VISIBLE cvar_t *in_pre_amp;
|
2001-08-17 07:26:01 +00:00
|
|
|
cvar_t *in_freelook;
|
2001-08-15 23:04:54 +00:00
|
|
|
cvar_t *in_mouse_filter;
|
2001-08-17 07:26:01 +00:00
|
|
|
cvar_t *in_mouse_amp;
|
|
|
|
cvar_t *in_mouse_pre_amp;
|
2001-04-15 07:18:04 +00:00
|
|
|
cvar_t *lookstrafe;
|
|
|
|
|
2021-09-26 06:11:07 +00:00
|
|
|
int64_t in_timeout = 10000;//10ms default timeout
|
2001-04-15 07:18:04 +00:00
|
|
|
kbutton_t in_mlook, in_klook;
|
|
|
|
kbutton_t in_strafe;
|
|
|
|
kbutton_t in_speed;
|
|
|
|
|
|
|
|
qboolean in_mouse_avail;
|
|
|
|
float in_mouse_x, in_mouse_y;
|
|
|
|
static float in_old_mouse_x, in_old_mouse_y;
|
|
|
|
|
2021-08-30 00:54:36 +00:00
|
|
|
int
|
|
|
|
IN_RegisterDriver (in_driver_t *driver, void *data)
|
|
|
|
{
|
|
|
|
in_regdriver_t rdriver = { *driver, data };
|
|
|
|
DARRAY_APPEND (&in_drivers, rdriver);
|
|
|
|
return in_drivers.size - 1;
|
|
|
|
}
|
|
|
|
|
2021-08-27 00:10:21 +00:00
|
|
|
void
|
2021-08-30 00:54:36 +00:00
|
|
|
IN_DriverData (int handle, void *data)
|
2021-08-27 00:10:21 +00:00
|
|
|
{
|
2021-08-30 00:54:36 +00:00
|
|
|
in_drivers.a[handle].data = data;
|
2021-08-27 00:10:21 +00:00
|
|
|
}
|
|
|
|
|
2021-08-30 05:40:19 +00:00
|
|
|
static int
|
2021-09-20 06:21:11 +00:00
|
|
|
in_add_device (int driver, in_device_t *device, const char *name,
|
|
|
|
const char *id)
|
2021-08-30 05:40:19 +00:00
|
|
|
{
|
|
|
|
size_t devid;
|
2021-09-20 06:21:11 +00:00
|
|
|
in_device_t indev = {
|
|
|
|
.driverid = driver,
|
|
|
|
.device = device,
|
|
|
|
.name = name,
|
|
|
|
.id = id,
|
|
|
|
};
|
2021-08-30 05:40:19 +00:00
|
|
|
|
|
|
|
for (devid = 0; devid < in_devices.size; devid++) {
|
2021-09-20 06:21:11 +00:00
|
|
|
if (in_devices.a[devid].driverid == -1) {
|
|
|
|
in_devices.a[devid] = indev;
|
2021-08-30 05:40:19 +00:00
|
|
|
return devid;
|
|
|
|
}
|
|
|
|
}
|
2021-09-20 06:21:11 +00:00
|
|
|
DARRAY_APPEND (&in_devices, indev);
|
2021-08-30 05:40:19 +00:00
|
|
|
return devid;
|
|
|
|
}
|
|
|
|
|
|
|
|
int
|
2021-09-20 06:21:11 +00:00
|
|
|
IN_AddDevice (int driver, void *device, const char *name, const char *id)
|
2021-08-30 05:40:19 +00:00
|
|
|
{
|
2021-09-20 06:21:11 +00:00
|
|
|
if ((size_t) driver >= in_drivers.size) {
|
|
|
|
Sys_Error ("IN_AddDevice: invalid driver: %d", driver);
|
|
|
|
}
|
|
|
|
|
|
|
|
int devid = in_add_device (driver, device, name, id);
|
2021-08-30 05:40:19 +00:00
|
|
|
|
|
|
|
IE_event_t event = {
|
|
|
|
.type = ie_add_device,
|
|
|
|
.when = Sys_LongTime (),
|
|
|
|
.device = {
|
|
|
|
.devid = devid,
|
|
|
|
},
|
|
|
|
};
|
|
|
|
IE_Send_Event (&event);
|
|
|
|
return devid;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
IN_RemoveDevice (int devid)
|
|
|
|
{
|
2021-09-20 06:21:11 +00:00
|
|
|
if ((size_t) devid >= in_devices.size) {
|
2021-08-30 05:40:19 +00:00
|
|
|
Sys_Error ("IN_RemoveDevice: invalid devid: %d", devid);
|
|
|
|
}
|
|
|
|
|
|
|
|
IE_event_t event = {
|
|
|
|
.type = ie_remove_device,
|
|
|
|
.when = Sys_LongTime (),
|
|
|
|
.device = {
|
|
|
|
.devid = devid,
|
|
|
|
},
|
|
|
|
};
|
|
|
|
IE_Send_Event (&event);
|
|
|
|
|
2021-09-20 06:21:11 +00:00
|
|
|
in_devices.a[devid].device = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
IN_SendConnectedDevices (void)
|
|
|
|
{
|
|
|
|
for (size_t devid = 0; devid < in_devices.size; devid++) {
|
|
|
|
if (in_devices.a[devid].driverid >= 0
|
|
|
|
&& in_devices.a[devid].device) {
|
|
|
|
IE_event_t event = {
|
|
|
|
.type = ie_add_device,
|
|
|
|
.when = Sys_LongTime (),//FIXME actual time?
|
|
|
|
.device = {
|
|
|
|
.devid = devid,
|
|
|
|
},
|
|
|
|
};
|
|
|
|
IE_Send_Event (&event);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
const char *
|
|
|
|
IN_GetDeviceName (int devid)
|
|
|
|
{
|
|
|
|
if ((size_t) devid >= in_devices.size) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
if (!in_devices.a[devid].device || in_devices.a[devid].driverid == -1) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
return in_devices.a[devid].name;
|
|
|
|
}
|
|
|
|
|
|
|
|
const char *
|
|
|
|
IN_GetDeviceId (int devid)
|
|
|
|
{
|
|
|
|
if ((size_t) devid >= in_devices.size) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
if (!in_devices.a[devid].device || in_devices.a[devid].driverid == -1) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
return in_devices.a[devid].id;
|
|
|
|
}
|
|
|
|
|
|
|
|
int
|
|
|
|
IN_AxisInfo (int devid, in_axisinfo_t *axes, int *numaxes)
|
|
|
|
{
|
|
|
|
if ((size_t) devid >= in_devices.size) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
if (!in_devices.a[devid].device || in_devices.a[devid].driverid == -1) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
int driver = in_devices.a[devid].driverid;
|
|
|
|
in_regdriver_t *rd = &in_drivers.a[driver];
|
|
|
|
if (!rd->driver.axis_info) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
rd->driver.axis_info (rd->data, in_devices.a[devid].device, axes, numaxes);
|
|
|
|
if (axes) {
|
|
|
|
for (int i = 0; i < *numaxes; i++) {
|
|
|
|
axes[i].deviceid = devid;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int
|
|
|
|
IN_ButtonInfo (int devid, in_buttoninfo_t *buttons, int *numbuttons)
|
|
|
|
{
|
|
|
|
if ((size_t) devid >= in_devices.size) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
if (!in_devices.a[devid].device || in_devices.a[devid].driverid == -1) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
int driver = in_devices.a[devid].driverid;
|
|
|
|
in_regdriver_t *rd = &in_drivers.a[driver];
|
|
|
|
if (!rd->driver.button_info) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
rd->driver.button_info (rd->data, in_devices.a[devid].device,
|
|
|
|
buttons, numbuttons);
|
|
|
|
if (buttons) {
|
|
|
|
for (int i = 0; i < *numbuttons; i++) {
|
|
|
|
buttons[i].deviceid = devid;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return 0;
|
2021-08-30 05:40:19 +00:00
|
|
|
}
|
|
|
|
|
2003-02-13 19:03:48 +00:00
|
|
|
void
|
|
|
|
IN_UpdateGrab (cvar_t *var) // called from context_*.c
|
2001-08-30 20:04:13 +00:00
|
|
|
{
|
2003-02-25 06:04:42 +00:00
|
|
|
if (var) {
|
2021-08-27 00:10:21 +00:00
|
|
|
for (size_t i = 0; i < in_drivers.size; i++) {
|
2021-08-30 00:54:36 +00:00
|
|
|
in_regdriver_t *rd = &in_drivers.a[i];
|
|
|
|
if (rd->driver.grab_input) {
|
|
|
|
rd->driver.grab_input (rd->data, var->int_val);
|
2021-08-27 00:10:21 +00:00
|
|
|
}
|
|
|
|
}
|
2001-08-30 20:04:13 +00:00
|
|
|
}
|
|
|
|
}
|
2001-08-17 07:26:01 +00:00
|
|
|
|
2001-04-15 07:18:04 +00:00
|
|
|
void
|
2003-04-08 18:45:12 +00:00
|
|
|
IN_ProcessEvents (void)
|
2001-04-15 07:18:04 +00:00
|
|
|
{
|
2021-09-28 01:53:51 +00:00
|
|
|
qf_fd_set fdset;
|
2021-09-26 06:11:07 +00:00
|
|
|
int maxfd = -1;
|
|
|
|
int64_t timeout = in_timeout;
|
|
|
|
|
2021-09-28 01:53:51 +00:00
|
|
|
QF_FD_ZERO (&fdset);
|
2021-08-27 00:10:21 +00:00
|
|
|
for (size_t i = 0; i < in_drivers.size; i++) {
|
2021-08-30 00:54:36 +00:00
|
|
|
in_regdriver_t *rd = &in_drivers.a[i];
|
2021-09-26 06:11:07 +00:00
|
|
|
if (rd->driver.add_select) {
|
|
|
|
rd->driver.add_select (&fdset, &maxfd, rd->data);
|
|
|
|
}
|
|
|
|
if (rd->driver.process_events) {
|
|
|
|
rd->driver.process_events (rd->data);
|
|
|
|
// if a driver can't use select, then we can't block in select
|
|
|
|
timeout = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (maxfd >= 0 && Sys_Select (maxfd, &fdset, timeout) > 0) {
|
|
|
|
for (size_t i = 0; i < in_drivers.size; i++) {
|
|
|
|
in_regdriver_t *rd = &in_drivers.a[i];
|
|
|
|
if (rd->driver.check_select) {
|
|
|
|
rd->driver.check_select (&fdset, rd->data);
|
|
|
|
}
|
|
|
|
}
|
2021-08-27 00:10:21 +00:00
|
|
|
}
|
2001-04-15 07:18:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
IN_Move (void)
|
|
|
|
{
|
2021-08-27 00:10:21 +00:00
|
|
|
//JOY_Move ();
|
2001-04-15 07:18:04 +00:00
|
|
|
|
|
|
|
if (!in_mouse_avail)
|
|
|
|
return;
|
|
|
|
|
2001-09-08 03:34:28 +00:00
|
|
|
in_mouse_x *= in_mouse_pre_amp->value * in_pre_amp->value;
|
|
|
|
in_mouse_y *= in_mouse_pre_amp->value * in_pre_amp->value;
|
2001-08-06 05:01:31 +00:00
|
|
|
|
2001-08-15 23:04:54 +00:00
|
|
|
if (in_mouse_filter->int_val) {
|
2001-04-15 07:18:04 +00:00
|
|
|
in_mouse_x = (in_mouse_x + in_old_mouse_x) * 0.5;
|
|
|
|
in_mouse_y = (in_mouse_y + in_old_mouse_y) * 0.5;
|
2001-08-15 23:04:54 +00:00
|
|
|
|
|
|
|
in_old_mouse_x = in_mouse_x;
|
|
|
|
in_old_mouse_y = in_mouse_y;
|
2001-04-15 07:18:04 +00:00
|
|
|
}
|
|
|
|
|
2001-09-08 03:33:57 +00:00
|
|
|
in_mouse_x *= in_mouse_amp->value * in_amp->value;
|
|
|
|
in_mouse_y *= in_mouse_amp->value * in_amp->value;
|
2001-04-15 07:18:04 +00:00
|
|
|
|
|
|
|
if ((in_strafe.state & 1) || (lookstrafe->int_val && freelook))
|
|
|
|
viewdelta.position[0] += in_mouse_x;
|
|
|
|
else
|
|
|
|
viewdelta.angles[YAW] -= in_mouse_x;
|
|
|
|
|
|
|
|
if (freelook && !(in_strafe.state & 1)) {
|
|
|
|
viewdelta.angles[PITCH] += in_mouse_y;
|
|
|
|
} else {
|
|
|
|
viewdelta.position[2] -= in_mouse_y;
|
|
|
|
}
|
|
|
|
in_mouse_x = in_mouse_y = 0.0;
|
|
|
|
}
|
|
|
|
|
2001-08-27 01:00:03 +00:00
|
|
|
/* Called at shutdown */
|
2019-07-12 14:15:26 +00:00
|
|
|
static void
|
2020-06-25 05:03:52 +00:00
|
|
|
IN_shutdown (void *data)
|
2001-04-15 07:18:04 +00:00
|
|
|
{
|
2021-08-27 00:10:21 +00:00
|
|
|
//JOY_Shutdown ();
|
2001-04-15 07:18:04 +00:00
|
|
|
|
2021-03-29 10:58:00 +00:00
|
|
|
Sys_MaskPrintf (SYS_vid, "IN_Shutdown\n");
|
2021-08-27 00:10:21 +00:00
|
|
|
for (size_t i = in_drivers.size; i-- > 0; ) {
|
2021-08-30 00:54:36 +00:00
|
|
|
in_regdriver_t *rd = &in_drivers.a[i];
|
|
|
|
if (rd->driver.shutdown) {
|
|
|
|
rd->driver.shutdown (rd->data);
|
2021-08-27 00:10:21 +00:00
|
|
|
}
|
|
|
|
}
|
2001-04-15 07:18:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2012-02-01 12:52:47 +00:00
|
|
|
IN_Init (cbuf_t *cbuf)
|
2001-04-15 07:18:04 +00:00
|
|
|
{
|
2020-06-25 05:03:52 +00:00
|
|
|
Sys_RegisterShutdown (IN_shutdown, 0);
|
2019-07-12 14:15:26 +00:00
|
|
|
|
2021-08-27 00:10:21 +00:00
|
|
|
for (size_t i = 0; i < in_drivers.size; i++) {
|
2021-08-30 00:54:36 +00:00
|
|
|
in_regdriver_t *rd = &in_drivers.a[i];
|
|
|
|
rd->driver.init (rd->data);
|
2021-08-27 00:10:21 +00:00
|
|
|
}
|
2012-02-01 12:52:47 +00:00
|
|
|
Key_Init (cbuf);
|
2021-08-27 00:10:21 +00:00
|
|
|
//JOY_Init ();
|
2001-04-15 07:18:04 +00:00
|
|
|
|
|
|
|
in_mouse_x = in_mouse_y = 0.0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
IN_Init_Cvars (void)
|
|
|
|
{
|
2012-02-01 12:52:47 +00:00
|
|
|
Key_Init_Cvars ();
|
2021-08-27 00:10:21 +00:00
|
|
|
//JOY_Init_Cvars ();
|
2001-09-18 04:53:01 +00:00
|
|
|
in_grab = Cvar_Get ("in_grab", "0", CVAR_ARCHIVE, IN_UpdateGrab,
|
2003-01-28 21:31:55 +00:00
|
|
|
"With this set to 1, quake will grab the mouse, "
|
|
|
|
"preventing loss of input focus.");
|
2001-08-17 07:26:01 +00:00
|
|
|
in_amp = Cvar_Get ("in_amp", "1", CVAR_ARCHIVE, NULL,
|
|
|
|
"global in_amp multiplier");
|
|
|
|
in_pre_amp = Cvar_Get ("in_pre_amp", "1", CVAR_ARCHIVE, NULL,
|
|
|
|
"global in_pre_amp multiplier");
|
2001-04-17 03:08:52 +00:00
|
|
|
in_freelook = Cvar_Get ("freelook", "0", CVAR_ARCHIVE, NULL,
|
2001-08-17 07:26:01 +00:00
|
|
|
"force +mlook");
|
|
|
|
in_mouse_filter = Cvar_Get ("in_mouse_filter", "0", CVAR_ARCHIVE, NULL,
|
|
|
|
"Toggle mouse input filtering.");
|
|
|
|
in_mouse_amp = Cvar_Get ("in_mouse_amp", "15", CVAR_ARCHIVE, NULL,
|
|
|
|
"mouse in_mouse_amp multiplier");
|
|
|
|
in_mouse_pre_amp = Cvar_Get ("in_mouse_pre_amp", "1", CVAR_ARCHIVE, NULL,
|
|
|
|
"mouse in_mouse_pre_amp multiplier");
|
2001-04-15 07:18:04 +00:00
|
|
|
lookstrafe = Cvar_Get ("lookstrafe", "0", CVAR_ARCHIVE, NULL,
|
|
|
|
"when mlook/klook on player will strafe");
|
|
|
|
}
|
2001-04-15 21:11:41 +00:00
|
|
|
|
2001-04-17 15:55:33 +00:00
|
|
|
void
|
|
|
|
IN_ClearStates (void)
|
|
|
|
{
|
2021-08-27 00:10:21 +00:00
|
|
|
for (size_t i = 0; i < in_drivers.size; i++) {
|
2021-08-30 00:54:36 +00:00
|
|
|
in_regdriver_t *rd = &in_drivers.a[i];
|
|
|
|
if (rd->driver.clear_states) {
|
|
|
|
rd->driver.clear_states (rd->data);
|
2021-08-27 00:10:21 +00:00
|
|
|
}
|
|
|
|
}
|
2001-04-17 15:55:33 +00:00
|
|
|
Key_ClearStates ();
|
|
|
|
}
|
2021-08-27 00:10:21 +00:00
|
|
|
|
2021-09-28 01:57:12 +00:00
|
|
|
#ifdef HAVE_EVDEV
|
2021-08-27 00:10:21 +00:00
|
|
|
extern int in_evdev_force_link;
|
|
|
|
static __attribute__((used)) int *evdev_force_link = &in_evdev_force_link;
|
2021-09-28 01:57:12 +00:00
|
|
|
#endif
|