More Gui
19 Jul 2017 | ourMachinery imgui redom hyperApp elmI like UI! I said it multiple times. And I recently found three new posts on exciting new UI technology.
Our Machinery IMGUI UI
The original creators of Stingray: Niklas and Tobias have a new venture. It is called Our Machinery. Their development blog is incredibly interesting. And Niklas struck gold (with me at least) with its last post on IMGUI. He explains in great great details how they implemented their UI library. The whole thing is both clean and well architectured (thanks to Niklas) and super efficient graphically speaking (thanks to Tobias). This UI example showcases what type of widgets and functionalities (docking system FTW!) they support:
Obviously this is programmer art. If only we had access to the source of their library to see the internal beauty of that UI system :)
Redom
Virtual DOM is all the craze in web UI framework (React, Preact, Mithril, Inferno…). Virtual DOM is really the IMGUI of web UI. Then on the total opposite we have the new, tiny and shiny new library called Re:dom. This library DOESN’T use a virtual DOM. Instead it provides a set of helpers that makes it easier to create web view using an API might sound similar to React but that creates DOM element directly. It is thus really performant since it is basically a thin wrapper over Vanilla JS.
This small examples shows how to create a new component and mount it on the DOM:
import { el, mount } from 'redom';
// define Login component
class Login {
constructor () {
this.el = el('form#login',
this.email = el('input.email', { type: 'email' }),
this.pass = el('input.pass', { type: 'password' }),
this.submit = el('button', { type: 'submit' },
'Sign in'
)
);
this.el.onsubmit = e => {
e.preventDefault();
const email = this.email.value;
const pass = this.pass.value;
console.log(email, pass);
};
}
}
// create login
const login = new Login();
// mount to DOM
mount(document.body, login);
el
is an hyperscript syntax to create DOM element (tough you can use JSX if you prefer).
Return to HyperApp
I talked a little bit about HyperApp a while ago. This is a nice super small (about 1Kb) library that combines Virtual DOM and state management through a totally functional and stateless set of components.
Small aparte on Elm
The whole workflow of an HyperApp webapp is based on the Elm Architecture. As a side note, Elm is a totally functional language that transpiles to javascript and that can be used to create webapps. A simple “counter app” would look like this:
-- Read more about this program in the official Elm guide:
-- https://guide.elm-lang.org/architecture/user_input/buttons.html
import Html exposing (beginnerProgram, div, button, text)
import Html.Events exposing (onClick)
main =
beginnerProgram { model = 0, view = view, update = update }
view model =
div []
[ button [ onClick Decrement ] [ text "-" ]
, div [] [ text (toString model) ]
, button [ onClick Increment ] [ text "+" ]
]
type Msg = Increment | Decrement
update msg model =
case msg of
Increment ->
model + 1
Decrement ->
model - 1
The codepen for this app can be seen here.
Getting Started with Hyperapp
Jorge Bucaran, the developer of HyperApp has created a great Getting Started article. If we look at the Hyperapp code of the same counter app:
app({
state: 0,
// the view function make use of JSX but you can
// use an hyperscript notation as well
view: (state, actions) => (
<main>
<h1>{state}</h1>
<button onclick={actions.add}>+</button>
<button onclick={actions.sub}>-</button>
</main>
),
actions: {
add: state => state + 1,
sub: state => state - 1
}
})
We see that the actions are similar to Redux Reducers: pure function that computes a new state from a previous state according to a set of input. This is how HyperApp achieves its stateless components.