<lambda>null1 package org.jetbrains.dokka.Formats
2
3 import com.google.inject.Inject
4 import org.jetbrains.dokka.*
5 import java.net.URI
6 import com.google.inject.name.Named
7 import org.jetbrains.kotlin.cfg.pseudocode.AllTypes
8
9
10 interface DacOutlineFormatService {
11 fun computeOutlineURI(node: DocumentationNode): URI
12 fun format(to: Appendable, node: DocumentationNode)
13 }
14
15 class DacOutlineFormatter @Inject constructor(
16 uriProvider: JavaLayoutHtmlUriProvider,
17 languageService: LanguageService,
18 @Named("dacRoot") dacRoot: String,
19 @Named("generateClassIndex") generateClassIndex: Boolean,
20 @Named("generatePackageIndex") generatePackageIndex: Boolean
21 ) : JavaLayoutHtmlFormatOutlineFactoryService {
22 val tocOutline = TocOutlineService(uriProvider, languageService, dacRoot, generateClassIndex, generatePackageIndex)
23 val outlines = listOf(tocOutline)
24
generateOutlinesnull25 override fun generateOutlines(outputProvider: (URI) -> Appendable, nodes: Iterable<DocumentationNode>) {
26 for (node in nodes) {
27 for (outline in outlines) {
28 val uri = outline.computeOutlineURI(node)
29 val output = outputProvider(uri)
30 outline.format(output, node)
31 }
32 }
33 }
34 }
35
36 /**
37 * Outline service for generating a _toc.yaml file, responsible for pointing to the paths of each
38 * index.html file in the doc tree.
39 */
40 class BookOutlineService(
41 val uriProvider: JavaLayoutHtmlUriProvider,
42 val languageService: LanguageService,
43 val dacRoot: String,
44 val generateClassIndex: Boolean,
45 val generatePackageIndex: Boolean
46 ) : DacOutlineFormatService {
computeOutlineURInull47 override fun computeOutlineURI(node: DocumentationNode): URI = uriProvider.outlineRootUri(node).resolve("_book.yaml")
48
49 override fun format(to: Appendable, node: DocumentationNode) {
50 appendOutline(to, listOf(node))
51 }
52
53 var outlineLevel = 0
54
55 /** Appends formatted outline to [StringBuilder](to) using specified [location] */
appendOutlinenull56 fun appendOutline(to: Appendable, nodes: Iterable<DocumentationNode>) {
57 if (outlineLevel == 0) to.appendln("reference:")
58 for (node in nodes) {
59 appendOutlineHeader(node, to)
60 val subPackages = node.members.filter {
61 it.kind == NodeKind.Package
62 }
63 if (subPackages.any()) {
64 val sortedMembers = subPackages.sortedBy { it.name.toLowerCase() }
65 appendOutlineLevel(to) {
66 appendOutline(to, sortedMembers)
67 }
68 }
69
70 }
71 }
72
appendOutlineHeadernull73 fun appendOutlineHeader(node: DocumentationNode, to: Appendable) {
74 if (node is DocumentationModule) {
75 to.appendln("- title: Package Index")
76 to.appendln(" path: $dacRoot${uriProvider.outlineRootUri(node).resolve("packages.html")}")
77 to.appendln(" status_text: no-toggle")
78 } else {
79 to.appendln("- title: ${languageService.renderName(node)}")
80 to.appendln(" path: $dacRoot${uriProvider.mainUriOrWarn(node)}")
81 to.appendln(" status_text: no-toggle")
82 }
83 }
84
appendOutlineLevelnull85 fun appendOutlineLevel(to: Appendable, body: () -> Unit) {
86 outlineLevel++
87 body()
88 outlineLevel--
89 }
90 }
91
92 /**
93 * Outline service for generating a _toc.yaml file, responsible for pointing to the paths of each
94 * index.html file in the doc tree.
95 */
96 class TocOutlineService(
97 val uriProvider: JavaLayoutHtmlUriProvider,
98 val languageService: LanguageService,
99 val dacRoot: String,
100 val generateClassIndex: Boolean,
101 val generatePackageIndex: Boolean
102 ) : DacOutlineFormatService {
computeOutlineURInull103 override fun computeOutlineURI(node: DocumentationNode): URI = uriProvider.outlineRootUri(node).resolve("_toc.yaml")
104
105 override fun format(to: Appendable, node: DocumentationNode) {
106 appendOutline(to, listOf(node))
107 }
108
109 var outlineLevel = 0
110
111 /** Appends formatted outline to [StringBuilder](to) using specified [location] */
appendOutlinenull112 fun appendOutline(to: Appendable, nodes: Iterable<DocumentationNode>) {
113 if (outlineLevel == 0) to.appendln("toc:")
114 for (node in nodes) {
115 appendOutlineHeader(node, to)
116 val subPackages = node.members.filter {
117 it.kind == NodeKind.Package
118 }
119 if (subPackages.any()) {
120 val sortedMembers = subPackages.sortedBy { it.nameWithOuterClass() }
121 appendOutlineLevel {
122 appendOutline(to, sortedMembers)
123 }
124 }
125 }
126 }
127
appendOutlineHeadernull128 fun appendOutlineHeader(node: DocumentationNode, to: Appendable) {
129 if (node is DocumentationModule) {
130 if (generateClassIndex) {
131 node.members.filter { it.kind == NodeKind.AllTypes }.firstOrNull()?.let {
132 to.appendln("- title: Class Index")
133 to.appendln(" path: $dacRoot${uriProvider.outlineRootUri(it).resolve("classes.html")}")
134 to.appendln()
135 }
136 }
137 if (generatePackageIndex) {
138 to.appendln("- title: Package Index")
139 to.appendln(" path: $dacRoot${uriProvider.outlineRootUri(node).resolve("packages.html")}")
140 to.appendln()
141 }
142 } else if (node.kind != NodeKind.AllTypes && !(node is DocumentationModule)) {
143 to.appendln("- title: ${languageService.renderName(node)}")
144 to.appendln(" path: $dacRoot${uriProvider.mainUriOrWarn(node)}")
145 to.appendln()
146 var addedSectionHeader = false
147 for (kind in NodeKind.classLike) {
148 val members = node.getMembersOfKinds(kind)
149 if (members.isNotEmpty()) {
150 if (!addedSectionHeader) {
151 to.appendln(" section:")
152 addedSectionHeader = true
153 }
154 to.appendln(" - title: ${kind.pluralizedName()}")
155 to.appendln()
156 to.appendln(" section:")
157 members.sortedBy { it.nameWithOuterClass().toLowerCase() }.forEach { member ->
158 to.appendln(" - title: ${languageService.renderNameWithOuterClass(member)}")
159 to.appendln(" path: $dacRoot${uriProvider.mainUriOrWarn(member)}".trimEnd('#'))
160 to.appendln()
161 }
162 }
163 }
164 to.appendln().appendln()
165 }
166 }
167
appendOutlineLevelnull168 fun appendOutlineLevel(body: () -> Unit) {
169 outlineLevel++
170 body()
171 outlineLevel--
172 }
173 }
174
175 class DacNavOutlineService constructor(
176 val uriProvider: JavaLayoutHtmlUriProvider,
177 val languageService: LanguageService,
178 val dacRoot: String
179 ) : DacOutlineFormatService {
computeOutlineURInull180 override fun computeOutlineURI(node: DocumentationNode): URI =
181 uriProvider.outlineRootUri(node).resolve("navtree_data.js")
182
183 override fun format(to: Appendable, node: DocumentationNode) {
184 to.append("var NAVTREE_DATA = ").appendNavTree(node.members).append(";")
185 }
186
appendNavTreenull187 private fun Appendable.appendNavTree(nodes: Iterable<DocumentationNode>): Appendable {
188 append("[ ")
189 var first = true
190 for (node in nodes) {
191 if (!first) append(", ")
192 first = false
193 val interfaces = node.getMembersOfKinds(NodeKind.Interface)
194 val classes = node.getMembersOfKinds(NodeKind.Class)
195 val objects = node.getMembersOfKinds(NodeKind.Object)
196 val annotations = node.getMembersOfKinds(NodeKind.AnnotationClass)
197 val enums = node.getMembersOfKinds(NodeKind.Enum)
198 val exceptions = node.getMembersOfKinds(NodeKind.Exception)
199
200 append("[ \"${node.name}\", \"$dacRoot${uriProvider.tryGetMainUri(node)}\", [ ")
201 var needComma = false
202 if (interfaces.firstOrNull() != null) {
203 appendNavTreePagesOfKind("Interfaces", interfaces)
204 needComma = true
205 }
206 if (classes.firstOrNull() != null) {
207 if (needComma) append(", ")
208 appendNavTreePagesOfKind("Classes", classes)
209 needComma = true
210 }
211 if (objects.firstOrNull() != null) {
212 if (needComma) append(", ")
213 appendNavTreePagesOfKind("Objects", objects)
214 }
215 if (annotations.firstOrNull() != null) {
216 if (needComma) append(", ")
217 appendNavTreePagesOfKind("Annotations", annotations)
218 needComma = true
219 }
220 if (enums.firstOrNull() != null) {
221 if (needComma) append(", ")
222 appendNavTreePagesOfKind("Enums", enums)
223 needComma = true
224 }
225 if (exceptions.firstOrNull() != null) {
226 if (needComma) append(", ")
227 appendNavTreePagesOfKind("Exceptions", exceptions)
228 }
229 append(" ] ]")
230 }
231 append(" ]")
232 return this
233 }
234
appendNavTreePagesOfKindnull235 private fun Appendable.appendNavTreePagesOfKind(kindTitle: String,
236 nodesOfKind: Iterable<DocumentationNode>): Appendable {
237 append("[ \"$kindTitle\", null, [ ")
238 var started = false
239 for (node in nodesOfKind) {
240 if (started) append(", ")
241 started = true
242 appendNavTreeChild(node)
243 }
244 append(" ], null, null ]")
245 return this
246 }
247
appendNavTreeChildnull248 private fun Appendable.appendNavTreeChild(node: DocumentationNode): Appendable {
249 append("[ \"${node.nameWithOuterClass()}\", \"${dacRoot}${uriProvider.tryGetMainUri(node)}\"")
250 append(", null, null, null ]")
251 return this
252 }
253 }
254
255 class DacSearchOutlineService(
256 val uriProvider: JavaLayoutHtmlUriProvider,
257 val languageService: LanguageService,
258 val dacRoot: String
259 ) : DacOutlineFormatService {
260
computeOutlineURInull261 override fun computeOutlineURI(node: DocumentationNode): URI =
262 uriProvider.outlineRootUri(node).resolve("lists.js")
263
264 override fun format(to: Appendable, node: DocumentationNode) {
265 val pageNodes = node.getAllPageNodes()
266 var id = 0
267 to.append("var KTX_CORE_DATA = [\n")
268 var first = true
269 for (pageNode in pageNodes) {
270 if (pageNode.kind == NodeKind.Module) continue
271 if (!first) to.append(", \n")
272 first = false
273 to.append(" { " +
274 "id:$id, " +
275 "label:\"${pageNode.qualifiedName()}\", " +
276 "link:\"${dacRoot}${uriProvider.tryGetMainUri(pageNode)}\", " +
277 "type:\"${pageNode.getClassOrPackage()}\", " +
278 "deprecated:\"false\" }")
279 id++
280 }
281 to.append("\n];")
282 }
283
getClassOrPackagenull284 private fun DocumentationNode.getClassOrPackage(): String =
285 if (hasOwnPage())
286 "class"
287 else if (isPackage()) {
288 "package"
289 } else {
290 ""
291 }
292
DocumentationNodenull293 private fun DocumentationNode.getAllPageNodes(): Iterable<DocumentationNode> {
294 val allPageNodes = mutableListOf<DocumentationNode>()
295 recursiveSetAllPageNodes(allPageNodes)
296 return allPageNodes
297 }
298
recursiveSetAllPageNodesnull299 private fun DocumentationNode.recursiveSetAllPageNodes(
300 allPageNodes: MutableList<DocumentationNode>) {
301 for (child in members) {
302 if (child.hasOwnPage() || child.isPackage()) {
303 allPageNodes.add(child)
304 child.qualifiedName()
305 child.recursiveSetAllPageNodes(allPageNodes)
306 }
307 }
308 }
309
310 }
311
312 /**
313 * Return all children of the node who are one of the selected `NodeKind`s. It recursively fetches
314 * all offspring, not just immediate children.
315 */
getMembersOfKindsnull316 fun DocumentationNode.getMembersOfKinds(vararg kinds: NodeKind): MutableList<DocumentationNode> {
317 val membersOfKind = mutableListOf<DocumentationNode>()
318 recursiveSetMembersOfKinds(kinds, membersOfKind)
319 return membersOfKind
320 }
321
DocumentationNodenull322 private fun DocumentationNode.recursiveSetMembersOfKinds(kinds: Array<out NodeKind>,
323 membersOfKind: MutableList<DocumentationNode>) {
324 for (member in members) {
325 if (member.kind in kinds) {
326 membersOfKind.add(member)
327 }
328 member.recursiveSetMembersOfKinds(kinds, membersOfKind)
329 }
330 }
331
332 /**
333 * Returns whether or not this node owns a page. The criteria for whether a node owns a page is
334 * similar to the way javadoc is structured. Classes, Interfaces, Enums, AnnotationClasses,
335 * Exceptions, and Objects (Kotlin-specific) meet the criteria.
336 */
hasOwnPagenull337 fun DocumentationNode.hasOwnPage() =
338 kind == NodeKind.Class || kind == NodeKind.Interface || kind == NodeKind.Enum ||
339 kind == NodeKind.AnnotationClass || kind == NodeKind.Exception ||
340 kind == NodeKind.Object
341
342 /**
343 * In most cases, this returns the short name of the `Type`. When the Type is an inner Type, it
344 * prepends the name with the containing Type name(s).
345 *
346 * For example, if you have a class named OuterClass and an inner class named InnerClass, this would
347 * return OuterClass.InnerClass.
348 *
349 */
350 fun DocumentationNode.nameWithOuterClass(): String {
351 val nameBuilder = StringBuilder(name)
352 var parent = owner
353 if (hasOwnPage()) {
354 while (parent != null && parent.hasOwnPage()) {
355 nameBuilder.insert(0, "${parent.name}.")
356 parent = parent.owner
357 }
358 }
359 return nameBuilder.toString()
360 }
361
362 /**
363 * Return whether the node is a package.
364 */
isPackagenull365 fun DocumentationNode.isPackage(): Boolean {
366 return kind == NodeKind.Package
367 }
368
369 /**
370 * Return the 'page owner' of this node. `DocumentationNode.hasOwnPage()` defines the criteria for
371 * a page owner. If this node is not a page owner, then it iterates up through its ancestors to
372 * find the first page owner.
373 */
pageOwnernull374 fun DocumentationNode.pageOwner(): DocumentationNode {
375 if (hasOwnPage() || owner == null) {
376 return this
377 } else {
378 var parent: DocumentationNode = owner!!
379 while (!parent.hasOwnPage() && !parent.isPackage()) {
380 parent = parent.owner!!
381 }
382 return parent
383 }
384 }
385
NodeKindnull386 fun NodeKind.pluralizedName() = when(this) {
387 NodeKind.Class -> "Classes"
388 NodeKind.Interface -> "Interfaces"
389 NodeKind.AnnotationClass -> "Annotations"
390 NodeKind.Enum -> "Enums"
391 NodeKind.Exception -> "Exceptions"
392 NodeKind.Object -> "Objects"
393 NodeKind.TypeAlias -> "TypeAliases"
394 else -> "${name}s"
395 }