reagent/demo/reagentdemo/news/clockpost.cljs

125 lines
4.6 KiB
Plaintext
Raw Normal View History

2014-02-22 11:36:34 +00:00
(ns reagentdemo.news.clockpost
(:require [reagent.core :as r :refer [atom]]
[reagent.interop :refer-macros [.' .!]]
2014-02-26 12:17:18 +00:00
[reagent.debug :refer-macros [dbg]]
2014-02-22 11:36:34 +00:00
[reagentdemo.syntax :refer-macros [get-source]]
[sitetools :as tools :refer [title link]]
2014-02-22 11:36:34 +00:00
[reagentdemo.common :as common :refer [demo-component]]
[reagentdemo.news.binaryclock :as binaryclock]))
(def url "news/binary-clock.html")
2014-02-22 16:24:22 +00:00
(def funmap (-> "reagentdemo/news/binaryclock.cljs"
get-source common/fun-map))
2014-02-22 11:36:34 +00:00
(def src-for (partial common/src-for funmap))
2014-02-22 16:24:22 +00:00
(defn fn-src [& parts]
[demo-component {:src (src-for (vec parts))
:no-heading true}])
2014-02-22 11:36:34 +00:00
(defn main [{:keys [summary]}]
2014-02-26 12:17:18 +00:00
(let [head "A binary clock"
lexclock {:href "http://www.lexicallyscoped.com/2014/01/23/clojurescript-react-om-binary-clock.html"}
hopclock {:href "http://pmbauer.github.io/2014/01/27/hoplon-binary-clock/"}
om {:href "https://github.com/swannodette/om"}
2014-02-26 13:17:02 +00:00
hoplon {:href "http://hoplon.io"}
clocksrc {:href "https://github.com/reagent-project/reagent/blob/master/demo/reagentdemo/news/binaryclock.cljs"}]
2014-02-22 11:36:34 +00:00
[:div.reagent-demo
[:h1 [link {:href url} head]]
2014-02-22 16:24:22 +00:00
[title head]
2014-02-22 11:36:34 +00:00
[:div.demo-text
2014-02-22 16:24:22 +00:00
(when-not summary
[:div
2014-02-26 12:17:18 +00:00
[:div.clearfix
[binaryclock/main]]
[:div [:strong "Click to toggle 1/100th seconds."]]])
2014-02-22 11:36:34 +00:00
2014-02-26 12:17:18 +00:00
[:p "Fredrik Dyrkell wrote a very nice " [:a lexclock "binary
clock"] " using " [:a om "Om"] ". I thought Id replicate that
using Reagent for fun (another re-write, using " [:a
hoplon "Hoplon"] ", can be seen " [:a hopclock "here"] ")."]
2014-02-22 16:24:22 +00:00
2014-02-26 12:17:18 +00:00
[:p "So, without further ado, here is a binary clock using Reagent."]
2014-02-22 11:36:34 +00:00
(if summary
[link {:href url
:class 'news-read-mode} "Read more"]
2014-02-22 11:36:34 +00:00
[:div.demo-text
2014-02-26 12:17:18 +00:00
[fn-src :nsr]
2014-02-22 16:24:22 +00:00
[:p "We start with the basics: The clock is built out of
2014-02-26 12:17:18 +00:00
cells, with a light colour if the bit the cell corresponds to
2014-02-22 16:24:22 +00:00
is set."]
[fn-src :cell]
[:p "Cells are combined into columns of four bits, with a
decimal digit at the bottom."]
[fn-src :column]
[:p "Columns are in turn combined into pairs:"]
[fn-src :column-pair]
[:p "We'll also need the legend on the left side:"]
[fn-src :legend]
[:p "We combine these element into a component that shows the
legend, hours, minutes and seconds; and optionally 1/100
seconds. It also responds to clicks."]
[fn-src :clock]
[:p "We also need to keep track of the time, and of the
detail shown, in a Reagent atom. And a function to update the
time."]
[fn-src :clock-state :update-time]
2014-02-26 12:17:18 +00:00
[:p "And finally we use the " [:code "clock"] " component.
The current time is scheduled to be updated, after a suitable
delay, every time the main component is rendered ("
[:code "reagent.core/next-tick"] " is just a front for "
[:code "requestAnimationFrame"] "):"]
[fn-src :main]
2014-02-26 13:17:02 +00:00
[:p "The entire source is also available " [:a
clocksrc "here"] "."]
2014-02-26 12:17:18 +00:00
[:h2 "How it all works"]
[:p "Reading through the source, it may look like the entire
clock component is recreated from scratch whenever the time
changes. "]
[:p "That is an illusion: Reagent and React together
makes sure that only the parts of the DOM that actually need
to change are updated. For example, the "
[:code "column-pair"] " function corresponding to hours only
runs once every hour."]
[:p "And thats what makes Reagent and React fast. Try
clicking on the clock to toggle the display of 1/100th
seconds. Most browsers should have no trouble at all keeping
up (even if they wont actually show every 1/100th second:
they are typically limited to roughly 60 fps)."]
[:p "But it is a very handy illusion. Almost the entire UI is
made up of pure functions, transforming immutable data into
other immutable data structures. That makes them easy to
reason about, and trivial to test. You dont have to care
about ”model objects”, or about how to update the DOM
efficiently. "]
[:p "Just pass arguments to component functions, return a UI
description that corresponds to those arguments, and leave it
to React to actually display that UI."]])]]))
2014-02-22 11:36:34 +00:00
(tools/register-page url (fn [] [main]))