Gem Version Build Status Dependency Status Code Climate Coverage Status

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

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!