blob: 983354746a48dec07e2b8bffdfdd62386abe5605 [file] [log] [blame]
/*
* Copyright 2000-2014 JetBrains s.r.o.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package org.jetbrains.plugins.groovy.formatter;
import com.intellij.lang.ASTNode;
import com.intellij.openapi.diagnostic.Logger;
import com.intellij.psi.PsiElement;
import com.intellij.psi.TokenType;
import com.intellij.psi.tree.IElementType;
import com.intellij.psi.util.PsiTreeUtil;
import org.jetbrains.annotations.Nullable;
import org.jetbrains.plugins.groovy.lang.lexer.GroovyTokenTypes;
import org.jetbrains.plugins.groovy.lang.parser.GroovyElementTypes;
import org.jetbrains.plugins.groovy.lang.psi.api.statements.blocks.GrClosableBlock;
import org.jetbrains.plugins.groovy.lang.psi.api.statements.expressions.GrExpression;
import org.jetbrains.plugins.groovy.lang.psi.api.statements.expressions.GrMethodCall;
import org.jetbrains.plugins.groovy.lang.psi.api.statements.expressions.GrReferenceExpression;
import org.jetbrains.plugins.groovy.lang.psi.impl.PsiImplUtil;
import org.jetbrains.plugins.groovy.lang.psi.util.PsiUtil;
/**
* @author Max Medvedev
*/
public class GeeseUtil {
private static final Logger LOG = Logger.getInstance(GeeseUtil.class);
private GeeseUtil() {
}
@Nullable
public static ASTNode getClosureRBraceAtTheEnd(ASTNode node) {
IElementType elementType = node.getElementType();
if (elementType == GroovyElementTypes.CLOSABLE_BLOCK) {
PsiElement rBrace = ((GrClosableBlock)node.getPsi()).getRBrace();
return rBrace != null ? rBrace.getNode() : null;
}
ASTNode lastChild = node.getLastChildNode();
while (lastChild != null && PsiImplUtil.isWhiteSpaceOrNls(lastChild)) {
lastChild = lastChild.getTreePrev();
}
if (lastChild == null) return null;
return getClosureRBraceAtTheEnd(lastChild);
}
public static boolean isClosureRBrace(PsiElement e) {
return e != null && e.getNode().getElementType() == GroovyTokenTypes.mRCURLY &&
e.getParent() instanceof GrClosableBlock &&
((GrClosableBlock)e.getParent()).getRBrace() == e;
}
@Nullable
public static PsiElement getNextNonWhitespaceToken(PsiElement e) {
PsiElement next = PsiTreeUtil.nextLeaf(e);
while (next != null && next.getNode().getElementType() == TokenType.WHITE_SPACE) next = PsiTreeUtil.nextLeaf(next);
return next;
}
static void calculateRBraceAlignment(PsiElement rBrace, AlignmentProvider alignments) {
int leadingBraceCount = 0;
PsiElement next;
if (!isClosureContainLF(rBrace)) return;
for (next = PsiUtil.getPreviousNonWhitespaceToken(rBrace);
isClosureRBrace(next) && isClosureContainLF(next);
next = PsiUtil.getPreviousNonWhitespaceToken(next)) {
leadingBraceCount++;
}
PsiElement cur = rBrace;
for (next = getNextNonWhitespaceToken(cur); isClosureRBrace(next); next = getNextNonWhitespaceToken(cur)) {
cur = next;
}
for (; leadingBraceCount > 0; leadingBraceCount--) {
cur = PsiUtil.getPreviousNonWhitespaceToken(cur);
}
PsiElement parent = cur.getParent();
LOG.assertTrue(parent instanceof GrClosableBlock);
//search for start of the line
cur = parent;
if (cur.getParent() instanceof GrMethodCall) {
GrMethodCall call = (GrMethodCall)cur.getParent();
GrExpression invoked = call.getInvokedExpression();
if (invoked instanceof GrReferenceExpression && ((GrReferenceExpression)invoked).getReferenceNameElement() != null) {
cur = ((GrReferenceExpression)invoked).getReferenceNameElement();
}
else {
cur = call;
}
}
cur = PsiTreeUtil.getDeepestFirst(cur);
while (!PsiUtil.isNewLine(next = PsiTreeUtil.prevLeaf(cur, true))) {
if (next == null) break;
if (next.getNode().getElementType() == TokenType.WHITE_SPACE && PsiTreeUtil.prevLeaf(next) == null) {
break; //if cur is first word in the text, whitespace could be before it
}
cur = next;
}
int startOffset = cur.getTextRange().getStartOffset();
int endOffset = rBrace.getTextRange().getStartOffset();
if (rBrace.getContainingFile().getText().substring(startOffset, endOffset).indexOf('\n') < 0) {
return;
}
while (true) {
final PsiElement p = cur.getParent();
if (p != null && p.getTextOffset() == cur.getTextOffset()) {
cur = p;
}
else {
break;
}
}
alignments.addPair(rBrace, cur, true);
}
public static boolean isClosureContainLF(PsiElement rBrace) {
PsiElement parent = rBrace.getParent();
return parent.getText().indexOf('\n') >= 0;
}
}