org.jetbrains.kotlin.javac.wrappers.trees.TreeBasedClass.kt Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of kotlin-compiler-embeddable Show documentation
Show all versions of kotlin-compiler-embeddable Show documentation
the Kotlin compiler embeddable
/*
* 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: Sequence
get() = emptySequence()
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
}
}