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.
252 lines
8.9 KiB
252 lines
8.9 KiB
'use strict' |
|
|
|
/** |
|
* @fileoverview Disallows or enforces spaces inside of object literals. |
|
* @author Jamund Ferguson |
|
* @copyright 2014 Brandyn Bennett. All rights reserved. |
|
* @copyright 2014 Michael Ficarra. No rights reserved. |
|
* @copyright 2014 Vignesh Anand. All rights reserved. |
|
* @copyright 2015 Jamund Ferguson. All rights reserved. |
|
*/ |
|
// ------------------------------------------------------------------------------ |
|
// Rule Definition |
|
// ------------------------------------------------------------------------------ |
|
|
|
module.exports = { |
|
meta: { |
|
docs: { |
|
url: 'https://github.com/standard/eslint-plugin-standard#rules-explanations' |
|
} |
|
}, |
|
|
|
create: function (context) { |
|
var spaced = context.options[0] === 'always' |
|
var either = context.options[0] === 'either' |
|
|
|
/** |
|
* Determines whether an option is set, relative to the spacing option. |
|
* If spaced is "always", then check whether option is set to false. |
|
* If spaced is "never", then check whether option is set to true. |
|
* @param {Object} option - The option to exclude. |
|
* @returns {boolean} Whether or not the property is excluded. |
|
*/ |
|
function isOptionSet (option) { |
|
return context.options[1] != null ? context.options[1][option] === !spaced : false |
|
} |
|
|
|
var options = { |
|
spaced: spaced, |
|
either: either, |
|
arraysInObjectsException: isOptionSet('arraysInObjects'), |
|
objectsInObjectsException: isOptionSet('objectsInObjects') |
|
} |
|
|
|
// -------------------------------------------------------------------------- |
|
// Helpers |
|
// -------------------------------------------------------------------------- |
|
|
|
/** |
|
* Determines whether two adjacent tokens are have whitespace between them. |
|
* @param {Object} left - The left token object. |
|
* @param {Object} right - The right token object. |
|
* @returns {boolean} Whether or not there is space between the tokens. |
|
*/ |
|
function isSpaced (left, right) { |
|
return left.range[1] < right.range[0] |
|
} |
|
|
|
/** |
|
* Determines whether two adjacent tokens are on the same line. |
|
* @param {Object} left - The left token object. |
|
* @param {Object} right - The right token object. |
|
* @returns {boolean} Whether or not the tokens are on the same line. |
|
*/ |
|
function isSameLine (left, right) { |
|
return left.loc.start.line === right.loc.start.line |
|
} |
|
|
|
/** |
|
* Reports that there shouldn't be a space after the first token |
|
* @param {ASTNode} node - The node to report in the event of an error. |
|
* @param {Token} token - The token to use for the report. |
|
* @returns {void} |
|
*/ |
|
function reportNoBeginningSpace (node, token) { |
|
context.report(node, token.loc.start, |
|
"There should be no space after '" + token.value + "'") |
|
} |
|
|
|
/** |
|
* Reports that there shouldn't be a space before the last token |
|
* @param {ASTNode} node - The node to report in the event of an error. |
|
* @param {Token} token - The token to use for the report. |
|
* @returns {void} |
|
*/ |
|
function reportNoEndingSpace (node, token) { |
|
context.report(node, token.loc.start, |
|
"There should be no space before '" + token.value + "'") |
|
} |
|
|
|
/** |
|
* Reports that there should be a space after the first token |
|
* @param {ASTNode} node - The node to report in the event of an error. |
|
* @param {Token} token - The token to use for the report. |
|
* @returns {void} |
|
*/ |
|
function reportRequiredBeginningSpace (node, token) { |
|
context.report(node, token.loc.start, |
|
"A space is required after '" + token.value + "'") |
|
} |
|
|
|
/** |
|
* Reports that there should be a space before the last token |
|
* @param {ASTNode} node - The node to report in the event of an error. |
|
* @param {Token} token - The token to use for the report. |
|
* @returns {void} |
|
*/ |
|
function reportRequiredEndingSpace (node, token) { |
|
context.report(node, token.loc.start, |
|
"A space is required before '" + token.value + "'") |
|
} |
|
|
|
/** |
|
* Checks if a start and end brace in a node are spaced evenly |
|
* and not too long (>1 space) |
|
* @param node |
|
* @param start |
|
* @param end |
|
* @returns {boolean} |
|
*/ |
|
function isEvenlySpacedAndNotTooLong (node, start, end) { |
|
var expectedSpace = start[1].range[0] - start[0].range[1] |
|
var endSpace = end[1].range[0] - end[0].range[1] |
|
return endSpace === expectedSpace && endSpace <= 1 |
|
} |
|
|
|
/** |
|
* Determines if spacing in curly braces is valid. |
|
* @param {ASTNode} node The AST node to check. |
|
* @param {Token} first The first token to check (should be the opening brace) |
|
* @param {Token} second The second token to check (should be first after the opening brace) |
|
* @param {Token} penultimate The penultimate token to check (should be last before closing brace) |
|
* @param {Token} last The last token to check (should be closing brace) |
|
* @returns {void} |
|
*/ |
|
function validateBraceSpacing (node, first, second, penultimate, last) { |
|
var closingCurlyBraceMustBeSpaced = |
|
(options.arraysInObjectsException && penultimate.value === ']') || |
|
(options.objectsInObjectsException && penultimate.value === '}') |
|
? !options.spaced : options.spaced |
|
|
|
// we only care about evenly spaced things |
|
if (options.either) { |
|
// newlines at any point means return |
|
if (!isSameLine(first, last)) { |
|
return |
|
} |
|
|
|
// confirm that the object expression/literal is spaced evenly |
|
if (!isEvenlySpacedAndNotTooLong(node, [first, second], [penultimate, last])) { |
|
context.report(node, 'Expected consistent spacing') |
|
} |
|
|
|
return |
|
} |
|
|
|
// { and key are on same line |
|
if (isSameLine(first, second)) { |
|
if (options.spaced && !isSpaced(first, second)) { |
|
reportRequiredBeginningSpace(node, first) |
|
} |
|
if (!options.spaced && isSpaced(first, second)) { |
|
reportNoBeginningSpace(node, first) |
|
} |
|
} |
|
|
|
// final key and } ore on the same line |
|
if (isSameLine(penultimate, last)) { |
|
if (closingCurlyBraceMustBeSpaced && !isSpaced(penultimate, last)) { |
|
reportRequiredEndingSpace(node, last) |
|
} |
|
if (!closingCurlyBraceMustBeSpaced && isSpaced(penultimate, last)) { |
|
reportNoEndingSpace(node, last) |
|
} |
|
} |
|
} |
|
|
|
// -------------------------------------------------------------------------- |
|
// Public |
|
// -------------------------------------------------------------------------- |
|
|
|
return { |
|
// var {x} = y |
|
ObjectPattern: function (node) { |
|
if (node.properties.length === 0) { |
|
return |
|
} |
|
|
|
var firstSpecifier = node.properties[0] |
|
var lastSpecifier = node.properties[node.properties.length - 1] |
|
|
|
var first = context.getTokenBefore(firstSpecifier) |
|
var second = context.getFirstToken(firstSpecifier) |
|
var penultimate = context.getLastToken(lastSpecifier) |
|
var last = context.getTokenAfter(lastSpecifier) |
|
|
|
// support trailing commas |
|
if (last.value === ',') { |
|
penultimate = last |
|
last = context.getTokenAfter(last) |
|
} |
|
|
|
validateBraceSpacing(node, first, second, penultimate, last) |
|
}, |
|
|
|
// import {y} from 'x' |
|
ImportDeclaration: function (node) { |
|
var firstSpecifier = node.specifiers[0] |
|
var lastSpecifier = node.specifiers[node.specifiers.length - 1] |
|
|
|
// don't do anything for namespace or default imports |
|
if (firstSpecifier && lastSpecifier && firstSpecifier.type === 'ImportSpecifier' && lastSpecifier.type === 'ImportSpecifier') { |
|
var first = context.getTokenBefore(firstSpecifier) |
|
var second = context.getFirstToken(firstSpecifier) |
|
var penultimate = context.getLastToken(lastSpecifier) |
|
var last = context.getTokenAfter(lastSpecifier) |
|
|
|
validateBraceSpacing(node, first, second, penultimate, last) |
|
} |
|
}, |
|
|
|
// export {name} from 'yo' |
|
ExportNamedDeclaration: function (node) { |
|
if (!node.specifiers.length) { |
|
return |
|
} |
|
|
|
var firstSpecifier = node.specifiers[0] |
|
var lastSpecifier = node.specifiers[node.specifiers.length - 1] |
|
var first = context.getTokenBefore(firstSpecifier) |
|
var second = context.getFirstToken(firstSpecifier) |
|
var penultimate = context.getLastToken(lastSpecifier) |
|
var last = context.getTokenAfter(lastSpecifier) |
|
|
|
validateBraceSpacing(node, first, second, penultimate, last) |
|
}, |
|
|
|
// var y = {x: 'y'} |
|
ObjectExpression: function (node) { |
|
if (node.properties.length === 0) { |
|
return |
|
} |
|
|
|
var first = context.getFirstToken(node) |
|
var second = context.getFirstToken(node, 1) |
|
var penultimate = context.getLastToken(node, 1) |
|
var last = context.getLastToken(node) |
|
|
|
validateBraceSpacing(node, first, second, penultimate, last) |
|
} |
|
} |
|
} |
|
}
|
|
|