Last active
June 2, 2020 08:02
-
-
Save allgress/7ae4ae0261314582451a to your computer and use it in GitHub Desktop.
Use experimental fork of DataScript from https://github.com/allgress/datascript to handle undo and per-query subscriptions
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
;;; Use experimental fork of DataScript from https://github.com/allgress/datascript to handle undo and per-query subscriptions | |
(ns reagent-test.core | |
(:require [reagent.core :as reagent :refer [atom]] | |
[datascript :as d] | |
[cljs-uuid-utils :as uuid])) | |
(enable-console-print!) | |
(defn bind | |
([conn q] | |
(bind conn q (atom nil))) | |
([conn q state] | |
(let [k (uuid/make-random-uuid)] | |
(reset! state (d/q q @conn)) | |
(d/listen! conn k #(reset! state (d/q q @conn)) q [@conn]) | |
(set! (.-__key state) k) | |
state))) | |
(defn unbind | |
[conn state] | |
(d/unlisten! conn (.-__key state))) | |
;;; Creates a DataScript "connection" (really an atom with the current DB value) | |
(def conn (d/create-conn)) | |
;;; Add some data | |
(d/transact! conn [{:db/id -1 :name "Bob" :age 30} | |
{:db/id -2 :name "Sally" :age 25} | |
{:db/id -3 :spacy/name "Clorp" :spacy/age 12997573}]) | |
;;; Maintain DB history. | |
(def history (atom [])) | |
(d/listen! conn :history (fn [tx-report] | |
(condp = (:type tx-report) | |
:transact (swap! history conj tx-report) | |
:revert (swap! history pop)))) | |
(defn undo | |
[] | |
(when (not-empty @history) | |
(d/revert! conn (:tx-data (peek @history))))) | |
;;; Query to get name and age of peeps in the DB | |
(def q-peeps '[:find ?n ?a | |
:where | |
[?e :name ?n] | |
[?e :age ?a]]) | |
;; Simple reagent component. Returns a function that performs render | |
(defn peeps-view | |
[] | |
(let [peeps (bind conn q-peeps) | |
temp (atom {:name "" :age ""})] | |
(fn [] | |
[:div | |
[:h2 "Peeps!"] | |
[:ul | |
(map (fn [[n a]] [:li [:span (str "Name: " n " Age: " a)]]) @peeps)] | |
[:div | |
[:span "Name"][:input {:type "text" | |
:value (:name @temp) | |
:on-change #(swap! temp assoc-in [:name] (.. % -target -value))}]] | |
[:div | |
[:span "Age"][:input {:type "text" | |
:value (:age @temp) | |
:on-change #(swap! temp assoc-in [:age] (.. % -target -value))}]] | |
[:button | |
{:onClick (fn [] | |
(d/transact! conn [{:db/id -1 :name (:name @temp) :age (js/parseInt (:age @temp))}]) | |
(reset! temp {:name "" :age ""}))} | |
"Add Peep"] | |
[:button {:on-click undo :disabled (= 0 (count @history))} "Undo"]]))) | |
;;; Query to find peeps whose age is less than 18 | |
(def q-young '[:find ?n | |
:where | |
[?e :name ?n] | |
[?e :age ?a] | |
[(< ?a 18)]]) | |
;;; Uses reagent/create-class to create a React component with lifecyle functions | |
(defn younguns-view | |
[] | |
(let [y (atom nil)] | |
(reagent/create-class | |
{ | |
;; Subscribe to db transactions. | |
:component-will-mount (fn [] (bind conn q-young y)) | |
;; Unsubscribe from db transactions. | |
:component-will-unmount (fn [] (unbind conn y)) | |
:render | |
(fn [_] | |
[:div | |
[:h2 "Young 'uns (under 18)"] | |
[:ul | |
(map (fn [[n]] [:li [:span n]]) @y)]])}))) | |
;;; Some non-DB state | |
(def state (atom {:show-younguns false})) | |
;;; Uber component, contains/controls stuff and younguns. | |
(defn uber | |
[] | |
[:div | |
[:div [peeps-view]] | |
[:div {:style {:margin-top "20px"}} | |
[:input {:type "checkbox" | |
:name "younguns" | |
:onChange #(swap! state assoc-in [:show-younguns] (.. % -target -checked))} | |
"Show Young'uns"]] | |
(when (:show-younguns @state) | |
[:div [younguns-view]]) | |
]) | |
;;; Initial render | |
(reagent/render-component [uber] (.-body js/document)) |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment