]> git.mxchange.org Git - flightgear.git/blobdiff - src/Input/input.cxx
Modified Files:
[flightgear.git] / src / Input / input.cxx
index 618edc75e6d5b73debd1a1a84ebfb8e0797e56cc..fdb13de5e5e828b81051838bd9bfdaf2bed09e9a 100644 (file)
@@ -16,7 +16,7 @@
 //
 // 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., 675 Mass Ave, Cambridge, MA 02139, USA.
+// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301, USA.
 //
 // $Id$
 
@@ -41,7 +41,9 @@
 
 #include <simgear/constants.h>
 #include <simgear/debug/logstream.hxx>
+#include <simgear/math/SGMath.hxx>
 #include <simgear/props/props.hxx>
+#include <simgear/scene/util/SGSceneUserData.hxx>
 
 #include <Aircraft/aircraft.hxx>
 #include <Autopilot/xmlauto.hxx>
 
 #include "input.hxx"
 
+#include <Scenery/scenery.hxx>
+#include <Main/renderer.hxx>
+
 SG_USING_STD(ifstream);
 SG_USING_STD(string);
 SG_USING_STD(vector);
 
-void mouseClickHandler(int button, int updown, int x, int y);
+void mouseClickHandler(int button, int updown, int x, int y, bool mainWindow, const osgGA::GUIEventAdapter*);
 void mouseMotionHandler(int x, int y);
 void keyHandler(int key, int keymod, int mousex, int mousey);
 
@@ -72,88 +77,29 @@ void keyHandler(int key, int keymod, int mousex, int mousey);
 
 static FGInput * default_input = 0;
 
-
 \f
 ////////////////////////////////////////////////////////////////////////
-// Implementation of FGBinding.
+// Local functions.
 ////////////////////////////////////////////////////////////////////////
 
-FGBinding::FGBinding ()
-  : _command(0),
-    _arg(new SGPropertyNode),
-    _setting(0)
+static bool
+getModShift ()
 {
+  return bool(fgGetKeyModifiers() & KEYMOD_SHIFT);
 }
 
-FGBinding::FGBinding (const SGPropertyNode * node)
-  : _command(0),
-    _arg(0),
-    _setting(0)
+static bool
+getModCtrl ()
 {
-  read(node);
+  return bool(fgGetKeyModifiers() & KEYMOD_CTRL);
 }
 
-FGBinding::~FGBinding ()
+static bool
+getModAlt ()
 {
-  _arg->getParent()->removeChild(_arg->getName(), _arg->getIndex());
-}
-
-void
-FGBinding::read (const SGPropertyNode * node)
-{
-  const SGPropertyNode * conditionNode = node->getChild("condition");
-  if (conditionNode != 0)
-    setCondition(sgReadCondition(globals->get_props(), conditionNode));
-
-  _command_name = node->getStringValue("command", "");
-  if (_command_name.empty()) {
-    SG_LOG(SG_INPUT, SG_WARN, "No command supplied for binding.");
-    _command = 0;
-    return;
-  }
-
-  _arg = (SGPropertyNode *)node;
-  _setting = 0;
+  return bool(fgGetKeyModifiers() & KEYMOD_ALT);
 }
 
-void
-FGBinding::fire () const
-{
-  if (test()) {
-    if (_command == 0)
-      _command = globals->get_commands()->getCommand(_command_name);
-    if (_command == 0) {
-      SG_LOG(SG_INPUT, SG_WARN, "No command attached to binding");
-    } else if (!(*_command)(_arg)) {
-      SG_LOG(SG_INPUT, SG_ALERT, "Failed to execute command "
-             << _command_name);
-    }
-  }
-}
-
-void
-FGBinding::fire (double offset, double max) const
-{
-  if (test()) {
-    _arg->setDoubleValue("offset", offset/max);
-    fire();
-  }
-}
-
-void
-FGBinding::fire (double setting) const
-{
-  if (test()) {
-                                // A value is automatically added to
-                                // the args
-    if (_setting == 0)          // save the setting node for efficiency
-      _setting = _arg->getChild("setting", 0, true);
-    _setting->setDoubleValue(setting);
-    fire();
-  }
-}
-
-
 \f
 ////////////////////////////////////////////////////////////////////////
 // Implementation of FGInput.
@@ -196,6 +142,22 @@ FGInput::postinit ()
   _postinit_joystick();
 }
 
+void
+FGInput::bind ()
+{
+  fgTie("/devices/status/keyboard/shift", getModShift);
+  fgTie("/devices/status/keyboard/ctrl", getModCtrl);
+  fgTie("/devices/status/keyboard/alt", getModAlt);
+}
+
+void
+FGInput::unbind ()
+{
+  fgUntie("/devices/status/keyboard/shift");
+  fgUntie("/devices/status/keyboard/ctrl");
+  fgUntie("/devices/status/keyboard/alt");
+}
+
 void 
 FGInput::update (double dt)
 {
@@ -249,7 +211,7 @@ FGInput::doKey (int k, int modifiers, int x, int y)
     if (!b.last_state || b.is_repeatable) {
       const binding_list_t &bindings = _find_key_bindings(k, modifiers);
 
-      for (int i = 0; i < bindings.size(); i++)
+      for (unsigned int i = 0; i < bindings.size(); i++)
         bindings[i]->fire();
       b.last_state = 1;
     }
@@ -260,15 +222,32 @@ FGInput::doKey (int k, int modifiers, int x, int y)
            << " with modifiers " << modifiers);
     if (b.last_state) {
       const binding_list_t &bindings = _find_key_bindings(k, modifiers);
-      for (int i = 0; i < bindings.size(); i++)
+      for (unsigned int i = 0; i < bindings.size(); i++)
         bindings[i]->fire();
       b.last_state = 0;
+    } else {
+      if (k >= 1 && k <= 26) {
+        if (_key_bindings[k + '@'].last_state)
+          doKey(k + '@', KEYMOD_RELEASED, x, y);
+        if (_key_bindings[k + '`'].last_state)
+          doKey(k + '`', KEYMOD_RELEASED, x, y);
+      } else if (k >= 'A' && k <= 'Z') {
+        if (_key_bindings[k - '@'].last_state)
+          doKey(k - '@', KEYMOD_RELEASED, x, y);
+        if (_key_bindings[tolower(k)].last_state)
+          doKey(tolower(k), KEYMOD_RELEASED, x, y);
+      } else if (k >= 'a' && k <= 'z') {
+        if (_key_bindings[k - '`'].last_state)
+          doKey(k - '`', KEYMOD_RELEASED, x, y);
+        if (_key_bindings[toupper(k)].last_state)
+          doKey(toupper(k), KEYMOD_RELEASED, x, y);
+      }
     }
   }
 }
 
 void
-FGInput::doMouseClick (int b, int updown, int x, int y)
+FGInput::doMouseClick (int b, int updown, int x, int y, bool mainWindow, const osgGA::GUIEventAdapter* ea)
 {
   int modifiers = fgGetKeyModifiers();
 
@@ -282,18 +261,52 @@ FGInput::doMouseClick (int b, int updown, int x, int y)
                                 // Pass on to PUI and the panel if
                                 // requested, and return if one of
                                 // them consumes the event.
+
+  if (updown != MOUSE_BUTTON_DOWN) {
+    // Execute the mouse up event in any case, may be we should
+    // stop processing here?
+    while (!_activePickCallbacks[b].empty()) {
+      _activePickCallbacks[b].front()->buttonReleased();
+      _activePickCallbacks[b].pop_front();
+    }
+  }
+
   if (mode.pass_through) {
-    if (puMouse(b, updown, x, y))
+    // The pu stuff seems to need that. May be it does opengl picking ...
+    fgMakeCurrent();
+    if (0 <= x && 0 <= y && puMouse(b, updown, x, y))
       return;
-    else if ((globals->get_current_panel() != 0) &&
+    else if (0 <= x && 0 <= y && (globals->get_current_panel() != 0) &&
              globals->get_current_panel()->getVisibility() &&
              globals->get_current_panel()->doMouseAction(b, updown, x, y))
       return;
-    else if (fgHandle3DPanelMouseEvent(b, updown, x, y))
+    else if (0 <= x && 0 <= y && fgHandle3DPanelMouseEvent(b, updown, x, y))
       return;
+    else {
+      // pui didn't want the click event so compute a
+      // scenegraph intersection point corresponding to the mouse click
+      if (updown == MOUSE_BUTTON_DOWN) {
+
+        // Get the list of hit callbacks. Take the first callback that
+        // accepts the mouse button press and ignore the rest of them
+        // That is they get sorted by distance and by scenegraph depth.
+        // The nearest one is the first one and the deepest
+        // (the most specialized one in the scenegraph) is the first.
+        std::vector<SGSceneryPick> pickList;
+        if (FGRenderer::pick(x, y, pickList, ea)) {
+          std::vector<SGSceneryPick>::const_iterator i;
+          for (i = pickList.begin(); i != pickList.end(); ++i) {
+            if (i->callback->buttonPressed(b, i->info)) {
+              _activePickCallbacks[b].push_back(i->callback);
+              return;
+            }
+          }
+        }
+      }
+    }
   }
 
-                                // OK, PUI and the panel didn't want the click
+  // OK, PUI and the panel didn't want the click
   if (b >= MAX_MOUSE_BUTTONS) {
     SG_LOG(SG_INPUT, SG_ALERT, "Mouse button " << b
            << " where only " << MAX_MOUSE_BUTTONS << " expected");
@@ -347,27 +360,25 @@ FGInput::doMouseMotion (int x, int y)
                                 // Constrain the mouse if requested
   if (mode.constrained) {
     bool need_warp = false;
-    if (x <= 0) {
-      x = xsize - 2;
-      need_warp = true;
-    } else if (x >= (xsize-1)) {
-      x = 1;
+    if (x <= (xsize * .25) || x >= (xsize * .75)) {
+      x = int(xsize * .5);
       need_warp = true;
     }
 
-    if (y <= 0) {
-      y = ysize - 2;
-      need_warp = true;
-    } else if (y >= (ysize-1)) {
-      y = 1;
+    if (y <= (ysize * .25) || y >= (ysize * .75)) {
+      y = int(ysize * .5);
       need_warp = true;
     }
 
     if (need_warp)
       fgWarpMouse(x, y);
   }
-  m.x = x;
-  m.y = y;
+
+  if (m.x != x)
+      fgSetInt("/devices/status/mice/mouse/x", m.x = x);
+
+  if (m.y != y)
+      fgSetInt("/devices/status/mice/mouse/y", m.y = y);
 }
 
 void
@@ -486,8 +497,7 @@ FGInput::_init_joystick ()
   }
 
   // get rid of unused config nodes
-  for (unsigned int m = 0; m < js_named.size(); m++)
-    js_nodes->removeChild("js-named", js_named[m]->getIndex(), false);
+  js_nodes->removeChildren("js-named", false);
 }
 
 
@@ -852,6 +862,16 @@ FGInput::_update_mouse ( double dt )
       m.save_x = m.x;
       m.save_y = m.y;
   }
+
+  // handle repeatable mouse press events
+  std::map<int, std::list<SGSharedPtr<SGPickCallback> > >::iterator mi;
+  for (mi = _activePickCallbacks.begin();
+       mi != _activePickCallbacks.end(); ++mi) {
+    std::list<SGSharedPtr<SGPickCallback> >::iterator li;
+    for (li = mi->second.begin(); li != mi->second.end(); ++li) {
+      (*li)->update(dt);
+    }
+  }
 }
 
 void
@@ -862,8 +882,9 @@ FGInput::_update_button (button &b, int modifiers, bool pressed,
                                 // The press event may be repeated.
     if (!b.last_state || b.is_repeatable) {
       SG_LOG( SG_INPUT, SG_DEBUG, "Button has been pressed" );
-      for (unsigned int k = 0; k < b.bindings[modifiers].size(); k++)
+      for (unsigned int k = 0; k < b.bindings[modifiers].size(); k++) {
         b.bindings[modifiers][k]->fire(x, y);
+      }
     }
   } else {
                                 // The release event is never repeated.
@@ -891,7 +912,7 @@ FGInput::_read_bindings (const SGPropertyNode * node,
 
     if (!strcmp(cmd, "nasal") && _module[0])
       bindings[i]->setStringValue("module", _module);
-    binding_list[modifiers].push_back(new FGBinding(bindings[i]));
+    binding_list[modifiers].push_back(new SGBinding(bindings[i], globals->get_props()));
   }
 
                                 // Read nested bindings for modifiers
@@ -913,7 +934,7 @@ FGInput::_read_bindings (const SGPropertyNode * node,
 }
 
 
-const vector<FGBinding *> &
+const FGInput::binding_list_t&
 FGInput::_find_key_bindings (unsigned int k, int modifiers)
 {
   unsigned char kc = (unsigned char)k;
@@ -957,7 +978,9 @@ FGInput::_find_key_bindings (unsigned int k, int modifiers)
 
 FGInput::button::button ()
   : is_repeatable(false),
-    last_state(-1)
+    interval_sec(0),
+    last_dt(0),
+    last_state(0)
 {
 }
 
@@ -979,7 +1002,9 @@ FGInput::axis::axis ()
   : last_value(9999999),
     tolerance(0.002),
     low_threshold(-0.9),
-    high_threshold(0.9)
+    high_threshold(0.9),
+    interval_sec(0),
+    last_dt(0)
 {
 }
 
@@ -997,6 +1022,12 @@ FGInput::axis::~axis ()
 ////////////////////////////////////////////////////////////////////////
 
 FGInput::joystick::joystick ()
+  : jsnum(0),
+    js(0),
+    naxes(0),
+    nbuttons(0),
+    axes(0),
+    buttons(0)
 {
 }
 
@@ -1068,10 +1099,10 @@ void keyHandler(int key, int keymod, int mousex, int mousey)
         default_input->doKey(key, keymod, mousex, mousey);
 }
 
-void mouseClickHandler(int button, int updown, int x, int y)
+void mouseClickHandler(int button, int updown, int x, int y, bool mainWindow, const osgGA::GUIEventAdapter* ea)
 {
     if(default_input)
-        default_input->doMouseClick(button, updown, x, y);
+      default_input->doMouseClick(button, updown, x, y, mainWindow, ea);
 }
 
 void mouseMotionHandler(int x, int y)