All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.jetbrains.kotlin.javac.wrappers.trees.TreeBasedClass.kt Maven / Gradle / Ivy

There is a newer version: 2.1.0-Beta1
Show newest version
/*
 * Copyright 2010-2017 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 org.jetbrains.kotlin.javac.wrappers.trees

import com.intellij.openapi.vfs.VirtualFile
import com.intellij.psi.search.SearchScope
import com.sun.source.tree.CompilationUnitTree
import com.sun.source.tree.Tree
import com.sun.tools.javac.code.Flags
import com.sun.tools.javac.tree.JCTree
import com.sun.tools.javac.tree.TreeInfo
import org.jetbrains.kotlin.descriptors.Visibilities
import org.jetbrains.kotlin.descriptors.Visibility
import org.jetbrains.kotlin.javac.JavaClassWithClassId
import org.jetbrains.kotlin.javac.JavacWrapper
import org.jetbrains.kotlin.load.java.structure.*
import org.jetbrains.kotlin.name.ClassId
import org.jetbrains.kotlin.name.FqName
import org.jetbrains.kotlin.name.Name

class TreeBasedClass(
        tree: JCTree.JCClassDecl,
        compilationUnit: CompilationUnitTree,
        javac: JavacWrapper,
        override val classId: ClassId?,
        override val outerClass: JavaClass?
) : TreeBasedElement(tree, compilationUnit, javac), JavaClassWithClassId {

    override val isFromSource: Boolean
        get() = true

    override val name: Name
        get() = Name.identifier(tree.simpleName.toString())

    override val annotations: Collection by lazy {
        tree.annotations().map { annotation -> TreeBasedAnnotation(annotation, compilationUnit, javac, this) }
    }

    override fun findAnnotation(fqName: FqName) =
            annotations.find { it.classId?.asSingleFqName() == fqName }

    override val isDeprecatedInJavaDoc: Boolean
        get() = javac.isDeprecatedInJavaDoc(tree, compilationUnit)

    override val isAbstract: Boolean
        get() = tree.modifiers.isAbstract || ((isAnnotationType || isEnum) && methods.any { it.isAbstract })

    override val isStatic: Boolean
        get() = isEnum || isInterface || (outerClass?.isInterface ?: false) || tree.modifiers.isStatic

    override val isFinal: Boolean
        get() = isEnum || tree.modifiers.isFinal

    override val visibility: Visibility
        get() = if (outerClass?.isInterface == true) Visibilities.Public else tree.modifiers.visibility

    override val typeParameters: List
        get() = tree.typeParameters.map { parameter ->
            TreeBasedTypeParameter(parameter, compilationUnit, javac, this)
        }

    override val fqName: FqName
        get() = classId?.asSingleFqName() ?: throw UnsupportedOperationException("classId of $name is null")

    override val supertypes: Collection
            by lazy {
                arrayListOf().also { list ->
                    if (isEnum) {
                        list.add(EnumSupertype(this, javac))
                    }
                    else if (isAnnotationType) {
                        javac.JAVA_LANG_ANNOTATION_ANNOTATION?.let { list.add(it) }
                    }

                    tree.extending?.let {
                        (TreeBasedType.create(it, compilationUnit, javac, emptyList(), this) as? JavaClassifierType)
                                ?.let { list.add(it) }
                    }

                    if (list.isEmpty() && !isInterface) {
                        javac.JAVA_LANG_OBJECT?.let { list.add(it) }
                    }

                    tree.implementing?.mapNotNullTo(list) {
                        TreeBasedType.create(it, compilationUnit, javac, emptyList(), this) as? JavaClassifierType
                    }
                }
            }

    val innerClasses: Map by lazy {
        tree.members
                .filterIsInstance(JCTree.JCClassDecl::class.java)
                .map { TreeBasedClass(it, compilationUnit, javac, classId?.createNestedClassId(Name.identifier(it.simpleName.toString())), this) }
                .associateBy(JavaClass::name)
    }

    override val isInterface: Boolean
        get() = tree.modifiers.flags and Flags.INTERFACE.toLong() != 0L

    override val isAnnotationType: Boolean
        get() = tree.modifiers.flags and Flags.ANNOTATION.toLong() != 0L

    override val isEnum: Boolean
        get() = tree.modifiers.flags and Flags.ENUM.toLong() != 0L

    // TODO: Support
    override val isRecord: Boolean
        get() = false

    // TODO
    override val isSealed: Boolean
        get() = false

    override val permittedTypes: Collection
        get() = emptyList()

    override val lightClassOriginKind: LightClassOriginKind?
        get() = null

    override val methods: Collection
        get() = tree.members
                .filter { it.kind == Tree.Kind.METHOD && !TreeInfo.isConstructor(it) }
                .map { TreeBasedMethod(it as JCTree.JCMethodDecl, compilationUnit,this, javac) }

    override val fields: Collection
        get() = tree.members
                .filterIsInstance(JCTree.JCVariableDecl::class.java)
                .map { TreeBasedField(it, compilationUnit, this, javac) }

    override val constructors: Collection
        get() = tree.members
                .filter { member -> TreeInfo.isConstructor(member) }
                .map { constructor ->
                    TreeBasedConstructor(constructor as JCTree.JCMethodDecl, compilationUnit, this, javac)
                }

    override val recordComponents: Collection
        get() = emptyList()

    override fun hasDefaultConstructor() = !isInterface && constructors.isEmpty()

    override val innerClassNames: Collection
        get() = innerClasses.keys

    override val virtualFile: VirtualFile? by lazy {
        javac.toVirtualFile(compilationUnit.sourceFile)
    }

    override fun isFromSourceCodeInScope(scope: SearchScope): Boolean = true

    override fun findInnerClass(name: Name) = innerClasses[name]

}

private class EnumSupertype(private val javaClass: JavaClass,
                            private val javac: JavacWrapper) : JavaClassifierType {

    override val classifier: JavaClass?
        get() = javac.JAVA_LANG_ENUM

    override val typeArguments: List
        get() = listOf(TypeArgument())

    override val isRaw: Boolean
        get() = false
    override val annotations: Collection
        get() = emptyList()
    override val classifierQualifiedName: String
        get() = classifier?.fqName?.asString() ?: ""
    override val presentableText: String
        get() = classifierQualifiedName
    override val isDeprecatedInJavaDoc: Boolean
        get() = false

    override fun findAnnotation(fqName: FqName) = null

    private inner class TypeArgument : JavaClassifierType {
        override val classifier: JavaClassifier?
            get() = [email protected]
        override val typeArguments: List
            get() = emptyList()
        override val isRaw: Boolean
            get() = false
        override val annotations: Collection
            get() = emptyList()
        override val classifierQualifiedName: String
            get() = [email protected]!!.asString()
        override val presentableText: String
            get() = classifierQualifiedName
        override val isDeprecatedInJavaDoc: Boolean
            get() = false

        override fun findAnnotation(fqName: FqName) = null

    }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy