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 🙏

© 2025 – Pkg Stats / Ryan Hefner

turnish

v1.3.0

Published

HTML to Markdown converter

Readme

Turnish

Test NPM version

Turnish is a HTML to Markdown converter written in JavaScript.

This is a fork of Turndown, originally created by Dom Christie.

Demo

You can try Turnish online here.

Turnish vs Turndown

Turnish has been created to address various issues found in Turndown, while keeping its core functionality and API intact.

Library user perspective:

  • Fixed various issues, such as:
    • escaping unwanted characters that can break the Markdown rendering.
    • removing new lines in link texts.
  • Added more options, such as:
    • htmlRetentionMode: configures how non-standard or unsupported HTML fragments are handled during conversion. For example, whether to convert them to Markdown where possible, preserve them as raw HTML, or retain them (adding a markdown="1" attribute when the HTML contains Markdown) so Markdown processors can parse any embedded Markdown.
    • bulletListMarker: configures the number of spaces after bullet list markers.
  • Changed the default behavior to better comply with de facto standards.
  • Remains compatible with Turndown plugins, such as @joplin/turndown-plugin-gfm.
  • Remains licensed under the MIT License.

Library/plugin developer perspective:

  • Full TypeScript support
  • Modernized build system using Vite
  • Improved code structure for better maintainability

Installation

npm:

npm install turnish

pnpm:

pnpm install turnish

Browser:

<script src="https://cdn.jsdelivr.net/npm/turnish@latest/dist/index.iife.js"></script>

Usage

For Node.js:

var Turnish = require('turnish')

var turnish = new Turnish()
var markdown = turnish.render('<h1>Hello world!</h1>')

ES module import (Node with ESM, bundlers, or browsers supporting modules):

import Turnish from 'turnish'

const turnish = new Turnish()
const markdown = turnish.render('<h1>Hello world!</h1>')

Turnish also accepts DOM nodes as input (either element nodes, document nodes, or document fragment nodes):

var markdown = turnish.render(document.getElementById('content'))

Options

Options can be passed in to the constructor on instantiation. For example:

var turnish = new Turnish({ option: 'value' })

| Option | Description | Default | | :--------------------------- | :------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------ | :--------- | | headingStyle | Style of headings such as h1, h2...setext: underlined headingsatx: hash-prefixed headings | atx | | hr | Any Thematic break | --- | | bulletListMarker | -, +, or * | - | | listMarkerSpaceCount | Number of spaces after the bullet list marker.1 - 4 | 1 | | listItemIndent | tab or space | space | | listItemIndentSpaceCount | Number of spaces for the nested list item indentation.1 - 4 | 4 | | codeBlockStyle | indented or fenced | fenced | | fence | ``` or ~~~ | ``` | | emDelimiter | _ or * | * | | strongDelimiter | ** or __ | ** | | linkStyle | inlined or referenced | inlined | | linkReferenceStyle | full, collapsed, or shortcut | full | | linkReferenceDeduplication | How duplicate link reference definitions are handled.none: preserve every duplicate reference as-is.full: merge identical references into one. | full | | htmlRetentionMode | How non-standard HTML fragments are handled.standard: does not preserve non-standard HTMLpreserveAll: preserve non-standard HTML tags and its childrenmarkdownIncludingHtml: preserve non-standard HTML while including Markdown as its child. markdown="1" attribute is added when the HTML contains Markdown. | standard | | preformattedCode | false or true | false |

Advanced Options

| Option | Valid values | Default | | :------------------- | :------------------------ | :-------------------------- | | blankReplacement | rule replacement function | See Special Rules below | | keepReplacement | rule replacement function | See Special Rules below | | defaultReplacement | rule replacement function | See Special Rules below |

Methods

addRule(key, rule)

The key parameter is a unique name for the rule for easy reference. Example:

turnish.addRule('strikethrough', {
  filter: ['del', 's', 'strike'],
  replacement: function (content) {
    return '~' + content + '~'
  }
})

addRule returns the Turnish instance for chaining.

See Extending with Rules below.

keep(filter)

Determines which elements are to be kept and rendered as HTML. By default, Turndown does not keep any elements. The filter parameter works like a rule filter (see section on filters belows). Example:

turnish.keep(['del', 'ins'])
turnish.render('<p>Hello <del>world</del><ins>World</ins></p>') // 'Hello <del>world</del><ins>World</ins>'

This will render <del> and <ins> elements as HTML when converted.

keep can be called multiple times, with the newly added keep filters taking precedence over older ones. Keep filters will be overridden by the standard CommonMark rules and any added rules. To keep elements that are normally handled by those rules, add a rule with the desired behaviour.

keep returns the Turnish instance for chaining.

remove(filter)

Determines which elements are to be removed altogether i.e. converted to an empty string. By default, Turnish does not remove any elements. The filter parameter works like a rule filter (see section on filters belows). Example:

turnish.remove('del')
turnish.render('<p>Hello <del>world</del><ins>World</ins></p>') // 'Hello World'

This will remove <del> elements (and contents).

remove can be called multiple times, with the newly added remove filters taking precedence over older ones. Remove filters will be overridden by the keep filters, standard CommonMark rules, and any added rules. To remove elements that are normally handled by those rules, add a rule with the desired behaviour.

remove returns the Turnish instance for chaining.

use(plugin|array)

Use a plugin, or an array of plugins. Example:

// Import plugins from @joplin/turndown-plugin-gfm
var turndownPluginGfm = require('@joplin/turndown-plugin-gfm')
var gfm = turndownPluginGfm.gfm
var tables = turndownPluginGfm.tables
var strikethrough = turndownPluginGfm.strikethrough

// Use the gfm plugin
turnish.use(gfm)

// Use the table and strikethrough plugins only
turnish.use([tables, strikethrough])

use returns the Turnish instance for chaining.

See Plugins below.

Extending with Rules

Turnish can be extended by adding rules. A rule is a plain JavaScript object with filter and replacement properties. For example, the rule for converting <p> elements is as follows:

{
  filter: 'p',
  replacement: function (content) {
    return '\n\n' + content + '\n\n'
  }
}

The filter selects <p> elements, and the replacement function returns the <p> contents separated by two new lines.

filter string|Array|Function

The filter property determines whether or not an element should be replaced with the rule's replacement. DOM nodes can be selected simply using a tag name or an array of tag names:

  • filter: 'p' will select <p> elements
  • filter: ['em', 'i'] will select <em> or <i> elements

The tag names in the filter property are expected in lowercase, regardless of their form in the document.

Alternatively, the filter can be a function that returns a boolean depending on whether a given node should be replaced. The function is passed a DOM node as well as the Turnish options. For example, the following rule selects <a> elements (with an href) when the linkStyle option is inlined:

filter: function (node, options) {
  return (
    options.linkStyle === 'inlined' &&
    node.nodeName === 'A' &&
    node.getAttribute('href')
  )
}

replacement Function

The replacement function determines how an element should be converted. It should return the Markdown string for a given node. The function is passed the node's content, the node itself, and the Turnish options.

The following rule shows how <em> elements are converted:

rules.emphasis = {
  filter: ['em', 'i'],

  replacement: function (content, node, options) {
    return options.emDelimiter + content + options.emDelimiter
  }
}

Special Rules

Blank rule determines how to handle blank elements. It overrides every rule (even those added via addRule). A node is blank if it only contains whitespace, and it's not an <a>, <td>,<th> or a void element. Its behaviour can be customised using the blankReplacement option.

Keep rules determine how to handle the elements that should not be converted, i.e. rendered as HTML in the Markdown output. By default, no elements are kept. Block-level elements will be separated from surrounding content by blank lines. Its behaviour can be customised using the keepReplacement option.

Remove rules determine which elements to remove altogether. By default, no elements are removed.

Default rule handles nodes which are not recognised by any other rule. By default, it outputs the node's text content (separated by blank lines if it is a block-level element). Its behaviour can be customised with the defaultReplacement option.

Rule Precedence

Turnish iterates over the set of rules, and picks the first one that matches the filter. The following list describes the order of precedence:

  1. Blank rule
  2. Added rules (optional)
  3. Commonmark rules
  4. Keep rules
  5. Remove rules
  6. Default rule

Plugins

The plugin API provides a convenient way for developers to apply multiple extensions. A plugin is just a function that is called with the Turnish instance.

Escaping Markdown Characters

Turnish uses backslashes (\) to escape Markdown characters in the HTML input. This ensures that these characters are not interpreted as Markdown when the output is compiled back to HTML. For example, the contents of <h1>1. Hello world</h1> needs to be escaped to 1\. Hello world, otherwise it will be interpreted as a list item rather than a heading.

To avoid the complexity and the performance implications of parsing the content of every HTML element as Markdown, Turnish uses a group of regular expressions to escape potential Markdown syntax. As a result, the escaping rules can be quite aggressive.

Overriding Turnish.escape()

If you are confident in doing so, you may want to customise the escaping behaviour to suit your needs. This can be done by overriding the escape() method of Turnish. escape() takes the text of each HTML element and should return a version with the Markdown characters escaped.

Note: text in code elements is never passed toescape().

License

MIT License

  • Copyright (c) 2025- Manabu Nakazawa
  • Copyright (c) 2017-2025 Dom Christie

Turnish is originally based on Turndown by Dom Christie, and is licensed under the MIT License.