Skip to content

Commit

Permalink
feat(rules) object-curly-even-spacing rule (fixes standard#182)
Browse files Browse the repository at this point in the history
  • Loading branch information
Jamund Ferguson committed Jul 17, 2015
1 parent c91a46c commit 337b83d
Show file tree
Hide file tree
Showing 3 changed files with 252 additions and 1 deletion.
1 change: 1 addition & 0 deletions index.js
Expand Up @@ -25,6 +25,7 @@ var DEFAULT_IGNORE_PATTERNS = [

var DEFAULT_CONFIG = {
configFile: path.join(__dirname, 'rc', '.eslintrc'),
rulePaths: [ path.join(__dirname, 'rules') ],
useEslintrc: false
}

Expand Down
5 changes: 4 additions & 1 deletion rc/.eslintrc
@@ -1,3 +1,6 @@
{
"extends": ["standard", "standard-react"]
"extends": ["standard", "standard-react"],
"rules": {
"object-curly-even-spacing": [2, "either"]
}
}
247 changes: 247 additions & 0 deletions rules/object-curly-even-spacing.js
@@ -0,0 +1,247 @@
/**
* @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.
*/
'use strict'

// ------------------------------------------------------------------------------
// Rule Definition
// ------------------------------------------------------------------------------

module.exports = 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 + "'")
}

function isEvenlySpaced (node, start, end) {
var expectedSpace = start[0].range[1] - start[1].range[0]
var endSpace = end[0].range[1] - end[1].range[0]
return endSpace === expectedSpace
}

/**
* 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) {
// make sure newlines are the same on top and on bottom
var startsSame = isSameLine(first, second)
var endsSame = isSameLine(penultimate, last)

// two newlines means return
if (!startsSame && !endsSame) {
return
}

// inconsistent newlines are evil
if (startsSame !== endsSame) {
context.report(node, 'Expected consistent spacing')
return
}

// confirm that the object expression/literal is spaced evenly
if (!isEvenlySpaced(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) {
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)
}

}

}

0 comments on commit 337b83d

Please sign in to comment.