Now let's change the app's main page from "Hello, World" to something a little more chatty.
We need to write code that will generate HTML. To do this, we will use a library called hiccup
. We don't have this library yet, so we're going to add it. Adding a new library requires two steps:
- Add the library to the dependency section of the
project.clj
file.
Add hiccup by updating the project.clj
dependencies vector to include this:
[hiccup "1.0.5"]
- Import the hiccup libraries into the handler namespace by adding to the
ns :require
declaration. Include these in(:require )
form.
[hiccup.core :as hiccup]
[hiccup.page :as page]
Let's use hiccup to generate the html by changing app-routes
:
(defroutes app-routes
(GET "/" []
(page/html5
[:head
[:title "chatter"]]
[:body
[:h1 "Our Chat App"]]))
(route/not-found "Not Found"))
Once the code is updated, let's try it out.
$: lein ring server-headless
Refresh your browser.
Now http://localhost:3000
displays "Our Chat App". Right-click and select View Page Source
to see it's now proper HTML complete with head
, title
, and body
.
The hiccup function page/html5
generates an HTML page. It expects Clojure vectors with symbols representing corresponding HTML tags. Hiccup will automatically add the closing tag when it reaches the end of the vector.
Vectors are a Clojure data structure used to contain sequences of things, including other vectors. Vectors are written using square brackets or the
(vec)
function. For example,[1 2 3]
or(vec 1 2 3)
are vectors containing the numbers 1, 2, and 3. Hiccup uses vectors of keywords to represent sections of HTML.
Keywords are names that begin with a colon.
:title
,:x
, and:favorite-color
are all keywords. Clojure often uses keywords where other languages use strings. If you were to use Clojure to query a database, Clojure would probably use keywords to represent the column names. Hiccup uses keywords to represent the names of HTML elements.
Where HTML uses
<body>
, hiccup would expect:body
. Where HTML uses<title>
, hiccup uses:title
. Because the keywords are enclosed in a vector, the closing of the HTML tag is unnecessary. The closing of the surrounding vector signals where the HTML section ends.
Ok before we fix some problems lets add some passing tests. I know right, well better late than never, and especially as a scaffold for our upcoming changes. Open test/chatter/handler_test.clj
file and then replace the contents with:
(ns chatter.handler-test
(:use [kerodon.core]
[kerodon.test]
[clojure.test])
(:require [chatter.handler :refer [app]]))
(deftest test-app
(testing "main route"
(-> (session app)
(visit "/")
(has (status? 200) "page is found")
(has (some-text? "Our Chat App"))))
(testing "not-found route"
(-> (session app)
(visit "/invalid")
(has (status? 404) "page not found"))))
These are two interaction tests, two tests covering the projected generated scenarios, plus one additional to cover code we have already written.
$: lein test
Confirm that all three assertions have passed and that there are no failures or errors, then continue on to the refactor.
A problem with our new app-routes
is that it has two different functions right now. Its main role is to take the incoming request and decide what to do. Right now it's doing that, but it is also generating a full HTML page. As we add more pages, this will become too complicated to manage. We'll get ahead of the game by splitting out the task of generating the HTML into a helper function. Add a function index-view
and call it in the GET
function:
(defn index-view
"This generates the HTML for displaying messages"
[]
(page/html5
[:head
[:title "chatter"]]
[:body
[:h1 "Our Chat App"]]))
(defroutes app-routes
(GET "/" [] (index-view))
(route/not-found "Not Found"))
The function index-view
takes no arguments (for now!). It calls a hiccup function page/html5
to generate html from a vector representing the head
sections and a vector representing the body
elements of the html.
Save handler.clj
, and refresh the browser to make sure our page still works. From the outside, we shouldn't see a change. The page should still display "Our Chat App" and the html should be identical.
Confirm that all three assertions have passed and that there are no failures or errors, pat yourself on the back.
Our app is not displaying messages, nor do we have a way of adding messages. Let's make that happen now.
Let's change the app so it displays messages. We'll represent the messages as a vector of maps. Each map will have a :name
and :message
key and the corresponding value, in quotes. For example, the code below will represent blue's first post.
(def messages {:name "blue" :message "blue's first post"})
This is a map with two keys named messages
.
:name
is "blue", because blue posted it.
:message
is the content of the post.
To get a value from a map, pass the map and key into the get
function. For example,
(get messages :name)
When the keys are keywords, you can also use the keyword as a function that takes the map and returns the values.
(:name messages)
returns "blue"
.
Maps are everywhere in Clojure and are used for many things where other languages might use objects.
Let's call the vector simply chat-messages
and hard code some samples to get started. Add a chat-messages variable to handler.clj
.
Before your index-view
function add:
(def chat-messages [{:name "Bob" :message "hello, world"}
{:name "George" :message "What's up internet"}
{:name "Sally" :message "Hungry for some pizza?"}])
Next, we'll modify the HTML to display the messages. We will also add a parameter to the index-view
function so that we can give it the messages we want displayed.
(defn index-view
"This generates the HTML for displaying messages"
[messages]
(page/html5
[:head
[:title "chatter"]]
[:body
[:h1 "Our Chat App"]
[:p messages]]))
(defroutes app-routes
(GET "/" [] (index-view chat-messages))
(route/not-found "Not Found"))
Save handler.clj
and refresh the browser.
This blows up spectacularly.
This is a stack trace - it gives us an idea what the program was doing when it hit the problem. Ignore all the files that aren't ones you wrote for the project. In my case, the first file of interest is
handler.clj
, line 14, the index-view
function.
The exception message on the top, "... is not a valid element name"
, is a clue to what's wrong. Elements are what fragments of html are called. Hiccup is responsible for generating html from Clojure symbols. The problem is that we've got a map with symbols in it and hiccup thinks they're html. They're not, so it creates an error.
We can fix the issue by converting our maps to strings.
(defn index-view
"This generates the HTML for displaying messages"
[messages]
(page/html5
[:head
[:title "chatter"]]
[:body
[:h1 "Our Chat App"]
[:p (str messages)]]))
Save handler.clj
, and refresh the browser.
This fixes the exception but it's ugly.
Let's take the messages and put them in a bootstrap components.
(defn message-view
"This generates the HTML for displaying a single message"
[message]
[:div.panel.panel-default
[:div.panel-heading (hiccup/h (:name message))]
[:div.panel-body (hiccup/h (:message message))]])
Now our index-view
looks like: (we added boostrap css and called map around the list of messages).
(defn index-view
"This generates the HTML for the index page"
[messages]
(page/html5
[:head
[:title "chatter"]
(page/include-css "https://maxcdn.bootstrapcdn.com/bootstrap/3.3.7/css/bootstrap.min.css")]
[:body
[:div.container
[:h1 "Our Chat App"]
[:div.row (map message-view messages)]]]))
Save handler.clj
, then refresh the browser. Our hard-coded messages should now display in the page.
Next Chapter 7: Add Form