blob: 246ea2eb9b930d83816dbef44ced46e7144f7b3c [file] [log] [blame]
/*
Copyright (C) 1996-1997 Id Software, Inc.
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., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
*/
// in_sun.c -- SUN/X mouse input handler
#include <sys/time.h>
#include <sys/types.h>
#include <unistd.h>
#include <signal.h>
#include <stdlib.h>
#include <stdio.h>
#include <string.h>
#include <sys/ipc.h>
#include <sys/shm.h>
#include <X11/Xlib.h>
#include <X11/Xutil.h>
#include <X11/Xatom.h>
#include <X11/keysym.h>
#include "quakedef.h"
//
// typedefs and defines
//
#define MOUSE_SCALE 4
//
// externs
//
extern Display *x_disp;
extern int x_screen, x_screen_width, x_screen_height;
extern int x_center_height, x_center_width;
extern int x_std_event_mask;
extern Window x_win, x_root_win;
extern qboolean x_fullscreen;
extern qboolean x_focus;
extern int global_dx, global_dy;
//
// globals
//
cvar_t _windowed_mouse = {"_windowed_mouse","1", true};
int x_root, y_root;
int x_root_old, y_root_old;
//
// locals
//
static int x_mouse_num, x_mouse_denom, x_mouse_thresh;
static qboolean x_grabbed = false;
//
// IN_CenterMouse - center the mouse in the screen
//
void IN_CenterMouse( void )
{
CheckMouseState();
if (!x_grabbed)
return;
XSelectInput( x_disp, x_win, x_std_event_mask & ~PointerMotionMask );
XWarpPointer( x_disp, None, x_root_win, 0, 0, 0, 0, x_center_width,
x_center_height );
XSelectInput( x_disp, x_win, x_std_event_mask );
}
//
// Check to see if we have grabbed the mouse or not and deal with it
// appropriately
//
static void CheckMouseState(void)
{
if (x_focus && _windowed_mouse.value && !x_grabbed) {
x_grabbed = true;
printf("fooling with mouse!\n");
if (XGetPointerControl( x_disp, &x_mouse_num, &x_mouse_denom, &x_mouse_thresh ))
printf( "XGetPointerControl failed!\n" );
//printf( "mouse %d/%d thresh %d\n", x_mouse_num, x_mouse_denom, x_mouse_thresh );
// make input rawer
XAutoRepeatOff(x_disp);
XGrabKeyboard(x_disp, x_win, True, GrabModeAsync, GrabModeAsync, CurrentTime);
XGrabPointer(x_disp, x_win, True,
PointerMotionMask | ButtonPressMask | ButtonReleaseMask,
GrabModeAsync, GrabModeAsync, None, None, CurrentTime);
// if (XChangePointerControl( x_disp, True, True, 1, MOUSE_SCALE, x_mouse_thresh ))
// printf( "XChangePointerControl failed!\n" );
IN_CenterMouse();
// safe initial values
x_root = x_root_old = vid.width >> 1;
y_root = y_root_old = vid.height >> 1;
} else if (x_grabbed && (!_windowed_mouse.value || !x_focus)) {
printf("fooling with mouse!\n");
x_grabbed = false;
// undo mouse warp
if (XChangePointerControl( x_disp, True, True, x_mouse_num, x_mouse_denom, x_mouse_thresh ))
printf( "XChangePointerControl failed!\n" );
XUngrabPointer( x_disp, CurrentTime );
XUngrabKeyboard( x_disp, CurrentTime );
XAutoRepeatOn( x_disp );
}
}
//
// IN_Init - setup mouse input
//
void IN_Init (void)
{
if (!x_disp) Sys_Error( "X display not open!\n" );
Cvar_RegisterVariable (&_windowed_mouse);
// we really really want to clean these up...
atexit( IN_Shutdown );
}
//
// IN_Shutdown - clean up mouse settings (must be done from signal handler too!)
//
void IN_Shutdown (void)
{
if (!x_disp) return;
// undo mouse warp
if (XChangePointerControl( x_disp, True, True, x_mouse_num, x_mouse_denom, x_mouse_thresh ))
printf( "XChangePointerControl failed!\n" );
XUngrabPointer( x_disp, CurrentTime );
XUngrabKeyboard( x_disp, CurrentTime );
XAutoRepeatOn( x_disp );
}
//
// IN_Commands - process buttons
//
void IN_Commands (void)
{
// done in X event handler
}
//
// IN_Move - process mouse moves
//
void
IN_Move (usercmd_t *cmd)
{
static int last_dx, last_dy;
static long long last_movement;
long long now, gethrtime();
int dx, dy;
CheckMouseState();
if (!x_grabbed)
return; // no mouse movement
now = gethrtime();
dx = global_dx;
global_dx = 0;
dy = global_dy;
global_dy = 0;
// printf("GOT: dx %d dy %d\n", dx, dy);
dx *= sensitivity.value;
dy *= sensitivity.value;
//
// implement low pass filter to smooth motion a bit
//
if (now - last_movement > 100000000) {
dx = .6 * dx;
dy = .6 * dy;
}
last_movement = now;
dx = .6 * dx + .4 * last_dx;
dy = .6 * dy + .4 * last_dy;
last_dx = dx;
last_dy = dy;
if (!dx && !dy) {
if (in_mlook.state & 1)
V_StopPitchDrift ();
return;
}
// add mouse X/Y movement to cmd
if ((in_strafe.state & 1) || (lookstrafe.value && (in_mlook.state & 1)))
cmd->sidemove += m_side.value * dx;
else
cl.viewangles[YAW] -= m_yaw.value * dx;
if (in_mlook.state & 1)
V_StopPitchDrift ();
if ((in_mlook.state & 1) && !(in_strafe.state & 1)) {
cl.viewangles[PITCH] += m_pitch.value * dy;
if (cl.viewangles[PITCH] > 80) cl.viewangles[PITCH] = 80;
if (cl.viewangles[PITCH] < -70) cl.viewangles[PITCH] = -70;
}
else {
if ((in_strafe.state & 1) && noclip_anglehack) cmd->upmove -= m_forward.value * dy;
else cmd->forwardmove -= m_forward.value * dy;
}
}