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

rollup-plugin-handlebars-compiler

v1.2.1

Published

Rollup plugin to compile Handlebars templates to JavaScript with partials, helpers, assets, and Handlebars compile options.

Downloads

24

Readme

rollup-plugin-handlebars-compiler

A Rollup plugin to compile Handlebars templates to JavaScript with partials, helpers, assets, and Handlebars compile options.

This plugin supports:

  • Handlebars Compile options
  • Passing Template Data
  • Partials - through plugin options or by path
  • Helpers - through plugin options or by path
  • Assets - resolving and emitting assets to the output directory

Table of Contents

  1. Installation
  2. Usage
  3. Plugin Options
  4. Details
  5. Change Log
  6. Attribution
  7. License

Installation

npm i rollup-plugin-handlebars-compiler

This plugin and this plugin's version of Handlebars runtime is dependent on NodeJS modules. To use:

npm i @rollup/plugin-node-resolve @rollup/plugin-commonjs

Usage

Rollup Configuration

// rollup.config.js
import nodeResolve from '@rollup/plugin-node-resolve'
import commonJS from '@rollup/plugin-commonjs';
import handlebarsCompiler from 'rollup-plugin-handlebars-compiler'

export default {
    ...
    plugins: [
        nodeResolve(),
        commonJS(),
        handlebarsCompiler()
    ]
}

Example

{{! src/page.hbs }}
<h1>
    {{Title}}
</h1>
{{> ./partials/content }}
{{! src/partials/content.hbs }}
<p>{{Body}}</p>
// src/js/main.js
import PageTemplate from '../views/page.hbs';
document.body.append(PageTemplate({
    Title: 'Hello world!'
    Body: 'Lorem ipsum sit dolor amet'
}));

Output:

<h1>
    Hello World
</h1>
<p>Lorem ipsum sit dolor amet</p>

Plugin Options

In addition to the Handlebars Compiler plugin options, any compile option can also be passed.

See Handlebars Compile Options

export default {
    ...
    plugins: [
        ...
        handlebarsCompiler({

            /* Rollup Plugin Handlebars Compiler Options */

            partials, // An object with one or more items of `partialName: partialHandlebarsTemplate`
            helpers, // An object with one or more items of `helperName: function`
            templateData, // An object that passes data to the Handlebars template
            rootDir, // A string with the absolute path of the root directory of assets and partials (defaults to `process.cwd()`)
            assets: {
                resolve, // set to true to resolve asset paths in Handlebars templates (`true` by default)
                emit, // set to true to emit assets to output directory (`true` by default)
                contextPath, // A string to remove this sub-path when resolving assets (relative to `rootDir`)
                outputDir, // A string to set the output directory of all assets
                external, // A string or array of string glob patterns (relative to `rootDir`)
            }


            /* Handlebars compile options */

            data, // Set to false to disable @data tracking.
            compat, // Set to true to enable recursive field lookup.
            knownHelpers, // Hash containing list of helpers that are known to exist (truthy) at template execution time. Passing this allows the compiler to optimize a number of cases. Builtin helpers are automatically included in this list and may be omitted by setting that value to false.
            knownHelpersOnly, // Set to true to allow further optimizations based on the known helpers list.
            noEscape, // Set to true to not HTML escape any content.
            strict, // Run in strict mode. In this mode, templates will throw rather than silently ignore missing fields. This has the side effect of disabling inverse operations such as {{^foo}}{{/foo}} unless fields are explicitly included in the source object.
            assumeObjects, // Removes object existence checks when traversing paths. This is a subset of strict mode that generates optimized templates when the data inputs are known to be safe.
            preventIndent, // By default, an indented partial-call causes the output of the whole partial being indented by the same amount. This can lead to unexpected behavior when the partial writes pre-tags. Setting this option to true will disable the auto-indent feature.
            ignoreStandalone, // Disables standalone tag removal when set to true. When set, blocks and partials that are on their own line will not remove the whitespace on that line.
            explicitPartialContext, // Disables implicit context for partials. When enabled, partials that are not passed a context value will execute against an empty object.
        })
    ]
}

Details

Data

To pass data to templates, you can either:

  1. Pass data globally to all templates through the templateData object to the plugin options.
  2. Pass data as a parameter to the JS template function.

Pass data through plugin options

To pass global data accessible by all templates, simply pass an object to templateData in the plugin options.

// rollup.config.js
export default {
    plugins: [
        ...
        handlebarsCompiler({
            templateData: {
                firstname: 'John',
                lastname: 'Doe'
            },
        })
    ]
}

Pass data through the template function

To pass data specific to the template function, simply pass an object to the first parameter.

// src/js/main.js
import PageTemplate from '../views/page.hbs';
document.body.append(PageTemplate({
    firstname: 'Jane',
    lastname: 'Doe'
}));

Both the data from the plugin options and the template function parameter will be accessible in the template. If the data in the template function parameter has the same key as the global data from the plugin options, the data from the template function will override that of the plugin options.

{{! src/page.hbs }}
{{! templateData has `firstname: 'John'` and `lastname: 'Doe'` }}
{{! function data parameter has `firstname: 'Jane'` and `lastname: 'Doe'` }}
{{firstname}} {{lastname}}

Output:

Jane Doe

Partials

To pass partials, you can either:

  1. Pass partials to the partials object to the plugin options.
  2. Use the relative path in the handlebars template/partial.

Register partials through plugin options

To register through plugin options, simply pass the partial content with the partial name as the ID.

// rollup.config.js
export default {
    plugins: [
        ...
        handlebarsCompiler({
            partials: {
                titlePartial: fs.readFileSync(path.join(__dirname, './src/title-partial.hbs')).toString()
            },
        })
    ]
}
{{! src/title-partial.hbs }}
<h1>{{Title}}</h1>

Output:

<h1>
    Hello World
</h1>

Reference partials by relative path

The plugin resolves partials and helpers automatically. They are looked up relative to the current directory.

{{! src/partial/file.hbs }}
{{> template }} will reference "src/partials/template.hbs" if this file exists.
{{> ../template }} will reference "src/template.hbs" if this file exists.
{{> /src/template }} will reference "src/template.hbs" if this file exists.

The plugin will automatically resolve any nested partials. However, if a relative partial path's name collides with any partial name passed through the plugin options, the partial passed through the plugin options takes precedence.

For example:

// rollup.config.js
export default {
    plugins: [
        ...
        handlebarsCompiler({
            partials: {
                'partials/template': `<h1>I am from the plugin options</h1>`
            },
        })
    ]
}
{{! src/page.hbs }}
{{> ./partials/template }}
{{! src/partials/template.hbs }}
<h1>I am from the partial file</h1>

Output:

<h1>I am from the plugin options</h1>

Helpers

To pass helpers, simply pass a helpers object to the plugin options.

Register helpers through plugin options

// src/helpers.js
export function fullname(firstName, lastName) {
    return `${firstName} ${lastName}`
}

export function list(items, options) {
  const itemsAsHtml = items.map(item => "<li>" + options.fn(item) + "</li>");
  return "<ul>\n" + itemsAsHtml.join("\n") + "\n</ul>";
}
// rollup.config.js
export default {
    import { fullname, list } from 'src/helpers.js'

    ...
    plugins: [
        ...
        handlebarsCompiler({
            helpers: {
                fullname,
                list,
            },
        })
    ]
}

Reference helpers by relative path

The plugin resolves partials and helpers automatically. They are looked up relative to the current directory.

{{! src/template.hbs }}
{{helper}} will reference the helper "src/helper.js" if this file exists.
{{../helper}} will reference "helper.js" if this file exists.
{{./nested/helper}} will reference the helper "src/nested/helper.js" if this file exists.
{{[nested/helper] 'helper parameter'}} will reference the helper "src/nested/helper.js" if this file exists, passes 'helper parameter' as first parameter to helper.

Assets

By default, the Handlebars Compiler plugin will scan and resolve any assets referenced in HTML tags and emit them in the Rollup output directory (output.dir option in the Rollup config).

{{! src/partials/template.hbs }}
<img src="../images/logo.png">

Output:

<!-- resolves to relative root path (relative to cwd by default) -->
<!-- With `output.dir` set to `dist`, outputs file as `dist/src/images/logo.png` -->
<img src="/src/images/logo.png">

To disable emitting files to the output directory, set assets.emit to false.

// rollup.config.js
export default {
    plugins: [
        ...
        handlebarsCompiler({
            assets: {
                emit: false // disables emitting files
            }
        })
    ]
}

In addition, if assets.emit is enabled, asset paths will always be resolved according to rootDir and assets.contextPath. To disable resolving, set assets.resolve to false.

// rollup.config.js
export default {
    plugins: [
        ...
        handlebarsCompiler({
            assets: {
                emit: false, // No assets are emitted to the output directory
                resolve: true // Asset paths 
            }
        })
    ]
}
// rollup.config.js
export default {
    plugins: [
        ...
        handlebarsCompiler({
            assets: {
                emit: false, // No assets are emitted to the output directory
                resolve: false // Asset paths keep their original paths
            }
        })
    ]
}

Because assets need to be resolved to reference emitting files, if assets.resolve is set to false, no files will be emitted.

// rollup.config.js
export default {
    plugins: [
        ...
        handlebarsCompiler({
            assets: {
                emit: true,
                resolve: false // No assets are emitted to the output directory, and asset paths keep their original paths
            }
        })
    ]
}

It may be desired to output assets to a specific directory in the output.dir. To do so, a directory can be set in assets.outputDir (relative to output.dir).

// rollup.config.js
export default {
    plugins: [
        ...
        handlebarsCompiler({
            assets: {
                outputDir: 'assets', // If `output.dir` is set to `dist`, all assets are output to `dist/assets`
            }
        })
    ]
}

It may also be desired to use root relative paths for assets relative to the project root but resolve it to something else with assets.contextPath to define the context path of the asset paths in the project folder.

// rollup.config.js
export default {
    plugins: [
        ...
        handlebarsCompiler({
            assets: {
                contextPath: 'src',
                // e.g. `/src/images/logo.png` will be resolved and emitted to the output directory as `/images/logo.png`
            }
        })
    ]
}

Below is an example of using both assets.contextPath and assets.outputDir to customize the assets' paths in the output directory (with dist as the Rollup output.dir).

// rollup.config.js
export default {
    plugins: [
        ...
        handlebarsCompiler({
            assets: {
                outputDir: 'assets',
                contextPath: 'src',
                // `/src/images/logo.png`   - Original src reference
                // `/assets/logo.png`       - Resolved src reference
                // `dist/assets/logo.png`   - Emitted file
            }
        })
    ]
}

You can specify assets to treat as externals by providing glob patterns in the assets.external option. Doing so will skip resolving and emitting any matching files.

// rollup.config.js
export default {
    plugins: [
        ...
        handlebarsCompiler({
            assets: {
                emit: true,
                resolve: true,
                external: 'src/**/*.png',
                // `./src/images/logo.png` will not be resolved and emitted
                // `./src/images/picture.jpeg` will be resolved and emitted to the output directory as `/src/images/picture.jpeg`
            }
        })
    ]
}

Change Log

See the CHANGELOG.md file.

Attribution

This project is heavily inspired by pcardune/handlebars-loader for Webpack and a rollup plugin code snippet to resolve Handlebars partial paths ba55ie/rollup-plugin-handlebars-nested.js.

License

Mozilla Public License 2.0 (https://opensource.org/license/mpl-2-0nse)