blaze-colonnade-0.1: Helper functions for using blaze-html with colonnade

Safe HaskellNone
LanguageHaskell2010

Text.Blaze.Colonnade

Contents

Description

Build HTML tables using blaze-html and colonnade. The bottom of this page has a tutorial that walks through a full example, illustrating how to meet typical needs with this library. It is recommended that users read the documentation for colonnade first, since this library builds on the abstractions introduced there. A concise example of this library's use:

>>> :set -XOverloadedStrings
>>> :module + Colonnade Text.Blaze.Html Text.Blaze.Colonnade
>>> let col = headed "Grade" (toHtml . fst) <> headed "Letter" (toHtml . snd)
>>> let rows = [("90-100",'A'),("80-89",'B'),("70-79",'C')]
>>> printVeryCompactHtml (encodeHeadedHtmlTable mempty col rows)
<table>
    <thead><th>Grade</th><th>Letter</th></thead>
    <tbody>
        <tr><td>90-100</td><td>A</td></tr>
        <tr><td>80-89</td><td>B</td></tr>
        <tr><td>70-79</td><td>C</td></tr>
    </tbody>
</table>

Synopsis

Apply

encodeHeadedHtmlTable Source #

Arguments

:: Foldable f 
=> Attribute

Attributes of <table> element

-> Colonnade Headed Html a

How to encode data as columns

-> f a

Collection of data

-> Html 

Encode a table with a header. Table cells cannot have attributes applied to them.

encodeHeadlessHtmlTable Source #

Arguments

:: Foldable f 
=> Attribute

Attributes of <table> element

-> Colonnade Headless Html a

How to encode data as columns

-> f a

Collection of data

-> Html 

Encode a table without a header. Table cells cannot have attributes applied to them.

encodeHeadedCellTable Source #

Arguments

:: Foldable f 
=> Attribute

Attributes of <table> element

-> Colonnade Headed Cell a

How to encode data as columns

-> f a

Collection of data

-> Html 

Encode a table with a header. Table cells may have attributes applied to them.

encodeHeadlessCellTable Source #

Arguments

:: Foldable f 
=> Attribute

Attributes of <table> element

-> Colonnade Headless Cell a

How to encode data as columns

-> f a

Collection of data

-> Html 

Encode a table without a header. Table cells may have attributes applied to them.

encodeTable Source #

Arguments

:: (Foldable f, Foldable h) 
=> Maybe Attribute

Attributes of <thead>, pass Nothing to omit <thead>

-> Attribute

Attributes of <tbody> element

-> (a -> Attribute)

Attributes of each <tr> element

-> ((Html -> Html) -> c -> Html)

Wrap content and convert to Html

-> Attribute

Attributes of <table> element

-> Colonnade h c a

How to encode data as a row

-> f a

Collection of data

-> Html 

Encode a table. This handles a very general case and is seldom needed by users. One of the arguments provided is used to add attributes to the generated <tr> elements.

Cell

The Cell type is used to build a Colonnade that has Html content inside table cells and may optionally have attributes added to the <td> or <th> elements that wrap this HTML content.

data Cell Source #

The attributes that will be applied to a <td> and the HTML content that will go inside it. When using this type, remember that Attribute, defined in blaze-markup, is actually a collection of attributes, not a single attribute.

Constructors

Cell 

Instances

htmlCell :: Html -> Cell Source #

Create a Cell from a Widget

stringCell :: String -> Cell Source #

Create a Cell from a String

textCell :: Text -> Cell Source #

Create a Cell from a Text

lazyTextCell :: Text -> Cell Source #

Create a Cell from a lazy text

builderCell :: Builder -> Cell Source #

Create a Cell from a text builder

Interactive

printCompactHtml :: Html -> IO () Source #

Pretty print an HTML table, stripping whitespace from inside <td>, <th>, and common inline tags. The implementation is inefficient and is incorrect in many corner cases. It is only provided to reduce the line count of the HTML printed by GHCi examples in this module's documentation. Use of this function is discouraged.

printVeryCompactHtml :: Html -> IO () Source #

Similar to printCompactHtml. Additionally strips all whitespace inside <tr> elements and <thead> elements.

Tutorial

We start with a few necessary imports and some example data types:

>>> :set -XOverloadedStrings
>>> import Data.Monoid (mconcat,(<>))
>>> import Data.Char (toLower)
>>> import Data.Functor.Contravariant (Contravariant(contramap))
>>> import Colonnade (Colonnade,Headed,Headless,headed)
>>> import Text.Blaze.Html (Html, toHtml, toValue)
>>> import qualified Colonnade as C
>>> import qualified Text.Blaze.Html5 as H
>>> data Department = Management | Sales | Engineering deriving (Show,Eq)
>>> data Employee = Employee { name :: String, department :: Department, age :: Int }

We define some employees that we will display in a table:

>>> :{
let employees = 
      [ Employee "Thaddeus" Sales 34
      , Employee "Lucia" Engineering 33
      , Employee "Pranav" Management 57
      ]
:}

Let's build a table that displays the name and the age of an employee. Additionally, we will emphasize the names of engineers using a <strong> tag.

>>> :{
let tableEmpA :: Colonnade Headed Html Employee
    tableEmpA = mconcat
      [ headed "Name" $ \emp -> case department emp of
          Engineering -> H.strong (toHtml (name emp))
          _ -> toHtml (name emp)
      , headed "Age" (toHtml . show . age)
      ]
:}

The type signature of tableEmpA is inferrable but is written out for clarity in this example. Additionally, note that the first argument to headed is of type Html, so OverloadedStrings is necessary for the above example to compile. To avoid using this extension, it is possible to instead use toHtml to convert a String to Html. Let's continue:

>>> let customAttrs = HA.class_ "stylish-table" <> HA.id "main-table"
>>> printCompactHtml (encodeHeadedHtmlTable customAttrs tableEmpA employees)
<table class="stylish-table" id="main-table">
    <thead>
        <th>Name</th>
        <th>Age</th>
    </thead>
    <tbody>
        <tr>
            <td>Thaddeus</td>
            <td>34</td>
        </tr>
        <tr>
            <td><strong>Lucia</strong></td>
            <td>33</td>
        </tr>
        <tr>
            <td>Pranav</td>
            <td>57</td>
        </tr>
    </tbody>
</table>

Excellent. As expected, Lucia's name is wrapped in a <strong> tag since she is an engineer.

One limitation of using Html as the content type of a Colonnade is that we are unable to add attributes to the <td> and <th> elements. This library provides the Cell type to work around this problem. A Cell is just Html content and a set of attributes to be applied to its parent th or td. To illustrate how its use, another employee table will be built. This table will contain a single column indicating the department of each employ. Each cell will be assigned a class name based on the department. To start off, let's build a table that encodes departments:

>>> :{
let tableDept :: Colonnade Headed Cell Department
    tableDept = mconcat
      [ headed "Dept." $ \d -> Cell
          (HA.class_ (toValue (map toLower (show d))))
          (toHtml (show d))
      ]
:}

Again, OverloadedStrings plays a role, this time allowing the literal "Dept." to be accepted as a value of type Cell. To avoid this extension, stringCell could be used to upcast the String. To try out our Colonnade on a list of departments, we need to use encodeHeadedCellTable instead of encodeHeadedHtmlTable:

>>> let twoDepts = [Sales,Management]
>>> printCompactHtml (encodeHeadedCellTable customAttrs tableDept twoDepts)
<table class="stylish-table" id="main-table">
    <thead>
        <th>Dept.</th>
    </thead>
    <tbody>
        <tr>
            <td class="sales">Sales</td>
        </tr>
        <tr>
            <td class="management">Management</td>
        </tr>
    </tbody>
</table>

The attributes on the <td> elements show up as they are expected to. Now, we take advantage of the Contravariant instance of Colonnade to allow this to work on Employee's instead:

>>> :t contramap
contramap :: Contravariant f => (a -> b) -> f b -> f a
>>> let tableEmpB = contramap department tableDept
>>> :t tableEmpB
tableEmpB :: Colonnade Headed Cell Employee
>>> printCompactHtml (encodeHeadedCellTable customAttrs tableEmpB employees)
<table class="stylish-table" id="main-table">
    <thead>
        <th>Dept.</th>
    </thead>
    <tbody>
        <tr>
            <td class="sales">Sales</td>
        </tr>
        <tr>
            <td class="engineering">Engineering</td>
        </tr>
        <tr>
            <td class="management">Management</td>
        </tr>
    </tbody>
</table>

This table shows the department of each of our three employees, additionally making a lowercased version of the department into a class name for the <td>. This table is nice for illustrative purposes, but it does not provide all the information that we have about the employees. If we combine it with the earlier table we wrote, we can present everything in the table. One small roadblock is that the types of tableEmpA and tableEmpB do not match, which prevents a straightforward monoidal append:

>>> :t tableEmpA
tableEmpA :: Colonnade Headed Html Employee
>>> :t tableEmpB
tableEmpB :: Colonnade Headed Cell Employee

We can upcast the content type with mapContent. Monoidal append is then well-typed, and the resulting Colonnade can be applied to the employees:

>>> let tableEmpC = C.mapContent htmlCell tableEmpA <> tableEmpB
>>> :t tableEmpC
tableEmpC :: Colonnade Headed Cell Employee
>>> printCompactHtml (encodeHeadedCellTable customAttrs tableEmpC employees)
<table class="stylish-table" id="main-table">
    <thead>
        <th>Name</th>
        <th>Age</th>
        <th>Dept.</th>
    </thead>
    <tbody>
        <tr>
            <td>Thaddeus</td>
            <td>34</td>
            <td class="sales">Sales</td>
        </tr>
        <tr>
            <td><strong>Lucia</strong></td>
            <td>33</td>
            <td class="engineering">Engineering</td>
        </tr>
        <tr>
            <td>Pranav</td>
            <td>57</td>
            <td class="management">Management</td>
        </tr>
    </tbody>
</table>

Discussion

In this module, some of the functions for applying a Colonnade to some values to build a table have roughly this type signature:

Foldable a => Colonnade Headedness Cell a -> f a -> Html

The Colonnade content type is Cell, but the content type of the result is Html. It may not be immidiately clear why this is useful done. Another strategy, which this library also uses, is to write these functions to take a Colonnade whose content is Html:

Foldable a => Colonnade Headedness Html a -> f a -> Html

When the Colonnade content type is Html, then the header content is rendered as the child of a <th> and the row content the child of a <td>. However, it is not possible to add attributes to these parent elements. To accomodate this situation, it is necessary to introduce Cell, which includes the possibility of attributes on the parent node.