2018-01-21 21:44:25 -08:00
|
|
|
// @ts-check
|
|
|
|
|
|
|
|
"use strict";
|
|
|
|
|
2024-06-08 20:41:40 -07:00
|
|
|
const { addErrorContext, frontMatterHasTitle } = require("../helpers");
|
|
|
|
const { filterByTypes, getHeadingLevel, inHtmlFlow } =
|
|
|
|
require("../helpers/micromark.cjs");
|
2018-01-21 21:44:25 -08:00
|
|
|
|
2024-02-27 20:42:09 -08:00
|
|
|
// eslint-disable-next-line jsdoc/valid-types
|
|
|
|
/** @type import("./markdownlint").Rule */
|
2018-01-21 21:44:25 -08:00
|
|
|
module.exports = {
|
2019-03-16 20:21:57 -07:00
|
|
|
"names": [ "MD025", "single-title", "single-h1" ],
|
2020-12-28 13:28:38 -08:00
|
|
|
"description": "Multiple top-level headings in the same document",
|
2023-11-09 20:05:30 -08:00
|
|
|
"tags": [ "headings" ],
|
2024-06-08 20:41:40 -07:00
|
|
|
"parser": "micromark",
|
2018-01-21 21:44:25 -08:00
|
|
|
"function": function MD025(params, onError) {
|
2020-01-25 18:40:39 -08:00
|
|
|
const level = Number(params.config.level || 1);
|
2019-03-16 20:21:57 -07:00
|
|
|
const foundFrontMatterTitle =
|
2019-04-13 11:18:57 -07:00
|
|
|
frontMatterHasTitle(
|
2019-03-16 20:21:57 -07:00
|
|
|
params.frontMatterLines,
|
|
|
|
params.config.front_matter_title
|
|
|
|
);
|
2018-04-27 22:05:34 -07:00
|
|
|
let hasTopLevelHeading = false;
|
2024-06-08 20:41:40 -07:00
|
|
|
const headings = filterByTypes(
|
|
|
|
params.parsers.micromark.tokens,
|
|
|
|
[ "atxHeading", "setextHeading" ]
|
|
|
|
);
|
|
|
|
for (const heading of headings) {
|
|
|
|
const headingLevel = getHeadingLevel(heading);
|
|
|
|
if ((headingLevel === level) && !inHtmlFlow(heading)) {
|
2019-03-16 20:21:57 -07:00
|
|
|
if (hasTopLevelHeading || foundFrontMatterTitle) {
|
2024-06-08 20:41:40 -07:00
|
|
|
const headingTexts = filterByTypes(
|
|
|
|
heading.children,
|
|
|
|
[ "atxHeadingText", "setextHeadingText" ]
|
|
|
|
);
|
|
|
|
const headingText = headingTexts.
|
|
|
|
map((token) => token.text).
|
|
|
|
join(" ").
|
|
|
|
replace(/[\r\n]+/g, " ");
|
|
|
|
addErrorContext(
|
|
|
|
onError,
|
|
|
|
heading.startLine,
|
|
|
|
headingText
|
|
|
|
);
|
|
|
|
} else if (heading.startLine === 1) {
|
2018-01-21 21:44:25 -08:00
|
|
|
hasTopLevelHeading = true;
|
|
|
|
}
|
|
|
|
}
|
2024-06-08 20:41:40 -07:00
|
|
|
}
|
2018-01-21 21:44:25 -08:00
|
|
|
}
|
|
|
|
};
|