| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608 | //-----------------------------------------------------------------------------// Copyright (c) 2012 GarageGames, LLC//// Permission is hereby granted, free of charge, to any person obtaining a copy// of this software and associated documentation files (the "Software"), to// deal in the Software without restriction, including without limitation the// rights to use, copy, modify, merge, publish, distribute, sublicense, and/or// sell copies of the Software, and to permit persons to whom the Software is// furnished to do so, subject to the following conditions://// The above copyright notice and this permission notice shall be included in// all copies or substantial portions of the Software.//// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING// FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS// IN THE SOFTWARE.//-----------------------------------------------------------------------------#ifndef TORQUE_SDL#include "platformX86UNIX/platformX86UNIX.h"#include "platform/platformInput.h"#include "platform/platformVideo.h"#include "platform/event.h"#include "platform/gameInterface.h"#include "console/console.h"#include "platformX86UNIX/x86UNIXState.h"#include "platformX86UNIX/x86UNIXInputManager.h"#include <X11/Xlib.h>#include <X11/Xatom.h>#include <X11/keysym.h>#include <SDL/SDL.h>#ifdef LOG_INPUT#include <time.h>#include <stdarg.h>#include <fcntl.h>#include <platformX86UNIX/x86UNIXUtils.h>extern int x86UNIXOpen(const char *path, int oflag);extern int x86UNIXClose(int fd);extern ssize_t x86UNIXWrite(int fd, const void *buf, size_t nbytes);#endifclass XClipboard{   private:      Atom mClipboardProperty;      Atom mClipboard;      Atom mPrimary;      bool mInitialized;      U8 *mXData;      char *mTData;      S32 mTDataSize;      void init();      void freeXData();      void freeTData();      void checkTDataSize(S32 requestedSize);   public:      XClipboard();      ~XClipboard();      bool setClipboard(const char *text);      const char* getClipboard();      void handleSelectionRequest(XSelectionRequestEvent& request);};// Static class variables:InputManager*  Input::smManager;// smActive is not maintained under unix.  Use Input::isActive()// insteadbool           Input::smActive = false;// unix platform stateextern x86UNIXPlatformState * x86UNIXState;extern AsciiData AsciiTable[NUM_KEYS];static XClipboard xclipboard;#ifdef LOG_INPUTS32 gInputLog = -1;#endif //------------------------------------------------------------------------------void Input::init(){   Con::printf( "Input Init:" );   destroy();#ifdef LOG_INPUT   struct tm* newTime;   time_t aclock;   time( &aclock );   newTime = localtime( &aclock );   asctime( newTime );   gInputLog = x86UNIXOpen("input.log", O_WRONLY | O_CREAT);   log("Input log opened at %s\n", asctime( newTime ) );   log("Operating System:\n" );   log("  %s", UUtils->getOSName());   log("\n");#endif   smActive = false;   smManager = NULL;   UInputManager *uInputManager = new UInputManager;   if ( !uInputManager->enable() )   {      Con::errorf( "   Failed to enable Input Manager." );      delete uInputManager;      return;   }   uInputManager->init();   smManager = uInputManager;   Con::printf("   Input initialized");   Con::printf(" ");}//------------------------------------------------------------------------------ConsoleFunction( isJoystickDetected, bool, 1, 1, "isJoystickDetected()" ){   argc; argv;   UInputManager* manager = dynamic_cast<UInputManager*>(Input::getManager());   if (manager)      return manager->joystickDetected();   else      return false;}//------------------------------------------------------------------------------ConsoleFunction( getJoystickAxes, const char*, 2, 2, "getJoystickAxes( instance )" ){   argc; argv;   UInputManager* manager = dynamic_cast<UInputManager*>(Input::getManager());   if (manager)      return manager->getJoystickAxesString(dAtoi(argv[1]));   else      return "";}//------------------------------------------------------------------------------U16 Input::getKeyCode( U16 asciiCode ){   U16 keyCode = 0;   U16 i;      // This is done three times so the lowerkey will always   // be found first. Some foreign keyboards have duplicate   // chars on some keys.   for ( i = KEY_FIRST; i < NUM_KEYS && !keyCode; i++ )   {      if ( AsciiTable[i].lower.ascii == asciiCode )      {         keyCode = i;         break;      };   }   for ( i = KEY_FIRST; i < NUM_KEYS && !keyCode; i++ )   {      if ( AsciiTable[i].upper.ascii == asciiCode )      {         keyCode = i;         break;      };   }   for ( i = KEY_FIRST; i < NUM_KEYS && !keyCode; i++ )   {      if ( AsciiTable[i].goofy.ascii == asciiCode )      {         keyCode = i;         break;      };   }   return( keyCode );}//-----------------------------------------------------------------------------//// This function gets the standard ASCII code corresponding to our key code// and the existing modifier key state.////-----------------------------------------------------------------------------U16 Input::getAscii( U16 keyCode, KEY_STATE keyState ){   if ( keyCode >= NUM_KEYS )      return 0;   switch ( keyState )   {      case STATE_LOWER:         return AsciiTable[keyCode].lower.ascii;      case STATE_UPPER:         return AsciiTable[keyCode].upper.ascii;      case STATE_GOOFY:         return AsciiTable[keyCode].goofy.ascii;      default:         return(0);               }}//------------------------------------------------------------------------------void Input::destroy(){   #ifdef LOG_INPUT   if ( gInputLog != -1 )   {      log( "*** CLOSING LOG ***\n" );      x86UNIXClose(gInputLog);      gInputLog = -1;   }#endif   if ( smManager && smManager->isEnabled() )   {      smManager->disable();      delete smManager;      smManager = NULL;   }}//------------------------------------------------------------------------------bool Input::enable(){      if ( smManager && !smManager->isEnabled() )      return( smManager->enable() );      return( false );}//------------------------------------------------------------------------------void Input::disable(){   if ( smManager && smManager->isEnabled() )      smManager->disable();}//------------------------------------------------------------------------------void Input::activate(){   if ( smManager && smManager->isEnabled() && !isActive())   {#ifdef LOG_INPUT      Input::log( "Activating Input...\n" );#endif      UInputManager* uInputManager = dynamic_cast<UInputManager*>( smManager );      if ( uInputManager )         uInputManager->activate();   }}//------------------------------------------------------------------------------void Input::deactivate(){   if ( smManager && smManager->isEnabled() && isActive() )   {#ifdef LOG_INPUT      Input::log( "Deactivating Input...\n" );#endif      UInputManager* uInputManager = dynamic_cast<UInputManager*>( smManager );      if ( uInputManager )         uInputManager->deactivate();   }}//------------------------------------------------------------------------------void Input::reactivate(){   Input::deactivate();   Input::activate();}//------------------------------------------------------------------------------bool Input::isEnabled(){   if ( smManager )      return smManager->isEnabled();   return false;}//------------------------------------------------------------------------------bool Input::isActive(){   UInputManager* uInputManager = dynamic_cast<UInputManager*>( smManager );   if ( uInputManager )      return uInputManager->isActive();   return false;}//------------------------------------------------------------------------------void Input::process(){   if ( smManager )      smManager->process();}//------------------------------------------------------------------------------InputManager* Input::getManager(){   return smManager;}#ifdef LOG_INPUT//------------------------------------------------------------------------------void Input::log( const char* format, ... ){   if ( gInputLog == -1)      return;      va_list argptr;   va_start( argptr, format );   const int BufSize = 4096;   char buffer[BufSize];   dVsprintf( buffer, BufSize, format, argptr );   x86UNIXWrite(gInputLog, buffer, dStrlen( buffer ));   va_end( argptr );}ConsoleFunction( inputLog, void, 2, 2, "inputLog( string )" ){   argc;   Input::log( "%s\n", (const char*)argv[1] );}#endif // LOG_INPUT//------------------------------------------------------------------------------void NotifySelectionEvent(XEvent& event){   // somebody sent us a select event   if (event.type == SelectionRequest)      xclipboard.handleSelectionRequest(event.xselectionrequest);}//------------------------------------------------------------------------------const char* Platform::getClipboard(){   return xclipboard.getClipboard();}//------------------------------------------------------------------------------bool Platform::setClipboard(const char *text){   return xclipboard.setClipboard(text);}//-----------------------------------------------------------------------------// XClipboard membersXClipboard::XClipboard(){   mInitialized = false;   mXData = 0;   mTData = 0;   mTDataSize = 0;}//------------------------------------------------------------------------------XClipboard::~XClipboard(){   freeXData();   freeTData();}//------------------------------------------------------------------------------void XClipboard::init(){   DisplayPtrManager xdisplay;   Display* display = xdisplay.getDisplayPointer();   mClipboardProperty = XInternAtom(display,       "TORQUE_CLIPBOARD_ATOM", False);   mClipboard = XInternAtom(display, "CLIPBOARD",       False);   mPrimary = XA_PRIMARY; //XInternAtom(display, "PRIMARY", False);   mXData = NULL;   mTData = NULL;   mTDataSize = 0;   mInitialized = true;}//------------------------------------------------------------------------------inline void XClipboard::freeXData(){   if (mXData != NULL)   {      XFree(mXData);      mXData = NULL;   }}//------------------------------------------------------------------------------inline void XClipboard::freeTData(){   if (mTData != NULL)   {      dRealFree(mTData);      mTData = NULL;      mTDataSize = 0;   }}//// JMQ: As you might expect, X clipboard usage is bizarre.  I // found this document to be useful.//// http://www.freedesktop.org/standards/clipboards.txt //// JMQ: later note: programming the X clipboard is not just// bizarre, it SUCKS.  No wonder so many apps have// clipboard problems.////------------------------------------------------------------------------------const char* XClipboard::getClipboard(){   DisplayPtrManager xdisplay;   Display* display = xdisplay.getDisplayPointer();   if (!mInitialized)      init();   // find the owner of the clipboard   Atom targetSelection = mClipboard;   Window clipOwner = XGetSelectionOwner(display,       targetSelection);   if (clipOwner == None)   {      // It seems like KDE/QT reads the clipboard but doesn't set it.      // This is a bug, that supposedly will be fixed in QT3.      // I tried working around this by using      // PRIMARY instead of CLIPBOARD, but this has some nonintuitive      // side effects.  So, no pasting from KDE apps for now.      //targetSelection = mPrimary;      //clipOwner = XGetSelectionOwner(display, targetSelection);   }   if (clipOwner == None)      // oh well      return "";   // request that the owner convert the selection to a string   XConvertSelection(display, targetSelection,       XA_STRING, mClipboardProperty, x86UNIXState->getWindow(), CurrentTime);   // flush the output buffer to make sure the selection request event gets    // sent now   XFlush(display);   XEvent xevent;   // if our window is the current owner, (e.g. copy from one part of   // torque and paste to another), then we just sent an event to our   // window that won't get processed until we get back to the event   // loop in x86Unixwindow.  So look for selection request events in   // the event queue immediately and handle them.   while (XCheckTypedWindowEvent(display,              x86UNIXState->getWindow(), SelectionRequest, &xevent))      handleSelectionRequest(xevent.xselectionrequest);     // poll for the SelectionNotify event for 5 seconds.  in most cases    // we should get the event very quickly   U32 startTime = Platform::getRealMilliseconds();   bool timeOut = false;   while (!XCheckTypedWindowEvent(display,              x86UNIXState->getWindow(), SelectionNotify, &xevent) &&      !timeOut)   {      // we'll be spinning here, but who cares      if ((Platform::getRealMilliseconds() - startTime) > 5000)         timeOut = true;   }   if (timeOut)   {      Con::warnf(ConsoleLogEntry::General,          "XClipboard: waited too long for owner to convert selection");      return "";   }   if (xevent.xselection.property == None)      return "";   // free the X data from a previous get   freeXData();   // grab the string data from the property    Atom actual_type;   int actual_format;   unsigned long bytes_after;   unsigned long nitems;   // query the property length the 250000 is "the length in 32-bit   // multiples of the data to be retrieved".  so we support up to a   // million bytes of returned data.   int numToRetrieve = 250000;   int status = XGetWindowProperty(display,       x86UNIXState->getWindow(),      mClipboardProperty, 0, numToRetrieve, True, XA_STRING,       &actual_type, &actual_format, &nitems, &bytes_after, &mXData);   // we should have returned OK, with string type, 8bit data,   // and > 0 items.   if ((status != Success) || (actual_type != XA_STRING) ||       (actual_format != 8) || (nitems == 0))      return "";   // if there is data left in the clipboard, warn about it   if (bytes_after > 0)      Con::warnf(ConsoleLogEntry::General,          "XClipboard: some data was not retrieved");   return reinterpret_cast<const char *>(mXData);}//------------------------------------------------------------------------------void XClipboard::checkTDataSize(S32 requestedSize){   if (mTDataSize < requestedSize)   {      freeTData();      mTData = static_cast<char*>(dRealMalloc(sizeof(char) * requestedSize));      AssertFatal(mTData, "unable to allocate clipboard buffer data!");      mTDataSize = requestedSize;   }}//------------------------------------------------------------------------------bool XClipboard::setClipboard(const char *text){   DisplayPtrManager xdisplay;   Display* display = xdisplay.getDisplayPointer();   if (!mInitialized)      init();   // get the length of the text   S32 len = dStrlen(text) + 1;      // reallocate the storage buffer if necessary   checkTDataSize(len);   // copy the data into the storage buffer   dStrcpy(mTData, text);   // tell X that we own the clipboard.  (we'll get events   // if an app tries to paste)   XSetSelectionOwner(display, mClipboard,       x86UNIXState->getWindow(), CurrentTime);   return true;}//------------------------------------------------------------------------------void XClipboard::handleSelectionRequest(XSelectionRequestEvent& request){   DisplayPtrManager xdisplay;   Display* display = xdisplay.getDisplayPointer();   // init our response   XSelectionEvent notify;   notify.type = SelectionNotify;   notify.display = display;   notify.requestor = request.requestor;   notify.selection = request.selection;   notify.target = XA_STRING;   notify.property = None;   notify.time = CurrentTime;   // make sure the owner is our window, and that the   // requestor wants the clipboard   if (request.owner == x86UNIXState->getWindow() &&       request.selection == mClipboard)   {      notify.property = request.property;      // check to see if they did not set the property      if (notify.property == None)         notify.property = mClipboardProperty;      // get the length of the data in the clipboard      S32 length = dStrlen(mTData);      // set the property on the requestor window      XChangeProperty(display, request.requestor,          notify.property, XA_STRING,         8, PropModeReplace, reinterpret_cast<const unsigned char*>(mTData),          length);   }   XSendEvent(display, notify.requestor, False, 0,       reinterpret_cast<XEvent*>(¬ify));   // flush the output buffer to send the event now   XFlush(display);}#endif
 |