Skip to content

Latest commit

 

History

History
132 lines (95 loc) · 5.31 KB

README.md

File metadata and controls

132 lines (95 loc) · 5.31 KB

root

What is root?

root is a recursive UI resolver.

  • Approaches to UI development like react.js + graphQL require UI components to request data in a shape that satisfies the UI tree. This means the shape of the data is determined by the UI tree. Root takes an inverse approach where the UI tree is determined by the shape of the data.
  • A major benefit of this approach is that the UI layout is thus entirely determined by data, data that can be traversed, transformed and stored in arbitrary ways and by arbitrary means.
  • This is powerful for unstructured, user-determined, block-based UI's like rich documents (think Roam Research, Notion etc.) enabling queries and functions that, based on users' demands, derive the optimal presentation of a document.

Does root manage state?

No, root is agnostic to your state and state management. It cares only for a lookup function.

What is developing a root like?

root is data first. It recurses through your data resolving UI components along the way. With data in place, it lets you develop incrementally, even for data you haven't defined components for yet.

live examples

code example

Root is data-first. Without data it doesn't render. So let's define data to be:

(def data
  {1 {:type       :user
      :first-name "Eva"
      :last-name  "Luator"}})

And a minimal root with a :lookup into data and a :dispatch-fn on the :type key of each data-node:

(ns my-ns
  (:require [root.impl.core :as rc]))

(def root
  (rc/ui-root
   {:lookup       (fn [k] (get data k))
    :dispatch-fn  :type
    :content-keys [:content]
    :content-spec integer?}))

Rendering [root :resolve {:root-id 1}] results in

Note that even though we haven't defined any UI componenets, root renders a UI. This is the default-view in root. It always tries to render all keys and values on a given data node.

root acts like multimethod. Add new components like this:

(root :view :profile-pic
  (fn [{:keys [src]}]
    [:img.br-100 {:src src}]))

:profile-pic is the dispatch-value, the function is the UI component definition in hiccup.

If we now define data to be

(def data
  {1 {:type       :user
      :first-name "Eva"
      :last-name  "Luator"
      :content    {:profile-pic 3
                   :address     2}}
   2 {:type   :address
      :street "1 Long Infinite Loop"}
   3 {:type :profile-pic
      :src  "https://picsum.photos/id/1005/200"}})

root will use content-keys to examine each node for child-content and when such content exists pass it to lookup in turn. lookup is expected to return a node that might again have children under content-keys and from there the root grows (recurses).

When lookup returns a non-nil value, root invokes it with dispatch-fn looking for a matching UI component. In the case of :profile-pic it found the component we defined above. However, for :user and :address it fell back on default-view.

Now the rendered result is:

Here's the full code and live example. I recommend playing with it yourself.

problems with current UI development conventions
  • anti-ui-reuse: data-fetching inside components or through wrapper components
    • solution: data-first. components are defined as dispatches on data but oblivious to data-origin.
  • defocused UI: component tree as the result of numerous control flow decisions spread out over many components
    • child-components rendered as a reaction: state change → parent-component subscription → control flow expression → child-component
    • solution: component tree is derived from data traversal by dispatching nodes along the way. Derived component tree has broadly the same shape as the data.
  • lacking oversight: dense component tree (DOM elements and css classes and elements) makes semantics unclear
    • solution: print the (denormalized) data. Done.
  • application data is invisible in the UI until components are defined and instantiated
    • solution: Data first, incremental UI development! Render views for your data even when components for it are not yet defined.
problems with root
  • loose UIX dependency
    • for a reactive/dynamic root: single DB in xframe doesn't currently allow for multiple roots to render within each other
    • xframe (re-frame like event handling+subscription sub-lib) is overkill. Reagent ratom + cursor like abstraction would do.

Status

This software is so alpha you don't even...

License

Copyright © 2020 Dennis Heihoff

Distributed under the Eclipse Public License either version 1.0 or (at your option) any later version.