Skip to main content
Version: Next

Modules

By default the modules should only perform safe transforms, see the module documentation below for details. You can disable modules by passing false as option, and enable them by passing true.

The order in which the modules are documented is also the order in which they are applied.

Attributes

normalizeAttributeValues

  • Normalize casing of specific attribute values that are case-insensitive (for example form[method], img[crossorigin], script[type], link[sizes]).
  • Trim surrounding whitespace for normalized values.
  • Apply the invalid value default for specific attributes (for example img[loading], img[decoding], track[kind], button[type], textarea[wrap], crossorigin, referrerpolicy, hidden, autocapitalize, marquee[behavior], marquee[direction]).

Invalid value defaults are only applied to the specific elements listed above (for example button[type] is normalized, but input[type] is not changed).

Example

Source:

<form method="GET"></form>
<img loading="">
<button type="EXAMPLE"></button>

Minified:

<form method="get"></form>
<img loading="eager">
<button type="submit"></button>

removeEmptyAttributes

Removes empty attributes when it is safe, based on the tag and attribute name. The removal rules are a fixed allowlist in the module.

Notes

  • Attributes are removed when the value is empty or whitespace-only.
  • Event handler attributes (for example onclick, onfocus) are always removed when empty.
  • Some attributes are only removed on specific tags, such as cols on <textarea> or minlength/maxlength on <input> and <textarea>.
  • Attributes like alt are intentionally not removed, even when empty.

Side effects

This module could break your styles or JS if you use selectors with attributes:

img[style=""] {
margin: 10px;
}

Example

Source:

<div id="" class="" title=""></div>
<button onclick="" onfocus=" "></button>
<textarea cols=""></textarea>
<img src="foo.jpg" alt="" style="">

Minified:

<div></div>
<button></button>
<textarea></textarea>
<img src="foo.jpg" alt="">

collapseAttributeWhitespace

Collapse redundant whitespace in attribute values where it is safe:

  • List-like attributes are normalized by collapsing internal whitespace and trimming ends (class, rel, ping, sandbox, headers, dropzone, sizes on <link>).
  • Single-value attributes are trimmed (for example href, style, src, width, height) when they are on the correct elements.
  • Event handler attributes (like onclick) are trimmed only at the ends; inner whitespace is preserved.

sizes on <img> is not modified.

Example

Source:

<a class=" content  page  " style="  display: block;    " href="   https://example.com"></a>

Minified:

<a class="content page" style="display: block;" href="https://example.com"></a>

removeRedundantAttributes

Removes redundant attributes from tags when they match HTML defaults:

  • method="get" from <form>
  • type="text" from <input>
  • type="submit" from <button>
  • language="javascript" and redundant JS type values from <script>
  • charset from inline <script> (only when src is not present)
  • media="all" from <style> and <link>
  • type="text/css" from <style>
  • type="text/css" from <link rel="stylesheet">
  • loading="eager" from <img> and <iframe>
  • decoding="auto" from <img>
  • kind="subtitles" from <track>
  • wrap="soft" from <textarea>
  • shape="rect" from <area>

Attribute values are matched case-insensitively with surrounding whitespace ignored. Script type="module" is preserved, and link[rel] is treated as a space-separated token list when checking for rel="stylesheet".

Options

This module is disabled by default, change option to true to enable this module.

Side effects

This module could break your styles or JS if you use selectors with attributes:

form[method="get"] {
color: red;
}

Example

Source:

<form method="get">
<input type="text">
</form>
<script type="module"></script>
<script type="text/javascript" charset="utf-8"></script>
<script src="app.js" charset="utf-8"></script>

Minified:

<form>
<input>
</form>
<script type="module"></script>
<script></script>
<script src="app.js" charset="utf-8"></script>

collapseBooleanAttributes

  • Collapses HTML boolean attributes (like disabled, checked, readonly) to the minimized form, regardless of their string value (for example checked="false" still becomes checked).
  • Collapses attributes whose value is an empty string to the minimized form (for example href="" becomes href).
  • Collapses missing value default attributes when they match the default, currently audio[preload=auto] and video[preload=auto].
  • Collapses crossorigin="anonymous" (case-insensitive) to crossorigin.
  • Leaves visible untouched on A-Frame elements (<a-*>) to avoid breaking visible="false".

Options

If your document uses AMP, set the amphtml flag to collapse additional, AMP-specific boolean attributes:

"collapseBooleanAttributes": {
"amphtml": true
}

When amphtml is enabled, AMP boolean attributes are collapsed when their value is empty, true, or matches the attribute name.

Side effects

This module could break your styles or JS if you use selectors with attributes:

button[disabled="disabled"] {
color: red;
}

Example

Source:

<button disabled="disabled">click</button>
<script defer=""></script>
<a href=""></a>
<script src="example-framework.js" crossorigin="anonymous"></script>
<video preload="auto"></video>
<a-entity visible="false"></a-entity>

Minified:

<button disabled>click</button>
<script defer></script>
<a href></a>
<script src="example-framework.js" crossorigin></script>
<video preload></video>
<a-entity visible="false"></a-entity>

deduplicateAttributeValues

Remove duplicate values from list-like attributes (class, rel, ping, sandbox, dropzone, sizes on link, headers). For case-insensitive token lists (rel, sandbox, dropzone, sizes), duplicates are removed regardless of casing (first occurrence kept). Whitespace is preserved where possible: repeated tokens are removed, but existing spacing between remaining tokens is kept.

Non-list-like attributes are not modified.

Example

Source:

<link rel="nofollow NoFoLlOw noopener">
<a class="foo foo bar">click</a>

Minified:

<link rel="nofollow noopener">
<a class="foo bar">click</a>

minifyAttributes

Minify specific attribute values. Currently this module targets meta[http-equiv="refresh"] by removing the url= prefix when present, trimming whitespace, and dropping empty URLs.

Example

Source:

<meta http-equiv="refresh" content="5; url=">
<meta http-equiv="refresh" content="5; url=http://example.com/">

Minified:

<meta http-equiv="refresh" content="5">
<meta http-equiv="refresh" content="5; http://example.com/">

minifyUrls

Convert absolute URL to relative URL using relateurl.

You have to install relateurl, terser and srcset in order to use this feature:

npm install --save-dev relateurl terser srcset
# if you prefer yarn
# yarn add --dev relateurl terser srcset
# if you prefer pnpm
# pnpm install --save-dev relateurl terser srcset

Options

The base URL to resolve against. Support String & URL.

htmlnano.process(html, {
minifyUrls: 'https://example.com' // Valid configuration
});
htmlnano.process(html, {
minifyUrls: new URL('https://example.com') // Valid configuration
});
htmlnano.process(html, {
minifyUrls: false // The module will be disabled
});
htmlnano.process(html, {
minifyUrls: true // Invalid configuration, the module will be disabled
});

Notes

  • Only http(s) and relative URLs are related. Non-HTTP schemes (e.g. mailto:, tel:, data:) are left untouched.
  • Hash-only (#...) and query-only (?...) URLs are left untouched.
  • link[rel="canonical"] is never rewritten.
  • srcset and imagesrcset are processed when srcset is installed; invalid srcset strings are left unchanged.
  • javascript: URLs are minified with terser when available, preserving leading whitespace and normalizing the protocol to lowercase.
  • This module is not enabled by the default presets; you must pass a base URL to turn it on.

Example

Basic Usage

Configuration:

htmlnano.process(html, {
minifyUrls: 'https://example.com'
});

Source:

<a href="https://example.com/foo/bar/baz">bar</a>

Minified:

<a href="foo/bar/baz">bar</a>

With sub-directory

Configuration:

htmlnano.process(html, {
minifyUrls: 'https://example.com/foo/baz/'
});

Source:

<a href="https://example.com/foo/bar">bar</a>

Minified:

<a href="../bar">bar</a>

Srcset

Configuration:

htmlnano.process(html, {
minifyUrls: 'https://example.com/foo/baz/'
});

Source:

<img srcset="https://example.com/foo/bar/image.png 1x, https://example.com/foo/bar/image2.png.png 2x">

Minified:

<img srcset="../bar/image.png 1x, ../bar/image2.png.png 2x">

sortAttributes

Sort attributes inside elements.

The module won't impact the plain-text size of the output. However it will improve the compression ratio of gzip/brotli used in HTTP compression.

Options

  • alphabetical: Default option. Sort attributes in alphabetical order.
  • frequency: Sort attributes by frequency across the whole document.
  • true: Alias for alphabetical.
  • false: Disable the module.

Frequency sorting preserves the original attribute name casing (for example viewBox stays viewBox) and uses alphabetical ordering on lowercased attribute names to break ties. When attribute names collide by case, the first occurrence wins.

Example

alphabetical

Source:

<input type="text" class="form-control" name="testInput" autofocus="" autocomplete="off" id="testId">

Processed:

<input autocomplete="off" autofocus="" class="form-control" id="testId" name="testInput" type="text">

frequency

Source:

<input type="text" class="form-control" name="testInput" id="testId">
<a id="testId" href="#" class="testClass"></a>
<img width="20" src="../images/image.png" height="40" alt="image" class="cls" id="id2">

Processed:

<input class="form-control" id="testId" type="text" name="testInput">
<a class="testClass" id="testId" href="#"></a>
<img class="cls" id="id2" alt="image" height="40" src="../images/image.png" width="20">

sortAttributesWithLists

Sort values in list-like attributes (class, rel, ping, sandbox, dropzone, headers, and sizes on <link>). sizes on <img> is not modified.

The module won't impact the plain-text size of the output. However it will improve the compression ratio of gzip/brotli used in HTTP compression.

Options

  • alphabetical: Default option. Sort attribute values in alphabetical order.
  • frequency: Sort attribute values by frequency across the document.
  • true: Alias for alphabetical.
  • false: Disable the module.

Frequency sorting preserves duplicates, ignores redundant whitespace in the attribute value, and uses alphabetical ordering to break ties. Attribute name casing is preserved.

Example

alphabetical

Source:

<div class="foo baz bar">click</div>

Processed:

<div class="bar baz foo">click</div>

frequency

Source:

<div class="foo baz bar"></div><div class="bar foo"></div>

Processed:

<div class="foo bar baz"></div><div class="foo bar"></div>

HTML Content

collapseWhitespace

Collapses redundant white spaces (including new lines). It doesn’t affect white spaces in the elements <style>, <textarea>, <script>, <pre>, and <template>.

Options

  • conservative — collapses all redundant whitespace to 1 space (default). Whitespace around inline elements (like <a>, <span>, <code>) is preserved when possible.
  • aggressive — collapses redundant whitespace and trims around nodes when it is safe. This may remove indentation and drop whitespace-only text nodes between comments and non-inline elements.
  • all — collapses all redundant whitespace and trims text nodes. This is the most aggressive behavior and can remove meaningful spacing between inline elements.

Notes

  • Comments are preserved, but whitespace around them can be collapsed depending on the option.
  • Template content is left untouched.

Side effects

all <i>hello</i> <i>world</i> or <i>hello</i><br><i>world</i> after minification will be rendered as helloworld. To prevent that use either the default conservative option, or the aggressive option.

Example

Source:

<div>
hello world!
<a href="#">answer</a>
<style>div { color: red; } </style>
<main></main>
</div>

Minified (with all):

<div>hello world!<a href="#">answer</a><style>div  { color: red; }  </style><main></main></div>

Minified (with aggressive):

<div>hello world! <a href="#">answer</a><style>div  { color: red; }  </style><main></main></div>

Minified (with conservative):

<div> hello world! <a href="#">answer</a> <style>div  { color: red; }  </style> <main></main> </div>

removeComments

Options

  • safe – removes HTML comments but keeps:
    • conditional comments (<!--[if ...]>...<![endif]--> and downlevel-revealed forms)
    • <!--noindex-->...<!--/noindex--> (case/spacing tolerant)
    • <!--sse-->...<!--/sse--> Server-Side Excludes markers (case/spacing tolerant)
    • excerpt markers that start with more (case/spacing tolerant), e.g. <!-- more -->, <!-- MORE -->, <!-- more Read more --> (default)
  • all — removes all HTML comments, including conditional/noindex/sse/excerpt comments
  • A RegExp — removes HTML comments that match the regexp (non-matching comments are kept)
  • A string — treated as a regexp pattern. Supports /pattern/flags or a plain pattern string (useful in JSON config files)
  • A Function that returns boolean — removes HTML comments for which the callback returns a truthy value

Example

Source:

{
removeComments: 'all'
}
<div><!-- test --></div>

Minified:

<div></div>

Source:

{
removeComments: 'safe'
}
<!--noindex-->indexed?<!--/noindex-->
<!--[if IE 8]><link href="ie8only.css" rel="stylesheet"><![endif]-->
Lorem ipsum <!-- more --> dolor sit amet <!-- comment -->

Minified:

<!--noindex-->indexed?<!--/noindex-->
<!--[if IE 8]><link href="ie8only.css" rel="stylesheet"><![endif]-->
Lorem ipsum <!-- more --> dolor sit amet

Source:

{
removeComments: /<!--(\/)?noindex-->/
}
<div><!--noindex-->this text will not be indexed<!--/noindex-->Lorem ipsum dolor sit amet<!--more-->Lorem ipsum dolor sit amet</div>

Minified:

<div>this text will not be indexedLorem ipsum dolor sit amet<!--more-->Lorem ipsum dolor sit amet</div>

Source (JSON config):

{
"removeComments": "/<!--(\\/)?noindex-->/"
}

Source:

{
removeComments: (comment) => {
if (comment.includes('noindex')) return true;
return false;
}
}
<div><!--noindex-->this text will not be indexed<!--/noindex-->Lorem ipsum dolor sit amet<!--more-->Lorem ipsum dolor sit amet</div>

Minified:

<div>this text will not be indexedLorem ipsum dolor sit amet<!--more-->Lorem ipsum dolor sit amet</div>

removeEmptyElements

Removes elements that have no meaningful content.

Options

  • true — removes empty elements without attributes.
  • { removeWithAttributes: true } — removes empty elements even if they have attributes.

Empty elements are defined as elements with no child elements and only whitespace/comments as content. Void elements (like <img> or <br>) are never removed.

Side effects

This module can remove elements that are used for styling or scripting (for example <span class="icon"></span>). It is disabled by default.

Example

Source:

<div>hello<span><b></b></span></div>
<div><span class="icon"></span></div>

Minified (removeEmptyElements: true):

<div>hello</div>
<div><span class="icon"></span></div>

Minified (removeEmptyElements: { removeWithAttributes: true }):

<div>hello</div>
<div></div>

minifyConditionalComments

Minifies HTML inside IE conditional comments (both downlevel-hidden and downlevel-revealed forms). The conditional comment wrappers are preserved while the inner HTML is processed with the same htmlnano options, so other modules (like collapseWhitespace, minifyCss, or minifyJs) can apply within the conditional block.

Notes

  • Empty conditional comments and standalone <!--<![endif]--> markers are left untouched.
  • If the comment content includes an opening <html> without a closing tag, any auto-inserted </html> from the internal parse step is removed so it is not injected into the conditional block.

Example

Source:

<!--[if lte IE 7]>
<style type="text/css">
.title {
color: red;
}
</style>
<![endif]-->

Minified:

<!--[if lte IE 7]><style type="text/css">.title{color:red}</style><![endif]-->

removeOptionalTags

Remove certain tags that can be omitted, see HTML Standard - 13.1.2.4 Optional tags.

Only tags without attributes are eligible. If the element has any attributes, the tag is preserved.

Notes

  • htmlnano can only remove a tag when both its start and end tags can be omitted.
  • Due to the limitation of PostHTML, htmlnano can’t remove only the start tag or only the end tag of an element.
  • Supported optional tags are limited to the ones that can be removed as a pair.

Supported tags and key rules:

  • html
    • Start tag can be omitted when the first child is not a comment.
    • End tag can be omitted when the html element is not immediately followed by a comment.
  • head
    • Start tag can be omitted when the element is empty or the first child is an element.
    • End tag can be omitted when head is not immediately followed by ASCII whitespace or a comment.
  • body
    • Start tag can be omitted when the element is empty or the first child is not ASCII whitespace or a comment.
    • Start tag can’t be omitted if the first child element is meta, link, script, style, or template.
    • End tag can be omitted when body is not immediately followed by a comment.
  • colgroup
    • Start tag can be omitted when the first child element is col, and the element is not immediately preceded by another colgroup.
    • End tag can be omitted when colgroup is not immediately followed by ASCII whitespace or a comment.
  • tbody
    • Start tag can be omitted when the first child element is tr, and the element is not immediately preceded by tbody, thead, or tfoot.
    • End tag can be omitted when the element is not immediately followed by tbody or tfoot.

Example

Source:

<html><head><title>Title</title></head><body><p>Hi</p></body></html>

Minified:

<title>Title</title><p>Hi</p>

removeAttributeQuotes

Remove quotes around attributes when possible, see HTML Standard - 12.1.2.3 Attributes - Unquoted attribute value syntax.

Options

  • force — if true, forces quoteAllAttributes to false even when other PostHTML options or plugins set it to true.

Example

Source:

<div class="foo" title="hello world"></div>

Minified:

<div class=foo title="hello world"></div>

Notice

The feature is implemented by posthtml-render's quoteAllAttributes, which is a PostHTML option. removeAttributeQuotes sets this option to false only when it is not already defined, so other PostHTML plugins and configuration can override it.

For example:

posthtml([
htmlnano({
removeAttributeQuotes: true
})
]).process(html, {
quoteAllAttributes: true
})

removeAttributeQuotes will not work because PostHTML's quoteAllAttributes takes the priority.

If you need to ensure quotes are removed even when quoteAllAttributes is already true, enable the force option:

posthtml([
htmlnano({
removeAttributeQuotes: { force: true }
})
]).process(html, {
quoteAllAttributes: true
})

<style>, <script> and <svg> Tags

mergeStyles

Merges multiple <style> with the same normalized media and type into one tag, as long as all other attributes match (for example: nonce, title, data-*). Attribute matching is strict: any differing attribute (including nonce) prevents merging.

Normalization details:

  • Missing or empty type is treated as text/css (case-insensitive).
  • Missing or empty media is treated as all, and internal whitespace is collapsed.
  • Boolean attributes like amp-custom and disabled are treated as present when set.

Skipped styles:

  • <style scoped>...</style>
  • <style integrity>...</style>
  • AMP boilerplate styles (amp-boilerplate, amp4ads-boilerplate, amp4email-boilerplate)

Example

Source:

<style>h1 { color: red }</style>
<style media="print">div { color: blue }</style>

<style type="text/css" media="print">a {}</style>
<style>div { font-size: 20px }</style>

Minified:

<style>h1 { color: red } div { font-size: 20px }</style>
<style media="print">div { color: blue } a {}</style>

mergeScripts

Merge adjacent inline <script> tags when they share the same normalized attributes (all except src, integrity, and type). The merged content is appended into the last script in the group and the earlier scripts are removed.

Notes

  • Only inline scripts with mergeable types are considered: text/javascript and application/javascript (default is text/javascript). Other types (including type="module") are left untouched.
  • Scripts with src or integrity are never merged and they break a merge group, so code on each side stays separate.
  • Boolean attributes (async, defer, nomodule) are normalized and treated as present, so defer and defer="defer" match.
  • Scripts are separated by nonce value, by nomodule, and by async/defer differences.
  • A missing trailing semicolon is added when concatenating. If a script ends with a line comment, the merger inserts \n; before the next script to avoid comment swallowing.

Side effects

Merging changes where the code physically lives in the document. This can break code when different attribute sets must remain isolated (for example async or nomodule), or when external tools rely on script boundaries.

Example

Source:

<script>const foo = 'A:1';</script>
<script class="test">foo = 'B:1';</script>
<script type="text/javascript">foo = 'A:2';</script>
<script defer>foo = 'C:1';</script>
<script>foo = 'A:3';</script>
<script defer="defer">foo = 'C:2';</script>
<script class="test" type="text/javascript">foo = 'B:2';</script>

Minified:

<script>const foo = 'A:1';foo = 'A:2';foo = 'A:3';</script>
<script defer="defer">foo = 'C:1';foo = 'C:2';</script>
<script class="test" type="text/javascript">foo = 'B:1';foo = 'B:2';</script>

minifyCss

Minifies CSS with cssnano inside <style> tags and style attributes.

Only <style> tags with a CSS type (text/css or no type attribute) are minified. Other style types (for example text/less) are left untouched. CDATA wrappers are preserved, even when surrounded by whitespace.

Skipped nodes:

  • Any element with an integrity attribute (covers both <style> tags and style attributes).
  • AMP boilerplate styles (amp-boilerplate, amp4ads-boilerplate, amp4email-boilerplate).

Notes:

  • style attributes are wrapped in a temporary selector (a{...}) before minification so cssnano can parse them, then the wrapper is removed.

You have to install cssnano and postcss in order to use this feature:

npm install --save-dev cssnano postcss
# if you prefer yarn
# yarn add --dev cssnano postcss
# if you prefer pnpm
# pnpm install --save-dev cssnano postcss

Options

See the documentation of cssnano for all supported optimizations. By default CSS is minified with preset default, which shouldn't have any side-effects.

To use another preset or disable some optimizations pass options to minifyCss module:

htmlnano.process(html, {
minifyCss: {
preset: ['default', {
discardComments: {
removeAll: true,
},
}]
}
});

Example

Source:

<div>
<style>
h1 {
margin: 10px 10px 10px 10px;
color: #ff0000;
}
</style>
</div>

Minified:

<div>
<style>h1{margin:10px;color:red}</style>
</div>

minifyJs

Minifies JS using Terser inside <script> tags.

You have to install terser in order to use this feature:

npm install --save-dev terser
# if you prefer yarn
# yarn add --dev terser
# if you prefer pnpm
# pnpm install --save-dev terser

Options

See the documentation of Terser for all supported options. Terser options can be passed directly to the minifyJs module:

htmlnano.process(html, {
minifyJs: {
output: { quote_style: 1 },
},
});

The module treats script types with parameters (for example text/javascript; charset=utf-8) as JavaScript. For type="module" scripts, it enables Terser's module option unless you explicitly set module yourself.

Notes

  • Only JavaScript script types are processed: the default type, text/javascript, application/javascript, and legacy text/ecmascript. Other types (for example application/json) are left untouched.
  • Any <script> with an integrity attribute is skipped to preserve SRI safety. Generate SRI after minification if you rely on it.
  • Inline event handler attributes (like onclick or onClick) are minified as JavaScript.
  • CDATA wrappers inside <script> in SVG are preserved; the inner JS is minified and re-wrapped.
  • For AMP documents, inline handlers like on="tap:..." are not modified when using the AMP-safe preset.

Example

Source:

<div>
<script>
/* comment */
const foo = function () {

};
</script>
</div>

Minified:

<div>
<script>const foo=function(){};</script>
</div>

minifyJson

Minifies JSON inside <script> tags whose type ends in /json or +json (case-insensitive), including when MIME parameters are present.

Notes

  • Only <script> tags with an explicit JSON MIME type are processed. Tags without a type attribute are left untouched.
  • MIME parameters are allowed (application/json; charset=utf-8) and casing is ignored.
  • Tags with integrity are skipped to preserve SRI safety.
  • Invalid JSON is left unchanged.
  • JSON-like suffixes such as application/jsonp are not considered JSON and are skipped.

Example

Source:

<script type="application/json">
{
"user": "me"
}
</script>

Minified:

<script type="application/json">{"user":"me"}</script>

Source:

<script type="application/ld+json; charset=UTF-8">
{
"id": 1
}
</script>

Minified:

<script type="application/ld+json; charset=UTF-8">{"id":1}</script>

minifySvg

Minifies SVG inside <svg> tags using SVGO.

SVGO is an optional dependency. If it is not installed, SVG content is left untouched.

Options

See the documentation of SVGO for all supported options. SVGO options can be passed directly to the minifySvg module:

htmlnano.process(html, {
minifySvg: {
plugins: [
{
name: 'preset-default',
params: {
overrides: {
builtinPluginName: {
optionName: 'optionValue'
},
},
},
}
]
}
});

minifySvg: true enables SVGO with its default configuration. htmlnano enables SVGO multipass by default; set multipass: false to disable it.

Notes

  • SVGs are rendered with quoted attributes before SVGO runs to keep the output stable.
  • Parser errors from SVGO leave the original <svg> as-is. Other SVGO errors are logged and the module falls back to a no-plugin SVGO run; if that also fails, the original SVG is preserved.
  • Set skipInternalWarnings: true to suppress SVGO error logging.

Example

Source:

<svg version="1.1" baseProfile="full" width="300" height="200" xmlns="http://www.w3.org/2000/svg">
<rect width="100%" height="100%" fill="red" />

<circle cx="150" cy="100" r="80" fill="green" />

<text x="150" y="125" font-size="60" text-anchor="middle" fill="white">SVG</text>
</svg>

Minified:

<svg baseProfile="full" width="300" height="200" xmlns="http://www.w3.org/2000/svg"><rect width="100%" height="100%" fill="red"/><circle cx="150" cy="100" r="80" fill="green"/><text x="150" y="125" font-size="60" text-anchor="middle" fill="#fff">SVG</text></svg>

removeUnusedCss

Removes unused CSS inside <style> tags with either uncss or PurgeCSS.

With uncss

You have to install uncss in order to use this feature:

npm install --save-dev uncss
# if you prefer yarn
# yarn add --dev uncss
# if you prefer pnpm
# pnpm install --save-dev uncss

You can also use a maintained fork @novaatwarren/uncss instead.

Options

See the documentation of uncss for all supported options.

uncss options can be passed directly to the removeUnusedCss module:

htmlnano.process(html, {
removeUnusedCss: {
ignore: ['.do-not-remove']
}
});

The following uncss options are ignored if passed to the module:

  • stylesheets
  • ignoreSheets
  • raw

With PurgeCSS

Use PurgeCSS instead of uncss by adding tool: 'purgeCSS' to the options.

You have to install purgecss in order to use this feature:

npm install --save-dev purgecss
# if you prefer yarn
# yarn add --dev purgecss
# if you prefer pnpm
# pnpm install --save-dev purgecss
Options

See the documentation of PurgeCSS for all supported options.

PurgeCSS options can be passed directly to the removeUnusedCss module:

htmlnano.process(html, {
removeUnusedCss: {
tool: 'purgeCSS',
safelist: ['.do-not-remove']
}
});

The following PurgeCSS options are ignored if passed to the module:

  • content
  • css
  • extractors

Notes

  • Only <style> tags with a CSS type (text/css or no type attribute) are processed. Other style types are left untouched.
  • CDATA wrappers in <style> tags are preserved after removing unused rules.
  • Empty styles are removed entirely when all rules are stripped.
  • AMP boilerplate styles (amp-boilerplate, amp4ads-boilerplate, amp4email-boilerplate) are not touched.
  • PurgeCSS keeps tag selectors based on the HTML (for example section {}), and class/id extraction treats whitespace and newlines as separators.

Example

Source:

<div class="b">
<style>
.a {
margin: 10px 10px 10px 10px;
}
.b {
color: #ff0000;
}
</style>
</div>

Optimized:

<div class="b">
<style>
.b {
color: #ff0000;
}
</style>
</div>

Miscellaneous

custom

It's also possible to pass custom modules in the minifier. As a function:

const options = {
custom: function (tree, options) {
// Some minification
return tree;
}
};

Or as a list of functions:

const options = {
custom: [
function (tree, options) {
// Some minification
return tree;
},

function (tree, options) {
// Some other minification
return tree;
}
]
};

htmlnano's options are passed to your custom plugin by the second parameter options.