001 /*
002 * Copyright 2010-2015 JetBrains s.r.o.
003 *
004 * Licensed under the Apache License, Version 2.0 (the "License");
005 * you may not use this file except in compliance with the License.
006 * You may obtain a copy of the License at
007 *
008 * http://www.apache.org/licenses/LICENSE-2.0
009 *
010 * Unless required by applicable law or agreed to in writing, software
011 * distributed under the License is distributed on an "AS IS" BASIS,
012 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
013 * See the License for the specific language governing permissions and
014 * limitations under the License.
015 */
016
017 package org.jetbrains.kotlin.resolve.lazy;
018
019 import com.intellij.psi.PsiElement;
020 import org.jetbrains.annotations.NotNull;
021 import org.jetbrains.kotlin.incremental.components.NoLookupLocation;
022 import org.jetbrains.kotlin.psi.*;
023 import org.jetbrains.kotlin.psi.psiUtil.PsiUtilPackage;
024 import org.jetbrains.kotlin.resolve.calls.smartcasts.DataFlowInfo;
025 import org.jetbrains.kotlin.resolve.lazy.descriptors.LazyClassDescriptor;
026 import org.jetbrains.kotlin.resolve.scopes.LexicalScope;
027
028 public class DeclarationScopeProviderImpl implements DeclarationScopeProvider {
029
030 private final LazyDeclarationResolver lazyDeclarationResolver;
031
032 private final FileScopeProvider fileScopeProvider;
033
034 public DeclarationScopeProviderImpl(
035 @NotNull LazyDeclarationResolver lazyDeclarationResolver,
036 @NotNull FileScopeProvider fileScopeProvider
037 ) {
038 this.lazyDeclarationResolver = lazyDeclarationResolver;
039 this.fileScopeProvider = fileScopeProvider;
040 }
041
042 @Override
043 @NotNull
044 public LexicalScope getResolutionScopeForDeclaration(@NotNull PsiElement elementOfDeclaration) {
045 JetDeclaration jetDeclaration = JetStubbedPsiUtil.getPsiOrStubParent(elementOfDeclaration, JetDeclaration.class, false);
046
047 assert !(elementOfDeclaration instanceof JetDeclaration) || jetDeclaration == elementOfDeclaration :
048 "For JetDeclaration element getParentOfType() should return itself.";
049 assert jetDeclaration != null : "Should be contained inside declaration.";
050
051 JetDeclaration parentDeclaration = JetStubbedPsiUtil.getContainingDeclaration(jetDeclaration);
052
053 if (jetDeclaration instanceof JetPropertyAccessor) {
054 parentDeclaration = JetStubbedPsiUtil.getContainingDeclaration(parentDeclaration, JetDeclaration.class);
055 }
056
057 if (parentDeclaration == null) {
058 return fileScopeProvider.getFileScope((JetFile) elementOfDeclaration.getContainingFile());
059 }
060
061 if (parentDeclaration instanceof JetClassOrObject) {
062 JetClassOrObject classOrObject = (JetClassOrObject) parentDeclaration;
063 LazyClassDescriptor classDescriptor = (LazyClassDescriptor) lazyDeclarationResolver.getClassDescriptor(classOrObject, NoLookupLocation.WHEN_GET_DECLARATION_SCOPE);
064 if (jetDeclaration instanceof JetClassInitializer || jetDeclaration instanceof JetProperty) {
065 return classDescriptor.getScopeForInitializerResolution();
066 }
067 if (jetDeclaration instanceof JetObjectDeclaration
068 || (jetDeclaration instanceof JetClass && !((JetClass) jetDeclaration).isInner())) {
069 return classDescriptor.getScopeForStaticMemberDeclarationResolution();
070 }
071
072 return classDescriptor.getScopeForMemberDeclarationResolution();
073 }
074
075 throw new IllegalStateException("Don't call this method for local declarations: " + jetDeclaration + "\n" +
076 PsiUtilPackage.getElementTextWithContext(jetDeclaration));
077 }
078
079 @NotNull
080 @Override
081 public DataFlowInfo getOuterDataFlowInfoForDeclaration(@NotNull PsiElement elementOfDeclaration) {
082 return DataFlowInfo.EMPTY;
083 }
084 }