/* 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; } }