#Rollup Plugin HTML

Plugin for generating HTML files from rollup.

  • Generate one or more HTML pages from a rollup build
  • Inject rollup bundle into an HTML page
  • Optionally use HTML as rollup input, bundling any module scripts inside
  • Minify HTML and inline JS and CSS
  • Suitable for single page and multi-page apps

Jump to:

#Installation

npm i -D @open-wc/rollup-plugin-html

#Examples

#Simple HTML page

When used without any options, the plugin will inject your rollup bundle into a basic HTML page. Useful for developing a simple application.

Show example
import html from '@open-wc/rollup-plugin-html';
export default {
  input: './my-app.js',
  output: { dir: 'dist' },
  plugins: [html()],
};

#Input from file

During development, you will probably already have an HTML file which imports your application's modules. You can use this same file as the input of the html plugin, which will bundle any modules inside and output the same HTML minified optimized.

To do this, you can set the html file as input for rollup:

Show example
import html from '@open-wc/rollup-plugin-html';
export default {
  input: 'index.html',
  output: { dir: 'dist' },
  plugins: [html()],
};

You can also set the inputPath property on the html plugin. This is useful if you are generating multiple html files, which each have their own entrypoints:

Show example
import html from '@open-wc/rollup-plugin-html';
export default {
  output: { dir: 'dist' },
  plugins: [
    html({
      inputPath: 'index.html',
    }),
    html({
      inputPath: 'another-index.html',
    }),
  ],
};

#Input from string

Sometimes the HTML you want to use as input is not available on the file system. With the inputHtml option you can provide the HTML as a string directly. This is useful for example when using rollup from javascript directly.

Show example
import html from '@open-wc/rollup-plugin-html';
export default {
  output: { dir: 'dist' },
  plugins: [
    html({
      inputHtml: '<html><script type="module" src="./app.js"></script></html>',
    }),
  ],
};

#Template

With the template option, you can let the plugin know where to inject the rollup build into. This option can be a string or an (async) function which returns a string.

Show example

Template as a string:

import html from '@open-wc/rollup-plugin-html';
export default {
  output: { dir: 'dist' },
  plugins: [
    html({
      template: `
      <html>
        <head><title>My app</title></head>
        <body></body>
      </html>`,
    }),
  ],
};

Template as a function:

import fs from 'fs';
import html from '@open-wc/rollup-plugin-html';

export default {
  output: { dir: 'dist' },
  plugins: [
    html({
      template() {
        return new Promise((resolve) => {
          const indexPath = path.join(__dirname, 'index.html');
          fs.readFile(indexPath, 'utf-8', (err, data) => {
            resolve(data);
          });
        });
      }
    }
  ],
};

#Multiple HTML pages

With this plugin, you can generate as many HTML pages as you want. Rollup will efficiently create shared chunks between pages, allowing you to serve from cache between navigations.

View example

The easiest way is to have the HTML files with module scripts on disk, for each one you can create an instance of the plugin which will bundle the different entry points automatically share common code.

By default, the output filename is taken from the input filename. If you want to create a specific directory structure you need to provide an explicit name:

import html from '@open-wc/rollup-plugin-html';

export default {
  output: { dir: 'dist' },
  plugins: [
    html({
      inputPath: './home.html',
    }),
    html({
      inputPath: './about.html',
    }),
    html({
      name: 'articles/a.html',
      inputPath: './articles/a.html',
    }),
    html({
      name: 'articles/b.html',
      inputPath: './articles/b.html',
    }),
    html({
      name: 'articles/c.html',
      inputPath: './articles/c.html',
    }),
  ],
};

#Manually inject build output

If you want to control how the build output is injected on the page, disable the inject option and use the arguments provided to the template function.

Show example

With a regular template function:

import html from '@open-wc/rollup-plugin-html';
export default {
  input: './app.js',
  output: { dir: 'dist' },
  plugins: [
    html({
      name: 'index.html',
      inject: false,
      template({ bundle }) {
        return `
        <html>
          <head>
            ${bundle.entrypoints.map(bundle => e =>
              `<script type="module" src="${e.importPath}"></script>`,
            )}
          </head>
        </html>
      `;
      },
    }),
  ],
};

When one of the input options is used, the input html is available in the template function. You can use this to inject the bundle into your existing HTML page:

import html from '@open-wc/rollup-plugin-html';
export default {
  input: './app.js',
  output: { dir: 'dist' },
  plugins: [
    html({
      inputPath: './index.html',
      inject: false,
      template({ inputHtml, bundle }) {
        return inputHtml.replace(
          '</body>',
          `<script type="module" src="${bundle[0].entrypoints[0].importPath}"></script></body>`,
        );
      },
    }),
  ],
};

#Transform output HTML

You can use the transform option to manipulate the output HTML before it's written to disk. This is useful for setting meta tags or environment variables based on input from other sources.

transform can be a single function or an array. This makes it easy to compose transformations.

View example

Inject language attribute:

import html from '@open-wc/rollup-plugin-html';
export default {
  output: { dir: 'dist' },
  plugins: [
    html({
      inputPath: './index.html',
      transform: html => html.replace('<html>', '<html lang="en-GB">'),
    }),
  ],
};

Inject language attributes and environment variables:

import html from '@open-wc/rollup-plugin-html';
import packageJson from './package.json';

const watchMode = process.env.ROLLUP_WATCH === 'true';

export default {
  output: { dir: 'dist' },
  plugins: [
    html({
      inputPath: './index.html',
      transform: [
        html => html.replace('<html>', '<html lang="en-GB">'),
        html =>
          html.replace(
            '<head>',
            `<head>
              <script>
                window.ENVIRONMENT = "${watchMode ? 'DEVELOPMENT' : 'PRODUCTION'}";
                window.APP_VERSION = "${packageJson.version}";
              </script>`,
          ),
      ],
    }),
  ],
};

#Public path

By default, all imports are made relative to the HTML file and expect files to be in the rollup output directory. With the publicPath option you can modify where files from the HTML file are requested from.

View example
import html from '@open-wc/rollup-plugin-html';

export default {
  output: { dir: 'dist' },
  plugins: [
    html({
      inputPath: './index.html',
      publicPath: '/static/',
    }),
  ],
};

#Multiple build outputs

It is possible to create multiple rollup build outputs and inject both bundles into the same HTML file. This way you can ship multiple bundles to your users, and load the most optimal version for the user's browser.

View example

When you configure rollup to generate multiple build outputs you can inject all outputs into a single HTML file.

To do this, create one parent @open-wc/rollup-plugin-html instance and use addOutput to create two child plugins for each separate rollup output.

Each output defines a unique name, this can be used to retrieve the correct bundle from bundles argument when creating the HTML template.

import html from '@open-wc/rollup-plugin-html';

const htmlPlugin = html({
  name: 'index.html',
  inject: false,
  template({ bundles }) {
    return `
      <html>
        <body>
        ${bundles.modern.entrypoints.map(
          e => `<script type="module" src="${e.importPath}"></script>`,
        )}

        <script nomodule src="./systemjs.js"></script>
        ${bundles.legacy.entrypoints.map(
          e => `<script nomodule>System.import("${e.importPath}")</script>`,
        )}
        </body>
      </html>
    `;
  },
});

export default {
  input: './app.js',
  output: [
    {
      format: 'es',
      dir: 'dist',
      plugins: [htmlPlugin.addOutput('modern')],
    },
    {
      format: 'system',
      dir: 'dist',
      plugins: [htmlPlugin.addOutput('legacy')],
    },
  ],
  plugins: [htmlPlugin],
};

If your outputs use different outputs directories, you need to set the outputBundleName option to specify which build to use to output the HTML file.

View example
import html from '@open-wc/rollup-plugin-html';

const htmlPlugin = html({
  name: 'index.html',
  inject: false,
  outputBundleName: 'modern',
  template({ bundles }) {
    return `
      ...
    `;
  },
});

export default {
  input: './app.js',
  output: [
    {
      format: 'es',
      dir: 'dist',
      plugins: [htmlPlugin.addOutput('modern')],
    },
    {
      format: 'system',
      dir: 'dist',
      plugins: [htmlPlugin.addOutput('legacy')],
    },
  ],
  plugins: [htmlPlugin],
};

#Configuration options

All configuration options are optional if an option is not set the plugin will fall back to smart defaults. See below example use cases.

#name

Type: string

Name of the generated HTML file. If inputPath is set, defaults to the inputPath filename, otherwise defaults to index.html.

#inputPath

Type: string

Path to the HTML file to use as input. Modules in this file are bundled and the HTML is used as the template for the generated HTML file.

#inputHtml

Type: string

Same as inputPath, but provides the HTML as a string directly.

#outputBundleName

Type: string

When using multiple build outputs, this is the name of the build that will be used to emit the generated HTML file.

#dir

Type: string

The directory to output the HTML file into. This defaults to the main output directory of your rollup build. If your build has multiple outputs in different directories, this defaults to the lowest directory on the file system.

#publicPath

Type: string

The public path where static resources are hosted. Any file requests (CSS, js, etc.) from the index.html will be prefixed with the public path.

#inject

Type: boolean

Whether to inject the rollup bundle into the output HTML. If using one of the input options, only the bundled modules in the HTML file are injected. Otherwise, all rollup bundles are injected. Default true. Set this to false if you need to apply some custom logic to how the bundle is injected.

#minify

Type: boolean | object | (html: string) => string | Promise<string>

When false, does not do any minification. When true, does minification with default settings. When an object, does minification with a custom config. When a function, the function is called with the html and should return the minified html. Defaults to true.

Default minification is done using html-minifier. When passing an object, the object is given to html-minifier directly so you can use any of the regular minify options.

#template

Type: string | (args: TemplateArgs) => string | Promise<string>

Template to inject js bundle into. It can be a string or an (async) function. If an input is set, that is used as the default output template. Otherwise defaults to a simple html file.

For more info see the configuration type definitions.

#transform

Type: TransformFunction | TransformFunction[]

TransformFunction: (html: string, args: TransformArgs) => string | Promise<string>

Function or array of functions that transform the final HTML output.

For more info see the configuration type definitions.

#Configuration types

Full typescript definitions of configuration options
import { OutputChunk, OutputOptions, OutputBundle, Plugin } from 'rollup';

export interface PluginOptions {
  name?: string;
  inputPath?: string;
  inputHtml?: string;
  outputBundleName?: string;
  publicPath?: string;
  inject?: boolean;
  minify?: boolean | object | MinifyFunction;
  template?: string | TemplateFunction;
  transform?: TransformFunction | TransformFunction[];
}

export type MinifyFunction = (html: string) => string | Promise<string>;

export interface GeneratedBundle {
  name: string;
  options: OutputOptions;
  bundle: OutputBundle;
}

export interface EntrypointBundle extends GeneratedBundle {
  entrypoints: {
    // path to import the entrypoint, can be used in an import statement
    // or script tag directly
    importPath: string;
    // associated rollup chunk, useful if you need to get more information
    // about the chunk. See the rollup docs for type definitions
    chunk: OutputChunk;
  }[];
}

export interface TemplateArgs {
  // if one of the input options was set, this references the HTML set as input
  inputHtml?: string;
  // the rollup bundle to be injected on the page. if there are multiple
  // rollup output options, this will reference the first bundle
  //
  // if one of the input options was set, only the bundled module script contained
  // in the HTML input are available to be injected in both the bundle and bundles
  // options
  bundle: EntrypointBundle;
  // the rollup bundles to be injected on the page. if there is only one
  // build output options, this will be an array with one option
  bundles: Record<string, EntrypointBundle>;
}

export interface TransformArgs {
  // see TemplateArgs
  bundle: EntrypointBundle;
  // see TemplateArgs
  bundles: Record<string, EntrypointBundle>;
}

export type TransformFunction = (html: string, args: TransformArgs) => string | Promise<string>;

export type TemplateFunction = (args: TemplateArgs) => string | Promise<string>;

export interface InputHtmlData {
  name?: string;
  rootDir: string;
  inputHtml: string;
}

export interface RollupPluginHtml extends Plugin {
  getHtmlFileName(): string;
  addHtmlTransformer(transform: TransformFunction): void;
  addOutput(name: string): Plugin;
}