multi-lang by file base
Create a multi-lang site by writing articles with the following structure.
- posts/2024-05-28-filename-en.md
- posts/2024-05-28-filename-jp.md
specific example
This site is being created with the above structure. The engine part will be separated and released at a later date.
modify Route
First, change the Route
. The following is an image of the change.
- path: original source
- posts/2024-05-28-filename-en.md
- posts/2024-05-28-filename-jp.md
- path: converted html
- en/2024-05-28-filename.html
- jp/2024-05-28-filename.html
Excerpt : codes of Route
"posts/*") $ do
match (fromArticle route langRoute
langRoute :: Routes
= customRoute ((`replaceExtension` ".html") . getLangPath)
langRoute
langseperator :: String
= "-"
langseperator
takeLang :: Identifier -> String
= last . splitAll langseperator . takeBaseName . toFilePath
takeLang
dropLang :: Identifier -> String
= L.intercalate langseperator . init . splitAll langseperator . toFilePath
dropLang
getLangPath :: Identifier -> String
id = lang `combine` dropLang id
getLangPath where
= takeLang id lang
The operations being performed are simple and include the following
- get filepath from
Identifier
- pick lang-code such as like
-en
,-jp
- swap and re-concat with
/
embed link which another lang
It is achieved as follows.
- search same
filename
fromIdentifier
such as :filename-lang.md
- put link which search result
anothorLangFields :: String -> Snapshot -> Context String
= listFieldWith name baseCtx f
anothorLangFields name snapshot where
= loadAllSnapshots (fromGlob (mainname ++ "*")) snapshot
f item where
id = itemIdentifier item
= dropLang id
mainname = takeLang id lang
below is how to use anothorLangFields
.
- make
snapshot
- apply
template
withpostCtx
"posts/*") $ do
match (fromArticle
route langRoute$ do
compile
pandocCompiler>>= saveSnapshot "posts-content"
>>= loadAndApplyTemplate (idBase "templates/post.html") postCtx
postCtx
is to connectbaseCtx
andanothorLangFields
.
postCtx :: Context String
=
postCtx
baseCtx`mappend` anothorLangFields "anothorLangs" "posts-content"
baseCtx
: a context to embed basic info
baseCtx :: Context String
=
baseCtx "date" "%Y/%m/%d"
dateField `mappend` dateFieldMeta "last-modified" "last-modified" "%Y/%m/%d"
`mappend` readTimeField "readtime" "posts-content"
`mappend` langField "lang"
`mappend` sameLangField "home" "index"
`mappend` defaultContext
The context now has the following structure
- date
- last-modified
- readtime
- lang
- home
- (defaultContext)
- anotherlangs
- date
- last-modified
- readtime
- lang
- home
- (defaultContext)
It can be called as like below in template
.
$for(anothorLangs)$<tr>
<td>
<a href="$url$">$title$ ($lang$)</a>
</td>
<td>$date$</td>
<td>$if(last-modified)$ $last-modified$ $else$ $date$ $endif$</td>
</tr>
$endfor$
to use inter link in md with modified Route
This is separated into a separate page due to its more general content.
to use inter link in md with modified Route by Hakyll
to reconstruct post list
-en.md
and -jp.md
will be lined up respectively if posts/*.md
read in this file-based system.
but necessary is unieque one of same file
in posts/file-lang.md
.
apply filterUniqs
againstposts/*
like the following and save only unique file
.
$ do
compile <- recentFirst . filterUniqs =<< loadAllSnapshots "posts/*" "posts-content" posts
filterUniqs
is a function to reconstruct [Item String]
.
pick files which filename dropped a part -lang
is not same.
and concat picked items. and make a list to concated it.
filterUniqs :: [Item String] -> [Item String]
= []
filterUniqs [] : xs) = x : filterUniqs (filter ((/= f x) . f) xs)
filterUniqs (x where
= dropLang . itemIdentifier f
This will achieve the expected operation.