From c0d1d62d2119f4ca4f0b095d786fb68de278a22a Mon Sep 17 00:00:00 2001 From: Oleg Yukhnevich Date: Tue, 21 Nov 2023 16:27:55 +0200 Subject: [PATCH 1/4] Render member extensions correctly --- .../dokka/model/documentableUtils.kt | 4 + .../plugin-base/api/plugin-base.api | 5 +- .../documentables/DefaultPageCreator.kt | 354 +++++++++++------- 3 files changed, 222 insertions(+), 141 deletions(-) diff --git a/dokka-subprojects/core/src/main/kotlin/org/jetbrains/dokka/model/documentableUtils.kt b/dokka-subprojects/core/src/main/kotlin/org/jetbrains/dokka/model/documentableUtils.kt index c9d75bf419..100d70ecb7 100644 --- a/dokka-subprojects/core/src/main/kotlin/org/jetbrains/dokka/model/documentableUtils.kt +++ b/dokka-subprojects/core/src/main/kotlin/org/jetbrains/dokka/model/documentableUtils.kt @@ -24,4 +24,8 @@ public fun DTypeParameter.filter(filteredSet: Set): DTypeParamet ) } +@Deprecated( + "Deprecated for removal", + ReplaceWith("this is Callable && this.receiver != null") +) public fun Documentable.isExtension(): Boolean = this is Callable && receiver != null diff --git a/dokka-subprojects/plugin-base/api/plugin-base.api b/dokka-subprojects/plugin-base/api/plugin-base.api index 8d768b4222..d0ff8ffe5e 100644 --- a/dokka-subprojects/plugin-base/api/plugin-base.api +++ b/dokka-subprojects/plugin-base/api/plugin-base.api @@ -1371,12 +1371,9 @@ public class org/jetbrains/dokka/base/translators/documentables/DefaultPageCreat protected fun contentForModule (Lorg/jetbrains/dokka/model/DModule;)Lorg/jetbrains/dokka/pages/ContentGroup; protected fun contentForPackage (Lorg/jetbrains/dokka/model/DPackage;)Lorg/jetbrains/dokka/pages/ContentGroup; protected fun contentForProperty (Lorg/jetbrains/dokka/model/DProperty;)Lorg/jetbrains/dokka/pages/ContentGroup; - protected fun contentForScope (Lorg/jetbrains/dokka/model/WithScope;Lorg/jetbrains/dokka/links/DRI;Ljava/util/Set;Ljava/util/List;)Lorg/jetbrains/dokka/pages/ContentGroup; - public static synthetic fun contentForScope$default (Lorg/jetbrains/dokka/base/translators/documentables/DefaultPageCreator;Lorg/jetbrains/dokka/model/WithScope;Lorg/jetbrains/dokka/links/DRI;Ljava/util/Set;Ljava/util/List;ILjava/lang/Object;)Lorg/jetbrains/dokka/pages/ContentGroup; + protected fun contentForScope (Lorg/jetbrains/dokka/model/WithScope;Lorg/jetbrains/dokka/links/DRI;Ljava/util/Set;)Lorg/jetbrains/dokka/pages/ContentGroup; protected fun contentForScopes (Ljava/util/List;Ljava/util/Set;Ljava/util/List;)Lorg/jetbrains/dokka/pages/ContentGroup; public static synthetic fun contentForScopes$default (Lorg/jetbrains/dokka/base/translators/documentables/DefaultPageCreator;Ljava/util/List;Ljava/util/Set;Ljava/util/List;ILjava/lang/Object;)Lorg/jetbrains/dokka/pages/ContentGroup; - protected fun divergentBlock (Lorg/jetbrains/dokka/base/translators/documentables/PageContentBuilder$DocumentableContentBuilder;Ljava/lang/String;Ljava/util/Collection;Lorg/jetbrains/dokka/pages/ContentKind;Lorg/jetbrains/dokka/model/properties/PropertyContainer;)V - public static synthetic fun divergentBlock$default (Lorg/jetbrains/dokka/base/translators/documentables/DefaultPageCreator;Lorg/jetbrains/dokka/base/translators/documentables/PageContentBuilder$DocumentableContentBuilder;Ljava/lang/String;Ljava/util/Collection;Lorg/jetbrains/dokka/pages/ContentKind;Lorg/jetbrains/dokka/model/properties/PropertyContainer;ILjava/lang/Object;)V protected fun getContentBuilder ()Lorg/jetbrains/dokka/base/translators/documentables/PageContentBuilder; public final fun getCustomTagContentProviders ()Ljava/util/List; public final fun getDocumentableAnalyzer ()Lorg/jetbrains/dokka/analysis/kotlin/internal/DocumentableSourceLanguageParser; diff --git a/dokka-subprojects/plugin-base/src/main/kotlin/org/jetbrains/dokka/base/translators/documentables/DefaultPageCreator.kt b/dokka-subprojects/plugin-base/src/main/kotlin/org/jetbrains/dokka/base/translators/documentables/DefaultPageCreator.kt index dac5144d8c..1955c9796a 100644 --- a/dokka-subprojects/plugin-base/src/main/kotlin/org/jetbrains/dokka/base/translators/documentables/DefaultPageCreator.kt +++ b/dokka-subprojects/plugin-base/src/main/kotlin/org/jetbrains/dokka/base/translators/documentables/DefaultPageCreator.kt @@ -71,7 +71,8 @@ public open class DefaultPageCreator( * @see ActualTypealias */ private fun List.filterOutActualTypeAlias(): List { - fun List.hasExpectClass(dri: DRI) = find { it is DClasslike && it.dri == dri && it.expectPresentInSet != null } != null + fun List.hasExpectClass(dri: DRI) = + find { it is DClasslike && it.dri == dri && it.expectPresentInSet != null } != null return this.filterNot { it is DTypeAlias && this.hasExpectClass(it.dri) } } @@ -382,28 +383,42 @@ public open class DefaultPageCreator( extensions: List = emptyList() ): ContentGroup { val types = scopes.flatMap { it.classlikes } + scopes.filterIsInstance().flatMap { it.typealiases } + val (extensionProperties, extensionFunctions) = extensions.splitPropsAndFuns() return contentForScope( - @Suppress("UNCHECKED_CAST") + dri = @Suppress("UNCHECKED_CAST") (scopes as List).dri, - sourceSets, - types, - scopes.flatMap { it.functions }, - scopes.flatMap { it.properties }, - extensions + sourceSets = sourceSets, + types = types, + functions = scopes.flatMap { it.functions }, + properties = scopes.flatMap { it.properties }, + extensionFunctions = extensionFunctions, + extensionProperties = extensionProperties ) } protected open fun contentForScope( s: WithScope, dri: DRI, - sourceSets: Set, - extensions: List = emptyList() + sourceSets: Set ): ContentGroup { val types = listOf( s.classlikes, (s as? DPackage)?.typealiases ?: emptyList() ).flatten() - return contentForScope(setOf(dri), sourceSets, types, s.functions, s.properties, extensions) + + // can be used only for packages + val (extensionFunctions, functions) = s.functions.partition { it.receiver != null } + val (extensionProperties, properties) = s.properties.partition { it.receiver != null } + + return contentForScope( + dri = setOf(dri), + sourceSets = sourceSets, + types = types, + functions = functions, + properties = properties, + extensionFunctions = extensionFunctions, + extensionProperties = extensionProperties + ) } private fun contentForScope( @@ -412,41 +427,54 @@ public open class DefaultPageCreator( types: List, functions: List, properties: List, - extensions: List + extensionFunctions: List, + extensionProperties: List, ) = contentBuilder.contentFor(dri, sourceSets) { - divergentBlock( - "Types", - types, - ContentKind.Classlikes - ) - val (extensionProps, extensionFuns) = extensions.splitPropsAndFuns() + typesBlock(types) if (separateInheritedMembers) { val (inheritedFunctions, memberFunctions) = functions.splitInherited() val (inheritedProperties, memberProperties) = properties.splitInherited() - val (inheritedExtensionFunctions, extensionFunctions) = extensionFuns.splitInheritedExtension(dri) - val (inheritedExtensionProperties, extensionProperties) = extensionProps.splitInheritedExtension(dri) - propertiesBlock( - "Properties", memberProperties + extensionProperties + val ( + inheritedExtensionFunctions, + directExtensionFunctions + ) = extensionFunctions.splitInheritedExtension(dri) + + val ( + inheritedExtensionProperties, + directExtensionProperties + ) = extensionProperties.splitInheritedExtension(dri) + + declarationsBlock( + name = "Properties", + isFunctions = false, + declarations = memberProperties, + extensions = directExtensionProperties ) - propertiesBlock( - "Inherited properties", inheritedProperties + inheritedExtensionProperties + declarationsBlock( + name = "Inherited properties", + isFunctions = false, + declarations = inheritedProperties, + extensions = inheritedExtensionProperties ) - functionsBlock("Functions", memberFunctions + extensionFunctions) - functionsBlock( - "Inherited functions", inheritedFunctions + inheritedExtensionFunctions + declarationsBlock( + name = "Functions", + isFunctions = true, + declarations = memberFunctions, + extensions = directExtensionFunctions ) - } else { - propertiesBlock( - "Properties", properties + extensionProps + declarationsBlock( + name = "Inherited functions", + isFunctions = true, + declarations = inheritedFunctions, + extensions = inheritedExtensionFunctions ) - functionsBlock("Functions", functions + extensionFuns) + } else { + declarationsBlock("Properties", isFunctions = false, properties, extensionProperties) + declarationsBlock("Functions", isFunctions = true, functions, extensionFunctions) } } - private fun Iterable.sorted() = - sortedWith(compareBy({ it.name }, { it.parameters.size }, { it.dri.toString() })) - /** * @param documentables a list of [DClasslike] and [DEnumEntry] and [DTypeAlias] with the same dri in different sourceSets */ @@ -491,6 +519,7 @@ public open class DefaultPageCreator( +contentForScopes(scopes, documentables.sourceSets, extensions) } } + protected open fun contentForConstructors( constructorsToDocumented: List, dri: Set, @@ -560,7 +589,6 @@ public open class DefaultPageCreator( } - protected open fun contentForDescription( d: Documentable ): List { @@ -611,7 +639,7 @@ public open class DefaultPageCreator( tag: TagWrapper ) { val language = documentableAnalyzer.getLanguage(documentable, sourceSet) - when(language) { + when (language) { DocumentableLanguage.JAVA -> firstSentenceComment(tag.root) DocumentableLanguage.KOTLIN -> firstParagraphComment(tag.root) else -> firstParagraphComment(tag.root) @@ -643,128 +671,180 @@ public open class DefaultPageCreator( } } - private fun DocumentableContentBuilder.functionsBlock( - name: String, - list: Collection - ) { + private fun DocumentableContentBuilder.typesBlock(types: List) { + if (types.isEmpty()) return + + val grouped = types + // This groupBy should probably use LocationProvider + .groupBy(Documentable::name) + .mapValues { (_, elements) -> + // This hacks displaying actual typealias signatures along classlike ones + if (elements.any { it is DClasslike }) elements.filter { it !is DTypeAlias } else elements + } + + val groups = grouped.entries + .sortedWith(compareBy(nullsFirst(canonicalAlphabeticalOrder)) { it.key }) + .map { (name, elements) -> + DivergentElementGroup( + name = name, + kind = ContentKind.Classlikes, + contentTypeOverride = null, + elements = elements + ) + } + divergentBlock( - name, - list.sorted(), - ContentKind.Functions, - extra = mainExtra + name = "Types", + kind = ContentKind.Classlikes, + extra = mainExtra, + contentType = BasicTabbedContentType.TYPE, + groups = groups ) } - private fun DocumentableContentBuilder.propertiesBlock( + private fun DocumentableContentBuilder.declarationsBlock( name: String, - list: Collection + isFunctions: Boolean, + declarations: List, + extensions: List ) { + if (declarations.isEmpty() && extensions.isEmpty()) return + + val contentKind = when { + isFunctions -> ContentKind.Functions + else -> ContentKind.Properties + } + + val declarationContentType = when { + isFunctions -> BasicTabbedContentType.FUNCTION + else -> BasicTabbedContentType.PROPERTY + } + + val extensionContentType = when { + isFunctions -> BasicTabbedContentType.EXTENSION_FUNCTION + else -> BasicTabbedContentType.EXTENSION_PROPERTY + } + + // This groupBy should probably use LocationProvider + val grouped = declarations.groupBy { + NameAndIsExtension(it.name, isExtension = false) + } + extensions.groupBy { + NameAndIsExtension(it.name, isExtension = true) + } + + val groups = grouped.entries + .sortedWith(compareBy(NameAndIsExtension.comparator) { it.key }) + .map { (nameAndIsExtension, elements) -> + DivergentElementGroup( + name = nameAndIsExtension.name, + kind = when { + nameAndIsExtension.isExtension -> ContentKind.Extensions + else -> contentKind + }, + contentTypeOverride = when { + nameAndIsExtension.isExtension -> extensionContentType + else -> null + }, + elements = elements + ) + } + divergentBlock( - name, - list, - ContentKind.Properties, - extra = mainExtra + name = name, + kind = contentKind, + extra = mainExtra, + contentType = when { + // if only extensions + declarations.isEmpty() -> extensionContentType + else -> declarationContentType + }, + groups = groups ) - } - private data class NameAndIsExtension(val name:String?, val isExtension: Boolean) - - private fun groupAndSortDivergentCollection(collection: Collection): List>> { - val groupKeyComparator: Comparator>> = - compareBy>, String?>( - nullsFirst(canonicalAlphabeticalOrder) - ) { it.key.name } - .thenBy { it.key.isExtension } - - return collection - .groupBy { - NameAndIsExtension( - it.name, - it.isExtension() - ) - } // This groupBy should probably use LocationProvider - // This hacks displaying actual typealias signatures along classlike ones - .mapValues { if (it.value.any { it is DClasslike }) it.value.filter { it !is DTypeAlias } else it.value } - .entries.sortedWith(groupKeyComparator) + + private data class NameAndIsExtension(val name: String?, val isExtension: Boolean) { + companion object { + val comparator = compareBy( + comparator = nullsFirst(canonicalAlphabeticalOrder), + selector = NameAndIsExtension::name + ).thenBy(NameAndIsExtension::isExtension) + } } - protected open fun DocumentableContentBuilder.divergentBlock( + private class DivergentElementGroup( + val name: String?, + val kind: ContentKind, + val contentTypeOverride: BasicTabbedContentType?, + val elements: List + ) + + private fun DocumentableContentBuilder.divergentBlock( name: String, - collection: Collection, kind: ContentKind, - extra: PropertyContainer = mainExtra + extra: PropertyContainer, + contentType: BasicTabbedContentType, + groups: List, ) { - if (collection.any()) { - val onlyExtensions = collection.all { it.isExtension() } - val groupExtra = when(kind) { - ContentKind.Functions -> extra + TabbedContentTypeExtra(if (onlyExtensions) BasicTabbedContentType.EXTENSION_FUNCTION else BasicTabbedContentType.FUNCTION) - ContentKind.Properties -> extra + TabbedContentTypeExtra(if (onlyExtensions) BasicTabbedContentType.EXTENSION_PROPERTY else BasicTabbedContentType.PROPERTY) - ContentKind.Classlikes -> extra + TabbedContentTypeExtra(BasicTabbedContentType.TYPE) - else -> extra - } - - group(extra = groupExtra) { - // be careful: groupExtra will be applied for children by default - header(2, name, kind = kind, extra = extra) { } - val isFunctions = collection.any { it is DFunction } - table(kind, extra = extra, styles = emptySet()) { - header { - group { text("Name") } - group { text("Summary") } - } - groupAndSortDivergentCollection(collection) - .forEach { (elementNameAndIsExtension, elements) -> // This groupBy should probably use LocationProvider - val elementName = elementNameAndIsExtension.name - val isExtension = elementNameAndIsExtension.isExtension - val rowExtra = - if (isExtension) extra + TabbedContentTypeExtra(if(isFunctions) BasicTabbedContentType.EXTENSION_FUNCTION else BasicTabbedContentType.EXTENSION_PROPERTY) else extra - val rowKind = if (isExtension) ContentKind.Extensions else kind - val sortedElements = sortDivergentElementsDeterministically(elements) - row( - dri = sortedElements.map { it.dri }.toSet(), - sourceSets = sortedElements.flatMap { it.sourceSets }.toSet(), - kind = rowKind, - styles = emptySet(), - extra = elementName?.let { name -> rowExtra + SymbolAnchorHint(name, kind) } ?: rowExtra - ) { - link( - text = elementName.orEmpty(), - address = sortedElements.first().dri, - kind = rowKind, - styles = setOf(ContentStyle.RowTitle), - sourceSets = sortedElements.sourceSets.toSet(), - extra = extra - ) - divergentGroup( - ContentDivergentGroup.GroupID(name), - sortedElements.map { it.dri }.toSet(), - kind = rowKind, - extra = extra + if (groups.isEmpty()) return + group(extra = extra + TabbedContentTypeExtra(contentType)) { + // be careful: groupExtra will be applied for children by default + header(2, name, kind = kind, extra = extra) { } + table(kind, extra = extra, styles = emptySet()) { + header { + group { text("Name") } + group { text("Summary") } + } + groups.forEach { group -> + val elementName = group.name + val rowKind = group.kind + val sortedElements = sortDivergentElementsDeterministically(group.elements) + + row( + dri = sortedElements.map { it.dri }.toSet(), + sourceSets = sortedElements.flatMap { it.sourceSets }.toSet(), + kind = rowKind, + styles = emptySet(), + extra = extra.addAll( + listOfNotNull( + group.contentTypeOverride?.let(::TabbedContentTypeExtra), + elementName?.let { name -> SymbolAnchorHint(name, kind) } + ) + ) + ) { + link( + text = elementName.orEmpty(), + address = sortedElements.first().dri, + kind = rowKind, + styles = setOf(ContentStyle.RowTitle), + sourceSets = sortedElements.sourceSets.toSet(), + extra = extra + ) + divergentGroup( + ContentDivergentGroup.GroupID(name), + sortedElements.map { it.dri }.toSet(), + kind = rowKind, + extra = extra + ) { + sortedElements.map { element -> + instance( + setOf(element.dri), + element.sourceSets.toSet() ) { - sortedElements.map { element -> - instance( - setOf(element.dri), - element.sourceSets.toSet(), - extra = PropertyContainer.withAll( - SymbolAnchorHint(element.name ?: "", rowKind) - ) - ) { - divergent(extra = PropertyContainer.empty()) { - group { - +buildSignature(element) - } - } - after( - extra = PropertyContainer.empty() - ) { - contentForBrief(element) - contentForCustomTagsBrief(element) - } + divergent(extra = PropertyContainer.empty()) { + group { + +buildSignature(element) } } + after( + extra = PropertyContainer.empty() + ) { + contentForBrief(element) + contentForCustomTagsBrief(element) + } } } } + } } } } From c29bb99a302f36cdf6e01512e550fad67af770d2 Mon Sep 17 00:00:00 2001 From: Oleg Yukhnevich Date: Mon, 4 Dec 2023 20:30:37 +0200 Subject: [PATCH 2/4] add a lot of tests --- .../src/test/kotlin/content/ExtensionsTest.kt | 714 ++++++++++++++++++ 1 file changed, 714 insertions(+) create mode 100644 dokka-subprojects/plugin-base/src/test/kotlin/content/ExtensionsTest.kt diff --git a/dokka-subprojects/plugin-base/src/test/kotlin/content/ExtensionsTest.kt b/dokka-subprojects/plugin-base/src/test/kotlin/content/ExtensionsTest.kt new file mode 100644 index 0000000000..8157c2c04f --- /dev/null +++ b/dokka-subprojects/plugin-base/src/test/kotlin/content/ExtensionsTest.kt @@ -0,0 +1,714 @@ +/* + * Copyright 2014-2023 JetBrains s.r.o. Use of this source code is governed by the Apache 2.0 license. + */ + +package content + +import matchers.content.* +import org.jetbrains.dokka.base.testApi.testRunner.BaseAbstractTest +import org.jetbrains.dokka.pages.* +import kotlin.test.Test +import kotlin.test.assertEquals + +class ExtensionsTest : BaseAbstractTest() { + private val configuration = dokkaConfiguration { + sourceSets { + sourceSet { + sourceRoots = listOf("src/") + } + } + } + + // function tests + + @Test + fun `should render top-level extension function`() { + testInline( + """ + /src/A.kt + fun String.extension() {} + """.trimIndent(), + configuration + ) { + pagesGenerationStage = { modulePage -> + val pkgNode = modulePage.children.single() as ContentPage + + pkgNode.content.assertTabbedNode { + group { + assertTabGroup("Functions", BasicTabbedContentType.EXTENSION_FUNCTION) { + assertTableWithTabs( + "extension" to BasicTabbedContentType.EXTENSION_FUNCTION + ) + } + } + } + } + } + } + + @Test + fun `should differentiate top-level function and extension function`() { + testInline( + """ + /src/A.kt + fun function() {} + fun String.extension() {} + """.trimIndent(), + configuration + ) { + pagesGenerationStage = { modulePage -> + val pkgNode = modulePage.children.single() as ContentPage + + pkgNode.content.assertTabbedNode { + group { + assertTabGroup("Functions", BasicTabbedContentType.FUNCTION) { + assertTableWithTabs( + "extension" to BasicTabbedContentType.EXTENSION_FUNCTION, + "function" to null + ) + } + } + } + } + } + } + + @Test + fun `should render member extension function for object`() { + testInline( + """ + /src/A.kt + object A { + fun String.memberExtension() {} + } + """.trimIndent(), + configuration + ) { + pagesGenerationStage = { modulePage -> + val pkgNode = modulePage.children.single() + val objectNode = pkgNode.children.single { it.name == "A" } as ContentPage + + objectNode.content.assertTabbedNode { + group { + assertTabGroup("Functions", BasicTabbedContentType.FUNCTION) { + assertTableWithTabs( + "memberExtension" to null, + ) + } + } + } + } + } + } + + @Test + fun `should render top-level extension function for object`() { + testInline( + """ + /src/A.kt + object A + fun A.topLevelExtension() {} + """.trimIndent(), + configuration + ) { + pagesGenerationStage = { modulePage -> + val pkgNode = modulePage.children.single() + val objectNode = pkgNode.children.single { it.name == "A" } as ContentPage + + objectNode.content.assertTabbedNode { + group { + assertTabGroup("Functions", BasicTabbedContentType.EXTENSION_FUNCTION) { + assertTableWithTabs( + "topLevelExtension" to BasicTabbedContentType.EXTENSION_FUNCTION + ) + } + } + } + } + } + } + + @Test + fun `should differentiate functions and extension functions for object`() { + testInline( + """ + /src/A.kt + object A { + fun function() {} + fun String.memberExtension() {} + } + fun A.extension() {} + """.trimIndent(), + configuration + ) { + pagesGenerationStage = { modulePage -> + val pkgNode = modulePage.children.single() + val objectNode = pkgNode.children.single { it.name == "A" } as ContentPage + + objectNode.content.assertTabbedNode { + group { + assertTabGroup("Functions", BasicTabbedContentType.FUNCTION) { + assertTableWithTabs( + "extension" to BasicTabbedContentType.EXTENSION_FUNCTION, + "function" to null, + "memberExtension" to null + ) + } + } + } + } + } + } + + @Test + fun `should render member extensions function for class`() { + testInline( + """ + /src/A.kt + class A { + fun String.memberExtension() {} + fun A.memberSelfExtension() {} + } + """.trimIndent(), + configuration + ) { + pagesGenerationStage = { modulePage -> + val pkgNode = modulePage.children.single() + val objectNode = pkgNode.children.single { it.name == "A" } as ContentPage + + objectNode.content.assertTabbedNode { + group { + assertTabGroup("Constructors", BasicTabbedContentType.CONSTRUCTOR) { skipAllNotMatching() } + } + group { + assertTabGroup("Functions", BasicTabbedContentType.FUNCTION) { + assertTableWithTabs( + "memberExtension" to null, + "memberSelfExtension" to null, + ) + } + } + } + } + } + } + + @Test + fun `should render top-level extension functions for class`() { + testInline( + """ + /src/A.kt + class A + fun A.topLevelExtension() {} + """.trimIndent(), + configuration + ) { + pagesGenerationStage = { modulePage -> + val pkgNode = modulePage.children.single() + val objectNode = pkgNode.children.single { it.name == "A" } as ContentPage + + objectNode.content.assertTabbedNode { + group { + assertTabGroup("Constructors", BasicTabbedContentType.CONSTRUCTOR) { skipAllNotMatching() } + } + group { + assertTabGroup("Functions", BasicTabbedContentType.EXTENSION_FUNCTION) { + assertTableWithTabs( + "topLevelExtension" to BasicTabbedContentType.EXTENSION_FUNCTION + ) + } + } + } + } + } + } + + @Test + fun `should render extension functions from object for class`() { + testInline( + """ + /src/A.kt + class A + object B { + fun A.extensionFromB() {} + } + """.trimIndent(), + configuration + ) { + pagesGenerationStage = { modulePage -> + val pkgNode = modulePage.children.single() + val objectNode = pkgNode.children.single { it.name == "A" } as ContentPage + + objectNode.content.assertTabbedNode { + group { + assertTabGroup("Constructors", BasicTabbedContentType.CONSTRUCTOR) { skipAllNotMatching() } + } + group { + assertTabGroup("Functions", BasicTabbedContentType.EXTENSION_FUNCTION) { + assertTableWithTabs( + "extensionFromB" to BasicTabbedContentType.EXTENSION_FUNCTION + ) + } + } + } + } + } + } + + @Test + fun `should differentiate functions and extension functions for class`() { + testInline( + """ + /src/A.kt + class A { + fun function() {} + fun String.memberExtension() {} + fun A.memberSelfExtension() {} + } + fun A.extension() {} + object B { + fun A.extensionFromB() {} + } + """.trimIndent(), + configuration + ) { + pagesGenerationStage = { modulePage -> + val pkgNode = modulePage.children.single() + val objectNode = pkgNode.children.single { it.name == "A" } as ContentPage + + objectNode.content.assertTabbedNode { + group { + assertTabGroup("Constructors", BasicTabbedContentType.CONSTRUCTOR) { skipAllNotMatching() } + } + group { + assertTabGroup("Functions", BasicTabbedContentType.FUNCTION) { + assertTableWithTabs( + "extension" to BasicTabbedContentType.EXTENSION_FUNCTION, + "extensionFromB" to BasicTabbedContentType.EXTENSION_FUNCTION, + "function" to null, + "memberExtension" to null, + "memberSelfExtension" to null, + ) + } + } + } + } + } + } + + @Test + fun `should render member extension functions for companion object of class`() { + testInline( + """ + /src/A.kt + class A { + companion object { + fun Int.companionMemberExtensionForInt() {} + fun A.companionMemberExtensionForA() {} + } + } + """.trimIndent(), + configuration + ) { + pagesGenerationStage = { modulePage -> + val pkgNode = modulePage.children.single() + val objectNode = pkgNode.children.single { it.name == "A" } as ContentPage + + objectNode.content.assertTabbedNode { + group { + assertTabGroup("Constructors", BasicTabbedContentType.CONSTRUCTOR) { skipAllNotMatching() } + } + group { + assertTabGroup("Types", BasicTabbedContentType.TYPE) { skipAllNotMatching() } + assertTabGroup("Functions", BasicTabbedContentType.EXTENSION_FUNCTION) { + assertTableWithTabs( + "companionMemberExtensionForA" to BasicTabbedContentType.EXTENSION_FUNCTION, + ) + } + } + } + + val companionPage = objectNode.children.single { it.name == "Companion" } as ContentPage + + companionPage.content.assertTabbedNode { + group { + assertTabGroup("Functions", BasicTabbedContentType.FUNCTION) { + assertTableWithTabs( + "companionMemberExtensionForA" to null, + "companionMemberExtensionForInt" to null + ) + } + } + } + } + } + } + + // property tests + + @Test + fun `should render top-level extension property`() { + testInline( + """ + /src/A.kt + val String.extension: String get() = "" + """.trimIndent(), + configuration + ) { + pagesGenerationStage = { modulePage -> + val pkgNode = modulePage.children.single() as ContentPage + + pkgNode.content.assertTabbedNode { + group { + assertTabGroup("Properties", BasicTabbedContentType.EXTENSION_PROPERTY) { + assertTableWithTabs( + "extension" to BasicTabbedContentType.EXTENSION_PROPERTY + ) + } + } + } + } + } + } + + @Test + fun `should differentiate top-level property and extension property`() { + testInline( + """ + /src/A.kt + val property: String get() = "" + val String.extension: String get() = "" + """.trimIndent(), + configuration + ) { + pagesGenerationStage = { modulePage -> + val pkgNode = modulePage.children.single() as ContentPage + + pkgNode.content.assertTabbedNode { + group { + assertTabGroup("Properties", BasicTabbedContentType.PROPERTY) { + assertTableWithTabs( + "extension" to BasicTabbedContentType.EXTENSION_PROPERTY, + "property" to null + ) + } + } + } + } + } + } + + @Test + fun `should render member extension property for object`() { + testInline( + """ + /src/A.kt + object A { + val String.memberExtension: String get() = "" + } + """.trimIndent(), + configuration + ) { + pagesGenerationStage = { modulePage -> + val pkgNode = modulePage.children.single() + val objectNode = pkgNode.children.single { it.name == "A" } as ContentPage + + objectNode.content.assertTabbedNode { + group { + assertTabGroup("Properties", BasicTabbedContentType.PROPERTY) { + assertTableWithTabs( + "memberExtension" to null, + ) + } + } + } + } + } + } + + @Test + fun `should render top-level extension property for object`() { + testInline( + """ + /src/A.kt + object A + val A.topLevelExtension: String get() = "" + """.trimIndent(), + configuration + ) { + pagesGenerationStage = { modulePage -> + val pkgNode = modulePage.children.single() + val objectNode = pkgNode.children.single { it.name == "A" } as ContentPage + + objectNode.content.assertTabbedNode { + group { + assertTabGroup("Properties", BasicTabbedContentType.EXTENSION_PROPERTY) { + assertTableWithTabs( + "topLevelExtension" to BasicTabbedContentType.EXTENSION_PROPERTY + ) + } + } + } + } + } + } + + @Test + fun `should differentiate properties and extension properties for object`() { + testInline( + """ + /src/A.kt + object A { + val property: String get() = "" + val String.memberExtension: String get() = "" + } + val A.extension: String get() = "" + """.trimIndent(), + configuration + ) { + pagesGenerationStage = { modulePage -> + val pkgNode = modulePage.children.single() + val objectNode = pkgNode.children.single { it.name == "A" } as ContentPage + + objectNode.content.assertTabbedNode { + group { + assertTabGroup("Properties", BasicTabbedContentType.PROPERTY) { + assertTableWithTabs( + "extension" to BasicTabbedContentType.EXTENSION_PROPERTY, + "memberExtension" to null, + "property" to null + ) + } + } + } + } + } + } + + @Test + fun `should render member extension properties for class`() { + testInline( + """ + /src/A.kt + class A { + val String.memberExtension: String get() = "" + val A.memberSelfExtension: String get() = "" + } + """.trimIndent(), + configuration + ) { + pagesGenerationStage = { modulePage -> + val pkgNode = modulePage.children.single() + val objectNode = pkgNode.children.single { it.name == "A" } as ContentPage + + objectNode.content.assertTabbedNode { + group { + assertTabGroup("Constructors", BasicTabbedContentType.CONSTRUCTOR) { skipAllNotMatching() } + } + group { + assertTabGroup("Properties", BasicTabbedContentType.PROPERTY) { + assertTableWithTabs( + "memberExtension" to null, + "memberSelfExtension" to null, + ) + } + } + } + } + } + } + + @Test + fun `should render top-level extension properties for class`() { + testInline( + """ + /src/A.kt + class A + val A.topLevelExtension: String get() = "" + """.trimIndent(), + configuration + ) { + pagesGenerationStage = { modulePage -> + val pkgNode = modulePage.children.single() + val objectNode = pkgNode.children.single { it.name == "A" } as ContentPage + + objectNode.content.assertTabbedNode { + group { + assertTabGroup("Constructors", BasicTabbedContentType.CONSTRUCTOR) { skipAllNotMatching() } + } + group { + assertTabGroup("Properties", BasicTabbedContentType.EXTENSION_PROPERTY) { + assertTableWithTabs( + "topLevelExtension" to BasicTabbedContentType.EXTENSION_PROPERTY + ) + } + } + } + } + } + } + + @Test + fun `should render extension properties from object for class`() { + testInline( + """ + /src/A.kt + class A + object B { + val A.extensionFromB: String get() = "" + } + """.trimIndent(), + configuration + ) { + pagesGenerationStage = { modulePage -> + val pkgNode = modulePage.children.single() + val objectNode = pkgNode.children.single { it.name == "A" } as ContentPage + + objectNode.content.assertTabbedNode { + group { + assertTabGroup("Constructors", BasicTabbedContentType.CONSTRUCTOR) { skipAllNotMatching() } + } + group { + assertTabGroup("Properties", BasicTabbedContentType.EXTENSION_PROPERTY) { + assertTableWithTabs( + "extensionFromB" to BasicTabbedContentType.EXTENSION_PROPERTY + ) + } + } + } + } + } + } + + @Test + fun `should differentiate properties and extension properties for class`() { + testInline( + """ + /src/A.kt + class A { + val property: String get() = "" + val String.memberExtension: String get() = "" + val A.memberSelfExtension: String get() = "" + } + val A.extension: String get() = "" + object B { + val A.extensionFromB: String get() = "" + } + """.trimIndent(), + configuration + ) { + pagesGenerationStage = { modulePage -> + val pkgNode = modulePage.children.single() + val objectNode = pkgNode.children.single { it.name == "A" } as ContentPage + + objectNode.content.assertTabbedNode { + group { + assertTabGroup("Constructors", BasicTabbedContentType.CONSTRUCTOR) { skipAllNotMatching() } + } + group { + assertTabGroup("Properties", BasicTabbedContentType.PROPERTY) { + assertTableWithTabs( + "extension" to BasicTabbedContentType.EXTENSION_PROPERTY, + "extensionFromB" to BasicTabbedContentType.EXTENSION_PROPERTY, + "memberExtension" to null, + "memberSelfExtension" to null, + "property" to null, + ) + } + } + } + } + } + } + + @Test + fun `should render member extension properties for companion object of class`() { + testInline( + """ + /src/A.kt + class A { + companion object { + val Int.companionMemberExtensionForInt: String get() = "" + val A.companionMemberExtensionForA: String get() = "" + } + } + """.trimIndent(), + configuration + ) { + pagesGenerationStage = { modulePage -> + val pkgNode = modulePage.children.single() + val objectNode = pkgNode.children.single { it.name == "A" } as ContentPage + + objectNode.content.assertTabbedNode { + group { + assertTabGroup("Constructors", BasicTabbedContentType.CONSTRUCTOR) { skipAllNotMatching() } + } + group { + assertTabGroup("Types", BasicTabbedContentType.TYPE) { skipAllNotMatching() } + assertTabGroup("Properties", BasicTabbedContentType.EXTENSION_PROPERTY) { + assertTableWithTabs( + "companionMemberExtensionForA" to BasicTabbedContentType.EXTENSION_PROPERTY, + ) + } + } + } + + val companionPage = objectNode.children.single { it.name == "Companion" } as ContentPage + + companionPage.content.assertTabbedNode { + group { + assertTabGroup("Properties", BasicTabbedContentType.PROPERTY) { + assertTableWithTabs( + "companionMemberExtensionForA" to null, + "companionMemberExtensionForInt" to null + ) + } + } + } + } + } + } + + private fun ContentMatcherBuilder.assertTableWithTabs( + vararg expected: Pair + ) { + table { + expected.forEach { (name, tabType) -> + group { + assertTabbedContentType(tabType) + link { +name } + skipAllNotMatching() + } + } + } + } + + private fun ContentMatcherBuilder.assertTabbedContentType(expected: TabbedContentType?) { + check { + assertEquals(expected, extra[TabbedContentTypeExtra]?.value) + } + } + + private fun ContentNode.assertTabbedNode(block: ContentMatcherBuilder.() -> Unit) { + assertNode { + group { + header { } + skipAllNotMatching() + } + tabbedGroup(block) + } + } + + private fun ContentMatcherBuilder.assertTabGroup( + name: String, + type: TabbedContentType, + block: ContentMatcherBuilder.() -> Unit + ) { + group { + assertTabbedContentType(type) + header { +name } + block() + } + } + +} From 75fe3a386edbf08a20590cfa71a54524522c9511 Mon Sep 17 00:00:00 2001 From: Oleg Yukhnevich Date: Mon, 11 Dec 2023 15:10:47 +0200 Subject: [PATCH 3/4] revert deprecation --- .../kotlin/org/jetbrains/dokka/model/documentableUtils.kt | 4 ---- 1 file changed, 4 deletions(-) diff --git a/dokka-subprojects/core/src/main/kotlin/org/jetbrains/dokka/model/documentableUtils.kt b/dokka-subprojects/core/src/main/kotlin/org/jetbrains/dokka/model/documentableUtils.kt index 100d70ecb7..c9d75bf419 100644 --- a/dokka-subprojects/core/src/main/kotlin/org/jetbrains/dokka/model/documentableUtils.kt +++ b/dokka-subprojects/core/src/main/kotlin/org/jetbrains/dokka/model/documentableUtils.kt @@ -24,8 +24,4 @@ public fun DTypeParameter.filter(filteredSet: Set): DTypeParamet ) } -@Deprecated( - "Deprecated for removal", - ReplaceWith("this is Callable && this.receiver != null") -) public fun Documentable.isExtension(): Boolean = this is Callable && receiver != null From 2bcd3da3fb2726fd56a316bb9d78ee4faa5b5b16 Mon Sep 17 00:00:00 2001 From: Oleg Yukhnevich Date: Mon, 11 Dec 2023 15:23:44 +0200 Subject: [PATCH 4/4] Refactor separation of members and extensions to be more understandable * cleanup logic a bit * add documentation for unclear things * revert some ABI changes --- .../dokka/pages/contentNodeProperties.kt | 25 ++- .../plugin-base/api/plugin-base.api | 3 +- .../documentables/DefaultPageCreator.kt | 179 +++++++++--------- .../src/test/kotlin/content/ExtensionsTest.kt | 20 +- 4 files changed, 121 insertions(+), 106 deletions(-) diff --git a/dokka-subprojects/core/src/main/kotlin/org/jetbrains/dokka/pages/contentNodeProperties.kt b/dokka-subprojects/core/src/main/kotlin/org/jetbrains/dokka/pages/contentNodeProperties.kt index 64f1957291..f3699ee23c 100644 --- a/dokka-subprojects/core/src/main/kotlin/org/jetbrains/dokka/pages/contentNodeProperties.kt +++ b/dokka-subprojects/core/src/main/kotlin/org/jetbrains/dokka/pages/contentNodeProperties.kt @@ -16,7 +16,30 @@ public class SimpleAttr( } public enum class BasicTabbedContentType : TabbedContentType { - TYPE, CONSTRUCTOR, FUNCTION, PROPERTY, ENTRY, EXTENSION_PROPERTY, EXTENSION_FUNCTION + TYPE, CONSTRUCTOR, + + // property/function here means a different things depending on parent: + // - if parent=package - describes just `top-level` property/function without receiver + // - if parent=classlike - describes `member` property/function, + // it could have receiver (becoming member extension property/function) or not (ordinary member property/function) + // for examples look at docs for `EXTENSION_PROPERTY`, `EXTENSION_FUNCTION` + FUNCTION, PROPERTY, + + ENTRY, + + // property/function here means a different things depending on parent, + // and not just `an extension property/function`: + // example 1: `fun Foo.bar()` - top-level extension function + // - on a page describing `Foo` class `bar` will have type=`EXTENSION_FUNCTION` + // - on a page describing package declarations `bar` will have type=`EXTENSION_FUNCTION` + // example 2: `object Namespace { fun Foo.bar() }` - member extension function + // - on a page describing `Foo` class `bar` will have type=`EXTENSION_FUNCTION` + // - on a page describing `Namespace` object `bar` will have type=`FUNCTION` + // + // These types are needed to separate member functions and extension function on classlike pages. + // The same split rules are also used + // when grouping functions/properties with the same name on pages for classlike and package + EXTENSION_PROPERTY, EXTENSION_FUNCTION } /** diff --git a/dokka-subprojects/plugin-base/api/plugin-base.api b/dokka-subprojects/plugin-base/api/plugin-base.api index d0ff8ffe5e..bdcbdc2718 100644 --- a/dokka-subprojects/plugin-base/api/plugin-base.api +++ b/dokka-subprojects/plugin-base/api/plugin-base.api @@ -1371,7 +1371,8 @@ public class org/jetbrains/dokka/base/translators/documentables/DefaultPageCreat protected fun contentForModule (Lorg/jetbrains/dokka/model/DModule;)Lorg/jetbrains/dokka/pages/ContentGroup; protected fun contentForPackage (Lorg/jetbrains/dokka/model/DPackage;)Lorg/jetbrains/dokka/pages/ContentGroup; protected fun contentForProperty (Lorg/jetbrains/dokka/model/DProperty;)Lorg/jetbrains/dokka/pages/ContentGroup; - protected fun contentForScope (Lorg/jetbrains/dokka/model/WithScope;Lorg/jetbrains/dokka/links/DRI;Ljava/util/Set;)Lorg/jetbrains/dokka/pages/ContentGroup; + protected fun contentForScope (Lorg/jetbrains/dokka/model/WithScope;Lorg/jetbrains/dokka/links/DRI;Ljava/util/Set;Ljava/util/List;)Lorg/jetbrains/dokka/pages/ContentGroup; + public static synthetic fun contentForScope$default (Lorg/jetbrains/dokka/base/translators/documentables/DefaultPageCreator;Lorg/jetbrains/dokka/model/WithScope;Lorg/jetbrains/dokka/links/DRI;Ljava/util/Set;Ljava/util/List;ILjava/lang/Object;)Lorg/jetbrains/dokka/pages/ContentGroup; protected fun contentForScopes (Ljava/util/List;Ljava/util/Set;Ljava/util/List;)Lorg/jetbrains/dokka/pages/ContentGroup; public static synthetic fun contentForScopes$default (Lorg/jetbrains/dokka/base/translators/documentables/DefaultPageCreator;Ljava/util/List;Ljava/util/Set;Ljava/util/List;ILjava/lang/Object;)Lorg/jetbrains/dokka/pages/ContentGroup; protected fun getContentBuilder ()Lorg/jetbrains/dokka/base/translators/documentables/PageContentBuilder; diff --git a/dokka-subprojects/plugin-base/src/main/kotlin/org/jetbrains/dokka/base/translators/documentables/DefaultPageCreator.kt b/dokka-subprojects/plugin-base/src/main/kotlin/org/jetbrains/dokka/base/translators/documentables/DefaultPageCreator.kt index 1955c9796a..a8b3639011 100644 --- a/dokka-subprojects/plugin-base/src/main/kotlin/org/jetbrains/dokka/base/translators/documentables/DefaultPageCreator.kt +++ b/dokka-subprojects/plugin-base/src/main/kotlin/org/jetbrains/dokka/base/translators/documentables/DefaultPageCreator.kt @@ -372,7 +372,14 @@ public open class DefaultPageCreator( } } group(styles = setOf(ContentStyle.TabbedContent), extra = mainExtra) { - +contentForScope(p, p.dri, p.sourceSets) + val (functions, extensionFunctions) = p.functions.partition { it.receiver == null } + val (properties, extensionProperties) = p.properties.partition { it.receiver == null } + +contentForScope( + s = p.copy(functions = functions, properties = properties), + dri = p.dri, + sourceSets = p.sourceSets, + extensions = extensionFunctions + extensionProperties + ) } } } @@ -381,45 +388,22 @@ public open class DefaultPageCreator( scopes: List, sourceSets: Set, extensions: List = emptyList() - ): ContentGroup { - val types = scopes.flatMap { it.classlikes } + scopes.filterIsInstance().flatMap { it.typealiases } - val (extensionProperties, extensionFunctions) = extensions.splitPropsAndFuns() - return contentForScope( - dri = @Suppress("UNCHECKED_CAST") - (scopes as List).dri, - sourceSets = sourceSets, - types = types, - functions = scopes.flatMap { it.functions }, - properties = scopes.flatMap { it.properties }, - extensionFunctions = extensionFunctions, - extensionProperties = extensionProperties - ) - } + ): ContentGroup = contentForScope( + dri = @Suppress("UNCHECKED_CAST") (scopes as List).dri, + sourceSets = sourceSets, + types = scopes.flatMap { it.classlikes } + + scopes.filterIsInstance().flatMap { it.typealiases }, + functions = scopes.flatMap { it.functions }, + properties = scopes.flatMap { it.properties }, + extensions = extensions, + ) protected open fun contentForScope( s: WithScope, dri: DRI, - sourceSets: Set - ): ContentGroup { - val types = listOf( - s.classlikes, - (s as? DPackage)?.typealiases ?: emptyList() - ).flatten() - - // can be used only for packages - val (extensionFunctions, functions) = s.functions.partition { it.receiver != null } - val (extensionProperties, properties) = s.properties.partition { it.receiver != null } - - return contentForScope( - dri = setOf(dri), - sourceSets = sourceSets, - types = types, - functions = functions, - properties = properties, - extensionFunctions = extensionFunctions, - extensionProperties = extensionProperties - ) - } + sourceSets: Set, + extensions: List = emptyList() + ): ContentGroup = contentForScopes(listOf(s), sourceSets, extensions) private fun contentForScope( dri: Set, @@ -427,10 +411,10 @@ public open class DefaultPageCreator( types: List, functions: List, properties: List, - extensionFunctions: List, - extensionProperties: List, + extensions: List, ) = contentBuilder.contentFor(dri, sourceSets) { typesBlock(types) + val (extensionProperties, extensionFunctions) = extensions.splitPropsAndFuns() if (separateInheritedMembers) { val (inheritedFunctions, memberFunctions) = functions.splitInherited() val (inheritedProperties, memberProperties) = properties.splitInherited() @@ -445,33 +429,14 @@ public open class DefaultPageCreator( directExtensionProperties ) = extensionProperties.splitInheritedExtension(dri) - declarationsBlock( - name = "Properties", - isFunctions = false, - declarations = memberProperties, - extensions = directExtensionProperties - ) - declarationsBlock( - name = "Inherited properties", - isFunctions = false, - declarations = inheritedProperties, - extensions = inheritedExtensionProperties - ) - declarationsBlock( - name = "Functions", - isFunctions = true, - declarations = memberFunctions, - extensions = directExtensionFunctions - ) - declarationsBlock( - name = "Inherited functions", - isFunctions = true, - declarations = inheritedFunctions, - extensions = inheritedExtensionFunctions - ) + propertiesBlock("Properties", memberProperties, directExtensionProperties) + propertiesBlock("Inherited properties", inheritedProperties, inheritedExtensionProperties) + + functionsBlock("Functions", memberFunctions, directExtensionFunctions) + functionsBlock("Inherited functions", inheritedFunctions, inheritedExtensionFunctions) } else { - declarationsBlock("Properties", isFunctions = false, properties, extensionProperties) - declarationsBlock("Functions", isFunctions = true, functions, extensionFunctions) + propertiesBlock("Properties", properties, extensionProperties) + functionsBlock("Functions", functions, extensionFunctions) } } @@ -688,7 +653,6 @@ public open class DefaultPageCreator( DivergentElementGroup( name = name, kind = ContentKind.Classlikes, - contentTypeOverride = null, elements = elements ) } @@ -702,29 +666,49 @@ public open class DefaultPageCreator( ) } - private fun DocumentableContentBuilder.declarationsBlock( + private fun DocumentableContentBuilder.functionsBlock( + name: String, + declarations: List, + extensions: List + ) { + functionsOrPropertiesBlock( + name = name, + contentKind = ContentKind.Functions, + contentType = when { + declarations.isEmpty() -> BasicTabbedContentType.EXTENSION_FUNCTION + else -> BasicTabbedContentType.FUNCTION + }, + declarations = declarations, + extensions = extensions + ) + } + + private fun DocumentableContentBuilder.propertiesBlock( name: String, - isFunctions: Boolean, + declarations: List, + extensions: List + ) { + functionsOrPropertiesBlock( + name = name, + contentKind = ContentKind.Properties, + contentType = when { + declarations.isEmpty() -> BasicTabbedContentType.EXTENSION_PROPERTY + else -> BasicTabbedContentType.PROPERTY + }, + declarations = declarations, + extensions = extensions + ) + } + + private fun DocumentableContentBuilder.functionsOrPropertiesBlock( + name: String, + contentKind: ContentKind, + contentType: BasicTabbedContentType, declarations: List, extensions: List ) { if (declarations.isEmpty() && extensions.isEmpty()) return - val contentKind = when { - isFunctions -> ContentKind.Functions - else -> ContentKind.Properties - } - - val declarationContentType = when { - isFunctions -> BasicTabbedContentType.FUNCTION - else -> BasicTabbedContentType.PROPERTY - } - - val extensionContentType = when { - isFunctions -> BasicTabbedContentType.EXTENSION_FUNCTION - else -> BasicTabbedContentType.EXTENSION_PROPERTY - } - // This groupBy should probably use LocationProvider val grouped = declarations.groupBy { NameAndIsExtension(it.name, isExtension = false) @@ -741,10 +725,6 @@ public open class DefaultPageCreator( nameAndIsExtension.isExtension -> ContentKind.Extensions else -> contentKind }, - contentTypeOverride = when { - nameAndIsExtension.isExtension -> extensionContentType - else -> null - }, elements = elements ) } @@ -753,11 +733,7 @@ public open class DefaultPageCreator( name = name, kind = contentKind, extra = mainExtra, - contentType = when { - // if only extensions - declarations.isEmpty() -> extensionContentType - else -> declarationContentType - }, + contentType = contentType, groups = groups ) } @@ -774,7 +750,6 @@ public open class DefaultPageCreator( private class DivergentElementGroup( val name: String?, val kind: ContentKind, - val contentTypeOverride: BasicTabbedContentType?, val elements: List ) @@ -786,8 +761,9 @@ public open class DefaultPageCreator( groups: List, ) { if (groups.isEmpty()) return + + // be careful: extra here will be applied for children by default group(extra = extra + TabbedContentTypeExtra(contentType)) { - // be careful: groupExtra will be applied for children by default header(2, name, kind = kind, extra = extra) { } table(kind, extra = extra, styles = emptySet()) { header { @@ -799,6 +775,21 @@ public open class DefaultPageCreator( val rowKind = group.kind val sortedElements = sortDivergentElementsDeterministically(group.elements) + // This override here is needed to be able to split members and extensions into separate tabs in HTML renderer. + // The idea is that `contentType` is set to the `tab group` itself to `FUNCTION` or `PROPERTY` (above in the code), + // and then for `extensions` we override it - in this case we are able to create 2 tabs in HTML renderer: + // - `Members` - which show ONLY member functions/properties + // - `Members & Extensions` - which show BOTH member functions/properties and extensions for this classlike + val rowContentTypeOverride = when (rowKind) { + ContentKind.Extensions -> when (contentType) { + BasicTabbedContentType.FUNCTION -> BasicTabbedContentType.EXTENSION_FUNCTION + BasicTabbedContentType.PROPERTY -> BasicTabbedContentType.EXTENSION_PROPERTY + else -> null + } + + else -> null + } + row( dri = sortedElements.map { it.dri }.toSet(), sourceSets = sortedElements.flatMap { it.sourceSets }.toSet(), @@ -806,7 +797,7 @@ public open class DefaultPageCreator( styles = emptySet(), extra = extra.addAll( listOfNotNull( - group.contentTypeOverride?.let(::TabbedContentTypeExtra), + rowContentTypeOverride?.let(::TabbedContentTypeExtra), elementName?.let { name -> SymbolAnchorHint(name, kind) } ) ) diff --git a/dokka-subprojects/plugin-base/src/test/kotlin/content/ExtensionsTest.kt b/dokka-subprojects/plugin-base/src/test/kotlin/content/ExtensionsTest.kt index 8157c2c04f..fe6ace5948 100644 --- a/dokka-subprojects/plugin-base/src/test/kotlin/content/ExtensionsTest.kt +++ b/dokka-subprojects/plugin-base/src/test/kotlin/content/ExtensionsTest.kt @@ -37,7 +37,7 @@ class ExtensionsTest : BaseAbstractTest() { group { assertTabGroup("Functions", BasicTabbedContentType.EXTENSION_FUNCTION) { assertTableWithTabs( - "extension" to BasicTabbedContentType.EXTENSION_FUNCTION + "extension" to null ) } } @@ -119,7 +119,7 @@ class ExtensionsTest : BaseAbstractTest() { group { assertTabGroup("Functions", BasicTabbedContentType.EXTENSION_FUNCTION) { assertTableWithTabs( - "topLevelExtension" to BasicTabbedContentType.EXTENSION_FUNCTION + "topLevelExtension" to null ) } } @@ -214,7 +214,7 @@ class ExtensionsTest : BaseAbstractTest() { group { assertTabGroup("Functions", BasicTabbedContentType.EXTENSION_FUNCTION) { assertTableWithTabs( - "topLevelExtension" to BasicTabbedContentType.EXTENSION_FUNCTION + "topLevelExtension" to null ) } } @@ -246,7 +246,7 @@ class ExtensionsTest : BaseAbstractTest() { group { assertTabGroup("Functions", BasicTabbedContentType.EXTENSION_FUNCTION) { assertTableWithTabs( - "extensionFromB" to BasicTabbedContentType.EXTENSION_FUNCTION + "extensionFromB" to null ) } } @@ -322,7 +322,7 @@ class ExtensionsTest : BaseAbstractTest() { assertTabGroup("Types", BasicTabbedContentType.TYPE) { skipAllNotMatching() } assertTabGroup("Functions", BasicTabbedContentType.EXTENSION_FUNCTION) { assertTableWithTabs( - "companionMemberExtensionForA" to BasicTabbedContentType.EXTENSION_FUNCTION, + "companionMemberExtensionForA" to null, ) } } @@ -362,7 +362,7 @@ class ExtensionsTest : BaseAbstractTest() { group { assertTabGroup("Properties", BasicTabbedContentType.EXTENSION_PROPERTY) { assertTableWithTabs( - "extension" to BasicTabbedContentType.EXTENSION_PROPERTY + "extension" to null ) } } @@ -444,7 +444,7 @@ class ExtensionsTest : BaseAbstractTest() { group { assertTabGroup("Properties", BasicTabbedContentType.EXTENSION_PROPERTY) { assertTableWithTabs( - "topLevelExtension" to BasicTabbedContentType.EXTENSION_PROPERTY + "topLevelExtension" to null ) } } @@ -539,7 +539,7 @@ class ExtensionsTest : BaseAbstractTest() { group { assertTabGroup("Properties", BasicTabbedContentType.EXTENSION_PROPERTY) { assertTableWithTabs( - "topLevelExtension" to BasicTabbedContentType.EXTENSION_PROPERTY + "topLevelExtension" to null ) } } @@ -571,7 +571,7 @@ class ExtensionsTest : BaseAbstractTest() { group { assertTabGroup("Properties", BasicTabbedContentType.EXTENSION_PROPERTY) { assertTableWithTabs( - "extensionFromB" to BasicTabbedContentType.EXTENSION_PROPERTY + "extensionFromB" to null ) } } @@ -647,7 +647,7 @@ class ExtensionsTest : BaseAbstractTest() { assertTabGroup("Types", BasicTabbedContentType.TYPE) { skipAllNotMatching() } assertTabGroup("Properties", BasicTabbedContentType.EXTENSION_PROPERTY) { assertTableWithTabs( - "companionMemberExtensionForA" to BasicTabbedContentType.EXTENSION_PROPERTY, + "companionMemberExtensionForA" to null, ) } }