Metadata-Version: 2.4
Name: nb2wb
Version: 0.1.1
Summary: Write in Jupyter Notebooks. Publish anywhere.
Author: Tivadar Danka
License: MIT
Project-URL: Homepage, https://github.com/the-palindrome/nb2wb
Project-URL: Repository, https://github.com/the-palindrome/nb2wb
Project-URL: Issues, https://github.com/the-palindrome/nb2wb/issues
Keywords: jupyter,substack,latex,converter,technical-writing
Classifier: Development Status :: 4 - Beta
Classifier: Intended Audience :: Science/Research
Classifier: License :: OSI Approved :: MIT License
Classifier: Programming Language :: Python :: 3
Classifier: Programming Language :: Python :: 3.9
Classifier: Programming Language :: Python :: 3.10
Classifier: Programming Language :: Python :: 3.11
Classifier: Programming Language :: Python :: 3.12
Classifier: Programming Language :: Python :: 3.13
Classifier: Topic :: Text Processing :: Markup :: HTML
Requires-Python: >=3.9
Description-Content-Type: text/markdown
License-File: LICENSE
Requires-Dist: nbformat>=5.0
Requires-Dist: nbconvert>=6.0
Requires-Dist: ipykernel>=6.0
Requires-Dist: matplotlib>=3.5
Requires-Dist: Pillow>=9.0
Requires-Dist: Pygments>=2.10
Requires-Dist: PyYAML>=6.0
Requires-Dist: markdown>=3.4
Requires-Dist: unicodeit>=0.7
Provides-Extra: dev
Requires-Dist: pytest>=7.0; extra == "dev"
Requires-Dist: black; extra == "dev"
Requires-Dist: isort; extra == "dev"
Dynamic: license-file

# nb2wb

**Write in Jupyter Notebooks. Publish anywhere.**

`nb2wb` (short for *notebook to web*) converts Jupyter Notebooks (`.ipynb`), Quarto documents (`.qmd`),
and plain Markdown files (`.md`) into self-contained HTML files you can paste directly into publishing
platforms like **Substack**, **Medium**, and **X Articles** — with LaTeX, code, and outputs all rendered faithfully.

---

## Why

Most web publishing platforms strip MathJax and code-block formatting. `nb2wb` sidesteps
this by turning complex content into **images**, and converting simple inline math into
**Unicode text** that pastes cleanly as prose.

| Notebook element | nb2wb output |
|---|---|
| Inline LaTeX `$...$` | Unicode text (α, β, ∇, ℝ, …) |
| Display math `$$...$$`, `\[...\]`, `\begin{equation}` | PNG image |
| Code cells | Syntax-highlighted PNG image |
| Cell outputs (text, repr, …) | PNG image |
| Cell outputs (matplotlib figure, …) | Embedded PNG / SVG |

---

## Installation

```bash
pip install nb2wb
```

For development:

```bash
git clone https://github.com/the-palindrome/nb2wb.git
cd nb2wb
pip install -e ".[dev]"
```

---

## Usage

```bash
nb2wb notebook.ipynb                         # → notebook.html (Substack by default)
nb2wb notebook.ipynb -t medium               # → Medium format
nb2wb notebook.ipynb -t x                    # → X Articles format
nb2wb notebook.ipynb -c config.yaml          # with custom config
nb2wb notebook.ipynb -o out.html             # explicit output path
nb2wb notebook.ipynb --open                  # open in browser when done
nb2wb notebook.ipynb --serve                 # serve with ngrok tunnel (see below)
nb2wb article.md                             # → article.html from Markdown
nb2wb article.md --execute                   # execute code blocks via Jupyter kernel
nb2wb document.qmd                           # → document.html from Quarto
```

Then open the HTML file in your browser, click **Copy to clipboard**, and
paste into your platform's draft editor.

### Supported platforms

| Platform | Flag | How it works |
|---|---|---|
| **Substack** | `-t substack` (default) | Copy and paste — images transfer automatically |
| **Medium** | `-t medium` | Copy and paste text, then hover each image to copy it individually (or use `--serve` for one-click) |
| **X Articles** | `-t x` | Copy and paste text, then hover each image to copy it individually (or use `--serve` for one-click) |

### Platform tips

**Substack** works out of the box — click **Copy to clipboard**, paste into
your draft, and all text and images transfer in one step.

**Medium** strips embedded images from pasted HTML, so the default output
includes per-image **Copy image** buttons that appear on hover. For a
smoother workflow, use `--serve` mode (see below) which gives images public
URLs that Medium can fetch automatically — one click, everything pastes.

**X Articles** also strips embedded images. The same two workflows apply:
hover-to-copy each image individually, or use `--serve` for one-click pasting.

### Serve mode (`--serve`)

Some platforms (Medium, X Articles) strip base64-embedded images but will
fetch images from real HTTP URLs. The `--serve` flag starts a local HTTP
server and exposes it via an [ngrok](https://ngrok.com/) tunnel, giving every
image a public URL.

```bash
nb2wb notebook.ipynb -t medium --serve
```

This will:

1. Extract embedded images to an `images/` directory
2. Rewrite the HTML to reference those files
3. Start a local HTTP server and an ngrok tunnel
4. Open the public URL in your browser

Copy from the opened page and paste into your editor — images transfer
automatically because the platform can fetch them from the public URL.
Press **Ctrl-C** to stop the server when done.

**Requirements:** [ngrok](https://ngrok.com/download) must be installed and
authenticated (`ngrok config add-authtoken <TOKEN>`). Serve mode works with
any target platform (`-t substack`, `-t medium`, `-t x`).

---

## Configuration

All options are optional. Copy `example_config.yaml` and edit as needed:

```yaml
code:
  font_size: 14           # font size for code images
  theme: "monokai"        # any Pygments style: monokai, dracula, vs, solarized-dark, …
  line_numbers: true      # show line numbers
  font: "DejaVu Sans Mono"

latex:
  font_size: 16           # font size for LaTeX images
  dpi: 150                # image resolution
  color: "black"
  background: "white"
  padding: 0.15           # whitespace around each expression (inches)
  try_usetex: true        # use a full LaTeX install when available; falls back to mathtext
```

Pass it with `-c`:

```bash
nb2wb notebook.ipynb -c config.yaml
```

### LaTeX rendering modes

| Mode | Requirement | Quality |
|---|---|---|
| `usetex: true` (default) | LaTeX + dvipng installed | Full LaTeX, best quality |
| mathtext fallback | None (matplotlib built-in) | Most standard expressions |

If `try_usetex: true` and a LaTeX installation is found, full LaTeX is used
automatically. Otherwise `nb2wb` falls back to matplotlib's mathtext, which
handles most common expressions without any extra dependencies.

### Cell tags

Attach tags to cells to control what appears in the output.

| Tag | Effect |
|---|---|
| `hide-cell` | Entire cell omitted (input + output) |
| `hide-input` | Source code hidden; output shown |
| `hide-output` | Output hidden; source code shown |
| `latex-preamble` | Cell source used as LaTeX preamble; cell itself is hidden |
| `text-snippet` | Code rendered as copyable HTML text instead of a PNG image |

`hide-cell` also works on Markdown cells.

#### Notebook cell tags

In JupyterLab open **View → Cell Toolbar → Tags**; in Jupyter Notebook use
**View → Cell Toolbar → Tags** or edit the cell metadata directly.

#### Markdown cell tags

In `.md` files there are two ways to attach tags to code blocks.

**Fence-line tags** — add tags after the language on the opening fence:

````markdown
```python hide-input
secret = "not shown"
```

```python text-snippet
x = 1 + 1
print(x)
```
````

**HTML comment directives** — place a `<!-- nb2wb: ... -->` comment on its
own line immediately before the code block:

```markdown
<!-- nb2wb: hide-input -->
```python
secret = "not shown"
```
```

Multiple tags can be combined in a single comment:

```markdown
<!-- nb2wb: hide-input, hide-output -->
```python
x = 1
```
```

Both methods can be used together — the tags are merged.

#### Quarto cell tags

In `.qmd` files, use the standard `#|` options inside code chunks:

````markdown
```{python}
#| echo: false
secret = "not shown"
```
````

See the [Quarto documentation](https://quarto.org/) for all available options.

### LaTeX preamble

Custom LaTeX packages and macros can be supplied in several ways (all are
combined when rendering):

**1. Notebook cell (`.ipynb`)**

Tag any Markdown cell with `latex-preamble`. Its raw source is injected into
every formula's LaTeX document. The cell is invisible in the output.

```
\usepackage{xcolor}
\definecolor{accent}{HTML}{E8C547}
```

**2. Markdown file (`.md`)**

Use a fenced code block with the special language `latex-preamble`:

````markdown
```latex-preamble
\usepackage{xcolor}
\definecolor{accent}{HTML}{E8C547}
```
````

The block is consumed by the converter and does not appear in the output.

**3. Config file (for project-wide defaults)**

```yaml
latex:
  preamble: |
    \usepackage{xcolor}
    \definecolor{accent}{HTML}{E8C547}
```

> The preamble is only used when `try_usetex: true` and a LaTeX installation
> is found. The mathtext fallback ignores it.

---

### Code themes

Any [Pygments style](https://pygments.org/styles/) works:

```bash
python -c "from pygments.styles import get_all_styles; print(list(get_all_styles()))"
```

Popular choices: `monokai`, `dracula`, `nord`, `solarized-dark`, `vs`,
`github-dark`, `one-dark`.

---

## How it works

```
notebook.ipynb / document.qmd / article.md
      │
      ▼
  [nbformat]  parse cells
      │
      ├─ Markdown cell
      │      ├─ display math   →  [matplotlib]  →  PNG image
      │      ├─ inline math    →  [unicodeit]   →  Unicode text
      │      └─ prose          →  [markdown]    →  HTML
      │
      └─ Code cell
             ├─ source         →  [PIL + Pygments]  →  PNG image
             └─ outputs
                    ├─ stream / text/plain  →  [PIL]  →  PNG image
                    ├─ image/png            →  embedded as-is
                    └─ image/svg+xml        →  embedded as-is
                                               │
                                               ▼
                                        page.html  (self-contained)
```

All images are base64-encoded and embedded in the HTML file — no external
assets, no server required.

---

## Requirements

### Core (installed automatically)

- Python ≥ 3.9
- [nbformat](https://nbformat.readthedocs.io/) — notebook parsing
- [nbconvert](https://nbconvert.readthedocs.io/) + [ipykernel](https://ipykernel.readthedocs.io/) — `.qmd` cell execution
- [matplotlib](https://matplotlib.org/) — LaTeX / mathtext rendering
- [Pillow](https://pillow.readthedocs.io/) — image compositing
- [Pygments](https://pygments.org/) — syntax highlighting
- [PyYAML](https://pyyaml.org/) — config and `.qmd` front matter parsing
- [Markdown](https://python-markdown.github.io/) — prose conversion
- [unicodeit](https://github.com/svenkreiss/unicodeit) — inline LaTeX → Unicode

### Optional

| Extra | Install | What it adds |
|---|---|---|
| `examples` | `pip install -e ".[examples]"` | NumPy — required by the bundled example notebooks |
| `dev` | `pip install -e ".[dev]"` | pytest, black, isort |

For the best LaTeX rendering, also install a LaTeX distribution
([TeX Live](https://tug.org/texlive/) or [MiKTeX](https://miktex.org/)) plus
`dvipng`.

---

## License

MIT
