A bit about hexps
What on earth is a Hexp?
Hexps are basically snippets of HTML written in nothing but Ruby, here’s an example.
ruby
@message = "Hexps are fun for the whole family, from 9 to 99 years old."
@hexp = H[:div, {class: 'hexp-intro'}, [
[:p, @message]
]
]
For more info to get you up and running have a look at the API documentation for Hexp::Node.
Don’t people use templates for this kind of thing?
They do, this is an alternative approach. With templates you need to think about which parts need to be HTML-escaped, or you can make errors like forgetting a closing tag. With hexps you no longer need to think about escaping.
Wait how is that?
With traditional approaches you can insert plain text in your template, or snippets of HTML. The first must be escaped, the second should not. For your template they are all just strings, so you, the programmer, need to distinguish between the two in a way. For example by using html_escape
on one (explicit escaping), or html_safe
on the other (implicit escaping).
When using hexps you never deal with strings that actually contain HTML. Helper methods would return hexps instead, and you can combine those into bigger hexps. Strings inside a hexp are always just that, so they will always be escaped without you thinking about it.
So that’s it, easier escaping?
Well that’s not all, by having a simple lightweight representation of HTML that is a real data structure, you can really start programming your HTML. If you have an object that responds to to_hexp
, you can use that object inside a hexp, so you can use Object Orientation for your user interface. Like so
````ruby class ProfileLink < Struct.new(:user) def to_hexp H[:a, “profile-link”, href: “/user/#{user.id”}, user.name] end end
class Layout < Struct.new(:content) def to_hexp H[:html, [ [:body, [content]] ] end end
render inline: Layout.new(ProfileLink.new(@user)).to_html ````
Does it get any better?
It does! The really neat part is having filters that process this HTML tree before it gets serialized to text. This could be good for
- populating form fields
- adding extra admin buttons when logged in
- cleanly separate aspects of your app (e.g. discount codes) from ‘core’ implementation
- becoming filthy rich and/or ridiculously happy
What’s up with the funny name?
Hexp stands for HTML expressions. It’s a reference to s-expressions as they are known in LISP languages, a simple way to represent data as nested lists.
How to use it
Hexp objects come in two flavors : Hexp::Node
and Hexp::List
. A Node
consists of three parts : its tag
, attributes
and children
. A List
is just that, a list (of nodes).
To construct a Node
use H[tag, attributes, children]
. Use a Symbol
for the tag
, a Hash
for the attributes
, and an Array
for the children
. Attributes or children can be omitted when they are empty.
The list of children will automatically be converted to Hexp::List
, similarly for any nested nodes you can simply use [tag, attributes, children]
without the H
, all nodes in the tree will be converted to proper Hexp::Node
objects.
The entire API is centered around these two classes, and one of them you can think of as essentially just an Array
, in other words Hexp is super easy to learn. Try it out in irb
, have a look at the examples, and build cool stuff!
A note on immutability
All Hexp objects are deep frozen on creation, you can never alter them afterwards. Operations always return a new Hexp::Node
rather than working in place.
This might seem stringent when you are not used to this style of coding, but it’s a pattern that generally promotes good code.
Can I already use it
Hold your horses, this is very alpha level code! Feedback is very much appreciated, so try it out, and let me know what you think. Basic functionality is there, but it’s far from finished, and the API will probably still change based on feedback. You have been warned.
Is it any good?
Yes
How to install
At this point you’re best off grabbing the Git repo, e.g. with bundler
````sh # Gemfile
gem ‘hexp’, github: ‘plexus/hexp’ ````
Who is behind this
Hexp is being actively developed by Arne Brasseur. You can supportme on Gittip!