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.js.translate.reference;
018    
019    import com.google.dart.compiler.backend.js.ast.JsExpression;
020    import com.google.dart.compiler.backend.js.ast.JsNode;
021    import org.jetbrains.annotations.NotNull;
022    import org.jetbrains.annotations.Nullable;
023    import org.jetbrains.kotlin.descriptors.DeclarationDescriptor;
024    import org.jetbrains.kotlin.descriptors.PackageViewDescriptor;
025    import org.jetbrains.kotlin.js.translate.context.TemporaryVariable;
026    import org.jetbrains.kotlin.js.translate.context.TranslationContext;
027    import org.jetbrains.kotlin.js.translate.utils.ErrorReportingUtils;
028    import org.jetbrains.kotlin.js.translate.utils.JsAstUtils;
029    import org.jetbrains.kotlin.psi.*;
030    import org.jetbrains.kotlin.resolve.bindingContextUtil.BindingContextUtilPackage;
031    
032    import static org.jetbrains.kotlin.js.translate.general.Translation.translateAsExpression;
033    import static org.jetbrains.kotlin.js.translate.reference.CallExpressionTranslator.shouldBeInlined;
034    import static org.jetbrains.kotlin.js.translate.utils.BindingUtils.getDescriptorForReferenceExpression;
035    import static org.jetbrains.kotlin.js.translate.utils.PsiUtils.getNotNullSimpleNameSelector;
036    import static org.jetbrains.kotlin.js.translate.utils.PsiUtils.getSelector;
037    
038    public final class QualifiedExpressionTranslator {
039    
040        private QualifiedExpressionTranslator() {
041        }
042    
043        @NotNull
044        public static AccessTranslator getAccessTranslator(@NotNull JetQualifiedExpression expression,
045                                                           @NotNull TranslationContext context, boolean forceOrderOfEvaluation) {
046            JsExpression receiver = translateReceiver(expression, context);
047            if (forceOrderOfEvaluation && receiver != null) {
048                TemporaryVariable temporaryVariable = context.declareTemporary(null);
049                context.addStatementToCurrentBlock(JsAstUtils.assignment(temporaryVariable.reference(), receiver).makeStmt());
050                receiver = temporaryVariable.reference();
051            }
052            return VariableAccessTranslator.newInstance(context, getNotNullSimpleNameSelector(expression), receiver);
053        }
054    
055        @NotNull
056        public static JsNode translateQualifiedExpression(
057                @NotNull JetQualifiedExpression expression,
058                @NotNull TranslationContext context
059        ) {
060            JsExpression receiver = translateReceiver(expression, context);
061            JetExpression selector = getSelector(expression);
062            return dispatchToCorrectTranslator(receiver, selector, context);
063        }
064    
065        @NotNull
066        private static JsNode dispatchToCorrectTranslator(
067                @Nullable JsExpression receiver,
068                @NotNull JetExpression selector,
069                @NotNull TranslationContext context
070        ) {
071            if (ReferenceTranslator.canBePropertyAccess(selector, context)) {
072                assert selector instanceof JetSimpleNameExpression : "Selectors for properties must be simple names.";
073                return VariableAccessTranslator.newInstance(context, (JetSimpleNameExpression)selector, receiver).translateAsGet();
074            }
075            if (selector instanceof JetCallExpression) {
076                return invokeCallExpressionTranslator(receiver, selector, context);
077            }
078            //TODO: never get there
079            if (selector instanceof JetSimpleNameExpression) {
080                return ReferenceTranslator.translateSimpleNameWithQualifier((JetSimpleNameExpression) selector, receiver, context);
081            }
082            throw new AssertionError("Unexpected qualified expression: " + selector.getText());
083        }
084    
085        @NotNull
086        private static JsNode invokeCallExpressionTranslator(
087                @Nullable JsExpression receiver,
088                @NotNull JetExpression selector,
089                @NotNull TranslationContext context
090        ) {
091            try {
092                return CallExpressionTranslator.translate((JetCallExpression) selector, receiver, context);
093            } catch (RuntimeException e) {
094                throw  ErrorReportingUtils.reportErrorWithLocation(selector, e);
095            }
096        }
097    
098        @Nullable
099        private static JsExpression translateReceiver(@NotNull JetQualifiedExpression expression,
100                                                      @NotNull TranslationContext context) {
101            JetExpression receiverExpression = expression.getReceiverExpression();
102            if (isFullQualifierForExpression(receiverExpression, context)) {
103                return null;
104            }
105            return translateAsExpression(receiverExpression, context);
106        }
107    
108        //TODO: prove correctness
109        private static boolean isFullQualifierForExpression(@Nullable JetExpression receiverExpression, @NotNull TranslationContext context) {
110            if (receiverExpression == null) {
111                return false;
112            }
113            if (receiverExpression instanceof JetReferenceExpression) {
114                DeclarationDescriptor descriptorForReferenceExpression =
115                    getDescriptorForReferenceExpression(context.bindingContext(), (JetReferenceExpression)receiverExpression);
116                if (descriptorForReferenceExpression instanceof PackageViewDescriptor) {
117                    return true;
118                }
119            }
120            if (receiverExpression instanceof JetQualifiedExpression) {
121                return isFullQualifierForExpression(((JetQualifiedExpression)receiverExpression).getSelectorExpression(), context);
122            }
123            return false;
124        }
125    }