001 /*
002 * Copyright 2010-2013 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.k2js.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 jet.runtime.typeinfo.KotlinSignature;
024 import org.jetbrains.annotations.NotNull;
025 import org.jetbrains.jet.lang.descriptors.ClassDescriptor;
026 import org.jetbrains.jet.lang.descriptors.FunctionDescriptor;
027 import org.jetbrains.jet.lang.descriptors.Modality;
028 import org.jetbrains.jet.lang.descriptors.PropertyDescriptor;
029 import org.jetbrains.jet.lang.psi.*;
030 import org.jetbrains.jet.lang.types.JetType;
031 import org.jetbrains.k2js.translate.context.TranslationContext;
032 import org.jetbrains.k2js.translate.general.Translation;
033 import org.jetbrains.k2js.translate.general.TranslatorVisitor;
034 import org.jetbrains.k2js.translate.initializer.ClassInitializerTranslator;
035 import org.jetbrains.k2js.translate.utils.BindingUtils;
036 import org.jetbrains.k2js.translate.utils.TranslationUtils;
037
038 import java.util.List;
039
040 import static org.jetbrains.k2js.translate.initializer.InitializerUtils.createClassObjectInitializer;
041 import static org.jetbrains.k2js.translate.utils.BindingUtils.getClassDescriptor;
042 import static org.jetbrains.k2js.translate.utils.BindingUtils.getFunctionDescriptor;
043 import static org.jetbrains.k2js.translate.utils.JsDescriptorUtils.getSupertypesWithoutFakes;
044
045 public class DeclarationBodyVisitor extends TranslatorVisitor<Void> {
046 @KotlinSignature("val result: MutableList<JsPropertyInitializer>")
047 protected final List<JsPropertyInitializer> result;
048 protected final List<JsPropertyInitializer> staticResult;
049 protected final List<JsPropertyInitializer> enumEntryList = new SmartList<JsPropertyInitializer>();
050
051 public DeclarationBodyVisitor(@NotNull List<JsPropertyInitializer> result, @NotNull List<JsPropertyInitializer> staticResult) {
052 this.result = result;
053 this.staticResult = staticResult;
054 }
055
056 @NotNull
057 public List<JsPropertyInitializer> getResult() {
058 return result;
059 }
060
061 public List<JsPropertyInitializer> getEnumEntryList() {
062 return enumEntryList;
063 }
064
065 @Override
066 public Void visitClass(@NotNull JetClass expression, TranslationContext context) {
067 return null;
068 }
069
070 @Override
071 public Void visitEnumEntry(@NotNull JetEnumEntry enumEntry, TranslationContext data) {
072 JsExpression jsEnumEntryCreation;
073 ClassDescriptor descriptor = getClassDescriptor(data.bindingContext(), enumEntry);
074 List<JetType> supertypes = getSupertypesWithoutFakes(descriptor);
075 if (enumEntry.getBody() != null || supertypes.size() > 1) {
076 jsEnumEntryCreation = ClassTranslator.generateClassCreation(enumEntry, descriptor, data);
077 } else {
078 assert supertypes.size() == 1 : "Simple Enum entry must have one supertype";
079 jsEnumEntryCreation = new ClassInitializerTranslator(enumEntry, data).generateEnumEntryInstanceCreation(supertypes.get(0));
080 }
081 enumEntryList.add(new JsPropertyInitializer(data.getNameForDescriptor(descriptor).makeRef(), jsEnumEntryCreation));
082 return null;
083 }
084
085 @Override
086 public Void visitClassObject(@NotNull JetClassObject classObject, TranslationContext context) {
087 JetObjectDeclaration declaration = classObject.getObjectDeclaration();
088 ClassDescriptor descriptor = getClassDescriptor(context.bindingContext(), declaration);
089 JsExpression value = ClassTranslator.generateClassCreation(declaration, descriptor, context);
090
091 JsFunction fun = TranslationUtils.simpleReturnFunction(context.getScopeForDescriptor(descriptor), value);
092 staticResult.add(createClassObjectInitializer(fun, context));
093 return null;
094 }
095
096 @Override
097 public Void visitObjectDeclaration(@NotNull JetObjectDeclaration declaration, TranslationContext context) {
098 // parsed it in initializer visitor => no additional actions are needed
099 return null;
100 }
101
102 @Override
103 public Void visitNamedFunction(@NotNull JetNamedFunction expression, TranslationContext context) {
104 FunctionDescriptor descriptor = getFunctionDescriptor(context.bindingContext(), expression);
105 if (descriptor.getModality() == Modality.ABSTRACT) {
106 return null;
107 }
108
109 JsPropertyInitializer methodAsPropertyInitializer = Translation.functionTranslator(expression, context).translateAsMethod();
110 result.add(methodAsPropertyInitializer);
111 return null;
112 }
113
114 @Override
115 public Void visitProperty(@NotNull JetProperty expression, TranslationContext context) {
116 PropertyDescriptor propertyDescriptor = BindingUtils.getPropertyDescriptor(context.bindingContext(), expression);
117 PropertyTranslator.translateAccessors(propertyDescriptor, expression, result, context);
118 return null;
119 }
120
121 @Override
122 public Void visitAnonymousInitializer(@NotNull JetClassInitializer expression, TranslationContext context) {
123 // parsed it in initializer visitor => no additional actions are needed
124 return null;
125 }
126 }