npm package discovery and stats viewer.

Discover Tips

  • General search

    [free text search, go nuts!]

  • Package details

    pkg:[package-name]

  • User packages

    @[username]

Sponsor

Optimize Toolset

I’ve always been into building performant and accessible sites, but lately I’ve been taking it extremely seriously. So much so that I’ve been building a tool to help me optimize and monitor the sites that I build to make sure that I’m making an attempt to offer the best experience to those who visit them. If you’re into performant, accessible and SEO friendly sites, you might like it too! You can check it out at Optimize Toolset.

About

Hi, 👋, I’m Ryan Hefner  and I built this site for me, and you! The goal of this site was to provide an easy way for me to check the stats on my npm packages, both for prioritizing issues and updates, and to give me a little kick in the pants to keep up on stuff.

As I was building it, I realized that I was actually using the tool to build the tool, and figured I might as well put this out there and hopefully others will find it to be a fast and useful way to search and browse npm packages as I have.

If you’re interested in other things I’m working on, follow me on Twitter or check out the open source projects I’ve been publishing on GitHub.

I am also working on a Twitter bot for this site to tweet the most popular, newest, random packages from npm. Please follow that account now and it will start sending out packages soon–ish.

Open Software & Tools

This site wouldn’t be possible without the immense generosity and tireless efforts from the people who make contributions to the world and share their work via open source initiatives. Thank you 🙏

© 2024 – Pkg Stats / Ryan Hefner

nodoku-core

v0.1.5

Published

basic foundation for nodoku static site generator

Downloads

227

Readme

GitHub link: https://github.com/nodoku/nodoku-core

Nodoku is a static site generator, where the content is provided via MD (markdown) files, and the visual representation is controlled using Yaml files - called skin.

Nodoku promotes the content-first approach, when the content creation process is not being distracted by the considerations related to the visual representation and design.

Instead, the content is created first as an MD file, demarcated by the special markers into content blocks, and then each block is rendered using the configuration provided in a Yaml file called skin.

Figure 1 shows a screenshot of a part of a landing page created with Nodoku.

Nodoku is a set of libraries, the most important of which is nodoku-core, intended to be used with the NextJS framework for generation of static sites.

Nodoku is intended to be used in server side rendering, it is not suitable for client side.

nodoku-core doesn't contain the visual representation for content blocks. Instead, the visual representation is supplied via separated dependencies, such as nodoku-flowbite (components based on Flowbite) and nodoku-mambaui (components based on MambaUI).

More set of components can be added, and included in the project as required.

The structure of the skin files is organized by rows, each row having one or more components.

The mapping between visual representation and the content block is performed using the concept of selector. Selector is a set of matching conditions, attached to a visual component in the skin Yaml file.

The actual rendering is performed in two steps:

  • first, for a given visual component a set of matching content blocks is determined, using the selector and the meta-data of the content block

  • second, this flow of content blocks is provided to the rendering mechanism of the visual component for actual rendering.

This decoupling allows for great level of flexibility and reuse between the content and the visual representation.

If the visual components support Tailwind, the Tailwind customization can be provided in the skin Yaml file to fine tune the visual representation.

You can learn more about the rationale behind Nodoku and the main principles in the blog article: Nodoku: a lo-code static site generator, promoting a content-first approach based on Markdown files

Nodoku foundation

Nodoku is organized around two flows of data:

  • the content flow (supplied via a Markdown file)
  • the visual representation flow (supplied via Yaml file called skin)

The Nodoku engine will take care of parsing these files and supply them to the designated visual components for rendering.

Nodoku content flow

As has been mentioned above, the Nodoku content flow is supplied via an MD file.

The content flow in Nodoku is organized around an entity called content block, which is a single piece of content suitable for rendering by a Nodoku visual component. The set of such components constitute the Nodoku content flow.

The content block has a predefined structure. It is designed to be more or less generic, and visual representation agnostic. In particular this is important to implement the content-first principle, where the content can (and should!) be created first, without any visual design considerations.

Nodoku engine parses the MD file to extract the set of content blocks, that are contained in such file. The content blocks are delimited by a special markers - the content block delimiters.

The content block delimiter is a small piece of Yaml code snippet, embedded directly into the MD file.

For example, the content, that has been used to generate the screeshnot on Figure 1 is the following:


```yaml
nd-block:
  attributes:
    sectionName: nodoku-way
``

# Step 1: _Think_
## Create content promoting your product or service as an **MD file**
...
Concentrate on the subject of your product / service to highlight its advantages.
...
|Get started|

# Step 2: _Skin_
## Skin the MD file using simple **Yaml config** and available components
...
Once you are happy with the message your landing page conveys,
start by skinning it up.
...
|Get started|

# Step 3: _Fine tune_
## Use configuration options to fine tune your landing page presentation
...
If the default presentation doesn't suit your needs, you can tweak it up
using the config options of each component to fine tune it for your needs.
...
|Get started|

The first thing one would notice in this MD excerpt is the small Yaml code snippet.

nd-block:
  attributes:
    sectionName: nodoku-way

This Yaml code snippet is a content block delimiter, and it contains the content block meta-data, such as id, attributes and tags.

The schema for this Yaml code snippet can be found in the Json schema file : nodoku-core/schemas/md-content-block-delimiter.json

The content block has the following predefined, fixed structure:

class NdContentBlock {
    // content block meta-data
    id: string;
    lng: string;
    attributes: {key: string, value: string}[] = [];
    tags: string[] = []
    namespace: string;
    
    // the actual textual content
    title?: NdTranslatedText;       // a title
    subTitle?: NdTranslatedText;    // a subtitle
    h3?: NdTranslatedText;          // h3 header
    h4?: NdTranslatedText;          // h4 header
    h5?: NdTranslatedText;          // h5 header
    h6?: NdTranslatedText;          // h6 header
    footer?: NdTranslatedText;      // footer
    paragraphs: NdTranslatedText[]; // set of paragraphs
    bgImageUrl?: NdTranslatedText;  // background image 
    images: NdContentImage[] = [];  // set of images 
}

where NdTranslatedText represents a piece of text, that can be used for i18next translation (see below)

class NdTranslatedText {
    key: string;    // the translation key
    ns: string;     // the translation namespace
    text: string;   // the fallback text, extracted from content
}

As one can see, the content block cannot contain more than instance of each type of headers: title (h1), subtitle (h2), h3, etc.

But it can contain several headers of different types, for example, a title (h1) and subtitle (h2)

Hence, the parsing of the MD file goes as follows:

  • if the Yaml nd-block is encountered, a new content block is started, which would absord all the content elements that are discovered below, until a new content block is started
  • if a header (of any kind) is encountered
    • if a header of the same kind or below exists in the current block
      • finalize the current block
      • start a new block
      • and copy the metadata from the current block to the new one
    • otherwise
      • add the corresponding header to the current block

This approach allows for smooth content definition, where a common content block metadata is defined only once, and all the subsequent pieces of content are treated as new blocks with the same metadata.

Thanks to this parsing approach, the excerpt of the MD file, shown above, defines 3 content blocks - one per card shown on the screenshot of Figure 1 - instead of only one.

Nodoku skin

Nodoku skin is a Yaml file which configures the visual representation of the content defined in a markdown file.

The Nodoku page layout is organized as a set of rows, each row having its configured set of components.

Each layout row can have one or more visual components.

Here is an example of a typical Nodoku skin file:

rows:
  - row:
      components:
        - mambaui/card:
            selector:
              attributes:
                sectionName: nodoku-way

In this example, we define a row containing components of type mambaui/card

The selector defines the content blocks that should be rendered using this visual component. In our case, these are all the content blocks having an attribute sectionName equal to nodoku-way.

Recall, that according to our MD file we are actually having 3 content blocks matching this criteria:

  • Step 1: Think
  • Step 2: Skin
  • Step 3: Fine tune

Naturally, a single card component cannot display more than one content block.

Consequently, according to the skin Yaml file, the Nodoku engine will apply the same visual component definition to all the three matching content blocks.

And this process will end up rendering the screenshot presented on Figure 1.

It's worth noting that a visual component can support arbitrary number of content blocks. For example, the flowbite/carousel component can display any number of content blocks.

Consequently, if the skin Yaml file had the following configuration:

rows:
  - row:
      components:
        - flowbite/carousel:
            selector:
              attributes:
                sectionName: nodoku-way

it would have had the following visual rendering:

Getting started

Prerequisites

As has been mentioned above, Nodoku is a library intended to be used within the NextJS framework. The creation of a NextJS project is out of scope for the current documentation.

We assume that the user is already familiar with the following concepts:

  • NextJS:
    • Nodoku is written to be used in server side rendering in NextJS
  • Typescript:
    • Nodoku is written using Typescript (5.x), it uses the ECMAScript Modules (ESM) as module standard. It doesn't ship as CommonJS or UMD modules.
  • Tailwind CSS:
    • Nodoku heavily relies on Tailwind for visual appearance. All the Nodoku components are using Tailwind for their default visual configuration. And Tailwind can be used for skin customization.
  • React and JSX:
    • you need a basic knowledge of React and JSX as all the Nodoku visual compnents are async functions returning JSX (see the declaration of AsyncFunctionComponent).
  • Webpack config:
    • when using Nodoku with flowbite (see nodoku-flowbite) it might be required to modify the default Webpack configuration to make sure only one instance of flowbite is used

Installation

In order to use Nodoku one needs to install the nodoku-core library (this one) and at least one component library for Nodoku (for example, nodoku-flowbite)

npm install nodoku-core nodoku-flowbite

Integrating Nodoku into a project

The entry point of a Nodoku library is the RenderingPage component.

This component receives as properties the flow of content blocks and the skin, and renders accordingly.

Here is a typical example of the usage of the RenderingPage TSX component:


// load and parse the content MD file
const content: NdContentBlock[] = await contentMarkdownProvider("<url location of the content file>.md", "en", "nodoku-landing")

// load the Yaml skin file
const skin: NdPageSkin = await skinYamlProvider("<url location of the skin file>.yaml")

...

<RenderingPage
        lng={lng}
        renderingPriority={RenderingPriority.skin_first}
        skin={skin}
        content={content}
        componentResolver={defaultComponentResolver}
/>

Loding and parsing of the content MD file

First, we parse the content MD file, using the predefined function contentMarkdownProvider.

The implementation of this function is intentionally kept simple, so that it can easily be replaced by a custom provider, if for some reason the standard one cannot be used.

async function contentMarkdownProvider(
    mdFileUrl: string,
    contentLng: string, 
    ns: string): Promise<NdContentBlock[]> {
    
    return await fetch(mdFileUrl)
        .then(response => response.text())
        .then(fileContent => {
            return parseMarkdownAsContent(fileContent, contentLng, ns)
        })
}

The real parsing happens in the function parseMarkdownAsContent, which can also be used directly, if for some reason the URL of the MD file is not available.

The result of this phase is the set of content blocks, represented as an array of NdContentBlock items.

Each content block that is extracted from the content MD file has an id - called blcokId.

The content block id is an important concept as it serves as a prefix for the translation keys, generated for that content block.

The Nodoku content blocks are made available for translation and localization out of the box.

Hence, the unique translation keys given for each piece of textual information is important.

The contenb blockId can

  • either be provided directly as metadata in the delimiting Yaml code snippet
  • or, if not explicitly provided, it is generated automatically, using the metadata available

Explicitly providing content blockId

The content blockId can be provided directly in the Yaml code snippet, delimiting the content blocks:

nd-block:
  id: nodoku-way-think

As this:

```yaml
nd-block:
  id: nodoku-way-think
  attributes:
    sectionName: nodoku-way
``

# Step 1: _Think_
## Create content promoting your product or service as an **MD file**

This would generate the content block with blockId nodoku-way-think

and consequently the following translation keys will be generated for this block:

  • nodoku-way-think.title
  • nodoku-way-think.subTitle

Automatic generation of content blockId

It might be tedious to require the end user to explicitly provide a metadata, including blockId, for each and every content.

After all, Nodoku's philosophy is a content first approach, where the textual content creation should be as seamless and straightforward, as possible.

Hence, Nodoku parser will generate the blockId automatically, if not provided, using the sequential index of the content block.

Consider this example:

```yaml
nd-block:
  attributes:
    sectionName: nodoku-way
``

# Step 1: _Think_
## Create content promoting your product or service as an **MD file**

Loading of the skin Yaml file

Likewise, there is a readily available parser for Yaml files, used as Nodoku skin. Recall, that skin is a configuration for visual page rendering.

async function skinYamlProvider(yamlFileUrl: string): Promise<NdPageSkin> {
    return await fetch(yamlFileUrl)
        .then(response => response.text())
        .then(parseYamlContentAsSkin);
}

If, for some reason, the standard implementation is not suitable for particular needs, one can easily replace this function by another one, using parseYamlContentAsSkin as the actual parser.

The result of this parsing is an instance of NdPageSkin, which conveys the necessary information regarding the Nodoku skin.

Rendering Nodoku

Finally, when the skin and content are loaded, we can invoke the actual rendering using the provided component RenderingPage.

RenderingPage is an async JSX function, which is suitable for usage in the NextJS environment.

Let's have a closer look at its properties:

class RenderingPageProps {
    lng: string;
    content: NdContentBlock[];
    skin: NdPageSkin | undefined = undefined;
    renderingPriority: RenderingPriority = RenderingPriority.content_first;
    componentResolver: ComponentResolver | undefined = undefined;
    imageUrlProvider: ImageUrlProvider | undefined = undefined;
    i18nextProvider: I18nextProvider | undefined = undefined;
}
  • lng: the language in which the page should be rendered. Nodoku supports localization out of the box (see more on Nodoku localization in nodoku-i18n)

  • content: the content flow, represented as an array of NdContentBlock instances. The content blocks are usually parsed from a Markdown file using the provided parser contentMarkdownProvider.

  • skin: an instance of NdPageSkin class representing the Nodoku skin - configuration of the visual representation of the page. It is usually loaded from a Yaml file using the supplied loader skinYamlProvider.

  • renderingPriority: this parameter is an enum that can have two values:

    • content_first: the content is rendered as it appears in the markdown file, sequentially, block by block, from top to bottom. If a visual component is configured in the skin Yaml file, this visual component is used for rendering the content block. Otherwise, a default visual component is used

    • skin_first: the rendering is fully prescribed by the skin Yaml file. The components are rendered in the order they appear in the Yaml file If a content block is not matched by any of the visual components in the skin Yaml file, it is not rendered at all. If a content block matches more than one visual component, each visual component is rendered, and the same content block will appear several times on the page.

  • componentProvider: the function returning an actual implementation of the component, given its name, as specified in the skin. The signature is as follows:

    (componentName: string) => Promise<AsyncFunctionComponent>

    where AsyncFunctionComponent is the following function:

    (props: NdSkinComponentProps) => Promise<JSX.Element>

    the actual implementations, respecting the AsyncFunctionComponent signature, are usually supplied via the component bundles, such as nodoku-flowbite and nodoku-mambaui

  • imageUrlProvider: the function allowing to customize the image URL conversion for rendering. It may so happen that the URL of images appearing in the MD file are different from those appearing on the page. For example, we often use the relative notation for images in the MD file, whereas this is not suitable for page rendering. The conversion between URL's in the MD file, and the URL's on the page can be provided using this parameter. The default implementation strips the leading dots, thus naively converting a relative url to an absolute one. Like this (more sophisticated patterns can be implemented, if required):

    ../images/my-image-123.png will be converted to /images/my-image-123.png

  • i18nextProvider: this parameter can be used to provide the localization mechanism for Nodoku. This function is supposed to return an object containing the t() function, which will further be used for translating the text.

    type I18nextProvider = (lng: string) => Promise<{t: (text: NdTranslatedText) => string}>

    Note, that the provided t() function takes as an argument the whole NdTranslatedText structure, which contains the translation namespace, the key and the default value, extracted from the content.

    This parameter is optional. If not provided, the text from the content MD file is used.

    For more details see nodoku-i18n

Nodoku component resolver

The component resolver is expected to resolve each component name in a textual form, as extract from the skin file, to an actual component definition, among other things containing the actual Javascript function that should be called to render the component.

As the components are provided as external dependencies, Nodoku supplies a special script - nodoku-gen-component-resolver - that should be launched from command line to automatically generate the component resolver.

Here is the typical example of the generated component resolver, that is crucial for successful functioning of Nodoku:


import {AsyncFunctionComponent, DummyComp, NdComponentDefinition} from "nodoku-core";

import { NodokuFlowbite } from "nodoku-flowbite";

const components: Map<string, {compo: AsyncFunctionComponent, compoDef: NdComponentDefinition}> = 
        new Map<string, {compo: AsyncFunctionComponent, compoDef: NdComponentDefinition}>();

components.set("flowbite/card", {
    compo: NodokuFlowbite.Card, 
    compoDef: new NdComponentDefinition(1, 
            "./schemas/nodoku-flowbite/dist/schemas/components/card/default-theme.yml")
});

// other component definitions go here

export function nodokuComponentResolver(componentName: string): Promise<{compo: AsyncFunctionComponent, compoDef: NdComponentDefinition}> {
    const f: {compo: AsyncFunctionComponent, compoDef: NdComponentDefinition} | undefined = 
            components.get(componentName);
    return Promise.resolve(f ? f : {compo: DummyComp, compoDef: new NdComponentDefinition(1)});
}

Recall that the componentResolver is one of the properties of the RenderingPage component.

Thanks to this function Nodoku can find the correspondence between the component name specified in the skin Yaml file, and the actual component implementation.

It is worth noting here that nodoku-core is the Nodoku engine, which contains the resolving and parsing functionality, but it contains no visual components.

The visual components are supposed to be supplied as external library dependency, such as nodoku-flowbite and nodoku-mambaui.

The static map, that is generated by the scirpt nodoku-gen-component-resolver, associates with each component name two attributes:

  • compo: the function that is to be called to render the component
  • compoDef: the definition of the component. The component defintion is loaded from the Nodoku manifest, supplied in the bundle where this component is defined. The component definition includes the following fields:
    • numBlocks: the maximum number of blocks this component supports
    • defaultThemeYaml: the Yaml file defining the default component visual configuration (usually uses Tailwind class names)

These components parameters are defined in a special file - nodoku.manifest.json - which is supposed to be shipped within the component bundle. See below for more details.

Configuring Nodoku project

Tailwind configuration

Since Nodoku is heavily relying on Tailwind, the Nodoku project is supposed to be configured accordingly. Otherwise, the Tailwind classes won't be parsed, and the page would not be rendered correctly.

To understand how to configure Tailwind we first need to understand how Tailwind functions.

Tailwind is a set of predefined CSS classes, each having its own responsibility and value.

For example, mt-10 results in the following CSS being genrated by Tailwind:

.mt-10 {
  margin-top: 2.5rem;
}

Since there are a lot of those Tailwind classes, not all of them are included in the final bindle, but only those that are actually used in the project.

Hence, Tailwind needs to know what files to parse to look for actual Tailwind classes that are being used. Only the CSS classes discovered during this parsing, will be included in the final bundle.

In order for Nodoku to function properly, one needs to include all the places where potentially Tailwind classes can be encountered.

Here is the typical tailwind.config.ts

import flowbite from "flowbite-react/tailwind";
import type {Config} from "tailwindcss";
import * as typo from '@tailwindcss/typography';


const config: Config = {
    content: [
        "./src/**/*.ts",
        "./src/**/*.tsx",
        "./src/**/*.js",
        "./src/**/*.jsx",

        "./node_modules/nodoku-core/dist/esm/**/*.js",
        "./node_modules/nodoku-core/dist/esm/**/*.jsx",
        "./node_modules/nodoku-components/dist/esm/**/*.js",
        "./node_modules/nodoku-components/dist/esm/**/*.jsx",
        "./node_modules/nodoku-flowbite/dist/esm/**/*.js",
        "./node_modules/nodoku-flowbite/dist/esm/**/*.jsx",
        "./node_modules/nodoku-flowbite/dist/schemas/**/*.yml",
        "./node_modules/nodoku-mambaui/dist/esm/**/*.js",
        "./node_modules/nodoku-mambaui/dist/esm/**/*.jsx",
        "./node_modules/nodoku-mambaui/dist/schemas/**/*.yml",
        "./public/**/*.html",
        "./src/**/*.{html,js}",
        "./public/site/**/*.yaml",
        flowbite.content(),
    ],

    theme: {
        extend: {
            aspectRatio: {
                'carousel': '4 / 1.61',
            },
            typography: {
                DEFAULT: {
                    css: {
                        maxWidth: 'unset',
                    }
                }
            }
        },
    },
    plugins: [
        flowbite.plugin(),
        typo.default(),
    ],
};

export default config;

Let's have a closer look at this configuration:

  • content: should contain all the paths where Tailwind classes are located, and consequently will be parsed by Tailwind for class extraction.
    • for each Nodoku component bundle it should contain the all the .js, .jsx and .yml files, because it might contain the Tailwind classes
    • if a component bundle using flowbite-react is used (such as nodoku-flowbite), its plugin should be included in the content
  • theme: this is a standard Tailwind's theme definition. For example, in this example we are defining a new aspectRation value called carousel, which later can be used for styling Nodoku components
  • plugins: this section includes plugins needed for Nodoku components to work properly
    • flowbite.plugin() - if components based on flowbite-react are used, this plugin should be included
    • typo.default() - if the Nodoku Typography plugin is used, this plugin should be included

For more details on Tailwind config see the section "Configuring Tailwind" in one of the Nodoku component bundles - nodoku-flowbite.

Webpack configuration

It might so happen that we'll need to configure the Webpack in order for Nodoku to run properly.

In particular this problem becomes apparent if we are using flowbite-react, in order to ensure that only instance of flowbite-react is included in the final bundle.

Otherwise, the flowbite-react theme might not work properly, and all the customizations of this theme will not be applied.

Webpack configuration in a NextJS project is not trivial, as we don't have access directly to the file webpack.config.js.

Instead, we should be redefining the nextConfig in next.config.mjs as follows:

import path from "node:path";

const nextConfig = (phase, {defaultConfig}) => {
    /**
     * @type {import('next').NextConfig}
     */
    const nextConfig = {
        /* config options here */
        ...defaultConfig,
        webpack: ((config, opts) => {
            config.resolve.alias["flowbite-react"] = path.resolve('./node_modules/flowbite-react');
            config.resolve.alias["flowbite"] = path.resolve('./node_modules/flowbite');
            return config;
        })
    }

    return nextConfig
}

export default nextConfig;

Note that we pin down the resolution of flowbite and flowbite-react to the particular dependencies in our node_modules folder.

Nodoku component bundle

As has already been mentioned above, Nodoku visual components are supplied as a component bundle - a standard NPM dependency, that can be installed using the standard NPM installation process

npm install <nodoku component bundle>

for example:

npm install nodoku-flowbite

Nodoku manifest

Each component bundle should be shipped with a special file called nodoku.manifest.json.

This json file is a visit card of the component bundle: it contains all the information required for Nodoku engine to treat correctly the component and use it following the configuration in the Nodoku skin Yaml file.

Here is an excerpt from such file for nodoku-flowbite:

{
  "namespace": "NodokuFlowbite",
  "components": {
    ..
    "flowbite/carousel": {
      "schemaFile": "./dist/schemas/components/carousel/visual-schema.json",
      "optionsFile": "./dist/schemas/components/carousel/options-schema.json",
      "defaultThemeFile": "./dist/schemas/components/carousel/default-theme.yml",
      "implementation": "Carousel",
      "numBlocks": "unlimited"
    },
    ...
  }
}

Let's have a closer look at the attributes of each Nodoku visual component:

  • namespace: the Javascript namespace where out of which the component is being exported.

    The presence of the namespace attribute instructs the generator of the component resolver (nodoku-gen-component-resolver) to insert the component definition prefixed with the namespace.

    import { NodokuFlowbite } from "nodoku-flowbite";
    components.set("flowbite/carousel", {compo: NodokuFlowbite.Carousel, ...});

    The Nodoku manifest cannot have more than one namespace.

  • components: is a object associating a textual component name with the data structure that is used to extract the component definition. This data structure include:

  • schemaFile: is a Json schema file representing the data structure of the visual theme of the component (see Schema for Nodoku skin Yaml file)

  • optionsFile: is a Json schema file representing the data structure of the functional options of the component (see Schema for Nodoku skin Yaml file)

  • defaultThemeFile: a Yaml file containing the default Tailwind configuration of the component

  • implementation: the actual Javascript function rendering this compnent. (the signature should correspond to AsyncFunctionComponent)

  • numBlocks: number of content blocks this component can accept for rendering. This parameter can either be a positive number of the predefined value 'unlimited'

The numBlocks parameter is used by the Nodoku engine to calculate the correct number of content blocks this visual component is capable of rendering. This parameter is normally either 1 or 'unlimited'.

The defaultThemeYaml attribute is an important parameter designating the Yaml file where the default Tailwind configuration is located. Normally each Nodoku visual component comes as an empty JSX scaffolding, which delegates the styling - the actual Tailwind classes - to the external data structure, the component default theme.

Nodoku skin

The Nodoku skin is a Yaml file which configures the Nodoku rendering.

Recall, that in Nodoku there is a strict separation between the content and the visual representation.

The content is provided via an MD file, whereas the mapping between the content blocks and visual components are defined in the skin.

The skin also defines the necessary customizations and fine-tuning of the page visual representation, if required.

Nodoku visual component theme

Normally, each JSX block in a visual component is styled using the Tailwind class names. This styling is typically composed of two parts: base and decoration.

Consider the following example:

<h3 className={`${effectiveTheme.titleStyle?.base} ${effectiveTheme.titleStyle?.decoration}`}>
  {t(block.title)}
</h3>

<h4 className={`${effectiveTheme.subTitleStyle?.base} ${effectiveTheme.subTitleStyle?.decoration}`}>
  {t(block.subTitle)}
</h4>

The component theme is supplied to the component by the Nodoku engine, and comes in as a data structure containing two elements:

  • one for title styling effectiveTheme.titleStyle
  • and one for subtitle styling effectiveTheme.subTitleStyle

Each such element is called ThemeStyle, and is composed of two string attributes: base and decoration

The component theme consists of several attributes of type ThemeStyle.

Here is an excerpt from the component theme flowbite/card

export class CardTheme {
    ...
    titleStyle?: ThemeStyle;
    subTitleStyle?: ThemeStyle;
    ...
}

where ThemeStyle is defined as follows:

export class ThemeStyle {
    base: string;
    decoration: string;
}

For each component there is a default theme supplied in a Yaml file, and this file is defined for the given visual component in the Nodoku manifest.

In addition, the theme of each component can further be customized in the skin Yaml file.

The Nodoku skin Yaml file

The skin Yaml file has a predefined structure, which can be summarized as follows:

global:
  renderingPage:
    # styles applied to the whole page
  theme:
    # attributes defined on the global level for all the component themes
  themes:
    # contains the list of theme customizations, where each theme from the list is applied sequentially on each content block rendered by the component of a given type in a single row
  components:
    <component-name>:
      theme:
        # defines a theme customization for a particular component
      themes:
        # defines a list of theme customizations for a particular component, where each theme from the list is applied sequentially on each content block rendered by the component of this type in a single row

rows:
  # list of rows
  - row:
      theme:
        # styling classes applied on the row level
      maxCols:
        # optional: a limit on a number of visual components in a row. By default, the max number of elements in a row is 12 (grid-cols-12 from Tailwind) but it can be reduced to 1, when we need content blocks to be located one beneat the other
      components:
        <component-name>:
          selector:
            # the data structure defining the selector to be applied on content blocks to select the matching content blocks that would be rendered by this component
          options:
            # a component may accept an options object, which can be defined here 
          theme:
            # the customization of the component's scheme
          themes:
            # a list of theme customizations for the given component, where each theme from the list is applied sequentially on each content block rendered by the component of this type in a single row

The customization is applied progressively, from the least specific theme to the most specific, starting from the default theme of the component.

Since the Nodoku skin file is large, and can contain many customization options, it is desirable to have a schema file that would guide the user through the process of defining and editing of the skin.

Schema for Nodoku skin Yaml file

In order to make the Nodoku skin creation process more convenient and predictable, Nodoku provides a possibility to create a schema file for a project, with a given set of Nodoku component bundles, defined in NPM dependencies (in package.json).

Similar to the generation script for Nodoku component resolver, there is a script for generation of the Nodoku skin schema: nodoku-gen-skin-schema

One can run this script to automatically generate a Nodoku skin schema file.

The schema file by default is generated in the folder

/schemas

After the schema generation script completes, the ./schemas folder should look as follows:

The generated schema file - ./schemas/visual-schema.json - can be applied to the skin Yaml file in several ways:

  • use the IDE mechanism of schema application

  • use the yaml-language-server header (the first row) for the Yaml file as follows:

    # yaml-language-server: $schema=../../../schemas/visual-schema.json
      
    global:
      renderingPage:
        base: dark:text-gray-200 text-gray-700
    
    rows:
    - row:
        theme:
          decoration: mb-10
    

Customizing Nodoku page appearance

The main principle of component customization in Nodoku consists of cascading application of themes, defined on different levels:

  • global level: the section global in the Yaml file. This level can further be divided to
    • theme level: where any attribute for any component can be defined
    • component level: where attributes for a given component can be defined
  • rows level: where the list of rows defines each its set of components
    • component in a row level: the most specific customization, where a particular component can be customized

The process of customization starts from the component's default theme and works its way through all the defined customizations, until the most specific level is reached.

Nodoku generation scripts

The nodoku-core provides the following scripts, that are used to generate component resolver and visual schema, by scanning the node_modules folder of the project

  • nodoku-gen-component-resolver: generates the component resolver by scanning node_modules and searching for nodoku component libraries - the libraries providing the nodoku.manifest.json file. For more details see Nodoku component resolver

  • nodoku-gen-visual-schema: generates the json schema file thate can be used to validate the Nodoku skin schema file. For more details see Schema for Nodoku skin Yaml file

To simplify the use of these script it is recommended to add them in the project's package.json file as follows:

{
  ...
  "scripts": {
    "gen-component-resolver": "nodoku-gen-component-resolver",
    "gen-skin-schema": "nodoku-gen-skin-schema"
  },
  ...          
}