1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
|
package org.jetbrains.dokka.plugability
data class ExtensionPoint<T : Any> internal constructor(
internal val pluginClass: String,
internal val pointName: String
) {
override fun toString() = "ExtensionPoint: $pluginClass/$pointName"
}
abstract class Extension<T : Any> internal constructor(
internal val extensionPoint: ExtensionPoint<T>,
internal val pluginClass: String,
internal val extensionName: String,
val action: T,
internal val ordering: (OrderDsl.() -> Unit)? = null
) {
override fun toString() = "Extension: $pluginClass/$extensionName"
override fun equals(other: Any?) =
if (other is Extension<*>) this.pluginClass == other.pluginClass && this.extensionName == other.extensionName
else false
override fun hashCode() = listOf(pluginClass, extensionName).hashCode()
}
class ExtensionOrdered<T : Any>(
extensionPoint: ExtensionPoint<T>,
pluginClass: String,
extensionName: String,
action: T,
ordering: (OrderDsl.() -> Unit)
) : Extension<T>(
extensionPoint,
pluginClass,
extensionName,
action,
ordering
)
class ExtensionUnordered<T : Any>(
extensionPoint: ExtensionPoint<T>,
pluginClass: String,
extensionName: String,
action: T
) : Extension<T>(
extensionPoint,
pluginClass,
extensionName,
action,
null
)
internal data class Ordering(val previous: Set<Extension<*>>, val following: Set<Extension<*>>)
@DslMarker
annotation class ExtensionsDsl
@ExtensionsDsl
class ExtendingDSL(private val pluginClass: String, private val extensionName: String) {
infix fun <T: Any> ExtensionPoint<T>.with(action: T) =
ExtensionUnordered(this, this@ExtendingDSL.pluginClass, extensionName, action)
infix fun <T: Any> ExtensionUnordered<T>.order(block: OrderDsl.() -> Unit) =
ExtensionOrdered(extensionPoint, pluginClass, extensionName, action, block)
}
@ExtensionsDsl
class OrderDsl {
internal val previous = mutableSetOf<Extension<*>>()
internal val following = mutableSetOf<Extension<*>>()
fun after(vararg extensions: Extension<*>) {
previous += extensions
}
fun before(vararg extensions: Extension<*>) {
following += extensions
}
}
|