module Ideas.Text.HTML
( HTMLPage, HTMLBuilder
, addCSS, addScript, showHTML
, string, text
, htmlPage, link
, h1, h2, h3, h4, h5, h6
, preText, ul, table, keyValueTable
, image, space, spaces, highlightXML
, para, ttText, hr, br, pre, bullet
, divClass, spanClass
, idA, classA, styleA, titleA
, tt, italic, bold, big, small
, module Data.Monoid
) where
import Data.Char
import Data.List
import Data.Monoid
import Ideas.Text.XML
import Prelude hiding (div)
import qualified Ideas.Text.XML as XML
type HTMLBuilder = XMLBuilder
data HTMLPage = HTMLPage
{ title :: String
, styleSheets :: [FilePath]
, scripts :: [FilePath]
, htmlContent :: HTMLBuilder
}
instance InXML HTMLPage where
toXML page = makeXML "html" $
element "head"
[ tag "title" (string (title page))
, mconcat
[ element "link"
[ "rel" .=. "STYLESHEET"
, "href" .=. css
, "type" .=. "text/css"
]
| css <- styleSheets page
]
, mconcat
[ tag "script" ("src" .=. js)
| js <- scripts page
]
]
<> tag "body" (htmlContent page)
fromXML _ = fail "HTMLPage.fromXML"
showHTML :: HTMLPage -> String
showHTML = compactXML . toXML
addCSS :: FilePath -> HTMLPage -> HTMLPage
addCSS css page = page { styleSheets = css : styleSheets page }
addScript :: FilePath -> HTMLPage -> HTMLPage
addScript js page = page { scripts = js : scripts page }
htmlPage :: String -> HTMLBuilder -> HTMLPage
htmlPage s = HTMLPage s [] []
link :: BuildXML a => String -> a -> a
link url body = tag "a" $
("href" .=. url) <> body
h1, h2, h3, h4, h5, h6 :: BuildXML a => String -> a
h1 = tag "h1" . string
h2 = tag "h2" . string
h3 = tag "h3" . string
h4 = tag "h4" . string
h5 = tag "h5" . string
h6 = tag "h6" . string
para :: BuildXML a => a -> a
para = tag "p"
preText :: BuildXML a => String -> a
preText = pre . string
pre :: BuildXML a => a -> a
pre = tag "pre"
hr :: BuildXML a => a
hr = emptyTag "hr"
br :: BuildXML a => a
br = emptyTag "br"
ttText :: BuildXML a => String -> a
ttText = tt . string
ul :: BuildXML a => [a] -> a
ul = element "ul" . map (tag "li")
table :: BuildXML a => Bool -> [[a]] -> a
table b rows = element "table" $
("border" .=. "1") :
[ element "tr" $
("class" .=. getClass i) :
[ tag "td" c | c <- row ]
| (i, row) <- zip [0::Int ..] rows
]
where
getClass i
| i == 0 && b = "top-row"
| even i = "even-row"
| otherwise = "odd-row"
keyValueTable :: BuildXML a => [(String, a)] -> a
keyValueTable =
let f (s, a) = [spanClass "table-key" (string s), a]
in para . table False . map f
spaces :: BuildXML a => Int -> a
spaces n = mconcat (replicate n space)
space, bullet :: BuildXML a => a
space = XML.unescaped " "
bullet = XML.unescaped "•"
image :: BuildXML a => String -> a
image n = tag "img" ("src" .=. n)
divClass :: BuildXML a => String -> a -> a
divClass n a = tag "div" (classA n <> a)
spanClass :: BuildXML a => String -> a -> a
spanClass n a = tag "span" (classA n <> a)
highlightXML :: Bool -> XML -> HTMLBuilder
highlightXML nice
| nice = builder . highlight . makeXML "pre" . string . show
| otherwise = builder . highlight . makeXML "tt" . string . compactXML
where
highlight :: XML -> XML
highlight html = html {content = map (either (Left . f) Right) (content html)}
f :: String -> String
f [] = []
f list@(x:xs)
| "</" `isPrefixOf` list =
let (as, bs) = span isAlphaNum (drop 5 list)
in "<font color='blue'></" ++ as ++ "<font color='green'>" ++ g bs
| "<" `isPrefixOf` list =
let (as, bs) = span isAlphaNum (drop 4 list)
in "<font color='blue'><" ++ as ++ "<font color='green'>" ++ g bs
| otherwise = x : f xs
g [] = []
g list@(x:xs)
| "/>" `isPrefixOf` list =
"</font>/></font>" ++ f (drop 5 list)
| ">" `isPrefixOf` list =
"</font>></font>" ++ f (drop 4 list)
| x=='=' = "<font color='orange'>=</font>" ++ g xs
| otherwise = x : g xs
idA, classA, styleA, titleA :: BuildXML a => String -> a
idA = ("id" .=.)
classA = ("class" .=.)
styleA = ("style" .=.)
titleA = ("title" .=.)
tt :: BuildXML a => a -> a
tt = tag "tt"
italic :: BuildXML a => a -> a
italic = tag "i"
bold :: BuildXML a => a -> a
bold = tag "b"
big :: BuildXML a => a -> a
big = tag "big"
small :: BuildXML a => a -> a
small = tag "small"