diff options
-rw-r--r-- | src/Hakyll/Web/Page/Metadata.hs | 67 |
1 files changed, 54 insertions, 13 deletions
diff --git a/src/Hakyll/Web/Page/Metadata.hs b/src/Hakyll/Web/Page/Metadata.hs index efae390..948c349 100644 --- a/src/Hakyll/Web/Page/Metadata.hs +++ b/src/Hakyll/Web/Page/Metadata.hs @@ -12,6 +12,7 @@ module Hakyll.Web.Page.Metadata , copyField , renderDateField , renderDateFieldWith + , comparePagesByDate , renderModificationTime , renderModificationTimeWith , copyBodyToField @@ -22,7 +23,7 @@ import Prelude hiding (id) import Control.Category (id) import Control.Arrow (Arrow, arr, (>>>), (***), (&&&)) import Data.List (intercalate) -import Data.Maybe (fromMaybe) +import Data.Maybe (fromMaybe, fromJust, isJust) import Data.Time.Clock (UTCTime) import Data.Time.Format (parseTime, formatTime) import qualified Data.Map as M @@ -118,11 +119,23 @@ copyField :: String -- ^ Key to copy -> Page a -- ^ Resulting page copyField src dst = renderField src dst id --- | When the metadata has a field called @path@ in a --- @folder/yyyy-mm-dd-title.extension@ format (the convention for pages), --- this function can render the date. +-- | When the metadata has a field called @published@ in one of the +-- following formats then this function can render the date. -- --- > renderDate "date" "%B %e, %Y" "Date unknown" +-- * @Sun, 01 Feb 2000 13:00:00 UT@ (RSS date format) +-- +-- * @2000-02-01T13:00:00Z@ (Atom date format) +-- +-- * @February 1, 2000 1:00 PM@ (PM is usually uppercase) +-- +-- * @February 1, 2000@ (assumes 12:00 AM for the time) +-- +-- Alternatively, when the metadata has a field called @path@ in a +-- @folder/yyyy-mm-dd-title.extension@ format (the convention for pages) +-- and no @published@ metadata field set, this function can render +-- the date. +-- +-- > renderDateField "date" "%B %e, %Y" "Date unknown" -- -- Will render something like @January 32, 2010@. -- @@ -143,19 +156,47 @@ renderDateFieldWith :: TimeLocale -- ^ Output time locale -> String -- ^ Default value -> Page a -- ^ Target page -> Page a -- ^ Resulting page -renderDateFieldWith locale key format defaultValue = - renderField "path" key renderDate' +renderDateFieldWith locale key format defaultValue page = + setField key renderTimeString page where - renderDate' filePath = fromMaybe defaultValue $ do + renderTimeString = fromMaybe defaultValue $ do + time <- getUTCMaybe locale page + return $ formatTime locale format time + +-- | Parser to try to extract and parse the time from the @published@ +-- field or from the filename. See 'renderDateField' for more information. +getUTCMaybe :: TimeLocale -- ^ Output time locale + -> Page a -- ^ Input page + -> Maybe UTCTime -- ^ Parsed UTCTime +getUTCMaybe locale page = getUTCMaybe' formats + where + formats = [ "%a, %d %b %Y %H:%M:%S UT" -- RSS format + , "%Y-%m-%dT%H:%M:%SZ" -- Atom format + , "%B %e, %Y %l:%M %p" + , "%B %e, %Y" + ] + getUTCMaybe' [] = do + filePath <- getFieldMaybe "path" page let dateString = intercalate "-" $ take 3 $ splitAll "-" $ takeFileName filePath - time <- parseTime defaultTimeLocale - "%Y-%m-%d" - dateString :: Maybe UTCTime - return $ formatTime locale format time + parseTime locale "%Y-%m-%d" dateString :: Maybe UTCTime + getUTCMaybe' (f:fs) = if isJust timeMaybe + then timeMaybe + else getUTCMaybe' fs + where timeMaybe = do dateString <- getFieldMaybe "published" page + parseTime locale f dateString :: Maybe UTCTime + +-- | Compare pages by the date and time parsed as in 'renderDateField', +-- where 'LT' implies earlier, and 'GT' implies later. For more details, +-- see 'renderDateField'. +comparePagesByDate :: Page a -> Page a -> Ordering +comparePagesByDate p1 p2 = compare p1UTC p2UTC + where p1UTC = fromMaybe defaultTime $ getUTCMaybe defaultTimeLocale p1 + p2UTC = fromMaybe defaultTime $ getUTCMaybe defaultTimeLocale p2 + defaultTime = fromJust $ parseTime defaultTimeLocale + "%B %e, %Y %l:%M %p" "January 1, 1900 12:00 AM" -- | Set the modification time as a field in the page --- renderModificationTime :: String -- ^ Destination key -> String |