website/app/Route/Blog/Slug_.elm

150 lines
3.6 KiB
Elm
Raw Normal View History

2023-10-24 05:23:25 +07:00
module Route.Blog.Slug_ exposing (ActionData, Data, Model, Msg, route)
import Article
import BackendTask exposing (BackendTask)
import Date exposing (Date)
import FatalError exposing (FatalError)
import Head
import Head.Seo as Seo
import Html.Styled exposing (..)
2023-12-10 03:59:01 +07:00
import Html.Styled.Attributes exposing (style)
import Link exposing (Link)
2023-10-24 05:23:25 +07:00
import Json.Decode as Decode exposing (Decoder)
import Json.Decode.Extra
import Pages.Url
import PagesMsg exposing (PagesMsg)
2023-12-10 03:59:01 +07:00
import Route
2023-10-24 05:23:25 +07:00
import RouteBuilder exposing (App, StatelessRoute)
import Shared
import View exposing (View)
import Markdown.Block
import Markdown.Renderer
import MarkdownCodec
import TailwindMarkdownRenderer
import Tailwind.Utilities as Tw
type alias Model =
{}
type alias Msg =
()
type alias RouteParams =
{ slug : String }
route : StatelessRoute RouteParams Data ActionData
route =
RouteBuilder.preRender
{ head = head
, pages = pages
, data = data
}
|> RouteBuilder.buildNoState { view = view }
pages : BackendTask FatalError (List RouteParams)
pages =
Article.blogPostsGlob
|> BackendTask.map
(List.map
(\globData ->
{ slug = globData.slug }
)
)
type alias Data =
{ metadata : ArticleMetadata
, body : List Markdown.Block.Block
}
type alias ActionData =
{}
data : RouteParams -> BackendTask FatalError Data
data routeParams =
MarkdownCodec.withFrontmatter Data
frontmatterDecoder
TailwindMarkdownRenderer.renderer
("content/blog/" ++ routeParams.slug ++ ".md")
type alias ArticleMetadata =
{ title : String
2024-01-29 23:35:05 +07:00
, subtitle : String
2023-10-24 05:23:25 +07:00
, description : String
, published : Date
-- , image : Pages.Url.Url
, draft : Bool
}
frontmatterDecoder : Decoder ArticleMetadata
frontmatterDecoder =
2024-01-29 23:35:05 +07:00
Decode.map5 ArticleMetadata
2023-10-24 05:23:25 +07:00
(Decode.field "title" Decode.string)
2024-01-29 23:35:05 +07:00
(Decode.field "subtitle" Decode.string)
(Decode.field "summary" Decode.string)
2023-10-24 05:23:25 +07:00
(Decode.field "published"
(Decode.string
|> Decode.andThen
(\isoString ->
Date.fromIsoString isoString
|> Json.Decode.Extra.fromResult
)
)
)
-- (Decode.oneOf
-- [ Decode.field "image" imageDecoder
-- , Decode.field "unsplash" UnsplashImage.decoder |> Decode.map UnsplashImage.imagePath
-- ]
-- )
(Decode.field "draft" Decode.bool
|> Decode.maybe
|> Decode.map (Maybe.withDefault False)
)
head :
App Data ActionData RouteParams
-> List Head.Tag
head app =
Seo.summary
{ canonicalUrlOverride = Nothing
2023-11-06 00:20:32 +07:00
, siteName = "nganhkhoa blogs"
2023-10-24 05:23:25 +07:00
, image =
2023-11-06 00:20:32 +07:00
{ url = Pages.Url.external ""
, alt = app.data.metadata.title
2023-10-24 05:23:25 +07:00
, dimensions = Nothing
, mimeType = Nothing
}
2023-11-06 00:20:32 +07:00
, description = ""
2023-10-24 05:23:25 +07:00
, locale = Nothing
2023-11-06 00:20:32 +07:00
, title = app.data.metadata.title
2023-10-24 05:23:25 +07:00
}
|> Seo.website
view :
App Data ActionData RouteParams
-> Shared.Model
-> View (PagesMsg Msg)
view app shared =
2023-12-10 03:59:01 +07:00
let rendered = (app.data.body |> Markdown.Renderer.render TailwindMarkdownRenderer.renderer) |> Result.withDefault []
in
2023-11-06 00:20:32 +07:00
{ title = app.data.metadata.title
2023-10-24 05:23:25 +07:00
, body =
2024-06-16 02:13:12 +07:00
[ section []
2024-01-29 23:35:05 +07:00
[ h1 [] [ text app.data.metadata.title ]
, h1 [ style "font-size" "1.5em" ] [ text app.data.metadata.subtitle ]
2024-06-16 02:13:12 +07:00
, section [] rendered
2024-01-29 23:35:05 +07:00
]
2023-12-10 03:59:01 +07:00
]
2023-10-24 05:23:25 +07:00
}