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

org.jetbrains.kotlin.javac.wrappers.symbols.SymbolBasedClass.kt Maven / Gradle / Ivy

There is a newer version: 2.1.20-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.symbols

import com.intellij.openapi.vfs.VirtualFile
import com.intellij.psi.CommonClassNames
import com.intellij.psi.search.SearchScope
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
import javax.lang.model.element.ElementKind
import javax.lang.model.element.ExecutableElement
import javax.lang.model.element.TypeElement
import javax.lang.model.element.VariableElement
import javax.lang.model.type.NoType
import javax.lang.model.type.TypeKind
import javax.lang.model.type.TypeMirror
import javax.tools.JavaFileObject

class SymbolBasedClass(
    element: TypeElement,
    javac: JavacWrapper,
    override val classId: ClassId?,
    val file: JavaFileObject?
) : SymbolBasedClassifier(element, javac), JavaClassWithClassId {

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

    override val isAbstract: Boolean
        get() = element.isAbstract

    override val isStatic: Boolean
        get() = element.isStatic

    override val isFinal: Boolean
        get() = element.isFinal

    override val visibility: Visibility
        get() = element.getVisibility()

    override val typeParameters: List
            by lazy { element.typeParameters.map { SymbolBasedTypeParameter(it, javac) } }

    override val fqName: FqName
        get() = FqName(element.qualifiedName.toString())

    override val supertypes: Collection
            by lazy {
                arrayListOf()
                    .apply {
                        element.superclass.takeIf { it !is NoType }?.let(this::add)
                        addAll(element.interfaces)
                    }
                    .mapTo(arrayListOf()) { SymbolBasedClassifierType(it, javac) }
                    .apply {
                        if (isEmpty() && element.qualifiedName.toString() != CommonClassNames.JAVA_LANG_OBJECT) {
                            javac.JAVA_LANG_OBJECT?.let { add(it) }
                        }
                    }
            }

    val innerClasses: Map
            by lazy {
                enclosedElements
                    .filterIsInstance(TypeElement::class.java)
                    .map { SymbolBasedClass(it, javac, classId?.createNestedClassId(Name.identifier(it.simpleName.toString())), file) }
                    .associateBy(JavaClass::name)
            }

    override val outerClass: JavaClass?
            by lazy {
                element.enclosingElement?.let {
                    if (it.asType().kind != TypeKind.DECLARED) null else SymbolBasedClass(
                        it as TypeElement,
                        javac,
                        classId?.outerClassId,
                        file
                    )
                }
            }

    override val isInterface: Boolean
        get() = element.kind == ElementKind.INTERFACE

    override val isAnnotationType: Boolean
        get() = element.kind == ElementKind.ANNOTATION_TYPE

    override val isEnum: Boolean
        get() = element.kind == ElementKind.ENUM

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

    override val permittedTypes: Sequence
        get() = emptySequence()

    override val lightClassOriginKind: LightClassOriginKind?
        get() = null

    override val methods: Collection
        get() = enclosedElements
            .filter { it.kind == ElementKind.METHOD && !isEnumValuesOrValueOf(it as ExecutableElement) }
            .map { SymbolBasedMethod(it as ExecutableElement, this, javac) }

    private fun isEnumValuesOrValueOf(method: ExecutableElement): Boolean {
        return isEnum && when (method.simpleName.toString()) {
            "values" -> method.parameters.isEmpty()
            "valueOf" -> method.parameters.let { it.size == 1 && it.first().asType().toString() == "java.lang.String" }
            else -> false
        }
    }

    override val fields: Collection
        get() = enclosedElements
            .filter { it.kind.isField && Name.isValidIdentifier(it.simpleName.toString()) }
            .map { SymbolBasedField(it as VariableElement, this, javac) }

    override val constructors: Collection
        get() = enclosedElements
            .filter { it.kind == ElementKind.CONSTRUCTOR }
            .map { SymbolBasedConstructor(it as ExecutableElement, this, javac) }

    override val isRecord: Boolean
        get() = false

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

    override fun hasDefaultConstructor() = false // default constructors are explicit in symbols

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

    override val virtualFile: VirtualFile? by lazy {
        file?.let { javac.toVirtualFile(it) }
    }

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

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

    private val enclosedElements by lazy { element.enclosedElements }

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy