Add dynamic subscriptions to re-frame

Dynamic subscriptions allow the user to specify subscriptions that
depend on Ratoms/Reactions and will be rerun when they change. Users
will subscribe with v and a vector of dynamic values. The dynamic values
are dereffed and passed to the handler-fn. Dynamic subscriptions need to
pass a fn which takes app-db, v, and the dereffed dynamic values.

Every time a dynamic value changes, handler-fn will be rerun. This is in
contrast to standard subscriptions where handler-fn will only be run
once, although the reaction that it produces will change over time.

A concrete example of the need for this is:
1. You want to subscribe to a query on a remote server which will return
a Reaction which changes in response to server changes.
2. You want this subscription to be able to be rerun when you change one
of the query parameters.

In the current system, all views need to be aware of the possibility of
changing parameters and provide them in their subscriptions.

Example usage code:

(register-sub
  :todo-dynamic
  (fn todo-dynamic [_ _ [active-list]]
    (let [q (q/get-query active-list)]
      q)))

(register-sub
  :todos
  (fn todos [db _]
    (let [active-list (subscribe [:active-list])
          todos       (subscribe [:todo-dynamic] [active-list])]
      (make-reaction (fn todo-vals [] (update @todos :result #(vals (:list %))))))))
This commit is contained in:
Daniel Compton 2015-08-03 15:25:15 +12:00
parent bbc6bc4abe
commit 7e492930e3
1 changed files with 19 additions and 8 deletions

View File

@ -1,6 +1,6 @@
(ns re-frame.subs (ns re-frame.subs
(:require (:require
[reagent.ratom :refer [make-reaction]] [reagent.ratom :refer [make-reaction] :refer-macros [reaction run!]]
[re-frame.db :refer [app-db]] [re-frame.db :refer [app-db]]
[re-frame.utils :refer [first-in-vector warn error]])) [re-frame.utils :refer [first-in-vector warn error]]))
@ -25,10 +25,21 @@
(defn subscribe (defn subscribe
"Returns a reagent/reaction which observes a part of app-db" "Returns a reagent/reaction which observes a part of app-db"
[v] ([v]
(let [key-v (first-in-vector v) (let [key-v (first-in-vector v)
handler-fn (get @key->fn key-v)] handler-fn (get @key->fn key-v)]
(if (nil? handler-fn) (if (nil? handler-fn)
(error "re-frame: no subscription handler registered for: \"" key-v "\". Returning a nil subscription.") (error "re-frame: no subscription handler registered for: \"" key-v "\". Returning a nil subscription.")
(handler-fn app-db v)))) (handler-fn app-db v))))
([v dynv]
(let [key-v (first-in-vector v)
handler-fn (get @key->fn key-v)]
(if (nil? handler-fn)
(error "re-frame: no subscription handler registered for: \"" key-v "\". Returning a nil subscription.")
(let [result (reagent.ratom/atom nil)
dyn-vals (reaction (mapv deref dynv))
sub (reaction (handler-fn app-db v @dyn-vals))
;; handler-fn returns a reaction which is then wrapped in the sub reaction
;; need to double deref it to get to the actual value.
_ (run! (reset! result @@sub))] ;; run! here to force this to be started, won't ever run otherwise
result)))))