From 5e24c92b5efaf147fba0444d8d702d59bb7796d1 Mon Sep 17 00:00:00 2001 From: Oleg Bask Date: Wed, 14 Aug 2024 12:26:20 +0100 Subject: [PATCH] Add strikethrough markdown support --- gradle/libs.versions.toml | 1 + markdown/README.md | 2 + markdown/core/api/core.api | 6 ++- .../MarkdownInlineRendererExtension.kt | 8 ++- .../markdown/processing/ProcessingUtil.kt | 4 +- .../DefaultInlineMarkdownRenderer.kt | 5 +- .../strikethrough/api/strikethrough.api | 29 +++++++++++ .../extension/strikethrough/build.gradle.kts | 17 +++++++ .../strikethrough/CustomStrikethroughNode.kt | 34 +++++++++++++ .../StrikethroughProcessorExtension.kt | 25 ++++++++++ .../StrikethroughRendererExtension.kt | 49 +++++++++++++++++++ .../StrikethroughProcessorExtensionTest.kt | 21 ++++++++ samples/standalone/build.gradle.kts | 3 +- .../view/markdown/MarkdownPreview.kt | 18 +++++-- settings.gradle.kts | 1 + 15 files changed, 213 insertions(+), 10 deletions(-) create mode 100644 markdown/extension/strikethrough/api/strikethrough.api create mode 100644 markdown/extension/strikethrough/build.gradle.kts create mode 100644 markdown/extension/strikethrough/src/main/kotlin/org/jetbrains/jewel/markdown/extension/strikethrough/CustomStrikethroughNode.kt create mode 100644 markdown/extension/strikethrough/src/main/kotlin/org/jetbrains/jewel/markdown/extension/strikethrough/StrikethroughProcessorExtension.kt create mode 100644 markdown/extension/strikethrough/src/main/kotlin/org/jetbrains/jewel/markdown/extension/strikethrough/StrikethroughRendererExtension.kt create mode 100644 markdown/extension/strikethrough/src/test/kotlin/org/jetbrains/jewel/markdown/extension/autolink/StrikethroughProcessorExtensionTest.kt diff --git a/gradle/libs.versions.toml b/gradle/libs.versions.toml index fe2e9ce0d..b9bc47fc5 100644 --- a/gradle/libs.versions.toml +++ b/gradle/libs.versions.toml @@ -19,6 +19,7 @@ poko = "0.15.3" [libraries] commonmark-core = { module = "org.commonmark:commonmark", version.ref = "commonmark" } commonmark-ext-autolink = { module = "org.commonmark:commonmark-ext-autolink", version.ref = "commonmark" } +commonmark-ext-strikethrough = { module = "org.commonmark:commonmark-ext-gfm-strikethrough", version.ref = "commonmark" } filePicker = { module = "com.darkrockstudios:mpfilepicker", version = "3.1.0" } diff --git a/markdown/README.md b/markdown/README.md index c41644a86..bee6c7d63 100644 --- a/markdown/README.md +++ b/markdown/README.md @@ -12,8 +12,10 @@ Additional supported Markdown, via extensions: * Alerts ([GitHub Flavored Markdown][alerts-specs]) — see [`extension-gfm-alerts`](extension/gfm-alerts) * Autolink (standard CommonMark, but provided as extension) — see [`extension-autolink`](extension/autolink) +* Strikethrough ([GitHub Flavored Markdown][strikethrough-specs]) — see [`extension-strikethrough`](extension/strikethrough) [alerts-specs]: https://github.com/orgs/community/discussions/16925 +[strikethrough-specs]: https://github.github.com/gfm/#strikethrough-extension- On the roadmap, but not currently supported — in no particular order: diff --git a/markdown/core/api/core.api b/markdown/core/api/core.api index 4c08ee61c..bee33c94c 100644 --- a/markdown/core/api/core.api +++ b/markdown/core/api/core.api @@ -322,7 +322,7 @@ public abstract interface class org/jetbrains/jewel/markdown/extensions/Markdown public abstract interface class org/jetbrains/jewel/markdown/extensions/MarkdownInlineRendererExtension { public abstract fun canRender (Lorg/jetbrains/jewel/markdown/InlineMarkdown$CustomNode;)Z - public abstract fun render (Lorg/jetbrains/jewel/markdown/InlineMarkdown$CustomNode;Lorg/jetbrains/jewel/markdown/rendering/InlineMarkdownRenderer;Z)V + public abstract fun render (Landroidx/compose/ui/text/AnnotatedString$Builder;Lorg/jetbrains/jewel/markdown/InlineMarkdown$CustomNode;Lorg/jetbrains/jewel/markdown/rendering/InlineMarkdownRenderer;Z)V } public final class org/jetbrains/jewel/markdown/extensions/MarkdownKt { @@ -374,6 +374,10 @@ public final class org/jetbrains/jewel/markdown/processing/MarkdownProcessor { public final fun processMarkdownDocument (Ljava/lang/String;)Ljava/util/List; } +public final class org/jetbrains/jewel/markdown/processing/ProcessingUtilKt { + public static final fun toInlineMarkdownOrNull (Lorg/commonmark/node/Node;Lorg/jetbrains/jewel/markdown/processing/MarkdownProcessor;Ljava/util/List;)Lorg/jetbrains/jewel/markdown/InlineMarkdown; +} + public class org/jetbrains/jewel/markdown/rendering/DefaultInlineMarkdownRenderer : org/jetbrains/jewel/markdown/rendering/InlineMarkdownRenderer { public static final field $stable I public static final field Companion Lorg/jetbrains/jewel/markdown/rendering/DefaultInlineMarkdownRenderer$Companion; diff --git a/markdown/core/src/main/kotlin/org/jetbrains/jewel/markdown/extensions/MarkdownInlineRendererExtension.kt b/markdown/core/src/main/kotlin/org/jetbrains/jewel/markdown/extensions/MarkdownInlineRendererExtension.kt index 723d0a2c0..cbfcc2cff 100644 --- a/markdown/core/src/main/kotlin/org/jetbrains/jewel/markdown/extensions/MarkdownInlineRendererExtension.kt +++ b/markdown/core/src/main/kotlin/org/jetbrains/jewel/markdown/extensions/MarkdownInlineRendererExtension.kt @@ -1,5 +1,6 @@ package org.jetbrains.jewel.markdown.extensions +import androidx.compose.ui.text.AnnotatedString import org.jetbrains.jewel.markdown.InlineMarkdown import org.jetbrains.jewel.markdown.InlineMarkdown.CustomNode import org.jetbrains.jewel.markdown.rendering.InlineMarkdownRenderer @@ -13,5 +14,10 @@ public interface MarkdownInlineRendererExtension { * Render a [CustomNode] as an annotated string. Note that if [canRender] returns `false` for [inline], the * implementation might throw. */ - public fun render(inline: CustomNode, inlineRenderer: InlineMarkdownRenderer, enabled: Boolean) + public fun render( + builder: AnnotatedString.Builder, + inline: CustomNode, + inlineRenderer: InlineMarkdownRenderer, + enabled: Boolean, + ) } diff --git a/markdown/core/src/main/kotlin/org/jetbrains/jewel/markdown/processing/ProcessingUtil.kt b/markdown/core/src/main/kotlin/org/jetbrains/jewel/markdown/processing/ProcessingUtil.kt index b0f65dcfb..25687d44f 100644 --- a/markdown/core/src/main/kotlin/org/jetbrains/jewel/markdown/processing/ProcessingUtil.kt +++ b/markdown/core/src/main/kotlin/org/jetbrains/jewel/markdown/processing/ProcessingUtil.kt @@ -32,10 +32,10 @@ internal fun Node.readInlineContent( } @VisibleForTesting -internal fun Node.toInlineMarkdownOrNull( +public fun Node.toInlineMarkdownOrNull( markdownProcessor: MarkdownProcessor, extensions: List, -) = +): InlineMarkdown? = when (this) { is CMText -> InlineMarkdown.Text(literal) is CMLink -> diff --git a/markdown/core/src/main/kotlin/org/jetbrains/jewel/markdown/rendering/DefaultInlineMarkdownRenderer.kt b/markdown/core/src/main/kotlin/org/jetbrains/jewel/markdown/rendering/DefaultInlineMarkdownRenderer.kt index 60fc29271..651493927 100644 --- a/markdown/core/src/main/kotlin/org/jetbrains/jewel/markdown/rendering/DefaultInlineMarkdownRenderer.kt +++ b/markdown/core/src/main/kotlin/org/jetbrains/jewel/markdown/rendering/DefaultInlineMarkdownRenderer.kt @@ -90,11 +90,12 @@ public open class DefaultInlineMarkdownRenderer(private val rendererExtensions: ) } - is InlineMarkdown.CustomNode -> + is InlineMarkdown.CustomNode -> { rendererExtensions .find { it.inlineRenderer?.canRender(child) == true } ?.inlineRenderer - ?.render(child, inlineRenderer = this@DefaultInlineMarkdownRenderer, enabled) + ?.render(builder = this, child, inlineRenderer = this@DefaultInlineMarkdownRenderer, enabled) + } } } } diff --git a/markdown/extension/strikethrough/api/strikethrough.api b/markdown/extension/strikethrough/api/strikethrough.api new file mode 100644 index 000000000..5e1051543 --- /dev/null +++ b/markdown/extension/strikethrough/api/strikethrough.api @@ -0,0 +1,29 @@ +public final class org/jetbrains/jewel/markdown/extension/strikethrough/CustomStrikethroughNode : org/jetbrains/jewel/markdown/InlineMarkdown$CustomNode, org/jetbrains/jewel/markdown/WithInlineMarkdown { + public static final field $stable I + public fun (Ljava/lang/String;Ljava/lang/String;Ljava/util/List;)V + public fun (Lorg/commonmark/ext/gfm/strikethrough/Strikethrough;Lorg/jetbrains/jewel/markdown/processing/MarkdownProcessor;)V + public fun contentOrNull ()Ljava/lang/String; + public fun equals (Ljava/lang/Object;)Z + public final fun getClosingDelimiter ()Ljava/lang/String; + public fun getInlineContent ()Ljava/util/List; + public final fun getOpeningDelimiter ()Ljava/lang/String; + public fun hashCode ()I + public fun toString ()Ljava/lang/String; +} + +public final class org/jetbrains/jewel/markdown/extension/strikethrough/StrikethroughProcessorExtension : org/jetbrains/jewel/markdown/extensions/MarkdownProcessorExtension { + public static final field $stable I + public static final field INSTANCE Lorg/jetbrains/jewel/markdown/extension/strikethrough/StrikethroughProcessorExtension; + public fun getBlockProcessorExtension ()Lorg/jetbrains/jewel/markdown/extensions/MarkdownBlockProcessorExtension; + public fun getInlineProcessorExtension ()Lorg/jetbrains/jewel/markdown/extensions/MarkdownInlineProcessorExtension; + public fun getParserExtension ()Lorg/commonmark/parser/Parser$ParserExtension; + public fun getTextRendererExtension ()Lorg/commonmark/renderer/text/TextContentRenderer$TextContentRendererExtension; +} + +public final class org/jetbrains/jewel/markdown/extension/strikethrough/StrikethroughRendererExtension : org/jetbrains/jewel/markdown/extensions/MarkdownRendererExtension { + public static final field $stable I + public fun (Lorg/jetbrains/jewel/markdown/rendering/InlinesStyling;)V + public fun getBlockRenderer ()Lorg/jetbrains/jewel/markdown/extensions/MarkdownBlockRendererExtension; + public fun getInlineRenderer ()Lorg/jetbrains/jewel/markdown/extensions/MarkdownInlineRendererExtension; +} + diff --git a/markdown/extension/strikethrough/build.gradle.kts b/markdown/extension/strikethrough/build.gradle.kts new file mode 100644 index 000000000..3b9482557 --- /dev/null +++ b/markdown/extension/strikethrough/build.gradle.kts @@ -0,0 +1,17 @@ +plugins { + jewel + `jewel-publish` + `jewel-check-public-api` + alias(libs.plugins.composeDesktop) +} + +dependencies { + implementation(projects.markdown.core) + implementation(libs.commonmark.ext.strikethrough) + testImplementation(compose.desktop.uiTestJUnit4) +} + +publishing.publications.named("main") { + val ijpTarget = project.property("ijp.target") as String + artifactId = "jewel-markdown-extension-${project.name}-$ijpTarget" +} diff --git a/markdown/extension/strikethrough/src/main/kotlin/org/jetbrains/jewel/markdown/extension/strikethrough/CustomStrikethroughNode.kt b/markdown/extension/strikethrough/src/main/kotlin/org/jetbrains/jewel/markdown/extension/strikethrough/CustomStrikethroughNode.kt new file mode 100644 index 000000000..f79b6e7cf --- /dev/null +++ b/markdown/extension/strikethrough/src/main/kotlin/org/jetbrains/jewel/markdown/extension/strikethrough/CustomStrikethroughNode.kt @@ -0,0 +1,34 @@ +package org.jetbrains.jewel.markdown.extension.strikethrough + +import org.commonmark.ext.gfm.strikethrough.Strikethrough +import org.commonmark.node.Node +import org.jetbrains.jewel.foundation.GenerateDataFunctions +import org.jetbrains.jewel.markdown.InlineMarkdown +import org.jetbrains.jewel.markdown.WithInlineMarkdown +import org.jetbrains.jewel.markdown.processing.MarkdownProcessor +import org.jetbrains.jewel.markdown.processing.toInlineMarkdownOrNull + +@GenerateDataFunctions +public class CustomStrikethroughNode( + public val openingDelimiter: String, + public val closingDelimiter: String, + public override val inlineContent: List, +) : InlineMarkdown.CustomNode, WithInlineMarkdown { + public constructor( + node: Strikethrough, + processor: MarkdownProcessor, + ) : this( + node.openingDelimiter, + node.closingDelimiter, + node.children().mapNotNull { it.toInlineMarkdownOrNull(processor, emptyList()) }, + ) +} + +private fun Node.children() = + buildList { + var current = this@children.firstChild + while (current != null) { + this.add(current) + current = current.next + } + } diff --git a/markdown/extension/strikethrough/src/main/kotlin/org/jetbrains/jewel/markdown/extension/strikethrough/StrikethroughProcessorExtension.kt b/markdown/extension/strikethrough/src/main/kotlin/org/jetbrains/jewel/markdown/extension/strikethrough/StrikethroughProcessorExtension.kt new file mode 100644 index 000000000..cc360f990 --- /dev/null +++ b/markdown/extension/strikethrough/src/main/kotlin/org/jetbrains/jewel/markdown/extension/strikethrough/StrikethroughProcessorExtension.kt @@ -0,0 +1,25 @@ +package org.jetbrains.jewel.markdown.extension.strikethrough + +import org.commonmark.ext.gfm.strikethrough.Strikethrough +import org.commonmark.ext.gfm.strikethrough.StrikethroughExtension +import org.commonmark.node.CustomNode +import org.commonmark.parser.Parser.ParserExtension +import org.jetbrains.jewel.markdown.InlineMarkdown +import org.jetbrains.jewel.markdown.extensions.MarkdownInlineProcessorExtension +import org.jetbrains.jewel.markdown.extensions.MarkdownProcessorExtension +import org.jetbrains.jewel.markdown.processing.MarkdownProcessor + +public object StrikethroughProcessorExtension : MarkdownProcessorExtension { + override val parserExtension: ParserExtension = StrikethroughExtension.create() as ParserExtension + + override val inlineProcessorExtension: MarkdownInlineProcessorExtension + get() = + object : MarkdownInlineProcessorExtension { + override fun canProcess(node: CustomNode): Boolean = node is Strikethrough + + override fun processInlineMarkdown( + node: CustomNode, + processor: MarkdownProcessor, + ): InlineMarkdown.CustomNode = CustomStrikethroughNode(node as Strikethrough, processor) + } +} diff --git a/markdown/extension/strikethrough/src/main/kotlin/org/jetbrains/jewel/markdown/extension/strikethrough/StrikethroughRendererExtension.kt b/markdown/extension/strikethrough/src/main/kotlin/org/jetbrains/jewel/markdown/extension/strikethrough/StrikethroughRendererExtension.kt new file mode 100644 index 000000000..7082f8042 --- /dev/null +++ b/markdown/extension/strikethrough/src/main/kotlin/org/jetbrains/jewel/markdown/extension/strikethrough/StrikethroughRendererExtension.kt @@ -0,0 +1,49 @@ +package org.jetbrains.jewel.markdown.extension.strikethrough + +import androidx.compose.ui.text.AnnotatedString.Builder +import androidx.compose.ui.text.SpanStyle +import androidx.compose.ui.text.style.TextDecoration +import org.jetbrains.jewel.markdown.InlineMarkdown +import org.jetbrains.jewel.markdown.WithInlineMarkdown +import org.jetbrains.jewel.markdown.extensions.MarkdownInlineRendererExtension +import org.jetbrains.jewel.markdown.extensions.MarkdownRendererExtension +import org.jetbrains.jewel.markdown.rendering.InlineMarkdownRenderer +import org.jetbrains.jewel.markdown.rendering.InlinesStyling + +public class StrikethroughRendererExtension(private val inlinesStyling: InlinesStyling) : MarkdownRendererExtension { + override val inlineRenderer: MarkdownInlineRendererExtension + get() = + object : MarkdownInlineRendererExtension { + override fun canRender(inline: InlineMarkdown.CustomNode): Boolean = inline is CustomStrikethroughNode + + override fun render( + builder: Builder, + inline: InlineMarkdown.CustomNode, + inlineRenderer: InlineMarkdownRenderer, + enabled: Boolean, + ) { + with(builder) { + val popTo = + pushStyle( + inlinesStyling.textStyle + .copy(textDecoration = TextDecoration.LineThrough) + .toSpanStyle() + .copy() + ) + val inlineMarkdowns = (inline as? WithInlineMarkdown)?.inlineContent ?: emptyList() + for (markdown in inlineMarkdowns) { + append( + inlineRenderer.renderAsAnnotatedString(listOf(markdown), inlinesStyling, enabled = true) + ) + } + pop(popTo) + } + } + } +} + +private inline fun Builder.withStyles(spanStyle: SpanStyle, action: Builder.() -> Unit) { + val popTo = pushStyle(spanStyle) + action() + pop(popTo) +} diff --git a/markdown/extension/strikethrough/src/test/kotlin/org/jetbrains/jewel/markdown/extension/autolink/StrikethroughProcessorExtensionTest.kt b/markdown/extension/strikethrough/src/test/kotlin/org/jetbrains/jewel/markdown/extension/autolink/StrikethroughProcessorExtensionTest.kt new file mode 100644 index 000000000..cf127790e --- /dev/null +++ b/markdown/extension/strikethrough/src/test/kotlin/org/jetbrains/jewel/markdown/extension/autolink/StrikethroughProcessorExtensionTest.kt @@ -0,0 +1,21 @@ +package org.jetbrains.jewel.markdown.extension.autolink + +import org.jetbrains.jewel.markdown.InlineMarkdown +import org.jetbrains.jewel.markdown.MarkdownBlock +import org.jetbrains.jewel.markdown.extension.strikethrough.StrikethroughProcessorExtension +import org.jetbrains.jewel.markdown.processing.MarkdownProcessor +import org.junit.Assert.assertTrue +import org.junit.Test + +class StrikethroughProcessorExtensionTest { + // testing a simple case to assure wiring up our AutolinkProcessorExtension works correctly + @Test + fun `https text gets processed into a link`() { + val processor = MarkdownProcessor(listOf(StrikethroughProcessorExtension)) + val rawMarkDown = "~~fdsa~~" + val processed = processor.processMarkdownDocument(rawMarkDown) + val paragraph = processed.first() as MarkdownBlock.Paragraph + + assertTrue(paragraph.inlineContent.first() is InlineMarkdown.Link) + } +} diff --git a/samples/standalone/build.gradle.kts b/samples/standalone/build.gradle.kts index 1aa6754cd..2a26927ed 100644 --- a/samples/standalone/build.gradle.kts +++ b/samples/standalone/build.gradle.kts @@ -13,8 +13,9 @@ dependencies { implementation(projects.intUi.intUiStandalone) implementation(projects.intUi.intUiDecoratedWindow) implementation(projects.markdown.intUiStandaloneStyling) - implementation(projects.markdown.extension.gfmAlerts) implementation(projects.markdown.extension.autolink) + implementation(projects.markdown.extension.gfmAlerts) + implementation(projects.markdown.extension.strikethrough) implementation(compose.desktop.currentOs) { exclude(group = "org.jetbrains.compose.material") } implementation(libs.intellijPlatform.icons) } diff --git a/samples/standalone/src/main/kotlin/org/jetbrains/jewel/samples/standalone/view/markdown/MarkdownPreview.kt b/samples/standalone/src/main/kotlin/org/jetbrains/jewel/samples/standalone/view/markdown/MarkdownPreview.kt index 98f5ab893..0adc36546 100644 --- a/samples/standalone/src/main/kotlin/org/jetbrains/jewel/samples/standalone/view/markdown/MarkdownPreview.kt +++ b/samples/standalone/src/main/kotlin/org/jetbrains/jewel/samples/standalone/view/markdown/MarkdownPreview.kt @@ -27,6 +27,8 @@ import org.jetbrains.jewel.intui.markdown.standalone.styling.light import org.jetbrains.jewel.markdown.LazyMarkdown import org.jetbrains.jewel.markdown.MarkdownBlock import org.jetbrains.jewel.markdown.extension.autolink.AutolinkProcessorExtension +import org.jetbrains.jewel.markdown.extension.strikethrough.StrikethroughProcessorExtension +import org.jetbrains.jewel.markdown.extension.strikethrough.StrikethroughRendererExtension import org.jetbrains.jewel.markdown.extensions.github.alerts.AlertStyling import org.jetbrains.jewel.markdown.extensions.github.alerts.GitHubAlertProcessorExtension import org.jetbrains.jewel.markdown.extensions.github.alerts.GitHubAlertRendererExtension @@ -43,7 +45,9 @@ internal fun MarkdownPreview(modifier: Modifier = Modifier, rawMarkdown: CharSeq val markdownStyling = remember(isDark) { if (isDark) MarkdownStyling.dark() else MarkdownStyling.light() } var markdownBlocks by remember { mutableStateOf(emptyList()) } - val extensions = remember { listOf(GitHubAlertProcessorExtension, AutolinkProcessorExtension) } + val extensions = remember { + listOf(GitHubAlertProcessorExtension, AutolinkProcessorExtension, StrikethroughProcessorExtension) + } // We are doing this here for the sake of simplicity. // In a real-world scenario you would be doing this outside your Composables, @@ -63,12 +67,20 @@ internal fun MarkdownPreview(modifier: Modifier = Modifier, rawMarkdown: CharSeq if (isDark) { MarkdownBlockRenderer.dark( styling = markdownStyling, - rendererExtensions = listOf(GitHubAlertRendererExtension(AlertStyling.dark(), markdownStyling)), + rendererExtensions = + listOf( + GitHubAlertRendererExtension(AlertStyling.dark(), markdownStyling), + StrikethroughRendererExtension(markdownStyling.paragraph.inlinesStyling), + ), ) } else { MarkdownBlockRenderer.light( styling = markdownStyling, - rendererExtensions = listOf(GitHubAlertRendererExtension(AlertStyling.light(), markdownStyling)), + rendererExtensions = + listOf( + GitHubAlertRendererExtension(AlertStyling.light(), markdownStyling), + StrikethroughRendererExtension(markdownStyling.paragraph.inlinesStyling), + ), ) } } diff --git a/settings.gradle.kts b/settings.gradle.kts index 608a13983..ad95660c7 100644 --- a/settings.gradle.kts +++ b/settings.gradle.kts @@ -41,6 +41,7 @@ include( ":markdown:core", ":markdown:extension:autolink", ":markdown:extension:gfm-alerts", + ":markdown:extension:strikethrough", ":markdown:int-ui-standalone-styling", ":markdown:ide-laf-bridge-styling", ":samples:ide-plugin",