macana

Static site generator for Obsidian Vault

  1. 1
  2. 2
  3. 3
  4. 4
  5. 5
  6. 6
  7. 7
  8. 8
  9. 9
  10. 10
  11. 11
  12. 12
  13. 13
  14. 14
  15. 15
  16. 16
  17. 17
  18. 18
  19. 19
  20. 20
  21. 21
  22. 22
  23. 23
  24. 24
  25. 25
  26. 26
  27. 27
  28. 28
  29. 29
  30. 30
  31. 31
  32. 32
  33. 33
  34. 34
  35. 35
  36. 36
  37. 37
  38. 38
  39. 39
  40. 40
  41. 41
  42. 42
  43. 43
  44. 44
  45. 45
  46. 46
  47. 47
  48. 48
  49. 49
  50. 50
  51. 51
  52. 52
  53. 53
  54. 54
// SPDX-FileCopyrightText: 2024 Shota FUJI <pockawoooh@gmail.com>
//
// SPDX-License-Identifier: Apache-2.0

import type * as Mdast from "../../deps/npm/mdast/types.ts";
import type { Extension } from "../../deps/npm/mdast-util-from-markdown/mod.ts";
import { SKIP, visit } from "../../deps/npm/unist-util-visit/mod.ts";

export class HeadingLevelExceedsLimitError extends Error {
	constructor(level: number, magnitude: number) {
		super(
			`Attempt to down-level h${level} by ${magnitude},` +
				` but resulting level (${level + magnitude}) exceeds limit (6)`,
		);
	}
}

export interface DownlevelHeadingsOptions {
	magnitude?: 1 | 2 | 3 | 4 | 5;
}

/**
 * Down-level headings by given amount (`magnitude`).
 * e.g. `# Foo` -> `## Foo`
 *
 * This functions throws `HeadingLevelExceedsLimitError` when the final
 * heading level exceeds 6 (`<h6>`).
 */
export function downlevelHeadings(
	nodes: Mdast.Nodes,
	{ magnitude = 1 }: DownlevelHeadingsOptions = {},
): void {
	visit(nodes, (node) => node.type === "heading", (node) => {
		if (node.type !== "heading") {
			return SKIP;
		}

		if (node.depth + magnitude > 6) {
			throw new HeadingLevelExceedsLimitError(node.depth, magnitude);
		}

		node.depth = (node.depth + magnitude) as 1 | 2 | 3 | 4 | 5 | 6;
	});
}

export function downlevelHeadingsFromMarkdown(
	options: DownlevelHeadingsOptions = {},
): Extension {
	return {
		transforms: [(nodes) => {
			downlevelHeadings(nodes, options);
		}],
	};
}