;; This Source Code Form is subject to the terms of the Mozilla Public ;; License, v. 2.0. If a copy of the MPL was not distributed with this ;; file, You can obtain one at http://mozilla.org/MPL/2.0/. ;; ;; This Source Code Form is "Incompatible With Secondary Licenses", as ;; defined by the Mozilla Public License, v. 2.0. ;; ;; Copyright (c) 2020 UXBOX Labs S.L (ns app.main.ui.hooks "A collection of general purpose react hooks." (:require ["mousetrap" :as mousetrap] [app.common.spec :as us] [app.main.data.shortcuts :refer [bind-shortcuts]] [app.util.dom :as dom] [app.util.dom.dnd :as dnd] [app.util.logging :as log] [app.util.timers :as ts] [app.util.transit :as t] [app.util.webapi :as wapi] [beicon.core :as rx] [cljs.spec.alpha :as s] [goog.events :as events] [rumext.alpha :as mf]) (:import goog.events.EventType)) (log/set-level! :warn) (defn use-rxsub [ob] (let [[state reset-state!] (mf/useState @ob)] (mf/useEffect (fn [] (let [sub (rx/subscribe ob #(reset-state! %))] #(rx/cancel! sub))) #js [ob]) state)) (defn use-shortcuts [shortcuts] (mf/use-effect (fn [] (bind-shortcuts shortcuts mousetrap/bind (fn [key cb] (fn [event] (log/debug :msg (str "Shortcut" key)) (.preventDefault event) (cb event)))) (fn [] (mousetrap/reset)))) nil) (defn invisible-image [] (let [img (js/Image.) imd ""] (set! (.-src img) imd) img)) (defn- set-timer [state ms func] (assoc state :timer (ts/schedule ms func))) (defn- cancel-timer [state] (let [timer (:timer state)] (if timer (do (rx/dispose! timer) (dissoc state :timer)) state))) (def sortable-ctx (mf/create-context nil)) (mf/defc sortable-container [{:keys [children] :as props}] (let [global-drag-end (mf/use-memo #(rx/subject))] [:& (mf/provider sortable-ctx) {:value global-drag-end} children])) ;; The dnd API is problematic for nested elements, such a sortable items tree. ;; The approach used here to solve bad situations is: ;; - Capture all events in the leaf draggable elements, and stop propagation. ;; - Ignore events originated in non-draggable children. ;; - At drag operation end, all elements that have received some enter/over ;; event and have not received the corresponding leave event, are notified ;; so they can clean up. This can be occur, for example, if ;; * some leave events are throttled out because of a slow computer ;; * some corner cases of mouse entering a container element, and then ;; moving into a contained element. This is anyway mitigated by not ;; stopping propagation of leave event. ;; ;; Do not remove commented out lines, they are useful to debug events when ;; things go weird. (defn use-sortable [& {:keys [data-type data on-drop on-drag on-hold detect-center?] :as opts}] (let [ref (mf/use-ref) state (mf/use-state {:over nil :timer nil :subscr nil}) global-drag-end (mf/use-ctx sortable-ctx) cleanup (fn [] ;; (js/console.log "cleanup" (:name data)) (when-let [subscr (:subscr @state)] ;; (js/console.log "unsubscribing" (:name data)) (rx/unsub! (:subscr @state))) (swap! state (fn [state] (-> state (cancel-timer) (dissoc :over :subscr))))) subscribe-to-drag-end (fn [] (when (nil? (:subscr @state)) ;; (js/console.log "subscribing" (:name data)) (swap! state #(assoc % :subscr (rx/sub! global-drag-end cleanup))))) on-drag-start (fn [event] (dom/stop-propagation event) ;; (dnd/trace event data "drag-start") (dnd/set-data! event data-type data) (dnd/set-drag-image! event (invisible-image)) (dnd/set-allowed-effect! event "move") (when (fn? on-drag) (on-drag data))) on-drag-enter (fn [event] (dom/prevent-default event) ;; prevent default to allow drag enter (when-not (dnd/from-child? event) (dom/stop-propagation event) (subscribe-to-drag-end) ;; (dnd/trace event data "drag-enter") (when (fn? on-hold) (swap! state (fn [state] (-> state (cancel-timer) (set-timer 1000 on-hold))))))) on-drag-over (fn [event] (when (dnd/has-type? event data-type) (dom/prevent-default event) ;; prevent default to allow drag over (when-not (dnd/from-child? event) (dom/stop-propagation event) (subscribe-to-drag-end) ;; (dnd/trace event data "drag-over") (let [side (dnd/drop-side event detect-center?)] (swap! state assoc :over side))))) on-drag-leave (fn [event] (when-not (dnd/from-child? event) ;; (dnd/trace event data "drag-leave") (cleanup))) on-drop' (fn [event] (dom/stop-propagation event) ;; (dnd/trace event data "drop") (let [side (dnd/drop-side event detect-center?) drop-data (dnd/get-data event data-type)] (cleanup) (rx/push! global-drag-end nil) (when (fn? on-drop) (on-drop side drop-data)))) on-drag-end (fn [event] (dom/stop-propagation event) ;; (dnd/trace event data "drag-end") (rx/push! global-drag-end nil) (cleanup)) on-mount (fn [] (let [dom (mf/ref-val ref)] (.setAttribute dom "draggable" true) ;; Register all events in the (default) bubble mode, so that they ;; are captured by the most leaf item. The handler will stop ;; propagation, so they will not go up in the containment tree. (.addEventListener dom "dragstart" on-drag-start false) (.addEventListener dom "dragenter" on-drag-enter false) (.addEventListener dom "dragover" on-drag-over false) (.addEventListener dom "dragleave" on-drag-leave false) (.addEventListener dom "drop" on-drop' false) (.addEventListener dom "dragend" on-drag-end false) #(do (.removeEventListener dom "dragstart" on-drag-start) (.removeEventListener dom "dragenter" on-drag-enter) (.removeEventListener dom "dragover" on-drag-over) (.removeEventListener dom "dragleave" on-drag-leave) (.removeEventListener dom "drop" on-drop') (.removeEventListener dom "dragend" on-drag-end))))] (mf/use-effect (mf/deps data on-drop) on-mount) [(deref state) ref])) (defn use-stream "Wraps the subscription to a strem into a `use-effect` call" ([stream on-subscribe] (use-stream stream (mf/deps) on-subscribe)) ([stream deps on-subscribe] (mf/use-effect deps (fn [] (let [sub (->> stream (rx/subs on-subscribe))] #(rx/dispose! sub)))))) ;; https://reactjs.org/docs/hooks-faq.html#how-to-get-the-previous-props-or-state (defn use-previous [value] (let [ref (mf/use-ref value)] (mf/use-effect (mf/deps value) (fn [] (mf/set-ref-val! ref value))) (mf/ref-val ref)))