You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
124 lines
3.4 KiB
124 lines
3.4 KiB
/** |
|
* @fileoverview Define a style for the props casing in templates. |
|
* @author Armano |
|
*/ |
|
'use strict' |
|
|
|
const utils = require('../utils') |
|
const casing = require('../utils/casing') |
|
const svgAttributes = require('../utils/svg-attributes-weird-case.json') |
|
|
|
// ------------------------------------------------------------------------------ |
|
// Rule Definition |
|
// ------------------------------------------------------------------------------ |
|
|
|
module.exports = { |
|
meta: { |
|
type: 'suggestion', |
|
docs: { |
|
description: |
|
'enforce attribute naming style on custom components in template', |
|
categories: ['vue3-strongly-recommended', 'strongly-recommended'], |
|
url: 'https://eslint.vuejs.org/rules/attribute-hyphenation.html' |
|
}, |
|
fixable: 'code', |
|
schema: [ |
|
{ |
|
enum: ['always', 'never'] |
|
}, |
|
{ |
|
type: 'object', |
|
properties: { |
|
ignore: { |
|
type: 'array', |
|
items: { |
|
allOf: [ |
|
{ type: 'string' }, |
|
{ not: { type: 'string', pattern: ':exit$' } }, |
|
{ not: { type: 'string', pattern: '^\\s*$' } } |
|
] |
|
}, |
|
uniqueItems: true, |
|
additionalItems: false |
|
} |
|
}, |
|
additionalProperties: false |
|
} |
|
] |
|
}, |
|
/** @param {RuleContext} context */ |
|
create(context) { |
|
const sourceCode = context.getSourceCode() |
|
const option = context.options[0] |
|
const optionsPayload = context.options[1] |
|
const useHyphenated = option !== 'never' |
|
let ignoredAttributes = ['data-', 'aria-', 'slot-scope'].concat( |
|
svgAttributes |
|
) |
|
|
|
if (optionsPayload && optionsPayload.ignore) { |
|
ignoredAttributes = ignoredAttributes.concat(optionsPayload.ignore) |
|
} |
|
|
|
const caseConverter = casing.getExactConverter( |
|
useHyphenated ? 'kebab-case' : 'camelCase' |
|
) |
|
|
|
/** |
|
* @param {VDirective | VAttribute} node |
|
* @param {string} name |
|
*/ |
|
function reportIssue(node, name) { |
|
const text = sourceCode.getText(node.key) |
|
|
|
context.report({ |
|
node: node.key, |
|
loc: node.loc, |
|
message: useHyphenated |
|
? "Attribute '{{text}}' must be hyphenated." |
|
: "Attribute '{{text}}' can't be hyphenated.", |
|
data: { |
|
text |
|
}, |
|
fix: (fixer) => |
|
fixer.replaceText(node.key, text.replace(name, caseConverter(name))) |
|
}) |
|
} |
|
|
|
/** |
|
* @param {string} value |
|
*/ |
|
function isIgnoredAttribute(value) { |
|
const isIgnored = ignoredAttributes.some((attr) => { |
|
return value.includes(attr) |
|
}) |
|
|
|
if (isIgnored) { |
|
return true |
|
} |
|
|
|
return useHyphenated ? value.toLowerCase() === value : !/-/.test(value) |
|
} |
|
|
|
// ---------------------------------------------------------------------- |
|
// Public |
|
// ---------------------------------------------------------------------- |
|
|
|
return utils.defineTemplateBodyVisitor(context, { |
|
VAttribute(node) { |
|
if (!utils.isCustomComponent(node.parent.parent)) return |
|
|
|
const name = !node.directive |
|
? node.key.rawName |
|
: node.key.name.name === 'bind' |
|
? node.key.argument && |
|
node.key.argument.type === 'VIdentifier' && |
|
node.key.argument.rawName |
|
: /* otherwise */ false |
|
if (!name || isIgnoredAttribute(name)) return |
|
|
|
reportIssue(node, name) |
|
} |
|
}) |
|
} |
|
}
|
|
|