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.

281 lines
11 KiB

4 years ago
  1. /**
  2. * @fileoverview enforce consistent line breaks inside function parentheses
  3. * @author Teddy Katz
  4. */
  5. "use strict";
  6. //------------------------------------------------------------------------------
  7. // Requirements
  8. //------------------------------------------------------------------------------
  9. const astUtils = require("./utils/ast-utils");
  10. //------------------------------------------------------------------------------
  11. // Rule Definition
  12. //------------------------------------------------------------------------------
  13. module.exports = {
  14. meta: {
  15. type: "layout",
  16. docs: {
  17. description: "enforce consistent line breaks inside function parentheses",
  18. category: "Stylistic Issues",
  19. recommended: false,
  20. url: "https://eslint.org/docs/rules/function-paren-newline"
  21. },
  22. fixable: "whitespace",
  23. schema: [
  24. {
  25. oneOf: [
  26. {
  27. enum: ["always", "never", "consistent", "multiline", "multiline-arguments"]
  28. },
  29. {
  30. type: "object",
  31. properties: {
  32. minItems: {
  33. type: "integer",
  34. minimum: 0
  35. }
  36. },
  37. additionalProperties: false
  38. }
  39. ]
  40. }
  41. ],
  42. messages: {
  43. expectedBefore: "Expected newline before ')'.",
  44. expectedAfter: "Expected newline after '('.",
  45. expectedBetween: "Expected newline between arguments/params.",
  46. unexpectedBefore: "Unexpected newline before ')'.",
  47. unexpectedAfter: "Unexpected newline after '('."
  48. }
  49. },
  50. create(context) {
  51. const sourceCode = context.getSourceCode();
  52. const rawOption = context.options[0] || "multiline";
  53. const multilineOption = rawOption === "multiline";
  54. const multilineArgumentsOption = rawOption === "multiline-arguments";
  55. const consistentOption = rawOption === "consistent";
  56. let minItems;
  57. if (typeof rawOption === "object") {
  58. minItems = rawOption.minItems;
  59. } else if (rawOption === "always") {
  60. minItems = 0;
  61. } else if (rawOption === "never") {
  62. minItems = Infinity;
  63. } else {
  64. minItems = null;
  65. }
  66. //----------------------------------------------------------------------
  67. // Helpers
  68. //----------------------------------------------------------------------
  69. /**
  70. * Determines whether there should be newlines inside function parens
  71. * @param {ASTNode[]} elements The arguments or parameters in the list
  72. * @param {boolean} hasLeftNewline `true` if the left paren has a newline in the current code.
  73. * @returns {boolean} `true` if there should be newlines inside the function parens
  74. */
  75. function shouldHaveNewlines(elements, hasLeftNewline) {
  76. if (multilineArgumentsOption && elements.length === 1) {
  77. return hasLeftNewline;
  78. }
  79. if (multilineOption || multilineArgumentsOption) {
  80. return elements.some((element, index) => index !== elements.length - 1 && element.loc.end.line !== elements[index + 1].loc.start.line);
  81. }
  82. if (consistentOption) {
  83. return hasLeftNewline;
  84. }
  85. return elements.length >= minItems;
  86. }
  87. /**
  88. * Validates parens
  89. * @param {Object} parens An object with keys `leftParen` for the left paren token, and `rightParen` for the right paren token
  90. * @param {ASTNode[]} elements The arguments or parameters in the list
  91. * @returns {void}
  92. */
  93. function validateParens(parens, elements) {
  94. const leftParen = parens.leftParen;
  95. const rightParen = parens.rightParen;
  96. const tokenAfterLeftParen = sourceCode.getTokenAfter(leftParen);
  97. const tokenBeforeRightParen = sourceCode.getTokenBefore(rightParen);
  98. const hasLeftNewline = !astUtils.isTokenOnSameLine(leftParen, tokenAfterLeftParen);
  99. const hasRightNewline = !astUtils.isTokenOnSameLine(tokenBeforeRightParen, rightParen);
  100. const needsNewlines = shouldHaveNewlines(elements, hasLeftNewline);
  101. if (hasLeftNewline && !needsNewlines) {
  102. context.report({
  103. node: leftParen,
  104. messageId: "unexpectedAfter",
  105. fix(fixer) {
  106. return sourceCode.getText().slice(leftParen.range[1], tokenAfterLeftParen.range[0]).trim()
  107. // If there is a comment between the ( and the first element, don't do a fix.
  108. ? null
  109. : fixer.removeRange([leftParen.range[1], tokenAfterLeftParen.range[0]]);
  110. }
  111. });
  112. } else if (!hasLeftNewline && needsNewlines) {
  113. context.report({
  114. node: leftParen,
  115. messageId: "expectedAfter",
  116. fix: fixer => fixer.insertTextAfter(leftParen, "\n")
  117. });
  118. }
  119. if (hasRightNewline && !needsNewlines) {
  120. context.report({
  121. node: rightParen,
  122. messageId: "unexpectedBefore",
  123. fix(fixer) {
  124. return sourceCode.getText().slice(tokenBeforeRightParen.range[1], rightParen.range[0]).trim()
  125. // If there is a comment between the last element and the ), don't do a fix.
  126. ? null
  127. : fixer.removeRange([tokenBeforeRightParen.range[1], rightParen.range[0]]);
  128. }
  129. });
  130. } else if (!hasRightNewline && needsNewlines) {
  131. context.report({
  132. node: rightParen,
  133. messageId: "expectedBefore",
  134. fix: fixer => fixer.insertTextBefore(rightParen, "\n")
  135. });
  136. }
  137. }
  138. /**
  139. * Validates a list of arguments or parameters
  140. * @param {Object} parens An object with keys `leftParen` for the left paren token, and `rightParen` for the right paren token
  141. * @param {ASTNode[]} elements The arguments or parameters in the list
  142. * @returns {void}
  143. */
  144. function validateArguments(parens, elements) {
  145. const leftParen = parens.leftParen;
  146. const tokenAfterLeftParen = sourceCode.getTokenAfter(leftParen);
  147. const hasLeftNewline = !astUtils.isTokenOnSameLine(leftParen, tokenAfterLeftParen);
  148. const needsNewlines = shouldHaveNewlines(elements, hasLeftNewline);
  149. for (let i = 0; i <= elements.length - 2; i++) {
  150. const currentElement = elements[i];
  151. const nextElement = elements[i + 1];
  152. const hasNewLine = currentElement.loc.end.line !== nextElement.loc.start.line;
  153. if (!hasNewLine && needsNewlines) {
  154. context.report({
  155. node: currentElement,
  156. messageId: "expectedBetween",
  157. fix: fixer => fixer.insertTextBefore(nextElement, "\n")
  158. });
  159. }
  160. }
  161. }
  162. /**
  163. * Gets the left paren and right paren tokens of a node.
  164. * @param {ASTNode} node The node with parens
  165. * @returns {Object} An object with keys `leftParen` for the left paren token, and `rightParen` for the right paren token.
  166. * Can also return `null` if an expression has no parens (e.g. a NewExpression with no arguments, or an ArrowFunctionExpression
  167. * with a single parameter)
  168. */
  169. function getParenTokens(node) {
  170. switch (node.type) {
  171. case "NewExpression":
  172. if (!node.arguments.length && !(
  173. astUtils.isOpeningParenToken(sourceCode.getLastToken(node, { skip: 1 })) &&
  174. astUtils.isClosingParenToken(sourceCode.getLastToken(node))
  175. )) {
  176. // If the NewExpression does not have parens (e.g. `new Foo`), return null.
  177. return null;
  178. }
  179. // falls through
  180. case "CallExpression":
  181. return {
  182. leftParen: sourceCode.getTokenAfter(node.callee, astUtils.isOpeningParenToken),
  183. rightParen: sourceCode.getLastToken(node)
  184. };
  185. case "FunctionDeclaration":
  186. case "FunctionExpression": {
  187. const leftParen = sourceCode.getFirstToken(node, astUtils.isOpeningParenToken);
  188. const rightParen = node.params.length
  189. ? sourceCode.getTokenAfter(node.params[node.params.length - 1], astUtils.isClosingParenToken)
  190. : sourceCode.getTokenAfter(leftParen);
  191. return { leftParen, rightParen };
  192. }
  193. case "ArrowFunctionExpression": {
  194. const firstToken = sourceCode.getFirstToken(node);
  195. if (!astUtils.isOpeningParenToken(firstToken)) {
  196. // If the ArrowFunctionExpression has a single param without parens, return null.
  197. return null;
  198. }
  199. return {
  200. leftParen: firstToken,
  201. rightParen: sourceCode.getTokenBefore(node.body, astUtils.isClosingParenToken)
  202. };
  203. }
  204. case "ImportExpression": {
  205. const leftParen = sourceCode.getFirstToken(node, 1);
  206. const rightParen = sourceCode.getLastToken(node);
  207. return { leftParen, rightParen };
  208. }
  209. default:
  210. throw new TypeError(`unexpected node with type ${node.type}`);
  211. }
  212. }
  213. //----------------------------------------------------------------------
  214. // Public
  215. //----------------------------------------------------------------------
  216. return {
  217. [[
  218. "ArrowFunctionExpression",
  219. "CallExpression",
  220. "FunctionDeclaration",
  221. "FunctionExpression",
  222. "ImportExpression",
  223. "NewExpression"
  224. ]](node) {
  225. const parens = getParenTokens(node);
  226. let params;
  227. if (node.type === "ImportExpression") {
  228. params = [node.source];
  229. } else if (astUtils.isFunction(node)) {
  230. params = node.params;
  231. } else {
  232. params = node.arguments;
  233. }
  234. if (parens) {
  235. validateParens(parens, params);
  236. if (multilineArgumentsOption) {
  237. validateArguments(parens, params);
  238. }
  239. }
  240. }
  241. };
  242. }
  243. };