blob: a83605b3814b9b75467dcd32558619f5ca67f732 [file] [log] [blame]
/*
* Copyright 2000-2009 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 com.intellij.psi.impl.source.tree.java;
import com.intellij.lang.ASTNode;
import com.intellij.openapi.diagnostic.Logger;
import com.intellij.psi.*;
import com.intellij.psi.impl.PsiImplUtil;
import com.intellij.psi.impl.source.SourceTreeToPsiMap;
import com.intellij.psi.impl.source.Constants;
import com.intellij.psi.impl.source.tree.ChildRole;
import com.intellij.psi.impl.source.tree.CompositePsiElement;
import com.intellij.psi.impl.source.tree.TreeElement;
import com.intellij.psi.scope.PsiScopeProcessor;
import com.intellij.psi.scope.util.PsiScopesUtil;
import com.intellij.psi.tree.IElementType;
import com.intellij.psi.tree.ChildRoleBase;
import com.intellij.util.IncorrectOperationException;
import org.jetbrains.annotations.NotNull;
public class PsiForStatementImpl extends CompositePsiElement implements PsiForStatement, Constants {
private static final Logger LOG = Logger.getInstance("#com.intellij.psi.impl.source.tree.java.PsiForStatementImpl");
public PsiForStatementImpl() {
super(FOR_STATEMENT);
}
@Override
public PsiStatement getInitialization(){
return (PsiStatement)findChildByRoleAsPsiElement(ChildRole.FOR_INITIALIZATION);
}
@Override
public PsiExpression getCondition(){
return (PsiExpression)findChildByRoleAsPsiElement(ChildRole.CONDITION);
}
@Override
public PsiStatement getUpdate(){
return (PsiStatement)findChildByRoleAsPsiElement(ChildRole.FOR_UPDATE);
}
@Override
public PsiStatement getBody(){
return (PsiStatement)findChildByRoleAsPsiElement(ChildRole.LOOP_BODY);
}
@Override
public PsiJavaToken getLParenth() {
return (PsiJavaToken) findChildByRoleAsPsiElement(ChildRole.LPARENTH);
}
@Override
public PsiJavaToken getRParenth() {
return (PsiJavaToken) findChildByRoleAsPsiElement(ChildRole.RPARENTH);
}
@Override
public ASTNode findChildByRole(int role){
LOG.assertTrue(ChildRole.isUnique(role));
switch(role){
default:
return null;
case ChildRole.FOR_KEYWORD:
return findChildByType(FOR_KEYWORD);
case ChildRole.LPARENTH:
return findChildByType(LPARENTH);
case ChildRole.FOR_INITIALIZATION:
final ASTNode initialization = PsiImplUtil.findStatementChild(this);
// should be inside parens
ASTNode paren = findChildByRole(ChildRole.LPARENTH);
for(ASTNode child = paren; child != null; child = child.getTreeNext()){
if (child == initialization) return initialization;
if (child.getElementType() == RPARENTH) return null;
}
return null;
case ChildRole.CONDITION:
return findChildByType(EXPRESSION_BIT_SET);
case ChildRole.FOR_SEMICOLON:
return findChildByType(SEMICOLON);
case ChildRole.FOR_UPDATE:
{
ASTNode semicolon = findChildByRole(ChildRole.FOR_SEMICOLON);
for(ASTNode child = semicolon; child != null; child = child.getTreeNext()){
if (child.getPsi() instanceof PsiStatement) {
return child;
}
if (child.getElementType() == RPARENTH) break;
}
return null;
}
case ChildRole.RPARENTH:
return findChildByType(RPARENTH);
case ChildRole.LOOP_BODY:
{
ASTNode rparenth = findChildByRole(ChildRole.RPARENTH);
for(ASTNode child = rparenth; child != null; child = child.getTreeNext()){
if (child.getPsi() instanceof PsiStatement) {
return child;
}
}
return null;
}
}
}
@Override
public int getChildRole(ASTNode child) {
LOG.assertTrue(child.getTreeParent() == this);
IElementType i = child.getElementType();
if (i == FOR_KEYWORD) {
return ChildRole.FOR_KEYWORD;
}
else if (i == LPARENTH) {
return ChildRole.LPARENTH;
}
else if (i == RPARENTH) {
return ChildRole.RPARENTH;
}
else if (i == SEMICOLON) {
return ChildRole.FOR_SEMICOLON;
}
else {
if (EXPRESSION_BIT_SET.contains(child.getElementType())) {
return ChildRole.CONDITION;
}
else if (child.getPsi() instanceof PsiStatement) {
int role = getChildRole(child, ChildRole.FOR_INITIALIZATION);
if (role != ChildRoleBase.NONE) return role;
role = getChildRole(child, ChildRole.FOR_UPDATE);
if (role != ChildRoleBase.NONE) return role;
return ChildRole.LOOP_BODY;
}
else {
return ChildRoleBase.NONE;
}
}
}
@Override
public void accept(@NotNull PsiElementVisitor visitor){
if (visitor instanceof JavaElementVisitor) {
((JavaElementVisitor)visitor).visitForStatement(this);
}
else {
visitor.visitElement(this);
}
}
public String toString(){
return "PsiForStatement";
}
@Override
public void deleteChildInternal(@NotNull ASTNode child) {
final boolean isForInitialization = getChildRole(child) == ChildRole.FOR_INITIALIZATION;
if (isForInitialization) {
try {
final PsiStatement emptyStatement = JavaPsiFacade.getInstance(getProject()).getElementFactory().createStatementFromText(";", null);
super.replaceChildInternal(child, (TreeElement)SourceTreeToPsiMap.psiElementToTree(emptyStatement));
}
catch (IncorrectOperationException e) {
LOG.error(e);
}
}
else {
super.deleteChildInternal(child);
}
}
@Override
public boolean processDeclarations(@NotNull PsiScopeProcessor processor, @NotNull ResolveState state, PsiElement lastParent, @NotNull PsiElement place) {
processor.handleEvent(PsiScopeProcessor.Event.SET_DECLARATION_HOLDER, this);
if (lastParent == null || lastParent.getParent() != this)
// Parent element should not see our vars
return true;
return PsiScopesUtil.walkChildrenScopes(this, processor, state, lastParent, place);
}
}