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.declaration;
018
019 import com.google.dart.compiler.backend.js.ast.JsExpression;
020 import com.google.dart.compiler.backend.js.ast.JsFunction;
021 import com.google.dart.compiler.backend.js.ast.JsPropertyInitializer;
022 import com.intellij.util.SmartList;
023 import org.jetbrains.annotations.NotNull;
024 import org.jetbrains.kotlin.descriptors.ClassDescriptor;
025 import org.jetbrains.kotlin.descriptors.FunctionDescriptor;
026 import org.jetbrains.kotlin.descriptors.Modality;
027 import org.jetbrains.kotlin.descriptors.PropertyDescriptor;
028 import org.jetbrains.kotlin.js.translate.context.TranslationContext;
029 import org.jetbrains.kotlin.js.translate.declaration.propertyTranslator.PropertyTranslatorKt;
030 import org.jetbrains.kotlin.js.translate.general.Translation;
031 import org.jetbrains.kotlin.js.translate.general.TranslatorVisitor;
032 import org.jetbrains.kotlin.js.translate.initializer.ClassInitializerTranslator;
033 import org.jetbrains.kotlin.js.translate.utils.BindingUtils;
034 import org.jetbrains.kotlin.js.translate.utils.TranslationUtils;
035 import org.jetbrains.kotlin.psi.*;
036 import org.jetbrains.kotlin.types.KotlinType;
037
038 import java.util.List;
039
040 import static org.jetbrains.kotlin.js.translate.initializer.InitializerUtils.createCompanionObjectInitializer;
041 import static org.jetbrains.kotlin.js.translate.utils.BindingUtils.getClassDescriptor;
042 import static org.jetbrains.kotlin.js.translate.utils.BindingUtils.getFunctionDescriptor;
043 import static org.jetbrains.kotlin.js.translate.utils.JsDescriptorUtils.getSupertypesWithoutFakes;
044
045 public class DeclarationBodyVisitor extends TranslatorVisitor<Void> {
046 protected final List<JsPropertyInitializer> result;
047 protected final List<JsPropertyInitializer> staticResult;
048 protected final List<JsPropertyInitializer> enumEntryList = new SmartList<JsPropertyInitializer>();
049
050 public DeclarationBodyVisitor(@NotNull List<JsPropertyInitializer> result, @NotNull List<JsPropertyInitializer> staticResult) {
051 this.result = result;
052 this.staticResult = staticResult;
053 }
054
055 @NotNull
056 public List<JsPropertyInitializer> getResult() {
057 return result;
058 }
059
060 public List<JsPropertyInitializer> getEnumEntryList() {
061 return enumEntryList;
062 }
063
064 @Override
065 protected Void emptyResult(@NotNull TranslationContext context) {
066 return null;
067 }
068
069 @Override
070 public Void visitClass(@NotNull KtClass expression, TranslationContext context) {
071 return null;
072 }
073
074 @Override
075 public Void visitEnumEntry(@NotNull KtEnumEntry enumEntry, TranslationContext data) {
076 JsExpression jsEnumEntryCreation;
077 ClassDescriptor descriptor = getClassDescriptor(data.bindingContext(), enumEntry);
078 List<KotlinType> supertypes = getSupertypesWithoutFakes(descriptor);
079 if (enumEntry.getBody() != null || supertypes.size() > 1) {
080 jsEnumEntryCreation = ClassTranslator.generateClassCreation(enumEntry, data);
081 } else {
082 assert supertypes.size() == 1 : "Simple Enum entry must have one supertype";
083 jsEnumEntryCreation = new ClassInitializerTranslator(enumEntry, data).generateEnumEntryInstanceCreation(supertypes.get(0));
084 }
085 enumEntryList.add(new JsPropertyInitializer(data.getNameForDescriptor(descriptor).makeRef(), jsEnumEntryCreation));
086 return null;
087 }
088
089 @Override
090 public Void visitObjectDeclaration(@NotNull KtObjectDeclaration declaration, TranslationContext context) {
091 if (!declaration.isCompanion()) {
092 // parsed it in initializer visitor => no additional actions are needed
093 return null;
094 }
095 JsExpression value = ClassTranslator.generateClassCreation(declaration, context);
096
097 ClassDescriptor descriptor = getClassDescriptor(context.bindingContext(), declaration);
098 JsFunction fun = TranslationUtils.simpleReturnFunction(context.getScopeForDescriptor(descriptor), value);
099 staticResult.add(createCompanionObjectInitializer(fun, context));
100 return null;
101 }
102
103 @Override
104 public Void visitNamedFunction(@NotNull KtNamedFunction expression, TranslationContext context) {
105 FunctionDescriptor descriptor = getFunctionDescriptor(context.bindingContext(), expression);
106 if (descriptor.getModality() == Modality.ABSTRACT) {
107 return null;
108 }
109
110 JsPropertyInitializer methodAsPropertyInitializer = Translation.functionTranslator(expression, context).translateAsMethod();
111 result.add(methodAsPropertyInitializer);
112 return null;
113 }
114
115 @Override
116 public Void visitProperty(@NotNull KtProperty expression, TranslationContext context) {
117 PropertyDescriptor propertyDescriptor = BindingUtils.getPropertyDescriptor(context.bindingContext(), expression);
118 PropertyTranslatorKt.translateAccessors(propertyDescriptor, expression, result, context);
119 return null;
120 }
121
122 @Override
123 public Void visitAnonymousInitializer(@NotNull KtClassInitializer expression, TranslationContext context) {
124 // parsed it in initializer visitor => no additional actions are needed
125 return null;
126 }
127
128 @Override
129 public Void visitSecondaryConstructor(@NotNull KtSecondaryConstructor constructor, TranslationContext data) {
130 return null;
131 }
132 }