UNPKG

3.24 kBJavaScriptView Raw
1/**
2 * @fileoverview Validates newlines before and after dots
3 * @author Greg Cochard
4 */
5
6"use strict";
7
8const astUtils = require("../ast-utils");
9
10//------------------------------------------------------------------------------
11// Rule Definition
12//------------------------------------------------------------------------------
13
14module.exports = {
15 meta: {
16 docs: {
17 description: "enforce consistent newlines before and after dots",
18 category: "Best Practices",
19 recommended: false,
20 url: "https://eslint.org/docs/rules/dot-location"
21 },
22
23 schema: [
24 {
25 enum: ["object", "property"]
26 }
27 ],
28
29 fixable: "code",
30
31 messages: {
32 expectedDotAfterObject: "Expected dot to be on same line as object.",
33 expectedDotBeforeProperty: "Expected dot to be on same line as property."
34 }
35 },
36
37 create(context) {
38
39 const config = context.options[0];
40
41 // default to onObject if no preference is passed
42 const onObject = config === "object" || !config;
43
44 const sourceCode = context.getSourceCode();
45
46 /**
47 * Reports if the dot between object and property is on the correct loccation.
48 * @param {ASTNode} obj The object owning the property.
49 * @param {ASTNode} prop The property of the object.
50 * @param {ASTNode} node The corresponding node of the token.
51 * @returns {void}
52 */
53 function checkDotLocation(obj, prop, node) {
54 const dot = sourceCode.getTokenBefore(prop);
55 const textBeforeDot = sourceCode.getText().slice(obj.range[1], dot.range[0]);
56 const textAfterDot = sourceCode.getText().slice(dot.range[1], prop.range[0]);
57
58 if (dot.type === "Punctuator" && dot.value === ".") {
59 if (onObject) {
60 if (!astUtils.isTokenOnSameLine(obj, dot)) {
61 const neededTextAfterObj = astUtils.isDecimalInteger(obj) ? " " : "";
62
63 context.report({
64 node,
65 loc: dot.loc.start,
66 messageId: "expectedDotAfterObject",
67 fix: fixer => fixer.replaceTextRange([obj.range[1], prop.range[0]], `${neededTextAfterObj}.${textBeforeDot}${textAfterDot}`)
68 });
69 }
70 } else if (!astUtils.isTokenOnSameLine(dot, prop)) {
71 context.report({
72 node,
73 loc: dot.loc.start,
74 messageId: "expectedDotBeforeProperty",
75 fix: fixer => fixer.replaceTextRange([obj.range[1], prop.range[0]], `${textBeforeDot}${textAfterDot}.`)
76 });
77 }
78 }
79 }
80
81 /**
82 * Checks the spacing of the dot within a member expression.
83 * @param {ASTNode} node The node to check.
84 * @returns {void}
85 */
86 function checkNode(node) {
87 checkDotLocation(node.object, node.property, node);
88 }
89
90 return {
91 MemberExpression: checkNode
92 };
93 }
94};