blob: 257ac9975ce1c2ede0f56dcd32b846f7e48faddd [file] [log] [blame]
/*
* Copyright 2000-2013 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.editor.selection;
import com.intellij.codeInsight.editorActions.ExtendWordSelectionHandlerBase;
import com.intellij.openapi.editor.Editor;
import com.intellij.openapi.util.TextRange;
import com.intellij.psi.PsiElement;
import com.intellij.psi.PsiWhiteSpace;
import org.jetbrains.plugins.groovy.lang.lexer.GroovyTokenTypes;
import org.jetbrains.plugins.groovy.lang.psi.api.statements.GrStatement;
import org.jetbrains.plugins.groovy.lang.psi.api.statements.expressions.GrExpression;
import org.jetbrains.plugins.groovy.lang.psi.impl.PsiImplUtil;
import org.jetbrains.plugins.groovy.lang.psi.util.PsiUtil;
import java.util.List;
/**
* @author Max Medvedev
*/
public class GroovyStatementSelectioner extends ExtendWordSelectionHandlerBase {
@Override
public boolean canSelect(PsiElement e) {
return e instanceof GrExpression && PsiUtil.isExpressionStatement(e) ||
(e instanceof GrStatement && !(e instanceof GrExpression)) ||
e.getNode().getElementType() == GroovyTokenTypes.mSEMI;
}
@Override
public List<TextRange> select(PsiElement e, CharSequence editorText, int cursorOffset, Editor editor) {
TextRange originalRange;
PsiElement first;
PsiElement last;
if (e instanceof GrStatement) {
first = e;
PsiElement next = e.getNextSibling();
next = skipWhitespacesForward(next);
if (next != null && next.getNode().getElementType() == GroovyTokenTypes.mSEMI) {
originalRange = new TextRange(e.getTextRange().getStartOffset(), next.getTextRange().getEndOffset());
last = next;
}
else {
originalRange = e.getTextRange();
last = e;
}
}
else {
last = e;
PsiElement prev = e.getPrevSibling();
prev = skipWhitespaceBack(prev);
if (prev instanceof GrStatement) {
originalRange = new TextRange(prev.getTextRange().getStartOffset(), e.getTextRange().getEndOffset());
first = prev;
}
else {
originalRange = e.getTextRange();
first = e;
}
}
final List<TextRange> ranges = ExtendWordSelectionHandlerBase.expandToWholeLine(editorText, originalRange);
final TextRange blockRange = inferBlockRange(first, last);
if (!blockRange.equals(originalRange)) {
ranges.addAll(ExtendWordSelectionHandlerBase.expandToWholeLine(editorText, blockRange, true));
}
return ranges;
}
private static TextRange inferBlockRange(PsiElement first, PsiElement last) {
while (true) {
PsiElement prev = first.getPrevSibling();
prev = skipWhitespaceBack(prev);
if (isOneLineFeed(prev)) prev = prev.getPrevSibling();
prev = skipWhitespaceBack(prev);
if (prev != null && prev.getNode().getElementType() == GroovyTokenTypes.mSEMI || prev instanceof GrStatement) {
first = prev;
}
else {
break;
}
}
while (true) {
PsiElement next = last.getNextSibling();
next = skipWhitespacesForward(next);
if (isOneLineFeed(next)) next = next.getNextSibling();
next = skipWhitespacesForward(next);
if (next != null && next.getNode().getElementType() == GroovyTokenTypes.mSEMI || next instanceof GrStatement) {
last = next;
}
else {
break;
}
}
return new TextRange(first.getTextRange().getStartOffset(), last.getTextRange().getEndOffset());
}
private static PsiElement skipWhitespacesForward(PsiElement next) {
while (isSpaceWithoutLineFeed(next)) next = next.getNextSibling();
return next;
}
private static PsiElement skipWhitespaceBack(PsiElement prev) {
while (isSpaceWithoutLineFeed(prev)) prev = prev.getPrevSibling();
return prev;
}
private static boolean isOneLineFeed(PsiElement e) {
if (e == null) return false;
if (!PsiImplUtil.isWhiteSpaceOrNls(e)) return false;
final String text = e.getText();
final int i = text.indexOf('\n');
return i >= 0 && i == text.lastIndexOf('\n');
}
private static boolean isSpaceWithoutLineFeed(PsiElement e) {
return e instanceof PsiWhiteSpace && e.getText().indexOf('\n') == -1;
}
}