123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254 |
- '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: {
- type: 'layout',
- docs: {
- url: 'https://github.com/standard/eslint-plugin-standard#rules-explanations'
- }
- },
- create: function (context) {
- const spaced = context.options[0] === 'always'
- const 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
- }
- const options = {
- spaced,
- 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) {
- const expectedSpace = start[1].range[0] - start[0].range[1]
- const 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) {
- const 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
- }
- const firstSpecifier = node.properties[0]
- const lastSpecifier = node.properties[node.properties.length - 1]
- const first = context.getTokenBefore(firstSpecifier)
- const second = context.getFirstToken(firstSpecifier)
- let penultimate = context.getLastToken(lastSpecifier)
- let 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) {
- const firstSpecifier = node.specifiers[0]
- const 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') {
- const first = context.getTokenBefore(firstSpecifier)
- const second = context.getFirstToken(firstSpecifier)
- const penultimate = context.getLastToken(lastSpecifier)
- const last = context.getTokenAfter(lastSpecifier)
- validateBraceSpacing(node, first, second, penultimate, last)
- }
- },
- // export {name} from 'yo'
- ExportNamedDeclaration: function (node) {
- if (!node.specifiers.length) {
- return
- }
- const firstSpecifier = node.specifiers[0]
- const lastSpecifier = node.specifiers[node.specifiers.length - 1]
- const first = context.getTokenBefore(firstSpecifier)
- const second = context.getFirstToken(firstSpecifier)
- const penultimate = context.getLastToken(lastSpecifier)
- const last = context.getTokenAfter(lastSpecifier)
- validateBraceSpacing(node, first, second, penultimate, last)
- },
- // var y = {x: 'y'}
- ObjectExpression: function (node) {
- if (node.properties.length === 0) {
- return
- }
- const first = context.getFirstToken(node)
- const second = context.getFirstToken(node, 1)
- const penultimate = context.getLastToken(node, 1)
- const last = context.getLastToken(node)
- validateBraceSpacing(node, first, second, penultimate, last)
- }
- }
- }
- }
|