@html-eslint/eslint-plugin
Version:
ESLint plugin for html
91 lines (82 loc) • 1.9 kB
JavaScript
/**
* @typedef { import("@html-eslint/types").Tag } Tag
*
* @typedef {Object} Option
* @property {string[]} [Option.substitute]
*
* @typedef { import("../types").RuleModule<[Option]> } RuleModule
*/
const { RULE_CATEGORY } = require("../constants");
const { createVisitors } = require("./utils/visitors");
const MESSAGE_IDS = {
MISSING_ALT: "missingAlt",
};
/**
* @type {RuleModule}
*/
module.exports = {
meta: {
type: "code",
docs: {
description: "Require `alt` attribute at `<img>` tag",
category: RULE_CATEGORY.ACCESSIBILITY,
recommended: true,
},
fixable: null,
schema: [
{
type: "object",
properties: {
substitute: {
type: "array",
items: {
type: "string",
},
},
},
},
],
messages: {
[MESSAGE_IDS.MISSING_ALT]: "Missing `alt` attribute at `<img>` tag",
},
},
create(context) {
const substitute =
(context.options &&
context.options[0] &&
context.options[0].substitute) ||
[];
return createVisitors(context, {
Tag(node) {
if (node.name !== "img") {
return;
}
if (!hasAltAttrAndValue(node, substitute)) {
context.report({
loc: {
start: node.openStart.loc.start,
end: node.openEnd.loc.end,
},
messageId: MESSAGE_IDS.MISSING_ALT,
});
}
},
});
},
};
/**
* @param {Tag} node
* @param {string[]} substitute
* @returns
*/
function hasAltAttrAndValue(node, substitute = []) {
return node.attributes.some((attr) => {
if (attr.key && attr.value) {
return (
(attr.key.value === "alt" || substitute.includes(attr.key.value)) &&
typeof attr.value.value === "string"
);
}
return false;
});
}