# HG changeset patch
# User Mikhail Kryshen <mikhail@kryshen.net>
# Date 1335279097 -14400
# Node ID 16643a84b9e4976593bc89833ce87bc2e6638856
# Parent  2c2e4c63931118b3aa7d63151faa656d946e604e
Refactored the event dispatcher. Update hover states at every commit.

diff -r 2c2e4c639311 -r 16643a84b9e4 src/net/kryshen/indyvon/core.clj
--- a/src/net/kryshen/indyvon/core.clj	Tue Apr 24 18:49:06 2012 +0400
+++ b/src/net/kryshen/indyvon/core.clj	Tue Apr 24 18:51:37 2012 +0400
@@ -615,7 +615,7 @@
 
 (defn- under-cursor
   "Returns a vector of child nodes under cursor."
-  [x y tree node]
+  [^long x ^long y tree node]
   (some #(if (and (:clip %)
                   (.contains ^Shape (:clip %) x y))
            (conj (vec (under-cursor x y tree %)) %))
@@ -652,26 +652,40 @@
              (recur (rest nodes) false event id)))
          (recur (rest nodes) first-only event id)))))
 
+(defn- process-mouse-position
+  [dispatcher event]
+  (dosync
+   (let [{hovered-ref :hovered
+          last-ref :last-motion
+          tree-ref :tree} dispatcher
+         ^java.awt.event.MouseEvent event (or event @last-ref)]
+     (when event
+       (let [x (.getX event)
+             y (.getY event)
+             old-hovered @hovered-ref
+             hovered (under-cursor x y @tree-ref dispatcher)
+             pred #(= (:handle %1) (:handle %2))
+             exited (remove-all old-hovered hovered pred)
+             entered (remove-all hovered old-hovered pred)
+             moved (remove-all hovered entered pred)]
+         (ref-set hovered-ref hovered)
+         (ref-set last-ref event)
+         [exited entered moved event])))))
+
 (defn- dispatch-mouse-motion
-  "Dispatches mouse motion events."
-  [hovered-ref tree root ^java.awt.event.MouseEvent event]
-  (let [x (.getX event)
-        y (.getY event)
-        [hovered hovered2] (dosync
-                            [@hovered-ref
-                             (ref-set hovered-ref
-                                      (under-cursor x y tree root))])
-        pred #(= (:handle %1) (:handle %2))
-        exited (remove-all hovered hovered2 pred)
-        entered (remove-all hovered2 hovered pred)
-        moved (remove-all hovered2 entered pred)]
+  [dispatcher source-event]
+  (when-let [[exited entered moved event] (process-mouse-position
+                                           dispatcher source-event)]
     (translate-and-dispatch exited false event :mouse-exited)
     (translate-and-dispatch entered false event :mouse-entered)
-    (translate-and-dispatch moved true event :mouse-moved)))
+    (when source-event
+      (translate-and-dispatch moved true event :mouse-moved))))
 
 (defn- dispatch-mouse-button
-  [picked-ref hovered-ref ^java.awt.event.MouseEvent event]
-  (let [id (awt-events (.getID event))
+  [dispatcher ^java.awt.event.MouseEvent event]
+  (let [{picked-ref :picked
+         hovered-ref :hovered} dispatcher
+        id (awt-events (.getID event))
         nodes (case id
                :mouse-pressed
                (dosync
@@ -684,50 +698,55 @@
                @hovered-ref)]
     (translate-and-dispatch nodes true event id)))
 
+(defrecord RootEventDispatcher [tree-r ; register
+                                tree   ; dispatch
+                                hovered
+                                picked
+                                last-motion]
+  EventDispatcher
+  (listen! [dispatcher component]
+    (doto ^Component component
+          (.addMouseListener dispatcher)
+          (.addMouseWheelListener dispatcher)
+          (.addMouseMotionListener dispatcher)))
+  (create-dispatcher [dispatcher handle handlers]
+    (let [node (make-node handle handlers)]
+      (dosync (alter tree-r add-node node))
+      node))
+  (commit [dispatcher]
+    ;; TODO: retain contexts that do not intersect graphics
+    ;; clipping area in tree.
+    (dosync (ref-set tree @tree-r)
+            (ref-set tree-r {}))
+    ;; Send mouse entered and exited events if necessary due to
+    ;; updated layout.
+    (dispatch-mouse-motion dispatcher nil))
+  (handle-picked? [dispatcher handle]
+    (some #(= handle (:handle %)) @picked))
+  (handle-hovered? [dispatcher handle]
+    (some #(= handle (:handle %)) @hovered))
+  MouseListener
+  (mouseEntered [dispatcher event]
+    (dispatch-mouse-motion dispatcher event))
+  (mouseExited [dispatcher event]
+    (dispatch-mouse-motion dispatcher event))
+  (mouseClicked [dispatcher event]
+    (dispatch-mouse-button dispatcher event))
+  (mousePressed [dispatcher event]
+    (dispatch-mouse-button dispatcher event))
+  (mouseReleased [dispatcher event]
+    (dispatch-mouse-button dispatcher event))
+  MouseWheelListener
+  (mouseWheelMoved [dispatcher event]
+    (dispatch-mouse-button dispatcher event))
+  MouseMotionListener
+  (mouseDragged [dispatcher event]
+    (translate-and-dispatch @picked true event))
+  (mouseMoved [dispatcher event]
+    (dispatch-mouse-motion dispatcher event)))
+
 (defn root-event-dispatcher []
-  (let [tree-r (ref {})   ; register
-        tree (ref {})     ; dispatch
-        hovered (ref '())
-        picked (ref '())]
-    (reify
-     EventDispatcher
-     (listen! [this component]
-       (doto ^Component component
-         (.addMouseListener this)
-         (.addMouseWheelListener this)
-         (.addMouseMotionListener this)))
-     (create-dispatcher [this handle handlers]
-       (let [node (make-node handle handlers)]
-         (dosync (alter tree-r add-node node))
-         node))
-     (commit [this]
-       ;; TODO: retain contexts that do not intersect graphics
-       ;; clipping area in tree.
-       (dosync (ref-set tree @tree-r)
-               (ref-set tree-r {})))
-     (handle-picked? [this handle]
-       (some #(= handle (:handle %)) @picked))
-     (handle-hovered? [this handle]
-       (some #(= handle (:handle %)) @hovered))
-     MouseListener
-     (mouseEntered [this event]
-       (dispatch-mouse-motion hovered @tree this event))
-     (mouseExited [this event]
-       (dispatch-mouse-motion hovered @tree this event))
-     (mouseClicked [this event]
-       (dispatch-mouse-button picked hovered event))
-     (mousePressed [this event]
-       (dispatch-mouse-button picked hovered event))
-     (mouseReleased [this event]
-       (dispatch-mouse-button picked hovered event))
-     MouseWheelListener
-     (mouseWheelMoved [this event]
-       (dispatch-mouse-button picked hovered event))
-     MouseMotionListener
-     (mouseDragged [this event]
-       (translate-and-dispatch @picked true event))
-     (mouseMoved [this event]
-       (dispatch-mouse-motion hovered @tree this event)))))
+  (->RootEventDispatcher (ref {}) (ref {}) (ref '()) (ref '()) (ref nil)))
 
 ;;
 ;; Scene