| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200 | 'use strict'/** * @fileoverview Disallows or enforces spaces inside of array brackets. * @author Jamund Ferguson * @copyright 2015 Jamund Ferguson. All rights reserved. * @copyright 2014 Brandyn Bennett. All rights reserved. * @copyright 2014 Michael Ficarra. No rights reserved. * @copyright 2014 Vignesh Anand. 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 = {      either,      spaced,      singleElementException: isOptionSet('singleValue'),      objectsInArraysException: isOptionSet('objectsInArrays'),      arraysInArraysException: isOptionSet('arraysInArrays')    }    // --------------------------------------------------------------------------    // 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    }    /**     * Validates the spacing around array brackets     * @param {ASTNode} node - The node we're checking for spacing     * @returns {void}     */    function validateArraySpacing (node) {      if (node.elements.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)      const openingBracketMustBeSpaced =      (options.objectsInArraysException && second.value === '{') ||      (options.arraysInArraysException && second.value === '[') ||      (options.singleElementException && node.elements.length === 1)        ? !options.spaced        : options.spaced      const closingBracketMustBeSpaced =      (options.objectsInArraysException && penultimate.value === '}') ||      (options.arraysInArraysException && penultimate.value === ']') ||      (options.singleElementException && node.elements.length === 1)        ? !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      }      if (isSameLine(first, second)) {        if (openingBracketMustBeSpaced && !isSpaced(first, second)) {          reportRequiredBeginningSpace(node, first)        }        if (!openingBracketMustBeSpaced && isSpaced(first, second)) {          reportNoBeginningSpace(node, first)        }      }      if (isSameLine(penultimate, last)) {        if (closingBracketMustBeSpaced && !isSpaced(penultimate, last)) {          reportRequiredEndingSpace(node, last)        }        if (!closingBracketMustBeSpaced && isSpaced(penultimate, last)) {          reportNoEndingSpace(node, last)        }      }    }    // --------------------------------------------------------------------------    // Public    // --------------------------------------------------------------------------    return {      ArrayPattern: validateArraySpacing,      ArrayExpression: validateArraySpacing    }  }}
 |