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.
214 lines
5.9 KiB
214 lines
5.9 KiB
/** |
|
* @fileoverview Prevent missing displayName in a React component definition |
|
* @author Yannick Croissant |
|
*/ |
|
|
|
'use strict'; |
|
|
|
const Components = require('../util/Components'); |
|
const astUtil = require('../util/ast'); |
|
const docsUrl = require('../util/docsUrl'); |
|
const propsUtil = require('../util/props'); |
|
|
|
// ------------------------------------------------------------------------------ |
|
// Rule Definition |
|
// ------------------------------------------------------------------------------ |
|
|
|
module.exports = { |
|
meta: { |
|
docs: { |
|
description: 'Prevent missing displayName in a React component definition', |
|
category: 'Best Practices', |
|
recommended: true, |
|
url: docsUrl('display-name') |
|
}, |
|
|
|
schema: [{ |
|
type: 'object', |
|
properties: { |
|
ignoreTranspilerName: { |
|
type: 'boolean' |
|
} |
|
}, |
|
additionalProperties: false |
|
}] |
|
}, |
|
|
|
create: Components.detect((context, components, utils) => { |
|
const config = context.options[0] || {}; |
|
const ignoreTranspilerName = config.ignoreTranspilerName || false; |
|
|
|
const MISSING_MESSAGE = 'Component definition is missing display name'; |
|
|
|
/** |
|
* Mark a prop type as declared |
|
* @param {ASTNode} node The AST node being checked. |
|
*/ |
|
function markDisplayNameAsDeclared(node) { |
|
components.set(node, { |
|
hasDisplayName: true |
|
}); |
|
} |
|
|
|
/** |
|
* Reports missing display name for a given component |
|
* @param {Object} component The component to process |
|
*/ |
|
function reportMissingDisplayName(component) { |
|
context.report({ |
|
node: component.node, |
|
message: MISSING_MESSAGE, |
|
data: { |
|
component: component.name |
|
} |
|
}); |
|
} |
|
|
|
/** |
|
* Checks if the component have a name set by the transpiler |
|
* @param {ASTNode} node The AST node being checked. |
|
* @returns {Boolean} True if component has a name, false if not. |
|
*/ |
|
function hasTranspilerName(node) { |
|
const namedObjectAssignment = ( |
|
node.type === 'ObjectExpression' && |
|
node.parent && |
|
node.parent.parent && |
|
node.parent.parent.type === 'AssignmentExpression' && |
|
( |
|
!node.parent.parent.left.object || |
|
node.parent.parent.left.object.name !== 'module' || |
|
node.parent.parent.left.property.name !== 'exports' |
|
) |
|
); |
|
const namedObjectDeclaration = ( |
|
node.type === 'ObjectExpression' && |
|
node.parent && |
|
node.parent.parent && |
|
node.parent.parent.type === 'VariableDeclarator' |
|
); |
|
const namedClass = ( |
|
(node.type === 'ClassDeclaration' || node.type === 'ClassExpression') && |
|
node.id && |
|
node.id.name |
|
); |
|
|
|
const namedFunctionDeclaration = ( |
|
(node.type === 'FunctionDeclaration' || node.type === 'FunctionExpression') && |
|
node.id && |
|
node.id.name |
|
); |
|
|
|
const namedFunctionExpression = ( |
|
astUtil.isFunctionLikeExpression(node) && |
|
node.parent && |
|
(node.parent.type === 'VariableDeclarator' || node.parent.method === true) && |
|
(!node.parent.parent || !utils.isES5Component(node.parent.parent)) |
|
); |
|
|
|
if ( |
|
namedObjectAssignment || namedObjectDeclaration || |
|
namedClass || |
|
namedFunctionDeclaration || namedFunctionExpression |
|
) { |
|
return true; |
|
} |
|
return false; |
|
} |
|
|
|
// -------------------------------------------------------------------------- |
|
// Public |
|
// -------------------------------------------------------------------------- |
|
|
|
return { |
|
|
|
ClassProperty(node) { |
|
if (!propsUtil.isDisplayNameDeclaration(node)) { |
|
return; |
|
} |
|
markDisplayNameAsDeclared(node); |
|
}, |
|
|
|
MemberExpression(node) { |
|
if (!propsUtil.isDisplayNameDeclaration(node.property)) { |
|
return; |
|
} |
|
const component = utils.getRelatedComponent(node); |
|
if (!component) { |
|
return; |
|
} |
|
markDisplayNameAsDeclared(component.node); |
|
}, |
|
|
|
FunctionExpression(node) { |
|
if (ignoreTranspilerName || !hasTranspilerName(node)) { |
|
return; |
|
} |
|
if (components.get(node)) { |
|
markDisplayNameAsDeclared(node); |
|
} |
|
}, |
|
|
|
FunctionDeclaration(node) { |
|
if (ignoreTranspilerName || !hasTranspilerName(node)) { |
|
return; |
|
} |
|
if (components.get(node)) { |
|
markDisplayNameAsDeclared(node); |
|
} |
|
}, |
|
|
|
ArrowFunctionExpression(node) { |
|
if (ignoreTranspilerName || !hasTranspilerName(node)) { |
|
return; |
|
} |
|
if (components.get(node)) { |
|
markDisplayNameAsDeclared(node); |
|
} |
|
}, |
|
|
|
MethodDefinition(node) { |
|
if (!propsUtil.isDisplayNameDeclaration(node.key)) { |
|
return; |
|
} |
|
markDisplayNameAsDeclared(node); |
|
}, |
|
|
|
ClassExpression(node) { |
|
if (ignoreTranspilerName || !hasTranspilerName(node)) { |
|
return; |
|
} |
|
markDisplayNameAsDeclared(node); |
|
}, |
|
|
|
ClassDeclaration(node) { |
|
if (ignoreTranspilerName || !hasTranspilerName(node)) { |
|
return; |
|
} |
|
markDisplayNameAsDeclared(node); |
|
}, |
|
|
|
ObjectExpression(node) { |
|
if (ignoreTranspilerName || !hasTranspilerName(node)) { |
|
// Search for the displayName declaration |
|
node.properties.forEach((property) => { |
|
if (!property.key || !propsUtil.isDisplayNameDeclaration(property.key)) { |
|
return; |
|
} |
|
markDisplayNameAsDeclared(node); |
|
}); |
|
return; |
|
} |
|
markDisplayNameAsDeclared(node); |
|
}, |
|
|
|
'Program:exit': function () { |
|
const list = components.list(); |
|
// Report missing display name for all components |
|
Object.keys(list).filter(component => !list[component].hasDisplayName).forEach((component) => { |
|
reportMissingDisplayName(list[component]); |
|
}); |
|
} |
|
}; |
|
}) |
|
};
|
|
|