package org.jetbrains.dokka.base.renderers.html
import kotlinx.html.*
import kotlinx.html.stream.createHTML
import org.jetbrains.dokka.base.renderers.pageId
import org.jetbrains.dokka.base.templating.AddToNavigationCommand
import org.jetbrains.dokka.links.DRI
import org.jetbrains.dokka.model.DisplaySourceSet
import org.jetbrains.dokka.model.WithChildren
import org.jetbrains.dokka.pages.*
import org.jetbrains.dokka.plugability.DokkaContext
class NavigationPage(
val root: NavigationNode,
val moduleName: String,
val context: DokkaContext
) : RendererSpecificPage {
override val name = "navigation"
override val children = emptyList()
override fun modified(name: String, children: List) = this
override val strategy = RenderingStrategy {
createHTML().visit(root, this)
}
private fun TagConsumer.visit(node: NavigationNode, renderer: HtmlRenderer): R = with(renderer) {
if (context.configuration.delayTemplateSubstitution) {
templateCommand(AddToNavigationCommand(moduleName)) {
visit(node, "${moduleName}-nav-submenu", renderer)
}
} else {
visit(node, "${moduleName}-nav-submenu", renderer)
}
}
private fun TagConsumer.visit(node: NavigationNode, navId: String, renderer: HtmlRenderer): R =
with(renderer) {
div("sideMenuPart") {
id = navId
attributes["pageId"] = "${moduleName}::${node.pageId}"
div("overview") {
if (node.children.isNotEmpty()) {
span("navButton") {
onClick = """document.getElementById("$navId").classList.toggle("hidden");"""
span("navButtonContent")
}
}
buildLink(node.dri, node.sourceSets.toList()) {
val withIcon = node.icon != null
if (withIcon) {
// in case link text is so long that it needs to have word breaks,
// and it stretches to two or more lines, make sure the icon
// is always on the left in the grid and is not wrapped with text
span("nav-link-grid") {
span("nav-link-child ${node.icon?.style()}")
span("nav-link-child") {
nodeText(node)
}
}
} else {
nodeText(node)
}
}
}
node.children.withIndex().forEach { (n, p) -> visit(p, "$navId-$n", renderer) }
}
}
private fun FlowContent.nodeText(node: NavigationNode) {
if (node.styles.contains(TextStyle.Strikethrough)) {
strike {
buildBreakableText(node.name)
}
} else {
buildBreakableText(node.name)
}
}
}
data class NavigationNode(
val name: String,
val dri: DRI,
val sourceSets: Set,
val icon: NavigationNodeIcon?,
val styles: Set