blob: dc7cca4c903c58fcb52e2c846f1b22c062337f59 [file] [log] [blame]
/*
* Copyright 2003-2008 Dave Griffith, Bas Leijdekkers
*
* 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 com.siyeh.ipp.increment;
import com.intellij.openapi.project.Project;
import com.intellij.psi.*;
import com.intellij.psi.codeStyle.JavaCodeStyleManager;
import com.intellij.psi.search.LocalSearchScope;
import com.intellij.psi.search.SearchScope;
import com.intellij.psi.util.PsiTreeUtil;
import com.intellij.util.IncorrectOperationException;
import com.siyeh.IntentionPowerPackBundle;
import com.siyeh.ig.PsiReplacementUtil;
import com.siyeh.ipp.base.MutablyNamedIntention;
import com.siyeh.ipp.base.PsiElementPredicate;
import org.jetbrains.annotations.NotNull;
import org.jetbrains.annotations.Nullable;
public class ExtractIncrementIntention extends MutablyNamedIntention {
@Override
public String getTextForElement(PsiElement element) {
final PsiJavaToken sign;
if (element instanceof PsiPostfixExpression) {
sign = ((PsiPostfixExpression)element).getOperationSign();
}
else {
sign = ((PsiPrefixExpression)element).getOperationSign();
}
final String operator = sign.getText();
return IntentionPowerPackBundle.message(
"extract.increment.intention.name", operator);
}
@Override
@NotNull
public PsiElementPredicate getElementPredicate() {
return new ExtractIncrementPredicate();
}
@Override
public void processIntention(@NotNull PsiElement element)
throws IncorrectOperationException {
final PsiExpression operand;
if (element instanceof PsiPostfixExpression) {
final PsiPostfixExpression postfixExpression =
(PsiPostfixExpression)element;
operand = postfixExpression.getOperand();
}
else {
final PsiPrefixExpression prefixExpression =
(PsiPrefixExpression)element;
operand = prefixExpression.getOperand();
}
if (operand == null) {
return;
}
final PsiStatement statement =
PsiTreeUtil.getParentOfType(element, PsiStatement.class);
if (statement == null) {
return;
}
final PsiElement parent = statement.getParent();
if (parent == null) {
return;
}
final Project project = element.getProject();
final PsiElementFactory factory =
JavaPsiFacade.getInstance(project).getElementFactory();
final String newStatementText = element.getText() + ';';
final String operandText = operand.getText();
if (parent instanceof PsiIfStatement ||
parent instanceof PsiLoopStatement) {
// need to add braces because
// in/decrement is inside braceless control statement body
final StringBuilder text = new StringBuilder();
text.append('{');
final String elementText =
getElementText(statement, element, operandText);
if (element instanceof PsiPostfixExpression) {
text.append(elementText);
text.append(newStatementText);
}
else {
text.append(newStatementText);
text.append(elementText);
}
text.append('}');
final PsiCodeBlock codeBlock =
factory.createCodeBlockFromText(text.toString(), parent);
statement.replace(codeBlock);
return;
}
final PsiStatement newStatement =
factory.createStatementFromText(newStatementText, element);
if (statement instanceof PsiReturnStatement) {
if (element instanceof PsiPostfixExpression) {
// special handling of postfix expression in return statement
final PsiReturnStatement returnStatement =
(PsiReturnStatement)statement;
final PsiExpression returnValue =
returnStatement.getReturnValue();
if (returnValue == null) {
return;
}
final JavaCodeStyleManager javaCodeStyleManager =
JavaCodeStyleManager.getInstance(project);
final String variableName =
javaCodeStyleManager.suggestUniqueVariableName(
"result", returnValue, true);
final PsiType type = returnValue.getType();
if (type == null) {
return;
}
final String newReturnValueText = getElementText(
returnValue, element, operandText);
final String declarationStatementText =
type.getCanonicalText() + ' ' + variableName +
'=' + newReturnValueText + ';';
final PsiStatement declarationStatement =
factory.createStatementFromText(declarationStatementText,
returnStatement);
parent.addBefore(declarationStatement, statement);
parent.addBefore(newStatement, statement);
final PsiStatement newReturnStatement =
factory.createStatementFromText(
"return " + variableName + ';',
returnStatement);
returnStatement.replace(newReturnStatement);
return;
}
else {
parent.addBefore(newStatement, statement);
}
}
else if (statement instanceof PsiThrowStatement) {
if (element instanceof PsiPostfixExpression) {
// special handling of postfix expression in throw statement
final PsiThrowStatement returnStatement =
(PsiThrowStatement)statement;
final PsiExpression exception =
returnStatement.getException();
if (exception == null) {
return;
}
final JavaCodeStyleManager javaCodeStyleManager =
JavaCodeStyleManager.getInstance(project);
final String variableName =
javaCodeStyleManager.suggestUniqueVariableName(
"e", exception, true);
final PsiType type = exception.getType();
if (type == null) {
return;
}
final String newReturnValueText = getElementText(
exception, element, operandText);
final String declarationStatementText =
type.getCanonicalText() + ' ' + variableName +
'=' + newReturnValueText + ';';
final PsiStatement declarationStatement =
factory.createStatementFromText(declarationStatementText,
returnStatement);
parent.addBefore(declarationStatement, statement);
parent.addBefore(newStatement, statement);
final PsiStatement newReturnStatement =
factory.createStatementFromText(
"throw " + variableName + ';',
returnStatement);
returnStatement.replace(newReturnStatement);
return;
}
else {
parent.addBefore(newStatement, statement);
}
}
else if (!(statement instanceof PsiForStatement)) {
if (element instanceof PsiPostfixExpression) {
parent.addAfter(newStatement, statement);
}
else {
parent.addBefore(newStatement, statement);
}
}
else if (operand instanceof PsiReferenceExpression) {
final PsiReferenceExpression referenceExpression =
(PsiReferenceExpression)operand;
final PsiElement target = referenceExpression.resolve();
if (target != null) {
final SearchScope useScope = target.getUseScope();
if (!new LocalSearchScope(statement).equals(useScope)) {
if (element instanceof PsiPostfixExpression) {
parent.addAfter(newStatement, statement);
}
else {
parent.addBefore(newStatement, statement);
}
}
}
}
if (statement instanceof PsiLoopStatement) {
// in/decrement inside loop statement condition
final PsiLoopStatement loopStatement = (PsiLoopStatement)statement;
final PsiStatement body = loopStatement.getBody();
if (body instanceof PsiBlockStatement) {
final PsiBlockStatement blockStatement =
(PsiBlockStatement)body;
final PsiCodeBlock codeBlock = blockStatement.getCodeBlock();
if (element instanceof PsiPostfixExpression) {
final PsiElement firstElement =
codeBlock.getFirstBodyElement();
codeBlock.addBefore(newStatement, firstElement);
}
else {
codeBlock.add(newStatement);
}
}
else {
final StringBuilder blockText = new StringBuilder();
blockText.append('{');
if (element instanceof PsiPostfixExpression) {
blockText.append(newStatementText);
if (body != null) {
blockText.append(body.getText());
}
}
else {
if (body != null) {
blockText.append(body.getText());
}
blockText.append(newStatementText);
}
blockText.append('}');
final PsiStatement blockStatement =
factory.createStatementFromText(blockText.toString(),
statement);
if (body == null) {
loopStatement.add(blockStatement);
}
else {
body.replace(blockStatement);
}
}
}
PsiReplacementUtil.replaceExpression((PsiExpression)element, operandText);
}
private static String getElementText(@NotNull PsiElement element,
@Nullable PsiElement elementToReplace,
@Nullable String replacement) {
final StringBuilder out = new StringBuilder();
getElementText(element, elementToReplace, replacement, out);
return out.toString();
}
private static void getElementText(
@NotNull PsiElement element,
@Nullable PsiElement elementToReplace,
@Nullable String replacement,
@NotNull StringBuilder out) {
if (element.equals(elementToReplace)) {
out.append(replacement);
return;
}
final PsiElement[] children = element.getChildren();
if (children.length == 0) {
out.append(element.getText());
return;
}
for (PsiElement child : children) {
getElementText(child, elementToReplace, replacement, out);
}
}
}