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
  55. 55
  56. 56
  57. 57
  58. 58
  59. 59
  60. 60
  61. 61
// 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";
import { toString } from "../../deps/npm/mdast-util-to-string/mod.ts";
import { fastUslug } from "../../deps/npm/@shelf/fast-uslug/mod.ts";

function hasHProperties(
	data: Mdast.Data | undefined,
): data is Mdast.Data & { hProperties: Record<string, unknown> } {
	return !!(data && "hProperties" in data &&
		typeof data.hProperties === "object" && data.hProperties);
}

/**
 * This function mutates the given node tree.
 */
export function autoHeadingId(nodes: Mdast.Nodes): void {
	const counts = new Map<string, number>();

	visit(nodes, (node) => node.type === "heading", (node) => {
		if (node.type !== "heading") {
			return SKIP;
		}

		if (hasHProperties(node.data) && node.data.hProperties.id) {
			return SKIP;
		}

		const id = fastUslug(toString(node), {
			lower: false,
		});

		const count = counts.get(id) ?? 0;
		counts.set(id, count + 1);

		const finalId = count > 0 ? id + "__" + count : id;

		node.data = {
			...node.data,
			hProperties: {
				...(node.data && "hProperties" in node.data &&
						typeof node.data.hProperties === "object"
					? node.data.hProperties
					: {}),
				id: finalId,
			},
		};
	});
}

export function autoHeadingIdFromMarkdown(): Extension {
	return {
		transforms: [(nodes) => {
			autoHeadingId(nodes);
		}],
	};
}