1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20 package com.puppycrawl.tools.checkstyle.checks.whitespace;
21
22 import java.util.Optional;
23
24 import com.puppycrawl.tools.checkstyle.StatelessCheck;
25 import com.puppycrawl.tools.checkstyle.api.AbstractCheck;
26 import com.puppycrawl.tools.checkstyle.api.DetailAST;
27 import com.puppycrawl.tools.checkstyle.api.TokenTypes;
28 import com.puppycrawl.tools.checkstyle.utils.CommonUtil;
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69 @StatelessCheck
70 public class NoWhitespaceAfterCheck extends AbstractCheck {
71
72
73
74
75
76 public static final String MSG_KEY = "ws.followed";
77
78
79 private boolean allowLineBreaks = true;
80
81 @Override
82 public int[] getDefaultTokens() {
83 return new int[] {
84 TokenTypes.ARRAY_INIT,
85 TokenTypes.AT,
86 TokenTypes.INC,
87 TokenTypes.DEC,
88 TokenTypes.UNARY_MINUS,
89 TokenTypes.UNARY_PLUS,
90 TokenTypes.BNOT,
91 TokenTypes.LNOT,
92 TokenTypes.DOT,
93 TokenTypes.ARRAY_DECLARATOR,
94 TokenTypes.INDEX_OP,
95 };
96 }
97
98 @Override
99 public int[] getAcceptableTokens() {
100 return new int[] {
101 TokenTypes.ARRAY_INIT,
102 TokenTypes.AT,
103 TokenTypes.INC,
104 TokenTypes.DEC,
105 TokenTypes.UNARY_MINUS,
106 TokenTypes.UNARY_PLUS,
107 TokenTypes.BNOT,
108 TokenTypes.LNOT,
109 TokenTypes.DOT,
110 TokenTypes.TYPECAST,
111 TokenTypes.ARRAY_DECLARATOR,
112 TokenTypes.INDEX_OP,
113 TokenTypes.LITERAL_SYNCHRONIZED,
114 TokenTypes.METHOD_REF,
115 };
116 }
117
118 @Override
119 public int[] getRequiredTokens() {
120 return CommonUtil.EMPTY_INT_ARRAY;
121 }
122
123
124
125
126
127
128
129
130 public void setAllowLineBreaks(boolean allowLineBreaks) {
131 this.allowLineBreaks = allowLineBreaks;
132 }
133
134 @Override
135 public void visitToken(DetailAST ast) {
136 if (shouldCheckWhitespaceAfter(ast)) {
137 final DetailAST whitespaceFollowedAst = getWhitespaceFollowedNode(ast);
138 final int whitespaceColumnNo = getPositionAfter(whitespaceFollowedAst);
139 final int whitespaceLineNo = whitespaceFollowedAst.getLineNo();
140
141 if (hasTrailingWhitespace(ast, whitespaceColumnNo, whitespaceLineNo)) {
142 log(ast, MSG_KEY, whitespaceFollowedAst.getText());
143 }
144 }
145 }
146
147
148
149
150
151
152
153
154
155 private static DetailAST getWhitespaceFollowedNode(DetailAST ast) {
156 return switch (ast.getType()) {
157 case TokenTypes.TYPECAST -> ast.findFirstToken(TokenTypes.RPAREN);
158 case TokenTypes.ARRAY_DECLARATOR -> getArrayDeclaratorPreviousElement(ast);
159 case TokenTypes.INDEX_OP -> getIndexOpPreviousElement(ast);
160 default -> ast;
161 };
162 }
163
164
165
166
167
168
169
170
171
172 private static boolean shouldCheckWhitespaceAfter(DetailAST ast) {
173 final DetailAST previousSibling = ast.getPreviousSibling();
174 final boolean isSynchronizedMethod = ast.getType() == TokenTypes.LITERAL_SYNCHRONIZED
175 && ast.getFirstChild() == null;
176 return !isSynchronizedMethod
177 && (previousSibling == null || previousSibling.getType() != TokenTypes.ANNOTATIONS);
178 }
179
180
181
182
183
184
185
186 private static int getPositionAfter(DetailAST ast) {
187 final int after;
188
189 if (ast.getType() == TokenTypes.IDENT
190 && ast.getNextSibling() != null
191 && ast.getNextSibling().getType() == TokenTypes.LPAREN) {
192 final DetailAST methodDef = ast.getParent();
193 final DetailAST endOfParams = methodDef.findFirstToken(TokenTypes.RPAREN);
194 after = endOfParams.getColumnNo() + 1;
195 }
196 else {
197 after = ast.getColumnNo() + ast.getText().length();
198 }
199 return after;
200 }
201
202
203
204
205
206
207
208
209
210
211
212
213 private boolean hasTrailingWhitespace(DetailAST ast,
214 int whitespaceColumnNo, int whitespaceLineNo) {
215 final boolean result;
216 final int astLineNo = ast.getLineNo();
217 final int[] line = getLineCodePoints(astLineNo - 1);
218 if (astLineNo == whitespaceLineNo && whitespaceColumnNo < line.length) {
219 result = CommonUtil.isCodePointWhitespace(line, whitespaceColumnNo);
220 }
221 else {
222 result = !allowLineBreaks;
223 }
224 return result;
225 }
226
227
228
229
230
231
232
233
234
235
236
237 private static DetailAST getArrayDeclaratorPreviousElement(DetailAST ast) {
238 final DetailAST previousElement;
239
240 if (ast.getPreviousSibling() != null
241 && ast.getPreviousSibling().getType() == TokenTypes.ARRAY_DECLARATOR) {
242
243 previousElement = getPreviousElementOfMultiDimArray(ast);
244 }
245 else {
246
247 final DetailAST parent = ast.getParent();
248
249 previousElement = switch (parent.getType()) {
250
251 case TokenTypes.TYPE_UPPER_BOUNDS, TokenTypes.TYPE_LOWER_BOUNDS ->
252 ast.getPreviousSibling();
253
254 case TokenTypes.LITERAL_NEW, TokenTypes.TYPE_ARGUMENT, TokenTypes.DOT ->
255 getTypeLastNode(ast);
256
257
258 case TokenTypes.TYPE -> getPreviousNodeWithParentOfTypeAst(ast, parent);
259
260
261 case TokenTypes.METHOD_REF -> {
262 final DetailAST ident = getIdentLastToken(ast);
263 if (ident == null) {
264
265 yield ast.getParent().getFirstChild();
266 }
267 yield ident;
268 }
269
270 default -> throw new IllegalStateException("unexpected ast syntax " + parent);
271 };
272 }
273
274 return previousElement;
275 }
276
277
278
279
280
281
282
283
284 private static DetailAST getPreviousElementOfMultiDimArray(DetailAST leftBracket) {
285 final DetailAST previousRightBracket = leftBracket.getPreviousSibling().getLastChild();
286
287 DetailAST ident = null;
288
289 DetailAST parent = leftBracket.getParent().getParent();
290 while (ident == null) {
291 ident = parent.findFirstToken(TokenTypes.IDENT);
292 parent = parent.getParent();
293 }
294
295 final DetailAST previousElement;
296 if (ident.getColumnNo() > previousRightBracket.getColumnNo()
297 && ident.getColumnNo() < leftBracket.getColumnNo()) {
298
299 previousElement = ident;
300 }
301 else {
302
303 previousElement = previousRightBracket;
304 }
305 return previousElement;
306 }
307
308
309
310
311
312
313
314
315
316
317 private static DetailAST getIndexOpPreviousElement(DetailAST ast) {
318 final DetailAST result;
319 final DetailAST firstChild = ast.getFirstChild();
320 if (firstChild.getType() == TokenTypes.INDEX_OP) {
321
322 result = firstChild.findFirstToken(TokenTypes.RBRACK);
323 }
324 else if (firstChild.getType() == TokenTypes.IDENT) {
325 result = firstChild;
326 }
327 else {
328 final DetailAST ident = getIdentLastToken(ast);
329 if (ident == null) {
330 final DetailAST rparen = ast.findFirstToken(TokenTypes.RPAREN);
331
332 if (rparen == null) {
333 final DetailAST lastChild = firstChild.getLastChild();
334 result = lastChild.findFirstToken(TokenTypes.RCURLY);
335 }
336
337 else {
338 result = rparen;
339 }
340 }
341 else {
342 result = ident;
343 }
344 }
345 return result;
346 }
347
348
349
350
351
352
353
354
355
356 private static DetailAST getTypeLastNode(DetailAST ast) {
357 final DetailAST typeLastNode;
358 final DetailAST parent = ast.getParent();
359 final boolean isPrecededByTypeArgs =
360 parent.findFirstToken(TokenTypes.TYPE_ARGUMENTS) != null;
361 final Optional<DetailAST> objectArrayType = Optional.ofNullable(getIdentLastToken(ast));
362
363 if (isPrecededByTypeArgs) {
364 typeLastNode = parent.findFirstToken(TokenTypes.TYPE_ARGUMENTS)
365 .findFirstToken(TokenTypes.GENERIC_END);
366 }
367 else if (objectArrayType.isPresent()) {
368 typeLastNode = objectArrayType.orElseThrow();
369 }
370 else {
371 typeLastNode = parent.getFirstChild();
372 }
373
374 return typeLastNode;
375 }
376
377
378
379
380
381
382
383
384
385
386
387 private static DetailAST getPreviousNodeWithParentOfTypeAst(DetailAST ast, DetailAST parent) {
388 final DetailAST previousElement;
389 final DetailAST ident = getIdentLastToken(parent.getParent());
390 final DetailAST lastTypeNode = getTypeLastNode(ast);
391
392
393
394
395
396 if (ident == null || ident.getLineNo() > ast.getLineNo()) {
397 previousElement = lastTypeNode;
398 }
399 else if (ident.getLineNo() < ast.getLineNo()) {
400 previousElement = ident;
401 }
402
403 else {
404 final int instanceOfSize = 13;
405
406 if (ident.getColumnNo() >= ast.getColumnNo() + 2
407
408
409 || lastTypeNode.getColumnNo() >= ident.getColumnNo() + instanceOfSize) {
410 previousElement = lastTypeNode;
411 }
412 else {
413 previousElement = ident;
414 }
415 }
416 return previousElement;
417 }
418
419
420
421
422
423
424
425
426 private static DetailAST getIdentLastToken(DetailAST ast) {
427 final DetailAST result;
428 final Optional<DetailAST> dot = getPrecedingDot(ast);
429
430 if (dot.isEmpty() || ast.getFirstChild().getType() == TokenTypes.METHOD_CALL) {
431 final DetailAST methodCall = ast.findFirstToken(TokenTypes.METHOD_CALL);
432 if (methodCall == null) {
433 result = ast.findFirstToken(TokenTypes.IDENT);
434 }
435 else {
436 result = methodCall.findFirstToken(TokenTypes.RPAREN);
437 }
438 }
439
440 else {
441 result = dot.orElseThrow().getFirstChild().getNextSibling();
442 }
443 return result;
444 }
445
446
447
448
449
450
451
452
453 private static Optional<DetailAST> getPrecedingDot(DetailAST leftBracket) {
454 final DetailAST referencedMemberDot = leftBracket.findFirstToken(TokenTypes.DOT);
455 final Optional<DetailAST> result = Optional.ofNullable(referencedMemberDot);
456 return result.or(() -> getReferencedClassDot(leftBracket));
457 }
458
459
460
461
462
463
464
465 private static Optional<DetailAST> getReferencedClassDot(DetailAST leftBracket) {
466 final DetailAST parent = leftBracket.getParent();
467 Optional<DetailAST> classDot = Optional.empty();
468 if (parent.getType() != TokenTypes.ASSIGN) {
469 classDot = Optional.ofNullable(parent.findFirstToken(TokenTypes.DOT));
470 }
471 return classDot;
472 }
473 }