summaryrefslogtreecommitdiff
path: root/src/Hakyll/Web/CompressCss.hs
blob: aac48b1dd5314166c6ba7bb95695195744f0b092 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
--------------------------------------------------------------------------------
-- | Module used for CSS compression. The compression is currently in a simple
-- state, but would typically reduce the number of bytes by about 25%.
module Hakyll.Web.CompressCss
    ( compressCssCompiler
    , compressCss
    ) where


--------------------------------------------------------------------------------
import           Data.Char               (isSpace)
import           Data.List               (isPrefixOf)


--------------------------------------------------------------------------------
import           Hakyll.Core.Compiler
import           Hakyll.Core.Item
import           Hakyll.Core.Util.String


--------------------------------------------------------------------------------
-- | Compiler form of 'compressCss'
compressCssCompiler :: Compiler (Item String)
compressCssCompiler = fmap compressCss <$> getResourceString


--------------------------------------------------------------------------------
-- | Compress CSS to speed up your site.
compressCss :: String -> String
compressCss = compressSeparators . stripComments . compressWhitespace


--------------------------------------------------------------------------------
-- | Compresses certain forms of separators.
compressSeparators :: String -> String
compressSeparators [] = []
compressSeparators str
    | isPrefixOf "\"" str = head str : retainConstants compressSeparators "\"" (drop 1 str)
    | isPrefixOf "'" str = head str : retainConstants compressSeparators "'" (drop 1 str)
    | stripFirst = compressSeparators (drop 1 str)
    | stripSecond = compressSeparators (head str : (drop 2 str))
    | isPrefixOf ";}" str = '}' : compressSeparators (drop 2 str)
    | otherwise = head str : compressSeparators (drop 1 str)
  where
    prefix p = isPrefixOf p str
    stripFirst = or $ map prefix ["  ", " {", " }", ";;"]
    stripSecond = or $ map prefix ["{ ", "} ", "; "]

--------------------------------------------------------------------------------
-- | Compresses all whitespace.
compressWhitespace :: String -> String
compressWhitespace [] = []
compressWhitespace str
    | isPrefixOf "\"" str = head str : retainConstants compressWhitespace "\"" (drop 1 str)
    | isPrefixOf "'" str = head str : retainConstants compressWhitespace "'" (drop 1 str)
    | otherwise = replaceAll "[ \t\n\r]+" (const " ") str


--------------------------------------------------------------------------------
-- | Function that strips CSS comments away.
stripComments :: String -> String
stripComments [] = []
stripComments str
    | isPrefixOf "/*" str = stripComments $ eatComments $ drop 2 str
    | otherwise = head str : stripComments (drop 1 str)
  where
    eatComments str'
        | null str' = []
        | isPrefixOf "*/" str' = drop 2 str'
        | otherwise = eatComments $ drop 1 str'

--------------------------------------------------------------------------------
-- | Helper function to handle string constants correctly.
retainConstants :: (String -> String) -> String -> String -> String
retainConstants f delim str
    | null str = []
    | isPrefixOf delim str = head str : f (drop 1 str)
    | otherwise = head str : retainConstants f delim (drop 1 str)