nodoku-demo-page is a demo project for the Nodoku static site builder.
This repository demonstrates the use of Nodoku libraries to build static pages using the Markdown files as content provider, and Yaml files as visual representation configuration.
Here are the steps to run the nodoku-demo-page project locally:
git clone https://github.com/epanikas/nodoku-demo-page.git
cd nodoku-demo-page
npm install
.env.local
as follows:DEFAULT_LNG=en
SIMPLELOCALIZE_API_KEY=<my api key>
SIMPLELOCALIZE_PROJECT_TOKEN=<my project token>
npm run dev
npm run build
npm start
Once this is done, and evrything went well, you can navigate in your browser to the url: http://localhos:3001 and you should see the site working
The project is organized around the Next JS page router, based on the file structure, starting from the src/app
folder.
Each page is nothing but ac parameterized call to the Nodoku RenderingPage JSX component, which is the main entry point for the Nodoku library.
As follows (see for example src/app/[lng]/page.tsx
):
export default async function Home({params}: { params: Promise<{ lng: string }> }): Promise<JSX.Element> {
const {lng} = await params;
const skin: NdPageSkin = parseYamlContentAsSkin(fs.readFileSync("./public/site/nodoku-landing.yaml").toString());
const content: NdContentBlock[] = parseMarkdownAsContent(fs.readFileSync("./public/site/nodoku-landing.md").toString(), "en", "nodoku-landing")
return <RenderingPage
lng={lng}
renderingPriority={RenderingPriority.skin_first}
skin={skin}
content={content}
componentResolver={nodokuComponentResolver}
imageProvider={commonImageProvider}
htmlSanitizer={commonHtmlSanitizer}
i18nextProvider={NodokuI18n.i18nForNodoku(i18nStore)}
i18nextPostProcessor={NodokuIcons.iconTextPostProcessorFactory(nameToIconConverters)}
clientSideComponentProvider={undefined}
/>;
}
As one can see, the content for this page is retrieved from the ./public/site/nodoku-landing.md
and the skin - from ./public/site/nodoku-landing.yaml
This is a common pattern in this project when the structure of content and skin files follow the structure of the pages in src/app
router.
Here is a brief description of the parameters for RenderingPage (for more detailed explanation please refer to https://github.com/nodoku/nodoku-core)
{icon:nd-react-icons/fa6:FaGithub}
to an actual icon representationflowbite/nav-header
accepts two client side components, that can be displayed on the navbar:
language-switcher
and
user-account
The current project uses a cloud based localization solution, provided by Simplelocalize.
In order for this project to work one would need to provide the Simplelocalize PROJECT TOKEN and Simplelocalize API KEY in the file .env.local
.
The translation engine is specified as follows:
import {i18nStore} from "@/app/components/nodoku-server-i18n-config";
...
<RenderingPage
...
i18nextProvider={NodokuI18n.i18nForNodoku(i18nStore)}
...
/>
The store for the localization is defined in the src/app/components/nodoku-server-i18n-config.ts
as follows:
const client: SimplelocalizeBackendApiClientImpl =
new SimplelocalizeBackendApiClientImpl(apiKey, projectToken, translationFetchMode);
const missingKeyStorage: MissingKeyStorage =
new SimplelocalizeMissingKeyStorage(client, () => Promise.resolve(), onMissingKeyStrategy, onFallbackLngTextUpdateStrategy);
export const i18nStore: NodokuI18n.I18nStore = await NodokuI18n.initI18nStore(
"all", // all languages to be fetched
[ // the namespaces to be used
"nodoku-landing",
"getting-started",
"showcase",
"footer",
"nav-header"
],
'en', // the default language (fallback language), used to detect the changes in the source text
translationFetchMode, // should we use CDN or API to fetch the translation from the cloud
saveMissing, // should the missing keys be saved
loadOnInit, // whether the translations should be loaded immediately
client, // the actual translation client to be used, in our case SimplelocalizeBackendApiClientImpl
missingKeyStorage // where the missing keys should be stored
);
Nodoku is a library that heavily relies on the Tailwind CSS framework for styling and visual appearance.
The following tailwind.config.ts
should be used:
import type {Config} from "tailwindcss";
import * as typo from '@tailwindcss/typography';
import {NodokuFlowbiteTailwind} from "nodoku-flowbite/tailwind";
import {NodokuMambaUiTailwind} from "nodoku-mambaui/tailwind";
import {NodokuCoreTailwind} from "nodoku-core/tailwind";
import {NodokuComponentsTailwind} from "nodoku-components/tailwind";
const config: Config = {
darkMode: 'class', // specifies that the dark mode button is activated via the class parameter. This is the default mode for the Flowbite theme switcher button
content: [
"./src/**/*.ts", // includin tailwind classes found in the code
"./src/**/*.tsx",
"./src/**/*.js",
"./src/**/*.jsx",
"./schemas/**/*.yml", // including tailwind classes for the default component themes
"./public/**/*.html",
"./src/**/*.{html,js}",
"./public/site/**/*.yaml", // including tailwind classes found in the page skin Yaml files
]
.concat(NodokuCoreTailwind.tailwindConfig()) // add the required config for components from nodoku-core
.concat(NodokuComponentsTailwind.tailwindConfig()) // add the required config for components from nodoku-components
.concat(NodokuFlowbiteTailwind.tailwindConfig()) // add the required config for components from nodoku-flowbite
.concat(NodokuMambaUiTailwind.tailwindConfig()), // add the required config for components from nodoku-mambaui
theme: {
extend: {
typography: {
DEFAULT: {
css: {
maxWidth: 'unset', // removing default width for Typography Tailwind plugin
}
}
}
},
},
plugins: [
typo.default(), // make sure to include the Typography Tailwind plugin, since the component core/typography relies on that
],
};
export default config;
Note, that when using Nodoku, the correct Tailwind configuration is crucial for the correct work.
Each Nodoku component bundle, such as nodoku-flowbite and nodoku-mambaui are shipped with the Tailwind config to be included in the Tailwind config of the project.
For example,
import {NodokuMambaUiTailwind} from "nodoku-mambaui/tailwind";
...
const twMambaUi = NodokuMambaUiTailwind.tailwindConfig()
const allContent = [standard project configuration].concat(twMambaUi)
const config: Config = {
content: allContent
};
export default config;
Nodoku framework allows using icons by specifying them directly in the text, or as images.
The component nodoku-icons provides the necessary mechanism to enable icons
An icon can be specified directly in the text content using the following format {icon:<name of the icons bundle>/<name of the icons set>:<name of the icon>}
:
For example:
{icon:nd-react-icons/fa6:FaAt}
This text pattern is postprocessed by the specifed text postprocessor, specified as a parameter to the RenderingPage component:
import {NodokuIcons} from "nodoku-icons";
import {nameToIconConverters} from "@/app/components/common-provider";
<RenderingPage
...
i18nextPostProcessor={NodokuIcons.iconTextPostProcessorFactory(nameToIconConverters)}
...
/>;
In this project, see src/app/components/common-provider.tsx
for details
An icon can be encoded in the image url as follows:

One would use the same icon notation, as in case of text, with the only exception being that instead of curly braces - {}, the brackets are used - ().
The RenderingPage parameter imageProvider should be configured for this url notation to work.
See the implementation of the function commonImageProvider in the file src/app/components/common-provider.tsx
Note that you can control the icon size in the Yaml skin file using the parameter imageStyle::imageWidth
as follows:
rows:
- row:
components:
- flowbite/horizontal-card:
theme:
imageStyle:
# controls the icon size
imageWidth: '150'
selector:
attributes:
sectionName: example
See the skin file public/site/nodoku-landing.yaml
for more details
This site uses the following components from third party creative contributions:
Font Awesome 6 - https://fontawesome.com/ License: CC BY 4.0 License https://creativecommons.org/licenses/by/4.0/
Heroicons - https://github.com/tailwindlabs/heroicons License: MIT https://opensource.org/licenses/MIT
Material Design icons - http://google.github.io/material-design-icons/ License: Apache License Version 2.0 https://github.com/google/material-design-icons/blob/master/LICENSE